• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* Copyright JS Foundation and other contributors, http://js.foundation
2  *
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "ecma-alloc.h"
17 #include "ecma-builtins.h"
18 #include "ecma-exceptions.h"
19 #include "ecma-gc.h"
20 #include "ecma-globals.h"
21 #include "ecma-helpers.h"
22 #include "ecma-objects.h"
23 #include "ecma-objects-general.h"
24 #include "ecma-symbol-object.h"
25 #include "lit-char-helpers.h"
26 
27 #if ENABLED (JERRY_ES2015)
28 
29 /** \addtogroup ecma ECMA
30  * @{
31  *
32  * \addtogroup ecmasymbolobject ECMA Symbol object related routines
33  * @{
34  */
35 
36 /**
37  * Symbol creation operation.
38  *
39  * See also: ECMA-262 v6, 6.1.5.1
40  *
41  * @return ecma value
42  *         Returned value must be freed with ecma_free_value
43  */
44 ecma_value_t
ecma_op_create_symbol(const ecma_value_t * arguments_list_p,ecma_length_t arguments_list_len)45 ecma_op_create_symbol (const ecma_value_t *arguments_list_p, /**< list of arguments */
46                        ecma_length_t arguments_list_len) /**< length of the arguments' list */
47 {
48   JERRY_ASSERT (arguments_list_len == 0 || arguments_list_p != NULL);
49 
50   ecma_value_t string_desc;
51 
52   /* 1-3. */
53   if (arguments_list_len == 0)
54   {
55     string_desc = ecma_make_magic_string_value (LIT_MAGIC_STRING__EMPTY);
56   }
57   else
58   {
59     ecma_string_t *str_p = ecma_op_to_string (arguments_list_p[0]);
60 
61     /* 4. */
62     if (JERRY_UNLIKELY (str_p == NULL))
63     {
64       return ECMA_VALUE_ERROR;
65     }
66 
67     string_desc = ecma_make_string_value (str_p);
68   }
69 
70   /* 5. */
71   return ecma_make_symbol_value (ecma_new_symbol_from_descriptor_string (string_desc));
72 } /* ecma_op_create_symbol */
73 
74 /**
75  * Symbol object creation operation.
76  *
77  * See also: ECMA-262 v6, 19.4.1
78  *
79  * @return ecma value
80  *         Returned value must be freed with ecma_free_value
81  */
82 ecma_value_t
ecma_op_create_symbol_object(const ecma_value_t value)83 ecma_op_create_symbol_object (const ecma_value_t value) /**< symbol value */
84 {
85   JERRY_ASSERT (ecma_is_value_symbol (value));
86 
87   ecma_object_t *prototype_obj_p = ecma_builtin_get (ECMA_BUILTIN_ID_SYMBOL_PROTOTYPE);
88   ecma_object_t *object_p = ecma_create_object (prototype_obj_p,
89                                                 sizeof (ecma_extended_object_t),
90                                                 ECMA_OBJECT_TYPE_CLASS);
91 
92   ecma_extended_object_t *ext_object_p = (ecma_extended_object_t *) object_p;
93   ext_object_p->u.class_prop.class_id = LIT_MAGIC_STRING_SYMBOL_UL;
94   ext_object_p->u.class_prop.u.value = ecma_copy_value (value);
95 
96   return ecma_make_object_value (object_p);
97 } /* ecma_op_create_symbol_object */
98 
99 /**
100  * Get the symbol descriptor ecma-string from an ecma-symbol
101  *
102  * @return pointer to ecma-string descriptor
103  */
104 ecma_string_t *
ecma_get_symbol_description(ecma_string_t * symbol_p)105 ecma_get_symbol_description (ecma_string_t *symbol_p) /**< ecma-symbol */
106 {
107   JERRY_ASSERT (symbol_p != NULL);
108   JERRY_ASSERT (ecma_prop_name_is_symbol (symbol_p));
109 
110   return ecma_get_string_from_value (((ecma_extended_string_t *) symbol_p)->u.symbol_descriptor);
111 } /* ecma_get_symbol_description */
112 
113 /**
114  * Get the descriptive string of the Symbol.
115  *
116  * See also: ECMA-262 v6, 19.4.3.2.1
117  *
118  * @return ecma value
119  *         Returned value must be freed with ecma_free_value.
120  */
121 ecma_value_t
ecma_get_symbol_descriptive_string(ecma_value_t symbol_value)122 ecma_get_symbol_descriptive_string (ecma_value_t symbol_value) /**< symbol to stringify */
123 {
124   /* 1. */
125   JERRY_ASSERT (ecma_is_value_symbol (symbol_value));
126 
127   /* 2 - 3. */
128   ecma_string_t *symbol_p = ecma_get_symbol_from_value (symbol_value);
129   ecma_string_t *string_desc_p = ecma_get_symbol_description (symbol_p);
130 
131   /* 5. */
132   ecma_stringbuilder_t builder = ecma_stringbuilder_create_raw ((lit_utf8_byte_t *) ("Symbol("), 7);
133   ecma_stringbuilder_append (&builder, string_desc_p);
134   ecma_stringbuilder_append_byte (&builder, LIT_CHAR_RIGHT_PAREN);
135 
136   return ecma_make_string_value (ecma_stringbuilder_finalize (&builder));
137 } /* ecma_get_symbol_descriptive_string */
138 
139 /**
140  * Helper for Symbol.prototype.{toString, valueOf} routines
141  *
142  * See also: 19.4.3.2, 19.4.3.3
143  *
144  * @return ecma value
145  *         Returned value must be freed with ecma_free_value.
146  */
147 ecma_value_t
ecma_symbol_to_string_helper(ecma_value_t this_arg,bool is_to_string)148 ecma_symbol_to_string_helper (ecma_value_t this_arg, /**< this argument value */
149                               bool is_to_string) /**< true - perform the 'toString' routine steps
150                                                   *   false - perform the 'valueOf' routine steps */
151 {
152   if (ecma_is_value_symbol (this_arg))
153   {
154     return is_to_string ? ecma_get_symbol_descriptive_string (this_arg) : ecma_copy_value (this_arg);
155   }
156 
157   if (ecma_is_value_object (this_arg))
158   {
159     ecma_object_t *object_p = ecma_get_object_from_value (this_arg);
160 
161     if (ecma_get_object_type (object_p) == ECMA_OBJECT_TYPE_CLASS)
162     {
163       ecma_extended_object_t *ext_object_p = (ecma_extended_object_t *) object_p;
164 
165       if (ext_object_p->u.class_prop.class_id == LIT_MAGIC_STRING_SYMBOL_UL)
166       {
167         return (is_to_string ? ecma_get_symbol_descriptive_string (ext_object_p->u.class_prop.u.value)
168                              : ecma_copy_value (ext_object_p->u.class_prop.u.value));
169       }
170     }
171   }
172 
173   return ecma_raise_type_error (ECMA_ERR_MSG ("Argument 'this' is must be a Symbol."));
174 } /* ecma_symbol_to_string_helper */
175 
176 #endif /* ENABLED (JERRY_ES2015) */
177 
178 /**
179  * @}
180  * @}
181  */
182