Linux&UVC驱动
+ -

LINUX&UVC视频控制请求与V4L2映射关系

2024-03-28 142 0

从 Linux&UVC驱动栈 https://www.usbzh.com/article/detail-1322.html 一文可知,UVC驱动其实是一个中间层驱动程序,其启着承上启下的功能。

  • UVC驱动的下层是USBCore驱动,其实现的是USB设备初始化及通讯相关的功能实现。
  • 在UVC驱动的上层,则是V4L2,是LINUX关于视频流的通用驱动程序。假如从Windows的角度来讲,V4L2是类驱动程序,UVC驱动是端口驱动程序。这和Windows系统中HIDCLASS.SYS和HIDUSB.SYS的关系类似。

在UVC视频控制请求描述符和UVC视频流控制请求中,存在着大量的UVC特定类请求。如果只是单纯的是UVC驱动,其实完全可以不需要V4L2,但由于存在其它总线的相机的问题,如MIPI之类。所以关于发向这些UVC拓扑结构或者视频流接口特定类请求需要在V4L2这里有一个更加广泛而统一的接口。这些统一的调用接口在系统层应可以更加方便的使用于其它总线的设备。这就类似于系统层所有设备都被当作文件对待一下,都可以支持VFS层想关的fops相关回调函数一样。

从LINUX&UVC驱动的代码来看,主要汲到以下请求:

static const u8 uvc_processing_guid[16] = UVC_GUID_UVC_PROCESSING;
static const u8 uvc_camera_guid[16] = UVC_GUID_UVC_CAMERA;
static const u8 uvc_media_transport_input_guid[16] = UVC_GUID_UVC_MEDIA_TRANSPORT_INPUT;

static int uvc_entity_match_guid(const struct uvc_entity *entity,
    const u8 guid[16])
{
    switch (UVC_ENTITY_TYPE(entity)) {
    case UVC_ITT_CAMERA:
        return memcmp(uvc_camera_guid, guid, 16) == 0;

    case UVC_ITT_MEDIA_TRANSPORT_INPUT:
        return memcmp(uvc_media_transport_input_guid, guid, 16) == 0;

    case UVC_VC_PROCESSING_UNIT:
        return memcmp(uvc_processing_guid, guid, 16) == 0;

    case UVC_VC_EXTENSION_UNIT:
        return memcmp(entity->extension.guidExtensionCode,guid, 16) == 0;

    default:
        return 0;
    }
}
  • UVC_ITT_CAMERA:UVC视频输入端相机特定类请求,如PTZ
  • UVC_ITT_MEDIA_TRANSPORT_INPUT:这种类型的设备本人没有接触过。不做分析。
  • UVC_VC_PROCESSING_UNIT:UVC控制接口UVC特定类请求。
  • UVC_VC_EXTENSION_UNIT:UVC用于用户自定义通讯的扩展单元请求。

从定义来看,除过扩展单元以外,其它所有请求都定义了统一的GUID,用于惟一的标识。而扩展单元可由固件开发者自行定义。这样就做到了UVC请求在格式上的统一。

那么,为什么要做这些处理呢?这是因为不同的相机固件对以上的拓扑结构支持的功能是不同的。以UVC_ITT_CAMERA为示例。其最多可能支持以下的请求:

bmControls:使用位图来表示支持的视频流。
D0:扫描模式 //扫描模式(逐行扫描或隔行扫描)
D1:自动曝光模式
D2:自动曝光优先级
D3:曝光时间(绝对值)
D4:曝光时间(相对)
D5:焦点(绝对)
D6:焦点(相对)
D7:虹膜光圈(绝对)
D8:虹膜光圈(相对)
D9:缩放(绝对)
D10:缩放(相对)
D11:PanTilt(绝对) PT是左右和上下移动
D12:PanTilt(相对)
D13:滚动(绝对)
D14:滚动(相对)
D15:预留
D16:预留
D17:自动对焦
D18:隐私
D19:专注,简单
D20:窗口
D21:关注区域,感兴趣区域
D22–D23:保留,设置为零

详见

但固件一般大概率是不会支持全部的,故只会根据其设备的实际情况选择性的支持。例如这里可能只支持PTZ相关的,其余不支持。故在设备初始化的时候,会对这此是否支持进行处理:

/*
 * Initialize device controls.
 */
