v4l2.c 33.9 KB
Newer Older
Luca Abeni's avatar
Luca Abeni committed
1
/*
2 3
 * Copyright (c) 2000,2001 Fabrice Bellard
 * Copyright (c) 2006 Luca Abeni
Luca Abeni's avatar
Luca Abeni committed
4
 *
5 6 7
 * This file is part of FFmpeg.
 *
 * FFmpeg is free software; you can redistribute it and/or
Luca Abeni's avatar
Luca Abeni committed
8 9
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
10
 * version 2.1 of the License, or (at your option) any later version.
Luca Abeni's avatar
Luca Abeni committed
11
 *
12
 * FFmpeg is distributed in the hope that it will be useful,
Luca Abeni's avatar
Luca Abeni committed
13 14 15 16 17
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
18
 * License along with FFmpeg; if not, write to the Free Software
Luca Abeni's avatar
Luca Abeni committed
19 20
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */
21

22 23 24 25 26
/**
 * @file
 * Video4Linux2 grab interface
 *
 * Part of this file is based on the V4L2 video capture example
27
 * (http://linuxtv.org/downloads/v4l-dvb-apis/capture-example.html)
28 29
 *
 * Thanks to Michael Niedermayer for providing the mapping between
30
 * V4L2_PIX_FMT_* and AV_PIX_FMT_*
31 32
 */

33
#include "v4l2-common.h"
Luca Abeni's avatar
Luca Abeni committed
34

35 36 37 38
#if CONFIG_LIBV4L2
#include <libv4l2.h>
#endif

Luca Abeni's avatar
Luca Abeni committed
39 40
static const int desired_video_buffers = 256;

41 42 43
#define V4L_ALLFORMATS  3
#define V4L_RAWFORMATS  1
#define V4L_COMPFORMATS 2
Luca Abeni's avatar
Luca Abeni committed
44

45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
/**
 * Return timestamps to the user exactly as returned by the kernel
 */
#define V4L_TS_DEFAULT  0
/**
 * Autodetect the kind of timestamps returned by the kernel and convert to
 * absolute (wall clock) timestamps.
 */
#define V4L_TS_ABS      1
/**
 * Assume kernel timestamps are from the monotonic clock and convert to
 * absolute timestamps.
 */
#define V4L_TS_MONO2ABS 2

/**
 * Once the kind of timestamps returned by the kernel have been detected,
 * the value of the timefilter (NULL or not) determines whether a conversion
 * takes place.
 */
#define V4L_TS_CONVERT_READY V4L_TS_DEFAULT

Luca Abeni's avatar
Luca Abeni committed
67
struct video_data {
68
    AVClass *class;
Luca Abeni's avatar
Luca Abeni committed
69 70 71 72
    int fd;
    int frame_format; /* V4L2_PIX_FMT_* */
    int width, height;
    int frame_size;
73
    int interlaced;
Luca Abeni's avatar
Luca Abeni committed
74
    int top_field_first;
75 76 77
    int ts_mode;
    TimeFilter *timefilter;
    int64_t last_time_m;
Luca Abeni's avatar
Luca Abeni committed
78 79

    int buffers;
80
    volatile int buffers_queued;
Luca Abeni's avatar
Luca Abeni committed
81 82
    void **buf_start;
    unsigned int *buf_len;
83
    char *standard;
84
    v4l2_std_id std_id;
85
    int channel;
86
    char *pixel_format; /**< Set by a private option. */
87
    int list_format;    /**< Set by a private option. */
88
    int list_standard;  /**< Set by a private option. */
89
    char *framerate;    /**< Set by a private option. */
90 91 92 93 94 95 96 97 98

    int use_libv4l2;
    int (*open_f)(const char *file, int oflag, ...);
    int (*close_f)(int fd);
    int (*dup_f)(int fd);
    int (*ioctl_f)(int fd, unsigned long int request, ...);
    ssize_t (*read_f)(int fd, void *buffer, size_t n);
    void *(*mmap_f)(void *start, size_t length, int prot, int flags, int fd, int64_t offset);
    int (*munmap_f)(void *_start, size_t length);
Luca Abeni's avatar
Luca Abeni committed
99 100
};

101
struct buff_data {
102
    struct video_data *s;
103 104 105
    int index;
};

106
static int device_open(AVFormatContext *ctx)
Luca Abeni's avatar
Luca Abeni committed
107
{
108
    struct video_data *s = ctx->priv_data;
Luca Abeni's avatar
Luca Abeni committed
109 110
    struct v4l2_capability cap;
    int fd;
111
    int ret;
112
    int flags = O_RDWR;
Luca Abeni's avatar
Luca Abeni committed
113

114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
#define SET_WRAPPERS(prefix) do {       \
    s->open_f   = prefix ## open;       \
    s->close_f  = prefix ## close;      \
    s->dup_f    = prefix ## dup;        \
    s->ioctl_f  = prefix ## ioctl;      \
    s->read_f   = prefix ## read;       \
    s->mmap_f   = prefix ## mmap;       \
    s->munmap_f = prefix ## munmap;     \
} while (0)

    if (s->use_libv4l2) {
#if CONFIG_LIBV4L2
        SET_WRAPPERS(v4l2_);
#else
        av_log(ctx, AV_LOG_ERROR, "libavdevice is not build with libv4l2 support.\n");
        return AVERROR(EINVAL);
#endif
    } else {
        SET_WRAPPERS();
    }

#define v4l2_open   s->open_f
#define v4l2_close  s->close_f
#define v4l2_dup    s->dup_f
#define v4l2_ioctl  s->ioctl_f
#define v4l2_read   s->read_f
#define v4l2_mmap   s->mmap_f
#define v4l2_munmap s->munmap_f

143 144 145
    if (ctx->flags & AVFMT_FLAG_NONBLOCK) {
        flags |= O_NONBLOCK;
    }
Luca Barbato's avatar
Luca Barbato committed
146

147
    fd = v4l2_open(ctx->filename, flags, 0);
Luca Abeni's avatar
Luca Abeni committed
148
    if (fd < 0) {
149
        ret = AVERROR(errno);
150
        av_log(ctx, AV_LOG_ERROR, "Cannot open video device %s: %s\n",
151 152
               ctx->filename, av_err2str(ret));
        return ret;
Luca Abeni's avatar
Luca Abeni committed
153 154
    }

