librashader_test/render/
wgpu.rs

1use crate::render::{CommonFrameOptions, RenderTest};
2use anyhow::anyhow;
3use image::RgbaImage;
4use librashader::runtime::wgpu::*;
5use librashader::runtime::{Size, Viewport};
6use librashader_runtime::image::{Image, UVDirection};
7use std::io::{Cursor, Write};
8use std::ops::DerefMut;
9use std::path::Path;
10use std::sync::Arc;
11use wgpu::{Adapter, Device, Instance, Queue, TexelCopyBufferInfo, Texture};
12use wgpu_types::{
13    BufferAddress, BufferDescriptor, BufferUsages, CommandEncoderDescriptor, TexelCopyBufferLayout,
14    TextureDescriptor, TextureDimension, TextureFormat, TextureUsages,
15};
16
17use librashader::presets::ShaderPreset;
18use librashader::runtime::{FilterChainParameters, RuntimeParameters};
19use parking_lot::Mutex;
20
21pub struct Wgpu {
22    _instance: Instance,
23    _adapter: Adapter,
24    device: Arc<Device>,
25    queue: Arc<Queue>,
26    image: Image,
27    texture: Arc<Texture>,
28}
29
30struct BufferDimensions {
31    height: usize,
32    unpadded_bytes_per_row: usize,
33    padded_bytes_per_row: usize,
34}
35
36impl BufferDimensions {
37    fn new(width: usize, height: usize) -> Self {
38        let bytes_per_pixel = std::mem::size_of::<u32>();
39        let unpadded_bytes_per_row = width * bytes_per_pixel;
40        let align = wgpu::COPY_BYTES_PER_ROW_ALIGNMENT as usize;
41        let padded_bytes_per_row_padding = (align - unpadded_bytes_per_row % align) % align;
42        let padded_bytes_per_row = unpadded_bytes_per_row + padded_bytes_per_row_padding;
43        Self {
44            height,
45            unpadded_bytes_per_row,
46            padded_bytes_per_row,
47        }
48    }
49}
50
51impl RenderTest for Wgpu {
52    fn new(path: &Path) -> anyhow::Result<Self>
53    where
54        Self: Sized,
55    {
56        Wgpu::new(path)
57    }
58
59    fn image_size(&self) -> Size<u32> {
60        self.image.size
61    }
62
63    fn render_with_preset_and_params(
64        &mut self,
65        preset: ShaderPreset,
66        frame_count: usize,
67        output_size: Option<Size<u32>>,
68        param_setter: Option<&dyn Fn(&RuntimeParameters)>,
69        frame_options: Option<CommonFrameOptions>,
70    ) -> anyhow::Result<image::RgbaImage> {
71        let mut chain = FilterChain::load_from_preset(
72            preset,
73            &self.device,
74            &self.queue,
75            Some(&FilterChainOptions {
76                force_no_mipmaps: false,
77                enable_cache: true,
78                adapter_info: None,
79            }),
80        )?;
81        if let Some(setter) = param_setter {
82            setter(&chain.parameters());
83        }
84        let mut cmd = self
85            .device
86            .create_command_encoder(&CommandEncoderDescriptor { label: None });
87
88        let output_tex = self.device.create_texture(&TextureDescriptor {
89            label: None,
90            size: output_size.map_or(self.texture.size(), |size| size.into()),
91            mip_level_count: 1,
92            sample_count: 1,
93            dimension: TextureDimension::D2,
94            format: TextureFormat::Rgba8Unorm,
95            usage: TextureUsages::RENDER_ATTACHMENT | TextureUsages::COPY_SRC,
96            view_formats: &[wgpu::TextureFormat::Rgba8Unorm],
97        });
98
99        let buffer_dimensions =
100            BufferDimensions::new(output_tex.width() as usize, output_tex.height() as usize);
101        let output_buf = Arc::new(self.device.create_buffer(&BufferDescriptor {
102            label: None,
103            size: (buffer_dimensions.padded_bytes_per_row * buffer_dimensions.height)
104                as BufferAddress, // 4bpp
105            usage: BufferUsages::COPY_DST | BufferUsages::MAP_READ,
106            mapped_at_creation: false,
107        }));
108
109        let view = output_tex.create_view(&wgpu::TextureViewDescriptor::default());
110        let output = WgpuOutputView::new_from_raw(
111            &view,
112            output_tex.size().into(),
113            TextureFormat::Rgba8Unorm,
114        );
115
116        let viewport = Viewport::new_render_target_sized_origin(output, None)?;
117        let options = frame_options.map(|options| FrameOptions {
118            clear_history: options.clear_history,
119            frame_direction: options.frame_direction,
120            rotation: options.rotation,
121            total_subframes: options.total_subframes,
122            current_subframe: options.current_subframe,
123            aspect_ratio: options.aspect_ratio,
124            frametime_delta: options.frametime_delta,
125            frames_per_second: options.frames_per_second,
126        });
127
128        for frame in 0..=frame_count {
129            chain.frame(&self.texture, &viewport, &mut cmd, frame, options.as_ref())?;
130        }
131
132        cmd.copy_texture_to_buffer(
133            output_tex.as_image_copy(),
134            TexelCopyBufferInfo {
135                buffer: &output_buf,
136                layout: TexelCopyBufferLayout {
137                    offset: 0,
138                    bytes_per_row: Some(buffer_dimensions.padded_bytes_per_row as u32),
139                    rows_per_image: None,
140                },
141            },
142            output_tex.size(),
143        );
144
145        let si = self.queue.submit([cmd.finish()]);
146        self.device.poll(wgpu::PollType::Wait {
147            submission_index: Some(si),
148            timeout: None,
149        })?;
150
151        let capturable = Arc::clone(&output_buf);
152
153        let pixels = Arc::new(Mutex::new(Vec::new()));
154
155        let pixels_async = Arc::clone(&pixels);
156        output_buf
157            .slice(..)
158            .map_async(wgpu::MapMode::Read, move |r| {
159                if r.is_ok() {
160                    let buffer = capturable.slice(..).get_mapped_range();
161                    let mut pixels = pixels_async.lock();
162                    pixels.resize(buffer.len(), 0);
163
164                    let mut cursor = Cursor::new(pixels.deref_mut());
165                    for chunk in buffer.chunks(buffer_dimensions.padded_bytes_per_row) {
166                        cursor
167                            .write_all(&chunk[..buffer_dimensions.unpadded_bytes_per_row])
168                            .unwrap()
169                    }
170
171                    cursor.into_inner();
172                }
173                capturable.unmap();
174            });
175
176        self.device.poll(wgpu::PollType::Wait {
177            submission_index: None,
178            timeout: None,
179        })?;
180
181        if pixels.lock().len() == 0 {
182            return Err(anyhow!("failed to copy pixels from buffer"));
183        }
184
185        let image = RgbaImage::from_raw(
186            output_tex.width(),
187            output_tex.height(),
188            pixels.lock().to_vec(),
189        )
190        .ok_or(anyhow!("Unable to create image from data"))?;
191
192        Ok(image)
193    }
194}
195
196impl Wgpu {
197    pub fn new(image: &Path) -> anyhow::Result<Self> {
198        pollster::block_on(async {
199            let instance = wgpu::Instance::default();
200            let adapter = instance
201                .request_adapter(&wgpu::RequestAdapterOptions {
202                    power_preference: wgpu::PowerPreference::default(),
203                    compatible_surface: None,
204                    force_fallback_adapter: false,
205                })
206                .await?;
207
208            let (device, queue) = adapter
209                .request_device(&wgpu::DeviceDescriptor {
210                    required_features: wgpu::Features::ADDRESS_MODE_CLAMP_TO_BORDER
211                        | wgpu::Features::PIPELINE_CACHE
212                        | wgpu::Features::TEXTURE_ADAPTER_SPECIFIC_FORMAT_FEATURES
213                        | wgpu::Features::FLOAT32_FILTERABLE,
214                    required_limits: wgpu::Limits::default(),
215                    label: None,
216                    memory_hints: Default::default(),
217                    ..Default::default()
218                })
219                .await?;
220            let (image, texture) = Self::load_image(&device, &queue, image)?;
221
222            Ok(Self {
223                _instance: instance,
224                _adapter: adapter,
225                device: Arc::new(device),
226                queue: Arc::new(queue),
227                image,
228                texture: Arc::new(texture),
229            })
230        })
231    }
232
233    fn load_image(device: &Device, queue: &Queue, path: &Path) -> anyhow::Result<(Image, Texture)> {
234        let image = Image::load(path, UVDirection::TopLeft)?;
235        let texture = device.create_texture(&TextureDescriptor {
236            size: image.size.into(),
237            mip_level_count: 1,
238            sample_count: 1,
239            dimension: TextureDimension::D2,
240            usage: TextureUsages::TEXTURE_BINDING | TextureUsages::COPY_DST,
241            format: wgpu::TextureFormat::Rgba8Unorm,
242            view_formats: &[wgpu::TextureFormat::Rgba8Unorm],
243            label: None,
244        });
245
246        queue.write_texture(
247            wgpu::TexelCopyTextureInfo {
248                texture: &texture,
249                mip_level: 0,
250                origin: wgpu::Origin3d::ZERO,
251                aspect: wgpu::TextureAspect::All,
252            },
253            &image.bytes,
254            wgpu::TexelCopyBufferLayout {
255                offset: 0,
256                bytes_per_row: Some(4 * image.size.width),
257                rows_per_image: None,
258            },
259            image.size.into(),
260        );
261
262        let si = queue.submit([]);
263
264        device.poll(wgpu::PollType::Wait {
265            submission_index: Some(si),
266            timeout: None,
267        })?;
268
269        Ok((image, texture))
270    }
271}