MCS-51单片机模拟I2C软件包(转)

注意:普通M51单片机机器周期=12个时钟周期,f020机器周期=时钟周期

对于nop指令,F020是一个时钟周期,例如f020晶振为22.1184M, 约为45ns

 

/*函数是采用软件延时的方法产生SCL脉冲,固对高晶振频率要作 一定的修改....(本软件包是1us机器周期,即晶振频率要小于12MHZ)总线时序符合I2C标准模式,100Kbit/S。*/

#include <reg51.h>
#include <intrins.h>

#define  uchar unsigned char  /*宏定义*/
#define  uint  unsigned int

#define  _Nop()  _nop_()        /*定义空指令 1us*/


sbit SDA=P3^4;             /*模拟I2C数据传送位*/
sbit SCL=P3^5;             /*模拟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);
功能:      主控器进行应答信号(可以是应答或非应答信号,由位参数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  ISendStr(uchar sla,ucahr *s,uchar no); 
功能:     从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件
          地址sla,发送内容是s指向的内容,发送no个字节。
          如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
********************************************************************/
bit ISendStrExt(uchar sla,uchar *s,uchar no)
{
   uchar i;

   Start_I2c();               /*启动总线*/
   SendByte(sla);             /*发送器件地址*/
   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);
}


/*******************************************************************
                    向无子地址器件读取多字节数据函数              
函数原型: bit  ISendStrExt(uchar sla,ucahr *s,uchar no); 
功能:     从启动总线到发送地址,读数据,结束总线的全过程.
          从器件地址sla,读出的内容放入s指向的存储区,
          读no个字节。如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
********************************************************************/
bit IRcvStrExt(uchar sla,uchar *s,uchar no)
{
   uchar i;

   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);
}

一般来说,用汇编语言编写的程序兼容性、可移植性、可读性都很差。不同程序员编写的程序在函数参数传递上采取的策略各不相同,直接造成移植方面的困难。因此,复制本软件包到您的工程中去,未必马上就能调通,还需要做些必要的调整。 程序文件说明: main.ASM 主程序文件,请按照注释在适当的地方添加您的代码。 Branch.INC 实用的分支判断程序,好好利用能够大大简化编程工作。 Calc.INC 实用的运算类程序。 Pointer.INC 通用指针处理程序。如果您不了解“通用指针”是什么,请打开该文件看开头的简要说明。如果想深入了解,请研究“C:\Keil\C51\HLP\C51.PDF”中的相关内容。 I2C.INC 简单实用的I/O模拟I2C总线的驱动程序,是在C51软件包的基础上翻译成汇编的。 附:本软件包采用的函数参数传递规则 【系统寄存器】 PC、PSW、SP、A、B、C、R0、R1、DPTR 【函数返回值】 ============================================================ 返回类型 保存在 说明 ------------------------------------------------------------ bit(位) CY byte(字节) ACC word(字) R6,R7 R6-高8位,R7-低8位 ptr(通用指针) R1,DPTR R1-存储类型,DPTR-存储地址 ============================================================ 【函数参数】 ============================================================ bit byte word ptr ------------------------------------------------------------ 第1参数 20H.0 R7 R6,R7 08H,09H,0AH 第2参数 20H.1 R6 R4,R5 0BH,0CH,0DH 第3参数 20H.2 R5 0EH,0FH 第4参数 20H.3 R4 0CH,0DH 第5参数 20H.4 0FH 第6参数 20H.5 0EH 第7参数 20H.6 0DH 第8参数 20H.7 0CH ============================================================ 系统寄存器不允许用于函数参数的传递 参数的传递顺序有严格要求,必须遵守 函数的参数通常都比较少,对偶尔出现参数过多的情况,请自行约定传递方式 【局部变量】 bit变量从21H.0~21H.7分配,分配前21H要入栈保护 byte、word变量从R2、R3、18H~1FH分配,分配前要入栈保护 参数传递剩余的寄存器可作为寄存器变量使用 系统寄存器总是临时性的,进入函数后,不必入栈保护 为了提高效率,系统寄存器也可作为变量使用,但中间不能调用任何函数 变量的分配比较灵活,分配顺序无严格要求 局部IRAM数组从堆栈空间进行动态分配 【全局变量、堆栈】 全局bit变量从22H.0开始分配 全局byte、word变量(含全局数组)从30H开始分配 堆栈起始位置紧跟全局变量之后 函数的局部静态变量按全局变量来处理(命名时可前缀函数名)
#include<reg52> //包含单片机寄存器的头文件 #include <intrins> #define ulong unsigned long #define uchar unsigned char #define uint unsigned int #define LCD1602_PORT P2 //用LED1602_PORT表示P2总线接口 #define PCF8591 0x90 //PCF8591 地址 #define NOP() _nop_() /* 定义空指令 */ #define _Nop() _nop_() /*定义空指令*/ // 位定义 sbit SCL=P1^0; //I2C 时钟 sbit SDA=P1^1; //I2C 数据 sbit LCD1602_RS = P0^5; //定义1602液晶显示屏的数据/命令选择端,数据/命令(H/L) sbit LCD1602_RW = P0^6; //定义1602液晶显示屏的读/写选择端,读/写(H/L) sbit LCD1602_EN = P0^7; //定义1602液晶显示屏的使能端 bit ack; /*应答标志位*/ // 变量定义 uchar AD_CHANNEL; ulong xdata LedOut[8]; //单片机内部存取器 ulong v,a,ss; uchar date; // 函数申明 extern bit ack;//起动总线函数 extern void Start_I2c();//结束总线函数 extern void Stop_I2c();//应答子函数 extern void Ack_I2c(bit a);//字节数据发送函数 extern void SendByte(uchar c);//有子地址发送多字节数据函数 extern bit ISendStr(uchar sla,uchar suba,uchar *s,uchar no) ;//无子地址发送多字节数据函数 extern bit ISendStrExt(uchar sla,uchar *s,uchar no);//无子地址读字节数据函数 extern uchar RcvByte(); void LCD1602_delay_ms(uint n); void LCD1602_write_com(uchar com); void LCD1602_write_data(ulong dat); void LCD1602_write_word(uchar *s); void Init_LCD1602(); bit ISendByte(uchar sla,uchar c); uchar IRcvByte(uchar sla); //MS延时函数(12M晶振下测试) void delay_ms(uint n) { unsigned int i,j; for(i=0;i<n i++) for(j=0;j xss=removed xss=removed xss=removed xss=removed xss=removed xss=removed xss=removed xss=removed>0) { LCD1602_write_data(*s); s++; } } //1602初始化函数 void Init_LCD1602() { LCD1602_EN = 0; LCD1602_RW = 0;//设置为写状态 LCD1602_write_com(0x38);//显示模式设定 LCD1602_write_com(0x0c);//开关显示、光标有无设置、光标闪烁设置 LCD1602_write_com(0x06);//写
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值