https://blog.csdn.net/xiaodingqq/article/details/80415082

一、内容介绍

在驱动层添加一个虚拟键盘驱动程序(cc-vkeyboard),该驱动是一个字符设备+输入子系统的组合,主要功能:

  1. 接收write操作用户发送的内容
  2. 以键盘按键方式写入用户子系统

二、驱动层实现

vkeyboard.c

  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. #include <linux/uaccess.h>
  17. #define DRVNAME "cc-vkeyboard"
  18. #define DEVNAME "cc_vkeyboard"
  19. int major = 0;
  20. struct keyboard_event
  21. {
  22. int press;
  23. int key;
  24. };
  25. static struct input_dev *vkeyboard_input_dev = NULL;
  26. static struct class *fir_driv_class;
  27. static int vkeyboard_open(struct inode* inode, struct file* filp)
  28. {
  29. return 0;
  30. }
  31. static int vkeyboard_release(struct inode* inode, struct file* filp)
  32. {
  33. return 0;
  34. }
  35. static ssize_t vkeyboard_read(struct file* filp, char __user *buf, size_t count, loff_t* f_pos)
  36. {
  37. printk(KERN_INFO"%s\n", __func__);
  38. return count;
  39. }
  40. static ssize_t vkeyboard_write(struct file* filp, const char __user * buf, size_t count, loff_t* f_pos)
  41. {
  42. int ret = 0;
  43. struct keyboard_event event;
  44. while(ret < count)
  45. {
  46. if(copy_from_user(&event, buf + ret, sizeof(event)))
  47. {
  48. return -EFAULT;
  49. }
  50. ret += sizeof(event);
  51. input_event(vkeyboard_input_dev, EV_MSC, MSC_SCAN, event.key);
  52. input_report_key(vkeyboard_input_dev, event.key, event.press);
  53. input_sync (vkeyboard_input_dev);
  54. printk(KERN_INFO"%s p=%d key=%d with scan code\n", __func__, event.press, event.key);
  55. }
  56. return ret;
  57. }
  58. static struct file_operations vkeyboard_fops =
  59. {
  60. .owner = THIS_MODULE,
  61. .open = vkeyboard_open,
  62. .release = vkeyboard_release,
  63. .read = vkeyboard_read,
  64. .write = vkeyboard_write,
  65. };
  66. static int vkeyboard_input_dev_open(struct input_dev* idev)
  67. {
  68. printk(KERN_INFO"%s\n", __func__);
  69. return 0;
  70. }
  71. static void vkeyboard_input_dev_close(struct input_dev* idev)
  72. {
  73. printk(KERN_INFO"%s\n", __func__);
  74. return;
  75. }
  76. static int vkeyboard_input_dev_setup(void)
  77. {
  78. int ret = 0;
  79. vkeyboard_input_dev = input_allocate_device();
  80. if(vkeyboard_input_dev == NULL)
  81. {
  82. return -ENOMEM;
  83. }
  84. set_bit(EV_KEY, vkeyboard_input_dev->evbit);
  85. set_bit(EV_REP, vkeyboard_input_dev->evbit);//设置可以产生重复类事件
  86. //KEY_MAX是多个值,不能使用set_bit,可以使用bitmap_fill
  87. //set_bit(KEY_MAX, vkeyboard_input_dev->keybit);
  88. bitmap_fill(vkeyboard_input_dev->keybit, KEY_MAX);
  89. vkeyboard_input_dev->name = "vkeyboard";
  90. vkeyboard_input_dev->phys = "vkeyboard/input0";
  91. vkeyboard_input_dev->open = vkeyboard_input_dev_open;
  92. vkeyboard_input_dev->close = vkeyboard_input_dev_close;
  93. #if 0
  94. //bitmap_fill(vkeyboard_input_dev->relbit, REL_MAX);
  95. //bitmap_fill(vkeyboard_input_dev->absbit, ABS_MAX);
  96. int i = 0;
  97. for(i = 32; i < KEY_MAX; i++)
  98. {
  99. input_set_capability(vkeyboard_input_dev, EV_KEY, i);
  100. }
  101. __set_bit(EV_KEY, vkeyboard_input_dev->evbit);
  102. #endif
  103. ret = input_register_device(vkeyboard_input_dev);
  104. return ret;
  105. }
  106. static int __init vkeyboard_init(void)
  107. {
  108. //注册字符设备驱动
  109. major = register_chrdev(0, DRVNAME, &vkeyboard_fops);
  110. //注册输入子系统设备
  111. vkeyboard_input_dev_setup();
  112. //新建类
  113. fir_driv_class = class_create(THIS_MODULE, DRVNAME);
  114. if(IS_ERR(fir_driv_class))
  115. {
  116. return PTR_ERR(fir_driv_class);
  117. }
  118. //创建设备 /dev/fir_dev
  119. device_create(fir_driv_class,NULL,MKDEV(major, 0),NULL,DEVNAME);
  120. printk(KERN_INFO"%s,%d\n", __func__,major);
  121. return 0;
  122. }
  123. static void __exit vkeyboard_cleanup(void)
  124. {
  125. //删除设备结点
  126. device_destroy(fir_driv_class,MKDEV(major, 0));
  127. class_destroy(fir_driv_class);
  128. input_unregister_device(vkeyboard_input_dev);
  129. input_free_device(vkeyboard_input_dev);
  130. unregister_chrdev(major, DRVNAME);
  131. return;
  132. }
  133. module_init(vkeyboard_init);
  134. module_exit(vkeyboard_cleanup);
  135. module_param(major, int, 0);
  136. MODULE_AUTHOR("zhangsan");
  137. MODULE_LICENSE("GPL");

makefile

ifeq ($(KERNELRELEASE),)
    KERNELDIR := /home/pi/linux-rpi-4.19.y-x86
    KERNELDIR := /lib/modules/`uname -r`/build
    PWD := $(shell pwd)
    ARCH = x86
    CROSS_COMPILE =
modules:
    $(MAKE) -C $(KERNELDIR) M=$(PWD) modules ARCH=$(ARCH) CROSS_COMPILE=$(CROSS_COMPILE)
modules_install:
    $(MAKE) -C $(KERNELDIR) M=$(PWD) modules_install ARCH=$(ARCH) CROSS_COMPILE=$(CROSS_COMPILE)
clean:
    rm -rf *.o *~core.depend .*.cmd *.mod.c *.tmp_version *.order *.symvers
else
    obj-m := vkeyboard.o
endif

将驱动添加到内核:

sudo insmod vkeyboard.ko

结果如下:

  1. 自动生成设备文件: /dev/cc_vkeyboard
  2. 自动生成输入子系统文件: /dev/input/eventx

三、驱动测试

#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>

#include <linux/input.h>

#define DEVICE_NAME "/dev/cc_vkeyboard"

struct keyboard_event
{
    int press;
    int key;
};

int main(void)
{
    struct keyboard_event event;
    int fd, i = 0, j = 50;

    event.key = KEY_A;
    fd=open(DEVICE_NAME,O_RDWR|O_NDELAY);
    if(fd<0)
    {
        perror("open error");
        exit(-1);
    }
    while(j--)
    {
        event.press=1;
        write(fd,&event,sizeof(event));
        event.press=0;
        write(fd,&event,sizeof(event));
        sleep(1);

        printf("%d \n",i++);
    }
    close(fd);
    return 0;
}