Linux第91步_了解“platform总线,platform驱动和platform设备”,以及驱动框架和设备框架

plattorm是为了驱动的分离与分层而提出来的一种框架,其驱动的具体实现还是需要字符设备驱动、块设备驱动或网络设备驱动

对于一个完整的驱动程序,必须提供“有设备树”和“无设备树”两种匹配方法。

1、总线

Linux系统内核使用bus_type结构体表示总线,需要包含“#include <linux/device.h>”,结构体如下:

struct bus_type {

    const char    *name;

    const char    *dev_name;

    struct device     *dev_root;

    const struct attribute_group **bus_groups;

    const struct attribute_group **dev_groups;

    const struct attribute_group **drv_groups;

    int (*match)(struct device *dev, struct device_driver *drv);

//dev是设备,drv是驱动,根据“注册的驱动”查找相应的设备

    int (*uevent)(struct device *dev, struct kobj_uevent_env *env);

    int (*probe)(struct device *dev);

    int (*remove)(struct device *dev);

    void (*shutdown)(struct device *dev);

    int (*online)(struct device *dev);

    int (*offline)(struct device *dev);

    int (*suspend)(struct device *dev, pm_message_t state);

    int (*resume)(struct device *dev);

    int (*num_vf)(struct device *dev);

    int (*dma_configure)(struct device *dev);

    const struct dev_pm_ops *pm;

    const struct iommu_ops *iommu_ops;

    struct subsys_private *p;

    struct lock_class_key lock_key;

    bool need_parent_lock;

};

2、platform总线

platform_bus_type是platform平台总线,它是bus_type的一个具体实例,需要包含“#include <linux/platform_device.h>”,platform总线定义如下:

struct bus_type  platform_bus_type = {

    .name      = "platform",

    .dev_groups   = platform_dev_groups,

    .match     = platform_match,

    .uevent       = platform_uevent,

    .dma_configure    = platform_dma_configure,

    .pm    = &platform_dev_pm_ops,

};

#include <linux/of_device.h>

static int platform_match(struct device *dev,struct device_driver *drv)

{

struct platform_device *pdev = to_platform_device(dev);

struct platform_driver *pdrv = to_platform_driver(drv);

/*When driver_override is set,only bind to the matching driver*/

if (pdev->driver_override)//当设置driver_override时,只绑定到匹配的驱动程序

return !strcmp(pdev->driver_override, drv->name);

/* Attempt an OF style match first */

if (of_driver_match_device(dev, drv))

//第1种匹配方式:OF类型的匹配,就是设备树采用的匹配方式

return 1;

/* Then try ACPI style match */

if (acpi_driver_match_device(dev, drv)) //第2种匹配方式:ACPI匹配方式

return 1;

/* Then try to match against the id table */

if (pdrv->id_table)//第3种匹配方式:id_table匹配

return platform_match_id(pdrv->id_table, pdev) != NULL;

/* fall-back to driver name match */

return (strcmp(pdev->name, drv->name) == 0);

/*

第4种匹配方式:如果第3种匹配方式的id_table不存在,就直接比较驱动和设备的name 字段,看看是不是相等,如果相等的话就匹配成功。

*/

}

3、platform驱动

对于一个完整的驱动程序,必须提供有设备树和无设备树两种匹配方法。

platform_driver结构体表示“platform驱动”,需要包含“#include <linux/platform_device.h>”,结构体如下:

struct platform_driver {

    int (*probe)(struct platform_device *);

/*probe()函数,当“驱动”与“设备”匹配成功以后,probe()函数就会执行*/

    int (*remove)(struct platform_device *);

    void (*shutdown)(struct platform_device *);

    int (*suspend)(struct platform_device *, pm_message_t state);

    int (*resume)(struct platform_device *);

    struct device_driver  driver;

    const struct platform_device_id  *id_table;

/*id_table是个表(也就是数组),每个元素的类型为platfomm_device_id*/

    bool prevent_deferred_probe;

};

platfomm_device_id结构体如下:

struct platform_device_id {

    char name[PLATFORM_NAME_SIZE];

    kernel_ulong_t driver_data;

};

device_driver结构体,需要包含“#include <linux/device.h>”如下:

struct device_driver {

    const char    *name;

    struct bus_type      *bus;

    struct module     *owner;

    const char    *mod_name; /* used for built-in modules */

    bool suppress_bind_attrs;   /* disables bind/unbind via sysfs */

    enum probe_type probe_type;

    const struct of_device_id   *of_match_table;

/*of_match_table是采用设备树时,驱动使用的匹配表,同样是数组,每个匹配项都为of_device_id结构体类型*/

    const struct acpi_device_id *acpi_match_table;

    int (*probe) (struct device *dev);

    int (*remove) (struct device *dev);

    void (*shutdown) (struct device *dev);

    int (*suspend) (struct device *dev, pm_message_t state);

    int (*resume) (struct device *dev);

    const struct attribute_group **groups;

    const struct attribute_group **dev_groups;

    const struct dev_pm_ops *pm;

    void (*coredump) (struct device *dev);

    struct driver_private *p;

};

使用of_device_id结构体,需要包含“#include <linux/mod_devicetable.h>”,结构体如下:

struct of_device_id {

    char   name[32];

    char   type[32];

    char   compatible[128];

/*

对于设备树而言,就是通过设备节点的 compatible 属性值和of_match_table中每个项目的 compatible成员变量进行比较,如果有相等的就表示设备和此驱动匹配成功;

*/

    const void *data;

};

int platform_driver_register (struct platform_driver *driver)

//向Linux内核注册一个platform驱动

//driver:要注册的 platform 驱动;

//返回值:负数,失败;0,成功;

void platform_driver_unregister(struct platform_driver *drv)

//卸载一个platform驱动

//drv:要卸载的platform驱动;

4、platform驱动框架举例

/* 设备结构体 */

struct xxx_dev{

dev_t devid; /*声明32位变量devid用来给保存设备号*/

int major;   /*主设备号*/

int minor;   /*次设备号*/

struct cdev  cdev; /*字符设备结构变量cdev */

struct class *class;     /*类*/

struct device *device;  /*设备*/

atomic_t lock;  /*原子变量*/

struct fasync_struct *async_queue;

/* fasync_struct结构体,即“异步通知”结构体 */

};

struct xxx_dev xxxdev; /* 定义个设备结构体变量 */

static int xxx_open(struct inode *inode, struct file *filp)

{

  /* 函数具体内容 */

  return 0;

}

static ssize_t xxx_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)

{

  /* 函数具体内容 */

  return 0;

}

/*字符设备驱动操作集*/

static struct file_operations xxx_fops = {

   .owner = THIS_MODULE,

   .open = xxx_open,

   .write = xxx_write,

};

/*platform驱动的probe函数

驱动与设备匹配成功以后此函数就会执行

*/

static int xxx_probe(struct platform_device *dev)

{

  ......

  cdev_init(&xxxdev.cdev, &xxx_fops); /* 注册字符设备驱动 */

  /* 函数具体内容 */

  return 0;

}

static int xxx_remove(struct platform_device *dev)

{

  .....

  cdev_del(&xxxdev.cdev);/* 删除cdev */

  /* 函数具体内容 */

  return 0;

}

/*对于一个完整的驱动程序,必须提供有设备树和无设备树两种匹配方法*/

/* 匹配列表 */

static const struct of_device_id  xxx_of_match[] = {

  { .compatible = "xxx-gpio" },

  { /* Sentinel */ }

};

/*platform平台驱动结构体*/

static struct platform_driver xxx_driver = {

  .driver = {

  .name = "xxx",//用于传统的驱动与设备匹配

  .of_match_table = xxx_of_match, //用于设备树下的驱动与设备检査

  },

  .probe = xxx_probe,

  .remove = xxx_remove,

};

/* 驱动模块加载 */

static int __init xxxdriver_init(void)

{

   return platform_driver_register(&xxx_driver);

   //向Linux内核注册一个platform驱动

}

/* 驱动模块卸载 */

static void __exit xxxdriver_exit(void)

{

  platform_driver_unregister(&xxx_driver);

  //卸载一个platform驱动

}

module_init(xxxdriver_init);

module_exit(xxxdriver_exit);

MODULE_LICENSE("GPL");

MODULE_AUTHOR("zgq");

5、platform设备

platform_device结构体,需要包含“#include <linux/device.h>”如下:

struct platform_device {

    const char *name;

//设备名字,要和platform驱动的name字段相同,这样,设备和驱动才能匹配

