I2C总线协议的软件模拟实现方法

I2C总线协议的软件模拟实现方法

在上一篇博客中已经讲过I2C总线通信协议,本文讲述I2C总线协议的软件模拟实现方法。

1. 简述

所谓的I2C总线协议的软件模拟实现方法,就是用软件控制GPIO的输入、输出和高低电平变化,来模拟I2C总线通讯过程中SCL、SDA的电平变化来实现的。

2. I2C总线的封装

每个处理器对应的GPIO操作都有差异,即使是同一款处理器,不同的人也会有不同的GPIO封装风格,就以我个人习惯用的GPIO方法为例来进行讲解。我习惯上将GPIO的组和位封装为一个结构体,这样使用方便,看起来也更直观。

typedef struct {
    unsigned char group;
    unsigned char bit;
} gpio_t;

将I2C总线中使用的SCL和SDA的GPIO进一步进行封装。

typedef struct {
    gpio_t scl;
    gpio_t sda;
} i2c_gpio_t;

将I2C总线软件模拟部分当做驱动程序中的一个模块来使用,定义一个结构体来封装I2C模块中的一些全局变量,如:GPIO、锁等等。本文中的锁只是为了保证I2C的一个操作步骤是原子的,所有锁的使用可以忽略,如果想要了解更过关于锁的使用方法,请关注另外一篇博客(还没来得及写,以后会补充)。

typedef struct {
    i2c_gpio_t gpio;
    spinlock_t lock;
    struct mutex i2c_mutex;
} i2c_info_t;

3. 软件模拟实现

3.1 I2C总线的初始化

1)先初始化I2C总线,具体要做的内容是,先把外部调用I2C模块时要使用的GPIO引脚,作为参数传递到I2C模块,用来进行一系列的操作。在这里将GPIO作为参数传递到I2C模块后,保存在全局变量的结构体中。

2)再初始化I2C总线的GPIO引脚,即将用来代替模拟I2C总线中SCL、SDA引脚的GPIO设置为输出,并输出高电平,因为两条线上都接有上拉电阻,I2C总线空闲时默认SCL、SDA都处于高电平,也就是空闲状态。

3)如果要使用锁机制,需要在这一步中将锁初始化。

// I2C模块初始化
int i2c_init(i2c_gpio_t *gpio)
{
    i2c_debug("i2c_init");

    // 初始化锁
    spin_lock_init(&i2c_info.lock);
    mutex_init(&i2c_info.i2c_mutex);

    // 初始化全局变量中I2C的GPIO
    i2c_info.gpio.scl = gpio->scl;
    i2c_info.gpio.sda = gpio->sda;

    i2c_gpio_init();

    return 0;
}
// I2C的GPIO初始化
static void i2c_gpio_init(void)
{
    i2c_debug("i2c_gpio_init");
    i2c_sda_init();
    i2c_scl_init();
}
// I2C的SCL初始化
static void i2c_scl_init(void)
{
    i2c_debug("scl init");
    SET_SCL_OUT;
    SET_SCL_HIGH;
}
// I2C的SDA初始化
static void i2c_sda_init(void)
{
    i2c_debug("sda init");
    SET_SDA_OUT;
    SET_SDA_HIGH;
}

3.2 I2C总线的起始位

I2C总线在开始通信时要先发送一个起始位标志,起始位是在SCL为高电平时,SDA由高电平变为低电平。

// I2C总线的起始位
int i2c_start(void)
{
    mutex_lock(&i2c_info.i2c_mutex);
    SET_SDA_OUT;
    udelay(I2C_DELAY);

    SET_SDA_HIGH;
    udelay(I2C_DELAY);

    SET_SCL_HIGH;
    udelay(I2C_DELAY);

    SET_SDA_LOW;
    udelay(I2C_DELAY);

    SET_SCL_LOW;
    udelay(I2C_DELAY);
    mutex_unlock(&i2c_info.i2c_mutex);

    return 0;
}

3.3 I2C总线的结束位