155 156
    if (v4l2_ioctl(fd, VIDIOC_QUERYCAP, &cap) < 0) {
        ret = AVERROR(errno);
157
        av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_QUERYCAP): %s\n",
158
               av_err2str(ret));
159
        goto fail;
Luca Abeni's avatar
Luca Abeni committed
160
    }
Luca Barbato's avatar
Luca Barbato committed
161

162
    av_log(ctx, AV_LOG_VERBOSE, "fd:%d capabilities:%x\n",
163 164 165 166
           fd, cap.capabilities);

    if (!(cap.capabilities & V4L2_CAP_VIDEO_CAPTURE)) {
        av_log(ctx, AV_LOG_ERROR, "Not a video capture device.\n");
167
        ret = AVERROR(ENODEV);
168
        goto fail;
Luca Abeni's avatar
Luca Abeni committed
169
    }
Luca Barbato's avatar
Luca Barbato committed
170

171 172 173
    if (!(cap.capabilities & V4L2_CAP_STREAMING)) {
        av_log(ctx, AV_LOG_ERROR,
               "The device does not support the streaming I/O method.\n");
174
        ret = AVERROR(ENOSYS);
175
        goto fail;
Luca Abeni's avatar
Luca Abeni committed
176 177 178
    }

    return fd;
179 180

fail:
181
    v4l2_close(fd);
182
    return ret;
Luca Abeni's avatar
Luca Abeni committed
183 184
}

Luca Barbato's avatar
Luca Barbato committed
185 186
static int device_init(AVFormatContext *ctx, int *width, int *height,
                       uint32_t pix_fmt)
Luca Abeni's avatar
Luca Abeni committed
187
{
188 189
    struct video_data *s = ctx->priv_data;
    int fd = s->fd;
190
    struct v4l2_format fmt = { .type = V4L2_BUF_TYPE_VIDEO_CAPTURE };
Luca Barbato's avatar
Luca Barbato committed
191 192
    struct v4l2_pix_format *pix = &fmt.fmt.pix;

193
    int res = 0;
Luca Abeni's avatar
Luca Abeni committed
194

Luca Barbato's avatar
Luca Barbato committed
195 196 197 198 199
    pix->width = *width;
    pix->height = *height;
    pix->pixelformat = pix_fmt;
    pix->field = V4L2_FIELD_ANY;

200 201
    if (v4l2_ioctl(fd, VIDIOC_S_FMT, &fmt) < 0)
        res = AVERROR(errno);
Luca Barbato's avatar
Luca Barbato committed
202

203
    if ((*width != fmt.fmt.pix.width) || (*height != fmt.fmt.pix.height)) {
Luca Barbato's avatar
Luca Barbato committed
204 205 206
        av_log(ctx, AV_LOG_INFO,
               "The V4L2 driver changed the video from %dx%d to %dx%d\n",
               *width, *height, fmt.fmt.pix.width, fmt.fmt.pix.height);
207 208 209 210
        *width = fmt.fmt.pix.width;
        *height = fmt.fmt.pix.height;
    }

211
    if (pix_fmt != fmt.fmt.pix.pixelformat) {
Luca Barbato's avatar
Luca Barbato committed
212 213 214 215
        av_log(ctx, AV_LOG_DEBUG,
               "The V4L2 driver changed the pixel format "
               "from 0x%08X to 0x%08X\n",
               pix_fmt, fmt.fmt.pix.pixelformat);
216
        res = AVERROR(EINVAL);
217 218
    }

219
    if (fmt.fmt.pix.field == V4L2_FIELD_INTERLACED) {
220 221
        av_log(ctx, AV_LOG_DEBUG,
               "The V4L2 driver is using the interlaced mode\n");
222 223 224
        s->interlaced = 1;
    }

225
    return res;
Luca Abeni's avatar
Luca Abeni committed
226 227
}

228
static int first_field(const struct video_data *s, int fd)
Luca Abeni's avatar
Luca Abeni committed
229 230 231 232
{
    int res;
    v4l2_std_id std;

233
    res = v4l2_ioctl(fd, VIDIOC_G_STD, &std);
Luca Abeni's avatar
Luca Abeni committed
234 235 236 237 238 239 240 241 242 243
    if (res < 0) {
        return 0;
    }
    if (std & V4L2_STD_NTSC) {
        return 0;
    }

    return 1;
}

244 245 246
#if HAVE_STRUCT_V4L2_FRMIVALENUM_DISCRETE
static void list_framesizes(AVFormatContext *ctx, int fd, uint32_t pixelformat)
{
247
    const struct video_data *s = ctx->priv_data;
248 249
    struct v4l2_frmsizeenum vfse = { .pixel_format = pixelformat };

250
    while(!v4l2_ioctl(fd, VIDIOC_ENUM_FRAMESIZES, &vfse)) {
251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272
        switch (vfse.type) {
        case V4L2_FRMSIZE_TYPE_DISCRETE:
            av_log(ctx, AV_LOG_INFO, " %ux%u",
                   vfse.discrete.width, vfse.discrete.height);
        break;
        case V4L2_FRMSIZE_TYPE_CONTINUOUS:
        case V4L2_FRMSIZE_TYPE_STEPWISE:
            av_log(ctx, AV_LOG_INFO, " {%u-%u, %u}x{%u-%u, %u}",
                   vfse.stepwise.min_width,
                   vfse.stepwise.max_width,
                   vfse.stepwise.step_width,
                   vfse.stepwise.min_height,
                   vfse.stepwise.max_height,
                   vfse.stepwise.step_height);
        }
        vfse.index++;
    }
}
#endif

