Ignore:
Timestamp:
Dec 12, 2016, 6:38:13 PM (8 years ago)
Author:
Silvan Scherrer
Message:

libusb: update trunk to version 1.0.21

Location:
libusb1/trunk
Files:
2 edited

Legend:

Unmodified
Added
Removed
  • libusb1/trunk

  • TabularUnified libusb1/trunk/libusb/descriptor.c

    r1876 r1882  
     1/* -*- Mode: C; indent-tabs-mode:t ; c-basic-offset:8 -*- */
    12/*
    23 * 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>
    66 *
    77 * This library is free software; you can redistribute it and/or
     
    2020 */
    2121
     22#include <config.h>
     23
    2224#include <errno.h>
    2325#include <stdint.h>
    2426#include <stdlib.h>
    2527#include <string.h>
    26 #include <stdio.h>
    27 #include <assert.h>
    2828
    2929#include "libusbi.h"
     
    3636#define ENDPOINT_AUDIO_DESC_LENGTH      9
    3737
    38 /** @defgroup desc USB descriptors
     38/** @defgroup libusb_desc USB descriptors
    3939 * This page details how to examine the various standard USB descriptors
    4040 * for detected devices
     
    6969                                dp += 2;
    7070                                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 */
    7573
    7674                                if (host_endian) {
     
    8482                                dp += 4;
    8583                                break;
     84                        case 'u':       /* 16 byte UUID */
     85                                memcpy(dp, sp, 16);
     86                                sp += 16;
     87                                dp += 16;
     88                                break;
    8689                }
    8790        }
     
    9295static void clear_endpoint(struct libusb_endpoint_descriptor *endpoint)
    9396{
    94         if (endpoint->extra)
    95                 free((unsigned char *) endpoint->extra);
     97        free((void *) endpoint->extra);
    9698}
    9799
     
    106108        int len;
    107109
     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
    108116        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 
    117117        if (header.bDescriptorType != LIBUSB_DT_ENDPOINT) {
    118118                usbi_err(ctx, "unexpected descriptor %x (expected %x)",
     
    120120                return parsed;
    121121        }
    122 
     122        if (header.bLength > size) {
     123                usbi_warn(ctx, "short endpoint descriptor read %d/%d",
     124                          size, header.bLength);
     125                return parsed;
     126        }
    123127        if (header.bLength >= ENDPOINT_AUDIO_DESC_LENGTH)
    124128                usbi_parse_descriptor(buffer, "bbbbwbbb", endpoint, host_endian);
    125129        else if (header.bLength >= ENDPOINT_DESC_LENGTH)
    126130                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        }
    127135
    128136        buffer += header.bLength;
     
    135143        while (size >= DESC_HEADER_LENGTH) {
    136144                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;
    141153                }
    142154
    143155                /* If we find another "proper" descriptor then we're done  */
    144156                if ((header.bDescriptorType == LIBUSB_DT_ENDPOINT) ||
    145                     (header.bDescriptorType == LIBUSB_DT_INTERFACE) ||
    146                     (header.bDescriptorType == LIBUSB_DT_CONFIG) ||
    147                     (header.bDescriptorType == LIBUSB_DT_DEVICE))
     157                                (header.bDescriptorType == LIBUSB_DT_INTERFACE) ||
     158                                (header.bDescriptorType == LIBUSB_DT_CONFIG) ||
     159                                (header.bDescriptorType == LIBUSB_DT_DEVICE))
    148160                        break;
    149161
     
    186198                                (struct libusb_interface_descriptor *)
    187199                                usb_interface->altsetting + i;
    188                         if (ifp->extra)
    189                                 free((void *) ifp->extra);
     200                        free((void *) ifp->extra);
    190201                        if (ifp->endpoint) {
    191202                                for (j = 0; j < ifp->bNumEndpoints; j++)
    192203                                        clear_endpoint((struct libusb_endpoint_descriptor *)
    193                                                 ifp->endpoint + j);
    194                                 free((void *) ifp->endpoint);
     204                                                       ifp->endpoint + j);
    195205                        }
    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;
    201211}
    202212
     
    209219        int r;
    210220        int parsed = 0;
    211         size_t tmp;
     221        int interface_number = -1;
    212222        struct usb_descriptor_header header;
    213223        struct libusb_interface_descriptor *ifp;
     
    219229                struct libusb_interface_descriptor *altsetting =
    220230                        (struct libusb_interface_descriptor *) usb_interface->altsetting;
    221                 altsetting = realloc(altsetting,
     231                altsetting = usbi_reallocf(altsetting,
    222232                        sizeof(struct libusb_interface_descriptor) *
    223233                        (usb_interface->num_altsetting + 1));
     
    229239
    230240                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
    231264                usb_interface->num_altsetting++;
    232                 usbi_parse_descriptor(buffer, "bbbbbbbbb", ifp, 0);
    233265                ifp->extra = NULL;
    234266                ifp->extra_length = 0;
    235267                ifp->endpoint = NULL;
     268
     269                if (interface_number == -1)
     270                        interface_number = ifp->bInterfaceNumber;
    236271
    237272                /* Skip over the interface */
     
    245280                while (size >= DESC_HEADER_LENGTH) {
    246281                        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);
    250286                                r = LIBUSB_ERROR_IO;
    251287                                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;
    252293                        }
    253294
    254295                        /* If we find another "proper" descriptor then we're done */
    255296                        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))
    261300                                break;
    262301
     
    279318                }
    280319
    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 
    296320                if (ifp->bNumEndpoints > 0) {
    297321                        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));
    300323                        ifp->endpoint = endpoint;
    301324                        if (!endpoint) {
     
    304327                        }
    305328
    306                         memset(endpoint, 0, tmp);
    307329                        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 
    316330                                r = parse_endpoint(ctx, endpoint + i, buffer, size,
    317331                                        host_endian);
    318332                                if (r < 0)
    319333                                        goto err;
     334                                if (r == 0) {
     335                                        ifp->bNumEndpoints = (uint8_t)i;
     336                                        break;;
     337                                }
    320338
    321339                                buffer += r;
     
    329347                if (size < LIBUSB_DT_INTERFACE_SIZE ||
    330348                                ifp->bDescriptorType != LIBUSB_DT_INTERFACE ||
    331                                 !ifp->bAlternateSetting)
     349                                ifp->bInterfaceNumber != interface_number)
    332350                        return parsed;
    333351        }
     
    341359static void clear_configuration(struct libusb_config_descriptor *config)
    342360{
     361        int i;
    343362        if (config->interface) {
    344                 int i;
    345363                for (i = 0; i < config->bNumInterfaces; i++)
    346364                        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);
    352369}
    353370
    354371static int parse_configuration(struct libusb_context *ctx,
    355372        struct libusb_config_descriptor *config, unsigned char *buffer,
    356         int host_endian)
     373        int size, int host_endian)
    357374{
    358375        int i;
    359376        int r;
    360         int size;
    361         size_t tmp;
    362377        struct usb_descriptor_header header;
    363378        struct libusb_interface *usb_interface;
    364379
     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
    365386        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        }
    368401        if (config->bNumInterfaces > USB_MAXINTERFACES) {
    369402                usbi_err(ctx, "too many interfaces (%d)", config->bNumInterfaces);
     
    371404        }
    372405
    373         tmp = config->bNumInterfaces * sizeof(struct libusb_interface);
    374         usb_interface = malloc(tmp);
     406        usb_interface = calloc(config->bNumInterfaces, sizeof(struct libusb_interface));
    375407        config->interface = usb_interface;
    376         if (!config->interface)
     408        if (!usb_interface)
    377409                return LIBUSB_ERROR_NO_MEM;
    378410
    379         memset(usb_interface, 0, tmp);
    380411        buffer += config->bLength;
    381412        size -= config->bLength;
     
    394425                        usbi_parse_descriptor(buffer, "bb", &header, 0);
    395426
    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);
    400431                                r = LIBUSB_ERROR_IO;
    401432                                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;
    402439                        }
    403440
    404441                        /* If we find another "proper" descriptor then we're done */
    405442                        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))
    411446                                break;
    412447
    413                         usbi_dbg("skipping descriptor 0x%x\n", header.bDescriptorType);
     448                        usbi_dbg("skipping descriptor 0x%x", header.bDescriptorType);
    414449                        buffer += header.bLength;
    415450                        size -= header.bLength;
     
    436471                if (r < 0)
    437472                        goto err;
     473                if (r == 0) {
     474                        config->bNumInterfaces = (uint8_t)i;
     475                        break;
     476                }
    438477
    439478                buffer += r;
     
    448487}
    449488
     489static 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
    450512int usbi_device_cache_descriptor(libusb_device *dev)
    451513{
    452         int r, host_endian;
     514        int r, host_endian = 0;
    453515
    454516        r = usbi_backend->get_device_descriptor(dev, (unsigned char *) &dev->device_descriptor,
     
    467529}
    468530
    469 /** \ingroup desc
     531/** \ingroup libusb_desc
    470532 * Get the USB device descriptor for a given device.
    471533 *
    472534 * 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.
    473538 *
    474539 * \param dev the device
     
    485550}
    486551
    487 /** \ingroup desc
     552/** \ingroup libusb_desc
    488553 * Get the USB configuration descriptor for the currently active configuration.
    489554 * This is a non-blocking function which does not involve any requests being
     
    502567        struct libusb_config_descriptor **config)
    503568{
    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];
    506571        unsigned char *buf = NULL;
    507572        int host_endian = 0;
    508573        int r;
    509574
    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)
    512588                return LIBUSB_ERROR_NO_MEM;
    513589
    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 
    526590        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);
    538594
    539595        free(buf);
    540         *config = _config;
    541         return 0;
    542 
    543 err:
    544         free(_config);
    545         if (buf)
    546                 free(buf);
    547596        return r;
    548597}
    549598
    550 /** \ingroup desc
     599/** \ingroup libusb_desc
    551600 * Get a USB configuration descriptor based on its index.
    552601 * This is a non-blocking function which does not involve any requests being
     
    567616        uint8_t config_index, struct libusb_config_descriptor **config)
    568617{
    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];
    571620        unsigned char *buf = NULL;
    572621        int host_endian = 0;
     
    577626                return LIBUSB_ERROR_NOT_FOUND;
    578627
    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)
    581641                return LIBUSB_ERROR_NO_MEM;
    582642
    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;
    596643        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);
    608647
    609648        free(buf);
    610         *config = _config;
    611         return 0;
    612 
    613 err:
    614         free(_config);
    615         if (buf)
    616                 free(buf);
    617649        return r;
    618650}
     
    621653 * matching a specific bConfigurationValue in the idx output parameter, or -1
    622654 * if the config was not found.
    623  * returns 0 or a LIBUSB_ERROR code
     655 * returns 0 on success or a LIBUSB_ERROR code
    624656 */
    625657int usbi_get_config_index_by_value(struct libusb_device *dev,
     
    634666                int r = usbi_backend->get_config_descriptor(dev, i, tmp, sizeof(tmp),
    635667                        &host_endian);
    636                 if (r < 0)
     668                if (r < 0) {
     669                        *idx = -1;
    637670                        return r;
     671                }
    638672                if (tmp[5] == bConfigurationValue) {
    639673                        *idx = i;
     
    646680}
    647681
    648 /** \ingroup desc
     682/** \ingroup libusb_desc
    649683 * Get a USB configuration descriptor with a specific bConfigurationValue.
    650684 * This is a non-blocking function which does not involve any requests being
     
    666700        uint8_t bConfigurationValue, struct libusb_config_descriptor **config)
    667701{
    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);
    670714        if (r < 0)
    671715                return r;
     
    676720}
    677721
    678 /** \ingroup desc
     722/** \ingroup libusb_desc
    679723 * Free a configuration descriptor obtained from
    680724 * libusb_get_active_config_descriptor() or libusb_get_config_descriptor().
     
    694738}
    695739
    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 */
     753int 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 */
     798void API_EXPORTED libusb_free_ss_endpoint_companion_descriptor(
     799        struct libusb_ss_endpoint_companion_descriptor *ep_comp)
     800{
     801        free(ep_comp);
     802}
     803
     804static 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 */
     894int 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 */
     943void 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 */
     968int 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 */
     1007void 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 */
     1026int 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 */
     1065void 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 */
     1084int 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 */
     1122void 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
    6971129 * Retrieve a string descriptor in C style ASCII.
    6981130 *
     
    7001132 * supported by the device.
    7011133 *
    702  * \param dev a device handle
     1134 * \param dev_handle a device handle
    7031135 * \param desc_index the index of the descriptor to retrieve
    7041136 * \param data output buffer for ASCII string descriptor
     
    7061138 * \returns number of bytes returned in data, or LIBUSB_ERROR code on failure
    7071139 */
    708 int API_EXPORTED libusb_get_string_descriptor_ascii(libusb_device_handle *dev,
     1140int API_EXPORTED libusb_get_string_descriptor_ascii(libusb_device_handle *dev_handle,
    7091141        uint8_t desc_index, unsigned char *data, int length)
    7101142{
     
    7251157                return LIBUSB_ERROR_INVALID_PARAM;
    7261158
    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));
    7281160        if (r < 0)
    7291161                return r;
     
    7341166        langid = tbuf[2] | (tbuf[3] << 8);
    7351167
    736         r = libusb_get_string_descriptor(dev, desc_index, langid, tbuf,
     1168        r = libusb_get_string_descriptor(dev_handle, desc_index, langid, tbuf,
    7371169                sizeof(tbuf));
    7381170        if (r < 0)
     
    7491181                        break;
    7501182
    751                 if (tbuf[si + 1]) /* high byte */
     1183                if ((tbuf[si] & 0x80) || (tbuf[si + 1])) /* non-ASCII */
    7521184                        data[di++] = '?';
    7531185                else
     
    7581190        return di;
    7591191}
    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                         } else
    826                                 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                         } else
    839                                 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.