STM32通信:IIC (二)

有几个参考资料写得挺好的
一个是NXP出的I2C规范和用户手册,直接搜索UM10204 pdf即可
还有一个是ADI出的技术文章
https://www.analog.com/en/technical-articles/i2c-primer-what-is-i2c-part-1.html
https://www.analog.com/en/technical-articles/i2c-timing-definition-and-specification-guide-part-2.html

一、硬件部分

I2C使用两根信号线,一根SDA用于传输数据,一根SCL用于产生时钟
一条总线上可以有一个主设备和多个从设备,时钟由主设备控制,因此软件模拟I2C主设备较为容易
SCL和SDA在实际电路中需要配置成开漏/开集输出并使用上拉电阻,从而支持线与逻辑

STM32作为主设备时可配置开漏输出并使用GPIO自带的上拉功能。如果使用推挽输出,则可能会在与一些设备通信时产生异常,同时这种操作也不符合I2C规范
下图可以看到推挽输出具有输出高低电平的能力,同时切换速度较快;开漏输出只能输出低电平,其高电平靠上拉电阻实现,而它的上升时间也更长

二、基本时序

先是几个比较明显的特点:
1.正常传输数据时,当SCL线处于低电平期时,SDA线上的电平允许变动
2.正常传输数据时,当SCL线处于高电平期时,SDA线上的电平不变
3.如果在SCL线的高电平期,SDA线由高电平向低电平跳变,则表示开始传输数据(START信号)
4.如果在SCL线的高电平期,SDA线由低电平向高电平跳变,则表示停止传输数据(STOP信号)

然后看一下具体时序要求

这里以普通I2C模式举例(SCL时钟频率为100kHz),其它模式可以查阅上表

1.START信号

在单片机实现时不需要太关注下降时间,但需要关注在SCL保持高电平时SDA下降沿之前的setup时间和hold时间。最小值分别为4.7us和4.0us

2.STOP信号

接着是STOP信号,可以看到STOP信号只有一个setup时间要求,最小值为4.0us
然而,在STOP信号和START信号之间会有一个buffer时间用于让其它设备检测总线状态,buffer时间最小值为4.7us,因此SDA上升沿之后至少要跟4.7us的延迟

3.数据位

图中可以看到在SDA稳定到SCL变为高电平之间有setup时间要求,同时在SDA跳变到下一个SCL上升沿之间也有setup时间要求,最小值均为250ns
需要注意的是,在SDA稳定且SCL处于高电平时会有一个hold时间要求,表格中给出的是至少300ns,但同时还要考虑到SCL的低电平时间和高电平时间也有最小值要求,分别是4.7us和4.0us

三、读写时序

1.发送一个字节

MSB在前,LSB在后,每次只能发送长度为8的字节,然后接收方会返回NACK/ACK作为回应
NACK(高电平)表示数据错误或者无响应(被外部上拉电阻拉高)
ACK(低电平)表示成功接收

2.读单个字节

START信号 -> 发送7位数据地址+1位写标识(HIGH) -> 从设备返回ACK -> 发送寄存器地址/命令 -> 从设备返回ACK -> STOP信号(部分设备不需要在这里发送STOP信号) -> START信号 -> 发送7位数据地址+1位读标识(LOW) -> 从设备返回ACK -> 从设备返回1个字节 -> 主设备发送ACK -> STOP信号
如果按照7位设备地址来计算此图对应的设备地址是0x29,将地址左移1位,或0得到0x52(写),或1得到0x53(读)
图中Ac是从设备发送的ACK,Am是主设备发送的ACK

3.写单个字节

START信号 -> 发送7位数据地址+1位写标识(HIGH) -> 从设备返回ACK -> 发送寄存器地址/命令 -> 从设备返回ACK -> 发送写入的字节 -> 从设备返回ACK -> STOP信号

4.读多个字节

如图

5.写多个字节

如图

四、代码

使用HAL库编写,稍作修改也可移植到使用库函数开发的项目上

Delay_ticks()函数:

void Delay_ticks(uint32_t ticks)
{
  uint32_t told, tnow, tcnt = 0;
  uint32_t reload = SysTick->LOAD;
  told = SysTick->VAL;
  while (1)
  {
    tnow = SysTick->VAL;
    if (tnow != told)
    {
      if (tnow < told)
        tcnt += told - tnow;
      else
        tcnt += reload - tnow + told;
      told = tnow;
      if (tcnt >= ticks)
        break;
    }
  }
}

