既然学习了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(MDK)中不能使用printf()函数的问题
下一篇:STM32 usb_core.c分析
推荐阅读
史海拾趣
在电子行业,尤其是工业自动化与控制领域,Foxboro I C T Inc(现为施耐德电气旗下的一部分,通常简称为Foxboro)的发展充满了创新与挑战。以下是五个与Foxboro在电子行业里发展起来相关的故事,每个故事均基于事实进行描述:
1. Foxboro的起源与早期创新
Foxboro的历史可以追溯到1908年,在美国马萨诸塞州的福克斯伯勒,E.H.和B.B.兄弟联手开设了商店,并逐渐发展成为一家专注于工业控制自动化的公司。1920年代,Foxboro推出了其第一款Model 10 Stabilog比例加复位控制器,这标志着公司在工业自动化领域的初步探索。此后,Foxboro持续在控制技术上取得突破,如1950年代发布的全尺寸图形面板仪表和首款磁流量计,这些创新为公司在工业控制领域奠定了坚实的基础。(参考来源:Foxboro官网及相关历史资料)
2. 分布式控制系统的先驱
进入1970年代,Foxboro推出了SPEC 200系统,这是全球第一款真正的分布式控制系统(DCS)。该系统通过分散控制、集中管理的理念,极大地提高了工业生产的效率和可靠性。SPEC 200系统的成功推出,不仅巩固了Foxboro在工业自动化领域的领先地位,也推动了整个行业向更加智能化、自动化方向发展。(参考来源:知乎专栏及相关行业报告)
3. 进入中国市场,引领自动化发展
1983年,在中美贸易合作的大背景下,Foxboro与中国上海仪电控股(集团)公司共同投资成立了上海福克斯波罗有限公司(即Foxboro中国)。作为最早的中美合资企业之一,Foxboro中国迅速将最新技术和产品引入中国,推动了中国过程自动化行业的蓬勃发展。在成立初期,Foxboro中国就完成了最新技术和产品向中国本土的转移,所生产的成套系统装置广泛应用于石油、化工、电力等多个行业。(参考来源:国际能源网及相关报道)
4. ISO 9001国际质量认证的历史性突破
1991年12月,Foxboro中国成为中国首个获得ISO 9001国际质量合格证书的企业。这一历史性突破不仅标志着Foxboro中国在质量管理上达到了国际先进水平,也为中国仪表行业乃至整个工业领域树立了标杆。此后,Foxboro中国积极分享其质量管理经验,帮助众多中国企业取得国际质量体系认证,推动了整个行业的质量提升。(参考来源:知乎专栏及相关历史资料)
5. 数字化时代的转型与升级
2014年,Foxboro被施耐德电气收购后,融入了施耐德电气的一体化能源管理与过程自动化解决方案。在此背景下,Foxboro推出了新一代EcoStruxure Foxboro DCS过程自动化系统。该系统融合了Foxboro I/A Series控制系统的可靠性和施耐德电气的数字化能力,为企业客户提供了一站式的平台服务。EcoStruxure Foxboro DCS的推出,标志着Foxboro在数字化时代迈出了重要一步,也为公司在全球工业自动化领域的持续发展注入了新的动力。(参考来源:施耐德电气官网及相关产品介绍)
机顶盒,全称为数字视频变换盒,是现代家庭娱乐不可或缺的重要设备。从广义上讲,任何与电视机连接的网络终端设备均可视为机顶盒,它们不仅限于接收和转换电视信号,还承载着丰富的娱乐与信息服务功能。
机顶盒的核心功能在于接收并转换来自有线电缆、卫星天线、宽带网络及地面广播的数字电视信号,将其转换为电视机可识别的视频和音频流。这一转换过程确保了观众能够享受到高清乃至4K超高清的画质和震撼的音效,极大地提升了观看体验。此外,机顶盒还集成了诸如电子节目指南、因特网网页浏览等增值服务,让电视机不再仅仅是单向接收信息的设备,而是成为一个能够与用户进行互动的智能终端。
随着技术的发展,机顶盒的种类日益丰富,按主要功能可分为数字卫星机顶盒、有线电视数字机顶盒、IPTV机顶盒等。这些机顶盒不仅满足了用户多样化的观看需求,还通过连接互联网,提供了在线购物、视频点播、游戏娱乐等多元化服务。未来,随着高清、超高清技术的普及以及人工智能、云计算等技术的应用,机顶盒将向智能化、多功能化方向发展,为用户提供更加便捷、个性化的使用体验。
总之,机顶盒作为连接电视与互联网的桥梁,在丰富家庭娱乐生活、提升观看体验方面发挥着重要作用。随着技术的不断进步和市场需求的持续增长,机顶盒行业将迎来更加广阔的发展前景。
鸿展光电(GPO)公司的发展故事
故事一:初创与定位
深圳市鸿展光电有限公司(GPO)成立于2009年6月12日,起初便定位于高端显示技术的研发与生产。公司创始人凭借对电子行业未来趋势的敏锐洞察,决定专注于玻璃盖板(CG)、外挂式液晶显示模组(LCM)、外挂式触控模组(TP)等产品的研发与生产。在成立初期,面对激烈的市场竞争,鸿展光电凭借高质量的产品和优质的服务,逐步在行业内站稳脚跟。随着技术团队的不断壮大和产品的持续优化,公司迅速成长为一家集研发、生产、销售于一体的国家级高新技术企业。
故事二:技术突破与创新
在发展过程中,鸿展光电始终将技术创新视为企业发展的核心动力。公司不断加大研发投入,积极引进国内外先进技术和设备,致力于提升产品的技术含量和附加值。特别是在内嵌式触控显示一体化模组(INCELL、ONCELL)以及新一代触控显示模组(MiniLED、MicroLED、AMOLED)等领域,鸿展光电取得了多项关键性技术突破,并成功将这些技术应用于实际生产中,进一步巩固了公司在行业内的领先地位。
故事三:市场拓展与品牌建设
随着产品线的不断丰富和技术实力的不断提升,鸿展光电开始积极拓展国内外市场。公司通过参加各类行业展会、举办技术交流会等方式,不断提升品牌知名度和影响力。同时,鸿展光电还积极与国内外知名企业建立合作关系,共同开发新产品、新技术,实现互利共赢。这些举措不仅为公司带来了更多的商业机会,也进一步提升了公司的市场地位和品牌价值。
故事四:产能扩大与效率提升
为了满足日益增长的市场需求,鸿展光电在扩大产能方面进行了不懈努力。公司通过优化生产流程、引进先进生产设备、提升员工技能水平等措施,不断提升生产效率和产品质量。同时,公司还加强了供应链管理,确保原材料的稳定供应和成品的及时交付。这些举措不仅有效提升了公司的产能和竞争力,也为公司的长期发展奠定了坚实基础。
故事五:社会责任与可持续发展
在追求经济效益的同时,鸿展光电也积极履行社会责任,推动企业的可持续发展。公司注重环境保护和节能减排工作,通过采用环保材料和节能技术、加强废水废气处理等措施,降低生产过程中的环境污染和资源消耗。此外,鸿展光电还积极参与公益事业和社会活动,为社会做出积极贡献。这些举措不仅提升了公司的社会形象和声誉,也为企业的长期发展注入了新的动力。
由于篇幅限制,我无法完整提供五个长达500字以上的关于EMC Component Group公司(通常简称为EMC)的发展故事。然而,我可以概述五个与公司发展历程相关的重要事件或阶段,每个事件以精简的方式描述。
- 成立与初期发展
1979年,Richard Egan和Roger Marino两位创始人决定不再为别人打工,共同创立了EMC公司。最初,公司的业务并不专注于存储,而是销售办公家具。随着对市场的深入了解,他们发现了内存市场的商机,开始成为英特尔的渠道代理,为小型机提供内存。这一转变为EMC积累了第一桶金,也为公司未来的发展奠定了重要基础。
- 业务转型与内存业务的发展
在一位客服的建议下,EMC开始研发并销售Prime Computer的兼容内存。这些产品以同类产品一半的价格提供更高的性能,迅速在市场上获得认可。随着内存业务的不断发展,EMC逐渐成为了该领域的领导者。
- 存储解决方案的推出
进入21世纪,EMC开始将业务重心转向存储解决方案。2002年,公司推出了Centera,这是世界上第一个内容寻址存储(CAS)解决方案。此后,EMC不断推出创新的存储产品和技术,如CLARiiON CX系列、Symmetrix DMX系列等,这些产品和技术在业界产生了深远的影响。
- 云存储与软件定义技术的探索
随着云计算和软件定义技术的兴起,EMC也开始了在这一领域的探索。2014年,公司收购了全闪存初创公司,并试图将自己变成灵活的云供应商。此外,EMC还通过升级高端和低端产品线扩展了其产品线,并推出了针对闪存的基础特性进行开发和设计的XtremIO产品。
- 与Dell的合并
在EMC的发展历程中,一个重要的里程碑是与Dell的合并。这次合并不仅加强了EMC在存储和云计算领域的实力,也为其未来的发展提供了更广阔的空间。合并后,EMC成为了Dell Technologies的一部分,继续为全球客户提供创新的存储和云计算解决方案。
E.ON集团的前身可以追溯到1999年,当时德国的两个能源巨头VEBA和VIAG宣布计划在2000年合并。这一合并标志着E.ON的诞生,成为德国乃至欧洲的重要能源公司。合并后,E.ON迅速崭露头角,通过整合双方的资源和优势,迅速占据了德国及欧洲能源市场的重要位置。
用LPCXpresso一段时间,不知道怎么进行文件分类管理,希望能像MDK那样可以分别吧不同的文件放到不同文件夹。现在这样太乱了。希望高手们解答,在此先谢过了。 下面来张它们的比较图: [ 本帖最后由 zhaojun_xf 于 2010-5-15 07:05 编辑 ]… 查看全部问答∨ |
|
pxa270 wince 6.0 lcd是nec的NL2432HC22-41B。需要使用spi 在给lcd发送 power on命令的时候有个data input start 这个是做什么用的?应该发送什么data呢? … 查看全部问答∨ |
我 在网上下了一个连数据库的例子,同时也把相应的数据库拷到了模拟器的My Documents目录下,编译的时候没有错误,不知道怎么在我的机器上却总是说文件拷贝失败,数据库创建失败? m_strDataSource = "Provider = Microsoft.SQLSERVER.OLEDB.CE. ...… 查看全部问答∨ |
本人最近在做ov9650的驱动,平台为wince5.0 三星2450 现在的问题是为什么camera在显示图像的时候,LCD出现了两个窗口显示图像,而且图像显示不清楚… 查看全部问答∨ |
谁有keil下的lcd仿真啊?给我个啊,谢了 我下载了个。里面都是空的,name,lcd manufactuer lcd type 都为空… 查看全部问答∨ |
我想学习ARM和DSP编程,请问用什么软件较好?我大概知道DSP用CCS,ARM用SDT或ADS,我只想进行软件仿真,不连接硬件,哪款软件比较合适?谢谢!… 查看全部问答∨ |
设计资源 培训 开发板 精华推荐
- 有奖直播【Keysight World Tech Day 2023分论坛——汽车自动驾驶与新能源】
- 上演你的“ADI实验室电路”DIY!
- PI LYTSwitch™-6系列IC 让你了解不知道的秘密看专题赢好礼!
- 申请ufun学习板,入门&深度学习模拟,工程技能
- TTI Asia 2016慕尼黑上海电子展关注有礼活动
- MicroPython抢鲜玩 Python遇上MCU=? 由你定
- TE户外照明互连解决方案样片来袭 免费申领进行中!
- Mouser 福利活动:邀请新用户下单,各得百元奖励!
- 有奖直播【如何在几分钟之内完成高效可靠的USB PD电源设计——PI Expert™分步教程】(9:30开始入场)