/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ #ifndef TEST_ALLOCATOR_HPP #define TEST_ALLOCATOR_HPP #include #include #include // this allocator keeps the total allocated size in a global variable for testing namespace datasketches { extern long long test_allocator_total_bytes; extern long long test_allocator_net_allocations; template class test_allocator { public: using value_type = T; using pointer = value_type*; using const_pointer = const value_type*; using reference = value_type&; using const_reference = const value_type&; using size_type = std::size_t; using difference_type = std::ptrdiff_t; template struct rebind { using other = test_allocator; }; // this is to test that a given instance of an allocator is used instead of instantiating static const bool DISALLOW_DEFAULT_CONSTRUCTOR = true; test_allocator() { if (DISALLOW_DEFAULT_CONSTRUCTOR) throw std::runtime_error("test_allocator: default constructor"); } // call this constructor in tests and pass an allocator instance test_allocator(int) {} test_allocator(const test_allocator&) {} template test_allocator(const test_allocator&) {} test_allocator(test_allocator&&) {} ~test_allocator() {} test_allocator& operator=(const test_allocator&) { return *this; } test_allocator& operator=(test_allocator&&) { return *this; } pointer address(reference x) const { return &x; } const_pointer address(const_reference x) const { return x; } pointer allocate(size_type n, const_pointer = 0) { void* p = ::operator new(n * sizeof(value_type)); if (!p) throw std::bad_alloc(); test_allocator_total_bytes += n * sizeof(value_type); ++test_allocator_net_allocations; return static_cast(p); } void deallocate(pointer p, size_type n) { if (p) ::operator delete(p); test_allocator_total_bytes -= n * sizeof(value_type); --test_allocator_net_allocations; } size_type max_size() const { return static_cast(-1) / sizeof(value_type); } template void construct(pointer p, Args&&... args) { new(p) value_type(std::forward(args)...); } void destroy(pointer p) { p->~value_type(); } }; template<> class test_allocator { public: using value_type = void; using pointer = void*; using const_pointer = const void*; template struct rebind { using other = test_allocator; }; }; template inline bool operator==(const test_allocator&, const test_allocator&) { return true; } template inline bool operator!=(const test_allocator&, const test_allocator&) { return false; } } /* namespace datasketches */ #endif