1 //===------ Interpreter.cpp - Incremental Compilation and Execution -------===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 // 9 // This file implements the component which performs incremental code 10 // compilation and execution. 11 // 12 //===----------------------------------------------------------------------===// 13 14 #include "clang/Interpreter/Interpreter.h" 15 16 #include "IncrementalExecutor.h" 17 #include "IncrementalParser.h" 18 19 #include "clang/AST/ASTContext.h" 20 #include "clang/Basic/TargetInfo.h" 21 #include "clang/CodeGen/ModuleBuilder.h" 22 #include "clang/CodeGen/ObjectFilePCHContainerOperations.h" 23 #include "clang/Driver/Compilation.h" 24 #include "clang/Driver/Driver.h" 25 #include "clang/Driver/Job.h" 26 #include "clang/Driver/Options.h" 27 #include "clang/Driver/Tool.h" 28 #include "clang/Frontend/CompilerInstance.h" 29 #include "clang/Frontend/TextDiagnosticBuffer.h" 30 #include "clang/Lex/PreprocessorOptions.h" 31 32 #include "llvm/IR/Module.h" 33 #include "llvm/Support/Errc.h" 34 #include "llvm/Support/Host.h" 35 36 using namespace clang; 37 38 // FIXME: Figure out how to unify with namespace init_convenience from 39 // tools/clang-import-test/clang-import-test.cpp 40 namespace { 41 /// Retrieves the clang CC1 specific flags out of the compilation's jobs. 42 /// \returns NULL on error. 43 static llvm::Expected<const llvm::opt::ArgStringList *> 44 GetCC1Arguments(DiagnosticsEngine *Diagnostics, 45 driver::Compilation *Compilation) { 46 // We expect to get back exactly one Command job, if we didn't something 47 // failed. Extract that job from the Compilation. 48 const driver::JobList &Jobs = Compilation->getJobs(); 49 if (!Jobs.size() || !isa<driver::Command>(*Jobs.begin())) 50 return llvm::createStringError(llvm::errc::not_supported, 51 "Driver initialization failed. " 52 "Unable to create a driver job"); 53 54 // The one job we find should be to invoke clang again. 55 const driver::Command *Cmd = cast<driver::Command>(&(*Jobs.begin())); 56 if (llvm::StringRef(Cmd->getCreator().getName()) != "clang") 57 return llvm::createStringError(llvm::errc::not_supported, 58 "Driver initialization failed"); 59 60 return &Cmd->getArguments(); 61 } 62 63 static llvm::Expected<std::unique_ptr<CompilerInstance>> 64 CreateCI(const llvm::opt::ArgStringList &Argv) { 65 std::unique_ptr<CompilerInstance> Clang(new CompilerInstance()); 66 IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); 67 68 // Register the support for object-file-wrapped Clang modules. 69 // FIXME: Clang should register these container operations automatically. 70 auto PCHOps = Clang->getPCHContainerOperations(); 71 PCHOps->registerWriter(std::make_unique<ObjectFilePCHContainerWriter>()); 72 PCHOps->registerReader(std::make_unique<ObjectFilePCHContainerReader>()); 73 74 // Buffer diagnostics from argument parsing so that we can output them using 75 // a well formed diagnostic object. 76 IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); 77 TextDiagnosticBuffer *DiagsBuffer = new TextDiagnosticBuffer; 78 DiagnosticsEngine Diags(DiagID, &*DiagOpts, DiagsBuffer); 79 bool Success = CompilerInvocation::CreateFromArgs( 80 Clang->getInvocation(), llvm::makeArrayRef(Argv.begin(), Argv.size()), 81 Diags); 82 83 // Infer the builtin include path if unspecified. 84 if (Clang->getHeaderSearchOpts().UseBuiltinIncludes && 85 Clang->getHeaderSearchOpts().ResourceDir.empty()) 86 Clang->getHeaderSearchOpts().ResourceDir = 87 CompilerInvocation::GetResourcesPath(Argv[0], nullptr); 88 89 // Create the actual diagnostics engine. 90 Clang->createDiagnostics(); 91 if (!Clang->hasDiagnostics()) 92 return llvm::createStringError(llvm::errc::not_supported, 93 "Initialization failed. " 94 "Unable to create diagnostics engine"); 95 96 DiagsBuffer->FlushDiagnostics(Clang->getDiagnostics()); 97 if (!Success) 98 return llvm::createStringError(llvm::errc::not_supported, 99 "Initialization failed. " 100 "Unable to flush diagnostics"); 101 102 // FIXME: Merge with CompilerInstance::ExecuteAction. 103 llvm::MemoryBuffer *MB = llvm::MemoryBuffer::getMemBuffer("").release(); 104 Clang->getPreprocessorOpts().addRemappedFile("<<< inputs >>>", MB); 105 106 Clang->setTarget(TargetInfo::CreateTargetInfo( 107 Clang->getDiagnostics(), Clang->getInvocation().TargetOpts)); 108 if (!Clang->hasTarget()) 109 return llvm::createStringError(llvm::errc::not_supported, 110 "Initialization failed. " 111 "Target is missing"); 112 113 Clang->getTarget().adjust(Clang->getDiagnostics(), Clang->getLangOpts()); 114 115 // Don't clear the AST before backend codegen since we do codegen multiple 116 // times, reusing the same AST. 117 Clang->getCodeGenOpts().ClearASTBeforeBackend = false; 118 119 return std::move(Clang); 120 } 121 122 } // anonymous namespace 123 124 llvm::Expected<std::unique_ptr<CompilerInstance>> 125 IncrementalCompilerBuilder::create(std::vector<const char *> &ClangArgv) { 126 127 // If we don't know ClangArgv0 or the address of main() at this point, try 128 // to guess it anyway (it's possible on some platforms). 129 std::string MainExecutableName = 130 llvm::sys::fs::getMainExecutable(nullptr, nullptr); 131 132 ClangArgv.insert(ClangArgv.begin(), MainExecutableName.c_str()); 133 134 // Prepending -c to force the driver to do something if no action was 135 // specified. By prepending we allow users to override the default 136 // action and use other actions in incremental mode. 137 // FIXME: Print proper driver diagnostics if the driver flags are wrong. 138 ClangArgv.insert(ClangArgv.begin() + 1, "-c"); 139 140 if (!llvm::is_contained(ClangArgv, " -x")) { 141 // We do C++ by default; append right after argv[0] if no "-x" given 142 ClangArgv.push_back("-x"); 143 ClangArgv.push_back("c++"); 144 } 145 146 // Put a dummy C++ file on to ensure there's at least one compile job for the 147 // driver to construct. 148 ClangArgv.push_back("<<< inputs >>>"); 149 150 // Buffer diagnostics from argument parsing so that we can output them using a 151 // well formed diagnostic object. 152 IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); 153 IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = 154 CreateAndPopulateDiagOpts(ClangArgv); 155 TextDiagnosticBuffer *DiagsBuffer = new TextDiagnosticBuffer; 156 DiagnosticsEngine Diags(DiagID, &*DiagOpts, DiagsBuffer); 157 158 driver::Driver Driver(/*MainBinaryName=*/ClangArgv[0], 159 llvm::sys::getProcessTriple(), Diags); 160 Driver.setCheckInputsExist(false); // the input comes from mem buffers 161 llvm::ArrayRef<const char *> RF = llvm::makeArrayRef(ClangArgv); 162 std::unique_ptr<driver::Compilation> Compilation(Driver.BuildCompilation(RF)); 163 164 if (Compilation->getArgs().hasArg(driver::options::OPT_v)) 165 Compilation->getJobs().Print(llvm::errs(), "\n", /*Quote=*/false); 166 167 auto ErrOrCC1Args = GetCC1Arguments(&Diags, Compilation.get()); 168 if (auto Err = ErrOrCC1Args.takeError()) 169 return std::move(Err); 170 171 return CreateCI(**ErrOrCC1Args); 172 } 173 174 Interpreter::Interpreter(std::unique_ptr<CompilerInstance> CI, 175 llvm::Error &Err) { 176 llvm::ErrorAsOutParameter EAO(&Err); 177 auto LLVMCtx = std::make_unique<llvm::LLVMContext>(); 178 TSCtx = std::make_unique<llvm::orc::ThreadSafeContext>(std::move(LLVMCtx)); 179 IncrParser = std::make_unique<IncrementalParser>(std::move(CI), 180 *TSCtx->getContext(), Err); 181 } 182 183 Interpreter::~Interpreter() {} 184 185 llvm::Expected<std::unique_ptr<Interpreter>> 186 Interpreter::create(std::unique_ptr<CompilerInstance> CI) { 187 llvm::Error Err = llvm::Error::success(); 188 auto Interp = 189 std::unique_ptr<Interpreter>(new Interpreter(std::move(CI), Err)); 190 if (Err) 191 return std::move(Err); 192 return std::move(Interp); 193 } 194 195 const CompilerInstance *Interpreter::getCompilerInstance() const { 196 return IncrParser->getCI(); 197 } 198 199 const llvm::orc::LLJIT *Interpreter::getExecutionEngine() const { 200 if (IncrExecutor) 201 return IncrExecutor->getExecutionEngine(); 202 return nullptr; 203 } 204 205 llvm::Expected<PartialTranslationUnit &> 206 Interpreter::Parse(llvm::StringRef Code) { 207 return IncrParser->Parse(Code); 208 } 209 210 llvm::Error Interpreter::Execute(PartialTranslationUnit &T) { 211 assert(T.TheModule); 212 if (!IncrExecutor) { 213 const llvm::Triple &Triple = 214 getCompilerInstance()->getASTContext().getTargetInfo().getTriple(); 215 llvm::Error Err = llvm::Error::success(); 216 IncrExecutor = std::make_unique<IncrementalExecutor>(*TSCtx, Err, Triple); 217 218 if (Err) 219 return Err; 220 } 221 // FIXME: Add a callback to retain the llvm::Module once the JIT is done. 222 if (auto Err = IncrExecutor->addModule(std::move(T.TheModule))) 223 return Err; 224 225 if (auto Err = IncrExecutor->runCtors()) 226 return Err; 227 228 return llvm::Error::success(); 229 } 230 231 llvm::Expected<llvm::JITTargetAddress> 232 Interpreter::getSymbolAddress(GlobalDecl GD) const { 233 if (!IncrExecutor) 234 return llvm::make_error<llvm::StringError>("Operation failed. " 235 "No execution engine", 236 std::error_code()); 237 llvm::StringRef MangledName = IncrParser->GetMangledName(GD); 238 return getSymbolAddress(MangledName); 239 } 240 241 llvm::Expected<llvm::JITTargetAddress> 242 Interpreter::getSymbolAddress(llvm::StringRef IRName) const { 243 if (!IncrExecutor) 244 return llvm::make_error<llvm::StringError>("Operation failed. " 245 "No execution engine", 246 std::error_code()); 247 248 return IncrExecutor->getSymbolAddress(IRName, IncrementalExecutor::IRName); 249 } 250 251 llvm::Expected<llvm::JITTargetAddress> 252 Interpreter::getSymbolAddressFromLinkerName(llvm::StringRef Name) const { 253 if (!IncrExecutor) 254 return llvm::make_error<llvm::StringError>("Operation failed. " 255 "No execution engine", 256 std::error_code()); 257 258 return IncrExecutor->getSymbolAddress(Name, IncrementalExecutor::LinkerName); 259 } 260