Skip to content

Commit cc74bcc

Browse files
author
Robert Fancsik
committed
Rework VM dispatch
Overview: - `VM_OC_*` opcodes are removed, now the dispatch is based on the `CBC`/`CBC_EXT` opcode - The common argument decoding is removed, now each opcode resolves it's arguments individually using `VM_DECODE_{EXT}_...` macros. Whenever an argument is decoded the dispatcher continues with the same opcode increased by `VM_OPCODE_DECODED_ARG`. - E.g.: A opcode with 2 literal arguments dispatches as the following: - step_1: `case opcode: goto resolve_first_literal`, opcode increment by `VM_OPCODE_DECODED_ARG`, dispatch again (Hidden by macro) - step_2: `case VM_OPCODE_ONE_LITERAL(opcode): goto resolve_second_literal`, opcode increment by `VM_OPCODE_DECODED_ARG`, dispatch again (Hidden by macro) - step_3: `case VM_OPCODE_TWO_LITERALS (opcode):` Opcode handler implementation goes here. - The opcode handler implementation goes as the following: ```c case VM_OPCODE_TWO_LITERALS (opcode): { VM_DECODE_LITERAL_LITERAL (opcode); /* use left and right value as before */ } ``` - The put result block is optimized, each assignment knows whether an ident or property reference should be resolved - `free_left_value` and `free_both_values` labels are removed, due to the extra long jumps they caused to execute 1 or 2 simple calls. JerryScript-DCO-1.0-Signed-off-by: Robert Fancsik [email protected]
1 parent 0ef5094 commit cc74bcc

12 files changed

+2123
-1953
lines changed

docs/05.PORT-API.md

