用51单片机做一个模拟I2c的实验,E2PROM用的是at24c512b,但是程序始终有问题,求高手帮助,谢谢 30
#include<reg51.h>sbitsda=P1^0;sbitscl=P1^1;#definedelay60#definefalse0#definetrue1voi...
#include<reg51.h>
sbit sda=P1^0;
sbit scl=P1^1;
#define delay 60
#define false 0
#define true 1
void delayus(unsigned char t)
{
for(t=0;t<60;t++)
{;}
}
void start(void)
{
sda=1;
scl=1;
delayus(delay);
sda=0;
delayus(delay);
scl=0;
delayus(delay);
}
void stop(void)
{
sda=0;
scl=1;
delayus(delay);
sda=1;
delayus(delay);
scl=0;
delayus(delay);
}
void send0(void)
{
sda=0;
scl=1;
delayus(delay);
scl=0;
delayus(delay);
}
void send1(void)
{
sda=1;
scl=1;
delayus(delay);
scl=0;
delayus(delay);
}
bit ack(void)
{
char F0;
sda=1;
scl=1;
delayus(delay/2);
F0=sda;
delayus(delay/2);
scl=0;
delayus(delay);
if(F0==1)
return false;
else
return true;
}
void writeI2Cbyte(char b)
{
char i;
for(i=0;i<8;i++)
{
if((b<<i)&0x80)
send1();
else
send0();
}
}
char readI2Cbyte(void)
{
char b=0,i,F0;
for(i=0;i<8;i++)
{
sda=1;
scl=1;
delayus(10);
F0=sda;
delayus(10);
scl=0;
if(F0==1)
{
b=b<<1;
b=b|0x01;
}
else
b=b<<1;
}
return b;
}
void writeonebyte(int addr,char thedata)
{
bit acktemp=1;
start();
writeI2Cbyte(0xa0);
acktemp=ack();
writeI2Cbyte(addr);
acktemp=ack();
writeI2Cbyte(thedata);
acktemp=ack();
stop();
}
char readonebyte(int addr)
{
bit acktemp=1;
char indata;
start();
writeI2Cbyte(0xa0);
acktemp=ack();
writeI2Cbyte(addr);
acktemp=ack();
start();
writeI2Cbyte(0xa1);
acktemp=ack();
indata=readI2Cbyte();
acktemp=ack();
return indata;
stop();
}
void led(char t)
{
unsigned char num[10]={0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF};
char i;
for(i=0;i<10;i++)
{
if(t==i)
P2=num[i];
}
}
void main()
{
char i;
while(1)
{
writeonebyte(0x1000,3);
i=readonebyte(0x1000);
led(i);
}
} 展开
sbit sda=P1^0;
sbit scl=P1^1;
#define delay 60
#define false 0
#define true 1
void delayus(unsigned char t)
{
for(t=0;t<60;t++)
{;}
}
void start(void)
{
sda=1;
scl=1;
delayus(delay);
sda=0;
delayus(delay);
scl=0;
delayus(delay);
}
void stop(void)
{
sda=0;
scl=1;
delayus(delay);
sda=1;
delayus(delay);
scl=0;
delayus(delay);
}
void send0(void)
{
sda=0;
scl=1;
delayus(delay);
scl=0;
delayus(delay);
}
void send1(void)
{
sda=1;
scl=1;
delayus(delay);
scl=0;
delayus(delay);
}
bit ack(void)
{
char F0;
sda=1;
scl=1;
delayus(delay/2);
F0=sda;
delayus(delay/2);
scl=0;
delayus(delay);
if(F0==1)
return false;
else
return true;
}
void writeI2Cbyte(char b)
{
char i;
for(i=0;i<8;i++)
{
if((b<<i)&0x80)
send1();
else
send0();
}
}
char readI2Cbyte(void)
{
char b=0,i,F0;
for(i=0;i<8;i++)
{
sda=1;
scl=1;
delayus(10);
F0=sda;
delayus(10);
scl=0;
if(F0==1)
{
b=b<<1;
b=b|0x01;
}
else
b=b<<1;
}
return b;
}
void writeonebyte(int addr,char thedata)
{
bit acktemp=1;
start();
writeI2Cbyte(0xa0);
acktemp=ack();
writeI2Cbyte(addr);
acktemp=ack();
writeI2Cbyte(thedata);
acktemp=ack();
stop();
}
char readonebyte(int addr)
{
bit acktemp=1;
char indata;
start();
writeI2Cbyte(0xa0);
acktemp=ack();
writeI2Cbyte(addr);
acktemp=ack();
start();
writeI2Cbyte(0xa1);
acktemp=ack();
indata=readI2Cbyte();
acktemp=ack();
return indata;
stop();
}
void led(char t)
{
unsigned char num[10]={0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF};
char i;
for(i=0;i<10;i++)
{
if(t==i)
P2=num[i];
}
}
void main()
{
char i;
while(1)
{
writeonebyte(0x1000,3);
i=readonebyte(0x1000);
led(i);
}
} 展开
3个回答
展开全部
24C512的地址位是16位的,不能只传送八位地址位
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
展开全部
/*----------------------------------------------------------------------------------------------------------
IIC总线读写EEPROM(串行扩展eeprom,24c02)
(STC12C系列单片机自带eeprom,且有另外的eeprom操作方式)
作者:Allen.H(帮同学修改的一个程序)
时间:2010.11.5
----------------------------------------------------------------------------------------------------------*/
#include <reg52.h>
#include <intrins.h>//是用括号还是双引号看情况,本地头文件用双引号,系统头文件用括号
//这里使用了_nop_()函数,所以调用此头文件
#define TRUE 1/*define宏定义一般用大写,宏定义并不会减少最终代码空间
define多行语句时,每一行末尾写上\,最后一行可以不写,
有时比较短的语句写成一个子函数会牺牲更多的时间,
因为函数调用耗时比较多,这个时候用一个define语句更好*/
#define FALSE 0
typedef unsigned char uchar;//良好的程序风格,不应该用#define
//#define uchar unsigned char
sbit sda=P2^0; //---------你把sda和scl引脚可能定反了,我换过来了-------------------------------
sbit scl=P2^1;//等号对其,变量名长短不一时,注意,且测试等于号"=="或者其他双目关系运算符两边都空一格
//-----------------------------------------------------------------
void delay(uchar z)//带参数很好
{//大括号所在行不要写代码
uchar i,j;//局部变量中用来自加自减可以用i,j之类的定义,计数建议不要用i,j
//局部变量不占内存,函数调用时生成堆栈,不应该定义局部变量时作初始化
//----局部变量命名后空一格,写正式代码
for(i=z;i>0;i--)
for(j=100;j>0;j--);//注明多少时间,在调试模式下,看窗口左边的SEC值
}
//函数与函数之间空一格
void delay_7nop()//子程序命名最好顾名思义,比如delay_1ms(),这里考虑都是使用7nop,不带参数
{/*程序代码每进一层逻辑就缩进一格TAB键,TAB设置为3,4格,
在keil的view->options里面设置,不要使用几个空格来缩进,统一使用TAB键*/
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();//这里0-1000多个_nop_都可以
}
//delay函数都放在一起,函数顺序不要乱放,相关的放一起,
//--------------------------------------------------------------------
void init()
{
sda=1;
delay_7nop();
scl=1;
delay_7nop();
}
//---SCL线为高电平期间,SDA线由高电平向低电平的变化表示起始信号;
//SCL线为高电平期间,SDA线由低电平向高电平的变化表示终止信号。
//但更具体还是得看时序图,下面就没有都先把scl先拉高再去变sda
void start()
{
sda=1;
delay_7nop(); //这里sda和第三行的scl信号哪个放上没什么区别,主要起始和停止信号风格保持一致就行了
scl=1;
delay_7nop();
sda=0;
delay_7nop();
//scl=0; //允许数据变化,传数据的时候拉低才允许数据变化,
//但是在开始信号和停止信号scl都为高,这里看时序图就知道了
}
void stop()
{
sda=0;
delay_7nop();
scl=1;
delay_7nop();
sda=1;
delay_7nop();
}
bit ask()//应答信号,return是什么类型这里函数就是什么类型,
{ //每一个被传送的字节后面都必须跟随一位应答位(即一帧共有9位)
//如果一段时间内没有收到从机的应答信号,则自动认为从机已正确接收到数据
bit flag;//真假判断,或只有0/1取值的标志位设置为bit
sda=1;
scl=1;
delay_7nop();
flag=sda;
delay_7nop();
scl=0;
delay_7nop();
if(flag==1)
return FALSE;//非应答
else
return TRUE;//应答
}
//用下面屏蔽的的应答信号也可以,上面的应答信号考虑更周全
/*
void ask() //应答
{
uchar i;
scl=1;
delay_7nop();
while((sda==1)&&(i<250))i++;
scl=0;
delay_7nop();
}
*/
//----------------------------------------------------
void writedata(uchar dat)//下面是readdata()和readadd()保持程序风格的一致性,命名不该命为writecurrent
{//函数参数不要乱用P,q之类的毫无意义的名字,这里用dat,date是关键字,不能用
uchar i;
// scl=0;//此句可有可无
for(i=0;i<8;i++)//按位写
{
dat=dat<<1;//左移一位
scl=0;
delay_7nop();
sda=CY;//psw位中的CY进位标识位,左移后最高位移入CY
delay_7nop();
scl=1;//scl高电平,数据稳定
delay_7nop();
}
scl=0;
delay_7nop();
sda=1;//总线释放
delay_7nop();
}
void writeadd(uchar add,uchar infor)
{
start();
writedata(0xa0);//器件地址
ask();
writedata(add);//器件内部存储区的地址
ask();
writedata(infor);//数据
ask();
stop();
}
//----------------------------------------------------
uchar readdata()
{
uchar i,dat;
scl=0;
delay_7nop();
sda=1;//数据总线释放
delay_7nop();
for(i=0;i<8;i++)
{
scl=1;
delay_7nop();
dat=(dat<<1)|(uchar)sda;//此处的强制类型转换表现思维考虑到了
//dat左移一位,最低位为0,此时与sda按位或运算就把sda数据读到了最低位
scl=0;
delay_7nop();
}
//密切相关的代码紧接着写,不很相关的空一格再写
return dat;
}
uchar readadd(uchar add)
{
uchar r=0;//局部变量小写,全局变量首字母大写
start();
writedata(0xa0);
ask();
writedata(add);
ask();
start();
writedata(0xa1);
ask();
r=readdata();
stop();
return r;
}
//-------------------------------------------------------
void main()
{
while(1)//在keil的调试仿真窗口(Perpherals->I/O-ports->)看不出P2口的变换,
//因为这里是外部EEPROM,要仿真芯片或者硬件的支持才能观察结果,本程序测试无误
{
init();//这里初始化一下
writeadd(25,0xaa);
delay(50);//此处最少要delay(7);
P1=readadd(25);//P1还是P2还是P3主要是看你的硬件用哪个来测试
}
}
//主函数放最后是省去了函数申明,但在工程应用中建议放在最上面
//这样一眼就能看到该工程是做什么的,且功能函数本身就应该在头文件中作申明
//以便其他点C文件能方便调用,每写一个功能函数都在头文件中作申明,这是一个好习惯
//方便其他点C文件随时调用
//-------------------------------------------------------
//总结:
// 1.你的程序最初可能把scl和sda可能定反了,
// 2.你没写ask函数(屏蔽了),应答信号必须写,
// 3.你的代码风格,变量名命名,函数名命名,函数排放顺序,无注释,
// 书写排版有很大问题,看你的程序很吃力,且不美观
// 4.能用子函数代替的就写成子函数,用那么多nop看上去代码真丑
// 5.有关读写的4个函数你函数名命名风格没统一
//建议:
// 1.看时序图的能力和对IIC总线的理解有待加强
// 2.要慢慢形成规范的代码风格
// 3.keil软件对你还有很大学习空间,要学习用更多的keil调试和用protus仿真
//相关提示:
// 1.要学些使用下列对内存和存储的理解:
// code :程序存储区(64KB)
// data :可直接寻址的内部数据存储区(128B) 默认的变量存储区
// idata:不可直接寻址的内部数据存储区(256B) 当全局变量定义太多的时候
// 使用,比如:idata int 或int idata来使用这个区域存储全局变量
// bdata:可位寻址内部数据存储区(16B)
// xdata:外部数据存储区(64KB)当使用来了外部RAM时,应用它,keil中options for target 的memory mode可选
// pdata:分页的外部数据存储区
这是一个可以用的,和你的有点区别,而且注释得很全,希望能对你有所帮助
IIC总线读写EEPROM(串行扩展eeprom,24c02)
(STC12C系列单片机自带eeprom,且有另外的eeprom操作方式)
作者:Allen.H(帮同学修改的一个程序)
时间:2010.11.5
----------------------------------------------------------------------------------------------------------*/
#include <reg52.h>
#include <intrins.h>//是用括号还是双引号看情况,本地头文件用双引号,系统头文件用括号
//这里使用了_nop_()函数,所以调用此头文件
#define TRUE 1/*define宏定义一般用大写,宏定义并不会减少最终代码空间
define多行语句时,每一行末尾写上\,最后一行可以不写,
有时比较短的语句写成一个子函数会牺牲更多的时间,
因为函数调用耗时比较多,这个时候用一个define语句更好*/
#define FALSE 0
typedef unsigned char uchar;//良好的程序风格,不应该用#define
//#define uchar unsigned char
sbit sda=P2^0; //---------你把sda和scl引脚可能定反了,我换过来了-------------------------------
sbit scl=P2^1;//等号对其,变量名长短不一时,注意,且测试等于号"=="或者其他双目关系运算符两边都空一格
//-----------------------------------------------------------------
void delay(uchar z)//带参数很好
{//大括号所在行不要写代码
uchar i,j;//局部变量中用来自加自减可以用i,j之类的定义,计数建议不要用i,j
//局部变量不占内存,函数调用时生成堆栈,不应该定义局部变量时作初始化
//----局部变量命名后空一格,写正式代码
for(i=z;i>0;i--)
for(j=100;j>0;j--);//注明多少时间,在调试模式下,看窗口左边的SEC值
}
//函数与函数之间空一格
void delay_7nop()//子程序命名最好顾名思义,比如delay_1ms(),这里考虑都是使用7nop,不带参数
{/*程序代码每进一层逻辑就缩进一格TAB键,TAB设置为3,4格,
在keil的view->options里面设置,不要使用几个空格来缩进,统一使用TAB键*/
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();//这里0-1000多个_nop_都可以
}
//delay函数都放在一起,函数顺序不要乱放,相关的放一起,
//--------------------------------------------------------------------
void init()
{
sda=1;
delay_7nop();
scl=1;
delay_7nop();
}
//---SCL线为高电平期间,SDA线由高电平向低电平的变化表示起始信号;
//SCL线为高电平期间,SDA线由低电平向高电平的变化表示终止信号。
//但更具体还是得看时序图,下面就没有都先把scl先拉高再去变sda
void start()
{
sda=1;
delay_7nop(); //这里sda和第三行的scl信号哪个放上没什么区别,主要起始和停止信号风格保持一致就行了
scl=1;
delay_7nop();
sda=0;
delay_7nop();
//scl=0; //允许数据变化,传数据的时候拉低才允许数据变化,
//但是在开始信号和停止信号scl都为高,这里看时序图就知道了
}
void stop()
{
sda=0;
delay_7nop();
scl=1;
delay_7nop();
sda=1;
delay_7nop();
}
bit ask()//应答信号,return是什么类型这里函数就是什么类型,
{ //每一个被传送的字节后面都必须跟随一位应答位(即一帧共有9位)
//如果一段时间内没有收到从机的应答信号,则自动认为从机已正确接收到数据
bit flag;//真假判断,或只有0/1取值的标志位设置为bit
sda=1;
scl=1;
delay_7nop();
flag=sda;
delay_7nop();
scl=0;
delay_7nop();
if(flag==1)
return FALSE;//非应答
else
return TRUE;//应答
}
//用下面屏蔽的的应答信号也可以,上面的应答信号考虑更周全
/*
void ask() //应答
{
uchar i;
scl=1;
delay_7nop();
while((sda==1)&&(i<250))i++;
scl=0;
delay_7nop();
}
*/
//----------------------------------------------------
void writedata(uchar dat)//下面是readdata()和readadd()保持程序风格的一致性,命名不该命为writecurrent
{//函数参数不要乱用P,q之类的毫无意义的名字,这里用dat,date是关键字,不能用
uchar i;
// scl=0;//此句可有可无
for(i=0;i<8;i++)//按位写
{
dat=dat<<1;//左移一位
scl=0;
delay_7nop();
sda=CY;//psw位中的CY进位标识位,左移后最高位移入CY
delay_7nop();
scl=1;//scl高电平,数据稳定
delay_7nop();
}
scl=0;
delay_7nop();
sda=1;//总线释放
delay_7nop();
}
void writeadd(uchar add,uchar infor)
{
start();
writedata(0xa0);//器件地址
ask();
writedata(add);//器件内部存储区的地址
ask();
writedata(infor);//数据
ask();
stop();
}
//----------------------------------------------------
uchar readdata()
{
uchar i,dat;
scl=0;
delay_7nop();
sda=1;//数据总线释放
delay_7nop();
for(i=0;i<8;i++)
{
scl=1;
delay_7nop();
dat=(dat<<1)|(uchar)sda;//此处的强制类型转换表现思维考虑到了
//dat左移一位,最低位为0,此时与sda按位或运算就把sda数据读到了最低位
scl=0;
delay_7nop();
}
//密切相关的代码紧接着写,不很相关的空一格再写
return dat;
}
uchar readadd(uchar add)
{
uchar r=0;//局部变量小写,全局变量首字母大写
start();
writedata(0xa0);
ask();
writedata(add);
ask();
start();
writedata(0xa1);
ask();
r=readdata();
stop();
return r;
}
//-------------------------------------------------------
void main()
{
while(1)//在keil的调试仿真窗口(Perpherals->I/O-ports->)看不出P2口的变换,
//因为这里是外部EEPROM,要仿真芯片或者硬件的支持才能观察结果,本程序测试无误
{
init();//这里初始化一下
writeadd(25,0xaa);
delay(50);//此处最少要delay(7);
P1=readadd(25);//P1还是P2还是P3主要是看你的硬件用哪个来测试
}
}
//主函数放最后是省去了函数申明,但在工程应用中建议放在最上面
//这样一眼就能看到该工程是做什么的,且功能函数本身就应该在头文件中作申明
//以便其他点C文件能方便调用,每写一个功能函数都在头文件中作申明,这是一个好习惯
//方便其他点C文件随时调用
//-------------------------------------------------------
//总结:
// 1.你的程序最初可能把scl和sda可能定反了,
// 2.你没写ask函数(屏蔽了),应答信号必须写,
// 3.你的代码风格,变量名命名,函数名命名,函数排放顺序,无注释,
// 书写排版有很大问题,看你的程序很吃力,且不美观
// 4.能用子函数代替的就写成子函数,用那么多nop看上去代码真丑
// 5.有关读写的4个函数你函数名命名风格没统一
//建议:
// 1.看时序图的能力和对IIC总线的理解有待加强
// 2.要慢慢形成规范的代码风格
// 3.keil软件对你还有很大学习空间,要学习用更多的keil调试和用protus仿真
//相关提示:
// 1.要学些使用下列对内存和存储的理解:
// code :程序存储区(64KB)
// data :可直接寻址的内部数据存储区(128B) 默认的变量存储区
// idata:不可直接寻址的内部数据存储区(256B) 当全局变量定义太多的时候
// 使用,比如:idata int 或int idata来使用这个区域存储全局变量
// bdata:可位寻址内部数据存储区(16B)
// xdata:外部数据存储区(64KB)当使用来了外部RAM时,应用它,keil中options for target 的memory mode可选
// pdata:分页的外部数据存储区
这是一个可以用的,和你的有点区别,而且注释得很全,希望能对你有所帮助
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询