完整代码下载地址: http://www.51hei.com/f/2402avr.rar
main函数: /* 程序功能:ATMEGA16读写24C02 作者:朱波 时间:2012年1月30日 说明:程序运行后会从0开始自动计数到256(对字节操作的结果) 当复位或者断电时24C02保存断电时的数据,当重新启动后 计数值会从断电时的值开始计数 */ #include <iom16v.h> #include <macros.h> #include"delay.h" #include"display.h" #include"24C02.h" #define uchar unsigned char #define uint unsigned int uint num;//定义变量用于读取24C02中的数值,注意必须为uint型 void main() { LCD_init(); twi_init(); while(1) { num=IIC_Read(0x00);//从器件中读取(0x00是器件地址) num=num+1;//加1 num=num%1000;//关键语句,没有此句则不工作*/ IIC_Write(num,0x00);//加1后写入器件 display(); } } 延时函数delay.c #include"delay.h" void delay(unsigned int ms) { unsigned int i,j; for(i=0;i<ms;i++) { for(j=0;j<1141;j++); } } 显示函数display.c #include <iom16v.h> #include <macros.h> #include"display.h" #define uchar unsigned char #define uint unsigned int uchar table1[]="0123456789"; uchar table2[]="zhubo-24C02"; uchar table3[]="result:"; extern uint num;//主函数中的值 void write_com(uchar com) { PORTD&=~BIT(4); PORTD&=~BIT(5); PORTB=com; PORTD|=BIT(6); delay(1); PORTD&=~BIT(6); } void write_dat(uchar dat) { PORTD|=BIT(4); PORTD&=~BIT(5); PORTB=dat; PORTD|=BIT(6); delay(1); PORTD&=~BIT(6); } void LCD_init() { DDRB=0XFF; DDRD|=BIT(4)|BIT(5)|BIT(6); PORTD&=~BIT(6); write_com(0X38); delay(5); write_com(0X01); delay(5); write_com(0X0C); delay(5); write_com(0X06); delay(5); } void display() { uint i; write_com(0X80+0); delay(5); for(i=0;i<7;i++) { write_dat(table3[i]); delay(5); } //这是在1602液晶中显示数值的方法 write_com(0X80+8); delay(5); write_dat(table1[num/1000]); delay(5); write_com(0X80+9); delay(5); write_dat(table1[num%1000/100]); delay(5); write_com(0X80+10); delay(5); write_dat(table1[num%100/10]); delay(5); write_com(0X80+11); delay(5); write_dat(table1[num%10]); delay(5); write_com(0X80+0X40);//仿真与实际1602有区别 delay(5); for(i=0;i<11;i++) { write_dat(table2[i]); delay(5); } } 24C02.c: #include <iom16v.h> #include <macros.h> #include"24C02.h" #define uchar unsigned char #define uint unsigned int //主机模式启动状态码 #define I2C_START 0x08 //启动总线 #define I2C_RESTART 0x10 //重新启动总线 //主机发送模式状态码 #define I2C_MT_SLA_ACK 0x18 //SLA+W写地址已发送,收到应答位 #define I2C_MT_SLA_NACK 0x20 //SLA+W写地址已发送,收到非应答位 #define I2C_MT_DATA_ACK 0x28 //写入数据已发送,收到应答位 #define I2C_MT_DATA_NACK 0x30 //写入数据已发送,收到应答位 #define I2C_MT_ARB_LOST 0x38 //SLA+W或数据仲裁失败 //主机接收模式状态码 #define I2C_MR_ARB_LOST 0x38 //SLA+R或NOT ACK的仲裁失败 #define I2C_MR_SLA_ACK 0x40 //SLA+R已发送,收到应答位 #define I2C_MR_SLA_NACK 0x48 //SLA+R已发送,收到非应答位 #define I2C_MR_DATA_ACK 0x50 //接收到数据,应答位已返回 #define I2C_MR_DATA_NACK 0x58 //接收到数据,非应答位已返回 //从机接收模式状态码 #define I2C_SR_SLA_ACK 0x60 //自己的SLA+W已经被接收ACK已返回 #define I2C_SR_ARB_LOST_SLA_ACK 0x68 //SLA+R/W作为主机的仲裁失败,自己的SLA+W已经被接收ACK已返回 #define I2C_SR_GCALL_ACK 0x70 //接收到广播地址ACK已返回 #define I2C_SR_ARB_LOST_GCALL_ACK 0x78 //SLA+R/W作为主机的仲裁失败,接收到广播地址ACK已返回 #define I2C_SR_DATA_ACK 0x80 //以前以自己的SLA+W被寻址,数据已经被接收ACK已返回 #define I2C_SR_DATA_NACK 0x88 //以前以自己的SLA+W被寻址,数据已经被接收NOT ACK已返回 #define I2C_SR_GCALL_DATA_ACK 0x90 //以前以广播方式被寻址,数据已经被接收ACK已返回 #define I2C_SR_GCALL_DATA_NACK 0x98 //以前以广播方式被寻址,数据已经被接收NOT ACK已返回 #define I2C_SR_STOP 0xA0 //在以从机工作时接收到STOP或重复START //从机发送模式状态码 #define I2C_ST_SLA_ACK 0xA8 //自己的SLA+R已经被接收ACK已返回 #define I2C_ST_ARB_LOST_SLA_ACK 0xB0 //SLA+R/W作为主机的仲裁失败,自己的SLA+R已经被接收ACK已返回 #define I2C_ST_DATA_ACK 0xB8 //TWDR里数据已经被发送接收到ACK #define I2C_ST_DATA_NACK 0xC0 //TWDR里数据已经被发送接收到NOT ACK #define I2C_ST_LAST_DATA 0xC8 //TWDR的一字节数据已经发送(TWAE='0'),接收到ACK //其他状态码 #define I2C_NO_INFO 0xF8 //没有相关的状态信息,TWINT='0' #define I2C_BUS_ERROR 0x00 //由于非法的START或STOP引起的总线错误 //定义SLA中读写控制位极性 #define I2C_READ 1 #define I2C_WRITE 0 #define RD_DEVICE_ADDR 0XA1 //写的器件地址 #define WD_DEVICE_ADDR 0XA0 //读的器件地址 //常用TWI操作(主模式写和读) #define Start() (TWCR=(1<<TWINT)|(1<<TWSTA)|(1<<TWEN)) #define Stop() (TWCR=(1<<TWINT)|(1<<TWEN)|(1<<TWSTO)) #define Wait() {while(!(TWCR&(1<<TWINT)));} #define TestAck() (TWSR&0XF8) #define SetAck (TWCR|=(1<<TWEA)) #define SetNoAck (TWCR&=~(1<<TWEA)) #define Twi() (TWCR=(1<<TWINT)|(1<<TWEN))//重新启动发送数据 #define Write8Bit(x) {TWDR=(x);TWCR=(1<<TWINT)|(1<<TWEN);}//装入SLA_W到TWDR //寄存器,TWINT位清零,启动发送地址 //定义运行状态返回值极性 #define I2C_ERR 0 //错误 #define I2C_CRR 1 //正确 //定义TWINT位置位查询等待时间 #define WAITCOUNT 50 //3.6864M时,此处必须大于60,验证得知小于60,程序均不能正常工作 void twi_init()//硬件初始化 { TWBR=0X20;//TWBR的值不小于10 TWCR=0X04;//使能TWI操作 TWSR=0;//1分频 } uchar IIC_Write(uchar wdata,uchar addr)//字节写操作 { //按照时序来写 Start();//开始 Wait();//等待 if(TestAck()!=I2C_START)//测试 return 1;// 如果出错则返回1 Write8Bit(WD_DEVICE_ADDR);//写器件地址(SLAVE ADDRESS) Wait();//等待 if(TestAck()!=I2C_MT_SLA_ACK )//检验TWI状态寄存器,屏蔽 return 1;//预分频位,如果状态字不是SLA+R已发送,收到应答位 //转出错处理 ,(SLA针对的是器件地址)此时时序中第一个ACK结束 Write8Bit(addr);//写数据地址(要存储到哪个字节上BYTE ADDRESS) Wait();//等待 if(TestAck()!=I2C_MT_DATA_ACK )//检验TWI状态寄存器,屏蔽 return 1;//预分频位,如果状态字不是写入数据已发送,收到应答位 //转出错,到此时序中第二个ACK结束 Write8Bit(wdata);//写数据(DATA) Wait();//等待 if(TestAck()!=I2C_MT_DATA_ACK)//DATA和ADDRESS是EEPROM内部的就用 return 1; //I2C_MT_DATA_ACK Stop(); return 0;//运行到STOP处都没有出错就返回0 } uchar IIC_Read(uchar addr)//根据选择读时序来写的 { uchar temp;//定义此变量从单片机中读取数据 //此段与上个函数相同 Start(); Wait(); if(TestAck()!=I2C_START)//测试 return 1; //此段与上个函数相同 Write8Bit(WD_DEVICE_ADDR);//写器件地址(SLAVE ADDRESS) Wait(); if(TestAck()!=I2C_MT_SLA_ACK)//sla是器件外部 return 1;//此处时序下方的第一个ACK完成 Write8Bit(addr);//写存储地址(BYTE ADDRESS) Wait(); if(TestAck()!=I2C_MT_DATA_ACK)//器件内部用I2C_MT_DATA_ACK return 1;//此处时序下方的第二个ACK完成 Start();//根据时序又出现了START Wait(); if(TestAck()!=I2C_RESTART)//判断是否是重复START出现 return 1;//如果出错则返回1 Write8Bit(RD_DEVICE_ADDR);//读器件地址(SLAVE ADDRESS) Wait(); if(TestAck()!=I2C_MR_SLA_ACK)//现在是主机接收所以不是T而是R return 1;//此处时序下方的第三个ACK完成 Twi();//重新启动TWI总线 Wait(); if(TestAck()!=I2C_MR_DATA_NACK)//向器件里读数据,时序中为NO ACK return 1;//程序到了这里数据已经到了单片机里边了 temp=TWDR;//从数据寄存器中读取数据 Stop();//发送停止信号 return temp; } 以下是接口函数代码: delay.h: #ifndef delay_h #define delay_h extern void delay(unsigned int ms); #endif display.h: #define uchar unsigned char #define uint unsigned int #ifndef display_h 24C02.h: #define uchar unsigned char #define uint unsigned int extern void twi_init(); extern uchar IIC_Write(uchar wdata,uchar addr); extern uchar IIC_Read(uchar addr); #define display_h extern void write_com(uchar com); extern void write_dat(uchar dat); extern void LCD_init(); extern void display(); extern uchar table[]; #endif