1 2 Linux USB HID gadget driver 3 4Introduction 5 6 The HID Gadget driver provides emulation of USB Human Interface 7 Devices (HID). The basic HID handling is done in the kernel, 8 and HID reports can be sent/received through I/O on the 9 /dev/hidgX character devices. 10 11 For more details about HID, see the developer page on 12 http://www.usb.org/developers/hidpage/ 13 14Configuration 15 16 g_hid is a platform driver, so to use it you need to add 17 struct platform_device(s) to your platform code defining the 18 HID function descriptors you want to use - E.G. something 19 like: 20 21#include <linux/platform_device.h> 22#include <linux/usb/g_hid.h> 23 24/* hid descriptor for a keyboard */ 25static struct hidg_func_descriptor my_hid_data = { 26 .subclass = 0, /* No subclass */ 27 .protocol = 1, /* Keyboard */ 28 .report_length = 8, 29 .report_desc_length = 63, 30 .report_desc = { 31 0x05, 0x01, /* USAGE_PAGE (Generic Desktop) */ 32 0x09, 0x06, /* USAGE (Keyboard) */ 33 0xa1, 0x01, /* COLLECTION (Application) */ 34 0x05, 0x07, /* USAGE_PAGE (Keyboard) */ 35 0x19, 0xe0, /* USAGE_MINIMUM (Keyboard LeftControl) */ 36 0x29, 0xe7, /* USAGE_MAXIMUM (Keyboard Right GUI) */ 37 0x15, 0x00, /* LOGICAL_MINIMUM (0) */ 38 0x25, 0x01, /* LOGICAL_MAXIMUM (1) */ 39 0x75, 0x01, /* REPORT_SIZE (1) */ 40 0x95, 0x08, /* REPORT_COUNT (8) */ 41 0x81, 0x02, /* INPUT (Data,Var,Abs) */ 42 0x95, 0x01, /* REPORT_COUNT (1) */ 43 0x75, 0x08, /* REPORT_SIZE (8) */ 44 0x81, 0x03, /* INPUT (Cnst,Var,Abs) */ 45 0x95, 0x05, /* REPORT_COUNT (5) */ 46 0x75, 0x01, /* REPORT_SIZE (1) */ 47 0x05, 0x08, /* USAGE_PAGE (LEDs) */ 48 0x19, 0x01, /* USAGE_MINIMUM (Num Lock) */ 49 0x29, 0x05, /* USAGE_MAXIMUM (Kana) */ 50 0x91, 0x02, /* OUTPUT (Data,Var,Abs) */ 51 0x95, 0x01, /* REPORT_COUNT (1) */ 52 0x75, 0x03, /* REPORT_SIZE (3) */ 53 0x91, 0x03, /* OUTPUT (Cnst,Var,Abs) */ 54 0x95, 0x06, /* REPORT_COUNT (6) */ 55 0x75, 0x08, /* REPORT_SIZE (8) */ 56 0x15, 0x00, /* LOGICAL_MINIMUM (0) */ 57 0x25, 0x65, /* LOGICAL_MAXIMUM (101) */ 58 0x05, 0x07, /* USAGE_PAGE (Keyboard) */ 59 0x19, 0x00, /* USAGE_MINIMUM (Reserved) */ 60 0x29, 0x65, /* USAGE_MAXIMUM (Keyboard Application) */ 61 0x81, 0x00, /* INPUT (Data,Ary,Abs) */ 62 0xc0 /* END_COLLECTION */ 63 } 64}; 65 66static struct platform_device my_hid = { 67 .name = "hidg", 68 .id = 0, 69 .num_resources = 0, 70 .resource = 0, 71 .dev.platform_data = &my_hid_data, 72}; 73 74 You can add as many HID functions as you want, only limited by 75 the amount of interrupt endpoints your gadget driver supports. 76 77Send and receive HID reports 78 79 HID reports can be sent/received using read/write on the 80 /dev/hidgX character devices. See below for an example program 81 to do this. 82 83 hid_gadget_test is a small interactive program to test the HID 84 gadget driver. To use, point it at a hidg device and set the 85 device type (keyboard / mouse / joystick) - E.G.: 86 87 # hid_gadget_test /dev/hidg0 keyboard 88 89 You are now in the prompt of hid_gadget_test. You can type any 90 combination of options and values. Available options and 91 values are listed at program start. In keyboard mode you can 92 send up to six values. 93 94 For example type: g i s t r --left-shift 95 96 Hit return and the corresponding report will be sent by the 97 HID gadget. 98 99 Another interesting example is the caps lock test. Type 100 --caps-lock and hit return. A report is then sent by the 101 gadget and you should receive the host answer, corresponding 102 to the caps lock LED status. 103 104 --caps-lock 105 recv report:2 106 107 With this command: 108 109 # hid_gadget_test /dev/hidg1 mouse 110 111 You can test the mouse emulation. Values are two signed numbers. 112 113 114Sample code 115 116/* hid_gadget_test */ 117 118#include <pthread.h> 119#include <string.h> 120#include <stdio.h> 121#include <ctype.h> 122#include <fcntl.h> 123#include <errno.h> 124#include <stdio.h> 125#include <stdlib.h> 126#include <unistd.h> 127 128#define BUF_LEN 512 129 130struct options { 131 const char *opt; 132 unsigned char val; 133}; 134 135static struct options kmod[] = { 136 {.opt = "--left-ctrl", .val = 0x01}, 137 {.opt = "--right-ctrl", .val = 0x10}, 138 {.opt = "--left-shift", .val = 0x02}, 139 {.opt = "--right-shift", .val = 0x20}, 140 {.opt = "--left-alt", .val = 0x04}, 141 {.opt = "--right-alt", .val = 0x40}, 142 {.opt = "--left-meta", .val = 0x08}, 143 {.opt = "--right-meta", .val = 0x80}, 144 {.opt = NULL} 145}; 146 147static struct options kval[] = { 148 {.opt = "--return", .val = 0x28}, 149 {.opt = "--esc", .val = 0x29}, 150 {.opt = "--bckspc", .val = 0x2a}, 151 {.opt = "--tab", .val = 0x2b}, 152 {.opt = "--spacebar", .val = 0x2c}, 153 {.opt = "--caps-lock", .val = 0x39}, 154 {.opt = "--f1", .val = 0x3a}, 155 {.opt = "--f2", .val = 0x3b}, 156 {.opt = "--f3", .val = 0x3c}, 157 {.opt = "--f4", .val = 0x3d}, 158 {.opt = "--f5", .val = 0x3e}, 159 {.opt = "--f6", .val = 0x3f}, 160 {.opt = "--f7", .val = 0x40}, 161 {.opt = "--f8", .val = 0x41}, 162 {.opt = "--f9", .val = 0x42}, 163 {.opt = "--f10", .val = 0x43}, 164 {.opt = "--f11", .val = 0x44}, 165 {.opt = "--f12", .val = 0x45}, 166 {.opt = "--insert", .val = 0x49}, 167 {.opt = "--home", .val = 0x4a}, 168 {.opt = "--pageup", .val = 0x4b}, 169 {.opt = "--del", .val = 0x4c}, 170 {.opt = "--end", .val = 0x4d}, 171 {.opt = "--pagedown", .val = 0x4e}, 172 {.opt = "--right", .val = 0x4f}, 173 {.opt = "--left", .val = 0x50}, 174 {.opt = "--down", .val = 0x51}, 175 {.opt = "--kp-enter", .val = 0x58}, 176 {.opt = "--up", .val = 0x52}, 177 {.opt = "--num-lock", .val = 0x53}, 178 {.opt = NULL} 179}; 180 181int keyboard_fill_report(char report[8], char buf[BUF_LEN], int *hold) 182{ 183 char *tok = strtok(buf, " "); 184 int key = 0; 185 int i = 0; 186 187 for (; tok != NULL; tok = strtok(NULL, " ")) { 188 189 if (strcmp(tok, "--quit") == 0) 190 return -1; 191 192 if (strcmp(tok, "--hold") == 0) { 193 *hold = 1; 194 continue; 195 } 196 197 if (key < 6) { 198 for (i = 0; kval[i].opt != NULL; i++) 199 if (strcmp(tok, kval[i].opt) == 0) { 200 report[2 + key++] = kval[i].val; 201 break; 202 } 203 if (kval[i].opt != NULL) 204 continue; 205 } 206 207 if (key < 6) 208 if (islower(tok[0])) { 209 report[2 + key++] = (tok[0] - ('a' - 0x04)); 210 continue; 211 } 212 213 for (i = 0; kmod[i].opt != NULL; i++) 214 if (strcmp(tok, kmod[i].opt) == 0) { 215 report[0] = report[0] | kmod[i].val; 216 break; 217 } 218 if (kmod[i].opt != NULL) 219 continue; 220 221 if (key < 6) 222 fprintf(stderr, "unknown option: %s\n", tok); 223 } 224 return 8; 225} 226 227static struct options mmod[] = { 228 {.opt = "--b1", .val = 0x01}, 229 {.opt = "--b2", .val = 0x02}, 230 {.opt = "--b3", .val = 0x04}, 231 {.opt = NULL} 232}; 233 234int mouse_fill_report(char report[8], char buf[BUF_LEN], int *hold) 235{ 236 char *tok = strtok(buf, " "); 237 int mvt = 0; 238 int i = 0; 239 for (; tok != NULL; tok = strtok(NULL, " ")) { 240 241 if (strcmp(tok, "--quit") == 0) 242 return -1; 243 244 if (strcmp(tok, "--hold") == 0) { 245 *hold = 1; 246 continue; 247 } 248 249 for (i = 0; mmod[i].opt != NULL; i++) 250 if (strcmp(tok, mmod[i].opt) == 0) { 251 report[0] = report[0] | mmod[i].val; 252 break; 253 } 254 if (mmod[i].opt != NULL) 255 continue; 256 257 if (!(tok[0] == '-' && tok[1] == '-') && mvt < 2) { 258 errno = 0; 259 report[1 + mvt++] = (char)strtol(tok, NULL, 0); 260 if (errno != 0) { 261 fprintf(stderr, "Bad value:'%s'\n", tok); 262 report[1 + mvt--] = 0; 263 } 264 continue; 265 } 266 267 fprintf(stderr, "unknown option: %s\n", tok); 268 } 269 return 3; 270} 271 272static struct options jmod[] = { 273 {.opt = "--b1", .val = 0x10}, 274 {.opt = "--b2", .val = 0x20}, 275 {.opt = "--b3", .val = 0x40}, 276 {.opt = "--b4", .val = 0x80}, 277 {.opt = "--hat1", .val = 0x00}, 278 {.opt = "--hat2", .val = 0x01}, 279 {.opt = "--hat3", .val = 0x02}, 280 {.opt = "--hat4", .val = 0x03}, 281 {.opt = "--hatneutral", .val = 0x04}, 282 {.opt = NULL} 283}; 284 285int joystick_fill_report(char report[8], char buf[BUF_LEN], int *hold) 286{ 287 char *tok = strtok(buf, " "); 288 int mvt = 0; 289 int i = 0; 290 291 *hold = 1; 292 293 /* set default hat position: neutral */ 294 report[3] = 0x04; 295 296 for (; tok != NULL; tok = strtok(NULL, " ")) { 297 298 if (strcmp(tok, "--quit") == 0) 299 return -1; 300 301 for (i = 0; jmod[i].opt != NULL; i++) 302 if (strcmp(tok, jmod[i].opt) == 0) { 303 report[3] = (report[3] & 0xF0) | jmod[i].val; 304 break; 305 } 306 if (jmod[i].opt != NULL) 307 continue; 308 309 if (!(tok[0] == '-' && tok[1] == '-') && mvt < 3) { 310 errno = 0; 311 report[mvt++] = (char)strtol(tok, NULL, 0); 312 if (errno != 0) { 313 fprintf(stderr, "Bad value:'%s'\n", tok); 314 report[mvt--] = 0; 315 } 316 continue; 317 } 318 319 fprintf(stderr, "unknown option: %s\n", tok); 320 } 321 return 4; 322} 323 324void print_options(char c) 325{ 326 int i = 0; 327 328 if (c == 'k') { 329 printf(" keyboard options:\n" 330 " --hold\n"); 331 for (i = 0; kmod[i].opt != NULL; i++) 332 printf("\t\t%s\n", kmod[i].opt); 333 printf("\n keyboard values:\n" 334 " [a-z] or\n"); 335 for (i = 0; kval[i].opt != NULL; i++) 336 printf("\t\t%-8s%s", kval[i].opt, i % 2 ? "\n" : ""); 337 printf("\n"); 338 } else if (c == 'm') { 339 printf(" mouse options:\n" 340 " --hold\n"); 341 for (i = 0; mmod[i].opt != NULL; i++) 342 printf("\t\t%s\n", mmod[i].opt); 343 printf("\n mouse values:\n" 344 " Two signed numbers\n" 345 "--quit to close\n"); 346 } else { 347 printf(" joystick options:\n"); 348 for (i = 0; jmod[i].opt != NULL; i++) 349 printf("\t\t%s\n", jmod[i].opt); 350 printf("\n joystick values:\n" 351 " three signed numbers\n" 352 "--quit to close\n"); 353 } 354} 355 356int main(int argc, const char *argv[]) 357{ 358 const char *filename = NULL; 359 int fd = 0; 360 char buf[BUF_LEN]; 361 int cmd_len; 362 char report[8]; 363 int to_send = 8; 364 int hold = 0; 365 fd_set rfds; 366 int retval, i; 367 368 if (argc < 3) { 369 fprintf(stderr, "Usage: %s devname mouse|keyboard|joystick\n", 370 argv[0]); 371 return 1; 372 } 373 374 if (argv[2][0] != 'k' && argv[2][0] != 'm' && argv[2][0] != 'j') 375 return 2; 376 377 filename = argv[1]; 378 379 if ((fd = open(filename, O_RDWR, 0666)) == -1) { 380 perror(filename); 381 return 3; 382 } 383 384 print_options(argv[2][0]); 385 386 while (42) { 387 388 FD_ZERO(&rfds); 389 FD_SET(STDIN_FILENO, &rfds); 390 FD_SET(fd, &rfds); 391 392 retval = select(fd + 1, &rfds, NULL, NULL, NULL); 393 if (retval == -1 && errno == EINTR) 394 continue; 395 if (retval < 0) { 396 perror("select()"); 397 return 4; 398 } 399 400 if (FD_ISSET(fd, &rfds)) { 401 cmd_len = read(fd, buf, BUF_LEN - 1); 402 printf("recv report:"); 403 for (i = 0; i < cmd_len; i++) 404 printf(" %02x", buf[i]); 405 printf("\n"); 406 } 407 408 if (FD_ISSET(STDIN_FILENO, &rfds)) { 409 memset(report, 0x0, sizeof(report)); 410 cmd_len = read(STDIN_FILENO, buf, BUF_LEN - 1); 411 412 if (cmd_len == 0) 413 break; 414 415 buf[cmd_len - 1] = '\0'; 416 hold = 0; 417 418 memset(report, 0x0, sizeof(report)); 419 if (argv[2][0] == 'k') 420 to_send = keyboard_fill_report(report, buf, &hold); 421 else if (argv[2][0] == 'm') 422 to_send = mouse_fill_report(report, buf, &hold); 423 else 424 to_send = joystick_fill_report(report, buf, &hold); 425 426 if (to_send == -1) 427 break; 428 429 if (write(fd, report, to_send) != to_send) { 430 perror(filename); 431 return 5; 432 } 433 if (!hold) { 434 memset(report, 0x0, sizeof(report)); 435 if (write(fd, report, to_send) != to_send) { 436 perror(filename); 437 return 6; 438 } 439 } 440 } 441 } 442 443 close(fd); 444 return 0; 445} 446