// 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/ast/function.h" #include #include "src/ast/clone_context.h" #include "src/ast/decorated_variable.h" #include "src/ast/module.h" #include "src/ast/stage_decoration.h" #include "src/ast/type/multisampled_texture_type.h" #include "src/ast/type/sampled_texture_type.h" #include "src/ast/type/texture_type.h" #include "src/ast/workgroup_decoration.h" namespace tint { namespace ast { Function::Function() = default; Function::Function(const std::string& name, VariableList params, type::Type* return_type, BlockStatement* body) : Base(), name_(name), params_(std::move(params)), return_type_(return_type), body_(body) {} Function::Function(const Source& source, const std::string& name, VariableList params, type::Type* return_type, BlockStatement* body) : Base(source), name_(name), params_(std::move(params)), return_type_(return_type), body_(body) {} Function::Function(Function&&) = default; Function::~Function() = default; std::tuple Function::workgroup_size() const { for (auto* deco : decorations_) { if (auto* workgroup = deco->As()) { return workgroup->values(); } } return {1, 1, 1}; } PipelineStage Function::pipeline_stage() const { for (auto* deco : decorations_) { if (auto* stage = deco->As()) { return stage->value(); } } return PipelineStage::kNone; } void Function::add_referenced_module_variable(Variable* var) { for (const auto* v : referenced_module_vars_) { if (v->name() == var->name()) { return; } } referenced_module_vars_.push_back(var); } const std::vector> Function::referenced_location_variables() const { std::vector> ret; for (auto* var : referenced_module_variables()) { if (auto* decos = var->As()) { for (auto* deco : decos->decorations()) { if (auto* location = deco->As()) { ret.push_back({var, location}); break; } } } } return ret; } const std::vector> Function::referenced_uniform_variables() const { std::vector> ret; for (auto* var : referenced_module_variables()) { if (var->storage_class() != StorageClass::kUniform) { continue; } if (auto* decorated = var->As()) { BindingDecoration* binding = nullptr; SetDecoration* set = nullptr; for (auto* deco : decorated->decorations()) { if (auto* b = deco->As()) { binding = b; } else if (auto* s = deco->As()) { set = s; } } if (binding == nullptr || set == nullptr) { continue; } ret.push_back({var, BindingInfo{binding, set}}); } } return ret; } const std::vector> Function::referenced_storagebuffer_variables() const { std::vector> ret; for (auto* var : referenced_module_variables()) { if (var->storage_class() != StorageClass::kStorageBuffer) { continue; } if (auto* decorated = var->As()) { BindingDecoration* binding = nullptr; SetDecoration* set = nullptr; for (auto* deco : decorated->decorations()) { if (auto* b = deco->As()) { binding = b; } else if (auto* s = deco->As()) { set = s; } } if (binding == nullptr || set == nullptr) { continue; } ret.push_back({var, BindingInfo{binding, set}}); } } return ret; } const std::vector> Function::referenced_builtin_variables() const { std::vector> ret; for (auto* var : referenced_module_variables()) { if (auto* decorated = var->As()) { for (auto* deco : decorated->decorations()) { if (auto* builtin = deco->As()) { ret.push_back({var, builtin}); break; } } } } return ret; } const std::vector> Function::referenced_sampler_variables() const { return ReferencedSamplerVariablesImpl(type::SamplerKind::kSampler); } const std::vector> Function::referenced_comparison_sampler_variables() const { return ReferencedSamplerVariablesImpl(type::SamplerKind::kComparisonSampler); } const std::vector> Function::referenced_sampled_texture_variables() const { return ReferencedSampledTextureVariablesImpl(false); } const std::vector> Function::referenced_multisampled_texture_variables() const { return ReferencedSampledTextureVariablesImpl(true); } void Function::add_ancestor_entry_point(const std::string& ep) { for (const auto& point : ancestor_entry_points_) { if (point == ep) { return; } } ancestor_entry_points_.push_back(ep); } bool Function::HasAncestorEntryPoint(const std::string& name) const { for (const auto& point : ancestor_entry_points_) { if (point == name) { return true; } } return false; } const Statement* Function::get_last_statement() const { return body_->last(); } Function* Function::Clone(CloneContext* ctx) const { auto* cloned = ctx->mod->create( ctx->Clone(source()), name_, ctx->Clone(params_), ctx->Clone(return_type_), ctx->Clone(body_)); cloned->set_decorations(ctx->Clone(decorations_)); return cloned; } bool Function::IsValid() const { for (auto* param : params_) { if (param == nullptr || !param->IsValid()) return false; } if (body_ == nullptr || !body_->IsValid()) { return false; } if (name_.length() == 0) { return false; } if (return_type_ == nullptr) { return false; } return true; } void Function::to_str(std::ostream& out, size_t indent) const { make_indent(out, indent); out << "Function " << name_ << " -> " << return_type_->type_name() << std::endl; for (auto* deco : decorations()) { deco->to_str(out, indent); } make_indent(out, indent); out << "("; if (params_.size() > 0) { out << std::endl; for (auto* param : params_) param->to_str(out, indent + 2); make_indent(out, indent); } out << ")" << std::endl; make_indent(out, indent); out << "{" << std::endl; if (body_ != nullptr) { for (auto* stmt : *body_) { stmt->to_str(out, indent + 2); } } make_indent(out, indent); out << "}" << std::endl; } std::string Function::type_name() const { std::ostringstream out; out << "__func" + return_type_->type_name(); for (auto* param : params_) { out << param->type()->type_name(); } return out.str(); } const std::vector> Function::ReferencedSamplerVariablesImpl(type::SamplerKind kind) const { std::vector> ret; for (auto* var : referenced_module_variables()) { auto* unwrapped_type = var->type()->UnwrapIfNeeded(); auto* sampler = unwrapped_type->As(); if (sampler == nullptr || sampler->kind() != kind) { continue; } if (auto* decorated = var->As()) { BindingDecoration* binding = nullptr; SetDecoration* set = nullptr; for (auto* deco : decorated->decorations()) { if (auto* b = deco->As()) { binding = b; } if (auto* s = deco->As()) { set = s; } } if (binding == nullptr || set == nullptr) { continue; } ret.push_back({var, BindingInfo{binding, set}}); } } return ret; } const std::vector> Function::ReferencedSampledTextureVariablesImpl(bool multisampled) const { std::vector> ret; for (auto* var : referenced_module_variables()) { auto* unwrapped_type = var->type()->UnwrapIfNeeded(); auto* texture = unwrapped_type->As(); if (texture == nullptr) { continue; } auto is_multisampled = texture->Is(); auto is_sampled = texture->Is(); if ((multisampled && !is_multisampled) || (!multisampled && !is_sampled)) { continue; } if (auto* decorated = var->As()) { BindingDecoration* binding = nullptr; SetDecoration* set = nullptr; for (auto* deco : decorated->decorations()) { if (auto* b = deco->As()) { binding = b; } else if (auto* s = deco->As()) { set = s; } } if (binding == nullptr || set == nullptr) { continue; } ret.push_back({var, BindingInfo{binding, set}}); } } return ret; } } // namespace ast } // namespace tint