11--多线程探索08--GCD源码之dispatch_once

概述

dispatch_once能保证任务只会被执行一次,即使同时多线程调用也是线程安全的。常用于创建单例、swizzeld method等功能。

static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
    //创建单例、method swizzled或其他任务
});

源码(老)

在10之后的源码中,隐藏了很多实现细节,不利于解读过程,可以先看一份早期的源码:

void dispatch_once_f(dispatch_once_t *val, void *ctxt, dispatch_function_t func) {
    struct _dispatch_once_waiter_s * volatile *vval =
            (struct _dispatch_once_waiter_s**)val;
    struct _dispatch_once_waiter_s dow = { NULL, 0 };
    struct _dispatch_once_waiter_s *tail, *tmp;
    _dispatch_thread_semaphore_t sema;

    if (dispatch_atomic_cmpxchg(vval, NULL, &dow, acquire)) {
        _dispatch_client_callout(ctxt, func);

        dispatch_atomic_maximally_synchronizing_barrier();
        // above assumed to contain release barrier
        tmp = dispatch_atomic_xchg(vval, DISPATCH_ONCE_DONE, relaxed);
        tail = &dow;
        while (tail != tmp) {
            while (!tmp->dow_next) {
                dispatch_hardware_pause();
            }
            sema = tmp->dow_sema;
            tmp = (struct _dispatch_once_waiter_s*)tmp->dow_next;
            _dispatch_thread_semaphore_signal(sema);
        }
    } else {
        dow.dow_sema = _dispatch_get_thread_semaphore();
        tmp = *vval;
        for (;;) {
            if (tmp == DISPATCH_ONCE_DONE) {
                break;
            }
            if (dispatch_atomic_cmpxchgvw(vval, tmp, &dow, &tmp, release)) {
                dow.dow_next = tmp;
                _dispatch_thread_semaphore_wait(dow.dow_sema);
                break;
            }
        }
        _dispatch_put_thread_semaphore(dow.dow_sema);
    }
}

构造的链表,这个结构体在最新的源码中是找不到的

struct _dispatch_once_waiter_s {
  truevolatile struct _dispatch_once_waiter_s *volatile dow_next;
  true_dispatch_thread_semaphore_t dow_sema;
};

block执行的源码,这段代码后面也会分析

_dispatch_client_callout(ctxt, func);

我们都知道dispatch_once的魅力在于它可以保证代码块只执行一次,且是线程安全的

image.png

场景一:第一个线程第一次进入

  1. 刚进来时:dispatch_once_t==nilif (dispatch_atomic_cmpxchg(vval, NULL, &dow, acquire))条件成立,进入if流程;
  2. 调用:_dispatch_client_callout,执行block;
  3. 调用:dispatch_atomic_xchg(vval, DISPATCH_ONCE_DONE, relaxed);,将vval的值更新成DISPATCH_ONCE_DONE表示任务已完成;
  4. 遍历链表的节点并调用_dispatch_thread_semaphore_signal来唤醒等待中的信号量;

场景二:第二个线程进入

在第一个线程结束之前,第二个线程就已经进来了,但因为dispatch_atomic_cmpxchg(vval, NULL, &dow, acquire)里面的流程都是原子操作,原子表示最小的操作,不可分割的操作,所以其他进来的线程需要进入else流程

  1. if (dispatch_atomic_cmpxchg(vval, NULL, &dow, acquire))条件不成立,进入else流程
  2. 获取当前线程信号量:dow.dow_sema = _dispatch_get_thread_semaphore();
  3. 开始无限循环:for (;;)
  4. 阻塞线程:_dispatch_thread_semaphore_wait(dow.dow_sema);
  5. 循环退出条件:if (tmp == DISPATCH_ONCE_DONE) { break; },在场景一中有最后一步,当任务完成时,会vval的值更新成DISPATCH_ONCE_DONE,然后遍历链表中的所有节点,并并调用_dispatch_thread_semaphore_signal来唤醒等待中的信号量,这个时候,循环就会退出。
  6. 更新当前线程信号量:_dispatch_put_thread_semaphore(dow.dow_sema)

场景三:同一个线程第二次进入

