Property setters should not be called for bound arguments list entries.
[WebKit-https.git] / Source / JavaScriptCore / runtime / IntlNumberFormatPrototype.cpp
1 /*
2  * Copyright (C) 2015 Andy VanWagoner (thetalecrafter@gmail.com)
3  * Copyright (C) 2016 Apple Inc. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
15  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
16  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
18  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
19  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
20  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
21  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
22  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
23  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
24  * THE POSSIBILITY OF SUCH DAMAGE.
25  */
26
27 #include "config.h"
28 #include "IntlNumberFormatPrototype.h"
29
30 #if ENABLE(INTL)
31
32 #include "BuiltinNames.h"
33 #include "Error.h"
34 #include "IntlNumberFormat.h"
35 #include "JSBoundFunction.h"
36 #include "JSCInlines.h"
37 #include "JSObjectInlines.h"
38
39 namespace JSC {
40
41 static EncodedJSValue JSC_HOST_CALL IntlNumberFormatPrototypeGetterFormat(ExecState*);
42 static EncodedJSValue JSC_HOST_CALL IntlNumberFormatPrototypeFuncResolvedOptions(ExecState*);
43
44 }
45
46 #include "IntlNumberFormatPrototype.lut.h"
47
48 namespace JSC {
49
50 const ClassInfo IntlNumberFormatPrototype::s_info = { "Object", &IntlNumberFormat::s_info, &numberFormatPrototypeTable, CREATE_METHOD_TABLE(IntlNumberFormatPrototype) };
51
52 /* Source for IntlNumberFormatPrototype.lut.h
53 @begin numberFormatPrototypeTable
54   format           IntlNumberFormatPrototypeGetterFormat         DontEnum|Accessor
55   resolvedOptions  IntlNumberFormatPrototypeFuncResolvedOptions  DontEnum|Function 0
56 @end
57 */
58
59 IntlNumberFormatPrototype* IntlNumberFormatPrototype::create(VM& vm, JSGlobalObject*, Structure* structure)
60 {
61     IntlNumberFormatPrototype* object = new (NotNull, allocateCell<IntlNumberFormatPrototype>(vm.heap)) IntlNumberFormatPrototype(vm, structure);
62     object->finishCreation(vm, structure);
63     return object;
64 }
65
66 Structure* IntlNumberFormatPrototype::createStructure(VM& vm, JSGlobalObject* globalObject, JSValue prototype)
67 {
68     return Structure::create(vm, globalObject, prototype, TypeInfo(ObjectType, StructureFlags), info());
69 }
70
71 IntlNumberFormatPrototype::IntlNumberFormatPrototype(VM& vm, Structure* structure)
72     : IntlNumberFormat(vm, structure)
73 {
74 }
75
76 void IntlNumberFormatPrototype::finishCreation(VM& vm, Structure*)
77 {
78     Base::finishCreation(vm);
79 }
80
81 static EncodedJSValue JSC_HOST_CALL IntlNumberFormatFuncFormatNumber(ExecState* state)
82 {
83     VM& vm = state->vm();
84     auto scope = DECLARE_THROW_SCOPE(vm);
85     // 11.3.4 Format Number Functions (ECMA-402 2.0)
86     // 1. Let nf be the this value.
87     // 2. Assert: Type(nf) is Object and nf has an [[initializedNumberFormat]] internal slot whose value  true.
88     IntlNumberFormat* numberFormat = jsCast<IntlNumberFormat*>(state->thisValue());
89
90     // 3. If value is not provided, let value be undefined.
91     // 4. Let x be ToNumber(value).
92     double number = state->argument(0).toNumber(state);
93     // 5. ReturnIfAbrupt(x).
94     RETURN_IF_EXCEPTION(scope, encodedJSValue());
95
96     // 6. Return FormatNumber(nf, x).
97     scope.release();
98     return JSValue::encode(numberFormat->formatNumber(*state, number));
99 }
100
101 EncodedJSValue JSC_HOST_CALL IntlNumberFormatPrototypeGetterFormat(ExecState* state)
102 {
103     VM& vm = state->vm();
104     auto scope = DECLARE_THROW_SCOPE(vm);
105
106     // 11.3.3 Intl.NumberFormat.prototype.format (ECMA-402 2.0)
107     // 1. Let nf be this NumberFormat object.
108     IntlNumberFormat* nf = jsDynamicCast<IntlNumberFormat*>(state->thisValue());
109
110     // FIXME: Workaround to provide compatibility with ECMA-402 1.0 call/apply patterns.
111     // https://bugs.webkit.org/show_bug.cgi?id=153679
112     if (!nf) {
113         JSValue value = state->thisValue().get(state, vm.propertyNames->builtinNames().intlSubstituteValuePrivateName());
114         RETURN_IF_EXCEPTION(scope, encodedJSValue());
115         nf = jsDynamicCast<IntlNumberFormat*>(value);
116     }
117
118     if (!nf)
119         return JSValue::encode(throwTypeError(state, scope, ASCIILiteral("Intl.NumberFormat.prototype.format called on value that's not an object initialized as a NumberFormat")));
120     
121     JSBoundFunction* boundFormat = nf->boundFormat();
122     // 2. If nf.[[boundFormat]] is undefined,
123     if (!boundFormat) {
124         JSGlobalObject* globalObject = nf->globalObject();
125         // a. Let F be a new built-in function object as defined in 11.3.4.
126         // b. The value of F’s length property is 1.
127         JSFunction* targetObject = JSFunction::create(vm, globalObject, 1, ASCIILiteral("format"), IntlNumberFormatFuncFormatNumber, NoIntrinsic);
128         // c. Let bf be BoundFunctionCreate(F, «this value»).
129         boundFormat = JSBoundFunction::create(vm, state, globalObject, targetObject, nf, nullptr, 1, ASCIILiteral("format"));
130         RETURN_IF_EXCEPTION(scope, encodedJSValue());
131         // d. Set nf.[[boundFormat]] to bf.
132         nf->setBoundFormat(vm, boundFormat);
133     }
134     // 3. Return nf.[[boundFormat]].
135     return JSValue::encode(boundFormat);
136 }
137
138 EncodedJSValue JSC_HOST_CALL IntlNumberFormatPrototypeFuncResolvedOptions(ExecState* state)
139 {
140     VM& vm = state->vm();
141     auto scope = DECLARE_THROW_SCOPE(vm);
142
143     // 11.3.5 Intl.NumberFormat.prototype.resolvedOptions() (ECMA-402 2.0)
144     IntlNumberFormat* numberFormat = jsDynamicCast<IntlNumberFormat*>(state->thisValue());
145
146     // FIXME: Workaround to provide compatibility with ECMA-402 1.0 call/apply patterns.
147     // https://bugs.webkit.org/show_bug.cgi?id=153679
148     if (!numberFormat) {
149         JSValue value = state->thisValue().get(state, vm.propertyNames->builtinNames().intlSubstituteValuePrivateName());
150         RETURN_IF_EXCEPTION(scope, encodedJSValue());
151         numberFormat = jsDynamicCast<IntlNumberFormat*>(value);
152     }
153
154     if (!numberFormat)
155         return JSValue::encode(throwTypeError(state, scope, ASCIILiteral("Intl.NumberFormat.prototype.resolvedOptions called on value that's not an object initialized as a NumberFormat")));
156
157     scope.release();
158     return JSValue::encode(numberFormat->resolvedOptions(*state));
159 }
160
161 } // namespace JSC
162
163 #endif // ENABLE(INTL)