RK3568驱动指南|第十五篇 I2C-第171章 I2C 驱动框架编写

瑞芯微RK3568芯片是一款定位中高端的通用型SOC,采用22nm制程工艺,搭载一颗四核Cortex-A55处理器和Mali G52 2EE 图形处理器。RK3568 支持4K 解码和 1080P 编码,支持SATA/PCIE/USB3.0 外围接口。RK3568内置独立NPU,可用于轻量级人工智能应用。RK3568 支持安卓 11 和 linux 系统,主要面向物联网网关、NVR 存储、工控平板、工业检测、工控盒、卡拉 OK、云终端、车载中控等行业。


第171章I2C 驱动框架编写

在上一个章节中,讲解了I2C Client设备树的编写,在I2C控制器节点中添加了FT5X06触摸芯片对应的节点,接下来将开始一步步完善I2C驱动程序,首先在本章节中将会完善I2C驱动框架,注册一个最简单的I2C驱动设备,并匹配上一章节中编写的设备树,在后面的章节中将会逐步对该驱动程序进行内容的填充。

171.1 注册I2C设备

I2C设备的注册使用的函数为i2c_add_driver,被定义在内核源码的“include/linux/i2c.h”目录下,具体内容如下所示:

#define i2c_add_driver(driver) \
	i2c_register_driver(THIS_MODULE, driver)

可以看到i2c_add_driver是一个宏定义,这个宏定义的作用是为了简化注册 I2C 设备驱动程序的过程,实际注册I2C设备的函数为i2c_register_driver,该函数定义在“drivers/i2c/i2c-core-base.c”文件中,具体内容如下所示:

int i2c_register_driver(struct module *owner, struct i2c_driver *driver)
{
    int res;

    /* 在驱动模型初始化完成之前,无法注册驱动程序 */
    if (WARN_ON(!is_registered))
        return -EAGAIN;

    /* 将驱动程序添加到驱动核心的 i2c 驱动列表中 */
    driver->driver.owner = owner;
    driver->driver.bus = &i2c_bus_type;
    INIT_LIST_HEAD(&driver->clients);

    /* 当注册返回时,驱动核心会为所有匹配但尚未绑定的设备调用 probe() 函数 */
    res = driver_register(&driver->driver);
    if (res)
        return res;

    pr_debug("driver [%s] registered\n", driver->driver.name);

    /* 遍历所有已经存在的适配器 */
    i2c_for_each_dev(driver, __process_new_driver);

    return 0;
}

该函数的主要作用是将 I2C 设备驱动程序注册到驱动核心中,并初始化相关数据结构,这里传入的数据结构类型为i2c_driver,需要我们在驱动程序编写的时候进行填充,该结构体定义在“include/linux/i2c.h”头文件中,具体内容如下所示:

struct i2c_driver {
    unsigned int class;  // 驱动程序所属的设备类型

   int (*probe)(struct i2c_client *, const struct i2c_device_id *); // 探测并绑定设备的回调函数
    int (*remove)(struct i2c_client *); // 从设备上解绑驱动程序的回调函数
    int (*probe_new)(struct i2c_client *); // 新的探测设备并绑定的回调函数
    void (*shutdown)(struct i2c_client *); // 设备关闭时调用的回调函数
    void (*alert)(struct i2c_client *, enum i2c_alert_protocol protocol,
               unsigned int data); // 设备报警时调用的回调函数,格式和含义取决于所使用的协议
    int (*command)(struct i2c_client *client, unsigned int cmd, void *arg); // 用于执行设备特定功能的命令回调函数
    struct device_driver driver; // 设备驱动程序基础结构
    const struct i2c_device_id *id_table; // 与该驱动程序匹配的设备 ID 表
    int (*detect)(struct i2c_client *, struct i2c_board_info *); // 用于自动创建设备的探测回调函数
    const unsigned short *address_list; // 与该驱动程序匹配的设备地址列表
    struct list_head clients; // 与该驱动程序绑定的 I2C 设备列表
    bool disable_i2c_core_irq_mapping; // 禁用 I2C 核心中断映射的标志
};

