Linux--IIC驱动编程实验

本文详细阐述了Linux中的I2C驱动架构,涉及I2C总线驱动、设备驱动的结构,以及如何通过i2c_client和i2c_driver进行设备信息描述和驱动内容编写,包括probe函数的使用和i2c_transfer在数据收发中的作用。
摘要由CSDN通过智能技术生成
对于 I2C 主机驱动,一旦编写完成就不需要再做修改,其他的 I2C 设备直接调用主机驱动提供的 API 函数完成读写操作即可。这个正好符合 Linux 的驱动分离与分层的思想,因此 Linux内核也将 I2C 驱动分为两部分:
①、 I2C 总线驱动, I2C 总线驱动就是 SOC I2C 控制器驱动,也叫做 I2C 适配器驱动。
②、 I2C 设备驱动, I2C 设备驱动就是针对具体的 I2C 设备而编写的驱动。

 总的来说,IIC的驱动分为IIC控制器(主机)的驱动和IIC设备驱动(从机),一般控制器的驱动芯片原厂已经写好,我们要做的就是根据自己要用的IIC设备写设备驱动。

一、I2C 设备驱动

I2C 设备驱动重点关注两个数据结构: i2c_client i2c_driver ,根据总线、设备和驱动模型。还剩下设备和驱动, i2c_client 就是描述设备信息的, i2c_driver 描述驱动内容,类似于platform_driver。

1.1  i2c_client 结构体

i2c_client 结构体定义在 include/linux/i2c.h 文件中,内容如下:
一个设备对应一个 i2c_client ,每检测到一个 I2C 设备就会给这个 I2C 设备分配一个i2c_client。

1.2   i2c_driver 结构体

i2c_driver 类似 platform_driver ,是我们编写 I2C 设备驱动重点要处理的内容, i2c_driver 结构体定义在 include/linux/i2c.h 文件中,内容如下:
示例代码 61.1.2.2 i2c_driver 结构体
161 struct i2c_driver {
162 unsigned int class;
163
164 /* Notifies the driver that a new bus has appeared. You should 
165 * avoid using this, it will be removed in a near future.
166 */
167 int (*attach_adapter)(struct i2c_adapter *) __deprecated;
168
169 /* Standard driver model interfaces */
170 int (*probe)(struct i2c_client *, const struct i2c_device_id *);
171 int (*remove)(struct i2c_client *);
172
173 /* driver model interfaces that don't relate to enumeration */
174 void (*shutdown)(struct i2c_client *);
175
176 /* Alert callback, for example for the SMBus alert protocol.
177 * The format and meaning of the data value depends on the 
178 * protocol.For the SMBus alert protocol, there is a single bit 
179 * of data passed as the alert response's low bit ("event 
180 flag"). */
181 void (*alert)(struct i2c_client *, unsigned int data);
182
183 /* a ioctl like command that can be used to perform specific 
184 * functions with the device.
185 */
186 int (*command)(struct i2c_client *client, unsigned int cmd,
void *arg);
187
188 struct device_driver driver;
189 const struct i2c_device_id *id_table;
190
191 /* Device detection callback for automatic device creation */
192 int (*detect)(struct i2c_client *, struct i2c_board_info *);
193 const unsigned short *address_list;
194 struct list_head clients;
195 };
170 行,当 I2C 设备和驱动匹配成功以后 probe 函数就会执行,和 platform 驱动一样。
188 行, device_driver 驱动结构体,如果使用设备树的话,需要设置 device_driver 的 of_match_table 成员变量,也就是驱动的兼容 (compatible) 属性。
对于我们 I2C 设备驱动编写人来说,重点工作就是构建 i2c_driver ,构建完成以后需要向Linux 内核注册这个 i2c_driver i2c_driver 注册函数为 int i2c_register_driver

二、I2C设备驱动编写流程

2.1   I2C 设备信息描述,此处主要叙述使用设备树的时候

使用设备树的时候 I2C 设备信息通过创建相应的节点就行了,比如 NXP 官方的 EVK 开发板在 I2C1 上接了 mag3110 这个磁力计芯片,因此必须在 i2c1 节点下创建 mag3110 子节点,然后在这个子节点内描述 mag3110 这个芯片的相关信息。
示例代码 61.3.1.4 mag3110 子节点
1 &i2c1 {
2 clock-frequency = <100000>;
3 pinctrl-names = "default"; 4 pinctrl-0 = <&pinctrl_i2c1>;
5 status = "okay"; 6 
7 mag3110@0e { 8 compatible = "fsl,mag3110"; 9 reg = <0x0e>;
10 position = <2>;
11 };
......
20 };
7~11 行,向 i2c1 添加 mag3110 子节点,第 7 行“ mag3110@0e ”是子节点名字,“ @ ” 后面“0e ”就是 mag3110 I2C 器件地址。第 8 行设置 compatible 属性值为“ fsl,mag3110 ”。
9 行的 reg 属性也是设置 mag3110 的器件地址的,因此值为 0x0e I2C 设备节点的创建重点 是 compatible 属性和 reg 属性的设置,一个用于匹配驱动,一个用于设置器件地址。
2.2   I2C 设备数据收发处理流程
I2C 设备驱动首先要做的就是初始化 i2c_driver 并向 Linux 内核注册。当设备和驱动匹配以后 i2c_driver 里面的 probe 函数就会执行, probe 函数里面所做的就是字符设备驱动那一套了。一般需要在 probe 函数里面初始化 I2C 设备,要初始化 I2C 设备就必须能够对 I2C 设备寄存器进行读写操作,这里就要用到 i2c_transfer 函数了。 i2c_transfer 函数最终会调用 I2C 适配器中 i2c_algorithm 里面的 master_xfer 函数,对于 I.MX6U 而言就是i2c_imx_xfer 这个函数。i2c_transfer 函数原型如下:
int i2c_transfer(struct i2c_adapter *adap, 
struct i2c_msg *msgs, 
int num)
adap 所使用的 I2C 适配器, i2c_client 会保存其对应的 i2c_adapter
msgs I2C 要发送的一个或多个消息。
num 消息数量,也就是 msgs 的数量。
返回值: 负值,失败,其他非负值,发送的 msgs 数量。
我们重点来看一下 msgs 这个参数,这是一个 i2c_msg 类型的指针参数, I2C 进行数据收发说白了就是消息的传递,Linux 内核使用 i2c_msg 结构体来描述一个消息。
使用 i2c_transfer 函数发送数据之前要先构建好 i2c_msg

