1 /*
2 * Copyright © 2021 Intel Corporation
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice (including the next
12 * paragraph) shall be included in all copies or substantial portions of the
13 * Software.
14 *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21 * IN THE SOFTWARE.
22 */
23
24 #include "vk_image.h"
25
26 #if DETECT_OS_LINUX || DETECT_OS_BSD
27 #include <drm-uapi/drm_fourcc.h>
28 #endif
29
30 #include "vk_alloc.h"
31 #include "vk_common_entrypoints.h"
32 #include "vk_device.h"
33 #include "vk_format.h"
34 #include "vk_format_info.h"
35 #include "vk_log.h"
36 #include "vk_physical_device.h"
37 #include "vk_render_pass.h"
38 #include "vk_util.h"
39 #include "vulkan/wsi/wsi_common.h"
40
41 #if DETECT_OS_ANDROID
42 #include "vk_android.h"
43 #include <vulkan/vulkan_android.h>
44 #endif
45
46 void
vk_image_init(struct vk_device * device,struct vk_image * image,const VkImageCreateInfo * pCreateInfo)47 vk_image_init(struct vk_device *device,
48 struct vk_image *image,
49 const VkImageCreateInfo *pCreateInfo)
50 {
51 vk_object_base_init(device, &image->base, VK_OBJECT_TYPE_IMAGE);
52
53 assert(pCreateInfo->sType == VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO);
54 assert(pCreateInfo->mipLevels > 0);
55 assert(pCreateInfo->arrayLayers > 0);
56 assert(pCreateInfo->samples > 0);
57 assert(pCreateInfo->extent.width > 0);
58 assert(pCreateInfo->extent.height > 0);
59 assert(pCreateInfo->extent.depth > 0);
60
61 if (pCreateInfo->flags & VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT)
62 assert(pCreateInfo->imageType == VK_IMAGE_TYPE_2D);
63 if (pCreateInfo->flags & VK_IMAGE_CREATE_2D_ARRAY_COMPATIBLE_BIT)
64 assert(pCreateInfo->imageType == VK_IMAGE_TYPE_3D);
65
66 image->create_flags = pCreateInfo->flags;
67 image->image_type = pCreateInfo->imageType;
68 vk_image_set_format(image, pCreateInfo->format);
69 image->extent = vk_image_sanitize_extent(image, pCreateInfo->extent);
70 image->mip_levels = pCreateInfo->mipLevels;
71 image->array_layers = pCreateInfo->arrayLayers;
72 image->samples = pCreateInfo->samples;
73 image->tiling = pCreateInfo->tiling;
74 image->usage = pCreateInfo->usage;
75 image->sharing_mode = pCreateInfo->sharingMode;
76
77 if (image->aspects & VK_IMAGE_ASPECT_STENCIL_BIT) {
78 const VkImageStencilUsageCreateInfo *stencil_usage_info =
79 vk_find_struct_const(pCreateInfo->pNext,
80 IMAGE_STENCIL_USAGE_CREATE_INFO);
81 image->stencil_usage =
82 stencil_usage_info ? stencil_usage_info->stencilUsage :
83 pCreateInfo->usage;
84 } else {
85 image->stencil_usage = 0;
86 }
87
88 const VkExternalMemoryImageCreateInfo *ext_mem_info =
89 vk_find_struct_const(pCreateInfo->pNext, EXTERNAL_MEMORY_IMAGE_CREATE_INFO);
90 if (ext_mem_info)
91 image->external_handle_types = ext_mem_info->handleTypes;
92 else
93 image->external_handle_types = 0;
94
95 const struct wsi_image_create_info *wsi_info =
96 vk_find_struct_const(pCreateInfo->pNext, WSI_IMAGE_CREATE_INFO_MESA);
97 image->wsi_legacy_scanout = wsi_info && wsi_info->scanout;
98
99 #if DETECT_OS_LINUX || DETECT_OS_BSD
100 image->drm_format_mod = ((1ULL << 56) - 1) /* DRM_FORMAT_MOD_INVALID */;
101 #endif
102
103 #if DETECT_OS_ANDROID
104 const VkExternalFormatANDROID *ext_format =
105 vk_find_struct_const(pCreateInfo->pNext, EXTERNAL_FORMAT_ANDROID);
106 if (ext_format && ext_format->externalFormat != 0) {
107 assert(image->format == VK_FORMAT_UNDEFINED);
108 assert(image->external_handle_types &
109 VK_EXTERNAL_MEMORY_HANDLE_TYPE_ANDROID_HARDWARE_BUFFER_BIT_ANDROID);
110 vk_image_set_format(image, (VkFormat)ext_format->externalFormat);
111 }
112
113 image->ahb_format = vk_image_format_to_ahb_format(image->format);
114 #endif
115 }
116
117 void *
vk_image_create(struct vk_device * device,const VkImageCreateInfo * pCreateInfo,const VkAllocationCallbacks * alloc,size_t size)118 vk_image_create(struct vk_device *device,
119 const VkImageCreateInfo *pCreateInfo,
120 const VkAllocationCallbacks *alloc,
121 size_t size)
122 {
123 struct vk_image *image =
124 vk_zalloc2(&device->alloc, alloc, size, 8,
125 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
126 if (image == NULL)
127 return NULL;
128
129 vk_image_init(device, image, pCreateInfo);
130
131 return image;
132 }
133
134 void
vk_image_finish(struct vk_image * image)135 vk_image_finish(struct vk_image *image)
136 {
137 vk_object_base_finish(&image->base);
138 }
139
140 void
vk_image_destroy(struct vk_device * device,const VkAllocationCallbacks * alloc,struct vk_image * image)141 vk_image_destroy(struct vk_device *device,
142 const VkAllocationCallbacks *alloc,
143 struct vk_image *image)
144 {
145 vk_object_free(device, alloc, image);
146 }
147
148 #if DETECT_OS_LINUX || DETECT_OS_BSD
149 VKAPI_ATTR VkResult VKAPI_CALL
vk_common_GetImageDrmFormatModifierPropertiesEXT(UNUSED VkDevice device,VkImage _image,VkImageDrmFormatModifierPropertiesEXT * pProperties)150 vk_common_GetImageDrmFormatModifierPropertiesEXT(UNUSED VkDevice device,
151 VkImage _image,
152 VkImageDrmFormatModifierPropertiesEXT *pProperties)
153 {
154 VK_FROM_HANDLE(vk_image, image, _image);
155
156 assert(pProperties->sType ==
157 VK_STRUCTURE_TYPE_IMAGE_DRM_FORMAT_MODIFIER_PROPERTIES_EXT);
158
159 assert(image->tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT);
160 pProperties->drmFormatModifier = image->drm_format_mod;
161
162 return VK_SUCCESS;
163 }
164 #endif
165
166 VKAPI_ATTR void VKAPI_CALL
vk_common_GetImageSubresourceLayout(VkDevice _device,VkImage _image,const VkImageSubresource * pSubresource,VkSubresourceLayout * pLayout)167 vk_common_GetImageSubresourceLayout(VkDevice _device, VkImage _image,
168 const VkImageSubresource *pSubresource,
169 VkSubresourceLayout *pLayout)
170 {
171 VK_FROM_HANDLE(vk_device, device, _device);
172
173 const VkImageSubresource2KHR subresource = {
174 .sType = VK_STRUCTURE_TYPE_IMAGE_SUBRESOURCE_2_KHR,
175 .imageSubresource = *pSubresource,
176 };
177
178 VkSubresourceLayout2KHR layout = {
179 .sType = VK_STRUCTURE_TYPE_SUBRESOURCE_LAYOUT_2_KHR
180 };
181
182 device->dispatch_table.GetImageSubresourceLayout2KHR(_device, _image,
183 &subresource, &layout);
184
185 *pLayout = layout.subresourceLayout;
186 }
187
188 void
vk_image_set_format(struct vk_image * image,VkFormat format)189 vk_image_set_format(struct vk_image *image, VkFormat format)
190 {
191 image->format = format;
192 image->aspects = vk_format_aspects(format);
193 }
194
195 VkImageUsageFlags
vk_image_usage(const struct vk_image * image,VkImageAspectFlags aspect_mask)196 vk_image_usage(const struct vk_image *image,
197 VkImageAspectFlags aspect_mask)
198 {
199 /* From the Vulkan 1.2.131 spec:
200 *
201 * "If the image was has a depth-stencil format and was created with
202 * a VkImageStencilUsageCreateInfo structure included in the pNext
203 * chain of VkImageCreateInfo, the usage is calculated based on the
204 * subresource.aspectMask provided:
205 *
206 * - If aspectMask includes only VK_IMAGE_ASPECT_STENCIL_BIT, the
207 * implicit usage is equal to
208 * VkImageStencilUsageCreateInfo::stencilUsage.
209 *
210 * - If aspectMask includes only VK_IMAGE_ASPECT_DEPTH_BIT, the
211 * implicit usage is equal to VkImageCreateInfo::usage.
212 *
213 * - If both aspects are included in aspectMask, the implicit usage
214 * is equal to the intersection of VkImageCreateInfo::usage and
215 * VkImageStencilUsageCreateInfo::stencilUsage.
216 */
217 if (aspect_mask == VK_IMAGE_ASPECT_STENCIL_BIT) {
218 return image->stencil_usage;
219 } else if (aspect_mask == (VK_IMAGE_ASPECT_DEPTH_BIT |
220 VK_IMAGE_ASPECT_STENCIL_BIT)) {
221 return image->usage & image->stencil_usage;
222 } else {
223 /* This also handles the color case */
224 return image->usage;
225 }
226 }
227
228 #define VK_IMAGE_ASPECT_ANY_COLOR_MASK_MESA ( \
229 VK_IMAGE_ASPECT_COLOR_BIT | \
230 VK_IMAGE_ASPECT_PLANE_0_BIT | \
231 VK_IMAGE_ASPECT_PLANE_1_BIT | \
232 VK_IMAGE_ASPECT_PLANE_2_BIT)
233
234 /** Expands the given aspect mask relative to the image
235 *
236 * If the image has color plane aspects VK_IMAGE_ASPECT_COLOR_BIT has been
237 * requested, this returns the aspects of the underlying image.
238 *
239 * For example,
240 *
241 * VK_IMAGE_ASPECT_COLOR_BIT
242 *
243 * will be converted to
244 *
245 * VK_IMAGE_ASPECT_PLANE_0_BIT |
246 * VK_IMAGE_ASPECT_PLANE_1_BIT |
247 * VK_IMAGE_ASPECT_PLANE_2_BIT
248 *
249 * for an image of format VK_FORMAT_G8_B8_R8_3PLANE_420_UNORM.
250 */
251 VkImageAspectFlags
vk_image_expand_aspect_mask(const struct vk_image * image,VkImageAspectFlags aspect_mask)252 vk_image_expand_aspect_mask(const struct vk_image *image,
253 VkImageAspectFlags aspect_mask)
254 {
255 if (aspect_mask == VK_IMAGE_ASPECT_COLOR_BIT) {
256 assert(image->aspects & VK_IMAGE_ASPECT_ANY_COLOR_MASK_MESA);
257 return image->aspects;
258 } else {
259 assert(aspect_mask && !(aspect_mask & ~image->aspects));
260 return aspect_mask;
261 }
262 }
263
264 VkExtent3D
vk_image_extent_to_elements(const struct vk_image * image,VkExtent3D extent)265 vk_image_extent_to_elements(const struct vk_image *image, VkExtent3D extent)
266 {
267 const struct util_format_description *fmt =
268 vk_format_description(image->format);
269
270 extent = vk_image_sanitize_extent(image, extent);
271 extent.width = DIV_ROUND_UP(extent.width, fmt->block.width);
272 extent.height = DIV_ROUND_UP(extent.height, fmt->block.height);
273 extent.depth = DIV_ROUND_UP(extent.depth, fmt->block.depth);
274
275 return extent;
276 }
277
278 VkOffset3D
vk_image_offset_to_elements(const struct vk_image * image,VkOffset3D offset)279 vk_image_offset_to_elements(const struct vk_image *image, VkOffset3D offset)
280 {
281 const struct util_format_description *fmt =
282 vk_format_description(image->format);
283
284 offset = vk_image_sanitize_offset(image, offset);
285
286 assert(offset.x % fmt->block.width == 0);
287 assert(offset.y % fmt->block.height == 0);
288 assert(offset.z % fmt->block.depth == 0);
289
290 offset.x /= fmt->block.width;
291 offset.y /= fmt->block.height;
292 offset.z /= fmt->block.depth;
293
294 return offset;
295 }
296
297 struct vk_image_buffer_layout
vk_image_buffer_copy_layout(const struct vk_image * image,const VkBufferImageCopy2 * region)298 vk_image_buffer_copy_layout(const struct vk_image *image,
299 const VkBufferImageCopy2* region)
300 {
301 VkExtent3D extent = vk_image_sanitize_extent(image, region->imageExtent);
302
303 const uint32_t row_length = region->bufferRowLength ?
304 region->bufferRowLength : extent.width;
305 const uint32_t image_height = region->bufferImageHeight ?
306 region->bufferImageHeight : extent.height;
307
308 const VkImageAspectFlags aspect = region->imageSubresource.aspectMask;
309 VkFormat format = vk_format_get_aspect_format(image->format, aspect);
310 const struct util_format_description *fmt = vk_format_description(format);
311
312 assert(fmt->block.bits % 8 == 0);
313 const uint32_t element_size_B = fmt->block.bits / 8;
314
315 const uint32_t row_stride_B =
316 DIV_ROUND_UP(row_length, fmt->block.width) * element_size_B;
317 const uint64_t image_stride_B =
318 DIV_ROUND_UP(image_height, fmt->block.height) * (uint64_t)row_stride_B;
319
320 return (struct vk_image_buffer_layout) {
321 .row_length = row_length,
322 .image_height = image_height,
323 .element_size_B = element_size_B,
324 .row_stride_B = row_stride_B,
325 .image_stride_B = image_stride_B,
326 };
327 }
328
329 struct vk_image_buffer_layout
vk_memory_to_image_copy_layout(const struct vk_image * image,const VkMemoryToImageCopyEXT * region)330 vk_memory_to_image_copy_layout(const struct vk_image *image,
331 const VkMemoryToImageCopyEXT* region)
332 {
333 const VkBufferImageCopy2 bic = {
334 .bufferOffset = 0,
335 .bufferRowLength = region->memoryRowLength,
336 .bufferImageHeight = region->memoryImageHeight,
337 .imageSubresource = region->imageSubresource,
338 .imageOffset = region->imageOffset,
339 .imageExtent = region->imageExtent,
340 };
341 return vk_image_buffer_copy_layout(image, &bic);
342 }
343
344 struct vk_image_buffer_layout
vk_image_to_memory_copy_layout(const struct vk_image * image,const VkImageToMemoryCopyEXT * region)345 vk_image_to_memory_copy_layout(const struct vk_image *image,
346 const VkImageToMemoryCopyEXT* region)
347 {
348 const VkBufferImageCopy2 bic = {
349 .bufferOffset = 0,
350 .bufferRowLength = region->memoryRowLength,
351 .bufferImageHeight = region->memoryImageHeight,
352 .imageSubresource = region->imageSubresource,
353 .imageOffset = region->imageOffset,
354 .imageExtent = region->imageExtent,
355 };
356 return vk_image_buffer_copy_layout(image, &bic);
357 }
358
359 static VkComponentSwizzle
remap_swizzle(VkComponentSwizzle swizzle,VkComponentSwizzle component)360 remap_swizzle(VkComponentSwizzle swizzle, VkComponentSwizzle component)
361 {
362 return swizzle == VK_COMPONENT_SWIZZLE_IDENTITY ? component : swizzle;
363 }
364
365 void
vk_image_view_init(struct vk_device * device,struct vk_image_view * image_view,bool driver_internal,const VkImageViewCreateInfo * pCreateInfo)366 vk_image_view_init(struct vk_device *device,
367 struct vk_image_view *image_view,
368 bool driver_internal,
369 const VkImageViewCreateInfo *pCreateInfo)
370 {
371 vk_object_base_init(device, &image_view->base, VK_OBJECT_TYPE_IMAGE_VIEW);
372
373 assert(pCreateInfo->sType == VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO);
374 VK_FROM_HANDLE(vk_image, image, pCreateInfo->image);
375
376 image_view->create_flags = pCreateInfo->flags;
377 image_view->image = image;
378 image_view->view_type = pCreateInfo->viewType;
379
380 image_view->format = pCreateInfo->format;
381 if (image_view->format == VK_FORMAT_UNDEFINED)
382 image_view->format = image->format;
383
384 if (!driver_internal) {
385 switch (image_view->view_type) {
386 case VK_IMAGE_VIEW_TYPE_1D:
387 case VK_IMAGE_VIEW_TYPE_1D_ARRAY:
388 assert(image->image_type == VK_IMAGE_TYPE_1D);
389 break;
390 case VK_IMAGE_VIEW_TYPE_2D:
391 case VK_IMAGE_VIEW_TYPE_2D_ARRAY:
392 if (image->create_flags & (VK_IMAGE_CREATE_2D_ARRAY_COMPATIBLE_BIT |
393 VK_IMAGE_CREATE_2D_VIEW_COMPATIBLE_BIT_EXT))
394 assert(image->image_type == VK_IMAGE_TYPE_3D);
395 else
396 assert(image->image_type == VK_IMAGE_TYPE_2D);
397 break;
398 case VK_IMAGE_VIEW_TYPE_3D:
399 assert(image->image_type == VK_IMAGE_TYPE_3D);
400 break;
401 case VK_IMAGE_VIEW_TYPE_CUBE:
402 case VK_IMAGE_VIEW_TYPE_CUBE_ARRAY:
403 assert(image->image_type == VK_IMAGE_TYPE_2D);
404 assert(image->create_flags & VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT);
405 break;
406 default:
407 unreachable("Invalid image view type");
408 }
409 }
410
411 const VkImageSubresourceRange *range = &pCreateInfo->subresourceRange;
412
413 if (driver_internal) {
414 image_view->aspects = range->aspectMask;
415 image_view->view_format = image_view->format;
416 } else {
417 image_view->aspects =
418 vk_image_expand_aspect_mask(image, range->aspectMask);
419
420 assert(!(image_view->aspects & ~image->aspects));
421
422 /* From the Vulkan 1.2.184 spec:
423 *
424 * "If the image has a multi-planar format and
425 * subresourceRange.aspectMask is VK_IMAGE_ASPECT_COLOR_BIT, and image
426 * has been created with a usage value not containing any of the
427 * VK_IMAGE_USAGE_VIDEO_DECODE_DST_BIT_KHR,
428 * VK_IMAGE_USAGE_VIDEO_DECODE_SRC_BIT_KHR,
429 * VK_IMAGE_USAGE_VIDEO_DECODE_DPB_BIT_KHR,
430 * VK_IMAGE_USAGE_VIDEO_ENCODE_DST_BIT_KHR,
431 * VK_IMAGE_USAGE_VIDEO_ENCODE_SRC_BIT_KHR, and
432 * VK_IMAGE_USAGE_VIDEO_ENCODE_DPB_BIT_KHR flags, then the format must
433 * be identical to the image format, and the sampler to be used with the
434 * image view must enable sampler Y′CBCR conversion."
435 *
436 * Since no one implements video yet, we can ignore the bits about video
437 * create flags and assume YCbCr formats match.
438 */
439 if ((image->aspects & VK_IMAGE_ASPECT_PLANE_1_BIT) &&
440 (range->aspectMask == VK_IMAGE_ASPECT_COLOR_BIT))
441 assert(image_view->format == image->format);
442
443 /* From the Vulkan 1.2.184 spec:
444 *
445 * "Each depth/stencil format is only compatible with itself."
446 */
447 if (image_view->aspects & (VK_IMAGE_ASPECT_DEPTH_BIT |
448 VK_IMAGE_ASPECT_STENCIL_BIT))
449 assert(image_view->format == image->format);
450
451 if (!(image->create_flags & VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT))
452 assert(image_view->format == image->format);
453
454 /* Restrict the format to only the planes chosen.
455 *
456 * For combined depth and stencil images, this means the depth-only or
457 * stencil-only format if only one aspect is chosen and the full
458 * combined format if both aspects are chosen.
459 *
460 * For single-plane color images, we just take the format as-is. For
461 * multi-plane views of multi-plane images, this means we want the full
462 * multi-plane format. For single-plane views of multi-plane images, we
463 * want a format compatible with the one plane. Fortunately, this is
464 * already what the client gives us. The Vulkan 1.2.184 spec says:
465 *
466 * "If image was created with the VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT
467 * and the image has a multi-planar format, and if
468 * subresourceRange.aspectMask is VK_IMAGE_ASPECT_PLANE_0_BIT,
469 * VK_IMAGE_ASPECT_PLANE_1_BIT, or VK_IMAGE_ASPECT_PLANE_2_BIT,
470 * format must be compatible with the corresponding plane of the
471 * image, and the sampler to be used with the image view must not
472 * enable sampler Y′CBCR conversion."
473 */
474 if (image_view->aspects == VK_IMAGE_ASPECT_STENCIL_BIT) {
475 image_view->view_format = vk_format_stencil_only(image_view->format);
476 } else if (image_view->aspects == VK_IMAGE_ASPECT_DEPTH_BIT) {
477 image_view->view_format = vk_format_depth_only(image_view->format);
478 } else {
479 image_view->view_format = image_view->format;
480 }
481 }
482
483 image_view->swizzle = (VkComponentMapping) {
484 .r = remap_swizzle(pCreateInfo->components.r, VK_COMPONENT_SWIZZLE_R),
485 .g = remap_swizzle(pCreateInfo->components.g, VK_COMPONENT_SWIZZLE_G),
486 .b = remap_swizzle(pCreateInfo->components.b, VK_COMPONENT_SWIZZLE_B),
487 .a = remap_swizzle(pCreateInfo->components.a, VK_COMPONENT_SWIZZLE_A),
488 };
489
490 assert(range->layerCount > 0);
491 assert(range->baseMipLevel < image->mip_levels);
492
493 image_view->base_mip_level = range->baseMipLevel;
494 image_view->level_count = vk_image_subresource_level_count(image, range);
495 image_view->base_array_layer = range->baseArrayLayer;
496 image_view->layer_count = vk_image_subresource_layer_count(image, range);
497
498 const VkImageViewMinLodCreateInfoEXT *min_lod_info =
499 vk_find_struct_const(pCreateInfo, IMAGE_VIEW_MIN_LOD_CREATE_INFO_EXT);
500 image_view->min_lod = min_lod_info ? min_lod_info->minLod : 0.0f;
501
502 /* From the Vulkan 1.3.215 spec:
503 *
504 * VUID-VkImageViewMinLodCreateInfoEXT-minLod-06456
505 *
506 * "minLod must be less or equal to the index of the last mipmap level
507 * accessible to the view."
508 */
509 assert(image_view->min_lod <= image_view->base_mip_level +
510 image_view->level_count - 1);
511
512 image_view->extent =
513 vk_image_mip_level_extent(image, image_view->base_mip_level);
514
515 /* By default storage uses the same as the image properties, but it can be
516 * overriden with VkImageViewSlicedCreateInfoEXT.
517 */
518 image_view->storage.z_slice_offset = 0;
519 image_view->storage.z_slice_count = image_view->extent.depth;
520
521 const VkImageViewSlicedCreateInfoEXT *sliced_info =
522 vk_find_struct_const(pCreateInfo, IMAGE_VIEW_SLICED_CREATE_INFO_EXT);
523 assert(image_view->base_mip_level + image_view->level_count
524 <= image->mip_levels);
525 switch (image->image_type) {
526 default:
527 unreachable("bad VkImageType");
528 case VK_IMAGE_TYPE_1D:
529 case VK_IMAGE_TYPE_2D:
530 assert(image_view->base_array_layer + image_view->layer_count
531 <= image->array_layers);
532 break;
533 case VK_IMAGE_TYPE_3D:
534 if (sliced_info && image_view->view_type == VK_IMAGE_VIEW_TYPE_3D) {
535 unsigned total = image_view->extent.depth;
536 image_view->storage.z_slice_offset = sliced_info->sliceOffset;
537 assert(image_view->storage.z_slice_offset < total);
538 if (sliced_info->sliceCount == VK_REMAINING_3D_SLICES_EXT) {
539 image_view->storage.z_slice_count = total - image_view->storage.z_slice_offset;
540 } else {
541 image_view->storage.z_slice_count = sliced_info->sliceCount;
542 }
543 } else if (image_view->view_type != VK_IMAGE_VIEW_TYPE_3D) {
544 image_view->storage.z_slice_offset = image_view->base_array_layer;
545 image_view->storage.z_slice_count = image_view->layer_count;
546 }
547 assert(image_view->storage.z_slice_offset + image_view->storage.z_slice_count
548 <= image->extent.depth);
549 assert(image_view->base_array_layer + image_view->layer_count
550 <= image_view->extent.depth);
551 break;
552 }
553
554 /* If we are creating a color view from a depth/stencil image we compute
555 * usage from the underlying depth/stencil aspects.
556 */
557 const VkImageUsageFlags image_usage =
558 vk_image_usage(image, image_view->aspects);
559 const VkImageViewUsageCreateInfo *usage_info =
560 vk_find_struct_const(pCreateInfo, IMAGE_VIEW_USAGE_CREATE_INFO);
561 image_view->usage = usage_info ? usage_info->usage : image_usage;
562 assert(driver_internal || !(image_view->usage & ~image_usage));
563 }
564
565 void
vk_image_view_finish(struct vk_image_view * image_view)566 vk_image_view_finish(struct vk_image_view *image_view)
567 {
568 vk_object_base_finish(&image_view->base);
569 }
570
571 void *
vk_image_view_create(struct vk_device * device,bool driver_internal,const VkImageViewCreateInfo * pCreateInfo,const VkAllocationCallbacks * alloc,size_t size)572 vk_image_view_create(struct vk_device *device,
573 bool driver_internal,
574 const VkImageViewCreateInfo *pCreateInfo,
575 const VkAllocationCallbacks *alloc,
576 size_t size)
577 {
578 struct vk_image_view *image_view =
579 vk_zalloc2(&device->alloc, alloc, size, 8,
580 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
581 if (image_view == NULL)
582 return NULL;
583
584 vk_image_view_init(device, image_view, driver_internal, pCreateInfo);
585
586 return image_view;
587 }
588
589 void
vk_image_view_destroy(struct vk_device * device,const VkAllocationCallbacks * alloc,struct vk_image_view * image_view)590 vk_image_view_destroy(struct vk_device *device,
591 const VkAllocationCallbacks *alloc,
592 struct vk_image_view *image_view)
593 {
594 vk_object_free(device, alloc, image_view);
595 }
596
597 bool
vk_image_layout_is_read_only(VkImageLayout layout,VkImageAspectFlagBits aspect)598 vk_image_layout_is_read_only(VkImageLayout layout,
599 VkImageAspectFlagBits aspect)
600 {
601 assert(util_bitcount(aspect) == 1);
602
603 switch (layout) {
604 case VK_IMAGE_LAYOUT_UNDEFINED:
605 case VK_IMAGE_LAYOUT_PREINITIALIZED:
606 return true; /* These are only used for layout transitions */
607
608 case VK_IMAGE_LAYOUT_GENERAL:
609 case VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL:
610 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL:
611 case VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL:
612 case VK_IMAGE_LAYOUT_SHARED_PRESENT_KHR:
613 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL:
614 case VK_IMAGE_LAYOUT_STENCIL_ATTACHMENT_OPTIMAL:
615 case VK_IMAGE_LAYOUT_ATTACHMENT_OPTIMAL:
616 case VK_IMAGE_LAYOUT_ATTACHMENT_FEEDBACK_LOOP_OPTIMAL_EXT:
617 case VK_IMAGE_LAYOUT_RENDERING_LOCAL_READ_KHR:
618 return false;
619
620 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL:
621 case VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL:
622 case VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL:
623 case VK_IMAGE_LAYOUT_PRESENT_SRC_KHR:
624 case VK_IMAGE_LAYOUT_FRAGMENT_SHADING_RATE_ATTACHMENT_OPTIMAL_KHR:
625 case VK_IMAGE_LAYOUT_FRAGMENT_DENSITY_MAP_OPTIMAL_EXT:
626 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_OPTIMAL:
627 case VK_IMAGE_LAYOUT_STENCIL_READ_ONLY_OPTIMAL:
628 case VK_IMAGE_LAYOUT_READ_ONLY_OPTIMAL:
629 return true;
630
631 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_STENCIL_ATTACHMENT_OPTIMAL:
632 return aspect == VK_IMAGE_ASPECT_DEPTH_BIT;
633
634 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_STENCIL_READ_ONLY_OPTIMAL:
635 return aspect == VK_IMAGE_ASPECT_STENCIL_BIT;
636
637 case VK_IMAGE_LAYOUT_MAX_ENUM:
638 case VK_IMAGE_LAYOUT_VIDEO_DECODE_DST_KHR:
639 case VK_IMAGE_LAYOUT_VIDEO_DECODE_SRC_KHR:
640 case VK_IMAGE_LAYOUT_VIDEO_DECODE_DPB_KHR:
641 case VK_IMAGE_LAYOUT_VIDEO_ENCODE_DST_KHR:
642 case VK_IMAGE_LAYOUT_VIDEO_ENCODE_SRC_KHR:
643 case VK_IMAGE_LAYOUT_VIDEO_ENCODE_DPB_KHR:
644 unreachable("Invalid image layout.");
645 }
646
647 unreachable("Invalid image layout.");
648 }
649
650 bool
vk_image_layout_is_depth_only(VkImageLayout layout)651 vk_image_layout_is_depth_only(VkImageLayout layout)
652 {
653 switch (layout) {
654 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_OPTIMAL:
655 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL:
656 return true;
657
658 default:
659 return false;
660 }
661 }
662
663 static VkResult
vk_image_create_get_format_list_uncompressed(struct vk_device * device,const VkImageCreateInfo * pCreateInfo,const VkAllocationCallbacks * pAllocator,VkFormat ** formats,uint32_t * format_count)664 vk_image_create_get_format_list_uncompressed(struct vk_device *device,
665 const VkImageCreateInfo *pCreateInfo,
666 const VkAllocationCallbacks *pAllocator,
667 VkFormat **formats,
668 uint32_t *format_count)
669 {
670 const struct vk_format_class_info *class =
671 vk_format_get_class_info(pCreateInfo->format);
672
673 *formats = NULL;
674 *format_count = 0;
675
676 if (class->format_count < 2)
677 return VK_SUCCESS;
678
679 *formats = vk_alloc2(&device->alloc, pAllocator,
680 sizeof(VkFormat) * class->format_count,
681 alignof(VkFormat), VK_SYSTEM_ALLOCATION_SCOPE_COMMAND);
682 if (*formats == NULL)
683 return vk_error(device, VK_ERROR_OUT_OF_HOST_MEMORY);
684
685 memcpy(*formats, class->formats, sizeof(VkFormat) * class->format_count);
686 *format_count = class->format_count;
687
688 return VK_SUCCESS;
689 }
690
691 static VkResult
vk_image_create_get_format_list_compressed(struct vk_device * device,const VkImageCreateInfo * pCreateInfo,const VkAllocationCallbacks * pAllocator,VkFormat ** formats,uint32_t * format_count)692 vk_image_create_get_format_list_compressed(struct vk_device *device,
693 const VkImageCreateInfo *pCreateInfo,
694 const VkAllocationCallbacks *pAllocator,
695 VkFormat **formats,
696 uint32_t *format_count)
697 {
698 if ((pCreateInfo->flags & VK_IMAGE_CREATE_BLOCK_TEXEL_VIEW_COMPATIBLE_BIT) == 0) {
699 return vk_image_create_get_format_list_uncompressed(device,
700 pCreateInfo,
701 pAllocator,
702 formats,
703 format_count);
704 }
705
706 const struct vk_format_class_info *class =
707 vk_format_get_class_info(pCreateInfo->format);
708 const struct vk_format_class_info *uncompr_class = NULL;
709
710 switch (vk_format_get_blocksizebits(pCreateInfo->format)) {
711 case 64:
712 uncompr_class = vk_format_class_get_info(MESA_VK_FORMAT_CLASS_64_BIT);
713 break;
714 case 128:
715 uncompr_class = vk_format_class_get_info(MESA_VK_FORMAT_CLASS_128_BIT);
716 break;
717 }
718
719 if (!uncompr_class)
720 return vk_error(device, VK_ERROR_FORMAT_NOT_SUPPORTED);
721
722 uint32_t fmt_count = class->format_count + uncompr_class->format_count;
723
724 *formats = vk_alloc2(&device->alloc, pAllocator,
725 sizeof(VkFormat) * fmt_count,
726 alignof(VkFormat), VK_SYSTEM_ALLOCATION_SCOPE_COMMAND);
727 if (*formats == NULL)
728 return vk_error(device, VK_ERROR_OUT_OF_HOST_MEMORY);
729
730 memcpy(*formats, class->formats, sizeof(VkFormat) * class->format_count);
731 memcpy(*formats + class->format_count, uncompr_class->formats,
732 sizeof(VkFormat) * uncompr_class->format_count);
733 *format_count = class->format_count + uncompr_class->format_count;
734
735 return VK_SUCCESS;
736 }
737
738 /* Get a list of compatible formats when VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT
739 * or VK_IMAGE_CREATE_BLOCK_TEXEL_VIEW_COMPATIBLE_BIT is set. This list is
740 * either retrieved from a VkImageFormatListCreateInfo passed to the creation
741 * chain, or forged from the default compatible list specified in the
742 * "formats-compatibility-classes" section of the spec.
743 *
744 * The value returned in *formats must be freed with
745 * vk_free2(&device->alloc, pAllocator), and should not live past the
746 * vkCreateImage() call (allocated in the COMMAND scope).
747 */
748 VkResult
vk_image_create_get_format_list(struct vk_device * device,const VkImageCreateInfo * pCreateInfo,const VkAllocationCallbacks * pAllocator,VkFormat ** formats,uint32_t * format_count)749 vk_image_create_get_format_list(struct vk_device *device,
750 const VkImageCreateInfo *pCreateInfo,
751 const VkAllocationCallbacks *pAllocator,
752 VkFormat **formats,
753 uint32_t *format_count)
754 {
755 *formats = NULL;
756 *format_count = 0;
757
758 if (!(pCreateInfo->flags &
759 (VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT |
760 VK_IMAGE_CREATE_BLOCK_TEXEL_VIEW_COMPATIBLE_BIT))) {
761 return VK_SUCCESS;
762 }
763
764 /* "Each depth/stencil format is only compatible with itself." */
765 if (vk_format_is_depth_or_stencil(pCreateInfo->format))
766 return VK_SUCCESS;
767
768 const VkImageFormatListCreateInfo *format_list = (const VkImageFormatListCreateInfo *)
769 vk_find_struct_const(pCreateInfo->pNext, IMAGE_FORMAT_LIST_CREATE_INFO);
770
771 if (format_list) {
772 if (!format_list->viewFormatCount)
773 return VK_SUCCESS;
774
775 *formats = vk_alloc2(&device->alloc, pAllocator,
776 sizeof(VkFormat) * format_list->viewFormatCount,
777 alignof(VkFormat), VK_SYSTEM_ALLOCATION_SCOPE_COMMAND);
778 if (*formats == NULL)
779 return vk_error(device, VK_ERROR_OUT_OF_HOST_MEMORY);
780
781 memcpy(*formats, format_list->pViewFormats, sizeof(VkFormat) * format_list->viewFormatCount);
782 *format_count = format_list->viewFormatCount;
783 return VK_SUCCESS;
784 }
785
786 if (vk_format_is_compressed(pCreateInfo->format))
787 return vk_image_create_get_format_list_compressed(device,
788 pCreateInfo,
789 pAllocator,
790 formats,
791 format_count);
792
793 return vk_image_create_get_format_list_uncompressed(device,
794 pCreateInfo,
795 pAllocator,
796 formats,
797 format_count);
798 }
799
800 /* From the Vulkan Specification 1.2.166 - VkAttachmentReference2:
801 *
802 * "If layout only specifies the layout of the depth aspect of the
803 * attachment, the layout of the stencil aspect is specified by the
804 * stencilLayout member of a VkAttachmentReferenceStencilLayout structure
805 * included in the pNext chain. Otherwise, layout describes the layout for
806 * all relevant image aspects."
807 */
808 VkImageLayout
vk_att_ref_stencil_layout(const VkAttachmentReference2 * att_ref,const VkAttachmentDescription2 * attachments)809 vk_att_ref_stencil_layout(const VkAttachmentReference2 *att_ref,
810 const VkAttachmentDescription2 *attachments)
811 {
812 /* From VUID-VkAttachmentReference2-attachment-04755:
813 * "If attachment is not VK_ATTACHMENT_UNUSED, and the format of the
814 * referenced attachment is a depth/stencil format which includes both
815 * depth and stencil aspects [...]
816 */
817 if (att_ref->attachment == VK_ATTACHMENT_UNUSED ||
818 !vk_format_has_stencil(attachments[att_ref->attachment].format))
819 return VK_IMAGE_LAYOUT_UNDEFINED;
820
821 const VkAttachmentReferenceStencilLayout *stencil_ref =
822 vk_find_struct_const(att_ref->pNext, ATTACHMENT_REFERENCE_STENCIL_LAYOUT);
823
824 if (stencil_ref)
825 return stencil_ref->stencilLayout;
826
827 /* From VUID-VkAttachmentReference2-attachment-04755:
828 * "If attachment is not VK_ATTACHMENT_UNUSED, and the format of the
829 * referenced attachment is a depth/stencil format which includes both
830 * depth and stencil aspects, and layout is
831 * VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL or
832 * VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_OPTIMAL, the pNext chain must include
833 * a VkAttachmentReferenceStencilLayout structure."
834 */
835 assert(!vk_image_layout_is_depth_only(att_ref->layout));
836
837 return att_ref->layout;
838 }
839
840 /* From the Vulkan Specification 1.2.184:
841 *
842 * "If the pNext chain includes a VkAttachmentDescriptionStencilLayout
843 * structure, then the stencilInitialLayout and stencilFinalLayout members
844 * specify the initial and final layouts of the stencil aspect of a
845 * depth/stencil format, and initialLayout and finalLayout only apply to the
846 * depth aspect. For depth-only formats, the
847 * VkAttachmentDescriptionStencilLayout structure is ignored. For
848 * stencil-only formats, the initial and final layouts of the stencil aspect
849 * are taken from the VkAttachmentDescriptionStencilLayout structure if
850 * present, or initialLayout and finalLayout if not present."
851 *
852 * "If format is a depth/stencil format, and either initialLayout or
853 * finalLayout does not specify a layout for the stencil aspect, then the
854 * application must specify the initial and final layouts of the stencil
855 * aspect by including a VkAttachmentDescriptionStencilLayout structure in
856 * the pNext chain."
857 */
858 VkImageLayout
vk_att_desc_stencil_layout(const VkAttachmentDescription2 * att_desc,bool final)859 vk_att_desc_stencil_layout(const VkAttachmentDescription2 *att_desc, bool final)
860 {
861 if (!vk_format_has_stencil(att_desc->format))
862 return VK_IMAGE_LAYOUT_UNDEFINED;
863
864 const VkAttachmentDescriptionStencilLayout *stencil_desc =
865 vk_find_struct_const(att_desc->pNext, ATTACHMENT_DESCRIPTION_STENCIL_LAYOUT);
866
867 if (stencil_desc) {
868 return final ?
869 stencil_desc->stencilFinalLayout :
870 stencil_desc->stencilInitialLayout;
871 }
872
873 const VkImageLayout main_layout =
874 final ? att_desc->finalLayout : att_desc->initialLayout;
875
876 /* From VUID-VkAttachmentDescription2-format-03302/03303:
877 * "If format is a depth/stencil format which includes both depth and
878 * stencil aspects, and initial/finalLayout is
879 * VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL or
880 * VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_OPTIMAL, the pNext chain must include
881 * a VkAttachmentDescriptionStencilLayout structure."
882 */
883 assert(!vk_image_layout_is_depth_only(main_layout));
884
885 return main_layout;
886 }
887
888 VkImageUsageFlags
vk_image_layout_to_usage_flags(VkImageLayout layout,VkImageAspectFlagBits aspect)889 vk_image_layout_to_usage_flags(VkImageLayout layout,
890 VkImageAspectFlagBits aspect)
891 {
892 assert(util_bitcount(aspect) == 1);
893
894 switch (layout) {
895 case VK_IMAGE_LAYOUT_UNDEFINED:
896 case VK_IMAGE_LAYOUT_PREINITIALIZED:
897 return 0u;
898
899 case VK_IMAGE_LAYOUT_GENERAL:
900 return ~0u;
901
902 case VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL:
903 assert(aspect & VK_IMAGE_ASPECT_ANY_COLOR_MASK_MESA);
904 return VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
905
906 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL:
907 assert(aspect & (VK_IMAGE_ASPECT_DEPTH_BIT |
908 VK_IMAGE_ASPECT_STENCIL_BIT));
909 return VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT;
910
911 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL:
912 assert(aspect & VK_IMAGE_ASPECT_DEPTH_BIT);
913 return vk_image_layout_to_usage_flags(
914 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
915
916 case VK_IMAGE_LAYOUT_STENCIL_ATTACHMENT_OPTIMAL:
917 assert(aspect & VK_IMAGE_ASPECT_STENCIL_BIT);
918 return vk_image_layout_to_usage_flags(
919 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
920
921 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL:
922 assert(aspect & (VK_IMAGE_ASPECT_DEPTH_BIT |
923 VK_IMAGE_ASPECT_STENCIL_BIT));
924 return VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT |
925 VK_IMAGE_USAGE_SAMPLED_BIT |
926 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT;
927
928 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_OPTIMAL:
929 assert(aspect & VK_IMAGE_ASPECT_DEPTH_BIT);
930 return vk_image_layout_to_usage_flags(
931 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
932
933 case VK_IMAGE_LAYOUT_STENCIL_READ_ONLY_OPTIMAL:
934 assert(aspect & VK_IMAGE_ASPECT_STENCIL_BIT);
935 return vk_image_layout_to_usage_flags(
936 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
937
938 case VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL:
939 return VK_IMAGE_USAGE_SAMPLED_BIT |
940 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT;
941
942 case VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL:
943 return VK_IMAGE_USAGE_TRANSFER_SRC_BIT;
944
945 case VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL:
946 return VK_IMAGE_USAGE_TRANSFER_DST_BIT;
947
948 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_STENCIL_ATTACHMENT_OPTIMAL:
949 if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT) {
950 return vk_image_layout_to_usage_flags(
951 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
952 } else if (aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
953 return vk_image_layout_to_usage_flags(
954 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
955 } else {
956 assert(!"Must be a depth/stencil aspect");
957 return 0;
958 }
959
960 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_STENCIL_READ_ONLY_OPTIMAL:
961 if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT) {
962 return vk_image_layout_to_usage_flags(
963 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
964 } else if (aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
965 return vk_image_layout_to_usage_flags(
966 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
967 } else {
968 assert(!"Must be a depth/stencil aspect");
969 return 0;
970 }
971
972 case VK_IMAGE_LAYOUT_PRESENT_SRC_KHR:
973 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
974 /* This needs to be handled specially by the caller */
975 return 0;
976
977 case VK_IMAGE_LAYOUT_SHARED_PRESENT_KHR:
978 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
979 return vk_image_layout_to_usage_flags(VK_IMAGE_LAYOUT_GENERAL, aspect);
980
981 case VK_IMAGE_LAYOUT_FRAGMENT_SHADING_RATE_ATTACHMENT_OPTIMAL_KHR:
982 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
983 return VK_IMAGE_USAGE_FRAGMENT_SHADING_RATE_ATTACHMENT_BIT_KHR;
984
985 case VK_IMAGE_LAYOUT_FRAGMENT_DENSITY_MAP_OPTIMAL_EXT:
986 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
987 return VK_IMAGE_USAGE_FRAGMENT_DENSITY_MAP_BIT_EXT;
988
989 case VK_IMAGE_LAYOUT_ATTACHMENT_OPTIMAL:
990 if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT ||
991 aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
992 return VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT;
993 } else {
994 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
995 return VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
996 }
997
998 case VK_IMAGE_LAYOUT_READ_ONLY_OPTIMAL:
999 return VK_IMAGE_USAGE_SAMPLED_BIT |
1000 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT;
1001
1002 case VK_IMAGE_LAYOUT_ATTACHMENT_FEEDBACK_LOOP_OPTIMAL_EXT:
1003 case VK_IMAGE_LAYOUT_RENDERING_LOCAL_READ_KHR:
1004 if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT ||
1005 aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
1006 return VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT |
1007 VK_IMAGE_USAGE_SAMPLED_BIT |
1008 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT |
1009 VK_IMAGE_USAGE_ATTACHMENT_FEEDBACK_LOOP_BIT_EXT;
1010 } else {
1011 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
1012 return VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
1013 VK_IMAGE_USAGE_SAMPLED_BIT |
1014 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT |
1015 VK_IMAGE_USAGE_ATTACHMENT_FEEDBACK_LOOP_BIT_EXT;
1016 }
1017
1018 case VK_IMAGE_LAYOUT_VIDEO_DECODE_DST_KHR:
1019 return VK_IMAGE_USAGE_VIDEO_DECODE_DST_BIT_KHR;
1020 case VK_IMAGE_LAYOUT_VIDEO_DECODE_SRC_KHR:
1021 return VK_IMAGE_USAGE_VIDEO_DECODE_SRC_BIT_KHR;
1022 case VK_IMAGE_LAYOUT_VIDEO_DECODE_DPB_KHR:
1023 return VK_IMAGE_USAGE_VIDEO_DECODE_DPB_BIT_KHR;
1024 case VK_IMAGE_LAYOUT_VIDEO_ENCODE_DST_KHR:
1025 return VK_IMAGE_USAGE_VIDEO_ENCODE_DST_BIT_KHR;
1026 case VK_IMAGE_LAYOUT_VIDEO_ENCODE_SRC_KHR:
1027 return VK_IMAGE_USAGE_VIDEO_ENCODE_SRC_BIT_KHR;
1028 case VK_IMAGE_LAYOUT_VIDEO_ENCODE_DPB_KHR:
1029 return VK_IMAGE_USAGE_VIDEO_ENCODE_DPB_BIT_KHR;
1030 case VK_IMAGE_LAYOUT_MAX_ENUM:
1031 unreachable("Invalid image layout.");
1032 }
1033
1034 unreachable("Invalid image layout.");
1035 }
1036