2014-08-10 17:33:03 +02:00
|
|
|
/*
|
|
|
|
Copyright (C) 2014 by Leonhard Oelke <leonhard@in-verted.de>
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/mman.h>
|
|
|
|
|
|
|
|
#include <util/bmem.h>
|
|
|
|
|
|
|
|
#include "v4l2-helpers.h"
|
|
|
|
|
|
|
|
#define blog(level, msg, ...) blog(level, "v4l2-helpers: " msg, ##__VA_ARGS__)
|
|
|
|
|
|
|
|
int_fast32_t v4l2_start_capture(int_fast32_t dev, struct v4l2_buffer_data *buf)
|
|
|
|
{
|
|
|
|
enum v4l2_buf_type type;
|
|
|
|
struct v4l2_buffer enq;
|
|
|
|
|
|
|
|
memset(&enq, 0, sizeof(enq));
|
|
|
|
enq.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
enq.memory = V4L2_MEMORY_MMAP;
|
|
|
|
|
|
|
|
for (enq.index = 0; enq.index < buf->count; ++enq.index) {
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_QBUF, &enq) < 0) {
|
|
|
|
blog(LOG_ERROR, "unable to queue buffer");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_STREAMON, &type) < 0) {
|
|
|
|
blog(LOG_ERROR, "unable to start stream");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int_fast32_t v4l2_stop_capture(int_fast32_t dev)
|
|
|
|
{
|
|
|
|
enum v4l2_buf_type type;
|
|
|
|
|
|
|
|
type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_STREAMOFF, &type) < 0) {
|
|
|
|
blog(LOG_ERROR, "unable to stop stream");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-01-02 15:53:40 -08:00
|
|
|
int_fast32_t v4l2_reset_capture(int_fast32_t dev, struct v4l2_buffer_data *buf)
|
|
|
|
{
|
|
|
|
blog(LOG_DEBUG, "attempting to reset capture");
|
|
|
|
if (v4l2_stop_capture(dev) < 0)
|
|
|
|
return -1;
|
|
|
|
if (v4l2_start_capture(dev, buf) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-01-02 14:44:09 -08:00
|
|
|
#ifdef _DEBUG
|
|
|
|
int_fast32_t v4l2_query_all_buffers(int_fast32_t dev,
|
|
|
|
struct v4l2_buffer_data *buf_data)
|
|
|
|
{
|
|
|
|
struct v4l2_buffer buf;
|
|
|
|
|
|
|
|
blog(LOG_DEBUG, "attempting to read buffer data for %ld buffers",
|
|
|
|
buf_data->count);
|
|
|
|
|
|
|
|
for (uint_fast32_t i = 0; i < buf_data->count; i++) {
|
|
|
|
buf.index = i;
|
|
|
|
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
buf.memory = V4L2_MEMORY_MMAP;
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_QUERYBUF, &buf) < 0) {
|
|
|
|
blog(LOG_DEBUG,
|
|
|
|
"failed to read buffer data for buffer #%ld", i);
|
|
|
|
} else {
|
|
|
|
blog(LOG_DEBUG,
|
|
|
|
"query buf #%ld info: ts: %06ld buf id #%d, flags 0x%08X, seq #%d, len %d, used %d",
|
|
|
|
i, buf.timestamp.tv_usec, buf.index, buf.flags,
|
|
|
|
buf.sequence, buf.length, buf.bytesused);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-08-10 17:33:03 +02:00
|
|
|
int_fast32_t v4l2_create_mmap(int_fast32_t dev, struct v4l2_buffer_data *buf)
|
|
|
|
{
|
|
|
|
struct v4l2_requestbuffers req;
|
|
|
|
struct v4l2_buffer map;
|
|
|
|
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
|
|
req.count = 4;
|
|
|
|
req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
req.memory = V4L2_MEMORY_MMAP;
|
|
|
|
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_REQBUFS, &req) < 0) {
|
|
|
|
blog(LOG_ERROR, "Request for buffers failed !");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req.count < 2) {
|
|
|
|
blog(LOG_ERROR, "Device returned less than 2 buffers");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf->count = req.count;
|
|
|
|
buf->info = bzalloc(req.count * sizeof(struct v4l2_mmap_info));
|
|
|
|
|
|
|
|
memset(&map, 0, sizeof(map));
|
|
|
|
map.type = req.type;
|
|
|
|
map.memory = req.memory;
|
|
|
|
|
|
|
|
for (map.index = 0; map.index < req.count; ++map.index) {
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_QUERYBUF, &map) < 0) {
|
|
|
|
blog(LOG_ERROR, "Failed to query buffer details");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf->info[map.index].length = map.length;
|
|
|
|
buf->info[map.index].start =
|
|
|
|
v4l2_mmap(NULL, map.length, PROT_READ | PROT_WRITE,
|
|
|
|
MAP_SHARED, dev, map.m.offset);
|
|
|
|
|
|
|
|
if (buf->info[map.index].start == MAP_FAILED) {
|
|
|
|
blog(LOG_ERROR, "mmap for buffer failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int_fast32_t v4l2_destroy_mmap(struct v4l2_buffer_data *buf)
|
|
|
|
{
|
|
|
|
for (uint_fast32_t i = 0; i < buf->count; ++i) {
|
2014-08-11 23:35:13 +02:00
|
|
|
if (buf->info[i].start != MAP_FAILED && buf->info[i].start != 0)
|
2014-08-10 17:33:03 +02:00
|
|
|
v4l2_munmap(buf->info[i].start, buf->info[i].length);
|
|
|
|
}
|
|
|
|
|
2014-08-11 23:35:13 +02:00
|
|
|
if (buf->count) {
|
|
|
|
bfree(buf->info);
|
|
|
|
buf->count = 0;
|
|
|
|
}
|
2014-08-10 17:33:03 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-15 23:59:41 +02:00
|
|
|
int_fast32_t v4l2_set_input(int_fast32_t dev, int *input)
|
|
|
|
{
|
|
|
|
if (!dev || !input)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return (*input == -1) ? v4l2_ioctl(dev, VIDIOC_G_INPUT, input)
|
|
|
|
: v4l2_ioctl(dev, VIDIOC_S_INPUT, input);
|
|
|
|
}
|
|
|
|
|
2015-02-16 22:29:39 +01:00
|
|
|
int_fast32_t v4l2_get_input_caps(int_fast32_t dev, int input, uint32_t *caps)
|
|
|
|
{
|
|
|
|
if (!dev || !caps)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (input == -1) {
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_G_INPUT, &input) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct v4l2_input in;
|
|
|
|
memset(&in, 0, sizeof(in));
|
|
|
|
in.index = input;
|
|
|
|
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_ENUMINPUT, &in) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
*caps = in.capabilities;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-16 00:05:11 +02:00
|
|
|
int_fast32_t v4l2_set_format(int_fast32_t dev, int *resolution,
|
|
|
|
int *pixelformat, int *bytesperline)
|
|
|
|
{
|
|
|
|
bool set = false;
|
|
|
|
int width, height;
|
|
|
|
struct v4l2_format fmt;
|
|
|
|
|
|
|
|
if (!dev || !resolution || !pixelformat || !bytesperline)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* We need to set the type in order to query the settings */
|
|
|
|
fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_G_FMT, &fmt) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (*resolution != -1) {
|
|
|
|
v4l2_unpack_tuple(&width, &height, *resolution);
|
|
|
|
fmt.fmt.pix.width = width;
|
|
|
|
fmt.fmt.pix.height = height;
|
|
|
|
set = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*pixelformat != -1) {
|
|
|
|
fmt.fmt.pix.pixelformat = *pixelformat;
|
|
|
|
set = true;
|
|
|
|
}
|
2014-09-15 23:59:41 +02:00
|
|
|
|
2014-09-16 00:05:11 +02:00
|
|
|
if (set && (v4l2_ioctl(dev, VIDIOC_S_FMT, &fmt) < 0))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
*resolution = v4l2_pack_tuple(fmt.fmt.pix.width, fmt.fmt.pix.height);
|
|
|
|
*pixelformat = fmt.fmt.pix.pixelformat;
|
|
|
|
*bytesperline = fmt.fmt.pix.bytesperline;
|
|
|
|
return 0;
|
|
|
|
}
|
2014-09-16 00:06:42 +02:00
|
|
|
|
|
|
|
int_fast32_t v4l2_set_framerate(int_fast32_t dev, int *framerate)
|
|
|
|
{
|
|
|
|
bool set = false;
|
|
|
|
int num, denom;
|
|
|
|
struct v4l2_streamparm par;
|
|
|
|
|
|
|
|
if (!dev || !framerate)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* We need to set the type in order to query the stream settings */
|
|
|
|
par.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_G_PARM, &par) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (*framerate != -1) {
|
|
|
|
v4l2_unpack_tuple(&num, &denom, *framerate);
|
|
|
|
par.parm.capture.timeperframe.numerator = num;
|
|
|
|
par.parm.capture.timeperframe.denominator = denom;
|
|
|
|
set = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (set && (v4l2_ioctl(dev, VIDIOC_S_PARM, &par) < 0))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
*framerate = v4l2_pack_tuple(par.parm.capture.timeperframe.numerator,
|
|
|
|
par.parm.capture.timeperframe.denominator);
|
|
|
|
return 0;
|
|
|
|
}
|
2015-02-23 21:08:57 +01:00
|
|
|
|
|
|
|
int_fast32_t v4l2_set_standard(int_fast32_t dev, int *standard)
|
|
|
|
{
|
|
|
|
if (!dev || !standard)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (*standard == -1) {
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_G_STD, standard) < 0)
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_S_STD, standard) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2015-02-24 22:27:00 +01:00
|
|
|
|
|
|
|
int_fast32_t v4l2_enum_dv_timing(int_fast32_t dev, struct v4l2_dv_timings *dvt,
|
|
|
|
int index)
|
|
|
|
{
|
2015-05-02 15:05:38 +02:00
|
|
|
#if !defined(VIDIOC_ENUM_DV_TIMINGS) || !defined(V4L2_IN_CAP_DV_TIMINGS)
|
2015-04-22 20:09:18 +02:00
|
|
|
UNUSED_PARAMETER(dev);
|
|
|
|
UNUSED_PARAMETER(dvt);
|
|
|
|
UNUSED_PARAMETER(index);
|
|
|
|
return -1;
|
|
|
|
#else
|
2015-02-24 22:27:00 +01:00
|
|
|
if (!dev || !dvt)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
struct v4l2_enum_dv_timings iter;
|
|
|
|
memset(&iter, 0, sizeof(iter));
|
|
|
|
iter.index = index;
|
|
|
|
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_ENUM_DV_TIMINGS, &iter) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
memcpy(dvt, &iter.timings, sizeof(struct v4l2_dv_timings));
|
|
|
|
|
|
|
|
return 0;
|
2015-04-22 20:09:18 +02:00
|
|
|
#endif
|
2015-02-24 22:27:00 +01:00
|
|
|
}
|
2015-02-24 22:18:24 +01:00
|
|
|
|
|
|
|
int_fast32_t v4l2_set_dv_timing(int_fast32_t dev, int *timing)
|
|
|
|
{
|
|
|
|
if (!dev || !timing)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (*timing == -1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
struct v4l2_dv_timings dvt;
|
|
|
|
|
|
|
|
if (v4l2_enum_dv_timing(dev, &dvt, *timing) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (v4l2_ioctl(dev, VIDIOC_S_DV_TIMINGS, &dvt) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|