17 bool GetHardwareBufferProperties(
18 const vk::Device& device,
19 struct AHardwareBuffer* hardware_buffer,
20 ::impeller::vk::AndroidHardwareBufferPropertiesANDROID* ahb_props,
21 ::impeller::vk::AndroidHardwareBufferFormatPropertiesANDROID*
23 FML_CHECK(ahb_format_props !=
nullptr);
24 FML_CHECK(ahb_props !=
nullptr);
25 ahb_props->pNext = ahb_format_props;
26 ::impeller::vk::Result result =
27 device.getAndroidHardwareBufferPropertiesANDROID(hardware_buffer,
29 if (result != impeller::vk::Result::eSuccess) {
35 vk::ExternalFormatANDROID MakeExternalFormat(
36 const vk::AndroidHardwareBufferFormatPropertiesANDROID& format_props) {
37 vk::ExternalFormatANDROID external_format;
38 external_format.pNext =
nullptr;
39 external_format.externalFormat = 0;
40 if (format_props.format == vk::Format::eUndefined) {
41 external_format.externalFormat = format_props.externalFormat;
43 return external_format;
47 int FindMemoryTypeIndex(
48 const vk::AndroidHardwareBufferPropertiesANDROID& props) {
49 uint32_t memory_type_bits = props.memoryTypeBits;
50 int32_t type_index = -1;
51 for (uint32_t i = 0; memory_type_bits;
52 memory_type_bits = memory_type_bits >> 0x1, ++i) {
53 if (memory_type_bits & 0x1) {
63 AndroidHardwareBufferTextureSourceVK::AndroidHardwareBufferTextureSourceVK(
64 TextureDescriptor desc,
65 const vk::Device& device,
66 struct AHardwareBuffer* hardware_buffer,
67 const AHardwareBuffer_Desc& hardware_buffer_desc)
68 : TextureSourceVK(desc), device_(device) {
69 vk::AndroidHardwareBufferFormatPropertiesANDROID ahb_format_props;
70 vk::AndroidHardwareBufferPropertiesANDROID ahb_props;
71 if (!GetHardwareBufferProperties(device, hardware_buffer, &ahb_props,
75 vk::ExternalFormatANDROID external_format =
76 MakeExternalFormat(ahb_format_props);
77 vk::ExternalMemoryImageCreateInfo external_memory_image_info;
78 external_memory_image_info.pNext = &external_format;
79 external_memory_image_info.handleTypes =
80 vk::ExternalMemoryHandleTypeFlagBits::eAndroidHardwareBufferANDROID;
81 const int memory_type_index = FindMemoryTypeIndex(ahb_props);
82 if (memory_type_index < 0) {
83 FML_LOG(ERROR) <<
"Could not find memory type.";
87 vk::ImageCreateFlags image_create_flags;
88 vk::ImageUsageFlags image_usage_flags;
89 if (hardware_buffer_desc.usage & AHARDWAREBUFFER_USAGE_GPU_SAMPLED_IMAGE) {
90 image_usage_flags |= impeller::vk::ImageUsageFlagBits::eSampled |
91 impeller::vk::ImageUsageFlagBits::eInputAttachment;
93 if (hardware_buffer_desc.usage & AHARDWAREBUFFER_USAGE_GPU_COLOR_OUTPUT) {
94 image_usage_flags |= impeller::vk::ImageUsageFlagBits::eColorAttachment;
96 if (hardware_buffer_desc.usage & AHARDWAREBUFFER_USAGE_PROTECTED_CONTENT) {
97 image_create_flags |= impeller::vk::ImageCreateFlagBits::eProtected;
100 vk::ImageCreateInfo image_create_info;
101 image_create_info.pNext = &external_memory_image_info;
102 image_create_info.imageType = vk::ImageType::e2D;
103 image_create_info.format = ahb_format_props.format;
104 image_create_info.extent.width = hardware_buffer_desc.width;
105 image_create_info.extent.height = hardware_buffer_desc.height;
106 image_create_info.extent.depth = 1;
107 image_create_info.mipLevels = 1;
108 image_create_info.arrayLayers = 1;
109 image_create_info.samples = vk::SampleCountFlagBits::e1;
110 image_create_info.tiling = vk::ImageTiling::eOptimal;
111 image_create_info.usage = image_usage_flags;
112 image_create_info.flags = image_create_flags;
113 image_create_info.sharingMode = vk::SharingMode::eExclusive;
114 image_create_info.initialLayout = vk::ImageLayout::eUndefined;
116 vk::ResultValue<impeller::vk::Image> maybe_image =
117 device.createImage(image_create_info);
118 if (maybe_image.result != vk::Result::eSuccess) {
119 FML_LOG(ERROR) <<
"device.createImage failed: "
120 <<
static_cast<int>(maybe_image.result);
123 vk::Image image = maybe_image.value;
125 vk::ImportAndroidHardwareBufferInfoANDROID ahb_import_info;
126 ahb_import_info.pNext =
nullptr;
127 ahb_import_info.buffer = hardware_buffer;
129 vk::MemoryDedicatedAllocateInfo dedicated_alloc_info;
130 dedicated_alloc_info.pNext = &ahb_import_info;
131 dedicated_alloc_info.image = image;
132 dedicated_alloc_info.buffer = VK_NULL_HANDLE;
134 vk::MemoryAllocateInfo mem_alloc_info;
135 mem_alloc_info.pNext = &dedicated_alloc_info;
136 mem_alloc_info.allocationSize = ahb_props.allocationSize;
137 mem_alloc_info.memoryTypeIndex = memory_type_index;
139 vk::ResultValue<vk::DeviceMemory> allocate_result =
140 device.allocateMemory(mem_alloc_info);
141 if (allocate_result.result != vk::Result::eSuccess) {
142 FML_LOG(ERROR) <<
"vkAllocateMemory failed : "
143 <<
static_cast<int>(allocate_result.result);
144 device.destroyImage(image);
147 vk::DeviceMemory device_memory = allocate_result.value;
150 vk::Result bind_image_result =
151 device.bindImageMemory(image, device_memory, 0);
152 if (bind_image_result != vk::Result::eSuccess) {
153 FML_LOG(ERROR) <<
"vkBindImageMemory failed : "
154 <<
static_cast<int>(bind_image_result);
155 device.destroyImage(image);
156 device.freeMemory(device_memory);
160 device_memory_ = device_memory;
163 vk::ImageViewCreateInfo view_info;
164 view_info.image = image_;
165 view_info.viewType = vk::ImageViewType::e2D;
167 view_info.subresourceRange.aspectMask = vk::ImageAspectFlagBits::eColor;
168 view_info.subresourceRange.baseMipLevel = 0u;
169 view_info.subresourceRange.baseArrayLayer = 0u;
170 view_info.subresourceRange.levelCount = desc.mip_count;
172 auto [view_result, view] = device.createImageViewUnique(view_info);
173 if (view_result != vk::Result::eSuccess) {
174 FML_LOG(ERROR) <<
"createImageViewUnique failed : "
175 <<
static_cast<int>(view_result);
178 image_view_ = std::move(view);
183 AndroidHardwareBufferTextureSourceVK::~AndroidHardwareBufferTextureSourceVK() {
184 device_.destroyImage(image_);
185 device_.freeMemory(device_memory_);
188 bool AndroidHardwareBufferTextureSourceVK::IsValid()
const {
193 vk::Image AndroidHardwareBufferTextureSourceVK::GetImage()
const {
194 FML_CHECK(IsValid());
199 vk::ImageView AndroidHardwareBufferTextureSourceVK::GetImageView()
const {
200 FML_CHECK(IsValid());
201 return image_view_.get();