/* * Copyright (C) 2016 Charybdis Development Team * Copyright (C) 2016 Jason Volk * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice is present in all copies. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #pragma once #define HAVE_IRCD_JS_FUNCTION_H namespace ircd { namespace js { string decompile(const JS::Handle &, const bool &pretty = false); string display_name(const JSFunction &); string name(const JSFunction &); uint16_t arity(const JSFunction &f); namespace basic { template struct function :root { operator JSObject *() const; explicit operator script() const; explicit operator string() const; // js::value/js::object == lifetime::stack js::value operator()(const js::object &, const JS::HandleValueArray &args) const; js::value operator()(const js::object &) const; // new function function(JS::AutoObjectVector &stack, const JS::CompileOptions &opts, const char *const &name, const std::vector &args, const std::string &src); using root::root; explicit function(const value &); function(JSFunction *const &); function(JSFunction &); }; } // namespace basic using function = basic::function; using heap_function = basic::function; // // Implementation // namespace basic { template function::function(JSFunction &func) :function::root::type{&func} { } template function::function(JSFunction *const &func) :function::root::type{func} { if(unlikely(!this->get())) throw internal_error("NULL function"); } template function::function(const value &val) :function::root::type { JS_ValueToFunction(*cx, val) } { if(!this->get()) throw type_error("value is not a function"); } template function::function(JS::AutoObjectVector &stack, const JS::CompileOptions &opts, const char *const &name, const std::vector &args, const std::string &src) :function::root::type{} { std::vector argp(args.size()); std::transform(begin(args), end(args), begin(argp), [] (const std::string &arg) { return arg.data(); }); if(!JS::CompileFunction(*cx, stack, opts, name, argp.size(), &argp.front(), src.data(), src.size(), &(*this))) { throw syntax_error("Failed to compile function"); } } template js::value function::operator()(const js::object &that) const { return this->operator()(that, JS::HandleValueArray::empty()); } template js::value function::operator()(const js::object &that, const JS::HandleValueArray &args) const { js::value ret; if(!JS_CallFunction(*cx, that, *this, args, &ret)) throw jserror(jserror::pending); return ret; } template function::operator string() const { return decompile(*this, true); } template function::operator script() const { return JS_GetFunctionScript(*cx, *this); } template function::operator JSObject *() const { const auto ret(JS_GetFunctionObject(this->get())); if(unlikely(!ret)) throw type_error("function cannot cast to Object"); return ret; } } // namespace basic inline uint16_t arity(const JSFunction &f) { return JS_GetFunctionArity(const_cast(&f)); } inline string name(const JSFunction &f) { const auto ret(JS_GetFunctionId(const_cast(&f))); return ret? string(ret) : string(""); } inline string display_name(const JSFunction &f) { const auto ret(JS_GetFunctionDisplayId(const_cast(&f))); return ret? string(ret) : string(""); } inline string decompile(const JS::Handle &f, const bool &pretty) { uint flags(0); flags |= pretty? 0 : JS_DONT_PRETTY_PRINT; return JS_DecompileFunction(*cx, f, flags); } } // namespace js } // namespace ircd