// Copyright 2020 The Tint 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 "src/inspector/inspector.h" #include #include #include #include "src/ast/bool_literal.h" #include "src/ast/constructor_expression.h" #include "src/ast/decorated_variable.h" #include "src/ast/float_literal.h" #include "src/ast/function.h" #include "src/ast/null_literal.h" #include "src/ast/scalar_constructor_expression.h" #include "src/ast/sint_literal.h" #include "src/ast/type/access_control_type.h" #include "src/ast/type/array_type.h" #include "src/ast/type/matrix_type.h" #include "src/ast/type/multisampled_texture_type.h" #include "src/ast/type/sampled_texture_type.h" #include "src/ast/type/struct_type.h" #include "src/ast/type/texture_type.h" #include "src/ast/type/type.h" #include "src/ast/type/vector_type.h" #include "src/ast/uint_literal.h" namespace tint { namespace inspector { Inspector::Inspector(const ast::Module& module) : ctx_(new Context()), context_is_owned_(true), module_(module) {} Inspector::Inspector(Context* ctx, const ast::Module& module) : ctx_(ctx), context_is_owned_(false), module_(module) { assert(ctx); } Inspector::~Inspector() { if (context_is_owned_) delete ctx_; } std::vector Inspector::GetEntryPoints() { std::vector result; for (auto* func : module_.functions()) { if (!func->IsEntryPoint()) { continue; } EntryPoint entry_point; entry_point.name = func->name(); entry_point.remapped_name = ctx_->namer()->NameFor(func->name()); entry_point.stage = func->pipeline_stage(); std::tie(entry_point.workgroup_size_x, entry_point.workgroup_size_y, entry_point.workgroup_size_z) = func->workgroup_size(); for (auto* var : func->referenced_module_variables()) { if (var->storage_class() == ast::StorageClass::kInput) { entry_point.input_variables.push_back(var->name()); } else { entry_point.output_variables.push_back(var->name()); } } result.push_back(std::move(entry_point)); } return result; } std::string Inspector::GetRemappedNameForEntryPoint( const std::string& entry_point) { // TODO(rharrison): Reenable once all of the backends are using the renamed // entry points. // auto* func = FindEntryPointByName(entry_point); // if (!func) { // return {}; // } // return ctx_->namer()->NameFor(entry_point); return entry_point; } std::map Inspector::GetConstantIDs() { std::map result; for (auto* var : module_.global_variables()) { if (!var->IsDecorated()) { continue; } auto* decorated = var->AsDecorated(); if (!decorated->HasConstantIdDecoration()) { continue; } // If there are conflicting defintions for a constant id, that is invalid // WGSL, so the validator should catch it. Thus here the inspector just // assumes all definitians of the constant id are the same, so only needs // to find the first reference to constant id. uint32_t constant_id = decorated->constant_id(); if (result.find(constant_id) != result.end()) { continue; } if (!var->has_constructor()) { result[constant_id] = Scalar(); continue; } auto* expression = var->constructor(); if (!expression->IsConstructor()) { // This is invalid WGSL, but handling gracefully. result[constant_id] = Scalar(); continue; } auto* constructor = expression->AsConstructor(); if (!constructor->IsScalarConstructor()) { // This is invalid WGSL, but handling gracefully. result[constant_id] = Scalar(); continue; } auto* literal = constructor->AsScalarConstructor()->literal(); if (!literal) { // This is invalid WGSL, but handling gracefully. result[constant_id] = Scalar(); continue; } if (literal->IsBool()) { result[constant_id] = Scalar(literal->AsBool()->IsTrue()); continue; } if (literal->IsUint()) { result[constant_id] = Scalar(literal->AsUint()->value()); continue; } if (literal->IsSint()) { result[constant_id] = Scalar(literal->AsSint()->value()); continue; } if (literal->IsFloat()) { result[constant_id] = Scalar(literal->AsFloat()->value()); continue; } result[constant_id] = Scalar(); } return result; } std::vector Inspector::GetUniformBufferResourceBindings( const std::string& entry_point) { auto* func = FindEntryPointByName(entry_point); if (!func) { return {}; } std::vector result; for (auto& ruv : func->referenced_uniform_variables()) { ResourceBinding entry; ast::Variable* var = nullptr; ast::Function::BindingInfo binding_info; std::tie(var, binding_info) = ruv; if (!var->type()->Is()) { continue; } auto* unwrapped_type = var->type()->UnwrapIfNeeded(); if (!unwrapped_type->IsStruct()) { continue; } if (!unwrapped_type->AsStruct()->IsBlockDecorated()) { continue; } entry.bind_group = binding_info.set->value(); entry.binding = binding_info.binding->value(); entry.min_buffer_binding_size = var->type()->MinBufferBindingSize( ast::type::MemoryLayout::kUniformBuffer); result.push_back(entry); } return result; } std::vector Inspector::GetStorageBufferResourceBindings( const std::string& entry_point) { return GetStorageBufferResourceBindingsImpl(entry_point, false); } std::vector Inspector::GetReadOnlyStorageBufferResourceBindings( const std::string& entry_point) { return GetStorageBufferResourceBindingsImpl(entry_point, true); } std::vector Inspector::GetSamplerResourceBindings( const std::string& entry_point) { auto* func = FindEntryPointByName(entry_point); if (!func) { return {}; } std::vector result; for (auto& rs : func->referenced_sampler_variables()) { ResourceBinding entry; ast::Variable* var = nullptr; ast::Function::BindingInfo binding_info; std::tie(var, binding_info) = rs; entry.bind_group = binding_info.set->value(); entry.binding = binding_info.binding->value(); result.push_back(entry); } return result; } std::vector Inspector::GetComparisonSamplerResourceBindings( const std::string& entry_point) { auto* func = FindEntryPointByName(entry_point); if (!func) { return {}; } std::vector result; for (auto& rcs : func->referenced_comparison_sampler_variables()) { ResourceBinding entry; ast::Variable* var = nullptr; ast::Function::BindingInfo binding_info; std::tie(var, binding_info) = rcs; entry.bind_group = binding_info.set->value(); entry.binding = binding_info.binding->value(); result.push_back(entry); } return result; } std::vector Inspector::GetSampledTextureResourceBindings( const std::string& entry_point) { return GetSampledTextureResourceBindingsImpl(entry_point, false); } std::vector Inspector::GetMultisampledTextureResourceBindings( const std::string& entry_point) { return GetSampledTextureResourceBindingsImpl(entry_point, true); } ast::Function* Inspector::FindEntryPointByName(const std::string& name) { auto* func = module_.FindFunctionByName(name); if (!func) { error_ += name + " was not found!"; return nullptr; } if (!func->IsEntryPoint()) { error_ += name + " is not an entry point!"; return nullptr; } return func; } std::vector Inspector::GetStorageBufferResourceBindingsImpl( const std::string& entry_point, bool read_only) { auto* func = FindEntryPointByName(entry_point); if (!func) { return {}; } std::vector result; for (auto& rsv : func->referenced_storagebuffer_variables()) { ResourceBinding entry; ast::Variable* var = nullptr; ast::Function::BindingInfo binding_info; std::tie(var, binding_info) = rsv; if (!var->type()->Is()) { continue; } auto* ac_type = var->type()->As(); if (read_only != ac_type->IsReadOnly()) { continue; } if (!var->type()->UnwrapIfNeeded()->IsStruct()) { continue; } entry.bind_group = binding_info.set->value(); entry.binding = binding_info.binding->value(); entry.min_buffer_binding_size = var->type()->MinBufferBindingSize( ast::type::MemoryLayout::kStorageBuffer); result.push_back(entry); } return result; } std::vector Inspector::GetSampledTextureResourceBindingsImpl( const std::string& entry_point, bool multisampled_only) { auto* func = FindEntryPointByName(entry_point); if (!func) { return {}; } std::vector result; auto& referenced_variables = multisampled_only ? func->referenced_multisampled_texture_variables() : func->referenced_sampled_texture_variables(); for (auto& ref : referenced_variables) { ResourceBinding entry; ast::Variable* var = nullptr; ast::Function::BindingInfo binding_info; std::tie(var, binding_info) = ref; entry.bind_group = binding_info.set->value(); entry.binding = binding_info.binding->value(); auto* texture_type = var->type()->UnwrapIfNeeded()->AsTexture(); switch (texture_type->dim()) { case ast::type::TextureDimension::k1d: entry.dim = ResourceBinding::TextureDimension::k1d; break; case ast::type::TextureDimension::k1dArray: entry.dim = ResourceBinding::TextureDimension::k1dArray; break; case ast::type::TextureDimension::k2d: entry.dim = ResourceBinding::TextureDimension::k2d; break; case ast::type::TextureDimension::k2dArray: entry.dim = ResourceBinding::TextureDimension::k2dArray; break; case ast::type::TextureDimension::k3d: entry.dim = ResourceBinding::TextureDimension::k3d; break; case ast::type::TextureDimension::kCube: entry.dim = ResourceBinding::TextureDimension::kCube; break; case ast::type::TextureDimension::kCubeArray: entry.dim = ResourceBinding::TextureDimension::kCubeArray; break; default: entry.dim = ResourceBinding::TextureDimension::kNone; break; } ast::type::Type* base_type = nullptr; if (multisampled_only) { base_type = texture_type->AsMultisampled()->type()->UnwrapIfNeeded(); } else { base_type = texture_type->AsSampled()->type()->UnwrapIfNeeded(); } if (base_type->IsArray()) { base_type = base_type->AsArray()->type(); } else if (base_type->IsMatrix()) { base_type = base_type->AsMatrix()->type(); } else if (base_type->IsVector()) { base_type = base_type->AsVector()->type(); } if (base_type->IsF32()) { entry.sampled_kind = ResourceBinding::SampledKind::kFloat; } else if (base_type->IsU32()) { entry.sampled_kind = ResourceBinding::SampledKind::kUInt; } else if (base_type->IsI32()) { entry.sampled_kind = ResourceBinding::SampledKind::kSInt; } else { entry.sampled_kind = ResourceBinding::SampledKind::kUnknown; } result.push_back(entry); } return result; } } // namespace inspector } // namespace tint