I2C总线在数据传输完成后,需要发送一个结束位,来结束I2C通讯,并释放I2C总线,结束位是在SCL为高电平时,SDA由低电平变为高电平

// I2C总线的结束位
int i2c_stop(void)
{
    mutex_lock(&i2c_info.i2c_mutex);
    SET_SDA_OUT;
    udelay(I2C_DELAY);

    SET_SCL_LOW;
    udelay(I2C_DELAY);

    SET_SDA_LOW;
    udelay(I2C_DELAY);

    SET_SCL_HIGH;
    udelay(I2C_DELAY);

    SET_SDA_HIGH;
    udelay(I2C_DELAY);
    mutex_unlock(&i2c_info.i2c_mutex);

    return 0;
}

3.4 I2C总线的应答

为了统一管理和使用方便,将I2C总线的等待应答、发送应答信号、发送非应答信号封装在一起进行管理。

1)I2C总线的等待应答

在I2C总线通讯时,主设备给从设备发送一个字节的数据后,要等待从设备的一个应答信号,这时候主设备处于等待应答状态,需要检测从设备的应答信号是否到来,如果从设备的应答信号到来,主设备就继续给从设备发送下一个字节的数据,或者发送停止位结束I2C通讯;如果在主设备等待超时后,从设备的应答信号时钟不到来,就说明I2C总线通讯中出现问题,主设备跳出等待,直接发送结束位,以结束I2C总线通讯。

// I2C总线的等待应答
static int i2c_wait_ack(void)
{
    int ack_times = 0;
    int ret = 0;

    mutex_lock(&i2c_info.i2c_mutex);
    SET_SDA_OUT;
    udelay(I2C_DELAY);

    SET_SDA_HIGH;
    udelay(I2C_DELAY);

    SET_SDA_IN;
    udelay(I2C_DELAY);

    SET_SCL_LOW;
    udelay(I2C_DELAY);

    SET_SCL_HIGH;
    udelay(I2C_DELAY);

    ack_times = 0;
    // 检测从设备应答信号
    while (GET_SDA_VAL) {
        ack_times++;
        // 判断等待是否超时
        if (ack_times == 10) {
            ret = 1;
            i2c_error("i2c ack error, no ack");
            break;
        }
    }

    SET_SCL_LOW;
    mutex_unlock(&i2c_info.i2c_mutex);

    return ret;
}

2)I2C总线的发送应答

在I2C总线通信的时候,主设备每次接收到从设备发送的一个字节数据后,要给从设备发送应答信号(ACK)以继续接收从设备的数据,或者给从设备发送非应答信号(NOACK)以结束接收从设备的数据。

应答信号(ACK)就是先拉低SDA线,并在SCL为高电平期间保持SDA线为低电平

// I2C总线发送应答信号
static int i2c_send_ack(void)
{
    i2c_debug("i2c_send_ack");

    mutex_lock(&i2c_info.i2c_mutex);
    SET_SDA_OUT;
    udelay(I2C_DELAY);

    SET_SCL_LOW;
    udelay(I2C_DELAY);

    SET_SDA_LOW;
    udelay(I2C_DELAY);

    SET_SCL_HIGH;
    udelay(I2C_DELAY);

    SET_SCL_LOW;
    udelay(I2C_DELAY);
    mutex_unlock(&i2c_info.i2c_mutex);

    return 0;
}

非应答信号(NOACK)就是不要拉低SDA线(此时SDA线为高电平),并在SCL为高电平期间保持SDA线为高电平。

// I2C总线发送非应答信号
static int i2c_send_noack(void)
{
    i2c_debug("i2c_send_noack");

    mutex_lock(&i2c_info.i2c_mutex);
    SET_SDA_OUT;
    udelay(I2C_DELAY);

    SET_SCL_LOW;
    udelay(I2C_DELAY);

    SET_SDA_HIGH;
    udelay(I2C_DELAY);

    SET_SCL_HIGH;
    udelay(I2C_DELAY);

    SET_SCL_LOW;
    udelay(I2C_DELAY);
    mutex_unlock(&i2c_info.i2c_mutex);

    return 0;
}

