当前位置:   article > 正文

Linux--V4L2摄像头驱动框架及UVC浅析

Linux--V4L2摄像头驱动框架及UVC浅析

 一、前言

对于一个usb摄像头,它的内核驱动源码位于/drivers/media/usb/uvc/

核心层:V4L2_dev.c文件

硬件相关层: uvc_driver.c文件

本篇记录基于对6.8.8.8内核下vivid-core.c文件(虚拟视频驱动程序)的分析,梳理Linux系统中vedio视频设备的驱动框架。

二、源码分析

1、入口函数

  1. module_init(vivid_init);
  2. vivid_init
  3. //注册vivid设备和驱动
  4. platform_device_register(&vivid_pdev);
  5. platform_driver_register(&vivid_pdrv);

通过入口函数将平台设备和平台驱动结构体注册到内核中,关于具体的设备和驱动注册先关内容可以看Linux--平台设备、平台驱动的注册源码分析-CSDN博客这篇。

在代码中,vivid_pdrv 平台驱动没有 of_match_table 字段,因为该驱动程序不依赖设备树进行匹配和初始化。如果是真实的硬件驱动程序需要支持设备树匹配,则需要定义 of_match_table 字段,并提供一个设备树兼容字符串数组,也就是compatible属性值。

2、probe()函数

vivid_init()里分别注册了vivid_pdevvivid_pdrv,注册后,当他们匹配时,内核会自动则会调用该驱动程序probe()初始化该设备。在probe()里面主要进行初始化、注册等相关流程。

  1. vivid_probe
  2. 1) vivid_create_instance /* 创建实例 */
  3. dev = kzalloc(sizeof(*vivid_dev), GFP_KERNEL);/* 分配video_devicede */
  4. v4l2_device_register /* 初始化v4l2_device */
  5. 2) static int vivid_create_devnodes(struct platform_device *pdev,
  6. struct vivid_dev *dev, int inst,
  7. unsigned int cec_tx_bus_cnt,
  8. v4l2_std_id tvnorms_cap,
  9. v4l2_std_id tvnorms_out,
  10. unsigned in_type_counter[4],
  11. unsigned out_type_counter[4])
  12. {
  13. /* 设置video_device */
  14. 1.vfd->fops = &vivid_fops; /*也就是在probe里的这一步实现设备操作和驱动操作的绑定*/
  15. vfd->ioctl_ops = &vivid_ioctl_ops;
  16. vfd->release = video_device_release_empty;
  17. 2.vfd->v4l2_dev = &dev->v4l2_dev;
  18. 3.设置"ctrl属性"(用于APP的ioctl)
  19. v4l2_ctrl_handler_setup(&dev->ctrl_hdl_vid_cap);
  20. v4l2_ctrl_handler_setup(&dev->ctrl_hdl_vid_out);
  21. ...
  22. /*注册vedio_device*/ /*分配主次设备号*/
  23. video_register_device(video_device, type:VFL_TYPE_GRABBER, nr)
  24. __video_register_device
  25. vdev->cdev = cdev_alloc();
  26. vdev->cdev->ops = &v4l2_fops;
  27. video_devices[vdev->minor] = vdev;//将video_device放入全局数组中
  28. ret = cdev_add(vdev->cdev, MKDEV(VIDEO_MAJOR, vdev->minor), 1);
  29. if (vdev->ctrl_handler == NULL)
  30. vdev->ctrl_handler = vdev->v4l2_dev->ctrl_handler;

关键点:

1、分配video_device结构体内存          video_device_alloc()kzalloc()

2、设置video_device       .fops.ioctl_opsdev;、

也就是在probe里的这一步实现设备操作和驱动操作的绑定

3、注册video_device结构体      video_register_device()

3、操作函数(open、read、wirite等)

1、open

  1. app: open("/dev/video0",....)
  2. ---------------------------------------------------
  3. drv: v4l2_fops.v4l2_open
  4. vdev = video_devdata(filp); // 根据次设备号从数组中得到video_device
  5. ret = vdev->fops->open(filp);
  6. vivi_ioctl_ops.open
  7. v4l2_fh_open

当应用层发生系统调用时,会先调用到v4l2_fops中的v4l2_open,v4l2_open会找到对应设备video_device的file opr中的open,而设备中的open是在驱动注册后调用probe函数与驱动函数定义的open绑定的,这样就实现了video设备用户空间open调用驱动open的过程。

2、ioctl

  1. app: read ....
  2. -----------------------------------------------
  3. drv: v4l2_fops.v4l2_read
  4. struct video_device *vdev = video_devdata(filp);
  5. ret = vdev->fops->read(filp, buf, sz, off);

核心层v4l2_ctrl_handler会找到video_device的fops 。过程如下(在上文中通过v4l2_ctrl_handler_setup设置了某些属性的ioctl处理函数)

  1. __video_do_ioctl
  2. struct video_device *vfd = video_devdata(file);
  3. v4l2_is_known_ioctl(cmd)
  4. info = &v4l2_ioctls[_IOC_NR(cmd)]; /* v4l2_ioctls数组存有全部IOCTL_INFO,含有 ioctl函数指针 */
  5. ret = info->func(ops, file, fh, arg); /* 调用对应的ioctl函数 */

三、UVC(USB Video Class)

UVC全称为USB Video Class,即:USB视频类,是一种为USB视频捕获设备定义的协议标准。是Microsoft与另外几家设备厂商联合推出的为USB视频捕获设备定义的协议标准,已成为USB org标准之一。

最新的UVC版本为UVC 1.5,由USB Implementers Forum定义包括基本协议及负载格式。             

USB摄像头示例如下:

对于框架具体细节可以看这篇UVC 1.5 Class Specification 简解_uvc1.5-CSDN博客

UVC驱动的重点在于:
   描述符的分析
   属性的控制: 通过VideoControl Interface来设置
   格式的选择:通过VideoStreaming Interface来设置
   数据的获得:通过VideoStreaming Interface的URB来获得

参考文章:

​​​​​​【Linux驱动】Linux--虚拟摄像头vivid驱动分析(基于5.4内核)-CSDN博客

【转载】linux摄像头驱动_linux 摄像头驱动-CSDN博客                 

本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号