Skip to content
Snippets Groups Projects
usb_desc.c 14 KiB
Newer Older
  • Learn to ignore specific revisions
  • /* Teensyduino Core Library
     * http://www.pjrc.com/teensy/
     * Copyright (c) 2013 PJRC.COM, LLC.
     *
     * Permission is hereby granted, free of charge, to any person obtaining
     * a copy of this software and associated documentation files (the
     * "Software"), to deal in the Software without restriction, including
     * without limitation the rights to use, copy, modify, merge, publish,
     * distribute, sublicense, and/or sell copies of the Software, and to
     * permit persons to whom the Software is furnished to do so, subject to
     * the following conditions:
     *
     * 1. The above copyright notice and this permission notice shall be 
     * included in all copies or substantial portions of the Software.
     *
     * 2. If the Software is incorporated into a build system that allows 
     * selection among a list of target devices, then similar target
     * devices manufactured by PJRC.COM must be included in the list of
     * target devices and selectable in the same manner.
     *
     * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
     * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
     * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
     * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
     * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
     * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
     * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
     * SOFTWARE.
     */
    
    #include "usb_desc.h"
    #include "mk20dx128.h"
    
    // USB Descriptors are binary data which the USB host reads to
    // automatically detect a USB device's capabilities.  The format
    // and meaning of every field is documented in numerous USB
    // standards.  When working with USB descriptors, despite the
    // complexity of the standards and poor writing quality in many
    // of those documents, remember descriptors are nothing more
    // than constant binary data that tells the USB host what the
    // device can do.  Computers will load drivers based on this data.
    // Those drivers then communicate on the endpoints specified by
    // the descriptors.
    
    // To configure a new combination of interfaces or make minor
    // changes to existing configuration (eg, change the name or ID
    // numbers), usually you would edit "usb_desc.h".  This file
    // is meant to be configured by the header, so generally it is
    // only edited to add completely new USB interfaces or features.
    
    
    
    // **************************************************************
    //   USB Device
    // **************************************************************
    
    #define LSB(n) ((n) & 255)
    #define MSB(n) (((n) >> 8) & 255)
    
    // USB Device Descriptor.  The USB host reads this first, to learn
    // what type of device is connected.
    
    static const uint8_t device_descriptor[] = {
    
            18,                                     // bLength
            1,                                      // bDescriptorType
            0x00, 0x02,                             // bcdUSB
    
            0,                                      // bDeviceClass
            0,                                      // bDeviceSubClass
            0,                                      // bDeviceProtocol
    
            EP0_SIZE,                               // bMaxPacketSize0
            LSB(VENDOR_ID), MSB(VENDOR_ID),         // idVendor
            LSB(PRODUCT_ID), MSB(PRODUCT_ID),       // idProduct
    
            LSB(DEVICE_VER), MSB(DEVICE_VER),       // bcdDevice
    
            1,                                      // iManufacturer
            2,                                      // iProduct
            3,                                      // iSerialNumber
            1                                       // bNumConfigurations
    };
    
    // **************************************************************
    //   USB Configuration
    // **************************************************************
    
    // USB Configuration Descriptor.  This huge descriptor tells all
    // of the devices capbilities.
    
    static const uint8_t config_descriptor[CONFIG_DESC_SIZE] = {
    
            // configuration descriptor, USB spec 9.6.3, page 264-266, Table 9-10
            9,                                      // bLength;
            2,                                      // bDescriptorType;
            LSB(CONFIG_DESC_SIZE),                 // wTotalLength
            MSB(CONFIG_DESC_SIZE),
            NUM_INTERFACE,                          // bNumInterfaces
            1,                                      // bConfigurationValue
            0,                                      // iConfiguration
    
            50,                                     // bMaxPower
    
    #ifdef FC_INTERFACE
            // interface descriptor, USB spec 9.6.5, page 267-269, Table 9-12
            9,                                      // bLength
            4,                                      // bDescriptorType
            FC_INTERFACE,                           // bInterfaceNumber
            0,                                      // bAlternateSetting
            1,                                      // bNumEndpoints
            0xff,                                   // bInterfaceClass (Vendor specific)
            0x00,                                   // bInterfaceSubClass
            0x00,                                   // bInterfaceProtocol
            0,                                      // iInterface
            // endpoint descriptor, USB spec 9.6.6, page 269-271, Table 9-13
            7,                                      // bLength
            5,                                      // bDescriptorType
            FC_OUT_ENDPOINT,                        // bEndpointAddress
            0x02,                                   // bmAttributes (0x02=bulk)
            FC_OUT_SIZE, 0,                         // wMaxPacketSize
            0,                                      // bInterval
    #endif // FC_INTERFACE
    
    
    #ifdef DFU_INTERFACE
            // interface descriptor, DFU Mode (DFU spec Table 4.4)
            9,                                      // bLength
            4,                                      // bDescriptorType
            DFU_INTERFACE,                          // bInterfaceNumber
            0,                                      // bAlternateSetting
            0,                                      // bNumEndpoints
            0xFE,                                   // bInterfaceClass
            0x01,                                   // bInterfaceSubClass
            0x01,                                   // bInterfaceProtocol (Runtime)
            4,                                      // iInterface
            // DFU Functional Descriptor (DFU spec TAble 4.2)
            9,                                      // bLength
            0x21,                                   // bDescriptorType
            0x0D,                                   // bmAttributes
            LSB(DFU_DETACH_TIMEOUT),                // wDetachTimeOut
            MSB(DFU_DETACH_TIMEOUT),
            LSB(DFU_TRANSFER_SIZE),                 // wTransferSize
            MSB(DFU_TRANSFER_SIZE),
            0x01,0x01,                              // bcdDFUVersion
    #endif // DFU_INTERFACE
    
    };
    
    
    // **************************************************************
    //   String Descriptors
    // **************************************************************
    
    struct usb_string_descriptor_struct {
    
        uint8_t bLength;
        uint8_t bDescriptorType;
        uint16_t wString[];
    
    static const struct usb_string_descriptor_struct string0 = {
    
    static const struct usb_string_descriptor_struct usb_string_manufacturer_name = {
    
        2 + MANUFACTURER_NAME_LEN * 2,
        3,
        MANUFACTURER_NAME
    
    static const struct usb_string_descriptor_struct usb_string_product_name = {
    
        2 + PRODUCT_NAME_LEN * 2,
        3,
        PRODUCT_NAME
    
    static const struct usb_string_descriptor_struct usb_string_dfu_name = {
    
        2 + DFU_NAME_LEN * 2,
        3,
        DFU_NAME
    };
    
    // Microsoft OS String Descriptor. See: https://github.com/pbatard/libwdi/wiki/WCID-Devices
    
    static const struct usb_string_descriptor_struct usb_string_microsoft = {
    
        18, 3,
        {'M','S','F','T','1','0','0', MSFT_VENDOR_CODE}
    };
    
    // Microsoft WCID
    
        16 + 2*24, 0, 0, 0,             // Length
    
        0x00, 0x01,                     // Version
        0x04, 0x00,                     // Compatibility ID descriptor index
    
        0, 0, 0, 0, 0, 0, 0,            // Reserved (7 bytes)
    
    
        FC_INTERFACE,                   // Interface number
    
        0x01,                           // Reserved
        'W','I','N','U','S','B',0,0,    // Compatible ID
        0,0,0,0,0,0,0,0,                // Sub-compatible ID (unused)
        0,0,0,0,0,0,                    // Reserved
    
    
        DFU_INTERFACE,                  // Interface number
        0x01,                           // Reserved
        'W','I','N','U','S','B',0,0,    // Compatible ID
        0,0,0,0,0,0,0,0,                // Sub-compatible ID (unused)
        0,0,0,0,0,0,                    // Reserved
    };
    
    // Microsoft Extended Properties descriptor, necessary for WinUSB on a multi-interface device.
    
    const uint8_t usb_microsoft_extprop[] = {
    
        146, 0, 0, 0,                   // Length
        0x00, 0x01,                     // Version
        0x05, 0x00,                     // Extended Properties descriptor index
        1, 0,                           // Number of sections
    
        136, 0, 0, 0,                   // Length of section
        7, 0, 0, 0,                     // Data type (REG_MULTI_SZ)
        42, 0,                          // Property name length
        // L"DeviceInterfaceGUIDs"
        'D',0,'e',0,'v',0,'i',0,'c',0,'e',0,'I',0,'n',0,'t',0,'e',0,'r',0,'f',0,'a',0,'c',0,'e',0,'G',0,'U',0,'I',0,'D',0,'s',0,0,0,
        80, 0, 0, 0,                    // Property data length
        // L"{62fd4123-87e3-47fe-946a-e044f36f2fb3}\0"
        '{',0,'6',0,'2',0,'f',0,'d',0,'4',0,'1',0,'2',0,'3',0,'-',0,'8',0,'7',0,'e',0,'3',0,'-',0,'4',0,'7',0,'f',0,'e',0,'-',0,'9',0,'4',0,'6',0,'a',0,'-',0,'e',0,'0',0,'4',0,'4',0,'f',0,'3',0,'6',0,'f',0,'2',0,'f',0,'b',0,'3',0,'}',0,
        0,0,0,0,
    
    
    // 32-digit hex string, corresponding to the MK20DX128's built-in unique 128-bit ID.
    
    Micah Elizabeth Scott's avatar
    Micah Elizabeth Scott committed
    static FLEXRAM_DATA struct {
        uint8_t bLength;
        uint8_t bDescriptorType;
        uint16_t wString[16];
    } usb_string_serial_number;
    
    void usb_init_serialnumber(void)
    
        /*
         * The CPU has a 128-bit unique serial number. But it seems to encode
         * some manufacturing information in a way that makes it hard to visually
         * parse. Distinct devices may have very similar-looking serial numbers.
         *
         * To make this more user-friendly, we mix up the serial number using
         * an FNV hash and alphabetically encode it.
         */
    
    
    Micah Elizabeth Scott's avatar
    Micah Elizabeth Scott committed
        usb_string_serial_number.bLength = 2 + 16 * 2;
        usb_string_serial_number.bDescriptorType = 3;
    
    
        union {
            uint8_t bytes[16];
            struct {
                uint32_t a, b, c, d;
            };
        } serial;
    
    
        uint32_t hash = 0x811c9dc5;
    
        serial.d = SIM_UIDH;
        serial.c = SIM_UIDMH;
        serial.b = SIM_UIDML;
        serial.a = SIM_UIDL;
    
        // Initial hash
        for (i = 0; i < 16; ++i) {
            hash = (hash ^ serial.bytes[i]) * 0x1000193;
    
        // Output letters
        for (i = 0; i < 16; ++i) {
            hash = (hash ^ serial.bytes[i]) * 0x1000193;
    
            usb_string_serial_number.wString[i] = 'A' + (hash % 26);
    
    }
    
    
    // **************************************************************
    //   Descriptors List
    // **************************************************************
    
    // This table provides access to all the descriptor data above.
    
    const usb_descriptor_list_t usb_descriptor_list[] = {
    
        {0x0100, device_descriptor, sizeof(device_descriptor)},
        {0x0200, config_descriptor, sizeof(config_descriptor)},
        {0x0300, (const uint8_t *)&string0, 0},
        {0x0301, (const uint8_t *)&usb_string_manufacturer_name, 0},
        {0x0302, (const uint8_t *)&usb_string_product_name, 0},
        {0x0303, (const uint8_t *)&usb_string_serial_number, 0},
        {0x0304, (const uint8_t *)&usb_string_dfu_name, 0},
        {0x03EE, (const uint8_t *)&usb_string_microsoft, 0},
        {0, NULL, 0}
    
    };
    
    
    // **************************************************************
    //   Endpoint Configuration
    // **************************************************************
    
    const uint8_t usb_endpoint_config_table[NUM_ENDPOINTS] = 
    {
    #if (defined(ENDPOINT1_CONFIG) && NUM_ENDPOINTS >= 1)
    
    #elif (NUM_ENDPOINTS >= 1)
    
    #endif
    #if (defined(ENDPOINT2_CONFIG) && NUM_ENDPOINTS >= 2)
    
    #elif (NUM_ENDPOINTS >= 2)
    
    #endif
    #if (defined(ENDPOINT3_CONFIG) && NUM_ENDPOINTS >= 3)
    
    #elif (NUM_ENDPOINTS >= 3)
    
    #endif
    #if (defined(ENDPOINT4_CONFIG) && NUM_ENDPOINTS >= 4)
    
    #elif (NUM_ENDPOINTS >= 4)
    
    #endif
    #if (defined(ENDPOINT5_CONFIG) && NUM_ENDPOINTS >= 5)
    
    #elif (NUM_ENDPOINTS >= 5)
    
    #endif
    #if (defined(ENDPOINT6_CONFIG) && NUM_ENDPOINTS >= 6)
    
    #elif (NUM_ENDPOINTS >= 6)
    
    #endif
    #if (defined(ENDPOINT7_CONFIG) && NUM_ENDPOINTS >= 7)
    
    #elif (NUM_ENDPOINTS >= 7)
    
    #endif
    #if (defined(ENDPOINT8_CONFIG) && NUM_ENDPOINTS >= 8)
    
    #elif (NUM_ENDPOINTS >= 8)
    
    #endif
    #if (defined(ENDPOINT9_CONFIG) && NUM_ENDPOINTS >= 9)
    
    #elif (NUM_ENDPOINTS >= 9)
    
    #endif
    #if (defined(ENDPOINT10_CONFIG) && NUM_ENDPOINTS >= 10)
    
    #elif (NUM_ENDPOINTS >= 10)
    
    #endif
    #if (defined(ENDPOINT11_CONFIG) && NUM_ENDPOINTS >= 11)
    
    #elif (NUM_ENDPOINTS >= 11)
    
    #endif
    #if (defined(ENDPOINT12_CONFIG) && NUM_ENDPOINTS >= 12)
    
    #elif (NUM_ENDPOINTS >= 12)
    
    #endif
    #if (defined(ENDPOINT13_CONFIG) && NUM_ENDPOINTS >= 13)
    
    #elif (NUM_ENDPOINTS >= 13)
    
    #endif
    #if (defined(ENDPOINT14_CONFIG) && NUM_ENDPOINTS >= 14)
    
    #elif (NUM_ENDPOINTS >= 14)
    
    #endif
    #if (defined(ENDPOINT15_CONFIG) && NUM_ENDPOINTS >= 15)
    
    #elif (NUM_ENDPOINTS >= 15)