blob: 9ed930d1bf6fbeb377ce450e4213b0cf34c4056f [file] [log] [blame]
// 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 "src/main/native/windows/util.h"
#include <stdio.h>
#include <stdlib.h>
#include <windows.h>
#include <algorithm>
#include <sstream>
#include <string>
namespace bazel {
namespace windows {
using std::wstring;
using std::wstringstream;
wstring MakeErrorMessage(const wchar_t* file, int line,
const wchar_t* failed_func, const wstring& func_arg,
const wstring& message) {
wstringstream result;
result << L"ERROR: " << file << L"(" << line << L"): " << failed_func << L"("
<< func_arg << L"): " << message;
return result.str();
}
wstring MakeErrorMessage(const wchar_t* file, int line,
const wchar_t* failed_func, const wstring& func_arg,
DWORD error_code) {
return MakeErrorMessage(file, line, failed_func, func_arg,
GetLastErrorString(error_code));
}
wstring GetLastErrorString(DWORD error_code) {
if (error_code == 0) {
return L"";
}
LPWSTR message = NULL;
DWORD size = FormatMessageW(
FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS |
FORMAT_MESSAGE_ALLOCATE_BUFFER,
NULL, error_code, LANG_USER_DEFAULT, (LPWSTR)&message, 0, NULL);
if (size == 0) {
wstringstream err;
DWORD format_message_error = GetLastError();
err << L"Error code " << error_code
<< L"; cannot format message due to error code "
<< format_message_error;
return err.str();
}
wstring result(message);
HeapFree(GetProcessHeap(), LMEM_FIXED, message);
return result;
}
static void QuotePath(const wstring& path, wstring* result) {
*result = wstring(L"\"") + path + L"\"";
}
static bool IsSeparator(WCHAR c) { return c == L'/' || c == L'\\'; }
static bool HasSeparator(const wstring& s) {
return s.find_first_of(L'/') != wstring::npos ||
s.find_first_of(L'\\') != wstring::npos;
}
static bool Contains(const wstring& s, const WCHAR* substr) {
return s.find(substr) != wstring::npos;
}
wstring AsShortPath(wstring path, wstring* result) {
if (path.empty()) {
result->clear();
return L"";
}
if (path[0] == '"') {
return MakeErrorMessage(WSTR(__FILE__), __LINE__, L"AsShortPath", path,
L"path should not be quoted");
}
if (IsSeparator(path[0])) {
return MakeErrorMessage(WSTR(__FILE__), __LINE__, L"AsShortPath", path,
L"path is absolute without a drive letter");
}
if (Contains(path, L"/./") || Contains(path, L"\\.\\") ||
Contains(path, L"/..") || Contains(path, L"\\..")) {
return MakeErrorMessage(WSTR(__FILE__), __LINE__, L"AsShortPath", path,
L"path is not normalized");
}
if (path.size() >= MAX_PATH && !HasSeparator(path)) {
return MakeErrorMessage(WSTR(__FILE__), __LINE__, L"AsShortPath", path,
L"path is just a file name but too long");
}
if (HasSeparator(path) &&
!(isalpha(path[0]) && path[1] == L':' && IsSeparator(path[2]))) {
return MakeErrorMessage(WSTR(__FILE__), __LINE__, L"AsShortPath", path,
L"path is not absolute");
}
// 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 L"";
}
// At this point we know that the path is at least MAX_PATH long and that it's
// absolute, normalized, and Windows-style.
wstring wlong = wstring(L"\\\\?\\") + path;
// 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) {
DWORD err_code = GetLastError();
wstring res = MakeErrorMessage(WSTR(__FILE__), __LINE__,
L"GetShortPathNameW", wlong, err_code);
return res;
}
if (wshort_size >= kMaxShortPath) {
return MakeErrorMessage(WSTR(__FILE__), __LINE__, L"GetShortPathNameW",
wlong, L"cannot shorten the path enough");
}
GetShortPathNameW(wlong.c_str(), wshort, kMaxShortPath);
result->assign(wshort + 4);
return L"";
}
wstring AsExecutablePathForCreateProcess(const wstring& path, wstring* result) {
if (path.empty()) {
return MakeErrorMessage(WSTR(__FILE__), __LINE__,
L"AsExecutablePathForCreateProcess", path,
L"path should not be empty");
}
wstring error = AsShortPath(path, result);
if (!error.empty()) {
return MakeErrorMessage(WSTR(__FILE__), __LINE__,
L"AsExecutablePathForCreateProcess", path, error);
}
// 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 L"";
}
} // namespace windows
} // namespace bazel