static void list_formats(AVFormatContext *ctx, int fd, int type)
{
273
    const struct video_data *s = ctx->priv_data;
274 275
    struct v4l2_fmtdesc vfd = { .type = V4L2_BUF_TYPE_VIDEO_CAPTURE };

276
    while(!v4l2_ioctl(fd, VIDIOC_ENUM_FMT, &vfd)) {
277 278
        enum AVCodecID codec_id = avpriv_fmt_v4l2codec(vfd.pixelformat);
        enum AVPixelFormat pix_fmt = avpriv_fmt_v4l2ff(vfd.pixelformat, codec_id);
279 280 281 282 283 284

        vfd.index++;

        if (!(vfd.flags & V4L2_FMT_FLAG_COMPRESSED) &&
            type & V4L_RAWFORMATS) {
            const char *fmt_name = av_get_pix_fmt_name(pix_fmt);
285
            av_log(ctx, AV_LOG_INFO, "Raw       : %9s : %20s :",
286 287 288 289
                   fmt_name ? fmt_name : "Unsupported",
                   vfd.description);
        } else if (vfd.flags & V4L2_FMT_FLAG_COMPRESSED &&
                   type & V4L_COMPFORMATS) {
290
            AVCodec *codec = avcodec_find_decoder(codec_id);
291
            av_log(ctx, AV_LOG_INFO, "Compressed: %9s : %20s :",
292 293 294 295 296 297
                   codec ? codec->name : "Unsupported",
                   vfd.description);
        } else {
            continue;
        }

298
#ifdef V4L2_FMT_FLAG_EMULATED
299 300
        if (vfd.flags & V4L2_FMT_FLAG_EMULATED)
            av_log(ctx, AV_LOG_INFO, " Emulated :");
301
#endif
302 303 304 305 306 307 308
#if HAVE_STRUCT_V4L2_FRMIVALENUM_DISCRETE
        list_framesizes(ctx, fd, vfd.pixelformat);
#endif
        av_log(ctx, AV_LOG_INFO, "\n");
    }
}

309 310 311 312 313 314 315 316 317 318
static void list_standards(AVFormatContext *ctx)
{
    int ret;
    struct video_data *s = ctx->priv_data;
    struct v4l2_standard standard;

    if (s->std_id == 0)
        return;

    for (standard.index = 0; ; standard.index++) {
319 320 321
        if (v4l2_ioctl(s->fd, VIDIOC_ENUMSTD, &standard) < 0) {
            ret = AVERROR(errno);
            if (ret == AVERROR(EINVAL)) {
322
                break;
323 324
            } else {
                av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_ENUMSTD): %s\n", av_err2str(ret));
325 326 327 328 329 330 331 332
                return;
            }
        }
        av_log(ctx, AV_LOG_INFO, "%2d, %16llx, %s\n",
               standard.index, standard.id, standard.name);
    }
}

333
static int mmap_init(AVFormatContext *ctx)
Luca Abeni's avatar
Luca Abeni committed
334 335
{
    int i, res;
336 337 338 339 340 341
    struct video_data *s = ctx->priv_data;
    struct v4l2_requestbuffers req = {
        .type   = V4L2_BUF_TYPE_VIDEO_CAPTURE,
        .count  = desired_video_buffers,
        .memory = V4L2_MEMORY_MMAP
    };
Luca Abeni's avatar
Luca Abeni committed
342

343 344 345 346
    if (v4l2_ioctl(s->fd, VIDIOC_REQBUFS, &req) < 0) {
        res = AVERROR(errno);
        av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_REQBUFS): %s\n", av_err2str(res));
        return res;
Luca Abeni's avatar
Luca Abeni committed
347 348 349
    }

    if (req.count < 2) {
350
        av_log(ctx, AV_LOG_ERROR, "Insufficient buffer memory\n");
351
        return AVERROR(ENOMEM);
Luca Abeni's avatar
Luca Abeni committed
352 353 354 355
    }
    s->buffers = req.count;
    s->buf_start = av_malloc(sizeof(void *) * s->buffers);
    if (s->buf_start == NULL) {
356
        av_log(ctx, AV_LOG_ERROR, "Cannot allocate buffer pointers\n");
357
        return AVERROR(ENOMEM);
Luca Abeni's avatar
Luca Abeni committed
358 359 360
    }
    s->buf_len = av_malloc(sizeof(unsigned int) * s->buffers);
    if (s->buf_len == NULL) {
361
        av_log(ctx, AV_LOG_ERROR, "Cannot allocate buffer sizes\n");
Luca Abeni's avatar
Luca Abeni committed
362
        av_free(s->buf_start);
363
        return AVERROR(ENOMEM);
Luca Abeni's avatar
Luca Abeni committed
364 365 366
    }

    for (i = 0; i < req.count; i++) {
367 368 369 370 371
        struct v4l2_buffer buf = {
            .type   = V4L2_BUF_TYPE_VIDEO_CAPTURE,
            .index  = i,
            .memory = V4L2_MEMORY_MMAP
        };
372 373 374 375
        if (v4l2_ioctl(s->fd, VIDIOC_QUERYBUF, &buf) < 0) {
            res = AVERROR(errno);
            av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_QUERYBUF): %s\n", av_err2str(res));
            return res;
Luca Abeni's avatar
Luca Abeni committed
376 377 378
        }

        s->buf_len[i] = buf.length;
379
        if (s->frame_size > 0 && s->buf_len[i] < s->frame_size) {
Luca Barbato's avatar
Luca Barbato committed
380
            av_log(ctx, AV_LOG_ERROR,
381
                   "buf_len[%d] = %d < expected frame size %d\n",
Luca Barbato's avatar
Luca Barbato committed
382
                   i, s->buf_len[i], s->frame_size);
383
            return AVERROR(ENOMEM);
Luca Abeni's avatar
Luca Abeni committed
384
        }
385
        s->buf_start[i] = v4l2_mmap(NULL, buf.length,
Luca Barbato's avatar
Luca Barbato committed
386 387 388
                               PROT_READ | PROT_WRITE, MAP_SHARED,
                               s->fd, buf.m.offset);

Luca Abeni's avatar
Luca Abeni committed
389
        if (s->buf_start[i] == MAP_FAILED) {
390 391 392
            res = AVERROR(errno);
            av_log(ctx, AV_LOG_ERROR, "mmap: %s\n", av_err2str(res));
            return res;
Luca Abeni's avatar
Luca Abeni committed
393 394 395 396 397 398
        }
    }

    return 0;
}

399 400 401 402 403 404 405 406
#if FF_API_DESTRUCT_PACKET
static void dummy_release_buffer(AVPacket *pkt)
{
    av_assert0(0);
}
#endif

