#ifndef AsyncCallStackTracker_h
#define AsyncCallStackTracker_h
-#include "bindings/v8/ScriptValue.h"
+#include "bindings/core/v8/ScriptValue.h"
+#include "core/dom/ContextLifecycleObserver.h"
#include "wtf/Deque.h"
#include "wtf/HashMap.h"
#include "wtf/HashSet.h"
#include "wtf/PassRefPtr.h"
#include "wtf/RefPtr.h"
-namespace WebCore {
+namespace blink {
class Event;
class EventListener;
class EventTarget;
class ExecutionContext;
+class ExecutionContextTask;
class MutationObserver;
class XMLHttpRequest;
-class AsyncCallStackTracker {
+class AsyncCallStackTracker FINAL : public NoBaseWillBeGarbageCollectedFinalized<AsyncCallStackTracker> {
WTF_MAKE_NONCOPYABLE(AsyncCallStackTracker);
public:
- class AsyncCallStack : public RefCounted<AsyncCallStack> {
+ class AsyncCallStack FINAL : public RefCountedWillBeGarbageCollectedFinalized<AsyncCallStack> {
public:
AsyncCallStack(const String&, const ScriptValue&);
~AsyncCallStack();
+ void trace(Visitor*) { }
String description() const { return m_description; }
ScriptValue callFrames() const { return m_callFrames; }
private:
ScriptValue m_callFrames;
};
- typedef Deque<RefPtr<AsyncCallStack>, 4> AsyncCallStackVector;
+ typedef WillBeHeapDeque<RefPtrWillBeMember<AsyncCallStack>, 4> AsyncCallStackVector;
- class AsyncCallChain : public RefCounted<AsyncCallChain> {
+ class AsyncCallChain FINAL : public RefCountedWillBeGarbageCollected<AsyncCallChain> {
public:
AsyncCallChain() { }
AsyncCallChain(const AsyncCallChain& t) : m_callStacks(t.m_callStacks) { }
AsyncCallStackVector callStacks() const { return m_callStacks; }
+ void trace(Visitor*);
private:
friend class AsyncCallStackTracker;
AsyncCallStackVector m_callStacks;
};
+ class ExecutionContextData FINAL : public NoBaseWillBeGarbageCollectedFinalized<ExecutionContextData>, public ContextLifecycleObserver {
+ WTF_MAKE_FAST_ALLOCATED_WILL_BE_REMOVED;
+ public:
+ ExecutionContextData(AsyncCallStackTracker* tracker, ExecutionContext* executionContext)
+ : ContextLifecycleObserver(executionContext)
+ , m_circularSequentialID(0)
+ , m_tracker(tracker)
+ {
+ }
+
+ virtual void contextDestroyed() OVERRIDE;
+
+ int circularSequentialID();
+
+ void trace(Visitor*);
+
+ private:
+ int m_circularSequentialID;
+
+ public:
+ RawPtrWillBeMember<AsyncCallStackTracker> m_tracker;
+ HashSet<int> m_intervalTimerIds;
+ WillBeHeapHashMap<int, RefPtrWillBeMember<AsyncCallChain> > m_timerCallChains;
+ WillBeHeapHashMap<int, RefPtrWillBeMember<AsyncCallChain> > m_animationFrameCallChains;
+ WillBeHeapHashMap<RawPtrWillBeMember<Event>, RefPtrWillBeMember<AsyncCallChain> > m_eventCallChains;
+ WillBeHeapHashMap<RawPtrWillBeMember<EventTarget>, RefPtrWillBeMember<AsyncCallChain> > m_xhrCallChains;
+ WillBeHeapHashMap<RawPtrWillBeMember<MutationObserver>, RefPtrWillBeMember<AsyncCallChain> > m_mutationObserverCallChains;
+ WillBeHeapHashMap<ExecutionContextTask*, RefPtrWillBeMember<AsyncCallChain> > m_executionContextTaskCallChains;
+ WillBeHeapHashMap<String, RefPtrWillBeMember<AsyncCallChain> > m_v8AsyncTaskCallChains;
+ WillBeHeapHashMap<int, RefPtrWillBeMember<AsyncCallChain> > m_asyncOperationCallChains;
+ };
+
AsyncCallStackTracker();
bool isEnabled() const { return m_maxAsyncCallStackDepth; }
void didEnqueueEvent(EventTarget*, Event*, const ScriptValue& callFrames);
void didRemoveEvent(EventTarget*, Event*);
void willHandleEvent(EventTarget*, Event*, EventListener*, bool useCapture);
+
void willLoadXHR(XMLHttpRequest*, const ScriptValue& callFrames);
+ void didLoadXHR(XMLHttpRequest*);
void didEnqueueMutationRecord(ExecutionContext*, MutationObserver*, const ScriptValue& callFrames);
bool hasEnqueuedMutationRecord(ExecutionContext*, MutationObserver*);
void didClearAllMutationRecords(ExecutionContext*, MutationObserver*);
void willDeliverMutationRecords(ExecutionContext*, MutationObserver*);
+ void didPostExecutionContextTask(ExecutionContext*, ExecutionContextTask*, const ScriptValue& callFrames);
+ void didKillAllExecutionContextTasks(ExecutionContext*);
+ void willPerformExecutionContextTask(ExecutionContext*, ExecutionContextTask*);
+
+ void didEnqueueV8AsyncTask(ExecutionContext*, const String& eventName, int id, const ScriptValue& callFrames);
+ void willHandleV8AsyncTask(ExecutionContext*, const String& eventName, int id);
+
+ int traceAsyncOperationStarting(ExecutionContext*, const String& operationName, const ScriptValue& callFrames);
+ void traceAsyncOperationCompleted(ExecutionContext*, int operationId);
+ void traceAsyncCallbackStarting(ExecutionContext*, int operationId);
+
void didFireAsyncCall();
void clear();
+ void trace(Visitor*);
+
private:
- void willHandleXHREvent(XMLHttpRequest*, EventTarget*, Event*);
+ void willHandleXHREvent(XMLHttpRequest*, Event*);
- PassRefPtr<AsyncCallChain> createAsyncCallChain(const String& description, const ScriptValue& callFrames);
- void setCurrentAsyncCallChain(PassRefPtr<AsyncCallChain>);
+ PassRefPtrWillBeRawPtr<AsyncCallChain> createAsyncCallChain(const String& description, const ScriptValue& callFrames);
+ void setCurrentAsyncCallChain(ExecutionContext*, PassRefPtrWillBeRawPtr<AsyncCallChain>);
void clearCurrentAsyncCallChain();
static void ensureMaxAsyncCallChainDepth(AsyncCallChain*, unsigned);
- static bool validateCallFrames(const ScriptValue& callFrames);
+ bool validateCallFrames(const ScriptValue& callFrames);
- class ExecutionContextData;
ExecutionContextData* createContextDataIfNeeded(ExecutionContext*);
unsigned m_maxAsyncCallStackDepth;
- RefPtr<AsyncCallChain> m_currentAsyncCallChain;
+ RefPtrWillBeMember<AsyncCallChain> m_currentAsyncCallChain;
unsigned m_nestedAsyncCallCount;
- typedef HashMap<ExecutionContext*, ExecutionContextData*> ExecutionContextDataMap;
+ typedef WillBeHeapHashMap<RawPtrWillBeMember<ExecutionContext>, OwnPtrWillBeMember<ExecutionContextData> > ExecutionContextDataMap;
ExecutionContextDataMap m_executionContextDataMap;
};
-} // namespace WebCore
+} // namespace blink
#endif // !defined(AsyncCallStackTracker_h)