Implement bash.exe detection logic.
The logic is as follows:
1) search for msys installation
2) search got git-on-Windows installation
3) search in PATH.
This happens on every client startup unless BAZEL_SH enviornment
variable is set.
My measurements show that the time required for this detection is
negligible (<10 msec in the worst case).
Change-Id: If130e2491a9df5a23954d303f2ccdb932eeed1db
PiperOrigin-RevId: 162466913
diff --git a/src/main/cpp/blaze_util_windows.cc b/src/main/cpp/blaze_util_windows.cc
index 802b3c2..7f0ca75 100644
--- a/src/main/cpp/blaze_util_windows.cc
+++ b/src/main/cpp/blaze_util_windows.cc
@@ -1448,4 +1448,263 @@
return true; // Nothing to do so assume success.
}
+static const int MAX_KEY_LENGTH = 255;
+// We do not care about registry values longer than MAX_PATH
+static const int REG_VALUE_BUFFER_SIZE = MAX_PATH;
+
+// Implements heuristics to discover msys2 installation.
+static string GetMsysBash() {
+ HKEY h_uninstall;
+
+ // MSYS2 installer writes its registry into HKCU, although documentation
+ // (https://msdn.microsoft.com/en-us/library/ms954376.aspx)
+ // clearly states that it should go to HKLM.
+ static const char* const key =
+ "Software\\Microsoft\\Windows\\CurrentVersion\\Uninstall";
+ if (RegOpenKeyExA(HKEY_CURRENT_USER, // _In_ HKEY hKey,
+ key, // _In_opt_ LPCTSTR lpSubKey,
+ 0, // _In_ DWORD ulOptions,
+ KEY_ENUMERATE_SUB_KEYS |
+ KEY_QUERY_VALUE, // _In_ REGSAM samDesired,
+ &h_uninstall // _Out_ PHKEY phkResult
+ )) {
+ debug_log("Cannot open HKCU\\%s", key);
+ return string();
+ }
+ AutoHandle auto_uninstall(h_uninstall);
+
+ // Since MSYS2 decided to generate a new product key for each installation,
+ // we enumerate all keys under
+ // HKCU\Software\Microsoft\Windows\CurrentVersion\Uninstall and find the first
+ // with MSYS2 64bit display name.
+ static const char* const msys_display_name = "MSYS2 64bit";
+ DWORD n_subkeys;
+
+ if (RegQueryInfoKey(h_uninstall, // _In_ HKEY hKey,
+ 0, // _Out_opt_ LPTSTR lpClass,
+ 0, // _Inout_opt_ LPDWORD lpcClass,
+ 0, // _Reserved_ LPDWORD lpReserved,
+ &n_subkeys, // _Out_opt_ LPDWORD lpcSubKeys,
+ 0, // _Out_opt_ LPDWORD lpcMaxSubKeyLen,
+ 0, // _Out_opt_ LPDWORD lpcMaxClassLen,
+ 0, // _Out_opt_ LPDWORD lpcValues,
+ 0, // _Out_opt_ LPDWORD lpcMaxValueNameLen,
+ 0, // _Out_opt_ LPDWORD lpcMaxValueLen,
+ 0, // _Out_opt_ LPDWORD lpcbSecurityDescriptor,
+ 0 // _Out_opt_ PFILETIME lpftLastWriteTime
+ )) {
+ debug_log("Cannot query HKCU\\%s", key);
+ return string();
+ }
+
+ for (DWORD key_index = 0; key_index < n_subkeys; key_index++) {
+ char subkey_name[MAX_KEY_LENGTH];
+ if (RegEnumKeyA(h_uninstall, // _In_ HKEY hKey,
+ key_index, // _In_ DWORD dwIndex,
+ subkey_name, // _Out_ LPTSTR lpName,
+ sizeof(subkey_name) // _In_ DWORD cchName
+ )) {
+ debug_log("Cannot get %d subkey of HKCU\\%s", key_index, key);
+ continue; // try next subkey
+ }
+
+ HKEY h_subkey;
+ if (RegOpenKeyEx(h_uninstall, // _In_ HKEY hKey,
+ subkey_name, // _In_opt_ LPCTSTR lpSubKey,
+ 0, // _In_ DWORD ulOptions,
+ KEY_QUERY_VALUE, // _In_ REGSAM samDesired,
+ &h_subkey // _Out_ PHKEY phkResult
+ )) {
+ debug_log("Failed to open subkey HKCU\\%s\\%s", key, subkey_name);
+ continue; // try next subkey
+ }
+ AutoHandle auto_subkey(h_subkey);
+
+ BYTE value[REG_VALUE_BUFFER_SIZE];
+ DWORD value_length = sizeof(value);
+ DWORD value_type;
+
+ if (RegQueryValueEx(h_subkey, // _In_ HKEY hKey,
+ "DisplayName", // _In_opt_ LPCTSTR lpValueName,
+ 0, // _Reserved_ LPDWORD lpReserved,
+ &value_type, // _Out_opt_ LPDWORD lpType,
+ value, // _Out_opt_ LPBYTE lpData,
+ &value_length // _Inout_opt_ LPDWORD lpcbData
+ )) {
+ debug_log("Failed to query DisplayName of HKCU\\%s\\%s", key,
+ subkey_name);
+ continue; // try next subkey
+ }
+
+ if (value_type == REG_SZ &&
+ 0 == memcmp(msys_display_name, value, sizeof(msys_display_name))) {
+ debug_log("Getting install location of HKCU\\%s\\%s", key, subkey_name);
+ BYTE path[REG_VALUE_BUFFER_SIZE];
+ DWORD path_length = sizeof(path);
+ DWORD path_type;
+ if (RegQueryValueEx(
+ h_subkey, // _In_ HKEY hKey,
+ "InstallLocation", // _In_opt_ LPCTSTR lpValueName,
+ 0, // _Reserved_ LPDWORD lpReserved,
+ &path_type, // _Out_opt_ LPDWORD lpType,
+ path, // _Out_opt_ LPBYTE lpData,
+ &path_length // _Inout_opt_ LPDWORD lpcbData
+ )) {
+ debug_log("Failed to query InstallLocation of HKCU\\%s\\%s", key,
+ subkey_name);
+ continue; // try next subkey
+ }
+
+ if (path_length == 0 || path_type != REG_SZ) {
+ debug_log("Zero-length (%d) install location or wrong type (%d)",
+ path_length, path_type);
+ continue; // try next subkey
+ }
+
+ debug_log("Install location of HKCU\\%s\\%s is %s", key, subkey_name,
+ path);
+ string path_as_string(path, path + path_length - 1);
+ string bash_exe = path_as_string + "\\usr\\bin\\bash.exe";
+ if (!blaze_util::PathExists(bash_exe)) {
+ debug_log("%s does not exist", bash_exe.c_str());
+ continue; // try next subkey
+ }
+
+ debug_log("Detected msys bash at %s", bash_exe.c_str());
+ return bash_exe;
+ }
+ }
+ return string();
+}
+
+// Implements heuristics to discover Git-on-Win installation.
+static string GetBashFromGitOnWin() {
+ HKEY h_GitOnWin_uninstall;
+
+ // Well-known registry key for Git-on-Windows.
+ static const char* const key =
+ "Software\\Microsoft\\Windows\\CurrentVersion\\Uninstall\\Git_is1";
+ if (RegOpenKeyExA(HKEY_LOCAL_MACHINE, // _In_ HKEY hKey,
+ key, // _In_opt_ LPCTSTR lpSubKey,
+ 0, // _In_ DWORD ulOptions,
+ KEY_QUERY_VALUE, // _In_ REGSAM samDesired,
+ &h_GitOnWin_uninstall // _Out_ PHKEY phkResult
+ )) {
+ debug_log("Cannot open HKCU\\%s", key);
+ return string();
+ }
+ AutoHandle auto_h_GitOnWin_uninstall(h_GitOnWin_uninstall);
+
+ debug_log("Getting install location of HKLM\\%s", key);
+ BYTE path[REG_VALUE_BUFFER_SIZE];
+ DWORD path_length = sizeof(path);
+ DWORD path_type;
+ if (RegQueryValueEx(h_GitOnWin_uninstall, // _In_ HKEY hKey,
+ "InstallLocation", // _In_opt_ LPCTSTR lpValueName,
+ 0, // _Reserved_ LPDWORD lpReserved,
+ &path_type, // _Out_opt_ LPDWORD lpType,
+ path, // _Out_opt_ LPBYTE lpData,
+ &path_length // _Inout_opt_ LPDWORD lpcbData
+ )) {
+ debug_log("Failed to query InstallLocation of HKLM\\%s", key);
+ return string();
+ }
+
+ if (path_length == 0 || path_type != REG_SZ) {
+ debug_log("Zero-length (%d) install location or wrong type (%d)",
+ path_length, path_type);
+ return string();
+ }
+
+ debug_log("Install location of HKLM\\%s is %s", key, path);
+ string path_as_string(path, path + path_length - 1);
+ string bash_exe = path_as_string + "\\usr\\bin\\bash.exe";
+ if (!blaze_util::PathExists(bash_exe)) {
+ debug_log("%s does not exist", bash_exe.c_str());
+ return string();
+ }
+
+ debug_log("Detected msys bash at %s", bash_exe.c_str());
+ return bash_exe;
+}
+
+static string GetBashFromPath() {
+ char found[MAX_PATH];
+ string path_list = blaze::GetEnv("PATH");
+
+ // We do not fully replicate all the quirks of search in PATH.
+ // There is no system function to do so, and that way lies madness.
+ size_t start = 0;
+ do {
+ // This ignores possibly quoted semicolons in PATH etc.
+ size_t end = path_list.find_first_of(";", start);
+ string path = path_list.substr(
+ start, end != string::npos ? end - start : string::npos);
+ // Handle one typical way of quoting (where.exe does not handle this, but
+ // CreateProcess does).
+ if (path.size() > 1 && path[0] == '"' && path[path.size() - 1] == '"') {
+ path = path.substr(1, path.size() - 2);
+ }
+ if (SearchPathA(path.c_str(), // _In_opt_ LPCTSTR lpPath,
+ "bash.exe", // _In_ LPCTSTR lpFileName,
+ 0, // LPCTSTR lpExtension,
+ sizeof(found), // DWORD nBufferLength,
+ found, // _Out_ LPTSTR lpBuffer,
+ 0 // _Out_opt_ LPTSTR *lpFilePart
+ )) {
+ debug_log("bash.exe found on PATH: %s", found);
+ return string(found);
+ }
+ if (end == string::npos) {
+ break;
+ }
+ start = end + 1;
+ } while (true);
+
+ debug_log("bash.exe not found on PATH");
+ return string();
+}
+
+static string LocateBash() {
+ string msys_bash = GetMsysBash();
+ if (!msys_bash.empty()) {
+ return msys_bash;
+ }
+
+ string git_on_win_bash = GetBashFromGitOnWin();
+ if (!git_on_win_bash.empty()) {
+ return git_on_win_bash;
+ }
+
+ return GetBashFromPath();
+}
+
+void DetectBashOrDie() {
+ if (!blaze::GetEnv("BAZEL_SH").empty()) return;
+
+ uint64_t start = blaze::GetMillisecondsMonotonic();
+
+ string bash = LocateBash();
+ uint64_t end = blaze::GetMillisecondsMonotonic();
+ debug_log("BAZEL_SH detection took %lu msec", end - start);
+
+ if (!bash.empty()) {
+ blaze::SetEnv("BAZEL_SH", bash);
+ } else {
+ printf(
+ "Bazel on Windows requires bash.exe and other Unix tools, but we could "
+ "not find them.\n"
+ "If you do not have them installed, the easiest is to install MSYS2 "
+ "from\n"
+ " http://repo.msys2.org/distrib/msys2-x86_64-latest.exe\n"
+ "or git-on-Windows from\n"
+ " https://git-scm.com/download/win\n"
+ "\n"
+ "If you already have bash.exe installed but Bazel cannot find it,\n"
+ "set BAZEL_SH environment variable to it's location:\n"
+ " set BAZEL_SH=c:\\path\\to\\bash.exe\n");
+ exit(1);
+ }
+}
+
} // namespace blaze