static void mmap_release_buffer(void *opaque, uint8_t *data)
407
{
408
    struct v4l2_buffer buf = { 0 };
409
    int res;
410 411
    struct buff_data *buf_descriptor = opaque;
    struct video_data *s = buf_descriptor->s;
412

413 414 415 416 417
    buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    buf.memory = V4L2_MEMORY_MMAP;
    buf.index = buf_descriptor->index;
    av_free(buf_descriptor);

418
    if (v4l2_ioctl(s->fd, VIDIOC_QBUF, &buf) < 0) {
419
        res = AVERROR(errno);
Luca Barbato's avatar
Luca Barbato committed
420
        av_log(NULL, AV_LOG_ERROR, "ioctl(VIDIOC_QBUF): %s\n",
421
               av_err2str(res));
422
    }
423

424
    avpriv_atomic_int_add_and_fetch(&s->buffers_queued, 1);
425 426
}

427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452
#if HAVE_CLOCK_GETTIME && defined(CLOCK_MONOTONIC)
static int64_t av_gettime_monotonic(void)
{
    struct timespec tv;

    clock_gettime(CLOCK_MONOTONIC, &tv);
    return (int64_t)tv.tv_sec * 1000000 + tv.tv_nsec / 1000;
}
#endif

static int init_convert_timestamp(AVFormatContext *ctx, int64_t ts)
{
    struct video_data *s = ctx->priv_data;
    int64_t now;

    now = av_gettime();
    if (s->ts_mode == V4L_TS_ABS &&
        ts <= now + 1 * AV_TIME_BASE && ts >= now - 10 * AV_TIME_BASE) {
        av_log(ctx, AV_LOG_INFO, "Detected absolute timestamps\n");
        s->ts_mode = V4L_TS_CONVERT_READY;
        return 0;
    }
#if HAVE_CLOCK_GETTIME && defined(CLOCK_MONOTONIC)
    now = av_gettime_monotonic();
    if (s->ts_mode == V4L_TS_MONO2ABS ||
        (ts <= now + 1 * AV_TIME_BASE && ts >= now - 10 * AV_TIME_BASE)) {
453 454
        AVRational tb = {AV_TIME_BASE, 1};
        int64_t period = av_rescale_q(1, tb, ctx->streams[0]->avg_frame_rate);
455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486
        av_log(ctx, AV_LOG_INFO, "Detected monotonic timestamps, converting\n");
        /* microseconds instead of seconds, MHz instead of Hz */
        s->timefilter = ff_timefilter_new(1, period, 1.0E-6);
        s->ts_mode = V4L_TS_CONVERT_READY;
        return 0;
    }
#endif
    av_log(ctx, AV_LOG_ERROR, "Unknown timestamps\n");
    return AVERROR(EIO);
}

static int convert_timestamp(AVFormatContext *ctx, int64_t *ts)
{
    struct video_data *s = ctx->priv_data;

    if (s->ts_mode) {
        int r = init_convert_timestamp(ctx, *ts);
        if (r < 0)
            return r;
    }
#if HAVE_CLOCK_GETTIME && defined(CLOCK_MONOTONIC)
    if (s->timefilter) {
        int64_t nowa = av_gettime();
        int64_t nowm = av_gettime_monotonic();
        ff_timefilter_update(s->timefilter, nowa, nowm - s->last_time_m);
        s->last_time_m = nowm;
        *ts = ff_timefilter_eval(s->timefilter, *ts - nowm);
    }
#endif
    return 0;
}

487
static int mmap_read_frame(AVFormatContext *ctx, AVPacket *pkt)
Luca Abeni's avatar
Luca Abeni committed
488
{
489
    struct video_data *s = ctx->priv_data;
490 491 492 493
    struct v4l2_buffer buf = {
        .type   = V4L2_BUF_TYPE_VIDEO_CAPTURE,
        .memory = V4L2_MEMORY_MMAP
    };
Luca Abeni's avatar
Luca Abeni committed
494 495 496
    int res;

    /* FIXME: Some special treatment might be needed in case of loss of signal... */
497
    while ((res = v4l2_ioctl(s->fd, VIDIOC_DQBUF, &buf)) < 0 && (errno == EINTR));
Luca Abeni's avatar
Luca Abeni committed
498
    if (res < 0) {
499 500 501 502
        if (errno == EAGAIN) {
            pkt->size = 0;
            return AVERROR(EAGAIN);
        }
503 504 505
        res = AVERROR(errno);
        av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_DQBUF): %s\n", av_err2str(res));
        return res;
Luca Abeni's avatar
Luca Abeni committed
506
    }
507 508 509 510 511

    if (buf.index >= s->buffers) {
        av_log(ctx, AV_LOG_ERROR, "Invalid buffer index received.\n");
        return AVERROR(EINVAL);
    }
512 513 514
    avpriv_atomic_int_add_and_fetch(&s->buffers_queued, -1);
    // always keep at least one buffer queued
    av_assert0(avpriv_atomic_int_get(&s->buffers_queued) >= 1);
Stephan Hilb's avatar
Stephan Hilb committed
515 516 517 518 519 520 521

    /* CPIA is a compressed format and we don't know the exact number of bytes
     * used by a frame, so set it here as the driver announces it.
     */
    if (ctx->video_codec_id == AV_CODEC_ID_CPIA)
        s->frame_size = buf.bytesused;

522
    if (s->frame_size > 0 && buf.bytesused != s->frame_size) {
Luca Barbato's avatar
Luca Barbato committed
523 524 525
        av_log(ctx, AV_LOG_ERROR,
               "The v4l2 frame is %d bytes, but %d bytes are expected\n",
               buf.bytesused, s->frame_size);
526
        return AVERROR_INVALIDDATA;
527 528
    }

Luca Abeni's avatar
Luca Abeni committed
529
    /* Image is at s->buff_start[buf.index] */
530
    if (avpriv_atomic_int_get(&s->buffers_queued) == FFMAX(s->buffers / 8, 1)) {
Diego Biurrun's avatar
Diego Biurrun committed
531
        /* when we start getting low on queued buffers, fall back on copying data */
532 533 534
        res = av_new_packet(pkt, buf.bytesused);
        if (res < 0) {
            av_log(ctx, AV_LOG_ERROR, "Error allocating a packet.\n");
535 536
            if (v4l2_ioctl(s->fd, VIDIOC_QBUF, &buf) == 0)
                avpriv_atomic_int_add_and_fetch(&s->buffers_queued, 1);
537 538 539
            return res;
        }
        memcpy(pkt->data, s->buf_start[buf.index], buf.bytesused);
540

541 542 543
        if (v4l2_ioctl(s->fd, VIDIOC_QBUF, &buf) < 0) {
            res = AVERROR(errno);
            av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_QBUF): %s\n", av_err2str(res));
544
            av_free_packet(pkt);
545
            return res;
546 547
        }
        avpriv_atomic_int_add_and_fetch(&s->buffers_queued, 1);
