2#include <boost/container/vector.hpp>
3#include <boost/predef.h>
13#if defined(_MSC_VER) && defined(_DEBUG)
19 template <
typename... Args>
20 explicit pod_allocator(Args&&...) noexcept
24 pod_allocator() noexcept = default;
25 pod_allocator(const pod_allocator&) noexcept = default;
26 pod_allocator(pod_allocator&&) noexcept = default;
27 pod_allocator& operator=(const pod_allocator&) noexcept = default;
28 pod_allocator& operator=(pod_allocator&&) noexcept = default;
30 static inline T* allocate(std::
size_t num) noexcept {
return new T[num]; }
32 static inline void deallocate(T* p, std::size_t)
noexcept {
delete[] p; }
35 operator==(
const pod_allocator& lhs,
const pod_allocator& rhs)
noexcept
40 operator!=(
const pod_allocator& lhs,
const pod_allocator& rhs)
noexcept
46template <
class T, std::
size_t Align>
47using aligned_pod_allocator = pod_allocator<T>;
55 pod_allocator() noexcept = default;
56 pod_allocator(const pod_allocator&) noexcept = default;
57 pod_allocator(pod_allocator&&) noexcept = default;
58 pod_allocator& operator=(const pod_allocator&) noexcept = default;
59 pod_allocator& operator=(pod_allocator&&) noexcept = default;
61 static inline T* allocate(std::
size_t num) noexcept
64 std::is_standard_layout_v<T> && std::is_trivial_v<T>,
65 "can only be used with POD types");
67 alignof(T) <=
alignof(std::max_align_t),
68 "type must not have specific alignment requirements");
70 return (T*)std::malloc(
sizeof(T) * num);
73 static inline void deallocate(T* p, std::size_t)
noexcept { std::free(p); }
75 friend inline bool operator==(pod_allocator lhs, pod_allocator rhs)
noexcept
79 friend inline bool operator!=(pod_allocator lhs, pod_allocator rhs)
noexcept
85template <
class T, std::
size_t Align>
86struct aligned_pod_allocator
91 using other = aligned_pod_allocator<U, Align>;
96 aligned_pod_allocator() noexcept = default;
97 aligned_pod_allocator(const aligned_pod_allocator&) noexcept = default;
98 aligned_pod_allocator(aligned_pod_allocator&&) noexcept = default;
99 aligned_pod_allocator& operator=(const aligned_pod_allocator&) noexcept = default;
100 aligned_pod_allocator& operator=(aligned_pod_allocator&&) noexcept = default;
102#if defined(__linux__) || defined(__unix__) || defined(__unix) || defined(__APPLE__) \
103 || defined(__EMSCRIPTEN__)
104#define OSSIA_HAS_POSIX_MEMALIGN 1
107 template <
typename U>
108 static constexpr bool has_posix_memalign() noexcept
110#if defined(__linux__) || defined(__unix__) || defined(__unix) || defined(__APPLE__) \
111 || defined(__EMSCRIPTEN__)
118 static inline T* allocate(std::size_t num)
noexcept
121 std::is_standard_layout_v<T> && std::is_trivial_v<T>,
122 "can only be used with POD types");
124 alignof(T) <=
alignof(std::max_align_t),
125 "type must not have specific alignment requirements");
127 if constexpr(Align != __STDCPP_DEFAULT_NEW_ALIGNMENT__)
130 return static_cast<T*
>(::_aligned_malloc(
sizeof(T) * num, Align));
133#if OSSIA_HAS_POSIX_MEMALIGN
135 posix_memalign(&p, Align,
sizeof(T) * num);
139 static_assert(Align < 128);
140 p = std::malloc(
sizeof(T) * num + Align + 1);
141 const auto pb = uintptr_t(p);
142 auto pptr = uintptr_t(p);
144 while((pptr % Align) != 0)
152 *(
unsigned char*)(pptr - 1) = (pptr - pb);
156 return static_cast<T*
>(p);
161 return (T*)std::malloc(
sizeof(T) * num);
165 static inline void deallocate(T* p, std::size_t)
noexcept
168 if constexpr(Align != __STDCPP_DEFAULT_NEW_ALIGNMENT__)
177#if OSSIA_HAS_POSIX_MEMALIGN
183 auto count = *(((
unsigned char*)p) - 1);
184 auto root = ((
unsigned char*)p - count);
192 operator==(aligned_pod_allocator lhs, aligned_pod_allocator rhs)
noexcept
197 operator!=(aligned_pod_allocator lhs, aligned_pod_allocator rhs)
noexcept
205struct pod_allocator_avx2 : aligned_pod_allocator<T, 32>
210 using other = pod_allocator_avx2<U>;
212 using aligned_pod_allocator<T, 32>::aligned_pod_allocator;
216using pod_vector = boost::container::vector<T, pod_allocator_avx2<T>>;
218using int_vector = pod_vector<int>;
219using float_vector = pod_vector<float>;
220using double_vector = pod_vector<double>;