• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===- WindowsSupport.h - Common Windows Include File -----------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
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.
14 //
15 //===----------------------------------------------------------------------===//
16 
17 //===----------------------------------------------------------------------===//
18 //=== WARNING: Implementation here must contain only generic Win32 code that
19 //===          is guaranteed to work on *all* Win32 variants.
20 //===----------------------------------------------------------------------===//
21 
22 #ifndef LLVM_SUPPORT_WINDOWSSUPPORT_H
23 #define LLVM_SUPPORT_WINDOWSSUPPORT_H
24 
25 // mingw-w64 tends to define it as 0x0502 in its headers.
26 #undef _WIN32_WINNT
27 #undef _WIN32_IE
28 
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.
32 #define WIN32_LEAN_AND_MEAN
33 #ifndef NOMINMAX
34 #define NOMINMAX
35 #endif
36 
37 #include "llvm/ADT/SmallVector.h"
38 #include "llvm/ADT/StringExtras.h"
39 #include "llvm/ADT/StringRef.h"
40 #include "llvm/ADT/Twine.h"
41 #include "llvm/Config/config.h" // Get build system configuration settings
42 #include "llvm/Support/Chrono.h"
43 #include "llvm/Support/Compiler.h"
44 #include <cassert>
45 #include <string>
46 #include <system_error>
47 #include <windows.h>
48 #include <wincrypt.h> // Must be included after windows.h
49 
50 /// Determines if the program is running on Windows 8 or newer. This
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.
RunningWindows8OrGreater()54 inline bool RunningWindows8OrGreater() {
55   // Windows 8 is version 6.2, service pack 0.
56   OSVERSIONINFOEXW osvi = {};
57   osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
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;
70 }
71 
MakeErrMsg(std::string * ErrMsg,const std::string & prefix)72 inline bool MakeErrMsg(std::string *ErrMsg, const std::string &prefix) {
73   if (!ErrMsg)
74     return true;
75   char *buffer = NULL;
76   DWORD LastError = GetLastError();
77   DWORD R = FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER |
78                                FORMAT_MESSAGE_FROM_SYSTEM |
79                                FORMAT_MESSAGE_MAX_WIDTH_MASK,
80                            NULL, LastError, 0, (LPSTR)&buffer, 1, NULL);
81   if (R)
82     *ErrMsg = prefix + ": " + buffer;
83   else
84     *ErrMsg = prefix + ": Unknown error";
85   *ErrMsg += " (0x" + llvm::utohexstr(LastError) + ")";
86 
87   LocalFree(buffer);
88   return R != 0;
89 }
90 
91 template <typename HandleTraits>
92 class ScopedHandle {
93   typedef typename HandleTraits::handle_type handle_type;
94   handle_type Handle;
95 
96   ScopedHandle(const ScopedHandle &other); // = delete;
97   void operator=(const ScopedHandle &other); // = delete;
98 public:
ScopedHandle()99   ScopedHandle()
100     : Handle(HandleTraits::GetInvalid()) {}
101 
ScopedHandle(handle_type h)102   explicit ScopedHandle(handle_type h)
103     : Handle(h) {}
104 
~ScopedHandle()105   ~ScopedHandle() {
106     if (HandleTraits::IsValid(Handle))
107       HandleTraits::Close(Handle);
108   }
109 
take()110   handle_type take() {
111     handle_type t = Handle;
112     Handle = HandleTraits::GetInvalid();
113     return t;
114   }
115 
116   ScopedHandle &operator=(handle_type h) {
117     if (HandleTraits::IsValid(Handle))
118       HandleTraits::Close(Handle);
119     Handle = h;
120     return *this;
121   }
122 
123   // True if Handle is valid.
124   explicit operator bool() const {
125     return HandleTraits::IsValid(Handle) ? true : false;
126   }
127 
handle_type()128   operator handle_type() const {
129     return Handle;
130   }
131 };
132 
133 struct CommonHandleTraits {
134   typedef HANDLE handle_type;
135 
GetInvalidCommonHandleTraits136   static handle_type GetInvalid() {
137     return INVALID_HANDLE_VALUE;
138   }
139 
CloseCommonHandleTraits140   static void Close(handle_type h) {
141     ::CloseHandle(h);
142   }
143 
IsValidCommonHandleTraits144   static bool IsValid(handle_type h) {
145     return h != GetInvalid();
146   }
147 };
148 
149 struct JobHandleTraits : CommonHandleTraits {
GetInvalidJobHandleTraits150   static handle_type GetInvalid() {
151     return NULL;
152   }
153 };
154 
155 struct CryptContextTraits : CommonHandleTraits {
156   typedef HCRYPTPROV handle_type;
157 
GetInvalidCryptContextTraits158   static handle_type GetInvalid() {
159     return 0;
160   }
161 
CloseCryptContextTraits162   static void Close(handle_type h) {
163     ::CryptReleaseContext(h, 0);
164   }
165 
IsValidCryptContextTraits166   static bool IsValid(handle_type h) {
167     return h != GetInvalid();
168   }
169 };
170 
171 struct RegTraits : CommonHandleTraits {
172   typedef HKEY handle_type;
173 
GetInvalidRegTraits174   static handle_type GetInvalid() {
175     return NULL;
176   }
177 
CloseRegTraits178   static void Close(handle_type h) {
179     ::RegCloseKey(h);
180   }
181 
IsValidRegTraits182   static bool IsValid(handle_type h) {
183     return h != GetInvalid();
184   }
185 };
186 
187 struct FindHandleTraits : CommonHandleTraits {
CloseFindHandleTraits188   static void Close(handle_type h) {
189     ::FindClose(h);
190   }
191 };
192 
193 struct FileHandleTraits : CommonHandleTraits {};
194 
195 typedef ScopedHandle<CommonHandleTraits> ScopedCommonHandle;
196 typedef ScopedHandle<FileHandleTraits>   ScopedFileHandle;
197 typedef ScopedHandle<CryptContextTraits> ScopedCryptContext;
198 typedef ScopedHandle<RegTraits>          ScopedRegHandle;
199 typedef ScopedHandle<FindHandleTraits>   ScopedFindHandle;
200 typedef ScopedHandle<JobHandleTraits>    ScopedJobHandle;
201 
202 namespace llvm {
203 template <class T>
204 class SmallVectorImpl;
205 
206 template <class T>
207 typename SmallVectorImpl<T>::const_pointer
c_str(SmallVectorImpl<T> & str)208 c_str(SmallVectorImpl<T> &str) {
209   str.push_back(0);
210   str.pop_back();
211   return str.data();
212 }
213 
214 namespace sys {
215 
toDuration(FILETIME Time)216 inline std::chrono::nanoseconds toDuration(FILETIME Time) {
217   ULARGE_INTEGER TimeInteger;
218   TimeInteger.LowPart = Time.dwLowDateTime;
219   TimeInteger.HighPart = Time.dwHighDateTime;
220 
221   // FILETIME's are # of 100 nanosecond ticks (1/10th of a microsecond)
222   return std::chrono::nanoseconds(100 * TimeInteger.QuadPart);
223 }
224 
toTimePoint(FILETIME Time)225 inline TimePoint<> toTimePoint(FILETIME Time) {
226   ULARGE_INTEGER TimeInteger;
227   TimeInteger.LowPart = Time.dwLowDateTime;
228   TimeInteger.HighPart = Time.dwHighDateTime;
229 
230   // Adjust for different epoch
231   TimeInteger.QuadPart -= 11644473600ll * 10000000;
232 
233   // FILETIME's are # of 100 nanosecond ticks (1/10th of a microsecond)
234   return TimePoint<>(std::chrono::nanoseconds(100 * TimeInteger.QuadPart));
235 }
236 
toFILETIME(TimePoint<> TP)237 inline FILETIME toFILETIME(TimePoint<> TP) {
238   ULARGE_INTEGER TimeInteger;
239   TimeInteger.QuadPart = TP.time_since_epoch().count() / 100;
240   TimeInteger.QuadPart += 11644473600ll * 10000000;
241 
242   FILETIME Time;
243   Time.dwLowDateTime = TimeInteger.LowPart;
244   Time.dwHighDateTime = TimeInteger.HighPart;
245   return Time;
246 }
247 
248 namespace path {
249 std::error_code widenPath(const Twine &Path8,
250                           SmallVectorImpl<wchar_t> &Path16);
251 } // end namespace path
252 
253 namespace windows {
254 std::error_code UTF8ToUTF16(StringRef utf8, SmallVectorImpl<wchar_t> &utf16);
255 std::error_code UTF16ToUTF8(const wchar_t *utf16, size_t utf16_len,
256                             SmallVectorImpl<char> &utf8);
257 /// Convert from UTF16 to the current code page used in the system
258 std::error_code UTF16ToCurCP(const wchar_t *utf16, size_t utf16_len,
259                              SmallVectorImpl<char> &utf8);
260 } // end namespace windows
261 } // end namespace sys
262 } // end namespace llvm.
263 
264 #endif
265