// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 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. The // full license for this software is available in the LICENSE file. #pragma once #define HAVE_IRCD_JS_TRAP_H namespace ircd { namespace js { struct trap { struct property; // Properties (trap_property.h) struct function; // Functions (trap_function.h) struct constint; // Constant integers (trap_constint.h) std::string name; // classp name trap *prototrap; // Parent trap // Static specifications std::map<std::string, property *> sprop; std::map<std::string, function *> sfunc; // Member specifications std::map<std::string, property *> memprop; std::map<std::string, function *> memfunc; 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: protected: 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; // Aggregate structure specifying internal callback surface static const JSClassOps cops; // Used for regular objects static const JSClassOps gcops; // Used for global objects const JSClass classp; // Instance uses one of the above JSClassOps public: static trap &from(const JSObject *const &); static trap &from(const JSObject &); auto &jsclass() const { return classp; } auto &jsclass() { return classp; } 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(const std::string &name, const uint &flags = 0, const uint &prop_flags = 0); trap(trap &&) = delete; trap(const trap &) = delete; virtual ~trap() noexcept; }; } // namespace js } // namespace ircd template<class... args> ircd::js::object ircd::js::trap::operator()(args&&... a) { const vector<value> argv { std::forward<args>(a)... }; return construct(argv); }