news 2026/5/9 4:36:08

LLVM Pass快速入门(二):运行第一个pass

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
LLVM Pass快速入门(二):运行第一个pass

认识Pass层级结构

Pass范围从上到下一共分为5个层级:

  • 模块层级:单个.ll.bc文件
  • 调用图层级:函数调用的关系。
  • 函数层级:单个函数。
  • 基本块层级:单个代码块。例如C语言中{}括起来的最小代码。
  • 指令层级:单个IR指令。

注意:下面代码最好不要用中文,使用起来非常麻烦,控制台,编译,目标文件的编码不同会造成乱码。

项目目录如下

/MyProject ├── CMakeLists.txt # CMake 配置文件 ├── build/ #构建目录 │ └── test.c #测试编译代码 └── mypass1.cpp # pass 项目代码

一,测试代码示例

test.c

/* by 01130.hk - online tools website : 01130.hk/zh/blood.html */ #include <stdio.h> void secret_function() { printf("I am secret\n"); } int main() { secret_function(); return 0; }

二,Pass编写

项目描述:通过解析下面代码的IR,将下面代码中的函数名打印出来。
mypass1.cpp

/* by 01130.hk - online tools website : 01130.hk/zh/blood.html */ #include "llvm/IR/PassManager.h" #include "llvm/Passes/PassBuilder.h" #include "llvm/Passes/PassPlugin.h" #include "llvm/Support/raw_ostream.h" using namespace llvm; // 这个结构体基本是固定模板 namespace { struct mypass1 : public PassInfoMixin<mypass1> { //函数回调,每次遇到函数时调用(这里有重载,存在多种入口方式,可以以模块为入口) PreservedAnalyses run(Function &F, FunctionAnalysisManager &) { //这里是主要的逻辑代码,我们主要学习的代码在这 //打印出当前函数的名字 errs() << "Found Function: " << F.getName() << "\n"; //只读时返回:PreservedAnalyses::all() //存在修改时:PreservedAnalyses::none() return PreservedAnalyses::all(); } }; } //下面基本上是固定的模板,每个pass没什么变化,可以直接复制粘贴,或者背熟。 //直接使用需要修改的是下面的<模块名称,版本号,调用参数,和调用的pass结构体> extern "C" LLVM_ATTRIBUTE_WEAK::llvm::PassPluginLibraryInfo llvmGetPassPluginInfo() { return { LLVM_PLUGIN_API_VERSION,//版本信息 "mypass1",//模块信息,自定义 "v0.1",//版本号,自定义 [](PassBuilder &PB) { PB.registerPipelineParsingCallback( [](StringRef Name, FunctionPassManager &FPM, ArrayRef<PassBuilder::PipelineElement>) { if (Name == "mypass1") {//调用参数 FPM.addPass(mypass1());//上面pass结构体 return true; } return false; } ); } }; }

三,Pass的构建

构建LLVM Pass需要写CMakeLists.txt构建声明

1. 配置CMake配置文件

CMakeLists.txt
下面的cmake配置可以直接拿去用,我已经标注好需要修改的位置

#cmake 版本,可通过 cmake --version 判断 cmake_minimum_required(VERSION 4.1.1) #---->修改 cmake版本号 #项目名字 project(mypass1) #---->修改 项目名称 #导入项目的 LLVM cmake 配置文件路径(如果根据我之前文章安装这里就相同) set(LLVM_DIR "D:/LLVM/llvm-project/build/lib/cmake/llvm")#---->修改 llvm cmake配置路径 #寻找 LLVM 的包文件 #REQUIRED 找不到 LLVM 则停止构建 #强制使用 LLVM 安装时生成的配置文件进行定位 find_package(LLVM REQUIRED CONFIG) #将 LLVM 的 CMake 模块路径添加到当前 CMake 搜索路径中,以便后续使用 include(AddLLVM)。 list(APPEND CMAKE_MODULE_PATH "${LLVM_CMAKE_DIR}") #引入 LLVM 提供的专用 CMake 宏 include(AddLLVM) #将 LLVM 的头文件目录(如 llvm/IR/Function.h)加入编译器的搜索路径 include_directories(${LLVM_INCLUDE_DIRS}) #导入 LLVM 编译时使用的宏定义 add_definitions(${LLVM_DEFINITIONS}) #设置 C++ 标准为 C++17。(这里如果不用17编译会报错) set(CMAKE_CXX_STANDARD 17) #强制要求必须支持 C++17,如果编译器不支持则失败。 set(CMAKE_CXX_STANDARD_REQUIRED ON) #创建一个模块化的库(.dll) add_library(mypass8 MODULE mypass8.cpp) #---->修改 项目名称,文件名 #windows不用会报错:导出符号 #LLVM Pass 需要暴露一些特定的入口点(如 getAnalysisUsage)给 opt 工具调用。 set_target_properties(mypass8 PROPERTIES WINDOWS_EXPORT_ALL_SYMBOLS ON) #---->修改 项目名称 # 指定该 Pass 需要链接的 LLVM 核心组件。 # LLVMCore: 提供 IR、Function、Module 等核心类。 # LLVMSupport: 提供各种辅助工具类(如 errs() 输出)。 target_link_libraries(mypass8 LLVMCore LLVMSupport) #---->修改 项目名称,文件名 # 为该目标设置特定的编译器选项。 # /utf-8: 告诉 MSVC 编译器使用 UTF-8 编码处理源代码,防止中文注释引起的乱码或编译错误。 target_compile_options(mypass8 PRIVATE /utf-8)#---->修改 项目名称,文件名
2.编译并构建Pass

