2 * Copyright (C) 1999-2000,2003 Harri Porten (porten@kde.org)
3 * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
23 #include "number_object.h"
24 #include "number_object.lut.h"
27 #include "error_object.h"
28 #include "operations.h"
29 #include <wtf/Assertions.h>
30 #include <wtf/MathExtras.h>
31 #include <wtf/Vector.h>
35 // ------------------------------ NumberInstance ----------------------------
37 const ClassInfo
NumberInstance::info
= { "Number", 0, 0 };
39 NumberInstance::NumberInstance(JSObject
* proto
)
40 : JSWrapperObject(proto
)
44 // ------------------------------ NumberPrototype ---------------------------
46 static JSValue
* numberProtoFuncToString(ExecState
*, JSObject
*, const List
&);
47 static JSValue
* numberProtoFuncToLocaleString(ExecState
*, JSObject
*, const List
&);
48 static JSValue
* numberProtoFuncValueOf(ExecState
*, JSObject
*, const List
&);
49 static JSValue
* numberProtoFuncToFixed(ExecState
*, JSObject
*, const List
&);
50 static JSValue
* numberProtoFuncToExponential(ExecState
*, JSObject
*, const List
&);
51 static JSValue
* numberProtoFuncToPrecision(ExecState
*, JSObject
*, const List
&);
55 NumberPrototype::NumberPrototype(ExecState
* exec
, ObjectPrototype
* objectPrototype
, FunctionPrototype
* functionPrototype
)
56 : NumberInstance(objectPrototype
)
58 setInternalValue(jsNumber(0));
60 // The constructor will be added later, after NumberObjectImp has been constructed
62 putDirectFunction(new PrototypeFunction(exec
, functionPrototype
, 1, exec
->propertyNames().toString
, numberProtoFuncToString
), DontEnum
);
63 putDirectFunction(new PrototypeFunction(exec
, functionPrototype
, 0, exec
->propertyNames().toLocaleString
, numberProtoFuncToLocaleString
), DontEnum
);
64 putDirectFunction(new PrototypeFunction(exec
, functionPrototype
, 0, exec
->propertyNames().valueOf
, numberProtoFuncValueOf
), DontEnum
);
65 putDirectFunction(new PrototypeFunction(exec
, functionPrototype
, 1, exec
->propertyNames().toFixed
, numberProtoFuncToFixed
), DontEnum
);
66 putDirectFunction(new PrototypeFunction(exec
, functionPrototype
, 1, exec
->propertyNames().toExponential
, numberProtoFuncToExponential
), DontEnum
);
67 putDirectFunction(new PrototypeFunction(exec
, functionPrototype
, 1, exec
->propertyNames().toPrecision
, numberProtoFuncToPrecision
), DontEnum
);
70 // ------------------------------ Functions ---------------------------
72 // ECMA 15.7.4.2 - 15.7.4.7
74 static UString
integer_part_noexp(double d
)
78 char* result
= kjs_dtoa(d
, 0, 0, &decimalPoint
, &sign
, NULL
);
79 bool resultIsInfOrNan
= (decimalPoint
== 9999);
80 size_t length
= strlen(result
);
82 UString str
= sign
? "-" : "";
85 else if (decimalPoint
<= 0)
88 Vector
<char, 1024> buf(decimalPoint
+ 1);
90 // FIXME: Remove use of strcpy() and strncpy()
91 if (static_cast<int>(length
) <= decimalPoint
) {
92 strcpy(buf
.data(), result
);
93 memset(buf
.data() + length
, '0', decimalPoint
- length
);
95 strncpy(buf
.data(), result
, decimalPoint
);
97 buf
[decimalPoint
] = '\0';
98 str
+= UString(buf
.data());
101 kjs_freedtoa(result
);
106 static UString
char_sequence(char c
, int count
)
108 Vector
<char, 2048> buf(count
+ 1, c
);
111 return UString(buf
.data());
114 static double intPow10(int e
)
116 // This function uses the "exponentiation by squaring" algorithm and
117 // long double to quickly and precisely calculate integer powers of 10.0.
119 // This is a handy workaround for <rdar://problem/4494756>
124 bool negative
= e
< 0;
125 unsigned exp
= negative
? -e
: e
;
127 long double result
= 10.0;
128 bool foundOne
= false;
129 for (int bit
= 31; bit
>= 0; bit
--) {
131 if ((exp
>> bit
) & 1)
134 result
= result
* result
;
135 if ((exp
>> bit
) & 1)
136 result
= result
* 10.0;
141 return static_cast<double>(1.0 / result
);
142 return static_cast<double>(result
);
146 JSValue
* numberProtoFuncToString(ExecState
* exec
, JSObject
* thisObj
, const List
& args
)
148 if (!thisObj
->inherits(&NumberInstance::info
))
149 return throwError(exec
, TypeError
);
151 JSValue
* v
= static_cast<NumberInstance
*>(thisObj
)->internalValue();
153 double radixAsDouble
= args
[0]->toInteger(exec
); // nan -> 0
154 if (radixAsDouble
== 10 || args
[0]->isUndefined())
155 return jsString(v
->toString(exec
));
157 if (radixAsDouble
< 2 || radixAsDouble
> 36)
158 return throwError(exec
, RangeError
, "toString() radix argument must be between 2 and 36");
160 int radix
= static_cast<int>(radixAsDouble
);
161 const char digits
[] = "0123456789abcdefghijklmnopqrstuvwxyz";
162 // INT_MAX results in 1024 characters left of the dot with radix 2
163 // give the same space on the right side. safety checks are in place
164 // unless someone finds a precise rule.
166 const char* lastCharInString
= s
+ sizeof(s
) - 1;
167 double x
= v
->toNumber(exec
);
168 if (isnan(x
) || isinf(x
))
169 return jsString(UString::from(x
));
171 bool isNegative
= x
< 0.0;
175 double integerPart
= floor(x
);
176 char* decimalPoint
= s
+ sizeof(s
) / 2;
178 // convert integer portion
179 char* p
= decimalPoint
;
180 double d
= integerPart
;
182 int remainderDigit
= static_cast<int>(fmod(d
, radix
));
183 *--p
= digits
[remainderDigit
];
185 } while ((d
<= -1.0 || d
>= 1.0) && s
< p
);
189 char* startOfResultString
= p
;
190 ASSERT(s
<= startOfResultString
);
194 const double epsilon
= 0.001; // TODO: guessed. base on radix ?
195 bool hasFractionalPart
= (d
< -epsilon
|| d
> epsilon
);
196 if (hasFractionalPart
) {
200 const int digit
= static_cast<int>(d
);
201 *p
++ = digits
[digit
];
203 } while ((d
< -epsilon
|| d
> epsilon
) && p
< lastCharInString
);
206 ASSERT(p
< s
+ sizeof(s
));
208 return jsString(startOfResultString
);
211 JSValue
* numberProtoFuncToLocaleString(ExecState
* exec
, JSObject
* thisObj
, const List
&)
213 if (!thisObj
->inherits(&NumberInstance::info
))
214 return throwError(exec
, TypeError
);
217 return jsString(static_cast<NumberInstance
*>(thisObj
)->internalValue()->toString(exec
));
220 JSValue
* numberProtoFuncValueOf(ExecState
* exec
, JSObject
* thisObj
, const List
&)
222 if (!thisObj
->inherits(&NumberInstance::info
))
223 return throwError(exec
, TypeError
);
225 return static_cast<NumberInstance
*>(thisObj
)->internalValue()->toJSNumber(exec
);
228 JSValue
* numberProtoFuncToFixed(ExecState
* exec
, JSObject
* thisObj
, const List
& args
)
230 if (!thisObj
->inherits(&NumberInstance::info
))
231 return throwError(exec
, TypeError
);
233 JSValue
* v
= static_cast<NumberInstance
*>(thisObj
)->internalValue();
235 JSValue
* fractionDigits
= args
[0];
236 double df
= fractionDigits
->toInteger(exec
);
237 if (!(df
>= 0 && df
<= 20))
238 return throwError(exec
, RangeError
, "toFixed() digits argument must be between 0 and 20");
241 double x
= v
->toNumber(exec
);
243 return jsString("NaN");
249 } else if (x
== -0.0)
252 if (x
>= pow(10.0, 21.0))
253 return jsString(s
+ UString::from(x
));
255 const double tenToTheF
= pow(10.0, f
);
256 double n
= floor(x
* tenToTheF
);
257 if (fabs(n
/ tenToTheF
- x
) >= fabs((n
+ 1) / tenToTheF
- x
))
260 UString m
= integer_part_noexp(n
);
265 for (int i
= 0; i
< f
+ 1 - k
; i
++)
269 ASSERT(k
== m
.size());
272 if (kMinusf
< m
.size())
273 return jsString(s
+ m
.substr(0, kMinusf
) + "." + m
.substr(kMinusf
));
274 return jsString(s
+ m
.substr(0, kMinusf
));
277 static void fractionalPartToString(char* buf
, int& i
, const char* result
, int resultLength
, int fractionalDigits
)
279 if (fractionalDigits
<= 0)
282 int fDigitsInResult
= static_cast<int>(resultLength
) - 1;
284 if (fDigitsInResult
> 0) {
285 if (fractionalDigits
< fDigitsInResult
) {
286 strncpy(buf
+ i
, result
+ 1, fractionalDigits
);
287 i
+= fractionalDigits
;
289 // FIXME: Remove use of strcpy()
290 strcpy(buf
+ i
, result
+ 1);
291 i
+= static_cast<int>(resultLength
) - 1;
295 for (int j
= 0; j
< fractionalDigits
- fDigitsInResult
; j
++)
299 static void exponentialPartToString(char* buf
, int& i
, int decimalPoint
)
302 buf
[i
++] = (decimalPoint
>= 0) ? '+' : '-';
303 // decimalPoint can't be more than 3 digits decimal given the
304 // nature of float representation
305 int exponential
= decimalPoint
- 1;
308 if (exponential
>= 100)
309 buf
[i
++] = static_cast<char>('0' + exponential
/ 100);
310 if (exponential
>= 10)
311 buf
[i
++] = static_cast<char>('0' + (exponential
% 100) / 10);
312 buf
[i
++] = static_cast<char>('0' + exponential
% 10);
315 JSValue
* numberProtoFuncToExponential(ExecState
* exec
, JSObject
* thisObj
, const List
& args
)
317 if (!thisObj
->inherits(&NumberInstance::info
))
318 return throwError(exec
, TypeError
);
320 JSValue
* v
= static_cast<NumberInstance
*>(thisObj
)->internalValue();
322 double x
= v
->toNumber(exec
);
324 if (isnan(x
) || isinf(x
))
325 return jsString(UString::from(x
));
327 JSValue
* fractionalDigitsValue
= args
[0];
328 double df
= fractionalDigitsValue
->toInteger(exec
);
329 if (!(df
>= 0 && df
<= 20))
330 return throwError(exec
, RangeError
, "toExponential() argument must between 0 and 20");
331 int fractionalDigits
= (int)df
;
332 bool includeAllDigits
= fractionalDigitsValue
->isUndefined();
334 int decimalAdjust
= 0;
335 if (x
&& !includeAllDigits
) {
336 double logx
= floor(log10(fabs(x
)));
337 x
/= pow(10.0, logx
);
338 const double tenToTheF
= pow(10.0, fractionalDigits
);
339 double fx
= floor(x
* tenToTheF
) / tenToTheF
;
340 double cx
= ceil(x
* tenToTheF
) / tenToTheF
;
342 if (fabs(fx
- x
) < fabs(cx
- x
))
347 decimalAdjust
= static_cast<int>(logx
);
351 return jsString("NaN");
353 if (x
== -0.0) // (-0.0).toExponential() should print as 0 instead of -0
358 char* result
= kjs_dtoa(x
, 0, 0, &decimalPoint
, &sign
, NULL
);
359 size_t resultLength
= strlen(result
);
360 decimalPoint
+= decimalAdjust
;
363 char buf
[80]; // digit + '.' + fractionDigits (max 20) + 'e' + sign + exponent (max?)
367 if (decimalPoint
== 999) // ? 9999 is the magical "result is Inf or NaN" value. what's 999??
368 // FIXME: Remove magic number 80
369 strlcpy(buf
+ i
, result
, 80 - i
);
371 buf
[i
++] = result
[0];
373 if (includeAllDigits
)
374 fractionalDigits
= static_cast<int>(resultLength
) - 1;
376 fractionalPartToString(buf
, i
, result
, resultLength
, fractionalDigits
);
377 exponentialPartToString(buf
, i
, decimalPoint
);
382 kjs_freedtoa(result
);
384 return jsString(buf
);
387 JSValue
* numberProtoFuncToPrecision(ExecState
* exec
, JSObject
* thisObj
, const List
& args
)
389 if (!thisObj
->inherits(&NumberInstance::info
))
390 return throwError(exec
, TypeError
);
392 JSValue
* v
= static_cast<NumberInstance
*>(thisObj
)->internalValue();
394 double doublePrecision
= args
[0]->toIntegerPreserveNaN(exec
);
395 double x
= v
->toNumber(exec
);
396 if (args
[0]->isUndefined() || isnan(x
) || isinf(x
))
397 return jsString(v
->toString(exec
));
405 if (!(doublePrecision
>= 1 && doublePrecision
<= 21)) // true for NaN
406 return throwError(exec
, RangeError
, "toPrecision() argument must be between 1 and 21");
407 int precision
= (int)doublePrecision
;
412 e
= static_cast<int>(log10(x
));
413 double tens
= intPow10(e
- precision
+ 1);
414 double n
= floor(x
/ tens
);
415 if (n
< intPow10(precision
- 1)) {
417 tens
= intPow10(e
- precision
+ 1);
421 if (fabs((n
+ 1.0) * tens
- x
) <= fabs(n
* tens
- x
))
423 // maintain n < 10^(precision)
424 if (n
>= intPow10(precision
)) {
428 ASSERT(intPow10(precision
- 1) <= n
);
429 ASSERT(n
< intPow10(precision
));
431 m
= integer_part_noexp(n
);
432 if (e
< -6 || e
>= precision
) {
434 m
= m
.substr(0, 1) + "." + m
.substr(1);
436 return jsString(s
+ m
+ "e+" + UString::from(e
));
437 return jsString(s
+ m
+ "e-" + UString::from(-e
));
440 m
= char_sequence('0', precision
);
444 if (e
== precision
- 1)
445 return jsString(s
+ m
);
447 if (e
+ 1 < m
.size())
448 return jsString(s
+ m
.substr(0, e
+ 1) + "." + m
.substr(e
+ 1));
449 return jsString(s
+ m
);
451 return jsString(s
+ "0." + char_sequence('0', -(e
+ 1)) + m
);
454 // ------------------------------ NumberObjectImp ------------------------------
456 const ClassInfo
NumberObjectImp::info
= { "Function", &InternalFunctionImp::info
, &numberTable
};
458 /* Source for number_object.lut.h
460 NaN NumberObjectImp::NaNValue DontEnum|DontDelete|ReadOnly
461 NEGATIVE_INFINITY NumberObjectImp::NegInfinity DontEnum|DontDelete|ReadOnly
462 POSITIVE_INFINITY NumberObjectImp::PosInfinity DontEnum|DontDelete|ReadOnly
463 MAX_VALUE NumberObjectImp::MaxValue DontEnum|DontDelete|ReadOnly
464 MIN_VALUE NumberObjectImp::MinValue DontEnum|DontDelete|ReadOnly
467 NumberObjectImp::NumberObjectImp(ExecState
* exec
, FunctionPrototype
* funcProto
, NumberPrototype
* numberProto
)
468 : InternalFunctionImp(funcProto
, numberProto
->classInfo()->className
)
471 putDirect(exec
->propertyNames().prototype
, numberProto
, DontEnum
|DontDelete
|ReadOnly
);
473 // no. of arguments for constructor
474 putDirect(exec
->propertyNames().length
, jsNumber(1), ReadOnly
|DontDelete
|DontEnum
);
477 bool NumberObjectImp::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
479 return getStaticValueSlot
<NumberObjectImp
, InternalFunctionImp
>(exec
, &numberTable
, this, propertyName
, slot
);
482 JSValue
* NumberObjectImp::getValueProperty(ExecState
*, int token
) const
489 return jsNumberCell(-Inf
);
491 return jsNumberCell(Inf
);
493 return jsNumberCell(1.7976931348623157E+308);
495 return jsNumberCell(5E-324);
497 ASSERT_NOT_REACHED();
501 bool NumberObjectImp::implementsConstruct() const
507 JSObject
* NumberObjectImp::construct(ExecState
* exec
, const List
& args
)
509 JSObject
* proto
= exec
->lexicalGlobalObject()->numberPrototype();
510 NumberInstance
* obj
= new NumberInstance(proto
);
512 // FIXME: Check args[0]->isUndefined() instead of args.isEmpty()?
513 double n
= args
.isEmpty() ? 0 : args
[0]->toNumber(exec
);
514 obj
->setInternalValue(jsNumber(n
));
519 JSValue
* NumberObjectImp::callAsFunction(ExecState
* exec
, JSObject
*, const List
& args
)
521 // FIXME: Check args[0]->isUndefined() instead of args.isEmpty()?
522 return jsNumber(args
.isEmpty() ? 0 : args
[0]->toNumber(exec
));