xref: /aosp_15_r20/external/swiftshader/third_party/subzero/src/IceBrowserCompileServer.cpp (revision 03ce13f70fcc45d86ee91b7ee4cab1936a95046e)
1*03ce13f7SAndroid Build Coastguard Worker //===- subzero/src/IceBrowserCompileServer.cpp - Browser compile server ---===//
2*03ce13f7SAndroid Build Coastguard Worker //
3*03ce13f7SAndroid Build Coastguard Worker //                        The Subzero Code Generator
4*03ce13f7SAndroid Build Coastguard Worker //
5*03ce13f7SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*03ce13f7SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*03ce13f7SAndroid Build Coastguard Worker //
8*03ce13f7SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*03ce13f7SAndroid Build Coastguard Worker ///
10*03ce13f7SAndroid Build Coastguard Worker /// \file
11*03ce13f7SAndroid Build Coastguard Worker /// \brief Defines the browser-based compile server.
12*03ce13f7SAndroid Build Coastguard Worker ///
13*03ce13f7SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
14*03ce13f7SAndroid Build Coastguard Worker 
15*03ce13f7SAndroid Build Coastguard Worker // Can only compile this with the NaCl compiler (needs irt.h, and the
16*03ce13f7SAndroid Build Coastguard Worker // unsandboxed LLVM build using the trusted compiler does not have irt.h).
17*03ce13f7SAndroid Build Coastguard Worker #include "IceBrowserCompileServer.h"
18*03ce13f7SAndroid Build Coastguard Worker #include "IceRangeSpec.h"
19*03ce13f7SAndroid Build Coastguard Worker 
20*03ce13f7SAndroid Build Coastguard Worker #if PNACL_BROWSER_TRANSLATOR
21*03ce13f7SAndroid Build Coastguard Worker 
22*03ce13f7SAndroid Build Coastguard Worker // Headers which are not properly part of the SDK are included by their path in
23*03ce13f7SAndroid Build Coastguard Worker // the NaCl tree.
24*03ce13f7SAndroid Build Coastguard Worker #ifdef __pnacl__
25*03ce13f7SAndroid Build Coastguard Worker #include "native_client/src/untrusted/nacl/pnacl.h"
26*03ce13f7SAndroid Build Coastguard Worker #endif // __pnacl__
27*03ce13f7SAndroid Build Coastguard Worker 
28*03ce13f7SAndroid Build Coastguard Worker #include "llvm/Support/QueueStreamer.h"
29*03ce13f7SAndroid Build Coastguard Worker 
30*03ce13f7SAndroid Build Coastguard Worker #include <cstring>
31*03ce13f7SAndroid Build Coastguard Worker #include <fstream>
32*03ce13f7SAndroid Build Coastguard Worker #include <irt.h>
33*03ce13f7SAndroid Build Coastguard Worker #include <irt_dev.h>
34*03ce13f7SAndroid Build Coastguard Worker #include <pthread.h>
35*03ce13f7SAndroid Build Coastguard Worker #include <thread>
36*03ce13f7SAndroid Build Coastguard Worker 
37*03ce13f7SAndroid Build Coastguard Worker namespace Ice {
38*03ce13f7SAndroid Build Coastguard Worker 
39*03ce13f7SAndroid Build Coastguard Worker // Create C wrappers around callback handlers for the IRT interface.
40*03ce13f7SAndroid Build Coastguard Worker namespace {
41*03ce13f7SAndroid Build Coastguard Worker 
42*03ce13f7SAndroid Build Coastguard Worker BrowserCompileServer *gCompileServer;
43*03ce13f7SAndroid Build Coastguard Worker struct nacl_irt_private_pnacl_translator_compile gIRTFuncs;
44*03ce13f7SAndroid Build Coastguard Worker 
getIRTInterfaces()45*03ce13f7SAndroid Build Coastguard Worker void getIRTInterfaces() {
46*03ce13f7SAndroid Build Coastguard Worker   size_t QueryResult =
47*03ce13f7SAndroid Build Coastguard Worker       nacl_interface_query(NACL_IRT_PRIVATE_PNACL_TRANSLATOR_COMPILE_v0_1,
48*03ce13f7SAndroid Build Coastguard Worker                            &gIRTFuncs, sizeof(gIRTFuncs));
49*03ce13f7SAndroid Build Coastguard Worker   if (QueryResult != sizeof(gIRTFuncs))
50*03ce13f7SAndroid Build Coastguard Worker     llvm::report_fatal_error("Failed to get translator compile IRT interface");
51*03ce13f7SAndroid Build Coastguard Worker }
52*03ce13f7SAndroid Build Coastguard Worker 
53*03ce13f7SAndroid Build Coastguard Worker // Allow pnacl-sz arguments to be supplied externally, instead of coming from
54*03ce13f7SAndroid Build Coastguard Worker // the browser.  This is meant to be used for debugging.
55*03ce13f7SAndroid Build Coastguard Worker //
56*03ce13f7SAndroid Build Coastguard Worker // NOTE: This functionality is only enabled in non-MINIMAL Subzero builds, for
57*03ce13f7SAndroid Build Coastguard Worker // security/safety reasons.
58*03ce13f7SAndroid Build Coastguard Worker //
59*03ce13f7SAndroid Build Coastguard Worker // If the SZARGFILE environment variable is set to a file name, arguments are
60*03ce13f7SAndroid Build Coastguard Worker // read from that file, one argument per line.  This requires setting 3
61*03ce13f7SAndroid Build Coastguard Worker // environment variables before starting the browser:
62*03ce13f7SAndroid Build Coastguard Worker //
63*03ce13f7SAndroid Build Coastguard Worker // NACL_ENV_PASSTHROUGH=NACL_DANGEROUS_ENABLE_FILE_ACCESS,NACLENV_SZARGFILE
64*03ce13f7SAndroid Build Coastguard Worker // NACL_DANGEROUS_ENABLE_FILE_ACCESS=1
65*03ce13f7SAndroid Build Coastguard Worker // NACLENV_SZARGFILE=/path/to/myargs.txt
66*03ce13f7SAndroid Build Coastguard Worker //
67*03ce13f7SAndroid Build Coastguard Worker // In addition, Chrome needs to be launched with the "--no-sandbox" argument.
68*03ce13f7SAndroid Build Coastguard Worker //
69*03ce13f7SAndroid Build Coastguard Worker // If the SZARGLIST environment variable is set, arguments are extracted from
70*03ce13f7SAndroid Build Coastguard Worker // that variable's value, separated by the '|' character (being careful to
71*03ce13f7SAndroid Build Coastguard Worker // escape/quote special shell characters).  This requires setting 2 environment
72*03ce13f7SAndroid Build Coastguard Worker // variables before starting the browser:
73*03ce13f7SAndroid Build Coastguard Worker //
74*03ce13f7SAndroid Build Coastguard Worker // NACL_ENV_PASSTHROUGH=NACLENV_SZARGLIST
75*03ce13f7SAndroid Build Coastguard Worker // NACLENV_SZARGLIST=arg
76*03ce13f7SAndroid Build Coastguard Worker //
77*03ce13f7SAndroid Build Coastguard Worker // This does not require the "--no-sandbox" argument, and is therefore much
78*03ce13f7SAndroid Build Coastguard Worker // safer, but does require restarting the browser to change the arguments.
79*03ce13f7SAndroid Build Coastguard Worker //
80*03ce13f7SAndroid Build Coastguard Worker // If external arguments are supplied, the browser's NumThreads specification is
81*03ce13f7SAndroid Build Coastguard Worker // ignored, to allow -threads to be specified as an external argument.  Note
82*03ce13f7SAndroid Build Coastguard Worker // that the browser normally supplies the "-O2" argument, so externally supplied
83*03ce13f7SAndroid Build Coastguard Worker // arguments might want to provide an explicit -O argument.
84*03ce13f7SAndroid Build Coastguard Worker //
85*03ce13f7SAndroid Build Coastguard Worker // See Chrome's src/components/nacl/zygote/nacl_fork_delegate_linux.cc for the
86*03ce13f7SAndroid Build Coastguard Worker // NACL_ENV_PASSTHROUGH mechanism.
87*03ce13f7SAndroid Build Coastguard Worker //
88*03ce13f7SAndroid Build Coastguard Worker // See NaCl's src/trusted/service_runtime/env_cleanser.c for the NACLENV_
89*03ce13f7SAndroid Build Coastguard Worker // mechanism.
getExternalArgs()90*03ce13f7SAndroid Build Coastguard Worker std::vector<std::string> getExternalArgs() {
91*03ce13f7SAndroid Build Coastguard Worker   std::vector<std::string> ExternalArgs;
92*03ce13f7SAndroid Build Coastguard Worker   if (BuildDefs::minimal())
93*03ce13f7SAndroid Build Coastguard Worker     return ExternalArgs;
94*03ce13f7SAndroid Build Coastguard Worker   char ArgsFileVar[] = "SZARGFILE";
95*03ce13f7SAndroid Build Coastguard Worker   char ArgsListVar[] = "SZARGLIST";
96*03ce13f7SAndroid Build Coastguard Worker   if (const char *ArgsFilename = getenv(ArgsFileVar)) {
97*03ce13f7SAndroid Build Coastguard Worker     std::ifstream ArgsStream(ArgsFilename);
98*03ce13f7SAndroid Build Coastguard Worker     std::string Arg;
99*03ce13f7SAndroid Build Coastguard Worker     while (ArgsStream >> std::ws, std::getline(ArgsStream, Arg)) {
100*03ce13f7SAndroid Build Coastguard Worker       if (!Arg.empty() && Arg[0] == '#')
101*03ce13f7SAndroid Build Coastguard Worker         continue;
102*03ce13f7SAndroid Build Coastguard Worker       ExternalArgs.emplace_back(Arg);
103*03ce13f7SAndroid Build Coastguard Worker     }
104*03ce13f7SAndroid Build Coastguard Worker     if (ExternalArgs.empty()) {
105*03ce13f7SAndroid Build Coastguard Worker       llvm::report_fatal_error("Failed to read arguments from file '" +
106*03ce13f7SAndroid Build Coastguard Worker                                std::string(ArgsFilename) + "'");
107*03ce13f7SAndroid Build Coastguard Worker     }
108*03ce13f7SAndroid Build Coastguard Worker   } else if (const char *ArgsList = getenv(ArgsListVar)) {
109*03ce13f7SAndroid Build Coastguard Worker     // Leverage the RangeSpec tokenizer.
110*03ce13f7SAndroid Build Coastguard Worker     auto Args = RangeSpec::tokenize(ArgsList, '|');
111*03ce13f7SAndroid Build Coastguard Worker     ExternalArgs.insert(ExternalArgs.end(), Args.begin(), Args.end());
112*03ce13f7SAndroid Build Coastguard Worker   }
113*03ce13f7SAndroid Build Coastguard Worker   return ExternalArgs;
114*03ce13f7SAndroid Build Coastguard Worker }
115*03ce13f7SAndroid Build Coastguard Worker 
onInitCallback(uint32_t NumThreads,int * ObjFileFDs,size_t ObjFileFDCount,char ** CLArgs,size_t CLArgsLen)116*03ce13f7SAndroid Build Coastguard Worker char *onInitCallback(uint32_t NumThreads, int *ObjFileFDs,
117*03ce13f7SAndroid Build Coastguard Worker                      size_t ObjFileFDCount, char **CLArgs, size_t CLArgsLen) {
118*03ce13f7SAndroid Build Coastguard Worker   if (ObjFileFDCount < 1) {
119*03ce13f7SAndroid Build Coastguard Worker     std::string Buffer;
120*03ce13f7SAndroid Build Coastguard Worker     llvm::raw_string_ostream StrBuf(Buffer);
121*03ce13f7SAndroid Build Coastguard Worker     StrBuf << "Invalid number of FDs for onInitCallback " << ObjFileFDCount
122*03ce13f7SAndroid Build Coastguard Worker            << "\n";
123*03ce13f7SAndroid Build Coastguard Worker     return strdup(StrBuf.str().c_str());
124*03ce13f7SAndroid Build Coastguard Worker   }
125*03ce13f7SAndroid Build Coastguard Worker   int ObjFileFD = ObjFileFDs[0];
126*03ce13f7SAndroid Build Coastguard Worker   if (ObjFileFD < 0) {
127*03ce13f7SAndroid Build Coastguard Worker     std::string Buffer;
128*03ce13f7SAndroid Build Coastguard Worker     llvm::raw_string_ostream StrBuf(Buffer);
129*03ce13f7SAndroid Build Coastguard Worker     StrBuf << "Invalid FD given for onInitCallback " << ObjFileFD << "\n";
130*03ce13f7SAndroid Build Coastguard Worker     return strdup(StrBuf.str().c_str());
131*03ce13f7SAndroid Build Coastguard Worker   }
132*03ce13f7SAndroid Build Coastguard Worker   // CLArgs is almost an "argv", but is missing the argv[0] program name.
133*03ce13f7SAndroid Build Coastguard Worker   std::vector<const char *> Argv;
134*03ce13f7SAndroid Build Coastguard Worker   constexpr static char ProgramName[] = "pnacl-sz.nexe";
135*03ce13f7SAndroid Build Coastguard Worker   Argv.reserve(CLArgsLen + 1);
136*03ce13f7SAndroid Build Coastguard Worker   Argv.push_back(ProgramName);
137*03ce13f7SAndroid Build Coastguard Worker 
138*03ce13f7SAndroid Build Coastguard Worker   bool UseNumThreadsFromBrowser = true;
139*03ce13f7SAndroid Build Coastguard Worker   auto ExternalArgs = getExternalArgs();
140*03ce13f7SAndroid Build Coastguard Worker   if (ExternalArgs.empty()) {
141*03ce13f7SAndroid Build Coastguard Worker     for (size_t i = 0; i < CLArgsLen; ++i) {
142*03ce13f7SAndroid Build Coastguard Worker       Argv.push_back(CLArgs[i]);
143*03ce13f7SAndroid Build Coastguard Worker     }
144*03ce13f7SAndroid Build Coastguard Worker   } else {
145*03ce13f7SAndroid Build Coastguard Worker     for (auto &Arg : ExternalArgs) {
146*03ce13f7SAndroid Build Coastguard Worker       Argv.emplace_back(Arg.c_str());
147*03ce13f7SAndroid Build Coastguard Worker     }
148*03ce13f7SAndroid Build Coastguard Worker     UseNumThreadsFromBrowser = false;
149*03ce13f7SAndroid Build Coastguard Worker   }
150*03ce13f7SAndroid Build Coastguard Worker   // NOTE: strings pointed to by argv are owned by the caller, but we parse
151*03ce13f7SAndroid Build Coastguard Worker   // here before returning and don't store them.
152*03ce13f7SAndroid Build Coastguard Worker   gCompileServer->getParsedFlags(UseNumThreadsFromBrowser, NumThreads,
153*03ce13f7SAndroid Build Coastguard Worker                                  Argv.size(), Argv.data());
154*03ce13f7SAndroid Build Coastguard Worker   gCompileServer->startCompileThread(ObjFileFD);
155*03ce13f7SAndroid Build Coastguard Worker   return nullptr;
156*03ce13f7SAndroid Build Coastguard Worker }
157*03ce13f7SAndroid Build Coastguard Worker 
onDataCallback(const void * Data,size_t NumBytes)158*03ce13f7SAndroid Build Coastguard Worker int onDataCallback(const void *Data, size_t NumBytes) {
159*03ce13f7SAndroid Build Coastguard Worker   return gCompileServer->pushInputBytes(Data, NumBytes) ? 1 : 0;
160*03ce13f7SAndroid Build Coastguard Worker }
161*03ce13f7SAndroid Build Coastguard Worker 
onEndCallback()162*03ce13f7SAndroid Build Coastguard Worker char *onEndCallback() {
163*03ce13f7SAndroid Build Coastguard Worker   gCompileServer->endInputStream();
164*03ce13f7SAndroid Build Coastguard Worker   gCompileServer->waitForCompileThread();
165*03ce13f7SAndroid Build Coastguard Worker   // TODO(jvoung): Also return UMA data.
166*03ce13f7SAndroid Build Coastguard Worker   if (gCompileServer->getErrorCode().value()) {
167*03ce13f7SAndroid Build Coastguard Worker     const std::string Error = gCompileServer->getErrorStream().getContents();
168*03ce13f7SAndroid Build Coastguard Worker     return strdup(Error.empty() ? "Some error occurred" : Error.c_str());
169*03ce13f7SAndroid Build Coastguard Worker   }
170*03ce13f7SAndroid Build Coastguard Worker   return nullptr;
171*03ce13f7SAndroid Build Coastguard Worker }
172*03ce13f7SAndroid Build Coastguard Worker 
173*03ce13f7SAndroid Build Coastguard Worker struct nacl_irt_pnacl_compile_funcs SubzeroCallbacks {
174*03ce13f7SAndroid Build Coastguard Worker   &onInitCallback, &onDataCallback, &onEndCallback
175*03ce13f7SAndroid Build Coastguard Worker };
176*03ce13f7SAndroid Build Coastguard Worker 
getOutputStream(int FD)177*03ce13f7SAndroid Build Coastguard Worker std::unique_ptr<llvm::raw_fd_ostream> getOutputStream(int FD) {
178*03ce13f7SAndroid Build Coastguard Worker   if (FD <= 0)
179*03ce13f7SAndroid Build Coastguard Worker     llvm::report_fatal_error("Invalid output FD");
180*03ce13f7SAndroid Build Coastguard Worker   constexpr bool CloseOnDtor = true;
181*03ce13f7SAndroid Build Coastguard Worker   constexpr bool Unbuffered = false;
182*03ce13f7SAndroid Build Coastguard Worker   return std::unique_ptr<llvm::raw_fd_ostream>(
183*03ce13f7SAndroid Build Coastguard Worker       new llvm::raw_fd_ostream(FD, CloseOnDtor, Unbuffered));
184*03ce13f7SAndroid Build Coastguard Worker }
185*03ce13f7SAndroid Build Coastguard Worker 
fatalErrorHandler(void * UserData,const std::string & Reason,bool GenCrashDialog)186*03ce13f7SAndroid Build Coastguard Worker void fatalErrorHandler(void *UserData, const std::string &Reason,
187*03ce13f7SAndroid Build Coastguard Worker                        bool GenCrashDialog) {
188*03ce13f7SAndroid Build Coastguard Worker   (void)GenCrashDialog;
189*03ce13f7SAndroid Build Coastguard Worker   BrowserCompileServer *Server =
190*03ce13f7SAndroid Build Coastguard Worker       reinterpret_cast<BrowserCompileServer *>(UserData);
191*03ce13f7SAndroid Build Coastguard Worker   Server->setFatalError(Reason);
192*03ce13f7SAndroid Build Coastguard Worker   // Only kill the current thread instead of the whole process. We need the
193*03ce13f7SAndroid Build Coastguard Worker   // server thread to remain alive in order to respond with the error message.
194*03ce13f7SAndroid Build Coastguard Worker   // We could also try to pthread_kill all other worker threads, but
195*03ce13f7SAndroid Build Coastguard Worker   // pthread_kill / raising signals is not supported by NaCl. We'll have to
196*03ce13f7SAndroid Build Coastguard Worker   // assume that the worker/emitter threads will be well behaved after a fatal
197*03ce13f7SAndroid Build Coastguard Worker   // error in other threads, and either get stuck waiting on input from a
198*03ce13f7SAndroid Build Coastguard Worker   // previous stage, or also call report_fatal_error.
199*03ce13f7SAndroid Build Coastguard Worker   pthread_exit(0);
200*03ce13f7SAndroid Build Coastguard Worker }
201*03ce13f7SAndroid Build Coastguard Worker 
202*03ce13f7SAndroid Build Coastguard Worker /// Adapted from pnacl-llc's AddDefaultCPU() in srpc_main.cpp.
getTargetArch()203*03ce13f7SAndroid Build Coastguard Worker TargetArch getTargetArch() {
204*03ce13f7SAndroid Build Coastguard Worker #if defined(__pnacl__)
205*03ce13f7SAndroid Build Coastguard Worker   switch (__builtin_nacl_target_arch()) {
206*03ce13f7SAndroid Build Coastguard Worker   case PnaclTargetArchitectureX86_32:
207*03ce13f7SAndroid Build Coastguard Worker   case PnaclTargetArchitectureX86_32_NonSFI:
208*03ce13f7SAndroid Build Coastguard Worker     return Target_X8632;
209*03ce13f7SAndroid Build Coastguard Worker   case PnaclTargetArchitectureX86_64:
210*03ce13f7SAndroid Build Coastguard Worker     return Target_X8664;
211*03ce13f7SAndroid Build Coastguard Worker   case PnaclTargetArchitectureARM_32:
212*03ce13f7SAndroid Build Coastguard Worker   case PnaclTargetArchitectureARM_32_NonSFI:
213*03ce13f7SAndroid Build Coastguard Worker     return Target_ARM32;
214*03ce13f7SAndroid Build Coastguard Worker   case PnaclTargetArchitectureMips_32:
215*03ce13f7SAndroid Build Coastguard Worker     return Target_MIPS32;
216*03ce13f7SAndroid Build Coastguard Worker   default:
217*03ce13f7SAndroid Build Coastguard Worker     llvm::report_fatal_error("no target architecture match.");
218*03ce13f7SAndroid Build Coastguard Worker   }
219*03ce13f7SAndroid Build Coastguard Worker #elif defined(__i386__)
220*03ce13f7SAndroid Build Coastguard Worker   return Target_X8632;
221*03ce13f7SAndroid Build Coastguard Worker #elif defined(__x86_64__)
222*03ce13f7SAndroid Build Coastguard Worker   return Target_X8664;
223*03ce13f7SAndroid Build Coastguard Worker #elif defined(__arm__)
224*03ce13f7SAndroid Build Coastguard Worker   return Target_ARM32;
225*03ce13f7SAndroid Build Coastguard Worker #else
226*03ce13f7SAndroid Build Coastguard Worker // TODO(stichnot): Add mips.
227*03ce13f7SAndroid Build Coastguard Worker #error "Unknown architecture"
228*03ce13f7SAndroid Build Coastguard Worker #endif
229*03ce13f7SAndroid Build Coastguard Worker }
230*03ce13f7SAndroid Build Coastguard Worker 
231*03ce13f7SAndroid Build Coastguard Worker } // end of anonymous namespace
232*03ce13f7SAndroid Build Coastguard Worker 
233*03ce13f7SAndroid Build Coastguard Worker BrowserCompileServer::~BrowserCompileServer() = default;
234*03ce13f7SAndroid Build Coastguard Worker 
run()235*03ce13f7SAndroid Build Coastguard Worker void BrowserCompileServer::run() {
236*03ce13f7SAndroid Build Coastguard Worker   gCompileServer = this;
237*03ce13f7SAndroid Build Coastguard Worker   getIRTInterfaces();
238*03ce13f7SAndroid Build Coastguard Worker   gIRTFuncs.serve_translate_request(&SubzeroCallbacks);
239*03ce13f7SAndroid Build Coastguard Worker }
240*03ce13f7SAndroid Build Coastguard Worker 
getParsedFlags(bool UseNumThreadsFromBrowser,uint32_t NumThreads,int argc,const char * const * argv)241*03ce13f7SAndroid Build Coastguard Worker void BrowserCompileServer::getParsedFlags(bool UseNumThreadsFromBrowser,
242*03ce13f7SAndroid Build Coastguard Worker                                           uint32_t NumThreads, int argc,
243*03ce13f7SAndroid Build Coastguard Worker                                           const char *const *argv) {
244*03ce13f7SAndroid Build Coastguard Worker   ClFlags::parseFlags(argc, argv);
245*03ce13f7SAndroid Build Coastguard Worker   ClFlags::getParsedClFlags(ClFlags::Flags);
246*03ce13f7SAndroid Build Coastguard Worker   // Set some defaults which aren't specified via the argv string.
247*03ce13f7SAndroid Build Coastguard Worker   if (UseNumThreadsFromBrowser)
248*03ce13f7SAndroid Build Coastguard Worker     ClFlags::Flags.setNumTranslationThreads(NumThreads);
249*03ce13f7SAndroid Build Coastguard Worker   ClFlags::Flags.setUseSandboxing(true);
250*03ce13f7SAndroid Build Coastguard Worker   ClFlags::Flags.setOutFileType(FT_Elf);
251*03ce13f7SAndroid Build Coastguard Worker   ClFlags::Flags.setTargetArch(getTargetArch());
252*03ce13f7SAndroid Build Coastguard Worker   ClFlags::Flags.setInputFileFormat(llvm::PNaClFormat);
253*03ce13f7SAndroid Build Coastguard Worker }
254*03ce13f7SAndroid Build Coastguard Worker 
pushInputBytes(const void * Data,size_t NumBytes)255*03ce13f7SAndroid Build Coastguard Worker bool BrowserCompileServer::pushInputBytes(const void *Data, size_t NumBytes) {
256*03ce13f7SAndroid Build Coastguard Worker   // If there was an earlier error, do not attempt to push bytes to the
257*03ce13f7SAndroid Build Coastguard Worker   // QueueStreamer. Otherwise the thread could become blocked.
258*03ce13f7SAndroid Build Coastguard Worker   if (HadError.load())
259*03ce13f7SAndroid Build Coastguard Worker     return true;
260*03ce13f7SAndroid Build Coastguard Worker   return InputStream->PutBytes(
261*03ce13f7SAndroid Build Coastguard Worker              const_cast<unsigned char *>(
262*03ce13f7SAndroid Build Coastguard Worker                  reinterpret_cast<const unsigned char *>(Data)),
263*03ce13f7SAndroid Build Coastguard Worker              NumBytes) != NumBytes;
264*03ce13f7SAndroid Build Coastguard Worker }
265*03ce13f7SAndroid Build Coastguard Worker 
setFatalError(const std::string & Reason)266*03ce13f7SAndroid Build Coastguard Worker void BrowserCompileServer::setFatalError(const std::string &Reason) {
267*03ce13f7SAndroid Build Coastguard Worker   HadError.store(true);
268*03ce13f7SAndroid Build Coastguard Worker   Ctx->getStrError() << Reason;
269*03ce13f7SAndroid Build Coastguard Worker   // Make sure that the QueueStreamer is not stuck by signaling an early end.
270*03ce13f7SAndroid Build Coastguard Worker   InputStream->SetDone();
271*03ce13f7SAndroid Build Coastguard Worker }
272*03ce13f7SAndroid Build Coastguard Worker 
getErrorCode()273*03ce13f7SAndroid Build Coastguard Worker ErrorCode &BrowserCompileServer::getErrorCode() {
274*03ce13f7SAndroid Build Coastguard Worker   if (HadError.load()) {
275*03ce13f7SAndroid Build Coastguard Worker     // HadError means report_fatal_error is called. Make sure that the
276*03ce13f7SAndroid Build Coastguard Worker     // LastError is not EC_None. We don't know the type of error so just pick
277*03ce13f7SAndroid Build Coastguard Worker     // some error category.
278*03ce13f7SAndroid Build Coastguard Worker     LastError.assign(EC_Translation);
279*03ce13f7SAndroid Build Coastguard Worker   }
280*03ce13f7SAndroid Build Coastguard Worker   return LastError;
281*03ce13f7SAndroid Build Coastguard Worker }
282*03ce13f7SAndroid Build Coastguard Worker 
endInputStream()283*03ce13f7SAndroid Build Coastguard Worker void BrowserCompileServer::endInputStream() { InputStream->SetDone(); }
284*03ce13f7SAndroid Build Coastguard Worker 
startCompileThread(int ObjFD)285*03ce13f7SAndroid Build Coastguard Worker void BrowserCompileServer::startCompileThread(int ObjFD) {
286*03ce13f7SAndroid Build Coastguard Worker   InputStream = new llvm::QueueStreamer();
287*03ce13f7SAndroid Build Coastguard Worker   bool LogStreamFailure = false;
288*03ce13f7SAndroid Build Coastguard Worker   int LogFD = STDOUT_FILENO;
289*03ce13f7SAndroid Build Coastguard Worker   if (getFlags().getLogFilename() == "-") {
290*03ce13f7SAndroid Build Coastguard Worker     // Common case, do nothing.
291*03ce13f7SAndroid Build Coastguard Worker   } else if (getFlags().getLogFilename() == "/dev/stderr") {
292*03ce13f7SAndroid Build Coastguard Worker     LogFD = STDERR_FILENO;
293*03ce13f7SAndroid Build Coastguard Worker   } else {
294*03ce13f7SAndroid Build Coastguard Worker     LogStreamFailure = true;
295*03ce13f7SAndroid Build Coastguard Worker   }
296*03ce13f7SAndroid Build Coastguard Worker   LogStream = getOutputStream(LogFD);
297*03ce13f7SAndroid Build Coastguard Worker   LogStream->SetUnbuffered();
298*03ce13f7SAndroid Build Coastguard Worker   if (LogStreamFailure) {
299*03ce13f7SAndroid Build Coastguard Worker     *LogStream
300*03ce13f7SAndroid Build Coastguard Worker         << "Warning: Log file name must be either '-' or '/dev/stderr'\n";
301*03ce13f7SAndroid Build Coastguard Worker   }
302*03ce13f7SAndroid Build Coastguard Worker   EmitStream = getOutputStream(ObjFD);
303*03ce13f7SAndroid Build Coastguard Worker   EmitStream->SetBufferSize(1 << 14);
304*03ce13f7SAndroid Build Coastguard Worker   std::unique_ptr<StringStream> ErrStrm(new StringStream());
305*03ce13f7SAndroid Build Coastguard Worker   ErrorStream = std::move(ErrStrm);
306*03ce13f7SAndroid Build Coastguard Worker   ELFStream.reset(new ELFFileStreamer(*EmitStream.get()));
307*03ce13f7SAndroid Build Coastguard Worker   Ctx.reset(new GlobalContext(LogStream.get(), EmitStream.get(),
308*03ce13f7SAndroid Build Coastguard Worker                               &ErrorStream->getStream(), ELFStream.get()));
309*03ce13f7SAndroid Build Coastguard Worker   CompileThread = std::thread([this]() {
310*03ce13f7SAndroid Build Coastguard Worker     llvm::install_fatal_error_handler(fatalErrorHandler, this);
311*03ce13f7SAndroid Build Coastguard Worker     Ctx->initParserThread();
312*03ce13f7SAndroid Build Coastguard Worker     this->getCompiler().run(ClFlags::Flags, *Ctx.get(),
313*03ce13f7SAndroid Build Coastguard Worker                             // Retain original reference, but the compiler
314*03ce13f7SAndroid Build Coastguard Worker                             // (LLVM's MemoryObject) wants to handle deletion.
315*03ce13f7SAndroid Build Coastguard Worker                             std::unique_ptr<llvm::DataStreamer>(InputStream));
316*03ce13f7SAndroid Build Coastguard Worker   });
317*03ce13f7SAndroid Build Coastguard Worker }
318*03ce13f7SAndroid Build Coastguard Worker 
319*03ce13f7SAndroid Build Coastguard Worker } // end of namespace Ice
320*03ce13f7SAndroid Build Coastguard Worker 
321*03ce13f7SAndroid Build Coastguard Worker #else // !PNACL_BROWSER_TRANSLATOR
322*03ce13f7SAndroid Build Coastguard Worker 
323*03ce13f7SAndroid Build Coastguard Worker #include "llvm/Support/ErrorHandling.h"
324*03ce13f7SAndroid Build Coastguard Worker 
325*03ce13f7SAndroid Build Coastguard Worker namespace Ice {
326*03ce13f7SAndroid Build Coastguard Worker 
~BrowserCompileServer()327*03ce13f7SAndroid Build Coastguard Worker BrowserCompileServer::~BrowserCompileServer() {}
328*03ce13f7SAndroid Build Coastguard Worker 
run()329*03ce13f7SAndroid Build Coastguard Worker void BrowserCompileServer::run() {
330*03ce13f7SAndroid Build Coastguard Worker   llvm::report_fatal_error("no browser hookups");
331*03ce13f7SAndroid Build Coastguard Worker }
332*03ce13f7SAndroid Build Coastguard Worker 
getErrorCode()333*03ce13f7SAndroid Build Coastguard Worker ErrorCode &BrowserCompileServer::getErrorCode() {
334*03ce13f7SAndroid Build Coastguard Worker   llvm::report_fatal_error("no browser hookups");
335*03ce13f7SAndroid Build Coastguard Worker }
336*03ce13f7SAndroid Build Coastguard Worker 
337*03ce13f7SAndroid Build Coastguard Worker } // end of namespace Ice
338*03ce13f7SAndroid Build Coastguard Worker 
339*03ce13f7SAndroid Build Coastguard Worker #endif // PNACL_BROWSER_TRANSLATOR
340