1425bb815Sopenharmony_ci/* Copyright JS Foundation and other contributors, http://js.foundation 2425bb815Sopenharmony_ci * 3425bb815Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License"); 4425bb815Sopenharmony_ci * you may not use this file except in compliance with the License. 5425bb815Sopenharmony_ci * You may obtain a copy of the License at 6425bb815Sopenharmony_ci * 7425bb815Sopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0 8425bb815Sopenharmony_ci * 9425bb815Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software 10425bb815Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS 11425bb815Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12425bb815Sopenharmony_ci * See the License for the specific language governing permissions and 13425bb815Sopenharmony_ci * limitations under the License. 14425bb815Sopenharmony_ci */ 15425bb815Sopenharmony_ci 16425bb815Sopenharmony_ci#include "ecma-alloc.h" 17425bb815Sopenharmony_ci#include "ecma-builtins.h" 18425bb815Sopenharmony_ci#include "ecma-exceptions.h" 19425bb815Sopenharmony_ci#include "ecma-gc.h" 20425bb815Sopenharmony_ci#include "ecma-globals.h" 21425bb815Sopenharmony_ci#include "ecma-helpers.h" 22425bb815Sopenharmony_ci#include "ecma-objects.h" 23425bb815Sopenharmony_ci#include "ecma-objects-general.h" 24425bb815Sopenharmony_ci#include "ecma-symbol-object.h" 25425bb815Sopenharmony_ci#include "lit-char-helpers.h" 26425bb815Sopenharmony_ci 27425bb815Sopenharmony_ci#if ENABLED (JERRY_ES2015) 28425bb815Sopenharmony_ci 29425bb815Sopenharmony_ci/** \addtogroup ecma ECMA 30425bb815Sopenharmony_ci * @{ 31425bb815Sopenharmony_ci * 32425bb815Sopenharmony_ci * \addtogroup ecmasymbolobject ECMA Symbol object related routines 33425bb815Sopenharmony_ci * @{ 34425bb815Sopenharmony_ci */ 35425bb815Sopenharmony_ci 36425bb815Sopenharmony_ci/** 37425bb815Sopenharmony_ci * Symbol creation operation. 38425bb815Sopenharmony_ci * 39425bb815Sopenharmony_ci * See also: ECMA-262 v6, 6.1.5.1 40425bb815Sopenharmony_ci * 41425bb815Sopenharmony_ci * @return ecma value 42425bb815Sopenharmony_ci * Returned value must be freed with ecma_free_value 43425bb815Sopenharmony_ci */ 44425bb815Sopenharmony_ciecma_value_t 45425bb815Sopenharmony_ciecma_op_create_symbol (const ecma_value_t *arguments_list_p, /**< list of arguments */ 46425bb815Sopenharmony_ci ecma_length_t arguments_list_len) /**< length of the arguments' list */ 47425bb815Sopenharmony_ci{ 48425bb815Sopenharmony_ci JERRY_ASSERT (arguments_list_len == 0 || arguments_list_p != NULL); 49425bb815Sopenharmony_ci 50425bb815Sopenharmony_ci ecma_value_t string_desc; 51425bb815Sopenharmony_ci 52425bb815Sopenharmony_ci /* 1-3. */ 53425bb815Sopenharmony_ci if (arguments_list_len == 0) 54425bb815Sopenharmony_ci { 55425bb815Sopenharmony_ci string_desc = ecma_make_magic_string_value (LIT_MAGIC_STRING__EMPTY); 56425bb815Sopenharmony_ci } 57425bb815Sopenharmony_ci else 58425bb815Sopenharmony_ci { 59425bb815Sopenharmony_ci ecma_string_t *str_p = ecma_op_to_string (arguments_list_p[0]); 60425bb815Sopenharmony_ci 61425bb815Sopenharmony_ci /* 4. */ 62425bb815Sopenharmony_ci if (JERRY_UNLIKELY (str_p == NULL)) 63425bb815Sopenharmony_ci { 64425bb815Sopenharmony_ci return ECMA_VALUE_ERROR; 65425bb815Sopenharmony_ci } 66425bb815Sopenharmony_ci 67425bb815Sopenharmony_ci string_desc = ecma_make_string_value (str_p); 68425bb815Sopenharmony_ci } 69425bb815Sopenharmony_ci 70425bb815Sopenharmony_ci /* 5. */ 71425bb815Sopenharmony_ci return ecma_make_symbol_value (ecma_new_symbol_from_descriptor_string (string_desc)); 72425bb815Sopenharmony_ci} /* ecma_op_create_symbol */ 73425bb815Sopenharmony_ci 74425bb815Sopenharmony_ci/** 75425bb815Sopenharmony_ci * Symbol object creation operation. 76425bb815Sopenharmony_ci * 77425bb815Sopenharmony_ci * See also: ECMA-262 v6, 19.4.1 78425bb815Sopenharmony_ci * 79425bb815Sopenharmony_ci * @return ecma value 80425bb815Sopenharmony_ci * Returned value must be freed with ecma_free_value 81425bb815Sopenharmony_ci */ 82425bb815Sopenharmony_ciecma_value_t 83425bb815Sopenharmony_ciecma_op_create_symbol_object (const ecma_value_t value) /**< symbol value */ 84425bb815Sopenharmony_ci{ 85425bb815Sopenharmony_ci JERRY_ASSERT (ecma_is_value_symbol (value)); 86425bb815Sopenharmony_ci 87425bb815Sopenharmony_ci ecma_object_t *prototype_obj_p = ecma_builtin_get (ECMA_BUILTIN_ID_SYMBOL_PROTOTYPE); 88425bb815Sopenharmony_ci ecma_object_t *object_p = ecma_create_object (prototype_obj_p, 89425bb815Sopenharmony_ci sizeof (ecma_extended_object_t), 90425bb815Sopenharmony_ci ECMA_OBJECT_TYPE_CLASS); 91425bb815Sopenharmony_ci 92425bb815Sopenharmony_ci ecma_extended_object_t *ext_object_p = (ecma_extended_object_t *) object_p; 93425bb815Sopenharmony_ci ext_object_p->u.class_prop.class_id = LIT_MAGIC_STRING_SYMBOL_UL; 94425bb815Sopenharmony_ci ext_object_p->u.class_prop.u.value = ecma_copy_value (value); 95425bb815Sopenharmony_ci 96425bb815Sopenharmony_ci return ecma_make_object_value (object_p); 97425bb815Sopenharmony_ci} /* ecma_op_create_symbol_object */ 98425bb815Sopenharmony_ci 99425bb815Sopenharmony_ci/** 100425bb815Sopenharmony_ci * Get the symbol descriptor ecma-string from an ecma-symbol 101425bb815Sopenharmony_ci * 102425bb815Sopenharmony_ci * @return pointer to ecma-string descriptor 103425bb815Sopenharmony_ci */ 104425bb815Sopenharmony_ciecma_string_t * 105425bb815Sopenharmony_ciecma_get_symbol_description (ecma_string_t *symbol_p) /**< ecma-symbol */ 106425bb815Sopenharmony_ci{ 107425bb815Sopenharmony_ci JERRY_ASSERT (symbol_p != NULL); 108425bb815Sopenharmony_ci JERRY_ASSERT (ecma_prop_name_is_symbol (symbol_p)); 109425bb815Sopenharmony_ci 110425bb815Sopenharmony_ci return ecma_get_string_from_value (((ecma_extended_string_t *) symbol_p)->u.symbol_descriptor); 111425bb815Sopenharmony_ci} /* ecma_get_symbol_description */ 112425bb815Sopenharmony_ci 113425bb815Sopenharmony_ci/** 114425bb815Sopenharmony_ci * Get the descriptive string of the Symbol. 115425bb815Sopenharmony_ci * 116425bb815Sopenharmony_ci * See also: ECMA-262 v6, 19.4.3.2.1 117425bb815Sopenharmony_ci * 118425bb815Sopenharmony_ci * @return ecma value 119425bb815Sopenharmony_ci * Returned value must be freed with ecma_free_value. 120425bb815Sopenharmony_ci */ 121425bb815Sopenharmony_ciecma_value_t 122425bb815Sopenharmony_ciecma_get_symbol_descriptive_string (ecma_value_t symbol_value) /**< symbol to stringify */ 123425bb815Sopenharmony_ci{ 124425bb815Sopenharmony_ci /* 1. */ 125425bb815Sopenharmony_ci JERRY_ASSERT (ecma_is_value_symbol (symbol_value)); 126425bb815Sopenharmony_ci 127425bb815Sopenharmony_ci /* 2 - 3. */ 128425bb815Sopenharmony_ci ecma_string_t *symbol_p = ecma_get_symbol_from_value (symbol_value); 129425bb815Sopenharmony_ci ecma_string_t *string_desc_p = ecma_get_symbol_description (symbol_p); 130425bb815Sopenharmony_ci 131425bb815Sopenharmony_ci /* 5. */ 132425bb815Sopenharmony_ci ecma_stringbuilder_t builder = ecma_stringbuilder_create_raw ((lit_utf8_byte_t *) ("Symbol("), 7); 133425bb815Sopenharmony_ci ecma_stringbuilder_append (&builder, string_desc_p); 134425bb815Sopenharmony_ci ecma_stringbuilder_append_byte (&builder, LIT_CHAR_RIGHT_PAREN); 135425bb815Sopenharmony_ci 136425bb815Sopenharmony_ci return ecma_make_string_value (ecma_stringbuilder_finalize (&builder)); 137425bb815Sopenharmony_ci} /* ecma_get_symbol_descriptive_string */ 138425bb815Sopenharmony_ci 139425bb815Sopenharmony_ci/** 140425bb815Sopenharmony_ci * Helper for Symbol.prototype.{toString, valueOf} routines 141425bb815Sopenharmony_ci * 142425bb815Sopenharmony_ci * See also: 19.4.3.2, 19.4.3.3 143425bb815Sopenharmony_ci * 144425bb815Sopenharmony_ci * @return ecma value 145425bb815Sopenharmony_ci * Returned value must be freed with ecma_free_value. 146425bb815Sopenharmony_ci */ 147425bb815Sopenharmony_ciecma_value_t 148425bb815Sopenharmony_ciecma_symbol_to_string_helper (ecma_value_t this_arg, /**< this argument value */ 149425bb815Sopenharmony_ci bool is_to_string) /**< true - perform the 'toString' routine steps 150425bb815Sopenharmony_ci * false - perform the 'valueOf' routine steps */ 151425bb815Sopenharmony_ci{ 152425bb815Sopenharmony_ci if (ecma_is_value_symbol (this_arg)) 153425bb815Sopenharmony_ci { 154425bb815Sopenharmony_ci return is_to_string ? ecma_get_symbol_descriptive_string (this_arg) : ecma_copy_value (this_arg); 155425bb815Sopenharmony_ci } 156425bb815Sopenharmony_ci 157425bb815Sopenharmony_ci if (ecma_is_value_object (this_arg)) 158425bb815Sopenharmony_ci { 159425bb815Sopenharmony_ci ecma_object_t *object_p = ecma_get_object_from_value (this_arg); 160425bb815Sopenharmony_ci 161425bb815Sopenharmony_ci if (ecma_get_object_type (object_p) == ECMA_OBJECT_TYPE_CLASS) 162425bb815Sopenharmony_ci { 163425bb815Sopenharmony_ci ecma_extended_object_t *ext_object_p = (ecma_extended_object_t *) object_p; 164425bb815Sopenharmony_ci 165425bb815Sopenharmony_ci if (ext_object_p->u.class_prop.class_id == LIT_MAGIC_STRING_SYMBOL_UL) 166425bb815Sopenharmony_ci { 167425bb815Sopenharmony_ci return (is_to_string ? ecma_get_symbol_descriptive_string (ext_object_p->u.class_prop.u.value) 168425bb815Sopenharmony_ci : ecma_copy_value (ext_object_p->u.class_prop.u.value)); 169425bb815Sopenharmony_ci } 170425bb815Sopenharmony_ci } 171425bb815Sopenharmony_ci } 172425bb815Sopenharmony_ci 173425bb815Sopenharmony_ci return ecma_raise_type_error (ECMA_ERR_MSG ("Argument 'this' is must be a Symbol.")); 174425bb815Sopenharmony_ci} /* ecma_symbol_to_string_helper */ 175425bb815Sopenharmony_ci 176425bb815Sopenharmony_ci#endif /* ENABLED (JERRY_ES2015) */ 177425bb815Sopenharmony_ci 178425bb815Sopenharmony_ci/** 179425bb815Sopenharmony_ci * @} 180425bb815Sopenharmony_ci * @} 181425bb815Sopenharmony_ci */ 182