可执行文件压缩
前言
最近看Rust相关东西的时候看到一篇关于压缩可执行文件的文章。压缩可执行文件对嵌入式开发特别有用,但是延伸一下用来减少我们游戏行业里预编译的工具二进制包大小和Android/iOS的库也是蛮有用的。
原文见这里: https://jamesmunns.com/blog/tinyrocket/
基本流程
- Release编译,移除调试符号文件,开启最小化size优化(-Oz)
- 使用LLVM的全量LTO
- 使用xargo重新编译标准库(std)和核心库(core)(这个C/C++不容易模仿,而且编译选项十分难搞)
- 移除jemalloc(服务器程序还是留着比较好,内置的malloc实现一般碎片比较厉害。虽然C/C++默认也不是jemalloc,很多项目为了新能还是会用它)
- 移除panic的详情信息(这个仅适用于Rust)
- strip(由GNU的binutils提供),参考命令:
strip [二进制]
- UPX进一步压缩加壳
尝试改造优化
然后尝试使用上面的流程改造我们的 gmtools-cli 。原先我是直接开LTO+Release编译的,编出的文件大小为4.4MB(4520728字节)。
- Release编译只要构建命令用
cargo build --release
就可以了,开size优化需要加个配置选项 ``` [profile.release] opt-level = “z”
[profile.dev] opt-level = “z”
2. LTO加几个配置选项就行了
[profile.release] lto = true codegen-units = 1 incremental = false
[profile.dev] lto = true codegen-units = 1 incremental = false
3. 重新编译标准库(std)和核心库(core)比较麻烦,而且我本地的环境报```could not find native static library `c`, perhaps an -L flag is missing?```。原文里自己编译这两个库反而体积变大了,我就先忽略这个了
4. 这个要改源码和配置文件
首先是 **Cargo.toml** 里要增加:
[features] system-alloc = []
然后代码增加:
```rust
#![feature(global_allocator)]
#![feature(allocator_api)]
// When the `system-alloc` feature is used, use the System Allocator
#[cfg(feature = "system-alloc")]
mod allocator {
use std::heap::System;
#[global_allocator]
pub static mut THE_ALLOC: System = System;
}
// When the `system-alloc` feature is not used, do nothing,
// retaining the default functionality (using jemalloc)
#[cfg(not(feature = "system-alloc"))]
mod allocator {
#[allow(dead_code)]
pub static THE_ALLOC: () = ();
}
#[allow(unused_imports)]
use allocator::THE_ALLOC;
最后构建命令加 --features system-alloc
- 这个就是移除调试信息,把
[profile.release]
的配置 panic = “abort” 就可以了 ``` [profile.release] panic = “abort”
[profile.dev] panic = “abort” ```
- 直接执行
strip 二进制
即可 - 参考命令
upx --ultra-brute 二进制
最后执行完,成果很惊人。压缩完后的大小是274K(280264字节)。
来个更直观的对比。
对比项 |
压缩前 |
压缩后 |
---|---|---|
编译选项 |
release,opt-level=3,lto=true,codegen-units=3,panic=“unwind” |
release,opt-level=“z”,lto=true,codegen-units=1,panic=“abort”,strip |
原始编译结果 |
4.4MB(4520728字节) |
2.1MB(2187784字节) – 减少51.6% |
仅执行strip |
4.4MB(4520728字节) |
844K(863312字节) – 减少80.9% |
执行strip和upx |
4.4MB(4520728字节) |
274K(280264字节) – 减少93.8% |
其他C/C++的压缩
其实上面效果最大的是Release编译移除调试符号、strip和upx,这三项都可以直接用再C/C++项目里的。唯一不同的就是可以编译的时候保留调试符号,然后用 objcopy
来代替 strip
把调试符号导出来并且移除了。
关于UPX和WSL和Android
UPX的原理是压缩代码,然后加入一些初始化函数再运行时解压,以前被一些病毒拿来做加壳处理,所以可能有些杀毒软件会报。其实不用UPX只strip也有不错的压缩率了。
在WSL环境下,现在的版本不支持UPX压缩后的可执行程序,会报 exec format error
,但是马上要发布的春季更新后就支持了。 这里有个Issue说这个问题的 https://github.com/Microsoft/WSL/issues/330 。
Android下用UPX看到说需要几个小patch(我没试,这里只是记录一下):
- UPX需要二进制文件大于40K,如果不够大可以加个全局变量搞大这个.so。
- 在native代码中需要声明
extern "C" {void _init(void){}}
函数,用于在编译时生成 _init 段。(UPX要求二进制文件必须存在init段,但是android的.so可能没有)- 或者也可以自定义初始化代码,
extern "C" {void my_init(void){}}
,然后编译时在Android.mk
里加入LOCAL_LDFLAGS += -Wl,-init=my_init
。来指定自己的初始化加载函数
- 或者也可以自定义初始化代码,
我的博客即将搬运同步至腾讯云+社区,邀请大家一同入驻:https://cloud.tencent.com/developer/support-plan?invite_code=3n1gmsmrgq2ok
- 工具| 手把手教你制作信息收集器之端口扫描
- 厚土Go学习笔记 | 14. switch 的条件写的有点灵活,不过风格还是go的一贯风格
- Nodejs学习笔记(十四)— Mongoose介绍和入门
- 厚土Go学习笔记 | 13. 用循环和函数 实现Sqrt(x)
- 代码审计| 这是一款适合练手的漏洞
- 工具| NSE漏洞审计和渗透脚本的demo
- Windows Server 2008 R2 配置Exchange 2010邮件服务器并使用EWS发送邮件
- 厚土Go学习笔记 | 12. if 语句
- 厚土Go学习笔记 | 11. for循环 go语言只有for循环
- 从编译原理看一个解释器的实现
- 厚土Go学习笔记 | 10. 常量 与 数值常量
- Unity应用架构设计(7)——IoC工厂理念先行
- 厚土Go学习笔记 | 09. 类型转换 与 类型推导
- 厚积薄发,拥抱 .NET 2016
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Android自定义橡皮擦效果
- Android Canvas drawText文字居中的一些事(图解)
- Nginx+keepalived一主一从高可用,手把手带你一步一步配置!
- Behave!:一款针对页面活动的浏览器监控插件
- R可视化 | 地理信息空间(上)
- Nautilus:一款基于语法的反馈式模糊测试工具
- R可视化 | 地理信息空间(下)
- 21页优雅读博指南:佐治亚理工学院助理教授Eric Gilbert撰写,入坑前必读
- Too old resource version 引起 Flink JobManager 崩溃的问题定位
- SharpHose:一款基于C#开发的Windows异步密码喷射工具
- PyTorch版YOLOv4更新了,不仅适用于自定义数据集,还集成了注意力和MobileNet
- 一行代码不用写,就可以训练、测试、使用模型,这个star量1.5k的项目帮你做到
- Android画板开发之橡皮擦功能
- Android画板开发之基本画笔功能
- python递归函数求n的阶乘,优缺点及递归次数设置方式