linux 内核驱动–Platform Device和Platform_driver注册过程[通俗易懂]

linux 内核驱动–Platform Device和Platform_driver注册过程[通俗易懂]

转自网络,自己学习用
 
从Linux2.6起引入了一套新的驱动管理和注册机制:Platform_device和Platform_driver。
Linux中大部分的设备驱动,都可以使用这套机制, 设备用Platform_device表示,驱动用Platform_driver进行注册。

Linuxplatformdriver机制和传统的devicedriver 机制(通过driver_register函数进行注册

大家好,又见面了,我是你们的朋友全栈君。

转自网络,自己学习用

 

Linux 2.6 起引入了一套新的驱动管理和注册机制 :Platform_device Platform_driver

Linux 中大部分的设备驱动,都可以使用这套机制 ,  设备用 Platform_device 表示,驱动用 Platform_driver 进行注册。


Linux platform driver 机制和传统的 device driver  机制 ( 通过 driver_register 函数进行注册 ) 相比,一个十分明显的优势在于 platform 机制将设备本身的资源注册进内核,由内核统一管理,在驱动程序中使用这些资源时通过 platform device 提供的标准接口进行申请并使用。这样提高了驱动和资源管理的独立性,并且拥有较好的可移植性和安全性 ( 这些标准接口是安全的 )


Platform 机制的本身使用并不复杂,由两部分组成: platform_device platfrom_driver

通过 Platform 机制开发发底层驱动的大致流程为   定义   platform_add_devices 注册   platform_device, 定义   platform_add_driver 注册   platform_driver

 

1、platform_device注册过程:


首先要确认的就是设备的资源信息,例如设备的地址,中断号等。

2.6 内核中 platform 设备用结构体 platform_device 来描述,该结构体定义在 kernel/include/linux/platform_device.h 中,

struct platform_device {

    const char    * name;
    int        id;
    struct device    dev;
    u32        num_resources;
    struct resource    *  resource ;
    struct platform_device_id    *id_entry;
};

该结构一个重要的元素是 resource ,该元素存入了最为重要的设备资源信息,定义在 kernel/include/linux/ioport.h 中,

struct resource {

    resource_size_t start;
    resource_size_t end;
    const char *name;
    unsigned long flags;
    struct resource *parent, *sibling, *child;
};

 

下面举 s3c6410 平台的 i2c 驱动作为例子来说明:

static struct platform_device *smdk6410_devices [] __initdata = {

#ifdef CONFIG_SMDK6410_SD_CH0
    &s3c_device_hsmmc0,
#endif
#ifdef CONFIG_SMDK6410_SD_CH1
    &s3c_device_hsmmc1,
#endif
    &s3c_device_i2c0 ,
    &s3c_device_i2c1,
    &s3c_device_fb,
    &s3c_device_usb,
    &s3c_device_usb_hsotg,
    &smdk6410_lcd_powerdev,

    &smdk6410_smsc911x,
};
把一个或几个设备资源放在一起,便于集中管理,其中IIC设备

platform_device如下:

struct platform_device  s3c_device_i2c0   = {

    .name          = “s3c2410-i2c”,
#ifdef CONFIG_S3C_DEV_I2C1
    .id          = 0,
#else
    .id          = -1,
#endif
    .num_resources      = ARRAY_SIZE(s3c_i2c_resource ),
    .resource      = s3c_i2c_resource,
};
具体resource如下:

static struct resource  s3c_i2c_resource [] = {

    [0] = {

        .start = S3C_PA_IIC,
        .end   = S3C_PA_IIC + SZ_4K – 1,
        .flags = IORESOURCE_MEM,
    },
    [1] = {

        .start = IRQ_IIC,
        .end   = IRQ_IIC,
        .flags = IORESOURCE_IRQ,
    },
};

这里定义了两组 resource ,它描述了一个 I2C 设备的资源,第 1 组描述了这个 I2C 设备所占用的总线地址范围, IORESOURCE_MEM 表示第 1 组描述的是内存类型的资源信息,第 2 组描述了这个 I2C 设备的中断号, IORESOURCE_IRQ 表示第 2 组描述的是中断资源信息。设备驱动会根据 flags 来获取相应的资源信息。



定义好了 platform_device 结构体后就可以调用函数 platform_add_devices 向系统中添加该设备了,之后可以调用 platform_driver_register() 进行设备注册。

s3c6410-i2c的platform_device是在系统启动时,在mach-smdk6410.c里的smdk6410_machine_init()函数里进行注册的,这个函数申明为arch_initcall的函数调用,arch_initcall的优先级高于module_init。所以会在Platform驱动注册之前调用。(详细参考imach-smdk6410.c)


static void __init smdk6410_machine_init(void)
{

    s3c_i2c0_set_platdata(NULL);
    s3c_i2c1_set_platdata(NULL);
    s3c_fb_set_platdata(&smdk6410_lcd_pdata);

    gpio_request(S3C64XX_GPN(5), “LCD power”);
    gpio_request(S3C64XX_GPF(13), “LCD power”);
    gpio_request(S3C64XX_GPF(15), “LCD power”);

    i2c_register_board_info(0, i2c_devs0, ARRAY_SIZE(i2c_devs0));
    i2c_register_board_info(1, i2c_devs1, ARRAY_SIZE(i2c_devs1));

    platform_add_devices(smdk6410_devices, ARRAY_SIZE(smdk6410_devices));
    //添加多设备

}
int platform_add_devices(struct platform_device **devs, int num)
{

    int i, ret = 0;

    for (i = 0; i < num; i++) {

         ret = platform_device_register(devs[i]);
        if (ret) {

            while (–i >= 0)
                platform_device_unregister(devs[i]);
            break;
        }
    }

    return ret;
}
int platform_device_register(struct platform_device *pdev)
{

    device_initialize(&pdev->dev);
     return platform_device_add(pdev);
}
int platform_device_add(struct platform_device *pdev)
{

    int i, ret = 0;

    if (!pdev)
        return -EINVAL;

    if (!pdev->dev.parent)
        pdev->dev.parent = &platform_bus;

    pdev->dev.bus = &platform_bus_type;

    if (pdev->id != -1)
        dev_set_name(&pdev->dev, “%s.%d”, pdev->name,  pdev->id);//如果有id 表示有多个同类设备用

pdev->name和
pdev->id标识该设备


    else
        dev_set_name(&pdev->dev, “%s”, pdev->name);
//否则,只用

pdev->name标识该设备



    for (i = 0; i < pdev->num_resources; i++) {

        struct resource *p, *r = &pdev->resource[i];

        if (r->name == NULL)
            r->name = dev_name(&pdev->dev);

        p = r->parent;
        if (!p) {

            if (resource_type(r) == IORESOURCE_MEM)
                 p = &iomem_resource;

//   作为 IOMEM 资源分配  

            else if (resource_type(r) == IORESOURCE_IO)
                 p = &ioport_resource;

//   作为 IO PORT资源分配  

        }

         if (p && insert_resource(p, r)) {

// 将新的 resource 插入内核 resource tree

            printk(KERN_ERR
                   “%s: failed to claim resource %d/n”,
                   dev_name(&pdev->dev), i);
            ret = -EBUSY;
            goto failed;
        }
    }

    pr_debug(“Registering platform device ‘%s’. Parent at %s/n”,
         dev_name(&pdev->dev), dev_name(pdev->dev.parent));

     ret = device_add(&pdev->dev);//添加设备到设备树
    if (ret == 0)
        return ret;

 failed:
    while (–i >= 0) {

        struct resource *r = &pdev->resource[i];
        unsigned long type = resource_type(r);

        if (type == IORESOURCE_MEM || type == IORESOURCE_IO)
            release_resource(r);
    }

    return ret;
}

1、platform_driver注册过程:


在platform_device注册完成后,就可以进行platform_driver注册了,在驱动初始化函数中调用函数 platform_driver_register() 注册 platform_driver ,需要注意的是 s3c_device_i2c 结构中 name 元素和 s3c6410_i2c_driver 结构中 driver.name 必须是相同的,这样在 platform_driver_register() 注册时会对所有已注册的所有 platform_device 中的 name 和当前注册的 platform_driver driver.name 进行比较,只有找到相同的名称的 platfomr_device 才能注册成功。

static struct platform_driver s3c24xx_i2c_driver = {

    .probe        = s3c24xx_i2c_probe,
    .remove        = s3c24xx_i2c_remove,
    .suspend_late    = s3c24xx_i2c_suspend_late,
    .resume        = s3c24xx_i2c_resume,
    .id_table    = s3c24xx_driver_ids,
    .driver        = {

        .owner    = THIS_MODULE,
        .name    = “s3c-i2c”,
    },
};

static int __init i2c_adap_s3c_init(void)
{

     return platform_driver_register(&s3c24xx_i2c_driver);//注册IIC驱动
}

nt platform_driver_register(struct platform_driver *drv)
{

    drv->driver.bus = &platform_bus_type;
    if (drv->probe)
        drv->driver.probe = platform_drv_probe;
    if (drv->remove)
        drv->driver.remove = platform_drv_remove;
    if (drv->shutdown)
        drv->driver.shutdown = platform_drv_shutdown;

     return driver_register(&drv->driver);
}

int driver_register(struct device_driver *drv)
{

    int ret;
    struct device_driver *other;

    BUG_ON(!drv->bus->p);

    if ((drv->bus->probe && drv->probe) ||
        (drv->bus->remove && drv->remove) ||
        (drv->bus->shutdown && drv->shutdown))
        printk(KERN_WARNING “Driver ‘%s’ needs updating – please use “
            “bus_type methods/n”, drv->name);

    other = driver_find(drv->name, drv->bus);//检查Driver是否已经存在
    if (other) {

        put_driver(other);
        printk(KERN_ERR “Error: Driver ‘%s’ is already registered, “
            “aborting…/n”, drv->name);
        return -EBUSY;
    }
    //若不存在,则添加驱动到驱动树。
    ret = bus_add_driver(drv);
    if (ret)
        return ret;
    ret = driver_add_groups(drv, drv->groups);
    if (ret)
        bus_remove_driver(drv);
    return ret;
}
int  bus_add_driver (struct device_driver *drv)
{

    struct bus_type *bus;
    struct driver_private *priv;
    int error = 0;

    bus = bus_get(drv->bus);
    if (!bus)
        return -EINVAL;

    pr_debug(“bus: ‘%s’: add driver %s/n”, bus->name, drv->name);

    priv = kzalloc(sizeof(*priv), GFP_KERNEL);
    if (!priv) {

        error = -ENOMEM;
        goto out_put_bus;
    }
    klist_init(&priv->klist_devices, NULL, NULL);
    priv->driver = drv;
    drv->p = priv;
    priv->kobj.kset = bus->p->drivers_kset;
    error = kobject_init_and_add(&priv->kobj, &driver_ktype, NULL,
                     “%s”, drv->name);
    if (error)
        goto out_unregister;

    if (drv->bus->p->drivers_autoprobe) {

        error = driver_attach(drv);
        if (error)
            goto out_unregister;
    }
    klist_add_tail(&priv->knode_bus, &bus->p->klist_drivers);
    module_add_driver(drv->owner, drv);

    error = driver_create_file(drv, &driver_attr_uevent);
    if (error) {

        printk(KERN_ERR “%s: uevent attr (%s) failed/n”,
            __func__, drv->name);
    }
    error = driver_add_attrs(bus, drv);
    if (error) {

        /* How the hell do we get out of this pickle? Give up */
        printk(KERN_ERR “%s: driver_add_attrs(%s) failed/n”,
            __func__, drv->name);
    }
    error = add_bind_files(drv);
    if (error) {

        /* Ditto */
        printk(KERN_ERR “%s: add_bind_files(%s) failed/n”,
            __func__, drv->name);
    }

    kobject_uevent(&priv->kobj, KOBJ_ADD);
    return 0;
out_unregister:
    kfree(drv->p);
    drv->p = NULL;
    kobject_put(&priv->kobj);
out_put_bus:
    bus_put(bus);
    return error;
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/163811.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • leetcode 颜色分类_LEETCODE

    leetcode 颜色分类_LEETCODE给定一个包含红色、白色和蓝色,一共 n 个元素的数组,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色、白色、蓝色顺序排列。此题中,我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。示例 1:输入:nums = [2,0,2,1,1,0]输出:[0,0,1,1,2,2]示例 2:输入:nums = [2,0,1]输出:[0,1,2]示例 3:输入:nums = [0]输出:[0]示例 4:输入:nums = [1]输出:[1] 提示:n == num

    2022年8月9日
    10
  • 文本分类算法比较与总结

    文本分类算法比较与总结本文对常用的几种文本分类算法进行了比较与总结 主要阐述它们之间的优劣 为算法的选择提供依据 nbsp 一 Rocchio 算法 nbsp Rocchio 算法应该算是人们思考文本分类问题最先能想到的 也是最符合直觉的解决方法 基本的思路是把一个类别里的样本文档各项取个平均值 例如把所有 体育 类文档中词汇 篮球 出现的次数取个平均值 再把 裁判 取个平均值 依次做下去 就可以得到一个新的向量 形象的称之为 质心 质

    2025年7月17日
    5
  • 【matlab安装】手把手图文并茂安装matlab2021(win10版)

    【matlab安装】手把手图文并茂安装matlab2021(win10版)本文介绍matlab的安装。

    2022年6月18日
    43
  • SMTP协议详解及工作过程

    SMTP协议详解及工作过程

    2022年3月5日
    47
  • pcep协议什么意思_SDN学习笔记

    pcep协议什么意思_SDN学习笔记SDN什么是SDNSDN是一种框架和思想,核心诉求是通过软件控制网络,实现业务的自动化部署,为方便软件来控制网络,希望控制面和转发面是分离的。例如,传统的交换机内部,由交换机负责具体的网络流量往哪里转发,在SDN中,有一个控制器进行流量转发的计算,然后将结果发送给交换机,交换机只进行简单的转发,从分布式的控制转发过程称为集中式的控制,使得控制和转发平面相分离。SDN的特点网络开放可编程、数控分离(…

    2025年9月29日
    2
  • eclipse环境的搭建以及JDK的安装步骤详细[通俗易懂]

    java学习之旅java开发环境:eclipse可以从这里下载链接:https://pan.baidu.com/s/1yPTV4UrAyWnc12NbAIj-MA密码:bnma安装步骤如下:从上链接获取压缩包下载到电脑之后,解压,此时的eclipse是打不开的,需要安装JDK和JRE,jdk的详细安装步骤在下面出现以下情况都是因为还没有安装JDK的原因:…

    2022年4月8日
    43

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号