在运行时动态调用C函数而不知道其原型

时间:2013-04-27 14:34:52

标签: c linux shared-libraries ld

我想知道是否有可能实现上述目标。显然,可以在Linux中使用dlopen, dlsym方法加载库并调用它的方法。但它需要知道函数原型,以便在调用之前将void *指针强制转换为相应的类型。

假设可以在外部提供原型元数据(使用某些描述符文件等)

有没有办法实现这个目标?

1 个答案:

答案 0 :(得分:4)

可以肯定,但不要指望任何便携。例如,您可以使用臭名昭着的libffi库。伪C中的虚拟示例:

// We make some kind of descriptor structure and possible return and argument types
enum c_type {
    C_TYPE_CHAR,
    C_TYPE_INT,
    C_TYPE_FLOAT,
    C_TYPE_PTR,
    C_TYPE_VOID
};

struct func_proto_desc {
    enum c_type ret_type;
    int n_args; // Reasonable convention: -1 for variadic
    c_type *arg_types;
};

// Imaginary function that parses textual metadata and returns a function descriptor
void parse_func_desc(const char *str, struct func_proto_desc *desc);

// this is how to use it:
struct func_proto_desc fproto;
parse_func_desc("void (*)(int, float, const char *)", &fproto);

ffi_cif cif;
ffi_type *args[3];
void *vals[3];

int n = 42;
float f = 3.1415927;
const char *s = "Hello world!";

vals[0] = &n;
vals[1] = &f;
vals[2] = &s;

// Here you can set up the types according to the type description
// that the parser function returned
// (this one is an imaginary function too)
populate_ffi_types_from_desc(args, &fproto);

// Use libffi to call the function
ffi_prep_cif(&cif, FFI_DEFAULT_ABI, fproto->n_args, &ffi_type_void, args);
ffi_call(&cif, func_ptr, NULL, vals);

这样的事情应该让你开始。