2 * Copyright (C) 2015 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23 * THE POSSIBILITY OF SUCH DAMAGE.
27 #import "Regress141275.h"
29 #import <Foundation/Foundation.h>
31 #import <objc/runtime.h>
33 #if JSC_OBJC_API_ENABLED
35 extern "C" void JSSynchronousGarbageCollectForDebugging(JSContextRef);
39 static const NSUInteger scriptToEvaluate = 50;
41 @interface JSTEvaluator : NSObject
42 - (instancetype)initWithScript:(NSString*)script;
44 - (void)insertSignPostWithCompletion:(void(^)(NSError* error))completionHandler;
46 - (void)evaluateScript:(NSString*)script completion:(void(^)(NSError* error))completionHandler;
47 - (void)evaluateBlock:(void(^)(JSContext* context))evaluationBlock completion:(void(^)(NSError* error))completionHandler;
49 - (void)waitForTasksDoneAndReportResults;
53 static const NSString* JSTEvaluatorThreadContextKey = @"JSTEvaluatorThreadContextKey";
56 * A JSTEvaluatorThreadContext is kept in the thread dictionary of threads used by JSEvaluator.
58 * This includes the run loop thread, and any threads used by _jsSourcePerformQueue to execute a task.
60 @interface JSTEvaluatorThreadContext : NSObject
61 @property (weak) JSTEvaluator* evaluator;
62 @property (strong) JSContext* jsContext;
65 @implementation JSTEvaluatorThreadContext
70 * A JSTEvaluatorTask is a single task to be executed.
72 * JSTEvaluator keeps a list of pending tasks. The run loop thread is repsonsible for feeding pending tasks to the _jsSourcePerformQueue, while respecting sign posts.
74 @interface JSTEvaluatorTask : NSObject
76 @property (nonatomic, copy) void (^evaluateBlock)(JSContext* jsContext);
77 @property (nonatomic, copy) void (^completionHandler)(NSError* error);
78 @property (nonatomic, copy) NSError* error;
80 + (instancetype)evaluatorTaskWithEvaluateBlock:(void (^)(JSContext*))block completionHandler:(void (^)(NSError* error))completionBlock;
84 @implementation JSTEvaluatorTask
86 + (instancetype)evaluatorTaskWithEvaluateBlock:(void (^)(JSContext*))evaluationBlock completionHandler:(void (^)(NSError* error))completionHandler
88 JSTEvaluatorTask* task = [self new];
89 task.evaluateBlock = evaluationBlock;
90 task.completionHandler = completionHandler;
96 @implementation JSTEvaluator {
97 dispatch_queue_t _jsSourcePerformQueue;
98 dispatch_semaphore_t _allScriptsDone;
99 CFRunLoopRef _jsThreadRunLoop;
100 CFRunLoopSourceRef _jsThreadRunLoopSource;
101 JSContext* _jsContext;
102 NSMutableArray* __pendingTasks;
109 _jsSourcePerformQueue = dispatch_queue_create("JSTEval", DISPATCH_QUEUE_CONCURRENT);
111 _allScriptsDone = dispatch_semaphore_create(0);
113 _jsContext = [JSContext new];
114 _jsContext.name = @"JSTEval";
115 __pendingTasks = [NSMutableArray new];
117 NSThread* jsThread = [[NSThread alloc] initWithTarget:self selector:@selector(_jsThreadMain) object:nil];
118 [jsThread setName:@"JSTEval"];
125 - (instancetype)initWithScript:(NSString*)script
129 __block NSError* scriptError = nil;
130 dispatch_semaphore_t dsema = dispatch_semaphore_create(0);
131 [self evaluateScript:script
132 completion:^(NSError* error) {
134 dispatch_semaphore_signal(dsema);
136 dispatch_semaphore_wait(dsema, DISPATCH_TIME_FOREVER);
141 - (void)_accessPendingTasksWithBlock:(void(^)(NSMutableArray* pendingTasks))block
143 @synchronized(self) {
144 block(__pendingTasks);
145 if (__pendingTasks.count > 0) {
146 if (_jsThreadRunLoop && _jsThreadRunLoopSource) {
147 CFRunLoopSourceSignal(_jsThreadRunLoopSource);
148 CFRunLoopWakeUp(_jsThreadRunLoop);
154 - (void)insertSignPostWithCompletion:(void(^)(NSError* error))completionHandler
156 [self _accessPendingTasksWithBlock:^(NSMutableArray* pendingTasks) {
157 JSTEvaluatorTask* task = [JSTEvaluatorTask evaluatorTaskWithEvaluateBlock:nil
158 completionHandler:completionHandler];
160 [pendingTasks addObject:task];
164 - (void)evaluateScript:(NSString*)script completion:(void(^)(NSError* error))completionHandler
166 [self evaluateBlock:^(JSContext* context) {
167 [context evaluateScript:script];
168 } completion:completionHandler];
171 - (void)evaluateBlock:(void(^)(JSContext* context))evaluationBlock completion:(void(^)(NSError* error))completionHandler
173 NSParameterAssert(evaluationBlock != nil);
174 [self _accessPendingTasksWithBlock:^(NSMutableArray* pendingTasks) {
175 JSTEvaluatorTask* task = [JSTEvaluatorTask evaluatorTaskWithEvaluateBlock:evaluationBlock
176 completionHandler:completionHandler];
178 [pendingTasks addObject:task];
182 - (void)waitForTasksDoneAndReportResults
184 NSString* passFailString = @"PASSED";
186 if (!dispatch_semaphore_wait(_allScriptsDone, dispatch_time(DISPATCH_TIME_NOW, 30 * NSEC_PER_SEC))) {
187 int totalScriptsRun = [_jsContext[@"counter"] toInt32];
189 if (totalScriptsRun != scriptToEvaluate) {
190 passFailString = @"FAILED";
194 NSLog(@" Ran a total of %d scripts: %@", totalScriptsRun, passFailString);
196 passFailString = @"FAILED";
198 NSLog(@" Error, timeout waiting for all tasks to complete: %@", passFailString);
202 static void __JSTRunLoopSourceScheduleCallBack(void* info, CFRunLoopRef rl, CFStringRef)
205 [(__bridge JSTEvaluator*)info _sourceScheduledOnRunLoop:rl];
209 static void __JSTRunLoopSourcePerformCallBack(void* info )
212 [(__bridge JSTEvaluator*)info _sourcePerform];
216 static void __JSTRunLoopSourceCancelCallBack(void* info, CFRunLoopRef rl, CFStringRef)
219 [(__bridge JSTEvaluator*)info _sourceCanceledOnRunLoop:rl];
223 - (void)_jsThreadMain
226 const CFIndex kRunLoopSourceContextVersion = 0;
227 CFRunLoopSourceContext sourceContext = {
228 kRunLoopSourceContextVersion, (__bridge void*)(self),
229 NULL, NULL, NULL, NULL, NULL,
230 __JSTRunLoopSourceScheduleCallBack,
231 __JSTRunLoopSourceCancelCallBack,
232 __JSTRunLoopSourcePerformCallBack
235 @synchronized(self) {
236 _jsThreadRunLoop = CFRunLoopGetCurrent();
237 CFRetain(_jsThreadRunLoop);
239 _jsThreadRunLoopSource = CFRunLoopSourceCreate(kCFAllocatorDefault, 0, &sourceContext);
240 CFRunLoopAddSource(_jsThreadRunLoop, _jsThreadRunLoopSource, kCFRunLoopDefaultMode);
245 @synchronized(self) {
246 NSMutableDictionary* threadDict = [[NSThread currentThread] threadDictionary];
247 [threadDict removeObjectForKey:threadDict[JSTEvaluatorThreadContextKey]];
249 CFRelease(_jsThreadRunLoopSource);
250 _jsThreadRunLoopSource = NULL;
252 CFRelease(_jsThreadRunLoop);
253 _jsThreadRunLoop = NULL;
255 __pendingTasks = nil;
260 - (void)_sourceScheduledOnRunLoop:(CFRunLoopRef)runLoop
262 UNUSED_PARAM(runLoop);
263 assert([[[NSThread currentThread] name] isEqualToString:@"JSTEval"]);
265 // Wake up the run loop in case requests were submitted prior to the
266 // run loop & run loop source getting created.
267 CFRunLoopSourceSignal(_jsThreadRunLoopSource);
268 CFRunLoopWakeUp(_jsThreadRunLoop);
271 - (void)_setupEvaluatorThreadContextIfNeeded
273 NSMutableDictionary* threadDict = [[NSThread currentThread] threadDictionary];
274 JSTEvaluatorThreadContext* context = threadDict[JSTEvaluatorThreadContextKey];
275 // The evaluator may be other evualuator, or nil if this thread has not been used before. Eaither way take ownership.
276 if (context.evaluator != self) {
277 context = [JSTEvaluatorThreadContext new];
278 context.evaluator = self;
279 threadDict[JSTEvaluatorThreadContextKey] = context;
283 - (void)_callCompletionHandler:(void(^)(NSError* error))completionHandler ifNeededWithError:(NSError*)error
285 if (completionHandler) {
286 dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
287 completionHandler(error);
292 - (void)_sourcePerform
294 assert([[[NSThread currentThread] name] isEqualToString:@"JSTEval"]);
296 __block NSArray* tasks = nil;
297 [self _accessPendingTasksWithBlock:^(NSMutableArray* pendingTasks) {
298 // No signpost, take all tasks.
299 tasks = [pendingTasks copy];
300 [pendingTasks removeAllObjects];
303 if (tasks.count > 0) {
304 for (JSTEvaluatorTask* task in tasks) {
305 dispatch_block_t block = ^{
306 NSError* error = nil;
307 if (task.evaluateBlock) {
308 [self _setupEvaluatorThreadContextIfNeeded];
309 task.evaluateBlock(_jsContext);
310 if (_jsContext.exception) {
311 NSLog(@"Did fail on JSContext: %@", _jsContext.name);
312 NSDictionary* userInfo = @{ NSLocalizedDescriptionKey : [_jsContext.exception[@"message"] toString] };
313 error = [NSError errorWithDomain:@"JSTEvaluator" code:1 userInfo:userInfo];
314 _jsContext.exception = nil;
317 [self _callCompletionHandler:task.completionHandler ifNeededWithError:error];
320 if (task.evaluateBlock)
321 dispatch_async(_jsSourcePerformQueue, block);
323 dispatch_barrier_async(_jsSourcePerformQueue, block);
326 dispatch_barrier_sync(_jsSourcePerformQueue, ^{
327 if ([_jsContext[@"counter"] toInt32] == scriptToEvaluate)
328 dispatch_semaphore_signal(_allScriptsDone);
333 - (void)_sourceCanceledOnRunLoop:(CFRunLoopRef)runLoop
335 UNUSED_PARAM(runLoop);
336 assert([[[NSThread currentThread] name] isEqualToString:@"JSTEval"]);
338 @synchronized(self) {
339 assert(_jsThreadRunLoop);
340 assert(_jsThreadRunLoopSource);
342 CFRunLoopRemoveSource(_jsThreadRunLoop, _jsThreadRunLoopSource, kCFRunLoopDefaultMode);
343 CFRunLoopStop(_jsThreadRunLoop);
349 void runRegress141275()
351 // Test that we can execute the same script from multiple threads with a shared context.
352 // See <https://webkit.org/b/141275>
353 NSLog(@"TEST: Testing multiple threads executing the same script with a shared context");
356 JSTEvaluator* evaluator = [[JSTEvaluator alloc] initWithScript:@"this['counter'] = 0;"];
358 void (^showErrorIfNeeded)(NSError* error) = ^(NSError* error) {
360 dispatch_async(dispatch_get_main_queue(), ^{
361 NSLog(@"Error: %@", error);
366 [evaluator evaluateBlock:^(JSContext* context) {
367 JSSynchronousGarbageCollectForDebugging([context JSGlobalContextRef]);
368 } completion:showErrorIfNeeded];
370 [evaluator evaluateBlock:^(JSContext* context) {
371 context[@"wait"] = ^{
372 [NSThread sleepForTimeInterval:0.01];
374 } completion:^(NSError* error) {
376 dispatch_async(dispatch_get_main_queue(), ^{
377 NSLog(@"Error: %@", error);
380 for (unsigned i = 0; i < scriptToEvaluate; i++)
381 [evaluator evaluateScript:@"this['counter']++; this['wait']();" completion:showErrorIfNeeded];
384 [evaluator waitForTasksDoneAndReportResults];
388 #endif // JSC_OBJC_API_ENABLED