// 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. #include <ircd/js/js.h> #include <js/Initialization.h> // JS_Init() / JS_ShutDown() #include <mozilla/ThreadLocal.h> // For GetThreadType() linkage hack (see: down) // Must be defined until we address issues with SpiderMonkey fixed inside any // IRCD_JS_FIX blocks. #define IRCD_JS_FIX // This was only ever defined for the SpiderMonkey headers and some of our hacks, // but we need to undef it to not step on the log facility log::DEBUG. // Use JS_DEBUG as an analog instead. #undef DEBUG // Logging facility for this submodule with SNOMASK. decltype(ircd::js::log) ircd::js::log { "js", 'J' }; namespace ircd { namespace js { // Location of the thread_local runtext (js/context.h) // If this is null, js is not available on your thread. __thread context *cx; __thread trap *tree; // Whenever a JSClass is seen by the runtime it has to remain reachable for the lifetime // of the runtimet. They don't give any further access or callbacks to free the object, // expecting it to be static or something (yea right). What we have here is a place for // traps to dump their JSClass on destruction and then this can be reaped later. std::forward_list<std::unique_ptr<JSClass>> class_drain; // Internal prototypes const char *reflect(const ::js::CTypesActivityType &); } // namespace js } // namespace ircd /////////////////////////////////////////////////////////////////////////////// // // ircd/js.h - Without 3rd party (JSAPI) symbols // ircd::js::init::init() { if(ircd::nojs) { log::warning { log, "Not initializing the JS engine due to -nojs set by user." }; return; } log::info { log, "Initializing the JS engine [%s: %s]", "SpiderMonkey", version(ver::IMPLEMENTATION) }; const unwind exit([this] { // Ensure ~init() is always safe to call at any intermediate state if(std::current_exception()) this->~init(); }); { const char *errmsg; if((errmsg = JS_InitWithFailureDiagnostic())) throw error("JS_Init(): %s", errmsg); } struct context::opts context_opts; log::info { log, "Initializing the main JS context (main_maxbytes: %zu)", context_opts.max_bytes }; assert(!cx); cx = new context(context_opts); // Additional options //set(*cx, JSGC_MODE, JSGC_MODE_INCREMENTAL); log::info { log, "Initialized main JS context (version: '%s')", version(*cx) }; } ircd::js::init::~init() noexcept { if(!cx) return; log::info { log, "Terminating the main JS context" }; delete cx; cx = nullptr; log::info { log, "Terminating the JS Engine" }; JS_ShutDown(); } const char * ircd::js::version(const ver &type) { switch(type) { case ver::IMPLEMENTATION: return JS_GetImplementationVersion(); default: throw error("version(): Unknown version type requested"); } } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/js.h - With 3rd party (JSAPI) symbols // /////////////////////////////////////////////////////////////////////////////// // // ircd/js/task.h // ircd::js::task::task(const std::string &source) :task{locale::char16::conv(source)} { } ircd::js::task::task(const std::u16string &source) try :pid { 0 } ,global{[this] { JS::CompartmentCreationOptions opts; // Global object is constructed using the root trap (JSClass) at *tree; // This is a thread_local registered by the kernel.so module. struct global global(*tree, nullptr, opts); // The root trap is configured with HAS_PRIVATE and that slot is set so we can find this // `struct task` from the global object using task::get(object). The global object // can first be found from a context or active compartment. As a convenience `struct task` // can be found contextually with task::get(void). JS_SetPrivate(global, this); return global; }()} ,main{[this, &source] () -> struct module { // A compartment for the global must be entered to compile in this scope const compartment c(this->global); // TODO: options JS::CompileOptions opts(*cx); //opts.setCanLazilyParse(true); //opts.setSourceIsLazy(true); //opts.setIntroductionType("GeneratorFunction"); //opts.forceAsync = true; // The compilation is also conducted asynchronously: it will yield the current // ircd::ctx until it is complete. const auto instantiate{true}; return { module::yielding, opts, source, nullptr, instantiate }; }()} { //const compartment c(this->global); } catch(const std::exception &e) { throw; } ircd::js::task::~task() noexcept { run_gc(*cx); } bool ircd::js::task::enter(const std::weak_ptr<task> &ptr, const std::function<void (task &)> &closure) try { const life_guard<struct task> task(ptr); return enter(*task, closure); } catch(const std::bad_weak_ptr &e) { ircd::js::log.warning("task::enter(%p, closure: %p): expired task", (const void *)&ptr, (const void *)&closure); return false; } bool ircd::js::task::enter(task &t, const std::function<void (task &)> &closure) { const std::lock_guard<context> lock{*cx}; const compartment compartment(t.global); run([&closure, &t] { closure(t); }); return true; } ircd::js::object ircd::js::reflect(const task &task) { const object global(task.global); const object reflect(get(global, "Reflect")); const function parse(get(reflect, "parse")); return parse(global, decompile(task)); } ircd::js::string ircd::js::decompile(const task &task, const bool &pretty) { return decompile(task.main, "main", pretty); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/global.h // ircd::js::global::global(trap &trap, JSPrincipals *const &principals, JS::CompartmentCreationOptions copts, JS::CompartmentBehaviors bopts) :object{[&trap, &principals, &copts, &bopts] { //copts.setMergeable(true); //copts.setInvisibleToDebugger(true); copts.setTrace(handle_trace); const JS::CompartmentOptions opts { copts, bopts }; return JS_NewGlobalObject(*cx, &trap.jsclass(), principals, JS::DontFireOnNewGlobalHook, opts); }()} ,module_resolve_hook{[this] { const compartment c(*this); return std::make_unique<function::native>("ModuleResolveHook", 0, 0, [this] (object::handle func, value::handle that, const args &args) { assert(args.size() == 2); const object requestor(args.at(0)); const string requesting(args.at(1)); module &importer(module::our(requestor)); return import(importer, requesting, object(that)); }); }()} { const compartment c(*this); if(!JS_InitStandardClasses(*cx, *this)) throw error("Failed to init standard classes for global object"); JS_InitReflectParse(*cx, *this); JS::SetModuleResolveHook(*cx, *module_resolve_hook); JS_FireOnNewGlobalObject(*cx, *this); } ircd::js::global::~global() noexcept { } void ircd::js::global::handle_trace(JSTracer *const tracer, JSObject *const obj) noexcept { //assert(tracer->runtime() == cx->get()); log.debug("context(%p): global tracer(%p) object(%p)", nullptr,//(const void *)&our(tracer->context()), (const void *)tracer, (const void *)obj); } ircd::js::object ircd::js::global::import(module &importer, const string &requesting, const object &that) { log.debug("context(%p): module '%s' import '%s' (that: %p existing: %d)", (const void *)cx, std::string("<fixme>"), std::string(requesting), (const void *)that.address(), imports.find(requesting) != end(imports)); const auto it(imports.find(requesting)); if(it != end(imports)) { const auto &exporter(*it->second); const string name(exporter.trap->name); js::set(that, name, exporter.trap->construct()); return static_cast<const object &>(exporter); } static const std::string prefix { std::string(fs::MODPATH) + "/" }; string fname(prefix); const string delim("_"); tokens(requesting, '.', [&fname, &delim] (const string &token) { fname += token; fname += delim; }); fname = substr(fname, 0, fname.size() - 1); fname += ".so"; ircd::module *const m(new ircd::module(std::string(fname))); const auto exporter(m->get<module *>("IRCD_JS_MODULE")); const auto iit(imports.emplace(requesting, exporter)); const string name(exporter->trap->name); js::set(that, name, exporter->trap->construct()); return static_cast<const object &>(*exporter); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/trap_property.h // ircd::js::trap::property::property(struct trap &trap, const std::string &name, const uint &flags) try :trap{&trap} ,name{name} ,spec{0} { spec.name = this->name.data(), spec.flags = flags, spec.accessors.setter.native = { handle_set }; spec.accessors.getter.native = { handle_get }; const auto iit(trap.memprop.emplace(this->name, this)); if(!iit.second) throw error("trap property '%s' already exists on '%s'", this->name, trap.name); log.debug("Registered property '%s' on trap '%s'", this->name, trap.name); } catch(const error &e) { throw error("Failed to register property '%s': out slots on trap '%s': %s", this->name.c_str(), trap.name.c_str(), e.what()); } ircd::js::trap::property::~property() noexcept { const auto erased(trap->memprop.erase(name)); assert(erased); } bool ircd::js::trap::property::handle_get(JSContext *const c, const unsigned argc, JS::Value *const argv) noexcept try { using js::function; const struct args args(argc, argv); object that(args.computeThis(c)); function func(args.callee()); const string name(js::name(func)); auto &trap(from(that)); trap.debug(that.get(), "get '%s' (property)", name.c_str()); property &prop(*trap.memprop.at(name)); args.rval().set(prop.on_get(func, that)); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto ca(JS::CallArgsFromVp(argc, argv)); object that(ca.computeThis(c)); auto &trap(from(that)); trap.host_exception(that.get(), "property get: %s", e.what()); return false; } bool ircd::js::trap::property::handle_set(JSContext *const c, const unsigned argc, JS::Value *const argv) noexcept try { using js::function; const struct args args(argc, argv); object that(args.computeThis(c)); function func(args.callee()); const string name(js::name(func)); auto &trap(from(that)); trap.debug(that.get(), "set '%s' (property)", name.c_str()); property &prop(*trap.memprop.at(name)); args.rval().set(prop.on_get(func, that)); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto ca(JS::CallArgsFromVp(argc, argv)); object that(ca.computeThis(c)); auto &trap(from(that)); trap.host_exception(that.get(), "property set: %s", e.what()); return false; } ircd::js::value ircd::js::trap::property::on_get(function::handle, object::handle that) { return {}; } ircd::js::value ircd::js::trap::property::on_set(function::handle, object::handle that, value::handle val) { return val; } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/trap_function.h // ircd::js::trap::function::function(struct trap &trap, const std::string &name, const uint16_t &flags, const uint16_t &arity, const closure &lambda) :trap{&trap} ,name{name} ,lambda{lambda} ,spec { this->name.data(), { handle_call }, arity, flags } { const auto iit(trap.memfunc.emplace(this->name, this)); if(!iit.second) throw error("trap function '%s' already exists on '%s'", this->name, trap.name); log.debug("Registered function '%s' on trap '%s'", this->name, trap.name); } ircd::js::trap::function::~function() noexcept { const auto erased(trap->memfunc.erase(this->name)); assert(erased); } ircd::js::function ircd::js::trap::function::operator()(const object::handle &obj) const { const auto jsf(::js::DefineFunctionWithReserved(*cx, obj, spec.name, spec.call.op, spec.nargs, spec.flags)); if(unlikely(!jsf)) throw internal_error("Failed to create trap::function"); js::function ret(jsf); ::js::SetFunctionNativeReserved(ret, 0, pointer_value(this)); return ret; } bool ircd::js::trap::function::handle_call(JSContext *const c, const unsigned argc, JS::Value *const argv) noexcept try { assert(&our(c) == cx); const struct args args(argc, argv); const object func(args.callee()); auto &trap(from(func)); if(args.isConstructing()) { log.debug("trap(%p) %s() ctor argv[%u]", (const void *)&trap, trap.name, argc); const value that(trap.on_new(func, args)); args.rval().set(that); log.debug("trap(%p) this(%p) %s() leave", (const void *)&trap, (const void *)that.address(), trap.name); } else { const value that(args.computeThis(c)); log.debug("trap(%p) this(%p) %s() call argv[%u]", (const void *)&trap, (const void *)that.address(), trap.name, argc); args.rval().set(trap.on_call(func, that, args)); log.debug("trap(%p) this(%p) %s() leave", (const void *)&trap, (const void *)that.address(), trap.name); } return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { const struct args args(argc, argv); auto &func(args.callee()); auto &trap(from(&func)); log.error("trap(%p) \"%s()\": %s", reinterpret_cast<const void *>(&trap), trap.name, e.what()); JS_ReportErrorUTF8(*cx, "BUG: trap(%p) \"%s()\": %s", reinterpret_cast<const void *>(&trap), trap.name.c_str(), e.what()); return false; } ircd::js::trap::function & ircd::js::trap::function::from(JSObject *const &func) { const auto tval(::js::GetFunctionNativeReserved(func, 0)); return *pointer_value<function>(tval); } ircd::js::value ircd::js::trap::function::on_call(object::handle obj, value::handle val, const args &args) { return lambda(obj, val, args); } ircd::js::value ircd::js::trap::function::on_new(object::handle obj, const args &args) { value ud; return on_call(obj, ud, args); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/trap.h // namespace ircd { namespace js { } // namespace js } // namespace ircd ircd::js::trap::trap(const std::string &name, const uint &flags, const uint &prop_flags) :name{name} ,prototrap{nullptr} ,classp { this->name.data(), flags | JSCLASS_FOREGROUND_FINALIZE, (flags & JSCLASS_IS_GLOBAL)? &gcops : &cops, { this } // reserved[0] TODO: ????????? } { if(this->name == "[global]") ircd::js::tree = this; log.debug("Registered trap '%s' @ %p", jsclass().name, (const void *)this); } ircd::js::trap::~trap() noexcept { log.debug("Unregistered trap '%s' @ %p", jsclass().name, (const void *)this); } const JSClassOps ircd::js::trap::cops { handle_add, handle_del, handle_get, handle_set, handle_enu, handle_has, nullptr, // JSConvertOp - Obsolete since SpiderMonkey 44 // 45 = mayResolve? handle_dtor, handle_call, handle_inst, handle_ctor, handle_trace, }; const JSClassOps ircd::js::trap::gcops { handle_add, // flags & JSCLASS_GLOBAL_FLAGS? nullptr : handle_add, handle_del, handle_get, handle_set, handle_enu, handle_has, nullptr, // JSConvertOp - Obsolete since SpiderMonkey 44 // 45 = mayResolve? handle_dtor, handle_call, handle_inst, handle_ctor, JS_GlobalObjectTraceHook, }; ircd::js::object ircd::js::trap::construct(const vector<value>::handle &argv) { const object globals; return construct(globals, argv); } ircd::js::object ircd::js::trap::construct(const object::handle &globals, const vector<value>::handle &argv) { const object prototype(this->prototype(globals)); return JS_New(*cx, prototype, argv); } ircd::js::object ircd::js::trap::prototype(const object::handle &globals) { // Create parent object of this object in the prototype chain const object super { prototrap? prototrap->construct() : object{object::uninitialized} }; // Build static property descriptor array std::vector<JSPropertySpec> sprop { this->sprop.size() + 1, // JS requires classical null terminator { 0 } }; // Build static function descriptor array std::vector<JSFunctionSpec> sfunc { this->sfunc.size() + 1, // JS requires classical null terminator { 0 } }; // Build member property descriptor array std::vector<JSPropertySpec> memprop { this->memprop.size() + 1, // JS requires classical null terminator { 0 } }; std::transform(begin(this->memprop), end(this->memprop), begin(memprop), [] (const auto &it) { const auto &prop(*it.second); return prop.spec; }); // Mozilla really screwed the pooch here. There is no way to specify a function // with a reserved slot in order to find our trap::function instance from the // JSNative callback. Their API surface is too broad and there are too many different // and mutually exclusive ways to define functions. So we have to be inconsistent // here and not use InitClass()'s facility for defining member functions. // Build member function descriptor array std::vector<JSFunctionSpec> memfunc { this->memfunc.size() + 1, // JS requires classical vector terminator { 0 } }; // std::transform(begin(this->memfunc), end(this->memfunc), begin(memfunc), [] // (const auto &it) // { // const auto &function(*it.second); // return function.spec; // }); // Create object const object proto { JS_InitClass(*cx, globals, super, &jsclass(), nullptr, 0, memprop.data(), memfunc.data(), sprop.data(), sfunc.data()) }; // Member functions defined after the fact here. std::for_each(begin(this->memfunc), end(this->memfunc), [&proto] (const auto &it) { const auto &deffun(*it.second); const auto func(deffun(proto)); }); //JS_DefineConstIntegers(*cx, proto, cis.data()); //JS_DefineConstDoubles(*cx, proto, cds.data()); return proto; } void ircd::js::trap::handle_dtor(JSFreeOp *const op, JSObject *const obj) noexcept try { assert(op); assert(obj); //assert(&our_runtime(*op) == rt); auto &trap(from(*obj)); trap.debug(obj, "dtor"); trap.on_gc(obj); } catch(const std::exception &e) { auto &trap(from(*obj)); log.critical("Unhandled on GC (fop: %p obj: %p): %s", (const void *)op, (const void *)obj, e.what()); assert(0); } bool ircd::js::trap::handle_ctor(JSContext *const c, unsigned argc, JS::Value *const argv) noexcept try { assert(&our(c) == cx); assert(!pending_exception(*cx)); const struct args args(argc, argv); object that(args.callee()); auto &trap(from(that)); trap.debug(that.get(), "ctor '%s' argv[%zu]", trap.name.c_str(), args.size()); object ret(JS_NewObjectWithGivenProto(*cx, &trap.jsclass(), that)); trap.on_new(that, ret, args); args.rval().set(JS::Value(ret)); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto ca(JS::CallArgsFromVp(argc, argv)); object that(ca.callee()); auto &trap(from(that)); trap.host_exception(that.get(), "ctor: %s", e.what()); return false; } bool ircd::js::trap::handle_call(JSContext *const c, unsigned argc, JS::Value *const argv) noexcept try { assert(&our(c) == cx); assert(!pending_exception(*cx)); const struct args args(argc, argv); value that(args.computeThis(c)); object func(args.callee()); //auto &trap_that(from(object(that))); auto &trap_func(from(func)); //trap_that.debug(that.get(), "call: '%s'", trap_func.name().c_str()); trap_func.debug(func.get(), "call argv[%zu]", args.size()); args.rval().set(trap_func.on_call(func, that, args)); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto ca(JS::CallArgsFromVp(argc, argv)); object func(ca.callee()); auto &trap(from(func)); trap.host_exception(func.get(), "call: %s", e.what()); return false; } bool ircd::js::trap::handle_enu(JSContext *const c, JS::HandleObject obj) noexcept try { assert(&our(c) == cx); auto &trap(from(obj)); trap.debug(obj.get(), "enumerate"); trap.on_enu(obj); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto &trap(from(obj)); trap.host_exception(obj.get(), "enu: %s", e.what()); return false; } bool ircd::js::trap::handle_has(JSContext *const c, JS::HandleObject obj, JS::HandleId id, bool *const resolved) noexcept try { assert(&our(c) == cx); assert(!pending_exception(*cx)); auto &trap(trap::from(obj)); trap.debug(obj.get(), "has '%s'", string(id).c_str()); *resolved = trap.on_has(obj, id); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto &trap(from(obj)); trap.host_exception(obj.get(), "has '%s': %s", string(id).c_str(), e.what()); return false; } bool ircd::js::trap::handle_del(JSContext *const c, JS::HandleObject obj, JS::HandleId id, JS::ObjectOpResult &res) noexcept try { assert(&our(c) == cx); assert(!pending_exception(*cx)); auto &trap(from(obj)); trap.debug(obj.get(), "del '%s'", string(id).c_str()); if(trap.on_del(obj, id)) res.succeed(); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto &trap(from(obj)); trap.host_exception(obj.get(), "del '%s': %s", string(id).c_str(), e.what()); return false; } bool ircd::js::trap::handle_getter(JSContext *const c, unsigned argc, JS::Value *const argv) noexcept try { using js::function; const struct args args(argc, argv); const object that(args.computeThis(c)); const function func(args.callee()); const string name(js::name(func)); auto &trap(from(that)); trap.debug(that.get(), "get '%s' (getter)", name.c_str()); //value &val(it->second); //args.rval().set(val); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { const auto ca(JS::CallArgsFromVp(argc, argv)); const object that(ca.computeThis(c)); auto &trap(from(that)); trap.host_exception(that.get(), "getter: %s", e.what()); return false; } bool ircd::js::trap::handle_setter(JSContext *const c, unsigned argc, JS::Value *const argv) noexcept try { using js::function; const struct args args(argc, argv); const object that(args.computeThis(c)); const function func(args.callee()); const value val(args[0]); const auto type(js::type(val)); const string name(js::name(func)); auto &trap(from(that)); trap.debug(that.get(), "set '%s' (%s) (setter)", name.c_str(), reflect(type)); switch(js::type(type)) { case jstype::OBJECT: { //const auto flags(JSPROP_SHARED); //object ret(JS_DefineObject(*cx, object(val), "", &trap.jsclass(), flags)); //args.rval().set(ret); //return true; } default: args.rval().set(val); return true; } } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { const auto ca(JS::CallArgsFromVp(argc, argv)); const object that(ca.computeThis(c)); auto &trap(from(that)); trap.host_exception(that.get(), "setter: %s", e.what()); return false; } bool ircd::js::trap::handle_get(JSContext *const c, JS::HandleObject obj, JS::HandleId id, JS::MutableHandleValue val) noexcept try { assert(&our(c) == cx); assert(!pending_exception(*cx)); auto &trap(from(obj)); trap.debug(obj.get(), "get '%s'", string(id).c_str()); const value ret(trap.on_get(obj, id, val)); val.set(ret.get()); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto &trap(from(obj)); trap.host_exception(obj.get(), "get: '%s': %s", string(id).c_str(), e.what()); return false; } bool ircd::js::trap::handle_set(JSContext *const c, JS::HandleObject obj, JS::HandleId id, JS::MutableHandleValue val, JS::ObjectOpResult &res) noexcept try { assert(&our(c) == cx); assert(!pending_exception(*cx)); auto &trap(from(obj)); trap.debug(obj.get(), "set '%s'", string(id).c_str()); const value ret(trap.on_set(obj, id, val)); val.set(ret.get()); if(!val.isUndefined()) res.succeed(); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto &trap(from(obj)); trap.host_exception(obj.get(), "set '%s': %s", string(id).c_str(), e.what()); return false; } bool ircd::js::trap::handle_add(JSContext *const c, JS::HandleObject obj, JS::HandleId id, JS::HandleValue val) noexcept try { assert(&our(c) == cx); assert(!pending_exception(*cx)); auto &trap(from(obj)); const string name(id); trap.debug(obj.get(), "add '%s' %s @%p", name.c_str(), reflect(type(val)), (const void *)val.address()); trap.on_add(obj, id, val); return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto &trap(from(obj)); trap.host_exception(obj.get(), "add '%s': %s", string(id).c_str(), e.what()); return false; } bool ircd::js::trap::handle_inst(JSContext *const c, JS::HandleObject obj, JS::MutableHandleValue val, bool *const has_instance) noexcept try { assert(&our(c) == cx); auto &trap(from(obj)); trap.debug(obj.get(), "inst"); return false; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { auto &trap(from(obj)); trap.host_exception(obj.get(), "inst: %s", e.what()); return false; } void ircd::js::trap::handle_trace(JSTracer *const tracer, JSObject *const obj) noexcept try { assert(cx); assert(tracer); assert(obj); auto &trap(from(*obj)); trap.debug(obj, "trace"); trap.on_trace(obj); } catch(const jserror &e) { e.set_pending(); return; } catch(const std::exception &e) { auto &trap(from(*obj)); trap.host_exception(obj, "trace: %s", e.what()); return; } ircd::js::trap & ircd::js::trap::from(const JSObject &o) { return from(&o); } ircd::js::trap & ircd::js::trap::from(const JSObject *const &o) { auto *const c(JS_GetClass(const_cast<JSObject *>(o))); if(!c) { log.critical("trap::from(): Trapped on an object without a JSClass!"); ircd::terminate(); //TODO: exception } if(!c->reserved[0]) { log.critical("trap::from(): Trap called on a trap instance that has gone out of scope!"); ircd::terminate(); //TODO: exception } return *static_cast<trap *>(c->reserved[0]); //TODO: ??? } void ircd::js::trap::debug(const void *const &that, const char *const fmt, ...) const { va_list ap; va_start(ap, fmt); char buf[1024]; vsnprintf(buf, sizeof(buf), fmt, ap); log.debug("trap(%p) this(%p) '%s' %s", reinterpret_cast<const void *>(this), that, jsclass().name, buf); va_end(ap); } void ircd::js::trap::host_exception(const void *const &that, const char *const fmt, ...) const { va_list ap; va_start(ap, fmt); char buf[1024]; vsnprintf(buf, sizeof(buf), fmt, ap); log.error("trap(%p) this(%p) '%s' %s", reinterpret_cast<const void *>(this), that, jsclass().name, buf); JS_ReportErrorUTF8(*cx, "BUG: trap(%p) this(%p) '%s' %s", reinterpret_cast<const void *>(this), that, jsclass().name, buf); va_end(ap); } void ircd::js::trap::on_gc(JSObject *const &that) { if(jsclass().flags & JSCLASS_IS_GLOBAL) //TODO: leak?? return; if(jsclass().flags & JSCLASS_HAS_PRIVATE) del(that, priv); } void ircd::js::trap::on_trace(const JSObject *const &) { } void ircd::js::trap::on_new(object::handle, object &, const args &) { } void ircd::js::trap::on_enu(object::handle) { } bool ircd::js::trap::on_has(object::handle, id::handle id) { // const string sid(id); // if(children.count(sid)) // return false; // // if(std::any_of(begin(memfun), end(memfun), [&sid] // (const auto &it) // { // const auto &memfun(*it.second); // return sid == memfun.name; // })) // return false; // // value val; // const auto flags(JSPROP_SHARED | JSPROP_ENUMERATE); // if(!JS_DefinePropertyById(*cx, obj, id, val, flags, handle_getter, handle_setter)) // throw jserror("Failed to define property '%s'", sid.c_str()); // // const auto flags(0); // if(!JS_DefineObject(*cx, obj, sid.c_str(), &jsclass(), flags)) // throw jserror("Failed to define property '%s'", sid.c_str()); return false; } bool ircd::js::trap::on_del(object::handle, id::handle) { return true; } void ircd::js::trap::on_add(object::handle, id::handle, value::handle) { } ircd::js::value ircd::js::trap::on_get(object::handle obj, id::handle id, value::handle val) { return val; } ircd::js::value ircd::js::trap::on_set(object::handle, id::handle, value::handle val) { return val; } ircd::js::value ircd::js::trap::on_call(object::handle, value::handle, const args &) { return {}; } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/module.h // /////////////////////////////////////////////////////////////////////////////// // // ircd/js/script.h // namespace ircd { namespace js { void handle_compile_async(void *, void *) noexcept; } // namespace js } // namespace ircd bool ircd::js::compilable(const std::string &str, const object &stack) { return compilable(str.c_str(), str.size(), stack); } bool ircd::js::compilable(const char *const &str, const size_t &len, const object &stack) { return JS_BufferIsCompilableUnit(*cx, stack, str, len); } size_t ircd::js::bytecodes(const JS::Handle<JSScript *> &s, uint8_t *const &buf, const size_t &max) { JS::TranscodeBuffer tcbuf; const auto code { JS::EncodeScript(*cx, tcbuf, s) }; switch(code) { case JS::TranscodeResult_Ok: { if(unlikely(tcbuf.length() > max)) throw internal_error { "Insufficient buffer size %zu where %zu is required.", max, tcbuf.length() }; memcpy(buf, tcbuf.begin(), tcbuf.length()); return tcbuf.length(); } case JS::TranscodeResult_Throw: throw jserror(jserror::pending); default: throw internal_error { "Failed to encode script to XDR (error %d)", code }; }; } ircd::js::string ircd::js::decompile(const JS::Handle<JSScript *> &s, const char *const &name, const bool &pretty) { uint flags(0); flags |= pretty; return JS_DecompileScript(*cx, s, name, flags); } ircd::ctx::future<void *> ircd::js::compile_async(const JS::ReadOnlyCompileOptions &opts, const std::u16string &src, const bool &module) { auto promise(std::make_unique<ctx::promise<void *>>()); if(!JS::CanCompileOffThread(*cx, opts, src.size())) { log.debug("context(%p): Rejected asynchronous script compile (script size: %zu)", (const void *)cx, src.size()); ctx::future<void *> ret(*promise); promise->set_value(nullptr); return ret; } const auto &compile { module? JS::CompileOffThreadModule : JS::CompileOffThread }; if(!compile(*cx, opts, src.data(), src.size(), handle_compile_async, promise.get())) throw internal_error("Failed to compile %s concurrently", module? "module" : "script"); return *promise.release(); } void ircd::js::handle_compile_async(void *const token, void *const priv) noexcept { // This frame is entered on a thread owned by SpiderMonkey, not IRCd. Do not call // ircd::log from here, it is not thread-safe. // printf("[thread %s]: context(%p): compile(%p) READY (priv: %p)\n", // ircd::string(std::this_thread::get_id()).c_str(), // (const void *)cx, // token, // priv); // Setting the value of the promise and then deleting the promise is thread-safe. // Note that JS::FinishOffThreadScript(); will need to be called on the main thread. auto *const _promise(reinterpret_cast<ctx::promise<void *> *>(priv)); const std::unique_ptr<ctx::promise<void *>> promise(_promise); promise->set_value(token); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/function.h // ircd::js::value ircd::js::function::operator()(const object::handle &that, const vector<value>::handle &argv) const { return call(*this, that, argv); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/function_literal.h // ircd::js::function::literal::literal(const char *const &name, const std::initializer_list<const char *> &prototype, const char *const &text) :root<JSFunction *>{} ,name{name} ,text{text} ,prototype{prototype} { JS::CompileOptions opts{*cx}; JS::AutoObjectVector stack{*cx}; if(!JS::CompileFunction(*cx, stack, opts, name, this->prototype.size(), &this->prototype.front(), text, strlen(text), &(*this))) { throw syntax_error("Failed to compile function literal"); } } ircd::js::function::literal::literal(literal &&other) noexcept :root<JSFunction *> { std::move(other) } ,name{std::move(other.name)} ,text{std::move(other.text)} ,prototype{std::move(other.prototype)} { } ircd::js::function::literal operator ""_function(const char *const text, const size_t len) { return { "<literal>", {}, text }; } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/function_native.h // ircd::js::function::native::native(const char *const &name, const uint &flags, const uint &arity, const closure &lambda) :native::root::type{[this, &name, &flags, &arity] { const auto jsf(::js::DefineFunctionWithReserved(*cx, object(), name, handle_call, arity, flags)); if(unlikely(!jsf)) throw internal_error("Failed to create function::native '%s'", name); js::function ret(jsf); ::js::SetFunctionNativeReserved(ret, 0, pointer_value(this)); return ret; }()} ,_name{name} ,lambda{lambda} { } ircd::js::function::native::~native() noexcept { } ircd::js::value ircd::js::function::native::operator()(const object::handle &that, const vector<value>::handle &argv) const { return call(*this, that, argv); } ircd::js::string ircd::js::function::native::name() const { return JS_GetFunctionId(*this); } ircd::js::string ircd::js::function::native::display_name() const { return JS_GetFunctionDisplayId(*this); } uint16_t ircd::js::function::native::arity() const { return JS_GetFunctionArity(*this); } bool ircd::js::function::native::handle_call(JSContext *const c, const unsigned argc, JS::Value *const argv) noexcept try { assert(&our(c) == cx); const struct args args(argc, argv); const object func(args.callee()); auto &instance(from(func)); if(args.isConstructing()) { log.debug("native(%p) %s() ctor argv[%u]", (const void *)&instance, instance._name, argc); const value that(instance.on_new(func, args)); args.rval().set(that); log.debug("native(%p) this(%p) %s() leave", (const void *)&instance, (const void *)that.address(), instance._name); } else { const value that(args.computeThis(c)); log.debug("native(%p) this(%p) %s() call argv[%u]", (const void *)&instance, (const void *)that.address(), instance._name, argc); args.rval().set(instance.on_call(func, that, args)); log.debug("native(%p) this(%p) %s() leave", (const void *)&instance, (const void *)that.address(), instance._name); } return true; } catch(const jserror &e) { e.set_pending(); return false; } catch(const std::exception &e) { const struct args args(argc, argv); const object func(args.callee()); const function::native &instance(from(func)); log.error("native(%p) \"%s()\": %s", std::addressof(instance), instance._name, e.what()); JS_ReportErrorUTF8(*cx, "BUG: native(%p) \"%s()\": %s", std::addressof(instance), instance._name, e.what()); return false; } ircd::js::function::native & ircd::js::function::native::from(JSObject *const &func) { const auto tval(::js::GetFunctionNativeReserved(func, 0)); return *pointer_value<function::native>(tval); } ircd::js::value ircd::js::function::native::on_call(object::handle obj, value::handle val, const args &args) { return lambda(obj, val, args); } ircd::js::value ircd::js::function::native::on_new(object::handle obj, const args &args) { value ud; return on_call(obj, ud, args); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/for_each.h // void ircd::js::for_each(object::handle obj, const each_key_val &closure) { for_each(obj, iter::none, closure); } void ircd::js::for_each(object::handle obj, const iter &flags, const each_key_val &closure) { for_each(obj, flags, each_id([&obj, &closure] (const id &key) { const value val(get(obj, key)); closure(key, val); })); } void ircd::js::for_each(object::handle obj, const each_key &closure) { for_each(obj, iter::none, closure); } void ircd::js::for_each(object::handle obj, const iter &flags, const each_key &closure) { for_each(obj, flags, each_id([&obj, &closure] (const id &id) { closure(id); })); } void ircd::js::for_each(object::handle obj, const each_id &closure) { for_each(obj, iter::none, closure); } void ircd::js::for_each(object::handle obj, const iter &flags, const each_id &closure) { JS::AutoIdVector props(*cx); if(::js::GetPropertyKeys(*cx, obj, uint(flags), &props)) for(size_t i(0); i < props.length(); ++i) closure(props[i]); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/call.h // ircd::js::value ircd::js::call(const std::string &name, const object::handle &that, const vector<value>::handle &args) { return call(name.c_str(), that, args); } ircd::js::value ircd::js::call(const char *const &name, const object::handle &that, const vector<value>::handle &args) { value ret; if(!JS_CallFunctionName(*cx, that, name, args, &ret)) throw jserror(jserror::pending); return ret; } ircd::js::value ircd::js::call(const value::handle &val, const object::handle &that, const vector<value>::handle &args) { value ret; if(!JS_CallFunctionValue(*cx, that, val, args, &ret)) throw jserror(jserror::pending); return ret; } ircd::js::value ircd::js::call(const function::handle &func, const object::handle &that, const vector<value>::handle &args) { value ret; if(!JS_CallFunction(*cx, that, func, args, &ret)) throw jserror(jserror::pending); return ret; } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/del.h // void ircd::js::del(const object::handle &src, const char *const path) { value val; object obj(src); const char *fail(nullptr); char buffer[strlen(path) + 1]; ircd::tokens(path, ".", mutable_buffer{buffer, sizeof(buffer)}, [&path, &val, &obj, &fail] (const string_view &part) { if(fail) throw type_error("cannot recurse through non-object '%s' in `%s'", fail, path); if(!JS_GetProperty(*cx, obj, part.data(), &val) || undefined(val)) throw reference_error("%s", part.data()); object tmp(obj.get()); if(!JS_ValueToObject(*cx, val, &obj) || !obj.get()) { fail = part.data(); obj = std::move(tmp); } }); del(obj, id(val)); } void ircd::js::del(const object::handle &obj, const uint32_t &idx) { JS::ObjectOpResult res; if(!JS_DeleteElement(*cx, obj, idx, res)) throw jserror(jserror::pending); if(!res.checkStrict(*cx, obj)) throw jserror(jserror::pending); } void ircd::js::del(const object::handle &obj, const id &id) { return del(obj, id::handle(id)); } void ircd::js::del(const object::handle &obj, const id::handle &id) { JS::ObjectOpResult res; if(!JS_DeletePropertyById(*cx, obj, id, res)) throw jserror(jserror::pending); if(!res.checkStrict(*cx, obj, id)) throw jserror(jserror::pending); } void ircd::js::del(JSObject *const &obj, priv_t) { if(unlikely(~flags(obj) & JSCLASS_HAS_PRIVATE)) throw error("del(priv): Object has no private slot"); void *const existing(JS_GetPrivate(obj)); delete reinterpret_cast<priv_ptr *>(existing); JS_SetPrivate(obj, nullptr); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/set.h // void ircd::js::set(const object::handle &src, const char *const path, const value &val) { object obj(src); char buffer[strlen(path) + 1]; const char *fail(nullptr), *key(nullptr); ircd::tokens(path, ".", mutable_buffer{buffer, sizeof(buffer)}, [&path, &obj, &fail, &key] (const string_view &part) { if(fail) throw type_error("cannot recurse through non-object '%s' in `%s'", fail, path); if(key) throw reference_error("%s", part.data()); key = part.data(); if(strcmp(key, path) == 0) return; value tmp; if(!JS_GetProperty(*cx, obj, part.data(), &tmp) || undefined(tmp)) return; if(!JS_ValueToObject(*cx, tmp, &obj) || !obj.get()) fail = part.data(); }); if(!key) return; if(!JS_SetProperty(*cx, obj, key, val)) throw jserror(jserror::pending); } void ircd::js::set(const object::handle &obj, const id &id, const value &val) { set(obj, id::handle(id), val); } void ircd::js::set(const object::handle &obj, const id::handle &id, const value &val) { set(obj, id, value::handle(val)); } void ircd::js::set(const object::handle &obj, const id::handle &id, const value::handle &val) { if(!JS_SetPropertyById(*cx, obj, id, val)) throw jserror(jserror::pending); } void ircd::js::set(JSObject *const &obj, priv_data &data) { set(obj, shared_from(data)); } void ircd::js::set(JSObject *const &obj, const std::shared_ptr<priv_data> &data) { if(unlikely(~flags(obj) & JSCLASS_HAS_PRIVATE)) throw error("set(priv): Object has no private slot"); void *const existing(JS_GetPrivate(obj)); delete reinterpret_cast<priv_ptr *>(existing); JS_SetPrivate(obj, new priv_ptr(data)); } void ircd::js::set(JSObject *const &obj, const reserved &slot, const JS::Value &val) { JS_SetReservedSlot(obj, uint(slot), val); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/get.h // ircd::js::value ircd::js::get(const object::handle &src, const char *const path) { value ret; object obj(src); const char *fail(nullptr); char buffer[strlen(path) + 1]; ircd::tokens(path, ".", mutable_buffer{buffer, sizeof(buffer)}, [&obj, &path, &ret, &fail] (const string_view &part) { if(fail) throw type_error("cannot recurse through non-object '%s' in `%s'", fail, path); if(!JS_GetProperty(*cx, obj, part.data(), &ret) || undefined(ret)) throw reference_error("%s", part.data()); if(!JS_ValueToObject(*cx, ret, &obj) || !obj.get()) fail = part.data(); }); return ret; } ircd::js::value ircd::js::get(const object::handle &obj, const uint32_t &idx) { value ret; if(!JS_GetElement(*cx, obj, idx, &ret) || undefined(ret)) throw reference_error("[%u]", idx); return ret; } ircd::js::value ircd::js::get(const object::handle &obj, const id &id) { return get(obj, id::handle(id)); } ircd::js::value ircd::js::get(const object::handle &obj, const id::handle &id) { value ret; if(!JS_GetPropertyById(*cx, obj, id, &ret) || undefined(ret)) throw reference_error("%s", string(id).c_str()); return ret; } JS::Value ircd::js::get(JSObject *const &obj, const reserved &slot) { return JS_GetReservedSlot(obj, uint(slot)); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/has.h // bool ircd::js::has(const object::handle &src, const char *const path) { bool ret(true); object obj(src); const char *fail(nullptr); char buffer[strlen(path) + 1]; ircd::tokens(path, ".", mutable_buffer{buffer, sizeof(buffer)}, [&obj, &path, &ret, &fail] (const string_view &part) { if(fail) throw type_error("cannot recurse through non-object '%s' in `%s'", fail, path); if(!JS_HasProperty(*cx, obj, part.data(), &ret)) throw jserror(jserror::pending); if(!ret) return; value tmp; if(!JS_GetProperty(*cx, obj, part.data(), &tmp) || undefined(tmp)) { ret = false; return; } if(!JS_ValueToObject(*cx, tmp, &obj) || !obj.get()) fail = part.data(); }); return ret; } bool ircd::js::has(const object::handle &obj, const uint32_t &idx) { bool ret; if(!JS_HasElement(*cx, obj, idx, &ret)) throw jserror(jserror::pending); return ret; } bool ircd::js::has(const object::handle &obj, const id &id) { return has(obj, id::handle(id)); } bool ircd::js::has(const object::handle &obj, const id::handle &id) { bool ret; if(!JS_HasPropertyById(*cx, obj, id, &ret)) throw jserror(jserror::pending); return ret; } bool ircd::js::has(const JSObject *const &obj, priv_t) { if(~flags(obj) & JSCLASS_HAS_PRIVATE) return false; const auto vp(JS_GetPrivate(const_cast<JSObject *>(obj))); const auto sp(reinterpret_cast<const priv_ptr *>(vp)); return sp && !!*sp; } bool ircd::js::has(const JSObject *const &obj, const reserved &slot) { return flags(obj) & JSCLASS_HAS_RESERVED_SLOTS(uint(slot)); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/object.h // /////////////////////////////////////////////////////////////////////////////// // // ircd/js/priv.h // // Anchor the struct priv_data vtable here. ircd::js::priv_data::~priv_data() noexcept { } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/string.h // char * ircd::js::c_str(const JSString *const &str) { static thread_local char cbuf[CSTR_BUFS][CSTR_BUFSIZE]; static thread_local size_t ctr; char *const buf(cbuf[ctr]); native(str, buf, CSTR_BUFSIZE); ctr = (ctr + 1) % CSTR_BUFS; return buf; } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/value.h // /////////////////////////////////////////////////////////////////////////////// // // ircd/js/json.h // namespace ircd { namespace js { namespace json { bool write_callback(const char16_t *, uint32_t, void *) noexcept; } // namespace json } // namespace js } // namespace ircd ircd::js::value ircd::js::json::parse(const string &string) { value ret; if(!JS_ParseJSON(*cx, string, &ret)) throw jserror(jserror::pending); return ret; } ircd::js::value ircd::js::json::parse(const char *const &str) { return parse(locale::char16::conv(str)); } ircd::js::value ircd::js::json::parse(const std::string &str) { return parse(locale::char16::conv(str)); } ircd::js::value ircd::js::json::parse(const std::u16string &str) { return parse(str.c_str(), str.size()); } ircd::js::value ircd::js::json::parse(const char16_t *const &str, const size_t &size) { value ret; if(!JS_ParseJSON(*cx, str, size, &ret)) throw jserror(jserror::pending); return ret; } std::u16string ircd::js::json::stringify(const value &val, const bool &pretty) { value v(val); return stringify(value::handle_mutable(&v), pretty); } std::u16string ircd::js::json::stringify(value &v, const bool &pretty) { return stringify(value::handle_mutable(&v), pretty); } std::u16string ircd::js::json::stringify(const value::handle_mutable &val, const bool &pretty) { const object fmtr; const string sp(string::literal, pretty? u"\t" : u""); return stringify(val, fmtr, sp); } std::u16string ircd::js::json::stringify(const value::handle_mutable &val, const JS::HandleObject &fmtr, const value &sp) { std::u16string ret; stringify(val, fmtr, sp, [&ret] (const char16_t *const &ptr, const uint &len) { ret.assign(ptr, len); return true; }); return ret; } void ircd::js::json::stringify(const value::handle_mutable &val, const closure &cont) { value sp; object fmtr; return stringify(val, fmtr, sp, cont); } void ircd::js::json::stringify(const value::handle_mutable &val, const JS::HandleObject &fmtr, const value &sp, const closure &cont) { if(!JS_Stringify(*cx, val, fmtr, sp, write_callback, const_cast<closure *>(&cont))) throw jserror(jserror::pending); } bool ircd::js::json::write_callback(const char16_t *const str, const uint32_t len, void *const priv) noexcept { const auto &func(*reinterpret_cast<const closure *>(priv)); return func(str, len); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/native.h // namespace ircd { namespace js { void native_external_noop(JS::Zone *const zone, const JSStringFinalizer *const fin, char16_t *const buf); void native_external_deleter(JS::Zone *const zone, const JSStringFinalizer *const fin, char16_t *const buf); JSStringFinalizer native_external_delete { native_external_deleter }; JSStringFinalizer native_external_static { native_external_noop }; } // namespace js } // namespace ircd std::string ircd::js::native(const JSString *const &s) { std::string ret(native_size(s) + 1, char()); native(s, &ret.front(), ret.size()); ret.resize(ret.size() - 1); return ret; } size_t ircd::js::native_size(const JSString *const &s) { return JS_GetStringEncodingLength(*cx, const_cast<JSString *>(s)); } size_t ircd::js::native(const JSString *const &s, char *const &buf, const size_t &max) { if(unlikely(!max)) return 0; ssize_t ret(s? JS_EncodeStringToBuffer(*cx, const_cast<JSString *>(s), buf, max) : 0); ret = std::max(ret, ssize_t(0)); ret = std::min(ret, ssize_t(max - 1)); buf[ret] = '\0'; return ret; } void ircd::js::native_external_deleter(JS::Zone *const zone, const JSStringFinalizer *const fin, char16_t *const buf) { log.debug("context(%p) zone(%p): string(%p) delete (dtor @%p) \"%s\"", (const void *)cx, (const void *)zone, (const void *)buf, (const void *)fin, locale::char16::conv(buf).c_str()); delete[] buf; } void ircd::js::native_external_noop(JS::Zone *const zone, const JSStringFinalizer *const fin, char16_t *const buf) { log.debug("string literal release (zone: %p fin: %p buf: %p)", (const void *)zone, (const void *)fin, (const void *)buf); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/error.h // ircd::js::jserror::jserror(const JS::Value &val) :ircd::js::error{generate_skip} ,val{val} { } ircd::js::jserror::jserror(JSObject *const &obj) :ircd::js::error{generate_skip} ,val{obj? JS::ObjectValue(*obj) : JS::UndefinedValue() } { } ircd::js::jserror::jserror(JSObject &obj) :ircd::js::error{generate_skip} ,val{JS::ObjectValue(obj)} { } ircd::js::jserror::jserror(generate_skip_t) :ircd::js::error(generate_skip) ,val{} { } ircd::js::jserror::jserror(const JSErrorReport &report) :ircd::js::error{generate_skip} ,val{} { create(report); } ircd::js::jserror::jserror(pending_t) :ircd::js::error{generate_skip} ,val{} { //if(unlikely(!restore_exception(*cx))) // throw error("No exception pending"); value val; if(unlikely(!JS_GetPendingException(*cx, &val))) throw error("Failed to recover pending exception"); object obj(val); const auto rp(JS_ErrorFromException(*cx, obj)); if(unlikely(!rp)) throw error("Error report from exception failed"); auto &report(*rp); generate_what_our(report); log.debug("jserror(%p): from pending [%s]: %s", (const void *)this, debug(report), what()); this->val = val; clear_exception(*cx); } void ircd::js::jserror::set_uncatchable() const { set_pending(); clear_exception(*cx); } void ircd::js::jserror::set_pending() const { JS_SetPendingException(*cx, &val); save_exception(*cx); } void ircd::js::jserror::generate(const JSExnType &type, const char *const &fmt, const va_rtti &ap) { ircd::exception::generate(fmt, ap); const auto msg(locale::char16::conv(what())); JSErrorReport report; //report.message = msg.c_str(); report.exnType = type; create(report); } void ircd::js::jserror::create(const JSErrorReport &report) { JSErrorReport cpy(report); create(cpy); } void ircd::js::jserror::create(JSErrorReport &report) { JS::AutoFilename fn; const auto col(report.column? nullptr : &report.column); const auto line(report.lineno? nullptr : &report.lineno); DescribeScriptedCaller(*cx, &fn, line, col); JS::RootedString msg { *cx, //JS_NewStringCopyUTF8N(*cx, report.message()) }; JS::RootedString file { *cx, JS_NewStringCopyZ(*cx, fn.get()?: "<unknown>") }; JS::RootedObject stack(*cx, nullptr); const auto type((JSExnType)report.exnType); if(!JS::CreateError(*cx, type, stack, file, report.lineno, report.column, &report, msg, &val)) { throw error("Failed to construct jserror exception!"); } JS::Rooted<JSObject *> obj(*cx); if(unlikely(!JS_ValueToObject(*cx, &val, &obj))) throw error("Failed to convert value to object on exception!"); JS::Rooted<JS::Value> msgv(*cx, JS::StringValue(msg)); if(unlikely(!JS_SetProperty(*cx, obj, "message", msgv))) throw error("Failed to set jserror.message property on exception!"); generate_what_our(report); } void ircd::js::jserror::generate_what_our(const JSErrorReport &report) { char linebuf[64]; snprintf(linebuf, sizeof(linebuf), "@%u+%u: ", report.lineno, report.column); const auto msg(report.message()); const auto empty(false); snprintf(ircd::exception::buf, sizeof(ircd::exception::buf), "%s%s%s%s", reflect((JSExnType)report.exnType), empty? "." : ": ", empty? "" : !report.lineno && !report.column? "" : linebuf, msg.c_str()); } void ircd::js::jserror::generate_what_js(const JSErrorReport &report) { //const auto str(native(::js::ErrorReportToString(*cx, const_cast<JSErrorReport *>(&report)))); //snprintf(ircd::exception::buf, sizeof(ircd::exception::buf), "%s", str.c_str()); } void ircd::js::replace_message(JSErrorReport &report, const char *fmt, ...) { va_list ap; va_start(ap, fmt); thread_local char buf[1024]; vsnprintf(buf, sizeof(buf), fmt, ap); const size_t ucsz(sizeof(buf) * 2); custom_ptr<char16_t> ucbuf(reinterpret_cast<char16_t *>(js_malloc(ucsz)), js_free); locale::char16::conv(buf, ucbuf.get(), ucsz); //custom_ptr<void> existing(const_cast<char16_t *>(report.message()), js_free); //report.ucmessage = ucbuf.release(); va_end(ap); } #if defined(IRCD_JS_FIX) void ::JSErrorReport::freeLinebuf() { js_free(const_cast<char16_t *>(linebuf())); } #endif // IRCD_JS_FIX #if defined(IRCD_JS_FIX) void ::JSErrorReport::freeMessage() { } #endif // IRCD_JS_FIX /////////////////////////////////////////////////////////////////////////////// // // ircd/js/compartment.h // ircd::js::compartment::compartment(const JSVersion &ver) :compartment{*cx, ver} { } ircd::js::compartment::compartment(context &c, const JSVersion &ver) :compartment { current_global(c)?: throw error("Cannot enter compartment without global"), c, ver } { } ircd::js::compartment::compartment(JSObject *const &obj, const JSVersion &ver) :compartment{obj, *cx, ver} { } ircd::js::compartment::compartment(JSObject *const &obj, context &c, const JSVersion &ver) :c{&c} ,prev{JS_EnterCompartment(c, obj)} ,ours{current_compartment(c)} ,cprev{static_cast<compartment *>(JS_GetCompartmentPrivate(ours))} { JS_SetCompartmentPrivate(ours, this); JS_SetVersionForCompartment(ours, ver); } ircd::js::compartment::compartment(compartment &&other) noexcept :c{std::move(other.c)} ,prev{std::move(other.prev)} ,ours{std::move(other.ours)} ,cprev{std::move(other.cprev)} { JS_SetCompartmentPrivate(ours, this); other.ours = nullptr; } ircd::js::compartment::~compartment() noexcept { // branch not taken on std::move() if(ours) { JS_SetCompartmentPrivate(ours, cprev); JS_LeaveCompartment(*c, prev); } } void ircd::js::for_each_compartment_our(const compartment::closure_our &closure) { for_each_compartment([&closure] (JSCompartment *const &c) { if(our(c)) closure(*our(c)); }); } void ircd::js::for_each_compartment(const compartment::closure &closure) { JS_IterateCompartments(*cx, const_cast<compartment::closure *>(&closure), compartment::handle_iterate); } void ircd::js::compartment::handle_iterate(JSContext *const cx, void *const priv, JSCompartment *const c) noexcept { const auto &closure(*static_cast<compartment::closure *>(priv)); closure(c); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/context.h // namespace ircd { namespace js { void handle_activity_ctypes(JSContext *, enum ::js::CTypesActivityType) noexcept; } // namespace js } // namespace ircd ircd::js::context::context(const struct opts &opts, JSContext *const &parent) :context { std::make_unique<struct opts>(opts), parent } { } ircd::js::context::context(std::unique_ptr<struct opts> opts, JSContext *const &parent) :custom_ptr<JSContext> { // Construct the context [this, &opts, &parent] { const auto ret(JS_NewContext(opts->max_bytes, opts->max_nursery_bytes, parent)); // Use their privdata pointer to point to our instance. We can then use our(JSContext*) // to get back to `this` instance. JS_SetContextPrivate(ret, this); return ret; }(), // Plant the destruction [](JSContext *const ctx) noexcept { if(!ctx) return; // Free the user's privdata managed object //delete static_cast<const priv_data *>(JS_GetSecondContextPrivate(ctx)); JS_DestroyContext(ctx); } } ,opts{std::move(opts)} ,tid{std::this_thread::get_id()} ,except{nullptr} ,state {{ 0, // Semaphore value starting at 0. phase::LEAVE, // LEAVE phase indicates nothing is running. irq::JS, // irq::JS is otherwise here in case JS triggers an interrupt. }} ,timer { std::bind(&context::handle_timeout, get()) } ,star{nullptr} { JS::SetWarningReporter(get(), handle_error); JS_AddInterruptCallback(get(), handle_interrupt); //JS_SetAccumulateTelemetryCallback(get(), handle_telemetry); JS::SetOutOfMemoryCallback(get(), handle_out_of_memory, nullptr); JS::SetLargeAllocationFailureCallback(get(), handle_large_allocation_failure, nullptr); JS_SetGCCallback(get(), handle_gc, nullptr); JS::SetGCSliceCallback(get(), handle_slice); JS_SetObjectsTenuredCallback(get(), handle_objects_tenured, nullptr); JS_AddFinalizeCallback(get(), handle_finalize, nullptr); JS_SetCompartmentNameCallback(get(), handle_compartment_name); JS_SetDestroyCompartmentCallback(get(), handle_compartment_destroy); JS_SetSweepZoneCallback(get(), handle_zone_sweep); JS_SetDestroyZoneCallback(get(), handle_zone_destroy); ::js::SetPreserveWrapperCallback(get(), handle_preserve_wrapper); JS_SetGrayGCRootsTracer(get(), handle_trace_gray, nullptr); JS_AddExtraGCRootsTracer(get(), handle_trace_extra, nullptr); ::js::SetActivityCallback(get(), handle_activity, this); ::js::SetCTypesActivityCallback(get(), handle_activity_ctypes); JS::SetEnqueuePromiseJobCallback(get(), handle_promise_enqueue_job, this); JS::SetPromiseRejectionTrackerCallback(get(), handle_promise_rejection_tracker, this); JS::SetAsyncTaskCallbacks(get(), handle_start_async_task, handle_finish_async_task); JS::SetGetIncumbentGlobalCallback(get(), handle_get_incumbent_global); ::js::SetStopwatchStartCallback(get(), handle_stopwatch_start, this); ::js::SetStopwatchCommitCallback(get(), handle_stopwatch_commit, this); ::js::SetGetPerformanceGroupsCallback(get(), handle_get_performance_groups, nullptr); JS::SetBuildIdOp(get(), handle_set_build_id_op); timer.set(this->opts->timer_limit); JS_SetNativeStackQuota(get(), this->opts->code_stack_max, this->opts->trusted_stack_max, this->opts->untrusted_stack_max); JS_SetParallelParsingEnabled(get(), this->opts->concurrent_parsing); JS_SetOffthreadIonCompilationEnabled(get(), this->opts->concurrent_jit); JS_SetGCZeal(get(), this->opts->gc_zeal_mode, this->opts->gc_zeal_freq); if(!JS::InitSelfHostedCode(get())) throw error("JS::InitSelfHostedCode: failure."); } ircd::js::context::~context() noexcept { // If items still exist on the tracing lists at this point (runtime shutdown): // that is bad. The objects are still reachable but should have removed themselves. if(unlikely(!tracing.heap.empty())) { // When the context terminates with an active exception the exception itself // can be ignored and leak. This case should eventually lead to IRCd termination... if(std::uncaught_exception() && tracing.heap.size() == 1) return; log.critical("context(%p): !!! LEAK !!! %zu traceable items still reachable on the heap", (const void *)this, tracing.heap.size()); assert(0); } } void ircd::js::leave(context &c) { c.timer.cancel(); // Load the state to keep the current sem value up to date. // This thread is the only writer to that value. auto state(c.state.load(std::memory_order_relaxed)); // The LEAVE phase locks out the interruptor state.phase = phase::LEAVE; // The LEAVE is released and the current phase seen by interruptor is acquired. state = c.state.exchange(state, std::memory_order_acq_rel); // The executor (us) must check if the interruptor (them) has committed to an interrupt // targeting the JS run we are now leaving. JS may have exited after the commitment // and before the interrupt arrival. if(state.phase == phase::INTR) assert(interrupt_poll(c)); } void ircd::js::enter(context &c) { // State was already acquired by the last leave(); auto state(c.state.load(std::memory_order_relaxed)); // Increment the semaphore for the next execution; ++state.sem; // Set the IRQ to JS in case we don't trigger an interrupt, the handler // will see a correct value. state.irq = irq::JS; state.phase = phase::ENTER; // Commit to next execution. c.state.store(state, std::memory_order_release); c.timer.start(); } bool ircd::js::interrupt(context &c, const irq &req) { if(req == irq::NONE) return false; // Acquire the execution state. Proceed if something was running. const auto state(c.state.load(std::memory_order_acquire)); if(state.phase != phase::ENTER) return false; // The expected value of the state to transact. struct context::state in { state.sem, // Expect the sem value to match, else the execution has changed. phase::ENTER, // Expect the execution to be occurring. irq::JS, // JS is always expected here instead of NONE. }; // The restatement after the transaction. const struct context::state out { state.sem, // Keep the same sem value. phase::INTR, // Indicate our commitment to interrupting. req // Include the irq type. }; // Attempt commitment to interrupt here. static const auto order_fail(std::memory_order_relaxed); static const auto order_success(std::memory_order_acq_rel); if(!c.state.compare_exchange_strong(in, out, order_success, order_fail)) { // To reliably read from `in` here, order_fail should not be relaxed. return false; } // Commitment now puts the burden on the executor to not allow this interrupt to bleed into // the next execution, even if JS has already exited before its arrival. JS_RequestInterruptCallback(c); return true; } bool ircd::js::interrupt_poll(const context &c) { return JS_CheckForInterrupt(c); } bool ircd::js::restore_exception(context &c) { if(unlikely(!c.except)) return false; JS_RestoreExceptionState(c, c.except); c.except = nullptr; return true; } void ircd::js::save_exception(context &c) { if(c.except) { //log.warning("save_exception(): Dropping unrestored exception @ %p", (const void *)c.except); //JS_DropExceptionState(*cx, c.except); throw error("(internal error): Won't overwrite saved exception @ %p", (const void *)c.except); } c.except = JS_SaveExceptionState(c); } bool ircd::js::maybe_gc(context &c) noexcept { // JS_MaybeGC dereferences the context's current zone without checking if the context // is even in a compartment/has a current zone; we must check here. if(!current_zone(c)) return false; JS_MaybeGC(c); return true; } void ircd::js::out_of_memory(context &c) { JS_ReportOutOfMemory(c); } void ircd::js::allocation_overflow(context &c) { JS_ReportAllocationOverflow(c); } uint32_t ircd::js::get(context &c, const JSGCParamKey &key) { //return JS_GetGCParameterForThread(c, key); // broken return JS_GetGCParameter(c, key); } void ircd::js::set(context &c, const JSGCParamKey &key, const uint32_t &val) { //JS_SetGCParameterForThread(c, key, val); // broken JS_SetGCParameter(c, key, val); } bool ircd::js::run_gc(context &c) noexcept { JS_GC(c); return true; } // // Callback surface // bool ircd::js::context::handle_set_build_id_op(JS::BuildIdCharVector *const vector) noexcept { static const uint32_t build_id { 0 }; assert(vector); const bool resized(vector->resize(sizeof(build_id))); assert(resized); memcpy(vector->begin(), &build_id, sizeof(build_id)); return true; } JSObject * ircd::js::context::handle_get_incumbent_global(JSContext *const cx) noexcept { auto &c(our(cx)); log.debug("context(%p): get incumbent global (current_global: %p)", (const void *)cx, current_global(c)); return current_global(c); } bool ircd::js::context::handle_start_async_task(JSContext *const cx, JS::AsyncTask *const task) noexcept { log.debug("context(%p): async task(%p) START", (const void *)cx, task); return true; } bool ircd::js::context::handle_finish_async_task(JS::AsyncTask *const task) noexcept { log.debug("context(%p): async task(%p) FINISH", nullptr, task); return true; } bool ircd::js::context::handle_promise_enqueue_job(JSContext *const c, JS::HandleObject job, JS::HandleObject allocation_site, JS::HandleObject incumbent_global, void *const priv) noexcept { log.debug("context(%p): promise enqueue job (priv: %p)", (const void *)c, priv); // dump(job); // printf("--\n"); // dump(allocation_site); return true; } void ircd::js::context::handle_promise_rejection_tracker(JSContext *const c, JS::HandleObject promise, PromiseRejectionHandlingState state, void *const priv) noexcept { log.debug("context(%p): promise rejection track (state: %s priv: %p)", (const void *)c, reflect(state), priv); } void ircd::js::context::handle_activity(void *const priv, const bool active) noexcept { assert(priv); auto &c(*static_cast<struct context *>(priv)); //const auto tid(std::this_thread::get_id()); const auto &msg(active? "ENTER" : "LEAVE"); log.debug("context(%p): %s", //ircd::string(tid).c_str(), (const void *)c.ptr(), msg); } void ircd::js::handle_activity_ctypes(JSContext *const cx, const ::js::CTypesActivityType t) noexcept { log.debug("context(%p): %s", (const void *)cx, reflect(t)); } bool ircd::js::context::handle_preserve_wrapper(JSContext *const cx, JSObject *const obj) noexcept { log.debug("context(%p): (object: %p) preserve wrapper", (const void *)cx, (const void *)obj); return true; } void ircd::js::context::handle_gc(JSContext *const cx, const JSGCStatus status, void *const priv) noexcept { log.debug("context(%p): GC %s (priv: %p)", (const void *)cx, reflect(status), (const void *)priv); } void ircd::js::context::handle_slice(JSContext *const cx, JS::GCProgress progress, const JS::GCDescription &d) noexcept { log.debug("context(%p): SLICE: %s (description: %p)", (const void *)cx, reflect(progress), (const void *)&d); } void ircd::js::context::handle_objects_tenured(JSContext *const cx, void *const priv) noexcept { log.debug("context(%p): objects tenured (priv: %p)", (const void *)cx, (const void *)priv); } void ircd::js::context::handle_finalize(JSFreeOp *const fop, const JSFinalizeStatus status, const bool is_compartment, void *const priv) noexcept { log.debug("context(%p): fop(%p): %s %s (priv: %p)", (const void *)cx, (const void *)fop, reflect(status), is_compartment? "COMPARTMENT" : "", priv); } void ircd::js::context::handle_compartment_destroy(JSFreeOp *const fop, JSCompartment *const compartment) noexcept { auto *const c(our(compartment)); auto *const cx(c? &static_cast<context &>(*c) : (context *)nullptr); log.debug("context(%p): compartment: %p %s%sdestroy: fop(%p)", cx? (const void *)cx->ptr() : (const void *)nullptr, (const void *)compartment, ::js::IsSystemCompartment(compartment)? "[system] " : "", ::js::IsAtomsCompartment(compartment)? "[atoms] " : "", (const void *)fop); } void ircd::js::context::handle_compartment_name(JSContext *const cx, JSCompartment *const compartment, char *const buf, const size_t max) noexcept { log.debug("context(%p): comaprtment: %p (buf@%p: max: %zu)", (const void *)cx, (const void *)compartment, (const void *)buf, max); } void ircd::js::context::handle_zone_destroy(JS::Zone *const zone) noexcept { log.debug("context(%p): zone: %p %s%sdestroy", (const void *)cx, (const void *)zone, ::js::IsSystemZone(zone)? "[system] " : "", ::js::IsAtomsZone(zone)? "[atoms] " : ""); } void ircd::js::context::handle_zone_sweep(JS::Zone *const zone) noexcept { log.debug("context(%p): zone: %p %s%ssweep", (const void *)cx, (const void *)zone, ::js::IsSystemZone(zone)? "[system] " : "", ::js::IsAtomsZone(zone)? "[atoms] " : ""); } void ircd::js::context::handle_weak_pointer_compartment(JSContext *const cx, JSCompartment *const comp, void *const data) noexcept { log.debug("context(%p): weak pointer compartment(%p) %p", (const void *)cx, (const void *)comp, data); } void ircd::js::context::handle_weak_pointer_zone(JSContext *const cx, void *const data) noexcept { log.debug("context(%p): weak pointer zone %p", (const void *)cx, data); } void ircd::js::context::handle_trace_extra(JSTracer *const tracer, void *const priv) noexcept { log.debug("context(%p): tracer(%p) %s: extra (priv: %p) count: %zu", (const void *)cx, (const void *)tracer, debug(*tracer).c_str(), priv, cx->tracing.heap.size()); if(unlikely(std::uncaught_exception() && cx->tracing.heap.size() == 1)) { log.warning("context(%p): tracer(%p) %s: extra skipped due to uncaught exception", (const void *)cx, (const void *)tracer, debug(*tracer).c_str()); return; } cx->tracing(tracer); } void ircd::js::context::handle_trace_gray(JSTracer *const tracer, void *const priv) noexcept { log.debug("context(%p): tracer(%p): gray (priv: %p)", (const void *)cx, (const void *)tracer, priv); } void ircd::js::context::handle_large_allocation_failure(void *const priv) noexcept { log.error("context(%p): Large allocation failure (priv: %p)", (const void *)cx, priv); assert(0); //TODO: XXX } void ircd::js::context::handle_out_of_memory(JSContext *const cx, void *const priv) noexcept { log.error("context(%p): out of memory", (const void *)cx); assert(0); //TODO: XXX } bool ircd::js::context::handle_stopwatch_start(const uint64_t us, void *const priv) noexcept { assert(priv); auto &c(*static_cast<context *>(priv)); log.debug("context(%p): stopwatch start (microseconds: %lu priv: %p)", (const void *)c.get(), us, priv); return true; } bool ircd::js::context::handle_stopwatch_commit(const uint64_t us, ::js::PerformanceGroupVector &vec, void *const priv) noexcept { assert(priv); auto &c(*static_cast<context *>(priv)); log.debug("context(%p): stopwatch commit (microseconds: %lu priv: %p)", (const void *)c.get(), us, priv); return true; } bool ircd::js::context::handle_get_performance_groups(JSContext *const cx, ::js::PerformanceGroupVector &vec, void *const priv) noexcept { log.debug("context(%p): get performance groups (priv: %p)", (const void *)cx, priv); return true; } void ircd::js::context::handle_telemetry(const int id, const uint32_t sample, const char *const key) noexcept { //const auto tid(std::this_thread::get_id()); log.debug("context(%p): telemetry(%02d) %s: %u %s", //ircd::string(tid).c_str(), (const void *)cx, id, reflect_telemetry(id), sample, key?: ""); } void ircd::js::context::handle_timeout(JSContext *const cx) noexcept { // At this time there is no yield logic so if the timer calls the script is terminated. auto &c(our(cx)); interrupt(c, irq::TERMINATE); } bool ircd::js::context::handle_interrupt(JSContext *const cx) noexcept { auto &c(our(cx)); auto state(c.state.load(std::memory_order_acquire)); log.debug("context(%p): Interrupt: IRQ[%u] phase[%u]", (const void *)cx, uint(state.irq), uint(state.phase)); // Spurious interrupt; ignore. if(unlikely(state.phase != phase::INTR && state.phase != phase::ENTER)) { log.warning("context(%p): Spurious interrupt (irq: %02x)", (const void *)cx, uint(state.irq)); return true; } // After the interrupt is handled the phase indicates entry back to JS, // IRQ is left indicating JS in case we don't trigger the next interrupt. const unwind interrupt_return([&c, &state] { state.phase = phase::ENTER; state.irq = irq::JS; c.state.store(state, std::memory_order_release); }); // Call the user hook if available if(c.on_intr) { // The user's handler returns -1 for non-overriding behavior const auto ret(c.on_intr(state.irq)); if(ret != -1) return ret; } switch(state.irq) { case irq::JS: case irq::USER: return true; case irq::YIELD: ctx::yield(); return true; case irq::TERMINATE: return false; default: case irq::NONE: assert(0); return false; } } void ircd::js::context::handle_error(JSContext *const cx, JSErrorReport *const report) noexcept try { assert(report); const log::facility facility { JSREPORT_IS_WARNING(report->flags)? log::WARNING: log::DEBUG }; log(facility, "context(%p): %s", (const void *)cx, debug(*report).c_str()); auto &c(our(cx)); if(JSREPORT_IS_EXCEPTION(report->flags)) { // If except state is saved this is a redundant report from an exception as our // intertwined c++ -> js -> c++ stack blows up. if(c.except) return; // This is likely an uncaught exception from a throw in JS. We create a new exception // object because we lost the one from the user and make that pending now in case some // other opportunity for the user to catch this is presented. jserror e(*report); e.set_pending(); return; } if(report->exnType == JSEXN_INTERNALERR) { static const std::string msg("god save jsapi"); internal_error ie("%s", msg.c_str()); ie.set_pending(); return; } switch(report->errorNumber) { case 61: // JSAPI's code for interruption { report->exnType = JSEXN_ERR; report->flags |= JSREPORT_EXCEPTION; replace_message(*report, "interrupted @ line[%u] col[%u]", report->lineno, report->column); jserror e(*report); e.set_uncatchable(); return; } case 105: // JSAPI's code for user reported error { report->exnType = JSEXN_INTERNALERR; report->flags |= JSREPORT_EXCEPTION; replace_message(*report, "(BUG) Host exception"); jserror e(*report); e.set_uncatchable(); return; } } } catch(const std::exception &e) { log.critical("triple fault: %s\n", e.what()); ircd::terminate(); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/timer.h // ircd::js::timer::timer(const callback &timeout) :finished{false} ,timeout{timeout} ,started{time_point::min()} ,limit{0us} ,state {{ 0, false }} ,thread{std::bind(&timer::worker, this)} { } ircd::js::timer::~timer() noexcept { // This is all on the main IRCd thread, and the only point when it waits on the timer // thread. That's okay as long as this is on IRCd shutdown etc. std::unique_lock<decltype(mutex)> lock(mutex); finished = true; // Wait for timer thread to exit. Notify before the unlock() could double tap, // but should always make for a reliable way to do this shutdown. cond.notify_all(); lock.unlock(); thread.join(); } ircd::js::timer::time_point ircd::js::timer::start() { // The counter is incremented indicating a new timing request, invalidating // anything the timer was previously doing. auto state(this->state.load(std::memory_order_relaxed)); ++state.sem; state.running = true; // Commit to starting a new timer operation, unconditionally overwrite previous. started = steady_clock::now(); this->state.store(state, std::memory_order_release); // The timing thread is notified here. It may have already started on our new // request. However, this notifcation must not be delayed and the timing thread // must wake up soon/now. cond.notify_one(); return started; } bool ircd::js::timer::cancel() { // Relaxed acquire of the state to get the sem value. auto state(this->state.load(std::memory_order_relaxed)); // Expect the timer to be running and order the timer to idle with an invalidation. struct state in { state.sem, true }; const struct state out { state.sem + 1, false }; // Commit to cancellation. On a successful state transition, wake up the thread. static const auto order_fail(std::memory_order_relaxed); static const auto order_success(std::memory_order_acq_rel); if(this->state.compare_exchange_strong(in, out, order_success, order_fail)) { cond.notify_one(); return true; } else return false; } void ircd::js::timer::set(const callback &timeout) { this->timeout = timeout; std::atomic_thread_fence(std::memory_order_release); } void ircd::js::timer::set(const microseconds &limit) { this->limit.store(limit, std::memory_order_release); } void ircd::js::timer::worker() { // This lock is only ever held by this thread except during a finish condition. // Notifications to the condition are only broadcast by the main thread. std::unique_lock<decltype(mutex)> lock(mutex); while(likely(!finished)) handle(lock); } void ircd::js::timer::handle(std::unique_lock<std::mutex> &lock) { struct state ours, theirs; const auto idle_condition([this, &ours] { if(unlikely(finished)) return true; // Acquire the request operation ours = this->state.load(std::memory_order_acquire); // Require a start time if(started == time_point::min()) return false; return ours.running; }); const auto cancel_condition([this, &ours, &theirs] { if(unlikely(finished)) return true; // Acquire the request state and compare it to our saved state to see // if the counter has changed or if there is cancellation. theirs = this->state.load(std::memory_order_consume); return ours.sem != theirs.sem || !theirs.running; }); // Wait for a running condition into `in` cond.wait(lock, idle_condition); if(unlikely(finished)) return; // Wait for timeout or cancellation const auto limit(this->limit.load(std::memory_order_consume)); if(cond.wait_until(lock, started + limit, cancel_condition)) { // A cancel or increment of the counter into the next request has occurred. // This thread will go back to sleep in the idle_condition unless or until // the next start() is triggered. return; } // A timeout has occurred. This is the last chance for a belated cancellation to be // observed. If a transition from running to !running can take place on this counter // value, the user has not invalidated this request and desires the timeout. assert(ours.running); const struct state out { ours.sem, false }; static const auto order_fail(std::memory_order_relaxed); static const auto order_success(std::memory_order_acq_rel); if(state.compare_exchange_strong(ours, out, order_success, order_fail)) if(likely(timeout)) timeout(); } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/tracing.h // namespace ircd { namespace js { void trace_heap(JSTracer *const &tracer, tracing::thing &thing); } // namespace js } // namespace ircd ircd::js::tracing::tracing() { } ircd::js::tracing::~tracing() noexcept { if(std::uncaught_exception() && heap.size() == 1) return; assert(heap.empty()); } void ircd::js::tracing::operator()(JSTracer *const &tracer) { for(auto &thing : cx->tracing.heap) trace_heap(tracer, thing); } void ircd::js::trace_heap(JSTracer *const &tracer, tracing::thing &thing) { if(thing.type != jstype::OBJECT) log.debug("context(%p): tracer(%p): heap<%s> @ %p", nullptr, //(const void *)tracer->runtime(), (const void *)tracer, reflect(thing.type), (const void *)thing.ptr); if(likely(thing.ptr)) switch(thing.type) { case jstype::VALUE: { const auto ptr(reinterpret_cast<JS::Heap<JS::Value> *>(thing.ptr)); if(!ptr->address()) break; //JS_CallValueTracer(tracer, ptr, "heap"); break; } case jstype::OBJECT: { const auto ptr(reinterpret_cast<JS::Heap<JSObject *> *>(thing.ptr)); //if(!ptr->get()) break; log.debug("context(%p): tracer(%p): heap<%s> @ %p object(%p trap: %p '%s')", nullptr, //(const void *)tracer->runtime(), (const void *)tracer, reflect(thing.type), (const void *)thing.ptr, (const void *)ptr->get(), (const void *)(has_jsclass(*ptr)? &jsclass(*ptr) : nullptr), has_jsclass(*ptr)? jsclass(*ptr).name : "<no trap>"); //JS_CallObjectTracer(tracer, ptr, "heap"); break; } case jstype::FUNCTION: { const auto ptr(reinterpret_cast<JS::Heap<JSFunction *> *>(thing.ptr)); //if(!ptr->get()) break; //JS_CallFunctionTracer(tracer, ptr, "heap"); break; } case jstype::SCRIPT: { const auto ptr(reinterpret_cast<JS::Heap<JSScript *> *>(thing.ptr)); //if(!ptr->get()) break; //JS_CallScriptTracer(tracer, ptr, "heap"); break; } case jstype::STRING: { const auto ptr(reinterpret_cast<JS::Heap<JSString *> *>(thing.ptr)); //if(!ptr->get()) break; //JS_CallStringTracer(tracer, ptr, "heap"); break; } case jstype::ID: { const auto ptr(reinterpret_cast<JS::Heap<jsid> *>(thing.ptr)); //if(!ptr->address()) break; //JS_CallIdTracer(tracer, ptr, "heap"); break; } case jstype::SYMBOL: { const auto ptr(reinterpret_cast<JS::Heap<JS::Symbol *> *>(thing.ptr)); //if(!ptr->get()) break; break; } default: { log.warning("context(%p): tracer(%p): heap<%s> @ %p", nullptr, //(const void *)tracer->runtime(), (const void *)tracer, reflect(thing.type), (const void *)thing.ptr); break; } } } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/debug.h // void ircd::js::log_gcparams() { for(int i(0); i < 50; ++i) { const auto key(static_cast<JSGCParamKey>(i)); const char *const name(reflect(key)); if(!strlen(name)) continue; // These trigger assertion failures switch(key) { case JSGC_NUMBER: case JSGC_MAX_MALLOC_BYTES: case JSGC_ALLOCATION_THRESHOLD: continue; default: break; } log.debug("context(%p) %s => %u", (const void *)cx, name, get(*cx, key)); } } void ircd::js::backtrace() { #ifdef JS_DEBUG ::js::DumpBacktrace(*cx); #endif } void ircd::js::dump_promise(const JS::HandleObject &promise) { #ifdef JS_DEBUG JS::DumpPromiseAllocationSite(*cx, promise); JS::DumpPromiseResolutionSite(*cx, promise); #endif } void ircd::js::dump(const JSString *const &v) { #ifdef JS_DEBUG ::js::DumpString(const_cast<JSString *>(v)); #endif } void ircd::js::dump(const JSAtom *const &v) { #ifdef JS_DEBUG ::js::DumpAtom(const_cast<JSAtom *>(v)); #endif } void ircd::js::dump(const JSObject *const &v) { #ifdef JS_DEBUG ::js::DumpObject(const_cast<JSObject *>(v)); #endif } void ircd::js::dump(const JS::Value &v) { #ifdef JS_DEBUG ::js::DumpValue(v); #endif } void ircd::js::dump(const jsid &v) { #ifdef JS_DEBUG ::js::DumpId(v); #endif } void ircd::js::dump(const JSContext *v) { #ifdef JS_DEBUG ::js::DumpPC(const_cast<JSContext *>(v)); #endif } void ircd::js::dump(const JSScript *const &v) { #ifdef JS_DEBUG ::js::DumpScript(*cx, const_cast<JSScript *>(v)); #endif } void ircd::js::dump(const char16_t *const &v, const size_t &len) { #ifdef JS_DEBUG ::js::DumpChars(v, len); #endif } void ircd::js::dump(const ::js::InterpreterFrame *v) { #ifdef JS_DEBUG ::js::DumpInterpreterFrame(*cx, const_cast<::js::InterpreterFrame *>(v)); #endif } std::string ircd::js::debug(const JSTracer &t) { return t.isMarkingTracer()? "MARKING": t.isWeakMarkingTracer()? "WEAKMARKING": t.isTenuringTracer()? "TENURING": t.isCallbackTracer()? "CALLBACK": "UNKNOWN"; } std::string ircd::js::debug(const JSErrorReport &r) { std::stringstream ss; if(JSREPORT_IS_WARNING(r.flags)) ss << "WARNING "; if(JSREPORT_IS_EXCEPTION(r.flags)) ss << "EXCEPTION "; if(JSREPORT_IS_STRICT(r.flags)) ss << "STRICT "; if(r.isMuted) ss << "MUTED "; if(r.filename) ss << "file[" << r.filename << "] "; if(r.lineno) ss << "line[" << r.lineno << "] "; if(r.column) ss << "col[" << r.column << "] "; if(r.linebuf()) ss << "code[" << std::u16string(r.linebuf()) << "] "; //if(r.tokenptr) // ss << "toke[" << r.tokenptr << "] "; if(r.errorNumber) ss << "errnum[" << r.errorNumber << "] "; if(r.exnType) ss << reflect(JSExnType(r.exnType)) << " "; // if(r.message()) // ss << "\"" << locale::char16::conv(r.message()) << "\" "; // for(auto it(r.messageArgs); it && *it; ++it) // ss << "\"" << locale::char16::conv(*it) << "\" "; return ss.str(); } std::string ircd::js::debug(const JS::HandleObject &o) { std::stringstream ss; if(JS_IsGlobalObject(o)) ss << "Global "; if(JS_IsNative(o)) ss << "Native "; if(JS::IsCallable(o)) ss << "Callable "; if(JS::IsConstructor(o)) ss << "Constructor "; if(JS_ObjectIsFunction(*cx, o)) ss << "Function "; bool ret; if(JS_IsExtensible(*cx, o, &ret)) if(ret) ss << "Extensible "; if(JS_IsArrayObject(*cx, o, &ret)) if(ret) ss << "Array "; if(JS::IsArray(*cx, o, &ret)) if(ret) ss << "Array.isArray "; if(JS_ObjectIsRegExp(*cx, o, &ret)) if(ret) ss << "RegExp "; if(JS_ObjectIsDate(*cx, o, &ret)) if(ret) ss << "Date "; if(JS::IsPromiseObject(o)) { ss << "Promise[#" << JS::GetPromiseID(o); ss << " " << reflect(JS::GetPromiseState(o)); ss << "] "; } if(JS_IsArrayBufferObject(o)) { ss << "ArrayBuffer[" << JS_GetArrayBufferByteLength(o); ss << " " << JS_IsDetachedArrayBufferObject(o)? "DETACHED" : "ATTACHED"; ss << " " << JS_IsMappedArrayBufferObject(o)? "MAPPED" : "UNMAPPED"; ss << "] "; } if(JS_IsSharedArrayBufferObject(o)) { ss << "SharedArrayBuffer[" << JS_GetSharedArrayBufferByteLength(o); ss << " " << JS_IsDetachedArrayBufferObject(o)? "DETACHED" : "ATTACHED"; ss << " " << JS_IsMappedArrayBufferObject(o)? "MAPPED" : "UNMAPPED"; ss << "] "; } if(JS_IsArrayBufferViewObject(o)) { ss << "ArrayBufferView[" << JS_GetArrayBufferViewByteLength(o); ss << "] "; } if(JS_IsDataViewObject(o)) { ss << "DataView[" << JS_GetDataViewByteLength(o); ss << " @" << JS_GetDataViewByteOffset(o); ss << "] "; } return ss.str(); } std::string ircd::js::debug(const JS::Value &v) { std::stringstream ss; if(v.isNull()) ss << "Null "; if(v.isUndefined()) ss << "Undefined "; if(v.isBoolean()) ss << "Boolean "; if(v.isTrue()) ss << "TrueValue "; if(v.isFalse()) ss << "FalseValue "; if(v.isNumber()) ss << "Number "; if(v.isDouble()) ss << "Double "; if(v.isInt32()) ss << "Int32 "; if(v.isString()) ss << "String "; if(v.isSymbol()) ss << "Symbol "; if(v.isObject()) ss << "Object "; if(v.isObject()) { JS::RootedObject obj(*cx); JS::RootedValue rv(*cx, v); if(JS_ValueToObject(*cx, rv, &obj)) ss << "(" << debug(obj) << ") "; } return ss.str(); } const char * ircd::js::reflect_telemetry(const int &id) { switch(id) { case JS_TELEMETRY_GC_REASON: return "GC_REASON"; case JS_TELEMETRY_GC_MS: return "GC_MS"; case JS_TELEMETRY_GC_BUDGET_MS: return "GC_BUDGET_MS"; case JS_TELEMETRY_GC_ANIMATION_MS: return "GC_ANIMATION_MS"; case JS_TELEMETRY_GC_MAX_PAUSE_MS: return "GC_MAX_PAUSE_MS"; case JS_TELEMETRY_GC_MARK_MS: return "GC_MARK_MS"; case JS_TELEMETRY_GC_SWEEP_MS: return "GC_SWEEP_MS"; case JS_TELEMETRY_GC_MARK_ROOTS_MS: return "GC_MARK_ROOTS_MS"; case JS_TELEMETRY_GC_MARK_GRAY_MS: return "GC_MARK_GRAY_MS"; case JS_TELEMETRY_GC_SLICE_MS: return "GC_SLICE_MS"; case JS_TELEMETRY_GC_SLOW_PHASE: return "GC_SLOW_PHASE"; case JS_TELEMETRY_GC_MMU_50: return "GC_MMU_50"; case JS_TELEMETRY_GC_RESET: return "GC_RESET"; case JS_TELEMETRY_GC_INCREMENTAL_DISABLED: return "GC_INCREMENTAL_DISABLED"; case JS_TELEMETRY_GC_NON_INCREMENTAL: return "GC_NON_INCREMENTAL"; case JS_TELEMETRY_GC_SCC_SWEEP_TOTAL_MS: return "GC_SCC_SWEEP_TOTAL_MS"; case JS_TELEMETRY_GC_SCC_SWEEP_MAX_PAUSE_MS: return "GC_SCC_SWEEP_MAX_PAUSE_MS"; case JS_TELEMETRY_GC_MINOR_REASON: return "GC_MINOR_REASON"; case JS_TELEMETRY_GC_MINOR_REASON_LONG: return "GC_MINOR_REASON_LONG"; case JS_TELEMETRY_GC_MINOR_US: return "GC_MINOR_US"; case JS_TELEMETRY_DEPRECATED_LANGUAGE_EXTENSIONS_IN_CONTENT: return "DEPRECATED_LANGUAGE_EXTENSIONS_IN_CONTENT"; case JS_TELEMETRY_DEPRECATED_LANGUAGE_EXTENSIONS_IN_ADDONS: return "DEPRECATED_LANGUAGE_EXTENSIONS_IN_ADDONS"; case JS_TELEMETRY_ADDON_EXCEPTIONS: return "ADDON_EXCEPTIONS"; } return ""; } const char * ircd::js::reflect_prop(const uint &flag) { switch(flag) { case JSPROP_ENUMERATE: return "JSPROP_ENUMERATE"; case JSPROP_READONLY: return "JSPROP_READONLY"; case JSPROP_PERMANENT: return "JSPROP_PERMANENT"; case JSPROP_PROPOP_ACCESSORS: return "JSPROP_PROPOP_ACCESSORS"; case JSPROP_GETTER: return "JSPROP_GETTER"; case JSPROP_SETTER: return "JSPROP_SETTER"; case JSPROP_SHARED: return "JSPROP_SHARED"; case JSPROP_INTERNAL_USE_BIT: return "JSPROP_INTERNAL_USE_BIT"; case JSFUN_STUB_GSOPS: return "JSFUN_STUB_GSOPS"; case JSFUN_CONSTRUCTOR: return "JSFUN_CONSTRUCTOR"; case JSPROP_REDEFINE_NONCONFIGURABLE: return "JSPROP_REDEFINE_NONCONFIGURABLE"; case JSPROP_RESOLVING: return "JSPROP_RESOLVING"; case JSPROP_IGNORE_ENUMERATE: return "JSPROP_IGNORE_ENUMERATE"; case JSPROP_IGNORE_READONLY: return "JSPROP_IGNORE_READONLY"; case JSPROP_IGNORE_PERMANENT: return "JSPROP_IGNORE_PERMANENT"; case JSPROP_IGNORE_VALUE: return "JSPROP_IGNORE_VALUE"; } return ""; } const char * ircd::js::reflect(const JS::PromiseState &state) { switch(state) { case JS::PromiseState::Pending: return "Pending"; case JS::PromiseState::Fulfilled: return "Fulfilled"; case JS::PromiseState::Rejected: return "Rejected"; } return ""; } const char * ircd::js::reflect(const PromiseRejectionHandlingState &state) { switch(state) { case PromiseRejectionHandlingState::Unhandled: return "Unhandled"; case PromiseRejectionHandlingState::Handled: return "Handled"; } return ""; } const char * ircd::js::reflect(const ::js::CTypesActivityType &t) { using namespace ::js; switch(t) { case CTYPES_CALL_BEGIN: return "CTYPES_CALL_BEGIN"; case CTYPES_CALL_END: return "CTYPES_CALL_END"; case CTYPES_CALLBACK_BEGIN: return "CTYPES_CALLBACK_BEGIN"; case CTYPES_CALLBACK_END: return "CTYPES_CALLBACK_END"; } return ""; } const char * ircd::js::reflect(const JSFinalizeStatus &s) { switch(s) { case JSFINALIZE_GROUP_START: return "GROUP_START"; case JSFINALIZE_GROUP_END: return "GROUP_END"; case JSFINALIZE_COLLECTION_END: return "COLLECTION_END"; } return ""; } const char * ircd::js::reflect(const JSGCParamKey &s) { switch(s) { case JSGC_MAX_BYTES: return "JSGC_MAX_BYTES"; case JSGC_MAX_MALLOC_BYTES: return "JSGC_MAX_MALLOC_BYTES"; case JSGC_BYTES: return "JSGC_BYTES"; case JSGC_NUMBER: return "JSGC_NUMBER"; case JSGC_MODE: return "JSGC_MODE"; case JSGC_UNUSED_CHUNKS: return "JSGC_UNUSED_CHUNKS"; case JSGC_TOTAL_CHUNKS: return "JSGC_TOTAL_CHUNKS"; case JSGC_SLICE_TIME_BUDGET: return "JSGC_SLICE_TIME_BUDGET"; case JSGC_MARK_STACK_LIMIT: return "JSGC_MARK_STACK_LIMIT"; case JSGC_HIGH_FREQUENCY_TIME_LIMIT: return "JSGC_HIGH_FREQUENCY_TIME_LIMIT"; case JSGC_HIGH_FREQUENCY_LOW_LIMIT: return "JSGC_HIGH_FREQUENCY_LOW_LIMIT"; case JSGC_HIGH_FREQUENCY_HIGH_LIMIT: return "JSGC_HIGH_FREQUENCY_HIGH_LIMIT"; case JSGC_HIGH_FREQUENCY_HEAP_GROWTH_MAX: return "JSGC_HIGH_FREQUENCY_HEAP_GROWTH_MAX"; case JSGC_HIGH_FREQUENCY_HEAP_GROWTH_MIN: return "JSGC_HIGH_FREQUENCY_HEAP_GROWTH_MIN"; case JSGC_LOW_FREQUENCY_HEAP_GROWTH: return "JSGC_LOW_FREQUENCY_HEAP_GROWTH"; case JSGC_DYNAMIC_HEAP_GROWTH: return "JSGC_DYNAMIC_HEAP_GROWTH"; case JSGC_DYNAMIC_MARK_SLICE: return "JSGC_DYNAMIC_MARK_SLICE"; case JSGC_ALLOCATION_THRESHOLD: return "JSGC_ALLOCATION_THRESHOLD"; case JSGC_MIN_EMPTY_CHUNK_COUNT: return "JSGC_MIN_EMPTY_CHUNK_COUNT"; case JSGC_MAX_EMPTY_CHUNK_COUNT: return "JSGC_MAX_EMPTY_CHUNK_COUNT"; case JSGC_COMPACTING_ENABLED: return "JSGC_COMPACTING_ENABLED"; case JSGC_REFRESH_FRAME_SLICES_ENABLED: return "JSGC_REFRESH_FRAME_SLICES_ENABLED"; } return ""; } const char * ircd::js::reflect(const JSGCStatus &s) { switch(s) { case JSGC_BEGIN: return "BEGIN"; case JSGC_END: return "END"; } return ""; } const char * ircd::js::reflect(const JSGCMode &s) { switch(s) { case JSGC_MODE_GLOBAL: return "GLOBAL"; case JSGC_MODE_INCREMENTAL: return "INCREMENTAL"; case JSGC_MODE_ZONE: return "ZONE"; } return ""; } const char * ircd::js::reflect(const JS::GCProgress &s) { switch(s) { case JS::GC_CYCLE_BEGIN: return "CYCLE_BEGIN"; case JS::GC_SLICE_BEGIN: return "SLICE_BEGIN"; case JS::GC_SLICE_END: return "SLICE_END"; case JS::GC_CYCLE_END: return "CYCLE_END"; } return ""; } const char * ircd::js::reflect(const JSExnType &e) { switch(e) { case JSEXN_WARN: return "Warning"; case JSEXN_ERR: return "Error"; case JSEXN_INTERNALERR: return "InternalError"; case JSEXN_EVALERR: return "EvalError"; case JSEXN_RANGEERR: return "RangeError"; case JSEXN_REFERENCEERR: return "ReferenceError"; case JSEXN_SYNTAXERR: return "SyntaxError"; case JSEXN_TYPEERR: return "TypeError"; case JSEXN_URIERR: return "URIError"; case JSEXN_LIMIT: return "?LIMIT?"; case JSEXN_DEBUGGEEWOULDRUN: return "DebugeeWouldRun"; case JSEXN_WASMCOMPILEERROR: return "WASMCompileError"; case JSEXN_WASMRUNTIMEERROR: return "WASMRuntimeError"; } return ""; } const char * ircd::js::reflect(const JSType &t) { switch(t) { case JSTYPE_VOID: return "VOID"; case JSTYPE_OBJECT: return "OBJECT"; case JSTYPE_FUNCTION: return "FUNCTION"; case JSTYPE_STRING: return "STRING"; case JSTYPE_NUMBER: return "NUMBER"; case JSTYPE_BOOLEAN: return "BOOLEAN"; case JSTYPE_NULL: return "NULL"; case JSTYPE_SYMBOL: return "SYMBOL"; case JSTYPE_LIMIT: return "LIMIT"; } return ""; } const char * ircd::js::reflect(const jstype &t) { switch(t) { case jstype::VALUE: return "VALUE"; case jstype::OBJECT: return "OBJECT"; case jstype::FUNCTION: return "FUNCTION"; case jstype::SCRIPT: return "SCRIPT"; case jstype::STRING: return "STRING"; case jstype::SYMBOL: return "SYMBOL"; case jstype::ID: return "ID"; } return ""; } /////////////////////////////////////////////////////////////////////////////// // // ircd/js/type.h // /////////////////////////////////////////////////////////////////////////////// // // ircd/js/version.h // /////////////////////////////////////////////////////////////////////////////// // // Misc / Fixes / Temp // #if defined(IRCD_JS_FIX) void __attribute__((noreturn)) js::ReportOutOfMemory(ExclusiveContext *const c) { ircd::js::log.critical("jsalloc(): Reported out of memory (ExclusiveContext: %p)", (const void *)c); ircd::terminate(); } #endif //IRCD_JS_FIX // // This DEBUG section is a fix for linkage errors when SpiderMonkey is compiled // in debug mode. // #if defined(JS_DEBUG) && defined(IRCD_JS_FIX) namespace js { namespace oom { extern mozilla::detail::ThreadLocal<uint32_t> threadType; uint32_t GetThreadType() { assert(0); return threadType.get(); } } // namespace oom } // namespace js #endif // JS_DEBUG && IRCD_JS_FIX