    int    id;

    bool       id_auto;

    struct device dev;

    u64    platform_dma_mask;

    u32    num_resources; //资源数量

    struct resource   *resource;//表示资源,也是设备信息

    const struct platform_device_id *id_entry;

    char *driver_override; /* Driver name to force a match */

    /* MFD cell pointer */

    struct mfd_cell *mfd_cell;

    /* arch specific additions */

    struct pdev_archdata archdata;

};

resource结构体,需要包含“#include <linux/ioport.h>”如下:

struct resource {

    resource_size_t start;   //资源的起始信息,若为内存资源,则表示内存的起始地址

    resource_size_t end;    //资源的终止信息,若为内存资源,则表示内存的终止地址

    const char *name;      //资源的名字

    unsigned long flags;     //资源的內型

    unsigned long desc;     //

    struct resource *parent, *sibling, *child;

};

资源內型

#define IORESOURCE_BITS     0x000000ff   /* Bus-specific bits */

#define IORESOURCE_TYPE_BITS    0x00001f00  /* Resource type */

#define IORESOURCE_IO          0x00000100

   /*IO口的资源,PCI/ISA I/O ports */

#define IORESOURCE_MEM      0x00000200  //内存地址

#define IORESOURCE_REG      0x00000300 /* Register offsets */

#define IORESOURCE_IRQ      0x00000400  //中断号

#define IORESOURCE_DMA      0x00000800  //DMA通道号

#define IORESOURCE_BUS      0x00001000  //总线号

#define IORESOURCE_PREFETCH   0x00002000 /* No side effects */

#define IORESOURCE_READONLY   0x00004000

#define IORESOURCE_CACHEABLE    0x00008000

#define IORESOURCE_RANGELENGTH  0x00010000

#define IORESOURCE_SHADOWABLE   0x00020000

int platform_device_register(struct platform_device *pdev)

/*在linux不支持设备树时,需要采用手动方式将“platform设备信息”注册到Linux内核*/

//pdev:要注册的platform设备。

//返回值:负数,失败;0,成功。

void platform_device_unregister(struct platform_device *pdev)

//在linux不支持设备树时,需要手动注销掉相应的“platform设备”

//pdev:要注销的platform设备。

6、platform设备框架举例:

/* 寄存器地址定义*/

#define PERIPH1_REGISTER_BASE (0X20000000)   /* 外设1寄存器首地址 */

#define PERIPH2_REGISTER_BASE (0X020E0068)   /* 外设2寄存器首地址 */

#define REGISTER_LENGTH    4                 /* 设备资源 */

static struct resource xxx_resources[] = {

   [0] = { .start = PERIPH1_REGISTER_BASE,//设备外设1起始地址

           .end = (PERIPH1_REGISTER_BASE + REGISTER_LENGTH - 1),

           //设备外设1终止地址

           .flags = IORESOURCE_MEM, //资源內型:内存地址

         },

   [1] = { .start = PERIPH2_REGISTER_BASE, //设备外设2起始地址

           .end = (PERIPH2_REGISTER_BASE + REGISTER_LENGTH - 1),

           //设备外设2终止地址

           .flags = IORESOURCE_MEM, //资源內型:内存地址

         },

};

/* platform设备结构体 */

static struct platform_device xxxdevice = {

   .name = "xxx-gpio",  //必须保证和驱动中的名字一致

   .id = -1,

   .num_resources = ARRAY_SIZE(xxx_resources),

   //数组xxx_resources[]的大小,ARRAY_SIZE()用来计算数组元素的个数

   .resource = xxx_resources,//数组xxx_resources[]

};

/* 设备模块加载 */

static int __init xxxdevice_init(void)

{

  return platform_device_register(&xxxdevice);

//在linux不支持设备树时,需要将“platform设备信息”注册到Linux内核

}

/* 设备模块注销 */

static void __exit xxx_resourcesdevice_exit(void)

{

   platform_device_unregister(&xxxdevice);

   //注销掉相应的“platform设备”

   //pdev=&xxxdevice:要注销的platform设备。

}

module_init(xxxdevice_init);

module_exit(xxxdevice_exit);

MODULE_LICENSE("GPL");

MODULE_AUTHOR("zgq");

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/823412.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

