当前位置: 首页 > news >正文

Android SurfaceFlinger做Layer合成时,如何与HAL层进行交互

目录

  • 零、本文讨论问题的范围
  • 一、问题:SurfaceFlinger图层合成选择实现方式的两难
    • 1.1 从OpenGL ES、HWC本身来讲
    • 1.2 以HWC为主导的判断逻辑
  • 二、SurfaceFlinger与HAL层进行交互的具体实现框架
    • 2.1 SurfaceFlinger 调用 OpenGL ES 流程
    • 2.2 FrameBuffer
    • 2.3 SurfaceFlinger 调用 HWC 的流程
  • 三、通过HAL定义,理解上述概念
    • 3.1 hwcomposer.h
    • 3.2 fb.h
    • 3.3 gralloc.h

零、本文讨论问题的范围

Surface 如何被渲染图
上图表示了,Surface 如何被渲染。
借用官网的一段话:“无论开发者使用什么渲染 API,一切内容都会渲染到 Surface 上。Surface 表示缓冲区队列中的生产方,而缓冲区队列通常会被 SurfaceFlinger 消耗。在 Android 平台上创建的每个窗口都由 Surface 提供支持。所有被渲染的可见 Surface 都被 SurfaceFlinger 合成到屏幕。”

本文关注的范围是上图红框中部分,SurfaceFlinger做Layer合成时,如何与HAL层的交互?

一、问题:SurfaceFlinger图层合成选择实现方式的两难

在处理Layer合成时,SurfaceFlinger会与OpenGL ES、HWC进行交互。对于HWC,由于各OEM的实现不一样,所支持的能力也不一样,很难直接用API表示硬件设备支持合成的Layer数量,Layer是否可以进行旋转和混合模式操作,以及对图层定位和硬件合成的限制等。这就导致了SurfaceFlinger作为调用端,很难做决策:根据当前的要合成的Layer数量,数据量大小,我该选择使用OpenGL ES?还是HWC?

1.1 从OpenGL ES、HWC本身来讲

在Android系统中,相对于HWC,OpenGL ES并不很擅长Layer的合成。
因为OpenGL ES主要是为游戏开发和图形渲染而设计的,它并不针对图层合成进行优化。如果使用OpenGL ES进行图层合成,会占用大量的GPU资源,导致应用程序无法使用GPU进行自己的渲染。因此,在需要进行高效图层合成的场景下,HWC更适合使用。

1.2 以HWC为主导的判断逻辑

既然各OEM提供的HWC的能力是不同的,有强、有弱,但Layer合成任务又想优先使用HWC,那么SurfaceFlinger的选择策略就交给HWC判断好了:
SurfaceFlinger根据HWC硬件能力决定使用OpenGL ES还是使用HWC来处理Layer合成任务。

具体逻辑如下:
SurfaceFlinger向HWC提供所有Layer的完整列表,让HWC根据其硬件能力决定如何处理这些Layer。
HWC会为每个Layer标注合成方式,是通过GPU还是通过HWC合成。
SurfaceFlinger负责先把所有注明GPU合成的Layer合成(使用OpenGL ES)到一个输出Buffer,然后把这个输出Buffer和其他注明HWC合成的Layer一起交给HWC,让HWC完成剩余Layer的合成和显示。

二、SurfaceFlinger与HAL层进行交互的具体实现框架

2.1 SurfaceFlinger 调用 OpenGL ES 流程

SurfaceFlinger 与 OpenGL ES
Gralloc 即 Graphics Alloc 图形分配 。 Android 系统在HAL层中提供了一个 Gralloc 模块,封装了对 Framebuffer 的所有访问操作。
Gralloc 模块包含 fb 和 gralloc 两个设备:fb 负责打开内核中的 Framebuffer 、初始化配置,以及提供 post, setSwapInterval 等操作,它是底层显卡的HAL层抽象;gralloc 则管理帧缓冲区的分配和释放,上层应用只能通过 Gralloc 访问 fb。

