Reid Kleckner | d59e2fa | 2014-02-12 21:26:20 +0000 | [diff] [blame] | 1 | //===- WindowsSupport.h - Common Windows Include File -----------*- C++ -*-===// |
Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 2 | // |
Reid Spencer | 0f0c5cf | 2004-09-15 05:48:11 +0000 | [diff] [blame] | 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
Chris Lattner | f3ebc3f | 2007-12-29 20:36:04 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 7 | // |
Reid Spencer | 0f0c5cf | 2004-09-15 05:48:11 +0000 | [diff] [blame] | 8 | //===----------------------------------------------------------------------===// |
| 9 | // |
Reid Kleckner | d59e2fa | 2014-02-12 21:26:20 +0000 | [diff] [blame] | 10 | // This file defines things specific to Windows implementations. In addition to |
| 11 | // providing some helpers for working with win32 APIs, this header wraps |
| 12 | // <windows.h> with some portability macros. Always include WindowsSupport.h |
| 13 | // instead of including <windows.h> directly. |
Reid Spencer | 0f0c5cf | 2004-09-15 05:48:11 +0000 | [diff] [blame] | 14 | // |
| 15 | //===----------------------------------------------------------------------===// |
| 16 | |
| 17 | //===----------------------------------------------------------------------===// |
Jeff Cohen | 3800a28 | 2005-07-13 02:15:18 +0000 | [diff] [blame] | 18 | //=== WARNING: Implementation here must contain only generic Win32 code that |
| 19 | //=== is guaranteed to work on *all* Win32 variants. |
Reid Spencer | 0f0c5cf | 2004-09-15 05:48:11 +0000 | [diff] [blame] | 20 | //===----------------------------------------------------------------------===// |
| 21 | |
Yaron Keren | 3f02c14 | 2015-01-21 16:20:38 +0000 | [diff] [blame] | 22 | #ifndef LLVM_SUPPORT_WINDOWSSUPPORT_H |
| 23 | #define LLVM_SUPPORT_WINDOWSSUPPORT_H |
| 24 | |
NAKAMURA Takumi | aeb4c0d | 2011-02-09 04:18:30 +0000 | [diff] [blame] | 25 | // mingw-w64 tends to define it as 0x0502 in its headers. |
| 26 | #undef _WIN32_WINNT |
Aaron Ballman | 42f6622 | 2014-02-03 17:20:26 +0000 | [diff] [blame] | 27 | #undef _WIN32_IE |
NAKAMURA Takumi | aeb4c0d | 2011-02-09 04:18:30 +0000 | [diff] [blame] | 28 | |
Pawel Bylica | 6b129bd | 2015-10-15 14:50:31 +0000 | [diff] [blame] | 29 | // Require at least Windows 7 API. |
| 30 | #define _WIN32_WINNT 0x0601 |
| 31 | #define _WIN32_IE 0x0800 // MinGW at it again. FIXME: verify if still needed. |
Michael J. Spencer | bb6e51c | 2010-11-09 15:11:07 +0000 | [diff] [blame] | 32 | #define WIN32_LEAN_AND_MEAN |
Yunzhong Gao | d7009f3 | 2016-01-06 01:36:45 +0000 | [diff] [blame] | 33 | #ifndef NOMINMAX |
Yunzhong Gao | fb2a9c4 | 2016-01-06 00:50:06 +0000 | [diff] [blame] | 34 | #define NOMINMAX |
Yunzhong Gao | d7009f3 | 2016-01-06 01:36:45 +0000 | [diff] [blame] | 35 | #endif |
Jeff Cohen | 3800a28 | 2005-07-13 02:15:18 +0000 | [diff] [blame] | 36 | |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 37 | #include "llvm/ADT/SmallVector.h" |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 38 | #include "llvm/ADT/StringExtras.h" |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 39 | #include "llvm/ADT/StringRef.h" |
Paul Robinson | c38deee | 2014-11-24 18:05:29 +0000 | [diff] [blame] | 40 | #include "llvm/ADT/Twine.h" |
Michael J. Spencer | 4b263dd | 2010-11-09 15:10:29 +0000 | [diff] [blame] | 41 | #include "llvm/Config/config.h" // Get build system configuration settings |
Timur Iskhodzhanov | 0588513 | 2013-05-15 09:00:30 +0000 | [diff] [blame] | 42 | #include "llvm/Support/Compiler.h" |
Rafael Espindola | a6e9c3e | 2014-06-12 17:38:55 +0000 | [diff] [blame] | 43 | #include <system_error> |
NAKAMURA Takumi | 5a3ff5b | 2011-02-04 12:53:04 +0000 | [diff] [blame] | 44 | #include <windows.h> |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 45 | #include <wincrypt.h> |
Reid Spencer | 0f0c5cf | 2004-09-15 05:48:11 +0000 | [diff] [blame] | 46 | #include <cassert> |
| 47 | #include <string> |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 48 | #include <vector> |
Reid Spencer | 0f0c5cf | 2004-09-15 05:48:11 +0000 | [diff] [blame] | 49 | |
Reid Kleckner | ffbe12f | 2016-01-11 20:35:45 +0000 | [diff] [blame] | 50 | /// Determines if the program is running on Windows 8 or newer. This |
| 51 | /// reimplements the helpers in the Windows 8.1 SDK, which are intended to |
| 52 | /// supercede raw calls to GetVersionEx, because old Windows SDKs, Cygwin, and |
| 53 | /// MinGW don't have VersionSupport.h yet. |
Yunzhong Gao | fb2a9c4 | 2016-01-06 00:50:06 +0000 | [diff] [blame] | 54 | inline bool IsWindows8OrGreater() { |
| 55 | OSVERSIONINFO osvi = {}; |
| 56 | osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); |
| 57 | if (!::GetVersionEx(&osvi)) |
| 58 | return false; |
| 59 | return (osvi.dwMajorVersion > 6 || |
| 60 | (osvi.dwMajorVersion == 6 && osvi.dwMinorVersion >= 2)); |
| 61 | } |
Yunzhong Gao | fb2a9c4 | 2016-01-06 00:50:06 +0000 | [diff] [blame] | 62 | |
Reid Spencer | 879ed5a | 2006-08-23 07:30:48 +0000 | [diff] [blame] | 63 | inline bool MakeErrMsg(std::string* ErrMsg, const std::string& prefix) { |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 64 | if (!ErrMsg) |
Reid Spencer | 879ed5a | 2006-08-23 07:30:48 +0000 | [diff] [blame] | 65 | return true; |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 66 | char *buffer = NULL; |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 67 | DWORD LastError = GetLastError(); |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 68 | DWORD R = FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 69 | FORMAT_MESSAGE_FROM_SYSTEM | |
| 70 | FORMAT_MESSAGE_MAX_WIDTH_MASK, |
| 71 | NULL, LastError, 0, (LPSTR)&buffer, 1, NULL); |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 72 | if (R) |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 73 | *ErrMsg = prefix + ": " + buffer; |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 74 | else |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 75 | *ErrMsg = prefix + ": Unknown error"; |
| 76 | *ErrMsg += " (0x" + llvm::utohexstr(LastError) + ")"; |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 77 | |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 78 | LocalFree(buffer); |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 79 | return R != 0; |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 80 | } |
Jeff Cohen | a531d04 | 2007-03-05 05:22:08 +0000 | [diff] [blame] | 81 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 82 | template <typename HandleTraits> |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 83 | class ScopedHandle { |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 84 | typedef typename HandleTraits::handle_type handle_type; |
| 85 | handle_type Handle; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 86 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 87 | ScopedHandle(const ScopedHandle &other); // = delete; |
| 88 | void operator=(const ScopedHandle &other); // = delete; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 89 | public: |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 90 | ScopedHandle() |
| 91 | : Handle(HandleTraits::GetInvalid()) {} |
| 92 | |
| 93 | explicit ScopedHandle(handle_type h) |
| 94 | : Handle(h) {} |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 95 | |
| 96 | ~ScopedHandle() { |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 97 | if (HandleTraits::IsValid(Handle)) |
| 98 | HandleTraits::Close(Handle); |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 99 | } |
| 100 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 101 | handle_type take() { |
| 102 | handle_type t = Handle; |
| 103 | Handle = HandleTraits::GetInvalid(); |
| 104 | return t; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 105 | } |
| 106 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 107 | ScopedHandle &operator=(handle_type h) { |
| 108 | if (HandleTraits::IsValid(Handle)) |
| 109 | HandleTraits::Close(Handle); |
| 110 | Handle = h; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 111 | return *this; |
| 112 | } |
| 113 | |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 114 | // True if Handle is valid. |
Aaron Ballman | b46962f | 2015-02-15 22:00:20 +0000 | [diff] [blame] | 115 | explicit operator bool() const { |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 116 | return HandleTraits::IsValid(Handle) ? true : false; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 117 | } |
| 118 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 119 | operator handle_type() const { |
| 120 | return Handle; |
Michael J. Spencer | 7ecd94c | 2010-12-06 04:28:42 +0000 | [diff] [blame] | 121 | } |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 122 | }; |
Michael J. Spencer | 7ecd94c | 2010-12-06 04:28:42 +0000 | [diff] [blame] | 123 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 124 | struct CommonHandleTraits { |
| 125 | typedef HANDLE handle_type; |
| 126 | |
| 127 | static handle_type GetInvalid() { |
| 128 | return INVALID_HANDLE_VALUE; |
| 129 | } |
| 130 | |
| 131 | static void Close(handle_type h) { |
| 132 | ::CloseHandle(h); |
| 133 | } |
| 134 | |
| 135 | static bool IsValid(handle_type h) { |
| 136 | return h != GetInvalid(); |
| 137 | } |
| 138 | }; |
| 139 | |
| 140 | struct JobHandleTraits : CommonHandleTraits { |
| 141 | static handle_type GetInvalid() { |
| 142 | return NULL; |
| 143 | } |
| 144 | }; |
| 145 | |
| 146 | struct CryptContextTraits : CommonHandleTraits { |
| 147 | typedef HCRYPTPROV handle_type; |
| 148 | |
| 149 | static handle_type GetInvalid() { |
| 150 | return 0; |
| 151 | } |
| 152 | |
| 153 | static void Close(handle_type h) { |
| 154 | ::CryptReleaseContext(h, 0); |
| 155 | } |
| 156 | |
| 157 | static bool IsValid(handle_type h) { |
| 158 | return h != GetInvalid(); |
| 159 | } |
| 160 | }; |
| 161 | |
| 162 | struct FindHandleTraits : CommonHandleTraits { |
| 163 | static void Close(handle_type h) { |
| 164 | ::FindClose(h); |
| 165 | } |
| 166 | }; |
| 167 | |
| 168 | struct FileHandleTraits : CommonHandleTraits {}; |
| 169 | |
| 170 | typedef ScopedHandle<CommonHandleTraits> ScopedCommonHandle; |
| 171 | typedef ScopedHandle<FileHandleTraits> ScopedFileHandle; |
| 172 | typedef ScopedHandle<CryptContextTraits> ScopedCryptContext; |
| 173 | typedef ScopedHandle<FindHandleTraits> ScopedFindHandle; |
| 174 | typedef ScopedHandle<JobHandleTraits> ScopedJobHandle; |
Michael J. Spencer | 751e9aa | 2010-12-09 17:37:18 +0000 | [diff] [blame] | 175 | |
| 176 | namespace llvm { |
| 177 | template <class T> |
| 178 | class SmallVectorImpl; |
| 179 | |
| 180 | template <class T> |
| 181 | typename SmallVectorImpl<T>::const_pointer |
| 182 | c_str(SmallVectorImpl<T> &str) { |
| 183 | str.push_back(0); |
| 184 | str.pop_back(); |
| 185 | return str.data(); |
| 186 | } |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 187 | |
| 188 | namespace sys { |
Paul Robinson | c38deee | 2014-11-24 18:05:29 +0000 | [diff] [blame] | 189 | namespace path { |
| 190 | std::error_code widenPath(const Twine &Path8, |
| 191 | SmallVectorImpl<wchar_t> &Path16); |
| 192 | } // end namespace path |
| 193 | |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 194 | namespace windows { |
Rafael Espindola | 885719f | 2014-06-12 17:49:35 +0000 | [diff] [blame] | 195 | std::error_code UTF8ToUTF16(StringRef utf8, SmallVectorImpl<wchar_t> &utf16); |
| 196 | std::error_code UTF16ToUTF8(const wchar_t *utf16, size_t utf16_len, |
| 197 | SmallVectorImpl<char> &utf8); |
Rafael Espindola | 9c35966 | 2014-09-03 20:02:00 +0000 | [diff] [blame] | 198 | /// Convert from UTF16 to the current code page used in the system |
| 199 | std::error_code UTF16ToCurCP(const wchar_t *utf16, size_t utf16_len, |
| 200 | SmallVectorImpl<char> &utf8); |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 201 | } // end namespace windows |
| 202 | } // end namespace sys |
Michael J. Spencer | 751e9aa | 2010-12-09 17:37:18 +0000 | [diff] [blame] | 203 | } // end namespace llvm. |
Yaron Keren | 3f02c14 | 2015-01-21 16:20:38 +0000 | [diff] [blame] | 204 | |
| 205 | #endif |