重生奇迹mu恶魔来袭副本

在游戏重生奇迹mu中&#xff0c;恶魔来袭副本是玩家能够组队通过的副本。但是因为手游组队的不方便性&#xff0c;部分玩家对其还是非常苦手。而今天&#xff0c;我们就给大家讲解一下这个游戏的双人通关攻略。 1、挂机找怪手动输出 (1)对于普通剧情副本而言&#xff0c;挂机…

python爬虫原理和编程实战:爬取CSDN博主的账号信息

&#x1f9d1; 作者简介&#xff1a;阿里巴巴嵌入式技术专家&#xff0c;深耕嵌入式人工智能领域&#xff0c;具备多年的嵌入式硬件产品研发管理经验。 &#x1f4d2; 博客介绍&#xff1a;分享嵌入式开发领域的相关知识、经验、思考和感悟,欢迎关注。提供嵌入式方向的学习指导…

黑马点评(四) -- 分布式锁

1 . 分布式锁基本原理和实现方式对比 分布式锁&#xff1a;满足分布式系统或集群模式下多进程可见并且互斥的锁。 分布式锁的核心思想就是让大家都使用同一把锁&#xff0c;只要大家使用的是同一把锁&#xff0c;那么我们就能锁住线程&#xff0c;不让线程进行&#xff0c;让…

4.1 返回JSON数据

1. 默认实现方式 JSON是目前主流的前后端数据传输方式&#xff0c;Spring MVC中使用消息转换器HttpMessageConverter对JSON的转换提供了很好的支持&#xff0c;在Spring Boot中更进一步&#xff0c;对相关配置做了更进一步的简化。 默认情况下&#xff0c;当开发者新创建一个S…

24.4.17 驱动开发定时器作业,消抖

定时器消抖工作原理 在按键按下之后&#xff0c;进入中断处理函数&#xff0c;在中断处理函数中&#xff0c;定时时间10ms当定时时间到&#xff0c;执行定时器处理函数&#xff0c;在定时器处理函数中&#xff0c;读取管脚的电平状态如果读到的是低电平&#xff0c;表示按键按下…

x264 编码器 refine_subpel 函数

x264 264是一个开源的视频编码库,用于将视频压缩为H.264/AVC(Advanced Video Coding)格式。它是一种广泛使用的视频编码标准,能够提供高质量的视频压缩和较低的比特率。x264库提供了一个编码器,可以将原始视频序列转换为H.264/AVC压缩的比特流。它实现了各种H.264编码算法…

刷题日记——质因数的个数

题目 分析&#xff08;从质数的判断角度出发&#xff09;&#xff08;递归&#xff09; 判断n是否是质数需要&#xff1a; 遍历&#xff0c;i从2开始到sqrt(n)&#xff0c;每次判断n是否能整除i&#xff0c;若能则不是质数&#xff0c;若不能则是 思路&#xff1a; 先判断n…

【MATLAB源码-第26期】基于matlab的FBMC/OQAM的误码率仿真。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 FBMC&#xff08;Filter Bank Multicarrier&#xff09;是一种多载波调制技术&#xff0c;它采用滤波器组来处理频域内的子载波&#xff0c;以在有限带宽内实现高效的数据传输。OQAM&#xff08;Offset Quadrature Amplitude…

数据结构初阶:二叉树(一)

树概念及结构 树的概念 树是一种 非线性 的数据结构&#xff0c;它是由 n &#xff08; n>0 &#xff09;个有限结点组成一个具有层次关系的集合。 把它叫做树是因 为它看起来像一棵倒挂的树&#xff0c;也就是说它是根朝上&#xff0c;而叶朝下的 。 有一个特殊的结点&a…

微前端 qiankun 框架接入问题记录

背景&#xff1a;需要搭建一个平台&#xff0c;这个平台的主要功能是集成各个子系统&#xff0c;方面对系统之间的统一管理。在搭建这样一个平台时&#xff0c;前端考虑使用微前端架构方式实现&#xff0c;使用的框架是 qiankun&#xff0c;本文主要记录在 qiankun 框架使用过程…

【Leetcode每日一题】 分治 - 颜色分类(难度⭐⭐)(57)

