banner
yono

yono

哈喽~欢迎光临
follow
github

USB TMC Enumeration - A Must-Have in the Instrumentation Industry

Basic Introduction#

The standard instrument equipment interface is used for communication between the host computer and instruments, such as oscilloscopes, power supplies, signal generators, etc. The commonly used standard instrument interfaces and their corresponding communication protocols are as follows.

Communication InterfaceCommunication Protocol
LANVXI-11
USBUSB-TMC
GPIBIEEE488

The communication protocol for LAN is essentially the same as TCP Client/Server and RAW Socket of TCP/IP; as long as one of them is supported, it can be recognized normally.

The communication protocol for GPIB is relatively complex, and the physical layer uses parallel transmission, which can almost only communicate using dedicated transceiver chips. The control method for dedicated transceivers is similar to SRAM, and there are no particularly difficult problems to solve.

The communication method for USB must be appreciated.

As a company that provides testing devices and instruments, it is surprising that we have no technical accumulation in this area, so let me take a closer look at what USB-TMC enumeration is.

Overall Idea#

First and foremost, it is very important to review the standard documents. I have two bilingual versions that I translated myself, which need to be viewed in a two-page layout.

USBTMC_1_00_EnCh.pdf | yono's file
USBTMC_usb488_subclass_1_00_EnCh.pdf | yono's file

Secondly, it is very unimportant to find a competitor's standard instrument to capture data packets and see what the situation is.

ps: In fact, I have not even glanced at the second document; I completely relied on capturing data and cross-referencing with the first document to complete the driver development.

It is worth noting that the USB-TMC class enumeration actually has only one subclass, USB488, and there are still many reserved data locations and communication processes, so I believe that this TMC enumeration can still be used for many years.

Enumeration Process#

As we all know, during the enumeration phase of USB devices, the driver needs to query descriptors, and how this descriptor is written is key to enumerating as a TMC device.

Enumeration Process Data Packet

After capturing packets, an example device descriptor has emerged, and it should be exactly the same as follows.

UCHAR device_framework_full_speed_demo[] = {
    /* Device descriptor */
    0x12, 		// len(static)
    0x01, 		// device(static)
    0x00, 0x02,	// USB2.0(static)
    0x00,		// DeviceClass(static)
    0x00,		// DeviceSubClass(static)
    0x00,		// bDeviceProtocol(static)
    0x40,		// bMaxPacketSize0
    0x69, 0x0A,	// idVendor (VID): 0x0A69
    0x8A, 0x08,	// idProduct (PID): 0x088A
    0x00, 0x01,	// bcdDevice: Device release number 0x0100
    0x01,		// iManufacturer (static)
    0x02,		// iProduct (static)
    0x03,		// iSerialNumber (static)
    0x01,		// bNumConfigurations(static)

    /* Device qualifier descriptor */
    0x0a,		// len(static)
    0x06,		// bDescriptorType: Device Qualifier Descriptor(static)
    0x00, 0x02,	// USB2.0(static)
    0x00,		// DeviceClass(static)
    0x00,		// DeviceSubClass(static)
    0x00,		// bDeviceProtocol(static)
    0x40,		// bMaxPacketSize0
    0x01,		// bNumConfigurations(static)
    0x00,		// reserve(static)

    /* Configuration descriptor */
    0x09, 		// len(static)
    0x02, 		// bDescriptorType: Configuration Descriptor(static)
    0x20, 0x00,	// wTotalLength: Total length of data for this configuration (32 bytes)
    0x01, 		// bNumInterfaces(static)
    0x01, 		// bConfigurationValue(static)
    0x00, 		// iConfiguration: Index of configuration string descriptor
    0x80, 		// bmAttributes: Self-powered
    0x64, 		// bMaxPower: 200 mA (0x64 * 2 mA)

    /* Interface descriptor */
    0x09, 		// len(static)
    0x04, 		// bDescriptorType: Interface Descriptor(static)
    0x00, 		// bInterfaceNumber: Interface 0
    0x00, 		// bAlternateSetting
    0x02, 		// bNumEndpoints: 2 endpoints(static)
    0xFE, 		// bInterfaceClass: TMC(static)
    0x03, 		// bInterfaceSubClass: USB488(static)
    0x01, 		// bInterfaceProtocol(static)
    0x00, 		// iInterface: Index of interface string descriptor(static)

    /* Endpoint descriptor (Bulk Out) */
    0x07, 		// len(static)
    0x05,		// bDescriptorType: Endpoint Descriptor(static)
    0x81,		// bEndpointAddress: Endpoint address (IN direction, endpoint number 1)
    0x02,		// bmAttributes: Transfer type is Bulk
    0x40, 0x00,	// wMaxPacketSize: Maximum packet size 64 bytes
    0x00,		// bInterval: Polling interval (0 for bulk transfer)

    /* Endpoint descriptor (Bulk In) */
    0x07,		// len(static)
    0x05,		// bDescriptorType: Endpoint Descriptor(static)
    0x01,		// bEndpointAddress: Endpoint address (OUT direction, endpoint number 1)
    0x02,		// bmAttributes: Transfer type is Bulk
    0x40, 0x00,	// wMaxPacketSize: Maximum packet size 64 bytes
    0x00,		// bInterval: Polling interval (0 for bulk transfer)
};

The most critical parts are bInterfaceClass and bInterfaceSubClass in the Interface descriptor, which directly determine our enumeration type. Of course, all other parts marked with (static) in the descriptor must be exactly the same as the example, as per standard regulations. Some are TMC standards, and some are USB standards; in any case, the parts marked with (static) must be identical.

Notably, the USB-TMC protocol does not distinguish between high-speed and full-speed; regardless of what USB line it operates on, a device qualifier is required, which is different from other modern USB protocol stacks.

Another special point is that USB-TMC devices cannot use composite enumeration, as it will disrupt NI-VISA's recognition and resource invocation for USB-TMC. For example, I tried to composite enumerate as TMC + serial device, and it showed up in the device manager (enumeration successful), but when using VISA calls, only the serial port could be invoked, and the TMC device could not be called.

Sending Commands to the Instrument#

In communication, messages are sent to the instrument through the Bulk Out endpoint described earlier. Therefore, the device should repeatedly open this Bulk Out endpoint to receive any possible messages. If the device does not open the Bulk Out endpoint, it will generate a large number of NAK packets, triggering a setup command to restart the endpoint.

For example, a classic message is as follows:

0x01, 0x02, 0xFD, 0x00, 0x06, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x2A, 0x49, 0x44, 0x4E, 0x3F, 0x0A, 0x00, 0x00

This message can be parsed according to the description in the USBTMC_1_00 standard document, divided into the following segments.

Message SegmentMeaning
0x01Indicates that this message is an OUT request, meaning the instrument needs to parse the content of this message to take action.
0x02, 0xFDMessage tag and its complement; this tag will be different for each message.
0x00Fixed 0.
0x06, 0x00, 0x00, 0x00Indicates that the content length is 6, transmitted in little-endian 32-bit.
0x01, 0x00, 0x00, 0x00Only the first bit of the first byte is meaningful; 1 indicates that the content statement has been fully transmitted, while 0 indicates that there is more content to follow.
0x2A, 0x49, 0x44, 0x4E, 0x3F, 0x0AThe formal content of 6 bytes, which is *IDN?\r, where \r is the escape character for carriage return.
0x00, 0x00Padding zeros to ensure the total message length is 4-byte aligned.

In simple terms, it means parsing the first byte of the message data received from the Bulk Out endpoint; if it is 0x01, then proceed with further parsing to extract the formal SCPI command "*IDN?" and process it.

Receiving Replies from the Instrument#

In communication, the instrument cannot arbitrarily send replies to the host via USB, which is different from serial communication. When the host expects to receive reply data, it will send a message on the Bulk Out endpoint indicating that the instrument can now send a reply. A classic message to enable replies is as follows:

0x02, 0x03, 0xFC, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,

This message can be parsed according to the description in the USBTMC_1_00 standard document, divided into the following segments.

Message SegmentMeaning
0x02Indicates that this message is an IN request, meaning the instrument should send any possible replies.
0x03, 0xFCMessage tag and its complement; this tag will be different for each message.
0x00Fixed 0.
0x00, 0x04, 0x00, 0x00Indicates that the host has opened a 0x0400 buffer for receiving 1024 bytes; this value does not need to be considered; the host has abundant resources, and if communication fails, it can just increase the size.
0x00The D1 bit, which is the &0x2 bit, is meaningful; if it is 1, it means that the next byte is a terminator that the host supports and the device must support; if it is 0, it can be ignored. Other bits are reserved.
0x00If the previous byte's D1 bit is 1, then this byte indicates the supported terminator, such as \r.
0x00, 0x00Padding reserved characters, which also ensures that the total message length is 4-byte aligned.

