STM32 USB数据接收与数据发送过程

news/2024/11/18 3:36:19/

      既然学习了USB,那就必须的搞懂USB设备与USB主机数据是怎么通讯的。这里主要讲设备端,因为我们的代码是做USB设备用的。

我们需要必须要定义了USB中断。起始在STM32的中断向量表中给USB两个中断,我们可以在stm32f10x.h中找到这两个中断:

 
  1. USB_HP_CAN1_TX_IRQn         = 19,   /*!< USB Device High Priority or CAN1 TX Interrupts  */

  2.   USB_LP_CAN1_RX0_IRQn        = 20,   /*!< USB Device Low Priority or CAN1 RX0 Interrupts  */

这两个中断是USB与CAN复用的中断,在做USB用时,表示USB设备的高优先级与低优先级中断。在我的工程中,我选择用低优先级的USB中断。代码如下:

 
  1. void USB_LP_CAN1_RX0_IRQHandler(void)

  2. {

  3.   USB_Istr();

  4. }

中断服务程序很简单,就是在发生中断的时候调用USB_istr()函数。USB_istr()这个函数我们之前说过的,在usb_istr.c中定义的。这个函数处理ISTR中断状态寄存器中定义的中断,包括:CTR正确传输中断、RESET复位中断,DOVR分组缓冲溢出中断、ERR错误中断、WAKEUP中断、SUSP挂起中断、SOF帧首中断、ESOF期望帧首中断。这里重点是CTR中断,在USB在正确发送或正确接收数据后,USB模块自动回将ISTR寄存器的该位置1,触发中断CTR中断。在USB_istr()中CTR的处理代码如下:

 
  1. #if (IMR_MSK & ISTR_CTR) //正确传输中断CTR标志

  2.   if (wIstr & ISTR_CTR & wInterrupt_Mask)//读出的中断标志是CRT中断标志,且CRT中断使能了

  3.   {

  4.     CTR_LP(); //调用正确传输中断服务程序

  5. #ifdef CTR_CALLBACK

  6.     CTR_Callback(); //当定义了CTR_CALLBACK,则调用CTR_Callback,像钩子函数一样,在发生CRT中断时做点什么

  7. #endif

  8.   }

首先要解释下 #if (IMR_MSK & ISTR_CTR) 这句话。

#define IMR_MSK (CNTR_CTRM | CNTR_WKUPM | CNTR_SUSPM | CNTR_ERRM | CNTR_SOFM \ | CNTR_ESOFM | CNTR_RESETM )

这是IMR_MSK的定义,表示包含所有中断的掩码,IMR_MSK & ISTR_CTR表示:如果ISTR_CTR是规定的中断类别,则编译#if与#endif之间的代码。很明显这里符合。然后,判断下从CNTR寄存器中读出来的中断值是CRT中断,且该中断已经在CNTR中使能了。接着调用CTR_LP()函数处理,如果定义了CTR_CALLBACK,则调用CTR_Callback()函数,该函数是个钩子函数,让用户在正确接收到数据后能够做些什么,比如说亮下灯或通过串口打印些消息。

