UVC设备也是一个usb设备,在uvc_driver.c中的init函数会调用usb_register注册,根据id_table发送可支持的设备后调用probe函数,其会去uvc_register_chains注册所有uvc_device,前面说的根据id_table匹配的过程是usb_bus总线,uvc是另一个东西了

vivi.c是一个摄像头驱动模板或者实例,安装vivi模块后会产生一个虚拟的摄像头,使用应用程序可以获得其提供的虚拟数据并显示。

vivi_init
  vivi_create_instance
    v4l2_device_register // 不是主要, 只是用于初始化一些东西,比如自旋锁、引用计数

    hdl = &dev->ctrl_handler;//hdl 是用于保存子设备控制方法集的结构体,对于视频设备这些ctrls包括设置亮度、饱和度、对比度和 清晰度                                                                                      等,用链表的方式来保存ctrls,可以通过v4l2_ctrl_new_std函数向链表添加ctrls。在下面的代码中用到了这个函数。
    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属性",并挂接到hdl的链表上(用于应用的ioctl)://后续会根据这里设置的属性调用真正的ioctl,用户空间可以通过ioctl的VIDIOC_S_CTRL指令调用到hdl(v4l2_ctrl_handler结构体)
        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_device
        vdev->cdev = cdev_alloc();
        vdev->cdev->ops = &v4l2_fops;//真正的fops是vivi_fops
        cdev_add

        video_device[vdev->minor] = vdev;//把包含真正fops结构体的变量放入video_device数组,以次设备号为下标

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

分析vivi.c的open,read,write,ioctl过程
1. open
app: open("/dev/video0",....)
---------------------------------------------------
drv: v4l2_fops.v4l2_open
    vdev = video_devdata(filp); // 根据次设备号从数组中得到video_device
    ret = vdev->fops->open(filp);
      vivi_ioctl_ops.open
        v4l2_fh_open

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

3. ioctl
app: ioctl
----------------------------------------------------
drv: v4l2_fops.unlocked_ioctl
  v4l2_ioctl
    struct video_device *vdev = video_devdata(filp);
    ret = vdev->fops->unlocked_ioctl(filp, cmd, arg);
      video_ioctl2
        video_usercopy(file, cmd, arg, __video_do_ioctl);
          __video_do_ioctl
            struct video_device *vfd = video_devdata(file);
            根据应用传入的cmd来获得、设置"某些属性"

