阅读:4291回复:4
我的51单片机模拟IIC总线访问24c04的程序,楼下高手看看哪里不行啊?!!
/*----------------------------------------------------------------
Acess the eeprom--24c04 ----------------------------------------------------------------*/ #include <intrins.h> #ifndef INT8U #define INT8U unsigned char #endif #ifndef INT8S #define INT8S signed char #endif #ifndef INT16U #define INT16U unsigned int #endif #define I2C_DELAY; _nop_();_nop_();_nop_();_nop_();_nop_(); // >=4.7uS //---------------------------------------------------------------- // delay 100us //---------------------------------------------------------------- void mDelay(INT8U k) { INT16U i ; for(; k>0; k--) { for(i=0; i<93; i++) ; } } //---------------------------------------------------------------- //OK //---------------------------------------------------------------- void I2C_Start(void) { SDA = 1; I2C_DELAY; SCL = 1; I2C_DELAY; SDA = 0; I2C_DELAY; I2C_DELAY; } //---------------------------------------------------------------- //OK //---------------------------------------------------------------- void I2C_Stop(void) { SDA = 0 ; I2C_DELAY; SCL = 1 ; I2C_DELAY; SDA = 1 ; I2C_DELAY; I2C_DELAY; } //---------------------------------------------------------------- // //---------------------------------------------------------------- void sendAck(void) { SCL = 0; I2C_DELAY; SDA = 0; I2C_DELAY; SCL = 1; I2C_DELAY; } //---------------------------------------------------------------- // //---------------------------------------------------------------- void sendNoAck(void) { SCL = 0; I2C_DELAY; SDA = 1; I2C_DELAY; SCL = 1; I2C_DELAY; } //---------------------------------------------------------------- // 0 = noACK; 1 = ACK ; //---------------------------------------------------------------- bit checkAck() { bit tempbit; /*发送完一个字节后检验设备的应答信号*/ SDA = 1; I2C_DELAY; SCL = 0; I2C_DELAY; tempbit = SDA; SCL = 1; I2C_DELAY; if(tempbit==1) { return 0; //noACK } else { return 1; //ACK } } //---------------------------------------------------------------- //OK // a positive clock edge clock a bit into the ROM //---------------------------------------------------------------- void writeByte(INT8U datum) { INT8U bitCnt = 0 ; for(bitCnt=0; bitCnt<8; bitCnt++) { SCL = 0 ; I2C_DELAY; if ((datum&0x80) == 0x80) //if the MSb is 1 SDA = 1 ; else SDA = 0 ; I2C_DELAY; SCL = 1 ; I2C_DELAY; datum<<=1 ; } } //---------------------------------------------------------------- //OK //---------------------------------------------------------------- INT8U readByte(void) { bit tempbit = 1 ; INT8U temp = 0 ; INT8U bitCnt ; SDA = 1 ; // release the bus,ready to receive byte?????????????? I2C_DELAY; for(bitCnt=0; bitCnt<8; bitCnt++) { SCL = 0; //?????????????????????????huan??????????????? I2C_DELAY; tempbit = SDA ; if (tempbit) temp |= 0x01 ; else temp &= 0xfe ; SCL = 1 ; I2C_DELAY; if(bitCnt<7) temp <<= 1 ; } return(temp) ; } /*~~~~~~~~~~~~~~~~~~~~~~~ API ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~*/ /*----------------------------------------------------------------- write some bytes to sequential address -----------------------------------------------------------------*/ void writeToROM(INT8U datum[], INT8U address, INT8U num) { bit tempbit ; INT8U i ; INT8U *datum_P ; datum_P = datum ; I2C_Start() ; writeByte(0xa0) ; tempbit = checkAck(); writeByte(address) ; tempbit = checkAck(); for(i=0; i<num; i++) { writeByte(*(datum_P+i)) ; if(!checkAck()) { I2C_Stop() ; mDelay(100) ; } } I2C_Stop() ; } /*----------------------------------------------------------------- read some bytes from ROM`s sequential address -----------------------------------------------------------------*/ void readFromROM(INT8U datum[], INT8U address, INT8U num) { bit tempbit ; INT8U i ; INT8U *datum_P ; datum_P = datum; I2C_Start() ; writeByte(0xa0) ; tempbit = checkAck(); writeByte(address) ; tempbit = checkAck(); I2C_Start() ; writeByte(0xa1) ; tempbit = checkAck(); for(i=0; i<num; i++) { *(datum_P+i) = readByte() ; if(i!=num-1) { sendAck() ; } else { sendNoAck() ; } } I2C_Stop() ; } /*----------------------------------------------------------------- wirte one byte to ROM --random write -----------------------------------------------------------------*/ void writeOneByte(INT8U addr, INT8U datum) { bit tempbit ; /*write a byte to mem*/ I2C_Start(); writeByte(0xa0); tempbit = checkAck(); writeByte(addr); /*address*/ tempbit = checkAck(); writeByte(datum); /*the data*/ tempbit = checkAck(); I2C_Stop(); mDelay(100) ; } /*----------------------------------------------------------------- read one byte from rom --random read -----------------------------------------------------------------*/ INT8U readOneByte(INT8U addr) { bit tempbit = 1; INT8U mydata; /*read a byte from mem*/ I2C_Start(); writeByte(0xa0); tempbit = checkAck(); writeByte(addr); /*address*/ tempbit = checkAck(); I2C_Start(); writeByte(0xa1); tempbit = checkAck(); mydata = readByte(); tempbit = checkAck(); return (mydata) ; I2C_Stop(); } |
|
|
沙发#
发布于:2005-08-05 16:25
你用 KEIL C 的LA看输出波形
|
|
板凳#
发布于:2005-08-05 16:25
你用 KEIL C 的LA看输出波形
|
|
地板#
发布于:2007-04-01 22:43
问题可能出在checkAck()
上了!有可能 |
|
地下室#
发布于:2007-04-09 13:44
/********************************************************************
VI2C_C51.C 此程序是I2C操作平台(主方式的软件平台)的底层的C子程序,如发送数据 及接收数据,应答位发送,并提供了几个直接面对器件的操作函数,它很方便的 与用户程序连接并扩展..... 注意:函数是采用软件延时的方法产生SCL脉冲,固对高晶振频率要作 一定的修改....(本例是1us机器周期,即晶振频率要小于12MHZ) ********************************************************************/ #include <reg.51h> /*头文件的包含*/ #include <intrins.h> #define uchar unsigned char /*宏定义*/ #define uint unsigned int #define _Nop() _nop_() /*定义空指令*/ /* 常,变量定义区 */ /*端口位定义*/ sbit SDA=P1^3; /*模拟I2C数据传送位*/ sbit SCL=P1^2; /*模拟I2C时钟控制位*/ /*状态标志*/ bit ack; /*应答标志位*/ /******************************************************************* 起动总线函数 函数原型: 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<<BitCnt)&0x80)SDA=1; /*判断发送位*/ else SDA=0; _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(); } /******************************************************************* 向无子地址器件发送字节数据函数 函数原型: bit ISendByte(uchar sla,ucahr c); 功能: 从启动总线到发送地址,数据,结束总线的全过程,从器件地址sla. 如果返回1表示操作成功,否则操作有误。 注意: 使用前必须已结束总线。 ********************************************************************/ bit ISendByte(uchar sla,uchar c) { Start_I2c(); /*启动总线*/ 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(uchar sla,uchar suba,uchar *s,uchar no) { uchar i; Start_I2c(); /*启动总线*/ SendByte(sla); /*发送器件地址*/ if(ack==0)return(0); SendByte(suba); /*发送器件子地址*/ if(ack==0)return(0); for(i=0;i<no;i++) { SendByte(*s); /*发送数据*/ if(ack==0)return(0); s++; } Stop_I2c(); /*结束总线*/ return(1); } /******************************************************************* 向无子地址器件读字节数据函数 函数原型: bit IRcvByte(uchar sla,ucahr *c); 功能: 从启动总线到发送地址,读数据,结束总线的全过程,从器件地 址sla,返回值在c. 如果返回1表示操作成功,否则操作有误。 注意: 使用前必须已结束总线。 ********************************************************************/ bit IRcvByte(uchar sla,uchar *c) { Start_I2c(); /*启动总线*/ SendByte(sla+1); /*发送器件地址*/ if(ack==0)return(0); *c=RcvByte(); /*读取数据*/ Ack_I2c(1); /*发送非就答位*/ Stop_I2c(); /*结束总线*/ return(1); } /******************************************************************* 向有子地址器件读取多字节数据函数 函数原型: bit ISendStr(uchar sla,uchar suba,ucahr *s,uchar no); 功能: 从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件 地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。 如果返回1表示操作成功,否则操作有误。 注意: 使用前必须已结束总线。 ********************************************************************/ bit IRcvStr(uchar sla,uchar suba,uchar *s,uchar no) { uchar i; Start_I2c(); /*启动总线*/ SendByte(sla); /*发送器件地址*/ if(ack==0)return(0); SendByte(suba); /*发送器件子地址*/ if(ack==0)return(0); Start_I2c(); SendByte(sla+1); if(ack==0)return(0); for(i=0;i<no-1;i++) { *s=RcvByte(); /*发送数据*/ Ack_I2c(0); /*发送就答位*/ s++; } *s=RcvByte(); Ack_I2c(1); /*发送非应位*/ Stop_I2c(); /*结束总线*/ return(1); } /* 完毕 */ |
|
|