免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1285 | 回复: 0
打印 上一主题 下一主题

关于device 与device_driver [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-02-22 22:54 |只看该作者 |倒序浏览


/**
*    driver_register - register driver with bus
*    @drv:     driver to register
*
*    We pass off most of the work to the bus_add_driver() call,
*    since most of the things we have to do deal with the bus
*    structures.
*
*    The one interesting aspect is that we setup @drv->unloaded
*    as a completion that gets complete when the driver reference
*    count reaches 0.
*/
int driver_register(struct device_driver * drv)
{
       klist_init(&drv->klist_devices, klist_devices_get, klist_devices_put);
       init_completion(&drv->unloaded);
       return bus_add_driver(drv);
}
int bus_add_driver(struct device_driver * drv)
{
       struct bus_type * bus = get_bus(drv->bus);
       int error = 0;

       if (bus) {
              pr_debug("bus %s: add driver %s\n", bus->name, drv->name);
              error = kobject_set_name(&drv->kobj, "%s", drv->name);
              if (error) {
                     put_bus(bus);
                     return error;
              }
              drv->kobj.kset = &bus->drivers;
              if ((error = kobject_register(&drv->kobj))) {
                     put_bus(bus);
                     return error;
              }

              driver_attach(drv);//这个函数用来匹配device与 device_driver机构
              klist_add_tail(&drv->knode_bus, &bus->klist_drivers);
              module_add_driver(drv->owner, drv);

              driver_add_attrs(bus, drv);
              driver_create_file(drv, &driver_attr_unbind);
              driver_create_file(drv, &driver_attr_bind);
       }
       return error;
}
/**
*    driver_attach - try to bind driver to devices.
*    @drv:     driver.
*
*    Walk the list of devices that the bus has on it and try to
*    match the driver with each one.  If driver_probe_device()
*    returns 0 and the @dev->driver is set, we've found a
*    compatible pair.
*/
void driver_attach(struct device_driver * drv)
{
       bus_for_each_dev(drv->bus, NULL, drv, __driver_attach);
}

static int __driver_attach(struct device * dev, void * data)
{
       struct device_driver * drv = data;

       /*
        * Lock device and try to bind to it. We drop the error
        * here and always return 0, because we need to keep trying
        * to bind to devices and some drivers will return an error
        * simply if it didn't support the device.
        *
        * driver_probe_device() will spit a warning if there
        * is an error.
        */

       down(&dev->sem);
       if (!dev->driver)
              driver_probe_device(drv, dev);
       up(&dev->sem);


       return 0;
}
/**
*    driver_probe_device - attempt to bind device & driver.
*    @drv:     driver.
*    @dev:     device.
*
*    First, we call the bus's match function, if one present, which
*    should compare the device IDs the driver supports with the
*    device IDs of the device. Note we don't do this ourselves
*    because we don't know the format of the ID structures, nor what
*    is to be considered a match and what is not.
*
*
*    This function returns 1 if a match is found, an error if one
*    occurs (that is not -ENODEV or -ENXIO), and 0 otherwise.
*
*    This function must be called with @dev->sem held.
*/
int driver_probe_device(struct device_driver * drv, struct device * dev)
{
       int ret = 0;

       if (drv->bus->match && !drv->bus->match(dev, drv))//由此可见匹配是有总线模型中定义的match函数决定的
              goto Done;

       pr_debug("%s: Matched Device %s with Driver %s\n",
               drv->bus->name, dev->bus_id, drv->name);
       dev->driver = drv;
       if (drv->probe) {
              ret = drv->probe(dev);//在此处 device_driver中的probe函数被调用所以一些初始化的东西可以放在这个函数中实现。
              if (ret) {
                     dev->driver = NULL;
                     goto ProbeFailed;
              }
       }
       device_bind_driver(dev);//将两个结构联系在一起
       ret = 1;
       pr_debug("%s: Bound Device %s to Driver %s\n",
               drv->bus->name, dev->bus_id, drv->name);
       goto Done;

ProbeFailed:
       if (ret == -ENODEV || ret == -ENXIO) {
              /* Driver matched, but didn't support device
               * or device not found.
               * Not an error; keep going.
               */
              ret = 0;
       } else {
              /* driver matched but the probe failed */
              printk(KERN_WARNING
                     "%s: probe of %s failed with error %d\n",
                     drv->name, dev->bus_id, ret);
       }
Done:
       return ret;
}
/*************************************************/
/**
*    driver_register - register driver with bus
*    @drv:     driver to register
*
*    We pass off most of the work to the bus_add_driver() call,
*    since most of the things we have to do deal with the bus
*    structures.
*
*    The one interesting aspect is that we setup @drv->unloaded
*    as a completion that gets complete when the driver reference
*    count reaches 0.
*/
int driver_register(struct device_driver * drv)
{
       klist_init(&drv->klist_devices, klist_devices_get, klist_devices_put);
       init_completion(&drv->unloaded);
       return bus_add_driver(drv);
}

int bus_add_driver(struct device_driver * drv)
{
       struct bus_type * bus = get_bus(drv->bus);
       int error = 0;

       if (bus) {
              pr_debug("bus %s: add driver %s\n", bus->name, drv->name);
              error = kobject_set_name(&drv->kobj, "%s", drv->name);
              if (error) {
                     put_bus(bus);
                     return error;
              }
              drv->kobj.kset = &bus->drivers;
              if ((error = kobject_register(&drv->kobj))) {
                     put_bus(bus);
                     return error;
              }

              driver_attach(drv);// 匹配device与 device_driver结构构
              klist_add_tail(&drv->knode_bus, &bus->klist_drivers);
              module_add_driver(drv->owner, drv);

              driver_add_attrs(bus, drv);
              driver_create_file(drv, &driver_attr_unbind);
              driver_create_file(drv, &driver_attr_bind);
       }
       return error;
}

总线类型注册其实也差不多啦?易2410为例内核的代码如下:
struct bus_type platform_bus_type = {
       .name             = "platform",
       .match           = platform_match,//这个函数将决定你的device 和 device_driver的匹配方式2410中 只是strcmp一下 name
       .suspend  = platform_suspend,
       .resume          = platform_resume,
};

struct device platform_bus = {
       .bus_id           = "platform",
};

int __init platform_bus_init(void)
{
       device_register(&platform_bus);
       return bus_register(&platform_bus_type);
}

上面是设备模型的注册要完成用户空间的系统还要调用各种设备的自己的注册过程。
比如 字符设备。
内核在调用 字符设备驱动时(如 open ) 要通过设备节点的主设备号来查找到对应的驱动程序所以你还是要调用 cdev_add 把你的字符设备注册到系统中。这个可以参考http://blog.chinaunix.net/u2/71164/showart_1887233.html


本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u3/102839/showart_2184247.html
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP