2 * Copyright (C) 2011 Apple Inc. All rights reserved.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20 #ifndef StringRecursionChecker_h
21 #define StringRecursionChecker_h
23 #include "Interpreter.h"
24 #include <wtf/StackStats.h>
25 #include <wtf/WTFThreadData.h>
29 class StringRecursionChecker
{
30 WTF_MAKE_NONCOPYABLE(StringRecursionChecker
);
33 StringRecursionChecker(ExecState
*, JSObject
* thisObject
);
34 ~StringRecursionChecker();
36 JSValue
earlyReturnValue() const; // 0 if everything is OK, value to return for failure cases
39 JSValue
throwStackOverflowError();
40 JSValue
emptyString();
41 JSValue
performCheck();
44 JSObject
* m_thisObject
;
45 JSValue m_earlyReturnValue
;
47 StackStats::CheckPoint stackCheckpoint
;
50 inline JSValue
StringRecursionChecker::performCheck()
52 const StackBounds
& nativeStack
= wtfThreadData().stack();
53 if (!nativeStack
.isSafeToRecurse())
54 return throwStackOverflowError();
55 bool alreadyVisited
= !m_exec
->vm().stringRecursionCheckVisitedObjects
.add(m_thisObject
).isNewEntry
;
57 return emptyString(); // Return empty string to avoid infinite recursion.
58 return JSValue(); // Indicate success.
61 inline StringRecursionChecker::StringRecursionChecker(ExecState
* exec
, JSObject
* thisObject
)
63 , m_thisObject(thisObject
)
64 , m_earlyReturnValue(performCheck())
68 inline JSValue
StringRecursionChecker::earlyReturnValue() const
70 return m_earlyReturnValue
;
73 inline StringRecursionChecker::~StringRecursionChecker()
75 if (m_earlyReturnValue
)
77 ASSERT(m_exec
->vm().stringRecursionCheckVisitedObjects
.contains(m_thisObject
));
78 m_exec
->vm().stringRecursionCheckVisitedObjects
.remove(m_thisObject
);