1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
|
/*
* Copyright (C) 2013 Google Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#include "core/dom/MainThreadTaskRunner.h"
#include "core/dom/ExecutionContext.h"
#include "core/dom/ExecutionContextTask.h"
#include "core/inspector/InspectorInstrumentation.h"
#include "platform/ThreadSafeFunctional.h"
#include "public/platform/Platform.h"
#include "wtf/Assertions.h"
namespace blink {
MainThreadTaskRunner::MainThreadTaskRunner(ExecutionContext* context)
: m_context(context)
#if !ENABLE(OILPAN)
, m_weakFactory(this)
#endif
, m_pendingTasksTimer(this, &MainThreadTaskRunner::pendingTasksTimerFired)
, m_suspended(false)
, m_weakFactory(this)
{
}
MainThreadTaskRunner::~MainThreadTaskRunner()
{
}
void MainThreadTaskRunner::postTaskInternal(const WebTraceLocation& location, PassOwnPtr<ExecutionContextTask> task, bool isInspectorTask)
{
Platform::current()->mainThread()->getWebTaskRunner()->postTask(location, threadSafeBind(
&MainThreadTaskRunner::perform,
AllowCrossThreadAccess(m_weakFactory.createWeakPtr()),
task,
isInspectorTask));
}
void MainThreadTaskRunner::postTask(const WebTraceLocation& location, PassOwnPtr<ExecutionContextTask> task)
{
if (!task->taskNameForInstrumentation().isEmpty())
InspectorInstrumentation::asyncTaskScheduled(m_context, task->taskNameForInstrumentation(), task.get());
postTaskInternal(location, task, false);
}
void MainThreadTaskRunner::postInspectorTask(const WebTraceLocation& location, PassOwnPtr<ExecutionContextTask> task)
{
postTaskInternal(location, task, true);
}
void MainThreadTaskRunner::perform(PassOwnPtr<ExecutionContextTask> task, bool isInspectorTask)
{
// If the owner m_context is about to be swept then it
// is no longer safe to access.
if (Heap::willObjectBeLazilySwept(m_context.get()))
return;
if (!isInspectorTask && (m_context->tasksNeedSuspension() || !m_pendingTasks.isEmpty())) {
m_pendingTasks.append(task);
return;
}
InspectorInstrumentation::AsyncTask asyncTask(m_context, task.get(), !isInspectorTask);
task->performTask(m_context);
}
void MainThreadTaskRunner::suspend()
{
DCHECK(!m_suspended);
m_pendingTasksTimer.stop();
m_suspended = true;
}
void MainThreadTaskRunner::resume()
{
DCHECK(m_suspended);
if (!m_pendingTasks.isEmpty())
m_pendingTasksTimer.startOneShot(0, BLINK_FROM_HERE);
m_suspended = false;
}
void MainThreadTaskRunner::pendingTasksTimerFired(Timer<MainThreadTaskRunner>*)
{
// If the owner m_context is about to be swept then it
// is no longer safe to access.
if (Heap::willObjectBeLazilySwept(m_context.get()))
return;
while (!m_pendingTasks.isEmpty()) {
OwnPtr<ExecutionContextTask> task = m_pendingTasks[0].release();
m_pendingTasks.remove(0);
const bool instrumenting = !task->taskNameForInstrumentation().isEmpty();
InspectorInstrumentation::AsyncTask asyncTask(m_context, task.get(), instrumenting);
task->performTask(m_context);
}
}
} // namespace blink
|