In simple terms, it means parsing the first byte of the message data received from the Bulk Out endpoint; if it is 0x02, then open the device's sending channel. For example, if the host previously transmitted the formal SCPI command "*IDN?", then the device needs to transmit the reply generated.

As for the specified terminator function, it will not be supported; in the subsequent necessary setup packet introduction, it will be explained how to let the host know that we do not support this specification.

A classic reply transmission message is as follows:

0x02, 0x03, 0xFC, 0x00, 0x06, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x2A, 0x49, 0x44, 0x4E, 0x3F, 0x0A, 0x00, 0x00,

Message SegmentMeaning
0x02Indicates that this message is a reply to an IN request.
0x03, 0xFCMessage tag and its complement; this tag should replicate the request message.
0x00Fixed 0.
0x06, 0x00, 0x00, 0x00Indicates that the valid reply data is a total of 6 bytes, transmitted in little-endian 32-bit.
0x01, 0x00, 0x00, 0x00Only the first bit of the first byte is meaningful; 1 indicates that the content statement has been fully transmitted, while 0 indicates that there is more content to follow.
0x2A, 0x49, 0x44, 0x4E, 0x3F, 0x0A*IDN?\r, this is a loopback test; my instrument will copy the last received command to respond to any reply.
0x00, 0x00Padding zeros to ensure the total message length is 4-byte aligned.

An Excellent Method for 4-Byte Alignment#

Using bitwise operations is a very convenient and efficient way to achieve 4-byte alignment.

uint8_t *data;  				// Original message buffer
uint32_t len =  OriginalLen;	// Original length

for(uint32_t i = len; i < ((len + 0x3) & ~0x3U); i++)
{
    data[i] = 0x00;
}

_write(data, ((len + 0x3) & ~0x3U));

Handling Setup Packets#

The standard setup packets starting with 0x80,0x06 have the following categories; I am also unable to assist much, as I completely rely on the usbx protocol stack for automatic processing.

#define UX_GET_STATUS                                                   0u
#define UX_CLEAR_FEATURE                                                1u
#define UX_SET_FEATURE                                                  3u 
#define UX_SET_ADDRESS                                                  5u
#define UX_GET_DESCRIPTOR                                               6u
#define UX_SET_DESCRIPTOR                                               7u
#define UX_GET_CONFIGURATION                                            8u
#define UX_SET_CONFIGURATION                                            9u
#define UX_GET_INTERFACE                                                10u
#define UX_SET_INTERFACE                                                11u
#define UX_SYNCH_FRAME                                                  12u

The COMMAND_REQUEST packets starting with 0xA1 or 0xA2 have the following content that we must handle.

Starting with 0xA1,0x07, it indicates querying the device's features. An example reply packet is as follows. The content is quite extensive; you can refer to the document yourself, but it is recommended to directly respond with this.

0x01, 0x00, 0x10, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,

Starting with 0xA2,0x01, it indicates clearing the bulk-out transfer; we don't need to worry about anything, just reply successfully, and the reply packet is as follows.

0x01, tag (copy the second byte of the setup packet starting from 0).

Starting with 0xA2,0x02, it indicates inquiring about the status of the previously cleared bulk-out transfer request; we don't need to worry about anything, just reply successfully, and the reply packet is as follows.

0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00.

As for other COMMAND_REQUEST packets, they are not commonly used and can be ignored.

[!NOTE]

First, you should be somewhat familiar with the practice of usbx, at least having used existing examples like CDC_ACM; otherwise, the following content will be meaningless. The following is an example of how to add TMC enumeration into the usbx USB protocol stack framework.

Practice of Enumeration Descriptors in usbx#

Pay attention to the _ux_device_stack_initialize function, where there is such a part, which means that it will only register based on the position of the Interface descriptor and Configuration descriptor in the overall descriptor. Everything before the Configuration descriptor is treated as the ==device descriptor==, and everything after the Interface descriptor is treated as the ==endpoint descriptor==.

            switch(descriptor_type)
            {

            case UX_INTERFACE_DESCRIPTOR_ITEM:
				......
                break;

            case UX_CONFIGURATION_DESCRIPTOR_ITEM:
				......
                break;

            default:
                break;
            }

Then, it is necessary to slightly modify the content of the original ux_device_descriptors.c file in the USBD_Device_Framework_Builder function, which originally should have content similar to the following. It specifies that only high-speed USB requires a device qualifier; however, our USB-TMC usually operates on full-speed USB, and since the USB-TMC protocol does not distinguish between high-speed and full-speed, we need to remove this if condition so that regardless of high-speed or full-speed, this device qualifier is included in the descriptor.

    /* Check if USBx is in high speed mode to add qualifier descriptor */
    if(Speed == USBD_HIGH_SPEED)
    {
        pDevQualDesc                     = (USBD_DevQualiDescTypedef *)(pDevFrameWorkDesc + pdev->CurrDevDescSz);
        pDevQualDesc->bLength            = (uint8_t)sizeof(USBD_DevQualiDescTypedef);
        pDevQualDesc->bDescriptorType    = UX_DEVICE_QUALIFIER_DESCRIPTOR_ITEM;
        pDevQualDesc->bcdDevice          = 0x0200;
        pDevQualDesc->Class              = 0x00;
        pDevQualDesc->SubClass           = 0x00;
        pDevQualDesc->Protocol           = 0x00;
        pDevQualDesc->bMaxPacketSize     = 0x40;
        pDevQualDesc->bNumConfigurations = 0x01;
        pDevQualDesc->bReserved          = 0x00;
        pdev->CurrDevDescSz += (uint32_t)sizeof(USBD_DevQualiDescTypedef);
    }

Other parts do not require major adjustments; just modify the parameters segment by segment according to the example descriptor. The modified ux_device_descriptors.c and ux_device_descriptors.h files based on the original file are as follows.

/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file    ux_device_descriptors.c
  * @author  MCD Application Team
  * @brief   USBX Device descriptor header file
  ******************************************************************************
  * @attention
  *
  * Copyright (c) 2020-2021 STMicroelectronics.
  * All rights reserved.
  *
  * This software is licensed under terms that can be found in the LICENSE file
  * in the root directory of this software component.
  * If no LICENSE file comes with this software, it is provided AS-IS.
  *
  ******************************************************************************
  */
/* USER CODE END Header */

/* Includes ------------------------------------------------------------------*/
#include "ux_device_descriptors.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */

/* USER CODE END Includes */

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

/* USER CODE END PTD */

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */

/* USER CODE END PD */

/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */

/* USER CODE END PM */

/* Private variables ---------------------------------------------------------*/
USBD_DevClassHandleTypeDef USBD_Device_FS, USBD_Device_HS;

uint8_t                    UserClassInstance[USBD_MAX_CLASS_INTERFACES] = {
    CLASS_TYPE_TMC,
};

uint8_t UserTMCInterface[] = {
    INTERFACE_TMC_USB488,
};

/* The generic device descriptor buffer that will be filled by builder
   Size of the buffer is the maximum possible device FS descriptor size. */
#if defined(__ICCARM__) /* IAR Compiler */
#pragma data_alignment = 4
#endif /* defined ( __ICCARM__ ) */
__ALIGN_BEGIN static uint8_t DevFrameWorkDesc_FS[USBD_FRAMEWORK_MAX_DESC_SZ] __ALIGN_END = {0};

/* The generic device descriptor buffer that will be filled by builder
   Size of the buffer is the maximum possible device HS descriptor size. */
#if defined(__ICCARM__) /* IAR Compiler */
#pragma data_alignment = 4
#endif /* defined ( __ICCARM__ ) */
__ALIGN_BEGIN static uint8_t DevFrameWorkDesc_HS[USBD_FRAMEWORK_MAX_DESC_SZ] __ALIGN_END = {0};

static uint8_t              *pDevFrameWorkDesc_FS                                        = DevFrameWorkDesc_FS;

static uint8_t              *pDevFrameWorkDesc_HS                                        = DevFrameWorkDesc_HS;
/* USER CODE BEGIN PV0 */

/* USER CODE END PV0 */

