2018-07-18 09:40:26 +00:00
|
|
|
// Copyright 2017 The Dawn Authors
|
2017-04-20 18:38:20 +00:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2018-07-24 11:53:51 +00:00
|
|
|
#include "dawn_native/ShaderModule.h"
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2019-05-01 13:27:07 +00:00
|
|
|
#include "common/HashUtils.h"
|
2018-07-24 11:53:51 +00:00
|
|
|
#include "dawn_native/BindGroupLayout.h"
|
|
|
|
#include "dawn_native/Device.h"
|
|
|
|
#include "dawn_native/Pipeline.h"
|
|
|
|
#include "dawn_native/PipelineLayout.h"
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2018-09-06 15:25:46 +00:00
|
|
|
#include <spirv-tools/libspirv.hpp>
|
2019-09-04 08:47:14 +00:00
|
|
|
#include <spirv_cross.hpp>
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2019-04-11 14:52:55 +00:00
|
|
|
#include <sstream>
|
|
|
|
|
2018-07-24 14:45:45 +00:00
|
|
|
namespace dawn_native {
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2019-09-26 00:12:41 +00:00
|
|
|
namespace {
|
|
|
|
Format::Type SpirvCrossBaseTypeToFormatType(spirv_cross::SPIRType::BaseType spirvBaseType) {
|
|
|
|
switch (spirvBaseType) {
|
|
|
|
case spirv_cross::SPIRType::Float:
|
|
|
|
return Format::Float;
|
|
|
|
case spirv_cross::SPIRType::Int:
|
|
|
|
return Format::Sint;
|
|
|
|
case spirv_cross::SPIRType::UInt:
|
|
|
|
return Format::Uint;
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return Format::Other;
|
|
|
|
}
|
|
|
|
}
|
2019-11-22 16:40:22 +00:00
|
|
|
|
|
|
|
wgpu::TextureViewDimension SpirvDimToTextureViewDimension(spv::Dim dim, bool arrayed) {
|
|
|
|
switch (dim) {
|
|
|
|
case spv::Dim::Dim1D:
|
|
|
|
return wgpu::TextureViewDimension::e1D;
|
|
|
|
case spv::Dim::Dim2D:
|
|
|
|
if (arrayed) {
|
|
|
|
return wgpu::TextureViewDimension::e2DArray;
|
|
|
|
} else {
|
|
|
|
return wgpu::TextureViewDimension::e2D;
|
|
|
|
}
|
|
|
|
case spv::Dim::Dim3D:
|
|
|
|
return wgpu::TextureViewDimension::e3D;
|
|
|
|
case spv::Dim::DimCube:
|
|
|
|
if (arrayed) {
|
|
|
|
return wgpu::TextureViewDimension::CubeArray;
|
|
|
|
} else {
|
|
|
|
return wgpu::TextureViewDimension::Cube;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return wgpu::TextureViewDimension::Undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-14 16:29:41 +00:00
|
|
|
wgpu::TextureViewDimension ToWGPUTextureViewDimension(
|
|
|
|
shaderc_spvc_texture_view_dimension dim) {
|
|
|
|
switch (dim) {
|
|
|
|
case shaderc_spvc_texture_view_dimension_undefined:
|
|
|
|
return wgpu::TextureViewDimension::Undefined;
|
|
|
|
case shaderc_spvc_texture_view_dimension_e1D:
|
|
|
|
return wgpu::TextureViewDimension::e1D;
|
|
|
|
case shaderc_spvc_texture_view_dimension_e2D:
|
|
|
|
return wgpu::TextureViewDimension::e2D;
|
|
|
|
case shaderc_spvc_texture_view_dimension_e2D_array:
|
|
|
|
return wgpu::TextureViewDimension::e2DArray;
|
|
|
|
case shaderc_spvc_texture_view_dimension_cube:
|
|
|
|
return wgpu::TextureViewDimension::Cube;
|
|
|
|
case shaderc_spvc_texture_view_dimension_cube_array:
|
|
|
|
return wgpu::TextureViewDimension::CubeArray;
|
|
|
|
case shaderc_spvc_texture_view_dimension_e3D:
|
|
|
|
return wgpu::TextureViewDimension::e3D;
|
|
|
|
}
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
Format::Type ToDawnFormatType(shaderc_spvc_texture_format_type type) {
|
|
|
|
switch (type) {
|
|
|
|
case shaderc_spvc_texture_format_type_float:
|
|
|
|
return Format::Type::Float;
|
|
|
|
case shaderc_spvc_texture_format_type_sint:
|
|
|
|
return Format::Type::Sint;
|
|
|
|
case shaderc_spvc_texture_format_type_uint:
|
|
|
|
return Format::Type::Uint;
|
|
|
|
case shaderc_spvc_texture_format_type_other:
|
|
|
|
return Format::Type::Other;
|
|
|
|
}
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
wgpu::BindingType ToWGPUBindingType(shaderc_spvc_binding_type type) {
|
|
|
|
switch (type) {
|
|
|
|
case shaderc_spvc_binding_type_uniform_buffer:
|
|
|
|
return wgpu::BindingType::UniformBuffer;
|
|
|
|
case shaderc_spvc_binding_type_storage_buffer:
|
|
|
|
return wgpu::BindingType::StorageBuffer;
|
|
|
|
case shaderc_spvc_binding_type_readonly_storage_buffer:
|
|
|
|
return wgpu::BindingType::ReadonlyStorageBuffer;
|
|
|
|
case shaderc_spvc_binding_type_sampler:
|
|
|
|
return wgpu::BindingType::Sampler;
|
|
|
|
case shaderc_spvc_binding_type_sampled_texture:
|
|
|
|
return wgpu::BindingType::SampledTexture;
|
2020-03-17 10:28:07 +00:00
|
|
|
|
|
|
|
// TODO(jiawei.shao@intel.com): add convertion to read-only and write-only storage
|
|
|
|
// textures when they are supported as shaderc_spvc binding types.
|
2020-01-14 16:29:41 +00:00
|
|
|
case shaderc_spvc_binding_type_storage_texture:
|
|
|
|
return wgpu::BindingType::StorageTexture;
|
2020-03-17 18:12:57 +00:00
|
|
|
default:
|
|
|
|
// TODO(rharrison): Remove this case once I am done changing the
|
|
|
|
// values in shaderc_spvc_binding_type
|
|
|
|
UNREACHABLE();
|
|
|
|
return wgpu::BindingType::StorageTexture;
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
ResultOrError<SingleShaderStage> ToSingleShaderStage(
|
|
|
|
shaderc_spvc_execution_model execution_model) {
|
2020-01-14 16:29:41 +00:00
|
|
|
switch (execution_model) {
|
|
|
|
case shaderc_spvc_execution_model_vertex:
|
2020-01-15 14:56:22 +00:00
|
|
|
return {SingleShaderStage::Vertex};
|
2020-01-14 16:29:41 +00:00
|
|
|
case shaderc_spvc_execution_model_fragment:
|
2020-01-15 14:56:22 +00:00
|
|
|
return {SingleShaderStage::Fragment};
|
2020-01-14 16:29:41 +00:00
|
|
|
case shaderc_spvc_execution_model_glcompute:
|
2020-01-15 14:56:22 +00:00
|
|
|
return {SingleShaderStage::Compute};
|
2020-01-14 16:29:41 +00:00
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"Attempted to convert invalid spvc execution model to SingleShaderStage");
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
}
|
2020-03-17 10:28:07 +00:00
|
|
|
|
|
|
|
wgpu::TextureFormat ToWGPUTextureFormat(spv::ImageFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case spv::ImageFormatR8:
|
|
|
|
return wgpu::TextureFormat::R8Unorm;
|
|
|
|
case spv::ImageFormatR8Snorm:
|
|
|
|
return wgpu::TextureFormat::R8Snorm;
|
|
|
|
case spv::ImageFormatR8ui:
|
|
|
|
return wgpu::TextureFormat::R8Uint;
|
|
|
|
case spv::ImageFormatR8i:
|
|
|
|
return wgpu::TextureFormat::R8Sint;
|
|
|
|
case spv::ImageFormatR16ui:
|
|
|
|
return wgpu::TextureFormat::R16Uint;
|
|
|
|
case spv::ImageFormatR16i:
|
|
|
|
return wgpu::TextureFormat::R16Sint;
|
|
|
|
case spv::ImageFormatR16f:
|
|
|
|
return wgpu::TextureFormat::R16Float;
|
|
|
|
case spv::ImageFormatRg8:
|
|
|
|
return wgpu::TextureFormat::RG8Unorm;
|
|
|
|
case spv::ImageFormatRg8Snorm:
|
|
|
|
return wgpu::TextureFormat::RG8Snorm;
|
|
|
|
case spv::ImageFormatRg8ui:
|
|
|
|
return wgpu::TextureFormat::RG8Uint;
|
|
|
|
case spv::ImageFormatRg8i:
|
|
|
|
return wgpu::TextureFormat::RG8Sint;
|
|
|
|
case spv::ImageFormatR32f:
|
|
|
|
return wgpu::TextureFormat::R32Float;
|
|
|
|
case spv::ImageFormatR32ui:
|
|
|
|
return wgpu::TextureFormat::R32Uint;
|
|
|
|
case spv::ImageFormatR32i:
|
|
|
|
return wgpu::TextureFormat::R32Sint;
|
|
|
|
case spv::ImageFormatRg16ui:
|
|
|
|
return wgpu::TextureFormat::RG16Uint;
|
|
|
|
case spv::ImageFormatRg16i:
|
|
|
|
return wgpu::TextureFormat::RG16Sint;
|
|
|
|
case spv::ImageFormatRg16f:
|
|
|
|
return wgpu::TextureFormat::RG16Float;
|
|
|
|
case spv::ImageFormatRgba8:
|
|
|
|
return wgpu::TextureFormat::RGBA8Unorm;
|
|
|
|
case spv::ImageFormatRgba8Snorm:
|
|
|
|
return wgpu::TextureFormat::RGBA8Snorm;
|
|
|
|
case spv::ImageFormatRgba8ui:
|
|
|
|
return wgpu::TextureFormat::RGBA8Uint;
|
|
|
|
case spv::ImageFormatRgba8i:
|
|
|
|
return wgpu::TextureFormat::RGBA8Sint;
|
|
|
|
case spv::ImageFormatRgb10A2:
|
|
|
|
return wgpu::TextureFormat::RGB10A2Unorm;
|
|
|
|
case spv::ImageFormatR11fG11fB10f:
|
|
|
|
return wgpu::TextureFormat::RG11B10Float;
|
|
|
|
case spv::ImageFormatRg32f:
|
|
|
|
return wgpu::TextureFormat::RG32Float;
|
|
|
|
case spv::ImageFormatRg32ui:
|
|
|
|
return wgpu::TextureFormat::RG32Uint;
|
|
|
|
case spv::ImageFormatRg32i:
|
|
|
|
return wgpu::TextureFormat::RG32Sint;
|
|
|
|
case spv::ImageFormatRgba16ui:
|
|
|
|
return wgpu::TextureFormat::RGBA16Uint;
|
|
|
|
case spv::ImageFormatRgba16i:
|
|
|
|
return wgpu::TextureFormat::RGBA16Sint;
|
|
|
|
case spv::ImageFormatRgba16f:
|
|
|
|
return wgpu::TextureFormat::RGBA16Float;
|
|
|
|
case spv::ImageFormatRgba32f:
|
|
|
|
return wgpu::TextureFormat::RGBA32Float;
|
|
|
|
case spv::ImageFormatRgba32ui:
|
|
|
|
return wgpu::TextureFormat::RGBA32Uint;
|
|
|
|
case spv::ImageFormatRgba32i:
|
|
|
|
return wgpu::TextureFormat::RGBA32Sint;
|
|
|
|
default:
|
|
|
|
return wgpu::TextureFormat::Undefined;
|
|
|
|
}
|
|
|
|
}
|
2019-09-26 00:12:41 +00:00
|
|
|
} // anonymous namespace
|
|
|
|
|
2018-08-22 13:37:29 +00:00
|
|
|
MaybeError ValidateShaderModuleDescriptor(DeviceBase*,
|
|
|
|
const ShaderModuleDescriptor* descriptor) {
|
2018-09-10 14:17:24 +00:00
|
|
|
if (descriptor->nextInChain != nullptr) {
|
|
|
|
return DAWN_VALIDATION_ERROR("nextInChain must be nullptr");
|
|
|
|
}
|
2018-09-06 15:25:46 +00:00
|
|
|
|
2018-11-28 16:54:31 +00:00
|
|
|
spvtools::SpirvTools spirvTools(SPV_ENV_VULKAN_1_1);
|
2018-09-06 15:25:46 +00:00
|
|
|
|
|
|
|
std::ostringstream errorStream;
|
|
|
|
errorStream << "SPIRV Validation failure:" << std::endl;
|
|
|
|
|
|
|
|
spirvTools.SetMessageConsumer([&errorStream](spv_message_level_t level, const char*,
|
|
|
|
const spv_position_t& position,
|
|
|
|
const char* message) {
|
|
|
|
switch (level) {
|
|
|
|
case SPV_MSG_FATAL:
|
|
|
|
case SPV_MSG_INTERNAL_ERROR:
|
|
|
|
case SPV_MSG_ERROR:
|
|
|
|
errorStream << "error: line " << position.index << ": " << message << std::endl;
|
|
|
|
break;
|
|
|
|
case SPV_MSG_WARNING:
|
|
|
|
errorStream << "warning: line " << position.index << ": " << message
|
|
|
|
<< std::endl;
|
|
|
|
break;
|
|
|
|
case SPV_MSG_INFO:
|
|
|
|
errorStream << "info: line " << position.index << ": " << message << std::endl;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!spirvTools.Validate(descriptor->code, descriptor->codeSize)) {
|
2018-09-10 14:17:24 +00:00
|
|
|
return DAWN_VALIDATION_ERROR(errorStream.str().c_str());
|
2018-09-06 15:25:46 +00:00
|
|
|
}
|
|
|
|
|
2018-08-20 15:01:20 +00:00
|
|
|
return {};
|
2020-03-17 10:28:07 +00:00
|
|
|
} // namespace
|
2018-08-20 15:01:20 +00:00
|
|
|
|
|
|
|
// ShaderModuleBase
|
|
|
|
|
2019-10-30 00:20:03 +00:00
|
|
|
ShaderModuleBase::ShaderModuleBase(DeviceBase* device, const ShaderModuleDescriptor* descriptor)
|
|
|
|
: CachedObject(device), mCode(descriptor->code, descriptor->code + descriptor->codeSize) {
|
2019-09-26 00:12:41 +00:00
|
|
|
mFragmentOutputFormatBaseTypes.fill(Format::Other);
|
2020-02-07 16:30:27 +00:00
|
|
|
if (GetDevice()->IsToggleEnabled(Toggle::UseSpvcParser)) {
|
|
|
|
mSpvcContext.SetUseSpvcParser(true);
|
|
|
|
}
|
2017-07-26 16:35:35 +00:00
|
|
|
}
|
|
|
|
|
2019-02-13 13:09:18 +00:00
|
|
|
ShaderModuleBase::ShaderModuleBase(DeviceBase* device, ObjectBase::ErrorTag tag)
|
2019-10-30 00:20:03 +00:00
|
|
|
: CachedObject(device, tag) {
|
2019-02-13 13:09:18 +00:00
|
|
|
}
|
|
|
|
|
2019-05-01 13:27:07 +00:00
|
|
|
ShaderModuleBase::~ShaderModuleBase() {
|
2019-10-30 00:20:03 +00:00
|
|
|
if (IsCachedReference()) {
|
2019-05-01 13:27:07 +00:00
|
|
|
GetDevice()->UncacheShaderModule(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-13 13:09:18 +00:00
|
|
|
// static
|
|
|
|
ShaderModuleBase* ShaderModuleBase::MakeError(DeviceBase* device) {
|
|
|
|
return new ShaderModuleBase(device, ObjectBase::kError);
|
|
|
|
}
|
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
MaybeError ShaderModuleBase::ExtractSpirvInfo(const spirv_cross::Compiler& compiler) {
|
2019-02-13 13:09:18 +00:00
|
|
|
ASSERT(!IsError());
|
2020-01-14 16:29:41 +00:00
|
|
|
if (GetDevice()->IsToggleEnabled(Toggle::UseSpvc)) {
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(ExtractSpirvInfoWithSpvc());
|
2020-01-14 16:29:41 +00:00
|
|
|
} else {
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(ExtractSpirvInfoWithSpirvCross(compiler));
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
2020-01-15 14:56:22 +00:00
|
|
|
return {};
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
MaybeError ShaderModuleBase::ExtractSpirvInfoWithSpvc() {
|
2020-01-14 16:29:41 +00:00
|
|
|
shaderc_spvc_execution_model execution_model;
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(CheckSpvcSuccess(mSpvcContext.GetExecutionModel(&execution_model),
|
|
|
|
"Unable to get execution model for shader."));
|
2020-01-14 16:29:41 +00:00
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY_ASSIGN(mExecutionModel, ToSingleShaderStage(execution_model));
|
2020-01-14 16:29:41 +00:00
|
|
|
|
|
|
|
size_t push_constant_buffers_count;
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(
|
|
|
|
CheckSpvcSuccess(mSpvcContext.GetPushConstantBufferCount(&push_constant_buffers_count),
|
|
|
|
"Unable to get push constant buffer count for shader."));
|
2020-01-14 16:29:41 +00:00
|
|
|
|
|
|
|
// TODO(rharrison): This should be handled by spirv-val pass in spvc,
|
|
|
|
// but need to confirm.
|
|
|
|
if (push_constant_buffers_count > 0) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Push constants aren't supported.");
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Fill in bindingInfo with the SPIRV bindings
|
2020-01-15 14:56:22 +00:00
|
|
|
auto ExtractResourcesBinding =
|
|
|
|
[this](std::vector<shaderc_spvc_binding_info> bindings) -> MaybeError {
|
2020-01-14 16:29:41 +00:00
|
|
|
for (const auto& binding : bindings) {
|
|
|
|
if (binding.binding >= kMaxBindingsPerGroup || binding.set >= kMaxBindGroups) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Binding over limits in the SPIRV");
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BindingInfo* info = &mBindingInfo[binding.set][binding.binding];
|
|
|
|
*info = {};
|
|
|
|
info->used = true;
|
|
|
|
info->id = binding.id;
|
|
|
|
info->base_type_id = binding.base_type_id;
|
|
|
|
if (binding.binding_type == shaderc_spvc_binding_type_sampled_texture) {
|
|
|
|
info->multisampled = binding.multisampled;
|
|
|
|
info->textureDimension = ToWGPUTextureViewDimension(binding.texture_dimension);
|
|
|
|
info->textureComponentType = ToDawnFormatType(binding.texture_component_type);
|
|
|
|
}
|
|
|
|
info->type = ToWGPUBindingType(binding.binding_type);
|
|
|
|
}
|
2020-01-15 14:56:22 +00:00
|
|
|
return {};
|
2020-01-14 16:29:41 +00:00
|
|
|
};
|
|
|
|
|
2020-03-17 10:28:07 +00:00
|
|
|
// TODO(jiawei.shao@intel.com): extract binding information about storage textures.
|
2020-01-14 16:29:41 +00:00
|
|
|
std::vector<shaderc_spvc_binding_info> resource_bindings;
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(CheckSpvcSuccess(mSpvcContext.GetBindingInfo(
|
|
|
|
shaderc_spvc_shader_resource_uniform_buffers,
|
|
|
|
shaderc_spvc_binding_type_uniform_buffer, &resource_bindings),
|
|
|
|
"Unable to get binding info for uniform buffers from shader"));
|
|
|
|
DAWN_TRY(ExtractResourcesBinding(resource_bindings));
|
|
|
|
|
|
|
|
DAWN_TRY(CheckSpvcSuccess(
|
|
|
|
mSpvcContext.GetBindingInfo(shaderc_spvc_shader_resource_separate_images,
|
|
|
|
shaderc_spvc_binding_type_sampled_texture,
|
|
|
|
&resource_bindings),
|
|
|
|
"Unable to get binding info for sampled textures from shader"));
|
|
|
|
DAWN_TRY(ExtractResourcesBinding(resource_bindings));
|
|
|
|
|
|
|
|
DAWN_TRY(CheckSpvcSuccess(
|
|
|
|
mSpvcContext.GetBindingInfo(shaderc_spvc_shader_resource_separate_samplers,
|
|
|
|
shaderc_spvc_binding_type_sampler, &resource_bindings),
|
|
|
|
"Unable to get binding info for samples from shader"));
|
|
|
|
DAWN_TRY(ExtractResourcesBinding(resource_bindings));
|
|
|
|
|
|
|
|
DAWN_TRY(CheckSpvcSuccess(mSpvcContext.GetBindingInfo(
|
|
|
|
shaderc_spvc_shader_resource_storage_buffers,
|
|
|
|
shaderc_spvc_binding_type_storage_buffer, &resource_bindings),
|
|
|
|
"Unable to get binding info for storage buffers from shader"));
|
|
|
|
DAWN_TRY(ExtractResourcesBinding(resource_bindings));
|
2020-01-14 16:29:41 +00:00
|
|
|
|
|
|
|
std::vector<shaderc_spvc_resource_location_info> input_stage_locations;
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(CheckSpvcSuccess(mSpvcContext.GetInputStageLocationInfo(&input_stage_locations),
|
|
|
|
"Unable to get input stage location information from shader"));
|
2019-02-13 13:09:18 +00:00
|
|
|
|
2020-01-14 16:29:41 +00:00
|
|
|
for (const auto& input : input_stage_locations) {
|
|
|
|
if (mExecutionModel == SingleShaderStage::Vertex) {
|
|
|
|
if (input.location >= kMaxVertexAttributes) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Attribute location over limits in the SPIRV");
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
mUsedVertexAttributes.set(input.location);
|
|
|
|
} else if (mExecutionModel == SingleShaderStage::Fragment) {
|
|
|
|
// Without a location qualifier on vertex inputs, spirv_cross::CompilerMSL gives
|
|
|
|
// them all the location 0, causing a compile error.
|
|
|
|
if (!input.has_location) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Need location qualifier on fragment input");
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<shaderc_spvc_resource_location_info> output_stage_locations;
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(CheckSpvcSuccess(mSpvcContext.GetOutputStageLocationInfo(&output_stage_locations),
|
|
|
|
"Unable to get output stage location information from shader"));
|
2020-01-14 16:29:41 +00:00
|
|
|
|
|
|
|
for (const auto& output : output_stage_locations) {
|
|
|
|
if (mExecutionModel == SingleShaderStage::Vertex) {
|
|
|
|
// Without a location qualifier on vertex outputs, spirv_cross::CompilerMSL
|
|
|
|
// gives them all the location 0, causing a compile error.
|
|
|
|
if (!output.has_location) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Need location qualifier on vertex output");
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
} else if (mExecutionModel == SingleShaderStage::Fragment) {
|
|
|
|
if (output.location >= kMaxColorAttachments) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"Fragment output location over limits in the SPIRV");
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mExecutionModel == SingleShaderStage::Fragment) {
|
|
|
|
std::vector<shaderc_spvc_resource_type_info> output_types;
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(CheckSpvcSuccess(mSpvcContext.GetOutputStageTypeInfo(&output_types),
|
|
|
|
"Unable to get output stage type information from shader"));
|
2020-01-14 16:29:41 +00:00
|
|
|
|
|
|
|
for (const auto& output : output_types) {
|
2020-01-15 14:56:22 +00:00
|
|
|
if (output.type == shaderc_spvc_texture_format_type_other) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Unexpected Fragment output type");
|
|
|
|
}
|
2020-01-14 16:29:41 +00:00
|
|
|
mFragmentOutputFormatBaseTypes[output.location] = ToDawnFormatType(output.type);
|
|
|
|
}
|
|
|
|
}
|
2020-01-15 14:56:22 +00:00
|
|
|
return {};
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
MaybeError ShaderModuleBase::ExtractSpirvInfoWithSpirvCross(
|
|
|
|
const spirv_cross::Compiler& compiler) {
|
2019-04-01 21:48:38 +00:00
|
|
|
// TODO(cwallez@chromium.org): make errors here creation errors
|
2017-06-05 20:23:18 +00:00
|
|
|
// currently errors here do not prevent the shadermodule from being used
|
2017-04-20 18:38:20 +00:00
|
|
|
const auto& resources = compiler.get_shader_resources();
|
|
|
|
|
|
|
|
switch (compiler.get_execution_model()) {
|
|
|
|
case spv::ExecutionModelVertex:
|
2019-08-27 08:42:29 +00:00
|
|
|
mExecutionModel = SingleShaderStage::Vertex;
|
2017-04-20 18:38:20 +00:00
|
|
|
break;
|
|
|
|
case spv::ExecutionModelFragment:
|
2019-08-27 08:42:29 +00:00
|
|
|
mExecutionModel = SingleShaderStage::Fragment;
|
2017-04-20 18:38:20 +00:00
|
|
|
break;
|
|
|
|
case spv::ExecutionModelGLCompute:
|
2019-08-27 08:42:29 +00:00
|
|
|
mExecutionModel = SingleShaderStage::Compute;
|
2017-04-20 18:38:20 +00:00
|
|
|
break;
|
|
|
|
default:
|
2017-07-11 01:48:12 +00:00
|
|
|
UNREACHABLE();
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Unexpected shader execution model");
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (resources.push_constant_buffers.size() > 0) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Push constants aren't supported.");
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Fill in bindingInfo with the SPIRV bindings
|
2020-01-14 16:29:41 +00:00
|
|
|
auto ExtractResourcesBinding =
|
|
|
|
[this](const spirv_cross::SmallVector<spirv_cross::Resource>& resources,
|
2020-01-15 14:56:22 +00:00
|
|
|
const spirv_cross::Compiler& compiler,
|
|
|
|
wgpu::BindingType bindingType) -> MaybeError {
|
|
|
|
for (const auto& resource : resources) {
|
|
|
|
if (!compiler.get_decoration_bitset(resource.id).get(spv::DecorationBinding)) {
|
|
|
|
return DAWN_VALIDATION_ERROR("No Binding decoration set for resource");
|
|
|
|
}
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
if (!compiler.get_decoration_bitset(resource.id)
|
|
|
|
.get(spv::DecorationDescriptorSet)) {
|
|
|
|
return DAWN_VALIDATION_ERROR("No Descriptor Decoration set for resource");
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t binding = compiler.get_decoration(resource.id, spv::DecorationBinding);
|
|
|
|
uint32_t set = compiler.get_decoration(resource.id, spv::DecorationDescriptorSet);
|
|
|
|
|
|
|
|
if (binding >= kMaxBindingsPerGroup || set >= kMaxBindGroups) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Binding over limits in the SPIRV");
|
|
|
|
}
|
|
|
|
|
|
|
|
BindingInfo* info = &mBindingInfo[set][binding];
|
|
|
|
*info = {};
|
|
|
|
info->used = true;
|
|
|
|
info->id = resource.id;
|
|
|
|
info->base_type_id = resource.base_type_id;
|
|
|
|
switch (bindingType) {
|
|
|
|
case wgpu::BindingType::SampledTexture: {
|
|
|
|
spirv_cross::SPIRType::ImageType imageType =
|
|
|
|
compiler.get_type(info->base_type_id).image;
|
|
|
|
spirv_cross::SPIRType::BaseType textureComponentType =
|
|
|
|
compiler.get_type(imageType.type).basetype;
|
|
|
|
|
|
|
|
info->multisampled = imageType.ms;
|
|
|
|
info->textureDimension =
|
|
|
|
SpirvDimToTextureViewDimension(imageType.dim, imageType.arrayed);
|
|
|
|
info->textureComponentType =
|
|
|
|
SpirvCrossBaseTypeToFormatType(textureComponentType);
|
|
|
|
info->type = bindingType;
|
|
|
|
} break;
|
|
|
|
case wgpu::BindingType::StorageBuffer: {
|
|
|
|
// Differentiate between readonly storage bindings and writable ones
|
|
|
|
// based on the NonWritable decoration
|
|
|
|
spirv_cross::Bitset flags = compiler.get_buffer_block_flags(resource.id);
|
|
|
|
if (flags.get(spv::DecorationNonWritable)) {
|
|
|
|
info->type = wgpu::BindingType::ReadonlyStorageBuffer;
|
|
|
|
} else {
|
|
|
|
info->type = wgpu::BindingType::StorageBuffer;
|
|
|
|
}
|
|
|
|
} break;
|
2020-03-11 01:28:48 +00:00
|
|
|
case wgpu::BindingType::StorageTexture: {
|
|
|
|
spirv_cross::Bitset flags = compiler.get_decoration_bitset(resource.id);
|
|
|
|
if (flags.get(spv::DecorationNonReadable)) {
|
|
|
|
info->type = wgpu::BindingType::WriteonlyStorageTexture;
|
|
|
|
} else if (flags.get(spv::DecorationNonWritable)) {
|
|
|
|
info->type = wgpu::BindingType::ReadonlyStorageTexture;
|
|
|
|
} else {
|
|
|
|
info->type = wgpu::BindingType::StorageTexture;
|
|
|
|
}
|
2020-03-17 10:28:07 +00:00
|
|
|
|
|
|
|
spirv_cross::SPIRType::ImageType imageType =
|
|
|
|
compiler.get_type(info->base_type_id).image;
|
|
|
|
wgpu::TextureFormat storageTextureFormat =
|
|
|
|
ToWGPUTextureFormat(imageType.format);
|
|
|
|
if (storageTextureFormat == wgpu::TextureFormat::Undefined) {
|
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"Invalid image format declaration on storage image");
|
|
|
|
}
|
|
|
|
const Format& format =
|
|
|
|
GetDevice()->GetValidInternalFormat(storageTextureFormat);
|
|
|
|
if (!format.supportsStorageUsage) {
|
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"The storage texture format is not supported");
|
|
|
|
}
|
|
|
|
info->storageTextureFormat = storageTextureFormat;
|
2020-03-11 01:28:48 +00:00
|
|
|
} break;
|
2020-01-15 14:56:22 +00:00
|
|
|
default:
|
|
|
|
info->type = bindingType;
|
2019-10-31 09:51:11 +00:00
|
|
|
}
|
2020-01-15 14:56:22 +00:00
|
|
|
}
|
|
|
|
return {};
|
|
|
|
};
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_TRY(ExtractResourcesBinding(resources.uniform_buffers, compiler,
|
|
|
|
wgpu::BindingType::UniformBuffer));
|
|
|
|
DAWN_TRY(ExtractResourcesBinding(resources.separate_images, compiler,
|
|
|
|
wgpu::BindingType::SampledTexture));
|
|
|
|
DAWN_TRY(ExtractResourcesBinding(resources.separate_samplers, compiler,
|
|
|
|
wgpu::BindingType::Sampler));
|
|
|
|
DAWN_TRY(ExtractResourcesBinding(resources.storage_buffers, compiler,
|
|
|
|
wgpu::BindingType::StorageBuffer));
|
2020-03-11 01:28:48 +00:00
|
|
|
DAWN_TRY(ExtractResourcesBinding(resources.storage_images, compiler,
|
|
|
|
wgpu::BindingType::StorageTexture));
|
2017-04-20 18:38:20 +00:00
|
|
|
|
|
|
|
// Extract the vertex attributes
|
2019-08-27 08:42:29 +00:00
|
|
|
if (mExecutionModel == SingleShaderStage::Vertex) {
|
2017-04-20 18:38:20 +00:00
|
|
|
for (const auto& attrib : resources.stage_inputs) {
|
2020-01-15 14:56:22 +00:00
|
|
|
if (!(compiler.get_decoration_bitset(attrib.id).get(spv::DecorationLocation))) {
|
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"Unable to find Location decoration for Vertex input");
|
|
|
|
}
|
2017-04-20 18:38:20 +00:00
|
|
|
uint32_t location = compiler.get_decoration(attrib.id, spv::DecorationLocation);
|
|
|
|
|
|
|
|
if (location >= kMaxVertexAttributes) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Attribute location over limits in the SPIRV");
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
2017-11-23 18:32:51 +00:00
|
|
|
mUsedVertexAttributes.set(location);
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
2020-01-14 16:29:41 +00:00
|
|
|
// Without a location qualifier on vertex outputs, spirv_cross::CompilerMSL gives
|
|
|
|
// them all the location 0, causing a compile error.
|
2017-04-20 18:38:20 +00:00
|
|
|
for (const auto& attrib : resources.stage_outputs) {
|
2018-08-24 12:51:21 +00:00
|
|
|
if (!compiler.get_decoration_bitset(attrib.id).get(spv::DecorationLocation)) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Need location qualifier on vertex output");
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-27 08:42:29 +00:00
|
|
|
if (mExecutionModel == SingleShaderStage::Fragment) {
|
2020-01-14 16:29:41 +00:00
|
|
|
// Without a location qualifier on vertex inputs, spirv_cross::CompilerMSL gives
|
|
|
|
// them all the location 0, causing a compile error.
|
2017-04-20 18:38:20 +00:00
|
|
|
for (const auto& attrib : resources.stage_inputs) {
|
2018-08-24 12:51:21 +00:00
|
|
|
if (!compiler.get_decoration_bitset(attrib.id).get(spv::DecorationLocation)) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR("Need location qualifier on fragment input");
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
}
|
2019-09-25 01:21:28 +00:00
|
|
|
|
|
|
|
for (const auto& fragmentOutput : resources.stage_outputs) {
|
2020-01-15 14:56:22 +00:00
|
|
|
if (!compiler.get_decoration_bitset(fragmentOutput.id)
|
|
|
|
.get(spv::DecorationLocation)) {
|
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"Unable to find Location decoration for Fragment output");
|
|
|
|
}
|
2019-09-25 01:21:28 +00:00
|
|
|
uint32_t location =
|
|
|
|
compiler.get_decoration(fragmentOutput.id, spv::DecorationLocation);
|
|
|
|
if (location >= kMaxColorAttachments) {
|
2020-01-15 14:56:22 +00:00
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"Fragment output location over limits in the SPIRV");
|
2019-09-25 01:21:28 +00:00
|
|
|
}
|
2019-09-26 00:12:41 +00:00
|
|
|
|
|
|
|
spirv_cross::SPIRType::BaseType shaderFragmentOutputBaseType =
|
|
|
|
compiler.get_type(fragmentOutput.base_type_id).basetype;
|
|
|
|
Format::Type formatType =
|
|
|
|
SpirvCrossBaseTypeToFormatType(shaderFragmentOutputBaseType);
|
2020-01-15 14:56:22 +00:00
|
|
|
if (formatType == Format::Type::Other) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Unexpected Fragment output type");
|
|
|
|
};
|
2019-09-26 00:12:41 +00:00
|
|
|
mFragmentOutputFormatBaseTypes[location] = formatType;
|
2019-09-25 01:21:28 +00:00
|
|
|
}
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
2020-01-15 14:56:22 +00:00
|
|
|
return {};
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const ShaderModuleBase::ModuleBindingInfo& ShaderModuleBase::GetBindingInfo() const {
|
2019-02-13 13:09:18 +00:00
|
|
|
ASSERT(!IsError());
|
2017-11-23 18:32:51 +00:00
|
|
|
return mBindingInfo;
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const std::bitset<kMaxVertexAttributes>& ShaderModuleBase::GetUsedVertexAttributes() const {
|
2019-02-13 13:09:18 +00:00
|
|
|
ASSERT(!IsError());
|
2017-11-23 18:32:51 +00:00
|
|
|
return mUsedVertexAttributes;
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
2019-09-26 00:12:41 +00:00
|
|
|
const ShaderModuleBase::FragmentOutputBaseTypes& ShaderModuleBase::GetFragmentOutputBaseTypes()
|
|
|
|
const {
|
|
|
|
ASSERT(!IsError());
|
|
|
|
return mFragmentOutputFormatBaseTypes;
|
|
|
|
}
|
|
|
|
|
2019-08-27 08:42:29 +00:00
|
|
|
SingleShaderStage ShaderModuleBase::GetExecutionModel() const {
|
2019-02-13 13:09:18 +00:00
|
|
|
ASSERT(!IsError());
|
2017-11-23 18:32:51 +00:00
|
|
|
return mExecutionModel;
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
2019-11-22 17:02:22 +00:00
|
|
|
bool ShaderModuleBase::IsCompatibleWithPipelineLayout(const PipelineLayoutBase* layout) const {
|
2019-02-13 13:09:18 +00:00
|
|
|
ASSERT(!IsError());
|
|
|
|
|
2019-02-11 23:35:33 +00:00
|
|
|
for (uint32_t group : IterateBitSet(layout->GetBindGroupLayoutsMask())) {
|
2017-04-20 18:38:20 +00:00
|
|
|
if (!IsCompatibleWithBindGroupLayout(group, layout->GetBindGroupLayout(group))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2019-02-11 23:35:33 +00:00
|
|
|
|
|
|
|
for (uint32_t group : IterateBitSet(~layout->GetBindGroupLayoutsMask())) {
|
|
|
|
for (size_t i = 0; i < kMaxBindingsPerGroup; ++i) {
|
|
|
|
if (mBindingInfo[group][i].used) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-20 18:38:20 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-11-22 17:02:22 +00:00
|
|
|
bool ShaderModuleBase::IsCompatibleWithBindGroupLayout(
|
|
|
|
size_t group,
|
|
|
|
const BindGroupLayoutBase* layout) const {
|
2019-02-13 13:09:18 +00:00
|
|
|
ASSERT(!IsError());
|
|
|
|
|
2017-04-20 18:38:20 +00:00
|
|
|
const auto& layoutInfo = layout->GetBindingInfo();
|
|
|
|
for (size_t i = 0; i < kMaxBindingsPerGroup; ++i) {
|
2017-11-23 18:32:51 +00:00
|
|
|
const auto& moduleInfo = mBindingInfo[group][i];
|
2019-05-17 02:05:37 +00:00
|
|
|
const auto& layoutBindingType = layoutInfo.types[i];
|
2017-04-20 18:38:20 +00:00
|
|
|
|
|
|
|
if (!moduleInfo.used) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-07-09 07:58:57 +00:00
|
|
|
if (layoutBindingType != moduleInfo.type) {
|
2019-11-11 20:05:28 +00:00
|
|
|
// Binding mismatch between shader and bind group is invalid. For example, a
|
|
|
|
// writable binding in the shader with a readonly storage buffer in the bind group
|
|
|
|
// layout is invalid. However, a readonly binding in the shader with a writable
|
|
|
|
// storage buffer in the bind group layout is valid.
|
|
|
|
bool validBindingConversion =
|
|
|
|
layoutBindingType == wgpu::BindingType::StorageBuffer &&
|
|
|
|
moduleInfo.type == wgpu::BindingType::ReadonlyStorageBuffer;
|
|
|
|
if (!validBindingConversion) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
2019-05-17 02:05:37 +00:00
|
|
|
|
2017-11-23 18:32:51 +00:00
|
|
|
if ((layoutInfo.visibilities[i] & StageBit(mExecutionModel)) == 0) {
|
2017-04-20 18:38:20 +00:00
|
|
|
return false;
|
|
|
|
}
|
2019-10-31 09:51:11 +00:00
|
|
|
|
|
|
|
if (layoutBindingType == wgpu::BindingType::SampledTexture) {
|
|
|
|
Format::Type layoutTextureComponentType =
|
|
|
|
Format::TextureComponentTypeToFormatType(layoutInfo.textureComponentTypes[i]);
|
|
|
|
if (layoutTextureComponentType != moduleInfo.textureComponentType) {
|
|
|
|
return false;
|
|
|
|
}
|
2019-11-22 16:40:22 +00:00
|
|
|
|
|
|
|
if (layoutInfo.textureDimensions[i] != moduleInfo.textureDimension) {
|
|
|
|
return false;
|
|
|
|
}
|
2019-10-31 09:51:11 +00:00
|
|
|
}
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-05-01 13:27:07 +00:00
|
|
|
size_t ShaderModuleBase::HashFunc::operator()(const ShaderModuleBase* module) const {
|
|
|
|
size_t hash = 0;
|
|
|
|
|
|
|
|
for (uint32_t word : module->mCode) {
|
|
|
|
HashCombine(&hash, word);
|
|
|
|
}
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ShaderModuleBase::EqualityFunc::operator()(const ShaderModuleBase* a,
|
|
|
|
const ShaderModuleBase* b) const {
|
|
|
|
return a->mCode == b->mCode;
|
|
|
|
}
|
|
|
|
|
2020-01-15 14:56:22 +00:00
|
|
|
MaybeError ShaderModuleBase::CheckSpvcSuccess(shaderc_spvc_status status,
|
|
|
|
const char* error_msg) {
|
2020-01-14 16:29:41 +00:00
|
|
|
if (status != shaderc_spvc_status_success) {
|
2020-01-15 14:56:22 +00:00
|
|
|
DAWN_VALIDATION_ERROR(error_msg);
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
2020-01-15 14:56:22 +00:00
|
|
|
return {};
|
2020-01-14 16:29:41 +00:00
|
|
|
}
|
|
|
|
|
2020-02-06 17:26:46 +00:00
|
|
|
shaderc_spvc::CompileOptions ShaderModuleBase::GetCompileOptions() {
|
|
|
|
shaderc_spvc::CompileOptions options;
|
|
|
|
options.SetValidate(GetDevice()->IsValidationEnabled());
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
|
2018-07-24 14:45:45 +00:00
|
|
|
} // namespace dawn_native
|