单片机: EEPROM和串口通信

Posted wzjhoutai

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了单片机: EEPROM和串口通信相关的知识,希望对你有一定的参考价值。

 名称:IIC协议 EEPROM24c02 通过串口通信存数读取数据

 内容:此程序用于检測EEPROM性能,測试方法例如以下:写入24c02一个数据,然后在内存中改变这些数据。 掉电后主内存将失去这些信息,然后从24c02中调入这些数据。看是否与写入的同样。

电脑通过串口发送一个十六进制的数据到单片机,存储进24c02,要求断电重新启动后在数码管上显示上一次发送的数据。

(本例是1us机器周期,即晶振频率要小于12MHZ)

#include <reg52.h>          //头文件的包括
#include <intrins.h>

#define  _Nop()  _nop_()        //定义空指令
#define DataPort P0
sbit WEI=P2^7;
sbit DUAN=P2^6;
// 常,变量定义区
unsigned char code dofly_DuanMa[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,
		                  	         0x77,0x7c,0x39,0x5e,0x79,0x71,0x40,0x00};// 显示段码值0~F,-,全空
unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别相应相应的数码管点亮,即位码

unsigned char TempData[8];
                                              
sbit SDA=P2^1;            //模拟I2C数据传送位
sbit SCL=P2^0;            //模拟I2C时钟控制位

                          
bit ack;	              //应答标志位
   
unsigned char res;
void DelayUs2x(unsigned char t);//函数声明 
void DelayMs(unsigned char t);

void Delay(unsigned int t)
{
	while(t--);	
}

void InitUART(void)
{
	SCON=0x50;
	TMOD|=0x20;
	TH1=0xFD;
	TR1=1;
	EA=1;
}

void DelayUs2x(unsigned char t)
{   
 while(--t);
}

void DelayMs(unsigned char t)
{
     
 while(t--)
 {
     //大致延时1mS
     DelayUs2x(245);
	 DelayUs2x(245);
 }
}


void Display(unsigned char FirstBit,unsigned char Num)
{
	unsigned char i;
	for(i=0;i<Num;i++)
	{ 
	   	DataPort=0;   //清空数据。防止有交替重影
      	DUAN=1;     //段锁存
      	DUAN=0;

       	DataPort=dofly_WeiMa[i+FirstBit]; //取位码 
       	WEI=1;     //位锁存
       	WEI=0;

       	DataPort=TempData[i]; //取显示数据,段码
       	DUAN=1;     //段锁存
       	DUAN=0;
       
	   	Delay(200); // 扫描间隙延时。时间太长会闪烁,
                   //太短会造成重影

    }
}


/*------------------------------------------------
                    启动总线
------------------------------------------------*/
void Start_I2c()
{
  SDA=1;   //发送起始条件的数据信号
  _Nop();
  SCL=1;
  _Nop();    //起始条件建立时间大于4.7us,延时
  _Nop();
  _Nop();
  _Nop();
  _Nop();    
  SDA=0;     //发送起始信号
  _Nop();    //起始条件锁定时间大于4μ
  _Nop();
  _Nop();
  _Nop();
  _Nop();       
  SCL=0;    //钳住I2C总线。准备发送或接收数据
  _Nop();
  _Nop();
}
/*------------------------------------------------
                    结束总线
------------------------------------------------*/
void Stop_I2c()
{
  SDA=0;    //发送结束条件的数据信号
  _Nop();   //发送结束条件的时钟信号
  SCL=1;    //结束条件建立时间大于4μ
  _Nop();
  _Nop();
  _Nop();
  _Nop();
  _Nop();
  SDA=1;    //发送I2C总线结束信号
  _Nop();
  _Nop();
  _Nop();
  _Nop();
}

void  SendByte(unsigned char c)
{
 unsigned char BitCnt;
 
 for(BitCnt=0;BitCnt<8;BitCnt++)  //要传送的数据长度为8位
    {
     if((c<<BitCnt)&0x80)SDA=1;   //推断发送位
       else  SDA=0;                
     _Nop();
     SCL=1;               //置时钟线为高,通知被控器開始接收数据位
      _Nop(); 
      _Nop();             //保证时钟高电平周期大于4μ
      _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();
}
	
unsigned char  RcvByte()
{
  unsigned char retc;
  unsigned char BitCnt;
  
  retc=0; 
  SDA=1;             //置数据线为输入方式
  for(BitCnt=0;BitCnt<8;BitCnt++)
      {
        _Nop();           
        SCL=0;       //置时钟线为低,准备接收数据位
        _Nop();
        _Nop();      //时钟低电平周期大于4.7us
        _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(void);
 
----------------------------------------------------------------*/
void Ack_I2c(void)
{
  
  SDA=0;     
  _Nop();
  _Nop();
  _Nop();      
  SCL=1;
  _Nop();
  _Nop();              //时钟低电平周期大于4μ
  _Nop();
  _Nop();
  _Nop();  
  SCL=0;               //清时钟线,钳住I2C总线以便继续接收
  _Nop();
  _Nop();    
}
/*----------------------------------------------------------------
                     非应答子函数
原型:  void NoAck_I2c(void);
 
----------------------------------------------------------------*/
void NoAck_I2c(void)
{
  
  SDA=1;
  _Nop();
  _Nop();
  _Nop();      
  SCL=1;
  _Nop();
  _Nop();              //时钟低电平周期大于4μ
  _Nop();
  _Nop();
  _Nop();  
  SCL=0;                //清时钟线。钳住I2C总线以便继续接收
  _Nop();
  _Nop();    
}






/*----------------------------------------------------------------
                    向无子地址器件发送字节数据函数               
函数原型: bit  ISendByte(unsigned char sla,ucahr c);  
功能:     从启动总线到发送地址。数据,结束总线的全过程,从器件地址sla.
           假设返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。

----------------------------------------------------------------*/ /*bit ISendByte(unsigned char sla,unsigned char c) { Start_I2c(); //启动总线 SendByte(sla); //发送器件地址 if(ack==0)return(0); SendByte(c); //发送数据 if(ack==0)return(0); Stop_I2c(); //结束总线 return(1); } */ /*---------------------------------------------------------------- 向有子地址器件发送多字节数据函数 函数原型: bit ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no); 功能: 从启动总线到发送地址。子地址,数据,结束总线的全过程,从器件 地址sla,子地址suba。发送内容是s指向的内容,发送no个字节。 假设返回1表示操作成功。否则操作有误。 注意: 使用前必须已结束总线。 ----------------------------------------------------------------*/ bit ISendStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no) { unsigned char 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(unsigned char sla,ucahr *c); 功能: 从启动总线到发送地址,读数据。结束总线的全过程,从器件地 址sla,返回值在c. 假设返回1表示操作成功,否则操作有误。

注意: 使用前必须已结束总线。 ----------------------------------------------------------------*/ /*bit IRcvByte(unsigned char sla,unsigned char *c) { Start_I2c(); //启动总线 SendByte(sla+1); //发送器件地址 if(ack==0)return(0); *c=RcvByte(); //读取数据 NoAck_I2c(); //发送非就答位 Stop_I2c(); //结束总线 return(1); } */ /*---------------------------------------------------------------- 向有子地址器件读取多字节数据函数 函数原型: bit ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no); 功能: 从启动总线到发送地址。子地址,读数据,结束总线的全过程,从器件 地址sla。子地址suba,读出的内容放入s指向的存储区。读no个字节。

假设返回1表示操作成功,否则操作有误。

注意: 使用前必须已结束总线。

----------------------------------------------------------------*/ bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no) { unsigned char 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(); //发送就答位 s++; } *s=RcvByte(); NoAck_I2c(); //发送非应位 Stop_I2c(); //结束总线 return(1); } /*------------------------------------------------ 主函数 ------------------------------------------------*/ void main() { unsigned char doflye; // 定义暂时变量 unsigned char i; IRcvStr(0xae,4,&doflye,1); //调用存储数据 TempData[0]=dofly_DuanMa[doflye/16]; TempData[1]=dofly_DuanMa[doflye%16]; InitUART(); ES=1; while(1) { Display(0,2); doflye=res; ISendStr(0xae,4,&doflye,1); //写入24c02 } } void UART_SER(void) interrupt 4 { unsigned char Temp; // unsigned char i; if(RI) { RI=0; Temp=SBUF; res=Temp; TempData[0]=dofly_DuanMa[Temp/16]; TempData[1]=dofly_DuanMa[Temp%16]; } if(TI) TI=0; }



以上是关于单片机: EEPROM和串口通信的主要内容,如果未能解决你的问题,请参考以下文章

51单片机串口通讯

谁能把单片机串口通信SFR介绍下?

5V单片机与3.3V单片机串口通信问题

Arduino和C51之串口通信

单片机串口通信控制led灯的点亮

5V单片机与3.3V单片机串口通信问题