20回答

0收藏

急,500铜板求STM8硬件IIC双机通信程序,,,

 

回答|共 20 个

倒序浏览

11#

cr531585

发表于 2014-1-26 08:43:23 | 只看该作者

分享到:
                              
12#

玉石俱焚

发表于 2014-1-26 09:42:24 | 只看该作者

这是我发送的程序:
void IIC_Init(void)
{
  I2C_FREQR = 0x0a;   //外设时钟频率10M
  I2C_CCRH = 0x03;    //
  I2C_CCRL = 0xe8;    //10KHZ
  I2C_TRISER = 0x04;  //上升时间
  I2C_CR2 |=0x04;    //收到一个字节后应答
  I2C_CR1 |=0x01;   //启用IIC模块
  
  I2C_OARH=0x40;
  I2C_OARL=0x03;
}

//主模式下的发送
void IIC_Host_TX(uchar add,uchar data)
{
  uchar temp;
  
  while(I2C_SR3 & 0x02);      //等待总线空闲
  UART_Send(2);
  I2C_CR2 |= 0x01;            //产生起始位
  while(!(I2C_SR1 & 0x01));   //等待起始位发送完成
  UART_Send(3);
  I2C_DR = add;               //写入器件地址
  while(!(I2C_SR1&0x02));     //等待器件地址发送结束
  UART_Send(4);
  temp = I2C_SR1;           
  temp = I2C_SR3;
  I2C_DR =data;               //发送数据
  while(!(I2C_SR1 & 0x84));   //等待数据字节发送完成
  UART_Send(5);
  I2C_CR2 |= 0x02;            //产生停止位   
}

这是我接收的程序:
void IIC_Init(void)
{
  I2C_ITR=0x06;
  I2C_FREQR = 0x0a;   //外设时钟频率10M
  I2C_CCRH = 0x03;    //
  I2C_CCRL = 0xe8;    //10KHZ
  I2C_TRISER = 0x04;  //上升时间
  I2C_CR2 |=0x04;    //收到一个字节后应答
  I2C_CR1 |=0x01;   //启用IIC模块
  
  I2C_OARH=0x40;
  I2C_OARL=0x02;
}

#pragma vector=0x15
__interrupt void I2C_IRQHandler(void)
{

  UART_Send(1);
  if(I2C_SR1&0x02)    //地址匹配
  {
    UART_Send(2);
    temp = I2C_SR1;           
    temp = I2C_SR3;
  }
  
  if(I2C_SR1&0x40)    //接收数据寄存器非空
  {
    UART_Send(3);
    IIC_Data=I2C_DR;
    UART_Send(IIC_Data);
  }
  
  if(I2C_SR1&0x10)    //检测到停止位
  {
    UART_Send(4);
    I2C_CR2|=0x02;    //释放总线
  }
}


发送在  while(!(I2C_SR1&0x02));     //等待器件地址发送结束  这里就卡了
接收只打印了1和4,
13#

Mandelbrot_Set

发表于 2014-1-26 10:21:21 | 只看该作者

虽然我不知道为什么,
但我觉得可以抓波形分析下,另外可以用已知i2c完全没有问题的设备辅助调试
▽·▽×A = 0
14#

abu315

发表于 2014-1-26 11:28:08 | 只看该作者

本帖最后由 abu315 于 2014-1-26 11:41 编辑

这是我以前用C51写的模拟IIC协议,希望对你有用
/********************************************************************
                       VIIC_C51.C
  此程序是I2C操作平台(主方式的软件平台)的底层的C子程序,如发送数据
及接收数据,应答位发送,并提供了几个直接面对器件的操作函数,它很方便的
与用户程序连接并扩展.....  
  
    注意:函数是采用软件延时的方法产生SCL脉冲,固对高晶振频率要作
一定的修改....(本例是1us机器周期,即晶振频率要小于12MHZ)

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

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

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

/* 常,变量定义区 */

/*端口位定义*/
sbit SDA=P3^3;            /*模拟I2C数据传送位*/
sbit SCL=P3^2;            /*模拟I2C时钟控制位*/


/*******************************************************************
                     起动总线函数               
函数原型: 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表示被控器无应答或损坏。
********************************************************************/

/*******************************************************************
                 字节数据传送函数               
函数原型: bit  SendByte(uchar c);
功能:  将数据c发送出去,可以是地址,也可以是数据,发完后等待应答,并对
     此状态位进行操作.(不应答或非应答都使ack=0 假)     
     发送数据正常,返回1; 返回0表示被控器无应答或损坏。
********************************************************************/
bit  SendByte(uchar c)
{
   
uchar BitCnt;
bit ack;

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();
   
    return ack;
}






/*******************************************************************
                 字节数据接收函数               
函数原型: 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);

功能:主控器进行应答信号,(可以是应答或非应答信号)
********************************************************************/
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();               //启动总线   
    if(SendByte(sla)==0)       //发送器件地址
        return(0);   
    if(SendByte(c)==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();                         /*启动总线*/
    if( SendByte(sla)==0)return(0);      /*发送器件地址*/
    if( SendByte(suba)==0)return(0);     /*发送器件子地址*/
    for(i=0;i<no;i++)
    {   
        if( SendByte(*s)==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();                        //启动总线   
    if(SendByte(sla+1)==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();                        /*启动总线*/   
    if(SendByte(sla)==0)return(0);      /*发送器件地址*/               
    if(SendByte(suba)==0)return(0);     /*发送器件子地址*/
    Start_I2c();
    if( SendByte(sla+1)==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);
}
15#

motodefy

发表于 2014-1-26 16:33:14 | 只看该作者

双机通信倒是没试过··不过这个应该要上拉电阻什么的吧···猜的
16#

稻爷的学徒

发表于 2014-1-26 16:42:46 | 只看该作者

直接 单片机 A 发数据,单片机 B 接到数据 回发 给 A , A校验,岂不是简单
17#

陈克雄

发表于 2014-2-2 16:11:54 | 只看该作者

没用过STM8呢,可以先试试模拟IIC呀
18#

玉石俱焚

发表于 2014-2-2 16:32:26 | 只看该作者

Kite 发表于 2014-2-2 16:11
没用过STM8呢,可以先试试模拟IIC呀

模拟IIC怎么做从机呢,
19#

zizhou

发表于 2014-2-11 19:55:51 | 只看该作者

还是没人回答、、、、、
20#

骑蜗牛闯红灯

发表于 2015-7-27 09:12:08 | 只看该作者

卡在哪里要不就是初始化的问题 要不就是硬件的问题 你仔细看看例程应该对你有帮助 官方例程值得我们学习
您需要登录后才可以回帖 注册/登录

本版积分规则

关闭

站长推荐上一条 /3 下一条