官方库例程:

..\STM32Cube\Repository\STM32Cube_FW_F4_V1.23.0\Projects\STM324xG_EVAL\Applications\USB_Device\CDC_Standalone\MDK-ARM

环境:

STM32CubeMX   STM32F429IGT6  STlink

首先要确保硬件电路USB部分没问题;USB相关的概念知识大概需要了解一下,网上挺多这类文章的,自行百度。

STM32CubeMX 配置:

点击USB_OTG_FS,模式选择Device_Only,其他保持默认。

STM32虚拟化串口识别USB设备_stm32

点击USB_DEVICE,选择IP 为VPC(虚拟串口),其他保持默认。

STM32虚拟化串口识别USB设备_STM32虚拟化串口识别USB设备_02

我使用的芯片是F429IGT6,最大时钟180MHz,但是USB时钟必须为48MHz(详情看STM32中文参考手册930页),180MHz是分频不出来48MHz的USB时钟,所以把系统配置成168MHz就能分频出48MHz的USB时钟。

STM32虚拟化串口识别USB设备_arm_03

STM32虚拟化串口识别USB设备_arm_04

堆空间需要改大一点,不然在USB插入电脑的时候,设备管理器会显示虚拟串口设备黄色感叹号。

因为在USB插入电脑,STM32会创建一个实例,malloc申请内存,但内存不足的时候就失败了,驱动工作不正常了。

造成这一现象的原因:源码在usbd_cdc.c中: pdev->pClassData = USBD_malloc(sizeof (USBD_CDC_HandleTypeDef));

如果pdev->pClassData = NULL 即出现设备黄色感叹号。

STM32虚拟化串口识别USB设备_数据_05

最后生成代码即可,打开工程。

STM32虚拟化串口识别USB设备_stm32_06

图中Application/User文件夹中多了几个文件:

usb_device.czhi只有一个USB设备函数初始化函数 MX_USB_DEVICE_Init()。

usb_conf.c是USB协议参数、IO初始化、中断回调函数、端点打开关闭停止操作等等函数。

usbd_cdc_if.c有虚拟串口的接收和发送等函数。

usb_desc.c有USB的描述符和USB枚举处理等。

文件夹Middlewares/USB_Device_Library是STM32Cube库。

检测VCP连接状态

在中断函数OTG_FS_IRQHandler
                         HAL_PCD_IRQHandler(&hpcd_USB_OTG_FS);
                                       HAL_PCD_ConnectCallback(hpcd);    //连接事件回调
                                       HAL_PCD_DisconnectCallback(hpcd);    //断开事件回调

发现两个连接状态事件回调函数,但是经过测试发现这两个回调函数根本不会发生。具体往下就没去深究了。

USBD_StatusTypeDef USBD_LL_DevDisconnected(USBD_HandleTypeDef  *pdev)
{
  /* Free Class Resources */
  pdev->dev_state = USBD_STATE_DEFAULT;
  pdev->pClass->DeInit(pdev, pdev->dev_config);  
  return USBD_OK;
}



void HAL_PCD_DisconnectCallback(PCD_HandleTypeDef *hpcd)
{
    USBD_LL_DevDisconnected((USBD_HandleTypeDef*)hpcd->pData);
}

但是偶然发现断开事件有改变一个变量pdev->dev_state = USBD_STATE_DEFAULT;

所以发现hUsbDeviceFS.dev_state的状态才是真正的连接状态标志位。

/*  Device Status */
#define USBD_STATE_DEFAULT                                1              //初始化状态
#define USBD_STATE_ADDRESSED                              2              //建立地址
#define USBD_STATE_CONFIGURED                             3              //配置完成,连接成功
#define USBD_STATE_SUSPENDED                              4              //usb挂起,断开成功

检测USB状态的函数

void VCP_Status(void)
{
    static uint8_t old_status = 0;

    if(hUsbDeviceFS.dev_state != old_status)
    {
        if(hUsbDeviceFS.dev_state == USBD_STATE_CONFIGURED)
            printf("连接成功\r\n");
        else if (hUsbDeviceFS.dev_state == USBD_STATE_SUSPENDED)
            printf("断开成功\r\n");
        old_status = hUsbDeviceFS.dev_state;
    }    
}

打印函数

写一个usb_printf打印函数,在usbd_cdc_if.c里面末尾USER CODE BEGIN及USER CODE END之间添加

/* USER CODE BEGIN PRIVATE_FUNCTIONS_IMPLEMENTATION */

#include <stdarg.h>

void usb_printf(const char *format, ...)
{
    va_list args;
    uint32_t length;
 
    va_start(args, format);
    length = vsnprintf((char *)UserTxBufferFS, APP_TX_DATA_SIZE, (char *)format, args);
    va_end(args);
    CDC_Transmit_FS(UserTxBufferFS, length);
}

/* USER CODE END PRIVATE_FUNCTIONS_IMPLEMENTATION */

关于接收有几点注意事项,认真看看。

1、usb虚拟串口每次接收最大的数据包ReceivePacket是64个字节;且每包数据以末尾追加  \r\n 表示一包数据接收完整。

当包长小于64个字节的时候

修改USB接收打印函数,在usbd_cdc_if.c里先找到static int8_t CDC_Receive_FS(uint8_t* Buf, uint32_t *Len)这个函数,

把它改成下图类似:

PC的usb虚拟串口收到MCU的数据通过UARTx发送回PC。演示一下发现什么问题:

//Len是每包数据的有效数据长度。值不会超过64
static int8_t CDC_Receive_FS(uint8_t* Buf, uint32_t *Len)
{
    USBD_CDC_ReceivePacket(&hUsbDeviceFS);
    printf("%s\r\n", Buf);
    memset(Buf, 0, APP_RX_DATA_SIZE);
    return (USBD_OK);
}

