如何在Linux下模拟键盘输入 利用驱动程序实现

参考:http://blog.csdn.net/absurd/archive/2009/09/13/4549514.aspx

驱动程序:

  1. #include <linux/module.h>
  2. #include <linux/moduleparam.h>
  3. #include <linux/init.h>
  4. #include <linux/kernel.h>
  5. #include <linux/slab.h>
  6. #include <linux/fs.h>
  7. #include <linux/errno.h>
  8. #include <linux/types.h>
  9. #include <linux/proc_fs.h>
  10. #include <linux/fcntl.h>
  11. #include <linux/aio.h>
  12. #include <asm/uaccess.h>
  13. #include <linux/ioctl.h>
  14. #include <linux/cdev.h>
  15. #include <linux/input.h>
  16.  
  17. int vkeyboard_major = 201;
  18. module_param(vkeyboard_major, int, 0);
  19. MODULE_AUTHOR("Li XianJing <xianjimli@hotmail.com>");
  20. MODULE_LICENSE("DUAL BSD/GPL");
  21. static struct input_dev* vkeyboard_idev = NULL;
  22. static int vkeyboard_input_dev_setup(void);
  23. static int vkeyboard_open(struct inode* inode, struct file* filp)
  24. {
  25.     return 0;
  26. }
  27. static int vkeyboard_release(struct inode* inode, struct file* filp)
  28. {
  29.     return 0;
  30. }
  31. static ssize_t vkeyboard_read(struct file* filp, char __user *buf, size_t count, loff_t* f_pos)
  32. {
  33.     printk(KERN_INFO"%s\n", __func__);
  34.     return count;
  35. }
  36. struct keyboard_event
  37. {
  38.     int press;
  39.     int key;
  40. };
  41. static ssize_t vkeyboard_write(struct file* filp, const char __user * buf, size_t count, loff_t* f_pos)
  42. {
  43.     int ret = 0;
  44.     struct keyboard_event event;
  45.     while(ret < count)
  46.     {
  47.         if(copy_from_user(&event, buf + ret, sizeof(event)))
  48.         {
  49.             return -EFAULT;
  50.         }
  51.         ret += sizeof(event);
  52.         input_event(vkeyboard_idev, EV_MSC, MSC_SCAN, event.key);
  53.         input_report_key(vkeyboard_idev, event.key, event.press);
  54.         input_sync (vkeyboard_idev);
  55.         printk(KERN_INFO"%s p=%d key=%d with scan code\n", __func__, event.press, event.key);
  56.     }
  57.     return ret;
  58. }
  59. static struct file_operations vkeyboard_fops =
  60. {
  61.     .owner = THIS_MODULE,
  62.     .open = vkeyboard_open,
  63.     .release = vkeyboard_release,
  64.     .read = vkeyboard_read,
  65.     .write = vkeyboard_write,
  66. };
  67. static int __init vkeyboard_init(void)
  68. {
  69.     int result = register_chrdev(vkeyboard_major, "vkeyboard", &vkeyboard_fops);
  70.     vkeyboard_input_dev_setup();
  71.     return result;
  72. }
  73. static void __exit vkeyboard_cleanup(void)
  74. {
  75.     input_unregister_device(vkeyboard_idev);
  76.     unregister_chrdev(vkeyboard_major, "vkeyboard");
  77.     return;
  78. }
  79. module_init(vkeyboard_init);
  80. module_exit(vkeyboard_cleanup);
  81. static int vkeyboard_input_dev_open(struct input_dev* idev)
  82. {
  83.     printk(KERN_INFO"%s\n", __func__);
  84.     return 0;
  85. }
  86. static void vkeyboard_input_dev_close(struct input_dev* idev)
  87. {
  88.     printk(KERN_INFO"%s\n", __func__);
  89.     return;
  90. }
  91. static int vkeyboard_input_dev_setup(void)
  92. {
  93.     int ret = 0;
  94.     int i = 0;
  95.     vkeyboard_idev = input_allocate_device();
  96.     if(vkeyboard_idev == NULL)
  97.     {
  98.         return -ENOMEM;
  99.     }
  100.     vkeyboard_idev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS) | BIT_MASK(EV_MSC);
  101.     vkeyboard_idev->mscbit[0] = BIT_MASK(MSC_SCAN) | BIT_MASK(MSC_SERIAL) | BIT_MASK(MSC_RAW);
  102.     bitmap_fill(vkeyboard_idev->keybit, KEY_MAX);
  103.     bitmap_fill(vkeyboard_idev->relbit, REL_MAX);
  104.     bitmap_fill(vkeyboard_idev->absbit, ABS_MAX);
  105.     vkeyboard_idev->name = "vkeyboard";
  106.     vkeyboard_idev->phys = "vkeyboard/input0";
  107.     vkeyboard_idev->open = vkeyboard_input_dev_open;
  108.     vkeyboard_idev->close = vkeyboard_input_dev_close;
  109.     for(i = 32; i < KEY_MAX; i++)
  110.     {
  111.         input_set_capability(vkeyboard_idev, EV_KEY, i);
  112.     }
  113.     __set_bit(EV_KEY, vkeyboard_idev->evbit);
  114.     ret = input_register_device(vkeyboard_idev);
  115.     return ret;
  116. }


Makefile:

  1. obj-m := vkeyboard.o
  2. KDIR := /lib/modules/$(shell uname -r)/build
  3. PWD := $(shell pwd)
  4. default:
  5. $(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules
  6. install:
  7. $(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules_install
  8. clean:
  9. rm -f *.mod.c *.ko *.o Module.markers modules.order Module.symvers



线创建节点:
sudo mknod /dev/vkeyboard c 201 0

用户空间测试程序:

    1. #include <fcntl.h>
    2. #include <stdlib.h>
    3. #include <stdio.h>
    4. #include <errno.h>
    5. #include <linux/input.h>
    6. #define DEVICE_NAME "/dev/vkeyboard"
    7. struct keyboard_event
    8. {
    9.     int press;
    10.     int key;
    11. };
    12. int main(void)
    13. {
    14.     struct keyboard_event event;
    15.     int fd;
    16.     int i=0, j=50;
    17.     event.key=KEY_UP;
    18.     fd=open(DEVICE_NAME,O_RDWR|O_NDELAY);
    19.     if(fd<0)
    20.     {
    21.         perror("open error");
    22.         exit(-1);
    23.     }
    24.     while(j--)
    25.     {
    26.         event.press=1;
    27.         write(fd,&event,sizeof(event));
    28.         event.press=0;
    29.         write(fd,&event,sizeof(event));
    30.         sleep(1);
    31.         printf("%d \n",i++);
    32.     }
    33.     close(fd);
    34.     return 0;
    35. }
posted @ 2014-11-23 01:45  alxe_yu  阅读(745)  评论(0)    收藏  举报