文章目录
一、hid核心初始化
在linux内核中,HID核心是完成HID功能的关键组件,如果内核支持HID,在启动过程中,则会对HID进行初始化,完成该操作的函数是hid_init()
,实现在/drivers/hid/hid-core.c中:
static int __init hid_init(void)
{
int ret;
ret = bus_register(&hid_bus_type);
if (ret) {
pr_err("can't register hid bus\n");
goto err;
}
#ifdef CONFIG_HID_BPF
hid_bpf_ops = &hid_ops;
#endif
ret = hidraw_init();
if (ret)
goto err_bus;
hid_debug_init();
return 0;
err_bus:
bus_unregister(&hid_bus_type);
err:
return ret;
}
- (1)调用
bus_register()
注册hid总线,在总线类型定义中指定了总线名称、dev_groups
、drv_groups
、.match
、.probe
、.remove
和.uevent
:
const struct bus_type hid_bus_type = {
.name = "hid",
.dev_groups = hid_dev_groups,
.drv_groups = hid_drv_groups,
.match = hid_bus_match,
.probe = hid_device_probe,
.remove = hid_device_remove,
.uevent = hid_uevent,
};
- (2)调用
hidraw_init()
初始化hidraw模块支持,hidraw模块提供了对hid原始数据的直接访问接口。 - (3)调用
hid_debug_init()
创建debugfs中的调试条目hid
。
上述则是hid初始化的具体步骤,以模块的方式构建进内核,在内核启动过程中自动完成。
二、hid总线probe过程分析
从hid_bus_type
总线可以知道,hid总线的probe是hid_device_probe()
,函数定义如下:
static int hid_device_probe(struct device *dev)
{
struct hid_device *hdev = to_hid_device(dev);
struct hid_driver *hdrv = to_hid_driver(dev->driver);
int ret = 0;
if (down_interruptible(&hdev->driver_input_lock))
return -EINTR;
hdev->io_started = false;
clear_bit(ffs(HID_STAT_REPROBED), &hdev->status);
if (!hdev->driver)
ret = __hid_device_probe(hdev, hdrv);
if (!hdev->io_started)
up(&hdev->driver_input_lock);
return ret;
}
上述函数将对HID设备进行探测,如果设备没有驱动程序,则尝试调用 __hid_device_probe()
函数来进行探测。__hid_device_probe()
实现如下:
static int __hid_device_probe(struct hid_device *hdev, struct hid_driver *hdrv)
{
const struct hid_device_id *id;
int ret;
if (!hid_check_device_match(hdev, hdrv, &id))
return -ENODEV;
hdev-></