尝试遵循FFI示例时出现“找不到功能”

时间:2019-05-04 04:02:27

标签: rust ffi

我试图开始使用Rust FFI,所以我去了the nomicon's documentation on FFI,并尝试复制“国外呼叫约定”示例。

我做了一些简单的代码,只是将示例复制粘贴,并尝试使用声明的函数:

extern crate libc;

#[cfg(all(target_os = "win32", target_arch = "x86"))]
#[link(name = "kernel32")]
#[allow(non_snake_case)]
extern "stdcall" {
    fn SetEnvironmentVariableA(n: *const u8, v: *const u8) -> libc::c_int;
}

fn main() {
    println!("Enter main");
    unsafe {
        let ret = SetEnvironmentVariableA("SOME_NEW_ENV_VAR\0", "NEW_ENV_VAR_VAL\0");
        println!("ret: {}", ret);
    }
    println!("Exit main");
}

这实际上并没有做任何有用的事情,我只是想看看我是否可以链接并调用一个函数而不会发生任何爆炸。

我从编译器收到此错误:

error[E0425]: cannot find function `SetEnvironmentVariableA` in this scope
  --> src\main.rs:13:19
   |
13 |         let ret = SetEnvironmentVariableA("SOME_NEW_ENV_VAR\0","NEW_ENV_VAR_VAL\0");
   |                   ^^^^^^^^^^^^^^^^^^^^^^^ not found in this scope

示例是否过时了?我是否缺少明显的东西?看来无论如何,我声明的SetEnvironmentVariableA应该在范围内。

我知道winapi板条箱,但是我正在尝试学习Rust FFI;使用现有的解决问题的板条箱将无法达到目的。

1 个答案:

答案 0 :(得分:0)

Stargateur的评论是正确的方向。删除以下行可解决问题:

#[cfg(all(target_os = "win32", target_arch = "x86"))]

在本教程的“外部调用约定”部分中使用了cfg attribute,因此我认为使链接和调用约定正常工作是魔术的一部分。根本不是cfg的工作方式。仅文档的URL解释了为什么我遇到“找不到函数”错误:“条件编译”。

我能够将示例扩展到某种问候世界:

#[link(name = "kernel32")]
#[allow(non_snake_case)]
extern "stdcall" {
    fn SetEnvironmentVariableA(n: *const u8, v: *const u8) -> i32;
    fn GetEnvironmentVariableA(n: *const u8, v: *mut u8, s : u32) -> i32;
}

fn main() {
    println!("Enter main");
    unsafe {
        let ret = SetEnvironmentVariableA("HELLO_WORLD_VAR\0".as_ptr(),"hellow world!\0".as_ptr());
        println!("ret: {}", ret);
        let mut dummy : u8 = 0;
        let ret1 = GetEnvironmentVariableA("HELLO_WORLD_VAR\0".as_ptr(), &mut dummy, 0);
        println!("ret1: {}", ret1);
        if ret1 > 0 {
            let mut read = vec![0; ret1 as usize];
            let ret2 = GetEnvironmentVariableA("HELLO_WORLD_VAR\0".as_ptr(),read.as_mut_ptr(), ret1 as u32);
            let read_str = String::from_utf8(read).unwrap();
            println!("ret2: {}", ret2);
            println!("read_str: {}", read_str);
            println!("read_str.len(): {}", read_str.len());
        }
    }
    println!("Exit main");
}