//===- Translation.cpp - Translation registry -----------------------------===// // // 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 // //===----------------------------------------------------------------------===// // // Definitions of the translation registry. // //===----------------------------------------------------------------------===// #include "mlir/Tools/mlir-translate/Translation.h" #include "mlir/IR/AsmState.h" #include "mlir/IR/BuiltinOps.h" #include "mlir/IR/Dialect.h" #include "mlir/IR/Verifier.h" #include "mlir/Parser/Parser.h" #include "mlir/Tools/ParseUtilties.h" #include "llvm/Support/SourceMgr.h" using namespace mlir; //===----------------------------------------------------------------------===// // Translation Registry //===----------------------------------------------------------------------===// struct TranslationBundle { TranslateFunction translateFunction; StringRef translateDescription; }; /// Get the mutable static map between registered file-to-file MLIR translations /// and TranslateFunctions with its description that perform those translations. static llvm::StringMap &getTranslationRegistry() { static llvm::StringMap translationBundle; return translationBundle; } /// Register the given translation. static void registerTranslation(StringRef name, StringRef description, const TranslateFunction &function) { auto &translationRegistry = getTranslationRegistry(); if (translationRegistry.find(name) != translationRegistry.end()) llvm::report_fatal_error( "Attempting to overwrite an existing function"); assert(function && "Attempting to register an empty translate function"); translationRegistry[name].translateFunction = function; translationRegistry[name].translateDescription = description; } TranslateRegistration::TranslateRegistration( StringRef name, StringRef description, const TranslateFunction &function) { registerTranslation(name, description, function); } //===----------------------------------------------------------------------===// // Translation to MLIR //===----------------------------------------------------------------------===// // Puts `function` into the to-MLIR translation registry unless there is already // a function registered for the same name. static void registerTranslateToMLIRFunction( StringRef name, StringRef description, const TranslateSourceMgrToMLIRFunction &function) { auto wrappedFn = [function](llvm::SourceMgr &sourceMgr, raw_ostream &output, MLIRContext *context) { OwningOpRef op = function(sourceMgr, context); if (!op || failed(verify(*op))) return failure(); op.get()->print(output); return success(); }; registerTranslation(name, description, wrappedFn); } TranslateToMLIRRegistration::TranslateToMLIRRegistration( StringRef name, StringRef description, const TranslateSourceMgrToMLIRFunction &function) { registerTranslateToMLIRFunction(name, description, function); } /// Wraps `function` with a lambda that extracts a StringRef from a source /// manager and registers the wrapper lambda as a to-MLIR conversion. TranslateToMLIRRegistration::TranslateToMLIRRegistration( StringRef name, StringRef description, const TranslateStringRefToMLIRFunction &function) { registerTranslateToMLIRFunction( name, description, [function](llvm::SourceMgr &sourceMgr, MLIRContext *ctx) { const llvm::MemoryBuffer *buffer = sourceMgr.getMemoryBuffer(sourceMgr.getMainFileID()); return function(buffer->getBuffer(), ctx); }); } //===----------------------------------------------------------------------===// // Translation from MLIR //===----------------------------------------------------------------------===// TranslateFromMLIRRegistration::TranslateFromMLIRRegistration( StringRef name, StringRef description, const TranslateFromMLIRFunction &function, const std::function &dialectRegistration) { static llvm::cl::opt noImplicitModule{ "no-implicit-module", llvm::cl::desc("Disable the parsing of an implicit top-level module op"), llvm::cl::init(false)}; registerTranslation(name, description, [function, dialectRegistration]( llvm::SourceMgr &sourceMgr, raw_ostream &output, MLIRContext *context) { DialectRegistry registry; dialectRegistration(registry); context->appendDialectRegistry(registry); OwningOpRef op = parseSourceFileForTool( sourceMgr, context, !noImplicitModule); if (!op || failed(verify(*op))) return failure(); return function(op.get(), output); }); } //===----------------------------------------------------------------------===// // Translation Parser //===----------------------------------------------------------------------===// TranslationParser::TranslationParser(llvm::cl::Option &opt) : llvm::cl::parser(opt) { for (const auto &kv : getTranslationRegistry()) { addLiteralOption(kv.first(), &kv.second.translateFunction, kv.second.translateDescription); } } void TranslationParser::printOptionInfo(const llvm::cl::Option &o, size_t globalWidth) const { TranslationParser *tp = const_cast(this); llvm::array_pod_sort(tp->Values.begin(), tp->Values.end(), [](const TranslationParser::OptionInfo *lhs, const TranslationParser::OptionInfo *rhs) { return lhs->Name.compare(rhs->Name); }); llvm::cl::parser::printOptionInfo(o, globalWidth); }