/* String Device Framework :
 Byte 0 and 1 : Word containing the language ID : 0x0904 for US
 Byte 2       : Byte containing the index of the descriptor
 Byte 3       : Byte containing the length of the descriptor string
*/
#if defined(__ICCARM__) /* IAR Compiler */
#pragma data_alignment = 4
#endif /* defined ( __ICCARM__ ) */
__ALIGN_BEGIN UCHAR USBD_string_framework[USBD_STRING_FRAMEWORK_MAX_LENGTH] __ALIGN_END = {0};

/* Multiple languages are supported on the device, to add
   a language besides English, the Unicode language code must
   be appended to the language_id_framework array and the length
   adjusted accordingly. */

#if defined(__ICCARM__) /* IAR Compiler */
#pragma data_alignment = 4
#endif /* defined ( __ICCARM__ ) */
__ALIGN_BEGIN UCHAR USBD_language_id_framework[LANGUAGE_ID_MAX_LENGTH] __ALIGN_END = {0};

/* USER CODE BEGIN PV1 */

/* USER CODE END PV1 */

/* Private function prototypes -----------------------------------------------*/
static void     USBD_Desc_GetString(uint8_t *desc, uint8_t *Buffer, uint16_t *len);
static uint8_t  USBD_Desc_GetLen(uint8_t *buf);

static uint8_t *USBD_Device_Framework_Builder(USBD_DevClassHandleTypeDef *pdev, uint8_t *pDevFrameWorkDesc, uint8_t *UserClassInstance, uint8_t Speed);

static uint8_t  USBD_FrameWork_AddToConfDesc(USBD_DevClassHandleTypeDef *pdev, uint8_t Speed, uint8_t *pCmpstConfDesc);

static uint8_t  USBD_FrameWork_AddClass(USBD_DevClassHandleTypeDef *pdev, USBD_CompositeClassTypeDef class, uint8_t cfgidx, uint8_t Speed, uint8_t *pCmpstConfDesc);

static uint8_t  USBD_FrameWork_FindFreeIFNbr(USBD_DevClassHandleTypeDef *pdev);

static void     USBD_FrameWork_AddConfDesc(uint32_t Conf, uint32_t *pSze);

static void     USBD_FrameWork_AssignEp(USBD_DevClassHandleTypeDef *pdev, uint8_t Add, uint8_t Type, uint32_t Sze);

static void     USBD_FrameWork_TMC_Desc(USBD_DevClassHandleTypeDef *pdev, uint32_t pConf, uint32_t *Sze);
/* USER CODE BEGIN PFP */

/* USER CODE END PFP */

/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */

/* USER CODE END 0 */

/**
  * @brief  USBD_Get_Device_Framework_Speed
  *         Return the device speed descriptor
  * @param  Speed : HIGH or FULL SPEED flag
  * @param  length : length of HIGH or FULL SPEED array
  * @retval Pointer to descriptor buffer
  */
uint8_t *USBD_Get_Device_Framework_Speed(uint8_t Speed, ULONG *Length)
{
    uint8_t *pFrameWork = NULL;
    /* USER CODE BEGIN Device_Framework0 */

    /* USER CODE END Device_Framework0 */

    if(USBD_FULL_SPEED == Speed)
    {
        USBD_Device_Framework_Builder(&USBD_Device_FS, pDevFrameWorkDesc_FS, UserClassInstance, Speed);

        /* Get the length of USBD_device_framework_full_speed */
        *Length    = (ULONG)(USBD_Device_FS.CurrDevDescSz + USBD_Device_FS.CurrConfDescSz);

        pFrameWork = pDevFrameWorkDesc_FS;
    }
    else
    {
        USBD_Device_Framework_Builder(&USBD_Device_HS, pDevFrameWorkDesc_HS, UserClassInstance, Speed);

        /* Get the length of USBD_device_framework_high_speed */
        *Length    = (ULONG)(USBD_Device_HS.CurrDevDescSz + USBD_Device_HS.CurrConfDescSz);

        pFrameWork = pDevFrameWorkDesc_HS;
    }
    /* USER CODE BEGIN Device_Framework1 */

    /* USER CODE END Device_Framework1 */
    return pFrameWork;
}

/**
  * @brief  USBD_Get_String_Framework
  *         Return the language_id_framework
  * @param  Length : Length of String_Framework
  * @retval Pointer to language_id_framework buffer
  */
uint8_t *USBD_Get_String_Framework(ULONG *Length)
{
    uint16_t len   = 0U;
    uint8_t  count = 0U;

    /* USER CODE BEGIN String_Framework0 */

    /* USER CODE END String_Framework0 */

    /* Set the Manufacturer language Id and index in USBD_string_framework */
    USBD_string_framework[count++] = USBD_LANGID_STRING & 0xFF;
    USBD_string_framework[count++] = USBD_LANGID_STRING >> 8;
    USBD_string_framework[count++] = USBD_IDX_MFC_STR;

    /* Set the Manufacturer string in string_framework */
    USBD_Desc_GetString((uint8_t *)USBD_MANUFACTURER_STRING, USBD_string_framework + count, &len);

    /* Set the Product language Id and index in USBD_string_framework */
    count += len + 1;
    USBD_string_framework[count++] = USBD_LANGID_STRING & 0xFF;
    USBD_string_framework[count++] = USBD_LANGID_STRING >> 8;
    USBD_string_framework[count++] = USBD_IDX_PRODUCT_STR;

    /* Set the Product string in USBD_string_framework */
    USBD_Desc_GetString((uint8_t *)USBD_PRODUCT_STRING, USBD_string_framework + count, &len);

    /* Set Serial language Id and index in string_framework */
    count += len + 1;
    USBD_string_framework[count++] = USBD_LANGID_STRING & 0xFF;
    USBD_string_framework[count++] = USBD_LANGID_STRING >> 8;
    USBD_string_framework[count++] = USBD_IDX_SERIAL_STR;

    /* Set the Serial number in USBD_string_framework */
    USBD_Desc_GetString((uint8_t *)USBD_SERIAL_NUMBER, USBD_string_framework + count, &len);

    /* USER CODE BEGIN String_Framework1 */

    /* USER CODE END String_Framework1 */

    /* Get the length of USBD_string_framework */
    *Length = strlen((const char *)USBD_string_framework);

    return USBD_string_framework;
}

/**
  * @brief  USBD_Get_Language_Id_Framework
  *         Return the language_id_framework
  * @param  Length : Length of Language_Id_Framework
  * @retval Pointer to language_id_framework buffer
  */
uint8_t *USBD_Get_Language_Id_Framework(ULONG *Length)
{
    uint8_t count = 0U;

    /* Set the language Id in USBD_language_id_framework */
    USBD_language_id_framework[count++] = USBD_LANGID_STRING & 0xFF;
    USBD_language_id_framework[count++] = USBD_LANGID_STRING >> 8;

    /* Get the length of USBD_language_id_framework */
    *Length = strlen((const char *)USBD_language_id_framework);

    return USBD_language_id_framework;
}

/**
  * @brief  USBD_Get_Interface_Number
  *         Return interface number
  * @param  class_type : Device class type
  * @param  interface_type : Device interface type
  * @retval interface number
  */
uint16_t USBD_Get_Interface_Number(uint8_t class_type, uint8_t interface_type)
{
    uint8_t itf_num = 0U;
    uint8_t idx     = 0U;

    /* USER CODE BEGIN USBD_Get_Interface_Number0 */

    /* USER CODE END USBD_Get_Interface_Number0 */

    for(idx = 0; idx < USBD_MAX_SUPPORTED_CLASS; idx++)
    {
        if((USBD_Device_FS.tclasslist[idx].ClassType == class_type) && (USBD_Device_FS.tclasslist[idx].InterfaceType == interface_type))
        {
            itf_num = USBD_Device_FS.tclasslist[idx].Ifs[0];
        }
    }

    /* USER CODE BEGIN USBD_Get_Interface_Number1 */

    /* USER CODE END USBD_Get_Interface_Number1 */

    return itf_num;
}

/**
  * @brief  USBD_Get_Configuration_Number
  *         Return configuration number
  * @param  class_type : Device class type
  * @param  interface_type : Device interface type
  * @retval configuration number
  */
uint16_t USBD_Get_Configuration_Number(uint8_t class_type, uint8_t interface_type)
{
    uint8_t cfg_num = 1U;

    /* USER CODE BEGIN USBD_Get_CONFIGURATION_Number0 */

    /* USER CODE END USBD_Get_CONFIGURATION_Number0 */

    /* USER CODE BEGIN USBD_Get_CONFIGURATION_Number1 */

    /* USER CODE END USBD_Get_CONFIGURATION_Number1 */

    return cfg_num;
}

