/* Copyright (C) 2019 by Gary Kramlich 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 . */ #include #include #include #include "v4l2-controls.h" #define blog(level, msg, ...) blog(level, "v4l2-controls: " msg, ##__VA_ARGS__) #if defined(__LP64__) #define UINT_TO_POINTER(val) ((void *)(unsigned long)(val)) #define POINTER_TO_UINT(p) ((unsigned int)(unsigned long)(p)) #else #define UINT_TO_POINTER(val) ((void *)(unsigned int)(val)) #define POINTER_TO_UINT(p) ((unsigned int)(unsigned int)(p)) #endif static bool v4l2_control_changed(void *data, obs_properties_t *props, obs_property_t *prop, obs_data_t *settings) { int dev = v4l2_open(obs_data_get_string(settings, "device_id"), O_RDWR | O_NONBLOCK); bool ret = false; (void)props; if (dev == -1) return false; struct v4l2_control control; control.id = POINTER_TO_UINT(data); control.value = obs_data_get_int(settings, obs_property_name(prop)); if (0 != v4l2_ioctl(dev, VIDIOC_S_CTRL, &control)) { ret = true; } v4l2_close(dev); return ret; } static int_fast32_t v4l2_update_controls_menu(int_fast32_t dev, obs_properties_t *props, struct v4l2_queryctrl *qctrl) { obs_property_t *prop; struct v4l2_querymenu qmenu; prop = obs_properties_add_list(props, (char *)qctrl->name, (char *)qctrl->name, OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT); obs_property_set_modified_callback2(prop, v4l2_control_changed, UINT_TO_POINTER(qctrl->id)); memset(&qmenu, 0, sizeof(qmenu)); qmenu.id = qctrl->id; for (qmenu.index = qctrl->minimum; qmenu.index <= (uint32_t)qctrl->maximum; qmenu.index += qctrl->step) { if (0 == v4l2_ioctl(dev, VIDIOC_QUERYMENU, &qmenu)) { obs_property_list_add_int(prop, (char *)qmenu.name, qmenu.value); } } return 0; } #define INVALID_CONTROL_FLAGS \ (V4L2_CTRL_FLAG_DISABLED | V4L2_CTRL_FLAG_READ_ONLY | \ V4L2_CTRL_FLAG_VOLATILE) static inline bool valid_control(struct v4l2_queryctrl *qctrl) { return (qctrl->flags & INVALID_CONTROL_FLAGS) == 0; } static inline bool add_control_property(obs_properties_t *props, obs_data_t *settings, int_fast32_t dev, struct v4l2_queryctrl *qctrl) { obs_property_t *prop = NULL; if (!valid_control(qctrl)) { return; } switch (qctrl->type) { case V4L2_CTRL_TYPE_INTEGER: prop = obs_properties_add_int_slider( props, (char *)qctrl->name, (char *)qctrl->name, qctrl->minimum, qctrl->maximum, qctrl->step); obs_data_set_default_int(settings, (char *)qctrl->name, qctrl->default_value); obs_property_set_modified_callback2(prop, v4l2_control_changed, UINT_TO_POINTER(qctrl->id)); break; case V4L2_CTRL_TYPE_BOOLEAN: prop = obs_properties_add_bool(props, (char *)qctrl->name, (char *)qctrl->name); obs_data_set_default_bool(settings, (char *)qctrl->name, qctrl->default_value); obs_property_set_modified_callback2(prop, v4l2_control_changed, UINT_TO_POINTER(qctrl->id)); break; case V4L2_CTRL_TYPE_MENU: case V4L2_CTRL_TYPE_INTEGER_MENU: v4l2_update_controls_menu(dev, props, qctrl); obs_data_set_default_int(settings, (char *)qctrl->name, qctrl->default_value); blog(LOG_INFO, "setting default for %s to %d", (char *)qctrl->name, qctrl->default_value); break; } } int_fast32_t v4l2_update_controls(int_fast32_t dev, obs_properties_t *props, obs_data_t *settings) { struct v4l2_queryctrl qctrl; if (!dev || !props) return -1; memset(&qctrl, 0, sizeof(qctrl)); qctrl.id = V4L2_CTRL_FLAG_NEXT_CTRL; while (0 == v4l2_ioctl(dev, VIDIOC_QUERYCTRL, &qctrl)) { add_control_property(props, settings, dev, &qctrl); qctrl.id |= V4L2_CTRL_FLAG_NEXT_CTRL; } return 0; }