int uvc_ctrl_init_device(struct uvc_device *dev)
{
...
/* Walk the entities list and instantiate controls */
    list_for_each_entry(entity, &dev->entities, list) {
        struct uvc_control *ctrl;
        unsigned int bControlSize = 0, ncontrols;
        u8 *bmControls = NULL;

        //找到各个相请的控制变量
        if (UVC_ENTITY_TYPE(entity) == UVC_VC_EXTENSION_UNIT) {
            bmControls = entity->extension.bmControls;
            bControlSize = entity->extension.bControlSize;
        } else if (UVC_ENTITY_TYPE(entity) == UVC_VC_PROCESSING_UNIT) {
            bmControls = entity->processing.bmControls;
            bControlSize = entity->processing.bControlSize;
        } else if (UVC_ENTITY_TYPE(entity) == UVC_ITT_CAMERA) {
            bmControls = entity->camera.bmControls;
            bControlSize = entity->camera.bControlSize;
        }

        /* Remove bogus/blacklisted controls */
        uvc_ctrl_prune_entity(dev, entity);

        /* Count supported controls and allocate the controls array */
        ncontrols = memweight(bmControls, bControlSize);
        if (ncontrols == 0)
            continue;

        //分配控制请求
        entity->controls = kcalloc(ncontrols, sizeof(*ctrl),  GFP_KERNEL);
        if (entity->controls == NULL)
            return -ENOMEM;
        entity->ncontrols = ncontrols;

        /* Initialize all supported controls */
        ctrl = entity->controls;
        for (i = 0; i < bControlSize * 8; ++i) {
            //位检查,是否支持某个特性
            if (uvc_test_bit(bmControls, i) == 0)
                continue;

            ctrl->entity = entity;
            ctrl->index = i;
            //如果支持,调用uvc_ctrl_init_ctrl进行初始化
            uvc_ctrl_init_ctrl(dev, ctrl);
            ctrl++;
        }
    }

    return 0;
}

在uvc_ctrl_init_ctrl函数中,通过uvc_ctrl_add_info调用uvc_ctrl_get_flags获取该选择子支持的操作。该操作是通过GET_INFO特定类请求实现的。

static int uvc_ctrl_get_flags(struct uvc_device *dev,
                  const struct uvc_control *ctrl,
                  struct uvc_control_info *info)
{
    u8 *data;
    int ret;

    data = kmalloc(1, GFP_KERNEL);
    if (data == NULL)
        return -ENOMEM;

    ret = uvc_query_ctrl(dev, UVC_GET_INFO, ctrl->entity->id, dev->intfnum,info->selector, data, 1);
    if (!ret)
        info->flags |= (data[0] & UVC_CONTROL_CAP_GET ?    UVC_CTRL_FLAG_GET_CUR : 0)
                |  (data[0] & UVC_CONTROL_CAP_SET ?    UVC_CTRL_FLAG_SET_CUR : 0)   
                |  (data[0] & UVC_CONTROL_CAP_AUTOUPDATE ?    UVC_CTRL_FLAG_AUTO_UPDATE : 0)
                |  (data[0] & UVC_CONTROL_CAP_ASYNCHRONOUS ?UVC_CTRL_FLAG_ASYNCHRONOUS : 0);

    kfree(data);
    return ret;
}

0 篇笔记 写笔记

