v4l2.c 34.6 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
                return;
            }
        }
328 329
        av_log(ctx, AV_LOG_INFO, "%2d, %16"PRIx64", %s\n",
               standard.index, (uint64_t)standard.id, standard.name);
330 331 332
    }
}

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
#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)
450 451 452 453 454 455 456 457 458 459 460 461 462 463
    if (ctx->streams[0]->avg_frame_rate.num) {
        now = av_gettime_monotonic();
        if (s->ts_mode == V4L_TS_MONO2ABS ||
            (ts <= now + 1 * AV_TIME_BASE && ts >= now - 10 * AV_TIME_BASE)) {
            AVRational tb = {AV_TIME_BASE, 1};
            int64_t period = av_rescale_q(1, tb, ctx->streams[0]->avg_frame_rate);
            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);
            if (!s->timefilter)
                return AVERROR(ENOMEM);
            s->ts_mode = V4L_TS_CONVERT_READY;
            return 0;
        }
464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490
    }
#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;
}

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

    /* FIXME: Some special treatment might be needed in case of loss of signal... */
501
    while ((res = v4l2_ioctl(s->fd, VIDIOC_DQBUF, &buf)) < 0 && (errno == EINTR));
Luca Abeni's avatar
Luca Abeni committed
502
    if (res < 0) {
503 504 505 506
        if (errno == EAGAIN) {
            pkt->size = 0;
            return AVERROR(EAGAIN);
        }
507 508 509
        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
510
    }
511 512 513 514 515

    if (buf.index >= s->buffers) {
        av_log(ctx, AV_LOG_ERROR, "Invalid buffer index received.\n");
        return AVERROR(EINVAL);
    }
516 517 518
    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
519 520 521 522 523 524 525

    /* 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;

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

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

545 546 547
        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));
548
            av_free_packet(pkt);
549
            return res;
550 551
        }
        avpriv_atomic_int_add_and_fetch(&s->buffers_queued, 1);
552
    } else {
553
        struct buff_data *buf_descriptor;
Luca Abeni's avatar
Luca Abeni committed
554

555 556 557
        pkt->data     = s->buf_start[buf.index];
        pkt->size     = buf.bytesused;
#if FF_API_DESTRUCT_PACKET
558
FF_DISABLE_DEPRECATION_WARNINGS
559
        pkt->destruct = dummy_release_buffer;
560
FF_ENABLE_DEPRECATION_WARNINGS
561 562 563 564 565 566 567 568
#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");
569 570
            if (v4l2_ioctl(s->fd, VIDIOC_QBUF, &buf) == 0)
                avpriv_atomic_int_add_and_fetch(&s->buffers_queued, 1);
571 572 573

            return AVERROR(ENOMEM);
        }
574
        buf_descriptor->index = buf.index;
575 576 577 578 579
        buf_descriptor->s     = s;

        pkt->buf = av_buffer_create(pkt->data, pkt->size, mmap_release_buffer,
                                    buf_descriptor, 0);
        if (!pkt->buf) {
580 581 582
            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);
583 584 585
            av_freep(&buf_descriptor);
            return AVERROR(ENOMEM);
        }
586 587
    }
    pkt->pts = buf.timestamp.tv_sec * INT64_C(1000000) + buf.timestamp.tv_usec;
588
    convert_timestamp(ctx, &pkt->pts);
Luca Abeni's avatar
Luca Abeni committed
589 590 591 592

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

593
static int mmap_start(AVFormatContext *ctx)
Luca Abeni's avatar
Luca Abeni committed
594
{
595
    struct video_data *s = ctx->priv_data;
Luca Abeni's avatar
Luca Abeni committed
596 597 598 599
    enum v4l2_buf_type type;
    int i, res;

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

606 607 608 609
        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
610 611
        }
    }
612
    s->buffers_queued = s->buffers;
Luca Abeni's avatar
Luca Abeni committed
613 614

    type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
615 616 617 618
    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
619 620 621 622 623 624 625 626 627 628 629 630 631 632
    }

    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...
     */
633
    v4l2_ioctl(s->fd, VIDIOC_STREAMOFF, &type);
