gitbook/编译原理之美/docs/151939.md
2022-09-03 22:05:03 +08:00

18 KiB
Raw Permalink Blame History

24 | 中间代码:兼容不同的语言和硬件

前几节课我带你尝试不通过IR直接生成汇编代码这是为了帮你快速破冰建立直觉。在这个过程中你也遇到了一些挑战比如
  • 你要对生成的代码进行优化,才有可能更好地使用寄存器和内存,同时也能减少代码量;

  • 另外针对不同的CPU和操作系统你需要调整生成汇编代码的逻辑。

这些实际体验,都进一步验证了20讲IR的作用我们能基于IR对接不同语言的前端也能对接不同的硬件架构还能做很多的优化。

既然IR有这些作用那你可能会问IR都是什么样子的呢有什么特点如何生成IR呢

本节课我就带你了解IR的特点认识常见的三地址代码学会如何把高级语言的代码翻译成IR。然后我还会特别介绍LLVM的IR以便后面使用LLVM这个工具。

首先来看看IR的特征。

介于中间的语言

IR的意思是中间表达方式它在高级语言和汇编语言的中间这意味着它的特征也是处于二者之间的。

与高级语言相比IR丢弃了大部分高级语言的语法特征和语义特征比如循环语句、if语句、作用域、面向对象等等它更像高层次的汇编语言而相比真正的汇编语言它又不会有那么多琐碎的、与具体硬件相关的细节。

相信你在学习汇编语言的时候,会发现汇编语言的细节特别多。比如,你要知道很多指令的名字和用法,还要记住很多不同的寄存器。在22讲我提到如果你想完整地掌握x86-64架构还需要接触很多指令集以及调用约定的细节、内存使用的细节等等参见Intel的手册)。

仅仅拿指令的数量来说据有人统计Intel指令的助记符有981个之多都记住怎么可能啊。所以说,汇编语言并不难,而是麻烦。

IR不会像x86-64汇编语言那么繁琐但它却包含了足够的细节信息能方便我们实现优化算法以及生成针对目标机器的汇编代码。

另外我在20讲提到IR有很多种类AST也是一种IR每种IR都有不同的特点和用途有的编译器甚至要用到几种不同的IR。

我们在后端部分所讲的IR目的是方便执行各种优化算法并有利于生成汇编。这种IR可以看做是一种高层次的汇编语言主要体现在

  • 它可以使用寄存器,但寄存器的数量没有限制;
  • 控制结构也跟汇编语言比较像,比如有跳转语句,分成多个程序块,用标签来标识程序块等;
  • 使用相当于汇编指令的操作码。这些操作码可以一对一地翻译成汇编代码,但有时一个操作码会对应多个汇编指令。

下面来看看一个典型IR三地址代码简称TAC。

认识典型的IR三地址代码TAC

下面是一种常见的IR的格式它叫做三地址代码Three Address Code, TAC它的优点是很简洁所以适合用来讨论算法

x := y op z   //二元操作
x := op y     //一元操作

每条三地址代码最多有三个地址其中两个是源地址比如第一行代码的y和z一个是目的地址也就是x每条代码最多有一个操作op

我来举几个例子,带你熟悉一下三地址代码,这样,你能掌握三地址代码的特点,从高级语言的代码转换生成三地址代码。

1.基本的算术运算:

int a, b, c, d;
a = b + c * d;

TAC

t1 := c * d
a  := b + t1

t1是新产生的临时变量。当源代码的表达式中包含一个以上的操作符时就需要引入临时变量并把原来的一条代码拆成多条代码。

2.布尔值的计算:

int a, b;
bool x, y;
x = a * 2 < b;
y = a + 3 == b;

TAC

t1 := a * 2;
x  := t1 < b;
t2 := a + 3;
y  := t2 == b;

布尔值实际上是用整数表示的0代表false非0值代表true。

3.条件语句:

int a, b c;
if (a < b )
    c = b;
else
    c = a;  
c = c * 2;      

TAC

  t1 := a < b;
  IfZ t1 Goto L1;
  c := a;
  Goto L2;
L1:
  c := b;
L2:
  c := c * 2;  

IfZ是检查后面的操作数是否是0“Z”就是“Zero”的意思。这里使用了标签和Goto语句来进行指令的跳转Goto相当于x86-64的汇编指令jmp

4.循环语句:

