阅读:6156回复:10
24C64子程序
/*24C64子程序 */
bit I2C_Start(void); void I2C_Stop(void); void I2C_Ack(void); void I2C_Nack(void); bit I2C_Send_Byte( uchar); uchar I2C_Receive_Byte(void); void AT24C64_R(void *mcu_address,uint AT24C64_address,uint count); void AT24C64_W(void *mcu_address,uint AT24C64_address,uint count); void Delay_10_uS(void) { char i=10; while(i--); } void Delay_N_mS( uint n_milisecond) /* n mS delay */ { uchar i; while(n_milisecond--) { i=37; while(i--); } } bit I2C_Start(void) { Delay_10_uS(); I2C_SDA =1; Delay_10_uS(); I2C_SCK =1; Delay_10_uS(); if ( I2C_SDA == 0) return 0; if ( I2C_SCK == 0) return 0; I2C_SDA = 0; Delay_10_uS(); I2C_SCK = 0; Delay_10_uS(); return 1; } void I2C_Stop(void) { Delay_10_uS(); I2C_SDA = 0; Delay_10_uS(); I2C_SCK = 1; Delay_10_uS(); I2C_SDA = 1; Delay_10_uS(); } void I2C_Ack(void) { Delay_10_uS(); I2C_SDA=0; Delay_10_uS(); I2C_SCK=1; Delay_10_uS(); I2C_SCK=0; Delay_10_uS(); } void I2C_Nack(void) { Delay_10_uS(); I2C_SDA=1; Delay_10_uS(); I2C_SCK=1; Delay_10_uS(); I2C_SCK=0; Delay_10_uS(); } bit I2C_Send_Byte( uchar d) { uchar i = 8; bit bit_ack; while( i-- ) { Delay_10_uS(); if ( d &0x80 ) I2C_SDA =1; else I2C_SDA =0; Delay_10_uS(); I2C_SCK = 1; Delay_10_uS(); I2C_SCK = 0; d = d << 1; } Delay_10_uS(); I2C_SDA = 1; Delay_10_uS(); I2C_SCK = 1; Delay_10_uS(); bit_ack = I2C_SDA; I2C_SCK =0; Delay_10_uS(); return bit_ack; } uchar I2C_Receive_Byte(void) { uchar i = 8, d; Delay_10_uS(); I2C_SDA = 1; while ( i--) { d = d << 1; Delay_10_uS(); I2C_SCK =1; if ( I2C_SDA ) d++; Delay_10_uS(); I2C_SCK =0; } return d; } void AT24C64_W(void *mcu_address,uint AT24C64_address,uint count) { DOG_WDI=!DOG_WDI; DOGTIME=0; while(count--) { I2C_Start(); /*I2C_Send_Byte( 0xa0 + AT24C64_address /256 *2);*/ /* 24C16 USE */ I2C_Send_Byte( 0xa0 ); I2C_Send_Byte( AT24C64_address/256 ); I2C_Send_Byte( AT24C64_address %256 ); I2C_Send_Byte( *(uchar*)mcu_address ); I2C_Stop(); Delay_N_mS(10); /* waiting for write cycle to be completed */ ((uchar*)mcu_address)++; AT24C64_address++; } } void AT24C64_R(void *mcu_address,uint AT24C64_address,uint count) { DOG_WDI=!DOG_WDI; DOGTIME=0; while(count--) { I2C_Start(); /*I2C_Send_Byte( 0xa0 + AT24C64_address / 256 *2 );*/ /* 24C16 USE */ I2C_Send_Byte( 0xa0 ); I2C_Send_Byte( AT24C64_address/256 ); I2C_Send_Byte( AT24C64_address % 256 ); I2C_Start(); /*I2C_Send_Byte( 0xa1 + AT24C64_address /256 *2 );*/ I2C_Send_Byte( 0xa1 ); *(uchar*)mcu_address = I2C_Receive_Byte(); I2C_Nack(); I2C_Stop(); ((uchar*)mcu_address)++; AT24C64_address++; } } |
|
|
沙发#
发布于:2005-04-24 13:30
一个通用的24C01-24C256共9种EEPROM的字节读写操作程序,
此程序有五个入口条件,分别为读写数据缓冲区指针, 进行读写的字节数,EEPROM首址,EEPROM控制字节, 以及EEPROM类型。此程序结构性良好,具有极好的容错性,程序机器码也不多: #pragma ot(6,SIZE) #include <reg52.h> #include <intrins.h> #define ERRORCOUNT 10 sbit SDA=P0^0; sbit SCL=P0^1; enum eepromtype {M2401,M2402,M2404,M2408,M2416,M2432,M2464,M24128,M24256}; enum eepromtype EepromType; //DataBuff为读写数据输入/输出缓冲区的首址 //ByteQuantity 为要读写数据的字节数量 //Address 为EEPROM的片内地址 //ControlByte 为EEPROM的控制字节,具体形式为(1)(0)(1)(0)(A2)(A1)(A0)(R/W),其中R/W=1, //表示读操作,R/W=0为写操作,A2,A1,A0为EEPROM的页选或片选地址; //EepromType为枚举变量,需为M2401至M24256中的一种,分别对应24C01至24C256; //函数返回值为一个位变量,若返回1表示此次操作失效,0表示操作成功; //ERRORCOUNT为允许最大次数,若出现ERRORCOUNT次操作失效后,则函数中止操作,并返回1 //SDA和SCL由用户自定义,这里暂定义为P0^0和P0^1; //其余的用户不用管,只要把只子程序放在你的程序中并调用它就可以了; /***********************************************************************************/ bit RW24XX(unsigned char *DataBuff,unsigned char ByteQuantity,unsigned int Address, unsigned char ControlByte,enum eepromtype EepromType) { void Delay(unsigned char DelayCount); void IICStart(void); void IICStop(void); bit IICRecAck(void); void IICNoAck(void); void IICAck(void); unsigned char IICReceiveByte(void); void IICSendByte(unsigned char sendbyte); unsigned char data j,i=ERRORCOUNT; bit errorflag=1; while(i--) { IICStart(); IICSendByte(ControlByte&0xfe); if(IICRecAck()) continue; if(EepromType>M2416) { IICSendByte((unsigned char)(Address>>8)); if(IICRecAck()) continue; } IICSendByte((unsigned char)Address); if(IICRecAck()) continue; if(!(ControlByte&0x01)) { j=ByteQuantity; errorflag=0; //********clr errorflag while(j--) { IICSendByte(*DataBuff++); if(!IICRecAck()) continue; errorflag=1; break; } if(errorflag==1) continue; break; } else { IICStart(); IICSendByte(ControlByte); if(IICRecAck()) continue; while(--ByteQuantity) { *DataBuff++=IICReceiveByte(); IICAck(); } *DataBuff=IICReceiveByte(); //read last byte data IICNoAck(); errorflag=0; break; } } IICStop(); if(!(ControlByte&0x01)) { Delay(255); Delay(255); Delay(255); Delay(255); } return(errorflag); } /*****************以下是对IIC总线的操作子程序***/ /*****************启动总线**********************/ void IICStart(void) { SCL=0; // SDA=1; SCL=1; _nop_(); _nop_(); _nop_(); SDA=0; _nop_(); _nop_(); _nop_(); _nop_(); SCL=0; SDA=1; // } /*****************停止IIC总线****************/ void IICStop(void) { SCL=0; SDA=0; SCL=1; _nop_(); _nop_(); _nop_(); SDA=1; _nop_(); _nop_(); _nop_(); SCL=0; } /**************检查应答位*******************/ bit IICRecAck(void) { SCL=0; SDA=1; SCL=1; _nop_(); _nop_(); _nop_(); _nop_(); CY=SDA; //因为返回值总是放在CY中的 SCL=0; return(CY); } /***************对IIC总线产生应答*******************/ void IICACK(void) { SDA=0; SCL=1; _nop_(); _nop_(); _nop_(); _nop_(); SCL=0; _nop_(); SDA=1; } /*****************不对IIC总线产生应答***************/ void IICNoAck(void) { SDA=1; SCL=1; _nop_(); _nop_(); _nop_(); _nop_(); SCL=0; } /*******************向IIC总线写数据*********************/ void IICSendByte(unsigned char sendbyte) { unsigned char data j=8; for(;j>0;j--) { SCL=0; sendbyte<<=1; //无论C51怎样实现这个操作,始终会使CY=sendbyte^7; SDA=CY; SCL=1; } SCL=0; } /**********************从IIC总线上读数据子程序**********/ unsigned char IICReceiveByte(void) { register receivebyte,i=8; SCL=0; while(i--) { SCL=1; receivebyte=(receivebyte<<1)|SDA; SCL=0; } return(receivebyte); } /***************一个简单延时程序************************/ void Delay(unsigned char DelayCount) { while(DelayCount--); } |
|
|
板凳#
发布于:2005-04-24 13:47
FM24C64程序
#define WDI P3_4 #define WP P1_1 #define SCL P3_7 #define SDA P1_0 #define FM24c64_address 0xa0 //A2,A1,A0,接GND。 void Fm24c64_start()//iic总线开始 { SDA=1; _nop_();//6M.晶振这一指令执行2us SCL=1; _nop_(); SDA=0; _nop_(); SCL=0; _nop_(); } void Fm24c64_stop() //IIC总线停止 { SDA=0; _nop_();//6M.晶振这一指令执行2us SCL=1; _nop_(); SDA=1; _nop_(); } void Fm24c64_send_8(BYTE dat)//内部函数,发送8位数据 { BYTE j=8; for(j=0;j<8;j++) { SDA=(bit)(dat&0x80); dat<<=1; SCL=1; _nop_(); SCL=0; } SDA=1; _nop_(); _nop_(); while(SDA); SCL=1; _nop_(); SCL=0; } BYTE Fm24c64_receive_8() { BYTE value=0,j=8; for(j=0;j<8;j++) { SCL=1; value<<=1; if(SDA) { value|=0x01; } SCL=0; _nop_(); } return(value); } void Fm24c64_write_byte(WORD address,BYTE dat) { BYTE value; Fm24c64_start(); value=FM24c64_address&0xfe; //发送器件地址 Fm24c64_send_8(value); value=address/0x100;//发送地址高字节 Fm24c64_send_8(value); value=address%0x100;//发送地址低字节 Fm24c64_send_8(value); value=dat; Fm24c64_send_8(value);//发送数据 Fm24c64_stop(); } void Fm24c64_write_bytes(WORD address,BYTE *pdat,BYTE length) { BYTE value,i; Fm24c64_start(); value=FM24c64_address&0xfe; //发送器件地址 Fm24c64_send_8(value); value=address/0x100;//发送地址高字节 Fm24c64_send_8(value); value=address%0x100;//发送地址低字节 Fm24c64_send_8(value); for(i=0;i<length;i++) { value=*pdat; Fm24c64_send_8(value);//发送数据 pdat++; } Fm24c64_stop(); } BYTE Fm24c64_read_byte(WORD address)//读指定的地址的数据 { BYTE value; Fm24c64_start(); value=FM24c64_address&0xfe; //发送器件地址 Fm24c64_send_8(value); value=address/0x100;//发送地址高字节 Fm24c64_send_8(value); value=address%0x100;//发送地址低字节 Fm24c64_send_8(value); Fm24c64_stop(); Fm24c64_start(); value=FM24c64_address|0x01; //发送器件地址 Fm24c64_send_8(value); value=Fm24c64_receive_8(); SDA=1; //主不应答信号 SCL=1; _nop_(); SCL=0; Fm24c64_stop(); return(value); } void Fm24c64_read_bytes(WORD address,BYTE *pdat,BYTE length)//读指定地址后的连续的数据 { BYTE value,i; Fm24c64_start(); value=FM24c64_address&0xfe; //发送器件地址 Fm24c64_send_8(value); value=address/0x100;//发送地址高字节 Fm24c64_send_8(value); value=address%0x100;//发送地址低字节 Fm24c64_send_8(value); Fm24c64_stop(); Fm24c64_start(); value=FM24c64_address|0x01; //发送器件地址 Fm24c64_send_8(value); i=length-1; while(i--) { *pdat=Fm24c64_receive_8(); _nop_(); SDA=1; // _nop_(); SDA=0; //主应答信号 SCL=1; _nop_(); SCL=0; SDA=1; pdat++; } *pdat=Fm24c64_receive_8(); SDA=1; //主不应答信号 SCL=1; _nop_(); SCL=0; Fm24c64_stop(); } 大侠们指点,有时间可以交流,可以互相帮助,有项目可以同做!!!!!!! 祝你们圣诞快乐。 本人信箱,16778@sina.com.cn 本人还想写一个用Protel制作的从SCH到PCB的流程步骤,画电路板的话,减少反复工作,制板一次成功,望大家互相提好的建议!!!!!!!!!!!! |
|
|
地板#
发布于:2005-04-24 13:47
HT48 MCU 读写HT24 系列EEPROM 的应用范例
HT48 MCU 读写HT24 系列EEPROM 的应用范例 文件编码:HA0016s 简介: HT24 系列的EEPROM 是通过I2C 协议控制其读写的。HT48 系列单片机的接口部分是CMOS I/O 口,可以用来很方便地采用I2C 协议控制周边器件。 HT24 系列的EEPROM 总共8 个管脚,三个为芯片地址脚A0、A1、A2,在单片机对它进行操 作时,从SDA 输入A0、A1、A2 数据和芯片外部A0、A1、A2 所接地址需一一对应。一个为芯片写 保护脚WP,WP 脚接低电平时,芯片可进行读写操作;WP 脚接高时,芯片只可进行读,不可进行写。 另外两个管脚为电源脚VCC,VSS。 用单片机对HT24 系列的EEPROM 进行控制时,HT24 系列的EEPROM 的外部管脚VCC、VSS、 WP、A0、A1、A2 根据需要,对应接上,SDA、SCL 接到单片机控制脚上。 引脚名称I/O 功能描述 A0~A2 I 地址输入 VSS I 电源负极输入 SDA I/O 串行数据输入/输出 SCL I 串行数据传送时钟信号输入 WP I 写保护 VCC I 电源正极输入 HT24 系列的EEPROM 根据型号不同,EEPROM 的容量大小不同,当EEPROM 的空间大于1 页(256bytes)时,即大于2048bits,则HT48 MCU 需要控制需要控制A0、A1、A2 来确定写HT24 系列的EEPROM 的第几页,HT24 系列的EEPROM 空间大小如下表所示: 型号引脚A0、A1 及A2 使用方法容量大小 HT24LC02 A0、A1、A2 引脚作为器件地址输入,从SDA 输入A0、A1、 A2 数据和芯片引脚A0、A1、A2 所接状态需一一对应 2K(256×8) HT24LC04 A1、A2 引脚作为器件地址输入,从SDA 输入A1、A2 数据 和芯片引脚A1、A2 所接状态需一一对应,A0 引脚浮空 4K(512×8, 2pages) HT24LC08 A2 引脚器件地址输入,从SDA 输入A2 数据和芯片引脚A2 所接状态需一一对应,其余引脚浮空 8K(1024×8, 4pages) HT24LC16 A0、A1、A2 全部浮空,不必接 16K(2048×8, 8pages) HT48 MCU 读写HT24 系列EEPROM 的应用范例 使用说明: 本文是以HT48R30A-1 控制HT24LC04 为例的。I2C 协议只需要两根线进行控制,一根时钟线 SCL,一根数据线SDA。用单片机对HT24LC04 进行控制时,HT24LC04 外部管脚VCC、VSS、WP、 A1、A2 根据需要,对应接上(本例中是与VSS 相接,即A1、A2=00),SDA、SCL 接到单片机控 制脚上。在这个例程中SCL 接到pa.3 脚,SDA 接到pa.1 脚。程序的过程是这样的:先向EEPROM 中某个地址写55H,写完后,再将EEPROM 中内容读出来,并将读出数据和55H 进行比较,若数据 不相等程序跳到fail_out 中;若相等,最后程序跳到ok_end 中。 例程中只是对第0 页中的特定地址进行写操作后,再将该地址的内容读出。关于HT24 系列的 EEPROM 其它操作只要严格按其时序要求,并注意从SDA 输入A0、A1、A2 数据和芯片引脚A0、 A1、A2 所接状态需一一对应即可。 例程: ; |
|
|
地下室#
发布于:2005-04-24 13:48
到词一游
|
|
|
5楼#
发布于:2005-04-24 18:41
一个通用的24C01-24C256共9种EEPROM的字节读写操作程序, 推荐 :P |
|
|
6楼#
发布于:2005-05-11 01:10
不错,我最近就在做这方面的东西,真是受益非浅。推荐进精华
|
|
7楼#
发布于:2005-05-13 16:06
不错,我最近就在做这方面的东西,真是受益非浅。推荐进精华 谢谢....你用什么语言写的/ :P |
|
|
8楼#
发布于:2005-05-28 10:31
不错
24C64是采用I2C总线进行通讯的 以前,我是使用汇编编写的,今天,算是长了见识了 呵呵 :D |
|
9楼#
发布于:2005-06-26 10:39
好贴。顶!
我正在写i2c读写cat24wc64,数据有些不稳定。可能是写完之后延时不够。居然要延时10ms,真是太失望了!这样随机读写的速度就上不去了。要提高速度,可能只有用页读写才行。所谓的200k和400k的i2c总线,也只是i2c通讯的速度,根本不是读写存储器的速度。哎!!! |
|
10楼#
发布于:2005-07-31 02:51
好贴。顶!
|
|
|