Luca Abeni's avatar
Luca Abeni committed
634
    for (i = 0; i < s->buffers; i++) {
635
        v4l2_munmap(s->buf_start[i], s->buf_len[i]);
Luca Abeni's avatar
Luca Abeni committed
636 637 638 639 640
    }
    av_free(s->buf_start);
    av_free(s->buf_len);
}

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

Luca Barbato's avatar
Luca Barbato committed
650 651 652 653
    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);
654 655
        return ret;
    }
656

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

            if (v4l2_ioctl(s->fd, VIDIOC_S_STD, &standard.id) < 0) {
677 678 679
                ret = AVERROR(errno);
                av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_S_STD): %s\n", av_err2str(ret));
                return ret;
680 681 682 683 684 685 686 687 688 689 690
            }
        } 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++) {
691
            standard.index = i;
692 693
            if (v4l2_ioctl(s->fd, VIDIOC_ENUMSTD, &standard) < 0) {
                ret = AVERROR(errno);
694 695 696 697
                if (ret == AVERROR(EINVAL)) {
                    tpf = &streamparm.parm.capture.timeperframe;
                    break;
                }
698 699
                av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_ENUMSTD): %s\n", av_err2str(ret));
                return ret;
700 701 702
            }
            if (standard.id == s->std_id) {
                av_log(s1, AV_LOG_DEBUG,
703
                       "Current standard: %s, id: %"PRIx64", frameperiod: %d/%d\n",
704
                       standard.name, (uint64_t)standard.id, tpf->numerator, tpf->denominator);
705
                break;
706
            }
707
        }
708 709 710
    } else {
        tpf = &streamparm.parm.capture.timeperframe;
    }
711

712 713
    streamparm.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    if (v4l2_ioctl(s->fd, VIDIOC_G_PARM, &streamparm) < 0) {
714 715 716
        ret = AVERROR(errno);
        av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_G_PARM): %s\n", av_err2str(ret));
        return ret;
717
    }
718

719
    if (framerate_q.num && framerate_q.den) {
720 721 722 723
        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",
724
                   framerate_q.den, framerate_q.num);
725 726
            tpf->numerator   = framerate_q.den;
            tpf->denominator = framerate_q.num;
727

728
            if (v4l2_ioctl(s->fd, VIDIOC_S_PARM, &streamparm) < 0) {
729 730 731
                ret = AVERROR(errno);
                av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_S_PARM): %s\n", av_err2str(ret));
                return ret;
732 733 734 735 736 737 738 739 740 741 742 743 744
            }

            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");
745
        }
746
    }
747 748 749 750 751 752
    if (tpf->denominator > 0 && tpf->numerator > 0) {
        s1->streams[0]->avg_frame_rate.num = tpf->denominator;
        s1->streams[0]->avg_frame_rate.den = tpf->numerator;
        s1->streams[0]->r_frame_rate = s1->streams[0]->avg_frame_rate;
    } else
        av_log(s1, AV_LOG_WARNING, "Time per frame unknown\n");
753

754 755 756
    return 0;
}

757 758 759 760 761 762
static int device_try_init(AVFormatContext *s1,
                           enum AVPixelFormat pix_fmt,
                           int *width,
                           int *height,
                           uint32_t *desired_format,
                           enum AVCodecID *codec_id)
763
{
764
    int ret, i;
765

766
    *desired_format = avpriv_fmt_ff2v4l(pix_fmt, s1->video_codec_id);
767

768 769 770 771 772 773 774 775 776 777
    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) {
778
        for (i = 0; avpriv_fmt_conversion_table[i].codec_id != AV_CODEC_ID_NONE; i++) {
779
            if (s1->video_codec_id == AV_CODEC_ID_NONE ||
780
                avpriv_fmt_conversion_table[i].codec_id == s1->video_codec_id) {
781
                av_log(s1, AV_LOG_DEBUG, "Trying to set codec:%s pix_fmt:%s\n",
782 783
                       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"));
784

785
                *desired_format = avpriv_fmt_conversion_table[i].v4l2_fmt;
786 787
                ret = device_init(s1, width, height, *desired_format);
                if (ret >= 0)
788
                    break;
789 790 791
                else if (ret != AVERROR(EINVAL))
                    return ret;
                *desired_format = 0;
792 793
            }
        }
Luca Barbato's avatar
Luca Barbato committed
794

795 796 797 798 799 800 801
        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);
        }
