网站品牌栏目建设情况广州竞价托管代运营
文章目录
- input_handle结构体详解
- 配对的input设备与input事件处理器实例
- input核心层对驱动层和事件层之间的框架建立流程图
本文章中与input子系统相关的结构体可参考input子系统结构体解析
input函数路径:drivers/input/input.c
input_handle结构体详解
input_handle结构体属于核心层,代表一个配对的input设备与input事件处理器。
int input_register_handle(struct input_handle *handle)
{struct input_handler *handler = handle->handler;struct input_dev *dev = handle->dev;int error;/* 获取互斥锁 */error = mutex_lock_interruptible(&dev->mutex);if (error)return error;/* 将handle的d_node,链接到其相关的input_dev的h_list链表中 */if (handler->filter)list_add_rcu(&handle->d_node, &dev->h_list);elselist_add_tail_rcu(&handle->d_node, &dev->h_list);/* 释放锁 */mutex_unlock(&dev->mutex);/* 将handle的h_node,链接到其相关的input_handler的h_list链表中 */list_add_tail_rcu(&handle->h_node, &handler->h_list);if (handler->start)handler->start(handle);return 0;
}
EXPORT_SYMBOL(input_register_handle);
配对的input设备与input事件处理器实例
当 input设备注册Input_register_device 和 一个新的input事件注册input_register_handler 匹配上,都会调用回调函数handler->connect(handler, dev, id)
以事件处理器evdev为例:
/* drivers/input/evdev.c */
static int evdev_connect(struct input_handler *handler, struct input_dev *dev,const struct input_device_id *id)
{struct evdev *evdev;int minor;int dev_no;int error;/* 获取次设备号,从evdev_table中找到一个未使用的最小的数组项,最大值32 */minor = input_get_new_minor(EVDEV_MINOR_BASE, EVDEV_MINORS, true);if (minor < 0) {error = minor;pr_err("failed to reserve new minor: %d\n", error);return error;}/* 分配空间 */evdev = kzalloc(sizeof(struct evdev), GFP_KERNEL);if (!evdev) {error = -ENOMEM;goto err_free_minor;}/* 初始化client_list链表头,代表多少应用读写这个设备 */INIT_LIST_HEAD(&evdev->client_list); spin_lock_init(&evdev->client_lock); /* 加锁 */ mutex_init(&evdev->mutex); /* */init_waitqueue_head(&evdev->wait); /* 初始化等待队列,当evdev没有数据可读时,就 在 该队列上睡眠 */evdev->exist = true; /* 设备存在 */dev_no = minor;if (dev_no < EVDEV_MINOR_BASE + EVDEV_MINORS)dev_no -= EVDEV_MINOR_BASE;dev_set_name(&evdev->dev, "event%d", dev_no); /* 设置设备名为eventX */evdev->handle.dev = input_get_device(dev); /* 获取设备 */evdev->handle.name = dev_name(&evdev->dev); /* 设备名称 */evdev->handle.handler = handler; /* handler绑定 */ evdev->handle.private = evdev; /* evdev数据指向 */evdev->dev.devt = MKDEV(INPUT_MAJOR, minor); /* sysfs下的设备号 */evdev->dev.class = &input_class; /* 将input_class作为设备类 */evdev->dev.parent = &dev->dev; /* input_dev作为evdev的父设备 */evdev->dev.release = evdev_free; /* 释放函数 */device_initialize(&evdev->dev); /* 初始化设备 *//* 注册一个handle处理事件 */error = input_register_handle(&evdev->handle); if (error)goto err_free_evdev;cdev_init(&evdev->cdev, &evdev_fops); /* 字符设备初始化 */error = cdev_device_add(&evdev->cdev, &evdev->dev); /* 添加字符设备 */if (error)goto err_cleanup_evdev;return 0;err_cleanup_evdev:evdev_cleanup(evdev);
err_unregister_handle:input_unregister_handle(&evdev->handle);
err_free_evdev:put_device(&evdev->dev);
err_free_minor:input_free_minor(minor);return error;
}
(1)保存驱动设备名字,event0是表示input子系统,驱动名字就由event1、event2…递增
(2)保存驱动设备的主次设备号,其中主设备号INPUT_MAJOR=13,次设备号=EVSEV_MINOR_BASE+驱动程序本身设备号。
(3)会在/sys/class/input类下创建驱动设备event%d,比如键盘驱动event1
(4)最终进入input_register_handler()函数来注册handle。
input核心层对驱动层和事件层之间的框架建立流程图
input核心层对驱动层和事件层之间的框架建立流程图