#include "alloc/raw/aligned/foreign.hpp" #include "alloc/raw/linear.hpp" #include "pointer.hpp" #include "tap.hpp" int main () { static std::byte buffer[1024*1024]; static constexpr std::size_t alignment = 3; static constexpr std::size_t increment = 1; // ensure we have an base pointer that's off-by-one to a likely natural // system alignment std::byte* base = cruft::align ( std::data (buffer), alignof (std::max_align_t) ) + increment; cruft::alloc::raw::aligned::foreign<cruft::alloc::raw::linear> alloc ( cruft::view(base,std::end(buffer)), alignment ); cruft::TAP::logger tap; // ensure the first element allocated falls at the base address tap.expect_eq (base, alloc.data (), "allocator base address is the supplied base address"); tap.expect_eq (base, alloc.allocate<std::byte> (8).data (), "first allocation is the supplied base address"); // allocate a range of values and make sure they all satisfy our alignment. // don't choose values which are likely to combine with the testing // alignment to produce a likely system alignment. eg, 3 + 5 == 8 which is // a power-of-2. static const struct { size_t size; const char *message; } TESTS[] = { { 9, "just over a power of two" }, { 1, "a single byte" }, { 64, "a cache line" }, { 250, "multiple cache lines, but not a power of two" }, }; for (const auto &t: TESTS) { auto ptr = reinterpret_cast<uintptr_t> (alloc.allocate<char> (t.size).data ()); auto offset = ptr - reinterpret_cast<uintptr_t> (base); tap.expect_mod (offset, alignment, "%s", t.message); } return tap.status (); };