802 803
    }

804
    *codec_id = avpriv_fmt_v4l2codec(*desired_format);
805 806
    av_assert0(*codec_id != AV_CODEC_ID_NONE);
    return ret;
807 808
}

809
static int v4l2_read_header(AVFormatContext *s1)
Luca Abeni's avatar
Luca Abeni committed
810 811 812
{
    struct video_data *s = s1->priv_data;
    AVStream *st;
813
    int res = 0;
814
    uint32_t desired_format;
Stephan Hilb's avatar
Stephan Hilb committed
815
    enum AVCodecID codec_id = AV_CODEC_ID_NONE;
816
    enum AVPixelFormat pix_fmt = AV_PIX_FMT_NONE;
817
    struct v4l2_input input = { 0 };
Luca Abeni's avatar
Luca Abeni committed
818

819
    st = avformat_new_stream(s1, NULL);
820 821
    if (!st)
        return AVERROR(ENOMEM);
Luca Abeni's avatar
Luca Abeni committed
822

823 824 825
#if CONFIG_LIBV4L2
    /* silence libv4l2 logging. if fopen() fails v4l2_log_file will be NULL
       and errors will get sent to stderr */
826 827
    if (s->use_libv4l2)
        v4l2_log_file = fopen("/dev/null", "w");
828 829
#endif

830
    s->fd = device_open(s1);
831 832
    if (s->fd < 0)
        return s->fd;
833

834 835 836 837 838 839 840 841 842 843 844 845 846 847 848
    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;
        }
849 850
    }

851
    /* enum input */
852 853
    input.index = s->channel;
    if (v4l2_ioctl(s->fd, VIDIOC_ENUMINPUT, &input) < 0) {
854 855 856
        res = AVERROR(errno);
        av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_ENUMINPUT): %s\n", av_err2str(res));
        return res;
857 858
    }
    s->std_id = input.std;
859 860
    av_log(s1, AV_LOG_DEBUG, "Current input_channel: %d, input_name: %s, input_std: %"PRIx64"\n",
           s->channel, input.name, (uint64_t)input.std);
861

862 863
    if (s->list_format) {
        list_formats(s1, s->fd, s->list_format);
864
        return AVERROR_EXIT;
865
    }
Luca Abeni's avatar
Luca Abeni committed
866

867 868 869 870 871
    if (s->list_standard) {
        list_standards(s1);
        return AVERROR_EXIT;
    }

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

874 875 876
    if ((res = v4l2_set_parameters(s1)) < 0)
        return res;

877
    if (s->pixel_format) {
878 879 880 881
        AVCodec *codec = avcodec_find_decoder_by_name(s->pixel_format);

        if (codec)
            s1->video_codec_id = codec->id;
882 883 884

        pix_fmt = av_get_pix_fmt(s->pixel_format);

885
        if (pix_fmt == AV_PIX_FMT_NONE && !codec) {
886
            av_log(s1, AV_LOG_ERROR, "No such input format: %s.\n",
887 888
                   s->pixel_format);

889
            return AVERROR(EINVAL);
890
        }
891
    }
Luca Abeni's avatar
Luca Abeni committed
892

893
    if (!s->width && !s->height) {
894
        struct v4l2_format fmt = { .type = V4L2_BUF_TYPE_VIDEO_CAPTURE };
895

Luca Barbato's avatar
Luca Barbato committed
896 897
        av_log(s1, AV_LOG_VERBOSE,
               "Querying the device for the current frame size\n");
898
        if (v4l2_ioctl(s->fd, VIDIOC_G_FMT, &fmt) < 0) {
899 900 901
            res = AVERROR(errno);
            av_log(s1, AV_LOG_ERROR, "ioctl(VIDIOC_G_FMT): %s\n", av_err2str(res));
            return res;
902
        }
Luca Barbato's avatar
Luca Barbato committed
903

904 905
        s->width  = fmt.fmt.pix.width;
        s->height = fmt.fmt.pix.height;
Luca Barbato's avatar
Luca Barbato committed
906 907
        av_log(s1, AV_LOG_VERBOSE,
               "Setting frame size to %dx%d\n", s->width, s->height);
908 909
    }

