mirror of
https://github.com/mutouyun/cpp-ipc.git
synced 2025-12-06 16:56:45 +08:00
214 lines
5.4 KiB
C++
214 lines
5.4 KiB
C++
#pragma once
|
|
|
|
#include <limits>
|
|
#include <new>
|
|
#include <tuple>
|
|
#include <map>
|
|
#include <vector>
|
|
#include <functional>
|
|
#include <utility>
|
|
#include <cstddef>
|
|
#include <type_traits>
|
|
|
|
#include "def.h"
|
|
#include "rw_lock.h"
|
|
#include "tls_pointer.h"
|
|
|
|
#include "memory/alloc.h"
|
|
#include "memory/detail.h"
|
|
#include "platform/detail.h"
|
|
|
|
namespace ipc {
|
|
namespace mem {
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
/// The allocator wrapper class for STL
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
template <typename T, typename AllocP>
|
|
class allocator_wrapper {
|
|
|
|
template <typename U, typename AllocU>
|
|
friend class allocator_wrapper;
|
|
|
|
public:
|
|
// type definitions
|
|
typedef T value_type;
|
|
typedef value_type* pointer;
|
|
typedef const value_type* const_pointer;
|
|
typedef value_type& reference;
|
|
typedef const value_type& const_reference;
|
|
typedef std::size_t size_type;
|
|
typedef AllocP alloc_policy;
|
|
|
|
private:
|
|
alloc_policy alloc_;
|
|
|
|
public:
|
|
allocator_wrapper(void) noexcept = default;
|
|
|
|
allocator_wrapper(const allocator_wrapper<T, AllocP>& rhs) noexcept
|
|
: alloc_(rhs.alloc_)
|
|
{}
|
|
|
|
template <typename U>
|
|
allocator_wrapper(const allocator_wrapper<U, AllocP>& rhs) noexcept
|
|
: alloc_(rhs.alloc_)
|
|
{}
|
|
|
|
allocator_wrapper(allocator_wrapper<T, AllocP>&& rhs) noexcept
|
|
: alloc_(std::move(rhs.alloc_))
|
|
{}
|
|
|
|
template <typename U>
|
|
allocator_wrapper(allocator_wrapper<U, AllocP>&& rhs) noexcept
|
|
: alloc_(std::move(rhs.alloc_))
|
|
{}
|
|
|
|
allocator_wrapper(const AllocP& rhs) noexcept
|
|
: alloc_(rhs)
|
|
{}
|
|
|
|
allocator_wrapper(AllocP&& rhs) noexcept
|
|
: alloc_(std::move(rhs))
|
|
{}
|
|
|
|
public:
|
|
// the other type of std_allocator
|
|
template <typename U>
|
|
struct rebind { typedef allocator_wrapper<U, AllocP> other; };
|
|
|
|
constexpr size_type max_size(void) const noexcept {
|
|
return (std::numeric_limits<size_type>::max)() / sizeof(T);
|
|
}
|
|
|
|
public:
|
|
pointer allocate(size_type count) noexcept {
|
|
if (count == 0) return nullptr;
|
|
if (count > this->max_size()) return nullptr;
|
|
return static_cast<pointer>(alloc_.alloc(count * sizeof(T)));
|
|
}
|
|
|
|
void deallocate(pointer p, size_type count) noexcept {
|
|
alloc_.free(p, count * sizeof(T));
|
|
}
|
|
|
|
template <typename... P>
|
|
static void construct(pointer p, P&&... params) {
|
|
::new (static_cast<void*>(p)) T(std::forward<P>(params)...);
|
|
}
|
|
|
|
static void destroy(pointer p) {
|
|
p->~T();
|
|
}
|
|
};
|
|
|
|
template <class AllocP>
|
|
class allocator_wrapper<void, AllocP> {
|
|
public:
|
|
typedef void value_type;
|
|
typedef AllocP alloc_policy;
|
|
};
|
|
|
|
template <typename T, typename U, class AllocP>
|
|
constexpr bool operator==(const allocator_wrapper<T, AllocP>&, const allocator_wrapper<U, AllocP>&) noexcept {
|
|
return true;
|
|
}
|
|
|
|
template <typename T, typename U, class AllocP>
|
|
constexpr bool operator!=(const allocator_wrapper<T, AllocP>&, const allocator_wrapper<U, AllocP>&) noexcept {
|
|
return false;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
/// Thread-safe allocation wrapper
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
template <std::size_t BlockSize>
|
|
using page_fixed = ipc::mem::detail::fixed<BlockSize, page_fixed_alloc>;
|
|
|
|
using page_pool_alloc = detail::pool_alloc<page_fixed>;
|
|
|
|
template <typename T>
|
|
using page_allocator = allocator_wrapper<T, page_pool_alloc>;
|
|
|
|
template <typename AllocP>
|
|
class synchronized {
|
|
public:
|
|
using alloc_policy = AllocP;
|
|
|
|
private:
|
|
spin_lock lc_;
|
|
std::multimap<std::size_t, alloc_policy*, std::less<std::size_t>,
|
|
page_allocator<std::pair<const std::size_t, alloc_policy*>>> allocs_;
|
|
|
|
struct alloc_t {
|
|
synchronized* t_;
|
|
std::size_t s_ = 0;
|
|
alloc_policy* a_ = nullptr;
|
|
|
|
alloc_t(synchronized* t)
|
|
: t_ { t } {
|
|
{
|
|
IPC_UNUSED_ auto guard = ipc::detail::unique_lock(t_->lc_);
|
|
auto it = t_->allocs_.begin();
|
|
if (it != t_->allocs_.end()) {
|
|
std::tie(s_, a_) = *it;
|
|
t_->allocs_.erase(it);
|
|
}
|
|
if (a_ == nullptr) {
|
|
a_ = static_cast<alloc_policy*>(page_pool_alloc::alloc(sizeof(alloc_policy)));
|
|
}
|
|
}
|
|
::new (a_) alloc_policy;
|
|
}
|
|
|
|
~alloc_t() {
|
|
IPC_UNUSED_ auto guard = ipc::detail::unique_lock(t_->lc_);
|
|
t_->allocs_.emplace(s_, a_);
|
|
}
|
|
|
|
void* alloc(std::size_t size) {
|
|
void* p = a_->alloc(size);
|
|
if ((p != nullptr) && (s_ > 0)) {
|
|
--s_;
|
|
}
|
|
return p;
|
|
}
|
|
|
|
void free(void* p) {
|
|
a_->free(p);
|
|
++s_;
|
|
}
|
|
};
|
|
|
|
auto& alc_info() {
|
|
static tls::pointer<alloc_t> alc;
|
|
return *alc.create(this);
|
|
}
|
|
|
|
public:
|
|
void clear() {
|
|
IPC_UNUSED_ auto guard = ipc::detail::unique_lock(lc_);
|
|
for (auto& pair : allocs_) {
|
|
pair.second->~AllocP();
|
|
page_pool_alloc::free(pair.second, sizeof(alloc_policy));
|
|
}
|
|
}
|
|
|
|
void* alloc(std::size_t size) {
|
|
return alc_info().alloc(size);
|
|
}
|
|
|
|
void free(void* p) {
|
|
alc_info().free(p);
|
|
}
|
|
|
|
void free(void* p, std::size_t /*size*/) {
|
|
free(p);
|
|
}
|
|
};
|
|
|
|
} // namespace mem
|
|
} // namespace ipc
|