UVC系列5-编写Android jni代码实现控制PTZ

在Android kernel层完成定制之后,需要写app实现对摄像头的控制,主要通过jni代码实现。
在jni代码中主要定义这几个函数:
jintArrayJava_com_chuck_android_uvccamera_MainActivity1_startControlCamera(JNIEnv*env,jobject thiz , jint controlId ,jint value)
实现控制的主函数:

//实现获取当前控制参数的值
jintJava_com_chuck_android_uvccamera_MainActivity1_getCurrentControlValue
//判断当前摄像头是否支持PTZ控制。
jbooleanJava_com_chuck_android_uvccamera_UVCCameraPreview_isSupportPtz

现在逐个函数说明,startControlCamera函数如下:

//返回控制是否成功
void Java_com_chuck_android_uvccamera_MainActivity_startControlCamera(JNIEnv* env,jobject thiz , jint controlId ,jint value){
    switch (controlId) {
        case V4L2_CID_PAN_RELATIVE:
            //控制左右转动
            __android_log_print(ANDROID_LOG_ERROR , TAG , "控制左右的值是 = %d" , value);
//          controlId = V4L2_CID_PAN_ABSOLUTE;
            controlId = V4L2_CID_PAN_RELATIVE;
//          controlId = PAN_SPEED_ID;
//          controlType = "V4L2_CID_PAN_ABSOLUTE ";
            break;
        case V4L2_CID_TILT_RELATIVE:
            //控制上下转动
            __android_log_print(ANDROID_LOG_ERROR , TAG , "控制上下的值是 = %d" , value);
//          controlId = V4L2_CID_TILT_ABSOLUTE;
            controlId = V4L2_CID_TILT_RELATIVE;
//          controlId = TILT_SPEED_ID;
//          controlType = "V4L2_CID_TILT_ABSOLUTE ";
            break;
        case ZOOM_RELATIVE_ID:
            //控制聚焦
            controlId = ZOOM_RELATIVE_ID;
            break;
        case PAN_SPEED_ID:
            controlId = PAN_SPEED_ID;
            break;
        case TILT_SPEED_ID:
            controlId = TILT_SPEED_ID;
            break;
        default:
            break;
    }
    int result = startControlPanTilt(controlId , value);
    if(result != 0){
        __android_log_print(ANDROID_LOG_ERROR , TAG , "控制失败");
    }
}

这里的controlId就是我们在kernel添加的id,value就是每个控制id对应的值,值要按照UVC协议规定的传递。

接下来看看startControlPanTilt函数:

int startControlPanTilt(int controlId , int value){
    //getControlValue(controlId);
    __android_log_print(ANDROID_LOG_ERROR , TAG , "start controlId = %d , value = %d" , controlId , value);
    //jint setBefore;
    if(fd < 0){
        __android_log_print(ANDROID_LOG_ERROR , TAG , "device open fail");
        return -1;
    }
    struct v4l2_control ctrl;
 
    CLEAR(ctrl);
 
    ctrl.id = controlId;
    ctrl.value = value;
 
    int result = ioctl(fd, VIDIOC_S_CTRL, &ctrl);
    __android_log_print(ANDROID_LOG_ERROR , TAG , "result = %d , controlId = %d , controlValue = %d" , result , ctrl.id , ctrl.value);
    if (result == -1){
        __android_log_print(ANDROID_LOG_ERROR , TAG , "VIDIOC_S_EXT_CTRLS failed while tilting down , reason = %s" , strerror (errno));
        return -1;
    }else{
        __android_log_print(ANDROID_LOG_ERROR , TAG , "VIDIOC_S_EXT_CTRLS success while tilting down , after value = %d" , ctrl.value);
    }
    return 0;
}