打开UVC摄像头(特定类请求)数据分析
通过上节可以知道,设备支持4种的视频格式,且每种格式如下:MPJPG数据格式bFormatIndexbFrameIndex分辩率及及帧率11640x480x(30,15,10)121280x720x(30,15,10)131920x1080x(30,15......
UVC特定类请求get cur返回长度是26或者是34的是什么?
从返回的长度来看,你应该是在应用打开摄像头,进行视频格式及图形分辨率协商返回的数据长度.UVC视频流接口这两个特别重要的选择子,分别用于协商过程和提交数据格式,让固件开始发送数据。ControlSelector ValueVS_PROBE_CONTROL0x01VS_COMMIT_CONTROL0x......
USB标准请求、类特定请求总结
做USB开发,最常见的就是USB的控制请求的那8个字节的分析,如果抓包工具有协议分析那还好说,但是如果不带速协议分析,那其实还是挺郁闷的。所以说,如何快速的通过USB请求的那几个字段定位出是何请求,是可以大大的提高我们的协议分析效率的。USB的请求这里包括USB的标准请求和特定类请求,其包括以下几......
UAC 特定类请求
特定类请求一般用于设置或获取音频控制。这些控制又分为两大部分:音频控制请求:对音频控制单元或终端进行控制。音频流请求: 如对音频流控制的请求,如音频采相率等。音频设备类也支持其他特定于类的请求:内存请求(Memory Request),每个可寻址的实体或终端,可导出一个内存映射接口。提供对该实体内存......
UVC 再议特定类请求
通过特定请求的支持 GET_INFO可以获取设备支持的特定类请求。当然在UVC规范中,有些特定类请求是可选择的,有些是必须的,这种情况因特定类请求的使用环境不同而定。我们知道,特定类请求一般包括:名称值 说明RC_UNDEFINED0x00未定义SET_CUR0x01设置属性GET_CUR0x81获......
UVC 特定请求的支持 GET_INFO
GET_INFO请求用于获取设备支持的特定请求。GET_INFO的具体使用,可参见:处理单元特定类请求示例一节。使用此请求获取设备支持的特定请求属性时,wLength字段为1,其返回值各位代表的含义见下表:位描述位状态D01=Supports GET value requestsCapability......
USB 标准请求
USB定义了8个字节的标准请求,通过这些请求,可以对设备的状态进行更改或对设备进行枚举。USB的标准请求的数据传输方式都是控制传输方式,所以使用的端点是设备的默认端点0。USB这8个字节的的控制请求不包括传输过程中的数据,但包括了控制请求的数据长度。当数据长度为不为0时,设备或主机向对方传输相应长......
UVC 特定类请求概述
本节我们来讲述UVC的特定类请求。UVC特定类请求的功能UVC特定类请求的主要功能用于控制UVC摄像头,实现对UVC摄像头的打开,关闭及摄像头参数的控制。通过对UVC规范的学习,使用我们可详细的了解UVC是怎样通过这些特定类请求实现对UVC摄像头的控制。UVC特定类请求的分类大多数特定于类的请求用于......
UVC 处理单元特定类请求示例
笔者手中有一UVC摄像头,其处理单元描述符bUnitID为2,其处理单元描述符内容如下: -------- Video Control Processing Unit Descriptor -----------------------bLength ......
U盘枚举失败-该设备无法启动(GET_MAX_LUN请求)
群里有人用STM32搞了一个U盘,但是U盘在插入电脑后在设备管理器是枚举失败。我让他看一下设备状态:又是熟悉的错误码10,表示设备启动失败。设备的启动失败,一般在设备获取描述符获取之后,初始会失败,我之前在弄USB虚拟鼠标的时候也遇到此类情况。不过由于这个设备是U盘,本人还没有研究USB存储协议,只......
BUSHOUND抓包stall pid的USTS c0000004错误
对USB设备进行数据分析,使用最多的也就是BUSHOUND了,不过经常遇到一个问题就是 USTS c0000004 stall pid 错误。USTS c0000004 stall pid在本站中搜......
BOT MASS_STORAGE_RESET
Bulk-Only Mass Storage Reset类特定请求是USB大容量存储设备独有的。 该特定类请求的功能用于复位大容量存储设备和与之关联的接口。通知设备接下来的批量端点输出数据为命令块包(CBW)。由于该请求是控制请求,所以是通过端点0发送的。在设备完成该请求即复位之前,设备应......
UVC摄像头视频特定类请求
UVC摄像头特定类请求包括视频控制接口请求和视频流请求。/* Video Class-specific Request codes */#define SET_CUR 0x01#define GET_CUR 0x81#define GET_MIN 0x82#define GET_MAX 0......
USB2.0集线器HUB特定类请求大全
RequestbmRequestTypebRequestwValuewIndexwLengthDataClearHubFeature00100000B (0x20)CLEAR_FEATURE(0x01)FeatureSelector00ClearPort......
LINUX&UVC视频控制请求与V4L2映射关系
从 Linux&UVC驱动栈 https://www.usbzh.com/article/detail-1322.html 一文可知,UVC驱动其实是一个中间层驱动程序,其启着承上启下的功能。在UVC驱动的下层是USBCore驱动,其实现的是USB设备初始化及通讯相关的功能实现。在UVC......
关注公众号
  • HID人机交互
  • Linux&USB
  • UAC音频
  • TYPE-C
  • USB规范
  • USB大容量存储
  • USB百科
  • USB周边
  • UVC摄像头
  • Windows系统USB
  • 音视频博客
  • 取消
    感谢您的支持,我会继续努力的!
    扫码支持
    扫码打赏,你说多少就多少

    打开支付宝扫一扫,即可进行扫码打赏哦

    您的支持,是我们前进的动力!