安卓11-设置HDMI分辨率流程

安卓11中从设置-显示设置hdmi分辨率流程:framework层通过jni控制底层驱动实现,标准驱动模型


packages\apps\Settings\src\com\android\settings\display\HdmiSettings.javaprivate void updateResolution(final ITEM_CONTROL control, final int index) {showWaitingDialog(R.string.dialog_update_resolution);mEnableDisplayListener = false;new Thread() {@Overridepublic void run() {if (ITEM_CONTROL.CHANGE_RESOLUTION == control) {synchronized (mLock) {int display = mSelectDisplayInfo.getDisplayNo();DrmDisplaySetting.updateDisplayInfos();DrmDisplaySetting.updateDisplayModesInfo(mSelectDisplayInfo);int status = DrmDisplaySetting.getCurrentDpyConnState(display);mSelectDisplayInfo.setStatus(status);String[] modes = mSelectDisplayInfo.getOrginModes();Log.v(TAG, "display " + display + ", status=" + status + ", modes=" + modes);if (DPY_STATUS_CONNECTED == status && null != modes && modes.length > 0) {DrmDisplaySetting.setDisplayModeTemp(mSelectDisplayInfo, index);
//                            String mode = Arrays.asList(modes).get(index);
//                            DrmDisplaySetting.setMode(display, mode);if (USED_OFFON_RESOLUTION) {sendSwitchDeviceOffOnMsg(control, SWITCH_STATUS_OFF_ON);} else {Message message = new Message();message.what = MSG_SHOW_CONFIRM_DIALOG;message.obj = control;mHandler.sendMessageDelayed(message, 300);}} else {Message message = new Message();message.what = MSG_UPDATE_STATUS_UI;message.obj = ITEM_CONTROL.REFRESH_DISPLAY_STATUS_INFO;mHandler.sendMessage(message);}}}}}.start();}packages/apps/Settings/src/com/android/settings/display/DrmDisplaySetting.javapublic static void setDisplayModeTemp(DisplayInfo di, int index) {String mode = Arrays.asList(di.getOrginModes()).get(index);setMode(di.getDisplayNo(), mode);tmpSetMode = mode;}public static void setDisplayModeTemp(DisplayInfo di, int index) {String mode = Arrays.asList(di.getOrginModes()).get(index);setMode(di.getDisplayNo(), mode);tmpSetMode = mode;}public static void setMode(int display, String mode) {RkDisplayOutputManager manager = new RkDisplayOutputManager();int result = manager.updateDisplayInfos();int type = manager.getCurrentInterface(display);logd("setMode display=" + display + ", type=" + type + ", mode=" + mode);manager.setMode(display, type, mode);}
frameworks$ vim ./base/core/java/android/os/RkDisplayOutputManager.java
public void setMode(int display, int type, String mode) {String iface = typetoface(type);try {mService.setMode(display, iface, mode);}catch (Exception e) {Log.e(TAG, "Error set mode :" + e);return;}}public RkDisplayOutputManager() {IBinder b = ServiceManager.getService("drm_device_management");if(b == null) {Log.e(TAG, "Unable to connect to display device management service! - is it running yet?");return;}mService = IRkDisplayDeviceManagementService.Stub.asInterface(b);int mMainState = getCurrentDpyConnState(MAIN_DISPLAY);try {// Get main display interfaceString[] display_iface = mService.listInterfaces(MAIN_DISPLAY);if(DBG) Log.d(TAG, "main display iface num is " + display_iface.length);if(display_iface != null && display_iface.length > 0 &&  mMainState==DRM_MODE_CONNECTED) {m_main_iface = new int[display_iface.length];for(int i = 0; i < m_main_iface.length; i++) {if(DBG) Log.d(TAG, display_iface[i]);m_main_iface[i] = ifacetotype(display_iface[i]);}}./base/services/core/java/com/android/server/RkDisplayDeviceManagementService.javapublic void setMode(int display, String iface, String mode) {boolean isSameProperty = false;String lastMode;mdrmModes.setMode(display, iface, mode);}public RkDisplayDeviceManagementService(Context context) {mContext = context;mdrmModes = new RkDisplayModes();mdrmModes.init();IntentFilter hdmiFilter = new IntentFilter();hdmiFilter.addAction(ACTION_PLUGGED);mHdmiReceiver = new HdmiReceiver(mdrmModes);mContext.registerReceiver(mHdmiReceiver,hdmiFilter);}./base/services/core/java/com/android/server/rkdisplay/RkDisplayModes.javapublic void setMode(int display, String iface, String mode){int ifaceType = ifacetotype(iface);String[] mode_str;int idx=0;RkDisplayModes.RkPhysicalDisplayInfo info;Log.w(TAG, "setMode " + mode + " display " + display);if (mode.contains("Auto")) {nativeSetMode(display, ifaceType, mode);//通过本地方法实现} else {mode_str = mode.split("-");for (String mval: mode_str){Log.e(TAG, "setMode split:  " + mval);}if (mode_str.length != 2){return;}idx = Integer.parseInt(mode_str[1]);if (mMainDisplayInfos!=null && idx >= mMainDisplayInfos.length && display==0)idx=0;else if (mAuxDisplayInfos!=null && idx >= mAuxDisplayInfos.length && display==1)idx=0;if (display == 0)info = mMainDisplayInfos[idx];elseinfo = mAuxDisplayInfos[idx];StringBuilder builder = new StringBuilder();builder.append(info.width).append("x").append(info.height);
/*if (info.interlaceFlag == true)builder.append("i");elsebuilder.append("p");
*/builder.append("@");builder.append(String.format(Locale.ENGLISH, "%.2f", info.refreshRate));builder.append("-");builder.append(info.hsync_start);builder.append("-");builder.append(info.hsync_end);builder.append("-");builder.append(info.htotal);builder.append("-");builder.append(info.vsync_start).append("-").append(info.vsync_end).append("-").append(info.vtotal).append("-").append(String.format(Locale.ENGLISH, "%x", info.flags)).append("-").append(info.clock);nativeSetMode(display, ifaceType, builder.toString());}}
base/services/core/jni/com_android_server_rkdisplay_RkDisplayModes.cpp 
#include <rockchip/hardware/outputmanager/1.0/IRkOutputManager.h>  
static void nativeSetMode(JNIEnv* env, jobject obj, jint dpy, jint iface_type, jstring mode)
{const char* mMode = env->GetStringUTFChars(mode, NULL);if (mComposer != nullptr)mComposer->setMode(dpy, mMode); //这里又调用了hal层的方法env->ReleaseStringUTFChars(mode, mMode);
}
static void nativeInit(JNIEnv* env, jobject obj) {mComposer = IRkOutputManager::getService();//通过IRkOutputManager实现hdmi分辨率写入的if (mComposer != nullptr) {mComposer->initial();} else {ALOGD("nativeInit failed to get IRkOutputManager");}
}
./vendor/rockchip/hardware/interfaces/outputmanager/1.0/default/RkOutputManager.cpp
Return<Result> RkOutputManager::setMode(Display display, const hidl_string& mode)
{std::string modeStd(mode.c_str());mHwOutput->setMode(mHwOutput, display, modeStd.c_str());  //找到设备写入的地方return Result::OK;
}
IRkOutputManager* HIDL_FETCH_IRkOutputManager(const char* /* name */) {struct hw_output_device*   mHwOutput;const hw_module_t* hw_module = nullptr;int ret = hw_get_module(HW_OUTPUT_HARDWARE_MODULE_ID, &hw_module);//找到这个设备,获取设备,设备的具体实现在hw_output.cppif (ret == 0) {ret = hw_output_open(hw_module, &mHwOutput);if (ret == 0) {return new RkOutputManager(mHwOutput);} else {LOG(ERROR) << "Passthrough failed to load legacy HAL.";return nullptr;}}else {LOG(ERROR) << "hw_get_module " << HWC_HARDWARE_MODULE_ID<< " failed: " << ret;return nullptr;}
}
hardware/libhardware/modules/hw_output/hw_output.cpp
static int hw_output_device_open(const struct hw_module_t* module,const char* name, struct hw_device_t** device)
{int status = -EINVAL;if (!strcmp(name, HW_OUTPUT_DEFAULT_DEVICE)) {hw_output_private_t* dev = (hw_output_private_t*)malloc(sizeof(*dev));/* initialize our state here *///memset(dev, 0, sizeof(*dev));/* initialize the procs */dev->device.common.tag = HARDWARE_DEVICE_TAG;dev->device.common.version = HW_OUTPUT_DEVICE_API_VERSION_0_1;dev->device.common.module = const_cast<hw_module_t*>(module);dev->device.common.close = hw_output_device_close;dev->device.initialize = hw_output_initialize;dev->device.setMode = hw_output_set_mode; //这里传递一个函数指针,具体设置分辨率的地方
..........
static int hw_output_set_mode(struct hw_output_device* dev, int dpy, const char* mode)
{hw_output_private_t* priv = (hw_output_private_t*)dev;DrmConnector* conn = getValidDrmConnector(priv, dpy);BaseParameter* mBaseParameter = priv->mBaseParmeter;char property[PROPERTY_VALUE_MAX];std::string propertyStr;propertyStr = getPropertySuffix(priv, "persist.vendor.resolution.", dpy);ALOGD("nativeSetMode %s display %d", mode, dpy);if (strcmp(mode, property) !=0) {property_set(propertyStr.c_str(), mode);updateTimeline();struct disp_info info;float vfresh=0.0f;int slot = 0;mBaseParameter->get_disp_info(conn->get_type(), conn->connector_id(), &info);slot = findSuitableInfoSlot(&info, conn->get_type(), conn->connector_id());info.screen_info[slot].type = conn->get_type();info.screen_info[slot].id = conn->connector_id();if (strncmp(mode, "Auto", 4) != 0 && strncmp(mode, "0x0p0-0", 7) !=0) {sscanf(mode,"%dx%d@%f-%d-%d-%d-%d-%d-%d-%x-%d",&info.screen_info[slot].resolution.hdisplay, &info.screen_info[slot].resolution.vdisplay,&vfresh, &info.screen_info[slot].resolution.hsync_start,&info.screen_info[slot].resolution.hsync_end,&info.screen_info[slot].resolution.htotal,&info.screen_info[slot].resolution.vsync_start,&info.screen_info[slot].resolution.vsync_end, &info.screen_info[slot].resolution.vtotal,&info.screen_info[slot].resolution.flags, &info.screen_info[slot].resolution.clock);info.screen_info[slot].resolution.vrefresh = (int)vfresh;} else {info.screen_info[slot].feature|= RESOLUTION_AUTO;memset(&info.screen_info[slot].resolution, 0, sizeof(info.screen_info[slot].resolution));}mBaseParameter->set_disp_info(conn->get_type(), conn->connector_id(), &info);}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/698295.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【主题广范|见刊快】2024年科技,绿色能源和可持续发展国际会议(ICTGESD 2024)

【主题广范|见刊快】2024年科技&#xff0c;绿色能源和可持续发展国际会议&#xff08;ICTGESD 2024&#xff09; 2024 International Conference on Technology, Green Energy, and Sustainable Development ⊙会议简介&#xff1a; 2024年科技&#xff0c;绿色能源和可持续发…

音视频技术-声反馈啸叫的产生与消除

目录 1.均衡调节: 2.移频法: 3.移相法: 4.比较法: 在扩音系统中,产生啸叫危害很大,一方面影响会议、演出等活动的正常进行,另一方面严重的啸叫会导致音响设备的损坏。 “啸叫”是“声反馈”的俗称,形成的机制复杂,消除的手段多样,专业调音师也对

Unity中URP实现水效果(水的深度)

文章目录 前言一、搭建预备场景1、新建一个面片&#xff0c;使其倾斜一个角度&#xff0c;来模拟水底和岸边的效果2、随便创建几个物体&#xff0c;作为与水面接触的物体3、再新建一个面片&#xff0c;作为水面 二、开始编写水体的Shader效果1、新建一个URP基础Shader2、把水体…

最优传输(Optimal Transport)

最优传输&#xff08;Optimal Transport&#xff09;是一种数学理论和计算方法&#xff0c;用于描述两个概率分布之间的距离或者对应关系。它的核心概念是如何以最佳方式将一组资源&#xff08;如质量、能量等&#xff09;从一个位置传输到另一个位置。 基本概念&#xff1a; …

Qt 5.12.12 如何使用 cmake

首先 我们需要 Qt6 的help 这里面有所有使用cmake的说明 Qt6 help 下载地址 : 链接: https://pan.baidu.com/s/1jhwdYLtFaAa7tq5Gly0gAQ?pwd6666 提取码: 6666 然后 通过 qt creator 安装help qt-creater打开 Tools -> options -> Help -> Documentation 选择 …

MongoDB聚合运算符:$avg

$avg运算符返回给定数值的平均值 $avg可用于以下阶段&#xff1a; $addFields阶段(从MongoDB 3.4开始可用)$bucket阶段$bucketAuto阶段$group阶段包含$expr表达式的$match阶段$project阶段$replaceRoot阶段(从MongoDB 3.4开始可用)$replaceWith阶段(从MongoDB 4.2开始可用)$s…

【Linux】设置线程名字prctl、pthread_setname_np

1、prctl 设置线程名字 1.1 说明 prctl 对线程和进程的操作,详见:https://man7.org/linux/man-pages/man2/prctl.2.html #include <sys/prctl.h>int prctl(int option , ... /* unsigned long arg2 , unsigned long arg3 , unsigned long arg4 , unsigned long arg5…

Java编程实战:构建医疗信息管理新平台

✍✍计算机编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java实战 |…

代码随想录算法训练营day25|216.组合总和III

216.组合总和III 题目链接/文章讲解&#xff1a;代码随想录 视频讲解&#xff1a;和组合问题有啥区别&#xff1f;回溯算法如何剪枝&#xff1f;| LeetCode&#xff1a;216.组合总和III_哔哩哔哩_bilibili 跟77题差不多&#xff0c;要搞清楚k确定了递归的深度 依旧用回溯三部…

cesium 高德路网数据加载到WGS-84坐标系,解决偏移问题

首先用到cesium.map.min.js 修改cesium.map.min.js中数据类型的瓦片路径 比如类型是img 搜索M.GCJ02ToWGS84&#xff0c;初步定位&#xff0c;确定是火星坐标系转wGS-84&#xff0c;然后修改 var S{img:"//webst{s}.is.autonavi.com/appmaptile?style6&x{x}&y{y…

sql注入 [极客大挑战 2019]FinalSQL1

打开题目 点击1到5号的结果 1号 2号 3号 4号 5号 这里直接令传入的id6 传入id1^1^1 逻辑符号|会被检测到&#xff0c;而&感觉成了注释符&#xff0c;&之后的内容都被替换掉了。 传入id1|1 直接盲注比较慢&#xff0c;还需要利用二分法来编写脚本 这里利用到大佬的脚…

英伟达推出免训练,可生成连贯图片的文生图模型

目前&#xff0c;多数文生图模型皆使用的是随机采样模式&#xff0c;使得每次生成的图像效果皆不同&#xff0c;在生成连贯的图像方面非常差。 例如&#xff0c;想通过AI生成一套图像连环画&#xff0c;即便使用同类的提示词也很难实现。虽然DALLE 3和Midjourney可以对图像实现…

linux0.11 源码阅读 head.s setup.s bootsect.s加载位置

从github上下载linux0.11源码 linux0.11源码 将0x10000处的代码往下复制到0开始的地址处。 移动后的内存布局如下 setup中存在gdt和idt的相关数据。此时需要用gdtr和idtr寄存器指向对应的数据。 实模式下&#xff0c;访问内存方式。最多访问1M内存。

有哪些适合程序员的副业?

如果你经常玩知乎、看公众号&#xff08;软件、工具、互联网这几类的&#xff09;你就会发现&#xff0c;好多资源连接都变成了夸克网盘、迅雷网盘的资源链接。 例如&#xff1a;天涯神贴&#xff0c;基本上全是夸克、UC、迅雷网盘的资源链接。 有资源的前提下&#xff0c;迅雷…

人工智能 — 图像滤波器

目录 一、图像噪声1、高斯噪声2、椒盐噪声3、泊松噪声4、乘性噪声5、瑞利噪声6、伽马噪声 二、图像滤波三、各种滤波器1、均值滤波2、中值滤波3、最大最小值滤波4、引导滤波 四、图像增强1、点处理1、线性变换2、分段线性变换3、对数变换4、幂律变换/伽马变换 2、领域处理3、图…

IP设置教程

Win 7 固定Ip设定 https://jingyan.baidu.com/article/4b07be3cbc8e7348b380f31d.html Win 10 固定Ip设定 Win10 固定IP地址方法_win10设置固定ip地址怎么设置-CSDN博客 Win 11 固定Ip设定 https://jingyan.baidu.com/article/cb5d6105be5354415c2fe0d3.html TP-LINK…

LightGBM中的特征选择与重要性评估

导言 在机器学习任务中&#xff0c;特征选择是提高模型性能和减少过拟合的重要步骤之一。LightGBM作为一种高效的梯度提升决策树算法&#xff0c;提供了内置的特征重要性评估功能&#xff0c;帮助用户选择最重要的特征进行模型训练。本教程将详细介绍如何在Python中使用LightG…

图表征学习——Graph Embedding

图表征学习的目的是将图中的节点嵌入低维的表征&#xff0c;并有效地保留图的结构信息。 Graph Embedding是实现图表征学习的方法&#xff0c;即Graph Embedding的目的也是将图结构转换为节点的低维嵌入表示&#xff0c;在这个过程中&#xff0c;保留图的拓扑结构信息尤为重要。…

2006-2021年地级市资本存量数据(含原始数据+计算过程+计算结果)(以2006年为基期)

2006-2021年地级市资本存量数据&#xff08;含原始数据计算过程计算结果&#xff09;&#xff08;以2006年为基期&#xff09; 1、时间&#xff1a;2006-2021年 2、来源&#xff1a;城市年鉴、统计年鉴、各省年鉴、各市年鉴和公报、2017-2021年利用固定资产投资增速计算获取 …

【C语言】内存操作,内存函数篇---memcpy,memmove,memset和memcmp内存函数的使用和模拟实现【图文详解】

欢迎来CILMY23的博客喔&#xff0c;本篇为​【C语言】内存操作&#xff0c;内存函数篇---memcpy&#xff0c;memmove&#xff0c;memset和memcmp内存函数的使用和模拟实现【图文详解】&#xff0c;图文讲解四种内存函数&#xff0c;带大家更深刻理解C语言中内存函数的操作&…