250 lines
6.1 KiB
C
250 lines
6.1 KiB
C
#include "device/hw/device.h"
|
|
#include "device/hw/buffer.h"
|
|
#include "device/hw/buffer_list.h"
|
|
|
|
device_t *device_open(const char *name, const char *path) {
|
|
device_t *dev = calloc(1, sizeof(device_t));
|
|
dev->name = strdup(name);
|
|
dev->path = strdup(path);
|
|
dev->fd = open(path, O_RDWR|O_NONBLOCK);
|
|
dev->subdev_fd = -1;
|
|
dev->allow_dma = true;
|
|
if(dev->fd < 0) {
|
|
E_LOG_ERROR(dev, "Can't open device: %s", path);
|
|
}
|
|
|
|
E_LOG_DEBUG(dev, "Querying device capabilities ...");
|
|
E_XIOCTL(dev, dev->fd, VIDIOC_QUERYCAP, &dev->v4l2_cap, "Can't query device capabilities");\
|
|
|
|
if (!(dev->v4l2_cap.capabilities & V4L2_CAP_STREAMING)) {
|
|
E_LOG_ERROR(dev, "Device doesn't support streaming IO");
|
|
}
|
|
|
|
E_LOG_INFO(dev, "Device path=%s fd=%d opened", dev->path, dev->fd);
|
|
|
|
dev->subdev_fd = device_open_v4l2_subdev(dev, 0);
|
|
return dev;
|
|
|
|
error:
|
|
device_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
void device_close(device_t *dev) {
|
|
if(dev == NULL) {
|
|
return;
|
|
}
|
|
|
|
if (dev->capture_list) {
|
|
buffer_list_close(dev->capture_list);
|
|
dev->capture_list = NULL;
|
|
}
|
|
|
|
if (dev->output_list) {
|
|
buffer_list_close(dev->output_list);
|
|
dev->output_list = NULL;
|
|
}
|
|
|
|
if (dev->subdev_fd >= 0) {
|
|
close(dev->subdev_fd);
|
|
}
|
|
|
|
if(dev->fd >= 0) {
|
|
close(dev->fd);
|
|
}
|
|
|
|
free(dev->name);
|
|
free(dev->path);
|
|
free(dev);
|
|
}
|
|
|
|
int device_open_buffer_list(device_t *dev, bool do_capture, unsigned width, unsigned height, unsigned format, unsigned bytesperline, int nbufs, bool do_mmap)
|
|
{
|
|
unsigned type;
|
|
char name[64];
|
|
struct buffer_list_s **buf_list = NULL;
|
|
|
|
if (!dev) {
|
|
return -1;
|
|
}
|
|
|
|
if (!dev->allow_dma) {
|
|
do_mmap = true;
|
|
}
|
|
|
|
if (do_capture) {
|
|
buf_list = &dev->capture_list;
|
|
|
|
if (dev->capture_list) {
|
|
E_LOG_ERROR(dev, "The capture_list is already created.");
|
|
}
|
|
|
|
if (dev->v4l2_cap.capabilities & V4L2_CAP_VIDEO_CAPTURE) {
|
|
type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
sprintf(name, "%s:capture", dev->name);
|
|
} else if (dev->v4l2_cap.capabilities & (V4L2_CAP_VIDEO_CAPTURE_MPLANE | V4L2_CAP_VIDEO_M2M_MPLANE)) {
|
|
type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
|
|
sprintf(name, "%s:capture:mplane", dev->name);
|
|
} else {
|
|
E_LOG_ERROR(dev, "Video capture is not supported by device: %08x", dev->v4l2_cap.capabilities);
|
|
}
|
|
} else {
|
|
buf_list = &dev->output_list;
|
|
|
|
if (dev->output_list) {
|
|
E_LOG_ERROR(dev, "The output_list is already created.");
|
|
}
|
|
|
|
if (dev->v4l2_cap.capabilities & V4L2_CAP_VIDEO_OUTPUT) {
|
|
type = V4L2_BUF_TYPE_VIDEO_OUTPUT;
|
|
sprintf(name, "%s:output", dev->name);
|
|
} else if (dev->v4l2_cap.capabilities & (V4L2_CAP_VIDEO_OUTPUT_MPLANE | V4L2_CAP_VIDEO_M2M_MPLANE)) {
|
|
type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
|
|
sprintf(name, "%s:output:mplane", dev->name);
|
|
} else {
|
|
E_LOG_ERROR(dev, "Video output is not supported by device: %08x", dev->v4l2_cap.capabilities);
|
|
}
|
|
}
|
|
|
|
*buf_list = buffer_list_open(name, dev, type, do_mmap);
|
|
if (!*buf_list) {
|
|
goto error;
|
|
}
|
|
|
|
if (buffer_list_set_format(*buf_list, width, height, format, bytesperline) < 0) {
|
|
goto error;
|
|
}
|
|
|
|
if (buffer_list_request(*buf_list, nbufs) < 0) {
|
|
goto error;
|
|
}
|
|
|
|
return 0;
|
|
|
|
error:
|
|
buffer_list_close(*buf_list);
|
|
*buf_list = NULL;
|
|
return -1;
|
|
}
|
|
|
|
int device_open_buffer_list_output(device_t *dev, buffer_list_t *capture_list)
|
|
{
|
|
return device_open_buffer_list(dev, false,
|
|
capture_list->fmt_width, capture_list->fmt_height,
|
|
capture_list->fmt_format, capture_list->fmt_bytesperline,
|
|
capture_list->nbufs,
|
|
capture_list->device->allow_dma ? !capture_list->do_mmap : true);
|
|
}
|
|
|
|
int device_open_buffer_list_capture(device_t *dev, buffer_list_t *output_list, float div, unsigned format, bool do_mmap)
|
|
{
|
|
if (!output_list) {
|
|
output_list = dev->output_list;
|
|
}
|
|
if (!output_list) {
|
|
return -1;
|
|
}
|
|
|
|
return device_open_buffer_list(dev, true,
|
|
output_list->fmt_width / div, output_list->fmt_height / div,
|
|
format, 0, output_list->nbufs, do_mmap);
|
|
}
|
|
|
|
int device_set_stream(device_t *dev, bool do_on)
|
|
{
|
|
struct v4l2_event_subscription sub = {0};
|
|
sub.type = V4L2_EVENT_SOURCE_CHANGE;
|
|
|
|
E_LOG_DEBUG(dev, "Subscribing to DV-timings events ...");
|
|
xioctl(dev_name(dev), dev->fd, do_on ? VIDIOC_SUBSCRIBE_EVENT : VIDIOC_UNSUBSCRIBE_EVENT, &sub);
|
|
|
|
if (dev->capture_list) {
|
|
if (buffer_list_stream(dev->capture_list, do_on) < 0) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (dev->output_list) {
|
|
if (buffer_list_stream(dev->output_list, do_on) < 0) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int device_set_decoder_start(device_t *dev, bool do_on)
|
|
{
|
|
struct v4l2_decoder_cmd cmd = {0};
|
|
|
|
cmd.cmd = do_on ? V4L2_DEC_CMD_START : V4L2_DEC_CMD_STOP;
|
|
|
|
E_LOG_DEBUG(dev, "Setting decoder state %s...", do_on ? "Start" : "Stop");
|
|
E_XIOCTL(dev, dev->fd, VIDIOC_DECODER_CMD, &cmd, "Cannot set decoder state");
|
|
dev->decoder_started = do_on;
|
|
return 0;
|
|
|
|
error:
|
|
return -1;
|
|
}
|
|
|
|
int device_consume_event(device_t *dev)
|
|
{
|
|
struct v4l2_event event;
|
|
|
|
if (!dev) {
|
|
return -1;
|
|
}
|
|
|
|
E_LOG_DEBUG(dev, "Consuming V4L2 event ...");
|
|
E_XIOCTL(dev, dev->fd, VIDIOC_DQEVENT, &event, "Got some V4L2 device event, but where is it?");
|
|
|
|
switch (event.type) {
|
|
case V4L2_EVENT_SOURCE_CHANGE:
|
|
E_LOG_INFO(dev, "Got V4L2_EVENT_SOURCE_CHANGE: source changed");
|
|
return -1;
|
|
case V4L2_EVENT_EOS:
|
|
E_LOG_INFO(dev, "Got V4L2_EVENT_EOS: end of stream (ignored)");
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
|
|
error:
|
|
return -1;
|
|
}
|
|
|
|
int device_video_force_key(device_t *dev)
|
|
{
|
|
if (!dev) {
|
|
return -1;
|
|
}
|
|
|
|
struct v4l2_control ctl = {0};
|
|
ctl.id = V4L2_CID_MPEG_VIDEO_FORCE_KEY_FRAME;
|
|
ctl.value = 1;
|
|
E_LOG_DEBUG(dev, "Forcing keyframe ...");
|
|
E_XIOCTL(dev, dev->fd, VIDIOC_S_CTRL, &ctl, "Can't force keyframe");
|
|
return 0;
|
|
error:
|
|
return -1;
|
|
}
|
|
|
|
int device_set_fps(device_t *dev, int desired_fps)
|
|
{
|
|
struct v4l2_streamparm setfps = {0};
|
|
|
|
if (!dev) {
|
|
return -1;
|
|
}
|
|
|
|
setfps.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
setfps.parm.output.timeperframe.numerator = 1;
|
|
setfps.parm.output.timeperframe.denominator = desired_fps;
|
|
E_LOG_DEBUG(dev, "Configuring FPS ...");
|
|
E_XIOCTL(dev, dev->fd, VIDIOC_S_PARM, &setfps, "Can't set FPS");
|
|
return 0;
|
|
error:
|
|
return -1;
|
|
}
|