⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 atkbd.c

📁 at91_ps2_驱动~at91sam9261的IO口模拟PS2键盘驱动程序设计——最新
💻 C
📖 第 1 页 / 共 3 页
字号:
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 + -