Support/Windows: Add ScopedHandle and move some clients over to it.
authorMichael J. Spencer <bigcheesegs@gmail.com>
Mon, 6 Dec 2010 04:28:13 +0000 (04:28 +0000)
committerMichael J. Spencer <bigcheesegs@gmail.com>
Mon, 6 Dec 2010 04:28:13 +0000 (04:28 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@120987 91177308-0d34-0410-b5e6-96231b3b80d8

lib/Support/Windows/PathV2.inc
lib/Support/Windows/Windows.h

index f6f53714b066c3a08a948eaca54ae19b29a5331d..6bd541e49cc885203702f9814f240bde04dda1aa 100644 (file)
@@ -113,15 +113,14 @@ namespace {
     return success;
   }
 
-  struct AutoCryptoProvider {
-    HCRYPTPROV CryptoProvider;
-
-    ~AutoCryptoProvider() {
-      ::CryptReleaseContext(CryptoProvider, 0);
-    }
+  // Forwarder for ScopedHandle.
+  BOOL WINAPI CryptReleaseContext(HCRYPTPROV Provider) {
+    return ::CryptReleaseContext(Provider, 0);
+  }
 
-    operator HCRYPTPROV() const {return CryptoProvider;}
-  };
+  typedef ScopedHandle<HCRYPTPROV, HCRYPTPROV(INVALID_HANDLE_VALUE),
+                       BOOL (WINAPI*)(HCRYPTPROV), CryptReleaseContext>
+    ScopedCryptContext;
 }
 
 namespace llvm {
@@ -503,13 +502,14 @@ error_code unique_file(const Twine &model, int &result_fd,
   SmallVector<wchar_t, 128> random_path_utf16;
 
   // Get a Crypto Provider for CryptGenRandom.
-  AutoCryptoProvider CryptoProvider;
-  if (!::CryptAcquireContextW(&CryptoProvider.CryptoProvider,
+  HCRYPTPROV HCPC;
+  if (!::CryptAcquireContextW(&HCPC,
                               NULL,
                               NULL,
                               PROV_RSA_FULL,
                               0))
     return windows_error(::GetLastError());
+  ScopedCryptContext CryptoProvider(HCPC);
 
 retry_random_path:
   random_path_utf16.set_size(0);
index 00a7e75fc2a0c05a1c788b09b3e4f20f642fe046..0233a420512ec4d786bde9c93e31b35abf2b43f3 100644 (file)
@@ -58,3 +58,43 @@ public:
     return *this;
   }
 };
+
+template <class HandleType, HandleType InvalidHandle,
+          class DeleterType, DeleterType D>
+class ScopedHandle {
+  HandleType Handle;
+
+public:
+  ScopedHandle() : Handle(InvalidHandle) {}
+  ScopedHandle(HandleType handle) : Handle(handle) {}
+
+  ~ScopedHandle() {
+    if (Handle != InvalidHandle)
+      D(Handle);
+  }
+
+  HandleType take() {
+    HandleType temp = Handle;
+    Handle = InvalidHandle;
+    return temp;
+  }
+
+  operator HandleType() const { return Handle; }
+
+  ScopedHandle &operator=(HandleType handle) {
+    Handle = handle;
+    return *this;
+  }
+
+  typedef void (*unspecified_bool_type)();
+  static void unspecified_bool_true() {}
+
+  // True if Handle is valid.
+  operator unspecified_bool_type() const {
+    return Handle == InvalidHandle ? 0 : unspecified_bool_true;
+  }
+
+  typedef ScopedHandle<HANDLE, INVALID_HANDLE_VALUE,
+                       BOOL (WINAPI*)(HANDLE), ::FindClose>
+    ScopedFindHandle;
+};