From 3a0c2eac16c9a8a53ad0b54cdec16cd77d395f9c Mon Sep 17 00:00:00 2001 From: Brandon Jones Date: Wed, 18 Dec 2024 11:01:54 -0800 Subject: [PATCH 1/2] Loosen Viewport validation tests to match spec update --- .../cmds/render/dynamic_state.spec.ts | 120 +++++++++++------- src/webgpu/listing_meta.json | 4 +- 2 files changed, 75 insertions(+), 49 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..e8d5eff4078d 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,97 @@ 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 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,xy_rect_contained_in_bounds') .desc( - 'Test that the rectangle defined by x, y, width, height must be contained in the attachments' + `Test that the rectangle defined by x, y, width, height must be contained in the maximum viewport bounds +and that the viewport size cannot exceed the maximum.` ) .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 }, - ]) + u.combineWithParams([ + // 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') diff --git a/src/webgpu/listing_meta.json b/src/webgpu/listing_meta.json index 618753748601..894164403209 100644 --- a/src/webgpu/listing_meta.json +++ b/src/webgpu/listing_meta.json @@ -499,8 +499,8 @@ "webgpu:api,validation,encoding,cmds,render,dynamic_state:setScissorRect,xy_rect_contained_in_attachment:*": { "subcaseMS": 1.325 }, "webgpu:api,validation,encoding,cmds,render,dynamic_state:setStencilReference:*": { "subcaseMS": 3.450 }, "webgpu:api,validation,encoding,cmds,render,dynamic_state:setViewport,depth_rangeAndOrder:*": { "subcaseMS": 1.667 }, - "webgpu:api,validation,encoding,cmds,render,dynamic_state:setViewport,x_y_width_height_nonnegative:*": { "subcaseMS": 0.400 }, - "webgpu:api,validation,encoding,cmds,render,dynamic_state:setViewport,xy_rect_contained_in_attachment:*": { "subcaseMS": 0.200 }, + "webgpu:api,validation,encoding,cmds,render,dynamic_state:setViewport,width_height_nonnegative:*": { "subcaseMS": 0.400 }, + "webgpu:api,validation,encoding,cmds,render,dynamic_state:setViewport,xy_rect_contained_in_bounds:*": { "subcaseMS": 0.200 }, "webgpu:api,validation,encoding,cmds,render,indirect_draw:indirect_buffer,device_mismatch:*": { "subcaseMS": 2.000 }, "webgpu:api,validation,encoding,cmds,render,indirect_draw:indirect_buffer_state:*": { "subcaseMS": 2.708 }, "webgpu:api,validation,encoding,cmds,render,indirect_draw:indirect_buffer_usage:*": { "subcaseMS": 2.733 }, From 764466539316e7aa6f2caf43d2b2783407be421f Mon Sep 17 00:00:00 2001 From: Brandon Jones Date: Thu, 19 Dec 2024 13:52:15 -0800 Subject: [PATCH 2/2] Address review feedback --- .../cmds/render/dynamic_state.spec.ts | 76 ++++++++++--------- 1 file changed, 40 insertions(+), 36 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 e8d5eff4078d..f118be37f3a6 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 @@ -23,6 +23,7 @@ TODO: ensure existing tests cover these notes. Note many of these may be operati `; import { makeTestGroup } from '../../../../../../common/framework/test_group.js'; +import { MaxLimitsTestMixin } from '../../../../../gpu_test.js'; import { ValidationTest } from '../../../validation_test.js'; interface ViewportCall { @@ -128,7 +129,7 @@ class F extends ValidationTest { } } -export const g = makeTestGroup(F); +export const g = makeTestGroup(MaxLimitsTestMixin(F)); g.test('setViewport,width_height_nonnegative') .desc( @@ -139,16 +140,20 @@ TODO Test the first value smaller than -0` ) .paramsSubcasesOnly([ // Control case: everything to 0 is ok, covers the empty viewport case. - { w: 0, h: 0 }, + { x: 0, y: 0, w: 0, h: 0 }, - // Test -1 - { w: -1, h: 0 }, - { w: 0, h: -1 }, + // Negative width/height is invalid + { x: 0, y: 0, w: -1, h: 0 }, + { x: 0, y: 0, w: 0, h: -1 }, + + // Negative width/height is invalid even if the resulting bounds are positive + { x: 1, y: 0, w: -1, h: 0 }, + { x: 0, y: 1, w: 0, h: -1 }, ]) .fn(t => { - const { w, h } = t.params; + const { x, y, w, h } = t.params; const success = w >= 0 && h >= 0; - t.testViewportCall(success, { x: 0, y: 0, w, h, minDepth: 0, maxDepth: 1 }); + t.testViewportCall(success, { x, y, w, h, minDepth: 0, maxDepth: 1 }); }); g.test('setViewport,exceeds_attachment_size') @@ -172,46 +177,45 @@ g.test('setViewport,xy_rect_contained_in_bounds') and that the viewport size cannot exceed the maximum.` ) .paramsSubcasesOnly(u => - u.combineWithParams([ + u.combine('dimension', [0, 1]).combineWithParams([ // Control case: max viewport is valid. - { mx: 0, my: 0, dx: 0, dy: 0, dw: 0, dh: 0 }, + { om: 0, od: 0, sd: 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 }, + { om: -1, od: 0, sd: 0 }, + { om: -2, od: 0, sd: 0 }, + { om: 1, od: -1, sd: 0 }, + { om: 0, od: -1, sd: 0 }, + { om: 0, od: 1, sd: 0 }, + { om: 1, od: 0, sd: -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 }, + { om: -2, od: -1, sd: 0 }, + { om: 1, od: 0, sd: 0 }, + { om: 1, od: 1, sd: -1 }, + { om: 1, od: -0.1, sd: 0 }, + + // Case that exceeds the max viewport size + { om: 0, od: 0, sd: 1 }, + { om: 0, od: 0, sd: 0.1 }, ]) ) .fn(t => { - const { mx, my, dx, dy } = t.params; + const { dimension, om, od, sd } = 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 xy = [0, 0]; + const wh = [maxViewportSize, maxViewportSize]; + + xy[dimension] = maxViewportSize * om + od; + wh[dimension] += sd; + + const x = xy[0]; + const y = xy[1]; + const w = wh[0]; + const h = wh[1]; const inBounds = x >= -maxViewportBounds &&