这里需要着分析下CTR_LP()这个函数在usb_int.c中定义。代码如下:

 
  1. /*******************************************************************************

  2. * Function Name  : CTR_LP.

  3. * Description    : 低优先级的端点正确传输中断服务程序

  4. * Input          : None.

  5. * Output         : None.

  6. * Return         : None.

  7. *******************************************************************************/

  8. void CTR_LP(void)

  9. {

  10.   __IO uint16_t wEPVal = 0;

  11.   while (((wIstr = _GetISTR()) & ISTR_CTR) != 0) //读取中断状态寄存器的值,看是否是CRT(正确传输中断)

  12.   {

  13.     EPindex = (uint8_t)(wIstr & ISTR_EP_ID); //获取产生中断的端点号,

  14.     if (EPindex == 0) //如果端点0

  15.     {

  16.     SaveRState = _GetENDPOINT(ENDP0); //读取端点0的状态寄存器

  17.     SaveTState = SaveRState & EPTX_STAT; //保存端点0发送状态

  18.     SaveRState &=  EPRX_STAT; //保存端点0接收状态

  19.  
  20.     _SetEPRxTxStatus(ENDP0,EP_RX_NAK,EP_TX_NAK);//设置端点0对主机以NAK方式响应所有的接收和发送请求

  21.       if ((wIstr & ISTR_DIR) == 0) //如果是IN令牌

  22.       {

  23.         _ClearEP_CTR_TX(ENDP0); //清除端点0正确发送标志位

  24.         In0_Process(); //处理IN令牌包

  25.  
  26.            /* before terminate set Tx & Rx status */

  27.  
  28.             _SetEPRxTxStatus(ENDP0,SaveRState,SaveTState);//在传输之前设置端点0接收发送状态位

  29.   return;

  30.       }

  31.       else //OUT令牌

  32.       {

  33.         wEPVal = _GetENDPOINT(ENDP0); //获取端点0的端点寄存器的值

  34.         

  35.         if ((wEPVal &EP_SETUP) != 0) //SETUP分组传输完成标志位

  36.         {

  37.           _ClearEP_CTR_RX(ENDP0);  //清除端点0的接收标志位

  38.           Setup0_Process(); //端点0建立阶段的数据处理

  39.  
  40.       _SetEPRxTxStatus(ENDP0,SaveRState,SaveTState);//设置端点0阶接收发送标志位

  41.           return;

  42.         }

  43.  
  44.         else if ((wEPVal & EP_CTR_RX) != 0) //正确接收标志位

  45.         {

  46.           _ClearEP_CTR_RX(ENDP0); //清除端点0正确标志位

  47.           Out0_Process(); //处理OUT令牌包

  48.      

  49.      _SetEPRxTxStatus(ENDP0,SaveRState,SaveTState);//设置端点0的接收发送状态

  50.           return;

  51.         }

  52.       }

  53.     }/* if(EPindex == 0) */

  54.     else //如果非0端点

  55.     {

  56.       wEPVal = _GetENDPOINT(EPindex); //获取该端点的端点寄存器的值

  57.       if ((wEPVal & EP_CTR_RX) != 0) //正确接收标志

  58.       {

  59.         _ClearEP_CTR_RX(EPindex); //清除端点正确接收标志

  60.  
  61.         (*pEpInt_OUT[EPindex-1])(); //调用注册过的端点OUT处理函数

  62.  
  63.       } /* if((wEPVal & EP_CTR_RX) */

  64.  
  65.       if ((wEPVal & EP_CTR_TX) != 0) //正确发送标志

  66.       {

  67.         _ClearEP_CTR_TX(EPindex); //清除正确发送标志

  68.  
  69.         (*pEpInt_IN[EPindex-1])(); //调用注册过的端点IN处理函数

  70.       } /* if((wEPVal & EP_CTR_TX) != 0) */

  71.  
  72.     }/* if(EPindex == 0) else */

  73.  
  74.   }/* while(...) */

  75. }

这个函数首先会判断是否真的CTR中断,如果是,执行while()中的代码,用EPindex来保存产生中断的端点号。EPindex为0表示是端点0产生的中断,说明此时USB还处于枚举阶段。EPindex不为0,表示枚举已经成功了,USB处于正常工作状态。

在枚举阶段,SaveRState保存端点0寄存器的值,接着SaveTState = SaveRState & EPTX_STAT;和SaveRState &=  EPRX_STAT;这两句,SaveTState保存当前发送端点0的状态, SaveRState 保存当前接收端点的状态。接着设置接收端点0为NAk状态,发送端点0也设置成NAK状态,也就是说当主机发送任何数据,从机只以NAK回应,从机也只能发送NAK数据,即不允许在数据处理阶段进行数据通讯。然后判断是输入还是输出。如果是输入(注意这里的输入是相对于主机来说的)则清除端点寄存器的EP_CTR_TX标志位,并且调用IN令牌包处理函数In0_Process()(在usb_core.c中定义)。如果是输出(注意这里的输出是相对于主机来说的),则还要判断接收到是SETUP包还是OUT令牌包,如果是SETUP包,清除端点0寄存器的EP_SETUP位,并且调动SETUP处理函数Setup0_Process(),同时还要回复原来的接发端点的状态,准备处理下一次的中断处理。如果是OUT令牌包,清除端点0寄存器的EP_CRT_RX位,调用OUT处理函数Out0_Process(),同时还要回复原来接法端口的状态,准备处理下一次的中断处理。

