//===-- Mangler.cpp -------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "flang/Lower/Mangler.h" #include "flang/Common/reference.h" #include "flang/Lower/Support/Utils.h" #include "flang/Optimizer/Builder/Todo.h" #include "flang/Optimizer/Dialect/FIRType.h" #include "flang/Optimizer/Support/InternalNames.h" #include "flang/Semantics/tools.h" #include "llvm/ADT/ArrayRef.h" #include "llvm/ADT/SmallVector.h" #include "llvm/ADT/StringRef.h" #include "llvm/ADT/Twine.h" #include "llvm/Support/MD5.h" #include // recursively build the vector of module scopes static void moduleNames(const Fortran::semantics::Scope &scope, llvm::SmallVector &result) { if (scope.IsTopLevel()) return; moduleNames(scope.parent(), result); if (scope.kind() == Fortran::semantics::Scope::Kind::Module) if (const Fortran::semantics::Symbol *symbol = scope.symbol()) result.emplace_back(toStringRef(symbol->name())); } static llvm::SmallVector moduleNames(const Fortran::semantics::Symbol &symbol) { const Fortran::semantics::Scope &scope = symbol.owner(); llvm::SmallVector result; moduleNames(scope, result); return result; } static std::optional hostName(const Fortran::semantics::Symbol &symbol) { const Fortran::semantics::Scope &scope = symbol.owner(); if (scope.kind() == Fortran::semantics::Scope::Kind::Subprogram) { assert(scope.symbol() && "subprogram scope must have a symbol"); return toStringRef(scope.symbol()->name()); } if (scope.kind() == Fortran::semantics::Scope::Kind::MainProgram) // Do not use the main program name, if any, because it may lead to name // collision with procedures with the same name in other compilation units // (technically illegal, but all compilers are able to compile and link // properly these programs). return llvm::StringRef(""); return {}; } // Mangle the name of `symbol` to make it unique within FIR's symbol table using // the FIR name mangler, `mangler` std::string Fortran::lower::mangle::mangleName(const Fortran::semantics::Symbol &symbol, bool keepExternalInScope) { // Resolve host and module association before mangling const auto &ultimateSymbol = symbol.GetUltimate(); auto symbolName = toStringRef(ultimateSymbol.name()); // The Fortran and BIND(C) namespaces are counterintuitive. A // BIND(C) name is substituted early having precedence over the // Fortran name of the subprogram. By side-effect, this allows // multiple subprocedures with identical Fortran names to be legally // present in the program. Assume the BIND(C) name is unique. if (auto *overrideName = ultimateSymbol.GetBindName()) return *overrideName; // TODO: the case of procedure that inherits the BIND(C) through another // interface (procedure(iface)), should be dealt within GetBindName() // directly, or some semantics wrapper. if (!Fortran::semantics::IsPointer(ultimateSymbol) && Fortran::semantics::IsBindCProcedure(ultimateSymbol) && Fortran::semantics::ClassifyProcedure(symbol) != Fortran::semantics::ProcedureDefinitionClass::Internal) return ultimateSymbol.name().ToString(); return std::visit( Fortran::common::visitors{ [&](const Fortran::semantics::MainProgramDetails &) { return fir::NameUniquer::doProgramEntry().str(); }, [&](const Fortran::semantics::SubprogramDetails &subpDetails) { // Mangle external procedure without any scope prefix. if (!keepExternalInScope && Fortran::semantics::IsExternal(ultimateSymbol)) return fir::NameUniquer::doProcedure(std::nullopt, std::nullopt, symbolName); // A separate module procedure must be mangled according to its // declaration scope, not its definition scope. const Fortran::semantics::Symbol *interface = &ultimateSymbol; if (interface->attrs().test(Fortran::semantics::Attr::MODULE) && interface->owner().IsSubmodule() && !subpDetails.isInterface()) interface = subpDetails.moduleInterface(); assert(interface && "Separate module procedure must be declared"); llvm::SmallVector modNames = moduleNames(*interface); return fir::NameUniquer::doProcedure(modNames, hostName(*interface), symbolName); }, [&](const Fortran::semantics::ProcEntityDetails &) { // Mangle procedure pointers and dummy procedures as variables if (Fortran::semantics::IsPointer(ultimateSymbol) || Fortran::semantics::IsDummy(ultimateSymbol)) return fir::NameUniquer::doVariable(moduleNames(ultimateSymbol), hostName(ultimateSymbol), symbolName); // Otherwise, this is an external procedure, even if it does not // have an explicit EXTERNAL attribute. Mangle it without any // prefix. return fir::NameUniquer::doProcedure(std::nullopt, std::nullopt, symbolName); }, [&](const Fortran::semantics::ObjectEntityDetails &) { llvm::SmallVector modNames = moduleNames(ultimateSymbol); std::optional optHost = hostName(ultimateSymbol); if (Fortran::semantics::IsNamedConstant(ultimateSymbol)) return fir::NameUniquer::doConstant(modNames, optHost, symbolName); return fir::NameUniquer::doVariable(modNames, optHost, symbolName); }, [&](const Fortran::semantics::NamelistDetails &) { llvm::SmallVector modNames = moduleNames(ultimateSymbol); std::optional optHost = hostName(ultimateSymbol); return fir::NameUniquer::doNamelistGroup(modNames, optHost, symbolName); }, [&](const Fortran::semantics::CommonBlockDetails &) { return fir::NameUniquer::doCommonBlock(symbolName); }, [&](const Fortran::semantics::DerivedTypeDetails &) -> std::string { // Derived type mangling must used mangleName(DerivedTypeSpec&) so // that kind type parameter values can be mangled. llvm::report_fatal_error( "only derived type instances can be mangled"); }, [&](const Fortran::semantics::ProcBindingDetails &procBinding) -> std::string { return mangleName(procBinding.symbol(), keepExternalInScope); }, [](const auto &) -> std::string { TODO_NOLOC("symbol mangling"); }, }, ultimateSymbol.details()); } std::string Fortran::lower::mangle::mangleName( const Fortran::semantics::DerivedTypeSpec &derivedType) { // Resolve host and module association before mangling const Fortran::semantics::Symbol &ultimateSymbol = derivedType.typeSymbol().GetUltimate(); llvm::StringRef symbolName = toStringRef(ultimateSymbol.name()); llvm::SmallVector modNames = moduleNames(ultimateSymbol); std::optional optHost = hostName(ultimateSymbol); llvm::SmallVector kinds; for (const auto ¶m : Fortran::semantics::OrderParameterDeclarations(ultimateSymbol)) { const auto ¶mDetails = param->get(); if (paramDetails.attr() == Fortran::common::TypeParamAttr::Kind) { const Fortran::semantics::ParamValue *paramValue = derivedType.FindParameter(param->name()); assert(paramValue && "derived type kind parameter value not found"); const Fortran::semantics::MaybeIntExpr paramExpr = paramValue->GetExplicit(); assert(paramExpr && "derived type kind param not explicit"); std::optional init = Fortran::evaluate::ToInt64(paramValue->GetExplicit()); assert(init && "derived type kind param is not constant"); kinds.emplace_back(*init); } } return fir::NameUniquer::doType(modNames, optHost, symbolName, kinds); } std::string Fortran::lower::mangle::demangleName(llvm::StringRef name) { auto result = fir::NameUniquer::deconstruct(name); return result.second.name; } //===----------------------------------------------------------------------===// // Array Literals Mangling //===----------------------------------------------------------------------===// static std::string typeToString(Fortran::common::TypeCategory cat, int kind, llvm::StringRef derivedName) { switch (cat) { case Fortran::common::TypeCategory::Integer: return "i" + std::to_string(kind); case Fortran::common::TypeCategory::Real: return "r" + std::to_string(kind); case Fortran::common::TypeCategory::Complex: return "z" + std::to_string(kind); case Fortran::common::TypeCategory::Logical: return "l" + std::to_string(kind); case Fortran::common::TypeCategory::Character: return "c" + std::to_string(kind); case Fortran::common::TypeCategory::Derived: return derivedName.str(); } llvm_unreachable("bad TypeCategory"); } std::string Fortran::lower::mangle::mangleArrayLiteral( const uint8_t *addr, size_t size, const Fortran::evaluate::ConstantSubscripts &shape, Fortran::common::TypeCategory cat, int kind, Fortran::common::ConstantSubscript charLen, llvm::StringRef derivedName) { std::string typeId; for (Fortran::evaluate::ConstantSubscript extent : shape) typeId.append(std::to_string(extent)).append("x"); if (charLen >= 0) typeId.append(std::to_string(charLen)).append("x"); typeId.append(typeToString(cat, kind, derivedName)); std::string name = fir::NameUniquer::doGenerated("ro."s.append(typeId).append(".")); if (!size) return name += "null"; llvm::MD5 hashValue{}; hashValue.update(llvm::ArrayRef{addr, size}); llvm::MD5::MD5Result hashResult; hashValue.final(hashResult); llvm::SmallString<32> hashString; llvm::MD5::stringifyResult(hashResult, hashString); return name += hashString.c_str(); } //===----------------------------------------------------------------------===// // Intrinsic Procedure Mangling //===----------------------------------------------------------------------===// /// Helper to encode type into string for intrinsic procedure names. /// Note: mlir has Type::dump(ostream) methods but it may add "!" that is not /// suitable for function names. static std::string typeToString(mlir::Type t) { if (auto refT{t.dyn_cast()}) return "ref_" + typeToString(refT.getEleTy()); if (auto i{t.dyn_cast()}) { return "i" + std::to_string(i.getWidth()); } if (auto cplx{t.dyn_cast()}) { return "z" + std::to_string(cplx.getFKind()); } if (auto real{t.dyn_cast()}) { return "r" + std::to_string(real.getFKind()); } if (auto f{t.dyn_cast()}) { return "f" + std::to_string(f.getWidth()); } if (auto logical{t.dyn_cast()}) { return "l" + std::to_string(logical.getFKind()); } if (auto character{t.dyn_cast()}) { return "c" + std::to_string(character.getFKind()); } if (auto boxCharacter{t.dyn_cast()}) { return "bc" + std::to_string(boxCharacter.getEleTy().getFKind()); } llvm_unreachable("no mangling for type"); } std::string fir::mangleIntrinsicProcedure(llvm::StringRef intrinsic, mlir::FunctionType funTy) { std::string name = "fir."; name.append(intrinsic.str()).append("."); assert(funTy.getNumResults() == 1 && "only function mangling supported"); name.append(typeToString(funTy.getResult(0))); unsigned e = funTy.getNumInputs(); for (decltype(e) i = 0; i < e; ++i) name.append(".").append(typeToString(funTy.getInput(i))); return name; } std::string Fortran::lower::mangle::globalNamelistDescriptorName( const Fortran::semantics::Symbol &sym) { std::string name = mangleName(sym); return IsAllocatableOrPointer(sym) ? name : name + ".desc"s; }