Damien Martin-Guillerez | f88f4d8 | 2015-09-25 13:56:55 +0000 | [diff] [blame] | 1 | // Copyright 2014 The Bazel Authors. All rights reserved. |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 2 | // |
| 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-Moore | 44e8d10 | 2015-11-10 11:27:39 +0000 | [diff] [blame] | 15 | #include <errno.h> // errno, ENAMETOOLONG |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 16 | #include <limits.h> |
| 17 | #include <string.h> // strerror |
Mostyn Bramley-Moore | 44e8d10 | 2015-11-10 11:27:39 +0000 | [diff] [blame] | 18 | #include <sys/cygwin.h> |
Googler | 11565c1 | 2015-07-23 12:03:53 +0000 | [diff] [blame] | 19 | #include <sys/socket.h> |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 20 | #include <sys/statfs.h> |
| 21 | #include <unistd.h> |
| 22 | |
Dmitry Lomov | 78c0cc7 | 2015-08-11 16:44:21 +0000 | [diff] [blame] | 23 | #include <windows.h> |
| 24 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 25 | #include <cstdlib> |
| 26 | #include <cstdio> |
| 27 | |
Han-Wen Nienhuys | 36fbe63 | 2015-04-21 13:58:08 +0000 | [diff] [blame] | 28 | #include "src/main/cpp/blaze_util.h" |
Thiago Farina | 7f9357f | 2015-04-23 13:57:43 +0000 | [diff] [blame] | 29 | #include "src/main/cpp/blaze_util_platform.h" |
Han-Wen Nienhuys | 36fbe63 | 2015-04-21 13:58:08 +0000 | [diff] [blame] | 30 | #include "src/main/cpp/util/errors.h" |
Thiago Farina | 7f9357f | 2015-04-23 13:57:43 +0000 | [diff] [blame] | 31 | #include "src/main/cpp/util/exit_code.h" |
Han-Wen Nienhuys | 36fbe63 | 2015-04-21 13:58:08 +0000 | [diff] [blame] | 32 | #include "src/main/cpp/util/file.h" |
| 33 | #include "src/main/cpp/util/strings.h" |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 34 | |
| 35 | namespace blaze { |
| 36 | |
Thiago Farina | 241f46c | 2015-04-13 14:33:30 +0000 | [diff] [blame] | 37 | using blaze_util::die; |
| 38 | using blaze_util::pdie; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 39 | using std::string; |
Dmitry Lomov | f7d3eb7 | 2015-08-13 20:45:09 +0000 | [diff] [blame] | 40 | using std::vector; |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 41 | |
| 42 | void WarnFilesystemType(const string& output_base) { |
| 43 | } |
| 44 | |
| 45 | string GetSelfPath() { |
| 46 | char buffer[PATH_MAX] = {}; |
Dmitry Lomov | 3157412 | 2016-02-17 16:07:31 +0000 | [diff] [blame] | 47 | if (!GetModuleFileName(0, buffer, sizeof(buffer))) { |
| 48 | pdie(255, "Error %u getting executable file name\n", GetLastError()); |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 49 | } |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 50 | return string(buffer); |
| 51 | } |
| 52 | |
Kristina Chodorow | 9396335 | 2015-03-20 19:11:19 +0000 | [diff] [blame] | 53 | string GetOutputRoot() { |
Dmitry Lomov | bc84cc8 | 2016-04-15 14:05:24 +0000 | [diff] [blame^] | 54 | char* tmpdir = getenv("TMPDIR"); |
| 55 | if (tmpdir == 0 || strlen(tmpdir) == 0) { |
| 56 | return "/var/tmp"; |
| 57 | } else { |
| 58 | return string(tmpdir); |
| 59 | } |
Kristina Chodorow | 9396335 | 2015-03-20 19:11:19 +0000 | [diff] [blame] | 60 | } |
| 61 | |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 62 | pid_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 Farina | 8a67da4 | 2015-05-05 18:04:50 +0000 | [diff] [blame] | 72 | uint64_t MonotonicClock() { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 73 | struct timespec ts = {}; |
| 74 | clock_gettime(CLOCK_MONOTONIC, &ts); |
| 75 | return ts.tv_sec * 1000000000LL + ts.tv_nsec; |
| 76 | } |
| 77 | |
Thiago Farina | 8a67da4 | 2015-05-05 18:04:50 +0000 | [diff] [blame] | 78 | uint64_t ProcessClock() { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 79 | struct timespec ts = {}; |
| 80 | clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &ts); |
| 81 | return ts.tv_sec * 1000000000LL + ts.tv_nsec; |
| 82 | } |
| 83 | |
| 84 | void SetScheduling(bool batch_cpu_scheduling, int io_nice_level) { |
| 85 | // TODO(bazel-team): There should be a similar function on Windows. |
| 86 | } |
| 87 | |
| 88 | string GetProcessCWD(int pid) { |
| 89 | char server_cwd[PATH_MAX] = {}; |
| 90 | if (readlink( |
Googler | 9588b81 | 2015-07-23 11:49:37 +0000 | [diff] [blame] | 91 | ("/proc/" + ToString(pid) + "/cwd").c_str(), |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 92 | server_cwd, sizeof(server_cwd)) < 0) { |
| 93 | return ""; |
| 94 | } |
| 95 | |
| 96 | return string(server_cwd); |
| 97 | } |
| 98 | |
Thiago Farina | 01f3600 | 2015-04-08 15:59:08 +0000 | [diff] [blame] | 99 | bool IsSharedLibrary(const string &filename) { |
Han-Wen Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 100 | return blaze_util::ends_with(filename, ".dll"); |
| 101 | } |
| 102 | |
| 103 | string 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 Lomov | b5ff50b | 2016-02-03 19:35:42 +0000 | [diff] [blame] | 112 | namespace { |
| 113 | void 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 Lomov | b5ff50b | 2016-02-03 19:35:42 +0000 | [diff] [blame] | 123 | |
Dmitry Lomov | 47afaab | 2016-02-19 08:21:13 +0000 | [diff] [blame] | 124 | // Run the given program in the current working directory, |
| 125 | // using the given argument vector. |
| 126 | DWORD CreateProcessWrapper( |
| 127 | const string& exe, const vector<string>& args_vector) { |
Dmitry Lomov | 78c0cc7 | 2015-08-11 16:44:21 +0000 | [diff] [blame] | 128 | 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 Lomov | b5ff50b | 2016-02-03 19:35:42 +0000 | [diff] [blame] | 160 | |
| 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 Lomov | 78c0cc7 | 2015-08-11 16:44:21 +0000 | [diff] [blame] | 175 | } |
| 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 Lomov | 45d07a1 | 2016-02-03 21:43:47 +0000 | [diff] [blame] | 194 | // 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 Lomov | 78c0cc7 | 2015-08-11 16:44:21 +0000 | [diff] [blame] | 199 | 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 Lomov | 47afaab | 2016-02-19 08:21:13 +0000 | [diff] [blame] | 219 | return exit_code; |
| 220 | } |
| 221 | } // namespace |
Dmitry Lomov | 78c0cc7 | 2015-08-11 16:44:21 +0000 | [diff] [blame] | 222 | |
Dmitry Lomov | 47afaab | 2016-02-19 08:21:13 +0000 | [diff] [blame] | 223 | // 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. |
| 226 | void ExecuteProgram(const string& exe, const vector<string>& args_vector) { |
Dmitry Lomov | 78c0cc7 | 2015-08-11 16:44:21 +0000 | [diff] [blame] | 227 | // Emulate execv. |
Dmitry Lomov | 47afaab | 2016-02-19 08:21:13 +0000 | [diff] [blame] | 228 | exit(CreateProcessWrapper(exe, args_vector)); |
Dmitry Lomov | 78c0cc7 | 2015-08-11 16:44:21 +0000 | [diff] [blame] | 229 | } |
| 230 | |
| 231 | string ListSeparator() { return ";"; } |
| 232 | |
| 233 | string 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 Lomov | 47afaab | 2016-02-19 08:21:13 +0000 | [diff] [blame] | 241 | bool 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 Nienhuys | d08b27f | 2015-02-25 16:45:20 +0100 | [diff] [blame] | 254 | } // namespace blaze |