头文件:
softi2c1.h

#ifndef _SOFTI2C1_H
#define _SOFTI2C1_H

#include "main.h"
#include "DELAY/delay.h"
#include "softi2c_common.h"

// ****** configuration start ******
#define SOFTI2C1_SCL_CLKEN() __HAL_RCC_GPIOB_CLK_ENABLE()
#define SOFTI2C1_SDA_CLKEN() __HAL_RCC_GPIOB_CLK_ENABLE()

#define SOFTI2C1_SCL_GPIO GPIOB
#define SOFTI2C1_SDA_GPIO GPIOB

#define SOFTI2C1_SCL_PINID 8
#define SOFTI2C1_SDA_PINID 9

#define SOFTI2C1_RETRYTIMES 3
// ****** configuration end ******

#define SOFTI2C1_SCL_PIN ((uint16_t)1u << SOFTI2C1_SCL_PINID)
#define SOFTI2C1_SDA_PIN ((uint16_t)1u << SOFTI2C1_SDA_PINID)

#define SOFTI2C1_SCL(__PINSTATE__) (SOFTI2C1_SCL_GPIO->BSRR = (uint32_t)SOFTI2C1_SCL_PIN << ((__PINSTATE__) ? (0u) : (16u)) )
#define SOFTI2C1_SDA(__PINSTATE__) (SOFTI2C1_SDA_GPIO->BSRR = (uint32_t)SOFTI2C1_SDA_PIN << ((__PINSTATE__) ? (0u) : (16u)) )
#define SOFTI2C1_READSDA() ((SOFTI2C1_SDA_GPIO->IDR >> SOFTI2C1_SDA_PINID) & 1u)

#define SOFTI2C1_SDA_IN() {SOFTI2C1_SDA_GPIO->MODER &= ~(3 << (SOFTI2C1_SDA_PINID * 2u)); SOFTI2C1_SDA_GPIO->MODER |= 0 <<  (SOFTI2C1_SDA_PINID * 2u);}
#define SOFTI2C1_SDA_OUT() {SOFTI2C1_SDA_GPIO->MODER &= ~(3 << (SOFTI2C1_SDA_PINID * 2u)); SOFTI2C1_SDA_GPIO->MODER |= 1    << (SOFTI2C1_SDA_PINID * 2u);}

// For most of the cases
void SoftI2C1_Init(uint32_t speed);
uint8_t SoftI2C1_SendAddr(uint16_t addr, uint8_t addrLen, uint8_t RorW);
uint8_t SoftI2C1_Read(uint16_t deviceAddr, uint8_t deviceAddrLen, uint8_t memAddr, uint8_t *dataBuf, uint32_t dataSize);
uint8_t SoftI2C1_Write(uint16_t deviceAddr, uint8_t deviceAddrLen, uint8_t memAddr, uint8_t *dataBuf, uint32_t dataSize);

// Low Layer
void SoftI2C1_Start(void);
void SoftI2C1_Stop(void);
void SoftI2C1_SendACK(uint8_t ACK);
uint8_t SoftI2C1_WaitACK(void);
void SoftI2C1_SendByte(uint8_t byte);
uint8_t SoftI2C1_ReadByte(void);
uint8_t SoftI2C1_SendByte_ACK(uint8_t byte, uint8_t handleACK);
uint8_t SoftI2C1_ReadByte_ACK(uint8_t ACK);

#endif

softi2c_common.h

#ifndef _SOFTI2C_COMMON_H
#define _SOFTI2C_COMMON_H

#define SI2C_ACK 0
#define SI2C_NACK 1

#define SI2C_WRITE 0
#define SI2C_READ 1

#define SI2C_ADDR_7b 0
#define SI2C_ADDR_10b 1

#endif

源码:
softi2c1.c

#include "softi2c1.h"

uint16_t SoftI2C1_delayTicks = 0;

