完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
扫一扫,分享给好友
我在调试一块PCF8563,I2C总线的时钟芯片,但是能够写入时钟寄存器,不能够读出时间,各位看看是什么原因?
现程序如下: I2C程序部分: #include #include #include"I2C.H" #define _Nop() _nop_() /*定义空指令*/ #define uchar unsigned char /*宏定义*/ #define uint unsigned int ***it SCL=P2^1; //I2C 时钟 ***it SDA=P2^0; //I2C 数据 static bit ack; /*应答标志位*/ /***********此部分为I2C总线的驱动程序*******************/ /******************************************************************* 起动总线函数 函数原型: void Start_I2c(); 功能: 启动I2C总线,即发送I2C起始条件. ********************************************************************/ void Start_I2c() { SDA=1; /*发送起始条件的数据信号*/ _Nop(); SCL=1; _Nop(); /*起始条件建立时间大于4.7us,延时*/ _Nop(); _Nop(); _Nop(); _Nop(); SDA=0; /*发送起始信号*/ _Nop(); /* 起始条件锁定时间大于4μs*/ _Nop(); _Nop(); _Nop(); _Nop(); SCL=0; /*钳住I2C总线,准备发送或接收数据 */ _Nop(); _Nop(); } /******************************************************************* 结束总线函数 函数原型: void Stop_I2c(); 功能: 结束I2C总线,即发送I2C结束条件. ********************************************************************/ void Stop_I2c() { SDA=0; /*发送结束条件的数据信号*/ _Nop(); /*发送结束条件的时钟信号*/ SCL=1; /*结束条件建立时间大于4μs*/ _Nop(); _Nop(); _Nop(); _Nop(); _Nop(); SDA=1; /*发送I2C总线结束信号*/ _Nop(); _Nop(); _Nop(); _Nop(); } /******************************************************************* 字节数据传送函数 函数原型: void SendByte(uchar c); 功能: 将数据c发送出去,可以是地址,也可以是数据,发完后等待应答,并对 此状态位进行操作.(不应答或非应答都使ack=0 假) 发送数据正常,ack=1; ack=0表示被控器无应答或损坏。 ********************************************************************/ void SendByte(uchar c) { uchar BitCnt; for(BitCnt=0;BitCnt<8;BitCnt++) /*要传送的数据长度为8位*/ { if((c< _Nop(); SCL=1; /*置时钟线为高,通知被控器开始接收数据位*/ _Nop(); _Nop(); /*保证时钟高电平周期大于4μs*/ _Nop(); _Nop(); _Nop(); SCL=0; } _Nop(); _Nop(); SDA=1; /*8位发送完后释放数据线,准备接收应答位*/ _Nop(); _Nop(); SCL=1; _Nop(); _Nop(); _Nop(); if(SDA==1)ack=0; else ack=1; /*判断是否接收到应答信号*/ SCL=0; _Nop(); _Nop(); } /******************************************************************* 字节数据传送函数 函数原型: uchar RcvByte(); 功能: 用来接收从器件传来的数据,并判断总线错误(不发应答信号), 发完后请用应答函数。 ********************************************************************/ uchar RcvByte() { uchar retc; uchar BitCnt; retc=0; SDA=1; /*置数据线为输入方式*/ for(BitCnt=0;BitCnt<8;BitCnt++) { _Nop(); SCL=0; /*置时钟线为低,准备接收数据位 */ _Nop(); _Nop(); /*时钟低电平周期大于4.7μs*/ _Nop(); _Nop(); _Nop(); SCL=1; /*置时钟线为高使数据线上数据有效*/ _Nop(); _Nop(); retc=retc<<1; if(SDA==1)retc=retc+1; /*读数据位,接收的数据位放入retc中 */ _Nop(); _Nop(); } SCL=0; _Nop(); _Nop(); return(retc); } /******************************************************************** 应答子函数 原型: void Ack_I2c(bit a); 功能:主控器进行应答信号,(可以是应答或非应答信号) ********************************************************************/ void Ack_I2c(bit a) { if(a==0)SDA=0; /*在此发出应答或非应答信号 */ else SDA=1; _Nop(); _Nop(); _Nop(); SCL=1; _Nop(); _Nop(); /*时钟低电平周期大于4μs*/ _Nop(); _Nop(); _Nop(); SCL=0; /*清时钟线,钳住I2C总线以便继续接收*/ _Nop(); _Nop(); } /********************此部分为PCF8563操作程序************************/ /******************************************************************* 写一个字节数据到PCF8563指定寄存器中 函数原型: bit ISendByte(uchar sla,ucahr c); 功能: 从启动总线到发送地址,数据,结束总线的全过程,从器件地址sla. 如果返回1表示操作成功,否则操作有误。 注意: 使用前必须已结束总线。 ********************************************************************/ bit ISendByte(unsigned char sla,unsigned char c) { Start_I2c(); /*启动总线*/ SendByte(0xa2); /*发送写指令*/ if(ack==0)return(0); SendByte(sla); /*发送器件地址*/ if(ack==0)return(0); SendByte(c); /*发送数据*/ if(ack==0)return(0); Stop_I2c(); /*结束总线*/ return(1); } /******************************************************************* 向有子地址器件发送多字节数据函数 函数原型: bit ISendStr(uchar sla,uchar suba,ucahr *s,uchar no); 功能: 从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件 地址sla,子地址suba,发送内容是s指向的内容,发送no个字节。 如果返回1表示操作成功,否则操作有误。 注意: 使用前必须已结束总线。 ********************************************************************/ bit ISendStr(unsigned char suba,unsigned char *s,unsigned char no) { unsigned char i; Start_I2c(); /*启动总线*/ SendByte(0xa2); /*发送器件地址*/ if(ack==0)return(0); SendByte(suba); /*发送器件子地址*/ if(ack==0)return(0); for(i=0;i SendByte(*s); /*发送数据*/ if(ack==0)return(0); s++; } Stop_I2c(); /*结束总线*/ return(1); } /******************************************************************* 向有子地址器件读取多字节数据函数 函数原型: bit ISendStr(uchar sla,uchar suba,ucahr *s,uchar no); 功能: 从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件 地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。 如果返回1表示操作成功,否则操作有误。 注意: 使用前必须已结束总线。 ********************************************************************/ bit IRcvStr(unsigned char suba,unsigned char *s,unsigned char no) { unsigned char i; Start_I2c(); /*启动总线*/ SendByte(0xa2); /*发送器件地址*/ if(ack==0)return(0); SendByte(suba); /*发送器件子地址*/ if(ack==0)return(0); Start_I2c(); SendByte(0xa3); if(ack==0)return(0); for(i=0;i *s=RcvByte(); /*发送数据*/ Ack_I2c(0); /*发送就答位*/ s++; } *s=RcvByte(); Ack_I2c(1); /*发送非应位*/ Stop_I2c(); /*结束总线*/ return(1); } I2C头文件: #ifndef _I2C_H_ #define _I2C_H_ extern bit ISendByte(unsigned char sla,unsigned char c); //向PCF8563寄存器发送一个字节 extern bit ISendStr(unsigned char suba,unsigned char *s,unsigned char no);//发送多字节数据 extern bit IRcvStr(unsigned char suba,unsigned char *s,unsigned char no); //接收多字节数据 #endif 主函数: #include #include"I2C.h" typedef unsigned int uint; typedef unsigned char uchar; ***it LS138A=P2^2; ***it LS138B=P2^3; ***it LS138C=P2^4; uchar tmpdisplay[8],gettime[8]; uchar tmpdate[3]={0,59,7}; //秒、分、时、日、星期、月、年;2012-11-1 07:59:00 uchar rtc_address[3]={0x02,0x03,0x04}; //秒、分、时、日、星期、月、年寄存器地址 unsigned char code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x40}; //数码管显示 void set_rtc() //设定时钟数据 { uchar i,tmp; for(i=0;i<3;i++) //BCD处理,将十进制数转化为BCD编码(按十六进制数计算) { tmp=tmpdate[i]/10; tmpdate[i]=tmpdate[i]%10; tmpdate[i]=tmpdate[i]+tmp*16; } ISendStr(0x02,tmpdate,0x03); } void rtc_init() //时钟初始化 { set_rtc(); ISendByte(0x00,0x00); //控制寄存器0 ISendByte(0x01,0x11); //控制寄存器1 ISendByte(0x0D,0xf0); } void read_rtc() //读时钟函数,将读出的数据存入tmpdate缓存数组中 { uchar i,tmp; IRcvStr(0x02,gettime,0x03); gettime[0]=gettime[0]&0x7f; //秒 gettime[1]=gettime[1]&0x7f; //分 gettime[2]=gettime[2]&0x3f; //时 for(i=0;i<3;i++) //将BCD码转化为十六进制数(十进制数) { tmp=gettime[i]/16; gettime[i]=gettime[i]%10; gettime[i]=gettime[i]+tmp*10; } } void delay(uint a) { uint j; for(a;a>0;a--) for(j=110;j>0;j--); } void display() { uint i; for(i=0;i<8;i++) { P0=table[tmpdisplay[i]]; //查表法得到要显示数字的数码段 switch(i) { case 0:LS138A=0; LS138B=0; LS138C=0; break; case 1:LS138A=1; LS138B=0; LS138C=0; break; case 2:LS138A=0; LS138B=1; LS138C=0; break; case 3:LS138A=1; LS138B=1; LS138C=0; break; case 4:LS138A=0; LS138B=0; LS138C=1; break; case 5:LS138A=1; LS138B=0; LS138C=1; break; case 6:LS138A=0; LS138B=1; LS138C=1; break; case 7:LS138A=1; LS138B=1; LS138C=1; break; } delay(2); } } void main() { rtc_init(); while(1) { read_rtc(); tmpdisplay[0]=gettime[2]/10; //数据的转换,因我们采用数码管0~9的显示,将数据分开 tmpdisplay[1]=gettime[2]%10; tmpdisplay[2]=10;//加入"-" tmpdisplay[3]=gettime[1]/10; tmpdisplay[4]=gettime[1]%10; tmpdisplay[5]=10;//加入"-" tmpdisplay[6]=gettime[0]/10; tmpdisplay[7]=gettime[0]%10; display(); } } |
|
相关推荐
7个回答
|
|
不太清楚
|
|
|
|
|
|
|
|
|
|
|
|
first,这代码如果是你自己写的,那么你能操作写,肯定能操作读出。
问题就这么几种,延时不够,时序不严谨,LSB的读写位弄错了,地址发错,重新启动总线时的错误。 自己找找吧。 |
|
|
|
好的,我再仔细看看! |
|
|
|
|
|
我也是读出来是FF
|
|
|
|
你正在撰写答案
如果你是对答案或其他答案精选点评或询问,请使用“评论”功能。
【每周推荐】采用11代Intel CPU,基于youyeetoo X1开发板搭建少儿AI智能STEAM积木平台
752 浏览 0 评论
2208 浏览 2 评论
【youyeetoo X1 windows 开发板体验】+ 影音处理和AI模型移植
2084 浏览 5 评论
I.MX6ULL-飞凌 ElfBoard ELF1板卡- 移植zbar的方法
1682 浏览 0 评论
2636 浏览 3 评论
【youyeetoo X1 windows 开发板体验】少儿AI智能STEAM积木平台
5443 浏览 31 评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-4-18 03:25 , Processed in 0.438520 second(s), Total 73, Slave 57 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号