cc3fee965e5a880d8209b9524400f41b4627866a
[WebKit-https.git] / Source / WTF / wtf / mac / MainThreadMac.mm
1 /*
2  * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  *
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  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
14  *     its contributors may be used to endorse or promote products derived
15  *     from this software without specific prior written permission. 
16  *
17  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
18  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28  
29 #import "config.h"
30 #import "MainThread.h"
31
32 #import <CoreFoundation/CoreFoundation.h>
33 #import <Foundation/NSThread.h>
34 #import <stdio.h>
35 #import <wtf/Assertions.h>
36 #import <wtf/HashSet.h>
37 #import <wtf/Threading.h>
38
39 #if USE(WEB_THREAD)
40 #include <wtf/ios/WebCoreThread.h>
41 #endif
42
43 @interface JSWTFMainThreadCaller : NSObject {
44 }
45 - (void)call;
46 @end
47
48 @implementation JSWTFMainThreadCaller
49
50 - (void)call
51 {
52     WTF::dispatchFunctionsFromMainThread();
53 }
54
55 @end // implementation JSWTFMainThreadCaller
56
57 namespace WTF {
58
59 static JSWTFMainThreadCaller* staticMainThreadCaller;
60 static bool isTimerPosted; // This is only accessed on the 'main' thread.
61 static bool mainThreadEstablishedAsPthreadMain;
62 static pthread_t mainThreadPthread;
63 static NSThread* mainThreadNSThread;
64
65 #if USE(WEB_THREAD)
66 static ThreadIdentifier sApplicationUIThreadIdentifier;
67 static ThreadIdentifier sWebThreadIdentifier;
68 #endif
69
70 void initializeMainThreadPlatform()
71 {
72     ASSERT(!staticMainThreadCaller);
73     staticMainThreadCaller = [[JSWTFMainThreadCaller alloc] init];
74
75     mainThreadEstablishedAsPthreadMain = false;
76     mainThreadPthread = pthread_self();
77     mainThreadNSThread = [[NSThread currentThread] retain];
78     
79     initializeGCThreads();
80 }
81
82 #if !USE(WEB_THREAD)
83 void initializeMainThreadToProcessMainThreadPlatform()
84 {
85     if (!pthread_main_np())
86         NSLog(@"WebKit Threading Violation - initial use of WebKit from a secondary thread.");
87
88     ASSERT(!staticMainThreadCaller);
89     staticMainThreadCaller = [[JSWTFMainThreadCaller alloc] init];
90
91     mainThreadEstablishedAsPthreadMain = true;
92     mainThreadPthread = 0;
93     mainThreadNSThread = nil;
94     
95     initializeGCThreads();
96 }
97 #endif // !USE(WEB_THREAD)
98
99 static void timerFired(CFRunLoopTimerRef timer, void*)
100 {
101     CFRelease(timer);
102     isTimerPosted = false;
103     WTF::dispatchFunctionsFromMainThread();
104 }
105
106 static void postTimer()
107 {
108     ASSERT(isMainThread());
109
110     if (isTimerPosted)
111         return;
112
113     isTimerPosted = true;
114     CFRunLoopAddTimer(CFRunLoopGetCurrent(), CFRunLoopTimerCreate(0, 0, 0, 0, 0, timerFired, 0), kCFRunLoopCommonModes);
115 }
116
117 void scheduleDispatchFunctionsOnMainThread()
118 {
119     ASSERT(staticMainThreadCaller);
120
121     if (isMainThread()) {
122         postTimer();
123         return;
124     }
125
126     if (mainThreadEstablishedAsPthreadMain) {
127         ASSERT(!mainThreadNSThread);
128         [staticMainThreadCaller performSelectorOnMainThread:@selector(call) withObject:nil waitUntilDone:NO];
129         return;
130     }
131
132     ASSERT(mainThreadNSThread);
133     [staticMainThreadCaller performSelector:@selector(call) onThread:mainThreadNSThread withObject:nil waitUntilDone:NO];
134 }
135
136 #if USE(WEB_THREAD)
137 bool isMainThread()
138 {
139     ASSERT(!mainThreadEstablishedAsPthreadMain);
140     return (isWebThread() || pthread_main_np()) && WebCoreWebThreadIsLockedOrDisabled();
141 }
142
143 bool isUIThread()
144 {
145     return pthread_main_np();
146 }
147
148 bool isWebThread()
149 {
150     return pthread_equal(pthread_self(), mainThreadPthread);
151 }
152
153 void initializeApplicationUIThreadIdentifier()
154 {
155     ASSERT(pthread_main_np());
156     sApplicationUIThreadIdentifier = currentThread();
157 }
158
159 void initializeWebThreadIdentifier()
160 {
161     ASSERT(!pthread_main_np());
162     sWebThreadIdentifier = currentThread();
163 }
164
165 bool canAccessThreadLocalDataForThread(ThreadIdentifier threadId)
166 {
167     ThreadIdentifier currentThreadId = currentThread();
168     if (threadId == currentThreadId)
169         return true;
170
171     if (threadId == sWebThreadIdentifier || threadId == sApplicationUIThreadIdentifier)
172         return (currentThreadId == sWebThreadIdentifier || currentThreadId == sApplicationUIThreadIdentifier) && WebCoreWebThreadIsLockedOrDisabled();
173
174     return false;
175 }
176 #else
177 bool isMainThread()
178 {
179     if (mainThreadEstablishedAsPthreadMain) {
180         ASSERT(!mainThreadPthread);
181         return pthread_main_np();
182     }
183
184     ASSERT(mainThreadPthread);
185     return pthread_equal(pthread_self(), mainThreadPthread);
186 }
187 #endif // USE(WEB_THREAD)
188
189 } // namespace WTF