| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 | /*Copyright (C) 2019 by Gary Kramlich <[email protected]>This program is free software: you can redistribute it and/or modifyit under the terms of the GNU General Public License as published bythe 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 ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See theGNU General Public License for more details.You should have received a copy of the GNU General Public Licensealong 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))#endifstatic 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;}
 |