blob: aef027270d0b8d87a4dc21b376d2ffddb390d792 [file] [log] [blame]
// Copyright 2025 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// One-off "spot"/regression/smoke tests for Emdawnwebgpu.
#include <dawn/webgpu_cpp_print.h>
#include <emscripten.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <webgpu/webgpu_cpp.h>
#include <array>
#include <string>
#include <utility>
namespace {
using testing::_;
using testing::HasSubstr;
class SpotTests : public testing::Test {
public:
void SetUp() override {
static constexpr auto kInstanceFeatures =
std::array{wgpu::InstanceFeatureName::TimedWaitAny};
wgpu::InstanceDescriptor instanceDesc{.requiredFeatureCount = kInstanceFeatures.size(),
.requiredFeatures = kInstanceFeatures.data()};
instance = wgpu::CreateInstance(&instanceDesc);
wgpu::Adapter adapter;
EXPECT_EQ(wgpu::WaitStatus::Success,
instance.WaitAny(instance.RequestAdapter(
nullptr, wgpu::CallbackMode::WaitAnyOnly,
[&adapter](wgpu::RequestAdapterStatus, wgpu::Adapter a,
wgpu::StringView) { adapter = std::move(a); }),
UINT64_MAX));
EXPECT_TRUE(adapter);
wgpu::SupportedFeatures features;
adapter.GetFeatures(&features);
wgpu::DeviceDescriptor deviceDesc;
// Enable all available features
deviceDesc.requiredFeatureCount = features.featureCount;
deviceDesc.requiredFeatures = features.features;
wgpu::Device device;
EXPECT_EQ(wgpu::WaitStatus::Success,
instance.WaitAny(
adapter.RequestDevice(&deviceDesc, wgpu::CallbackMode::WaitAnyOnly,
[&device](wgpu::RequestDeviceStatus, wgpu::Device d,
wgpu::StringView) { device = std::move(d); }),
UINT64_MAX));
EXPECT_TRUE(device);
this->adapter = adapter;
this->device = device;
}
protected:
wgpu::Instance instance;
wgpu::Adapter adapter;
wgpu::Device device;
};
TEST_F(SpotTests, QuerySet) {
// Spot test wgpuQuerySetGetType which uses indexOf on an int-to-string table.
wgpu::QuerySetDescriptor querySetDesc{.type = wgpu::QueryType::Timestamp, .count = 1};
wgpu::QuerySet querySet = device.CreateQuerySet(&querySetDesc);
EXPECT_TRUE(querySet);
EXPECT_EQ(querySet.GetType(), querySetDesc.type);
}
TEST_F(SpotTests, BufferGetMapState) {
// Spot test one of the string-to-int tables (Int_BufferMapState) to make sure
// that Closure's minification didn't minify its keys.
wgpu::BufferDescriptor bufferDesc{.usage = wgpu::BufferUsage::CopyDst, .size = 4};
wgpu::Buffer buffer = device.CreateBuffer(&bufferDesc);
EXPECT_EQ(buffer.GetMapState(), wgpu::BufferMapState::Unmapped);
}
TEST_F(SpotTests, GetCompilationInfo) {
for (bool valid : {true, false}) {
wgpu::ShaderSourceWGSL wgslDesc{};
wgslDesc.code = valid ? "" : "some invalid code";
wgpu::ShaderModuleDescriptor descriptor{};
descriptor.nextInChain = &wgslDesc;
auto sm = device.CreateShaderModule(&descriptor);
auto future = sm.GetCompilationInfo(
wgpu::CallbackMode::WaitAnyOnly,
[](wgpu::CompilationInfoRequestStatus, const wgpu::CompilationInfo* compilationInfo) {
// We shouldn't have tried to allocate stuff if there were no messages.
EXPECT_EQ(compilationInfo->messageCount == 0, compilationInfo->messages == nullptr);
// After this, any compilation info will be freed. (There was a bug here which
// this test catches, but only in ASAN builds.)
});
EXPECT_EQ(wgpu::WaitStatus::Success, instance.WaitAny(future, UINT64_MAX));
}
}
TEST_F(SpotTests, ExternalRefCount) {
wgpu::BufferDescriptor bufferDesc{
.usage = wgpu::BufferUsage::MapRead, .size = 16, .mappedAtCreation = true};
wgpu::Buffer buffer = device.CreateBuffer(&bufferDesc);
ASSERT_TRUE(buffer);
EXPECT_EQ(buffer.GetMapState(), wgpu::BufferMapState::Mapped);
{
// Add and then release an extra external ref.
wgpu::Buffer tmp = buffer;
}
// Make sure the device wasn't implicitly destroyed (because we thought
// the last external ref was dropped).
EXPECT_EQ(buffer.GetMapState(), wgpu::BufferMapState::Mapped);
}
template <typename T>
void TestGetFeatures(T o) { // o is either wgpu::Adapter or wgpu::Device.
wgpu::SupportedFeatures f;
o.GetFeatures(&f);
auto features = std::span(f.features, f.featureCount);
for (auto feature : features) {
// GetFeatures should filter out any unknown features.
EXPECT_NE(feature, wgpu::FeatureName{0});
EXPECT_TRUE(o.HasFeature(feature));
}
// Test some specific features to make sure minification worked.
bool haveCompressedTexture = false;
if (EM_ASM_INT(
{ return WebGPU.getJsObject($0).features.has('texture-compression-bc'); }, o.Get())) {
auto feature = wgpu::FeatureName::TextureCompressionBC;
EXPECT_NE(std::find(features.begin(), features.end(), feature), features.end());
EXPECT_TRUE(o.HasFeature(feature));
haveCompressedTexture = true;
}
if (EM_ASM_INT(
{ return WebGPU.getJsObject($0).features.has('texture-compression-etc2'); }, o.Get())) {
auto feature = wgpu::FeatureName::TextureCompressionETC2;
EXPECT_NE(std::find(features.begin(), features.end(), feature), features.end());
EXPECT_TRUE(o.HasFeature(feature));
haveCompressedTexture = true;
}
EXPECT_TRUE(haveCompressedTexture);
// "subgroups" is a valid JS identifier (no hyphens), so it's
// vulnerable to Closure minification.
if (EM_ASM_INT({ return WebGPU.getJsObject($0).features.has('subgroups'); }, o.Get())) {
auto feature = wgpu::FeatureName::Subgroups;
EXPECT_NE(std::find(features.begin(), features.end(), feature), features.end());
EXPECT_TRUE(o.HasFeature(feature));
}
}
// Test GetFeatures and HasFeature enum lookups.
TEST_F(SpotTests, GetFeatures) {
TestGetFeatures(adapter);
TestGetFeatures(device);
}
TEST_F(SpotTests, GetWGSLLanguageFeatures) {
wgpu::SupportedWGSLLanguageFeatures f;
instance.GetWGSLLanguageFeatures(&f);
auto features = std::span(f.features, f.featureCount);
for (auto feature : features) {
// GetWGSLLanguageFeatures should filter out any unknown features.
EXPECT_NE(feature, wgpu::WGSLLanguageFeatureName{0});
EXPECT_TRUE(instance.HasWGSLLanguageFeature(feature));
}
// Test a specific feature to make sure minification worked.
// WGSL feature names are valid JS identifiers (they use underscores instead
// of hyphens), so they're vulnerable to Closure minification.
if (EM_ASM_INT({
return navigator.gpu.wgslLanguageFeatures.has('unrestricted_pointer_parameters');
})) {
auto feature = wgpu::WGSLLanguageFeatureName::UnrestrictedPointerParameters;
EXPECT_NE(std::find(features.begin(), features.end(), feature), features.end());
EXPECT_TRUE(instance.HasWGSLLanguageFeature(feature));
}
}
} // namespace