热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

i2c扩展32路gpio_【STM32Cube_13】使用硬件I2C读写EEPROM(AT24C02)

寻求更好的阅读体验,请移步Mculover666的个人博客:【STM32Cube_13】使用硬件I2C读写EEPROM(AT24C02&#
9827a2059e2a21e8f41ac25c1d56d79a.png

寻求更好的阅读体验,请移步Mculover666的个人博客:

【STM32Cube_13】使用硬件I2C读写EEPROM(AT24C02)​www.mculover666.cn
f6e775d21695398ab8a91c29a12f194f.png

本篇详细的记录了如何使用STM32CubeMX配置STM32L431RCT6的硬件I2C外设读取EEPROM数据(以AT24C02为例)。

1. 准备工作

硬件准备

  • 开发板

首先需要准备一个开发板,这里我准备的是STM32L4的开发板(BearPi):

a508db256c6658e5e4ed244d1a05ee08.png
  • EEPROM

小熊派开发板左边的接口是E53接口,用来连接E53接口的扩展板,每个扩展板都板载了一块EEPROM用来保存信息,如图:

b59e45b595e94681a83655b486811687.png

AT24C02的原理图如下(该原理图中有bug,A0的上拉电阻无效,实际A0为低电平):

d6519003464b964d075d39716f0a5707.png

软件准备

  • 需要安装好Keil - MDK及芯片对应的包,以便编译和下载生成的代码;
Keil MDK和串口助手Serial Port Utility 的安装包都可以在文末关注公众号获取,回复关键字获取相应的安装包:
cc1901670a11b36d8bedfee40cc07cbb.png

2.生成MDK工程

选择芯片型号

打开STM32CubeMX,打开MCU选择器:

14695f389e7fc0dc64eca38f306a87e7.png

搜索并选中芯片STM32L431RCT6:

83da32db92c1be29247181bddd71951d.png

配置时钟源

  • 如果选择使用外部高速时钟(HSE),则需要在System Core中配置RCC;
  • 如果使用默认内部时钟(HSI),这一步可以略过;

这里我都使用外部时钟:

fe7e792db0c1775c0289f56b373dfa94.png

配置串口

小熊派开发板板载ST-Link并且虚拟了一个串口,原理图如下:

4bc5151aa0370bb3d85bbf9543414ff8.png

这里我将开关拨到AT-MCU模式,使PC的串口与USART1之间连接。

接下来开始配置USART1:

562219632088df82e303d06358960666.png

配置硬件I2C

首先查看小熊派开发板的原理图,确定EEPROM接在哪个I2C接口上,如图:

559ed85e43ab2f5070d3882a7847ab2c.png

接下来开始配置I2C接口1:

9751fe5d27e8ff2557ac28ab02887bcb.png

配置时钟树

STM32L4的最高主频到80M,所以配置PLL,最后使HCLK = 80Mhz即可:

ae27df34580db536d2f623c02792c29c.png

生成工程设置

e2c956ebb48563d73ad216aaded60bac.png

代码生成设置

最后设置生成独立的初始化文件:

c0e6a702d0e715bc6e5d604e36bdb6ee.png

生成代码

点击GENERATE CODE即可生成MDK-V5工程:

af3bf8c673586d0ce7a13ce59135fd06.png

3. 在MDK中编写、编译、下载用户代码

修改I2C初始化代码的小BUG

4e54a5bc1b440cbf974a63230510d478.png

重定向printf( )函数

参考:【STM32Cube_09】重定向printf函数到串口输出的多种方法。

编写EEPROM驱动程序

EEPROM的驱动编写篇幅过多,单独分出来一节讲述。

4. AT24C02驱动的编写

确定IIC器件地址

根据AT24C02的 Datasheet 可知AT24C02有2K bit,即256B,分为32页,每页8个字节,结合数据手册和原理图可以得知,板载AT24C02的读地址为0xA2,写地址为0xA3:

371b4fff1f92c744ccc3ad209f0955af.png

首先在at24c02_i2c_drv.h中编写AT24C02相关的宏定义:

#define AT24C02_ADDR_WRITE 0xA0
#define AT24C02_ADDR_READ 0xA1

然后在at24c02_i2c_drv.c中引入i2c.h,基于HAL提供的硬件IIC操作函数,编写AT24C02的一些底层函数,如下。

任意地址写一个字节

根据AT24C02的数据手册可知,AT24C02写一个字节的格式如下:

bf3541cf6fa22c73d9ba01cd4685a5c2.png

编写的函数如下:

/*** @brief AT24C02任意地址写一个字节数据* @param addr —— 写数据的地址(0-255)* @param dat —— 存放写入数据的地址* @retval 成功 —— HAL_OK
*/
uint8_t At24c02_Write_Byte(uint16_t addr, uint8_t* dat)
{return HAL_I2C_Mem_Write(&hi2c1, AT24C02_ADDR_WRITE, addr, I2C_MEMADD_SIZE_8BIT, dat, 1, 0xFFFF);
}

任意地址读一个字节

根据AT24C02的数据手册可知,AT24C02读一个字节的格式如下:

496dfc1cfcaad2f08a48597e98d6b6c2.png

编写的函数如下:

/*** @brief AT24C02任意地址读一个字节数据* @param addr —— 读数据的地址(0-255)* @param read_buf —— 存放读取数据的地址* @retval 成功 —— HAL_OK
*/
uint8_t At24c02_Read_Byte(uint16_t addr, uint8_t* read_buf)
{return HAL_I2C_Mem_Read(&hi2c1, AT24C02_ADDR_READ, addr, I2C_MEMADD_SIZE_8BIT, read_buf, 1, 0xFFFF);
}

测试字节读写函数

main.c中测试:

int main(void)
{uint8_t write_dat = 0xa5;uint8_t recv_buf = 0;HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_I2C1_Init();MX_USART1_UART_Init();if(HAL_OK == At24c02_Write_Byte(10,&write_dat)){printf("Write okn");}else{printf("Write failn");}HAL_Delay(50); //写一次和读一次之间需要短暂的延时if(HAL_OK == At24c02_Read_Byte(10,&recv_buf)){printf("Read ok, recv_buf = 0x%02Xn", recv_buf);}else{printf("Read failn");}while(1);

测试结果如下:

8d8bf44661b457e3b52bfe2ced568923.png

任意地址连续写多个字节

AT24C02连续写字节的时候需要注意,不能使用写单个字节函数连续的写入,因为AT24C02分为了32页,每页是8个字节,如果连续的单字节写入8个字节后,会重复的继续往该页写数据,所以要使用如下的写一页的格式:

222764d5a8476851ccd8bc60ff24f137.png

/*** @brief AT24C02任意地址连续写多个字节数据* @param addr —— 写数据的地址(0-255)* @param dat —— 存放写入数据的地址* @retval 成功 —— HAL_OK
*/
uint8_t At24c02_Write_Amount_Byte(uint16_t addr, uint8_t* dat, uint16_t size)
{uint8_t i &#61; 0;uint16_t cnt &#61; 0; //写入字节计数/* 对于起始地址&#xff0c;有两种情况&#xff0c;分别判断 */if(0 &#61;&#61; addr % 8 ){/* 起始地址刚好是页开始地址 *//* 对于写入的字节数&#xff0c;有两种情况&#xff0c;分别判断 */if(size <&#61; 8){//写入的字节数不大于一页&#xff0c;直接写入return HAL_I2C_Mem_Write(&hi2c1, AT24C02_ADDR_WRITE, addr, I2C_MEMADD_SIZE_8BIT, dat, size, 0xFFFF);}else{//写入的字节数大于一页&#xff0c;先将整页循环写入for(i &#61; 0;i }

任意地址连续读多个字节

AT24C02连续读多个字节没有限制&#xff0c;直接读取即可&#xff0c;代码如下&#xff1a;

/*** &#64;brief AT24C02任意地址连续读多个字节数据* &#64;param addr —— 读数据的地址&#xff08;0-255&#xff09;* &#64;param dat —— 存放读出数据的地址* &#64;retval 成功 —— HAL_OK
*/
uint8_t At24c02_Read_Amount_Byte(uint16_t addr, uint8_t* recv_buf, uint16_t size)
{return HAL_I2C_Mem_Read(&hi2c1, AT24C02_ADDR_READ, addr, I2C_MEMADD_SIZE_8BIT, recv_buf, size, 0xFFFF);
}

测试任意地址连续读写多个字节

main.c中测试&#xff1a;

int main(void)
{uint8_t write_dat[22] &#61; {0};uint8_t recv_buf[22] &#61; {0};HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_I2C1_Init();MX_USART1_UART_Init();for(i &#61; 0;i <22; i&#43;&#43;){write_dat[i] &#61; i;printf("%02X ", write_dat[i]);if((i&#43;1) % 16 &#61;&#61; 0){printf("n");}}if(HAL_OK &#61;&#61; At24c02_Write_Amount_Byte(0, write_dat, 22)){printf("write okn");}else{printf("write failn");}HAL_Delay(50);if(HAL_OK &#61;&#61; HAL_I2C_Mem_Read(&hi2c1, AT24C02_ADDR_READ, 0, I2C_MEMADD_SIZE_8BIT, recv_buf, 22, 0xFFFF)){printf("read okn");for(i &#61; 0; i <22; i&#43;&#43;){printf("0x%02X ", recv_buf[i]);if((i&#43;1) % 8 &#61;&#61; 0){printf("n");}}}else{printf("read failn");}while(1);

测试结果&#xff1a;

7e3e4b3273832fc04f47f7c48e8c2556.png

将上面的读写地址由0改为5&#xff0c;再次测试&#xff1a;

if(HAL_OK &#61;&#61; At24c02_Write_Amount_Byte(5, write_dat, 22))

测试结果&#xff1a;

0e6132e97ee8a2831d0a1623a18db369.png

至此&#xff0c;我们已经学会如何使用硬件IIC接口读写EEPROM&#xff0c;下一节将讲述如何使用硬件IIC接口读取环境光强度传感器数据&#xff08;BH1750&#xff09;。

更多精彩文章及资源&#xff0c;请关注我的微信公众号&#xff1a;『mculover666』。

e9b84a19b1d0977bbba6d8b279ac2932.png



推荐阅读
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 360SRC安全应急响应:从漏洞提交到修复的全过程
    本文详细介绍了360SRC平台处理一起关键安全事件的过程,涵盖从漏洞提交、验证、排查到最终修复的各个环节。通过这一案例,展示了360在安全应急响应方面的专业能力和严谨态度。 ... [详细]
  • 本文详细解析了Python中的os和sys模块,介绍了它们的功能、常用方法及其在实际编程中的应用。 ... [详细]
  • Java编程实践:深入理解方法重载
    本文介绍了Java中方法重载的概念及其应用。通过多个示例,详细讲解了如何在同一类中定义具有相同名称但不同参数列表的方法,以实现更灵活的功能调用。 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文详细介绍了macOS系统的核心组件,包括如何管理其安全特性——系统完整性保护(SIP),并探讨了不同版本的更新亮点。对于使用macOS系统的用户来说,了解这些信息有助于更好地管理和优化系统性能。 ... [详细]
  • Google最新推出的嵌入AI技术的便携式相机Clips现已上架,旨在通过人工智能技术自动捕捉用户生活中值得纪念的时刻,帮助人们减少照片数量过多的问题。 ... [详细]
  • 本文探讨了 C++ 中普通数组和标准库类型 vector 的初始化方法。普通数组具有固定长度,而 vector 是一种可扩展的容器,允许动态调整大小。文章详细介绍了不同初始化方式及其应用场景,并提供了代码示例以加深理解。 ... [详细]
  • 深入理解Java泛型:JDK 5的新特性
    本文详细介绍了Java泛型的概念及其在JDK 5中的应用,通过具体代码示例解释了泛型的引入、作用和优势。同时,探讨了泛型类、泛型方法和泛型接口的实现,并深入讲解了通配符的使用。 ... [详细]
author-avatar
沉醉在温柔箱
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有