wip: Implementing external function api
This commit is contained in:
@@ -89,6 +89,7 @@ DECLARE_ROUTINES_FOR (collection_chunk)
|
|||||||
DECLARE_ROUTINES_FOR (string)
|
DECLARE_ROUTINES_FOR (string)
|
||||||
DECLARE_ROUTINES_FOR (label_descriptor)
|
DECLARE_ROUTINES_FOR (label_descriptor)
|
||||||
DECLARE_ROUTINES_FOR (getter_setter_pointers)
|
DECLARE_ROUTINES_FOR (getter_setter_pointers)
|
||||||
|
DECLARE_ROUTINES_FOR (external_pointer)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
|
|||||||
@@ -121,6 +121,18 @@ extern ecma_getter_setter_pointers_t *ecma_alloc_getter_setter_pointers (void);
|
|||||||
*/
|
*/
|
||||||
extern void ecma_dealloc_getter_setter_pointers (ecma_getter_setter_pointers_t *pointer_pair_p);
|
extern void ecma_dealloc_getter_setter_pointers (ecma_getter_setter_pointers_t *pointer_pair_p);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allocate memory for external pointer
|
||||||
|
*
|
||||||
|
* @return pointer to allocated memory
|
||||||
|
*/
|
||||||
|
extern ecma_external_pointer_t *ecma_alloc_external_pointer (void);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dealloc memory from external pointer
|
||||||
|
*/
|
||||||
|
extern void ecma_dealloc_external_pointer (ecma_external_pointer_t *ptr_p);
|
||||||
|
|
||||||
|
|
||||||
#endif /* JERRY_ECMA_ALLOC_H */
|
#endif /* JERRY_ECMA_ALLOC_H */
|
||||||
|
|
||||||
|
|||||||
@@ -372,6 +372,8 @@ typedef enum
|
|||||||
and not host objects */
|
and not host objects */
|
||||||
ECMA_OBJECT_TYPE_STRING, /**< String objects (15.5) */
|
ECMA_OBJECT_TYPE_STRING, /**< String objects (15.5) */
|
||||||
ECMA_OBJECT_TYPE_FUNCTION, /**< Function objects (15.3), created through 13.2 routine */
|
ECMA_OBJECT_TYPE_FUNCTION, /**< Function objects (15.3), created through 13.2 routine */
|
||||||
|
ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION, /**< Function object (15.3), created through 13.2 routine
|
||||||
|
but [[Code]] is in external handler. */
|
||||||
ECMA_OBJECT_TYPE_BOUND_FUNCTION, /**< Function objects (15.3), created through 15.3.4.5 routine */
|
ECMA_OBJECT_TYPE_BOUND_FUNCTION, /**< Function objects (15.3), created through 15.3.4.5 routine */
|
||||||
ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION, /** One of built-in functions described in section 15
|
ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION, /** One of built-in functions described in section 15
|
||||||
of ECMA-262 v5 specification */
|
of ECMA-262 v5 specification */
|
||||||
@@ -810,6 +812,9 @@ typedef struct ecma_string_t
|
|||||||
} u;
|
} u;
|
||||||
} ecma_string_t;
|
} ecma_string_t;
|
||||||
|
|
||||||
|
|
||||||
|
typedef uintptr_t ecma_external_pointer_t;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -25,6 +25,7 @@
|
|||||||
#include "ecma-objects-general.h"
|
#include "ecma-objects-general.h"
|
||||||
#include "ecma-try-catch-macro.h"
|
#include "ecma-try-catch-macro.h"
|
||||||
#include "jrt.h"
|
#include "jrt.h"
|
||||||
|
#include "jerry-api.h"
|
||||||
|
|
||||||
/** \addtogroup ecma ECMA
|
/** \addtogroup ecma ECMA
|
||||||
* @{
|
* @{
|
||||||
@@ -78,6 +79,51 @@ ecma_unpack_code_internal_property_value (uint32_t value, /**< packed value */
|
|||||||
return opcode_idx;
|
return opcode_idx;
|
||||||
} /* ecma_unpack_code_internal_property_value */
|
} /* ecma_unpack_code_internal_property_value */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pack the pointer to external handler to value
|
||||||
|
* that can be stored in an [[Code]] internal property.
|
||||||
|
*
|
||||||
|
* @return packed value
|
||||||
|
*/
|
||||||
|
static uint32_t
|
||||||
|
ecma_pack_external_code_internal_property_value (jerry_external_handler_t handler)
|
||||||
|
{
|
||||||
|
uint32_t value;
|
||||||
|
if (sizeof (jerry_external_handler_t) == sizeof (uint32_t))
|
||||||
|
{
|
||||||
|
value = static_cast <uint32_t> (reinterpret_cast <uintptr_t> (handler));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ecma_external_pointer_t *handler_p = ecma_alloc_external_pointer ();
|
||||||
|
*handler_p = reinterpret_cast <ecma_external_pointer_t> (handler);
|
||||||
|
ECMA_SET_POINTER (value, handler_p);
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
} /* ecma_pack_external_code_internal_property_value */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unpack 'is_strict' flag and opcode index from value
|
||||||
|
* that can be stored in an [[Code]] internal property.
|
||||||
|
*
|
||||||
|
* @return opcode index
|
||||||
|
*/
|
||||||
|
static jerry_external_handler_t
|
||||||
|
ecma_unpack_external_code_internal_property_value (uint32_t value) /**< packed value */
|
||||||
|
{
|
||||||
|
jerry_external_handler_t handler;
|
||||||
|
if (sizeof (ecma_external_pointer_t) == sizeof (uint32_t))
|
||||||
|
{
|
||||||
|
handler = reinterpret_cast <jerry_external_handler_t> (static_cast <uintptr_t> (value));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ecma_external_pointer_t* handler_p = ECMA_GET_POINTER (ecma_external_pointer_t, value);
|
||||||
|
handler = reinterpret_cast <jerry_external_handler_t> (*handler_p);
|
||||||
|
}
|
||||||
|
return handler;
|
||||||
|
} /* ecma_unpack_code_internal_property_value */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* IsCallable operation.
|
* IsCallable operation.
|
||||||
*
|
*
|
||||||
@@ -139,12 +185,21 @@ ecma_op_create_function_object (ecma_string_t* formal_parameter_list_p[], /**< f
|
|||||||
ecma_length_t formal_parameters_number, /**< formal parameters list's length */
|
ecma_length_t formal_parameters_number, /**< formal parameters list's length */
|
||||||
ecma_object_t *scope_p, /**< function's scope */
|
ecma_object_t *scope_p, /**< function's scope */
|
||||||
bool is_strict, /**< 'strict' flag */
|
bool is_strict, /**< 'strict' flag */
|
||||||
opcode_counter_t first_opcode_idx) /**< index of first opcode of function's body */
|
bool is_external, /**< true if the function is an external */
|
||||||
|
opcode_counter_t first_opcode_idx, /**< index of first opcode of function's body */
|
||||||
|
jerry_external_handler_t external_handler) /**< external handler */
|
||||||
{
|
{
|
||||||
|
JERRY_ASSERT (!is_external || external_handler != NULL);
|
||||||
|
|
||||||
// 1., 4., 13.
|
// 1., 4., 13.
|
||||||
ecma_object_t *prototype_obj_p = ecma_builtin_get (ECMA_BUILTIN_ID_FUNCTION_PROTOTYPE);
|
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_object_type_t object_type = ECMA_OBJECT_TYPE_FUNCTION;
|
||||||
|
if (is_external)
|
||||||
|
{
|
||||||
|
object_type = ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION;
|
||||||
|
}
|
||||||
|
ecma_object_t *f = ecma_create_object (prototype_obj_p, true, object_type);
|
||||||
|
|
||||||
ecma_deref_object (prototype_obj_p);
|
ecma_deref_object (prototype_obj_p);
|
||||||
|
|
||||||
@@ -179,8 +234,16 @@ ecma_op_create_function_object (ecma_string_t* formal_parameter_list_p[], /**< f
|
|||||||
|
|
||||||
// 12.
|
// 12.
|
||||||
ecma_property_t *code_prop_p = ecma_create_internal_property (f, ECMA_INTERNAL_PROPERTY_CODE);
|
ecma_property_t *code_prop_p = ecma_create_internal_property (f, ECMA_INTERNAL_PROPERTY_CODE);
|
||||||
code_prop_p->u.internal_property.value = ecma_pack_code_internal_property_value (is_strict,
|
if (is_external)
|
||||||
first_opcode_idx);
|
{
|
||||||
|
code_prop_p->u.internal_property.value = ecma_pack_external_code_internal_property_value (external_handler);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
code_prop_p->u.internal_property.value = ecma_pack_code_internal_property_value (is_strict,
|
||||||
|
first_opcode_idx);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// 14.
|
// 14.
|
||||||
ecma_number_t* len_p = ecma_alloc_number ();
|
ecma_number_t* len_p = ecma_alloc_number ();
|
||||||
@@ -467,7 +530,8 @@ ecma_op_function_call (ecma_object_t *func_obj_p, /**< Function object */
|
|||||||
|
|
||||||
ecma_completion_value_t ret_value;
|
ecma_completion_value_t ret_value;
|
||||||
|
|
||||||
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
|
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION
|
||||||
|
|| ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION)
|
||||||
{
|
{
|
||||||
if (unlikely (ecma_get_object_is_builtin (func_obj_p)))
|
if (unlikely (ecma_get_object_is_builtin (func_obj_p)))
|
||||||
{
|
{
|
||||||
@@ -480,15 +544,24 @@ ecma_op_function_call (ecma_object_t *func_obj_p, /**< Function object */
|
|||||||
{
|
{
|
||||||
/* Entering Function Code (ECMA-262 v5, 10.4.3) */
|
/* 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 *scope_prop_p = ecma_get_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_SCOPE);
|
||||||
ecma_property_t *code_prop_p = ecma_get_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_CODE);
|
|
||||||
|
|
||||||
ecma_object_t *scope_p = ECMA_GET_NON_NULL_POINTER (ecma_object_t,
|
ecma_object_t *scope_p = ECMA_GET_NON_NULL_POINTER (ecma_object_t,
|
||||||
scope_prop_p->u.internal_property.value);
|
scope_prop_p->u.internal_property.value);
|
||||||
|
|
||||||
|
// 8.
|
||||||
|
bool is_strict;
|
||||||
|
opcode_counter_t code_first_opcode_idx = 0;
|
||||||
|
|
||||||
|
ecma_property_t *code_prop_p = ecma_get_internal_property (func_obj_p, ECMA_INTERNAL_PROPERTY_CODE);
|
||||||
uint32_t code_prop_value = code_prop_p->u.internal_property.value;
|
uint32_t code_prop_value = code_prop_p->u.internal_property.value;
|
||||||
|
|
||||||
bool is_strict;
|
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
|
||||||
// 8.
|
{
|
||||||
opcode_counter_t code_first_opcode_idx = ecma_unpack_code_internal_property_value (code_prop_value, &is_strict);
|
code_first_opcode_idx = ecma_unpack_code_internal_property_value (code_prop_value, &is_strict);
|
||||||
|
}
|
||||||
|
else /* ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION */
|
||||||
|
{
|
||||||
|
is_strict = false;
|
||||||
|
}
|
||||||
|
|
||||||
ecma_value_t this_binding;
|
ecma_value_t this_binding;
|
||||||
// 1.
|
// 1.
|
||||||
@@ -514,7 +587,7 @@ ecma_op_function_call (ecma_object_t *func_obj_p, /**< Function object */
|
|||||||
// 5.
|
// 5.
|
||||||
ecma_object_t *local_env_p = ecma_create_decl_lex_env (scope_p);
|
ecma_object_t *local_env_p = ecma_create_decl_lex_env (scope_p);
|
||||||
|
|
||||||
// 9.
|
|
||||||
ECMA_TRY_CATCH (args_var_declaration_ret,
|
ECMA_TRY_CATCH (args_var_declaration_ret,
|
||||||
ecma_function_call_setup_args_variables (func_obj_p,
|
ecma_function_call_setup_args_variables (func_obj_p,
|
||||||
local_env_p,
|
local_env_p,
|
||||||
@@ -523,18 +596,29 @@ ecma_op_function_call (ecma_object_t *func_obj_p, /**< Function object */
|
|||||||
is_strict),
|
is_strict),
|
||||||
ret_value);
|
ret_value);
|
||||||
|
|
||||||
ecma_completion_value_t completion = run_int_from_pos (code_first_opcode_idx,
|
// 9.
|
||||||
this_binding,
|
if (ecma_get_object_type (func_obj_p) == ECMA_OBJECT_TYPE_FUNCTION)
|
||||||
local_env_p,
|
|
||||||
is_strict,
|
|
||||||
false);
|
|
||||||
if (ecma_is_completion_value_return (completion))
|
|
||||||
{
|
{
|
||||||
ret_value = ecma_make_normal_completion_value (ecma_get_completion_value_value (completion));
|
ecma_completion_value_t completion = run_int_from_pos (code_first_opcode_idx,
|
||||||
|
this_binding,
|
||||||
|
local_env_p,
|
||||||
|
is_strict,
|
||||||
|
false);
|
||||||
|
if (ecma_is_completion_value_return (completion))
|
||||||
|
{
|
||||||
|
ret_value = ecma_make_normal_completion_value (ecma_get_completion_value_value (completion));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ret_value = completion;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ret_value = completion;
|
jerry_external_handler_t handler = ecma_unpack_external_code_internal_property_value (code_prop_value);
|
||||||
|
handler (NULL, NULL, 0, NULL);
|
||||||
|
ret_value = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
ECMA_FINALIZE (args_var_declaration_ret);
|
ECMA_FINALIZE (args_var_declaration_ret);
|
||||||
@@ -681,7 +765,9 @@ ecma_op_function_declaration (ecma_object_t *lex_env_p, /**< lexical environment
|
|||||||
formal_parameter_list_length,
|
formal_parameter_list_length,
|
||||||
lex_env_p,
|
lex_env_p,
|
||||||
is_strict,
|
is_strict,
|
||||||
function_code_opcode_idx);
|
false,
|
||||||
|
function_code_opcode_idx,
|
||||||
|
0);
|
||||||
|
|
||||||
// c.
|
// c.
|
||||||
bool func_already_declared = ecma_op_has_binding (lex_env_p, function_name_p);
|
bool func_already_declared = ecma_op_has_binding (lex_env_p, function_name_p);
|
||||||
|
|||||||
@@ -18,6 +18,7 @@
|
|||||||
|
|
||||||
#include "ecma-globals.h"
|
#include "ecma-globals.h"
|
||||||
#include "vm.h"
|
#include "vm.h"
|
||||||
|
#include "jerry-api.h"
|
||||||
|
|
||||||
/** \addtogroup ecma ECMA
|
/** \addtogroup ecma ECMA
|
||||||
* @{
|
* @{
|
||||||
@@ -34,7 +35,9 @@ ecma_op_create_function_object (ecma_string_t* formal_parameter_list_p[],
|
|||||||
ecma_length_t formal_parameters_number,
|
ecma_length_t formal_parameters_number,
|
||||||
ecma_object_t *scope_p,
|
ecma_object_t *scope_p,
|
||||||
bool is_strict,
|
bool is_strict,
|
||||||
opcode_counter_t first_opcode_idx);
|
bool is_external,
|
||||||
|
opcode_counter_t first_opcode_idx,
|
||||||
|
jerry_external_handler_t external_handler);
|
||||||
|
|
||||||
extern ecma_completion_value_t
|
extern ecma_completion_value_t
|
||||||
ecma_op_function_call (ecma_object_t *func_obj_p,
|
ecma_op_function_call (ecma_object_t *func_obj_p,
|
||||||
|
|||||||
@@ -483,6 +483,7 @@ ecma_op_object_has_instance (ecma_object_t *obj_p, /**< the object */
|
|||||||
}
|
}
|
||||||
|
|
||||||
case ECMA_OBJECT_TYPE_FUNCTION:
|
case ECMA_OBJECT_TYPE_FUNCTION:
|
||||||
|
case ECMA_OBJECT_TYPE_EXTERNAL_FUNCTION:
|
||||||
case ECMA_OBJECT_TYPE_BOUND_FUNCTION:
|
case ECMA_OBJECT_TYPE_BOUND_FUNCTION:
|
||||||
case ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION:
|
case ECMA_OBJECT_TYPE_BUILT_IN_FUNCTION:
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -80,6 +80,14 @@ typedef struct jerry_api_value_t
|
|||||||
};
|
};
|
||||||
} jerry_api_value_t;
|
} jerry_api_value_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Jerry external function handler type
|
||||||
|
*/
|
||||||
|
typedef bool (*jerry_external_handler_t) (const jerry_api_value_t *this_p,
|
||||||
|
const jerry_api_value_t *args_p [],
|
||||||
|
const int16_t args_cnt,
|
||||||
|
jerry_api_value_t *ret_val_p);
|
||||||
|
|
||||||
extern EXTERN_C ssize_t
|
extern EXTERN_C ssize_t
|
||||||
jerry_api_string_to_char_buffer (const jerry_api_string_t *string_p,
|
jerry_api_string_to_char_buffer (const jerry_api_string_t *string_p,
|
||||||
char *buffer_p,
|
char *buffer_p,
|
||||||
@@ -101,6 +109,8 @@ extern EXTERN_C
|
|||||||
jerry_api_string_t* jerry_api_create_string (const char *v);
|
jerry_api_string_t* jerry_api_create_string (const char *v);
|
||||||
extern EXTERN_C
|
extern EXTERN_C
|
||||||
jerry_api_object_t* jerry_api_create_object (void);
|
jerry_api_object_t* jerry_api_create_object (void);
|
||||||
|
extern EXTERN_C
|
||||||
|
jerry_api_object_t* jerry_api_create_external_function (jerry_external_handler_t handler);
|
||||||
|
|
||||||
extern EXTERN_C
|
extern EXTERN_C
|
||||||
bool jerry_api_is_function (const jerry_api_object_t *object_p);
|
bool jerry_api_is_function (const jerry_api_object_t *object_p);
|
||||||
|
|||||||
@@ -23,6 +23,7 @@
|
|||||||
#include "ecma-gc.h"
|
#include "ecma-gc.h"
|
||||||
#include "ecma-helpers.h"
|
#include "ecma-helpers.h"
|
||||||
#include "ecma-init-finalize.h"
|
#include "ecma-init-finalize.h"
|
||||||
|
#include "ecma-lex-env.h"
|
||||||
#include "ecma-objects.h"
|
#include "ecma-objects.h"
|
||||||
#include "ecma-objects-general.h"
|
#include "ecma-objects-general.h"
|
||||||
#include "jerry.h"
|
#include "jerry.h"
|
||||||
@@ -341,6 +342,26 @@ jerry_api_create_object (void)
|
|||||||
return ecma_op_create_object_object_noarg ();
|
return ecma_op_create_object_object_noarg ();
|
||||||
} /* jerry_api_create_object */
|
} /* jerry_api_create_object */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an external function object
|
||||||
|
*
|
||||||
|
* Note:
|
||||||
|
* caller should release the object with jerry_api_release_object, just when the value becomes unnecessary.
|
||||||
|
*
|
||||||
|
* @return pointer to created external function object
|
||||||
|
*/
|
||||||
|
jerry_api_object_t*
|
||||||
|
jerry_api_create_external_function (jerry_external_handler_t handler)
|
||||||
|
{
|
||||||
|
return ecma_op_create_function_object (NULL,
|
||||||
|
0,
|
||||||
|
ecma_get_globl_lexical_environment (),
|
||||||
|
false,
|
||||||
|
0,
|
||||||
|
true,
|
||||||
|
handler);
|
||||||
|
} /* jerry_api_create_object */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if the specified object is a function object.
|
* Check if the specified object is a function object.
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -580,7 +580,9 @@ opfunc_func_expr_n (opcode_t opdata, /**< operation data */
|
|||||||
params_number,
|
params_number,
|
||||||
scope_p,
|
scope_p,
|
||||||
is_strict,
|
is_strict,
|
||||||
int_data->pos);
|
false,
|
||||||
|
int_data->pos,
|
||||||
|
0);
|
||||||
|
|
||||||
ret_value = set_variable_value (int_data, lit_oc,
|
ret_value = set_variable_value (int_data, lit_oc,
|
||||||
dst_var_idx,
|
dst_var_idx,
|
||||||
|
|||||||
@@ -59,6 +59,17 @@ test_api_init_api_value_string (jerry_api_value_t *out_value_p, /**< out: API va
|
|||||||
out_value_p->v_string = jerry_api_create_string (v);
|
out_value_p->v_string = jerry_api_create_string (v);
|
||||||
} /* test_api_init_api_value_string */
|
} /* test_api_init_api_value_string */
|
||||||
|
|
||||||
|
static bool
|
||||||
|
handler (const jerry_api_value_t *this_p,
|
||||||
|
const jerry_api_value_t *args_p [],
|
||||||
|
const int16_t args_cnt,
|
||||||
|
jerry_api_value_t *ret_val_p)
|
||||||
|
{
|
||||||
|
printf("ok %p %p %d %p\n", this_p, args_p, args_cnt, ret_val_p);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int
|
int
|
||||||
main (void)
|
main (void)
|
||||||
{
|
{
|
||||||
@@ -68,6 +79,7 @@ main (void)
|
|||||||
ssize_t sz;
|
ssize_t sz;
|
||||||
jerry_api_value_t val_t, val_foo, val_bar, val_A, val_A_prototype, val_a, val_a_foo;
|
jerry_api_value_t val_t, val_foo, val_bar, val_A, val_A_prototype, val_a, val_a_foo;
|
||||||
jerry_api_object_t* global_obj_p;
|
jerry_api_object_t* global_obj_p;
|
||||||
|
jerry_api_object_t* external_func_p;
|
||||||
jerry_api_value_t res, args [2];
|
jerry_api_value_t res, args [2];
|
||||||
char buffer [16];
|
char buffer [16];
|
||||||
|
|
||||||
@@ -180,6 +192,14 @@ main (void)
|
|||||||
jerry_api_release_value (&res);
|
jerry_api_release_value (&res);
|
||||||
jerry_api_release_value (&val_a_foo);
|
jerry_api_release_value (&val_a_foo);
|
||||||
|
|
||||||
|
external_func_p = jerry_api_create_external_function (handler);
|
||||||
|
assert (external_func_p != NULL);
|
||||||
|
|
||||||
|
is_ok = jerry_api_call_function (external_func_p, global_obj_p, &res, NULL, 0);
|
||||||
|
assert (is_ok);
|
||||||
|
|
||||||
|
jerry_api_release_object (external_func_p);
|
||||||
|
|
||||||
jerry_api_release_value (&val_a);
|
jerry_api_release_value (&val_a);
|
||||||
|
|
||||||
jerry_api_release_object (global_obj_p);
|
jerry_api_release_object (global_obj_p);
|
||||||
|
|||||||
Reference in New Issue
Block a user