1use alloc::{borrow::ToOwned as _, boxed::Box, ffi::CString, string::String, sync::Arc, vec::Vec};
2use core::{
3 ffi::{c_void, CStr},
4 marker::PhantomData,
5 slice,
6 str::FromStr,
7};
8use std::thread;
9
10use arrayvec::ArrayVec;
11use ash::{ext, khr, vk};
12use parking_lot::RwLock;
13
14unsafe extern "system" fn debug_utils_messenger_callback(
15 message_severity: vk::DebugUtilsMessageSeverityFlagsEXT,
16 message_type: vk::DebugUtilsMessageTypeFlagsEXT,
17 callback_data_ptr: *const vk::DebugUtilsMessengerCallbackDataEXT,
18 user_data: *mut c_void,
19) -> vk::Bool32 {
20 use alloc::borrow::Cow;
21
22 if thread::panicking() {
23 return vk::FALSE;
24 }
25
26 let cd = unsafe { &*callback_data_ptr };
27 let user_data = unsafe { &*user_data.cast::<super::DebugUtilsMessengerUserData>() };
28
29 const VUID_VKCMDENDDEBUGUTILSLABELEXT_COMMANDBUFFER_01912: i32 = 0x56146426;
30 if cd.message_id_number == VUID_VKCMDENDDEBUGUTILSLABELEXT_COMMANDBUFFER_01912 {
31 if let Some(layer_properties) = user_data.validation_layer_properties.as_ref() {
35 if layer_properties.layer_description.as_ref() == c"Khronos Validation Layer"
36 && layer_properties.layer_spec_version >= vk::make_api_version(0, 1, 3, 240)
37 && layer_properties.layer_spec_version <= vk::make_api_version(0, 1, 3, 250)
38 {
39 return vk::FALSE;
40 }
41 }
42 }
43
44 const VUID_VKSWAPCHAINCREATEINFOKHR_PNEXT_07781: i32 = 0x4c8929c1;
48 if cd.message_id_number == VUID_VKSWAPCHAINCREATEINFOKHR_PNEXT_07781 {
49 return vk::FALSE;
50 }
51
52 const VUID_VKRENDERPASSBEGININFO_FRAMEBUFFER_04627: i32 = 0x45125641;
59 if cd.message_id_number == VUID_VKRENDERPASSBEGININFO_FRAMEBUFFER_04627
60 && user_data.has_obs_layer
61 {
62 return vk::FALSE;
63 }
64
65 const VUID_VKCMDCOPYIMAGETOBUFFER_PREGIONS_00184: i32 = 0x45ef177c;
69 if cd.message_id_number == VUID_VKCMDCOPYIMAGETOBUFFER_PREGIONS_00184 {
70 return vk::FALSE;
71 }
72
73 const VUID_STANDALONESPIRV_NONE_10684: i32 = 0xb210f7c2_u32 as i32;
78 if cd.message_id_number == VUID_STANDALONESPIRV_NONE_10684 {
79 return vk::FALSE;
80 }
81
82 let level = match message_severity {
83 vk::DebugUtilsMessageSeverityFlagsEXT::VERBOSE => log::Level::Trace,
86 vk::DebugUtilsMessageSeverityFlagsEXT::INFO => log::Level::Debug,
87 vk::DebugUtilsMessageSeverityFlagsEXT::WARNING => log::Level::Warn,
88 vk::DebugUtilsMessageSeverityFlagsEXT::ERROR => log::Level::Error,
89 _ => log::Level::Warn,
90 };
91
92 let message_id_name =
93 unsafe { cd.message_id_name_as_c_str() }.map_or(Cow::Borrowed(""), CStr::to_string_lossy);
94 let message = unsafe { cd.message_as_c_str() }.map_or(Cow::Borrowed(""), CStr::to_string_lossy);
95
96 let _ = std::panic::catch_unwind(|| {
97 log::log!(
98 level,
99 "{:?} [{} (0x{:x})]\n\t{}",
100 message_type,
101 message_id_name,
102 cd.message_id_number,
103 message,
104 );
105 });
106
107 if cd.queue_label_count != 0 {
108 let labels =
109 unsafe { slice::from_raw_parts(cd.p_queue_labels, cd.queue_label_count as usize) };
110 let names = labels
111 .iter()
112 .flat_map(|dul_obj| unsafe { dul_obj.label_name_as_c_str() }.map(CStr::to_string_lossy))
113 .collect::<Vec<_>>();
114
115 let _ = std::panic::catch_unwind(|| {
116 log::log!(level, "\tqueues: {}", names.join(", "));
117 });
118 }
119
120 if cd.cmd_buf_label_count != 0 {
121 let labels =
122 unsafe { slice::from_raw_parts(cd.p_cmd_buf_labels, cd.cmd_buf_label_count as usize) };
123 let names = labels
124 .iter()
125 .flat_map(|dul_obj| unsafe { dul_obj.label_name_as_c_str() }.map(CStr::to_string_lossy))
126 .collect::<Vec<_>>();
127
128 let _ = std::panic::catch_unwind(|| {
129 log::log!(level, "\tcommand buffers: {}", names.join(", "));
130 });
131 }
132
133 if cd.object_count != 0 {
134 let labels = unsafe { slice::from_raw_parts(cd.p_objects, cd.object_count as usize) };
135 let names = labels
137 .iter()
138 .map(|obj_info| {
139 let name = unsafe { obj_info.object_name_as_c_str() }
140 .map_or(Cow::Borrowed("?"), CStr::to_string_lossy);
141
142 format!(
143 "(type: {:?}, hndl: 0x{:x}, name: {})",
144 obj_info.object_type, obj_info.object_handle, name
145 )
146 })
147 .collect::<Vec<_>>();
148 let _ = std::panic::catch_unwind(|| {
149 log::log!(level, "\tobjects: {}", names.join(", "));
150 });
151 }
152
153 #[cfg(feature = "validation_canary")]
154 if cfg!(debug_assertions) && level == log::Level::Error {
155 use alloc::string::ToString as _;
156
157 crate::VALIDATION_CANARY.add(message.to_string());
159 }
160
161 vk::FALSE
162}
163
164impl super::DebugUtilsCreateInfo {
165 fn to_vk_create_info(&self) -> vk::DebugUtilsMessengerCreateInfoEXT<'_> {
166 let user_data_ptr: *const super::DebugUtilsMessengerUserData = &*self.callback_data;
167 vk::DebugUtilsMessengerCreateInfoEXT::default()
168 .message_severity(self.severity)
169 .message_type(self.message_type)
170 .user_data(user_data_ptr as *mut _)
171 .pfn_user_callback(Some(debug_utils_messenger_callback))
172 }
173}
174
175impl super::InstanceShared {
176 pub fn entry(&self) -> &ash::Entry {
177 &self.entry
178 }
179
180 pub fn raw_instance(&self) -> &ash::Instance {
181 &self.raw
182 }
183
184 pub fn instance_api_version(&self) -> u32 {
185 self.instance_api_version
186 }
187
188 pub fn extensions(&self) -> &[&'static CStr] {
189 &self.extensions[..]
190 }
191}
192
193impl super::Instance {
194 pub fn shared_instance(&self) -> &super::InstanceShared {
195 &self.shared
196 }
197
198 fn enumerate_instance_extension_properties(
199 entry: &ash::Entry,
200 layer_name: Option<&CStr>,
201 ) -> Result<Vec<vk::ExtensionProperties>, crate::InstanceError> {
202 let instance_extensions = {
203 profiling::scope!("vkEnumerateInstanceExtensionProperties");
204 unsafe { entry.enumerate_instance_extension_properties(layer_name) }
205 };
206 instance_extensions.map_err(|e| {
207 crate::InstanceError::with_source(
208 String::from("enumerate_instance_extension_properties() failed"),
209 e,
210 )
211 })
212 }
213
214 pub fn desired_extensions(
228 entry: &ash::Entry,
229 _instance_api_version: u32,
230 flags: wgt::InstanceFlags,
231 ) -> Result<Vec<&'static CStr>, crate::InstanceError> {
232 let instance_extensions = Self::enumerate_instance_extension_properties(entry, None)?;
233
234 let mut extensions: Vec<&'static CStr> = Vec::new();
236
237 extensions.push(khr::surface::NAME);
239
240 if cfg!(all(
242 unix,
243 not(target_os = "android"),
244 not(target_os = "macos")
245 )) {
246 extensions.push(khr::xlib_surface::NAME);
248 extensions.push(khr::xcb_surface::NAME);
250 extensions.push(khr::wayland_surface::NAME);
252 }
253 if cfg!(target_os = "android") {
254 extensions.push(khr::android_surface::NAME);
256 }
257 if cfg!(target_os = "windows") {
258 extensions.push(khr::win32_surface::NAME);
260 }
261 if cfg!(target_os = "macos") {
262 extensions.push(ext::metal_surface::NAME);
264 extensions.push(khr::portability_enumeration::NAME);
265 }
266 if cfg!(drm) {
267 extensions.push(ext::acquire_drm_display::NAME);
269 extensions.push(ext::direct_mode_display::NAME);
270 extensions.push(khr::display::NAME);
271 extensions.push(khr::get_physical_device_properties2::NAME);
272 extensions.push(khr::get_display_properties2::NAME);
273 }
274
275 if flags.contains(wgt::InstanceFlags::DEBUG) {
276 extensions.push(ext::debug_utils::NAME);
278 }
279
280 extensions.push(ext::swapchain_colorspace::NAME);
283
284 extensions.push(khr::get_physical_device_properties2::NAME);
288
289 extensions.retain(|&ext| {
291 if instance_extensions
292 .iter()
293 .any(|inst_ext| inst_ext.extension_name_as_c_str() == Ok(ext))
294 {
295 true
296 } else {
297 log::debug!("Unable to find extension: {}", ext.to_string_lossy());
298 false
299 }
300 });
301 Ok(extensions)
302 }
303
304 #[allow(clippy::too_many_arguments)]
317 pub unsafe fn from_raw(
318 entry: ash::Entry,
319 raw_instance: ash::Instance,
320 instance_api_version: u32,
321 android_sdk_version: u32,
322 debug_utils_create_info: Option<super::DebugUtilsCreateInfo>,
323 extensions: Vec<&'static CStr>,
324 flags: wgt::InstanceFlags,
325 memory_budget_thresholds: wgt::MemoryBudgetThresholds,
326 has_nv_optimus: bool,
327 drop_callback: Option<crate::DropCallback>,
328 ) -> Result<Self, crate::InstanceError> {
329 log::debug!("Instance version: 0x{instance_api_version:x}");
330
331 let debug_utils = if let Some(debug_utils_create_info) = debug_utils_create_info {
332 if extensions.contains(&ext::debug_utils::NAME) {
333 log::debug!("Enabling debug utils");
334
335 let extension = ext::debug_utils::Instance::new(&entry, &raw_instance);
336 let vk_info = debug_utils_create_info.to_vk_create_info();
337 let messenger =
338 unsafe { extension.create_debug_utils_messenger(&vk_info, None) }.unwrap();
339
340 Some(super::DebugUtils {
341 extension,
342 messenger,
343 callback_data: debug_utils_create_info.callback_data,
344 })
345 } else {
346 log::debug!("Debug utils not enabled: extension not listed");
347 None
348 }
349 } else {
350 log::debug!(
351 "Debug utils not enabled: \
352 debug_utils_user_data not passed to Instance::from_raw"
353 );
354 None
355 };
356
357 let get_physical_device_properties =
358 if extensions.contains(&khr::get_physical_device_properties2::NAME) {
359 log::debug!("Enabling device properties2");
360 Some(khr::get_physical_device_properties2::Instance::new(
361 &entry,
362 &raw_instance,
363 ))
364 } else {
365 None
366 };
367
368 let drop_guard = crate::DropGuard::from_option(drop_callback);
369
370 Ok(Self {
371 shared: Arc::new(super::InstanceShared {
372 raw: raw_instance,
373 extensions,
374 drop_guard,
375 flags,
376 memory_budget_thresholds,
377 debug_utils,
378 get_physical_device_properties,
379 entry,
380 has_nv_optimus,
381 instance_api_version,
382 android_sdk_version,
383 }),
384 })
385 }
386
387 fn create_surface_from_xlib(
388 &self,
389 dpy: *mut vk::Display,
390 window: vk::Window,
391 ) -> Result<super::Surface, crate::InstanceError> {
392 if !self.shared.extensions.contains(&khr::xlib_surface::NAME) {
393 return Err(crate::InstanceError::new(String::from(
394 "Vulkan driver does not support VK_KHR_xlib_surface",
395 )));
396 }
397
398 let surface = {
399 let xlib_loader =
400 khr::xlib_surface::Instance::new(&self.shared.entry, &self.shared.raw);
401 let info = vk::XlibSurfaceCreateInfoKHR::default()
402 .flags(vk::XlibSurfaceCreateFlagsKHR::empty())
403 .window(window)
404 .dpy(dpy);
405
406 unsafe { xlib_loader.create_xlib_surface(&info, None) }
407 .expect("XlibSurface::create_xlib_surface() failed")
408 };
409
410 Ok(self.create_surface_from_vk_surface_khr(surface))
411 }
412
413 fn create_surface_from_xcb(
414 &self,
415 connection: *mut vk::xcb_connection_t,
416 window: vk::xcb_window_t,
417 ) -> Result<super::Surface, crate::InstanceError> {
418 if !self.shared.extensions.contains(&khr::xcb_surface::NAME) {
419 return Err(crate::InstanceError::new(String::from(
420 "Vulkan driver does not support VK_KHR_xcb_surface",
421 )));
422 }
423
424 let surface = {
425 let xcb_loader = khr::xcb_surface::Instance::new(&self.shared.entry, &self.shared.raw);
426 let info = vk::XcbSurfaceCreateInfoKHR::default()
427 .flags(vk::XcbSurfaceCreateFlagsKHR::empty())
428 .window(window)
429 .connection(connection);
430
431 unsafe { xcb_loader.create_xcb_surface(&info, None) }
432 .expect("XcbSurface::create_xcb_surface() failed")
433 };
434
435 Ok(self.create_surface_from_vk_surface_khr(surface))
436 }
437
438 fn create_surface_from_wayland(
439 &self,
440 display: *mut vk::wl_display,
441 surface: *mut vk::wl_surface,
442 ) -> Result<super::Surface, crate::InstanceError> {
443 if !self.shared.extensions.contains(&khr::wayland_surface::NAME) {
444 return Err(crate::InstanceError::new(String::from(
445 "Vulkan driver does not support VK_KHR_wayland_surface",
446 )));
447 }
448
449 let surface = {
450 let w_loader =
451 khr::wayland_surface::Instance::new(&self.shared.entry, &self.shared.raw);
452 let info = vk::WaylandSurfaceCreateInfoKHR::default()
453 .flags(vk::WaylandSurfaceCreateFlagsKHR::empty())
454 .display(display)
455 .surface(surface);
456
457 unsafe { w_loader.create_wayland_surface(&info, None) }.expect("WaylandSurface failed")
458 };
459
460 Ok(self.create_surface_from_vk_surface_khr(surface))
461 }
462
463 fn create_surface_android(
464 &self,
465 window: *mut vk::ANativeWindow,
466 ) -> Result<super::Surface, crate::InstanceError> {
467 if !self.shared.extensions.contains(&khr::android_surface::NAME) {
468 return Err(crate::InstanceError::new(String::from(
469 "Vulkan driver does not support VK_KHR_android_surface",
470 )));
471 }
472
473 let surface = {
474 let a_loader =
475 khr::android_surface::Instance::new(&self.shared.entry, &self.shared.raw);
476 let info = vk::AndroidSurfaceCreateInfoKHR::default()
477 .flags(vk::AndroidSurfaceCreateFlagsKHR::empty())
478 .window(window);
479
480 unsafe { a_loader.create_android_surface(&info, None) }.expect("AndroidSurface failed")
481 };
482
483 Ok(self.create_surface_from_vk_surface_khr(surface))
484 }
485
486 fn create_surface_from_hwnd(
487 &self,
488 hinstance: vk::HINSTANCE,
489 hwnd: vk::HWND,
490 ) -> Result<super::Surface, crate::InstanceError> {
491 if !self.shared.extensions.contains(&khr::win32_surface::NAME) {
492 return Err(crate::InstanceError::new(String::from(
493 "Vulkan driver does not support VK_KHR_win32_surface",
494 )));
495 }
496
497 let surface = {
498 let info = vk::Win32SurfaceCreateInfoKHR::default()
499 .flags(vk::Win32SurfaceCreateFlagsKHR::empty())
500 .hinstance(hinstance)
501 .hwnd(hwnd);
502 let win32_loader =
503 khr::win32_surface::Instance::new(&self.shared.entry, &self.shared.raw);
504 unsafe {
505 win32_loader
506 .create_win32_surface(&info, None)
507 .expect("Unable to create Win32 surface")
508 }
509 };
510
511 Ok(self.create_surface_from_vk_surface_khr(surface))
512 }
513
514 #[cfg(target_vendor = "apple")]
515 fn create_surface_from_layer(
516 &self,
517 layer: raw_window_metal::Layer,
518 ) -> Result<super::Surface, crate::InstanceError> {
519 if !self.shared.extensions.contains(&ext::metal_surface::NAME) {
520 return Err(crate::InstanceError::new(String::from(
521 "Vulkan driver does not support VK_EXT_metal_surface",
522 )));
523 }
524
525 let surface = {
528 let metal_loader =
529 ext::metal_surface::Instance::new(&self.shared.entry, &self.shared.raw);
530 let vk_info = vk::MetalSurfaceCreateInfoEXT::default()
531 .flags(vk::MetalSurfaceCreateFlagsEXT::empty())
532 .layer(layer.as_ptr().as_ptr());
533
534 unsafe { metal_loader.create_metal_surface(&vk_info, None).unwrap() }
535 };
536
537 Ok(self.create_surface_from_vk_surface_khr(surface))
538 }
539
540 pub(super) fn create_surface_from_vk_surface_khr(
541 &self,
542 surface: vk::SurfaceKHR,
543 ) -> super::Surface {
544 let native_surface =
545 crate::vulkan::swapchain::NativeSurface::from_vk_surface_khr(self, surface);
546
547 super::Surface {
548 swapchain: RwLock::new(None),
549 inner: Box::new(native_surface),
550 }
551 }
552
553 pub unsafe fn init_with_callback(
562 desc: &crate::InstanceDescriptor<'_>,
563 callback: Option<Box<super::CreateInstanceCallback>>,
564 ) -> Result<Self, crate::InstanceError> {
565 profiling::scope!("Init Vulkan Backend");
566
567 let entry = unsafe {
568 profiling::scope!("Load vk library");
569 ash::Entry::load()
570 }
571 .map_err(|err| {
572 crate::InstanceError::with_source(String::from("missing Vulkan entry points"), err)
573 })?;
574 let version = {
575 profiling::scope!("vkEnumerateInstanceVersion");
576 unsafe { entry.try_enumerate_instance_version() }
577 };
578 let instance_api_version = match version {
579 Ok(Some(version)) => version,
581 Ok(None) => vk::API_VERSION_1_0,
582 Err(err) => {
583 return Err(crate::InstanceError::with_source(
584 String::from("try_enumerate_instance_version() failed"),
585 err,
586 ));
587 }
588 };
589
590 let app_name = CString::new(desc.name).unwrap();
591 let app_info = vk::ApplicationInfo::default()
592 .application_name(app_name.as_c_str())
593 .application_version(1)
594 .engine_name(c"wgpu-hal")
595 .engine_version(2)
596 .api_version(
597 if instance_api_version < vk::API_VERSION_1_1 {
599 vk::API_VERSION_1_0
600 } else {
601 vk::API_VERSION_1_3
610 },
611 );
612
613 let mut extensions = Self::desired_extensions(&entry, instance_api_version, desc.flags)?;
614 let mut create_info = vk::InstanceCreateInfo::default();
615
616 if let Some(callback) = callback {
617 callback(super::CreateInstanceCallbackArgs {
618 extensions: &mut extensions,
619 create_info: &mut create_info,
620 entry: &entry,
621 _phantom: PhantomData,
622 });
623 }
624
625 let instance_layers = {
626 profiling::scope!("vkEnumerateInstanceLayerProperties");
627 unsafe { entry.enumerate_instance_layer_properties() }
628 };
629 let instance_layers = instance_layers.map_err(|e| {
630 log::debug!("enumerate_instance_layer_properties: {e:?}");
631 crate::InstanceError::with_source(
632 String::from("enumerate_instance_layer_properties() failed"),
633 e,
634 )
635 })?;
636
637 fn find_layer<'layers>(
638 instance_layers: &'layers [vk::LayerProperties],
639 name: &CStr,
640 ) -> Option<&'layers vk::LayerProperties> {
641 instance_layers
642 .iter()
643 .find(|inst_layer| inst_layer.layer_name_as_c_str() == Ok(name))
644 }
645
646 let validation_layer_name = c"VK_LAYER_KHRONOS_validation";
647 let validation_layer_properties = find_layer(&instance_layers, validation_layer_name);
648
649 let validation_features_are_enabled = if validation_layer_properties.is_some() {
652 let exts =
654 Self::enumerate_instance_extension_properties(&entry, Some(validation_layer_name))?;
655 let mut ext_names = exts
657 .iter()
658 .filter_map(|ext| ext.extension_name_as_c_str().ok());
659 ext_names.any(|ext_name| ext_name == ext::validation_features::NAME)
661 } else {
662 false
663 };
664
665 let should_enable_gpu_based_validation = desc
666 .flags
667 .intersects(wgt::InstanceFlags::GPU_BASED_VALIDATION)
668 && validation_features_are_enabled;
669
670 let has_nv_optimus = find_layer(&instance_layers, c"VK_LAYER_NV_optimus").is_some();
671
672 let has_obs_layer = find_layer(&instance_layers, c"VK_LAYER_OBS_HOOK").is_some();
673
674 let mut layers: Vec<&'static CStr> = Vec::new();
675
676 let has_debug_extension = extensions.contains(&ext::debug_utils::NAME);
677 let mut debug_user_data = has_debug_extension.then(|| {
678 Box::new(super::DebugUtilsMessengerUserData {
681 validation_layer_properties: None,
682 has_obs_layer,
683 })
684 });
685
686 if desc.flags.intersects(wgt::InstanceFlags::VALIDATION)
688 || should_enable_gpu_based_validation
689 {
690 if let Some(layer_properties) = validation_layer_properties {
691 layers.push(validation_layer_name);
692
693 if let Some(debug_user_data) = debug_user_data.as_mut() {
694 debug_user_data.validation_layer_properties =
695 Some(super::ValidationLayerProperties {
696 layer_description: layer_properties
697 .description_as_c_str()
698 .unwrap()
699 .to_owned(),
700 layer_spec_version: layer_properties.spec_version,
701 });
702 }
703 } else {
704 log::debug!(
705 "InstanceFlags::VALIDATION requested, but unable to find layer: {}",
706 validation_layer_name.to_string_lossy()
707 );
708 }
709 }
710 let mut debug_utils = if let Some(callback_data) = debug_user_data {
711 let mut severity = vk::DebugUtilsMessageSeverityFlagsEXT::ERROR;
713 if log::max_level() >= log::LevelFilter::Debug {
714 severity |= vk::DebugUtilsMessageSeverityFlagsEXT::VERBOSE;
715 }
716 if log::max_level() >= log::LevelFilter::Info {
717 severity |= vk::DebugUtilsMessageSeverityFlagsEXT::INFO;
718 }
719 if log::max_level() >= log::LevelFilter::Warn {
720 severity |= vk::DebugUtilsMessageSeverityFlagsEXT::WARNING;
721 }
722
723 let message_type = vk::DebugUtilsMessageTypeFlagsEXT::GENERAL
724 | vk::DebugUtilsMessageTypeFlagsEXT::VALIDATION
725 | vk::DebugUtilsMessageTypeFlagsEXT::PERFORMANCE;
726
727 let create_info = super::DebugUtilsCreateInfo {
728 severity,
729 message_type,
730 callback_data,
731 };
732
733 Some(create_info)
734 } else {
735 None
736 };
737
738 #[cfg(target_os = "android")]
739 let android_sdk_version = {
740 let properties = android_system_properties::AndroidSystemProperties::new();
741 if let Some(val) = properties.get("ro.build.version.sdk") {
743 match val.parse::<u32>() {
744 Ok(sdk_ver) => sdk_ver,
745 Err(err) => {
746 log::error!(
747 concat!(
748 "Couldn't parse Android's ",
749 "ro.build.version.sdk system property ({}): {}",
750 ),
751 val,
752 err,
753 );
754 0
755 }
756 }
757 } else {
758 log::error!("Couldn't read Android's ro.build.version.sdk system property");
759 0
760 }
761 };
762 #[cfg(not(target_os = "android"))]
763 let android_sdk_version = 0;
764
765 let mut flags = vk::InstanceCreateFlags::empty();
766
767 if extensions.contains(&khr::portability_enumeration::NAME) {
771 flags |= vk::InstanceCreateFlags::ENUMERATE_PORTABILITY_KHR;
772 }
773 let vk_instance = {
774 let str_pointers = layers
775 .iter()
776 .chain(extensions.iter())
777 .map(|&s: &&'static _| {
778 s.as_ptr()
780 })
781 .collect::<Vec<_>>();
782
783 create_info = create_info
784 .flags(flags)
785 .application_info(&app_info)
786 .enabled_layer_names(&str_pointers[..layers.len()])
787 .enabled_extension_names(&str_pointers[layers.len()..]);
788
789 let mut debug_utils_create_info = debug_utils
790 .as_mut()
791 .map(|create_info| create_info.to_vk_create_info());
792 if let Some(debug_utils_create_info) = debug_utils_create_info.as_mut() {
793 create_info = create_info.push_next(debug_utils_create_info);
794 }
795
796 let mut validation_features;
798 let mut validation_feature_list: ArrayVec<_, 3>;
799 if validation_features_are_enabled {
800 validation_feature_list = ArrayVec::new();
801
802 validation_feature_list
804 .push(vk::ValidationFeatureEnableEXT::SYNCHRONIZATION_VALIDATION);
805
806 if should_enable_gpu_based_validation {
808 validation_feature_list.push(vk::ValidationFeatureEnableEXT::GPU_ASSISTED);
809 validation_feature_list
810 .push(vk::ValidationFeatureEnableEXT::GPU_ASSISTED_RESERVE_BINDING_SLOT);
811 }
812
813 validation_features = vk::ValidationFeaturesEXT::default()
814 .enabled_validation_features(&validation_feature_list);
815 create_info = create_info.push_next(&mut validation_features);
816 }
817
818 unsafe {
819 profiling::scope!("vkCreateInstance");
820 entry.create_instance(&create_info, None)
821 }
822 .map_err(|e| {
823 crate::InstanceError::with_source(
824 String::from("Entry::create_instance() failed"),
825 e,
826 )
827 })?
828 };
829
830 unsafe {
831 Self::from_raw(
832 entry,
833 vk_instance,
834 instance_api_version,
835 android_sdk_version,
836 debug_utils,
837 extensions,
838 desc.flags,
839 desc.memory_budget_thresholds,
840 has_nv_optimus,
841 None,
842 )
843 }
844 }
845}
846
847impl Drop for super::InstanceShared {
848 fn drop(&mut self) {
849 unsafe {
850 let _du = self.debug_utils.take().inspect(|du| {
852 du.extension
853 .destroy_debug_utils_messenger(du.messenger, None);
854 });
855 if self.drop_guard.is_none() {
856 self.raw.destroy_instance(None);
857 }
858 }
859 }
860}
861
862impl crate::Instance for super::Instance {
863 type A = super::Api;
864
865 unsafe fn init(desc: &crate::InstanceDescriptor<'_>) -> Result<Self, crate::InstanceError> {
866 unsafe { Self::init_with_callback(desc, None) }
867 }
868
869 unsafe fn create_surface(
870 &self,
871 display_handle: raw_window_handle::RawDisplayHandle,
872 window_handle: raw_window_handle::RawWindowHandle,
873 ) -> Result<super::Surface, crate::InstanceError> {
874 use raw_window_handle::{RawDisplayHandle as Rdh, RawWindowHandle as Rwh};
875
876 match (window_handle, display_handle) {
879 (Rwh::Wayland(handle), Rdh::Wayland(display)) => {
880 self.create_surface_from_wayland(display.display.as_ptr(), handle.surface.as_ptr())
881 }
882 (Rwh::Xlib(handle), Rdh::Xlib(display)) => {
883 let display = display.display.expect("Display pointer is not set.");
884 self.create_surface_from_xlib(display.as_ptr(), handle.window)
885 }
886 (Rwh::Xcb(handle), Rdh::Xcb(display)) => {
887 let connection = display.connection.expect("Pointer to X-Server is not set.");
888 self.create_surface_from_xcb(connection.as_ptr(), handle.window.get())
889 }
890 #[cfg(drm)]
891 (Rwh::Drm(handle), Rdh::Drm(display)) => {
892 self.create_surface_from_drm_plane(display.fd, handle.plane)
893 }
894 (Rwh::AndroidNdk(handle), _) => {
895 self.create_surface_android(handle.a_native_window.as_ptr())
896 }
897 (Rwh::Win32(handle), _) => {
898 let hinstance = handle.hinstance.ok_or_else(|| {
899 crate::InstanceError::new(String::from(
900 "Vulkan requires raw-window-handle's Win32::hinstance to be set",
901 ))
902 })?;
903 self.create_surface_from_hwnd(hinstance.get(), handle.hwnd.get())
904 }
905 #[cfg(target_vendor = "apple")]
906 (Rwh::AppKit(handle), _)
907 if self.shared.extensions.contains(&ext::metal_surface::NAME) =>
908 {
909 let layer = unsafe { raw_window_metal::Layer::from_ns_view(handle.ns_view) };
910 self.create_surface_from_layer(layer)
911 }
912 #[cfg(target_vendor = "apple")]
913 (Rwh::UiKit(handle), _)
914 if self.shared.extensions.contains(&ext::metal_surface::NAME) =>
915 {
916 let layer = unsafe { raw_window_metal::Layer::from_ui_view(handle.ui_view) };
917 self.create_surface_from_layer(layer)
918 }
919 (_, _) => Err(crate::InstanceError::new(format!(
920 "window handle {window_handle:?} is not a Vulkan-compatible handle"
921 ))),
922 }
923 }
924
925 unsafe fn enumerate_adapters(
926 &self,
927 _surface_hint: Option<&super::Surface>,
928 ) -> Vec<crate::ExposedAdapter<super::Api>> {
929 use crate::auxil::db;
930
931 let raw_devices = match unsafe { self.shared.raw.enumerate_physical_devices() } {
932 Ok(devices) => devices,
933 Err(err) => {
934 log::error!("enumerate_adapters: {err}");
935 Vec::new()
936 }
937 };
938
939 let mut exposed_adapters = raw_devices
940 .into_iter()
941 .flat_map(|device| self.expose_adapter(device))
942 .collect::<Vec<_>>();
943
944 let has_nvidia_dgpu = exposed_adapters.iter().any(|exposed| {
946 exposed.info.device_type == wgt::DeviceType::DiscreteGpu
947 && exposed.info.vendor == db::nvidia::VENDOR
948 });
949 if cfg!(target_os = "linux") && has_nvidia_dgpu && self.shared.has_nv_optimus {
950 for exposed in exposed_adapters.iter_mut() {
951 if exposed.info.device_type == wgt::DeviceType::IntegratedGpu
952 && exposed.info.vendor == db::intel::VENDOR
953 {
954 if let Some(version) = exposed.info.driver_info.split_once("Mesa ").map(|s| {
956 let mut components = s.1.split('.');
957 let major = components.next().and_then(|s| u8::from_str(s).ok());
958 let minor = components.next().and_then(|s| u8::from_str(s).ok());
959 if let (Some(major), Some(minor)) = (major, minor) {
960 (major, minor)
961 } else {
962 (0, 0)
963 }
964 }) {
965 if version < (21, 2) {
966 log::debug!(
968 concat!(
969 "Disabling presentation on '{}' (id {:?}) ",
970 "due to NV Optimus and Intel Mesa < v21.2"
971 ),
972 exposed.info.name,
973 exposed.adapter.raw
974 );
975 exposed.adapter.private_caps.can_present = false;
976 }
977 }
978 }
979 }
980 }
981
982 exposed_adapters
983 }
984}
985
986impl crate::Surface for super::Surface {
987 type A = super::Api;
988
989 unsafe fn configure(
990 &self,
991 device: &super::Device,
992 config: &crate::SurfaceConfiguration,
993 ) -> Result<(), crate::SurfaceError> {
994 let mut swap_chain = self.swapchain.write();
996
997 let mut old = swap_chain.take();
998 if let Some(ref mut old) = old {
999 unsafe { old.release_resources(device) };
1000 }
1001
1002 let swapchain = unsafe { self.inner.create_swapchain(device, config, old)? };
1003 *swap_chain = Some(swapchain);
1004
1005 Ok(())
1006 }
1007
1008 unsafe fn unconfigure(&self, device: &super::Device) {
1009 if let Some(mut sc) = self.swapchain.write().take() {
1010 unsafe { sc.release_resources(device) };
1012 }
1013 }
1014
1015 unsafe fn acquire_texture(
1016 &self,
1017 timeout: Option<core::time::Duration>,
1018 fence: &super::Fence,
1019 ) -> Result<crate::AcquiredSurfaceTexture<super::Api>, crate::SurfaceError> {
1020 let mut swapchain = self.swapchain.write();
1021 let swapchain = swapchain.as_mut().unwrap();
1022
1023 unsafe { swapchain.acquire(timeout, fence) }
1024 }
1025
1026 unsafe fn discard_texture(&self, texture: super::SurfaceTexture) {
1027 unsafe {
1028 self.swapchain
1029 .write()
1030 .as_mut()
1031 .unwrap()
1032 .discard_texture(texture)
1033 .unwrap()
1034 };
1035 }
1036}