自学内容网 自学内容网

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

目录

一、操作说明 

二、工程配置

1、时钟、DEBUG、GPIO、USART6、NVIC、Code Generator

2、 IIC2

(1)Master Features组,主设备参数

(2)Slave Features组,从设备参数

三、软件设计

1、KELED

2、EEPROM

(1)24cxx.c

(2)24cxx.h

3、main.c

四、下载与运行


        本文旨在通过使用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、KELED

        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"
#defineEP24C_TIMEOUT 200//超时等待时间,单位:ms
#defineEP24C_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);
returnresult;
}

//从任意地址读出一个字节的数据, 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
#defineDEV_ADDR_24CXX 0x00A0//24C02的写地址
// EEPROM存储器参数
#definePAGE_SIZE_24CXX 0x0008//24C02的Page大小为8字节
#defineMEM_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键,执行复位。


原文地址:https://blog.csdn.net/wenchm/article/details/144723242

免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!