3.5 I2C总线的写操作

// I2C总线的写操作
int i2c_write_byte(u8 data)
{
    unsigned long flag = 0;
    u8 i = 0;

    local_irq_save(flag);
    preempt_disable();
    mutex_lock(&i2c_info.i2c_mutex);
    SET_SDA_OUT;
    udelay(I2C_DELAY);

    for (i = 0; i < 8; i++) {
        if (data & 0x80) {
            SET_SDA_HIGH;
        } else {
            SET_SDA_LOW;
        }
        udelay(I2C_DELAY);

        SET_SCL_HIGH;
        udelay(I2C_DELAY);

        SET_SCL_LOW;
        udelay(I2C_DELAY);

        data <<= 0x1;
    }
    mutex_unlock(&i2c_info.i2c_mutex);
    preempt_enable();
    local_irq_restore(flag);

    return 0;
}
int i2c_write_byte_with_ack(u8 data)
{
    i2c_write_byte(data);
    if (i2c_ack(I2C_WAIT_ACK)) {
        i2c_error("wait ack failed, no ack");
        i2c_stop();
        return -1;
    }

    return 0;
}

3.6 I2C总线的读操作

// I2C总线的读操作
int i2c_read_byte(u8 *data)
{
    unsigned long flag = 0;
    u8 ret = 0;
    u8 i = 0;

    local_irq_save(flag);
    preempt_disable();
    mutex_lock(&i2c_info.i2c_mutex);

    SET_SDA_IN;
    udelay(I2C_DELAY);

    for (i = 0; i < 8; i++) {
        SET_SCL_HIGH;
        udelay(I2C_DELAY);

        ret <<= 1;

        if (GET_SDA_VAL) {
            ret |= 0x01;
        }

        SET_SCL_LOW;
        udelay(I2C_DELAY);
    }

    mutex_unlock(&i2c_info.i2c_mutex);
    preempt_enable();
    local_irq_restore(flag);

    *data = ret;

    return 0;
}

原文地址:https://www.cnblogs.com/microxiami/p/8528459.html

时间: 2024-11-05 01:39:07

I2C总线协议的软件模拟实现方法的相关文章

I2C 总线协议分析

由于大规模集成电路技术的发展,在单个芯片集成 CPU 以及组成一个单独工作系统所必须的 ROM.RAM.I/O 端口.A/D.D/A 等外围电路和已经实现,这就是常说的单片机或微控制器.目前,世界上许多公司生产单片机,品种很多:包括各种字长的 CPU,各种容量和品种的 ROM.RAM,以及功能各异的 I/O 等等.但是,单片机品种规格有限,所以只能选用某种单片机再进行扩展.扩展的方法有两种:一种是并行总线,另一种是串行总线.由于串行总线连线少,结构简单,往往不用专用的母板和插座而直接用导线连接各

[I2C]I2C总线协议图解

转自:http://blog.csdn.net/w89436838/article/details/38660631 1  I2C总线物理拓扑结构      I2C 总线在物理连接上非常简单,分别由SDA(串行数据线)和SCL(串行时钟线)及上拉电阻组成.通信原理是通过对SCL和SDA线高低电平时序的控制,来产生I2C总线协议所需要的信号进行数据的传递.在总线空闲状态时,这两根线一般被上面所接的上拉电阻拉高,保持着高电平. 2  I2C总线特征 I2C总线上的每一个设备都可以作为主设备或者从设备

I2C总线协议详解

1.1 I2C总线知识 1.1.1  I2C总线物理拓扑结构     I2C 总线在物理连接上非常简单,分别由SDA(串行数据线)和SCL(串行时钟线)及上拉电阻组成.通信原理是通过对SCL和SDA线高低电平时序的控制,来 产生I2C总线协议所需要的信号进行数据的传递.在总线空闲状态时,这两根线一般被上面所接的上拉电阻拉高,保持着高电平. 1.1.2  I2C总线特征    I2C总线上的每一个设备都可以作为主设备或者从设备,而且每一个设备都会对应一个唯一的地址(可以从I2C器件的数据手册得知)

