iOS 符号化方法

介绍 iOS 堆栈符号化常用的方法。

符号化可以将上面的得到的代码内存地址转为可读的函数签名信息,包括函数名、文件名、行数等等,方便我们快速定位问题。

符号化常用方法

使用 backtrace_symbols_fd 方法进行运行时符号化

backtrace_symbols_fd 函数能够将从backtrace 中得到的原始堆栈信息进行符号化,并输出到控制台。

vm_address_t *stacks_2[128];
int depth = backtrace((void **)stacks_2, 128);
backtrace_symbols_fd((void **)stacks_2, depth, STDOUT_FILENO);     // STDOUT_FILENO 表示向屏幕输出
💡
注意,这里用的是 backtrace_symbols_fd 而不是 backtrace_symbols。backtrace_symbols 内部实现也有 alloc/malloc 相关的调用,在 malloc_logger 里面调用会陷入递归死循环

使用 atos 和 dSYM 对堆栈信息符号化

假设我们的崩溃日志系统收集到了下面这样的崩溃日志:

"0 libsystem_kernel.dylib 0x1b00df000 0x1b00e1d0c" "1 libsystem_kernel.dylib 0x1b00df000 0x1b00ffe70" "2 Keep 0x104584000 0x10704e864" "3 Keep 0x104584000 0x10704ac5c" "4 libsystem_malloc.dylib 0x192e77000 0x192e89b4c" "5 Keep 0x104584000 0x10a1d9c2c" "6 Keep 0x104584000 0x10a209ba4" "7 Keep 0x104584000 0x10a209e48" "8 Keep 0x104584000 0x108c877b8" "9 Keep 0x104584000 0x108c86c04" "10 Keep 0x104584000 0x108c6c8d0" "11 Keep 0x104584000 0x108c68938" "12 Keep 0x104584000 0x108c664d8" "13 Keep 0x104584000 0x108c66d5c" "14 Keep 0x104584000 0x108c66ee8" "15 libdispatch.dylib 0x183f33000 0x183f932b0" "16 libdispatch.dylib 0x183f33000 0x183f94298" "17 libdispatch.dylib 0x183f33000 0x183f3ca40" "18 libdispatch.dylib 0x183f33000 0x183f3d518" "19 libdispatch.dylib 0x183f33000 0x183f46fac" "20 libsystem_pthread.dylib 0x1cc79e000 0x1cc7a95bc" 

我们需要先拿到 App 的 dSYM 文件。之后在终端使用 xcrun atos 进行符号化。

xcrun atos -arch arm64 -o Keep.app.dSYM/Contents/Resources/DWARF/Keep -l 0x104584000
0x108c66ee8
# 第一行指令输入一次即可,随后可以逐次输入调用方法的内存地址

逐次输入各个调用方法的内存地址后,控制台会输出相应的解析结果。

符号化

动态库堆栈符号化

iOS 静态库是在程序编译的时候就进行了链接,拷贝到了 App 的可执行文件里。而动态库并不会在编译时置入 App 中,而是在 App 运行的时候去动态查找动态库并进行链接。

静态库代码发生崩溃时,堆栈信息也是和 App 放在一起的,内存加载地址一致,可以用 dSYM 解析。

但是动态库堆栈信息无法使用 dSYM 解析,需要利用动态库的 framework 进行解析。

堆栈信息:

"0 libdispatch.dylib 0x1210e4000 0x1210eabb8" "5 KPlayerCore 0x121f48000 0x121f61134" "6 KPlayerCore 0x121f48000 0x121f954a8" "7 KPlayerCore 0x121f48000 0x121f95634" "8 KPlayerCore 0x121f48000 0x121f74e5c" "9 libsystem_pthread.dylib 0x213dcc000 0x213dcd6b8"

找到工程使用的动态库 framework:

img.png

使用 atos 工具解析堆栈:

img.png

Read more

碎碎念——投资,不确定性沟通定语

碎碎念——投资,不确定性沟通定语

