给定 mach_header 如何找到二进制图像名称?

Given a mach_header how do I find the binary image name?

我正在研究 Mach 动态链接器 dyld。这个问题适用于所有 Apple 平台,但很高兴得到特定于平台的答案;我正在使用 ObjC,但如果适合您,我也很乐意翻译 Swift。 The relevant Apple docs can be found here

我正在尝试解决我维护的库中的死锁问题,因此需要捕获有关加载的二进制图像的信息。 (对于外行来说,这些不是 JPEG 类型的图像)。

我已经通过 _dyld_register_func_for_add_image() 注册了一个回调来监视二进制图像添加,并且在加载图像时调用它。我可以看到传入的 mach_header 结构以及 vmaddr_slide 值。

出于各种避免死锁的原因,我不想使用 _dyld_image_count()/_dyld_get_image_header() post-hoc 遍历图像,而是希望在图像出现时跟踪图像loaded/unloaded 仅使用 add()/remove() 回调。通常我可以在迭代图像时调用 _dyld_get_image_name(index) 来检索图像名称,但是,就像我说的,我不能使用迭代模型。

我的问题是:根据传递给 func_for_add_image() 回调的值(使用任何必要的额外系统调用)我如何确定图像名称?

一个答案,但可能不是答案如下:

// Required header:
#include <dlfcn.h>

// This is the callback, added via _dyld_register_func_for_add_image()
void add_binary_image(const struct mach_header *header, intptr_t slide)
{
    // ...

    Dl_info  DlInfo;
    dladdr(header, &DlInfo);
    const char* image_name = DlInfo.dli_fname;

    // ...
}

The docs 声明 dladdr() 函数仅在动态链接程序中可用。我希望在其中使用的框架也作为静态库存在。我已经达到了我的知识极限,所以想仔细检查我是否正确地误解了动态链接,以及使用 dladdr() 在所有现代 Mac、iOS 等中都可以。应用程序。

[编辑:上面的 dladdr() 调用已参考 this answer]

修复