0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-12-29 08:54:02 +01:00
construct/include/ircd/buffer/unique_buffer.h
Jason Volk 30796e5729 ircd::buffer: Fix template name conflicts for clang-11; apply inline linkages.
ircd::json::tuple: Fix template name related for clang-11.

ircd::ctx: Fix template related for clang-11; inline linkages.

ircd:Ⓜ️🪝 Fix template related for clang-11.
2020-10-29 04:06:59 -07:00

129 lines
3 KiB
C++

// 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
{
std::unique_ptr<char, decltype(&std::free)>
aligned_alloc(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)
:buffer_type
{
allocator::aligned_alloc(0, ircd::buffer::size(src)).release(), ircd::buffer::size(src)
}
{
using ircd::buffer::size;
assert(this->begin() != nullptr);
assert(size(src) == size(*this));
const mutable_buffer dst
{
const_cast<char *>(this->begin()), 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 &align)
:buffer_type
{
allocator::aligned_alloc(align, size).release(), 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
{
std::free(const_cast<char *>(this->begin()));
}
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();
}