在最近的一个 Issue 中 (#9387),Databend 社区的朋友希望能够利用 PGO 技术构建性能优化的可执行文件。让我们一起来看一下如何使用 Rust 构建 PGO 优化后的 Databend 吧!
背景知识
Profile-guided optimization 是一种编译器优化技术,它会收集程序运行过程中的典型执行数据(可能执行的分支),然后针对内联、条件分支、机器代码布局、寄存器分配等进行优化。
引入这一技术的背景是:编译器中的静态分析技术能够在不执行代码的情况下考虑代码优化,从而提高编译产物的性能;但是这些优化并不一定能够完全有效,在缺乏运行时信息的情况下,编译器无法考虑到程序的实际执行情况。
PGO 可以基于应用程序在生产环境中的场景收集数据,从而允许优化器针对较热的代码路径优化速度并针对较冷的代码路径优化大小,为应用程序生成更快和更小的代码。
rustc 支持 PGO,允许创建内置数据收集的二进制文件,然后在运行过程中收集数据,从而为最终的编译优化做准备。其实现完全依赖 LLVM。
典型过程
构建 PGO 优化的二进制文件通常需要进行以下几步工作:
- 构建内置数据收集的二进制文件
- 运行并收集数据,数据会以 的形式存在
.proraw
- 将 文件转换为
.proraw
文件.prodata
- 根据 文件进行构建优化
.prodata
前置准备
运行过程中的收集到的数据最终需要使用
llvm-profdata
rustup
llvm-tools-preview
llvm-profdata
rustup component add llvm-tools-preview
安装之后的
llvm-profdata
PATH
~/.rustup/toolchains/<toolchain>/lib/rustlib/<target-triple>/bin/
具体步骤
这里并没有选用某个具体生产环境的工作负载,而是使用 Databend 的 SQL 逻辑测试作为一个示范。在性能上可能并不具有积极意义,但可以帮助我们了解如何进行这一过程。
特别提示: 提供给程序的数据样本必须在统计学上代表典型的使用场景; 否则,反馈有可能损害最终构建的整体性能。
-
清除旧数据
rm -rf /tmp/pgo-data
-
编译支持收集 profile 数据的 release,使用
可以将 PGO 编译标志应用到所有 crates 的编译中。RUSTFLAGS
RUSTFLAGS="-Cprofile-generate=/tmp/pgo-data" \
cargo build --release --target=x86_64-unknown-linux-gnu -
运行编译好的程序,实际情况下推荐使用符合生产环境典型工 作负载的数据集和查询。
- 通过脚本启动 Databend 单机节点,考虑到生产环境更多是以集群模式运行,也可以启动 Databend 集群。
- 导入数据集并运行典型的查询工作负载。
示例中选择执行 SQL 逻辑测试,仅供参考。
BUILD_PROFILE=release ./scripts/ci/deploy/databend-query-standalone.sh
ulimit -n 10000;ulimit -s 16384; cargo run -p sqllogictests --release -- --enable_sandbox --parallel 16 --no-fail-fast -
使用
合并 profile 数据llvm-profdata
llvm-profdata merge -o /tmp/pgo-data/merged.profdata /tmp/pgo-data
-
在 profile 数据指导下进行编译,其实可以注意到,两次编译都使用
标志,因为实际运行情况下,我们总是使用 release 构建的二进制。--release
RUSTFLAGS="-Cprofile-use=/tmp/pgo-data/merged.profdata -Cllvm-args=-pgo-warn-missing-function" \
cargo build --release --target=x86_64-unknown-linux-gnu -
再次运行编译好的程序,运行之前的工作负载以检查性能。
BUILD_PROFILE=release ./scripts/ci/deploy/databend-query-standalone.sh
ulimit -n 10000;ulimit -s 16384; cargo run -p sqllogictests --release -- --enable_sandbox --parallel 16 --no-fail-fast
参考资料
- https://doc.rust-lang.org/rustc/profile-guided-optimization.html
- https://en.wikipedia.org/wiki/Profile-guided_optimization
- https://learn.microsoft.com/en-us/cpp/build/profile-guided-optimizations?view=msvc-170
- https://clang.llvm.org/docs/UsersManual.html#profile-guided-optimization
订阅我们的新闻简报
及时了解功能发布、产品规划、支持服务和云服务的最新信息!