mirror of
https://github.com/KingDuckZ/kamokan.git
synced 2025-02-09 09:23:56 +00:00
IncRedis is now held by Response, if requested by the base class. Response objects know their names, and they use it to load the html (soon to be mustache) data from disk. Main only prepares a factory, the factory then instantiates the actual Response. The code now correctly serves index.cgi when the request is /. Remove kakoune's safe_ptr from inside src and put an updated one in lib/kakoune.
115 lines
2.5 KiB
C++
115 lines
2.5 KiB
C++
#ifndef ref_ptr_hh_INCLUDED
|
|
#define ref_ptr_hh_INCLUDED
|
|
|
|
#include <utility>
|
|
|
|
namespace Kakoune
|
|
{
|
|
|
|
struct RefCountable
|
|
{
|
|
int refcount = 0;
|
|
virtual ~RefCountable() = default;
|
|
};
|
|
|
|
struct RefCountablePolicy
|
|
{
|
|
static void inc_ref(RefCountable* r, void*) noexcept { ++r->refcount; }
|
|
static void dec_ref(RefCountable* r, void*) { if (--r->refcount == 0) delete r; }
|
|
static void ptr_moved(RefCountable*, void*, void*) noexcept {}
|
|
};
|
|
|
|
template<typename T, typename Policy = RefCountablePolicy>
|
|
struct RefPtr
|
|
{
|
|
RefPtr() = default;
|
|
explicit RefPtr(T* ptr) : m_ptr(ptr) { acquire(); }
|
|
~RefPtr() { release(); }
|
|
RefPtr(const RefPtr& other) : m_ptr(other.m_ptr) { acquire(); }
|
|
RefPtr(RefPtr&& other)
|
|
noexcept(noexcept(std::declval<RefPtr>().moved(nullptr)))
|
|
: m_ptr(other.m_ptr) { other.m_ptr = nullptr; moved(&other); }
|
|
|
|
RefPtr& operator=(const RefPtr& other)
|
|
{
|
|
if (other.m_ptr != m_ptr)
|
|
{
|
|
release();
|
|
m_ptr = other.m_ptr;
|
|
acquire();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
RefPtr& operator=(RefPtr&& other)
|
|
{
|
|
release();
|
|
m_ptr = other.m_ptr;
|
|
other.m_ptr = nullptr;
|
|
moved(&other);
|
|
return *this;
|
|
}
|
|
|
|
RefPtr& operator=(T* ptr)
|
|
{
|
|
if (ptr != m_ptr)
|
|
{
|
|
release();
|
|
m_ptr = ptr;
|
|
acquire();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
[[gnu::always_inline]]
|
|
T* operator->() const { return m_ptr; }
|
|
[[gnu::always_inline]]
|
|
T& operator*() const { return *m_ptr; }
|
|
|
|
[[gnu::always_inline]]
|
|
T* get() const { return m_ptr; }
|
|
|
|
[[gnu::always_inline]]
|
|
explicit operator bool() const { return m_ptr; }
|
|
|
|
void reset(T* ptr = nullptr)
|
|
{
|
|
if (ptr == m_ptr)
|
|
return;
|
|
release();
|
|
m_ptr = ptr;
|
|
acquire();
|
|
}
|
|
|
|
friend bool operator==(const RefPtr& lhs, const RefPtr& rhs) { return lhs.m_ptr == rhs.m_ptr; }
|
|
friend bool operator!=(const RefPtr& lhs, const RefPtr& rhs) { return lhs.m_ptr != rhs.m_ptr; }
|
|
|
|
private:
|
|
T* m_ptr = nullptr;
|
|
|
|
[[gnu::always_inline]]
|
|
void acquire()
|
|
{
|
|
if (m_ptr)
|
|
Policy::inc_ref(m_ptr, this);
|
|
}
|
|
|
|
[[gnu::always_inline]]
|
|
void release()
|
|
{
|
|
if (m_ptr)
|
|
Policy::dec_ref(m_ptr, this);
|
|
}
|
|
|
|
[[gnu::always_inline]]
|
|
void moved(void* from)
|
|
noexcept(noexcept(Policy::ptr_moved(nullptr, nullptr, nullptr)))
|
|
{
|
|
if (m_ptr)
|
|
Policy::ptr_moved(m_ptr, from, this);
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
#endif // ref_ptr_hh_INCLUDED
|