2022-09-06 19:14:07 -04:00
|
|
|
// Copyright 2015 The Crashpad Authors
|
2015-12-08 15:38:17 -05:00
|
|
|
//
|
|
|
|
// Licensed 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 CRASHPAD_UTIL_STDLIB_ALIGNED_ALLOCATOR_H_
|
|
|
|
#define CRASHPAD_UTIL_STDLIB_ALIGNED_ALLOCATOR_H_
|
|
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
|
|
|
|
#include <limits>
|
|
|
|
#include <memory>
|
|
|
|
#include <new>
|
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace crashpad {
|
|
|
|
|
|
|
|
//! \brief Allocates memory with the specified alignment constraint.
|
|
|
|
//!
|
|
|
|
//! This function wraps `posix_memalign()` or `_aligned_malloc()`. Memory
|
|
|
|
//! allocated by this function must be released by AlignFree().
|
|
|
|
void* AlignedAllocate(size_t alignment, size_t size);
|
|
|
|
|
|
|
|
//! \brief Frees memory allocated by AlignedAllocate().
|
|
|
|
//!
|
|
|
|
//! This function wraps `free()` or `_aligned_free()`.
|
|
|
|
void AlignedFree(void* pointer);
|
|
|
|
|
|
|
|
//! \brief A standard allocator that aligns its allocations as requested,
|
|
|
|
//! suitable for use as an allocator in standard containers.
|
|
|
|
//!
|
|
|
|
//! This is similar to `std::allocator<T>`, with the addition of an alignment
|
|
|
|
//! guarantee. \a Alignment must be a power of 2. If \a Alignment is not
|
|
|
|
//! specified, the default alignment for type \a T is used.
|
2017-06-13 13:48:07 -04:00
|
|
|
template <class T, size_t Alignment = alignof(T)>
|
2015-12-08 15:38:17 -05:00
|
|
|
struct AlignedAllocator {
|
|
|
|
public:
|
|
|
|
using value_type = T;
|
|
|
|
using pointer = T*;
|
|
|
|
using const_pointer = const T*;
|
|
|
|
using reference = T&;
|
|
|
|
using const_reference = const T&;
|
|
|
|
using size_type = size_t;
|
|
|
|
using difference_type = ptrdiff_t;
|
|
|
|
|
|
|
|
template <class U>
|
|
|
|
struct rebind {
|
|
|
|
using other = AlignedAllocator<U, Alignment>;
|
|
|
|
};
|
|
|
|
|
2017-10-13 10:35:45 -04:00
|
|
|
AlignedAllocator() noexcept {}
|
|
|
|
AlignedAllocator(const AlignedAllocator& other) noexcept {}
|
2015-12-08 15:38:17 -05:00
|
|
|
|
|
|
|
template <typename U>
|
2017-10-13 10:35:45 -04:00
|
|
|
AlignedAllocator(const AlignedAllocator<U, Alignment>& other) noexcept {}
|
2015-12-08 15:38:17 -05:00
|
|
|
|
|
|
|
~AlignedAllocator() {}
|
|
|
|
|
2017-10-13 10:35:45 -04:00
|
|
|
pointer address(reference x) const noexcept { return &x; }
|
|
|
|
const_pointer address(const_reference x) const noexcept { return &x; }
|
2015-12-08 15:38:17 -05:00
|
|
|
|
2020-12-08 07:23:47 -08:00
|
|
|
pointer allocate(size_type n, const void* hint = 0) {
|
2015-12-08 15:38:17 -05:00
|
|
|
return reinterpret_cast<pointer>(
|
2017-10-31 22:25:34 -04:00
|
|
|
AlignedAllocate(Alignment, sizeof(value_type) * n));
|
2015-12-08 15:38:17 -05:00
|
|
|
}
|
|
|
|
|
2017-10-31 22:25:34 -04:00
|
|
|
void deallocate(pointer p, size_type n) { AlignedFree(p); }
|
2015-12-08 15:38:17 -05:00
|
|
|
|
2017-10-13 10:35:45 -04:00
|
|
|
size_type max_size() const noexcept {
|
2015-12-08 15:38:17 -05:00
|
|
|
return std::numeric_limits<size_type>::max() / sizeof(value_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class U, class... Args>
|
|
|
|
void construct(U* p, Args&&... args) {
|
|
|
|
new (reinterpret_cast<void*>(p)) U(std::forward<Args>(args)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class U>
|
|
|
|
void destroy(U* p) {
|
|
|
|
p->~U();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class T1, class T2, size_t Alignment>
|
|
|
|
bool operator==(const AlignedAllocator<T1, Alignment>& lhs,
|
2017-10-13 10:35:45 -04:00
|
|
|
const AlignedAllocator<T2, Alignment>& rhs) noexcept {
|
2015-12-08 15:38:17 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T1, class T2, size_t Alignment>
|
|
|
|
bool operator!=(const AlignedAllocator<T1, Alignment>& lhs,
|
2017-10-13 10:35:45 -04:00
|
|
|
const AlignedAllocator<T2, Alignment>& rhs) noexcept {
|
2015-12-08 15:38:17 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! \brief A `std::vector` using AlignedAllocator.
|
|
|
|
//!
|
|
|
|
//! This is similar to `std::vector<T>`, with the addition of an alignment
|
|
|
|
//! guarantee. \a Alignment must be a power of 2. If \a Alignment is not
|
|
|
|
//! specified, the default alignment for type \a T is used.
|
2017-06-13 13:48:07 -04:00
|
|
|
template <typename T, size_t Alignment = alignof(T)>
|
2015-12-08 15:38:17 -05:00
|
|
|
using AlignedVector = std::vector<T, AlignedAllocator<T, Alignment>>;
|
|
|
|
|
|
|
|
} // namespace crashpad
|
|
|
|
|
|
|
|
#endif // CRASHPAD_UTIL_STDLIB_ALIGNED_ALLOCATOR_H_
|