打印
[STM32F4]

STM32F407单片机通过IIC读写EEPROM 24C02

[复制链接]
36|0
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
磨砂|  楼主 | 2025-1-12 08:11 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式
  本文旨在通过使用STM32F407的IIC总线读写 EEPROM 24C02。

        一些细节可以参考本文作者写的其他文章。

        参考文章1:细说STM32F407单片机轮询方式读写SPI FLASH W25Q16BV-CSDN博客  https://wenchm.blog.csdn.net/article/details/144587209

         参考文章2,细说STM32F407单片机IIC总线基础知识-CSDN博客  https://wenchm.blog.csdn.net/article/details/144717358

一、操作说明
        项目使用旺宝红龙开发板,STM32F407ZGT6 KIT V1.0。 使用开发板上的按键S2、S3、S4、S5,依次按下诸键后执行如下操作,并且使用开发板上的LED灯,依次显示操作状态。S6是复位键。



[S2]KeyUp   = Write a number     LED1
[S3]KeyDown = Read the number    LED2
[S4]KeyLeft = Write a string     LED3
[S5]KeyRight= Read the string    LED4
二、工程配置
1、时钟、DEBUG、GPIO、USART6、NVIC、Code Generator
        与参考文章2相同。

        I2C的中断事件主要是表示传输过程和错误的一些事件,由于I2C通信是一种应答式通信,与其他外设的轮询式操作类似,本例不开启I2C2的中断。I2C也具有DMA功能,但是24C02操作的数据量小,没有使用DMA的必要。如果需要使用I2C接口的中断或DMA数据传输功能,可看参考文章中介绍的中断方式和DMA方式相关函数。

2、 IIC2
        占用管脚PF0、PF1。默认全部设置。

(1)Master Features组,主设备参数
I2C Speed Mode,速度模式。可选标准模式(Standard Mode)或快速模式(Fast Mode)。
I2C Clock Speed (Hz),I2C时钟速度。标准模式最大值为100kHz,快速模式最大值为400kHz。
Fast Mode Duty Cycle,快速模式占空比。选择快速模式后这个参数会出现,用于设置时钟信号的占空比,是一个周期内低电平与高电平的时间比,有2:1和16:9两种选项。
        本示例中,速度模式选择标准模式。

(2)Slave Features组,从设备参数
Clock No Stretch Mode,禁止时钟延长。设置为Disabled表示允许时钟延长。
Primary AddressLength selection,设备主地址长度。可选7-bit或10-bit,本例选择7-bit。
DualAddress Acknowledge,双地址确认。从设备可以有两个地址,如果设置为Enabled。还会出现一个Secondary slave address参数,用于设置从设备副地址。
Primary slave address,从设备主地址。设置从设备主地址,作为I2C从设备时才需要设置。
General Call address detection,广播呼叫检测。设置为Disabled表示禁止广播呼叫,不对地址0x00应答;否则,就是允许广播呼叫,对地址0x00应答。
       STM32F407是I2C主设备,无须设置从设备地址。24C02是I2C从设备,其从设备地址是0xA0。

三、软件设计
1、KEYLED
        keyled.c和keyled.h与参考文章相同。

2、EEPROM
       在项目中创建EEPROM文件夹,并在其中设计24cxx.c和24cxx.h。

        为便于程序的移植,在文件24cxx.h中定义一个宏I2C_HANDLE替代hi2c2。如果使用了不同的I2C接口,只需修改这个宏定义即可,而I2C接口的外设初始化由CubeMX自动生成的函数完成。在24cxx.h中还定义了表示24C02地址的宏DEV_ADDR_24CXX,如果实际电路中的24C02的I2C地址被修改了,修改这个宏即可。

        在文件24cxx.h中定义了5个函数,前面4个都是直接封装I2C的HAL传输函数实现的,最后1个函数EP24C_WriteLongData()能自动将一个长的数据拆分为多个页(每页8字节)写入,这5个函数都使用I2C的阻塞式存储器数据传输函数。

(1)24cxx.c

