// Copyright 2016 The Bazel Authors. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include #include #include #include #include #include #include #include "src/main/native/windows/util.h" namespace bazel { namespace windows { using std::function; using std::string; using std::unique_ptr; using std::wstring; string GetLastErrorString(const string& cause) { DWORD last_error = GetLastError(); if (last_error == 0) { return ""; } LPSTR message; DWORD size = FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, last_error, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPSTR)&message, 0, NULL); if (size == 0) { char buf[256]; snprintf(buf, sizeof(buf), "%s: Error %d (cannot format message due to error %d)", cause.c_str(), last_error, GetLastError()); buf[sizeof(buf) - 1] = 0; } string result = string(message); LocalFree(message); return cause + ": " + result; } static void QuotePath(const string& path, string* result) { *result = string("\"") + path + "\""; } static bool IsSeparator(char c) { return c == '/' || c == '\\'; } static bool HasSeparator(const string& s) { return s.find_first_of('/') != string::npos || s.find_first_of('\\') != string::npos; } static bool Contains(const string& s, const char* substr) { return s.find(substr) != string::npos; } string AsShortPath(string path, function path_as_wstring, string* result) { if (path.empty()) { result->clear(); return ""; } if (path[0] == '"') { return string("path should not be quoted"); } if (IsSeparator(path[0])) { return string("path='") + path + "' is absolute"; } if (Contains(path, "/./") || Contains(path, "\\.\\") || Contains(path, "/..") || Contains(path, "\\..")) { return string("path='") + path + "' is not normalized"; } if (path.size() >= MAX_PATH && !HasSeparator(path)) { return string("path='") + path + "' is just a file name but too long"; } if (HasSeparator(path) && !(isalpha(path[0]) && path[1] == ':' && IsSeparator(path[2]))) { return string("path='") + path + "' is not an absolute path"; } // At this point we know the path is either just a file name (shorter than // MAX_PATH), or an absolute, normalized, Windows-style path (of any length). std::replace(path.begin(), path.end(), '/', '\\'); // Fast-track: the path is already short. if (path.size() < MAX_PATH) { *result = path; return ""; } // At this point we know that the path is at least MAX_PATH long and that it's // absolute, normalized, and Windows-style. // Retrieve string as UTF-16 path, add "\\?\" prefix. wstring wlong = wstring(L"\\\\?\\") + path_as_wstring(); // Experience shows that: // - GetShortPathNameW's result has a "\\?\" prefix if and only if the input // did too (though this behavior is not documented on MSDN) // - CreateProcess{A,W} only accept an executable of MAX_PATH - 1 length // Therefore for our purposes the acceptable shortened length is // MAX_PATH + 4 (null-terminated). That is, MAX_PATH - 1 for the shortened // path, plus a potential "\\?\" prefix that's only there if `wlong` also had // it and which we'll omit from `result`, plus a null terminator. static const size_t kMaxShortPath = MAX_PATH + 4; WCHAR wshort[kMaxShortPath]; DWORD wshort_size = ::GetShortPathNameW(wlong.c_str(), NULL, 0); if (wshort_size == 0) { return GetLastErrorString(string("GetShortPathName failed (path=") + path + ")"); } if (wshort_size >= kMaxShortPath) { return string("GetShortPathName would not shorten the path enough (path=") + path + ")"; } GetShortPathNameW(wlong.c_str(), wshort, kMaxShortPath); // Convert the result to UTF-8. char mbs_short[MAX_PATH]; size_t mbs_size = wcstombs( mbs_short, wshort + 4, // we know it has a "\\?\" prefix, because `wlong` also did MAX_PATH); if (mbs_size < 0 || mbs_size >= MAX_PATH) { return string("wcstombs failed (path=") + path + ")"; } mbs_short[mbs_size] = 0; *result = mbs_short; return ""; } string AsExecutablePathForCreateProcess(const string& path, function path_as_wstring, string* result) { if (path.empty()) { return string("path should not be empty"); } string error = AsShortPath(path, path_as_wstring, result); if (error.empty()) { // Quote the path in case it's something like "c:\foo\app name.exe". // Do this unconditionally, there's no harm in quoting. Quotes are not // allowed inside paths so we don't need to escape quotes. QuotePath(*result, result); } return error; } } // namespace windows } // namespace bazel