0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-16 17:46:54 +01:00
construct/include/ircd/js/trap.h

126 lines
5.4 KiB
C
Raw Normal View History

/*
* 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_TRAP_H
namespace ircd {
namespace js {
struct trap
{
struct property;
struct function;
trap *parent;
const std::string _name; // don't touch
std::map<std::string, property *> member;
std::map<std::string, function *> memfun;
std::map<std::string, trap *> children;
std::array<JSConstIntegerSpec, 32> cis; // static const integer spec
std::array<JSConstDoubleSpec, 32> cds; // static const double spec
std::array<JSPropertySpec, 32> sps; // static property spec
std::array<JSFunctionSpec, 32> sfs; // static function spec
std::array<JSPropertySpec, 32> ps; // property spec
std::array<JSFunctionSpec, 32> fs; // function spec
std::unique_ptr<JSClass> _class; // class spec
trap *prototrap; // pointer to __proto__ trap
static trap &from(const JSObject &);
static trap &from(const JSObject *const &);
protected:
void debug(const void *const &that, const char *fmt, ...) const AFP(3, 4);
// Override these to define JS objects in C
virtual value on_call(object::handle, value::handle, const args &);
virtual value on_set(object::handle, id::handle, value::handle);
virtual value on_get(object::handle, id::handle, value::handle);
virtual void on_add(object::handle, id::handle, value::handle);
virtual bool on_del(object::handle, id::handle);
virtual bool on_has(object::handle, id::handle);
virtual void on_enu(object::handle);
virtual void on_new(object::handle, object &, const args &);
virtual void on_trace(const JSObject *const &);
virtual void on_gc(JSObject *const &);
private:
void add_this();
void del_this();
void host_exception(const void *const &that, const char *fmt, ...) const AFP(3, 4);
// Internal callback interface
static void handle_trace(JSTracer *, JSObject *) noexcept;
static bool handle_inst(JSContext *, JS::HandleObject, JS::MutableHandleValue, bool *yesno) noexcept;
static bool handle_add(JSContext *, JS::HandleObject, JS::HandleId, JS::HandleValue) noexcept;
static bool handle_set(JSContext *, JS::HandleObject, JS::HandleId, JS::MutableHandleValue, JS::ObjectOpResult &) noexcept;
static bool handle_get(JSContext *, JS::HandleObject, JS::HandleId, JS::MutableHandleValue) noexcept;
static bool handle_del(JSContext *, JS::HandleObject, JS::HandleId, JS::ObjectOpResult &) noexcept;
static bool handle_has(JSContext *, JS::HandleObject, JS::HandleId, bool *resolved) noexcept;
static bool handle_enu(JSContext *, JS::HandleObject) noexcept;
static bool handle_setter(JSContext *, unsigned argc, JS::Value *argv) noexcept;
static bool handle_getter(JSContext *, unsigned argc, JS::Value *argv) noexcept;
static bool handle_call(JSContext *, unsigned argc, JS::Value *argv) noexcept;
static bool handle_ctor(JSContext *, unsigned argc, JS::Value *argv) noexcept;
static void handle_dtor(JSFreeOp *, JSObject *) noexcept;
public:
auto &name() const { return _name; }
auto &jsclass() const { return *_class; }
auto &jsclass() { return *_class; }
// Get child by name (NOT PATH)
const trap &child(const std::string &name) const;
trap &child(const std::string &name);
// Path is absolute to root
static trap &find(const string::handle &path);
static trap &find(const std::string &path);
operator const JSClass &() const { return jsclass(); }
operator const JSClass *() const { return &jsclass(); }
object prototype(const object::handle &globals);
object construct(const object::handle &globals, const vector<value>::handle &argv = {});
object construct(const vector<value>::handle &argv = {});
template<class... args> object operator()(args&&...);
trap(trap &parent, const std::string &name, const uint &flags = 0, const uint &prop_flags = 0);
trap(const std::string &name, const uint &flags = 0, const uint &prop_flags = 0);
trap(trap &&) = delete;
trap(const trap &) = delete;
virtual ~trap() noexcept;
};
extern __thread trap *tree;
} // namespace js
} // namespace ircd
template<class... args>
ircd::js::object
ircd::js::trap::operator()(args&&... a)
{
vector<value> argv{std::forward<args>(a)...};
return construct(argv);
}