V4L2框架分析

       V4L2是Video for linux2的简称,为linux中关于视频设备的内核驱动。v4L2是针对uvc(USB Video Class)免驱usb设备的编程框架,主要用于采集usb摄像头等。

      下图是V4L2的框架,首先系统核心层分配设置注册一个名为cdev结构体变量(cdev结构体是video_device结构体里的一部分),并设置cdev->ops = v4l2_fops;在硬件层我们分配设置注册了一个名为vfd结构体变量(video_device结构体),并设置vfd->fops = &vivi_fops,vfd->ioctl_ops  = &vivi_ioctl_ops;当应用程序(APP)调用read、open等函数时,会调用到v4l2_fops里的read、open函数,然后v4l2_fops里的read、open函数会再调用到硬件层相关的vfd->fops里的read、open函数。ioctl函数也类似。

        下面我们从程序入手来分析V4L2的框架,本文借助Linux内核目录下的drivers\medio\video里的虚拟视频驱动程序vivi.c(这段代码使用v4l2 api模拟真实的视频设备)来分析V4L2的框架。它的总体框架如下所示:

vivi_initvivi_create_instancev4l2_device_register   // 不是主要, 只是用于初始化一些东西,比如自旋锁、引用计数video_device_alloc// 设置1. vfd:.fops           = &vivi_fops,.ioctl_ops 	= &vivi_ioctl_ops,.release	= video_device_release,2.vfd->v4l2_dev = &dev->v4l2_dev;3. 设置"ctrl属性"(用于APP的ioctl):v4l2_ctrl_handler_init(hdl, 11);dev->volume = v4l2_ctrl_new_std(hdl, &vivi_ctrl_ops,V4L2_CID_AUDIO_VOLUME, 0, 255, 1, 200);dev->brightness = v4l2_ctrl_new_std(hdl, &vivi_ctrl_ops,V4L2_CID_BRIGHTNESS, 0, 255, 1, 127);dev->contrast = v4l2_ctrl_new_std(hdl, &vivi_ctrl_ops,V4L2_CID_CONTRAST, 0, 255, 1, 16);                        video_register_device(video_device, type:VFL_TYPE_GRABBER, nr)__video_register_devicevdev->cdev = cdev_alloc();vdev->cdev->ops = &v4l2_fops;cdev_addvideo_device[vdev->minor] = vdev;if (vdev->ctrl_handler == NULL)vdev->ctrl_handler = vdev->v4l2_dev->ctrl_handler;

①我们从vivi.c里的vivi_init函数入手发现它调用了v4l2_device_register,该函数用于初始化一些东西,比如自旋锁、引用计数,这个并不是必需的;②调用了video_device_alloc分配video_device结构体并对其进行相应的设置,例如

.fops             = &vivi_fops,
.ioctl_ops     = &vivi_ioctl_ops,
.release       = video_device_release,

等设置,然后video_register_device注册该结构体;

③video_register_device函数调用了__video_register_device实现了如下操作:

vdev->cdev = cdev_alloc();
vdev->cdev->ops = &v4l2_fops;
cdev_add
                
video_device[vdev->minor] = vdev;

if (vdev->ctrl_handler == NULL)
    vdev->ctrl_handler = vdev->v4l2_dev->ctrl_handler;

 

 

上图是vivi_create_instance函数的一部分,首先分配一个video_device结构体的变量vfd,然后设置*vfd = vivi_template;其中vivi_template是一个video_device的结构体变量,它本身设置好了一些如.fops之类信息(如下图),此操作便相当于设置

 1. vfd:

.fops             = &vivi_fops,
.ioctl_ops     = &vivi_ioctl_ops,
.release       = video_device_release,

static struct video_device vivi_template = {.name		= "vivi",.fops           = &vivi_fops,.ioctl_ops 	= &vivi_ioctl_ops,.release	= video_device_release,.tvnorms              = V4L2_STD_525_60,.current_norm         = V4L2_STD_NTSC_M,
};

然后进入video_register_device函数,下面是video_register_device里的一部分源码,首先分配一个cdev结构体

然后设置cdev->ops = &v4l2_fops;v4l2_fops本身指向了一些函数(如下图),这样cdev便也指向了这些函数,当APP调用read函数时,便会调用cdev里面的read函数

