/* usbreset -- send a USB port reset to a USB device */ /* http://marc.info/?l=linux-usb-users&m=116827193506484&w=2 and needs mounted usbfs filesystem sudo mount -t usbfs none /proc/bus/usb There is a way to suspend a USB device. In order to use it, you must have a kernel with CONFIG_PM_SYSFS_DEPRECATED turned on. To suspend a device, do (as root): echo -n 2 >/sys/bus/usb/devices/.../power/state where the "..." is the ID for your device. To unsuspend, do the same thing but with a "0" instead of the "2" above. Note that this mechanism is slated to be removed from the kernel within the next year. Hopefully some other mechanism will take its place. > To reset a > device? Here's a program to do it. You invoke it as either usbreset /proc/bus/usb/BBB/DDD or usbreset /dev/usbB.D depending on how your system is set up, where BBB and DDD are the bus and device address numbers. Alan Stern */ #include <stdio.h> #include <stdbool.h> #include <unistd.h> #include <fcntl.h> #include <errno.h> #include <string.h> #include <sys/ioctl.h> #include <linux/usbdevice_fs.h> static char *usbfs = NULL; struct usbentry { int bus_num; int dev_num; int vendor_id; int product_id; char vendor_name[128]; char product_name[128]; }; static bool find_usbfs(void) { FILE *mtab; char buf[1024], type[32]; static char path[1024]; if ((mtab = fopen("/proc/mounts", "r")) != NULL) { while (fgets(buf, sizeof(buf), mtab)) { if (sscanf(buf, "%*s %1023s %31s ", path, type) == 2 && !strncmp(type, "usbfs", 5)) { usbfs = path; break; } } fclose(mtab); } return !!usbfs; } static FILE * open_devlist(void) { char buf[1024]; snprintf(buf, sizeof(buf), "%s/devices", usbfs); return fopen(buf, "r"); } static void close_devlist(FILE *devs) { fclose(devs); } static struct usbentry * parse_devlist(FILE *devs) { char buf[1024]; static struct usbentry dev; memset(&dev, 0, sizeof(dev)); while (fgets(buf, sizeof(buf), devs)) { buf[strlen(buf)-1] = 0; switch (buf[0]) { case 'T': sscanf(buf, "T: Bus=%d Lev=%*d Prnt=%*d Port=%*d Cnt=%*d Dev#=%d", &dev.bus_num, &dev.dev_num); break; case 'P': sscanf(buf, "P: Vendor=%x ProdID=%x", &dev.vendor_id, &dev.product_id); break; case 'S': if (!strncmp(buf, "S: Manufacturer=", 17)) snprintf(dev.vendor_name, sizeof(dev.vendor_name), "%s", buf+17); else if (!strncmp(buf, "S: Product=", 12)) snprintf(dev.product_name, sizeof(dev.product_name), "%s", buf+12); break; } if (dev.product_name[0]) return &dev; } return NULL; } static void list_devices(void) { FILE *devs = open_devlist(); struct usbentry *dev; if (!devs) return; while ((dev = parse_devlist(devs)) != NULL) { printf(" Number %03d/%03d ID %04x:%04x %s\n", dev->bus_num, dev->dev_num, dev->vendor_id, dev->product_id, dev->product_name); } close_devlist(devs); } struct usbentry * find_device(int *bus, int *dev, int *vid, int *pid, const char *product) { FILE *devs = open_devlist(); struct usbentry *e, *match = NULL; if (!devs) return NULL; while ((e = parse_devlist(devs)) != NULL) { if ((bus && (e->bus_num == *bus) && (e->dev_num == *dev)) || (vid && (e->vendor_id == *vid) && (e->product_id == *pid)) || (product && !strcasecmp(e->product_name, product))) { match = e; break; } } close_devlist(devs); return match; } static void reset_device(struct usbentry *dev) { int fd; char path[1024]; snprintf(path, sizeof(path), "%s/%03d/%03d", usbfs, dev->bus_num, dev->dev_num); printf("Resetting %s ... ", dev->product_name); if ((fd = open(path, O_WRONLY)) > -1) { if (ioctl(fd, USBDEVFS_RESET, 0) < 0) printf("failed [%s]\n", strerror(errno)); else printf("ok\n"); close(fd); } else { printf("can't open [%s]\n", strerror(errno)); } } int main(int argc, char **argv) { int id1, id2; struct usbentry *dev; if (!find_usbfs()) { fprintf(stderr, "Unable to find usbfs, is it mounted?\n"); return 1; } if ((argc == 2) && (sscanf(argv[1], "%3d/%3d", &id1, &id2) == 2)) { dev = find_device(&id1, &id2, NULL, NULL, NULL); } else if ((argc == 2) && (sscanf(argv[1], "%4x:%4x", &id1, &id2) == 2)) { dev = find_device(NULL, NULL, &id1, &id2, NULL); } else if ((argc == 2) && strlen(argv[1]) < 128) { dev = find_device(NULL, NULL, NULL, NULL, argv[1]); } else { printf("Usage:\n" " usbreset PPPP:VVVV - reset by product and vendor id\n" " usbreset BBB/DDD - reset by bus and device number\n" " usbreset \"Product\" - reset by product name\n\n" "Devices:\n"); list_devices(); return 1; } if (!dev) { fprintf(stderr, "No such device found\n"); return 1; } reset_device(dev); return 0; }