/* 文件: 24cxx.c
* 描述: 24C02驱动程序源程序文件
*/

#include "24cxx.h"
#define        EP24C_TIMEOUT 200                                                //超时等待时间,单位:ms
#define        EP24C_MEMADD_SIZE I2C_MEMADD_SIZE_8BIT  //存储器地址大小,8位地址

//检查设备是否准备好I2C通讯,返回HAL_OK 表示OK
HAL_StatusTypeDef EP24C_IsDeviceReady(void)
{
        uint32_t Trials=10;                //尝试次数
        HAL_StatusTypeDef result=HAL_I2C_IsDeviceReady(&I2C_HANDLE,DEV_ADDR_24CXX,Trials,EP24C_TIMEOUT);
        return result;
}

//向任意地址写入一个字节的数据, memAddr是存储器内部地址,byteData是需要写入的1个字节的数据
HAL_StatusTypeDef EP24C_WriteOneByte(uint16_t memAddress,uint8_t byteData)
{
        HAL_StatusTypeDef result=HAL_I2C_Mem_Write(&I2C_HANDLE,DEV_ADDR_24CXX,memAddress,
                        EP24C_MEMADD_SIZE,&byteData,1,EP24C_TIMEOUT);
        return        result;
}

//从任意地址读出一个字节的数据, memAddr是存储器内部地址,byteData是读出的1个字节的数据,若返回HAL_OK表示读取成功
HAL_StatusTypeDef EP24C_ReadOneByte(uint16_t memAddress,uint8_t *byteData)
{
        HAL_StatusTypeDef result=HAL_I2C_Mem_Read(&I2C_HANDLE,DEV_ADDR_24CXX,memAddress,
                        EP24C_MEMADD_SIZE,byteData,1,EP24C_TIMEOUT);
        return result;
}

//连续读取数据,任意地址,任意长度,不受页的限制
HAL_StatusTypeDef EP24C_ReadBytes(uint16_t memAddress,uint8_t *pBuffer,uint16_t bufferLen)
{
        if(bufferLen>MEM_SIZE_24CXX)        //超过总存储容量
                return HAL_ERROR;

        HAL_StatusTypeDef result=HAL_I2C_Mem_Read(&I2C_HANDLE,DEV_ADDR_24CXX,memAddress,
                        EP24C_MEMADD_SIZE,pBuffer,bufferLen,EP24C_TIMEOUT);
        return result;
}

//限定在一个页内写入连续数据,最多8字节。任意起始地址,但是起始地址+数据长度不能超过页边界
HAL_StatusTypeDef EP24C_WriteInOnePage(uint16_t memAddress,uint8_t *pBuffer,uint16_t bufferLen)
{
        if(bufferLen>PAGE_SIZE_24CXX)        //数据长度不能大于页的大小
                return HAL_ERROR;

        HAL_StatusTypeDef result=HAL_I2C_Mem_Write(&I2C_HANDLE,DEV_ADDR_24CXX,memAddress,
                        EP24C_MEMADD_SIZE,pBuffer,bufferLen,EP24C_TIMEOUT);
        return result;
}

//写任意长的数据,可以超过8字节,但数据地址必须从页首开始,即 8*N。自动分解为多次写入
HAL_StatusTypeDef EP24C_WriteLongData(uint16_t memAddress,uint8_t *pBuffer,uint16_t bufferLen)
{
        if(bufferLen>MEM_SIZE_24CXX)        //超过总存储容量
                return HAL_ERROR;

        HAL_StatusTypeDef result=HAL_ERROR;
        if(bufferLen<=PAGE_SIZE_24CXX)        //不超过1个page,直接写入后退出
        {
                result=HAL_I2C_Mem_Write(&I2C_HANDLE,DEV_ADDR_24CXX,memAddress,
                                EP24C_MEMADD_SIZE,pBuffer,bufferLen,EP24C_TIMEOUT);
                return result;
        }

        uint8_t *pt=pBuffer;                        //临时指针,不能改变传入的指针
        uint16_t pageCount=bufferLen/PAGE_SIZE_24CXX;        //Page个数
        for(uint16_t i=0; i<pageCount; i++)        //一次写入一个page的数据
        {
                result=HAL_I2C_Mem_Write(&I2C_HANDLE,DEV_ADDR_24CXX,memAddress,
                                EP24C_MEMADD_SIZE,pt,PAGE_SIZE_24CXX,EP24C_TIMEOUT);
                pt += PAGE_SIZE_24CXX;
                memAddress += PAGE_SIZE_24CXX;
                HAL_Delay(5);                                        //必须有延时,以等待页写完
                if (result != HAL_OK)
                        return result;
        }

        uint16_t leftBytes=bufferLen % PAGE_SIZE_24CXX;  //余数
        if (leftBytes>0)                                        //写入剩余的数据
                result=HAL_I2C_Mem_Write(&I2C_HANDLE,DEV_ADDR_24CXX,memAddress,
                                EP24C_MEMADD_SIZE,pt,leftBytes,EP24C_TIMEOUT);
        return result;
}


