菜鸟教程小白 发表于 2022-12-13 02:35:21

ios - CFBundleGetFunctionPointerForName 和 dlsym 为导出的函数返回 NULL


                                            <p><p>我有一个 JavaScriptCore 框架的分支,我在其中添加了一个自己的函数,该函数被导出。框架编译只是find。在框架上运行 <code>nm</code> 会发现函数 (<code>JSContextCreateBacktrace_unsafe</code>) 确实已导出:</p>

<pre><code>Leo-Natans-Wix-MPB:JavaScriptCore.framework lnatan$ nm -gU JavaScriptCore.framework/JavaScriptCore | grep JSContextCreateBacktrace
00000000004cb860 T _JSContextCreateBacktrace
00000000004cba10 T _JSContextCreateBacktrace_unsafe
</code></pre>

<p>但是,我无法使用 <code>CFBundleGetFunctionPointerForName</code> 或 <code>dlsym</code> 获取该函数的指针;两者都返回 <code>NULL</code>。起初,我使用 <code>dlopen</code> 打开我的框架,然后尝试使用 <code>CFBundleCreate</code> 和 <code>CFBundleGetFunctionPointerForName</code> 但也返回 NULL。</p>

<p>这是什么原因造成的?</p>

<p><strong>更新</strong></p>

<p>发生了一些可疑的事情。我重命名了其中一个 JSC 函数,<code>nm</code> 反射(reflect)了这一点。但是,<code>dlsym</code> 仍然能够找到具有原始名称的函数,而不是重命名的函数。</p></p>
                                    <br><hr><h1><strong>Best Answer-推荐答案</ strong></h1><br>
                                            <p><p>这很难追踪,因为它高度依赖于您的特定环境和情况,但您很可能遇到此问题,因为系统镜像已经加载并且您没有更改框架。</p>

<p>如果您查看 <a href="https://github.com/opensource-apple/dyld/blob/195030646877261f0c8c7ad8b001f52d6a26f514/src/dyldAPIs.cpp#L1458-L1471" rel="noreferrer noopener nofollow">dyld/dyldAPIS.cpp:1458</a> 中 <code>dlopen</code> 的源代码,您会注意到传递给 dyld 的上下文是使用 <code>matchByInstallName = true</code> 配置的。然后将此上下文传递给 <code>load</code>,后者执行图像加载所需的各个阶段。有几个阶段值得注意:</p>

<ul>
<li><p><code>loadPhase2</code> in <a href="https://github.com/opensource-apple/dyld/blob/195030646877261f0c8c7ad8b001f52d6a26f514/src/dyld.cpp#L2896" rel="noreferrer noopener nofollow">dyld/dyld.cpp:2896</a>提取框架路径的结尾并在搜索路径中搜索</p></li>
<li><p><code>loadPhase5check</code> in <a href="https://github.com/opensource-apple/dyld/blob/195030646877261f0c8c7ad8b001f52d6a26f514/src/dyld.cpp#L2712-L2731" rel="noreferrer noopener nofollow">dyld/dyld:2712</a>遍历所有加载的图像并确定其中是否有匹配的安装名称,如果有,则返回该名称而不是加载新的。</p></li>
<li><p><code>loadPhase5load</code> in <a href="https://github.com/opensource-apple/dyld/blob/195030646877261f0c8c7ad8b001f52d6a26f514/src/dyld.cpp#L2601" rel="noreferrer noopener nofollow">dyld/dyld:2601</a>如果之前的任何步骤都没有加载/找到图像,则最终加载图像。 (值得注意的是,首先执行 <code>loadPhase5check</code>,因为图像加载是一个两遍过程。)</p></li>
</ul>

<p>鉴于以上所有情况,我会尝试将您的框架重命名为 <code>JavaScriptCore.framework</code> 之外的其他名称。根据系统框架和您的框架的安装名称,我还建议更改安装名称。 (有大量的博客文章和 StackOverflow 帖子记录了如何使用 <code>install_name_tool -id</code> 进行此操作。)</p></p>
                                   
                                                <p style="font-size: 20px;">关于ios - CFBundleGetFunctionPointerForName 和 dlsym 为导出的函数返回 NULL,我们在Stack Overflow上找到一个类似的问题:
                                                        <a href="https://stackoverflow.com/questions/46148634/" rel="noreferrer noopener nofollow" style="color: red;">
                                                                https://stackoverflow.com/questions/46148634/
                                                        </a>
                                                </p>
                                       
页: [1]
查看完整版本: ios - CFBundleGetFunctionPointerForName 和 dlsym 为导出的函数返回 NULL