platform 设备

platform_device

并不是与字符设备、块设备并列的概念,而是Linux系统提供的一种管理设备的手段,所有SOC系统中集成的独立的外设控制器、挂接在SOC内存空间的外设等都属Platform设备。如把内部集成的I2C、PTC、SPI、LCD、看门狗控制器都归纳为Platform设备,而这些设备本身就是字符设备。由此可见Platform中的设备是不分字符设备还是块设备。

bus ,device ,driver

platform总线是一种虚拟的总线,相应的设备则为platform_device,
而驱动则为platform_driver
把I2C、RTC、LCD,leds-pwm等都归纳为platform_device。

对platform_device的定义通常在BSP的板文件中实现,在板文件中,将platform_device归纳为一个数组,最终通过 platform_add_devices() / platform_device_register函数统一注册。

2.6内核加入了platform虚拟总线。platform机制将设备本身的资源注册进内核,有内核统一管理,在驱动程序使用这些资源时使用统一的接口,这样提高了程序的可移植性。

1、platform总线:

linux在系统启动时就注册了platform总线
在设备总线驱动模型的中,BUS像一个月老一样,通过它的match函数,将注册到bus中的device与driver进行配对,那么每一个不同的bus 都有自己的match函数
platform_device_driver->name 与 platform_device->name 进行比较,
将驱动注册到内核中去,如果有同名设备,则调用driver->probe函数....

platform_device_dirver

从 device_driver 封装而来的

platform_device

从device封装而来

dev真正的设备

通过 container_of ,就能找到整个platform_device

总线中定义了成员名字和match函数,当有总线或者设备注册到platform总线时,内核自动调用match函数,判断设备和驱动的name是否一致。

static int platform_match(struct device *dev, struct device_driver *drv)
{
struct platform_device *pdev;

    pdev = container_of(dev, struct platform_device, dev);
     return (strcmp(pdev->name, drv->name) == 0); //配对函数检验名字是否一致
 }
 struct bus_type platform_bus_type = {
     .name = "platform", //定义了总线名字为platform,总线注册后新建目录sys/bus/platform
     .dev_attrs = platform_dev_attrs,
     .match = platform_match, //指定配对函数
     .uevent = platform_uevent,
     .pm = PLATFORM_PM_OPS_PTR,
 };

2、platform设备:

platform设备对应的结构体paltform_device:

/*linux/platform_device.h*/
 struct platform_device {
    const char * name; //设备的名字,这将代替device->dev_id,用作sys/device下显示的目录名
     int id; //设备id,用于给插入给该总线并且具有相同name的设备编号,如果只有一个设备的话填-1。
    struct device dev; //结构体中内嵌的device结构体。
     u32 num_resources; //资源数。
     struct resource * resource; //用于存放资源的数组。
 };
struct resource {  
    resource_size_t start;      //资源的起始值  
    resource_size_t end;        //资源的结束值  
    const char *name;  
    unsigned long flags;        //资源的类型,如IORESOURCE_IO,IORESOURCE_MEM,IORESOURCE_IRQ,IORESOURCE_DMA  
    struct resource *parent, *sibling, *child;  
}; 

其中有个重要的成员是resource,是设备的资源信息,如IO地址,中断号等。
设备可能有多个资源,通常使用platform_get_resource函数来获取资源

设备注册

platform_device的封装就是指定了一个目录的名字name,并且内嵌device。

platform_device的注册和注销使用以下函数

 int platform_device_register(struct platform_device *pdev) 

 void platform_device_unregister(struct platform_device *pdev)

注册后,同样会在/sys/device/目录下创建一个以name命名的目录,并且创建软连接到/sys/bus/platform/device下。

3、platform驱动:

先看一下platform驱动对应的结构体paltform_driver:

/*linux/platform_device.h*/
 struct platform_driver {
     int (*probe)(struct platform_device *);
     int (*remove)(struct platform_device *);
     void (*shutdown)(struct platform_device *);
     int (*suspend)(struct platform_device *, pm_message_t state);
     int (*suspend_late)(struct platform_device *, pm_message_t state);
     int (*resume_early)(struct platform_device *);
     int (*resume)(struct platform_device *);
     struct device_driver driver;
 };

可以看到,platform_driver结构体内嵌了device_driver,并且实现了probe、remove等操作。其实,当内核需要调用probe函数时,它会调用driver->probe,在driver->probe中再调用platform_driver->probe。
注册成功后内核会在/sys/bus/platform/driver/目录下创建一个名字为driver->name的目录。

./sys/devices/platform/plat_usb_mouse.0
./sys/bus/platform/devices/plat_usb_mouse.0
./sys/bus/platform/drivers/plat_usb_mouse
./sys/bus/platform/drivers/plat_usb_mouse/plat_usb_mouse.0

当我查找usb_mouse时出现了四个目录,但是为什么后面有个“0”?这个0代表设备的编号,是由paltform_device->id指定的,默认为0。如果你不想的你的目录名字没有后缀,那你就设置platform_device->id = -1;

platform设备的资源和数据

 struct platform_device {
     const char * name;
     int id;
     struct device dev;
     u32 num_resources;
     struct resource * resource;
 };

resource是一个指向platform资源数组的指针,该数组中有num_resource个资源,看一下资源结构体:

/*linux/ioport.h"*/
 struct resource {
     resource_size_t start;
     resource_size_t end;
     const char *name;
     unsigned long flags;
     struct resource *parent, *sibling, *child;
24 };

常见的flags有IORESOURCE_MEM和IORESOURCE_IRQ。其他的可以自己查看include/linux/ioport.h
如果flags为IORESOURCE_MEM,start和end分别是该设备的连续的开始和结束地址,如果不连续你可以定义两个或者更多的资源结构体。
如果flags为IORESOURCE_IRQ,start和end分别是该设备连续的开始和结束的连续中断号,如果不连续可以分开定义,当然,如果地址或者中断只有一个,你可以将start和end定义成一样。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容