297 lines
6.6 KiB
C
297 lines
6.6 KiB
C
#include "device/device.h"
|
|
#include "device/buffer.h"
|
|
#include "device/buffer_list.h"
|
|
#include "util/opts/log.h"
|
|
#include "util/opts/opts.h"
|
|
|
|
device_t *device_open(const char *name, const char *path, device_hw_t *hw) {
|
|
device_t *dev = calloc(1, sizeof(device_t));
|
|
dev->name = strdup(name);
|
|
dev->path = strdup(path);
|
|
dev->hw = hw;
|
|
dev->opts.allow_dma = true;
|
|
|
|
if (dev->hw->device_open(dev) < 0) {
|
|
LOG_ERROR(dev, "Can't open device: %s", path);
|
|
}
|
|
|
|
return dev;
|
|
|
|
error:
|
|
device_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
void device_close(device_t *dev) {
|
|
if(dev == NULL) {
|
|
return;
|
|
}
|
|
|
|
if (dev->capture_lists) {
|
|
for (int i = 0; i < dev->n_capture_list; i++) {
|
|
buffer_list_close(dev->capture_lists[i]);
|
|
dev->capture_lists[i] = NULL;
|
|
}
|
|
free(dev->capture_lists);
|
|
dev->capture_lists = NULL;
|
|
}
|
|
|
|
if (dev->output_list) {
|
|
buffer_list_close(dev->output_list);
|
|
dev->output_list = NULL;
|
|
}
|
|
|
|
dev->hw->device_close(dev);
|
|
free(dev->name);
|
|
free(dev->path);
|
|
free(dev);
|
|
}
|
|
|
|
buffer_list_t *device_open_buffer_list(device_t *dev, bool do_capture, buffer_format_t fmt, bool do_mmap)
|
|
{
|
|
return device_open_buffer_list2(dev, NULL, do_capture, fmt, do_mmap);
|
|
}
|
|
|
|
buffer_list_t *device_open_buffer_list2(device_t *dev, const char *path, bool do_capture, buffer_format_t fmt, bool do_mmap)
|
|
{
|
|
char name[64];
|
|
int index = 0;
|
|
buffer_list_t *buf_list;
|
|
|
|
if (!dev) {
|
|
return NULL;
|
|
}
|
|
|
|
if (!dev->opts.allow_dma) {
|
|
do_mmap = true;
|
|
}
|
|
|
|
if (do_capture) {
|
|
index = dev->n_capture_list;
|
|
|
|
if (index > 0)
|
|
sprintf(name, "%s:capture:%d", dev->name, index);
|
|
else
|
|
sprintf(name, "%s:capture", dev->name);
|
|
} else {
|
|
if (dev->output_list) {
|
|
LOG_ERROR(dev, "The output_list is already created.");
|
|
}
|
|
|
|
sprintf(name, "%s:output", dev->name);
|
|
}
|
|
|
|
buf_list = buffer_list_open(name, index, dev, path, fmt, do_capture, do_mmap);
|
|
if (!buf_list) {
|
|
goto error;
|
|
}
|
|
|
|
if (do_capture) {
|
|
dev->capture_lists = reallocarray(dev->capture_lists, dev->n_capture_list+1, sizeof(buffer_list_t*));
|
|
dev->capture_lists[dev->n_capture_list++] = buf_list;
|
|
} else {
|
|
dev->output_list = buf_list;
|
|
}
|
|
|
|
return buf_list;
|
|
|
|
error:
|
|
buffer_list_close(buf_list);
|
|
return NULL;
|
|
}
|
|
|
|
buffer_list_t *device_open_buffer_list_output(device_t *dev, buffer_list_t *capture_list)
|
|
{
|
|
if (!dev || !capture_list) {
|
|
return NULL;
|
|
}
|
|
|
|
buffer_format_t fmt = capture_list->fmt;
|
|
|
|
fmt.interval_us = 0;
|
|
|
|
bool do_mmap = capture_list->dev->opts.allow_dma ? !capture_list->do_mmap : true;
|
|
|
|
// If manually allocating buffers, ensure that `sizeimage` is at least `buf->length`
|
|
if (do_mmap) {
|
|
for (int i = 0; i < capture_list->nbufs; i++) {
|
|
buffer_t *buf = capture_list->bufs[i];
|
|
if (fmt.sizeimage < buf->length)
|
|
fmt.sizeimage = buf->length;
|
|
}
|
|
} else {
|
|
fmt.sizeimage = 0;
|
|
}
|
|
|
|
return device_open_buffer_list(dev, false,
|
|
fmt,
|
|
do_mmap);
|
|
}
|
|
|
|
buffer_list_t *device_open_buffer_list_capture(device_t *dev, const char *path, buffer_list_t *output_list, buffer_format_t fmt, bool do_mmap)
|
|
{
|
|
if (!dev || !output_list) {
|
|
return NULL;
|
|
}
|
|
|
|
if (!fmt.width)
|
|
fmt.width = output_list->fmt.width;
|
|
if (!fmt.height)
|
|
fmt.height = output_list->fmt.height;
|
|
if (!fmt.nbufs)
|
|
fmt.nbufs = output_list->fmt.nbufs;
|
|
|
|
return device_open_buffer_list2(dev, path, true, fmt, do_mmap);
|
|
}
|
|
|
|
buffer_list_t *device_open_buffer_list_capture2(device_t *dev, const char *path, buffer_list_t *output_list, unsigned choosen_format, bool do_mmap)
|
|
{
|
|
buffer_format_t fmt = {
|
|
.format = choosen_format
|
|
};
|
|
|
|
return device_open_buffer_list_capture(dev, path, output_list, fmt, do_mmap);
|
|
}
|
|
|
|
int device_set_stream(device_t *dev, bool do_on)
|
|
{
|
|
// TODO: support events
|
|
#if 0
|
|
struct v4l2_event_subscription sub = {0};
|
|
sub.type = V4L2_EVENT_SOURCE_CHANGE;
|
|
|
|
LOG_DEBUG(dev, "Subscribing to DV-timings events ...");
|
|
ioctl_retried(dev_name(dev), dev->v4l2->dev_fd, do_on ? VIDIOC_SUBSCRIBE_EVENT : VIDIOC_UNSUBSCRIBE_EVENT, &sub);
|
|
#endif
|
|
|
|
for (int i = 0; i < dev->n_capture_list; i++) {
|
|
if (buffer_list_set_stream(dev->capture_lists[i], do_on) < 0) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (dev->output_list) {
|
|
if (buffer_list_set_stream(dev->output_list, do_on) < 0) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int device_video_force_key(device_t *dev)
|
|
{
|
|
if (!dev || dev->hw->device_video_force_key(dev) < 0)
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void device_dump_options(device_t *dev, FILE *stream)
|
|
{
|
|
if (dev && dev->hw->device_dump_options) {
|
|
dev->hw->device_dump_options(dev, stream);
|
|
}
|
|
}
|
|
|
|
int device_set_fps(device_t *dev, int desired_fps)
|
|
{
|
|
if (!dev)
|
|
return -1;
|
|
|
|
unsigned interval_us = 0;
|
|
|
|
if (desired_fps > 0) {
|
|
interval_us = 1000 * 1000 / desired_fps;
|
|
}
|
|
|
|
// try to use HW fps setting
|
|
if (dev->hw->device_set_fps && dev->hw->device_set_fps(dev, desired_fps) >= 0) {
|
|
interval_us = 0;
|
|
}
|
|
|
|
LOG_INFO(dev, "Setting frame interval_us=%d for FPS=%d", interval_us, desired_fps);
|
|
|
|
for (int i = 0; i < dev->n_capture_list; i++) {
|
|
dev->capture_lists[i]->fmt.interval_us = interval_us;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int device_set_rotation(device_t *dev, bool vflip, bool hflip)
|
|
{
|
|
if (!dev)
|
|
return -1;
|
|
|
|
if (dev->hw->device_set_rotation) {
|
|
return dev->hw->device_set_rotation(dev, vflip, hflip);
|
|
}
|
|
|
|
int hret = device_set_option_string(dev, "horizontal_flip", hflip ? "1" : "0");
|
|
int vret = device_set_option_string(dev, "vertical_flip", vflip ? "1" : "0");
|
|
return hret ? hret : vret;
|
|
}
|
|
|
|
int device_set_option_string(device_t *dev, const char *key, const char *value)
|
|
{
|
|
if (!dev) {
|
|
return 0;
|
|
}
|
|
|
|
return dev->hw->device_set_option(dev, key, value);
|
|
}
|
|
|
|
void device_set_option_list(device_t *dev, const char *option_list)
|
|
{
|
|
if (!dev || !option_list || !option_list[0]) {
|
|
return;
|
|
}
|
|
|
|
char *start = strdup(option_list);
|
|
char *string = start;
|
|
char *option;
|
|
|
|
while ((option = strsep(&string, OPTION_VALUE_LIST_SEP)) != NULL) {
|
|
char *value = option;
|
|
char *key = strsep(&value, "=");
|
|
|
|
if (value) {
|
|
device_set_option_string(dev, key, value);
|
|
} else {
|
|
LOG_INFO(dev, "Missing 'key=value' for '%s'", option);
|
|
continue;
|
|
}
|
|
|
|
// consume all separators
|
|
while (strsep(&value, "="));
|
|
}
|
|
|
|
free(start);
|
|
}
|
|
|
|
int device_output_enqueued(device_t *dev)
|
|
{
|
|
if (dev->output_list)
|
|
return buffer_list_count_enqueued(dev->output_list);
|
|
return 0;
|
|
}
|
|
|
|
int device_capture_enqueued(device_t *dev, int *max)
|
|
{
|
|
int min_val = 100;
|
|
int max_val = 0;
|
|
|
|
for (int i = 0; i < dev->n_capture_list; i++) {
|
|
int count = buffer_list_count_enqueued(dev->capture_lists[i]);
|
|
min_val = MIN(min_val, count);
|
|
max_val = MAX(max_val, count);
|
|
}
|
|
|
|
min_val = MIN(min_val, max_val);
|
|
if (max) {
|
|
*max = max_val;
|
|
}
|
|
return min_val;
|
|
}
|