Changeset 1882 for libusb1/trunk/libusb/descriptor.c
- Timestamp:
- Dec 12, 2016, 6:38:13 PM (8 years ago)
- Location:
- libusb1/trunk
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
libusb1/trunk ¶
-
Property svn:mergeinfo
set to
/libusb1/vendor/current merged eligible
-
Property svn:mergeinfo
set to
-
TabularUnified libusb1/trunk/libusb/descriptor.c ¶
r1876 r1882 1 /* -*- Mode: C; indent-tabs-mode:t ; c-basic-offset:8 -*- */ 1 2 /* 2 3 * USB descriptor handling functions for libusb 3 * Copyright (C) 2007 Daniel Drake <dsd@gentoo.org> 4 * Copyright (c) 2001 Johannes Erdfelt <johannes@erdfelt.com> 5 * Copyright (c) 2012-2013 Nathan Hjelm <hjelmn@cs.unm.edu> 4 * Copyright © 2007 Daniel Drake <dsd@gentoo.org> 5 * Copyright © 2001 Johannes Erdfelt <johannes@erdfelt.com> 6 6 * 7 7 * This library is free software; you can redistribute it and/or … … 20 20 */ 21 21 22 #include <config.h> 23 22 24 #include <errno.h> 23 25 #include <stdint.h> 24 26 #include <stdlib.h> 25 27 #include <string.h> 26 #include <stdio.h>27 #include <assert.h>28 28 29 29 #include "libusbi.h" … … 36 36 #define ENDPOINT_AUDIO_DESC_LENGTH 9 37 37 38 /** @defgroup desc USB descriptors38 /** @defgroup libusb_desc USB descriptors 39 39 * This page details how to examine the various standard USB descriptors 40 40 * for detected devices … … 69 69 dp += 2; 70 70 break; 71 /* 32-bit word, convert from little endian to CPU */ 72 case 'd': 73 /* Align to word boundary */ 74 dp += ((unsigned long)dp & 1); 71 case 'd': /* 32-bit word, convert from little endian to CPU */ 72 dp += ((uintptr_t)dp & 1); /* Align to word boundary */ 75 73 76 74 if (host_endian) { … … 84 82 dp += 4; 85 83 break; 84 case 'u': /* 16 byte UUID */ 85 memcpy(dp, sp, 16); 86 sp += 16; 87 dp += 16; 88 break; 86 89 } 87 90 } … … 92 95 static void clear_endpoint(struct libusb_endpoint_descriptor *endpoint) 93 96 { 94 if (endpoint->extra) 95 free((unsigned char *) endpoint->extra); 97 free((void *) endpoint->extra); 96 98 } 97 99 … … 106 108 int len; 107 109 110 if (size < DESC_HEADER_LENGTH) { 111 usbi_err(ctx, "short endpoint descriptor read %d/%d", 112 size, DESC_HEADER_LENGTH); 113 return LIBUSB_ERROR_IO; 114 } 115 108 116 usbi_parse_descriptor(buffer, "bb", &header, 0); 109 110 /* Everything should be fine being passed into here, but we sanity */111 /* check JIC */112 if (header.bLength > size) {113 usbi_err(ctx, "ran out of descriptors parsing");114 return -1;115 }116 117 117 if (header.bDescriptorType != LIBUSB_DT_ENDPOINT) { 118 118 usbi_err(ctx, "unexpected descriptor %x (expected %x)", … … 120 120 return parsed; 121 121 } 122 122 if (header.bLength > size) { 123 usbi_warn(ctx, "short endpoint descriptor read %d/%d", 124 size, header.bLength); 125 return parsed; 126 } 123 127 if (header.bLength >= ENDPOINT_AUDIO_DESC_LENGTH) 124 128 usbi_parse_descriptor(buffer, "bbbbwbbb", endpoint, host_endian); 125 129 else if (header.bLength >= ENDPOINT_DESC_LENGTH) 126 130 usbi_parse_descriptor(buffer, "bbbbwb", endpoint, host_endian); 131 else { 132 usbi_err(ctx, "invalid endpoint bLength (%d)", header.bLength); 133 return LIBUSB_ERROR_IO; 134 } 127 135 128 136 buffer += header.bLength; … … 135 143 while (size >= DESC_HEADER_LENGTH) { 136 144 usbi_parse_descriptor(buffer, "bb", &header, 0); 137 138 if (header.bLength < 2) { 139 usbi_err(ctx, "invalid descriptor length %d", header.bLength); 140 return -1; 145 if (header.bLength < DESC_HEADER_LENGTH) { 146 usbi_err(ctx, "invalid extra ep desc len (%d)", 147 header.bLength); 148 return LIBUSB_ERROR_IO; 149 } else if (header.bLength > size) { 150 usbi_warn(ctx, "short extra ep desc read %d/%d", 151 size, header.bLength); 152 return parsed; 141 153 } 142 154 143 155 /* If we find another "proper" descriptor then we're done */ 144 156 if ((header.bDescriptorType == LIBUSB_DT_ENDPOINT) || 145 146 147 157 (header.bDescriptorType == LIBUSB_DT_INTERFACE) || 158 (header.bDescriptorType == LIBUSB_DT_CONFIG) || 159 (header.bDescriptorType == LIBUSB_DT_DEVICE)) 148 160 break; 149 161 … … 186 198 (struct libusb_interface_descriptor *) 187 199 usb_interface->altsetting + i; 188 if (ifp->extra) 189 free((void *) ifp->extra); 200 free((void *) ifp->extra); 190 201 if (ifp->endpoint) { 191 202 for (j = 0; j < ifp->bNumEndpoints; j++) 192 203 clear_endpoint((struct libusb_endpoint_descriptor *) 193 ifp->endpoint + j); 194 free((void *) ifp->endpoint); 204 ifp->endpoint + j); 195 205 } 196 }197 free((void *) usb_interface->altsetting);198 usb_interface->altsetting = NULL;199 }200 206 free((void *) ifp->endpoint); 207 } 208 } 209 free((void *) usb_interface->altsetting); 210 usb_interface->altsetting = NULL; 201 211 } 202 212 … … 209 219 int r; 210 220 int parsed = 0; 211 size_t tmp;221 int interface_number = -1; 212 222 struct usb_descriptor_header header; 213 223 struct libusb_interface_descriptor *ifp; … … 219 229 struct libusb_interface_descriptor *altsetting = 220 230 (struct libusb_interface_descriptor *) usb_interface->altsetting; 221 altsetting = realloc(altsetting,231 altsetting = usbi_reallocf(altsetting, 222 232 sizeof(struct libusb_interface_descriptor) * 223 233 (usb_interface->num_altsetting + 1)); … … 229 239 230 240 ifp = altsetting + usb_interface->num_altsetting; 241 usbi_parse_descriptor(buffer, "bbbbbbbbb", ifp, 0); 242 if (ifp->bDescriptorType != LIBUSB_DT_INTERFACE) { 243 usbi_err(ctx, "unexpected descriptor %x (expected %x)", 244 ifp->bDescriptorType, LIBUSB_DT_INTERFACE); 245 return parsed; 246 } 247 if (ifp->bLength < INTERFACE_DESC_LENGTH) { 248 usbi_err(ctx, "invalid interface bLength (%d)", 249 ifp->bLength); 250 r = LIBUSB_ERROR_IO; 251 goto err; 252 } 253 if (ifp->bLength > size) { 254 usbi_warn(ctx, "short intf descriptor read %d/%d", 255 size, ifp->bLength); 256 return parsed; 257 } 258 if (ifp->bNumEndpoints > USB_MAXENDPOINTS) { 259 usbi_err(ctx, "too many endpoints (%d)", ifp->bNumEndpoints); 260 r = LIBUSB_ERROR_IO; 261 goto err; 262 } 263 231 264 usb_interface->num_altsetting++; 232 usbi_parse_descriptor(buffer, "bbbbbbbbb", ifp, 0);233 265 ifp->extra = NULL; 234 266 ifp->extra_length = 0; 235 267 ifp->endpoint = NULL; 268 269 if (interface_number == -1) 270 interface_number = ifp->bInterfaceNumber; 236 271 237 272 /* Skip over the interface */ … … 245 280 while (size >= DESC_HEADER_LENGTH) { 246 281 usbi_parse_descriptor(buffer, "bb", &header, 0); 247 if (header.bLength < 2) { 248 usbi_err(ctx, "invalid descriptor of length %d", 249 header.bLength); 282 if (header.bLength < DESC_HEADER_LENGTH) { 283 usbi_err(ctx, 284 "invalid extra intf desc len (%d)", 285 header.bLength); 250 286 r = LIBUSB_ERROR_IO; 251 287 goto err; 288 } else if (header.bLength > size) { 289 usbi_warn(ctx, 290 "short extra intf desc read %d/%d", 291 size, header.bLength); 292 return parsed; 252 293 } 253 294 254 295 /* If we find another "proper" descriptor then we're done */ 255 296 if ((header.bDescriptorType == LIBUSB_DT_INTERFACE) || 256 (header.bDescriptorType == LIBUSB_DT_ENDPOINT) || 257 (header.bDescriptorType == LIBUSB_DT_CONFIG) || 258 (header.bDescriptorType == LIBUSB_DT_DEVICE) || 259 (header.bDescriptorType == 260 LIBUSB_DT_SS_ENDPOINT_COMPANION)) 297 (header.bDescriptorType == LIBUSB_DT_ENDPOINT) || 298 (header.bDescriptorType == LIBUSB_DT_CONFIG) || 299 (header.bDescriptorType == LIBUSB_DT_DEVICE)) 261 300 break; 262 301 … … 279 318 } 280 319 281 /* Did we hit an unexpected descriptor? */282 if (size >= DESC_HEADER_LENGTH) {283 usbi_parse_descriptor(buffer, "bb", &header, 0);284 if ((header.bDescriptorType == LIBUSB_DT_CONFIG) ||285 (header.bDescriptorType == LIBUSB_DT_DEVICE)) {286 return parsed;287 }288 }289 290 if (ifp->bNumEndpoints > USB_MAXENDPOINTS) {291 usbi_err(ctx, "too many endpoints (%d)", ifp->bNumEndpoints);292 r = LIBUSB_ERROR_IO;293 goto err;294 }295 296 320 if (ifp->bNumEndpoints > 0) { 297 321 struct libusb_endpoint_descriptor *endpoint; 298 tmp = ifp->bNumEndpoints * sizeof(struct libusb_endpoint_descriptor); 299 endpoint = malloc(tmp); 322 endpoint = calloc(ifp->bNumEndpoints, sizeof(struct libusb_endpoint_descriptor)); 300 323 ifp->endpoint = endpoint; 301 324 if (!endpoint) { … … 304 327 } 305 328 306 memset(endpoint, 0, tmp);307 329 for (i = 0; i < ifp->bNumEndpoints; i++) { 308 usbi_parse_descriptor(buffer, "bb", &header, 0);309 310 if (header.bLength > size) {311 usbi_err(ctx, "ran out of descriptors parsing");312 r = LIBUSB_ERROR_IO;313 goto err;314 }315 316 330 r = parse_endpoint(ctx, endpoint + i, buffer, size, 317 331 host_endian); 318 332 if (r < 0) 319 333 goto err; 334 if (r == 0) { 335 ifp->bNumEndpoints = (uint8_t)i; 336 break;; 337 } 320 338 321 339 buffer += r; … … 329 347 if (size < LIBUSB_DT_INTERFACE_SIZE || 330 348 ifp->bDescriptorType != LIBUSB_DT_INTERFACE || 331 !ifp->bAlternateSetting)349 ifp->bInterfaceNumber != interface_number) 332 350 return parsed; 333 351 } … … 341 359 static void clear_configuration(struct libusb_config_descriptor *config) 342 360 { 361 int i; 343 362 if (config->interface) { 344 int i;345 363 for (i = 0; i < config->bNumInterfaces; i++) 346 364 clear_interface((struct libusb_interface *) 347 config->interface + i); 348 free((void *) config->interface); 349 } 350 if (config->extra) 351 free((void *) config->extra); 365 config->interface + i); 366 } 367 free((void *) config->interface); 368 free((void *) config->extra); 352 369 } 353 370 354 371 static int parse_configuration(struct libusb_context *ctx, 355 372 struct libusb_config_descriptor *config, unsigned char *buffer, 356 int host_endian)373 int size, int host_endian) 357 374 { 358 375 int i; 359 376 int r; 360 int size;361 size_t tmp;362 377 struct usb_descriptor_header header; 363 378 struct libusb_interface *usb_interface; 364 379 380 if (size < LIBUSB_DT_CONFIG_SIZE) { 381 usbi_err(ctx, "short config descriptor read %d/%d", 382 size, LIBUSB_DT_CONFIG_SIZE); 383 return LIBUSB_ERROR_IO; 384 } 385 365 386 usbi_parse_descriptor(buffer, "bbwbbbbb", config, host_endian); 366 size = config->wTotalLength; 367 387 if (config->bDescriptorType != LIBUSB_DT_CONFIG) { 388 usbi_err(ctx, "unexpected descriptor %x (expected %x)", 389 config->bDescriptorType, LIBUSB_DT_CONFIG); 390 return LIBUSB_ERROR_IO; 391 } 392 if (config->bLength < LIBUSB_DT_CONFIG_SIZE) { 393 usbi_err(ctx, "invalid config bLength (%d)", config->bLength); 394 return LIBUSB_ERROR_IO; 395 } 396 if (config->bLength > size) { 397 usbi_err(ctx, "short config descriptor read %d/%d", 398 size, config->bLength); 399 return LIBUSB_ERROR_IO; 400 } 368 401 if (config->bNumInterfaces > USB_MAXINTERFACES) { 369 402 usbi_err(ctx, "too many interfaces (%d)", config->bNumInterfaces); … … 371 404 } 372 405 373 tmp = config->bNumInterfaces * sizeof(struct libusb_interface); 374 usb_interface = malloc(tmp); 406 usb_interface = calloc(config->bNumInterfaces, sizeof(struct libusb_interface)); 375 407 config->interface = usb_interface; 376 if (! config->interface)408 if (!usb_interface) 377 409 return LIBUSB_ERROR_NO_MEM; 378 410 379 memset(usb_interface, 0, tmp);380 411 buffer += config->bLength; 381 412 size -= config->bLength; … … 394 425 usbi_parse_descriptor(buffer, "bb", &header, 0); 395 426 396 if ( (header.bLength > size) ||397 (header.bLength < DESC_HEADER_LENGTH)) {398 usbi_err(ctx, "invalid descriptor length of %d",399 header.bLength);427 if (header.bLength < DESC_HEADER_LENGTH) { 428 usbi_err(ctx, 429 "invalid extra config desc len (%d)", 430 header.bLength); 400 431 r = LIBUSB_ERROR_IO; 401 432 goto err; 433 } else if (header.bLength > size) { 434 usbi_warn(ctx, 435 "short extra config desc read %d/%d", 436 size, header.bLength); 437 config->bNumInterfaces = (uint8_t)i; 438 return size; 402 439 } 403 440 404 441 /* If we find another "proper" descriptor then we're done */ 405 442 if ((header.bDescriptorType == LIBUSB_DT_ENDPOINT) || 406 (header.bDescriptorType == LIBUSB_DT_INTERFACE) || 407 (header.bDescriptorType == LIBUSB_DT_CONFIG) || 408 (header.bDescriptorType == LIBUSB_DT_DEVICE) || 409 (header.bDescriptorType == 410 LIBUSB_DT_SS_ENDPOINT_COMPANION)) 443 (header.bDescriptorType == LIBUSB_DT_INTERFACE) || 444 (header.bDescriptorType == LIBUSB_DT_CONFIG) || 445 (header.bDescriptorType == LIBUSB_DT_DEVICE)) 411 446 break; 412 447 413 usbi_dbg("skipping descriptor 0x%x \n", header.bDescriptorType);448 usbi_dbg("skipping descriptor 0x%x", header.bDescriptorType); 414 449 buffer += header.bLength; 415 450 size -= header.bLength; … … 436 471 if (r < 0) 437 472 goto err; 473 if (r == 0) { 474 config->bNumInterfaces = (uint8_t)i; 475 break; 476 } 438 477 439 478 buffer += r; … … 448 487 } 449 488 489 static int raw_desc_to_config(struct libusb_context *ctx, 490 unsigned char *buf, int size, int host_endian, 491 struct libusb_config_descriptor **config) 492 { 493 struct libusb_config_descriptor *_config = malloc(sizeof(*_config)); 494 int r; 495 496 if (!_config) 497 return LIBUSB_ERROR_NO_MEM; 498 499 r = parse_configuration(ctx, _config, buf, size, host_endian); 500 if (r < 0) { 501 usbi_err(ctx, "parse_configuration failed with error %d", r); 502 free(_config); 503 return r; 504 } else if (r > 0) { 505 usbi_warn(ctx, "still %d bytes of descriptor data left", r); 506 } 507 508 *config = _config; 509 return LIBUSB_SUCCESS; 510 } 511 450 512 int usbi_device_cache_descriptor(libusb_device *dev) 451 513 { 452 int r, host_endian ;514 int r, host_endian = 0; 453 515 454 516 r = usbi_backend->get_device_descriptor(dev, (unsigned char *) &dev->device_descriptor, … … 467 529 } 468 530 469 /** \ingroup desc531 /** \ingroup libusb_desc 470 532 * Get the USB device descriptor for a given device. 471 533 * 472 534 * This is a non-blocking function; the device descriptor is cached in memory. 535 * 536 * Note since libusb-1.0.16, \ref LIBUSB_API_VERSION >= 0x01000102, this 537 * function always succeeds. 473 538 * 474 539 * \param dev the device … … 485 550 } 486 551 487 /** \ingroup desc552 /** \ingroup libusb_desc 488 553 * Get the USB configuration descriptor for the currently active configuration. 489 554 * This is a non-blocking function which does not involve any requests being … … 502 567 struct libusb_config_descriptor **config) 503 568 { 504 struct libusb_config_descriptor *_config = malloc(sizeof(*_config));505 unsigned char tmp[ 8];569 struct libusb_config_descriptor _config; 570 unsigned char tmp[LIBUSB_DT_CONFIG_SIZE]; 506 571 unsigned char *buf = NULL; 507 572 int host_endian = 0; 508 573 int r; 509 574 510 usbi_dbg(""); 511 if (!_config) 575 r = usbi_backend->get_active_config_descriptor(dev, tmp, 576 LIBUSB_DT_CONFIG_SIZE, &host_endian); 577 if (r < 0) 578 return r; 579 if (r < LIBUSB_DT_CONFIG_SIZE) { 580 usbi_err(dev->ctx, "short config descriptor read %d/%d", 581 r, LIBUSB_DT_CONFIG_SIZE); 582 return LIBUSB_ERROR_IO; 583 } 584 585 usbi_parse_descriptor(tmp, "bbw", &_config, host_endian); 586 buf = malloc(_config.wTotalLength); 587 if (!buf) 512 588 return LIBUSB_ERROR_NO_MEM; 513 589 514 r = usbi_backend->get_active_config_descriptor(dev, tmp, sizeof(tmp),515 &host_endian);516 if (r < 0)517 goto err;518 519 usbi_parse_descriptor(tmp, "bbw", _config, host_endian);520 buf = malloc(_config->wTotalLength);521 if (!buf) {522 r = LIBUSB_ERROR_NO_MEM;523 goto err;524 }525 526 590 r = usbi_backend->get_active_config_descriptor(dev, buf, 527 _config->wTotalLength, &host_endian); 528 if (r < 0) 529 goto err; 530 531 r = parse_configuration(dev->ctx, _config, buf, host_endian); 532 if (r < 0) { 533 usbi_err(dev->ctx, "parse_configuration failed with error %d", r); 534 goto err; 535 } else if (r > 0) { 536 usbi_warn(dev->ctx, "descriptor data still left"); 537 } 591 _config.wTotalLength, &host_endian); 592 if (r >= 0) 593 r = raw_desc_to_config(dev->ctx, buf, r, host_endian, config); 538 594 539 595 free(buf); 540 *config = _config;541 return 0;542 543 err:544 free(_config);545 if (buf)546 free(buf);547 596 return r; 548 597 } 549 598 550 /** \ingroup desc599 /** \ingroup libusb_desc 551 600 * Get a USB configuration descriptor based on its index. 552 601 * This is a non-blocking function which does not involve any requests being … … 567 616 uint8_t config_index, struct libusb_config_descriptor **config) 568 617 { 569 struct libusb_config_descriptor *_config;570 unsigned char tmp[ 8];618 struct libusb_config_descriptor _config; 619 unsigned char tmp[LIBUSB_DT_CONFIG_SIZE]; 571 620 unsigned char *buf = NULL; 572 621 int host_endian = 0; … … 577 626 return LIBUSB_ERROR_NOT_FOUND; 578 627 579 _config = malloc(sizeof(*_config)); 580 if (!_config) 628 r = usbi_backend->get_config_descriptor(dev, config_index, tmp, 629 LIBUSB_DT_CONFIG_SIZE, &host_endian); 630 if (r < 0) 631 return r; 632 if (r < LIBUSB_DT_CONFIG_SIZE) { 633 usbi_err(dev->ctx, "short config descriptor read %d/%d", 634 r, LIBUSB_DT_CONFIG_SIZE); 635 return LIBUSB_ERROR_IO; 636 } 637 638 usbi_parse_descriptor(tmp, "bbw", &_config, host_endian); 639 buf = malloc(_config.wTotalLength); 640 if (!buf) 581 641 return LIBUSB_ERROR_NO_MEM; 582 642 583 r = usbi_backend->get_config_descriptor(dev, config_index, tmp,584 sizeof(tmp), &host_endian);585 if (r < 0)586 goto err;587 588 usbi_parse_descriptor(tmp, "bbw", _config, host_endian);589 buf = malloc(_config->wTotalLength);590 if (!buf) {591 r = LIBUSB_ERROR_NO_MEM;592 goto err;593 }594 595 host_endian = 0;596 643 r = usbi_backend->get_config_descriptor(dev, config_index, buf, 597 _config->wTotalLength, &host_endian); 598 if (r < 0) 599 goto err; 600 601 r = parse_configuration(dev->ctx, _config, buf, host_endian); 602 if (r < 0) { 603 usbi_err(dev->ctx, "parse_configuration failed with error %d", r); 604 goto err; 605 } else if (r > 0) { 606 usbi_warn(dev->ctx, "descriptor data still left"); 607 } 644 _config.wTotalLength, &host_endian); 645 if (r >= 0) 646 r = raw_desc_to_config(dev->ctx, buf, r, host_endian, config); 608 647 609 648 free(buf); 610 *config = _config;611 return 0;612 613 err:614 free(_config);615 if (buf)616 free(buf);617 649 return r; 618 650 } … … 621 653 * matching a specific bConfigurationValue in the idx output parameter, or -1 622 654 * if the config was not found. 623 * returns 0 o r a LIBUSB_ERROR code655 * returns 0 on success or a LIBUSB_ERROR code 624 656 */ 625 657 int usbi_get_config_index_by_value(struct libusb_device *dev, … … 634 666 int r = usbi_backend->get_config_descriptor(dev, i, tmp, sizeof(tmp), 635 667 &host_endian); 636 if (r < 0) 668 if (r < 0) { 669 *idx = -1; 637 670 return r; 671 } 638 672 if (tmp[5] == bConfigurationValue) { 639 673 *idx = i; … … 646 680 } 647 681 648 /** \ingroup desc682 /** \ingroup libusb_desc 649 683 * Get a USB configuration descriptor with a specific bConfigurationValue. 650 684 * This is a non-blocking function which does not involve any requests being … … 666 700 uint8_t bConfigurationValue, struct libusb_config_descriptor **config) 667 701 { 668 int idx; 669 int r = usbi_get_config_index_by_value(dev, bConfigurationValue, &idx); 702 int r, idx, host_endian; 703 unsigned char *buf = NULL; 704 705 if (usbi_backend->get_config_descriptor_by_value) { 706 r = usbi_backend->get_config_descriptor_by_value(dev, 707 bConfigurationValue, &buf, &host_endian); 708 if (r < 0) 709 return r; 710 return raw_desc_to_config(dev->ctx, buf, r, host_endian, config); 711 } 712 713 r = usbi_get_config_index_by_value(dev, bConfigurationValue, &idx); 670 714 if (r < 0) 671 715 return r; … … 676 720 } 677 721 678 /** \ingroup desc722 /** \ingroup libusb_desc 679 723 * Free a configuration descriptor obtained from 680 724 * libusb_get_active_config_descriptor() or libusb_get_config_descriptor(). … … 694 738 } 695 739 696 /** \ingroup desc 740 /** \ingroup libusb_desc 741 * Get an endpoints superspeed endpoint companion descriptor (if any) 742 * 743 * \param ctx the context to operate on, or NULL for the default context 744 * \param endpoint endpoint descriptor from which to get the superspeed 745 * endpoint companion descriptor 746 * \param ep_comp output location for the superspeed endpoint companion 747 * descriptor. Only valid if 0 was returned. Must be freed with 748 * libusb_free_ss_endpoint_companion_descriptor() after use. 749 * \returns 0 on success 750 * \returns LIBUSB_ERROR_NOT_FOUND if the configuration does not exist 751 * \returns another LIBUSB_ERROR code on error 752 */ 753 int API_EXPORTED libusb_get_ss_endpoint_companion_descriptor( 754 struct libusb_context *ctx, 755 const struct libusb_endpoint_descriptor *endpoint, 756 struct libusb_ss_endpoint_companion_descriptor **ep_comp) 757 { 758 struct usb_descriptor_header header; 759 int size = endpoint->extra_length; 760 const unsigned char *buffer = endpoint->extra; 761 762 *ep_comp = NULL; 763 764 while (size >= DESC_HEADER_LENGTH) { 765 usbi_parse_descriptor(buffer, "bb", &header, 0); 766 if (header.bLength < 2 || header.bLength > size) { 767 usbi_err(ctx, "invalid descriptor length %d", 768 header.bLength); 769 return LIBUSB_ERROR_IO; 770 } 771 if (header.bDescriptorType != LIBUSB_DT_SS_ENDPOINT_COMPANION) { 772 buffer += header.bLength; 773 size -= header.bLength; 774 continue; 775 } 776 if (header.bLength < LIBUSB_DT_SS_ENDPOINT_COMPANION_SIZE) { 777 usbi_err(ctx, "invalid ss-ep-comp-desc length %d", 778 header.bLength); 779 return LIBUSB_ERROR_IO; 780 } 781 *ep_comp = malloc(sizeof(**ep_comp)); 782 if (*ep_comp == NULL) 783 return LIBUSB_ERROR_NO_MEM; 784 usbi_parse_descriptor(buffer, "bbbbw", *ep_comp, 0); 785 return LIBUSB_SUCCESS; 786 } 787 return LIBUSB_ERROR_NOT_FOUND; 788 } 789 790 /** \ingroup libusb_desc 791 * Free a superspeed endpoint companion descriptor obtained from 792 * libusb_get_ss_endpoint_companion_descriptor(). 793 * It is safe to call this function with a NULL ep_comp parameter, in which 794 * case the function simply returns. 795 * 796 * \param ep_comp the superspeed endpoint companion descriptor to free 797 */ 798 void API_EXPORTED libusb_free_ss_endpoint_companion_descriptor( 799 struct libusb_ss_endpoint_companion_descriptor *ep_comp) 800 { 801 free(ep_comp); 802 } 803 804 static int parse_bos(struct libusb_context *ctx, 805 struct libusb_bos_descriptor **bos, 806 unsigned char *buffer, int size, int host_endian) 807 { 808 struct libusb_bos_descriptor bos_header, *_bos; 809 struct libusb_bos_dev_capability_descriptor dev_cap; 810 int i; 811 812 if (size < LIBUSB_DT_BOS_SIZE) { 813 usbi_err(ctx, "short bos descriptor read %d/%d", 814 size, LIBUSB_DT_BOS_SIZE); 815 return LIBUSB_ERROR_IO; 816 } 817 818 usbi_parse_descriptor(buffer, "bbwb", &bos_header, host_endian); 819 if (bos_header.bDescriptorType != LIBUSB_DT_BOS) { 820 usbi_err(ctx, "unexpected descriptor %x (expected %x)", 821 bos_header.bDescriptorType, LIBUSB_DT_BOS); 822 return LIBUSB_ERROR_IO; 823 } 824 if (bos_header.bLength < LIBUSB_DT_BOS_SIZE) { 825 usbi_err(ctx, "invalid bos bLength (%d)", bos_header.bLength); 826 return LIBUSB_ERROR_IO; 827 } 828 if (bos_header.bLength > size) { 829 usbi_err(ctx, "short bos descriptor read %d/%d", 830 size, bos_header.bLength); 831 return LIBUSB_ERROR_IO; 832 } 833 834 _bos = calloc (1, 835 sizeof(*_bos) + bos_header.bNumDeviceCaps * sizeof(void *)); 836 if (!_bos) 837 return LIBUSB_ERROR_NO_MEM; 838 839 usbi_parse_descriptor(buffer, "bbwb", _bos, host_endian); 840 buffer += bos_header.bLength; 841 size -= bos_header.bLength; 842 843 /* Get the device capability descriptors */ 844 for (i = 0; i < bos_header.bNumDeviceCaps; i++) { 845 if (size < LIBUSB_DT_DEVICE_CAPABILITY_SIZE) { 846 usbi_warn(ctx, "short dev-cap descriptor read %d/%d", 847 size, LIBUSB_DT_DEVICE_CAPABILITY_SIZE); 848 break; 849 } 850 usbi_parse_descriptor(buffer, "bbb", &dev_cap, host_endian); 851 if (dev_cap.bDescriptorType != LIBUSB_DT_DEVICE_CAPABILITY) { 852 usbi_warn(ctx, "unexpected descriptor %x (expected %x)", 853 dev_cap.bDescriptorType, LIBUSB_DT_DEVICE_CAPABILITY); 854 break; 855 } 856 if (dev_cap.bLength < LIBUSB_DT_DEVICE_CAPABILITY_SIZE) { 857 usbi_err(ctx, "invalid dev-cap bLength (%d)", 858 dev_cap.bLength); 859 libusb_free_bos_descriptor(_bos); 860 return LIBUSB_ERROR_IO; 861 } 862 if (dev_cap.bLength > size) { 863 usbi_warn(ctx, "short dev-cap descriptor read %d/%d", 864 size, dev_cap.bLength); 865 break; 866 } 867 868 _bos->dev_capability[i] = malloc(dev_cap.bLength); 869 if (!_bos->dev_capability[i]) { 870 libusb_free_bos_descriptor(_bos); 871 return LIBUSB_ERROR_NO_MEM; 872 } 873 memcpy(_bos->dev_capability[i], buffer, dev_cap.bLength); 874 buffer += dev_cap.bLength; 875 size -= dev_cap.bLength; 876 } 877 _bos->bNumDeviceCaps = (uint8_t)i; 878 *bos = _bos; 879 880 return LIBUSB_SUCCESS; 881 } 882 883 /** \ingroup libusb_desc 884 * Get a Binary Object Store (BOS) descriptor 885 * This is a BLOCKING function, which will send requests to the device. 886 * 887 * \param dev_handle the handle of an open libusb device 888 * \param bos output location for the BOS descriptor. Only valid if 0 was returned. 889 * Must be freed with \ref libusb_free_bos_descriptor() after use. 890 * \returns 0 on success 891 * \returns LIBUSB_ERROR_NOT_FOUND if the device doesn't have a BOS descriptor 892 * \returns another LIBUSB_ERROR code on error 893 */ 894 int API_EXPORTED libusb_get_bos_descriptor(libusb_device_handle *dev_handle, 895 struct libusb_bos_descriptor **bos) 896 { 897 struct libusb_bos_descriptor _bos; 898 uint8_t bos_header[LIBUSB_DT_BOS_SIZE] = {0}; 899 unsigned char *bos_data = NULL; 900 const int host_endian = 0; 901 int r; 902 903 /* Read the BOS. This generates 2 requests on the bus, 904 * one for the header, and one for the full BOS */ 905 r = libusb_get_descriptor(dev_handle, LIBUSB_DT_BOS, 0, bos_header, 906 LIBUSB_DT_BOS_SIZE); 907 if (r < 0) { 908 if (r != LIBUSB_ERROR_PIPE) 909 usbi_err(HANDLE_CTX(dev_handle), "failed to read BOS (%d)", r); 910 return r; 911 } 912 if (r < LIBUSB_DT_BOS_SIZE) { 913 usbi_err(HANDLE_CTX(dev_handle), "short BOS read %d/%d", 914 r, LIBUSB_DT_BOS_SIZE); 915 return LIBUSB_ERROR_IO; 916 } 917 918 usbi_parse_descriptor(bos_header, "bbwb", &_bos, host_endian); 919 usbi_dbg("found BOS descriptor: size %d bytes, %d capabilities", 920 _bos.wTotalLength, _bos.bNumDeviceCaps); 921 bos_data = calloc(_bos.wTotalLength, 1); 922 if (bos_data == NULL) 923 return LIBUSB_ERROR_NO_MEM; 924 925 r = libusb_get_descriptor(dev_handle, LIBUSB_DT_BOS, 0, bos_data, 926 _bos.wTotalLength); 927 if (r >= 0) 928 r = parse_bos(HANDLE_CTX(dev_handle), bos, bos_data, r, host_endian); 929 else 930 usbi_err(HANDLE_CTX(dev_handle), "failed to read BOS (%d)", r); 931 932 free(bos_data); 933 return r; 934 } 935 936 /** \ingroup libusb_desc 937 * Free a BOS descriptor obtained from libusb_get_bos_descriptor(). 938 * It is safe to call this function with a NULL bos parameter, in which 939 * case the function simply returns. 940 * 941 * \param bos the BOS descriptor to free 942 */ 943 void API_EXPORTED libusb_free_bos_descriptor(struct libusb_bos_descriptor *bos) 944 { 945 int i; 946 947 if (!bos) 948 return; 949 950 for (i = 0; i < bos->bNumDeviceCaps; i++) 951 free(bos->dev_capability[i]); 952 free(bos); 953 } 954 955 /** \ingroup libusb_desc 956 * Get an USB 2.0 Extension descriptor 957 * 958 * \param ctx the context to operate on, or NULL for the default context 959 * \param dev_cap Device Capability descriptor with a bDevCapabilityType of 960 * \ref libusb_capability_type::LIBUSB_BT_USB_2_0_EXTENSION 961 * LIBUSB_BT_USB_2_0_EXTENSION 962 * \param usb_2_0_extension output location for the USB 2.0 Extension 963 * descriptor. Only valid if 0 was returned. Must be freed with 964 * libusb_free_usb_2_0_extension_descriptor() after use. 965 * \returns 0 on success 966 * \returns a LIBUSB_ERROR code on error 967 */ 968 int API_EXPORTED libusb_get_usb_2_0_extension_descriptor( 969 struct libusb_context *ctx, 970 struct libusb_bos_dev_capability_descriptor *dev_cap, 971 struct libusb_usb_2_0_extension_descriptor **usb_2_0_extension) 972 { 973 struct libusb_usb_2_0_extension_descriptor *_usb_2_0_extension; 974 const int host_endian = 0; 975 976 if (dev_cap->bDevCapabilityType != LIBUSB_BT_USB_2_0_EXTENSION) { 977 usbi_err(ctx, "unexpected bDevCapabilityType %x (expected %x)", 978 dev_cap->bDevCapabilityType, 979 LIBUSB_BT_USB_2_0_EXTENSION); 980 return LIBUSB_ERROR_INVALID_PARAM; 981 } 982 if (dev_cap->bLength < LIBUSB_BT_USB_2_0_EXTENSION_SIZE) { 983 usbi_err(ctx, "short dev-cap descriptor read %d/%d", 984 dev_cap->bLength, LIBUSB_BT_USB_2_0_EXTENSION_SIZE); 985 return LIBUSB_ERROR_IO; 986 } 987 988 _usb_2_0_extension = malloc(sizeof(*_usb_2_0_extension)); 989 if (!_usb_2_0_extension) 990 return LIBUSB_ERROR_NO_MEM; 991 992 usbi_parse_descriptor((unsigned char *)dev_cap, "bbbd", 993 _usb_2_0_extension, host_endian); 994 995 *usb_2_0_extension = _usb_2_0_extension; 996 return LIBUSB_SUCCESS; 997 } 998 999 /** \ingroup libusb_desc 1000 * Free a USB 2.0 Extension descriptor obtained from 1001 * libusb_get_usb_2_0_extension_descriptor(). 1002 * It is safe to call this function with a NULL usb_2_0_extension parameter, 1003 * in which case the function simply returns. 1004 * 1005 * \param usb_2_0_extension the USB 2.0 Extension descriptor to free 1006 */ 1007 void API_EXPORTED libusb_free_usb_2_0_extension_descriptor( 1008 struct libusb_usb_2_0_extension_descriptor *usb_2_0_extension) 1009 { 1010 free(usb_2_0_extension); 1011 } 1012 1013 /** \ingroup libusb_desc 1014 * Get a SuperSpeed USB Device Capability descriptor 1015 * 1016 * \param ctx the context to operate on, or NULL for the default context 1017 * \param dev_cap Device Capability descriptor with a bDevCapabilityType of 1018 * \ref libusb_capability_type::LIBUSB_BT_SS_USB_DEVICE_CAPABILITY 1019 * LIBUSB_BT_SS_USB_DEVICE_CAPABILITY 1020 * \param ss_usb_device_cap output location for the SuperSpeed USB Device 1021 * Capability descriptor. Only valid if 0 was returned. Must be freed with 1022 * libusb_free_ss_usb_device_capability_descriptor() after use. 1023 * \returns 0 on success 1024 * \returns a LIBUSB_ERROR code on error 1025 */ 1026 int API_EXPORTED libusb_get_ss_usb_device_capability_descriptor( 1027 struct libusb_context *ctx, 1028 struct libusb_bos_dev_capability_descriptor *dev_cap, 1029 struct libusb_ss_usb_device_capability_descriptor **ss_usb_device_cap) 1030 { 1031 struct libusb_ss_usb_device_capability_descriptor *_ss_usb_device_cap; 1032 const int host_endian = 0; 1033 1034 if (dev_cap->bDevCapabilityType != LIBUSB_BT_SS_USB_DEVICE_CAPABILITY) { 1035 usbi_err(ctx, "unexpected bDevCapabilityType %x (expected %x)", 1036 dev_cap->bDevCapabilityType, 1037 LIBUSB_BT_SS_USB_DEVICE_CAPABILITY); 1038 return LIBUSB_ERROR_INVALID_PARAM; 1039 } 1040 if (dev_cap->bLength < LIBUSB_BT_SS_USB_DEVICE_CAPABILITY_SIZE) { 1041 usbi_err(ctx, "short dev-cap descriptor read %d/%d", 1042 dev_cap->bLength, LIBUSB_BT_SS_USB_DEVICE_CAPABILITY_SIZE); 1043 return LIBUSB_ERROR_IO; 1044 } 1045 1046 _ss_usb_device_cap = malloc(sizeof(*_ss_usb_device_cap)); 1047 if (!_ss_usb_device_cap) 1048 return LIBUSB_ERROR_NO_MEM; 1049 1050 usbi_parse_descriptor((unsigned char *)dev_cap, "bbbbwbbw", 1051 _ss_usb_device_cap, host_endian); 1052 1053 *ss_usb_device_cap = _ss_usb_device_cap; 1054 return LIBUSB_SUCCESS; 1055 } 1056 1057 /** \ingroup libusb_desc 1058 * Free a SuperSpeed USB Device Capability descriptor obtained from 1059 * libusb_get_ss_usb_device_capability_descriptor(). 1060 * It is safe to call this function with a NULL ss_usb_device_cap 1061 * parameter, in which case the function simply returns. 1062 * 1063 * \param ss_usb_device_cap the USB 2.0 Extension descriptor to free 1064 */ 1065 void API_EXPORTED libusb_free_ss_usb_device_capability_descriptor( 1066 struct libusb_ss_usb_device_capability_descriptor *ss_usb_device_cap) 1067 { 1068 free(ss_usb_device_cap); 1069 } 1070 1071 /** \ingroup libusb_desc 1072 * Get a Container ID descriptor 1073 * 1074 * \param ctx the context to operate on, or NULL for the default context 1075 * \param dev_cap Device Capability descriptor with a bDevCapabilityType of 1076 * \ref libusb_capability_type::LIBUSB_BT_CONTAINER_ID 1077 * LIBUSB_BT_CONTAINER_ID 1078 * \param container_id output location for the Container ID descriptor. 1079 * Only valid if 0 was returned. Must be freed with 1080 * libusb_free_container_id_descriptor() after use. 1081 * \returns 0 on success 1082 * \returns a LIBUSB_ERROR code on error 1083 */ 1084 int API_EXPORTED libusb_get_container_id_descriptor(struct libusb_context *ctx, 1085 struct libusb_bos_dev_capability_descriptor *dev_cap, 1086 struct libusb_container_id_descriptor **container_id) 1087 { 1088 struct libusb_container_id_descriptor *_container_id; 1089 const int host_endian = 0; 1090 1091 if (dev_cap->bDevCapabilityType != LIBUSB_BT_CONTAINER_ID) { 1092 usbi_err(ctx, "unexpected bDevCapabilityType %x (expected %x)", 1093 dev_cap->bDevCapabilityType, 1094 LIBUSB_BT_CONTAINER_ID); 1095 return LIBUSB_ERROR_INVALID_PARAM; 1096 } 1097 if (dev_cap->bLength < LIBUSB_BT_CONTAINER_ID_SIZE) { 1098 usbi_err(ctx, "short dev-cap descriptor read %d/%d", 1099 dev_cap->bLength, LIBUSB_BT_CONTAINER_ID_SIZE); 1100 return LIBUSB_ERROR_IO; 1101 } 1102 1103 _container_id = malloc(sizeof(*_container_id)); 1104 if (!_container_id) 1105 return LIBUSB_ERROR_NO_MEM; 1106 1107 usbi_parse_descriptor((unsigned char *)dev_cap, "bbbbu", 1108 _container_id, host_endian); 1109 1110 *container_id = _container_id; 1111 return LIBUSB_SUCCESS; 1112 } 1113 1114 /** \ingroup libusb_desc 1115 * Free a Container ID descriptor obtained from 1116 * libusb_get_container_id_descriptor(). 1117 * It is safe to call this function with a NULL container_id parameter, 1118 * in which case the function simply returns. 1119 * 1120 * \param container_id the USB 2.0 Extension descriptor to free 1121 */ 1122 void API_EXPORTED libusb_free_container_id_descriptor( 1123 struct libusb_container_id_descriptor *container_id) 1124 { 1125 free(container_id); 1126 } 1127 1128 /** \ingroup libusb_desc 697 1129 * Retrieve a string descriptor in C style ASCII. 698 1130 * … … 700 1132 * supported by the device. 701 1133 * 702 * \param dev a device handle1134 * \param dev_handle a device handle 703 1135 * \param desc_index the index of the descriptor to retrieve 704 1136 * \param data output buffer for ASCII string descriptor … … 706 1138 * \returns number of bytes returned in data, or LIBUSB_ERROR code on failure 707 1139 */ 708 int API_EXPORTED libusb_get_string_descriptor_ascii(libusb_device_handle *dev ,1140 int API_EXPORTED libusb_get_string_descriptor_ascii(libusb_device_handle *dev_handle, 709 1141 uint8_t desc_index, unsigned char *data, int length) 710 1142 { … … 725 1157 return LIBUSB_ERROR_INVALID_PARAM; 726 1158 727 r = libusb_get_string_descriptor(dev , 0, 0, tbuf, sizeof(tbuf));1159 r = libusb_get_string_descriptor(dev_handle, 0, 0, tbuf, sizeof(tbuf)); 728 1160 if (r < 0) 729 1161 return r; … … 734 1166 langid = tbuf[2] | (tbuf[3] << 8); 735 1167 736 r = libusb_get_string_descriptor(dev , desc_index, langid, tbuf,1168 r = libusb_get_string_descriptor(dev_handle, desc_index, langid, tbuf, 737 1169 sizeof(tbuf)); 738 1170 if (r < 0) … … 749 1181 break; 750 1182 751 if ( tbuf[si + 1]) /* high byte*/1183 if ((tbuf[si] & 0x80) || (tbuf[si + 1])) /* non-ASCII */ 752 1184 data[di++] = '?'; 753 1185 else … … 758 1190 return di; 759 1191 } 760 761 int API_EXPORTED libusb_parse_ss_endpoint_comp(const void *buf, int len,762 struct libusb_ss_endpoint_companion_descriptor **ep_comp)763 {764 struct libusb_ss_endpoint_companion_descriptor *ep_comp_desc;765 struct usb_descriptor_header header;766 767 usbi_parse_descriptor(buf, "bb", &header, 0);768 769 /* Everything should be fine being passed into here, but we sanity */770 /* check JIC */771 if (header.bLength > len) {772 usbi_err(NULL, "ran out of descriptors parsing");773 return LIBUSB_ERROR_NO_MEM;774 }775 776 if (header.bDescriptorType != LIBUSB_DT_SS_ENDPOINT_COMPANION) {777 usbi_err(NULL, "unexpected descriptor %x (expected %x)",778 header.bDescriptorType, LIBUSB_DT_SS_ENDPOINT_COMPANION);779 return LIBUSB_ERROR_INVALID_PARAM;780 }781 782 ep_comp_desc = calloc(1, sizeof (*ep_comp_desc));783 if (!ep_comp_desc) {784 return LIBUSB_ERROR_NO_MEM;785 }786 787 if (header.bLength >= LIBUSB_DT_SS_ENDPOINT_COMPANION_SIZE)788 usbi_parse_descriptor(buf, "bbbbw", ep_comp_desc, 0);789 790 *ep_comp = ep_comp_desc;791 792 return LIBUSB_SUCCESS;793 }794 795 void API_EXPORTED libusb_free_ss_endpoint_comp(struct libusb_ss_endpoint_companion_descriptor *ep_comp)796 {797 assert(ep_comp);798 free(ep_comp);799 }800 801 int API_EXPORTED libusb_parse_bos_descriptor(const void *buf, int len,802 struct libusb_bos_descriptor **bos)803 {804 const unsigned char *buffer = (const unsigned char *) buf;805 struct libusb_bos_descriptor *bos_desc;806 int i;807 808 bos_desc = calloc (1, sizeof (*bos_desc));809 if (!bos_desc) {810 return LIBUSB_ERROR_NO_MEM;811 }812 813 usbi_parse_descriptor(buffer, "bbwb", bos_desc, 0);814 buffer += LIBUSB_DT_BOS_SIZE;815 816 /* Get the device capability descriptors */817 for (i = 0; i < bos_desc->bNumDeviceCaps; ++i) {818 if (buffer[2] == LIBUSB_USB_CAP_TYPE_EXT) {819 if (!bos_desc->usb_2_0_ext_cap) {820 bos_desc->usb_2_0_ext_cap =821 (struct libusb_usb_2_0_device_capability_descriptor *)822 malloc(sizeof(*bos_desc->usb_2_0_ext_cap));823 usbi_parse_descriptor(buffer, "bbbd",824 bos_desc->usb_2_0_ext_cap, 0);825 } else826 usbi_warn(NULL,827 "usb_2_0_ext_cap was already allocated");828 829 /* move to the next device capability descriptor */830 buffer += LIBUSB_USB_2_0_EXTENSION_DEVICE_CAPABILITY_SIZE;831 } else if (buffer[2] == LIBUSB_SS_USB_CAP_TYPE) {832 if (!bos_desc->ss_usb_cap) {833 bos_desc->ss_usb_cap =834 (struct libusb_ss_usb_device_capability_descriptor *)835 malloc(sizeof(*bos_desc->ss_usb_cap));836 usbi_parse_descriptor(buffer, "bbbbwbbw",837 bos_desc->ss_usb_cap, 0);838 } else839 usbi_warn(NULL,840 "ss_usb_cap was already allocated");841 842 /* move to the next device capability descriptor */843 buffer += LIBUSB_SS_USB_DEVICE_CAPABILITY_SIZE;844 } else {845 usbi_info(NULL, "wireless/container_id capability "846 "descriptor");847 848 /* move to the next device capability descriptor */849 buffer += buffer[0];850 }851 }852 853 *bos = bos_desc;854 855 return LIBUSB_SUCCESS;856 }857 858 void API_EXPORTED libusb_free_bos_descriptor(struct libusb_bos_descriptor *bos)859 {860 assert(bos);861 862 if (bos->usb_2_0_ext_cap) {863 free(bos->usb_2_0_ext_cap);864 }865 866 if (bos->ss_usb_cap) {867 free(bos->ss_usb_cap);868 }869 870 free(bos);871 }
Note:
See TracChangeset
for help on using the changeset viewer.