mirror of
https://github.com/jerryscript-project/jerryscript.git
synced 2025-12-15 16:29:21 +00:00
922 lines
33 KiB
C
922 lines
33 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);
|
|
ECMA_SET_INTERNAL_VALUE_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
|
|
uint32_t len = 0;
|
|
|
|
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 = ECMA_GET_INTERNAL_VALUE_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 = args_p->argument_end;
|
|
}
|
|
else
|
|
{
|
|
cbc_uint8_arguments_t *args_p = (cbc_uint8_arguments_t *) bytecode_data_p;
|
|
len = args_p->argument_end;
|
|
}
|
|
|
|
// 15
|
|
ecma_property_t *length_prop_p = ecma_create_named_data_property (obj_p,
|
|
property_name_p,
|
|
ECMA_PROPERTY_FIXED);
|
|
|
|
ecma_named_data_property_assign_value (obj_p, length_prop_p, ecma_make_uint32_value (len));
|
|
|
|
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,
|
|
ECMA_PROPERTY_FLAG_WRITABLE);
|
|
|
|
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 = ECMA_GET_INTERNAL_VALUE_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_INTERNAL_VALUE_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;
|
|
|
|
JMEM_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);
|
|
|
|
JMEM_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_INTERNAL_VALUE_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;
|
|
|
|
JMEM_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);
|
|
|
|
JMEM_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 */
|
|
|
|
/**
|
|
* @}
|
|
* @}
|
|
*/
|