//===- ExternalNameConversion.cpp -- convert name with external convention ===// // // 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/Common/Fortran.h" #include "flang/Optimizer/Dialect/FIRDialect.h" #include "flang/Optimizer/Dialect/FIROps.h" #include "flang/Optimizer/Dialect/FIROpsSupport.h" #include "flang/Optimizer/Support/InternalNames.h" #include "flang/Optimizer/Transforms/Passes.h" #include "mlir/IR/Attributes.h" #include "mlir/IR/SymbolTable.h" #include "mlir/Pass/Pass.h" namespace fir { #define GEN_PASS_DEF_EXTERNALNAMECONVERSION #include "flang/Optimizer/Transforms/Passes.h.inc" } // namespace fir using namespace mlir; //===----------------------------------------------------------------------===// // Helper functions //===----------------------------------------------------------------------===// /// Mangle the name with gfortran convention. std::string mangleExternalName(const std::pair result, bool appendUnderscore) { if (result.first == fir::NameUniquer::NameKind::COMMON && result.second.name.empty()) return Fortran::common::blankCommonObjectName; return Fortran::common::GetExternalAssemblyName(result.second.name, appendUnderscore); } namespace { class ExternalNameConversionPass : public fir::impl::ExternalNameConversionBase { public: ExternalNameConversionPass(bool appendUnderscoring) : appendUnderscores(appendUnderscoring) {} ExternalNameConversionPass() { usePassOpt = true; } mlir::ModuleOp getModule() { return getOperation(); } void runOnOperation() override; private: bool appendUnderscores; bool usePassOpt = false; }; } // namespace void ExternalNameConversionPass::runOnOperation() { auto op = getOperation(); auto *context = &getContext(); appendUnderscores = (usePassOpt) ? appendUnderscoreOpt : appendUnderscores; llvm::DenseMap remappings; // Update names of external Fortran functions and names of Common Block // globals. for (auto &funcOrGlobal : op->getRegion(0).front()) { if (llvm::isa(funcOrGlobal) || llvm::isa(funcOrGlobal)) { auto symName = funcOrGlobal.getAttrOfType( mlir::SymbolTable::getSymbolAttrName()); auto deconstructedName = fir::NameUniquer::deconstruct(symName); if (fir::NameUniquer::isExternalFacingUniquedName(deconstructedName)) { auto newName = mangleExternalName(deconstructedName, appendUnderscores); auto newAttr = mlir::StringAttr::get(context, newName); mlir::SymbolTable::setSymbolName(&funcOrGlobal, newAttr); auto newSymRef = mlir::FlatSymbolRefAttr::get(newAttr); remappings.try_emplace(symName, newSymRef); if (llvm::isa(funcOrGlobal)) funcOrGlobal.setAttr(fir::getInternalFuncNameAttrName(), symName); } } } if (remappings.empty()) return; // Update all uses of the functions and globals that have been renamed. op.walk([&remappings](mlir::Operation *nestedOp) { llvm::SmallVector> updates; for (const mlir::NamedAttribute &attr : nestedOp->getAttrDictionary()) if (auto symRef = llvm::dyn_cast(attr.getValue())) if (auto remap = remappings.find(symRef.getRootReference()); remap != remappings.end()) updates.emplace_back(std::pair{ attr.getName(), mlir::SymbolRefAttr(remap->second)}); for (auto update : updates) nestedOp->setAttr(update.first, update.second); }); } std::unique_ptr fir::createExternalNameConversionPass() { return std::make_unique(); } std::unique_ptr fir::createExternalNameConversionPass(bool appendUnderscoring) { return std::make_unique(appendUnderscoring); }