// -*- C++ -*- // Testing allocator for the C++ library testsuite. // // Copyright (C) 2002-2016 Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License along // with this library; see the file COPYING3. If not see // . // // This file provides an test instrumentation allocator that can be // used to verify allocation functionality of standard library // containers. 2002.11.25 smw #ifndef _GLIBCXX_TESTSUITE_ALLOCATOR_H #define _GLIBCXX_TESTSUITE_ALLOCATOR_H #include #include #include #include #include namespace __gnu_test { class tracker_allocator_counter { public: typedef std::size_t size_type; static void allocate(size_type blocksize) { allocationCount_ += blocksize; } static void construct() { ++constructCount_; } static void destroy() { ++destructCount_; } static void deallocate(size_type blocksize) { deallocationCount_ += blocksize; } static size_type get_allocation_count() { return allocationCount_; } static size_type get_deallocation_count() { return deallocationCount_; } static int get_construct_count() { return constructCount_; } static int get_destruct_count() { return destructCount_; } static void reset() { allocationCount_ = 0; deallocationCount_ = 0; constructCount_ = 0; destructCount_ = 0; } private: static size_type allocationCount_; static size_type deallocationCount_; static int constructCount_; static int destructCount_; }; // Helper to detect inconsistency between type used to instantiate an // allocator and the underlying allocator value_type. template struct check_consistent_alloc_value_type; template struct check_consistent_alloc_value_type { typedef T value_type; }; // An allocator facade that intercepts allocate/deallocate/construct/destroy // calls and track them through the tracker_allocator_counter class. This // class is templated on the target object type, but tracker isn't. template > class tracker_allocator : public Alloc { private: typedef tracker_allocator_counter counter_type; typedef __gnu_cxx::__alloc_traits AllocTraits; public: typedef typename check_consistent_alloc_value_type::value_type value_type; typedef typename AllocTraits::pointer pointer; typedef typename AllocTraits::size_type size_type; template struct rebind { typedef tracker_allocator::other> other; }; #if __cplusplus >= 201103L tracker_allocator() = default; tracker_allocator(const tracker_allocator&) = default; tracker_allocator(tracker_allocator&&) = default; tracker_allocator& operator=(const tracker_allocator&) = default; tracker_allocator& operator=(tracker_allocator&&) = default; // Perfect forwarding constructor. template tracker_allocator(_Args&&... __args) : Alloc(std::forward<_Args>(__args)...) { } #else tracker_allocator() { } tracker_allocator(const tracker_allocator&) { } ~tracker_allocator() { } #endif template tracker_allocator(const tracker_allocator::other>& alloc) _GLIBCXX_USE_NOEXCEPT : Alloc(alloc) { } pointer allocate(size_type n, const void* = 0) { pointer p = AllocTraits::allocate(*this, n); counter_type::allocate(n * sizeof(T)); return p; } #if __cplusplus >= 201103L template void construct(U* p, Args&&... args) { AllocTraits::construct(*this, p, std::forward(args)...); counter_type::construct(); } template void destroy(U* p) { AllocTraits::destroy(*this, p); counter_type::destroy(); } #else void construct(pointer p, const T& value) { AllocTraits::construct(*this, p, value); counter_type::construct(); } void destroy(pointer p) { AllocTraits::destroy(*this, p); counter_type::destroy(); } #endif void deallocate(pointer p, size_type num) { counter_type::deallocate(num * sizeof(T)); AllocTraits::deallocate(*this, p, num); } // Implement swap for underlying allocators that might need it. friend inline void swap(tracker_allocator& a, tracker_allocator& b) { using std::swap; Alloc& aa = a; Alloc& ab = b; swap(aa, ab); } }; template bool operator==(const tracker_allocator& lhs, const tracker_allocator& rhs) throw() { const Alloc1& alloc1 = lhs; const Alloc2& alloc2 = rhs; return alloc1 == alloc2; } template bool operator!=(const tracker_allocator& lhs, const tracker_allocator& rhs) throw() { return !(lhs == rhs); } bool check_construct_destroy(const char* tag, int expected_c, int expected_d); template bool check_deallocate_null() { // Let's not core here... Alloc a; a.deallocate(0, 1); a.deallocate(0, 10); return true; } template bool check_allocate_max_size() { Alloc a; try { a.allocate(a.max_size() + 1); } catch(std::bad_alloc&) { return true; } catch(...) { throw; } throw; } // A simple allocator which can be constructed endowed of a given // "personality" (an integer), queried in operator== to simulate the // behavior of realworld "unequal" allocators (i.e., not exploiting // the provision in 20.1.5/4, first bullet). A global unordered_map, // filled at allocation time with (pointer, personality) pairs, is // then consulted to enforce the requirements in Table 32 about // deallocation vs allocator equality. Note that this allocator is // swappable, not copy assignable, consistently with Option 3 of DR 431 // (see N1599). struct uneq_allocator_base { typedef std::tr1::unordered_map map_type; // Avoid static initialization troubles and/or bad interactions // with tests linking testsuite_allocator.o and playing globally // with operator new/delete. static map_type& get_map() { static map_type alloc_map; return alloc_map; } }; template > class uneq_allocator : private uneq_allocator_base, public Alloc { typedef __gnu_cxx::__alloc_traits AllocTraits; Alloc& base() { return *this; } const Alloc& base() const { return *this; } void swap_base(Alloc& b) { swap(b, this->base()); } public: typedef typename check_consistent_alloc_value_type::value_type value_type; typedef typename AllocTraits::size_type size_type; typedef typename AllocTraits::pointer pointer; #if __cplusplus >= 201103L typedef std::true_type propagate_on_container_swap; typedef std::false_type is_always_equal; #endif template struct rebind { typedef uneq_allocator::other> other; }; uneq_allocator() _GLIBCXX_USE_NOEXCEPT : personality(0) { } uneq_allocator(int person) _GLIBCXX_USE_NOEXCEPT : personality(person) { } #if __cplusplus >= 201103L uneq_allocator(const uneq_allocator&) = default; uneq_allocator(uneq_allocator&&) = default; #endif template uneq_allocator(const uneq_allocator::other>& b) _GLIBCXX_USE_NOEXCEPT : personality(b.get_personality()) { } ~uneq_allocator() _GLIBCXX_USE_NOEXCEPT { } int get_personality() const { return personality; } pointer allocate(size_type n, const void* hint = 0) { pointer p = AllocTraits::allocate(*this, n); try { get_map().insert(map_type::value_type(reinterpret_cast(p), personality)); } catch(...) { AllocTraits::deallocate(*this, p, n); __throw_exception_again; } return p; } void deallocate(pointer p, size_type n) { bool test __attribute__((unused)) = true; VERIFY( p ); map_type::iterator it = get_map().find(reinterpret_cast(p)); VERIFY( it != get_map().end() ); // Enforce requirements in Table 32 about deallocation vs // allocator equality. VERIFY( it->second == personality ); get_map().erase(it); AllocTraits::deallocate(*this, p, n); } #if __cplusplus >= 201103L // Not copy assignable... uneq_allocator& operator=(const uneq_allocator&) = delete; // ... but still moveable if base allocator is. uneq_allocator& operator=(uneq_allocator&&) = default; #else private: // Not assignable... uneq_allocator& operator=(const uneq_allocator&); #endif private: // ... yet swappable! friend inline void swap(uneq_allocator& a, uneq_allocator& b) { std::swap(a.personality, b.personality); a.swap_base(b); } template friend inline bool operator==(const uneq_allocator& a, const uneq_allocator::other>& b) { return a.personality == b.personality; } template friend inline bool operator!=(const uneq_allocator& a, const uneq_allocator::other>& b) { return !(a == b); } int personality; }; #if __cplusplus >= 201103L // An uneq_allocator which can be used to test allocator propagation. template> class propagating_allocator : public uneq_allocator { typedef __gnu_cxx::__alloc_traits AllocTraits; typedef uneq_allocator base_alloc; base_alloc& base() { return *this; } const base_alloc& base() const { return *this; } void swap_base(base_alloc& b) { swap(b, this->base()); } typedef std::integral_constant trait_type; public: // default allocator_traits::rebind_alloc would select // uneq_allocator::rebind so we must define rebind here template struct rebind { typedef propagating_allocator::other> other; }; propagating_allocator(int i) noexcept : base_alloc(i) { } template propagating_allocator(const propagating_allocator::other>& a) noexcept : base_alloc(a) { } propagating_allocator() noexcept = default; propagating_allocator(const propagating_allocator&) noexcept = default; propagating_allocator& operator=(const propagating_allocator& a) noexcept { static_assert(Propagate, "assigning propagating_allocator"); propagating_allocator(a).swap_base(*this); return *this; } template propagating_allocator& operator=(const propagating_allocator& a) noexcept { static_assert(P2, "assigning propagating_allocator"); propagating_allocator(a).swap_base(*this); return *this; } // postcondition: a.get_personality() == 0 propagating_allocator(propagating_allocator&& a) noexcept : base_alloc() { swap_base(a); } // postcondition: a.get_personality() == 0 propagating_allocator& operator=(propagating_allocator&& a) noexcept { propagating_allocator(std::move(a)).swap_base(*this); return *this; } typedef trait_type propagate_on_container_copy_assignment; typedef trait_type propagate_on_container_move_assignment; typedef trait_type propagate_on_container_swap; propagating_allocator select_on_container_copy_construction() const { return Propagate ? *this : propagating_allocator(); } }; // Class template supporting the minimal interface that satisfies the // Allocator requirements, from example in [allocator.requirements] template struct SimpleAllocator { typedef Tp value_type; SimpleAllocator() noexcept { } template SimpleAllocator(const SimpleAllocator&) { } Tp *allocate(std::size_t n) { return std::allocator().allocate(n); } void deallocate(Tp *p, std::size_t n) { std::allocator().deallocate(p, n); } }; template bool operator==(const SimpleAllocator&, const SimpleAllocator&) { return true; } template bool operator!=(const SimpleAllocator&, const SimpleAllocator&) { return false; } #endif template struct ExplicitConsAlloc : std::allocator { ExplicitConsAlloc() { } template explicit ExplicitConsAlloc(const ExplicitConsAlloc&) { } template struct rebind { typedef ExplicitConsAlloc other; }; }; #if __cplusplus >= 201103L template class CustomPointerAlloc : public std::allocator { template> using Ptr = __gnu_cxx::_Pointer_adapter; public: CustomPointerAlloc() = default; template CustomPointerAlloc(const CustomPointerAlloc&) { } template struct rebind { typedef CustomPointerAlloc other; }; typedef Ptr pointer; typedef Ptr const_pointer; typedef Ptr void_pointer; typedef Ptr const_void_pointer; pointer allocate(std::size_t n, pointer = {}) { return pointer(std::allocator::allocate(n)); } void deallocate(pointer p, std::size_t n) { std::allocator::deallocate(std::addressof(*p), n); } }; // Utility for use as CRTP base class of custom pointer types template struct PointerBase { typedef T element_type; // typedefs for iterator_traits typedef T value_type; typedef std::ptrdiff_t difference_type; typedef std::random_access_iterator_tag iterator_category; typedef Derived pointer; typedef T& reference; T* value; explicit PointerBase(T* p = nullptr) : value(p) { } template(std::declval()))> PointerBase(const PointerBase& p) : value(p.value) { } T& operator*() const { return *value; } T* operator->() const { return value; } T& operator[](difference_type n) const { return value[n]; } Derived& operator++() { ++value; return derived(); } Derived operator++(int) { Derived tmp(derived()); ++value; return tmp; } Derived& operator--() { --value; return derived(); } Derived operator--(int) { Derived tmp(derived()); --value; return tmp; } Derived& operator+=(difference_type n) { value += n; return derived(); } Derived& operator-=(difference_type n) { value -= n; return derived(); } explicit operator bool() const { return value != nullptr; } Derived operator+(difference_type n) const { Derived p(derived()); return p += n; } Derived operator-(difference_type n) const { Derived p(derived()); return p -= n; } private: Derived& derived() { return static_cast(*this); } }; template std::ptrdiff_t operator-(PointerBase l, PointerBase r) { return l.value - r.value; } template bool operator==(PointerBase l, PointerBase r) { return l.value == r.value; } template bool operator!=(PointerBase l, PointerBase r) { return l.value != r.value; } // implementation for void specializations template struct PointerBase_void { typedef T element_type; // typedefs for iterator_traits typedef T value_type; typedef std::ptrdiff_t difference_type; typedef std::random_access_iterator_tag iterator_category; T* value; explicit PointerBase_void(T* p = nullptr) : value(p) { } template(std::declval()))> PointerBase_void(const PointerBase& p) : value(p.value) { } explicit operator bool() const { return value != nullptr; } }; template struct PointerBase : PointerBase_void { using PointerBase_void::PointerBase_void; typedef Derived pointer; }; template struct PointerBase : PointerBase_void { using PointerBase_void::PointerBase_void; typedef Derived pointer; }; #endif } // namespace __gnu_test #endif // _GLIBCXX_TESTSUITE_ALLOCATOR_H