]>
git.saurik.com Git - apple/javascriptcore.git/blob - kjs/value.cpp
55da40b320c4324b59a3cc53b75645346c15245b
2 * Copyright (C) 1999-2001 Harri Porten (porten@kde.org)
3 * Copyright (C) 2001 Peter Kelly (pmk@post.com)
4 * Copyright (C) 2003, 2007, 2008 Apple Inc. All rights reserved.
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Library General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Library General Public License for more details.
16 * You should have received a copy of the GNU Library General Public License
17 * along with this library; see the file COPYING.LIB. If not, write to
18 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19 * Boston, MA 02110-1301, USA.
26 #include "error_object.h"
30 #include <wtf/MathExtras.h>
34 #if defined NAN && defined INFINITY
36 extern const double NaN
= NAN
;
37 extern const double Inf
= INFINITY
;
39 #else // !(defined NAN && defined INFINITY)
41 // The trick is to define the NaN and Inf globals with a different type than the declaration.
42 // This trick works because the mangled name of the globals does not include the type, although
43 // I'm not sure that's guaranteed. There could be alignment issues with this, since arrays of
44 // characters don't necessarily need the same alignment doubles do, but for now it seems to work.
45 // It would be good to figure out a 100% clean way that still avoids code that runs at init time.
47 // Note, we have to use union to ensure alignment. Otherwise, NaN_Bytes can start anywhere,
48 // while NaN_double has to be 4-byte aligned for 32-bits.
49 // With -fstrict-aliasing enabled, unions are the only safe way to do type masquerading.
53 unsigned char NaN_Bytes
[8];
54 unsigned char Inf_Bytes
[8];
63 #if PLATFORM(BIG_ENDIAN)
64 { 0x7f, 0xf8, 0, 0, 0, 0, 0, 0 },
65 { 0x7f, 0xf0, 0, 0, 0, 0, 0, 0 }
66 #elif PLATFORM(MIDDLE_ENDIAN)
67 { 0, 0, 0xf8, 0x7f, 0, 0, 0, 0 },
68 { 0, 0, 0xf0, 0x7f, 0, 0, 0, 0 }
70 { 0, 0, 0, 0, 0, 0, 0xf8, 0x7f },
71 { 0, 0, 0, 0, 0, 0, 0xf0, 0x7f }
75 extern const double NaN
= NaNInf
.doubles
.NaN_Double
;
76 extern const double Inf
= NaNInf
.doubles
.Inf_Double
;
78 #endif // !(defined NAN && defined INFINITY)
80 static const double D16
= 65536.0;
81 static const double D32
= 4294967296.0;
83 void *JSCell::operator new(size_t size
)
85 return Collector::allocate(size
);
88 bool JSCell::getUInt32(uint32_t&) const
93 bool JSCell::getTruncatedInt32(int32_t&) const
98 bool JSCell::getTruncatedUInt32(uint32_t&) const
104 double JSValue::toInteger(ExecState
*exec
) const
107 if (getTruncatedInt32(i
))
109 double d
= toNumber(exec
);
110 return isnan(d
) ? 0.0 : trunc(d
);
113 double JSValue::toIntegerPreserveNaN(ExecState
*exec
) const
116 if (getTruncatedInt32(i
))
118 return trunc(toNumber(exec
));
121 int32_t JSValue::toInt32SlowCase(double d
, bool& ok
)
125 if (d
>= -D32
/ 2 && d
< D32
/ 2)
126 return static_cast<int32_t>(d
);
128 if (isnan(d
) || isinf(d
)) {
133 double d32
= fmod(trunc(d
), D32
);
136 else if (d32
< -D32
/ 2)
138 return static_cast<int32_t>(d32
);
141 int32_t JSValue::toInt32SlowCase(ExecState
* exec
, bool& ok
) const
143 return JSValue::toInt32SlowCase(toNumber(exec
), ok
);
146 uint32_t JSValue::toUInt32SlowCase(double d
, bool& ok
)
150 if (d
>= 0.0 && d
< D32
)
151 return static_cast<uint32_t>(d
);
153 if (isnan(d
) || isinf(d
)) {
158 double d32
= fmod(trunc(d
), D32
);
161 return static_cast<uint32_t>(d32
);
164 uint32_t JSValue::toUInt32SlowCase(ExecState
* exec
, bool& ok
) const
166 return JSValue::toUInt32SlowCase(toNumber(exec
), ok
);
169 float JSValue::toFloat(ExecState
* exec
) const
171 return static_cast<float>(toNumber(exec
));
174 bool JSCell::getNumber(double &numericValue
) const
178 numericValue
= static_cast<const NumberImp
*>(this)->value();
182 double JSCell::getNumber() const
184 return isNumber() ? static_cast<const NumberImp
*>(this)->value() : NaN
;
187 bool JSCell::getString(UString
&stringValue
) const
191 stringValue
= static_cast<const StringImp
*>(this)->value();
195 UString
JSCell::getString() const
197 return isString() ? static_cast<const StringImp
*>(this)->value() : UString();
200 JSObject
*JSCell::getObject()
202 return isObject() ? static_cast<JSObject
*>(this) : 0;
205 const JSObject
*JSCell::getObject() const
207 return isObject() ? static_cast<const JSObject
*>(this) : 0;
210 JSCell
* jsString(const char* s
)
212 return new StringImp(s
? s
: "");
215 JSCell
* jsString(const UString
& s
)
217 return s
.isNull() ? new StringImp("") : new StringImp(s
);
220 JSCell
* jsOwnedString(const UString
& s
)
222 return s
.isNull() ? new StringImp("", StringImp::HasOtherOwner
) : new StringImp(s
, StringImp::HasOtherOwner
);
225 // This method includes a PIC branch to set up the NumberImp's vtable, so we quarantine
226 // it in a separate function to keep the normal case speedy.
227 JSValue
*jsNumberCell(double d
)
229 return new NumberImp(d
);