(2)24cxx.h

/*
*  文件名:24cxx.h
*         功能描述: 24C02 EEPROM驱动程序,使用HAL库
*
*         (1)24C02是 256字节EEPROM,可以单个字节读写,连续读写时按页读写,每页8字节。所以,按页读写时最多8字节,页号0--31
*
*         (2)连续写入数据时要注意不要超过页的边界,否则从页的开始处重新写,会覆盖原来的内容。
*
*         (3)24C02地址是 0b1010xxxy, 其中xxx由芯片的地址引脚A2,A1,A0决定,一般接地,所以是0xA0, HAL库在读写时自动在最后一位写0或1进行读或写操作
*
*         (4)24C02的地址数据长度是8位,使用宏定义符号I2C_MEMADD_SIZE_8BIT
*/


#ifndef __24cxx_H
#define __24cxx_H

#include "stm32f4xx_hal.h"
#include "i2c.h"                                 //i2c.h中定义了hi2c2
/* 两个与硬件相关的定义,   */
#define I2C_HANDLE hi2c2                //I2C外设对象变量,利用i2c.h中定义的hi2c2
#define        DEV_ADDR_24CXX 0x00A0        //24C02的写地址
// EEPROM存储器参数
#define        PAGE_SIZE_24CXX 0x0008        //24C02的Page大小为8字节
#define        MEM_SIZE_24CXX (uint16_t)256 //24C02总共容量字节数,8*32=256字节

//检查设备是否准备好
HAL_StatusTypeDef EP24C_IsDeviceReady(void);

//在任意地址写入一个字节
HAL_StatusTypeDef EP24C_WriteOneByte(uint16_t memAddress, uint8_t byteData);

//在任意地址读出一个字节
HAL_StatusTypeDef EP24C_ReadOneByte(uint16_t memAddress, uint8_t *byteData);

//连续读取数据,任意地址,任意长度,不受页的限制
HAL_StatusTypeDef EP24C_ReadBytes(uint16_t memAddress, uint8_t *pBuffer, uint16_t bufferLen);

/*限定在一个页内写入连续数据,最多8字节。任意起始地址,
*但是起始地址+数据长度不能超过页边界,即不能超过地址8*N-1 */
HAL_StatusTypeDef EP24C_WriteInOnePage(uint16_t memAddress, uint8_t *pBuffer, uint16_t bufferLen);

//写任意长的数据,可以超过8字节,但数据地址必须从页首开始,即 8*N
HAL_StatusTypeDef EP24C_WriteLongData(uint16_t memAddress, uint8_t *pBuffer, uint16_t bufferLen);

#endif