void SoftI2C1_Init(uint32_t speed)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};

  SOFTI2C1_SCL_CLKEN();
  SOFTI2C1_SDA_CLKEN();

  SOFTI2C1_SCL(1);
  SOFTI2C1_SDA(1);
  GPIO_InitStruct.Pin = SOFTI2C1_SCL_PIN;
  GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
  HAL_GPIO_Init(SOFTI2C1_SCL_GPIO, &GPIO_InitStruct);

  GPIO_InitStruct.Pin = SOFTI2C1_SDA_PIN;
  HAL_GPIO_Init(SOFTI2C1_SDA_GPIO, &GPIO_InitStruct);

  // 这里用系统主频来替换Delay_GetSYSFreq(),例如 72000000,168000000,480000000
  SoftI2C1_delayTicks = Delay_GetSYSFreq() / speed / 2;
}

uint8_t SoftI2C1_SendAddr(uint16_t addr, uint8_t addrLen, uint8_t RorW)
{
  uint8_t buf;
  if (addrLen == SI2C_ADDR_7b)
  {
    buf = ((addr & 0x007F) << 1u) | RorW;
    return SoftI2C1_SendByte_ACK(buf, SI2C_ACK);
  }
  else
  {
    // 111100XX with first 2 bits
    buf = ((addr & 0x0003) << 1u) | RorW;
    buf |= 0x78;
    if (!SoftI2C1_SendByte_ACK(buf, SI2C_ACK))
      return 0;

    // the last 8 bits
    buf = addr & 0x00FF;
    return SoftI2C1_SendByte_ACK(buf, SI2C_ACK);
  }
}

uint8_t SoftI2C1_Read(uint16_t deviceAddr, uint8_t deviceAddrLen, uint8_t memAddr, uint8_t *dataBuf, uint32_t dataSize)
{
  uint32_t i;

  SoftI2C1_Start();
  if (!SoftI2C1_SendAddr(deviceAddr, deviceAddrLen, SI2C_WRITE))
    return 0;
  if (!SoftI2C1_SendByte_ACK(memAddr, SI2C_ACK))
    return 0;
// SoftI2C1_Stop(); // A STOP signal is required on some devices.

  SoftI2C1_Start();
  if (!SoftI2C1_SendAddr(deviceAddr, deviceAddrLen, SI2C_READ))
    return 0;
  for (i = 0; i < dataSize; i++)
    *(dataBuf + i) = SoftI2C1_ReadByte_ACK(SI2C_ACK);
  SoftI2C1_Stop();

  return 1;
}
uint8_t SoftI2C1_Write(uint16_t deviceAddr, uint8_t deviceAddrLen, uint8_t memAddr, uint8_t *dataBuf, uint32_t dataSize)
{
  uint32_t i;

  SoftI2C1_Start();
  if (!SoftI2C1_SendAddr(deviceAddr, deviceAddrLen, SI2C_WRITE))
    return 0;
  if (!SoftI2C1_SendByte_ACK(memAddr, SI2C_ACK))
    return 0;
  for (i = 0; i < dataSize; i++)
    if (!SoftI2C1_SendByte_ACK(*(dataBuf + i), SI2C_ACK))
      return 0;
  SoftI2C1_Stop();

  return 1;
}

void SoftI2C1_Start(void)
{
  SOFTI2C1_SDA_OUT();
  SOFTI2C1_SCL(1);
  SOFTI2C1_SDA(1);
  Delay_ticks(SoftI2C1_delayTicks); // setup time
  SOFTI2C1_SDA(0);                  // START: when CLK is high,DATA change form HIGH to LOW
  Delay_ticks(SoftI2C1_delayTicks); // hold time

  SOFTI2C1_SCL(0); // cannot be read
  Delay_ticks(SoftI2C1_delayTicks);
}

void SoftI2C1_Stop(void)
{
  SOFTI2C1_SDA_OUT();
  SOFTI2C1_SCL(1);
  SOFTI2C1_SDA(0);
  Delay_ticks(SoftI2C1_delayTicks);     // setup time
  SOFTI2C1_SDA(1);                      // STOP: when CLK is high,DATA change form LOW to HIGH
  Delay_ticks(SoftI2C1_delayTicks * 2); // hold time(not necessary in most of the situations) and buff time(necessary)

  // when the transmition is stopped, the SCL should be high
}