在调用i2c_add_driver函数注册I2C设备之前,需要先填充i2c_driver结构体,然后实现的各种回调函数,跟前面讲解的平台总线内容相同。

171.2驱动程序的编写

本实验驱动对应的网盘路径为:iTOP-RK3568开发板【底板V1.7版本】\03_【iTOP-RK3568开发板】指南教程\02_Linux驱动配套资料\04_Linux驱动程序\102_ft5x06_01

本实验旨在搭建最简单的I2C驱动程序框架,申请注册一个I2C设备,所以只填充了最简单的probe初始化函数和remove移除函数,在后面的章节中会对该驱动程序进行填充。

编写完成的ft5x06_driver.c代码如下所示:

#include <linux/init.h>
#include <linux/module.h>
#include <linux/i2c.h>
#include <linux/of_device.h>

// ft5x06设备的初始化函数
int ft5x06_probe(struct i2c_client *client, const struct i2c_device_id *id) {
    printk("This is ft5x06 probe\n");
    return 0;
}

// ft5x06设备的移除函数
int ft5x06_remove(struct i2c_client *client) {
    return 0;
}

// 设备树匹配表
static const struct of_device_id ft5x06_id[] = {
    { .compatible = "my-ft5x06" },
    { }, 
};

// ft5x06设备驱动结构体
static struct i2c_driver ft5x06_driver = {
    .driver = {
        .owner = THIS_MODULE,
        .name = "my-ft5x06",
        .of_match_table = ft5x06_id, // 添加设备树匹配表
    },    
    .probe = ft5x06_probe,
    .remove = ft5x06_remove,    
};

// 驱动初始化函数
static int __init ft5x06_driver_init(void) {
    int ret;
    // 注册I2C设备驱动
    ret = i2c_add_driver(&ft5x06_driver);
    if (ret < 0) {
        printk("i2c_add_driver is error\n");
        return ret;
    }
    return 0;
}

// 驱动退出函数
static void __exit ft5x06_driver_exit(void) {
    // 注销I2C设备驱动
    i2c_del_driver(&ft5x06_driver);
}

module_init(ft5x06_driver_init);
module_exit(ft5x06_driver_exit);
MODULE_LICENSE("GPL");

171.3 运行测试

171.3.1 编译驱动程序

在上一小节中的ft5x06_driver.c代码同一目录下创建 Makefile 文件,Makefile 文件内容如下所示:

export ARCH=arm64#设置平台架构
export CROSS_COMPILE=aarch64-linux-gnu-#交叉编译器前缀
obj-m += ft5x06_driver.o    #此处要和你的驱动源文件同名
KDIR :=/home/topeet/Linux/linux_sdk/kernel    #这里是你的内核目录                                                                                                                            
PWD ?= $(shell pwd)
all:
    make -C $(KDIR) M=$(PWD) modules    #make操作
clean:
    make -C $(KDIR) M=$(PWD) clean    #make clean操作

对于Makefile的内容注释已在上图添加,保存退出之后,来到存放platform_driver.c和Makefile文件目录下,如下图所示:

然后使用命令“make”进行驱动的编译,编译完成如下图所示:

编译完生成ft5x06_driver.ko目标文件,如下图所示:

至此驱动模块就编译成功了。

171.3.2 运行测试

在进行实验之前,首先要确保开发板烧写的是我们在170.3小节中编译出来的boot.img。开发板启动之后,首先进入到“/proc/device-tree/i2c@fe5a0000/”目录下确保存在 my-ft5x06@38目录,如下图所示:

只有在设备树节点编写正确的前提下,这里才会生成my-ft5x06@38目录,如果没有出现my-ft5x06@38目录就要回头检查看看了。

然后使用以下命令进行驱动模块的加载,如下图所示:

insmod ft5x06_driver.ko

可以看到成功打印了在probe函数中的打印,证明我们添加的设备树节点和ft5x06_driver驱动匹配成功了。

然后使用以下命令进行驱动模块的卸载,如下图所示:

rmmod ft5x06_driver.ko

由于没有在remove卸载函数中添加打印相关内容,所以使用rmmod命令卸载驱动之后,没有任何打印,至此,最简单的I2C驱动实验就完成了。 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值