12

我有一个 C 源/头文件,它是一个更大项目的一部分。我想将其作为一个单元进行测试,独立于实际项目。虽然可以在 C 中通过使用不同main()的 .框架等

我的印象是我可以用 CFFI 做到这一点。这是一个示例 C 文件:

// magic.c
// Implementation of magic.
int add(int a, int b)
{
    return a;
}

标题:

// magic.h
// Add two numbers (where a + b is not greater than INT_MAX).
int add(int a, int b);

这是一个试图编译它的脚本,所以我可以调用一些函数:

# cffi_test.py
import cffi

INCLUDE_DIRS = ('.',)

SOURCES = ('magic.c',)

ffi = cffi.FFI()

ffi.set_source(
    '_magic_tests',
    '#include "magic.h"',
    include_dirs = INCLUDE_DIRS,
    sources = SOURCES,
    libraries = [],
    )

ffi.compile()

最终,我计划在一组单元测试之前将其作为设置的一部分,例如。纯 Python 函数将通过在测试设置中构造的对象test_add()调用并检查 C 函数的结果。add()ffi

上面的脚本似乎有效;它运行没有错误,它创建一个_magic_tests.c文件、一个_magic_tests.cp35-win32.pyd文件和一个Release目录。我也可以import _magic_tests没有错误。

但我不知道如何通过 CFFI 实际调用 C 函数。我找不到该set_source()函数的任何文档,它似乎是整个过程不可或缺的一部分。概述提到了很多,但参考文献中出现的次数为零。文档确实有一个关于调用函数的部分,但它引用了某个lib对象而没有显示它是如何创建的。如果我查看前面的示例,则有一个从 中lib创建的对象ffi.dlopen(),但我不知道如何将其应用于 CFFI 本身正在生成的东西。

我的大问题(即我的X 问题)是:

  • CFFI 是否是用于以跨平台(Windows 7-10、Linux、OS X)方式调用和测试 C 函数的合理工具,如果是,如何?

我目前的方法(即我的Y 问题)引起的问题是:

  • 文档在哪里set_source()?我怎样才能找出它需要什么论据?
  • 如何生成lib包含我要调用的函数的对象?
  • 这是使用 CFFI 调用 C 函数的最简单方法吗?我并不特别需要或不希望生成共享库或可再发行包;如果它必须发生,那很好,但没有必要。我还可以尝试哪些其他方法?

我目前的设置是:

  • 操作系统:Windows 10
  • Python:CPython 3.5.1 32 位
  • 点数:8.1.2
  • CFFI:1.6.0
  • C 编译器:Visual C++ Build Tools 2015 附带的任何内容,链接自此 MSDN 帖子

我正在使用Christoph Gohlke 的存储库中的 CFFI 和 pycparser 。

4

1 回答 1

14

对于我的一个项目,我cffi用来测试我的 C 代码。恕我直言cffi,它是为 C 代码生成 python 绑定的好工具,因此认为它是用于从 python 调用和测试 C 函数的合理工具。但是,您的代码只能像 C 代码那样跨平台,因为您必须为每个平台编译绑定。

您可以在下面找到一些对应该回答您的问题的文档的参考。此外,我编写了一些示例代码来说明您将如何使用cffi. 对于更大的示例,您可以在https://github.com/ntruessel/qcgc/tree/master/test找到我的项目。

四个你的例子,build_magic_tests.py看起来像这样:

from cffi import FFI

ffibuilder = FFI()

# For every function that you want to have a python binding,
# specify its declaration here
ffibuilder.cdef("""
    int add(int a, int b);
                """)

# Here go the sources, most likely only includes and additional functions if necessary
ffibuilder.set_source("magic_tests",
    """
    #include "magic.h"
    """, sources=["magic.c"])

if __name__ == "__main__":
    ffibuilder.compile()

要生成 magic_tests 模块,您必须运行python build_magic_tests.py. 生成的模块可以像这样导入和使用:

from magic_tests import ffi, lib

def run_add():
    assert 4 == lib.add(4, 5)
于 2016-06-21T08:22:36.423 回答