123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174 |
- /*
- Copyright (C) 2019 by Gary Kramlich <[email protected]>
- 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 <fcntl.h>
- #include <linux/videodev2.h>
- #include <libv4l2.h>
- #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);
- switch (obs_property_get_type(prop)) {
- case OBS_PROPERTY_BOOL:
- control.value =
- obs_data_get_bool(settings, obs_property_name(prop));
- break;
- case OBS_PROPERTY_INT:
- case OBS_PROPERTY_LIST:
- control.value =
- obs_data_get_int(settings, obs_property_name(prop));
- break;
- default:
- blog(LOG_ERROR, "unknown property type for %s",
- obs_property_name(prop));
- v4l2_close(dev);
- return ret;
- }
- if (0 != v4l2_ioctl(dev, VIDIOC_S_CTRL, &control)) {
- ret = true;
- }
- v4l2_close(dev);
- return ret;
- }
- static bool 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.index);
- }
- }
- if (obs_property_list_item_count(prop) == 0) {
- obs_properties_remove_by_name(props, (char *)qctrl->name);
- return false;
- }
- return true;
- }
- #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 void 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:
- if (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;
- }
|