2016-10-15 07:44:42 +02:00
|
|
|
/*
|
|
|
|
* 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 {
|
|
|
|
|
2016-11-15 03:33:36 +01:00
|
|
|
struct trap
|
2016-10-15 07:44:42 +02:00
|
|
|
{
|
2016-11-25 04:27:47 +01:00
|
|
|
struct property;
|
2016-11-15 03:33:36 +01:00
|
|
|
struct function;
|
|
|
|
|
2016-11-25 04:27:47 +01:00
|
|
|
trap *parent;
|
2016-10-15 07:44:42 +02:00
|
|
|
const std::string _name; // don't touch
|
2016-11-25 04:27:47 +01:00
|
|
|
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
|
2016-11-15 03:33:36 +01:00
|
|
|
std::array<JSPropertySpec, 32> sps; // static property spec
|
|
|
|
std::array<JSFunctionSpec, 32> sfs; // static function spec
|
2016-11-25 04:27:47 +01:00
|
|
|
std::array<JSPropertySpec, 32> ps; // property spec
|
2016-11-15 03:33:36 +01:00
|
|
|
std::array<JSFunctionSpec, 32> fs; // function spec
|
2016-11-25 04:27:47 +01:00
|
|
|
std::unique_ptr<JSClass> _class; // class spec
|
|
|
|
trap *prototrap; // pointer to __proto__ trap
|
2016-10-15 07:44:42 +02:00
|
|
|
|
2016-11-13 02:47:10 +01:00
|
|
|
static trap &from(const JSObject &);
|
2016-11-25 04:27:47 +01:00
|
|
|
static trap &from(const JSObject *const &);
|
2016-11-13 02:47:10 +01:00
|
|
|
|
2016-11-25 04:23:54 +01:00
|
|
|
protected:
|
|
|
|
void debug(const void *const &that, const char *fmt, ...) const AFP(3, 4);
|
2016-11-13 02:47:10 +01:00
|
|
|
|
2016-10-15 07:44:42 +02:00
|
|
|
// Override these to define JS objects in C
|
2016-11-06 23:40:03 +01:00
|
|
|
virtual value on_call(object::handle, value::handle, const args &);
|
2016-10-25 07:14:39 +02:00
|
|
|
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);
|
2016-10-31 20:08:36 +01:00
|
|
|
virtual void on_enu(object::handle);
|
2016-11-04 00:48:01 +01:00
|
|
|
virtual void on_new(object::handle, object &, const args &);
|
2016-11-14 00:08:03 +01:00
|
|
|
virtual void on_trace(const JSObject *const &);
|
|
|
|
virtual void on_gc(JSObject *const &);
|
2016-10-15 07:44:42 +02:00
|
|
|
|
2016-11-29 16:23:38 +01:00
|
|
|
private: protected:
|
2016-10-25 07:22:19 +02:00
|
|
|
void add_this();
|
|
|
|
void del_this();
|
2016-11-25 04:23:54 +01:00
|
|
|
void host_exception(const void *const &that, const char *fmt, ...) const AFP(3, 4);
|
2016-10-15 07:44:42 +02:00
|
|
|
|
|
|
|
// Internal callback interface
|
2016-10-18 03:43:23 +02:00
|
|
|
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;
|
2016-11-25 04:27:47 +01:00
|
|
|
static bool handle_setter(JSContext *, unsigned argc, JS::Value *argv) noexcept;
|
|
|
|
static bool handle_getter(JSContext *, unsigned argc, JS::Value *argv) noexcept;
|
2016-10-18 03:43:23 +02:00
|
|
|
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;
|
2016-10-15 07:44:42 +02:00
|
|
|
|
|
|
|
public:
|
|
|
|
auto &name() const { return _name; }
|
2016-10-23 03:46:27 +02:00
|
|
|
auto &jsclass() const { return *_class; }
|
2016-11-25 04:27:47 +01:00
|
|
|
auto &jsclass() { return *_class; }
|
2016-10-15 07:44:42 +02:00
|
|
|
|
2016-10-30 13:57:10 +01:00
|
|
|
// Get child by name (NOT PATH)
|
2016-11-08 01:30:28 +01:00
|
|
|
const trap &child(const std::string &name) const;
|
|
|
|
trap &child(const std::string &name);
|
2016-10-25 07:22:19 +02:00
|
|
|
|
|
|
|
// Path is absolute to root
|
2016-10-30 13:57:10 +01:00
|
|
|
static trap &find(const string::handle &path);
|
2016-10-25 07:22:19 +02:00
|
|
|
static trap &find(const std::string &path);
|
|
|
|
|
2016-10-18 03:43:23 +02:00
|
|
|
operator const JSClass &() const { return jsclass(); }
|
|
|
|
operator const JSClass *() const { return &jsclass(); }
|
|
|
|
|
2016-11-25 04:27:47 +01:00
|
|
|
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&&...);
|
2016-10-15 07:44:42 +02:00
|
|
|
|
2016-11-25 04:27:47 +01:00
|
|
|
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);
|
2016-10-15 07:44:42 +02:00
|
|
|
trap(trap &&) = delete;
|
|
|
|
trap(const trap &) = delete;
|
|
|
|
virtual ~trap() noexcept;
|
|
|
|
};
|
|
|
|
|
2016-10-29 07:04:03 +02:00
|
|
|
extern __thread trap *tree;
|
|
|
|
|
2016-10-15 07:44:42 +02:00
|
|
|
} // namespace js
|
|
|
|
} // namespace ircd
|
2016-11-25 04:27:47 +01:00
|
|
|
|
|
|
|
template<class... args>
|
|
|
|
ircd::js::object
|
|
|
|
ircd::js::trap::operator()(args&&... a)
|
|
|
|
{
|
2016-11-26 07:52:11 +01:00
|
|
|
vector<value> argv{std::forward<args>(a)...};
|
2016-11-25 04:27:47 +01:00
|
|
|
return construct(argv);
|
|
|
|
}
|