|  | // 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. | 
|  |  | 
|  | #ifndef BAZEL_SRC_MAIN_NATIVE_WINDOWS_UTIL_H__ | 
|  | #define BAZEL_SRC_MAIN_NATIVE_WINDOWS_UTIL_H__ | 
|  |  | 
|  | #ifndef WIN32_LEAN_AND_MEAN | 
|  | #define WIN32_LEAN_AND_MEAN | 
|  | #endif | 
|  |  | 
|  | #include <windows.h> | 
|  |  | 
|  | #include <memory> | 
|  | #include <string> | 
|  |  | 
|  | namespace bazel { | 
|  | namespace windows { | 
|  |  | 
|  | using std::wstring; | 
|  |  | 
|  | // A wrapper for the `HANDLE` type that calls CloseHandle in its d'tor. | 
|  | // WARNING: do not use for HANDLE returned by FindFirstFile; those must be | 
|  | // closed with FindClose (otherwise they aren't closed properly). | 
|  | class AutoHandle { | 
|  | public: | 
|  | explicit AutoHandle(HANDLE handle = INVALID_HANDLE_VALUE) : handle_(handle) {} | 
|  | AutoHandle(const AutoHandle&) = delete; | 
|  | AutoHandle(AutoHandle&& other) = delete; | 
|  | AutoHandle& operator=(const AutoHandle&) = delete; | 
|  | AutoHandle& operator=(AutoHandle&& other) = delete; | 
|  |  | 
|  | ~AutoHandle() { | 
|  | if (IsValid()) { | 
|  | ::CloseHandle(handle_); | 
|  | } | 
|  | } | 
|  |  | 
|  | AutoHandle(AutoHandle* other) : handle_(other->handle_) { | 
|  | other->handle_ = INVALID_HANDLE_VALUE; | 
|  | } | 
|  |  | 
|  | bool IsValid() const { | 
|  | return handle_ != INVALID_HANDLE_VALUE && handle_ != nullptr; | 
|  | } | 
|  |  | 
|  | AutoHandle& operator=(const HANDLE& rhs) { | 
|  | if (IsValid()) { | 
|  | ::CloseHandle(handle_); | 
|  | } | 
|  | handle_ = rhs; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | operator HANDLE() const { return handle_; } | 
|  |  | 
|  | private: | 
|  | HANDLE handle_; | 
|  | }; | 
|  |  | 
|  | class AutoAttributeList { | 
|  | public: | 
|  | AutoAttributeList() {} | 
|  |  | 
|  | static bool Create(HANDLE stdin_h, HANDLE stdout_h, HANDLE stderr_h, | 
|  | std::unique_ptr<AutoAttributeList>* result, | 
|  | std::wstring* error_msg = nullptr); | 
|  | ~AutoAttributeList(); | 
|  |  | 
|  | bool InheritAnyHandles() const { return handles_.ValidHandlesCount() > 0; } | 
|  |  | 
|  | void InitStartupInfoExW(STARTUPINFOEXW* startup_info) const; | 
|  |  | 
|  | bool HasConsoleHandle() const { return handles_.HasConsoleHandle(); } | 
|  |  | 
|  | private: | 
|  | class StdHandles { | 
|  | public: | 
|  | StdHandles(); | 
|  | StdHandles(HANDLE stdin_h, HANDLE stdout_h, HANDLE stderr_h); | 
|  | size_t ValidHandlesCount() const { return valid_handles_; } | 
|  | HANDLE* ValidHandles() { return valid_handle_array_; } | 
|  | HANDLE StdIn() const { return stdin_h_; } | 
|  | HANDLE StdOut() const { return stdout_h_; } | 
|  | HANDLE StdErr() const { return stderr_h_; } | 
|  |  | 
|  | bool HasConsoleHandle() const { | 
|  | for (size_t i = 0; i < valid_handles_; ++i) { | 
|  | if (GetFileType(valid_handle_array_[i]) == FILE_TYPE_CHAR) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | private: | 
|  | size_t valid_handles_; | 
|  | HANDLE valid_handle_array_[3]; | 
|  | HANDLE stdin_h_; | 
|  | HANDLE stdout_h_; | 
|  | HANDLE stderr_h_; | 
|  | }; | 
|  |  | 
|  | AutoAttributeList(std::unique_ptr<uint8_t[]>&& data, HANDLE stdin_h, | 
|  | HANDLE stdout_h, HANDLE stderr_h); | 
|  | AutoAttributeList(const AutoAttributeList&) = delete; | 
|  | AutoAttributeList& operator=(const AutoAttributeList&) = delete; | 
|  |  | 
|  | operator LPPROC_THREAD_ATTRIBUTE_LIST() const; | 
|  |  | 
|  | std::unique_ptr<uint8_t[]> data_; | 
|  | StdHandles handles_; | 
|  | }; | 
|  |  | 
|  | #define WSTR1(x) L##x | 
|  | #define WSTR(x) WSTR1(x) | 
|  |  | 
|  | wstring MakeErrorMessage(const wchar_t* file, int line, | 
|  | const wchar_t* failed_func, const wstring& func_arg, | 
|  | const wstring& message); | 
|  | wstring MakeErrorMessage(const wchar_t* file, int line, | 
|  | const wchar_t* failed_func, const wstring& func_arg, | 
|  | DWORD error_code); | 
|  | wstring GetLastErrorString(DWORD error_code); | 
|  |  | 
|  | // Same as `AsExecutablePathForCreateProcess` except it won't quote the result. | 
|  | wstring AsShortPath(wstring path, wstring* result); | 
|  |  | 
|  | // Computes a path suitable as the executable part in CreateProcessA's cmdline. | 
|  | // | 
|  | // The null-terminated executable path for CreateProcessA has to fit into | 
|  | // MAX_PATH, therefore the limit for the executable's path is MAX_PATH - 1 | 
|  | // (not including null terminator). This method attempts to convert the input | 
|  | // `path` to a short format to fit it into the MAX_PATH - 1 limit. | 
|  | // | 
|  | // `path` must be either an absolute, normalized, Windows-style path with drive | 
|  | // letter (e.g. "c:\foo\bar.exe", but no "\foo\bar.exe"), or must be just a file | 
|  | // name (e.g. "cmd.exe") that's shorter than MAX_PATH (without null-terminator). | 
|  | // In both cases, `path` must be unquoted. | 
|  | // | 
|  | // If this function succeeds, it returns an empty string (indicating no error), | 
|  | // and sets `result` to the resulting path, which is always quoted, and is | 
|  | // always at most MAX_PATH + 1 long (MAX_PATH - 1 without null terminator, plus | 
|  | // two quotes). If there's any error, this function returns the error message. | 
|  | // | 
|  | // If `path` is at most MAX_PATH - 1 long (not including null terminator), the | 
|  | // result will be that (plus quotes). | 
|  | // Otherwise this method attempts to compute an 8dot3 style short name for | 
|  | // `path`, and if that succeeds and the result is at most MAX_PATH - 1 long (not | 
|  | // including null terminator), then that will be the result (plus quotes). | 
|  | // Otherwise this function fails and returns an error message. | 
|  | wstring AsExecutablePathForCreateProcess(wstring path, wstring* result); | 
|  |  | 
|  | }  // namespace windows | 
|  | }  // namespace bazel | 
|  |  | 
|  | #endif  // BAZEL_SRC_MAIN_NATIVE_WINDOWS_UTIL_H__ |