libusb: hotplug, use events instead of a timer when possible

Try to fetch events from nlsysevent or devd to determine when
to scan the usb bus for devices addition or removal.
if none are available fallback on the regular timer based (4s)
scanner

if devd socket or netlink socket is closed or error fallback on the
timer based method.

Reviewed by:	kevans
Differential Revision:	https://reviews.freebsd.org/D48300
This commit is contained in:
Baptiste Daroussin
2025-01-04 11:23:12 +01:00
parent e3b4a51580
commit 9dc96d8bc3
3 changed files with 196 additions and 1 deletions
+8
View File
@@ -155,6 +155,7 @@ libusb_init_context(libusb_context **context,
return (LIBUSB_ERROR_INVALID_PARAM);
memset(ctx, 0, sizeof(*ctx));
ctx->devd_pipe = -1;
debug = getenv("LIBUSB_DEBUG");
if (debug != NULL) {
@@ -280,6 +281,13 @@ libusb_exit(libusb_context *ctx)
HOTPLUG_LOCK(ctx);
td = ctx->hotplug_handler;
ctx->hotplug_handler = NO_THREAD;
if (ctx->usb_event_mode == usb_event_devd) {
close(ctx->devd_pipe);
ctx->devd_pipe = -1;
} else if (ctx->usb_event_mode == usb_event_netlink) {
close(ctx->ss.fd);
ctx->ss.fd = -1;
}
HOTPLUG_UNLOCK(ctx);
pthread_join(td, &ptr);
+16
View File
@@ -30,6 +30,12 @@
#ifndef LIBUSB_GLOBAL_INCLUDE_FILE
#include <sys/queue.h>
#include <netlink/netlink.h>
#include <netlink/netlink_generic.h>
#include <netlink/netlink_snl.h>
#include <netlink/netlink_snl_generic.h>
#include <netlink/netlink_sysevent.h>
#endif
#define GET_CONTEXT(ctx) (((ctx) == NULL) ? usbi_default_context : (ctx))
@@ -90,12 +96,22 @@ struct libusb_hotplug_callback_handle_struct {
TAILQ_HEAD(libusb_device_head, libusb_device);
typedef enum {
usb_event_none,
usb_event_scan,
usb_event_devd,
usb_event_netlink
} usb_event_mode_t;
struct libusb_context {
int debug;
int debug_fixed;
int ctrl_pipe[2];
int tr_done_ref;
int tr_done_gen;
usb_event_mode_t usb_event_mode;
int devd_pipe;
struct snl_state ss;
pthread_mutex_t ctx_lock;
pthread_mutex_t hotplug_lock;
+172 -1
View File
@@ -23,6 +23,7 @@
* SUCH DAMAGE.
*/
#include <netlink/netlink_snl_generic.h>
#ifdef LIBUSB_GLOBAL_INCLUDE_FILE
#include LIBUSB_GLOBAL_INCLUDE_FILE
#else
@@ -39,6 +40,10 @@
#include <sys/ioctl.h>
#include <sys/queue.h>
#include <sys/endian.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <sys/module.h>
#include <sys/linker.h>
#endif
#define libusb_device_handle libusb20_device
@@ -49,6 +54,118 @@
#include "libusb.h"
#include "libusb10.h"
#define DEVDPIPE "/var/run/devd.seqpacket.pipe"
#define DEVCTL_MAXBUF 1024
typedef enum {
broken_event,
invalid_event,
valid_event,
} event_t;
static bool
netlink_init(libusb_context *ctx)
{
struct _getfamily_attrs attrs;
if (modfind("nlsysevent") < 0)
kldload("nlsysevent");
if (modfind("nlsysevent") < 0)
return (false);
if (!snl_init(&ctx->ss, NETLINK_GENERIC))
return (false);
if (!snl_get_genl_family_info(&ctx->ss, "nlsysevent", &attrs))
return (false);
for (unsigned int i = 0; i < attrs.mcast_groups.num_groups; i++) {
if (strcmp(attrs.mcast_groups.groups[i]->mcast_grp_name,
"USB") == 0) {
if (setsockopt(ctx->ss.fd, SOL_NETLINK,
NETLINK_ADD_MEMBERSHIP,
&attrs.mcast_groups.groups[i]->mcast_grp_id,
sizeof(attrs.mcast_groups.groups[i]->mcast_grp_id))
== -1) {
return (false);
}
}
}
ctx->usb_event_mode = usb_event_netlink;
return (true);
}
static bool
devd_init(libusb_context *ctx)
{
struct sockaddr_un devd_addr;
bzero(&devd_addr, sizeof(devd_addr));
if ((ctx->devd_pipe = socket(PF_LOCAL, SOCK_SEQPACKET|SOCK_NONBLOCK, 0)) < 0)
return (false);
devd_addr.sun_family = PF_LOCAL;
strlcpy(devd_addr.sun_path, DEVDPIPE, sizeof(devd_addr.sun_path));
if (connect(ctx->devd_pipe, (struct sockaddr *)&devd_addr,
sizeof(devd_addr)) == -1) {
close(ctx->devd_pipe);
ctx->devd_pipe = -1;
return (false);
}
ctx->usb_event_mode = usb_event_devd;
return (true);
}
struct nlevent {
const char *name;
const char *subsystem;
const char *type;
const char *data;
};
#define _OUT(_field) offsetof(struct nlevent, _field)
static struct snl_attr_parser ap_nlevent_get[] = {
{ .type = NLSE_ATTR_SYSTEM, .off = _OUT(name), .cb = snl_attr_get_string },
{ .type = NLSE_ATTR_SUBSYSTEM, .off = _OUT(subsystem), .cb = snl_attr_get_string },
{ .type = NLSE_ATTR_TYPE, .off = _OUT(type), .cb = snl_attr_get_string },
{ .type = NLSE_ATTR_DATA, .off = _OUT(data), .cb = snl_attr_get_string },
};
#undef _OUT
SNL_DECLARE_GENL_PARSER(nlevent_get_parser, ap_nlevent_get);
static event_t
verify_event_validity(libusb_context *ctx)
{
if (ctx->usb_event_mode == usb_event_netlink) {
struct nlmsghdr *hdr;
struct nlevent ne;
hdr = snl_read_message(&ctx->ss);
if (hdr != NULL && hdr->nlmsg_type != NLMSG_ERROR) {
memset(&ne, 0, sizeof(ne));
if (!snl_parse_nlmsg(&ctx->ss, hdr, &nlevent_get_parser, &ne))
return (broken_event);
if (strcmp(ne.subsystem, "DEVICE") == 0)
return (valid_event);
return (invalid_event);
}
return (invalid_event);
} else if (ctx->usb_event_mode == usb_event_devd) {
char buf[DEVCTL_MAXBUF];
ssize_t len;
len = read(ctx->devd_pipe, buf, sizeof(buf));
if (len == 0 || (len < 0 && errno != EWOULDBLOCK))
return (broken_event);
if (len > 0 && strstr(buf, "system=USB") != NULL &&
strstr(buf, "subsystem=DEVICE") != NULL)
return (valid_event);
return (invalid_event);
}
return (broken_event);
}
static int
libusb_hotplug_equal(libusb_device *_adev, libusb_device *_bdev)
{
@@ -105,6 +222,7 @@ libusb_hotplug_enumerate(libusb_context *ctx, struct libusb_device_head *phead)
static void *
libusb_hotplug_scan(void *arg)
{
struct pollfd pfd;
struct libusb_device_head hotplug_devs;
libusb_hotplug_callback_handle acbh;
libusb_hotplug_callback_handle bcbh;
@@ -112,9 +230,51 @@ libusb_hotplug_scan(void *arg)
libusb_device *temp;
libusb_device *adev;
libusb_device *bdev;
int timeout = INFTIM;
int nfds;
memset(&pfd, 0, sizeof(pfd));
if (ctx->usb_event_mode == usb_event_devd) {
pfd.fd = ctx->devd_pipe;
pfd.events = POLLIN | POLLERR;
nfds = 1;
} else if (ctx->usb_event_mode == usb_event_netlink) {
pfd.fd = ctx->ss.fd;
pfd.events = POLLIN | POLLERR;
nfds = 1;
} else {
nfds = 0;
timeout = 4000;
}
for (;;) {
usleep(4000000);
pfd.revents = 0;
if (poll(&pfd, nfds, timeout) > 0) {
switch (verify_event_validity(ctx)) {
case invalid_event:
continue;
case valid_event:
break;
case broken_event:
/* There are 2 cases for broken events:
* - devd and netlink sockets are not available
* anymore (devd restarted, nlsysevent unloaded)
* - libusb_exit has been called as it sets NO_THREAD
* this will result in exiting this loop and this thread
* immediately
*/
nfds = 0;
if (ctx->usb_event_mode == usb_event_devd) {
if (ctx->devd_pipe != -1)
close(ctx->devd_pipe);
} else if (ctx->usb_event_mode == usb_event_netlink) {
if (ctx->ss.fd != -1)
close(ctx->ss.fd);
}
ctx->usb_event_mode = usb_event_scan;
timeout = 4000;
break;
}
}
HOTPLUG_LOCK(ctx);
if (ctx->hotplug_handler == NO_THREAD) {
@@ -122,6 +282,10 @@ libusb_hotplug_scan(void *arg)
TAILQ_REMOVE(&ctx->hotplug_devs, adev, hotplug_entry);
libusb_unref_device(adev);
}
if (ctx->usb_event_mode == usb_event_devd)
close(ctx->devd_pipe);
else if (ctx->usb_event_mode == usb_event_netlink)
close(ctx->ss.fd);
HOTPLUG_UNLOCK(ctx);
break;
}
@@ -192,6 +356,13 @@ int libusb_hotplug_register_callback(libusb_context *ctx,
ctx = GET_CONTEXT(ctx);
if (ctx->usb_event_mode == usb_event_none) {
HOTPLUG_LOCK(ctx);
if (!netlink_init(ctx) && !devd_init(ctx))
ctx->usb_event_mode = usb_event_scan;
HOTPLUG_UNLOCK(ctx);
}
if (ctx == NULL || cb_fn == NULL || events == 0 ||
vendor_id < -1 || vendor_id > 0xffff ||
product_id < -1 || product_id > 0xffff ||