1. 题目解析 题目链接&#xff1a;75. 颜色分类 这个问题的理解其实相当简单&#xff0c;只需看一下示例&#xff0c;基本就能明白其含义了。 2.算法原理 算法思路解析 本算法采用三指针法&#xff0c;将数组划分为三个区域&#xff0c;分别用于存放值为0、1和2的元素。通过…

Ubuntu修改DNS

【永久修改DNS】 临时修改DNS的方法是在 /etc/resolv.conf 添加&#xff1a;nameserver 8.8.8.8 nameserver 8.8.8.8 注意到/etc/resolv.conf最上面有这么一行&#xff1a; DO NOT EDIT THIS FILE BY HAND -- YOUR CHANGES WILL BE OVERWRITTEN 说明重启之后这个文件会被自动…

CTFHUB-技能树-Web前置技能-文件上传(无验证,JS前端验证,前端验证)

CTFHUB-技能树-Web前置技能-文件上传&#xff08;无验证&#xff0c;JS前端验证&#xff0c;前端验证—.htaccess&#xff09; 文章目录 CTFHUB-技能树-Web前置技能-文件上传&#xff08;无验证&#xff0c;JS前端验证&#xff0c;前端验证—.htaccess&#xff09;文件上传无验…

【在线OJ系统】自定义注解实现自增ID的无感插入

实现思路 首先自定义参数注解&#xff0c;然后根据AOP思想&#xff0c;找到该注解作用的切点&#xff0c;也就是mapper层对于mapper层的接口在执行前都会执行该aop操作&#xff1a;获取到对于的方法对象&#xff0c;根据方法对象获取参数列表&#xff0c;根据参数列表判断某个…

电商数据采集的网页抓取数据、淘宝、天猫、京东等平台的电商数据抓取|电商数据API接口网页爬虫、采集网站数据

电商数据采集的网页抓取数据、淘宝、天猫、京东等平台的电商数据抓取&#xff0c;网页爬虫、采集网站数据、网页数据采集软件、python爬虫、HTM网页提取、APP数据抓包、APP数据采集、一站式网站采集技术、BI数据的数据分析、数据标注等成为大数据发展中的热门技术关键词。那么电…

深入理解同步与异步编程及协程管理在Python中的应用

文章目录 1. 同步与异步函数的对比1.1 同步函数1.2 异步函数1.3 对比 2. 管理多个协程与异常处理2.1 并发执行多个协程2.2 错误处理2.3 任务取消 本文将探索Python中同步与异步编程的基本概念及其区别。还会详细介绍如何使用asyncio库来有效管理协程&#xff0c;包括任务的创建…

最新的网易星球GEC挖矿系统修复版 章鱼星球挖矿系统源码 区块链虚拟币交易源码 基于ThinkPHP5开发

区块链系统介绍 2018.12.10更新增加聚合数据短信接口 2018.11.19更新增加短信宝接口 2018.08.17修复Linux系统搭建验证码不显示问题 2018.08.09修复后台某处溢出数据库账号密码BUG 2018.08.06修复票卷BUG 源码介绍&#xff1a; 区块链系统中用户共九个等级&#xff0c;依…

旧衣服回收小程序,旧衣回收行业的必然发展趋势

近年来&#xff0c;旧衣回收行业成为了一个新型的创业项目&#xff0c;因其投资成本低、回报高的优势&#xff0c;也成为了当下年轻人的创业新选择。 一、旧衣服回收市场发展趋势 当下人们对衣物淘汰的速度逐渐加快&#xff0c;每年产生的废旧衣物高达百万吨&#xff0c;加之…

延迟阻塞队列DelayQueue的用法例子(延迟任务)

DelayQueue: JDK自带的延迟队列&#xff0c;基于阻塞队列实现。 import java.util.Calendar; import java.util.Date; import java.util.concurrent.DelayQueue; import java.util.concurrent.Delayed; import java.util.concurrent.TimeUnit;//延迟任务 例子&#xff1a; pu…

.cur 鼠标光标编辑器

详解透明贴图和三元光栅操作 - CodeBus 鼠标指针文件格式解析——Windows&#xff08;二&#xff09; (qq.com) [C/C] RGBA数组生成Windows下的ico文件_c ico格式-CSDN博客 色环设计 - CodeBus 左键绘制 右键选颜色 ctrl右键设置鼠标热点 F1导出.cur文件 //代码来源&…