专注电子技术学习与研究
当前位置:单片机教程网 >> MCU设计实例 >> 浏览文章

ATMEGA16读写24C02 C51程序

作者:佚名   来源:本站原创   点击数:  更新时间:2012年03月02日   【字体:



 
完整代码下载地址: 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
关闭窗口

相关文章