Android Display Graphics #1 整体框架介绍一

  1. 软件基础

Android的framework层提供了一系列的图像渲染API,可绘制2D和3D。简单理解就是上层开发APP的小伙伴提供了接口,开发者可以直接显示对应的自己内容。但如果掌握了Display底层逻辑再写上层app,会有掌控力,出问题可以根据log定位。

比如app用调用了

public class MainActivity extends AppCompatActivity {

或者

public class MainActivity extends Activity {

创建一个APP类型窗口,或者通过主动调用WindowManager.addView的方式来创建非App类型的窗口,都会调到ViewRootImpl。ViewRootImpl中有两个关键对象 mSurface和mSurfaceControl。

frameworks/base/core/java/android/view/ViewRootImpl.java

652      public final Surface mSurface = new Surface();

653      private final SurfaceControl mSurfaceControl = new SurfaceControl();

通过jni调到native层SurfaceControl。创建一个SurfaceFlinger的client端,new SurfaceComposerClient。通过这个client创建buffer和surface。这个surface就给到了app端。用户可以在上面绘制要显示的内容。绘制完成后将Surface(即buffer)传递到SurfaceFlinger合成,再给到HardwareComposer送显,最终显示到显示屏上。

几个基础概念:

Canvas

翻译过来就是 “画布”,也可以理解为绘制规则。是Java层的一个类,提供了一系列API,使用这些API可以实现各种的产品经理要求的效果。Canvas包含bitmap、线条、圆形、矩形、文字等绘制方法,是绘制 2D 对象的最简便方法,并且和bitmap、Surface相关联。

可通过如下几种方法创建:

1、通过构造方法。

kotlin

val canvas = Canvas()

canvas.setBitmap(bitmap)

绘制完的内容保存到Canvas的同期bitmap中。

kotlin

val canvas = Canvas(bitmap)  效果同上

2、重写View.onDraw

override fun onDraw(canvas: Canvas?) {

super.onDraw(canvas)

}

这也是我们使用最多的方式,自定义View时通过重写onDraw方法来实现。Canvas绘制的内容是显示在自定义View中。

3、通过SurfaceView

kotlin

val surfaceView = SurfaceView(context)   //从surfaceView的surfaceHolder里锁定获取Canvas

val canvas = surfaceView.holder.lockCanvas()  //进行canvas操作

......

Canvas surfaceView.holder.unlockCanvasAndPost(canvas)   //Canvas操作结束后解锁并执行

Kotlin Example:

canvas?.drawColor(context.getColor(R.color.blue))

OpenGL ES

目前电脑和手机都离不开GPU,但每个大厂都有自己的GPU,高通的adreno,ARM的mali,英伟达,Apple的等,如果各厂商的驱动接口各不相同,上层用户开发显示就太难了。如果有一套接口将复杂的、 各种各样的 GPU 硬件包装起来,各个电脑公司编写自家的驱动,然后提供出来 一套统一的接口,供上层软件工程师调用。这样,世界就和平了。

没错,OpenGL就是这套标准接口。OpenGL ES 为嵌入式设备 GPU 驱动的标准接口,比如手机, OpenGL ES 全称:OpenGL for Embedded Systems。

那谁这么牛逼定义了这套标准呢?Khronos 组织,这个众多大公司联合组建而来,比如 Apple、 Intel、AMD、Google、ARM、Qualcomm、Nvidia 等。针对 GPU 又提出了另外一套更底层的接口 Vulkan,这是一套比 OpenGL 更底层的接口。手机屏上显示的内容,2D或3D ,如果是 3D 的,不管是 App 还是游戏,或者简单的图片界面,底层都是通过 GPU、 通过 OpenGL(ES)绘制出来的。

对于 2D 渲染,Canvas是更简单的选择。Android NDK 中提供 OpenGL ES。android.opengl 和 javax.microedition.khronos.opengles 软件包提供 OpenGL ES 功能。

跟OpenGL ES协作的还有EGL(Embedded Graphic Interface)和GLSL(OpenGL Shading Language)两个好伙伴,后面有需要再分析。

Surface

这里其实理解为”画布”更合适,但会和Canvas混淆,所以不翻译。Surface表示一个用于合成的内存块。在电脑上编辑一张图片,首先要加载到内存,操作内存数据,再保存到本地。这个Surface就相当于你向系统申请的那个内存(Android 中是SurfaceFlinger),手机分辨率是1920*1080,那你申请的Surface大小和范围只要在[0,0]-[1920,1080]之间就可以。

把Surface理解为一个buffer也可以,Native层的Surface定义在/frameworks/native/libs/gui/include/gui/Surface.h,可以看到很多操作buffer的接口。

显示的不同阶段对这块内存的叫法不同,在上层可以是Window、ANnativeWindow、Surface、SurfaceView、layer、buffer,这些本质上没啥区别,可以广泛意义上理解为用于显示内容的那块内存,只不过各个阶段的操作接口不同。

Vulkan

Vulkan 是一个适用于高性能 3D 图形的低开销、跨平台 API。前面不是已经有OpenGL ES了,怎么又出来一个Vulkan呢?因为OpenGL虽然迭代了25年以上,并不断满足行业需求,但是现在已经逐渐满足不了行业的需要。且Khronos Group的各大佬成员都同意需要新一代的跨平台GPU API,由此Vulkan诞生了。

Vulkan诞生之初就决定了它一定要有跨平台属性,目的就是成为行业内的统一标准。

GPU的可编程性越来越强,越来越多的平台开始支持加速图形,计算,视觉和深度学习。灵活性和可移植性变得很重要。

性能上,OpenGL也不能充分发挥现代CPU多核多线程的性能优势。

Vulkan-新一代GPU API的特性

·Explicit(明确、透明)

GPU driver做更少的事情,把更多的控制权交给开发者

·Streamlined(精简)

更快的性能,更低的开销,更少的延迟

·Portable(可移植)

Cloud, desktop, console, mobile and embedded

·Extensible (可扩展)

支持新功能的扩展,推动行业技术进步

Vulkan有哪些优势呢?

·显式的GPU控制

OpenGL驱动中,驱动会帮你做API验证,内存管理,线程管理等大部分工作。OpenGL驱动大包大揽什么事情都管,即使应用使用API出错,也会帮忙解决处理,保证应用正常运行。开发者使用起来非常简单。Vulkan则不然。Vulkan把API验证、内存管理、多线程管理等工作交由开发者负责。一旦API使用出错,应用就会出现crash。没人帮应用兜底,所有事情都交由应用打理。这种方式无疑增加了API使用的复杂度和困难度,但换来的是性能上巨大的提升。

·CPU多线程的效率提升

在OpenGL中,所有的渲染操作都放在一个线程,其他线程就算处于空闲状态,也只能围观。

Vulkan中引入了 Command Buffer 的概念,每个线程都可以往Command Buffer 提交渲染命令,给开发者提供了充分发挥CPU多核多线程的优势。在复杂场景下,性能的提升非常客观!

· 可移植性

Vulkan可以支持各种平台。

Vulkan和OpenGL性能对比

Khronos给出了一张各个图形API理论性能对比:

PowerVR做了一个视频做了Vulkan和OpenGL的性能对比:

为防止视频丢失,可以关注微信公众号AndroidAVWorld 回复:OpenglVSVulkan 获取。

如何决定是使用OpenGL还是Vulkan呢?如果你定位到确实是OpenGL导致的性能问题,而不是GPU瓶颈,可以考虑使用Vulkan。但你要准备好适应更大难度的代码复杂度,更多开发量。

Android 给的画图方式有3种,即Canvas、OpenGL ES、Vulkan。原文如下:

App developers draw images to the screen in three ways: with Canvas, OpenGL ES, or Vulkan.

