Android 9.0 对后台程序使用Mic、Camera做了限制?

01

发现问题

在android9.0 上修改原生的SnapdragonCamera 应用,需要修改成能支持后台录像,也就是Activity处于OnPause状态的时候,还是在录像(现在市面上支持后台录像的应用,基本都是在后台service中去开启录像,这个就不在本文的讨论场景了)。

原先的代码预览采用的surfaceView,后面修改成了使用textureView。不过发现按home键退出返回到桌面的时候,后台只能录制1分钟的视频,然后上报了camera error 3的错误。

这个error 3的错误,就是camera关闭了,camera 处于不可用状态下上报的异常。

02

问题分析

1 )根据上报的error来入手,这个error是从hardware层上报,经过CameraService.cpp,然后notify给上层,我们先来看下Camera error状态有哪些:目前我们遇到的这个error 3,就是camera disable了,那就是那里把camera给关了。

2)第一步,我先在应用代码添加了日志,确认了上层没有自动关闭camera的操作。

接下来第二步是在CameraService.cpp中notifyError 的地方添加多点日志打印,确认是哪里回调的error信息。

根据打印的信息,跟踪下相关代码,发现CameraService这边在接收到uid 相关状态发生了变化的时候,是自动做了notify error 3,然后执行了disconnect Camera的操作。

void CameraService::UidPolicy::updateOverrideUid(uid_t uid, String16 callingPackage,
        bool active, bool insert) {
    ...
//判断到uid处于非active状态,然后执行了上报error3,并disconnect camera
    if (wasActive != isActive && !isActive) {
        sp<CameraService> service = mService.promote();
        if (service != nullptr) {
            service->blockClientsForUid(uid);
        }
    }
}


void CameraService::blockClientsForUid(uid_t uid) {
    const auto clients = mActiveClientManager.getAll();
    for (auto& current : clients) {
        if (current != nullptr) {
            const auto basicClient = current->getValue();
            if (basicClient.get() != nullptr && basicClient->getClientUid() == uid) {
                basicClient->block();
            }
        }
    }
}


void CameraService::BasicClient::block() {
    ...
    notifyError(hardware::camera2::ICameraDeviceCallbacks::ERROR_CAMERA_DISABLED, resultExtras);
    disconnect();
}




void CameraService::Client::notifyError(int32_t errorCode,
        const CaptureResultExtras& resultExtras) {
    (void) resultExtras;
    if (mRemoteCallback != NULL) {
        int32_t api1ErrorCode = CAMERA_ERROR_RELEASED;
        if (errorCode == hardware::camera2::ICameraDeviceCallbacks::ERROR_CAMERA_DISABLED) {
            api1ErrorCode = CAMERA_ERROR_DISABLED;
        }
        mRemoteCallback->notifyCallback(CAMERA_MSG_ERROR, api1ErrorCode, 0);
    } else {
        ALOGE("mRemoteCallback is NULL!!");
    }
}

3)根据上面的信息,网上查了下相关资料(https://blog.csdn.net/leonxu_sjtu/article/details/82787997),android 9.0上的确对后台程序使用mic、camera做了限制,如果uid处于idle状态,则会断开相应的硬件设备连接。

03

修改方案

因为我们可以修改framework层代码,所以在CameraService.cpp中把自动断开camera连接的地方给屏蔽掉.

自定义property属性ro.backgroundrecord.enable ,然后根据property属性值来判断是否需要屏蔽Android9.0的这个限制。

+++ b/frameworks/av/services/camera/libcameraservice/CameraService.cpp


@@ -2425,7 +2425,15 @@ void CameraService::UidPolicy::onUidIdle(uid_t uid, bool /* disabled */) {
     if (deleted) {
         sp<CameraService> service = mService.promote();
         if (service != nullptr) {
-            service->blockClientsForUid(uid);
+            char value[PROPERTY_VALUE_MAX];
+            property_get("ro.backgroundrecord.enable", value, "0");
+            int32_t enable = atoi(value);


+            if(enable){
+              ALOGV("---- enable background record,not need to block uid!!!");
+            }else{
+              service->blockClientsForUid(uid);
+            }
         }
     }
 }

推荐阅读:

Android Camera2 对焦梳理

Android Camera 测光梳理

高通Camera数字成像系统简介

一篇文章带你了解Android 最新Camera框架

这可能是介绍Android UvcCamera最详细的文章了

深圳上班,

生活简简单单,

14年开始从事Android Camera相关软件开发工作,

做过车载、手机、执法记录仪......

公众号记录生活和工作的点滴,

关注“小驰笔记”,期待和你相遇~

客官,点个赞呗~

posted @ 2021-08-09 08:00  小驰行动派  阅读(47)  评论(0编辑  收藏  举报  来源