1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker * Copyright © 2015 Intel Corporation
3*61046927SAndroid Build Coastguard Worker *
4*61046927SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
5*61046927SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
6*61046927SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
7*61046927SAndroid Build Coastguard Worker * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*61046927SAndroid Build Coastguard Worker * and/or sell copies of the Software, and to permit persons to whom the
9*61046927SAndroid Build Coastguard Worker * Software is furnished to do so, subject to the following conditions:
10*61046927SAndroid Build Coastguard Worker *
11*61046927SAndroid Build Coastguard Worker * The above copyright notice and this permission notice (including the next
12*61046927SAndroid Build Coastguard Worker * paragraph) shall be included in all copies or substantial portions of the
13*61046927SAndroid Build Coastguard Worker * Software.
14*61046927SAndroid Build Coastguard Worker *
15*61046927SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*61046927SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*61046927SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18*61046927SAndroid Build Coastguard Worker * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*61046927SAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*61046927SAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21*61046927SAndroid Build Coastguard Worker * IN THE SOFTWARE.
22*61046927SAndroid Build Coastguard Worker */
23*61046927SAndroid Build Coastguard Worker
24*61046927SAndroid Build Coastguard Worker #include <assert.h>
25*61046927SAndroid Build Coastguard Worker #include <stdbool.h>
26*61046927SAndroid Build Coastguard Worker #include <string.h>
27*61046927SAndroid Build Coastguard Worker #include <unistd.h>
28*61046927SAndroid Build Coastguard Worker #include <fcntl.h>
29*61046927SAndroid Build Coastguard Worker #include <sys/mman.h>
30*61046927SAndroid Build Coastguard Worker #include "drm-uapi/drm_fourcc.h"
31*61046927SAndroid Build Coastguard Worker
32*61046927SAndroid Build Coastguard Worker #include "anv_private.h"
33*61046927SAndroid Build Coastguard Worker #include "common/intel_aux_map.h"
34*61046927SAndroid Build Coastguard Worker #include "util/u_debug.h"
35*61046927SAndroid Build Coastguard Worker #include "vk_util.h"
36*61046927SAndroid Build Coastguard Worker #include "util/u_math.h"
37*61046927SAndroid Build Coastguard Worker
38*61046927SAndroid Build Coastguard Worker #include "vk_format.h"
39*61046927SAndroid Build Coastguard Worker
40*61046927SAndroid Build Coastguard Worker #define ANV_OFFSET_IMPLICIT UINT64_MAX
41*61046927SAndroid Build Coastguard Worker
42*61046927SAndroid Build Coastguard Worker static const enum isl_surf_dim
43*61046927SAndroid Build Coastguard Worker vk_to_isl_surf_dim[] = {
44*61046927SAndroid Build Coastguard Worker [VK_IMAGE_TYPE_1D] = ISL_SURF_DIM_1D,
45*61046927SAndroid Build Coastguard Worker [VK_IMAGE_TYPE_2D] = ISL_SURF_DIM_2D,
46*61046927SAndroid Build Coastguard Worker [VK_IMAGE_TYPE_3D] = ISL_SURF_DIM_3D,
47*61046927SAndroid Build Coastguard Worker };
48*61046927SAndroid Build Coastguard Worker
49*61046927SAndroid Build Coastguard Worker static uint64_t MUST_CHECK UNUSED
memory_range_end(struct anv_image_memory_range memory_range)50*61046927SAndroid Build Coastguard Worker memory_range_end(struct anv_image_memory_range memory_range)
51*61046927SAndroid Build Coastguard Worker {
52*61046927SAndroid Build Coastguard Worker assert(anv_is_aligned(memory_range.offset, memory_range.alignment));
53*61046927SAndroid Build Coastguard Worker return memory_range.offset + memory_range.size;
54*61046927SAndroid Build Coastguard Worker }
55*61046927SAndroid Build Coastguard Worker
56*61046927SAndroid Build Coastguard Worker /**
57*61046927SAndroid Build Coastguard Worker * Get binding for VkImagePlaneMemoryRequirementsInfo,
58*61046927SAndroid Build Coastguard Worker * VkBindImagePlaneMemoryInfo and VkDeviceImageMemoryRequirements.
59*61046927SAndroid Build Coastguard Worker */
60*61046927SAndroid Build Coastguard Worker struct anv_image_binding *
anv_image_aspect_to_binding(struct anv_image * image,VkImageAspectFlags aspect)61*61046927SAndroid Build Coastguard Worker anv_image_aspect_to_binding(struct anv_image *image,
62*61046927SAndroid Build Coastguard Worker VkImageAspectFlags aspect)
63*61046927SAndroid Build Coastguard Worker {
64*61046927SAndroid Build Coastguard Worker uint32_t plane = 0;
65*61046927SAndroid Build Coastguard Worker
66*61046927SAndroid Build Coastguard Worker assert(image->disjoint);
67*61046927SAndroid Build Coastguard Worker
68*61046927SAndroid Build Coastguard Worker if (image->vk.tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT) {
69*61046927SAndroid Build Coastguard Worker /* Spec requires special aspects for modifier images. */
70*61046927SAndroid Build Coastguard Worker assert(aspect == VK_IMAGE_ASPECT_MEMORY_PLANE_0_BIT_EXT ||
71*61046927SAndroid Build Coastguard Worker aspect == VK_IMAGE_ASPECT_MEMORY_PLANE_1_BIT_EXT ||
72*61046927SAndroid Build Coastguard Worker aspect == VK_IMAGE_ASPECT_MEMORY_PLANE_2_BIT_EXT ||
73*61046927SAndroid Build Coastguard Worker aspect == VK_IMAGE_ASPECT_MEMORY_PLANE_3_BIT_EXT);
74*61046927SAndroid Build Coastguard Worker
75*61046927SAndroid Build Coastguard Worker /* We don't advertise DISJOINT for modifiers with aux, and therefore we
76*61046927SAndroid Build Coastguard Worker * don't handle queries of the modifier's "aux plane" here.
77*61046927SAndroid Build Coastguard Worker */
78*61046927SAndroid Build Coastguard Worker assert(!isl_drm_modifier_has_aux(image->vk.drm_format_mod));
79*61046927SAndroid Build Coastguard Worker
80*61046927SAndroid Build Coastguard Worker switch(aspect) {
81*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_MEMORY_PLANE_0_BIT_EXT: plane = 0; break;
82*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_MEMORY_PLANE_1_BIT_EXT: plane = 1; break;
83*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_MEMORY_PLANE_2_BIT_EXT: plane = 2; break;
84*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_MEMORY_PLANE_3_BIT_EXT: plane = 3; break;
85*61046927SAndroid Build Coastguard Worker }
86*61046927SAndroid Build Coastguard Worker } else {
87*61046927SAndroid Build Coastguard Worker plane = anv_image_aspect_to_plane(image, aspect);
88*61046927SAndroid Build Coastguard Worker }
89*61046927SAndroid Build Coastguard Worker
90*61046927SAndroid Build Coastguard Worker return &image->bindings[ANV_IMAGE_MEMORY_BINDING_PLANE_0 + plane];
91*61046927SAndroid Build Coastguard Worker }
92*61046927SAndroid Build Coastguard Worker
93*61046927SAndroid Build Coastguard Worker /**
94*61046927SAndroid Build Coastguard Worker * Extend the memory binding's range by appending a new memory range with `size`
95*61046927SAndroid Build Coastguard Worker * and `alignment` at `offset`. Return the appended range.
96*61046927SAndroid Build Coastguard Worker *
97*61046927SAndroid Build Coastguard Worker * Offset is ignored if ANV_OFFSET_IMPLICIT.
98*61046927SAndroid Build Coastguard Worker *
99*61046927SAndroid Build Coastguard Worker * The given binding must not be ANV_IMAGE_MEMORY_BINDING_MAIN. The function
100*61046927SAndroid Build Coastguard Worker * converts to MAIN as needed.
101*61046927SAndroid Build Coastguard Worker */
102*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
image_binding_grow(const struct anv_device * device,struct anv_image * image,enum anv_image_memory_binding binding,uint64_t offset,uint64_t size,uint32_t alignment,struct anv_image_memory_range * out_range)103*61046927SAndroid Build Coastguard Worker image_binding_grow(const struct anv_device *device,
104*61046927SAndroid Build Coastguard Worker struct anv_image *image,
105*61046927SAndroid Build Coastguard Worker enum anv_image_memory_binding binding,
106*61046927SAndroid Build Coastguard Worker uint64_t offset,
107*61046927SAndroid Build Coastguard Worker uint64_t size,
108*61046927SAndroid Build Coastguard Worker uint32_t alignment,
109*61046927SAndroid Build Coastguard Worker struct anv_image_memory_range *out_range)
110*61046927SAndroid Build Coastguard Worker {
111*61046927SAndroid Build Coastguard Worker /* We overwrite 'offset' but need to remember if it was implicit. */
112*61046927SAndroid Build Coastguard Worker const bool has_implicit_offset = (offset == ANV_OFFSET_IMPLICIT);
113*61046927SAndroid Build Coastguard Worker
114*61046927SAndroid Build Coastguard Worker assert(size > 0);
115*61046927SAndroid Build Coastguard Worker assert(util_is_power_of_two_or_zero(alignment));
116*61046927SAndroid Build Coastguard Worker
117*61046927SAndroid Build Coastguard Worker switch (binding) {
118*61046927SAndroid Build Coastguard Worker case ANV_IMAGE_MEMORY_BINDING_MAIN:
119*61046927SAndroid Build Coastguard Worker /* The caller must not pre-translate BINDING_PLANE_i to BINDING_MAIN. */
120*61046927SAndroid Build Coastguard Worker unreachable("ANV_IMAGE_MEMORY_BINDING_MAIN");
121*61046927SAndroid Build Coastguard Worker case ANV_IMAGE_MEMORY_BINDING_PLANE_0:
122*61046927SAndroid Build Coastguard Worker case ANV_IMAGE_MEMORY_BINDING_PLANE_1:
123*61046927SAndroid Build Coastguard Worker case ANV_IMAGE_MEMORY_BINDING_PLANE_2:
124*61046927SAndroid Build Coastguard Worker if (!image->disjoint)
125*61046927SAndroid Build Coastguard Worker binding = ANV_IMAGE_MEMORY_BINDING_MAIN;
126*61046927SAndroid Build Coastguard Worker break;
127*61046927SAndroid Build Coastguard Worker case ANV_IMAGE_MEMORY_BINDING_PRIVATE:
128*61046927SAndroid Build Coastguard Worker assert(offset == ANV_OFFSET_IMPLICIT);
129*61046927SAndroid Build Coastguard Worker break;
130*61046927SAndroid Build Coastguard Worker case ANV_IMAGE_MEMORY_BINDING_END:
131*61046927SAndroid Build Coastguard Worker unreachable("ANV_IMAGE_MEMORY_BINDING_END");
132*61046927SAndroid Build Coastguard Worker }
133*61046927SAndroid Build Coastguard Worker
134*61046927SAndroid Build Coastguard Worker struct anv_image_memory_range *container =
135*61046927SAndroid Build Coastguard Worker &image->bindings[binding].memory_range;
136*61046927SAndroid Build Coastguard Worker
137*61046927SAndroid Build Coastguard Worker if (has_implicit_offset) {
138*61046927SAndroid Build Coastguard Worker offset = align64(container->offset + container->size, alignment);
139*61046927SAndroid Build Coastguard Worker } else {
140*61046927SAndroid Build Coastguard Worker /* Offset must be validated because it comes from
141*61046927SAndroid Build Coastguard Worker * VkImageDrmFormatModifierExplicitCreateInfoEXT.
142*61046927SAndroid Build Coastguard Worker */
143*61046927SAndroid Build Coastguard Worker if (unlikely(!anv_is_aligned(offset, alignment))) {
144*61046927SAndroid Build Coastguard Worker return vk_errorf(device,
145*61046927SAndroid Build Coastguard Worker VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT,
146*61046927SAndroid Build Coastguard Worker "VkImageDrmFormatModifierExplicitCreateInfoEXT::"
147*61046927SAndroid Build Coastguard Worker "pPlaneLayouts[]::offset is misaligned");
148*61046927SAndroid Build Coastguard Worker }
149*61046927SAndroid Build Coastguard Worker }
150*61046927SAndroid Build Coastguard Worker
151*61046927SAndroid Build Coastguard Worker /* Surfaces can be added out of memory-order. Track the end of each memory
152*61046927SAndroid Build Coastguard Worker * plane to update the binding size properly.
153*61046927SAndroid Build Coastguard Worker */
154*61046927SAndroid Build Coastguard Worker uint64_t memory_range_end;
155*61046927SAndroid Build Coastguard Worker if (__builtin_add_overflow(offset, size, &memory_range_end)) {
156*61046927SAndroid Build Coastguard Worker if (has_implicit_offset) {
157*61046927SAndroid Build Coastguard Worker assert(!"overflow");
158*61046927SAndroid Build Coastguard Worker return vk_errorf(device, VK_ERROR_UNKNOWN,
159*61046927SAndroid Build Coastguard Worker "internal error: overflow in %s", __func__);
160*61046927SAndroid Build Coastguard Worker } else {
161*61046927SAndroid Build Coastguard Worker return vk_errorf(device,
162*61046927SAndroid Build Coastguard Worker VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT,
163*61046927SAndroid Build Coastguard Worker "VkImageDrmFormatModifierExplicitCreateInfoEXT::"
164*61046927SAndroid Build Coastguard Worker "pPlaneLayouts[]::offset is too large");
165*61046927SAndroid Build Coastguard Worker }
166*61046927SAndroid Build Coastguard Worker }
167*61046927SAndroid Build Coastguard Worker
168*61046927SAndroid Build Coastguard Worker container->size = MAX2(container->size, memory_range_end);
169*61046927SAndroid Build Coastguard Worker container->alignment = MAX2(container->alignment, alignment);
170*61046927SAndroid Build Coastguard Worker
171*61046927SAndroid Build Coastguard Worker *out_range = (struct anv_image_memory_range) {
172*61046927SAndroid Build Coastguard Worker .binding = binding,
173*61046927SAndroid Build Coastguard Worker .alignment = alignment,
174*61046927SAndroid Build Coastguard Worker .size = size,
175*61046927SAndroid Build Coastguard Worker .offset = offset,
176*61046927SAndroid Build Coastguard Worker };
177*61046927SAndroid Build Coastguard Worker
178*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
179*61046927SAndroid Build Coastguard Worker }
180*61046927SAndroid Build Coastguard Worker
181*61046927SAndroid Build Coastguard Worker /**
182*61046927SAndroid Build Coastguard Worker * Adjust range 'a' to contain range 'b'.
183*61046927SAndroid Build Coastguard Worker *
184*61046927SAndroid Build Coastguard Worker * For simplicity's sake, the offset of 'a' must be 0 and remains 0.
185*61046927SAndroid Build Coastguard Worker * If 'a' and 'b' target different bindings, then no merge occurs.
186*61046927SAndroid Build Coastguard Worker */
187*61046927SAndroid Build Coastguard Worker static void
memory_range_merge(struct anv_image_memory_range * a,const struct anv_image_memory_range b)188*61046927SAndroid Build Coastguard Worker memory_range_merge(struct anv_image_memory_range *a,
189*61046927SAndroid Build Coastguard Worker const struct anv_image_memory_range b)
190*61046927SAndroid Build Coastguard Worker {
191*61046927SAndroid Build Coastguard Worker if (b.size == 0)
192*61046927SAndroid Build Coastguard Worker return;
193*61046927SAndroid Build Coastguard Worker
194*61046927SAndroid Build Coastguard Worker if (a->binding != b.binding)
195*61046927SAndroid Build Coastguard Worker return;
196*61046927SAndroid Build Coastguard Worker
197*61046927SAndroid Build Coastguard Worker assert(a->offset == 0);
198*61046927SAndroid Build Coastguard Worker assert(anv_is_aligned(a->offset, a->alignment));
199*61046927SAndroid Build Coastguard Worker assert(anv_is_aligned(b.offset, b.alignment));
200*61046927SAndroid Build Coastguard Worker
201*61046927SAndroid Build Coastguard Worker a->alignment = MAX2(a->alignment, b.alignment);
202*61046927SAndroid Build Coastguard Worker a->size = MAX2(a->size, b.offset + b.size);
203*61046927SAndroid Build Coastguard Worker }
204*61046927SAndroid Build Coastguard Worker
205*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t
anv_image_choose_isl_surf_usage(struct anv_physical_device * device,VkImageCreateFlags vk_create_flags,VkImageUsageFlags vk_usage,isl_surf_usage_flags_t isl_extra_usage,VkImageAspectFlagBits aspect,VkImageCompressionFlagsEXT comp_flags)206*61046927SAndroid Build Coastguard Worker anv_image_choose_isl_surf_usage(struct anv_physical_device *device,
207*61046927SAndroid Build Coastguard Worker VkImageCreateFlags vk_create_flags,
208*61046927SAndroid Build Coastguard Worker VkImageUsageFlags vk_usage,
209*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_extra_usage,
210*61046927SAndroid Build Coastguard Worker VkImageAspectFlagBits aspect,
211*61046927SAndroid Build Coastguard Worker VkImageCompressionFlagsEXT comp_flags)
212*61046927SAndroid Build Coastguard Worker {
213*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_usage = isl_extra_usage;
214*61046927SAndroid Build Coastguard Worker
215*61046927SAndroid Build Coastguard Worker /* On platform like MTL, we choose to allocate additional CCS memory at the
216*61046927SAndroid Build Coastguard Worker * back of the VkDeviceMemory objects since different images can share the
217*61046927SAndroid Build Coastguard Worker * AUX-TT PTE because the HW doesn't care about the image format in the
218*61046927SAndroid Build Coastguard Worker * PTE. That means we can always ignore the AUX-TT alignment requirement
219*61046927SAndroid Build Coastguard Worker * from an ISL point of view.
220*61046927SAndroid Build Coastguard Worker */
221*61046927SAndroid Build Coastguard Worker if (device->alloc_aux_tt_mem)
222*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_NO_AUX_TT_ALIGNMENT_BIT;
223*61046927SAndroid Build Coastguard Worker
224*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_SAMPLED_BIT)
225*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_TEXTURE_BIT;
226*61046927SAndroid Build Coastguard Worker
227*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT)
228*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_TEXTURE_BIT;
229*61046927SAndroid Build Coastguard Worker
230*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_STORAGE_BIT)
231*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_STORAGE_BIT;
232*61046927SAndroid Build Coastguard Worker
233*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT)
234*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_RENDER_TARGET_BIT;
235*61046927SAndroid Build Coastguard Worker
236*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_FRAGMENT_SHADING_RATE_ATTACHMENT_BIT_KHR)
237*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_CPB_BIT;
238*61046927SAndroid Build Coastguard Worker
239*61046927SAndroid Build Coastguard Worker /* TODO: consider whether compression with sparse is workable. */
240*61046927SAndroid Build Coastguard Worker if (vk_create_flags & VK_IMAGE_CREATE_SPARSE_BINDING_BIT)
241*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_SPARSE_BIT |
242*61046927SAndroid Build Coastguard Worker ISL_SURF_USAGE_DISABLE_AUX_BIT;
243*61046927SAndroid Build Coastguard Worker
244*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_VIDEO_DECODE_DST_BIT_KHR ||
245*61046927SAndroid Build Coastguard Worker vk_usage & VK_IMAGE_USAGE_VIDEO_DECODE_DPB_BIT_KHR ||
246*61046927SAndroid Build Coastguard Worker vk_usage & VK_IMAGE_USAGE_VIDEO_ENCODE_DPB_BIT_KHR ||
247*61046927SAndroid Build Coastguard Worker vk_usage & VK_IMAGE_USAGE_VIDEO_ENCODE_SRC_BIT_KHR)
248*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_VIDEO_DECODE_BIT;
249*61046927SAndroid Build Coastguard Worker
250*61046927SAndroid Build Coastguard Worker if (vk_create_flags & VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT)
251*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_CUBE_BIT;
252*61046927SAndroid Build Coastguard Worker
253*61046927SAndroid Build Coastguard Worker if (vk_create_flags & (VK_IMAGE_CREATE_2D_VIEW_COMPATIBLE_BIT_EXT |
254*61046927SAndroid Build Coastguard Worker VK_IMAGE_CREATE_2D_ARRAY_COMPATIBLE_BIT))
255*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_2D_3D_COMPATIBLE_BIT;
256*61046927SAndroid Build Coastguard Worker
257*61046927SAndroid Build Coastguard Worker if (vk_create_flags & VK_IMAGE_CREATE_PROTECTED_BIT)
258*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_PROTECTED_BIT;
259*61046927SAndroid Build Coastguard Worker
260*61046927SAndroid Build Coastguard Worker /* Even if we're only using it for transfer operations, clears to depth and
261*61046927SAndroid Build Coastguard Worker * stencil images happen as depth and stencil so they need the right ISL
262*61046927SAndroid Build Coastguard Worker * usage bits or else things will fall apart.
263*61046927SAndroid Build Coastguard Worker */
264*61046927SAndroid Build Coastguard Worker switch (aspect) {
265*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_DEPTH_BIT:
266*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_DEPTH_BIT;
267*61046927SAndroid Build Coastguard Worker break;
268*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_STENCIL_BIT:
269*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_STENCIL_BIT;
270*61046927SAndroid Build Coastguard Worker break;
271*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_COLOR_BIT:
272*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_PLANE_0_BIT:
273*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_PLANE_1_BIT:
274*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_PLANE_2_BIT:
275*61046927SAndroid Build Coastguard Worker break;
276*61046927SAndroid Build Coastguard Worker default:
277*61046927SAndroid Build Coastguard Worker unreachable("bad VkImageAspect");
278*61046927SAndroid Build Coastguard Worker }
279*61046927SAndroid Build Coastguard Worker
280*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_TRANSFER_SRC_BIT) {
281*61046927SAndroid Build Coastguard Worker /* blorp implements transfers by sampling from the source image. */
282*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_TEXTURE_BIT;
283*61046927SAndroid Build Coastguard Worker }
284*61046927SAndroid Build Coastguard Worker
285*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_TRANSFER_DST_BIT &&
286*61046927SAndroid Build Coastguard Worker aspect == VK_IMAGE_ASPECT_COLOR_BIT) {
287*61046927SAndroid Build Coastguard Worker /* blorp implements transfers by rendering into the destination image.
288*61046927SAndroid Build Coastguard Worker * Only request this with color images, as we deal with depth/stencil
289*61046927SAndroid Build Coastguard Worker * formats differently. */
290*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_RENDER_TARGET_BIT;
291*61046927SAndroid Build Coastguard Worker }
292*61046927SAndroid Build Coastguard Worker
293*61046927SAndroid Build Coastguard Worker if (comp_flags & VK_IMAGE_COMPRESSION_DISABLED_EXT)
294*61046927SAndroid Build Coastguard Worker isl_usage |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
295*61046927SAndroid Build Coastguard Worker
296*61046927SAndroid Build Coastguard Worker return isl_usage;
297*61046927SAndroid Build Coastguard Worker }
298*61046927SAndroid Build Coastguard Worker
299*61046927SAndroid Build Coastguard Worker static isl_tiling_flags_t
choose_isl_tiling_flags(const struct intel_device_info * devinfo,const struct anv_image_create_info * anv_info,const struct isl_drm_modifier_info * isl_mod_info,bool legacy_scanout)300*61046927SAndroid Build Coastguard Worker choose_isl_tiling_flags(const struct intel_device_info *devinfo,
301*61046927SAndroid Build Coastguard Worker const struct anv_image_create_info *anv_info,
302*61046927SAndroid Build Coastguard Worker const struct isl_drm_modifier_info *isl_mod_info,
303*61046927SAndroid Build Coastguard Worker bool legacy_scanout)
304*61046927SAndroid Build Coastguard Worker {
305*61046927SAndroid Build Coastguard Worker const VkImageCreateInfo *base_info = anv_info->vk_info;
306*61046927SAndroid Build Coastguard Worker isl_tiling_flags_t flags = 0;
307*61046927SAndroid Build Coastguard Worker
308*61046927SAndroid Build Coastguard Worker assert((isl_mod_info != NULL) ==
309*61046927SAndroid Build Coastguard Worker (base_info->tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT));
310*61046927SAndroid Build Coastguard Worker
311*61046927SAndroid Build Coastguard Worker switch (base_info->tiling) {
312*61046927SAndroid Build Coastguard Worker default:
313*61046927SAndroid Build Coastguard Worker unreachable("bad VkImageTiling");
314*61046927SAndroid Build Coastguard Worker case VK_IMAGE_TILING_OPTIMAL:
315*61046927SAndroid Build Coastguard Worker flags = ISL_TILING_ANY_MASK;
316*61046927SAndroid Build Coastguard Worker break;
317*61046927SAndroid Build Coastguard Worker case VK_IMAGE_TILING_LINEAR:
318*61046927SAndroid Build Coastguard Worker flags = ISL_TILING_LINEAR_BIT;
319*61046927SAndroid Build Coastguard Worker break;
320*61046927SAndroid Build Coastguard Worker case VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT:
321*61046927SAndroid Build Coastguard Worker flags = 1 << isl_mod_info->tiling;
322*61046927SAndroid Build Coastguard Worker }
323*61046927SAndroid Build Coastguard Worker
324*61046927SAndroid Build Coastguard Worker if (anv_info->isl_tiling_flags) {
325*61046927SAndroid Build Coastguard Worker assert(isl_mod_info == NULL);
326*61046927SAndroid Build Coastguard Worker flags &= anv_info->isl_tiling_flags;
327*61046927SAndroid Build Coastguard Worker }
328*61046927SAndroid Build Coastguard Worker
329*61046927SAndroid Build Coastguard Worker if (legacy_scanout) {
330*61046927SAndroid Build Coastguard Worker isl_tiling_flags_t legacy_mask = ISL_TILING_LINEAR_BIT;
331*61046927SAndroid Build Coastguard Worker if (devinfo->has_tiling_uapi)
332*61046927SAndroid Build Coastguard Worker legacy_mask |= ISL_TILING_X_BIT;
333*61046927SAndroid Build Coastguard Worker flags &= legacy_mask;
334*61046927SAndroid Build Coastguard Worker }
335*61046927SAndroid Build Coastguard Worker
336*61046927SAndroid Build Coastguard Worker assert(flags);
337*61046927SAndroid Build Coastguard Worker
338*61046927SAndroid Build Coastguard Worker return flags;
339*61046927SAndroid Build Coastguard Worker }
340*61046927SAndroid Build Coastguard Worker
341*61046927SAndroid Build Coastguard Worker /**
342*61046927SAndroid Build Coastguard Worker * Add the surface to the binding at the given offset.
343*61046927SAndroid Build Coastguard Worker *
344*61046927SAndroid Build Coastguard Worker * \see image_binding_grow()
345*61046927SAndroid Build Coastguard Worker */
346*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
add_surface(struct anv_device * device,struct anv_image * image,struct anv_surface * surf,enum anv_image_memory_binding binding,uint64_t offset)347*61046927SAndroid Build Coastguard Worker add_surface(struct anv_device *device,
348*61046927SAndroid Build Coastguard Worker struct anv_image *image,
349*61046927SAndroid Build Coastguard Worker struct anv_surface *surf,
350*61046927SAndroid Build Coastguard Worker enum anv_image_memory_binding binding,
351*61046927SAndroid Build Coastguard Worker uint64_t offset)
352*61046927SAndroid Build Coastguard Worker {
353*61046927SAndroid Build Coastguard Worker /* isl surface must be initialized */
354*61046927SAndroid Build Coastguard Worker assert(surf->isl.size_B > 0);
355*61046927SAndroid Build Coastguard Worker
356*61046927SAndroid Build Coastguard Worker return image_binding_grow(device, image, binding, offset,
357*61046927SAndroid Build Coastguard Worker surf->isl.size_B,
358*61046927SAndroid Build Coastguard Worker surf->isl.alignment_B,
359*61046927SAndroid Build Coastguard Worker &surf->memory_range);
360*61046927SAndroid Build Coastguard Worker }
361*61046927SAndroid Build Coastguard Worker
362*61046927SAndroid Build Coastguard Worker static bool
can_fast_clear_with_non_zero_color(const struct intel_device_info * devinfo,const struct anv_image * image,uint32_t plane,const VkImageFormatListCreateInfo * fmt_list)363*61046927SAndroid Build Coastguard Worker can_fast_clear_with_non_zero_color(const struct intel_device_info *devinfo,
364*61046927SAndroid Build Coastguard Worker const struct anv_image *image,
365*61046927SAndroid Build Coastguard Worker uint32_t plane,
366*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *fmt_list)
367*61046927SAndroid Build Coastguard Worker {
368*61046927SAndroid Build Coastguard Worker /* If we don't have an AUX surface where fast clears apply, we can return
369*61046927SAndroid Build Coastguard Worker * early.
370*61046927SAndroid Build Coastguard Worker */
371*61046927SAndroid Build Coastguard Worker if (!isl_aux_usage_has_fast_clears(image->planes[plane].aux_usage))
372*61046927SAndroid Build Coastguard Worker return false;
373*61046927SAndroid Build Coastguard Worker
374*61046927SAndroid Build Coastguard Worker /* On TGL (< C0), if a block of fragment shader outputs match the surface's
375*61046927SAndroid Build Coastguard Worker * clear color, the HW may convert them to fast-clears (see HSD 1607794140).
376*61046927SAndroid Build Coastguard Worker * This can lead to rendering corruptions if not handled properly. We
377*61046927SAndroid Build Coastguard Worker * restrict the clear color to zero to avoid issues that can occur with:
378*61046927SAndroid Build Coastguard Worker * - Texture view rendering (including blorp_copy calls)
379*61046927SAndroid Build Coastguard Worker * - Images with multiple levels or array layers
380*61046927SAndroid Build Coastguard Worker */
381*61046927SAndroid Build Coastguard Worker if (image->planes[plane].aux_usage == ISL_AUX_USAGE_FCV_CCS_E)
382*61046927SAndroid Build Coastguard Worker return false;
383*61046927SAndroid Build Coastguard Worker
384*61046927SAndroid Build Coastguard Worker /* Turning on non zero fast clears for CCS_E introduces a performance
385*61046927SAndroid Build Coastguard Worker * regression for games such as F1 22 and RDR2 by introducing additional
386*61046927SAndroid Build Coastguard Worker * partial resolves. Let's turn non zero fast clears back off till we can
387*61046927SAndroid Build Coastguard Worker * fix performance.
388*61046927SAndroid Build Coastguard Worker */
389*61046927SAndroid Build Coastguard Worker if (image->planes[plane].aux_usage == ISL_AUX_USAGE_CCS_E &&
390*61046927SAndroid Build Coastguard Worker devinfo->ver >= 12)
391*61046927SAndroid Build Coastguard Worker return false;
392*61046927SAndroid Build Coastguard Worker
393*61046927SAndroid Build Coastguard Worker /* Non mutable image, we can fast clear with any color supported by HW.
394*61046927SAndroid Build Coastguard Worker */
395*61046927SAndroid Build Coastguard Worker if (!(image->vk.create_flags & VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT))
396*61046927SAndroid Build Coastguard Worker return true;
397*61046927SAndroid Build Coastguard Worker
398*61046927SAndroid Build Coastguard Worker /* Mutable image with no format list, we have to assume all formats */
399*61046927SAndroid Build Coastguard Worker if (!fmt_list || fmt_list->viewFormatCount == 0)
400*61046927SAndroid Build Coastguard Worker return false;
401*61046927SAndroid Build Coastguard Worker
402*61046927SAndroid Build Coastguard Worker enum isl_format img_format = image->planes[plane].primary_surface.isl.format;
403*61046927SAndroid Build Coastguard Worker
404*61046927SAndroid Build Coastguard Worker /* Check bit compatibility for clear color components */
405*61046927SAndroid Build Coastguard Worker for (uint32_t i = 0; i < fmt_list->viewFormatCount; i++) {
406*61046927SAndroid Build Coastguard Worker if (fmt_list->pViewFormats[i] == VK_FORMAT_UNDEFINED)
407*61046927SAndroid Build Coastguard Worker continue;
408*61046927SAndroid Build Coastguard Worker
409*61046927SAndroid Build Coastguard Worker struct anv_format_plane view_format_plane =
410*61046927SAndroid Build Coastguard Worker anv_get_format_plane(devinfo, fmt_list->pViewFormats[i],
411*61046927SAndroid Build Coastguard Worker plane, image->vk.tiling);
412*61046927SAndroid Build Coastguard Worker
413*61046927SAndroid Build Coastguard Worker enum isl_format view_format = view_format_plane.isl_format;
414*61046927SAndroid Build Coastguard Worker
415*61046927SAndroid Build Coastguard Worker if (!isl_formats_have_same_bits_per_channel(img_format, view_format))
416*61046927SAndroid Build Coastguard Worker return false;
417*61046927SAndroid Build Coastguard Worker }
418*61046927SAndroid Build Coastguard Worker
419*61046927SAndroid Build Coastguard Worker return true;
420*61046927SAndroid Build Coastguard Worker }
421*61046927SAndroid Build Coastguard Worker
422*61046927SAndroid Build Coastguard Worker /**
423*61046927SAndroid Build Coastguard Worker * Return true if the storage image could be used with atomics.
424*61046927SAndroid Build Coastguard Worker *
425*61046927SAndroid Build Coastguard Worker * If the image was created with an explicit format, we check it for typed
426*61046927SAndroid Build Coastguard Worker * atomic support. If MUTABLE_FORMAT_BIT is set, then we check the optional
427*61046927SAndroid Build Coastguard Worker * format list, seeing if /any/ of the formats support typed atomics. If no
428*61046927SAndroid Build Coastguard Worker * list is supplied, we fall back to using the bpb, as the application could
429*61046927SAndroid Build Coastguard Worker * make an image view with a format that does use atomics.
430*61046927SAndroid Build Coastguard Worker */
431*61046927SAndroid Build Coastguard Worker static bool
storage_image_format_supports_atomic(const struct intel_device_info * devinfo,VkImageCreateFlags create_flags,enum isl_format format,VkImageTiling vk_tiling,const VkImageFormatListCreateInfo * fmt_list)432*61046927SAndroid Build Coastguard Worker storage_image_format_supports_atomic(const struct intel_device_info *devinfo,
433*61046927SAndroid Build Coastguard Worker VkImageCreateFlags create_flags,
434*61046927SAndroid Build Coastguard Worker enum isl_format format,
435*61046927SAndroid Build Coastguard Worker VkImageTiling vk_tiling,
436*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *fmt_list)
437*61046927SAndroid Build Coastguard Worker {
438*61046927SAndroid Build Coastguard Worker if (isl_format_supports_typed_atomics(devinfo, format))
439*61046927SAndroid Build Coastguard Worker return true;
440*61046927SAndroid Build Coastguard Worker
441*61046927SAndroid Build Coastguard Worker if (!(create_flags & VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT))
442*61046927SAndroid Build Coastguard Worker return false;
443*61046927SAndroid Build Coastguard Worker
444*61046927SAndroid Build Coastguard Worker if (fmt_list) {
445*61046927SAndroid Build Coastguard Worker for (uint32_t i = 0; i < fmt_list->viewFormatCount; i++) {
446*61046927SAndroid Build Coastguard Worker if (fmt_list->pViewFormats[i] == VK_FORMAT_UNDEFINED)
447*61046927SAndroid Build Coastguard Worker continue;
448*61046927SAndroid Build Coastguard Worker
449*61046927SAndroid Build Coastguard Worker enum isl_format view_format =
450*61046927SAndroid Build Coastguard Worker anv_get_isl_format(devinfo, fmt_list->pViewFormats[i],
451*61046927SAndroid Build Coastguard Worker VK_IMAGE_ASPECT_COLOR_BIT, vk_tiling);
452*61046927SAndroid Build Coastguard Worker
453*61046927SAndroid Build Coastguard Worker if (isl_format_supports_typed_atomics(devinfo, view_format))
454*61046927SAndroid Build Coastguard Worker return true;
455*61046927SAndroid Build Coastguard Worker }
456*61046927SAndroid Build Coastguard Worker
457*61046927SAndroid Build Coastguard Worker return false;
458*61046927SAndroid Build Coastguard Worker }
459*61046927SAndroid Build Coastguard Worker
460*61046927SAndroid Build Coastguard Worker /* No explicit format list. Any 16/32/64bpp format could be used with atomics. */
461*61046927SAndroid Build Coastguard Worker unsigned bpb = isl_format_get_layout(format)->bpb;
462*61046927SAndroid Build Coastguard Worker return bpb == 16 || bpb == 32 || bpb == 64;
463*61046927SAndroid Build Coastguard Worker }
464*61046927SAndroid Build Coastguard Worker
465*61046927SAndroid Build Coastguard Worker static bool
formats_ccs_e_compatible(const struct intel_device_info * devinfo,VkImageCreateFlags create_flags,enum isl_format format,VkImageTiling vk_tiling,const VkImageFormatListCreateInfo * fmt_list)466*61046927SAndroid Build Coastguard Worker formats_ccs_e_compatible(const struct intel_device_info *devinfo,
467*61046927SAndroid Build Coastguard Worker VkImageCreateFlags create_flags,
468*61046927SAndroid Build Coastguard Worker enum isl_format format, VkImageTiling vk_tiling,
469*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *fmt_list)
470*61046927SAndroid Build Coastguard Worker {
471*61046927SAndroid Build Coastguard Worker if (!anv_format_supports_ccs_e(devinfo, format))
472*61046927SAndroid Build Coastguard Worker return false;
473*61046927SAndroid Build Coastguard Worker
474*61046927SAndroid Build Coastguard Worker /* For images created without MUTABLE_FORMAT_BIT set, we know that they will
475*61046927SAndroid Build Coastguard Worker * always be used with the original format. In particular, they will always
476*61046927SAndroid Build Coastguard Worker * be used with a format that supports color compression.
477*61046927SAndroid Build Coastguard Worker */
478*61046927SAndroid Build Coastguard Worker if (!(create_flags & VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT))
479*61046927SAndroid Build Coastguard Worker return true;
480*61046927SAndroid Build Coastguard Worker
481*61046927SAndroid Build Coastguard Worker if (!fmt_list || fmt_list->viewFormatCount == 0)
482*61046927SAndroid Build Coastguard Worker return false;
483*61046927SAndroid Build Coastguard Worker
484*61046927SAndroid Build Coastguard Worker for (uint32_t i = 0; i < fmt_list->viewFormatCount; i++) {
485*61046927SAndroid Build Coastguard Worker if (fmt_list->pViewFormats[i] == VK_FORMAT_UNDEFINED)
486*61046927SAndroid Build Coastguard Worker continue;
487*61046927SAndroid Build Coastguard Worker
488*61046927SAndroid Build Coastguard Worker enum isl_format view_format =
489*61046927SAndroid Build Coastguard Worker anv_get_isl_format(devinfo, fmt_list->pViewFormats[i],
490*61046927SAndroid Build Coastguard Worker VK_IMAGE_ASPECT_COLOR_BIT, vk_tiling);
491*61046927SAndroid Build Coastguard Worker
492*61046927SAndroid Build Coastguard Worker if (!isl_formats_are_ccs_e_compatible(devinfo, format, view_format))
493*61046927SAndroid Build Coastguard Worker return false;
494*61046927SAndroid Build Coastguard Worker }
495*61046927SAndroid Build Coastguard Worker
496*61046927SAndroid Build Coastguard Worker return true;
497*61046927SAndroid Build Coastguard Worker }
498*61046927SAndroid Build Coastguard Worker
499*61046927SAndroid Build Coastguard Worker bool
anv_format_supports_ccs_e(const struct intel_device_info * devinfo,const enum isl_format format)500*61046927SAndroid Build Coastguard Worker anv_format_supports_ccs_e(const struct intel_device_info *devinfo,
501*61046927SAndroid Build Coastguard Worker const enum isl_format format)
502*61046927SAndroid Build Coastguard Worker {
503*61046927SAndroid Build Coastguard Worker /* CCS_E for YCRCB_NORMAL and YCRCB_SWAP_UV is not currently supported by
504*61046927SAndroid Build Coastguard Worker * ANV so leave it disabled for now.
505*61046927SAndroid Build Coastguard Worker */
506*61046927SAndroid Build Coastguard Worker if (isl_format_is_yuv(format))
507*61046927SAndroid Build Coastguard Worker return false;
508*61046927SAndroid Build Coastguard Worker
509*61046927SAndroid Build Coastguard Worker return isl_format_supports_ccs_e(devinfo, format);
510*61046927SAndroid Build Coastguard Worker }
511*61046927SAndroid Build Coastguard Worker
512*61046927SAndroid Build Coastguard Worker bool
anv_formats_ccs_e_compatible(const struct intel_device_info * devinfo,VkImageCreateFlags create_flags,VkFormat vk_format,VkImageTiling vk_tiling,VkImageUsageFlags vk_usage,const VkImageFormatListCreateInfo * fmt_list)513*61046927SAndroid Build Coastguard Worker anv_formats_ccs_e_compatible(const struct intel_device_info *devinfo,
514*61046927SAndroid Build Coastguard Worker VkImageCreateFlags create_flags,
515*61046927SAndroid Build Coastguard Worker VkFormat vk_format, VkImageTiling vk_tiling,
516*61046927SAndroid Build Coastguard Worker VkImageUsageFlags vk_usage,
517*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *fmt_list)
518*61046927SAndroid Build Coastguard Worker {
519*61046927SAndroid Build Coastguard Worker u_foreach_bit(b, vk_format_aspects(vk_format)) {
520*61046927SAndroid Build Coastguard Worker VkImageAspectFlagBits aspect = 1 << b;
521*61046927SAndroid Build Coastguard Worker enum isl_format format =
522*61046927SAndroid Build Coastguard Worker anv_get_isl_format(devinfo, vk_format, aspect, vk_tiling);
523*61046927SAndroid Build Coastguard Worker
524*61046927SAndroid Build Coastguard Worker if (!formats_ccs_e_compatible(devinfo, create_flags, format, vk_tiling,
525*61046927SAndroid Build Coastguard Worker fmt_list))
526*61046927SAndroid Build Coastguard Worker return false;
527*61046927SAndroid Build Coastguard Worker
528*61046927SAndroid Build Coastguard Worker if (vk_usage & VK_IMAGE_USAGE_STORAGE_BIT) {
529*61046927SAndroid Build Coastguard Worker if (devinfo->verx10 < 125)
530*61046927SAndroid Build Coastguard Worker return false;
531*61046927SAndroid Build Coastguard Worker
532*61046927SAndroid Build Coastguard Worker /* Disable compression when surface can be potentially used for
533*61046927SAndroid Build Coastguard Worker * atomic operation.
534*61046927SAndroid Build Coastguard Worker */
535*61046927SAndroid Build Coastguard Worker if (storage_image_format_supports_atomic(devinfo, create_flags,
536*61046927SAndroid Build Coastguard Worker format, vk_tiling,
537*61046927SAndroid Build Coastguard Worker fmt_list))
538*61046927SAndroid Build Coastguard Worker return false;
539*61046927SAndroid Build Coastguard Worker }
540*61046927SAndroid Build Coastguard Worker }
541*61046927SAndroid Build Coastguard Worker
542*61046927SAndroid Build Coastguard Worker return true;
543*61046927SAndroid Build Coastguard Worker }
544*61046927SAndroid Build Coastguard Worker
545*61046927SAndroid Build Coastguard Worker /**
546*61046927SAndroid Build Coastguard Worker * For color images that have an auxiliary surface, request allocation for an
547*61046927SAndroid Build Coastguard Worker * additional buffer that mainly stores fast-clear values. Use of this buffer
548*61046927SAndroid Build Coastguard Worker * allows us to access the image's subresources while being aware of their
549*61046927SAndroid Build Coastguard Worker * fast-clear values in non-trivial cases (e.g., outside of a render pass in
550*61046927SAndroid Build Coastguard Worker * which a fast clear has occurred).
551*61046927SAndroid Build Coastguard Worker *
552*61046927SAndroid Build Coastguard Worker * In order to avoid having multiple clear colors for a single plane of an
553*61046927SAndroid Build Coastguard Worker * image (hence a single RENDER_SURFACE_STATE), we only allow fast-clears on
554*61046927SAndroid Build Coastguard Worker * the first slice (level 0, layer 0). At the time of our testing (Jan 17,
555*61046927SAndroid Build Coastguard Worker * 2018), there were no known applications which would benefit from fast-
556*61046927SAndroid Build Coastguard Worker * clearing more than just the first slice.
557*61046927SAndroid Build Coastguard Worker *
558*61046927SAndroid Build Coastguard Worker * The fast clear portion of the image is laid out in the following order:
559*61046927SAndroid Build Coastguard Worker *
560*61046927SAndroid Build Coastguard Worker * * 1 or 4 dwords (depending on hardware generation) for the clear color
561*61046927SAndroid Build Coastguard Worker * * 1 dword for the anv_fast_clear_type of the clear color
562*61046927SAndroid Build Coastguard Worker * * On gfx9+, 1 dword per level and layer of the image (3D levels count
563*61046927SAndroid Build Coastguard Worker * multiple layers) in level-major order for compression state.
564*61046927SAndroid Build Coastguard Worker *
565*61046927SAndroid Build Coastguard Worker * For the purpose of discoverability, the algorithm used to manage
566*61046927SAndroid Build Coastguard Worker * compression and fast-clears is described here:
567*61046927SAndroid Build Coastguard Worker *
568*61046927SAndroid Build Coastguard Worker * * On a transition from UNDEFINED or PREINITIALIZED to a defined layout,
569*61046927SAndroid Build Coastguard Worker * all of the values in the fast clear portion of the image are initialized
570*61046927SAndroid Build Coastguard Worker * to default values.
571*61046927SAndroid Build Coastguard Worker *
572*61046927SAndroid Build Coastguard Worker * * On fast-clear, the clear value is written into surface state and also
573*61046927SAndroid Build Coastguard Worker * into the buffer and the fast clear type is set appropriately. Both
574*61046927SAndroid Build Coastguard Worker * setting the fast-clear value in the buffer and setting the fast-clear
575*61046927SAndroid Build Coastguard Worker * type happen from the GPU using MI commands.
576*61046927SAndroid Build Coastguard Worker *
577*61046927SAndroid Build Coastguard Worker * * Whenever a render or blorp operation is performed with CCS_E, we call
578*61046927SAndroid Build Coastguard Worker * genX(cmd_buffer_mark_image_written) to set the compression state to
579*61046927SAndroid Build Coastguard Worker * true (which is represented by UINT32_MAX).
580*61046927SAndroid Build Coastguard Worker *
581*61046927SAndroid Build Coastguard Worker * * On pipeline barrier transitions, the worst-case transition is computed
582*61046927SAndroid Build Coastguard Worker * from the image layouts. The command streamer inspects the fast clear
583*61046927SAndroid Build Coastguard Worker * type and compression state dwords and constructs a predicate. The
584*61046927SAndroid Build Coastguard Worker * worst-case resolve is performed with the given predicate and the fast
585*61046927SAndroid Build Coastguard Worker * clear and compression state is set accordingly.
586*61046927SAndroid Build Coastguard Worker *
587*61046927SAndroid Build Coastguard Worker * See anv_layout_to_aux_usage and anv_layout_to_fast_clear_type functions for
588*61046927SAndroid Build Coastguard Worker * details on exactly what is allowed in what layouts.
589*61046927SAndroid Build Coastguard Worker *
590*61046927SAndroid Build Coastguard Worker * On gfx7-9, we do not have a concept of indirect clear colors in hardware.
591*61046927SAndroid Build Coastguard Worker * In order to deal with this, we have to do some clear color management.
592*61046927SAndroid Build Coastguard Worker *
593*61046927SAndroid Build Coastguard Worker * * For LOAD_OP_LOAD at the top of a renderpass, we have to copy the clear
594*61046927SAndroid Build Coastguard Worker * value from the buffer into the surface state with MI commands.
595*61046927SAndroid Build Coastguard Worker *
596*61046927SAndroid Build Coastguard Worker * * For any blorp operations, we pass the address to the clear value into
597*61046927SAndroid Build Coastguard Worker * blorp and it knows to copy the clear color.
598*61046927SAndroid Build Coastguard Worker */
599*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
add_aux_state_tracking_buffer(struct anv_device * device,struct anv_image * image,uint64_t state_offset,uint32_t plane)600*61046927SAndroid Build Coastguard Worker add_aux_state_tracking_buffer(struct anv_device *device,
601*61046927SAndroid Build Coastguard Worker struct anv_image *image,
602*61046927SAndroid Build Coastguard Worker uint64_t state_offset,
603*61046927SAndroid Build Coastguard Worker uint32_t plane)
604*61046927SAndroid Build Coastguard Worker {
605*61046927SAndroid Build Coastguard Worker assert(image && device);
606*61046927SAndroid Build Coastguard Worker /* Xe2+ platforms don't use aux tracking buffers. We shouldn't get here. */
607*61046927SAndroid Build Coastguard Worker assert(device->info->ver < 20);
608*61046927SAndroid Build Coastguard Worker assert(image->planes[plane].aux_usage != ISL_AUX_USAGE_NONE &&
609*61046927SAndroid Build Coastguard Worker image->vk.aspects & (VK_IMAGE_ASPECT_ANY_COLOR_BIT_ANV |
610*61046927SAndroid Build Coastguard Worker VK_IMAGE_ASPECT_DEPTH_BIT));
611*61046927SAndroid Build Coastguard Worker
612*61046927SAndroid Build Coastguard Worker unsigned clear_color_state_size;
613*61046927SAndroid Build Coastguard Worker if (device->info->ver >= 11) {
614*61046927SAndroid Build Coastguard Worker /* When importing an image from another source with a drm modifier that
615*61046927SAndroid Build Coastguard Worker * supports clear color, the clear color values are in a 32-byte struct
616*61046927SAndroid Build Coastguard Worker * defined in drm_fourcc.h. The fast clear type and compression state
617*61046927SAndroid Build Coastguard Worker * are not defined in these drm_fourcc.h, so there won't be memory
618*61046927SAndroid Build Coastguard Worker * allocated for these extra meta data by the source.
619*61046927SAndroid Build Coastguard Worker *
620*61046927SAndroid Build Coastguard Worker * We use the last 2 dwords of the clear color struct's memory to store
621*61046927SAndroid Build Coastguard Worker * the fast clear type and the first compression state, so the driver
622*61046927SAndroid Build Coastguard Worker * doesn't assume the extra size or need another allocation later.
623*61046927SAndroid Build Coastguard Worker *
624*61046927SAndroid Build Coastguard Worker * So far, the 2 stolen dwords are either not used in the clear color
625*61046927SAndroid Build Coastguard Worker * struct or for features not enabled. There should be no side effect to
626*61046927SAndroid Build Coastguard Worker * the hardware and destinations of images exported by this driver.
627*61046927SAndroid Build Coastguard Worker *
628*61046927SAndroid Build Coastguard Worker * Images with multiple levels or layers are not supported by drm
629*61046927SAndroid Build Coastguard Worker * modifiers, so we don't have to apply the above approach or face a
630*61046927SAndroid Build Coastguard Worker * bigger shortage from multiple compression states. We just apply the
631*61046927SAndroid Build Coastguard Worker * approach to all cases to keep the design unified.
632*61046927SAndroid Build Coastguard Worker *
633*61046927SAndroid Build Coastguard Worker * As a result, the state starts 8 bytes lower than where it should be.
634*61046927SAndroid Build Coastguard Worker */
635*61046927SAndroid Build Coastguard Worker assert(device->isl_dev.ss.clear_color_state_size >= 32);
636*61046927SAndroid Build Coastguard Worker clear_color_state_size = device->isl_dev.ss.clear_color_state_size - 8;
637*61046927SAndroid Build Coastguard Worker } else {
638*61046927SAndroid Build Coastguard Worker clear_color_state_size = device->isl_dev.ss.clear_value_size;
639*61046927SAndroid Build Coastguard Worker }
640*61046927SAndroid Build Coastguard Worker
641*61046927SAndroid Build Coastguard Worker /* Clear color and fast clear type */
642*61046927SAndroid Build Coastguard Worker unsigned state_size = clear_color_state_size + 4;
643*61046927SAndroid Build Coastguard Worker
644*61046927SAndroid Build Coastguard Worker /* We only need to track compression on CCS_E surfaces. */
645*61046927SAndroid Build Coastguard Worker if (isl_aux_usage_has_ccs_e(image->planes[plane].aux_usage)) {
646*61046927SAndroid Build Coastguard Worker if (image->vk.image_type == VK_IMAGE_TYPE_3D) {
647*61046927SAndroid Build Coastguard Worker for (uint32_t l = 0; l < image->vk.mip_levels; l++)
648*61046927SAndroid Build Coastguard Worker state_size += u_minify(image->vk.extent.depth, l) * 4;
649*61046927SAndroid Build Coastguard Worker } else {
650*61046927SAndroid Build Coastguard Worker state_size += image->vk.mip_levels * image->vk.array_layers * 4;
651*61046927SAndroid Build Coastguard Worker }
652*61046927SAndroid Build Coastguard Worker }
653*61046927SAndroid Build Coastguard Worker
654*61046927SAndroid Build Coastguard Worker enum anv_image_memory_binding binding =
655*61046927SAndroid Build Coastguard Worker ANV_IMAGE_MEMORY_BINDING_PLANE_0 + plane;
656*61046927SAndroid Build Coastguard Worker
657*61046927SAndroid Build Coastguard Worker /* If an auxiliary surface is used for an externally-shareable image,
658*61046927SAndroid Build Coastguard Worker * we have to hide this from the memory of the image since other
659*61046927SAndroid Build Coastguard Worker * processes with access to the memory may not be aware of it or of
660*61046927SAndroid Build Coastguard Worker * its current state. So put that auxiliary data into a separate
661*61046927SAndroid Build Coastguard Worker * buffer (ANV_IMAGE_MEMORY_BINDING_PRIVATE).
662*61046927SAndroid Build Coastguard Worker *
663*61046927SAndroid Build Coastguard Worker * But when the image is created with a drm modifier that supports
664*61046927SAndroid Build Coastguard Worker * clear color, it will be exported along with main surface.
665*61046927SAndroid Build Coastguard Worker */
666*61046927SAndroid Build Coastguard Worker if (anv_image_is_externally_shared(image)
667*61046927SAndroid Build Coastguard Worker && !isl_drm_modifier_get_info(image->vk.drm_format_mod)->supports_clear_color) {
668*61046927SAndroid Build Coastguard Worker binding = ANV_IMAGE_MEMORY_BINDING_PRIVATE;
669*61046927SAndroid Build Coastguard Worker }
670*61046927SAndroid Build Coastguard Worker
671*61046927SAndroid Build Coastguard Worker /* The indirect clear color BO requires 64B-alignment on gfx11+. */
672*61046927SAndroid Build Coastguard Worker return image_binding_grow(device, image, binding,
673*61046927SAndroid Build Coastguard Worker state_offset, state_size, 64,
674*61046927SAndroid Build Coastguard Worker &image->planes[plane].fast_clear_memory_range);
675*61046927SAndroid Build Coastguard Worker }
676*61046927SAndroid Build Coastguard Worker
677*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
add_compression_control_buffer(struct anv_device * device,struct anv_image * image,uint32_t plane,uint32_t binding,uint64_t offset)678*61046927SAndroid Build Coastguard Worker add_compression_control_buffer(struct anv_device *device,
679*61046927SAndroid Build Coastguard Worker struct anv_image *image,
680*61046927SAndroid Build Coastguard Worker uint32_t plane,
681*61046927SAndroid Build Coastguard Worker uint32_t binding,
682*61046927SAndroid Build Coastguard Worker uint64_t offset)
683*61046927SAndroid Build Coastguard Worker {
684*61046927SAndroid Build Coastguard Worker assert(device->info->has_aux_map);
685*61046927SAndroid Build Coastguard Worker
686*61046927SAndroid Build Coastguard Worker return image_binding_grow(device, image, binding, offset,
687*61046927SAndroid Build Coastguard Worker image->planes[plane].primary_surface.isl.size_B /
688*61046927SAndroid Build Coastguard Worker INTEL_AUX_MAP_MAIN_SIZE_SCALEDOWN,
689*61046927SAndroid Build Coastguard Worker INTEL_AUX_MAP_META_ALIGNMENT_B,
690*61046927SAndroid Build Coastguard Worker &image->planes[plane].compr_ctrl_memory_range);
691*61046927SAndroid Build Coastguard Worker }
692*61046927SAndroid Build Coastguard Worker
693*61046927SAndroid Build Coastguard Worker static bool
want_hiz_wt_for_image(const struct intel_device_info * devinfo,const struct anv_image * image)694*61046927SAndroid Build Coastguard Worker want_hiz_wt_for_image(const struct intel_device_info *devinfo,
695*61046927SAndroid Build Coastguard Worker const struct anv_image *image)
696*61046927SAndroid Build Coastguard Worker {
697*61046927SAndroid Build Coastguard Worker /* Gen12 only supports single-sampled while Gen20+ supports
698*61046927SAndroid Build Coastguard Worker * multi-sampled images.
699*61046927SAndroid Build Coastguard Worker */
700*61046927SAndroid Build Coastguard Worker if (devinfo->ver < 20 && image->vk.samples > 1)
701*61046927SAndroid Build Coastguard Worker return false;
702*61046927SAndroid Build Coastguard Worker
703*61046927SAndroid Build Coastguard Worker if ((image->vk.usage & (VK_IMAGE_USAGE_SAMPLED_BIT |
704*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT)) == 0)
705*61046927SAndroid Build Coastguard Worker return false;
706*61046927SAndroid Build Coastguard Worker
707*61046927SAndroid Build Coastguard Worker /* If this image has the maximum number of samples supported by
708*61046927SAndroid Build Coastguard Worker * running platform and will be used as a texture, put the HiZ surface
709*61046927SAndroid Build Coastguard Worker * in write-through mode so that we can sample from it.
710*61046927SAndroid Build Coastguard Worker *
711*61046927SAndroid Build Coastguard Worker * TODO: This is a heuristic trade-off; we haven't tuned it at all.
712*61046927SAndroid Build Coastguard Worker */
713*61046927SAndroid Build Coastguard Worker return true;
714*61046927SAndroid Build Coastguard Worker }
715*61046927SAndroid Build Coastguard Worker
716*61046927SAndroid Build Coastguard Worker /**
717*61046927SAndroid Build Coastguard Worker * The return code indicates whether creation of the VkImage should continue
718*61046927SAndroid Build Coastguard Worker * or fail, not whether the creation of the aux surface succeeded. If the aux
719*61046927SAndroid Build Coastguard Worker * surface is not required (for example, by neither hardware nor DRM format
720*61046927SAndroid Build Coastguard Worker * modifier), then this may return VK_SUCCESS when creation of the aux surface
721*61046927SAndroid Build Coastguard Worker * fails.
722*61046927SAndroid Build Coastguard Worker *
723*61046927SAndroid Build Coastguard Worker * @param offset See add_surface()
724*61046927SAndroid Build Coastguard Worker */
725*61046927SAndroid Build Coastguard Worker static VkResult
add_aux_surface_if_supported(struct anv_device * device,struct anv_image * image,uint32_t plane,struct anv_format_plane plane_format,const VkImageFormatListCreateInfo * fmt_list,uint64_t offset,uint32_t stride,uint64_t aux_state_offset)726*61046927SAndroid Build Coastguard Worker add_aux_surface_if_supported(struct anv_device *device,
727*61046927SAndroid Build Coastguard Worker struct anv_image *image,
728*61046927SAndroid Build Coastguard Worker uint32_t plane,
729*61046927SAndroid Build Coastguard Worker struct anv_format_plane plane_format,
730*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *fmt_list,
731*61046927SAndroid Build Coastguard Worker uint64_t offset,
732*61046927SAndroid Build Coastguard Worker uint32_t stride,
733*61046927SAndroid Build Coastguard Worker uint64_t aux_state_offset)
734*61046927SAndroid Build Coastguard Worker {
735*61046927SAndroid Build Coastguard Worker VkImageAspectFlags aspect = plane_format.aspect;
736*61046927SAndroid Build Coastguard Worker VkResult result;
737*61046927SAndroid Build Coastguard Worker bool ok;
738*61046927SAndroid Build Coastguard Worker
739*61046927SAndroid Build Coastguard Worker /* The aux surface must not be already added. */
740*61046927SAndroid Build Coastguard Worker assert(!anv_surface_is_valid(&image->planes[plane].aux_surface));
741*61046927SAndroid Build Coastguard Worker
742*61046927SAndroid Build Coastguard Worker if (image->planes[plane].primary_surface.isl.usage &
743*61046927SAndroid Build Coastguard Worker ISL_SURF_USAGE_DISABLE_AUX_BIT)
744*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
745*61046927SAndroid Build Coastguard Worker
746*61046927SAndroid Build Coastguard Worker uint32_t binding;
747*61046927SAndroid Build Coastguard Worker if (image->vk.drm_format_mod == DRM_FORMAT_MOD_INVALID ||
748*61046927SAndroid Build Coastguard Worker isl_drm_modifier_has_aux(image->vk.drm_format_mod)) {
749*61046927SAndroid Build Coastguard Worker binding = ANV_IMAGE_MEMORY_BINDING_PLANE_0 + plane;
750*61046927SAndroid Build Coastguard Worker } else {
751*61046927SAndroid Build Coastguard Worker binding = ANV_IMAGE_MEMORY_BINDING_PRIVATE;
752*61046927SAndroid Build Coastguard Worker }
753*61046927SAndroid Build Coastguard Worker
754*61046927SAndroid Build Coastguard Worker if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT) {
755*61046927SAndroid Build Coastguard Worker /* We don't advertise that depth buffers could be used as storage
756*61046927SAndroid Build Coastguard Worker * images.
757*61046927SAndroid Build Coastguard Worker */
758*61046927SAndroid Build Coastguard Worker assert(!(image->vk.usage & VK_IMAGE_USAGE_STORAGE_BIT));
759*61046927SAndroid Build Coastguard Worker
760*61046927SAndroid Build Coastguard Worker ok = isl_surf_get_hiz_surf(&device->isl_dev,
761*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl,
762*61046927SAndroid Build Coastguard Worker &image->planes[plane].aux_surface.isl);
763*61046927SAndroid Build Coastguard Worker if (!ok)
764*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
765*61046927SAndroid Build Coastguard Worker
766*61046927SAndroid Build Coastguard Worker if (!isl_surf_supports_ccs(&device->isl_dev,
767*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl,
768*61046927SAndroid Build Coastguard Worker &image->planes[plane].aux_surface.isl)) {
769*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_HIZ;
770*61046927SAndroid Build Coastguard Worker } else if (want_hiz_wt_for_image(device->info, image)) {
771*61046927SAndroid Build Coastguard Worker assert(device->info->ver >= 12);
772*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_HIZ_CCS_WT;
773*61046927SAndroid Build Coastguard Worker } else {
774*61046927SAndroid Build Coastguard Worker assert(device->info->ver >= 12);
775*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_HIZ_CCS;
776*61046927SAndroid Build Coastguard Worker }
777*61046927SAndroid Build Coastguard Worker
778*61046927SAndroid Build Coastguard Worker result = add_surface(device, image, &image->planes[plane].aux_surface,
779*61046927SAndroid Build Coastguard Worker binding, ANV_OFFSET_IMPLICIT);
780*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
781*61046927SAndroid Build Coastguard Worker return result;
782*61046927SAndroid Build Coastguard Worker
783*61046927SAndroid Build Coastguard Worker if (anv_image_plane_uses_aux_map(device, image, plane)) {
784*61046927SAndroid Build Coastguard Worker result = add_compression_control_buffer(device, image, plane,
785*61046927SAndroid Build Coastguard Worker binding,
786*61046927SAndroid Build Coastguard Worker ANV_OFFSET_IMPLICIT);
787*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
788*61046927SAndroid Build Coastguard Worker return result;
789*61046927SAndroid Build Coastguard Worker }
790*61046927SAndroid Build Coastguard Worker
791*61046927SAndroid Build Coastguard Worker if (device->info->ver == 12 &&
792*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage == ISL_AUX_USAGE_HIZ_CCS_WT) {
793*61046927SAndroid Build Coastguard Worker return add_aux_state_tracking_buffer(device, image, aux_state_offset,
794*61046927SAndroid Build Coastguard Worker plane);
795*61046927SAndroid Build Coastguard Worker }
796*61046927SAndroid Build Coastguard Worker } else if (aspect == VK_IMAGE_ASPECT_STENCIL_BIT) {
797*61046927SAndroid Build Coastguard Worker if (!isl_surf_supports_ccs(&device->isl_dev,
798*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl,
799*61046927SAndroid Build Coastguard Worker NULL))
800*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
801*61046927SAndroid Build Coastguard Worker
802*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_STC_CCS;
803*61046927SAndroid Build Coastguard Worker
804*61046927SAndroid Build Coastguard Worker if (device->info->has_aux_map) {
805*61046927SAndroid Build Coastguard Worker result = add_compression_control_buffer(device, image, plane,
806*61046927SAndroid Build Coastguard Worker binding,
807*61046927SAndroid Build Coastguard Worker ANV_OFFSET_IMPLICIT);
808*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
809*61046927SAndroid Build Coastguard Worker return result;
810*61046927SAndroid Build Coastguard Worker }
811*61046927SAndroid Build Coastguard Worker } else if ((aspect & VK_IMAGE_ASPECT_ANY_COLOR_BIT_ANV) && image->vk.samples == 1) {
812*61046927SAndroid Build Coastguard Worker
813*61046927SAndroid Build Coastguard Worker if (device->info->has_flat_ccs || device->info->has_aux_map) {
814*61046927SAndroid Build Coastguard Worker ok = isl_surf_supports_ccs(&device->isl_dev,
815*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl,
816*61046927SAndroid Build Coastguard Worker NULL);
817*61046927SAndroid Build Coastguard Worker } else {
818*61046927SAndroid Build Coastguard Worker ok = isl_surf_get_ccs_surf(&device->isl_dev,
819*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl,
820*61046927SAndroid Build Coastguard Worker &image->planes[plane].aux_surface.isl,
821*61046927SAndroid Build Coastguard Worker stride);
822*61046927SAndroid Build Coastguard Worker }
823*61046927SAndroid Build Coastguard Worker if (!ok)
824*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
825*61046927SAndroid Build Coastguard Worker
826*61046927SAndroid Build Coastguard Worker /* Choose aux usage. */
827*61046927SAndroid Build Coastguard Worker if (device->info->verx10 == 125 && !device->physical->disable_fcv) {
828*61046927SAndroid Build Coastguard Worker /* FCV is enabled via 3DSTATE_3D_MODE. We'd expect plain CCS_E to
829*61046927SAndroid Build Coastguard Worker * perform better because it allows for non-zero fast clear colors,
830*61046927SAndroid Build Coastguard Worker * but we've run into regressions in several benchmarks (F1 22 and
831*61046927SAndroid Build Coastguard Worker * RDR2) when trying to enable it. When non-zero clear colors are
832*61046927SAndroid Build Coastguard Worker * enabled, we've observed many partial resolves. We haven't yet
833*61046927SAndroid Build Coastguard Worker * root-caused what layout transitions are causing these resolves,
834*61046927SAndroid Build Coastguard Worker * so in the meantime, we choose to reduce our clear color support.
835*61046927SAndroid Build Coastguard Worker * With only zero clear colors being supported, we might as well
836*61046927SAndroid Build Coastguard Worker * turn on FCV.
837*61046927SAndroid Build Coastguard Worker */
838*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_FCV_CCS_E;
839*61046927SAndroid Build Coastguard Worker } else if (intel_needs_workaround(device->info, 1607794140)) {
840*61046927SAndroid Build Coastguard Worker /* FCV is permanently enabled on this hardware. */
841*61046927SAndroid Build Coastguard Worker assert(device->info->verx10 == 120);
842*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_FCV_CCS_E;
843*61046927SAndroid Build Coastguard Worker } else if (device->info->ver >= 12) {
844*61046927SAndroid Build Coastguard Worker /* Support for CCS_E was already checked for in anv_image_init(). */
845*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_CCS_E;
846*61046927SAndroid Build Coastguard Worker } else if (anv_formats_ccs_e_compatible(device->info,
847*61046927SAndroid Build Coastguard Worker image->vk.create_flags,
848*61046927SAndroid Build Coastguard Worker image->vk.format,
849*61046927SAndroid Build Coastguard Worker image->vk.tiling,
850*61046927SAndroid Build Coastguard Worker image->vk.usage, fmt_list)) {
851*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_CCS_E;
852*61046927SAndroid Build Coastguard Worker } else {
853*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_CCS_D;
854*61046927SAndroid Build Coastguard Worker }
855*61046927SAndroid Build Coastguard Worker
856*61046927SAndroid Build Coastguard Worker if (device->info->has_flat_ccs) {
857*61046927SAndroid Build Coastguard Worker result = VK_SUCCESS;
858*61046927SAndroid Build Coastguard Worker } else if (device->info->has_aux_map) {
859*61046927SAndroid Build Coastguard Worker result = add_compression_control_buffer(device, image, plane,
860*61046927SAndroid Build Coastguard Worker binding, offset);
861*61046927SAndroid Build Coastguard Worker } else {
862*61046927SAndroid Build Coastguard Worker result = add_surface(device, image,
863*61046927SAndroid Build Coastguard Worker &image->planes[plane].aux_surface, binding,
864*61046927SAndroid Build Coastguard Worker offset);
865*61046927SAndroid Build Coastguard Worker }
866*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
867*61046927SAndroid Build Coastguard Worker return result;
868*61046927SAndroid Build Coastguard Worker
869*61046927SAndroid Build Coastguard Worker if (device->info->ver <= 12)
870*61046927SAndroid Build Coastguard Worker return add_aux_state_tracking_buffer(device, image, aux_state_offset,
871*61046927SAndroid Build Coastguard Worker plane);
872*61046927SAndroid Build Coastguard Worker } else if ((aspect & VK_IMAGE_ASPECT_ANY_COLOR_BIT_ANV) && image->vk.samples > 1) {
873*61046927SAndroid Build Coastguard Worker assert(!(image->vk.usage & VK_IMAGE_USAGE_STORAGE_BIT));
874*61046927SAndroid Build Coastguard Worker ok = isl_surf_get_mcs_surf(&device->isl_dev,
875*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl,
876*61046927SAndroid Build Coastguard Worker &image->planes[plane].aux_surface.isl);
877*61046927SAndroid Build Coastguard Worker if (!ok)
878*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
879*61046927SAndroid Build Coastguard Worker
880*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_MCS;
881*61046927SAndroid Build Coastguard Worker
882*61046927SAndroid Build Coastguard Worker result = add_surface(device, image, &image->planes[plane].aux_surface,
883*61046927SAndroid Build Coastguard Worker binding, ANV_OFFSET_IMPLICIT);
884*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
885*61046927SAndroid Build Coastguard Worker return result;
886*61046927SAndroid Build Coastguard Worker
887*61046927SAndroid Build Coastguard Worker if (device->info->ver <= 12)
888*61046927SAndroid Build Coastguard Worker return add_aux_state_tracking_buffer(device, image, aux_state_offset,
889*61046927SAndroid Build Coastguard Worker plane);
890*61046927SAndroid Build Coastguard Worker }
891*61046927SAndroid Build Coastguard Worker
892*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
893*61046927SAndroid Build Coastguard Worker }
894*61046927SAndroid Build Coastguard Worker
895*61046927SAndroid Build Coastguard Worker static VkResult
add_video_buffers(struct anv_device * device,struct anv_image * image,const struct VkVideoProfileListInfoKHR * profile_list)896*61046927SAndroid Build Coastguard Worker add_video_buffers(struct anv_device *device,
897*61046927SAndroid Build Coastguard Worker struct anv_image *image,
898*61046927SAndroid Build Coastguard Worker const struct VkVideoProfileListInfoKHR *profile_list)
899*61046927SAndroid Build Coastguard Worker {
900*61046927SAndroid Build Coastguard Worker ASSERTED bool ok;
901*61046927SAndroid Build Coastguard Worker unsigned size = 0;
902*61046927SAndroid Build Coastguard Worker
903*61046927SAndroid Build Coastguard Worker for (unsigned i = 0; i < profile_list->profileCount; i++) {
904*61046927SAndroid Build Coastguard Worker if (profile_list->pProfiles[i].videoCodecOperation == VK_VIDEO_CODEC_OPERATION_DECODE_H264_BIT_KHR) {
905*61046927SAndroid Build Coastguard Worker unsigned w_mb = DIV_ROUND_UP(image->vk.extent.width, ANV_MB_WIDTH);
906*61046927SAndroid Build Coastguard Worker unsigned h_mb = DIV_ROUND_UP(image->vk.extent.height, ANV_MB_HEIGHT);
907*61046927SAndroid Build Coastguard Worker size = w_mb * h_mb * 128;
908*61046927SAndroid Build Coastguard Worker }
909*61046927SAndroid Build Coastguard Worker else if (profile_list->pProfiles[i].videoCodecOperation == VK_VIDEO_CODEC_OPERATION_DECODE_H265_BIT_KHR ||
910*61046927SAndroid Build Coastguard Worker profile_list->pProfiles[i].videoCodecOperation == VK_VIDEO_CODEC_OPERATION_ENCODE_H265_BIT_KHR) {
911*61046927SAndroid Build Coastguard Worker unsigned w_mb = DIV_ROUND_UP(image->vk.extent.width, 32);
912*61046927SAndroid Build Coastguard Worker unsigned h_mb = DIV_ROUND_UP(image->vk.extent.height, 32);
913*61046927SAndroid Build Coastguard Worker size = ALIGN(w_mb * h_mb, 2) << 6;
914*61046927SAndroid Build Coastguard Worker }
915*61046927SAndroid Build Coastguard Worker }
916*61046927SAndroid Build Coastguard Worker
917*61046927SAndroid Build Coastguard Worker if (size == 0)
918*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
919*61046927SAndroid Build Coastguard Worker
920*61046927SAndroid Build Coastguard Worker ok = image_binding_grow(device, image, ANV_IMAGE_MEMORY_BINDING_PRIVATE,
921*61046927SAndroid Build Coastguard Worker ANV_OFFSET_IMPLICIT, size, 65536, &image->vid_dmv_top_surface);
922*61046927SAndroid Build Coastguard Worker return ok;
923*61046927SAndroid Build Coastguard Worker }
924*61046927SAndroid Build Coastguard Worker
925*61046927SAndroid Build Coastguard Worker /**
926*61046927SAndroid Build Coastguard Worker * Initialize the anv_image::*_surface selected by \a aspect. Then update the
927*61046927SAndroid Build Coastguard Worker * image's memory requirements (that is, the image's size and alignment).
928*61046927SAndroid Build Coastguard Worker *
929*61046927SAndroid Build Coastguard Worker * @param offset See add_surface()
930*61046927SAndroid Build Coastguard Worker */
931*61046927SAndroid Build Coastguard Worker static VkResult
add_primary_surface(struct anv_device * device,struct anv_image * image,uint32_t plane,struct anv_format_plane plane_format,uint64_t offset,uint32_t stride,isl_tiling_flags_t isl_tiling_flags,isl_surf_usage_flags_t isl_usage)932*61046927SAndroid Build Coastguard Worker add_primary_surface(struct anv_device *device,
933*61046927SAndroid Build Coastguard Worker struct anv_image *image,
934*61046927SAndroid Build Coastguard Worker uint32_t plane,
935*61046927SAndroid Build Coastguard Worker struct anv_format_plane plane_format,
936*61046927SAndroid Build Coastguard Worker uint64_t offset,
937*61046927SAndroid Build Coastguard Worker uint32_t stride,
938*61046927SAndroid Build Coastguard Worker isl_tiling_flags_t isl_tiling_flags,
939*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_usage)
940*61046927SAndroid Build Coastguard Worker {
941*61046927SAndroid Build Coastguard Worker struct anv_surface *anv_surf = &image->planes[plane].primary_surface;
942*61046927SAndroid Build Coastguard Worker bool ok;
943*61046927SAndroid Build Coastguard Worker
944*61046927SAndroid Build Coastguard Worker uint32_t width = image->vk.extent.width;
945*61046927SAndroid Build Coastguard Worker uint32_t height = image->vk.extent.height;
946*61046927SAndroid Build Coastguard Worker const struct vk_format_ycbcr_info *ycbcr_info =
947*61046927SAndroid Build Coastguard Worker vk_format_get_ycbcr_info(image->vk.format);
948*61046927SAndroid Build Coastguard Worker if (ycbcr_info) {
949*61046927SAndroid Build Coastguard Worker assert(plane < ycbcr_info->n_planes);
950*61046927SAndroid Build Coastguard Worker width /= ycbcr_info->planes[plane].denominator_scales[0];
951*61046927SAndroid Build Coastguard Worker height /= ycbcr_info->planes[plane].denominator_scales[1];
952*61046927SAndroid Build Coastguard Worker }
953*61046927SAndroid Build Coastguard Worker
954*61046927SAndroid Build Coastguard Worker ok = isl_surf_init(&device->isl_dev, &anv_surf->isl,
955*61046927SAndroid Build Coastguard Worker .dim = vk_to_isl_surf_dim[image->vk.image_type],
956*61046927SAndroid Build Coastguard Worker .format = plane_format.isl_format,
957*61046927SAndroid Build Coastguard Worker .width = width,
958*61046927SAndroid Build Coastguard Worker .height = height,
959*61046927SAndroid Build Coastguard Worker .depth = image->vk.extent.depth,
960*61046927SAndroid Build Coastguard Worker .levels = image->vk.mip_levels,
961*61046927SAndroid Build Coastguard Worker .array_len = image->vk.array_layers,
962*61046927SAndroid Build Coastguard Worker .samples = image->vk.samples,
963*61046927SAndroid Build Coastguard Worker .min_alignment_B = 0,
964*61046927SAndroid Build Coastguard Worker .row_pitch_B = stride,
965*61046927SAndroid Build Coastguard Worker .usage = isl_usage,
966*61046927SAndroid Build Coastguard Worker .tiling_flags = isl_tiling_flags);
967*61046927SAndroid Build Coastguard Worker
968*61046927SAndroid Build Coastguard Worker if (!ok) {
969*61046927SAndroid Build Coastguard Worker /* TODO: Should return
970*61046927SAndroid Build Coastguard Worker * VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT in come cases.
971*61046927SAndroid Build Coastguard Worker */
972*61046927SAndroid Build Coastguard Worker return VK_ERROR_OUT_OF_DEVICE_MEMORY;
973*61046927SAndroid Build Coastguard Worker }
974*61046927SAndroid Build Coastguard Worker
975*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage = ISL_AUX_USAGE_NONE;
976*61046927SAndroid Build Coastguard Worker
977*61046927SAndroid Build Coastguard Worker return add_surface(device, image, anv_surf,
978*61046927SAndroid Build Coastguard Worker ANV_IMAGE_MEMORY_BINDING_PLANE_0 + plane, offset);
979*61046927SAndroid Build Coastguard Worker }
980*61046927SAndroid Build Coastguard Worker
981*61046927SAndroid Build Coastguard Worker #ifndef NDEBUG
982*61046927SAndroid Build Coastguard Worker static bool MUST_CHECK
memory_range_is_aligned(struct anv_image_memory_range memory_range)983*61046927SAndroid Build Coastguard Worker memory_range_is_aligned(struct anv_image_memory_range memory_range)
984*61046927SAndroid Build Coastguard Worker {
985*61046927SAndroid Build Coastguard Worker return anv_is_aligned(memory_range.offset, memory_range.alignment);
986*61046927SAndroid Build Coastguard Worker }
987*61046927SAndroid Build Coastguard Worker
988*61046927SAndroid Build Coastguard Worker static bool MUST_CHECK
memory_ranges_equal(struct anv_image_memory_range a,struct anv_image_memory_range b)989*61046927SAndroid Build Coastguard Worker memory_ranges_equal(struct anv_image_memory_range a,
990*61046927SAndroid Build Coastguard Worker struct anv_image_memory_range b)
991*61046927SAndroid Build Coastguard Worker {
992*61046927SAndroid Build Coastguard Worker return a.binding == b.binding &&
993*61046927SAndroid Build Coastguard Worker a.alignment == b.alignment &&
994*61046927SAndroid Build Coastguard Worker a.size == b.size &&
995*61046927SAndroid Build Coastguard Worker a.offset == b.offset;
996*61046927SAndroid Build Coastguard Worker }
997*61046927SAndroid Build Coastguard Worker #endif
998*61046927SAndroid Build Coastguard Worker
999*61046927SAndroid Build Coastguard Worker struct check_memory_range_params {
1000*61046927SAndroid Build Coastguard Worker struct anv_image_memory_range *accum_ranges;
1001*61046927SAndroid Build Coastguard Worker const struct anv_surface *test_surface;
1002*61046927SAndroid Build Coastguard Worker const struct anv_image_memory_range *test_range;
1003*61046927SAndroid Build Coastguard Worker enum anv_image_memory_binding expect_binding;
1004*61046927SAndroid Build Coastguard Worker };
1005*61046927SAndroid Build Coastguard Worker
1006*61046927SAndroid Build Coastguard Worker #define check_memory_range(...) \
1007*61046927SAndroid Build Coastguard Worker check_memory_range_s(&(struct check_memory_range_params) { __VA_ARGS__ })
1008*61046927SAndroid Build Coastguard Worker
1009*61046927SAndroid Build Coastguard Worker static void UNUSED
check_memory_range_s(const struct check_memory_range_params * p)1010*61046927SAndroid Build Coastguard Worker check_memory_range_s(const struct check_memory_range_params *p)
1011*61046927SAndroid Build Coastguard Worker {
1012*61046927SAndroid Build Coastguard Worker assert((p->test_surface == NULL) != (p->test_range == NULL));
1013*61046927SAndroid Build Coastguard Worker
1014*61046927SAndroid Build Coastguard Worker const struct anv_image_memory_range *test_range =
1015*61046927SAndroid Build Coastguard Worker p->test_range ?: &p->test_surface->memory_range;
1016*61046927SAndroid Build Coastguard Worker
1017*61046927SAndroid Build Coastguard Worker struct anv_image_memory_range *accum_range =
1018*61046927SAndroid Build Coastguard Worker &p->accum_ranges[p->expect_binding];
1019*61046927SAndroid Build Coastguard Worker
1020*61046927SAndroid Build Coastguard Worker assert(test_range->binding == p->expect_binding);
1021*61046927SAndroid Build Coastguard Worker assert(test_range->offset >= memory_range_end(*accum_range));
1022*61046927SAndroid Build Coastguard Worker assert(memory_range_is_aligned(*test_range));
1023*61046927SAndroid Build Coastguard Worker
1024*61046927SAndroid Build Coastguard Worker if (p->test_surface) {
1025*61046927SAndroid Build Coastguard Worker assert(anv_surface_is_valid(p->test_surface));
1026*61046927SAndroid Build Coastguard Worker assert(p->test_surface->memory_range.alignment ==
1027*61046927SAndroid Build Coastguard Worker p->test_surface->isl.alignment_B);
1028*61046927SAndroid Build Coastguard Worker }
1029*61046927SAndroid Build Coastguard Worker
1030*61046927SAndroid Build Coastguard Worker memory_range_merge(accum_range, *test_range);
1031*61046927SAndroid Build Coastguard Worker }
1032*61046927SAndroid Build Coastguard Worker
1033*61046927SAndroid Build Coastguard Worker /**
1034*61046927SAndroid Build Coastguard Worker * Validate the image's memory bindings *after* all its surfaces and memory
1035*61046927SAndroid Build Coastguard Worker * ranges are final.
1036*61046927SAndroid Build Coastguard Worker *
1037*61046927SAndroid Build Coastguard Worker * For simplicity's sake, we do not validate free-form layout of the image's
1038*61046927SAndroid Build Coastguard Worker * memory bindings. We validate the layout described in the comments of struct
1039*61046927SAndroid Build Coastguard Worker * anv_image.
1040*61046927SAndroid Build Coastguard Worker */
1041*61046927SAndroid Build Coastguard Worker static void
check_memory_bindings(const struct anv_device * device,const struct anv_image * image)1042*61046927SAndroid Build Coastguard Worker check_memory_bindings(const struct anv_device *device,
1043*61046927SAndroid Build Coastguard Worker const struct anv_image *image)
1044*61046927SAndroid Build Coastguard Worker {
1045*61046927SAndroid Build Coastguard Worker #if MESA_DEBUG
1046*61046927SAndroid Build Coastguard Worker /* As we inspect each part of the image, we merge the part's memory range
1047*61046927SAndroid Build Coastguard Worker * into these accumulation ranges.
1048*61046927SAndroid Build Coastguard Worker */
1049*61046927SAndroid Build Coastguard Worker struct anv_image_memory_range accum_ranges[ANV_IMAGE_MEMORY_BINDING_END];
1050*61046927SAndroid Build Coastguard Worker for (int i = 0; i < ANV_IMAGE_MEMORY_BINDING_END; ++i) {
1051*61046927SAndroid Build Coastguard Worker accum_ranges[i] = (struct anv_image_memory_range) {
1052*61046927SAndroid Build Coastguard Worker .binding = i,
1053*61046927SAndroid Build Coastguard Worker };
1054*61046927SAndroid Build Coastguard Worker }
1055*61046927SAndroid Build Coastguard Worker
1056*61046927SAndroid Build Coastguard Worker for (uint32_t p = 0; p < image->n_planes; ++p) {
1057*61046927SAndroid Build Coastguard Worker const struct anv_image_plane *plane = &image->planes[p];
1058*61046927SAndroid Build Coastguard Worker
1059*61046927SAndroid Build Coastguard Worker /* The binding that must contain the plane's primary surface. */
1060*61046927SAndroid Build Coastguard Worker const enum anv_image_memory_binding primary_binding = image->disjoint
1061*61046927SAndroid Build Coastguard Worker ? ANV_IMAGE_MEMORY_BINDING_PLANE_0 + p
1062*61046927SAndroid Build Coastguard Worker : ANV_IMAGE_MEMORY_BINDING_MAIN;
1063*61046927SAndroid Build Coastguard Worker
1064*61046927SAndroid Build Coastguard Worker /* Aliasing is incompatible with the private binding because it does not
1065*61046927SAndroid Build Coastguard Worker * live in a VkDeviceMemory. The exception is either swapchain images or
1066*61046927SAndroid Build Coastguard Worker * that the private binding is for a video motion vector buffer.
1067*61046927SAndroid Build Coastguard Worker */
1068*61046927SAndroid Build Coastguard Worker assert(!(image->vk.create_flags & VK_IMAGE_CREATE_ALIAS_BIT) ||
1069*61046927SAndroid Build Coastguard Worker image->from_wsi ||
1070*61046927SAndroid Build Coastguard Worker (plane->primary_surface.isl.usage & ISL_SURF_USAGE_VIDEO_DECODE_BIT) ||
1071*61046927SAndroid Build Coastguard Worker image->bindings[ANV_IMAGE_MEMORY_BINDING_PRIVATE].memory_range.size == 0);
1072*61046927SAndroid Build Coastguard Worker
1073*61046927SAndroid Build Coastguard Worker /* Check primary surface */
1074*61046927SAndroid Build Coastguard Worker check_memory_range(accum_ranges,
1075*61046927SAndroid Build Coastguard Worker .test_surface = &plane->primary_surface,
1076*61046927SAndroid Build Coastguard Worker .expect_binding = primary_binding);
1077*61046927SAndroid Build Coastguard Worker
1078*61046927SAndroid Build Coastguard Worker /* Check aux_surface */
1079*61046927SAndroid Build Coastguard Worker const struct anv_image_memory_range *aux_mem_range =
1080*61046927SAndroid Build Coastguard Worker anv_image_get_aux_memory_range(image, p);
1081*61046927SAndroid Build Coastguard Worker if (aux_mem_range->size > 0) {
1082*61046927SAndroid Build Coastguard Worker enum anv_image_memory_binding binding = primary_binding;
1083*61046927SAndroid Build Coastguard Worker
1084*61046927SAndroid Build Coastguard Worker /* If an auxiliary surface is used for an externally-shareable image,
1085*61046927SAndroid Build Coastguard Worker * we have to hide this from the memory of the image since other
1086*61046927SAndroid Build Coastguard Worker * processes with access to the memory may not be aware of it or of
1087*61046927SAndroid Build Coastguard Worker * its current state. So put that auxiliary data into a separate
1088*61046927SAndroid Build Coastguard Worker * buffer (ANV_IMAGE_MEMORY_BINDING_PRIVATE).
1089*61046927SAndroid Build Coastguard Worker */
1090*61046927SAndroid Build Coastguard Worker if (anv_image_is_externally_shared(image) &&
1091*61046927SAndroid Build Coastguard Worker !isl_drm_modifier_has_aux(image->vk.drm_format_mod)) {
1092*61046927SAndroid Build Coastguard Worker binding = ANV_IMAGE_MEMORY_BINDING_PRIVATE;
1093*61046927SAndroid Build Coastguard Worker }
1094*61046927SAndroid Build Coastguard Worker
1095*61046927SAndroid Build Coastguard Worker /* Display hardware requires that the aux surface start at
1096*61046927SAndroid Build Coastguard Worker * a higher address than the primary surface. The 3D hardware
1097*61046927SAndroid Build Coastguard Worker * doesn't care, but we enforce the display requirement in case
1098*61046927SAndroid Build Coastguard Worker * the image is sent to display.
1099*61046927SAndroid Build Coastguard Worker */
1100*61046927SAndroid Build Coastguard Worker check_memory_range(accum_ranges,
1101*61046927SAndroid Build Coastguard Worker .test_range = aux_mem_range,
1102*61046927SAndroid Build Coastguard Worker .expect_binding = binding);
1103*61046927SAndroid Build Coastguard Worker }
1104*61046927SAndroid Build Coastguard Worker
1105*61046927SAndroid Build Coastguard Worker /* Check fast clear state */
1106*61046927SAndroid Build Coastguard Worker if (plane->fast_clear_memory_range.size > 0) {
1107*61046927SAndroid Build Coastguard Worker enum anv_image_memory_binding binding = primary_binding;
1108*61046927SAndroid Build Coastguard Worker
1109*61046927SAndroid Build Coastguard Worker /* If an auxiliary surface is used for an externally-shareable image,
1110*61046927SAndroid Build Coastguard Worker * we have to hide this from the memory of the image since other
1111*61046927SAndroid Build Coastguard Worker * processes with access to the memory may not be aware of it or of
1112*61046927SAndroid Build Coastguard Worker * its current state. So put that auxiliary data into a separate
1113*61046927SAndroid Build Coastguard Worker * buffer (ANV_IMAGE_MEMORY_BINDING_PRIVATE).
1114*61046927SAndroid Build Coastguard Worker *
1115*61046927SAndroid Build Coastguard Worker * But when the image is created with a drm modifier that supports
1116*61046927SAndroid Build Coastguard Worker * clear color, it will be exported along with main surface.
1117*61046927SAndroid Build Coastguard Worker */
1118*61046927SAndroid Build Coastguard Worker if (anv_image_is_externally_shared(image)
1119*61046927SAndroid Build Coastguard Worker && !isl_drm_modifier_get_info(image->vk.drm_format_mod)->supports_clear_color) {
1120*61046927SAndroid Build Coastguard Worker binding = ANV_IMAGE_MEMORY_BINDING_PRIVATE;
1121*61046927SAndroid Build Coastguard Worker }
1122*61046927SAndroid Build Coastguard Worker
1123*61046927SAndroid Build Coastguard Worker /* The indirect clear color BO requires 64B-alignment on gfx11+. */
1124*61046927SAndroid Build Coastguard Worker assert(plane->fast_clear_memory_range.alignment == 64);
1125*61046927SAndroid Build Coastguard Worker check_memory_range(accum_ranges,
1126*61046927SAndroid Build Coastguard Worker .test_range = &plane->fast_clear_memory_range,
1127*61046927SAndroid Build Coastguard Worker .expect_binding = binding);
1128*61046927SAndroid Build Coastguard Worker }
1129*61046927SAndroid Build Coastguard Worker }
1130*61046927SAndroid Build Coastguard Worker #endif
1131*61046927SAndroid Build Coastguard Worker }
1132*61046927SAndroid Build Coastguard Worker
1133*61046927SAndroid Build Coastguard Worker /**
1134*61046927SAndroid Build Coastguard Worker * Check that the fully-initialized anv_image is compatible with its DRM format
1135*61046927SAndroid Build Coastguard Worker * modifier.
1136*61046927SAndroid Build Coastguard Worker *
1137*61046927SAndroid Build Coastguard Worker * Checking compatibility at the end of image creation is prudent, not
1138*61046927SAndroid Build Coastguard Worker * superfluous, because usage of modifiers triggers numerous special cases
1139*61046927SAndroid Build Coastguard Worker * throughout queries and image creation, and because
1140*61046927SAndroid Build Coastguard Worker * vkGetPhysicalDeviceImageFormatProperties2 has difficulty detecting all
1141*61046927SAndroid Build Coastguard Worker * incompatibilities.
1142*61046927SAndroid Build Coastguard Worker *
1143*61046927SAndroid Build Coastguard Worker * Return VK_ERROR_UNKNOWN if the incompatibility is difficult to detect in
1144*61046927SAndroid Build Coastguard Worker * vkGetPhysicalDeviceImageFormatProperties2. Otherwise, assert fail.
1145*61046927SAndroid Build Coastguard Worker *
1146*61046927SAndroid Build Coastguard Worker * Ideally, if vkGetPhysicalDeviceImageFormatProperties2() succeeds with a given
1147*61046927SAndroid Build Coastguard Worker * modifier, then vkCreateImage() produces an image that is compatible with the
1148*61046927SAndroid Build Coastguard Worker * modifier. However, it is difficult to reconcile the two functions to agree
1149*61046927SAndroid Build Coastguard Worker * due to their complexity. For example, isl_surf_get_ccs_surf() may
1150*61046927SAndroid Build Coastguard Worker * unexpectedly fail in vkCreateImage(), eliminating the image's aux surface
1151*61046927SAndroid Build Coastguard Worker * even when the modifier requires one. (Maybe we should reconcile the two
1152*61046927SAndroid Build Coastguard Worker * functions despite the difficulty).
1153*61046927SAndroid Build Coastguard Worker */
1154*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
check_drm_format_mod(const struct anv_device * device,const struct anv_image * image)1155*61046927SAndroid Build Coastguard Worker check_drm_format_mod(const struct anv_device *device,
1156*61046927SAndroid Build Coastguard Worker const struct anv_image *image)
1157*61046927SAndroid Build Coastguard Worker {
1158*61046927SAndroid Build Coastguard Worker /* Image must have a modifier if and only if it has modifier tiling. */
1159*61046927SAndroid Build Coastguard Worker assert((image->vk.drm_format_mod != DRM_FORMAT_MOD_INVALID) ==
1160*61046927SAndroid Build Coastguard Worker (image->vk.tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT));
1161*61046927SAndroid Build Coastguard Worker
1162*61046927SAndroid Build Coastguard Worker if (image->vk.drm_format_mod == DRM_FORMAT_MOD_INVALID)
1163*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1164*61046927SAndroid Build Coastguard Worker
1165*61046927SAndroid Build Coastguard Worker const struct isl_drm_modifier_info *isl_mod_info =
1166*61046927SAndroid Build Coastguard Worker isl_drm_modifier_get_info(image->vk.drm_format_mod);
1167*61046927SAndroid Build Coastguard Worker
1168*61046927SAndroid Build Coastguard Worker /* Driver must support the modifier. */
1169*61046927SAndroid Build Coastguard Worker assert(isl_drm_modifier_get_score(device->info, isl_mod_info->modifier));
1170*61046927SAndroid Build Coastguard Worker
1171*61046927SAndroid Build Coastguard Worker /* Enforced by us, not the Vulkan spec. */
1172*61046927SAndroid Build Coastguard Worker assert(image->vk.image_type == VK_IMAGE_TYPE_2D);
1173*61046927SAndroid Build Coastguard Worker assert(!(image->vk.aspects & VK_IMAGE_ASPECT_DEPTH_BIT));
1174*61046927SAndroid Build Coastguard Worker assert(!(image->vk.aspects & VK_IMAGE_ASPECT_STENCIL_BIT));
1175*61046927SAndroid Build Coastguard Worker assert(image->vk.mip_levels == 1);
1176*61046927SAndroid Build Coastguard Worker assert(image->vk.array_layers == 1);
1177*61046927SAndroid Build Coastguard Worker assert(image->vk.samples == 1);
1178*61046927SAndroid Build Coastguard Worker
1179*61046927SAndroid Build Coastguard Worker for (int i = 0; i < image->n_planes; ++i) {
1180*61046927SAndroid Build Coastguard Worker const struct anv_image_plane *plane = &image->planes[i];
1181*61046927SAndroid Build Coastguard Worker ASSERTED const struct isl_format_layout *isl_layout =
1182*61046927SAndroid Build Coastguard Worker isl_format_get_layout(plane->primary_surface.isl.format);
1183*61046927SAndroid Build Coastguard Worker
1184*61046927SAndroid Build Coastguard Worker /* Enforced by us, not the Vulkan spec. */
1185*61046927SAndroid Build Coastguard Worker assert(isl_layout->txc == ISL_TXC_NONE);
1186*61046927SAndroid Build Coastguard Worker assert(isl_layout->colorspace == ISL_COLORSPACE_LINEAR ||
1187*61046927SAndroid Build Coastguard Worker isl_layout->colorspace == ISL_COLORSPACE_SRGB);
1188*61046927SAndroid Build Coastguard Worker
1189*61046927SAndroid Build Coastguard Worker if (isl_drm_modifier_has_aux(isl_mod_info->modifier)) {
1190*61046927SAndroid Build Coastguard Worker /* Reject DISJOINT for consistency with the GL driver. */
1191*61046927SAndroid Build Coastguard Worker assert(!image->disjoint);
1192*61046927SAndroid Build Coastguard Worker
1193*61046927SAndroid Build Coastguard Worker /* The modifier's required aux usage mandates the image's aux usage.
1194*61046927SAndroid Build Coastguard Worker * The inverse, however, does not hold; if the modifier has no aux
1195*61046927SAndroid Build Coastguard Worker * usage, then we may enable a private aux surface.
1196*61046927SAndroid Build Coastguard Worker */
1197*61046927SAndroid Build Coastguard Worker if ((isl_mod_info->supports_media_compression &&
1198*61046927SAndroid Build Coastguard Worker plane->aux_usage != ISL_AUX_USAGE_MC) ||
1199*61046927SAndroid Build Coastguard Worker (isl_mod_info->supports_render_compression &&
1200*61046927SAndroid Build Coastguard Worker !isl_aux_usage_has_ccs_e(plane->aux_usage))) {
1201*61046927SAndroid Build Coastguard Worker return vk_errorf(device, VK_ERROR_UNKNOWN,
1202*61046927SAndroid Build Coastguard Worker "image with modifier unexpectedly has wrong aux "
1203*61046927SAndroid Build Coastguard Worker "usage");
1204*61046927SAndroid Build Coastguard Worker }
1205*61046927SAndroid Build Coastguard Worker }
1206*61046927SAndroid Build Coastguard Worker }
1207*61046927SAndroid Build Coastguard Worker
1208*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1209*61046927SAndroid Build Coastguard Worker }
1210*61046927SAndroid Build Coastguard Worker
1211*61046927SAndroid Build Coastguard Worker /**
1212*61046927SAndroid Build Coastguard Worker * Use when the app does not provide
1213*61046927SAndroid Build Coastguard Worker * VkImageDrmFormatModifierExplicitCreateInfoEXT.
1214*61046927SAndroid Build Coastguard Worker */
1215*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
add_all_surfaces_implicit_layout(struct anv_device * device,struct anv_image * image,const VkImageFormatListCreateInfo * format_list_info,uint32_t stride,isl_tiling_flags_t isl_tiling_flags,isl_surf_usage_flags_t isl_extra_usage_flags)1216*61046927SAndroid Build Coastguard Worker add_all_surfaces_implicit_layout(
1217*61046927SAndroid Build Coastguard Worker struct anv_device *device,
1218*61046927SAndroid Build Coastguard Worker struct anv_image *image,
1219*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *format_list_info,
1220*61046927SAndroid Build Coastguard Worker uint32_t stride,
1221*61046927SAndroid Build Coastguard Worker isl_tiling_flags_t isl_tiling_flags,
1222*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_extra_usage_flags)
1223*61046927SAndroid Build Coastguard Worker {
1224*61046927SAndroid Build Coastguard Worker const struct intel_device_info *devinfo = device->info;
1225*61046927SAndroid Build Coastguard Worker VkResult result;
1226*61046927SAndroid Build Coastguard Worker
1227*61046927SAndroid Build Coastguard Worker const struct vk_format_ycbcr_info *ycbcr_info =
1228*61046927SAndroid Build Coastguard Worker vk_format_get_ycbcr_info(image->vk.format);
1229*61046927SAndroid Build Coastguard Worker if (ycbcr_info)
1230*61046927SAndroid Build Coastguard Worker assert(ycbcr_info->n_planes == image->n_planes);
1231*61046927SAndroid Build Coastguard Worker
1232*61046927SAndroid Build Coastguard Worker unsigned num_aspects = 0;
1233*61046927SAndroid Build Coastguard Worker VkImageAspectFlagBits aspects[3];
1234*61046927SAndroid Build Coastguard Worker u_foreach_bit(b, image->vk.aspects) {
1235*61046927SAndroid Build Coastguard Worker assert(num_aspects < 3);
1236*61046927SAndroid Build Coastguard Worker aspects[num_aspects++] = 1 << b;
1237*61046927SAndroid Build Coastguard Worker }
1238*61046927SAndroid Build Coastguard Worker assert(num_aspects == image->n_planes);
1239*61046927SAndroid Build Coastguard Worker
1240*61046927SAndroid Build Coastguard Worker /* The Android hardware buffer YV12 format has the planes ordered as Y-Cr-Cb,
1241*61046927SAndroid Build Coastguard Worker * while Vulkan expects VK_FORMAT_G8_B8_R8_3PLANE_420_UNORM to be in Y-Cb-Cr.
1242*61046927SAndroid Build Coastguard Worker * Adjust the order we add the ISL surfaces accordingly so the implicit
1243*61046927SAndroid Build Coastguard Worker * offset gets calculated correctly.
1244*61046927SAndroid Build Coastguard Worker */
1245*61046927SAndroid Build Coastguard Worker if (image->from_ahb && image->vk.format == VK_FORMAT_G8_B8_R8_3PLANE_420_UNORM) {
1246*61046927SAndroid Build Coastguard Worker assert(num_aspects == 3);
1247*61046927SAndroid Build Coastguard Worker assert(aspects[1] == VK_IMAGE_ASPECT_PLANE_1_BIT);
1248*61046927SAndroid Build Coastguard Worker assert(aspects[2] == VK_IMAGE_ASPECT_PLANE_2_BIT);
1249*61046927SAndroid Build Coastguard Worker aspects[1] = VK_IMAGE_ASPECT_PLANE_2_BIT;
1250*61046927SAndroid Build Coastguard Worker aspects[2] = VK_IMAGE_ASPECT_PLANE_1_BIT;
1251*61046927SAndroid Build Coastguard Worker }
1252*61046927SAndroid Build Coastguard Worker
1253*61046927SAndroid Build Coastguard Worker for (unsigned i = 0; i < num_aspects; i++) {
1254*61046927SAndroid Build Coastguard Worker VkImageAspectFlagBits aspect = aspects[i];
1255*61046927SAndroid Build Coastguard Worker const uint32_t plane = anv_image_aspect_to_plane(image, aspect);
1256*61046927SAndroid Build Coastguard Worker const struct anv_format_plane plane_format =
1257*61046927SAndroid Build Coastguard Worker anv_get_format_plane(devinfo, image->vk.format, plane, image->vk.tiling);
1258*61046927SAndroid Build Coastguard Worker
1259*61046927SAndroid Build Coastguard Worker enum isl_format isl_fmt = plane_format.isl_format;
1260*61046927SAndroid Build Coastguard Worker assert(isl_fmt != ISL_FORMAT_UNSUPPORTED);
1261*61046927SAndroid Build Coastguard Worker
1262*61046927SAndroid Build Coastguard Worker uint32_t plane_stride = stride * isl_format_get_layout(isl_fmt)->bpb / 8;
1263*61046927SAndroid Build Coastguard Worker if (ycbcr_info)
1264*61046927SAndroid Build Coastguard Worker plane_stride /= ycbcr_info->planes[plane].denominator_scales[0];
1265*61046927SAndroid Build Coastguard Worker
1266*61046927SAndroid Build Coastguard Worker VkImageUsageFlags vk_usage = vk_image_usage(&image->vk, aspect);
1267*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_usage =
1268*61046927SAndroid Build Coastguard Worker anv_image_choose_isl_surf_usage(device->physical,
1269*61046927SAndroid Build Coastguard Worker image->vk.create_flags, vk_usage,
1270*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags, aspect,
1271*61046927SAndroid Build Coastguard Worker image->vk.compr_flags);
1272*61046927SAndroid Build Coastguard Worker
1273*61046927SAndroid Build Coastguard Worker result = add_primary_surface(device, image, plane, plane_format,
1274*61046927SAndroid Build Coastguard Worker ANV_OFFSET_IMPLICIT, plane_stride,
1275*61046927SAndroid Build Coastguard Worker isl_tiling_flags, isl_usage);
1276*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
1277*61046927SAndroid Build Coastguard Worker return result;
1278*61046927SAndroid Build Coastguard Worker
1279*61046927SAndroid Build Coastguard Worker result = add_aux_surface_if_supported(device, image, plane, plane_format,
1280*61046927SAndroid Build Coastguard Worker format_list_info,
1281*61046927SAndroid Build Coastguard Worker ANV_OFFSET_IMPLICIT, plane_stride,
1282*61046927SAndroid Build Coastguard Worker ANV_OFFSET_IMPLICIT);
1283*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
1284*61046927SAndroid Build Coastguard Worker return result;
1285*61046927SAndroid Build Coastguard Worker }
1286*61046927SAndroid Build Coastguard Worker
1287*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1288*61046927SAndroid Build Coastguard Worker }
1289*61046927SAndroid Build Coastguard Worker
1290*61046927SAndroid Build Coastguard Worker /**
1291*61046927SAndroid Build Coastguard Worker * Use when the app provides VkImageDrmFormatModifierExplicitCreateInfoEXT.
1292*61046927SAndroid Build Coastguard Worker */
1293*61046927SAndroid Build Coastguard Worker static VkResult
add_all_surfaces_explicit_layout(struct anv_device * device,struct anv_image * image,const VkImageFormatListCreateInfo * format_list_info,const VkImageDrmFormatModifierExplicitCreateInfoEXT * drm_info,isl_tiling_flags_t isl_tiling_flags,isl_surf_usage_flags_t isl_extra_usage_flags)1294*61046927SAndroid Build Coastguard Worker add_all_surfaces_explicit_layout(
1295*61046927SAndroid Build Coastguard Worker struct anv_device *device,
1296*61046927SAndroid Build Coastguard Worker struct anv_image *image,
1297*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *format_list_info,
1298*61046927SAndroid Build Coastguard Worker const VkImageDrmFormatModifierExplicitCreateInfoEXT *drm_info,
1299*61046927SAndroid Build Coastguard Worker isl_tiling_flags_t isl_tiling_flags,
1300*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_extra_usage_flags)
1301*61046927SAndroid Build Coastguard Worker {
1302*61046927SAndroid Build Coastguard Worker const struct intel_device_info *devinfo = device->info;
1303*61046927SAndroid Build Coastguard Worker const uint32_t mod_plane_count = drm_info->drmFormatModifierPlaneCount;
1304*61046927SAndroid Build Coastguard Worker const bool mod_has_aux =
1305*61046927SAndroid Build Coastguard Worker isl_drm_modifier_has_aux(drm_info->drmFormatModifier);
1306*61046927SAndroid Build Coastguard Worker VkResult result;
1307*61046927SAndroid Build Coastguard Worker
1308*61046927SAndroid Build Coastguard Worker /* Currently there is no way to properly map memory planes to format planes
1309*61046927SAndroid Build Coastguard Worker * and aux planes due to the lack of defined ABI for external multi-planar
1310*61046927SAndroid Build Coastguard Worker * images.
1311*61046927SAndroid Build Coastguard Worker */
1312*61046927SAndroid Build Coastguard Worker if (image->n_planes == 1)
1313*61046927SAndroid Build Coastguard Worker assert(image->vk.aspects == VK_IMAGE_ASPECT_COLOR_BIT);
1314*61046927SAndroid Build Coastguard Worker else
1315*61046927SAndroid Build Coastguard Worker assert(!(image->vk.aspects & ~VK_IMAGE_ASPECT_PLANES_BITS_ANV));
1316*61046927SAndroid Build Coastguard Worker
1317*61046927SAndroid Build Coastguard Worker if (mod_has_aux) {
1318*61046927SAndroid Build Coastguard Worker assert(image->n_planes == 1);
1319*61046927SAndroid Build Coastguard Worker
1320*61046927SAndroid Build Coastguard Worker /* About valid usage in the Vulkan spec:
1321*61046927SAndroid Build Coastguard Worker *
1322*61046927SAndroid Build Coastguard Worker * Unlike vanilla vkCreateImage, which produces undefined behavior on user
1323*61046927SAndroid Build Coastguard Worker * error, here the spec requires the implementation to return
1324*61046927SAndroid Build Coastguard Worker * VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT if the app provides
1325*61046927SAndroid Build Coastguard Worker * a bad plane layout. However, the spec does require
1326*61046927SAndroid Build Coastguard Worker * drmFormatModifierPlaneCount to be valid.
1327*61046927SAndroid Build Coastguard Worker *
1328*61046927SAndroid Build Coastguard Worker * Most validation of plane layout occurs in add_surface().
1329*61046927SAndroid Build Coastguard Worker */
1330*61046927SAndroid Build Coastguard Worker uint32_t n_mod_planes =
1331*61046927SAndroid Build Coastguard Worker isl_drm_modifier_get_plane_count(devinfo,
1332*61046927SAndroid Build Coastguard Worker drm_info->drmFormatModifier,
1333*61046927SAndroid Build Coastguard Worker image->n_planes);
1334*61046927SAndroid Build Coastguard Worker assert(n_mod_planes == mod_plane_count);
1335*61046927SAndroid Build Coastguard Worker } else {
1336*61046927SAndroid Build Coastguard Worker assert(image->n_planes == mod_plane_count);
1337*61046927SAndroid Build Coastguard Worker }
1338*61046927SAndroid Build Coastguard Worker
1339*61046927SAndroid Build Coastguard Worker /* Reject special values in the app-provided plane layouts. */
1340*61046927SAndroid Build Coastguard Worker for (uint32_t i = 0; i < mod_plane_count; ++i) {
1341*61046927SAndroid Build Coastguard Worker if (drm_info->pPlaneLayouts[i].rowPitch == 0) {
1342*61046927SAndroid Build Coastguard Worker return vk_errorf(device,
1343*61046927SAndroid Build Coastguard Worker VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT,
1344*61046927SAndroid Build Coastguard Worker "VkImageDrmFormatModifierExplicitCreateInfoEXT::"
1345*61046927SAndroid Build Coastguard Worker "pPlaneLayouts[%u]::rowPitch is 0", i);
1346*61046927SAndroid Build Coastguard Worker }
1347*61046927SAndroid Build Coastguard Worker
1348*61046927SAndroid Build Coastguard Worker if (drm_info->pPlaneLayouts[i].offset == ANV_OFFSET_IMPLICIT) {
1349*61046927SAndroid Build Coastguard Worker return vk_errorf(device,
1350*61046927SAndroid Build Coastguard Worker VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT,
1351*61046927SAndroid Build Coastguard Worker "VkImageDrmFormatModifierExplicitCreateInfoEXT::"
1352*61046927SAndroid Build Coastguard Worker "pPlaneLayouts[%u]::offset is %" PRIu64,
1353*61046927SAndroid Build Coastguard Worker i, ANV_OFFSET_IMPLICIT);
1354*61046927SAndroid Build Coastguard Worker }
1355*61046927SAndroid Build Coastguard Worker }
1356*61046927SAndroid Build Coastguard Worker
1357*61046927SAndroid Build Coastguard Worker u_foreach_bit(b, image->vk.aspects) {
1358*61046927SAndroid Build Coastguard Worker const VkImageAspectFlagBits aspect = 1 << b;
1359*61046927SAndroid Build Coastguard Worker const uint32_t plane = anv_image_aspect_to_plane(image, aspect);
1360*61046927SAndroid Build Coastguard Worker const struct anv_format_plane format_plane =
1361*61046927SAndroid Build Coastguard Worker anv_get_format_plane(devinfo, image->vk.format, plane, image->vk.tiling);
1362*61046927SAndroid Build Coastguard Worker const VkSubresourceLayout *primary_layout = &drm_info->pPlaneLayouts[plane];
1363*61046927SAndroid Build Coastguard Worker
1364*61046927SAndroid Build Coastguard Worker result = add_primary_surface(device, image, plane,
1365*61046927SAndroid Build Coastguard Worker format_plane,
1366*61046927SAndroid Build Coastguard Worker primary_layout->offset,
1367*61046927SAndroid Build Coastguard Worker primary_layout->rowPitch,
1368*61046927SAndroid Build Coastguard Worker isl_tiling_flags,
1369*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags);
1370*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
1371*61046927SAndroid Build Coastguard Worker return result;
1372*61046927SAndroid Build Coastguard Worker
1373*61046927SAndroid Build Coastguard Worker if (mod_has_aux) {
1374*61046927SAndroid Build Coastguard Worker const VkSubresourceLayout flat_ccs_layout = {
1375*61046927SAndroid Build Coastguard Worker .offset = ANV_OFFSET_IMPLICIT,
1376*61046927SAndroid Build Coastguard Worker };
1377*61046927SAndroid Build Coastguard Worker
1378*61046927SAndroid Build Coastguard Worker const VkSubresourceLayout *aux_layout;
1379*61046927SAndroid Build Coastguard Worker
1380*61046927SAndroid Build Coastguard Worker uint64_t aux_state_offset = ANV_OFFSET_IMPLICIT;
1381*61046927SAndroid Build Coastguard Worker
1382*61046927SAndroid Build Coastguard Worker /* We already asserted on image->n_planes == 1 when mod_has_aux is
1383*61046927SAndroid Build Coastguard Worker * true above, so the indexes of aux and clear color are just hard-
1384*61046927SAndroid Build Coastguard Worker * coded without ambiguity.
1385*61046927SAndroid Build Coastguard Worker */
1386*61046927SAndroid Build Coastguard Worker if (devinfo->has_flat_ccs) {
1387*61046927SAndroid Build Coastguard Worker aux_layout = &flat_ccs_layout;
1388*61046927SAndroid Build Coastguard Worker if (isl_drm_modifier_get_info(
1389*61046927SAndroid Build Coastguard Worker drm_info->drmFormatModifier)->supports_clear_color) {
1390*61046927SAndroid Build Coastguard Worker aux_state_offset = drm_info->pPlaneLayouts[1].offset;
1391*61046927SAndroid Build Coastguard Worker }
1392*61046927SAndroid Build Coastguard Worker } else {
1393*61046927SAndroid Build Coastguard Worker aux_layout = &drm_info->pPlaneLayouts[1];
1394*61046927SAndroid Build Coastguard Worker if (isl_drm_modifier_get_info(
1395*61046927SAndroid Build Coastguard Worker drm_info->drmFormatModifier)->supports_clear_color) {
1396*61046927SAndroid Build Coastguard Worker aux_state_offset = drm_info->pPlaneLayouts[2].offset;
1397*61046927SAndroid Build Coastguard Worker }
1398*61046927SAndroid Build Coastguard Worker }
1399*61046927SAndroid Build Coastguard Worker
1400*61046927SAndroid Build Coastguard Worker result = add_aux_surface_if_supported(device, image, plane,
1401*61046927SAndroid Build Coastguard Worker format_plane,
1402*61046927SAndroid Build Coastguard Worker format_list_info,
1403*61046927SAndroid Build Coastguard Worker aux_layout->offset,
1404*61046927SAndroid Build Coastguard Worker aux_layout->rowPitch,
1405*61046927SAndroid Build Coastguard Worker aux_state_offset);
1406*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
1407*61046927SAndroid Build Coastguard Worker return result;
1408*61046927SAndroid Build Coastguard Worker
1409*61046927SAndroid Build Coastguard Worker assert(isl_aux_usage_has_ccs(image->planes[plane].aux_usage));
1410*61046927SAndroid Build Coastguard Worker }
1411*61046927SAndroid Build Coastguard Worker }
1412*61046927SAndroid Build Coastguard Worker
1413*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1414*61046927SAndroid Build Coastguard Worker }
1415*61046927SAndroid Build Coastguard Worker
1416*61046927SAndroid Build Coastguard Worker static const struct isl_drm_modifier_info *
choose_drm_format_mod(const struct anv_physical_device * device,uint32_t modifier_count,const uint64_t * modifiers)1417*61046927SAndroid Build Coastguard Worker choose_drm_format_mod(const struct anv_physical_device *device,
1418*61046927SAndroid Build Coastguard Worker uint32_t modifier_count, const uint64_t *modifiers)
1419*61046927SAndroid Build Coastguard Worker {
1420*61046927SAndroid Build Coastguard Worker uint64_t best_mod = UINT64_MAX;
1421*61046927SAndroid Build Coastguard Worker uint32_t best_score = 0;
1422*61046927SAndroid Build Coastguard Worker
1423*61046927SAndroid Build Coastguard Worker for (uint32_t i = 0; i < modifier_count; ++i) {
1424*61046927SAndroid Build Coastguard Worker uint32_t score = isl_drm_modifier_get_score(&device->info, modifiers[i]);
1425*61046927SAndroid Build Coastguard Worker if (score > best_score) {
1426*61046927SAndroid Build Coastguard Worker best_mod = modifiers[i];
1427*61046927SAndroid Build Coastguard Worker best_score = score;
1428*61046927SAndroid Build Coastguard Worker }
1429*61046927SAndroid Build Coastguard Worker }
1430*61046927SAndroid Build Coastguard Worker
1431*61046927SAndroid Build Coastguard Worker if (best_score > 0)
1432*61046927SAndroid Build Coastguard Worker return isl_drm_modifier_get_info(best_mod);
1433*61046927SAndroid Build Coastguard Worker else
1434*61046927SAndroid Build Coastguard Worker return NULL;
1435*61046927SAndroid Build Coastguard Worker }
1436*61046927SAndroid Build Coastguard Worker
1437*61046927SAndroid Build Coastguard Worker static VkImageUsageFlags
anv_image_create_usage(const VkImageCreateInfo * pCreateInfo,VkImageUsageFlags usage)1438*61046927SAndroid Build Coastguard Worker anv_image_create_usage(const VkImageCreateInfo *pCreateInfo,
1439*61046927SAndroid Build Coastguard Worker VkImageUsageFlags usage)
1440*61046927SAndroid Build Coastguard Worker {
1441*61046927SAndroid Build Coastguard Worker /* Add TRANSFER_SRC usage for multisample attachment images. This is
1442*61046927SAndroid Build Coastguard Worker * because we might internally use the TRANSFER_SRC layout on them for
1443*61046927SAndroid Build Coastguard Worker * blorp operations associated with resolving those into other attachments
1444*61046927SAndroid Build Coastguard Worker * at the end of a subpass.
1445*61046927SAndroid Build Coastguard Worker *
1446*61046927SAndroid Build Coastguard Worker * Without this additional usage, we compute an incorrect AUX state in
1447*61046927SAndroid Build Coastguard Worker * anv_layout_to_aux_state().
1448*61046927SAndroid Build Coastguard Worker */
1449*61046927SAndroid Build Coastguard Worker if (pCreateInfo->samples > VK_SAMPLE_COUNT_1_BIT &&
1450*61046927SAndroid Build Coastguard Worker (usage & (VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
1451*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT)))
1452*61046927SAndroid Build Coastguard Worker usage |= VK_IMAGE_USAGE_TRANSFER_SRC_BIT;
1453*61046927SAndroid Build Coastguard Worker return usage;
1454*61046927SAndroid Build Coastguard Worker }
1455*61046927SAndroid Build Coastguard Worker
1456*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
alloc_private_binding(struct anv_device * device,struct anv_image * image,const VkImageCreateInfo * create_info)1457*61046927SAndroid Build Coastguard Worker alloc_private_binding(struct anv_device *device,
1458*61046927SAndroid Build Coastguard Worker struct anv_image *image,
1459*61046927SAndroid Build Coastguard Worker const VkImageCreateInfo *create_info)
1460*61046927SAndroid Build Coastguard Worker {
1461*61046927SAndroid Build Coastguard Worker struct anv_image_binding *binding =
1462*61046927SAndroid Build Coastguard Worker &image->bindings[ANV_IMAGE_MEMORY_BINDING_PRIVATE];
1463*61046927SAndroid Build Coastguard Worker
1464*61046927SAndroid Build Coastguard Worker if (binding->memory_range.size == 0)
1465*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1466*61046927SAndroid Build Coastguard Worker
1467*61046927SAndroid Build Coastguard Worker const VkImageSwapchainCreateInfoKHR *swapchain_info =
1468*61046927SAndroid Build Coastguard Worker vk_find_struct_const(create_info->pNext, IMAGE_SWAPCHAIN_CREATE_INFO_KHR);
1469*61046927SAndroid Build Coastguard Worker
1470*61046927SAndroid Build Coastguard Worker if (swapchain_info && swapchain_info->swapchain != VK_NULL_HANDLE) {
1471*61046927SAndroid Build Coastguard Worker /* The image will be bound to swapchain memory. */
1472*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1473*61046927SAndroid Build Coastguard Worker }
1474*61046927SAndroid Build Coastguard Worker
1475*61046927SAndroid Build Coastguard Worker VkResult result = anv_device_alloc_bo(device, "image-binding-private",
1476*61046927SAndroid Build Coastguard Worker binding->memory_range.size, 0, 0,
1477*61046927SAndroid Build Coastguard Worker &binding->address.bo);
1478*61046927SAndroid Build Coastguard Worker if (result == VK_SUCCESS) {
1479*61046927SAndroid Build Coastguard Worker pthread_mutex_lock(&device->mutex);
1480*61046927SAndroid Build Coastguard Worker list_addtail(&image->link, &device->image_private_objects);
1481*61046927SAndroid Build Coastguard Worker pthread_mutex_unlock(&device->mutex);
1482*61046927SAndroid Build Coastguard Worker }
1483*61046927SAndroid Build Coastguard Worker
1484*61046927SAndroid Build Coastguard Worker return result;
1485*61046927SAndroid Build Coastguard Worker }
1486*61046927SAndroid Build Coastguard Worker
1487*61046927SAndroid Build Coastguard Worker static void
anv_image_finish_sparse_bindings(struct anv_image * image)1488*61046927SAndroid Build Coastguard Worker anv_image_finish_sparse_bindings(struct anv_image *image)
1489*61046927SAndroid Build Coastguard Worker {
1490*61046927SAndroid Build Coastguard Worker struct anv_device *device =
1491*61046927SAndroid Build Coastguard Worker container_of(image->vk.base.device, struct anv_device, vk);
1492*61046927SAndroid Build Coastguard Worker
1493*61046927SAndroid Build Coastguard Worker assert(anv_image_is_sparse(image));
1494*61046927SAndroid Build Coastguard Worker
1495*61046927SAndroid Build Coastguard Worker for (int i = 0; i < ANV_IMAGE_MEMORY_BINDING_END; i++) {
1496*61046927SAndroid Build Coastguard Worker struct anv_image_binding *b = &image->bindings[i];
1497*61046927SAndroid Build Coastguard Worker
1498*61046927SAndroid Build Coastguard Worker if (b->sparse_data.size != 0) {
1499*61046927SAndroid Build Coastguard Worker assert(b->memory_range.size == b->sparse_data.size);
1500*61046927SAndroid Build Coastguard Worker assert(b->address.offset == b->sparse_data.address);
1501*61046927SAndroid Build Coastguard Worker anv_free_sparse_bindings(device, &b->sparse_data);
1502*61046927SAndroid Build Coastguard Worker }
1503*61046927SAndroid Build Coastguard Worker }
1504*61046927SAndroid Build Coastguard Worker }
1505*61046927SAndroid Build Coastguard Worker
1506*61046927SAndroid Build Coastguard Worker static VkResult MUST_CHECK
anv_image_init_sparse_bindings(struct anv_image * image,const struct anv_image_create_info * create_info)1507*61046927SAndroid Build Coastguard Worker anv_image_init_sparse_bindings(struct anv_image *image,
1508*61046927SAndroid Build Coastguard Worker const struct anv_image_create_info *create_info)
1509*61046927SAndroid Build Coastguard Worker {
1510*61046927SAndroid Build Coastguard Worker struct anv_device *device =
1511*61046927SAndroid Build Coastguard Worker container_of(image->vk.base.device, struct anv_device, vk);
1512*61046927SAndroid Build Coastguard Worker VkResult result;
1513*61046927SAndroid Build Coastguard Worker
1514*61046927SAndroid Build Coastguard Worker assert(anv_image_is_sparse(image));
1515*61046927SAndroid Build Coastguard Worker
1516*61046927SAndroid Build Coastguard Worker enum anv_bo_alloc_flags alloc_flags = 0;
1517*61046927SAndroid Build Coastguard Worker uint64_t explicit_address = 0;
1518*61046927SAndroid Build Coastguard Worker if (image->vk.create_flags & VK_IMAGE_CREATE_DESCRIPTOR_BUFFER_CAPTURE_REPLAY_BIT_EXT) {
1519*61046927SAndroid Build Coastguard Worker alloc_flags |= ANV_BO_ALLOC_FIXED_ADDRESS;
1520*61046927SAndroid Build Coastguard Worker
1521*61046927SAndroid Build Coastguard Worker const VkOpaqueCaptureDescriptorDataCreateInfoEXT *opaque_info =
1522*61046927SAndroid Build Coastguard Worker vk_find_struct_const(create_info->vk_info->pNext,
1523*61046927SAndroid Build Coastguard Worker OPAQUE_CAPTURE_DESCRIPTOR_DATA_CREATE_INFO_EXT);
1524*61046927SAndroid Build Coastguard Worker if (opaque_info)
1525*61046927SAndroid Build Coastguard Worker explicit_address = *((const uint64_t *)opaque_info->opaqueCaptureDescriptorData);
1526*61046927SAndroid Build Coastguard Worker }
1527*61046927SAndroid Build Coastguard Worker
1528*61046927SAndroid Build Coastguard Worker for (int i = 0; i < ANV_IMAGE_MEMORY_BINDING_END; i++) {
1529*61046927SAndroid Build Coastguard Worker struct anv_image_binding *b = &image->bindings[i];
1530*61046927SAndroid Build Coastguard Worker
1531*61046927SAndroid Build Coastguard Worker if (b->memory_range.size != 0) {
1532*61046927SAndroid Build Coastguard Worker assert(b->sparse_data.size == 0);
1533*61046927SAndroid Build Coastguard Worker
1534*61046927SAndroid Build Coastguard Worker /* From the spec, Custom Sparse Image Block Shapes section:
1535*61046927SAndroid Build Coastguard Worker * "... the size in bytes of the custom sparse image block shape
1536*61046927SAndroid Build Coastguard Worker * will be reported in VkMemoryRequirements::alignment."
1537*61046927SAndroid Build Coastguard Worker *
1538*61046927SAndroid Build Coastguard Worker * ISL should have set this for us, so just assert it here.
1539*61046927SAndroid Build Coastguard Worker */
1540*61046927SAndroid Build Coastguard Worker assert(b->memory_range.alignment == ANV_SPARSE_BLOCK_SIZE);
1541*61046927SAndroid Build Coastguard Worker assert(b->memory_range.size % ANV_SPARSE_BLOCK_SIZE == 0);
1542*61046927SAndroid Build Coastguard Worker
1543*61046927SAndroid Build Coastguard Worker result = anv_init_sparse_bindings(device,
1544*61046927SAndroid Build Coastguard Worker b->memory_range.size,
1545*61046927SAndroid Build Coastguard Worker &b->sparse_data,
1546*61046927SAndroid Build Coastguard Worker alloc_flags,
1547*61046927SAndroid Build Coastguard Worker explicit_address,
1548*61046927SAndroid Build Coastguard Worker &b->address);
1549*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS) {
1550*61046927SAndroid Build Coastguard Worker anv_image_finish_sparse_bindings(image);
1551*61046927SAndroid Build Coastguard Worker return result;
1552*61046927SAndroid Build Coastguard Worker }
1553*61046927SAndroid Build Coastguard Worker }
1554*61046927SAndroid Build Coastguard Worker }
1555*61046927SAndroid Build Coastguard Worker
1556*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1557*61046927SAndroid Build Coastguard Worker }
1558*61046927SAndroid Build Coastguard Worker
1559*61046927SAndroid Build Coastguard Worker VkResult
anv_image_init(struct anv_device * device,struct anv_image * image,const struct anv_image_create_info * create_info)1560*61046927SAndroid Build Coastguard Worker anv_image_init(struct anv_device *device, struct anv_image *image,
1561*61046927SAndroid Build Coastguard Worker const struct anv_image_create_info *create_info)
1562*61046927SAndroid Build Coastguard Worker {
1563*61046927SAndroid Build Coastguard Worker const VkImageCreateInfo *pCreateInfo = create_info->vk_info;
1564*61046927SAndroid Build Coastguard Worker const struct VkImageDrmFormatModifierExplicitCreateInfoEXT *mod_explicit_info = NULL;
1565*61046927SAndroid Build Coastguard Worker const struct isl_drm_modifier_info *isl_mod_info = NULL;
1566*61046927SAndroid Build Coastguard Worker VkResult r;
1567*61046927SAndroid Build Coastguard Worker
1568*61046927SAndroid Build Coastguard Worker vk_image_init(&device->vk, &image->vk, pCreateInfo);
1569*61046927SAndroid Build Coastguard Worker
1570*61046927SAndroid Build Coastguard Worker image->vk.usage = anv_image_create_usage(pCreateInfo, image->vk.usage);
1571*61046927SAndroid Build Coastguard Worker image->vk.stencil_usage =
1572*61046927SAndroid Build Coastguard Worker anv_image_create_usage(pCreateInfo, image->vk.stencil_usage);
1573*61046927SAndroid Build Coastguard Worker
1574*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_extra_usage_flags =
1575*61046927SAndroid Build Coastguard Worker create_info->isl_extra_usage_flags;
1576*61046927SAndroid Build Coastguard Worker if (pCreateInfo->tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT) {
1577*61046927SAndroid Build Coastguard Worker assert(!image->vk.wsi_legacy_scanout);
1578*61046927SAndroid Build Coastguard Worker mod_explicit_info =
1579*61046927SAndroid Build Coastguard Worker vk_find_struct_const(pCreateInfo->pNext,
1580*61046927SAndroid Build Coastguard Worker IMAGE_DRM_FORMAT_MODIFIER_EXPLICIT_CREATE_INFO_EXT);
1581*61046927SAndroid Build Coastguard Worker if (mod_explicit_info) {
1582*61046927SAndroid Build Coastguard Worker isl_mod_info = isl_drm_modifier_get_info(mod_explicit_info->drmFormatModifier);
1583*61046927SAndroid Build Coastguard Worker } else {
1584*61046927SAndroid Build Coastguard Worker const struct VkImageDrmFormatModifierListCreateInfoEXT *mod_list_info =
1585*61046927SAndroid Build Coastguard Worker vk_find_struct_const(pCreateInfo->pNext,
1586*61046927SAndroid Build Coastguard Worker IMAGE_DRM_FORMAT_MODIFIER_LIST_CREATE_INFO_EXT);
1587*61046927SAndroid Build Coastguard Worker isl_mod_info = choose_drm_format_mod(device->physical,
1588*61046927SAndroid Build Coastguard Worker mod_list_info->drmFormatModifierCount,
1589*61046927SAndroid Build Coastguard Worker mod_list_info->pDrmFormatModifiers);
1590*61046927SAndroid Build Coastguard Worker }
1591*61046927SAndroid Build Coastguard Worker
1592*61046927SAndroid Build Coastguard Worker assert(isl_mod_info);
1593*61046927SAndroid Build Coastguard Worker assert(image->vk.drm_format_mod == DRM_FORMAT_MOD_INVALID);
1594*61046927SAndroid Build Coastguard Worker image->vk.drm_format_mod = isl_mod_info->modifier;
1595*61046927SAndroid Build Coastguard Worker
1596*61046927SAndroid Build Coastguard Worker if (isl_drm_modifier_needs_display_layout(image->vk.drm_format_mod))
1597*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISPLAY_BIT;
1598*61046927SAndroid Build Coastguard Worker
1599*61046927SAndroid Build Coastguard Worker /* Disable compression on gen12+ if the selected/requested modifier
1600*61046927SAndroid Build Coastguard Worker * doesn't support it. Prior to that we can use a private binding for
1601*61046927SAndroid Build Coastguard Worker * the aux surface and it should be transparent to users.
1602*61046927SAndroid Build Coastguard Worker */
1603*61046927SAndroid Build Coastguard Worker if (device->info->ver >= 12 &&
1604*61046927SAndroid Build Coastguard Worker !isl_drm_modifier_has_aux(image->vk.drm_format_mod)) {
1605*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1606*61046927SAndroid Build Coastguard Worker }
1607*61046927SAndroid Build Coastguard Worker }
1608*61046927SAndroid Build Coastguard Worker
1609*61046927SAndroid Build Coastguard Worker for (int i = 0; i < ANV_IMAGE_MEMORY_BINDING_END; ++i) {
1610*61046927SAndroid Build Coastguard Worker image->bindings[i] = (struct anv_image_binding) {
1611*61046927SAndroid Build Coastguard Worker .memory_range = { .binding = i },
1612*61046927SAndroid Build Coastguard Worker };
1613*61046927SAndroid Build Coastguard Worker }
1614*61046927SAndroid Build Coastguard Worker
1615*61046927SAndroid Build Coastguard Worker /* In case of AHardwareBuffer import, we don't know the layout yet */
1616*61046927SAndroid Build Coastguard Worker if (image->vk.external_handle_types &
1617*61046927SAndroid Build Coastguard Worker VK_EXTERNAL_MEMORY_HANDLE_TYPE_ANDROID_HARDWARE_BUFFER_BIT_ANDROID) {
1618*61046927SAndroid Build Coastguard Worker image->from_ahb = true;
1619*61046927SAndroid Build Coastguard Worker #if DETECT_OS_ANDROID
1620*61046927SAndroid Build Coastguard Worker image->vk.ahb_format = anv_ahb_format_for_vk_format(image->vk.format);
1621*61046927SAndroid Build Coastguard Worker #endif
1622*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1623*61046927SAndroid Build Coastguard Worker }
1624*61046927SAndroid Build Coastguard Worker
1625*61046927SAndroid Build Coastguard Worker image->n_planes = anv_get_format_planes(image->vk.format);
1626*61046927SAndroid Build Coastguard Worker
1627*61046927SAndroid Build Coastguard Worker #ifdef VK_USE_PLATFORM_ANDROID_KHR
1628*61046927SAndroid Build Coastguard Worker /* In the case of gralloc-backed swap chain image, we don't know the
1629*61046927SAndroid Build Coastguard Worker * layout yet.
1630*61046927SAndroid Build Coastguard Worker */
1631*61046927SAndroid Build Coastguard Worker if (vk_find_struct_const(pCreateInfo->pNext,
1632*61046927SAndroid Build Coastguard Worker IMAGE_SWAPCHAIN_CREATE_INFO_KHR) != NULL)
1633*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1634*61046927SAndroid Build Coastguard Worker #endif
1635*61046927SAndroid Build Coastguard Worker
1636*61046927SAndroid Build Coastguard Worker image->from_wsi =
1637*61046927SAndroid Build Coastguard Worker vk_find_struct_const(pCreateInfo->pNext, WSI_IMAGE_CREATE_INFO_MESA) != NULL;
1638*61046927SAndroid Build Coastguard Worker
1639*61046927SAndroid Build Coastguard Worker /* The Vulkan 1.2.165 glossary says:
1640*61046927SAndroid Build Coastguard Worker *
1641*61046927SAndroid Build Coastguard Worker * A disjoint image consists of multiple disjoint planes, and is created
1642*61046927SAndroid Build Coastguard Worker * with the VK_IMAGE_CREATE_DISJOINT_BIT bit set.
1643*61046927SAndroid Build Coastguard Worker */
1644*61046927SAndroid Build Coastguard Worker image->disjoint = image->n_planes > 1 &&
1645*61046927SAndroid Build Coastguard Worker (pCreateInfo->flags & VK_IMAGE_CREATE_DISJOINT_BIT);
1646*61046927SAndroid Build Coastguard Worker
1647*61046927SAndroid Build Coastguard Worker if (anv_is_format_emulated(device->physical, pCreateInfo->format)) {
1648*61046927SAndroid Build Coastguard Worker assert(image->n_planes == 1 &&
1649*61046927SAndroid Build Coastguard Worker vk_format_is_compressed(image->vk.format));
1650*61046927SAndroid Build Coastguard Worker assert(!(image->vk.create_flags & VK_IMAGE_CREATE_SPARSE_RESIDENCY_BIT));
1651*61046927SAndroid Build Coastguard Worker
1652*61046927SAndroid Build Coastguard Worker image->emu_plane_format =
1653*61046927SAndroid Build Coastguard Worker anv_get_emulation_format(device->physical, image->vk.format);
1654*61046927SAndroid Build Coastguard Worker
1655*61046927SAndroid Build Coastguard Worker /* for fetching the raw copmressed data and storing the decompressed
1656*61046927SAndroid Build Coastguard Worker * data
1657*61046927SAndroid Build Coastguard Worker */
1658*61046927SAndroid Build Coastguard Worker image->vk.create_flags |=
1659*61046927SAndroid Build Coastguard Worker VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT |
1660*61046927SAndroid Build Coastguard Worker VK_IMAGE_CREATE_BLOCK_TEXEL_VIEW_COMPATIBLE_BIT;
1661*61046927SAndroid Build Coastguard Worker if (image->vk.image_type == VK_IMAGE_TYPE_3D)
1662*61046927SAndroid Build Coastguard Worker image->vk.create_flags |= VK_IMAGE_CREATE_2D_VIEW_COMPATIBLE_BIT_EXT;
1663*61046927SAndroid Build Coastguard Worker image->vk.usage |=
1664*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_STORAGE_BIT;
1665*61046927SAndroid Build Coastguard Worker
1666*61046927SAndroid Build Coastguard Worker /* TODO: enable compression on emulation plane */
1667*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1668*61046927SAndroid Build Coastguard Worker }
1669*61046927SAndroid Build Coastguard Worker
1670*61046927SAndroid Build Coastguard Worker /* Disable aux if image supports export without modifiers. */
1671*61046927SAndroid Build Coastguard Worker if (image->vk.external_handle_types != 0 &&
1672*61046927SAndroid Build Coastguard Worker image->vk.tiling != VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT)
1673*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1674*61046927SAndroid Build Coastguard Worker
1675*61046927SAndroid Build Coastguard Worker if (device->queue_count > 1) {
1676*61046927SAndroid Build Coastguard Worker /* Notify ISL that the app may access this image from different engines.
1677*61046927SAndroid Build Coastguard Worker * Note that parallel access to the surface will occur regardless of the
1678*61046927SAndroid Build Coastguard Worker * sharing mode.
1679*61046927SAndroid Build Coastguard Worker */
1680*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_MULTI_ENGINE_PAR_BIT;
1681*61046927SAndroid Build Coastguard Worker
1682*61046927SAndroid Build Coastguard Worker /* If the resource is created with the CONCURRENT sharing mode, we can't
1683*61046927SAndroid Build Coastguard Worker * support compression because we aren't allowed barriers in order to
1684*61046927SAndroid Build Coastguard Worker * construct the main surface data with FULL_RESOLVE/PARTIAL_RESOLVE.
1685*61046927SAndroid Build Coastguard Worker */
1686*61046927SAndroid Build Coastguard Worker if (image->vk.sharing_mode == VK_SHARING_MODE_CONCURRENT)
1687*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1688*61046927SAndroid Build Coastguard Worker }
1689*61046927SAndroid Build Coastguard Worker
1690*61046927SAndroid Build Coastguard Worker /* Aux is pointless if it will never be used as an attachment. */
1691*61046927SAndroid Build Coastguard Worker if (vk_format_is_depth_or_stencil(image->vk.format) &&
1692*61046927SAndroid Build Coastguard Worker !(image->vk.usage & VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT))
1693*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1694*61046927SAndroid Build Coastguard Worker
1695*61046927SAndroid Build Coastguard Worker /* TODO: Adjust blorp for multi-LOD HiZ surface on Gen9. */
1696*61046927SAndroid Build Coastguard Worker if (vk_format_has_depth(image->vk.format) &&
1697*61046927SAndroid Build Coastguard Worker image->vk.mip_levels > 1 && device->info->ver == 9) {
1698*61046927SAndroid Build Coastguard Worker anv_perf_warn(VK_LOG_OBJS(&image->vk.base), "Enable multi-LOD HiZ");
1699*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1700*61046927SAndroid Build Coastguard Worker }
1701*61046927SAndroid Build Coastguard Worker
1702*61046927SAndroid Build Coastguard Worker /* Mark WSI images with the right surf usage. */
1703*61046927SAndroid Build Coastguard Worker if (image->from_wsi)
1704*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISPLAY_BIT;
1705*61046927SAndroid Build Coastguard Worker
1706*61046927SAndroid Build Coastguard Worker const isl_tiling_flags_t isl_tiling_flags =
1707*61046927SAndroid Build Coastguard Worker choose_isl_tiling_flags(device->info, create_info, isl_mod_info,
1708*61046927SAndroid Build Coastguard Worker image->vk.wsi_legacy_scanout);
1709*61046927SAndroid Build Coastguard Worker
1710*61046927SAndroid Build Coastguard Worker const VkImageFormatListCreateInfo *fmt_list =
1711*61046927SAndroid Build Coastguard Worker vk_find_struct_const(pCreateInfo->pNext,
1712*61046927SAndroid Build Coastguard Worker IMAGE_FORMAT_LIST_CREATE_INFO);
1713*61046927SAndroid Build Coastguard Worker
1714*61046927SAndroid Build Coastguard Worker if ((image->vk.aspects & VK_IMAGE_ASPECT_ANY_COLOR_BIT_ANV) &&
1715*61046927SAndroid Build Coastguard Worker image->vk.samples == 1) {
1716*61046927SAndroid Build Coastguard Worker if (image->n_planes != 1) {
1717*61046927SAndroid Build Coastguard Worker /* Multiplanar images seem to hit a sampler bug with CCS and R16G16
1718*61046927SAndroid Build Coastguard Worker * format. (Putting the clear state a page/4096bytes further fixes
1719*61046927SAndroid Build Coastguard Worker * the issue).
1720*61046927SAndroid Build Coastguard Worker */
1721*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1722*61046927SAndroid Build Coastguard Worker }
1723*61046927SAndroid Build Coastguard Worker
1724*61046927SAndroid Build Coastguard Worker if ((image->vk.create_flags & VK_IMAGE_CREATE_ALIAS_BIT) &&
1725*61046927SAndroid Build Coastguard Worker !image->from_wsi) {
1726*61046927SAndroid Build Coastguard Worker /* The image may alias a plane of a multiplanar image. Above we ban
1727*61046927SAndroid Build Coastguard Worker * CCS on multiplanar images.
1728*61046927SAndroid Build Coastguard Worker *
1729*61046927SAndroid Build Coastguard Worker * We must also reject aliasing of any image that uses
1730*61046927SAndroid Build Coastguard Worker * ANV_IMAGE_MEMORY_BINDING_PRIVATE. Since we're already rejecting
1731*61046927SAndroid Build Coastguard Worker * all aliasing here, there's no need to further analyze if the image
1732*61046927SAndroid Build Coastguard Worker * needs a private binding.
1733*61046927SAndroid Build Coastguard Worker */
1734*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1735*61046927SAndroid Build Coastguard Worker }
1736*61046927SAndroid Build Coastguard Worker
1737*61046927SAndroid Build Coastguard Worker if (device->info->ver >= 12 &&
1738*61046927SAndroid Build Coastguard Worker !anv_formats_ccs_e_compatible(device->info, image->vk.create_flags,
1739*61046927SAndroid Build Coastguard Worker image->vk.format, image->vk.tiling,
1740*61046927SAndroid Build Coastguard Worker image->vk.usage, fmt_list)) {
1741*61046927SAndroid Build Coastguard Worker /* CCS_E is the only aux-mode supported for single sampled color
1742*61046927SAndroid Build Coastguard Worker * surfaces on gfx12+. If we can't support it, we should configure
1743*61046927SAndroid Build Coastguard Worker * the main surface without aux support.
1744*61046927SAndroid Build Coastguard Worker */
1745*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags |= ISL_SURF_USAGE_DISABLE_AUX_BIT;
1746*61046927SAndroid Build Coastguard Worker }
1747*61046927SAndroid Build Coastguard Worker }
1748*61046927SAndroid Build Coastguard Worker
1749*61046927SAndroid Build Coastguard Worker if (mod_explicit_info) {
1750*61046927SAndroid Build Coastguard Worker r = add_all_surfaces_explicit_layout(device, image, fmt_list,
1751*61046927SAndroid Build Coastguard Worker mod_explicit_info, isl_tiling_flags,
1752*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags);
1753*61046927SAndroid Build Coastguard Worker } else {
1754*61046927SAndroid Build Coastguard Worker r = add_all_surfaces_implicit_layout(device, image, fmt_list, create_info->stride,
1755*61046927SAndroid Build Coastguard Worker isl_tiling_flags,
1756*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags);
1757*61046927SAndroid Build Coastguard Worker }
1758*61046927SAndroid Build Coastguard Worker
1759*61046927SAndroid Build Coastguard Worker if (r != VK_SUCCESS)
1760*61046927SAndroid Build Coastguard Worker goto fail;
1761*61046927SAndroid Build Coastguard Worker
1762*61046927SAndroid Build Coastguard Worker if (image->emu_plane_format != VK_FORMAT_UNDEFINED) {
1763*61046927SAndroid Build Coastguard Worker const struct intel_device_info *devinfo = device->info;
1764*61046927SAndroid Build Coastguard Worker const uint32_t plane = image->n_planes;
1765*61046927SAndroid Build Coastguard Worker const struct anv_format_plane plane_format = anv_get_format_plane(
1766*61046927SAndroid Build Coastguard Worker devinfo, image->emu_plane_format, 0, image->vk.tiling);
1767*61046927SAndroid Build Coastguard Worker
1768*61046927SAndroid Build Coastguard Worker isl_surf_usage_flags_t isl_usage = anv_image_choose_isl_surf_usage(
1769*61046927SAndroid Build Coastguard Worker device->physical, image->vk.create_flags, image->vk.usage,
1770*61046927SAndroid Build Coastguard Worker isl_extra_usage_flags, VK_IMAGE_ASPECT_COLOR_BIT,
1771*61046927SAndroid Build Coastguard Worker image->vk.compr_flags);
1772*61046927SAndroid Build Coastguard Worker
1773*61046927SAndroid Build Coastguard Worker r = add_primary_surface(device, image, plane, plane_format,
1774*61046927SAndroid Build Coastguard Worker ANV_OFFSET_IMPLICIT, 0,
1775*61046927SAndroid Build Coastguard Worker isl_tiling_flags, isl_usage);
1776*61046927SAndroid Build Coastguard Worker if (r != VK_SUCCESS)
1777*61046927SAndroid Build Coastguard Worker goto fail;
1778*61046927SAndroid Build Coastguard Worker }
1779*61046927SAndroid Build Coastguard Worker
1780*61046927SAndroid Build Coastguard Worker const VkVideoProfileListInfoKHR *video_profile =
1781*61046927SAndroid Build Coastguard Worker vk_find_struct_const(pCreateInfo->pNext,
1782*61046927SAndroid Build Coastguard Worker VIDEO_PROFILE_LIST_INFO_KHR);
1783*61046927SAndroid Build Coastguard Worker if (video_profile) {
1784*61046927SAndroid Build Coastguard Worker r = add_video_buffers(device, image, video_profile);
1785*61046927SAndroid Build Coastguard Worker if (r != VK_SUCCESS)
1786*61046927SAndroid Build Coastguard Worker goto fail;
1787*61046927SAndroid Build Coastguard Worker }
1788*61046927SAndroid Build Coastguard Worker
1789*61046927SAndroid Build Coastguard Worker if (!create_info->no_private_binding_alloc) {
1790*61046927SAndroid Build Coastguard Worker r = alloc_private_binding(device, image, pCreateInfo);
1791*61046927SAndroid Build Coastguard Worker if (r != VK_SUCCESS)
1792*61046927SAndroid Build Coastguard Worker goto fail;
1793*61046927SAndroid Build Coastguard Worker }
1794*61046927SAndroid Build Coastguard Worker
1795*61046927SAndroid Build Coastguard Worker check_memory_bindings(device, image);
1796*61046927SAndroid Build Coastguard Worker
1797*61046927SAndroid Build Coastguard Worker r = check_drm_format_mod(device, image);
1798*61046927SAndroid Build Coastguard Worker if (r != VK_SUCCESS)
1799*61046927SAndroid Build Coastguard Worker goto fail;
1800*61046927SAndroid Build Coastguard Worker
1801*61046927SAndroid Build Coastguard Worker /* Once we have all the bindings, determine whether we can do non 0 fast
1802*61046927SAndroid Build Coastguard Worker * clears for each plane.
1803*61046927SAndroid Build Coastguard Worker */
1804*61046927SAndroid Build Coastguard Worker for (uint32_t p = 0; p < image->n_planes; p++) {
1805*61046927SAndroid Build Coastguard Worker image->planes[p].can_non_zero_fast_clear =
1806*61046927SAndroid Build Coastguard Worker can_fast_clear_with_non_zero_color(device->info, image, p, fmt_list);
1807*61046927SAndroid Build Coastguard Worker }
1808*61046927SAndroid Build Coastguard Worker
1809*61046927SAndroid Build Coastguard Worker if (anv_image_is_sparse(image)) {
1810*61046927SAndroid Build Coastguard Worker r = anv_image_init_sparse_bindings(image, create_info);
1811*61046927SAndroid Build Coastguard Worker if (r != VK_SUCCESS)
1812*61046927SAndroid Build Coastguard Worker goto fail;
1813*61046927SAndroid Build Coastguard Worker }
1814*61046927SAndroid Build Coastguard Worker
1815*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
1816*61046927SAndroid Build Coastguard Worker
1817*61046927SAndroid Build Coastguard Worker fail:
1818*61046927SAndroid Build Coastguard Worker vk_image_finish(&image->vk);
1819*61046927SAndroid Build Coastguard Worker return r;
1820*61046927SAndroid Build Coastguard Worker }
1821*61046927SAndroid Build Coastguard Worker
1822*61046927SAndroid Build Coastguard Worker void
anv_image_finish(struct anv_image * image)1823*61046927SAndroid Build Coastguard Worker anv_image_finish(struct anv_image *image)
1824*61046927SAndroid Build Coastguard Worker {
1825*61046927SAndroid Build Coastguard Worker struct anv_device *device =
1826*61046927SAndroid Build Coastguard Worker container_of(image->vk.base.device, struct anv_device, vk);
1827*61046927SAndroid Build Coastguard Worker
1828*61046927SAndroid Build Coastguard Worker if (anv_image_is_sparse(image))
1829*61046927SAndroid Build Coastguard Worker anv_image_finish_sparse_bindings(image);
1830*61046927SAndroid Build Coastguard Worker
1831*61046927SAndroid Build Coastguard Worker /* Unmap a CCS so that if the bound region of the image is rebound to
1832*61046927SAndroid Build Coastguard Worker * another image, the AUX tables will be cleared to allow for a new
1833*61046927SAndroid Build Coastguard Worker * mapping.
1834*61046927SAndroid Build Coastguard Worker */
1835*61046927SAndroid Build Coastguard Worker for (int p = 0; p < image->n_planes; ++p) {
1836*61046927SAndroid Build Coastguard Worker if (image->planes[p].aux_tt.mapped) {
1837*61046927SAndroid Build Coastguard Worker intel_aux_map_del_mapping(device->aux_map_ctx,
1838*61046927SAndroid Build Coastguard Worker image->planes[p].aux_tt.addr,
1839*61046927SAndroid Build Coastguard Worker image->planes[p].aux_tt.size);
1840*61046927SAndroid Build Coastguard Worker }
1841*61046927SAndroid Build Coastguard Worker }
1842*61046927SAndroid Build Coastguard Worker
1843*61046927SAndroid Build Coastguard Worker if (image->from_gralloc) {
1844*61046927SAndroid Build Coastguard Worker assert(!image->disjoint);
1845*61046927SAndroid Build Coastguard Worker assert(image->n_planes == 1);
1846*61046927SAndroid Build Coastguard Worker assert(image->planes[0].primary_surface.memory_range.binding ==
1847*61046927SAndroid Build Coastguard Worker ANV_IMAGE_MEMORY_BINDING_MAIN);
1848*61046927SAndroid Build Coastguard Worker assert(image->bindings[ANV_IMAGE_MEMORY_BINDING_MAIN].address.bo != NULL);
1849*61046927SAndroid Build Coastguard Worker anv_device_release_bo(device, image->bindings[ANV_IMAGE_MEMORY_BINDING_MAIN].address.bo);
1850*61046927SAndroid Build Coastguard Worker }
1851*61046927SAndroid Build Coastguard Worker
1852*61046927SAndroid Build Coastguard Worker struct anv_bo *private_bo = image->bindings[ANV_IMAGE_MEMORY_BINDING_PRIVATE].address.bo;
1853*61046927SAndroid Build Coastguard Worker if (private_bo) {
1854*61046927SAndroid Build Coastguard Worker pthread_mutex_lock(&device->mutex);
1855*61046927SAndroid Build Coastguard Worker list_del(&image->link);
1856*61046927SAndroid Build Coastguard Worker pthread_mutex_unlock(&device->mutex);
1857*61046927SAndroid Build Coastguard Worker anv_device_release_bo(device, private_bo);
1858*61046927SAndroid Build Coastguard Worker }
1859*61046927SAndroid Build Coastguard Worker
1860*61046927SAndroid Build Coastguard Worker vk_image_finish(&image->vk);
1861*61046927SAndroid Build Coastguard Worker }
1862*61046927SAndroid Build Coastguard Worker
1863*61046927SAndroid Build Coastguard Worker static struct anv_image *
anv_swapchain_get_image(VkSwapchainKHR swapchain,uint32_t index)1864*61046927SAndroid Build Coastguard Worker anv_swapchain_get_image(VkSwapchainKHR swapchain,
1865*61046927SAndroid Build Coastguard Worker uint32_t index)
1866*61046927SAndroid Build Coastguard Worker {
1867*61046927SAndroid Build Coastguard Worker VkImage image = wsi_common_get_image(swapchain, index);
1868*61046927SAndroid Build Coastguard Worker return anv_image_from_handle(image);
1869*61046927SAndroid Build Coastguard Worker }
1870*61046927SAndroid Build Coastguard Worker
1871*61046927SAndroid Build Coastguard Worker static VkResult
anv_image_init_from_create_info(struct anv_device * device,struct anv_image * image,const VkImageCreateInfo * pCreateInfo,bool no_private_binding_alloc)1872*61046927SAndroid Build Coastguard Worker anv_image_init_from_create_info(struct anv_device *device,
1873*61046927SAndroid Build Coastguard Worker struct anv_image *image,
1874*61046927SAndroid Build Coastguard Worker const VkImageCreateInfo *pCreateInfo,
1875*61046927SAndroid Build Coastguard Worker bool no_private_binding_alloc)
1876*61046927SAndroid Build Coastguard Worker {
1877*61046927SAndroid Build Coastguard Worker if (pCreateInfo->flags & VK_IMAGE_CREATE_SPARSE_RESIDENCY_BIT) {
1878*61046927SAndroid Build Coastguard Worker VkResult result =
1879*61046927SAndroid Build Coastguard Worker anv_sparse_image_check_support(device->physical,
1880*61046927SAndroid Build Coastguard Worker pCreateInfo->flags,
1881*61046927SAndroid Build Coastguard Worker pCreateInfo->tiling,
1882*61046927SAndroid Build Coastguard Worker pCreateInfo->samples,
1883*61046927SAndroid Build Coastguard Worker pCreateInfo->imageType,
1884*61046927SAndroid Build Coastguard Worker pCreateInfo->format);
1885*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
1886*61046927SAndroid Build Coastguard Worker return result;
1887*61046927SAndroid Build Coastguard Worker }
1888*61046927SAndroid Build Coastguard Worker
1889*61046927SAndroid Build Coastguard Worker const VkNativeBufferANDROID *gralloc_info =
1890*61046927SAndroid Build Coastguard Worker vk_find_struct_const(pCreateInfo->pNext, NATIVE_BUFFER_ANDROID);
1891*61046927SAndroid Build Coastguard Worker if (gralloc_info)
1892*61046927SAndroid Build Coastguard Worker return anv_image_init_from_gralloc(device, image, pCreateInfo,
1893*61046927SAndroid Build Coastguard Worker gralloc_info);
1894*61046927SAndroid Build Coastguard Worker
1895*61046927SAndroid Build Coastguard Worker struct anv_image_create_info create_info = {
1896*61046927SAndroid Build Coastguard Worker .vk_info = pCreateInfo,
1897*61046927SAndroid Build Coastguard Worker .no_private_binding_alloc = no_private_binding_alloc,
1898*61046927SAndroid Build Coastguard Worker };
1899*61046927SAndroid Build Coastguard Worker
1900*61046927SAndroid Build Coastguard Worker return anv_image_init(device, image, &create_info);
1901*61046927SAndroid Build Coastguard Worker }
1902*61046927SAndroid Build Coastguard Worker
anv_CreateImage(VkDevice _device,const VkImageCreateInfo * pCreateInfo,const VkAllocationCallbacks * pAllocator,VkImage * pImage)1903*61046927SAndroid Build Coastguard Worker VkResult anv_CreateImage(
1904*61046927SAndroid Build Coastguard Worker VkDevice _device,
1905*61046927SAndroid Build Coastguard Worker const VkImageCreateInfo* pCreateInfo,
1906*61046927SAndroid Build Coastguard Worker const VkAllocationCallbacks* pAllocator,
1907*61046927SAndroid Build Coastguard Worker VkImage* pImage)
1908*61046927SAndroid Build Coastguard Worker {
1909*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
1910*61046927SAndroid Build Coastguard Worker
1911*61046927SAndroid Build Coastguard Worker if ((device->physical->sparse_type == ANV_SPARSE_TYPE_NOT_SUPPORTED) &&
1912*61046927SAndroid Build Coastguard Worker INTEL_DEBUG(DEBUG_SPARSE) &&
1913*61046927SAndroid Build Coastguard Worker pCreateInfo->flags & (VK_IMAGE_CREATE_SPARSE_BINDING_BIT |
1914*61046927SAndroid Build Coastguard Worker VK_IMAGE_CREATE_SPARSE_RESIDENCY_BIT |
1915*61046927SAndroid Build Coastguard Worker VK_IMAGE_CREATE_SPARSE_ALIASED_BIT))
1916*61046927SAndroid Build Coastguard Worker fprintf(stderr, "=== %s %s:%d flags:0x%08x\n", __func__, __FILE__,
1917*61046927SAndroid Build Coastguard Worker __LINE__, pCreateInfo->flags);
1918*61046927SAndroid Build Coastguard Worker
1919*61046927SAndroid Build Coastguard Worker #ifndef VK_USE_PLATFORM_ANDROID_KHR
1920*61046927SAndroid Build Coastguard Worker /* Skip the WSI common swapchain creation here on Android. Similar to ahw,
1921*61046927SAndroid Build Coastguard Worker * this case is handled by a partial image init and then resolved when the
1922*61046927SAndroid Build Coastguard Worker * image is bound and gralloc info is passed.
1923*61046927SAndroid Build Coastguard Worker */
1924*61046927SAndroid Build Coastguard Worker const VkImageSwapchainCreateInfoKHR *swapchain_info =
1925*61046927SAndroid Build Coastguard Worker vk_find_struct_const(pCreateInfo->pNext, IMAGE_SWAPCHAIN_CREATE_INFO_KHR);
1926*61046927SAndroid Build Coastguard Worker if (swapchain_info && swapchain_info->swapchain != VK_NULL_HANDLE) {
1927*61046927SAndroid Build Coastguard Worker return wsi_common_create_swapchain_image(&device->physical->wsi_device,
1928*61046927SAndroid Build Coastguard Worker pCreateInfo,
1929*61046927SAndroid Build Coastguard Worker swapchain_info->swapchain,
1930*61046927SAndroid Build Coastguard Worker pImage);
1931*61046927SAndroid Build Coastguard Worker }
1932*61046927SAndroid Build Coastguard Worker #endif
1933*61046927SAndroid Build Coastguard Worker
1934*61046927SAndroid Build Coastguard Worker struct anv_image *image =
1935*61046927SAndroid Build Coastguard Worker vk_object_zalloc(&device->vk, pAllocator, sizeof(*image),
1936*61046927SAndroid Build Coastguard Worker VK_OBJECT_TYPE_IMAGE);
1937*61046927SAndroid Build Coastguard Worker if (!image)
1938*61046927SAndroid Build Coastguard Worker return vk_error(device, VK_ERROR_OUT_OF_HOST_MEMORY);
1939*61046927SAndroid Build Coastguard Worker
1940*61046927SAndroid Build Coastguard Worker VkResult result = anv_image_init_from_create_info(device, image,
1941*61046927SAndroid Build Coastguard Worker pCreateInfo,
1942*61046927SAndroid Build Coastguard Worker false);
1943*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS) {
1944*61046927SAndroid Build Coastguard Worker vk_object_free(&device->vk, pAllocator, image);
1945*61046927SAndroid Build Coastguard Worker return result;
1946*61046927SAndroid Build Coastguard Worker }
1947*61046927SAndroid Build Coastguard Worker
1948*61046927SAndroid Build Coastguard Worker ANV_RMV(image_create, device, false, image);
1949*61046927SAndroid Build Coastguard Worker
1950*61046927SAndroid Build Coastguard Worker *pImage = anv_image_to_handle(image);
1951*61046927SAndroid Build Coastguard Worker
1952*61046927SAndroid Build Coastguard Worker return result;
1953*61046927SAndroid Build Coastguard Worker }
1954*61046927SAndroid Build Coastguard Worker
1955*61046927SAndroid Build Coastguard Worker void
anv_DestroyImage(VkDevice _device,VkImage _image,const VkAllocationCallbacks * pAllocator)1956*61046927SAndroid Build Coastguard Worker anv_DestroyImage(VkDevice _device, VkImage _image,
1957*61046927SAndroid Build Coastguard Worker const VkAllocationCallbacks *pAllocator)
1958*61046927SAndroid Build Coastguard Worker {
1959*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
1960*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_image, image, _image);
1961*61046927SAndroid Build Coastguard Worker
1962*61046927SAndroid Build Coastguard Worker if (!image)
1963*61046927SAndroid Build Coastguard Worker return;
1964*61046927SAndroid Build Coastguard Worker
1965*61046927SAndroid Build Coastguard Worker ANV_RMV(image_destroy, device, image);
1966*61046927SAndroid Build Coastguard Worker
1967*61046927SAndroid Build Coastguard Worker assert(&device->vk == image->vk.base.device);
1968*61046927SAndroid Build Coastguard Worker anv_image_finish(image);
1969*61046927SAndroid Build Coastguard Worker
1970*61046927SAndroid Build Coastguard Worker vk_free2(&device->vk.alloc, pAllocator, image);
1971*61046927SAndroid Build Coastguard Worker }
1972*61046927SAndroid Build Coastguard Worker
1973*61046927SAndroid Build Coastguard Worker /* We are binding AHardwareBuffer. Get a description, resolve the
1974*61046927SAndroid Build Coastguard Worker * format and prepare anv_image properly.
1975*61046927SAndroid Build Coastguard Worker */
1976*61046927SAndroid Build Coastguard Worker static void
resolve_ahw_image(struct anv_device * device,struct anv_image * image,struct anv_device_memory * mem)1977*61046927SAndroid Build Coastguard Worker resolve_ahw_image(struct anv_device *device,
1978*61046927SAndroid Build Coastguard Worker struct anv_image *image,
1979*61046927SAndroid Build Coastguard Worker struct anv_device_memory *mem)
1980*61046927SAndroid Build Coastguard Worker {
1981*61046927SAndroid Build Coastguard Worker #if DETECT_OS_ANDROID && ANDROID_API_LEVEL >= 26
1982*61046927SAndroid Build Coastguard Worker assert(mem->vk.ahardware_buffer);
1983*61046927SAndroid Build Coastguard Worker AHardwareBuffer_Desc desc;
1984*61046927SAndroid Build Coastguard Worker AHardwareBuffer_describe(mem->vk.ahardware_buffer, &desc);
1985*61046927SAndroid Build Coastguard Worker VkResult result;
1986*61046927SAndroid Build Coastguard Worker
1987*61046927SAndroid Build Coastguard Worker /* Check tiling. */
1988*61046927SAndroid Build Coastguard Worker enum isl_tiling tiling;
1989*61046927SAndroid Build Coastguard Worker const native_handle_t *handle =
1990*61046927SAndroid Build Coastguard Worker AHardwareBuffer_getNativeHandle(mem->vk.ahardware_buffer);
1991*61046927SAndroid Build Coastguard Worker struct u_gralloc_buffer_handle gr_handle = {
1992*61046927SAndroid Build Coastguard Worker .handle = handle,
1993*61046927SAndroid Build Coastguard Worker .hal_format = desc.format,
1994*61046927SAndroid Build Coastguard Worker .pixel_stride = desc.stride,
1995*61046927SAndroid Build Coastguard Worker };
1996*61046927SAndroid Build Coastguard Worker result = anv_android_get_tiling(device, &gr_handle, &tiling);
1997*61046927SAndroid Build Coastguard Worker assert(result == VK_SUCCESS);
1998*61046927SAndroid Build Coastguard Worker isl_tiling_flags_t isl_tiling_flags = (1u << tiling);
1999*61046927SAndroid Build Coastguard Worker
2000*61046927SAndroid Build Coastguard Worker /* Check format. */
2001*61046927SAndroid Build Coastguard Worker VkFormat vk_format = vk_format_from_android(desc.format, desc.usage);
2002*61046927SAndroid Build Coastguard Worker assert(vk_format != VK_FORMAT_UNDEFINED);
2003*61046927SAndroid Build Coastguard Worker
2004*61046927SAndroid Build Coastguard Worker /* Now we are able to fill anv_image fields properly and create
2005*61046927SAndroid Build Coastguard Worker * isl_surface for it.
2006*61046927SAndroid Build Coastguard Worker */
2007*61046927SAndroid Build Coastguard Worker vk_image_set_format(&image->vk, vk_format);
2008*61046927SAndroid Build Coastguard Worker image->n_planes = anv_get_format_planes(image->vk.format);
2009*61046927SAndroid Build Coastguard Worker
2010*61046927SAndroid Build Coastguard Worker result = add_all_surfaces_implicit_layout(device, image, NULL, desc.stride,
2011*61046927SAndroid Build Coastguard Worker isl_tiling_flags,
2012*61046927SAndroid Build Coastguard Worker ISL_SURF_USAGE_DISABLE_AUX_BIT);
2013*61046927SAndroid Build Coastguard Worker assert(result == VK_SUCCESS);
2014*61046927SAndroid Build Coastguard Worker #endif
2015*61046927SAndroid Build Coastguard Worker }
2016*61046927SAndroid Build Coastguard Worker
2017*61046927SAndroid Build Coastguard Worker static void
resolve_anb_image(struct anv_device * device,struct anv_image * image,const VkNativeBufferANDROID * gralloc_info)2018*61046927SAndroid Build Coastguard Worker resolve_anb_image(struct anv_device *device,
2019*61046927SAndroid Build Coastguard Worker struct anv_image *image,
2020*61046927SAndroid Build Coastguard Worker const VkNativeBufferANDROID *gralloc_info)
2021*61046927SAndroid Build Coastguard Worker {
2022*61046927SAndroid Build Coastguard Worker #if DETECT_OS_ANDROID && ANDROID_API_LEVEL >= 29
2023*61046927SAndroid Build Coastguard Worker VkResult result;
2024*61046927SAndroid Build Coastguard Worker
2025*61046927SAndroid Build Coastguard Worker /* Check tiling. */
2026*61046927SAndroid Build Coastguard Worker enum isl_tiling tiling;
2027*61046927SAndroid Build Coastguard Worker struct u_gralloc_buffer_handle gr_handle = {
2028*61046927SAndroid Build Coastguard Worker .handle = gralloc_info->handle,
2029*61046927SAndroid Build Coastguard Worker .hal_format = gralloc_info->format,
2030*61046927SAndroid Build Coastguard Worker .pixel_stride = gralloc_info->stride,
2031*61046927SAndroid Build Coastguard Worker };
2032*61046927SAndroid Build Coastguard Worker result = anv_android_get_tiling(device, &gr_handle, &tiling);
2033*61046927SAndroid Build Coastguard Worker assert(result == VK_SUCCESS);
2034*61046927SAndroid Build Coastguard Worker
2035*61046927SAndroid Build Coastguard Worker isl_tiling_flags_t isl_tiling_flags = (1u << tiling);
2036*61046927SAndroid Build Coastguard Worker
2037*61046927SAndroid Build Coastguard Worker /* Now we are able to fill anv_image fields properly and create
2038*61046927SAndroid Build Coastguard Worker * isl_surface for it.
2039*61046927SAndroid Build Coastguard Worker */
2040*61046927SAndroid Build Coastguard Worker result = add_all_surfaces_implicit_layout(device, image, NULL, gralloc_info->stride,
2041*61046927SAndroid Build Coastguard Worker isl_tiling_flags,
2042*61046927SAndroid Build Coastguard Worker ISL_SURF_USAGE_DISABLE_AUX_BIT);
2043*61046927SAndroid Build Coastguard Worker assert(result == VK_SUCCESS);
2044*61046927SAndroid Build Coastguard Worker #endif
2045*61046927SAndroid Build Coastguard Worker }
2046*61046927SAndroid Build Coastguard Worker
2047*61046927SAndroid Build Coastguard Worker static bool
anv_image_is_pat_compressible(struct anv_device * device,struct anv_image * image)2048*61046927SAndroid Build Coastguard Worker anv_image_is_pat_compressible(struct anv_device *device, struct anv_image *image)
2049*61046927SAndroid Build Coastguard Worker {
2050*61046927SAndroid Build Coastguard Worker if (INTEL_DEBUG(DEBUG_NO_CCS))
2051*61046927SAndroid Build Coastguard Worker return false;
2052*61046927SAndroid Build Coastguard Worker
2053*61046927SAndroid Build Coastguard Worker if (device->info->ver < 20)
2054*61046927SAndroid Build Coastguard Worker return false;
2055*61046927SAndroid Build Coastguard Worker
2056*61046927SAndroid Build Coastguard Worker /*
2057*61046927SAndroid Build Coastguard Worker * Be aware that Vulkan spec requires that Images with some properties
2058*61046927SAndroid Build Coastguard Worker * always returns the same memory types, so this function also needs to
2059*61046927SAndroid Build Coastguard Worker * have the same return for the same set of properties.
2060*61046927SAndroid Build Coastguard Worker *
2061*61046927SAndroid Build Coastguard Worker * For images created with a color format, the memoryTypeBits member is
2062*61046927SAndroid Build Coastguard Worker * identical for all VkImage objects created with the same combination
2063*61046927SAndroid Build Coastguard Worker * of values for the tiling member, the
2064*61046927SAndroid Build Coastguard Worker * VK_IMAGE_CREATE_SPARSE_BINDING_BIT bit of the flags member, the
2065*61046927SAndroid Build Coastguard Worker * VK_IMAGE_CREATE_SPLIT_INSTANCE_BIND_REGIONS_BIT bit of the flags
2066*61046927SAndroid Build Coastguard Worker * member, handleTypes member of VkExternalMemoryImageCreateInfo, and
2067*61046927SAndroid Build Coastguard Worker * the VK_IMAGE_USAGE_TRANSIENT_ATTACHMENT_BIT of the usage member in
2068*61046927SAndroid Build Coastguard Worker * the VkImageCreateInfo structure passed to vkCreateImage.
2069*61046927SAndroid Build Coastguard Worker *
2070*61046927SAndroid Build Coastguard Worker * For images created with a depth/stencil format, the memoryTypeBits
2071*61046927SAndroid Build Coastguard Worker * member is identical for all VkImage objects created with the same
2072*61046927SAndroid Build Coastguard Worker * combination of values for the format member, the tiling member, the
2073*61046927SAndroid Build Coastguard Worker * VK_IMAGE_CREATE_SPARSE_BINDING_BIT bit of the flags member, the
2074*61046927SAndroid Build Coastguard Worker * VK_IMAGE_CREATE_SPLIT_INSTANCE_BIND_REGIONS_BIT bit of the flags
2075*61046927SAndroid Build Coastguard Worker * member, handleTypes member of VkExternalMemoryImageCreateInfo, and
2076*61046927SAndroid Build Coastguard Worker * the VK_IMAGE_USAGE_TRANSIENT_ATTACHMENT_BIT of the usage member in
2077*61046927SAndroid Build Coastguard Worker * the VkImageCreateInfo structure passed to vkCreateImage.
2078*61046927SAndroid Build Coastguard Worker */
2079*61046927SAndroid Build Coastguard Worker
2080*61046927SAndroid Build Coastguard Worker /* There are no compression-enabled modifiers on Xe2, and all legacy
2081*61046927SAndroid Build Coastguard Worker * modifiers are not defined with compression. We simply disable
2082*61046927SAndroid Build Coastguard Worker * compression on all modifiers.
2083*61046927SAndroid Build Coastguard Worker *
2084*61046927SAndroid Build Coastguard Worker * We disable this in anv_AllocateMemory() as well.
2085*61046927SAndroid Build Coastguard Worker */
2086*61046927SAndroid Build Coastguard Worker if (image->vk.tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT)
2087*61046927SAndroid Build Coastguard Worker return false;
2088*61046927SAndroid Build Coastguard Worker
2089*61046927SAndroid Build Coastguard Worker return true;
2090*61046927SAndroid Build Coastguard Worker }
2091*61046927SAndroid Build Coastguard Worker
2092*61046927SAndroid Build Coastguard Worker void
anv_image_get_memory_requirements(struct anv_device * device,struct anv_image * image,VkImageAspectFlags aspects,VkMemoryRequirements2 * pMemoryRequirements)2093*61046927SAndroid Build Coastguard Worker anv_image_get_memory_requirements(struct anv_device *device,
2094*61046927SAndroid Build Coastguard Worker struct anv_image *image,
2095*61046927SAndroid Build Coastguard Worker VkImageAspectFlags aspects,
2096*61046927SAndroid Build Coastguard Worker VkMemoryRequirements2 *pMemoryRequirements)
2097*61046927SAndroid Build Coastguard Worker {
2098*61046927SAndroid Build Coastguard Worker /* The Vulkan spec (git aaed022) says:
2099*61046927SAndroid Build Coastguard Worker *
2100*61046927SAndroid Build Coastguard Worker * memoryTypeBits is a bitfield and contains one bit set for every
2101*61046927SAndroid Build Coastguard Worker * supported memory type for the resource. The bit `1<<i` is set if and
2102*61046927SAndroid Build Coastguard Worker * only if the memory type `i` in the VkPhysicalDeviceMemoryProperties
2103*61046927SAndroid Build Coastguard Worker * structure for the physical device is supported.
2104*61046927SAndroid Build Coastguard Worker */
2105*61046927SAndroid Build Coastguard Worker uint32_t memory_types;
2106*61046927SAndroid Build Coastguard Worker
2107*61046927SAndroid Build Coastguard Worker if (image->vk.create_flags & VK_IMAGE_CREATE_PROTECTED_BIT) {
2108*61046927SAndroid Build Coastguard Worker memory_types = device->physical->memory.protected_mem_types;
2109*61046927SAndroid Build Coastguard Worker } else {
2110*61046927SAndroid Build Coastguard Worker memory_types = device->physical->memory.default_buffer_mem_types;
2111*61046927SAndroid Build Coastguard Worker if (anv_image_is_pat_compressible(device, image))
2112*61046927SAndroid Build Coastguard Worker memory_types |= device->physical->memory.compressed_mem_types;
2113*61046927SAndroid Build Coastguard Worker }
2114*61046927SAndroid Build Coastguard Worker
2115*61046927SAndroid Build Coastguard Worker vk_foreach_struct(ext, pMemoryRequirements->pNext) {
2116*61046927SAndroid Build Coastguard Worker switch (ext->sType) {
2117*61046927SAndroid Build Coastguard Worker case VK_STRUCTURE_TYPE_MEMORY_DEDICATED_REQUIREMENTS: {
2118*61046927SAndroid Build Coastguard Worker VkMemoryDedicatedRequirements *requirements = (void *)ext;
2119*61046927SAndroid Build Coastguard Worker if (image->vk.wsi_legacy_scanout ||
2120*61046927SAndroid Build Coastguard Worker image->from_ahb ||
2121*61046927SAndroid Build Coastguard Worker (isl_drm_modifier_has_aux(image->vk.drm_format_mod) &&
2122*61046927SAndroid Build Coastguard Worker anv_image_uses_aux_map(device, image))) {
2123*61046927SAndroid Build Coastguard Worker /* If we need to set the tiling for external consumers or the
2124*61046927SAndroid Build Coastguard Worker * modifier involves AUX tables, we need a dedicated allocation.
2125*61046927SAndroid Build Coastguard Worker *
2126*61046927SAndroid Build Coastguard Worker * See also anv_AllocateMemory.
2127*61046927SAndroid Build Coastguard Worker */
2128*61046927SAndroid Build Coastguard Worker requirements->prefersDedicatedAllocation = true;
2129*61046927SAndroid Build Coastguard Worker requirements->requiresDedicatedAllocation = true;
2130*61046927SAndroid Build Coastguard Worker } else {
2131*61046927SAndroid Build Coastguard Worker requirements->prefersDedicatedAllocation = false;
2132*61046927SAndroid Build Coastguard Worker requirements->requiresDedicatedAllocation = false;
2133*61046927SAndroid Build Coastguard Worker }
2134*61046927SAndroid Build Coastguard Worker break;
2135*61046927SAndroid Build Coastguard Worker }
2136*61046927SAndroid Build Coastguard Worker
2137*61046927SAndroid Build Coastguard Worker default:
2138*61046927SAndroid Build Coastguard Worker vk_debug_ignored_stype(ext->sType);
2139*61046927SAndroid Build Coastguard Worker break;
2140*61046927SAndroid Build Coastguard Worker }
2141*61046927SAndroid Build Coastguard Worker }
2142*61046927SAndroid Build Coastguard Worker
2143*61046927SAndroid Build Coastguard Worker /* If the image is disjoint, then we must return the memory requirements for
2144*61046927SAndroid Build Coastguard Worker * the single plane specified in VkImagePlaneMemoryRequirementsInfo. If
2145*61046927SAndroid Build Coastguard Worker * non-disjoint, then exactly one set of memory requirements exists for the
2146*61046927SAndroid Build Coastguard Worker * whole image.
2147*61046927SAndroid Build Coastguard Worker *
2148*61046927SAndroid Build Coastguard Worker * This is enforced by the Valid Usage for VkImageMemoryRequirementsInfo2,
2149*61046927SAndroid Build Coastguard Worker * which requires that the app provide VkImagePlaneMemoryRequirementsInfo if
2150*61046927SAndroid Build Coastguard Worker * and only if the image is disjoint (that is, multi-planar format and
2151*61046927SAndroid Build Coastguard Worker * VK_IMAGE_CREATE_DISJOINT_BIT).
2152*61046927SAndroid Build Coastguard Worker */
2153*61046927SAndroid Build Coastguard Worker const struct anv_image_binding *binding;
2154*61046927SAndroid Build Coastguard Worker if (image->disjoint) {
2155*61046927SAndroid Build Coastguard Worker assert(util_bitcount(aspects) == 1);
2156*61046927SAndroid Build Coastguard Worker assert(aspects & image->vk.aspects);
2157*61046927SAndroid Build Coastguard Worker binding = anv_image_aspect_to_binding(image, aspects);
2158*61046927SAndroid Build Coastguard Worker } else {
2159*61046927SAndroid Build Coastguard Worker assert(aspects == image->vk.aspects);
2160*61046927SAndroid Build Coastguard Worker binding = &image->bindings[ANV_IMAGE_MEMORY_BINDING_MAIN];
2161*61046927SAndroid Build Coastguard Worker }
2162*61046927SAndroid Build Coastguard Worker
2163*61046927SAndroid Build Coastguard Worker pMemoryRequirements->memoryRequirements = (VkMemoryRequirements) {
2164*61046927SAndroid Build Coastguard Worker .size = binding->memory_range.size,
2165*61046927SAndroid Build Coastguard Worker .alignment = binding->memory_range.alignment,
2166*61046927SAndroid Build Coastguard Worker .memoryTypeBits = memory_types,
2167*61046927SAndroid Build Coastguard Worker };
2168*61046927SAndroid Build Coastguard Worker }
2169*61046927SAndroid Build Coastguard Worker
anv_GetImageMemoryRequirements2(VkDevice _device,const VkImageMemoryRequirementsInfo2 * pInfo,VkMemoryRequirements2 * pMemoryRequirements)2170*61046927SAndroid Build Coastguard Worker void anv_GetImageMemoryRequirements2(
2171*61046927SAndroid Build Coastguard Worker VkDevice _device,
2172*61046927SAndroid Build Coastguard Worker const VkImageMemoryRequirementsInfo2* pInfo,
2173*61046927SAndroid Build Coastguard Worker VkMemoryRequirements2* pMemoryRequirements)
2174*61046927SAndroid Build Coastguard Worker {
2175*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
2176*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_image, image, pInfo->image);
2177*61046927SAndroid Build Coastguard Worker
2178*61046927SAndroid Build Coastguard Worker VkImageAspectFlags aspects = image->vk.aspects;
2179*61046927SAndroid Build Coastguard Worker
2180*61046927SAndroid Build Coastguard Worker vk_foreach_struct_const(ext, pInfo->pNext) {
2181*61046927SAndroid Build Coastguard Worker switch (ext->sType) {
2182*61046927SAndroid Build Coastguard Worker case VK_STRUCTURE_TYPE_IMAGE_PLANE_MEMORY_REQUIREMENTS_INFO: {
2183*61046927SAndroid Build Coastguard Worker assert(image->disjoint);
2184*61046927SAndroid Build Coastguard Worker const VkImagePlaneMemoryRequirementsInfo *plane_reqs =
2185*61046927SAndroid Build Coastguard Worker (const VkImagePlaneMemoryRequirementsInfo *) ext;
2186*61046927SAndroid Build Coastguard Worker aspects = plane_reqs->planeAspect;
2187*61046927SAndroid Build Coastguard Worker break;
2188*61046927SAndroid Build Coastguard Worker }
2189*61046927SAndroid Build Coastguard Worker
2190*61046927SAndroid Build Coastguard Worker default:
2191*61046927SAndroid Build Coastguard Worker vk_debug_ignored_stype(ext->sType);
2192*61046927SAndroid Build Coastguard Worker break;
2193*61046927SAndroid Build Coastguard Worker }
2194*61046927SAndroid Build Coastguard Worker }
2195*61046927SAndroid Build Coastguard Worker
2196*61046927SAndroid Build Coastguard Worker anv_image_get_memory_requirements(device, image, aspects,
2197*61046927SAndroid Build Coastguard Worker pMemoryRequirements);
2198*61046927SAndroid Build Coastguard Worker }
2199*61046927SAndroid Build Coastguard Worker
anv_GetDeviceImageMemoryRequirements(VkDevice _device,const VkDeviceImageMemoryRequirements * pInfo,VkMemoryRequirements2 * pMemoryRequirements)2200*61046927SAndroid Build Coastguard Worker void anv_GetDeviceImageMemoryRequirements(
2201*61046927SAndroid Build Coastguard Worker VkDevice _device,
2202*61046927SAndroid Build Coastguard Worker const VkDeviceImageMemoryRequirements* pInfo,
2203*61046927SAndroid Build Coastguard Worker VkMemoryRequirements2* pMemoryRequirements)
2204*61046927SAndroid Build Coastguard Worker {
2205*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
2206*61046927SAndroid Build Coastguard Worker struct anv_image image = { 0 };
2207*61046927SAndroid Build Coastguard Worker
2208*61046927SAndroid Build Coastguard Worker if ((device->physical->sparse_type == ANV_SPARSE_TYPE_NOT_SUPPORTED) &&
2209*61046927SAndroid Build Coastguard Worker INTEL_DEBUG(DEBUG_SPARSE) &&
2210*61046927SAndroid Build Coastguard Worker pInfo->pCreateInfo->flags & (VK_IMAGE_CREATE_SPARSE_BINDING_BIT |
2211*61046927SAndroid Build Coastguard Worker VK_IMAGE_CREATE_SPARSE_RESIDENCY_BIT |
2212*61046927SAndroid Build Coastguard Worker VK_IMAGE_CREATE_SPARSE_ALIASED_BIT))
2213*61046927SAndroid Build Coastguard Worker fprintf(stderr, "=== %s %s:%d flags:0x%08x\n", __func__, __FILE__,
2214*61046927SAndroid Build Coastguard Worker __LINE__, pInfo->pCreateInfo->flags);
2215*61046927SAndroid Build Coastguard Worker
2216*61046927SAndroid Build Coastguard Worker ASSERTED VkResult result =
2217*61046927SAndroid Build Coastguard Worker anv_image_init_from_create_info(device, &image, pInfo->pCreateInfo, true);
2218*61046927SAndroid Build Coastguard Worker assert(result == VK_SUCCESS);
2219*61046927SAndroid Build Coastguard Worker
2220*61046927SAndroid Build Coastguard Worker VkImageAspectFlags aspects =
2221*61046927SAndroid Build Coastguard Worker image.disjoint ? pInfo->planeAspect : image.vk.aspects;
2222*61046927SAndroid Build Coastguard Worker
2223*61046927SAndroid Build Coastguard Worker anv_image_get_memory_requirements(device, &image, aspects,
2224*61046927SAndroid Build Coastguard Worker pMemoryRequirements);
2225*61046927SAndroid Build Coastguard Worker anv_image_finish(&image);
2226*61046927SAndroid Build Coastguard Worker }
2227*61046927SAndroid Build Coastguard Worker
2228*61046927SAndroid Build Coastguard Worker static void
anv_image_get_sparse_memory_requirements(struct anv_device * device,struct anv_image * image,VkImageAspectFlags aspects,uint32_t * pSparseMemoryRequirementCount,VkSparseImageMemoryRequirements2 * pSparseMemoryRequirements)2229*61046927SAndroid Build Coastguard Worker anv_image_get_sparse_memory_requirements(
2230*61046927SAndroid Build Coastguard Worker struct anv_device *device,
2231*61046927SAndroid Build Coastguard Worker struct anv_image *image,
2232*61046927SAndroid Build Coastguard Worker VkImageAspectFlags aspects,
2233*61046927SAndroid Build Coastguard Worker uint32_t *pSparseMemoryRequirementCount,
2234*61046927SAndroid Build Coastguard Worker VkSparseImageMemoryRequirements2 *pSparseMemoryRequirements)
2235*61046927SAndroid Build Coastguard Worker {
2236*61046927SAndroid Build Coastguard Worker VK_OUTARRAY_MAKE_TYPED(VkSparseImageMemoryRequirements2, reqs,
2237*61046927SAndroid Build Coastguard Worker pSparseMemoryRequirements,
2238*61046927SAndroid Build Coastguard Worker pSparseMemoryRequirementCount);
2239*61046927SAndroid Build Coastguard Worker
2240*61046927SAndroid Build Coastguard Worker /* From the spec:
2241*61046927SAndroid Build Coastguard Worker * "The sparse image must have been created using the
2242*61046927SAndroid Build Coastguard Worker * VK_IMAGE_CREATE_SPARSE_RESIDENCY_BIT flag to retrieve valid sparse
2243*61046927SAndroid Build Coastguard Worker * image memory requirements."
2244*61046927SAndroid Build Coastguard Worker */
2245*61046927SAndroid Build Coastguard Worker if (!(image->vk.create_flags & VK_IMAGE_CREATE_SPARSE_RESIDENCY_BIT))
2246*61046927SAndroid Build Coastguard Worker return;
2247*61046927SAndroid Build Coastguard Worker
2248*61046927SAndroid Build Coastguard Worker VkSparseImageMemoryRequirements ds_mem_reqs = {};
2249*61046927SAndroid Build Coastguard Worker VkSparseImageMemoryRequirements2 *ds_reqs_ptr = NULL;
2250*61046927SAndroid Build Coastguard Worker
2251*61046927SAndroid Build Coastguard Worker u_foreach_bit(b, aspects) {
2252*61046927SAndroid Build Coastguard Worker VkImageAspectFlagBits aspect = 1 << b;
2253*61046927SAndroid Build Coastguard Worker const uint32_t plane = anv_image_aspect_to_plane(image, aspect);
2254*61046927SAndroid Build Coastguard Worker struct isl_surf *surf = &image->planes[plane].primary_surface.isl;
2255*61046927SAndroid Build Coastguard Worker
2256*61046927SAndroid Build Coastguard Worker VkSparseImageFormatProperties format_props =
2257*61046927SAndroid Build Coastguard Worker anv_sparse_calc_image_format_properties(device->physical, aspect,
2258*61046927SAndroid Build Coastguard Worker image->vk.image_type,
2259*61046927SAndroid Build Coastguard Worker image->vk.samples, surf);
2260*61046927SAndroid Build Coastguard Worker
2261*61046927SAndroid Build Coastguard Worker uint32_t miptail_first_lod;
2262*61046927SAndroid Build Coastguard Worker VkDeviceSize miptail_size, miptail_offset, miptail_stride;
2263*61046927SAndroid Build Coastguard Worker anv_sparse_calc_miptail_properties(device, image, aspect,
2264*61046927SAndroid Build Coastguard Worker &miptail_first_lod, &miptail_size,
2265*61046927SAndroid Build Coastguard Worker &miptail_offset, &miptail_stride);
2266*61046927SAndroid Build Coastguard Worker
2267*61046927SAndroid Build Coastguard Worker VkSparseImageMemoryRequirements mem_reqs = {
2268*61046927SAndroid Build Coastguard Worker .formatProperties = format_props,
2269*61046927SAndroid Build Coastguard Worker .imageMipTailFirstLod = miptail_first_lod,
2270*61046927SAndroid Build Coastguard Worker .imageMipTailSize = miptail_size,
2271*61046927SAndroid Build Coastguard Worker .imageMipTailOffset = miptail_offset,
2272*61046927SAndroid Build Coastguard Worker .imageMipTailStride = miptail_stride,
2273*61046927SAndroid Build Coastguard Worker };
2274*61046927SAndroid Build Coastguard Worker
2275*61046927SAndroid Build Coastguard Worker /* If both depth and stencil are the same, unify them if possible. */
2276*61046927SAndroid Build Coastguard Worker if (aspect & (VK_IMAGE_ASPECT_DEPTH_BIT |
2277*61046927SAndroid Build Coastguard Worker VK_IMAGE_ASPECT_STENCIL_BIT)) {
2278*61046927SAndroid Build Coastguard Worker if (!ds_reqs_ptr) {
2279*61046927SAndroid Build Coastguard Worker ds_mem_reqs = mem_reqs;
2280*61046927SAndroid Build Coastguard Worker } else if (ds_mem_reqs.formatProperties.imageGranularity.width ==
2281*61046927SAndroid Build Coastguard Worker mem_reqs.formatProperties.imageGranularity.width &&
2282*61046927SAndroid Build Coastguard Worker ds_mem_reqs.formatProperties.imageGranularity.height ==
2283*61046927SAndroid Build Coastguard Worker mem_reqs.formatProperties.imageGranularity.height &&
2284*61046927SAndroid Build Coastguard Worker ds_mem_reqs.formatProperties.imageGranularity.depth ==
2285*61046927SAndroid Build Coastguard Worker mem_reqs.formatProperties.imageGranularity.depth &&
2286*61046927SAndroid Build Coastguard Worker ds_mem_reqs.imageMipTailFirstLod ==
2287*61046927SAndroid Build Coastguard Worker mem_reqs.imageMipTailFirstLod &&
2288*61046927SAndroid Build Coastguard Worker ds_mem_reqs.imageMipTailSize ==
2289*61046927SAndroid Build Coastguard Worker mem_reqs.imageMipTailSize &&
2290*61046927SAndroid Build Coastguard Worker ds_mem_reqs.imageMipTailOffset ==
2291*61046927SAndroid Build Coastguard Worker mem_reqs.imageMipTailOffset &&
2292*61046927SAndroid Build Coastguard Worker ds_mem_reqs.imageMipTailStride ==
2293*61046927SAndroid Build Coastguard Worker mem_reqs.imageMipTailStride) {
2294*61046927SAndroid Build Coastguard Worker ds_reqs_ptr->memoryRequirements.formatProperties.aspectMask |=
2295*61046927SAndroid Build Coastguard Worker aspect;
2296*61046927SAndroid Build Coastguard Worker continue;
2297*61046927SAndroid Build Coastguard Worker }
2298*61046927SAndroid Build Coastguard Worker }
2299*61046927SAndroid Build Coastguard Worker
2300*61046927SAndroid Build Coastguard Worker vk_outarray_append_typed(VkSparseImageMemoryRequirements2, &reqs, r) {
2301*61046927SAndroid Build Coastguard Worker r->memoryRequirements = mem_reqs;
2302*61046927SAndroid Build Coastguard Worker if (aspect & (VK_IMAGE_ASPECT_DEPTH_BIT |
2303*61046927SAndroid Build Coastguard Worker VK_IMAGE_ASPECT_STENCIL_BIT))
2304*61046927SAndroid Build Coastguard Worker ds_reqs_ptr = r;
2305*61046927SAndroid Build Coastguard Worker }
2306*61046927SAndroid Build Coastguard Worker }
2307*61046927SAndroid Build Coastguard Worker }
2308*61046927SAndroid Build Coastguard Worker
anv_GetImageSparseMemoryRequirements2(VkDevice _device,const VkImageSparseMemoryRequirementsInfo2 * pInfo,uint32_t * pSparseMemoryRequirementCount,VkSparseImageMemoryRequirements2 * pSparseMemoryRequirements)2309*61046927SAndroid Build Coastguard Worker void anv_GetImageSparseMemoryRequirements2(
2310*61046927SAndroid Build Coastguard Worker VkDevice _device,
2311*61046927SAndroid Build Coastguard Worker const VkImageSparseMemoryRequirementsInfo2* pInfo,
2312*61046927SAndroid Build Coastguard Worker uint32_t* pSparseMemoryRequirementCount,
2313*61046927SAndroid Build Coastguard Worker VkSparseImageMemoryRequirements2* pSparseMemoryRequirements)
2314*61046927SAndroid Build Coastguard Worker {
2315*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
2316*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_image, image, pInfo->image);
2317*61046927SAndroid Build Coastguard Worker
2318*61046927SAndroid Build Coastguard Worker if (!anv_sparse_residency_is_enabled(device)) {
2319*61046927SAndroid Build Coastguard Worker if ((device->physical->sparse_type == ANV_SPARSE_TYPE_NOT_SUPPORTED) &&
2320*61046927SAndroid Build Coastguard Worker INTEL_DEBUG(DEBUG_SPARSE))
2321*61046927SAndroid Build Coastguard Worker fprintf(stderr, "=== [%s:%d] [%s]\n", __FILE__, __LINE__, __func__);
2322*61046927SAndroid Build Coastguard Worker
2323*61046927SAndroid Build Coastguard Worker *pSparseMemoryRequirementCount = 0;
2324*61046927SAndroid Build Coastguard Worker return;
2325*61046927SAndroid Build Coastguard Worker }
2326*61046927SAndroid Build Coastguard Worker
2327*61046927SAndroid Build Coastguard Worker anv_image_get_sparse_memory_requirements(device, image, image->vk.aspects,
2328*61046927SAndroid Build Coastguard Worker pSparseMemoryRequirementCount,
2329*61046927SAndroid Build Coastguard Worker pSparseMemoryRequirements);
2330*61046927SAndroid Build Coastguard Worker }
2331*61046927SAndroid Build Coastguard Worker
anv_GetDeviceImageSparseMemoryRequirements(VkDevice _device,const VkDeviceImageMemoryRequirements * pInfo,uint32_t * pSparseMemoryRequirementCount,VkSparseImageMemoryRequirements2 * pSparseMemoryRequirements)2332*61046927SAndroid Build Coastguard Worker void anv_GetDeviceImageSparseMemoryRequirements(
2333*61046927SAndroid Build Coastguard Worker VkDevice _device,
2334*61046927SAndroid Build Coastguard Worker const VkDeviceImageMemoryRequirements* pInfo,
2335*61046927SAndroid Build Coastguard Worker uint32_t* pSparseMemoryRequirementCount,
2336*61046927SAndroid Build Coastguard Worker VkSparseImageMemoryRequirements2* pSparseMemoryRequirements)
2337*61046927SAndroid Build Coastguard Worker {
2338*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
2339*61046927SAndroid Build Coastguard Worker struct anv_image image = { 0 };
2340*61046927SAndroid Build Coastguard Worker
2341*61046927SAndroid Build Coastguard Worker if (!anv_sparse_residency_is_enabled(device)) {
2342*61046927SAndroid Build Coastguard Worker if ((device->physical->sparse_type == ANV_SPARSE_TYPE_NOT_SUPPORTED) &&
2343*61046927SAndroid Build Coastguard Worker INTEL_DEBUG(DEBUG_SPARSE))
2344*61046927SAndroid Build Coastguard Worker fprintf(stderr, "=== [%s:%d] [%s]\n", __FILE__, __LINE__, __func__);
2345*61046927SAndroid Build Coastguard Worker
2346*61046927SAndroid Build Coastguard Worker *pSparseMemoryRequirementCount = 0;
2347*61046927SAndroid Build Coastguard Worker return;
2348*61046927SAndroid Build Coastguard Worker }
2349*61046927SAndroid Build Coastguard Worker
2350*61046927SAndroid Build Coastguard Worker /* This function is similar to anv_GetDeviceImageMemoryRequirements, in
2351*61046927SAndroid Build Coastguard Worker * which it actually creates an image, gets the properties and then
2352*61046927SAndroid Build Coastguard Worker * destroys the image.
2353*61046927SAndroid Build Coastguard Worker *
2354*61046927SAndroid Build Coastguard Worker * We could one day refactor things to allow us to gather the properties
2355*61046927SAndroid Build Coastguard Worker * without having to actually create the image, maybe by reworking ISL to
2356*61046927SAndroid Build Coastguard Worker * separate creation from parameter computing.
2357*61046927SAndroid Build Coastguard Worker */
2358*61046927SAndroid Build Coastguard Worker VkResult result =
2359*61046927SAndroid Build Coastguard Worker anv_image_init_from_create_info(device, &image, pInfo->pCreateInfo,
2360*61046927SAndroid Build Coastguard Worker true /* no_private_binding_alloc */);
2361*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS) {
2362*61046927SAndroid Build Coastguard Worker *pSparseMemoryRequirementCount = 0;
2363*61046927SAndroid Build Coastguard Worker return;
2364*61046927SAndroid Build Coastguard Worker }
2365*61046927SAndroid Build Coastguard Worker
2366*61046927SAndroid Build Coastguard Worker /* The spec says:
2367*61046927SAndroid Build Coastguard Worker * "planeAspect is a VkImageAspectFlagBits value specifying the aspect
2368*61046927SAndroid Build Coastguard Worker * corresponding to the image plane to query. This parameter is ignored
2369*61046927SAndroid Build Coastguard Worker * unless pCreateInfo::tiling is VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT,
2370*61046927SAndroid Build Coastguard Worker * or pCreateInfo::flags has VK_IMAGE_CREATE_DISJOINT_BIT set."
2371*61046927SAndroid Build Coastguard Worker */
2372*61046927SAndroid Build Coastguard Worker VkImageAspectFlags aspects =
2373*61046927SAndroid Build Coastguard Worker (pInfo->pCreateInfo->flags & VK_IMAGE_CREATE_DISJOINT_BIT) ||
2374*61046927SAndroid Build Coastguard Worker (pInfo->pCreateInfo->tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT)
2375*61046927SAndroid Build Coastguard Worker ? pInfo->planeAspect : image.vk.aspects;
2376*61046927SAndroid Build Coastguard Worker
2377*61046927SAndroid Build Coastguard Worker anv_image_get_sparse_memory_requirements(device, &image, aspects,
2378*61046927SAndroid Build Coastguard Worker pSparseMemoryRequirementCount,
2379*61046927SAndroid Build Coastguard Worker pSparseMemoryRequirements);
2380*61046927SAndroid Build Coastguard Worker
2381*61046927SAndroid Build Coastguard Worker anv_image_finish(&image);
2382*61046927SAndroid Build Coastguard Worker }
2383*61046927SAndroid Build Coastguard Worker
2384*61046927SAndroid Build Coastguard Worker static bool
anv_image_map_aux_tt(struct anv_device * device,struct anv_image * image,uint32_t plane)2385*61046927SAndroid Build Coastguard Worker anv_image_map_aux_tt(struct anv_device *device,
2386*61046927SAndroid Build Coastguard Worker struct anv_image *image, uint32_t plane)
2387*61046927SAndroid Build Coastguard Worker {
2388*61046927SAndroid Build Coastguard Worker const struct anv_address main_addr = anv_image_address(
2389*61046927SAndroid Build Coastguard Worker image, &image->planes[plane].primary_surface.memory_range);
2390*61046927SAndroid Build Coastguard Worker struct anv_bo *bo = main_addr.bo;
2391*61046927SAndroid Build Coastguard Worker assert(bo != NULL);
2392*61046927SAndroid Build Coastguard Worker
2393*61046927SAndroid Build Coastguard Worker /* If the additional memory padding was added at the end of the BO for CCS
2394*61046927SAndroid Build Coastguard Worker * data, map this region at the granularity of the main/CCS pages.
2395*61046927SAndroid Build Coastguard Worker *
2396*61046927SAndroid Build Coastguard Worker * Otherwise the image should have additional CCS data at the computed
2397*61046927SAndroid Build Coastguard Worker * offset.
2398*61046927SAndroid Build Coastguard Worker */
2399*61046927SAndroid Build Coastguard Worker if (device->physical->alloc_aux_tt_mem &&
2400*61046927SAndroid Build Coastguard Worker (bo->alloc_flags & ANV_BO_ALLOC_AUX_CCS)) {
2401*61046927SAndroid Build Coastguard Worker uint64_t main_aux_alignment =
2402*61046927SAndroid Build Coastguard Worker intel_aux_map_get_alignment(device->aux_map_ctx);
2403*61046927SAndroid Build Coastguard Worker assert(bo->offset % main_aux_alignment == 0);
2404*61046927SAndroid Build Coastguard Worker const struct anv_address start_addr = (struct anv_address) {
2405*61046927SAndroid Build Coastguard Worker .bo = bo,
2406*61046927SAndroid Build Coastguard Worker .offset = ROUND_DOWN_TO(main_addr.offset, main_aux_alignment),
2407*61046927SAndroid Build Coastguard Worker };
2408*61046927SAndroid Build Coastguard Worker const struct anv_address aux_addr = (struct anv_address) {
2409*61046927SAndroid Build Coastguard Worker .bo = bo,
2410*61046927SAndroid Build Coastguard Worker .offset = bo->ccs_offset +
2411*61046927SAndroid Build Coastguard Worker intel_aux_main_to_aux_offset(device->aux_map_ctx,
2412*61046927SAndroid Build Coastguard Worker start_addr.offset),
2413*61046927SAndroid Build Coastguard Worker };
2414*61046927SAndroid Build Coastguard Worker const struct isl_surf *surf = &image->planes[plane].primary_surface.isl;
2415*61046927SAndroid Build Coastguard Worker const uint64_t format_bits =
2416*61046927SAndroid Build Coastguard Worker intel_aux_map_format_bits_for_isl_surf(surf);
2417*61046927SAndroid Build Coastguard Worker /* Make sure to have the mapping cover the entire image from the aux
2418*61046927SAndroid Build Coastguard Worker * aligned start.
2419*61046927SAndroid Build Coastguard Worker */
2420*61046927SAndroid Build Coastguard Worker const uint64_t main_size = align(
2421*61046927SAndroid Build Coastguard Worker (main_addr.offset - start_addr.offset) + surf->size_B,
2422*61046927SAndroid Build Coastguard Worker main_aux_alignment);
2423*61046927SAndroid Build Coastguard Worker
2424*61046927SAndroid Build Coastguard Worker if (intel_aux_map_add_mapping(device->aux_map_ctx,
2425*61046927SAndroid Build Coastguard Worker anv_address_physical(start_addr),
2426*61046927SAndroid Build Coastguard Worker anv_address_physical(aux_addr),
2427*61046927SAndroid Build Coastguard Worker main_size, format_bits)) {
2428*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_tt.mapped = true;
2429*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_tt.addr = anv_address_physical(start_addr);
2430*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_tt.size = main_size;
2431*61046927SAndroid Build Coastguard Worker return true;
2432*61046927SAndroid Build Coastguard Worker }
2433*61046927SAndroid Build Coastguard Worker } else {
2434*61046927SAndroid Build Coastguard Worker if (anv_address_allows_aux_map(device, main_addr)) {
2435*61046927SAndroid Build Coastguard Worker const struct anv_address aux_addr =
2436*61046927SAndroid Build Coastguard Worker anv_image_address(image,
2437*61046927SAndroid Build Coastguard Worker &image->planes[plane].compr_ctrl_memory_range);
2438*61046927SAndroid Build Coastguard Worker const struct isl_surf *surf =
2439*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl;
2440*61046927SAndroid Build Coastguard Worker const uint64_t format_bits =
2441*61046927SAndroid Build Coastguard Worker intel_aux_map_format_bits_for_isl_surf(surf);
2442*61046927SAndroid Build Coastguard Worker if (intel_aux_map_add_mapping(device->aux_map_ctx,
2443*61046927SAndroid Build Coastguard Worker anv_address_physical(main_addr),
2444*61046927SAndroid Build Coastguard Worker anv_address_physical(aux_addr),
2445*61046927SAndroid Build Coastguard Worker surf->size_B, format_bits)) {
2446*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_tt.mapped = true;
2447*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_tt.addr = anv_address_physical(main_addr);
2448*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_tt.size = surf->size_B;
2449*61046927SAndroid Build Coastguard Worker return true;
2450*61046927SAndroid Build Coastguard Worker }
2451*61046927SAndroid Build Coastguard Worker }
2452*61046927SAndroid Build Coastguard Worker }
2453*61046927SAndroid Build Coastguard Worker
2454*61046927SAndroid Build Coastguard Worker return false;
2455*61046927SAndroid Build Coastguard Worker
2456*61046927SAndroid Build Coastguard Worker }
2457*61046927SAndroid Build Coastguard Worker
2458*61046927SAndroid Build Coastguard Worker static VkResult
anv_bind_image_memory(struct anv_device * device,const VkBindImageMemoryInfo * bind_info)2459*61046927SAndroid Build Coastguard Worker anv_bind_image_memory(struct anv_device *device,
2460*61046927SAndroid Build Coastguard Worker const VkBindImageMemoryInfo *bind_info)
2461*61046927SAndroid Build Coastguard Worker {
2462*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device_memory, mem, bind_info->memory);
2463*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_image, image, bind_info->image);
2464*61046927SAndroid Build Coastguard Worker bool did_bind = false;
2465*61046927SAndroid Build Coastguard Worker
2466*61046927SAndroid Build Coastguard Worker const VkBindMemoryStatusKHR *bind_status =
2467*61046927SAndroid Build Coastguard Worker vk_find_struct_const(bind_info->pNext, BIND_MEMORY_STATUS_KHR);
2468*61046927SAndroid Build Coastguard Worker
2469*61046927SAndroid Build Coastguard Worker assert(!anv_image_is_sparse(image));
2470*61046927SAndroid Build Coastguard Worker
2471*61046927SAndroid Build Coastguard Worker /* Resolve will alter the image's aspects, do this first. */
2472*61046927SAndroid Build Coastguard Worker if (mem && mem->vk.ahardware_buffer)
2473*61046927SAndroid Build Coastguard Worker resolve_ahw_image(device, image, mem);
2474*61046927SAndroid Build Coastguard Worker
2475*61046927SAndroid Build Coastguard Worker vk_foreach_struct_const(s, bind_info->pNext) {
2476*61046927SAndroid Build Coastguard Worker switch (s->sType) {
2477*61046927SAndroid Build Coastguard Worker case VK_STRUCTURE_TYPE_BIND_IMAGE_PLANE_MEMORY_INFO: {
2478*61046927SAndroid Build Coastguard Worker const VkBindImagePlaneMemoryInfo *plane_info =
2479*61046927SAndroid Build Coastguard Worker (const VkBindImagePlaneMemoryInfo *) s;
2480*61046927SAndroid Build Coastguard Worker
2481*61046927SAndroid Build Coastguard Worker /* Workaround for possible spec bug.
2482*61046927SAndroid Build Coastguard Worker *
2483*61046927SAndroid Build Coastguard Worker * Unlike VkImagePlaneMemoryRequirementsInfo, which requires that
2484*61046927SAndroid Build Coastguard Worker * the image be disjoint (that is, multi-planar format and
2485*61046927SAndroid Build Coastguard Worker * VK_IMAGE_CREATE_DISJOINT_BIT), VkBindImagePlaneMemoryInfo allows
2486*61046927SAndroid Build Coastguard Worker * the image to be non-disjoint and requires only that the image
2487*61046927SAndroid Build Coastguard Worker * have the DISJOINT flag. In this case, regardless of the value of
2488*61046927SAndroid Build Coastguard Worker * VkImagePlaneMemoryRequirementsInfo::planeAspect, the behavior is
2489*61046927SAndroid Build Coastguard Worker * the same as if VkImagePlaneMemoryRequirementsInfo were omitted.
2490*61046927SAndroid Build Coastguard Worker */
2491*61046927SAndroid Build Coastguard Worker if (!image->disjoint)
2492*61046927SAndroid Build Coastguard Worker break;
2493*61046927SAndroid Build Coastguard Worker
2494*61046927SAndroid Build Coastguard Worker struct anv_image_binding *binding =
2495*61046927SAndroid Build Coastguard Worker anv_image_aspect_to_binding(image, plane_info->planeAspect);
2496*61046927SAndroid Build Coastguard Worker
2497*61046927SAndroid Build Coastguard Worker binding->address = (struct anv_address) {
2498*61046927SAndroid Build Coastguard Worker .bo = mem->bo,
2499*61046927SAndroid Build Coastguard Worker .offset = bind_info->memoryOffset,
2500*61046927SAndroid Build Coastguard Worker };
2501*61046927SAndroid Build Coastguard Worker
2502*61046927SAndroid Build Coastguard Worker ANV_RMV(image_bind, device, image,
2503*61046927SAndroid Build Coastguard Worker binding - image->bindings);
2504*61046927SAndroid Build Coastguard Worker
2505*61046927SAndroid Build Coastguard Worker did_bind = true;
2506*61046927SAndroid Build Coastguard Worker break;
2507*61046927SAndroid Build Coastguard Worker }
2508*61046927SAndroid Build Coastguard Worker case VK_STRUCTURE_TYPE_BIND_IMAGE_MEMORY_SWAPCHAIN_INFO_KHR: {
2509*61046927SAndroid Build Coastguard Worker /* Ignore this struct on Android, we cannot access swapchain
2510*61046927SAndroid Build Coastguard Worker * structures there.
2511*61046927SAndroid Build Coastguard Worker */
2512*61046927SAndroid Build Coastguard Worker #ifndef VK_USE_PLATFORM_ANDROID_KHR
2513*61046927SAndroid Build Coastguard Worker const VkBindImageMemorySwapchainInfoKHR *swapchain_info =
2514*61046927SAndroid Build Coastguard Worker (const VkBindImageMemorySwapchainInfoKHR *) s;
2515*61046927SAndroid Build Coastguard Worker struct anv_image *swapchain_image =
2516*61046927SAndroid Build Coastguard Worker anv_swapchain_get_image(swapchain_info->swapchain,
2517*61046927SAndroid Build Coastguard Worker swapchain_info->imageIndex);
2518*61046927SAndroid Build Coastguard Worker assert(swapchain_image);
2519*61046927SAndroid Build Coastguard Worker assert(image->vk.aspects == swapchain_image->vk.aspects);
2520*61046927SAndroid Build Coastguard Worker assert(mem == NULL);
2521*61046927SAndroid Build Coastguard Worker
2522*61046927SAndroid Build Coastguard Worker for (int j = 0; j < ARRAY_SIZE(image->bindings); ++j) {
2523*61046927SAndroid Build Coastguard Worker assert(memory_ranges_equal(image->bindings[j].memory_range,
2524*61046927SAndroid Build Coastguard Worker swapchain_image->bindings[j].memory_range));
2525*61046927SAndroid Build Coastguard Worker image->bindings[j].address = swapchain_image->bindings[j].address;
2526*61046927SAndroid Build Coastguard Worker }
2527*61046927SAndroid Build Coastguard Worker
2528*61046927SAndroid Build Coastguard Worker /* We must bump the private binding's bo's refcount because, unlike the other
2529*61046927SAndroid Build Coastguard Worker * bindings, its lifetime is not application-managed.
2530*61046927SAndroid Build Coastguard Worker */
2531*61046927SAndroid Build Coastguard Worker struct anv_bo *private_bo =
2532*61046927SAndroid Build Coastguard Worker image->bindings[ANV_IMAGE_MEMORY_BINDING_PRIVATE].address.bo;
2533*61046927SAndroid Build Coastguard Worker if (private_bo)
2534*61046927SAndroid Build Coastguard Worker anv_bo_ref(private_bo);
2535*61046927SAndroid Build Coastguard Worker
2536*61046927SAndroid Build Coastguard Worker did_bind = true;
2537*61046927SAndroid Build Coastguard Worker #endif
2538*61046927SAndroid Build Coastguard Worker break;
2539*61046927SAndroid Build Coastguard Worker }
2540*61046927SAndroid Build Coastguard Worker #pragma GCC diagnostic push
2541*61046927SAndroid Build Coastguard Worker #pragma GCC diagnostic ignored "-Wswitch"
2542*61046927SAndroid Build Coastguard Worker case VK_STRUCTURE_TYPE_NATIVE_BUFFER_ANDROID: {
2543*61046927SAndroid Build Coastguard Worker const VkNativeBufferANDROID *gralloc_info =
2544*61046927SAndroid Build Coastguard Worker (const VkNativeBufferANDROID *)s;
2545*61046927SAndroid Build Coastguard Worker VkResult result = anv_image_bind_from_gralloc(device, image,
2546*61046927SAndroid Build Coastguard Worker gralloc_info);
2547*61046927SAndroid Build Coastguard Worker if (result != VK_SUCCESS)
2548*61046927SAndroid Build Coastguard Worker return result;
2549*61046927SAndroid Build Coastguard Worker
2550*61046927SAndroid Build Coastguard Worker resolve_anb_image(device, image, gralloc_info);
2551*61046927SAndroid Build Coastguard Worker did_bind = true;
2552*61046927SAndroid Build Coastguard Worker break;
2553*61046927SAndroid Build Coastguard Worker }
2554*61046927SAndroid Build Coastguard Worker #pragma GCC diagnostic pop
2555*61046927SAndroid Build Coastguard Worker default:
2556*61046927SAndroid Build Coastguard Worker vk_debug_ignored_stype(s->sType);
2557*61046927SAndroid Build Coastguard Worker break;
2558*61046927SAndroid Build Coastguard Worker }
2559*61046927SAndroid Build Coastguard Worker }
2560*61046927SAndroid Build Coastguard Worker
2561*61046927SAndroid Build Coastguard Worker if (!did_bind) {
2562*61046927SAndroid Build Coastguard Worker assert(!image->disjoint);
2563*61046927SAndroid Build Coastguard Worker
2564*61046927SAndroid Build Coastguard Worker image->bindings[ANV_IMAGE_MEMORY_BINDING_MAIN].address =
2565*61046927SAndroid Build Coastguard Worker (struct anv_address) {
2566*61046927SAndroid Build Coastguard Worker .bo = mem->bo,
2567*61046927SAndroid Build Coastguard Worker .offset = bind_info->memoryOffset,
2568*61046927SAndroid Build Coastguard Worker };
2569*61046927SAndroid Build Coastguard Worker
2570*61046927SAndroid Build Coastguard Worker ANV_RMV(image_bind, device, image,
2571*61046927SAndroid Build Coastguard Worker ANV_IMAGE_MEMORY_BINDING_MAIN);
2572*61046927SAndroid Build Coastguard Worker
2573*61046927SAndroid Build Coastguard Worker did_bind = true;
2574*61046927SAndroid Build Coastguard Worker }
2575*61046927SAndroid Build Coastguard Worker
2576*61046927SAndroid Build Coastguard Worker /* Now that we have the BO, finalize CCS setup. */
2577*61046927SAndroid Build Coastguard Worker for (int p = 0; p < image->n_planes; ++p) {
2578*61046927SAndroid Build Coastguard Worker enum anv_image_memory_binding binding =
2579*61046927SAndroid Build Coastguard Worker image->planes[p].primary_surface.memory_range.binding;
2580*61046927SAndroid Build Coastguard Worker const struct anv_bo *bo =
2581*61046927SAndroid Build Coastguard Worker image->bindings[binding].address.bo;
2582*61046927SAndroid Build Coastguard Worker
2583*61046927SAndroid Build Coastguard Worker if (!bo || !isl_aux_usage_has_ccs(image->planes[p].aux_usage))
2584*61046927SAndroid Build Coastguard Worker continue;
2585*61046927SAndroid Build Coastguard Worker
2586*61046927SAndroid Build Coastguard Worker /* Do nothing if flat CCS requirements are satisfied.
2587*61046927SAndroid Build Coastguard Worker *
2588*61046927SAndroid Build Coastguard Worker * Also, assume that imported BOs with a modifier including
2589*61046927SAndroid Build Coastguard Worker * CCS live only in local memory. Otherwise the exporter should
2590*61046927SAndroid Build Coastguard Worker * have failed the creation of the BO.
2591*61046927SAndroid Build Coastguard Worker */
2592*61046927SAndroid Build Coastguard Worker if (device->info->has_flat_ccs &&
2593*61046927SAndroid Build Coastguard Worker (anv_bo_is_vram_only(bo) ||
2594*61046927SAndroid Build Coastguard Worker (bo->alloc_flags & ANV_BO_ALLOC_IMPORTED)))
2595*61046927SAndroid Build Coastguard Worker continue;
2596*61046927SAndroid Build Coastguard Worker
2597*61046927SAndroid Build Coastguard Worker /* If the AUX-TT mapping succeeds, there is nothing else to do. */
2598*61046927SAndroid Build Coastguard Worker if (device->info->has_aux_map && anv_image_map_aux_tt(device, image, p))
2599*61046927SAndroid Build Coastguard Worker continue;
2600*61046927SAndroid Build Coastguard Worker
2601*61046927SAndroid Build Coastguard Worker /* Do nothing except for gfx12. There are no special requirements. */
2602*61046927SAndroid Build Coastguard Worker if (device->info->ver != 12)
2603*61046927SAndroid Build Coastguard Worker continue;
2604*61046927SAndroid Build Coastguard Worker
2605*61046927SAndroid Build Coastguard Worker /* The plane's BO cannot support CCS, disable compression on it. */
2606*61046927SAndroid Build Coastguard Worker assert(!isl_drm_modifier_has_aux(image->vk.drm_format_mod));
2607*61046927SAndroid Build Coastguard Worker
2608*61046927SAndroid Build Coastguard Worker anv_perf_warn(VK_LOG_OBJS(&image->vk.base),
2609*61046927SAndroid Build Coastguard Worker "BO lacks CCS support. Disabling the CCS aux usage.");
2610*61046927SAndroid Build Coastguard Worker
2611*61046927SAndroid Build Coastguard Worker if (image->planes[p].aux_surface.memory_range.size > 0) {
2612*61046927SAndroid Build Coastguard Worker assert(image->planes[p].aux_usage == ISL_AUX_USAGE_HIZ_CCS ||
2613*61046927SAndroid Build Coastguard Worker image->planes[p].aux_usage == ISL_AUX_USAGE_HIZ_CCS_WT);
2614*61046927SAndroid Build Coastguard Worker image->planes[p].aux_usage = ISL_AUX_USAGE_HIZ;
2615*61046927SAndroid Build Coastguard Worker } else {
2616*61046927SAndroid Build Coastguard Worker assert(image->planes[p].aux_usage == ISL_AUX_USAGE_CCS_E ||
2617*61046927SAndroid Build Coastguard Worker image->planes[p].aux_usage == ISL_AUX_USAGE_FCV_CCS_E ||
2618*61046927SAndroid Build Coastguard Worker image->planes[p].aux_usage == ISL_AUX_USAGE_STC_CCS);
2619*61046927SAndroid Build Coastguard Worker image->planes[p].aux_usage = ISL_AUX_USAGE_NONE;
2620*61046927SAndroid Build Coastguard Worker }
2621*61046927SAndroid Build Coastguard Worker }
2622*61046927SAndroid Build Coastguard Worker
2623*61046927SAndroid Build Coastguard Worker if (bind_status)
2624*61046927SAndroid Build Coastguard Worker *bind_status->pResult = VK_SUCCESS;
2625*61046927SAndroid Build Coastguard Worker
2626*61046927SAndroid Build Coastguard Worker return VK_SUCCESS;
2627*61046927SAndroid Build Coastguard Worker }
2628*61046927SAndroid Build Coastguard Worker
anv_BindImageMemory2(VkDevice _device,uint32_t bindInfoCount,const VkBindImageMemoryInfo * pBindInfos)2629*61046927SAndroid Build Coastguard Worker VkResult anv_BindImageMemory2(
2630*61046927SAndroid Build Coastguard Worker VkDevice _device,
2631*61046927SAndroid Build Coastguard Worker uint32_t bindInfoCount,
2632*61046927SAndroid Build Coastguard Worker const VkBindImageMemoryInfo* pBindInfos)
2633*61046927SAndroid Build Coastguard Worker {
2634*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
2635*61046927SAndroid Build Coastguard Worker VkResult result = VK_SUCCESS;
2636*61046927SAndroid Build Coastguard Worker
2637*61046927SAndroid Build Coastguard Worker for (uint32_t i = 0; i < bindInfoCount; i++) {
2638*61046927SAndroid Build Coastguard Worker VkResult res = anv_bind_image_memory(device, &pBindInfos[i]);
2639*61046927SAndroid Build Coastguard Worker if (result == VK_SUCCESS && res != VK_SUCCESS)
2640*61046927SAndroid Build Coastguard Worker result = res;
2641*61046927SAndroid Build Coastguard Worker }
2642*61046927SAndroid Build Coastguard Worker
2643*61046927SAndroid Build Coastguard Worker return result;
2644*61046927SAndroid Build Coastguard Worker }
2645*61046927SAndroid Build Coastguard Worker
2646*61046927SAndroid Build Coastguard Worker static void
anv_get_image_subresource_layout(const struct anv_image * image,const VkImageSubresource2KHR * subresource,VkSubresourceLayout2KHR * layout)2647*61046927SAndroid Build Coastguard Worker anv_get_image_subresource_layout(const struct anv_image *image,
2648*61046927SAndroid Build Coastguard Worker const VkImageSubresource2KHR *subresource,
2649*61046927SAndroid Build Coastguard Worker VkSubresourceLayout2KHR *layout)
2650*61046927SAndroid Build Coastguard Worker {
2651*61046927SAndroid Build Coastguard Worker const struct anv_image_memory_range *mem_range;
2652*61046927SAndroid Build Coastguard Worker
2653*61046927SAndroid Build Coastguard Worker assert(__builtin_popcount(subresource->imageSubresource.aspectMask) == 1);
2654*61046927SAndroid Build Coastguard Worker
2655*61046927SAndroid Build Coastguard Worker /* The Vulkan spec requires that aspectMask be
2656*61046927SAndroid Build Coastguard Worker * VK_IMAGE_ASPECT_MEMORY_PLANE_i_BIT_EXT if tiling is
2657*61046927SAndroid Build Coastguard Worker * VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT.
2658*61046927SAndroid Build Coastguard Worker *
2659*61046927SAndroid Build Coastguard Worker * For swapchain images, the Vulkan spec says that every swapchain image has
2660*61046927SAndroid Build Coastguard Worker * tiling VK_IMAGE_TILING_OPTIMAL, but we may choose
2661*61046927SAndroid Build Coastguard Worker * VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT internally. Vulkan doesn't allow
2662*61046927SAndroid Build Coastguard Worker * vkGetImageSubresourceLayout for images with VK_IMAGE_TILING_OPTIMAL,
2663*61046927SAndroid Build Coastguard Worker * therefore it's invalid for the application to call this on a swapchain
2664*61046927SAndroid Build Coastguard Worker * image. The WSI code, however, knows when it has internally created
2665*61046927SAndroid Build Coastguard Worker * a swapchain image with VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT,
2666*61046927SAndroid Build Coastguard Worker * so it _should_ correctly use VK_IMAGE_ASPECT_MEMORY_PLANE_* in that case.
2667*61046927SAndroid Build Coastguard Worker * But it incorrectly uses VK_IMAGE_ASPECT_PLANE_*, so we have a temporary
2668*61046927SAndroid Build Coastguard Worker * workaround.
2669*61046927SAndroid Build Coastguard Worker *
2670*61046927SAndroid Build Coastguard Worker * https://gitlab.freedesktop.org/mesa/mesa/-/issues/10176
2671*61046927SAndroid Build Coastguard Worker */
2672*61046927SAndroid Build Coastguard Worker if (image->vk.tiling == VK_IMAGE_TILING_DRM_FORMAT_MODIFIER_EXT) {
2673*61046927SAndroid Build Coastguard Worker /* TODO(chadv): Drop this workaround when WSI gets fixed. */
2674*61046927SAndroid Build Coastguard Worker uint32_t mem_plane;
2675*61046927SAndroid Build Coastguard Worker switch (subresource->imageSubresource.aspectMask) {
2676*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_MEMORY_PLANE_0_BIT_EXT:
2677*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_PLANE_0_BIT:
2678*61046927SAndroid Build Coastguard Worker mem_plane = 0;
2679*61046927SAndroid Build Coastguard Worker break;
2680*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_MEMORY_PLANE_1_BIT_EXT:
2681*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_PLANE_1_BIT:
2682*61046927SAndroid Build Coastguard Worker mem_plane = 1;
2683*61046927SAndroid Build Coastguard Worker break;
2684*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_MEMORY_PLANE_2_BIT_EXT:
2685*61046927SAndroid Build Coastguard Worker case VK_IMAGE_ASPECT_PLANE_2_BIT:
2686*61046927SAndroid Build Coastguard Worker mem_plane = 2;
2687*61046927SAndroid Build Coastguard Worker break;
2688*61046927SAndroid Build Coastguard Worker default:
2689*61046927SAndroid Build Coastguard Worker unreachable("bad VkImageAspectFlags");
2690*61046927SAndroid Build Coastguard Worker }
2691*61046927SAndroid Build Coastguard Worker
2692*61046927SAndroid Build Coastguard Worker uint32_t row_pitch_B;
2693*61046927SAndroid Build Coastguard Worker if (isl_drm_modifier_plane_is_clear_color(image->vk.drm_format_mod,
2694*61046927SAndroid Build Coastguard Worker mem_plane)) {
2695*61046927SAndroid Build Coastguard Worker assert(image->n_planes == 1);
2696*61046927SAndroid Build Coastguard Worker
2697*61046927SAndroid Build Coastguard Worker mem_range = &image->planes[0].fast_clear_memory_range;
2698*61046927SAndroid Build Coastguard Worker row_pitch_B = ISL_DRM_CC_PLANE_PITCH_B;
2699*61046927SAndroid Build Coastguard Worker } else if (mem_plane == 1 &&
2700*61046927SAndroid Build Coastguard Worker image->planes[0].compr_ctrl_memory_range.size > 0) {
2701*61046927SAndroid Build Coastguard Worker assert(image->n_planes == 1);
2702*61046927SAndroid Build Coastguard Worker assert(isl_drm_modifier_has_aux(image->vk.drm_format_mod));
2703*61046927SAndroid Build Coastguard Worker
2704*61046927SAndroid Build Coastguard Worker mem_range = &image->planes[0].compr_ctrl_memory_range;
2705*61046927SAndroid Build Coastguard Worker row_pitch_B = image->planes[0].primary_surface.isl.row_pitch_B /
2706*61046927SAndroid Build Coastguard Worker INTEL_AUX_MAP_MAIN_PITCH_SCALEDOWN;
2707*61046927SAndroid Build Coastguard Worker } else if (mem_plane == 1 &&
2708*61046927SAndroid Build Coastguard Worker image->planes[0].aux_surface.memory_range.size > 0) {
2709*61046927SAndroid Build Coastguard Worker assert(image->n_planes == 1);
2710*61046927SAndroid Build Coastguard Worker assert(image->vk.drm_format_mod == I915_FORMAT_MOD_Y_TILED_CCS);
2711*61046927SAndroid Build Coastguard Worker
2712*61046927SAndroid Build Coastguard Worker mem_range = &image->planes[0].aux_surface.memory_range;
2713*61046927SAndroid Build Coastguard Worker row_pitch_B = image->planes[0].aux_surface.isl.row_pitch_B;
2714*61046927SAndroid Build Coastguard Worker } else {
2715*61046927SAndroid Build Coastguard Worker assert(mem_plane < image->n_planes);
2716*61046927SAndroid Build Coastguard Worker
2717*61046927SAndroid Build Coastguard Worker mem_range = &image->planes[mem_plane].primary_surface.memory_range;
2718*61046927SAndroid Build Coastguard Worker row_pitch_B =
2719*61046927SAndroid Build Coastguard Worker image->planes[mem_plane].primary_surface.isl.row_pitch_B;
2720*61046927SAndroid Build Coastguard Worker }
2721*61046927SAndroid Build Coastguard Worker
2722*61046927SAndroid Build Coastguard Worker /* If the memory binding differs between the primary plane and the
2723*61046927SAndroid Build Coastguard Worker * specified memory plane, the returned offset will be incorrect.
2724*61046927SAndroid Build Coastguard Worker */
2725*61046927SAndroid Build Coastguard Worker assert(mem_range->binding ==
2726*61046927SAndroid Build Coastguard Worker image->planes[0].primary_surface.memory_range.binding);
2727*61046927SAndroid Build Coastguard Worker
2728*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.offset = mem_range->offset;
2729*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.size = mem_range->size;
2730*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.rowPitch = row_pitch_B;
2731*61046927SAndroid Build Coastguard Worker /* The spec for VkSubresourceLayout says,
2732*61046927SAndroid Build Coastguard Worker *
2733*61046927SAndroid Build Coastguard Worker * The value of arrayPitch is undefined for images that were not
2734*61046927SAndroid Build Coastguard Worker * created as arrays. depthPitch is defined only for 3D images.
2735*61046927SAndroid Build Coastguard Worker *
2736*61046927SAndroid Build Coastguard Worker * We are working with a non-arrayed 2D image. So, we leave the
2737*61046927SAndroid Build Coastguard Worker * remaining pitches undefined.
2738*61046927SAndroid Build Coastguard Worker */
2739*61046927SAndroid Build Coastguard Worker assert(image->vk.image_type == VK_IMAGE_TYPE_2D);
2740*61046927SAndroid Build Coastguard Worker assert(image->vk.array_layers == 1);
2741*61046927SAndroid Build Coastguard Worker } else {
2742*61046927SAndroid Build Coastguard Worker const uint32_t plane =
2743*61046927SAndroid Build Coastguard Worker anv_image_aspect_to_plane(image,
2744*61046927SAndroid Build Coastguard Worker subresource->imageSubresource.aspectMask);
2745*61046927SAndroid Build Coastguard Worker const struct isl_surf *isl_surf =
2746*61046927SAndroid Build Coastguard Worker &image->planes[plane].primary_surface.isl;
2747*61046927SAndroid Build Coastguard Worker mem_range = &image->planes[plane].primary_surface.memory_range;
2748*61046927SAndroid Build Coastguard Worker
2749*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.offset = mem_range->offset;
2750*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.rowPitch = isl_surf->row_pitch_B;
2751*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.depthPitch =
2752*61046927SAndroid Build Coastguard Worker isl_surf_get_array_pitch(isl_surf);
2753*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.arrayPitch =
2754*61046927SAndroid Build Coastguard Worker isl_surf_get_array_pitch(isl_surf);
2755*61046927SAndroid Build Coastguard Worker
2756*61046927SAndroid Build Coastguard Worker const uint32_t level = subresource->imageSubresource.mipLevel;
2757*61046927SAndroid Build Coastguard Worker const uint32_t layer = subresource->imageSubresource.arrayLayer;
2758*61046927SAndroid Build Coastguard Worker if (level > 0 || layer > 0) {
2759*61046927SAndroid Build Coastguard Worker assert(isl_surf->tiling == ISL_TILING_LINEAR);
2760*61046927SAndroid Build Coastguard Worker
2761*61046927SAndroid Build Coastguard Worker uint64_t offset_B;
2762*61046927SAndroid Build Coastguard Worker isl_surf_get_image_offset_B_tile_sa(isl_surf, level, layer,
2763*61046927SAndroid Build Coastguard Worker 0 /* logical_z_offset_px */,
2764*61046927SAndroid Build Coastguard Worker &offset_B, NULL, NULL);
2765*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.offset += offset_B;
2766*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.size =
2767*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.rowPitch *
2768*61046927SAndroid Build Coastguard Worker u_minify(image->vk.extent.height, level) *
2769*61046927SAndroid Build Coastguard Worker image->vk.extent.depth;
2770*61046927SAndroid Build Coastguard Worker } else {
2771*61046927SAndroid Build Coastguard Worker layout->subresourceLayout.size = mem_range->size;
2772*61046927SAndroid Build Coastguard Worker }
2773*61046927SAndroid Build Coastguard Worker }
2774*61046927SAndroid Build Coastguard Worker
2775*61046927SAndroid Build Coastguard Worker VkImageCompressionPropertiesEXT *comp_props =
2776*61046927SAndroid Build Coastguard Worker vk_find_struct(layout->pNext, IMAGE_COMPRESSION_PROPERTIES_EXT);
2777*61046927SAndroid Build Coastguard Worker if (comp_props) {
2778*61046927SAndroid Build Coastguard Worker comp_props->imageCompressionFixedRateFlags =
2779*61046927SAndroid Build Coastguard Worker VK_IMAGE_COMPRESSION_FIXED_RATE_NONE_EXT;
2780*61046927SAndroid Build Coastguard Worker comp_props->imageCompressionFlags = VK_IMAGE_COMPRESSION_DISABLED_EXT;
2781*61046927SAndroid Build Coastguard Worker for (uint32_t p = 0; p < image->n_planes; p++) {
2782*61046927SAndroid Build Coastguard Worker if (image->planes[p].aux_usage != ISL_AUX_USAGE_NONE) {
2783*61046927SAndroid Build Coastguard Worker comp_props->imageCompressionFlags = VK_IMAGE_COMPRESSION_DEFAULT_EXT;
2784*61046927SAndroid Build Coastguard Worker break;
2785*61046927SAndroid Build Coastguard Worker }
2786*61046927SAndroid Build Coastguard Worker }
2787*61046927SAndroid Build Coastguard Worker }
2788*61046927SAndroid Build Coastguard Worker }
2789*61046927SAndroid Build Coastguard Worker
anv_GetDeviceImageSubresourceLayoutKHR(VkDevice _device,const VkDeviceImageSubresourceInfoKHR * pInfo,VkSubresourceLayout2KHR * pLayout)2790*61046927SAndroid Build Coastguard Worker void anv_GetDeviceImageSubresourceLayoutKHR(
2791*61046927SAndroid Build Coastguard Worker VkDevice _device,
2792*61046927SAndroid Build Coastguard Worker const VkDeviceImageSubresourceInfoKHR* pInfo,
2793*61046927SAndroid Build Coastguard Worker VkSubresourceLayout2KHR* pLayout)
2794*61046927SAndroid Build Coastguard Worker {
2795*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_device, device, _device);
2796*61046927SAndroid Build Coastguard Worker
2797*61046927SAndroid Build Coastguard Worker struct anv_image image = { 0 };
2798*61046927SAndroid Build Coastguard Worker
2799*61046927SAndroid Build Coastguard Worker if (anv_image_init_from_create_info(device, &image, pInfo->pCreateInfo,
2800*61046927SAndroid Build Coastguard Worker true) != VK_SUCCESS) {
2801*61046927SAndroid Build Coastguard Worker pLayout->subresourceLayout = (VkSubresourceLayout) { 0, };
2802*61046927SAndroid Build Coastguard Worker return;
2803*61046927SAndroid Build Coastguard Worker }
2804*61046927SAndroid Build Coastguard Worker
2805*61046927SAndroid Build Coastguard Worker anv_get_image_subresource_layout(&image, pInfo->pSubresource, pLayout);
2806*61046927SAndroid Build Coastguard Worker }
2807*61046927SAndroid Build Coastguard Worker
anv_GetImageSubresourceLayout2KHR(VkDevice device,VkImage _image,const VkImageSubresource2KHR * pSubresource,VkSubresourceLayout2KHR * pLayout)2808*61046927SAndroid Build Coastguard Worker void anv_GetImageSubresourceLayout2KHR(
2809*61046927SAndroid Build Coastguard Worker VkDevice device,
2810*61046927SAndroid Build Coastguard Worker VkImage _image,
2811*61046927SAndroid Build Coastguard Worker const VkImageSubresource2KHR* pSubresource,
2812*61046927SAndroid Build Coastguard Worker VkSubresourceLayout2KHR* pLayout)
2813*61046927SAndroid Build Coastguard Worker {
2814*61046927SAndroid Build Coastguard Worker ANV_FROM_HANDLE(anv_image, image, _image);
2815*61046927SAndroid Build Coastguard Worker
2816*61046927SAndroid Build Coastguard Worker anv_get_image_subresource_layout(image, pSubresource, pLayout);
2817*61046927SAndroid Build Coastguard Worker }
2818*61046927SAndroid Build Coastguard Worker
2819*61046927SAndroid Build Coastguard Worker static VkImageUsageFlags
anv_image_flags_filter_for_queue(VkImageUsageFlags usages,VkQueueFlagBits queue_flags)2820*61046927SAndroid Build Coastguard Worker anv_image_flags_filter_for_queue(VkImageUsageFlags usages,
2821*61046927SAndroid Build Coastguard Worker VkQueueFlagBits queue_flags)
2822*61046927SAndroid Build Coastguard Worker {
2823*61046927SAndroid Build Coastguard Worker /* Eliminate graphics usages if the queue is not graphics capable */
2824*61046927SAndroid Build Coastguard Worker if (!(queue_flags & VK_QUEUE_GRAPHICS_BIT)) {
2825*61046927SAndroid Build Coastguard Worker usages &= ~(VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
2826*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT |
2827*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_TRANSIENT_ATTACHMENT_BIT |
2828*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT |
2829*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_FRAGMENT_DENSITY_MAP_BIT_EXT |
2830*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_FRAGMENT_SHADING_RATE_ATTACHMENT_BIT_KHR |
2831*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_ATTACHMENT_FEEDBACK_LOOP_BIT_EXT);
2832*61046927SAndroid Build Coastguard Worker }
2833*61046927SAndroid Build Coastguard Worker
2834*61046927SAndroid Build Coastguard Worker /* Eliminate sampling & storage usages if the queue is neither graphics nor
2835*61046927SAndroid Build Coastguard Worker * compute capable
2836*61046927SAndroid Build Coastguard Worker */
2837*61046927SAndroid Build Coastguard Worker if (!(queue_flags & (VK_QUEUE_GRAPHICS_BIT | VK_QUEUE_COMPUTE_BIT))) {
2838*61046927SAndroid Build Coastguard Worker usages &= ~(VK_IMAGE_USAGE_SAMPLED_BIT |
2839*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_STORAGE_BIT);
2840*61046927SAndroid Build Coastguard Worker }
2841*61046927SAndroid Build Coastguard Worker
2842*61046927SAndroid Build Coastguard Worker /* Eliminate transfer usages if the queue is neither transfer, compute or
2843*61046927SAndroid Build Coastguard Worker * graphics capable
2844*61046927SAndroid Build Coastguard Worker */
2845*61046927SAndroid Build Coastguard Worker if (!(queue_flags & (VK_QUEUE_TRANSFER_BIT |
2846*61046927SAndroid Build Coastguard Worker VK_QUEUE_COMPUTE_BIT |
2847*61046927SAndroid Build Coastguard Worker VK_QUEUE_GRAPHICS_BIT))) {
2848*61046927SAndroid Build Coastguard Worker usages &= ~(VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
2849*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_TRANSFER_DST_BIT);
2850*61046927SAndroid Build Coastguard Worker }
2851*61046927SAndroid Build Coastguard Worker
2852*61046927SAndroid Build Coastguard Worker return usages;
2853*61046927SAndroid Build Coastguard Worker }
2854*61046927SAndroid Build Coastguard Worker
2855*61046927SAndroid Build Coastguard Worker /**
2856*61046927SAndroid Build Coastguard Worker * This function returns the assumed isl_aux_state for a given VkImageLayout.
2857*61046927SAndroid Build Coastguard Worker * Because Vulkan image layouts don't map directly to isl_aux_state enums, the
2858*61046927SAndroid Build Coastguard Worker * returned enum is the assumed worst case.
2859*61046927SAndroid Build Coastguard Worker *
2860*61046927SAndroid Build Coastguard Worker * @param devinfo The device information of the Intel GPU.
2861*61046927SAndroid Build Coastguard Worker * @param image The image that may contain a collection of buffers.
2862*61046927SAndroid Build Coastguard Worker * @param aspect The aspect of the image to be accessed.
2863*61046927SAndroid Build Coastguard Worker * @param layout The current layout of the image aspect(s).
2864*61046927SAndroid Build Coastguard Worker *
2865*61046927SAndroid Build Coastguard Worker * @return The primary buffer that should be used for the given layout.
2866*61046927SAndroid Build Coastguard Worker */
2867*61046927SAndroid Build Coastguard Worker enum isl_aux_state ATTRIBUTE_PURE
anv_layout_to_aux_state(const struct intel_device_info * const devinfo,const struct anv_image * const image,const VkImageAspectFlagBits aspect,const VkImageLayout layout,const VkQueueFlagBits queue_flags)2868*61046927SAndroid Build Coastguard Worker anv_layout_to_aux_state(const struct intel_device_info * const devinfo,
2869*61046927SAndroid Build Coastguard Worker const struct anv_image * const image,
2870*61046927SAndroid Build Coastguard Worker const VkImageAspectFlagBits aspect,
2871*61046927SAndroid Build Coastguard Worker const VkImageLayout layout,
2872*61046927SAndroid Build Coastguard Worker const VkQueueFlagBits queue_flags)
2873*61046927SAndroid Build Coastguard Worker {
2874*61046927SAndroid Build Coastguard Worker /* Validate the inputs. */
2875*61046927SAndroid Build Coastguard Worker
2876*61046927SAndroid Build Coastguard Worker /* The devinfo is needed as the optimal buffer varies across generations. */
2877*61046927SAndroid Build Coastguard Worker assert(devinfo != NULL);
2878*61046927SAndroid Build Coastguard Worker
2879*61046927SAndroid Build Coastguard Worker /* The layout of a NULL image is not properly defined. */
2880*61046927SAndroid Build Coastguard Worker assert(image != NULL);
2881*61046927SAndroid Build Coastguard Worker
2882*61046927SAndroid Build Coastguard Worker /* The aspect must be exactly one of the image aspects. */
2883*61046927SAndroid Build Coastguard Worker assert(util_bitcount(aspect) == 1 && (aspect & image->vk.aspects));
2884*61046927SAndroid Build Coastguard Worker
2885*61046927SAndroid Build Coastguard Worker /* Determine the optimal buffer. */
2886*61046927SAndroid Build Coastguard Worker
2887*61046927SAndroid Build Coastguard Worker const uint32_t plane = anv_image_aspect_to_plane(image, aspect);
2888*61046927SAndroid Build Coastguard Worker
2889*61046927SAndroid Build Coastguard Worker /* If we don't have an aux buffer then aux state makes no sense */
2890*61046927SAndroid Build Coastguard Worker const enum isl_aux_usage aux_usage = image->planes[plane].aux_usage;
2891*61046927SAndroid Build Coastguard Worker assert(aux_usage != ISL_AUX_USAGE_NONE);
2892*61046927SAndroid Build Coastguard Worker
2893*61046927SAndroid Build Coastguard Worker /* All images that use an auxiliary surface are required to be tiled. */
2894*61046927SAndroid Build Coastguard Worker assert(image->planes[plane].primary_surface.isl.tiling != ISL_TILING_LINEAR);
2895*61046927SAndroid Build Coastguard Worker
2896*61046927SAndroid Build Coastguard Worker /* Handle a few special cases */
2897*61046927SAndroid Build Coastguard Worker switch (layout) {
2898*61046927SAndroid Build Coastguard Worker /* Invalid layouts */
2899*61046927SAndroid Build Coastguard Worker case VK_IMAGE_LAYOUT_MAX_ENUM:
2900*61046927SAndroid Build Coastguard Worker unreachable("Invalid image layout.");
2901*61046927SAndroid Build Coastguard Worker
2902*61046927SAndroid Build Coastguard Worker /* Undefined layouts
2903*61046927SAndroid Build Coastguard Worker *
2904*61046927SAndroid Build Coastguard Worker * The pre-initialized layout is equivalent to the undefined layout for
2905*61046927SAndroid Build Coastguard Worker * optimally-tiled images. We can only do color compression (CCS or HiZ)
2906*61046927SAndroid Build Coastguard Worker * on tiled images.
2907*61046927SAndroid Build Coastguard Worker */
2908*61046927SAndroid Build Coastguard Worker case VK_IMAGE_LAYOUT_UNDEFINED:
2909*61046927SAndroid Build Coastguard Worker case VK_IMAGE_LAYOUT_PREINITIALIZED:
2910*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_AUX_INVALID;
2911*61046927SAndroid Build Coastguard Worker
2912*61046927SAndroid Build Coastguard Worker case VK_IMAGE_LAYOUT_PRESENT_SRC_KHR: {
2913*61046927SAndroid Build Coastguard Worker assert(image->vk.aspects == VK_IMAGE_ASPECT_COLOR_BIT);
2914*61046927SAndroid Build Coastguard Worker
2915*61046927SAndroid Build Coastguard Worker enum isl_aux_state aux_state =
2916*61046927SAndroid Build Coastguard Worker isl_drm_modifier_get_default_aux_state(image->vk.drm_format_mod);
2917*61046927SAndroid Build Coastguard Worker
2918*61046927SAndroid Build Coastguard Worker switch (aux_state) {
2919*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_AUX_INVALID:
2920*61046927SAndroid Build Coastguard Worker /* The modifier does not support compression. But, if we arrived
2921*61046927SAndroid Build Coastguard Worker * here, then we have enabled compression on it anyway, in which case
2922*61046927SAndroid Build Coastguard Worker * we must resolve the aux surface before we release ownership to the
2923*61046927SAndroid Build Coastguard Worker * presentation engine (because, having no modifier, the presentation
2924*61046927SAndroid Build Coastguard Worker * engine will not be aware of the aux surface). The presentation
2925*61046927SAndroid Build Coastguard Worker * engine will not access the aux surface (because it is unware of
2926*61046927SAndroid Build Coastguard Worker * it), and so the aux surface will still be resolved when we
2927*61046927SAndroid Build Coastguard Worker * re-acquire ownership.
2928*61046927SAndroid Build Coastguard Worker *
2929*61046927SAndroid Build Coastguard Worker * Therefore, at ownership transfers in either direction, there does
2930*61046927SAndroid Build Coastguard Worker * exist an aux surface despite the lack of modifier and its state is
2931*61046927SAndroid Build Coastguard Worker * pass-through.
2932*61046927SAndroid Build Coastguard Worker */
2933*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_PASS_THROUGH;
2934*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_COMPRESSED_CLEAR:
2935*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_COMPRESSED_CLEAR;
2936*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_COMPRESSED_NO_CLEAR:
2937*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_COMPRESSED_NO_CLEAR;
2938*61046927SAndroid Build Coastguard Worker default:
2939*61046927SAndroid Build Coastguard Worker unreachable("unexpected isl_aux_state");
2940*61046927SAndroid Build Coastguard Worker }
2941*61046927SAndroid Build Coastguard Worker }
2942*61046927SAndroid Build Coastguard Worker
2943*61046927SAndroid Build Coastguard Worker default:
2944*61046927SAndroid Build Coastguard Worker break;
2945*61046927SAndroid Build Coastguard Worker }
2946*61046927SAndroid Build Coastguard Worker
2947*61046927SAndroid Build Coastguard Worker const bool read_only = vk_image_layout_is_read_only(layout, aspect);
2948*61046927SAndroid Build Coastguard Worker
2949*61046927SAndroid Build Coastguard Worker const VkImageUsageFlags image_aspect_usage =
2950*61046927SAndroid Build Coastguard Worker anv_image_flags_filter_for_queue(
2951*61046927SAndroid Build Coastguard Worker vk_image_usage(&image->vk, aspect), queue_flags);
2952*61046927SAndroid Build Coastguard Worker const VkImageUsageFlags usage =
2953*61046927SAndroid Build Coastguard Worker vk_image_layout_to_usage_flags(layout, aspect) & image_aspect_usage;
2954*61046927SAndroid Build Coastguard Worker
2955*61046927SAndroid Build Coastguard Worker bool aux_supported = true;
2956*61046927SAndroid Build Coastguard Worker bool clear_supported = isl_aux_usage_has_fast_clears(aux_usage);
2957*61046927SAndroid Build Coastguard Worker
2958*61046927SAndroid Build Coastguard Worker if ((usage & (VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT |
2959*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_ATTACHMENT_FEEDBACK_LOOP_BIT_EXT)) &&
2960*61046927SAndroid Build Coastguard Worker !read_only) {
2961*61046927SAndroid Build Coastguard Worker /* This image could be used as both an input attachment and a render
2962*61046927SAndroid Build Coastguard Worker * target (depth, stencil, or color) at the same time and this can cause
2963*61046927SAndroid Build Coastguard Worker * corruption.
2964*61046927SAndroid Build Coastguard Worker *
2965*61046927SAndroid Build Coastguard Worker * We currently only disable aux in this way for depth even though we
2966*61046927SAndroid Build Coastguard Worker * disable it for color in GL.
2967*61046927SAndroid Build Coastguard Worker *
2968*61046927SAndroid Build Coastguard Worker * TODO: Should we be disabling this in more cases?
2969*61046927SAndroid Build Coastguard Worker */
2970*61046927SAndroid Build Coastguard Worker if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT && devinfo->ver <= 9) {
2971*61046927SAndroid Build Coastguard Worker aux_supported = false;
2972*61046927SAndroid Build Coastguard Worker clear_supported = false;
2973*61046927SAndroid Build Coastguard Worker }
2974*61046927SAndroid Build Coastguard Worker }
2975*61046927SAndroid Build Coastguard Worker
2976*61046927SAndroid Build Coastguard Worker if (usage & (VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
2977*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_SAMPLED_BIT |
2978*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT)) {
2979*61046927SAndroid Build Coastguard Worker switch (aux_usage) {
2980*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_HIZ:
2981*61046927SAndroid Build Coastguard Worker if (!anv_can_sample_with_hiz(devinfo, image)) {
2982*61046927SAndroid Build Coastguard Worker aux_supported = false;
2983*61046927SAndroid Build Coastguard Worker clear_supported = false;
2984*61046927SAndroid Build Coastguard Worker }
2985*61046927SAndroid Build Coastguard Worker break;
2986*61046927SAndroid Build Coastguard Worker
2987*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_HIZ_CCS:
2988*61046927SAndroid Build Coastguard Worker aux_supported = false;
2989*61046927SAndroid Build Coastguard Worker clear_supported = false;
2990*61046927SAndroid Build Coastguard Worker break;
2991*61046927SAndroid Build Coastguard Worker
2992*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_HIZ_CCS_WT:
2993*61046927SAndroid Build Coastguard Worker break;
2994*61046927SAndroid Build Coastguard Worker
2995*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_CCS_D:
2996*61046927SAndroid Build Coastguard Worker aux_supported = false;
2997*61046927SAndroid Build Coastguard Worker clear_supported = false;
2998*61046927SAndroid Build Coastguard Worker break;
2999*61046927SAndroid Build Coastguard Worker
3000*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_MCS:
3001*61046927SAndroid Build Coastguard Worker if (!anv_can_sample_mcs_with_clear(devinfo, image))
3002*61046927SAndroid Build Coastguard Worker clear_supported = false;
3003*61046927SAndroid Build Coastguard Worker break;
3004*61046927SAndroid Build Coastguard Worker
3005*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_CCS_E:
3006*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_FCV_CCS_E:
3007*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_STC_CCS:
3008*61046927SAndroid Build Coastguard Worker break;
3009*61046927SAndroid Build Coastguard Worker
3010*61046927SAndroid Build Coastguard Worker default:
3011*61046927SAndroid Build Coastguard Worker unreachable("Unsupported aux usage");
3012*61046927SAndroid Build Coastguard Worker }
3013*61046927SAndroid Build Coastguard Worker }
3014*61046927SAndroid Build Coastguard Worker
3015*61046927SAndroid Build Coastguard Worker switch (aux_usage) {
3016*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_HIZ:
3017*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_HIZ_CCS:
3018*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_HIZ_CCS_WT:
3019*61046927SAndroid Build Coastguard Worker if (aux_supported) {
3020*61046927SAndroid Build Coastguard Worker assert(clear_supported);
3021*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_COMPRESSED_CLEAR;
3022*61046927SAndroid Build Coastguard Worker } else if (read_only) {
3023*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_RESOLVED;
3024*61046927SAndroid Build Coastguard Worker } else {
3025*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_AUX_INVALID;
3026*61046927SAndroid Build Coastguard Worker }
3027*61046927SAndroid Build Coastguard Worker
3028*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_CCS_D:
3029*61046927SAndroid Build Coastguard Worker /* We only support clear in exactly one state */
3030*61046927SAndroid Build Coastguard Worker if (layout == VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL ||
3031*61046927SAndroid Build Coastguard Worker layout == VK_IMAGE_LAYOUT_ATTACHMENT_OPTIMAL) {
3032*61046927SAndroid Build Coastguard Worker assert(aux_supported);
3033*61046927SAndroid Build Coastguard Worker assert(clear_supported);
3034*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_PARTIAL_CLEAR;
3035*61046927SAndroid Build Coastguard Worker } else {
3036*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_PASS_THROUGH;
3037*61046927SAndroid Build Coastguard Worker }
3038*61046927SAndroid Build Coastguard Worker
3039*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_CCS_E:
3040*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_FCV_CCS_E:
3041*61046927SAndroid Build Coastguard Worker if (aux_supported) {
3042*61046927SAndroid Build Coastguard Worker assert(clear_supported);
3043*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_COMPRESSED_CLEAR;
3044*61046927SAndroid Build Coastguard Worker } else {
3045*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_PASS_THROUGH;
3046*61046927SAndroid Build Coastguard Worker }
3047*61046927SAndroid Build Coastguard Worker
3048*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_MCS:
3049*61046927SAndroid Build Coastguard Worker assert(aux_supported);
3050*61046927SAndroid Build Coastguard Worker if (clear_supported) {
3051*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_COMPRESSED_CLEAR;
3052*61046927SAndroid Build Coastguard Worker } else {
3053*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_COMPRESSED_NO_CLEAR;
3054*61046927SAndroid Build Coastguard Worker }
3055*61046927SAndroid Build Coastguard Worker
3056*61046927SAndroid Build Coastguard Worker case ISL_AUX_USAGE_STC_CCS:
3057*61046927SAndroid Build Coastguard Worker assert(aux_supported);
3058*61046927SAndroid Build Coastguard Worker assert(!clear_supported);
3059*61046927SAndroid Build Coastguard Worker return ISL_AUX_STATE_COMPRESSED_NO_CLEAR;
3060*61046927SAndroid Build Coastguard Worker
3061*61046927SAndroid Build Coastguard Worker default:
3062*61046927SAndroid Build Coastguard Worker unreachable("Unsupported aux usage");
3063*61046927SAndroid Build Coastguard Worker }
3064*61046927SAndroid Build Coastguard Worker }
3065*61046927SAndroid Build Coastguard Worker
3066*61046927SAndroid Build Coastguard Worker /**
3067*61046927SAndroid Build Coastguard Worker * This function determines the optimal buffer to use for a given
3068*61046927SAndroid Build Coastguard Worker * VkImageLayout and other pieces of information needed to make that
3069*61046927SAndroid Build Coastguard Worker * determination. This does not determine the optimal buffer to use
3070*61046927SAndroid Build Coastguard Worker * during a resolve operation.
3071*61046927SAndroid Build Coastguard Worker *
3072*61046927SAndroid Build Coastguard Worker * @param devinfo The device information of the Intel GPU.
3073*61046927SAndroid Build Coastguard Worker * @param image The image that may contain a collection of buffers.
3074*61046927SAndroid Build Coastguard Worker * @param aspect The aspect of the image to be accessed.
3075*61046927SAndroid Build Coastguard Worker * @param usage The usage which describes how the image will be accessed.
3076*61046927SAndroid Build Coastguard Worker * @param layout The current layout of the image aspect(s).
3077*61046927SAndroid Build Coastguard Worker *
3078*61046927SAndroid Build Coastguard Worker * @return The primary buffer that should be used for the given layout.
3079*61046927SAndroid Build Coastguard Worker */
3080*61046927SAndroid Build Coastguard Worker enum isl_aux_usage ATTRIBUTE_PURE
anv_layout_to_aux_usage(const struct intel_device_info * const devinfo,const struct anv_image * const image,const VkImageAspectFlagBits aspect,const VkImageUsageFlagBits usage,const VkImageLayout layout,const VkQueueFlagBits queue_flags)3081*61046927SAndroid Build Coastguard Worker anv_layout_to_aux_usage(const struct intel_device_info * const devinfo,
3082*61046927SAndroid Build Coastguard Worker const struct anv_image * const image,
3083*61046927SAndroid Build Coastguard Worker const VkImageAspectFlagBits aspect,
3084*61046927SAndroid Build Coastguard Worker const VkImageUsageFlagBits usage,
3085*61046927SAndroid Build Coastguard Worker const VkImageLayout layout,
3086*61046927SAndroid Build Coastguard Worker const VkQueueFlagBits queue_flags)
3087*61046927SAndroid Build Coastguard Worker {
3088*61046927SAndroid Build Coastguard Worker const uint32_t plane = anv_image_aspect_to_plane(image, aspect);
3089*61046927SAndroid Build Coastguard Worker
3090*61046927SAndroid Build Coastguard Worker /* If there is no auxiliary surface allocated, we must use the one and only
3091*61046927SAndroid Build Coastguard Worker * main buffer.
3092*61046927SAndroid Build Coastguard Worker */
3093*61046927SAndroid Build Coastguard Worker if (image->planes[plane].aux_usage == ISL_AUX_USAGE_NONE)
3094*61046927SAndroid Build Coastguard Worker return ISL_AUX_USAGE_NONE;
3095*61046927SAndroid Build Coastguard Worker
3096*61046927SAndroid Build Coastguard Worker enum isl_aux_state aux_state =
3097*61046927SAndroid Build Coastguard Worker anv_layout_to_aux_state(devinfo, image, aspect, layout, queue_flags);
3098*61046927SAndroid Build Coastguard Worker
3099*61046927SAndroid Build Coastguard Worker switch (aux_state) {
3100*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_CLEAR:
3101*61046927SAndroid Build Coastguard Worker unreachable("We never use this state");
3102*61046927SAndroid Build Coastguard Worker
3103*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_PARTIAL_CLEAR:
3104*61046927SAndroid Build Coastguard Worker assert(image->vk.aspects & VK_IMAGE_ASPECT_ANY_COLOR_BIT_ANV);
3105*61046927SAndroid Build Coastguard Worker assert(image->planes[plane].aux_usage == ISL_AUX_USAGE_CCS_D);
3106*61046927SAndroid Build Coastguard Worker assert(image->vk.samples == 1);
3107*61046927SAndroid Build Coastguard Worker return ISL_AUX_USAGE_CCS_D;
3108*61046927SAndroid Build Coastguard Worker
3109*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_COMPRESSED_CLEAR:
3110*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_COMPRESSED_NO_CLEAR:
3111*61046927SAndroid Build Coastguard Worker return image->planes[plane].aux_usage;
3112*61046927SAndroid Build Coastguard Worker
3113*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_RESOLVED:
3114*61046927SAndroid Build Coastguard Worker /* We can only use RESOLVED in read-only layouts because any write will
3115*61046927SAndroid Build Coastguard Worker * either land us in AUX_INVALID or COMPRESSED_NO_CLEAR. We can do
3116*61046927SAndroid Build Coastguard Worker * writes in PASS_THROUGH without destroying it so that is allowed.
3117*61046927SAndroid Build Coastguard Worker */
3118*61046927SAndroid Build Coastguard Worker assert(vk_image_layout_is_read_only(layout, aspect));
3119*61046927SAndroid Build Coastguard Worker assert(util_is_power_of_two_or_zero(usage));
3120*61046927SAndroid Build Coastguard Worker if (usage == VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT) {
3121*61046927SAndroid Build Coastguard Worker /* If we have valid HiZ data and are using the image as a read-only
3122*61046927SAndroid Build Coastguard Worker * depth/stencil attachment, we should enable HiZ so that we can get
3123*61046927SAndroid Build Coastguard Worker * faster depth testing.
3124*61046927SAndroid Build Coastguard Worker */
3125*61046927SAndroid Build Coastguard Worker return image->planes[plane].aux_usage;
3126*61046927SAndroid Build Coastguard Worker } else {
3127*61046927SAndroid Build Coastguard Worker return ISL_AUX_USAGE_NONE;
3128*61046927SAndroid Build Coastguard Worker }
3129*61046927SAndroid Build Coastguard Worker
3130*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_PASS_THROUGH:
3131*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_AUX_INVALID:
3132*61046927SAndroid Build Coastguard Worker return ISL_AUX_USAGE_NONE;
3133*61046927SAndroid Build Coastguard Worker }
3134*61046927SAndroid Build Coastguard Worker
3135*61046927SAndroid Build Coastguard Worker unreachable("Invalid isl_aux_state");
3136*61046927SAndroid Build Coastguard Worker }
3137*61046927SAndroid Build Coastguard Worker
3138*61046927SAndroid Build Coastguard Worker /**
3139*61046927SAndroid Build Coastguard Worker * This function returns the level of unresolved fast-clear support of the
3140*61046927SAndroid Build Coastguard Worker * given image in the given VkImageLayout.
3141*61046927SAndroid Build Coastguard Worker *
3142*61046927SAndroid Build Coastguard Worker * @param devinfo The device information of the Intel GPU.
3143*61046927SAndroid Build Coastguard Worker * @param image The image that may contain a collection of buffers.
3144*61046927SAndroid Build Coastguard Worker * @param aspect The aspect of the image to be accessed.
3145*61046927SAndroid Build Coastguard Worker * @param usage The usage which describes how the image will be accessed.
3146*61046927SAndroid Build Coastguard Worker * @param layout The current layout of the image aspect(s).
3147*61046927SAndroid Build Coastguard Worker */
3148*61046927SAndroid Build Coastguard Worker enum anv_fast_clear_type ATTRIBUTE_PURE
anv_layout_to_fast_clear_type(const struct intel_device_info * const devinfo,const struct anv_image * const image,const VkImageAspectFlagBits aspect,const VkImageLayout layout,const VkQueueFlagBits queue_flags)3149*61046927SAndroid Build Coastguard Worker anv_layout_to_fast_clear_type(const struct intel_device_info * const devinfo,
3150*61046927SAndroid Build Coastguard Worker const struct anv_image * const image,
3151*61046927SAndroid Build Coastguard Worker const VkImageAspectFlagBits aspect,
3152*61046927SAndroid Build Coastguard Worker const VkImageLayout layout,
3153*61046927SAndroid Build Coastguard Worker const VkQueueFlagBits queue_flags)
3154*61046927SAndroid Build Coastguard Worker {
3155*61046927SAndroid Build Coastguard Worker if (INTEL_DEBUG(DEBUG_NO_FAST_CLEAR))
3156*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_NONE;
3157*61046927SAndroid Build Coastguard Worker
3158*61046927SAndroid Build Coastguard Worker /* Xe2+ platforms don't have fast clear type and can always support
3159*61046927SAndroid Build Coastguard Worker * arbitrary fast-clear values.
3160*61046927SAndroid Build Coastguard Worker */
3161*61046927SAndroid Build Coastguard Worker if (devinfo->ver >= 20)
3162*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_ANY;
3163*61046927SAndroid Build Coastguard Worker
3164*61046927SAndroid Build Coastguard Worker const uint32_t plane = anv_image_aspect_to_plane(image, aspect);
3165*61046927SAndroid Build Coastguard Worker
3166*61046927SAndroid Build Coastguard Worker /* If there is no auxiliary surface allocated, there are no fast-clears */
3167*61046927SAndroid Build Coastguard Worker if (image->planes[plane].aux_usage == ISL_AUX_USAGE_NONE)
3168*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_NONE;
3169*61046927SAndroid Build Coastguard Worker
3170*61046927SAndroid Build Coastguard Worker enum isl_aux_state aux_state =
3171*61046927SAndroid Build Coastguard Worker anv_layout_to_aux_state(devinfo, image, aspect, layout, queue_flags);
3172*61046927SAndroid Build Coastguard Worker
3173*61046927SAndroid Build Coastguard Worker const VkImageUsageFlags layout_usage =
3174*61046927SAndroid Build Coastguard Worker vk_image_layout_to_usage_flags(layout, aspect);
3175*61046927SAndroid Build Coastguard Worker
3176*61046927SAndroid Build Coastguard Worker switch (aux_state) {
3177*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_CLEAR:
3178*61046927SAndroid Build Coastguard Worker unreachable("We never use this state");
3179*61046927SAndroid Build Coastguard Worker
3180*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_PARTIAL_CLEAR:
3181*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_COMPRESSED_CLEAR:
3182*61046927SAndroid Build Coastguard Worker if (aspect == VK_IMAGE_ASPECT_DEPTH_BIT) {
3183*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_DEFAULT_VALUE;
3184*61046927SAndroid Build Coastguard Worker } else if (layout == VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL ||
3185*61046927SAndroid Build Coastguard Worker layout == VK_IMAGE_LAYOUT_ATTACHMENT_OPTIMAL) {
3186*61046927SAndroid Build Coastguard Worker /* The image might not support non zero fast clears when mutable. */
3187*61046927SAndroid Build Coastguard Worker if (!image->planes[plane].can_non_zero_fast_clear)
3188*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_DEFAULT_VALUE;
3189*61046927SAndroid Build Coastguard Worker
3190*61046927SAndroid Build Coastguard Worker /* When we're in a render pass we have the clear color data from the
3191*61046927SAndroid Build Coastguard Worker * VkRenderPassBeginInfo and we can use arbitrary clear colors. They
3192*61046927SAndroid Build Coastguard Worker * must get partially resolved before we leave the render pass.
3193*61046927SAndroid Build Coastguard Worker */
3194*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_ANY;
3195*61046927SAndroid Build Coastguard Worker } else if (layout_usage & (VK_IMAGE_USAGE_TRANSFER_SRC_BIT |
3196*61046927SAndroid Build Coastguard Worker VK_IMAGE_USAGE_TRANSFER_DST_BIT)) {
3197*61046927SAndroid Build Coastguard Worker /* Fast clear with non zero color is not supported during transfer
3198*61046927SAndroid Build Coastguard Worker * operations since transfer may do format reinterpretation.
3199*61046927SAndroid Build Coastguard Worker */
3200*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_DEFAULT_VALUE;
3201*61046927SAndroid Build Coastguard Worker } else if (image->planes[plane].aux_usage == ISL_AUX_USAGE_MCS ||
3202*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage == ISL_AUX_USAGE_CCS_E ||
3203*61046927SAndroid Build Coastguard Worker image->planes[plane].aux_usage == ISL_AUX_USAGE_FCV_CCS_E) {
3204*61046927SAndroid Build Coastguard Worker if (devinfo->ver >= 11) {
3205*61046927SAndroid Build Coastguard Worker /* The image might not support non zero fast clears when mutable. */
3206*61046927SAndroid Build Coastguard Worker if (!image->planes[plane].can_non_zero_fast_clear)
3207*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_DEFAULT_VALUE;
3208*61046927SAndroid Build Coastguard Worker
3209*61046927SAndroid Build Coastguard Worker /* On ICL and later, the sampler hardware uses a copy of the clear
3210*61046927SAndroid Build Coastguard Worker * value that is encoded as a pixel value. Therefore, we can use
3211*61046927SAndroid Build Coastguard Worker * any clear color we like for sampling.
3212*61046927SAndroid Build Coastguard Worker */
3213*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_ANY;
3214*61046927SAndroid Build Coastguard Worker } else {
3215*61046927SAndroid Build Coastguard Worker /* If the image has MCS or CCS_E enabled all the time then we can
3216*61046927SAndroid Build Coastguard Worker * use fast-clear as long as the clear color is the default value
3217*61046927SAndroid Build Coastguard Worker * of zero since this is the default value we program into every
3218*61046927SAndroid Build Coastguard Worker * surface state used for texturing.
3219*61046927SAndroid Build Coastguard Worker */
3220*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_DEFAULT_VALUE;
3221*61046927SAndroid Build Coastguard Worker }
3222*61046927SAndroid Build Coastguard Worker } else {
3223*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_NONE;
3224*61046927SAndroid Build Coastguard Worker }
3225*61046927SAndroid Build Coastguard Worker
3226*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_COMPRESSED_NO_CLEAR:
3227*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_RESOLVED:
3228*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_PASS_THROUGH:
3229*61046927SAndroid Build Coastguard Worker case ISL_AUX_STATE_AUX_INVALID:
3230*61046927SAndroid Build Coastguard Worker return ANV_FAST_CLEAR_NONE;
3231*61046927SAndroid Build Coastguard Worker }
3232*61046927SAndroid Build Coastguard Worker
3233*61046927SAndroid Build Coastguard Worker unreachable("Invalid isl_aux_state");
3234*61046927SAndroid Build Coastguard Worker }
3235*61046927SAndroid Build Coastguard Worker
3236*61046927SAndroid Build Coastguard Worker
3237*61046927SAndroid Build Coastguard Worker /**
3238*61046927SAndroid Build Coastguard Worker * This function determines if the layout & usage of an image can have
3239*61046927SAndroid Build Coastguard Worker * untracked aux writes. When we see a transition that matches this criteria,
3240*61046927SAndroid Build Coastguard Worker * we need to mark the image as compressed written so that our predicated
3241*61046927SAndroid Build Coastguard Worker * resolves work properly.
3242*61046927SAndroid Build Coastguard Worker *
3243*61046927SAndroid Build Coastguard Worker * @param devinfo The device information of the Intel GPU.
3244*61046927SAndroid Build Coastguard Worker * @param image The image that may contain a collection of buffers.
3245*61046927SAndroid Build Coastguard Worker * @param aspect The aspect of the image to be accessed.
3246*61046927SAndroid Build Coastguard Worker * @param layout The current layout of the image aspect(s).
3247*61046927SAndroid Build Coastguard Worker */
3248*61046927SAndroid Build Coastguard Worker bool
anv_layout_has_untracked_aux_writes(const struct intel_device_info * const devinfo,const struct anv_image * const image,const VkImageAspectFlagBits aspect,const VkImageLayout layout,const VkQueueFlagBits queue_flags)3249*61046927SAndroid Build Coastguard Worker anv_layout_has_untracked_aux_writes(const struct intel_device_info * const devinfo,
3250*61046927SAndroid Build Coastguard Worker const struct anv_image * const image,
3251*61046927SAndroid Build Coastguard Worker const VkImageAspectFlagBits aspect,
3252*61046927SAndroid Build Coastguard Worker const VkImageLayout layout,
3253*61046927SAndroid Build Coastguard Worker const VkQueueFlagBits queue_flags)
3254*61046927SAndroid Build Coastguard Worker {
3255*61046927SAndroid Build Coastguard Worker const VkImageUsageFlags image_aspect_usage =
3256*61046927SAndroid Build Coastguard Worker vk_image_usage(&image->vk, aspect);
3257*61046927SAndroid Build Coastguard Worker const VkImageUsageFlags usage =
3258*61046927SAndroid Build Coastguard Worker vk_image_layout_to_usage_flags(layout, aspect) & image_aspect_usage;
3259*61046927SAndroid Build Coastguard Worker
3260*61046927SAndroid Build Coastguard Worker /* Storage is the only usage where we do not write the image through a
3261*61046927SAndroid Build Coastguard Worker * render target but through a descriptor. Since VK_EXT_descriptor_indexing
3262*61046927SAndroid Build Coastguard Worker * and the update-after-bind feature, it has become impossible to track
3263*61046927SAndroid Build Coastguard Worker * writes to images in descriptor at the command buffer build time. So it's
3264*61046927SAndroid Build Coastguard Worker * not possible to mark an image as compressed like we do in
3265*61046927SAndroid Build Coastguard Worker * genX_cmd_buffer.c(EndRendering) or anv_blorp.c for all transfer
3266*61046927SAndroid Build Coastguard Worker * operations.
3267*61046927SAndroid Build Coastguard Worker */
3268*61046927SAndroid Build Coastguard Worker if (!(usage & VK_IMAGE_USAGE_STORAGE_BIT))
3269*61046927SAndroid Build Coastguard Worker return false;
3270*61046927SAndroid Build Coastguard Worker
3271*61046927SAndroid Build Coastguard Worker /* No AUX, no writes to the AUX surface :) */
3272*61046927SAndroid Build Coastguard Worker const uint32_t plane = anv_image_aspect_to_plane(image, aspect);
3273*61046927SAndroid Build Coastguard Worker const enum isl_aux_usage aux_usage = image->planes[plane].aux_usage;
3274*61046927SAndroid Build Coastguard Worker if (aux_usage == ISL_AUX_USAGE_NONE)
3275*61046927SAndroid Build Coastguard Worker return false;
3276*61046927SAndroid Build Coastguard Worker
3277*61046927SAndroid Build Coastguard Worker return true;
3278*61046927SAndroid Build Coastguard Worker }
3279*61046927SAndroid Build Coastguard Worker
anv_GetRenderingAreaGranularityKHR(VkDevice _device,const VkRenderingAreaInfoKHR * pRenderingAreaInfo,VkExtent2D * pGranularity)3280*61046927SAndroid Build Coastguard Worker void anv_GetRenderingAreaGranularityKHR(
3281*61046927SAndroid Build Coastguard Worker VkDevice _device,
3282*61046927SAndroid Build Coastguard Worker const VkRenderingAreaInfoKHR* pRenderingAreaInfo,
3283*61046927SAndroid Build Coastguard Worker VkExtent2D* pGranularity)
3284*61046927SAndroid Build Coastguard Worker {
3285*61046927SAndroid Build Coastguard Worker *pGranularity = (VkExtent2D) {
3286*61046927SAndroid Build Coastguard Worker .width = 1,
3287*61046927SAndroid Build Coastguard Worker .height = 1,
3288*61046927SAndroid Build Coastguard Worker };
3289*61046927SAndroid Build Coastguard Worker }
3290