[iOS] ASSERTION FAILED: m_scriptExecutionContext->isContextThread() in ContextDestruc...
[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 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 <dispatch/dispatch.h>
35 #import <stdio.h>
36 #import <wtf/Assertions.h>
37 #import <wtf/HashSet.h>
38 #import <wtf/Threading.h>
39
40 #if USE(WEB_THREAD)
41 #include <wtf/ios/WebCoreThread.h>
42 #endif
43
44 @interface JSWTFMainThreadCaller : NSObject {
45 }
46 - (void)call;
47 @end
48
49 @implementation JSWTFMainThreadCaller
50
51 - (void)call
52 {
53     WTF::dispatchFunctionsFromMainThread();
54 }
55
56 @end // implementation JSWTFMainThreadCaller
57
58 namespace WTF {
59
60 static JSWTFMainThreadCaller* staticMainThreadCaller;
61 static bool isTimerPosted; // This is only accessed on the 'main' thread.
62 static bool mainThreadEstablishedAsPthreadMain;
63 static pthread_t mainThreadPthread;
64 static NSThread* mainThreadNSThread;
65
66 #if USE(WEB_THREAD)
67 static ThreadIdentifier sApplicationUIThreadIdentifier;
68 static ThreadIdentifier sWebThreadIdentifier;
69 #endif
70
71 void initializeMainThreadPlatform()
72 {
73     ASSERT(!staticMainThreadCaller);
74     staticMainThreadCaller = [[JSWTFMainThreadCaller alloc] init];
75
76 #if !USE(WEB_THREAD)
77     mainThreadEstablishedAsPthreadMain = false;
78     mainThreadPthread = pthread_self();
79     mainThreadNSThread = [[NSThread currentThread] retain];
80 #else
81     mainThreadEstablishedAsPthreadMain = true;
82     ASSERT(!mainThreadPthread);
83     ASSERT(!mainThreadNSThread);
84 #endif
85
86     initializeGCThreads();
87 }
88
89 #if !USE(WEB_THREAD)
90 void initializeMainThreadToProcessMainThreadPlatform()
91 {
92     if (!pthread_main_np())
93         NSLog(@"WebKit Threading Violation - initial use of WebKit from a secondary thread.");
94
95     ASSERT(!staticMainThreadCaller);
96     staticMainThreadCaller = [[JSWTFMainThreadCaller alloc] init];
97
98     mainThreadEstablishedAsPthreadMain = true;
99     mainThreadPthread = 0;
100     mainThreadNSThread = nil;
101     
102     initializeGCThreads();
103 }
104 #endif // !USE(WEB_THREAD)
105
106 static void timerFired(CFRunLoopTimerRef timer, void*)
107 {
108     CFRelease(timer);
109     isTimerPosted = false;
110
111     @autoreleasepool {
112         WTF::dispatchFunctionsFromMainThread();
113     }
114 }
115
116 static void postTimer()
117 {
118     ASSERT(isMainThread());
119
120     if (isTimerPosted)
121         return;
122
123     isTimerPosted = true;
124     CFRunLoopAddTimer(CFRunLoopGetCurrent(), CFRunLoopTimerCreate(0, 0, 0, 0, 0, timerFired, 0), kCFRunLoopCommonModes);
125 }
126
127 void scheduleDispatchFunctionsOnMainThread()
128 {
129     ASSERT(staticMainThreadCaller);
130
131     if (isWebThread()) {
132         postTimer();
133         return;
134     }
135
136     if (mainThreadEstablishedAsPthreadMain) {
137         ASSERT(!mainThreadNSThread);
138         [staticMainThreadCaller performSelectorOnMainThread:@selector(call) withObject:nil waitUntilDone:NO];
139         return;
140     }
141
142     ASSERT(mainThreadNSThread);
143     [staticMainThreadCaller performSelector:@selector(call) onThread:mainThreadNSThread withObject:nil waitUntilDone:NO];
144 }
145
146 void callOnWebThreadOrDispatchAsyncOnMainThread(void (^block)())
147 {
148 #if USE(WEB_THREAD)
149     if (WebCoreWebThreadIsEnabled && WebCoreWebThreadIsEnabled()) {
150         WebCoreWebThreadRun(block);
151         return;
152     }
153 #endif
154     dispatch_async(dispatch_get_main_queue(), block);
155 }
156
157 #if USE(WEB_THREAD)
158 static bool webThreadIsUninitializedOrLockedOrDisabled()
159 {
160     return !WebCoreWebThreadIsLockedOrDisabled || WebCoreWebThreadIsLockedOrDisabled();
161 }
162
163 bool isMainThread()
164 {
165     return (isWebThread() || pthread_main_np()) && webThreadIsUninitializedOrLockedOrDisabled();
166 }
167
168 bool isUIThread()
169 {
170     return pthread_main_np();
171 }
172
173 bool isWebThread()
174 {
175     return pthread_equal(pthread_self(), mainThreadPthread);
176 }
177
178 void initializeApplicationUIThreadIdentifier()
179 {
180     ASSERT(pthread_main_np());
181     sApplicationUIThreadIdentifier = currentThread();
182 }
183
184 void initializeWebThreadIdentifier()
185 {
186     ASSERT(!pthread_main_np());
187     sWebThreadIdentifier = currentThread();
188 }
189
190 void initializeWebThreadPlatform()
191 {
192     ASSERT(!pthread_main_np());
193
194     mainThreadEstablishedAsPthreadMain = false;
195     mainThreadPthread = pthread_self();
196     mainThreadNSThread = [[NSThread currentThread] retain];
197 }
198
199 bool canAccessThreadLocalDataForThread(ThreadIdentifier threadId)
200 {
201     ThreadIdentifier currentThreadId = currentThread();
202     if (threadId == currentThreadId)
203         return true;
204
205     if (threadId == sWebThreadIdentifier || threadId == sApplicationUIThreadIdentifier)
206         return (currentThreadId == sWebThreadIdentifier || currentThreadId == sApplicationUIThreadIdentifier) && webThreadIsUninitializedOrLockedOrDisabled();
207
208     return false;
209 }
210 #else
211 bool isMainThread()
212 {
213     if (mainThreadEstablishedAsPthreadMain) {
214         ASSERT(!mainThreadPthread);
215         return pthread_main_np();
216     }
217
218     ASSERT(mainThreadPthread);
219     return pthread_equal(pthread_self(), mainThreadPthread);
220 }
221 #endif // USE(WEB_THREAD)
222
223 } // namespace WTF