Skip to content

Commit 26581b9

Browse files
authored
Add JS_NewObjectFrom and JS_NewObjectFromStr (#871)
Combines JS_NewObject + JS_SetProperty into a single, optimized API call. Much more efficient when creating many objects or objects with many properties. Refs: #868
1 parent 0b38708 commit 26581b9

File tree

3 files changed

+103
-22
lines changed

3 files changed

+103
-22
lines changed

quickjs.c

+78
Original file line numberDiff line numberDiff line change
@@ -5038,6 +5038,84 @@ JSValue JS_NewObjectProto(JSContext *ctx, JSValue proto)
50385038
return JS_NewObjectProtoClass(ctx, proto, JS_CLASS_OBJECT);
50395039
}
50405040

5041+
JSValue JS_NewObjectFrom(JSContext *ctx, int count, const JSAtom *props,
5042+
const JSValue *values)
5043+
{
5044+
JSShapeProperty *pr;
5045+
uint32_t *hash;
5046+
JSRuntime *rt;
5047+
JSObject *p;
5048+
JSShape *sh;
5049+
JSValue obj;
5050+
JSAtom atom;
5051+
intptr_t h;
5052+
int i;
5053+
5054+
rt = ctx->rt;
5055+
obj = JS_NewObject(ctx);
5056+
if (JS_IsException(obj))
5057+
return JS_EXCEPTION;
5058+
if (count > 0) {
5059+
p = JS_VALUE_GET_OBJ(obj);
5060+
sh = p->shape;
5061+
assert(sh->is_hashed);
5062+
assert(sh->header.ref_count == 1);
5063+
js_shape_hash_unlink(rt, sh);
5064+
if (resize_properties(ctx, &sh, p, count)) {
5065+
js_shape_hash_link(rt, sh);
5066+
JS_FreeValue(ctx, obj);
5067+
return JS_EXCEPTION;
5068+
}
5069+
p->shape = sh;
5070+
for (i = 0; i < count; i++) {
5071+
atom = props[i];
5072+
pr = &sh->prop[i];
5073+
sh->hash = shape_hash(shape_hash(sh->hash, atom), JS_PROP_C_W_E);
5074+
sh->has_small_array_index |= __JS_AtomIsTaggedInt(atom);
5075+
h = atom & sh->prop_hash_mask;
5076+
hash = &prop_hash_end(sh)[-h - 1];
5077+
pr->hash_next = *hash;
5078+
*hash = i + 1;
5079+
pr->atom = JS_DupAtom(ctx, atom);
5080+
pr->flags = JS_PROP_C_W_E;
5081+
p->prop[i].u.value = values[i];
5082+
}
5083+
js_shape_hash_link(rt, sh);
5084+
sh->prop_count = count;
5085+
}
5086+
return obj;
5087+
}
5088+
5089+
JSValue JS_NewObjectFromStr(JSContext *ctx, int count, const char **props,
5090+
const JSValue *values)
5091+
{
5092+
JSAtom atoms_s[16], *atoms = atoms_s;
5093+
JSValue ret;
5094+
int i;
5095+
5096+
i = 0;
5097+
ret = JS_EXCEPTION;
5098+
if (count < 1)
5099+
goto out;
5100+
if (count > (int)countof(atoms_s)) {
5101+
atoms = js_malloc(ctx, count * sizeof(*atoms));
5102+
if (!atoms)
5103+
return JS_EXCEPTION;
5104+
}
5105+
for (i = 0; i < count; i++) {
5106+
atoms[i] = JS_NewAtom(ctx, props[i]);
5107+
if (atoms[i] == JS_ATOM_NULL)
5108+
goto out;
5109+
}
5110+
ret = JS_NewObjectFrom(ctx, count, atoms, values);
5111+
out:
5112+
while (i-- > 0)
5113+
JS_FreeAtom(ctx, atoms[i]);
5114+
if (atoms != atoms_s)
5115+
js_free(ctx, atoms);
5116+
return ret;
5117+
}
5118+
50415119
JSValue JS_NewArray(JSContext *ctx)
50425120
{
50435121
return JS_NewObjectFromShape(ctx, js_dup_shape(ctx->array_shape),

quickjs.h

+6
Original file line numberDiff line numberDiff line change
@@ -717,6 +717,12 @@ JS_EXTERN JSValue JS_NewObjectProtoClass(JSContext *ctx, JSValue proto, JSClassI
717717
JS_EXTERN JSValue JS_NewObjectClass(JSContext *ctx, int class_id);
718718
JS_EXTERN JSValue JS_NewObjectProto(JSContext *ctx, JSValue proto);
719719
JS_EXTERN JSValue JS_NewObject(JSContext *ctx);
720+
JS_EXTERN JSValue JS_NewObjectFrom(JSContext *ctx, int count,
721+
const JSAtom *props,
722+
const JSValue *values);
723+
JS_EXTERN JSValue JS_NewObjectFromStr(JSContext *ctx, int count,
724+
const char **props,
725+
const JSValue *values);
720726
JS_EXTERN JSValue JS_ToObject(JSContext *ctx, JSValue val);
721727
JS_EXTERN JSValue JS_ToObjectString(JSContext *ctx, JSValue val);
722728

run-test262.c

+19-22
Original file line numberDiff line numberDiff line change
@@ -931,36 +931,33 @@ static JSValue js_IsHTMLDDA(JSContext *ctx, JSValue this_val,
931931
static JSValue add_helpers1(JSContext *ctx)
932932
{
933933
JSValue global_obj;
934-
JSValue obj262, obj;
934+
JSValue obj262, is_html_dda;
935935

936936
global_obj = JS_GetGlobalObject(ctx);
937937

938938
JS_SetPropertyStr(ctx, global_obj, "print",
939939
JS_NewCFunction(ctx, js_print_262, "print", 1));
940940

941+
is_html_dda = JS_NewCFunction(ctx, js_IsHTMLDDA, "IsHTMLDDA", 0);
942+
JS_SetIsHTMLDDA(ctx, is_html_dda);
943+
#define N 7
944+
static const char *props[N] = {
945+
"detachArrayBuffer", "evalScript", "codePointRange",
946+
"agent", "global", "createRealm", "IsHTMLDDA",
947+
};
948+
JSValue values[N] = {
949+
JS_NewCFunction(ctx, js_detachArrayBuffer, "detachArrayBuffer", 1),
950+
JS_NewCFunction(ctx, js_evalScript_262, "evalScript", 1),
951+
JS_NewCFunction(ctx, js_string_codePointRange, "codePointRange", 2),
952+
js_new_agent(ctx),
953+
JS_DupValue(ctx, global_obj),
954+
JS_NewCFunction(ctx, js_createRealm, "createRealm", 0),
955+
is_html_dda,
956+
};
941957
/* $262 special object used by the tests */
942-
obj262 = JS_NewObject(ctx);
943-
JS_SetPropertyStr(ctx, obj262, "detachArrayBuffer",
944-
JS_NewCFunction(ctx, js_detachArrayBuffer,
945-
"detachArrayBuffer", 1));
946-
JS_SetPropertyStr(ctx, obj262, "evalScript",
947-
JS_NewCFunction(ctx, js_evalScript_262,
948-
"evalScript", 1));
949-
JS_SetPropertyStr(ctx, obj262, "codePointRange",
950-
JS_NewCFunction(ctx, js_string_codePointRange,
951-
"codePointRange", 2));
952-
JS_SetPropertyStr(ctx, obj262, "agent", js_new_agent(ctx));
953-
954-
JS_SetPropertyStr(ctx, obj262, "global",
955-
JS_DupValue(ctx, global_obj));
956-
JS_SetPropertyStr(ctx, obj262, "createRealm",
957-
JS_NewCFunction(ctx, js_createRealm,
958-
"createRealm", 0));
959-
obj = JS_NewCFunction(ctx, js_IsHTMLDDA, "IsHTMLDDA", 0);
960-
JS_SetIsHTMLDDA(ctx, obj);
961-
JS_SetPropertyStr(ctx, obj262, "IsHTMLDDA", obj);
962-
958+
obj262 = JS_NewObjectFromStr(ctx, N, props, values);
963959
JS_SetPropertyStr(ctx, global_obj, "$262", JS_DupValue(ctx, obj262));
960+
#undef N
964961

965962
JS_FreeValue(ctx, global_obj);
966963
return obj262;

0 commit comments

Comments
 (0)