【转帖】STM32开发板入门教程(六) - I2C--24Cxx

时间:2021-12-17 04:00:40

我们所用来示范的24Cxx系列是最常用的EEPROM芯片。
前面提到了一个地址码,
24Cxx的地址码是固定的,
8位如下:
1  0  1  0  A2 A1 A0  0
A2 A1 A0分别是它三个管脚的电平

24Cxx 理解起来有一个特别之处。
24Cxx 包括 01/02/04/08/16 四种,容量关系刚好和数字一样。1K 2K 4K 8K 16K
24C02 最为常见, 它的三个地址管脚A2 A1 A0都是可用的,
A2 A1 A0 有8中电平组合,也就是说,可以有8个 24C02 挂载同一个I2C总线上。
24C04呢, A0管脚就失效了,只有A2 和 A1 有用,四种组合,最多有4个24C04在总线上,
以此类推。24C16只能有一个在总线上。
这里就不好理解了,为什么要这样呢。
事实是一片 24C16 == 8片24C02 总线挂到一起。A2 A1 A0虽然起不到设置作用了,但你使用地址码还是会访问到特定的区域。
明白了吧。所以其实24C系列的代码是通用的。
地址码也是固定的。就是 0xA0 0xA2 0xA4 0xA6 0xA8 0xAA 0xAC 0xAE
好,我们以24C16为范例吧。

IO设置在I2C1上,无Remap,复用开漏输出。I2C 总线是挂4.7k电阻上拉到高电平的。
        //-----------------------I2C--------------------------------------------
            /* Configure I2C1 pins: SCL and SDA */
          GPIO_InitStruct.GPIO_Pin =  GPIO_Pin_6 | GPIO_Pin_7;
          GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
          GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_OD;   //复用开漏输出
          GPIO_Init(GPIOB, &GPIO_InitStruct);

I2C init函数:

void i2c_24c_init(I2C_TypeDef *I2Cx)
{
        I2C_InitTypeDef I2C_InitStruct;

        I2C_InitStruct.I2C_Mode = I2C_Mode_I2C;    // I2C模式
        I2C_InitStruct.I2C_Ack = I2C_Ack_Enable;    // ACK 在通讯中常见,握手包,即发送到了一个数据,接收方回一句,我收到鸟。
        I2C_InitStruct.I2C_ClockSpeed = I2C_Speed;  // I2C 速度设置,一般是40KHZ,400KHZ是极限,一般到不了那么高
        I2C_InitStruct.I2C_DutyCycle = I2C_DutyCycle_2;  //快速模式下的选项,这里先不讲,100KHZ以上才有用
        I2C_InitStruct.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;  //应答地址码长度,7位或者10位,24C是7位
      
// EEprom Block Select;
        I2C_InitStruct.I2C_OwnAddress1 = I2C_Slave_Adress7;   //第一个设备自身地址

        I2C_Cmd(I2Cx,ENABLE);   //开启I2C

        I2C_Init(I2Cx,&I2C_InitStruct);  //将刚刚的设置送进去
}

注意:现在的片子一般都不止一个I2C。所以用了上述模式,请详细看注释。

写一个字节进EEPROM:
参数解释:Byte待写的字节,WriteAddr预计写入的地址,ByteToWrite写多少给字节,EE24cBlockSelect选择EEPROM相应的区域(I2C地址),*I2Cx,I2C设备指针
void i2c_24c_byte_write(unsigned char Byte, unsigned char WriteAddr, unsigned int ByteToWrite, unsigned char EE24cBlockSelect,I2C_TypeDef *I2Cx)
{
        // Start the I2C
        I2C_GenerateSTART(I2Cx,ENABLE); //打开I2C,开始发送过程

      //not recommanded, stupid way
        while(!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_MODE_SELECT));  //设置主机模式

        I2C_Send7bitAddress(I2Cx,EE24cBlockSelect,I2C_Direction_Transmitter); //发送片选,选择哪一片区域写。i2C地址区分

        // when get ACK, means Set Success
        while(!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); //等待这次选择过程完成

        I2C_SendData(I2Cx, WriteAddr);  //发送要写入的地址码

        while(!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //等待字节发送完成

        I2C_SendData(I2Cx, Byte); //发送要写的字节

        while(!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //等待直到字节发送完成
        
        I2C_GenerateSTOP(I2Cx, ENABLE); //发送过程结束。
}

读EEPROM函数类似,却稍微复杂。
参数说明:pBuffer接收I2C数据的缓冲区,Addr读的地址,NumToRead读多少个字节,ee24cblockselect读哪个区域,I2Cx i2c设备指针
void i2c_24c_buffer_read(unsigned char *pBuffer, unsigned char Addr,unsigned char NumToRead,unsigned char EE24cBlockSelect, I2C_TypeDef *I2Cx)
{
        //open I2C
        I2C_GenerateSTART(I2Cx, ENABLE);  //开始发送

        while(!(I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_MODE_SELECT))); //设置自己为主机

        I2C_Send7bitAddress(I2Cx,EE24cBlockSelect,I2C_Direction_Transmitter);  //设置自己为发送

        while(!(I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED))); //等待主机发送模式设置成功

        I2C_Cmd(I2Cx,ENABLE);  //使能I2C

        I2C_SendData(I2Cx, Addr); //发送地址码,即要读的地址

        while(!(I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_TRANSMITTED)));  //等待主机发送过程完成

        I2C_GenerateSTART(I2Cx, ENABLE); //I2C开始发送

        while(!(I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_MODE_SELECT)));  //设置主机模式

        I2C_Send7bitAddress(I2Cx,EE24cBlockSelect,I2C_Direction_Receiver);   //设置从机地址,并设置主机为接收模式

        while(!(I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED))); //确认该过程完成

        while(NumToRead)
        {
                if(NumToRead==1)
                        {
                                I2C_AcknowledgeConfig(I2Cx, DISABLE);  //关闭I2C的应答功能

                                I2C_GenerateSTOP(I2Cx, ENABLE); //发送结束信息
                        }

                if((I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_RECEIVED)))  //如果接收到信息了
                        {
                                *pBuffer = I2C_ReceiveData(I2Cx);  //把接收到的数据 填进缓冲区当中

                                pBuffer++;

                                NumToRead--;
                        }
        }

        I2C_AcknowledgeConfig(I2Cx, ENABLE);  //开启主机I2C的应答功能
        
}

 

在写i2C和读i2C之间要插入下面函数等待,否则会有问题

I2C_EE_WaitEepromStandbyState();