/**
  * @brief  USBD_Desc_GetString
  *         Convert ASCII string into Unicode one
  * @param  desc : descriptor buffer
  * @param  Unicode : Formatted string buffer (Unicode)
  * @param  len : descriptor length
  * @retval None
  */
static void USBD_Desc_GetString(uint8_t *desc, uint8_t *unicode, uint16_t *len)
{
    uint8_t  idx = 0U;
    uint8_t *pdesc;

    if(desc == NULL)
    {
        return;
    }

    pdesc          = desc;
    *len           = (uint16_t)USBD_Desc_GetLen(pdesc);

    unicode[idx++] = *(uint8_t *)len;

    while(*pdesc != (uint8_t)'\0')
    {
        unicode[idx++] = *pdesc;
        pdesc++;
    }
}

/**
  * @brief  USBD_Desc_GetLen
  *         return the string length
  * @param  buf : pointer to the ASCII string buffer
  * @retval string length
  */
static uint8_t USBD_Desc_GetLen(uint8_t *buf)
{
    uint8_t  len   = 0U;
    uint8_t *pbuff = buf;

    while(*pbuff != (uint8_t)'\0')
    {
        len++;
        pbuff++;
    }

    return len;
}

/**
  * @brief  USBD_Device_Framework_Builder
  *         Device Framework builder
  * @param  pdev: device instance
  * @param  pDevFrameWorkDesc: Pointer to the device framework descriptor
  * @param  UserClassInstance: type of the class to be added
  * @param  Speed: Speed parameter HS or FS
  * @retval status
  */
static uint8_t *USBD_Device_Framework_Builder(USBD_DevClassHandleTypeDef *pdev, uint8_t *pDevFrameWorkDesc, uint8_t *UserClassInstance, uint8_t Speed)
{
    static USBD_DeviceDescTypedef   *pDevDesc;
    static USBD_DevQualiDescTypedef *pDevQualDesc;
    uint8_t                          Idx_Instance = 0U;

    /* Set Dev and conf descriptors size to 0 */
    pdev->CurrConfDescSz = 0U;
    pdev->CurrDevDescSz  = 0U;

    /* Set the pointer to the device descriptor area*/
    pDevDesc = (USBD_DeviceDescTypedef *)pDevFrameWorkDesc;

    /* Start building the generic device descriptor common part */
    pDevDesc->bLength            = (uint8_t)sizeof(USBD_DeviceDescTypedef);
    pDevDesc->bDescriptorType    = UX_DEVICE_DESCRIPTOR_ITEM;
    pDevDesc->bcdUSB             = USB_BCDUSB;
    pDevDesc->bDeviceClass       = 0x00;
    pDevDesc->bDeviceSubClass    = 0x00;
    pDevDesc->bDeviceProtocol    = 0x00;
    pDevDesc->bMaxPacketSize     = USBD_MAX_EP0_SIZE;
    pDevDesc->idVendor           = USBD_VID;
    pDevDesc->idProduct          = USBD_PID;
    pDevDesc->bcdDevice          = 0x0100;
    pDevDesc->iManufacturer      = USBD_IDX_MFC_STR;
    pDevDesc->iProduct           = USBD_IDX_PRODUCT_STR;
    pDevDesc->iSerialNumber      = USBD_IDX_SERIAL_STR;
    pDevDesc->bNumConfigurations = USBD_MAX_NUM_CONFIGURATION;
    pdev->CurrDevDescSz += (uint32_t)sizeof(USBD_DeviceDescTypedef);

    /* Check if USBx is in high speed mode to add qualifier descriptor */
    pDevQualDesc                     = (USBD_DevQualiDescTypedef *)(pDevFrameWorkDesc + pdev->CurrDevDescSz);
    pDevQualDesc->bLength            = (uint8_t)sizeof(USBD_DevQualiDescTypedef);
    pDevQualDesc->bDescriptorType    = UX_DEVICE_QUALIFIER_DESCRIPTOR_ITEM;
    pDevQualDesc->bcdDevice          = 0x0200;
    pDevQualDesc->Class              = 0x00;
    pDevQualDesc->SubClass           = 0x00;
    pDevQualDesc->Protocol           = 0x00;
    pDevQualDesc->bMaxPacketSize     = 0x40;
    pDevQualDesc->bNumConfigurations = 0x01;
    pDevQualDesc->bReserved          = 0x00;
    pdev->CurrDevDescSz += (uint32_t)sizeof(USBD_DevQualiDescTypedef);

    /* Build the device framework */
    while(Idx_Instance < USBD_MAX_SUPPORTED_CLASS)
    {
        if((pdev->classId < USBD_MAX_SUPPORTED_CLASS) && (pdev->NumClasses < USBD_MAX_SUPPORTED_CLASS) && (UserClassInstance[Idx_Instance] != CLASS_TYPE_NONE))
        {
            /* Call the composite class builder */
            (void)USBD_FrameWork_AddClass(pdev, (USBD_CompositeClassTypeDef)UserClassInstance[Idx_Instance], 0, Speed, (pDevFrameWorkDesc + pdev->CurrDevDescSz));

            /* Increment the ClassId for the next occurrence */
            pdev->classId++;
            pdev->NumClasses++;
        }

        Idx_Instance++;
    }

    /* Check if there is a composite class and update device class */
    if(pdev->NumClasses > 1)
    {
        pDevDesc->bDeviceClass    = 0xEF;
        pDevDesc->bDeviceSubClass = 0x02;
        pDevDesc->bDeviceProtocol = 0x01;
    }
    else
    {
        /* Check if the CDC ACM class is set and update device class */
        if(UserClassInstance[0] == CLASS_TYPE_CDC_ACM)
        {
            pDevDesc->bDeviceClass    = 0x02;
            pDevDesc->bDeviceSubClass = 0x02;
            pDevDesc->bDeviceProtocol = 0x00;
        }
    }

    return pDevFrameWorkDesc;
}

/**
  * @brief  USBD_FrameWork_AddToConfDesc
  *         Add a new class to the configuration descriptor
  * @param  pdev: device instance
  * @param  Speed: device speed
  * @param  pCmpstConfDesc: to composite device configuration descriptor
  * @retval status
  */
uint8_t USBD_FrameWork_AddToConfDesc(USBD_DevClassHandleTypeDef *pdev, uint8_t Speed, uint8_t *pCmpstConfDesc)
{
    uint8_t interface = 0U;

    pdev->Speed       = Speed;

    if(pdev->classId == 0U)
    {
        /* Add configuration and IAD descriptors */
        USBD_FrameWork_AddConfDesc((uint32_t)pCmpstConfDesc, &pdev->CurrConfDescSz);
    }

    switch(pdev->tclasslist[pdev->classId].ClassType)
    {
    case CLASS_TYPE_TMC:
        switch(pdev->tclasslist[pdev->classId].InterfaceType)
        {
        case INTERFACE_TMC_USB488:

            /* Find the first available interface slot and Assign number of interfaces */
            interface                              = USBD_FrameWork_FindFreeIFNbr(pdev);
            pdev->tclasslist[pdev->classId].NumIf  = 1U;
            pdev->tclasslist[pdev->classId].Ifs[0] = interface;

            /* Assign endpoint numbers */
            pdev->tclasslist[pdev->classId].NumEps = 2U; /* EP_IN, EP_OUT */

            /* Check the current speed to assign endpoints */
            if(pdev->Speed == USBD_HIGH_SPEED)
            {
                /* Assign IN Endpoint */
                USBD_FrameWork_AssignEp(pdev, USBD_TMC_NONE_EPIN_ADDR, USBD_EP_TYPE_BULK, USBD_TMC_NONE_EPIN_HS_MPS);

                /* Assign OUT Endpoint */
                USBD_FrameWork_AssignEp(pdev, USBD_TMC_NONE_EPOUT_ADDR, USBD_EP_TYPE_BULK, USBD_TMC_NONE_EPOUT_HS_MPS);
            }
            else
            {
                USBD_FrameWork_AssignEp(pdev, USBD_TMC_NONE_EPIN_ADDR, USBD_EP_TYPE_BULK, USBD_TMC_NONE_EPIN_FS_MPS);

                USBD_FrameWork_AssignEp(pdev, USBD_TMC_NONE_EPOUT_ADDR, USBD_EP_TYPE_BULK, USBD_TMC_NONE_EPOUT_FS_MPS);
            }

            USBD_FrameWork_TMC_Desc(pdev, (uint32_t)pCmpstConfDesc, &pdev->CurrConfDescSz);

            break;

        default:
            break;
        }
        break;

    default:
        break;
    }

    return UX_SUCCESS;
}