910 911 912 913 914
    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
915 916 917 918 919 920 921 922

    /* 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;

923
    if ((res = av_image_check_size(s->width, s->height, 0, s1)) < 0)
924
        return res;
Luca Barbato's avatar
Luca Barbato committed
925

Luca Abeni's avatar
Luca Abeni committed
926 927
    s->frame_format = desired_format;

928
    st->codec->pix_fmt = avpriv_fmt_v4l2ff(desired_format, codec_id);
Luca Barbato's avatar
Luca Barbato committed
929 930 931
    s->frame_size =
        avpicture_get_size(st->codec->pix_fmt, s->width, s->height);

932 933
    if ((res = mmap_init(s1)) ||
        (res = mmap_start(s1)) < 0) {
934
        v4l2_close(s->fd);
935
        return res;
Luca Abeni's avatar
Luca Abeni committed
936
    }
937

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

940
    st->codec->codec_type = AVMEDIA_TYPE_VIDEO;
941
    st->codec->codec_id = codec_id;
942
    if (codec_id == AV_CODEC_ID_RAWVIDEO)
943 944
        st->codec->codec_tag =
            avcodec_pix_fmt_to_codec_tag(st->codec->pix_fmt);
945 946 947
    else if (codec_id == AV_CODEC_ID_H264) {
        st->need_parsing = AVSTREAM_PARSE_HEADERS;
    }
948 949
    if (desired_format == V4L2_PIX_FMT_YVU420)
        st->codec->codec_tag = MKTAG('Y', 'V', '1', '2');
950 951
    else if (desired_format == V4L2_PIX_FMT_YVU410)
        st->codec->codec_tag = MKTAG('Y', 'V', 'U', '9');
952 953
    st->codec->width = s->width;
    st->codec->height = s->height;
954 955
    if (st->avg_frame_rate.den)
        st->codec->bit_rate = s->frame_size * av_q2d(st->avg_frame_rate) * 8;
Luca Abeni's avatar
Luca Abeni committed
956

957
    return 0;
Luca Abeni's avatar
Luca Abeni committed
958 959 960 961 962
}

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

966 967
    av_init_packet(pkt);
    if ((res = mmap_read_frame(s1, pkt)) < 0) {
968
        return res;
Luca Abeni's avatar
Luca Abeni committed
969 970
    }

Luca Barbato's avatar
Luca Barbato committed
971 972 973
    if (frame && s->interlaced) {
        frame->interlaced_frame = 1;
        frame->top_field_first = s->top_field_first;
Luca Abeni's avatar
Luca Abeni committed
974 975
    }

976
    return pkt->size;
Luca Abeni's avatar
Luca Abeni committed
977 978 979 980 981 982
}

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

983 984 985 986
    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");

987
    mmap_close(s);
Luca Abeni's avatar
Luca Abeni committed
988

989
    v4l2_close(s->fd);
Luca Abeni's avatar
Luca Abeni committed
990 991 992
    return 0;
}

993 994
#define OFFSET(x) offsetof(struct video_data, x)
#define DEC AV_OPT_FLAG_DECODING_PARAM
995

996
static const AVOption options[] = {
997
    { "standard",     "set TV standard, used only by analog frame grabber",       OFFSET(standard),     AV_OPT_TYPE_STRING, {.str = NULL }, 0, 0,       DEC },
998
    { "channel",      "set TV channel, used only by frame grabber",               OFFSET(channel),      AV_OPT_TYPE_INT,    {.i64 = -1 },  -1, INT_MAX, DEC },
999 1000 1001 1002 1003 1004 1005 1006 1007 1008
    { "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" },

1009 1010 1011
    { "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" },

1012 1013 1014 1015 1016
    { "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" },
1017
    { "use_libv4l2",  "use libv4l2 (v4l-utils) convertion functions",             OFFSET(use_libv4l2),  AV_OPT_TYPE_INT,    {.i64 = 0}, 0, 1, DEC },
1018 1019 1020 1021 1022 1023 1024 1025 1026 1027
    { NULL },
};

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

1028
AVInputFormat ff_v4l2_demuxer = {
1029
    .name           = "video4linux2,v4l2",
1030 1031 1032 1033 1034 1035 1036
    .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
1037
};