[GTK] Clean up RunLoop implementation
[WebKit-https.git] / Source / WTF / wtf / gtk / RunLoopGtk.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     // g_main_context_default() doesn't add an extra reference.
38     m_mainContext = isMainThread() ? g_main_context_default() : adoptGRef(g_main_context_new());
39     ASSERT(m_mainContext);
40     GRefPtr<GMainLoop> innermostLoop = adoptGRef(g_main_loop_new(m_mainContext.get(), FALSE));
41     ASSERT(innermostLoop);
42     m_mainLoops.append(innermostLoop);
43 }
44
45 RunLoop::~RunLoop()
46 {
47     for (int i = m_mainLoops.size() - 1; i >= 0; --i) {
48         if (!g_main_loop_is_running(m_mainLoops[i].get()))
49             continue;
50         g_main_loop_quit(m_mainLoops[i].get());
51     }
52 }
53
54 void RunLoop::run()
55 {
56     RunLoop& runLoop = RunLoop::current();
57
58     // The innermost main loop should always be there.
59     ASSERT(!runLoop.m_mainLoops.isEmpty());
60
61     GMainLoop* innermostLoop = runLoop.m_mainLoops[0].get();
62     if (!g_main_loop_is_running(innermostLoop)) {
63         g_main_loop_run(innermostLoop);
64         return;
65     }
66
67     // Create and run a nested loop if the innermost one was already running.
68     GMainLoop* nestedMainLoop = g_main_loop_new(runLoop.m_mainContext.get(), FALSE);
69     runLoop.m_mainLoops.append(adoptGRef(nestedMainLoop));
70     g_main_loop_run(nestedMainLoop);
71     runLoop.m_mainLoops.removeLast();
72 }
73
74 void RunLoop::stop()
75 {
76     // The innermost main loop should always be there.
77     ASSERT(!m_mainLoops.isEmpty());
78     GRefPtr<GMainLoop> lastMainLoop = m_mainLoops.last();
79     if (g_main_loop_is_running(lastMainLoop.get()))
80         g_main_loop_quit(lastMainLoop.get());
81 }
82
83 void RunLoop::wakeUp()
84 {
85     RefPtr<RunLoop> runLoop(this);
86     GMainLoopSource::scheduleAndDeleteOnDestroy("[WebKit] RunLoop work", std::function<void()>([runLoop] {
87         runLoop->performWork();
88     }), G_PRIORITY_DEFAULT, nullptr, m_mainContext.get());
89     g_main_context_wakeup(m_mainContext.get());
90 }
91
92 RunLoop::TimerBase::TimerBase(RunLoop& runLoop)
93     : m_runLoop(runLoop)
94 {
95 }
96
97 RunLoop::TimerBase::~TimerBase()
98 {
99     stop();
100 }
101
102 void RunLoop::TimerBase::start(double fireInterval, bool repeat)
103 {
104     m_timerSource.scheduleAfterDelay("[WebKit] RunLoop::Timer", std::function<bool ()>([this, repeat] { fired(); return repeat; }),
105         std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::duration<double>(fireInterval)), G_PRIORITY_DEFAULT, nullptr, m_runLoop.m_mainContext.get());
106 }
107
108 void RunLoop::TimerBase::stop()
109 {
110     m_timerSource.cancel();
111 }
112
113 bool RunLoop::TimerBase::isActive() const
114 {
115     return m_timerSource.isScheduled();
116 }
117
118 } // namespace WTF