找回密码
 立即注册
首页 业界区 业界 LLVM Pass快速入门(四):代码插桩

LLVM Pass快速入门(四):代码插桩

甦忻愉 2026-2-4 18:10:23
代码插桩

项目需求:在函数运行时打印出运行的函数名
项目目录如下
  1. /MyProject
  2. ├── CMakeLists.txt # CMake 配置文件
  3. ├── build/ #构建目录
  4. │   └── test.c #测试编译代码
  5. └── mypass3.cpp # pass 项目代码
复制代码
一,测试代码示例

test.c
  1. #include <stdio.h>
  2. void func_A() {
  3.     int a = 1;
  4. }
  5. void func_B() {
  6.     func_A();
  7. }
  8. int main() {
  9.     printf("#把.c文件编译为.ll
  10. #-O1 使用O1优化(这里我尝试-O0不优化,会导致我的pass无法应用)
  11. #-Xclang -disable-llvm-passes 不使用默认的pass优化
  12. clang -S -emit-llvm -O1 -Xclang -disable-llvm-passes test.c -S -o test.ll
  13. #使用pass
  14. opt -load-pass-plugin=mypass3.dll -passes=mypass3  test.ll -S -o test_opt.ll
  15. #编译使用pass后的exe
  16. clang test_opt.ll -o test_opt.exe
  17. #编译使用pass前的exe
  18. clang test.ll -o test.exe\n");
  19.     func_A();
  20.     func_B();
  21.     return 0;
  22. }
复制代码
二,编写Pass

其他的固定的模板之前文章注释有,这里我只注释当前项目重要的部分
代码流程: 遍历指令并匹配ADD指令->替换为sub指令
  1. #include "llvm/IR/PassManager.h"
  2. #include "llvm/Passes/PassBuilder.h"
  3. #include "llvm/Passes/PassPlugin.h"
  4. #include "llvm/Support/raw_ostream.h"
  5. #include "llvm/IR/Function.h"
  6. #include "llvm/IR/BasicBlock.h"
  7. #include "llvm/IR/Instruction.h"
  8. #include "llvm/IR/Instructions.h"
  9. #include "llvm/IR/IRBuilder.h"
  10. using namespace llvm;
  11. namespace {
  12. struct mypass3 : public PassInfoMixin<mypass3> {
  13.    
  14.     PreservedAnalyses run(Function &F, FunctionAnalysisManager &) {
  15.         //过滤函数
  16.         //过滤掉printf和printf有关的函数,防止在printf中插入printf造成递归(死循环)
  17.         if(F.isDeclaration() || F.getName().starts_with("_") || F.getName().contains("printf")){
  18.             return PreservedAnalyses::all();
  19.         }
  20.         errs() << "handle func:" << F.getName() << "\n";
  21.                
  22.                 //获取模块
  23.         Module *M = F.getParent();
  24.         //获取模块上下文
  25.         //上下文中包含了数据的类型
  26.         LLVMContext &Ctx = M->getContext();
  27.         
  28.         //下面是创建函数,类比java反射,或者frida的hook
  29.         //定义printf的参数类型,相当于函数括号中的内容,这里的PointerType是指针类型
  30.         std::vector<Type*> printfArgs = {PointerType::getUnqual(Ctx)};
  31.                 //定义函数类型,这里相当于定义:int (void*, ...)
  32.         FunctionType *printfType = FunctionType::get(
  33.             Type::getInt32Ty(Ctx),//函数返回值类型
  34.             printfArgs,//函数的参数类型(vector)
  35.             true//是否是可变参数
  36.         );
  37.                 //如果printf存在则引用,如果不存在,则创建一个新的printf
  38.         FunctionCallee printfFunc = M->getOrInsertFunction("printf", printfType);
  39.                
  40.                 //下面是插入函数
  41.                 //将修改的位置定位到要插桩函数的头部
  42.         IRBuilder<> builder(&F.getEntryBlock().front());
  43.         //声明全局变量(这里是要传给printf的格式化字符串)
  44.         Value* formatStr = builder.CreateGlobalStringPtr(">> enter function %s <<\n", "my_format");
  45.         //声明全局变量,这里定义了函数名称的字符串变量
  46.         Value* funcName = builder.CreateGlobalStringPtr(F.getName(), "my_func_name");
  47.         //将上面定义的实际参数传入
  48.         std::vector<Value*> printfArgsVec = {formatStr, funcName};
  49.         //创建函数调用
  50.         builder.CreateCall(printfFunc, printfArgsVec);
  51.         
  52.         return PreservedAnalyses::none();
  53.     }
  54. };
  55. }
  56. extern "C" LLVM_ATTRIBUTE_WEAK ::llvm::PassPluginLibraryInfo
  57. llvmGetPassPluginInfo() {
  58.     return {
  59.         LLVM_PLUGIN_API_VERSION,
  60.         "mypass3",
  61.         "v0.1",
  62.         [](PassBuilder &PB) {
  63.             PB.registerPipelineParsingCallback(
  64.                 [](StringRef Name, FunctionPassManager &FPM,
  65.                    ArrayRef<PassBuilder::PipelineElement>) {
  66.                     if (Name == "mypass3") {
  67.                         FPM.addPass(mypass3());
  68.                         return true;
  69.                     }
  70.                     return false;
  71.                 });
  72.         }};
  73. }
复制代码
2.编译并构建Pass

打开visual studio的工作台,我这里是x64 Native Tools Command Prompt for VS 2022`
进到build目录
  1. #cmake 版本,可通过 cmake --version 判断
  2. cmake_minimum_required(VERSION 4.1.1) #---->修改 cmake版本号
  3. #项目名字
  4. project(mypass3) #---->修改 项目名称
  5. #导入项目的 LLVM cmake 配置文件路径(如果根据我之前文章安装这里就相同)
  6. set(LLVM_DIR "D:/LLVM/llvm-project/build/lib/cmake/llvm")#---->修改 llvm cmake配置路径
  7. #寻找 LLVM 的包文件
  8. #REQUIRED 找不到 LLVM 则停止构建
  9. #强制使用 LLVM 安装时生成的配置文件进行定位
  10. find_package(LLVM REQUIRED CONFIG)
  11. #将 LLVM 的 CMake 模块路径添加到当前 CMake 搜索路径中,以便后续使用 include(AddLLVM)。
  12. list(APPEND CMAKE_MODULE_PATH "${LLVM_CMAKE_DIR}")
  13. #引入 LLVM 提供的专用 CMake 宏
  14. include(AddLLVM)
  15. #将 LLVM 的头文件目录(如 llvm/IR/Function.h)加入编译器的搜索路径
  16. include_directories(${LLVM_INCLUDE_DIRS})
  17. #导入 LLVM 编译时使用的宏定义
  18. add_definitions(${LLVM_DEFINITIONS})
  19. #设置 C++ 标准为 C++17。(这里如果不用17编译会报错)
  20. set(CMAKE_CXX_STANDARD 17)
  21. #强制要求必须支持 C++17,如果编译器不支持则失败。
  22. set(CMAKE_CXX_STANDARD_REQUIRED ON)
  23. #创建一个模块化的库(.dll)
  24. add_library(mypass3 MODULE mypass3.cpp) #---->修改 项目名称,文件名
  25. #windows不用会报错:导出符号
  26. #LLVM Pass 需要暴露一些特定的入口点(如 getAnalysisUsage)给 opt 工具调用。
  27. set_target_properties(mypass3 PROPERTIES WINDOWS_EXPORT_ALL_SYMBOLS ON) #---->修改 项目名称
  28. # 指定该 Pass 需要链接的 LLVM 核心组件。
  29. # LLVMCore: 提供 IR、Function、Module 等核心类。
  30. # LLVMSupport: 提供各种辅助工具类(如 errs() 输出)。
  31. target_link_libraries(mypass3 LLVMCore LLVMSupport) #---->修改 项目名称,文件名  
  32. # 为该目标设置特定的编译器选项。
  33. # /utf-8: 告诉 MSVC 编译器使用 UTF-8 编码处理源代码,防止中文注释引起的乱码或编译错误。  
  34. target_compile_options(mypass3 PRIVATE /utf-8)#---->修改 项目名称,文件名
复制代码
最后出现下面提示,即为编译成功
  1. #构建项目
  2. #其中-DCMAKE_BUILD_TYPE=RelWithDebInfo不选会报错,由于我之前编译的是带符号的relase版本
  3. cmake -G "Ninja"  -DCMAKE_BUILD_TYPE=RelWithDebInfo ..
  4. #编译
  5. ninja
复制代码
四,使用插桩Pass对源码进行插桩

进到build目录
  1. [2/2] Linking CXX shared module mypass3.dll
复制代码
输出结果
运行test.exe:不使用pass,输出结果如下:
  1. #把.c文件编译为.ll
  2. #-O1 使用O1优化(这里我尝试-O0不优化,会导致我的pass无法应用)
  3. #-Xclang -disable-llvm-passes 不使用默认的pass优化
  4. clang -S -emit-llvm -O1 -Xclang -disable-llvm-passes test.c -S -o test.ll
  5. #使用pass
  6. opt -load-pass-plugin=mypass3.dll -passes=mypass3  test.ll -S -o test_opt.ll
  7. #编译使用pass后的exe
  8. clang test_opt.ll -o test_opt.exe
  9. #编译使用pass前的exe
  10. clang test.ll -o test.exe
复制代码
运行test_opt.exe:使用pass后,输出结果如下:
[code]>> enter function main > enter function func_A > enter function func_B > enter function func_A

相关推荐

2026-2-6 06:05:45

举报

2026-2-6 11:59:07

举报

2026-2-8 09:12:21

举报

2026-2-9 03:12:15

举报

2026-2-10 07:12:21

举报

7 天前

举报

7 天前

举报

6 天前

举报

前天 07:54

举报

喜欢鼓捣这些软件,现在用得少,谢谢分享!
您需要登录后才可以回帖 登录 | 立即注册