Fix null getters/setters in jerry_property_descriptor_from_ecma (#4573)
JerryScript-DCO-1.0-Signed-off-by: Zoltan Herczeg zherczeg.u-szeged@partner.samsung.com
This commit is contained in:
+20
-14
@@ -3244,47 +3244,53 @@ jerry_property_descriptor_from_ecma (const ecma_property_descriptor_t *prop_desc
|
||||
jerry_property_descriptor_t prop_desc;
|
||||
jerry_init_property_descriptor_fields (&prop_desc);
|
||||
|
||||
if (prop_desc_p->flags & (ECMA_PROP_IS_ENUMERABLE_DEFINED))
|
||||
if (prop_desc_p->flags & ECMA_PROP_IS_ENUMERABLE_DEFINED)
|
||||
{
|
||||
prop_desc.is_enumerable_defined = true;
|
||||
prop_desc.is_enumerable = prop_desc_p->flags & (ECMA_PROP_IS_ENUMERABLE);
|
||||
prop_desc.is_enumerable = prop_desc_p->flags & ECMA_PROP_IS_ENUMERABLE;
|
||||
}
|
||||
|
||||
if (prop_desc_p->flags & (ECMA_PROP_IS_CONFIGURABLE_DEFINED))
|
||||
if (prop_desc_p->flags & ECMA_PROP_IS_CONFIGURABLE_DEFINED)
|
||||
{
|
||||
prop_desc.is_configurable_defined = true;
|
||||
prop_desc.is_configurable = prop_desc_p->flags & (ECMA_PROP_IS_CONFIGURABLE);
|
||||
prop_desc.is_configurable = prop_desc_p->flags & ECMA_PROP_IS_CONFIGURABLE;
|
||||
}
|
||||
|
||||
prop_desc.is_value_defined = prop_desc_p->flags & (ECMA_PROP_IS_VALUE_DEFINED);
|
||||
prop_desc.is_value_defined = prop_desc_p->flags & ECMA_PROP_IS_VALUE_DEFINED;
|
||||
|
||||
if (prop_desc.is_value_defined)
|
||||
{
|
||||
prop_desc.value = prop_desc_p->value;
|
||||
}
|
||||
|
||||
if (prop_desc_p->flags & (ECMA_PROP_IS_WRITABLE_DEFINED))
|
||||
if (prop_desc_p->flags & ECMA_PROP_IS_WRITABLE_DEFINED)
|
||||
{
|
||||
prop_desc.is_writable_defined = true;
|
||||
prop_desc.is_writable = prop_desc_p->flags & (ECMA_PROP_IS_WRITABLE);
|
||||
prop_desc.is_writable = prop_desc_p->flags & ECMA_PROP_IS_WRITABLE;
|
||||
}
|
||||
|
||||
if (prop_desc_p->flags & (ECMA_PROP_IS_GET_DEFINED))
|
||||
if (prop_desc_p->flags & ECMA_PROP_IS_GET_DEFINED)
|
||||
{
|
||||
ecma_value_t getter = ecma_make_object_value (prop_desc_p->get_p);
|
||||
prop_desc.getter = ECMA_VALUE_NULL;
|
||||
prop_desc.is_get_defined = true;
|
||||
|
||||
if (ecma_op_is_callable (getter))
|
||||
if (prop_desc_p->get_p != NULL)
|
||||
{
|
||||
prop_desc.getter = getter;
|
||||
prop_desc.getter = ecma_make_object_value (prop_desc_p->get_p);
|
||||
JERRY_ASSERT (ecma_op_is_callable (prop_desc.getter));
|
||||
}
|
||||
}
|
||||
|
||||
if (prop_desc_p->flags & (ECMA_PROP_IS_SET_DEFINED))
|
||||
if (prop_desc_p->flags & ECMA_PROP_IS_SET_DEFINED)
|
||||
{
|
||||
ecma_value_t setter = ecma_make_object_value (prop_desc_p->set_p);
|
||||
prop_desc.setter = ECMA_VALUE_NULL;
|
||||
prop_desc.is_set_defined = true;
|
||||
prop_desc.setter = setter;
|
||||
|
||||
if (prop_desc_p->set_p != NULL)
|
||||
{
|
||||
prop_desc.setter = ecma_make_object_value (prop_desc_p->set_p);
|
||||
JERRY_ASSERT (ecma_op_is_callable (prop_desc.setter));
|
||||
}
|
||||
}
|
||||
|
||||
return prop_desc;
|
||||
|
||||
@@ -17,6 +17,48 @@
|
||||
#include "jerryscript.h"
|
||||
#include "test-common.h"
|
||||
|
||||
static jerry_value_t
|
||||
create_property_descriptor (const char *script_p) /**< source code */
|
||||
{
|
||||
jerry_value_t result = jerry_eval ((const jerry_char_t *) script_p, strlen (script_p), 0);
|
||||
TEST_ASSERT (jerry_value_is_object (result));
|
||||
return result;
|
||||
} /* create_property_descriptor */
|
||||
|
||||
static void
|
||||
check_attribute (jerry_value_t attribute, /**< attribute to be checked */
|
||||
jerry_value_t object, /**< original object */
|
||||
const char *name_p) /**< name of the attribute */
|
||||
{
|
||||
jerry_value_t prop_name = jerry_create_string_from_utf8 ((const jerry_char_t *) name_p);
|
||||
jerry_value_t value = jerry_get_property (object, prop_name);
|
||||
|
||||
if (jerry_value_is_undefined (value))
|
||||
{
|
||||
TEST_ASSERT (jerry_value_is_null (attribute));
|
||||
}
|
||||
else
|
||||
{
|
||||
jerry_value_t result = jerry_binary_operation (JERRY_BIN_OP_STRICT_EQUAL, attribute, value);
|
||||
TEST_ASSERT (jerry_value_is_boolean (result) && jerry_get_boolean_value (result));
|
||||
jerry_release_value (result);
|
||||
}
|
||||
|
||||
jerry_release_value (value);
|
||||
jerry_release_value (prop_name);
|
||||
} /* check_attribute */
|
||||
|
||||
static void
|
||||
to_property_descriptor (jerry_value_t object, /**< object */
|
||||
jerry_property_descriptor_t *prop_desc_p) /**< property descriptor */
|
||||
{
|
||||
jerry_init_property_descriptor_fields (prop_desc_p);
|
||||
|
||||
jerry_value_t result = jerry_to_property_descriptor (object, prop_desc_p);
|
||||
TEST_ASSERT (jerry_value_is_boolean (result) && jerry_get_boolean_value (result));
|
||||
jerry_release_value (result);
|
||||
} /* to_property_descriptor */
|
||||
|
||||
int
|
||||
main (void)
|
||||
{
|
||||
@@ -24,55 +66,100 @@ main (void)
|
||||
|
||||
jerry_init (JERRY_INIT_EMPTY);
|
||||
|
||||
jerry_value_t object = jerry_create_object ();
|
||||
jerry_value_t prop_name;
|
||||
jerry_value_t value = jerry_create_boolean (true);
|
||||
jerry_property_descriptor_t prop_desc;
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "value");
|
||||
TEST_ASSERT (jerry_set_property (object, prop_name, value));
|
||||
/* Next test. */
|
||||
const char *source_p = "({ value:'X', writable:true, enumerable:true, configurable:true })";
|
||||
jerry_value_t object = create_property_descriptor (source_p);
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "writable");
|
||||
TEST_ASSERT (jerry_set_property (object, prop_name, value));
|
||||
to_property_descriptor (object, &prop_desc);
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "writable");
|
||||
TEST_ASSERT (jerry_set_property (object, prop_name, value));
|
||||
check_attribute (prop_desc.value, object, "value");
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "enumerable");
|
||||
TEST_ASSERT (jerry_set_property (object, prop_name, value));
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "configurable");
|
||||
TEST_ASSERT (jerry_set_property (object, prop_name, value));
|
||||
|
||||
jerry_value_t result = jerry_to_property_descriptor (object, &prop_desc);
|
||||
TEST_ASSERT (jerry_value_is_boolean (result) && jerry_get_boolean_value (result));
|
||||
jerry_release_value (result);
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "value");
|
||||
value = jerry_get_property (object, prop_name);
|
||||
TEST_ASSERT (value == prop_desc.value);
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "writable");
|
||||
value = jerry_get_property (object, prop_name);
|
||||
TEST_ASSERT (jerry_get_boolean_value (value) == prop_desc.is_writable);
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "writable");
|
||||
value = jerry_get_property (object, prop_name);
|
||||
TEST_ASSERT (jerry_get_boolean_value (value) == prop_desc.is_writable);
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "enumerable");
|
||||
value = jerry_get_property (object, prop_name);
|
||||
TEST_ASSERT (jerry_get_boolean_value (value) == prop_desc.is_enumerable);
|
||||
|
||||
prop_name = jerry_create_string_from_utf8 ((jerry_char_t *) "configurable");
|
||||
value = jerry_get_property (object, prop_name);
|
||||
TEST_ASSERT (jerry_get_boolean_value (value) == prop_desc.is_configurable);
|
||||
TEST_ASSERT (prop_desc.is_value_defined);
|
||||
TEST_ASSERT (!prop_desc.is_get_defined && !prop_desc.is_set_defined);
|
||||
TEST_ASSERT (prop_desc.is_writable_defined && prop_desc.is_writable);
|
||||
TEST_ASSERT (prop_desc.is_enumerable_defined && prop_desc.is_enumerable);
|
||||
TEST_ASSERT (prop_desc.is_configurable_defined && prop_desc.is_configurable);
|
||||
|
||||
jerry_release_value (object);
|
||||
jerry_release_value (prop_name);
|
||||
jerry_release_value (value);
|
||||
jerry_free_property_descriptor_fields (&prop_desc);
|
||||
|
||||
/* Next test. */
|
||||
source_p = "({ writable:false, configurable:true })";
|
||||
object = create_property_descriptor (source_p);
|
||||
|
||||
to_property_descriptor (object, &prop_desc);
|
||||
|
||||
TEST_ASSERT (!prop_desc.is_value_defined);
|
||||
TEST_ASSERT (!prop_desc.is_get_defined && !prop_desc.is_set_defined);
|
||||
TEST_ASSERT (prop_desc.is_writable_defined && !prop_desc.is_writable);
|
||||
TEST_ASSERT (!prop_desc.is_enumerable_defined);
|
||||
TEST_ASSERT (prop_desc.is_configurable_defined && prop_desc.is_configurable);
|
||||
|
||||
jerry_release_value (object);
|
||||
jerry_free_property_descriptor_fields (&prop_desc);
|
||||
|
||||
/* Next test. */
|
||||
/* Note: the 'set' property is defined, and it has a value of undefined.
|
||||
* This is different from not having a 'set' property. */
|
||||
source_p = "({ get: function() {}, set:undefined, configurable:true })";
|
||||
object = create_property_descriptor (source_p);
|
||||
|
||||
to_property_descriptor (object, &prop_desc);
|
||||
|
||||
check_attribute (prop_desc.getter, object, "get");
|
||||
check_attribute (prop_desc.setter, object, "set");
|
||||
|
||||
TEST_ASSERT (!prop_desc.is_value_defined && !prop_desc.is_writable_defined);
|
||||
TEST_ASSERT (prop_desc.is_get_defined && prop_desc.is_set_defined);
|
||||
TEST_ASSERT (!prop_desc.is_enumerable_defined);
|
||||
TEST_ASSERT (prop_desc.is_configurable_defined && prop_desc.is_configurable);
|
||||
|
||||
jerry_release_value (object);
|
||||
jerry_free_property_descriptor_fields (&prop_desc);
|
||||
|
||||
/* Next test. */
|
||||
source_p = "({ get: undefined, enumerable:false })";
|
||||
object = create_property_descriptor (source_p);
|
||||
|
||||
to_property_descriptor (object, &prop_desc);
|
||||
|
||||
check_attribute (prop_desc.getter, object, "get");
|
||||
|
||||
TEST_ASSERT (!prop_desc.is_value_defined && !prop_desc.is_writable_defined);
|
||||
TEST_ASSERT (prop_desc.is_get_defined && !prop_desc.is_set_defined);
|
||||
TEST_ASSERT (prop_desc.is_enumerable_defined && !prop_desc.is_enumerable);
|
||||
TEST_ASSERT (!prop_desc.is_configurable_defined);
|
||||
|
||||
jerry_release_value (object);
|
||||
jerry_free_property_descriptor_fields (&prop_desc);
|
||||
|
||||
/* Next test. */
|
||||
source_p = "({ set: function(v) {}, enumerable:true, configurable:false })";
|
||||
object = create_property_descriptor (source_p);
|
||||
|
||||
to_property_descriptor (object, &prop_desc);
|
||||
|
||||
check_attribute (prop_desc.setter, object, "set");
|
||||
|
||||
TEST_ASSERT (!prop_desc.is_value_defined && !prop_desc.is_writable_defined);
|
||||
TEST_ASSERT (!prop_desc.is_get_defined && prop_desc.is_set_defined);
|
||||
TEST_ASSERT (prop_desc.is_enumerable_defined && prop_desc.is_enumerable);
|
||||
TEST_ASSERT (prop_desc.is_configurable_defined && !prop_desc.is_configurable);
|
||||
|
||||
jerry_release_value (object);
|
||||
jerry_free_property_descriptor_fields (&prop_desc);
|
||||
|
||||
/* Next test. */
|
||||
source_p = "({ get: function(v) {}, writable:true })";
|
||||
object = create_property_descriptor (source_p);
|
||||
jerry_value_t result = jerry_to_property_descriptor (object, &prop_desc);
|
||||
TEST_ASSERT (jerry_value_is_error (result));
|
||||
jerry_release_value (result);
|
||||
jerry_release_value (object);
|
||||
|
||||
/* Next test. */
|
||||
object = jerry_create_null ();
|
||||
result = jerry_to_property_descriptor (object, &prop_desc);
|
||||
TEST_ASSERT (jerry_value_is_error (result));
|
||||
|
||||
Reference in New Issue
Block a user