Skip to content

Commit a50bb28

Browse files
author
Robert Fancsik
committed
Rework VM dispatch
JerryScript-DCO-1.0-Signed-off-by: Robert Fancsik [email protected]
1 parent 79fd540 commit a50bb28

File tree

10 files changed

+2086
-1896
lines changed

10 files changed

+2086
-1896
lines changed

docs/05.PORT-API.md

-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,6 @@ typedef enum
2626
{
2727
ERR_OUT_OF_MEMORY = 10,
2828
ERR_REF_COUNT_LIMIT = 12,
29-
ERR_DISABLED_BYTE_CODE = 13,
3029
ERR_FAILED_INTERNAL_ASSERTION = 120
3130
} jerry_fatal_code_t;
3231
```

jerry-core/include/jerryscript-port.h

-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,6 @@ typedef enum
4646
{
4747
ERR_OUT_OF_MEMORY = 10,
4848
ERR_REF_COUNT_LIMIT = 12,
49-
ERR_DISABLED_BYTE_CODE = 13,
5049
ERR_UNTERMINATED_GC_LOOPS = 14,
5150
ERR_FAILED_INTERNAL_ASSERTION = 120
5251
} jerry_fatal_code_t;

jerry-core/jrt/jrt-fatals.c

-5
Original file line numberDiff line numberDiff line change
@@ -47,11 +47,6 @@ jerry_fatal (jerry_fatal_code_t code) /**< status code */
4747
JERRY_ERROR_MSG ("Error: ERR_UNTERMINATED_GC_LOOPS\n");
4848
break;
4949
}
50-
case ERR_DISABLED_BYTE_CODE:
51-
{
52-
JERRY_ERROR_MSG ("Error: ERR_DISABLED_BYTE_CODE\n");
53-
break;
54-
}
5550
case ERR_FAILED_INTERNAL_ASSERTION:
5651
{
5752
JERRY_ERROR_MSG ("Error: ERR_FAILED_INTERNAL_ASSERTION\n");

jerry-core/parser/js/byte-code.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ JERRY_STATIC_ASSERT (CBC_EXT_END == 167, number_of_cbc_ext_opcodes_changed);
4545
/**
4646
* Compact bytecode definition
4747
*/
48-
#define CBC_OPCODE(arg1, arg2, arg3, arg4) ((arg2) | (((arg3) + CBC_STACK_ADJUST_BASE) << CBC_STACK_ADJUST_SHIFT)),
48+
#define CBC_OPCODE(arg1, arg2, arg3) ((arg2) | (((arg3) + CBC_STACK_ADJUST_BASE) << CBC_STACK_ADJUST_SHIFT)),
4949

5050
/**
5151
* Flags of the opcodes.
@@ -63,7 +63,7 @@ const uint8_t cbc_ext_flags[] = { CBC_EXT_OPCODE_LIST };
6363

6464
#if JERRY_PARSER_DUMP_BYTE_CODE
6565

66-
#define CBC_OPCODE(arg1, arg2, arg3, arg4) #arg1,
66+
#define CBC_OPCODE(arg1, arg2, arg3) #arg1,
6767

6868
/**
6969
* Names of the opcodes.

jerry-core/parser/js/byte-code.h

+360-575
Large diffs are not rendered by default.

jerry-core/parser/js/js-parser-expr.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -1420,7 +1420,7 @@ parser_parse_object_literal (parser_context_t *context_p) /**< context */
14201420
uint16_t function_literal_index = lexer_construct_function_object (context_p, status_flags);
14211421

14221422
#if JERRY_ESNEXT
1423-
if (opcode >= CBC_EXT_SET_COMPUTED_GETTER)
1423+
if (is_computed)
14241424
{
14251425
literal_index = function_literal_index;
14261426
}
@@ -3487,7 +3487,7 @@ parser_process_binary_assignment_token (parser_context_t *context_p, /**< contex
34873487

34883488
if (opcode == CBC_ASSIGN_PROP_THIS_LITERAL && (context_p->stack_depth >= context_p->stack_limit))
34893489
{
3490-
/* Stack limit is increased for VM_OC_ASSIGN_PROP_THIS. Needed by vm.c. */
3490+
/* Needed by vm.c. */
34913491
JERRY_ASSERT (context_p->stack_depth == context_p->stack_limit);
34923492

34933493
context_p->stack_limit++;

jerry-core/vm/opcodes.c

+90-135
Original file line numberDiff line numberDiff line change
@@ -59,10 +59,10 @@ opfunc_typeof (ecma_value_t left_value) /**< left value */
5959
/**
6060
* Update data property for object literals.
6161
*/
62-
void
63-
opfunc_set_data_property (ecma_object_t *object_p, /**< object */
64-
ecma_string_t *prop_name_p, /**< data property name */
65-
ecma_value_t value) /**< new value */
62+
static void
63+
opfunc_assing_data_property (ecma_object_t *object_p, /**< object */
64+
ecma_string_t *prop_name_p, /**< data property name */
65+
ecma_value_t value) /**< new value */
6666
{
6767
JERRY_ASSERT (!ecma_op_object_is_fast_array (object_p));
6868

@@ -96,7 +96,7 @@ opfunc_set_data_property (ecma_object_t *object_p, /**< object */
9696
}
9797

9898
ecma_named_data_property_assign_value (object_p, prop_value_p, value);
99-
} /* opfunc_set_data_property */
99+
} /* opfunc_assing_data_property */
100100

101101
/**
102102
* Update getter or setter for object literals.
@@ -320,16 +320,15 @@ opfunc_for_in (ecma_value_t iterable_value, /**< ideally an iterable value */
320320
#if JERRY_ESNEXT
321321

322322
/**
323-
* 'VM_OC_APPEND_ARRAY' opcode handler specialized for spread objects
323+
* 'CBC_EXT_SPREAD_ARRAY_APPEND' opcode handler specialized for spread objects
324324
*
325325
* @return ECMA_VALUE_ERROR - if the operation failed
326326
* ECMA_VALUE_EMPTY, otherwise
327327
*/
328-
static ecma_value_t JERRY_ATTR_NOINLINE
328+
ecma_value_t JERRY_ATTR_NOINLINE
329329
opfunc_append_to_spread_array (ecma_value_t *stack_top_p, /**< current stack top */
330330
uint16_t values_length) /**< number of elements to set */
331331
{
332-
JERRY_ASSERT (!(values_length & OPFUNC_HAS_SPREAD_ELEMENT));
333332

334333
ecma_object_t *array_obj_p = ecma_get_object_from_value (stack_top_p[-1]);
335334
JERRY_ASSERT (ecma_get_object_type (array_obj_p) == ECMA_OBJECT_TYPE_ARRAY);
@@ -501,23 +500,15 @@ opfunc_spread_arguments (ecma_value_t *stack_top_p, /**< pointer to the current
501500
#endif /* JERRY_ESNEXT */
502501

503502
/**
504-
* 'VM_OC_APPEND_ARRAY' opcode handler, for setting array object properties
503+
* 'CBC_ARRAY_APPEND' and opcode handler, for setting array object properties
505504
*
506505
* @return ECMA_VALUE_ERROR - if the operation failed
507506
* ECMA_VALUE_EMPTY, otherwise
508507
*/
509508
ecma_value_t JERRY_ATTR_NOINLINE
510509
opfunc_append_array (ecma_value_t *stack_top_p, /**< current stack top */
511-
uint16_t values_length) /**< number of elements to set
512-
* with potential OPFUNC_HAS_SPREAD_ELEMENT flag */
510+
uint16_t values_length) /**< number of elements to set */
513511
{
514-
#if JERRY_ESNEXT
515-
if (values_length >= OPFUNC_HAS_SPREAD_ELEMENT)
516-
{
517-
return opfunc_append_to_spread_array (stack_top_p, (uint16_t) (values_length & ~OPFUNC_HAS_SPREAD_ELEMENT));
518-
}
519-
#endif /* JERRY_ESNEXT */
520-
521512
ecma_object_t *array_obj_p = ecma_get_object_from_value (stack_top_p[-1]);
522513
JERRY_ASSERT (ecma_get_object_type (array_obj_p) == ECMA_OBJECT_TYPE_ARRAY);
523514

@@ -1158,8 +1149,8 @@ opfunc_add_computed_field (ecma_value_t class_object, /**< class object */
11581149
* @return - new external function ecma-object
11591150
*/
11601151
ecma_value_t
1161-
opfunc_create_implicit_class_constructor (uint8_t opcode, /**< current cbc opcode */
1162-
const ecma_compiled_code_t *bytecode_p) /**< current byte code */
1152+
opfunc_create_implicit_class_constructor (const ecma_compiled_code_t *bytecode_p, /**< current byte code */
1153+
bool is_herigate) /* true -if class heritage is present */
11631154
{
11641155
/* 8. */
11651156
ecma_value_t script_value = ((cbc_uint8_arguments_t *) bytecode_p)->script_value;
@@ -1181,7 +1172,7 @@ opfunc_create_implicit_class_constructor (uint8_t opcode, /**< current cbc opcod
11811172
constructor_object_p->u.constructor_function.flags = 0;
11821173

11831174
/* 10.a.i */
1184-
if (opcode == CBC_EXT_PUSH_IMPLICIT_CONSTRUCTOR_HERITAGE)
1175+
if (is_herigate)
11851176
{
11861177
constructor_object_p->u.constructor_function.flags |= ECMA_CONSTRUCTOR_FUNCTION_HAS_HERITAGE;
11871178
}
@@ -1574,9 +1565,6 @@ opfunc_collect_private_properties (ecma_value_t constructor, ecma_value_t prop_n
15741565

15751566
if (opcode == CBC_EXT_COLLECT_PRIVATE_METHOD)
15761567
{
1577-
prop_name ^= value;
1578-
value ^= prop_name;
1579-
prop_name ^= value;
15801568
kind = ECMA_PRIVATE_METHOD;
15811569
}
15821570
else if (opcode == CBC_EXT_COLLECT_PRIVATE_GETTER)
@@ -1977,10 +1965,7 @@ opfunc_finalize_class (vm_frame_ctx_t *frame_ctx_p, /**< frame context */
19771965
* ECMA_VALUE_EMPTY - otherwise
19781966
*/
19791967
ecma_value_t
1980-
opfunc_form_super_reference (ecma_value_t **vm_stack_top_p, /**< current vm stack top */
1981-
vm_frame_ctx_t *frame_ctx_p, /**< frame context */
1982-
ecma_value_t prop_name, /**< property name to resolve */
1983-
uint8_t opcode) /**< current cbc opcode */
1968+
opfunc_resolve_super (vm_frame_ctx_t *frame_ctx_p) /**< frame context */
19841969
{
19851970
ecma_environment_record_t *environment_record_p = ecma_op_get_environment_record (frame_ctx_p->lex_env_p);
19861971

@@ -2001,112 +1986,8 @@ opfunc_form_super_reference (ecma_value_t **vm_stack_top_p, /**< current vm stac
20011986
return ECMA_VALUE_ERROR;
20021987
}
20031988

2004-
ecma_value_t *stack_top_p = *vm_stack_top_p;
2005-
2006-
if (opcode >= CBC_EXT_SUPER_PROP_ASSIGNMENT_REFERENCE)
2007-
{
2008-
JERRY_ASSERT (opcode == CBC_EXT_SUPER_PROP_ASSIGNMENT_REFERENCE
2009-
|| opcode == CBC_EXT_SUPER_PROP_LITERAL_ASSIGNMENT_REFERENCE);
2010-
*stack_top_p++ = parent;
2011-
*stack_top_p++ = ecma_copy_value (prop_name);
2012-
*vm_stack_top_p = stack_top_p;
2013-
2014-
return ECMA_VALUE_EMPTY;
2015-
}
2016-
2017-
ecma_object_t *parent_p = ecma_get_object_from_value (parent);
2018-
ecma_string_t *prop_name_p = ecma_op_to_property_key (prop_name);
2019-
2020-
if (prop_name_p == NULL)
2021-
{
2022-
ecma_deref_object (parent_p);
2023-
return ECMA_VALUE_ERROR;
2024-
}
2025-
2026-
ecma_value_t result = ecma_op_object_get_with_receiver (parent_p, prop_name_p, frame_ctx_p->this_binding);
2027-
ecma_deref_ecma_string (prop_name_p);
2028-
ecma_deref_object (parent_p);
2029-
2030-
if (ECMA_IS_VALUE_ERROR (result))
2031-
{
2032-
return result;
2033-
}
2034-
2035-
if (opcode == CBC_EXT_SUPER_PROP_LITERAL_REFERENCE || opcode == CBC_EXT_SUPER_PROP_REFERENCE)
2036-
{
2037-
*stack_top_p++ = ecma_copy_value (frame_ctx_p->this_binding);
2038-
*stack_top_p++ = ECMA_VALUE_UNDEFINED;
2039-
}
2040-
2041-
*stack_top_p++ = result;
2042-
*vm_stack_top_p = stack_top_p;
2043-
2044-
return ECMA_VALUE_EMPTY;
2045-
} /* opfunc_form_super_reference */
2046-
2047-
/**
2048-
* Assignment operation for SuperRefence base
2049-
*
2050-
* @return ECMA_VALUE_ERROR - if the operation fails
2051-
* ECMA_VALUE_EMPTY - otherwise
2052-
*/
2053-
ecma_value_t
2054-
opfunc_assign_super_reference (ecma_value_t **vm_stack_top_p, /**< vm stack top */
2055-
vm_frame_ctx_t *frame_ctx_p, /**< frame context */
2056-
uint32_t opcode_data) /**< opcode data to store the result */
2057-
{
2058-
ecma_value_t *stack_top_p = *vm_stack_top_p;
2059-
2060-
ecma_value_t base_obj = ecma_op_to_object (stack_top_p[-3]);
2061-
2062-
if (ECMA_IS_VALUE_ERROR (base_obj))
2063-
{
2064-
return base_obj;
2065-
}
2066-
2067-
ecma_object_t *base_obj_p = ecma_get_object_from_value (base_obj);
2068-
ecma_string_t *prop_name_p = ecma_op_to_property_key (stack_top_p[-2]);
2069-
2070-
if (prop_name_p == NULL)
2071-
{
2072-
ecma_deref_object (base_obj_p);
2073-
return ECMA_VALUE_ERROR;
2074-
}
2075-
2076-
bool is_strict = (frame_ctx_p->status_flags & VM_FRAME_CTX_IS_STRICT) != 0;
2077-
2078-
ecma_value_t result =
2079-
ecma_op_object_put_with_receiver (base_obj_p, prop_name_p, stack_top_p[-1], frame_ctx_p->this_binding, is_strict);
2080-
2081-
ecma_deref_ecma_string (prop_name_p);
2082-
ecma_deref_object (base_obj_p);
2083-
2084-
if (ECMA_IS_VALUE_ERROR (result))
2085-
{
2086-
return result;
2087-
}
2088-
2089-
for (int32_t i = 1; i <= 3; i++)
2090-
{
2091-
ecma_free_value (stack_top_p[-i]);
2092-
}
2093-
2094-
stack_top_p -= 3;
2095-
2096-
if (opcode_data & VM_OC_PUT_STACK)
2097-
{
2098-
*stack_top_p++ = result;
2099-
}
2100-
else if (opcode_data & VM_OC_PUT_BLOCK)
2101-
{
2102-
ecma_fast_free_value (VM_GET_REGISTER (frame_ctx_p, 0));
2103-
VM_GET_REGISTERS (frame_ctx_p)[0] = result;
2104-
}
2105-
2106-
*vm_stack_top_p = stack_top_p;
2107-
2108-
return result;
2109-
} /* opfunc_assign_super_reference */
1989+
return parent;
1990+
} /* opfunc_resolve_super */
21101991

21111992
/**
21121993
* Copy data properties of an object
@@ -2222,7 +2103,7 @@ opfunc_copy_data_properties (ecma_value_t target_object, /**< target object */
22222103
}
22232104
}
22242105

2225-
opfunc_set_data_property (target_object_p, property_name_p, result);
2106+
opfunc_assing_data_property (target_object_p, property_name_p, result);
22262107
ecma_free_value (result);
22272108

22282109
result = ECMA_VALUE_EMPTY;
@@ -2303,8 +2184,82 @@ opfunc_lexical_scope_has_restricted_binding (vm_frame_ctx_t *frame_ctx_p, /**< f
23032184
&& !ecma_is_property_configurable (property));
23042185
} /* opfunc_lexical_scope_has_restricted_binding */
23052186

2187+
/**
2188+
* Create function name property to the given function object
2189+
*/
2190+
void
2191+
opfunc_set_function_name (ecma_value_t function_object, /**< function object */
2192+
ecma_value_t function_name, /**< function name */
2193+
char *prefix_p, /**< function name prefix */
2194+
lit_utf8_size_t prefix_size) /**< function name prefix's length */
2195+
{
2196+
ecma_object_t *func_obj_p = ecma_get_object_from_value (function_object);
2197+
2198+
if (ecma_find_named_property (func_obj_p, ecma_get_magic_string (LIT_MAGIC_STRING_NAME)) != NULL)
2199+
{
2200+
return;
2201+
}
2202+
2203+
ecma_property_value_t *value_p;
2204+
value_p = ecma_create_named_data_property (func_obj_p,
2205+
ecma_get_magic_string (LIT_MAGIC_STRING_NAME),
2206+
ECMA_PROPERTY_FLAG_CONFIGURABLE,
2207+
NULL);
2208+
2209+
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
2210+
{
2211+
ECMA_SET_SECOND_BIT_TO_POINTER_TAG (((ecma_extended_object_t *) func_obj_p)->u.function.scope_cp);
2212+
}
2213+
2214+
value_p->value = ecma_op_function_form_name (ecma_get_prop_name_from_value (function_name), prefix_p, prefix_size);
2215+
} /* opfunc_set_function_name */
2216+
23062217
#endif /* JERRY_ESNEXT */
23072218

2219+
/**
2220+
* Set data property to an object/class
2221+
*
2222+
* @return ECMA_VALUE_ERROR - if the operation fails
2223+
* ECMA_VALUE_EMPTY - otherwise
2224+
*/
2225+
ecma_value_t
2226+
opfunc_set_data_property (ecma_value_t *stack_top_p, /**< vm stack */
2227+
ecma_value_t prop_name, /**< property name to set */
2228+
ecma_value_t value, /**< value to set */
2229+
bool is_static) /**< true - if set class property
2230+
false - otherwise */
2231+
{
2232+
ecma_string_t *prop_name_p = ecma_op_to_property_key (prop_name);
2233+
2234+
if (JERRY_UNLIKELY (prop_name_p == NULL))
2235+
{
2236+
return ECMA_VALUE_ERROR;
2237+
}
2238+
2239+
int index = -1;
2240+
2241+
#if JERRY_ESNEXT
2242+
if (JERRY_UNLIKELY (is_static))
2243+
{
2244+
if (ecma_compare_ecma_string_to_magic_id (prop_name_p, LIT_MAGIC_STRING_PROTOTYPE))
2245+
{
2246+
return ecma_raise_type_error (ECMA_ERR_CLASS_IS_NON_CONFIGURABLE);
2247+
}
2248+
2249+
index--;
2250+
}
2251+
#else /* !JERRY_ESNEXT */
2252+
JERRY_UNUSED (is_static);
2253+
#endif /* JERRY_ESNEXT */
2254+
2255+
ecma_object_t *object_p = ecma_get_object_from_value (stack_top_p[index]);
2256+
2257+
opfunc_assing_data_property (object_p, prop_name_p, value);
2258+
ecma_deref_ecma_string (prop_name_p);
2259+
2260+
return ECMA_VALUE_EMPTY;
2261+
} /* opfunc_set_data_property */
2262+
23082263
/**
23092264
* @}
23102265
* @}

0 commit comments

Comments
 (0)