Unreviewed, rolling out r191728.
[WebKit-https.git] / Source / WTF / wtf / glib / RunLoopGLib.cpp
1 /*
2  * Copyright (C) 2010 Apple Inc. All rights reserved.
3  * Portions Copyright (c) 2010 Motorola Mobility, Inc.  All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
15  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
16  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
18  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
19  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
20  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
21  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
22  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
23  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
24  * THE POSSIBILITY OF SUCH DAMAGE.
25  */
26
27 #include "config.h"
28 #include "RunLoop.h"
29
30 #include <glib.h>
31 #include <wtf/MainThread.h>
32
33 namespace WTF {
34
35 RunLoop::RunLoop()
36 {
37     m_mainContext = g_main_context_get_thread_default();
38     if (!m_mainContext)
39         m_mainContext = isMainThread() ? g_main_context_default() : adoptGRef(g_main_context_new());
40     ASSERT(m_mainContext);
41
42     GRefPtr<GMainLoop> innermostLoop = adoptGRef(g_main_loop_new(m_mainContext.get(), FALSE));
43     ASSERT(innermostLoop);
44     m_mainLoops.append(innermostLoop);
45 }
46
47 RunLoop::~RunLoop()
48 {
49     for (int i = m_mainLoops.size() - 1; i >= 0; --i) {
50         if (!g_main_loop_is_running(m_mainLoops[i].get()))
51             continue;
52         g_main_loop_quit(m_mainLoops[i].get());
53     }
54 }
55
56 void RunLoop::run()
57 {
58     RunLoop& runLoop = RunLoop::current();
59     GMainContext* mainContext = runLoop.m_mainContext.get();
60
61     // The innermost main loop should always be there.
62     ASSERT(!runLoop.m_mainLoops.isEmpty());
63
64     GMainLoop* innermostLoop = runLoop.m_mainLoops[0].get();
65     if (!g_main_loop_is_running(innermostLoop)) {
66         g_main_context_push_thread_default(mainContext);
67         g_main_loop_run(innermostLoop);
68         g_main_context_pop_thread_default(mainContext);
69         return;
70     }
71
72     // Create and run a nested loop if the innermost one was already running.
73     GMainLoop* nestedMainLoop = g_main_loop_new(mainContext, FALSE);
74     runLoop.m_mainLoops.append(adoptGRef(nestedMainLoop));
75
76     g_main_context_push_thread_default(mainContext);
77     g_main_loop_run(nestedMainLoop);
78     g_main_context_pop_thread_default(mainContext);
79
80     runLoop.m_mainLoops.removeLast();
81 }
82
83 void RunLoop::stop()
84 {
85     // The innermost main loop should always be there.
86     ASSERT(!m_mainLoops.isEmpty());
87     GRefPtr<GMainLoop> lastMainLoop = m_mainLoops.last();
88     if (g_main_loop_is_running(lastMainLoop.get()))
89         g_main_loop_quit(lastMainLoop.get());
90 }
91
92 void RunLoop::wakeUp()
93 {
94     RefPtr<RunLoop> runLoop(this);
95     GMainLoopSource::scheduleAndDeleteOnDestroy("[WebKit] RunLoop work", std::function<void()>([runLoop] {
96         runLoop->performWork();
97     }), G_PRIORITY_DEFAULT, nullptr, m_mainContext.get());
98     g_main_context_wakeup(m_mainContext.get());
99 }
100
101 RunLoop::TimerBase::TimerBase(RunLoop& runLoop)
102     : m_runLoop(runLoop)
103 {
104 }
105
106 RunLoop::TimerBase::~TimerBase()
107 {
108     stop();
109 }
110
111 void RunLoop::TimerBase::start(double fireInterval, bool repeat)
112 {
113     m_timerSource.scheduleAfterDelay("[WebKit] RunLoop::Timer", std::function<bool ()>([this, repeat] { fired(); return repeat; }),
114         std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::duration<double>(fireInterval)), G_PRIORITY_DEFAULT, nullptr, m_runLoop.m_mainContext.get());
115 }
116
117 void RunLoop::TimerBase::stop()
118 {
119     m_timerSource.cancel();
120 }
121
122 bool RunLoop::TimerBase::isActive() const
123 {
124     return m_timerSource.isScheduled();
125 }
126
127 } // namespace WTF