| // Copyright 2017 The Dawn Authors |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #include "common/Constants.h" |
| #include "tests/unittests/validation/ValidationTest.h" |
| #include "utils/WGPUHelpers.h" |
| |
| // TODO(cwallez@chromium.org): Add a regression test for Disptach validation trying to acces the |
| // input state. |
| |
| class ComputeValidationTest : public ValidationTest { |
| protected: |
| void SetUp() override { |
| ValidationTest::SetUp(); |
| |
| wgpu::ShaderModule computeModule = utils::CreateShaderModule(device, R"( |
| [[stage(compute), workgroup_size(1)]] fn main() { |
| })"); |
| |
| // Set up compute pipeline |
| wgpu::PipelineLayout pl = utils::MakeBasicPipelineLayout(device, nullptr); |
| |
| wgpu::ComputePipelineDescriptor csDesc; |
| csDesc.layout = pl; |
| csDesc.compute.module = computeModule; |
| csDesc.compute.entryPoint = "main"; |
| pipeline = device.CreateComputePipeline(&csDesc); |
| } |
| |
| void TestDispatch(uint32_t x, uint32_t y, uint32_t z) { |
| wgpu::CommandEncoder encoder = device.CreateCommandEncoder(); |
| wgpu::ComputePassEncoder pass = encoder.BeginComputePass(); |
| pass.SetPipeline(pipeline); |
| pass.Dispatch(x, y, z); |
| pass.EndPass(); |
| encoder.Finish(); |
| } |
| |
| wgpu::ComputePipeline pipeline; |
| }; |
| |
| // Check that 1x1x1 dispatch is OK. |
| TEST_F(ComputeValidationTest, PerDimensionDispatchSizeLimits_SmallestValid) { |
| TestDispatch(1, 1, 1); |
| } |
| |
| // Check that the largest allowed dispatch is OK. |
| TEST_F(ComputeValidationTest, PerDimensionDispatchSizeLimits_LargestValid) { |
| const uint32_t max = GetSupportedLimits().limits.maxComputeWorkgroupsPerDimension; |
| TestDispatch(max, max, max); |
| } |
| |
| // Check that exceeding the maximum on the X dimension results in validation failure. |
| TEST_F(ComputeValidationTest, PerDimensionDispatchSizeLimits_InvalidX) { |
| const uint32_t max = GetSupportedLimits().limits.maxComputeWorkgroupsPerDimension; |
| ASSERT_DEVICE_ERROR(TestDispatch(max + 1, 1, 1)); |
| } |
| |
| // Check that exceeding the maximum on the Y dimension results in validation failure. |
| TEST_F(ComputeValidationTest, PerDimensionDispatchSizeLimits_InvalidY) { |
| const uint32_t max = GetSupportedLimits().limits.maxComputeWorkgroupsPerDimension; |
| ASSERT_DEVICE_ERROR(TestDispatch(1, max + 1, 1)); |
| } |
| |
| // Check that exceeding the maximum on the Z dimension results in validation failure. |
| TEST_F(ComputeValidationTest, PerDimensionDispatchSizeLimits_InvalidZ) { |
| const uint32_t max = GetSupportedLimits().limits.maxComputeWorkgroupsPerDimension; |
| ASSERT_DEVICE_ERROR(TestDispatch(1, 1, max + 1)); |
| } |
| |
| // Check that exceeding the maximum on all dimensions results in validation failure. |
| TEST_F(ComputeValidationTest, PerDimensionDispatchSizeLimits_InvalidAll) { |
| const uint32_t max = GetSupportedLimits().limits.maxComputeWorkgroupsPerDimension; |
| ASSERT_DEVICE_ERROR(TestDispatch(max + 1, max + 1, max + 1)); |
| } |