【问题标题】:Exception when call PyModule_GetDict() with a specific python import使用特定 python 导入调用 PyModule_GetDict() 时出现异常
【发布时间】:2018-04-28 17:06:12
【问题描述】:

我想通过调用 python 的 C 代码管理 LCD。我是这方面的新手。 我的 python 应用程序与 LCD 完美配合。我的 python 代码中使用的库没有问题。 Python2.7 当我调用 PyModule_GetDict 函数时,我的问题就出现了。如果我在 python 文件中添加了一个特定的库导入,我的 C 程序就会崩溃。 库是Adafruit_ILI9341,如果我不导入,C程序运行正确。

是否可能存在与 Python interpeter 不兼容的库? 我该如何调试这个问题?

非常感谢!

这是我的 C 代码:

// Set PYTHONPATH TO working directory
setenv("PYTHONPATH",".",1);

PyObject *pName, *pModule, *pDict, *pFunc, *presult;
// Initialize the Python Interpreter
Py_Initialize();

// Build the name object
pName = PyString_FromString((char*)"lcd");

// Load the module object
pModule = PyImport_Import(pName);

// pDict is a borrowed reference 
pDict = PyModule_GetDict(pModule);

// pFunc is also a borrowed reference 
pFunc = PyDict_GetItemString(pDict, (char*)"initLCD");

if (PyCallable_Check(pFunc)) {
    printf("Let's give this a shot!\n");
    presult=PyObject_CallObject(pFunc,NULL);
    PyErr_Print();
} else {
    PyErr_Print();
}
printf("Result is %d\n",PyInt_AsLong(presult));

// Clean up
Py_DECREF(pModule);
Py_DECREF(pName);

// Finish the Python Interpreter
Py_Finalize();

这是python代码:

from PIL import Image
from PIL import ImageDraw
from PIL import ImageFont
import os
import threading
from threading import Timer

import Adafruit_ILI9341 as ADAFRUIT_TFT
import Adafruit_GPIO as GPIO
import Adafruit_GPIO.SPI as SPI
import RPi.GPIO as LCD_GPIO

import time
import locale

def iniciaDisplay():
    LCD_GPIO.setwarnings(False)
    LCD_GPIO.setmode(LCD_GPIO.BCM)
    LCD_GPIO.setup(5, LCD_GPIO.OUT)
    p = LCD_GPIO.PWM(5, 100)
    p.start(1)
    DC = 24
    RST = 25
    SPI_PORT = 0
    SPI_DEVICE = 0

    disp = ADAFRUIT_TFT.ILI9341(DC, rst=RST, spi=SPI.SpiDev(SPI_PORT, SPI_DEVICE, max_speed_hz=64000000))
    disp.begin()
    disp.clear()
    disp.display(getLogo(0))

    return 0

编辑:

按照@BlackJack 的建议,我添加了 NULL 检查和 PyErr_Print,我看到了它失败的原因。问题是由 _io 库导入引起的。
这是检查 NULL 值和 PyErr_Print 的输出:

Traceback (most recent call last):
  File "/home/pi/pruebas/DESFIREXip32/tft.py", line 18, in <module>
    import Adafruit_ILI9341 as ADAFRUIT_TFT
  File "build/bdist.linux-armv7l/egg/Adafruit_ILI9341/__init__.py", line 21, in <module>
  File "build/bdist.linux-armv7l/egg/Adafruit_ILI9341/ILI9341.py", line 23, in <module>
  File "/usr/lib/python2.7/dist-packages/numpy/__init__.py", line 153, in <module>
    from . import add_newdocs
  File "/usr/lib/python2.7/dist-packages/numpy/add_newdocs.py", line 13, in <module>
    from numpy.lib import add_newdoc
  File "/usr/lib/python2.7/dist-packages/numpy/lib/__init__.py", line 22, in <module>
    from .npyio import *
  File "/usr/lib/python2.7/dist-packages/numpy/lib/npyio.py", line 4, in <module>
    from . import format
  File "/usr/lib/python2.7/dist-packages/numpy/lib/format.py", line 141, in <module>
    import io
  File "/usr/lib/python2.7/io.py", line 51, in <module>
    import _io
SystemError: _PyImport_FixupExtension: module _io not loaded

我认为这个错误应该在新线程中处理。

【问题讨论】:

  • 在标题中写着 exception,但在正文中却写着 crash - 这是什么?您没有检查PyImport_Import() 调用的结果。
  • @BlackJack 我不知道是不是例外。如果我在 python 文件中添加 import Adafruit_ILI9341,我的 C 程序在调用 PyModule_GetDict(pModule) 函数时关闭
  • 再次:您没有检查PyImport_Import() 调用的结果。

标签: python c python-2.7


【解决方案1】:

我发现至少有五个问题,除了两个没有检查NULL 值。

您没有检查PyImport_Import() 的返回值。如果在导入 lcd 模块时出现问题/异常,您的代码将继续执行,就好像什么都没发生一样。

PyDict_GetItemString() 的返回值也没有真正与NULL 进行对比。 PyCallable_Check() 被应用,但它有些多余,因为PyObject_CallObject() 已经检查了参数是否可调用。

当然你不是检查PyObject_CallObject()的返回值是否是NULL

PyErr_Print() 可能会被调用,即使没有文档明确指出会导致致命错误的错误/异常。

最后PyInt_AsLong() 被一个可能未初始化的presult 指针调用。给定示例代码,它肯定是未初始化的,因为 Python 代码不包含 initLCD() 所以 pFuncNULLpresult 永远不会被分配一个值。

所以你最好初始化每个指向NULL 的指针并检查每个返回值。

PyObject *pName = NULL;
PyObject *pModule = NULL;
PyObject *pFunc = NULL;
PyObject *presult = NULL;

setenv("PYTHONPATH", ".", 1);
Py_Initialize();

if ((pName = PyString_FromString("lcd"))) {
    if ((pModule = PyImport_Import(pName))) {   
        if ((pFunc = PyObject_GetAttrString(pModule, "initLCD"))) {
            if ((presult = PyObject_CallObject(pFunc, NULL))) {
                printf("Result is %d\n", (int) PyInt_AsLong(presult));
            }
        }
    }
}
if (PyErr_Occurred()) PyErr_Print();

Py_XDECREF(presult);
Py_XDECREF(pFunc);
Py_XDECREF(pModule);
Py_XDECREF(pName);

Py_Finalize();

【讨论】:

  • 谢谢@BlackJack。昨晚我添加了 NULL 检查和 PyErr_Print,我看到了它失败的原因,但似乎不容易修复。我正在编写其他模块,但在这里忘记了评论。
  • 我编辑了我的问题,但我认为我必须为此提出新问题
猜你喜欢
  • 1970-01-01
  • 2013-11-29
  • 1970-01-01
  • 2015-07-04
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多