《Linux驱动:I2C协议分析》

一,前言

I2C总线支持设备之间的短距离通信,用于处理器和一些外围设备之间数据传输,它只需要两根信号线来就能完成数据传输。在分析I2C总线驱动之前,需要先了解下i2c协议。

二,硬件电路

硬件上的连接一般如下图所示,主控mcu上两根信号线SDA和SCL,多个设备可以同时接入,SDA和SCL必须接上拉电阻,阻值一般为4.7K。从图上可看出,I2C为主从模式,发出SCL的为主,其他为从。两根信号线,一根时钟线、一根数据线决定了它是半双工的通信模式,同一时间只能单向传输数据。
在这里插入图片描述

三,数据传输

I2C协议把传输的消息分为两种类型的帧:一个地址帧和一个或多个数据帧。地址帧表示了主控将和哪个从设备进行通信。数据传输过程中,SDA线上的数据必须在SCL高电平期间保持稳定,在SCL低电平期间才能发生高低电平的转换。
在这里插入图片描述

START:起始信号,当SCL为高电平时,SDA从高电平到低电平转换。
ADDRESS+R/W+ACK:地址帧,包含将进行通信的从设备的地址,1~7位为从设备地址,第8位为读写位(高电平-读,低电平-写),ACK为从机的响应(此时SDA交由从机控制,从机输出低电平表示响应)。
DATA+ACK:将要写给从机的数据,或者将要从从机读取的数据。
STOP:停止信号,当SCL为高电平时,SDA从低电平到高电平转换。

3.1 写操作一般流程

  • 主设备发起一个起始信号
  • 开始传输从设备地址,高七位表示从设备地址,最后一位为0,表示写。
  • 从地址传输完成后,将SDA设为输入状态,读取SDA电平状态,如果为0则表示被从设备拉低,即为一个ACK信号,表示找到了从设备。如果不为0,则表示从设备未响应,主设备发起停止信号,终止传输。
  • 找到从设备后,即开始传输需要将数据写入到从设备内部位置的地址值,和上一步类似。
  • 传输完从设备内部地址后,便可传输需要写到内部地址的数据。判断需要写入的数据大小,以字节为单位传输。
  • 传输完成后,主机发起一个停止信号,终止本次写操作。
    在这里插入图片描述

3.2 读操作一般流程

读操作,需要先把要读取的从设备内部地址传输给从设备,再执行读操作。

  • 写入要读取的从设备内部地址。步骤见3.1。
  • 主设备发起一个起始信号
  • 开始传输从设备地址,高七位表示从设备地址,最后一位为1,表示读。
  • 从地址传输完成后,将SDA设为输入状态,读取SDA电平状态,如果为0则表示被从设备拉低,即为一个ACK信号,表示找到了从设备。如果不为0,则表示从设备未响应,主设备发起停止信号,终止传输。
  • 找到从设备后,即开始读取从设备内部地址处的数据。
  • 判断需要读取的数据大小,以字节为单位传输。
  • 传输完成后,主机发起一个停止信号,终止本次写操作。
    在这里插入图片描述

四,IO口模拟I2C

4.1 IO口初始化

void sys_i2c_init_board(void)
{
    //初始化IO口,SCL配置为输出高
    GPIO_InitTypeDef PB6={GPIO_Pin_6,GPIO_Speed_2MHz,GPIO_Mode_Out_OD};
    GPIO_Init(GPIOB, &PB6);
    //初始化IO口,SDA配置为输出高
    GPIO_InitTypeDef PB7={GPIO_Pin_7,GPIO_Speed_2MHz,GPIO_Mode_Out_OD};
    GPIO_Init(GPIOB,&PB7);
}

4.2 起始信号

//START: High -> Low on SDA while SCL is High
void soft_i2c_send_start(void)
{
    SET_SDA_OUT;
    I2C_DELAY();
    
    I2C_SDA(1);
    I2C_SCL(1);
    I2C_DELAY();
    
    I2C_SDA(0);  // scl高电平期间,SDA拉低
    I2C_DELAY();
    
    I2C_SCL(0);
    I2C_DELAY();
}

4.3 停止信号

//STOP: Low -> High on SDA while SCL is High
void soft_i2c_send_stop(void)
{
    I2C_SCL(0);
    SET_SDA_OUT;
    I2C_SDA(0);
    I2C_DELAY();
    
  
    I2C_SCL(1);
    I2C_DELAY();

    I2C_SDA(1);  // scl高电平期间,SDA拉高
    I2C_DELAY();
}

4.4 读一个字节(该I2C例程内部使用)

static u8 soft_i2c_read_byte(int ack)
{
  u8  data;
  int  j;
  
  SET_SDA_IN;
  
  data = 0;
  for(j = 0; j < 8; j++) 
  {
    I2C_DELAY();
    I2C_SCL(1);
    I2C_DELAY();
    data <<= 1;
    data |= I2C_SDA_VALUE_IN;
    I2C_DELAY();
    I2C_SCL(0);
    I2C_DELAY();
  }
  soft_i2c_send_ack(ack);
  
  return(data);
}

4.5 写一个字节(该I2C例程内部使用)

