0

我正在尝试将我的简单应用程序从 C 移植到 Rust。它只在我的 Mac 上运行,只有 Mac 上的库。这是 C 代码中失败部分的简化版本

// myLog.h
#include <os/log.h> // macOS header

void debug(const char *str);

//************************************

// myLog.c
#include "myLog.h"

void debug(const char* str) {
    // call the macOS log function
    os_log_debug(OS_LOG_DEFAULT, "%{public}s", str);
}

这段代码可以简单地调用编译gcc debug.c,它工作正常。

然后我将 .h 和 .c 添加到我的 rust 项目中,并指定了如下所示的 bindgen

fn main() {
    println!("cargo:rerun-if-changed=myLog.h");

    let bindings = bindgen::Builder::default()
        .header("myLog.h")
        .parse_callbacks(Box::new(bindgen::CargoCallbacks))
        .generate()
        .expect("Unable to build bindgen");
    let out_path = PathBuf::from(env::var("OUT_DIR").unwrap());
    bindings
        .write_to_file(out_path.join("mylog_bindings.rs"))
        .expect("Couldn't write bindings!");
}

并且main函数没有其他功能,只是暂时测试一下日志:

#![allow(non_upper_case_globals)]
#![allow(non_camel_case_types)]
#![allow(non_snake_case)]

use std::ffi::CString;
include!(concat!(env!("OUT_DIR"), "/mylog_bindings.rs"));

fn main() {
    let log_infomation = CString::new("Log from Rust").expect("Failed to create c string");
    let c_pointer = log_infomation.as_ptr();
    unsafe {
        debug(c_pointer);
    }
}

该程序因以下错误而失败:

error: linking with `cc` failed: exit code: 1
  |
  = note: "cc" "-m64" "-arch" "x86_64" "-L" ......
  = note: Undefined symbols for architecture x86_64:
            "_debug", referenced from:
                bindgen_test::main::hc0e5702b90adf92c in bindgen_test.3ccmhz8adio5obzw.rcgu.o
          ld: symbol(s) not found for architecture x86_64
          clang: error: linker command failed with exit code 1 (use -v to see invocation)
          

error: aborting due to previous error; 2 warnings emitted

error: could not compile `bindgen_test`.

我不确定为什么会失败,但我发现如果我删除整个 unsafe 块(不调用函数),编译将起作用。但是有人可以向我解释我做错了什么吗?是否需要添加一些内容才能使其编译?

非常感谢!

4

1 回答 1

0

问题是您没有在myLog.c任何地方包含文件,只有myLog.h标题。这就是这样bindgen做的:它将 C 头文件转换为 Rust 代码,但它不编译 C 代码本身。

为此,您需要cc箱子。您必须在文件cc中同时使用两者:bindgenbuild.rs

use std::env;
use std::path::PathBuf;

fn main() {
    println!("cargo:rerun-if-changed=myLog.h");
    println!("cargo:rerun-if-changed=myLog.c");  // new line here!!

    let bindings = bindgen::Builder::default()
        .header("myLog.h")
        .parse_callbacks(Box::new(bindgen::CargoCallbacks))
        .generate()
        .expect("Unable to build bindgen");
    let out_path = PathBuf::from(env::var("OUT_DIR").unwrap());
    bindings
        .write_to_file(out_path.join("mylog_bindings.rs"))
        .expect("Couldn't write bindings!");

    //Compile and link a static library named `myLog`:
    cc::Build::new()
        .file("myLog.c")
        .compile("myLog");
}

并且不要忘记将cc板条箱添加到您的build-dependencies.

于 2020-09-27T20:32:08.580 回答