编译过程-运行时机制

通常情况下,程序有两种执行模式

第一种执行模式是在物理机上运行。

针对的是 C、C++、Go 这样的语言,编译器直接将源代码编译成汇编代码(或直接生成机器码),然后生成能够在操作系统上运行的可执行程序。为了实现它们的后端,编译器需要理解程序在底层的运行环境,包括 CPU、内存、操作系统跟程序的互动关系,并要能理解汇编代码。

第二种执行模式是在虚拟机上运行。

针对的是 Java、Python、Erlang 和 Lua 等语言,它们能够在虚拟机上解释执行。这时候,编译器要理解该语言的虚拟机的运行机制,并生成能够被执行的 IR。

 

在计算机发展的早期,科学家们确立了计算机的结构,并一直延续至今,这种结构就是冯·诺依曼结构。它的主要特点是:数据和指令不加区别,混合存储在同一个存储器中(即主存,或叫做内存);用一个指令指针指向内存中指令的位置,CPU 就能自动加载这个位置的指令并执行。

 

 

 

计算机指令的执行基本上只跟两个硬件相关:一个是 CPU,一个是内存。

 

CPU 是计算机的核心。从硬件构成方面,我们需要知道它的三个信息:

第一,CPU 上面有寄存器,并且可以直接由指令访问。寄存器的读写速度非常快,大约是内存的 100 倍。所以我们编译后的代码,要尽量充分利用寄存器,而不是频繁地去访问内存。

第二,CPU 有高速缓存,并且可能是多级的。高速缓存也比内存快。CPU 在读取指令和数据的时候,不是一次读取一条,而是读取相邻的一批数据,放到高速缓存里。接下来要读取的数据,很可能已经在高速缓存里了,通过这种机制来提高运行性能。因此,编译器要尽量提高缓存的命中率。

第三,CPU 内部有多个功能单元,有的负责计算,有的负责解码,等等。所以,一个指令可以被切分成多个执行阶段,每个阶段在不同的功能单元上运行,这为实现指令级并行提供了硬件基础。

 

CPU 有多种不同的架构,比如 x86 架构、ARM 架构等。不同架构的 CPU,它的指令是不一样的。不过它们的共性之处在于,指令都是 01 这样的机器码。为了便于理解,我们通常会用汇编代码来表示机器指令


movl  -4(%rbp), %eax    #把%rbp-4内存地址的值拷贝到%eax寄存器
addl  $2, %eax          #把2加到%eax寄存器
movl  %eax, -8(%rbp)    #把%eax寄存器的值保存回内存,地址是%rbp-8

上面的汇编代码采用的是 GNU 汇编器规定的格式。每条指令都包含了两部分:操作码(opcode)和操作数(oprand)。

 

操作码是让 CPU 执行的动作。这段示例代码中,movl、addl 是助记符(Assembly Mnemonic),其中的 mov 和 add 是指令,l 是后缀,表示操作数的位数。而操作数是指令的操作对象,它可以是常数、寄存器和某个内存地址。示例的汇编代码中,“$2”就是个常数,在指令里我们把它叫做立即数;而“%eax”是访问一个寄存器,其中 eax 是寄存器的名称;而带有括号的“-4(%rbp)”,则是对内存的访问方式,这个内存的地址是在 rbp 寄存器的值的基础上减去 4。

 

操作系统是管理系统资源的,而 CPU 是计算机的核心资源,操作系统会把 CPU 的时间划分成多个时间片,分配给不同的程序使用,每个程序实际上都是在“断断续续”地使用 CPU,这就是操作系统的分时调度机制。

 

程序在运行时,操作系统会给它分配一块虚拟的内存空间,让它可以在运行期内使用。

内存中的每个位置都有一个地址,地址的长度决定了能够表示多大空间,这叫做寻址空间。

我们目前使用的都是 64 位的机器,理论上,你可以用一个 64 位的长整型来表示内存地址。

 

不过,由于我们根本用不了这么大的内存,所以 AMD64 架构的寻址空间只使用了 48 位。但这也有 256TB,远远超出了一般情况下的需求。所以,像 Windows 这样的操作系统还会给予进一步的限制,缩小程序的寻址空间。

程序在逻辑上可使用的内存一般也会大于实际的物理内存。不过进程不会一下子使用那么多的内存,只有在向操作系统申请内存的时候,操作系统才会把一块物理内存,映射成进程寻址空间内的一块内存。

 

 

对于已经分配给进程的内存,如果进程很长时间不用,操作系统会把它写到磁盘上,以便腾出更多可用的物理内存。在需要的时候,再把这块空间的数据从磁盘中读回来。这就是操作系统的虚拟内存机制。

 

本质上来说,你想怎么用就怎么用,并没有什么特别的限制。一个编译器的作者,可以决定在哪儿放代码,在哪儿放数据。当然了,别的作者也可能采用其他的策略。比如,C 语言和 Java 虚拟机对内存的管理和使用策略就是不同的。不过尽管如此,大多数语言还是会采用一些通用的内存管理模式。

以 C 语言为例,会把内存划分为代码区、静态数据区、栈和堆。

代码区(也叫做文本段),主要存放编译完成后的机器码,也就是 CPU 指令;静态数据区会保存程序中的全局变量和常量。这些内存是静态的、固定大小的,在编译完毕以后就能确定清楚所占用空间的大小、代码区每个函数的地址,以及静态数据区每个变量和常量的地址。这些内存在程序运行期间会一直被占用。而堆和栈,属于程序动态、按需获取的内存。

 

是指在调用一个函数时,如何传递参数、如何设定返回地址、如何获取返回值的这种约定,我们把它称之为 ABI(Application Binary Interface,应用程序二进制接口)。利用 ABI,使得我们可以用一种语言写的程序,去调用另外的语言写的程序。

另一些信息会保存在栈里。每个函数(或过程)在栈里保存的信息,叫做栈帧(Stack Frame)。我们可以自由设计栈帧的结构,比如,下图就是一种常见的设计

 

返回值:一般放在最顶上,这样它的地址是固定的。foo 函数返回以后,它的调用者可以到这里来取到返回值。在实际情况中,ABI 会规定优先通过寄存器来传递返回值,比通过内存传递性能更高。参数:在调用 foo 函数时,我们把它所需要一个整型参数写到栈帧的这个位置。同样,我们也可以通过寄存器来传递参数,而不是通过内存。控制链接:就是上一级栈帧(也就是 main 函数的栈帧)的地址。如果该函数用到了上一级作用域中的变量,那么就可以顺着这个链接找到上一级作用域的栈帧,并找到变量的值。返回地址: foo 函数执行完毕以后,继续执行哪条指令。同样,我们可以用寄存器来保存这个信息。本地变量: foo 函数的本地变量 b 的存储空间。寄存器信息:我们还经常在栈帧里保存寄存器的数据。如果在 foo 函数里要使用某个寄存器,可能需要先把它的值保存下来,防止破坏了别的代码保存在这里的数据。这种约定叫做被调用者责任,也就是使用寄存器的函数要保护好寄存器里原有的信息。某个函数如果使用了某个寄存器,但它又要调用别的函数,为了防止别的函数把自己放在寄存器中的数据覆盖掉,这个函数就要自己把寄存器信息保存在栈帧中。这种约定叫做调用者责任。

 

操作系统一般会提供一个 API,供应用申请内存。当应用程序用完之后,要通过另一个 API 释放。如果忘记释放,就会造成内存越用越少,这叫做内存泄漏。

相对于栈来说,这是堆的一个缺点。不过,相应的好处是,应用在堆里申请的对象的生存期,可以由自己控制,不会像栈里的内存那样,在退出作用域之后就被自动收回。所以,如果数据的生存期超过了创建它的作用域的生存期,就必须在堆中申请内存。

 