v4l2_ctrl_handler的使用过程:(v4l2_ctrl_handler是属性链表,在video_device_alloc被设置)
  __video_do_ioctl//往上面数5行,video_usercopy里面调用__video_do_ioctl这个函数,最终ioctrl会表用vivi_ioctl_ops结构体里面的ioctrl函数
    struct video_device *vfd = video_devdata(file);

    case VIDIOC_QUERYCTRL:
    {
      struct v4l2_queryctrl *p = arg;

      if (vfh && vfh->ctrl_handler)
        ret = v4l2_queryctrl(vfh->ctrl_handler, p);
      else if (vfd->ctrl_handler) // 在哪设置?在video_register_device中vdev->ctrl_handler = vdev->v4l2_dev->ctrl_handler,vdev->v4l2_dev在video_register_device前通过vfd->v4l2_dev = &dev->v4l2_dev复制,这里的dev是vivi_device,dev->v4l2_dev.ctrl_handler = hdl,这里的hdl就是vivi_init过程中在v4l2_ctrl_new_std里面被设置,vfd就是video_device结构体类型;ctrl_handler是v4l2_ctrl_handler

        ret = v4l2_queryctrl(vfd->ctrl_handler, p);
          // 根据ID在ctrl_handler里找到v4l2_ctrl,返回它的值,ID在v4l2_ctrl_new_std里面被设置

怎么写V4L2驱动?

1、分配、设置、注册V4L2_device  (v4l2_device_register,v4l2_device)(仅初始化,并提供一些属性参数供ioctl设置,辅助作用,属性在这个v4l2_ctrl_handler里面被管理,v4l2_ctrl来表示属性,v4l2_ctrl_new_std来创建v4l2_ctrl,放入v4l2_ctrl_handler链表)

2、分配video_device(video_device_alloc)

3、设置 video_device()

  vfd->v4l2_dev = &dev->v4l2_dev;//把video_device和第一部里面设置的属性关联起来,后面会用来ioctl设置属性,比如亮度、饱和度等参数

  fops=真真的v4l2_file_operations,v4l2_fops中的read等会指向这里设置的fops;

  ioctl_ops=真真的ioctl函数,v4l2_fops->unlocked_ioctl调用v4l2_file_operations->unlocked_ioctl,调用ioctl_ops

13、虚拟驱动vivi.c注册过程分析及怎么写V4L2驱动及启动过程的更多相关文章

  1. 2.1 摄像头V4L2驱动框架分析

    学习目标:学习V4L2(V4L2:vidio for linux version 2)摄像头驱动框架,分析vivi.c(虚拟视频硬件相关)驱动源码程序,总结V4L2硬件相关的驱动的步骤:  一.V4L ...

  2. v4l2驱动编写篇【转】

    转自:http://blog.csdn.net/michaelcao1980/article/details/53008418 大部分所需的信息都在这里.作为一个驱动作者,当挖掘头文件的时候,你可能也 ...

  3. 通过虚拟驱动vivi分析摄像头驱动

    Linux摄像头驱动学习之:(二)通过虚拟驱动vivi分析摄像头驱动 一.通过指令 "strace -o xawtv.log xawtv" 得到以下调用信息: // 1~7都是在v ...

  4. 彻底分析虚拟视频驱动vivi(三)

    在Ubuntu系统中接上usb摄像头设备时,系统会自动安装对应的usb设备驱动程序.我们现在要使用自己编译的vivi驱动,该怎么办呢? 1.先安装系统自带的vivi驱动和它所依赖的所有驱动:sudo ...

  5. V4L2(二)虚拟摄像头驱动vivi深入分析【转】

    转自:http://www.cnblogs.com/tureno/articles/6694463.html 转载于: http://blog.csdn.net/lizuobin2/article/d ...

  6. 初始v4l2(六)-------根据虚拟驱动vivi的使用彻底分析摄像头驱动

    前面的几篇文章已经分析了v4l2的框架,对框架的分析是比较粗浅的,能基本清楚函数之间的调用过程.但是很多内容并没有分析,比如说里面有很多ioctl,并没有分析哪些ioctl是必须的,也没有分析如何从应 ...

  7. 20、在PC上测试虚拟驱动vivi

    在Ubuntu上测试 准备工作:安装xawtv(是一个应用程序,用来在Ubuntu上捕获摄像头数据并显示)sudo apt-get install xawtv 源码xawtv-3.95.tar.gz: ...

  8. (四) 虚拟摄像头vivi体验

    目录 虚拟摄像头vivi体验 源码下载 修改Makefile 安装xawtv 测试体验 title: 虚拟摄像头vivi体验 date: 2019/4/23 19:20:00 toc: true -- ...

  9. 虚拟摄像头vivi的测试(二)

    (前一部分的基础操作来源于作者:LingXiaokai 的 Ubuntu 9.10 下如何使用笔记本摄像头以及虚拟摄像头vivi的测试) 自己仅对实际操作中需要注意的点进行阐述 一.先在Ubuntu ...

随机推荐

  1. 适合ASP.NET Web API使用的场景

    富客户端web应用程序:ASP.NET Web API适合大量使用AJAX调用的富客户端应用程序,如Silverlight应用程序,基于Adobe Flash的应用程序或单页应用程序(SPA)等. 本 ...

  2. C++ 补课(二)

    1,如果遇到派生类成员和基类成员的名称冲突的情况,程序会采用派生类成员执行相应的操作.如果需要使用基类中的同名成员,则必须在程序中使用全局分辨符“::” 虚基类 —— 派生类在继承基类时加入“virt ...

  3. POJ 1474 Video Surveillance 半平面交/多边形核是否存在

    http://poj.org/problem?id=1474 解法同POJ 1279 A一送一 缺点是还是O(n^2) ...nlogn的过几天补上... /********************* ...

  4. 【Educational Codeforces Round 36 B】Browser

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 分类在区间里面和左边.右边三种情况. 看看l的左边有没有标签.r的右边有没有标签. 就能做完了. [代码] #include < ...

  5. 决策树之C4.5算法学习

    决策树<Decision Tree>是一种预測模型,它由决策节点,分支和叶节点三个部分组成. 决策节点代表一个样本測试,通常代表待分类样本的某个属性,在该属性上的不同測试结果代表一个分支: ...

  6. AngularJS渲染性能分析

    作者:Jiang, Jilin AngularJS中,通过数据绑定.能够十分方便的构建页面.可是当面对复杂的循环嵌套结构时,渲染会遇到性能瓶颈.今天,我们将通过一些列实验,来測试AngularJS的渲 ...

  7. 用 cctld工具创建带有国家代码的IP地址表

    用 cctld工具创建带有国家代码的IP地址表 cctld tools is creating IP addresses table with Country Code 项目地址 https://gi ...

  8. [NOI.AC#31]MST 计数类DP

    链接 注意到 \(n\) 只有40,爆搜一下发现40的整数拆分(相当于把 \(n\) 分成几个联通块)很少 因此可以枚举联通块状态来转移,这个状态直接用vector存起来,再用map映射,反正40也不 ...

  9. 9.使用 npm 命令安装模块

    转自:http://www.runoob.com/nodejs/nodejs-tutorial.html npm 安装 Node.js 模块语法格式如下: $ npm install <Modu ...

  10. Flume的client

    Client:生产数据,运行在一个独立的线程.