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