| // Copyright 2023 The Dawn & Tint Authors |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are met: |
| // |
| // 1. Redistributions of source code must retain the above copyright notice, this |
| // list of conditions and the following disclaimer. |
| // |
| // 2. Redistributions in binary form must reproduce the above copyright notice, |
| // this list of conditions and the following disclaimer in the documentation |
| // and/or other materials provided with the distribution. |
| // |
| // 3. Neither the name of the copyright holder nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE |
| // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
| // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
| // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
| // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, |
| // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #include <variant> |
| |
| #include "src/tint/lang/core/ir/construct.h" |
| #include "src/tint/lang/core/ir/control_instruction.h" |
| #include "src/tint/lang/core/ir/core_builtin_call.h" |
| #include "src/tint/lang/core/ir/function.h" |
| #include "src/tint/lang/core/ir/instruction.h" |
| #include "src/tint/lang/core/ir/loop.h" |
| #include "src/tint/lang/core/ir/module.h" |
| #include "src/tint/lang/core/ir/multi_in_block.h" |
| #include "src/tint/lang/core/ir/validator.h" |
| #include "src/tint/lang/core/ir/var.h" |
| #include "src/tint/lang/core/type/matrix.h" |
| #include "src/tint/lang/core/type/pointer.h" |
| #include "src/tint/lang/core/type/scalar.h" |
| #include "src/tint/lang/core/type/struct.h" |
| #include "src/tint/lang/core/type/vector.h" |
| #include "src/tint/lang/wgsl/writer/raise/rename_conflicts.h" |
| #include "src/tint/utils/containers/hashset.h" |
| #include "src/tint/utils/containers/reverse.h" |
| #include "src/tint/utils/containers/scope_stack.h" |
| #include "src/tint/utils/macros/defer.h" |
| #include "src/tint/utils/rtti/switch.h" |
| #include "src/tint/utils/text/string.h" |
| |
| namespace tint::wgsl::writer::raise { |
| |
| namespace { |
| |
| /// PIMPL state for the transform, for a single function. |
| struct State { |
| /// Constructor |
| /// @param i the IR module |
| explicit State(core::ir::Module& i) : ir(i) {} |
| |
| /// Processes the module, renaming all declarations that would prevent an identifier resolving |
| /// to the correct declaration. |
| void Process() { |
| scopes.Push(Scope{}); |
| TINT_DEFER(scopes.Clear()); |
| |
| RegisterModuleScopeDecls(); |
| |
| // Process the module-scope variable declarations |
| for (auto* inst : *ir.root_block) { |
| Process(inst); |
| } |
| |
| // Process the functions |
| for (core::ir::Function* fn : ir.functions) { |
| scopes.Push(Scope{}); |
| TINT_DEFER(scopes.Pop()); |
| for (auto* param : fn->Params()) { |
| EnsureResolvable(param->Type()); |
| if (auto symbol = ir.NameOf(param); symbol.IsValid()) { |
| Declare(scopes.Back(), param, symbol.NameView()); |
| } |
| } |
| Process(fn->Block()); |
| } |
| } |
| |
| private: |
| /// Map of identifier to declaration. |
| /// The declarations may be one of an core::ir::Value or core::type::Struct. |
| using Scope = Hashmap<std::string_view, CastableBase*, 8>; |
| |
| /// The IR module. |
| core::ir::Module& ir; |
| |
| /// Stack of scopes |
| Vector<Scope, 8> scopes{}; |
| |
| /// Registers all the WGSL module-scope declarations in the root-scope. |
| /// Duplicate declarations with the same name will renamed. |
| void RegisterModuleScopeDecls() { |
| // Declare all the user types |
| for (auto* ty : ir.Types()) { |
| if (auto* str = ty->As<core::type::Struct>()) { |
| auto name = str->Name().NameView(); |
| if (!IsBuiltinStruct(str)) { |
| Declare(scopes.Front(), const_cast<core::type::Struct*>(str), name); |
| } |
| } |
| } |
| |
| // Declare all the module-scope vars |
| for (auto* inst : *ir.root_block) { |
| for (auto* result : inst->Results()) { |
| if (auto symbol = ir.NameOf(result)) { |
| Declare(scopes.Front(), result, symbol.NameView()); |
| } |
| } |
| } |
| |
| // Declare all the functions |
| for (core::ir::Function* fn : ir.functions) { |
| if (auto symbol = ir.NameOf(fn); symbol.IsValid()) { |
| Declare(scopes.Back(), fn, symbol.NameView()); |
| } |
| } |
| } |
| |
| /// Processes the instructions of the block |
| void Process(core::ir::Block* block) { |
| for (auto* inst : *block) { |
| Process(inst); |
| } |
| } |
| |
| /// Processes an instruction, ensuring that all identifier references resolve to the correct |
| /// declaration. This may involve renaming of declarations in the outer scopes. |
| void Process(core::ir::Instruction* inst) { |
| // Check resolving of operands |
| for (auto* operand : inst->Operands()) { |
| if (operand) { |
| // Ensure that named operands can be resolved. |
| if (auto symbol = ir.NameOf(operand)) { |
| EnsureResolvesTo(symbol.NameView(), operand); |
| } |
| // If the operand is a constant, then ensure that type name can be resolved. |
| if (auto* c = operand->As<core::ir::Constant>()) { |
| EnsureResolvable(c->Type()); |
| } |
| } |
| } |
| |
| Switch( |
| inst, // |
| [&](core::ir::Loop* loop) { |
| // Initializer's scope encompasses the body and continuing |
| scopes.Push(Scope{}); |
| TINT_DEFER(scopes.Pop()); |
| Process(loop->Initializer()); |
| { |
| // Body's scope encompasses the continuing |
| scopes.Push(Scope{}); |
| TINT_DEFER(scopes.Pop()); |
| Process(loop->Body()); |
| { |
| scopes.Push(Scope{}); |
| TINT_DEFER(scopes.Pop()); |
| Process(loop->Continuing()); |
| } |
| } |
| }, |
| [&](core::ir::ControlInstruction* ctrl) { |
| // Traverse into the control instruction's blocks |
| ctrl->ForeachBlock([&](core::ir::Block* block) { |
| scopes.Push(Scope{}); |
| TINT_DEFER(scopes.Pop()); |
| Process(block); |
| }); |
| }, |
| [&](core::ir::Var*) { |
| // Ensure the var's type is resolvable |
| EnsureResolvable(inst->Result(0)->Type()); |
| }, |
| [&](core::ir::Construct*) { |
| // Ensure the type of a type constructor is resolvable |
| EnsureResolvable(inst->Result(0)->Type()); |
| }, |
| [&](core::ir::CoreBuiltinCall* call) { |
| // Ensure builtin of a builtin call is resolvable |
| auto name = tint::ToString(call->Func()); |
| EnsureResolvesTo(name, nullptr); |
| }); |
| |
| // Register new operands and check their types can resolve |
| for (auto* result : inst->Results()) { |
| if (auto symbol = ir.NameOf(result); symbol.IsValid()) { |
| Declare(scopes.Back(), result, symbol.NameView()); |
| } |
| } |
| } |
| |
| /// Ensures that the type @p type can be resolved given its identifier(s) |
| void EnsureResolvable(const core::type::Type* type) { |
| while (type) { |
| type = tint::Switch( |
| type, // |
| [&](const core::type::Scalar* s) { |
| EnsureResolvesTo(s->FriendlyName(), nullptr); |
| return nullptr; |
| }, |
| [&](const core::type::Vector* v) { |
| EnsureResolvesTo("vec" + tint::ToString(v->Width()), nullptr); |
| return v->type(); |
| }, |
| [&](const core::type::Matrix* m) { |
| EnsureResolvesTo( |
| "mat" + tint::ToString(m->columns()) + "x" + tint::ToString(m->rows()), |
| nullptr); |
| return m->type(); |
| }, |
| [&](const core::type::Pointer* p) { |
| EnsureResolvesTo(tint::ToString(p->Access()), nullptr); |
| EnsureResolvesTo(tint::ToString(p->AddressSpace()), nullptr); |
| return p->StoreType(); |
| }, |
| [&](const core::type::Struct* s) { |
| auto name = s->Name().NameView(); |
| if (IsBuiltinStruct(s)) { |
| EnsureResolvesTo(name, nullptr); |
| } else { |
| EnsureResolvesTo(name, s); |
| } |
| return nullptr; |
| }); |
| } |
| } |
| |
| /// Ensures that the identifier @p identifier resolves to the declaration @p thing |
| void EnsureResolvesTo(std::string_view identifier, const CastableBase* thing) { |
| for (auto& scope : tint::Reverse(scopes)) { |
| if (auto decl = scope.Get(identifier)) { |
| if (*decl == thing) { |
| return; // Resolved to the right thing. |
| } |
| |
| // Operand is shadowed |
| scope.Remove(identifier); |
| Rename(*decl, identifier); |
| } |
| } |
| } |
| |
| /// Registers the declaration @p thing in the scope @p scope with the name @p name |
| /// If there is an existing declaration with the given name in @p scope then @p thing will be |
| /// renamed. |
| void Declare(Scope& scope, CastableBase* thing, std::string_view name) { |
| auto add = scope.Add(name, thing); |
| if (!add && add.value != thing) { |
| // Multiple declarations with the same name in the same scope. |
| // Rename the later declaration. |
| Rename(thing, name); |
| } |
| } |
| |
| /// Rename changes the name of @p thing with the old name of @p old_name |
| void Rename(CastableBase* thing, std::string_view old_name) { |
| Symbol new_name = ir.symbols.New(old_name); |
| Switch( |
| thing, // |
| [&](core::ir::Value* value) { ir.SetName(value, new_name); }, |
| [&](core::type::Struct* str) { str->SetName(new_name); }, // |
| TINT_ICE_ON_NO_MATCH); |
| } |
| |
| /// @return true if @p s is a builtin (non-user declared) structure. |
| bool IsBuiltinStruct(const core::type::Struct* s) { |
| // TODO(bclayton): Need to do better than this. |
| return tint::HasPrefix(s->Name().NameView(), "_"); |
| } |
| }; |
| |
| } // namespace |
| |
| Result<SuccessType> RenameConflicts(core::ir::Module& ir) { |
| auto result = ValidateAndDumpIfNeeded(ir, "RenameConflicts transform"); |
| if (result != Success) { |
| return result; |
| } |
| |
| State{ir}.Process(); |
| |
| return Success; |
| } |
| |
| } // namespace tint::wgsl::writer::raise |