1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #ifndef CRYPTO_SCOPED_CAPI_TYPES_H_ 6 #define CRYPTO_SCOPED_CAPI_TYPES_H_ 7 8 #include <windows.h> 9 10 #include <algorithm> 11 12 #include "base/logging.h" 13 #include "base/macros.h" 14 #include "crypto/wincrypt_shim.h" 15 16 namespace crypto { 17 18 // Simple destructor for the Free family of CryptoAPI functions, such as 19 // CryptDestroyHash, which take only a single argument to release. 20 template <typename CAPIHandle, BOOL (WINAPI *Destroyer)(CAPIHandle)> 21 struct CAPIDestroyer { operatorCAPIDestroyer22 void operator()(CAPIHandle handle) const { 23 if (handle) { 24 BOOL ok = Destroyer(handle); 25 DCHECK(ok); 26 } 27 } 28 }; 29 30 // Destructor for the Close/Release family of CryptoAPI functions, which take 31 // a second DWORD parameter indicating flags to use when closing or releasing. 32 // This includes functions like CertCloseStore or CryptReleaseContext. 33 template <typename CAPIHandle, BOOL (WINAPI *Destroyer)(CAPIHandle, DWORD), 34 DWORD flags> 35 struct CAPIDestroyerWithFlags { operatorCAPIDestroyerWithFlags36 void operator()(CAPIHandle handle) const { 37 if (handle) { 38 BOOL ok = Destroyer(handle, flags); 39 DCHECK(ok); 40 } 41 } 42 }; 43 44 // scoped_ptr-like class for the CryptoAPI cryptography and certificate 45 // handles. Because these handles are defined as integer types, and not 46 // pointers, the existing scoped classes, such as scoped_ptr, are insufficient. 47 // The semantics are the same as scoped_ptr. 48 template <class CAPIHandle, typename FreeProc> 49 class ScopedCAPIHandle { 50 public: handle_(handle)51 explicit ScopedCAPIHandle(CAPIHandle handle = NULL) : handle_(handle) {} 52 ~ScopedCAPIHandle()53 ~ScopedCAPIHandle() { 54 reset(); 55 } 56 57 void reset(CAPIHandle handle = NULL) { 58 if (handle_ != handle) { 59 FreeProc free_proc; 60 free_proc(handle_); 61 handle_ = handle; 62 } 63 } 64 CAPIHandle()65 operator CAPIHandle() const { return handle_; } get()66 CAPIHandle get() const { return handle_; } 67 receive()68 CAPIHandle* receive() { 69 CHECK(handle_ == NULL); 70 return &handle_; 71 } 72 73 bool operator==(CAPIHandle handle) const { 74 return handle_ == handle; 75 } 76 77 bool operator!=(CAPIHandle handle) const { 78 return handle_ != handle; 79 } 80 swap(ScopedCAPIHandle & b)81 void swap(ScopedCAPIHandle& b) { 82 CAPIHandle tmp = b.handle_; 83 b.handle_ = handle_; 84 handle_ = tmp; 85 } 86 release()87 CAPIHandle release() { 88 CAPIHandle tmp = handle_; 89 handle_ = NULL; 90 return tmp; 91 } 92 93 private: 94 CAPIHandle handle_; 95 96 DISALLOW_COPY_AND_ASSIGN(ScopedCAPIHandle); 97 }; 98 99 template<class CH, typename FP> inline 100 bool operator==(CH h, const ScopedCAPIHandle<CH, FP>& b) { 101 return h == b.get(); 102 } 103 104 template<class CH, typename FP> inline 105 bool operator!=(CH h, const ScopedCAPIHandle<CH, FP>& b) { 106 return h != b.get(); 107 } 108 109 typedef ScopedCAPIHandle< 110 HCRYPTPROV, 111 CAPIDestroyerWithFlags<HCRYPTPROV, 112 CryptReleaseContext, 0> > ScopedHCRYPTPROV; 113 114 typedef ScopedCAPIHandle< 115 HCRYPTKEY, CAPIDestroyer<HCRYPTKEY, CryptDestroyKey> > ScopedHCRYPTKEY; 116 117 typedef ScopedCAPIHandle< 118 HCRYPTHASH, CAPIDestroyer<HCRYPTHASH, CryptDestroyHash> > ScopedHCRYPTHASH; 119 120 } // namespace crypto 121 122 #endif // CRYPTO_SCOPED_CAPI_TYPES_H_ 123