| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 | 
							- /*
 
- Copyright (C) 2014 by Leonhard Oelke <[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 <libudev.h>
 
- #include <util/threading.h>
 
- #include <util/darray.h>
 
- #include <obs.h>
 
- #include "v4l2-udev.h"
 
- /** udev action enum */
 
- enum udev_action {
 
- 	UDEV_ACTION_ADDED,
 
- 	UDEV_ACTION_REMOVED,
 
- 	UDEV_ACTION_UNKNOWN
 
- };
 
- static const char *udev_signals[] = {"void device_added(string device)",
 
- 				     "void device_removed(string device)",
 
- 				     NULL};
 
- /* global data */
 
- static uint_fast32_t udev_refs = 0;
 
- static pthread_mutex_t udev_mutex = PTHREAD_MUTEX_INITIALIZER;
 
- static pthread_t udev_thread;
 
- static os_event_t *udev_event;
 
- static signal_handler_t *udev_signalhandler = NULL;
 
- /**
 
-  * udev gives us the device action as string, so we convert it here ...
 
-  *
 
-  * @param action the udev action as string
 
-  * @return the udev action as enum value
 
-  */
 
- static enum udev_action udev_action_to_enum(const char *action)
 
- {
 
- 	if (!action)
 
- 		return UDEV_ACTION_UNKNOWN;
 
- 	if (!strncmp("add", action, 3))
 
- 		return UDEV_ACTION_ADDED;
 
- 	if (!strncmp("remove", action, 6))
 
- 		return UDEV_ACTION_REMOVED;
 
- 	return UDEV_ACTION_UNKNOWN;
 
- }
 
- /**
 
-  * Call all registered callbacks with the event
 
-  *
 
-  * @param dev udev device that had an event occuring
 
-  */
 
- static inline void udev_signal_event(struct udev_device *dev)
 
- {
 
- 	const char *node;
 
- 	enum udev_action action;
 
- 	struct calldata data;
 
- 	pthread_mutex_lock(&udev_mutex);
 
- 	node = udev_device_get_devnode(dev);
 
- 	action = udev_action_to_enum(udev_device_get_action(dev));
 
- 	calldata_init(&data);
 
- 	calldata_set_string(&data, "device", node);
 
- 	switch (action) {
 
- 	case UDEV_ACTION_ADDED:
 
- 		signal_handler_signal(udev_signalhandler, "device_added",
 
- 				      &data);
 
- 		break;
 
- 	case UDEV_ACTION_REMOVED:
 
- 		signal_handler_signal(udev_signalhandler, "device_removed",
 
- 				      &data);
 
- 		break;
 
- 	default:
 
- 		break;
 
- 	}
 
- 	calldata_free(&data);
 
- 	pthread_mutex_unlock(&udev_mutex);
 
- }
 
- /**
 
-  * Event listener thread
 
-  */
 
- static void *udev_event_thread(void *vptr)
 
- {
 
- 	UNUSED_PARAMETER(vptr);
 
- 	int fd;
 
- 	fd_set fds;
 
- 	struct timeval tv;
 
- 	struct udev *udev;
 
- 	struct udev_monitor *mon;
 
- 	struct udev_device *dev;
 
- 	/* set up udev monitoring */
 
- 	udev = udev_new();
 
- 	mon = udev_monitor_new_from_netlink(udev, "udev");
 
- 	udev_monitor_filter_add_match_subsystem_devtype(mon, "video4linux",
 
- 							NULL);
 
- 	if (udev_monitor_enable_receiving(mon) < 0)
 
- 		return NULL;
 
- 	/* set up fds */
 
- 	fd = udev_monitor_get_fd(mon);
 
- 	while (os_event_try(udev_event) == EAGAIN) {
 
- 		FD_ZERO(&fds);
 
- 		FD_SET(fd, &fds);
 
- 		tv.tv_sec = 1;
 
- 		tv.tv_usec = 0;
 
- 		if (select(fd + 1, &fds, NULL, NULL, &tv) <= 0)
 
- 			continue;
 
- 		dev = udev_monitor_receive_device(mon);
 
- 		if (!dev)
 
- 			continue;
 
- 		udev_signal_event(dev);
 
- 		udev_device_unref(dev);
 
- 	}
 
- 	udev_monitor_unref(mon);
 
- 	udev_unref(udev);
 
- 	return NULL;
 
- }
 
- void v4l2_init_udev(void)
 
- {
 
- 	pthread_mutex_lock(&udev_mutex);
 
- 	/* set up udev */
 
- 	if (udev_refs == 0) {
 
- 		if (os_event_init(&udev_event, OS_EVENT_TYPE_MANUAL) != 0)
 
- 			goto fail;
 
- 		if (pthread_create(&udev_thread, NULL, udev_event_thread,
 
- 				   NULL) != 0)
 
- 			goto fail;
 
- 		udev_signalhandler = signal_handler_create();
 
- 		if (!udev_signalhandler)
 
- 			goto fail;
 
- 		signal_handler_add_array(udev_signalhandler, udev_signals);
 
- 	}
 
- 	udev_refs++;
 
- fail:
 
- 	pthread_mutex_unlock(&udev_mutex);
 
- }
 
- void v4l2_unref_udev(void)
 
- {
 
- 	pthread_mutex_lock(&udev_mutex);
 
- 	/* unref udev monitor */
 
- 	if (udev_refs && --udev_refs == 0) {
 
- 		os_event_signal(udev_event);
 
- 		pthread_join(udev_thread, NULL);
 
- 		os_event_destroy(udev_event);
 
- 		if (udev_signalhandler)
 
- 			signal_handler_destroy(udev_signalhandler);
 
- 		udev_signalhandler = NULL;
 
- 	}
 
- 	pthread_mutex_unlock(&udev_mutex);
 
- }
 
- signal_handler_t *v4l2_get_udev_signalhandler(void)
 
- {
 
- 	return udev_signalhandler;
 
- }
 
 
  |