
在我们日常开发过程中,常用到EEPROM作为数据存储。今天我以AT24C32为例,给大家接一下EEPROM的使用。1、在我们编写代码前,需要先确定EEPROM的连接STM32的I2C接口。 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-页内起始地址。' L e$ e6 v- u" O! ^4 k ![]() 5、EEPROM读时序如下图,通过时序图我们发现,读写操作是不一样的。读操作需要先写入设备地址,两字节页地址后,需要重新发送一个读取的start信号,然后就可以读取数据了。读数据时不需要考虑读的长度,但也不能超出EEPROM的范围。( @4 z0 Q' B: M" Y' a3 F6 q ![]() 6、结合上面的介绍,我们来看一下EEPROM的代码操作。首先我们需要对I2C进行初始化,初始化时需要考虑I2C的复用功能,407和103的使用方法不同,需要注意。我们以407为例,代码初始化如下所示: /* Peripheral clock enable */' m1 G) G* Q# B% t* N$ U7 {% T RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE);7 O) z( S7 N- {) J* Y /**I2C2 GPIO Configuration9 I$ l F7 g7 D0 S PB8 ------> I2C1_SCL PB9 ------> I2C1_SDA5 `) ~8 L- _. y, U) ? *// C/ p1 X) p6 ?) S( } j RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);' S! y* J0 {' b3 H/ _% K# s GPIO_PinAFConfig(GPIOB, GPIO_PinSource8, GPIO_AF_I2C1);/ M# e) I8 B2 w8 L; \ GPIO_PinAFConfig(GPIOB, GPIO_PinSource9, GPIO_AF_I2C1);; z$ |% `+ X) S7 P% l, b, Y5 ^+ [ ! G5 y( b- ], _6 C P: F /*!< Configure I2C2 pins: SCL,SDA */9 E, N9 r2 p( N6 T/ a% V | GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_OType = GPIO_OType_OD; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;$ f2 g1 @8 n6 U2 y" p8 ] X4 p% b) Z# P GPIO_Init(GPIOB, &GPIO_InitStructure); Std_NVIC_SetPriority(I2C1_EV_IRQn, 0, 0);* z- w% Z: O9 R Std_NVIC_SetPriority(I2C1_ER_IRQn, 0, 0);7、读操作代码 , R2 C9 k/ D5 j0 W$ K. D while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY));+ j: ^" z- X' p4 l6 z) k I2C_GenerateSTART(I2C1, ENABLE);" U7 o+ \$ k7 @* k- L9 s9 E while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); //EV5 1 o8 N1 t; z3 H. p I2C_Send7bitAddress(I2C1, addr_dev, I2C_Direction_Transmitter); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); //EV6 , g; k' n7 g8 | I2C_SendData(I2C1,addr>>8]);8 S! C2 i1 q* |' B! s while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING));//EV8 I2C_SendData(I2C1, addr);- H/ _+ w4 W/ @1 n while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BTF) == RESET);//EV8: u! Z2 R. U6 E I2C_GenerateSTART(I2C1, ENABLE); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); //EV5* l' ~4 x) F) d" s) e! } I2C_Send7bitAddress(I2C1, addr_dev|0x01, I2C_Direction_Receiver);, E$ g0 {+ V5 ]. r while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));//EV6 while(Num --> 1) {4 g8 N6 q# q3 {2 E { while (I2C_GetFlagStatus(I2C1, I2C_FLAG_BTF) == RESET);0 d% j2 ` h2 a+ S: |0 o6 H4 e *pBuffer++ = I2C_ReceiveData(I2C1);4 q, s+ R# w5 k }9 J& S6 z- u( @0 F& o I2C_AcknowledgeConfig(I2C1, DISABLE); [% A9 T, ]( C0 f, \. F. _" ~ I2C_GenerateSTOP(I2C1, ENABLE);2 p3 G5 }; [" q2 f while (I2C_GetFlagStatus(I2C1, I2C_FLAG_BTF) == RESET);. {: t) `! U0 n5 s8 |6 q8 L9 f *pBuffer++ = I2C_ReceiveData(I2C1); 8、写操作代码 ! x3 d5 j3 R2 z3 [" S$ _ while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY));' _8 H, n. t! b1 o I2C_GenerateSTART(I2C1, ENABLE);& q3 p* D' e8 M9 A) j while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); //EV5 n/ {1 L, f: O/ F& V0 o 1 q) n" H+ W' o8 p, m& M% P I2C_Send7bitAddress(I2C1, addr_dev, I2C_Direction_Transmitter); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); //EV6 (void)I2C1->SR1;//EV6 ( t. M* i" H3 X0 ]7 V. f; } I2C_SendData(I2C1, addr>>8]); //EV8-1/ e6 e6 N; a( D0 @# h* k while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING)); //EV8 I2C_SendData(I2C1, addr);9 }+ f# }8 `# A( ~4 G while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING)); //EV8% i, w7 _$ t+ }0 I: _ while(Num-->0) {' `0 V Y3 J, y( e0 W y I2C_SendData(I2C1,*pBuffer++);) j6 t0 [- V: q while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTING)); //EV8/ Z4 u. X! W* z1 X" x6 S }; A, \/ V% e3 `* r I2C_GenerateSTOP(I2C1,ENABLE);% [: \* o+ R0 j4 D2 u& I while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //EV8-2 2 L* O. W$ P/ q/ W! T) r # X& [$ J) _6 I$ u R 4 q8 `+ @, W# Z0 e$ K6 e ( l- y; e8 K3 |) @; s1 U3 I + T5 S$ G# G K/ j- n1 v, H |
【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