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 #include <vulkan/vulkan_android.h>
27
28 #ifndef _WIN32
29 #include <drm-uapi/drm_fourcc.h>
30 #endif
31
32 #include "vk_alloc.h"
33 #include "vk_common_entrypoints.h"
34 #include "vk_device.h"
35 #include "vk_format.h"
36 #include "vk_util.h"
37 #include "vulkan/wsi/wsi_common.h"
38
39 static VkExtent3D
sanitize_image_extent(const VkImageType imageType,const VkExtent3D imageExtent)40 sanitize_image_extent(const VkImageType imageType,
41 const VkExtent3D imageExtent)
42 {
43 switch (imageType) {
44 case VK_IMAGE_TYPE_1D:
45 return (VkExtent3D) { imageExtent.width, 1, 1 };
46 case VK_IMAGE_TYPE_2D:
47 return (VkExtent3D) { imageExtent.width, imageExtent.height, 1 };
48 case VK_IMAGE_TYPE_3D:
49 return imageExtent;
50 default:
51 unreachable("invalid image type");
52 }
53 }
54
55 void
vk_image_init(struct vk_device * device,struct vk_image * image,const VkImageCreateInfo * pCreateInfo)56 vk_image_init(struct vk_device *device,
57 struct vk_image *image,
58 const VkImageCreateInfo *pCreateInfo)
59 {
60 vk_object_base_init(device, &image->base, VK_OBJECT_TYPE_IMAGE);
61
62 assert(pCreateInfo->sType == VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO);
63 assert(pCreateInfo->mipLevels > 0);
64 assert(pCreateInfo->arrayLayers > 0);
65 assert(pCreateInfo->samples > 0);
66 assert(pCreateInfo->extent.width > 0);
67 assert(pCreateInfo->extent.height > 0);
68 assert(pCreateInfo->extent.depth > 0);
69
70 if (pCreateInfo->flags & VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT)
71 assert(pCreateInfo->imageType == VK_IMAGE_TYPE_2D);
72 if (pCreateInfo->flags & VK_IMAGE_CREATE_2D_ARRAY_COMPATIBLE_BIT)
73 assert(pCreateInfo->imageType == VK_IMAGE_TYPE_3D);
74
75 image->create_flags = pCreateInfo->flags;
76 image->image_type = pCreateInfo->imageType;
77 vk_image_set_format(image, pCreateInfo->format);
78 image->extent = sanitize_image_extent(pCreateInfo->imageType,
79 pCreateInfo->extent);
80 image->mip_levels = pCreateInfo->mipLevels;
81 image->array_layers = pCreateInfo->arrayLayers;
82 image->samples = pCreateInfo->samples;
83 image->tiling = pCreateInfo->tiling;
84 image->usage = pCreateInfo->usage;
85
86 if (image->aspects & VK_IMAGE_ASPECT_STENCIL_BIT) {
87 const VkImageStencilUsageCreateInfoEXT *stencil_usage_info =
88 vk_find_struct_const(pCreateInfo->pNext,
89 IMAGE_STENCIL_USAGE_CREATE_INFO_EXT);
90 image->stencil_usage =
91 stencil_usage_info ? stencil_usage_info->stencilUsage :
92 pCreateInfo->usage;
93 } else {
94 image->stencil_usage = 0;
95 }
96
97 const VkExternalMemoryImageCreateInfo *ext_mem_info =
98 vk_find_struct_const(pCreateInfo->pNext, EXTERNAL_MEMORY_IMAGE_CREATE_INFO);
99 if (ext_mem_info)
100 image->external_handle_types = ext_mem_info->handleTypes;
101 else
102 image->external_handle_types = 0;
103
104 const struct wsi_image_create_info *wsi_info =
105 vk_find_struct_const(pCreateInfo->pNext, WSI_IMAGE_CREATE_INFO_MESA);
106 image->wsi_legacy_scanout = wsi_info && wsi_info->scanout;
107
108 #ifndef _WIN32
109 image->drm_format_mod = ((1ULL << 56) - 1) /* DRM_FORMAT_MOD_INVALID */;
110 #endif
111
112 #ifdef ANDROID
113 const VkExternalFormatANDROID *ext_format =
114 vk_find_struct_const(pCreateInfo->pNext, EXTERNAL_FORMAT_ANDROID);
115 if (ext_format && ext_format->externalFormat != 0) {
116 assert(image->format == VK_FORMAT_UNDEFINED);
117 assert(image->external_handle_types &
118 VK_EXTERNAL_MEMORY_HANDLE_TYPE_ANDROID_HARDWARE_BUFFER_BIT_ANDROID);
119 image->android_external_format = ext_format->externalFormat;
120 } else {
121 image->android_external_format = 0;
122 }
123 #endif
124 }
125
126 void *
vk_image_create(struct vk_device * device,const VkImageCreateInfo * pCreateInfo,const VkAllocationCallbacks * alloc,size_t size)127 vk_image_create(struct vk_device *device,
128 const VkImageCreateInfo *pCreateInfo,
129 const VkAllocationCallbacks *alloc,
130 size_t size)
131 {
132 struct vk_image *image =
133 vk_zalloc2(&device->alloc, alloc, size, 8,
134 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
135 if (image == NULL)
136 return NULL;
137
138 vk_image_init(device, image, pCreateInfo);
139
140 return image;
141 }
142
143 void
vk_image_finish(struct vk_image * image)144 vk_image_finish(struct vk_image *image)
145 {
146 vk_object_base_finish(&image->base);
147 }
148
149 void
vk_image_destroy(struct vk_device * device,const VkAllocationCallbacks * alloc,struct vk_image * image)150 vk_image_destroy(struct vk_device *device,
151 const VkAllocationCallbacks *alloc,
152 struct vk_image *image)
153 {
154 vk_object_free(device, alloc, image);
155 }
156
157 #ifndef _WIN32
158 VKAPI_ATTR VkResult VKAPI_CALL
vk_common_GetImageDrmFormatModifierPropertiesEXT(UNUSED VkDevice device,VkImage _image,VkImageDrmFormatModifierPropertiesEXT * pProperties)159 vk_common_GetImageDrmFormatModifierPropertiesEXT(UNUSED VkDevice device,
160 VkImage _image,
161 VkImageDrmFormatModifierPropertiesEXT *pProperties)
162 {
163 VK_FROM_HANDLE(vk_image, image, _image);
164
165 assert(pProperties->sType ==
166 VK_STRUCTURE_TYPE_IMAGE_DRM_FORMAT_MODIFIER_PROPERTIES_EXT);
167
168 assert(image->tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT);
169 pProperties->drmFormatModifier = image->drm_format_mod;
170
171 return VK_SUCCESS;
172 }
173 #endif
174
175 void
vk_image_set_format(struct vk_image * image,VkFormat format)176 vk_image_set_format(struct vk_image *image, VkFormat format)
177 {
178 image->format = format;
179 image->aspects = vk_format_aspects(format);
180 }
181
182 VkImageUsageFlags
vk_image_usage(const struct vk_image * image,VkImageAspectFlags aspect_mask)183 vk_image_usage(const struct vk_image *image,
184 VkImageAspectFlags aspect_mask)
185 {
186 assert(!(aspect_mask & ~image->aspects));
187
188 /* From the Vulkan 1.2.131 spec:
189 *
190 * "If the image was has a depth-stencil format and was created with
191 * a VkImageStencilUsageCreateInfo structure included in the pNext
192 * chain of VkImageCreateInfo, the usage is calculated based on the
193 * subresource.aspectMask provided:
194 *
195 * - If aspectMask includes only VK_IMAGE_ASPECT_STENCIL_BIT, the
196 * implicit usage is equal to
197 * VkImageStencilUsageCreateInfo::stencilUsage.
198 *
199 * - If aspectMask includes only VK_IMAGE_ASPECT_DEPTH_BIT, the
200 * implicit usage is equal to VkImageCreateInfo::usage.
201 *
202 * - If both aspects are included in aspectMask, the implicit usage
203 * is equal to the intersection of VkImageCreateInfo::usage and
204 * VkImageStencilUsageCreateInfo::stencilUsage.
205 */
206 if (aspect_mask == VK_IMAGE_ASPECT_STENCIL_BIT) {
207 return image->stencil_usage;
208 } else if (aspect_mask == (VK_IMAGE_ASPECT_DEPTH_BIT |
209 VK_IMAGE_ASPECT_STENCIL_BIT)) {
210 return image->usage & image->stencil_usage;
211 } else {
212 /* This also handles the color case */
213 return image->usage;
214 }
215 }
216
217 #define VK_IMAGE_ASPECT_ANY_COLOR_MASK_MESA ( \
218 VK_IMAGE_ASPECT_COLOR_BIT | \
219 VK_IMAGE_ASPECT_PLANE_0_BIT | \
220 VK_IMAGE_ASPECT_PLANE_1_BIT | \
221 VK_IMAGE_ASPECT_PLANE_2_BIT)
222
223 /** Expands the given aspect mask relative to the image
224 *
225 * If the image has color plane aspects VK_IMAGE_ASPECT_COLOR_BIT has been
226 * requested, this returns the aspects of the underlying image.
227 *
228 * For example,
229 *
230 * VK_IMAGE_ASPECT_COLOR_BIT
231 *
232 * will be converted to
233 *
234 * VK_IMAGE_ASPECT_PLANE_0_BIT |
235 * VK_IMAGE_ASPECT_PLANE_1_BIT |
236 * VK_IMAGE_ASPECT_PLANE_2_BIT
237 *
238 * for an image of format VK_FORMAT_G8_B8_R8_3PLANE_420_UNORM.
239 */
240 VkImageAspectFlags
vk_image_expand_aspect_mask(const struct vk_image * image,VkImageAspectFlags aspect_mask)241 vk_image_expand_aspect_mask(const struct vk_image *image,
242 VkImageAspectFlags aspect_mask)
243 {
244 if (aspect_mask == VK_IMAGE_ASPECT_COLOR_BIT) {
245 assert(image->aspects & VK_IMAGE_ASPECT_ANY_COLOR_MASK_MESA);
246 return image->aspects;
247 } else {
248 assert(aspect_mask && !(aspect_mask & ~image->aspects));
249 return aspect_mask;
250 }
251 }
252
253 static VkComponentSwizzle
remap_swizzle(VkComponentSwizzle swizzle,VkComponentSwizzle component)254 remap_swizzle(VkComponentSwizzle swizzle, VkComponentSwizzle component)
255 {
256 return swizzle == VK_COMPONENT_SWIZZLE_IDENTITY ? component : swizzle;
257 }
258
259 void
vk_image_view_init(struct vk_device * device,struct vk_image_view * image_view,const VkImageViewCreateInfo * pCreateInfo)260 vk_image_view_init(struct vk_device *device,
261 struct vk_image_view *image_view,
262 const VkImageViewCreateInfo *pCreateInfo)
263 {
264 vk_object_base_init(device, &image_view->base, VK_OBJECT_TYPE_IMAGE_VIEW);
265
266 assert(pCreateInfo->sType == VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO);
267 VK_FROM_HANDLE(vk_image, image, pCreateInfo->image);
268
269 image_view->create_flags = pCreateInfo->flags;
270 image_view->image = image;
271 image_view->view_type = pCreateInfo->viewType;
272
273 switch (image_view->view_type) {
274 case VK_IMAGE_VIEW_TYPE_1D:
275 case VK_IMAGE_VIEW_TYPE_1D_ARRAY:
276 assert(image->image_type == VK_IMAGE_TYPE_1D);
277 break;
278 case VK_IMAGE_VIEW_TYPE_2D:
279 case VK_IMAGE_VIEW_TYPE_2D_ARRAY:
280 if (image->create_flags & VK_IMAGE_CREATE_2D_ARRAY_COMPATIBLE_BIT)
281 assert(image->image_type == VK_IMAGE_TYPE_3D);
282 else
283 assert(image->image_type == VK_IMAGE_TYPE_2D);
284 break;
285 case VK_IMAGE_VIEW_TYPE_3D:
286 assert(image->image_type == VK_IMAGE_TYPE_3D);
287 break;
288 case VK_IMAGE_VIEW_TYPE_CUBE:
289 case VK_IMAGE_VIEW_TYPE_CUBE_ARRAY:
290 assert(image->image_type == VK_IMAGE_TYPE_2D);
291 assert(image->create_flags & VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT);
292 break;
293 default:
294 unreachable("Invalid image view type");
295 }
296
297 const VkImageSubresourceRange *range = &pCreateInfo->subresourceRange;
298
299 /* Some drivers may want to create color views of depth/stencil images
300 * to implement certain operations, which is not strictly allowed by the
301 * Vulkan spec, so handle this case separately.
302 */
303 bool is_color_view_of_depth_stencil =
304 vk_format_is_depth_or_stencil(image->format) &&
305 vk_format_is_color(pCreateInfo->format);
306 if (is_color_view_of_depth_stencil) {
307 assert(range->aspectMask == VK_IMAGE_ASPECT_COLOR_BIT);
308 assert(util_format_get_blocksize(vk_format_to_pipe_format(image->format)) ==
309 util_format_get_blocksize(vk_format_to_pipe_format(pCreateInfo->format)));
310 image_view->aspects = range->aspectMask;
311 } else {
312 image_view->aspects =
313 vk_image_expand_aspect_mask(image, range->aspectMask);
314
315 /* From the Vulkan 1.2.184 spec:
316 *
317 * "If the image has a multi-planar format and
318 * subresourceRange.aspectMask is VK_IMAGE_ASPECT_COLOR_BIT, and image
319 * has been created with a usage value not containing any of the
320 * VK_IMAGE_USAGE_VIDEO_DECODE_DST_BIT_KHR,
321 * VK_IMAGE_USAGE_VIDEO_DECODE_SRC_BIT_KHR,
322 * VK_IMAGE_USAGE_VIDEO_DECODE_DPB_BIT_KHR,
323 * VK_IMAGE_USAGE_VIDEO_ENCODE_DST_BIT_KHR,
324 * VK_IMAGE_USAGE_VIDEO_ENCODE_SRC_BIT_KHR, and
325 * VK_IMAGE_USAGE_VIDEO_ENCODE_DPB_BIT_KHR flags, then the format must
326 * be identical to the image format, and the sampler to be used with the
327 * image view must enable sampler Y′CBCR conversion."
328 *
329 * Since no one implements video yet, we can ignore the bits about video
330 * create flags and assume YCbCr formats match.
331 */
332 if ((image->aspects & VK_IMAGE_ASPECT_PLANE_1_BIT) &&
333 (range->aspectMask == VK_IMAGE_ASPECT_COLOR_BIT))
334 assert(pCreateInfo->format == image->format);
335
336 /* From the Vulkan 1.2.184 spec:
337 *
338 * "Each depth/stencil format is only compatible with itself."
339 */
340 if (image_view->aspects & (VK_IMAGE_ASPECT_DEPTH_BIT |
341 VK_IMAGE_ASPECT_STENCIL_BIT))
342 assert(pCreateInfo->format == image->format);
343
344 if (!(image->create_flags & VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT))
345 assert(pCreateInfo->format == image->format);
346 }
347
348 /* Restrict the format to only the planes chosen.
349 *
350 * For combined depth and stencil images, this means the depth-only or
351 * stencil-only format if only one aspect is chosen and the full combined
352 * format if both aspects are chosen.
353 *
354 * For single-plane color images, we just take the format as-is. For
355 * multi-plane views of multi-plane images, this means we want the full
356 * multi-plane format. For single-plane views of multi-plane images, we
357 * want a format compatible with the one plane. Fortunately, this is
358 * already what the client gives us. The Vulkan 1.2.184 spec says:
359 *
360 * "If image was created with the VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT and
361 * the image has a multi-planar format, and if
362 * subresourceRange.aspectMask is VK_IMAGE_ASPECT_PLANE_0_BIT,
363 * VK_IMAGE_ASPECT_PLANE_1_BIT, or VK_IMAGE_ASPECT_PLANE_2_BIT, format
364 * must be compatible with the corresponding plane of the image, and the
365 * sampler to be used with the image view must not enable sampler Y′CBCR
366 * conversion."
367 */
368 if (image_view->aspects == VK_IMAGE_ASPECT_STENCIL_BIT) {
369 image_view->format = vk_format_stencil_only(pCreateInfo->format);
370 } else if (image_view->aspects == VK_IMAGE_ASPECT_DEPTH_BIT) {
371 image_view->format = vk_format_depth_only(pCreateInfo->format);
372 } else {
373 image_view->format = pCreateInfo->format;
374 }
375
376 image_view->swizzle = (VkComponentMapping) {
377 .r = remap_swizzle(pCreateInfo->components.r, VK_COMPONENT_SWIZZLE_R),
378 .g = remap_swizzle(pCreateInfo->components.g, VK_COMPONENT_SWIZZLE_G),
379 .b = remap_swizzle(pCreateInfo->components.b, VK_COMPONENT_SWIZZLE_B),
380 .a = remap_swizzle(pCreateInfo->components.a, VK_COMPONENT_SWIZZLE_A),
381 };
382
383 assert(range->layerCount > 0);
384 assert(range->baseMipLevel < image->mip_levels);
385
386 image_view->base_mip_level = range->baseMipLevel;
387 image_view->level_count = vk_image_subresource_level_count(image, range);
388 image_view->base_array_layer = range->baseArrayLayer;
389 image_view->layer_count = vk_image_subresource_layer_count(image, range);
390
391 image_view->extent =
392 vk_image_mip_level_extent(image, image_view->base_mip_level);
393
394 assert(image_view->base_mip_level + image_view->level_count
395 <= image->mip_levels);
396 switch (image->image_type) {
397 default:
398 unreachable("bad VkImageType");
399 case VK_IMAGE_TYPE_1D:
400 case VK_IMAGE_TYPE_2D:
401 assert(image_view->base_array_layer + image_view->layer_count
402 <= image->array_layers);
403 break;
404 case VK_IMAGE_TYPE_3D:
405 assert(image_view->base_array_layer + image_view->layer_count
406 <= image_view->extent.depth);
407 break;
408 }
409
410 /* If we are creating a color view from a depth/stencil image we compute
411 * usage from the underlying depth/stencil aspects.
412 */
413 const VkImageUsageFlags image_usage = is_color_view_of_depth_stencil ?
414 vk_image_usage(image, image->aspects) :
415 vk_image_usage(image, image_view->aspects);
416 const VkImageViewUsageCreateInfo *usage_info =
417 vk_find_struct_const(pCreateInfo, IMAGE_VIEW_USAGE_CREATE_INFO);
418 image_view->usage = usage_info ? usage_info->usage : image_usage;
419 assert(!(image_view->usage & ~image_usage));
420 }
421
422 void
vk_image_view_finish(struct vk_image_view * image_view)423 vk_image_view_finish(struct vk_image_view *image_view)
424 {
425 vk_object_base_finish(&image_view->base);
426 }
427
428 void *
vk_image_view_create(struct vk_device * device,const VkImageViewCreateInfo * pCreateInfo,const VkAllocationCallbacks * alloc,size_t size)429 vk_image_view_create(struct vk_device *device,
430 const VkImageViewCreateInfo *pCreateInfo,
431 const VkAllocationCallbacks *alloc,
432 size_t size)
433 {
434 struct vk_image_view *image_view =
435 vk_zalloc2(&device->alloc, alloc, size, 8,
436 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
437 if (image_view == NULL)
438 return NULL;
439
440 vk_image_view_init(device, image_view, pCreateInfo);
441
442 return image_view;
443 }
444
445 void
vk_image_view_destroy(struct vk_device * device,const VkAllocationCallbacks * alloc,struct vk_image_view * image_view)446 vk_image_view_destroy(struct vk_device *device,
447 const VkAllocationCallbacks *alloc,
448 struct vk_image_view *image_view)
449 {
450 vk_object_free(device, alloc, image_view);
451 }
452
453 bool
vk_image_layout_is_read_only(VkImageLayout layout,VkImageAspectFlagBits aspect)454 vk_image_layout_is_read_only(VkImageLayout layout,
455 VkImageAspectFlagBits aspect)
456 {
457 assert(util_bitcount(aspect) == 1);
458
459 switch (layout) {
460 case VK_IMAGE_LAYOUT_UNDEFINED:
461 case VK_IMAGE_LAYOUT_PREINITIALIZED:
462 return true; /* These are only used for layout transitions */
463
464 case VK_IMAGE_LAYOUT_GENERAL:
465 case VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL:
466 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL:
467 case VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL:
468 case VK_IMAGE_LAYOUT_SHARED_PRESENT_KHR:
469 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL:
470 case VK_IMAGE_LAYOUT_STENCIL_ATTACHMENT_OPTIMAL:
471 case VK_IMAGE_LAYOUT_ATTACHMENT_OPTIMAL_KHR:
472 return false;
473
474 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL:
475 case VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL:
476 case VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL:
477 case VK_IMAGE_LAYOUT_PRESENT_SRC_KHR:
478 case VK_IMAGE_LAYOUT_SHADING_RATE_OPTIMAL_NV:
479 case VK_IMAGE_LAYOUT_FRAGMENT_DENSITY_MAP_OPTIMAL_EXT:
480 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_OPTIMAL:
481 case VK_IMAGE_LAYOUT_STENCIL_READ_ONLY_OPTIMAL:
482 case VK_IMAGE_LAYOUT_READ_ONLY_OPTIMAL_KHR:
483 return true;
484
485 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_STENCIL_ATTACHMENT_OPTIMAL:
486 return aspect == VK_IMAGE_ASPECT_DEPTH_BIT;
487
488 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_STENCIL_READ_ONLY_OPTIMAL:
489 return aspect == VK_IMAGE_ASPECT_STENCIL_BIT;
490
491 case VK_IMAGE_LAYOUT_MAX_ENUM:
492 unreachable("Invalid image layout.");
493 }
494
495 unreachable("Invalid image layout.");
496 }
497
498 VkImageUsageFlags
vk_image_layout_to_usage_flags(VkImageLayout layout,VkImageAspectFlagBits aspect)499 vk_image_layout_to_usage_flags(VkImageLayout layout,
500 VkImageAspectFlagBits aspect)
501 {
502 assert(util_bitcount(aspect) == 1);
503
504 switch (layout) {
505 case VK_IMAGE_LAYOUT_UNDEFINED:
506 case VK_IMAGE_LAYOUT_PREINITIALIZED:
507 return 0u;
508
509 case VK_IMAGE_LAYOUT_GENERAL:
510 return ~0u;
511
512 case VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL:
513 assert(aspect & VK_IMAGE_ASPECT_ANY_COLOR_MASK_MESA);
514 return VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
515
516 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL:
517 assert(aspect & (VK_IMAGE_ASPECT_DEPTH_BIT |
518 VK_IMAGE_ASPECT_STENCIL_BIT));
519 return VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT;
520
521 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL:
522 assert(aspect & VK_IMAGE_ASPECT_DEPTH_BIT);
523 return vk_image_layout_to_usage_flags(
524 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
525
526 case VK_IMAGE_LAYOUT_STENCIL_ATTACHMENT_OPTIMAL:
527 assert(aspect & VK_IMAGE_ASPECT_STENCIL_BIT);
528 return vk_image_layout_to_usage_flags(
529 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
530
531 case VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL:
532 assert(aspect & (VK_IMAGE_ASPECT_DEPTH_BIT |
533 VK_IMAGE_ASPECT_STENCIL_BIT));
534 return VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT |
535 VK_IMAGE_USAGE_SAMPLED_BIT |
536 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT;
537
538 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_OPTIMAL:
539 assert(aspect & VK_IMAGE_ASPECT_DEPTH_BIT);
540 return vk_image_layout_to_usage_flags(
541 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
542
543 case VK_IMAGE_LAYOUT_STENCIL_READ_ONLY_OPTIMAL:
544 assert(aspect & VK_IMAGE_ASPECT_STENCIL_BIT);
545 return vk_image_layout_to_usage_flags(
546 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
547
548 case VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL:
549 return VK_IMAGE_USAGE_SAMPLED_BIT |
550 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT;
551
552 case VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL:
553 return VK_IMAGE_USAGE_TRANSFER_SRC_BIT;
554
555 case VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL:
556 return VK_IMAGE_USAGE_TRANSFER_DST_BIT;
557
558 case VK_IMAGE_LAYOUT_DEPTH_READ_ONLY_STENCIL_ATTACHMENT_OPTIMAL:
559 if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT) {
560 return vk_image_layout_to_usage_flags(
561 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
562 } else if (aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
563 return vk_image_layout_to_usage_flags(
564 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
565 } else {
566 assert(!"Must be a depth/stencil aspect");
567 return 0;
568 }
569
570 case VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_STENCIL_READ_ONLY_OPTIMAL:
571 if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT) {
572 return vk_image_layout_to_usage_flags(
573 VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, aspect);
574 } else if (aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
575 return vk_image_layout_to_usage_flags(
576 VK_IMAGE_LAYOUT_DEPTH_STENCIL_READ_ONLY_OPTIMAL, aspect);
577 } else {
578 assert(!"Must be a depth/stencil aspect");
579 return 0;
580 }
581
582 case VK_IMAGE_LAYOUT_PRESENT_SRC_KHR:
583 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
584 /* This needs to be handled specially by the caller */
585 return 0;
586
587 case VK_IMAGE_LAYOUT_SHARED_PRESENT_KHR:
588 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
589 return vk_image_layout_to_usage_flags(VK_IMAGE_LAYOUT_GENERAL, aspect);
590
591 case VK_IMAGE_LAYOUT_SHADING_RATE_OPTIMAL_NV:
592 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
593 return VK_IMAGE_USAGE_SHADING_RATE_IMAGE_BIT_NV;
594
595 case VK_IMAGE_LAYOUT_FRAGMENT_DENSITY_MAP_OPTIMAL_EXT:
596 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
597 return VK_IMAGE_USAGE_FRAGMENT_DENSITY_MAP_BIT_EXT;
598
599 case VK_IMAGE_LAYOUT_ATTACHMENT_OPTIMAL_KHR:
600 if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT ||
601 aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
602 return VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT;
603 } else {
604 assert(aspect == VK_IMAGE_ASPECT_COLOR_BIT);
605 return VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
606 }
607
608 case VK_IMAGE_LAYOUT_READ_ONLY_OPTIMAL_KHR:
609 return VK_IMAGE_USAGE_SAMPLED_BIT |
610 VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT;
611
612 case VK_IMAGE_LAYOUT_MAX_ENUM:
613 unreachable("Invalid image layout.");
614 }
615
616 unreachable("Invalid image layout.");
617 }
618