UVC系列2-探索Android UVC协议

2024-06-01 06:18
文章标签 android 协议 系列 探索 uvc

本文主要是介绍UVC系列2-探索Android UVC协议,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章选取android下linux-3.10作为分析对象,具体的UVC初始化过程可以参考csdn大神写的博客,地址是:http://blog.csdn.net/orz415678659。

uvc加载摄像头的过程无非是初始化设备,加载设备,获取设备相关参数并加载相关参数到buffer,此时就已经将视频和控制参数加载到buffer了,这篇文章主要关注的是控制相关的参数。
需要关注的两个核心文件是:

  • drivers\media\usb\uvc\uvc_ctrl.c
  • drivers\media\usb\uvc\uvc_v4l2.c

首先看看uvc_ctrl.c文件中的struct uvc_control_info uvc_ctrls[]结构体,这个结构体中定义了摄像头的控制参数详情,主要包含了各种类型的控制,比如白平衡,曝光度等。其中需要重点关注的参数是上下左右和焦距的控制参数,如下:

{.entity		= UVC_GUID_UVC_CAMERA,.selector	= UVC_CT_ZOOM_ABSOLUTE_CONTROL,.index		= 9,.size		= 2,.flags		= UVC_CTRL_FLAG_SET_CUR| UVC_CTRL_FLAG_GET_RANGE| UVC_CTRL_FLAG_RESTORE| UVC_CTRL_FLAG_AUTO_UPDATE,
},
{.entity		= UVC_GUID_UVC_CAMERA,.selector	= UVC_CT_ZOOM_RELATIVE_CONTROL,.index		= 10,.size		= 3,.flags		= UVC_CTRL_FLAG_SET_CUR| UVC_CTRL_FLAG_GET_MIN| UVC_CTRL_FLAG_GET_MAX | UVC_CTRL_FLAG_GET_RES| UVC_CTRL_FLAG_GET_DEF| UVC_CTRL_FLAG_AUTO_UPDATE,
},
{.entity		= UVC_GUID_UVC_CAMERA,.selector	= UVC_CT_PANTILT_ABSOLUTE_CONTROL,.index		= 11,.size		= 8,.flags		= UVC_CTRL_FLAG_SET_CUR| UVC_CTRL_FLAG_GET_RANGE| UVC_CTRL_FLAG_RESTORE| UVC_CTRL_FLAG_AUTO_UPDATE,
},
{.entity		= UVC_GUID_UVC_CAMERA,.selector	= UVC_CT_PANTILT_RELATIVE_CONTROL,.index		= 12,.size		= 4,.flags		= UVC_CTRL_FLAG_SET_CUR| UVC_CTRL_FLAG_GET_MIN| UVC_CTRL_FLAG_GET_MAX | UVC_CTRL_FLAG_GET_RES| UVC_CTRL_FLAG_GET_DEF| UVC_CTRL_FLAG_AUTO_UPDATE,
}

在这里可以看到这里定义了相对控制和绝对控制的参数,有焦距和上下左右,其中zoom是焦距调节,pan是左右,tilt是上下,缩写是ptz,所以云台摄像头也叫ptz摄像头。
另外需要重点关注的变量是struct uvc_control_mapping uvc_ctrl_mappings[],这个结构体枚举出来了所有的控制类型,比如id = V4L2_CID_PAN_ABSOLUTEid = V4L2_CID_TILT_ABSOLUTE的selector等于uvc_ctrls[]中的selector 是UVC_CT_PANTILT_RELATIVE_CONTROL参数,这两个id分别表示左右和上下的绝对控制,绝对控制的意思是给一个角度,摄像头转到指定的角度之后就停了。焦距的参数与pan和tilt的参数类似。稍微有点不一样的地方是zoom在3.10版本中已经添加了相对和绝对控制,参数id分别为V4L2_CID_ZOOM_ABSOLUTE,V4L2_CID_ZOOM_CONTINUOUS,其中V4L2_CID_ZOOM_CONTINUOUS是相对控制,从名称也可以看出continuous是持续的意思,相对控制就是发给摄像头一个参数,摄像头会一直不停的转动,直到最大角度才停止,中途如果给发送了停止位,才会停止。
在这么多控制参数中,重点看一下焦距的相对控制,如下:

