dawn-cmake/src/tests/NXTTest.cpp

481 lines
16 KiB
C++
Raw Normal View History

// 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 "tests/NXTTest.h"
#include "common/Assert.h"
#include "common/Constants.h"
#include "common/Math.h"
#include "utils/BackendBinding.h"
2018-07-18 12:00:56 +00:00
#include "utils/DawnHelpers.h"
2017-07-17 21:13:57 +00:00
#include "utils/SystemUtils.h"
#include "wire/TerribleCommandBuffer.h"
#include "wire/Wire.h"
#include <iostream>
#include "GLFW/glfw3.h"
namespace {
utils::BackendType ParamToBackendType(BackendType type) {
switch(type) {
case D3D12Backend:
return utils::BackendType::D3D12;
case MetalBackend:
return utils::BackendType::Metal;
case OpenGLBackend:
return utils::BackendType::OpenGL;
case VulkanBackend:
return utils::BackendType::Vulkan;
default:
2017-07-11 01:48:12 +00:00
UNREACHABLE();
}
}
std::string ParamName(BackendType type) {
switch(type) {
case D3D12Backend:
return "D3D12";
case MetalBackend:
return "Metal";
case OpenGLBackend:
return "OpenGL";
case VulkanBackend:
return "Vulkan";
default:
2017-07-11 01:48:12 +00:00
UNREACHABLE();
}
}
// Windows don't usually like to be bound to one API than the other, for example switching
// from Vulkan to OpenGL causes crashes on some drivers. Because of this, we lazily created
// a window for each backing API.
GLFWwindow* windows[NumBackendTypes];
// Creates a GLFW window set up for use with a given backend.
GLFWwindow* GetWindowForBackend(utils::BackendBinding* binding, BackendType type) {
GLFWwindow** window = &windows[type];
if (*window != nullptr) {
return *window;
}
if (!glfwInit()) {
return nullptr;
}
glfwDefaultWindowHints();
binding->SetupGLFWWindowHints();
std::string windowName = "NXT " + ParamName(type) + " test window";
*window = glfwCreateWindow(400, 400, windowName.c_str(), nullptr, nullptr);
return *window;
}
// End2end tests should test valid commands produce the expected result so no error
// should happen. Failure cases should be tested in the validation tests.
void DeviceErrorCauseTestFailure(const char* message, nxtCallbackUserdata) {
FAIL() << "Device level failure: " << message;
}
struct MapReadUserdata {
NXTTest* test;
size_t slot;
};
}
NXTTest::~NXTTest() {
// We need to destroy child objects before the Device
2017-11-23 19:51:16 +00:00
mReadbackSlots.clear();
2018-07-18 11:43:49 +00:00
queue = dawn::Queue();
swapchain = dawn::SwapChain();
device = dawn::Device();
2017-11-23 19:51:16 +00:00
delete mBinding;
mBinding = nullptr;
nxtSetProcs(nullptr);
}
bool NXTTest::IsD3D12() const {
return GetParam() == D3D12Backend;
}
2017-07-18 14:31:50 +00:00
bool NXTTest::IsMetal() const {
return GetParam() == MetalBackend;
}
bool NXTTest::IsOpenGL() const {
return GetParam() == OpenGLBackend;
}
bool NXTTest::IsVulkan() const {
return GetParam() == VulkanBackend;
}
bool gTestUsesWire = false;
void NXTTest::SetUp() {
2017-11-23 19:51:16 +00:00
mBinding = utils::CreateBinding(ParamToBackendType(GetParam()));
DAWN_ASSERT(mBinding != nullptr);
2017-11-23 19:51:16 +00:00
GLFWwindow* testWindow = GetWindowForBackend(mBinding, GetParam());
DAWN_ASSERT(testWindow != nullptr);
2017-11-23 19:51:16 +00:00
mBinding->SetWindow(testWindow);
nxtDevice backendDevice;
nxtProcTable backendProcs;
2017-11-23 19:51:16 +00:00
mBinding->GetProcAndDevice(&backendProcs, &backendDevice);
// Choose whether to use the backend procs and devices directly, or set up the wire.
nxtDevice cDevice = nullptr;
nxtProcTable procs;
if (gTestUsesWire) {
2018-07-18 11:43:49 +00:00
mC2sBuf = new dawn::wire::TerribleCommandBuffer();
mS2cBuf = new dawn::wire::TerribleCommandBuffer();
2018-07-18 11:43:49 +00:00
mWireServer = dawn::wire::NewServerCommandHandler(backendDevice, backendProcs, mS2cBuf);
mC2sBuf->SetHandler(mWireServer);
nxtDevice clientDevice;
nxtProcTable clientProcs;
2018-07-18 11:43:49 +00:00
mWireClient = dawn::wire::NewClientDevice(&clientProcs, &clientDevice, mC2sBuf);
mS2cBuf->SetHandler(mWireClient);
procs = clientProcs;
cDevice = clientDevice;
} else {
procs = backendProcs;
cDevice = backendDevice;
}
// Set up the device and queue because all tests need them, and NXTTest needs them too for the
// deferred expectations.
nxtSetProcs(&procs);
2018-07-18 11:43:49 +00:00
device = dawn::Device::Acquire(cDevice);
queue = device.CreateQueue();
// The swapchain isn't used by tests but is useful when debugging with graphics debuggers that
// capture at frame boundaries.
2017-07-28 01:30:57 +00:00
swapchain = device.CreateSwapChainBuilder()
2017-11-23 19:51:16 +00:00
.SetImplementation(mBinding->GetSwapChainImplementation())
2017-07-28 01:30:57 +00:00
.GetResult();
2018-07-18 11:43:49 +00:00
swapchain.Configure(static_cast<dawn::TextureFormat>(mBinding->GetPreferredSwapChainTextureFormat()),
dawn::TextureUsageBit::OutputAttachment, 400, 400);
2017-07-28 01:30:57 +00:00
// The end2end tests should never cause validation errors. These should be tested in unittests.
device.SetErrorCallback(DeviceErrorCauseTestFailure, 0);
}
void NXTTest::TearDown() {
FlushWire();
MapSlotsSynchronously();
ResolveExpectations();
2017-11-23 19:51:16 +00:00
for (size_t i = 0; i < mReadbackSlots.size(); ++i) {
mReadbackSlots[i].buffer.Unmap();
2017-07-25 15:29:28 +00:00
}
2017-11-23 19:51:16 +00:00
for (auto& expectation : mDeferredExpectations) {
delete expectation.expectation;
expectation.expectation = nullptr;
}
if (gTestUsesWire) {
delete mC2sBuf;
delete mS2cBuf;
delete mWireClient;
delete mWireServer;
}
}
2018-07-18 11:43:49 +00:00
std::ostringstream& NXTTest::AddBufferExpectation(const char* file, int line, const dawn::Buffer& buffer, uint32_t offset, uint32_t size, detail::Expectation* expectation) {
dawn::Buffer source = buffer.Clone();
auto readback = ReserveReadback(size);
// We need to enqueue the copy immediately because by the time we resolve the expectation,
// the buffer might have been modified.
2018-07-18 11:43:49 +00:00
dawn::CommandBuffer commands = device.CreateCommandBufferBuilder()
.CopyBufferToBuffer(source, offset, readback.buffer, readback.offset, size)
.GetResult();
queue.Submit(1, &commands);
DeferredExpectation deferred;
deferred.file = file;
deferred.line = line;
deferred.readbackSlot = readback.slot;
deferred.readbackOffset = readback.offset;
deferred.size = size;
deferred.rowBytes = size;
deferred.rowPitch = size;
deferred.expectation = expectation;
2017-11-23 19:51:16 +00:00
mDeferredExpectations.push_back(std::move(deferred));
mDeferredExpectations.back().message = std::make_unique<std::ostringstream>();
return *(mDeferredExpectations.back().message.get());
}
2018-07-18 11:43:49 +00:00
std::ostringstream& NXTTest::AddTextureExpectation(const char* file, int line, const dawn::Texture& texture, uint32_t x, uint32_t y, uint32_t width, uint32_t height, uint32_t level, uint32_t pixelSize, detail::Expectation* expectation) {
dawn::Texture source = texture.Clone();
uint32_t rowPitch = Align(width * pixelSize, kTextureRowPitchAlignment);
uint32_t size = rowPitch * (height - 1) + width * pixelSize;
2017-06-27 04:11:16 +00:00
auto readback = ReserveReadback(size);
// We need to enqueue the copy immediately because by the time we resolve the expectation,
// the texture might have been modified.
2018-07-18 11:43:49 +00:00
dawn::CommandBuffer commands = device.CreateCommandBufferBuilder()
.CopyTextureToBuffer(source, x, y, 0, width, height, 1, level, readback.buffer, readback.offset, rowPitch)
2017-06-27 04:11:16 +00:00
.GetResult();
queue.Submit(1, &commands);
DeferredExpectation deferred;
deferred.file = file;
deferred.line = line;
deferred.readbackSlot = readback.slot;
deferred.readbackOffset = readback.offset;
deferred.size = size;
deferred.rowBytes = width * pixelSize;
deferred.rowPitch = rowPitch;
2017-06-27 04:11:16 +00:00
deferred.expectation = expectation;
2017-11-23 19:51:16 +00:00
mDeferredExpectations.push_back(std::move(deferred));
mDeferredExpectations.back().message = std::make_unique<std::ostringstream>();
return *(mDeferredExpectations.back().message.get());
2017-06-27 04:11:16 +00:00
}
2017-07-17 21:13:57 +00:00
void NXTTest::WaitABit() {
device.Tick();
FlushWire();
2017-07-17 21:13:57 +00:00
utils::USleep(100);
}
2017-07-28 01:30:57 +00:00
void NXTTest::SwapBuffersForCapture() {
// Insert a frame boundary for API capture tools.
2018-07-18 11:43:49 +00:00
dawn::Texture backBuffer = swapchain.GetNextTexture();
2017-07-28 01:30:57 +00:00
swapchain.Present(backBuffer);
}
2017-06-27 04:11:16 +00:00
void NXTTest::FlushWire() {
if (gTestUsesWire) {
2018-06-21 01:54:18 +00:00
ASSERT(mC2sBuf->Flush());
ASSERT(mS2cBuf->Flush());
}
}
NXTTest::ReadbackReservation NXTTest::ReserveReadback(uint32_t readbackSize) {
// For now create a new MapRead buffer for each readback
// TODO(cwallez@chromium.org): eventually make bigger buffers and allocate linearly?
ReadbackSlot slot;
slot.bufferSize = readbackSize;
slot.buffer = device.CreateBufferBuilder()
.SetSize(readbackSize)
2018-07-18 11:43:49 +00:00
.SetAllowedUsage(dawn::BufferUsageBit::MapRead | dawn::BufferUsageBit::TransferDst)
.GetResult();
ReadbackReservation reservation;
reservation.buffer = slot.buffer.Clone();
2017-11-23 19:51:16 +00:00
reservation.slot = mReadbackSlots.size();
reservation.offset = 0;
2017-11-23 19:51:16 +00:00
mReadbackSlots.push_back(std::move(slot));
return reservation;
}
void NXTTest::MapSlotsSynchronously() {
// Initialize numPendingMapOperations before mapping, just in case the callback is called immediately.
2017-11-23 19:51:16 +00:00
mNumPendingMapOperations = mReadbackSlots.size();
// Map all readback slots
2017-11-23 19:51:16 +00:00
for (size_t i = 0; i < mReadbackSlots.size(); ++i) {
auto userdata = new MapReadUserdata{this, i};
2017-11-23 19:51:16 +00:00
auto& slot = mReadbackSlots[i];
2018-07-18 11:43:49 +00:00
slot.buffer.MapReadAsync(0, slot.bufferSize, SlotMapReadCallback, static_cast<dawn::CallbackUserdata>(reinterpret_cast<uintptr_t>(userdata)));
}
// Busy wait until all map operations are done.
2017-11-23 19:51:16 +00:00
while (mNumPendingMapOperations != 0) {
2017-07-17 21:13:57 +00:00
WaitABit();
}
}
// static
void NXTTest::SlotMapReadCallback(nxtBufferMapAsyncStatus status,
const void* data,
nxtCallbackUserdata userdata_) {
DAWN_ASSERT(status == NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS);
auto userdata = reinterpret_cast<MapReadUserdata*>(static_cast<uintptr_t>(userdata_));
2017-11-23 19:51:16 +00:00
userdata->test->mReadbackSlots[userdata->slot].mappedData = data;
userdata->test->mNumPendingMapOperations --;
delete userdata;
}
void NXTTest::ResolveExpectations() {
2017-11-23 19:51:16 +00:00
for (const auto& expectation : mDeferredExpectations) {
DAWN_ASSERT(mReadbackSlots[expectation.readbackSlot].mappedData != nullptr);
// Get a pointer to the mapped copy of the data for the expectation.
2017-11-23 19:51:16 +00:00
const char* data = reinterpret_cast<const char*>(mReadbackSlots[expectation.readbackSlot].mappedData);
data += expectation.readbackOffset;
uint32_t size;
std::vector<char> packedData;
if (expectation.rowBytes != expectation.rowPitch) {
DAWN_ASSERT(expectation.rowPitch > expectation.rowBytes);
uint32_t rowCount = (expectation.size + expectation.rowPitch - 1) / expectation.rowPitch;
uint32_t packedSize = rowCount * expectation.rowBytes;
packedData.resize(packedSize);
for (uint32_t r = 0; r < rowCount; ++r) {
for (uint32_t i = 0; i < expectation.rowBytes; ++i) {
packedData[i + r * expectation.rowBytes] = data[i + r * expectation.rowPitch];
}
}
data = packedData.data();
size = packedSize;
} else {
size = expectation.size;
}
// Get the result for the expectation and add context to failures
testing::AssertionResult result = expectation.expectation->Check(data, size);
if (!result) {
result << " Expectation created at " << expectation.file << ":" << expectation.line << std::endl;
result << expectation.message->str();
}
EXPECT_TRUE(result);
}
}
2017-06-27 04:11:16 +00:00
bool RGBA8::operator==(const RGBA8& other) const {
return r == other.r && g == other.g && b == other.b && a == other.a;
}
bool RGBA8::operator!=(const RGBA8& other) const {
return !(*this == other);
}
std::ostream& operator<< (std::ostream& stream, const RGBA8& color) {
return stream << "RGBA8(" <<
static_cast<int>(color.r) << ", " <<
static_cast<int>(color.g) << ", " <<
static_cast<int>(color.b) << ", " <<
static_cast<int>(color.a) << ")";
}
std::ostream &operator<<(std::ostream& stream, BackendType backend) {
return stream << ParamName(backend);
}
namespace detail {
bool IsBackendAvailable(BackendType type) {
switch (type) {
#if defined(DAWN_ENABLE_BACKEND_D3D12)
case D3D12Backend:
#endif
#if defined(DAWN_ENABLE_BACKEND_METAL)
case MetalBackend:
#endif
#if defined(DAWN_ENABLE_BACKEND_OPENGL)
case OpenGLBackend:
#endif
#if defined(DAWN_ENABLE_BACKEND_VULKAN)
case VulkanBackend:
#endif
return true;
default:
return false;
}
}
std::vector<BackendType> FilterBackends(const BackendType* types, size_t numParams) {
std::vector<BackendType> backends;
for (size_t i = 0; i < numParams; ++i) {
if (IsBackendAvailable(types[i])) {
backends.push_back(types[i]);
}
}
return backends;
}
// Helper classes to set expectations
template<typename T>
ExpectEq<T>::ExpectEq(T singleValue) {
2017-11-23 19:51:16 +00:00
mExpected.push_back(singleValue);
}
2017-07-04 14:53:42 +00:00
template<typename T>
ExpectEq<T>::ExpectEq(const T* values, const unsigned int count) {
2017-11-23 19:51:16 +00:00
mExpected.assign(values, values + count);
2017-07-04 14:53:42 +00:00
}
template<typename T>
testing::AssertionResult ExpectEq<T>::Check(const void* data, size_t size) {
DAWN_ASSERT(size == sizeof(T) * mExpected.size());
const T* actual = reinterpret_cast<const T*>(data);
testing::AssertionResult failure = testing::AssertionFailure();
2017-11-23 19:51:16 +00:00
for (size_t i = 0; i < mExpected.size(); ++i) {
if (actual[i] != mExpected[i]) {
testing::AssertionResult result = testing::AssertionFailure() << "Expected data[" << i << "] to be " << mExpected[i] << ", actual " << actual[i] << std::endl;
auto printBuffer = [&](const T* buffer) {
static constexpr unsigned int kBytes = sizeof(T);
2017-11-23 19:51:16 +00:00
for (size_t index = 0; index < mExpected.size(); ++index) {
auto byteView = reinterpret_cast<const uint8_t*>(buffer + index);
for (unsigned int b = 0; b < kBytes; ++b) {
char buf[4];
sprintf(buf, "%02X ", byteView[b]);
result << buf;
}
}
result << std::endl;
};
2017-11-23 19:51:16 +00:00
if (mExpected.size() <= 1024) {
result << "Expected:" << std::endl;
2017-11-23 19:51:16 +00:00
printBuffer(mExpected.data());
result << "Actual:" << std::endl;
printBuffer(actual);
}
return result;
}
}
return testing::AssertionSuccess();
}
template class ExpectEq<uint8_t>;
template class ExpectEq<uint32_t>;
2017-06-27 04:11:16 +00:00
template class ExpectEq<RGBA8>;
}