C DLL 中初始化 Python 虚拟机问题

2022-11-17 21:25:38 +08:00
 justou

请教一个 C 中调用 Python 函数的问题。A.dll 中导出了int start_python()(调用了Py_Initialize) 和void stop_python()(调用了Py_Finalize)。B.dll 导出了一些 C 函数,这些 C 函数中调用了一些 python 函数, 这些 Python 函数是通过 cython 的 cdef public 声明再由 cython 生成的(Embedding Cython modules in C/C++ applications),目的是通过 dll 将 Python 环境隐藏。应用 a.exe 通过 loadlibrary 的方式加载了 A.dll 和 B.dll (它们都依赖于 python3.dll ):

  1. 调用 A.dll 的start_python启动 python 解释器, 然后调用 B.dll 中导出的 C 函数, 产生 access violation ;
  2. 调用 A.dll 的start_python启动 python 解释器, 尝试在 B.dll 中也调用Py_Initialize启动 python 解释器,都启动成功, 调用 B.dll 中导出的 C 函数, 产生 access violation ;
  3. 仅在 B.dll 中通过Py_Initialize启动 python 解释器,调用 B.dll 中导出的 C 函数, Ok.

经过 debug 发现 access violation 的出现是 cython 生成的代码里某个函数指针为 NULL,然后调用了它,看上去像是 Python 虚拟机没有得到正确的初始化,导致那个函数指针也没有初始化。请问 1 、2 问题出在哪儿?

手头暂时没有 Windows 机器,之后我再简化一个可测试的 demo 。

1171 次点击
所在节点    C++
1 条回复
justou
2022-11-18 10:52:23 +08:00
做了个简易 demo, 所有依赖都在里面, CLion 可以直接打开工程. 其它开发环境可能要用里面的 CMakeLists.txt 生成工程文件 https://drive.google.com/file/d/1OElQ6j8LwA-A2A45G2OROepqywmvh3Hn/view?usp=sharing

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/896047

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX