video.py 10.3 KB
Newer Older
1
import re
2
3
4
import gc
import torch
import numpy as np
5
6
import math
import warnings
7
8
9

try:
    import av
10
    av.logging.set_level(av.logging.ERROR)
11
12
13
14
15
16
17
18
    if not hasattr(av.video.frame.VideoFrame, 'pict_type'):
        av = ImportError("""\
Your version of PyAV is too old for the necessary video operations in torchvision.
If you are on Python 3.5, you will have to build from source (the conda-forge
packages are not up-to-date).  See
https://github.com/mikeboers/PyAV#installation for instructions on how to
install PyAV on your system.
""")
19
except ImportError:
20
    av = ImportError("""\
21
22
23
24
25
26
PyAV is not installed, and is necessary for the video operations in torchvision.
See https://github.com/mikeboers/PyAV#installation for instructions on how to
install PyAV on your system.
""")


27
28
29
30
31
32
33
34
35
def _check_av_available():
    if isinstance(av, Exception):
        raise av


def _av_available():
    return not isinstance(av, Exception)


36
37
# PyAV has some reference cycles
_CALLED_TIMES = 0
38
_GC_COLLECTION_INTERVAL = 10
39
40


41
def write_video(filename, video_array, fps, video_codec='libx264', options=None):
42
43
44
    """
    Writes a 4d tensor in [T, H, W, C] format in a video file

45
46
47
48
49
50
51
52
    Parameters
    ----------
    filename : str
        path where the video will be saved
    video_array : Tensor[T, H, W, C]
        tensor containing the individual frames, as a uint8 tensor in [T, H, W, C] format
    fps : Number
        frames per second
53
54
55
56
57
58
    """
    _check_av_available()
    video_array = torch.as_tensor(video_array, dtype=torch.uint8).numpy()

    container = av.open(filename, mode='w')

59
    stream = container.add_stream(video_codec, rate=fps)
60
61
    stream.width = video_array.shape[2]
    stream.height = video_array.shape[1]
62
63
    stream.pix_fmt = 'yuv420p' if video_codec != 'libx264rgb' else 'rgb24'
    stream.options = options or {}
64
65
66

    for img in video_array:
        frame = av.VideoFrame.from_ndarray(img, format='rgb24')
67
        frame.pict_type = 'NONE'
68
69
70
71
72
73
74
75
76
77
78
        for packet in stream.encode(frame):
            container.mux(packet)

    # Flush stream
    for packet in stream.encode():
        container.mux(packet)

    # Close the file
    container.close()


79
def _read_from_stream(container, start_offset, end_offset, pts_unit, stream, stream_name):
80
81
82
83
84
    global _CALLED_TIMES, _GC_COLLECTION_INTERVAL
    _CALLED_TIMES += 1
    if _CALLED_TIMES % _GC_COLLECTION_INTERVAL == _GC_COLLECTION_INTERVAL - 1:
        gc.collect()

85
86
87
88
89
90
91
92
    if pts_unit == 'sec':
        start_offset = int(math.floor(start_offset * (1 / stream.time_base)))
        if end_offset != float("inf"):
            end_offset = int(math.ceil(end_offset * (1 / stream.time_base)))
    else:
        warnings.warn("The pts_unit 'pts' gives wrong results and will be removed in a " +
                      "follow-up version. Please use pts_unit 'sec'.")

93
94
95
96
    frames = {}
    should_buffer = False
    max_buffer_size = 5
    if stream.type == "video":
97
        # DivX-style packed B-frames can have out-of-order pts (2 frames in a single pkt)
98
99
        # so need to buffer some extra frames to sort everything
        # properly
100
101
102
103
104
105
106
107
108
109
110
111
        extradata = stream.codec_context.extradata
        # overly complicated way of finding if `divx_packed` is set, following
        # https://github.com/FFmpeg/FFmpeg/commit/d5a21172283572af587b3d939eba0091484d3263
        if extradata and b"DivX" in extradata:
            # can't use regex directly because of some weird characters sometimes...
            pos = extradata.find(b"DivX")
            d = extradata[pos:]
            o = re.search(br"DivX(\d+)Build(\d+)(\w)", d)
            if o is None:
                o = re.search(br"DivX(\d+)b(\d+)(\w)", d)
            if o is not None:
                should_buffer = o.group(3) == b"p"
112
    seek_offset = start_offset
113
114
    # some files don't seek to the right location, so better be safe here
    seek_offset = max(seek_offset - 1, 0)
115
116
117
118
    if should_buffer:
        # FIXME this is kind of a hack, but we will jump to the previous keyframe
        # so this will be safe
        seek_offset = max(seek_offset - max_buffer_size, 0)
119
120
121
122
    try:
        # TODO check if stream needs to always be the video stream here or not
        container.seek(seek_offset, any_frame=False, backward=True, stream=stream)
    except av.AVError:
123
124
        # TODO add some warnings in this case
        # print("Corrupted file?", container.name)
125
        return []
126
    buffer_count = 0
127
128
129
130
131
132
133
134
135
136
137
    try:
        for idx, frame in enumerate(container.decode(**stream_name)):
            frames[frame.pts] = frame
            if frame.pts >= end_offset:
                if should_buffer and buffer_count < max_buffer_size:
                    buffer_count += 1
                    continue
                break
    except av.AVError:
        # TODO add a warning
        pass
138
139
    # ensure that the results are sorted wrt the pts
    result = [frames[i] for i in sorted(frames) if start_offset <= frames[i].pts <= end_offset]
140
    if len(frames) > 0 and start_offset > 0 and start_offset not in frames:
