Skip to content

Commit ea01767

Browse files
committed
Remove ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION to free an object type.
JerryScript-DCO-1.0-Signed-off-by: Zoltan Herczeg [email protected]
1 parent cfa0c94 commit ea01767

File tree

6 files changed

+108
-91
lines changed

6 files changed

+108
-91
lines changed

jerry-core/ecma/base/ecma-globals.h

Lines changed: 4 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -430,13 +430,11 @@ typedef enum
430430
ECMA_OBJECT_TYPE_GENERAL = 0, /**< all objects that are not String (15.5), Function (15.3),
431431
Arguments (10.6), Array (15.4) specification-defined objects */
432432
ECMA_OBJECT_TYPE_FUNCTION = 1, /**< Function objects (15.3), created through 13.2 routine */
433-
ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION = 2, /** One of built-in functions described in section 15
434-
* of ECMA-262 v5 specification */
433+
ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION = 2, /**< External (host) function object */
435434
ECMA_OBJECT_TYPE_ARRAY = 3, /**< Array object (15.4) */
436435
ECMA_OBJECT_TYPE_STRING = 4, /**< String objects (15.5) */
437-
ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION = 5, /**< External (host) function object */
438-
ECMA_OBJECT_TYPE_BOUND_FUNCTION = 6, /**< Function objects (15.3), created through 15.3.4.5 routine */
439-
ECMA_OBJECT_TYPE_ARGUMENTS = 7, /**< Arguments object (10.6) */
436+
ECMA_OBJECT_TYPE_BOUND_FUNCTION = 5, /**< Function objects (15.3), created through 15.3.4.5 routine */
437+
ECMA_OBJECT_TYPE_ARGUMENTS = 6, /**< Arguments object (10.6) */
440438

