Home | History | Annotate | Download | only in Windows
      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 // mingw-w64 tends to define it as 0x0502 in its headers.
     23 #undef _WIN32_WINNT
     24 #undef _WIN32_IE
     25 
     26 // Require at least Windows XP(5.1) API.
     27 #define _WIN32_WINNT 0x0501
     28 #define _WIN32_IE    0x0600 // MinGW at it again.
     29 #define WIN32_LEAN_AND_MEAN
     30 
     31 #include "llvm/ADT/SmallVector.h"
     32 #include "llvm/ADT/StringRef.h"
     33 #include "llvm/Config/config.h" // Get build system configuration settings
     34 #include "llvm/Support/Compiler.h"
     35 #include <system_error>
     36 #include <windows.h>
     37 #include <wincrypt.h>
     38 #include <cassert>
     39 #include <string>
     40 #include <vector>
     41 
     42 inline bool MakeErrMsg(std::string* ErrMsg, const std::string& prefix) {
     43   if (!ErrMsg)
     44     return true;
     45   char *buffer = NULL;
     46   DWORD R = FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER |
     47                           FORMAT_MESSAGE_FROM_SYSTEM,
     48                           NULL, GetLastError(), 0, (LPSTR)&buffer, 1, NULL);
     49   if (R)
     50     *ErrMsg = prefix + buffer;
     51   else
     52     *ErrMsg = prefix + "Unknown error";
     53 
     54   LocalFree(buffer);
     55   return R != 0;
     56 }
     57 
     58 template <typename HandleTraits>
     59 class ScopedHandle {
     60   typedef typename HandleTraits::handle_type handle_type;
     61   handle_type Handle;
     62 
     63   ScopedHandle(const ScopedHandle &other); // = delete;
     64   void operator=(const ScopedHandle &other); // = delete;
     65 public:
     66   ScopedHandle()
     67     : Handle(HandleTraits::GetInvalid()) {}
     68 
     69   explicit ScopedHandle(handle_type h)
     70     : Handle(h) {}
     71 
     72   ~ScopedHandle() {
     73     if (HandleTraits::IsValid(Handle))
     74       HandleTraits::Close(Handle);
     75   }
     76 
     77   handle_type take() {
     78     handle_type t = Handle;
     79     Handle = HandleTraits::GetInvalid();
     80     return t;
     81   }
     82 
     83   ScopedHandle &operator=(handle_type h) {
     84     if (HandleTraits::IsValid(Handle))
     85       HandleTraits::Close(Handle);
     86     Handle = h;
     87     return *this;
     88   }
     89 
     90   // True if Handle is valid.
     91   LLVM_EXPLICIT operator bool() const {
     92     return HandleTraits::IsValid(Handle) ? true : false;
     93   }
     94 
     95   operator handle_type() const {
     96     return Handle;
     97   }
     98 };
     99 
    100 struct CommonHandleTraits {
    101   typedef HANDLE handle_type;
    102 
    103   static handle_type GetInvalid() {
    104     return INVALID_HANDLE_VALUE;
    105   }
    106 
    107   static void Close(handle_type h) {
    108     ::CloseHandle(h);
    109   }
    110 
    111   static bool IsValid(handle_type h) {
    112     return h != GetInvalid();
    113   }
    114 };
    115 
    116 struct JobHandleTraits : CommonHandleTraits {
    117   static handle_type GetInvalid() {
    118     return NULL;
    119   }
    120 };
    121 
    122 struct CryptContextTraits : CommonHandleTraits {
    123   typedef HCRYPTPROV handle_type;
    124 
    125   static handle_type GetInvalid() {
    126     return 0;
    127   }
    128 
    129   static void Close(handle_type h) {
    130     ::CryptReleaseContext(h, 0);
    131   }
    132 
    133   static bool IsValid(handle_type h) {
    134     return h != GetInvalid();
    135   }
    136 };
    137 
    138 struct FindHandleTraits : CommonHandleTraits {
    139   static void Close(handle_type h) {
    140     ::FindClose(h);
    141   }
    142 };
    143 
    144 struct FileHandleTraits : CommonHandleTraits {};
    145 
    146 typedef ScopedHandle<CommonHandleTraits> ScopedCommonHandle;
    147 typedef ScopedHandle<FileHandleTraits>   ScopedFileHandle;
    148 typedef ScopedHandle<CryptContextTraits> ScopedCryptContext;
    149 typedef ScopedHandle<FindHandleTraits>   ScopedFindHandle;
    150 typedef ScopedHandle<JobHandleTraits>    ScopedJobHandle;
    151 
    152 namespace llvm {
    153 template <class T>
    154 class SmallVectorImpl;
    155 
    156 template <class T>
    157 typename SmallVectorImpl<T>::const_pointer
    158 c_str(SmallVectorImpl<T> &str) {
    159   str.push_back(0);
    160   str.pop_back();
    161   return str.data();
    162 }
    163 
    164 namespace sys {
    165 namespace windows {
    166 std::error_code UTF8ToUTF16(StringRef utf8, SmallVectorImpl<wchar_t> &utf16);
    167 std::error_code UTF16ToUTF8(const wchar_t *utf16, size_t utf16_len,
    168                             SmallVectorImpl<char> &utf8);
    169 } // end namespace windows
    170 } // end namespace sys
    171 } // end namespace llvm.
    172