Skip to content

Commit ffc16fc

Browse files
hakonkfacebook-github-bot
authored andcommitted
Fix data races in RCTImageLoader and RCTNetworkTask with shared atomic counters (#45114)
Summary: In order to fix the data races described in #44715, I propose a simple solution by leveraging shared counter functions wherein `std::atomic` is the backing for the integer values. ## Changelog: [iOS] [Fixed] - Implement shared atomic counters and replace static integers in `RCTImageLoader` and `RCTNetworkTask` that were accessed concurrently, which in some cases lead to data races. Pull Request resolved: #45114 Test Plan: Added unit tests for the counters in `RCTSharedCounterTests`. Reviewed By: cipolleschi Differential Revision: D59155076 Pulled By: javache fbshipit-source-id: f73afce6a816ad3226ed8c123cb2ccf4183549a0
1 parent 9833338 commit ffc16fc

File tree

2 files changed

+6
-9
lines changed

2 files changed

+6
-9
lines changed

packages/react-native/Libraries/Image/RCTImageLoader.mm

+2-6
Original file line numberDiff line numberDiff line change
@@ -32,11 +32,7 @@ static NSInteger RCTImageBytesForImage(UIImage *image)
3232
return image.images ? image.images.count * singleImageBytes : singleImageBytes;
3333
}
3434

35-
static uint64_t getNextImageRequestCount(void)
36-
{
37-
static uint64_t requestCounter = 0;
38-
return requestCounter++;
39-
}
35+
static auto currentRequestCount = std::atomic<uint64_t>(0);
4036

4137
static NSError *addResponseHeadersToError(NSError *originalError, NSHTTPURLResponse *response)
4238
{
@@ -510,7 +506,7 @@ - (RCTImageURLLoaderRequest *)_loadImageOrDataWithURLRequest:(NSURLRequest *)req
510506
auto cancelled = std::make_shared<std::atomic<int>>(0);
511507
__block dispatch_block_t cancelLoad = nil;
512508
__block NSLock *cancelLoadLock = [NSLock new];
513-
NSString *requestId = [NSString stringWithFormat:@"%@-%llu", [[NSUUID UUID] UUIDString], getNextImageRequestCount()];
509+
NSString *requestId = [NSString stringWithFormat:@"%@-%llu", [[NSUUID UUID] UUIDString], currentRequestCount++];
514510

515511
void (^completionHandler)(NSError *, id, id, NSURLResponse *) =
516512
^(NSError *error, id imageOrData, id imageMetadata, NSURLResponse *response) {

packages/react-native/Libraries/Network/RCTNetworkTask.mm

+4-3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
* LICENSE file in the root directory of this source tree.
66
*/
77

8+
#import <atomic>
89
#import <mutex>
910

1011
#import <React/RCTLog.h>
@@ -20,6 +21,8 @@ @implementation RCTNetworkTask {
2021
RCTNetworkTask *_selfReference;
2122
}
2223

24+
static auto currentRequestId = std::atomic<NSUInteger>(0);
25+
2326
- (instancetype)initWithRequest:(NSURLRequest *)request
2427
handler:(id<RCTURLRequestHandler>)handler
2528
callbackQueue:(dispatch_queue_t)callbackQueue
@@ -28,10 +31,8 @@ - (instancetype)initWithRequest:(NSURLRequest *)request
2831
RCTAssertParam(handler);
2932
RCTAssertParam(callbackQueue);
3033

31-
static NSUInteger requestID = 0;
32-
3334
if ((self = [super init])) {
34-
_requestID = @(requestID++);
35+
_requestID = @(currentRequestId++);
3536
_request = request;
3637
_handler = handler;
3738
_callbackQueue = callbackQueue;

0 commit comments

Comments
 (0)