我试图开始使用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;使用现有的解决问题的板条箱将无法达到目的。
答案 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");
}