【N32L40X】学习笔记13-软件IIC读写EEPROM AT24C02

AT24C02

  1. 8个字节每页,累计32个页

  2. 通讯频率MAX = 400K

  3. AT24C02大小 2K

在这里插入图片描述

  1. 芯片地址

在这里插入图片描述
对于at24c02 A2A1A0 这三个引脚没有使用
在这里插入图片描述

在这里插入图片描述

  1. 写时序在这里插入图片描述

由于设备在写周期中不会产生ACK恢复,因此这可用于确定周期何时完成(此特性可用于最大限度地提高总线吞吐量)。一旦从主服务器发出了写命令的停止条件,设备就会启动内部定时的写周期,然后就可以立即启动ACK轮询。这涉及到主服务器发送一个开始条件,然后是一个写命令的控制字节(R/W =0)。如果设备仍忙于写入周期,则不会返回ACK。如果循环完成,设备将返回ACK,然后主命令可以继续下一个读或写命令。该操作的流程图见图5-1。

在这里插入图片描述

  1. 读时序
    在这里插入图片描述
    在这里插入图片描述

at24c2.h

#ifndef _24CXX_H
#define _24CXX_H
#include <stdint.h>#define AT24C01		127
#define AT24C02		255
#define AT24C04		511
#define AT24C08		1023
#define AT24C16		2047
#define AT24C32		4095
#define AT24C64	  8191
#define AT24C128	16383
#define AT24C256	32767//定义EE_TYPE为AT24C16
#define EE_TYPE AT24C16uint8_t AT24CXX_ReadOneByte(uint16_t ReadAddr);							//指定地址读取一个字节
void AT24CXX_WriteOneByte(uint16_t WriteAddr,uint8_t DataToWrite);		//指定地址写入一个字节void AT24CXX_WriteLenByte(uint16_t WriteAddr,uint32_t DataToWrite,uint8_t Len);//指定地址开始写入指定长度的数据
uint32_t AT24CXX_ReadLenByte(uint16_t ReadAddr,uint8_t Len);					//指定地址开始读取指定长度数据void AT24CXX_Write(uint16_t WriteAddr,uint8_t *pBuffer,uint16_t NumToWrite);	//从指定地址开始写入指定长度的数据
void AT24CXX_Read(uint16_t ReadAddr,uint8_t *pBuffer,uint16_t NumToRead);   	//从指定地址开始读出指定长度的数据
void AT24CXX_Erasure(uint16_t NumToWrite);  /*从地址0开始将指定个数的地址清0*/
uint8_t AT24CXX_Check(void);  //检查器件
void AT24CXX_Init(void); //初始化IIC#endif

at24c2.c

#include "at24c02.h"
#include "n32l40x.h"
//#include "delay.h"
//#include "debug.h"
//IO方向设置
//#define SDA_OUT {GPIOC->MODER |= 0x00040000;}  // 设置SDA为输出方向,对于双向I/O需切换为输出
//#define SDA_IN  {GPIOC->MODER &= 0xFFF3FFFF;}  // 设置SDA为输入方向,对于双向I/O需切换为输入
IO操作
//#define IIC_SCL   PAout(8) //SCL
//#define IIC_SDA   PCout(9) //SDA
//#define READ_SDA  PCin(9)  //输入SDA#define  SDA_OUT do{\GPIO_InitType GPIO_InitStructure;\GPIO_InitStruct(&GPIO_InitStructure);\GPIO_InitStructure.Pin            = GPIO_PIN_11;\GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_Out_PP;\GPIO_InitPeripheral(GPIOB, &GPIO_InitStructure);}while(0);#define  SDA_IN do{\GPIO_InitType GPIO_InitStructure;\GPIO_InitStruct(&GPIO_InitStructure);\GPIO_InitStructure.Pin            = GPIO_PIN_11;\GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_Input;\GPIO_InitPeripheral(GPIOB, &GPIO_InitStructure);}while(0);
//产生IIC起始信号
#define 	READ_SDA 	GPIO_ReadInputDataBit(GPIOB,GPIO_PIN_11)
#define 	IIC_SDA(x) 	GPIO_WriteBit(GPIOB,GPIO_PIN_11,x)
#define 	IIC_SCL(x) 	GPIO_WriteBit(GPIOB,GPIO_PIN_10,x)static void IIC_Init(void);
static void IIC_Start(void);
static void IIC_Stop(void);
static uint8_t IIC_Wait_Ack(void);
static void IIC_Ack(void);
static void IIC_NAck(void);
static void IIC_Send_Byte(uint8_t txd);
static uint8_t IIC_Read_Byte(unsigned char ack);void delay_us(int x)
{for(int i=0; i<x*60; i++);
}//初始化IIC接口
void AT24CXX_Init(void)
{
#if HW_WARE_24CXX >0printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifIIC_Init();//IIC初始化
}
//在AT24CXX指定地址读出一个数据
//ReadAddr:开始读数的地址
//返回值  :读到的数据
uint8_t AT24CXX_ReadOneByte(uint16_t ReadAddr)
{
#if HW_WARE_24CXX >1printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifuint8_t temp=0;IIC_Start();if(EE_TYPE>AT24C16){IIC_Send_Byte(0XA0);	   //发送写命令IIC_Wait_Ack();IIC_Send_Byte(ReadAddr>>8);//发送高地址} else IIC_Send_Byte(0XA0+((ReadAddr/256)<<1));   //发送器件地址0XA0,写数据IIC_Wait_Ack();IIC_Send_Byte(ReadAddr%256);   //发送低地址IIC_Wait_Ack();IIC_Start();IIC_Send_Byte(0XA1);           //进入接收模式IIC_Wait_Ack();temp=IIC_Read_Byte(0);IIC_Stop();//产生一个停止条件return temp;
}
//在AT24CXX指定地址写入一个数据
//WriteAddr  :写入数据的目的地址
//DataToWrite:要写入的数据
void AT24CXX_WriteOneByte(uint16_t WriteAddr,uint8_t DataToWrite)
{int page=0;
#if HW_WARE_24CXX >1printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifIIC_Start();if(EE_TYPE>AT24C16){IIC_Send_Byte(0XA0);	    //发送写命令10100000IIC_Wait_Ack();IIC_Send_Byte(WriteAddr>>8);//发送高地址} else{page = WriteAddr/256;
#if HW_WARE_24CXX >0printf("%s %s %d page = %02xH\r\n",__FILE__,__FUNCTION__,__LINE__,page);
#endifIIC_Send_Byte(0XA0+((WriteAddr/256)<<1));   //发送器件地址0XA0,写数据}IIC_Wait_Ack();IIC_Send_Byte(WriteAddr%256);   //发送低地址IIC_Wait_Ack();IIC_Send_Byte(DataToWrite);     //发送字节IIC_Wait_Ack();IIC_Stop();//产生一个停止条件delay_us(100);
}
//在AT24CXX里面的指定地址开始写入长度为Len的数据
//该函数用于写入16bit或者32bit的数据.
//WriteAddr  :开始写入的地址
//DataToWrite:数据数组首地址
//Len        :要写入数据的长度2,4
void AT24CXX_WriteLenByte(uint16_t WriteAddr,uint32_t DataToWrite,uint8_t Len)
{
#if HW_WARE_24CXX >0printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifuint8_t t;for(t=0; t<Len; t++){AT24CXX_WriteOneByte(WriteAddr+t,(DataToWrite>>(8*t))&0xff);}
}//在AT24CXX里面的指定地址开始读出长度为Len的数据
//该函数用于读出16bit或者32bit的数据.
//ReadAddr   :开始读出的地址
//返回值     :数据
//Len        :要读出数据的长度2,4
uint32_t AT24CXX_ReadLenByte(uint16_t ReadAddr,uint8_t Len)
{
#if HW_WARE_24CXX >0printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifuint8_t t;uint32_t temp=0;for(t=0; t<Len; t++){temp<<=8;temp+=AT24CXX_ReadOneByte(ReadAddr+Len-t-1);}return temp;
}
//检查AT24CXX是否正常
//这里用了24XX的最后一个地址(255)来存储标志字.
//如果用其他24C系列,这个地址要修改
//返回1:检测失败
//返回0:检测成功
uint8_t AT24CXX_Check(void)
{
#if HW_WARE_24CXX > 0printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifuint8_t temp;temp=AT24CXX_ReadOneByte(255);//避免每次开机都写AT24CXXif(temp==0X55)return 0;else//排除第一次初始化的情况{AT24CXX_WriteOneByte(255,0X55);temp=AT24CXX_ReadOneByte(255);if(temp==0X55)return 0;}return 1;
}//在AT24CXX里面的指定地址开始读出指定个数的数据
//ReadAddr :开始读出的地址 对24c02为0~255
//pBuffer  :数据数组首地址
//NumToRead:要读出数据的个数
void AT24CXX_Read(uint16_t ReadAddr,uint8_t *pBuffer,uint16_t NumToRead)
{
#if HW_WARE_24CXX >0printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifwhile(NumToRead){*pBuffer++=AT24CXX_ReadOneByte(ReadAddr++);NumToRead--;}
}
//在AT24CXX里面的指定地址开始写入指定个数的数据
//WriteAddr :开始写入的地址 对24c02为0~255
//pBuffer   :数据数组首地址
//NumToWrite:要写入数据的个数
void AT24CXX_Write(uint16_t WriteAddr,uint8_t *pBuffer,uint16_t NumToWrite)
{
#if HW_WARE_24CXX >1printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifwhile(NumToWrite--){AT24CXX_WriteOneByte(WriteAddr,*pBuffer);WriteAddr++;pBuffer++;}
}//在AT24CXX里面的指定地址开始写入指定个数的数据
//WriteAddr :开始写入的地址 对24c02为0~255
//pBuffer   :数据数组首地址
//NumToWrite:要写入数据的个数
void AT24CXX_Erasure(uint16_t NumToWrite)
{
#if HW_WARE_24CXX >0printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifuint16_t  WriteAddr=0;while(NumToWrite--){AT24CXX_WriteOneByte(WriteAddr,0);WriteAddr++;}
}/*软件模拟IIC*/
//IIC初始化
static void IIC_Init(void)
{
#if HW_WARE_24CXX >0printf("%s %s %d\r\n",__FILE__,__FUNCTION__,__LINE__);
#endifRCC_EnableAPB2PeriphClk(RCC_APB2_PERIPH_GPIOB,ENABLE);GPIO_InitType GPIO_InitStructure;/* Initialize GPIO_InitStructure */GPIO_InitStruct(&GPIO_InitStructure);/* Configure USARTx Tx as alternate function push-pull */GPIO_InitStructure.Pin            = GPIO_PIN_10;GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_Out_PP;GPIO_InitPeripheral(GPIOB, &GPIO_InitStructure);GPIO_InitStructure.Pin            = GPIO_PIN_11;GPIO_InitStructure.GPIO_Mode      = GPIO_Mode_Out_PP;GPIO_InitPeripheral(GPIOB, &GPIO_InitStructure);GPIO_WriteBit(GPIOB,GPIO_PIN_10,Bit_SET);GPIO_WriteBit(GPIOB,GPIO_PIN_11,Bit_SET);}static void IIC_Start(void)
{SDA_OUT;     //sda线输出IIC_SDA(1);IIC_SCL(1);delay_us(4);IIC_SDA(0);//START:when CLK is high,DATA change form high to lowdelay_us(4);IIC_SCL(0);//钳住I2C总线,准备发送或接收数据
}//产生IIC停止信号
static void IIC_Stop(void)
{SDA_OUT;//sda线输出IIC_SCL(0);IIC_SDA(0);//STOP:when CLK is high DATA change form low to highdelay_us(4);IIC_SCL(1);delay_us(4);IIC_SDA(1);//发送I2C总线结束信号
}//等待应答信号到来
//返回值:1,接收应答失败
//        0,接收应答成功
static uint8_t IIC_Wait_Ack(void)
{uint8_t ucErrTime=0;SDA_IN;      //SDA设置为输入IIC_SDA(1);delay_us(1);IIC_SCL(1);delay_us(1);while(READ_SDA){ucErrTime++;if(ucErrTime>250){IIC_Stop();return 1;}}IIC_SCL(0);//时钟输出0return 0;
}//产生ACK应答
static void IIC_Ack(void)
{IIC_SCL(0);SDA_OUT;IIC_SDA(0);delay_us(2);IIC_SCL(1);delay_us(2);IIC_SCL(0);
}
//不产生ACK应答
static void IIC_NAck(void)
{IIC_SCL(0);SDA_OUT;IIC_SDA(1);delay_us(2);IIC_SCL(1);delay_us(2);IIC_SCL(0);
}
//IIC发送一个字节
//返回从机有无应答
//1,有应答
//0,无应答
static void IIC_Send_Byte(uint8_t txd)
{uint8_t t;SDA_OUT;IIC_SCL(0);//拉低时钟开始数据传输for(t=0; t<8; t++){IIC_SDA((txd&0x80)>>7);txd<<=1;delay_us(2);   //对TEA5767这三个延时都是必须的IIC_SCL(1);delay_us(2);IIC_SCL(0);delay_us(2);}
}
//读1个字节,ack=1时,发送ACK,ack=0,发送nACK
static uint8_t IIC_Read_Byte(unsigned char ack)
{unsigned char i,receive=0;SDA_IN;//SDA设置为输入for(i=0; i<8; i++ ){IIC_SCL(0);delay_us(2);IIC_SCL(1);receive<<=1;if(READ_SDA)receive++;delay_us(1);}if (!ack)IIC_NAck();//发送nACKelseIIC_Ack(); //发送ACKreturn receive;
}

测试代码

    AT24CXX_Init();if(!AT24CXX_Check()){printf("AT24CXX_Check ok\r\n");}else{printf("AT24CXX_Check fail\r\n");}uint8_t buff[100];uint8_t buff1[100];for(int i=0; i<100; i++){buff[i]=i;}AT24CXX_Write(0,buff,100);AT24CXX_Read(0,buff1,100);for(int i=0; i<100; i++){printf("%02d ",buff1[i]);}printf("\r\n");

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/77348.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

机器学习笔记 - YOLO-NAS 最高效的目标检测算法之一

一、YOLO-NAS概述 YOLO(You Only Look Once)是一种对象检测算法,它使用深度神经网络模型,特别是卷积神经网络,来实时检测和分类对象。该算法首次在 2016 年由 Joseph Redmon、Santosh Divvala、Ross Girshick 和 Ali Farhadi 发表的论文《You Only Look Once: Unified, Re…

2023-08-05——JVM 栈

栈 stack 栈&#xff1a;数据结构 程序数据结构算法 栈&#xff1a;先进后出&#xff0c;后进先出 好比一个&#xff1a;桶 队列&#xff1a;先进先出&#xff08;FIFO &#xff1a;First Input First Out&#xff09; 好比一个&#xff1a;管道 栈&#xff1a;喝多了吐。队列…

抽象轻松JavaScript

随心所欲的数组切割与改变2.0版本 splice(开始位置&#xff0c;删除数量&#xff0c;添加内容) slice(开始位置&#xff0c;结束位置) 目的&#xff0c;上面的一串小字&#xff0c;切割与改变 切割代表删&#xff0c;改变代表增、改&#xff0c;随心所欲的切割与改变意味着不…

分布式应用:Zookeeper 集群与kafka 集群部署

目录 一、理论 1.Zookeeper 2.部署 Zookeeper 集群 3.消息队列 4.Kafka 5.部署 kafka 集群 6.FilebeatKafkaELK 二、实验 1.Zookeeper 集群部署 2.kafka集群部署 3.FilebeatKafkaELK 三、问题 1.解压文件异常 2.kafka集群建立失败 3.启动 filebeat报错 4.VIM报错…

页面技术基础-html

页面技术基础-html 环境准备&#xff1a;在JDBC中项目上完成代码定义 1. 新建一个 Module:filr->右键 -》Module -》Java-》next->名字(html_day1)->finish 2. 在 Moudle上右键-》第二个选项&#xff1a;add framework .. -> 选择JavaEE下第一个选项 Web Apllicat…

mysql高级(尚硅谷-夏磊)

目录 内容介绍 Linux下MySQL的安装与使用 Mysql逻辑架构 Mysql存储引擎 Sql预热 索引简介 内容介绍 1、Linux下MySQL的安装与使用 2、逻辑架构 3、sql预热 Linux下MySQL的安装与使用 1、docker安装docker run -d \-p 3309:3306 \-v /atguigu/mysql/mysql8/conf:/etc/my…

Xilinx A7开发板LVDS IO无输出问题解决方法

使用A7-35T FGG484的FPGA开发板bank16上的IO作为差分LVDS的输入输出&#xff0c;搭建输入输出测试工程发现LVDS可以输入、无法输出。查阅UG471&#xff0c;找到如下信息&#xff1a; 手册中已经针对A7的LVDS做了明确的应用说明&#xff1a; &#xff08;1&#xff09;HP bank上…

观察者模式——对象间的联动

1、简介 1.1、概述 在软件系统中&#xff0c;有些对象之间也存在类似交通信号灯和汽车之间的关系。一个对象的状态或行为的变化将导致其他对象的状态或行为也发生改变&#xff0c;它们之间将产生联动&#xff0c;正所谓“触一而牵百发”。为了更好地描述对象之间存在的这种一…

第一个maven项目(IDEA生成)

第一个maven项目&#xff08;IDEA生成&#xff09; 步骤1 配置Project SDK 步骤2 配置maven File->Settings搜索maven

CHI(六)独占访问

AMBA5 CHI Architecture Specification IssueF 1. overview 独占访问的原则是&#xff0c;执行独占序列的逻辑处理器&#xff08;LP&#xff09;执行以下操作&#xff1a; 对一个地址执行exclusive load。计算要存储到该位置的新值。对该地址进行exclusive store。 支持对可…

Java并发系列之二:悲观锁机制

什么是锁 在并发环境下&#xff0c;会出现多个线程对同一个资源进行争抢的情况&#xff0c;假设A线程对资源正在进行修改&#xff0c;此时B线程此时又对资源进行了修改&#xff0c;这就可能会导致数据不一致的问题。为了解决这个问题&#xff0c;很多编程语言引入了锁机制&…

RocketMQ发送消息超时异常

说明&#xff1a;在使用RocketMQ发送消息时&#xff0c;出现下面这个异常&#xff08;org.springframework.messging.MessgingException&#xff1a;sendDefaultImpl call timeout……&#xff09;&#xff1b; 解决&#xff1a;修改RocketMQ中broke.conf配置&#xff0c;添加下…

【JAVASE】多态

⭐ 作者&#xff1a;小胡_不糊涂 &#x1f331; 作者主页&#xff1a;小胡_不糊涂的个人主页 &#x1f4c0; 收录专栏&#xff1a;浅谈Java &#x1f496; 持续更文&#xff0c;关注博主少走弯路&#xff0c;谢谢大家支持 &#x1f496; 多态 1. 概念2. 实现条件3. 重写4. 向上…

wordpress 学习贴

安装问题 我的使用环境为docker环境&#xff0c;php、nginx、mysql分别处于3个容器中&#xff0c; 提示异常&#xff0c;打开debug模式&#xff0c;会发现 No such file or directory Warning: mysqli_real_connect(): (HY000/2002): No such file or directory 这个其实问题其…

【C++】开源:sqlite3数据库配置使用

&#x1f60f;★,:.☆(&#xffe3;▽&#xffe3;)/$:.★ &#x1f60f; 这篇文章主要介绍sqlite3数据库配置使用。 无专精则不能成&#xff0c;无涉猎则不能通。——梁启超 欢迎来到我的博客&#xff0c;一起学习&#xff0c;共同进步。 喜欢的朋友可以关注一下&#xff0c;下…

【FAQ】如何隐藏网页H.265播放器EasyPlayer.js的实时录像按钮?

目前我们TSINGSEE青犀视频所有的视频监控平台&#xff0c;集成的都是EasyPlayer.js版播放器&#xff0c;它属于一款高效、精炼、稳定且免费的流媒体播放器&#xff0c;可支持多种流媒体协议播放&#xff0c;包括WebSocket-FLV、HTTP-FLV&#xff0c;HLS&#xff08;m3u8&#x…

MySQL建表和增添改查

1.创建一个名为mydb的数据库 mysql> show database mydb; 查询 mysql> show database mydb; 2.创建一个学生信息表 mysql> create table mydb.student_informtion( -> student_id int UNSIGNED NOT NULL PRIMARY KEY, //非空&#xff08;不允许为空&#xff0…

unity如何手动更改脚本执行顺序

在Unity中&#xff0c;脚本的执行顺序是由脚本的执行顺序属性决定的。默认情况下&#xff0c;Unity根据脚本在项目中的加载顺序来确定它们的执行顺序。然而&#xff0c;你可以手动更改脚本的执行顺序&#xff0c;以下是一种方法&#xff1a; 在Unity编辑器中&#xff0c;选择你…

PCIE上位机用什么工具?

可以了解一下神电测控出器的My FPGA开发套件&#xff0c;它可以用来开发FPGA板卡与上位机之间PCIE通信&#xff0c;而且就是用LabVIEW FPGA开发。它使用的是Xillybus PCIe IP核&#xff0c;神电将其封装成了在 LabVIEW FPGA 下的 PCIe CLIP 组件&#xff0c;可以方便的使用。而…

深度学习之tensorboard可视化工具

(1)什么是tensorboard tensorboard是TensorFlow 的一个可视化工具包&#xff0c;提供机器学习实验所需的可视化和工具&#xff0c;该工具的功能如下&#xff1a; 跟踪和可视化指标&#xff0c;例如损失和精度可视化模型图&#xff08;操作和层&#xff09;查看权重、偏差或其…