340 lines
12 KiB
C++
340 lines
12 KiB
C++
|
|
//===- 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
|