本帖最后由 shiyongzhu 于 2015-8-14 14:55 编辑
6 Q3 g1 f2 y3 o6 r
0 G, J- Q4 I, r+ ]1 Z( VST的HAL库的I2C模块函数库中提供了大量的函数,方便了用户的使用。近期由于需要使用下I2C接口,对HAL库中的函数以及相关的例程进行了一下研究,发现当mcu处于从机的时候,其提供的函数只能满足从机处于接收或者发送状态之一(本人观点,欢迎指正),当主机既需要对从机既需要读又需要写时,特别的不方便。为此本人对HAL库中的I2C进行了一下改造,使其处于从机时收发皆可。 主要的思想是开启地址匹配中断,主机对从机进行读或写时进入该中断后,在中断中读取状态寄存器ISR ,看主机是读还是写,然后分别进入相关的子程序即可。
具体实现以stm32f072为例,初始化,注意开启中断: - void MX_I2C1_Init(void)
9 g0 k* A1 S5 k5 D - {) G" v- _ _0 T1 ~6 o2 c
1 j) g/ @* d h- hi2c1.Instance = I2C1;; r# I, @$ @2 y) x, a4 T+ B8 T
- // hi2c1.Init.Timing = 0x00700000; //1MHz,100ns,100ns,slave
( @ D% E: U2 K* t% N- i5 I - hi2c1.Init.Timing = 0x00900000; //400kHz,10ns,10ns,slave9 y- f; \. f) F# D
- hi2c1.Init.OwnAddress1 = ADDRESS; //µçµ÷µØÖ·
) B) T1 {3 }* q% v N* R, D - hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT; // µØַΪ7λ
. x) y; I. e" U: n8 k a - hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLED;$ e7 b6 E# D+ b
- hi2c1.Init.OwnAddress2 = 0;
- J! l+ `" w) m - hi2c1.Init.OwnAddress2Masks = I2C_OA2_NOMASK;. h' @! P( s6 q2 l) S& w8 C
- hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLED;8 {2 q8 H: O3 d- u7 x3 T
- hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLED;+ v m2 ^( p9 X) i7 o
- HAL_I2C_Init(&hi2c1);) a' T" N2 D- E9 b( d
- 0 a5 D4 {8 C& c* @0 a6 v
- /**Configure Analogue filter 3 ?( z# s7 V4 ^$ o7 s) H$ f- X+ ^
- */
+ H0 y* r% ^; x9 ^& m - HAL_I2CEx_AnalogFilter_Config(&hi2c1, I2C_ANALOGFILTER_ENABLED);9 _/ q" O0 c8 q9 h- s. t: r! ?0 _2 ?
- 2 l8 w% c! v( m; O$ Z1 w
- HAL_NVIC_SetPriority(I2C1_IRQn, 0, 0); //i2c15 r) ^! e, C9 _. I; g q
- HAL_NVIC_EnableIRQ(I2C1_IRQn);
9 S+ X7 w. y* M5 r - }
复制代码 ; C4 X* q# V2 @, X
然后写I2C中断服务函数void I2C1_IRQHandler(),具体程序如下 - void I2C1_IRQHandler(void)
# l5 ~6 \- h& F! Z1 r9 M - {
2 G8 t; t0 _$ c/ K - if (hi2c1.Instance->ISR & (I2C_FLAG_BERR | I2C_FLAG_ARLO | I2C_FLAG_OVR)) {
* O6 K( \. }) ]9 o; a - HAL_I2C_ER_IRQHandler(&hi2c1); Q! k6 C9 s7 v5 K% y% p
- } else {
' ?& X) a) j" o$ o+ B; \7 b - HAL_I2C_EV_IRQHandler(&hi2c1);
2 I7 i) W9 D9 F0 |# l - }) k f A7 q# j4 p y8 |
- }
复制代码 - E) K- n" v. r% N# Z
具体干活的程序如下,本处对原有的函数库进行了改造 - void HAL_I2C_EV_IRQHandler(I2C_HandleTypeDef *hi2c)1 V3 f+ w' a+ \- z( g
- {
, d. h1 z8 k2 j# I6 p, t- B1 q - if ((__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_ADDR) == SET) && (__HAL_I2C_GET_IT_SOURCE(hi2c, I2C_IT_ADDRI) == SET))
W. q( y6 ] p/ @ - {
* |) H8 [- v4 R+ p0 m - HAL_I2C_SlaveRxTxCallback(hi2c);
0 Y0 X& ?6 P \; r" M: l5 \- } - } / |' {' `4 O8 B
-
# D' a, X x1 K) r7 g5 W( ?+ O - }
复制代码 * r, P( V$ m5 z" }# J
HAL_I2C_SlaveRxTxCallback(hi2c)为自己写的函数,具体如下 - void HAL_I2C_SlaveRxTxCallback(I2C_HandleTypeDef *hi2c) . y6 B4 Q' |; y5 O: D/ X5 h8 x
- {
5 i u9 t* U3 m - if(hi2c->Instance==I2C1)
. B. i/ w/ @ X7 l* p - {1 z) q* U4 i# h; B) D7 a9 I- n0 z
- /* Slave mode selected */
5 V6 X$ H' T" E3 x2 y- z - if (hi2c->State == HAL_I2C_STATE_READY)5 [9 K5 q1 Z9 x9 |* y
- {
* h8 C0 v) t% i5 U0 L - __HAL_I2C_CLEAR_FLAG(hi2c, I2C_FLAG_ADDR); // ÇåÖжÏ
1 T/ Z2 }7 @4 a+ L! t M) T - if(__HAL_I2C_GET_FLAG(hi2c, I2C_FLAG_DIR)) // ÊÕ·¢·½Ïò
. w8 b+ B) ~2 |, S* O - {
& l3 ~) e7 _& W1 C, Z1 K" [$ P - HAL_I2C_Slave_Transmit_DMA_NEW(hi2c,(uint8_t*)i2c1_tx_buffer, sizeof(i2c1_tx_buffer));& ~" N+ t/ y* s* ~
- }# O! ]* s, g8 H5 D
- else
' C2 E8 T) e4 e% m( Y - {- q+ y& q# d8 ]) j
- HAL_I2C_Slave_Receive_DMA_NEW(hi2c,(uint8_t*)i2c1_rx_buffer, sizeof(i2c1_rx_buffer));
7 J* V; i7 c/ b - }; D/ q$ n: ?- S2 c$ r& \0 r9 B: G
- }# h1 n/ A8 Q2 B2 A
- }3 f* p. r1 E5 U. G/ v' `# w# S
# W# u* e: Y. f; B- }
复制代码
% ]" g' k: B4 J3 S% z- t3 g上面函数中的函数: HAL_I2C_Slave_Transmit_DMA_NEW(hi2c,(uint8_t*)i2c1_tx_buffer,sizeof(i2c1_tx_buffer)) HAL_I2C_Slave_Receive_DMA_NEW(hi2c,(uint8_t*)i2c1_rx_buffer,sizeof(i2c1_rx_buffer)); 为HAL_I2C_Slave_Transmit_DMA和HAL_I2C_Slave_Receive_DMA_NEW的改造,删除了原有函数中的地址匹配和方向判断的内容。 5 j1 Z3 W/ E$ D: w, v& }
使用时,只要开启地址匹配中断一切就OK了 - __HAL_I2C_ENABLE_IT(&hi2c1,I2C_IT_ADDRI);
复制代码
. A# w/ f: a9 B% ? ^2 i; B
% U# v) J: S- E* f _5 P a; m* Q7 S: e; q( b2 E/ n
|
IIC的结束符属于底层的细节(类似于物理层的协议),HAL库封装了IIC的实现细节,你直接关注上面的应用就行了。
至于收发数量,即使你自己实现物理层也不会知道具体的数量吧?!你只是知道来了数据了,每次读若干个。
如果上位机发过来的数据不定长怎么办? 我也不知道他要发多少字节。那我怎么知道应该接收多少个。
以前其他单片机都是已结束位判断一次接收已经完成?
是不是以结束位判断比较好呢?
IIC 都是以结束位结束,HAL库接收发送都要定长度,这很郁闷。
: _7 [& ?' y. Z: Q% }# \
楼主在编辑帖子的时候,右上角有个添加代码文字按钮,用来放代码特别好用
谢谢沐紫,用了下效果好很多!
,我只是路过打酱油的 b, T0 m- v. ~
在中断回调函数中设置读1个字节,然后根据你的需要判断就可以了。接受完1个字节后根据需要再次打开中断。