blob: b2863e79b76d315d08ba99a88878f828cc9c0fcd [file] [log] [blame]
/*
* 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 <new>
#include <utility>
#include <stdexcept>
// 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 T> 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 <class U>
struct rebind { using other = test_allocator<U>; };
// 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 <class U>
test_allocator(const test_allocator<U>&) {}
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<pointer>(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<size_type>(-1) / sizeof(value_type);
}
template<typename... Args>
void construct(pointer p, Args&&... args) {
new(p) value_type(std::forward<Args>(args)...);
}
void destroy(pointer p) { p->~value_type(); }
};
template<> class test_allocator<void> {
public:
using value_type = void;
using pointer = void*;
using const_pointer = const void*;
template <class U>
struct rebind { using other = test_allocator<U>; };
};
template <class T>
inline bool operator==(const test_allocator<T>&, const test_allocator<T>&) {
return true;
}
template <class T>
inline bool operator!=(const test_allocator<T>&, const test_allocator<T>&) {
return false;
}
} /* namespace datasketches */
#endif