Skip to content

Commit

Permalink
example-runner-wpgu: emulate push constants on wasm/WebGPU.
Browse files Browse the repository at this point in the history
  • Loading branch information
eddyb committed Apr 18, 2023
1 parent 1c05177 commit 93d3a57
Show file tree
Hide file tree
Showing 2 changed files with 111 additions and 13 deletions.
112 changes: 100 additions & 12 deletions examples/runners/wgpu/src/graphics.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ use crate::maybe_watch;

use super::Options;
use shared::ShaderConstants;
use std::slice;
use winit::{
event::{ElementState, Event, KeyboardInput, MouseButton, VirtualKeyCode, WindowEvent},
event_loop::{ControlFlow, EventLoop, EventLoopBuilder},
Expand Down Expand Up @@ -169,15 +170,60 @@ async fn run(
let mut surface_with_config = initial_surface
.map(|surface| auto_configure_surface(&adapter, &device, surface, window.inner_size()));

// Load the shaders from disk
// Describe the pipeline layout and build the initial pipeline.
let push_constants_or_rossbo_emulation = {
const PUSH_CONSTANTS_SIZE: usize = std::mem::size_of::<ShaderConstants>();
let stages = wgpu::ShaderStages::VERTEX | wgpu::ShaderStages::FRAGMENT;

if !options.emulate_push_constants_with_storage_buffer {
Ok(wgpu::PushConstantRange {
stages,
range: 0..PUSH_CONSTANTS_SIZE as u32,
})
} else {
let buffer = device.create_buffer(&wgpu::BufferDescriptor {
label: None,
size: PUSH_CONSTANTS_SIZE as u64,
usage: wgpu::BufferUsages::STORAGE | wgpu::BufferUsages::COPY_DST,
mapped_at_creation: false,
});
let binding0 = wgpu::BindGroupLayoutEntry {
binding: 0,
visibility: stages,
ty: wgpu::BindingType::Buffer {
ty: wgpu::BufferBindingType::Storage { read_only: true },
has_dynamic_offset: false,
min_binding_size: Some((PUSH_CONSTANTS_SIZE as u64).try_into().unwrap()),
},
count: None,
};
let bind_group_layout =
device.create_bind_group_layout(&wgpu::BindGroupLayoutDescriptor {
label: None,
entries: &[binding0],
});
let bind_group = device.create_bind_group(&wgpu::BindGroupDescriptor {
label: None,
layout: &bind_group_layout,
entries: &[wgpu::BindGroupEntry {
binding: 0,
resource: buffer.as_entire_binding(),
}],
});
Err((buffer, bind_group_layout, bind_group))
}
};
let pipeline_layout = device.create_pipeline_layout(&wgpu::PipelineLayoutDescriptor {
label: None,
bind_group_layouts: &[],
push_constant_ranges: &[wgpu::PushConstantRange {
stages: wgpu::ShaderStages::VERTEX | wgpu::ShaderStages::FRAGMENT,
range: 0..std::mem::size_of::<ShaderConstants>() as u32,
}],
bind_group_layouts: push_constants_or_rossbo_emulation
.as_ref()
.err()
.map(|(_, layout, _)| layout)
.as_ref()
.map_or(&[], slice::from_ref),
push_constant_ranges: push_constants_or_rossbo_emulation
.as_ref()
.map_or(&[], slice::from_ref),
});

let mut render_pipeline = create_pipeline(
Expand Down Expand Up @@ -310,11 +356,23 @@ async fn run(
};

rpass.set_pipeline(render_pipeline);
rpass.set_push_constants(
wgpu::ShaderStages::VERTEX | wgpu::ShaderStages::FRAGMENT,
0,
bytemuck::bytes_of(&push_constants),
);
let (push_constant_offset, push_constant_bytes) =
(0, bytemuck::bytes_of(&push_constants));
match &push_constants_or_rossbo_emulation {
Ok(_) => rpass.set_push_constants(
wgpu::ShaderStages::VERTEX | wgpu::ShaderStages::FRAGMENT,
push_constant_offset as u32,
push_constant_bytes,
),
Err((buffer, _, bind_group)) => {
queue.write_buffer(
buffer,
push_constant_offset,
push_constant_bytes,
);
rpass.set_bind_group(0, bind_group, &[]);
}
}
rpass.draw(0..3, 0..1);
}

Expand Down Expand Up @@ -390,8 +448,37 @@ fn create_pipeline(
device: &wgpu::Device,
pipeline_layout: &wgpu::PipelineLayout,
surface_format: wgpu::TextureFormat,
shader_binary: wgpu::ShaderModuleDescriptorSpirV<'_>,
mut shader_binary: wgpu::ShaderModuleDescriptorSpirV<'_>,
) -> wgpu::RenderPipeline {
if options.emulate_push_constants_with_storage_buffer {
let (ds, b) = (0, 0);

let w = shader_binary.source.to_mut();
assert_eq!((w[0], w[4]), (0x07230203, 0));
let mut last_op_decorate_start = None;
let mut i = 5;
while i < w.len() {
let (op, len) = (w[i] & 0xffff, w[i] >> 16);
match op {
71 => last_op_decorate_start = Some(i),
32 if w[i + 2] == 9 => w[i + 2] = 12,
59 if w[i + 3] == 9 => {
w[i + 3] = 12;
let id = w[i + 2];
let j = last_op_decorate_start.expect("no OpDecorate?");
w.splice(
j..j,
[0x4_0047, id, 34, ds, 0x4_0047, id, 33, b, 0x3_0047, id, 24],
);
i += 11;
}
54 => break,
_ => {}
}
i += len as usize;
}
}

// FIXME(eddyb) automate this decision by default.
let module = if options.force_spirv_passthru {
unsafe { device.create_shader_module_spirv(&shader_binary) }
Expand All @@ -402,6 +489,7 @@ fn create_pipeline(
source: wgpu::ShaderSource::SpirV(source),
})
};

device.create_render_pipeline(&wgpu::RenderPipelineDescriptor {
label: None,
layout: Some(pipeline_layout),
Expand Down
12 changes: 11 additions & 1 deletion examples/runners/wgpu/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -160,19 +160,29 @@ pub struct Options {

#[structopt(long)]
force_spirv_passthru: bool,

#[structopt(long)]
emulate_push_constants_with_storage_buffer: bool,
}

#[cfg_attr(target_os = "android", export_name = "android_main")]
pub fn main(
#[cfg(target_os = "android")] android_app: winit::platform::android::activity::AndroidApp,
) {
let options: Options = Options::from_args();
let mut options: Options = Options::from_args();

#[cfg(not(any(target_os = "android", target_arch = "wasm32")))]
if options.shader == RustGPUShader::Compute {
return compute::start(&options);
}

// HACK(eddyb) force push constant emulation using (read-only) SSBOs, on
// wasm->WebGPU, as push constants are currently not supported.
// FIXME(eddyb) could push constant support be automatically detected at runtime?
if cfg!(target_arch = "wasm32") {
options.emulate_push_constants_with_storage_buffer = true;
}

graphics::start(
#[cfg(target_os = "android")]
android_app,
Expand Down

0 comments on commit 93d3a57

Please sign in to comment.