Bitcoin Core  27.99.0
P2P Digital Currency
secure.h
Go to the documentation of this file.
1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2021 The Bitcoin Core developers
3 // Distributed under the MIT software license, see the accompanying
4 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
5 
6 #ifndef BITCOIN_SUPPORT_ALLOCATORS_SECURE_H
7 #define BITCOIN_SUPPORT_ALLOCATORS_SECURE_H
8 
9 #include <support/lockedpool.h>
10 #include <support/cleanse.h>
11 
12 #include <memory>
13 #include <string>
14 
15 //
16 // Allocator that locks its contents from being paged
17 // out of memory and clears its contents before deletion.
18 //
19 template <typename T>
21  using value_type = T;
22 
23  secure_allocator() = default;
24  template <typename U>
26 
27  T* allocate(std::size_t n)
28  {
29  T* allocation = static_cast<T*>(LockedPoolManager::Instance().alloc(sizeof(T) * n));
30  if (!allocation) {
31  throw std::bad_alloc();
32  }
33  return allocation;
34  }
35 
36  void deallocate(T* p, std::size_t n)
37  {
38  if (p != nullptr) {
39  memory_cleanse(p, sizeof(T) * n);
40  }
42  }
43 
44  template <typename U>
45  friend bool operator==(const secure_allocator&, const secure_allocator<U>&) noexcept
46  {
47  return true;
48  }
49  template <typename U>
50  friend bool operator!=(const secure_allocator&, const secure_allocator<U>&) noexcept
51  {
52  return false;
53  }
54 };
55 
56 // This is exactly like std::string, but with a custom allocator.
57 // TODO: Consider finding a way to make incoming RPC request.params[i] mlock()ed as well
58 typedef std::basic_string<char, std::char_traits<char>, secure_allocator<char> > SecureString;
59 
60 template<typename T>
62  void operator()(T* t) noexcept {
64  }
65 };
66 
67 template<typename T>
68 using secure_unique_ptr = std::unique_ptr<T, SecureUniqueDeleter<T>>;
69 
70 template<typename T, typename... Args>
72 {
73  T* p = secure_allocator<T>().allocate(1);
74 
75  // initialize in place, and return as secure_unique_ptr
76  try {
77  return secure_unique_ptr<T>(new (p) T(std::forward(as)...));
78  } catch (...) {
80  throw;
81  }
82 }
83 
84 #endif // BITCOIN_SUPPORT_ALLOCATORS_SECURE_H
void free(void *ptr)
Free a previously allocated chunk of memory.
Definition: lockedpool.cpp:309
void * alloc(size_t size)
Allocate size bytes from this arena.
Definition: lockedpool.cpp:287
static LockedPoolManager & Instance()
Return the current instance, or create it once.
Definition: lockedpool.h:222
void memory_cleanse(void *ptr, size_t len)
Secure overwrite a buffer (possibly containing secret data) with zero-bytes.
Definition: cleanse.cpp:14
#define T(expected, seed, data)
std::basic_string< char, std::char_traits< char >, secure_allocator< char > > SecureString
Definition: secure.h:58
secure_unique_ptr< T > make_secure_unique(Args &&... as)
Definition: secure.h:71
std::unique_ptr< T, SecureUniqueDeleter< T > > secure_unique_ptr
Definition: secure.h:68
void operator()(T *t) noexcept
Definition: secure.h:62
friend bool operator==(const secure_allocator &, const secure_allocator< U > &) noexcept
Definition: secure.h:45
secure_allocator(const secure_allocator< U > &) noexcept
Definition: secure.h:25
friend bool operator!=(const secure_allocator &, const secure_allocator< U > &) noexcept
Definition: secure.h:50
secure_allocator()=default
T * allocate(std::size_t n)
Definition: secure.h:27
void deallocate(T *p, std::size_t n)
Definition: secure.h:36