2019-05-23 00:45:12 +00:00
|
|
|
// Copyright 2019 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 "tests/DawnTest.h"
|
|
|
|
|
|
|
|
#include "utils/ComboRenderPipelineDescriptor.h"
|
|
|
|
#include "utils/DawnHelpers.h"
|
|
|
|
|
|
|
|
constexpr uint32_t kRTSize = 400;
|
|
|
|
constexpr uint32_t kBufferElementsCount = kMinDynamicBufferOffsetAlignment / sizeof(uint32_t) + 2;
|
|
|
|
constexpr uint32_t kBufferSize = kBufferElementsCount * sizeof(uint32_t);
|
2019-07-05 08:01:10 +00:00
|
|
|
constexpr uint32_t kBindingSize = 8;
|
2019-05-23 00:45:12 +00:00
|
|
|
|
|
|
|
class DynamicBufferOffsetTests : public DawnTest {
|
|
|
|
protected:
|
|
|
|
void SetUp() override {
|
|
|
|
DawnTest::SetUp();
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
// Mix up dynamic and non dynamic resources in one bind group and using not continuous
|
|
|
|
// binding number to cover more cases.
|
2019-05-23 00:45:12 +00:00
|
|
|
std::array<uint32_t, kBufferElementsCount> uniformData = {0};
|
|
|
|
uniformData[0] = 1;
|
|
|
|
uniformData[1] = 2;
|
|
|
|
|
|
|
|
mUniformBuffer = utils::CreateBufferFromData(device, uniformData.data(), kBufferSize,
|
|
|
|
dawn::BufferUsageBit::Uniform);
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
uniformData[uniformData.size() - 2] = 5;
|
|
|
|
uniformData[uniformData.size() - 1] = 6;
|
|
|
|
|
|
|
|
mDynamicUniformBuffer = utils::CreateBufferFromData(device, uniformData.data(), kBufferSize,
|
|
|
|
dawn::BufferUsageBit::Uniform);
|
|
|
|
|
2019-05-23 00:45:12 +00:00
|
|
|
dawn::BufferDescriptor storageBufferDescriptor;
|
|
|
|
storageBufferDescriptor.size = kBufferSize;
|
|
|
|
storageBufferDescriptor.usage = dawn::BufferUsageBit::Storage |
|
2019-07-08 10:05:46 +00:00
|
|
|
dawn::BufferUsageBit::CopyDst |
|
|
|
|
dawn::BufferUsageBit::CopySrc;
|
2019-05-23 00:45:12 +00:00
|
|
|
|
|
|
|
mStorageBuffer = device.CreateBuffer(&storageBufferDescriptor);
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
mDynamicStorageBuffer = device.CreateBuffer(&storageBufferDescriptor);
|
|
|
|
|
2019-08-08 22:41:00 +00:00
|
|
|
mBindGroupLayout = utils::MakeBindGroupLayout(
|
2019-05-23 00:45:12 +00:00
|
|
|
device, {{0, dawn::ShaderStageBit::Compute | dawn::ShaderStageBit::Fragment,
|
2019-07-31 01:29:42 +00:00
|
|
|
dawn::BindingType::UniformBuffer},
|
2019-05-23 00:45:12 +00:00
|
|
|
{1, dawn::ShaderStageBit::Compute | dawn::ShaderStageBit::Fragment,
|
2019-07-31 01:29:42 +00:00
|
|
|
dawn::BindingType::StorageBuffer},
|
|
|
|
{3, dawn::ShaderStageBit::Compute | dawn::ShaderStageBit::Fragment,
|
|
|
|
dawn::BindingType::UniformBuffer, true},
|
|
|
|
{4, dawn::ShaderStageBit::Compute | dawn::ShaderStageBit::Fragment,
|
2019-07-09 07:58:57 +00:00
|
|
|
dawn::BindingType::StorageBuffer, true}});
|
2019-05-23 00:45:12 +00:00
|
|
|
|
2019-08-08 22:41:00 +00:00
|
|
|
mBindGroup = utils::MakeBindGroup(device, mBindGroupLayout,
|
2019-07-31 01:29:42 +00:00
|
|
|
{{0, mUniformBuffer, 0, kBindingSize},
|
|
|
|
{1, mStorageBuffer, 0, kBindingSize},
|
|
|
|
{3, mDynamicUniformBuffer, 0, kBindingSize},
|
|
|
|
{4, mDynamicStorageBuffer, 0, kBindingSize}});
|
2019-05-23 00:45:12 +00:00
|
|
|
}
|
|
|
|
// Create objects to use as resources inside test bind groups.
|
|
|
|
|
|
|
|
dawn::BindGroup mBindGroup;
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::BindGroupLayout mBindGroupLayout;
|
2019-05-23 00:45:12 +00:00
|
|
|
dawn::Buffer mUniformBuffer;
|
|
|
|
dawn::Buffer mStorageBuffer;
|
2019-07-31 01:29:42 +00:00
|
|
|
dawn::Buffer mDynamicUniformBuffer;
|
|
|
|
dawn::Buffer mDynamicStorageBuffer;
|
2019-05-23 00:45:12 +00:00
|
|
|
dawn::Texture mColorAttachment;
|
|
|
|
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::RenderPipeline CreateRenderPipeline() {
|
|
|
|
dawn::ShaderModule vsModule =
|
|
|
|
utils::CreateShaderModule(device, utils::ShaderStage::Vertex, R"(
|
2019-05-23 00:45:12 +00:00
|
|
|
#version 450
|
|
|
|
void main() {
|
|
|
|
const vec2 pos[3] = vec2[3](vec2(-1.0f, 0.0f), vec2(-1.0f, -1.0f), vec2(0.0f, -1.0f));
|
|
|
|
gl_Position = vec4(pos[gl_VertexIndex], 0.0, 1.0);
|
|
|
|
})");
|
|
|
|
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::ShaderModule fsModule =
|
|
|
|
utils::CreateShaderModule(device, utils::ShaderStage::Fragment, R"(
|
2019-05-23 00:45:12 +00:00
|
|
|
#version 450
|
2019-07-31 01:29:42 +00:00
|
|
|
layout(std140, set = 0, binding = 0) uniform uBufferNotDynamic {
|
|
|
|
uvec2 notDynamicValue;
|
|
|
|
};
|
|
|
|
layout(std140, set = 0, binding = 1) buffer sBufferNotDynamic {
|
|
|
|
uvec2 notDynamicResult;
|
|
|
|
} mid;
|
|
|
|
layout(std140, set = 0, binding = 3) uniform uBuffer {
|
2019-05-23 00:45:12 +00:00
|
|
|
uvec2 value;
|
|
|
|
};
|
2019-07-31 01:29:42 +00:00
|
|
|
layout(std140, set = 0, binding = 4) buffer SBuffer {
|
2019-05-23 00:45:12 +00:00
|
|
|
uvec2 result;
|
|
|
|
} sBuffer;
|
|
|
|
layout(location = 0) out vec4 fragColor;
|
|
|
|
void main() {
|
2019-07-31 01:29:42 +00:00
|
|
|
mid.notDynamicResult.xy = notDynamicValue.xy;
|
|
|
|
sBuffer.result.xy = value.xy + mid.notDynamicResult.xy;
|
2019-05-23 00:45:12 +00:00
|
|
|
fragColor = vec4(value.x / 255.0f, value.y / 255.0f, 1.0f, 1.0f);
|
|
|
|
})");
|
|
|
|
|
2019-08-08 22:41:00 +00:00
|
|
|
utils::ComboRenderPipelineDescriptor pipelineDescriptor(device);
|
|
|
|
pipelineDescriptor.cVertexStage.module = vsModule;
|
|
|
|
pipelineDescriptor.cFragmentStage.module = fsModule;
|
|
|
|
pipelineDescriptor.cColorStates[0]->format = dawn::TextureFormat::RGBA8Unorm;
|
|
|
|
dawn::PipelineLayout pipelineLayout =
|
|
|
|
utils::MakeBasicPipelineLayout(device, &mBindGroupLayout);
|
|
|
|
pipelineDescriptor.layout = pipelineLayout;
|
2019-05-23 00:45:12 +00:00
|
|
|
|
2019-08-08 22:41:00 +00:00
|
|
|
return device.CreateRenderPipeline(&pipelineDescriptor);
|
2019-05-23 00:45:12 +00:00
|
|
|
}
|
|
|
|
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::ComputePipeline CreateComputePipeline() {
|
|
|
|
dawn::ShaderModule csModule =
|
|
|
|
utils::CreateShaderModule(device, utils::ShaderStage::Compute, R"(
|
2019-05-23 00:45:12 +00:00
|
|
|
#version 450
|
2019-08-08 22:41:00 +00:00
|
|
|
layout(local_size_x = 1, local_size_y = 1, local_size_z = 1) in;
|
2019-07-31 01:29:42 +00:00
|
|
|
layout(std140, set = 0, binding = 0) uniform uBufferNotDynamic {
|
|
|
|
uvec2 notDynamicValue;
|
2019-05-23 00:45:12 +00:00
|
|
|
};
|
2019-07-31 01:29:42 +00:00
|
|
|
layout(std140, set = 0, binding = 1) buffer sBufferNotDynamic {
|
|
|
|
uvec2 notDynamicResult;
|
|
|
|
} mid;
|
|
|
|
layout(std140, set = 0, binding = 3) uniform uBuffer {
|
|
|
|
uvec2 value;
|
|
|
|
};
|
|
|
|
layout(std140, set = 0, binding = 4) buffer SBuffer {
|
|
|
|
uvec2 result;
|
2019-05-23 00:45:12 +00:00
|
|
|
} sBuffer;
|
|
|
|
|
|
|
|
void main() {
|
2019-07-31 01:29:42 +00:00
|
|
|
mid.notDynamicResult.xy = notDynamicValue.xy;
|
2019-08-08 22:41:00 +00:00
|
|
|
sBuffer.result.xy = value.xy + mid.notDynamicResult.xy;
|
2019-05-23 00:45:12 +00:00
|
|
|
})");
|
|
|
|
|
|
|
|
dawn::ComputePipelineDescriptor csDesc;
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::PipelineLayout pipelineLayout =
|
|
|
|
utils::MakeBasicPipelineLayout(device, &mBindGroupLayout);
|
|
|
|
csDesc.layout = pipelineLayout;
|
2019-05-23 00:45:12 +00:00
|
|
|
|
|
|
|
dawn::PipelineStageDescriptor computeStage;
|
2019-08-08 22:41:00 +00:00
|
|
|
computeStage.module = csModule;
|
2019-05-23 00:45:12 +00:00
|
|
|
computeStage.entryPoint = "main";
|
|
|
|
csDesc.computeStage = &computeStage;
|
|
|
|
|
|
|
|
return device.CreateComputePipeline(&csDesc);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Dynamic offsets are all zero and no effect to result.
|
|
|
|
TEST_P(DynamicBufferOffsetTests, BasicRenderPipeline) {
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::RenderPipeline pipeline = CreateRenderPipeline();
|
2019-05-23 00:45:12 +00:00
|
|
|
utils::BasicRenderPass renderPass = utils::CreateBasicRenderPass(device, kRTSize, kRTSize);
|
|
|
|
|
|
|
|
dawn::CommandEncoder commandEncoder = device.CreateCommandEncoder();
|
|
|
|
std::array<uint64_t, 2> offsets = {0, 0};
|
|
|
|
dawn::RenderPassEncoder renderPassEncoder =
|
|
|
|
commandEncoder.BeginRenderPass(&renderPass.renderPassInfo);
|
|
|
|
renderPassEncoder.SetPipeline(pipeline);
|
|
|
|
renderPassEncoder.SetBindGroup(0, mBindGroup, offsets.size(), offsets.data());
|
|
|
|
renderPassEncoder.Draw(3, 1, 0, 0);
|
|
|
|
renderPassEncoder.EndPass();
|
|
|
|
dawn::CommandBuffer commands = commandEncoder.Finish();
|
|
|
|
queue.Submit(1, &commands);
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
std::vector<uint32_t> expectedData = {2, 4};
|
2019-05-23 00:45:12 +00:00
|
|
|
EXPECT_PIXEL_RGBA8_EQ(RGBA8(1, 2, 255, 255), renderPass.color, 0, 0);
|
2019-07-31 01:29:42 +00:00
|
|
|
EXPECT_BUFFER_U32_RANGE_EQ(expectedData.data(), mDynamicStorageBuffer, 0, expectedData.size());
|
2019-05-23 00:45:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Have non-zero dynamic offsets.
|
|
|
|
TEST_P(DynamicBufferOffsetTests, SetDynamicOffestsRenderPipeline) {
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::RenderPipeline pipeline = CreateRenderPipeline();
|
2019-05-23 00:45:12 +00:00
|
|
|
utils::BasicRenderPass renderPass = utils::CreateBasicRenderPass(device, kRTSize, kRTSize);
|
|
|
|
|
|
|
|
dawn::CommandEncoder commandEncoder = device.CreateCommandEncoder();
|
|
|
|
std::array<uint64_t, 2> offsets = {kMinDynamicBufferOffsetAlignment,
|
|
|
|
kMinDynamicBufferOffsetAlignment};
|
|
|
|
dawn::RenderPassEncoder renderPassEncoder =
|
|
|
|
commandEncoder.BeginRenderPass(&renderPass.renderPassInfo);
|
|
|
|
renderPassEncoder.SetPipeline(pipeline);
|
|
|
|
renderPassEncoder.SetBindGroup(0, mBindGroup, offsets.size(), offsets.data());
|
|
|
|
renderPassEncoder.Draw(3, 1, 0, 0);
|
|
|
|
renderPassEncoder.EndPass();
|
|
|
|
dawn::CommandBuffer commands = commandEncoder.Finish();
|
|
|
|
queue.Submit(1, &commands);
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
std::vector<uint32_t> expectedData = {6, 8};
|
2019-05-23 00:45:12 +00:00
|
|
|
EXPECT_PIXEL_RGBA8_EQ(RGBA8(5, 6, 255, 255), renderPass.color, 0, 0);
|
2019-07-31 01:29:42 +00:00
|
|
|
EXPECT_BUFFER_U32_RANGE_EQ(expectedData.data(), mDynamicStorageBuffer,
|
2019-05-23 00:45:12 +00:00
|
|
|
kMinDynamicBufferOffsetAlignment, expectedData.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Dynamic offsets are all zero and no effect to result.
|
|
|
|
TEST_P(DynamicBufferOffsetTests, BasicComputePipeline) {
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::ComputePipeline pipeline = CreateComputePipeline();
|
2019-05-23 00:45:12 +00:00
|
|
|
|
|
|
|
std::array<uint64_t, 2> offsets = {0, 0};
|
|
|
|
|
|
|
|
dawn::CommandEncoder commandEncoder = device.CreateCommandEncoder();
|
|
|
|
dawn::ComputePassEncoder computePassEncoder = commandEncoder.BeginComputePass();
|
|
|
|
computePassEncoder.SetPipeline(pipeline);
|
|
|
|
computePassEncoder.SetBindGroup(0, mBindGroup, offsets.size(), offsets.data());
|
|
|
|
computePassEncoder.Dispatch(1, 1, 1);
|
|
|
|
computePassEncoder.EndPass();
|
|
|
|
dawn::CommandBuffer commands = commandEncoder.Finish();
|
|
|
|
queue.Submit(1, &commands);
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
std::vector<uint32_t> expectedData = {2, 4};
|
|
|
|
EXPECT_BUFFER_U32_RANGE_EQ(expectedData.data(), mDynamicStorageBuffer, 0, expectedData.size());
|
2019-05-23 00:45:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Have non-zero dynamic offsets.
|
|
|
|
TEST_P(DynamicBufferOffsetTests, SetDynamicOffestsComputePipeline) {
|
2019-08-08 22:41:00 +00:00
|
|
|
dawn::ComputePipeline pipeline = CreateComputePipeline();
|
2019-05-23 00:45:12 +00:00
|
|
|
|
|
|
|
std::array<uint64_t, 2> offsets = {kMinDynamicBufferOffsetAlignment,
|
|
|
|
kMinDynamicBufferOffsetAlignment};
|
|
|
|
|
|
|
|
dawn::CommandEncoder commandEncoder = device.CreateCommandEncoder();
|
|
|
|
dawn::ComputePassEncoder computePassEncoder = commandEncoder.BeginComputePass();
|
|
|
|
computePassEncoder.SetPipeline(pipeline);
|
|
|
|
computePassEncoder.SetBindGroup(0, mBindGroup, offsets.size(), offsets.data());
|
|
|
|
computePassEncoder.Dispatch(1, 1, 1);
|
|
|
|
computePassEncoder.EndPass();
|
|
|
|
dawn::CommandBuffer commands = commandEncoder.Finish();
|
|
|
|
queue.Submit(1, &commands);
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
std::vector<uint32_t> expectedData = {6, 8};
|
|
|
|
EXPECT_BUFFER_U32_RANGE_EQ(expectedData.data(), mDynamicStorageBuffer,
|
2019-05-23 00:45:12 +00:00
|
|
|
kMinDynamicBufferOffsetAlignment, expectedData.size());
|
|
|
|
}
|
|
|
|
|
2019-07-31 01:29:42 +00:00
|
|
|
DAWN_INSTANTIATE_TEST(DynamicBufferOffsetTests,
|
|
|
|
D3D12Backend,
|
|
|
|
MetalBackend,
|
|
|
|
OpenGLBackend,
|
|
|
|
VulkanBackend);
|