blob: a98e7f8efb9043129d912e89d57a306742c8af1a [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.
#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__