当场景一完成之后,vval的值为DISPATCH_ONCE_DONE,这个时候if (dispatch_atomic_cmpxchg(vval, NULL, &dow, acquire))条件就不成立,同样会走到else流程

  • 场景一已完成,走进for循环,然后立马跳出循环,更新线程信号量,啥都没做;
  • 场景一未完成,跟场景二的流程一致,进入for循环,直到vval的值更新成DISPATCH_ONCE_DONE

源码(新)

在新的源码中看不到信号量的作用了。新的源码中用了更多的宏,流程更加抽象。但主体思路变化不大。

源码实现

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);
    if (likely(v == DLOCK_ONCE_DONE)) {
        return;
    }
#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);
}
  1. 先校验once状态是否为DLOCK_ONCE_DONE,如果是,则直接返回,啥也不做,否则进入第二步;
  2. 条件判断:_dispatch_once_gate_tryenter(l),其实内部实现也是跟旧的源码差别不大:os_atomic_cmpxchg(&l->dgo_once, DLOCK_ONCE_UNLOCKED, (uintptr_t)_dispatch_lock_value_for_self(), relaxed);
  3. 条件通过则调用:_dispatch_once_callout(l, ctxt, func);,这个流程跟旧源码的if流程一模一样。先调用block,然后广播信号量状态改变的消息;
  4. 其他条件不通过时,则进入:_dispatch_once_wait(l);流程,这个流程和else流程一致;

源码定义

为了更加清晰的认识新的源码,将宏定义展开、函数嵌套展开之后看过程。整体的流程仍然是完全一致,只是将信号量改成了iOS10之后的unfair_lock

void
dispatch_once_f(dispatch_once_t *val, void *ctxt, dispatch_function_t func)
{
    dispatch_once_gate_t l = (dispatch_once_gate_t)val;

    uintptr_t v = os_atomic_load(&l->dgo_once, acquire);
    if (likely(v == DLOCK_ONCE_DONE)) {
        return;
    }
    if (os_atomic_cmpxchg(&l->dgo_once, DLOCK_ONCE_UNLOCKED,
            (uintptr_t)_dispatch_lock_value_for_self(), relaxed)) {
         //_dispatch_once_callout(l, ctxt, func);
         func(ctxt);
         dispatch_lock value_self = _dispatch_lock_value_for_self();
         uintptr_t v;
         v = _dispatch_once_mark_done(l);
         if (likely((dispatch_lock)v == value_self)) return;
         _dispatch_gate_broadcast_slow(&l->dgo_gate, (dispatch_lock)v);
         return;
    }
    
    // _dispatch_once_wait
    dispatch_lock self = _dispatch_lock_value_for_self();
    uintptr_t old_v, new_v;
    uint32_t timeout = 1;

    for (;;) {
        os_atomic_rmw_loop(&dgo->dgo_once, old_v, new_v, relaxed, {
            if (likely(old_v == DLOCK_ONCE_DONE)) {
                os_atomic_rmw_loop_give_up(return);
            }
            new_v = old_v | (uintptr_t)DLOCK_WAITERS_BIT;
            if (new_v == old_v) os_atomic_rmw_loop_give_up(break);
        });
        if (unlikely(_dispatch_lock_is_locked_by((dispatch_lock)old_v, self))) {
            DISPATCH_CLIENT_CRASH(0, "trying to lock recursively");
        }
        _dispatch_unfair_lock_wait(lock, (dispatch_lock)new_v, 0,
                DLOCK_LOCK_NONE);
        (void)timeout;
    }
}

定义

dispatch_once_t

typedef intptr_t dispatch_once_t;
static dispatch_once_t _dispatch_build_pred;

dispatch_once_gate_s, *dispatch_once_gate_t

typedef struct dispatch_once_gate_s {
    union {
        dispatch_gate_s dgo_gate;
        uintptr_t dgo_once;
    };
} dispatch_once_gate_s, *dispatch_once_gate_t;

宽度 acquire = __mo_acquire = 2

