分享

usb_register_dev

 WUCANADA 2013-10-10

usb_register_dev

分类: Liux Kernel Function 764人阅读 评论(0) 收藏 举报

一个USBinterface对应一种USB逻辑设备,比如鼠标、键盘、音频流。所以,在USB范畴中,device一般就是指一个 interface。一个驱动只控制一个interface。这样,usb_register_dev自然是注册一个interface,所以 usb_register_dev的第一个参数是interface(usb_interface类型)。

 

usb_register_dev注册一次,获取一个次设备号。该次设备号从usb_class_driver -> minor_base开始分配。

usb_register_dev(interface, &skel_class),也就是说,一个usb_interface对应一个次设备号。结合上面举的interface例子,可以知道,鼠标、键盘各自对应一个不同的次设备号。


usb_register_dev创建USB设备,USB设备的主设备号都是180,次设备号决定没每个不同的设备。

usb_register_de一般会在probe函数中调用。

int usb_register_dev(struct usb_interface *intf,
             struct usb_class_driver *class_driver)
{
    int retval;
    int minor_base = class_driver->minor_base;
    int minor;
    char name[20];
    char *temp;

#ifdef CONFIG_USB_DYNAMIC_MINORS
    /*
     * We don't care what the device tries to start at, we want to start
     * at zero to pack the devices into the smallest available space with
     * no holes in the minor range.
     */
    minor_base = 0;
#endif


  //判断是否定义了file_operations结构体,没有定义则返回,不用创建USB设备
    if (class_driver->fops == NULL)
        return -EINVAL;

//次设备号必须大于0
    if (intf->minor >= 0) 
        return -EADDRINUSE;
//创建USB class类,首先会判断usb_class-是否为空,为空才创建,不为空说明已经有了这个class了,usb_class->class = class_create(THIS_MODULE, "usb");
    retval = init_usb_class();
    if (retval)
        return retval;

    dev_dbg(&intf->dev, "looking for a minor, starting at %d", minor_base);
//找到一个没有使用的次设备号,把file_operations放在usb_minors中
    down_write(&minor_rwsem);
    for (minor = minor_base; minor < MAX_USB_MINORS; ++minor) {
        if (usb_minors[minor])
            continue;

        usb_minors[minor] = class_driver->fops;
        intf->minor = minor;
        break;
    }
    up_write(&minor_rwsem);
    if (intf->minor < 0)
        return -EXFULL;
//提取usb设备的名字,class_driver->name是传入的值
    /* create a usb class device for this usb interface */
    snprintf(name, sizeof(name),class_driver->name, minor - minor_base);
    temp = strrchr(name, '/');
    if (temp && (temp[1] != '\0'))
        ++temp;
    else
        temp = name;

//创建usb设备
    intf->usb_dev = device_create(usb_class->class, &intf->dev,
                      MKDEV(USB_MAJOR, minor), class_driver,
                      "%s", temp);

//如果创建usb设备错误,则清空usb_minors
    if (IS_ERR(intf->usb_dev)) {
        down_write(&minor_rwsem);
        usb_minors[minor] = NULL;
        intf->minor = -1;
        up_write(&minor_rwsem);
        retval = PTR_ERR(intf->usb_dev);
    }
    return retval;
}

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多