548
    } else {
549
        struct buff_data *buf_descriptor;
Luca Abeni's avatar
Luca Abeni committed
550

551 552 553
        pkt->data     = s->buf_start[buf.index];
        pkt->size     = buf.bytesused;
#if FF_API_DESTRUCT_PACKET
554
FF_DISABLE_DEPRECATION_WARNINGS
555
        pkt->destruct = dummy_release_buffer;
556
FF_ENABLE_DEPRECATION_WARNINGS
557 558 559 560 561 562 563 564
#endif

        buf_descriptor = av_malloc(sizeof(struct buff_data));
        if (buf_descriptor == NULL) {
            /* Something went wrong... Since av_malloc() failed, we cannot even
             * allocate a buffer for memcpying into it
             */
            av_log(ctx, AV_LOG_ERROR, "Failed to allocate a buffer descriptor\n");
565 566
            if (v4l2_ioctl(s->fd, VIDIOC_QBUF, &buf) == 0)
                avpriv_atomic_int_add_and_fetch(&s->buffers_queued, 1);
567 568 569

            return AVERROR(ENOMEM);
        }
570
        buf_descriptor->index = buf.index;
571 572 573 574 575
        buf_descriptor->s     = s;

        pkt->buf = av_buffer_create(pkt->data, pkt->size, mmap_release_buffer,
                                    buf_descriptor, 0);
        if (!pkt->buf) {
576 577 578
            av_log(ctx, AV_LOG_ERROR, "Failed to create a buffer\n");
            if (v4l2_ioctl(s->fd, VIDIOC_QBUF, &buf) == 0)
                avpriv_atomic_int_add_and_fetch(&s->buffers_queued, 1);
579 580 581
            av_freep(&buf_descriptor);
            return AVERROR(ENOMEM);
        }
582 583
    }
    pkt->pts = buf.timestamp.tv_sec * INT64_C(1000000) + buf.timestamp.tv_usec;
584
    convert_timestamp(ctx, &pkt->pts);
Luca Abeni's avatar
Luca Abeni committed
585 586 587 588

    return s->buf_len[buf.index];
}

589
static int mmap_start(AVFormatContext *ctx)
Luca Abeni's avatar
Luca Abeni committed
590
{
591
    struct video_data *s = ctx->priv_data;
Luca Abeni's avatar
Luca Abeni committed
592 593 594 595
    enum v4l2_buf_type type;
    int i, res;

    for (i = 0; i < s->buffers; i++) {
596 597 598 599 600
        struct v4l2_buffer buf = {
            .type   = V4L2_BUF_TYPE_VIDEO_CAPTURE,
            .index  = i,
            .memory = V4L2_MEMORY_MMAP
        };
Luca Abeni's avatar
Luca Abeni committed
601

602 603 604 605
        if (v4l2_ioctl(s->fd, VIDIOC_QBUF, &buf) < 0) {
            res = AVERROR(errno);
            av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_QBUF): %s\n", av_err2str(res));
            return res;
Luca Abeni's avatar
Luca Abeni committed
606 607
        }
    }
608
    s->buffers_queued = s->buffers;
Luca Abeni's avatar
Luca Abeni committed
609 610

    type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
611 612 613 614
    if (v4l2_ioctl(s->fd, VIDIOC_STREAMON, &type) < 0) {
        res = AVERROR(errno);
        av_log(ctx, AV_LOG_ERROR, "ioctl(VIDIOC_STREAMON): %s\n", av_err2str(res));
        return res;
Luca Abeni's avatar
Luca Abeni committed
615 616 617 618 619 620 621 622 623 624 625 626 627 628
    }

    return 0;
}

static void mmap_close(struct video_data *s)
{
    enum v4l2_buf_type type;
    int i;

    type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    /* We do not check for the result, because we could
     * not do anything about it anyway...
     */
629
    v4l2_ioctl(s->fd, VIDIOC_STREAMOFF, &type);
Luca Abeni's avatar
Luca Abeni committed
630
    for (i = 0; i < s->buffers; i++) {
631
        v4l2_munmap(s->buf_start[i], s->buf_len[i]);
Luca Abeni's avatar
Luca Abeni committed
632 633 634 635 636
    }
    av_free(s->buf_start);
    av_free(s->buf_len);
}

637
static int v4l2_set_parameters(AVFormatContext *s1)
638 639
{
    struct video_data *s = s1->priv_data;
640
    struct v4l2_standard standard = { 0 };
641
    struct v4l2_streamparm streamparm = { 0 };
642
    struct v4l2_fract *tpf;
643
    AVRational framerate_q = { 0 };
644
    int i, ret;
645

Luca Barbato's avatar
Luca Barbato committed
646 647 648 649
    if (s->framerate &&
        (ret = av_parse_video_rate(&framerate_q, s->framerate)) < 0) {
        av_log(s1, AV_LOG_ERROR, "Could not parse framerate '%s'.\n",
               s->framerate);
650 651
        return ret;
    }
652

653
    if (s->standard) {
654
        if (s->std_id) {
655
            ret = 0;
656 657 658 659
            av_log(s1, AV_LOG_DEBUG, "Setting standard: %s\n", s->standard);
            /* set tv standard */
            for (i = 0; ; i++) {
                standard.index = i;
660
                if (v4l2_ioctl(s->fd, VIDIOC_ENUMSTD, &standard) < 0) {
661
                    ret = AVERROR(errno);
662 663 664
                    break;
                }
                if (!av_strcasecmp(standard.name, s->standard))
665 666 667 668
                    break;
            }
            if (ret < 0) {
                av_log(s1, AV_LOG_ERROR, "Unknown or unsupported standard '%s'\n", s->standard);
669
                return ret;
670 671 672
            }

            if (v4l2_ioctl(s->fd, VIDIOC_S_STD, &standard.id) < 0) {
673 674 675
                ret = AVERROR(errno);
                av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_S_STD): %s\n", av_err2str(ret));
                return ret;
676 677 678 679 680 681 682 683 684 685 686
            }
        } else {
            av_log(s1, AV_LOG_WARNING,
                   "This device does not support any standard\n");
        }
    }

    /* get standard */
    if (v4l2_ioctl(s->fd, VIDIOC_G_STD, &s->std_id) == 0) {
        tpf = &standard.frameperiod;
        for (i = 0; ; i++) {
687
            standard.index = i;
688 689 690 691
            if (v4l2_ioctl(s->fd, VIDIOC_ENUMSTD, &standard) < 0) {
                ret = AVERROR(errno);
                av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_ENUMSTD): %s\n", av_err2str(ret));
                return ret;
692 693 694 695 696
            }
            if (standard.id == s->std_id) {
                av_log(s1, AV_LOG_DEBUG,
                       "Current standard: %s, id: %"PRIu64", frameperiod: %d/%d\n",
                       standard.name, (uint64_t)standard.id, tpf->numerator, tpf->denominator);
697
                break;
698
            }
699
        }
