IOS underlying principles + reverse article summary

This paper analyzes the underlying principles of queue creation, synchronous/asynchronous functions, singleton, semaphore and scheduling groups

The queue to create

In the previous article iOS- Underlying Principle 26: Functions and Queues in the GCD, we learned about queues and functions. We know that queues are created using the dispatch_queue_create method in the GCD. Now we explore how queues are created in libdispatch.dylib (download link).

Underlying source analysis

  • Search in the source codedispatch_queue_create
dispatch_queue_t
dispatch_queue_create(const char *label, dispatch_queue_attr_t attr)
{
	return _dispatch_lane_create_with_target(label, attr, DISPATCH_TARGET_QUEUE_DEFAULT, true);
}
Copy the code
  • Enter the_dispatch_lane_create_with_target(
DISPATCH_NOINLINE static dispatch_queue_t _dispatch_lane_create_with_target(const char *label, dispatch_queue_attr_t dqa, dispatch_queue_t tq, Bool Legacy) {// dqai create -dispatch_queue_attr_info_t dqai = _dispatch_queue_attr_to_info(dQA); // Step 1: Normalize parameters, such as qos, overcommit, tq... // Concatenation queue name const void *vtable; dispatch_queue_flags_t dqf = legacy ? DQF_MUTABLE : 0; If (dqai.dqai_concurrent) {// Vtable indicates the type of the class // OS_dispatch_queue_concurrent Vtable = DISPATCH_VTABLE(queue_concurrent); } else { vtable = DISPATCH_VTABLE(queue_serial); }... // Create a queue and initialize dispatch_lane_t dq = _dispatch_object_alloc(vtable, sizeof(struct dispatch_lane_s)); // alloc // The value of dqai.dqai_concurrent determines whether the queue is serial or concurrent. _dispatch_queuE_init (dq, DQF, dqai.dqai_concurrent? DISPATCH_QUEUE_WIDTH_MAX : 1, DISPATCH_QUEUE_ROLE_INNER | (dqai.dqai_inactive ? DISPATCH_QUEUE_INACTIVE : 0)); Dq ->dq_label = label; Dq ->dq_priority = _dispatch_priority_make((dispatch_qOS_t)dqai.dqai_qos, dqai.dqai_relpri); // Priority processing... Dq ->do_targetq = tq; dq->do_targetq = tq; _dispatch_object_debug(dq, "%s", __func__); return _dispatch_trace_queue_create(dq)._dq; // Dq}Copy the code

_dispatch_lane_create_with_target analysis

  • Step 1: Pass_dispatch_queue_attr_to_infoMethods the incomingdqa(that is, queue type,Serial and concurrentEtc.) to createdispatch_queue_attr_info_tObject of typedqaiforStores attributes of queues

  • [Step 2] Set the attributes associated with the queue, such as quality of service (qos)

  • [Step 3] Splicing the queue name (vtable) with DISPATCH_VTABLE. DISPATCH_VTABLE is a macro definition, as shown below. Therefore, the queue type is spliced with OS_dispatch_+ queue type queue_concurrent

    • serialQueue type:OS_dispatch_queue_serial, verify as follows

    • concurrentQueue type:OS_dispatch_queue_concurrent, verify as follows

#define DISPATCH_VTABLE(name) DISPATCH_OBJC_CLASS(name) 👇 (&dispatch_class_symbol (name)) 👇 #define DISPATCH_CLASS(name) OS_dispatch_##nameCopy the code
  • [Step 4] Initialize the queue by alloc+init, namely DQ. In the _dispatch_queue_init parameter, the Boolean value of dqai.dqai_concurrent can be used to determine whether the queue is serial or concurrent, and vtable indicates the type of the queue. A queue is also an object

    • Enter the_dispatch_object_alloc -> _os_object_alloc_realizedMethod is set to isa, from here you can verifyQueues are objectsthe

    • Enter the_dispatch_queue_initMethod, the queue type isdispatch_queue_tAnd set the properties of the queue

  • [Step 5] The created queue is processed by _dispatch_trace_queuE_CREATE, where _dispatch_trace_queuE_CREATE is the macro definition encapsulated by _dispatch_introspection_queue_CREATE. The processed _dq is returned

– to enter_dispatch_introspection_queue_create_hook -> dispatch_introspection_queue_get_info -> _dispatch_introspection_lane_get_infoAs you can see, there is a difference between our custom classes,Create a queueThe implementation at the bottom isCreate through a templatethe

conclusion

  • Parameter two in the queue creation method dispatch_queue_CREATE, that is, the queue type, determines the underlying Max & 1 (used to distinguish between serial and concurrent), where 1 means serial

  • A queue is also an object that needs to be created by alloc + init, and there is also a class in alloc that is concatenated by macro definitions and that specifies isa points to

  • The underlying process of creating a queue is created using a template of type dispatch_introspection_queue_s structure

dispatch_queue_createThe following figure shows the underlying analysis process

Function underlying principle analysis

It mainly analyzes the asynchronous function dispatch_async and synchronous function dispatch_sync

An asynchronous function

Enter dispatch_async source code implementation, the main analysis of two functions

  • _dispatch_continuation_init: Task wrapper function

  • _dispatch_continuation_async: concurrency processing function

void dispatch_async(dispatch_queue_t dq, Dispatch_block_t work)// Work task {dispatch_continuation_t dc = _dispatch_continuation_alloc(); uintptr_t dc_flags = DC_FLAG_CONSUME; dispatch_qos_t qos; // Save block qos = _dispatch_continuation_init(dc, dq, work, 0) dc_flags); Continuation_async (dq, dc, qos, dc->dc_flags); }Copy the code

_dispatch_continuation_init Task wrapper

Enter the _dispatch_continuation_init source code implementation, mainly packaging tasks, and set the thread of the return function, equivalent to initialization

DISPATCH_ALWAYS_INLINE static inline dispatch_qos_t _dispatch_continuation_init(dispatch_continuation_t dc, dispatch_queue_class_t dqu, dispatch_block_t work, dispatch_block_flags_t flags, uintptr_t dc_flags) { void *ctxt = _dispatch_Block_copy(work); / / copy task dc_flags | = DC_FLAG_BLOCK | DC_FLAG_ALLOCATED; if (unlikely(_dispatch_block_has_private_data(work))) { dc->dc_flags = dc_flags; dc->dc_ctxt = ctxt; // will initialize all fields but requires dc_flags & dc_ctxt to be set return _dispatch_continuation_init_slow(dc,)  dqu, flags); } dispatch_function_t func = _dispatch_Block_invoke(work); If (dc_flags & DC_FLAG_CONSUME) {func = _dispatch_call_block_and_release; } return _dispatch_continuation_init_f(dc, dqu, CTXT, func, flags, dc_flags); }Copy the code

There are mainly the following steps

  • Copy tasks using the _dispatch_Block_copy command

  • The task is encapsulated with _dispatch_Block_invoke, where _dispatch_Block_invoke is a macro definition, which is an asynchronous callback according to the above analysis

#define _dispatch_Block_invoke(bb) \
		((dispatch_function_t)((struct Block_layout *)bb)->invoke)
Copy the code
  • If synchronous, the callback is assigned to _dispatch_call_block_and_release

  • The callback function is assigned via the _dispatch_continuation_init_f method, where f is func, and stored in the property

_dispatch_continuation_async Concurrent processing

In this function, the main function is to perform the block callback

  • Enter the_dispatch_continuation_asyncSource code implementation of
DISPATCH_ALWAYS_INLINE static inline void _dispatch_continuation_async(dispatch_queue_class_t dqu, dispatch_continuation_t dc, dispatch_qos_t qos, uintptr_t dc_flags) { #if DISPATCH_INTROSPECTION if (! (dc_flags & DC_FLAG_NO_INTROSPECTION)) { _dispatch_trace_item_push(dqu, dc); // trace log} #else (void)dc_flags; #endif return dx_push(dqu._dq, dc, qos); Dx_invoke ();}Copy the code
  • The key code here isdx_push(dqu._dq, dc, qos).dx_pushIs a macro definition, as shown below
#define dx_push(x, y, z) dx_vtable(x)->dq_push(x, y, z)
Copy the code
  • And one of thedq_pushDifferent functions need to be executed depending on the type of queue

Symbol breakpoint debugging execution function

  • Run Demo and passSymbol breakpoint, to determine which function is executed, because it is a concurrent queue, by increasing_dispatch_lane_concurrent_pushBreak point, let’s see if we get there
dispatch_queue_t conque = dispatch_queue_create("com.CJL.Queue", DISPATCH_QUEUE_CONCURRENT); Dispatch_async (conque, ^{NSLog(@" async ")); });Copy the code
  • The run found that the walk was indeed_dispatch_lane_concurrent_push

  • Enter the_dispatch_lane_concurrent_pushSource code, found to have two steps, continue through symbolic breakpoints_dispatch_continuation_redirect_pushand_dispatch_lane_pushDebugging, found out that the_dispatch_continuation_redirect_push

  • Enter the_dispatch_continuation_redirect_pushSource code, found to go againdx_pushWe can see from the previous queue creation that the queue is also an object, which has a parent class and a root class, so the method of the root class will be recursively executed

  • Next, through the root class_dispatch_root_queue_pushSign break point to verify that the conjecture is correct, and when we run it, it’s absolutely correct

  • Enter the_dispatch_root_queue_push -> _dispatch_root_queue_push_inline ->_dispatch_root_queue_poke -> _dispatch_root_queue_poke_slowSource code, after symbolic breakpoint verification, is really go here, view the method of the source code implementation, there are two main operations
    • The callbacks are registered with the _dispatch_root_queues_init method

    • The thread is created through a do-while loop, using the pthread_create method

DISPATCH_NOINLINE static void _dispatch_root_queue_poke_slow(dispatch_queue_global_t dq, int n, int floor) { int remaining = n; int r = ENOSYS; _dispatch_root_queues_init(); / / the key... // The do-while loop creates a thread do {_dispatch_retain(dq); // released in _dispatch_worker_thread while ((r = pthread_create(pthr, attr, _dispatch_worker_thread, dq))) { if (r ! = EAGAIN) { (void)dispatch_assume_zero(r); } _dispatch_temporary_resource_shortage(); } } while (--remaining); . }Copy the code
_dispatch_root_queues_init
  • Enter the_dispatch_root_queues_initSource code implementation, discovery is adispatch_once_fSingletons (see the underlying analysis of subsequent singletons, not described here), where the incomingfuncis_dispatch_root_queues_init_once.
DISPATCH_ALWAYS_INLINE
static inline void
_dispatch_root_queues_init(void)
{
	dispatch_once_f(&_dispatch_root_queues_pred, NULL, _dispatch_root_queues_init_once);
}

Copy the code
  • Enter the_dispatch_root_queues_init_onceThe source code of its internal different transaction call handles are_dispatch_worker_thread2

Its block callback executes the call path as follows: _dispatch_root_queues_init_once ->_dispatch_worker_thread2 -> _dispatch_root_queue_drain -> _dispatch_root_queue_drain -> _dispatch_continuation_pop_inline -> _dispatch_continuation_invoke_inline -> _dispatch_client_callout -> dispatch_call_block_and_release

This path can go through the breakpoint,btPrint the stack information

instructions

One thing to note here is that the block callbacks for singletons are different from the block callbacks for asynchronous functions

  • In the singleton, the func in the block callback is _dispatch_Block_invoke(block)

  • In an asynchronous function, the func in the block callback is dispatch_call_block_and_release

conclusion

So, in summary, the underlying analysis of asynchronous functions is as follows

  • [Preparation] : First, copy and encapsulate the asynchronous task and set the callback function func

  • [Block callback] : The underlying layer recurses through DX_push, redirects to the root queue, then creates the thread via pthread_creat, and finally executes the block callback via dx_invoke (note that dX_push and DX_invoke are paired).

The underlying analysis process of asynchronous functions is shown in the figure

Synchronization function

Enter the source implementation of dispatch_sync, whose underlying implementation is realized by the fence function (see below for the underlying analysis of the fence function)

DISPATCH_NOINLINE
void
dispatch_sync(dispatch_queue_t dq, dispatch_block_t work)
{
	uintptr_t dc_flags = DC_FLAG_BLOCK;
	if (unlikely(_dispatch_block_has_private_data(work))) {
		return _dispatch_sync_block_with_privdata(dq, work, dc_flags);
	}
	_dispatch_sync_f(dq, work, _dispatch_Block_invoke(work), dc_flags);
}
Copy the code
  • Enter the_dispatch_sync_fThe source code

  • To view_dispatch_sync_f_inlineSource code, wherewidth = 1Said isSerial queues, there are two key points:
    • Fence: _dispatch_barrier_sync_f (which can be explained by the fence function underlying analysis below), it can be concluded that the underlying implementation of the synchronization function is actually the synchronous fence function

    • Deadlock: _dispatch_sync_f_slow, deadlock if there is a mutual wait

DISPATCH_ALWAYS_INLINE static inline void _dispatch_sync_f_inline(dispatch_queue_t dq, void *ctxt, dispatch_function_t func, Uintptr_t dc_flags) {if (likely(dq->dq_width == 1)) {return _dispatch_barrier_sync_f dc_flags); } if (dx_metatype(dq)! = _DISPATCH_LANE_TYPE)) { DISPATCH_CLIENT_CRASH(0, "Queue type doesn't support dispatch_sync"); } dispatch_lane_t dl = upcast(dq)._dl; // Global concurrent queues and queues bound to non-dispatch threads // always fall into the slow case, see DISPATCH_ROOT_QUEUE_STATE_INIT_VALUE if (unlikely(! _dispatch_queue_try_reserve_sync_width(dl))) { return _dispatch_sync_f_slow(dl, ctxt, func, 0, dl, dc_flags); } if (dq-> do_targetQ -> do_targetQ) {return _dispatch_sync_recurse(dl, CTXT, func, dc_flags); } _dispatch_introspection_sync_begin(dl); // Process the current information _dispatch_sync_invoke_and_complete(dl, CTXT, func DISPATCH_TRACE_ARG(_dispatch_trace_ITEM_sync_push_pop (dq, ctxt, func, dc_flags))); // Block execute and release}Copy the code

_dispatch_sync_f_slow deadlock

  • Enter the_dispatch_sync_f_slowThe currentThe home side columnisHang, blockthe

  • To add a task to a queue, push it to the main queue, enter_dispatch_trace_item_push

  • Enter the__DISPATCH_WAIT_FOR_QUEUE__To determine whether dq is a waiting queue, and then give a state, and thenThe status of the DQ matches the queue on which the current task depends

  • Enter the_dq_state_drain_locked_by -> _dispatch_lock_is_locked_byThe source code
DISPATCH_ALWAYS_INLINE static inline bool _dispatch_lock_is_locked_by(dispatch_lock lock_value, Equivalent to equivalent or equivalent to dispatch_lock_owner(lock_value) == tid The same is 0, the different is 1, if the same, it is 0, 0 & any number is 0 // that is to determine whether the current waiting task and the executing task are the same, Return ((lock_value ^ tid) &dlock_owner_mask) == 0; }Copy the code

If you are waiting on the same queue as you are executing, this determines whether the thread ids are multiplied. If they are equal, a deadlock occurs

Reason for sequential execution of synchronous functions + concurrent queues

In the _dispatch_sync_invoke_and_complete -> _dispatch_sync_function_invoke_inline source code, there are three main steps:

  • To queue a task:_dispatch_thread_frame_push
  • Block callback for executing a task:_dispatch_client_callout
  • To send out a mission:_dispatch_thread_frame_pop

From the implementation, we can see that the task is pushed into the queue, then the block callback is executed, and then the task is pop, so the task is executed sequentially.

conclusion

The underlying implementation of the synchronization function is as follows:

  • The underlying implementation of the synchronization function is actually the synchronization fence function

  • In the synchronization function, if the queue that is currently executing and the queue that is waiting are the same, the situation of mutual waiting will result in deadlock

Therefore, to sum up, the underlying implementation process of the synchronization function is shown in the figure

The singleton

In daily development, we typically use GCD’s dispatch_once to create singletons, as shown below

static dispatch_once_t onceToken; Dispatch_once (&onceToken, ^{NSLog(@" singleton ")); });Copy the code

First, there are two things we need to know about singletons

  • [Reasons for executing one time]The singleton process is only executed once, how is the underlying control, that is, why can only be executed once?
  • 【 Block call timing 】When is the block of a singleton called?

With the following two questions, we will analyze the underlying singletons

  • Enter thedispatch_onceSource code implementation, the bottom is throughdispatch_once_fImplementation of the
    • Parameter 1: onceToken, which is a static variable. Because static variables are defined differently in different locations, static variables are unique

    • Parameter 2: block callback

void
dispatch_once(dispatch_once_t *val, dispatch_block_t block)
{
	dispatch_once_f(val, block, _dispatch_Block_invoke(block));
}
Copy the code
  • Enter thedispatch_once_fSource code, whichvalIt’s coming in from outsideonceTokenStatic variables, andfuncis_dispatch_Block_invoke(block), where the underlying singletons are divided into the following steps
    • Convert val, which is a static variable, to l of type dispatch_once_gate_t

    • Run the os_atomIC_LOAD command to obtain the task identifier v

      • If v is equal to DLOCK_ONCE_DONE, it indicates that the task has been executed and returns directly

      • If the locking fails after the task is executed, go to the _dispatch_once_mark_done_if_quiesced function and store again, setting the identifier to DLOCK_ONCE_DONE

      • Otherwise, it attempts to enter the task with _dispatch_once_gate_tryEnter, that is, to unlock it, and then executes the _dispatch_once_callout to execute the block callback

    • If a task is executing and a second one comes in, the _dispatch_once_wait function is used to put task 2 on an infinite wait

DISPATCH_NOINLINE void dispatch_once_f(dispatch_once_t *val, void *ctxt, dispatch_function_t func) { dispatch_once_gate_t l = (dispatch_once_gate_t)val; #if ! DISPATCH_ONCE_INLINE_FASTPATH || DISPATCH_ONCE_USE_QUIESCENT_COUNTER uintptr_t v = os_atomic_load(&l->dgo_once, acquire); //load if (v == DLOCK_ONCE_DONE)) { } #if DISPATCH_ONCE_USE_QUIESCENT_COUNTER if (likely(DISPATCH_ONCE_IS_GEN(v))) { return _dispatch_once_mark_done_if_quiesced(l, v); } #endif #endif if (_dispatch_once_gate_tryenter(l)) {return _dispatch_once_callout(l, CTXT, func); } return _dispatch_once_wait(l); // Wait an infinite number of times}Copy the code

_dispatch_once_gate_tryenter unlock

In the source code, compare the underlying OS_atomic_CMPxCHG method. If there is no problem with the comparison, the task is locked, that is, the identifier of the task is set as DLOCK_ONCE_UNLOCKED

DISPATCH_ALWAYS_INLINE static inline bool _dispatch_once_gate_tryenter(dispatch_once_gate_t l) { return os_atomic_cmpxchg(&l->dgo_once, DLOCK_ONCE_UNLOCKED, (uintptr_t)_dispatch_lock_value_for_self(), relaxed); // First compare, then change}Copy the code

_dispatch_once_callout callback

Enter the _dispatch_once_callout source code, there are two main steps

  • _dispatch_client_callout: block callback execution

  • _dispatch_once_gate_broadcast: indicates that broadcast is performed

DISPATCH_NOINLINE static void _dispatch_once_callout(dispatch_once_gate_t l, void *ctxt, dispatch_function_t func) { _dispatch_client_callout(ctxt, func); // The block call executes _dispatch_once_gate_broadcast(l); // Broadcast: tell others that they have a home, don't look for meCopy the code
  • Enter the_dispatch_client_calloutSource code, the main is to perform the block callback, whichfIs equal to the_dispatch_Block_invoke(block), that is, an asynchronous callback
#undef _dispatch_client_callout void _dispatch_client_callout(void *ctxt, dispatch_function_t f) { @try { return f(ctxt); } @catch (...) { objc_terminate(); }}Copy the code
  • Enter the_dispatch_once_gate_broadcast -> _dispatch_once_mark_doneSource code, is mainly todgo->dgo_onceValue, and then set the task identifier toDLOCK_ONCE_DONE, i.e.,unlock
Uintptr_t _dispatch_once_mark_done(dispatch_once_gate_t dgo) { Return os_atomic_xchg(&dgo-> dGO_once, DLOCK_ONCE_DONE, release); }Copy the code

conclusion

For the bottom implementation of singleton, the main description is as follows:

  • Singletons are executed only once: In the GCD singleton, there are two important parameters, onceToken and block. OnceToken is a static variable with uniqueness and is encapsulated as a variable l of type dispatch_once_gate_t at the bottom. L is mainly used to obtain the underlying atomic encapsulation association, that is, variable V, If v is equal to DLOCK_ONCE_DONE, it indicates that the task has been processed once and returns directly

  • [block call timing] : if the task has not been executed, the task will be locked through C++ function comparison, that is, the task state is set to DLOCK_ONCE_UNLOCK, in order to ensure the uniqueness of the current task and prevent multiple definitions in other places. After the lock is executed, the block callback function is executed. After the execution is complete, the current task is unlocked and the current task state is set to DLOCK_ONCE_DONE. The next time you come in, the task will not be executed and will return directly

  • [Multithreading impact] : If another task enters during the execution of the current task, it will enter an infinite wait. The reason is that the current task has acquired the lock, and the lock has been added, and other tasks cannot acquire the lock

The underlying process analysis of the singleton is shown below

Barrier function

There are two main types of fence functions commonly used in GCD

  • The synchronous fence function dispatch_barrier_sync (executed on the main thread) : this is where the previous task has completed, but the synchronous fence function blocks the thread and affects subsequent tasks

  • Asynchronous barrier function dispatch_barrier_async: this function will be used only after the previous task has completed

The most direct function of the fence function is to control the sequence of tasks to be executed synchronously.

In the meantime, there are a couple of things to note about the fence function

  • Barrier functiononlycanControl the same concurrent queue
  • Synchronous fenceWhen you add it to the queue,The current thread will be lockedThe current thread will not open and continue to execute the next line of code until the tasks before the synchronization fence and the synchronization fence task itself have completed.
  • When using the fence function. It makes sense to use a custom queueIf you use theSerial queuesOr the system provides itGlobal concurrent queueOf the fence functionFunctions as a synchronization functionIt doesn’t make any sense

Code debugging

There are four tasks in total, and the first two tasks have a dependency, that is, task 1 is finished, and task 2 is executed, and the fence function can be used

  • Asynchronous fence functions do not block the main thread, asynchronousIt is the queue that is blocked

  • The synchronous fence function willBlock main thread, the synchronization jam is the current thread

conclusion

  • The asynchronous fence function blocks the queue, and must be a custom concurrent queue, does not affect the main thread task execution

  • The synchronization fence function blocks the main thread and affects the execution of other tasks on the main thread

Usage scenarios

In addition to being used when tasks have dependencies, the fence function can also be used for data security

If you do this, it will crashThe reason for the crash is that the data is being retained and released continuously. Before the data is retained, the release has started, which is equivalent to adding an empty data and releasing it

Modify the

  • By adding the fence function
- (void)use041{ dispatch_queue_t concurrentQueue = dispatch_queue_create("com.CJL.Queue", DISPATCH_QUEUE_CONCURRENT); NSMutableArray *array = [NSMutableArray array]; for (int i = 0; i<100000; i++) { dispatch_async(concurrentQueue, ^{ dispatch_barrier_async(concurrentQueue, ^{ [array addObject:[NSString stringWithFormat:@"%d", i]]; }); }); }}Copy the code
  • Use mutex@synchronized (self) {}
- (void)use041{ dispatch_queue_t concurrentQueue = dispatch_queue_create("com.CJL.Queue", DISPATCH_QUEUE_CONCURRENT); NSMutableArray *array = [NSMutableArray array]; for (int i = 0; i<100000; i++) { dispatch_async(concurrentQueue, ^{ @synchronized (self) { [array addObject:[NSString stringWithFormat:@"%d", i]]; }; }); }}Copy the code

Pay attention to

  • If you use a global queue in a fence function, the operation will crash because the system is also using a global concurrent queue, and the fence will intercept the system at the same time, so it will crash

  • If you change your custom concurrent queue to a serial queue, which is itself sequential synchronization, you’re wasting performance by putting a fence on it

  • The fence function blocks only once

Bottom analysis of asynchronous fence function

Enter dispatch_barrier_Async source code implementation, its implementation and dispatch_async similar, here no longer do analysis, interested can explore below

#ifdef __BLOCKS__
void
dispatch_barrier_async(dispatch_queue_t dq, dispatch_block_t work)
{
	dispatch_continuation_t dc = _dispatch_continuation_alloc();
	uintptr_t dc_flags = DC_FLAG_CONSUME | DC_FLAG_BARRIER;
	dispatch_qos_t qos;

	qos = _dispatch_continuation_init(dc, dq, work, 0, dc_flags);
	_dispatch_continuation_async(dq, dc, qos, dc_flags);
}
#endif
Copy the code

Bottom analysis of synchronous fence function

Enter the source code of dispatch_barrier_sync, the implementation is as follows

void
dispatch_barrier_sync(dispatch_queue_t dq, dispatch_block_t work)
{
	uintptr_t dc_flags = DC_FLAG_BARRIER | DC_FLAG_BLOCK;
	if (unlikely(_dispatch_block_has_private_data(work))) {
		return _dispatch_sync_block_with_privdata(dq, work, dc_flags);
	}
	_dispatch_barrier_sync_f(dq, work, _dispatch_Block_invoke(work), dc_flags);
}
Copy the code

_dispatch_barrier_sync_f_inline

Enter the _dispatch_barrier_sync_f -> _dispatch_barrier_sync_F_inline source code

DISPATCH_ALWAYS_INLINE static inline void _dispatch_barrier_sync_f_inline(dispatch_queue_t dq, void *ctxt, dispatch_function_t func, uintptr_t dc_flags) { dispatch_tid tid = _dispatch_tid_self(); // Get the thread id, which is the thread's unique identifier... If (unlikely(!)); _dispatch_queue_try_acquire_barrier_sync(dl, tid)) {return _dispatch_sync_f_slow(dl, CTXT, func, DC_FLAG_BARRIER, dl, / / not recycling DC_FLAG_BARRIER | dc_flags); } // Verify that the target exists. If it does, If (unlikely(dl->do_targetq-> do_targetQ)) {return _dispatch_sync_recurse(dl, CTXT, func, DC_FLAG_BARRIER | dc_flags); } _dispatch_introspection_sync_begin(dl); _dispatch_lane_barrier_sync_invoke_and_complete(dl, ctxt, func DISPATCH_TRACE_ARG(_dispatch_trace_item_sync_push_pop( dq, ctxt, func, dc_flags | DC_FLAG_BARRIER))); / / execution}Copy the code

The source code is mainly divided into the following parts

  • through_dispatch_tid_selfGet thread ID
  • through_dispatch_queue_try_acquire_barrier_syncDetermine thread state

– to enter_dispatch_queue_try_acquire_barrier_sync_and_suspendI’m going to release it here

  • Find the target of the fence function recursively with _dispatch_sync_recurse

  • Forward information is processed with _dispatch_introspection_sync_begin

  • through_dispatch_lane_barrier_sync_invoke_and_completeExecute the block and release it

A semaphore

Semaphores are generally used to synchronize tasks, similar to a mutex. The user can control the maximum number of concurrent GCD as needed. This is generally used

Sem = dispatch_semaphore_create(1); dispatch_semaphore_wait(sem, DISPATCH_TIME_FOREVER); dispatch_semaphore_signal(sem);Copy the code

Let’s look at the underlying principles

Dispatch_semaphore_create create

The underlying implementation of this function is to initialize the semaphore and set the maximum concurrency of the GCD, which must be greater than 0

dispatch_semaphore_t
dispatch_semaphore_create(long value)
{
	dispatch_semaphore_t dsema;

	// If the internal value is negative, then the absolute of the value is
	// equal to the number of waiting threads. Therefore it is bogus to
	// initialize the semaphore with a negative value.
	if (value < 0) {
		return DISPATCH_BAD_INPUT;
	}

	dsema = _dispatch_object_alloc(DISPATCH_VTABLE(semaphore),
			sizeof(struct dispatch_semaphore_s));
	dsema->do_next = DISPATCH_OBJECT_LISTLESS;
	dsema->do_targetq = _dispatch_get_default_queue(false);
	dsema->dsema_value = value;
	_dispatch_sema4_init(&dsema->dsema_sema, _DSEMA4_POLICY_FIFO);
	dsema->dsema_orig = value;
	return dsema;
}
Copy the code

Dispatch_semaphore_wait lock

The function is implemented as follows. The main function is the semaphore dsema by os_atomic_dec2o, which is executed internally by the C++ atomic_fetch_sub_explicit method

  • If the value is greater than or equal to 0, the operation is invalid and the operation succeeds

  • If value equals LONG_MIN, the system throws a crash

  • If value is less than 0, the long wait is entered

long dispatch_semaphore_wait(dispatch_semaphore_t dsema, Long value = os_atomIC_DEC2O (dsema, dsema_value, acquire); If (likely(value >= 0)) {return 0; } return _dispatch_semaphore_wait_slow(dsema, timeout); // Wait a long time}Copy the code

The macro definition for OS_ATOMic_DEC2O is transformed as follows

Os_atomic_inc2o (p, f and m) 👇 os_atomic_sub2o (p, f, 1, m) 👇 _os_atomic_c11_op ((p), (v), m, sub, -) 👇 _os_atomic_c11_op ((p), (v), m, add, +) 👇 ({_os_atomic_basetypeof (p) _v = (v), _r = \ atomic_fetch_##o##_explicit(_os_atomic_c11_atomic(p), _v, \ memory_order_##m); (__typeof__(_r))(_r op _v); })Copy the code

So I’m going to substitute the actual value for

os_atomic_dec2o(dsema, dsema_value, acquire); os_atomic_sub2o(dsema, dsema_value, 1, m) os_atomic_sub(dsema->dsema_value, 1, m) _os_atomic_c11_op(dsema->dsema_value, 1, m, sub, -) _r = atomic_fetch_sub_explicit(dSEMa -> dSEMa_value, 1Copy the code
  • Enter the_dispatch_semaphore_wait_slowWhen value is less than 0, according to the wait eventtimeoutDo different things

Dispatch_semaphore_signal unlock

The source implementation of the function is as follows, its core is also through the os_atomic_inc2o function on the value of ++ operation, os_atomic_inc2o internal is through C++ atomic_fetch_add_explicit

  • If the value is greater than 0, the operation is invalid and the operation succeeds

  • If value equals 0, a long wait is entered

Long dispatch_semaphore_signal(dispatch_semaphore_t dsema) {// long value = os_atomic_inc2O (dsema, dsema_value, release); If (likely(value > 0)) {// return 0; } if (unlikely(value == LONG_MIN)) { DISPATCH_CLIENT_CRASH(value, "Unbalanced call to dispatch_semaphore_signal()"); } return _dispatch_semaphore_signal_slow(dsema); // Enter a long wait}Copy the code

The macro definition for OS_ATOMic_DEC2O is transformed as follows

Os_atomic_inc2o (p, f and m) 👇 os_atomic_add2o (p, f, 1, m) 👇 os_atomic_add (& (p) - > f (v), m) 👇 _os_atomic_c11_op ((p), (v), m, add, +) 👇 ({_os_atomic_basetypeof (p) _v = (v), _r = \ atomic_fetch_ # # # o# _explicit (_os_atomic_c11_atomic (p), _v, \ memory_order_##m); (__typeof__(_r))(_r op _v); })Copy the code

So I’m going to substitute the actual value for

os_atomic_inc2o(dsema, dsema_value, release); os_atomic_add2o(dsema, dsema_value, 1, m) os_atomic_add(&(dsema)->dsema_value, (1), m) _os_atomic_c11_op((dsema->dsema_value), (1), m, add, +) _r = atomic_fetch_add_explicit(dsema->dsema_value, 1), Equivalent to dsema->dsema_value + 1Copy the code

conclusion

  • Dispatch_semaphore_create is basically the initialization limit

  • Dispatch_semaphore_wait performs a lock operation on the value of the semaphore

  • Dispatch_semaphore_signal is a ++ operation that unlocks the value of the semaphore

Therefore, to sum up, the underlying operations of the semaphore correlation function are shown in the figure

Scheduling group

The most direct function of a scheduling group is to control the execution order of tasks

Dispatch_group_create Create a group dispatch_group_async Group task dispatch_group_notify Notify that a group task is completed dispatch_group_Wait Wait time for a group task to be executed // The incoming and outgoing groups are usually used in pairs. Dispatch_group_enter into the group and dispatch_group_leave out of the groupCopy the code

use

Let’s say I have two tasks, and I need to wait for both tasks to complete before I update the UI and make it usableScheduling group

  • [Amendment 1] If thedispatch_group_notifyMove to the front, can you do that?

Yes, but as long as there is an enter-leave pairing, notify will be executed without waiting for both groups to complete. Enter -leave: enter-leave: enter-leave: enter-leave

  • [Modify 2] Add another enter, i.eEnter: wait is 3:2, can notify be executed?

No, it will wait for a leave before the notify is executed

  • [Modify 3] If enter: wait is 2:3, can I execute notify?

It’s going to crash because the enter-leave doesn’t work, it’s going to crash because the async has a delay

Dispatch_group_create create a group

Basically, you create the group and set the property, and the value of the group is 0

  • Enter thedispatch_group_createThe source code
dispatch_group_t
dispatch_group_create(void)
{
	return _dispatch_group_create_with_count(0);
}
Copy the code
  • Enter the_dispatch_group_create_with_countSource code, which is to assign a value to the group object attribute, and return the group object, wherenIs equal to the0
DISPATCH_ALWAYS_INLINE static inline dispatch_group_t _dispatch_group_create_with_count(uint32_t n) { // Create a group object. The type of the object is OS_dispatch_group dispatch_group_t dg = _dispatch_object_alloc(DISPATCH_VTABLE(group), sizeof(struct dispatch_group_s)); Dg ->do_next = DISPATCH_OBJECT_LISTLESS; dg->do_targetq = _dispatch_get_default_queue(false); if (n) { os_atomic_store2o(dg, dg_bits, (uint32_t)-n * DISPATCH_GROUP_VALUE_INTERVAL, relaxed); os_atomic_store2o(dg, do_ref_cnt, 1, relaxed); // <rdar://22318411> } return dg; }Copy the code

Dispatch_group_enter into groups

Enter dispatch_group_Enter source code, os_atomic_sub_orig2O to dg->dg. Bits to make — operation, the value of processing

void dispatch_group_enter(dispatch_group_t dg) { // The value is decremented on a 32bits wide atomic so that the carry // for the 0 -> -1 transition is not propagated to the upper 32bits. uint32_t old_bits = os_atomic_sub_orig2o(dg, Dg_bits, 0 -> -1 DISPATCH_GROUP_VALUE_INTERVAL, acquire); uint32_t old_value = old_bits & DISPATCH_GROUP_VALUE_MASK; If (unlikely(old_value == 0)) {// if old_value _dispatch_retain(dg); // <rdar://problem/22318411>} if (unlikely(old_value == DISPATCH_GROUP_VALUE_MAX)) { DISPATCH_CLIENT_CRASH(old_bits, "Too many nested calls to dispatch_group_enter()"); }}Copy the code

Dispatch_group_leave out group

  • Enter thedispatch_group_leaveThe source code
    • Minus 1 to 0 is minus 1++operation
    • Depending on the state, the do-while loop wakes up the block task
    • If 0 + 1 = 1, enter-leave is unbalanced, that is, leave is called multiple times, crash will occur
void dispatch_group_leave(dispatch_group_t dg) { // The value is incremented on a 64bits wide atomic so that the carry for // the -1 -> 0 transition increments the generation atomically. uint64_t new_state, Old_state = os_atomic_add_orig2O (dg, dg_state, atom increment ++ DISPATCH_GROUP_VALUE_INTERVAL, release); uint32_t old_value = (uint32_t)(old_state & DISPATCH_GROUP_VALUE_MASK); If (old_value == DISPATCH_GROUP_VALUE_1)) {old_state += DISPATCH_GROUP_VALUE_INTERVAL; do { new_state = old_state; if ((old_state & DISPATCH_GROUP_VALUE_MASK) == 0) { new_state &= ~DISPATCH_GROUP_HAS_WAITERS; new_state &= ~DISPATCH_GROUP_HAS_NOTIFS; } else { // If the group was entered again since the atomic_add above, // we can't clear the waiters bit anymore as we don't know for // which generation the waiters are for new_state &= ~DISPATCH_GROUP_HAS_NOTIFS; } if (old_state == new_state) break; } while (unlikely(! os_atomic_cmpxchgv2o(dg, dg_state, old_state, new_state, &old_state, relaxed))); return _dispatch_group_wake(dg, old_state, true); } // leave = 0, 0+1 -> 1; If (unlikely(old_value == 0)) {DISPATCH_CLIENT_CRASH((uintptr_t)old_value, "Unbalanced call to dispatch_group_leave()"); }}Copy the code
  • Enter the_dispatch_group_wakeSource code, do-while loopasynchronousHit, call_dispatch_continuation_asyncperform
DISPATCH_NOINLINE static void _dispatch_group_wake(dispatch_group_t dg, uint64_t dg_state, bool needs_release) { uint16_t refs = needs_release ? 1:0; // <rdar://problem/22318411> if (dg_state & DISPATCH_GROUP_HAS_NOTIFS) { dispatch_continuation_t dc, next_dc, tail; // Snapshot before anything is notified/woken <rdar://problem/8554546> dc = os_mpsc_capture_snapshot(os_mpsc(dg, dg_notify), &tail); do { dispatch_queue_t dsn_queue = (dispatch_queue_t)dc->dc_data; next_dc = os_mpsc_pop_snapshot_head(dc, tail, do_next); _dispatch_continuation_async(dsn_queue, dc, _dispatch_qos_from_pp(dc->dc_priority), dc->dc_flags); // Block task execution _dispatch_release(dsn_queue); } while ((dc = next_dc)); //do-while loop, async task hits refs++; } if (dg_state & DISPATCH_GROUP_HAS_WAITERS) { _dispatch_wake_by_address(&dg->dg_gen); } if (refs) _dispatch_release_n(dg, refs); // Release the reference}Copy the code
  • Enter the_dispatch_continuation_asyncThe source code
DISPATCH_ALWAYS_INLINE static inline void _dispatch_continuation_async(dispatch_queue_class_t dqu, dispatch_continuation_t dc, dispatch_qos_t qos, uintptr_t dc_flags) { #if DISPATCH_INTROSPECTION if (! (dc_flags & DC_FLAG_NO_INTROSPECTION)) { _dispatch_trace_item_push(dqu, dc); // trace log} #else (void)dc_flags; #endif return dx_push(dqu._dq, dc, qos); Dx_invoke ();}Copy the code

This step is consistent with the block callback execution of an asynchronous function and will not be explained here

Dispatch_group_notify notice

  • Enter thedispatch_group_notifySource code, ifold_stateIs equal to the0“Can be released
DISPATCH_ALWAYS_INLINE static inline void _dispatch_group_notify(dispatch_group_t dg, dispatch_queue_t dq, dispatch_continuation_t dsn) { uint64_t old_state, new_state; dispatch_continuation_t prev; dsn->dc_data = dq; _dispatch_retain(dq); // Run the os_atomic_store2O command to obtain the underlying state identifier of dg. State prev = OS_MPSC_PUSH_update_tail (OS_MPSC (dg, dg_notify), DSN, do_next); if (os_mpsc_push_was_empty(prev)) _dispatch_retain(dg); os_mpsc_push_update_prev(os_mpsc(dg, dg_notify), prev, dsn, do_next); if (os_mpsc_push_was_empty(prev)) { os_atomic_rmw_loop2o(dg, dg_state, old_state, new_state, release, { new_state = old_state | DISPATCH_GROUP_HAS_NOTIFS; If ((uint32_t)old_state == 0) { Os_atomic_rmw_loop_give_up ({return _dispatch_group_wake(dg, new_state, false); / / wake up}); }}); }}Copy the code

In addition to leave can be woken up with _dispatch_group_wake, dispatch_group_notify can also be woken up

  • Among themos_mpsc_push_update_tailIs a macro definition used to get the status code of dg
#define os_mpsc_push_update_tail(Q, tail, _o_next) ({ \ os_mpsc_node_type(Q) _tl = (tail); \ os_atomic_store2o(_tl, _o_next, NULL, relaxed); \ os_atomic_xchg(_os_mpsc_tail Q, _tl, release); The \})Copy the code

dispatch_group_async

  • Enter thedispatch_group_asyncSource code, mainlyThe packing taskandAsynchronous processing tasks
#ifdef __BLOCKS__ void dispatch_group_async(dispatch_group_t dg, dispatch_queue_t dq, dispatch_block_t db) { dispatch_continuation_t dc = _dispatch_continuation_alloc(); uintptr_t dc_flags = DC_FLAG_CONSUME | DC_FLAG_GROUP_ASYNC; dispatch_qos_t qos; Qos = _dispatch_continuation_init(dc, dq, db, 0, dc_flags); Continuation_group_async (dg, dq, dc, qos); } #endifCopy the code
  • Enter the_dispatch_continuation_group_asyncSource code, is mainly encapsulateddispatch_group_enterInto a set of operations
DISPATCH_ALWAYS_INLINE static inline void _dispatch_continuation_group_async(dispatch_group_t dg, dispatch_queue_t dq, dispatch_continuation_t dc, dispatch_qos_t qos) { dispatch_group_enter(dg); Dc ->dc_data = dg; _dispatch_continuation_async(dq, dc, qos, dc->dc_flags); // Asynchronous operation}Copy the code
  • Enter the_dispatch_continuation_asyncSource code, the implementation of regular asynchronous functions underlying operations. sinceWith enter, there must be leaveWe guessThe block execution is followed by the implicit execution of leaveTo print stack information through breakpoint debugging

  • search_dispatch_client_calloutThe call in_dispatch_continuation_with_group_invokeIn the
DISPATCH_ALWAYS_INLINE static inline void _dispatch_continuation_with_group_invoke(dispatch_continuation_t dc) { struct dispatch_object_s *dou = dc->dc_data; unsigned long type = dx_type(dou); If (type == DISPATCH_GROUP_TYPE) {// If the dispatch group type is _dispatch_client_callout(dc-> dc_CTxt, dc->dc_func); / / block callback _dispatch_trace_item_complete (dc); dispatch_group_leave((dispatch_group_t)dou); } else {DISPATCH_INTERNAL_CRASH(dx_type(dou), "Unexpected object type"); }Copy the code

Therefore, it is perfectly confirmed that the underlying encapsulation of dispatch_group_async is enter-leave

conclusion

  • Enter -leave as long as the pair, regardless of distance

  • The “dispatch_group_Enter” function operates on the value of the group (i.e. 0 -> -1).

  • “Dispatch_group_leave” is a C++ function that operates on the value of the group (-1 -> 0).

  • The dispatch_group_notify function determines whether the state of the group is equal to 0 and notifies the group when it is equal to 0

  • You can wake up a block task using dispatch_group_leave or dispatch_group_notify

  • Dispatch_group_async is equivalent to enter-leave, and its underlying implementation is enter-leave

To sum up, the underlying analysis process of scheduling group is shown in the figure below

dispatch_source

The paper

Dispatch_source is the underlying data type used to coordinate the processing of specific underlying system events.

The dispatch_source function replaces the asynchronous callback function to handle system-specific events. When configuring a dispatch, you need to specify the events to monitor, the Dispatch queue, and the code (block or function) to handle the events. When an event occurs, the Dispatch Source commits your block or function to the specified queue for execution

The only reason to use Dispatch Source instead of dispatch_async is to take advantage of joins.

The general process of connection is as follows: When one of its functions, dispatch_source_merge_data, is called on any thread, the handle (which can be simply thought of as a block) defined by the Dispatch Source is executed. This process is called Custom Event, user event. Is the type of event that the Dispatch Source supports processing.

Simply put: This event is a signal that you signal yourself by calling the dispatch_source_merge_data function.

A HANDLE is a pointer to a pointer that points to a class or structure that is closely related to the system. There is also a common HANDLE, the HANDLE

  • Instance handle HINSTANCE
  • Bitmap handle HBITMAP
  • Device table handle HDC
  • Icon handle HICON

The characteristics of

  • Its CPU load is very small, jinling does not occupy resources

  • Advantages of connection

use

  • Create a dispatch source
dispatch_source_t source = dispatch_source_create(dispatch_source_type_t type, uintptr_t handle, unsigned long mask, dispatch_queue_t queue)
Copy the code
parameter instructions
type Dispatch Events that can be handled by the source
handle This can be interpreted as a handle, index, or ID. If you want to listen on a process, you need to pass in the process ID
mask You can think of it as a description, give it a more detailed description, so it knows exactly what it’s listening for, right
queue A queue required by the custom source to process all response handles

Dispatch the Source species

There are several types of type

species instructions
DISPATCH_SOURCE_TYPE_DATA_ADD Custom events, variables added
DISPATCH_SOURCE_TYPE_DATA_OR Custom event, variable OR
DISPATCH_SOURCE_TYPE_MACH_SEND MACH port send
DISPATCH_SOURCE_TYPE_MACH_RECV MACH port reception
DISPATCH_SOURCE_TYPE_MEMORYPRESSURE Memory pressure (note: available after iOS8)
DISPATCH_SOURCE_TYPE_PROC Processes listen, for example, when the process exits, when one or more child threads are created, when the process receives UNIX signals
DISPATCH_SOURCE_TYPE_READ IO operations, such as file operations and read responses to socket operations
DISPATCH_SOURCE_TYPE_SIGNAL Responds when receiving a UNIX signal
DISPATCH_SOURCE_TYPE_TIMER The timer
DISPATCH_SOURCE_TYPE_VNODE File status listening, files are deleted, moved, renamed
DISPATCH_SOURCE_TYPE_WRITE IO operations, such as file operations and write responses to socket operations

Note:

  • DISPATCH_SOURCE_TYPE_DATA_ADD

When an event is triggered at a high frequency at the same time, the Dispatch Source will accumulate these responses in the form of ADD and finally process them when the system is idle. If the triggering frequency is scattered, the Dispatch Source will respond to these events separately.

  • DISPATCH_SOURCE_TYPE_DATA_ORAs above, it is a custom event, but it is accumulated as an OR

Commonly used functions

Dispatch_suspend (queue) dispatch_suspend(queue) dispatch_suspend(queue) dispatch_suspend(queue) dispatch_suspend(queue) dispatch_suspend(queue) You cannot pass a value of 0 (the event will not be fired), nor can you pass a negative number. Dispatch_source_merge_data // Set the block that responds to dispatch source events, Run dispatch_source_set_event_handler on the queue specified by the dispatch source to get dispatch_source_get_data The second parameter of dispatch_source_create is called uintptr_t dispatch_source_get_handle(dispatch_source_t source); Dispatch_source_create unsigned long dispatch_source_get_mask(dispatch_source_t source); //// Cancel event handling of the dispatch source -- that is, block is no longer called. If you call dispatch_SUSPEND you just suspend the dispatch source. void dispatch_source_cancel(dispatch_source_t source); Long dispatch_source_testCancel (dispatch_source_t source); // Check whether the dispatch_source_t source is canceled. // Dispatch the block that is called when the source is cancelled. Release related resources void dispatch_source_set_cancel_handler(dispatch_source_t source, dispatch_block_t cancel_handler); // Can be used to set the block to be called when the dispatch source is started and to be released when the call is complete. This function can also be called at any time while the Dispatch source is running. void dispatch_source_set_registration_handler(dispatch_source_t source, dispatch_block_t registration_handler);Copy the code

Usage scenarios

Often used for captcha countdowns, because dispatch_source does not rely on Runloop, but directly interacts with the underlying kernel for higher accuracy.

- (void)use033{// countdown time __block int timeout = 3; Dispatch_queue_t globalQueue = dispatch_get_global_queue(0, 0); // Create timer dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, globalQueue); // Set the error when the timer is triggered once for 1s and the error when the timer is triggered for 0s /* -source dispatches the source-start number to control the time when the timer is triggered for the first time. The parameter type is dispatch_time_t, which is of opaque type and we cannot directly operate on it. We need the dispatch_time and dispatch_walltime functions to create them. In addition, constants DISPATCH_TIME_NOW and DISPATCH_TIME_FOREVER are often useful. - interval Interval - Accuracy of leeway timer triggering */ dispatch_sourCE_set_timer (timer,dispatch_walltime(NULL, 0),1.0*NSEC_PER_SEC, 0); Dispatch_source_set_event_handler (timer, ^{// Countdown is over, Disable if (timeout <= 0) {// Cancel dispatch source dispatch_source_cancel(timer); }else{ timeout--; NSLog(@" countdown - %d", timeout); }); }}); // Start the execution of dispatch source dispatch_resume(timer); }Copy the code

Use dispatch_source in GCD to implement custom countdown buttons

Implementation idea as long as it is inherited from UIButton, and then through the GCD dispatch_source to achieve the countdown button, the following is the demo download address

CJLCountDownButton- CJLCountDownButton