目录

银杏科技有限公司旗下技术文档发布平台
技术支持电话0379-69926675-801
技术支持邮件Gingko@vip.163.com
版本 日期 作者 修改内容
V1.0 2020-07-07 gingko 初次建立

实验十八:USB_VCP实验——虚拟串口

一、 实验目的与意义

  1. 了解STM32 USB VCP结构。
  2. 了解STM32 USB VCP特征。
  3. 掌握USB VCP的使用方法。
  4. 掌握STM32 HAL库中USB VCP属性的配置方法。
  5. 掌握KEIL MDK 集成开发环境使用方法。

二、 实验设备及平台

  1. iCore4 双核心板点击购买
  2. JLINK(或相同功能)仿真器点击购买
  3. Micro USB线缆。
  4. Keil MDK 开发平台。
  5. STM32CubeMX开发平台。
  6. 装有WIN XP(及更高版本)系统的计算机。

三、 实验原理

1、USB虚拟串口简介

2、USB CDC协议简介

3、CDC类软件框架

4、原理图

四、 实验程序

1、主函数

int main(void)
{
  /* MCU 配置 */
  /* 重置所有外围设备,初始化Flash接口和Systick. */
  HAL_Init();
  /* 系统时钟配置 */
  SystemClock_Config();
 /* 初始化所有已配置的外围设备 */
  MX_GPIO_Init();
  MX_USB_DEVICE_Init();
 
  LED_GREEN_ON;
  //接收发送处理均在usbd_cdc_if.c文件中
  while (1)
  {
  }
}

2、USB DEVICE初始化

void MX_USB_DEVICE_Init(void)
{
  /* 初始化设备库,添加支持的类并启动该库*/
  /* 初始化设备堆栈并加载类驱动程序*/
  USBD_Init(&hUsbDeviceHS, &HS_Desc, DEVICE_HS);/
  /* 将类驱动程序链接到设备核心*/
  USBD_RegisterClass(&hUsbDeviceHS, & USBD_Interface_fops_HS);
/* 启动USB设备核心 */
  USBD_Start(&hUsbDeviceHS);
}

3、CDC初始化函数介绍

/* 在USB HS IP上初始化CDC底层媒体 */
/* 如果所有操作均正常,则为USBD_OK,否则为USBD_FAIL  */
static int8_t CDC_Init_HS(void)
{
  /* 设置应用程序缓冲区 */
  USBD_CDC_SetTxBuffer(&hUsbDeviceHS, UserTxBufferHS, 0);
  USBD_CDC_SetRxBuffer(&hUsbDeviceHS, UserRxBufferHS);
  return (USBD_OK);
}
/* 取消初始化CDC媒体低层 */  
/* 如果所有操作均正常,则为USBD_OK,否则为USBD_FAIL */
static int8_t CDC_DeInit_HS(void)
{
  return (USBD_OK);
}

4、CDC类请求管理

/**
  * @brief  管理CDC类请求
  * @param  cmd: 指令码
  * @param  pbuf: 包含命令数据(请求参数)的缓冲区
  * @param  length:要发送的数据数(以字节为单位)
 * @retval 操作结果: 如果所有操作均正常,则为USBD_OK,否则为USBD_FAIL
  */
static int8_t CDC_Control_HS(uint8_t cmd, uint8_t* pbuf, uint16_t length)
{ 
  switch (cmd)
  {
  case CDC_SEND_ENCAPSULATED_COMMAND:
    break;
  case CDC_GET_ENCAPSULATED_RESPONSE:
    breakcase CDC_SET_COMM_FEATURE:
    breakcase CDC_GET_COMM_FEATURE:
    break;
  case CDC_CLEAR_COMM_FEATURE:
    break;   
  case CDC_SET_LINE_CODING:   
    break;
  case CDC_GET_LINE_CODING:     
    break;
  case CDC_SET_CONTROL_LINE_STATE:
    break;
  case CDC_SEND_BREAK:
    break;    
  default:
    break;
  }
  return (USBD_OK);
}

5、CDC接受数据

/**
  * @brief  USB OUT端点接收的数据通过此函数发送给CDC接口 
  * @note 此函数将阻止USB端点上的所有OUT数据包接收,直到退出此功能。如果在CDC接口上完成
            传输之前退出此功能(即使用DMA控制器),它将导致接收更多数据,而先前的数据仍未发送。
  * @param  Buf: 待接收数据的缓冲区
  * @param  Len: 接收的数据数(以字节为单位)
  * @retval 如果所有操作均正常,则为USBD_OK,否则为USBD_FAIL
  */
static int8_t CDC_Receive_HS (uint8_t* Buf, uint32_t *Len)
{
  USBD_CDC_SetRxBuffer(&hUsbDeviceHS, &Buf[0]);
  USBD_CDC_ReceivePacket(&hUsbDeviceHS);
  CDC_Transmit_HS(Buf,(uint16_t)(*Len));
  return (USBD_OK);
}

6、CDC发送数据

/**
* @brief  通过USB IN端点发送的数据通过此函数发送给CDC接口
* @param  Buf: 要发送的数据缓冲区
* @param  Len: 要发送的数据数(以字节为单位)
* @retval 操作结果: 如果所有操作均正常返回USBD_OK,否则返回USBD_FAIL或USBD_BUSY 
*/
uint8_t CDC_Transmit_HS(uint8_t* Buf, uint16_t Len)
{
  uint8_t result = USBD_OK; 
  USBD_CDC_HandleTypeDef *hcdc = (USBD_CDC_HandleTypeDef*)hUsbDeviceHS.pClassData;
 
  if (hcdc->TxState != 0){
    return USBD_BUSY;
  }
 
  USBD_CDC_SetTxBuffer(&hUsbDeviceHS, Buf, Len);
  result = USBD_CDC_TransmitPacket(&hUsbDeviceHS);
  return result;
}

五、 实验步骤

  1. 把仿真器与iCore4的SWD调试口相连(直接相连或者通过转接器相连);
  2. 将跳线帽插到USB_OTG;
  3. 把iCore4(USB_OTG)通过Micro USB线与计算机相连,为iCore4供电;
  4. 打开Keil MDK 开发环境,并打开本实验工程;
  5. 烧写程序到iCore4上;
  6. 打开Commix串口终端;
  7. 也可以进入Debug模式,单步运行或设置断点验证程序逻辑。

六、 实验现象