📄 atkbd.c
字号:
static int atkbd_connect(struct serio *serio, struct serio_driver *drv){ struct atkbd *atkbd; struct input_dev *dev; int err = -ENOMEM; atkbd = kzalloc(sizeof(struct atkbd), GFP_KERNEL); dev = input_allocate_device(); if (!atkbd || !dev) goto fail1; atkbd->dev = dev; ps2_init(&atkbd->ps2dev, serio); INIT_WORK(&atkbd->event_work, atkbd_event_work); mutex_init(&atkbd->event_mutex); switch (serio->id.type) { case SERIO_8042_XL: atkbd->translated = 1; case SERIO_8042: if (serio->write) atkbd->write = 1; break; } atkbd->softraw = atkbd_softraw; atkbd->softrepeat = atkbd_softrepeat; atkbd->scroll = atkbd_scroll; if (atkbd->softrepeat) atkbd->softraw = 1; serio_set_drvdata(serio, atkbd); err = serio_open(serio, drv); if (err) goto fail2; if (atkbd->write) { if (atkbd_probe(atkbd)) { err = -ENODEV; goto fail3; } atkbd->set = atkbd_select_set(atkbd, atkbd_set, atkbd_extra); atkbd_activate(atkbd); } else { atkbd->set = 2; atkbd->id = 0xab00; } atkbd_set_keycode_table(atkbd); atkbd_set_device_attrs(atkbd); err = sysfs_create_group(&serio->dev.kobj, &atkbd_attribute_group); if (err) goto fail3; atkbd_enable(atkbd); err = input_register_device(atkbd->dev); if (err) goto fail4; return 0; fail4: sysfs_remove_group(&serio->dev.kobj, &atkbd_attribute_group); fail3: serio_close(serio); fail2: serio_set_drvdata(serio, NULL); fail1: input_free_device(dev); kfree(atkbd); return err;}/* * atkbd_reconnect() tries to restore keyboard into a sane state and is * most likely called on resume. * 键盘重新连接 */static int atkbd_reconnect(struct serio *serio){ struct atkbd *atkbd = serio_get_drvdata(serio); struct serio_driver *drv = serio->drv; if (!atkbd || !drv) { printk(KERN_DEBUG "atkbd: reconnect request, but serio is disconnected, ignoring...\n"); return -1; } atkbd_disable(atkbd); if (atkbd->write) { if (atkbd_probe(atkbd)) return -1; if (atkbd->set != atkbd_select_set(atkbd, atkbd->set, atkbd->extra)) return -1; atkbd_activate(atkbd);/* * Restore repeat rate and LEDs (that were reset by atkbd_activate) * to pre-resume state */ if (!atkbd->softrepeat) atkbd_set_repeat_rate(atkbd); atkbd_set_leds(atkbd); } atkbd_enable(atkbd); return 0;}static struct serio_device_id atkbd_serio_ids[] = { { .type = SERIO_8042, .proto = SERIO_ANY, .id = SERIO_ANY, .extra = SERIO_ANY, }, { .type = SERIO_8042_XL, .proto = SERIO_ANY, .id = SERIO_ANY, .extra = SERIO_ANY, }, { .type = SERIO_RS232, .proto = SERIO_PS2SER, .id = SERIO_ANY, .extra = SERIO_ANY, }, { 0 }};MODULE_DEVICE_TABLE(serio, atkbd_serio_ids);static struct serio_driver atkbd_drv = { .driver = { .name = "atkbd", }, .description = DRIVER_DESC, .id_table = atkbd_serio_ids, .interrupt = atkbd_interrupt, .connect = atkbd_connect, .reconnect = atkbd_reconnect, .disconnect = atkbd_disconnect, .cleanup = atkbd_cleanup,};static ssize_t atkbd_attr_show_helper(struct device *dev, char *buf, ssize_t (*handler)(struct atkbd *, char *)){ struct serio *serio = to_serio_port(dev); int retval; retval = serio_pin_driver(serio); if (retval) return retval; if (serio->drv != &atkbd_drv) { retval = -ENODEV; goto out; } retval = handler((struct atkbd *)serio_get_drvdata(serio), buf);out: serio_unpin_driver(serio); return retval;}static ssize_t atkbd_attr_set_helper(struct device *dev, const char *buf, size_t count, ssize_t (*handler)(struct atkbd *, const char *, size_t)){ struct serio *serio = to_serio_port(dev); struct atkbd *atkbd; int retval; retval = serio_pin_driver(serio); if (retval) return retval; if (serio->drv != &atkbd_drv) { retval = -ENODEV; goto out; } atkbd = serio_get_drvdata(serio); atkbd_disable(atkbd); retval = handler(atkbd, buf, count); atkbd_enable(atkbd);out: serio_unpin_driver(serio); return retval;}static ssize_t atkbd_show_extra(struct atkbd *atkbd, char *buf){ return sprintf(buf, "%d\n", atkbd->extra ? 1 : 0);}static ssize_t atkbd_set_extra(struct atkbd *atkbd, const char *buf, size_t count){ struct input_dev *old_dev, *new_dev; unsigned long value; char *rest; int err; unsigned char old_extra, old_set; if (!atkbd->write) return -EIO; value = simple_strtoul(buf, &rest, 10); if (*rest || value > 1) return -EINVAL; if (atkbd->extra != value) { /* * Since device's properties will change we need to * unregister old device. But allocate and register * new one first to make sure we have it. * 关于新设备代替旧设备 */ old_dev = atkbd->dev; old_extra = atkbd->extra; old_set = atkbd->set; new_dev = input_allocate_device(); if (!new_dev) return -ENOMEM; atkbd->dev = new_dev; atkbd->set = atkbd_select_set(atkbd, atkbd->set, value); atkbd_activate(atkbd); atkbd_set_keycode_table(atkbd); atkbd_set_device_attrs(atkbd); err = input_register_device(atkbd->dev); if (err) { input_free_device(new_dev); atkbd->dev = old_dev; atkbd->set = atkbd_select_set(atkbd, old_set, old_extra); atkbd_set_keycode_table(atkbd); atkbd_set_device_attrs(atkbd); return err; } input_unregister_device(old_dev); } return count;}static ssize_t atkbd_show_scroll(struct atkbd *atkbd, char *buf){ return sprintf(buf, "%d\n", atkbd->scroll ? 1 : 0);}static ssize_t atkbd_set_scroll(struct atkbd *atkbd, const char *buf, size_t count){ struct input_dev *old_dev, *new_dev; unsigned long value; char *rest; int err; unsigned char old_scroll; value = simple_strtoul(buf, &rest, 10); if (*rest || value > 1) return -EINVAL; if (atkbd->scroll != value) { old_dev = atkbd->dev; old_scroll = atkbd->scroll; new_dev = input_allocate_device(); if (!new_dev) return -ENOMEM; atkbd->dev = new_dev; atkbd->scroll = value; atkbd_set_keycode_table(atkbd); atkbd_set_device_attrs(atkbd); err = input_register_device(atkbd->dev); if (err) { input_free_device(new_dev); atkbd->scroll = old_scroll; atkbd->dev = old_dev; atkbd_set_keycode_table(atkbd); atkbd_set_device_attrs(atkbd); return err; } input_unregister_device(old_dev); } return count;}static ssize_t atkbd_show_set(struct atkbd *atkbd, char *buf){ return sprintf(buf, "%d\n", atkbd->set);}static ssize_t atkbd_set_set(struct atkbd *atkbd, const char *buf, size_t count){ struct input_dev *old_dev, *new_dev; unsigned long value; char *rest; int err; unsigned char old_set, old_extra; if (!atkbd->write) return -EIO; value = simple_strtoul(buf, &rest, 10); if (*rest || (value != 2 && value != 3)) return -EINVAL; if (atkbd->set != value) { old_dev = atkbd->dev; old_extra = atkbd->extra; old_set = atkbd->set; new_dev = input_allocate_device(); if (!new_dev) return -ENOMEM; atkbd->dev = new_dev; atkbd->set = atkbd_select_set(atkbd, value, atkbd->extra); atkbd_activate(atkbd); atkbd_set_keycode_table(atkbd); atkbd_set_device_attrs(atkbd); err = input_register_device(atkbd->dev); if (err) { input_free_device(new_dev); atkbd->dev = old_dev; atkbd->set = atkbd_select_set(atkbd, old_set, old_extra); atkbd_set_keycode_table(atkbd); atkbd_set_device_attrs(atkbd); return err; } input_unregister_device(old_dev); } return count;}static ssize_t atkbd_show_softrepeat(struct atkbd *atkbd, char *buf){ return sprintf(buf, "%d\n", atkbd->softrepeat ? 1 : 0);}static ssize_t atkbd_set_softrepeat(struct atkbd *atkbd, const char *buf, size_t count){ struct input_dev *old_dev, *new_dev; unsigned long value; char *rest; int err; unsigned char old_softrepeat, old_softraw; if (!atkbd->write) return -EIO; value = simple_strtoul(buf, &rest, 10); if (*rest || value > 1) return -EINVAL; if (atkbd->softrepeat != value) { old_dev = atkbd->dev; old_softrepeat = atkbd->softrepeat; old_softraw = atkbd->softraw; new_dev = input_allocate_device(); if (!new_dev) return -ENOMEM; atkbd->dev = new_dev; atkbd->softrepeat = value; if (atkbd->softrepeat) atkbd->softraw = 1; atkbd_set_device_attrs(atkbd); err = input_register_device(atkbd->dev); if (err) { input_free_device(new_dev); atkbd->dev = old_dev; atkbd->softrepeat = old_softrepeat; atkbd->softraw = old_softraw; atkbd_set_device_attrs(atkbd); return err; } input_unregister_device(old_dev); } return count;}static ssize_t atkbd_show_softraw(struct atkbd *atkbd, char *buf){ return sprintf(buf, "%d\n", atkbd->softraw ? 1 : 0);}static ssize_t atkbd_set_softraw(struct atkbd *atkbd, const char *buf, size_t count){ struct input_dev *old_dev, *new_dev; unsigned long value; char *rest; int err; unsigned char old_softraw; value = simple_strtoul(buf, &rest, 10); if (*rest || value > 1) return -EINVAL; if (atkbd->softraw != value) { old_dev = atkbd->dev; old_softraw = atkbd->softraw; new_dev = input_allocate_device(); if (!new_dev) return -ENOMEM; atkbd->dev = new_dev; atkbd->softraw = value; atkbd_set_device_attrs(atkbd); err = input_register_device(atkbd->dev); if (err) { input_free_device(new_dev); atkbd->dev = old_dev; atkbd->softraw = old_softraw; atkbd_set_device_attrs(atkbd); return err; } input_unregister_device(old_dev); } return count;}static ssize_t atkbd_show_err_count(struct atkbd *atkbd, char *buf){ return sprintf(buf, "%lu\n", atkbd->err_count);}static int __init atkbd_init(void){ return serio_register_driver(&atkbd_drv);}static void __exit atkbd_exit(void){ serio_unregister_driver(&atkbd_drv);}module_init(atkbd_init);module_exit(atkbd_exit);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -