1#![allow(unused_variables)]
2
3use alloc::{string::String, sync::Arc, vec, vec::Vec};
4use core::{ptr, sync::atomic::Ordering, time::Duration};
5
6#[cfg(supports_64bit_atomics)]
7use core::sync::atomic::AtomicU64;
8#[cfg(not(supports_64bit_atomics))]
9use portable_atomic::AtomicU64;
10
11use crate::TlasInstance;
12
13mod buffer;
14pub use buffer::Buffer;
15mod command;
16pub use command::CommandBuffer;
17
18#[derive(Clone, Debug)]
19pub struct Api;
20
21pub struct Context {
22 options: Arc<wgt::NoopBackendOptions>,
23}
24#[derive(Debug)]
25pub struct Encoder;
26#[derive(Debug)]
27pub struct Resource;
28
29#[derive(Debug)]
30pub struct Fence {
31 value: AtomicU64,
32}
33
34type DeviceResult<T> = Result<T, crate::DeviceError>;
35
36impl crate::Api for Api {
37 const VARIANT: wgt::Backend = wgt::Backend::Noop;
38
39 type Instance = Context;
40 type Surface = Context;
41 type Adapter = Context;
42 type Device = Context;
43
44 type Queue = Context;
45 type CommandEncoder = CommandBuffer;
46 type CommandBuffer = CommandBuffer;
47
48 type Buffer = Buffer;
49 type Texture = Resource;
50 type SurfaceTexture = Resource;
51 type TextureView = Resource;
52 type Sampler = Resource;
53 type QuerySet = Resource;
54 type Fence = Fence;
55 type AccelerationStructure = Resource;
56 type PipelineCache = Resource;
57
58 type BindGroupLayout = Resource;
59 type BindGroup = Resource;
60 type PipelineLayout = Resource;
61 type ShaderModule = Resource;
62 type RenderPipeline = Resource;
63 type ComputePipeline = Resource;
64}
65
66crate::impl_dyn_resource!(Buffer, CommandBuffer, Context, Fence, Resource);
67
68impl crate::DynAccelerationStructure for Resource {}
69impl crate::DynBindGroup for Resource {}
70impl crate::DynBindGroupLayout for Resource {}
71impl crate::DynBuffer for Buffer {}
72impl crate::DynCommandBuffer for CommandBuffer {}
73impl crate::DynComputePipeline for Resource {}
74impl crate::DynFence for Fence {}
75impl crate::DynPipelineCache for Resource {}
76impl crate::DynPipelineLayout for Resource {}
77impl crate::DynQuerySet for Resource {}
78impl crate::DynRenderPipeline for Resource {}
79impl crate::DynSampler for Resource {}
80impl crate::DynShaderModule for Resource {}
81impl crate::DynSurfaceTexture for Resource {}
82impl crate::DynTexture for Resource {}
83impl crate::DynTextureView for Resource {}
84
85impl core::borrow::Borrow<dyn crate::DynTexture> for Resource {
86 fn borrow(&self) -> &dyn crate::DynTexture {
87 self
88 }
89}
90
91impl crate::Instance for Context {
92 type A = Api;
93
94 unsafe fn init(desc: &crate::InstanceDescriptor<'_>) -> Result<Self, crate::InstanceError> {
95 let options = Arc::new(desc.backend_options.noop.clone());
96 if options.enable {
97 Ok(Context { options })
98 } else {
99 Err(crate::InstanceError::new(String::from(
100 "noop backend disabled because NoopBackendOptions::enable is false",
101 )))
102 }
103 }
104 unsafe fn create_surface(
105 &self,
106 _display_handle: raw_window_handle::RawDisplayHandle,
107 _window_handle: raw_window_handle::RawWindowHandle,
108 ) -> Result<Context, crate::InstanceError> {
109 Ok(Context {
110 options: Arc::clone(&self.options),
111 })
112 }
113 unsafe fn enumerate_adapters(
114 &self,
115 _surface_hint: Option<&Context>,
116 ) -> Vec<crate::ExposedAdapter<Api>> {
117 let device_type = self.options.device_type.unwrap_or(wgt::DeviceType::Other);
118 let subgroup_min_size = self
119 .options
120 .subgroup_min_size
121 .unwrap_or(wgt::MINIMUM_SUBGROUP_MIN_SIZE);
122 let subgroup_max_size = self
123 .options
124 .subgroup_max_size
125 .unwrap_or(wgt::MAXIMUM_SUBGROUP_MAX_SIZE);
126 let features = self.options.features.unwrap_or(wgt::Features::all());
127 let limits = self
128 .options
129 .limits
130 .clone()
131 .unwrap_or(CAPABILITIES.limits.clone());
132
133 let info = wgt::AdapterInfo {
134 subgroup_min_size,
135 subgroup_max_size,
136 ..adapter_info()
137 };
138
139 let capabilities = crate::Capabilities {
140 limits,
141 ..CAPABILITIES
142 };
143
144 vec![crate::ExposedAdapter {
145 adapter: Context {
146 options: Arc::clone(&self.options),
147 },
148 info,
149 features,
150 capabilities,
151 }]
152 }
153}
154
155pub fn adapter_info() -> wgt::AdapterInfo {
160 wgt::AdapterInfo {
161 name: String::from("noop wgpu backend"),
162 driver: String::from("wgpu"),
163 ..wgt::AdapterInfo::new(wgt::DeviceType::Cpu, wgt::Backend::Noop)
164 }
165}
166
167pub const CAPABILITIES: crate::Capabilities = {
172 crate::Capabilities {
173 limits: wgt::Limits::unlimited(),
174 alignments: crate::Alignments {
175 buffer_copy_offset: wgt::BufferSize::MIN,
177 buffer_copy_pitch: wgt::BufferSize::MIN,
178 uniform_bounds_check_alignment: wgt::BufferSize::MIN,
179 raw_tlas_instance_size: 0,
180 ray_tracing_scratch_buffer_alignment: 1,
181 },
182 downlevel: wgt::DownlevelCapabilities {
183 flags: wgt::DownlevelFlags::all(),
184 limits: wgt::DownlevelLimits {},
185 shader_model: wgt::ShaderModel::Sm5,
186 },
187 cooperative_matrix_properties: Vec::new(),
188 }
189};
190
191impl crate::Surface for Context {
192 type A = Api;
193
194 unsafe fn configure(
195 &self,
196 device: &Context,
197 config: &crate::SurfaceConfiguration,
198 ) -> Result<(), crate::SurfaceError> {
199 Ok(())
200 }
201
202 unsafe fn unconfigure(&self, device: &Context) {}
203
204 unsafe fn acquire_texture(
205 &self,
206 _timeout: Option<Duration>,
207 _fence: &Fence,
208 ) -> Result<crate::AcquiredSurfaceTexture<Api>, crate::SurfaceError> {
209 Err(crate::SurfaceError::Timeout)
210 }
211 unsafe fn discard_texture(&self, texture: Resource) {}
212}
213
214impl crate::Adapter for Context {
215 type A = Api;
216
217 unsafe fn open(
218 &self,
219 features: wgt::Features,
220 _limits: &wgt::Limits,
221 _memory_hints: &wgt::MemoryHints,
222 ) -> DeviceResult<crate::OpenDevice<Api>> {
223 Ok(crate::OpenDevice {
224 device: Context {
225 options: Arc::clone(&self.options),
226 },
227 queue: Context {
228 options: Arc::clone(&self.options),
229 },
230 })
231 }
232 unsafe fn texture_format_capabilities(
233 &self,
234 format: wgt::TextureFormat,
235 ) -> crate::TextureFormatCapabilities {
236 crate::TextureFormatCapabilities::empty()
237 }
238
239 unsafe fn surface_capabilities(&self, surface: &Context) -> Option<crate::SurfaceCapabilities> {
240 None
241 }
242
243 unsafe fn get_presentation_timestamp(&self) -> wgt::PresentationTimestamp {
244 wgt::PresentationTimestamp::INVALID_TIMESTAMP
245 }
246
247 fn get_ordered_buffer_usages(&self) -> wgt::BufferUses {
248 wgt::BufferUses::INCLUSIVE | wgt::BufferUses::MAP_WRITE
249 }
250
251 fn get_ordered_texture_usages(&self) -> wgt::TextureUses {
252 wgt::TextureUses::INCLUSIVE
253 | wgt::TextureUses::COLOR_TARGET
254 | wgt::TextureUses::DEPTH_STENCIL_WRITE
255 }
256}
257
258impl crate::Queue for Context {
259 type A = Api;
260
261 unsafe fn submit(
262 &self,
263 command_buffers: &[&CommandBuffer],
264 surface_textures: &[&Resource],
265 (fence, fence_value): (&mut Fence, crate::FenceValue),
266 ) -> DeviceResult<()> {
267 for cb in command_buffers {
269 unsafe {
272 cb.execute();
273 }
274 }
275 fence.value.store(fence_value, Ordering::Release);
276 Ok(())
277 }
278 unsafe fn present(
279 &self,
280 surface: &Context,
281 texture: Resource,
282 ) -> Result<(), crate::SurfaceError> {
283 Ok(())
284 }
285
286 unsafe fn get_timestamp_period(&self) -> f32 {
287 1.0
288 }
289
290 unsafe fn wait_for_idle(&self) -> Result<(), crate::DeviceError> {
291 Ok(())
292 }
293}
294
295impl crate::Device for Context {
296 type A = Api;
297
298 unsafe fn create_buffer(&self, desc: &crate::BufferDescriptor) -> DeviceResult<Buffer> {
299 Buffer::new(desc)
300 }
301
302 unsafe fn destroy_buffer(&self, buffer: Buffer) {}
303 unsafe fn add_raw_buffer(&self, _buffer: &Buffer) {}
304
305 unsafe fn map_buffer(
306 &self,
307 buffer: &Buffer,
308 range: crate::MemoryRange,
309 ) -> DeviceResult<crate::BufferMapping> {
310 Ok(crate::BufferMapping {
314 ptr: ptr::NonNull::new(buffer.get_slice_ptr(range).cast::<u8>()).unwrap(),
315 is_coherent: true,
316 })
317 }
318 unsafe fn unmap_buffer(&self, buffer: &Buffer) {}
319 unsafe fn flush_mapped_ranges<I>(&self, buffer: &Buffer, ranges: I) {}
320 unsafe fn invalidate_mapped_ranges<I>(&self, buffer: &Buffer, ranges: I) {}
321
322 unsafe fn create_texture(&self, desc: &crate::TextureDescriptor) -> DeviceResult<Resource> {
323 Ok(Resource)
324 }
325 unsafe fn destroy_texture(&self, texture: Resource) {}
326 unsafe fn add_raw_texture(&self, _texture: &Resource) {}
327
328 unsafe fn create_texture_view(
329 &self,
330 texture: &Resource,
331 desc: &crate::TextureViewDescriptor,
332 ) -> DeviceResult<Resource> {
333 Ok(Resource)
334 }
335 unsafe fn destroy_texture_view(&self, view: Resource) {}
336 unsafe fn create_sampler(&self, desc: &crate::SamplerDescriptor) -> DeviceResult<Resource> {
337 Ok(Resource)
338 }
339 unsafe fn destroy_sampler(&self, sampler: Resource) {}
340
341 unsafe fn create_command_encoder(
342 &self,
343 desc: &crate::CommandEncoderDescriptor<Context>,
344 ) -> DeviceResult<CommandBuffer> {
345 Ok(CommandBuffer::new())
346 }
347
348 unsafe fn create_bind_group_layout(
349 &self,
350 desc: &crate::BindGroupLayoutDescriptor,
351 ) -> DeviceResult<Resource> {
352 Ok(Resource)
353 }
354 unsafe fn destroy_bind_group_layout(&self, bg_layout: Resource) {}
355 unsafe fn create_pipeline_layout(
356 &self,
357 desc: &crate::PipelineLayoutDescriptor<Resource>,
358 ) -> DeviceResult<Resource> {
359 Ok(Resource)
360 }
361 unsafe fn destroy_pipeline_layout(&self, pipeline_layout: Resource) {}
362 unsafe fn create_bind_group(
363 &self,
364 desc: &crate::BindGroupDescriptor<Resource, Buffer, Resource, Resource, Resource>,
365 ) -> DeviceResult<Resource> {
366 Ok(Resource)
367 }
368 unsafe fn destroy_bind_group(&self, group: Resource) {}
369
370 unsafe fn create_shader_module(
371 &self,
372 desc: &crate::ShaderModuleDescriptor,
373 shader: crate::ShaderInput,
374 ) -> Result<Resource, crate::ShaderError> {
375 Ok(Resource)
376 }
377 unsafe fn destroy_shader_module(&self, module: Resource) {}
378 unsafe fn create_render_pipeline(
379 &self,
380 desc: &crate::RenderPipelineDescriptor<Resource, Resource, Resource>,
381 ) -> Result<Resource, crate::PipelineError> {
382 Ok(Resource)
383 }
384 unsafe fn destroy_render_pipeline(&self, pipeline: Resource) {}
385 unsafe fn create_compute_pipeline(
386 &self,
387 desc: &crate::ComputePipelineDescriptor<Resource, Resource, Resource>,
388 ) -> Result<Resource, crate::PipelineError> {
389 Ok(Resource)
390 }
391 unsafe fn destroy_compute_pipeline(&self, pipeline: Resource) {}
392 unsafe fn create_pipeline_cache(
393 &self,
394 desc: &crate::PipelineCacheDescriptor<'_>,
395 ) -> Result<Resource, crate::PipelineCacheError> {
396 Ok(Resource)
397 }
398 unsafe fn destroy_pipeline_cache(&self, cache: Resource) {}
399
400 unsafe fn create_query_set(
401 &self,
402 desc: &wgt::QuerySetDescriptor<crate::Label>,
403 ) -> DeviceResult<Resource> {
404 Ok(Resource)
405 }
406 unsafe fn destroy_query_set(&self, set: Resource) {}
407 unsafe fn create_fence(&self) -> DeviceResult<Fence> {
408 Ok(Fence {
409 value: AtomicU64::new(0),
410 })
411 }
412 unsafe fn destroy_fence(&self, fence: Fence) {}
413 unsafe fn get_fence_value(&self, fence: &Fence) -> DeviceResult<crate::FenceValue> {
414 Ok(fence.value.load(Ordering::Acquire))
415 }
416 unsafe fn wait(
417 &self,
418 fence: &Fence,
419 value: crate::FenceValue,
420 timeout: Option<Duration>,
421 ) -> DeviceResult<bool> {
422 assert!(
426 fence.value.load(Ordering::Acquire) >= value,
427 "submission must have already been done"
428 );
429 Ok(true)
430 }
431
432 unsafe fn start_graphics_debugger_capture(&self) -> bool {
433 false
434 }
435 unsafe fn stop_graphics_debugger_capture(&self) {}
436 unsafe fn create_acceleration_structure(
437 &self,
438 desc: &crate::AccelerationStructureDescriptor,
439 ) -> DeviceResult<Resource> {
440 Ok(Resource)
441 }
442 unsafe fn get_acceleration_structure_build_sizes<'a>(
443 &self,
444 _desc: &crate::GetAccelerationStructureBuildSizesDescriptor<'a, Buffer>,
445 ) -> crate::AccelerationStructureBuildSizes {
446 Default::default()
447 }
448 unsafe fn get_acceleration_structure_device_address(
449 &self,
450 _acceleration_structure: &Resource,
451 ) -> wgt::BufferAddress {
452 Default::default()
453 }
454 unsafe fn destroy_acceleration_structure(&self, _acceleration_structure: Resource) {}
455
456 fn tlas_instance_to_bytes(&self, instance: TlasInstance) -> Vec<u8> {
457 vec![]
458 }
459
460 fn get_internal_counters(&self) -> wgt::HalCounters {
461 Default::default()
462 }
463
464 fn check_if_oom(&self) -> DeviceResult<()> {
465 Ok(())
466 }
467}