441439
ECMA_OBJECT_TYPE__MAX = ECMA_OBJECT_TYPE_ARGUMENTS /**< maximum value */
442440
} ecma_object_type_t;
@@ -448,7 +446,7 @@ typedef enum
448446
{
449447
/* ECMA_OBJECT_TYPE_GENERAL (0) with built-in flag. */
450448
/* ECMA_OBJECT_TYPE_FUNCTION (1) with built-in flag. */
451-
/* ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION (2) with built-in flag. */
449+
/* ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION (2) with built-in flag. */
452450
/* ECMA_OBJECT_TYPE_ARRAY (3) with built-in flag. */
453451
/* ECMA_OBJECT_TYPE_STRING (4) with built-in flag. */
454452
ECMA_LEXICAL_ENVIRONMENT_DECLARATIVE = 5, /**< declarative lexical environment */

jerry-core/ecma/builtin-objects/ecma-builtin-internal-routines-template.inc.h

Lines changed: 4 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -50,21 +50,11 @@
5050
#undef ROUTINE_ARG_LIST_0
5151
#undef ROUTINE_ARG
5252

53-
#define ECMA_BUILTIN_PROPERTY_NAME_INDEX(name) \
54-
PASTE (PASTE (PASTE (PASTE (ecma_builtin_property_names, _), BUILTIN_UNDERSCORED_ID), _), name)
55-
5653
enum
5754
{
58-
#define SIMPLE_VALUE(name, simple_value, prop_attributes) \
59-
ECMA_BUILTIN_PROPERTY_NAME_INDEX(name),
60-
#define NUMBER_VALUE(name, number_value, prop_attributes) \
61-
ECMA_BUILTIN_PROPERTY_NAME_INDEX(name),
62-
#define STRING_VALUE(name, magic_string_id, prop_attributes) \
63-
ECMA_BUILTIN_PROPERTY_NAME_INDEX(name),
64-
#define OBJECT_VALUE(name, obj_builtin_id, prop_attributes) \
65-
ECMA_BUILTIN_PROPERTY_NAME_INDEX(name),
55+
PASTE (ECMA_ROUTINE_START_, BUILTIN_UNDERSCORED_ID) = ECMA_BUILTIN_ID__COUNT - 1,
6656
#define ROUTINE(name, c_function_name, args_number, length_prop_value) \
67-
ECMA_BUILTIN_PROPERTY_NAME_INDEX(name),
57+
ECMA_ROUTINE_ ## name ## c_function_name,
6858
#include BUILTIN_INC_HEADER_NAME
6959
};
7060

@@ -78,7 +68,7 @@ const ecma_builtin_property_descriptor_t PROPERTY_DESCRIPTOR_LIST_NAME[] =
7868
name, \
7969
ECMA_BUILTIN_PROPERTY_ROUTINE, \
8070
ECMA_PROPERTY_CONFIGURABLE_WRITABLE, \
81-
length_prop_value \
71+
ECMA_ROUTINE_VALUE (ECMA_ROUTINE_ ## name ## c_function_name, length_prop_value) \
8272
},
8373
#define OBJECT_VALUE(name, obj_builtin_id, prop_attributes) \
8474
{ \
@@ -148,7 +138,7 @@ DISPATCH_ROUTINE_ROUTINE_NAME (uint16_t builtin_routine_id, /**< built-in wide r
148138
#define ROUTINE_ARG_LIST_3 ROUTINE_ARG_LIST_2, ROUTINE_ARG(3)
149139
#define ROUTINE_ARG_LIST_NON_FIXED , arguments_list, arguments_number
150140
#define ROUTINE(name, c_function_name, args_number, length_prop_value) \
151-
case name: \
141+
case ECMA_ROUTINE_ ## name ## c_function_name: \
152142
{ \
153143
return c_function_name (this_arg_value ROUTINE_ARG_LIST_ ## args_number); \
154144
}

jerry-core/ecma/builtin-objects/ecma-builtins.c

Lines changed: 33 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,22 @@ ecma_builtin_get (ecma_builtin_id_t builtin_id) /**< id of built-in to check on
8787
return ecma_builtin_objects[builtin_id];
8888
} /* ecma_builtin_get */
8989

90+
/**
91+
* Checks whether the given function is a built-in routine
92+
*
93+
* @return true if the function object is a built-in routine
94+
* false otherwise
95+
*/
96+
inline bool __attr_always_inline___
97+
ecma_builtin_function_is_routine (ecma_object_t *func_obj_p) /**< function object */
98+
{
99+
JERRY_ASSERT (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION);
100+
JERRY_ASSERT (ecma_get_object_is_builtin (func_obj_p));
101+
102+
ecma_extended_object_t *ext_func_obj_p = (ecma_extended_object_t *) func_obj_p;
103+
return (ext_func_obj_p->u.built_in.routine_id >= ECMA_BUILTIN_ID__COUNT);
104+
} /* ecma_builtin_function_is_routine */
105+
90106
/**
91107
* Initialize specified built-in object.
92108
*
@@ -113,6 +129,7 @@ ecma_builtin_init_object (ecma_builtin_id_t obj_builtin_id, /**< built-in ID */
113129

114130
ecma_extended_object_t *ext_obj_p = (ecma_extended_object_t *) obj_p;
115131
ext_obj_p->u.built_in.id = obj_builtin_id;
132+
ext_obj_p->u.built_in.routine_id = obj_builtin_id;
116133
ext_obj_p->u.built_in.instantiated_bitset = 0;
117134

118135
/** Initializing [[PrimitiveValue]] properties of built-in prototype objects */
@@ -285,12 +302,14 @@ ecma_builtin_make_function_object_for_routine (ecma_builtin_id_t builtin_id, /**
285302
{
286303
ecma_object_t *prototype_obj_p = ecma_builtin_get (ECMA_BUILTIN_ID_FUNCTION_PROTOTYPE);
287304

288-
ecma_object_t *func_obj_p = ecma_create_object (prototype_obj_p, true, true, ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION);
305+
ecma_object_t *func_obj_p = ecma_create_object (prototype_obj_p, true, true, ECMA_OBJECT_TYPE_FUNCTION);
289306

290307
ecma_deref_object (prototype_obj_p);
291308

292309
ecma_set_object_is_builtin (func_obj_p);
293310

311+
JERRY_ASSERT (routine_id >= ECMA_BUILTIN_ID__COUNT);
312+
294313
ecma_extended_object_t *ext_func_obj_p = (ecma_extended_object_t *) func_obj_p;
295314
ext_func_obj_p->u.built_in.id = builtin_id;
296315
ext_func_obj_p->u.built_in.length = length_prop_value;
@@ -328,9 +347,10 @@ ecma_builtin_try_to_instantiate_property (ecma_object_t *object_p, /**< object *
328347
{
329348
JERRY_ASSERT (ecma_get_object_is_builtin (object_p));
330349

331-
const ecma_object_type_t type = ecma_get_object_type (object_p);
350+
ecma_extended_object_t *ext_obj_p = (ecma_extended_object_t *) object_p;
332351

333-
if (type == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION)
352+
if (ecma_get_object_type (object_p) == ECMA_OBJECT_TYPE_FUNCTION
353+
&& ecma_builtin_function_is_routine (object_p))
334354
{
335355
ecma_string_t *magic_string_length_p = ecma_get_magic_string (LIT_MAGIC_STRING_LENGTH);
336356

@@ -348,8 +368,6 @@ ecma_builtin_try_to_instantiate_property (ecma_object_t *object_p, /**< object *
348368
* as it is non-configurable and so can't be deleted
349369
*/
350370

351-
ecma_extended_object_t *ext_obj_p = (ecma_extended_object_t *) object_p;
352-
353371
ecma_property_t *len_prop_p = ecma_create_named_data_property (object_p,
354372
string_p,
355373
ECMA_PROPERTY_FIXED);
@@ -371,8 +389,6 @@ ecma_builtin_try_to_instantiate_property (ecma_object_t *object_p, /**< object *
371389
return NULL;
372390
}
373391

374-
ecma_extended_object_t *ext_obj_p = (ecma_extended_object_t *) object_p;
375-
376392
ecma_builtin_id_t builtin_id = (ecma_builtin_id_t) ext_obj_p->u.built_in.id;
377393

378394
JERRY_ASSERT (builtin_id < ECMA_BUILTIN_ID__COUNT);
@@ -513,9 +529,10 @@ ecma_builtin_try_to_instantiate_property (ecma_object_t *object_p, /**< object *
513529
}
514530
case ECMA_BUILTIN_PROPERTY_ROUTINE:
515531
{
516-
ecma_object_t *func_obj_p = ecma_builtin_make_function_object_for_routine (builtin_id,
517-
magic_string_id,
518-
(uint8_t) curr_property_p->value);
532+
ecma_object_t *func_obj_p;
533+
func_obj_p = ecma_builtin_make_function_object_for_routine (builtin_id,
534+
ECMA_GET_ROUTINE_ID (curr_property_p->value),
535+
ECMA_GET_ROUTINE_LENGTH (curr_property_p->value));
519536
value = ecma_make_object_value (func_obj_p);
520537
break;
521538
}
@@ -559,9 +576,12 @@ ecma_builtin_list_lazy_property_names (ecma_object_t *object_p, /**< a built-in
559576
ecma_collection_header_t *non_enum_collection_p) /**< skipped 'non-enumerable'
560577
* collection */
561578
{
562-
const ecma_object_type_t type = ecma_get_object_type (object_p);
579+
JERRY_ASSERT (ecma_get_object_is_builtin (object_p));
580+
581+
ecma_extended_object_t *ext_obj_p = (ecma_extended_object_t *) object_p;
563582

564-
if (type == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION)
583+
if (ecma_get_object_type (object_p) == ECMA_OBJECT_TYPE_FUNCTION
584+
&& ecma_builtin_function_is_routine (object_p))
565585
{
566586
ecma_collection_header_t *for_enumerable_p = main_collection_p;
567587
(void) for_enumerable_p;
@@ -575,8 +595,6 @@ ecma_builtin_list_lazy_property_names (ecma_object_t *object_p, /**< a built-in
575595
}
576596
else
577597
{
578-
ecma_extended_object_t *ext_obj_p = (ecma_extended_object_t *) object_p;
579-
580598
ecma_builtin_id_t builtin_id = (ecma_builtin_id_t) ext_obj_p->u.built_in.id;
581599

582600
JERRY_ASSERT (builtin_id < ECMA_BUILTIN_ID__COUNT);
@@ -659,7 +677,7 @@ ecma_builtin_dispatch_call (ecma_object_t *obj_p, /**< built-in object */
659677
ecma_value_t ret_value = ecma_make_simple_value (ECMA_SIMPLE_VALUE_EMPTY);
660678
ecma_extended_object_t *ext_obj_p = (ecma_extended_object_t *) obj_p;
661679

662-
if (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION)
680+
if (ecma_builtin_function_is_routine (obj_p))
663681
{
664682
ret_value = ecma_builtin_dispatch_routine (ext_obj_p->u.built_in.id,
665683
ext_obj_p->u.built_in.routine_id,

jerry-core/ecma/builtin-objects/ecma-builtins.h

Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,21 @@ typedef enum
3434
ECMA_BUILTIN_ID__COUNT /**< number of built-in objects */
3535
} ecma_builtin_id_t;
3636

37+
/**
38+
* Construct a routine value
39+
*/
40+
#define ECMA_ROUTINE_VALUE(id, length) (((id) << 4) | length)
41+
42+
/**
43+
* Get routine length
44+
*/
45+
#define ECMA_GET_ROUTINE_LENGTH(value) ((uint8_t) ((value) & 0xf))
46+
47+
/**
48+
* Get routine ID
49+
*/
50+
#define ECMA_GET_ROUTINE_ID(value) ((uint16_t) ((value) >> 4))
51+
3752
/* ecma-builtins.c */
3853
extern void ecma_init_builtins (void);
3954
extern void ecma_finalize_builtins (void);
@@ -55,4 +70,7 @@ extern bool
5570
ecma_builtin_is (ecma_object_t *, ecma_builtin_id_t);
5671
extern ecma_object_t *
5772
ecma_builtin_get (ecma_builtin_id_t);
73+
extern bool
74+
ecma_builtin_function_is_routine (ecma_object_t *);
75+
5876
#endif /* !ECMA_BUILTINS_H */

jerry-core/ecma/operations/ecma-function-object.c

Lines changed: 19 additions & 17 deletions
Original file line numberDiff line numberDiff line change
@@ -59,9 +59,8 @@ ecma_op_is_callable (ecma_value_t value) /**< ecma value */
5959
JERRY_ASSERT (!ecma_is_lexical_environment (obj_p));
6060

6161
return (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_FUNCTION
62-
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION
6362
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION
64-
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION);
63+
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION);
6564
} /* ecma_op_is_callable */
6665

6766
/**
@@ -83,8 +82,13 @@ ecma_is_constructor (ecma_value_t value) /**< ecma value */
8382
JERRY_ASSERT (obj_p != NULL);
8483
JERRY_ASSERT (!ecma_is_lexical_environment (obj_p));
8584

86-
return (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_FUNCTION
87-
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION
85+
if (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
86+
{
87+
return (!ecma_get_object_is_builtin (obj_p)
88+
|| !ecma_builtin_function_is_routine (obj_p));
89+
}
90+
91+
return (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION
8892
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION);
8993
} /* ecma_is_constructor */
9094

@@ -444,7 +448,7 @@ ecma_op_create_external_function_object (ecma_external_pointer_t code_p) /**< po
444448
* created through 13.2 (ECMA_OBJECT_TYPE_FUNCTION)
445449
* or 15.3.4.5 (ECMA_OBJECT_TYPE_BOUND_FUNCTION),
446450
* and for built-in Function objects
447-
* from section 15 (ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION).
451+
* from section 15 (ECMA_OBJECT_TYPE_FUNCTION).
448452
*
449453
* @return ecma value
450454
* Returned value must be freed with ecma_free_value
@@ -460,6 +464,12 @@ ecma_op_function_has_instance (ecma_object_t *func_obj_p, /**< Function object *
460464

461465
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
462466
{
467+
if (ecma_get_object_is_builtin (func_obj_p)
468+
&& ecma_builtin_function_is_routine (func_obj_p))
469+
{
470+
return ecma_raise_type_error (ECMA_ERR_MSG (""));
471+
}
472+
463473
if (!ecma_is_value_object (value))
464474
{
465475
return ecma_make_simple_value (ECMA_SIMPLE_VALUE_FALSE);
@@ -505,8 +515,7 @@ ecma_op_function_has_instance (ecma_object_t *func_obj_p, /**< Function object *
505515

506516
ecma_deref_ecma_string (prototype_magic_string_p);
507517
}
508-
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION ||
509-
ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION)
518+
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION)
510519
{
511520
ret_value = ecma_raise_type_error (ECMA_ERR_MSG (""));
512521
}
@@ -535,7 +544,7 @@ ecma_op_function_has_instance (ecma_object_t *func_obj_p, /**< Function object *
535544
* created through 13.2 (ECMA_OBJECT_TYPE_FUNCTION)
536545
* or 15.3.4.5 (ECMA_OBJECT_TYPE_BOUND_FUNCTION),
537546
* and for built-in Function objects
538-
* from section 15 (ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION).
547+
* from section 15 (ECMA_OBJECT_TYPE_FUNCTION).
539548
*
540549
* @return ecma value
541550
* Returned value must be freed with ecma_free_value
@@ -636,13 +645,6 @@ ecma_op_function_call (ecma_object_t *func_obj_p, /**< Function object */
636645
ecma_free_value (this_binding);
637646
}
638647
}
639-
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION)
640-
{
641-
ret_value = ecma_builtin_dispatch_call (func_obj_p,
642-
this_arg_value,
643-
arguments_list_p,
644-
arguments_list_len);
645-
}
646648
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION)
647649
{
648650
ecma_extended_object_t *ext_func_obj_p = (ecma_extended_object_t *) func_obj_p;
@@ -822,8 +824,8 @@ ecma_op_function_construct (ecma_object_t *func_obj_p, /**< Function object */
822824

823825
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
824826
{
825-
if (unlikely (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION
826-
&& ecma_get_object_is_builtin (func_obj_p)))
827+
if (unlikely (ecma_get_object_is_builtin (func_obj_p)
828+
&& !ecma_builtin_function_is_routine (func_obj_p)))
827829
{
828830
ret_value = ecma_builtin_dispatch_construct (func_obj_p,
829831
arguments_list_p,

0 commit comments

Comments
 (0)