在工作阶段或者说是非枚举阶段,首先要判断下是EP_CTR_RX还EP_CTR_TX标志,如果是EP_CTR_RX正确接收标志,则清除该标志,调用对应端点的OUT处理函

数(*pEpInt_OUT[EPindex-1])()(在usb_istr中有注册过),如果是EP_CTR_TX标志,则清除该标志,调用对应端点的IN处理函数(*pEpInt_IN[EPindex-1])()(在usb_istr中有注册过)。

在usb_istr.c中非别注册了7个端点输入函数和端点输出函数。如下:

 
  1. /*定义指向指针的函数指针数组,函数指针分别指向7个端点输入服务程序*/

  2. void (*pEpInt_IN[7])(void) =

  3.   {

  4.     EP1_IN_Callback,

  5.     EP2_IN_Callback,

  6.     EP3_IN_Callback,

  7.     EP4_IN_Callback,

  8.     EP5_IN_Callback,

  9.     EP6_IN_Callback,

  10.     EP7_IN_Callback,

  11.   };

  12.  
  13. /*定义指向指针的函数指针数组,函数指针分别指向7个端点输出服务程序*/

  14. void (*pEpInt_OUT[7])(void) =

  15.   {

  16.     EP1_OUT_Callback,

  17.     EP2_OUT_Callback,

  18.     EP3_OUT_Callback,

  19.     EP4_OUT_Callback,

  20.     EP5_OUT_Callback,

  21.     EP6_OUT_Callback,

  22.     EP7_OUT_Callback,

  23.   };

而这些函数的定义在usb_endp.c中,我们拿EP1_OUT_Callback()函数分析。

 
  1. /*******************************************************************************

  2. * Function Name  : EP1_OUT_Callback.

  3. * Description    : 端点1输出回调函数

  4. * Input          : None.

  5. * Output         : None.

  6. * Return         : None.

  7. *******************************************************************************/

  8. void EP1_OUT_Callback(void)

  9. {

  10. PMAToUserBufferCopy(USB_Receive_Buffer, ENDP1_RXADDR, REPORT_COUNT);  //PMA缓冲区接收到的数据拷贝到用户自定义缓冲区USB_Receive_Buffer中

  11.    SetEPRxStatus(ENDP1, EP_RX_VALID); //设置端点的接收状态为有效,因为端点接收到数据后会端点状态自动设置成停止状态

  12.   USB_Received_Flag=1; //设置接收到数据标志位

  13. }

这个函数的工作很简单,首先因为数输出端点,是接收数据的,而USB模块接收到的数据又是暂存在PAM双缓冲区中,所以要线把数据从PMA中读取出来,放到用户自己缓冲区中。接着设置端点接收状态有效,因为当接收数据后,端点就会被关闭。最后置位接收带数据标志。

以上就是USB设备的接收的流程。接下去讲讲发送流程。发送比接收简单多了看看下面的代码就知道了。

 
  1. /**

  2.   * @brief  通过USB发送数据

  3.   * @param  data 数据存储首地址

  4.   * @param  dataNum 发送的数据字节数

  5.   * @retval 发送的字节数

  6.   */

  7. uint32_t USB_SendData(uint8_t *data,uint32_t dataNum)  

  8. {

  9. //将数据通过USB发送出去

  10. UserToPMABufferCopy(data, ENDP2_TXADDR, dataNum);//拷贝数据到PMA中

  11. SetEPTxCount(ENDP2, REPORT_COUNT); //从端点2发送64字节数据

  12. SetEPTxValid(ENDP2);    //使能端点2的发送状态

  13. return dataNum;  

  14. }

