函數(shù)調(diào)用堆棧

函數(shù)調(diào)用堆棧

函數(shù)調(diào)用堆棧最常用的是收集crash信息,解決問題用的悼做,這方面網(wǎng)上有很多的資料,也有成熟的第三方哗魂,如Fabric肛走、Bugly等。這里要說的是單純的收集當(dāng)前函數(shù)調(diào)用堆棧录别,用于收集卡頓檢測朽色、無法復(fù)現(xiàn)bug等。

函數(shù)調(diào)用堆棧的收集有3種方式组题,前2種方法比較簡單葫男,直接調(diào)用系統(tǒng)的方法就可以。

第一種方式

//返回的是當(dāng)前堆棧的地址數(shù)組
NSArray<NSNumber *> *arr = [NSThread callStackReturnAddresses];

符號地址

由于 ASLR 的原因崔列,在每次程序啟動(dòng)時(shí)梢褐,會(huì)在指定的進(jìn)程空間上加上一個(gè)隨機(jī)的偏移量(slide),導(dǎo)致我們看到的 stack address(日志中的地址)和 symbol address(符號對應(yīng)的地址)無法對應(yīng)。

隨機(jī)內(nèi)存布局(ASLR)是操作系統(tǒng)為防止緩沖區(qū)溢出攻擊而存在的內(nèi)存保護(hù)機(jī)制盈咳。該機(jī)制通過在程序載入內(nèi)存時(shí)耿眉,將地址進(jìn)行隨機(jī)偏移來實(shí)現(xiàn)。

簡單的說鱼响,symble addressstack addressslide鸣剪;

獲取ASLR

#import <dlfcn.h>
//針對32位和64位處理一下
#if __arm64__ || __x86_64__
#define FILE_BASE_ADDR 0x100000000
#else
#define FILE_BASE_ADDR 0x4000
#endif

Dl_info info;
if (dladdr((const void *)main, &info)){
    int * fbase = (int*)info.dli_fbase;
    printf("ASLR: 0x%lX\n", (unsigned long)(fbase) - FILE_BASE_ADDR);
}

我們可以通過在lldb里執(zhí)行:

image list -o -f

來驗(yàn)證我們所取得的地址的正確性。

解析

解析可以通過2個(gè)命令進(jìn)行热押,dwarfdump西傀、atos斤寇。

dwarfdump

這個(gè)命令顯示的比較詳情桶癣,需要自己更進(jìn)一步解析。

//0x100004260這個(gè)就是減過slide的地址
dwarfdump --arch arm64 --lookup 0x100004260 Backtrace.app.dSYM/

---------------------------------------------------------------
 File: Backtrace.app.dSYM/Contents/Resources/DWARF/Backtrace (arm64)
---------------------------------------------------------------
Looking up address: 0x0000000100004260 in .debug_info... found!

0x000382b7: Compile Unit: length = 0x000001b4  version = 0x0004  abbr_offset = 0x00000000  addr_size = 0x08  (next CU at 0x0003846f)

0x000382c2: TAG_compile_unit [110] *
             AT_producer( "Apple LLVM version 9.0.0 (clang-900.0.39.2)" )
             AT_language( DW_LANG_ObjC )
             AT_name( "..Backtrace/Backtrace/ViewController.m" )
             AT_stmt_list( 0x0000773c )
             AT_comp_dir( "../Backtrace" )
             AT_APPLE_optimized( true )
             AT_APPLE_major_runtime_vers( 0x02 )
             AT_low_pc( 0x0000000100004208 )
             AT_high_pc( 0x00000144 )

0x000383ad:     TAG_subprogram [115] *
                 AT_low_pc( 0x000000010000423c )
                 AT_high_pc( 0x00000070 )
                 AT_frame_base( reg29 )
                 AT_object_pointer( {0x000383c6} )
                 AT_name( "-[ViewController bar]" )
                 AT_decl_file( "../Backtrace/ViewController.m" )
                 AT_decl_line( 36 )
                 AT_prototyped( true )
                 AT_APPLE_optimized( true )
Line table dir : '../Backtrace'
Line table file: 'ViewController.m' line 0, column 20 with start address 0x0000000100004260

Looking up address: 0x0000000100004260 in .debug_frame... not found.

atos

atos -arch arm64 -o Backtrace 0x100004260
//這個(gè)行數(shù)不正確 暫不知原因
-[ViewController bar] (in BSBacktraceLogger) (ViewController.m:0)

第二種方式

#include <execinfo.h>
//定義一個(gè)指針數(shù)組
    void* callstack[128];
    //該函數(shù)用于獲取當(dāng)前線程的調(diào)用堆棧,獲取的信息將會(huì)被存放在callstack中娘锁。
    //參數(shù)128用來指定callstack中可以保存多少個(gè)void* 元素牙寞。
    //函數(shù)返回值是實(shí)際獲取的指針個(gè)數(shù),最大不超過128大小在callstack中的指針實(shí)際是從堆棧中獲取的返回地址,每一個(gè)堆棧框架有一個(gè)返回地址莫秆。
    int frames = backtrace(callstack, 128);
    //backtrace_symbols將從backtrace函數(shù)獲取的信息轉(zhuǎn)化為一個(gè)字符串?dāng)?shù)組.
    //參數(shù)callstack應(yīng)該是從backtrace函數(shù)獲取的數(shù)組指針,frames是該數(shù)組中的元素個(gè)數(shù)(backtrace的返回值)
    //函數(shù)返回值是一個(gè)指向字符串?dāng)?shù)組的指針,它的大小同callstack相同.每個(gè)字符串包含了一個(gè)相對于callstack中對應(yīng)元素的可打印信息.
    char **strs = backtrace_symbols(callstack, frames);

    NSMutableArray *backtrace = [NSMutableArray arrayWithCapacity:frames];

    for (int i=0;i<frames;i++)
    {
        [backtrace addObject:[NSString stringWithUTF8String:strs[i]]];
    }
    //注意釋放
    free(strs);

backtrace 源碼在這里间雀,需要的可以看下實(shí)現(xiàn)。

這個(gè)方式最簡單镊屎,而且是已經(jīng)解析過的惹挟,直接把這個(gè)數(shù)組上傳就可以了。

第三種方式

統(tǒng)計(jì)函數(shù)調(diào)用堆棧一般分為3步:

  • 獲取函數(shù)地址
  • 通過代碼或者dSYM獲取對應(yīng)的符號地址
  • 拿到對應(yīng)的代碼信息

前面2種都是通過調(diào)用系統(tǒng)函數(shù)來獲取的缝驳,第3種方式完全是通過自己的代碼邏輯來進(jìn)行獲取连锯。

通過thread_get_state函數(shù)獲取指定線程的riprbp用狱,再遍歷所有的動(dòng)態(tài)庫运怖,看rbp是否在當(dāng)前的動(dòng)態(tài)庫里,在的話就解析動(dòng)態(tài)庫獲取符號地址和字符地址夏伊。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末摇展,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子溺忧,更是在濱河造成了極大的恐慌咏连,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件鲁森,死亡現(xiàn)場離奇詭異捻勉,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)刀森,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進(jìn)店門踱启,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事埠偿⊥赴眨” “怎么了?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵冠蒋,是天一觀的道長羽圃。 經(jīng)常有香客問我,道長抖剿,這世上最難降的妖魔是什么朽寞? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮斩郎,結(jié)果婚禮上脑融,老公的妹妹穿的比我還像新娘。我一直安慰自己缩宜,他們只是感情好肘迎,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著锻煌,像睡著了一般妓布。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上宋梧,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天匣沼,我揣著相機(jī)與錄音,去河邊找鬼捂龄。 笑死释涛,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的跺讯。 我是一名探鬼主播枢贿,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼刀脏!你這毒婦竟也來了局荚?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤愈污,失蹤者是張志新(化名)和其女友劉穎耀态,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體暂雹,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡首装,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了杭跪。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片仙逻。...
    茶點(diǎn)故事閱讀 39,834評論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡驰吓,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出系奉,到底是詐尸還是另有隱情檬贰,我是刑警寧澤,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布缺亮,位于F島的核電站翁涤,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏萌踱。R本人自食惡果不足惜葵礼,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,167評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望并鸵。 院中可真熱鬧鸳粉,春花似錦、人聲如沸能真。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽粉铐。三九已至,卻和暖如春卤档,著一層夾襖步出監(jiān)牢的瞬間蝙泼,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工劝枣, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留汤踏,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓舔腾,卻偏偏與公主長得像溪胶,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子稳诚,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,779評論 2 354

推薦閱讀更多精彩內(nèi)容