STM32学习笔记之USB数据接收和发送流程分析

发布者:lqs1975最新更新时间:2018-09-20 来源: 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数据接收和发送流程分析

上一篇:STM32CubeMX教程之简介及基本使用
下一篇:STM32例程之USB HID双向数据传输

推荐阅读最新更新时间:2024-03-16 16:14

Nordic nRF52系列低端器件又添新产品,支持全速(12 Mbps) USB运作
Nordic Semiconductor宣布推出其广受欢迎且经过市场验证的nRF52系列中的第六款产品nRF52820蓝牙5.2系统级芯片(SoC)。这款器件是功耗超低的低功耗蓝牙 (Bluetooth® Low Energy /Bluetooth LE)、蓝牙mesh、Thread、Zigbee和2.4 GHz专有低端无线连接解决方案。nRF52820具有蓝牙5、5.1和5.2的全部功能,包括长距离传输和2 Mbps高吞吐量、寻向、低功耗功率控制和低功耗同步通道。该器件还包括一个全速(12 Mbps) USB 2.0接口。 nRF52820具有功能强大的64MHz 32位Arm®Cortex®-M4处理器,并具有256
[网络通信]
Nordic nRF52系列低端器件又添新产品,支持全速(12 Mbps) <font color='red'>USB</font>运作
STM32+果云GA6-GPRS/GSM模块+MQTT+HTTP协议连接中移OneNet上传GPS数据定位
一、环境介绍 MCU: STM32F103C8T6 GSM模块: GA6--果云 开发软件: Keil5 完整源码下载: https://download.csdn.net/download/xiaolong1126626497/18245590 其他参考文章: STM32+ESP8266使用MQTT协议连接阿里云物联网服务器 其他参考文章:STM32+ESP8266使用标准MQTT协议(MQTTS)连接中国移动OneNet物联网服务器 二、GA6-GSM模块介绍与调试 说明: GA6-B模块供电必须5V,采用电脑USB供电可能不稳定(没有5V,只有4.8V左右),导致模块使用不稳定,发送AT指令没有反应,
[单片机]
STM32+果云GA6-GPRS/GSM模块+MQTT+HTTP协议连接中移OneNet上传GPS<font color='red'>数据</font>定位
移植STM32固件库到IAR
作为嵌入式软件工程师应该有良好的习惯,建project应该层次清晰。 1. 建立一个文件夹用于存放工程,命名为GPIO 2. 打开GPIO文件夹,创建一个子文件夹project(用于存放与工程相关的文件),再创建一个readme.txt(用于对工程的说明,比如说硬件环境和功能) 3. 将ST公司提供的固件库:FWLb中library拷贝到GPIO目录中 4. 打开project,再创建一个文件夹EWARM(存放和开发环境密切相关的内容) 5. 打开STM32固件库:STM32F10xFWLib\FWLib\examples\GPIO\Example1(自己到ST官网上下载) 6. 复制
[单片机]
移植<font color='red'>STM32</font>固件库到IAR
STM32】HAL库 STM32CubeMX教程十一---DMA (串口DMA发送接收)
前言: 本系列教程将 对应外设原理,HAL库与STM32CubeMX结合在一起讲解,使您可以更快速的学会各个模块的使用 所用工具: 1、芯片: STM32F407ZET6/ STM32F103ZET6 2、STM32CubeMx软件 3、IDE: MDK-Keil软件 4、STM32F1xx/STM32F4xxHAL库 知识概括: 通过本篇博客您将学到: DMA工作原理 STM32CubeMX创建DMA例程 HAL库定时器DMA函数库 注意:关于cubemx的DMA配置,在DMA原理介绍中全部都有所讲解,如果有哪里不懂,请仔细阅读原理详解部分。 DMA的基本介绍 什么是DMA (DM
[单片机]
【<font color='red'>STM32</font>】HAL库 STM32CubeMX教程十一---DMA (串口DMA<font color='red'>发送</font><font color='red'>接收</font>)
基于WinCE操作系统的通用USB数控键盘设计
引言 随着网络时代的到来,机械制造行业也面临着如何适应网络化制造的问题。于是将计算机技术、网络技术和传统的控制技术相结合,以嵌入式系统为主的数控系统正日益成为机械制造领域的热点。对于一个具体的数控系统,在通过键盘作为人机交互工具时,按键的数目与功能上与其他的数控系统是不同的,实时性和高速率是工业现场所重视的,USB数控键盘相对于传统的PS/2键盘具有高速率、支持热插拔、可灵活配置等特点,所以要为嵌入式数控设备研发出通用的数控键盘,来满足现代化数控系统的需求。具体设计方案如图1所示。 图1 系统框图 系统硬件电路设计 系统(图1)有四个模块组成:键盘模块、USB接口模块、USB和MCU通信模块。USB数控键盘不
[单片机]
基于WinCE操作系统的通用<font color='red'>USB</font>数控键盘设计
STM32综合网上讲解的几种读保护措施
以下内容是在STM32F103系列单片机实验成功! STM32可以对存储在flash上的程序进行读保护. 启动读保护后,用户就不能再读写程序了. 所以,在烧写程序之前,需要程序调用关闭读保护.关闭读保护后,会自动清空flash上的程序 头文件位于:#include stm32f10x_flash.h 启动保护,用在main()函数初始化时调用: void Set_Protect(void) //启动保护 {   if(FLASH_GetReadOutProtectionStatus() != SET)   {     FLASH_Unlock(); //解锁     FLASH_ReadOutProtection(ENABLE
[单片机]
<font color='red'>STM32</font>综合网上讲解的几种读保护措施
恩智浦半导体(NXP)宣布将并购GloNav
中国,北京,2007 年 12月21日 –恩智浦半导体宣布将对GloNav 公司进行并购。GloNav公司是一家美国的无晶圆半导体公司,为全球定位系统(GPS)及其它卫星导航系统提供单芯片解决方案。恩智浦将为此次并购支付8,500万美元,并视情况追加高达2,500万美元现金,这将取决于今后两年GloNav的营业收入与产品发展状况。通过此次并购恩智浦可快速进军已被广泛认可的GPS产品与技术市场。本次并购预计会在 2008 年第一季度完成,具体时间取决于相关监管机构的批准。 恩智浦半导体首席执行官万豪敦表示:“这是恩智浦今年以来为加强手机与个人移动通信事业部完成的第二起重要收购,这使得互补性技术快速地增加到我们现有的产品线中,满足客
[焦点新闻]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved