当前位置:   article > 正文

Linux kernel: USB driver编写入门(三)_usb_class_driver

usb_class_driver

Linux kernel: USB driver编写入门(二)中写到struct usb_class_driver的变量usb_cd需要赋值。故添加代码如下:

  1. #define MYUSB_MINOR_BASE 250
  2. static int myusb_open(struct inode *inode, struct file *file)
  3. {
  4. pr_info("%s\n", __func__);
  5. return 0;
  6. }
  7. static int myusb_release(struct inode *inode, struct file *file)
  8. {
  9. pr_info("%s\n", __func__);
  10. return 0;
  11. }
  12. static ssize_t myusb_write(struct file *file, const char *user_buffer,
  13. size_t count, loff_t *ppos)
  14. {
  15. pr_info("%s\n", __func__);
  16. return count;
  17. }
  18. static ssize_t myusb_read(struct file *file, char *buffer, size_t count,
  19. loff_t *ppos)
  20. {
  21. pr_info("%s\n", __func__);
  22. return 0;
  23. }
  24. static const struct file_operations myusb_fops = {
  25. .owner = THIS_MODULE,
  26. .open = myusb_open,
  27. .release = myusb_release,
  28. .read = myusb_read,
  29. .write = myusb_write,
  30. };
  31. static struct usb_class_driver usb_cd = {
  32. .name = "ActionsUSB%d",
  33. .fops = &myusb_fops,
  34. .minor_base = MYUSB_MINOR_BASE,
  35. };

因为C语言编译器是按照源文件的代码顺序来编译的,所以引用的变量/常量或者函数需要先声明或者定义。

按照好理解的逻辑,我们先来看最下面,usb_cd的定义。我们先复习一下在(二)中提到的struct usb_class_driver的定义:

  1. struct usb_class_driver {
  2.     char *name;
  3.     char *(*devnode)(struct device *dev, umode_t *mode);
  4.     const struct file_operations *fops;
  5.     int minor_base;
  6. };

这里devnode函数指针先不给它赋值,因为根据(二),在usb_register_dev函数中只用到了struct usb_class_driver中的name, fops, minor_base。其中,最重要的就是fops,它定义了文件操作,因为Linux把几乎所有设备都当成file来看待。我这里的文件操作也很简单,就是把操作对应的函数名称在kernel消息里打印出来。

然后,在当前目录下make

$ make
make -C /lib/modules/5.19.0-rc3+/build M=/home/minipc/linux_usb_driver modules
make[1]: Entering directory '/home/minipc/stable_rc/linux-5.19.0'
  CC [M]  /home/minipc/linux_usb_driver/usb_test_drv.o
  MODPOST /home/minipc/linux_usb_driver/Module.symvers
  CC [M]  /home/minipc/linux_usb_driver/usb_test_drv.mod.o
  LD [M]  /home/minipc/linux_usb_driver/usb_test_drv.ko
  BTF [M] /home/minipc/linux_usb_driver/usb_test_drv.ko
make[1]: Leaving directory '/home/minipc/stable_rc/linux-5.19.0'

理想状况下应该是如上,没有error,没有warning。如果有问题,请按照(一)的内容排查。

如果USB设备已经连接Linux主机,请拔出,并重启Linux主机。如果在Linux主机本次运行期间曾经插上过USB设备,也需要在没有连接USB设备的状态下重启主机。

然后,load这个模块,用lsmod| grep usb和dmesg|tail查看是否load成功。

然后插上USB设备。运行dmesg。

$ dmesg| tail
[  400.931170] usb 2-4: New USB device found, idVendor=10d6, idProduct=1101, bcdDevice= 1.00
[  400.931183] usb 2-4: New USB device strings: Mfr=1, Product=2, SerialNumber=3
[  400.931189] usb 2-4: Product: USB CARDREADER
[  400.931194] usb 2-4: Manufacturer: ACTIONS
[  400.931198] usb 2-4: SerialNumber: ㉕捤稰眷㕳愳㤷湲
[  400.933716] USB info 0 now probed: (10d6:1101)
[  400.933726] ID->bNumEndpoints:02
[  400.933729] ID->bInterfaceClass:08
[  400.933874] Minor number = 1
[  401.011611] usbcore: registered new interface driver usb-storage

靠近底部的四行正是usb_drv_probe的输出kernel消息:

  1. printk(KERN_INFO "USB info %d now probed: (%04x:%04x)\n", interface_desc->desc.bInterfaceNumber, id->idVendor, id->idProduct);
  2. printk(KERN_INFO "ID->bNumEndpoints:%02x\n", interface_desc->desc.bNumEndpoints);
  3. printk(KERN_INFO "ID->bInterfaceClass:%02x\n", interface_desc->desc.bInterfaceClass);
  4. ret = usb_register_dev(interface,&usb_cd);
  5. if(ret)
  6. {
  7. printk(KERN_INFO "usb_register_dev erro: %d\n", ret);
  8. }
  9. else
  10. {
  11. printk(KERN_INFO "Minor number = %d\n", interface->minor);
  12. }

 从log消息来看,usb_register_dev调用成功。$ls /dev/ 可以看到ActionsUSB1, 恰恰是struct usb_class_driver中的name。

  1. static struct usb_class_driver usb_cd = {
  2. .name = "ActionsUSB%d",
  3. .fops = &myusb_fops,
  4. .minor_base = MYUSB_MINOR_BASE,
  5. };

向该设备写数据,如同一个文件write操作。

$ sudo sh -c "echo hello>/dev/ActionsUSB1"

成功后应该立即返回。然后再查看dmesg消息。

$ dmesg | tail
[  400.931194] usb 2-4: Manufacturer: ACTIONS
[  400.931198] usb 2-4: SerialNumber: ㉕捤稰眷㕳愳㤷湲
[  400.933716] USB info 0 now probed: (10d6:1101)
[  400.933726] ID->bNumEndpoints:02
[  400.933729] ID->bInterfaceClass:08
[  400.933874] Minor number = 1
[  401.011611] usbcore: registered new interface driver usb-storage
[ 1261.675975] myusb_open
[ 1261.675995] myusb_write
[ 1261.675998] myusb_release

最后三行是这个写操作,open, write , release,输出Log消息恰恰是对应文件操作函数的名称。

向这个设备读数据,如同一个文件读操作

$ sudo cat /dev/ActionsUSB1

 再查看dmesg消息:

$ dmesg | tail
[  400.933726] ID->bNumEndpoints:02
[  400.933729] ID->bInterfaceClass:08
[  400.933874] Minor number = 1
[  401.011611] usbcore: registered new interface driver usb-storage
[ 1261.675975] myusb_open
[ 1261.675995] myusb_write
[ 1261.675998] myusb_release
[ 1893.611703] myusb_open
[ 1893.611721] myusb_read
[ 1893.611736] myusb_release

同样是输出最后三行消息。

到这里,该usb driver已经实现了从Linux kernel层到user space的一个抽象。可以说具备了基本的驱动框架。

最后,把该USB设备从Linux主机中拔出。查看dmesg消息:

 $ dmesg | tail
[  400.933874] Minor number = 1
[  401.011611] usbcore: registered new interface driver usb-storage
[ 1261.675975] myusb_open
[ 1261.675995] myusb_write
[ 1261.675998] myusb_release
[ 1893.611703] myusb_open
[ 1893.611721] myusb_read
[ 1893.611736] myusb_release
[ 2146.812223] usb 2-4: USB disconnect, device number 4
[ 2146.812363] Disconneced and Release the MINOR number 1

最后一行消息 Disconneced and Release the MINOR number 1 恰恰是usb_drv_disconnect函数中的打印消息:

  1. static void usb_drv_disconnect(struct usb_interface *interface)
  2. {
  3. printk(KERN_INFO "Disconneced and Release the MINOR number %d\n", interface->minor);
  4. usb_deregister_dev(interface, &usb_cd);
  5. }

至此,整个usb驱动开发的最基础步骤已经完成。

本文内容由网友自发贡献,转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/246879
推荐阅读
相关标签
  

闽ICP备14008679号