STM23模拟I2C读写

发布者:MusicLover123最新更新时间:2015-10-21 来源: eefocus关键字:STM23  模拟I2C读写 手机看文章 扫描二维码
随时随地手机看文章
都说STM32的I2C有问题,不好用。我之前,在论坛上看到了Mcuplayer分享的一段代码,拿来测试了一下,读写正常,心想还挺不错。
但是等到真正做程序时,发现总是在while()循环处等待。无奈,只好用软件模拟I2C。
从网上找了一段程序,发现好多地方下载的代码都一样的。
代码如下:
#i nclude "includes.h"
#i nclude "24cxx.h"
#define ADDR_24CXX       0xA0
#define SCLH        GPIOB->BSRR = GPIO_Pin_6
#define SCLL        GPIOB->BRR  = GPIO_Pin_6 
   
#define SDAH        GPIOB->BSRR = GPIO_Pin_7
#define SDAL        GPIOB->BRR  = GPIO_Pin_7
#define SCLread     GPIOB->IDR  & GPIO_Pin_6
#define SDAread     GPIOB->IDR  & GPIO_Pin_7
static void I2C_GPIO_Config(void)
{
  GPIO_InitTypeDef  GPIO_InitStructure; 
  
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
  
  // Configure I2C1 pins: SCL and SDA 
  GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_6;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;  
  GPIO_Init(GPIOB, &GPIO_InitStructure);
  GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_7;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;
  GPIO_Init(GPIOB, &GPIO_InitStructure);  
}
void I2C_delay(void)

   u8 i=50; //这里可以优化速度 ,经测试最低到5还能写入
   while(i) 
  
     i--; 
  
}
bool I2C_Start(void)
{
 SDAH;
 SCLH;
 I2C_delay();
 if(!SDAread)return FALSE; //SDA线为低电平则总线忙,退出
 SDAL;
 I2C_delay();
 if(SDAread) return FALSE; //SDA线为高电平则总线出错,退出
 SDAL;
 I2C_delay();
 return TRUE;
}
void I2C_Stop(void)
{
 SCLL;
 I2C_delay();
 SDAL;
 I2C_delay();
 SCLH;
 I2C_delay();
 SDAH;
 I2C_delay();
}
void I2C_Ack(void)

 SCLL;
 I2C_delay();
 SDAL;
 I2C_delay();
 SCLH;
 I2C_delay();
 SCLL;
 I2C_delay();
}
void I2C_NoAck(void)

 SCLL;
 I2C_delay();
 SDAH;
 I2C_delay();
 SCLH;
 I2C_delay();
 SCLL;
 I2C_delay();
}
bool I2C_WaitAck(void)  //返回为:=1有ACK,=0无ACK
{
 SCLL;
 I2C_delay();
 SDAH;   
 I2C_delay();
 SCLH;
 I2C_delay();
 if(SDAread)
 {
         SCLL;
         return FALSE;
 }
 SCLL;
 return TRUE;
}
void I2C_SendByte(u8 SendByte) //数据从高位到低位//
{
    u8 i=8;
    while(i--)
    {
       SCLL;
       I2C_delay();
       if(SendByte&0x80)
         SDAH;  
       else 
       SDAL;   
       SendByte<<=1;
       I2C_delay();
 SCLH;
       I2C_delay();
    }
    SCLL;
}
u8 I2C_ReceiveByte(void) //数据从高位到低位//

    u8 i=8;
    u8 ReceiveByte=0;
   SDAH;    
    while(i--)
    {
      ReceiveByte<<=1;      
      SCLL;
      I2C_delay();
      SCLH;
      I2C_delay(); 
      if(SDAread)
      {
       ReceiveByte|=0x01;
      }
    }
    SCLL;
    return ReceiveByte;
}
//写入1字节数据       待写入数据    待写入地址       器件类型(24c16或SD2403)
bool I2C_WriteByte(u8 SendByte, u16 WriteAddress, u8 DeviceAddress)
 
    u32 j;
    if(!I2C_Start())return FALSE;
    //I2C_SendByte(((WriteAddress & 0x0700) >>7) | DeviceAddress & 0xFFFE);//设置高起始地址+器件地址 
    I2C_SendByte( DeviceAddress & 0xFE);//写器件地址 
    if(!I2C_WaitAck()){I2C_Stop(); return FALSE;}
    I2C_SendByte((u8)((WriteAddress>>8) & 0xFF));   //设置高起始地址      
    I2C_WaitAck(); 
    I2C_SendByte((u8)((WriteAddress) & 0xFF));   //设置低起始地址      
    I2C_WaitAck(); 
    I2C_SendByte(SendByte);          //写数据
    I2C_WaitAck();   
    I2C_Stop(); 
 //注意:因为这里要等待EEPROM写完,可以采用查询或延时方式(10ms)
    for(j=0;j<1500;j++)
      I2C_delay();
    return TRUE;
}
//读出1字节数据         存放读出数据  待读出长度      待读出地址       器件类型(24c16或SD2403) 
u8 I2C_ReadByte( u16 ReadAddress,  u8 DeviceAddress)
 
    u8 temp;
    if(!I2C_Start())return FALSE;
    //I2C_SendByte(((ReadAddress & 0x0700) >>7) | DeviceAddress & 0xFFFE);//设置高起始地址+器件地址 
    I2C_SendByte((DeviceAddress & 0xFE));//写器件地址 
    if(!I2C_WaitAck()){I2C_Stop(); return FALSE;}
    I2C_SendByte((u8)((ReadAddress>>8) & 0xFF));   //设置高起始地址  
    I2C_WaitAck();
    I2C_SendByte((u8)((ReadAddress) & 0xFF));   //设置低起始地址      
    I2C_WaitAck();
    I2C_Start();
    I2C_SendByte((DeviceAddress & 0xFE)|0x01);    //读器件地址
    I2C_WaitAck();
   
    //*pDat = I2C_ReceiveByte();
    temp = I2C_ReceiveByte();
    
    I2C_NoAck();
     
    I2C_Stop();
    return temp;
}
void I2C_24CXX_Init(void)
{
  I2C_GPIO_Config();
}
void I2C_24CXX_Write(u16 nAddr, u8* pDat, u16 nLen)
{
  u16 i,j;
  for(i=0;i
  {
    I2C_WriteByte(*(pDat+i), nAddr+i, ADDR_24CXX);
  }
}
void I2C_24CXX_Read(u16 nAddr, u8* pDat, u16 nLen)
{
    u16 i;
    for(i=0;i
      *(pDat+i)=I2C_ReadByte( nAddr+i, ADDR_24CXX);
}
可是,反复试了多次,貌似很不稳定,有时正确,有时错误。最后添加了红字处的的延时,读写完全正常。用示波器观察,延时大概7.5ms。 
关键字:STM23  模拟I2C读写 引用地址:STM23模拟I2C读写

上一篇:解决STM32调试,无法进入main函数的问题
下一篇:stm32f4 Vcap

推荐阅读最新更新时间:2024-03-16 14:36

STM23模拟I2C读写
都说STM32的I2C有问题,不好用。我之前,在论坛上看到了Mcuplayer分享的一段代码,拿来测试了一下,读写正常,心想还挺不错。 但是等到真正做程序时,发现总是在while()循环处等待。无奈,只好用软件模拟I2C。 从网上找了一段程序,发现好多地方下载的代码都一样的。 代码如下: #i nclude includes.h #i nclude 24cxx.h #define ADDR_24CXX 0xA0 #define SCLH GPIOB- BSRR = GPIO_Pin_6 #define SCLL GPIOB- BRR = GPIO_Pin_6 #define S
[单片机]
实用的单片机模拟I2C总线控制EEPROM读写程序
之前写的EEPROM程序虽然能够软仿成功,但烧到单片机里的时候却不能用,无疑是时序的问题,今天修正了时序,总算硬仿成功了。对照上次的程序可以发现就是添加了头函数: intrins.h ,这样就可以通过 _nop_() 指令较为准确的控制时序。 上次那个问题依然没有解决:就是接收缓冲区的数据是从readbuf 开始的,以这个程序为例: readbuf 中存放0x96 readbuf 中存放0x84 readbuf 中存放0xd5 readbuf 中存放0x63 readbuf 中存放0x7c readbuf 中存放0x8c 其实我是想把收到的数据存放在readbuf ~ 中的,我也不知道为什么结果会这样,有知道的麻烦指教
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved