OLD | NEW |
---|---|
1 //===- subzero/src/IceBrowserCompileServer.cpp - Browser compile server ---===// | 1 //===- subzero/src/IceBrowserCompileServer.cpp - Browser compile server ---===// |
2 // | 2 // |
3 // The Subzero Code Generator | 3 // The Subzero Code Generator |
4 // | 4 // |
5 // This file is distributed under the University of Illinois Open Source | 5 // This file is distributed under the University of Illinois Open Source |
6 // License. See LICENSE.TXT for details. | 6 // License. See LICENSE.TXT for details. |
7 // | 7 // |
8 //===----------------------------------------------------------------------===// | 8 //===----------------------------------------------------------------------===// |
9 /// | 9 /// |
10 /// \file | 10 /// \file |
11 /// \brief Defines the browser-based compile server. | 11 /// \brief Defines the browser-based compile server. |
12 /// | 12 /// |
13 //===----------------------------------------------------------------------===// | 13 //===----------------------------------------------------------------------===// |
14 | 14 |
15 // Can only compile this with the NaCl compiler (needs irt.h, and the | 15 // Can only compile this with the NaCl compiler (needs irt.h, and the |
16 // unsandboxed LLVM build using the trusted compiler does not have irt.h). | 16 // unsandboxed LLVM build using the trusted compiler does not have irt.h). |
17 #include "IceBrowserCompileServer.h" | 17 #include "IceBrowserCompileServer.h" |
18 #include "IceRangeSpec.h" | |
18 | 19 |
19 #if PNACL_BROWSER_TRANSLATOR | 20 #if PNACL_BROWSER_TRANSLATOR |
20 | 21 |
21 // Headers which are not properly part of the SDK are included by their path in | 22 // Headers which are not properly part of the SDK are included by their path in |
22 // the NaCl tree. | 23 // the NaCl tree. |
23 #ifdef __pnacl__ | 24 #ifdef __pnacl__ |
24 #include "native_client/src/untrusted/nacl/pnacl.h" | 25 #include "native_client/src/untrusted/nacl/pnacl.h" |
25 #endif // __pnacl__ | 26 #endif // __pnacl__ |
26 | 27 |
27 #include "llvm/Support/QueueStreamer.h" | 28 #include "llvm/Support/QueueStreamer.h" |
28 | 29 |
29 #include <cstring> | 30 #include <cstring> |
31 #include <fstream> | |
30 #include <irt.h> | 32 #include <irt.h> |
31 #include <irt_dev.h> | 33 #include <irt_dev.h> |
32 #include <pthread.h> | 34 #include <pthread.h> |
33 #include <thread> | 35 #include <thread> |
34 | 36 |
35 namespace Ice { | 37 namespace Ice { |
36 | 38 |
37 // Create C wrappers around callback handlers for the IRT interface. | 39 // Create C wrappers around callback handlers for the IRT interface. |
38 namespace { | 40 namespace { |
39 | 41 |
40 BrowserCompileServer *gCompileServer; | 42 BrowserCompileServer *gCompileServer; |
41 struct nacl_irt_private_pnacl_translator_compile gIRTFuncs; | 43 struct nacl_irt_private_pnacl_translator_compile gIRTFuncs; |
42 | 44 |
43 void getIRTInterfaces() { | 45 void getIRTInterfaces() { |
44 size_t QueryResult = | 46 size_t QueryResult = |
45 nacl_interface_query(NACL_IRT_PRIVATE_PNACL_TRANSLATOR_COMPILE_v0_1, | 47 nacl_interface_query(NACL_IRT_PRIVATE_PNACL_TRANSLATOR_COMPILE_v0_1, |
46 &gIRTFuncs, sizeof(gIRTFuncs)); | 48 &gIRTFuncs, sizeof(gIRTFuncs)); |
47 if (QueryResult != sizeof(gIRTFuncs)) | 49 if (QueryResult != sizeof(gIRTFuncs)) |
48 llvm::report_fatal_error("Failed to get translator compile IRT interface"); | 50 llvm::report_fatal_error("Failed to get translator compile IRT interface"); |
49 } | 51 } |
50 | 52 |
53 // Allow pnacl-sz arguments to be supplied externally, instead of coming from | |
54 // the browser. This is meant to be used for debugging. | |
55 // | |
56 // If the SZARGFILE environment variable is set to a file name, arguments are | |
57 // read from that file, one argument per line. This requires setting 3 | |
58 // environment variables before starting the browser: | |
59 // | |
60 // NACL_ENV_PASSTHROUGH=NACL_DANGEROUS_ENABLE_FILE_ACCESS,NACLENV_SZARGFILE | |
61 // NACL_DANGEROUS_ENABLE_FILE_ACCESS=1 | |
62 // NACLENV_SZARGFILE=/path/to/myargs.txt | |
63 // | |
64 // In addition, Chrome needs to be launched with the "--no-sandbox" argument. | |
65 // | |
66 // If the SZARGLIST environment variable is set, arguments are extracted from | |
67 // that variable's value, separated by the '|' character (being careful to | |
68 // escape/quote special shell characters). This requires setting 2 environment | |
69 // variables before starting the browser: | |
70 // | |
71 // NACL_ENV_PASSTHROUGH=NACLENV_SZARGLIST | |
72 // NACLENV_SZARGLIST=arg | |
73 // | |
74 // This does not require the "--no-sandbox" argument, and is therefore much | |
75 // safer, but does require restarting the browser to change the arguments. | |
76 // | |
77 // If external arguments are supplied, the browser's NumThreads specification is | |
78 // ignored, to allow -threads to be specified as an external argument. Note | |
79 // that the browser normally supplies the "-O2" argument, so externally supplied | |
80 // arguments might want to provide an explicit -O argument. | |
81 std::vector<std::string> getExternalArgs() { | |
John
2016/04/20 03:22:02
Maybe add a
if (BrowserBuild) return {};
Jim Stichnoth
2016/04/20 04:17:23
This whole file is already wrapped inside "#if PNA
| |
82 std::vector<std::string> ExternalArgs; | |
83 char ArgsFileVar[] = "SZARGFILE"; | |
84 char ArgsListVar[] = "SZARGLIST"; | |
85 if (const char *ArgsFilename = getenv(ArgsFileVar)) { | |
86 std::ifstream ArgsStream(ArgsFilename); | |
87 std::string Arg; | |
88 while (ArgsStream >> Arg) { | |
89 ExternalArgs.emplace_back(Arg); | |
90 } | |
91 if (ExternalArgs.empty()) { | |
92 llvm::report_fatal_error("Failed to read arguments from file '" + | |
93 std::string(ArgsFilename) + "'"); | |
94 } | |
95 } else if (const char *ArgsList = getenv(ArgsListVar)) { | |
96 // Leverage the RangeSpec tokenizer. | |
97 auto Args = RangeSpec::tokenize(ArgsList, '|'); | |
98 ExternalArgs.insert(ExternalArgs.end(), Args.begin(), Args.end()); | |
99 } | |
100 return ExternalArgs; | |
101 } | |
102 | |
51 char *onInitCallback(uint32_t NumThreads, int *ObjFileFDs, | 103 char *onInitCallback(uint32_t NumThreads, int *ObjFileFDs, |
52 size_t ObjFileFDCount, char **CLArgs, size_t CLArgsLen) { | 104 size_t ObjFileFDCount, char **CLArgs, size_t CLArgsLen) { |
53 if (ObjFileFDCount < 1) { | 105 if (ObjFileFDCount < 1) { |
54 std::string Buffer; | 106 std::string Buffer; |
55 llvm::raw_string_ostream StrBuf(Buffer); | 107 llvm::raw_string_ostream StrBuf(Buffer); |
56 StrBuf << "Invalid number of FDs for onInitCallback " << ObjFileFDCount | 108 StrBuf << "Invalid number of FDs for onInitCallback " << ObjFileFDCount |
57 << "\n"; | 109 << "\n"; |
58 return strdup(StrBuf.str().c_str()); | 110 return strdup(StrBuf.str().c_str()); |
59 } | 111 } |
60 int ObjFileFD = ObjFileFDs[0]; | 112 int ObjFileFD = ObjFileFDs[0]; |
61 if (ObjFileFD < 0) { | 113 if (ObjFileFD < 0) { |
62 std::string Buffer; | 114 std::string Buffer; |
63 llvm::raw_string_ostream StrBuf(Buffer); | 115 llvm::raw_string_ostream StrBuf(Buffer); |
64 StrBuf << "Invalid FD given for onInitCallback " << ObjFileFD << "\n"; | 116 StrBuf << "Invalid FD given for onInitCallback " << ObjFileFD << "\n"; |
65 return strdup(StrBuf.str().c_str()); | 117 return strdup(StrBuf.str().c_str()); |
66 } | 118 } |
67 // CLArgs is almost an "argv", but is missing the argv[0] program name. | 119 // CLArgs is almost an "argv", but is missing the argv[0] program name. |
68 std::vector<char *> Argv; | 120 std::vector<char *> Argv; |
69 char ProgramName[] = "pnacl-sz.nexe"; | 121 char ProgramName[] = "pnacl-sz.nexe"; |
70 Argv.reserve(CLArgsLen + 1); | 122 Argv.reserve(CLArgsLen + 1); |
71 Argv.push_back(ProgramName); | 123 Argv.push_back(ProgramName); |
72 for (size_t i = 0; i < CLArgsLen; ++i) { | 124 |
73 Argv.push_back(CLArgs[i]); | 125 bool UseNumThreadsFromBrowser = true; |
126 auto ExternalArgs = getExternalArgs(); | |
127 if (ExternalArgs.empty()) { | |
128 for (size_t i = 0; i < CLArgsLen; ++i) { | |
129 Argv.push_back(CLArgs[i]); | |
130 } | |
131 } else { | |
132 for (auto &Arg : ExternalArgs) { | |
133 Argv.emplace_back(const_cast<char *>(Arg.c_str())); | |
John
2016/04/20 03:22:02
This is, in general, bad: you're holding pointers
Jim Stichnoth
2016/04/20 04:17:23
Yeah, I dislike this, also the const_cast. (Fortu
| |
134 } | |
135 UseNumThreadsFromBrowser = false; | |
74 } | 136 } |
75 // NOTE: strings pointed to by argv are owned by the caller, but we parse | 137 // NOTE: strings pointed to by argv are owned by the caller, but we parse |
76 // here before returning and don't store them. | 138 // here before returning and don't store them. |
77 gCompileServer->getParsedFlags(NumThreads, Argv.size(), Argv.data()); | 139 gCompileServer->getParsedFlags(UseNumThreadsFromBrowser, NumThreads, |
140 Argv.size(), Argv.data()); | |
78 gCompileServer->startCompileThread(ObjFileFD); | 141 gCompileServer->startCompileThread(ObjFileFD); |
79 return nullptr; | 142 return nullptr; |
80 } | 143 } |
81 | 144 |
82 int onDataCallback(const void *Data, size_t NumBytes) { | 145 int onDataCallback(const void *Data, size_t NumBytes) { |
83 return gCompileServer->pushInputBytes(Data, NumBytes) ? 1 : 0; | 146 return gCompileServer->pushInputBytes(Data, NumBytes) ? 1 : 0; |
84 } | 147 } |
85 | 148 |
86 char *onEndCallback() { | 149 char *onEndCallback() { |
87 gCompileServer->endInputStream(); | 150 gCompileServer->endInputStream(); |
(...skipping 67 matching lines...) Loading... | |
155 } // end of anonymous namespace | 218 } // end of anonymous namespace |
156 | 219 |
157 BrowserCompileServer::~BrowserCompileServer() = default; | 220 BrowserCompileServer::~BrowserCompileServer() = default; |
158 | 221 |
159 void BrowserCompileServer::run() { | 222 void BrowserCompileServer::run() { |
160 gCompileServer = this; | 223 gCompileServer = this; |
161 getIRTInterfaces(); | 224 getIRTInterfaces(); |
162 gIRTFuncs.serve_translate_request(&SubzeroCallbacks); | 225 gIRTFuncs.serve_translate_request(&SubzeroCallbacks); |
163 } | 226 } |
164 | 227 |
165 void BrowserCompileServer::getParsedFlags(uint32_t NumThreads, int argc, | 228 void BrowserCompileServer::getParsedFlags(bool UseNumThreadsFromBrowser, |
229 uint32_t NumThreads, int argc, | |
166 char **argv) { | 230 char **argv) { |
167 ClFlags::parseFlags(argc, argv); | 231 ClFlags::parseFlags(argc, argv); |
168 ClFlags::getParsedClFlags(ClFlags::Flags); | 232 ClFlags::getParsedClFlags(ClFlags::Flags); |
169 // Set some defaults which aren't specified via the argv string. | 233 // Set some defaults which aren't specified via the argv string. |
170 ClFlags::Flags.setNumTranslationThreads(NumThreads); | 234 if (UseNumThreadsFromBrowser) |
235 ClFlags::Flags.setNumTranslationThreads(NumThreads); | |
171 ClFlags::Flags.setUseSandboxing(true); | 236 ClFlags::Flags.setUseSandboxing(true); |
172 ClFlags::Flags.setOutFileType(FT_Elf); | 237 ClFlags::Flags.setOutFileType(FT_Elf); |
173 ClFlags::Flags.setTargetArch(getTargetArch()); | 238 ClFlags::Flags.setTargetArch(getTargetArch()); |
174 ClFlags::Flags.setInputFileFormat(llvm::PNaClFormat); | 239 ClFlags::Flags.setInputFileFormat(llvm::PNaClFormat); |
175 } | 240 } |
176 | 241 |
177 bool BrowserCompileServer::pushInputBytes(const void *Data, size_t NumBytes) { | 242 bool BrowserCompileServer::pushInputBytes(const void *Data, size_t NumBytes) { |
178 // If there was an earlier error, do not attempt to push bytes to the | 243 // If there was an earlier error, do not attempt to push bytes to the |
179 // QueueStreamer. Otherwise the thread could become blocked. | 244 // QueueStreamer. Otherwise the thread could become blocked. |
180 if (HadError.load()) | 245 if (HadError.load()) |
(...skipping 58 matching lines...) Loading... | |
239 llvm::report_fatal_error("no browser hookups"); | 304 llvm::report_fatal_error("no browser hookups"); |
240 } | 305 } |
241 | 306 |
242 ErrorCode &BrowserCompileServer::getErrorCode() { | 307 ErrorCode &BrowserCompileServer::getErrorCode() { |
243 llvm::report_fatal_error("no browser hookups"); | 308 llvm::report_fatal_error("no browser hookups"); |
244 } | 309 } |
245 | 310 |
246 } // end of namespace Ice | 311 } // end of namespace Ice |
247 | 312 |
248 #endif // PNACL_BROWSER_TRANSLATOR | 313 #endif // PNACL_BROWSER_TRANSLATOR |
OLD | NEW |