{.id		= V4L2_CID_ZOOM_CONTINUOUS,.name		= "Zoom, Continuous",.entity		= UVC_GUID_UVC_CAMERA,.selector	= UVC_CT_ZOOM_RELATIVE_CONTROL,.size		= 0,.offset		= 0,.v4l2_type	= V4L2_CTRL_TYPE_INTEGER,.data_type	= UVC_CTRL_DATA_TYPE_SIGNED,.get		= uvc_ctrl_get_zoom,.set		= uvc_ctrl_set_zoom,
}

看看这个参数中定义了get和set,其实这个是映射到的具体get和set方法,具体方法是:

static __s32 uvc_ctrl_get_zoom(struct uvc_control_mapping *mapping,__u8 query, const __u8 *data)
{__s8 zoom = (__s8)data[0];switch (query) {case UVC_GET_CUR:return (zoom == 0) ? 0 : (zoom > 0 ? data[2] : -data[2]);case UVC_GET_MIN:case UVC_GET_MAX:case UVC_GET_RES:case UVC_GET_DEF:default:return data[2];}
}
static void uvc_ctrl_set_zoom(struct uvc_control_mapping *mapping,__s32 value, __u8 *data)
{data[0] = value == 0 ? 0 : (value > 0) ? 1 : 0xff;data[2] = min((int)abs(value), 0xff);
}

第一个函数是获取当前焦距所处的位置,在UVC_GET_CUR处,当前位置为0是,返回0,否则的话大于0,返回当前值,否则返回相反数。

第二个函数是设置zoom的值,data[0]填写的值有三种情况,0,1,0xff,其中0表示停止,1表示拉远,0xff表示拉近。

在这两个函数中可以看到UVC_GET_MINUVC_GET_MAX,意思是获取最大值和最小值,其实在设备初始化的时候,会将设备的这些各个参数的最大值和最小值读取并保存。

具体取值的地方在__uvc_query_v4l2_ctrl这个函数中,函数定义如下:

static int __uvc_query_v4l2_ctrl(struct uvc_video_chain *chain,struct uvc_control *ctrl,struct uvc_control_mapping *mapping,struct v4l2_queryctrl *v4l2_ctrl)
{struct uvc_control_mapping *master_map = NULL;struct uvc_control *master_ctrl = NULL;struct uvc_menu_info *menu;unsigned int i;memset(v4l2_ctrl, 0, sizeof *v4l2_ctrl);v4l2_ctrl->id = mapping->id;v4l2_ctrl->type = mapping->v4l2_type;strlcpy(v4l2_ctrl->name, mapping->name, sizeof v4l2_ctrl->name);v4l2_ctrl->flags = 0;if (!(ctrl->info.flags & UVC_CTRL_FLAG_GET_CUR))v4l2_ctrl->flags |= V4L2_CTRL_FLAG_WRITE_ONLY;if (!(ctrl->info.flags & UVC_CTRL_FLAG_SET_CUR))v4l2_ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY;if (mapping->master_id)__uvc_find_control(ctrl->entity, mapping->master_id,&master_map, &master_ctrl, 0);if (master_ctrl && (master_ctrl->info.flags & UVC_CTRL_FLAG_GET_CUR)) {s32 val;int ret = __uvc_ctrl_get(chain, master_ctrl, master_map, &val);if (ret < 0)return ret;if (val != mapping->master_manual)v4l2_ctrl->flags |= V4L2_CTRL_FLAG_INACTIVE;}if (!ctrl->cached) {int ret = uvc_ctrl_populate_cache(chain, ctrl);if (ret < 0)return ret;}if (ctrl->info.flags & UVC_CTRL_FLAG_GET_DEF) {v4l2_ctrl->default_value = mapping->get(mapping, UVC_GET_DEF,uvc_ctrl_data(ctrl,UVC_CTRL_DATA_DEF));}switch (mapping->v4l2_type) {case V4L2_CTRL_TYPE_MENU:v4l2_ctrl->minimum = 0;v4l2_ctrl->maximum = mapping->menu_count - 1;v4l2_ctrl->step = 1;menu = mapping->menu_info;for (i = 0; i < mapping->menu_count; ++i, ++menu) {if (menu->value == v4l2_ctrl->default_value) {v4l2_ctrl->default_value = i;break;}}return 0;case V4L2_CTRL_TYPE_BOOLEAN:v4l2_ctrl->minimum = 0;v4l2_ctrl->maximum = 1;v4l2_ctrl->step = 1;return 0;case V4L2_CTRL_TYPE_BUTTON:v4l2_ctrl->minimum = 0;v4l2_ctrl->maximum = 0;v4l2_ctrl->step = 0;return 0;default:break;}if (ctrl->info.flags & UVC_CTRL_FLAG_GET_MIN)v4l2_ctrl->minimum = mapping->get(mapping, UVC_GET_MIN,uvc_ctrl_data(ctrl, UVC_CTRL_DATA_MIN));if (ctrl->info.flags & UVC_CTRL_FLAG_GET_MAX)v4l2_ctrl->maximum = mapping->get(mapping, UVC_GET_MAX,uvc_ctrl_data(ctrl, UVC_CTRL_DATA_MAX));if (ctrl->info.flags & UVC_CTRL_FLAG_GET_RES)v4l2_ctrl->step = mapping->get(mapping, UVC_GET_RES,uvc_ctrl_data(ctrl, UVC_CTRL_DATA_RES));return 0;
}

