From 708b240ad39647ef0d58884bcfcb05dfc6864198 Mon Sep 17 00:00:00 2001 From: wwwcg Date: Mon, 17 Jun 2024 22:09:52 +0800 Subject: [PATCH] fix(ios): potential thread race in HippyJSExecutor --- .../ios/base/executors/HippyJSExecutor.mm | 37 +++++++++++++------ 1 file changed, 25 insertions(+), 12 deletions(-) diff --git a/framework/ios/base/executors/HippyJSExecutor.mm b/framework/ios/base/executors/HippyJSExecutor.mm index 21f305e39c6..6351b1bea37 100644 --- a/framework/ios/base/executors/HippyJSExecutor.mm +++ b/framework/ios/base/executors/HippyJSExecutor.mm @@ -78,14 +78,16 @@ @interface HippyJSExecutor () { // Set at setUp time: id _contextWrapper; - NSMutableArray *_pendingCalls; __weak HippyBridge *_bridge; #ifdef JS_JSC BOOL _isInspectable; #endif //JS_JSC } -@property(readwrite, assign) BOOL ready; +/// Whether JSExecutor has done setup. +@property (nonatomic, assign) BOOL ready; +/// Pending blocks to be executed on JS queue. +@property (nonatomic, strong) NSMutableArray *pendingCalls;; @end @@ -211,12 +213,19 @@ - (void)setup { strongSelf.contextCreatedBlock(strongSelf->_contextWrapper); } scope->SyncInitialize(); - strongSelf.ready = YES; - NSArray *pendingCalls = [strongSelf->_pendingCalls copy]; + + // execute pending blocks + NSArray *pendingCalls; + @synchronized (strongSelf) { + strongSelf.ready = YES; + pendingCalls = [strongSelf.pendingCalls copy]; + [strongSelf.pendingCalls removeAllObjects]; + } [pendingCalls enumerateObjectsUsingBlock:^(dispatch_block_t _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) { [strongSelf executeBlockOnJavaScriptQueue:obj]; }]; - [strongSelf->_pendingCalls removeAllObjects]; + + // performance record auto entry = scope->GetPerformance()->PerformanceNavigation(hippy::kPerfNavigationHippyInit); entry->SetHippyJsEngineInitStart(startPoint); entry->SetHippyJsEngineInitEnd(footstone::TimePoint::SystemNow()); @@ -245,7 +254,7 @@ - (instancetype)initWithEngineKey:(NSString *)engineKey bridge:(HippyBridge *)br self.bridge = bridge; self.ready = NO; - _pendingCalls = [NSMutableArray arrayWithCapacity:4]; + self.pendingCalls = [NSMutableArray array]; HippyLogInfo(@"[Hippy_OC_Log][Life_Circle],HippyJSCExecutor Init %p, engineKey:%@", self, engineKey); } @@ -595,9 +604,11 @@ static id executeApplicationScript(NSData *script, NSURL *sourceURL, SharedCtxPt } - (void)executeBlockOnJavaScriptQueue:(dispatch_block_t)block { - if (!self.ready) { - [_pendingCalls addObject:block]; - return; + @synchronized (self) { + if (!self.ready) { + [self.pendingCalls addObject:block]; + return; + } } auto engine = [[HippyJSEnginesMapper defaultInstance] JSEngineResourceForKey:self.enginekey]->GetEngine(); if (engine) { @@ -611,9 +622,11 @@ - (void)executeBlockOnJavaScriptQueue:(dispatch_block_t)block { } - (void)executeAsyncBlockOnJavaScriptQueue:(dispatch_block_t)block { - if (!self.ready) { - [_pendingCalls addObject:block]; - return; + @synchronized (self) { + if (!self.ready) { + [self.pendingCalls addObject:block]; + return; + } } auto engine = [[HippyJSEnginesMapper defaultInstance] JSEngineResourceForKey:self.enginekey]->GetEngine(); if (engine) {