2 * Copyright (C) 2009 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "JSONObject.h"
29 #include "BooleanObject.h"
31 #include "ExceptionHelpers.h"
33 #include "JSGlobalObject.h"
34 #include "LiteralParser.h"
36 #include "LocalScope.h"
38 #include "PropertyNameArray.h"
39 #include "UStringBuilder.h"
40 #include "UStringConcatenate.h"
41 #include <wtf/MathExtras.h>
45 ASSERT_CLASS_FITS_IN_CELL(JSONObject
);
47 static EncodedJSValue JSC_HOST_CALL
JSONProtoFuncParse(ExecState
*);
48 static EncodedJSValue JSC_HOST_CALL
JSONProtoFuncStringify(ExecState
*);
52 #include "JSONObject.lut.h"
56 JSONObject::JSONObject(JSGlobalObject
* globalObject
, Structure
* structure
)
57 : JSObjectWithGlobalObject(globalObject
, structure
)
59 ASSERT(inherits(&s_info
));
62 // PropertyNameForFunctionCall objects must be on the stack, since the JSValue that they create is not marked.
63 class PropertyNameForFunctionCall
{
65 PropertyNameForFunctionCall(const Identifier
&);
66 PropertyNameForFunctionCall(unsigned);
68 JSValue
value(ExecState
*) const;
71 const Identifier
* m_identifier
;
73 mutable JSValue m_value
;
77 WTF_MAKE_NONCOPYABLE(Stringifier
);
79 Stringifier(ExecState
*, const Local
<Unknown
>& replacer
, const Local
<Unknown
>& space
);
80 Local
<Unknown
> stringify(Handle
<Unknown
>);
82 void visitAggregate(SlotVisitor
&);
87 Holder(JSGlobalData
&, JSObject
*);
89 JSObject
* object() const { return m_object
.get(); }
91 bool appendNextProperty(Stringifier
&, UStringBuilder
&);
94 Local
<JSObject
> m_object
;
99 RefPtr
<PropertyNameArrayData
> m_propertyNames
;
104 static void appendQuotedString(UStringBuilder
&, const UString
&);
106 JSValue
toJSON(JSValue
, const PropertyNameForFunctionCall
&);
108 enum StringifyResult
{ StringifyFailed
, StringifySucceeded
, StringifyFailedDueToUndefinedValue
};
109 StringifyResult
appendStringifiedValue(UStringBuilder
&, JSValue
, JSObject
* holder
, const PropertyNameForFunctionCall
&);
111 bool willIndent() const;
114 void startNewLine(UStringBuilder
&) const;
116 ExecState
* const m_exec
;
117 const Local
<Unknown
> m_replacer
;
118 bool m_usingArrayReplacer
;
119 PropertyNameArray m_arrayReplacerPropertyNames
;
120 CallType m_replacerCallType
;
121 CallData m_replacerCallData
;
124 Vector
<Holder
, 16> m_holderStack
;
125 UString m_repeatedGap
;
129 // ------------------------------ helper functions --------------------------------
131 static inline JSValue
unwrapBoxedPrimitive(ExecState
* exec
, JSValue value
)
133 if (!value
.isObject())
135 JSObject
* object
= asObject(value
);
136 if (object
->inherits(&NumberObject::s_info
))
137 return jsNumber(object
->toNumber(exec
));
138 if (object
->inherits(&StringObject::s_info
))
139 return jsString(exec
, object
->toString(exec
));
140 if (object
->inherits(&BooleanObject::s_info
))
141 return object
->toPrimitive(exec
);
145 static inline UString
gap(ExecState
* exec
, JSValue space
)
147 const unsigned maxGapLength
= 10;
148 space
= unwrapBoxedPrimitive(exec
, space
);
150 // If the space value is a number, create a gap string with that number of spaces.
152 if (space
.getNumber(spaceCount
)) {
154 if (spaceCount
> maxGapLength
)
155 count
= maxGapLength
;
156 else if (!(spaceCount
> 0))
159 count
= static_cast<int>(spaceCount
);
160 UChar spaces
[maxGapLength
];
161 for (int i
= 0; i
< count
; ++i
)
163 return UString(spaces
, count
);
166 // If the space value is a string, use it as the gap string, otherwise use no gap string.
167 UString spaces
= space
.getString(exec
);
168 if (spaces
.length() > maxGapLength
) {
169 spaces
= spaces
.substringSharingImpl(0, maxGapLength
);
174 // ------------------------------ PropertyNameForFunctionCall --------------------------------
176 inline PropertyNameForFunctionCall::PropertyNameForFunctionCall(const Identifier
& identifier
)
177 : m_identifier(&identifier
)
181 inline PropertyNameForFunctionCall::PropertyNameForFunctionCall(unsigned number
)
187 JSValue
PropertyNameForFunctionCall::value(ExecState
* exec
) const
191 m_value
= jsString(exec
, m_identifier
->ustring());
193 m_value
= jsNumber(m_number
);
198 // ------------------------------ Stringifier --------------------------------
200 Stringifier::Stringifier(ExecState
* exec
, const Local
<Unknown
>& replacer
, const Local
<Unknown
>& space
)
202 , m_replacer(replacer
)
203 , m_usingArrayReplacer(false)
204 , m_arrayReplacerPropertyNames(exec
)
205 , m_replacerCallType(CallTypeNone
)
206 , m_gap(gap(exec
, space
.get()))
208 if (!m_replacer
.isObject())
211 if (m_replacer
.asObject()->inherits(&JSArray::s_info
)) {
212 m_usingArrayReplacer
= true;
213 Handle
<JSObject
> array
= m_replacer
.asObject();
214 unsigned length
= array
->get(exec
, exec
->globalData().propertyNames
->length
).toUInt32(exec
);
215 for (unsigned i
= 0; i
< length
; ++i
) {
216 JSValue name
= array
->get(exec
, i
);
217 if (exec
->hadException())
220 UString propertyName
;
221 if (name
.getString(exec
, propertyName
)) {
222 m_arrayReplacerPropertyNames
.add(Identifier(exec
, propertyName
));
227 if (name
.getNumber(value
)) {
228 m_arrayReplacerPropertyNames
.add(Identifier::from(exec
, value
));
232 if (name
.isObject()) {
233 if (!asObject(name
)->inherits(&NumberObject::s_info
) && !asObject(name
)->inherits(&StringObject::s_info
))
235 propertyName
= name
.toString(exec
);
236 if (exec
->hadException())
238 m_arrayReplacerPropertyNames
.add(Identifier(exec
, propertyName
));
244 m_replacerCallType
= m_replacer
.asObject()->getCallData(m_replacerCallData
);
247 Local
<Unknown
> Stringifier::stringify(Handle
<Unknown
> value
)
249 JSObject
* object
= constructEmptyObject(m_exec
);
250 if (m_exec
->hadException())
251 return Local
<Unknown
>(m_exec
->globalData(), jsNull());
253 PropertyNameForFunctionCall
emptyPropertyName(m_exec
->globalData().propertyNames
->emptyIdentifier
);
254 object
->putDirect(m_exec
->globalData(), m_exec
->globalData().propertyNames
->emptyIdentifier
, value
.get());
256 UStringBuilder result
;
257 if (appendStringifiedValue(result
, value
.get(), object
, emptyPropertyName
) != StringifySucceeded
)
258 return Local
<Unknown
>(m_exec
->globalData(), jsUndefined());
259 if (m_exec
->hadException())
260 return Local
<Unknown
>(m_exec
->globalData(), jsNull());
262 return Local
<Unknown
>(m_exec
->globalData(), jsString(m_exec
, result
.toUString()));
265 void Stringifier::appendQuotedString(UStringBuilder
& builder
, const UString
& value
)
267 int length
= value
.length();
271 const UChar
* data
= value
.characters();
272 for (int i
= 0; i
< length
; ++i
) {
274 while (i
< length
&& (data
[i
] > 0x1F && data
[i
] != '"' && data
[i
] != '\\'))
276 builder
.append(data
+ start
, i
- start
);
281 builder
.append('\\');
285 builder
.append('\\');
289 builder
.append('\\');
293 builder
.append('\\');
297 builder
.append('\\');
301 builder
.append('\\');
305 builder
.append('\\');
306 builder
.append('\\');
309 static const char hexDigits
[] = "0123456789abcdef";
311 UChar hex
[] = { '\\', 'u', hexDigits
[(ch
>> 12) & 0xF], hexDigits
[(ch
>> 8) & 0xF], hexDigits
[(ch
>> 4) & 0xF], hexDigits
[ch
& 0xF] };
312 builder
.append(hex
, WTF_ARRAY_LENGTH(hex
));
320 inline JSValue
Stringifier::toJSON(JSValue value
, const PropertyNameForFunctionCall
& propertyName
)
322 ASSERT(!m_exec
->hadException());
323 if (!value
.isObject() || !asObject(value
)->hasProperty(m_exec
, m_exec
->globalData().propertyNames
->toJSON
))
326 JSValue toJSONFunction
= asObject(value
)->get(m_exec
, m_exec
->globalData().propertyNames
->toJSON
);
327 if (m_exec
->hadException())
330 if (!toJSONFunction
.isObject())
333 JSObject
* object
= asObject(toJSONFunction
);
335 CallType callType
= object
->getCallData(callData
);
336 if (callType
== CallTypeNone
)
339 JSValue list
[] = { propertyName
.value(m_exec
) };
340 ArgList
args(list
, WTF_ARRAY_LENGTH(list
));
341 return call(m_exec
, object
, callType
, callData
, value
, args
);
344 Stringifier::StringifyResult
Stringifier::appendStringifiedValue(UStringBuilder
& builder
, JSValue value
, JSObject
* holder
, const PropertyNameForFunctionCall
& propertyName
)
346 // Call the toJSON function.
347 value
= toJSON(value
, propertyName
);
348 if (m_exec
->hadException())
349 return StringifyFailed
;
351 // Call the replacer function.
352 if (m_replacerCallType
!= CallTypeNone
) {
353 JSValue list
[] = { propertyName
.value(m_exec
), value
};
354 ArgList
args(list
, WTF_ARRAY_LENGTH(list
));
355 value
= call(m_exec
, m_replacer
.get(), m_replacerCallType
, m_replacerCallData
, holder
, args
);
356 if (m_exec
->hadException())
357 return StringifyFailed
;
360 if (value
.isUndefined() && !holder
->inherits(&JSArray::s_info
))
361 return StringifyFailedDueToUndefinedValue
;
363 if (value
.isNull()) {
364 builder
.append("null");
365 return StringifySucceeded
;
368 value
= unwrapBoxedPrimitive(m_exec
, value
);
370 if (m_exec
->hadException())
371 return StringifyFailed
;
373 if (value
.isBoolean()) {
374 builder
.append(value
.getBoolean() ? "true" : "false");
375 return StringifySucceeded
;
379 if (value
.getString(m_exec
, stringValue
)) {
380 appendQuotedString(builder
, stringValue
);
381 return StringifySucceeded
;
385 if (value
.getNumber(numericValue
)) {
386 if (!isfinite(numericValue
))
387 builder
.append("null");
389 builder
.append(UString::number(numericValue
));
390 return StringifySucceeded
;
393 if (!value
.isObject())
394 return StringifyFailed
;
396 JSObject
* object
= asObject(value
);
399 if (object
->getCallData(callData
) != CallTypeNone
) {
400 if (holder
->inherits(&JSArray::s_info
)) {
401 builder
.append("null");
402 return StringifySucceeded
;
404 return StringifyFailedDueToUndefinedValue
;
407 // Handle cycle detection, and put the holder on the stack.
408 for (unsigned i
= 0; i
< m_holderStack
.size(); i
++) {
409 if (m_holderStack
[i
].object() == object
) {
410 throwError(m_exec
, createTypeError(m_exec
, "JSON.stringify cannot serialize cyclic structures."));
411 return StringifyFailed
;
414 bool holderStackWasEmpty
= m_holderStack
.isEmpty();
415 m_holderStack
.append(Holder(m_exec
->globalData(), object
));
416 if (!holderStackWasEmpty
)
417 return StringifySucceeded
;
419 // If this is the outermost call, then loop to handle everything on the holder stack.
420 TimeoutChecker
localTimeoutChecker(m_exec
->globalData().timeoutChecker
);
421 localTimeoutChecker
.reset();
422 unsigned tickCount
= localTimeoutChecker
.ticksUntilNextCheck();
424 while (m_holderStack
.last().appendNextProperty(*this, builder
)) {
425 if (m_exec
->hadException())
426 return StringifyFailed
;
428 if (localTimeoutChecker
.didTimeOut(m_exec
)) {
429 throwError(m_exec
, createInterruptedExecutionException(&m_exec
->globalData()));
430 return StringifyFailed
;
432 tickCount
= localTimeoutChecker
.ticksUntilNextCheck();
435 m_holderStack
.removeLast();
436 } while (!m_holderStack
.isEmpty());
437 return StringifySucceeded
;
440 inline bool Stringifier::willIndent() const
442 return !m_gap
.isEmpty();
445 inline void Stringifier::indent()
447 // Use a single shared string, m_repeatedGap, so we don't keep allocating new ones as we indent and unindent.
448 unsigned newSize
= m_indent
.length() + m_gap
.length();
449 if (newSize
> m_repeatedGap
.length())
450 m_repeatedGap
= makeUString(m_repeatedGap
, m_gap
);
451 ASSERT(newSize
<= m_repeatedGap
.length());
452 m_indent
= m_repeatedGap
.substringSharingImpl(0, newSize
);
455 inline void Stringifier::unindent()
457 ASSERT(m_indent
.length() >= m_gap
.length());
458 m_indent
= m_repeatedGap
.substringSharingImpl(0, m_indent
.length() - m_gap
.length());
461 inline void Stringifier::startNewLine(UStringBuilder
& builder
) const
465 builder
.append('\n');
466 builder
.append(m_indent
);
469 inline Stringifier::Holder::Holder(JSGlobalData
& globalData
, JSObject
* object
)
470 : m_object(globalData
, object
)
471 , m_isArray(object
->inherits(&JSArray::s_info
))
476 bool Stringifier::Holder::appendNextProperty(Stringifier
& stringifier
, UStringBuilder
& builder
)
478 ASSERT(m_index
<= m_size
);
480 ExecState
* exec
= stringifier
.m_exec
;
482 // First time through, initialize.
485 m_isJSArray
= isJSArray(&exec
->globalData(), m_object
.get());
486 m_size
= m_object
->get(exec
, exec
->globalData().propertyNames
->length
).toUInt32(exec
);
489 if (stringifier
.m_usingArrayReplacer
)
490 m_propertyNames
= stringifier
.m_arrayReplacerPropertyNames
.data();
492 PropertyNameArray
objectPropertyNames(exec
);
493 m_object
->getOwnPropertyNames(exec
, objectPropertyNames
);
494 m_propertyNames
= objectPropertyNames
.releaseData();
496 m_size
= m_propertyNames
->propertyNameVector().size();
499 stringifier
.indent();
502 // Last time through, finish up and return false.
503 if (m_index
== m_size
) {
504 stringifier
.unindent();
505 if (m_size
&& builder
[builder
.length() - 1] != '{')
506 stringifier
.startNewLine(builder
);
507 builder
.append(m_isArray
? ']' : '}');
511 // Handle a single element of the array or object.
512 unsigned index
= m_index
++;
513 unsigned rollBackPoint
= 0;
514 StringifyResult stringifyResult
;
518 if (m_isJSArray
&& asArray(m_object
.get())->canGetIndex(index
))
519 value
= asArray(m_object
.get())->getIndex(index
);
521 PropertySlot
slot(m_object
.get());
522 if (!m_object
->getOwnPropertySlot(exec
, index
, slot
))
524 if (exec
->hadException())
526 value
= slot
.getValue(exec
, index
);
529 // Append the separator string.
532 stringifier
.startNewLine(builder
);
534 // Append the stringified value.
535 stringifyResult
= stringifier
.appendStringifiedValue(builder
, value
, m_object
.get(), index
);
538 PropertySlot
slot(m_object
.get());
539 Identifier
& propertyName
= m_propertyNames
->propertyNameVector()[index
];
540 if (!m_object
->getOwnPropertySlot(exec
, propertyName
, slot
))
542 JSValue value
= slot
.getValue(exec
, propertyName
);
543 if (exec
->hadException())
546 rollBackPoint
= builder
.length();
548 // Append the separator string.
549 if (builder
[rollBackPoint
- 1] != '{')
551 stringifier
.startNewLine(builder
);
553 // Append the property name.
554 appendQuotedString(builder
, propertyName
.ustring());
556 if (stringifier
.willIndent())
559 // Append the stringified value.
560 stringifyResult
= stringifier
.appendStringifiedValue(builder
, value
, m_object
.get(), propertyName
);
563 // From this point on, no access to the this pointer or to any members, because the
564 // Holder object may have moved if the call to stringify pushed a new Holder onto
567 switch (stringifyResult
) {
568 case StringifyFailed
:
569 builder
.append("null");
571 case StringifySucceeded
:
573 case StringifyFailedDueToUndefinedValue
:
574 // This only occurs when get an undefined value for an object property.
575 // In this case we don't want the separator and property name that we
576 // already appended, so roll back.
577 builder
.resize(rollBackPoint
);
584 // ------------------------------ JSONObject --------------------------------
586 const ClassInfo
JSONObject::s_info
= { "JSON", &JSObjectWithGlobalObject::s_info
, 0, ExecState::jsonTable
};
588 /* Source for JSONObject.lut.h
590 parse JSONProtoFuncParse DontEnum|Function 2
591 stringify JSONProtoFuncStringify DontEnum|Function 3
597 bool JSONObject::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
599 return getStaticFunctionSlot
<JSObject
>(exec
, ExecState::jsonTable(exec
), this, propertyName
, slot
);
602 bool JSONObject::getOwnPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
)
604 return getStaticFunctionDescriptor
<JSObject
>(exec
, ExecState::jsonTable(exec
), this, propertyName
, descriptor
);
609 Walker(ExecState
* exec
, Handle
<JSObject
> function
, CallType callType
, CallData callData
)
611 , m_function(exec
->globalData(), function
)
612 , m_callType(callType
)
613 , m_callData(callData
)
616 JSValue
walk(JSValue unfiltered
);
618 JSValue
callReviver(JSObject
* thisObj
, JSValue property
, JSValue unfiltered
)
620 JSValue args
[] = { property
, unfiltered
};
621 ArgList
argList(args
, 2);
622 return call(m_exec
, m_function
.get(), m_callType
, m_callData
, thisObj
, argList
);
628 Local
<JSObject
> m_function
;
633 // We clamp recursion well beyond anything reasonable, but we also have a timeout check
634 // to guard against "infinite" execution by inserting arbitrarily large objects.
635 static const unsigned maximumFilterRecursion
= 40000;
636 enum WalkerState
{ StateUnknown
, ArrayStartState
, ArrayStartVisitMember
, ArrayEndVisitMember
,
637 ObjectStartState
, ObjectStartVisitMember
, ObjectEndVisitMember
};
638 NEVER_INLINE JSValue
Walker::walk(JSValue unfiltered
)
640 Vector
<PropertyNameArray
, 16> propertyStack
;
641 Vector
<uint32_t, 16> indexStack
;
642 LocalStack
<JSObject
, 16> objectStack(m_exec
->globalData());
643 LocalStack
<JSArray
, 16> arrayStack(m_exec
->globalData());
645 Vector
<WalkerState
, 16> stateStack
;
646 WalkerState state
= StateUnknown
;
647 JSValue inValue
= unfiltered
;
648 JSValue outValue
= jsNull();
650 TimeoutChecker
localTimeoutChecker(m_exec
->globalData().timeoutChecker
);
651 localTimeoutChecker
.reset();
652 unsigned tickCount
= localTimeoutChecker
.ticksUntilNextCheck();
656 case ArrayStartState
: {
657 ASSERT(inValue
.isObject());
658 ASSERT(isJSArray(&m_exec
->globalData(), asObject(inValue
)) || asObject(inValue
)->inherits(&JSArray::s_info
));
659 if (objectStack
.size() + arrayStack
.size() > maximumFilterRecursion
)
660 return throwError(m_exec
, createStackOverflowError(m_exec
));
662 JSArray
* array
= asArray(inValue
);
663 arrayStack
.push(array
);
664 indexStack
.append(0);
667 arrayStartVisitMember
:
668 case ArrayStartVisitMember
: {
670 if (localTimeoutChecker
.didTimeOut(m_exec
))
671 return throwError(m_exec
, createInterruptedExecutionException(&m_exec
->globalData()));
672 tickCount
= localTimeoutChecker
.ticksUntilNextCheck();
675 JSArray
* array
= arrayStack
.peek();
676 uint32_t index
= indexStack
.last();
677 if (index
== array
->length()) {
680 indexStack
.removeLast();
683 if (isJSArray(&m_exec
->globalData(), array
) && array
->canGetIndex(index
))
684 inValue
= array
->getIndex(index
);
687 if (array
->getOwnPropertySlot(m_exec
, index
, slot
))
688 inValue
= slot
.getValue(m_exec
, index
);
690 inValue
= jsUndefined();
693 if (inValue
.isObject()) {
694 stateStack
.append(ArrayEndVisitMember
);
700 case ArrayEndVisitMember
: {
701 JSArray
* array
= arrayStack
.peek();
702 JSValue filteredValue
= callReviver(array
, jsString(m_exec
, UString::number(indexStack
.last())), outValue
);
703 if (filteredValue
.isUndefined())
704 array
->deleteProperty(m_exec
, indexStack
.last());
706 if (isJSArray(&m_exec
->globalData(), array
) && array
->canSetIndex(indexStack
.last()))
707 array
->setIndex(m_exec
->globalData(), indexStack
.last(), filteredValue
);
709 array
->put(m_exec
, indexStack
.last(), filteredValue
);
711 if (m_exec
->hadException())
714 goto arrayStartVisitMember
;
717 case ObjectStartState
: {
718 ASSERT(inValue
.isObject());
719 ASSERT(!isJSArray(&m_exec
->globalData(), asObject(inValue
)) && !asObject(inValue
)->inherits(&JSArray::s_info
));
720 if (objectStack
.size() + arrayStack
.size() > maximumFilterRecursion
)
721 return throwError(m_exec
, createStackOverflowError(m_exec
));
723 JSObject
* object
= asObject(inValue
);
724 objectStack
.push(object
);
725 indexStack
.append(0);
726 propertyStack
.append(PropertyNameArray(m_exec
));
727 object
->getOwnPropertyNames(m_exec
, propertyStack
.last());
730 objectStartVisitMember
:
731 case ObjectStartVisitMember
: {
733 if (localTimeoutChecker
.didTimeOut(m_exec
))
734 return throwError(m_exec
, createInterruptedExecutionException(&m_exec
->globalData()));
735 tickCount
= localTimeoutChecker
.ticksUntilNextCheck();
738 JSObject
* object
= objectStack
.peek();
739 uint32_t index
= indexStack
.last();
740 PropertyNameArray
& properties
= propertyStack
.last();
741 if (index
== properties
.size()) {
744 indexStack
.removeLast();
745 propertyStack
.removeLast();
749 if (object
->getOwnPropertySlot(m_exec
, properties
[index
], slot
))
750 inValue
= slot
.getValue(m_exec
, properties
[index
]);
752 inValue
= jsUndefined();
754 // The holder may be modified by the reviver function so any lookup may throw
755 if (m_exec
->hadException())
758 if (inValue
.isObject()) {
759 stateStack
.append(ObjectEndVisitMember
);
765 case ObjectEndVisitMember
: {
766 JSObject
* object
= objectStack
.peek();
767 Identifier prop
= propertyStack
.last()[indexStack
.last()];
768 PutPropertySlot slot
;
769 JSValue filteredValue
= callReviver(object
, jsString(m_exec
, prop
.ustring()), outValue
);
770 if (filteredValue
.isUndefined())
771 object
->deleteProperty(m_exec
, prop
);
773 object
->put(m_exec
, prop
, filteredValue
, slot
);
774 if (m_exec
->hadException())
777 goto objectStartVisitMember
;
781 if (!inValue
.isObject()) {
785 JSObject
* object
= asObject(inValue
);
786 if (isJSArray(&m_exec
->globalData(), object
) || object
->inherits(&JSArray::s_info
))
787 goto arrayStartState
;
788 goto objectStartState
;
790 if (stateStack
.isEmpty())
793 state
= stateStack
.last();
794 stateStack
.removeLast();
797 if (localTimeoutChecker
.didTimeOut(m_exec
))
798 return throwError(m_exec
, createInterruptedExecutionException(&m_exec
->globalData()));
799 tickCount
= localTimeoutChecker
.ticksUntilNextCheck();
802 JSObject
* finalHolder
= constructEmptyObject(m_exec
);
803 PutPropertySlot slot
;
804 finalHolder
->put(m_exec
, m_exec
->globalData().propertyNames
->emptyIdentifier
, outValue
, slot
);
805 return callReviver(finalHolder
, jsEmptyString(m_exec
), outValue
);
808 // ECMA-262 v5 15.12.2
809 EncodedJSValue JSC_HOST_CALL
JSONProtoFuncParse(ExecState
* exec
)
811 if (!exec
->argumentCount())
812 return throwVMError(exec
, createError(exec
, "JSON.parse requires at least one parameter"));
813 JSValue value
= exec
->argument(0);
814 UString source
= value
.toString(exec
);
815 if (exec
->hadException())
816 return JSValue::encode(jsNull());
818 LocalScope
scope(exec
->globalData());
819 LiteralParser
jsonParser(exec
, source
.characters(), source
.length(), LiteralParser::StrictJSON
);
820 JSValue unfiltered
= jsonParser
.tryLiteralParse();
822 return throwVMError(exec
, createSyntaxError(exec
, "Unable to parse JSON string"));
824 if (exec
->argumentCount() < 2)
825 return JSValue::encode(unfiltered
);
827 JSValue function
= exec
->argument(1);
829 CallType callType
= getCallData(function
, callData
);
830 if (callType
== CallTypeNone
)
831 return JSValue::encode(unfiltered
);
832 return JSValue::encode(Walker(exec
, Local
<JSObject
>(exec
->globalData(), asObject(function
)), callType
, callData
).walk(unfiltered
));
835 // ECMA-262 v5 15.12.3
836 EncodedJSValue JSC_HOST_CALL
JSONProtoFuncStringify(ExecState
* exec
)
838 if (!exec
->argumentCount())
839 return throwVMError(exec
, createError(exec
, "No input to stringify"));
840 LocalScope
scope(exec
->globalData());
841 Local
<Unknown
> value(exec
->globalData(), exec
->argument(0));
842 Local
<Unknown
> replacer(exec
->globalData(), exec
->argument(1));
843 Local
<Unknown
> space(exec
->globalData(), exec
->argument(2));
844 return JSValue::encode(Stringifier(exec
, replacer
, space
).stringify(value
).get());
847 UString
JSONStringify(ExecState
* exec
, JSValue value
, unsigned indent
)
849 LocalScope
scope(exec
->globalData());
850 Local
<Unknown
> result
= Stringifier(exec
, Local
<Unknown
>(exec
->globalData(), jsNull()), Local
<Unknown
>(exec
->globalData(), jsNumber(indent
))).stringify(Local
<Unknown
>(exec
->globalData(), value
));
851 if (result
.isUndefinedOrNull())
853 return result
.getString(exec
);