static const struct file_operations v4l2_fops = {.owner = THIS_MODULE,.read = v4l2_read,.write = v4l2_write,.open = v4l2_open,.get_unmapped_area = v4l2_get_unmapped_area,.mmap = v4l2_mmap,.unlocked_ioctl = v4l2_ioctl,
#ifdef CONFIG_COMPAT.compat_ioctl = v4l2_compat_ioctl32,
#endif.release = v4l2_release,.poll = v4l2_poll,.llseek = no_llseek,
};

cdev里面的read函数如下图,首先根据filp获取到video_device结构体,然后判断该video_device结构体里的read函数是否存在,若存在则调用它,所以最后便调用到了前面我们设置的vfd.fops里的read函数。

 

 

 

 

相比open、read函数,ioctl的调用过程更复杂一些,下面我们来看一下(我们以VIDIOC_QUERYCAP为例)。下图是v4l2_fops里的.unlocked_ioctl指向的v4l2_ioctl函数。

它调用了前面vivi_template的fops里面的ioctl。

vivi_template的fops里面的ioctl里调用到下图的__video_do_ioctl函数,该函数最终调用到vfd里的ioctl_ops成员里面的函数,即vivi_ioctl_ops里的函数

比如调用的是 VIDIOC_QUERYCAP,则最终会调用到下面的函数。

/* ------------------------------------------------------------------IOCTL vidioc handling------------------------------------------------------------------*/
static int vidioc_querycap(struct file *file, void  *priv,struct v4l2_capability *cap)
{struct vivi_dev *dev = video_drvdata(file);strcpy(cap->driver, "vivi");strcpy(cap->card, "vivi");strlcpy(cap->bus_info, dev->v4l2_dev.name, sizeof(cap->bus_info));cap->version = VIVI_VERSION;cap->capabilities = V4L2_CAP_VIDEO_CAPTURE | V4L2_CAP_STREAMING | \V4L2_CAP_READWRITE;return 0;
}

 

 

总结:ioctl的调用比open、read多了一层,当APP调用ioctl函数时,便会调用cdev里面的ioctl函数,然后调用到了前面我们设置的vfd.fops里的ioctl函数,即和read、open函数同一结构体里的v4l2_ioctl,然后最终再去调用到和  

.fops           = &vivi_fops,同一结构体里的
.ioctl_ops     = &vivi_ioctl_ops,里对应的函数。

 

 

相关文章:

https://blog.csdn.net/qingkongyeyue/article/details/53447331

https://blog.csdn.net/qingkongyeyue/article/details/52372960

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/384507.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux下IO多路复用之select函数的使用

select函数的作用: 如果我们的程序里有两个需要阻塞的地方,例如要从服务器读数据,同时还要从键盘上读数据(若不采用阻塞而用查询的方式则大量占用系统资源)。这个时候我们就有两处阻塞,你当然可以用多线程或…

条件变量实现线程同步

(1) 什么是条件变量实现线程同步?   假如我们的程序中有两个线程,一个是生产者线程,另一个是消费者线程,生产者线程每隔一段时间把数据写入到缓冲区buffer中,而消费者线程则每隔一段时间从buffer中取出数据,为了避免…

mjpg-streamer框架分析

mjpg-streamer程框架图如下所示: 程序运行起来后,主进程根据传入的参数设置的输入输出通道打开对应的输入输出动态链接库,并依次调用以下函数 1、输入---仓库-----输出(mjpg-streamer.h) (1)gl…

用strace工具跟踪系统调用

Linux下可以用strace工具查看应用程序的系统调用。 strace -h 查看能调用的参数 1.strace -o xwatv.log xwatv //-o xwatv.log 是指定将跟踪信息存放在xwatv.log中,xwatv是指要跟踪的命令或应用程序 2.把生成的log文件拷贝回windows下进行分析 主要分析open…

linux字符驱动之概念介绍

一、字符驱动框架 问:应用程序open、read、write如何找到驱动程序的open、read、write函数? 答:应用程序的open、read、write是在C库里面实现的,它里面通过swi val指令去触发一个异常,这个异常就会进入到内核空间,在内…

linux字符驱动之自动创建设备节点

上一节中,我们是手工创建设备节点,大家肯定也会觉得这样做太麻烦了。 上一节文章链接:https://blog.csdn.net/qq_37659294/article/details/104302700 问:能不能让系统自动创建设备节点? 答:可以&#x…