把要发送的数据拷贝到PMA中,之后设置端点计数,使能下端点,数据就发送出去了。

总结下:

数据发送:UserToPMABufferCopy--->SetEPTxCount--->SetEPTxValid

数据接收:USB_LP_CAN1_RX0_IRQHandler--->USB_Istr---->CTR_LP--->EPx_OUT_Callback


http://www.ppmy.cn/news/131468.html

相关文章

STM32 USB CDC 虚拟多串口

转自: http://www.stmcu.org.cn/module/forum/thread-613510-1-1.html 楼主 发表于 2017-9-28 22:30:04 | 只看该作者 |只看大图 本帖最后由 creep 于 2017-9-28 22:37 编辑 之前一直使用STM32的CDC虚拟串口和上位机进行数据通信&#xff0c;通常只枚举一个串口基本…

STM32 USB复合设备实现

可实现用一个USB接口实现多个USB设备&#xff0c;如 HIDMSC&#xff1b; HIDCDC&#xff1b; HIDCDCMSC等等 使用HAL库&#xff0c;以HIDMSC为例 具体步骤&#xff1a; 文章目录 一、增加端点二、更改设备描述符三、HIDMSC内核配置1、新建结构体2、配置描述符3、实现函数 四、…

高通SDX55平台:adb功能异常

高通SDX55平台&#xff1a;adb功能异常 1. 问题描述2. 问题分析2.1 测试环境2.2 初步分析2.3 USB驱动初始化2.3.1 USB驱动加载流程2.3.1.1 USB_init初始化2.3.1.2 usb_hub_init2.3.1.3 usb设备插入后usbufs驱动加载 2.3.2 adb请求后usbfs设备驱动初始化流程 2.4 adb接口枚举流程…

STM32F750成功运行Linux

论坛发帖太分散了&#xff0c;在此记录一下我开发STM32F750 uClinux开发板的进程&#xff0c;这是第一篇&#xff0c;发表于2018年12月。 前段时间ST推出了Value Line的STM32F750和H750两个系列&#xff0c;看了一下选型表&#xff0c;F750有LQFP144封装&#xff0c;正好适合我…

【BK3633】规格书

目录&#xff1a; 1. 资源预览1.1 特性1.2 引脚 2 功能描述2.1 GPIO2.2 定时器2.2.1 PWM 定时器2.2.2 看门狗定时器和RTC定时器2.2.3 亚微秒事件定时器 2.3 ADC2.4 UART、I2C和SPI2.5 USB2.6 真随机数生成器2.7 I2S音频数字接口2.8 代码加密和系统安全2.9 到达角和离开角 3 电力…

【R】【课程笔记】06 金融波动模型

本文是课程《数据科学与金融计算》第6章的学习笔记&#xff0c;主要介绍GARCH类、SV类模型和高频波动模型&#xff0c;用于知识点总结和代码练习&#xff0c;Q&A为问题及解决方案。 往期回顾&#xff1a; 博文内容【R】【课程笔记】01 R软件基础知识数据类型、数据结构、…

matlab地心坐标系和GPS坐标系的转换

** 卑微小兰的第一篇&#xff1a;matlab地心坐标系和GPS坐标系的转换&#xff0c;emmm GPS坐标系和大地坐标系的正反变换 a6378137; f0.00669437999014;%第一偏心率平方 B40; L50; H60; Na/((1-f*(sind(B)))^0.5)%N为地心到该点的空间距离 X(NH)cosd(B)cosd(L); Y(NH)cosd(B)…

62 stm32 usb自定义hid复合设备修改实验

1.引言 最近因为项目需要&#xff0c;我们希望单片机既能有hid键盘功能&#xff0c;又能有hid设备的功能。即单片机的一个usb接口插入电脑后&#xff0c;电脑能识别出键盘设备和hid设备&#xff0c;两者同时存在的。 基于项目只是要求实现功能&#xff0c;故本次只是对stm32usb…