int a, b;
while (a < b){
  a = a + 1;
}
a = a + b;

TAC

L1:
  t1 := a < b;
  IfZ t1 Goto L2;
  a := a + 1;
  Goto L1;
L2:
  a := a + b;  

三地址代码的规则相当简单我们可以通过比较简单的转换规则就能从AST生成TAC。

在课程中,三地址代码主要用来描述优化算法,因为它比较简洁易读,操作(指令)的类型很少,书写方式也符合我们的日常习惯。**不过,我并不用它来生成汇编代码,因为它含有的细节信息还是比较少,**比如整数是16位的、32位的还是64位的目标机器的架构和操作系统是什么生成二进制文件的布局是怎样的等等

**我会用LLVM的IR来承担生成汇编的任务**因为它有能力描述与目标机器CPU、操作系统相关的更加具体的信息准确地生成目标代码从而真正能够用于生产环境。

**在讲这个问题之前我想先延伸一下讲讲另外几种IR的格式**主要想帮你开拓思维如果你的项目需求恰好能用这种IR实现到时不妨拿来用一下

  • 首先是四元式。它是与三地址代码等价的另一种表达方式格式是OParg1arg2result所以“a := b + c” 就等价于(+bca

  • 另一种常用的格式是逆波兰表达式。它把操作符放到后面所以也叫做后缀表达式。“b + c”对应的逆波兰表达式是“b c +”而“a = b + c”对应的逆波兰表达式是“a b c + =”。

**逆波兰表达式特别适合用栈来做计算。**比如计算“b c +”先从栈里弹出加号知道要做加法操作然后从栈里弹出两个操作数执行加法运算即可。这个计算过程跟深度优先的遍历AST是等价的。所以采用逆波兰表达式有可能让你用一个很简单的方式就实现公式计算功能**如果你编写带有公式功能的软件时可以考虑使用它。**而且从AST生成逆波兰表达式也非常容易。

三地址代码主要是学习算法的工具或者用于实现比较简单的后端要实现工业级的后端充分发挥硬件的性能你还要学习LLVM的IR。

认识LLVM汇编码

**LLVM汇编码LLVM Assembly是LLVM的IR。**有的时候我们就简单地称呼它为LLVM语言因此我们可以把用LLVM汇编码书写的一个程序文件叫做LLVM程序。

我会在下一讲详细讲解LLVM这个开源项目。本节课作为铺垫告诉我们在使用LLVM之前要先了解它的核心——IR。

首先LLVM汇编码是采用静态单赋值代码形式的。

在三地址代码上再加一些限制就能得到另一种重要的代码即静态单赋值代码Static Single Assignment, SSA在静态单赋值代码中一个变量只能被赋值一次来看个例子。

“y = x1 + x2 + x3 + x4”的普通三地址代码如下

y := x1 + x2;
y := y + x3;
y := y + x4;

其中y被赋值了三次如果写成SSA的形式就只能写成下面的样子

t1 := x1 + x2;
t2 := t1 + x3;
y  := t2 + x4; 

为什么要费力写成这种形式呢还要为此多添加t1和t2两个临时变量原因是SSA的形式体现了精确的“使用-定义”关系。

每个变量很确定地只会被定义一次然后可以多次使用。这种特点使得基于SSA更容易做数据流分析而数据流分析又是很多代码优化技术的基础所以几乎所有语言的编译器、解释器或虚拟机中都使用了SSA因为有利于做代码优化。而LLVM的IR也是采用SSA的形式也是因为SSA方便做代码优化。

**其次LLVM IR比起三地址代码有更多的细节信息。**比如整型变量的字长、内存对齐方式等等所以使用LLVM IR能够更准确地翻译成汇编码。

看看下面这段C语言代码

int fun1(int a, int b){
  int c = 10;
  return a + b + c;
}

对应的LLLM汇编码如下这是我在macOS上生成的

; ModuleID = 'fun1.c'
source_filename = "fun1.c"
target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128"
target triple = "x86_64-apple-macosx10.14.0"
; Function Attrs: noinline nounwind optnone ssp uwtable
define i32 @fun1(i32, i32) #0 {
  %3 = alloca i32, align 4        //为3个变量申请空间
  %4 = alloca i32, align 4     
  %5 = alloca i32, align 4
  store i32 %0, i32* %3, align 4  //参数1赋值给变量1
  store i32 %1, i32* %4, align 4  //参数2赋值给变量2
  store i32 10, i32* %5, align 4  //常量10赋值给变量3
  %6 = load i32, i32* %3, align 4 //
  %7 = load i32, i32* %4, align 4
  %8 = add nsw i32 %6, %7
  %9 = load i32, i32* %5, align 4
  %10 = add nsw i32 %8, %9
  ret i32 %10
}
attributes #0 = { noinline nounwind optnone ssp uwtable "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf" "no-infs-fp-math"="false" "no-jump-tables"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="false" "stack-protector-buffer-size"="8" "target-cpu"="penryn" "target-features"="+cx16,+fxsr,+mmx,+sahf,+sse,+sse2,+sse3,+sse4.1,+ssse3,+x87" "unsafe-fp-math"="false" "use-soft-float"="false" }

