android閃光燈源碼分析

? ? ?

目錄

一、APP層源碼分析

??????????????????二,framework層代碼分析

???????????????2.1 binder溯源


????????這幾天擼了android11 aosp閃光燈源碼,本著前人栽樹后人乘涼的原則,有志于android系統開發的新同學們提供一盞明燈,照亮你們前行。

? ? ? ? 本人擼代碼風格,喜歡從app擼到kernel,啟航出發。

一、APP層源碼分析

? ? ? ? 下拉狀態欄,可見“手電筒”快捷開關,點擊后開啟手電筒,UI見下圖,

這部分代碼,在目錄:frameworks/base/packages/SystemUI/src/com/android/systemui/qs/tiles/FlashlightTile.java,點擊按鈕回調,

    protected void handleClick() {if (ActivityManager.isUserAMonkey()) {return;}boolean newState = !mState.value;refreshState(newState);//點擊“手電筒”核心核數mFlashlightController.setFlashlight(newState);}

通過?mFlashlightController.setFlashlight(newState)走到文件

frameworks/base/packages/SystemUI/src/com/android/systemui/statusbar/policy/FlashlightControllerImpl.java,

    public void setFlashlight(boolean enabled) {boolean pendingError = false;synchronized (this) {if (mCameraId == null) return;if (mFlashlightEnabled != enabled) {mFlashlightEnabled = enabled;try {//核心調用mCameraManager.setTorchMode(mCameraId, enabled);} catch (CameraAccessException e) {Log.e(TAG, "Couldn't set torch mode", e);mFlashlightEnabled = false;pendingError = true;}}}dispatchModeChanged(mFlashlightEnabled);if (pendingError) {dispatchError();}}

通過mCameraManager.setTorchMode(mCameraId, enabled)走到frameworks/base/core/java/android/hardware/camera2/CameraManager.java,

    public void setTorchMode(@NonNull String cameraId, boolean enabled)throws CameraAccessException {if (CameraManagerGlobal.sCameraServiceDisabled) {throw new IllegalArgumentException("No cameras available on device");}CameraManagerGlobal.get().setTorchMode(cameraId, enabled);}

CameraManagerGlobal.get().setTorchMode(cameraId, enabled)走到,

        public void setTorchMode(String cameraId, boolean enabled) throws CameraAccessException {synchronized(mLock) {if (cameraId == null) {throw new IllegalArgumentException("cameraId was null");}ICameraService cameraService = getCameraService();if (cameraService == null) {throw new CameraAccessException(CameraAccessException.CAMERA_DISCONNECTED,"Camera service is currently unavailable");}try {cameraService.setTorchMode(cameraId, enabled, mTorchClientBinder);} catch(ServiceSpecificException e) {throwAsPublicException(e);} catch (RemoteException e) {throw new CameraAccessException(CameraAccessException.CAMERA_DISCONNECTED,"Camera service is currently unavailable");}}}
????????ICameraService cameraService = getCameraService();看到這句是不是很熟悉,android獲取系統服務,跨進程調用。只是和我們app開發常用的 不同,這里直接拿到camera服務的proxy。對下面這行代碼不理解的,可以學習下android binder機制,本質就是拿到camera服務的proxy
ICameraService cameraService = ICameraService.Stub.asInterface(cameraServiceBinder);

? ? ? ? 原來操作“手電筒”的不是單獨模塊,而是通過camera模塊來控制的,666。

二,framework層代碼分析

2.1 binder溯源

? ? ? ? 由第一部分分析可知,“手電筒”通過cameraservice來操作,結合binder知識,可以在ICameraService.aidl接口定義中找到,

void setTorchMode(String cameraId, boolean enabled, IBinder clientBinder);

aidl文件會自動生成java文件,再通過jni調用到cpp文件,最終在ICameraService.cpp中實現了client的調用,

::android::binder::Status BpCameraService::setTorchMode(const ::android::String16& cameraId, bool enabled, const ::android::sp<::android::IBinder>& clientBinder) {::android::Parcel _aidl_data;::android::Parcel _aidl_reply;::android::status_t _aidl_ret_status = ::android::OK;::android::binder::Status _aidl_status;_aidl_ret_status = _aidl_data.writeInterfaceToken(getInterfaceDescriptor());if (((_aidl_ret_status) != (::android::OK))) {goto _aidl_error;}_aidl_ret_status = _aidl_data.writeString16(cameraId);if (((_aidl_ret_status) != (::android::OK))) {goto _aidl_error;}_aidl_ret_status = _aidl_data.writeBool(enabled);if (((_aidl_ret_status) != (::android::OK))) {goto _aidl_error;}_aidl_ret_status = _aidl_data.writeStrongBinder(clientBinder);if (((_aidl_ret_status) != (::android::OK))) {goto _aidl_error;}_aidl_ret_status = remote()->transact(::android::IBinder::FIRST_CALL_TRANSACTION + 15 /* setTorchMode */, _aidl_data, &_aidl_reply);if (UNLIKELY(_aidl_ret_status == ::android::UNKNOWN_TRANSACTION && ICameraService::getDefaultImpl())) {return ICameraService::getDefaultImpl()->setTorchMode(cameraId, enabled, clientBinder);}if (((_aidl_ret_status) != (::android::OK))) {goto _aidl_error;}_aidl_ret_status = _aidl_status.readFromParcel(_aidl_reply);if (((_aidl_ret_status) != (::android::OK))) {goto _aidl_error;}if (!_aidl_status.isOk()) {return _aidl_status;}_aidl_error:_aidl_status.setFromStatusT(_aidl_ret_status);return _aidl_status;
}

通過remote()->transact()發起IPC請求,事務碼FIRST_CALL_TRANSACTION + 15對應setTorchMode方法,數據通過_aidl_dataParcel傳遞,執行到CameraService.cpp文件,

Status CameraService::setTorchMode(const String16& cameraId, bool enabled,const sp<IBinder>& clientBinder) {Mutex::Autolock lock(mServiceLock);ATRACE_CALL();if (enabled && clientBinder == nullptr) {ALOGE("%s: torch client binder is NULL", __FUNCTION__);return STATUS_ERROR(ERROR_ILLEGAL_ARGUMENT,"Torch client Binder is null");}String8 id = String8(cameraId.string());int uid = CameraThreadState::getCallingUid();if (shouldRejectSystemCameraConnection(id)) {return STATUS_ERROR_FMT(ERROR_ILLEGAL_ARGUMENT, "Unable to set torch mode"" for system only device %s: ", id.string());}// verify id is valid.auto state = getCameraState(id);if (state == nullptr) {ALOGE("%s: camera id is invalid %s", __FUNCTION__, id.string());return STATUS_ERROR_FMT(ERROR_ILLEGAL_ARGUMENT,"Camera ID \"%s\" is a not valid camera ID", id.string());}StatusInternal cameraStatus = state->getStatus();if (cameraStatus != StatusInternal::PRESENT &&cameraStatus != StatusInternal::NOT_AVAILABLE) {ALOGE("%s: camera id is invalid %s, status %d", __FUNCTION__, id.string(), (int)cameraStatus);return STATUS_ERROR_FMT(ERROR_ILLEGAL_ARGUMENT,"Camera ID \"%s\" is a not valid camera ID", id.string());}{Mutex::Autolock al(mTorchStatusMutex);TorchModeStatus status;status_t err = getTorchStatusLocked(id, &status);if (err != OK) {if (err == NAME_NOT_FOUND) {return STATUS_ERROR_FMT(ERROR_ILLEGAL_ARGUMENT,"Camera \"%s\" does not have a flash unit", id.string());}ALOGE("%s: getting current torch status failed for camera %s",__FUNCTION__, id.string());return STATUS_ERROR_FMT(ERROR_INVALID_OPERATION,"Error updating torch status for camera \"%s\": %s (%d)", id.string(),strerror(-err), err);}if (status == TorchModeStatus::NOT_AVAILABLE) {if (cameraStatus == StatusInternal::NOT_AVAILABLE) {ALOGE("%s: torch mode of camera %s is not available because ""camera is in use", __FUNCTION__, id.string());return STATUS_ERROR_FMT(ERROR_CAMERA_IN_USE,"Torch for camera \"%s\" is not available due to an existing camera user",id.string());} else {ALOGE("%s: torch mode of camera %s is not available due to ""insufficient resources", __FUNCTION__, id.string());return STATUS_ERROR_FMT(ERROR_MAX_CAMERAS_IN_USE,"Torch for camera \"%s\" is not available due to insufficient resources",id.string());}}}{// Update UID map - this is used in the torch status changed callbacks, so must be done// before setTorchModeMutex::Autolock al(mTorchUidMapMutex);if (mTorchUidMap.find(id) == mTorchUidMap.end()) {mTorchUidMap[id].first = uid;mTorchUidMap[id].second = uid;} else {// Set the pending UIDmTorchUidMap[id].first = uid;}}status_t err = mFlashlight->setTorchMode(id, enabled);if (err != OK) {int32_t errorCode;String8 msg;switch (err) {case -ENOSYS:msg = String8::format("Camera \"%s\" has no flashlight",id.string());errorCode = ERROR_ILLEGAL_ARGUMENT;break;default:msg = String8::format("Setting torch mode of camera \"%s\" to %d failed: %s (%d)",id.string(), enabled, strerror(-err), err);errorCode = ERROR_INVALID_OPERATION;}ALOGE("%s: %s", __FUNCTION__, msg.string());return STATUS_ERROR(errorCode, msg.string());}{// update the link to client's deathMutex::Autolock al(mTorchClientMapMutex);ssize_t index = mTorchClientMap.indexOfKey(id);if (enabled) {if (index == NAME_NOT_FOUND) {mTorchClientMap.add(id, clientBinder);} else {mTorchClientMap.valueAt(index)->unlinkToDeath(this);mTorchClientMap.replaceValueAt(index, clientBinder);}clientBinder->linkToDeath(this);} else if (index != NAME_NOT_FOUND) {mTorchClientMap.valueAt(index)->unlinkToDeath(this);}}int clientPid = CameraThreadState::getCallingPid();const char *id_cstr = id.c_str();const char *torchState = enabled ? "on" : "off";ALOGI("Torch for camera id %s turned %s for client PID %d", id_cstr, torchState, clientPid);logTorchEvent(id_cstr, torchState , clientPid);return Status::ok();
}

其中,關鍵句status_t err = mFlashlight->setTorchMode(id, enabled);再由hal層交互可知,?
最終通過CameraProviderManager查詢HAL實現的setTorchMode()接口來控制device。

status_t CameraProviderManager::setTorchMode(const std::string &id, bool enabled) {std::lock_guard<std::mutex> lock(mInterfaceMutex);auto deviceInfo = findDeviceInfoLocked(id);if (deviceInfo == nullptr) return NAME_NOT_FOUND;// Pass the camera ID to start interface so that it will save it to the map of ICameraProviders// that are currently in use.sp<ProviderInfo> parentProvider = deviceInfo->mParentProvider.promote();if (parentProvider == nullptr) {return DEAD_OBJECT;}const sp<provider::V2_4::ICameraProvider> interface = parentProvider->startProviderInterface();if (interface == nullptr) {return DEAD_OBJECT;}saveRef(DeviceMode::TORCH, deviceInfo->mId, interface);return deviceInfo->setTorchMode(enabled);
}

最終,通過deviceInfo->setTorchMode(enabled)來實現對硬件的操作。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/90554.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/90554.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/90554.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

文心一言4.5開源部署指南及文學領域測評

&#x1f4dd;個人主頁&#xff1a;哈__ 期待您的關注 目錄 一、引言 二、文心一言開源模型 2.1 MoE架構 2.2 文心一言MoE架構 三、文心一言稠密模型部署 3.1 產品選擇 3.2 環境選擇 3.3 Python3.12安裝 3.3 PaddlePaddle-GPU安裝 3.4 FastDeploy-GPU安裝 ?編輯3.…

深入探討 C++ 中的浮點數數據類型

核心概念&#xff1a;IEEE 754 標準 C 中的浮點數&#xff08;float, double, long double&#xff09;在絕大多數現代系統上遵循 IEEE 754 標準。這個標準定義了浮點數在內存中的二進制表示方式、運算規則、特殊值&#xff08;如無窮大、NaN&#xff09;等。數據類型與精度 fl…

相機:以鼠標點為中心縮放(使用OpenGL+QT開發三維CAD)

很多軟件中&#xff08;Auto CAD、ODA等&#xff09;支持以鼠標點為中心進行放縮操作&#xff0c;有什么黑科技嗎&#xff1f; 本章節為相機原理和實現的補充內容&#xff0c;支持鼠標放縮時以鼠標點為中心進行放縮。 對應視頻課程已上線&#xff0c;歡迎觀看和支持~ https:…

??XAMPP安全升級指南:修復CVE-2024-4577漏洞,從PHP 8.2.12升級至PHP 8.4.10??

??1. 背景與漏洞概述?? 近期,PHP官方披露了一個高危漏洞 ??CVE-2024-4577??,該漏洞影響PHP 8.2.x及更早版本,可能導致遠程代碼執行(RCE)或信息泄露。由于XAMPP默認捆綁的PHP版本(如8.2.12)可能受此漏洞影響,建議用戶盡快升級至最新的??PHP 8.4.10??(或官…

ES 壓縮包安裝

以下是 Elasticsearch (ES) 通過 .tar.gz 壓縮包安裝的詳細步驟&#xff08;適用于 Linux/macOS 系統&#xff09;&#xff1a; 1. 準備工作 1.1 檢查系統依賴 Java 環境&#xff1a;ES 需要 JDK&#xff0c;推薦 OpenJDK 11/17&#xff08;ES 7.x/8.x 兼容版本&#xff09;。…

RoboRefer:面向機器人視覺-語言模型推理的空間參考

25年6月來自北航、北大和北京智源的論文“RoboRefer: Towards Spatial Referring with Reasoning in Vision-Language Models for Robotics”。 空間參考是實體機器人與三維物理世界交互的基本能力。然而&#xff0c;即使有了強大的預訓練視覺-語言模型 (VLM)&#xff0c;近期方…

【Unity】MiniGame編輯器小游戲(十)連連看【Link】

更新日期:2025年7月9日。 項目源碼:獲取項目源碼 索引 連連看【Link】一、游戲最終效果二、玩法簡介三、正式開始1.定義游戲窗口類2.規劃游戲窗口、視口區域3.方塊 Block①.定義方塊類②.生成方塊所有類型③.生成連連看棋盤④.繪制方塊陣列4.連線 Line①.點擊方塊連線②.嘗試…

Enable ADB Debugging Before Connect

If you don’t enable Developer Options and turn on USB Debugging before plugging in the cable, adb devices won’t detect the phone because the Android system doesn’t trust the connection yet. Here’s what you need to do step-by-step to fix this:? 1. Enab…

從互聯網電腦遷移Dify到內網部署Dify方法記錄

一、在互聯網電腦上準備遷移文件1. 保存 Docker 鏡像# 獲取所有 Dify 相關鏡像&#xff08;根據實際容器名調整&#xff09; docker ps --filter "namedify" --format "{{.Image}}" | sort -u > dify-images.list# 保存鏡像為 .tar 文件 docker save $(…

【EGSR2025】材質+擴散模型+神經網絡相關論文整理隨筆(一)

MatSwap: Light-aware material transfers in images介紹任務&#xff1a;輸入一張拍攝圖像、示例材質紋理圖像&#xff08;這里跟BRDF無關&#xff0c;通常我們講到材質一般指的是SVBRDF&#xff0c;但是這里的材質指的只是紋理&#xff09;、用戶為拍攝圖像指定的遮罩區域&am…

餓了么el-upload上傳組件報錯:TypeError: ***.upload.addEventListener is not a function

在本地上傳沒有報這個錯誤&#xff0c;部署到服務器后會報這個錯誤&#xff0c;一開始以為是服務器配置等什么原因&#xff0c;但是一想這個報錯應該還是在前端&#xff0c;接口都還沒請求&#xff0c;不可能到后臺去&#xff0c;后面搜了好幾個AI也沒有找到想要的答案或解決方…

淘寶直播與開源鏈動2+1模式AI智能名片S2B2C商城小程序的融合發展研究

摘要&#xff1a;本文聚焦于淘寶直播這一以“網紅”內容為主的社交電商平臺&#xff0c;深入分析其特點與流量入口優勢。同時&#xff0c;引入開源鏈動21模式AI智能名片S2B2C商城小程序這一新興概念&#xff0c;探討二者融合的可能性與潛在價值。通過分析融合過程中的技術、市場…

【macos用鏡像站體驗】Claude Code入門使用教程和常用命令

一、下載安裝nodejs # macOS 用戶安裝nodejs brew update brew install node二、安裝官方Claude Code # 安裝 Claude Code npm install -g anthropic-ai/claude-code # 查看版本 claude --version三、正式使用&#xff08;國內鏡像站&#xff09; 今天發現的一個鏡像站&…

算法學習筆記:11.冒泡排序——從原理到實戰,涵蓋 LeetCode 與考研 408 例題

在排序算法的大家族中&#xff0c;冒泡排序是最基礎也最經典的算法之一。它的核心思想簡單易懂&#xff0c;通過重復地走訪待排序序列&#xff0c;一次比較兩個相鄰的元素&#xff0c;若它們的順序錯誤就把它們交換過來&#xff0c;直到沒有需要交換的元素為止。雖然冒泡排序的…

Linux小白學習基礎內容

記錄第一天重新學習2025/7/10 15&#xff1a;467/10 17&#xff1a;02這里面一個命令帶多個參數舉例&#xff08;多個參數之間用空格隔開&#xff09;ls&#xff08;命令&#xff09; ~ / /etc/&#xff08;參數&#xff09; :這里就是同時查看主機的家目錄&#xff0c;根目…

從零開始搭建深度學習大廈系列-2.卷積神經網絡基礎(5-9)

(1)本人挑戰手寫代碼驗證理論&#xff0c;獲得一些AI工具無法提供的收獲和思考&#xff0c;對于一些我無法回答的疑問請大家在評論區指教&#xff1b; (2)本系列文章有很多細節需要弄清楚&#xff0c;但是考慮到讀者的吸收情況和文章篇幅限制&#xff0c;選擇重點進行分享&…

【iOS設計模式】深入理解MVC架構 - 重構你的第一個App

目錄 一、MVC模式概述 二、創建Model層 1. 新建Person模型類 2. 實現Person類 三、重構ViewController 1. 修改ViewController.h 2. 重構ViewController.m 四、MVC組件詳解 1. Model&#xff08;Person類&#xff09; 2. View&#xff08;Storyboard中的UI元素&#x…

前端項目集成lint-staged

lint-staged (lint-staged) 這個插件可以只針對進入git暫存區中的代碼進行代碼格式檢查與修復&#xff0c;極大提升效率&#xff0c;避免掃描整個項目文件&#xff0c;代碼風格控制 eslint prettier stylelint 看這兩篇文章 前端項目vue3項目集成eslint9.x跟prettier 前端項…

李宏毅genai筆記:模型編輯

0 和post training的區別直接用post training的方法是有挑戰的&#xff0c;因為通常訓練資料只有一筆而且之后不管問什么問題&#xff0c;都有可能只是這個答案了1 模型編輯的評估方案 reliability——同樣的問題&#xff0c;需要是目標答案generalization——問題&#xff08;…

Oracle:union all和union區別

UNION ALL和UNION在Oracle中的主要區別體現在處理重復記錄、性能及結果排序上&#xff1a;處理重復記錄?UNION?&#xff1a;自動去除重復記錄&#xff0c;確保最終結果唯一。?UNION ALL?&#xff1a;保留所有記錄&#xff0c;包括完全重復的行。性能表現?UNION?&#xff…