void SoftI2C1_SendACK(uint8_t ACK) // 0:ACK 1:NACK
{
  SOFTI2C1_SCL(0); // change start
  SOFTI2C1_SDA_OUT();
  SOFTI2C1_SDA(ACK);
  Delay_ticks(SoftI2C1_delayTicks / 8); // data setup time
  SOFTI2C1_SCL(1);                      // can be read
  Delay_ticks(SoftI2C1_delayTicks);     // hold
  SOFTI2C1_SCL(0);                      // cannot be read
  Delay_ticks(SoftI2C1_delayTicks);     // data setup time & SCL_LOW & SMBus requirement
}

uint8_t SoftI2C1_WaitACK(void) // 0:ACK 1:NACK/No response
{
  uint16_t waitTime = 0;
  uint8_t result = 0;
  SOFTI2C1_SDA_IN();
  SOFTI2C1_SCL(1);
  Delay_ticks(SoftI2C1_delayTicks / 8); // data setup time
  result = SOFTI2C1_READSDA();
  Delay_ticks(SoftI2C1_delayTicks);
  SOFTI2C1_SCL(0);
  Delay_ticks(SoftI2C1_delayTicks); // data setup time & SCL_LOW & SMBus requirement
  if (result == SI2C_NACK)
    SoftI2C1_Stop();
  return result;
}

// 8 SCL clock, just send a byte
void SoftI2C1_SendByte(uint8_t byte) // barely send a byte
{
  int8_t i;
  SOFTI2C1_SDA_OUT();
  SOFTI2C1_SCL(0);
  for (i = 7; i >= 0; i--)
  {
    SOFTI2C1_SDA((byte >> i) & 1u);
    Delay_ticks(SoftI2C1_delayTicks / 8); // data setup time
    SOFTI2C1_SCL(1);
    Delay_ticks(SoftI2C1_delayTicks); // data hold time
    SOFTI2C1_SCL(0);
    Delay_ticks(SoftI2C1_delayTicks); // data setup time & SCL_LOW & SMBus requirement
  }
}

// 8 SCL clock, just read a byte
uint8_t SoftI2C1_ReadByte(void) // barely read a byte
{
  uint8_t i, result = 0;
  SOFTI2C1_SDA_IN();
  for (i = 0; i < 8; i++)
  {
    SOFTI2C1_SCL(0);
    Delay_ticks(SoftI2C1_delayTicks); // data setup time & SCL_LOW & SMBus requirement
    SOFTI2C1_SCL(1);
    result <<= 1;
    Delay_ticks(SoftI2C1_delayTicks / 8); // data setup time
    result |= SOFTI2C1_READSDA();
    Delay_ticks(SoftI2C1_delayTicks); // data hold time
  }
  return result;
}

// 9 SCL clock, treat the 9th bit as ACK/NACK
// If SI2C_NACK is given, it will always return true(ignore ACK/NACK)
// If SI2C_ACK is given, it will retry SOFTI2C1_RETRYTIMES if no ACK is detected,
// and it will return whether the ACK is detected finally.
uint8_t SoftI2C1_SendByte_ACK(uint8_t byte, uint8_t handleACK) // handle ACK and retry
{
  uint8_t i;
  if (handleACK == SI2C_NACK)
  {
    SoftI2C1_SendByte(byte);
    SOFTI2C1_SDA_IN();
    SOFTI2C1_SCL(1);                      
    Delay_ticks(SoftI2C1_delayTicks / 8); // data setup time
    SOFTI2C1_SCL(1);                      // can be read
    Delay_ticks(SoftI2C1_delayTicks);     // hold
    SOFTI2C1_SCL(0);                      // cannot be read
    Delay_ticks(SoftI2C1_delayTicks);     // data setup time & SCL_LOW & SMBus requirement
    return 1;
  }
  else
  {
    for (i = 0; i < SOFTI2C1_RETRYTIMES; i++)
    {
      SoftI2C1_SendByte(byte);
      if (SoftI2C1_WaitACK() == SI2C_ACK)
        break;
    }
    return (i < SOFTI2C1_RETRYTIMES);
  }
}

// 9 SCL clock, send ACK/NACK after data received
uint8_t SoftI2C1_ReadByte_ACK(uint8_t ACK)
{
  uint8_t result;
  result = SoftI2C1_ReadByte();
  SoftI2C1_SendACK(ACK);
  return result;
}

2021.01.30

posted @ 2021-01-30 22:49  wh201906  阅读(523)  评论(0编辑  收藏  举报