可以看到控制的核心函数是linux的ioctl,传入的参数是v4l2_control,这个结构体里面有什么东西呢,可以看看其定义(https://www.linuxtv.org/downloads/legacy/video4linux/API/V4L2_API/spec-single/v4l2.html#v4l2-control):

image

结构体里面就id和value,按照UVC协议赋值,其他的交给ioctrl就可以了。此时ioctrl到了底层到了哪里呢?其实到了drivers/media/usb/uvc/uvc_v4l2.c文件中的uvc_v4l2_do_ioctl方法,在这个方法中需要关注的是以下几行代码:

case VIDIOC_S_CTRL:
{
    struct v4l2_control *ctrl = arg;
    struct v4l2_ext_control xctrl;

    memset(&xctrl, 0, sizeof xctrl);
    xctrl.id = ctrl->id;
    xctrl.value = ctrl->value;

    uvc_ctrl_begin(video);
    ret = uvc_ctrl_set(video, &xctrl);
    if (ret < 0) {
        uvc_ctrl_rollback(video);
        return ret;
    }
    ret = uvc_ctrl_commit(video);
    break;
}

v4l2_prio_check应该是检查是否有执行的优先级,没有执行的优先级的话返回当前繁忙设备,如下:

int v4l2_prio_check(struct v4l2_prio_state*global, enum v4l2_priority local)
{
    return(local < v4l2_prio_max(global)) ? -EBUSY : 0;
}

接下来的执行顺序与数据库的插入操作比较类似,开始执行控制,设置相关参数,检查控制结果,失败的话回滚操作,否则提交执行,同时记住当前的设定值。
第二个需要关注的是实现获取当前控制参数,getControlValue,有两种写法,一种是不用循环,一种是循环遍历UVC控制参数,查看设备当前控制参数的值,如下:

int getControlValues(int controlId){
    //an array of v4l2_ext_control
    struct v4l2_ext_control clist[1];
    struct v4l2_ext_controls ctrls;
 
    CLEAR(clist);
    CLEAR(ctrls);
 
    clist[0].id    = controlId;
    clist[0].value = 0;
 
    //v4l2_ext_controls with list of v4l2_ext_control
    ctrls.ctrl_class = V4L2_CTRL_CLASS_CAMERA;
    ctrls.count = 1;
    ctrls.controls = clist;
 
    //read back the value
    if (-1 == ioctl (fd, VIDIOC_G_EXT_CTRLS, &ctrls))
    {
        __android_log_print(ANDROID_LOG_ERROR,TAG,"get current value failed fd = %d,reason=%s" , fd,strerror(errno));
        return -1;
    }
    __android_log_print(ANDROID_LOG_ERROR,TAG,"get before value success , %d" , clist[0].value);
    return clist[0].value;
}
 
int getControlValue(int controlId){
    //an array of v4l2_ext_control
    struct v4l2_control ctrl;
 
    CLEAR(ctrl);
 
    ctrl.id    = controlId;
    ctrl.value = 0;
 
    //read back the value
    if (-1 == ioctl (fd, VIDIOC_G_CTRL, &ctrl))
    {
        __android_log_print(ANDROID_LOG_ERROR,TAG,"get current value failed fd = %d,reason=%s" , fd,strerror(errno));
        return -1;
    }
    __android_log_print(ANDROID_LOG_ERROR,TAG,"get before value success , %d" , ctrl.value);
    return ctrl.value;
}

这里获取控制id当前的值也是在drivers/media/usb/uvc/uvc_v4l2.c的uvc_v4l2_do_ioctl方法中,其中使用的获取参数是VIDIOC_G_EXT_CTRLS,其实查询单个控制参数的值用VIDIOC_G_CTRL也可以,只不过VIDIOC_G_EXT_CTRLS是多个控制参数一起查找,在底层源码里面是循环遍历查找,而VIDIOC_G_CTRL是直接查找,相关源码如下:

case VIDIOC_G_CTRL:
{
    struct v4l2_control *ctrl = arg;
    struct v4l2_ext_control xctrl;

    memset(&xctrl, 0, sizeof xctrl);
    xctrl.id = ctrl->id;

    uvc_ctrl_begin(video);
    ret = uvc_ctrl_get(video, &xctrl);
    uvc_ctrl_rollback(video);
    if (ret >= 0)
        ctrl->value = xctrl.value;
    break;
}
case VIDIOC_G_EXT_CTRLS:
{
    struct v4l2_ext_controls *ctrls = arg;
    struct v4l2_ext_control *ctrl = ctrls->controls;
    unsigned int i;

    uvc_ctrl_begin(video);
    for (i = 0; i < ctrls->count; ++ctrl, ++i) {
        ret = uvc_ctrl_get(video, ctrl);
        if (ret < 0) {
            uvc_ctrl_rollback(video);
            ctrls->error_idx = i;
            return ret;
        }
    }
    ctrls->error_idx = 0;
    ret = uvc_ctrl_rollback(video);
    break;
}

有关v4l2_ext_controls和v4l2_ext_control的控制参数见下图:


image

image

现在看看判断当前摄像头是否支持PTZ控制的函数,实际上是遍历摄像头的控制id,查看是否有PTZ相关的控制id,函数中定义的几个变量值定义如下:

const char* CONTROL_FLAG_PAN = "Pan(Absolute)";
const char* CONTROL_FLAG_TILT = "Tilt(Absolute)";
const char* CONTROL_FLAG_ZOOM = "Zoom,Absolute";
const int PAN_SPEED_ID = 10094880;
const int TILT_SPEED_ID = 10094881;
const int ZOOM_RELATIVE_ID = 10094863;

详细的函数如下:

jboolean queryControls(){
    jint canControl = 0;
    __android_log_print(ANDROID_LOG_ERROR , TAG , "设备号 = %d" , fd);
 
    struct v4l2_queryctrl qctrl;
    qctrl.id = V4L2_CTRL_CLASS_CAMERA | V4L2_CTRL_FLAG_NEXT_CTRL;
    int i = ioctl(fd, VIDIOC_QUERYCTRL, &qctrl);
    LOGE("error %d, %s , %d",errno, strerror (errno) , i);
    while (0 == i){
        __android_log_print(ANDROID_LOG_ERROR , TAG , "开始查找");
        if (V4L2_CTRL_ID2CLASS(qctrl.id) != V4L2_CTRL_CLASS_CAMERA)
            continue;
 
        if(strcmp(qctrl.name , CONTROL_FLAG_PAN) == 0 ){
            ++canControl;
            panMin = qctrl.maximum;
            panMax = qctrl.minimum;
            __android_log_print(ANDROID_LOG_ERROR , TAG , "panMin=%d , panMax=%d , panStep=%d" , panMin , panMax , qctrl.step);
        }
        if(strcmp(qctrl.name , CONTROL_FLAG_TILT) == 0){
            ++canControl;
            tiltMin = qctrl.maximum;
            tiltMax = qctrl.minimum;
            __android_log_print(ANDROID_LOG_ERROR , TAG , "tiltMin=%d , tiltMax=%d , tiltStep=%d" , tiltMin , tiltMax , qctrl.step);
        }
        if(strcmp(qctrl.name , CONTROL_FLAG_ZOOM) == 0){
            ++canControl;
            zoomMin = qctrl.maximum;
            zoomMax = qctrl.minimum;
            __android_log_print(ANDROID_LOG_ERROR , TAG , "zoomMin=%d , zoomMax=%d , zoomStep=%d" , zoomMin , zoomMax , qctrl.step);
        }
 
        __android_log_print(ANDROID_LOG_ERROR , TAG , "找到的控制函数是%s" , qctrl.name);
        __android_log_print(ANDROID_LOG_ERROR , TAG , "继续查找");
        __android_log_print(ANDROID_LOG_ERROR , TAG , "id = %d" , qctrl.id);
        __android_log_print(ANDROID_LOG_ERROR , TAG , "Next_Ctrl = %x" , V4L2_CTRL_FLAG_NEXT_CTRL);
        __android_log_print(ANDROID_LOG_ERROR , TAG , "Camera_Class = %x" , V4L2_CTRL_CLASS_CAMERA);
 
        qctrl.id |= V4L2_CTRL_FLAG_NEXT_CTRL;
 
        __android_log_print(ANDROID_LOG_ERROR , TAG , "id+ = %x" , qctrl.id);
 
        i = ioctl(fd, VIDIOC_QUERYCTRL, &qctrl);
        if(i != 0){
            __android_log_print(ANDROID_LOG_ERROR, TAG,"uvcioc ctrl add error: errno=%d (reason=%s)\n", errno,strerror(errno));
        }
    }
    //如果存在ptz控制的话,应该会有Pan,Tilt,Zoom字符串,变量自加三次
    return canControl == 3;
}

这里可以看到这个函数里面详细输出了支持哪些控制参数,各个控制参数的最大值,最小值,步长等信息,这些查找最后在底层的调用是在drivers\media\usb\uvc\uvc_ctrl.c的__uvc_query_v4l2_ctrl函数中,具体的描述在前一篇文章中有详细的讲述。

上述代码编写完毕,写MK文件,如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE    := UVCCamera
LOCAL_SRC_FILES := UVCCamera_Rel.cpp
LOCAL_LDLIBS    := -llog -ljnigraphics
include $(BUILD_SHARED_LIBRARY)

app层的代码编写完毕,在AS中编译运行生成so文件,在app中就可以使用了。同时要注意的是,这个app的权限级别要比较高,不然可能没有权限执行ioctrl。

另外这个代码中还涉及到了打开/dev/video0文件的操作,如下:

int opendevice(int i) {
    struct stat st;
 
    sprintf(dev_name,"/dev/video%d",i);
 
    if (-1 == stat (dev_name, &st)) {
        __android_log_print(ANDROID_LOG_ERROR , TAG , "Cannot identify '%s': %d, %s", dev_name, errno, strerror (errno));
        LOGE("Cannot identify '%s': %d, %s", dev_name, errno, strerror (errno));
        return ERROR_LOCAL;
    }
    __android_log_print(ANDROID_LOG_ERROR , TAG , "device is %s", dev_name);
    if (!S_ISCHR (st.st_mode)) {
        LOGE("%s is no device", dev_name);
        __android_log_print(ANDROID_LOG_ERROR , TAG , "%s is no device", dev_name);
        return ERROR_LOCAL;
    }
 
    fd = open (dev_name, O_RDWR | O_NONBLOCK, 0);
 
    if (-1 == fd) {
        LOGE("Cannot open '%s': %d, %s", dev_name, errno, strerror (errno));
        return ERROR_LOCAL;
    }else{
        LOGE("open device success %d" , fd);
 
        queryControls();
    }
    return SUCCESS_LOCAL;
}

至此,android UVC控制云台摄像头系列完成,在整个方案研发的过程中最大的阻碍是android kernel不支持相对控制,需要搜索相关资料自己去打patch,打了patch之后还要不断的调试试错;另外一个阻碍是摄像头厂商不同UVC版本的摄像头,让android适配的过程中吃尽苦头,如果碰上乐于合作的厂商问题就会很快解决。在这个技术实现的过程中深刻体会到有时候不逼自己一把,都不知道自己有多大的潜力。

在这个过程中涉及到了kernel层源码的解读,kernel源码的修改和调试,kernel层的编译和打包刷机,android jni代码等,整个过程还有boss每天一次的问候,查询进度。
微信公众号:Android部落格

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容