radv/winsys: Set winsys bo priority on creation.
[mesa.git] / src / amd / vulkan / radv_descriptor_set.c
1 /*
2 * Copyright © 2016 Red Hat.
3 * Copyright © 2016 Bas Nieuwenhuizen
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the "Software"),
7 * to deal in the Software without restriction, including without limitation
8 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9 * and/or sell copies of the Software, and to permit persons to whom the
10 * Software is furnished to do so, subject to the following conditions:
11 *
12 * The above copyright notice and this permission notice (including the next
13 * paragraph) shall be included in all copies or substantial portions of the
14 * Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22 * IN THE SOFTWARE.
23 */
24 #include <assert.h>
25 #include <stdbool.h>
26 #include <string.h>
27 #include <unistd.h>
28 #include <fcntl.h>
29
30 #include "util/mesa-sha1.h"
31 #include "radv_private.h"
32 #include "sid.h"
33 #include "vk_util.h"
34
35
36 static bool has_equal_immutable_samplers(const VkSampler *samplers, uint32_t count)
37 {
38 if (!samplers)
39 return false;
40 for(uint32_t i = 1; i < count; ++i) {
41 if (memcmp(radv_sampler_from_handle(samplers[0])->state,
42 radv_sampler_from_handle(samplers[i])->state, 16)) {
43 return false;
44 }
45 }
46 return true;
47 }
48
49 static int binding_compare(const void* av, const void *bv)
50 {
51 const VkDescriptorSetLayoutBinding *a = (const VkDescriptorSetLayoutBinding*)av;
52 const VkDescriptorSetLayoutBinding *b = (const VkDescriptorSetLayoutBinding*)bv;
53
54 return (a->binding < b->binding) ? -1 : (a->binding > b->binding) ? 1 : 0;
55 }
56
57 static VkDescriptorSetLayoutBinding *
58 create_sorted_bindings(const VkDescriptorSetLayoutBinding *bindings, unsigned count) {
59 VkDescriptorSetLayoutBinding *sorted_bindings = malloc(count * sizeof(VkDescriptorSetLayoutBinding));
60 if (!sorted_bindings)
61 return NULL;
62
63 memcpy(sorted_bindings, bindings, count * sizeof(VkDescriptorSetLayoutBinding));
64
65 qsort(sorted_bindings, count, sizeof(VkDescriptorSetLayoutBinding), binding_compare);
66
67 return sorted_bindings;
68 }
69
70 VkResult radv_CreateDescriptorSetLayout(
71 VkDevice _device,
72 const VkDescriptorSetLayoutCreateInfo* pCreateInfo,
73 const VkAllocationCallbacks* pAllocator,
74 VkDescriptorSetLayout* pSetLayout)
75 {
76 RADV_FROM_HANDLE(radv_device, device, _device);
77 struct radv_descriptor_set_layout *set_layout;
78
79 assert(pCreateInfo->sType == VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO);
80 const VkDescriptorSetLayoutBindingFlagsCreateInfoEXT *variable_flags =
81 vk_find_struct_const(pCreateInfo->pNext, DESCRIPTOR_SET_LAYOUT_BINDING_FLAGS_CREATE_INFO_EXT);
82
83 uint32_t max_binding = 0;
84 uint32_t immutable_sampler_count = 0;
85 for (uint32_t j = 0; j < pCreateInfo->bindingCount; j++) {
86 max_binding = MAX2(max_binding, pCreateInfo->pBindings[j].binding);
87 if (pCreateInfo->pBindings[j].pImmutableSamplers)
88 immutable_sampler_count += pCreateInfo->pBindings[j].descriptorCount;
89 }
90
91 uint32_t samplers_offset = sizeof(struct radv_descriptor_set_layout) +
92 (max_binding + 1) * sizeof(set_layout->binding[0]);
93 size_t size = samplers_offset + immutable_sampler_count * 4 * sizeof(uint32_t);
94
95 set_layout = vk_alloc2(&device->alloc, pAllocator, size, 8,
96 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
97 if (!set_layout)
98 return vk_error(device->instance, VK_ERROR_OUT_OF_HOST_MEMORY);
99
100 set_layout->flags = pCreateInfo->flags;
101
102 /* We just allocate all the samplers at the end of the struct */
103 uint32_t *samplers = (uint32_t*)&set_layout->binding[max_binding + 1];
104
105 VkDescriptorSetLayoutBinding *bindings = create_sorted_bindings(pCreateInfo->pBindings,
106 pCreateInfo->bindingCount);
107 if (!bindings) {
108 vk_free2(&device->alloc, pAllocator, set_layout);
109 return vk_error(device->instance, VK_ERROR_OUT_OF_HOST_MEMORY);
110 }
111
112 set_layout->binding_count = max_binding + 1;
113 set_layout->shader_stages = 0;
114 set_layout->dynamic_shader_stages = 0;
115 set_layout->has_immutable_samplers = false;
116 set_layout->size = 0;
117
118 memset(set_layout->binding, 0, size - sizeof(struct radv_descriptor_set_layout));
119
120 uint32_t buffer_count = 0;
121 uint32_t dynamic_offset_count = 0;
122
123 for (uint32_t j = 0; j < pCreateInfo->bindingCount; j++) {
124 const VkDescriptorSetLayoutBinding *binding = bindings + j;
125 uint32_t b = binding->binding;
126 uint32_t alignment;
127 unsigned binding_buffer_count = 0;
128
129 switch (binding->descriptorType) {
130 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
131 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC:
132 assert(!(pCreateInfo->flags & VK_DESCRIPTOR_SET_LAYOUT_CREATE_PUSH_DESCRIPTOR_BIT_KHR));
133 set_layout->binding[b].dynamic_offset_count = 1;
134 set_layout->dynamic_shader_stages |= binding->stageFlags;
135 set_layout->binding[b].size = 0;
136 binding_buffer_count = 1;
137 alignment = 1;
138 break;
139 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER:
140 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER:
141 case VK_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER:
142 case VK_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER:
143 set_layout->binding[b].size = 16;
144 binding_buffer_count = 1;
145 alignment = 16;
146 break;
147 case VK_DESCRIPTOR_TYPE_STORAGE_IMAGE:
148 case VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE:
149 case VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT:
150 /* main descriptor + fmask descriptor */
151 set_layout->binding[b].size = 64;
152 binding_buffer_count = 1;
153 alignment = 32;
154 break;
155 case VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER:
156 /* main descriptor + fmask descriptor + sampler */
157 set_layout->binding[b].size = 96;
158 binding_buffer_count = 1;
159 alignment = 32;
160 break;
161 case VK_DESCRIPTOR_TYPE_SAMPLER:
162 set_layout->binding[b].size = 16;
163 alignment = 16;
164 break;
165 default:
166 unreachable("unknown descriptor type\n");
167 break;
168 }
169
170 set_layout->size = align(set_layout->size, alignment);
171 set_layout->binding[b].type = binding->descriptorType;
172 set_layout->binding[b].array_size = binding->descriptorCount;
173 set_layout->binding[b].offset = set_layout->size;
174 set_layout->binding[b].buffer_offset = buffer_count;
175 set_layout->binding[b].dynamic_offset_offset = dynamic_offset_count;
176
177 if (variable_flags && binding->binding < variable_flags->bindingCount &&
178 (variable_flags->pBindingFlags[binding->binding] & VK_DESCRIPTOR_BINDING_VARIABLE_DESCRIPTOR_COUNT_BIT_EXT)) {
179 assert(!binding->pImmutableSamplers); /* Terribly ill defined how many samplers are valid */
180 assert(binding->binding == max_binding);
181
182 set_layout->has_variable_descriptors = true;
183 }
184
185 if (binding->pImmutableSamplers) {
186 set_layout->binding[b].immutable_samplers_offset = samplers_offset;
187 set_layout->binding[b].immutable_samplers_equal =
188 has_equal_immutable_samplers(binding->pImmutableSamplers, binding->descriptorCount);
189 set_layout->has_immutable_samplers = true;
190
191
192 for (uint32_t i = 0; i < binding->descriptorCount; i++)
193 memcpy(samplers + 4 * i, &radv_sampler_from_handle(binding->pImmutableSamplers[i])->state, 16);
194
195 /* Don't reserve space for the samplers if they're not accessed. */
196 if (set_layout->binding[b].immutable_samplers_equal) {
197 if (binding->descriptorType == VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER)
198 set_layout->binding[b].size -= 32;
199 else if (binding->descriptorType == VK_DESCRIPTOR_TYPE_SAMPLER)
200 set_layout->binding[b].size -= 16;
201 }
202 samplers += 4 * binding->descriptorCount;
203 samplers_offset += 4 * sizeof(uint32_t) * binding->descriptorCount;
204 }
205
206 set_layout->size += binding->descriptorCount * set_layout->binding[b].size;
207 buffer_count += binding->descriptorCount * binding_buffer_count;
208 dynamic_offset_count += binding->descriptorCount *
209 set_layout->binding[b].dynamic_offset_count;
210 set_layout->shader_stages |= binding->stageFlags;
211 }
212
213 free(bindings);
214
215 set_layout->buffer_count = buffer_count;
216 set_layout->dynamic_offset_count = dynamic_offset_count;
217
218 *pSetLayout = radv_descriptor_set_layout_to_handle(set_layout);
219
220 return VK_SUCCESS;
221 }
222
223 void radv_DestroyDescriptorSetLayout(
224 VkDevice _device,
225 VkDescriptorSetLayout _set_layout,
226 const VkAllocationCallbacks* pAllocator)
227 {
228 RADV_FROM_HANDLE(radv_device, device, _device);
229 RADV_FROM_HANDLE(radv_descriptor_set_layout, set_layout, _set_layout);
230
231 if (!set_layout)
232 return;
233
234 vk_free2(&device->alloc, pAllocator, set_layout);
235 }
236
237 void radv_GetDescriptorSetLayoutSupport(VkDevice device,
238 const VkDescriptorSetLayoutCreateInfo* pCreateInfo,
239 VkDescriptorSetLayoutSupport* pSupport)
240 {
241 VkDescriptorSetLayoutBinding *bindings = create_sorted_bindings(pCreateInfo->pBindings,
242 pCreateInfo->bindingCount);
243 if (!bindings) {
244 pSupport->supported = false;
245 return;
246 }
247
248 const VkDescriptorSetLayoutBindingFlagsCreateInfoEXT *variable_flags =
249 vk_find_struct_const(pCreateInfo->pNext, DESCRIPTOR_SET_LAYOUT_BINDING_FLAGS_CREATE_INFO_EXT);
250 VkDescriptorSetVariableDescriptorCountLayoutSupportEXT *variable_count =
251 vk_find_struct((void*)pCreateInfo->pNext, DESCRIPTOR_SET_VARIABLE_DESCRIPTOR_COUNT_LAYOUT_SUPPORT_EXT);
252 if (variable_count) {
253 variable_count->maxVariableDescriptorCount = 0;
254 }
255
256 bool supported = true;
257 uint64_t size = 0;
258 for (uint32_t i = 0; i < pCreateInfo->bindingCount; i++) {
259 const VkDescriptorSetLayoutBinding *binding = bindings + i;
260
261 uint64_t descriptor_size = 0;
262 uint64_t descriptor_alignment = 1;
263 switch (binding->descriptorType) {
264 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
265 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC:
266 break;
267 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER:
268 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER:
269 case VK_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER:
270 case VK_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER:
271 descriptor_size = 16;
272 descriptor_alignment = 16;
273 break;
274 case VK_DESCRIPTOR_TYPE_STORAGE_IMAGE:
275 case VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE:
276 case VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT:
277 descriptor_size = 64;
278 descriptor_alignment = 32;
279 break;
280 case VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER:
281 if (!has_equal_immutable_samplers(binding->pImmutableSamplers, binding->descriptorCount)) {
282 descriptor_size = 64;
283 } else {
284 descriptor_size = 96;
285 }
286 descriptor_alignment = 32;
287 break;
288 case VK_DESCRIPTOR_TYPE_SAMPLER:
289 if (!has_equal_immutable_samplers(binding->pImmutableSamplers, binding->descriptorCount)) {
290 descriptor_size = 16;
291 descriptor_alignment = 16;
292 }
293 break;
294 default:
295 unreachable("unknown descriptor type\n");
296 break;
297 }
298
299 if (size && !align_u64(size, descriptor_alignment)) {
300 supported = false;
301 }
302 size = align_u64(size, descriptor_alignment);
303
304 uint64_t max_count = UINT64_MAX;
305 if (descriptor_size)
306 max_count = (UINT64_MAX - size) / descriptor_size;
307
308 if (max_count < binding->descriptorCount) {
309 supported = false;
310 }
311 if (variable_flags && binding->binding <variable_flags->bindingCount && variable_count &&
312 (variable_flags->pBindingFlags[binding->binding] & VK_DESCRIPTOR_BINDING_VARIABLE_DESCRIPTOR_COUNT_BIT_EXT)) {
313 variable_count->maxVariableDescriptorCount = MIN2(UINT32_MAX, max_count);
314 }
315 size += binding->descriptorCount * descriptor_size;
316 }
317
318 free(bindings);
319
320 pSupport->supported = supported;
321 }
322
323 /*
324 * Pipeline layouts. These have nothing to do with the pipeline. They are
325 * just multiple descriptor set layouts pasted together.
326 */
327
328 VkResult radv_CreatePipelineLayout(
329 VkDevice _device,
330 const VkPipelineLayoutCreateInfo* pCreateInfo,
331 const VkAllocationCallbacks* pAllocator,
332 VkPipelineLayout* pPipelineLayout)
333 {
334 RADV_FROM_HANDLE(radv_device, device, _device);
335 struct radv_pipeline_layout *layout;
336 struct mesa_sha1 ctx;
337
338 assert(pCreateInfo->sType == VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO);
339
340 layout = vk_alloc2(&device->alloc, pAllocator, sizeof(*layout), 8,
341 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
342 if (layout == NULL)
343 return vk_error(device->instance, VK_ERROR_OUT_OF_HOST_MEMORY);
344
345 layout->num_sets = pCreateInfo->setLayoutCount;
346
347 unsigned dynamic_offset_count = 0;
348 uint16_t dynamic_shader_stages = 0;
349
350
351 _mesa_sha1_init(&ctx);
352 for (uint32_t set = 0; set < pCreateInfo->setLayoutCount; set++) {
353 RADV_FROM_HANDLE(radv_descriptor_set_layout, set_layout,
354 pCreateInfo->pSetLayouts[set]);
355 layout->set[set].layout = set_layout;
356
357 layout->set[set].dynamic_offset_start = dynamic_offset_count;
358 for (uint32_t b = 0; b < set_layout->binding_count; b++) {
359 dynamic_offset_count += set_layout->binding[b].array_size * set_layout->binding[b].dynamic_offset_count;
360 dynamic_shader_stages |= set_layout->dynamic_shader_stages;
361 if (set_layout->binding[b].immutable_samplers_offset)
362 _mesa_sha1_update(&ctx, radv_immutable_samplers(set_layout, set_layout->binding + b),
363 set_layout->binding[b].array_size * 4 * sizeof(uint32_t));
364 }
365 _mesa_sha1_update(&ctx, set_layout->binding,
366 sizeof(set_layout->binding[0]) * set_layout->binding_count);
367 }
368
369 layout->dynamic_offset_count = dynamic_offset_count;
370 layout->dynamic_shader_stages = dynamic_shader_stages;
371 layout->push_constant_size = 0;
372
373 for (unsigned i = 0; i < pCreateInfo->pushConstantRangeCount; ++i) {
374 const VkPushConstantRange *range = pCreateInfo->pPushConstantRanges + i;
375 layout->push_constant_size = MAX2(layout->push_constant_size,
376 range->offset + range->size);
377 }
378
379 layout->push_constant_size = align(layout->push_constant_size, 16);
380 _mesa_sha1_update(&ctx, &layout->push_constant_size,
381 sizeof(layout->push_constant_size));
382 _mesa_sha1_final(&ctx, layout->sha1);
383 *pPipelineLayout = radv_pipeline_layout_to_handle(layout);
384
385 return VK_SUCCESS;
386 }
387
388 void radv_DestroyPipelineLayout(
389 VkDevice _device,
390 VkPipelineLayout _pipelineLayout,
391 const VkAllocationCallbacks* pAllocator)
392 {
393 RADV_FROM_HANDLE(radv_device, device, _device);
394 RADV_FROM_HANDLE(radv_pipeline_layout, pipeline_layout, _pipelineLayout);
395
396 if (!pipeline_layout)
397 return;
398 vk_free2(&device->alloc, pAllocator, pipeline_layout);
399 }
400
401 #define EMPTY 1
402
403 static VkResult
404 radv_descriptor_set_create(struct radv_device *device,
405 struct radv_descriptor_pool *pool,
406 const struct radv_descriptor_set_layout *layout,
407 const uint32_t *variable_count,
408 struct radv_descriptor_set **out_set)
409 {
410 struct radv_descriptor_set *set;
411 unsigned range_offset = sizeof(struct radv_descriptor_set) +
412 sizeof(struct radeon_winsys_bo *) * layout->buffer_count;
413 unsigned mem_size = range_offset +
414 sizeof(struct radv_descriptor_range) * layout->dynamic_offset_count;
415
416 if (pool->host_memory_base) {
417 if (pool->host_memory_end - pool->host_memory_ptr < mem_size)
418 return vk_error(device->instance, VK_ERROR_OUT_OF_POOL_MEMORY);
419
420 set = (struct radv_descriptor_set*)pool->host_memory_ptr;
421 pool->host_memory_ptr += mem_size;
422 } else {
423 set = vk_alloc2(&device->alloc, NULL, mem_size, 8,
424 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
425
426 if (!set)
427 return vk_error(device->instance, VK_ERROR_OUT_OF_HOST_MEMORY);
428 }
429
430 memset(set, 0, mem_size);
431
432 if (layout->dynamic_offset_count) {
433 set->dynamic_descriptors = (struct radv_descriptor_range*)((uint8_t*)set + range_offset);
434 }
435
436 set->layout = layout;
437 uint32_t layout_size = align_u32(layout->size, 32);
438 if (layout_size) {
439 set->size = layout_size;
440
441 if (!pool->host_memory_base && pool->entry_count == pool->max_entry_count) {
442 vk_free2(&device->alloc, NULL, set);
443 return vk_error(device->instance, VK_ERROR_OUT_OF_POOL_MEMORY);
444 }
445
446 /* try to allocate linearly first, so that we don't spend
447 * time looking for gaps if the app only allocates &
448 * resets via the pool. */
449 if (pool->current_offset + layout_size <= pool->size) {
450 set->bo = pool->bo;
451 set->mapped_ptr = (uint32_t*)(pool->mapped_ptr + pool->current_offset);
452 set->va = radv_buffer_get_va(set->bo) + pool->current_offset;
453 if (!pool->host_memory_base) {
454 pool->entries[pool->entry_count].offset = pool->current_offset;
455 pool->entries[pool->entry_count].size = layout_size;
456 pool->entries[pool->entry_count].set = set;
457 pool->entry_count++;
458 }
459 pool->current_offset += layout_size;
460 } else if (!pool->host_memory_base) {
461 uint64_t offset = 0;
462 int index;
463
464 for (index = 0; index < pool->entry_count; ++index) {
465 if (pool->entries[index].offset - offset >= layout_size)
466 break;
467 offset = pool->entries[index].offset + pool->entries[index].size;
468 }
469
470 if (pool->size - offset < layout_size) {
471 vk_free2(&device->alloc, NULL, set);
472 return vk_error(device->instance, VK_ERROR_OUT_OF_POOL_MEMORY);
473 }
474 set->bo = pool->bo;
475 set->mapped_ptr = (uint32_t*)(pool->mapped_ptr + offset);
476 set->va = radv_buffer_get_va(set->bo) + offset;
477 memmove(&pool->entries[index + 1], &pool->entries[index],
478 sizeof(pool->entries[0]) * (pool->entry_count - index));
479 pool->entries[index].offset = offset;
480 pool->entries[index].size = layout_size;
481 pool->entries[index].set = set;
482 pool->entry_count++;
483 } else
484 return vk_error(device->instance, VK_ERROR_OUT_OF_POOL_MEMORY);
485 }
486
487 if (layout->has_immutable_samplers) {
488 for (unsigned i = 0; i < layout->binding_count; ++i) {
489 if (!layout->binding[i].immutable_samplers_offset ||
490 layout->binding[i].immutable_samplers_equal)
491 continue;
492
493 unsigned offset = layout->binding[i].offset / 4;
494 if (layout->binding[i].type == VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER)
495 offset += 16;
496
497 const uint32_t *samplers = (const uint32_t*)((const char*)layout + layout->binding[i].immutable_samplers_offset);
498 for (unsigned j = 0; j < layout->binding[i].array_size; ++j) {
499 memcpy(set->mapped_ptr + offset, samplers + 4 * j, 16);
500 offset += layout->binding[i].size / 4;
501 }
502
503 }
504 }
505 *out_set = set;
506 return VK_SUCCESS;
507 }
508
509 static void
510 radv_descriptor_set_destroy(struct radv_device *device,
511 struct radv_descriptor_pool *pool,
512 struct radv_descriptor_set *set,
513 bool free_bo)
514 {
515 assert(!pool->host_memory_base);
516
517 if (free_bo && set->size && !pool->host_memory_base) {
518 uint32_t offset = (uint8_t*)set->mapped_ptr - pool->mapped_ptr;
519 for (int i = 0; i < pool->entry_count; ++i) {
520 if (pool->entries[i].offset == offset) {
521 memmove(&pool->entries[i], &pool->entries[i+1],
522 sizeof(pool->entries[i]) * (pool->entry_count - i - 1));
523 --pool->entry_count;
524 break;
525 }
526 }
527 }
528 vk_free2(&device->alloc, NULL, set);
529 }
530
531 VkResult radv_CreateDescriptorPool(
532 VkDevice _device,
533 const VkDescriptorPoolCreateInfo* pCreateInfo,
534 const VkAllocationCallbacks* pAllocator,
535 VkDescriptorPool* pDescriptorPool)
536 {
537 RADV_FROM_HANDLE(radv_device, device, _device);
538 struct radv_descriptor_pool *pool;
539 uint64_t size = sizeof(struct radv_descriptor_pool);
540 uint64_t bo_size = 0, bo_count = 0, range_count = 0;
541
542
543 for (unsigned i = 0; i < pCreateInfo->poolSizeCount; ++i) {
544 if (pCreateInfo->pPoolSizes[i].type != VK_DESCRIPTOR_TYPE_SAMPLER)
545 bo_count += pCreateInfo->pPoolSizes[i].descriptorCount;
546
547 switch(pCreateInfo->pPoolSizes[i].type) {
548 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
549 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC:
550 range_count += pCreateInfo->pPoolSizes[i].descriptorCount;
551 break;
552 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER:
553 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER:
554 case VK_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER:
555 case VK_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER:
556 case VK_DESCRIPTOR_TYPE_SAMPLER:
557 /* 32 as we may need to align for images */
558 bo_size += 32 * pCreateInfo->pPoolSizes[i].descriptorCount;
559 break;
560 case VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE:
561 case VK_DESCRIPTOR_TYPE_STORAGE_IMAGE:
562 case VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT:
563 bo_size += 64 * pCreateInfo->pPoolSizes[i].descriptorCount;
564 break;
565 case VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER:
566 bo_size += 96 * pCreateInfo->pPoolSizes[i].descriptorCount;
567 break;
568 default:
569 unreachable("unknown descriptor type\n");
570 break;
571 }
572 }
573
574 if (!(pCreateInfo->flags & VK_DESCRIPTOR_POOL_CREATE_FREE_DESCRIPTOR_SET_BIT)) {
575 uint64_t host_size = pCreateInfo->maxSets * sizeof(struct radv_descriptor_set);
576 host_size += sizeof(struct radeon_winsys_bo*) * bo_count;
577 host_size += sizeof(struct radv_descriptor_range) * range_count;
578 size += host_size;
579 } else {
580 size += sizeof(struct radv_descriptor_pool_entry) * pCreateInfo->maxSets;
581 }
582
583 pool = vk_alloc2(&device->alloc, pAllocator, size, 8,
584 VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
585 if (!pool)
586 return vk_error(device->instance, VK_ERROR_OUT_OF_HOST_MEMORY);
587
588 memset(pool, 0, sizeof(*pool));
589
590 if (!(pCreateInfo->flags & VK_DESCRIPTOR_POOL_CREATE_FREE_DESCRIPTOR_SET_BIT)) {
591 pool->host_memory_base = (uint8_t*)pool + sizeof(struct radv_descriptor_pool);
592 pool->host_memory_ptr = pool->host_memory_base;
593 pool->host_memory_end = (uint8_t*)pool + size;
594 }
595
596 if (bo_size) {
597 pool->bo = device->ws->buffer_create(device->ws, bo_size, 32,
598 RADEON_DOMAIN_VRAM,
599 RADEON_FLAG_NO_INTERPROCESS_SHARING |
600 RADEON_FLAG_READ_ONLY |
601 RADEON_FLAG_32BIT,
602 RADV_BO_PRIORITY_DESCRIPTOR);
603 pool->mapped_ptr = (uint8_t*)device->ws->buffer_map(pool->bo);
604 }
605 pool->size = bo_size;
606 pool->max_entry_count = pCreateInfo->maxSets;
607
608 *pDescriptorPool = radv_descriptor_pool_to_handle(pool);
609 return VK_SUCCESS;
610 }
611
612 void radv_DestroyDescriptorPool(
613 VkDevice _device,
614 VkDescriptorPool _pool,
615 const VkAllocationCallbacks* pAllocator)
616 {
617 RADV_FROM_HANDLE(radv_device, device, _device);
618 RADV_FROM_HANDLE(radv_descriptor_pool, pool, _pool);
619
620 if (!pool)
621 return;
622
623 if (!pool->host_memory_base) {
624 for(int i = 0; i < pool->entry_count; ++i) {
625 radv_descriptor_set_destroy(device, pool, pool->entries[i].set, false);
626 }
627 }
628
629 if (pool->bo)
630 device->ws->buffer_destroy(pool->bo);
631 vk_free2(&device->alloc, pAllocator, pool);
632 }
633
634 VkResult radv_ResetDescriptorPool(
635 VkDevice _device,
636 VkDescriptorPool descriptorPool,
637 VkDescriptorPoolResetFlags flags)
638 {
639 RADV_FROM_HANDLE(radv_device, device, _device);
640 RADV_FROM_HANDLE(radv_descriptor_pool, pool, descriptorPool);
641
642 if (!pool->host_memory_base) {
643 for(int i = 0; i < pool->entry_count; ++i) {
644 radv_descriptor_set_destroy(device, pool, pool->entries[i].set, false);
645 }
646 pool->entry_count = 0;
647 }
648
649 pool->current_offset = 0;
650 pool->host_memory_ptr = pool->host_memory_base;
651
652 return VK_SUCCESS;
653 }
654
655 VkResult radv_AllocateDescriptorSets(
656 VkDevice _device,
657 const VkDescriptorSetAllocateInfo* pAllocateInfo,
658 VkDescriptorSet* pDescriptorSets)
659 {
660 RADV_FROM_HANDLE(radv_device, device, _device);
661 RADV_FROM_HANDLE(radv_descriptor_pool, pool, pAllocateInfo->descriptorPool);
662
663 VkResult result = VK_SUCCESS;
664 uint32_t i;
665 struct radv_descriptor_set *set = NULL;
666
667 const VkDescriptorSetVariableDescriptorCountAllocateInfoEXT *variable_counts =
668 vk_find_struct_const(pAllocateInfo->pNext, DESCRIPTOR_SET_VARIABLE_DESCRIPTOR_COUNT_ALLOCATE_INFO_EXT);
669 const uint32_t zero = 0;
670
671 /* allocate a set of buffers for each shader to contain descriptors */
672 for (i = 0; i < pAllocateInfo->descriptorSetCount; i++) {
673 RADV_FROM_HANDLE(radv_descriptor_set_layout, layout,
674 pAllocateInfo->pSetLayouts[i]);
675
676 const uint32_t *variable_count = NULL;
677 if (variable_counts) {
678 if (i < variable_counts->descriptorSetCount)
679 variable_count = variable_counts->pDescriptorCounts + i;
680 else
681 variable_count = &zero;
682 }
683
684 assert(!(layout->flags & VK_DESCRIPTOR_SET_LAYOUT_CREATE_PUSH_DESCRIPTOR_BIT_KHR));
685
686 result = radv_descriptor_set_create(device, pool, layout, variable_count, &set);
687 if (result != VK_SUCCESS)
688 break;
689
690 pDescriptorSets[i] = radv_descriptor_set_to_handle(set);
691 }
692
693 if (result != VK_SUCCESS)
694 radv_FreeDescriptorSets(_device, pAllocateInfo->descriptorPool,
695 i, pDescriptorSets);
696 return result;
697 }
698
699 VkResult radv_FreeDescriptorSets(
700 VkDevice _device,
701 VkDescriptorPool descriptorPool,
702 uint32_t count,
703 const VkDescriptorSet* pDescriptorSets)
704 {
705 RADV_FROM_HANDLE(radv_device, device, _device);
706 RADV_FROM_HANDLE(radv_descriptor_pool, pool, descriptorPool);
707
708 for (uint32_t i = 0; i < count; i++) {
709 RADV_FROM_HANDLE(radv_descriptor_set, set, pDescriptorSets[i]);
710
711 if (set && !pool->host_memory_base)
712 radv_descriptor_set_destroy(device, pool, set, true);
713 }
714 return VK_SUCCESS;
715 }
716
717 static void write_texel_buffer_descriptor(struct radv_device *device,
718 struct radv_cmd_buffer *cmd_buffer,
719 unsigned *dst,
720 struct radeon_winsys_bo **buffer_list,
721 const VkBufferView _buffer_view)
722 {
723 RADV_FROM_HANDLE(radv_buffer_view, buffer_view, _buffer_view);
724
725 memcpy(dst, buffer_view->state, 4 * 4);
726
727 if (cmd_buffer)
728 radv_cs_add_buffer(device->ws, cmd_buffer->cs, buffer_view->bo);
729 else
730 *buffer_list = buffer_view->bo;
731 }
732
733 static void write_buffer_descriptor(struct radv_device *device,
734 struct radv_cmd_buffer *cmd_buffer,
735 unsigned *dst,
736 struct radeon_winsys_bo **buffer_list,
737 const VkDescriptorBufferInfo *buffer_info)
738 {
739 RADV_FROM_HANDLE(radv_buffer, buffer, buffer_info->buffer);
740 uint64_t va = radv_buffer_get_va(buffer->bo);
741 uint32_t range = buffer_info->range;
742
743 if (buffer_info->range == VK_WHOLE_SIZE)
744 range = buffer->size - buffer_info->offset;
745
746 va += buffer_info->offset + buffer->offset;
747 dst[0] = va;
748 dst[1] = S_008F04_BASE_ADDRESS_HI(va >> 32);
749 dst[2] = range;
750 dst[3] = S_008F0C_DST_SEL_X(V_008F0C_SQ_SEL_X) |
751 S_008F0C_DST_SEL_Y(V_008F0C_SQ_SEL_Y) |
752 S_008F0C_DST_SEL_Z(V_008F0C_SQ_SEL_Z) |
753 S_008F0C_DST_SEL_W(V_008F0C_SQ_SEL_W) |
754 S_008F0C_NUM_FORMAT(V_008F0C_BUF_NUM_FORMAT_FLOAT) |
755 S_008F0C_DATA_FORMAT(V_008F0C_BUF_DATA_FORMAT_32);
756
757 if (cmd_buffer)
758 radv_cs_add_buffer(device->ws, cmd_buffer->cs, buffer->bo);
759 else
760 *buffer_list = buffer->bo;
761 }
762
763 static void write_dynamic_buffer_descriptor(struct radv_device *device,
764 struct radv_descriptor_range *range,
765 struct radeon_winsys_bo **buffer_list,
766 const VkDescriptorBufferInfo *buffer_info)
767 {
768 RADV_FROM_HANDLE(radv_buffer, buffer, buffer_info->buffer);
769 uint64_t va = radv_buffer_get_va(buffer->bo);
770 unsigned size = buffer_info->range;
771
772 if (buffer_info->range == VK_WHOLE_SIZE)
773 size = buffer->size - buffer_info->offset;
774
775 va += buffer_info->offset + buffer->offset;
776 range->va = va;
777 range->size = size;
778
779 *buffer_list = buffer->bo;
780 }
781
782 static void
783 write_image_descriptor(struct radv_device *device,
784 struct radv_cmd_buffer *cmd_buffer,
785 unsigned *dst,
786 struct radeon_winsys_bo **buffer_list,
787 VkDescriptorType descriptor_type,
788 const VkDescriptorImageInfo *image_info)
789 {
790 RADV_FROM_HANDLE(radv_image_view, iview, image_info->imageView);
791 uint32_t *descriptor;
792
793 if (descriptor_type == VK_DESCRIPTOR_TYPE_STORAGE_IMAGE) {
794 descriptor = iview->storage_descriptor;
795 } else {
796 descriptor = iview->descriptor;
797 }
798
799 memcpy(dst, descriptor, 16 * 4);
800
801 if (cmd_buffer)
802 radv_cs_add_buffer(device->ws, cmd_buffer->cs, iview->bo);
803 else
804 *buffer_list = iview->bo;
805 }
806
807 static void
808 write_combined_image_sampler_descriptor(struct radv_device *device,
809 struct radv_cmd_buffer *cmd_buffer,
810 unsigned *dst,
811 struct radeon_winsys_bo **buffer_list,
812 VkDescriptorType descriptor_type,
813 const VkDescriptorImageInfo *image_info,
814 bool has_sampler)
815 {
816 RADV_FROM_HANDLE(radv_sampler, sampler, image_info->sampler);
817
818 write_image_descriptor(device, cmd_buffer, dst, buffer_list, descriptor_type, image_info);
819 /* copy over sampler state */
820 if (has_sampler)
821 memcpy(dst + 16, sampler->state, 16);
822 }
823
824 static void
825 write_sampler_descriptor(struct radv_device *device,
826 unsigned *dst,
827 const VkDescriptorImageInfo *image_info)
828 {
829 RADV_FROM_HANDLE(radv_sampler, sampler, image_info->sampler);
830
831 memcpy(dst, sampler->state, 16);
832 }
833
834 void radv_update_descriptor_sets(
835 struct radv_device* device,
836 struct radv_cmd_buffer* cmd_buffer,
837 VkDescriptorSet dstSetOverride,
838 uint32_t descriptorWriteCount,
839 const VkWriteDescriptorSet* pDescriptorWrites,
840 uint32_t descriptorCopyCount,
841 const VkCopyDescriptorSet* pDescriptorCopies)
842 {
843 uint32_t i, j;
844 for (i = 0; i < descriptorWriteCount; i++) {
845 const VkWriteDescriptorSet *writeset = &pDescriptorWrites[i];
846 RADV_FROM_HANDLE(radv_descriptor_set, set,
847 dstSetOverride ? dstSetOverride : writeset->dstSet);
848 const struct radv_descriptor_set_binding_layout *binding_layout =
849 set->layout->binding + writeset->dstBinding;
850 uint32_t *ptr = set->mapped_ptr;
851 struct radeon_winsys_bo **buffer_list = set->descriptors;
852 /* Immutable samplers are not copied into push descriptors when they are
853 * allocated, so if we are writing push descriptors we have to copy the
854 * immutable samplers into them now.
855 */
856 const bool copy_immutable_samplers = cmd_buffer &&
857 binding_layout->immutable_samplers_offset && !binding_layout->immutable_samplers_equal;
858 const uint32_t *samplers = radv_immutable_samplers(set->layout, binding_layout);
859
860 ptr += binding_layout->offset / 4;
861 ptr += binding_layout->size * writeset->dstArrayElement / 4;
862 buffer_list += binding_layout->buffer_offset;
863 buffer_list += writeset->dstArrayElement;
864 for (j = 0; j < writeset->descriptorCount; ++j) {
865 switch(writeset->descriptorType) {
866 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
867 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC: {
868 unsigned idx = writeset->dstArrayElement + j;
869 idx += binding_layout->dynamic_offset_offset;
870 assert(!(set->layout->flags & VK_DESCRIPTOR_SET_LAYOUT_CREATE_PUSH_DESCRIPTOR_BIT_KHR));
871 write_dynamic_buffer_descriptor(device, set->dynamic_descriptors + idx,
872 buffer_list, writeset->pBufferInfo + j);
873 break;
874 }
875 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER:
876 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER:
877 write_buffer_descriptor(device, cmd_buffer, ptr, buffer_list,
878 writeset->pBufferInfo + j);
879 break;
880 case VK_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER:
881 case VK_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER:
882 write_texel_buffer_descriptor(device, cmd_buffer, ptr, buffer_list,
883 writeset->pTexelBufferView[j]);
884 break;
885 case VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE:
886 case VK_DESCRIPTOR_TYPE_STORAGE_IMAGE:
887 case VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT:
888 write_image_descriptor(device, cmd_buffer, ptr, buffer_list,
889 writeset->descriptorType,
890 writeset->pImageInfo + j);
891 break;
892 case VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER:
893 write_combined_image_sampler_descriptor(device, cmd_buffer, ptr, buffer_list,
894 writeset->descriptorType,
895 writeset->pImageInfo + j,
896 !binding_layout->immutable_samplers_offset);
897 if (copy_immutable_samplers) {
898 const unsigned idx = writeset->dstArrayElement + j;
899 memcpy(ptr + 16, samplers + 4 * idx, 16);
900 }
901 break;
902 case VK_DESCRIPTOR_TYPE_SAMPLER:
903 if (!binding_layout->immutable_samplers_offset) {
904 write_sampler_descriptor(device, ptr,
905 writeset->pImageInfo + j);
906 } else if (copy_immutable_samplers) {
907 unsigned idx = writeset->dstArrayElement + j;
908 memcpy(ptr, samplers + 4 * idx, 16);
909 }
910 break;
911 default:
912 unreachable("unimplemented descriptor type");
913 break;
914 }
915 ptr += binding_layout->size / 4;
916 ++buffer_list;
917 }
918
919 }
920
921 for (i = 0; i < descriptorCopyCount; i++) {
922 const VkCopyDescriptorSet *copyset = &pDescriptorCopies[i];
923 RADV_FROM_HANDLE(radv_descriptor_set, src_set,
924 copyset->srcSet);
925 RADV_FROM_HANDLE(radv_descriptor_set, dst_set,
926 copyset->dstSet);
927 const struct radv_descriptor_set_binding_layout *src_binding_layout =
928 src_set->layout->binding + copyset->srcBinding;
929 const struct radv_descriptor_set_binding_layout *dst_binding_layout =
930 dst_set->layout->binding + copyset->dstBinding;
931 uint32_t *src_ptr = src_set->mapped_ptr;
932 uint32_t *dst_ptr = dst_set->mapped_ptr;
933 struct radeon_winsys_bo **src_buffer_list = src_set->descriptors;
934 struct radeon_winsys_bo **dst_buffer_list = dst_set->descriptors;
935
936 src_ptr += src_binding_layout->offset / 4;
937 dst_ptr += dst_binding_layout->offset / 4;
938
939 src_ptr += src_binding_layout->size * copyset->srcArrayElement / 4;
940 dst_ptr += dst_binding_layout->size * copyset->dstArrayElement / 4;
941
942 src_buffer_list += src_binding_layout->buffer_offset;
943 src_buffer_list += copyset->srcArrayElement;
944
945 dst_buffer_list += dst_binding_layout->buffer_offset;
946 dst_buffer_list += copyset->dstArrayElement;
947
948 for (j = 0; j < copyset->descriptorCount; ++j) {
949 switch (src_binding_layout->type) {
950 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
951 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC: {
952 unsigned src_idx = copyset->srcArrayElement + j;
953 unsigned dst_idx = copyset->dstArrayElement + j;
954 struct radv_descriptor_range *src_range, *dst_range;
955 src_idx += src_binding_layout->dynamic_offset_offset;
956 dst_idx += dst_binding_layout->dynamic_offset_offset;
957
958 src_range = src_set->dynamic_descriptors + src_idx;
959 dst_range = dst_set->dynamic_descriptors + dst_idx;
960 *dst_range = *src_range;
961 break;
962 }
963 default:
964 memcpy(dst_ptr, src_ptr, src_binding_layout->size);
965 }
966 src_ptr += src_binding_layout->size / 4;
967 dst_ptr += dst_binding_layout->size / 4;
968 dst_buffer_list[j] = src_buffer_list[j];
969 ++src_buffer_list;
970 ++dst_buffer_list;
971 }
972 }
973 }
974
975 void radv_UpdateDescriptorSets(
976 VkDevice _device,
977 uint32_t descriptorWriteCount,
978 const VkWriteDescriptorSet* pDescriptorWrites,
979 uint32_t descriptorCopyCount,
980 const VkCopyDescriptorSet* pDescriptorCopies)
981 {
982 RADV_FROM_HANDLE(radv_device, device, _device);
983
984 radv_update_descriptor_sets(device, NULL, VK_NULL_HANDLE, descriptorWriteCount, pDescriptorWrites,
985 descriptorCopyCount, pDescriptorCopies);
986 }
987
988 VkResult radv_CreateDescriptorUpdateTemplate(VkDevice _device,
989 const VkDescriptorUpdateTemplateCreateInfo *pCreateInfo,
990 const VkAllocationCallbacks *pAllocator,
991 VkDescriptorUpdateTemplate *pDescriptorUpdateTemplate)
992 {
993 RADV_FROM_HANDLE(radv_device, device, _device);
994 RADV_FROM_HANDLE(radv_descriptor_set_layout, set_layout, pCreateInfo->descriptorSetLayout);
995 const uint32_t entry_count = pCreateInfo->descriptorUpdateEntryCount;
996 const size_t size = sizeof(struct radv_descriptor_update_template) +
997 sizeof(struct radv_descriptor_update_template_entry) * entry_count;
998 struct radv_descriptor_update_template *templ;
999 uint32_t i;
1000
1001 templ = vk_alloc2(&device->alloc, pAllocator, size, 8, VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
1002 if (!templ)
1003 return vk_error(device->instance, VK_ERROR_OUT_OF_HOST_MEMORY);
1004
1005 templ->entry_count = entry_count;
1006 templ->bind_point = pCreateInfo->pipelineBindPoint;
1007
1008 for (i = 0; i < entry_count; i++) {
1009 const VkDescriptorUpdateTemplateEntry *entry = &pCreateInfo->pDescriptorUpdateEntries[i];
1010 const struct radv_descriptor_set_binding_layout *binding_layout =
1011 set_layout->binding + entry->dstBinding;
1012 const uint32_t buffer_offset = binding_layout->buffer_offset + entry->dstArrayElement;
1013 const uint32_t *immutable_samplers = NULL;
1014 uint32_t dst_offset;
1015 uint32_t dst_stride;
1016
1017 /* dst_offset is an offset into dynamic_descriptors when the descriptor
1018 is dynamic, and an offset into mapped_ptr otherwise */
1019 switch (entry->descriptorType) {
1020 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
1021 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC:
1022 assert(pCreateInfo->templateType == VK_DESCRIPTOR_UPDATE_TEMPLATE_TYPE_DESCRIPTOR_SET);
1023 dst_offset = binding_layout->dynamic_offset_offset + entry->dstArrayElement;
1024 dst_stride = 0; /* Not used */
1025 break;
1026 default:
1027 switch (entry->descriptorType) {
1028 case VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER:
1029 case VK_DESCRIPTOR_TYPE_SAMPLER:
1030 /* Immutable samplers are copied into push descriptors when they are pushed */
1031 if (pCreateInfo->templateType == VK_DESCRIPTOR_UPDATE_TEMPLATE_TYPE_PUSH_DESCRIPTORS_KHR &&
1032 binding_layout->immutable_samplers_offset && !binding_layout->immutable_samplers_equal) {
1033 immutable_samplers = radv_immutable_samplers(set_layout, binding_layout) + entry->dstArrayElement * 4;
1034 }
1035 break;
1036 default:
1037 break;
1038 }
1039 dst_offset = binding_layout->offset / 4 + binding_layout->size * entry->dstArrayElement / 4;
1040 dst_stride = binding_layout->size / 4;
1041 break;
1042 }
1043
1044 templ->entry[i] = (struct radv_descriptor_update_template_entry) {
1045 .descriptor_type = entry->descriptorType,
1046 .descriptor_count = entry->descriptorCount,
1047 .src_offset = entry->offset,
1048 .src_stride = entry->stride,
1049 .dst_offset = dst_offset,
1050 .dst_stride = dst_stride,
1051 .buffer_offset = buffer_offset,
1052 .has_sampler = !binding_layout->immutable_samplers_offset,
1053 .immutable_samplers = immutable_samplers
1054 };
1055 }
1056
1057 *pDescriptorUpdateTemplate = radv_descriptor_update_template_to_handle(templ);
1058 return VK_SUCCESS;
1059 }
1060
1061 void radv_DestroyDescriptorUpdateTemplate(VkDevice _device,
1062 VkDescriptorUpdateTemplate descriptorUpdateTemplate,
1063 const VkAllocationCallbacks *pAllocator)
1064 {
1065 RADV_FROM_HANDLE(radv_device, device, _device);
1066 RADV_FROM_HANDLE(radv_descriptor_update_template, templ, descriptorUpdateTemplate);
1067
1068 if (!templ)
1069 return;
1070
1071 vk_free2(&device->alloc, pAllocator, templ);
1072 }
1073
1074 void radv_update_descriptor_set_with_template(struct radv_device *device,
1075 struct radv_cmd_buffer *cmd_buffer,
1076 struct radv_descriptor_set *set,
1077 VkDescriptorUpdateTemplate descriptorUpdateTemplate,
1078 const void *pData)
1079 {
1080 RADV_FROM_HANDLE(radv_descriptor_update_template, templ, descriptorUpdateTemplate);
1081 uint32_t i;
1082
1083 for (i = 0; i < templ->entry_count; ++i) {
1084 struct radeon_winsys_bo **buffer_list = set->descriptors + templ->entry[i].buffer_offset;
1085 uint32_t *pDst = set->mapped_ptr + templ->entry[i].dst_offset;
1086 const uint8_t *pSrc = ((const uint8_t *) pData) + templ->entry[i].src_offset;
1087 uint32_t j;
1088
1089 for (j = 0; j < templ->entry[i].descriptor_count; ++j) {
1090 switch (templ->entry[i].descriptor_type) {
1091 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
1092 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC: {
1093 const unsigned idx = templ->entry[i].dst_offset + j;
1094 assert(!(set->layout->flags & VK_DESCRIPTOR_SET_LAYOUT_CREATE_PUSH_DESCRIPTOR_BIT_KHR));
1095 write_dynamic_buffer_descriptor(device, set->dynamic_descriptors + idx,
1096 buffer_list, (struct VkDescriptorBufferInfo *) pSrc);
1097 break;
1098 }
1099 case VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER:
1100 case VK_DESCRIPTOR_TYPE_STORAGE_BUFFER:
1101 write_buffer_descriptor(device, cmd_buffer, pDst, buffer_list,
1102 (struct VkDescriptorBufferInfo *) pSrc);
1103 break;
1104 case VK_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER:
1105 case VK_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER:
1106 write_texel_buffer_descriptor(device, cmd_buffer, pDst, buffer_list,
1107 *(VkBufferView *) pSrc);
1108 break;
1109 case VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE:
1110 case VK_DESCRIPTOR_TYPE_STORAGE_IMAGE:
1111 case VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT:
1112 write_image_descriptor(device, cmd_buffer, pDst, buffer_list,
1113 templ->entry[i].descriptor_type,
1114 (struct VkDescriptorImageInfo *) pSrc);
1115 break;
1116 case VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER:
1117 write_combined_image_sampler_descriptor(device, cmd_buffer, pDst, buffer_list,
1118 templ->entry[i].descriptor_type,
1119 (struct VkDescriptorImageInfo *) pSrc,
1120 templ->entry[i].has_sampler);
1121 if (templ->entry[i].immutable_samplers)
1122 memcpy(pDst + 16, templ->entry[i].immutable_samplers + 4 * j, 16);
1123 break;
1124 case VK_DESCRIPTOR_TYPE_SAMPLER:
1125 if (templ->entry[i].has_sampler)
1126 write_sampler_descriptor(device, pDst,
1127 (struct VkDescriptorImageInfo *) pSrc);
1128 else if (templ->entry[i].immutable_samplers)
1129 memcpy(pDst, templ->entry[i].immutable_samplers + 4 * j, 16);
1130 break;
1131 default:
1132 unreachable("unimplemented descriptor type");
1133 break;
1134 }
1135 pSrc += templ->entry[i].src_stride;
1136 pDst += templ->entry[i].dst_stride;
1137 ++buffer_list;
1138 }
1139 }
1140 }
1141
1142 void radv_UpdateDescriptorSetWithTemplate(VkDevice _device,
1143 VkDescriptorSet descriptorSet,
1144 VkDescriptorUpdateTemplate descriptorUpdateTemplate,
1145 const void *pData)
1146 {
1147 RADV_FROM_HANDLE(radv_device, device, _device);
1148 RADV_FROM_HANDLE(radv_descriptor_set, set, descriptorSet);
1149
1150 radv_update_descriptor_set_with_template(device, NULL, set, descriptorUpdateTemplate, pData);
1151 }
1152
1153
1154 VkResult radv_CreateSamplerYcbcrConversion(VkDevice device,
1155 const VkSamplerYcbcrConversionCreateInfo* pCreateInfo,
1156 const VkAllocationCallbacks* pAllocator,
1157 VkSamplerYcbcrConversion* pYcbcrConversion)
1158 {
1159 *pYcbcrConversion = VK_NULL_HANDLE;
1160 return VK_SUCCESS;
1161 }
1162
1163
1164 void radv_DestroySamplerYcbcrConversion(VkDevice device,
1165 VkSamplerYcbcrConversion ycbcrConversion,
1166 const VkAllocationCallbacks* pAllocator)
1167 {
1168 /* Do nothing. */
1169 }