2 * Copyright (C) 1999-2002 Harri Porten (porten@kde.org)
3 * Copyright (C) 2001 Peter Kelly (pmk@post.com)
4 * Copyright (C) 2004, 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 "JSGlobalObject.h"
28 #include "Operations.h"
29 #include "StringObject.h"
30 #include "StringPrototype.h"
34 static const unsigned resolveRopeForSubstringCutoff
= 4;
36 // Overview: this methods converts a JSString from holding a string in rope form
37 // down to a simple UString representation. It does so by building up the string
38 // backwards, since we want to avoid recursion, we expect that the tree structure
39 // representing the rope is likely imbalanced with more nodes down the left side
40 // (since appending to the string is likely more common) - and as such resolving
41 // in this fashion should minimize work queue size. (If we built the queue forwards
42 // we would likely have to place all of the constituent UStringImpls into the
43 // Vector before performing any concatenation, but by working backwards we likely
44 // only fill the queue with the number of substrings at any given level in a
46 void JSString::resolveRope(ExecState
* exec
) const
50 // Allocate the buffer to hold the final string, position initially points to the end.
52 if (PassRefPtr
<UStringImpl
> newImpl
= UStringImpl::tryCreateUninitialized(m_length
, buffer
))
55 for (unsigned i
= 0; i
< m_fiberCount
; ++i
) {
56 RopeImpl::deref(m_other
.m_fibers
[i
]);
57 m_other
.m_fibers
[i
] = 0;
61 ASSERT(m_value
== UString());
63 throwOutOfMemoryError(exec
);
66 UChar
* position
= buffer
+ m_length
;
68 // Start with the current RopeImpl.
69 Vector
<RopeImpl::Fiber
, 32> workQueue
;
70 RopeImpl::Fiber currentFiber
;
71 for (unsigned i
= 0; i
< (m_fiberCount
- 1); ++i
)
72 workQueue
.append(m_other
.m_fibers
[i
]);
73 currentFiber
= m_other
.m_fibers
[m_fiberCount
- 1];
75 if (RopeImpl::isRope(currentFiber
)) {
76 RopeImpl
* rope
= static_cast<RopeImpl
*>(currentFiber
);
77 // Copy the contents of the current rope into the workQueue, with the last item in 'currentFiber'
78 // (we will be working backwards over the rope).
79 unsigned fiberCountMinusOne
= rope
->fiberCount() - 1;
80 for (unsigned i
= 0; i
< fiberCountMinusOne
; ++i
)
81 workQueue
.append(rope
->fibers()[i
]);
82 currentFiber
= rope
->fibers()[fiberCountMinusOne
];
84 UStringImpl
* string
= static_cast<UStringImpl
*>(currentFiber
);
85 unsigned length
= string
->length();
87 UStringImpl::copyChars(position
, string
->characters(), length
);
89 // Was this the last item in the work queue?
90 if (workQueue
.isEmpty()) {
91 // Create a string from the UChar buffer, clear the rope RefPtr.
92 ASSERT(buffer
== position
);
93 for (unsigned i
= 0; i
< m_fiberCount
; ++i
) {
94 RopeImpl::deref(m_other
.m_fibers
[i
]);
95 m_other
.m_fibers
[i
] = 0;
103 // No! - set the next item up to process.
104 currentFiber
= workQueue
.last();
105 workQueue
.removeLast();
110 // This function construsts a substring out of a rope without flattening by reusing the existing fibers.
111 // This can reduce memory usage substantially. Since traversing ropes is slow the function will revert
112 // back to flattening if the rope turns out to be long.
113 JSString
* JSString::substringFromRope(ExecState
* exec
, unsigned substringStart
, unsigned substringLength
)
117 JSGlobalData
* globalData
= &exec
->globalData();
119 UString substringFibers
[3];
121 unsigned fiberCount
= 0;
122 unsigned substringFiberCount
= 0;
123 unsigned substringEnd
= substringStart
+ substringLength
;
124 unsigned fiberEnd
= 0;
127 for (RopeIterator
it(m_other
.m_fibers
, m_fiberCount
); it
!= end
; ++it
) {
129 UStringImpl
* fiberString
= *it
;
130 unsigned fiberStart
= fiberEnd
;
131 fiberEnd
= fiberStart
+ fiberString
->length();
132 if (fiberEnd
<= substringStart
)
134 unsigned copyStart
= std::max(substringStart
, fiberStart
);
135 unsigned copyEnd
= std::min(substringEnd
, fiberEnd
);
136 if (copyStart
== fiberStart
&& copyEnd
== fiberEnd
)
137 substringFibers
[substringFiberCount
++] = UString(fiberString
);
139 substringFibers
[substringFiberCount
++] = UString(UStringImpl::create(fiberString
, copyStart
- fiberStart
, copyEnd
- copyStart
));
140 if (fiberEnd
>= substringEnd
)
142 if (fiberCount
> resolveRopeForSubstringCutoff
|| substringFiberCount
>= 3) {
143 // This turned out to be a really inefficient rope. Just flatten it.
145 return jsSubstring(&exec
->globalData(), m_value
, substringStart
, substringLength
);
148 ASSERT(substringFiberCount
&& substringFiberCount
<= 3);
150 if (substringLength
== 1) {
151 ASSERT(substringFiberCount
== 1);
152 UChar c
= substringFibers
[0].data()[0];
154 return globalData
->smallStrings
.singleCharacterString(globalData
, c
);
156 if (substringFiberCount
== 1)
157 return new (globalData
) JSString(globalData
, substringFibers
[0]);
158 if (substringFiberCount
== 2)
159 return new (globalData
) JSString(globalData
, substringFibers
[0], substringFibers
[1]);
160 return new (globalData
) JSString(globalData
, substringFibers
[0], substringFibers
[1], substringFibers
[2]);
163 JSValue
JSString::replaceCharacter(ExecState
* exec
, UChar character
, const UString
& replacement
)
166 unsigned matchPosition
= m_value
.find(character
);
167 if (matchPosition
== UString::NotFound
)
168 return JSValue(this);
169 return jsString(exec
, m_value
.substr(0, matchPosition
), replacement
, m_value
.substr(matchPosition
+ 1));
174 // Count total fibers and find matching string.
175 size_t fiberCount
= 0;
176 UStringImpl
* matchString
= 0;
177 int matchPosition
= -1;
178 for (RopeIterator
it(m_other
.m_fibers
, m_fiberCount
); it
!= end
; ++it
) {
183 UStringImpl
* string
= *it
;
184 matchPosition
= string
->find(character
);
185 if (matchPosition
== -1)
187 matchString
= string
;
193 RopeBuilder
builder(replacement
.size() ? fiberCount
+ 2 : fiberCount
+ 1);
194 if (UNLIKELY(builder
.isOutOfMemory()))
195 return throwOutOfMemoryError(exec
);
197 for (RopeIterator
it(m_other
.m_fibers
, m_fiberCount
); it
!= end
; ++it
) {
198 UStringImpl
* string
= *it
;
199 if (string
!= matchString
) {
200 builder
.append(UString(string
));
204 builder
.append(UString(string
).substr(0, matchPosition
));
205 if (replacement
.size())
206 builder
.append(replacement
);
207 builder
.append(UString(string
).substr(matchPosition
+ 1));
211 JSGlobalData
* globalData
= &exec
->globalData();
212 return JSValue(new (globalData
) JSString(globalData
, builder
.release()));
215 JSString
* JSString::getIndexSlowCase(ExecState
* exec
, unsigned i
)
219 // Return a safe no-value result, this should never be used, since the excetion will be thrown.
220 if (exec
->exception())
221 return jsString(exec
, "");
223 ASSERT(i
< m_value
.size());
224 return jsSingleCharacterSubstring(exec
, m_value
, i
);
227 JSValue
JSString::toPrimitive(ExecState
*, PreferredPrimitiveType
) const
229 return const_cast<JSString
*>(this);
232 bool JSString::getPrimitiveNumber(ExecState
* exec
, double& number
, JSValue
& result
)
235 number
= value(exec
).toDouble();
239 bool JSString::toBoolean(ExecState
*) const
244 double JSString::toNumber(ExecState
* exec
) const
246 return value(exec
).toDouble();
249 UString
JSString::toString(ExecState
* exec
) const
254 inline StringObject
* StringObject::create(ExecState
* exec
, JSString
* string
)
256 return new (exec
) StringObject(exec
->lexicalGlobalObject()->stringObjectStructure(), string
);
259 JSObject
* JSString::toObject(ExecState
* exec
) const
261 return StringObject::create(exec
, const_cast<JSString
*>(this));
264 JSObject
* JSString::toThisObject(ExecState
* exec
) const
266 return StringObject::create(exec
, const_cast<JSString
*>(this));
269 bool JSString::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
271 // The semantics here are really getPropertySlot, not getOwnPropertySlot.
272 // This function should only be called by JSValue::get.
273 if (getStringPropertySlot(exec
, propertyName
, slot
))
275 if (propertyName
== exec
->propertyNames().underscoreProto
) {
276 slot
.setValue(exec
->lexicalGlobalObject()->stringPrototype());
281 for (JSValue prototype
= exec
->lexicalGlobalObject()->stringPrototype(); !prototype
.isNull(); prototype
= object
->prototype()) {
282 object
= asObject(prototype
);
283 if (object
->getOwnPropertySlot(exec
, propertyName
, slot
))
290 bool JSString::getStringPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
)
292 if (propertyName
== exec
->propertyNames().length
) {
293 descriptor
.setDescriptor(jsNumber(exec
, m_length
), DontEnum
| DontDelete
| ReadOnly
);
298 unsigned i
= propertyName
.toStrictUInt32(&isStrictUInt32
);
299 if (isStrictUInt32
&& i
< m_length
) {
300 descriptor
.setDescriptor(getIndex(exec
, i
), DontDelete
| ReadOnly
);
307 bool JSString::getOwnPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
)
309 if (getStringPropertyDescriptor(exec
, propertyName
, descriptor
))
311 if (propertyName
!= exec
->propertyNames().underscoreProto
)
313 descriptor
.setDescriptor(exec
->lexicalGlobalObject()->stringPrototype(), DontEnum
);
317 bool JSString::getOwnPropertySlot(ExecState
* exec
, unsigned propertyName
, PropertySlot
& slot
)
319 // The semantics here are really getPropertySlot, not getOwnPropertySlot.
320 // This function should only be called by JSValue::get.
321 if (getStringPropertySlot(exec
, propertyName
, slot
))
323 return JSString::getOwnPropertySlot(exec
, Identifier::from(exec
, propertyName
), slot
);