700 701 702
    } else {
        tpf = &streamparm.parm.capture.timeperframe;
    }
703

704 705
    streamparm.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    if (v4l2_ioctl(s->fd, VIDIOC_G_PARM, &streamparm) < 0) {
706 707 708
        ret = AVERROR(errno);
        av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_G_PARM): %s\n", av_err2str(ret));
        return ret;
709
    }
710

711
    if (framerate_q.num && framerate_q.den) {
712 713 714 715
        if (streamparm.parm.capture.capability & V4L2_CAP_TIMEPERFRAME) {
            tpf = &streamparm.parm.capture.timeperframe;

            av_log(s1, AV_LOG_DEBUG, "Setting time per frame to %d/%d\n",
716
                   framerate_q.den, framerate_q.num);
717 718
            tpf->numerator   = framerate_q.den;
            tpf->denominator = framerate_q.num;
719

720
            if (v4l2_ioctl(s->fd, VIDIOC_S_PARM, &streamparm) < 0) {
721 722 723
                ret = AVERROR(errno);
                av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_S_PARM): %s\n", av_err2str(ret));
                return ret;
724 725 726 727 728 729 730 731 732 733 734 735 736
            }

            if (framerate_q.num != tpf->denominator ||
                framerate_q.den != tpf->numerator) {
                av_log(s1, AV_LOG_INFO,
                       "The driver changed the time per frame from "
                       "%d/%d to %d/%d\n",
                       framerate_q.den, framerate_q.num,
                       tpf->numerator, tpf->denominator);
            }
        } else {
            av_log(s1, AV_LOG_WARNING,
                   "The driver does not allow to change time per frame\n");
737
        }
738
    }
739 740
    s1->streams[0]->avg_frame_rate.num = tpf->denominator;
    s1->streams[0]->avg_frame_rate.den = tpf->numerator;
741
    s1->streams[0]->r_frame_rate = s1->streams[0]->avg_frame_rate;
742

743 744 745
    return 0;
}

746 747 748 749 750 751
static int device_try_init(AVFormatContext *s1,
                           enum AVPixelFormat pix_fmt,
                           int *width,
                           int *height,
                           uint32_t *desired_format,
                           enum AVCodecID *codec_id)
752
{
753
    int ret, i;
754

755
    *desired_format = avpriv_fmt_ff2v4l(pix_fmt, s1->video_codec_id);
756

757 758 759 760 761 762 763 764 765 766
    if (*desired_format) {
        ret = device_init(s1, width, height, *desired_format);
        if (ret < 0) {
            *desired_format = 0;
            if (ret != AVERROR(EINVAL))
                return ret;
        }
    }

    if (!*desired_format) {
767
        for (i = 0; avpriv_fmt_conversion_table[i].codec_id != AV_CODEC_ID_NONE; i++) {
768
            if (s1->video_codec_id == AV_CODEC_ID_NONE ||
769
                avpriv_fmt_conversion_table[i].codec_id == s1->video_codec_id) {
770
                av_log(s1, AV_LOG_DEBUG, "Trying to set codec:%s pix_fmt:%s\n",
771 772
                       avcodec_get_name(avpriv_fmt_conversion_table[i].codec_id),
                       (char *)av_x_if_null(av_get_pix_fmt_name(avpriv_fmt_conversion_table[i].ff_fmt), "none"));
773

774
                *desired_format = avpriv_fmt_conversion_table[i].v4l2_fmt;
775 776
                ret = device_init(s1, width, height, *desired_format);
                if (ret >= 0)
777
                    break;
778 779 780
                else if (ret != AVERROR(EINVAL))
                    return ret;
                *desired_format = 0;
781 782
            }
        }
Luca Barbato's avatar
Luca Barbato committed
783

784 785 786 787 788 789 790
        if (*desired_format == 0) {
            av_log(s1, AV_LOG_ERROR, "Cannot find a proper format for "
                   "codec '%s' (id %d), pixel format '%s' (id %d)\n",
                   avcodec_get_name(s1->video_codec_id), s1->video_codec_id,
                   (char *)av_x_if_null(av_get_pix_fmt_name(pix_fmt), "none"), pix_fmt);
            ret = AVERROR(EINVAL);
        }
791 792
    }

793
    *codec_id = avpriv_fmt_v4l2codec(*desired_format);
794 795
    av_assert0(*codec_id != AV_CODEC_ID_NONE);
    return ret;
796 797
}

