
在我们日常开发过程中,常用到EEPROM作为数据存储。今天我以AT24C32为例,给大家接一下EEPROM的使用。1、在我们编写代码前,需要先确定EEPROM的连接STM32的I2C接口。9 ^1 [9 S6 a' F# u% _ 2、我们需要确定EEPROM的设备地址,在技术手册中找到Device Address,如下图所示。我们介绍使用A2 A1 A0均连接到GND。因此我们使用的Device Address为0xA0。R/W这一位由我们操作I2C读写时再填入。 ![]() 3、确定EEPROM的内部地址访问规则,如下图所示,AT24C32的访问地址为16位,我们在发送start信号以后,需要连续写入两个字节的地址,再进行后续操作。需要确定2字节的内容存储的格式:AT24C32的页最大容量为32byte,bit0-4用来存写入页内地址,bit5-11填写页地址。比如访问AT24C32的地址为:35,输入页地址为addr=35%32 | ((uint16_t)(35/32)<<5)。换算后的结果为0x0023。 ![]() 4、EEPROM写时序如下图,通过时序图我们确定写入是,需要先写入硬件地址0xA0,再写入两字节页内地址,后面写入的是数据内容。注意数据内容长度要小于等于32-页内起始地址。; H% d: t8 j/ F2 R6 a ![]() 5、EEPROM读时序如下图,通过时序图我们发现,读写操作是不一样的。读操作需要先写入设备地址,两字节页地址后,需要重新发送一个读取的start信号,然后就可以读取数据了。读数据时不需要考虑读的长度,但也不能超出EEPROM的范围。 ![]() 6、结合上面的介绍,我们来看一下EEPROM的代码操作。首先我们需要对I2C进行初始化,初始化时需要考虑I2C的复用功能,407和103的使用方法不同,需要注意。我们以407为例,代码初始化如下所示: /* Peripheral clock enable */ RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE); /**I2C2 GPIO Configuration+ P( a5 N7 c3 T: _' N PB8 ------> I2C1_SCL PB9 ------> I2C1_SDA9 Y: ]. J2 G5 {, R */9 Q. a: i5 D8 V; v RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE); GPIO_PinAFConfig(GPIOB, GPIO_PinSource8, GPIO_AF_I2C1); ]8 N# V2 ?/ D+ h5 e; s1 K GPIO_PinAFConfig(GPIOB, GPIO_PinSource9, GPIO_AF_I2C1);& x, j$ n, T' d9 ?6 p /*!< Configure I2C2 pins: SCL,SDA */4 ?) r- |+ c, [8 q GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9;" B* V, o* c! s" N$ Q; D GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;% ~ ^. A- f2 B5 E k GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_OType = GPIO_OType_OD;9 @4 B$ {+ n% _0 z# [! p7 M4 G GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_Init(GPIOB, &GPIO_InitStructure); _! C3 Z- G3 R% g" ]+ h* M Std_NVIC_SetPriority(I2C1_EV_IRQn, 0, 0);5 p1 K4 G! v @6 j& M Std_NVIC_SetPriority(I2C1_ER_IRQn, 0, 0);7、读操作代码 % J: W/ D! p/ y/ \! o while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY));* t! u3 k8 n9 F8 d$ X, ~- O6 @5 M " x) u8 l) X [/ K I2C_GenerateSTART(I2C1, ENABLE);2 m7 i8 K6 Y: Y while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); //EV5& M+ N: ]: D. x: k I2C_Send7bitAddress(I2C1, addr_dev, I2C_Direction_Transmitter); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); //EV6 M+ l9 c' w/ D. P4 }" K7 U 8 d: _/ x2 D5 R% ?3 u* r/ k I2C_SendData(I2C1,addr>>8]);; u/ _: p. o% D. G6 N) }# G1 T while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING));//EV8# ^2 `* L! e" c9 y* w$ i0 W + m. |0 M. k, [, ^$ y4 O, h I I2C_SendData(I2C1, addr); while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BTF) == RESET);//EV8 I2C_GenerateSTART(I2C1, ENABLE);2 M* v& `0 F$ ?* f while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); //EV5 I2C_Send7bitAddress(I2C1, addr_dev|0x01, I2C_Direction_Receiver);/ H U% l* _$ N) ^) S1 _' ? while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));//EV60 F( I$ i) S; x" q, Y) @* o while(Num --> 1)) r$ F1 D. G9 M* o/ g {8 ]) T( E0 H4 | while (I2C_GetFlagStatus(I2C1, I2C_FLAG_BTF) == RESET); *pBuffer++ = I2C_ReceiveData(I2C1); } I2C_AcknowledgeConfig(I2C1, DISABLE);. b: w. ]3 V5 f0 q2 X# H% n I2C_GenerateSTOP(I2C1, ENABLE); while (I2C_GetFlagStatus(I2C1, I2C_FLAG_BTF) == RESET); *pBuffer++ = I2C_ReceiveData(I2C1);: i) ~. k7 g) S' Z# i% Y 8、写操作代码& \+ b4 m' j( S/ d* j# ^7 r # ~& l5 q. x1 q( o+ T: X0 A5 E while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY)); I2C_GenerateSTART(I2C1, ENABLE);7 A7 ?/ n1 e# v" Z5 {* `! M while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); //EV5 ( [0 e# D5 L0 z3 f8 o0 @4 s I2C_Send7bitAddress(I2C1, addr_dev, I2C_Direction_Transmitter); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); //EV6: C. C( z) A. `, T& `8 i (void)I2C1->SR1;//EV61 O$ b. K; j6 J* \- U s- H8 b * \; S( ]4 k7 j# ?1 y' y I2C_SendData(I2C1, addr>>8]); //EV8-18 }8 A- w4 m& `2 ? while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING)); //EV8 I2C_SendData(I2C1, addr);( w) O" P: N5 Y; k8 w while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING)); //EV8" r% a* Q, Q% \4 a* Y ; S6 D% W g- E% B while(Num-->0) {/ t7 `& C$ J4 Z; T. j! t- A L I2C_SendData(I2C1,*pBuffer++);# _- U! W3 s( ^0 L2 I% B; i while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING)); //EV87 T* Q+ V% d% `: {5 D } I2C_GenerateSTOP(I2C1,ENABLE);, P6 T' e. V, u: K* n$ @$ l while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //EV8-2 1 o3 ]5 E# M; {7 y! O' v1 Q . D/ r, R4 b) x F$ f / v- n8 P4 T! E3 _ " k" i3 Z' ^/ }2 N5 N) }4 { 8 [) E3 O. p* P/ h4 T 7 i9 J/ F9 e/ m + I: ^+ P4 V) E8 @ |
【NUCLEO-WB09KE评测】使用I2C点亮一个OLED
【NUCLEO-WB09KE评测】USART的printf和GPIO的使用
【STM32H7S78-DK评测】使用FatFs文件系统读取SD卡文件
【NUCLEO-H533RE评测】+04 串口中断使用心得
【NUCLEO-H533RE评测】+03 printf输出调试
传感器为什么需要智能?
关于STM32的SWD引脚错误配置,“No Target Connected”导致程序不能下载问题解决
【年中粉丝节】+18B20温度传感器
【年中粉丝节】STM32L476开发板-基于rtthread读取加速度传感器KX224数据
STM32F401CCU6快速移植littleVGL