/**
  * @brief  USBD_FrameWork_AddClass
  *         Register a class in the class builder
  * @param  pdev: device instance
  * @param  class: type of the class to be added (from USBD_CompositeClassTypeDef)
  * @param  cfgidx: configuration index
  * @param  speed: device speed
  * @param  pCmpstConfDesc: to composite device configuration descriptor
  * @retval status
  */
uint8_t USBD_FrameWork_AddClass(USBD_DevClassHandleTypeDef *pdev, USBD_CompositeClassTypeDef class, uint8_t cfgidx, uint8_t Speed, uint8_t *pCmpstConfDesc)
{
    static uint8_t interface_idx = 0U;

    if((pdev->classId < USBD_MAX_SUPPORTED_CLASS) && (pdev->tclasslist[pdev->classId].Active == 0U))
    {
        /* Store the class parameters in the global tab */
        pdev->tclasslist[pdev->classId].ClassId   = pdev->classId;
        pdev->tclasslist[pdev->classId].Active    = 1U;
        pdev->tclasslist[pdev->classId].ClassType = class;

        if(class == CLASS_TYPE_TMC)
        {
            pdev->tclasslist[pdev->classId].InterfaceType = UserTMCInterface[interface_idx];

            interface_idx++;

            if(interface_idx == sizeof(UserTMCInterface))
            {
                interface_idx = 0U;
            }
        }

        /* Call configuration descriptor builder and endpoint configuration builder */
        if(USBD_FrameWork_AddToConfDesc(pdev, Speed, pCmpstConfDesc) != UX_SUCCESS)
        {
            return UX_ERROR;
        }
    }

    UNUSED(cfgidx);

    return UX_SUCCESS;
}

/**
  * @brief  USBD_FrameWork_FindFreeIFNbr
  *         Find the first interface available slot
  * @param  pdev: device instance
  * @retval The interface number to be used
  */
static uint8_t USBD_FrameWork_FindFreeIFNbr(USBD_DevClassHandleTypeDef *pdev)
{
    uint32_t idx = 0U;

    /* Unroll all already activated classes */
    for(uint32_t i = 0U; i < pdev->NumClasses; i++)
    {
        /* Unroll each class interfaces */
        for(uint32_t j = 0U; j < pdev->tclasslist[i].NumIf; j++)
        {
            /* Increment the interface counter index */
            idx++;
        }
    }

    /* Return the first available interface slot */
    return (uint8_t)idx;
}

/**
  * @brief  USBD_FrameWork_AddConfDesc
  *         Add a new class to the configuration descriptor
  * @param  Conf: configuration descriptor
  * @param  pSze: pointer to the configuration descriptor size
  * @retval none
  */
static void USBD_FrameWork_AddConfDesc(uint32_t Conf, uint32_t *pSze)
{
    /* Intermediate variable to comply with MISRA-C Rule 11.3 */
    USBD_ConfigDescTypedef *ptr = (USBD_ConfigDescTypedef *)Conf;

    ptr->bLength                = (uint8_t)sizeof(USBD_ConfigDescTypedef);
    ptr->bDescriptorType        = UX_CONFIGURATION_DESCRIPTOR_ITEM;
    ptr->wDescriptorLength      = 0x20;
    ptr->bNumInterfaces         = 1U;
    ptr->bConfigurationValue    = 1U;
    ptr->iConfiguration         = USBD_CONFIG_STR_DESC_IDX;
    ptr->bmAttributes           = USBD_CONFIG_BMATTRIBUTES;
    ptr->bMaxPower              = USBD_CONFIG_MAXPOWER;
    *pSze += sizeof(USBD_ConfigDescTypedef);
}

/**
  * @brief  USBD_FrameWork_AssignEp
  *         Assign and endpoint
  * @param  pdev: device instance
  * @param  Add: Endpoint address
  * @param  Type: Endpoint type
  * @param  Sze: Endpoint max packet size
  * @retval none
  */
static void USBD_FrameWork_AssignEp(USBD_DevClassHandleTypeDef *pdev, uint8_t Add, uint8_t Type, uint32_t Sze)
{
    uint32_t idx = 0U;

    /* Find the first available endpoint slot */
    while(((idx < (pdev->tclasslist[pdev->classId]).NumEps) && ((pdev->tclasslist[pdev->classId].Eps[idx].is_used) != 0U)))
    {
        /* Increment the index */
        idx++;
    }

    /* Configure the endpoint */
    pdev->tclasslist[pdev->classId].Eps[idx].add     = Add;
    pdev->tclasslist[pdev->classId].Eps[idx].type    = Type;
    pdev->tclasslist[pdev->classId].Eps[idx].size    = (uint16_t)Sze;
    pdev->tclasslist[pdev->classId].Eps[idx].is_used = 1U;
}

/**
  * @brief  USBD_FrameWork_TMC_Desc
  *         Configure and Append the TMC Descriptor
  * @param  pdev: device instance
  * @param  pConf: Configuration descriptor pointer
  * @param  Sze: pointer to the current configuration descriptor size
  * @retval None
  */
static void USBD_FrameWork_TMC_Desc(USBD_DevClassHandleTypeDef *pdev, uint32_t pConf, uint32_t *Sze)
{
    static USBD_IfDescTypedef *pIfDesc;
    static USBD_EpDescTypedef *pEpDesc;

    switch(pdev->tclasslist[pdev->classId].InterfaceType)
    {
    case INTERFACE_TMC_USB488:

        /* Append Interface descriptor to Configuration descriptor */
        __USBD_FRAMEWORK_SET_IF(pdev->tclasslist[pdev->classId].Ifs[0], 0U, (uint8_t)(pdev->tclasslist[pdev->classId].NumEps), UX_DEVICE_CLASS_TMC_CLASS, UX_DEVICE_SUB_CLASS_USB488_CLASS, INTERFACE_TMC_USB488, 0U);

        if(pdev->Speed == USBD_HIGH_SPEED)
        {
            /* Append Endpoint descriptor to Configuration descriptor */
            __USBD_FRAMEWORK_SET_EP(pdev->tclasslist[pdev->classId].Eps[0].add, USBD_EP_TYPE_BULK, (uint16_t)pdev->tclasslist[pdev->classId].Eps[0].size, USBD_TMC_NONE_EPIN_FS_BINTERVAL, USBD_TMC_NONE_EPIN_HS_BINTERVAL);

            __USBD_FRAMEWORK_SET_EP(pdev->tclasslist[pdev->classId].Eps[1].add, USBD_EP_TYPE_BULK, (uint16_t)pdev->tclasslist[pdev->classId].Eps[1].size, USBD_TMC_NONE_EPOUT_HS_BINTERVAL, USBD_TMC_NONE_EPOUT_FS_BINTERVAL);
        }
        else
        {
            /* Append Endpoint descriptor to Configuration descriptor */
            __USBD_FRAMEWORK_SET_EP(pdev->tclasslist[pdev->classId].Eps[0].add, USBD_EP_TYPE_BULK, (uint16_t)pdev->tclasslist[pdev->classId].Eps[0].size, USBD_TMC_NONE_EPIN_FS_BINTERVAL, USBD_TMC_NONE_EPIN_HS_BINTERVAL);

            __USBD_FRAMEWORK_SET_EP(pdev->tclasslist[pdev->classId].Eps[1].add, USBD_EP_TYPE_BULK, (uint16_t)pdev->tclasslist[pdev->classId].Eps[1].size, USBD_TMC_NONE_EPOUT_HS_BINTERVAL, USBD_TMC_NONE_EPOUT_FS_BINTERVAL);
        }

        break;

    default:
        break;
    }
}
/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file    ux_device_descriptors.h
  * @author  MCD Application Team
  * @brief   USBX Device descriptor header file
  ******************************************************************************
  * @attention
  *
  * Copyright (c) 2020-2021 STMicroelectronics.
  * All rights reserved.
  *
  * This software is licensed under terms that can be found in the LICENSE file
  * in the root directory of this software component.
  * If no LICENSE file comes with this software, it is provided AS-IS.
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Define to prevent recursive inclusion -------------------------------------*/
#ifndef __UX_DEVICE_DESCRIPTORS_H__
#define __UX_DEVICE_DESCRIPTORS_H__

