// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk <jason@zemos.net> // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #pragma once #define HAVE_IRCD_BUFFER_UNIQUE_BUFFER_H // Fwd decl; circ dep. namespace ircd::allocator { [[using gnu: malloc, alloc_align(1), alloc_size(2), returns_nonnull, warn_unused_result]] char *allocate(const size_t align, const size_t size); } /// Like unique_ptr, this template holds ownership of an allocated buffer /// template<class buffer_type> struct ircd::buffer::unique_buffer :buffer_type { explicit operator bool() const; bool operator!() const; buffer_type release(); unique_buffer() = default; unique_buffer(const size_t &size, const size_t &align = 0); explicit unique_buffer(const const_buffer &); unique_buffer(unique_buffer &&) noexcept; unique_buffer(const unique_buffer &) = delete; unique_buffer &operator=(unique_buffer &&) & noexcept; unique_buffer &operator=(const unique_buffer &) = delete; ~unique_buffer() noexcept; }; template<class buffer_type> inline ircd::buffer::unique_buffer<buffer_type>::unique_buffer(const const_buffer &src) :unique_buffer { ircd::buffer::size(src) } { using ircd::buffer::size; assert(this->begin() != nullptr); assert(size(src) == size(*this)); const auto ptr(std::get<0>(*this)); const mutable_buffer dst { const_cast<char *>(ptr), size(src) }; copy(dst, src); } template<class buffer_type> inline ircd::buffer::unique_buffer<buffer_type>::unique_buffer(const size_t &size, const size_t &a) :buffer_type { allocator::allocate(a?: sizeof(void *), pad_to(size, a?: sizeof(void *))), size } {} template<class buffer_type> inline ircd::buffer::unique_buffer<buffer_type>::unique_buffer(unique_buffer &&other) noexcept :buffer_type { other.release() } { assert(std::get<0>(other) == nullptr); } template<class buffer_type> inline ircd::buffer::unique_buffer<buffer_type> & ircd::buffer::unique_buffer<buffer_type>::operator=(unique_buffer &&other) & noexcept { this->~unique_buffer(); static_cast<buffer_type &>(*this) = other.release(); assert(std::get<0>(other) == nullptr); return *this; } template<class buffer_type> inline ircd::buffer::unique_buffer<buffer_type>::~unique_buffer() noexcept { const auto ptr(std::get<0>(*this)); std::free(const_cast<char *>(ptr)); } template<class buffer_type> inline buffer_type ircd::buffer::unique_buffer<buffer_type>::release() { const buffer_type ret{*this}; std::get<0>(*this) = nullptr; std::get<1>(*this) = nullptr; return ret; } template<class buffer_type> inline bool ircd::buffer::unique_buffer<buffer_type>::operator!() const { return this->buffer_type::empty(); } template<class buffer_type> inline ircd::buffer::unique_buffer<buffer_type>::operator bool() const { return !this->buffer_type::empty(); }