Fix invalid buffer access errors caught by shader validation

Fixed several other shader validation errors caught when running
vello_shaders natively on Metal.

These were primarily caused by reading an invalid drawtag while
accessing the scene buffer. Scene buffer access in the offending
pipelines now initialize the draw tag to DRAWTAG_NOP if an
invocation ID would land beyond the valid index range of encoded draw
objects.
This commit is contained in:
Arman Uguray 2023-06-27 23:21:05 -07:00
parent a016fc19de
commit 1dea6c0ef0
6 changed files with 35 additions and 3 deletions

View file

@ -12,6 +12,7 @@
"segment": "${workspaceFolder}/shader/shared/segment.wgsl", "segment": "${workspaceFolder}/shader/shared/segment.wgsl",
"tile": "${workspaceFolder}/shader/shared/tile.wgsl", "tile": "${workspaceFolder}/shader/shared/tile.wgsl",
"transform": "${workspaceFolder}/shader/shared/transform.wgsl" "transform": "${workspaceFolder}/shader/shared/transform.wgsl"
"util": "${workspaceFolder}/shader/shared/util.wgsl",
}, },
"wgsl-analyzer.diagnostics.nagaVersion": "main", "wgsl-analyzer.diagnostics.nagaVersion": "main",
"wgsl-analyzer.preprocessor.shaderDefs": [ "wgsl-analyzer.preprocessor.shaderDefs": [

View file

@ -74,7 +74,10 @@ fn main(
let draw_monoid = draw_monoids[element_ix]; let draw_monoid = draw_monoids[element_ix];
var clip_bbox = vec4(-1e9, -1e9, 1e9, 1e9); var clip_bbox = vec4(-1e9, -1e9, 1e9, 1e9);
if draw_monoid.clip_ix > 0u { if draw_monoid.clip_ix > 0u {
clip_bbox = clip_bbox_buf[draw_monoid.clip_ix - 1u]; // TODO: `clip_ix` should always be valid as long as the monoids are correct. Leaving
// the bounds check in here for correctness but we should assert this condition instead
// once there is a debug-assertion mechanism.
clip_bbox = clip_bbox_buf[min(draw_monoid.clip_ix - 1u, config.n_clip - 1u)];
} }
// For clip elements, clip_box is the bbox of the clip path, // For clip elements, clip_box is the bbox of the clip path,
// intersected with enclosing clips. // intersected with enclosing clips.

View file

@ -29,6 +29,8 @@ var<storage, read_write> info: array<u32>;
@group(0) @binding(6) @group(0) @binding(6)
var<storage, read_write> clip_inp: array<ClipInp>; var<storage, read_write> clip_inp: array<ClipInp>;
#import util
let WG_SIZE = 256u; let WG_SIZE = 256u;
fn read_transform(transform_base: u32, ix: u32) -> Transform { fn read_transform(transform_base: u32, ix: u32) -> Transform {
@ -73,7 +75,7 @@ fn main(
workgroupBarrier(); workgroupBarrier();
var m = sh_scratch[0]; var m = sh_scratch[0];
workgroupBarrier(); workgroupBarrier();
let tag_word = scene[config.drawtag_base + ix]; let tag_word = read_draw_tag_from_scene(ix);
agg = map_draw_tag(tag_word); agg = map_draw_tag(tag_word);
sh_scratch[local_id.x] = agg; sh_scratch[local_id.x] = agg;
for (var i = 0u; i < firstTrailingBit(WG_SIZE); i += 1u) { for (var i = 0u; i < firstTrailingBit(WG_SIZE); i += 1u) {

View file

@ -16,13 +16,15 @@ let WG_SIZE = 256u;
var<workgroup> sh_scratch: array<DrawMonoid, WG_SIZE>; var<workgroup> sh_scratch: array<DrawMonoid, WG_SIZE>;
#import util
@compute @workgroup_size(256) @compute @workgroup_size(256)
fn main( fn main(
@builtin(global_invocation_id) global_id: vec3<u32>, @builtin(global_invocation_id) global_id: vec3<u32>,
@builtin(local_invocation_id) local_id: vec3<u32>, @builtin(local_invocation_id) local_id: vec3<u32>,
) { ) {
let ix = global_id.x; let ix = global_id.x;
let tag_word = scene[config.drawtag_base + ix]; let tag_word = read_draw_tag_from_scene(ix);
var agg = map_draw_tag(tag_word); var agg = map_draw_tag(tag_word);
sh_scratch[local_id.x] = agg; sh_scratch[local_id.x] = agg;
for (var i = 0u; i < firstTrailingBit(WG_SIZE); i += 1u) { for (var i = 0u; i < firstTrailingBit(WG_SIZE); i += 1u) {

23
shader/shared/util.wgsl Normal file
View file

@ -0,0 +1,23 @@
// SPDX-License-Identifier: Apache-2.0 OR MIT OR Unlicense
// This file defines utility functions that interact with host-shareable buffer objects. It should
// be imported once following the resource binding declarations in the shader module that access
// them.
// Reads a draw tag from the scene buffer, defaulting to DRAWTAG_NOP if the given `ix` is beyond the
// range of valid draw objects (e.g this can happen if `ix` is derived from an invocation ID in a
// workgroup that partially spans valid range).
//
// This function depends on the following global declarations:
// * `scene`: array<u32>
// * `config`: Config (see config.wgsl)
fn read_draw_tag_from_scene(ix: u32) -> u32 {
let tag_ix = config.drawtag_base + ix;
var tag_word: u32;
if tag_ix < config.drawtag_base + config.n_drawobj {
tag_word = scene[tag_ix];
} else {
tag_word = DRAWTAG_NOP;
}
return tag_word;
}

View file

@ -319,4 +319,5 @@ const SHARED_SHADERS: &[(&str, &str)] = &[
shared_shader!("segment"), shared_shader!("segment"),
shared_shader!("tile"), shared_shader!("tile"),
shared_shader!("transform"), shared_shader!("transform"),
shared_shader!("util"),
]; ];