#ifdef __cplusplus
extern "C"
{
#endif

/* Includes ------------------------------------------------------------------*/
#include "ux_api.h"
#include "ux_stm32_config.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "main.h"
/* USER CODE END Includes */

/* Private defines -----------------------------------------------------------*/
#define USBD_MAX_NUM_CONFIGURATION            1U
#define USBD_MAX_SUPPORTED_CLASS              3U
#define USBD_MAX_CLASS_ENDPOINTS              9U
#define USBD_MAX_CLASS_INTERFACES             12U

#define USBD_HID_CUSTOM_ACTIVATED             1U

#define USBD_CONFIG_MAXPOWER                  0x64 // bMaxPower: 200 mA (0x64 * 2 mA)
#define USBD_COMPOSITE_USE_IAD                1U
#define USBD_DEVICE_FRAMEWORK_BUILDER_ENABLED 1U

#define USBD_FRAMEWORK_MAX_DESC_SZ            200U
/* Exported types ------------------------------------------------------------*/
/* USER CODE BEGIN ET */

/* USER CODE END ET */

/* Enum Class Type */
typedef enum
{
    CLASS_TYPE_NONE     = 0,
    CLASS_TYPE_HID      = 1,
    CLASS_TYPE_CDC_ACM  = 2,
    CLASS_TYPE_MSC      = 3,
    CLASS_TYPE_CDC_ECM  = 4,
    CLASS_TYPE_DFU      = 5,
    CLASS_TYPE_VIDEO    = 6,
    CLASS_TYPE_PIMA_MTP = 7,
    CLASS_TYPE_CCID     = 8,
    CLASS_TYPE_PRINTER  = 9,
    CLASS_TYPE_RNDIS    = 10,
    CLASS_TYPE_AUDIO_10 = 11,
    CLASS_TYPE_AUDIO_20 = 12,
    CLASS_TYPE_TMC      = 40,
} USBD_CompositeClassTypeDef;

/* Enum NONE Interface Type */
typedef enum
{
    INTERFACE_NONE_DEFIN = 0
} USBD_NONEInterfaceTypeDef;

/* Enum HID Interface Type */
typedef enum
{
    INTERFACE_HID_CUSTOM   = 0,
    INTERFACE_HID_KEYBOARD = 1,
    INTERFACE_HID_MOUSE    = 2,
} USBD_HIDInterfaceTypeDef;

typedef enum
{
    INTERFACE_TMC_USB488 = 1,
} USBD_TMCInterfaceTypeDef;

/* USB Endpoint handle structure */
typedef struct
{
    uint32_t status;
    uint32_t total_length;
    uint32_t rem_length;
    uint32_t maxpacket;
    uint16_t is_used;
    uint16_t bInterval;
} USBD_EndpointTypeDef;

/* USB endpoint handle structure */
typedef struct
{
    uint8_t  add;
    uint8_t  type;
    uint16_t size;
    uint8_t  is_used;
} USBD_EPTypeDef;

/* USB Composite handle structure */
typedef struct
{
    USBD_CompositeClassTypeDef ClassType;
    uint32_t                   ClassId;
    uint8_t                    InterfaceType;
    uint32_t                   Active;
    uint32_t                   NumEps;
    uint32_t                   NumIf;
    USBD_EPTypeDef             Eps[USBD_MAX_CLASS_ENDPOINTS];
    uint8_t                    Ifs[USBD_MAX_CLASS_INTERFACES];
} USBD_CompositeElementTypeDef;

/* USB Device handle structure */
typedef struct _USBD_DevClassHandleTypeDef
{
    uint8_t                      Speed;
    uint32_t                     classId;
    uint32_t                     NumClasses;
    USBD_CompositeElementTypeDef tclasslist[USBD_MAX_SUPPORTED_CLASS];
    uint32_t                     CurrDevDescSz;
    uint32_t                     CurrConfDescSz;
} USBD_DevClassHandleTypeDef;

/* USB Device endpoint direction */
typedef enum
{
    OUT = 0x00,
    IN  = 0x80,
} USBD_EPDirectionTypeDef;

/* USB Device descriptors structure */
typedef struct
{
    uint8_t  bLength;
    uint8_t  bDescriptorType;
    uint16_t bcdUSB;
    uint8_t  bDeviceClass;
    uint8_t  bDeviceSubClass;
    uint8_t  bDeviceProtocol;
    uint8_t  bMaxPacketSize;
    uint16_t idVendor;
    uint16_t idProduct;
    uint16_t bcdDevice;
    uint8_t  iManufacturer;
    uint8_t  iProduct;
    uint8_t  iSerialNumber;
    uint8_t  bNumConfigurations;
} __PACKED USBD_DeviceDescTypedef;

/* USB Iad descriptors structure */
typedef struct
{
    uint8_t bLength;
    uint8_t bDescriptorType;
    uint8_t bFirstInterface;
    uint8_t bInterfaceCount;
    uint8_t bFunctionClass;
    uint8_t bFunctionSubClass;
    uint8_t bFunctionProtocol;
    uint8_t iFunction;
} __PACKED USBD_IadDescTypedef;

/* USB interface descriptors structure */
typedef struct
{
    uint8_t bLength;
    uint8_t bDescriptorType;
    uint8_t bInterfaceNumber;
    uint8_t bAlternateSetting;
    uint8_t bNumEndpoints;
    uint8_t bInterfaceClass;
    uint8_t bInterfaceSubClass;
    uint8_t bInterfaceProtocol;
    uint8_t iInterface;
} __PACKED USBD_IfDescTypedef;

/* USB endpoint descriptors structure */
typedef struct
{
    uint8_t  bLength;
    uint8_t  bDescriptorType;
    uint8_t  bEndpointAddress;
    uint8_t  bmAttributes;
    uint16_t wMaxPacketSize;
    uint8_t  bInterval;
} __PACKED USBD_EpDescTypedef;

/* USB Config descriptors structure */
typedef struct
{
    uint8_t  bLength;
    uint8_t  bDescriptorType;
    uint16_t wDescriptorLength;
    uint8_t  bNumInterfaces;
    uint8_t  bConfigurationValue;
    uint8_t  iConfiguration;
    uint8_t  bmAttributes;
    uint8_t  bMaxPower;
} __PACKED USBD_ConfigDescTypedef;

/* USB Qualifier descriptors structure */
typedef struct
{
    uint8_t  bLength;
    uint8_t  bDescriptorType;
    uint16_t bcdDevice;
    uint8_t  Class;
    uint8_t  SubClass;
    uint8_t  Protocol;
    uint8_t  bMaxPacketSize;
    uint8_t  bNumConfigurations;
    uint8_t  bReserved;
} __PACKED USBD_DevQualiDescTypedef;

typedef struct
{
    uint8_t  bLength;
    uint8_t  bDescriptorType;
    uint16_t bcdTMC;
    uint16_t bCountryCode;
    uint8_t  bNumDescriptors;
    uint8_t  bTMCDescriptorType;
    uint16_t wDescriptor;
} __PACKED USBD_TMCDescTypedef;
/* Exported functions prototypes ---------------------------------------------*/
/* USER CODE BEGIN EFP */

/* USER CODE END EFP */

uint8_t *USBD_Get_Device_Framework_Speed(uint8_t Speed, ULONG *Length);
uint8_t *USBD_Get_String_Framework(ULONG *Length);
uint8_t *USBD_Get_Language_Id_Framework(ULONG *Length);
uint16_t USBD_Get_Interface_Number(uint8_t class_type, uint8_t interface_type);
uint16_t USBD_Get_Configuration_Number(uint8_t class_type, uint8_t interface_type);

/* Private defines -----------------------------------------------------------*/
/* USER CODE BEGIN Private_defines */

/* USER CODE END Private_defines */

#define USBD_VID                         0x0A69
#define USBD_PID                         0x088A
#define USBD_LANGID_STRING               1033 /* English (United States) */
#define USBD_MANUFACTURER_STRING         "YONO"
#define USBD_PRODUCT_STRING              "yonoDevice"
#define USBD_SERIAL_NUMBER               "123456789"

#define USB_DESC_TYPE_INTERFACE          0x04U
#define USB_DESC_TYPE_ENDPOINT           0x05U

#define USBD_EP_TYPE_CTRL                0x00U
#define USBD_EP_TYPE_ISOC                0x01U
#define USBD_EP_TYPE_BULK                0x02U
#define USBD_EP_TYPE_INTR                0x03U

#define USBD_FULL_SPEED                  0x00U
#define USBD_HIGH_SPEED                  0x01U

#define USB_BCDUSB                       0x0200U
#define LANGUAGE_ID_MAX_LENGTH           2U

#define USBD_IDX_MFC_STR                 0x01U
#define USBD_IDX_PRODUCT_STR             0x02U
#define USBD_IDX_SERIAL_STR              0x03U

#define USBD_MAX_EP0_SIZE                64U
#define USBD_DEVICE_QUALIFIER_DESC_SIZE  0x0AU

#define USBD_STRING_FRAMEWORK_MAX_LENGTH 256U

/* */
#define UX_DEVICE_CLASS_TMC_CLASS        0xFEU
#define UX_DEVICE_SUB_CLASS_USB488_CLASS 0x03U

/* Device TMC none */
#define USBD_TMC_NONE_EPIN_ADDR          0x81U
#define USBD_TMC_NONE_EPOUT_ADDR         0x01U
#define USBD_TMC_NONE_EPIN_FS_MPS        0x40U
#define USBD_TMC_NONE_EPIN_HS_MPS        0x40U
#define USBD_TMC_NONE_EPIN_FS_BINTERVAL  5U
#define USBD_TMC_NONE_EPIN_HS_BINTERVAL  0U
#define USBD_TMC_NONE_EPOUT_FS_MPS       0x40U
#define USBD_TMC_NONE_EPOUT_HS_MPS       0x40U
#define USBD_TMC_NONE_EPOUT_FS_BINTERVAL 0U
#define USBD_TMC_NONE_EPOUT_HS_BINTERVAL 5U

#ifndef USBD_CONFIG_STR_DESC_IDX
#define USBD_CONFIG_STR_DESC_IDX 0x00
#endif /* USBD_CONFIG_STR_DESC_IDX */

#ifndef USBD_CONFIG_BMATTRIBUTES
#define USBD_CONFIG_BMATTRIBUTES 0x80
#endif /* USBD_CONFIG_BMATTRIBUTES */

/* Private macro -----------------------------------------------------------*/
/* USER CODE BEGIN Private_macro */

/* USER CODE END Private_macro */
#define __USBD_FRAMEWORK_SET_EP(epadd, eptype, epsize, HSinterval, FSinterval)        \
    do                                                                                \
    {                                                                                 \
        /* Append Endpoint descriptor to Configuration descriptor */                  \
        pEpDesc                   = ((USBD_EpDescTypedef *)((uint32_t)pConf + *Sze)); \
        pEpDesc->bLength          = (uint8_t)sizeof(USBD_EpDescTypedef);              \
        pEpDesc->bDescriptorType  = USB_DESC_TYPE_ENDPOINT;                           \
        pEpDesc->bEndpointAddress = (epadd);                                          \
        pEpDesc->bmAttributes     = (eptype);                                         \
        pEpDesc->wMaxPacketSize   = (epsize);                                         \
        if(pdev->Speed == USBD_HIGH_SPEED)                                            \
        {                                                                             \
            pEpDesc->bInterval = (HSinterval);                                        \
        }                                                                             \
        else                                                                          \
        {                                                                             \
            pEpDesc->bInterval = (FSinterval);                                        \
        }                                                                             \
        *Sze += (uint32_t)sizeof(USBD_EpDescTypedef);                                 \
    }                                                                                 \
    while(0)

#define __USBD_FRAMEWORK_SET_IF(ifnum, alt, eps, class, subclass, protocol, istring)    \
    do                                                                                  \
    {                                                                                   \
        /* Interface Descriptor */                                                      \
        pIfDesc                     = ((USBD_IfDescTypedef *)((uint32_t)pConf + *Sze)); \
        pIfDesc->bLength            = (uint8_t)sizeof(USBD_IfDescTypedef);              \
        pIfDesc->bDescriptorType    = USB_DESC_TYPE_INTERFACE;                          \
        pIfDesc->bInterfaceNumber   = (ifnum);                                          \
        pIfDesc->bAlternateSetting  = (alt);                                            \
        pIfDesc->bNumEndpoints      = (eps);                                            \
        pIfDesc->bInterfaceClass    = (class);                                          \
        pIfDesc->bInterfaceSubClass = (subclass);                                       \
        pIfDesc->bInterfaceProtocol = (protocol);                                       \
        pIfDesc->iInterface         = (istring);                                        \
        *Sze += (uint32_t)sizeof(USBD_IfDescTypedef);                                   \
    }                                                                                   \
    while(0)
#ifdef __cplusplus
}
#endif
#endif /* __UX_DEVICE_DESCRIPTORS_H__ */

