usb: Add environment based device ignorelist

Add the environment variable "usb_ignorelist" to prevent USB devices
listed in it from being bound to drivers. This allows to ignore devices
which are undesirable or trigger bugs in u-boot's USB stack.
Devices emulating keyboards are one example of undesirable devices as
u-boot currently supports only a single USB keyboard device. Most
commonly, people run into this with Yubikeys, so let's ignore those in
the default environment.

Based on previous USB keyboard specific patches for the same purpose.

Link: https://lore.kernel.org/u-boot/7ab604fb-0fec-4f5e-8708-7a3a7e2cb568@denx.de/
Reviewed-by: Neal Gompa <neal@gompa.dev>
Reviewed-by: Marek Vasut <marex@denx.de>
Signed-off-by: Janne Grunau <j@jannau.net>
This commit is contained in:
Janne Grunau 2024-04-04 08:25:52 +02:00 committed by Marek Vasut
parent 0b6775c402
commit 18f288b8d4
3 changed files with 94 additions and 0 deletions

View file

@ -28,6 +28,7 @@
#include <common.h>
#include <command.h>
#include <dm.h>
#include <dm/device_compat.h>
#include <log.h>
#include <malloc.h>
#include <memalign.h>
@ -1084,6 +1085,54 @@ static int usb_prepare_device(struct usb_device *dev, int addr, bool do_read,
return 0;
}
static int usb_device_is_ignored(u16 id_vendor, u16 id_product)
{
ulong vid, pid;
char *end;
const char *cur = NULL;
/* ignore list depends on env support */
if (!CONFIG_IS_ENABLED(ENV_SUPPORT))
return 0;
cur = env_get("usb_ignorelist");
/* parse "usb_ignorelist" strictly */
while (cur && cur[0] != '\0') {
vid = simple_strtoul(cur, &end, 0);
/*
* If strtoul did not parse a single digit or the next char is
* not ':' the ignore list is malformed.
*/
if (cur == end || end[0] != ':')
return -EINVAL;
cur = end + 1;
pid = simple_strtoul(cur, &end, 0);
/* Consider '*' as wildcard for the product ID */
if (cur == end && end[0] == '*') {
pid = U16_MAX + 1;
end++;
}
/*
* The ignore list is malformed if no product ID / wildcard was
* parsed or entries are not separated by ',' or terminated with
* '\0'.
*/
if (cur == end || (end[0] != ',' && end[0] != '\0'))
return -EINVAL;
if (id_vendor == vid && (pid > U16_MAX || id_product == pid))
return -ENODEV;
if (end[0] == '\0')
break;
cur = end + 1;
}
return 0;
}
int usb_select_config(struct usb_device *dev)
{
unsigned char *tmpbuf = NULL;
@ -1099,6 +1148,27 @@ int usb_select_config(struct usb_device *dev)
le16_to_cpus(&dev->descriptor.idProduct);
le16_to_cpus(&dev->descriptor.bcdDevice);
/* ignore devices from usb_ignorelist */
err = usb_device_is_ignored(dev->descriptor.idVendor,
dev->descriptor.idProduct);
if (err == -ENODEV) {
debug("Ignoring USB device 0x%x:0x%x\n",
dev->descriptor.idVendor, dev->descriptor.idProduct);
return err;
} else if (err == -EINVAL) {
/*
* Continue on "usb_ignorelist" parsing errors. The list is
* parsed for each device returning the error would result in
* ignoring all USB devices.
* Since the parsing error is independent of the probed device
* report errors with printf instead of dev_err.
*/
printf("usb_ignorelist parse error in \"%s\"\n",
env_get("usb_ignorelist"));
} else if (err < 0) {
return err;
}
/*
* Kingston DT Ultimate 32GB USB 3.0 seems to be extremely sensitive
* about this first Get Descriptor request. If there are any other

View file

@ -366,6 +366,19 @@ tftpwindowsize
This means the count of blocks we can receive before
sending ack to server.
usb_ignorelist
Ignore USB devices to prevent binding them to an USB device driver. This can
be used to ignore devices are for some reason undesirable or causes crashes
u-boot's USB stack.
An example for undesired behavior is the keyboard emulation of security keys
like Yubikeys. U-boot currently supports only a single USB keyboard device
so try to probe an useful keyboard device. The default environment blocks
Yubico devices as common devices emulating keyboards.
Devices are matched by idVendor and idProduct. The variable contains a comma
separated list of idVendor:idProduct pairs as hexadecimal numbers joined
by a colon. '*' functions as a wildcard for idProduct to block all devices
with the specified idVendor.
vlan
When set to a value < 4095 the traffic over
Ethernet is encapsulated/received over 802.1q

View file

@ -99,6 +99,17 @@ const char default_environment[] = {
#ifdef CONFIG_SYS_SOC
"soc=" CONFIG_SYS_SOC "\0"
#endif
#ifdef CONFIG_USB_HOST
"usb_ignorelist="
#ifdef CONFIG_USB_KEYBOARD
/* Ignore Yubico devices. Currently only a single USB keyboard device is
* supported and the emulated HID keyboard Yubikeys present is useless
* as keyboard.
*/
"0x1050:*,"
#endif
"\0"
#endif
#ifdef CONFIG_ENV_IMPORT_FDT
"env_fdt_path=" CONFIG_ENV_FDT_PATH "\0"
#endif