STM32 IIC详解

目录

1、IIC定义

2、IIC协议规范

2.1 SDA和SCL信号

2.2 数据有效性​

2.3 开始和结束信号​

2.4 字节格式

2.5 从机地址和读写位​

3、计算IIC的频率

4、PCF8536

4.1 Acknowledge

4.2 Addressing

4.3 读写时序


1、IIC定义

IIC 即Inter-Integrated Circuit(集成电路总线),这种总线类型是由飞利浦半导体公司(后被NXP收购)在八十年代初设计出来的一种简单、双向、二线制、同步串行总线,主要是用来连接整体电路(ICS) ,IIC是一种多向控制总线,也就是说多个芯片可以连接到同一总线结构下,同时每个芯片都可以作为实时数据传输的控制源。多主多从的通讯协议。

下文将结合NXP官方的IIC手册讲解IIC协议。下载链接见文末。

I2C串行总线一般有两根信号线,一根是双向的数据线SDA,另一根是时钟线SCL。所有接到I2C总线设备上的串行数据SDA都接到总线的SDA上,各设备的时钟线SCL接到总线的SCL上。速率最高400Kbit/s。

在1998年的修订中增加了高速模式,速率高达3.4Mbit/s。(这里不讲,只说快速模式)。

2、IIC协议规范

2.1 SDA和SCL信号

连接到总线的器件输出级必须是漏极开路或集电极开路才能执行线与的功能,当总线空

闲时这两条线路都是高电平。SDA 和SCL 都是双向线路都通过一个电流源或上拉电阻连接到正的电源电压。一般情况下我们都采用上拉电阻的方式

2.2 数据有效性

在SCL高电平的时候采样,也就是有效。低电平的时候切换数据

2.3 开始和结束信号

起始条件:SCL线是高电平时,SDA线从高电平向低电平切换。

停止条件:SCL线是高电平时,SDA线从低电平向高电平切换。

动画展示启动信号

代码实现

void I2C_Start(void)
{
  //IO输出
  SDA_OUT(); 
  SCL_OUT(); 
  I2C_DELAY();
  //IO置高
  SDA_SET();  
  SCL_SET(); 
  //延时
  I2C_DELAY();  
  //为低
  SDA_CLR();
  I2C_DELAY();
  I2C_DELAY();
  SCL_CLR();
}

结束信号时类似的方式(不是动图)

代码实现

void I2C_Stop(void)
{
  //IO输出
  SDA_OUT(); 
  SCL_OUT();
  //IO置0
  SDA_CLR();  
  SCL_CLR(); 
  I2C_DELAY();
  SCL_SET();
  //延时
  I2C_DELAY();  
  I2C_DELAY();
  I2C_DELAY();
  //SDA置1
  SDA_SET();
  I2C_DELAY();
  I2C_DELAY();
}

2.4 字节格式

SDA数据线上的每个字节必须是8位,每次传输的字节数量没有限制。每个字节后必须跟一个响应位(ACK)。首先传输的数据是最高位(MSB),SDA上的数据必须在SCL高电平周期时保持稳定,数据的高低电平翻转变化发生在SCL低电平时期。

每一个字节后面跟着一个ACK,有ACK就可以继续写或读。NACK,就停止

ACK:主机释放总线,传输完字节最后1位后的SCL的高电处,从机拉低电平。

NACK:主机释放总线,传输完字节最后1位后的SCL的高电处,从机无响应,总线为高电平。

动画描述写入的过程

代码实现

uint8_t I2C_Send_byte(uint8_t data)
{
  uint8_t k;
  //发送8bit数据
  for(k=0;k<8;k++){
    
    I2C_DELAY();
    if(data&0x80){
      SDA_SET();
    }
    else{
      SDA_CLR();
    }
    data=data<<1;
    I2C_DELAY();
    SCL_SET();
    I2C_DELAY();
    I2C_DELAY();
    SCL_CLR();
  }
  //延时读取ACK响应
  I2C_DELAY();
  SDA_SET();
  //置为输入线
  SDA_IN();
  I2C_DELAY();
  SCL_SET();   
  I2C_DELAY(); //这里出现了问题,延时变的无限大
  //读数据
  k=SDA_READ();
  if(k){ NACK响应
    return 0;
  }
  I2C_DELAY();
  SCL_CLR();
  I2C_DELAY();
  SDA_OUT();
  if(k){ NACK响应
    return 0;
  }
  return 1;
}

