1 //===----------------------------------------------------------------------===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is dual licensed under the MIT and the University of Illinois Open 6 // Source Licenses. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 10 #ifndef TEST_ALLOCATOR_H 11 #define TEST_ALLOCATOR_H 12 13 #include <cstddef> 14 #include <type_traits> 15 #include <cstdlib> 16 #include <new> 17 #include <climits> 18 #include <cassert> 19 20 #include "test_macros.h" 21 22 class test_alloc_base 23 { 24 protected: 25 static int time_to_throw; 26 public: 27 static int throw_after; 28 static int count; 29 static int alloc_count; 30 }; 31 32 int test_alloc_base::count = 0; 33 int test_alloc_base::time_to_throw = 0; 34 int test_alloc_base::alloc_count = 0; 35 int test_alloc_base::throw_after = INT_MAX; 36 37 template <class T> 38 class test_allocator 39 : public test_alloc_base 40 { 41 int data_; 42 43 template <class U> friend class test_allocator; 44 public: 45 46 typedef unsigned size_type; 47 typedef int difference_type; 48 typedef T value_type; 49 typedef value_type* pointer; 50 typedef const value_type* const_pointer; 51 typedef typename std::add_lvalue_reference<value_type>::type reference; 52 typedef typename std::add_lvalue_reference<const value_type>::type const_reference; 53 54 template <class U> struct rebind {typedef test_allocator<U> other;}; 55 56 test_allocator() throw() : data_(0) {++count;} 57 explicit test_allocator(int i) throw() : data_(i) {++count;} 58 test_allocator(const test_allocator& a) throw() 59 : data_(a.data_) {++count;} 60 template <class U> test_allocator(const test_allocator<U>& a) throw() 61 : data_(a.data_) {++count;} 62 ~test_allocator() throw() {assert(data_ >= 0); --count; data_ = -1;} 63 pointer address(reference x) const {return &x;} 64 const_pointer address(const_reference x) const {return &x;} 65 pointer allocate(size_type n, const void* = 0) 66 { 67 assert(data_ >= 0); 68 if (time_to_throw >= throw_after) { 69 #ifndef _LIBCPP_NO_EXCEPTIONS 70 throw std::bad_alloc(); 71 #else 72 std::terminate(); 73 #endif 74 } 75 ++time_to_throw; 76 ++alloc_count; 77 return (pointer)::operator new(n * sizeof(T)); 78 } 79 void deallocate(pointer p, size_type n) 80 {assert(data_ >= 0); --alloc_count; ::operator delete((void*)p);} 81 size_type max_size() const throw() 82 {return UINT_MAX / sizeof(T);} 83 void construct(pointer p, const T& val) 84 {::new(p) T(val);} 85 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES 86 void construct(pointer p, T&& val) 87 {::new(p) T(std::move(val));} 88 #endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES 89 void destroy(pointer p) {p->~T();} 90 91 friend bool operator==(const test_allocator& x, const test_allocator& y) 92 {return x.data_ == y.data_;} 93 friend bool operator!=(const test_allocator& x, const test_allocator& y) 94 {return !(x == y);} 95 }; 96 97 template <class T> 98 class non_default_test_allocator 99 : public test_alloc_base 100 { 101 int data_; 102 103 template <class U> friend class non_default_test_allocator; 104 public: 105 106 typedef unsigned size_type; 107 typedef int difference_type; 108 typedef T value_type; 109 typedef value_type* pointer; 110 typedef const value_type* const_pointer; 111 typedef typename std::add_lvalue_reference<value_type>::type reference; 112 typedef typename std::add_lvalue_reference<const value_type>::type const_reference; 113 114 template <class U> struct rebind {typedef non_default_test_allocator<U> other;}; 115 116 // non_default_test_allocator() throw() : data_(0) {++count;} 117 explicit non_default_test_allocator(int i) throw() : data_(i) {++count;} 118 non_default_test_allocator(const non_default_test_allocator& a) throw() 119 : data_(a.data_) {++count;} 120 template <class U> non_default_test_allocator(const non_default_test_allocator<U>& a) throw() 121 : data_(a.data_) {++count;} 122 ~non_default_test_allocator() throw() {assert(data_ >= 0); --count; data_ = -1;} 123 pointer address(reference x) const {return &x;} 124 const_pointer address(const_reference x) const {return &x;} 125 pointer allocate(size_type n, const void* = 0) 126 { 127 assert(data_ >= 0); 128 if (time_to_throw >= throw_after) { 129 #ifndef _LIBCPP_NO_EXCEPTIONS 130 throw std::bad_alloc(); 131 #else 132 std::terminate(); 133 #endif 134 } 135 ++time_to_throw; 136 ++alloc_count; 137 return (pointer)::operator new (n * sizeof(T)); 138 } 139 void deallocate(pointer p, size_type n) 140 {assert(data_ >= 0); --alloc_count; ::operator delete((void*)p); } 141 size_type max_size() const throw() 142 {return UINT_MAX / sizeof(T);} 143 void construct(pointer p, const T& val) 144 {::new(p) T(val);} 145 #ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES 146 void construct(pointer p, T&& val) 147 {::new(p) T(std::move(val));} 148 #endif // _LIBCPP_HAS_NO_RVALUE_REFERENCES 149 void destroy(pointer p) {p->~T();} 150 151 friend bool operator==(const non_default_test_allocator& x, const non_default_test_allocator& y) 152 {return x.data_ == y.data_;} 153 friend bool operator!=(const non_default_test_allocator& x, const non_default_test_allocator& y) 154 {return !(x == y);} 155 }; 156 157 template <> 158 class test_allocator<void> 159 : public test_alloc_base 160 { 161 int data_; 162 163 template <class U> friend class test_allocator; 164 public: 165 166 typedef unsigned size_type; 167 typedef int difference_type; 168 typedef void value_type; 169 typedef value_type* pointer; 170 typedef const value_type* const_pointer; 171 172 template <class U> struct rebind {typedef test_allocator<U> other;}; 173 174 test_allocator() throw() : data_(-1) {} 175 explicit test_allocator(int i) throw() : data_(i) {} 176 test_allocator(const test_allocator& a) throw() 177 : data_(a.data_) {} 178 template <class U> test_allocator(const test_allocator<U>& a) throw() 179 : data_(a.data_) {} 180 ~test_allocator() throw() {data_ = 0;} 181 182 friend bool operator==(const test_allocator& x, const test_allocator& y) 183 {return x.data_ == y.data_;} 184 friend bool operator!=(const test_allocator& x, const test_allocator& y) 185 {return !(x == y);} 186 }; 187 188 template <class T> 189 class other_allocator 190 { 191 int data_; 192 193 template <class U> friend class other_allocator; 194 195 public: 196 typedef T value_type; 197 198 other_allocator() : data_(-1) {} 199 explicit other_allocator(int i) : data_(i) {} 200 template <class U> other_allocator(const other_allocator<U>& a) 201 : data_(a.data_) {} 202 T* allocate(std::size_t n) 203 {return (T*)::operator new(n * sizeof(T));} 204 void deallocate(T* p, std::size_t n) 205 {::operator delete((void*)p);} 206 207 other_allocator select_on_container_copy_construction() const 208 {return other_allocator(-2);} 209 210 friend bool operator==(const other_allocator& x, const other_allocator& y) 211 {return x.data_ == y.data_;} 212 friend bool operator!=(const other_allocator& x, const other_allocator& y) 213 {return !(x == y);} 214 215 typedef std::true_type propagate_on_container_copy_assignment; 216 typedef std::true_type propagate_on_container_move_assignment; 217 typedef std::true_type propagate_on_container_swap; 218 219 #ifdef _LIBCPP_HAS_NO_ADVANCED_SFINAE 220 std::size_t max_size() const 221 {return UINT_MAX / sizeof(T);} 222 #endif // _LIBCPP_HAS_NO_ADVANCED_SFINAE 223 224 }; 225 226 #endif // TEST_ALLOCATOR_H 227