反汇编objc分析__block

"You can specify that an imported variable be mutable—that is, read-write— by applying the __block storage type modifier."文档已经清楚说明了它的作用。反汇编就是要看个究竟。

__block类型有着它自己的storage,是blocks编程的一部分,今天先来看一下它如何做storage的。

我们定义如下:

NSString* c = [@"ab" stringByAppendingString:@"c"];
    __block NSInteger bi = 0;
    __block NSString* bc = c;

得到反汇编代码:

0x10699054a <+90>:  movq   0x14bf7(%rip), %rsi       ; "stringByAppendingString:"
    0x106990551 <+97>:  leaq   0x13da8(%rip), %rdi       ; @"ab"
    0x106990558 <+104>: leaq   0x13dc1(%rip), %rdx       ; @"'c'"
    0x10699055f <+111>: movq   -0xf0(%rbp), %rcx
    0x106990566 <+118>: movq   %rax, -0xf8(%rbp)
    0x10699056d <+125>: callq  *%rcx
    0x10699056f <+127>: movq   %rax, -0x28(%rbp)
->  0x106990573 <+131>: movq   $0x0, -0x48(%rbp)
    0x10699057b <+139>: leaq   -0x48(%rbp), %rax
    0x10699057f <+143>: movq   %rax, -0x40(%rbp)
    0x106990583 <+147>: movl   $0x20000000, -0x38(%rbp)
    0x10699058a <+154>: movl   $0x20, -0x34(%rbp)
    0x106990591 <+161>: movq   $0x0, -0x30(%rbp)
    0x106990599 <+169>: movq   $0x0, -0x78(%rbp)
    0x1069905a1 <+177>: leaq   -0x78(%rbp), %rax
    0x1069905a5 <+181>: movq   %rax, -0x70(%rbp)
    0x1069905a9 <+185>: movl   $0x52000000, -0x68(%rbp)
    0x1069905b0 <+192>: movl   $0x30, -0x64(%rbp)
    0x1069905b7 <+199>: leaq   -0x21e(%rip), %rax        ; __Block_byref_object_copy_ at ViewController.mm:74
    0x1069905be <+206>: movq   %rax, -0x60(%rbp)
    0x1069905c2 <+210>: leaq   -0x1e9(%rip), %rax        ; __Block_byref_object_dispose_ at ViewController.mm:74
    0x1069905c9 <+217>: movq   %rax, -0x58(%rbp)
    0x1069905cd <+221>: movq   -0x28(%rbp), %rax
    0x1069905d1 <+225>: movq   %rax, -0x50(%rbp)

很容易就看到两处构造对像的地方,没有错就是__block NSInteger bi和__block NSString* bc。这样就可以写出反c++伪代码:

__Block_byref_object_copy_的汇编简单不贴了,_Block_object_assign就是根据最后一个标志参数选择执行_Block_retain,_Block_assign。此方法是被Block定义(就是NSBlock的一种),在构造私有调用栈时调用的。
另一个__Block_byref_object_dispose_就是__block在析构里调用的。
__block storage的地址被压入NSBlock的私有调用栈,地址不能改变,但是__block storage不在私有栈中,所以被它包装起来的真正的变量可以被修改。
例如NSString*, int在NSBlock的私有调用栈中分别是NSString* const, int count, 而__block NSString*, __block int则是__block<NSString*>* const, __block<int>* const。

上一篇介绍了《逆向分析objc,所有类的信息都能在动态调试中获取》。

最后,多谢大家观看本篇文章。下一篇将分析block的反汇编,还会看到__block变量是如何被block使用的。

更正修改:20150116
    :伪代码中少了一个成员,这个成员作用是__block变量从stack-based中脱离后,真实的指向(,或叫做重定向)。
posted on 2016-01-13 03:36  bbqz007  阅读(1322)  评论(1编辑  收藏  举报