您现在的位置: 主页 > MCU > 单片机技术应用 > STM32 USB数据接收与数据发送程序流程分析 -
本文所属标签:
为本文创立个标签吧:

STM32 USB数据接收与数据发送程序流程分析 -

来源: 网络用户发布,如有版权联系网管删除 2018-09-07 

[导读]既然学习了USB,那就必须的搞懂USB设备与USB主机数据是怎么通讯的。这里主要讲设备端,因为我们的代码是做USB设备用的。我们需要必须要定义了USB中断。起始在STM32的中断向量表中给USB两个中断,我们可以在stm32f10x

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

本文引用地址: http://www.21ic.com/app/mcu/201808/784113.htm

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

USB_HP_CAN1_TX_IRQn=19,/*!

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

voidUSB_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();//调用正确传输中断服务程序#ifdefCTR_CALLBACKCTR_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中定义。代码如下:

/********************************************************************************FunctionName:CTR_LP.*Description:低优先级的端点正确传输中断服务程序*Input:None.*Output:None.*Return:None.*******************************************************************************/voidCTR_LP(void){__IOuint16_twEPVal=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令牌包/*beforeterminatesetTx&Rxstatus*/_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;}elseif((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)*
			

              查看评论 回复



嵌入式交流网主页 > MCU > 单片机技术应用 > STM32 USB数据接收与数据发送程序流程分析 -
 

"STM32 USB数据接收与数据发送程序流程分析 -"的相关文章

围观()