mirror of
https://github.com/netsurf-browser/netsurf
synced 2025-01-06 11:02:22 +03:00
408 lines
11 KiB
C
408 lines
11 KiB
C
/*
|
|
* Copyright 2012 Vincent Sanders <vince@netsurf-browser.org>
|
|
* Copyright 2015 All of us.
|
|
*
|
|
* This file is part of NetSurf, http://www.netsurf-browser.org/
|
|
*
|
|
* NetSurf is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; version 2 of the License.
|
|
*
|
|
* NetSurf is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/** \file
|
|
* Duktapeish implementation of javascript engine functions.
|
|
*/
|
|
|
|
#include "content/content.h"
|
|
|
|
#include "utils/nsoption.h"
|
|
#include "utils/log.h"
|
|
#include "utils/corestrings.h"
|
|
|
|
#include "javascript/js.h"
|
|
#include "javascript/content.h"
|
|
|
|
#include "duktape/binding.h"
|
|
|
|
#include "duktape/duktape.h"
|
|
#include "dukky.h"
|
|
|
|
#include <dom/dom.h>
|
|
|
|
|
|
duk_bool_t
|
|
dukky_instanceof_at(duk_context *ctx, duk_idx_t _idx, const char *klass)
|
|
{
|
|
duk_idx_t idx = duk_normalize_index(ctx, _idx);
|
|
/* ... ??? */
|
|
if (!duk_check_type(ctx, idx, DUK_TYPE_OBJECT)) {
|
|
return false;
|
|
}
|
|
/* ... obj */
|
|
duk_get_global_string(ctx, "\xFF\xFFNETSURF_DUKTAPE_PROTOTYPES");
|
|
/* ... obj protos */
|
|
duk_get_prop_string(ctx, idx, klass);
|
|
/* ... obj protos goalproto */
|
|
duk_get_prototype(ctx, idx);
|
|
/* ... obj protos goalproto proto? */
|
|
while (!duk_is_undefined(ctx, -1)) {
|
|
if (duk_strict_equals(ctx, -1, -2)) {
|
|
duk_pop_3(ctx);
|
|
return true;
|
|
}
|
|
duk_get_prototype(ctx, -1);
|
|
/* ... obj protos goalproto proto proto? */
|
|
duk_replace(ctx, -2);
|
|
/* ... obj protos goalproto proto? */
|
|
}
|
|
duk_pop_3(ctx);
|
|
/* ... obj */
|
|
return false;
|
|
}
|
|
|
|
static duk_ret_t dukky_populate_object(duk_context *ctx)
|
|
{
|
|
/* ... obj args protoname nargs */
|
|
int nargs = duk_get_int(ctx, -1);
|
|
duk_pop(ctx);
|
|
/* ... obj args protoname */
|
|
duk_get_global_string(ctx, PROTO_MAGIC);
|
|
/* .. obj args protoname prototab */
|
|
duk_insert(ctx, -2);
|
|
/* ... obj args prototab protoname */
|
|
duk_get_prop(ctx, -2);
|
|
/* ... obj args prototab {proto/undefined} */
|
|
if (duk_is_undefined(ctx, -1)) {
|
|
LOG("RuhRoh, couldn't find a prototype, HTMLUnknownElement it is");
|
|
duk_pop(ctx);
|
|
duk_push_string(ctx, PROTO_NAME(HTMLUNKNOWNELEMENT));
|
|
duk_get_prop(ctx, -2);
|
|
}
|
|
/* ... obj args prototab proto */
|
|
duk_dup(ctx, -1);
|
|
/* ... obj args prototab proto proto */
|
|
duk_set_prototype(ctx, -(nargs+4));
|
|
/* ... obj[proto] args prototab proto */
|
|
duk_get_prop_string(ctx, -1, INIT_MAGIC);
|
|
/* ... obj[proto] args prototab proto initfn */
|
|
duk_insert(ctx, -(nargs+4));
|
|
/* ... initfn obj[proto] args prototab proto */
|
|
duk_pop_2(ctx);
|
|
/* ... initfn obj[proto] args */
|
|
LOG("Call the init function");
|
|
duk_call(ctx, nargs + 1);
|
|
return 1; /* The object */
|
|
}
|
|
|
|
duk_ret_t dukky_create_object(duk_context *ctx, const char *name, int args)
|
|
{
|
|
duk_ret_t ret;
|
|
LOG("name=%s nargs=%d", name+2, args);
|
|
/* ... args */
|
|
duk_push_object(ctx);
|
|
/* ... args obj */
|
|
duk_insert(ctx, -(args+1));
|
|
/* ... obj args */
|
|
duk_push_string(ctx, name);
|
|
/* ... obj args name */
|
|
duk_push_int(ctx, args);
|
|
/* ... obj args name nargs */
|
|
if ((ret = duk_safe_call(ctx, dukky_populate_object, args + 3, 1))
|
|
!= DUK_EXEC_SUCCESS)
|
|
return ret;
|
|
LOG("created");
|
|
return DUK_EXEC_SUCCESS;
|
|
}
|
|
|
|
|
|
|
|
duk_bool_t
|
|
dukky_push_node_stacked(duk_context *ctx)
|
|
{
|
|
int top_at_fail = duk_get_top(ctx) - 2;
|
|
/* ... nodeptr klass */
|
|
duk_get_global_string(ctx, NODE_MAGIC);
|
|
/* ... nodeptr klass nodes */
|
|
duk_dup(ctx, -3);
|
|
/* ... nodeptr klass nodes nodeptr */
|
|
duk_get_prop(ctx, -2);
|
|
/* ... nodeptr klass nodes node/undefined */
|
|
if (duk_is_undefined(ctx, -1)) {
|
|
/* ... nodeptr klass nodes undefined */
|
|
duk_pop(ctx);
|
|
/* ... nodeptr klass nodes */
|
|
duk_push_object(ctx);
|
|
/* ... nodeptr klass nodes obj */
|
|
duk_dup(ctx, -4);
|
|
/* ... nodeptr klass nodes obj nodeptr */
|
|
duk_dup(ctx, -4);
|
|
/* ... nodeptr klass nodes obj nodeptr klass */
|
|
duk_push_int(ctx, 1);
|
|
/* ... nodeptr klass nodes obj nodeptr klass 1 */
|
|
if (duk_safe_call(ctx, dukky_populate_object, 4, 1)
|
|
!= DUK_EXEC_SUCCESS) {
|
|
duk_set_top(ctx, top_at_fail);
|
|
LOG("Boo and also hiss");
|
|
return false;
|
|
}
|
|
/* ... nodeptr klass nodes node */
|
|
duk_dup(ctx, -4);
|
|
/* ... nodeptr klass nodes node nodeptr */
|
|
duk_dup(ctx, -2);
|
|
/* ... nodeptr klass nodes node nodeptr node */
|
|
duk_put_prop(ctx, -4);
|
|
/* ... nodeptr klass nodes node */
|
|
}
|
|
/* ... nodeptr klass nodes node */
|
|
duk_insert(ctx, -4);
|
|
/* ... node nodeptr klass nodes */
|
|
duk_pop_3(ctx);
|
|
/* ... node */
|
|
return true;
|
|
}
|
|
|
|
static void
|
|
dukky_push_node_klass(duk_context *ctx, struct dom_node *node)
|
|
{
|
|
dom_node_type nodetype;
|
|
dom_exception err;
|
|
|
|
err = dom_node_get_node_type(node, &nodetype);
|
|
if (err != DOM_NO_ERR) {
|
|
/* Oh bum, just node then */
|
|
duk_push_string(ctx, PROTO_NAME(NODE));
|
|
return;
|
|
}
|
|
|
|
switch(nodetype) {
|
|
case DOM_ELEMENT_NODE: {
|
|
dom_string *namespace, *tag;
|
|
err = dom_node_get_namespace(node, &namespace);
|
|
if (err != DOM_NO_ERR) {
|
|
/* Feck it, element */
|
|
LOG("dom_node_get_namespace() failed");
|
|
duk_push_string(ctx, PROTO_NAME(ELEMENT));
|
|
break;
|
|
}
|
|
if (namespace == NULL) {
|
|
/* No namespace, -> element */
|
|
LOG("no namespace");
|
|
duk_push_string(ctx, PROTO_NAME(ELEMENT));
|
|
break;
|
|
}
|
|
|
|
if (dom_string_isequal(namespace, corestring_dom_html_namespace) == false) {
|
|
/* definitely not an HTML element of some kind */
|
|
duk_push_string(ctx, PROTO_NAME(ELEMENT));
|
|
dom_string_unref(namespace);
|
|
break;
|
|
}
|
|
dom_string_unref(namespace);
|
|
|
|
err = dom_node_get_node_name(node, &tag);
|
|
if (err != DOM_NO_ERR) {
|
|
duk_push_string(ctx, PROTO_NAME(HTMLUNKNOWNELEMENT));
|
|
break;
|
|
}
|
|
|
|
duk_push_string(ctx, PROTO_NAME(HTML));
|
|
duk_push_lstring(ctx, dom_string_data(tag), dom_string_length(tag));
|
|
dom_string_unref(tag);
|
|
duk_push_string(ctx, "ELEMENT");
|
|
duk_concat(ctx, 3);
|
|
|
|
break;
|
|
}
|
|
case DOM_TEXT_NODE:
|
|
duk_push_string(ctx, PROTO_NAME(TEXT));
|
|
break;
|
|
case DOM_COMMENT_NODE:
|
|
duk_push_string(ctx, PROTO_NAME(COMMENT));
|
|
break;
|
|
case DOM_DOCUMENT_NODE:
|
|
duk_push_string(ctx, PROTO_NAME(DOCUMENT));
|
|
break;
|
|
case DOM_ATTRIBUTE_NODE:
|
|
case DOM_PROCESSING_INSTRUCTION_NODE:
|
|
case DOM_DOCUMENT_TYPE_NODE:
|
|
case DOM_DOCUMENT_FRAGMENT_NODE:
|
|
case DOM_NOTATION_NODE:
|
|
case DOM_ENTITY_REFERENCE_NODE:
|
|
case DOM_ENTITY_NODE:
|
|
case DOM_CDATA_SECTION_NODE:
|
|
default:
|
|
/* Oh bum, just node then */
|
|
duk_push_string(ctx, PROTO_NAME(NODE));
|
|
}
|
|
}
|
|
|
|
duk_bool_t
|
|
dukky_push_node(duk_context *ctx, struct dom_node *node)
|
|
{
|
|
LOG("Pushing node %p", node);
|
|
/* First check if we can find the node */
|
|
/* ... */
|
|
duk_get_global_string(ctx, NODE_MAGIC);
|
|
/* ... nodes */
|
|
duk_push_pointer(ctx, node);
|
|
/* ... nodes nodeptr */
|
|
duk_get_prop(ctx, -2);
|
|
/* ... nodes node/undefined */
|
|
if (!duk_is_undefined(ctx, -1)) {
|
|
/* ... nodes node */
|
|
duk_insert(ctx, -2);
|
|
/* ... node nodes */
|
|
duk_pop(ctx);
|
|
/* ... node */
|
|
LOG("Found it memoised");
|
|
return true;
|
|
}
|
|
/* ... nodes undefined */
|
|
duk_pop_2(ctx);
|
|
/* ... */
|
|
/* We couldn't, so now we determine the node type and then
|
|
* we ask for it to be created
|
|
*/
|
|
duk_push_pointer(ctx, node);
|
|
/* ... nodeptr */
|
|
dukky_push_node_klass(ctx, node);
|
|
/* ... nodeptr klass */
|
|
return dukky_push_node_stacked(ctx);
|
|
}
|
|
|
|
static duk_ret_t
|
|
dukky_bad_constructor(duk_context *ctx)
|
|
{
|
|
duk_error(ctx, DUK_ERR_ERROR, "Bad constructor");
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
dukky_inject_not_ctr(duk_context *ctx, int idx, const char *name)
|
|
{
|
|
/* ... p[idx] ... proto */
|
|
duk_push_c_function(ctx, dukky_bad_constructor, 0);
|
|
/* ... p[idx] ... proto cons */
|
|
duk_insert(ctx, -2);
|
|
/* ... p[idx] ... cons proto */
|
|
duk_put_prop_string(ctx, -2, "prototype");
|
|
/* ... p[idx] ... cons[proto] */
|
|
duk_put_prop_string(ctx, idx, name);
|
|
/* ... p ... */
|
|
return;
|
|
}
|
|
|
|
/**************************************** js.h ******************************/
|
|
struct jscontext {
|
|
duk_context *ctx;
|
|
duk_context *thread;
|
|
};
|
|
|
|
#define CTX (ctx->thread)
|
|
|
|
void js_initialise(void)
|
|
{
|
|
/** TODO: Forces JS on for our testing, needs changing before a release
|
|
* lest we incur the wrath of others.
|
|
*/
|
|
nsoption_set_bool(enable_javascript, true);
|
|
javascript_init();
|
|
}
|
|
|
|
void js_finalise(void)
|
|
{
|
|
/* NADA for now */
|
|
}
|
|
|
|
#define DUKKY_NEW_PROTOTYPE(klass, uklass, klass_name) \
|
|
dukky_create_prototype(ctx, dukky_##klass##___proto, PROTO_NAME(uklass), klass_name)
|
|
|
|
jscontext *js_newcontext(int timeout, jscallback *cb, void *cbctx)
|
|
{
|
|
duk_context *ctx;
|
|
jscontext *ret = calloc(1, sizeof(*ret));
|
|
LOG("Creating new duktape javascript context");
|
|
if (ret == NULL) return NULL;
|
|
ctx = ret->ctx = duk_create_heap_default();
|
|
if (ret->ctx == NULL) { free(ret); return NULL; }
|
|
/* Create the prototype stuffs */
|
|
duk_push_global_object(ctx);
|
|
duk_push_boolean(ctx, true);
|
|
duk_put_prop_string(ctx, -2, "protos");
|
|
duk_put_global_string(ctx, PROTO_MAGIC);
|
|
/* Create prototypes here */
|
|
dukky_create_prototypes(ctx);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void js_destroycontext(jscontext *ctx)
|
|
{
|
|
LOG("Destroying duktape javascript context");
|
|
duk_destroy_heap(ctx->ctx);
|
|
free(ctx);
|
|
}
|
|
|
|
jsobject *js_newcompartment(jscontext *ctx, void *win_priv, void *doc_priv)
|
|
{
|
|
/* Pop any active thread off */
|
|
LOG("Yay, new compartment, win_priv=%p, doc_priv=%p", win_priv, doc_priv);
|
|
duk_set_top(ctx->ctx, 0);
|
|
duk_push_thread(ctx->ctx);
|
|
ctx->thread = duk_require_context(ctx->ctx, -1);
|
|
duk_push_int(CTX, 0);
|
|
duk_push_int(CTX, 1);
|
|
duk_push_int(CTX, 2);
|
|
/* Manufacture a Window object */
|
|
/* win_priv is a browser_window, doc_priv is an html content struct */
|
|
duk_push_pointer(CTX, win_priv);
|
|
duk_push_pointer(CTX, doc_priv);
|
|
dukky_create_object(CTX, PROTO_NAME(WINDOW), 2);
|
|
duk_push_global_object(CTX);
|
|
duk_put_prop_string(CTX, -2, PROTO_MAGIC);
|
|
duk_set_global_object(CTX);
|
|
|
|
/* Now we need to prepare our node mapping table */
|
|
duk_push_object(CTX);
|
|
duk_put_global_string(CTX, NODE_MAGIC);
|
|
|
|
return (jsobject *)ctx;
|
|
}
|
|
|
|
static duk_ret_t eval_top_string(duk_context *ctx)
|
|
{
|
|
duk_eval(ctx);
|
|
return 0;
|
|
}
|
|
|
|
bool js_exec(jscontext *ctx, const char *txt, size_t txtlen)
|
|
{
|
|
assert(ctx);
|
|
if (txt == NULL || txtlen == 0) return false;
|
|
duk_set_top(CTX, 0);
|
|
duk_push_lstring(CTX, txt, txtlen);
|
|
|
|
if (duk_safe_call(CTX, eval_top_string, 1, 1) == DUK_EXEC_ERROR) {
|
|
LOG("JAVASCRIPT WENT BANG: %s", duk_safe_to_string(CTX, 0));
|
|
return false;
|
|
}
|
|
if (duk_get_top(CTX) == 0) duk_push_boolean(CTX, false);
|
|
LOG("Returning %s", duk_get_boolean(CTX, 0) ? "true" : "false");
|
|
return duk_get_boolean(CTX, 0);
|
|
}
|
|
|
|
bool js_fire_event(jscontext *ctx, const char *type, struct dom_document *doc, struct dom_node *target)
|
|
{
|
|
/* La La La */
|
|
LOG("Oh dear, an event: %s", type);
|
|
return true;
|
|
}
|