Linux 驱动开发基础知识——总线设备驱动模型(七)

 个人名片:

作者简介:学生
个人主页:妄北y

个人QQ:2061314755

个人邮箱:[email protected]
个人WeChat:Vir2021GKBS
本文由妄北y原创,首发CSDN
座右铭:大多数人想要改造这个世界,但却罕有人想改造自己。

专栏导航:

妄北y系列专栏导航:

C/C++的基础算法:C/C++是一种常用的编程语言,可以用于实现各种算法,这里我们对一些基础算法进行了详细的介绍与分享。

QT基础入门学习:对QT的基础图形化页面设计进行了一个简单的学习与认识,利用QT的基础知识进行了翻金币小游戏的制作

Linux基础编程:初步认识什么是Linux,为什么学Linux,安装环境,进行基础命令的学习,入门级的shell编程。

Linux应用开发基础开发:分享Linux的基本概念、命令行操作、文件系统、用户和权限管理等,网络编程相关知识,TCP/IP 协议、套接字(Socket)编程等,可以实现网络通信功能。

Linux项目开发:Linux基础知识的实践,做项目是最锻炼能力的一个学习方法,这里我们会学习到一些简单基础的项目开发与应用,而且都是毕业设计级别的哦。


非常期待和您一起在这个小小的互联网世界里共同探索、学习和成长。 ✨✨ 欢迎订阅本专栏 ✨✨ 

文章介绍:

本篇文章对Linux驱动基础学习的相关知识进行分享!

前面我们已经对驱动的基础写法有了一个初步的了解,但是我们之前的写法扩展性特别低,当我们将我们的驱动用在其他开发板时候就可能需要再次修改代码,现在我们需要对驱动的框架进行更细致的学习,这样才能更增强它的扩展性,更加易于扩展。

如果您觉得文章不错,期待你的一键三连哦,你的鼓励是我创作动力的源泉,让我们一起加油,一起奔跑,让我们顶峰相见!!!

感谢大家点赞收藏⭐评论✍️

一、驱动编写的 3 种方法

        以 LED 驱动为例。

1.1 传统方式

优缺点:

        使用哪个引脚,怎么操作引脚,都写死在代码中。

        最简单,不考虑扩展性,可以快速实现功能。

        修改引脚时,需要重新编译。

1.2 总线设备驱动模型

Linux 驱动开发基础知识——总线设备驱动模型(七)_第1张图片

         引入 platform_device/platform_driver,将“资源”与“驱动”分离开

优缺点:

        代码稍微复杂,但是易于扩展。

        冗余代码太多,修改引脚时设备端的代码需要重新编译。

        更换引脚时,图 9.3 中的 led_drv.c 基本不用改,但是需要修改 led_dev.c

1.3 设备树

Linux 驱动开发基础知识——总线设备驱动模型(七)_第2张图片

优缺点:

        通过配置文件──设备树来定义“资源”

        代码稍微复杂,但是易于扩展。

        无冗余代码,修改引脚时只需要修改 dts 文件并编译得到 dtb 文件,把它传 给内核。

        无需重新编译内核/驱动。

二、在 Linux 中实现“分离”:Bus/Dev/Drv 模型

Linux 驱动开发基础知识——总线设备驱动模型(七)_第3张图片

2.1 匹配规则 

2.1.1 最先比较

        platform_device.driver_override platform_driver.driver.name

        可以设置 platform_device driver_override,强制选择某个 platform_driver

2.1.2 然后比较

        platform_device. nameplatform_driver.id_table[i].name

        platform_driver.id_table 是“platform_device_id”指针,表示该 drv 支持若干个 device,它里面列出了各个 device{.name, .driver_data},其中的“name”表示该 drv 支持的设备的名字driver_data 是些提供给该 device 的私有数据

2.1.3  最后比较

        platform_device.name platform_driver.driver.name

        platform_driver.id_table 可能为空, 这时可以根据 platform_driver.driver.name 来寻找同名的 platform_device

2.2.4 函数调用关系

platform_device_register
platform_device_add
     device_add
         bus_add_device // 放入链表
         bus_probe_device // probe 枚举设备,即找到匹配的(dev, drv)
             device_initial_probe
                 __device_attach
                     bus_for_each_drv(...,__device_attach_driver,...)
                         __device_attach_driver
                             driver_match_device(drv, dev) // 是否匹配
                             driver_probe_device // 调用 drv 的 probe
platform_driver_register
__platform_driver_register
     driver_register
         bus_add_driver // 放入链表
             driver_attach(drv)
                 bus_for_each_dev(drv->bus, NULL, drv, __driver_attach);
                     __driver_attach
                         driver_match_device(drv, dev) // 是否匹配
                         driver_probe_device // 调用 drv 的 probe

2.2 常用函数

2.2.1 注册/反注册

platform_device_register/ platform_device_unregister
platform_driver_register/ platform_driver_unregister
platform_add_devices // 注册多个 device

2.2.2  获得资源

返回该 dev 中某类型(type)资源中的第几个(num)

struct resource *platform_get_resource(struct platform_device *dev, unsigned int 
                                        type,
                                        unsigned int num)

返回该 dev 所用的第几个(num)中断: 

int platform_get_irq(struct platform_device *dev, unsigned int num)

通过名字(name)返回该dev 的某类型(type)资源: 

struct resource *platform_get_resource_byname(struct platform_device *dev, 
                                                unsigned int type, 
                                                const char *name)

通过名字(name)返回该 dev 的中断号: 

int platform_get_irq_byname(struct platform_device *dev, const char *name)

三、怎么写程序 

3.1 分配/设置/注册 platform_device 结构体

         在里面定义所用资源,指定设备名字。

3.2 分配/设置/注册 platform_driver 结构体

        在其中的 probe 函数里,分配/设置/注册 file_operations 结构体。

        并从 platform_device 中确实所用硬件资源。

        指定 platform_driver 的名字

大佬觉得有用的话点个赞 呗。
❤️❤️❤️本人水平有限,如有纰漏,欢迎各位大佬评论批评指正!

如果觉得这篇文对你有帮助的话,也请给个点赞、收藏下吧,非常感谢!

任务在无形中完成,价值在无形中升华,让我们一起加油吧!

你可能感兴趣的:(Linux,驱动开发基础知识,linux,嵌入式硬件,imx6ull,总线设备驱动模型,驱动程序,驱动编写方法,设备树)