+1-2
Original file line numberDiff line numberDiff line change
@@ -26,8 +26,7 @@ typedef enum
2626
{
2727
JERRY_FATAL_OUT_OF_MEMORY = 10,
2828
JERRY_FATAL_REF_COUNT_LIMIT = 12,
29-
JERRY_FATAL_DISABLED_BYTE_CODE = 13,
30-
JERRY_FATAL_UNTERMINATED_GC_LOOPS = 14,
29+
JERRY_FATAL_UNTERMINATED_GC_LOOPS = 13,
3130
JERRY_FATAL_FAILED_ASSERTION = 120
3231
} jerry_fatal_code_t;
3332
```

jerry-core/api/jerry-snapshot.c

+27-46
Original file line numberDiff line numberDiff line change
@@ -34,46 +34,43 @@
3434
#if JERRY_SNAPSHOT_SAVE || JERRY_SNAPSHOT_EXEC
3535

3636
/**
37-
* Get snapshot configuration flags.
37+
* Get snapshot feature configuration flags.
3838
*
39-
* @return configuration flags
39+
* @return feature configuration flags
4040
*/
41-
static inline uint32_t JERRY_ATTR_ALWAYS_INLINE
42-
snapshot_get_global_flags (bool has_regex, /**< regex literal is present */
43-
bool has_class) /**< class literal is present */
41+
static inline jerry_snapshot_feature_flags_t
42+
snapshot_get_feature_flags (void)
4443
{
45-
JERRY_UNUSED (has_regex);
46-
JERRY_UNUSED (has_class);
47-
48-
uint32_t flags = 0;
44+
jerry_snapshot_feature_flags_t flags = JERRY_SNAPSHOT_FEATURE_NONE;
4945

5046
#if JERRY_BUILTIN_REGEXP
51-
flags |= (has_regex ? JERRY_SNAPSHOT_HAS_REGEX_LITERAL : 0);
47+
flags |= JERRY_SNAPSHOT_FEATURE_REGEXP;
5248
#endif /* JERRY_BUILTIN_REGEXP */
49+
#if JERRY_MODULE_SYSTEM
50+
flags |= JERRY_SNAPSHOT_FEATURE_MODULE;
51+
#endif /* JERRY_MODULE_SYSTEM */
52+
#if JERRY_DEBUGGER
53+
flags |= JERRY_SNAPSHOT_FEATURE_DEBUGGER;
54+
#endif /* JERRY_DEBUGGER */
5355
#if JERRY_ESNEXT
54-
flags |= (has_class ? JERRY_SNAPSHOT_HAS_CLASS_LITERAL : 0);
56+
flags |= JERRY_SNAPSHOT_FEATURE_ESNEXT;
5557
#endif /* JERRY_ESNEXT */
5658

5759
return flags;
58-
} /* snapshot_get_global_flags */
60+
} /* snapshot_get_feature_flags */
5961

6062
/**
61-
* Checks whether the global_flags argument matches to the current feature set.
63+
* Validate snapshot header
6264
*
63-
* @return true if global_flags accepted, false otherwise
65+
* @return true - if the header is valid
66+
* false - otherwise
6467
*/
65-
static inline bool JERRY_ATTR_ALWAYS_INLINE
66-
snapshot_check_global_flags (uint32_t global_flags) /**< global flags */
68+
static bool
69+
snapshot_validate_header (const jerry_snapshot_header_t *header_p)
6770
{
68-
#if JERRY_BUILTIN_REGEXP
69-
global_flags &= (uint32_t) ~JERRY_SNAPSHOT_HAS_REGEX_LITERAL;
70-
#endif /* JERRY_BUILTIN_REGEXP */
71-
#if JERRY_ESNEXT
72-
global_flags &= (uint32_t) ~JERRY_SNAPSHOT_HAS_CLASS_LITERAL;
73-
#endif /* JERRY_ESNEXT */
74-
75-
return global_flags == snapshot_get_global_flags (false, false);
76-
} /* snapshot_check_global_flags */
71+
return (header_p->magic == JERRY_SNAPSHOT_MAGIC && header_p->version == JERRY_SNAPSHOT_VERSION
72+
&& (header_p->feature_flags & snapshot_get_feature_flags ()) == header_p->feature_flags);
73+
} /* snapshot_validate_header */
7774

7875
#endif /* JERRY_SNAPSHOT_SAVE || JERRY_SNAPSHOT_EXEC */
7976

@@ -86,8 +83,6 @@ typedef struct
8683
{
8784
size_t snapshot_buffer_write_offset;
8885
ecma_value_t snapshot_error;
89-
bool regex_found;
90-
bool class_found;
9186
} snapshot_globals_t;
9287

9388
/** \addtogroup jerrysnapshot Jerry snapshot operations
@@ -171,11 +166,6 @@ snapshot_add_compiled_code (const ecma_compiled_code_t *compiled_code_p, /**< co
171166
jerry_throw_sz (JERRY_ERROR_RANGE, ecma_get_error_msg (ECMA_ERR_TAGGED_TEMPLATE_LITERALS));
172167
return 0;
173168
}
174-
175-
if (CBC_FUNCTION_GET_TYPE (compiled_code_p->status_flags) == CBC_FUNCTION_CONSTRUCTOR)
176-
{
177-
globals_p->class_found = true;
178-
}
179169
#endif /* JERRY_ESNEXT */
180170

181171
#if JERRY_BUILTIN_REGEXP
@@ -216,7 +206,6 @@ snapshot_add_compiled_code (const ecma_compiled_code_t *compiled_code_p, /**< co
216206
return 0;
217207
}
218208

219-
globals_p->regex_found = true;
220209
globals_p->snapshot_buffer_write_offset = JERRY_ALIGNUP (globals_p->snapshot_buffer_write_offset, JMEM_ALIGNMENT);
221210

222211
/* Regexp character size is stored in refs. */
@@ -788,8 +777,6 @@ jerry_generate_snapshot (jerry_value_t compiled_code, /**< parsed script or func
788777

789778
globals.snapshot_buffer_write_offset = aligned_header_size;
790779
globals.snapshot_error = ECMA_VALUE_EMPTY;
791-
globals.regex_found = false;
792-
globals.class_found = false;
793780

794781
if (generate_snapshot_opts & JERRY_SNAPSHOT_SAVE_STATIC)
795782
{
@@ -808,7 +795,7 @@ jerry_generate_snapshot (jerry_value_t compiled_code, /**< parsed script or func
808795
jerry_snapshot_header_t header;
809796
header.magic = JERRY_SNAPSHOT_MAGIC;
810797
header.version = JERRY_SNAPSHOT_VERSION;
811-
header.global_flags = snapshot_get_global_flags (globals.regex_found, globals.class_found);
798+
header.feature_flags = (uint32_t) snapshot_get_feature_flags ();
812799
header.lit_table_offset = (uint32_t) globals.snapshot_buffer_write_offset;
813800
header.number_of_funcs = 1;
814801
header.func_offsets[0] = aligned_header_size;
@@ -897,8 +884,7 @@ jerry_exec_snapshot (const uint32_t *snapshot_p, /**< snapshot */
897884

898885
const jerry_snapshot_header_t *header_p = (const jerry_snapshot_header_t *) snapshot_data_p;
899886

900-
if (header_p->magic != JERRY_SNAPSHOT_MAGIC || header_p->version != JERRY_SNAPSHOT_VERSION
901-
|| !snapshot_check_global_flags (header_p->global_flags))
887+
if (!snapshot_validate_header (header_p))
902888
{
903889
return jerry_throw_sz (JERRY_ERROR_TYPE, ecma_get_error_msg (ECMA_ERR_INVALID_SNAPSHOT_VERSION_OR_FEATURES));
904890
}
@@ -1219,7 +1205,6 @@ jerry_merge_snapshots (const uint32_t **inp_buffers_p, /**< array of (pointers t
12191205
{
12201206
#if JERRY_SNAPSHOT_SAVE
12211207
uint32_t number_of_funcs = 0;
1222-
uint32_t merged_global_flags = 0;
12231208
size_t functions_size = sizeof (jerry_snapshot_header_t);
12241209

12251210
if (number_of_snapshots < 2)
@@ -1241,16 +1226,13 @@ jerry_merge_snapshots (const uint32_t **inp_buffers_p, /**< array of (pointers t
12411226

12421227
const jerry_snapshot_header_t *header_p = (const jerry_snapshot_header_t *) inp_buffers_p[i];
12431228

1244-
if (header_p->magic != JERRY_SNAPSHOT_MAGIC || header_p->version != JERRY_SNAPSHOT_VERSION
1245-
|| !snapshot_check_global_flags (header_p->global_flags))
1229+
if (!snapshot_validate_header (header_p))
12461230
{
12471231
*error_p = "invalid snapshot version or unsupported features present";
12481232
ecma_collection_destroy (lit_pool_p);
12491233
return 0;
12501234
}
12511235

1252-
merged_global_flags |= header_p->global_flags;
1253-
12541236
uint32_t start_offset = header_p->func_offsets[0];
12551237
const uint8_t *data_p = (const uint8_t *) inp_buffers_p[i];
12561238
const uint8_t *literal_base_p = data_p + header_p->lit_table_offset;
@@ -1278,7 +1260,7 @@ jerry_merge_snapshots (const uint32_t **inp_buffers_p, /**< array of (pointers t
12781260

12791261
header_p->magic = JERRY_SNAPSHOT_MAGIC;
12801262
header_p->version = JERRY_SNAPSHOT_VERSION;
1281-
header_p->global_flags = merged_global_flags;
1263+
header_p->feature_flags = snapshot_get_feature_flags ();
12821264
header_p->lit_table_offset = (uint32_t) functions_size;
12831265
header_p->number_of_funcs = number_of_funcs;
12841266

@@ -1555,8 +1537,7 @@ jerry_get_literals_from_snapshot (const uint32_t *snapshot_p, /**< input snapsho
15551537
const uint8_t *snapshot_data_p = (uint8_t *) snapshot_p;
15561538
const jerry_snapshot_header_t *header_p = (const jerry_snapshot_header_t *) snapshot_data_p;
15571539

1558-
if (snapshot_size <= sizeof (jerry_snapshot_header_t) || header_p->magic != JERRY_SNAPSHOT_MAGIC
1559-
|| header_p->version != JERRY_SNAPSHOT_VERSION || !snapshot_check_global_flags (header_p->global_flags))
1540+
if (snapshot_size <= sizeof (jerry_snapshot_header_t) || !snapshot_validate_header (header_p))
15601541
{
15611542
/* Invalid snapshot format */
15621543
return 0;

jerry-core/api/jerry-snapshot.h

+8-8
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ typedef struct
2727
* uint32_t alignment. Otherwise some bytes after the header are wasted. */
2828
uint32_t magic; /**< four byte magic number */
2929
uint32_t version; /**< version number */
30-
uint32_t global_flags; /**< global configuration and feature flags */
30+
uint32_t feature_flags; /**< combination of jerry_snapshot_global_flags_t */
3131
uint32_t lit_table_offset; /**< byte offset of the literal table */
3232
uint32_t number_of_funcs; /**< number of primary ECMAScript functions */
3333
uint32_t func_offsets[1]; /**< function offsets (lowest bit: global(0) or eval(1) context) */
@@ -39,15 +39,15 @@ typedef struct
3939
#define JERRY_SNAPSHOT_MAGIC (0x5952524Au)
4040

4141
/**
42-
* Snapshot configuration flags.
42+
* Feature flags for snapshot execution.
4343
*/
4444
typedef enum
4545
{
46-
/* 8 bits are reserved for dynamic features */
47-
JERRY_SNAPSHOT_HAS_REGEX_LITERAL = (1u << 0), /**< byte code has regex literal */
48-
JERRY_SNAPSHOT_HAS_CLASS_LITERAL = (1u << 1), /**< byte code has class literal */
49-
/* 24 bits are reserved for compile time features */
50-
JERRY_SNAPSHOT_FOUR_BYTE_CPOINTER = (1u << 8) /**< deprecated, an unused placeholder now */
51-
} jerry_snapshot_global_flags_t;
46+
JERRY_SNAPSHOT_FEATURE_NONE = 0,
47+
JERRY_SNAPSHOT_FEATURE_REGEXP = (1u << 0), /**< feature flag for JERRY_BUILTIN_REGEXP */
48+
JERRY_SNAPSHOT_FEATURE_MODULE = (1u << 1), /**< feature flag for JERRY_MODULE_SYSTEM */
49+
JERRY_SNAPSHOT_FEATURE_DEBUGGER = (1u << 2), /**< feature flag for JERRY_DEBUGGER */
50+
JERRY_SNAPSHOT_FEATURE_ESNEXT = (1u << 3), /**< feature flag for JERRY_ESNEXT */
51+
} jerry_snapshot_feature_flags_t;
5252

5353
#endif /* !JERRY_SNAPSHOT_H */

jerry-core/include/jerryscript-port.h

+1-2
Original file line numberDiff line numberDiff line change
@@ -44,8 +44,7 @@ typedef enum
4444
{
4545
JERRY_FATAL_OUT_OF_MEMORY = 10, /**< Out of memory */
4646
JERRY_FATAL_REF_COUNT_LIMIT = 12, /**< Reference count limit reached */
47-
JERRY_FATAL_DISABLED_BYTE_CODE = 13, /**< Executed disabled instruction */
48-
JERRY_FATAL_UNTERMINATED_GC_LOOPS = 14, /**< Garbage collection loop limit reached */
47+
JERRY_FATAL_UNTERMINATED_GC_LOOPS = 13, /**< Garbage collection loop limit reached */
4948
JERRY_FATAL_FAILED_ASSERTION = 120 /**< Assertion failed */
5049
} jerry_fatal_code_t;
5150

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: JERRY_FATAL_UNTERMINATED_GC_LOOPS\n");
4848
break;
4949
}
50-
case JERRY_FATAL_DISABLED_BYTE_CODE:
51-
{
52-
JERRY_ERROR_MSG ("Error: JERRY_FATAL_DISABLED_BYTE_CODE\n");
53-
break;
54-
}
5550
case JERRY_FATAL_FAILED_ASSERTION:
5651
{
5752
JERRY_ERROR_MSG ("Error: JERRY_FATAL_FAILED_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.

0 commit comments

Comments
 (0)