当前位置: 首页 > news >正文

Rust:实现仅通过索引(序数)导出 DLL 函数的功能

在 Rust 中,可以通过手动控制导出来实现仅通过索引(序数)导出 DLL 函数的功能。以下是具体方法和完整步骤:


解决方案

通过结合 .def 文件(模块定义文件)和 MSVC 链接器参数来实现函数名隐藏,只暴露序数编号。


具体步骤

1. 创建 Rust 动态库项目

Cargo.toml 中配置 cdylib 类型:

[lib]
crate-type = ["cdylib"]
2. 编写 Rust 函数

使用 #[no_mangle]extern "C" 定义导出函数:

// src/lib.rs
#[no_mangle]
pub extern "C" fn secret_function1() -> i32 {42
}#[no_mangle]
pub extern "C" fn secret_function2(x: i32) -> i32 {x * 2
}
3. 创建模块定义文件(.def

创建 exports.def 文件,用 NONAME 隐藏函数名并分配序数:

EXPORTS; 语法: 函数名 @序数 NONAMEsecret_function1 @1 NONAMEsecret_function2 @2 NONAME
4. 设置编译链接参数

修改 .cargo/config.toml,添加 MSVC 链接器标志:

# .cargo/config.toml
[target.x86_64-pc-windows-msvc]
rustflags = ["-C", "link-args=/DEF:exports.def /EXPORT:NONE"]

注意:路径需根据项目结构调整(也可用绝对路径)

5. 编译项目
cargo build --release

生成的 target/release/your_lib.dll 将隐藏函数名。


验证导出结果

使用 dumpbin 工具检查导出表(确保 VS Developer Command Prompt 中运行):

dumpbin /EXPORTS target/release/your_lib.dll

输出应类似:

ordinal hint RVA      name1    0 00001000 [NONAME]2    1 00001010 [NONAME]

从调用方通过索引加载

在 C/C++ 中通过序数加载函数(示例):

#include <windows.h>
#include <stdio.h>typedef int (*Func1)();
typedef int (*Func2)(int);int main() {HINSTANCE hDll = LoadLibraryA("your_lib.dll");if (!hDll) return 1;// 通过序数1加载第一个函数Func1 f1 = (Func1)GetProcAddress(hDll, (LPCSTR)1);// 通过序数2加载第二个函数Func2 f2 = (Func2)GetProcAddress(hDll, (LPCSTR)2);printf("f1: %d\n", f1());    // 输出 42printf("f2: %d\n", f2(10));  // 输出 20FreeLibrary(hDll);return 0;
}

关键点说明

  1. NONAME 关键字
    .def 文件中强制使用序数导出,隐藏函数名称。

  2. /EXPORT:NONE
    禁止 Rust 的默认名称导出规则,确保只有 .def 中的定义生效。

  3. 序数分配
    序数必须唯一且从 1 开始(序数 0 保留)。

  4. 工具链限制
    此方案仅适用于 MSVC 工具链x86_64-pc-windows-msvc)。如需 GNU 工具链,需改用 dlltool(流程较复杂)。


替代方案

如果需要在 GNU 工具链(如 x86_64-pc-windows-gnu)实现:

  1. 使用 dlltool 生成 .a 导入库
  2. 手动控制 .defdlltool 参数
    但 GNU 方案较为复杂,推荐优先使用 MSVC 链。

通过以上步骤,可在 Rust 中编译出仅通过索引导出的 DLL 文件,有效隐藏内部符号名称。

http://www.xdnf.cn/news/1320355.html

相关文章:

  • STM32单片机学习日记
  • 网络常识-SSE对比Websocket
  • 记一次安装OpenStack(Stein)-nova报错问题解决
  • 数据赋能(396)——大数据——抽象原则
  • 智能汽车领域研发,复用云原生开发范式?
  • 48.Seata认识、部署TC服务、微服务集成
  • http工作流程
  • C++算法竞赛:位运算
  • 前端项目练习-王者荣耀竞赛可视化大屏 -Vue纯前端静态页面项目
  • 服务器管理与配置学习总结
  • MYSQL-175. 组合两个表
  • JavaScript性能优化实战(四):资源加载优化
  • LeetCode 837.新 21 点:动态规划+滑动窗口
  • 【数据结构】堆和二叉树详解——上
  • 旋钮键盘项目---foc讲解(闭环位置控制)
  • 学习Python中Selenium模块的基本用法(5:程序基本步骤)
  • Linux817 shell:until,nfs,random
  • 力扣438:找到字符串中所有的字母异位词
  • Django前后端交互实现用户登录功能
  • [python学习记录2]变量
  • 脉冲计数实现
  • Docker之自定义jkd镜像上传阿里云
  • 排列组合+数量+资料
  • 25. 能否创建一个包含可变对象的不可变对象
  • 编程算法实例-Armstrong数(阿姆斯特朗数)
  • IDE/去读懂STM32CubeMX 时钟配置图(有源/无源晶振、旁路/晶振模式、倍频/分频)
  • 负载测试与压力测试详解
  • Rust Async 异步编程(五):执行器和系统 I/O
  • Spring 创建 Bean 的 8 种主要方式
  • MXFP4量化:如何在80GB GPU上运行1200亿参数的GPT-OSS模型