Skip to content

Commit

Permalink
Compat: refactor setBindGroup for 0 storage buffers in frag stage. (g…
Browse files Browse the repository at this point in the history
…puweb#4125)

Most of these tests didn't seem to require storage buffers
to test what they were testing so I switched them to uniform buffers.

One of them used both storage and uniform buffers so I added more
subcases so that it tests with both storage buffers and uniform
buffers in compute stage only (this is for devices that don't support
storage buffers in fragment stage). Then both compute+fragment stages
like it was, but this case is skipped if there are no storage buffers
in fragment shaders. I also added a case were it uses 2 uniform buffers.

It's not clear what it's testing needs to buffers to be of different
types.
  • Loading branch information
greggman authored Dec 31, 2024
1 parent b72bd61 commit 58f9d5d
Showing 1 changed file with 28 additions and 14 deletions.
42 changes: 28 additions & 14 deletions src/webgpu/api/validation/encoding/cmds/setBindGroup.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,8 @@ import {
kBufferBindingTypes,
kMinDynamicBufferOffsetAlignment,
} from '../../../../capability_info.js';
import { kResourceStates, ResourceState } from '../../../../gpu_test.js';
import { GPUConst } from '../../../../constants.js';
import { kResourceStates, MaxLimitsTestMixin, ResourceState } from '../../../../gpu_test.js';
import {
kProgrammableEncoderTypes,
ProgrammableEncoderType,
Expand Down Expand Up @@ -53,7 +54,7 @@ class F extends ValidationTest {
return {
buffer: this.createBufferWithState(state, {
size: 4,
usage: GPUBufferUsage.STORAGE,
usage: GPUBufferUsage.UNIFORM,
}),
};
default:
Expand All @@ -80,7 +81,7 @@ class F extends ValidationTest {
entries: indices.map(binding => ({
binding,
visibility: this.encoderTypeToStageFlag(encoderType),
...(resourceType === 'buffer' ? { buffer: { type: 'storage' } } : { texture: {} }),
...(resourceType === 'buffer' ? { buffer: { type: 'uniform' } } : { texture: {} }),
})),
});
const bindGroup = this.device.createBindGroup({
Expand All @@ -101,7 +102,7 @@ class F extends ValidationTest {
}
}

export const g = makeTestGroup(F);
export const g = makeTestGroup(MaxLimitsTestMixin(F));

g.test('state_and_binding_index')
.desc('Tests that setBindGroup correctly handles {valid, invalid, destroyed} bindGroups.')
Expand Down Expand Up @@ -153,7 +154,7 @@ g.test('bind_group,device_mismatch')
const buffer = t.trackForCleanup(
sourceDevice.createBuffer({
size: 4,
usage: GPUBufferUsage.STORAGE,
usage: GPUBufferUsage.UNIFORM,
})
);

Expand All @@ -162,7 +163,7 @@ g.test('bind_group,device_mismatch')
{
binding: 0,
visibility: t.encoderTypeToStageFlag(encoderType),
buffer: { type: 'storage', hasDynamicOffset: useU32Array },
buffer: { type: 'uniform', hasDynamicOffset: useU32Array },
},
],
});
Expand Down Expand Up @@ -224,25 +225,38 @@ g.test('dynamic_offsets_match_expectations_in_pass_encoder')
{ dynamicOffsets: [0, 0xffffffff], _success: false },
])
.combine('useU32array', [false, true])
.beginSubcases()
.combine('visibility', [
GPUConst.ShaderStage.COMPUTE,
GPUConst.ShaderStage.COMPUTE | GPUConst.ShaderStage.FRAGMENT,
] as const)
.combine('useStorage', [false, true] as const)
)
.fn(t => {
const { visibility, useStorage } = t.params;
t.skipIf(
t.isCompatibility &&
(visibility & GPUShaderStage.FRAGMENT) !== 0 &&
!(t.device.limits.maxStorageBuffersInFragmentStage! >= 1),
`maxStorageBuffersInFragmentStage${t.device.limits.maxStorageBuffersInFragmentStage} < 1`
);
const kBindingSize = 12;

const bindGroupLayout = t.device.createBindGroupLayout({
entries: [
{
binding: 0,
visibility: GPUShaderStage.COMPUTE | GPUShaderStage.FRAGMENT,
visibility,
buffer: {
type: 'uniform',
hasDynamicOffset: true,
},
},
{
binding: 1,
visibility: GPUShaderStage.COMPUTE | GPUShaderStage.FRAGMENT,
visibility,
buffer: {
type: 'storage',
type: useStorage ? 'storage' : 'uniform',
hasDynamicOffset: true,
},
},
Expand All @@ -254,9 +268,9 @@ g.test('dynamic_offsets_match_expectations_in_pass_encoder')
usage: GPUBufferUsage.UNIFORM,
});

const storageBuffer = t.createBufferTracked({
const storageOrUniformBuffer = t.createBufferTracked({
size: 2 * kMinDynamicBufferOffsetAlignment + 8,
usage: GPUBufferUsage.STORAGE,
usage: useStorage ? GPUBufferUsage.STORAGE : GPUBufferUsage.UNIFORM,
});

const bindGroup = t.device.createBindGroup({
Expand All @@ -272,7 +286,7 @@ g.test('dynamic_offsets_match_expectations_in_pass_encoder')
{
binding: 1,
resource: {
buffer: storageBuffer,
buffer: storageOrUniformBuffer,
size: kBindingSize,
},
},
Expand Down Expand Up @@ -335,7 +349,7 @@ g.test('u32array_start_and_length')
binding: i,
visibility: GPUShaderStage.FRAGMENT,
buffer: {
type: 'storage',
type: 'uniform',
hasDynamicOffset: true,
},
})),
Expand All @@ -348,7 +362,7 @@ g.test('u32array_start_and_length')
resource: {
buffer: t.createBufferWithState('valid', {
size: kBindingSize,
usage: GPUBufferUsage.STORAGE,
usage: GPUBufferUsage.UNIFORM,
}),
size: kBindingSize,
},
Expand Down

0 comments on commit 58f9d5d

Please sign in to comment.