投资理财 最近因为关税的冲击,美股正在经历一波大跌行情。我个人比较看好纳斯达克,也在一直定投纳斯达克。我是长期主义者,没有精力和时间在短期波动中挣钱,只想在下跌调整中「进货」。 定投分左侧定投和右侧定投。左侧定投是在下跌的过程中定投,而右侧定投是在上涨的过程中定投。左侧定投无法确认底部在哪里,需要源源不断往里投入金钱(行内成为「子弹」);右侧定投无法确认反弹是诱多还是形势已经逆转。我采用的是左侧定投,大跌大加,小跌小加,反弹时停止定投。不论采用哪种定投,殊途同归,都是尽量降低投资成本。 目前网上看衰美股的声音不少,不少人因为恐慌割肉卖出股票。但我们要知道目前美国仍旧是世界第一大国,消费潜力巨大,大型科技公司(苹果、英伟达等)的基本面并没有出现大问题。只是因为特朗普的「量子态」关税政策,导致市场恐慌抛售。我们无需担心纳斯达克、标普指数从此一蹶不振。恰恰相反,现在是买入美股的绝佳时机。苹果、英伟达等大型公司的 PE 值已经降到了合理位置,只要不买妖股,不投机,只关注纳斯达克、标普指数,只买大型公司股票,迟早会取得丰厚盈利的。

By Gray
怀念小时候吃过的食物

怀念小时候吃过的食物

前两天下班骑车回家的路上听到了路旁有人在讨论泡馍。他们口中的泡馍应该是类似西安羊肉泡馍之类的食物。但是我却想起来了小时候吃的不一样的泡馍以及其他吃食。 不一样的泡馍 小时候我们那里普遍比较贫穷,家家户户除了过年过节基本上很难吃到大块肉。小孩子饭量时小时大,中午吃的饭,半晌就又饿了。家里有大葱或者豆糁的话,可以拿着一个馍就着就吃了。整根的葱是最下馍的,葱白部分甜又辣,葱叶里面会有像鼻涕一样的粘液,要把它挤出来才下得嘴吃。豆糁是黄豆的发酵产物,煮熟的大豆加盐发酵几天,黏丝丝的时候团成球,放到发黑就能吃了。吃的时候从球上掰下来几小块就行。豆糁是咸的,因而也能下饭。不过最妙的吃法是将豆糁和鸡蛋一起炒。鸡蛋的香气和豆糁稍微发臭的味道混在一起,形成一种独特的香味。像北京的臭豆腐一样,闻着臭,吃着香。 如果家里没葱没豆糁了,馍又很干,那泡馍就是解决饿肚子的绝好办法。将干硬的馍掰成几瓣,不能太碎小,放到瓷碗里。倒入炒菜的肉味王佐料,或者是平时攒下来的方便面调料。再提溜着暖水瓶,倒进去冒着热气的水。当然香油是少不了的,拿着油光光的瓶子,滴进去几滴喷香的香油。最后用大碗盖住,或者干脆啥也不盖,静等

By Gray
Swift Server Push Notification 配置

Swift Server Push Notification 配置

获取证书 在 Apple Developer 开发者账号 Certificates, Identifiers & Profiles 里选择 Keys。新增一个 key, configure 里选择 Sandbox & Production。下载该 p8 证书,并且保存好(只能下载一次)。 终端 cd 到证书所在路径,输入下面指令。 openssl pkcs8 -nocrypt -in AuthKey_XXXXXXXXX.p8 -out ~/Downloads/key.pem cat key.pem 得到 PRIVATE KEY 字符串,复制好。 服务端配置 服务端有多种技术栈方案,包括 Java、

By Gray
香港游记——一个传统而又现代的城市

香港游记——一个传统而又现代的城市

这是 2024 年的最后一场旅行,从北京到香港,跨越了大半个中国。去香港,一方面是想领略一下它的文化和风光,另一方面是想办一个香港银行卡,买港美股以及海外收付款。 从北京到香港,动卧是一个不错的选择。乘坐 D903 次动车,晚上八点登车,睡一觉,第二天一早就到深圳北了。再从深圳北坐高铁过口岸到香港西九龙,差不多上午九点多就能到达香港。深圳北到西九龙的高铁车次非常多,不用担心买不到票。 密集的建筑 香港给我的初印象就是——这里的楼房真的很密集。不光是住宅区又高又密,商业区的建筑物与建筑物之间也几乎只有街道相隔,很少见到大型的公园或者绿化带。土地利用率很高。这一点和北京差别还是挺大的。北京虽然也是寸土寸金,但是市内绿化面积很高,大型公园也很常见。 街上密集的建筑,让人第一眼看就知道这是香港。 旧与新,传统与现代 在香港,不同地区的风格面貌会相差很多。你既能见到破旧不堪、需要修缮的古老楼房,也能见到银光闪闪、科技感十足的现代化大厦。这种新与旧的切换,传统和现代的反差,总是能给人强烈的震撼。这正是香港的魅力所在。 维多利亚港和中环摩天轮 维多利亚港是香港的中心,是香港旅游

By Gray