Android12 的源码中,HAL层头文件定义的路径在:源码根目录/hardware/libhardware/include/hardware/*.h

2.2 FrameBuffer

FrameBuffer 机制模仿显卡的功能,是显卡硬件的抽象,可以将 FrameBuffer 看成是显存的一个映像,将其映射到进程地址空间之后,就可以直接进行读写操作,简单理解就是一段内存。用户应用不停的向这段内存中写入数据,显示控制器不停地从中读取数据并显示出来。

2.3 SurfaceFlinger 调用 HWC 的流程

SurfaceFlinger 与 HWC

三、通过HAL定义,理解上述概念

3.1 hwcomposer.h

上面我们提到HWC会为每个Layer标注合成方式

typedef struct hwc_layer_1 {int32_t compositionType;
}

结构体 hwc_layer_1 中,定义了一个 compositionType 变量,表示当前Layer 的合成类型。可选的常量有 HWC_BACKGROUND、HWC_FRAMEBUFFER_TARGET、HWC_FRAMEBUFFER、HWC_OVERLAY、HWC_SIDEBAND、HWC_CURSOR_OVERLAY。
其中:HWC_FRAMEBUFFER,是这么注释的(节选):

     *   Set by the HWC implementation during (*prepare)(), this indicates*   that the layer will be drawn into the framebuffer using OpenGL ES.

HWC在准备阶段,可能会将Layer的类型设置成 HWC_FRAMEBUFFER,表示这个Layer会使用OpenGL ES画到 framebuffer 中。
再看 HWC_BACKGROUND

     *   Always set by the caller before calling (*prepare)(), this value*   indicates this is a special "background" layer. The only valid field*   is backgroundColor.*   The HWC can toggle this value to HWC_FRAMEBUFFER to indicate it CANNOT*   handle the background color.

总是由调用者在调用(*prepare)()之前设置这个值,表示这是一个特殊的“背景”图层。唯一有效的字段 backgroundColor。HWC可以将此值切换为HWC_FRAMEBUFFER来表示它不能处理背景颜色(HWC处理不了,就只能是用OpenGl ES来处理)。这里的调用者应该就是SurfaceFlinger了。

Hardware Composer 必须支持事件,其中之一是 VSYNC(另一个是支持即插即用 HDMI 的热插拔)

3.2 fb.h

我们抽出几个属性,有个大体印象

typedef struct framebuffer_device_t {/* dimensions of the framebuffer in pixels */const uint32_t  width;const uint32_t  height;/* framebuffer pixel format */const int       format;/* resolution of the framebuffer's display panel in pixel per inch*/const float     xdpi;const float     ydpi;/* framebuffer's display panel refresh rate in frames per second */const float     fps;......
}

FrameBuffer设备的属性有:宽高、像素格式、以dip为单位的分辨率、fps。
再看一个方法:

    /** Post <buffer> to the display (display it on the screen)* The buffer must have been allocated with the*   GRALLOC_USAGE_HW_FB usage flag.* buffer must be the same width and height as the display and must NOT* be locked.** The buffer is shown during the next VSYNC.** Returns 0 on success or -errno on error.*/int (*post)(struct framebuffer_device_t* dev, buffer_handle_t buffer);

post函数会将 buffer post到display上,显示在屏幕上。
buffer的宽度要与fb设备宽高一致,不能被锁。
在下一个VSYNC信号到来时,buffer会显示出来。

3.3 gralloc.h

/*** Name of the graphics device to open*/
#define GRALLOC_HARDWARE_GPU0 "gpu0"

看出 gralloc操作的设备就是GPU

enum {/* buffer will be used as an OpenGL ES texture */GRALLOC_USAGE_HW_TEXTURE            = 0x00000100U,/* buffer will be used as an OpenGL ES render target */GRALLOC_USAGE_HW_RENDER             = 0x00000200U,/* buffer will be used by the 2D hardware blitter */GRALLOC_USAGE_HW_2D                 = 0x00000400U,/* buffer will be used by the HWComposer HAL module */GRALLOC_USAGE_HW_COMPOSER           = 0x00000800U,/* buffer will be used with the framebuffer device */GRALLOC_USAGE_HW_FB                 = 0x00001000U,/* buffer may be used as a cursor */GRALLOC_USAGE_CURSOR                = 0x00008000U,/* buffer will be used with the HW video encoder */GRALLOC_USAGE_HW_VIDEO_ENCODER      = 0x00010000U,/* buffer will be written by the HW camera pipeline */GRALLOC_USAGE_HW_CAMERA_WRITE       = 0x00020000U,/* buffer will be read by the HW camera pipeline */GRALLOC_USAGE_HW_CAMERA_READ        = 0x00040000U,......
}