141
142
143
        # if there is no frame that exactly matches the pts of start_offset
        # add the last frame smaller than start_offset, to guarantee that
        # we will have all the necessary data. This is most useful for audio
144
145
146
147
        preceding_frames = [i for i in frames if i < start_offset]
        if len(preceding_frames) > 0:
            first_frame_pts = max(preceding_frames)
            result.insert(0, frames[first_frame_pts])
148
    return result
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163


def _align_audio_frames(aframes, audio_frames, ref_start, ref_end):
    start, end = audio_frames[0].pts, audio_frames[-1].pts
    total_aframes = aframes.shape[1]
    step_per_aframe = (end - start + 1) / total_aframes
    s_idx = 0
    e_idx = total_aframes
    if start < ref_start:
        s_idx = int((ref_start - start) / step_per_aframe)
    if end > ref_end:
        e_idx = int((ref_end - end) / step_per_aframe)
    return aframes[:, s_idx:e_idx]


164
def read_video(filename, start_pts=0, end_pts=None, pts_unit='pts'):
165
166
167
168
    """
    Reads a video from a file, returning both the video frames as well as
    the audio frames

169
170
171
172
    Parameters
    ----------
    filename : str
        path to the video file
173
174
    start_pts : int if pts_unit = 'pts', optional
        float / Fraction if pts_unit = 'sec', optional
175
        the start presentation time of the video
176
177
    end_pts : int if pts_unit = 'pts', optional
        float / Fraction if pts_unit = 'sec', optional
178
        the end presentation time
179
180
    pts_unit : str, optional
        unit in which start_pts and end_pts values will be interpreted, either 'pts' or 'sec'. Defaults to 'pts'.
181
182
183
184
185
186
187
188
189
190
191

    Returns
    -------
    vframes : Tensor[T, H, W, C]
        the `T` video frames
    aframes : Tensor[K, L]
        the audio frames, where `K` is the number of channels and `L` is the
        number of points
    info : Dict
        metadata for the video and audio. Can contain the fields video_fps (float)
        and audio_fps (int)
192
193
194
195
196
197
198
199
200
201
202
203
204
205
    """
    _check_av_available()

    if end_pts is None:
        end_pts = float("inf")

    if end_pts < start_pts:
        raise ValueError("end_pts should be larger than start_pts, got "
                         "start_pts={} and end_pts={}".format(start_pts, end_pts))

    info = {}
    video_frames = []
    audio_frames = []

206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
    try:
        container = av.open(filename, metadata_errors='ignore')
    except av.AVError:
        # TODO raise a warning?
        pass
    else:
        if container.streams.video:
            video_frames = _read_from_stream(container, start_pts, end_pts, pts_unit,
                                             container.streams.video[0], {'video': 0})
            video_fps = container.streams.video[0].average_rate
            # guard against potentially corrupted files
            if video_fps is not None:
                info["video_fps"] = float(video_fps)

        if container.streams.audio:
            audio_frames = _read_from_stream(container, start_pts, end_pts, pts_unit,
                                             container.streams.audio[0], {'audio': 0})
            info["audio_fps"] = container.streams.audio[0].rate

        container.close()
226
227
228

    vframes = [frame.to_rgb().to_ndarray() for frame in video_frames]
    aframes = [frame.to_ndarray() for frame in audio_frames]
229
230
231
232
233
234

    if vframes:
        vframes = torch.as_tensor(np.stack(vframes))
    else:
        vframes = torch.empty((0, 1, 1, 3), dtype=torch.uint8)

235
236
237
238
239
240
241
242
243
244
    if aframes:
        aframes = np.concatenate(aframes, 1)
        aframes = torch.as_tensor(aframes)
        aframes = _align_audio_frames(aframes, audio_frames, start_pts, end_pts)
    else:
        aframes = torch.empty((1, 0), dtype=torch.float32)

    return vframes, aframes, info


245
246
247
248
249
250
251
252
253
def _can_read_timestamps_from_packets(container):
    extradata = container.streams[0].codec_context.extradata
    if extradata is None:
        return False
    if b"Lavc" in extradata:
        return True
    return False


254
def read_video_timestamps(filename, pts_unit='pts'):
255
256
257
258
259
    """
    List the video frames timestamps.

    Note that the function decodes the whole video frame-by-frame.

260
261
262
263
    Parameters
    ----------
    filename : str
        path to the video file
264
265
    pts_unit : str, optional
        unit in which timestamp values will be returned either 'pts' or 'sec'. Defaults to 'pts'.
266
267
268

    Returns
    -------
269
270
    pts : List[int] if pts_unit = 'pts'
        List[Fraction] if pts_unit = 'sec'
271
272
273
        presentation timestamps for each one of the frames in the video.
    video_fps : int
        the frame rate for the video
274
275
276

    """
    _check_av_available()
277

278
    video_frames = []
279
    video_fps = None
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300

    try:
        container = av.open(filename, metadata_errors='ignore')
    except av.AVError:
        # TODO add a warning
        pass
    else:
        if container.streams.video:
            video_stream = container.streams.video[0]
            video_time_base = video_stream.time_base
            if _can_read_timestamps_from_packets(container):
                # fast path
                video_frames = [x for x in container.demux(video=0) if x.pts is not None]
            else:
                video_frames = _read_from_stream(container, 0, float("inf"), pts_unit,
                                                 video_stream, {'video': 0})
            video_fps = float(video_stream.average_rate)
        container.close()

    pts = [x.pts for x in video_frames]

301
    if pts_unit == 'sec':
302
303
304
        pts = [x * video_time_base for x in pts]

    return pts, video_fps