2022-11-26 14:27:19 -05:00
|
|
|
// Copyright 2022 Google LLC
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
//
|
|
|
|
// Also licensed under MIT license, at your choice.
|
|
|
|
|
|
|
|
//! Simple helpers for managing wgpu state and surfaces.
|
|
|
|
|
2023-01-24 20:25:44 -08:00
|
|
|
use std::future::Future;
|
|
|
|
|
2022-11-26 14:56:43 -05:00
|
|
|
use super::Result;
|
|
|
|
|
2022-11-26 14:27:19 -05:00
|
|
|
use raw_window_handle::{HasRawDisplayHandle, HasRawWindowHandle};
|
2022-12-13 23:35:56 +00:00
|
|
|
use wgpu::{
|
2023-03-05 11:33:30 +00:00
|
|
|
Adapter, Device, Instance, Limits, Queue, Surface, SurfaceConfiguration, TextureFormat,
|
2022-12-13 23:35:56 +00:00
|
|
|
};
|
2022-11-26 14:27:19 -05:00
|
|
|
|
|
|
|
/// Simple render context that maintains wgpu state for rendering the pipeline.
|
|
|
|
pub struct RenderContext {
|
|
|
|
pub instance: Instance,
|
2022-12-13 23:38:50 +00:00
|
|
|
pub devices: Vec<DeviceHandle>,
|
2022-12-13 23:35:56 +00:00
|
|
|
}
|
|
|
|
|
2022-12-13 23:38:50 +00:00
|
|
|
pub struct DeviceHandle {
|
2022-12-13 23:35:56 +00:00
|
|
|
adapter: Adapter,
|
2022-12-13 23:38:50 +00:00
|
|
|
pub device: Device,
|
|
|
|
pub queue: Queue,
|
2022-11-26 14:27:19 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl RenderContext {
|
2022-12-13 23:35:56 +00:00
|
|
|
pub fn new() -> Result<Self> {
|
2023-01-31 16:07:03 +00:00
|
|
|
let instance = Instance::new(wgpu::InstanceDescriptor {
|
|
|
|
backends: wgpu::Backends::PRIMARY,
|
|
|
|
dx12_shader_compiler: wgpu::Dx12Compiler::Fxc,
|
|
|
|
});
|
2022-11-26 14:27:19 -05:00
|
|
|
Ok(Self {
|
|
|
|
instance,
|
2022-12-13 23:35:56 +00:00
|
|
|
devices: Vec::new(),
|
2022-11-26 14:27:19 -05:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a new surface for the specified window and dimensions.
|
2022-12-13 23:35:56 +00:00
|
|
|
pub async fn create_surface<W>(&mut self, window: &W, width: u32, height: u32) -> RenderSurface
|
2022-11-26 14:27:19 -05:00
|
|
|
where
|
|
|
|
W: HasRawWindowHandle + HasRawDisplayHandle,
|
|
|
|
{
|
2023-01-31 16:07:03 +00:00
|
|
|
let surface = unsafe { self.instance.create_surface(window) }.unwrap();
|
2023-03-05 11:33:30 +00:00
|
|
|
let dev_id = self.device(Some(&surface)).await.unwrap();
|
|
|
|
|
|
|
|
let device_handle = &self.devices[dev_id];
|
|
|
|
let capabilities = surface.get_capabilities(&device_handle.adapter);
|
|
|
|
let format = capabilities
|
|
|
|
.formats
|
|
|
|
.into_iter()
|
|
|
|
.find(|it| matches!(it, TextureFormat::Rgba8Unorm | TextureFormat::Bgra8Unorm))
|
|
|
|
.expect("surface should support Rgba8Unorm or Bgra8Unorm");
|
|
|
|
|
2022-11-26 14:27:19 -05:00
|
|
|
let config = wgpu::SurfaceConfiguration {
|
|
|
|
usage: wgpu::TextureUsages::RENDER_ATTACHMENT,
|
|
|
|
format,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
present_mode: wgpu::PresentMode::Fifo,
|
|
|
|
alpha_mode: wgpu::CompositeAlphaMode::Auto,
|
2023-01-31 16:07:03 +00:00
|
|
|
view_formats: vec![],
|
2022-11-26 14:27:19 -05:00
|
|
|
};
|
2022-12-13 23:35:56 +00:00
|
|
|
surface.configure(&self.devices[dev_id].device, &config);
|
|
|
|
RenderSurface {
|
|
|
|
surface,
|
|
|
|
config,
|
|
|
|
dev_id,
|
2023-03-05 11:33:30 +00:00
|
|
|
format,
|
2022-12-13 23:35:56 +00:00
|
|
|
}
|
2022-11-26 14:27:19 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Resizes the surface to the new dimensions.
|
|
|
|
pub fn resize_surface(&self, surface: &mut RenderSurface, width: u32, height: u32) {
|
|
|
|
surface.config.width = width;
|
|
|
|
surface.config.height = height;
|
2022-12-13 23:35:56 +00:00
|
|
|
surface
|
|
|
|
.surface
|
|
|
|
.configure(&self.devices[surface.dev_id].device, &surface.config);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Finds or creates a compatible device handle id.
|
2023-02-09 09:34:53 +00:00
|
|
|
pub async fn device(&mut self, compatible_surface: Option<&Surface>) -> Option<usize> {
|
2022-12-13 23:35:56 +00:00
|
|
|
let compatible = match compatible_surface {
|
|
|
|
Some(s) => self
|
|
|
|
.devices
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.find(|(_, d)| d.adapter.is_surface_supported(s))
|
|
|
|
.map(|(i, _)| i),
|
|
|
|
None => (!self.devices.is_empty()).then_some(0),
|
|
|
|
};
|
|
|
|
if compatible.is_none() {
|
|
|
|
return self.new_device(compatible_surface).await;
|
|
|
|
}
|
2023-01-14 11:07:07 -05:00
|
|
|
compatible
|
2022-12-13 23:35:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a compatible device handle id.
|
|
|
|
async fn new_device(&mut self, compatible_surface: Option<&Surface>) -> Option<usize> {
|
2023-01-21 14:29:23 -06:00
|
|
|
let adapter = wgpu::util::initialize_adapter_from_env_or_default(
|
|
|
|
&self.instance,
|
|
|
|
wgpu::Backends::PRIMARY,
|
|
|
|
compatible_surface,
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-13 23:35:56 +00:00
|
|
|
let features = adapter.features();
|
|
|
|
let limits = Limits::default();
|
|
|
|
let (device, queue) = adapter
|
|
|
|
.request_device(
|
|
|
|
&wgpu::DeviceDescriptor {
|
|
|
|
label: None,
|
|
|
|
features: features
|
|
|
|
& (wgpu::Features::TIMESTAMP_QUERY | wgpu::Features::CLEAR_TEXTURE),
|
|
|
|
limits,
|
|
|
|
},
|
|
|
|
None,
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.ok()?;
|
|
|
|
let device_handle = DeviceHandle {
|
|
|
|
adapter,
|
|
|
|
device,
|
|
|
|
queue,
|
|
|
|
};
|
|
|
|
self.devices.push(device_handle);
|
|
|
|
Some(self.devices.len() - 1)
|
2022-11-26 14:27:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Combination of surface and its configuration.
|
2023-03-05 11:33:30 +00:00
|
|
|
#[derive(Debug)]
|
2022-11-26 14:27:19 -05:00
|
|
|
pub struct RenderSurface {
|
|
|
|
pub surface: Surface,
|
|
|
|
pub config: SurfaceConfiguration,
|
2022-12-13 23:38:50 +00:00
|
|
|
pub dev_id: usize,
|
2023-03-05 11:33:30 +00:00
|
|
|
pub format: TextureFormat,
|
2022-11-26 14:27:19 -05:00
|
|
|
}
|
2023-01-24 20:25:44 -08:00
|
|
|
|
|
|
|
struct NullWake;
|
|
|
|
|
|
|
|
impl std::task::Wake for NullWake {
|
|
|
|
fn wake(self: std::sync::Arc<Self>) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Block on a future, polling the device as needed.
|
|
|
|
///
|
|
|
|
/// This will deadlock if the future is awaiting anything other than GPU progress.
|
|
|
|
pub fn block_on_wgpu<F: Future>(device: &Device, mut fut: F) -> F::Output {
|
|
|
|
let waker = std::task::Waker::from(std::sync::Arc::new(NullWake));
|
|
|
|
let mut context = std::task::Context::from_waker(&waker);
|
|
|
|
// Same logic as `pin_mut!` macro from `pin_utils`.
|
|
|
|
let mut fut = unsafe { std::pin::Pin::new_unchecked(&mut fut) };
|
|
|
|
loop {
|
|
|
|
match fut.as_mut().poll(&mut context) {
|
|
|
|
std::task::Poll::Pending => {
|
|
|
|
device.poll(wgpu::Maintain::Wait);
|
|
|
|
}
|
|
|
|
std::task::Poll::Ready(item) => break item,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|