Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 1 | //===--------------------------- new.cpp ----------------------------------===// |
| 2 | // |
Howard Hinnant | c566dc3 | 2010-05-11 21:36:01 +0000 | [diff] [blame] | 3 | // The LLVM Compiler Infrastructure |
Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 4 | // |
Howard Hinnant | ee11c31 | 2010-11-16 22:09:02 +0000 | [diff] [blame] | 5 | // This file is dual licensed under the MIT and the University of Illinois Open |
| 6 | // Source Licenses. See LICENSE.TXT for details. |
Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | |
Marshall Clow | 3bf7713 | 2013-09-11 01:38:42 +0000 | [diff] [blame] | 10 | #define _LIBCPP_BUILDING_NEW |
| 11 | |
Howard Hinnant | 155c2af | 2010-05-24 17:49:41 +0000 | [diff] [blame] | 12 | #include <stdlib.h> |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 13 | |
Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 14 | #include "new" |
Weiming Zhao | b613db7 | 2017-09-19 23:18:03 +0000 | [diff] [blame] | 15 | #include "include/atomic_support.h" |
Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 16 | |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 17 | #if defined(_LIBCPP_ABI_MICROSOFT) |
Shoaib Meenai | cfd1960 | 2017-10-09 19:25:17 +0000 | [diff] [blame^] | 18 | #if defined(_LIBCPP_NO_VCRUNTIME) |
| 19 | #include "support/runtime/new_handler_fallback.ipp" |
| 20 | #endif |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 21 | #elif defined(LIBCXX_BUILDING_LIBCXXABI) |
| 22 | #include <cxxabi.h> |
| 23 | #elif defined(LIBCXXRT) |
| 24 | #include <cxxabi.h> |
| 25 | #include "support/runtime/new_handler_fallback.ipp" |
| 26 | #elif defined(__GLIBCXX__) |
| 27 | // nothing todo |
| 28 | #else |
Eric Fiselier | 0fbd44d | 2017-02-10 09:16:29 +0000 | [diff] [blame] | 29 | # if defined(__APPLE__) && !defined(_LIBCPP_BUILDING_HAS_NO_ABI_LIBRARY) |
| 30 | # include <cxxabi.h> // FIXME: remove this once buildit is gone. |
| 31 | # else |
| 32 | # include "support/runtime/new_handler_fallback.ipp" |
| 33 | # endif |
Eric Fiselier | 88d1171 | 2017-02-10 07:43:08 +0000 | [diff] [blame] | 34 | #endif |
Eric Fiselier | 4e5fccc | 2017-02-10 04:25:33 +0000 | [diff] [blame] | 35 | |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 36 | namespace std |
| 37 | { |
| 38 | |
Peter Collingbourne | dc00995 | 2013-10-06 22:13:16 +0000 | [diff] [blame] | 39 | #ifndef __GLIBCXX__ |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 40 | const nothrow_t nothrow = {}; |
| 41 | #endif |
| 42 | |
| 43 | #ifndef LIBSTDCXX |
| 44 | |
| 45 | void |
| 46 | __throw_bad_alloc() |
| 47 | { |
| 48 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 49 | throw bad_alloc(); |
| 50 | #else |
| 51 | _VSTD::abort(); |
| 52 | #endif |
| 53 | } |
| 54 | |
| 55 | #endif // !LIBSTDCXX |
| 56 | |
| 57 | } // std |
| 58 | |
Shoaib Meenai | cfd1960 | 2017-10-09 19:25:17 +0000 | [diff] [blame^] | 59 | #if !defined(__GLIBCXX__) && \ |
| 60 | (!defined(_LIBCPP_ABI_MICROSOFT) || defined(_LIBCPP_NO_VCRUNTIME)) && \ |
Eric Fiselier | 95555c9 | 2017-03-02 19:35:33 +0000 | [diff] [blame] | 61 | !defined(_LIBCPP_DISABLE_NEW_DELETE_DEFINITIONS) |
Peter Collingbourne | dc00995 | 2013-10-06 22:13:16 +0000 | [diff] [blame] | 62 | |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 63 | // Implement all new and delete operators as weak definitions |
Eric Fiselier | cd4496b | 2015-08-20 05:23:16 +0000 | [diff] [blame] | 64 | // in this shared library, so that they can be overridden by programs |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 65 | // that define non-weak copies of the functions. |
| 66 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 67 | _LIBCPP_WEAK |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 68 | void * |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 69 | operator new(std::size_t size) _THROW_BAD_ALLOC |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 70 | { |
| 71 | if (size == 0) |
| 72 | size = 1; |
| 73 | void* p; |
| 74 | while ((p = ::malloc(size)) == 0) |
| 75 | { |
Howard Hinnant | 34468d4 | 2010-08-22 13:53:14 +0000 | [diff] [blame] | 76 | // If malloc fails and there is a new_handler, |
| 77 | // call it to try free up memory. |
Howard Hinnant | 8c65b45 | 2010-12-04 19:56:43 +0000 | [diff] [blame] | 78 | std::new_handler nh = std::get_new_handler(); |
Howard Hinnant | f64dfce | 2010-12-04 19:54:11 +0000 | [diff] [blame] | 79 | if (nh) |
| 80 | nh(); |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 81 | else |
Howard Hinnant | 72f7358 | 2010-08-11 17:04:31 +0000 | [diff] [blame] | 82 | #ifndef _LIBCPP_NO_EXCEPTIONS |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 83 | throw std::bad_alloc(); |
Howard Hinnant | 72f7358 | 2010-08-11 17:04:31 +0000 | [diff] [blame] | 84 | #else |
| 85 | break; |
| 86 | #endif |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 87 | } |
| 88 | return p; |
| 89 | } |
| 90 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 91 | _LIBCPP_WEAK |
Eric Fiselier | 850652f | 2017-01-20 01:13:49 +0000 | [diff] [blame] | 92 | void* |
| 93 | operator new(size_t size, const std::nothrow_t&) _NOEXCEPT |
| 94 | { |
| 95 | void* p = 0; |
| 96 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 97 | try |
| 98 | { |
| 99 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 100 | p = ::operator new(size); |
| 101 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 102 | } |
| 103 | catch (...) |
| 104 | { |
| 105 | } |
| 106 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 107 | return p; |
| 108 | } |
| 109 | |
| 110 | _LIBCPP_WEAK |
| 111 | void* |
| 112 | operator new[](size_t size) _THROW_BAD_ALLOC |
| 113 | { |
| 114 | return ::operator new(size); |
| 115 | } |
| 116 | |
| 117 | _LIBCPP_WEAK |
| 118 | void* |
| 119 | operator new[](size_t size, const std::nothrow_t&) _NOEXCEPT |
| 120 | { |
| 121 | void* p = 0; |
| 122 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 123 | try |
| 124 | { |
| 125 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 126 | p = ::operator new[](size); |
| 127 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 128 | } |
| 129 | catch (...) |
| 130 | { |
| 131 | } |
| 132 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 133 | return p; |
| 134 | } |
| 135 | |
| 136 | _LIBCPP_WEAK |
| 137 | void |
| 138 | operator delete(void* ptr) _NOEXCEPT |
| 139 | { |
| 140 | if (ptr) |
| 141 | ::free(ptr); |
| 142 | } |
| 143 | |
| 144 | _LIBCPP_WEAK |
| 145 | void |
| 146 | operator delete(void* ptr, const std::nothrow_t&) _NOEXCEPT |
| 147 | { |
| 148 | ::operator delete(ptr); |
| 149 | } |
| 150 | |
| 151 | _LIBCPP_WEAK |
| 152 | void |
| 153 | operator delete(void* ptr, size_t) _NOEXCEPT |
| 154 | { |
| 155 | ::operator delete(ptr); |
| 156 | } |
| 157 | |
| 158 | _LIBCPP_WEAK |
| 159 | void |
| 160 | operator delete[] (void* ptr) _NOEXCEPT |
| 161 | { |
| 162 | ::operator delete(ptr); |
| 163 | } |
| 164 | |
| 165 | _LIBCPP_WEAK |
| 166 | void |
| 167 | operator delete[] (void* ptr, const std::nothrow_t&) _NOEXCEPT |
| 168 | { |
| 169 | ::operator delete[](ptr); |
| 170 | } |
| 171 | |
| 172 | _LIBCPP_WEAK |
| 173 | void |
| 174 | operator delete[] (void* ptr, size_t) _NOEXCEPT |
| 175 | { |
| 176 | ::operator delete[](ptr); |
| 177 | } |
| 178 | |
Eric Fiselier | 44832fb | 2017-01-20 01:47:26 +0000 | [diff] [blame] | 179 | #if !defined(_LIBCPP_HAS_NO_ALIGNED_ALLOCATION) |
| 180 | |
Eric Fiselier | 850652f | 2017-01-20 01:13:49 +0000 | [diff] [blame] | 181 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 182 | void * |
| 183 | operator new(std::size_t size, std::align_val_t alignment) _THROW_BAD_ALLOC |
| 184 | { |
| 185 | if (size == 0) |
| 186 | size = 1; |
| 187 | if (static_cast<size_t>(alignment) < sizeof(void*)) |
| 188 | alignment = std::align_val_t(sizeof(void*)); |
| 189 | void* p; |
Eric Fiselier | bb999f9 | 2017-05-31 22:14:05 +0000 | [diff] [blame] | 190 | #if defined(_LIBCPP_MSVCRT_LIKE) |
Eric Fiselier | 29aaae1 | 2016-12-23 20:17:23 +0000 | [diff] [blame] | 191 | while ((p = _aligned_malloc(size, static_cast<size_t>(alignment))) == nullptr) |
| 192 | #else |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 193 | while (::posix_memalign(&p, static_cast<size_t>(alignment), size) != 0) |
Eric Fiselier | 29aaae1 | 2016-12-23 20:17:23 +0000 | [diff] [blame] | 194 | #endif |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 195 | { |
| 196 | // If posix_memalign fails and there is a new_handler, |
| 197 | // call it to try free up memory. |
| 198 | std::new_handler nh = std::get_new_handler(); |
| 199 | if (nh) |
| 200 | nh(); |
| 201 | else { |
| 202 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 203 | throw std::bad_alloc(); |
| 204 | #else |
| 205 | p = nullptr; // posix_memalign doesn't initialize 'p' on failure |
| 206 | break; |
| 207 | #endif |
| 208 | } |
| 209 | } |
| 210 | return p; |
| 211 | } |
| 212 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 213 | _LIBCPP_WEAK |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 214 | void* |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 215 | operator new(size_t size, std::align_val_t alignment, const std::nothrow_t&) _NOEXCEPT |
| 216 | { |
| 217 | void* p = 0; |
| 218 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 219 | try |
| 220 | { |
| 221 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 222 | p = ::operator new(size, alignment); |
| 223 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 224 | } |
| 225 | catch (...) |
| 226 | { |
| 227 | } |
| 228 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 229 | return p; |
| 230 | } |
| 231 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 232 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 233 | void* |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 234 | operator new[](size_t size, std::align_val_t alignment) _THROW_BAD_ALLOC |
| 235 | { |
| 236 | return ::operator new(size, alignment); |
| 237 | } |
| 238 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 239 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 240 | void* |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 241 | operator new[](size_t size, std::align_val_t alignment, const std::nothrow_t&) _NOEXCEPT |
| 242 | { |
| 243 | void* p = 0; |
| 244 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 245 | try |
| 246 | { |
| 247 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 248 | p = ::operator new[](size, alignment); |
| 249 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 250 | } |
| 251 | catch (...) |
| 252 | { |
| 253 | } |
| 254 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 255 | return p; |
| 256 | } |
| 257 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 258 | _LIBCPP_WEAK |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 259 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 260 | operator delete(void* ptr, std::align_val_t) _NOEXCEPT |
| 261 | { |
| 262 | if (ptr) |
Eric Fiselier | bb999f9 | 2017-05-31 22:14:05 +0000 | [diff] [blame] | 263 | #if defined(_LIBCPP_MSVCRT_LIKE) |
Shoaib Meenai | 35d3e69 | 2017-01-12 06:22:36 +0000 | [diff] [blame] | 264 | ::_aligned_free(ptr); |
| 265 | #else |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 266 | ::free(ptr); |
Shoaib Meenai | 35d3e69 | 2017-01-12 06:22:36 +0000 | [diff] [blame] | 267 | #endif |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 268 | } |
| 269 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 270 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 271 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 272 | operator delete(void* ptr, std::align_val_t alignment, const std::nothrow_t&) _NOEXCEPT |
| 273 | { |
| 274 | ::operator delete(ptr, alignment); |
| 275 | } |
| 276 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 277 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 278 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 279 | operator delete(void* ptr, size_t, std::align_val_t alignment) _NOEXCEPT |
| 280 | { |
| 281 | ::operator delete(ptr, alignment); |
| 282 | } |
| 283 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 284 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 285 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 286 | operator delete[] (void* ptr, std::align_val_t alignment) _NOEXCEPT |
| 287 | { |
| 288 | ::operator delete(ptr, alignment); |
| 289 | } |
| 290 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 291 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 292 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 293 | operator delete[] (void* ptr, std::align_val_t alignment, const std::nothrow_t&) _NOEXCEPT |
| 294 | { |
| 295 | ::operator delete[](ptr, alignment); |
| 296 | } |
| 297 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 298 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 299 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 300 | operator delete[] (void* ptr, size_t, std::align_val_t alignment) _NOEXCEPT |
| 301 | { |
| 302 | ::operator delete[](ptr, alignment); |
| 303 | } |
| 304 | |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 305 | #endif // !_LIBCPP_HAS_NO_ALIGNED_ALLOCATION |
Shoaib Meenai | cfd1960 | 2017-10-09 19:25:17 +0000 | [diff] [blame^] | 306 | #endif // !__GLIBCXX__ && (!_LIBCPP_ABI_MICROSOFT || _LIBCPP_NO_VCRUNTIME) && !_LIBCPP_DISABLE_NEW_DELETE_DEFINITIONS |