3、main.c
/* USER CODE BEGIN Includes */
#include "keyled.h"
#include "24cxx.h"
#include <stdio.h>
/* USER CODE END Includes */
/* USER CODE BEGIN 2 */
  printf("Demo17_1:I2C Interface\r\n");
  printf("24C02:EEPROM, 256 bytes\r\n");
  printf("8 bytes/page, 32 pages\r\n");
  printf("I2C Device Address=0xA0\r\n");
  if (EP24C_IsDeviceReady() == HAL_OK)
          printf("Device is ready.\r\n");

  //显示菜单
  printf("[S2]KeyUp   = Write a number\r\n");
  printf("[S3]KeyDown = Read the number\r\n");
  printf("[S4]KeyLeft = Write a string\r\n");
  printf("[S5]KeyRight= Read the string\r\n");

  // MCU output low level LED light is on
  LED1_OFF();
  LED2_OFF();
  LED3_OFF();
  LED4_OFF();
  /* USER CODE END 2 */

  /* USER CODE BEGIN WHILE */
  uint8_t num1 = 107,num2;
  uint16_t addr_any = 4;        //任意地址, 0-255
  uint16_t addr_page = 2*8;        //Page2起始地址
  //uint8_t infoStr[50];        //用于生成显示信息字符��?
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
        KEYS curKey = ScanPressedKey(KEY_WAIT_ALWAYS);

        switch(curKey)
        {
          case KEY_UP:
          {
                if (EP24C_WriteOneByte(addr_any,num1) == HAL_OK)
                  printf("Write %d at Address %d.\r\n",num1,addr_any);
                LED1_ON();
                LED2_OFF();
                LED3_OFF();
                LED4_OFF();
          }
          break;

          case KEY_DOWN:
          {
                if (EP24C_ReadOneByte(addr_any,&num2) == HAL_OK)
                   printf("Read out %d at Address %d.\r\n",num2,addr_any);
                LED1_OFF();
                LED2_ON();
                LED3_OFF();
                LED4_OFF();
          }
          break;

          case KEY_LEFT:
          {
                uint8_t strIn[] = "University of Petroleum";        //自动加'\0'
                if (EP24C_WriteLongData(addr_page,strIn,sizeof(strIn)) == HAL_OK)
                  printf("Write string from Page2:%s\r\n",strIn);
                LED1_OFF();
                LED2_OFF();
                LED3_ON();
                LED4_OFF();
          }
          break;

          case KEY_RIGHT:
          {
                uint8_t strOut[50];
                if (EP24C_ReadBytes(addr_page,strOut,50) == HAL_OK)
                  printf("Read string from Page2:%s\r\n",strOut);
                LED1_OFF();
                LED2_OFF();
                LED3_OFF();
                LED4_ON();
          }
          break;

          case KEY_NONE:
          {
                LED1_OFF();
                LED2_OFF();
                LED3_OFF();
                LED4_OFF();
          }
          break;

          default:
          break;
        }
        printf("** Reselect menu or reset **\r\n");
        HAL_Delay(500);        //延时500,消除按键后抖动
  }
  /* USER CODE END 3 */
}

        上述程序中有一个I2C_HandleTypeDef类型的结构体变量hi2c2,这是表示I2C2接口的外设对象变量,24C02的驱动程序文件24cxx.c中就使用这个外设对象变量访问I2C2接口。函数MX_I2C2_Init()中对hi2c2的各成员变量赋值,各赋值语句与STM32CubeIDE里的设置是对应的。完成hi2c2的赋值后,执行HAL_I2C_Init(&hi2c2)对I2C2接口进行初始化。

        HAL_I2C_MspInit()是I2C接口的MSP初始化函数,在函数HAL_I2C_Init()里被调用。函数HAL_I2C_MspInit()的主要功能是对I2C2接口的复用引脚PF0和PF1进行GPIO引脚配置。

         由STM32CubeIDE自动生成的代码,这里省略。

/* USER CODE BEGIN 4 */

//串口打印
int __io_putchar(int ch)
{
        HAL_UART_Transmit(&huart6,(uint8_t*)&ch,1,0xFFFF);
        return ch;
}
/* USER CODE END 4 */
四、下载与运行
        下载,运行。依次按下开发板上的S2、S3、S4、S5按键,执行写入1字节,读出1字节、写入页字符串,读出页字符串。按下S6键,执行复位。



————————————————

                            版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

原文链接:https://blog.csdn.net/wenchm/article/details/144723242

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

98

主题

4181

帖子

2

粉丝