#ifndef AsyncCallStackTracker_h
#define AsyncCallStackTracker_h
#include "bindings/v8/ScriptValue.h"
#include "wtf/Deque.h"
#include "wtf/HashMap.h"
#include "wtf/HashSet.h"
#include "wtf/Noncopyable.h"
#include "wtf/PassRefPtr.h"
#include "wtf/RefPtr.h"
namespace WebCore {
class EventListener;
class EventTarget;
class ExecutionContext;
class ExecutionContextTask;
class MutationObserver;
class XMLHttpRequest;
class AsyncCallStackTracker {
WTF_MAKE_NONCOPYABLE(AsyncCallStackTracker);
public:
class AsyncCallStack : public RefCounted<AsyncCallStack> {
public:
AsyncCallStack(const String&, const ScriptValue&);
~AsyncCallStack();
String description() const { return m_description; }
ScriptValue callFrames() const { return m_callFrames; }
private:
String m_description;
ScriptValue m_callFrames;
};
typedef Deque<RefPtr<AsyncCallStack>, 4> AsyncCallStackVector;
class AsyncCallChain : public RefCounted<AsyncCallChain> {
public:
AsyncCallChain() { }
AsyncCallChain(const AsyncCallChain& t) : m_callStacks(t.m_callStacks) { }
AsyncCallStackVector callStacks() const { return m_callStacks; }
private:
friend class AsyncCallStackTracker;
AsyncCallStackVector m_callStacks;
};
AsyncCallStackTracker();
bool isEnabled() const { return m_maxAsyncCallStackDepth; }
void setAsyncCallStackDepth(int);
const AsyncCallChain* currentAsyncCallChain() const;
void didInstallTimer(ExecutionContext*, int timerId, bool singleShot, const ScriptValue& callFrames);
void didRemoveTimer(ExecutionContext*, int timerId);
void willFireTimer(ExecutionContext*, int timerId);
void didRequestAnimationFrame(ExecutionContext*, int callbackId, const ScriptValue& callFrames);
void didCancelAnimationFrame(ExecutionContext*, int callbackId);
void willFireAnimationFrame(ExecutionContext*, int callbackId);
void didAddEventListener(EventTarget*, const AtomicString& eventType, EventListener*, bool useCapture, const ScriptValue& callFrames);
void didRemoveEventListener(EventTarget*, const AtomicString& eventType, EventListener*, bool useCapture);
void didRemoveAllEventListeners(EventTarget*);
void willHandleEvent(EventTarget*, const AtomicString& eventType, EventListener*, bool useCapture);
void willLoadXHR(XMLHttpRequest*, const ScriptValue& callFrames);
void didEnqueueMutationRecord(ExecutionContext*, MutationObserver*, const ScriptValue& callFrames);
bool hasEnqueuedMutationRecord(ExecutionContext*, MutationObserver*);
void didClearAllMutationRecords(ExecutionContext*, MutationObserver*);
void willDeliverMutationRecords(ExecutionContext*, MutationObserver*);
void didPostPromiseTask(ExecutionContext*, ExecutionContextTask*, bool isResolved, const ScriptValue& callFrames);
void willPerformPromiseTask(ExecutionContext*, ExecutionContextTask*);
void didFireAsyncCall();
void clear();
private:
void willHandleXHREvent(XMLHttpRequest*, EventTarget*, const AtomicString& eventType);
PassRefPtr<AsyncCallChain> createAsyncCallChain(const String& description, const ScriptValue& callFrames);
void setCurrentAsyncCallChain(PassRefPtr<AsyncCallChain>);
void clearCurrentAsyncCallChain();
static void ensureMaxAsyncCallChainDepth(AsyncCallChain*, unsigned);
static bool validateCallFrames(const ScriptValue& callFrames);
class ExecutionContextData;
ExecutionContextData* createContextDataIfNeeded(ExecutionContext*);
unsigned m_maxAsyncCallStackDepth;
RefPtr<AsyncCallChain> m_currentAsyncCallChain;
unsigned m_nestedAsyncCallCount;
typedef HashMap<ExecutionContext*, ExecutionContextData*> ExecutionContextDataMap;
ExecutionContextDataMap m_executionContextDataMap;
};
}
#endif