/** @file IAllocator_Xfer.hpp * * @author Roland Conybeare, Dec 2025 **/ #pragma once #include "AAllocator.hpp" namespace xo { namespace mm { /** @class IAllocator_Xfer * * Adapts typed allocator implementation @tparam IAllocator_DRepr * to type-erased @ref AAllocator interface **/ template struct IAllocator_Xfer : public AAllocator { // parallel interface to AAllocator, with specific data type using Impl = IAllocator_DRepr; using size_type = std::size_t; static const DRepr & _dcast(Copaque d) { return *(const DRepr *)d; } static DRepr & _dcast(Opaque d) { return *(DRepr *)d; } // from AAllocator int32_t _typeseq() const noexcept override { return s_typeseq; } const std::string & name(Copaque d) const noexcept override { return Impl::name(_dcast(d)); } size_type reserved(Copaque d) const noexcept override { return Impl::reserved(_dcast(d)); } size_type size(Copaque d) const noexcept override { return Impl::size(_dcast(d)); } size_type committed(Copaque d) const noexcept override { return Impl::committed(_dcast(d)); } size_type available(Copaque d) const noexcept override { return I::available(_dcast(d)); } size_type allocated(Copaque d) const noexcept override { return I::allocated(_dcast(d)); } bool contains(Copaque d, const void * p) const noexcept override { return Impl::contains(_dcast(d), p); } bool expand(Opaque d, std::size_t z) const noexcept override { return Impl::expand(_dcast(d), z); } std::byte * alloc(Opaque d, std::size_t z) const override { return Impl::alloc(*(DRepr*)d, z); } void clear(Opaque d) const override { return Impl::clear(*(DRepr*)d); } void destruct_data(Opaque d) const override { return Impl::destruct_data(*(DRepr*)d); } private: using I = Impl; public: static int32_t s_typeseq; static bool _valid; }; template int32_t IAllocator_Xfer::s_typeseq = facet::typeseq::id(); template bool IAllocator_Xfer::_valid = facet::valid_facet_implementation(); } /*namespace mm*/ } /*namespace xo*/ /* end IAllocator_Xfer.hpp */