反之,如果数据的生存期跟创建它的作用域一致的话,那么在栈里和堆里申请都是可以的。当然,肯定在栈里申请更划算。所以,编译优化中的逃逸分析,本质就是分析出哪些对象的生存期是跟函数或方法的生存期一致的,那么就不需要到堆里申请了。

 

在并发的场景下,由于栈是线程独享的,而堆是多个线程共享的,所以在堆里申请内存的效率会更低,因为需要在多个线程之间同步,避免出现竞争。

 

为了避免内存泄漏,在设计一门语言的时候,通常需要提供内存管理的方案。

一种方案是像 C 和 C++ 那样,由程序员自己负责内存的释放,这对程序员的要求就比较高。另一种方案是,像 Java 语言那样自动地管理内存,这个特性也叫做垃圾收集。垃圾收集是语言的运行时功能,能够通过一定的算法来回收不用的内存。

 

对于把源代码编译成机器码在操作系统上运行的语言来说(比如 C、C++),操作系统本身就可以看做是它们的运行时系统。它可以帮助程序调度 CPU 资源、内存资源,以及其他一些资源,如 IO 端口。

 

在虚拟机上运行

虚拟机是计算机语言的一种运行时系统。虚拟机上运行的是中间代码,而不是 CPU 可以直接认识的指令。

虚拟机有两种模型:一种叫做栈机(Stack Machine),一种叫做寄存器机(Register Machine)。它们的区别,主要在于如何获取指令的操作数。栈机是从栈里获取,而寄存器机是从寄存器里获取。这两种虚拟机各有优缺点。

 

首先说说栈机。JVM 和 Python 中的解释器,都采用了栈机的模型。

JVM 中,每一个线程都有一个 JVM 栈,每次调用一个方法都会生成一个栈帧,来支持这个方法的运行。这跟 C 语言很相似。但 JVM 的栈帧比 C 语言的复杂,它包含了一个本地变量数组(包括方法的参数和本地变量)、操作数栈、到运行时常量池的引用等信息。

 

2+3*5

要从 AST 生成上面的代码,你只需要对 AST 做深度优先的遍历即可。先后经过的节点是:2->3->5->*->+(注:这种把操作符放在后面的写法,叫做逆波兰表达式,也叫后缀表达式)。

生成上述栈机代码,只需要深度优先地遍历 AST,并且只需要进行两种操作:在遇到字面量或者变量的时候,生成 push 指令;在遇到操作符的时候,生成相应的操作指令即可。

 

像 imul 和 iadd 这样的指令,不需要带操作数,因为指令所需的操作数就在栈顶。这是栈机的指令跟汇编语言的指令的最大区别。

imul 和 iadd 中的 i,代表这两个指令是对整型值做操作。对浮点型、长整型等不同类型,分别对应不同的指令前缀。

 

 

栈机之外,另一种虚拟机是寄存器机。寄存器机使用寄存器名称来表示操作数,所以它的指令也跟汇编代码相似,像 add 这样的操作码后面要跟操作数。

早期版本的安卓系统中,用于解释执行代码的 Dalvik 虚拟机,就采用了寄存器模式,而 Erlang 和 Lua 语言的虚拟机也是寄存器机。JavaScript 引擎 V8 的比较新的版本中,也引入了一个解释器 Ignition,它也是个寄存器机。

 

 

与栈机相比,利用寄存器机编译所生成的代码更少,因为省去了很多 push 指令。不过,寄存器机所指的寄存器,不一定是真正的物理寄存器,有可能只是栈帧中的一个位置。当然,有的寄存器机在实现的时候,确实会用到物理寄存器,从而提高计算性能。我们在后面研究 V8 的 Ignition 解释器时,会看到这种实现。

 

 

像 Java 等语言,既可以解释执行字节码,又能够即时编译成本地代码运行,所以它们的运行时机制就更复杂一些。你要综合两种运行时机制的知识,才能完整地理解 JVM。

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值