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;
}
I2C总线协议的软件模拟实现方法的更多相关文章
- I2C总线协议的简要说明
为了快速的了解I2C总线协议,此处采用另类的方式进行说明. 倘若你和另外一个人只能通过一个开关加灯泡的装置在不同的两个房间进行交流,以下是很简单能说明的一个模型: 你的房间有一个开关,另外一间房间有一 ...
- [I2C]I2C总线协议图解
转自:http://blog.csdn.net/w89436838/article/details/38660631 1 I2C总线物理拓扑结构 I2C 总线在物理连接上非常简单,分别由S ...
- I2C总线协议的总结介绍
在看天翔哥的视频之后,他强调要把I2C协议好好研究一下,那么就对一些基本的通信手段是十分有帮助的..那么就来了解一下I2C总线协议的一些知识吧. I2C(Inter-Integrated Circui ...
- I2C总线协议图解(转载)
转自:http://blog.csdn.net/w89436838/article/details/38660631 另外,https://blog.csdn.net/qq_38410730/arti ...
- 【转】I2C总线协议
I2C总线(Inter Integrated-Circuit)是由PHILIPS公司在上世纪80年代发明的一种电路板级串行总线标准,通过两根信号线——时钟线SCL和数据线SDA——即可完成主从机的单工 ...
- I2C总线协议详解
I2C总线定义 I2C(Inter-Integrated Circuit)总线是一种由PHILIPS公司开发的两线式串行总线,用于连接微控制器及其外围设备.I2C总线产生于在80年代,最初为音 ...
- I2C 总线协议
1.I2C协议 2条双向串行线,一条数据线SDA,一条时钟线SCL. SDA传输数据是大端传输,每次传输8bit,即一字节. 支持多主控(multimastering),任何时间点只能 ...
- I2C总线协议
1.I2C协议 2条双向串行线,一条数据线SDA,一条时钟线SCL. SDA传输数据是大端传输,每次传输8bit,即一字节. 支持多主控(multimastering),任何时间点只能有 ...
- I2C总线协议学习笔记 (转载)
1.I2C协议 2条双向串行线,一条数据线SDA,一条时钟线SCL. SDA传输数据是大端传输,每次传输8bit,即一字节. 支持多主控(multimastering),任何时间点只能有一 ...
随机推荐
- 游戏2048源代码 - C语言控制台界面版
一.游戏介绍 <2048>是最近比较流行的一款数字游戏.原版2048首先在github上发布,原作者是Gabriele Cirulli.它是基于<1024>和<小3传奇& ...
- LeetCode - 690. Employee Importance
You are given a data structure of employee information, which includes the employee's unique id, his ...
- [Python Study Notes]psutil模块
系统性能信息模块psutil psutil是一个跨平台库,能够轻松实现获取系统运行的进程和系统利用率(CPU,内存,磁盘,网络等)信息,主要应用于系统监控,分析和限制系统资源及进程的管理,它实现了同等 ...
- 阿里云CentOS使用iptables禁止某IP访问
在CentOS下封停IP,有封杀网段和封杀单个IP两种形式.一般来说,现在的攻击者不会使用一个网段的IP来攻击(太招摇了),IP一般都是散列的.于是下面就详细说明一下封杀单个IP的命令,和解封单个IP ...
- codeforces 940D 比赛总结
这次比赛总体还行,但是并没发挥到极致 A题 速度正常 题解 B题 这个题先是没注意时间复杂度,tle了,好不容易优化了没多测几组就交了,很开心的wa了,查了一边发现没特判k,改好后有草率地交了,又wa ...
- [NOIP]2016天天爱跑步
[NOIP]2016天天爱跑步 标签: LCA 树上差分 NOIP Description 小C同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.<天天爱跑步>是 ...
- 在C#中几种常见数组复制方法的效率对比
原文是在http://blog.csdn.net/jiangzhanchang/article/details/9998229 看到的,本文在原文基础上增加了新的方法,并对多种数据类型做了更全面的对比 ...
- Entity Framework——记录执行的命令信息
有两种方法可以记录执行的SQl语句: 使用DbContext.Database.Log属性 实现IDbCommandInterceptor接口 一 使用DbContext.Database.Log属性 ...
- 登录功能(MD5加密)
登录这个功能,是不管哪个项目都会用到的,登录做的好坏,安全性的保障将直接影响到整个系统的成败,尤其是一些安全性要求比较严格的项目 1.首先需要对密码进行加密,这里用到的是md5加密,需要在login. ...
- PendingIntent
PendingIntent表示一种即将发生的意图,和Intent的区别在于:PendingIntent是在将来的某个不确定的时刻发生,而Intent是立刻发生 典型使用场景是给RemoteViews添 ...