2018-10-09 00:31:58 +00:00
|
|
|
// Copyright 2018 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 "common/Assert.h"
|
|
|
|
#include "common/Constants.h"
|
2018-12-05 03:22:04 +00:00
|
|
|
#include "common/Math.h"
|
2018-12-10 19:47:22 +00:00
|
|
|
#include "utils/ComboRenderPipelineDescriptor.h"
|
2018-10-09 00:31:58 +00:00
|
|
|
#include "utils/DawnHelpers.h"
|
|
|
|
|
2018-11-16 12:11:20 +00:00
|
|
|
#include <array>
|
|
|
|
|
2018-10-09 00:31:58 +00:00
|
|
|
constexpr static unsigned int kRTSize = 64;
|
2019-06-19 09:26:07 +00:00
|
|
|
constexpr dawn::TextureFormat kDefaultFormat = dawn::TextureFormat::RGBA8Unorm;
|
2018-12-05 03:22:04 +00:00
|
|
|
constexpr uint32_t kBytesPerTexel = 4;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
dawn::Texture Create2DTexture(dawn::Device device,
|
|
|
|
uint32_t width,
|
|
|
|
uint32_t height,
|
2019-02-21 00:45:19 +00:00
|
|
|
uint32_t arrayLayerCount,
|
|
|
|
uint32_t mipLevelCount,
|
2019-08-27 08:21:39 +00:00
|
|
|
dawn::TextureUsage usage) {
|
2018-12-05 03:22:04 +00:00
|
|
|
dawn::TextureDescriptor descriptor;
|
|
|
|
descriptor.dimension = dawn::TextureDimension::e2D;
|
|
|
|
descriptor.size.width = width;
|
|
|
|
descriptor.size.height = height;
|
|
|
|
descriptor.size.depth = 1;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.arrayLayerCount = arrayLayerCount;
|
2018-12-12 09:27:16 +00:00
|
|
|
descriptor.sampleCount = 1;
|
2018-12-05 03:22:04 +00:00
|
|
|
descriptor.format = kDefaultFormat;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.mipLevelCount = mipLevelCount;
|
2018-12-05 03:22:04 +00:00
|
|
|
descriptor.usage = usage;
|
|
|
|
return device.CreateTexture(&descriptor);
|
|
|
|
}
|
|
|
|
|
|
|
|
dawn::ShaderModule CreateDefaultVertexShaderModule(dawn::Device device) {
|
2019-08-27 08:42:29 +00:00
|
|
|
return utils::CreateShaderModule(device, utils::SingleShaderStage::Vertex, R"(
|
2018-12-05 03:22:04 +00:00
|
|
|
#version 450
|
|
|
|
layout (location = 0) out vec2 o_texCoord;
|
|
|
|
void main() {
|
|
|
|
const vec2 pos[6] = vec2[6](vec2(-2.f, -2.f),
|
|
|
|
vec2(-2.f, 2.f),
|
|
|
|
vec2( 2.f, -2.f),
|
|
|
|
vec2(-2.f, 2.f),
|
|
|
|
vec2( 2.f, -2.f),
|
|
|
|
vec2( 2.f, 2.f));
|
|
|
|
const vec2 texCoord[6] = vec2[6](vec2(0.f, 0.f),
|
|
|
|
vec2(0.f, 1.f),
|
|
|
|
vec2(1.f, 0.f),
|
|
|
|
vec2(0.f, 1.f),
|
|
|
|
vec2(1.f, 0.f),
|
|
|
|
vec2(1.f, 1.f));
|
|
|
|
gl_Position = vec4(pos[gl_VertexIndex], 0.f, 1.f);
|
|
|
|
o_texCoord = texCoord[gl_VertexIndex];
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
}
|
|
|
|
} // anonymous namespace
|
2018-10-09 00:31:58 +00:00
|
|
|
|
2018-12-05 03:22:04 +00:00
|
|
|
class TextureViewSamplingTest : public DawnTest {
|
2018-10-09 00:31:58 +00:00
|
|
|
protected:
|
2018-10-23 11:08:20 +00:00
|
|
|
// Generates an arbitrary pixel value per-layer-per-level, used for the "actual" uploaded
|
|
|
|
// textures and the "expected" results.
|
|
|
|
static int GenerateTestPixelValue(uint32_t layer, uint32_t level) {
|
|
|
|
return static_cast<int>(level * 10) + static_cast<int>(layer + 1);
|
|
|
|
}
|
|
|
|
|
2018-10-09 00:31:58 +00:00
|
|
|
void SetUp() override {
|
|
|
|
DawnTest::SetUp();
|
|
|
|
|
|
|
|
mRenderPass = utils::CreateBasicRenderPass(device, kRTSize, kRTSize);
|
|
|
|
|
|
|
|
dawn::FilterMode kFilterMode = dawn::FilterMode::Nearest;
|
|
|
|
dawn::AddressMode kAddressMode = dawn::AddressMode::ClampToEdge;
|
|
|
|
|
|
|
|
dawn::SamplerDescriptor samplerDescriptor;
|
|
|
|
samplerDescriptor.minFilter = kFilterMode;
|
|
|
|
samplerDescriptor.magFilter = kFilterMode;
|
|
|
|
samplerDescriptor.mipmapFilter = kFilterMode;
|
|
|
|
samplerDescriptor.addressModeU = kAddressMode;
|
|
|
|
samplerDescriptor.addressModeV = kAddressMode;
|
|
|
|
samplerDescriptor.addressModeW = kAddressMode;
|
2019-01-04 04:56:08 +00:00
|
|
|
samplerDescriptor.lodMinClamp = kLodMin;
|
|
|
|
samplerDescriptor.lodMaxClamp = kLodMax;
|
2019-07-08 08:29:57 +00:00
|
|
|
samplerDescriptor.compare = dawn::CompareFunction::Never;
|
2018-10-09 00:31:58 +00:00
|
|
|
mSampler = device.CreateSampler(&samplerDescriptor);
|
|
|
|
|
2018-12-05 03:22:04 +00:00
|
|
|
mVSModule = CreateDefaultVertexShaderModule(device);
|
2018-10-09 00:31:58 +00:00
|
|
|
}
|
|
|
|
|
2019-02-21 00:45:19 +00:00
|
|
|
void initTexture(uint32_t arrayLayerCount, uint32_t mipLevelCount) {
|
|
|
|
ASSERT(arrayLayerCount > 0 && mipLevelCount > 0);
|
2018-10-23 11:08:20 +00:00
|
|
|
|
2019-02-21 00:45:19 +00:00
|
|
|
const uint32_t textureWidthLevel0 = 1 << mipLevelCount;
|
|
|
|
const uint32_t textureHeightLevel0 = 1 << mipLevelCount;
|
2019-08-27 08:21:39 +00:00
|
|
|
constexpr dawn::TextureUsage kUsage =
|
|
|
|
dawn::TextureUsage::CopyDst | dawn::TextureUsage::Sampled;
|
2018-12-05 03:22:04 +00:00
|
|
|
mTexture = Create2DTexture(
|
2019-02-21 00:45:19 +00:00
|
|
|
device, textureWidthLevel0, textureHeightLevel0, arrayLayerCount, mipLevelCount, kUsage);
|
2018-10-09 00:31:58 +00:00
|
|
|
|
2018-10-23 11:08:20 +00:00
|
|
|
mDefaultTextureViewDescriptor.dimension = dawn::TextureViewDimension::e2DArray;
|
2018-12-05 03:22:04 +00:00
|
|
|
mDefaultTextureViewDescriptor.format = kDefaultFormat;
|
2018-10-23 11:08:20 +00:00
|
|
|
mDefaultTextureViewDescriptor.baseMipLevel = 0;
|
2019-02-21 00:45:19 +00:00
|
|
|
mDefaultTextureViewDescriptor.mipLevelCount = mipLevelCount;
|
2018-10-23 11:08:20 +00:00
|
|
|
mDefaultTextureViewDescriptor.baseArrayLayer = 0;
|
2019-02-21 00:45:19 +00:00
|
|
|
mDefaultTextureViewDescriptor.arrayLayerCount = arrayLayerCount;
|
2018-10-23 11:08:20 +00:00
|
|
|
|
|
|
|
// Create a texture with pixel = (0, 0, 0, level * 10 + layer + 1) at level `level` and
|
|
|
|
// layer `layer`.
|
|
|
|
static_assert((kTextureRowPitchAlignment % sizeof(RGBA8)) == 0,
|
|
|
|
"Texture row pitch alignment must be a multiple of sizeof(RGBA8).");
|
|
|
|
constexpr uint32_t kPixelsPerRowPitch = kTextureRowPitchAlignment / sizeof(RGBA8);
|
|
|
|
ASSERT_LE(textureWidthLevel0, kPixelsPerRowPitch);
|
2018-10-09 00:31:58 +00:00
|
|
|
|
2019-02-15 12:54:08 +00:00
|
|
|
dawn::CommandEncoder encoder = device.CreateCommandEncoder();
|
2019-02-21 00:45:19 +00:00
|
|
|
for (uint32_t layer = 0; layer < arrayLayerCount; ++layer) {
|
|
|
|
for (uint32_t level = 0; level < mipLevelCount; ++level) {
|
2018-10-23 11:08:20 +00:00
|
|
|
const uint32_t texWidth = textureWidthLevel0 >> level;
|
|
|
|
const uint32_t texHeight = textureHeightLevel0 >> level;
|
|
|
|
|
|
|
|
const int pixelValue = GenerateTestPixelValue(layer, level);
|
|
|
|
|
|
|
|
constexpr uint32_t kPaddedTexWidth = kPixelsPerRowPitch;
|
|
|
|
std::vector<RGBA8> data(kPaddedTexWidth * texHeight, RGBA8(0, 0, 0, pixelValue));
|
2019-08-27 08:21:39 +00:00
|
|
|
dawn::Buffer stagingBuffer = utils::CreateBufferFromData(
|
|
|
|
device, data.data(), data.size() * sizeof(RGBA8), dawn::BufferUsage::CopySrc);
|
2018-11-28 17:54:13 +00:00
|
|
|
dawn::BufferCopyView bufferCopyView =
|
|
|
|
utils::CreateBufferCopyView(stagingBuffer, 0, kTextureRowPitchAlignment, 0);
|
2018-12-12 09:27:46 +00:00
|
|
|
dawn::TextureCopyView textureCopyView =
|
|
|
|
utils::CreateTextureCopyView(mTexture, level, layer, {0, 0, 0});
|
2018-11-28 17:54:13 +00:00
|
|
|
dawn::Extent3D copySize = {texWidth, texHeight, 1};
|
2019-02-15 12:54:08 +00:00
|
|
|
encoder.CopyBufferToTexture(&bufferCopyView, &textureCopyView, ©Size);
|
2018-10-23 11:08:20 +00:00
|
|
|
}
|
2018-10-09 00:31:58 +00:00
|
|
|
}
|
2019-02-15 12:54:08 +00:00
|
|
|
dawn::CommandBuffer copy = encoder.Finish();
|
2018-10-09 00:31:58 +00:00
|
|
|
queue.Submit(1, ©);
|
|
|
|
}
|
|
|
|
|
2019-09-10 08:58:28 +00:00
|
|
|
void Verify(const dawn::TextureView& textureView,
|
|
|
|
dawn::TextureViewDimension dimension,
|
|
|
|
const char* fragmentShader,
|
|
|
|
int expected) {
|
|
|
|
dawn::BindGroupLayout bindGroupLayout = utils::MakeBindGroupLayout(
|
|
|
|
device, {
|
|
|
|
{0, dawn::ShaderStage::Fragment, dawn::BindingType::Sampler},
|
|
|
|
{1, dawn::ShaderStage::Fragment, dawn::BindingType::SampledTexture, false,
|
|
|
|
false, dimension, dawn::TextureComponentType::Float},
|
|
|
|
});
|
|
|
|
|
|
|
|
dawn::BindGroup bindGroup =
|
|
|
|
utils::MakeBindGroup(device, bindGroupLayout, {{0, mSampler}, {1, textureView}});
|
2018-10-09 00:31:58 +00:00
|
|
|
|
|
|
|
dawn::ShaderModule fsModule =
|
2019-08-27 08:42:29 +00:00
|
|
|
utils::CreateShaderModule(device, utils::SingleShaderStage::Fragment, fragmentShader);
|
2018-10-09 00:31:58 +00:00
|
|
|
|
2018-12-10 19:47:22 +00:00
|
|
|
utils::ComboRenderPipelineDescriptor textureDescriptor(device);
|
2019-09-05 09:35:07 +00:00
|
|
|
textureDescriptor.vertexStage.module = mVSModule;
|
2018-12-10 19:47:22 +00:00
|
|
|
textureDescriptor.cFragmentStage.module = fsModule;
|
2019-09-10 08:58:28 +00:00
|
|
|
textureDescriptor.layout = utils::MakeBasicPipelineLayout(device, &bindGroupLayout);
|
2019-02-20 13:00:36 +00:00
|
|
|
textureDescriptor.cColorStates[0]->format = mRenderPass.colorFormat;
|
2018-12-10 19:47:22 +00:00
|
|
|
|
|
|
|
dawn::RenderPipeline pipeline = device.CreateRenderPipeline(&textureDescriptor);
|
2018-10-09 00:31:58 +00:00
|
|
|
|
2019-02-15 12:54:08 +00:00
|
|
|
dawn::CommandEncoder encoder = device.CreateCommandEncoder();
|
2018-10-09 00:31:58 +00:00
|
|
|
{
|
2019-02-27 09:21:56 +00:00
|
|
|
dawn::RenderPassEncoder pass = encoder.BeginRenderPass(&mRenderPass.renderPassInfo);
|
2018-12-21 10:40:26 +00:00
|
|
|
pass.SetPipeline(pipeline);
|
2019-03-18 06:01:37 +00:00
|
|
|
pass.SetBindGroup(0, bindGroup, 0, nullptr);
|
2018-12-10 05:20:19 +00:00
|
|
|
pass.Draw(6, 1, 0, 0);
|
2018-10-09 00:31:58 +00:00
|
|
|
pass.EndPass();
|
|
|
|
}
|
|
|
|
|
2019-02-15 12:54:08 +00:00
|
|
|
dawn::CommandBuffer commands = encoder.Finish();
|
2018-10-09 00:31:58 +00:00
|
|
|
queue.Submit(1, &commands);
|
|
|
|
|
2018-10-23 11:08:20 +00:00
|
|
|
RGBA8 expectedPixel(0, 0, 0, expected);
|
|
|
|
EXPECT_PIXEL_RGBA8_EQ(expectedPixel, mRenderPass.color, 0, 0);
|
|
|
|
EXPECT_PIXEL_RGBA8_EQ(
|
|
|
|
expectedPixel, mRenderPass.color, mRenderPass.width - 1, mRenderPass.height - 1);
|
2018-10-09 00:31:58 +00:00
|
|
|
// TODO(jiawei.shao@intel.com): add tests for 3D textures once Dawn supports 3D textures
|
|
|
|
}
|
|
|
|
|
2018-10-23 11:08:20 +00:00
|
|
|
void Texture2DViewTest(uint32_t textureArrayLayers,
|
|
|
|
uint32_t textureMipLevels,
|
|
|
|
uint32_t textureViewBaseLayer,
|
|
|
|
uint32_t textureViewBaseMipLevel) {
|
|
|
|
ASSERT(textureViewBaseLayer < textureArrayLayers);
|
|
|
|
ASSERT(textureViewBaseMipLevel < textureMipLevels);
|
|
|
|
|
|
|
|
initTexture(textureArrayLayers, textureMipLevels);
|
|
|
|
|
|
|
|
dawn::TextureViewDescriptor descriptor = mDefaultTextureViewDescriptor;
|
|
|
|
descriptor.dimension = dawn::TextureViewDimension::e2D;
|
|
|
|
descriptor.baseArrayLayer = textureViewBaseLayer;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.arrayLayerCount = 1;
|
2018-10-23 11:08:20 +00:00
|
|
|
descriptor.baseMipLevel = textureViewBaseMipLevel;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.mipLevelCount = 1;
|
2019-04-09 16:57:00 +00:00
|
|
|
dawn::TextureView textureView = mTexture.CreateView(&descriptor);
|
2018-10-23 11:08:20 +00:00
|
|
|
|
|
|
|
const char* fragmentShader = R"(
|
|
|
|
#version 450
|
|
|
|
layout(set = 0, binding = 0) uniform sampler sampler0;
|
|
|
|
layout(set = 0, binding = 1) uniform texture2D texture0;
|
2018-11-16 12:11:20 +00:00
|
|
|
layout(location = 0) in vec2 texCoord;
|
2018-10-23 11:08:20 +00:00
|
|
|
layout(location = 0) out vec4 fragColor;
|
|
|
|
|
|
|
|
void main() {
|
|
|
|
fragColor =
|
2018-11-16 12:11:20 +00:00
|
|
|
texture(sampler2D(texture0, sampler0), texCoord);
|
2018-10-23 11:08:20 +00:00
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
const int expected = GenerateTestPixelValue(textureViewBaseLayer, textureViewBaseMipLevel);
|
2019-09-10 08:58:28 +00:00
|
|
|
Verify(textureView, dawn::TextureViewDimension::e2D, fragmentShader, expected);
|
2018-10-23 11:08:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Texture2DArrayViewTest(uint32_t textureArrayLayers,
|
|
|
|
uint32_t textureMipLevels,
|
|
|
|
uint32_t textureViewBaseLayer,
|
|
|
|
uint32_t textureViewBaseMipLevel) {
|
|
|
|
ASSERT(textureViewBaseLayer < textureArrayLayers);
|
|
|
|
ASSERT(textureViewBaseMipLevel < textureMipLevels);
|
|
|
|
|
|
|
|
// We always set the layer count of the texture view to be 3 to match the fragment shader in
|
|
|
|
// this test.
|
|
|
|
constexpr uint32_t kTextureViewLayerCount = 3;
|
|
|
|
ASSERT(textureArrayLayers >= textureViewBaseLayer + kTextureViewLayerCount);
|
|
|
|
|
|
|
|
initTexture(textureArrayLayers, textureMipLevels);
|
|
|
|
|
|
|
|
dawn::TextureViewDescriptor descriptor = mDefaultTextureViewDescriptor;
|
|
|
|
descriptor.dimension = dawn::TextureViewDimension::e2DArray;
|
|
|
|
descriptor.baseArrayLayer = textureViewBaseLayer;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.arrayLayerCount = kTextureViewLayerCount;
|
2018-10-23 11:08:20 +00:00
|
|
|
descriptor.baseMipLevel = textureViewBaseMipLevel;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.mipLevelCount = 1;
|
2019-04-09 16:57:00 +00:00
|
|
|
dawn::TextureView textureView = mTexture.CreateView(&descriptor);
|
2018-10-23 11:08:20 +00:00
|
|
|
|
|
|
|
const char* fragmentShader = R"(
|
|
|
|
#version 450
|
|
|
|
layout(set = 0, binding = 0) uniform sampler sampler0;
|
|
|
|
layout(set = 0, binding = 1) uniform texture2DArray texture0;
|
2018-11-16 12:11:20 +00:00
|
|
|
layout(location = 0) in vec2 texCoord;
|
2018-10-23 11:08:20 +00:00
|
|
|
layout(location = 0) out vec4 fragColor;
|
|
|
|
|
|
|
|
void main() {
|
|
|
|
fragColor =
|
2018-11-16 12:11:20 +00:00
|
|
|
texture(sampler2DArray(texture0, sampler0), vec3(texCoord, 0)) +
|
|
|
|
texture(sampler2DArray(texture0, sampler0), vec3(texCoord, 1)) +
|
|
|
|
texture(sampler2DArray(texture0, sampler0), vec3(texCoord, 2));
|
2018-10-23 11:08:20 +00:00
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
int expected = 0;
|
|
|
|
for (int i = 0; i < static_cast<int>(kTextureViewLayerCount); ++i) {
|
|
|
|
expected += GenerateTestPixelValue(textureViewBaseLayer + i, textureViewBaseMipLevel);
|
|
|
|
}
|
2019-09-10 08:58:28 +00:00
|
|
|
Verify(textureView, dawn::TextureViewDimension::e2DArray, fragmentShader, expected);
|
2018-10-23 11:08:20 +00:00
|
|
|
}
|
|
|
|
|
2018-11-16 12:11:20 +00:00
|
|
|
std::string CreateFragmentShaderForCubeMapFace(uint32_t layer, bool isCubeMapArray) {
|
|
|
|
// Reference: https://en.wikipedia.org/wiki/Cube_mapping
|
|
|
|
const std::array<std::string, 6> kCoordsToCubeMapFace = {{
|
|
|
|
" 1.f, tc, -sc", // Positive X
|
|
|
|
"-1.f, tc, sc", // Negative X
|
|
|
|
" sc, 1.f, -tc", // Positive Y
|
|
|
|
" sc, -1.f, tc", // Negative Y
|
|
|
|
" sc, tc, 1.f", // Positive Z
|
|
|
|
" -sc, tc, -1.f", // Negative Z
|
|
|
|
}};
|
|
|
|
|
|
|
|
const std::string textureType = isCubeMapArray ? "textureCubeArray" : "textureCube";
|
|
|
|
const std::string samplerType = isCubeMapArray ? "samplerCubeArray" : "samplerCube";
|
|
|
|
const uint32_t cubeMapArrayIndex = layer / 6;
|
|
|
|
const std::string coordToCubeMapFace = kCoordsToCubeMapFace[layer % 6];
|
|
|
|
|
|
|
|
std::ostringstream stream;
|
|
|
|
stream << R"(
|
|
|
|
#version 450
|
|
|
|
layout(set = 0, binding = 0) uniform sampler sampler0;
|
|
|
|
layout(set = 0, binding = 1) uniform )" << textureType << R"( texture0;
|
|
|
|
layout(location = 0) in vec2 texCoord;
|
|
|
|
layout(location = 0) out vec4 fragColor;
|
|
|
|
void main() {
|
|
|
|
float sc = 2.f * texCoord.x - 1.f;
|
|
|
|
float tc = 2.f * texCoord.y - 1.f;
|
|
|
|
fragColor = texture()" << samplerType << "(texture0, sampler0), ";
|
|
|
|
|
|
|
|
if (isCubeMapArray) {
|
|
|
|
stream << "vec4(" << coordToCubeMapFace << ", " << cubeMapArrayIndex;
|
|
|
|
} else {
|
|
|
|
stream << "vec3(" << coordToCubeMapFace;
|
|
|
|
}
|
|
|
|
|
|
|
|
stream << R"());
|
|
|
|
})";
|
|
|
|
|
|
|
|
return stream.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureCubeMapTest(uint32_t textureArrayLayers,
|
|
|
|
uint32_t textureViewBaseLayer,
|
|
|
|
uint32_t textureViewLayerCount,
|
|
|
|
bool isCubeMapArray) {
|
|
|
|
constexpr uint32_t kMipLevels = 1u;
|
|
|
|
initTexture(textureArrayLayers, kMipLevels);
|
|
|
|
|
|
|
|
ASSERT_TRUE((textureViewLayerCount == 6) ||
|
|
|
|
(isCubeMapArray && textureViewLayerCount % 6 == 0));
|
2019-09-10 08:58:28 +00:00
|
|
|
dawn::TextureViewDimension dimension = (isCubeMapArray)
|
|
|
|
? dawn::TextureViewDimension::CubeArray
|
|
|
|
: dawn::TextureViewDimension::Cube;
|
2018-11-16 12:11:20 +00:00
|
|
|
|
|
|
|
dawn::TextureViewDescriptor descriptor = mDefaultTextureViewDescriptor;
|
2019-09-10 08:58:28 +00:00
|
|
|
descriptor.dimension = dimension;
|
2018-11-16 12:11:20 +00:00
|
|
|
descriptor.baseArrayLayer = textureViewBaseLayer;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.arrayLayerCount = textureViewLayerCount;
|
2018-11-16 12:11:20 +00:00
|
|
|
|
2019-04-09 16:57:00 +00:00
|
|
|
dawn::TextureView cubeMapTextureView = mTexture.CreateView(&descriptor);
|
2018-11-16 12:11:20 +00:00
|
|
|
|
|
|
|
// Check the data in the every face of the cube map (array) texture view.
|
|
|
|
for (uint32_t layer = 0; layer < textureViewLayerCount; ++layer) {
|
|
|
|
const std::string &fragmentShader =
|
|
|
|
CreateFragmentShaderForCubeMapFace(layer, isCubeMapArray);
|
|
|
|
|
|
|
|
int expected = GenerateTestPixelValue(textureViewBaseLayer + layer, 0);
|
2019-09-10 08:58:28 +00:00
|
|
|
Verify(cubeMapTextureView, dimension, fragmentShader.c_str(), expected);
|
2018-11-16 12:11:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-09 00:31:58 +00:00
|
|
|
dawn::Sampler mSampler;
|
|
|
|
dawn::Texture mTexture;
|
2018-10-23 11:08:20 +00:00
|
|
|
dawn::TextureViewDescriptor mDefaultTextureViewDescriptor;
|
2018-10-09 00:31:58 +00:00
|
|
|
dawn::ShaderModule mVSModule;
|
|
|
|
utils::BasicRenderPass mRenderPass;
|
|
|
|
};
|
|
|
|
|
2018-10-23 11:08:20 +00:00
|
|
|
// Test drawing a rect with a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, Default2DArrayTexture) {
|
2018-10-16 09:16:15 +00:00
|
|
|
// TODO(cwallez@chromium.org) understand what the issue is
|
|
|
|
DAWN_SKIP_TEST_IF(IsVulkan() && IsNvidia());
|
|
|
|
|
2018-10-09 00:31:58 +00:00
|
|
|
constexpr uint32_t kLayers = 3;
|
2018-10-23 11:08:20 +00:00
|
|
|
constexpr uint32_t kMipLevels = 1;
|
|
|
|
initTexture(kLayers, kMipLevels);
|
2018-10-09 00:31:58 +00:00
|
|
|
|
2019-08-27 17:56:23 +00:00
|
|
|
dawn::TextureView textureView = mTexture.CreateView();
|
2018-10-09 00:31:58 +00:00
|
|
|
|
|
|
|
const char* fragmentShader = R"(
|
|
|
|
#version 450
|
|
|
|
layout(set = 0, binding = 0) uniform sampler sampler0;
|
|
|
|
layout(set = 0, binding = 1) uniform texture2DArray texture0;
|
2018-11-16 12:11:20 +00:00
|
|
|
layout(location = 0) in vec2 texCoord;
|
2018-10-09 00:31:58 +00:00
|
|
|
layout(location = 0) out vec4 fragColor;
|
|
|
|
|
|
|
|
void main() {
|
|
|
|
fragColor =
|
2018-11-16 12:11:20 +00:00
|
|
|
texture(sampler2DArray(texture0, sampler0), vec3(texCoord, 0)) +
|
|
|
|
texture(sampler2DArray(texture0, sampler0), vec3(texCoord, 1)) +
|
|
|
|
texture(sampler2DArray(texture0, sampler0), vec3(texCoord, 2));
|
2018-10-09 00:31:58 +00:00
|
|
|
}
|
|
|
|
)";
|
2018-10-23 11:08:20 +00:00
|
|
|
|
|
|
|
const int expected = GenerateTestPixelValue(0, 0) + GenerateTestPixelValue(1, 0) +
|
|
|
|
GenerateTestPixelValue(2, 0);
|
2019-09-10 08:58:28 +00:00
|
|
|
Verify(textureView, dawn::TextureViewDimension::e2DArray, fragmentShader, expected);
|
2018-10-23 11:08:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a 2D texture view created on a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, Texture2DViewOn2DArrayTexture) {
|
2018-10-23 11:08:20 +00:00
|
|
|
Texture2DViewTest(6, 1, 4, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a 2D array texture view created on a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, Texture2DArrayViewOn2DArrayTexture) {
|
2018-11-29 10:54:03 +00:00
|
|
|
DAWN_SKIP_TEST_IF(IsMetal() && IsIntel());
|
2018-10-23 11:08:20 +00:00
|
|
|
Texture2DArrayViewTest(6, 1, 2, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a 2D texture view created on a mipmap level of a 2D texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, Texture2DViewOnOneLevelOf2DTexture) {
|
2018-10-23 11:08:20 +00:00
|
|
|
Texture2DViewTest(1, 6, 0, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a 2D texture view created on a mipmap level of a 2D array texture layer.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, Texture2DViewOnOneLevelOf2DArrayTexture) {
|
2018-10-23 11:08:20 +00:00
|
|
|
Texture2DViewTest(6, 6, 3, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a 2D array texture view created on a mipmap level of a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, Texture2DArrayViewOnOneLevelOf2DArrayTexture) {
|
2018-11-29 10:54:03 +00:00
|
|
|
DAWN_SKIP_TEST_IF(IsMetal() && IsIntel());
|
2018-10-23 11:08:20 +00:00
|
|
|
Texture2DArrayViewTest(6, 6, 2, 4);
|
2018-10-09 00:31:58 +00:00
|
|
|
}
|
|
|
|
|
2018-11-16 12:11:20 +00:00
|
|
|
// Test sampling from a cube map texture view that covers a whole 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, TextureCubeMapOnWholeTexture) {
|
2018-11-16 12:11:20 +00:00
|
|
|
constexpr uint32_t kTotalLayers = 6;
|
|
|
|
TextureCubeMapTest(kTotalLayers, 0, kTotalLayers, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a cube map texture view that covers a sub part of a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, TextureCubeMapViewOnPartOfTexture) {
|
2018-11-16 12:11:20 +00:00
|
|
|
TextureCubeMapTest(10, 2, 6, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a cube map texture view that covers the last layer of a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, TextureCubeMapViewCoveringLastLayer) {
|
2018-11-16 12:11:20 +00:00
|
|
|
constexpr uint32_t kTotalLayers = 10;
|
|
|
|
constexpr uint32_t kBaseLayer = 4;
|
|
|
|
TextureCubeMapTest(kTotalLayers, kBaseLayer, kTotalLayers - kBaseLayer, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a cube map texture array view that covers a whole 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, TextureCubeMapArrayOnWholeTexture) {
|
2018-11-16 12:11:20 +00:00
|
|
|
constexpr uint32_t kTotalLayers = 12;
|
|
|
|
TextureCubeMapTest(kTotalLayers, 0, kTotalLayers, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a cube map texture array view that covers a sub part of a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, TextureCubeMapArrayViewOnPartOfTexture) {
|
2018-12-03 12:22:06 +00:00
|
|
|
// Test failing on the GPU FYI Mac Pro (AMD), see
|
|
|
|
// https://bugs.chromium.org/p/dawn/issues/detail?id=58
|
|
|
|
DAWN_SKIP_TEST_IF(IsMacOS() && IsMetal() && IsAMD());
|
|
|
|
|
2018-11-16 12:11:20 +00:00
|
|
|
TextureCubeMapTest(20, 3, 12, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a cube map texture array view that covers the last layer of a 2D array texture.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, TextureCubeMapArrayViewCoveringLastLayer) {
|
2018-12-03 12:22:06 +00:00
|
|
|
// Test failing on the GPU FYI Mac Pro (AMD), see
|
|
|
|
// https://bugs.chromium.org/p/dawn/issues/detail?id=58
|
|
|
|
DAWN_SKIP_TEST_IF(IsMacOS() && IsMetal() && IsAMD());
|
|
|
|
|
2018-11-16 12:11:20 +00:00
|
|
|
constexpr uint32_t kTotalLayers = 20;
|
|
|
|
constexpr uint32_t kBaseLayer = 8;
|
|
|
|
TextureCubeMapTest(kTotalLayers, kBaseLayer, kTotalLayers - kBaseLayer, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test sampling from a cube map array texture view that only has a single cube map.
|
2018-12-05 03:22:04 +00:00
|
|
|
TEST_P(TextureViewSamplingTest, TextureCubeMapArrayViewSingleCubeMap) {
|
2018-12-03 12:22:06 +00:00
|
|
|
// Test failing on the GPU FYI Mac Pro (AMD), see
|
|
|
|
// https://bugs.chromium.org/p/dawn/issues/detail?id=58
|
|
|
|
DAWN_SKIP_TEST_IF(IsMacOS() && IsMetal() && IsAMD());
|
|
|
|
|
2018-11-16 12:11:20 +00:00
|
|
|
TextureCubeMapTest(20, 7, 6, true);
|
|
|
|
}
|
|
|
|
|
2018-12-05 03:22:04 +00:00
|
|
|
class TextureViewRenderingTest : public DawnTest {
|
|
|
|
protected:
|
|
|
|
void TextureLayerAsColorAttachmentTest(dawn::TextureViewDimension dimension,
|
|
|
|
uint32_t layerCount,
|
|
|
|
uint32_t levelCount,
|
|
|
|
uint32_t textureViewBaseLayer,
|
|
|
|
uint32_t textureViewBaseLevel) {
|
|
|
|
ASSERT(dimension == dawn::TextureViewDimension::e2D ||
|
|
|
|
dimension == dawn::TextureViewDimension::e2DArray);
|
|
|
|
ASSERT_LT(textureViewBaseLayer, layerCount);
|
|
|
|
ASSERT_LT(textureViewBaseLevel, levelCount);
|
|
|
|
|
|
|
|
const uint32_t textureWidthLevel0 = 1 << levelCount;
|
|
|
|
const uint32_t textureHeightLevel0 = 1 << levelCount;
|
2019-08-27 08:21:39 +00:00
|
|
|
constexpr dawn::TextureUsage kUsage =
|
|
|
|
dawn::TextureUsage::OutputAttachment | dawn::TextureUsage::CopySrc;
|
2018-12-05 03:22:04 +00:00
|
|
|
dawn::Texture texture = Create2DTexture(
|
|
|
|
device, textureWidthLevel0, textureHeightLevel0, layerCount, levelCount, kUsage);
|
|
|
|
|
|
|
|
dawn::TextureViewDescriptor descriptor;
|
|
|
|
descriptor.format = kDefaultFormat;
|
|
|
|
descriptor.dimension = dimension;
|
|
|
|
descriptor.baseArrayLayer = textureViewBaseLayer;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.arrayLayerCount = 1;
|
2018-12-05 03:22:04 +00:00
|
|
|
descriptor.baseMipLevel = textureViewBaseLevel;
|
2019-02-21 00:45:19 +00:00
|
|
|
descriptor.mipLevelCount = 1;
|
2019-04-09 16:57:00 +00:00
|
|
|
dawn::TextureView textureView = texture.CreateView(&descriptor);
|
2018-12-05 03:22:04 +00:00
|
|
|
|
|
|
|
dawn::ShaderModule vsModule = CreateDefaultVertexShaderModule(device);
|
|
|
|
|
|
|
|
// Clear textureView with Red(255, 0, 0, 255) and render Green(0, 255, 0, 255) into it
|
2019-02-27 09:21:56 +00:00
|
|
|
utils::ComboRenderPassDescriptor renderPassInfo({textureView});
|
2019-09-20 22:59:47 +00:00
|
|
|
renderPassInfo.cColorAttachments[0].clearColor = {1.0f, 0.0f, 0.0f, 1.0f};
|
2018-12-05 03:22:04 +00:00
|
|
|
|
|
|
|
const char* oneColorFragmentShader = R"(
|
|
|
|
#version 450
|
|
|
|
layout(location = 0) out vec4 fragColor;
|
|
|
|
|
|
|
|
void main() {
|
|
|
|
fragColor = vec4(0.0, 1.0, 0.0, 1.0);
|
|
|
|
}
|
|
|
|
)";
|
2019-08-27 08:42:29 +00:00
|
|
|
dawn::ShaderModule oneColorFsModule = utils::CreateShaderModule(
|
|
|
|
device, utils::SingleShaderStage::Fragment, oneColorFragmentShader);
|
2018-12-05 03:22:04 +00:00
|
|
|
|
2018-12-10 19:47:22 +00:00
|
|
|
utils::ComboRenderPipelineDescriptor pipelineDescriptor(device);
|
2019-09-05 09:35:07 +00:00
|
|
|
pipelineDescriptor.vertexStage.module = vsModule;
|
2018-12-10 19:47:22 +00:00
|
|
|
pipelineDescriptor.cFragmentStage.module = oneColorFsModule;
|
2019-02-20 13:00:36 +00:00
|
|
|
pipelineDescriptor.cColorStates[0]->format = kDefaultFormat;
|
2018-12-10 19:47:22 +00:00
|
|
|
|
|
|
|
dawn::RenderPipeline oneColorPipeline = device.CreateRenderPipeline(&pipelineDescriptor);
|
|
|
|
|
2019-02-15 12:54:08 +00:00
|
|
|
dawn::CommandEncoder encoder = device.CreateCommandEncoder();
|
2018-12-05 03:22:04 +00:00
|
|
|
{
|
2019-02-27 09:21:56 +00:00
|
|
|
dawn::RenderPassEncoder pass = encoder.BeginRenderPass(&renderPassInfo);
|
2018-12-21 10:40:26 +00:00
|
|
|
pass.SetPipeline(oneColorPipeline);
|
2018-12-10 05:20:19 +00:00
|
|
|
pass.Draw(6, 1, 0, 0);
|
2018-12-05 03:22:04 +00:00
|
|
|
pass.EndPass();
|
|
|
|
}
|
|
|
|
|
2019-02-15 12:54:08 +00:00
|
|
|
dawn::CommandBuffer commands = encoder.Finish();
|
2018-12-05 03:22:04 +00:00
|
|
|
queue.Submit(1, &commands);
|
|
|
|
|
|
|
|
// Check if the right pixels (Green) have been written into the right part of the texture.
|
|
|
|
uint32_t textureViewWidth = textureWidthLevel0 >> textureViewBaseLevel;
|
|
|
|
uint32_t textureViewHeight = textureHeightLevel0 >> textureViewBaseLevel;
|
|
|
|
uint32_t rowPitch = Align(kBytesPerTexel * textureWidthLevel0, kTextureRowPitchAlignment);
|
|
|
|
uint32_t expectedDataSize =
|
|
|
|
rowPitch / kBytesPerTexel * (textureWidthLevel0 - 1) + textureHeightLevel0;
|
|
|
|
constexpr RGBA8 kExpectedPixel(0, 255, 0, 255);
|
|
|
|
std::vector<RGBA8> expected(expectedDataSize, kExpectedPixel);
|
|
|
|
EXPECT_TEXTURE_RGBA8_EQ(
|
|
|
|
expected.data(), texture, 0, 0, textureViewWidth, textureViewHeight,
|
|
|
|
textureViewBaseLevel, textureViewBaseLayer);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test rendering into a 2D texture view created on a mipmap level of a 2D texture.
|
|
|
|
TEST_P(TextureViewRenderingTest, Texture2DViewOnALevelOf2DTextureAsColorAttachment) {
|
|
|
|
constexpr uint32_t kLayers = 1;
|
|
|
|
constexpr uint32_t kMipLevels = 4;
|
|
|
|
constexpr uint32_t kBaseLayer = 0;
|
|
|
|
|
|
|
|
// Rendering into the first level
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLevel = 0;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2D, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rendering into the last level
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLevel = kMipLevels - 1;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2D, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test rendering into a 2D texture view created on a layer of a 2D array texture.
|
|
|
|
TEST_P(TextureViewRenderingTest, Texture2DViewOnALayerOf2DArrayTextureAsColorAttachment) {
|
|
|
|
constexpr uint32_t kMipLevels = 1;
|
|
|
|
constexpr uint32_t kBaseLevel = 0;
|
|
|
|
constexpr uint32_t kLayers = 10;
|
|
|
|
|
|
|
|
// Rendering into the first layer
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLayer = 0;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2D, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rendering into the last layer
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLayer = kLayers - 1;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2D, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test rendering into a 1-layer 2D array texture view created on a mipmap level of a 2D texture.
|
|
|
|
TEST_P(TextureViewRenderingTest, Texture2DArrayViewOnALevelOf2DTextureAsColorAttachment) {
|
|
|
|
constexpr uint32_t kLayers = 1;
|
|
|
|
constexpr uint32_t kMipLevels = 4;
|
|
|
|
constexpr uint32_t kBaseLayer = 0;
|
|
|
|
|
|
|
|
// Rendering into the first level
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLevel = 0;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2DArray, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rendering into the last level
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLevel = kMipLevels - 1;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2DArray, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test rendering into a 1-layer 2D array texture view created on a layer of a 2D array texture.
|
|
|
|
TEST_P(TextureViewRenderingTest, Texture2DArrayViewOnALayerOf2DArrayTextureAsColorAttachment) {
|
|
|
|
constexpr uint32_t kMipLevels = 1;
|
|
|
|
constexpr uint32_t kBaseLevel = 0;
|
|
|
|
constexpr uint32_t kLayers = 10;
|
|
|
|
|
|
|
|
// Rendering into the first layer
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLayer = 0;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2DArray, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rendering into the last layer
|
|
|
|
{
|
|
|
|
constexpr uint32_t kBaseLayer = kLayers - 1;
|
|
|
|
TextureLayerAsColorAttachmentTest(
|
|
|
|
dawn::TextureViewDimension::e2DArray, kLayers, kMipLevels, kBaseLayer, kBaseLevel);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-21 14:57:01 +00:00
|
|
|
DAWN_INSTANTIATE_TEST(TextureViewSamplingTest, D3D12Backend, MetalBackend, OpenGLBackend, VulkanBackend);
|
2018-12-05 03:22:04 +00:00
|
|
|
|
2019-02-21 14:57:01 +00:00
|
|
|
DAWN_INSTANTIATE_TEST(TextureViewRenderingTest, D3D12Backend, MetalBackend, OpenGLBackend, VulkanBackend);
|