19#include "lib/stdcompat/bit.h"
20#include "pw_allocator/block/allocatable.h"
21#include "pw_allocator/block/result.h"
22#include "pw_allocator/hardening.h"
23#include "pw_allocator/layout.h"
24#include "pw_bytes/alignment.h"
25#include "pw_status/status.h"
27namespace pw::allocator {
43template <
typename Derived>
47 static_assert(is_allocatable_v<Derived>,
48 "Types derived from AlignableBlock must also derive from "
64 constexpr const Derived* derived()
const {
65 return static_cast<const Derived*
>(
this);
71 size_t new_inner_size);
79template <
typename BlockType>
80struct is_alignable : std::is_base_of<internal::AlignableBase, BlockType> {};
83template <
typename BlockType>
90template <
typename Derived>
94 auto result = derived()->AllocatableBlock<Derived>::DoCanAlloc(layout);
98 size_t extra = result.
size();
101 if (layout.alignment() <= Derived::kAlignment) {
106 auto addr = cpp20::bit_cast<uintptr_t>(derived()->UsableSpace());
107 uintptr_t aligned_addr = addr;
108 Hardening::Increment(aligned_addr, extra);
109 aligned_addr =
AlignDown(aligned_addr, layout.alignment());
112 if (aligned_addr < addr) {
113 return StatusWithSize::ResourceExhausted();
117 size_t leading_outer_size = aligned_addr - addr;
118 if (leading_outer_size != 0 && leading_outer_size < Derived::kMinOuterSize &&
119 derived()->Prev() ==
nullptr) {
120 return StatusWithSize::ResourceExhausted();
126template <
typename Derived>
128 Derived*&& block,
Layout layout) {
130 if (layout.alignment() <= Derived::kAlignment) {
135 size_t size =
AlignUp(layout.size(), Derived::kAlignment);
136 layout =
Layout(size, layout.alignment());
138 if (!can_alloc.
ok()) {
141 size_t extra = can_alloc.
size();
142 size_t leading_outer_size = extra -
AlignDown(extra, layout.alignment());
145 if (leading_outer_size != 0 && leading_outer_size <= Derived::kMinOuterSize &&
146 block->Prev() ==
nullptr) {
147 leading_outer_size +=
AlignUp(Derived::kMinOuterSize - leading_outer_size,
150 if (leading_outer_size > extra) {
155 return DoAllocAligned(std::move(block), leading_outer_size, layout.size());
158template <
typename Derived>
160 Derived*&& block,
Layout layout) {
162 if (layout.alignment() <= Derived::kAlignment) {
167 size_t size =
AlignUp(layout.size(), Derived::kAlignment);
168 layout =
Layout(size, layout.alignment());
170 if (!can_alloc.
ok()) {
173 size_t leading_outer_size = can_alloc.
size();
176 return DoAllocAligned(std::move(block), leading_outer_size, layout.size());
179template <
typename Derived>
181 Derived*&& block,
size_t leading_outer_size,
size_t new_inner_size) {
183 Layout layout(block->InnerSize() - leading_outer_size, Derived::kAlignment);
186 if (!alloc_result.ok()) {
189 block = alloc_result.block();
193 block->AllocatableBlock<Derived>::DoResize(new_inner_size);
194 if (!resize_result.ok()) {
195 return resize_result;
198 return BlockResult<Derived>(
199 block, alloc_result.prev(), resize_result.next(), alloc_result.size());
static constexpr Status ResourceExhausted()
Definition: status.h:230
Definition: status_with_size.h:51
constexpr bool ok() const
True if status() == OkStatus().
Definition: status_with_size.h:154
constexpr size_t size() const
Definition: status_with_size.h:148
Definition: alignable.h:44
constexpr StatusWithSize DoCanAlloc(Layout layout) const
Definition: alignable.h:91
static constexpr BlockResult< Derived > DoAllocFirst(Derived *&&block, Layout layout)
Definition: alignable.h:127
static constexpr BlockResult< Derived > DoAllocLast(Derived *&&block, Layout layout)
Definition: alignable.h:159
static constexpr BlockResult< Derived > DoAllocLast(Derived *&&block, Layout layout)
Definition: allocatable.h:301
static constexpr BlockResult< Derived > DoAllocFirst(Derived *&&block, Layout layout)
Definition: allocatable.h:266
Definition: with_layout.h:50
constexpr bool is_alignable_v
Helper variable template for is_alignable<BlockType>::value.
Definition: alignable.h:84
constexpr size_t AlignUp(uintptr_t value, size_t alignment)
Returns the value rounded up to the nearest multiple of alignment.
Definition: alignment.h:54
constexpr size_t AlignDown(uintptr_t value, size_t alignment)
Returns the value rounded down to the nearest multiple of alignment.
Definition: alignment.h:42
Definition: alignable.h:31
Trait type that allows interrogating a block as to whether it is alignable.
Definition: alignable.h:80