Fixes to get example running in wasm

A number of things were wrong:

* The args were missing to `run`
* The robust memory changes introduced uniformity errors
* `clear_buffer` is a todo for wgpu on wasm
* Some more time calls crept in
* Initializing both env_logger and console_logger fails

In addition, we conditionally opt the shaders into
`workgroupUniformLoad`, as that's available on wasm but not yet native.

Some of the things (args, uniformity errors) are important fixes. Other
things (clear_buffer, wUL being optional) are workarounds for wgpu
limitations and have TODO items to be removed when wgpu catches up.
This commit is contained in:
Raph Levien 2023-01-26 12:19:12 -08:00
parent bf523e8845
commit d6cbae2a3f
6 changed files with 52 additions and 6 deletions

View file

@ -225,6 +225,9 @@ enum UserEvent {
fn main() { fn main() {
let args = Args::parse(); let args = Args::parse();
// TODO: initializing both env_logger and console_logger fails on wasm.
// Figure out a more principled approach.
#[cfg(not(target_arch = "wasm32"))]
env_logger::init(); env_logger::init();
#[cfg(not(target_arch = "wasm32"))] #[cfg(not(target_arch = "wasm32"))]
{ {
@ -261,6 +264,6 @@ fn main() {
.and_then(|doc| doc.body()) .and_then(|doc| doc.body())
.and_then(|body| body.append_child(&web_sys::Element::from(canvas)).ok()) .and_then(|body| body.append_child(&web_sys::Element::from(canvas)).ok())
.expect("couldn't append canvas to document body"); .expect("couldn't append canvas to document body");
wasm_bindgen_futures::spawn_local(run(event_loop, window)); wasm_bindgen_futures::spawn_local(run(event_loop, window, args));
} }
} }

View file

@ -82,9 +82,11 @@ pub fn render_svg_scene(
) { ) {
let scene_frag = scene.get_or_insert_with(|| { let scene_frag = scene.get_or_insert_with(|| {
use super::pico_svg::*; use super::pico_svg::*;
#[cfg(not(target_arch = "wasm32"))]
let start = Instant::now(); let start = Instant::now();
eprintln!("Starting to parse svg"); eprintln!("Starting to parse svg");
let svg = PicoSvg::load(svg, scale).unwrap(); let svg = PicoSvg::load(svg, scale).unwrap();
#[cfg(not(target_arch = "wasm32"))]
eprintln!("Parsing svg took {:?}", start.elapsed()); eprintln!("Parsing svg took {:?}", start.elapsed());
let mut new_scene = SceneFragment::new(); let mut new_scene = SceneFragment::new();
let mut builder = SceneBuilder::for_fragment(&mut new_scene); let mut builder = SceneBuilder::for_fragment(&mut new_scene);

View file

@ -148,7 +148,17 @@ fn main(
// Exit early if prior stages failed, as we can't run this stage. // Exit early if prior stages failed, as we can't run this stage.
// We need to check only prior stages, as if this stage has failed in another workgroup, // We need to check only prior stages, as if this stage has failed in another workgroup,
// we still want to know this workgroup's memory requirement. // we still want to know this workgroup's memory requirement.
if (atomicLoad(&bump.failed) & (STAGE_BINNING | STAGE_TILE_ALLOC | STAGE_PATH_COARSE)) != 0u { if local_id.x == 0u {
// Reuse sh_part_count to hold failed flag, shmem is tight
sh_part_count[0] = atomicLoad(&bump.failed);
}
#ifdef have_uniform
let failed = workgroupUniformLoad(&sh_part_count[0]);
#else
workgroupBarrier();
let failed = sh_part_count[0];
#endif
if (failed & (STAGE_BINNING | STAGE_TILE_ALLOC | STAGE_PATH_COARSE)) != 0u {
return; return;
} }
let width_in_bins = (config.width_in_tiles + N_TILE_X - 1u) / N_TILE_X; let width_in_bins = (config.width_in_tiles + N_TILE_X - 1u) / N_TILE_X;
@ -207,8 +217,12 @@ fn main(
workgroupBarrier(); workgroupBarrier();
} }
sh_part_count[local_id.x] = part_start_ix + count; sh_part_count[local_id.x] = part_start_ix + count;
#ifdef have_uniform
ready_ix = workgroupUniformLoad(&sh_part_count[WG_SIZE - 1u]);
#else
workgroupBarrier(); workgroupBarrier();
ready_ix = sh_part_count[WG_SIZE - 1u]; ready_ix = sh_part_count[WG_SIZE - 1u];
#endif
partition_ix += WG_SIZE; partition_ix += WG_SIZE;
} }
// use binary search to find draw object to read // use binary search to find draw object to read

View file

@ -29,6 +29,7 @@ let WG_SIZE = 256u;
var<workgroup> sh_tile_count: array<u32, WG_SIZE>; var<workgroup> sh_tile_count: array<u32, WG_SIZE>;
var<workgroup> sh_tile_offset: u32; var<workgroup> sh_tile_offset: u32;
var<workgroup> sh_atomic_failed: u32;
@compute @workgroup_size(256) @compute @workgroup_size(256)
fn main( fn main(
@ -38,7 +39,16 @@ fn main(
// Exit early if prior stages failed, as we can't run this stage. // Exit early if prior stages failed, as we can't run this stage.
// We need to check only prior stages, as if this stage has failed in another workgroup, // We need to check only prior stages, as if this stage has failed in another workgroup,
// we still want to know this workgroup's memory requirement. // we still want to know this workgroup's memory requirement.
if (atomicLoad(&bump.failed) & STAGE_BINNING) != 0u { if local_id.x == 0u {
sh_atomic_failed = atomicLoad(&bump.failed);
}
#ifdef have_uniform
let failed = workgroupUniformLoad(&sh_atomic_failed);
#else
workgroupBarrier();
let failed = sh_atomic_failed;
#endif
if (failed & STAGE_BINNING) != 0u {
return; return;
} }
// scale factors useful for converting coordinates to tiles // scale factors useful for converting coordinates to tiles

View file

@ -352,7 +352,19 @@ impl Engine {
} }
Command::Clear(proxy, offset, size) => { Command::Clear(proxy, offset, size) => {
let buffer = bind_map.get_or_create(*proxy, device, &mut self.pool)?; let buffer = bind_map.get_or_create(*proxy, device, &mut self.pool)?;
#[cfg(not(target_arch = "wasm32"))]
encoder.clear_buffer(buffer, *offset, *size); encoder.clear_buffer(buffer, *offset, *size);
#[cfg(target_arch = "wasm32")]
{
// TODO: remove this workaround when wgpu implements clear_buffer
// Also note: semantics are wrong, it's queue order rather than encoder.
let size = match size {
Some(size) => size.get(),
None => proxy.size,
};
let zeros = vec![0; size as usize];
queue.write_buffer(buffer, *offset, &zeros);
}
} }
} }
} }

View file

@ -160,6 +160,11 @@ pub fn full_shaders(device: &Device, engine: &mut Engine) -> Result<FullShaders,
let mut small_config = HashSet::new(); let mut small_config = HashSet::new();
small_config.insert("full".into()); small_config.insert("full".into());
small_config.insert("small".into()); small_config.insert("small".into());
// TODO: remove this workaround when workgroupUniformLoad lands in naga
#[allow(unused_mut)]
let mut uniform = HashSet::new();
#[cfg(target_arch = "wasm32")]
uniform.insert("have_uniform".into());
let pathtag_reduce = engine.add_shader( let pathtag_reduce = engine.add_shader(
device, device,
"pathtag_reduce", "pathtag_reduce",
@ -286,7 +291,7 @@ pub fn full_shaders(device: &Device, engine: &mut Engine) -> Result<FullShaders,
let tile_alloc = engine.add_shader( let tile_alloc = engine.add_shader(
device, device,
"tile_alloc", "tile_alloc",
preprocess::preprocess(shader!("tile_alloc"), &empty, &imports).into(), preprocess::preprocess(shader!("tile_alloc"), &uniform, &imports).into(),
&[ &[
BindType::Uniform, BindType::Uniform,
BindType::BufReadOnly, BindType::BufReadOnly,
@ -321,7 +326,7 @@ pub fn full_shaders(device: &Device, engine: &mut Engine) -> Result<FullShaders,
let coarse = engine.add_shader( let coarse = engine.add_shader(
device, device,
"coarse", "coarse",
preprocess::preprocess(shader!("coarse"), &empty, &imports).into(), preprocess::preprocess(shader!("coarse"), &uniform, &imports).into(),
&[ &[
BindType::Uniform, BindType::Uniform,
BindType::BufReadOnly, BindType::BufReadOnly,