首页 > 解决方案 > 如何将 C typedef 结构和函数与 Rust 中的该结构一起使用?

问题描述

我有这些 C 文件,我想使用pair_add,addPAIR来自 Rust。

加法器.c

#include <stdlib.h>
#include "adder.h"

int pair_add(PAIR * ppair) {
    return ppair->x + ppair->y;
}

int add(int x, int y) {
    return x + y;
}

加法器.h

typedef struct {
    int x;
    int y;
} PAIR;

int pair_add(PAIR * ppair);
int add(int, int);

我使用以下方法编译它们:

gcc -c adder.c
ar rc libadder.a adder.o  # Static link

文档没有详细说明如何集成 C typedef 结构,并且该示例适用于返回和接受的函数i32其他在线资源也有限

我尝试了以下但无法添加PAIRtypedef:

extern crate libc;

use libc::c_int;

#[link(name = "adder")]
extern "C" {
    // Define PAIR

    // int pair_add(PAIR * ppair);
    fn pair_add(input: Pair) -> c_int;

    // int add(int, int);
    fn add(input1: c_int) -> c_int;
}

fn main() {}

标签: rustexternffi

解决方案


第一的:

typedef struct {
    int x;
    int y;
} PAIR;

这声明了一个匿名结构,目前 Rust 不支持。有一个RFC建议添加匿名类型。

其次, atypedef只是一个别名,结构的名称对于兼容并不重要。这意味着您可以简单地执行以下操作:

extern crate libc;
use libc::c_int;

#[repr(C)]
struct PAIR {
    x: c_int,
    y: c_int,
}

// optional "equivalent" of typedef
type Pair = PAIR;

extern "C" {
    fn pair_add(input: *mut Pair) -> c_int;
    // could be
    // fn pair_add(input: *mut PAIR) -> c_int;
}

您可以轻松地忽略 typedef 并仅用PAIR作此结构的名称。您甚至可以只写struct PAIR;使其不透明。


推荐阅读