其中结构体v4l2_queryctrl的定义在\include\uapi\linux\videodev2.h文件中,

struct v4l2_queryctrl {__u32		     id;__u32		     type;	/* enum v4l2_ctrl_type */__u8		     name[32];	/* Whatever */__s32		     minimum;	/* Note signedness */__s32		     maximum;__s32		     step;__s32		     default_value;__u32                flags;__u32		     reserved[2];
};

包含了最大值,最小值,步长,默认值,控制名称等信息。
通过传递过来的控制id和类型遍历,并将遍历的结果保存在 v4l2_queryctrl中返回。
顺带说一下,在这个文件下面这个函数也很重要:
int uvc_ctrl_set(struct uvc_video_chain *chain,struct v4l2_ext_control *xctrl),这个函数掌控着控制的大权。其中v4l2_ext_control参数就是上层传递过来的控制参数,非常重要。
从相关控制参数一路分析过来,大致明白了控制的过程和控制代码的逻辑,下一步就是熟悉USB协议之后对这个参数做出定制,以适合不同厂商的摄像头。

这篇关于UVC系列2-探索Android UVC协议的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1020208

相关文章

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

【生成模型系列(初级)】嵌入(Embedding)方程——自然语言处理的数学灵魂【通俗理解】

【通俗理解】嵌入(Embedding)方程——自然语言处理的数学灵魂 关键词提炼 #嵌入方程 #自然语言处理 #词向量 #机器学习 #神经网络 #向量空间模型 #Siri #Google翻译 #AlexNet 第一节:嵌入方程的类比与核心概念【尽可能通俗】 嵌入方程可以被看作是自然语言处理中的“翻译机”,它将文本中的单词或短语转换成计算机能够理解的数学形式,即向量。 正如翻译机将一种语言

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

flume系列之:查看flume系统日志、查看统计flume日志类型、查看flume日志

遍历指定目录下多个文件查找指定内容 服务器系统日志会记录flume相关日志 cat /var/log/messages |grep -i oom 查找系统日志中关于flume的指定日志 import osdef search_string_in_files(directory, search_string):count = 0

【Linux】应用层http协议

一、HTTP协议 1.1 简要介绍一下HTTP        我们在网络的应用层中可以自己定义协议,但是,已经有大佬定义了一些现成的,非常好用的应用层协议,供我们直接使用,HTTP(超文本传输协议)就是其中之一。        在互联网世界中,HTTP(超文本传输协议)是一个至关重要的协议,他定义了客户端(如浏览器)与服务器之间如何进行通信,以交换或者传输超文本(比如HTML文档)。

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动