STM32虚拟化串口识别USB设备_数据_07

可以看到第一次发送内容  01                  >>>     MUC收到的数据是:01 81 7F 04 0D 0A 。数据异常,多了中间的 81 7F 04。

可以看到第二次发送内容  01 02 03 04   >>>     MUC收到的数据是:01 02 03 04 0D 0A 。数据正常。

注意static int8_t CDC_Receive_FS(uint8_t* Buf, uint32_t *Len)的  *Len 是每包数据真实有效长度;

第一次发送的内容,*Len是等于1的;

第二次发送的内容,*Len是等于4的;

*Len最大不会超过64。

出现第一种情况的原因是:当包长小于64个字节的时候,数据会强制32bit对齐。也就是发的字节必须要是4的倍数,不够补够4的倍数,最后末尾加上\r\n表示包的完整。

为什么会32位对齐,下面是STM32的源码:

typedef struct
 {
   uint32_t data[CDC_DATA_HS_MAX_PACKET_SIZE/4];      /* Force 32bits alignment */
   ...
   ...
 }


USBD_CDC_HandleTypeDef;

当包长大于64个字节的时候

STM32虚拟化串口识别USB设备_STM32虚拟化串口识别USB设备_08

发现一共发送了306个字节。但是MCU收到了318个字节。318怎么来的?

其实每64个字节加\r\n2个字节,最后一包数据不够4倍数再补充够4的倍数。

306 / 64 = 4......50
4 * 13 >= 50
也就是4 * (64 + \r\n) + 52个字节 + \r\n = 318。

小结一下:

1、当数据小于等于64字节,那么进行4字节对齐,故真实长度以(*Len)为准。评论说加'\0'其实是使用%s打印知道结束符。

2、当数据大于64字节,由于VCP每包数据最大只能接收64字节,超过64字节部分会重新清空刷新缓存区接收。故需新建用户缓存所有数据,直至收到数据中的标记位才表示该帧数据结束。过程中的(*Len)数据长度也要累加起来。

2019.07.24改进,上传的源码没更新。

下面这个函数作用是避免我们每次下载复位后需要拨出USB再插上才能用。如果不行的话可以把HAL_Delay()延时加大一些

原理:和usb硬件相关。PC的usb内部两根数据线都接着下拉电阻,当检测任一个任一根数据线有高电平代表有设备接入初始化。下面代码就是模拟,上电把两个STM32的USB IO拉低,相当于手动断开USB线,然后进行MX_USB_DEVICE_Init()初始化的时候会正确初始化这两个IO。

static void USB_Status_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStruct = {0};

    /* GPIO Ports Clock Enable */
    __HAL_RCC_GPIOA_CLK_ENABLE();

    /*Configure GPIO pin Output Level */
    HAL_GPIO_WritePin(GPIOA, GPIO_PIN_11 | GPIO_PIN_12, GPIO_PIN_RESET);

    /*Configure GPIO pin : W25Q256_CS_Pin */
    GPIO_InitStruct.Pin = GPIO_PIN_11 | GPIO_PIN_12;
    GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
    GPIO_InitStruct.Pull = GPIO_PULLDOWN;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

    //假如不行的话,下面的延时加长即可。
    HAL_Delay(2);
}

使用方法:

在main()函数中的 SystemClock_Config()函数调用后使用USB_Status_Init();

STM32虚拟化串口识别USB设备_STM32虚拟化串口识别USB设备_09

VCP通讯走的是pack,故和串口调试助手的波特率无关。无论波特率多少,对数据无影响。

接收函数:(更新2020.10.09)

故修改代码:以接收到\r\n为结束

typedef struct{
  uint32_t rxlen;
  uint32_t flag;
}VcpRx_t;

VcpRx_t temp = {
    .rxlen =0,
    .flag = 0
};

static int8_t CDC_Receive_FS(uint8_t* Buf, uint32_t *Len)
{
  /* USER CODE BEGIN 6 */   
    temp.rxlen = temp.rxlen + (*Len);
    
    if(temp.rxlen < APP_RX_DATA_SIZE && UserRxBufferFS[temp.rxlen - 2] != 0x0d  \
        &&  UserRxBufferFS[temp.rxlen - 1] != 0x0a)
    {
       //---继续接收---------------
       USBD_CDC_SetRxBuffer(&hUsbDeviceFS,UserRxBufferFS  + temp.rxlen); 
       USBD_CDC_ReceivePacket(&hUsbDeviceFS);
    }
    else 
    {
       temp.flag = 1;    //接收完成
       UserRxBufferFS[temp.rxlen] = 0;   //20201009新增,添加/0,方便打印
    }

    return (USBD_OK);
  /* USER CODE END 6 */
}




//接收完成后通过UART打印出来
void rxdata_printf(void)
{
  if(temp.flag)
  {
    printf("%s\r\n", UserRxBufferFS);
    temp.flag = 0;
    temp.rxlen = 0;
    memset(UserRxBufferFS, 0, APP_RX_DATA_SIZE);
    USBD_CDC_SetRxBuffer(&hUsbDeviceFS, UserRxBufferFS); 
    USBD_CDC_ReceivePacket(&hUsbDeviceFS);
  }
}

main函数

int main(void)
{
    HAL_Init();

    SystemClock_Config();

    MX_GPIO_Init();
    MX_USB_DEVICE_Init();
    MX_USART2_UART_Init();
    MX_USART1_UART_Init();
    /* USER CODE BEGIN 2 */
    printf("sudaroot\r\n");

    while (1)
    {
        rxdata_printf();       
    }
}