0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-11-16 15:00:51 +01:00
construct/include/ircd/js/root.h

162 lines
3.8 KiB
C++

/*
* Copyright (C) 2016 Charybdis Development Team
* Copyright (C) 2016 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.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#pragma once
#define HAVE_IRCD_JS_ROOT_H
namespace ircd {
namespace js {
template<class T>
using handle = typename T::handle;
// This conversion is missing in the jsapi. Is this object not gc rooted? Can it not have a handle?
template<class T>
JS::Handle<T>
operator&(const JS::Heap<T> &h)
{
return JS::Handle<T>::fromMarkedLocation(h.address());
}
// This conversion is missing in the jsapi. Is this object not gc rooted? Can it not have a handle?
template<class T>
JS::MutableHandle<T>
operator&(JS::Heap<T> &h)
{
const auto ptr(const_cast<T *>(h.address()));
return JS::MutableHandle<T>::fromMarkedLocation(ptr);
}
// This conversion is missing in the jsapi. Is this object not gc rooted? Can it not have a handle?
template<class T>
JS::Handle<T>
operator&(const JS::TenuredHeap<T> &h)
{
return JS::Handle<T>::fromMarkedLocation(h.address());
}
// This conversion is missing in the jsapi. Is this object not gc rooted? Can it not have a handle?
template<class T>
JS::MutableHandle<T>
operator&(JS::TenuredHeap<T> &h)
{
const auto ptr(const_cast<T *>(h.address()));
return JS::MutableHandle<T>::fromMarkedLocation(ptr);
}
template<class T>
struct root
:JS::Heap<T>
{
using value_type = T;
using base_type = JS::Heap<T>;
using handle = JS::Handle<T>;
using handle_mutable = JS::MutableHandle<T>;
using type = root<value_type>;
operator handle() const
{
const auto ptr(this->address());
return JS::Handle<T>::fromMarkedLocation(ptr);
}
operator handle_mutable()
{
const auto ptr(this->address());
return JS::MutableHandle<T>::fromMarkedLocation(ptr);
}
private:
tracing::list::iterator tracing_it;
tracing::list::iterator add(base_type *const &ptr)
{
return cx->tracing.heap.emplace(end(cx->tracing.heap), tracing::thing { ptr, js::type<T>() });
}
void del(const tracing::list::iterator &tracing_it)
{
if(tracing_it != end(cx->tracing.heap))
{
const void *ptr = tracing_it->ptr;
cx->tracing.heap.erase(tracing_it);
}
}
public:
root(const handle &h)
:base_type{h.get()}
,tracing_it{add(this)}
{
}
root(const handle_mutable &h)
:base_type{h.get()}
,tracing_it{add(this)}
{
}
explicit root(const T &t)
:base_type{t}
,tracing_it{add(this)}
{
}
root()
:base_type{}
,tracing_it{add(this)}
{
}
root(root &&other) noexcept
:base_type{other.get()}
,tracing_it{std::move(other.tracing_it)}
{
other.tracing_it = end(cx->tracing.heap);
tracing_it->ptr = static_cast<base_type *>(this);
}
root(const root &other)
:base_type{other.get()}
,tracing_it{add(this)}
{
}
root &operator=(root &&other) noexcept
{
base_type::operator=(std::move(other.get()));
return *this;
}
root &operator=(const root &other)
{
base_type::operator=(other.get());
return *this;
}
~root() noexcept
{
del(tracing_it);
}
};
} // namespace js
} // namespace ircd