【转】I2C总线协议

I2C总线(Inter Integrated-Circuit)是由PHILIPS公司在上世纪80年代发明的一种电路板级串行总线标准,通过两根信号线--时钟线SCL和数据线SDA--即可完成主从机的单工通信.总线硬件连接极其简单,不同I2C设备挂接在总线上,只需在信号线安装上拉电阻即可完成硬件线路的搭建.另外,I2C总线采用器件地址的硬件设置方法,通过软件寻址方式完全避免了片选寻址的弊端,从而使硬件系统扩展更为灵活. 由于简单有效,I2C在业界得到广泛应用.基于I2C衍生出来的标准有SMBus.P

I2C总线协议学习笔记 (转载)

1.I2C协议   2条双向串行线,一条数据线SDA,一条时钟线SCL.   SDA传输数据是大端传输,每次传输8bit,即一字节.   支持多主控(multimastering),任何时间点只能有一个主控.   总线上每个设备都有自己的一个addr,共7个bit,广播地址全0.   系统中可能有多个同种芯片,为此addr分为固定部分和可编程部份,细节视芯片而定,看datasheet. 1.1 I2C位传输   数据传输:SCL为高电平时,SDA线若保持稳定,那么SDA上是在传输数据bit: 

I2C总线协议

1.I2C协议   2条双向串行线,一条数据线SDA,一条时钟线SCL.   SDA传输数据是大端传输,每次传输8bit,即一字节.   支持多主控(multimastering),任何时间点只能有一个主控.   总线上每个设备都有自己的一个addr,共7个bit,广播地址全0.   系统中可能有多个同种芯片,为此addr分为固定部分和可编程部份,细节视芯片而定,看datasheet. 1.1 I2C位传输   数据传输:SCL为高电平时,SDA线若保持稳定,那么SDA上是在传输数据bit: 

ADXL345经验总结,采用SPI和I2C总线操作

一. ADXL345简介 ADXL345是ADI公司推出的三轴(x,y,z)iMEMS数字加速度计(digital accelerometer),具有在16G下高分辨率(13Bit)测量能力,同时具备16Bit数字输出.ADXL345 适用于静态倾角测量以及动态加速度测量,高达4mg/LSB的灵敏度允许测量小于1度的倾角. 该传感器还具备单击 /双击探测,自由落体探测,并允许用户设置一个加速度阀值,当加速度值超过设定阀值后可以产生一个信号输出.所有这些功能都可以映射到2个中断上.内置的32级FI

17、I2C总线和CAT24WCxx存储器

1.I2C串行总线概述 I2C总线是PHLIPS公司推出的一种串行总线,是具备多主机系统所需的总线裁决和高低速器件同步功能的高性能串行总线.I2C总线只有两根双向信号线.一根是数据线SDA,另一根是时钟线SCL. 2.I2C总线通过上拉电阻接正电源.当总线空闲时,两根线均为高电平.连到总线上的任一器件输出的低电平,都将使总线的信号变低,即各器件的SDA及SCL都是线“与”关系. 3.每个接到I2C总线上的器件都有唯一的地址.主机与其它器件间的数据传送可以是由主机发送数据到其它器件,这时主机即为发

I2C总线使用说明

I2C总线协议 概念: 1.I2C(Inter-IntegratedCircuit)总线是一种由PHILIPS公司开发的两线式串行总线,用于连接微控制器及其外围设备(特别是外部存储器件),最初为音频和视频的开发. 2.I2C总线是由数据线SDA和时钟SCL构成的串行总线,可发送和接收数据. 3.I2C总线在传送数据过程中共有三种特殊类型信号, 它们分别是:开始信号.结束信号和应答信号. 4.I2C先传高位,后传送低位数据. 时序: I2C总线发送数据的整个过程时序 (1)I2C的起始信号 SCL