#define StringRecursionChecker_h
#include "Interpreter.h"
+#include <wtf/StackStats.h>
+#include <wtf/WTFThreadData.h>
namespace JSC {
StringRecursionChecker(ExecState*, JSObject* thisObject);
~StringRecursionChecker();
- EncodedJSValue earlyReturnValue() const; // 0 if everything is OK, value to return for failure cases
+ JSValue earlyReturnValue() const; // 0 if everything is OK, value to return for failure cases
private:
- EncodedJSValue throwStackOverflowError();
- EncodedJSValue emptyString();
- EncodedJSValue performCheck();
+ JSValue throwStackOverflowError();
+ JSValue emptyString();
+ JSValue performCheck();
ExecState* m_exec;
JSObject* m_thisObject;
- EncodedJSValue m_earlyReturnValue;
+ JSValue m_earlyReturnValue;
+
+ StackStats::CheckPoint stackCheckpoint;
};
-inline EncodedJSValue StringRecursionChecker::performCheck()
+inline JSValue StringRecursionChecker::performCheck()
{
- int size = m_exec->globalData().stringRecursionCheckVisitedObjects.size();
- if (size >= MaxSmallThreadReentryDepth && size >= m_exec->globalData().maxReentryDepth)
+ VM& vm = m_exec->vm();
+ if (!vm.isSafeToRecurse())
return throwStackOverflowError();
- bool alreadyVisited = !m_exec->globalData().stringRecursionCheckVisitedObjects.add(m_thisObject).second;
+
+ bool alreadyVisited = false;
+ if (!vm.stringRecursionCheckFirstObject)
+ vm.stringRecursionCheckFirstObject = m_thisObject;
+ else if (vm.stringRecursionCheckFirstObject == m_thisObject)
+ alreadyVisited = true;
+ else
+ alreadyVisited = !vm.stringRecursionCheckVisitedObjects.add(m_thisObject).isNewEntry;
+
if (alreadyVisited)
return emptyString(); // Return empty string to avoid infinite recursion.
- return 0; // Indicate success.
+ return JSValue(); // Indicate success.
}
inline StringRecursionChecker::StringRecursionChecker(ExecState* exec, JSObject* thisObject)
{
}
-inline EncodedJSValue StringRecursionChecker::earlyReturnValue() const
+inline JSValue StringRecursionChecker::earlyReturnValue() const
{
return m_earlyReturnValue;
}
{
if (m_earlyReturnValue)
return;
- ASSERT(m_exec->globalData().stringRecursionCheckVisitedObjects.contains(m_thisObject));
- m_exec->globalData().stringRecursionCheckVisitedObjects.remove(m_thisObject);
+
+ VM& vm = m_exec->vm();
+ if (vm.stringRecursionCheckFirstObject == m_thisObject)
+ vm.stringRecursionCheckFirstObject = nullptr;
+ else {
+ ASSERT(vm.stringRecursionCheckVisitedObjects.contains(m_thisObject));
+ vm.stringRecursionCheckVisitedObjects.remove(m_thisObject);
+ }
}
}