798
static int v4l2_read_header(AVFormatContext *s1)
Luca Abeni's avatar
Luca Abeni committed
799 800 801
{
    struct video_data *s = s1->priv_data;
    AVStream *st;
802
    int res = 0;
803
    uint32_t desired_format;
Stephan Hilb's avatar
Stephan Hilb committed
804
    enum AVCodecID codec_id = AV_CODEC_ID_NONE;
805
    enum AVPixelFormat pix_fmt = AV_PIX_FMT_NONE;
806
    struct v4l2_input input = { 0 };
Luca Abeni's avatar
Luca Abeni committed
807

808
    st = avformat_new_stream(s1, NULL);
809 810
    if (!st)
        return AVERROR(ENOMEM);
Luca Abeni's avatar
Luca Abeni committed
811

812 813 814
#if CONFIG_LIBV4L2
    /* silence libv4l2 logging. if fopen() fails v4l2_log_file will be NULL
       and errors will get sent to stderr */
815 816
    if (s->use_libv4l2)
        v4l2_log_file = fopen("/dev/null", "w");
817 818
#endif

819
    s->fd = device_open(s1);
820 821
    if (s->fd < 0)
        return s->fd;
822

823 824 825 826 827 828 829 830 831 832 833 834 835 836 837
    if (s->channel != -1) {
        /* set video input */
        av_log(s1, AV_LOG_DEBUG, "Selecting input_channel: %d\n", s->channel);
        if (v4l2_ioctl(s->fd, VIDIOC_S_INPUT, &s->channel) < 0) {
            res = AVERROR(errno);
            av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_S_INPUT): %s\n", av_err2str(res));
            return res;
        }
    } else {
        /* get current video input */
        if (v4l2_ioctl(s->fd, VIDIOC_G_INPUT, &s->channel) < 0) {
            res = AVERROR(errno);
            av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_G_INPUT): %s\n", av_err2str(res));
            return res;
        }
838 839
    }

840
    /* enum input */
841 842
    input.index = s->channel;
    if (v4l2_ioctl(s->fd, VIDIOC_ENUMINPUT, &input) < 0) {
843 844 845
        res = AVERROR(errno);
        av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_ENUMINPUT): %s\n", av_err2str(res));
        return res;
846 847
    }
    s->std_id = input.std;
848
    av_log(s1, AV_LOG_DEBUG, "Current input_channel: %d, input_name: %s\n",
849 850
           s->channel, input.name);

851 852
    if (s->list_format) {
        list_formats(s1, s->fd, s->list_format);
853
        return AVERROR_EXIT;
854
    }
Luca Abeni's avatar
Luca Abeni committed
855

856 857 858 859 860
    if (s->list_standard) {
        list_standards(s1);
        return AVERROR_EXIT;
    }

861
    avpriv_set_pts_info(st, 64, 1, 1000000); /* 64 bits pts in us */
Luca Abeni's avatar
Luca Abeni committed
862

863
    if (s->pixel_format) {
864 865 866 867
        AVCodec *codec = avcodec_find_decoder_by_name(s->pixel_format);

        if (codec)
            s1->video_codec_id = codec->id;
868 869 870

        pix_fmt = av_get_pix_fmt(s->pixel_format);

871
        if (pix_fmt == AV_PIX_FMT_NONE && !codec) {
872
            av_log(s1, AV_LOG_ERROR, "No such input format: %s.\n",
873 874
                   s->pixel_format);

875
            return AVERROR(EINVAL);
876
        }
877
    }
Luca Abeni's avatar
Luca Abeni committed
878

879
    if (!s->width && !s->height) {
880
        struct v4l2_format fmt = { .type = V4L2_BUF_TYPE_VIDEO_CAPTURE };
881

Luca Barbato's avatar
Luca Barbato committed
882 883
        av_log(s1, AV_LOG_VERBOSE,
               "Querying the device for the current frame size\n");
884
        if (v4l2_ioctl(s->fd, VIDIOC_G_FMT, &fmt) < 0) {
885 886 887
            res = AVERROR(errno);
            av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_G_FMT): %s\n", av_err2str(res));
            return res;
888
        }
Luca Barbato's avatar
Luca Barbato committed
889

890 891
        s->width  = fmt.fmt.pix.width;
        s->height = fmt.fmt.pix.height;
Luca Barbato's avatar
Luca Barbato committed
892 893
        av_log(s1, AV_LOG_VERBOSE,
               "Setting frame size to %dx%d\n", s->width, s->height);
894 895
    }

896 897 898 899 900
    res = device_try_init(s1, pix_fmt, &s->width, &s->height, &desired_format, &codec_id);
    if (res < 0) {
        v4l2_close(s->fd);
        return res;
    }
Stephan Hilb's avatar
Stephan Hilb committed
901 902 903 904 905 906 907 908

    /* If no pixel_format was specified, the codec_id was not known up
     * until now. Set video_codec_id in the context, as codec_id will
     * not be available outside this function
     */
    if (codec_id != AV_CODEC_ID_NONE && s1->video_codec_id == AV_CODEC_ID_NONE)
        s1->video_codec_id = codec_id;

909
    if ((res = av_image_check_size(s->width, s->height, 0, s1)) < 0)
910
        return res;
Luca Barbato's avatar
Luca Barbato committed
911

Luca Abeni's avatar
Luca Abeni committed
912 913
    s->frame_format = desired_format;

914
    if ((res = v4l2_set_parameters(s1)) < 0)
915
        return res;
916

917
    st->codec->pix_fmt = avpriv_fmt_v4l2ff(desired_format, codec_id);
Luca Barbato's avatar
Luca Barbato committed
918 919 920
    s->frame_size =
        avpicture_get_size(st->codec->pix_fmt, s->width, s->height);

921 922
    if ((res = mmap_init(s1)) ||
        (res = mmap_start(s1)) < 0) {
923
        v4l2_close(s->fd);
924
        return res;
Luca Abeni's avatar
Luca Abeni committed
925
    }
926

927
    s->top_field_first = first_field(s, s->fd);
Luca Abeni's avatar
Luca Abeni committed
928

929
    st->codec->codec_type = AVMEDIA_TYPE_VIDEO;
930
    st->codec->codec_id = codec_id;
931
    if (codec_id == AV_CODEC_ID_RAWVIDEO)
932 933
        st->codec->codec_tag =
            avcodec_pix_fmt_to_codec_tag(st->codec->pix_fmt);
934 935
    if (desired_format == V4L2_PIX_FMT_YVU420)
        st->codec->codec_tag = MKTAG('Y', 'V', '1', '2');
936 937
    else if (desired_format == V4L2_PIX_FMT_YVU410)
        st->codec->codec_tag = MKTAG('Y', 'V', 'U', '9');
938 939
    st->codec->width = s->width;
    st->codec->height = s->height;
940
    st->codec->bit_rate = s->frame_size * av_q2d(st->avg_frame_rate) * 8;
Luca Abeni's avatar
Luca Abeni committed
941

942
    return 0;
Luca Abeni's avatar
Luca Abeni committed
943 944 945 946 947
}