打开visual studio的工作台,我这里是x64 Native Tools Command Prompt for VS 2022`

进到build目录

#构建项目 #其中-DCMAKE_BUILD_TYPE=RelWithDebInfo不选会报错,由于我之前编译的是带符号的relase版本 cmake -G "Ninja" -DCMAKE_BUILD_TYPE=RelWithDebInfo .. #编译 ninja

最后出现下面提示,即为编译成功

[2/2] Linking CXX shared module mypass1.dll

四,使用你第一个Pass

进到build目录

#把.c文件编译为.ll #-O1 使用O1优化(这里我尝试-O0不优化,会导致我的pass无法应用) #-Xclang -disable-llvm-passes 不使用默认的pass优化 clang -S -emit-llvm -O1 -Xclang -disable-llvm-passes test.c -S -o test.ll #使用pass opt -load-pass-plugin=mypass1.dll -passes=mypass1 test.ll -S -o test_opt.ll

输出结果

Found Function: sprintf Found Function: vsprintf Found Function: _snprintf Found Function: _vsnprintf Found Function: secret_function Found Function: printf Found Function: main Found Function: _vsprintf_l Found Function: _vsnprintf_l Found Function: __local_stdio_printf_options Found Function: _vfprintf_l

如果❤喜欢❤本系列教程,就点个关注吧,后续不定期更新~

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/4 4:43:51

DDoS 攻击深度解析:原理拆解与企业级防护方案实战

DDoS攻击&#xff08;分布式拒绝服务攻击&#xff09;是当今互联网世界最常见、最具破坏性的网络攻击之一。本文将深入解析其原理、类型、动机及防御策略。 一、核心概念&#xff1a;用简单比喻理解DDoS 通俗比喻&#xff1a; 想象一家热门咖啡馆&#xff08;你的网站/服务器…

作者头像 李华
网站建设 2026/4/27 7:38:58

【2026年美国大学生数学建模】ICM赛题Problem F: To Gen-AL, or Not To Gen-AI (or how to Gen-AD)? - 完整解题分析+完整源码+论文结构!

&#x1f393; 本文收录于 《滚雪球学数学建模》专栏&#xff0c;订阅即可获取2026年及历年数学建模笔记&#xff0c;每篇数十万字题解内容&#xff0c;且结合全球最新AI技术辅助&#xff0c;帮你轻松攻坚竞赛&#xff01;后续还将持续发布华为杯、高教社杯、华数杯、国赛、美赛…

作者头像 李华
网站建设 2026/5/2 18:09:20

<span class=“js_title_inner“>第二届开源产业大会专题分论坛“数字游民·OPC:重塑全球个人创新力与数字生产力”</span>

“数字游民OPC”重塑全球个人创新力与数字生产力&#xff0c;当开源、人工智能与全球流动叠加&#xff0c;人类正进入一个由“超级个体”驱动的新时代。个人不再只是组织的附属&#xff0c;而正在成为最小、最灵活、最具创造力的生产单元。 本次论坛聚焦“数字游民 OPC”&#…

作者头像 李华
网站建设 2026/5/9 3:08:26

python 面向对象基础入门

#定义教师类,将其作为具体教师对象的模板。该类中包含教师对象属性&#xff1a;name age gendar position属性。 #基于教师类&#xff0c;创建不同教师对象&#xff0c;要求其具有对应的姓名、年龄、性别和职务。 #1.init()方法 class Teacher: teacher_count0 …

作者头像 李华
网站建设 2026/5/1 16:08:20

引导型与文件型病毒:数字世界的两大经典威胁

引言&#xff1a;计算机病毒的两大“元老” 在计算机病毒的发展史上&#xff0c;引导型病毒和文件型病毒堪称两大元老级威胁。它们虽然基本原理不同&#xff0c;但都曾在计算机安全领域掀起过巨大波澜。理解这两类病毒的工作原理和历史演变&#xff0c;对我们认识计算机安全威…

作者头像 李华
网站建设 2026/4/23 17:22:54

P0951BA-0E FBP10处理器模块

P0951BA‑0E FBP10 处理器模块简介&#xff1a; P0951BA‑0E FBP10 处理器模块是用于工业自动化系统的核心运算单元&#xff0c;负责接收、处理和控制系统中各类输入输出信号&#xff0c;实现逻辑运算、数据处理和通讯任务。它在控制系统中承担主控角色&#xff0c;为复杂自动化…

作者头像 李华