三、实验程序编写(以AP3216C为例)

3.1     修改设备树

IO 修改或添加(配置要用到的总线的IO口)
首先肯定是要修改 IO AP3216C 用到了 I2C1 接口, I.MX6U-ALPHA 开发板上的 I2C1 接口使用到了 UART4_TXD UART4_RXD
示例代码 61.5.1.1 pinctrl_i2c1 子节点
1 pinctrl_i2c1: i2c1grp { 2 fsl,pins = < 3 MX6UL_PAD_UART4_TX_DATA__I2C1_SCL 0x4001b8b0
4 MX6UL_PAD_UART4_RX_DATA__I2C1_SDA 0x4001b8b0
5 >;
6 };
pinctrl_i2c1 就是 I2C1 IO 节点,这里将 UART4_TXD UART4_RXD 这两个 IO 分别复用为 I2C1_SCL I2C1_SDA
i2c1 节点追加 ap3216c 子节点
AP3216C 是连接到 I2C1 上的,因此需要在 i2c1 节点下添加 ap3216c 的设备子节点

3.2     AP3216C 驱动编写

补充:
上拉电阻的存在意义:
I2C是一个同步半双工硬件层次的串行通信协议、I2C总线总共只有两条信号线,串行时钟线SCL和串行数据线SDA。I2C总线上的各器件的数据线都接到SDA线上,I2C总线上的各器件的时钟线都接到SCL线上。
                   

当I2C总线空闲时,SDA和SCL均为高电平。由于连接到总线上的器件输出级必须是开漏输出的,因此只要有一个器件任意时刻输出低电平,都将总线上的信号都将总线上的信号变低,即各器件的SDA和SCL都是“线与“的关系。

由于各器件输出端为漏极开路,则SCL和SDA线必须通过上拉电阻接正电源,以保证SDA和SCL在空闲状态被上拉到高电平。

总线若一直保持在低电平状态,这可能导致以下问题:

  1. 通信混乱:在IIC总线上,设备通过拉低总线电平来发送数据。如果总线一直处于低电平状态,其他设备无法确定何时可以开始发送数据或进行通信,从而导致通信混乱。

  2. 总线锁定:如果总线一直处于低电平状态,其他设备无法发送数据或者尝试发送时会遭遇总线冲突。这可能导致总线被锁定,无法进行有效的通信。

总结:
对iic的驱动大概分为3部分:
核心层 drivers/i2c/i2c-core.c:I2C 设备和驱动的匹配过程是由 I2C 核心来完成的,drivers/i2c/i2c-core.c 就是 I2C 的核心部分,I2C 核心提供了一些与具体硬件无关的 API 函数,不依赖硬件平台的接口函数,实现注册I2C总线驱动和设备驱动的函数。
增加/删除i2c_driver;我们可以在i2c_driver的驱动入口函数和出口函数见到。
增加/删除i2c_adapter;
I2C传输、发送和接收;i2c_transfer()

i2c_transfer()用于I2C适配器与设备之间进行一组信息交互,其中第二个参数是指向一个i2c_msg数组的指针,所以i2c_transfer一次性可以传输多个i2c_msg。

对于时序简单的外设,i2c_master_send和i2c_master_recv会调用i2c_transfer分别完成一条写消息和读一条消息,但是i2c_transfer本身不具备驱动适配器物理硬件以完成信息交互的能力,只是寻找与i2c_adapter对应的i2c_algorithm,使用(主机驱动代码中)i2c_algorithm的x_fer()函数真正驱动硬件流程。

                        

主机驱动层 提供了I2C总线驱动和设备驱动的注册、注销、通信方法(Algorithm)上层与具体适配器无关的代码以及其他上层代码。
初始化 i2c_adapter 结构体变量,然后设置 i2c_algorithm 中的 master_xfer 函数。完成以后通过 i2c_add_numbered_adapter或 i2c_add_adapter 这两个函数向系统注册设置好的 i2c_adapter
设备驱动层 :2C 设备驱动重点关注两个数据结构: i2c_client 和 i2c_driver;
i2c_client 就是描述设备信息的,i2c_driver 描述驱动内容,类似于 platform_driver。
还有字符设备的常用框架:file_opr、probe()
设备驱动中写明了
1) read_regs函数来读取该I2C设备的寄存器参数,而read_regs函数中包括两大部分:1、实现msg数组存放帧格式数据等参数,比如该设备的从机地址,读取位flag,读取数据长度,读取到的数据等   2、使用transfer函数调用定义在主机驱动中的函数让参数代表的对应主机开启I2C通信并按msg信息执行操作
2) read_regs读到数据后将数据通过copy_to_user传递给用户空间。
             

原文链接:https://blog.csdn.net/m0_46573394/article/details/134859092

  • 10
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值