Android14 - 绘制系统 - 概览

从Android 12开始,Android绘制系统有结构性变化, 在绘制的生产消费者模式中,新增BLASTBufferQueue,客户端进程自行进行queue的生产和消费,随后通过Transation提交到SurfaceFlinger,如此可以使得各进程将缓存提交到SufrfaceFlinger后合并到同一事务后同步提交,在同一帧生效。实际上,从Android12到Android14整个绘制系统各个环节也都或大或小调整,比如Android13发布了1.3版本的Vulkan, Android14新增了TextureView,等等。本文基于Android14

Android 绘制系统整体架构:

从上到下可以理解为“生产者(Producer)”到“消费者(Consumer)”处理过程

首先WindowManagerService角度每个窗口称为Window一个Window一般一个APP页面或者Status Bar或者Navigation Bar或者WallPaper这些一个个Window。WindowManagerService(WMS)作为服务端对所有客户端窗口添加、层级、布局等进行统一管理WMS每个Window对应一个SurfaceSurface可以理解图像数据缓存持有者以及Canvas持有者Canvas画布提供绘制各种图形能力供开发使用。一个客户端窗口在建立之初,会先向WMS去申请一个SurfaceWMS创建Surface之后,通过binder返回客户端客户端Surface后,会去创建一个BLASTBufferQueue管理图像内存申请每次要使用Surface的Canvas进行绘制前,需要BLASTBufferQueue申请一块内存(dequeue),我们这里称为Buffer,然后生成图像数据写入Buffer。这个向BLASTBufferQueue申请Buffer并写入图像数据的过程,可以认为是“生产”阶段。随后,enqueue这个buffer,将其提交SurfaceFlinger去合成。这个阶段可以理解图像Buffer的“消费”阶段

SurfaceFlinger(SF负责Hardware沟通维护着设备挂载、VSync信号收发、Layer合成工作。WMS每个SurfaceSurfaceFlinger对应生成一个Layer对象客户端将某个Surface上的Buffer提交SurfaceFlinger,实际上就是更新对应LayerBuffer数据SurfaceFlinger调用HWComposer将这些Layer进行合成显示屏幕

AndroidHAL提供称为一个Hardware Composer组件用于隔离具体硬件的交互。Hardware Composer简称HWComposerHWC2(之所以2早期已有一个HWC版本支持软件合成)。SurfaceFlingerLayer数据交给HWComposer,各厂商来负责HWComposer合成接口具体实现合成完毕数据提交屏幕设备缓存(一般称为Frame Buffer)屏幕显示画面

上面过程可以拆解为几部分

  1. Surface创建与管理
  2. 客户端(EndPoint)绘制(Draw)渲染(Render)图像
  3. 第三部分是硬件Composition(合成)工作
  4. Vsync:由硬件产生信号用于同步framebuffer生产消费SurfaceFlingerVsync进行使用管理向上分发APPVsync是不断绘制驱动力,也是图像缓存有序投送到屏幕的重要机制。

现在分别讨论下四部分

  1. Surface的创建与管理

Surface创建过程中有几个角色贯穿其中

PhoneWindow一个Activity对应一个PhoneWindow代表一个应用窗口AMS创建Activity之初PhoneWindow服务端对应window对象(ActivityRecord)已经添加WMS

ViewRootImpl:其主要作用服务端通信承接外部触发绘制调用从而从上往下整个View树进行绘制可以把ViewRootImp理解为View的调度者ViewRootImp逻辑上View Hierarchy最顶层并不是一个真正View持有一个View--DecorViewDecorView才是真正ViewView最上层,包含着Activity的画面内容。在Activity的resume阶段,ViewRootImplrelayout方法会将DecorView添加到WMS中,这样Activity的内容就显示了出来。逻辑上,我们可以把DecorView也理解为一个Window。Activity对应一个PhoneWindow,通过ViewRootImplDecorView在WMS端添加PhoneWindowWindow。

WMS的Session客户端一个进程对应WMS一个Session客户端持有Sessionbinder客户端窗口添加事务客户端都是通过这个SessionWMS通信

WindowContainerWMS管理系统整体Window体系包括位置层级关系通过WindowContainer这个表达一个WindowDisplayContent代表一个屏幕级别WindowDisplayArea代表一块屏幕一块区域比如平板等大屏幕设备可能一块屏幕上同时显示多个应用区域此时就用DisplayArea表达WindowToken简单理解为对应一个客户端Window比如一个应用Activity,这里需要注意的是,Activity的WindowToken是作为ActivityRecord存在的,也就是说ActivityRecord是WindowToken的子类。而Activity具体内容承载者,DecorView对应WindowState上面所有DisplayContentDisplayAreaWindowTokenWindowState都是WindowContainer子类,这些Window在WMS内是以window树的形式组织起来的。事实上DisplayContent下面还有一个层级称为Feature具体层级结构Android12 - WMS之WindowContainer树(DisplayArea)_android windowcontainer-CSDN博客客户端通过Session接口调用添加DecorViewWMS生成一个对应WindowState对象将其作为Activity对应ActivityRecord(也就是WindowToken)window

SurfaceControl:在WMS端,每个WindowContainer对应一个SurfaceControlSurfaceControlWMS端管理Surface具体对象,在WMS端,可以理解一个SurfaceControl就代表一个Surface。SurfaceControl在SurfaceFlinge端对应一个Layer,持有一个layer的句柄handle。所有绘制动作最后提交SurfaceFinger作为Layer去合成。SurfaceControl作用或者Surface作用主要客户端窗口SurfaceFlingerLayer关联起来客户端Add一个DecorView WMS对应创建WindowState同时创建一个SurfaceControl、Layer随后SurfaceControl返回客户端客户端拿到SurfaceControl之后转换成Surface后续绘制就在这个Surface进行

SurfaceComposerClient是一个Binder主要作用是SurfaceControl调用SurfaceFlinger过程中,作为一个通道角色由于SurfaceControlWMS客户端持有所以客户端WMS都可以通过这个通道调用SF比如Layer创建、Graphihc Buffer提交

1. 客户端绘制和渲染

客户端通过Surface中提供的Canvas进行绘制Canvas基于Skia的SKCanvas。Skia(https://skia.org/)是Google管理的开源2D(也可以支持3D)图像库,目前AndroidGoogle ChromeChromeOS、Mozilla FireFoxFireFoxOS使用Skia作为绘制引擎。Skia可以集成OPEN GL和Vulkan进行3D绘制Android Q以后Skia作用加强即使硬件加速场景中绘制也会封装成Skia的GrOpList再提交给GPU。Android 14 Skia目录external/skia

渲染的过程是将画好的图像,进行栅格化(Rasterizer),变成一个个像素,这是一个非常耗时的过程。Android 3以前支持软件渲染Software Render过程如下:

APPViewonDraw阶段使用Canvas绘制通过Skia进行软件栅格化,通过CPU计算,将绘制内容转化成一个个像素信息,随后投送给屏幕进行显示。由于软件渲染效率低,当下软件渲染只是作为兼容方案得以保留,默认使用硬件加速。

硬件加速的流程简单表述如下:

Android硬件加速相关能力封装hwui组件hwui地址platform/frameworks/base/libs/hwui

硬件加速模式下APPonDraw通过Canvas绘制内容最终封装DisplayList一个个GrOp绘制命令然后通过OpenGL或者Vulkan交由GPU进行渲染随后结果投送屏幕显示具体使用OpenGL还是Vulkan可选择早期Android使用OpenGL由于Vulkan支持多线程渲染性能方面优势Android逐渐倾向使用Vulkan进行渲染另外,在哪些维度上进行硬件加速也是可选

整体使用硬件加速情况如果某个View绘制暂时不支持硬件加速,或者在某些位移动画上为了减少渲染成本,可以动过设置ViewlayerType = LAYER_TYPE_SOFTWARE单纯某个特定View使用Software Render。

硬件加速除了利用GPU加速渲染效率 本身计算渲染范围相较软件渲染更加高效软件渲染每次更新一个View局部将使得整个View hierarchy重新渲染硬件加速标注变化部分,所谓damage area,将绘制指令保存DisplayList中,如此大大提高渲染速度

OpenGL ES VS Vulkan

以下OpenGL ESVulkanAndroid发布版本历史

Vulkan作为一个面向更低级别规范、跨平台的API,可以提供更细粒度的内存管理和资源管理

以下VulkanOpenGL ES使用率(from GDC 2023 https://www.youtube.com/watch?v=C7OjI7CpjLw&t=1188s):

对于未来计划OpenGL ES将不会再有功能更新功能只会Vulkan支持因此Vulkan未来Android主推渲染引擎

无论OpenGL还是Vulkan都需要GPU的支持例如常见的车载高端芯片高通8155明确标明支持: OpenGL ES 3.xVulkanhttps://www.qualcomm.com/content/dam/qcomm-martech/dm-assets/documents/qul7413_sa8155_productbrief_r4.pdf

2. HW Composer(HWC2)图像合成

前面提到过,每个window对应一个SF中的Layer合成(Composition)工作就是这些Layer进程合并一个完整屏幕内容提交给硬件屏幕显示出来大概过程如下

页面LayerStatusBarNavigationBar中间APP内容页面,其中可能会有重叠的部分,称为Overlay。Composition工作就是将这三个Layer合并一个画面,计算重叠部分的颜色,提交屏幕显示出来

合成工作发生在渲染后的内容提交给SurfaceFlinger之后。大致流程如下

合成硬件合成的部分软件合成部分。硬件合成除了更高效的同时,可以合成工作GPU解放出来,提高GPU效率,节省能耗。嵌入式设备的SOC中,硬件合成一般独立DPU(Display Processing)完成

比如高通SA8155这款SOC布局如下

其中GPU部分负责渲染,“Dispay Processing”的部分用来处理合成工作

由于硬件合成Layer数量是限制例如高通QCS2290支持4个Layer、AMD有的芯片支持7)以及Layer的PixelFormat(比如支持PIXEL_FORMAT_RGBA_8888不支持YUV)是限制的,因此硬件合成之前如果合成Layer过多或者Format不满足需要使用GPU先进行一轮软件合成合并或转换一些Layer格式

软件合成过程。(Google I/O '18)

4. VSYNC

VSync简介:

首先关注两个重要概念

refresh rate - 60Hz 代表每秒钟屏幕可以更新多少次这一值早期是固定依赖于硬件现代旗舰设备屏幕支持多个刷新率60Hz~165Hz不等而且是可以App定制刷新率

frame rate秒钟GPU可以绘制多少越大越好

VSync一个通用概念LinuxPC移动设备实现

想象一下绘制过程是这样的:GPU绘制数据,将绘制结果投掷给屏幕显示出来

问题是refresh rateFrame Rate并不保证一致频率,也就是是说GPU渲染的时间并不能保证就正好是16ms(60Hz)内完成的。如果只有一块内存(Frame Buffer)用来交换数据,假如Refresh Rate大于Frame Rate由于GPU从上到下写这块内存的在当屏幕来取数据的时候,GPU刚刚在旧基础上一半此时就会出现图片撕裂问题

解决方法是双缓存方案

提供Back Buffer和Frame Buffer两个缓存,屏幕始终Frame Buffer数据显示GPU往Back BufferGPU完全数据写好Back Buffer整个拷贝Frame Buffer这样就能保证屏幕每次完整

此时仍有一个问题如果GPU的Frame Rate大于屏幕Refresh Rate那么屏幕下一可能GPU写完好几就会出现丢帧现象此时就需要VSync

屏幕根据自己的刷新频率,去给上层发送一个VSync信号GPU拿到这个VSync信号绘制这样就能同步屏幕上层绘制节奏

如果屏幕Refresh Rate大于GPUFrame Rate怎么

屏幕将会仍然显示旧帧。比如中间方框两次刷新屏幕仍然显示前一次内容

Android的VSYNC

实际上Android的VSync要复杂得多,主要由SurfaceFlinger负责实现。通过之前的介绍我们知道一帧的绘制过程有APP绘制渲染SurfaceFlinger合成Display硬件读取帧缓存显示图片三个阶段,如果每一个阶段都依赖VSync信号来执行,那可能会出现这种情况:

也就是VSync1时候APP正在绘制渲染SF还没有可以合成东西所以什么不做等到VSync2时候Render1工作已经完成可以合成VSync3时候合成做完了才可以显示屏幕上绘制渲染显示经历3VSync面对这种情况AndroidVSync设计如下

有三种信号

HW_VSYNC_[ID]底层硬件按Refresh Rate的频率发出,一般为60Hz、90Hz、120H等等,随后会通过HWC通知SurfaceFlinger

VSYNC-app:SurfaceFlinger通知给上层应用VSYNC用于控制和驱动应用绘制渲染

VSYNC-sf:通知给SurfaceFlinger自身的,用于合成Layer信号

VSYNC-appVSYNC-sfHW_VSYNC_[ID]并不是同步发送而是有一定延迟,称为相位差HW_VSYNC_[ID]VSYNC-app发出时间差称为app phaseHW_VSYNC_[ID]VSYNC-sf发出时间差称为sf phase这种设计好处是如果同一个VSync周期内,经sf phase后在执行合成时恰好前一步Render完成就可一个周期完成不用非得下一个VSync

另外Android并非直接硬件HW_VSYNC_[ID]信号直接分发应用SurfaceFlinger而是通过先收集HW_VSYNC_[ID]样本,再根据屏幕Refresh Rate、预先配置的相位差等信息,经过计算模拟出来VSYNC-appVSYNC-sf

由于只需要一定的硬件VSync样本便可以模拟出预期VSYNC-appVSYNC-sf因此并不需要一直HW_VSYNC_[ID]信号收到足够样本(在Android 14中为6个)就可以关闭硬件VSync每次合成数据提交屏幕返回一个硬件VSync时间戳(PresentFence),此时SF对比当前模拟VSync硬件VSync是否误差过大如果过大重新打开硬件VSync收集样本重新计算另外每次终端应用主动请求VSync判断前后两次模拟VSync时间差是否超过750ms如果重新请求打开硬件VSyncsystrace硬件VSyncTAGHW_VSYNC_ON_[ID]

参考资料https://source.android.com/docs/core/graphics/implement-vsync

可变刷新率

现代旗舰机屏幕刷新率可变,比如Pixel 5:

可以看到屏幕是支持60Hz90Hz两种刷新率

而且应用层可以在应用级别、窗口级别指定具体刷新率经过应用指定最终刷新率并不一定指定而是经过SurfaceFlinger综合计算得出具体见https://developer.android.com/media/optimize/performance/frame-rate

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

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

相关文章

【vue3+elementuiplus】el-select下拉框会自动触发校验规则

场景:编辑弹框省份字段下拉框必填,触发方式change,有值第一次打开不会触发校验提示,关闭弹框再次打开触发必填校验提示,但是该字段有值 问题的原因是:在关闭弹层事件中,我做了resetfileds&…

SpringBoot + MybatisPlus

SpringBoot MybatisPlus 整合记录 1. 硬件软件基本信息2. 相关链接3. 通过idea快速生成一个Springboot项目4. 启动报错问题解决问题一:Springboot启动的时候报错提示 “没有符合条件的Bean关于Mapper类型”问题二:启动的时候提示需要一个Bean&#xff0…

电磁仿真--CST网格介绍

1. 简介 网格会影响仿真的准确性和速度,花时间理解网格化过程是很重要的。 CST 中可用的数值方法包括FIT、TLM、FEM、MoM,使用不同类型的网格: FIT和TLM:六面体 FEM:四面体、平面 MoM:表面 CFD&#…

深入理解与防御跨站脚本攻击(XSS):从搭建实验环境到实战演练的全面教程

跨站脚本攻击(XSS)是一种常见的网络攻击手段,它允许攻击者在受害者的浏览器中执行恶意脚本。以下是一个XSS攻击的实操教程,包括搭建实验环境、编写测试程序代码、挖掘和攻击XSS漏洞的步骤。 搭建实验环境 1. 安装DVWA&#xff…

【408真题】2009-16

“接”是针对题目进行必要的分析,比较简略; “化”是对题目中所涉及到的知识点进行详细解释; “发”是对此题型的解题套路总结,并结合历年真题或者典型例题进行运用。 涉及到的知识全部来源于王道各科教材(2025版&…

推荐一个快速开发接私活神器

文章目录 前言一、项目介绍二、项目地址三、功能介绍四、页面显示登录页面菜单管理图表展示定时任务管理用户管理代码生成 五、视频讲解总结 前言 大家好!我是智航云科技,今天为大家分享一个快速开发接私活神器。 一、项目介绍 人人开源是一个提供多种…

Golang | Leetcode Golang题解之第112题路径总和

题目: 题解: func hasPathSum(root *TreeNode, sum int) bool {if root nil {return false}if root.Left nil && root.Right nil {return sum root.Val}return hasPathSum(root.Left, sum - root.Val) || hasPathSum(root.Right, sum - roo…

C++常见知识点总结

常见字符 * 注释:/* 这是一个注释*/乘法:a * b取值运算符:*指针变量,int a 4,*a ????指针变量:数据类型 *变量名, int *no &bh&#xff0…

SAP揭秘者-怎么执行生产订单ATP检查及其注意点

文章摘要: 上篇文章给大家介绍生产订单ATP检查的相关后台配置,大家可以按照配置步骤去进行配置,配置完之后,我们接下来就是要执行ATP检查。本篇文章具体给大家介绍怎么来执行生产 订单ATP检查及其注意点。 执行生产订单ATP检查的…

Qt for android 获取USB设备列表(二)JNI方式 获取

简介 基于上篇 [Qt for android 获取USB设备列表(一)Java方式 获取], 这篇就纯粹多了, 直接将上篇代码转换成JNI方式即可。即所有的设备连接与上篇一致。 (https://listentome.blog.csdn.net/article/details/139205850) 关键代码…

Android卡顿丢帧低内存与adb shell内存状态

Android卡顿丢帧低内存与adb shell内存状态 卡顿丢帧除了CPU/GPU层面,另外,也需要特别注意整机低内存情况。kswapd0 是一个内核工作线程,内存不足时会被唤醒,做内存回收工作。 当内存频繁在低水位的时候,kswapd0 会被频…

Linux基础(六):Linux 系统上 C 程序的编译与调试

本篇博客详细分析,Linux平台上C程序的编译过程与调试方法,这也是我们后续程序开发的基础。 目录 一、第一个hello world程序 1.1 创建.c文件 1.2 编译链接 运行可执行程序 二、编译链接过程 2.1 预编译阶段 2.2 编译阶段 2.3 汇编阶段 2.4 链…

qemu+gdb调试linux内核

打开CONFIG_DEBUG_INFO,编译内核 通过图形菜单配置该宏,执行make menuconfig。 kernel hacking —> compile-time checks and compiler options —> compile the kernel with debug info 验证是否打开成功,grep -nr “CONFIG_DEBUG_INFO” .config。 打开成功,然后…

plsql 学习

过程化编程语言 赋值:: ||:连接符号 dbms_output.put_line() :输出的语句 var_name ACCOUNTLIBRARY.USERNAME%type; 变量名;某个表的数据类型;赋值给变量名 用下面的方法更好用 异常exception 循…

力扣HOT100 - 75. 颜色分类

解题思路&#xff1a; 单指针&#xff0c;对数组进行两次遍历。 class Solution {public void sortColors(int[] nums) {int p 0;int n nums.length;for (int i 0; i < n; i) {if (nums[i] 0) {int tmp nums[i];nums[i] nums[p];nums[p] tmp;p;}}for (int i p; i …

存储+调优:存储-IP-SAN-EXTENSION

存储调优&#xff1a;存储-IP-SAN-EXTENSION 文件系统的锁标记 GFS&#xff08;锁表空间&#xff09; ----------- ------------ ------------- 节点 | ndoe1 | | node2 | | node3 | ---------- ------…

GNSS中的多路径效应原理及计算方法

1 多路径效应原理 图1 多路径效应原理图 2 计算方法 如需原文&#xff0c;可加多源融合定位与智能控制讨论群获取,QQ群号&#xff1a;51885949

【OceanBase诊断调优】—— 直连普通租户时遇到报错:Tenant not in this server

本文介绍了直连 OceanBase 数据库中的普通租户时&#xff0c;出现报错&#xff1a;ERROR 5150 (HY000) : Tenant not in this server 的处理方法。 问题描述 在 n-n 或者 n-n-n (n>1) 的部署架构中&#xff0c;使用 2881 端口 直连 OceanBase 集群的普通租户&#xff0c;可…

SpringBoot运维篇(打包,多环境,日志)

文章目录 一、SpringBoot程序的打包与运行二、配置高级三、多环境开发四、日志 一、SpringBoot程序的打包与运行 刚开始做开发学习的小伙伴可能在有一个知识上面有错误的认知&#xff0c;我们天天写程序是在Idea下写的&#xff0c;运行也是在Idea下运行的。 ​但是实际开发完成…

CDH6.3.2集成Flink1.17

直接运行脚本即可&#xff0c;一键输出相关依赖包 运行步骤已给到文档 下载地址