!llvm.module.flags = !{!0, !1, !2}
!llvm.ident = !{!3}

!0 = !{i32 2, !"SDK Version", [2 x i32] [i32 10, i32 14]}
!1 = !{i32 1, !"wchar_size", i32 4}
!2 = !{i32 7, !"PIC Level", i32 2}
!3 = !{!"Apple LLVM version 10.0.1 (clang-1001.0.46.4)"}

这些代码看上去确实比三地址代码复杂但还是比汇编精简多了比如LLVM IR的指令数量连x86-64汇编的十分之一都不到。

我们来熟悉一下里面的元素:

  • 模块

LLVM程序是由模块构成的这个文件就是一个模块。模块里可以包括函数、全局变量和符号表中的条目。链接的时候会把各个模块拼接到一起形成可执行文件或库文件。

在模块中你可以定义目标数据布局target datalayout。例如开头的小写“e”是低字节序Little Endian的意思对于超过一个字节的数据来说低位字节排放在内存的低地址端高位字节排放在内存的高地址端。

target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128"

“target triple”用来定义模块的目标主机它包括架构、厂商、操作系统三个部分。

target triple = "x86_64-apple-macosx10.14.0"

  • 函数

在示例代码中有一个以define开头的函数的声明还带着花括号。这有点儿像C语言的写法比汇编用采取标签来表示一个函数的可读性更好。

函数声明时可以带很多修饰成分比如链接类型、调用约定等。如果不写缺省的链接类型是external的也就是可以像23讲中做链接练习的那样暴露出来被其他模块链接。调用约定也有很多种选择缺省是“ccc”也就是C语言的调用约定C Calling Convention而“swiftcc”则是swift语言的调用约定。这些信息都是生成汇编时所需要的。

示例中函数fun1还带有“#0”的属性值定义了许多属性。这些也是生成汇编时所需要的。

  • 标识符

分为全局的Glocal和本地的Local全局标识符以@开头,包括函数和全局变量,前面代码中的@fun1就是本地标识符以%开头。

有的标识符是有名字的,比如@fun1或%a有的是没有名字的用数字表示就可以了如%1。

  • 操作码

alloca、store、load、add、ret这些都是操作码。它们的含义是

它们跟我们之前学到的汇编很相似。但是似乎函数体中的代码有点儿长。怎么一个简单的“a+b+c”就翻译成了10多行代码还用到了那么多临时变量不要担心**这只是完全没经过优化的格式,**带上优化参数稍加优化以后,它就会被精简成下面的样子:

define i32 @fun1(i32, i32) local_unnamed_addr #0 {
  %3 = add i32 %0, 10
  %4 = add i32 %3, %1
  ret i32 %4
}

  • 类型系统

汇编是无类型的。如果你用add指令它就认为你操作的是整数。而用fadd或addss指令就认为你操作的是浮点数。这样会有类型不安全的风险把整型当浮点数用了造成的后果是计算结果完全错误。

LLVM汇编则带有一个类型系统。它能避免不安全的数据操作并且有助于优化算法。这个类型系统包括基础数据类型、函数类型和void类型。

函数类型是包括对返回值和参数的定义比如i32 (i32)

void类型不代表任何值,也没有长度。

  • 全局变量和常量

在LLVM汇编中可以声明全局变量。全局变量所定义的内存是在编译时就分配好了的而不是在运行时例如下面这句定义了一个全局变量C

@c = global i32 100, align 4

你也可以声明常量,它的值在运行时不会被修改:

