| //===- subzero/src/IceBrowserCompileServer.cpp - Browser compile server ---===// |
| // |
| // The Subzero Code Generator |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| /// |
| /// \file |
| /// \brief Defines the browser-based compile server. |
| /// |
| //===----------------------------------------------------------------------===// |
| |
| // Can only compile this with the NaCl compiler (needs irt.h, and the |
| // unsandboxed LLVM build using the trusted compiler does not have irt.h). |
| #include "IceBrowserCompileServer.h" |
| #include "IceRangeSpec.h" |
| |
| #if PNACL_BROWSER_TRANSLATOR |
| |
| // Headers which are not properly part of the SDK are included by their path in |
| // the NaCl tree. |
| #ifdef __pnacl__ |
| #include "native_client/src/untrusted/nacl/pnacl.h" |
| #endif // __pnacl__ |
| |
| #include "llvm/Support/QueueStreamer.h" |
| |
| #include <cstring> |
| #include <fstream> |
| #include <irt.h> |
| #include <irt_dev.h> |
| #include <pthread.h> |
| #include <thread> |
| |
| namespace Ice { |
| |
| // Create C wrappers around callback handlers for the IRT interface. |
| namespace { |
| |
| BrowserCompileServer *gCompileServer; |
| struct nacl_irt_private_pnacl_translator_compile gIRTFuncs; |
| |
| void getIRTInterfaces() { |
| size_t QueryResult = |
| nacl_interface_query(NACL_IRT_PRIVATE_PNACL_TRANSLATOR_COMPILE_v0_1, |
| &gIRTFuncs, sizeof(gIRTFuncs)); |
| if (QueryResult != sizeof(gIRTFuncs)) |
| llvm::report_fatal_error("Failed to get translator compile IRT interface"); |
| } |
| |
| // Allow pnacl-sz arguments to be supplied externally, instead of coming from |
| // the browser. This is meant to be used for debugging. |
| // |
| // NOTE: This functionality is only enabled in non-MINIMAL Subzero builds, for |
| // security/safety reasons. |
| // |
| // If the SZARGFILE environment variable is set to a file name, arguments are |
| // read from that file, one argument per line. This requires setting 3 |
| // environment variables before starting the browser: |
| // |
| // NACL_ENV_PASSTHROUGH=NACL_DANGEROUS_ENABLE_FILE_ACCESS,NACLENV_SZARGFILE |
| // NACL_DANGEROUS_ENABLE_FILE_ACCESS=1 |
| // NACLENV_SZARGFILE=/path/to/myargs.txt |
| // |
| // In addition, Chrome needs to be launched with the "--no-sandbox" argument. |
| // |
| // If the SZARGLIST environment variable is set, arguments are extracted from |
| // that variable's value, separated by the '|' character (being careful to |
| // escape/quote special shell characters). This requires setting 2 environment |
| // variables before starting the browser: |
| // |
| // NACL_ENV_PASSTHROUGH=NACLENV_SZARGLIST |
| // NACLENV_SZARGLIST=arg |
| // |
| // This does not require the "--no-sandbox" argument, and is therefore much |
| // safer, but does require restarting the browser to change the arguments. |
| // |
| // If external arguments are supplied, the browser's NumThreads specification is |
| // ignored, to allow -threads to be specified as an external argument. Note |
| // that the browser normally supplies the "-O2" argument, so externally supplied |
| // arguments might want to provide an explicit -O argument. |
| // |
| // See Chrome's src/components/nacl/zygote/nacl_fork_delegate_linux.cc for the |
| // NACL_ENV_PASSTHROUGH mechanism. |
| // |
| // See NaCl's src/trusted/service_runtime/env_cleanser.c for the NACLENV_ |
| // mechanism. |
| std::vector<std::string> getExternalArgs() { |
| std::vector<std::string> ExternalArgs; |
| if (BuildDefs::minimal()) |
| return ExternalArgs; |
| char ArgsFileVar[] = "SZARGFILE"; |
| char ArgsListVar[] = "SZARGLIST"; |
| if (const char *ArgsFilename = getenv(ArgsFileVar)) { |
| std::ifstream ArgsStream(ArgsFilename); |
| std::string Arg; |
| while (ArgsStream >> std::ws, std::getline(ArgsStream, Arg)) { |
| if (!Arg.empty() && Arg[0] == '#') |
| continue; |
| ExternalArgs.emplace_back(Arg); |
| } |
| if (ExternalArgs.empty()) { |
| llvm::report_fatal_error("Failed to read arguments from file '" + |
| std::string(ArgsFilename) + "'"); |
| } |
| } else if (const char *ArgsList = getenv(ArgsListVar)) { |
| // Leverage the RangeSpec tokenizer. |
| auto Args = RangeSpec::tokenize(ArgsList, '|'); |
| ExternalArgs.insert(ExternalArgs.end(), Args.begin(), Args.end()); |
| } |
| return ExternalArgs; |
| } |
| |
| char *onInitCallback(uint32_t NumThreads, int *ObjFileFDs, |
| size_t ObjFileFDCount, char **CLArgs, size_t CLArgsLen) { |
| if (ObjFileFDCount < 1) { |
| std::string Buffer; |
| llvm::raw_string_ostream StrBuf(Buffer); |
| StrBuf << "Invalid number of FDs for onInitCallback " << ObjFileFDCount |
| << "\n"; |
| return strdup(StrBuf.str().c_str()); |
| } |
| int ObjFileFD = ObjFileFDs[0]; |
| if (ObjFileFD < 0) { |
| std::string Buffer; |
| llvm::raw_string_ostream StrBuf(Buffer); |
| StrBuf << "Invalid FD given for onInitCallback " << ObjFileFD << "\n"; |
| return strdup(StrBuf.str().c_str()); |
| } |
| // CLArgs is almost an "argv", but is missing the argv[0] program name. |
| std::vector<const char *> Argv; |
| constexpr static char ProgramName[] = "pnacl-sz.nexe"; |
| Argv.reserve(CLArgsLen + 1); |
| Argv.push_back(ProgramName); |
| |
| bool UseNumThreadsFromBrowser = true; |
| auto ExternalArgs = getExternalArgs(); |
| if (ExternalArgs.empty()) { |
| for (size_t i = 0; i < CLArgsLen; ++i) { |
| Argv.push_back(CLArgs[i]); |
| } |
| } else { |
| for (auto &Arg : ExternalArgs) { |
| Argv.emplace_back(Arg.c_str()); |
| } |
| UseNumThreadsFromBrowser = false; |
| } |
| // NOTE: strings pointed to by argv are owned by the caller, but we parse |
| // here before returning and don't store them. |
| gCompileServer->getParsedFlags(UseNumThreadsFromBrowser, NumThreads, |
| Argv.size(), Argv.data()); |
| gCompileServer->startCompileThread(ObjFileFD); |
| return nullptr; |
| } |
| |
| int onDataCallback(const void *Data, size_t NumBytes) { |
| return gCompileServer->pushInputBytes(Data, NumBytes) ? 1 : 0; |
| } |
| |
| char *onEndCallback() { |
| gCompileServer->endInputStream(); |
| gCompileServer->waitForCompileThread(); |
| // TODO(jvoung): Also return UMA data. |
| if (gCompileServer->getErrorCode().value()) { |
| const std::string Error = gCompileServer->getErrorStream().getContents(); |
| return strdup(Error.empty() ? "Some error occurred" : Error.c_str()); |
| } |
| return nullptr; |
| } |
| |
| struct nacl_irt_pnacl_compile_funcs SubzeroCallbacks { |
| &onInitCallback, &onDataCallback, &onEndCallback |
| }; |
| |
| std::unique_ptr<llvm::raw_fd_ostream> getOutputStream(int FD) { |
| if (FD <= 0) |
| llvm::report_fatal_error("Invalid output FD"); |
| constexpr bool CloseOnDtor = true; |
| constexpr bool Unbuffered = false; |
| return std::unique_ptr<llvm::raw_fd_ostream>( |
| new llvm::raw_fd_ostream(FD, CloseOnDtor, Unbuffered)); |
| } |
| |
| void fatalErrorHandler(void *UserData, const std::string &Reason, |
| bool GenCrashDialog) { |
| (void)GenCrashDialog; |
| BrowserCompileServer *Server = |
| reinterpret_cast<BrowserCompileServer *>(UserData); |
| Server->setFatalError(Reason); |
| // Only kill the current thread instead of the whole process. We need the |
| // server thread to remain alive in order to respond with the error message. |
| // We could also try to pthread_kill all other worker threads, but |
| // pthread_kill / raising signals is not supported by NaCl. We'll have to |
| // assume that the worker/emitter threads will be well behaved after a fatal |
| // error in other threads, and either get stuck waiting on input from a |
| // previous stage, or also call report_fatal_error. |
| pthread_exit(0); |
| } |
| |
| /// Adapted from pnacl-llc's AddDefaultCPU() in srpc_main.cpp. |
| TargetArch getTargetArch() { |
| #if defined(__pnacl__) |
| switch (__builtin_nacl_target_arch()) { |
| case PnaclTargetArchitectureX86_32: |
| case PnaclTargetArchitectureX86_32_NonSFI: |
| return Target_X8632; |
| case PnaclTargetArchitectureX86_64: |
| return Target_X8664; |
| case PnaclTargetArchitectureARM_32: |
| case PnaclTargetArchitectureARM_32_NonSFI: |
| return Target_ARM32; |
| case PnaclTargetArchitectureMips_32: |
| return Target_MIPS32; |
| default: |
| llvm::report_fatal_error("no target architecture match."); |
| } |
| #elif defined(__i386__) |
| return Target_X8632; |
| #elif defined(__x86_64__) |
| return Target_X8664; |
| #elif defined(__arm__) |
| return Target_ARM32; |
| #else |
| // TODO(stichnot): Add mips. |
| #error "Unknown architecture" |
| #endif |
| } |
| |
| } // end of anonymous namespace |
| |
| BrowserCompileServer::~BrowserCompileServer() = default; |
| |
| void BrowserCompileServer::run() { |
| gCompileServer = this; |
| getIRTInterfaces(); |
| gIRTFuncs.serve_translate_request(&SubzeroCallbacks); |
| } |
| |
| void BrowserCompileServer::getParsedFlags(bool UseNumThreadsFromBrowser, |
| uint32_t NumThreads, int argc, |
| const char *const *argv) { |
| ClFlags::parseFlags(argc, argv); |
| ClFlags::getParsedClFlags(ClFlags::Flags); |
| // Set some defaults which aren't specified via the argv string. |
| if (UseNumThreadsFromBrowser) |
| ClFlags::Flags.setNumTranslationThreads(NumThreads); |
| ClFlags::Flags.setUseSandboxing(true); |
| ClFlags::Flags.setOutFileType(FT_Elf); |
| ClFlags::Flags.setTargetArch(getTargetArch()); |
| ClFlags::Flags.setInputFileFormat(llvm::PNaClFormat); |
| } |
| |
| bool BrowserCompileServer::pushInputBytes(const void *Data, size_t NumBytes) { |
| // If there was an earlier error, do not attempt to push bytes to the |
| // QueueStreamer. Otherwise the thread could become blocked. |
| if (HadError.load()) |
| return true; |
| return InputStream->PutBytes( |
| const_cast<unsigned char *>( |
| reinterpret_cast<const unsigned char *>(Data)), |
| NumBytes) != NumBytes; |
| } |
| |
| void BrowserCompileServer::setFatalError(const std::string &Reason) { |
| HadError.store(true); |
| Ctx->getStrError() << Reason; |
| // Make sure that the QueueStreamer is not stuck by signaling an early end. |
| InputStream->SetDone(); |
| } |
| |
| ErrorCode &BrowserCompileServer::getErrorCode() { |
| if (HadError.load()) { |
| // HadError means report_fatal_error is called. Make sure that the |
| // LastError is not EC_None. We don't know the type of error so just pick |
| // some error category. |
| LastError.assign(EC_Translation); |
| } |
| return LastError; |
| } |
| |
| void BrowserCompileServer::endInputStream() { InputStream->SetDone(); } |
| |
| void BrowserCompileServer::startCompileThread(int ObjFD) { |
| InputStream = new llvm::QueueStreamer(); |
| bool LogStreamFailure = false; |
| int LogFD = STDOUT_FILENO; |
| if (getFlags().getLogFilename() == "-") { |
| // Common case, do nothing. |
| } else if (getFlags().getLogFilename() == "/dev/stderr") { |
| LogFD = STDERR_FILENO; |
| } else { |
| LogStreamFailure = true; |
| } |
| LogStream = getOutputStream(LogFD); |
| LogStream->SetUnbuffered(); |
| if (LogStreamFailure) { |
| *LogStream |
| << "Warning: Log file name must be either '-' or '/dev/stderr'\n"; |
| } |
| EmitStream = getOutputStream(ObjFD); |
| EmitStream->SetBufferSize(1 << 14); |
| std::unique_ptr<StringStream> ErrStrm(new StringStream()); |
| ErrorStream = std::move(ErrStrm); |
| ELFStream.reset(new ELFFileStreamer(*EmitStream.get())); |
| Ctx.reset(new GlobalContext(LogStream.get(), EmitStream.get(), |
| &ErrorStream->getStream(), ELFStream.get())); |
| CompileThread = std::thread([this]() { |
| llvm::install_fatal_error_handler(fatalErrorHandler, this); |
| Ctx->initParserThread(); |
| this->getCompiler().run(ClFlags::Flags, *Ctx.get(), |
| // Retain original reference, but the compiler |
| // (LLVM's MemoryObject) wants to handle deletion. |
| std::unique_ptr<llvm::DataStreamer>(InputStream)); |
| }); |
| } |
| |
| } // end of namespace Ice |
| |
| #else // !PNACL_BROWSER_TRANSLATOR |
| |
| #include "llvm/Support/ErrorHandling.h" |
| |
| namespace Ice { |
| |
| BrowserCompileServer::~BrowserCompileServer() {} |
| |
| void BrowserCompileServer::run() { |
| llvm::report_fatal_error("no browser hookups"); |
| } |
| |
| ErrorCode &BrowserCompileServer::getErrorCode() { |
| llvm::report_fatal_error("no browser hookups"); |
| } |
| |
| } // end of namespace Ice |
| |
| #endif // PNACL_BROWSER_TRANSLATOR |