blob: e27ae18a924c19a144fdae812c81579e113d89e1 [file] [log] [blame]
Damien Martin-Guillerezf88f4d82015-09-25 13:56:55 +00001// Copyright 2014 The Bazel Authors. All rights reserved.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01002//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
Mostyn Bramley-Moore44e8d102015-11-10 11:27:39 +000015#include <errno.h> // errno, ENAMETOOLONG
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010016#include <limits.h>
17#include <string.h> // strerror
Mostyn Bramley-Moore44e8d102015-11-10 11:27:39 +000018#include <sys/cygwin.h>
Googler11565c12015-07-23 12:03:53 +000019#include <sys/socket.h>
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010020#include <sys/statfs.h>
21#include <unistd.h>
22
Dmitry Lomov78c0cc72015-08-11 16:44:21 +000023#include <windows.h>
24
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010025#include <cstdlib>
26#include <cstdio>
27
Han-Wen Nienhuys36fbe632015-04-21 13:58:08 +000028#include "src/main/cpp/blaze_util.h"
Thiago Farina7f9357f2015-04-23 13:57:43 +000029#include "src/main/cpp/blaze_util_platform.h"
Han-Wen Nienhuys36fbe632015-04-21 13:58:08 +000030#include "src/main/cpp/util/errors.h"
Thiago Farina7f9357f2015-04-23 13:57:43 +000031#include "src/main/cpp/util/exit_code.h"
Han-Wen Nienhuys36fbe632015-04-21 13:58:08 +000032#include "src/main/cpp/util/file.h"
33#include "src/main/cpp/util/strings.h"
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010034
35namespace blaze {
36
Thiago Farina241f46c2015-04-13 14:33:30 +000037using blaze_util::die;
38using blaze_util::pdie;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010039using std::string;
Dmitry Lomovf7d3eb72015-08-13 20:45:09 +000040using std::vector;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010041
42void WarnFilesystemType(const string& output_base) {
43}
44
45string GetSelfPath() {
46 char buffer[PATH_MAX] = {};
Dmitry Lomov31574122016-02-17 16:07:31 +000047 if (!GetModuleFileName(0, buffer, sizeof(buffer))) {
48 pdie(255, "Error %u getting executable file name\n", GetLastError());
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010049 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010050 return string(buffer);
51}
52
Kristina Chodorow93963352015-03-20 19:11:19 +000053string GetOutputRoot() {
Dmitry Lomovbc84cc82016-04-15 14:05:24 +000054 char* tmpdir = getenv("TMPDIR");
55 if (tmpdir == 0 || strlen(tmpdir) == 0) {
56 return "/var/tmp";
57 } else {
58 return string(tmpdir);
59 }
Kristina Chodorow93963352015-03-20 19:11:19 +000060}
61
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010062pid_t GetPeerProcessId(int socket) {
63 struct ucred creds = {};
64 socklen_t len = sizeof creds;
65 if (getsockopt(socket, SOL_SOCKET, SO_PEERCRED, &creds, &len) == -1) {
66 pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
67 "can't get server pid from connection");
68 }
69 return creds.pid;
70}
71
Thiago Farina8a67da42015-05-05 18:04:50 +000072uint64_t MonotonicClock() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010073 struct timespec ts = {};
74 clock_gettime(CLOCK_MONOTONIC, &ts);
75 return ts.tv_sec * 1000000000LL + ts.tv_nsec;
76}
77
Thiago Farina8a67da42015-05-05 18:04:50 +000078uint64_t ProcessClock() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010079 struct timespec ts = {};
80 clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &ts);
81 return ts.tv_sec * 1000000000LL + ts.tv_nsec;
82}
83
84void SetScheduling(bool batch_cpu_scheduling, int io_nice_level) {
85 // TODO(bazel-team): There should be a similar function on Windows.
86}
87
88string GetProcessCWD(int pid) {
89 char server_cwd[PATH_MAX] = {};
90 if (readlink(
Googler9588b812015-07-23 11:49:37 +000091 ("/proc/" + ToString(pid) + "/cwd").c_str(),
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010092 server_cwd, sizeof(server_cwd)) < 0) {
93 return "";
94 }
95
96 return string(server_cwd);
97}
98
Thiago Farina01f36002015-04-08 15:59:08 +000099bool IsSharedLibrary(const string &filename) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100100 return blaze_util::ends_with(filename, ".dll");
101}
102
103string GetDefaultHostJavabase() {
104 const char *javahome = getenv("JAVA_HOME");
105 if (javahome == NULL) {
106 die(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
107 "Error: JAVA_HOME not set.");
108 }
109 return javahome;
110}
111
Dmitry Lomovb5ff50b2016-02-03 19:35:42 +0000112namespace {
113void ReplaceAll(
114 std::string* s, const std::string& pattern, const std::string with) {
115 size_t pos = 0;
116 while (true) {
117 size_t pos = s->find(pattern, pos);
118 if (pos == std::string::npos) return;
119 *s = s->replace(pos, pattern.length(), with);
120 pos += with.length();
121 }
122}
Dmitry Lomovb5ff50b2016-02-03 19:35:42 +0000123
Dmitry Lomov47afaab2016-02-19 08:21:13 +0000124// Run the given program in the current working directory,
125// using the given argument vector.
126DWORD CreateProcessWrapper(
127 const string& exe, const vector<string>& args_vector) {
Dmitry Lomov78c0cc72015-08-11 16:44:21 +0000128 if (VerboseLogging()) {
129 string dbg;
130 for (const auto& s : args_vector) {
131 dbg.append(s);
132 dbg.append(" ");
133 }
134
135 char cwd[PATH_MAX] = {};
136 if (getcwd(cwd, sizeof(cwd)) == NULL) {
137 pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR, "getcwd() failed");
138 }
139
140 fprintf(stderr, "Invoking binary %s in %s:\n %s\n", exe.c_str(), cwd,
141 dbg.c_str());
142 }
143
144 // Build full command line.
145 string cmdline;
146 bool first = true;
147 for (const auto& s : args_vector) {
148 if (first) {
149 first = false;
150 // Skip first argument, instead use quoted executable name with ".exe"
151 // suffix.
152 cmdline.append("\"");
153 cmdline.append(exe);
154 cmdline.append(".exe");
155 cmdline.append("\"");
156 continue;
157 } else {
158 cmdline.append(" ");
159 }
Dmitry Lomovb5ff50b2016-02-03 19:35:42 +0000160
161 string arg = s;
162 // Quote quotes.
163 if (s.find("\"") != string::npos) {
164 ReplaceAll(&arg, "\"", "\\\"");
165 }
166
167 // Quotize spaces.
168 if (arg.find(" ") != string::npos) {
169 cmdline.append("\"");
170 cmdline.append(arg);
171 cmdline.append("\"");
172 } else {
173 cmdline.append(arg);
174 }
Dmitry Lomov78c0cc72015-08-11 16:44:21 +0000175 }
176
177 // Copy command line into a mutable buffer.
178 // CreateProcess is allowed to mutate its command line argument.
179 // Max command line length is per CreateProcess documentation
180 // (https://msdn.microsoft.com/en-us/library/ms682425(VS.85).aspx)
181 static const int kMaxCmdLineLength = 32768;
182 char actual_line[kMaxCmdLineLength];
183 if (cmdline.length() >= kMaxCmdLineLength) {
184 pdie(255, "Command line too long: %s", cmdline.c_str());
185 }
186 strncpy(actual_line, cmdline.c_str(), kMaxCmdLineLength);
187 // Add trailing '\0' to be sure.
188 actual_line[kMaxCmdLineLength - 1] = '\0';
189
190 // Execute program.
191 STARTUPINFO startupinfo = {0};
192 PROCESS_INFORMATION pi = {0};
193
Dmitry Lomov45d07a12016-02-03 21:43:47 +0000194 // Propagate BAZEL_SH environment variable to a sub-process.
195 // todo(dslomov): More principled approach to propagating
196 // environment variables.
197 SetEnvironmentVariable("BAZEL_SH", getenv("BAZEL_SH"));
198
Dmitry Lomov78c0cc72015-08-11 16:44:21 +0000199 bool success = CreateProcess(
200 nullptr, // _In_opt_ LPCTSTR lpApplicationName,
201 actual_line, // _Inout_opt_ LPTSTR lpCommandLine,
202 nullptr, // _In_opt_ LPSECURITY_ATTRIBUTES lpProcessAttributes,
203 nullptr, // _In_opt_ LPSECURITY_ATTRIBUTES lpThreadAttributes,
204 true, // _In_ BOOL bInheritHandles,
205 0, // _In_ DWORD dwCreationFlags,
206 nullptr, // _In_opt_ LPVOID lpEnvironment,
207 nullptr, // _In_opt_ LPCTSTR lpCurrentDirectory,
208 &startupinfo, // _In_ LPSTARTUPINFO lpStartupInfo,
209 &pi); // _Out_ LPPROCESS_INFORMATION lpProcessInformation
210
211 if (!success) {
212 pdie(255, "Error %u executing: %s\n", GetLastError(), actual_line);
213 }
214 WaitForSingleObject(pi.hProcess, INFINITE);
215 DWORD exit_code;
216 GetExitCodeProcess(pi.hProcess, &exit_code);
217 CloseHandle(pi.hProcess);
218 CloseHandle(pi.hThread);
Dmitry Lomov47afaab2016-02-19 08:21:13 +0000219 return exit_code;
220}
221} // namespace
Dmitry Lomov78c0cc72015-08-11 16:44:21 +0000222
Dmitry Lomov47afaab2016-02-19 08:21:13 +0000223// Replace the current process with the given program in the current working
224// directory, using the given argument vector.
225// This function does not return on success.
226void ExecuteProgram(const string& exe, const vector<string>& args_vector) {
Dmitry Lomov78c0cc72015-08-11 16:44:21 +0000227 // Emulate execv.
Dmitry Lomov47afaab2016-02-19 08:21:13 +0000228 exit(CreateProcessWrapper(exe, args_vector));
Dmitry Lomov78c0cc72015-08-11 16:44:21 +0000229}
230
231string ListSeparator() { return ";"; }
232
233string ConvertPath(const string& path) {
234 char* wpath = static_cast<char*>(cygwin_create_path(
235 CCP_POSIX_TO_WIN_A, static_cast<const void*>(path.c_str())));
236 string result(wpath);
237 free(wpath);
238 return result;
239}
240
Dmitry Lomov47afaab2016-02-19 08:21:13 +0000241bool SymlinkDirectories(const string &target, const string &link) {
242 const string target_win = ConvertPath(target);
243 const string link_win = ConvertPath(link);
244 vector<string> args;
245 args.push_back("cmd");
246 args.push_back("/C");
247 args.push_back("mklink");
248 args.push_back("/J");
249 args.push_back(link_win);
250 args.push_back(target_win);
251 return CreateProcessWrapper("cmd", args) == 0;
252}
253
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100254} // namespace blaze