Files
meta.hpp/headers/meta.hpp/meta_base/memory_buffer.hpp
2023-02-02 14:40:02 +07:00

127 lines
3.8 KiB
C++

/*******************************************************************************
* This file is part of the "https://github.com/blackmatov/meta.hpp"
* For conditions of distribution and use, see copyright notice in LICENSE.md
* Copyright (C) 2021-2023, by Matvey Cherevko (blackmatov@gmail.com)
******************************************************************************/
#pragma once
#include "base.hpp"
#include "fnv1a_hash.hpp"
namespace meta_hpp::detail
{
class memory_buffer final {
public:
memory_buffer() = default;
memory_buffer(const memory_buffer&) = delete;
memory_buffer& operator=(const memory_buffer&) = delete;
memory_buffer(memory_buffer&& other) noexcept
: data_{other.data_}
, size_{other.size_}
, align_{other.align_} {
other.data_ = nullptr;
other.size_ = 0;
other.align_ = std::align_val_t{};
}
memory_buffer& operator=(memory_buffer&& other) noexcept {
if ( this != &other ) {
memory_buffer{std::move(other)}.swap(*this);
}
return *this;
}
explicit memory_buffer(std::size_t size, std::align_val_t align)
: data_{::operator new(size, align)}
, size_{size}
, align_{align} {}
explicit memory_buffer(const void* mem, std::size_t size, std::align_val_t align)
: memory_buffer{size, align} {
if ( mem != nullptr && size > 0 ) {
std::memcpy(data_, mem, size);
}
}
~memory_buffer() noexcept {
reset();
}
[[nodiscard]] bool is_valid() const noexcept {
return data_ != nullptr;
}
[[nodiscard]] explicit operator bool() const noexcept {
return is_valid();
}
void reset() noexcept {
if ( data_ != nullptr ) {
::operator delete(data_, align_);
data_ = nullptr;
size_ = 0;
align_ = std::align_val_t{};
}
}
void swap(memory_buffer& other) noexcept {
std::swap(data_, other.data_);
std::swap(size_, other.size_);
std::swap(align_, other.align_);
}
[[nodiscard]] int compare(const memory_buffer& other) const noexcept {
if ( size_ < other.size_ ) {
return -1;
}
if ( size_ > other.size_ ) {
return +1;
}
return size_ != 0
? std::memcmp(data_, other.data_, size_)
: 0;
}
[[nodiscard]] void* get_data() noexcept {
return data_;
}
[[nodiscard]] const void* get_data() const noexcept {
return data_;
}
[[nodiscard]] std::size_t get_size() const noexcept {
return size_;
}
[[nodiscard]] std::align_val_t get_align() const noexcept {
return align_;
}
private:
void* data_{};
std::size_t size_{};
std::align_val_t align_{};
};
inline void swap(memory_buffer& l, memory_buffer& r) noexcept { l.swap(r); }
[[nodiscard]] inline bool operator<(const memory_buffer& l, const memory_buffer& r) noexcept { return l.compare(r) < 0; }
[[nodiscard]] inline bool operator==(const memory_buffer& l, const memory_buffer& r) noexcept { return l.compare(r) == 0; }
[[nodiscard]] inline bool operator!=(const memory_buffer& l, const memory_buffer& r) noexcept { return l.compare(r) != 0; }
}
namespace std
{
template <>
struct hash<meta_hpp::detail::memory_buffer> {
size_t operator()(const meta_hpp::detail::memory_buffer& mb) const noexcept {
return meta_hpp::detail::fnv1a_hash(mb.get_data(), mb.get_size());
}
};
}