Commit 0f431527b79fbd28fb9fcd898ff379331258354a
1 parent
03b4fe7d
Add USB sys file-system support (v8) (TJ)
This patch adds support for host USB devices discovered via: /sys/bus/usb/devices/* and opened from /dev/bus/usb/*/* /dev/bus/usb/devices and opened from /dev/bus/usb/*/* in addition to the existing discovery via: /proc/bus/usb/devices and opened from /proc/bus/usb/*/* Signed-off-by: TJ <linux@tjworld.net> Signed-off-by: Anthony Liguori <aliguori> git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@5441 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
218 additions
and
10 deletions
usb-linux.c
| ... | ... | @@ -7,6 +7,10 @@ |
| 7 | 7 | * Support for host device auto connect & disconnect |
| 8 | 8 | * Major rewrite to support fully async operation |
| 9 | 9 | * |
| 10 | + * Copyright 2008 TJ <linux@tjworld.net> | |
| 11 | + * Added flexible support for /dev/bus/usb /sys/bus/usb/devices in addition | |
| 12 | + * to the legacy /proc/bus/usb USB device discovery and handling | |
| 13 | + * | |
| 10 | 14 | * Permission is hereby granted, free of charge, to any person obtaining a copy |
| 11 | 15 | * of this software and associated documentation files (the "Software"), to deal |
| 12 | 16 | * in the Software without restriction, including without limitation the rights |
| ... | ... | @@ -72,9 +76,20 @@ static int usb_host_find_device(int *pbus_num, int *paddr, |
| 72 | 76 | #define dprintf(...) |
| 73 | 77 | #endif |
| 74 | 78 | |
| 75 | -#define USBDEVFS_PATH "/proc/bus/usb" | |
| 79 | +#define USBPROCBUS_PATH "/proc/bus/usb" | |
| 76 | 80 | #define PRODUCT_NAME_SZ 32 |
| 77 | 81 | #define MAX_ENDPOINTS 16 |
| 82 | +#define USBDEVBUS_PATH "/dev/bus/usb" | |
| 83 | +#define USBSYSBUS_PATH "/sys/bus/usb" | |
| 84 | + | |
| 85 | +static char *usb_host_device_path; | |
| 86 | + | |
| 87 | +#define USB_FS_NONE 0 | |
| 88 | +#define USB_FS_PROC 1 | |
| 89 | +#define USB_FS_DEV 2 | |
| 90 | +#define USB_FS_SYS 3 | |
| 91 | + | |
| 92 | +static int usb_fs_type; | |
| 78 | 93 | |
| 79 | 94 | /* endpoint association data */ |
| 80 | 95 | struct endp_data { |
| ... | ... | @@ -890,13 +905,18 @@ static USBDevice *usb_host_device_open_addr(int bus_num, int addr, const char *p |
| 890 | 905 | |
| 891 | 906 | printf("husb: open device %d.%d\n", bus_num, addr); |
| 892 | 907 | |
| 893 | - snprintf(buf, sizeof(buf), USBDEVFS_PATH "/%03d/%03d", | |
| 908 | + if (!usb_host_device_path) { | |
| 909 | + perror("husb: USB Host Device Path not set"); | |
| 910 | + goto fail; | |
| 911 | + } | |
| 912 | + snprintf(buf, sizeof(buf), "%s/%03d/%03d", usb_host_device_path, | |
| 894 | 913 | bus_num, addr); |
| 895 | 914 | fd = open(buf, O_RDWR | O_NONBLOCK); |
| 896 | 915 | if (fd < 0) { |
| 897 | 916 | perror(buf); |
| 898 | 917 | goto fail; |
| 899 | 918 | } |
| 919 | + dprintf("husb: opened %s\n", buf); | |
| 900 | 920 | |
| 901 | 921 | /* read the device description */ |
| 902 | 922 | dev->descr_len = read(fd, dev->descr, sizeof(dev->descr)); |
| ... | ... | @@ -1038,23 +1058,33 @@ static int get_tag_value(char *buf, int buf_size, |
| 1038 | 1058 | return q - buf; |
| 1039 | 1059 | } |
| 1040 | 1060 | |
| 1041 | -static int usb_host_scan(void *opaque, USBScanFunc *func) | |
| 1061 | +/* | |
| 1062 | + * Use /proc/bus/usb/devices or /dev/bus/usb/devices file to determine | |
| 1063 | + * host's USB devices. This is legacy support since many distributions | |
| 1064 | + * are moving to /sys/bus/usb | |
| 1065 | + */ | |
| 1066 | +static int usb_host_scan_dev(void *opaque, USBScanFunc *func) | |
| 1042 | 1067 | { |
| 1043 | - FILE *f; | |
| 1068 | + FILE *f = 0; | |
| 1044 | 1069 | char line[1024]; |
| 1045 | 1070 | char buf[1024]; |
| 1046 | 1071 | int bus_num, addr, speed, device_count, class_id, product_id, vendor_id; |
| 1047 | - int ret; | |
| 1048 | 1072 | char product_name[512]; |
| 1073 | + int ret = 0; | |
| 1049 | 1074 | |
| 1050 | - f = fopen(USBDEVFS_PATH "/devices", "r"); | |
| 1075 | + if (!usb_host_device_path) { | |
| 1076 | + perror("husb: USB Host Device Path not set"); | |
| 1077 | + goto the_end; | |
| 1078 | + } | |
| 1079 | + snprintf(line, sizeof(line), "%s/devices", usb_host_device_path); | |
| 1080 | + f = fopen(line, "r"); | |
| 1051 | 1081 | if (!f) { |
| 1052 | - term_printf("husb: could not open %s\n", USBDEVFS_PATH "/devices"); | |
| 1053 | - return 0; | |
| 1082 | + perror("husb: cannot open devices file"); | |
| 1083 | + goto the_end; | |
| 1054 | 1084 | } |
| 1085 | + | |
| 1055 | 1086 | device_count = 0; |
| 1056 | 1087 | bus_num = addr = speed = class_id = product_id = vendor_id = 0; |
| 1057 | - ret = 0; | |
| 1058 | 1088 | for(;;) { |
| 1059 | 1089 | if (fgets(line, sizeof(line), f) == NULL) |
| 1060 | 1090 | break; |
| ... | ... | @@ -1111,7 +1141,185 @@ static int usb_host_scan(void *opaque, USBScanFunc *func) |
| 1111 | 1141 | product_id, product_name, speed); |
| 1112 | 1142 | } |
| 1113 | 1143 | the_end: |
| 1114 | - fclose(f); | |
| 1144 | + if (f) | |
| 1145 | + fclose(f); | |
| 1146 | + return ret; | |
| 1147 | +} | |
| 1148 | + | |
| 1149 | +/* | |
| 1150 | + * Read sys file-system device file | |
| 1151 | + * | |
| 1152 | + * @line address of buffer to put file contents in | |
| 1153 | + * @line_size size of line | |
| 1154 | + * @device_file path to device file (printf format string) | |
| 1155 | + * @device_name device being opened (inserted into device_file) | |
| 1156 | + * | |
| 1157 | + * @return 0 failed, 1 succeeded ('line' contains data) | |
| 1158 | + */ | |
| 1159 | +static int usb_host_read_file(char *line, size_t line_size, const char *device_file, const char *device_name) | |
| 1160 | +{ | |
| 1161 | + FILE *f; | |
| 1162 | + int ret = 0; | |
| 1163 | + char filename[PATH_MAX]; | |
| 1164 | + | |
| 1165 | + snprintf(filename, PATH_MAX, device_file, device_name); | |
| 1166 | + f = fopen(filename, "r"); | |
| 1167 | + if (f) { | |
| 1168 | + fgets(line, line_size, f); | |
| 1169 | + fclose(f); | |
| 1170 | + ret = 1; | |
| 1171 | + } else { | |
| 1172 | + term_printf("husb: could not open %s\n", filename); | |
| 1173 | + } | |
| 1174 | + | |
| 1175 | + return ret; | |
| 1176 | +} | |
| 1177 | + | |
| 1178 | +/* | |
| 1179 | + * Use /sys/bus/usb/devices/ directory to determine host's USB | |
| 1180 | + * devices. | |
| 1181 | + * | |
| 1182 | + * This code is based on Robert Schiele's original patches posted to | |
| 1183 | + * the Novell bug-tracker https://bugzilla.novell.com/show_bug.cgi?id=241950 | |
| 1184 | + */ | |
| 1185 | +static int usb_host_scan_sys(void *opaque, USBScanFunc *func) | |
| 1186 | +{ | |
| 1187 | + DIR *dir = 0; | |
| 1188 | + char line[1024]; | |
| 1189 | + int bus_num, addr, speed, class_id, product_id, vendor_id; | |
| 1190 | + int ret = 0; | |
| 1191 | + char product_name[512]; | |
| 1192 | + struct dirent *de; | |
| 1193 | + | |
| 1194 | + dir = opendir(USBSYSBUS_PATH "/devices"); | |
| 1195 | + if (!dir) { | |
| 1196 | + perror("husb: cannot open devices directory"); | |
| 1197 | + goto the_end; | |
| 1198 | + } | |
| 1199 | + | |
| 1200 | + while ((de = readdir(dir))) { | |
| 1201 | + if (de->d_name[0] != '.' && !strchr(de->d_name, ':')) { | |
| 1202 | + char *tmpstr = de->d_name; | |
| 1203 | + if (!strncmp(de->d_name, "usb", 3)) | |
| 1204 | + tmpstr += 3; | |
| 1205 | + bus_num = atoi(tmpstr); | |
| 1206 | + | |
| 1207 | + if (!usb_host_read_file(line, sizeof(line), USBSYSBUS_PATH "/devices/%s/devnum", de->d_name)) | |
| 1208 | + goto the_end; | |
| 1209 | + if (sscanf(line, "%d", &addr) != 1) | |
| 1210 | + goto the_end; | |
| 1211 | + | |
| 1212 | + if (!usb_host_read_file(line, sizeof(line), USBSYSBUS_PATH "/devices/%s/bDeviceClass", de->d_name)) | |
| 1213 | + goto the_end; | |
| 1214 | + if (sscanf(line, "%x", &class_id) != 1) | |
| 1215 | + goto the_end; | |
| 1216 | + | |
| 1217 | + if (!usb_host_read_file(line, sizeof(line), USBSYSBUS_PATH "/devices/%s/idVendor", de->d_name)) | |
| 1218 | + goto the_end; | |
| 1219 | + if (sscanf(line, "%x", &vendor_id) != 1) | |
| 1220 | + goto the_end; | |
| 1221 | + | |
| 1222 | + if (!usb_host_read_file(line, sizeof(line), USBSYSBUS_PATH "/devices/%s/idProduct", de->d_name)) | |
| 1223 | + goto the_end; | |
| 1224 | + if (sscanf(line, "%x", &product_id) != 1) | |
| 1225 | + goto the_end; | |
| 1226 | + | |
| 1227 | + if (!usb_host_read_file(line, sizeof(line), USBSYSBUS_PATH "/devices/%s/product", de->d_name)) { | |
| 1228 | + *product_name = 0; | |
| 1229 | + } else { | |
| 1230 | + if (strlen(line) > 0) | |
| 1231 | + line[strlen(line) - 1] = '\0'; | |
| 1232 | + pstrcpy(product_name, sizeof(product_name), line); | |
| 1233 | + } | |
| 1234 | + | |
| 1235 | + if (!usb_host_read_file(line, sizeof(line), USBSYSBUS_PATH "/devices/%s/speed", de->d_name)) | |
| 1236 | + goto the_end; | |
| 1237 | + if (!strcmp(line, "480\n")) | |
| 1238 | + speed = USB_SPEED_HIGH; | |
| 1239 | + else if (!strcmp(line, "1.5\n")) | |
| 1240 | + speed = USB_SPEED_LOW; | |
| 1241 | + else | |
| 1242 | + speed = USB_SPEED_FULL; | |
| 1243 | + | |
| 1244 | + ret = func(opaque, bus_num, addr, class_id, vendor_id, | |
| 1245 | + product_id, product_name, speed); | |
| 1246 | + if (ret) | |
| 1247 | + goto the_end; | |
| 1248 | + } | |
| 1249 | + } | |
| 1250 | + the_end: | |
| 1251 | + if (dir) | |
| 1252 | + closedir(dir); | |
| 1253 | + return ret; | |
| 1254 | +} | |
| 1255 | + | |
| 1256 | +/* | |
| 1257 | + * Determine how to access the host's USB devices and call the | |
| 1258 | + * specific support function. | |
| 1259 | + */ | |
| 1260 | +static int usb_host_scan(void *opaque, USBScanFunc *func) | |
| 1261 | +{ | |
| 1262 | + FILE *f = 0; | |
| 1263 | + DIR *dir = 0; | |
| 1264 | + int ret = 0; | |
| 1265 | + const char *devices = "/devices"; | |
| 1266 | + const char *opened = "husb: opened %s%s\n"; | |
| 1267 | + const char *fs_type[] = {"unknown", "proc", "dev", "sys"}; | |
| 1268 | + char devpath[PATH_MAX]; | |
| 1269 | + | |
| 1270 | + /* only check the host once */ | |
| 1271 | + if (!usb_fs_type) { | |
| 1272 | + f = fopen(USBPROCBUS_PATH "/devices", "r"); | |
| 1273 | + if (f) { | |
| 1274 | + /* devices found in /proc/bus/usb/ */ | |
| 1275 | + strcpy(devpath, USBPROCBUS_PATH); | |
| 1276 | + usb_fs_type = USB_FS_PROC; | |
| 1277 | + fclose(f); | |
| 1278 | + dprintf(opened, USBPROCBUS_PATH, devices); | |
| 1279 | + } | |
| 1280 | + /* try additional methods if an access method hasn't been found yet */ | |
| 1281 | + f = fopen(USBDEVBUS_PATH "/devices", "r"); | |
| 1282 | + if (!usb_fs_type && f) { | |
| 1283 | + /* devices found in /dev/bus/usb/ */ | |
| 1284 | + strcpy(devpath, USBDEVBUS_PATH); | |
| 1285 | + usb_fs_type = USB_FS_DEV; | |
| 1286 | + fclose(f); | |
| 1287 | + dprintf(opened, USBDEVBUS_PATH, devices); | |
| 1288 | + } | |
| 1289 | + dir = opendir(USBSYSBUS_PATH "/devices"); | |
| 1290 | + if (!usb_fs_type && dir) { | |
| 1291 | + /* devices found in /dev/bus/usb/ (yes - not a mistake!) */ | |
| 1292 | + strcpy(devpath, USBDEVBUS_PATH); | |
| 1293 | + usb_fs_type = USB_FS_SYS; | |
| 1294 | + closedir(dir); | |
| 1295 | + dprintf(opened, USBSYSBUS_PATH, devices); | |
| 1296 | + } else { | |
| 1297 | + term_printf("husb: unable to access USB devices\n"); | |
| 1298 | + goto the_end; | |
| 1299 | + } | |
| 1300 | + | |
| 1301 | + /* the module setting (used later for opening devices) */ | |
| 1302 | + usb_host_device_path = qemu_mallocz(strlen(devpath)+1); | |
| 1303 | + if (usb_host_device_path) { | |
| 1304 | + strcpy(usb_host_device_path, devpath); | |
| 1305 | + term_printf("husb: using %s file-system with %s\n", fs_type[usb_fs_type], usb_host_device_path); | |
| 1306 | + } else { | |
| 1307 | + /* out of memory? */ | |
| 1308 | + perror("husb: unable to allocate memory for device path"); | |
| 1309 | + goto the_end; | |
| 1310 | + } | |
| 1311 | + } | |
| 1312 | + | |
| 1313 | + switch (usb_fs_type) { | |
| 1314 | + case USB_FS_PROC: | |
| 1315 | + case USB_FS_DEV: | |
| 1316 | + ret = usb_host_scan_dev(opaque, func); | |
| 1317 | + break; | |
| 1318 | + case USB_FS_SYS: | |
| 1319 | + ret = usb_host_scan_sys(opaque, func); | |
| 1320 | + break; | |
| 1321 | + } | |
| 1322 | + the_end: | |
| 1115 | 1323 | return ret; |
| 1116 | 1324 | } |
| 1117 | 1325 | ... | ... |