Sending and Receiving in usbx Practice#

Regardless of the details, just look at the code. A total of 3 buffers are opened, which is necessary because the USB transmission line will occupy one buffer, one is needed to store the complete statement received, and one is needed for sending storage. It is worth noting that the _ux_device_class_dpump_tmc_read function, even after my modifications, is still a blocking function, so the TMC transmission and reception support in TMC_Engine() needs to occupy a separate thread.

The TMC_Dpump pointer will be bound and unbound in the activate and deactivate of the TMC_Device.

/********************************************************************************


 **** Copyright (C), 2024, Yuanlong Xu <[email protected]>    ****
 **** All rights reserved                                       ****

 ********************************************************************************
 * File Name     : ux_device_tmc.c
 * Author        : yono
 * Date          : 2025-01-07
 * Version       : 1.0
********************************************************************************/
/**************************************************************************/
/*
    TMC user parsing
*/

/* Includes ------------------------------------------------------------------*/
#include <stdint.h>
#include <stdlib.h>
#include <stdbool.h>
#include "ux_api.h"
#include "ux_device_class_dpump.h"
/* Private types -------------------------------------------------------------*/

/* Private variables ---------------------------------------------------------*/
#define UX_HOST_CLASS_DPUMP_PACKET_SIZE 255
extern UX_SLAVE_CLASS_DPUMP *TMC_Dpump;

/* Channel buffer */
uint8_t device_buffer[UX_HOST_CLASS_DPUMP_PACKET_SIZE];
/* Frame information */
uint8_t TransferEMO = 0; // Transmission end flag 0x00 continue transmission 0x01 transmission end
uint8_t TransferTag = 0; // Transmission token, copied during reply
/* Information buffer */
uint8_t  SCPI_Read[255];
uint32_t SCPI_Read_Length = 0;
uint8_t  SCPI_Write[255];
uint32_t SCPI_Write_Length = 0;

/* Private Constants ---------------------------------------------------------*/
/* Private macros ------------------------------------------------------------*/
/* Private function prototypes -----------------------------------------------*/
uint32_t    TMC_Write(uint8_t *data, uint32_t len);
extern UINT _ux_device_class_dpump_tmc_read(UX_SLAVE_CLASS_DPUMP *dpump, UCHAR *buffer, ULONG requested_length, ULONG *actual_length);
/* Private functions ---------------------------------------------------------*/

void TMC_Engine(void)
{
    UINT  status;
    ULONG actual_length;

    if(TMC_Dpump != UX_NULL)
    {
        /* Receive part */
        status = _ux_device_class_dpump_tmc_read(TMC_Dpump, device_buffer, UX_HOST_CLASS_DPUMP_PACKET_SIZE, &actual_length);
        if(status == UX_SUCCESS)
        {
            if((0xff ^ device_buffer[1] ^ device_buffer[2]) || device_buffer[3] != 0)
                return; // Bad message

            switch(device_buffer[0]) // Command
            {
            case 0x01:
                // Receiving branch
                ux_utility_memory_copy(&SCPI_Read[0 + SCPI_Read_Length],  // Compatible with segmented transmission, starting from the length of the previous transmission
                                       &device_buffer[12],                // Actual data starting position
                                       *(uint32_t *)(&device_buffer[4])); // Bytes 4~7 form the data length in little-endian
                SCPI_Read_Length += *(uint32_t *)(&device_buffer[4]);

                TransferEMO = device_buffer[8] & 0x01;

                if(TransferEMO)
                {
                    /* Loopback test */
                    TMC_Write(SCPI_Read, SCPI_Read_Length);
                    /* End compatibility */
                    // Transmission end triggers parsing
                    SCPI_Read_Length = 0;
                }
                else
                {
                    // Transmission not ended, do not process
                }
                break;

            case 0x02:
                TransferTag = device_buffer[1]; // Store token

                /* 0~3 are request and token, no need to change */
                *(uint32_t *)(&device_buffer[4]) = SCPI_Write_Length; // Bytes 4~7 form the data length in little-endian

                /* Fixed success data */
                device_buffer[8]  = 0x1; // Success reply
                device_buffer[9]  = 0x00;
                device_buffer[10] = 0x00;
                device_buffer[11] = 0x00;

                /* Fill in sending data */
                ux_utility_memory_copy(&device_buffer[12], SCPI_Write, SCPI_Write_Length);

                /* Fill 0x00; if originally 4-byte aligned, it will add 4 more zeros */
                for(uint32_t i = 12 + SCPI_Write_Length; i < ((12 + SCPI_Write_Length + 0x3) & ~0x3U); i++)
                {
                    device_buffer[i] = 0x00;
                }

                /* Request to send */
                status = _ux_device_class_dpump_write(TMC_Dpump, device_buffer, ((12 + SCPI_Write_Length + 0x3) & ~0x3U), &actual_length);

                /* Clear sending request */
                SCPI_Write_Length = 0;
                break;
            default:
                break;
            }
        }
    }
}

