如何利用AVR单片机实现AT24C256的数据高速稳定的读取

2020-01-15来源: elecfans关键字:AVR单片机  AT24C256  数据  读取

最近刚买回一块AT24C256 EEPROM ,容量为32K Byte ,数据地址宽度为 16Bit ,支持IIC 1M (5V)400K (2.7V) 速度模式 ,利用AVR M16 片内IIC 可以高速稳定地读取数据! 经过调试的,与各位大虾分享分享。

如何利用AVR单片机实现AT24C256的数据高速稳定的读取
 

程序如下:(winavr)

#include

#include

#include

#include

#define FREQ 8

#include

#include

#include

#define uchar unsigned char

#define uint unsigned int

#define ulong unsigned long

//管脚定义

#define pinSCL 0 //PC0 SCL

#define pinSDA 1 //PC1 SDA

//为保险起见,最好在SCL/SDA接上1~10K的外部上拉电阻到VCC。

#define fSCL 1000000 //TWI时钟为1000KHz

//预分频系数=1(TWPS=0)

#if F_CPU 《 fSCL*36

#define TWBR_SET 2; //TWBR必须大于等于10

#else

#define TWBR_SET (F_CPU/fSCL-16)/2; //计算TWBR值

#endif

#define TW_ACT (1《//TWCR只能IN/OUT,直接赋值比逻辑运算(|= &=)更节省空间

#define SLA_24CXX 0xA0 //24Cxx系列的厂商器件地址(高四位)

#define ADDR_24C256 0x00

// AT24C256的地址线A2/1/0全部接地,SLAW=0xA0+0x00《《1+0x00,SLAR=0xA0+0x00《《1+0x01

//TWI_操作状态

#define TW_BUSY 0

#define TW_OK 1

#define TW_FAIL 2

//TWI_读写命令状态

#define OP_BUSY 0

#define OP_RUN 1

//TWI读写操作公共步骤

#define ST_FAIL 0 //出错状态

#define ST_START 1 //START状态检查

#define ST_SLAW 2 //SLAW状态检查

#define ST_WADDR_H 3 //ADDR状态检查

#define ST_WADDR_L 4 //ADDR状态检查

//TWI读操作步骤

#define ST_RESTART 5 //RESTART状态检查

#define ST_SLAR 6 //SLAR状态检查

#define ST_RDATA 7 //读取数据状态检查,循环n字节

//TWI写操作步骤

#define ST_WDATA 8 //写数据状态检查,循环n字节

#define FAIL_MAX 1 //重试次数最大值

void delay_nms(uint ms)//若干毫秒延时