@c = constant i32 100, align 4

  • 元数据

在代码中你还看到以“!”开头的一些句子,这些是元数据。这些元数据定义了一些额外的信息,提供给优化器和代码生成器使用。

  • 基本块

函数中的代码会分成一个个的基本块可以用标签Label来标记一个基本块。下面这段代码有4个基本块其中第一个块有一个缺省的名字“entry”也就是作为入口的基本块这个基本块你不给它标签也可以。

define i32 @bb(i32) #0 {
  %2 = alloca i32, align 4
  %3 = alloca i32, align 4
  store i32 %0, i32* %3, align 4
  %4 = load i32, i32* %3, align 4
  %5 = icmp sgt i32 %4, 0
  br i1 %5, label %6, label %9

; <label>:6:                                      ; preds = %1
  %7 = load i32, i32* %3, align 4
  %8 = mul nsw i32 %7, 2
  store i32 %8, i32* %2, align 4
  br label %12

; <label>:9:                                      ; preds = %1
  %10 = load i32, i32* %3, align 4
  %11 = add nsw i32 %10, 3
  store i32 %11, i32* %2, align 4
  br label %12

; <label>:12:                                     ; preds = %9, %6
  %13 = load i32, i32* %2, align 4
  ret i32 %13
}

这段代码实际上相当于下面这段C语言的代码

int bb(int b){
    if (b > 0)
        return b * 2;
    else
        return b + 3;
}

每个基本块是一系列的指令。我们分析一下标签为9的基本块让你熟悉一下基本块和LLVM指令的特点

第一行(%10 = load i32, i32* %3, align 4的含义是把3号变量32位整型从内存加载到寄存器叫做10号变量其中内存对齐是4字节。

**我在这里延伸一下,**我们在内存里存放数据的时候有时会从2、4、8个字节的整数倍地址开始存。有些汇编指令要求必须从这样对齐的地址来取数据。另一些指令没做要求但如果是不对齐的比如是从0x03地址取数据就要花费更多的时钟周期。但缺点是内存对齐会浪费内存空间。

第一行是整个基本块的唯一入口,从其他基本块跳转过来的时候,只能跳转到这个入口行,不能跳转到基本块中的其他行。

第二行(%11 = add nsw i32 %10, 3的含义是把10号变量32位整型加上3保存到11号变量其中nsw是加法计算时没有符号环绕No Signed Wrap的意思。它的细节你可以查阅“LLVM语言参考手册”。

第三行store i32 %11, i32* %2, align 4的含义是把11号变量32位整型存入内存中的2号变量内存对齐4字节。

第四行br label %12的含义是跳转到标签为12的代码块。其中br指令是一条终结指令。终结指令要么是跳转到另一个基本块要么是从函数中返回ret指令基本块的最后一行必须是一条终结指令。

最后我要强调,从其他基本块不可以跳转到入口基本块,也就是函数中的第一个基本块。这个规定也是有利于做数据优化。

以上就是对LLVM汇编码的概要介绍更详细的信息了解可以参见“LLVM语言参考手册”

这样你实际上就可以用LLVM汇编码来编写程序了或者将AST翻译成LLVM汇编码。听上去有点让人犯怵因为LLVM汇编码的细节也相当不少好在LLVM提供了一个IR生成的API应用编程接口可以让我们更高效、更准确地生成IR。

课程小结

IR是我们后续做代码优化、汇编代码生成的基础在本节课中我想让你明确的要点如下

1.三地址代码是很常见的一种IR包含一个目的地址、一个操作符和至多两个源地址。它等价于四元式。我们在27讲和28讲中的优化算法会用三地址代码来讲解这样比较易于阅读。

2.LLVM IR的第一个特点是静态单赋值SSA也就是每个变量地址最多被赋值一次它这种特性有利于运行代码优化算法第二个特点是带有比较多的细节方便我们做优化和生成高质量的汇编代码。

通过本节课你应该对于编译器后端中常常提到的IR建立了直观的认识相信通过接下来的练习你一定会消除对IR的陌生感让它成为你得心应手的好工具

一课一思

我们介绍了IR的特点和几种基本的IR在你的领域比如人工智能领域你了解其他的IR吗它带来了什么好处欢迎分享你的经验和观点。

最后,感谢你的阅读,如果这篇文章让你有所收获,也欢迎你将它分享给更多的人。