  1. 硬件与显示流程 

      两大组成部分:  渲染(Render) + 合成 (SurfaceFlinger + HWC)

渲染(Render)也可以叫做绘制(drawFrame),这样理解起来更具体些。

渲染(Render):将Surface中的空GraphicBuffer交给CPU或GPU将数据绘制到Surface上。把绘图指令转化为二维像素数组的过程,就叫做“渲染”。CPU通过Skia库,GPU通过Canvas。

合成  (Compose): SurfaceFlinger将几块有数据的Buffer 叠合成一张Buffer,交个HWC送到panel显示。

图像渲染大致步骤:

渲染是通过GPU和CPU共同完成的,CPU主要工作:计算显示内容,比如视图的创建、布局计算、图片解码、文本绘制等。GPU主要工作:Rasterization (栅格化),渲染到帧缓冲区rasterization光栅化,将图转化为一个个栅格组成的图象,每一个元素也就是意味着帧缓冲区中的一个像素。如下图所示

光栅化是一种非常消耗资源的活动,GPU就是为了解决这个问题而在上个世纪90年代引入的单独的图像处理单元。CPU 负责把 UI 组件计算成 Polygons,Texture 纹理,然后交给 GPU 进行栅格化渲染。

硬件加速的本质就是使用GPU代替CPU完成Graphic Buffer绘制工作,以实现更好的性能。

GPU

GPU全称是GraphicProcessing Unit,中文是图形处理器,其最大的作用就是进行各种绘制计算机图形所需的运算,包括顶点设置、光影、像素操作等。也就是用来执行我们开发者调用的一个个的“绘图指令”。所以,GPU实际上是多组图形函数API的集合,这些函数由硬件驱动实现。

合成就是把每个图层的内容,计算重叠区域的像素后,最终按照Z轴的上下顺序把多个图层合并为一个图层。

举个栗子:

launcher桌面,它由“壁纸”、“顶部的状态栏”、“桌面的应用列表”以及“底部导航栏”4个图层组成。

Layer1:壁纸

Layer2:顶部状态栏,显示时间等

Layer3:桌面应用列表

Layer4:底部导航栏

上面的Layer1-Layer4图层,都由GPU渲染出来。但图层与图层之间发生重叠(比如状态栏、应用列表和导航栏这3个图层都是叠加在壁纸图层的上面),重叠部分的像素颜色就需要重新计算。

DPU

合成工作的本质是执行计算“脏区域”、格式转换、处理缩放等操作,这些任务可以调用GPU的接口来完成。“图层渲染”那一步GPU已经完成了3D处理,所以图层合成的过程中不需要3D操作。

那只需要为合成流程单独配置一块2D渲染引擎就可以了。目前Android设备中,基本都是DPU芯片负责这个任务。

DPU作为图形硬件的一部分,通常被封装在GPU模块当中,最主要的功能是将GPU渲染完成的图层输出到屏幕。兼任了合成功能以后,对于图层重叠的部分,DPU会自动计算出“脏区域”并更新像素颜色变化。

不过,DPU虽然可以执行合成工作,但它有合成数量的限制。

例如Arm Mali-DP550最多支持7层合成任务。

一幅图像的显示必须要经过渲染、合成、送显这三个阶段,才能展现给用户。GPU和DPU就是用于提供给应用程序渲染和合成能力的硬件设备,其中:

·GPU芯片负责执行图形显示流程中的“渲染”工作

·DPU芯片负责执行图形显示流程中的“合成”工作

HWC

上面说到GPU和DPU都可以做合成任务,厂商可以选择将合成工作放在DPU中,也可以选择在板子上加一块2D渲染芯片,将合成工作放在这块芯片中,或者直接使用GPU合成。这就需要一个规范把合成工作抽象成一个接口,由厂商自由选择合成方案。

Hardware Composer就是专门用来定义合成工作的抽象接口,它是Android Hardware Abstraction Layer(HAL)硬件抽象层的成员之一。在HWC中,厂商使用的是DPU还是其他的2D渲染芯片不重要,只需要实现HWC的接口即可。

那使用GPU和DPU合成有什么差异?合成策略怎么选择呢?

1、DPU合成比GPU高效,有由显示硬件原始设备制造商OEM完成;

2、hwc中的硬件供应商定制决策代码,大致流程为:

sf进程:有8个渲染好的图层过来了,改怎么合成这些图层呢?都给你行不?

hwc:老弟,我最多只能处理5个图层的合并工作,其余3个图层我标记为GPU合成了,你处理吧。

sf进程:调用GPU完成另外3个图层的合成工作,并将合并后的图层交给hwc。

3、当有一个layer是透明的,并且叠加到一起显示时,HWC的合成会低于OpenGL的合成效率。

4、超出HWC能力的图层会调用GPU合成,如果应用的图层太多会对性能产生影响。

具体layer的合成方式,可以通过dumpsys  SurfaceFlinger来查看。

HWC模块还负责送显以及发送VSync信号等。

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

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

相关文章

iOS自动连接已知Wi-Fi功能的实现

首先需要在配置文件申请的时候将hotspot勾选上&#xff0c;之后还要在x-code里添加对应的配置&#xff0c;由于我们并没有用到获取设备周边Wi-Fi的功能&#xff0c;所以就没申请相关权限 相关连接Wi-Fi代码如下&#xff1a; #import <NetworkExtension/NetworkExtension.h&…

探索k8s集群的存储卷 emptyDir hostPath nfs

目录 一 含义 查看支持的存储卷类型 emptyDir存储卷 1.1 特点 1.2 用途 1.3部署 二、hostPath存储卷 一 含义 容器磁盘上的文件的生命周期是短暂的&#xff0c;这就使得在容器中运行重要应用时会出现一些问题。首先&#xff0c;当容器崩溃时&#xff0c;kubelet 会重…

FreeRTOS【12】队列集使用

1.开发背景 基于以上的章节&#xff0c;了解了 FreeRTOS 多线程间的信号量、队列的使用&#xff0c;已经满足了日常使用场景。这个篇章要介绍的是队列集&#xff0c;实际上队列的升级版&#xff0c;存储信号量和队列等的触发事件。 队列集在实际的开发项目中应用相对比较少&…

chap4 simple neural network

全连接神经网络 问题描述 利用numpy和pytorch搭建全连接神经网络。使用numpy实现此练习需要自己手动求导&#xff0c;而pytorch具有自动求导机制。 我们首先先手动算一下反向传播的过程&#xff0c;使用的模型和初始化权重、偏差和训练用的输入和输出值如下&#xff1a; 我…

达梦数据库写文件的方式探索

0x01 前沿 这篇文章整体算是《达梦数据库手工注入笔记》的续集&#xff0c;达梦作为国内优秀的信创数据库&#xff0c;在关基单位中拥有越来越大的用户使用量。 通过SQL注入来写文件一直以来都是SQL注入漏洞深入利用的一种方式&#xff0c;对于不同的数据库通常写文件的方式也是…

刷代码随想录有感(86):贪心算法——跳跃游戏II(最小跳跃次数)

题干&#xff1a; 代码&#xff1a; class Solution { public:int jump(vector<int>& nums) {if(nums.size() 1)return 0;int curcover 0;int nextcover 0;int res 0;for(int i 0; i < curcover; i){nextcover max(i nums[i], nextcover);if(i curcover …

二叉树的链式存储

目录 1.二叉树的概念和性质2.二叉树的链式存储2.1二叉树的遍历2.1.1前中后遍历2.1.2层次遍历 2.2求节点的个数2.3求叶子节点的个数2.4求第k层节点个数2.5二叉树的销毁2.6怎样通过前序遍历构建二叉树2.7判断是否是满二叉树 1.二叉树的概念和性质 一&#xff0c;概念 1.五种形态…

掌握 JavaScript 基本输出方法

掌握 JavaScript 基本输出方法 前言 这是我在这个网站整理的笔记,有错误的地方请指出&#xff0c;关注我&#xff0c;接下来还会持续更新。 作者&#xff1a;神的孩子都在歌唱 JavaScript 是一种强大且灵活的编程语言&#xff0c;广泛用于 Web 开发。通过 JavaScript&#xff…

YOLOV8训练自己的数据集图文实战,包含voc数据集处理代码

yolov8官方链接: link 本文章是以labelimg标注好的voc数据集为基础,通过转换格式训练模型, 一,安装 pip install ultralyticsor pip install githttps://github.com/ultralytics/ultralytics.gitmainlink 二,数据集准备 数据集格式如下 ├── ultralytics └── datase…

RocketMq broker源码解析

broker 集群工作流程 NameSrv启动成功后&#xff0c;等待broker、Consumer和producer启动后也与NameSrv保持长连接, NameSrv相当于是路由控制中心。启动broker, broker与所有的NameSrv建立长连接, broker&#xff0c;通过定时线程定时向NameSrv发送心跳&#xff0c;broker信息…

LightDB pro*c迁移指南(游标模块)

文章目录 一、不使用SQLDA描述符范围的游标操作1.1 oracle 案例1.1.1 使用游标获取数据1.1.2 对于fetch结果集怎么去利用 1.2 LightDB 案例1.2.1 使用游标获取数据1.2.2 对于fetch结果集怎么去利用 3 总结&#xff1a;不同项 二、使用SQLDA描述符范围的游标操作2.1 Oracle样例2…

KMPlayer v2024.4.25.13 官方版 (万能播放器)

前言 KMPlaye通过各种插件扩展KMP可以支持层出不穷的新格式。KMPlaye强大的插件功能&#xff0c;直接从Winamp继承的插件功能&#xff0c;能够直接使用Winamp的音频&#xff0c;输入&#xff0c;视觉效果插件&#xff0c;而通过独有的扩展能力&#xff0c;只要你喜欢&#xff…

webman-admin多图上传预览和删除

前言 在webmen文档和论坛中都没找到多图上传的示例&#xff0c;自己找了一个&#xff0c;整合了一下凑合用 insert页面 引入css <link rel"stylesheet" href"/app/admin/admin/css/muti-upload.css" />muti-upload.css内容如下 .uploader-list .ha…

微信小程序学习

04.认识小程序项目的基本组成结构 把allow改成disallow,表示所有的页面不被微信进行索引。 比如修改首页的上面栏颜色

自制数据#国家2000投影带划分范围shp(高斯克吕格 3°/6°分带)

国家2000投影分带范围&#xff08;3&#xff09; https://www.123pan.com/s/lqEljv-xvCHA.html 国家2000投影分带范围&#xff08;6&#xff09; https://www.123pan.com/s/lqEljv-xvCHA.html 声明&#xff1a;转载此文不为商业用途。文字和图片版权归原作者所有&#xff0c;…

网络安全基础技术扫盲篇名词解释之“证书“

用通俗易懂的话说&#xff1a; 证书就好比是一张身份证&#xff08;类似&#xff0c;但不完全相同&#xff09;&#xff0c;用来证明一个网站的身份是否可信。就像你要确认一个陌生人的身份需要看他的身份证一样&#xff0c;电脑在连接一个网站时&#xff0c;也会查看网站的证…

延时性(过期/超时)和周期性的定时任务的实现方式

延时性&#xff08;过期/超时&#xff09;和周期性的定时任务的实现方式 一、延时性的定时任务&#xff08;例如订单超时30分钟后自动取消该订单&#xff09;1.使用DelayQueue实现任务即将到期提醒功能&#xff08;非分布式&#xff09;2.使用Redis实现任务即将到期提醒功能&am…

探索Web3工具:正确使用区块链平台工具的秘诀

在当今日新月异的数字时代&#xff0c;区块链技术正以惊人的速度改变着我们的生活和工作方式。尤其对于那些想要踏入区块链世界的人来说&#xff0c;正确使用区块链平台工具至关重要。本文将向您介绍一些关键的Web3工具&#xff0c;并以TestnetX.com为例&#xff0c;展示如何利…

数字化转型推动生物技术企业增长—纷享销客与集萃药康共探新动力

上周&#xff0c;在南京锦创书城&#xff0c;一场主题为“生物技术企业增长新动力&#xff1a;以客户为中心的数字化转型与创新”的研讨会圆满落幕。此次活动由纷享销客江苏分公司联合江苏集萃药康生物科技股份有限公司共同举办&#xff0c;吸引了众多生物技术领域企业的负责人…

斑消宝六周年大动作,斑小将将再迎高光时刻

如今&#xff0c;周年庆典已经成为众多品牌展示自身实力与影响力的重要舞台。这不仅仅是一个简单的庆祝活动&#xff0c;更是一次向外界展示品牌发展历程、未来规划以及团结合作伙伴的绝佳机会。在这样的背景下&#xff0c;广州斑消宝化妆品有限公司将打造别具一格的盛典&#…