linux字符驱动之点亮LED

上一节中,我们讲解了如何自动创建设备节点,这一节我们在上一节的基础上,实现点亮LED。 上一节文章链接:https://blog.csdn.net/qq_37659294/article/details/104308284 驱动里面能够用很多种方法实现LED驱动,其中有本…

USB摄像头视频监控项目学习笔记

一个摄像头监控应用程序的系统调用如下所示: /* open * VIDIOC_QUERYCAP 确定它是否视频捕捉设备,支持哪种接口(streaming/read,write) * VIDIOC_ENUM_FMT 查询支持哪种格式 * VIDIOC_S_FMT 设置摄像头使用哪种格式 * VIDIOC_REQBUFS 申请buffer 对于 str…

图片缩放算法

项目背景:博主之前做过一个摄像头采集数据,然后在LCD上显示视频数据的项目,假如我们摄像头采集的一帧数据的分辨率比我们的LCD的分辨率要大,那么LCD则无法显示整个图像,这时候我们就要把这么一帧图片进行缩放&#xff…

数码相框项目之显示一张可放大、缩小、拖拽的图片

之前我做过一个电子相框的项目,涉及到的重难点主要为:在LCD上放大、缩小、移动图片。 首先我们得明白的一点是:无论是放大或缩小,实际上都是对原图进行等比例的缩小,然后在LCD上面显示,只不过缩小的程度不…

TCP协议-如何保证传输可靠性

TCP协议传输的特点主要就是面向字节流、传输可靠、面向连接。这篇博客,我们就重点讨论一下TCP协议如何确保传输的可靠性的。 确保传输可靠性的方式 TCP协议保证数据传输可靠性的方式主要有: 校验和序列号确认应答超时重传连接管理流量控制拥塞控制 校…

TCP协议-握手与挥手

认识TCP协议 TCP全称为“传输控制协议”,这是传输层的一个协议,对数据的传输进行一个详细的控制。 特点: 面向字节流安全可靠面向连接 TCP协议段格式 源端口号与目的端口号:这里与UDP的一样,每个数据都要知道从哪个…

ASOC注册过程

一、什么是ASOC 在嵌入式系统里面的声卡驱动为ASOC(ALSA System on Chip) ,它是在ALSA 驱动程序上封装的一层,分为3大部分,Machine,Platform和Codec ,三部分的关系如下图所示:其中Machine是指我…

ASOC调用过程

上一篇文章我们将了嵌入式系统注册声卡的过程:https://blog.csdn.net/qq_37659294/article/details/104748747 这篇文章我们以打开一个声卡的播放节点为例,讲解一下在APP调用open时,最终会如何调用到硬件相关的函数。 在上一篇文章最后我们说…

编写声卡驱动(框架)

在前面两篇文章中,我们分别讲了嵌入式Linux系统声卡注册的过程和调用的过程: https://blog.csdn.net/qq_37659294/article/details/104748747 https://blog.csdn.net/qq_37659294/article/details/104802868 讲了那么多,我们最终的目的无非…

声卡学习笔记

分享几篇关于韦东山声卡驱动的学习笔记,作者写得非常详细。 ALSA驱动框架:https://blog.csdn.net/qingkongyeyue/article/details/52328991 ASoC驱动框架:https://blog.csdn.net/qingkongyeyue/article/details/52349120 ASoC驱动重要结构…

路由器、交换机、集线器的区别

https://blog.csdn.net/weibo1230123/article/details/82779040

$PATH环境变量的作用

echo $PATH 显示当前PATH环境变量,该变量的值由一系列以冒号分隔的目录名组成,如:/usr/local/bin:/bin:/usr/bin。(冒号:是路径分隔符) 在执行一个程序的时候如果没有PATH的话,就需要写出路径名(绝对或者相对&#xf…

dmesg

https://blog.csdn.net/zm_21/article/details/31760569

进程上下文与中断上下文的理解

一.什么是内核态和用户态 内核态:在内核空间执行,通常是驱动程序,中断相关程序,内核调度程序,内存管理及其操作程序。 用户态:用户程序运行空间。 二.什么是进程上下文与中断上下文 1.进程上下文&#xf…