// Copyright (C) 2013 The Android Open Source Project // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions // are met: // 1. Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // 2. Redistributions in binary form must reproduce the above copyright // notice, this list of conditions and the following disclaimer in the // documentation and/or other materials provided with the distribution. // 3. Neither the name of the project nor the names of its contributors // may be used to endorse or promote products derived from this software // without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND // ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE // ARE DISCLAIMED. IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS // OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) // HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT // LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY // OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF // SUCH DAMAGE. #include #include #include "cxxabi_defines.h" using std::size_t; namespace { using namespace __cxxabiv1; typedef __cxa_vec_constructor constructor_func; typedef __cxa_vec_copy_constructor copy_constructor_func; typedef __cxa_vec_destructor destructor_func; typedef void* (*alloc_func)(size_t); typedef void (*dealloc_func)(void*); typedef void (*dealloc2_func)(void*, size_t); // Helper class to ensure a ptr is deallocated on scope exit unless // the release() method has been called. class scoped_block { public: scoped_block(void* ptr, size_t size, dealloc2_func dealloc) : ptr_(ptr), size_(size), dealloc_(dealloc) {} ~scoped_block() { if (dealloc_) dealloc_(ptr_, size_); } void release() { dealloc_ = 0; } private: void* ptr_; size_t size_; dealloc2_func dealloc_; }; // Helper class to ensure a vector is cleaned up on scope exit // unless the release() method has been called. class scoped_cleanup { public: scoped_cleanup(void* array, size_t& index, size_t element_size, destructor_func destructor) : array_(array), index_(index), element_size_(element_size), destructor_(destructor) {} ~scoped_cleanup() { if (destructor_) __cxxabiv1::__cxa_vec_cleanup(array_, index_, element_size_, destructor_); } void release() { destructor_ = 0; } private: void* array_; size_t& index_; size_t element_size_; destructor_func destructor_; }; // Helper class that calls __fatal_error() with a given message if // it exits a scope without a previous call to release(). class scoped_catcher { public: scoped_catcher(const char* message) : message_(message) {} ~scoped_catcher() { if (message_) __gabixx::__fatal_error(message_); } void release() { message_ = 0; } private: const char* message_; }; } // namespace namespace __cxxabiv1 { extern "C" { void* __cxa_vec_new(size_t element_count, size_t element_size, size_t padding_size, constructor_func constructor, destructor_func destructor) { return __cxa_vec_new2(element_count, element_size, padding_size, constructor, destructor, &operator new[], &operator delete []); } void* __cxa_vec_new2(size_t element_count, size_t element_size, size_t padding_size, constructor_func constructor, destructor_func destructor, alloc_func alloc, dealloc_func dealloc) { // The only difference with __cxa_vec_new3 is the type of the // dealloc parameter. force a cast because on all supported // platforms, it is possible to call the dealloc function here // with two parameters. The second one will simply be ignored. return __cxa_vec_new3(element_count, element_size, padding_size, constructor, destructor, alloc, reinterpret_cast(dealloc)); } void* __cxa_vec_new3(size_t element_count, size_t element_size, size_t padding_size, constructor_func constructor, destructor_func destructor, alloc_func alloc, dealloc2_func dealloc) { // Compute the size of the needed memory block, and throw // std::bad_alloc() on overflow. bool overflow = false; size_t size = 0; if (element_size > 0 && element_count > size_t(-1) / element_size) overflow = true; else { size = element_count * element_size; if (size + padding_size < size) overflow = true; } if (overflow) throw std::bad_alloc(); // Allocate memory. Do not throw if NULL is returned. char* base = static_cast(alloc(size)); if (!base) return base; // Ensure the block is freed if construction throws. scoped_block block(base, size, dealloc); if (padding_size) { base += padding_size; reinterpret_cast(base)[-1] = element_count; #ifdef __arm__ // Required by the ARM C++ ABI. reinterpret_cast(base)[-2] = element_size; #endif } __cxa_vec_ctor(base, element_count, element_size, constructor, destructor); // Construction succeeded, no need to release the block. block.release(); return base; } #ifdef __arm__ // On ARM, __cxa_vec_ctor and __cxa_vec_cctor must return // their first parameter. Handle this here. #define _CXA_VEC_CTOR_RETURN(x) return x #else #define _CXA_VEC_CTOR_RETURN(x) return #endif __cxa_vec_ctor_return_type __cxa_vec_ctor(void* array_address, size_t element_count, size_t element_size, constructor_func constructor, destructor_func destructor) { if (constructor) { size_t n = 0; char* base = static_cast(array_address); scoped_cleanup cleanup(array_address, n, element_size, destructor); for (; n != element_count; ++n) { constructor(base); base += element_size; } cleanup.release(); } _CXA_VEC_CTOR_RETURN(array_address); } // Given the (data) address of an array, the number of elements, // and the size of its elements, call the given destructor on each // element. If the destructor throws an exception, rethrow after // destroying the remaining elements if possible. If the destructor // throws a second exception, call terminate(). The destructor // pointer may be NULL, in which case this routine does nothing. void __cxa_vec_dtor(void* array_address, size_t element_count, size_t element_size, destructor_func destructor) { if (!destructor) return; char* base = static_cast(array_address); size_t n = element_count; scoped_cleanup cleanup(array_address, n, element_size, destructor); base += element_count * element_size; // Note: n must be decremented before the destructor call // to avoid cleaning up one extra unwanted item. while (n--) { base -= element_size; destructor(base); } cleanup.release(); } // Given the (data) address of an array, the number of elements, // and the size of its elements, call the given destructor on each // element. If the destructor throws an exception, call terminate(). // The destructor pointer may be NULL, in which case this routine // does nothing. void __cxa_vec_cleanup(void* array_address, size_t element_count, size_t element_size, destructor_func destructor) { if (!destructor) return; char* base = static_cast(array_address); size_t n = element_count; base += n * element_size; scoped_catcher catcher("exception raised in vector destructor!"); while (n--) { base -= element_size; destructor(base); } catcher.release(); } // If the array_address is NULL, return immediately. Otherwise, // given the (data) address of an array, the non-negative size // of prefix padding for the cookie, and the size of its elements, // call the given destructor on each element, using the cookie to // determine the number of elements, and then delete the space by // calling ::operator delete[](void *). If the destructor throws an // exception, rethrow after (a) destroying the remaining elements, // and (b) deallocating the storage. If the destructor throws a // second exception, call terminate(). If padding_size is 0, the // destructor pointer must be NULL. If the destructor pointer is NULL, // no destructor call is to be made. void __cxa_vec_delete(void* array_address, size_t element_size, size_t padding_size, destructor_func destructor) { __cxa_vec_delete2(array_address, element_size, padding_size, destructor, &operator delete []); } // Same as __cxa_vec_delete, except that the given function is used // for deallocation instead of the default delete function. If dealloc // throws an exception, the result is undefined. The dealloc pointer // may not be NULL. void __cxa_vec_delete2(void* array_address, size_t element_size, size_t padding_size, destructor_func destructor, dealloc_func dealloc) { // Same trick than the one used on __cxa_vec_new2. __cxa_vec_delete3(array_address, element_size, padding_size, destructor, reinterpret_cast(dealloc)); } // Same as __cxa_vec_delete, except that the given function is used // for deallocation instead of the default delete function. The // deallocation function takes both the object address and its size. // If dealloc throws an exception, the result is undefined. The dealloc // pointer may not be NULL. void __cxa_vec_delete3(void* array_address, size_t element_size, size_t padding_size, destructor_func destructor, dealloc2_func dealloc) { if (!array_address) return; char* base = static_cast(array_address); if (!padding_size) { // If here is no padding size, asume the deallocator knows // how to handle this. Useful when called from __cxa_vec_delete2. dealloc(base, 0); return; } size_t element_count = reinterpret_cast(base)[-1]; base -= padding_size; size_t size = element_count * element_size + padding_size; // Always deallocate base on exit. scoped_block block(base, size, dealloc); if (padding_size > 0 && destructor != 0) __cxa_vec_dtor(array_address, element_count, element_size, destructor); } __cxa_vec_ctor_return_type __cxa_vec_cctor(void* dst_array, void* src_array, size_t element_count, size_t element_size, copy_constructor_func copy_constructor, destructor_func destructor) { if (copy_constructor) { size_t n = 0; char* dst = static_cast(dst_array); char* src = static_cast(src_array); scoped_cleanup cleanup(dst_array, n, element_size, destructor); for ( ; n != element_count; ++n) { copy_constructor(dst, src); dst += element_size; src += element_size; } cleanup.release(); } _CXA_VEC_CTOR_RETURN(dst_array); } } // extern "C" } // namespace __cxxabiv1 #if _GABIXX_ARM_ABI // The following functions are required by the ARM ABI, even // though neither GCC nor LLVM generate any code that uses it. // This may be important for machine code generated by other // compilers though (e.g. RCVT), which may depend on them. // They're supposed to simplify calling code. namespace __aeabiv1 { extern "C" { using namespace __cxxabiv1; void* __aeabi_vec_ctor_nocookie_nodtor(void* array_address, constructor_func constructor, size_t element_size, size_t element_count) { return __cxa_vec_ctor(array_address, element_count, element_size, constructor, /* destructor */ NULL); } void* __aeabi_vec_ctor_cookie_nodtor(void* array_address, constructor_func constructor, size_t element_size, size_t element_count) { if (!array_address) return array_address; size_t* base = reinterpret_cast(array_address) + 2; base[-2] = element_size; base[-1] = element_count; return __cxa_vec_ctor(base, element_count, element_size, constructor, /* destructor */ NULL); } void* __aeabi_vec_cctor_nocookie_nodtor( void* dst_array, void* src_array, size_t element_size, size_t element_count, copy_constructor_func copy_constructor) { return __cxa_vec_cctor(dst_array, src_array, element_count, element_size, copy_constructor, NULL); } void* __aeabi_vec_new_cookie_noctor(size_t element_size, size_t element_count) { return __cxa_vec_new(element_count, element_size, /* padding */ 2 * sizeof(size_t), /* constructor */ NULL, /* destructor */ NULL); } void* __aeabi_vec_new_nocookie(size_t element_size, size_t element_count, constructor_func constructor) { return __cxa_vec_new(element_count, element_size, /* padding */ 0, constructor, /* destructor */ NULL); } void* __aeabi_vec_new_cookie_nodtor(size_t element_size, size_t element_count, constructor_func constructor) { return __cxa_vec_new(element_count, element_size, /* padding */ 2 * sizeof(size_t), constructor, /* destructor */ NULL); } void* __aeabi_vec_new_cookie(size_t element_size, size_t element_count, constructor_func constructor, destructor_func destructor) { return __cxa_vec_new(element_count, element_size, /* padding */ 2 * sizeof(size_t), constructor, destructor); } void* __aeabi_vec_dtor(void* array_address, destructor_func destructor, size_t element_size, size_t element_count) { __cxa_vec_dtor(array_address, element_count, element_size, destructor); return reinterpret_cast(array_address) - 2; } void* __aeabi_vec_dtor_cookie(void* array_address, destructor_func destructor) { if (!array_address) return NULL; size_t* base = reinterpret_cast(array_address); __cxa_vec_dtor(array_address, /* element_count */ base[-1], /* element_size */ base[-2], destructor); return base - 2; } void __aeabi_vec_delete(void* array_address, destructor_func destructor) { if (array_address) { size_t* base = reinterpret_cast(array_address); __cxa_vec_delete(array_address, /* element_size */ base[-2], /* padding */ 2 * sizeof(size_t), destructor); } } void __aeabi_vec_delete3(void* array_address, destructor_func destructor, dealloc2_func dealloc) { if (array_address) { size_t* base = reinterpret_cast(array_address); __cxa_vec_delete3(array_address, /* element_size */ base[-2], /* padding */ 2 * sizeof(size_t), destructor, dealloc); } } void __aeabi_vec_delete3_nodtor(void* array_address, dealloc2_func dealloc) { if (array_address) { size_t* base = reinterpret_cast(array_address); __cxa_vec_delete3(array_address, /* element_size */ base[-2], /* padding */ 2 * sizeof(size_t), /* destructor */ NULL, dealloc); } } } // extern "C" } // namespace __aeabiv1 #endif // _GABIXX_ARM_ABI