mirror of
https://github.com/jerryscript-project/jerryscript.git
synced 2025-12-15 16:29:21 +00:00
The internals have surprisingly large number of unused functions at surprising places. This patch cleans them out to make maintenance easier and to prevent dead code having effect on future decisions. The patch intentionally does not try to clean up public API but focuses on internal modules only that are/should not be reachable from "outside". However, unit tests do access private API, thus tests of literal storage had to be adjusted. JerryScript-DCO-1.0-Signed-off-by: Akos Kiss akiss@inf.u-szeged.hu
928 lines
34 KiB
C
928 lines
34 KiB
C
/* Copyright 2014-2016 Samsung Electronics Co., Ltd.
|
|
* Copyright 2016 University of Szeged.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "ecma-alloc.h"
|
|
#include "ecma-builtin-helpers.h"
|
|
#include "ecma-builtins.h"
|
|
#include "ecma-exceptions.h"
|
|
#include "ecma-function-object.h"
|
|
#include "ecma-gc.h"
|
|
#include "ecma-helpers.h"
|
|
#include "ecma-lex-env.h"
|
|
#include "ecma-objects.h"
|
|
#include "ecma-objects-general.h"
|
|
#include "ecma-objects-arguments.h"
|
|
#include "ecma-try-catch-macro.h"
|
|
|
|
#define JERRY_INTERNAL
|
|
#include "jerry-internal.h"
|
|
|
|
/** \addtogroup ecma ECMA
|
|
* @{
|
|
*
|
|
* \addtogroup ecmafunctionobject ECMA Function object related routines
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
* IsCallable operation.
|
|
*
|
|
* See also: ECMA-262 v5, 9.11
|
|
*
|
|
* @return true, if value is callable object;
|
|
* false - otherwise.
|
|
*/
|
|
bool
|
|
ecma_op_is_callable (ecma_value_t value) /**< ecma value */
|
|
{
|
|
if (!ecma_is_value_object (value))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
ecma_object_t *obj_p = ecma_get_object_from_value (value);
|
|
|
|
JERRY_ASSERT (obj_p != NULL);
|
|
JERRY_ASSERT (!ecma_is_lexical_environment (obj_p));
|
|
|
|
return (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_FUNCTION
|
|
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION
|
|
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION
|
|
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION);
|
|
} /* ecma_op_is_callable */
|
|
|
|
/**
|
|
* Check whether the value is Object that implements [[Construct]].
|
|
*
|
|
* @return true, if value is constructor object;
|
|
* false - otherwise.
|
|
*/
|
|
bool
|
|
ecma_is_constructor (ecma_value_t value) /**< ecma value */
|
|
{
|
|
if (!ecma_is_value_object (value))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
ecma_object_t *obj_p = ecma_get_object_from_value (value);
|
|
|
|
JERRY_ASSERT (obj_p != NULL);
|
|
JERRY_ASSERT (!ecma_is_lexical_environment (obj_p));
|
|
|
|
return (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_FUNCTION
|
|
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION
|
|
|| ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION);
|
|
} /* ecma_is_constructor */
|
|
|
|
/**
|
|
* Helper function to merge argument lists
|
|
*
|
|
* See also:
|
|
* ECMA-262 v5, 15.3.4.5.1 step 4
|
|
* ECMA-262 v5, 15.3.4.5.2 step 4
|
|
*
|
|
* Used by:
|
|
* - [[Call]] implementation for Function objects.
|
|
* - [[Construct]] implementation for Function objects.
|
|
*/
|
|
static void
|
|
ecma_function_bind_merge_arg_lists (ecma_value_t *merged_args_list_p, /**< destination argument list */
|
|
ecma_collection_header_t *bound_arg_list_p, /**< bound argument list */
|
|
const ecma_value_t *arguments_list_p, /**< source arguments list */
|
|
ecma_length_t arguments_list_len) /**< length of source arguments list */
|
|
{
|
|
/* Performance optimization: only the values are copied. This is
|
|
* enough, since the original references keep these objects alive. */
|
|
|
|
ecma_collection_iterator_t bound_args_iterator;
|
|
ecma_collection_iterator_init (&bound_args_iterator, bound_arg_list_p);
|
|
|
|
for (ecma_length_t i = bound_arg_list_p->unit_number; i > 0; i--)
|
|
{
|
|
bool is_moved = ecma_collection_iterator_next (&bound_args_iterator);
|
|
JERRY_ASSERT (is_moved);
|
|
|
|
*merged_args_list_p++ = *bound_args_iterator.current_value_p;
|
|
}
|
|
|
|
while (arguments_list_len > 0)
|
|
{
|
|
*merged_args_list_p++ = *arguments_list_p++;
|
|
arguments_list_len--;
|
|
}
|
|
} /* ecma_function_bind_merge_arg_lists */
|
|
|
|
/**
|
|
* Function object creation operation.
|
|
*
|
|
* See also: ECMA-262 v5, 13.2
|
|
*
|
|
* @return pointer to newly created Function object
|
|
*/
|
|
ecma_object_t *
|
|
ecma_op_create_function_object (ecma_object_t *scope_p, /**< function's scope */
|
|
bool is_decl_in_strict_mode, /**< is function declared in strict mode code? */
|
|
const ecma_compiled_code_t *bytecode_data_p) /**< byte-code array */
|
|
{
|
|
bool is_strict_mode_code = is_decl_in_strict_mode;
|
|
|
|
if (bytecode_data_p->status_flags & CBC_CODE_FLAGS_STRICT_MODE)
|
|
{
|
|
is_strict_mode_code = true;
|
|
}
|
|
|
|
// 1., 4., 13.
|
|
ecma_object_t *prototype_obj_p = ecma_builtin_get (ECMA_BUILTIN_ID_FUNCTION_PROTOTYPE);
|
|
|
|
ecma_object_t *f = ecma_create_object (prototype_obj_p, true, ECMA_OBJECT_TYPE_FUNCTION);
|
|
|
|
ecma_deref_object (prototype_obj_p);
|
|
|
|
// 2., 6., 7., 8.
|
|
/*
|
|
* We don't setup [[Get]], [[Call]], [[Construct]], [[HasInstance]] for each function object.
|
|
* Instead we set the object's type to ECMA_OBJECT_TYPE_FUNCTION
|
|
* that defines which version of the routine should be used on demand.
|
|
*/
|
|
|
|
// 3.
|
|
/*
|
|
* [[Class]] property is not stored explicitly for objects of ECMA_OBJECT_TYPE_FUNCTION type.
|
|
*
|
|
* See also: ecma_object_get_class_name
|
|
*/
|
|
|
|
// 9.
|
|
ecma_property_t *scope_prop_p = ecma_create_internal_property (f, ECMA_INTERNAL_PROPERTY_SCOPE);
|
|
ECMA_SET_INTERNAL_VALUE_POINTER (ECMA_PROPERTY_VALUE_PTR (scope_prop_p)->value, scope_p);
|
|
|
|
// 10., 11., 12.
|
|
ecma_property_t *bytecode_prop_p = ecma_create_internal_property (f, ECMA_INTERNAL_PROPERTY_CODE_BYTECODE);
|
|
MEM_CP_SET_NON_NULL_POINTER (ECMA_PROPERTY_VALUE_PTR (bytecode_prop_p)->value, bytecode_data_p);
|
|
ecma_bytecode_ref ((ecma_compiled_code_t *) bytecode_data_p);
|
|
|
|
// 14., 15., 16., 17., 18.
|
|
/*
|
|
* 'length' and 'prototype' properties are instantiated lazily
|
|
*
|
|
* See also: ecma_op_function_object_get_own_property
|
|
* ecma_op_function_try_lazy_instantiate_property
|
|
*/
|
|
|
|
// 19.
|
|
if (is_strict_mode_code)
|
|
{
|
|
ecma_object_t *thrower_p = ecma_builtin_get (ECMA_BUILTIN_ID_TYPE_ERROR_THROWER);
|
|
|
|
ecma_property_descriptor_t prop_desc = ecma_make_empty_property_descriptor ();
|
|
{
|
|
prop_desc.is_enumerable_defined = true;
|
|
prop_desc.is_enumerable = false;
|
|
|
|
prop_desc.is_configurable_defined = true;
|
|
prop_desc.is_configurable = false;
|
|
|
|
prop_desc.is_get_defined = true;
|
|
prop_desc.get_p = thrower_p;
|
|
|
|
prop_desc.is_set_defined = true;
|
|
prop_desc.set_p = thrower_p;
|
|
}
|
|
|
|
ecma_string_t *magic_string_caller_p = ecma_get_magic_string (LIT_MAGIC_STRING_CALLER);
|
|
ecma_op_object_define_own_property (f,
|
|
magic_string_caller_p,
|
|
&prop_desc,
|
|
false);
|
|
ecma_deref_ecma_string (magic_string_caller_p);
|
|
|
|
ecma_string_t *magic_string_arguments_p = ecma_get_magic_string (LIT_MAGIC_STRING_ARGUMENTS);
|
|
ecma_op_object_define_own_property (f,
|
|
magic_string_arguments_p,
|
|
&prop_desc,
|
|
false);
|
|
ecma_deref_ecma_string (magic_string_arguments_p);
|
|
|
|
ecma_deref_object (thrower_p);
|
|
}
|
|
|
|
return f;
|
|
} /* ecma_op_create_function_object */
|
|
|
|
/**
|
|
* List names of a Function object's lazy instantiated properties,
|
|
* adding them to corresponding string collections
|
|
*
|
|
* See also:
|
|
* ecma_op_function_try_lazy_instantiate_property
|
|
*/
|
|
void
|
|
ecma_op_function_list_lazy_property_names (bool separate_enumerable, /**< true - list enumerable properties into
|
|
* main collection and non-enumerable
|
|
* to collection of 'skipped
|
|
* non-enumerable' properties,
|
|
* false - list all properties into main
|
|
* collection.
|
|
*/
|
|
ecma_collection_header_t *main_collection_p, /**< 'main' collection */
|
|
ecma_collection_header_t *non_enum_collection_p) /**< skipped
|
|
* 'non-enumerable'
|
|
* collection */
|
|
{
|
|
ecma_collection_header_t *for_enumerable_p = main_collection_p;
|
|
(void) for_enumerable_p;
|
|
|
|
ecma_collection_header_t *for_non_enumerable_p = separate_enumerable ? non_enum_collection_p : main_collection_p;
|
|
|
|
ecma_string_t *name_p;
|
|
|
|
/* 'length' property is non-enumerable (ECMA-262 v5, 13.2.5) */
|
|
name_p = ecma_get_magic_string (LIT_MAGIC_STRING_LENGTH);
|
|
ecma_append_to_values_collection (for_non_enumerable_p, ecma_make_string_value (name_p), true);
|
|
ecma_deref_ecma_string (name_p);
|
|
|
|
/* 'prototype' property is non-enumerable (ECMA-262 v5, 13.2.18) */
|
|
name_p = ecma_get_magic_string (LIT_MAGIC_STRING_PROTOTYPE);
|
|
ecma_append_to_values_collection (for_non_enumerable_p, ecma_make_string_value (name_p), true);
|
|
ecma_deref_ecma_string (name_p);
|
|
} /* ecma_op_function_list_lazy_property_names */
|
|
|
|
/**
|
|
* Lazy instantation of non-builtin ecma function object's properties
|
|
*
|
|
* Warning:
|
|
* Only non-configurable properties could be instantiated lazily in this function,
|
|
* as configurable properties could be deleted and it would be incorrect
|
|
* to reinstantiate them in the function in second time.
|
|
*
|
|
* @return pointer to newly instantiated property, if a property was instantiated,
|
|
* NULL - otherwise
|
|
*/
|
|
static ecma_property_t *
|
|
ecma_op_function_try_lazy_instantiate_property (ecma_object_t *obj_p, /**< the function object */
|
|
ecma_string_t *property_name_p) /**< property name */
|
|
{
|
|
JERRY_ASSERT (!ecma_get_object_is_builtin (obj_p));
|
|
|
|
ecma_string_t *magic_string_length_p = ecma_get_magic_string (LIT_MAGIC_STRING_LENGTH);
|
|
|
|
bool is_length_property = ecma_compare_ecma_strings (magic_string_length_p, property_name_p);
|
|
|
|
ecma_deref_ecma_string (magic_string_length_p);
|
|
|
|
if (is_length_property)
|
|
{
|
|
/* ECMA-262 v5, 13.2, 14-15 */
|
|
|
|
// 14
|
|
ecma_number_t *len_p = ecma_alloc_number ();
|
|
|
|
ecma_property_t *bytecode_prop_p = ecma_get_internal_property (obj_p, ECMA_INTERNAL_PROPERTY_CODE_BYTECODE);
|
|
|
|
const ecma_compiled_code_t *bytecode_data_p;
|
|
bytecode_data_p = MEM_CP_GET_POINTER (const ecma_compiled_code_t,
|
|
ECMA_PROPERTY_VALUE_PTR (bytecode_prop_p)->value);
|
|
|
|
if (bytecode_data_p->status_flags & CBC_CODE_FLAGS_UINT16_ARGUMENTS)
|
|
{
|
|
cbc_uint16_arguments_t *args_p = (cbc_uint16_arguments_t *) bytecode_data_p;
|
|
*len_p = args_p->argument_end;
|
|
}
|
|
else
|
|
{
|
|
cbc_uint8_arguments_t *args_p = (cbc_uint8_arguments_t *) bytecode_data_p;
|
|
*len_p = args_p->argument_end;
|
|
}
|
|
|
|
// 15
|
|
ecma_property_t *length_prop_p = ecma_create_named_data_property (obj_p,
|
|
property_name_p,
|
|
false,
|
|
false,
|
|
false);
|
|
|
|
ecma_named_data_property_assign_value (obj_p, length_prop_p, ecma_make_number_value (len_p));
|
|
|
|
ecma_dealloc_number (len_p);
|
|
|
|
JERRY_ASSERT (!ecma_is_property_configurable (length_prop_p));
|
|
return length_prop_p;
|
|
}
|
|
|
|
ecma_string_t *magic_string_prototype_p = ecma_get_magic_string (LIT_MAGIC_STRING_PROTOTYPE);
|
|
|
|
bool is_prototype_property = ecma_compare_ecma_strings (magic_string_prototype_p, property_name_p);
|
|
|
|
ecma_deref_ecma_string (magic_string_prototype_p);
|
|
|
|
if (is_prototype_property)
|
|
{
|
|
/* ECMA-262 v5, 13.2, 16-18 */
|
|
|
|
// 16.
|
|
ecma_object_t *proto_p = ecma_op_create_object_object_noarg ();
|
|
|
|
// 17.
|
|
ecma_string_t *magic_string_constructor_p = ecma_get_magic_string (LIT_MAGIC_STRING_CONSTRUCTOR);
|
|
ecma_builtin_helper_def_prop (proto_p,
|
|
magic_string_constructor_p,
|
|
ecma_make_object_value (obj_p),
|
|
true, /* Writable */
|
|
false, /* Enumerable */
|
|
true, /* Configurable */
|
|
false); /* Failure handling */
|
|
|
|
ecma_deref_ecma_string (magic_string_constructor_p);
|
|
|
|
// 18.
|
|
ecma_property_t *prototype_prop_p = ecma_create_named_data_property (obj_p,
|
|
property_name_p,
|
|
true,
|
|
false,
|
|
false);
|
|
|
|
ecma_named_data_property_assign_value (obj_p, prototype_prop_p, ecma_make_object_value (proto_p));
|
|
|
|
ecma_deref_object (proto_p);
|
|
|
|
JERRY_ASSERT (!ecma_is_property_configurable (prototype_prop_p));
|
|
return prototype_prop_p;
|
|
}
|
|
|
|
return NULL;
|
|
} /* ecma_op_function_try_lazy_instantiate_property */
|
|
|
|
/**
|
|
* Implementation-defined extension of [[GetOwnProperty]] ecma function object's operation
|
|
*
|
|
* Note:
|
|
* The [[GetOwnProperty]] is used only for lazy property instantiation,
|
|
* i.e. externally visible behaviour of [[GetOwnProperty]] is specification-defined
|
|
*
|
|
* @return pointer to a property - if it already existed
|
|
* or was lazy instantiated in context of
|
|
* current invocation,
|
|
* NULL (i.e. ecma-undefined) - otherwise.
|
|
*/
|
|
ecma_property_t *
|
|
ecma_op_function_object_get_own_property (ecma_object_t *obj_p, /**< the function object */
|
|
ecma_string_t *property_name_p) /**< property name */
|
|
{
|
|
JERRY_ASSERT (ecma_get_object_type (obj_p) == ECMA_OBJECT_TYPE_FUNCTION);
|
|
|
|
ecma_property_t *prop_p = ecma_op_general_object_get_own_property (obj_p, property_name_p);
|
|
|
|
if (prop_p != NULL)
|
|
{
|
|
return prop_p;
|
|
}
|
|
else if (!ecma_get_object_is_builtin (obj_p))
|
|
{
|
|
prop_p = ecma_op_function_try_lazy_instantiate_property (obj_p, property_name_p);
|
|
|
|
/*
|
|
* Only non-configurable properties could be instantiated lazily in the function,
|
|
* as configurable properties could be deleted and it would be incorrect
|
|
* to reinstantiate them in the function in second time.
|
|
*/
|
|
JERRY_ASSERT (prop_p == NULL || !ecma_is_property_configurable (prop_p));
|
|
}
|
|
|
|
return prop_p;
|
|
} /* ecma_op_function_object_get_own_property */
|
|
|
|
/**
|
|
* External function object creation operation.
|
|
*
|
|
* Note:
|
|
* external function object is implementation-defined object type
|
|
* that represent functions implemented in native code, using Embedding API
|
|
*
|
|
* @return pointer to newly created external function object
|
|
*/
|
|
ecma_object_t *
|
|
ecma_op_create_external_function_object (ecma_external_pointer_t code_p) /**< pointer to external native handler */
|
|
{
|
|
ecma_object_t *prototype_obj_p = ecma_builtin_get (ECMA_BUILTIN_ID_FUNCTION_PROTOTYPE);
|
|
|
|
ecma_object_t *function_obj_p = ecma_create_object (prototype_obj_p, true, ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION);
|
|
|
|
ecma_deref_object (prototype_obj_p);
|
|
|
|
/*
|
|
* [[Class]] property is not stored explicitly for objects of ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION type.
|
|
*
|
|
* See also: ecma_object_get_class_name
|
|
*/
|
|
|
|
bool is_created = ecma_create_external_pointer_property (function_obj_p,
|
|
ECMA_INTERNAL_PROPERTY_NATIVE_CODE,
|
|
(ecma_external_pointer_t) code_p);
|
|
JERRY_ASSERT (is_created);
|
|
|
|
ecma_string_t *magic_string_prototype_p = ecma_get_magic_string (LIT_MAGIC_STRING_PROTOTYPE);
|
|
ecma_builtin_helper_def_prop (function_obj_p,
|
|
magic_string_prototype_p,
|
|
ecma_make_simple_value (ECMA_SIMPLE_VALUE_UNDEFINED),
|
|
true, /* Writable */
|
|
false, /* Enumerable */
|
|
false, /* Configurable */
|
|
false); /* Failure handling */
|
|
|
|
ecma_deref_ecma_string (magic_string_prototype_p);
|
|
|
|
return function_obj_p;
|
|
} /* ecma_op_create_external_function_object */
|
|
|
|
/**
|
|
* [[Call]] implementation for Function objects,
|
|
* created through 13.2 (ECMA_OBJECT_TYPE_FUNCTION)
|
|
* or 15.3.4.5 (ECMA_OBJECT_TYPE_BOUND_FUNCTION),
|
|
* and for built-in Function objects
|
|
* from section 15 (ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION).
|
|
*
|
|
* @return ecma value
|
|
* Returned value must be freed with ecma_free_value
|
|
*/
|
|
ecma_value_t
|
|
ecma_op_function_has_instance (ecma_object_t *func_obj_p, /**< Function object */
|
|
ecma_value_t value) /**< argument 'V' */
|
|
{
|
|
JERRY_ASSERT (func_obj_p != NULL
|
|
&& !ecma_is_lexical_environment (func_obj_p));
|
|
|
|
ecma_value_t ret_value = ecma_make_simple_value (ECMA_SIMPLE_VALUE_EMPTY);
|
|
|
|
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
|
|
{
|
|
if (!ecma_is_value_object (value))
|
|
{
|
|
return ecma_make_simple_value (ECMA_SIMPLE_VALUE_FALSE);
|
|
}
|
|
|
|
ecma_object_t *v_obj_p = ecma_get_object_from_value (value);
|
|
|
|
ecma_string_t *prototype_magic_string_p = ecma_get_magic_string (LIT_MAGIC_STRING_PROTOTYPE);
|
|
|
|
ECMA_TRY_CATCH (prototype_obj_value,
|
|
ecma_op_object_get (func_obj_p, prototype_magic_string_p),
|
|
ret_value);
|
|
|
|
if (!ecma_is_value_object (prototype_obj_value))
|
|
{
|
|
ret_value = ecma_raise_type_error (ECMA_ERR_MSG (""));
|
|
}
|
|
else
|
|
{
|
|
ecma_object_t *prototype_obj_p = ecma_get_object_from_value (prototype_obj_value);
|
|
JERRY_ASSERT (prototype_obj_p != NULL);
|
|
|
|
do
|
|
{
|
|
v_obj_p = ecma_get_object_prototype (v_obj_p);
|
|
|
|
if (v_obj_p == NULL)
|
|
{
|
|
ret_value = ecma_make_simple_value (ECMA_SIMPLE_VALUE_FALSE);
|
|
|
|
break;
|
|
}
|
|
else if (v_obj_p == prototype_obj_p)
|
|
{
|
|
ret_value = ecma_make_simple_value (ECMA_SIMPLE_VALUE_TRUE);
|
|
|
|
break;
|
|
}
|
|
} while (true);
|
|
}
|
|
|
|
ECMA_FINALIZE (prototype_obj_value);
|
|
|
|
ecma_deref_ecma_string (prototype_magic_string_p);
|
|
}
|
|
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION ||
|
|
ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION)
|
|
{
|
|
ret_value = ecma_raise_type_error (ECMA_ERR_MSG (""));
|
|
}
|
|
else
|
|
{
|
|
JERRY_ASSERT (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION);
|
|
|
|
/* 1. */
|
|
ecma_property_t *target_function_prop_p;
|
|
target_function_prop_p = ecma_get_internal_property (func_obj_p,
|
|
ECMA_INTERNAL_PROPERTY_BOUND_FUNCTION_TARGET_FUNCTION);
|
|
|
|
ecma_object_t *target_func_obj_p;
|
|
target_func_obj_p = ECMA_GET_INTERNAL_VALUE_POINTER (ecma_object_t,
|
|
ECMA_PROPERTY_VALUE_PTR (target_function_prop_p)->value);
|
|
|
|
/* 3. */
|
|
ret_value = ecma_op_object_has_instance (target_func_obj_p, value);
|
|
}
|
|
|
|
return ret_value;
|
|
} /* ecma_op_function_has_instance */
|
|
|
|
/**
|
|
* [[Call]] implementation for Function objects,
|
|
* created through 13.2 (ECMA_OBJECT_TYPE_FUNCTION)
|
|
* or 15.3.4.5 (ECMA_OBJECT_TYPE_BOUND_FUNCTION),
|
|
* and for built-in Function objects
|
|
* from section 15 (ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION).
|
|
*
|
|
* @return ecma value
|
|
* Returned value must be freed with ecma_free_value
|
|
*/
|
|
ecma_value_t
|
|
ecma_op_function_call (ecma_object_t *func_obj_p, /**< Function object */
|
|
ecma_value_t this_arg_value, /**< 'this' argument's value */
|
|
const ecma_value_t *arguments_list_p, /**< arguments list */
|
|
ecma_length_t arguments_list_len) /**< length of arguments list */
|
|
{
|
|
JERRY_ASSERT (func_obj_p != NULL
|
|
&& !ecma_is_lexical_environment (func_obj_p));
|
|
JERRY_ASSERT (ecma_op_is_callable (ecma_make_object_value (func_obj_p)));
|
|
|
|
ecma_value_t ret_value = ecma_make_simple_value (ECMA_SIMPLE_VALUE_EMPTY);
|
|
|
|
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
|
|
{
|
|
if (unlikely (ecma_get_object_is_builtin (func_obj_p)))
|
|
{
|
|
ret_value = ecma_builtin_dispatch_call (func_obj_p,
|
|
this_arg_value,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
}
|
|
else
|
|
{
|
|
/* Entering Function Code (ECMA-262 v5, 10.4.3) */
|
|
ecma_property_t *scope_prop_p = ecma_get_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_SCOPE);
|
|
ecma_property_t *bytecode_prop_p = ecma_get_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_CODE_BYTECODE);
|
|
|
|
ecma_object_t *scope_p = ECMA_GET_INTERNAL_VALUE_POINTER (ecma_object_t,
|
|
ECMA_PROPERTY_VALUE_PTR (scope_prop_p)->value);
|
|
|
|
// 8.
|
|
ecma_value_t this_binding;
|
|
bool is_strict;
|
|
bool is_no_lex_env;
|
|
|
|
const ecma_compiled_code_t *bytecode_data_p;
|
|
bytecode_data_p = MEM_CP_GET_POINTER (const ecma_compiled_code_t,
|
|
ECMA_PROPERTY_VALUE_PTR (bytecode_prop_p)->value);
|
|
|
|
is_strict = (bytecode_data_p->status_flags & CBC_CODE_FLAGS_STRICT_MODE) ? true : false;
|
|
is_no_lex_env = (bytecode_data_p->status_flags & CBC_CODE_FLAGS_LEXICAL_ENV_NOT_NEEDED) ? true : false;
|
|
|
|
// 1.
|
|
if (is_strict)
|
|
{
|
|
this_binding = ecma_copy_value (this_arg_value);
|
|
}
|
|
else if (ecma_is_value_undefined (this_arg_value)
|
|
|| ecma_is_value_null (this_arg_value))
|
|
{
|
|
// 2.
|
|
this_binding = ecma_make_object_value (ecma_builtin_get (ECMA_BUILTIN_ID_GLOBAL));
|
|
}
|
|
else
|
|
{
|
|
// 3., 4.
|
|
this_binding = ecma_op_to_object (this_arg_value);
|
|
|
|
JERRY_ASSERT (!ecma_is_value_error (this_binding));
|
|
}
|
|
|
|
// 5.
|
|
ecma_object_t *local_env_p;
|
|
if (is_no_lex_env)
|
|
{
|
|
local_env_p = scope_p;
|
|
}
|
|
else
|
|
{
|
|
local_env_p = ecma_create_decl_lex_env (scope_p);
|
|
#ifndef CONFIG_ECMA_COMPACT_PROFILE
|
|
if (bytecode_data_p->status_flags & CBC_CODE_FLAGS_ARGUMENTS_NEEDED)
|
|
{
|
|
ecma_op_create_arguments_object (func_obj_p,
|
|
local_env_p,
|
|
arguments_list_p,
|
|
arguments_list_len,
|
|
bytecode_data_p);
|
|
}
|
|
#endif /* !CONFIG_ECMA_COMPACT_PROFILE */
|
|
}
|
|
|
|
ret_value = vm_run (bytecode_data_p,
|
|
this_binding,
|
|
local_env_p,
|
|
false,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
|
|
if (!is_no_lex_env)
|
|
{
|
|
ecma_deref_object (local_env_p);
|
|
}
|
|
|
|
ecma_free_value (this_binding);
|
|
}
|
|
}
|
|
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION)
|
|
{
|
|
ret_value = ecma_builtin_dispatch_call (func_obj_p,
|
|
this_arg_value,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
}
|
|
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION)
|
|
{
|
|
ecma_external_pointer_t handler_p;
|
|
bool is_retrieved = ecma_get_external_pointer_value (func_obj_p,
|
|
ECMA_INTERNAL_PROPERTY_NATIVE_CODE,
|
|
&handler_p);
|
|
JERRY_ASSERT (is_retrieved);
|
|
|
|
ecma_collection_header_t *arg_collection_p = ecma_new_values_collection (arguments_list_p,
|
|
arguments_list_len,
|
|
true);
|
|
|
|
ret_value = jerry_dispatch_external_function (func_obj_p,
|
|
handler_p,
|
|
this_arg_value,
|
|
arg_collection_p);
|
|
|
|
ecma_free_values_collection (arg_collection_p, true);
|
|
}
|
|
else
|
|
{
|
|
JERRY_ASSERT (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION);
|
|
|
|
/* 2-3. */
|
|
ecma_property_t *bound_this_prop_p;
|
|
ecma_property_t *target_function_prop_p;
|
|
|
|
bound_this_prop_p = ecma_get_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_BOUND_FUNCTION_BOUND_THIS);
|
|
target_function_prop_p = ecma_get_internal_property (func_obj_p,
|
|
ECMA_INTERNAL_PROPERTY_BOUND_FUNCTION_TARGET_FUNCTION);
|
|
|
|
ecma_object_t *target_func_obj_p;
|
|
target_func_obj_p = ECMA_GET_INTERNAL_VALUE_POINTER (ecma_object_t,
|
|
ECMA_PROPERTY_VALUE_PTR (target_function_prop_p)->value);
|
|
|
|
/* 4. */
|
|
ecma_property_t *bound_args_prop_p;
|
|
ecma_value_t bound_this_value = ECMA_PROPERTY_VALUE_PTR (bound_this_prop_p)->value;
|
|
bound_args_prop_p = ecma_find_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_BOUND_FUNCTION_BOUND_ARGS);
|
|
|
|
if (bound_args_prop_p != NULL)
|
|
{
|
|
ecma_collection_header_t *bound_arg_list_p;
|
|
bound_arg_list_p = ECMA_GET_NON_NULL_POINTER (ecma_collection_header_t,
|
|
ECMA_PROPERTY_VALUE_PTR (bound_args_prop_p)->value);
|
|
|
|
JERRY_ASSERT (bound_arg_list_p->unit_number > 0);
|
|
|
|
ecma_length_t merged_args_list_len = bound_arg_list_p->unit_number + arguments_list_len;
|
|
|
|
MEM_DEFINE_LOCAL_ARRAY (merged_args_list_p, merged_args_list_len, ecma_value_t);
|
|
|
|
ecma_function_bind_merge_arg_lists (merged_args_list_p,
|
|
bound_arg_list_p,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
|
|
/* 5. */
|
|
ret_value = ecma_op_function_call (target_func_obj_p,
|
|
bound_this_value,
|
|
merged_args_list_p,
|
|
merged_args_list_len);
|
|
|
|
MEM_FINALIZE_LOCAL_ARRAY (merged_args_list_p);
|
|
}
|
|
else
|
|
{
|
|
/* 5. */
|
|
ret_value = ecma_op_function_call (target_func_obj_p,
|
|
bound_this_value,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
}
|
|
}
|
|
|
|
JERRY_ASSERT (!ecma_is_value_empty (ret_value));
|
|
|
|
return ret_value;
|
|
} /* ecma_op_function_call */
|
|
|
|
/**
|
|
* [[Construct]] implementation for Function objects (13.2.2),
|
|
* created through 13.2 (ECMA_OBJECT_TYPE_FUNCTION) and
|
|
* externally defined (host) functions (ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION).
|
|
*
|
|
* @return ecma value
|
|
* Returned value must be freed with ecma_free_value
|
|
*/
|
|
static ecma_value_t
|
|
ecma_op_function_construct_simple_or_external (ecma_object_t *func_obj_p, /**< Function object */
|
|
const ecma_value_t *arguments_list_p, /**< arguments list */
|
|
ecma_length_t arguments_list_len) /**< length of arguments list */
|
|
{
|
|
JERRY_ASSERT (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION
|
|
|| ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION);
|
|
|
|
ecma_value_t ret_value = ecma_make_simple_value (ECMA_SIMPLE_VALUE_EMPTY);
|
|
|
|
ecma_string_t *prototype_magic_string_p = ecma_get_magic_string (LIT_MAGIC_STRING_PROTOTYPE);
|
|
|
|
// 5.
|
|
ECMA_TRY_CATCH (func_obj_prototype_prop_value,
|
|
ecma_op_object_get (func_obj_p,
|
|
prototype_magic_string_p),
|
|
ret_value);
|
|
|
|
// 1., 2., 4.
|
|
ecma_object_t *obj_p;
|
|
if (ecma_is_value_object (func_obj_prototype_prop_value))
|
|
{
|
|
// 6.
|
|
obj_p = ecma_create_object (ecma_get_object_from_value (func_obj_prototype_prop_value),
|
|
true,
|
|
ECMA_OBJECT_TYPE_GENERAL);
|
|
}
|
|
else
|
|
{
|
|
// 7.
|
|
ecma_object_t *prototype_p = ecma_builtin_get (ECMA_BUILTIN_ID_OBJECT_PROTOTYPE);
|
|
|
|
obj_p = ecma_create_object (prototype_p, true, ECMA_OBJECT_TYPE_GENERAL);
|
|
|
|
ecma_deref_object (prototype_p);
|
|
}
|
|
|
|
// 3.
|
|
/*
|
|
* [[Class]] property of ECMA_OBJECT_TYPE_GENERAL type objects
|
|
* without ECMA_INTERNAL_PROPERTY_CLASS internal property
|
|
* is "Object".
|
|
*
|
|
* See also: ecma_object_get_class_name.
|
|
*/
|
|
|
|
// 8.
|
|
ECMA_TRY_CATCH (call_completion,
|
|
ecma_op_function_call (func_obj_p,
|
|
ecma_make_object_value (obj_p),
|
|
arguments_list_p,
|
|
arguments_list_len),
|
|
ret_value);
|
|
|
|
// 9.
|
|
if (ecma_is_value_object (call_completion))
|
|
{
|
|
ret_value = ecma_copy_value (call_completion);
|
|
}
|
|
else
|
|
{
|
|
// 10.
|
|
ecma_ref_object (obj_p);
|
|
ret_value = ecma_make_object_value (obj_p);
|
|
}
|
|
|
|
ECMA_FINALIZE (call_completion);
|
|
|
|
ecma_deref_object (obj_p);
|
|
|
|
ECMA_FINALIZE (func_obj_prototype_prop_value);
|
|
|
|
ecma_deref_ecma_string (prototype_magic_string_p);
|
|
|
|
return ret_value;
|
|
} /* ecma_op_function_construct_simple_or_external */
|
|
|
|
/**
|
|
* [[Construct]] implementation:
|
|
* 13.2.2 - for Function objects, created through 13.2 (ECMA_OBJECT_TYPE_FUNCTION),
|
|
* and externally defined host functions (ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION);
|
|
* 15.3.4.5.1 - for Function objects, created through 15.3.4.5 (ECMA_OBJECT_TYPE_BOUND_FUNCTION).
|
|
*
|
|
* @return ecma value
|
|
* Returned value must be freed with ecma_free_value
|
|
*/
|
|
ecma_value_t
|
|
ecma_op_function_construct (ecma_object_t *func_obj_p, /**< Function object */
|
|
const ecma_value_t *arguments_list_p, /**< arguments list */
|
|
ecma_length_t arguments_list_len) /**< length of arguments list */
|
|
{
|
|
JERRY_ASSERT (func_obj_p != NULL
|
|
&& !ecma_is_lexical_environment (func_obj_p));
|
|
JERRY_ASSERT (ecma_is_constructor (ecma_make_object_value (func_obj_p)));
|
|
|
|
ecma_value_t ret_value = ecma_make_simple_value (ECMA_SIMPLE_VALUE_EMPTY);
|
|
|
|
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
|
|
{
|
|
if (unlikely (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION
|
|
&& ecma_get_object_is_builtin (func_obj_p)))
|
|
{
|
|
ret_value = ecma_builtin_dispatch_construct (func_obj_p,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
}
|
|
else
|
|
{
|
|
ret_value = ecma_op_function_construct_simple_or_external (func_obj_p,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
}
|
|
}
|
|
else if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION)
|
|
{
|
|
ret_value = ecma_op_function_construct_simple_or_external (func_obj_p,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
}
|
|
else
|
|
{
|
|
JERRY_ASSERT (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_BOUND_FUNCTION);
|
|
|
|
/* 1. */
|
|
ecma_property_t *target_function_prop_p;
|
|
target_function_prop_p = ecma_get_internal_property (func_obj_p,
|
|
ECMA_INTERNAL_PROPERTY_BOUND_FUNCTION_TARGET_FUNCTION);
|
|
|
|
ecma_object_t *target_func_obj_p;
|
|
target_func_obj_p = ECMA_GET_INTERNAL_VALUE_POINTER (ecma_object_t,
|
|
ECMA_PROPERTY_VALUE_PTR (target_function_prop_p)->value);
|
|
|
|
/* 2. */
|
|
if (!ecma_is_constructor (ecma_make_object_value (target_func_obj_p)))
|
|
{
|
|
ret_value = ecma_raise_type_error (ECMA_ERR_MSG (""));
|
|
}
|
|
else
|
|
{
|
|
/* 4. */
|
|
ecma_property_t *bound_args_prop_p;
|
|
bound_args_prop_p = ecma_find_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_BOUND_FUNCTION_BOUND_ARGS);
|
|
|
|
if (bound_args_prop_p != NULL)
|
|
{
|
|
ecma_collection_header_t *bound_arg_list_p;
|
|
bound_arg_list_p = ECMA_GET_NON_NULL_POINTER (ecma_collection_header_t,
|
|
ECMA_PROPERTY_VALUE_PTR (bound_args_prop_p)->value);
|
|
|
|
JERRY_ASSERT (bound_arg_list_p->unit_number > 0);
|
|
|
|
ecma_length_t merged_args_list_len = bound_arg_list_p->unit_number + arguments_list_len;
|
|
|
|
MEM_DEFINE_LOCAL_ARRAY (merged_args_list_p, merged_args_list_len, ecma_value_t);
|
|
|
|
ecma_function_bind_merge_arg_lists (merged_args_list_p,
|
|
bound_arg_list_p,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
|
|
/* 5. */
|
|
ret_value = ecma_op_function_construct (target_func_obj_p,
|
|
merged_args_list_p,
|
|
merged_args_list_len);
|
|
|
|
MEM_FINALIZE_LOCAL_ARRAY (merged_args_list_p);
|
|
}
|
|
else
|
|
{
|
|
/* 5. */
|
|
ret_value = ecma_op_function_construct (target_func_obj_p,
|
|
arguments_list_p,
|
|
arguments_list_len);
|
|
}
|
|
}
|
|
}
|
|
|
|
return ret_value;
|
|
} /* ecma_op_function_construct */
|
|
|
|
/**
|
|
* @}
|
|
* @}
|
|
*/
|