//===--- Legacy.h -- Adapters for ExecutionEngine API interop ---*- C++ -*-===// // // 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 // //===----------------------------------------------------------------------===// // // Contains core ORC APIs. // //===----------------------------------------------------------------------===// #ifndef LLVM_EXECUTIONENGINE_ORC_LEGACY_H #define LLVM_EXECUTIONENGINE_ORC_LEGACY_H #include "llvm/ExecutionEngine/JITSymbol.h" #include "llvm/ExecutionEngine/Orc/Core.h" namespace llvm { namespace orc { /// SymbolResolver is a composable interface for looking up symbol flags /// and addresses using the AsynchronousSymbolQuery type. It will /// eventually replace the LegacyJITSymbolResolver interface as the /// stardard ORC symbol resolver type. /// /// FIXME: SymbolResolvers should go away and be replaced with VSOs with /// defenition generators. class SymbolResolver { public: virtual ~SymbolResolver() = default; /// Returns the subset of the given symbols that the caller is responsible for /// materializing. virtual SymbolNameSet getResponsibilitySet(const SymbolNameSet &Symbols) = 0; /// For each symbol in Symbols that can be found, assigns that symbols /// value in Query. Returns the set of symbols that could not be found. virtual SymbolNameSet lookup(std::shared_ptr<AsynchronousSymbolQuery> Query, SymbolNameSet Symbols) = 0; private: virtual void anchor(); }; /// Implements SymbolResolver with a pair of supplied function objects /// for convenience. See createSymbolResolver. template <typename GetResponsibilitySetFn, typename LookupFn> class LambdaSymbolResolver final : public SymbolResolver { public: template <typename GetResponsibilitySetFnRef, typename LookupFnRef> LambdaSymbolResolver(GetResponsibilitySetFnRef &&GetResponsibilitySet, LookupFnRef &&Lookup) : GetResponsibilitySet( std::forward<GetResponsibilitySetFnRef>(GetResponsibilitySet)), Lookup(std::forward<LookupFnRef>(Lookup)) {} SymbolNameSet getResponsibilitySet(const SymbolNameSet &Symbols) final { return GetResponsibilitySet(Symbols); } SymbolNameSet lookup(std::shared_ptr<AsynchronousSymbolQuery> Query, SymbolNameSet Symbols) final { return Lookup(std::move(Query), std::move(Symbols)); } private: GetResponsibilitySetFn GetResponsibilitySet; LookupFn Lookup; }; /// Creates a SymbolResolver implementation from the pair of supplied /// function objects. template <typename GetResponsibilitySetFn, typename LookupFn> std::unique_ptr<LambdaSymbolResolver< typename std::remove_cv< typename std::remove_reference<GetResponsibilitySetFn>::type>::type, typename std::remove_cv< typename std::remove_reference<LookupFn>::type>::type>> createSymbolResolver(GetResponsibilitySetFn &&GetResponsibilitySet, LookupFn &&Lookup) { using LambdaSymbolResolverImpl = LambdaSymbolResolver< typename std::remove_cv< typename std::remove_reference<GetResponsibilitySetFn>::type>::type, typename std::remove_cv< typename std::remove_reference<LookupFn>::type>::type>; return llvm::make_unique<LambdaSymbolResolverImpl>( std::forward<GetResponsibilitySetFn>(GetResponsibilitySet), std::forward<LookupFn>(Lookup)); } /// Legacy adapter. Remove once we kill off the old ORC layers. class JITSymbolResolverAdapter : public JITSymbolResolver { public: JITSymbolResolverAdapter(ExecutionSession &ES, SymbolResolver &R, MaterializationResponsibility *MR); Expected<LookupSet> getResponsibilitySet(const LookupSet &Symbols) override; void lookup(const LookupSet &Symbols, OnResolvedFunction OnResolved) override; private: ExecutionSession &ES; std::set<SymbolStringPtr> ResolvedStrings; SymbolResolver &R; MaterializationResponsibility *MR; }; /// Use the given legacy-style FindSymbol function (i.e. a function that takes /// a const std::string& or StringRef and returns a JITSymbol) to get the /// subset of symbols that the caller is responsible for materializing. If any /// JITSymbol returned by FindSymbol is in an error state the function returns /// immediately with that error. /// /// Useful for implementing getResponsibilitySet bodies that query legacy /// resolvers. template <typename FindSymbolFn> Expected<SymbolNameSet> getResponsibilitySetWithLegacyFn(const SymbolNameSet &Symbols, FindSymbolFn FindSymbol) { SymbolNameSet Result; for (auto &S : Symbols) { if (JITSymbol Sym = FindSymbol(*S)) { if (!Sym.getFlags().isStrong()) Result.insert(S); } else if (auto Err = Sym.takeError()) return std::move(Err); } return Result; } /// Use the given legacy-style FindSymbol function (i.e. a function that /// takes a const std::string& or StringRef and returns a JITSymbol) to /// find the address and flags for each symbol in Symbols and store the /// result in Query. If any JITSymbol returned by FindSymbol is in an /// error then Query.notifyFailed(...) is called with that error and the /// function returns immediately. On success, returns the set of symbols /// not found. /// /// Useful for implementing lookup bodies that query legacy resolvers. template <typename FindSymbolFn> SymbolNameSet lookupWithLegacyFn(ExecutionSession &ES, AsynchronousSymbolQuery &Query, const SymbolNameSet &Symbols, FindSymbolFn FindSymbol) { SymbolNameSet SymbolsNotFound; bool NewSymbolsResolved = false; for (auto &S : Symbols) { if (JITSymbol Sym = FindSymbol(*S)) { if (auto Addr = Sym.getAddress()) { Query.resolve(S, JITEvaluatedSymbol(*Addr, Sym.getFlags())); Query.notifySymbolReady(); NewSymbolsResolved = true; } else { ES.legacyFailQuery(Query, Addr.takeError()); return SymbolNameSet(); } } else if (auto Err = Sym.takeError()) { ES.legacyFailQuery(Query, std::move(Err)); return SymbolNameSet(); } else SymbolsNotFound.insert(S); } if (NewSymbolsResolved && Query.isFullyResolved()) Query.handleFullyResolved(); if (NewSymbolsResolved && Query.isFullyReady()) Query.handleFullyReady(); return SymbolsNotFound; } /// An ORC SymbolResolver implementation that uses a legacy /// findSymbol-like function to perform lookup; template <typename LegacyLookupFn> class LegacyLookupFnResolver final : public SymbolResolver { public: using ErrorReporter = std::function<void(Error)>; LegacyLookupFnResolver(ExecutionSession &ES, LegacyLookupFn LegacyLookup, ErrorReporter ReportError) : ES(ES), LegacyLookup(std::move(LegacyLookup)), ReportError(std::move(ReportError)) {} SymbolNameSet getResponsibilitySet(const SymbolNameSet &Symbols) final { if (auto ResponsibilitySet = getResponsibilitySetWithLegacyFn(Symbols, LegacyLookup)) return std::move(*ResponsibilitySet); else { ReportError(ResponsibilitySet.takeError()); return SymbolNameSet(); } } SymbolNameSet lookup(std::shared_ptr<AsynchronousSymbolQuery> Query, SymbolNameSet Symbols) final { return lookupWithLegacyFn(ES, *Query, Symbols, LegacyLookup); } private: ExecutionSession &ES; LegacyLookupFn LegacyLookup; ErrorReporter ReportError; }; template <typename LegacyLookupFn> std::shared_ptr<LegacyLookupFnResolver<LegacyLookupFn>> createLegacyLookupResolver(ExecutionSession &ES, LegacyLookupFn LegacyLookup, std::function<void(Error)> ErrorReporter) { return std::make_shared<LegacyLookupFnResolver<LegacyLookupFn>>( ES, std::move(LegacyLookup), std::move(ErrorReporter)); } } // End namespace orc } // End namespace llvm #endif // LLVM_EXECUTIONENGINE_ORC_LEGACY_H