历史上的今天

今天是:2024年10月30日(星期三)

正在发生

2020年10月30日 | STM32 USB数据接收与数据发送程序流程分析

发布者:InnovateMind 来源: eefocus关键字:STM32  USB  数据接收  数据发送 手机看文章 扫描二维码
随时随地手机看文章

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


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

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

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


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

void USB_LP_CAN1_RX0_IRQHandler(void)

{

  USB_Istr();

}


中断服务程序很简单,就是在发生中断的时候调用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的处理代码如下:

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

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

  {

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

#ifdef CTR_CALLBACK

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

#endif

  }


首先要解释下  #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中定义。代码如下:

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

* Function Name  : CTR_LP.

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

* Input          : None.

* Output         : None.

* Return         : None.

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

void CTR_LP(void)

{

  __IO uint16_t wEPVal = 0;

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

  {

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

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

    {

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

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

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


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

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

      {

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

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


           /* before terminate set Tx & Rx status */


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

  return;

      }

      else //OUT令牌

      {

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

        

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

        {

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

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


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

          return;

        }


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

        {

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

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

     

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

          return;

        }

      }

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

    else //如果非0端点

    {

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

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

      {

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


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


      } /* if((wEPVal & EP_CTR_RX) */


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

      {

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


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

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


    }/* if(EPindex == 0) else */


  }/* while(...) */

}


  

这个函数首先会判断是否真的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个端点输入函数和端点输出函数。如下:

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

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

  {

    EP1_IN_Callback,

    EP2_IN_Callback,

    EP3_IN_Callback,

    EP4_IN_Callback,

    EP5_IN_Callback,

    EP6_IN_Callback,

    EP7_IN_Callback,

  };


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

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

  {

    EP1_OUT_Callback,

    EP2_OUT_Callback,

    EP3_OUT_Callback,

    EP4_OUT_Callback,

    EP5_OUT_Callback,

    EP6_OUT_Callback,

    EP7_OUT_Callback,

  };


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

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

* Function Name  : EP1_OUT_Callback.

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

* Input          : None.

* Output         : None.

* Return         : None.

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

void EP1_OUT_Callback(void)

{

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

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

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

}


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


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

/**

  * @brief  通过USB发送数据

  * @param  data 数据存储首地址

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

  * @retval 发送的字节数

  */

uint32_t USB_SendData(uint8_t *data,uint32_t dataNum)  

{

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

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

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

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

return dataNum;  

}


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



总结下:

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

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


关键字:STM32  USB  数据接收  数据发送 引用地址:STM32 USB数据接收与数据发送程序流程分析

上一篇:STM32(MDK)中不能使用printf()函数的问题
下一篇:STM32 usb_core.c分析

推荐阅读

行业领先的数字图像解决方案开发商豪威科技公司(OmniVision Technologies, Inc.)今日宣布推出基于PureCel®Plus堆栈芯片架构技术的第二代0.9微米像素、1/2.83英寸光学格式的2400万像素(MP)图像传感器。OV24B 传感器系列产品专为高分辨率智能手机的前置和后置摄像头而设计,具备尖端技术和增强设计,实现了差异化功能和卓越的图像性能。 图:OV24B...
1周时间,我从一个没用过STC单片机,不知道什么叫SPI接口的“文盲”,把nRF24L01的整个通信过程弄到完全没有bug.。兴奋之余来小屁一下。给那些正在奋斗着这个牛逼的芯片的小牛们小炫一下。希望有所帮助。屁话少说。正题:基本的东西我理解了,那就是:1.用5根线的SPI接口向2401发送数据或指令。2.芯片在每次上电的时候都需要进行一番配置。这些配置数据,...
自 iPhone 12 和 iPhone 12 Pro 上市以来,便有不少用户反映新设备的屏幕偏黄,虽然目前尚不知道问题的根源,但从网上的反馈来看,这两款型号或多或少都存在这样的问题。以下反馈来自知乎匿名用户,中间是 iPhone 12,旁边是 xr 和 11。仔细观看可以发现 iPhone 12 的上面偏红,下面就没有。明显发黄,关了原彩模式还是发黄。锋友@shaw911...
本周,A股三大指数出现分化行情,其中沪指、深证成指均有不同程度的跌幅,创业板指数则持续上涨。截至本周五收盘,沪指本周下跌35.26点,跌幅为0.98%,收报3547.34点;深证成指跌41.44点,跌幅为0.29%,收报14451.38点;创业板指数涨65.76点,涨幅为2%,收报3350.67点。Wind半导体指数本周掀起反攻潮。截至周五收盘,本周大涨322.09点,涨幅为4.49%,收报7...

史海拾趣

问答坊 | AI 解惑

NRF24L01的mega16程序

我的NRF24L01程序,已经调试通过,很实用!…

查看全部问答∨

LPCXpresso文件管理的问题

用LPCXpresso一段时间,不知道怎么进行文件分类管理,希望能像MDK那样可以分别吧不同的文件放到不同文件夹。现在这样太乱了。希望高手们解答,在此先谢过了。 下面来张它们的比较图: [ 本帖最后由 zhaojun_xf 于 2010-5-15 07:05 编辑 ]…

查看全部问答∨

AT89C2051P1口引脚的问题

我写的程序在keil上仿真调试时候, 而p1的0位和1位还是1的时候,p1口下面的ins的0位和1位为什么变成0, 菜鸟不知道表达的清楚不?…

查看全部问答∨

高手帮帮忙

我想在wince 5.0环境下开发数据库要怎么实现啊?用VS2005编,哪位高手帮帮忙。。。谢谢!…

查看全部问答∨

wince lcd 驱动

pxa270 wince 6.0 lcd是nec的NL2432HC22-41B。需要使用spi 在给lcd发送 power on命令的时候有个data input start 这个是做什么用的?应该发送什么data呢? …

查看全部问答∨

wince 下evc连数据库问题

我 在网上下了一个连数据库的例子,同时也把相应的数据库拷到了模拟器的My Documents目录下,编译的时候没有错误,不知道怎么在我的机器上却总是说文件拷贝失败,数据库创建失败? m_strDataSource = "Provider = Microsoft.SQLSERVER.OLEDB.CE. ...…

查看全部问答∨

ov9650显示问题

本人最近在做ov9650的驱动,平台为wince5.0 三星2450 现在的问题是为什么camera在显示图像的时候,LCD出现了两个窗口显示图像,而且图像显示不清楚…

查看全部问答∨

谁有keil下的lcd仿真啊?给我个啊,谢了

谁有keil下的lcd仿真啊?给我个啊,谢了 我下载了个。里面都是空的,name,lcd manufactuer  lcd  type  都为空…

查看全部问答∨

学习arm和dsp编程,用什么软件好

我想学习ARM和DSP编程,请问用什么软件较好?我大概知道DSP用CCS,ARM用SDT或ADS,我只想进行软件仿真,不连接硬件,哪款软件比较合适?谢谢!…

查看全部问答∨

7281 急

我用的是S3C2410的板子,用GPIO控制DATA,CLK和KEY,写入数据时,BC7281B时钟信号正常,就是data脚总是高电平,握手无法建立,查看以前的帖子有人问过但没有解决,不知到有没有人遇到同样的问题,怎么解决?很郁闷,多谢了!!!!…

查看全部问答∨
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved