将新的 PyObject * 从 C++ 返回给 Python 最终会出现段错误

Returning new PyObject * from C++ to Python eventually segfaults

本文关键字:段错误 错误 Python PyObject 返回 C++      更新时间:2023-10-16

我正在编写一个库的C++和Python端,该库公开了我们软件中的一些功能,这些功能C++编写为Python脚本。我正在将一些感兴趣的源文件和一个如下所示的包装器文件编译到一个共享库中,并使用 ctypes 加载该库。

extern "C" {
    PyObject *py_get_cxx_set_EXAMPLE(void)
    {
        std::set<long> cset = get_cxx_set_for_python();
        PyGILState_STATE gstate = PyGILState_Ensure();
        PyObject *pyset = PySet_New(NULL);
        for (long c_long: cset)
            PySet_Add(pyset, PyLong_FromLong(c_long));
        PyGILState_Release(gstate);
        return pyset;
    }
}

在蟒蛇方面:

example_lib.py_get_cxx_set_EXAMPLE.restype = ctypes.py_object
for i in range(0, 1000):
    ret = example_lib.py_get_cxx_set_EXAMPLE()

我发现前几次调用会成功,但C++代码会在循环中间出现段错误。在GDB上,我会发现调用堆栈的结尾是这样的:

#0  0x000055555563244a in PyErr_Occurred ()
#1  0x000055555562a387 in _PyObject_GC_Malloc ()
#2  0x0000555555629ebd in _PyObject_GC_New ()
#3  0x000055555562b23c in PyDict_New ()
#4  0x00007ffff66df9be in python::to_python_object<db::pmbus_diagnostics> (t=...) at python_wrapper/python.hpp:101

看起来 Python 运行时拒绝为我制作更多的 Python 对象(在本例中为字典)...!

我在C++代码中做错了什么?

编辑::更新,见答案

好的,我忘了添加代码来获取和释放某些函数类的全局解释器锁。对不起,这个愚蠢的问题。

对Python的孩子有信心。