1use alloc::{borrow::ToOwned as _, boxed::Box, collections::BTreeMap, sync::Arc, vec::Vec};
2use core::{ffi::CStr, marker::PhantomData};
3
4use ash::{ext, google, khr, vk};
5use parking_lot::Mutex;
6
7use crate::{vulkan::semaphore_list::SemaphoreList, AllocationSizes};
8
9use super::semaphore_list::SemaphoreListMode;
10
11fn depth_stencil_required_flags() -> vk::FormatFeatureFlags {
12 vk::FormatFeatureFlags::SAMPLED_IMAGE | vk::FormatFeatureFlags::DEPTH_STENCIL_ATTACHMENT
13}
14
15const INDEXING_FEATURES: wgt::Features = wgt::Features::TEXTURE_BINDING_ARRAY
16 .union(wgt::Features::BUFFER_BINDING_ARRAY)
17 .union(wgt::Features::STORAGE_RESOURCE_BINDING_ARRAY)
18 .union(wgt::Features::SAMPLED_TEXTURE_AND_STORAGE_BUFFER_ARRAY_NON_UNIFORM_INDEXING)
19 .union(wgt::Features::STORAGE_TEXTURE_ARRAY_NON_UNIFORM_INDEXING)
20 .union(wgt::Features::UNIFORM_BUFFER_BINDING_ARRAYS)
21 .union(wgt::Features::PARTIALLY_BOUND_BINDING_ARRAY);
22#[expect(rustdoc::private_intra_doc_links)]
23#[derive(Debug, Default)]
41pub struct PhysicalDeviceFeatures {
42 core: vk::PhysicalDeviceFeatures,
44
45 pub(super) descriptor_indexing:
47 Option<vk::PhysicalDeviceDescriptorIndexingFeaturesEXT<'static>>,
48
49 timeline_semaphore: Option<vk::PhysicalDeviceTimelineSemaphoreFeaturesKHR<'static>>,
51
52 image_robustness: Option<vk::PhysicalDeviceImageRobustnessFeaturesEXT<'static>>,
54
55 robustness2: Option<vk::PhysicalDeviceRobustness2FeaturesEXT<'static>>,
57
58 multiview: Option<vk::PhysicalDeviceMultiviewFeaturesKHR<'static>>,
60
61 sampler_ycbcr_conversion: Option<vk::PhysicalDeviceSamplerYcbcrConversionFeatures<'static>>,
63
64 astc_hdr: Option<vk::PhysicalDeviceTextureCompressionASTCHDRFeaturesEXT<'static>>,
66
67 shader_float16_int8: Option<vk::PhysicalDeviceShaderFloat16Int8Features<'static>>,
69
70 _16bit_storage: Option<vk::PhysicalDevice16BitStorageFeatures<'static>>,
72
73 acceleration_structure: Option<vk::PhysicalDeviceAccelerationStructureFeaturesKHR<'static>>,
75
76 buffer_device_address: Option<vk::PhysicalDeviceBufferDeviceAddressFeaturesKHR<'static>>,
91
92 ray_query: Option<vk::PhysicalDeviceRayQueryFeaturesKHR<'static>>,
102
103 zero_initialize_workgroup_memory:
106 Option<vk::PhysicalDeviceZeroInitializeWorkgroupMemoryFeatures<'static>>,
107 position_fetch: Option<vk::PhysicalDeviceRayTracingPositionFetchFeaturesKHR<'static>>,
108
109 shader_atomic_int64: Option<vk::PhysicalDeviceShaderAtomicInt64Features<'static>>,
111
112 shader_image_atomic_int64: Option<vk::PhysicalDeviceShaderImageAtomicInt64FeaturesEXT<'static>>,
114
115 shader_atomic_float: Option<vk::PhysicalDeviceShaderAtomicFloatFeaturesEXT<'static>>,
117
118 subgroup_size_control: Option<vk::PhysicalDeviceSubgroupSizeControlFeatures<'static>>,
120
121 maintenance4: Option<vk::PhysicalDeviceMaintenance4FeaturesKHR<'static>>,
123
124 mesh_shader: Option<vk::PhysicalDeviceMeshShaderFeaturesEXT<'static>>,
126
127 shader_integer_dot_product:
129 Option<vk::PhysicalDeviceShaderIntegerDotProductFeaturesKHR<'static>>,
130
131 shader_barycentrics: Option<vk::PhysicalDeviceFragmentShaderBarycentricFeaturesKHR<'static>>,
133
134 portability_subset: Option<vk::PhysicalDevicePortabilitySubsetFeaturesKHR<'static>>,
138
139 cooperative_matrix: Option<vk::PhysicalDeviceCooperativeMatrixFeaturesKHR<'static>>,
141
142 vulkan_memory_model: Option<vk::PhysicalDeviceVulkanMemoryModelFeaturesKHR<'static>>,
144
145 shader_draw_parameters: Option<vk::PhysicalDeviceShaderDrawParametersFeatures<'static>>,
146}
147
148impl PhysicalDeviceFeatures {
149 pub fn get_core(&self) -> vk::PhysicalDeviceFeatures {
150 self.core
151 }
152
153 pub fn add_to_device_create<'a>(
155 &'a mut self,
156 mut info: vk::DeviceCreateInfo<'a>,
157 ) -> vk::DeviceCreateInfo<'a> {
158 info = info.enabled_features(&self.core);
159 if let Some(ref mut feature) = self.descriptor_indexing {
160 info = info.push_next(feature);
161 }
162 if let Some(ref mut feature) = self.timeline_semaphore {
163 info = info.push_next(feature);
164 }
165 if let Some(ref mut feature) = self.image_robustness {
166 info = info.push_next(feature);
167 }
168 if let Some(ref mut feature) = self.robustness2 {
169 info = info.push_next(feature);
170 }
171 if let Some(ref mut feature) = self.multiview {
172 info = info.push_next(feature);
173 }
174 if let Some(ref mut feature) = self.astc_hdr {
175 info = info.push_next(feature);
176 }
177 if let Some(ref mut feature) = self.shader_float16_int8 {
178 info = info.push_next(feature);
179 }
180 if let Some(ref mut feature) = self._16bit_storage {
181 info = info.push_next(feature);
182 }
183 if let Some(ref mut feature) = self.zero_initialize_workgroup_memory {
184 info = info.push_next(feature);
185 }
186 if let Some(ref mut feature) = self.acceleration_structure {
187 info = info.push_next(feature);
188 }
189 if let Some(ref mut feature) = self.buffer_device_address {
190 info = info.push_next(feature);
191 }
192 if let Some(ref mut feature) = self.ray_query {
193 info = info.push_next(feature);
194 }
195 if let Some(ref mut feature) = self.shader_atomic_int64 {
196 info = info.push_next(feature);
197 }
198 if let Some(ref mut feature) = self.position_fetch {
199 info = info.push_next(feature);
200 }
201 if let Some(ref mut feature) = self.shader_image_atomic_int64 {
202 info = info.push_next(feature);
203 }
204 if let Some(ref mut feature) = self.shader_atomic_float {
205 info = info.push_next(feature);
206 }
207 if let Some(ref mut feature) = self.subgroup_size_control {
208 info = info.push_next(feature);
209 }
210 if let Some(ref mut feature) = self.maintenance4 {
211 info = info.push_next(feature);
212 }
213 if let Some(ref mut feature) = self.mesh_shader {
214 info = info.push_next(feature);
215 }
216 if let Some(ref mut feature) = self.shader_integer_dot_product {
217 info = info.push_next(feature);
218 }
219 if let Some(ref mut feature) = self.shader_barycentrics {
220 info = info.push_next(feature);
221 }
222 if let Some(ref mut feature) = self.portability_subset {
223 info = info.push_next(feature);
224 }
225 if let Some(ref mut feature) = self.cooperative_matrix {
226 info = info.push_next(feature);
227 }
228 if let Some(ref mut feature) = self.vulkan_memory_model {
229 info = info.push_next(feature);
230 }
231 if let Some(ref mut feature) = self.shader_draw_parameters {
232 info = info.push_next(feature);
233 }
234 info
235 }
236
237 fn supports_storage_input_output_16(&self) -> bool {
238 self._16bit_storage
239 .as_ref()
240 .map(|features| features.storage_input_output16 != 0)
241 .unwrap_or(false)
242 }
243
244 fn from_extensions_and_requested_features(
271 phd_capabilities: &PhysicalDeviceProperties,
272 phd_features: &PhysicalDeviceFeatures,
273 enabled_extensions: &[&'static CStr],
274 requested_features: wgt::Features,
275 downlevel_flags: wgt::DownlevelFlags,
276 private_caps: &super::PrivateCapabilities,
277 ) -> Self {
278 let device_api_version = phd_capabilities.device_api_version;
279 let needs_bindless = requested_features.intersects(
280 wgt::Features::TEXTURE_BINDING_ARRAY
281 | wgt::Features::BUFFER_BINDING_ARRAY
282 | wgt::Features::STORAGE_RESOURCE_BINDING_ARRAY
283 | wgt::Features::STORAGE_TEXTURE_ARRAY_NON_UNIFORM_INDEXING
284 | wgt::Features::SAMPLED_TEXTURE_AND_STORAGE_BUFFER_ARRAY_NON_UNIFORM_INDEXING,
285 );
286 let needs_partially_bound =
287 requested_features.intersects(wgt::Features::PARTIALLY_BOUND_BINDING_ARRAY);
288
289 Self {
290 core: vk::PhysicalDeviceFeatures::default()
293 .robust_buffer_access(private_caps.robust_buffer_access)
294 .independent_blend(downlevel_flags.contains(wgt::DownlevelFlags::INDEPENDENT_BLEND))
295 .sample_rate_shading(
296 downlevel_flags.contains(wgt::DownlevelFlags::MULTISAMPLED_SHADING),
297 )
298 .image_cube_array(
299 downlevel_flags.contains(wgt::DownlevelFlags::CUBE_ARRAY_TEXTURES),
300 )
301 .draw_indirect_first_instance(
302 requested_features.contains(wgt::Features::INDIRECT_FIRST_INSTANCE),
303 )
304 .multi_draw_indirect(phd_features.core.multi_draw_indirect != 0)
306 .fill_mode_non_solid(requested_features.intersects(
307 wgt::Features::POLYGON_MODE_LINE | wgt::Features::POLYGON_MODE_POINT,
308 ))
309 .sampler_anisotropy(
313 downlevel_flags.contains(wgt::DownlevelFlags::ANISOTROPIC_FILTERING),
314 )
315 .texture_compression_etc2(
316 requested_features.contains(wgt::Features::TEXTURE_COMPRESSION_ETC2),
317 )
318 .texture_compression_astc_ldr(
319 requested_features.contains(wgt::Features::TEXTURE_COMPRESSION_ASTC),
320 )
321 .texture_compression_bc(
322 requested_features.contains(wgt::Features::TEXTURE_COMPRESSION_BC),
323 )
325 .pipeline_statistics_query(
327 requested_features.contains(wgt::Features::PIPELINE_STATISTICS_QUERY),
328 )
329 .vertex_pipeline_stores_and_atomics(
330 requested_features.contains(wgt::Features::VERTEX_WRITABLE_STORAGE),
331 )
332 .fragment_stores_and_atomics(
333 downlevel_flags.contains(wgt::DownlevelFlags::FRAGMENT_WRITABLE_STORAGE),
334 )
335 .shader_uniform_buffer_array_dynamic_indexing(
338 requested_features.contains(wgt::Features::BUFFER_BINDING_ARRAY),
339 )
340 .shader_storage_buffer_array_dynamic_indexing(requested_features.contains(
341 wgt::Features::BUFFER_BINDING_ARRAY
342 | wgt::Features::STORAGE_RESOURCE_BINDING_ARRAY,
343 ))
344 .shader_sampled_image_array_dynamic_indexing(
345 requested_features.contains(wgt::Features::TEXTURE_BINDING_ARRAY),
346 )
347 .shader_storage_buffer_array_dynamic_indexing(requested_features.contains(
348 wgt::Features::TEXTURE_BINDING_ARRAY
349 | wgt::Features::STORAGE_RESOURCE_BINDING_ARRAY,
350 ))
351 .shader_clip_distance(requested_features.contains(wgt::Features::CLIP_DISTANCES))
353 .shader_float64(requested_features.contains(wgt::Features::SHADER_F64))
355 .shader_int64(requested_features.contains(wgt::Features::SHADER_INT64))
356 .shader_int16(requested_features.contains(wgt::Features::SHADER_I16))
357 .geometry_shader(requested_features.contains(wgt::Features::PRIMITIVE_INDEX))
359 .depth_clamp(requested_features.contains(wgt::Features::DEPTH_CLIP_CONTROL))
360 .dual_src_blend(requested_features.contains(wgt::Features::DUAL_SOURCE_BLENDING)),
361 descriptor_indexing: if requested_features.intersects(INDEXING_FEATURES) {
362 Some(
363 vk::PhysicalDeviceDescriptorIndexingFeaturesEXT::default()
364 .shader_sampled_image_array_non_uniform_indexing(needs_bindless)
365 .shader_storage_image_array_non_uniform_indexing(needs_bindless)
366 .shader_storage_buffer_array_non_uniform_indexing(needs_bindless)
367 .descriptor_binding_sampled_image_update_after_bind(needs_bindless)
368 .descriptor_binding_storage_image_update_after_bind(needs_bindless)
369 .descriptor_binding_storage_buffer_update_after_bind(needs_bindless)
370 .descriptor_binding_partially_bound(needs_partially_bound),
371 )
372 } else {
373 None
374 },
375 timeline_semaphore: if device_api_version >= vk::API_VERSION_1_2
376 || enabled_extensions.contains(&khr::timeline_semaphore::NAME)
377 {
378 Some(
379 vk::PhysicalDeviceTimelineSemaphoreFeaturesKHR::default()
380 .timeline_semaphore(private_caps.timeline_semaphores),
381 )
382 } else {
383 None
384 },
385 image_robustness: if device_api_version >= vk::API_VERSION_1_3
386 || enabled_extensions.contains(&ext::image_robustness::NAME)
387 {
388 Some(
389 vk::PhysicalDeviceImageRobustnessFeaturesEXT::default()
390 .robust_image_access(private_caps.robust_image_access),
391 )
392 } else {
393 None
394 },
395 robustness2: if enabled_extensions.contains(&ext::robustness2::NAME) {
396 Some(
397 vk::PhysicalDeviceRobustness2FeaturesEXT::default()
398 .robust_buffer_access2(private_caps.robust_buffer_access2)
399 .robust_image_access2(private_caps.robust_image_access2),
400 )
401 } else {
402 None
403 },
404 multiview: if device_api_version >= vk::API_VERSION_1_1
405 || enabled_extensions.contains(&khr::multiview::NAME)
406 {
407 Some(
408 vk::PhysicalDeviceMultiviewFeatures::default()
409 .multiview(requested_features.contains(wgt::Features::MULTIVIEW)),
410 )
411 } else {
412 None
413 },
414 sampler_ycbcr_conversion: if device_api_version >= vk::API_VERSION_1_1
415 || enabled_extensions.contains(&khr::sampler_ycbcr_conversion::NAME)
416 {
417 Some(
418 vk::PhysicalDeviceSamplerYcbcrConversionFeatures::default(), )
420 } else {
421 None
422 },
423 astc_hdr: if enabled_extensions.contains(&ext::texture_compression_astc_hdr::NAME) {
424 Some(
425 vk::PhysicalDeviceTextureCompressionASTCHDRFeaturesEXT::default()
426 .texture_compression_astc_hdr(true),
427 )
428 } else {
429 None
430 },
431 shader_float16_int8: match requested_features.contains(wgt::Features::SHADER_F16) {
432 shader_float16 if shader_float16 || private_caps.shader_int8 => Some(
433 vk::PhysicalDeviceShaderFloat16Int8Features::default()
434 .shader_float16(shader_float16)
435 .shader_int8(private_caps.shader_int8),
436 ),
437 _ => None,
438 },
439 _16bit_storage: if requested_features
440 .intersects(wgt::Features::SHADER_F16 | wgt::Features::SHADER_I16)
441 {
442 Some(
443 vk::PhysicalDevice16BitStorageFeatures::default()
444 .storage_buffer16_bit_access(true)
445 .storage_input_output16(phd_features.supports_storage_input_output_16())
446 .uniform_and_storage_buffer16_bit_access(true),
447 )
448 } else {
449 None
450 },
451 acceleration_structure: if enabled_extensions
452 .contains(&khr::acceleration_structure::NAME)
453 {
454 Some(
455 vk::PhysicalDeviceAccelerationStructureFeaturesKHR::default()
456 .acceleration_structure(true)
457 .descriptor_binding_acceleration_structure_update_after_bind(
458 requested_features
459 .contains(wgt::Features::ACCELERATION_STRUCTURE_BINDING_ARRAY),
460 ),
461 )
462 } else {
463 None
464 },
465 buffer_device_address: if enabled_extensions.contains(&khr::buffer_device_address::NAME)
466 {
467 Some(
468 vk::PhysicalDeviceBufferDeviceAddressFeaturesKHR::default()
469 .buffer_device_address(true),
470 )
471 } else {
472 None
473 },
474 ray_query: if enabled_extensions.contains(&khr::ray_query::NAME) {
475 Some(vk::PhysicalDeviceRayQueryFeaturesKHR::default().ray_query(true))
476 } else {
477 None
478 },
479 zero_initialize_workgroup_memory: if device_api_version >= vk::API_VERSION_1_3
480 || enabled_extensions.contains(&khr::zero_initialize_workgroup_memory::NAME)
481 {
482 Some(
483 vk::PhysicalDeviceZeroInitializeWorkgroupMemoryFeatures::default()
484 .shader_zero_initialize_workgroup_memory(
485 private_caps.zero_initialize_workgroup_memory,
486 ),
487 )
488 } else {
489 None
490 },
491 shader_atomic_int64: if device_api_version >= vk::API_VERSION_1_2
492 || enabled_extensions.contains(&khr::shader_atomic_int64::NAME)
493 {
494 let needed = requested_features.intersects(
495 wgt::Features::SHADER_INT64_ATOMIC_ALL_OPS
496 | wgt::Features::SHADER_INT64_ATOMIC_MIN_MAX,
497 );
498 Some(
499 vk::PhysicalDeviceShaderAtomicInt64Features::default()
500 .shader_buffer_int64_atomics(needed)
501 .shader_shared_int64_atomics(needed),
502 )
503 } else {
504 None
505 },
506 shader_image_atomic_int64: if enabled_extensions
507 .contains(&ext::shader_image_atomic_int64::NAME)
508 {
509 let needed = requested_features.intersects(wgt::Features::TEXTURE_INT64_ATOMIC);
510 Some(
511 vk::PhysicalDeviceShaderImageAtomicInt64FeaturesEXT::default()
512 .shader_image_int64_atomics(needed),
513 )
514 } else {
515 None
516 },
517 shader_atomic_float: if enabled_extensions.contains(&ext::shader_atomic_float::NAME) {
518 let needed = requested_features.contains(wgt::Features::SHADER_FLOAT32_ATOMIC);
519 Some(
520 vk::PhysicalDeviceShaderAtomicFloatFeaturesEXT::default()
521 .shader_buffer_float32_atomics(needed)
522 .shader_buffer_float32_atomic_add(needed),
523 )
524 } else {
525 None
526 },
527 subgroup_size_control: if device_api_version >= vk::API_VERSION_1_3
528 || enabled_extensions.contains(&ext::subgroup_size_control::NAME)
529 {
530 Some(
531 vk::PhysicalDeviceSubgroupSizeControlFeatures::default()
532 .subgroup_size_control(true),
533 )
534 } else {
535 None
536 },
537 position_fetch: if enabled_extensions.contains(&khr::ray_tracing_position_fetch::NAME) {
538 Some(
539 vk::PhysicalDeviceRayTracingPositionFetchFeaturesKHR::default()
540 .ray_tracing_position_fetch(true),
541 )
542 } else {
543 None
544 },
545 mesh_shader: if enabled_extensions.contains(&ext::mesh_shader::NAME) {
546 let needed = requested_features.contains(wgt::Features::EXPERIMENTAL_MESH_SHADER);
547 let multiview_needed =
548 requested_features.contains(wgt::Features::EXPERIMENTAL_MESH_SHADER_MULTIVIEW);
549 Some(
550 vk::PhysicalDeviceMeshShaderFeaturesEXT::default()
551 .mesh_shader(needed)
552 .task_shader(needed)
553 .multiview_mesh_shader(multiview_needed),
554 )
555 } else {
556 None
557 },
558 maintenance4: if device_api_version >= vk::API_VERSION_1_3
559 || enabled_extensions.contains(&khr::maintenance4::NAME)
560 {
561 let needed = requested_features.contains(wgt::Features::EXPERIMENTAL_MESH_SHADER);
562 Some(vk::PhysicalDeviceMaintenance4Features::default().maintenance4(needed))
563 } else {
564 None
565 },
566 shader_integer_dot_product: if device_api_version >= vk::API_VERSION_1_3
567 || enabled_extensions.contains(&khr::shader_integer_dot_product::NAME)
568 {
569 Some(
570 vk::PhysicalDeviceShaderIntegerDotProductFeaturesKHR::default()
571 .shader_integer_dot_product(private_caps.shader_integer_dot_product),
572 )
573 } else {
574 None
575 },
576 shader_barycentrics: if enabled_extensions
577 .contains(&khr::fragment_shader_barycentric::NAME)
578 {
579 let needed = requested_features.intersects(
580 wgt::Features::SHADER_BARYCENTRICS | wgt::Features::SHADER_PER_VERTEX,
581 );
582 Some(
583 vk::PhysicalDeviceFragmentShaderBarycentricFeaturesKHR::default()
584 .fragment_shader_barycentric(needed),
585 )
586 } else {
587 None
588 },
589 portability_subset: if enabled_extensions.contains(&khr::portability_subset::NAME) {
590 let multisample_array_needed =
591 requested_features.intersects(wgt::Features::MULTISAMPLE_ARRAY);
592
593 Some(
594 vk::PhysicalDevicePortabilitySubsetFeaturesKHR::default()
595 .multisample_array_image(multisample_array_needed),
596 )
597 } else {
598 None
599 },
600 cooperative_matrix: if enabled_extensions.contains(&khr::cooperative_matrix::NAME) {
601 let needed =
602 requested_features.contains(wgt::Features::EXPERIMENTAL_COOPERATIVE_MATRIX);
603 Some(
604 vk::PhysicalDeviceCooperativeMatrixFeaturesKHR::default()
605 .cooperative_matrix(needed),
606 )
607 } else {
608 None
609 },
610 vulkan_memory_model: if device_api_version >= vk::API_VERSION_1_2
611 || enabled_extensions.contains(&khr::vulkan_memory_model::NAME)
612 {
613 let needed =
614 requested_features.contains(wgt::Features::EXPERIMENTAL_COOPERATIVE_MATRIX);
615 Some(
616 vk::PhysicalDeviceVulkanMemoryModelFeaturesKHR::default()
617 .vulkan_memory_model(needed),
618 )
619 } else {
620 None
621 },
622 shader_draw_parameters: if device_api_version >= vk::API_VERSION_1_1 {
623 let needed = requested_features.contains(wgt::Features::SHADER_DRAW_INDEX);
624 Some(
625 vk::PhysicalDeviceShaderDrawParametersFeatures::default()
626 .shader_draw_parameters(needed),
627 )
628 } else {
629 None
630 },
631 }
632 }
633
634 fn to_wgpu(
643 &self,
644 instance: &ash::Instance,
645 phd: vk::PhysicalDevice,
646 caps: &PhysicalDeviceProperties,
647 queue_props: &vk::QueueFamilyProperties,
648 ) -> (wgt::Features, wgt::DownlevelFlags) {
649 use wgt::{DownlevelFlags as Df, Features as F};
650 let mut features = F::empty()
651 | F::MAPPABLE_PRIMARY_BUFFERS
652 | F::IMMEDIATES
653 | F::ADDRESS_MODE_CLAMP_TO_BORDER
654 | F::ADDRESS_MODE_CLAMP_TO_ZERO
655 | F::TEXTURE_ADAPTER_SPECIFIC_FORMAT_FEATURES
656 | F::CLEAR_TEXTURE
657 | F::PIPELINE_CACHE
658 | F::SHADER_EARLY_DEPTH_TEST
659 | F::TEXTURE_ATOMIC
660 | F::PASSTHROUGH_SHADERS
661 | F::MEMORY_DECORATION_COHERENT
662 | F::MEMORY_DECORATION_VOLATILE;
663
664 let mut dl_flags = Df::COMPUTE_SHADERS
665 | Df::BASE_VERTEX
666 | Df::READ_ONLY_DEPTH_STENCIL
667 | Df::NON_POWER_OF_TWO_MIPMAPPED_TEXTURES
668 | Df::COMPARISON_SAMPLERS
669 | Df::VERTEX_STORAGE
670 | Df::FRAGMENT_STORAGE
671 | Df::DEPTH_TEXTURE_AND_BUFFER_COPIES
672 | Df::BUFFER_BINDINGS_NOT_16_BYTE_ALIGNED
673 | Df::UNRESTRICTED_INDEX_BUFFER
674 | Df::INDIRECT_EXECUTION
675 | Df::VIEW_FORMATS
676 | Df::UNRESTRICTED_EXTERNAL_TEXTURE_COPIES
677 | Df::NONBLOCKING_QUERY_RESOLVE
678 | Df::SHADER_F16_IN_F32
679 | Df::MSL2_1;
680
681 dl_flags.set(
682 Df::SURFACE_VIEW_FORMATS,
683 caps.supports_extension(khr::swapchain_mutable_format::NAME),
684 );
685 dl_flags.set(Df::CUBE_ARRAY_TEXTURES, self.core.image_cube_array != 0);
686 dl_flags.set(Df::ANISOTROPIC_FILTERING, self.core.sampler_anisotropy != 0);
687 dl_flags.set(
688 Df::FRAGMENT_WRITABLE_STORAGE,
689 self.core.fragment_stores_and_atomics != 0,
690 );
691 dl_flags.set(Df::MULTISAMPLED_SHADING, self.core.sample_rate_shading != 0);
692 dl_flags.set(Df::INDEPENDENT_BLEND, self.core.independent_blend != 0);
693 dl_flags.set(
694 Df::FULL_DRAW_INDEX_UINT32,
695 self.core.full_draw_index_uint32 != 0,
696 );
697 dl_flags.set(Df::DEPTH_BIAS_CLAMP, self.core.depth_bias_clamp != 0);
698
699 features.set(
700 F::TIMESTAMP_QUERY
701 | F::TIMESTAMP_QUERY_INSIDE_ENCODERS
702 | F::TIMESTAMP_QUERY_INSIDE_PASSES,
703 queue_props.timestamp_valid_bits >= 36,
705 );
706 features.set(
707 F::INDIRECT_FIRST_INSTANCE,
708 self.core.draw_indirect_first_instance != 0,
709 );
710 features.set(F::POLYGON_MODE_LINE, self.core.fill_mode_non_solid != 0);
712 features.set(F::POLYGON_MODE_POINT, self.core.fill_mode_non_solid != 0);
713 features.set(
717 F::TEXTURE_COMPRESSION_ETC2,
718 self.core.texture_compression_etc2 != 0,
719 );
720 features.set(
721 F::TEXTURE_COMPRESSION_ASTC,
722 self.core.texture_compression_astc_ldr != 0,
723 );
724 features.set(
725 F::TEXTURE_COMPRESSION_BC,
726 self.core.texture_compression_bc != 0,
727 );
728 features.set(
729 F::TEXTURE_COMPRESSION_BC_SLICED_3D,
730 self.core.texture_compression_bc != 0, );
732 features.set(
733 F::PIPELINE_STATISTICS_QUERY,
734 self.core.pipeline_statistics_query != 0,
735 );
736 features.set(
737 F::VERTEX_WRITABLE_STORAGE,
738 self.core.vertex_pipeline_stores_and_atomics != 0,
739 );
740
741 features.set(F::SHADER_F64, self.core.shader_float64 != 0);
742 features.set(F::SHADER_INT64, self.core.shader_int64 != 0);
743 if let Some(ref bit16) = self._16bit_storage {
744 features.set(
745 F::SHADER_I16,
746 self.core.shader_int16 != 0
747 && bit16.storage_buffer16_bit_access != 0
748 && bit16.uniform_and_storage_buffer16_bit_access != 0,
749 );
750 }
751
752 features.set(F::PRIMITIVE_INDEX, self.core.geometry_shader != 0);
753
754 if let Some(ref shader_atomic_int64) = self.shader_atomic_int64 {
755 features.set(
756 F::SHADER_INT64_ATOMIC_ALL_OPS | F::SHADER_INT64_ATOMIC_MIN_MAX,
757 shader_atomic_int64.shader_buffer_int64_atomics != 0
758 && shader_atomic_int64.shader_shared_int64_atomics != 0,
759 );
760 }
761
762 if let Some(ref shader_image_atomic_int64) = self.shader_image_atomic_int64 {
763 features.set(
764 F::TEXTURE_INT64_ATOMIC,
765 shader_image_atomic_int64
766 .shader_image_int64_atomics(true)
767 .shader_image_int64_atomics
768 != 0,
769 );
770 }
771
772 if let Some(ref shader_atomic_float) = self.shader_atomic_float {
773 features.set(
774 F::SHADER_FLOAT32_ATOMIC,
775 shader_atomic_float.shader_buffer_float32_atomics != 0
776 && shader_atomic_float.shader_buffer_float32_atomic_add != 0,
777 );
778 }
779
780 if let Some(ref shader_barycentrics) = self.shader_barycentrics {
781 features.set(
782 F::SHADER_BARYCENTRICS | F::SHADER_PER_VERTEX,
783 shader_barycentrics.fragment_shader_barycentric != 0,
784 );
785 }
786
787 features.set(
790 F::MULTI_DRAW_INDIRECT_COUNT,
791 caps.supports_extension(khr::draw_indirect_count::NAME),
792 );
793 features.set(
794 F::CONSERVATIVE_RASTERIZATION,
795 caps.supports_extension(ext::conservative_rasterization::NAME),
796 );
797 features.set(
798 F::EXPERIMENTAL_RAY_HIT_VERTEX_RETURN,
799 caps.supports_extension(khr::ray_tracing_position_fetch::NAME),
800 );
801
802 if let Some(ref descriptor_indexing) = self.descriptor_indexing {
803 let supports_descriptor_indexing =
812 descriptor_indexing.shader_sampled_image_array_non_uniform_indexing != 0
814 && descriptor_indexing.descriptor_binding_sampled_image_update_after_bind != 0
815 && descriptor_indexing.shader_storage_image_array_non_uniform_indexing != 0
817 && descriptor_indexing.descriptor_binding_storage_image_update_after_bind != 0
818 && descriptor_indexing.shader_storage_buffer_array_non_uniform_indexing != 0
820 && descriptor_indexing.descriptor_binding_storage_buffer_update_after_bind != 0;
821
822 let descriptor_indexing_features = F::BUFFER_BINDING_ARRAY
823 | F::TEXTURE_BINDING_ARRAY
824 | F::STORAGE_RESOURCE_BINDING_ARRAY
825 | F::SAMPLED_TEXTURE_AND_STORAGE_BUFFER_ARRAY_NON_UNIFORM_INDEXING
826 | F::STORAGE_TEXTURE_ARRAY_NON_UNIFORM_INDEXING;
827
828 features.set(descriptor_indexing_features, supports_descriptor_indexing);
829
830 let supports_partially_bound =
831 descriptor_indexing.descriptor_binding_partially_bound != 0;
832
833 features.set(F::PARTIALLY_BOUND_BINDING_ARRAY, supports_partially_bound);
834 }
835
836 features.set(F::DEPTH_CLIP_CONTROL, self.core.depth_clamp != 0);
837 features.set(F::DUAL_SOURCE_BLENDING, self.core.dual_src_blend != 0);
838 features.set(F::CLIP_DISTANCES, self.core.shader_clip_distance != 0);
839
840 if let Some(ref multiview) = self.multiview {
841 features.set(F::MULTIVIEW, multiview.multiview != 0);
842 features.set(F::SELECTIVE_MULTIVIEW, multiview.multiview != 0);
843 }
844
845 features.set(
846 F::TEXTURE_FORMAT_16BIT_NORM,
847 is_format_16bit_norm_supported(instance, phd),
848 );
849
850 if let Some(ref astc_hdr) = self.astc_hdr {
851 features.set(
852 F::TEXTURE_COMPRESSION_ASTC_HDR,
853 astc_hdr.texture_compression_astc_hdr != 0,
854 );
855 }
856
857 if self.core.texture_compression_astc_ldr != 0 {
858 features.set(
859 F::TEXTURE_COMPRESSION_ASTC_SLICED_3D,
860 supports_astc_3d(instance, phd),
861 );
862 }
863
864 if let (Some(ref f16_i8), Some(ref bit16)) = (self.shader_float16_int8, self._16bit_storage)
865 {
866 features.set(
869 F::SHADER_F16,
870 f16_i8.shader_float16 != 0
871 && bit16.storage_buffer16_bit_access != 0
872 && bit16.uniform_and_storage_buffer16_bit_access != 0,
873 );
874 }
875
876 if let Some(ref subgroup) = caps.subgroup {
877 if (caps.device_api_version >= vk::API_VERSION_1_3
878 || caps.supports_extension(ext::subgroup_size_control::NAME))
879 && subgroup.supported_operations.contains(
880 vk::SubgroupFeatureFlags::BASIC
881 | vk::SubgroupFeatureFlags::VOTE
882 | vk::SubgroupFeatureFlags::ARITHMETIC
883 | vk::SubgroupFeatureFlags::BALLOT
884 | vk::SubgroupFeatureFlags::SHUFFLE
885 | vk::SubgroupFeatureFlags::SHUFFLE_RELATIVE
886 | vk::SubgroupFeatureFlags::QUAD,
887 )
888 {
889 features.set(
890 F::SUBGROUP,
891 subgroup
892 .supported_stages
893 .contains(vk::ShaderStageFlags::COMPUTE | vk::ShaderStageFlags::FRAGMENT),
894 );
895 features.set(
896 F::SUBGROUP_VERTEX,
897 subgroup
898 .supported_stages
899 .contains(vk::ShaderStageFlags::VERTEX),
900 );
901 features.insert(F::SUBGROUP_BARRIER);
902 }
903 }
904
905 let supports_depth_format = |format| {
906 supports_format(
907 instance,
908 phd,
909 format,
910 vk::ImageTiling::OPTIMAL,
911 depth_stencil_required_flags(),
912 )
913 };
914
915 let texture_s8 = supports_depth_format(vk::Format::S8_UINT);
916 let texture_d32 = supports_depth_format(vk::Format::D32_SFLOAT);
917 let texture_d24_s8 = supports_depth_format(vk::Format::D24_UNORM_S8_UINT);
918 let texture_d32_s8 = supports_depth_format(vk::Format::D32_SFLOAT_S8_UINT);
919
920 let stencil8 = texture_s8 || texture_d24_s8;
921 let depth24_plus_stencil8 = texture_d24_s8 || texture_d32_s8;
922
923 dl_flags.set(
924 Df::WEBGPU_TEXTURE_FORMAT_SUPPORT,
925 stencil8 && depth24_plus_stencil8 && texture_d32,
926 );
927
928 features.set(F::DEPTH32FLOAT_STENCIL8, texture_d32_s8);
929
930 let supports_acceleration_structures = caps
931 .supports_extension(khr::deferred_host_operations::NAME)
932 && caps.supports_extension(khr::acceleration_structure::NAME)
933 && caps.supports_extension(khr::buffer_device_address::NAME);
934
935 let supports_ray_query =
936 supports_acceleration_structures && caps.supports_extension(khr::ray_query::NAME);
937 let supports_acceleration_structure_binding_array = supports_ray_query
938 && self
939 .acceleration_structure
940 .as_ref()
941 .is_some_and(|features| {
942 features.descriptor_binding_acceleration_structure_update_after_bind != 0
943 });
944
945 features.set(
946 F::EXPERIMENTAL_RAY_QUERY
947 | F::EXTENDED_ACCELERATION_STRUCTURE_VERTEX_FORMATS,
950 supports_ray_query,
951 );
952
953 features.set(
958 F::ACCELERATION_STRUCTURE_BINDING_ARRAY,
959 supports_acceleration_structure_binding_array,
960 );
961
962 let rg11b10ufloat_renderable = supports_format(
963 instance,
964 phd,
965 vk::Format::B10G11R11_UFLOAT_PACK32,
966 vk::ImageTiling::OPTIMAL,
967 vk::FormatFeatureFlags::COLOR_ATTACHMENT
968 | vk::FormatFeatureFlags::COLOR_ATTACHMENT_BLEND,
969 );
970 features.set(F::RG11B10UFLOAT_RENDERABLE, rg11b10ufloat_renderable);
971
972 features.set(
973 F::BGRA8UNORM_STORAGE,
974 supports_bgra8unorm_storage(instance, phd, caps.device_api_version),
975 );
976
977 features.set(
978 F::FLOAT32_FILTERABLE,
979 is_float32_filterable_supported(instance, phd),
980 );
981
982 features.set(
983 F::FLOAT32_BLENDABLE,
984 is_float32_blendable_supported(instance, phd),
985 );
986
987 if let Some(ref _sampler_ycbcr_conversion) = self.sampler_ycbcr_conversion {
988 features.set(
989 F::TEXTURE_FORMAT_NV12,
990 supports_format(
991 instance,
992 phd,
993 vk::Format::G8_B8R8_2PLANE_420_UNORM,
994 vk::ImageTiling::OPTIMAL,
995 vk::FormatFeatureFlags::SAMPLED_IMAGE
996 | vk::FormatFeatureFlags::TRANSFER_SRC
997 | vk::FormatFeatureFlags::TRANSFER_DST,
998 ) && !caps
999 .driver
1000 .map(|driver| driver.driver_id == vk::DriverId::MOLTENVK)
1001 .unwrap_or_default(),
1002 );
1003 }
1004
1005 if let Some(ref _sampler_ycbcr_conversion) = self.sampler_ycbcr_conversion {
1006 features.set(
1007 F::TEXTURE_FORMAT_P010,
1008 supports_format(
1009 instance,
1010 phd,
1011 vk::Format::G10X6_B10X6R10X6_2PLANE_420_UNORM_3PACK16,
1012 vk::ImageTiling::OPTIMAL,
1013 vk::FormatFeatureFlags::SAMPLED_IMAGE
1014 | vk::FormatFeatureFlags::TRANSFER_SRC
1015 | vk::FormatFeatureFlags::TRANSFER_DST,
1016 ) && !caps
1017 .driver
1018 .map(|driver| driver.driver_id == vk::DriverId::MOLTENVK)
1019 .unwrap_or_default(),
1020 );
1021 }
1022
1023 features.set(
1024 F::VULKAN_GOOGLE_DISPLAY_TIMING,
1025 caps.supports_extension(google::display_timing::NAME),
1026 );
1027
1028 features.set(
1029 F::VULKAN_EXTERNAL_MEMORY_WIN32,
1030 caps.supports_extension(khr::external_memory_win32::NAME),
1031 );
1032 features.set(
1033 F::VULKAN_EXTERNAL_MEMORY_FD,
1034 caps.supports_extension(khr::external_memory_fd::NAME),
1035 );
1036 features.set(
1037 F::VULKAN_EXTERNAL_MEMORY_DMA_BUF,
1038 caps.supports_extension(khr::external_memory_fd::NAME)
1039 && caps.supports_extension(ext::external_memory_dma_buf::NAME)
1040 && caps.supports_extension(ext::image_drm_format_modifier::NAME),
1041 );
1042 features.set(
1043 F::EXPERIMENTAL_MESH_SHADER,
1044 caps.supports_extension(ext::mesh_shader::NAME),
1045 );
1046 features.set(
1047 F::EXPERIMENTAL_MESH_SHADER_POINTS,
1048 caps.supports_extension(ext::mesh_shader::NAME),
1049 );
1050 if let Some(ref mesh_shader) = self.mesh_shader {
1051 features.set(
1052 F::EXPERIMENTAL_MESH_SHADER_MULTIVIEW,
1053 mesh_shader.multiview_mesh_shader != 0,
1054 );
1055 }
1056
1057 features.set(
1059 F::MULTISAMPLE_ARRAY,
1060 self.portability_subset
1061 .map(|p| p.multisample_array_image == vk::TRUE)
1062 .unwrap_or(true),
1063 );
1064 features.set(
1066 F::EXPERIMENTAL_COOPERATIVE_MATRIX,
1067 !caps.cooperative_matrix_properties.is_empty(),
1068 );
1069
1070 features.set(
1071 F::SHADER_DRAW_INDEX,
1072 self.shader_draw_parameters
1073 .is_some_and(|a| a.shader_draw_parameters != 0)
1074 || caps.supports_extension(c"VK_KHR_shader_draw_parameters"),
1075 );
1076
1077 (features, dl_flags)
1078 }
1079}
1080
1081#[derive(Default, Debug)]
1102pub struct PhysicalDeviceProperties {
1103 supported_extensions: Vec<vk::ExtensionProperties>,
1106
1107 properties: vk::PhysicalDeviceProperties,
1110
1111 maintenance_3: Option<vk::PhysicalDeviceMaintenance3Properties<'static>>,
1114
1115 maintenance_4: Option<vk::PhysicalDeviceMaintenance4Properties<'static>>,
1118
1119 descriptor_indexing: Option<vk::PhysicalDeviceDescriptorIndexingPropertiesEXT<'static>>,
1122
1123 acceleration_structure: Option<vk::PhysicalDeviceAccelerationStructurePropertiesKHR<'static>>,
1126
1127 driver: Option<vk::PhysicalDeviceDriverPropertiesKHR<'static>>,
1130
1131 subgroup: Option<vk::PhysicalDeviceSubgroupProperties<'static>>,
1133
1134 subgroup_size_control: Option<vk::PhysicalDeviceSubgroupSizeControlProperties<'static>>,
1137
1138 robustness2: Option<vk::PhysicalDeviceRobustness2PropertiesEXT<'static>>,
1141
1142 mesh_shader: Option<vk::PhysicalDeviceMeshShaderPropertiesEXT<'static>>,
1145
1146 multiview: Option<vk::PhysicalDeviceMultiviewPropertiesKHR<'static>>,
1149
1150 pci_bus_info: Option<vk::PhysicalDevicePCIBusInfoPropertiesEXT<'static>>,
1152
1153 device_api_version: u32,
1159
1160 cooperative_matrix_properties: Vec<wgt::CooperativeMatrixProperties>,
1164}
1165
1166impl PhysicalDeviceProperties {
1167 pub fn properties(&self) -> vk::PhysicalDeviceProperties {
1168 self.properties
1169 }
1170
1171 pub fn supports_extension(&self, extension: &CStr) -> bool {
1172 self.supported_extensions
1173 .iter()
1174 .any(|ep| ep.extension_name_as_c_str() == Ok(extension))
1175 }
1176
1177 fn get_required_extensions(&self, requested_features: wgt::Features) -> Vec<&'static CStr> {
1179 let mut extensions = Vec::new();
1180
1181 extensions.push(khr::swapchain::NAME);
1186
1187 if self.device_api_version < vk::API_VERSION_1_1 {
1188 extensions.push(khr::maintenance1::NAME);
1190
1191 if self.supports_extension(khr::maintenance2::NAME) {
1193 extensions.push(khr::maintenance2::NAME);
1194 }
1195
1196 if self.supports_extension(khr::maintenance3::NAME) {
1198 extensions.push(khr::maintenance3::NAME);
1199 }
1200
1201 extensions.push(khr::storage_buffer_storage_class::NAME);
1203
1204 if requested_features.contains(wgt::Features::MULTIVIEW) {
1206 extensions.push(khr::multiview::NAME);
1207 }
1208
1209 if requested_features.contains(wgt::Features::TEXTURE_FORMAT_NV12) {
1211 extensions.push(khr::sampler_ycbcr_conversion::NAME);
1212 }
1213
1214 if requested_features.intersects(wgt::Features::SHADER_F16 | wgt::Features::SHADER_I16)
1216 {
1217 extensions.push(khr::_16bit_storage::NAME);
1222 }
1223
1224 if requested_features.contains(wgt::Features::SHADER_DRAW_INDEX) {
1225 extensions.push(khr::shader_draw_parameters::NAME);
1226 }
1227 }
1228
1229 if self.device_api_version < vk::API_VERSION_1_2 {
1230 if self.supports_extension(khr::image_format_list::NAME) {
1232 extensions.push(khr::image_format_list::NAME);
1233 }
1234
1235 if self.supports_extension(khr::driver_properties::NAME) {
1237 extensions.push(khr::driver_properties::NAME);
1238 }
1239
1240 if self.supports_extension(khr::timeline_semaphore::NAME) {
1242 extensions.push(khr::timeline_semaphore::NAME);
1243 }
1244
1245 if requested_features.intersects(INDEXING_FEATURES) {
1247 extensions.push(ext::descriptor_indexing::NAME);
1248 }
1249
1250 if requested_features.contains(wgt::Features::SHADER_F16)
1254 || self.supports_extension(khr::shader_float16_int8::NAME)
1255 {
1256 extensions.push(khr::shader_float16_int8::NAME);
1257 }
1258
1259 if requested_features.intersects(wgt::Features::EXPERIMENTAL_MESH_SHADER) {
1260 extensions.push(khr::spirv_1_4::NAME);
1261 }
1262
1263 }
1266
1267 if self.device_api_version < vk::API_VERSION_1_3 {
1268 if self.supports_extension(khr::maintenance4::NAME) {
1270 extensions.push(khr::maintenance4::NAME);
1271 }
1272
1273 if self.supports_extension(ext::image_robustness::NAME) {
1275 extensions.push(ext::image_robustness::NAME);
1276 }
1277
1278 if requested_features.contains(wgt::Features::SUBGROUP) {
1280 extensions.push(ext::subgroup_size_control::NAME);
1281 }
1282
1283 if self.supports_extension(khr::shader_integer_dot_product::NAME) {
1285 extensions.push(khr::shader_integer_dot_product::NAME);
1286 }
1287 }
1288
1289 if self.supports_extension(khr::swapchain_mutable_format::NAME) {
1291 extensions.push(khr::swapchain_mutable_format::NAME);
1292 }
1293
1294 if self.supports_extension(ext::robustness2::NAME) {
1296 extensions.push(ext::robustness2::NAME);
1297 }
1298
1299 if self.supports_extension(khr::external_memory_win32::NAME) {
1301 extensions.push(khr::external_memory_win32::NAME);
1302 }
1303
1304 if self.supports_extension(khr::external_memory_fd::NAME) {
1306 extensions.push(khr::external_memory_fd::NAME);
1307 }
1308
1309 if self.supports_extension(ext::external_memory_dma_buf::NAME) {
1311 extensions.push(ext::external_memory_dma_buf::NAME);
1312 }
1313
1314 if self.supports_extension(ext::image_drm_format_modifier::NAME) {
1316 extensions.push(ext::image_drm_format_modifier::NAME);
1317 }
1318
1319 if self.supports_extension(ext::memory_budget::NAME) {
1321 extensions.push(ext::memory_budget::NAME);
1322 } else {
1323 log::debug!("VK_EXT_memory_budget is not available.")
1324 }
1325
1326 if requested_features.contains(wgt::Features::MULTI_DRAW_INDIRECT_COUNT) {
1330 extensions.push(khr::draw_indirect_count::NAME);
1331 }
1332
1333 if requested_features.contains(wgt::Features::EXPERIMENTAL_RAY_QUERY) {
1335 extensions.push(khr::deferred_host_operations::NAME);
1336 extensions.push(khr::acceleration_structure::NAME);
1337 extensions.push(khr::buffer_device_address::NAME);
1338 extensions.push(khr::ray_query::NAME);
1339 }
1340
1341 if requested_features.contains(wgt::Features::EXPERIMENTAL_RAY_HIT_VERTEX_RETURN) {
1342 extensions.push(khr::ray_tracing_position_fetch::NAME)
1343 }
1344
1345 if requested_features.contains(wgt::Features::CONSERVATIVE_RASTERIZATION) {
1347 extensions.push(ext::conservative_rasterization::NAME);
1348 }
1349
1350 #[cfg(target_vendor = "apple")]
1352 extensions.push(khr::portability_subset::NAME);
1353
1354 if requested_features.contains(wgt::Features::TEXTURE_COMPRESSION_ASTC_HDR) {
1356 extensions.push(ext::texture_compression_astc_hdr::NAME);
1357 }
1358
1359 if requested_features.intersects(
1361 wgt::Features::SHADER_INT64_ATOMIC_ALL_OPS | wgt::Features::SHADER_INT64_ATOMIC_MIN_MAX,
1362 ) {
1363 extensions.push(khr::shader_atomic_int64::NAME);
1364 }
1365
1366 if requested_features.intersects(wgt::Features::TEXTURE_INT64_ATOMIC) {
1368 extensions.push(ext::shader_image_atomic_int64::NAME);
1369 }
1370
1371 if requested_features.contains(wgt::Features::SHADER_FLOAT32_ATOMIC) {
1373 extensions.push(ext::shader_atomic_float::NAME);
1374 }
1375
1376 if requested_features.contains(wgt::Features::VULKAN_GOOGLE_DISPLAY_TIMING) {
1378 extensions.push(google::display_timing::NAME);
1379 }
1380
1381 if requested_features.contains(wgt::Features::EXPERIMENTAL_MESH_SHADER) {
1382 extensions.push(ext::mesh_shader::NAME);
1383 }
1384
1385 if requested_features
1388 .intersects(wgt::Features::SHADER_BARYCENTRICS | wgt::Features::SHADER_PER_VERTEX)
1389 {
1390 extensions.push(khr::fragment_shader_barycentric::NAME);
1391 }
1392
1393 if requested_features.contains(wgt::Features::EXPERIMENTAL_COOPERATIVE_MATRIX) {
1395 extensions.push(khr::cooperative_matrix::NAME);
1396 }
1397
1398 extensions
1399 }
1400
1401 fn to_wgpu_limits(&self) -> wgt::Limits {
1402 let limits = &self.properties.limits;
1403
1404 let (
1406 mut max_task_workgroup_total_count,
1407 mut max_task_workgroups_per_dimension,
1408 mut max_mesh_workgroup_total_count,
1409 mut max_mesh_workgroups_per_dimension,
1410 ) = Default::default();
1411 let (
1412 mut max_task_invocations_per_workgroup,
1413 mut max_task_invocations_per_dimension,
1414 mut max_mesh_invocations_per_workgroup,
1415 mut max_mesh_invocations_per_dimension,
1416 mut max_task_payload_size,
1417 mut max_mesh_output_vertices,
1418 mut max_mesh_output_primitives,
1419 mut max_mesh_output_layers,
1420 mut max_mesh_multiview_view_count,
1421 ) = Default::default();
1422 if let Some(m) = self.mesh_shader {
1423 max_task_workgroup_total_count = m.max_task_work_group_total_count;
1424 max_task_workgroups_per_dimension =
1425 m.max_task_work_group_count.into_iter().min().unwrap();
1426 max_mesh_workgroup_total_count = m.max_mesh_work_group_total_count;
1427 max_mesh_workgroups_per_dimension =
1428 m.max_mesh_work_group_count.into_iter().min().unwrap();
1429 max_task_invocations_per_workgroup = m.max_task_work_group_invocations;
1430 max_task_invocations_per_dimension =
1431 m.max_task_work_group_size.into_iter().min().unwrap();
1432 max_mesh_invocations_per_workgroup = m.max_mesh_work_group_invocations;
1433 max_mesh_invocations_per_dimension =
1434 m.max_mesh_work_group_size.into_iter().min().unwrap();
1435 max_task_payload_size = m.max_task_payload_size;
1436 max_mesh_output_vertices = m.max_mesh_output_vertices;
1437 max_mesh_output_primitives = m.max_mesh_output_primitives;
1438 max_mesh_output_layers = m.max_mesh_output_layers;
1439 max_mesh_multiview_view_count = m.max_mesh_multiview_view_count;
1440 }
1441
1442 let max_memory_allocation_size = self
1443 .maintenance_3
1444 .map(|maintenance_3| maintenance_3.max_memory_allocation_size)
1445 .unwrap_or(u64::MAX);
1446 let max_buffer_size = self
1447 .maintenance_4
1448 .map(|maintenance_4| maintenance_4.max_buffer_size)
1449 .unwrap_or(u64::MAX);
1450 let max_buffer_size = max_buffer_size.min(max_memory_allocation_size);
1451
1452 let is_nvidia = self.properties.vendor_id == crate::auxil::db::nvidia::VENDOR;
1455 let max_buffer_size_cap =
1456 if (cfg!(target_os = "linux") || cfg!(target_os = "android")) && !is_nvidia {
1457 i32::MAX as u64
1458 } else {
1459 1u64 << 52
1460 };
1461
1462 let max_buffer_size = max_buffer_size.min(max_buffer_size_cap);
1463
1464 let mut max_binding_array_elements = 0;
1465 let mut max_sampler_binding_array_elements = 0;
1466 if let Some(ref descriptor_indexing) = self.descriptor_indexing {
1467 max_binding_array_elements = descriptor_indexing
1468 .max_descriptor_set_update_after_bind_sampled_images
1469 .min(descriptor_indexing.max_descriptor_set_update_after_bind_storage_images)
1470 .min(descriptor_indexing.max_descriptor_set_update_after_bind_storage_buffers)
1471 .min(descriptor_indexing.max_per_stage_descriptor_update_after_bind_sampled_images)
1472 .min(descriptor_indexing.max_per_stage_descriptor_update_after_bind_storage_images)
1473 .min(
1474 descriptor_indexing.max_per_stage_descriptor_update_after_bind_storage_buffers,
1475 );
1476
1477 max_sampler_binding_array_elements = descriptor_indexing
1478 .max_descriptor_set_update_after_bind_samplers
1479 .min(descriptor_indexing.max_per_stage_descriptor_update_after_bind_samplers);
1480 }
1481
1482 const MAX_SHADER_STAGES_PER_PIPELINE: u32 = 2;
1483
1484 let mut max_storage_textures_per_shader_stage = limits
1497 .max_per_stage_descriptor_storage_images
1498 .min(limits.max_descriptor_set_storage_images / MAX_SHADER_STAGES_PER_PIPELINE);
1499 let mut max_storage_buffers_per_shader_stage = limits
1500 .max_per_stage_descriptor_storage_buffers
1501 .min(limits.max_descriptor_set_storage_buffers / MAX_SHADER_STAGES_PER_PIPELINE);
1502 let mut max_color_attachments = limits
1503 .max_color_attachments
1504 .min(limits.max_fragment_output_attachments);
1505
1506 let ignore_max_fragment_combined_output_resources_by_device = [
1507 crate::auxil::db::intel::VENDOR,
1508 crate::auxil::db::nvidia::VENDOR,
1509 crate::auxil::db::amd::VENDOR,
1510 crate::auxil::db::imgtec::VENDOR,
1511 ]
1512 .contains(&self.properties.vendor_id);
1513 let ignore_max_fragment_combined_output_resources_by_driver = self
1514 .driver
1515 .map(|driver| [vk::DriverId::MESA_AGXV].contains(&driver.driver_id))
1516 .unwrap_or_default();
1517 let ignore_max_fragment_combined_output_resources =
1518 ignore_max_fragment_combined_output_resources_by_device
1519 || ignore_max_fragment_combined_output_resources_by_driver;
1520
1521 if !ignore_max_fragment_combined_output_resources {
1522 crate::auxil::cap_limits_to_be_under_the_sum_limit(
1523 [
1524 &mut max_storage_textures_per_shader_stage,
1525 &mut max_storage_buffers_per_shader_stage,
1526 &mut max_color_attachments,
1527 ],
1528 limits.max_fragment_combined_output_resources,
1529 );
1530 }
1531
1532 let mut max_sampled_textures_per_shader_stage = limits
1543 .max_per_stage_descriptor_sampled_images
1544 .min(limits.max_descriptor_set_sampled_images / MAX_SHADER_STAGES_PER_PIPELINE);
1545 let mut max_uniform_buffers_per_shader_stage = limits
1546 .max_per_stage_descriptor_uniform_buffers
1547 .min(limits.max_descriptor_set_uniform_buffers / MAX_SHADER_STAGES_PER_PIPELINE);
1548
1549 crate::auxil::cap_limits_to_be_under_the_sum_limit(
1550 [
1551 &mut max_sampled_textures_per_shader_stage,
1552 &mut max_uniform_buffers_per_shader_stage,
1553 &mut max_storage_textures_per_shader_stage,
1554 &mut max_storage_buffers_per_shader_stage,
1555 &mut max_color_attachments,
1556 ],
1557 limits.max_per_stage_resources,
1558 );
1559
1560 let mut max_blas_geometry_count = 0;
1562 let mut max_blas_primitive_count = 0;
1563 let mut max_tlas_instance_count = 0;
1564 let mut max_acceleration_structures_per_shader_stage = 0;
1565 if let Some(properties) = self.acceleration_structure {
1566 max_blas_geometry_count = properties.max_geometry_count as u32;
1567 max_blas_primitive_count = properties.max_primitive_count as u32;
1568 max_tlas_instance_count = properties.max_instance_count as u32;
1569 max_acceleration_structures_per_shader_stage = properties
1570 .max_per_stage_descriptor_acceleration_structures
1571 .min(
1572 properties.max_descriptor_set_acceleration_structures
1573 / MAX_SHADER_STAGES_PER_PIPELINE,
1574 );
1575 }
1576
1577 let max_per_set_descriptors = self
1591 .maintenance_3
1592 .map(|maintenance_3| maintenance_3.max_per_set_descriptors)
1593 .unwrap_or(256);
1597
1598 let mut max_samplers_per_shader_stage = limits
1599 .max_per_stage_descriptor_samplers
1600 .min(limits.max_descriptor_set_samplers / MAX_SHADER_STAGES_PER_PIPELINE);
1601
1602 crate::auxil::cap_limits_to_be_under_the_sum_limit(
1603 [
1604 &mut max_sampled_textures_per_shader_stage,
1605 &mut max_uniform_buffers_per_shader_stage,
1606 &mut max_storage_textures_per_shader_stage,
1607 &mut max_storage_buffers_per_shader_stage,
1608 &mut max_samplers_per_shader_stage,
1609 &mut max_acceleration_structures_per_shader_stage,
1610 ],
1611 max_per_set_descriptors / MAX_SHADER_STAGES_PER_PIPELINE,
1612 );
1613
1614 let max_bindings_per_bind_group = 1000.max(max_per_set_descriptors);
1619
1620 let max_color_attachment_bytes_per_sample =
1626 max_color_attachments * wgt::TextureFormat::MAX_TARGET_PIXEL_BYTE_COST;
1627
1628 let max_multiview_view_count = self
1629 .multiview
1630 .map(|a| a.max_multiview_view_count.min(32))
1631 .unwrap_or(0);
1632
1633 crate::auxil::adjust_raw_limits(wgt::Limits {
1634 max_texture_dimension_1d: limits.max_image_dimension1_d,
1639 max_texture_dimension_2d: limits
1640 .max_image_dimension2_d
1641 .min(limits.max_image_dimension_cube)
1642 .min(limits.max_framebuffer_width)
1643 .min(limits.max_framebuffer_height),
1644 max_texture_dimension_3d: limits.max_image_dimension3_d,
1645 max_texture_array_layers: limits.max_image_array_layers,
1646 max_bind_groups: limits.max_bound_descriptor_sets,
1647 max_bind_groups_plus_vertex_buffers: u32::MAX,
1649 max_bindings_per_bind_group,
1650 max_dynamic_uniform_buffers_per_pipeline_layout: limits
1651 .max_descriptor_set_uniform_buffers_dynamic,
1652 max_dynamic_storage_buffers_per_pipeline_layout: limits
1653 .max_descriptor_set_storage_buffers_dynamic,
1654 max_samplers_per_shader_stage,
1655 max_sampled_textures_per_shader_stage,
1656 max_storage_textures_per_shader_stage,
1657 max_storage_buffers_per_shader_stage,
1658 max_uniform_buffers_per_shader_stage,
1659 max_vertex_buffers: limits.max_vertex_input_bindings,
1660 max_buffer_size,
1661 max_uniform_buffer_binding_size: limits
1662 .max_uniform_buffer_range
1663 .min(crate::auxil::MAX_I32_BINDING_SIZE)
1664 .into(),
1665 max_storage_buffer_binding_size: limits
1666 .max_storage_buffer_range
1667 .min(crate::auxil::MAX_I32_BINDING_SIZE)
1668 .into(),
1669 min_uniform_buffer_offset_alignment: limits.min_uniform_buffer_offset_alignment as u32,
1670 min_storage_buffer_offset_alignment: limits.min_storage_buffer_offset_alignment as u32,
1671 max_vertex_attributes: limits.max_vertex_input_attributes,
1672 max_vertex_buffer_array_stride: limits.max_vertex_input_binding_stride,
1673 max_inter_stage_shader_variables: limits
1674 .max_vertex_output_components
1675 .min(limits.max_fragment_input_components)
1676 / 4
1677 - 1, max_color_attachments,
1679 max_color_attachment_bytes_per_sample,
1680 max_compute_workgroup_storage_size: limits.max_compute_shared_memory_size,
1681 max_compute_invocations_per_workgroup: limits.max_compute_work_group_invocations,
1682 max_compute_workgroup_size_x: limits.max_compute_work_group_size[0],
1683 max_compute_workgroup_size_y: limits.max_compute_work_group_size[1],
1684 max_compute_workgroup_size_z: limits.max_compute_work_group_size[2],
1685 max_compute_workgroups_per_dimension: limits.max_compute_work_group_count[0]
1686 .min(limits.max_compute_work_group_count[1])
1687 .min(limits.max_compute_work_group_count[2]),
1688 max_immediate_size: limits.max_push_constants_size,
1689 max_non_sampler_bindings: u32::MAX,
1693
1694 max_binding_array_elements_per_shader_stage: max_binding_array_elements,
1695 max_binding_array_sampler_elements_per_shader_stage: max_sampler_binding_array_elements,
1696 max_binding_array_acceleration_structure_elements_per_shader_stage: if self
1697 .descriptor_indexing
1698 .is_some()
1699 {
1700 max_acceleration_structures_per_shader_stage
1701 } else {
1702 0
1703 },
1704
1705 max_task_workgroup_total_count,
1706 max_task_workgroups_per_dimension,
1707 max_mesh_workgroup_total_count,
1708 max_mesh_workgroups_per_dimension,
1709
1710 max_task_invocations_per_workgroup,
1711 max_task_invocations_per_dimension,
1712
1713 max_mesh_invocations_per_workgroup,
1714 max_mesh_invocations_per_dimension,
1715
1716 max_task_payload_size,
1717 max_mesh_output_vertices,
1718 max_mesh_output_primitives,
1719 max_mesh_output_layers,
1720 max_mesh_multiview_view_count,
1721
1722 max_blas_primitive_count,
1723 max_blas_geometry_count,
1724 max_tlas_instance_count,
1725 max_acceleration_structures_per_shader_stage,
1726
1727 max_multiview_view_count,
1728 })
1729 }
1730
1731 fn to_hal_alignments(&self, using_robustness2: bool) -> crate::Alignments {
1746 let limits = &self.properties.limits;
1747 crate::Alignments {
1748 buffer_copy_offset: wgt::BufferSize::new(limits.optimal_buffer_copy_offset_alignment)
1749 .unwrap(),
1750 buffer_copy_pitch: wgt::BufferSize::new(limits.optimal_buffer_copy_row_pitch_alignment)
1751 .unwrap(),
1752 uniform_bounds_check_alignment: {
1753 let alignment = if using_robustness2 {
1754 self.robustness2
1755 .unwrap() .robust_uniform_buffer_access_size_alignment
1757 } else {
1758 1
1760 };
1761 wgt::BufferSize::new(alignment).unwrap()
1762 },
1763 raw_tlas_instance_size: 64,
1764 ray_tracing_scratch_buffer_alignment: self.acceleration_structure.map_or(
1765 0,
1766 |acceleration_structure| {
1767 acceleration_structure.min_acceleration_structure_scratch_offset_alignment
1768 },
1769 ),
1770 }
1771 }
1772}
1773
1774impl super::InstanceShared {
1775 fn inspect(
1776 &self,
1777 phd: vk::PhysicalDevice,
1778 ) -> (PhysicalDeviceProperties, PhysicalDeviceFeatures) {
1779 let capabilities = {
1780 let mut capabilities = PhysicalDeviceProperties::default();
1781 capabilities.supported_extensions =
1782 unsafe { self.raw.enumerate_device_extension_properties(phd).unwrap() };
1783 capabilities.properties = unsafe { self.raw.get_physical_device_properties(phd) };
1784 capabilities.device_api_version = capabilities.properties.api_version;
1785
1786 let supports_multiview = capabilities.device_api_version >= vk::API_VERSION_1_1
1787 || capabilities.supports_extension(khr::multiview::NAME);
1788
1789 if let Some(ref get_device_properties) = self.get_physical_device_properties {
1790 let supports_maintenance3 = capabilities.device_api_version >= vk::API_VERSION_1_1
1792 || capabilities.supports_extension(khr::maintenance3::NAME);
1793 let supports_maintenance4 = capabilities.device_api_version >= vk::API_VERSION_1_3
1794 || capabilities.supports_extension(khr::maintenance4::NAME);
1795 let supports_descriptor_indexing = capabilities.device_api_version
1796 >= vk::API_VERSION_1_2
1797 || capabilities.supports_extension(ext::descriptor_indexing::NAME);
1798 let supports_driver_properties = capabilities.device_api_version
1799 >= vk::API_VERSION_1_2
1800 || capabilities.supports_extension(khr::driver_properties::NAME);
1801 let supports_subgroup_size_control = capabilities.device_api_version
1802 >= vk::API_VERSION_1_3
1803 || capabilities.supports_extension(ext::subgroup_size_control::NAME);
1804 let supports_robustness2 = capabilities.supports_extension(ext::robustness2::NAME);
1805 let supports_pci_bus_info =
1806 capabilities.supports_extension(ext::pci_bus_info::NAME);
1807
1808 let supports_acceleration_structure =
1809 capabilities.supports_extension(khr::acceleration_structure::NAME);
1810
1811 let supports_mesh_shader = capabilities.supports_extension(ext::mesh_shader::NAME);
1812
1813 let mut properties2 = vk::PhysicalDeviceProperties2KHR::default();
1814 if supports_maintenance3 {
1815 let next = capabilities
1816 .maintenance_3
1817 .insert(vk::PhysicalDeviceMaintenance3Properties::default());
1818 properties2 = properties2.push_next(next);
1819 }
1820
1821 if supports_maintenance4 {
1822 let next = capabilities
1823 .maintenance_4
1824 .insert(vk::PhysicalDeviceMaintenance4Properties::default());
1825 properties2 = properties2.push_next(next);
1826 }
1827
1828 if supports_descriptor_indexing {
1829 let next = capabilities
1830 .descriptor_indexing
1831 .insert(vk::PhysicalDeviceDescriptorIndexingPropertiesEXT::default());
1832 properties2 = properties2.push_next(next);
1833 }
1834
1835 if supports_acceleration_structure {
1836 let next = capabilities
1837 .acceleration_structure
1838 .insert(vk::PhysicalDeviceAccelerationStructurePropertiesKHR::default());
1839 properties2 = properties2.push_next(next);
1840 }
1841
1842 if supports_driver_properties {
1843 let next = capabilities
1844 .driver
1845 .insert(vk::PhysicalDeviceDriverPropertiesKHR::default());
1846 properties2 = properties2.push_next(next);
1847 }
1848
1849 if capabilities.device_api_version >= vk::API_VERSION_1_1 {
1850 let next = capabilities
1851 .subgroup
1852 .insert(vk::PhysicalDeviceSubgroupProperties::default());
1853 properties2 = properties2.push_next(next);
1854 }
1855
1856 if supports_subgroup_size_control {
1857 let next = capabilities
1858 .subgroup_size_control
1859 .insert(vk::PhysicalDeviceSubgroupSizeControlProperties::default());
1860 properties2 = properties2.push_next(next);
1861 }
1862
1863 if supports_robustness2 {
1864 let next = capabilities
1865 .robustness2
1866 .insert(vk::PhysicalDeviceRobustness2PropertiesEXT::default());
1867 properties2 = properties2.push_next(next);
1868 }
1869
1870 if supports_pci_bus_info {
1871 let next = capabilities
1872 .pci_bus_info
1873 .insert(vk::PhysicalDevicePCIBusInfoPropertiesEXT::default());
1874 properties2 = properties2.push_next(next);
1875 }
1876
1877 if supports_mesh_shader {
1878 let next = capabilities
1879 .mesh_shader
1880 .insert(vk::PhysicalDeviceMeshShaderPropertiesEXT::default());
1881 properties2 = properties2.push_next(next);
1882 }
1883
1884 if supports_multiview {
1885 let next = capabilities
1886 .multiview
1887 .insert(vk::PhysicalDeviceMultiviewProperties::default());
1888 properties2 = properties2.push_next(next);
1889 }
1890
1891 unsafe {
1892 get_device_properties.get_physical_device_properties2(phd, &mut properties2)
1893 };
1894
1895 if capabilities.supports_extension(khr::cooperative_matrix::NAME) {
1897 let coop_matrix =
1898 khr::cooperative_matrix::Instance::new(&self.entry, &self.raw);
1899 capabilities.cooperative_matrix_properties =
1900 query_cooperative_matrix_properties(&coop_matrix, phd);
1901 }
1902
1903 if is_intel_igpu_outdated_for_robustness2(
1904 capabilities.properties,
1905 capabilities.driver,
1906 ) {
1907 capabilities
1908 .supported_extensions
1909 .retain(|&x| x.extension_name_as_c_str() != Ok(ext::robustness2::NAME));
1910 capabilities.robustness2 = None;
1911 }
1912 };
1913 capabilities
1914 };
1915
1916 let mut features = PhysicalDeviceFeatures::default();
1917 features.core = if let Some(ref get_device_properties) = self.get_physical_device_properties
1918 {
1919 let core = vk::PhysicalDeviceFeatures::default();
1920 let mut features2 = vk::PhysicalDeviceFeatures2KHR::default().features(core);
1921
1922 if capabilities.device_api_version >= vk::API_VERSION_1_1
1924 || capabilities.supports_extension(khr::multiview::NAME)
1925 {
1926 let next = features
1927 .multiview
1928 .insert(vk::PhysicalDeviceMultiviewFeatures::default());
1929 features2 = features2.push_next(next);
1930 }
1931
1932 if capabilities.device_api_version >= vk::API_VERSION_1_1
1934 || capabilities.supports_extension(khr::sampler_ycbcr_conversion::NAME)
1935 {
1936 let next = features
1937 .sampler_ycbcr_conversion
1938 .insert(vk::PhysicalDeviceSamplerYcbcrConversionFeatures::default());
1939 features2 = features2.push_next(next);
1940 }
1941
1942 if capabilities.supports_extension(ext::descriptor_indexing::NAME) {
1943 let next = features
1944 .descriptor_indexing
1945 .insert(vk::PhysicalDeviceDescriptorIndexingFeaturesEXT::default());
1946 features2 = features2.push_next(next);
1947 }
1948
1949 if capabilities.supports_extension(khr::timeline_semaphore::NAME) {
1952 let next = features
1953 .timeline_semaphore
1954 .insert(vk::PhysicalDeviceTimelineSemaphoreFeaturesKHR::default());
1955 features2 = features2.push_next(next);
1956 }
1957
1958 if capabilities.device_api_version >= vk::API_VERSION_1_2
1961 || capabilities.supports_extension(khr::shader_atomic_int64::NAME)
1962 {
1963 let next = features
1964 .shader_atomic_int64
1965 .insert(vk::PhysicalDeviceShaderAtomicInt64Features::default());
1966 features2 = features2.push_next(next);
1967 }
1968
1969 if capabilities.supports_extension(ext::shader_image_atomic_int64::NAME) {
1970 let next = features
1971 .shader_image_atomic_int64
1972 .insert(vk::PhysicalDeviceShaderImageAtomicInt64FeaturesEXT::default());
1973 features2 = features2.push_next(next);
1974 }
1975 if capabilities.supports_extension(ext::shader_atomic_float::NAME) {
1976 let next = features
1977 .shader_atomic_float
1978 .insert(vk::PhysicalDeviceShaderAtomicFloatFeaturesEXT::default());
1979 features2 = features2.push_next(next);
1980 }
1981 if capabilities.supports_extension(ext::image_robustness::NAME) {
1982 let next = features
1983 .image_robustness
1984 .insert(vk::PhysicalDeviceImageRobustnessFeaturesEXT::default());
1985 features2 = features2.push_next(next);
1986 }
1987 if capabilities.supports_extension(ext::robustness2::NAME) {
1988 let next = features
1989 .robustness2
1990 .insert(vk::PhysicalDeviceRobustness2FeaturesEXT::default());
1991 features2 = features2.push_next(next);
1992 }
1993 if capabilities.supports_extension(ext::texture_compression_astc_hdr::NAME) {
1994 let next = features
1995 .astc_hdr
1996 .insert(vk::PhysicalDeviceTextureCompressionASTCHDRFeaturesEXT::default());
1997 features2 = features2.push_next(next);
1998 }
1999
2000 if capabilities.device_api_version >= vk::API_VERSION_1_2
2002 || capabilities.supports_extension(khr::shader_float16_int8::NAME)
2003 {
2004 let next = features
2005 .shader_float16_int8
2006 .insert(vk::PhysicalDeviceShaderFloat16Int8FeaturesKHR::default());
2007 features2 = features2.push_next(next);
2008 }
2009
2010 if capabilities.supports_extension(khr::_16bit_storage::NAME) {
2011 let next = features
2012 ._16bit_storage
2013 .insert(vk::PhysicalDevice16BitStorageFeaturesKHR::default());
2014 features2 = features2.push_next(next);
2015 }
2016 if capabilities.supports_extension(khr::acceleration_structure::NAME) {
2017 let next = features
2018 .acceleration_structure
2019 .insert(vk::PhysicalDeviceAccelerationStructureFeaturesKHR::default());
2020 features2 = features2.push_next(next);
2021 }
2022
2023 if capabilities.supports_extension(khr::ray_tracing_position_fetch::NAME) {
2024 let next = features
2025 .position_fetch
2026 .insert(vk::PhysicalDeviceRayTracingPositionFetchFeaturesKHR::default());
2027 features2 = features2.push_next(next);
2028 }
2029
2030 if capabilities.device_api_version >= vk::API_VERSION_1_3
2032 || capabilities.supports_extension(khr::maintenance4::NAME)
2033 {
2034 let next = features
2035 .maintenance4
2036 .insert(vk::PhysicalDeviceMaintenance4Features::default());
2037 features2 = features2.push_next(next);
2038 }
2039
2040 if capabilities.device_api_version >= vk::API_VERSION_1_3
2042 || capabilities.supports_extension(khr::zero_initialize_workgroup_memory::NAME)
2043 {
2044 let next = features
2045 .zero_initialize_workgroup_memory
2046 .insert(vk::PhysicalDeviceZeroInitializeWorkgroupMemoryFeatures::default());
2047 features2 = features2.push_next(next);
2048 }
2049
2050 if capabilities.device_api_version >= vk::API_VERSION_1_3
2052 || capabilities.supports_extension(ext::subgroup_size_control::NAME)
2053 {
2054 let next = features
2055 .subgroup_size_control
2056 .insert(vk::PhysicalDeviceSubgroupSizeControlFeatures::default());
2057 features2 = features2.push_next(next);
2058 }
2059
2060 if capabilities.supports_extension(ext::mesh_shader::NAME) {
2061 let next = features
2062 .mesh_shader
2063 .insert(vk::PhysicalDeviceMeshShaderFeaturesEXT::default());
2064 features2 = features2.push_next(next);
2065 }
2066
2067 if capabilities.device_api_version >= vk::API_VERSION_1_3
2069 || capabilities.supports_extension(khr::shader_integer_dot_product::NAME)
2070 {
2071 let next = features
2072 .shader_integer_dot_product
2073 .insert(vk::PhysicalDeviceShaderIntegerDotProductFeatures::default());
2074 features2 = features2.push_next(next);
2075 }
2076
2077 if capabilities.supports_extension(khr::fragment_shader_barycentric::NAME) {
2078 let next = features
2079 .shader_barycentrics
2080 .insert(vk::PhysicalDeviceFragmentShaderBarycentricFeaturesKHR::default());
2081 features2 = features2.push_next(next);
2082 }
2083
2084 if capabilities.supports_extension(khr::portability_subset::NAME) {
2085 let next = features
2086 .portability_subset
2087 .insert(vk::PhysicalDevicePortabilitySubsetFeaturesKHR::default());
2088 features2 = features2.push_next(next);
2089 }
2090
2091 if capabilities.supports_extension(khr::cooperative_matrix::NAME) {
2092 let next = features
2093 .cooperative_matrix
2094 .insert(vk::PhysicalDeviceCooperativeMatrixFeaturesKHR::default());
2095 features2 = features2.push_next(next);
2096 }
2097
2098 if capabilities.device_api_version >= vk::API_VERSION_1_1 {
2099 let next = features
2100 .shader_draw_parameters
2101 .insert(vk::PhysicalDeviceShaderDrawParametersFeatures::default());
2102 features2 = features2.push_next(next);
2103 }
2104
2105 unsafe { get_device_properties.get_physical_device_features2(phd, &mut features2) };
2106 features2.features
2107 } else {
2108 unsafe { self.raw.get_physical_device_features(phd) }
2109 };
2110
2111 (capabilities, features)
2112 }
2113}
2114
2115impl super::Instance {
2116 pub fn expose_adapter(
2117 &self,
2118 phd: vk::PhysicalDevice,
2119 ) -> Option<crate::ExposedAdapter<super::Api>> {
2120 use crate::auxil::db;
2121
2122 let (phd_capabilities, phd_features) = self.shared.inspect(phd);
2123
2124 let mem_properties = {
2125 profiling::scope!("vkGetPhysicalDeviceMemoryProperties");
2126 unsafe { self.shared.raw.get_physical_device_memory_properties(phd) }
2127 };
2128 let memory_types = &mem_properties.memory_types_as_slice();
2129 let supports_lazily_allocated = memory_types.iter().any(|mem| {
2130 mem.property_flags
2131 .contains(vk::MemoryPropertyFlags::LAZILY_ALLOCATED)
2132 });
2133
2134 let device_type = match phd_capabilities.properties.device_type {
2135 vk::PhysicalDeviceType::OTHER => wgt::DeviceType::Other,
2136 vk::PhysicalDeviceType::INTEGRATED_GPU => wgt::DeviceType::IntegratedGpu,
2137 vk::PhysicalDeviceType::DISCRETE_GPU => wgt::DeviceType::DiscreteGpu,
2138 vk::PhysicalDeviceType::VIRTUAL_GPU => wgt::DeviceType::VirtualGpu,
2139 vk::PhysicalDeviceType::CPU => wgt::DeviceType::Cpu,
2140 _ => wgt::DeviceType::Other,
2141 };
2142 let info = wgt::AdapterInfo {
2143 name: {
2144 phd_capabilities
2145 .properties
2146 .device_name_as_c_str()
2147 .ok()
2148 .and_then(|name| name.to_str().ok())
2149 .unwrap_or("?")
2150 .to_owned()
2151 },
2152 vendor: phd_capabilities.properties.vendor_id,
2153 device: phd_capabilities.properties.device_id,
2154 device_pci_bus_id: phd_capabilities
2155 .pci_bus_info
2156 .filter(|info| info.pci_bus != 0 || info.pci_device != 0)
2157 .map(|info| {
2158 format!(
2159 "{:04x}:{:02x}:{:02x}.{}",
2160 info.pci_domain, info.pci_bus, info.pci_device, info.pci_function
2161 )
2162 })
2163 .unwrap_or_default(),
2164 driver: {
2165 phd_capabilities
2166 .driver
2167 .as_ref()
2168 .and_then(|driver| driver.driver_name_as_c_str().ok())
2169 .and_then(|name| name.to_str().ok())
2170 .unwrap_or("?")
2171 .to_owned()
2172 },
2173 driver_info: {
2174 phd_capabilities
2175 .driver
2176 .as_ref()
2177 .and_then(|driver| driver.driver_info_as_c_str().ok())
2178 .and_then(|name| name.to_str().ok())
2179 .unwrap_or("?")
2180 .to_owned()
2181 },
2182 subgroup_min_size: phd_capabilities
2183 .subgroup_size_control
2184 .map(|subgroup_size| subgroup_size.min_subgroup_size)
2185 .unwrap_or(wgt::MINIMUM_SUBGROUP_MIN_SIZE),
2186 subgroup_max_size: phd_capabilities
2187 .subgroup_size_control
2188 .map(|subgroup_size| subgroup_size.max_subgroup_size)
2189 .unwrap_or(wgt::MAXIMUM_SUBGROUP_MAX_SIZE),
2190 transient_saves_memory: supports_lazily_allocated,
2191 ..wgt::AdapterInfo::new(device_type, wgt::Backend::Vulkan)
2192 };
2193 let mut workarounds = super::Workarounds::empty();
2194 {
2195 workarounds |= super::Workarounds::SEPARATE_ENTRY_POINTS;
2197 workarounds.set(
2198 super::Workarounds::EMPTY_RESOLVE_ATTACHMENT_LISTS,
2199 phd_capabilities.properties.vendor_id == db::qualcomm::VENDOR,
2200 );
2201 workarounds.set(
2202 super::Workarounds::FORCE_FILL_BUFFER_WITH_SIZE_GREATER_4096_ALIGNED_OFFSET_16,
2203 phd_capabilities.properties.vendor_id == db::nvidia::VENDOR,
2204 );
2205 };
2206
2207 if let Some(driver) = phd_capabilities.driver {
2208 if driver.conformance_version.major == 0 {
2209 if driver.driver_id == vk::DriverId::MOLTENVK {
2210 log::debug!("Adapter is not Vulkan compliant, but is MoltenVK, continuing");
2211 } else if self
2212 .shared
2213 .flags
2214 .contains(wgt::InstanceFlags::ALLOW_UNDERLYING_NONCOMPLIANT_ADAPTER)
2215 {
2216 log::debug!("Adapter is not Vulkan compliant: {}", info.name);
2217 } else {
2218 log::debug!(
2219 "Adapter is not Vulkan compliant, hiding adapter: {}",
2220 info.name
2221 );
2222 return None;
2223 }
2224 }
2225 }
2226 if phd_capabilities.device_api_version == vk::API_VERSION_1_0
2227 && !phd_capabilities.supports_extension(khr::storage_buffer_storage_class::NAME)
2228 {
2229 log::debug!(
2230 "SPIR-V storage buffer class is not supported, hiding adapter: {}",
2231 info.name
2232 );
2233 return None;
2234 }
2235 if !phd_capabilities.supports_extension(khr::maintenance1::NAME)
2236 && phd_capabilities.device_api_version < vk::API_VERSION_1_1
2237 {
2238 log::debug!(
2239 "VK_KHR_maintenance1 is not supported, hiding adapter: {}",
2240 info.name
2241 );
2242 return None;
2243 }
2244
2245 let queue_families = unsafe {
2246 self.shared
2247 .raw
2248 .get_physical_device_queue_family_properties(phd)
2249 };
2250 let queue_family_properties = queue_families.first()?;
2251 let queue_flags = queue_family_properties.queue_flags;
2252 if !queue_flags.contains(vk::QueueFlags::GRAPHICS) {
2253 log::debug!("The first queue only exposes {queue_flags:?}");
2254 return None;
2255 }
2256
2257 let (available_features, mut downlevel_flags) = phd_features.to_wgpu(
2258 &self.shared.raw,
2259 phd,
2260 &phd_capabilities,
2261 queue_family_properties,
2262 );
2263
2264 if info.driver == "llvmpipe" {
2265 downlevel_flags.set(
2268 wgt::DownlevelFlags::SHADER_F16_IN_F32,
2269 available_features.contains(wgt::Features::SHADER_F16),
2270 );
2271 }
2272
2273 downlevel_flags.set(
2274 wgt::DownlevelFlags::TEXTURE_COMPRESSION,
2275 available_features.contains(wgt::Features::TEXTURE_COMPRESSION_BC)
2276 || available_features.contains(
2277 wgt::Features::TEXTURE_COMPRESSION_ETC2
2278 | wgt::Features::TEXTURE_COMPRESSION_ASTC,
2279 ),
2280 );
2281
2282 let has_robust_buffer_access2 = phd_features
2283 .robustness2
2284 .as_ref()
2285 .map(|r| r.robust_buffer_access2 == 1)
2286 .unwrap_or_default();
2287
2288 let alignments = phd_capabilities.to_hal_alignments(has_robust_buffer_access2);
2289
2290 let private_caps = super::PrivateCapabilities {
2291 image_view_usage: phd_capabilities.device_api_version >= vk::API_VERSION_1_1
2292 || phd_capabilities.supports_extension(khr::maintenance2::NAME),
2293 timeline_semaphores: match phd_features.timeline_semaphore {
2294 Some(features) => features.timeline_semaphore == vk::TRUE,
2295 None => phd_features
2296 .timeline_semaphore
2297 .is_some_and(|ext| ext.timeline_semaphore != 0),
2298 },
2299 texture_d24: supports_format(
2300 &self.shared.raw,
2301 phd,
2302 vk::Format::X8_D24_UNORM_PACK32,
2303 vk::ImageTiling::OPTIMAL,
2304 depth_stencil_required_flags(),
2305 ),
2306 texture_d24_s8: supports_format(
2307 &self.shared.raw,
2308 phd,
2309 vk::Format::D24_UNORM_S8_UINT,
2310 vk::ImageTiling::OPTIMAL,
2311 depth_stencil_required_flags(),
2312 ),
2313 texture_s8: supports_format(
2314 &self.shared.raw,
2315 phd,
2316 vk::Format::S8_UINT,
2317 vk::ImageTiling::OPTIMAL,
2318 depth_stencil_required_flags(),
2319 ),
2320 multi_draw_indirect: phd_features.core.multi_draw_indirect != 0,
2321 max_draw_indirect_count: phd_capabilities.properties.limits.max_draw_indirect_count,
2322 non_coherent_map_mask: phd_capabilities.properties.limits.non_coherent_atom_size - 1,
2323 can_present: true,
2324 robust_buffer_access: phd_features.core.robust_buffer_access != 0,
2326 robust_image_access: match phd_features.robustness2 {
2327 Some(ref f) => f.robust_image_access2 != 0,
2328 None => phd_features
2329 .image_robustness
2330 .is_some_and(|ext| ext.robust_image_access != 0),
2331 },
2332 robust_buffer_access2: has_robust_buffer_access2,
2333 robust_image_access2: phd_features
2334 .robustness2
2335 .as_ref()
2336 .map(|r| r.robust_image_access2 == 1)
2337 .unwrap_or_default(),
2338 zero_initialize_workgroup_memory: phd_features
2339 .zero_initialize_workgroup_memory
2340 .is_some_and(|ext| ext.shader_zero_initialize_workgroup_memory == vk::TRUE),
2341 image_format_list: phd_capabilities.device_api_version >= vk::API_VERSION_1_2
2342 || phd_capabilities.supports_extension(khr::image_format_list::NAME),
2343 maximum_samplers: phd_capabilities
2344 .properties
2345 .limits
2346 .max_sampler_allocation_count,
2347 shader_integer_dot_product: phd_features
2348 .shader_integer_dot_product
2349 .is_some_and(|ext| ext.shader_integer_dot_product != 0),
2350 shader_int8: phd_features
2351 .shader_float16_int8
2352 .is_some_and(|features| features.shader_int8 != 0),
2353 multiview_instance_index_limit: phd_capabilities
2354 .multiview
2355 .map(|a| a.max_multiview_instance_index)
2356 .unwrap_or(0),
2357 scratch_buffer_alignment: alignments.ray_tracing_scratch_buffer_alignment,
2358 };
2359 let capabilities = crate::Capabilities {
2360 limits: phd_capabilities.to_wgpu_limits(),
2361 alignments,
2362 downlevel: wgt::DownlevelCapabilities {
2363 flags: downlevel_flags,
2364 limits: wgt::DownlevelLimits {},
2365 shader_model: wgt::ShaderModel::Sm5, },
2367 cooperative_matrix_properties: phd_capabilities.cooperative_matrix_properties.clone(),
2368 };
2369
2370 let adapter = super::Adapter {
2371 raw: phd,
2372 instance: Arc::clone(&self.shared),
2373 known_memory_flags: vk::MemoryPropertyFlags::DEVICE_LOCAL
2375 | vk::MemoryPropertyFlags::HOST_VISIBLE
2376 | vk::MemoryPropertyFlags::HOST_COHERENT
2377 | vk::MemoryPropertyFlags::HOST_CACHED
2378 | vk::MemoryPropertyFlags::LAZILY_ALLOCATED,
2379 phd_capabilities,
2380 phd_features,
2381 downlevel_flags,
2382 private_caps,
2383 workarounds,
2384 };
2385
2386 Some(crate::ExposedAdapter {
2387 adapter,
2388 info,
2389 features: available_features,
2390 capabilities,
2391 })
2392 }
2393}
2394
2395impl super::Adapter {
2396 pub fn raw_physical_device(&self) -> vk::PhysicalDevice {
2397 self.raw
2398 }
2399
2400 pub fn get_physical_device_features(&self) -> &PhysicalDeviceFeatures {
2401 &self.phd_features
2402 }
2403
2404 pub fn physical_device_capabilities(&self) -> &PhysicalDeviceProperties {
2405 &self.phd_capabilities
2406 }
2407
2408 pub fn shared_instance(&self) -> &super::InstanceShared {
2409 &self.instance
2410 }
2411
2412 pub fn required_device_extensions(&self, features: wgt::Features) -> Vec<&'static CStr> {
2413 let (supported_extensions, unsupported_extensions) = self
2414 .phd_capabilities
2415 .get_required_extensions(features)
2416 .iter()
2417 .partition::<Vec<&CStr>, _>(|&&extension| {
2418 self.phd_capabilities.supports_extension(extension)
2419 });
2420
2421 if !unsupported_extensions.is_empty() {
2422 log::debug!("Missing extensions: {unsupported_extensions:?}");
2423 }
2424
2425 log::debug!("Supported extensions: {supported_extensions:?}");
2426 supported_extensions
2427 }
2428
2429 pub fn physical_device_features(
2444 &self,
2445 enabled_extensions: &[&'static CStr],
2446 features: wgt::Features,
2447 ) -> PhysicalDeviceFeatures {
2448 PhysicalDeviceFeatures::from_extensions_and_requested_features(
2449 &self.phd_capabilities,
2450 &self.phd_features,
2451 enabled_extensions,
2452 features,
2453 self.downlevel_flags,
2454 &self.private_caps,
2455 )
2456 }
2457
2458 #[allow(clippy::too_many_arguments)]
2466 pub unsafe fn device_from_raw(
2467 &self,
2468 raw_device: ash::Device,
2469 drop_callback: Option<crate::DropCallback>,
2470 enabled_extensions: &[&'static CStr],
2471 features: wgt::Features,
2472 limits: &wgt::Limits,
2473 memory_hints: &wgt::MemoryHints,
2474 family_index: u32,
2475 queue_index: u32,
2476 ) -> Result<crate::OpenDevice<super::Api>, crate::DeviceError> {
2477 let mem_properties = {
2478 profiling::scope!("vkGetPhysicalDeviceMemoryProperties");
2479 unsafe {
2480 self.instance
2481 .raw
2482 .get_physical_device_memory_properties(self.raw)
2483 }
2484 };
2485 let memory_types = &mem_properties.memory_types_as_slice();
2486 let valid_ash_memory_types = memory_types.iter().enumerate().fold(0, |u, (i, mem)| {
2487 if self.known_memory_flags.contains(mem.property_flags) {
2488 u | (1 << i)
2489 } else {
2490 u
2491 }
2492 });
2493
2494 let debug_utils_fn = if self.instance.extensions.contains(&ext::debug_utils::NAME) {
2498 Some(ext::debug_utils::Device::new(
2499 &self.instance.raw,
2500 &raw_device,
2501 ))
2502 } else {
2503 None
2504 };
2505 let indirect_count_fn = if enabled_extensions.contains(&khr::draw_indirect_count::NAME) {
2506 Some(khr::draw_indirect_count::Device::new(
2507 &self.instance.raw,
2508 &raw_device,
2509 ))
2510 } else {
2511 None
2512 };
2513 let timeline_semaphore_fn = if enabled_extensions.contains(&khr::timeline_semaphore::NAME) {
2514 Some(super::ExtensionFn::Extension(
2515 khr::timeline_semaphore::Device::new(&self.instance.raw, &raw_device),
2516 ))
2517 } else if self.phd_capabilities.device_api_version >= vk::API_VERSION_1_2 {
2518 Some(super::ExtensionFn::Promoted)
2519 } else {
2520 None
2521 };
2522 let ray_tracing_fns = if enabled_extensions.contains(&khr::acceleration_structure::NAME)
2523 && enabled_extensions.contains(&khr::buffer_device_address::NAME)
2524 {
2525 Some(super::RayTracingDeviceExtensionFunctions {
2526 acceleration_structure: khr::acceleration_structure::Device::new(
2527 &self.instance.raw,
2528 &raw_device,
2529 ),
2530 buffer_device_address: khr::buffer_device_address::Device::new(
2531 &self.instance.raw,
2532 &raw_device,
2533 ),
2534 })
2535 } else {
2536 None
2537 };
2538 let mesh_shading_fns = if enabled_extensions.contains(&ext::mesh_shader::NAME) {
2539 Some(ext::mesh_shader::Device::new(
2540 &self.instance.raw,
2541 &raw_device,
2542 ))
2543 } else {
2544 None
2545 };
2546 let external_memory_fd_fn = if enabled_extensions.contains(&khr::external_memory_fd::NAME) {
2547 Some(khr::external_memory_fd::Device::new(
2548 &self.instance.raw,
2549 &raw_device,
2550 ))
2551 } else {
2552 None
2553 };
2554
2555 let naga_options = {
2556 use naga::back::spv;
2557
2558 let mut capabilities = vec![
2561 spv::Capability::Shader,
2562 spv::Capability::Matrix,
2563 spv::Capability::Sampled1D,
2564 spv::Capability::Image1D,
2565 spv::Capability::ImageQuery,
2566 spv::Capability::DerivativeControl,
2567 spv::Capability::StorageImageExtendedFormats,
2568 ];
2569
2570 if self
2571 .downlevel_flags
2572 .contains(wgt::DownlevelFlags::CUBE_ARRAY_TEXTURES)
2573 {
2574 capabilities.push(spv::Capability::SampledCubeArray);
2575 }
2576
2577 if self
2578 .downlevel_flags
2579 .contains(wgt::DownlevelFlags::MULTISAMPLED_SHADING)
2580 {
2581 capabilities.push(spv::Capability::SampleRateShading);
2582 }
2583
2584 if features.contains(wgt::Features::MULTIVIEW) {
2585 capabilities.push(spv::Capability::MultiView);
2586 }
2587
2588 if features.contains(wgt::Features::PRIMITIVE_INDEX) {
2589 capabilities.push(spv::Capability::Geometry);
2590 }
2591
2592 if features.intersects(wgt::Features::SUBGROUP | wgt::Features::SUBGROUP_VERTEX) {
2593 capabilities.push(spv::Capability::GroupNonUniform);
2594 capabilities.push(spv::Capability::GroupNonUniformVote);
2595 capabilities.push(spv::Capability::GroupNonUniformArithmetic);
2596 capabilities.push(spv::Capability::GroupNonUniformBallot);
2597 capabilities.push(spv::Capability::GroupNonUniformShuffle);
2598 capabilities.push(spv::Capability::GroupNonUniformShuffleRelative);
2599 capabilities.push(spv::Capability::GroupNonUniformQuad);
2600 }
2601
2602 if features.intersects(
2603 wgt::Features::SAMPLED_TEXTURE_AND_STORAGE_BUFFER_ARRAY_NON_UNIFORM_INDEXING
2604 | wgt::Features::STORAGE_TEXTURE_ARRAY_NON_UNIFORM_INDEXING
2605 | wgt::Features::UNIFORM_BUFFER_BINDING_ARRAYS,
2606 ) {
2607 capabilities.push(spv::Capability::ShaderNonUniform);
2608 }
2609 if features.contains(wgt::Features::BGRA8UNORM_STORAGE) {
2610 capabilities.push(spv::Capability::StorageImageWriteWithoutFormat);
2611 }
2612
2613 if features.contains(wgt::Features::EXPERIMENTAL_RAY_QUERY) {
2614 capabilities.push(spv::Capability::RayQueryKHR);
2615 }
2616
2617 if features.contains(wgt::Features::SHADER_INT64) {
2618 capabilities.push(spv::Capability::Int64);
2619 }
2620
2621 if features.contains(wgt::Features::SHADER_F16) {
2622 capabilities.push(spv::Capability::Float16);
2623 }
2624
2625 if features.contains(wgt::Features::SHADER_I16) {
2626 capabilities.push(spv::Capability::Int16);
2627 }
2628
2629 if features.intersects(
2630 wgt::Features::SHADER_INT64_ATOMIC_ALL_OPS
2631 | wgt::Features::SHADER_INT64_ATOMIC_MIN_MAX
2632 | wgt::Features::TEXTURE_INT64_ATOMIC,
2633 ) {
2634 capabilities.push(spv::Capability::Int64Atomics);
2635 }
2636
2637 if features.intersects(wgt::Features::TEXTURE_INT64_ATOMIC) {
2638 capabilities.push(spv::Capability::Int64ImageEXT);
2639 }
2640
2641 if features.contains(wgt::Features::SHADER_FLOAT32_ATOMIC) {
2642 capabilities.push(spv::Capability::AtomicFloat32AddEXT);
2643 }
2644
2645 if features.contains(wgt::Features::CLIP_DISTANCES) {
2646 capabilities.push(spv::Capability::ClipDistance);
2647 }
2648
2649 if features
2651 .intersects(wgt::Features::SHADER_BARYCENTRICS | wgt::Features::SHADER_PER_VERTEX)
2652 {
2653 capabilities.push(spv::Capability::FragmentBarycentricKHR);
2654 }
2655
2656 if features.contains(wgt::Features::SHADER_DRAW_INDEX) {
2657 capabilities.push(spv::Capability::DrawParameters);
2658 }
2659
2660 let mut flags = spv::WriterFlags::empty();
2661 flags.set(
2662 spv::WriterFlags::DEBUG,
2663 self.instance.flags.contains(wgt::InstanceFlags::DEBUG),
2664 );
2665 flags.set(
2666 spv::WriterFlags::LABEL_VARYINGS,
2667 self.phd_capabilities.properties.vendor_id != crate::auxil::db::qualcomm::VENDOR,
2668 );
2669 flags.set(
2670 spv::WriterFlags::FORCE_POINT_SIZE,
2671 true, );
2676 flags.set(
2677 spv::WriterFlags::PRINT_ON_RAY_QUERY_INITIALIZATION_FAIL,
2678 self.instance.flags.contains(wgt::InstanceFlags::DEBUG)
2679 && (self.instance.instance_api_version >= vk::API_VERSION_1_3
2680 || enabled_extensions.contains(&khr::shader_non_semantic_info::NAME)),
2681 );
2682 if features.contains(wgt::Features::EXPERIMENTAL_RAY_QUERY) {
2683 capabilities.push(spv::Capability::RayQueryKHR);
2684 }
2685 if features.contains(wgt::Features::EXPERIMENTAL_RAY_HIT_VERTEX_RETURN) {
2686 capabilities.push(spv::Capability::RayQueryPositionFetchKHR)
2687 }
2688 if features.contains(wgt::Features::EXPERIMENTAL_MESH_SHADER) {
2689 capabilities.push(spv::Capability::MeshShadingEXT);
2690 }
2691 if features.contains(wgt::Features::EXPERIMENTAL_COOPERATIVE_MATRIX) {
2692 capabilities.push(spv::Capability::CooperativeMatrixKHR);
2693 capabilities.push(spv::Capability::VulkanMemoryModel);
2695 }
2696 if self.private_caps.shader_integer_dot_product {
2697 capabilities.extend(&[
2699 spv::Capability::DotProductInputAllKHR,
2700 spv::Capability::DotProductInput4x8BitKHR,
2701 spv::Capability::DotProductInput4x8BitPackedKHR,
2702 spv::Capability::DotProductKHR,
2703 ]);
2704 }
2705 if self.private_caps.shader_int8 {
2706 capabilities.extend(&[spv::Capability::Int8]);
2708 }
2709 spv::Options {
2710 lang_version: match self.phd_capabilities.device_api_version {
2711 vk::API_VERSION_1_0..vk::API_VERSION_1_1 => (1, 0),
2714 vk::API_VERSION_1_1..vk::API_VERSION_1_2 => (1, 3),
2715 vk::API_VERSION_1_2..vk::API_VERSION_1_3 => (1, 5),
2716 vk::API_VERSION_1_3.. => (1, 6),
2717 _ => unreachable!(),
2718 },
2719 flags,
2720 capabilities: Some(capabilities.iter().cloned().collect()),
2721 bounds_check_policies: naga::proc::BoundsCheckPolicies {
2722 index: naga::proc::BoundsCheckPolicy::Restrict,
2723 buffer: if self.private_caps.robust_buffer_access2 {
2724 naga::proc::BoundsCheckPolicy::Unchecked
2725 } else {
2726 naga::proc::BoundsCheckPolicy::Restrict
2727 },
2728 image_load: if self.private_caps.robust_image_access {
2729 naga::proc::BoundsCheckPolicy::Unchecked
2730 } else {
2731 naga::proc::BoundsCheckPolicy::Restrict
2732 },
2733 binding_array: naga::proc::BoundsCheckPolicy::Unchecked,
2735 },
2736 zero_initialize_workgroup_memory: if self
2737 .private_caps
2738 .zero_initialize_workgroup_memory
2739 {
2740 spv::ZeroInitializeWorkgroupMemoryMode::Native
2741 } else {
2742 spv::ZeroInitializeWorkgroupMemoryMode::Polyfill
2743 },
2744 force_loop_bounding: true,
2745 ray_query_initialization_tracking: true,
2746 use_storage_input_output_16: features.contains(wgt::Features::SHADER_F16)
2747 && self.phd_features.supports_storage_input_output_16(),
2748 fake_missing_bindings: false,
2749 binding_map: BTreeMap::default(),
2751 debug_info: None,
2752 task_dispatch_limits: Some(naga::back::TaskDispatchLimits {
2753 max_mesh_workgroups_per_dim: limits.max_mesh_workgroups_per_dimension,
2754 max_mesh_workgroups_total: limits.max_mesh_workgroup_total_count,
2755 }),
2756 mesh_shader_primitive_indices_clamp: true,
2757 trace_ray_argument_validation: true,
2758 emit_int_div_checks: true,
2759 }
2760 };
2761
2762 let raw_queue = {
2763 profiling::scope!("vkGetDeviceQueue");
2764 unsafe { raw_device.get_device_queue(family_index, queue_index) }
2765 };
2766
2767 let driver_version = self
2768 .phd_capabilities
2769 .properties
2770 .driver_version
2771 .to_be_bytes();
2772 #[rustfmt::skip]
2773 let pipeline_cache_validation_key = [
2774 driver_version[0], driver_version[1], driver_version[2], driver_version[3],
2775 0, 0, 0, 0,
2776 0, 0, 0, 0,
2777 0, 0, 0, 0,
2778 ];
2779
2780 let drop_guard = crate::DropGuard::from_option(drop_callback);
2781
2782 let empty_descriptor_set_layout = unsafe {
2783 raw_device
2784 .create_descriptor_set_layout(&vk::DescriptorSetLayoutCreateInfo::default(), None)
2785 .map_err(super::map_host_device_oom_err)?
2786 };
2787
2788 let shared = Arc::new(super::DeviceShared {
2789 raw: raw_device,
2790 family_index,
2791 queue_index,
2792 raw_queue,
2793 drop_guard,
2794 instance: Arc::clone(&self.instance),
2795 physical_device: self.raw,
2796 enabled_extensions: enabled_extensions.into(),
2797 extension_fns: super::DeviceExtensionFunctions {
2798 debug_utils: debug_utils_fn,
2799 draw_indirect_count: indirect_count_fn,
2800 timeline_semaphore: timeline_semaphore_fn,
2801 ray_tracing: ray_tracing_fns,
2802 mesh_shading: mesh_shading_fns,
2803 external_memory_fd: external_memory_fd_fn,
2804 },
2805 pipeline_cache_validation_key,
2806 vendor_id: self.phd_capabilities.properties.vendor_id,
2807 timestamp_period: self.phd_capabilities.properties.limits.timestamp_period,
2808 private_caps: self.private_caps.clone(),
2809 features,
2810 workarounds: self.workarounds,
2811 render_passes: Mutex::new(Default::default()),
2812 sampler_cache: Mutex::new(super::sampler::SamplerCache::new(
2813 self.private_caps.maximum_samplers,
2814 )),
2815 memory_allocations_counter: Default::default(),
2816
2817 texture_identity_factory: super::ResourceIdentityFactory::new(),
2818 texture_view_identity_factory: super::ResourceIdentityFactory::new(),
2819 empty_descriptor_set_layout,
2820 });
2821
2822 let relay_semaphores = super::RelaySemaphores::new(&shared)?;
2823
2824 let queue = super::Queue {
2825 raw: raw_queue,
2826 device: Arc::clone(&shared),
2827 family_index,
2828 relay_semaphores: Mutex::new(relay_semaphores),
2829 signal_semaphores: Mutex::new(SemaphoreList::new(SemaphoreListMode::Signal)),
2830 wait_semaphores: Mutex::new(SemaphoreList::new(SemaphoreListMode::Wait)),
2831 };
2832
2833 let allocation_sizes = AllocationSizes::from_memory_hints(memory_hints).into();
2834
2835 let buffer_device_address = enabled_extensions.contains(&khr::buffer_device_address::NAME);
2836
2837 let mem_allocator =
2838 gpu_allocator::vulkan::Allocator::new(&gpu_allocator::vulkan::AllocatorCreateDesc {
2839 instance: self.instance.raw.clone(),
2840 device: shared.raw.clone(),
2841 physical_device: self.raw,
2842 debug_settings: Default::default(),
2843 buffer_device_address,
2844 allocation_sizes,
2845 })?;
2846
2847 let desc_allocator = super::descriptor::DescriptorAllocator::new(
2848 if let Some(di) = self.phd_capabilities.descriptor_indexing {
2849 di.max_update_after_bind_descriptors_in_all_pools
2850 } else {
2851 0
2852 },
2853 );
2854
2855 let device = super::Device {
2856 shared,
2857 mem_allocator: Mutex::new(mem_allocator),
2858 desc_allocator: Mutex::new(desc_allocator),
2859 valid_ash_memory_types,
2860 naga_options,
2861 #[cfg(feature = "renderdoc")]
2862 render_doc: Default::default(),
2863 counters: Default::default(),
2864 };
2865
2866 Ok(crate::OpenDevice { device, queue })
2867 }
2868
2869 pub fn texture_format_as_raw(&self, texture_format: wgt::TextureFormat) -> vk::Format {
2870 self.private_caps.map_texture_format(texture_format)
2871 }
2872
2873 pub unsafe fn open_with_callback<'a>(
2878 &self,
2879 features: wgt::Features,
2880 limits: &wgt::Limits,
2881 memory_hints: &wgt::MemoryHints,
2882 callback: Option<Box<super::CreateDeviceCallback<'a>>>,
2883 ) -> Result<crate::OpenDevice<super::Api>, crate::DeviceError> {
2884 let mut enabled_extensions = self.required_device_extensions(features);
2885 let mut enabled_phd_features = self.physical_device_features(&enabled_extensions, features);
2886
2887 let family_index = 0; let family_info = vk::DeviceQueueCreateInfo::default()
2889 .queue_family_index(family_index)
2890 .queue_priorities(&[1.0]);
2891 let mut family_infos = Vec::from([family_info]);
2892
2893 let mut pre_info = vk::DeviceCreateInfo::default();
2894
2895 if let Some(callback) = callback {
2896 callback(super::CreateDeviceCallbackArgs {
2897 extensions: &mut enabled_extensions,
2898 device_features: &mut enabled_phd_features,
2899 queue_create_infos: &mut family_infos,
2900 create_info: &mut pre_info,
2901 _phantom: PhantomData,
2902 })
2903 }
2904
2905 let str_pointers = enabled_extensions
2906 .iter()
2907 .map(|&s| {
2908 s.as_ptr()
2910 })
2911 .collect::<Vec<_>>();
2912
2913 let pre_info = pre_info
2914 .queue_create_infos(&family_infos)
2915 .enabled_extension_names(&str_pointers);
2916 let info = enabled_phd_features.add_to_device_create(pre_info);
2917 let raw_device = {
2918 profiling::scope!("vkCreateDevice");
2919 unsafe {
2920 self.instance
2921 .raw
2922 .create_device(self.raw, &info, None)
2923 .map_err(map_err)?
2924 }
2925 };
2926 fn map_err(err: vk::Result) -> crate::DeviceError {
2927 match err {
2928 vk::Result::ERROR_TOO_MANY_OBJECTS => crate::DeviceError::OutOfMemory,
2929 vk::Result::ERROR_INITIALIZATION_FAILED => crate::DeviceError::Lost,
2930 vk::Result::ERROR_EXTENSION_NOT_PRESENT | vk::Result::ERROR_FEATURE_NOT_PRESENT => {
2931 crate::hal_usage_error(err)
2932 }
2933 other => super::map_host_device_oom_and_lost_err(other),
2934 }
2935 }
2936
2937 unsafe {
2938 self.device_from_raw(
2939 raw_device,
2940 None,
2941 &enabled_extensions,
2942 features,
2943 limits,
2944 memory_hints,
2945 family_info.queue_family_index,
2946 0,
2947 )
2948 }
2949 }
2950}
2951
2952impl crate::Adapter for super::Adapter {
2953 type A = super::Api;
2954
2955 unsafe fn open(
2956 &self,
2957 features: wgt::Features,
2958 limits: &wgt::Limits,
2959 memory_hints: &wgt::MemoryHints,
2960 ) -> Result<crate::OpenDevice<super::Api>, crate::DeviceError> {
2961 unsafe { self.open_with_callback(features, limits, memory_hints, None) }
2962 }
2963
2964 unsafe fn texture_format_capabilities(
2965 &self,
2966 format: wgt::TextureFormat,
2967 ) -> crate::TextureFormatCapabilities {
2968 use crate::TextureFormatCapabilities as Tfc;
2969
2970 let vk_format = self.private_caps.map_texture_format(format);
2971 let properties = unsafe {
2972 self.instance
2973 .raw
2974 .get_physical_device_format_properties(self.raw, vk_format)
2975 };
2976 let features = properties.optimal_tiling_features;
2977
2978 let mut flags = Tfc::empty();
2979 flags.set(
2980 Tfc::SAMPLED,
2981 features.contains(vk::FormatFeatureFlags::SAMPLED_IMAGE),
2982 );
2983 flags.set(
2984 Tfc::SAMPLED_LINEAR,
2985 features.contains(vk::FormatFeatureFlags::SAMPLED_IMAGE_FILTER_LINEAR),
2986 );
2987 flags.set(
2992 Tfc::STORAGE_READ_WRITE
2993 | Tfc::STORAGE_WRITE_ONLY
2994 | Tfc::STORAGE_READ_ONLY
2995 | Tfc::STORAGE_ATOMIC,
2996 features.contains(vk::FormatFeatureFlags::STORAGE_IMAGE),
2997 );
2998 flags.set(
2999 Tfc::STORAGE_ATOMIC,
3000 features.contains(vk::FormatFeatureFlags::STORAGE_IMAGE_ATOMIC),
3001 );
3002 flags.set(
3003 Tfc::COLOR_ATTACHMENT,
3004 features.contains(vk::FormatFeatureFlags::COLOR_ATTACHMENT),
3005 );
3006 flags.set(
3007 Tfc::COLOR_ATTACHMENT_BLEND,
3008 features.contains(vk::FormatFeatureFlags::COLOR_ATTACHMENT_BLEND),
3009 );
3010 flags.set(
3011 Tfc::DEPTH_STENCIL_ATTACHMENT,
3012 features.contains(vk::FormatFeatureFlags::DEPTH_STENCIL_ATTACHMENT),
3013 );
3014 flags.set(
3015 Tfc::COPY_SRC,
3016 features.intersects(vk::FormatFeatureFlags::TRANSFER_SRC),
3017 );
3018 flags.set(
3019 Tfc::COPY_DST,
3020 features.intersects(vk::FormatFeatureFlags::TRANSFER_DST),
3021 );
3022 flags.set(
3023 Tfc::STORAGE_ATOMIC,
3024 features.intersects(vk::FormatFeatureFlags::STORAGE_IMAGE_ATOMIC),
3025 );
3026 flags.set(Tfc::MULTISAMPLE_RESOLVE, !format.is_compressed());
3028
3029 let format_aspect = crate::FormatAspects::from(format);
3031 let limits = self.phd_capabilities.properties.limits;
3032
3033 let sample_flags = if format_aspect.contains(crate::FormatAspects::DEPTH) {
3034 limits
3035 .framebuffer_depth_sample_counts
3036 .min(limits.sampled_image_depth_sample_counts)
3037 } else if format_aspect.contains(crate::FormatAspects::STENCIL) {
3038 limits
3039 .framebuffer_stencil_sample_counts
3040 .min(limits.sampled_image_stencil_sample_counts)
3041 } else {
3042 let first_aspect = format_aspect
3043 .iter()
3044 .next()
3045 .expect("All texture should at least one aspect")
3046 .map();
3047
3048 assert_ne!(first_aspect, wgt::TextureAspect::DepthOnly);
3050 assert_ne!(first_aspect, wgt::TextureAspect::StencilOnly);
3051
3052 match format.sample_type(Some(first_aspect), None).unwrap() {
3053 wgt::TextureSampleType::Float { .. } => limits
3054 .framebuffer_color_sample_counts
3055 .min(limits.sampled_image_color_sample_counts),
3056 wgt::TextureSampleType::Sint | wgt::TextureSampleType::Uint => {
3057 limits.sampled_image_integer_sample_counts
3058 }
3059 _ => unreachable!(),
3060 }
3061 };
3062
3063 flags.set(
3064 Tfc::MULTISAMPLE_X2,
3065 sample_flags.contains(vk::SampleCountFlags::TYPE_2),
3066 );
3067 flags.set(
3068 Tfc::MULTISAMPLE_X4,
3069 sample_flags.contains(vk::SampleCountFlags::TYPE_4),
3070 );
3071 flags.set(
3072 Tfc::MULTISAMPLE_X8,
3073 sample_flags.contains(vk::SampleCountFlags::TYPE_8),
3074 );
3075 flags.set(
3076 Tfc::MULTISAMPLE_X16,
3077 sample_flags.contains(vk::SampleCountFlags::TYPE_16),
3078 );
3079
3080 flags
3081 }
3082
3083 unsafe fn surface_capabilities(
3084 &self,
3085 surface: &super::Surface,
3086 ) -> Option<crate::SurfaceCapabilities> {
3087 surface.inner.surface_capabilities(self)
3088 }
3089
3090 unsafe fn get_presentation_timestamp(&self) -> wgt::PresentationTimestamp {
3091 #[cfg(unix)]
3096 {
3097 let mut timespec = libc::timespec {
3098 tv_sec: 0,
3099 tv_nsec: 0,
3100 };
3101 unsafe {
3102 libc::clock_gettime(libc::CLOCK_MONOTONIC, &mut timespec);
3103 }
3104
3105 wgt::PresentationTimestamp(
3106 timespec.tv_sec as u128 * 1_000_000_000 + timespec.tv_nsec as u128,
3107 )
3108 }
3109 #[cfg(not(unix))]
3110 {
3111 wgt::PresentationTimestamp::INVALID_TIMESTAMP
3112 }
3113 }
3114
3115 fn get_ordered_buffer_usages(&self) -> wgt::BufferUses {
3116 wgt::BufferUses::INCLUSIVE | wgt::BufferUses::MAP_WRITE
3117 }
3118
3119 fn get_ordered_texture_usages(&self) -> wgt::TextureUses {
3124 wgt::TextureUses::INCLUSIVE
3125 }
3126}
3127
3128fn is_format_16bit_norm_supported(instance: &ash::Instance, phd: vk::PhysicalDevice) -> bool {
3129 [
3130 vk::Format::R16_UNORM,
3131 vk::Format::R16_SNORM,
3132 vk::Format::R16G16_UNORM,
3133 vk::Format::R16G16_SNORM,
3134 vk::Format::R16G16B16A16_UNORM,
3135 vk::Format::R16G16B16A16_SNORM,
3136 ]
3137 .into_iter()
3138 .all(|format| {
3139 supports_format(
3140 instance,
3141 phd,
3142 format,
3143 vk::ImageTiling::OPTIMAL,
3144 vk::FormatFeatureFlags::SAMPLED_IMAGE
3145 | vk::FormatFeatureFlags::STORAGE_IMAGE
3146 | vk::FormatFeatureFlags::TRANSFER_SRC
3147 | vk::FormatFeatureFlags::TRANSFER_DST,
3148 )
3149 })
3150}
3151
3152fn is_float32_filterable_supported(instance: &ash::Instance, phd: vk::PhysicalDevice) -> bool {
3153 [
3154 vk::Format::R32_SFLOAT,
3155 vk::Format::R32G32_SFLOAT,
3156 vk::Format::R32G32B32A32_SFLOAT,
3157 ]
3158 .into_iter()
3159 .all(|format| {
3160 supports_format(
3161 instance,
3162 phd,
3163 format,
3164 vk::ImageTiling::OPTIMAL,
3165 vk::FormatFeatureFlags::SAMPLED_IMAGE_FILTER_LINEAR,
3166 )
3167 })
3168}
3169
3170fn is_float32_blendable_supported(instance: &ash::Instance, phd: vk::PhysicalDevice) -> bool {
3171 [
3172 vk::Format::R32_SFLOAT,
3173 vk::Format::R32G32_SFLOAT,
3174 vk::Format::R32G32B32A32_SFLOAT,
3175 ]
3176 .into_iter()
3177 .all(|format| {
3178 supports_format(
3179 instance,
3180 phd,
3181 format,
3182 vk::ImageTiling::OPTIMAL,
3183 vk::FormatFeatureFlags::COLOR_ATTACHMENT_BLEND,
3184 )
3185 })
3186}
3187
3188fn supports_format(
3189 instance: &ash::Instance,
3190 phd: vk::PhysicalDevice,
3191 format: vk::Format,
3192 tiling: vk::ImageTiling,
3193 features: vk::FormatFeatureFlags,
3194) -> bool {
3195 let properties = unsafe { instance.get_physical_device_format_properties(phd, format) };
3196 match tiling {
3197 vk::ImageTiling::LINEAR => properties.linear_tiling_features.contains(features),
3198 vk::ImageTiling::OPTIMAL => properties.optimal_tiling_features.contains(features),
3199 _ => false,
3200 }
3201}
3202
3203fn supports_astc_3d(instance: &ash::Instance, phd: vk::PhysicalDevice) -> bool {
3204 [
3205 vk::Format::ASTC_4X4_UNORM_BLOCK,
3206 vk::Format::ASTC_4X4_SRGB_BLOCK,
3207 vk::Format::ASTC_5X4_UNORM_BLOCK,
3208 vk::Format::ASTC_5X4_SRGB_BLOCK,
3209 vk::Format::ASTC_5X5_UNORM_BLOCK,
3210 vk::Format::ASTC_5X5_SRGB_BLOCK,
3211 vk::Format::ASTC_6X5_UNORM_BLOCK,
3212 vk::Format::ASTC_6X5_SRGB_BLOCK,
3213 vk::Format::ASTC_6X6_UNORM_BLOCK,
3214 vk::Format::ASTC_6X6_SRGB_BLOCK,
3215 vk::Format::ASTC_8X5_UNORM_BLOCK,
3216 vk::Format::ASTC_8X5_SRGB_BLOCK,
3217 vk::Format::ASTC_8X6_UNORM_BLOCK,
3218 vk::Format::ASTC_8X6_SRGB_BLOCK,
3219 vk::Format::ASTC_8X8_UNORM_BLOCK,
3220 vk::Format::ASTC_8X8_SRGB_BLOCK,
3221 vk::Format::ASTC_10X5_UNORM_BLOCK,
3222 vk::Format::ASTC_10X5_SRGB_BLOCK,
3223 vk::Format::ASTC_10X6_UNORM_BLOCK,
3224 vk::Format::ASTC_10X6_SRGB_BLOCK,
3225 vk::Format::ASTC_10X8_UNORM_BLOCK,
3226 vk::Format::ASTC_10X8_SRGB_BLOCK,
3227 vk::Format::ASTC_10X10_UNORM_BLOCK,
3228 vk::Format::ASTC_10X10_SRGB_BLOCK,
3229 vk::Format::ASTC_12X10_UNORM_BLOCK,
3230 vk::Format::ASTC_12X10_SRGB_BLOCK,
3231 vk::Format::ASTC_12X12_UNORM_BLOCK,
3232 vk::Format::ASTC_12X12_SRGB_BLOCK,
3233 ]
3234 .into_iter()
3235 .all(|format| {
3236 unsafe {
3237 instance.get_physical_device_image_format_properties(
3238 phd,
3239 format,
3240 vk::ImageType::TYPE_3D,
3241 vk::ImageTiling::OPTIMAL,
3242 vk::ImageUsageFlags::SAMPLED,
3243 vk::ImageCreateFlags::empty(),
3244 )
3245 }
3246 .is_ok()
3247 })
3248}
3249
3250fn supports_bgra8unorm_storage(
3251 instance: &ash::Instance,
3252 phd: vk::PhysicalDevice,
3253 device_api_version: u32,
3254) -> bool {
3255 if device_api_version < vk::API_VERSION_1_3 {
3261 return false;
3262 }
3263
3264 unsafe {
3265 let mut properties3 = vk::FormatProperties3::default();
3266 let mut properties2 = vk::FormatProperties2::default().push_next(&mut properties3);
3267
3268 instance.get_physical_device_format_properties2(
3269 phd,
3270 vk::Format::B8G8R8A8_UNORM,
3271 &mut properties2,
3272 );
3273
3274 let features2 = properties2.format_properties.optimal_tiling_features;
3275 let features3 = properties3.optimal_tiling_features;
3276
3277 features2.contains(vk::FormatFeatureFlags::STORAGE_IMAGE)
3278 && features3.contains(vk::FormatFeatureFlags2::STORAGE_WRITE_WITHOUT_FORMAT)
3279 }
3280}
3281
3282fn is_intel_igpu_outdated_for_robustness2(
3286 props: vk::PhysicalDeviceProperties,
3287 driver: Option<vk::PhysicalDeviceDriverPropertiesKHR>,
3288) -> bool {
3289 const DRIVER_VERSION_WORKING: u32 = (101 << 14) | 2115; let is_outdated = props.vendor_id == crate::auxil::db::intel::VENDOR
3292 && props.device_type == vk::PhysicalDeviceType::INTEGRATED_GPU
3293 && props.driver_version < DRIVER_VERSION_WORKING
3294 && driver
3295 .map(|driver| driver.driver_id == vk::DriverId::INTEL_PROPRIETARY_WINDOWS)
3296 .unwrap_or_default();
3297
3298 if is_outdated {
3299 log::debug!(
3300 "Disabling robustBufferAccess2 and robustImageAccess2: IntegratedGpu Intel Driver is outdated. Found with version 0x{:X}, less than the known good version 0x{:X} (31.0.101.2115)",
3301 props.driver_version,
3302 DRIVER_VERSION_WORKING
3303 );
3304 }
3305 is_outdated
3306}
3307
3308fn map_vk_component_type(ty: vk::ComponentTypeKHR) -> Option<wgt::CooperativeScalarType> {
3310 match ty {
3311 vk::ComponentTypeKHR::FLOAT16 => Some(wgt::CooperativeScalarType::F16),
3312 vk::ComponentTypeKHR::FLOAT32 => Some(wgt::CooperativeScalarType::F32),
3313 vk::ComponentTypeKHR::SINT32 => Some(wgt::CooperativeScalarType::I32),
3314 vk::ComponentTypeKHR::UINT32 => Some(wgt::CooperativeScalarType::U32),
3315 _ => None,
3316 }
3317}
3318
3319fn map_vk_cooperative_size(size: u32) -> Option<u32> {
3321 match size {
3322 8 | 16 => Some(size),
3323 _ => None,
3324 }
3325}
3326
3327fn query_cooperative_matrix_properties(
3329 coop_matrix: &khr::cooperative_matrix::Instance,
3330 phd: vk::PhysicalDevice,
3331) -> Vec<wgt::CooperativeMatrixProperties> {
3332 let vk_properties =
3333 match unsafe { coop_matrix.get_physical_device_cooperative_matrix_properties(phd) } {
3334 Ok(props) => props,
3335 Err(e) => {
3336 log::warn!("Failed to query cooperative matrix properties: {e:?}");
3337 return Vec::new();
3338 }
3339 };
3340
3341 log::debug!(
3342 "Vulkan reports {} cooperative matrix configurations",
3343 vk_properties.len()
3344 );
3345
3346 let mut result = Vec::new();
3347 for prop in &vk_properties {
3348 log::debug!(
3349 " Vulkan coop matrix: M={} N={} K={} A={:?} B={:?} C={:?} Result={:?} scope={:?} saturating={}",
3350 prop.m_size,
3351 prop.n_size,
3352 prop.k_size,
3353 prop.a_type,
3354 prop.b_type,
3355 prop.c_type,
3356 prop.result_type,
3357 prop.scope,
3358 prop.saturating_accumulation
3359 );
3360
3361 if prop.scope != vk::ScopeKHR::SUBGROUP {
3363 log::debug!(" Skipped: scope is not SUBGROUP");
3364 continue;
3365 }
3366
3367 let m_size = match map_vk_cooperative_size(prop.m_size) {
3369 Some(s) => s,
3370 None => {
3371 log::debug!(" Skipped: M size {} not supported", prop.m_size);
3372 continue;
3373 }
3374 };
3375 let n_size = match map_vk_cooperative_size(prop.n_size) {
3376 Some(s) => s,
3377 None => {
3378 log::debug!(" Skipped: N size {} not supported", prop.n_size);
3379 continue;
3380 }
3381 };
3382 let k_size = match map_vk_cooperative_size(prop.k_size) {
3383 Some(s) => s,
3384 None => {
3385 log::debug!(" Skipped: K size {} not supported", prop.k_size);
3386 continue;
3387 }
3388 };
3389
3390 let ab_type = match map_vk_component_type(prop.a_type) {
3392 Some(t) if Some(t) == map_vk_component_type(prop.b_type) => t,
3393 _ => {
3394 log::debug!(
3395 " Skipped: A/B types {:?}/{:?} not supported or don't match",
3396 prop.a_type,
3397 prop.b_type
3398 );
3399 continue;
3400 }
3401 };
3402 let cr_type = match map_vk_component_type(prop.c_type) {
3403 Some(t) if Some(t) == map_vk_component_type(prop.result_type) => t,
3404 _ => {
3405 log::debug!(
3406 " Skipped: C/Result types {:?}/{:?} not supported or don't match",
3407 prop.c_type,
3408 prop.result_type
3409 );
3410 continue;
3411 }
3412 };
3413
3414 log::debug!(" Accepted!");
3415 result.push(wgt::CooperativeMatrixProperties {
3416 m_size,
3417 n_size,
3418 k_size,
3419 ab_type,
3420 cr_type,
3421 saturating_accumulation: prop.saturating_accumulation != 0,
3422 });
3423 }
3424
3425 log::info!(
3426 "Found {} cooperative matrix configurations supported by wgpu",
3427 result.len()
3428 );
3429 result
3430}