gralloc graphics allocation字面意思图形分配,其实就是开辟一段内存,操作的是GPU那就是显存。
这里定义了一个枚举,为分配出的 buffer 标记了分类,按使用用途分出的类别。
这里有做为 OpenGL ES texture 使用的、被 HWComposer HAL module 使用的、被硬件视频编码器使用的、被摄像头管道写入/读出的。还发现一个有意思的是,GRALLOC_USAGE_CURSOR 光标竟然是单独一种类型的 buffer。

核心方法:

typedef struct alloc_device_t {struct hw_device_t common;/* * (*alloc)() Allocates a buffer in graphic memory with the requested* parameters and returns a buffer_handle_t and the stride in pixels to* allow the implementation to satisfy hardware constraints on the width* of a pixmap (eg: it may have to be multiple of 8 pixels). * The CALLER TAKES OWNERSHIP of the buffer_handle_t.** If format is HAL_PIXEL_FORMAT_YCbCr_420_888, the returned stride must be* 0, since the actual strides are available from the android_ycbcr* structure.* * Returns 0 on success or -errno on error.*/int (*alloc)(struct alloc_device_t* dev,int w, int h, int format, int usage,buffer_handle_t* handle, int* stride);

这个方法就是分配显存,传入参数 宽、高、格式、使用用途。返回 buffer_handle_t、步长。

上述对 HAL层头文件简单浏览,有助于我们对这几个概念有更深的印象。

相关文章:

Android SurfaceFlinger做Layer合成时,如何与HAL层进行交互

目录 零、本文讨论问题的范围一、问题&#xff1a;SurfaceFlinger图层合成选择实现方式的两难1.1 从OpenGL ES、HWC本身来讲1.2 以HWC为主导的判断逻辑 二、SurfaceFlinger与HAL层进行交互的具体实现框架2.1 SurfaceFlinger 调用 OpenGL ES 流程2.2 FrameBuffer2.3 SurfaceFlin…...

华为eNSP配置专题-策略路由的配置

文章目录 华为eNSP配置专题-策略路由的配置0、概要介绍1、前置环境1.1、宿主机1.2、eNSP模拟器 2、基本环境搭建2.1、终端构成和连接2.2、终端的基本配置 3、配置接入交换机上的VLAN4、配置核心交换机为网关和DHCP服务器5、配置核心交换机和出口路由器互通6、配置PC和出口路由器…...

JAVA实现智能停车场管理系统 开源

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容A. 车主端功能B. 停车工作人员功能C. 系统管理员功能1. 停车位模块2. 车辆模块3. 停车记录模块4. IC卡模块5. IC卡挂失模块 三、界面展示3.1 登录注册3.2 车辆模块3.3 停车位模块3.4 停车数据模块3.5 IC卡档案模块3.6 IC卡挂…...

深入理解Docker之:存储卷相关概念详解和分析

深入理解Docker之&#xff1a;存储卷相关概念详解和分析 1. 为什么要使用存储卷 Docker镜像由多个只读层叠加而成&#xff0c;启动容器时&#xff0c;Docker会加载只读镜像层&#xff0c;并在镜像栈顶部添加一个读写层如果运行中的容器修改了现有的一个已经存在的文件&#x…...

Node.js的基本概念node -v 和npm -v 这两个命令的作用

Node.js 是一个开源且跨平台的 JavaScript 运行时环境&#xff0c;它可以让你在服务器端运行 JavaScript 代码。Node.js 使用了 Chrome 的 V8 JavaScript 引擎来执行代码&#xff0c;非常高效。 在 Node.js 出现之前&#xff0c;JavaScript 通常只在浏览器中运行&#xff0c;用…...

mysql bin_log日志恢复数据

1、开启bin_log日志 开启方式1 my.ini 下配置开启或者vi /etc/my.cnf log_binmysql-bin server_id1 2、参考文章 https://blog.csdn.net/DreamEhome/article/details/130010601 (重点) 【mysql】binlog日志_mysql binlog日志-CSDN博客 MySQL 开启binlog日志和windows服务…...

C++系列之list的模拟实现

&#x1f497; &#x1f497; 博客:小怡同学 &#x1f497; &#x1f497; 个人简介:编程小萌新 &#x1f497; &#x1f497; 如果博客对大家有用的话&#xff0c;请点赞关注再收藏 &#x1f31e; list的节点类 template struct list_Node { public: list_Node* _prev; list_…...

什么情况下你会使用AI工具(chatgpt、bard)?

在当今数字化和智能化的时代&#xff0c;AI工具已成为许多领域的常见工具。在本文中&#xff0c;我将探讨什么情况下会使用AI工具。前言 – 人工智能教程 ChatGPT是一款由OpenAI开发的大型语言模型&#xff0c;可以生成文本、翻译语言、编写不同类型的创意内容&#xff0c;并以…...

【go】两数求和

文章目录 题目代码解法2 代码仓库 题目 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数&#xff0c;并返回它们的数组下标。 你可以假设每种输入只会对应一个答案。但是&#xff0c;数组中同一个元素在答案…...

软考高项-成本管理

工具和技术 三点估算 通过考虑估算中的不确定性与风险&#xff0c;使用3种估算值来界定活动成本的近似区间&#xff0c;可以提高活动成本估算的准确性&#xff1b; 储备分析 为应对成本的不确定性&#xff0c;成本估算中可以包括应急储备。应急储备的管理方法&#xff1a; 将…...

24年FRM备考知识点以及一级公式表

FRM一级公示表以及备考知识点 链接&#xff1a;https://pan.baidu.com/s/17RpFF9OyfRk7FGtEQrxf3A?pwd1234 提取码&#xff1a;1234 FRM二级公示表以及备考知识点 链接&#xff1a;https://pan.baidu.com/s/175D05wV1p94dIfBZThutCQ?pwd1234 提取码&#xff1a;1234...

Spring Cloud学习:二【详细】

目录 Nacos的配置 Nacos的单机启动 服务注册 Nacos服务分级存储模型 优先访问同集群的服务 根据权重负载均衡 环境隔离Namespace Nacos调用流程 Nacos与Eureka注册对比 Nacos与Eureka的共同点 Nacos与Eureka的区别 Nacos配置管理 统一配置 配置自动刷新 多环境配…...

Unity的live2dgalgame多语言可配置剧情框架

这段代码用于读取表格 using System; using System.Collections; using System.Collections.Generic; using UnityEngine; using OfficeOpenXml; using System.IO; using UnityEngine.Networking; using UnityEngine.UI; using Random UnityEngine.Random;public class Plots…...

再畅通工程(最小生成树)

题目描述&#xff1a;还是畅通工程 某省调查乡村交通状况&#xff0c;得到的统计表中列出了任意两村庄间的距离。省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通&#xff08;但不一定有直接的公路相连&#xff0c;只要能间接通过公路可达即可&#xff09;&…...

前后端分离不可忽视的陷阱,深入剖析挑战,分享解决方案,助你顺利实施分离开发。

不管你设计的系统架构是怎么样&#xff0c;最后都是你的组织内的沟通结构胜出。这个观点一直在组织内不断地被证明&#xff0c;但也不断地被忽略。 前后端分离的利与弊 近几年&#xff0c;随着微服务架构风格的引入、前后端生态的快速发展、多端产品化的出现&#xff0c;前后…...

(四)库存超卖案例实战——优化redis分布式锁

前言 在上一节内容中&#xff0c;我们已经实现了使用redis分布式锁解决商品“超卖”的问题&#xff0c;本节内容是对redis分布式锁的优化。在上一节的redis分布式锁中&#xff0c;我们的锁有俩个可以优化的问题。第一&#xff0c;锁需要实现可重入&#xff0c;同一个线程不用重…...

【ROS入门】雷达、摄像头及kinect信息仿真以及显示

文章结构 雷达信息仿真以及显示Gazebo仿真雷达配置雷达传感器信息xacro文件集成启动仿真环境 Rviz显示雷达数据 摄像头信息仿真以及显示Gazebo仿真摄像头新建xacro文件&#xff0c;配置摄像头传感器信息xacro文件集成启动仿真环境 Rviz显示摄像头数据 kinect信息仿真以及显示Ga…...

实用篇-认识微服务

一、服务架构演变 1. 单体架构 单体架构&#xff1a;将业务的所有功能集中在一个项目中开发&#xff0c;打成一个包部署 单体架构的优点&#xff1a; 架构简单部署成本低 单体架构的缺点&#xff1a; 耦合度高 2. 分布式架构 分布式架构&#xff1a; 根据业务功能对系…...

【产品运营】产品需求应该如何管理

产品项目在进行时经常会有一些需求需要实现&#xff0c;需求是产品更新迭代的动力&#xff0c;需求也是从用户诉求转化而来&#xff1b;在做需求管理时&#xff0c;我们需要判断一个需求的优先级等方面&#xff0c;对产品进行优化&#xff1b; 目录&#xff1a; 一、 为什么要…...

Linux 系统调用IO口,利用光标偏移实现文件复制

用系统调用IO函数实现从一个文件读取最后2KB数据并复制到另一个文件中&#xff0c;源文件以只读方式打开&#xff0c;目标文件以只写的方式打开&#xff0c;若目标文件不存在&#xff0c;可以创建并设置初始值为0664&#xff0c;写出相应代码&#xff0c;要对出错情况有一定的处…...

【原创】指针变量作为函数参数要点注意

指针变量作为函数参数要点注意&#xff08;已写至笔记&#xff09; 1传参指针不加*&#xff08;main中函数&#xff09; 2收参指针要加*&#xff08;被main调用的函数&#xff09; 3传参指针名可与收参指针名不同&#xff0c;不影响 4【问】如何看主函数中指针所指内容是否改变…...

SpringMVC Day 04 : 数据绑定

前言 SpringMVC是一个非常流行的Java Web框架&#xff0c;它提供了很多方便的功能和工具来帮助我们构建高效、灵活的Web应用程序。其中&#xff0c;数据绑定就是SpringMVC中非常重要的一部分&#xff0c;它可以帮助我们方便地将请求参数绑定到Java对象上&#xff0c;从而简化了…...

2.3.1 协程设计原理与汇编实现

1.为什么要有协程&#xff1f; 同步的编程方式&#xff0c;异步的性能。同步编程时&#xff0c;我们需要等待io就绪。但是在协程这里&#xff0c;我们使用一种机制&#xff0c;当io需要等待时&#xff0c;就切到下一个io&#xff0c;之后当之前的io就绪时&#xff0c;再切换回来…...

J2EE项目部署与发布(Windows版本)->会议OA单体项目Windows部署,spa前后端分离项目Windows部署

会议OA单体项目Windows部署spa前后端分离项目Windows部署 1.会议OA单体项目Windows部署&#xff08;以实施的角度&#xff09; 将项目放入webapp&#xff0c;项目能够访问: 首先拿到war包和数据库脚本&#xff0c;并检查是否有什么问题。 如何查看项目报错信息&#xff08;当你…...

Lua脚本语言

1. 概念 Lua&#xff08;发音为"loo-ah"&#xff0c;葡萄牙语中的"lua"意为月亮&#xff09;是一种轻量级的、高效的、可嵌入的脚本编程语言。官网Lua最初由巴西计算机科学家Roberto Ierusalimschy、Waldemar Celes和Luiz Henrique de Figueiredo于1993年开…...

cat()函数和print()函数的区别

目录 区别一&#xff1a; 区别二&#xff1a; cat、print函数都是输出函数。 区别一&#xff1a; cat()函数不能赋值&#xff1b; print()函数可以赋值。 x<-cat("hello world") //赋值 hello world x //cat函数无返回值 NULLy<-print("hello …...

宝塔面板安装Python和Flask(新版Python项目)

&#xff08;一&#xff09;宝塔面板的项目菜单&#xff0c;打开Python项目的“项目版本管理” 安装Python版本3.10.0。 会创建一个Python版本的文件夹www/server/pyproject_evn/versions/ 会创建一个Python虚拟环境的文件夹www/server/pyproject_evn/python_venv/ &#xf…...

火柴排队.

题意&#xff1a;给两列火柴&#xff0c;可以交换任意相邻的火柴&#xff0c;使得&#xff08;ai-bi)^2的和最小&#xff0c;求最小交换次数。 分析&#xff1a;使得&#xff08;ai-bi)^2的和最小&#xff0c;即a^2-2abb^2的和最小&#xff0c;那么使得2ab最大&#xff0c;就可…...

改善游戏体验:数据分析与可视化的威力

当今&#xff0c;电子游戏已经超越了娱乐&#xff0c;成为一种文化现象&#xff0c;汇聚了全球数十亿的玩家。游戏制作公司正采用越来越复杂的技术来提高游戏质量&#xff0c;同时游戏数据分析和可视化工具变得不可或缺。 数据的力量&#xff1a;解析游戏体验 游戏制作涉及到大…...

GEE:本地影像上传到GEE的Assets中,并输入机器学习算法中作为特征变量

作者:CSDN @ _养乐多_ 当我们在 Google Earth Engine(GEE)中应用机器学习算法时,会输入一些影像作为特征变量数据,进一步根据这些特征影像去推理未知区域的数据。但是 GEE 平台上计算特征变量的 API 函数并不是非常全面,我们希望获得更多的特征用于分类。这个时候,我们…...