objc_msgSend源码解析(一)

objc_msgSend源码解析(一)
objc_msgSend源码解析(二)

1. objc_msgSend源码解析

在 objc4源码中全局搜索objc_msgSend可知该方法由汇编实现,在iOS平台下实现是在objc-msg-arm64.s文件中。

ENTRY _objc_msgSend 
//进入objc_msgSend流程

cmp p0, #0          // nil check and tagged pointer check
//p0为第一个参数消息接收者,判断消息接收者是否为nil或者为taggedPointer类型

#if SUPPORT_TAGGED_POINTERS
    b.le    LNilOrTagged        //  (MSB tagged pointer looks negative)
    //如果为taggedPointer 则跳转到LNilOrTagged
#else
    b.eq    LReturnZero
    //如果为nil 则跳转到LReturnZero
#endif

ldr p13, [x0]       // p13 = isa
//取出消息接收者的isa

    GetClassFromIsa_p16 p13     // p16 = class
//获取isa指向的class

LGetIsaDone:
    // calls imp or objc_msgSend_uncached
    CacheLookup NORMAL, _objc_msgSend
//调用CacheLookup的NORMAL模式

#if SUPPORT_TAGGED_POINTERS
LNilOrTagged:
    b.eq    LReturnZero     // nil check
    //taggedPointer作为消息接收者第一步同样也是跳转到LReturnZero

    // tagged 为taggedPointer的特殊处理
    adrp    x10, _objc_debug_taggedpointer_classes@PAGE
    add x10, x10, _objc_debug_taggedpointer_classes@PAGEOFF
    ubfx    x11, x0, #60, #4
    ldr x16, [x10, x11, LSL #3]
    adrp    x10, _OBJC_CLASS_$___NSUnrecognizedTaggedPointer@PAGE
    add x10, x10, _OBJC_CLASS_$___NSUnrecognizedTaggedPointer@PAGEOFF
    cmp x10, x16
    b.ne    LGetIsaDone

    // ext tagged
    adrp    x10, _objc_debug_taggedpointer_ext_classes@PAGE
    add x10, x10, _objc_debug_taggedpointer_ext_classes@PAGEOFF
    ubfx    x11, x0, #52, #8
    ldr x16, [x10, x11, LSL #3]
    b   LGetIsaDone
// SUPPORT_TAGGED_POINTERS
#endif

LReturnZero:
    //消息接收者为nil或者taggedPointer类型都是直接return
    // x0 is already zero
    mov x1, #0
    movi    d0, #0
    movi    d1, #0
    movi    d2, #0
    movi    d3, #0
    ret

    END_ENTRY _objc_msgSend

分析:

  1. p0为第一个参数消息接收者,判断消息接收者是否为nil或者为taggedPointer类型
  2. 如果为nil或者taggedPointer 则跳转到LReturnZero,直接return
  3. 如果不为nil或taggedPointer,则将消息接收者第一个地址的isa取出,赋值给p13
  4. 将p13即消息接收者的isa作为参数,调用GetClassFromIsa_p16,获取isa指向的类或者元类
  5. 然后调用CacheLookup函数的NORMAL模式
objc_msgSend.png

2. GetClassFromIsa_p16源码解析

.macro GetClassFromIsa_p16 /* src */

#if SUPPORT_INDEXED_ISA
    // Indexed isa
    mov p16, $0         // optimistically set dst = src
    tbz p16, #ISA_INDEX_IS_NPI_BIT, 1f  // done if not non-pointer isa
    // isa in p16 is indexed
    adrp    x10, _objc_indexed_classes@PAGE
    add x10, x10, _objc_indexed_classes@PAGEOFF
    ubfx    p16, p16, #ISA_INDEX_SHIFT, #ISA_INDEX_BITS  // extract index
    ldr p16, [x10, p16, UXTP #PTRSHIFT] // load class from array
1:

//在64位下根据传入的第一个参数 即 isa & ISA_MASK 取出class
#elif __LP64__
    // 64-bit packed isa
    and p16, $0, #ISA_MASK

#else
    // 32-bit raw isa
    mov p16, $0

#endif

.endmacro

分析: 在64位下根据传入的第一个参数 即 isa & ISA_MASK 取出class

3. CacheLookup源码解析

#define PTRSHIFT 3  // 1<<PTRSHIFT == PTRSIZE

.macro CacheLookup

LLookupStart$1:

    // p1 = SEL, p16 = isa
    ldr p11, [x16, #CACHE]              // p11 = mask|buckets
//获取isa指向class中的cache_t cache结构中的mask|buckets

//在iOS下走以下分支
#if CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_HIGH_16
//将p11中存的mask|buckets & 0x0000ffffffffffff,获取低48位的为buckets
    and p10, p11, #0x0000ffffffffffff   // p10 = buckets

//将p11中存的mask|buckets 逻辑右移48位,即将buckets顶出去,留下16位的mask;然后将获取到的mask & p1(_cmd),获取到buckets数组中要查找的index,即index = _cmd & mask
    and p12, p1, p11, LSR #48       // x12 = _cmd & mask
#elif CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_LOW_4
    and p10, p11, #~0xf         // p10 = buckets
    and p11, p11, #0xf          // p11 = maskShift
    mov p12, #0xffff
    lsr p11, p12, p11               // p11 = mask = 0xffff >> p11
    and p12, p1, p11                // x12 = _cmd & mask
#else
#error Unsupported cache mask storage for ARM64.
#endif


    add p12, p10, p12, LSL #(1+PTRSHIFT)
                     // p12 = buckets + ((_cmd & mask) << (1+PTRSHIFT))
//(1+PTRSHIFT) = (1+3) = 4,由于一个bucket结构包含 IMP _imp,SEL _sel,即16位,所以((_cmd & mask) << (1+PTRSHIFT))含义为index * 16,buckets + ((_cmd & mask) << (1+PTRSHIFT))含义为获取到index位置bucket的首地址赋值到p12

    ldp p17, p9, [x12]      // {imp, sel} = *bucket
//在iOS下bucket结构为{imp, sel},mac下为{sel, imp}

1:  cmp p9, p1          // if (bucket->sel != _cmd)
//比较bucket的sel和所需查找的_cmd是否相等

    b.ne    2f          //     scan more
//如果不相等 跳转到2

    CacheHit $0         // call or return imp
//如果缓存命中,则直接返回imp
    
2:  // not hit: p12 = not-hit bucket
    CheckMiss $0            // miss if bucket->sel == 0
//首先检查bucket的_sel是否等于0,如果等于0,跳转到CheckMiss

    cmp p12, p10        // wrap if bucket == buckets
//判断获取的bucket是否是buckets中的第一个

    b.eq    3f
//如果遍历到数组的首个元素,则跳转到3,之后不会返回来继续,否则形成死循环

    ldp p17, p9, [x12, #-BUCKET_SIZE]!  // {imp, sel} = *--bucket
//如果不是数组首个元素,则取得前一个bucket的imp和sel,继续进行比较

    b   1b          // loop

3:  // wrap: p12 = first bucket, w11 = mask
#if CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_HIGH_16
    add p12, p12, p11, LSR #(48 - (1+PTRSHIFT))
                    // p12 = buckets + (mask << 1+PTRSHIFT)
//mask为数组个数capacity减一,(mask << 1+PTRSHIFT)为mask * 16,即获取数组最后一个bucket赋值给p12

#elif CACHE_MASK_STORAGE == CACHE_MASK_STORAGE_LOW_4
    add p12, p12, p11, LSL #(1+PTRSHIFT)
                    // p12 = buckets + (mask << 1+PTRSHIFT)
#else
#error Unsupported cache mask storage for ARM64.
#endif

    // Clone scanning loop to miss instead of hang when cache is corrupt.
    // The slow path may detect any corruption and halt later.

    ldp p17, p9, [x12]      // {imp, sel} = *bucket
1:  cmp p9, p1          // if (bucket->sel != _cmd)
    b.ne    2f          //     scan more
    CacheHit $0         // call or return imp
    
2:  // not hit: p12 = not-hit bucket
    CheckMiss $0            // miss if bucket->sel == 0
    cmp p12, p10        // wrap if bucket == buckets
    b.eq    3f
    ldp p17, p9, [x12, #-BUCKET_SIZE]!  // {imp, sel} = *--bucket
    b   1b          // loop

LLookupEnd$1:
LLookupRecover$1:
3:  // double wrap
    JumpMiss $0

.endmacro

分析:

  1. 找到isa指向的类,偏移16字节后取到cache_t cache元素的首地址,首地址在iOS下为_maskAndBuckets赋值到p11
  2. p11 & 0x0000ffffffffffff,得到低48位的buckets
  3. p11逻辑右移48位,得到mask,mask & _cmd得到数组buckets起始查找位置的index
  4. index << 4,即index * 16字节,获取到index所指向bucket的首地址p12
  5. p17, p9 分别为imp,sel
    ///第一步
  6. 比较p9和p1(即传入的第二个参数,所要查找的SEL _cmp)
  7. 如果不想等,则跳转到2f,如果相等则缓存命中,返回sel所对应的函数指针地址imp
    ///第二步
  8. 首先检查bucket的sel是否等于0,如果等于0则跳转到CheckMiss
  9. 如果bucket的sel不等于0,判断bucket是否为数组第一个元素,如果是,则跳转到第三步,之后不会走下一步,而是直接走第三步之后的代码,否则形成死循环
  10. 如果bucket不是数组第一个元素,则获取前一个bucket,将新的imp和sel赋值到p17和p9,并跳转到第一步进行循环

///第三步

  1. mask为数组buckets元素个数(capacity-1),mask << 4,即获取数组最后一个元素bucket的首地址,赋值给p12

12.之后的第一步和上面第一步一样,第二步如果走到数组buckets的第一个元素仍然没有找到,则直接跳转到第三步即JumpMiss

cachelookup.png

4. CheckMiss源码分析

.macro CheckMiss
    // miss if bucket->sel == 0
.if $0 == GETIMP
    cbz p9, LGetImpMiss
.elseif $0 == NORMAL
    cbz p9, __objc_msgSend_uncached
.elseif $0 == LOOKUP
    cbz p9, __objc_msgLookup_uncached
.else
.abort oops
.endif
.endmacro

分析:此时为NORMAL模式,则进入__objc_msgSend_uncached流程

5. __objc_msgSend_uncached源码分析

STATIC_ENTRY __objc_msgSend_uncached
    UNWIND __objc_msgSend_uncached, FrameWithNoSaves

    // THIS IS NOT A CALLABLE C FUNCTION
    // Out-of-band p16 is the class to search
    
    MethodTableLookup
    TailCallFunctionPointer x17

    END_ENTRY __objc_msgSend_uncached

分析:此时快速缓存查找没有找到对应imp,调用MethodTableLookup函数中调用_lookUpImpOrForward,进入对应类或元类的methodlist查找流程

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