/* * Copyright (C) 2016 Charybdis Development Team * Copyright (C) 2016 Jason Volk * * 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 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 JS::Handle operator&(const JS::Heap &h) { return JS::Handle::fromMarkedLocation(h.address()); } // This conversion is missing in the jsapi. Is this object not gc rooted? Can it not have a handle? template JS::MutableHandle operator&(JS::Heap &h) { const auto ptr(const_cast(h.address())); return JS::MutableHandle::fromMarkedLocation(ptr); } // This conversion is missing in the jsapi. Is this object not gc rooted? Can it not have a handle? template JS::Handle operator&(const JS::TenuredHeap &h) { return JS::Handle::fromMarkedLocation(h.address()); } // This conversion is missing in the jsapi. Is this object not gc rooted? Can it not have a handle? template JS::MutableHandle operator&(JS::TenuredHeap &h) { const auto ptr(const_cast(h.address())); return JS::MutableHandle::fromMarkedLocation(ptr); } template struct root :JS::Heap { using value_type = T; using base_type = JS::Heap; using handle = JS::Handle; using handle_mutable = JS::MutableHandle; using type = root; operator handle() const { const auto ptr(this->address()); return JS::Handle::fromMarkedLocation(ptr); } operator handle_mutable() { const auto ptr(this->address()); return JS::MutableHandle::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() }); } 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(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