- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试使用 GCC 6.1.0 和 rustc 1.11 在 Windows 中使用 C 语言构建一个调用 Rust 函数的最小程序,最好使用
。这是我首先尝试的:#![no_std]
编译。 0-nightly (bb4a79b08 2016-06-15) x86_64-pc-windows-gnu
main.c
#include <stdio.h>
int sum(int, int);
int main()
{
printf("Sum is %d.\n", sum(2, 3));
return 0;
}
sum.rs
#![no_std]
#![feature(libc)]
extern crate libc;
#[no_mangle]
pub extern "C" fn sum(x: libc::c_int, y: libc::c_int) -> libc::c_int
{
x + y
}
然后我尝试运行:
rustc --crate-type=staticlib --emit=obj sum.rs
但是得到了:
error: language item required, but not found: `panic_fmt`
error: language item required, but not found: `eh_personality`
error: language item required, but not found: `eh_unwind_resume`
error: aborting due to 3 previous errors
好的,所以其中一些错误与 panic unwinding 有关。我发现了一个用于移除展开支持的 Rust 编译器设置,-C panic=abort
。使用它,关于 eh_personality
和 eh_unwind_resume
的错误消失了,但 Rust 仍然需要 panic_fmt
函数。所以我在 the Rust docs 找到了它的签名,然后我将其添加到文件中:
sum.rs
#![no_std]
#![feature(lang_items, libc)]
extern crate libc;
#[lang = "panic_fmt"]
pub fn panic_fmt(_fmt: core::fmt::Arguments, _file_line: &(&'static str, u32)) -> !
{ loop { } }
#[no_mangle]
pub extern "C" fn sum(x: libc::c_int, y: libc::c_int) -> libc::c_int
{
x + y
}
然后,我再次尝试构建整个程序:
rustc --crate-type=staticlib --emit=obj -C panic=abort sum.rs
gcc -c main.c
gcc sum.o main.o -o program.exe
但是得到了:
sum.o:(.text+0x3e): undefined reference to `core::panicking::panic::h907815f47e914305'
collect2.exe: error: ld returned 1 exit status
panic 函数引用可能来自 sum()
加法中的溢出检查。这一切都很好而且令人向往。根据this page ,我需要定义自己的 panic 函数来使用 libcore。但是我找不到关于如何这样做的说明:我应该为其提供定义的函数在文档中称为 panic_impl
,但是链接器提示 panic::h907815f47e914305
,不管它应该是什么。
使用 objdump
,我找到了缺失函数的名称,并将其破解到 C:
main.c
#include <stdio.h>
#include <stdlib.h>
int sum(int, int);
void _ZN4core9panicking5panic17h907815f47e914305E()
{
printf("Panic!\n");
abort();
}
int main()
{
printf("Sum is %d.\n", sum(2, 3));
return 0;
}
现在,整个程序编译链接成功,甚至可以正常运行。
如果我随后尝试在 Rust 中使用数组,则会生成另一种 panic 函数(用于边界检查),因此我也需要为其提供定义。每当我在 Rust 中尝试更复杂的东西时,就会出现新的错误。顺便说一下,panic_fmt
似乎永远不会被调用,即使确实发生了 panic 。
无论如何,这一切看起来都非常不可靠,并且与我通过 Google 可以找到的关于此事的所有信息相矛盾。有 this ,但我尝试按照说明进行操作,但无济于事。
这似乎是一件简单而基本的事情,但我无法让它以正确的方式工作。也许这是 Rust 的夜间错误?但我需要 libc
和 lang_items
。如何在没有展开或 panic 支持的情况下生成 Rust 目标文件/静态库?当它想要 panic 时,它可能应该只执行非法的处理器指令,或者调用我可以在 C 中安全定义的 panic 函数。
最佳答案
你不应该使用 --emit=obj
;只是 rustc --crate-type=staticlib -C panic=abort sum.rs
应该做正确的事情。 (这修复了 _ZN4core9panicking5panic17h907815f47e914305E 链接错误。)
要修复另一个链接错误,您需要正确编写 panic_fmt(注意 extern
的使用):
#[lang="panic_fmt"]
extern fn panic_fmt(_: ::core::fmt::Arguments, _: &'static str, _: u32) -> ! {
loop {}
}
有了这些更改,一切似乎都按照预期的方式工作。
您需要 panic_fmt 以便您可以决定在发生 panic 时要做什么:如果您使用 #![no_std]
,rustc 会假设没有标准库/libc/kernel,所以它可以'不要只是调用 abort() 或期望非法指令来做任何有用的事情。它应该以某种方式在稳定的 Rust 中公开,但我不知道是否有人正在努力稳定它。
你不需要使用#![feature(libc)]
来获取libc;你应该改用 crates.io 上发布的版本(或者你可以手动声明你需要的函数)。
关于c - Rust 和 C 与最小程序和 no_std 的链接问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37929165/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!