mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-05-13 10:51:35 +00:00
Fixed Kokoro failures when building Tint with all readers and writers disabled. Change-Id: I9bd041339236488dd563bbe687da883035f2a4b7 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/86181 Reviewed-by: Corentin Wallez <cwallez@chromium.org> Commit-Queue: Antonio Maiorano <amaiorano@google.com>
1345 lines
68 KiB
C++
1345 lines
68 KiB
C++
// 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 "dawn/native/ShaderModule.h"
|
|
|
|
#include "absl/strings/str_format.h"
|
|
#include "dawn/common/BitSetIterator.h"
|
|
#include "dawn/common/Constants.h"
|
|
#include "dawn/common/HashUtils.h"
|
|
#include "dawn/native/BindGroupLayout.h"
|
|
#include "dawn/native/ChainUtils_autogen.h"
|
|
#include "dawn/native/CompilationMessages.h"
|
|
#include "dawn/native/Device.h"
|
|
#include "dawn/native/ObjectContentHasher.h"
|
|
#include "dawn/native/Pipeline.h"
|
|
#include "dawn/native/PipelineLayout.h"
|
|
#include "dawn/native/RenderPipeline.h"
|
|
#include "dawn/native/TintUtils.h"
|
|
|
|
#include <tint/tint.h>
|
|
|
|
#include <sstream>
|
|
|
|
namespace dawn::native {
|
|
|
|
namespace {
|
|
|
|
tint::transform::VertexFormat ToTintVertexFormat(wgpu::VertexFormat format) {
|
|
switch (format) {
|
|
case wgpu::VertexFormat::Uint8x2:
|
|
return tint::transform::VertexFormat::kUint8x2;
|
|
case wgpu::VertexFormat::Uint8x4:
|
|
return tint::transform::VertexFormat::kUint8x4;
|
|
case wgpu::VertexFormat::Sint8x2:
|
|
return tint::transform::VertexFormat::kSint8x2;
|
|
case wgpu::VertexFormat::Sint8x4:
|
|
return tint::transform::VertexFormat::kSint8x4;
|
|
case wgpu::VertexFormat::Unorm8x2:
|
|
return tint::transform::VertexFormat::kUnorm8x2;
|
|
case wgpu::VertexFormat::Unorm8x4:
|
|
return tint::transform::VertexFormat::kUnorm8x4;
|
|
case wgpu::VertexFormat::Snorm8x2:
|
|
return tint::transform::VertexFormat::kSnorm8x2;
|
|
case wgpu::VertexFormat::Snorm8x4:
|
|
return tint::transform::VertexFormat::kSnorm8x4;
|
|
case wgpu::VertexFormat::Uint16x2:
|
|
return tint::transform::VertexFormat::kUint16x2;
|
|
case wgpu::VertexFormat::Uint16x4:
|
|
return tint::transform::VertexFormat::kUint16x4;
|
|
case wgpu::VertexFormat::Sint16x2:
|
|
return tint::transform::VertexFormat::kSint16x2;
|
|
case wgpu::VertexFormat::Sint16x4:
|
|
return tint::transform::VertexFormat::kSint16x4;
|
|
case wgpu::VertexFormat::Unorm16x2:
|
|
return tint::transform::VertexFormat::kUnorm16x2;
|
|
case wgpu::VertexFormat::Unorm16x4:
|
|
return tint::transform::VertexFormat::kUnorm16x4;
|
|
case wgpu::VertexFormat::Snorm16x2:
|
|
return tint::transform::VertexFormat::kSnorm16x2;
|
|
case wgpu::VertexFormat::Snorm16x4:
|
|
return tint::transform::VertexFormat::kSnorm16x4;
|
|
case wgpu::VertexFormat::Float16x2:
|
|
return tint::transform::VertexFormat::kFloat16x2;
|
|
case wgpu::VertexFormat::Float16x4:
|
|
return tint::transform::VertexFormat::kFloat16x4;
|
|
case wgpu::VertexFormat::Float32:
|
|
return tint::transform::VertexFormat::kFloat32;
|
|
case wgpu::VertexFormat::Float32x2:
|
|
return tint::transform::VertexFormat::kFloat32x2;
|
|
case wgpu::VertexFormat::Float32x3:
|
|
return tint::transform::VertexFormat::kFloat32x3;
|
|
case wgpu::VertexFormat::Float32x4:
|
|
return tint::transform::VertexFormat::kFloat32x4;
|
|
case wgpu::VertexFormat::Uint32:
|
|
return tint::transform::VertexFormat::kUint32;
|
|
case wgpu::VertexFormat::Uint32x2:
|
|
return tint::transform::VertexFormat::kUint32x2;
|
|
case wgpu::VertexFormat::Uint32x3:
|
|
return tint::transform::VertexFormat::kUint32x3;
|
|
case wgpu::VertexFormat::Uint32x4:
|
|
return tint::transform::VertexFormat::kUint32x4;
|
|
case wgpu::VertexFormat::Sint32:
|
|
return tint::transform::VertexFormat::kSint32;
|
|
case wgpu::VertexFormat::Sint32x2:
|
|
return tint::transform::VertexFormat::kSint32x2;
|
|
case wgpu::VertexFormat::Sint32x3:
|
|
return tint::transform::VertexFormat::kSint32x3;
|
|
case wgpu::VertexFormat::Sint32x4:
|
|
return tint::transform::VertexFormat::kSint32x4;
|
|
|
|
case wgpu::VertexFormat::Undefined:
|
|
break;
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
tint::transform::VertexStepMode ToTintVertexStepMode(wgpu::VertexStepMode mode) {
|
|
switch (mode) {
|
|
case wgpu::VertexStepMode::Vertex:
|
|
return tint::transform::VertexStepMode::kVertex;
|
|
case wgpu::VertexStepMode::Instance:
|
|
return tint::transform::VertexStepMode::kInstance;
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<SingleShaderStage> TintPipelineStageToShaderStage(
|
|
tint::ast::PipelineStage stage) {
|
|
switch (stage) {
|
|
case tint::ast::PipelineStage::kVertex:
|
|
return SingleShaderStage::Vertex;
|
|
case tint::ast::PipelineStage::kFragment:
|
|
return SingleShaderStage::Fragment;
|
|
case tint::ast::PipelineStage::kCompute:
|
|
return SingleShaderStage::Compute;
|
|
case tint::ast::PipelineStage::kNone:
|
|
break;
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
BindingInfoType TintResourceTypeToBindingInfoType(
|
|
tint::inspector::ResourceBinding::ResourceType type) {
|
|
switch (type) {
|
|
case tint::inspector::ResourceBinding::ResourceType::kUniformBuffer:
|
|
case tint::inspector::ResourceBinding::ResourceType::kStorageBuffer:
|
|
case tint::inspector::ResourceBinding::ResourceType::kReadOnlyStorageBuffer:
|
|
return BindingInfoType::Buffer;
|
|
case tint::inspector::ResourceBinding::ResourceType::kSampler:
|
|
case tint::inspector::ResourceBinding::ResourceType::kComparisonSampler:
|
|
return BindingInfoType::Sampler;
|
|
case tint::inspector::ResourceBinding::ResourceType::kSampledTexture:
|
|
case tint::inspector::ResourceBinding::ResourceType::kMultisampledTexture:
|
|
case tint::inspector::ResourceBinding::ResourceType::kDepthTexture:
|
|
case tint::inspector::ResourceBinding::ResourceType::kDepthMultisampledTexture:
|
|
return BindingInfoType::Texture;
|
|
case tint::inspector::ResourceBinding::ResourceType::kWriteOnlyStorageTexture:
|
|
return BindingInfoType::StorageTexture;
|
|
case tint::inspector::ResourceBinding::ResourceType::kExternalTexture:
|
|
return BindingInfoType::ExternalTexture;
|
|
|
|
default:
|
|
UNREACHABLE();
|
|
return BindingInfoType::Buffer;
|
|
}
|
|
}
|
|
|
|
wgpu::TextureFormat TintImageFormatToTextureFormat(
|
|
tint::inspector::ResourceBinding::TexelFormat format) {
|
|
switch (format) {
|
|
case tint::inspector::ResourceBinding::TexelFormat::kR32Uint:
|
|
return wgpu::TextureFormat::R32Uint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kR32Sint:
|
|
return wgpu::TextureFormat::R32Sint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kR32Float:
|
|
return wgpu::TextureFormat::R32Float;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba8Unorm:
|
|
return wgpu::TextureFormat::RGBA8Unorm;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba8Snorm:
|
|
return wgpu::TextureFormat::RGBA8Snorm;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba8Uint:
|
|
return wgpu::TextureFormat::RGBA8Uint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba8Sint:
|
|
return wgpu::TextureFormat::RGBA8Sint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRg32Uint:
|
|
return wgpu::TextureFormat::RG32Uint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRg32Sint:
|
|
return wgpu::TextureFormat::RG32Sint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRg32Float:
|
|
return wgpu::TextureFormat::RG32Float;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba16Uint:
|
|
return wgpu::TextureFormat::RGBA16Uint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba16Sint:
|
|
return wgpu::TextureFormat::RGBA16Sint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba16Float:
|
|
return wgpu::TextureFormat::RGBA16Float;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba32Uint:
|
|
return wgpu::TextureFormat::RGBA32Uint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba32Sint:
|
|
return wgpu::TextureFormat::RGBA32Sint;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kRgba32Float:
|
|
return wgpu::TextureFormat::RGBA32Float;
|
|
case tint::inspector::ResourceBinding::TexelFormat::kNone:
|
|
return wgpu::TextureFormat::Undefined;
|
|
|
|
default:
|
|
UNREACHABLE();
|
|
return wgpu::TextureFormat::Undefined;
|
|
}
|
|
}
|
|
|
|
wgpu::TextureViewDimension TintTextureDimensionToTextureViewDimension(
|
|
tint::inspector::ResourceBinding::TextureDimension dim) {
|
|
switch (dim) {
|
|
case tint::inspector::ResourceBinding::TextureDimension::k1d:
|
|
return wgpu::TextureViewDimension::e1D;
|
|
case tint::inspector::ResourceBinding::TextureDimension::k2d:
|
|
return wgpu::TextureViewDimension::e2D;
|
|
case tint::inspector::ResourceBinding::TextureDimension::k2dArray:
|
|
return wgpu::TextureViewDimension::e2DArray;
|
|
case tint::inspector::ResourceBinding::TextureDimension::k3d:
|
|
return wgpu::TextureViewDimension::e3D;
|
|
case tint::inspector::ResourceBinding::TextureDimension::kCube:
|
|
return wgpu::TextureViewDimension::Cube;
|
|
case tint::inspector::ResourceBinding::TextureDimension::kCubeArray:
|
|
return wgpu::TextureViewDimension::CubeArray;
|
|
case tint::inspector::ResourceBinding::TextureDimension::kNone:
|
|
return wgpu::TextureViewDimension::Undefined;
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
SampleTypeBit TintSampledKindToSampleTypeBit(
|
|
tint::inspector::ResourceBinding::SampledKind s) {
|
|
switch (s) {
|
|
case tint::inspector::ResourceBinding::SampledKind::kSInt:
|
|
return SampleTypeBit::Sint;
|
|
case tint::inspector::ResourceBinding::SampledKind::kUInt:
|
|
return SampleTypeBit::Uint;
|
|
case tint::inspector::ResourceBinding::SampledKind::kFloat:
|
|
return SampleTypeBit::Float | SampleTypeBit::UnfilterableFloat;
|
|
case tint::inspector::ResourceBinding::SampledKind::kUnknown:
|
|
return SampleTypeBit::None;
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<wgpu::TextureComponentType> TintComponentTypeToTextureComponentType(
|
|
tint::inspector::ComponentType type) {
|
|
switch (type) {
|
|
case tint::inspector::ComponentType::kFloat:
|
|
return wgpu::TextureComponentType::Float;
|
|
case tint::inspector::ComponentType::kSInt:
|
|
return wgpu::TextureComponentType::Sint;
|
|
case tint::inspector::ComponentType::kUInt:
|
|
return wgpu::TextureComponentType::Uint;
|
|
case tint::inspector::ComponentType::kUnknown:
|
|
return DAWN_VALIDATION_ERROR(
|
|
"Attempted to convert 'Unknown' component type from Tint");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<VertexFormatBaseType> TintComponentTypeToVertexFormatBaseType(
|
|
tint::inspector::ComponentType type) {
|
|
switch (type) {
|
|
case tint::inspector::ComponentType::kFloat:
|
|
return VertexFormatBaseType::Float;
|
|
case tint::inspector::ComponentType::kSInt:
|
|
return VertexFormatBaseType::Sint;
|
|
case tint::inspector::ComponentType::kUInt:
|
|
return VertexFormatBaseType::Uint;
|
|
case tint::inspector::ComponentType::kUnknown:
|
|
return DAWN_VALIDATION_ERROR(
|
|
"Attempted to convert 'Unknown' component type from Tint");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<wgpu::BufferBindingType> TintResourceTypeToBufferBindingType(
|
|
tint::inspector::ResourceBinding::ResourceType resource_type) {
|
|
switch (resource_type) {
|
|
case tint::inspector::ResourceBinding::ResourceType::kUniformBuffer:
|
|
return wgpu::BufferBindingType::Uniform;
|
|
case tint::inspector::ResourceBinding::ResourceType::kStorageBuffer:
|
|
return wgpu::BufferBindingType::Storage;
|
|
case tint::inspector::ResourceBinding::ResourceType::kReadOnlyStorageBuffer:
|
|
return wgpu::BufferBindingType::ReadOnlyStorage;
|
|
default:
|
|
return DAWN_VALIDATION_ERROR("Attempted to convert non-buffer resource type");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<wgpu::StorageTextureAccess> TintResourceTypeToStorageTextureAccess(
|
|
tint::inspector::ResourceBinding::ResourceType resource_type) {
|
|
switch (resource_type) {
|
|
case tint::inspector::ResourceBinding::ResourceType::kWriteOnlyStorageTexture:
|
|
return wgpu::StorageTextureAccess::WriteOnly;
|
|
default:
|
|
return DAWN_VALIDATION_ERROR(
|
|
"Attempted to convert non-storage texture resource type");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<InterStageComponentType> TintComponentTypeToInterStageComponentType(
|
|
tint::inspector::ComponentType type) {
|
|
switch (type) {
|
|
case tint::inspector::ComponentType::kFloat:
|
|
return InterStageComponentType::Float;
|
|
case tint::inspector::ComponentType::kSInt:
|
|
return InterStageComponentType::Sint;
|
|
case tint::inspector::ComponentType::kUInt:
|
|
return InterStageComponentType::Uint;
|
|
case tint::inspector::ComponentType::kUnknown:
|
|
return DAWN_VALIDATION_ERROR(
|
|
"Attempted to convert 'Unknown' component type from Tint");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<uint32_t> TintCompositionTypeToInterStageComponentCount(
|
|
tint::inspector::CompositionType type) {
|
|
switch (type) {
|
|
case tint::inspector::CompositionType::kScalar:
|
|
return 1u;
|
|
case tint::inspector::CompositionType::kVec2:
|
|
return 2u;
|
|
case tint::inspector::CompositionType::kVec3:
|
|
return 3u;
|
|
case tint::inspector::CompositionType::kVec4:
|
|
return 4u;
|
|
case tint::inspector::CompositionType::kUnknown:
|
|
return DAWN_VALIDATION_ERROR(
|
|
"Attempt to convert 'Unknown' composition type from Tint");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<InterpolationType> TintInterpolationTypeToInterpolationType(
|
|
tint::inspector::InterpolationType type) {
|
|
switch (type) {
|
|
case tint::inspector::InterpolationType::kPerspective:
|
|
return InterpolationType::Perspective;
|
|
case tint::inspector::InterpolationType::kLinear:
|
|
return InterpolationType::Linear;
|
|
case tint::inspector::InterpolationType::kFlat:
|
|
return InterpolationType::Flat;
|
|
case tint::inspector::InterpolationType::kUnknown:
|
|
return DAWN_VALIDATION_ERROR(
|
|
"Attempted to convert 'Unknown' interpolation type from Tint");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<InterpolationSampling> TintInterpolationSamplingToInterpolationSamplingType(
|
|
tint::inspector::InterpolationSampling type) {
|
|
switch (type) {
|
|
case tint::inspector::InterpolationSampling::kNone:
|
|
return InterpolationSampling::None;
|
|
case tint::inspector::InterpolationSampling::kCenter:
|
|
return InterpolationSampling::Center;
|
|
case tint::inspector::InterpolationSampling::kCentroid:
|
|
return InterpolationSampling::Centroid;
|
|
case tint::inspector::InterpolationSampling::kSample:
|
|
return InterpolationSampling::Sample;
|
|
case tint::inspector::InterpolationSampling::kUnknown:
|
|
return DAWN_VALIDATION_ERROR(
|
|
"Attempted to convert 'Unknown' interpolation sampling type from Tint");
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
EntryPointMetadata::OverridableConstant::Type FromTintOverridableConstantType(
|
|
tint::inspector::OverridableConstant::Type type) {
|
|
switch (type) {
|
|
case tint::inspector::OverridableConstant::Type::kBool:
|
|
return EntryPointMetadata::OverridableConstant::Type::Boolean;
|
|
case tint::inspector::OverridableConstant::Type::kFloat32:
|
|
return EntryPointMetadata::OverridableConstant::Type::Float32;
|
|
case tint::inspector::OverridableConstant::Type::kInt32:
|
|
return EntryPointMetadata::OverridableConstant::Type::Int32;
|
|
case tint::inspector::OverridableConstant::Type::kUint32:
|
|
return EntryPointMetadata::OverridableConstant::Type::Uint32;
|
|
}
|
|
UNREACHABLE();
|
|
}
|
|
|
|
ResultOrError<tint::Program> ParseWGSL(const tint::Source::File* file,
|
|
OwnedCompilationMessages* outMessages) {
|
|
#if TINT_BUILD_WGSL_READER
|
|
tint::Program program = tint::reader::wgsl::Parse(file);
|
|
if (outMessages != nullptr) {
|
|
outMessages->AddMessages(program.Diagnostics());
|
|
}
|
|
if (!program.IsValid()) {
|
|
return DAWN_FORMAT_VALIDATION_ERROR(
|
|
"Tint WGSL reader failure:\nParser: %s\nShader:\n%s\n",
|
|
program.Diagnostics().str(), file->content.data);
|
|
}
|
|
|
|
return std::move(program);
|
|
#else
|
|
return DAWN_FORMAT_VALIDATION_ERROR("TINT_BUILD_WGSL_READER is not defined.");
|
|
#endif
|
|
}
|
|
|
|
ResultOrError<tint::Program> ParseSPIRV(const std::vector<uint32_t>& spirv,
|
|
OwnedCompilationMessages* outMessages) {
|
|
#if TINT_BUILD_SPV_READER
|
|
tint::Program program = tint::reader::spirv::Parse(spirv);
|
|
if (outMessages != nullptr) {
|
|
outMessages->AddMessages(program.Diagnostics());
|
|
}
|
|
if (!program.IsValid()) {
|
|
return DAWN_FORMAT_VALIDATION_ERROR("Tint SPIR-V reader failure:\nParser: %s\n",
|
|
program.Diagnostics().str());
|
|
}
|
|
|
|
return std::move(program);
|
|
#else
|
|
return DAWN_FORMAT_VALIDATION_ERROR("TINT_BUILD_SPV_READER is not defined.");
|
|
|
|
#endif
|
|
}
|
|
|
|
std::vector<uint64_t> GetBindGroupMinBufferSizes(const BindingGroupInfoMap& shaderBindings,
|
|
const BindGroupLayoutBase* layout) {
|
|
std::vector<uint64_t> requiredBufferSizes(layout->GetUnverifiedBufferCount());
|
|
uint32_t packedIdx = 0;
|
|
|
|
for (BindingIndex bindingIndex{0}; bindingIndex < layout->GetBufferCount();
|
|
++bindingIndex) {
|
|
const BindingInfo& bindingInfo = layout->GetBindingInfo(bindingIndex);
|
|
if (bindingInfo.buffer.minBindingSize != 0) {
|
|
// Skip bindings that have minimum buffer size set in the layout
|
|
continue;
|
|
}
|
|
|
|
ASSERT(packedIdx < requiredBufferSizes.size());
|
|
const auto& shaderInfo = shaderBindings.find(bindingInfo.binding);
|
|
if (shaderInfo != shaderBindings.end()) {
|
|
requiredBufferSizes[packedIdx] = shaderInfo->second.buffer.minBindingSize;
|
|
} else {
|
|
// We have to include buffers if they are included in the bind group's
|
|
// packed vector. We don't actually need to check these at draw time, so
|
|
// if this is a problem in the future we can optimize it further.
|
|
requiredBufferSizes[packedIdx] = 0;
|
|
}
|
|
++packedIdx;
|
|
}
|
|
|
|
return requiredBufferSizes;
|
|
}
|
|
|
|
MaybeError ValidateCompatibilityOfSingleBindingWithLayout(
|
|
const DeviceBase* device,
|
|
const BindGroupLayoutBase* layout,
|
|
SingleShaderStage entryPointStage,
|
|
BindingNumber bindingNumber,
|
|
const ShaderBindingInfo& shaderInfo) {
|
|
const BindGroupLayoutBase::BindingMap& layoutBindings = layout->GetBindingMap();
|
|
|
|
// An external texture binding found in the shader will later be expanded into multiple
|
|
// bindings at compile time. This expansion will have already happened in the bgl - so
|
|
// the shader and bgl will always mismatch at this point. Expansion info is contained in
|
|
// the bgl object, so we can still verify the bgl used to have an external texture in
|
|
// the slot corresponding to the shader reflection.
|
|
if (shaderInfo.bindingType == BindingInfoType::ExternalTexture) {
|
|
// If an external texture binding used to exist in the bgl, it will be found as a
|
|
// key in the ExternalTextureBindingExpansions map.
|
|
ExternalTextureBindingExpansionMap expansions =
|
|
layout->GetExternalTextureBindingExpansionMap();
|
|
std::map<BindingNumber, dawn_native::ExternalTextureBindingExpansion>::iterator it =
|
|
expansions.find(bindingNumber);
|
|
// TODO(dawn:563): Provide info about the binding types.
|
|
DAWN_INVALID_IF(it == expansions.end(),
|
|
"Binding type in the shader (texture_external) doesn't match the "
|
|
"type in the layout.");
|
|
|
|
return {};
|
|
}
|
|
|
|
const auto& bindingIt = layoutBindings.find(bindingNumber);
|
|
DAWN_INVALID_IF(bindingIt == layoutBindings.end(), "Binding doesn't exist in %s.",
|
|
layout);
|
|
|
|
BindingIndex bindingIndex(bindingIt->second);
|
|
const BindingInfo& layoutInfo = layout->GetBindingInfo(bindingIndex);
|
|
|
|
// TODO(dawn:563): Provide info about the binding types.
|
|
DAWN_INVALID_IF(
|
|
layoutInfo.bindingType != shaderInfo.bindingType,
|
|
"Binding type (buffer vs. texture vs. sampler vs. external) doesn't match the type "
|
|
"in the layout.");
|
|
|
|
ExternalTextureBindingExpansionMap expansions =
|
|
layout->GetExternalTextureBindingExpansionMap();
|
|
DAWN_INVALID_IF(expansions.find(bindingNumber) != expansions.end(),
|
|
"Binding type (buffer vs. texture vs. sampler vs. external) doesn't "
|
|
"match the type in the layout.");
|
|
|
|
// TODO(dawn:563): Provide info about the visibility.
|
|
DAWN_INVALID_IF(
|
|
(layoutInfo.visibility & StageBit(entryPointStage)) == 0,
|
|
"Entry point's stage is not in the binding visibility in the layout (%s)",
|
|
layoutInfo.visibility);
|
|
|
|
switch (layoutInfo.bindingType) {
|
|
case BindingInfoType::Texture: {
|
|
DAWN_INVALID_IF(
|
|
layoutInfo.texture.multisampled != shaderInfo.texture.multisampled,
|
|
"Binding multisampled flag (%u) doesn't match the layout's multisampled "
|
|
"flag (%u)",
|
|
layoutInfo.texture.multisampled, shaderInfo.texture.multisampled);
|
|
|
|
// TODO(dawn:563): Provide info about the sample types.
|
|
DAWN_INVALID_IF((SampleTypeToSampleTypeBit(layoutInfo.texture.sampleType) &
|
|
shaderInfo.texture.compatibleSampleTypes) == 0,
|
|
"The sample type in the shader is not compatible with the "
|
|
"sample type of the layout.");
|
|
|
|
DAWN_INVALID_IF(
|
|
layoutInfo.texture.viewDimension != shaderInfo.texture.viewDimension,
|
|
"The shader's binding dimension (%s) doesn't match the shader's binding "
|
|
"dimension (%s).",
|
|
layoutInfo.texture.viewDimension, shaderInfo.texture.viewDimension);
|
|
break;
|
|
}
|
|
|
|
case BindingInfoType::StorageTexture: {
|
|
ASSERT(layoutInfo.storageTexture.format != wgpu::TextureFormat::Undefined);
|
|
ASSERT(shaderInfo.storageTexture.format != wgpu::TextureFormat::Undefined);
|
|
|
|
DAWN_INVALID_IF(
|
|
layoutInfo.storageTexture.access != shaderInfo.storageTexture.access,
|
|
"The layout's binding access (%s) isn't compatible with the shader's "
|
|
"binding access (%s).",
|
|
layoutInfo.storageTexture.access, shaderInfo.storageTexture.access);
|
|
|
|
DAWN_INVALID_IF(
|
|
layoutInfo.storageTexture.format != shaderInfo.storageTexture.format,
|
|
"The layout's binding format (%s) doesn't match the shader's binding "
|
|
"format (%s).",
|
|
layoutInfo.storageTexture.format, shaderInfo.storageTexture.format);
|
|
|
|
DAWN_INVALID_IF(layoutInfo.storageTexture.viewDimension !=
|
|
shaderInfo.storageTexture.viewDimension,
|
|
"The layout's binding dimension (%s) doesn't match the "
|
|
"shader's binding dimension (%s).",
|
|
layoutInfo.storageTexture.viewDimension,
|
|
shaderInfo.storageTexture.viewDimension);
|
|
break;
|
|
}
|
|
|
|
case BindingInfoType::Buffer: {
|
|
// 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. For internal usage with internal shaders, a storage
|
|
// binding in the shader with an internal storage buffer in the bind group
|
|
// layout is also valid.
|
|
bool validBindingConversion =
|
|
(layoutInfo.buffer.type == kInternalStorageBufferBinding &&
|
|
shaderInfo.buffer.type == wgpu::BufferBindingType::Storage);
|
|
|
|
DAWN_INVALID_IF(
|
|
layoutInfo.buffer.type != shaderInfo.buffer.type && !validBindingConversion,
|
|
"The buffer type in the shader (%s) is not compatible with the type in the "
|
|
"layout (%s).",
|
|
shaderInfo.buffer.type, layoutInfo.buffer.type);
|
|
|
|
DAWN_INVALID_IF(
|
|
layoutInfo.buffer.minBindingSize != 0 &&
|
|
shaderInfo.buffer.minBindingSize > layoutInfo.buffer.minBindingSize,
|
|
"The shader uses more bytes of the buffer (%u) than the layout's "
|
|
"minBindingSize (%u).",
|
|
shaderInfo.buffer.minBindingSize, layoutInfo.buffer.minBindingSize);
|
|
break;
|
|
}
|
|
|
|
case BindingInfoType::Sampler:
|
|
DAWN_INVALID_IF(
|
|
(layoutInfo.sampler.type == wgpu::SamplerBindingType::Comparison) !=
|
|
shaderInfo.sampler.isComparison,
|
|
"The sampler type in the shader (comparison: %u) doesn't match the type in "
|
|
"the layout (comparison: %u).",
|
|
shaderInfo.sampler.isComparison,
|
|
layoutInfo.sampler.type == wgpu::SamplerBindingType::Comparison);
|
|
break;
|
|
|
|
case BindingInfoType::ExternalTexture: {
|
|
UNREACHABLE();
|
|
break;
|
|
}
|
|
}
|
|
|
|
return {};
|
|
}
|
|
MaybeError ValidateCompatibilityWithBindGroupLayout(DeviceBase* device,
|
|
BindGroupIndex group,
|
|
const EntryPointMetadata& entryPoint,
|
|
const BindGroupLayoutBase* layout) {
|
|
// Iterate over all bindings used by this group in the shader, and find the
|
|
// corresponding binding in the BindGroupLayout, if it exists.
|
|
for (const auto& [bindingId, bindingInfo] : entryPoint.bindings[group]) {
|
|
DAWN_TRY_CONTEXT(ValidateCompatibilityOfSingleBindingWithLayout(
|
|
device, layout, entryPoint.stage, bindingId, bindingInfo),
|
|
"validating that the entry-point's declaration for @group(%u) "
|
|
"@binding(%u) matches %s",
|
|
static_cast<uint32_t>(group), static_cast<uint32_t>(bindingId),
|
|
layout);
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
ResultOrError<std::unique_ptr<EntryPointMetadata>> ReflectEntryPointUsingTint(
|
|
const DeviceBase* device,
|
|
tint::inspector::Inspector* inspector,
|
|
const tint::inspector::EntryPoint& entryPoint) {
|
|
const CombinedLimits& limits = device->GetLimits();
|
|
constexpr uint32_t kMaxInterStageShaderLocation = kMaxInterStageShaderVariables - 1;
|
|
|
|
std::unique_ptr<EntryPointMetadata> metadata = std::make_unique<EntryPointMetadata>();
|
|
|
|
// Returns the invalid argument, and if it is true additionally store the formatted
|
|
// error in metadata.infringedLimits. This is to delay the emission of these validation
|
|
// errors until the entry point is used.
|
|
#define DelayedInvalidIf(invalid, ...) \
|
|
([&]() { \
|
|
if (invalid) { \
|
|
metadata->infringedLimitErrors.push_back(absl::StrFormat(__VA_ARGS__)); \
|
|
} \
|
|
return invalid; \
|
|
})()
|
|
|
|
if (!entryPoint.overridable_constants.empty()) {
|
|
DAWN_INVALID_IF(device->IsToggleEnabled(Toggle::DisallowUnsafeAPIs),
|
|
"Pipeline overridable constants are disallowed because they "
|
|
"are partially implemented.");
|
|
|
|
const auto& name2Id = inspector->GetConstantNameToIdMap();
|
|
const auto& id2Scalar = inspector->GetConstantIDs();
|
|
|
|
for (auto& c : entryPoint.overridable_constants) {
|
|
uint32_t id = name2Id.at(c.name);
|
|
OverridableConstantScalar defaultValue;
|
|
if (c.is_initialized) {
|
|
// if it is initialized, the scalar must exist
|
|
const auto& scalar = id2Scalar.at(id);
|
|
if (scalar.IsBool()) {
|
|
defaultValue.b = scalar.AsBool();
|
|
} else if (scalar.IsU32()) {
|
|
defaultValue.u32 = scalar.AsU32();
|
|
} else if (scalar.IsI32()) {
|
|
defaultValue.i32 = scalar.AsI32();
|
|
} else if (scalar.IsFloat()) {
|
|
defaultValue.f32 = scalar.AsFloat();
|
|
} else {
|
|
UNREACHABLE();
|
|
}
|
|
}
|
|
EntryPointMetadata::OverridableConstant constant = {
|
|
id, FromTintOverridableConstantType(c.type), c.is_initialized,
|
|
defaultValue};
|
|
|
|
std::string identifier =
|
|
c.is_numeric_id_specified ? std::to_string(constant.id) : c.name;
|
|
metadata->overridableConstants[identifier] = constant;
|
|
|
|
if (!c.is_initialized) {
|
|
auto [_, inserted] = metadata->uninitializedOverridableConstants.emplace(
|
|
std::move(identifier));
|
|
// The insertion should have taken place
|
|
ASSERT(inserted);
|
|
} else {
|
|
auto [_, inserted] = metadata->initializedOverridableConstants.emplace(
|
|
std::move(identifier));
|
|
// The insertion should have taken place
|
|
ASSERT(inserted);
|
|
}
|
|
}
|
|
}
|
|
|
|
DAWN_TRY_ASSIGN(metadata->stage, TintPipelineStageToShaderStage(entryPoint.stage));
|
|
|
|
if (metadata->stage == SingleShaderStage::Compute) {
|
|
DelayedInvalidIf(
|
|
entryPoint.workgroup_size_x > limits.v1.maxComputeWorkgroupSizeX ||
|
|
entryPoint.workgroup_size_y > limits.v1.maxComputeWorkgroupSizeY ||
|
|
entryPoint.workgroup_size_z > limits.v1.maxComputeWorkgroupSizeZ,
|
|
"Entry-point uses workgroup_size(%u, %u, %u) that exceeds the "
|
|
"maximum allowed (%u, %u, %u).",
|
|
entryPoint.workgroup_size_x, entryPoint.workgroup_size_y,
|
|
entryPoint.workgroup_size_z, limits.v1.maxComputeWorkgroupSizeX,
|
|
limits.v1.maxComputeWorkgroupSizeY, limits.v1.maxComputeWorkgroupSizeZ);
|
|
|
|
// Dimensions have already been validated against their individual limits above.
|
|
// Cast to uint64_t to avoid overflow in this multiplication.
|
|
uint64_t numInvocations = static_cast<uint64_t>(entryPoint.workgroup_size_x) *
|
|
entryPoint.workgroup_size_y * entryPoint.workgroup_size_z;
|
|
DelayedInvalidIf(numInvocations > limits.v1.maxComputeInvocationsPerWorkgroup,
|
|
"The total number of workgroup invocations (%u) exceeds the "
|
|
"maximum allowed (%u).",
|
|
numInvocations, limits.v1.maxComputeInvocationsPerWorkgroup);
|
|
|
|
const size_t workgroupStorageSize =
|
|
inspector->GetWorkgroupStorageSize(entryPoint.name);
|
|
DelayedInvalidIf(workgroupStorageSize > limits.v1.maxComputeWorkgroupStorageSize,
|
|
"The total use of workgroup storage (%u bytes) is larger than "
|
|
"the maximum allowed (%u bytes).",
|
|
workgroupStorageSize, limits.v1.maxComputeWorkgroupStorageSize);
|
|
|
|
metadata->localWorkgroupSize.x = entryPoint.workgroup_size_x;
|
|
metadata->localWorkgroupSize.y = entryPoint.workgroup_size_y;
|
|
metadata->localWorkgroupSize.z = entryPoint.workgroup_size_z;
|
|
|
|
metadata->usesNumWorkgroups = entryPoint.num_workgroups_used;
|
|
}
|
|
|
|
if (metadata->stage == SingleShaderStage::Vertex) {
|
|
for (const auto& inputVar : entryPoint.input_variables) {
|
|
uint32_t unsanitizedLocation = inputVar.location_decoration;
|
|
if (DelayedInvalidIf(unsanitizedLocation >= kMaxVertexAttributes,
|
|
"Vertex input variable \"%s\" has a location (%u) that "
|
|
"exceeds the maximum (%u)",
|
|
inputVar.name, unsanitizedLocation,
|
|
kMaxVertexAttributes)) {
|
|
continue;
|
|
}
|
|
|
|
VertexAttributeLocation location(static_cast<uint8_t>(unsanitizedLocation));
|
|
DAWN_TRY_ASSIGN(
|
|
metadata->vertexInputBaseTypes[location],
|
|
TintComponentTypeToVertexFormatBaseType(inputVar.component_type));
|
|
metadata->usedVertexInputs.set(location);
|
|
}
|
|
|
|
// [[position]] must be declared in a vertex shader but is not exposed as an
|
|
// output variable by Tint so we directly add its components to the total.
|
|
uint32_t totalInterStageShaderComponents = 4;
|
|
for (const auto& outputVar : entryPoint.output_variables) {
|
|
EntryPointMetadata::InterStageVariableInfo variable;
|
|
DAWN_TRY_ASSIGN(variable.baseType, TintComponentTypeToInterStageComponentType(
|
|
outputVar.component_type));
|
|
DAWN_TRY_ASSIGN(
|
|
variable.componentCount,
|
|
TintCompositionTypeToInterStageComponentCount(outputVar.composition_type));
|
|
DAWN_TRY_ASSIGN(
|
|
variable.interpolationType,
|
|
TintInterpolationTypeToInterpolationType(outputVar.interpolation_type));
|
|
DAWN_TRY_ASSIGN(variable.interpolationSampling,
|
|
TintInterpolationSamplingToInterpolationSamplingType(
|
|
outputVar.interpolation_sampling));
|
|
totalInterStageShaderComponents += variable.componentCount;
|
|
|
|
uint32_t location = outputVar.location_decoration;
|
|
if (DelayedInvalidIf(location > kMaxInterStageShaderLocation,
|
|
"Vertex output variable \"%s\" has a location (%u) that "
|
|
"exceeds the maximum (%u).",
|
|
outputVar.name, location, kMaxInterStageShaderLocation)) {
|
|
continue;
|
|
}
|
|
|
|
metadata->usedInterStageVariables.set(location);
|
|
metadata->interStageVariables[location] = variable;
|
|
}
|
|
|
|
DelayedInvalidIf(
|
|
totalInterStageShaderComponents > kMaxInterStageShaderComponents,
|
|
"Total vertex output components count (%u) exceeds the maximum (%u).",
|
|
totalInterStageShaderComponents, kMaxInterStageShaderComponents);
|
|
}
|
|
|
|
if (metadata->stage == SingleShaderStage::Fragment) {
|
|
uint32_t totalInterStageShaderComponents = 0;
|
|
for (const auto& inputVar : entryPoint.input_variables) {
|
|
EntryPointMetadata::InterStageVariableInfo variable;
|
|
DAWN_TRY_ASSIGN(variable.baseType, TintComponentTypeToInterStageComponentType(
|
|
inputVar.component_type));
|
|
DAWN_TRY_ASSIGN(
|
|
variable.componentCount,
|
|
TintCompositionTypeToInterStageComponentCount(inputVar.composition_type));
|
|
DAWN_TRY_ASSIGN(
|
|
variable.interpolationType,
|
|
TintInterpolationTypeToInterpolationType(inputVar.interpolation_type));
|
|
DAWN_TRY_ASSIGN(variable.interpolationSampling,
|
|
TintInterpolationSamplingToInterpolationSamplingType(
|
|
inputVar.interpolation_sampling));
|
|
totalInterStageShaderComponents += variable.componentCount;
|
|
|
|
uint32_t location = inputVar.location_decoration;
|
|
if (DelayedInvalidIf(location > kMaxInterStageShaderLocation,
|
|
"Fragment input variable \"%s\" has a location (%u) that "
|
|
"exceeds the maximum (%u).",
|
|
inputVar.name, location, kMaxInterStageShaderLocation)) {
|
|
continue;
|
|
}
|
|
|
|
metadata->usedInterStageVariables.set(location);
|
|
metadata->interStageVariables[location] = variable;
|
|
}
|
|
|
|
if (entryPoint.front_facing_used) {
|
|
totalInterStageShaderComponents += 1;
|
|
}
|
|
if (entryPoint.input_sample_mask_used) {
|
|
totalInterStageShaderComponents += 1;
|
|
}
|
|
if (entryPoint.sample_index_used) {
|
|
totalInterStageShaderComponents += 1;
|
|
}
|
|
if (entryPoint.input_position_used) {
|
|
totalInterStageShaderComponents += 4;
|
|
}
|
|
|
|
DelayedInvalidIf(
|
|
totalInterStageShaderComponents > kMaxInterStageShaderComponents,
|
|
"Total fragment input components count (%u) exceeds the maximum (%u).",
|
|
totalInterStageShaderComponents, kMaxInterStageShaderComponents);
|
|
|
|
for (const auto& outputVar : entryPoint.output_variables) {
|
|
EntryPointMetadata::FragmentOutputVariableInfo variable;
|
|
DAWN_TRY_ASSIGN(variable.baseType, TintComponentTypeToTextureComponentType(
|
|
outputVar.component_type));
|
|
DAWN_TRY_ASSIGN(
|
|
variable.componentCount,
|
|
TintCompositionTypeToInterStageComponentCount(outputVar.composition_type));
|
|
ASSERT(variable.componentCount <= 4);
|
|
|
|
uint32_t unsanitizedAttachment = outputVar.location_decoration;
|
|
if (DelayedInvalidIf(unsanitizedAttachment >= kMaxColorAttachments,
|
|
"Fragment output variable \"%s\" has a location (%u) that "
|
|
"exceeds the maximum (%u).",
|
|
outputVar.name, unsanitizedAttachment,
|
|
kMaxColorAttachments)) {
|
|
continue;
|
|
}
|
|
|
|
ColorAttachmentIndex attachment(static_cast<uint8_t>(unsanitizedAttachment));
|
|
metadata->fragmentOutputVariables[attachment] = variable;
|
|
metadata->fragmentOutputsWritten.set(attachment);
|
|
}
|
|
}
|
|
|
|
for (const tint::inspector::ResourceBinding& resource :
|
|
inspector->GetResourceBindings(entryPoint.name)) {
|
|
ShaderBindingInfo info;
|
|
|
|
info.bindingType = TintResourceTypeToBindingInfoType(resource.resource_type);
|
|
|
|
switch (info.bindingType) {
|
|
case BindingInfoType::Buffer:
|
|
info.buffer.minBindingSize = resource.size_no_padding;
|
|
DAWN_TRY_ASSIGN(info.buffer.type, TintResourceTypeToBufferBindingType(
|
|
resource.resource_type));
|
|
break;
|
|
case BindingInfoType::Sampler:
|
|
switch (resource.resource_type) {
|
|
case tint::inspector::ResourceBinding::ResourceType::kSampler:
|
|
info.sampler.isComparison = false;
|
|
break;
|
|
case tint::inspector::ResourceBinding::ResourceType::kComparisonSampler:
|
|
info.sampler.isComparison = true;
|
|
break;
|
|
default:
|
|
UNREACHABLE();
|
|
}
|
|
break;
|
|
case BindingInfoType::Texture:
|
|
info.texture.viewDimension =
|
|
TintTextureDimensionToTextureViewDimension(resource.dim);
|
|
if (resource.resource_type ==
|
|
tint::inspector::ResourceBinding::ResourceType::kDepthTexture ||
|
|
resource.resource_type == tint::inspector::ResourceBinding::
|
|
ResourceType::kDepthMultisampledTexture) {
|
|
info.texture.compatibleSampleTypes = SampleTypeBit::Depth;
|
|
} else {
|
|
info.texture.compatibleSampleTypes =
|
|
TintSampledKindToSampleTypeBit(resource.sampled_kind);
|
|
}
|
|
info.texture.multisampled =
|
|
resource.resource_type == tint::inspector::ResourceBinding::
|
|
ResourceType::kMultisampledTexture ||
|
|
resource.resource_type == tint::inspector::ResourceBinding::
|
|
ResourceType::kDepthMultisampledTexture;
|
|
|
|
break;
|
|
case BindingInfoType::StorageTexture:
|
|
DAWN_TRY_ASSIGN(
|
|
info.storageTexture.access,
|
|
TintResourceTypeToStorageTextureAccess(resource.resource_type));
|
|
info.storageTexture.format =
|
|
TintImageFormatToTextureFormat(resource.image_format);
|
|
info.storageTexture.viewDimension =
|
|
TintTextureDimensionToTextureViewDimension(resource.dim);
|
|
|
|
break;
|
|
case BindingInfoType::ExternalTexture:
|
|
break;
|
|
default:
|
|
return DAWN_VALIDATION_ERROR("Unknown binding type in Shader");
|
|
}
|
|
|
|
BindingNumber bindingNumber(resource.binding);
|
|
BindGroupIndex bindGroupIndex(resource.bind_group);
|
|
|
|
if (DelayedInvalidIf(bindGroupIndex >= kMaxBindGroupsTyped,
|
|
"The entry-point uses a binding with a group decoration (%u) "
|
|
"that exceeds the maximum (%u).",
|
|
resource.bind_group, kMaxBindGroups) ||
|
|
DelayedInvalidIf(bindingNumber > kMaxBindingNumberTyped,
|
|
"Binding number (%u) exceeds the maximum binding number (%u).",
|
|
uint32_t(bindingNumber), uint32_t(kMaxBindingNumberTyped))) {
|
|
continue;
|
|
}
|
|
|
|
const auto& [binding, inserted] =
|
|
metadata->bindings[bindGroupIndex].emplace(bindingNumber, info);
|
|
DAWN_INVALID_IF(!inserted,
|
|
"Entry-point has a duplicate binding for (group:%u, binding:%u).",
|
|
resource.binding, resource.bind_group);
|
|
}
|
|
|
|
std::vector<tint::inspector::SamplerTexturePair> samplerTextureUses =
|
|
inspector->GetSamplerTextureUses(entryPoint.name);
|
|
metadata->samplerTexturePairs.reserve(samplerTextureUses.size());
|
|
std::transform(samplerTextureUses.begin(), samplerTextureUses.end(),
|
|
std::back_inserter(metadata->samplerTexturePairs),
|
|
[](const tint::inspector::SamplerTexturePair& pair) {
|
|
EntryPointMetadata::SamplerTexturePair result;
|
|
result.sampler = {BindGroupIndex(pair.sampler_binding_point.group),
|
|
BindingNumber(pair.sampler_binding_point.binding)};
|
|
result.texture = {BindGroupIndex(pair.texture_binding_point.group),
|
|
BindingNumber(pair.texture_binding_point.binding)};
|
|
return result;
|
|
});
|
|
|
|
#undef DelayedInvalidIf
|
|
return std::move(metadata);
|
|
}
|
|
|
|
ResultOrError<EntryPointMetadataTable> ReflectShaderUsingTint(
|
|
const DeviceBase* device,
|
|
const tint::Program* program) {
|
|
ASSERT(program->IsValid());
|
|
|
|
tint::inspector::Inspector inspector(program);
|
|
std::vector<tint::inspector::EntryPoint> entryPoints = inspector.GetEntryPoints();
|
|
DAWN_INVALID_IF(inspector.has_error(), "Tint Reflection failure: Inspector: %s\n",
|
|
inspector.error());
|
|
|
|
EntryPointMetadataTable result;
|
|
|
|
for (const tint::inspector::EntryPoint& entryPoint : entryPoints) {
|
|
std::unique_ptr<EntryPointMetadata> metadata;
|
|
DAWN_TRY_ASSIGN_CONTEXT(metadata,
|
|
ReflectEntryPointUsingTint(device, &inspector, entryPoint),
|
|
"processing entry point \"%s\".", entryPoint.name);
|
|
|
|
ASSERT(result.count(entryPoint.name) == 0);
|
|
result[entryPoint.name] = std::move(metadata);
|
|
}
|
|
return std::move(result);
|
|
}
|
|
} // anonymous namespace
|
|
|
|
ShaderModuleParseResult::ShaderModuleParseResult() = default;
|
|
ShaderModuleParseResult::~ShaderModuleParseResult() = default;
|
|
|
|
ShaderModuleParseResult::ShaderModuleParseResult(ShaderModuleParseResult&& rhs) = default;
|
|
|
|
ShaderModuleParseResult& ShaderModuleParseResult::operator=(ShaderModuleParseResult&& rhs) =
|
|
default;
|
|
|
|
bool ShaderModuleParseResult::HasParsedShader() const {
|
|
return tintProgram != nullptr;
|
|
}
|
|
|
|
// TintSource is a PIMPL container for a tint::Source::File, which needs to be kept alive for as
|
|
// long as tint diagnostics are inspected / printed.
|
|
class TintSource {
|
|
public:
|
|
template <typename... ARGS>
|
|
TintSource(ARGS&&... args) : file(std::forward<ARGS>(args)...) {
|
|
}
|
|
|
|
tint::Source::File file;
|
|
};
|
|
|
|
MaybeError ValidateShaderModuleDescriptor(DeviceBase* device,
|
|
const ShaderModuleDescriptor* descriptor,
|
|
ShaderModuleParseResult* parseResult,
|
|
OwnedCompilationMessages* outMessages) {
|
|
ASSERT(parseResult != nullptr);
|
|
|
|
const ChainedStruct* chainedDescriptor = descriptor->nextInChain;
|
|
DAWN_INVALID_IF(chainedDescriptor == nullptr,
|
|
"Shader module descriptor missing chained descriptor");
|
|
|
|
// For now only a single SPIRV or WGSL subdescriptor is allowed.
|
|
DAWN_TRY(ValidateSingleSType(chainedDescriptor, wgpu::SType::ShaderModuleSPIRVDescriptor,
|
|
wgpu::SType::ShaderModuleWGSLDescriptor));
|
|
|
|
ScopedTintICEHandler scopedICEHandler(device);
|
|
|
|
const ShaderModuleSPIRVDescriptor* spirvDesc = nullptr;
|
|
FindInChain(chainedDescriptor, &spirvDesc);
|
|
const ShaderModuleWGSLDescriptor* wgslDesc = nullptr;
|
|
FindInChain(chainedDescriptor, &wgslDesc);
|
|
|
|
// We have a temporary toggle to force the SPIRV ingestion to go through a WGSL
|
|
// intermediate step. It is done by switching the spirvDesc for a wgslDesc below.
|
|
ShaderModuleWGSLDescriptor newWgslDesc;
|
|
std::string newWgslCode;
|
|
if (spirvDesc && device->IsToggleEnabled(Toggle::ForceWGSLStep)) {
|
|
#if TINT_BUILD_WGSL_WRITER
|
|
std::vector<uint32_t> spirv(spirvDesc->code, spirvDesc->code + spirvDesc->codeSize);
|
|
tint::Program program;
|
|
DAWN_TRY_ASSIGN(program, ParseSPIRV(spirv, outMessages));
|
|
|
|
tint::writer::wgsl::Options options;
|
|
auto result = tint::writer::wgsl::Generate(&program, options);
|
|
DAWN_INVALID_IF(!result.success, "Tint WGSL failure: Generator: %s", result.error);
|
|
|
|
newWgslCode = std::move(result.wgsl);
|
|
newWgslDesc.source = newWgslCode.c_str();
|
|
|
|
spirvDesc = nullptr;
|
|
wgslDesc = &newWgslDesc;
|
|
#else
|
|
device->EmitLog(
|
|
WGPULoggingType_Info,
|
|
"Toggle::ForceWGSLStep skipped because TINT_BUILD_WGSL_WRITER is not defined\n");
|
|
#endif
|
|
}
|
|
|
|
if (spirvDesc) {
|
|
DAWN_INVALID_IF(device->IsToggleEnabled(Toggle::DisallowSpirv),
|
|
"SPIR-V is disallowed.");
|
|
|
|
std::vector<uint32_t> spirv(spirvDesc->code, spirvDesc->code + spirvDesc->codeSize);
|
|
tint::Program program;
|
|
DAWN_TRY_ASSIGN(program, ParseSPIRV(spirv, outMessages));
|
|
parseResult->tintProgram = std::make_unique<tint::Program>(std::move(program));
|
|
} else if (wgslDesc) {
|
|
auto tintSource = std::make_unique<TintSource>("", wgslDesc->source);
|
|
|
|
if (device->IsToggleEnabled(Toggle::DumpShaders)) {
|
|
std::ostringstream dumpedMsg;
|
|
dumpedMsg << "// Dumped WGSL:" << std::endl << wgslDesc->source;
|
|
device->EmitLog(WGPULoggingType_Info, dumpedMsg.str().c_str());
|
|
}
|
|
|
|
tint::Program program;
|
|
DAWN_TRY_ASSIGN(program, ParseWGSL(&tintSource->file, outMessages));
|
|
parseResult->tintProgram = std::make_unique<tint::Program>(std::move(program));
|
|
parseResult->tintSource = std::move(tintSource);
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
RequiredBufferSizes ComputeRequiredBufferSizesForLayout(const EntryPointMetadata& entryPoint,
|
|
const PipelineLayoutBase* layout) {
|
|
RequiredBufferSizes bufferSizes;
|
|
for (BindGroupIndex group : IterateBitSet(layout->GetBindGroupLayoutsMask())) {
|
|
bufferSizes[group] = GetBindGroupMinBufferSizes(entryPoint.bindings[group],
|
|
layout->GetBindGroupLayout(group));
|
|
}
|
|
|
|
return bufferSizes;
|
|
}
|
|
|
|
ResultOrError<tint::Program> RunTransforms(tint::transform::Transform* transform,
|
|
const tint::Program* program,
|
|
const tint::transform::DataMap& inputs,
|
|
tint::transform::DataMap* outputs,
|
|
OwnedCompilationMessages* outMessages) {
|
|
tint::transform::Output output = transform->Run(program, inputs);
|
|
if (outMessages != nullptr) {
|
|
outMessages->AddMessages(output.program.Diagnostics());
|
|
}
|
|
DAWN_INVALID_IF(!output.program.IsValid(), "Tint program failure: %s\n",
|
|
output.program.Diagnostics().str());
|
|
if (outputs != nullptr) {
|
|
*outputs = std::move(output.data);
|
|
}
|
|
return std::move(output.program);
|
|
}
|
|
|
|
void AddVertexPullingTransformConfig(const RenderPipelineBase& renderPipeline,
|
|
const std::string& entryPoint,
|
|
BindGroupIndex pullingBufferBindingSet,
|
|
tint::transform::DataMap* transformInputs) {
|
|
tint::transform::VertexPulling::Config cfg;
|
|
cfg.entry_point_name = entryPoint;
|
|
cfg.pulling_group = static_cast<uint32_t>(pullingBufferBindingSet);
|
|
|
|
cfg.vertex_state.resize(renderPipeline.GetVertexBufferCount());
|
|
for (VertexBufferSlot slot : IterateBitSet(renderPipeline.GetVertexBufferSlotsUsed())) {
|
|
const VertexBufferInfo& dawnInfo = renderPipeline.GetVertexBuffer(slot);
|
|
tint::transform::VertexBufferLayoutDescriptor* tintInfo =
|
|
&cfg.vertex_state[static_cast<uint8_t>(slot)];
|
|
|
|
tintInfo->array_stride = dawnInfo.arrayStride;
|
|
tintInfo->step_mode = ToTintVertexStepMode(dawnInfo.stepMode);
|
|
}
|
|
|
|
for (VertexAttributeLocation location :
|
|
IterateBitSet(renderPipeline.GetAttributeLocationsUsed())) {
|
|
const VertexAttributeInfo& dawnInfo = renderPipeline.GetAttribute(location);
|
|
tint::transform::VertexAttributeDescriptor tintInfo;
|
|
tintInfo.format = ToTintVertexFormat(dawnInfo.format);
|
|
tintInfo.offset = dawnInfo.offset;
|
|
tintInfo.shader_location = static_cast<uint32_t>(static_cast<uint8_t>(location));
|
|
|
|
uint8_t vertexBufferSlot = static_cast<uint8_t>(dawnInfo.vertexBufferSlot);
|
|
cfg.vertex_state[vertexBufferSlot].attributes.push_back(tintInfo);
|
|
}
|
|
|
|
transformInputs->Add<tint::transform::VertexPulling::Config>(cfg);
|
|
}
|
|
|
|
MaybeError ValidateCompatibilityWithPipelineLayout(DeviceBase* device,
|
|
const EntryPointMetadata& entryPoint,
|
|
const PipelineLayoutBase* layout) {
|
|
for (BindGroupIndex group : IterateBitSet(layout->GetBindGroupLayoutsMask())) {
|
|
DAWN_TRY_CONTEXT(ValidateCompatibilityWithBindGroupLayout(
|
|
device, group, entryPoint, layout->GetBindGroupLayout(group)),
|
|
"validating the entry-point's compatibility for group %u with %s",
|
|
static_cast<uint32_t>(group), layout->GetBindGroupLayout(group));
|
|
}
|
|
|
|
for (BindGroupIndex group : IterateBitSet(~layout->GetBindGroupLayoutsMask())) {
|
|
DAWN_INVALID_IF(entryPoint.bindings[group].size() > 0,
|
|
"The entry-point uses bindings in group %u but %s doesn't have a "
|
|
"BindGroupLayout for this index",
|
|
static_cast<uint32_t>(group), layout);
|
|
}
|
|
|
|
// Validate that filtering samplers are not used with unfilterable textures.
|
|
for (const auto& pair : entryPoint.samplerTexturePairs) {
|
|
const BindGroupLayoutBase* samplerBGL = layout->GetBindGroupLayout(pair.sampler.group);
|
|
const BindingInfo& samplerInfo =
|
|
samplerBGL->GetBindingInfo(samplerBGL->GetBindingIndex(pair.sampler.binding));
|
|
if (samplerInfo.sampler.type != wgpu::SamplerBindingType::Filtering) {
|
|
continue;
|
|
}
|
|
const BindGroupLayoutBase* textureBGL = layout->GetBindGroupLayout(pair.texture.group);
|
|
const BindingInfo& textureInfo =
|
|
textureBGL->GetBindingInfo(textureBGL->GetBindingIndex(pair.texture.binding));
|
|
|
|
ASSERT(textureInfo.bindingType != BindingInfoType::Buffer &&
|
|
textureInfo.bindingType != BindingInfoType::Sampler &&
|
|
textureInfo.bindingType != BindingInfoType::StorageTexture);
|
|
|
|
if (textureInfo.bindingType != BindingInfoType::Texture) {
|
|
continue;
|
|
}
|
|
|
|
// Uint/sint can't be statically used with a sampler, so they any
|
|
// texture bindings reflected must be float or depth textures. If
|
|
// the shader uses a float/depth texture but the bind group layout
|
|
// specifies a uint/sint texture binding,
|
|
// |ValidateCompatibilityWithBindGroupLayout| will fail since the
|
|
// sampleType does not match.
|
|
ASSERT(textureInfo.texture.sampleType != wgpu::TextureSampleType::Undefined &&
|
|
textureInfo.texture.sampleType != wgpu::TextureSampleType::Uint &&
|
|
textureInfo.texture.sampleType != wgpu::TextureSampleType::Sint);
|
|
|
|
DAWN_INVALID_IF(
|
|
textureInfo.texture.sampleType == wgpu::TextureSampleType::UnfilterableFloat,
|
|
"Texture binding (group:%u, binding:%u) is %s but used statically with a sampler "
|
|
"(group:%u, binding:%u) that's %s",
|
|
static_cast<uint32_t>(pair.texture.group),
|
|
static_cast<uint32_t>(pair.texture.binding),
|
|
wgpu::TextureSampleType::UnfilterableFloat,
|
|
static_cast<uint32_t>(pair.sampler.group),
|
|
static_cast<uint32_t>(pair.sampler.binding), wgpu::SamplerBindingType::Filtering);
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
// ShaderModuleBase
|
|
|
|
ShaderModuleBase::ShaderModuleBase(DeviceBase* device,
|
|
const ShaderModuleDescriptor* descriptor,
|
|
ApiObjectBase::UntrackedByDeviceTag tag)
|
|
: ApiObjectBase(device, descriptor->label), mType(Type::Undefined) {
|
|
ASSERT(descriptor->nextInChain != nullptr);
|
|
const ShaderModuleSPIRVDescriptor* spirvDesc = nullptr;
|
|
FindInChain(descriptor->nextInChain, &spirvDesc);
|
|
const ShaderModuleWGSLDescriptor* wgslDesc = nullptr;
|
|
FindInChain(descriptor->nextInChain, &wgslDesc);
|
|
ASSERT(spirvDesc || wgslDesc);
|
|
|
|
if (spirvDesc) {
|
|
mType = Type::Spirv;
|
|
mOriginalSpirv.assign(spirvDesc->code, spirvDesc->code + spirvDesc->codeSize);
|
|
} else if (wgslDesc) {
|
|
mType = Type::Wgsl;
|
|
mWgsl = std::string(wgslDesc->source);
|
|
}
|
|
}
|
|
|
|
ShaderModuleBase::ShaderModuleBase(DeviceBase* device, const ShaderModuleDescriptor* descriptor)
|
|
: ShaderModuleBase(device, descriptor, kUntrackedByDevice) {
|
|
TrackInDevice();
|
|
}
|
|
|
|
ShaderModuleBase::ShaderModuleBase(DeviceBase* device)
|
|
: ApiObjectBase(device, kLabelNotImplemented) {
|
|
TrackInDevice();
|
|
}
|
|
|
|
ShaderModuleBase::ShaderModuleBase(DeviceBase* device, ObjectBase::ErrorTag tag)
|
|
: ApiObjectBase(device, tag), mType(Type::Undefined) {
|
|
}
|
|
|
|
ShaderModuleBase::~ShaderModuleBase() = default;
|
|
|
|
void ShaderModuleBase::DestroyImpl() {
|
|
if (IsCachedReference()) {
|
|
// Do not uncache the actual cached object if we are a blueprint.
|
|
GetDevice()->UncacheShaderModule(this);
|
|
}
|
|
}
|
|
|
|
// static
|
|
Ref<ShaderModuleBase> ShaderModuleBase::MakeError(DeviceBase* device) {
|
|
return AcquireRef(new ShaderModuleBase(device, ObjectBase::kError));
|
|
}
|
|
|
|
ObjectType ShaderModuleBase::GetType() const {
|
|
return ObjectType::ShaderModule;
|
|
}
|
|
|
|
bool ShaderModuleBase::HasEntryPoint(const std::string& entryPoint) const {
|
|
return mEntryPoints.count(entryPoint) > 0;
|
|
}
|
|
|
|
const EntryPointMetadata& ShaderModuleBase::GetEntryPoint(const std::string& entryPoint) const {
|
|
ASSERT(HasEntryPoint(entryPoint));
|
|
return *mEntryPoints.at(entryPoint);
|
|
}
|
|
|
|
size_t ShaderModuleBase::ComputeContentHash() {
|
|
ObjectContentHasher recorder;
|
|
recorder.Record(mType);
|
|
recorder.Record(mOriginalSpirv);
|
|
recorder.Record(mWgsl);
|
|
return recorder.GetContentHash();
|
|
}
|
|
|
|
bool ShaderModuleBase::EqualityFunc::operator()(const ShaderModuleBase* a,
|
|
const ShaderModuleBase* b) const {
|
|
return a->mType == b->mType && a->mOriginalSpirv == b->mOriginalSpirv &&
|
|
a->mWgsl == b->mWgsl;
|
|
}
|
|
|
|
const tint::Program* ShaderModuleBase::GetTintProgram() const {
|
|
ASSERT(mTintProgram);
|
|
return mTintProgram.get();
|
|
}
|
|
|
|
void ShaderModuleBase::APIGetCompilationInfo(wgpu::CompilationInfoCallback callback,
|
|
void* userdata) {
|
|
if (callback == nullptr) {
|
|
return;
|
|
}
|
|
|
|
callback(WGPUCompilationInfoRequestStatus_Success,
|
|
mCompilationMessages->GetCompilationInfo(), userdata);
|
|
}
|
|
|
|
void ShaderModuleBase::InjectCompilationMessages(
|
|
std::unique_ptr<OwnedCompilationMessages> compilationMessages) {
|
|
// TODO(dawn:944): ensure the InjectCompilationMessages is properly handled for shader
|
|
// module returned from cache.
|
|
// InjectCompilationMessages should be called only once for a shader module, after it is
|
|
// created. However currently InjectCompilationMessages may be called on a shader module
|
|
// returned from cache rather than newly created, and violate the rule. We just skip the
|
|
// injection in this case for now, but a proper solution including ensure the cache goes
|
|
// before the validation is required.
|
|
if (mCompilationMessages != nullptr) {
|
|
return;
|
|
}
|
|
// Move the compilationMessages into the shader module and emit the tint errors and warnings
|
|
mCompilationMessages = std::move(compilationMessages);
|
|
|
|
// Emit the formatted Tint errors and warnings within the moved compilationMessages
|
|
const std::vector<std::string>& formattedTintMessages =
|
|
mCompilationMessages->GetFormattedTintMessages();
|
|
if (formattedTintMessages.empty()) {
|
|
return;
|
|
}
|
|
std::ostringstream t;
|
|
for (auto pMessage = formattedTintMessages.begin(); pMessage != formattedTintMessages.end();
|
|
pMessage++) {
|
|
if (pMessage != formattedTintMessages.begin()) {
|
|
t << std::endl;
|
|
}
|
|
t << *pMessage;
|
|
}
|
|
this->GetDevice()->EmitLog(WGPULoggingType_Warning, t.str().c_str());
|
|
}
|
|
|
|
OwnedCompilationMessages* ShaderModuleBase::GetCompilationMessages() const {
|
|
return mCompilationMessages.get();
|
|
}
|
|
|
|
// static
|
|
void ShaderModuleBase::AddExternalTextureTransform(const PipelineLayoutBase* layout,
|
|
tint::transform::Manager* transformManager,
|
|
tint::transform::DataMap* transformInputs) {
|
|
tint::transform::MultiplanarExternalTexture::BindingsMap newBindingsMap;
|
|
for (BindGroupIndex i : IterateBitSet(layout->GetBindGroupLayoutsMask())) {
|
|
const BindGroupLayoutBase* bgl = layout->GetBindGroupLayout(i);
|
|
|
|
for (const auto& expansion : bgl->GetExternalTextureBindingExpansionMap()) {
|
|
newBindingsMap[{static_cast<uint32_t>(i),
|
|
static_cast<uint32_t>(expansion.second.plane0)}] = {
|
|
{static_cast<uint32_t>(i), static_cast<uint32_t>(expansion.second.plane1)},
|
|
{static_cast<uint32_t>(i), static_cast<uint32_t>(expansion.second.params)}};
|
|
}
|
|
}
|
|
|
|
if (!newBindingsMap.empty()) {
|
|
transformManager->Add<tint::transform::MultiplanarExternalTexture>();
|
|
transformInputs->Add<tint::transform::MultiplanarExternalTexture::NewBindingPoints>(
|
|
newBindingsMap);
|
|
}
|
|
}
|
|
|
|
MaybeError ShaderModuleBase::InitializeBase(ShaderModuleParseResult* parseResult) {
|
|
mTintProgram = std::move(parseResult->tintProgram);
|
|
mTintSource = std::move(parseResult->tintSource);
|
|
|
|
DAWN_TRY_ASSIGN(mEntryPoints, ReflectShaderUsingTint(GetDevice(), mTintProgram.get()));
|
|
return {};
|
|
}
|
|
|
|
size_t PipelineLayoutEntryPointPairHashFunc::operator()(
|
|
const PipelineLayoutEntryPointPair& pair) const {
|
|
size_t hash = 0;
|
|
HashCombine(&hash, pair.first, pair.second);
|
|
return hash;
|
|
}
|
|
|
|
} // namespace dawn::native
|