Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 1 | //===--------------------------- new.cpp ----------------------------------===// |
| 2 | // |
Chandler Carruth | d201210 | 2019-01-19 10:56:40 +0000 | [diff] [blame] | 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
Howard Hinnant | 155c2af | 2010-05-24 17:49:41 +0000 | [diff] [blame] | 9 | #include <stdlib.h> |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 10 | |
Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 11 | #include "new" |
Weiming Zhao | b613db7 | 2017-09-19 23:18:03 +0000 | [diff] [blame] | 12 | #include "include/atomic_support.h" |
Howard Hinnant | c51e102 | 2010-05-11 19:42:16 +0000 | [diff] [blame] | 13 | |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 14 | #if defined(_LIBCPP_ABI_MICROSOFT) |
Louis Dionne | a810c33 | 2019-03-21 18:19:21 +0000 | [diff] [blame] | 15 | # if !defined(_LIBCPP_ABI_VCRUNTIME) |
| 16 | # include "support/runtime/new_handler_fallback.ipp" |
| 17 | # endif |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 18 | #elif defined(LIBCXX_BUILDING_LIBCXXABI) |
Louis Dionne | a810c33 | 2019-03-21 18:19:21 +0000 | [diff] [blame] | 19 | # include <cxxabi.h> |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 20 | #elif defined(LIBCXXRT) |
Louis Dionne | a810c33 | 2019-03-21 18:19:21 +0000 | [diff] [blame] | 21 | # include <cxxabi.h> |
Eric Fiselier | 0fbd44d | 2017-02-10 09:16:29 +0000 | [diff] [blame] | 22 | # include "support/runtime/new_handler_fallback.ipp" |
Louis Dionne | a810c33 | 2019-03-21 18:19:21 +0000 | [diff] [blame] | 23 | #elif defined(__GLIBCXX__) |
| 24 | // nothing to do |
Louis Dionne | 9fd1b09 | 2019-04-17 21:57:49 +0000 | [diff] [blame] | 25 | #else |
Louis Dionne | 549c055 | 2019-04-16 19:26:56 +0000 | [diff] [blame] | 26 | # include "support/runtime/new_handler_fallback.ipp" |
Eric Fiselier | 88d1171 | 2017-02-10 07:43:08 +0000 | [diff] [blame] | 27 | #endif |
Eric Fiselier | 4e5fccc | 2017-02-10 04:25:33 +0000 | [diff] [blame] | 28 | |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 29 | namespace std |
| 30 | { |
| 31 | |
Peter Collingbourne | dc00995 | 2013-10-06 22:13:16 +0000 | [diff] [blame] | 32 | #ifndef __GLIBCXX__ |
Thomas Anderson | 637a0ad | 2019-01-30 19:09:41 +0000 | [diff] [blame] | 33 | const nothrow_t nothrow{}; |
Eric Fiselier | ec3a167 | 2017-02-10 08:57:35 +0000 | [diff] [blame] | 34 | #endif |
| 35 | |
| 36 | #ifndef LIBSTDCXX |
| 37 | |
| 38 | void |
| 39 | __throw_bad_alloc() |
| 40 | { |
| 41 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 42 | throw bad_alloc(); |
| 43 | #else |
| 44 | _VSTD::abort(); |
| 45 | #endif |
| 46 | } |
| 47 | |
| 48 | #endif // !LIBSTDCXX |
| 49 | |
| 50 | } // std |
| 51 | |
Shoaib Meenai | cfd1960 | 2017-10-09 19:25:17 +0000 | [diff] [blame] | 52 | #if !defined(__GLIBCXX__) && \ |
Eric Fiselier | 85f6633 | 2019-03-05 01:57:01 +0000 | [diff] [blame] | 53 | !defined(_LIBCPP_ABI_VCRUNTIME) && \ |
Eric Fiselier | 95555c9 | 2017-03-02 19:35:33 +0000 | [diff] [blame] | 54 | !defined(_LIBCPP_DISABLE_NEW_DELETE_DEFINITIONS) |
Peter Collingbourne | dc00995 | 2013-10-06 22:13:16 +0000 | [diff] [blame] | 55 | |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 56 | // Implement all new and delete operators as weak definitions |
Eric Fiselier | cd4496b | 2015-08-20 05:23:16 +0000 | [diff] [blame] | 57 | // in this shared library, so that they can be overridden by programs |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 58 | // that define non-weak copies of the functions. |
| 59 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 60 | _LIBCPP_WEAK |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 61 | void * |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 62 | operator new(std::size_t size) _THROW_BAD_ALLOC |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 63 | { |
| 64 | if (size == 0) |
| 65 | size = 1; |
| 66 | void* p; |
| 67 | while ((p = ::malloc(size)) == 0) |
| 68 | { |
Howard Hinnant | 34468d4 | 2010-08-22 13:53:14 +0000 | [diff] [blame] | 69 | // If malloc fails and there is a new_handler, |
| 70 | // call it to try free up memory. |
Howard Hinnant | 8c65b45 | 2010-12-04 19:56:43 +0000 | [diff] [blame] | 71 | std::new_handler nh = std::get_new_handler(); |
Howard Hinnant | f64dfce | 2010-12-04 19:54:11 +0000 | [diff] [blame] | 72 | if (nh) |
| 73 | nh(); |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 74 | else |
Howard Hinnant | 72f7358 | 2010-08-11 17:04:31 +0000 | [diff] [blame] | 75 | #ifndef _LIBCPP_NO_EXCEPTIONS |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 76 | throw std::bad_alloc(); |
Howard Hinnant | 72f7358 | 2010-08-11 17:04:31 +0000 | [diff] [blame] | 77 | #else |
| 78 | break; |
| 79 | #endif |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 80 | } |
| 81 | return p; |
| 82 | } |
| 83 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 84 | _LIBCPP_WEAK |
Eric Fiselier | 850652f | 2017-01-20 01:13:49 +0000 | [diff] [blame] | 85 | void* |
| 86 | operator new(size_t size, const std::nothrow_t&) _NOEXCEPT |
| 87 | { |
| 88 | void* p = 0; |
| 89 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 90 | try |
| 91 | { |
| 92 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 93 | p = ::operator new(size); |
| 94 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 95 | } |
| 96 | catch (...) |
| 97 | { |
| 98 | } |
| 99 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 100 | return p; |
| 101 | } |
| 102 | |
| 103 | _LIBCPP_WEAK |
| 104 | void* |
| 105 | operator new[](size_t size) _THROW_BAD_ALLOC |
| 106 | { |
| 107 | return ::operator new(size); |
| 108 | } |
| 109 | |
| 110 | _LIBCPP_WEAK |
| 111 | void* |
| 112 | operator new[](size_t size, const std::nothrow_t&) _NOEXCEPT |
| 113 | { |
| 114 | void* p = 0; |
| 115 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 116 | try |
| 117 | { |
| 118 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 119 | p = ::operator new[](size); |
| 120 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 121 | } |
| 122 | catch (...) |
| 123 | { |
| 124 | } |
| 125 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 126 | return p; |
| 127 | } |
| 128 | |
| 129 | _LIBCPP_WEAK |
| 130 | void |
| 131 | operator delete(void* ptr) _NOEXCEPT |
| 132 | { |
Louis Dionne | 0ecb547 | 2020-11-11 16:33:38 -0500 | [diff] [blame] | 133 | if (ptr) |
| 134 | ::free(ptr); |
Eric Fiselier | 850652f | 2017-01-20 01:13:49 +0000 | [diff] [blame] | 135 | } |
| 136 | |
| 137 | _LIBCPP_WEAK |
| 138 | void |
| 139 | operator delete(void* ptr, const std::nothrow_t&) _NOEXCEPT |
| 140 | { |
| 141 | ::operator delete(ptr); |
| 142 | } |
| 143 | |
| 144 | _LIBCPP_WEAK |
| 145 | void |
| 146 | operator delete(void* ptr, size_t) _NOEXCEPT |
| 147 | { |
| 148 | ::operator delete(ptr); |
| 149 | } |
| 150 | |
| 151 | _LIBCPP_WEAK |
| 152 | void |
| 153 | operator delete[] (void* ptr) _NOEXCEPT |
| 154 | { |
| 155 | ::operator delete(ptr); |
| 156 | } |
| 157 | |
| 158 | _LIBCPP_WEAK |
| 159 | void |
| 160 | operator delete[] (void* ptr, const std::nothrow_t&) _NOEXCEPT |
| 161 | { |
| 162 | ::operator delete[](ptr); |
| 163 | } |
| 164 | |
| 165 | _LIBCPP_WEAK |
| 166 | void |
| 167 | operator delete[] (void* ptr, size_t) _NOEXCEPT |
| 168 | { |
| 169 | ::operator delete[](ptr); |
| 170 | } |
| 171 | |
Eric Fiselier | ff4c8a2 | 2018-10-11 00:17:24 +0000 | [diff] [blame] | 172 | #if !defined(_LIBCPP_HAS_NO_LIBRARY_ALIGNED_ALLOCATION) |
Eric Fiselier | 44832fb | 2017-01-20 01:47:26 +0000 | [diff] [blame] | 173 | |
Eric Fiselier | 850652f | 2017-01-20 01:13:49 +0000 | [diff] [blame] | 174 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 175 | void * |
| 176 | operator new(std::size_t size, std::align_val_t alignment) _THROW_BAD_ALLOC |
| 177 | { |
| 178 | if (size == 0) |
| 179 | size = 1; |
| 180 | if (static_cast<size_t>(alignment) < sizeof(void*)) |
| 181 | alignment = std::align_val_t(sizeof(void*)); |
Louis Dionne | a392217 | 2020-11-12 15:14:33 -0500 | [diff] [blame^] | 182 | |
| 183 | // Try allocating memory. If allocation fails and there is a new_handler, |
| 184 | // call it to try free up memory, and try again until it succeeds, or until |
| 185 | // the new_handler decides to terminate. |
| 186 | // |
| 187 | // If allocation fails and there is no new_handler, we throw bad_alloc |
| 188 | // (or return nullptr if exceptions are disabled). |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 189 | void* p; |
Louis Dionne | a392217 | 2020-11-12 15:14:33 -0500 | [diff] [blame^] | 190 | while ((p = std::__libcpp_aligned_alloc(static_cast<std::size_t>(alignment), size)) == nullptr) |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 191 | { |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 192 | std::new_handler nh = std::get_new_handler(); |
| 193 | if (nh) |
| 194 | nh(); |
| 195 | else { |
| 196 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 197 | throw std::bad_alloc(); |
| 198 | #else |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 199 | break; |
| 200 | #endif |
| 201 | } |
| 202 | } |
| 203 | return p; |
| 204 | } |
| 205 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 206 | _LIBCPP_WEAK |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 207 | void* |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 208 | operator new(size_t size, std::align_val_t alignment, const std::nothrow_t&) _NOEXCEPT |
| 209 | { |
| 210 | void* p = 0; |
| 211 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 212 | try |
| 213 | { |
| 214 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 215 | p = ::operator new(size, alignment); |
| 216 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 217 | } |
| 218 | catch (...) |
| 219 | { |
| 220 | } |
| 221 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 222 | return p; |
| 223 | } |
| 224 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 225 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 226 | void* |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 227 | operator new[](size_t size, std::align_val_t alignment) _THROW_BAD_ALLOC |
| 228 | { |
| 229 | return ::operator new(size, alignment); |
| 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, const std::nothrow_t&) _NOEXCEPT |
| 235 | { |
| 236 | void* p = 0; |
| 237 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 238 | try |
| 239 | { |
| 240 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 241 | p = ::operator new[](size, alignment); |
| 242 | #ifndef _LIBCPP_NO_EXCEPTIONS |
| 243 | } |
| 244 | catch (...) |
| 245 | { |
| 246 | } |
| 247 | #endif // _LIBCPP_NO_EXCEPTIONS |
| 248 | return p; |
| 249 | } |
| 250 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 251 | _LIBCPP_WEAK |
Nick Kledzik | d1a61bb | 2010-05-14 20:19:37 +0000 | [diff] [blame] | 252 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 253 | operator delete(void* ptr, std::align_val_t) _NOEXCEPT |
| 254 | { |
Louis Dionne | a392217 | 2020-11-12 15:14:33 -0500 | [diff] [blame^] | 255 | if (ptr) { |
| 256 | std::__libcpp_aligned_free(ptr); |
| 257 | } |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 258 | } |
| 259 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 260 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 261 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 262 | operator delete(void* ptr, std::align_val_t alignment, const std::nothrow_t&) _NOEXCEPT |
| 263 | { |
| 264 | ::operator delete(ptr, alignment); |
| 265 | } |
| 266 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 267 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 268 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 269 | operator delete(void* ptr, size_t, std::align_val_t alignment) _NOEXCEPT |
| 270 | { |
| 271 | ::operator delete(ptr, alignment); |
| 272 | } |
| 273 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 274 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 275 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 276 | operator delete[] (void* ptr, std::align_val_t alignment) _NOEXCEPT |
| 277 | { |
| 278 | ::operator delete(ptr, alignment); |
| 279 | } |
| 280 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 281 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 282 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 283 | operator delete[] (void* ptr, std::align_val_t alignment, const std::nothrow_t&) _NOEXCEPT |
| 284 | { |
| 285 | ::operator delete[](ptr, alignment); |
| 286 | } |
| 287 | |
Shoaib Meenai | 2d71db4 | 2016-11-16 22:18:10 +0000 | [diff] [blame] | 288 | _LIBCPP_WEAK |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 289 | void |
Eric Fiselier | 162922f | 2016-10-14 06:46:30 +0000 | [diff] [blame] | 290 | operator delete[] (void* ptr, size_t, std::align_val_t alignment) _NOEXCEPT |
| 291 | { |
| 292 | ::operator delete[](ptr, alignment); |
| 293 | } |
| 294 | |
Eric Fiselier | ff4c8a2 | 2018-10-11 00:17:24 +0000 | [diff] [blame] | 295 | #endif // !_LIBCPP_HAS_NO_LIBRARY_ALIGNED_ALLOCATION |
Eric Fiselier | 85f6633 | 2019-03-05 01:57:01 +0000 | [diff] [blame] | 296 | #endif // !__GLIBCXX__ && !_LIBCPP_ABI_VCRUNTIME && !_LIBCPP_DISABLE_NEW_DELETE_DEFINITIONS |