static int v4l2_read_packet(AVFormatContext *s1, AVPacket *pkt)
{
    struct video_data *s = s1->priv_data;
Luca Barbato's avatar
Luca Barbato committed
948
    AVFrame *frame = s1->streams[0]->codec->coded_frame;
Luca Abeni's avatar
Luca Abeni committed
949 950
    int res;

951 952
    av_init_packet(pkt);
    if ((res = mmap_read_frame(s1, pkt)) < 0) {
953
        return res;
Luca Abeni's avatar
Luca Abeni committed
954 955
    }

Luca Barbato's avatar
Luca Barbato committed
956 957 958
    if (frame && s->interlaced) {
        frame->interlaced_frame = 1;
        frame->top_field_first = s->top_field_first;
Luca Abeni's avatar
Luca Abeni committed
959 960
    }

961
    return pkt->size;
Luca Abeni's avatar
Luca Abeni committed
962 963 964 965 966 967
}

static int v4l2_read_close(AVFormatContext *s1)
{
    struct video_data *s = s1->priv_data;

968 969 970 971
    if (avpriv_atomic_int_get(&s->buffers_queued) != s->buffers)
        av_log(s1, AV_LOG_WARNING, "Some buffers are still owned by the caller on "
               "close.\n");

972
    mmap_close(s);
Luca Abeni's avatar
Luca Abeni committed
973

974
    v4l2_close(s->fd);
Luca Abeni's avatar
Luca Abeni committed
975 976 977
    return 0;
}

978 979
#define OFFSET(x) offsetof(struct video_data, x)
#define DEC AV_OPT_FLAG_DECODING_PARAM
980

981
static const AVOption options[] = {
982
    { "standard",     "set TV standard, used only by analog frame grabber",       OFFSET(standard),     AV_OPT_TYPE_STRING, {.str = NULL }, 0, 0,       DEC },
983
    { "channel",      "set TV channel, used only by frame grabber",               OFFSET(channel),      AV_OPT_TYPE_INT,    {.i64 = -1 },  -1, INT_MAX, DEC },
984 985 986 987 988 989 990 991 992 993
    { "video_size",   "set frame size",                                           OFFSET(width),        AV_OPT_TYPE_IMAGE_SIZE, {.str = NULL},  0, 0,   DEC },
    { "pixel_format", "set preferred pixel format",                               OFFSET(pixel_format), AV_OPT_TYPE_STRING, {.str = NULL},  0, 0,       DEC },
    { "input_format", "set preferred pixel format (for raw video) or codec name", OFFSET(pixel_format), AV_OPT_TYPE_STRING, {.str = NULL},  0, 0,       DEC },
    { "framerate",    "set frame rate",                                           OFFSET(framerate),    AV_OPT_TYPE_STRING, {.str = NULL},  0, 0,       DEC },

    { "list_formats", "list available formats and exit",                          OFFSET(list_format),  AV_OPT_TYPE_INT,    {.i64 = 0 },  0, INT_MAX, DEC, "list_formats" },
    { "all",          "show all available formats",                               OFFSET(list_format),  AV_OPT_TYPE_CONST,  {.i64 = V4L_ALLFORMATS  },    0, INT_MAX, DEC, "list_formats" },
    { "raw",          "show only non-compressed formats",                         OFFSET(list_format),  AV_OPT_TYPE_CONST,  {.i64 = V4L_RAWFORMATS  },    0, INT_MAX, DEC, "list_formats" },
    { "compressed",   "show only compressed formats",                             OFFSET(list_format),  AV_OPT_TYPE_CONST,  {.i64 = V4L_COMPFORMATS },    0, INT_MAX, DEC, "list_formats" },

994 995 996
    { "list_standards", "list supported standards and exit",                      OFFSET(list_standard), AV_OPT_TYPE_INT,   {.i64 = 0 },  0, 1, DEC, "list_standards" },
    { "all",            "show all supported standards",                           OFFSET(list_standard), AV_OPT_TYPE_CONST, {.i64 = 1 },  0, 0, DEC, "list_standards" },

997 998 999 1000 1001
    { "timestamps",   "set type of timestamps for grabbed frames",                OFFSET(ts_mode),      AV_OPT_TYPE_INT,    {.i64 = 0 }, 0, 2, DEC, "timestamps" },
    { "ts",           "set type of timestamps for grabbed frames",                OFFSET(ts_mode),      AV_OPT_TYPE_INT,    {.i64 = 0 }, 0, 2, DEC, "timestamps" },
    { "default",      "use timestamps from the kernel",                           OFFSET(ts_mode),      AV_OPT_TYPE_CONST,  {.i64 = V4L_TS_DEFAULT  }, 0, 2, DEC, "timestamps" },
    { "abs",          "use absolute timestamps (wall clock)",                     OFFSET(ts_mode),      AV_OPT_TYPE_CONST,  {.i64 = V4L_TS_ABS      }, 0, 2, DEC, "timestamps" },
    { "mono2abs",     "force conversion from monotonic to absolute timestamps",   OFFSET(ts_mode),      AV_OPT_TYPE_CONST,  {.i64 = V4L_TS_MONO2ABS }, 0, 2, DEC, "timestamps" },
1002
    { "use_libv4l2",  "use libv4l2 (v4l-utils) convertion functions",             OFFSET(use_libv4l2),  AV_OPT_TYPE_INT,    {.i64 = 0}, 0, 1, DEC },
1003 1004 1005 1006 1007 1008 1009 1010 1011 1012
    { NULL },
};

static const AVClass v4l2_class = {
    .class_name = "V4L2 indev",
    .item_name  = av_default_item_name,
    .option     = options,
    .version    = LIBAVUTIL_VERSION_INT,
};

1013
AVInputFormat ff_v4l2_demuxer = {
1014
    .name           = "video4linux2,v4l2",
1015 1016 1017 1018 1019 1020 1021
    .long_name      = NULL_IF_CONFIG_SMALL("Video4Linux2 device grab"),
    .priv_data_size = sizeof(struct video_data),
    .read_header    = v4l2_read_header,
    .read_packet    = v4l2_read_packet,
    .read_close     = v4l2_read_close,
    .flags          = AVFMT_NOFILE,
    .priv_class     = &v4l2_class,
Luca Abeni's avatar
Luca Abeni committed
1022
};