添加到 Android 的所有新像素格式都必须包含在 Android 接口定义语言 (AIDL) 和 Android 硬件缓冲区 (AHB) 中。AIDL 和 AHB 对稳定性和标准化有严格的要求,因此在扩展功能时需要谨慎处理。所有新的像素格式都必须在 AOSP 中落实,并且所有更新都必须由 AIDL 和 AHB 专家单独确认。这种谨慎确认的过程是在平台上标准化任何新像素格式的重要因素。
本页概述了在 AOSP 上添加新像素格式所需的 AOSP 代码变更和流程。
在添加新的像素格式之前,请按照提交补丁中的概述下载源代码并上传补丁。向 AIDL 添加新的像素格式
添加对新像素格式的支持需要同时更改 AIDL 中的两个 PixelFormat.aidl
文件。有关 AIDL 源代码,请参阅 hardware/interfaces/graphics/common/aidl/
。
要向 AIDL 添加新的像素格式,请按照以下步骤操作:
- 按照现有的代码约定,将新的像素格式作为新条目附加到
PixelFormat.aidl
中的PixelFormat
枚举的末尾,并将您的条目的十六进制值设置为比上一个条目大 1。使您的代码更改与之前的条目相匹配。有关RGBA_8888
像素格式条目的示例,请参阅以下内容:/** * 32-bit format that has 8-bit R, G, B, and A components, in that order, * from the lowest memory address to the highest memory address. * * The component values are unsigned normalized to the range [0, 1], whose * interpretation is defined by the dataspace. */ RGBA_8888 = 0x1,
在对
PixelFormat.aidl
进行更改后,当您构建代码时,会看到以下错误消息:android_developer:~/android/aosp-main: m ... ############################################################################### # ERROR: AIDL API change detected # ############################################################################### Above AIDL file(s) has changed. Run `m android.hardware.graphics.common-update-api` to reflect the changes to the current version so that it is reviewed by android-aidl-api-council@google.com And then you need to change dependency on android.hardware.graphics.common-V(n)-* to android.hardware.graphics.common-V(n+1)-* to use new APIs.
-
要清除此错误,请运行以下命令(如错误消息中所述)以更改
aidl_api
目录中的PixelFormat.aidl
:m android.hardware.graphics.common-update-api
运行上述命令会更新正确的文件,以便能够正常构建。
向 AHB 添加新的像素格式
添加对新像素格式的支持需要更改 hardware_buffer.h
和 AHardwareBuffer.cpp
。有关 AHB 源代码,请参阅 frameworks/native/libs/nativewindow
。
要向 AHB 添加新的像素格式,请按照以下步骤操作:
- 在
hardware_buffer.h
中,将新的像素格式作为新条目附加到AHardwareBuffer_Format
枚举的末尾。遵循现有的代码约定。使用
RGBA_8888
像素格式示例,添加新的像素格式条目,如下所示:/** * Corresponding formats: * Vulkan: VK_FORMAT_R8G8B8A8_UNORM * OpenGL ES: GL_RGBA8 */ AHARDWAREBUFFER_FORMAT_R8G8B8A8_UNORM = 1,
请注意,新的像素格式在 AHB 中被赋予一个名称,该名称必须以
AHARDWAREBUFFER_FORMAT_
开头,后跟通道缩写和位深度,并以编码结尾。此枚举条目必须具有与PixelFormat.aidl
中相同的十六进制值。像素格式应具有关联的 Vulkan 或 OpenGL ES 格式之一或两者。在适当的位置指定关联的格式。如果不存在关联的格式,请指定
N/A
。 -
如果像素格式具有关联的 OpenGL ES 格式,则将其添加到 CTS 下的可选测试中。为此,请使用新的格式的
FORMAT_CASE(...)
和GL_FORMAT_CASE(...)
将新的 GL 格式添加到AHBFormatAsString(int32_t format)
中的AHardwareBufferGLTest.cpp
中,如下所示:const char* AHBFormatAsString(int32_t format) { switch (format) { ... FORMAT_CASE(R8G8B8A8_UNORM); ... GL_FORMAT_CASE(GL_RGB8); } return ""; }
-
接下来,向
AHardwareBufferGLTest.cpp
添加一个新测试,如下所示:class RGBA8Test : public AHardwareBufferGLTest {}; // Verify that if we can allocate an RGBA8 AHB we can render to it. TEST_P(RGBA8Test, Write) { AHardwareBuffer_Desc desc = GetParam(); desc.usage = AHARDWAREBUFFER_USAGE_GPU_FRAMEBUFFER; if (!SetUpBuffer(desc)) { return; } ASSERT_NO_FATAL_FAILURE(SetUpFramebuffer(desc.width, desc.height, 0, kBufferAsRenderbuffer)); ASSERT_NO_FATAL_FAILURE( SetUpProgram(kVertexShader, kColorFragmentShader, kPyramidPositions, 0.5f)); glDrawArrays(GL_TRIANGLES, 0, kPyramidVertexCount); ASSERT_EQ(GLenum{GL_NO_ERROR}, glGetError()); } INSTANTIATE_TEST_CASE_P( SingleLayer, RGBA8Test, ::testing::Values( AHardwareBuffer_Desc{57, 33, 1, AHARDWAREBUFFER_FORMAT_R16G16_UINT, 0, 0, 0, 0}), &GetTestName);
至少指定一组
AHardwareBuffer_Desc
值。如果需要,添加更多值。 -
在
AHardwareBuffer.cpp
中,找到静态断言的结尾,它们位于// ---------------------------------------------------------------------------- // Validate hardware_buffer.h and PixelFormat.aidl agree // ----------------------------------------------------------------------------
添加一个新的
static_assert
用于新的像素格式,使用PixelFormat::
枚举,而不是HAL_PIXEL_FORMAT
常量。使用 向 AIDL 添加新的像素格式 中RGBA_8888
像素格式的相同示例,按如下方式添加新的像素格式条目static_assert(static_cast
(aidl::android::hardware::graphics::common::PixelFormat::RGBA_8888) == AHARDWAREBUFFER_FORMAT_R8G8B8A8_UNORM, "HAL and AHardwareBuffer pixel format don't match"); -
通过将新的像素格式附加到
AHardwareBufferTest.cpp
中的PrintAhbFormat()
的末尾,将新的像素格式添加到相应的测试中。遵循现有的代码约定,如下所示void PrintAhbFormat(std::ostream& os, uint64_t format) { switch (format) { ... FORMAT_CASE(R8G8B8A8_UNORM); default: os << "unknown"; break; } }
-
通过将新条目附加到
@IntDef
,将新的像素格式添加到HardwareBuffer.java
中的HardwareBuffer
SDK 中。例如,RGBA_8888
格式的条目显示如下@Retention(RetentionPolicy.SOURCE) @IntDef(prefix = { "RGB", "BLOB", "YCBCR_", "D_", "DS_", "S_" }, value = { ... RGBA_8888, })
如果组件值不是无符号归一化值,则在变量名称中显式指示该值。例如,仅用于无符号整数 16 位红色通道格式的变量名称必须为
R_16UI
,而具有附加的无符号整数 16 位绿色通道格式的相同格式必须为RG_16UI16UI
。 -
通过在
HardwareBuffer.java
中将新的像素格式添加为static int
,方法是在@Format
的末尾附加一个新的公共成员变量@Format ... /** Format: 8 bits each red, green, blue, alpha */ public static final int RGBA_8888 = 0x1;
此枚举条目必须具有与
PixelFormat.aidl
和hardware_buffer.h
中的十六进制值相同的值。遵循现有约定。 -
尝试使用这些代码更改进行构建会生成构建错误
android_developer:~/android/aosp-main: m ... ****************************** You have tried to change the API from what has been previously approved. To make these errors go away, you have two choices: 1. You can add '@hide' javadoc comments (and remove @SystemApi/@TestApi/etc) to the new methods, etc. shown in the above diff. 2. You can update current.txt and/or removed.txt by executing the following command: m api-stubs-docs-non-updatable-update-current-api To submit the revised current.txt to the main Android repository, you will need approval. ****************************** ...
要清除此错误,请运行错误消息中指定的以下命令,以更改
current.txt
m api-stubs-docs-non-updatable-update-current-api
运行上述命令会更新正确的文件,以便能够正常构建。
-
通过将新的像素格式附加到
HardwareBufferTest.java
中的paramsForTestCreateOptionalFormats()
的末尾,将新的像素格式添加到 Java 测试中,如下所示private static Object[] paramsForTestCreateOptionalFormats() { return new Integer[]{ HardwareBuffer.RGBA_8888 };
向窗口系统集成添加新的像素格式
要将新的像素格式用作图形 API 中帧缓冲区的格式,请将其添加到相关图形 API 的相应窗口系统集成 (WSI) 中。对于使用 Vulkan API 的应用或系统进程,请更新 Vulkan 交换链。对于使用 OpenGL ES API 的应用或系统进程,请更新 EGL API。
新像素格式的 Vulkan WSI 更改
按如下方式更新 Vulkan WSI-
在
swapchain.cpp
中的GetNativePixelFormat(VkFormat format)
函数中添加新的 caseandroid::PixelFormat GetNativePixelFormat(VkFormat format) { ... switch (format) { ... case VK_FORMAT_R8G8B8A8_UNORM: native_format = PixelFormat::RGBA_8888; break; ... default: ALOGV("unsupported swapchain format %d", format); break; } return native_format; }
- 如果像素格式需要 Vulkan 扩展才能运行,请查询 Vulkan 扩展。对于实例端扩展,请使用
instance_data
,如下所示bool colorspace_ext = instance_data.hook_extensions.test(ProcHook::EXT_swapchain_colorspace);
对于设备端扩展,请使用以下内容
bool rgba10x6_formats_ext = false; uint32_t exts_count; const auto& driver = GetData(pdev).driver; driver.EnumerateDeviceExtensionProperties(pdev, nullptr, &exts_count, nullptr); std::vector
props(exts_count); driver.EnumerateDeviceExtensionProperties(pdev, nullptr, &exts_count, props.data()); for (uint32_t i = 0; i < exts_count; i++) { VkExtensionProperties prop = props[i]; if (strcmp(prop.extensionName, VK_EXT_RGBA10X6_FORMATS_EXTENSION_NAME) == 0) { rgba10x6_formats_ext = true; } } Google 处理将实例或设备扩展公开给
swapchain.cpp
所需的基础架构。初始更改列表不需要从 Vulkan 加载程序正确设置扩展。 - 接下来,枚举格式和颜色空间对
desc.format = AHARDWAREBUFFER_FORMAT_R10G10B10A10_UNORM; if (AHardwareBuffer_isSupported(&desc) && rgba10x6_formats_ext) { all_formats.emplace_back( VkSurfaceFormatKHR{VK_FORMAT_R10X6G10X6B10X6A10X6_UNORM_4PACK16, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR}); if (colorspace_ext) { all_formats.emplace_back( VkSurfaceFormatKHR{VK_FORMAT_R10X6G10X6B10X6A10X6_UNORM_4PACK16, VK_COLOR_SPACE_PASS_THROUGH_EXT}); all_formats.emplace_back( VkSurfaceFormatKHR{VK_FORMAT_R10X6G10X6B10X6A10X6_UNORM_4PACK16, VK_COLOR_SPACE_DISPLAY_P3_NONLINEAR_EXT}); }
您必须了解兼容的格式和颜色空间对。
- 将新格式添加到位于
external/deqp
的dEQP-VK
。 - 通过从现有源代码推断所需的更改或联系您的 Android 支持以获取信息,更新
vktApiExternalMemoryTests.cpp
和vktExternalMemoryUtil.cpp
中的 Vulkan 一致性测试。
新像素格式的 EGL 更改
按如下方式更新 EGL
- 在
getNativePixelFormat()
函数中,修改if-else
树以返回新像素格式的 AIDL 枚举。使用RGBA_8888
像素格式的示例if (a == 0) { ... } else { if (componentType == EGL_COLOR_COMPONENT_TYPE_FIXED_EXT) { if (colorDepth > 24) { ... } else { *format = PixelFormat::RGBA_8888; } } else { ... } }
- 要将新格式添加到 dEQP,请向
androidFormats
枚举添加新条目,如下所示static const GLenum androidFormats[] = { ... GL_RGBA8, ... };
提交您的更新
按照 贡献者指南 启动您的更改列表并与相应的团队分享。