uintptr_t v = os_atomic_load(&l->dgo_once, acquire);
    rename everything to the wide variants
    重命名为指定宽度的变量 只需要考虑两位即可
    enum class memory_order : __memory_order_underlying_t {
      relaxed = __mo_relaxed,
      consume = __mo_consume,
      acquire = __mo_acquire,
      release = __mo_release,
      acq_rel = __mo_acq_rel,
      seq_cst = __mo_seq_cst
    };
    enum __legacy_memory_order {
        __mo_relaxed,
        __mo_consume,
        __mo_acquire,
        __mo_release,
        __mo_acq_rel,
        __mo_seq_cst
    };

宏定义

DLOCK_GATE_UNLOCKED
    #define DLOCK_GATE_UNLOCKED ((dispatch_lock)0)
        0
DLOCK_ONCE_UNLOCKED
    #define DLOCK_ONCE_UNLOCKED ((uintptr_t)0)
        0x00
        0b0000 0000
DLOCK_ONCE_DONE
    #define DLOCK_ONCE_DONE     (~(uintptr_t)0)
        0xff
        0b1111 1111
DLOCK_FAILED_TRYLOCK_BIT
    #define DLOCK_FAILED_TRYLOCK_BIT ((dispatch_lock)0x00000002)
        0x02
        0b0000 0010
DISPATCH_NOESCAPE
    #define DISPATCH_NOESCAPE __attribute__((__noescape__))
DISPATCH_EXPECT
    #define DISPATCH_EXPECT(x, v) __builtin_expect((x), (v))

函数定义

DISPATCH_ONCE_MAKE_GEN(gen)
    (((gen) << 2) + DLOCK_FAILED_TRYLOCK_BIT)
        左移两位
        + 0b0000 0010
DISPATCH_ONCE_IS_GEN(gen)
    (((gen) & 3) == DLOCK_FAILED_TRYLOCK_BIT)
        & 0b0000 0011
        比较 0b0000 0010
_dispatch_once_gate_tryenter
    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);
    }
        l->dgo_once == 0

atomic.h

os_atomic_cmpxchg(&l->dgo_once, DLOCK_ONCE_UNLOCKED,
            (uintptr_t)_dispatch_lock_value_for_self(), relaxed);
    比较+交换
    l->dgo_once 与 DLOCK_ONCE_UNLOCKED 比较
    将 _dispatch_lock_value_for_self() 赋值给 l->dgo_once

_dispatch_once_callout流程

_dispatch_client_callout(ctxt, func);
    return f(ctxt);
    调用block

_dispatch_once_gate_broadcast(l);

_dispatch_once_gate_broadcast(dispatch_once_gate_t l)
{
    dispatch_lock value_self = _dispatch_lock_value_for_self();
    uintptr_t v;
#if DISPATCH_ONCE_USE_QUIESCENT_COUNTER
    v = _dispatch_once_mark_quiescing(l);
#else
    v = _dispatch_once_mark_done(l);
#endif
    if (likely((dispatch_lock)v == value_self)) return;
    _dispatch_gate_broadcast_slow(&l->dgo_gate, (dispatch_lock)v);
}

最后两位结果为:最后两位0b10

static inline uintptr_t
_dispatch_once_mark_quiescing(dispatch_once_gate_t dgo)
{
    return os_atomic_xchg(&dgo->dgo_once, _dispatch_once_generation(), release);
}
    static inline uintptr_t
    _dispatch_once_generation(void)
    {
        uintptr_t value;
        value = *(volatile uintptr_t *)_COMM_PAGE_CPU_QUIESCENT_COUNTER;
        return (uintptr_t)DISPATCH_ONCE_MAKE_GEN(value);
    }

最后两位结果为:0b11

static inline uintptr_t
_dispatch_once_mark_done(dispatch_once_gate_t dgo)
{
    return os_atomic_xchg(&dgo->dgo_once, DLOCK_ONCE_DONE, release);
}

深入浅出 GCD 之 dispatch_once
iOS多线程:GCD源码分析<三>dispatch_once
atomic_cmpxchg

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,716评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,558评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,431评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,127评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,511评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,692评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,915评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,664评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,412评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,616评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,105评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,424评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,098评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,096评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,869评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,748评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,641评论 2 271

推荐阅读更多精彩内容