static int soft_i2c_write_byte(u8 data)
{
  int j;
  int nack;
  
  SET_SDA_OUT;
  for(j = 0; j < 8; j++) 
  {
    I2C_SDA(data & 0x80);
    I2C_DELAY();
    I2C_SCL(1);
    I2C_DELAY();
    I2C_DELAY();
    I2C_SCL(0);
    I2C_DELAY();
    
    data <<= 1;
  }

  /*
   * Look for an <ACK>(negative logic) and return it.
   */
  
  SET_SDA_IN;
  I2C_DELAY();
  I2C_SCL(1);
  I2C_DELAY();
  nack = I2C_SDA_VALUE_IN;
  I2C_DELAY();
  I2C_SCL(0);
  I2C_DELAY();
  
  SET_SDA_OUT;
  //I2C_SDA(1);
  return(nack);	/* not a nack is an ack */
}

4.6 I2C读数据(供外部调用)

int soft_i2c_read(u8 chipaddr, u32 internalAddr, u32 internalAddrLen, u8 *buffer, int len)
{
  int shift;

  /*
   * Do the addressing portion of a write cycle to set the
   * chip's address pointer.  If the address length is zero,
   * don't do the normal write cycle to set the address pointer,
   * there is no address pointer in this chip.
   */
  soft_i2c_send_start();
  if(internalAddrLen > 0) 
  {
    if(soft_i2c_write_byte(chipaddr << 1)) 	/* write cycle */
    {
      soft_i2c_send_stop();
      return(1);
    }
    shift = (internalAddrLen-1) * 8;
    while(internalAddrLen-- > 0) 
    {
      if(soft_i2c_write_byte(internalAddr >> shift)) 
      {
        //printf("i2c_read, internal address not <ACK>ed\n");
        return(1);
      }
      shift -= 8;
    }

  /* Some I2C chips need a stop/start sequence here,
   * other chips don't work with a full stop and need
   * only a start.  Default behaviour is to send the
   * stop/start sequence.
   */
#ifdef CONFIG_SOFT_I2C_READ_REPEATED_START
    soft_i2c_send_start();
#else
    soft_i2c_send_stop();
    soft_i2c_send_start();
#endif
  }
  /*
   * Send the chip address again, this time for a read cycle.
   * Then read the data.  On the last byte, we do a NACK instead
   * of an ACK(len == 0) to terminate the read.
   */
  soft_i2c_write_byte((chipaddr << 1) | 1);	/* read cycle */
  while(len-- > 0) 
  {
      *buffer++ = soft_i2c_read_byte(len == 0);
  }
  soft_i2c_send_stop();
  return(0);
}

4.7 I2C写入数据(供外部调用)

int soft_i2c_write(u8 chipaddr, u32 internalAddr, u32 internalAddrLen, u8 *buffer, int len)
{
    int shift, failures = 0;

    //printf("i2c_write: chip %02X addr %02X alen %d buffer %p len %d\n",chip, addr, alen, buffer, len);

    soft_i2c_send_start();
    if(soft_i2c_write_byte(chipaddr<<1)) 	/* write cycle */
    {	
        soft_i2c_send_stop();
        //printf("i2c_write, no chip responded %02X\n", chipaddr);
        return(1);
    }
    shift = (internalAddrLen-1) * 8;
    while(internalAddrLen-- > 0) 
    {
        if(soft_i2c_write_byte(internalAddr >> shift)) 
        {
            //printf("i2c_write, address not <ACK>ed\n");
            return(1);
        }
        shift -= 8;
    }

    while(len-- > 0) 
    {
        if(soft_i2c_write_byte(*buffer++)) 
        {
            failures++;
        }
    }
    soft_i2c_send_stop();
    return(failures);
}

4.8 完整例程

链接:https://pan.baidu.com/s/1DdKy33_3XhbQKfYtprmP_w
提取码:1nnl

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Linux驱动开发中的I2C(Inter-Integrated Circuit)是一种串行通信协议,用于连接微控制器、传感器、存储器等外设。下面是关于Linux驱动开发中的I2C的一些重要概念和步骤: 1. I2C总线:I2C总线是一种多主从设备的串行通信总线,可以同时连接多个设备。在Linux中,I2C总线由I2C控制器和I2C适配器组成。 2. I2C控制器:I2C控制器是硬件模块,负责控制I2C总线的通信。在Linux中,每个I2C控制器都有一个对应的驱动程序。 3. I2C适配器:I2C适配器是与I2C总线连接的硬件接口,可以是主机上的硬件接口或者通过GPIO模拟的软件接口。在Linux中,每个I2C适配器都有一个对应的驱动程序。 4. I2C设备驱动:每个连接到I2C总线上的设备都需要一个对应的设备驱动程序来进行控制和通信。在Linux中,每个I2C设备都有一个对应的驱动程序。 在Linux驱动开发中,使用I2C的步骤如下: 1. 注册I2C适配器:首先需要在Linux内核中注册I2C适配器,以便系统能够识别和管理I2C总线上的设备。 2. 创建I2C设备驱动:为每个连接到I2C总线上的设备编写对应的设备驱动程序,包括设备的初始化、读写操作等。 3. 注册I2C设备驱动:将设备驱动程序注册到Linux内核中,以便系统能够正确地加载和使用该驱动程序。 4. 使用I2C设备:在应用程序中通过打开设备文件、调用相应的读写接口来使用I2C设备。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

编程界的小学生、

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

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

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

打赏作者

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

抵扣说明:

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

余额充值