Skip to content

Commit

Permalink
Add an experiment that makes ASDataController to do everything on mai…
Browse files Browse the repository at this point in the history
…n thread

- Under this experiment, ASDataController will allocate and layout all nodes on the main thread. This helps to avoid deadlocks that would otherwise occur if some of the node allocations or layouts caused ASDataController's backgroud queue to block on main thread. As a bonus, this experiment also helps to measure how much perf wins we get from doing the work off main.
  • Loading branch information
nguyenhuy committed Sep 17, 2020
1 parent 35093fb commit c70d7b3
Show file tree
Hide file tree
Showing 4 changed files with 33 additions and 6 deletions.
1 change: 1 addition & 0 deletions Source/ASExperimentalFeatures.h
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ typedef NS_OPTIONS(NSUInteger, ASExperimentalFeatures) {
ASExperimentalDrawingGlobal = 1 << 8, // exp_drawing_global
ASExperimentalOptimizeDataControllerPipeline = 1 << 9, // exp_optimize_data_controller_pipeline
ASExperimentalDisableGlobalTextkitLock = 1 << 10, // exp_disable_global_textkit_lock
ASExperimentalMainThreadOnlyDataController = 1 << 11, // exp_main_thread_only_data_controller
ASExperimentalFeatureAll = 0xFFFFFFFF
};

Expand Down
3 changes: 2 additions & 1 deletion Source/ASExperimentalFeatures.mm
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,8 @@
@"exp_dispatch_apply",
@"exp_drawing_global",
@"exp_optimize_data_controller_pipeline",
@"exp_disable_global_textkit_lock"]));
@"exp_disable_global_textkit_lock",
@"exp_main_thread_only_data_controller"]));
if (flags == ASExperimentalFeatureAll) {
return allNames;
}
Expand Down
33 changes: 28 additions & 5 deletions Source/Details/ASDataController.mm
Original file line number Diff line number Diff line change
Expand Up @@ -622,12 +622,9 @@ - (void)updateWithChangeSet:(_ASHierarchyChangeSet *)changeSet
os_log_debug(ASCollectionLog(), "New content: %@", newMap.smallDescription);

Class<ASDataControllerLayoutDelegate> layoutDelegateClass = [self.layoutDelegate class];
++_editingTransactionGroupCount;
dispatch_group_async(_editingTransactionGroup, _editingTransactionQueue, ^{
__block __unused os_activity_scope_state_s preparationScope = {}; // unused if deployment target < iOS10
as_activity_scope_enter(as_activity_create("Prepare nodes for collection update", AS_ACTIVITY_CURRENT, OS_ACTIVITY_FLAG_DEFAULT), &preparationScope);

// Step 3: Call the layout delegate if possible. Otherwise, allocate and layout all elements
// Step 3: Call the layout delegate if possible. Otherwise, allocate and layout all elements
dispatch_block_t step3 = ^{
if (canDelegate) {
[layoutDelegateClass calculateLayoutWithContext:layoutContext];
} else {
Expand All @@ -644,6 +641,32 @@ - (void)updateWithChangeSet:(_ASHierarchyChangeSet *)changeSet
}
[self _allocateNodesFromElements:elementsToProcess];
}
};

// Step 3 can be done on the main thread or on _editingTransactionQueue
// depending on an experiment.
BOOL mainThreadOnly = ASActivateExperimentalFeature(ASExperimentalMainThreadOnlyDataController);
if (mainThreadOnly) {
// Doing it on main means _editingTransactionQueue will be empty all the time
// and blocking on it should be super quick/no-op.
//
// We'll still dispatch to _editingTransactionQueue only to schedule a block
// to _mainSerialQueue to execute next steps. This is not optimized because
// in theory we can skip _editingTransactionQueue entirely, but it's much safer
// because change sets will still flow through the pipeline in pretty the same way
// (main thread -> _editingTransactionQueue -> _mainSerialQueue) and so
// any methods that block on _editingTransactionQueue will still work.
step3();
}

++_editingTransactionGroupCount;
dispatch_group_async(_editingTransactionGroup, _editingTransactionQueue, ^{
__block __unused os_activity_scope_state_s preparationScope = {}; // unused if deployment target < iOS10
as_activity_scope_enter(as_activity_create("Prepare nodes for collection update", AS_ACTIVITY_CURRENT, OS_ACTIVITY_FLAG_DEFAULT), &preparationScope);

if (!mainThreadOnly) {
step3();
}

// Step 4: Inform the delegate on main thread
[self->_mainSerialQueue performBlockOnMainThread:^{
Expand Down
2 changes: 2 additions & 0 deletions Tests/ASConfigurationTests.mm
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,8 @@ + (NSArray *)names {
@"exp_dispatch_apply",
@"exp_drawing_global",
@"exp_optimize_data_controller_pipeline",
@"exp_disable_global_textkit_lock",
@"exp_main_thread_only_data_controller"
];
}

Expand Down

0 comments on commit c70d7b3

Please sign in to comment.