uint32_t TMC_Write(uint8_t *data, uint32_t len)
{
    if(sizeof(SCPI_Write) < (SCPI_Write_Length + len))
        return 1;

    ux_utility_memory_copy(&SCPI_Write[0 + SCPI_Write_Length], // Compatible with segmented filling
                           data,
                           len);
    SCPI_Write_Length += len;

    return 0;
}
/**************************************************************************/
/*                                                                        */
/*       Copyright (c) Microsoft Corporation. All rights reserved.        */
/*                                                                        */
/*       This software is licensed under the Microsoft Software License   */
/*       Terms for Microsoft Azure RTOS. Full text of the license can be  */
/*       found in the LICENSE file at https://aka.ms/AzureRTOS_EULA       */
/*       and in the root directory of this software.                      */
/*                                                                        */
/**************************************************************************/

/**************************************************************************/
/**************************************************************************/
/**                                                                       */
/** USBX Component                                                        */
/**                                                                       */
/**   Device DPUMP Class                                                  */
/**                                                                       */
/**************************************************************************/
/**************************************************************************/

#define UX_SOURCE_CODE

/* Include necessary system files.  */

#include "ux_api.h"
#include "ux_device_class_dpump.h"
#include "ux_device_stack.h"

/**
 * @brief A non-blocking version based on ux_device_class_dpump_read, suitable for TMC device reception
 */

/**************************************************************************/
/*                                                                        */
/*  FUNCTION                                               RELEASE        */
/*                                                                        */
/*    _ux_device_class_dpump_tmc_read                     PORTABLE C      */
/*                                                           6.1          */
/*  AUTHOR                                                                */
/*                                                                        */
/*    yono, yono233.cn                                                    */
/*                                                                        */
/*  DESCRIPTION                                                           */
/*                                                                        */
/*    This function reads from the DPUMP class of TMC,                    */
/*       non-blocking of function _ux_device_class_dpump_write.           */
/*                                                                        */
/*  INPUT                                                                 */
/*                                                                        */
/*    dpump                                   Address of dpump class      */
/*                                                instance                */
/*                                                                        */
/*  OUTPUT                                                                */
/*                                                                        */
/*    None                                                                */
/*                                                                        */
/*  CALLS                                                                 */
/*                                                                        */
/*    _ux_device_stack_transfer_request     Request transfer              */
/*    _ux_utility_memory_copy               Copy memory                   */
/*                                                                        */
/*  CALLED BY                                                             */
/*                                                                        */
/*    ThreadX                                                             */
/*                                                                        */
/*  RELEASE HISTORY                                                       */
/*                                                                        */
/*    DATE              NAME                      DESCRIPTION             */
/*                                                                        */
/*  05-19-2020     Chaoqiong Xiao           Initial Version 6.0           */
/*  09-30-2020     Chaoqiong Xiao           Modified comment(s),          */
/*                                            verified memset and memcpy  */
/*                                            cases,                      */
/*                                            resulting in version 6.1    */
/*  01-07-2025     yono                     Copy and modify to            */
/*                                                  non-blocking          */
/*                                                                        */
/**************************************************************************/
UINT _ux_device_class_dpump_tmc_read(UX_SLAVE_CLASS_DPUMP *dpump, UCHAR *buffer, ULONG requested_length, ULONG *actual_length)
{
    UX_SLAVE_ENDPOINT *endpoint;
    UX_SLAVE_DEVICE   *device;
    UX_SLAVE_TRANSFER *transfer_request;
    UINT               status;
    ULONG              local_requested_length;

    /* If trace is enabled, insert this event into the trace buffer.  */
    UX_TRACE_IN_LINE_INSERT(UX_TRACE_DEVICE_CLASS_DPUMP_READ, dpump, buffer, requested_length, 0, UX_TRACE_DEVICE_CLASS_EVENTS, 0, 0)

    /* Get the pointer to the device.  */
    device = &_ux_system_slave->ux_system_slave_device;

    /* As long as the device is in the CONFIGURED state.  */
    if(device->ux_slave_device_state != UX_DEVICE_CONFIGURED)
    {
        /* Error trap. */
        _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_CONFIGURATION_HANDLE_UNKNOWN);

        /* If trace is enabled, insert this event into the trace buffer.  */
        UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_CONFIGURATION_HANDLE_UNKNOWN, device, 0, 0, UX_TRACE_ERRORS, 0, 0)

        /* Cannot proceed with command, the interface is down.  */
        return (UX_CONFIGURATION_HANDLE_UNKNOWN);
    }

    /* Locate the OUT endpoint.  */
    endpoint = dpump->ux_slave_class_dpump_bulkout_endpoint;

    /* Check endpoint. If NULL, we have not yet received the proper SET_INTERFACE command.  */
    if(endpoint == UX_NULL)
    {
        /* Error trap. */
        _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_ENDPOINT_HANDLE_UNKNOWN);

        return (UX_ENDPOINT_HANDLE_UNKNOWN);
    }

    /* All DPUMP reading  are on the endpoint OUT, from the host.  */
    transfer_request = &endpoint->ux_slave_endpoint_transfer_request;

    /* Reset the actual length.  */
    *actual_length = 0;

    /* Set return status to SUCCESS to make certain compilers happy.  */
    status = UX_SUCCESS;

    /* Check if we need more transactions.  */
    while(device->ux_slave_device_state == UX_DEVICE_CONFIGURED && requested_length != 0)
    {
        /* Check if we have enough in the local buffer.  */
        if(requested_length > UX_SLAVE_REQUEST_DATA_MAX_LENGTH)

            /* We have too much to transfer.  */
            local_requested_length = UX_SLAVE_REQUEST_DATA_MAX_LENGTH;

        else

            /* We can proceed with the demanded length.  */
            local_requested_length = requested_length;

        /* Send the request to the device controller.  */
        status = _ux_device_stack_transfer_request(transfer_request, local_requested_length, local_requested_length);

        /* Check the status */
        if(status == UX_SUCCESS)
        {
            /* We need to copy the buffer locally.  */
            _ux_utility_memory_copy(buffer, transfer_request->ux_slave_transfer_request_data_pointer, local_requested_length); /* Use case of memcpy is verified. */

            /* Next buffer address.  */
            buffer += transfer_request->ux_slave_transfer_request_actual_length;

            /* Set the length actually received. */
            *actual_length += transfer_request->ux_slave_transfer_request_actual_length;

            /* Decrement what left has to be done.  */
            requested_length -= transfer_request->ux_slave_transfer_request_actual_length;

            /* actual_length > 0 */
            if(*actual_length > 0)
            {
                return UX_SUCCESS;
            }
        }
        else

            /* We got an error.  */
            return (status);
    }

    /* Check why we got here, either completion or device was extracted.  */
    if(device->ux_slave_device_state != UX_DEVICE_CONFIGURED)
    {
        /* Error trap. */
        _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_TRANSFER_NO_ANSWER);

        /* If trace is enabled, insert this event into the trace buffer.  */
        UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_TRANSFER_NO_ANSWER, transfer_request, 0, 0, UX_TRACE_ERRORS, 0, 0)

        /* Device must have been extracted.  */
        return (UX_TRANSFER_NO_ANSWER);
    }
    else

        /* Simply return the last transaction result.  */
        return (status);
}

Special Support for Setup in usbx Practice#

We know that to enable a device class in usbx, we must call the ux_device_stack_class_register() API, which is suitable for passing an _entry function as the class driver. According to the official examples and documentation of usbx, it is best to base custom class enumeration on the dpump class, so here we modify a TMC class driver based on _ux_device_class_dpump_entry.

The main modifications are needed in the **UX_SLAVE

Loading...
Ownership of this post data is guaranteed by blockchain and smart contracts to the creator alone.