From 97cc6e586b5bb045be8d57085fc7a596ce362806 Mon Sep 17 00:00:00 2001 From: Brandon Jones Date: Wed, 18 Dec 2024 11:01:54 -0800 Subject: [PATCH] Loosen Viewport validation tests to match spec update --- .../cmds/render/dynamic_state.spec.ts | 113 +++++++++++------- 1 file changed, 69 insertions(+), 44 deletions(-) diff --git a/src/webgpu/api/validation/encoding/cmds/render/dynamic_state.spec.ts b/src/webgpu/api/validation/encoding/cmds/render/dynamic_state.spec.ts index 935d94077296..e9b3365995ba 100644 --- a/src/webgpu/api/validation/encoding/cmds/render/dynamic_state.spec.ts +++ b/src/webgpu/api/validation/encoding/cmds/render/dynamic_state.spec.ts @@ -130,71 +130,96 @@ class F extends ValidationTest { export const g = makeTestGroup(F); -g.test('setViewport,x_y_width_height_nonnegative') +g.test('setViewport,width_height_nonnegative') .desc( - `Test that the parameters of setViewport to define the box must be non-negative. + `Test that the width and height parameters of setViewport must be non-negative. TODO Test -0 (it should be valid) but can't be tested because the harness complains about duplicate parameters. TODO Test the first value smaller than -0` ) .paramsSubcasesOnly([ // Control case: everything to 0 is ok, covers the empty viewport case. - { x: 0, y: 0, w: 0, h: 0 }, + { w: 0, h: 0 }, // Test -1 - { x: -1, y: 0, w: 0, h: 0 }, - { x: 0, y: -1, w: 0, h: 0 }, - { x: 0, y: 0, w: -1, h: 0 }, - { x: 0, y: 0, w: 0, h: -1 }, + { w: -1, h: 0 }, + { w: 0, h: -1 }, ]) .fn(t => { - const { x, y, w, h } = t.params; - const success = x >= 0 && y >= 0 && w >= 0 && h >= 0; - t.testViewportCall(success, { x, y, w, h, minDepth: 0, maxDepth: 1 }); + const { w, h } = t.params; + const success = w >= 0 && h >= 0; + t.testViewportCall(success, { x: 0, y: 0, w, h, minDepth: 0, maxDepth: 1 }); }); -g.test('setViewport,xy_rect_contained_in_attachment') +g.test('setViewport,exceeds_attachment_size') .desc( - 'Test that the rectangle defined by x, y, width, height must be contained in the attachments' + `Test that the viewport can exceed the attachment size` + ) + .paramsSubcasesOnly([ + { attachmentWidth: 3, attachmentHeight: 3 }, + { attachmentWidth: 1024, attachmentHeight: 1024 }, + ]) + .fn(t => { + const { attachmentWidth, attachmentHeight } = t.params; + t.testViewportCall( + true, + { x: 0, y: 0, w: attachmentWidth+1, h: attachmentHeight+1, minDepth: 0, maxDepth: 1 }, + { width: attachmentWidth, height: attachmentHeight, depthOrArrayLayers: 1 } + ); + }); + +g.test('setViewport,viewport_within_bounds') + .desc( + 'Test that the rectangle defined by x, y, width, height must be contained in the maximum viewport bounds' ) .paramsSubcasesOnly(u => u .combineWithParams([ - { attachmentWidth: 3, attachmentHeight: 5 }, - { attachmentWidth: 5, attachmentHeight: 3 }, - { attachmentWidth: 1024, attachmentHeight: 1 }, - { attachmentWidth: 1, attachmentHeight: 1024 }, - ]) - .combineWithParams([ - // Control case: a full viewport is valid. - { dx: 0, dy: 0, dw: 0, dh: 0 }, - - // Other valid cases with a partial viewport. - { dx: 1, dy: 0, dw: -1, dh: 0 }, - { dx: 0, dy: 1, dw: 0, dh: -1 }, - { dx: 0, dy: 0, dw: -1, dh: 0 }, - { dx: 0, dy: 0, dw: 0, dh: -1 }, - - // Test with a small value that causes the viewport to go outside the attachment. - { dx: 1, dy: 0, dw: 0, dh: 0 }, - { dx: 0, dy: 1, dw: 0, dh: 0 }, - { dx: 0, dy: 0, dw: 1, dh: 0 }, - { dx: 0, dy: 0, dw: 0, dh: 1 }, + // Control case: max viewport is valid. + { mx: 0, my: 0, dx: 0, dy: 0, dw: 0, dh: 0 }, + + // Other valid cases + { mx: -1, my: 0, dx: 0, dy: 0, dw: 0, dh: 0 }, + { mx: -2, my: 0, dx: 0, dy: 0, dw: 0, dh: 0 }, + { mx: 1, my: 0, dx: -1, dy: 0, dw: 0, dh: 0 }, + { mx: 0, my: -1, dx: 0, dy: 0, dw: 0, dh: 0 }, + { mx: 0, my: -2, dx: 0, dy: 0, dw: 0, dh: 0 }, + { mx: 0, my: 1, dx: 0, dy: -1, dw: 0, dh: 0 }, + { mx: 0, my: 0, dx: -1, dy: 0, dw: 0, dh: 0 }, + { mx: 0, my: 0, dx: 1, dy: 0, dw: 0, dh: 0 }, + { mx: 0, my: 0, dx: 0, dy: -1, dw: 0, dh: 0 }, + { mx: 0, my: 0, dx: 0, dy: 1, dw: 0, dh: 0 }, + { mx: 1, my: 0, dx: 0, dy: 0, dw: -1, dh: 0 }, + { mx: 0, my: 1, dx: 0, dy: 0, dw: 0, dh: -1 }, + + // Cases that go outside the allowed bounds + { mx: -2, my: 0, dx: -1, dy: 0, dw: 0, dh: 0 }, + { mx: 0, my: -2, dx: 0, dy: -1, dw: 0, dh: 0 }, + { mx: 1, my: 0, dx: 0, dy: 0, dw: 0, dh: 0 }, + { mx: 0, my: 1, dx: 0, dy: 0, dw: 0, dh: 0 }, + { mx: 1, my: 0, dx: 1, dy: 0, dw: -1, dh: 0 }, + { mx: 0, my: 1, dx: 0, dy: 1, dw: 0, dh: -1 }, + + // Cases that exceed the max viewport size + { mx: 0, my: 0, dx: 0, dy: 0, dw: 1, dh: 0 }, + { mx: 0, my: 0, dx: 0, dy: 0, dw: 0, dh: 1 }, ]) ) .fn(t => { - const { attachmentWidth, attachmentHeight, dx, dy, dw, dh } = t.params; - const x = dx; - const y = dy; - const w = attachmentWidth + dw; - const h = attachmentWidth + dh; - - const success = x + w <= attachmentWidth && y + h <= attachmentHeight; - t.testViewportCall( - success, - { x, y, w, h, minDepth: 0, maxDepth: 1 }, - { width: attachmentWidth, height: attachmentHeight, depthOrArrayLayers: 1 } - ); + const { mx, my, dx, dy } = t.params; + const maxViewportSize = t.device.limits.maxTextureDimension2D; + const maxViewportBounds = maxViewportSize * 2; + + const x = maxViewportSize * mx + dx; + const y = maxViewportSize * my + dy; + const w = maxViewportSize; + const h = maxViewportSize; + + const inBounds = x >= -maxViewportBounds && y >= -maxViewportBounds && + x + w <= maxViewportBounds - 1 && y + h <= maxViewportBounds -1; + const validSize = w <= maxViewportSize && h <= maxViewportSize; + const success = inBounds && validSize; + t.testViewportCall(success, { x, y, w, h, minDepth: 0, maxDepth: 1 }); }); g.test('setViewport,depth_rangeAndOrder')