uint8_t I2C_Receive_byte(uint8_t flg)
{
  uint8_t k,data;
  //接收8bit数据
  //置为输入线
  
  SDA_IN();
  data=0;
  for(k=0;k<8;k++){
    I2C_DELAY();
    SCL_SET();
    I2C_DELAY();
    //读数据
    data=data |SDA_READ();
    data=data<<1;
    I2C_DELAY();
    SCL_CLR();
    I2C_DELAY(); 
  }
  data=data>>1; //往回移动1次
  //返回ACK响应
  //置为输出线
  SDA_OUT();
  if(flg){
    SDA_SET(); //输出1-NACK
  }else{
    SDA_CLR();//输出0-ACK
  }
  I2C_DELAY();
  SCL_SET();
  I2C_DELAY();
  I2C_DELAY();
  SCL_CLR();
  I2C_DELAY();
  SDA_OUT();
  //返回读取的数据
  return (uint8_t)data;
}

2.5 从机地址和读写位

开始信号—>地址—>读写位—>ACK—>数据—>ACK.............—>停止位

这里只说7位地址,前7位为地址,最后一位为读写位,1表示读操作,0表示写操作

主机发给从机数据,也就是写,没有数据转向时

主机立即读从机数据,从第一个字节

(Combined)综合数据格式

3、计算IIC的频率

通信频率由主机掌控,也就是代码中的延时函数决定的

从上面,我们得知最高速为400Kbit/s。我们设计300Kbit/s。

速率300Kbit/s,对应周期1/300ms=10/3us≈3us,4分频就是3/4us。

我们使用的延时是,1/120MHZ*3*30 =3/4us

也就是频率是300Kbit/s

和SPI类似,时钟下降沿时,数据转换,时钟上升沿时,采样数据。也就是时钟高电平数据有效。

/*120Mhz时钟时,当ulCount为1时,函数耗时3个时钟,延时=3*1/120us=1/40us*/
/*
SystemCoreClock=120000000

us级延时,延时n微秒
SysCtlDelay(n*(SystemCoreClock/3000000));

ms级延时,延时n毫秒
SysCtlDelay(n*(SystemCoreClock/3000));

m级延时,延时n秒
SysCtlDelay(n*(SystemCoreClock/3));
*/

#if defined   (__CC_ARM) /*!< ARM Compiler */
__asm void
SysCtlDelay(unsigned long ulCount)
{
    subs    r0, #1;
    bne     SysCtlDelay;
    bx      lr;
}
#elif defined ( __ICCARM__ ) /*!< IAR Compiler */
void
SysCtlDelay(unsigned long ulCount)
{
    __asm("    subs    r0, #1\n"
       "    bne.n   SysCtlDelay\n"
       "    bx      lr");
}

#elif defined (__GNUC__) /*!< GNU Compiler */
void __attribute__((naked))
SysCtlDelay(unsigned long ulCount)
{
    __asm("    subs    r0, #1\n"
       "    bne     SysCtlDelay\n"
       "    bx      lr");
}

#elif defined  (__TASKING__) /*!< TASKING Compiler */                           
/*无*/
#endif /* __CC_ARM */


/*
 * @brief  SysCtlDelay
 * @param  ulCount 延时值,必须大于0
 * @retval None
 */
void SysCtlDelay_IIC(unsigned long ulCount)
{
	SysCtlDelay(ulCount);
}


/定义时钟频率,300KHz
#define I2C_DELAY()  SysCtlDelay_IIC(30)

4、PCF8536

4.1 Acknowledge

这个地方能看到关于2.4节关于ACK和NACk的说明

4.2 Addressing

这里直接给出读地址和写地址,也就是最后一位的区别

4.3 读写时序

其实就是按照IIC协议的

读指定器件的指定寄存器

主机设置完寄存器地址之后,再去读写

注意:读取多个字节,最后一个字节的回包应该是NACK

主机立即从机第一个字节读取

注意:读取多个字节,最后一个字节的回包应该是NACK

开源地址:

https://github.com/strongercjd/STM32F207VCT6

点击查看本文所在的专辑,STM32F207教程

资料下载:

百度网盘-链接不存在   提取码:cwsx

如果链接失效,请关注微信公众号,找到下载专区->博客附件,编号0012的下载资源,免费获得。

12

  • 128
    点赞
  • 806
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 28
    评论
STM32的IIC通信是通过SDA和SCL两个引脚进行的。在进行读取数据时,可以使用IIC_Read_Data函数来读取数据。该函数通过拉高SCK电平,并在每个时钟脉冲中读取一个bit的数据,最后返回读取到的数据。\[1\] 要检测IIC设备是否存在,可以使用ee_CheckDevice函数。该函数发送设备地址,并读取设备是否返回应答信号来判断设备是否存在。返回值为0表示设备存在,返回1表示未探测到IIC设备。\[2\] 在读取数据时,可以使用i2c_ReadByte函数。该函数通过拉高SCL引脚,并在每个时钟脉冲中读取一个bit的数据,最后将数据取出,完成一次8bit数据的读取。\[3\] #### 引用[.reference_title] - *1* [STM32----IIC详解](https://blog.csdn.net/qq_45604814/article/details/116099878)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [STM32之IIC详细解析](https://blog.csdn.net/qq_43940175/article/details/123437156)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 28
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

strongercjd

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值