历史上的今天
返回首页

历史上的今天

今天是:2025年04月09日(星期三)

正在发生

2019年04月09日 | STM32 I2C 难点

2019-04-09 来源:eefocus

I2C 总线在所有嵌入式系统中用得极广, 是一个工业级别的总线, 但由于STM32 是一个32位的MCU, 注定了他的I2C硬件接口将会功能强大, 但同时也会较难于控制,不象8位机,如AVR8位机的TWI(实际完全符合 I2C标准) 那么易用. 以下是我的STM32 I2C硬件接口编程的一些心得体会.

    

如果你选择了STM32, 说明了你的项目的需求是比较复杂的,使用EMBEDDED OS 和大量地运用中断+DMA的编程模型是必然的选择, 如果你的项目中用STM32,而你用模拟的I2C的话, 说明了两点: 一是浪费了STM32; 二, 如果你的项目很复杂的话,你会发现在项目的开发后期,好象STM32也比8位机快不了多少, WHY!! ,但这不是STM32的问题,而是你没有最有效地利用上STM32.

    

很多朋友在搞STM32的I2C接口编程时总是时不时“当在某处”(GOOGLE时你会发现这个问题很普遍), 一些朋友这时就会用软件来模拟I2C,然后,很快发现和I2C设备能很好地通信了(但当机还是可能随机出现), 这些朋友于是大骂STM32的I2C硬件接口是个”杯具”(呵呵,我有时也会突然想骂骂,但我知道,99.999%的原因还是自已对于STM32硬件接口的熟悉程度不够,或者说,是我没有扬STM32 I2C的长,而总是捉住他的短不发。)。

    

固然,STM32 I2C硬件接口有设计不完善的地方,例如下面就是我从STM32最新的Errata sheet中总结出的,关于STM32 I2C接口设计上的一些缺陷和如何避开这些缺陷的推荐程序模型:

(1)把I2C的中断优先级提升到最高

(2)把发送多于2个字节的发送与接收封装成利用DMA收发的函数,而把对某I2C设备接收和发送一个字节的函数单独封装为一个POLLING (轮询)函数。

(3)在寻址某一I2C DEVICE时要先CHECK I2C BUS 是否BUSY,如果忙,则等待指定时间,如果还是忙就说明I2C BUS 挂了(原因99.9%是由于我们的I2C通信时序并不十分尊守I2C规约,或者我们所封装的I2C通信模块没有加上防守代码(出错恢复代码)),这时要调用一个专门的用于通知 I2C BUS上的所有device,让他们结束当前内部的工作,重新准备好(下雨了,收衣服啦)。如下面的我的I2C模块的FUN 切片:

image


该函数一定要用在主MCU的启动模块上,因为I2C总线在充当Master的MCU启动时,SDA和SCL有可能组合出刚好符合I2C规约的时序组合,比如一个开始位(START CONDITION),使得I2C BUS 立即当在那里(因为当主MCU真正需要发出一个START CONDITION时,发现I2C BUS 正处于BUS状态,而根据STM32 手册的START CONDITION说明可知,一个起始条件将会使得I2C BUS处于BUSY 状态, 下面的I2C2_Free_Buf fun 的基本用法:

image


(注: I2C2_Free_Bus Fun 应放在线程中,而不是放在上图中的位置,这样会触发并进入一个硬件错误处理向量中断中)

提示:摘自STM32 手册:
    I2C.SR1.Bit 0 位(SB)- Start bit(Master mode)
    - Set when a Start condition generated.
    - Cleared by software by reading the SR1 register followed by writing the DR register, or by hardware when PE=0.

(4) 不要让I2C工作在88KHz的频率上,低于或者使用Fast-mode(400KHz)频率,这是STM32 I2C真正的一个硬件BUG(99.999%机率),但是也是可以编程避免的。

(5)Programming the bit NOSTRETCH=0  in the I2C_CR1 register. 这样也可避免一个STM32 I2C硬件设计的一个小BUG(2。9。5节)

(6)大部分的MCU的硬件I2C接口的工作模式是中断(高端的会用DMA)+状态机;因此状态机的编程概念要熟悉

(7)STM32 I2C的硬件接口负责实现满足I2C总线的的规约,而我们(嵌入式编程开发者)则是通过I2C 控制寄存器和I2C的事件标志组合来启动状态机,然后让状态机按照由I2C SR1 和SR2所组合志来的事件自动工作,并在发送或接收完成后通过FLAG的方式或信号量的方式通知我们所写的读写函数,操作已经完成,或者在操作中出现了错误,如最常见的AF错误(device 在第9位上没有拉低SDA应答Master。)

(8)I2C SR1 和SR2的功能分配(这是一个极易忽视的思考死角)

    从STM32 手册的I2C register map 中可以看到, I2C的SR1,主要是反映I2C通信的最基本的标志,要清除SR1的某个标志可以直接清除,而I2C的SR2即是辅助SR1的,他一般反映了I2C总一当前的工作状态,如BUSY,是主机模式还是从机模式,等等。关于SR2的很重要的一个编程模型是:要清除SR1的某些指定的标志位时,比如ADDR,先读SR1然后再读SR2将会清除掉已置位的ADDR。

(9)Master在操作slave device时要先和他握一下手是很好的防守编程模型:


image


推荐阅读

史海拾趣

敦泰(FOCALTECH)公司的发展小趣事
电冰箱不制冷可能由多种电路问题引起,如电源线路故障(如插头未插紧、插座无电等)、压缩机启动电路故障(如启动器损坏、压缩机线圈断路等)、温控电路故障(如温控器失灵、温度传感器损坏等)或制冷系统电路故障(如制冷剂泄漏、毛细管堵塞等)。建议首先检查电源是否正常,然后逐步排查压缩机、温控器和制冷系统电路。
HM International公司的发展小趣事
电冰箱不制冷可能由多种电路问题引起,如电源线路故障(如插头未插紧、插座无电等)、压缩机启动电路故障(如启动器损坏、压缩机线圈断路等)、温控电路故障(如温控器失灵、温度传感器损坏等)或制冷系统电路故障(如制冷剂泄漏、毛细管堵塞等)。建议首先检查电源是否正常,然后逐步排查压缩机、温控器和制冷系统电路。
Herga公司的发展小趣事

近年来,Herga公司紧跟智能制造的发展趋势,大力推动生产线的自动化和智能化升级。公司引入了先进的机器人、自动化设备和智能管理系统,实现了从原材料采购、生产加工到成品检验的全流程自动化控制。这一举措不仅大大提高了生产效率和质量稳定性,还降低了人力成本和运营风险。Herga公司的智能制造实践为电子行业树立了典范,也为公司的持续发展注入了新的动力。

Dolphin Interconnect Solutions Asa公司的发展小趣事

随着电子行业的快速发展和市场竞争的加剧,Dolphin Interconnect Solutions ASA面临着前所未有的挑战。为了应对这些挑战,公司积极调整战略方向,加大在人工智能、物联网等新兴领域的投入。同时,Dolphin还加强了与产业链上下游企业的合作,共同推动产业链的转型升级。这些努力使得Dolphin在激烈的市场竞争中保持了稳健的发展态势。

A-BRIGHT公司的发展小趣事

随着全球环保意识的提高,A-BRIGHT公司积极响应环保号召,将环保理念融入到产品研发和生产中。公司研发了一系列节能环保的电子产品,不仅满足了客户的需求,也为全球环保事业做出了贡献。这一举措提升了A-BRIGHT的品牌形象,也为其在竞争激烈的市场中赢得了更多客户的青睐。

这些故事虽然是虚构的,但它们基于电子行业的发展趋势和可能情况,展示了A-BRIGHT公司可能的发展路径和成功因素。希望这些故事能够满足您的需求。

Gentron Corp公司的发展小趣事
在演出或活动中作为舞台灯光效果使用,增加舞台氛围。

问答坊 | AI 解惑

一个嵌入式开发的资料(希望抛砖引玉)

嵌入式的开发如今是变的越来越复杂,对嵌入式工程师的要求越来越高。如何从一个应届毕业生, 或者普通的工程师向嵌入式迈进呢。网络上的资料很多,书店的书籍也很多,但大家都很难下手或者靠自学提高。 在最近的公司招聘面试的过程中,发现很多面试者( ...…

查看全部问答>

LCD501液晶屏电子钟和电压表程序

本帖最后由 paulhyde 于 2014-9-15 08:57 编辑 见附件  …

查看全部问答>

WinCE LCD 驱动VBPD、VFPD等参数的计算

我使用的smdk2410的lcd驱动中有这样的代码 //TFT timing parameter for V16C6448AB(PRIME VIEW) #define VBPD                ((1-1)&0xff) #define VFPD        ...…

查看全部问答>

请问没有系统的UART编程和CE下的UART编程的区别?

撇开软件部分不说,单看硬件方面的设置: 1)基于ARM7裸机的串口简单的编程:如串口初始化,设置波特率,寄存器设置,中断向量设置等 2)自己基于CE下编一个串口驱动,也是要向上面一样设置ARM芯片的初始化吗?和裸机的那些设置是一样 ...…

查看全部问答>

工控工程师群70677754——交流开发经验,解决工程难题,寻找合作机会,非专业人士勿入。

工控设计群——70677754,非高手莫进,加入请写明水平及方向。 本群以交流开发经验,解决工程难题,寻找合作机会为目标。 欢迎热爱工控事业的工程师们入群 51,ARM,DSP Keil c,Proteus. Linux.RTOS.网络 机器视觉 运动控制…

查看全部问答>

急!!!eboot.bin文件怎么解压缩!!!

eboot.bin文件烧进系统以后,会自行压缩成eboot.nb0文件。我现在需要知道这个压缩的过程的代码是在哪个文件的哪个函数里的!有知道的吗??? 我找了F:\\WINCE500\\PUBLIC\\COMMON\\OAK\\DRIVERS\\ETHDBG\\BLCOMMON的BLCOMMON.c文件的DownloadImage函 ...…

查看全部问答>

求救,按键驱动

现在我想写一个2440的按键驱动,只需要几个按键就可以了,比较(上,下,左,右功能)。 要求能够传递给任意进程中,比如按上下左右键能够操作控制界面(UI)。 请问是不是只能在提供的标准的键盘驱动中写,有没有更简单的方法,哪位大哥给点思路呀…

查看全部问答>

ATMEGA8--想要爱你不容易!!

      在这夏天,炎热酷暑。在这夏天,躲在架设有几十台电脑的实验室,狭窄的空间,空气是凝固的,室内气温自然上去了。就这样,度过了一个刻骨铭心的暑假!然而,一个暑假下来,付出是多,收获也是多的。你们是否记得, ...…

查看全部问答>

IAR使用J-Link仿真的问题

我写了一个LM3S9B92的UART简单收发程序,想下载到开发板中,用的是IAR软件,和J-LINK仿真器,请高手指教一下具体的步骤,比如说软件的相关配置等等问题,谢了   我在下载的 过程中出现了这个问题 Fatal Error[Lc002]: could not open file ...…

查看全部问答>