python - 在 ubuntu 中嵌入 python 时出现段错误

标签 python linux debugging shared-libraries

我有一个嵌入 python 的应用程序。它是在 windows 上开发的,运行良好,但现在我将它移植到 linux 上,但不太成功,因为它在 Py_Initialize() 中崩溃了。从 gdb 看,它似乎是在加载 os 模块时发生的。

gdb 在 seg 错误时报告这个调用栈:

#0  0x002384fc in import_submodule (mod=None, subname=0xb7d8a9bb "os", fullname=0xb7d8a9bb "os") at ../Python/import.c:2551
#1  0x0023893c in load_next (mod=<value optimized out>, altmod=<value optimized out>, p_name=0xb7d8a9ac, buf=0xb7d8a9bb "os", p_buflen=0xb7d8a9b4) at ../Python/import.c:2411
//.... etc...:
#65 0x002439de in Py_Initialize () at ../Python/pythonrun.c:361

Python/import.c中import_submodule的源码(从2.6.5 python下载):

static PyObject *
import_submodule(PyObject *mod, char *subname, char *fullname)
{ //***************** THIS IS LINE 2551***************************
    PyObject *modules = PyImport_GetModuleDict();
    PyObject *m = NULL;

    /* Require:
       if mod == None: subname == fullname
       else: mod.__name__ + "." + subname == fullname
    */

    if ((m = PyDict_GetItemString(modules, fullname)) != NULL) {
        Py_INCREF(m);
    }
    else {
        PyObject *path, *loader = NULL;
        char buf[MAXPATHLEN+1];
        struct filedescr *fdp;
        FILE *fp = NULL;

        if (mod == Py_None)
            path = NULL;
        else {
            path = PyObject_GetAttrString(mod, "__path__");
            if (path == NULL) {
                PyErr_Clear();
                Py_INCREF(Py_None);
                return Py_None;
            }
        }

        buf[0] = '\0';
        fdp = find_module(fullname, subname, path, buf, MAXPATHLEN+1,
                  &fp, &loader);
        Py_XDECREF(path);
        if (fdp == NULL) {
            if (!PyErr_ExceptionMatches(PyExc_ImportError))
                return NULL;
            PyErr_Clear();
            Py_INCREF(Py_None);
            return Py_None;
        }
        m = load_module(fullname, fp, buf, fdp->type, loader);
        Py_XDECREF(loader);
        if (fp)
            fclose(fp);
        if (!add_submodule(mod, m, fullname, subname, modules)) {
            Py_XDECREF(m);
            m = NULL;
        }
    }

    return m;
}

反汇编+部分源码+断点:

2550    import_submodule(PyObject *mod, char *subname, char *fullname)
2551    {
   0x002384f0 <+0>: push   %ebp
   0x002384f1 <+1>: mov    %esp,%ebp
   0x002384f3 <+3>: sub    $0x1058,%esp
   0x002384f9 <+9>: mov    %ebx,-0xc(%ebp)
=> 0x002384fc <+12>:    call   0x174787 <__i686.get_pc_thunk.bx>
   0x00238501 <+17>:    add    $0x112af3,%ebx
   0x00238507 <+23>:    mov    %esi,-0x8(%ebp)
   0x0023850a <+26>:    mov    0x8(%ebp),%esi
   0x0023850d <+29>:    mov    %edi,-0x4(%ebp)
   0x00238510 <+32>:    mov    %eax,-0x102c(%ebp)
   0x00238516 <+38>:    mov    %edx,-0x1030(%ebp)
   0x0023851c <+44>:    mov    %gs:0x14,%eax

这里发生了什么?显然 libpython2.6.so 中的一些代码已加载并运行。

最佳答案

在调用 Py_Initialize 之前尝试使用 Py_SetPythonHome 设置 Python home。从 python 目录的硬编码完整路径开始。还要确保您没有混合调试和发布版本。 Py_GetPath 是一个很好的 API,可以查看所有 python 在哪里寻找模块——但不知道它是否可以在 Py_Initialize 之前调用。

   char pySearchPath[] = "/users/abc/Python26";
   Py_SetPythonHome(pySearchPath);
   Py_Initialize();
   PyRun_SimpleString("from time import time,ctime\n"
                     "print 'Today is',ctime(time())\n");
   cerr << Py_GetPath() << endl;

关于python - 在 ubuntu 中嵌入 python 时出现段错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3094166/

相关文章:

c - 在c中获取物理硬盘列表

linux - 服务器端node.js请求无法连接到主机

android - 如何将 Visual Studio 2015 调试器附加到 Android 应用程序

c++ - 如何使用 visual studio 调试 Borland BCC55 编译的 EXE?

python - 按照 celery 教程,我收到此错误 No module named 'src'

python - 了解 Keras LSTM

python - Selenium Chrome WebDriver 不使用代理

python - 使用偏移量和限制对数组进行切片

c - linux:为什么 connect() 在接受调用失败时阻塞?

c++ - 使用 VS 调试 C++。追踪变量变化