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 |
Reid Kleckner | 5fb7a58 | 2016-01-11 23:33:03 +0000 | [diff] [blame^] | 51 | /// reimplements one of the helpers in the Windows 8.1 SDK, which are intended |
| 52 | /// to supercede raw calls to GetVersionEx. Old SDKs, Cygwin, and MinGW don't |
| 53 | /// yet have VersionHelpers.h, so we have our own helper. |
| 54 | inline bool RunningWindows8OrGreater() { |
| 55 | // Windows 8 is version 6.2, service pack 0. |
| 56 | OSVERSIONINFOEXW osvi = {}; |
Yunzhong Gao | fb2a9c4 | 2016-01-06 00:50:06 +0000 | [diff] [blame] | 57 | osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); |
Reid Kleckner | 5fb7a58 | 2016-01-11 23:33:03 +0000 | [diff] [blame^] | 58 | osvi.dwMajorVersion = 6; |
| 59 | osvi.dwMinorVersion = 2; |
| 60 | osvi.wServicePackMajor = 0; |
| 61 | |
| 62 | DWORDLONG Mask = 0; |
| 63 | Mask = VerSetConditionMask(Mask, VER_MAJORVERSION, VER_GREATER_EQUAL); |
| 64 | Mask = VerSetConditionMask(Mask, VER_MINORVERSION, VER_GREATER_EQUAL); |
| 65 | Mask = VerSetConditionMask(Mask, VER_SERVICEPACKMAJOR, VER_GREATER_EQUAL); |
| 66 | |
| 67 | return VerifyVersionInfoW(&osvi, VER_MAJORVERSION | VER_MINORVERSION | |
| 68 | VER_SERVICEPACKMAJOR, |
| 69 | Mask) != FALSE; |
Yunzhong Gao | fb2a9c4 | 2016-01-06 00:50:06 +0000 | [diff] [blame] | 70 | } |
Yunzhong Gao | fb2a9c4 | 2016-01-06 00:50:06 +0000 | [diff] [blame] | 71 | |
Reid Spencer | 879ed5a | 2006-08-23 07:30:48 +0000 | [diff] [blame] | 72 | inline bool MakeErrMsg(std::string* ErrMsg, const std::string& prefix) { |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 73 | if (!ErrMsg) |
Reid Spencer | 879ed5a | 2006-08-23 07:30:48 +0000 | [diff] [blame] | 74 | return true; |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 75 | char *buffer = NULL; |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 76 | DWORD LastError = GetLastError(); |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 77 | DWORD R = FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 78 | FORMAT_MESSAGE_FROM_SYSTEM | |
| 79 | FORMAT_MESSAGE_MAX_WIDTH_MASK, |
| 80 | NULL, LastError, 0, (LPSTR)&buffer, 1, NULL); |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 81 | if (R) |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 82 | *ErrMsg = prefix + ": " + buffer; |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 83 | else |
Paul Robinson | af19bc3 | 2015-11-23 17:34:20 +0000 | [diff] [blame] | 84 | *ErrMsg = prefix + ": Unknown error"; |
| 85 | *ErrMsg += " (0x" + llvm::utohexstr(LastError) + ")"; |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 86 | |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 87 | LocalFree(buffer); |
Aaron Ballman | 69c55ed | 2013-11-18 17:43:22 +0000 | [diff] [blame] | 88 | return R != 0; |
Reid Spencer | 42bcf6e | 2006-08-21 06:02:44 +0000 | [diff] [blame] | 89 | } |
Jeff Cohen | a531d04 | 2007-03-05 05:22:08 +0000 | [diff] [blame] | 90 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 91 | template <typename HandleTraits> |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 92 | class ScopedHandle { |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 93 | typedef typename HandleTraits::handle_type handle_type; |
| 94 | handle_type Handle; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 95 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 96 | ScopedHandle(const ScopedHandle &other); // = delete; |
| 97 | void operator=(const ScopedHandle &other); // = delete; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 98 | public: |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 99 | ScopedHandle() |
| 100 | : Handle(HandleTraits::GetInvalid()) {} |
| 101 | |
| 102 | explicit ScopedHandle(handle_type h) |
| 103 | : Handle(h) {} |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 104 | |
| 105 | ~ScopedHandle() { |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 106 | if (HandleTraits::IsValid(Handle)) |
| 107 | HandleTraits::Close(Handle); |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 108 | } |
| 109 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 110 | handle_type take() { |
| 111 | handle_type t = Handle; |
| 112 | Handle = HandleTraits::GetInvalid(); |
| 113 | return t; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 114 | } |
| 115 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 116 | ScopedHandle &operator=(handle_type h) { |
| 117 | if (HandleTraits::IsValid(Handle)) |
| 118 | HandleTraits::Close(Handle); |
| 119 | Handle = h; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 120 | return *this; |
| 121 | } |
| 122 | |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 123 | // True if Handle is valid. |
Aaron Ballman | b46962f | 2015-02-15 22:00:20 +0000 | [diff] [blame] | 124 | explicit operator bool() const { |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 125 | return HandleTraits::IsValid(Handle) ? true : false; |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 126 | } |
| 127 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 128 | operator handle_type() const { |
| 129 | return Handle; |
Michael J. Spencer | 7ecd94c | 2010-12-06 04:28:42 +0000 | [diff] [blame] | 130 | } |
Michael J. Spencer | 39c4621 | 2010-12-06 04:28:13 +0000 | [diff] [blame] | 131 | }; |
Michael J. Spencer | 7ecd94c | 2010-12-06 04:28:42 +0000 | [diff] [blame] | 132 | |
Michael J. Spencer | 513f1b6 | 2011-12-12 06:03:33 +0000 | [diff] [blame] | 133 | struct CommonHandleTraits { |
| 134 | typedef HANDLE handle_type; |
| 135 | |
| 136 | static handle_type GetInvalid() { |
| 137 | return INVALID_HANDLE_VALUE; |
| 138 | } |
| 139 | |
| 140 | static void Close(handle_type h) { |
| 141 | ::CloseHandle(h); |
| 142 | } |
| 143 | |
| 144 | static bool IsValid(handle_type h) { |
| 145 | return h != GetInvalid(); |
| 146 | } |
| 147 | }; |
| 148 | |
| 149 | struct JobHandleTraits : CommonHandleTraits { |
| 150 | static handle_type GetInvalid() { |
| 151 | return NULL; |
| 152 | } |
| 153 | }; |
| 154 | |
| 155 | struct CryptContextTraits : CommonHandleTraits { |
| 156 | typedef HCRYPTPROV handle_type; |
| 157 | |
| 158 | static handle_type GetInvalid() { |
| 159 | return 0; |
| 160 | } |
| 161 | |
| 162 | static void Close(handle_type h) { |
| 163 | ::CryptReleaseContext(h, 0); |
| 164 | } |
| 165 | |
| 166 | static bool IsValid(handle_type h) { |
| 167 | return h != GetInvalid(); |
| 168 | } |
| 169 | }; |
| 170 | |
| 171 | struct FindHandleTraits : CommonHandleTraits { |
| 172 | static void Close(handle_type h) { |
| 173 | ::FindClose(h); |
| 174 | } |
| 175 | }; |
| 176 | |
| 177 | struct FileHandleTraits : CommonHandleTraits {}; |
| 178 | |
| 179 | typedef ScopedHandle<CommonHandleTraits> ScopedCommonHandle; |
| 180 | typedef ScopedHandle<FileHandleTraits> ScopedFileHandle; |
| 181 | typedef ScopedHandle<CryptContextTraits> ScopedCryptContext; |
| 182 | typedef ScopedHandle<FindHandleTraits> ScopedFindHandle; |
| 183 | typedef ScopedHandle<JobHandleTraits> ScopedJobHandle; |
Michael J. Spencer | 751e9aa | 2010-12-09 17:37:18 +0000 | [diff] [blame] | 184 | |
| 185 | namespace llvm { |
| 186 | template <class T> |
| 187 | class SmallVectorImpl; |
| 188 | |
| 189 | template <class T> |
| 190 | typename SmallVectorImpl<T>::const_pointer |
| 191 | c_str(SmallVectorImpl<T> &str) { |
| 192 | str.push_back(0); |
| 193 | str.pop_back(); |
| 194 | return str.data(); |
| 195 | } |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 196 | |
| 197 | namespace sys { |
Paul Robinson | c38deee | 2014-11-24 18:05:29 +0000 | [diff] [blame] | 198 | namespace path { |
| 199 | std::error_code widenPath(const Twine &Path8, |
| 200 | SmallVectorImpl<wchar_t> &Path16); |
| 201 | } // end namespace path |
| 202 | |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 203 | namespace windows { |
Rafael Espindola | 885719f | 2014-06-12 17:49:35 +0000 | [diff] [blame] | 204 | std::error_code UTF8ToUTF16(StringRef utf8, SmallVectorImpl<wchar_t> &utf16); |
| 205 | std::error_code UTF16ToUTF8(const wchar_t *utf16, size_t utf16_len, |
| 206 | SmallVectorImpl<char> &utf8); |
Rafael Espindola | 9c35966 | 2014-09-03 20:02:00 +0000 | [diff] [blame] | 207 | /// Convert from UTF16 to the current code page used in the system |
| 208 | std::error_code UTF16ToCurCP(const wchar_t *utf16, size_t utf16_len, |
| 209 | SmallVectorImpl<char> &utf8); |
Rui Ueyama | 471d0c5 | 2013-09-10 19:45:51 +0000 | [diff] [blame] | 210 | } // end namespace windows |
| 211 | } // end namespace sys |
Michael J. Spencer | 751e9aa | 2010-12-09 17:37:18 +0000 | [diff] [blame] | 212 | } // end namespace llvm. |
Yaron Keren | 3f02c14 | 2015-01-21 16:20:38 +0000 | [diff] [blame] | 213 | |
| 214 | #endif |