编译Python扩展的C文件时出现exec格式错误

时间:2016-01-07 22:46:51

标签: c python-2.7 gcc archlinux-arm

我正在尝试使用简单的C文件扩展python。我成功创建了自己的python模块,但我还需要C文件本身作为独立的可执行文件进行编译和运行。我可以成功编译它,但是当我尝试运行它时,我得到错误"无法执行二进制文件:执行格式错误"。

这是我的C源文件(hellomodule.c):

#include <Python.h>

void print_hello(const char* name){
    printf("Hello %s!\n", name);
}

//Only used by Python
static PyObject*
say_hello(PyObject* self, PyObject* args)
{
    const char* name;

    if (!PyArg_ParseTuple(args, "s", &name))
        return NULL;

    //printf("Hello %s!\n", name);
    print_hello("World");

    Py_RETURN_NONE;
}

//Only used by Python
static PyMethodDef HelloMethods[] =
{
    {"say_hello", say_hello, METH_VARARGS, "Greet somebody."},
    {NULL, NULL, 0, NULL}
};

//Only used by Python
PyMODINIT_FUNC
inithello(void)
{
    (void) Py_InitModule("hello", HelloMethods);
}

int main(){
    print_hello("World");
}

我可以&#34;成功&#34;使用以下代码编译它没有任何错误或警告:

gcc -I /usr/include/python2.7 -c hellomodule.c -o hellomodule

制作&#34; hellomodule&#34;文件可执行文件,我运行它并得到错误:

-bash: ./hellomodule: cannot execute binary file: Exec format error

为什么会导致这样的错误?

1 个答案:

答案 0 :(得分:1)

您正在尝试执行不是可执行文件的目标文件。要将代码编译为模块,您需要类似的东西

gcc -Wall -Werror -Wextra -O2 -I/usr/include/python2.7 \
    -shared hellomodule.c -o hellomodule.so -lpython2.7

但要正确链接到所有内容并添加所有可能的包含目录,有一个名为python-config的脚本,您只需要像这样调用它

gcc -Wall -Werror -Wextra -O2 `python-config --includes` \
    -shared hellomodule.c -o hellomodule.so `python-config --libs`

甚至更好,该脚本也提供CFLAGSLDFLAGS,最后

gcc -Wall -Werror -Wextra -O2 `python-config --cflags` \
    -shared hellomodule.c -o hellomodule.so `python-config --ldflags`

然后,将生成的文件复制到/usr/lib/python2.7/site-packages

之后,您可以在这样的python脚本中加载模块

import hellomodule

目标文件是最终由链接器( ld可能)使用的中间二进制文件,以生成最终的二进制文件。 python模块没有main()函数,它必须是一个运行时可加载的共享对象,它导出一些预定义的符号,python解释器将用它们将模块加载到python脚本/程序中。

注意:要正确执行此操作而不是尝试创建一个Makefile,就像这样

CFLAGS = `python-config --cflags` -Wall -Werror # add more if needed
LDFLAGS = `python-config --ldflags` # add more if needed
OBJS = hellomodule.o # add more if needed

all: $(OBJS)
    $(CC) $(CFLAGS) $(LDFLAGS) -shared -o modulename.so $(OBJS)

%.o: %.c
    $(CC) -c $(CFLAGS) $<

确保使用标签而不是缩进空格,然后在 Makefile 和源文件所在的同一目录中运行make。< / p>