{

int i;

for(i=0;i{

_delay_loop_2(FREQ*250);

}

}

unsigned char TWI_RW(unsigned char sla,unsigned int addr,unsigned char *ptr,unsigned int len);

unsigned char BUFFER[256]; //缓冲区

void Test(void);

struct str_TWI //TWI数据结构

{

volatile unsigned char STATUS; //TWI_操作状态

unsigned char SLA; //从设备的器件地址

unsigned char ADDR_H; //从设备的数据地址

unsigned char ADDR_L; //从设备的数据地址

unsigned char *pBUF; //数据缓冲区指针

unsigned int DATALEN; //数据长度

unsigned char STATE; //TWI读写操作步骤

unsigned char FAILCNT; //失败重试次数

};

struct str_TWI strTWI; //TWI的数据结构变量

//AT24C256的读写函数(包括随机读,连续读,字节写,页写)

//根据sla的最低位决定(由中断程序中判断)

//bit0=1 TW_READ 读

//bit0=0 TW_WRITE 写

// sla 器件地址(不能搞错)

// addr EEPROM地址(0~32767)

// *ptr 读写数据缓冲区

// len 读数据长度(1~32768),写数据长度(1 or 8 or 16 or 32 or 64)

// 返回值 是否能执行当前操作

unsigned char TWI_RW(unsigned char sla,unsigned int addr,unsigned char *ptr,unsigned int len)

{

// unsigned char i;

if (strTWI.STATUS==TW_BUSY)

{//TWI忙,不能进行操作

return OP_BUSY;

}

strTWI.STATUS=TW_BUSY;

strTWI.SLA=sla;

strTWI.ADDR_H=(unsigned char)((addr》》8)&0xff);

strTWI.ADDR_L=(unsigned char)(addr&0xff);

strTWI.pBUF=ptr;

strTWI.DATALEN=len;

strTWI.STATE=ST_START;

strTWI.FAILCNT=0;

TWCR=(1《 return OP_RUN;

}

SIGNAL(SIG_2WIRE_SERIAL)

{//IIC中断

unsigned char acTIon,state,status;

acTIon=strTWI.SLA&TW_READ; //取操作模式

state=strTWI.STATE;

status=TWSR&0xF8; //屏蔽预分频位

if ((status》=0x60)||(status==0x00))

{//总线错误或从机模式引发的中断,不予处理

return;

}

switch(state)

{

case ST_START: //START状态检查

if(status==TW_START)

{//发送start信号成功

TWDR=strTWI.SLA&0xFE; //发送器件地址写SLAW

TWCR=TW_ACT; //触发下一步动作,同时清start发送标志

}

else

{//发送start信号出错

state=ST_FAIL;

}

break;

case ST_SLAW: //SLAW状态检查

if(status==TW_MT_SLA_ACK)

{//发送器件高位地址成功

TWDR=strTWI.ADDR_H; //发送eeprom地址

TWCR=TW_ACT; //触发下一步动作

}

else

{//发送器件地址出错

state=ST_FAIL;

}

break;

case ST_WADDR_H: //ADDR状态检查

if(status==TW_MT_DATA_ACK)

{//发送器件低位地址成功

TWDR=strTWI.ADDR_L; //发送eeprom地址

TWCR=TW_ACT; //触发下一步动作

}

else

{//发送器件地址出错

state=ST_FAIL;

}

break;

case ST_WADDR_L: //ADDR状态检查

if(status==TW_MT_DATA_ACK)

{//发送eeprom地址成功

if (acTIon==TW_READ)

{//读操作模式

TWCR=(1《 }

else

{//写操作模式

TWDR=*strTWI.pBUF++; //写第一个字节

strTWI.DATALEN--;

state=ST_WDATA-1; //下一步将跳到WDATA分支

TWCR=TW_ACT; //触发下一步动作

}

}

else

{//发送eeprom地址出错

state=ST_FAIL;

}

break;

case ST_RESTART: //RESTART状态检查,只有读操作模式才能跳到这里

if(status==TW_REP_START)

{//发送restart信号成功

TWDR=strTWI.SLA; //发器件地址读SLAR

TWCR=TW_ACT; //触发下一步动作,同时清start发送标志

}

else

{//重发start信号出错

state=ST_FAIL;

}

break;

case ST_SLAR: //SLAR状态检查,只有读操作模式才能跳到这里

if(status==TW_MR_SLA_ACK)

{//发送器件地址成功

if (strTWI.DATALEN--)

{//多个数据

TWCR=(1《 }

else

{//只有一个数据

TWCR=TW_ACT; //设定NAK,触发下一步动作

}

}

else

{//发送器件地址出错

state=ST_FAIL;

}

break;

case ST_RDATA: //读取数据状态检查,只有读操作模式才能跳到这里

state--; //循环,直到读完指定长度数据

if(status==TW_MR_DATA_ACK)

{//读取数据成功,但不是最后一个数据

*strTWI.pBUF++=TWDR;

if (strTWI.DATALEN--)

{//还有多个数据

TWCR=(1《 }

else

{//准备读最后一个数据

TWCR=TW_ACT; //设定NAK,触发下一步动作

}

}

else if(status==TW_MR_DATA_NACK)

{//已经读完最后一个数据

*strTWI.pBUF++=TWDR;

TWCR=(1《 strTWI.STATUS=TW_OK;

}

else

{//读取数据出错

state=ST_FAIL;

}

break;

case ST_WDATA: //写数据状态检查,只有写操作模式才能跳到这里

state--; //循环,直到写完指定长度数据

if(status==TW_MT_DATA_ACK)

{//写数据成功

if (strTWI.DATALEN)

{//还要写

TWDR=*strTWI.pBUF++;

strTWI.DATALEN--;

TWCR=TW_ACT; //触发下一步动作

}

else

{//写够了

TWCR=(1《 strTWI.STATUS=TW_OK;

//启动写命令后需要10ms(最大)的编程时间才能真正的把数据记录下来

//编程期间器件不响应任何命令

}

}

else

{//写数据失败

state=ST_FAIL;

}

break;

default:

//错误状态

state=ST_FAIL;

break;

}

关键字:AVR单片机  AT24C256  数据  读取 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/ic485759.html 本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:基于AVR单片机硬件多机通讯功能的实现
下一篇:avr单片机的数码管显示时钟设计

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

巧用avr单片机设计轮胎内径测量系统
。将光栅尺读头与激光测距仪固定在机械横梁上,运用步进电机控制横梁的运动,分别对模具不同层面的内径进行测量。    系统采用AVR单片机实现控制步进电机和光栅尺数据读取,通过接收上位机的控制命令,AVR单片机控制步进电机运动,数显表数据和激光控制器位移数据自动经串口发送给上位机,从而完成对模具内径的自动测量。    1 系统组成    基于AVR的轮胎内径测量系统主要由AVR单片机、上位机、光栅尺、数显表、激光测距仪、驱动器、步进电机、电子手轮、行程开关等组成。其功能框图如图1所示。    单片机选用的是爱特梅尔公司的ATmega16;上位机采用研华公司
发表于 2020-01-09
巧用avr单片机设计轮胎内径测量系统
AVR单片机汇编器的部分伪指令详解
伪指令不属于单片机的指令系统,而是由汇编器提供的指令,用于调整存储器中程序的位置、定义宏、初始化存储器等。AVR单片机的汇编器共提供18条伪指令(见附表)。其中,ORG、DB、DW、EQU读者比较熟悉,这里不再赘述。下面对部分伪指令加以说明。BYTE-保存单字节数据到SRAM中。BYTE伪指令仅用在数据存储器。为提供数据保存的位置,在BYTE前应有标号。在由CSEG、ESEG定义的代码段和E2PROM段中不能使用BYTE伪指令。格式LABEL:.BYTE表达式CSEG-定义程序存储器代码段的起始位置一个汇编文件可以包括若干个代码段,汇编时这些代码段被连成一个代码段。在代码段中不能使用BYTE伪指令。格式.CSEGDSEG-定义
发表于 2020-01-09
AVR单片机汇编器的部分伪指令详解
AVR单片机8位数码管显示的程序实现(两种方法介绍)
本文为大家介绍两个AVR单片机8位数码管显示的程序实现。AVR单片机595驱动8位数码管的显示的电路实现主程序代码#include >#include //GCC中的延时函数头文件#include “hc595.h”//unsigned char Led_Disbuf[10]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F}; //共阴极unsigned char Led_Disbuf[10]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90}; //共阳极unsigned char ComBuf
发表于 2020-01-09
AVR单片机8位数码管显示的程序实现(两种方法介绍)
为什么要选择AVR单片机?
什么是AVR单片机?AVR单片机有什么优点?为什么要选择AVR单片机?AVR单片机是ATMEL公司研制开发的一种新型单片机,它与51单片机、PIC单片机相比具有一系列的优点:1:在相同的系统时钟下AVR运行速度最快;2: 芯片内部的Flsah、EEPROM、SRAM容量较大;3:所有型号的Flash、EEPROM都可以反复烧写、全部支持在线编程烧写(ISP);4:多种频率的内部RC振荡器、上电自动复位、看门狗、启动延时等功能,零外围电路也可以工作;5:每个IO口都可以以推换驱动的方式输出高、低电平,驱动能力强;6:内部资源丰富,一般都集成AD、DA模数器;PWM;SPI、USART、TWI、I2C通信口;丰富的中断源等。目前支持
发表于 2020-01-09
如何让你的AVR单片机功耗超低
了俩小时,发现咋掉电休眠模式后电流还是800多uA,记得以前只有1uA左右的,咋回事?因为经过时间比较长了,重新看了有关休眠的PDF文档资料,完了,自说自话的加了句关闭全部外设的命令,就是功耗抑制寄存器PRR全部置位,发现电流还是大,第二天早上,偶调试了下,电流还是大,后来想想会不会这种垃圾杂牌铝电解漏电流太大,结果拆掉,休眠时的总功耗就3uA,其实,掉电休眠后,AVR的M48PA总消耗电流大约是0.5uA,LDO1uA左右,外围还有个AD欠压检测电流,大约消耗1uA左右,整体全部加起来大约就是2.5-3uA之间。达到预期目的接着,休眠唤醒后,发现键盘工作老不正常,查看原来的程序,除了扫描的矩阵阵列I/O口变化了,其他没啥变化
发表于 2020-01-09
如何让你的AVR单片机功耗超低
AVR复习笔记--AVR单片机SPI多机通讯【包括数据回传与接收】
VR复习笔记—SPI多机通信最近决定复习下AVR单片机,其实也是为了借此复习下几种简单的通信协议,包括串口,SPI,I2­C等。本来以为一两个晚上就能搞定的事儿,没想到竟耗费了一周晚上空余的时间。当然主要是这次的要求要提高点,实现SPI的多机通信,不但要发数据还要回传数据。实际中还是遇到了比我想象中要大的多的困难。即使是现在的实现方式也不是很理想。下面是spi部分的代码,由于spi接收发送用的同一终端,感觉使用起来形式不怎么样,还是采用了轮询标志位的方式#include "spi.h"static char mode=1;void spi_init(char flag){    char
发表于 2020-01-07
AVR复习笔记--AVR单片机SPI多机通讯【包括数据回传与接收】
小广播
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2020 EEWORLD.com.cn, Inc. All rights reserved