2 * Copyright (C) 1999-2001 Harri Porten (porten@kde.org)
3 * Copyright (C) 2001 Peter Kelly (pmk@post.com)
4 * Copyright (C) 2003, 2004, 2005, 2006, 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.
25 #include "CallFrame.h"
26 #include "CommonIdentifiers.h"
27 #include "Identifier.h"
28 #include "PropertyDescriptor.h"
29 #include "PropertySlot.h"
30 #include "Structure.h"
36 class LLIntOffsetsExtractor
;
38 JSString
* jsEmptyString(VM
*);
39 JSString
* jsEmptyString(ExecState
*);
40 JSString
* jsString(VM
*, const String
&); // returns empty string if passed null string
41 JSString
* jsString(ExecState
*, const String
&); // returns empty string if passed null string
43 JSString
* jsSingleCharacterString(VM
*, UChar
);
44 JSString
* jsSingleCharacterString(ExecState
*, UChar
);
45 JSString
* jsSingleCharacterSubstring(ExecState
*, const String
&, unsigned offset
);
46 JSString
* jsSubstring(VM
*, const String
&, unsigned offset
, unsigned length
);
47 JSString
* jsSubstring(ExecState
*, const String
&, unsigned offset
, unsigned length
);
49 // Non-trivial strings are two or more characters long.
50 // These functions are faster than just calling jsString.
51 JSString
* jsNontrivialString(VM
*, const String
&);
52 JSString
* jsNontrivialString(ExecState
*, const String
&);
54 // Should be used for strings that are owned by an object that will
55 // likely outlive the JSValue this makes, such as the parse tree or a
56 // DOM object that contains a String
57 JSString
* jsOwnedString(VM
*, const String
&);
58 JSString
* jsOwnedString(ExecState
*, const String
&);
60 JSRopeString
* jsStringBuilder(VM
*);
62 class JSString
: public JSCell
{
66 friend class SpecializedThunkJIT
;
67 friend class JSRopeString
;
68 friend class MarkStack
;
69 friend class SlotVisitor
;
70 friend struct ThunkHelpers
;
74 static const bool needsDestruction
= true;
75 static const bool hasImmortalStructure
= true;
76 static void destroy(JSCell
*);
79 JSString(VM
& vm
, PassRefPtr
<StringImpl
> value
)
80 : JSCell(vm
, vm
.stringStructure
.get())
87 : JSCell(vm
, vm
.stringStructure
.get())
92 void finishCreation(VM
& vm
, size_t length
)
94 ASSERT(!m_value
.isNull());
95 Base::finishCreation(vm
);
97 setIs8Bit(m_value
.impl()->is8Bit());
98 vm
.m_newStringsSinceLastHashCons
++;
101 void finishCreation(VM
& vm
, size_t length
, size_t cost
)
103 ASSERT(!m_value
.isNull());
104 Base::finishCreation(vm
);
106 setIs8Bit(m_value
.impl()->is8Bit());
107 Heap::heap(this)->reportExtraMemoryCost(cost
);
108 vm
.m_newStringsSinceLastHashCons
++;
112 void finishCreation(VM
& vm
)
114 Base::finishCreation(vm
);
117 vm
.m_newStringsSinceLastHashCons
++;
121 static JSString
* create(VM
& vm
, PassRefPtr
<StringImpl
> value
)
124 size_t length
= value
->length();
125 size_t cost
= value
->cost();
126 JSString
* newString
= new (NotNull
, allocateCell
<JSString
>(vm
.heap
)) JSString(vm
, value
);
127 newString
->finishCreation(vm
, length
, cost
);
130 static JSString
* createHasOtherOwner(VM
& vm
, PassRefPtr
<StringImpl
> value
)
133 size_t length
= value
->length();
134 JSString
* newString
= new (NotNull
, allocateCell
<JSString
>(vm
.heap
)) JSString(vm
, value
);
135 newString
->finishCreation(vm
, length
);
139 const String
& value(ExecState
*) const;
140 const String
& tryGetValue() const;
141 unsigned length() { return m_length
; }
143 JSValue
toPrimitive(ExecState
*, PreferredPrimitiveType
) const;
144 JS_EXPORT_PRIVATE
bool toBoolean() const;
145 bool getPrimitiveNumber(ExecState
*, double& number
, JSValue
&) const;
146 JSObject
* toObject(ExecState
*, JSGlobalObject
*) const;
147 double toNumber(ExecState
*) const;
149 bool getStringPropertySlot(ExecState
*, PropertyName
, PropertySlot
&);
150 bool getStringPropertySlot(ExecState
*, unsigned propertyName
, PropertySlot
&);
151 bool getStringPropertyDescriptor(ExecState
*, PropertyName
, PropertyDescriptor
&);
153 bool canGetIndex(unsigned i
) { return i
< m_length
; }
154 JSString
* getIndex(ExecState
*, unsigned);
156 static Structure
* createStructure(VM
& vm
, JSGlobalObject
* globalObject
, JSValue proto
)
158 return Structure::create(vm
, globalObject
, proto
, TypeInfo(StringType
, OverridesGetOwnPropertySlot
| InterceptsGetOwnPropertySlotByIndexEvenWhenLengthIsNotZero
), &s_info
);
161 static size_t offsetOfLength() { return OBJECT_OFFSETOF(JSString
, m_length
); }
162 static size_t offsetOfFlags() { return OBJECT_OFFSETOF(JSString
, m_flags
); }
163 static size_t offsetOfValue() { return OBJECT_OFFSETOF(JSString
, m_value
); }
165 static JS_EXPORTDATA
const ClassInfo s_info
;
167 static void visitChildren(JSCell
*, SlotVisitor
&);
170 HashConsLock
= 1u << 2,
171 IsHashConsSingleton
= 1u << 1,
176 friend class JSValue
;
178 bool isRope() const { return m_value
.isNull(); }
179 bool is8Bit() const { return m_flags
& Is8Bit
; }
180 void setIs8Bit(bool flag
)
187 bool shouldTryHashCons();
188 bool isHashConsSingleton() const { return m_flags
& IsHashConsSingleton
; }
189 void clearHashConsSingleton() { m_flags
&= ~IsHashConsSingleton
; }
190 void setHashConsSingleton() { m_flags
|= IsHashConsSingleton
; }
191 bool tryHashConsLock();
192 void releaseHashConsLock();
196 // A string is represented either by a String or a rope of fibers.
198 mutable String m_value
;
201 friend class LLIntOffsetsExtractor
;
203 static JSObject
* toThisObject(JSCell
*, ExecState
*);
205 // Actually getPropertySlot, not getOwnPropertySlot (see JSCell).
206 static bool getOwnPropertySlot(JSCell
*, ExecState
*, PropertyName
, PropertySlot
&);
207 static bool getOwnPropertySlotByIndex(JSCell
*, ExecState
*, unsigned propertyName
, PropertySlot
&);
209 String
& string() { ASSERT(!isRope()); return m_value
; }
211 friend JSValue
jsString(ExecState
*, JSString
*, JSString
*);
212 friend JSString
* jsSubstring(ExecState
*, JSString
*, unsigned offset
, unsigned length
);
215 class JSRopeString
: public JSString
{
216 friend class JSString
;
218 friend JSRopeString
* jsStringBuilder(VM
*);
224 , m_jsString(jsStringBuilder(&vm
))
229 void append(JSString
* jsString
)
231 if (m_index
== JSRopeString::s_maxInternalRopeLength
)
233 m_jsString
->append(m_vm
, m_index
++, jsString
);
236 JSRopeString
* release()
238 JSRopeString
* tmp
= m_jsString
;
243 unsigned length() { return m_jsString
->m_length
; }
249 JSRopeString
* m_jsString
;
259 void finishCreation(VM
& vm
, JSString
* s1
, JSString
* s2
)
261 Base::finishCreation(vm
);
262 m_length
= s1
->length() + s2
->length();
263 setIs8Bit(s1
->is8Bit() && s2
->is8Bit());
264 m_fibers
[0].set(vm
, this, s1
);
265 m_fibers
[1].set(vm
, this, s2
);
268 void finishCreation(VM
& vm
, JSString
* s1
, JSString
* s2
, JSString
* s3
)
270 Base::finishCreation(vm
);
271 m_length
= s1
->length() + s2
->length() + s3
->length();
272 setIs8Bit(s1
->is8Bit() && s2
->is8Bit() && s3
->is8Bit());
273 m_fibers
[0].set(vm
, this, s1
);
274 m_fibers
[1].set(vm
, this, s2
);
275 m_fibers
[2].set(vm
, this, s3
);
278 void finishCreation(VM
& vm
)
280 JSString::finishCreation(vm
);
283 void append(VM
& vm
, size_t index
, JSString
* jsString
)
285 m_fibers
[index
].set(vm
, this, jsString
);
286 m_length
+= jsString
->m_length
;
287 setIs8Bit(is8Bit() && jsString
->is8Bit());
290 static JSRopeString
* createNull(VM
& vm
)
292 JSRopeString
* newString
= new (NotNull
, allocateCell
<JSRopeString
>(vm
.heap
)) JSRopeString(vm
);
293 newString
->finishCreation(vm
);
298 static JSString
* create(VM
& vm
, JSString
* s1
, JSString
* s2
)
300 JSRopeString
* newString
= new (NotNull
, allocateCell
<JSRopeString
>(vm
.heap
)) JSRopeString(vm
);
301 newString
->finishCreation(vm
, s1
, s2
);
304 static JSString
* create(VM
& vm
, JSString
* s1
, JSString
* s2
, JSString
* s3
)
306 JSRopeString
* newString
= new (NotNull
, allocateCell
<JSRopeString
>(vm
.heap
)) JSRopeString(vm
);
307 newString
->finishCreation(vm
, s1
, s2
, s3
);
311 void visitFibers(SlotVisitor
&);
313 static ptrdiff_t offsetOfFibers() { return OBJECT_OFFSETOF(JSRopeString
, m_fibers
); }
315 static const unsigned s_maxInternalRopeLength
= 3;
318 friend JSValue
jsString(ExecState
*, Register
*, unsigned);
319 friend JSValue
jsStringFromArguments(ExecState
*, JSValue
);
321 JS_EXPORT_PRIVATE
void resolveRope(ExecState
*) const;
322 void resolveRopeSlowCase8(LChar
*) const;
323 void resolveRopeSlowCase(UChar
*) const;
324 void outOfMemory(ExecState
*) const;
326 JSString
* getIndexSlowCase(ExecState
*, unsigned);
328 mutable FixedArray
<WriteBarrier
<JSString
>, s_maxInternalRopeLength
> m_fibers
;
331 JSString
* asString(JSValue
);
333 inline JSString
* asString(JSValue value
)
335 ASSERT(value
.asCell()->isString());
336 return jsCast
<JSString
*>(value
.asCell());
339 inline JSString
* jsEmptyString(VM
* vm
)
341 return vm
->smallStrings
.emptyString();
344 ALWAYS_INLINE JSString
* jsSingleCharacterString(VM
* vm
, UChar c
)
346 if (c
<= maxSingleCharacterString
)
347 return vm
->smallStrings
.singleCharacterString(vm
, c
);
348 return JSString::create(*vm
, String(&c
, 1).impl());
351 ALWAYS_INLINE JSString
* jsSingleCharacterSubstring(ExecState
* exec
, const String
& s
, unsigned offset
)
353 VM
* vm
= &exec
->vm();
354 ASSERT(offset
< static_cast<unsigned>(s
.length()));
355 UChar c
= s
.characterAt(offset
);
356 if (c
<= maxSingleCharacterString
)
357 return vm
->smallStrings
.singleCharacterString(vm
, c
);
358 return JSString::create(*vm
, StringImpl::create(s
.impl(), offset
, 1));
361 inline JSString
* jsNontrivialString(VM
* vm
, const String
& s
)
363 ASSERT(s
.length() > 1);
364 return JSString::create(*vm
, s
.impl());
367 inline const String
& JSString::value(ExecState
* exec
) const
370 static_cast<const JSRopeString
*>(this)->resolveRope(exec
);
374 inline const String
& JSString::tryGetValue() const
377 static_cast<const JSRopeString
*>(this)->resolveRope(0);
381 inline JSString
* JSString::getIndex(ExecState
* exec
, unsigned i
)
383 ASSERT(canGetIndex(i
));
385 return static_cast<JSRopeString
*>(this)->getIndexSlowCase(exec
, i
);
386 ASSERT(i
< m_value
.length());
387 return jsSingleCharacterSubstring(exec
, m_value
, i
);
390 inline JSString
* jsString(VM
* vm
, const String
& s
)
392 int size
= s
.length();
394 return vm
->smallStrings
.emptyString();
396 UChar c
= s
.characterAt(0);
397 if (c
<= maxSingleCharacterString
)
398 return vm
->smallStrings
.singleCharacterString(vm
, c
);
400 return JSString::create(*vm
, s
.impl());
403 inline JSString
* jsSubstring(ExecState
* exec
, JSString
* s
, unsigned offset
, unsigned length
)
405 ASSERT(offset
<= static_cast<unsigned>(s
->length()));
406 ASSERT(length
<= static_cast<unsigned>(s
->length()));
407 ASSERT(offset
+ length
<= static_cast<unsigned>(s
->length()));
408 VM
* vm
= &exec
->vm();
410 return vm
->smallStrings
.emptyString();
411 return jsSubstring(vm
, s
->value(exec
), offset
, length
);
414 inline JSString
* jsSubstring8(VM
* vm
, const String
& s
, unsigned offset
, unsigned length
)
416 ASSERT(offset
<= static_cast<unsigned>(s
.length()));
417 ASSERT(length
<= static_cast<unsigned>(s
.length()));
418 ASSERT(offset
+ length
<= static_cast<unsigned>(s
.length()));
420 return vm
->smallStrings
.emptyString();
422 UChar c
= s
.characterAt(offset
);
423 if (c
<= maxSingleCharacterString
)
424 return vm
->smallStrings
.singleCharacterString(vm
, c
);
426 return JSString::createHasOtherOwner(*vm
, StringImpl::create8(s
.impl(), offset
, length
));
429 inline JSString
* jsSubstring(VM
* vm
, const String
& s
, unsigned offset
, unsigned length
)
431 ASSERT(offset
<= static_cast<unsigned>(s
.length()));
432 ASSERT(length
<= static_cast<unsigned>(s
.length()));
433 ASSERT(offset
+ length
<= static_cast<unsigned>(s
.length()));
435 return vm
->smallStrings
.emptyString();
437 UChar c
= s
.characterAt(offset
);
438 if (c
<= maxSingleCharacterString
)
439 return vm
->smallStrings
.singleCharacterString(vm
, c
);
441 return JSString::createHasOtherOwner(*vm
, StringImpl::create(s
.impl(), offset
, length
));
444 inline JSString
* jsOwnedString(VM
* vm
, const String
& s
)
446 int size
= s
.length();
448 return vm
->smallStrings
.emptyString();
450 UChar c
= s
.characterAt(0);
451 if (c
<= maxSingleCharacterString
)
452 return vm
->smallStrings
.singleCharacterString(vm
, c
);
454 return JSString::createHasOtherOwner(*vm
, s
.impl());
457 inline JSRopeString
* jsStringBuilder(VM
* vm
)
459 return JSRopeString::createNull(*vm
);
462 inline JSString
* jsEmptyString(ExecState
* exec
) { return jsEmptyString(&exec
->vm()); }
463 inline JSString
* jsString(ExecState
* exec
, const String
& s
) { return jsString(&exec
->vm(), s
); }
464 inline JSString
* jsSingleCharacterString(ExecState
* exec
, UChar c
) { return jsSingleCharacterString(&exec
->vm(), c
); }
465 inline JSString
* jsSubstring8(ExecState
* exec
, const String
& s
, unsigned offset
, unsigned length
) { return jsSubstring8(&exec
->vm(), s
, offset
, length
); }
466 inline JSString
* jsSubstring(ExecState
* exec
, const String
& s
, unsigned offset
, unsigned length
) { return jsSubstring(&exec
->vm(), s
, offset
, length
); }
467 inline JSString
* jsNontrivialString(ExecState
* exec
, const String
& s
) { return jsNontrivialString(&exec
->vm(), s
); }
468 inline JSString
* jsOwnedString(ExecState
* exec
, const String
& s
) { return jsOwnedString(&exec
->vm(), s
); }
470 ALWAYS_INLINE
bool JSString::getStringPropertySlot(ExecState
* exec
, PropertyName propertyName
, PropertySlot
& slot
)
472 if (propertyName
== exec
->propertyNames().length
) {
473 slot
.setValue(jsNumber(m_length
));
477 unsigned i
= propertyName
.asIndex();
479 ASSERT(i
!= PropertyName::NotAnIndex
); // No need for an explicit check, the above test would always fail!
480 slot
.setValue(getIndex(exec
, i
));
487 ALWAYS_INLINE
bool JSString::getStringPropertySlot(ExecState
* exec
, unsigned propertyName
, PropertySlot
& slot
)
489 if (propertyName
< m_length
) {
490 slot
.setValue(getIndex(exec
, propertyName
));
497 inline bool isJSString(JSValue v
) { return v
.isCell() && v
.asCell()->classInfo() == &JSString::s_info
; }
499 // --- JSValue inlines ----------------------------
501 inline bool JSValue::toBoolean(ExecState
* exec
) const
506 return asDouble() > 0.0 || asDouble() < 0.0; // false for NaN
508 return asCell()->toBoolean(exec
);
509 return isTrue(); // false, null, and undefined all convert to false.
512 inline JSString
* JSValue::toString(ExecState
* exec
) const
515 return jsCast
<JSString
*>(asCell());
516 return toStringSlowCase(exec
);
519 inline String
JSValue::toWTFString(ExecState
* exec
) const
522 return static_cast<JSString
*>(asCell())->value(exec
);
523 return toWTFStringSlowCase(exec
);
526 ALWAYS_INLINE String
inlineJSValueNotStringtoString(const JSValue
& value
, ExecState
* exec
)
530 return vm
.numericStrings
.add(value
.asInt32());
531 if (value
.isDouble())
532 return vm
.numericStrings
.add(value
.asDouble());
534 return vm
.propertyNames
->trueKeyword
.string();
536 return vm
.propertyNames
->falseKeyword
.string();
538 return vm
.propertyNames
->nullKeyword
.string();
539 if (value
.isUndefined())
540 return vm
.propertyNames
->undefinedKeyword
.string();
541 return value
.toString(exec
)->value(exec
);
544 ALWAYS_INLINE String
JSValue::toWTFStringInline(ExecState
* exec
) const
547 return static_cast<JSString
*>(asCell())->value(exec
);
549 return inlineJSValueNotStringtoString(*this, exec
);