深入计算机组成原理(二十)面向流水线的指令设计(上):一心多用的现代CPU

前面我们用了三讲,用一个个的电路组合,制作出了一个完整功能的CPU。这里面一下子给你引入了三个“周期”的概念,分别是指令周期、机器周期(或者CPU周期)以及时钟周期。

你可能会有点摸不着头脑,为什么小小一个CPU,有那么多的周期(Cycle)呢?我们在一开始,不是把CPU的性能定义得非常清楚了吗?我们说程序的性能,是由三个因素相乘来衡量的,我们还专门说过“指令数 x CPI x 时钟周期”这个公式。这里面和周期相关的只有一个时钟周期,也就是我们CPU的主频倒数。当时讲的时候我们说,一个CPU的时钟周期可以认为是可以完成一条最简单的计算机指令的时间。

那么,为什么我们在构造CPU的时候,一下子出来了那么多个周期呢?这一讲,我们就聊这个话题。

单指令周期处理器

学过前面三讲,你应该知道,一条CPU指令的执行,是由“取得指令(Fetch) - 指令译码(Decode) - 执行指令(Execute)”这样三个步骤组成的。这个执行过程,至少需要花费一个时钟周期。因为在取指令的时候,我们需要通过时钟周期的信号,来决定计数器的自增。

那么,很自然的,我们希望能确保让这样一整条指令的执行,在一个时钟周期内完成。这样,我们一个时钟周期可以执行一条指令,CPI也就是1,看起来就比执行一条指令需要多个时钟周期性能要好。采用这种设计思路的处理器,也叫做单指令周期处理器(Single Cycle Processor),也就是在一个时钟周期内,处理器正好能处理一条指令。

不过,我们的时钟周期是固定的,但是指令的电路复杂程度是不同的,所以实际一条指令执行的时间是不同的,在13讲和14讲讲加法器和乘法器电路的时候,我们讲过,随着门电路层数的增加,由于门延迟的存在,位数多、计算复杂的指令需要的执行时间就越长。

不同指令的执行时间不同,但是我们需要让所有指令都在一个时钟周期内完成,那就只好个时钟周期和执行时间最长的那个指令设成一样的。这就好比学校体育课1000米考试,我们要给考试预留的时间是和跑的最慢的同学时间一样。

深入计算机组成原理(二十)面向流水线的指令设计(上):一心多用的现代CPU_第1张图片

所以,在单指令周期处理器里面,无论是执行一条用不到ALU的无条件跳转指令,还是一条计算起来电路特别复杂的浮点数乘法运算,我们都要等满一个时钟周期。这个情况下,虽然CPI保持在1,但是我们的时钟频率却没法太高。因为太高的话,有些复杂指令没有办法在一个时钟周期内完成。那么在下一个时钟周期到来,开始执行下一条指令的时候,前一条指令的执行结果可能还没有写入到寄存器里面,那下一条指令独缺的数据就是不准确的,就会出现错误。

深入计算机组成原理(二十)面向流水线的指令设计(上):一心多用的现代CPU_第2张图片
到这里你会发现,这和我们在之前3讲和4讲讲时钟频率的时候不太一样。当时我们说,一个CPU时钟周期,可以认为是完成一条简单指令的时间,为什么到了这里,单指令周期处理器,反而变成了执行一条最复杂的指令的时间呢?

这是因为,无论PC上使用的Intel CPU,还是手机上使用的ARM CPU,都不是单指令周期处理器,而是采用了一种叫做指令流水线(Instruction Pipeline)的技术。

现代处理器的流水线设计

其实,CPU执行一条指令的过程和我们开发软件功能的过程很像。

如果我们想开发一个手机APP上的功能,并不是找来一个工程师,告诉他,“你把这个功能开发出来”,然后就能开发出来。真实情况是,无论只有一个工程师,还是有一个开发团队,我们都需要对开发功能的过程进行切分,把这个过程变成“撰写需求文档、开发后台API、开发客户端App、测试、发布上线”这样多个独立的过程。每一个后面的步骤,都要依赖前面的步骤。

我们的指令执行过程也是一样,它会拆分成“取指令、译码、执行”这样三大步骤。更细分一些的话,执行的过程,其实还包含从寄存器或者内存中读取数据,通过ALU进行运算,把结果写回到寄存器或者内存中。

如果我们有一个开发团队,我们不会让后端工程师开发完API之后,就等待前端开发、测试乃至发布,而是会在客户端App开发的同时,着手下一个需求的后端API开发。那么,同样的思路我们可以一样应用在CPU执行指令的过程中。

通过过去三讲,你应该以及知道了,CPU的指令执行过程,其实也是由各个电路模块组成的。我们在取指令的时候,需要一个译码器把数据从内存里面取出来,写入到寄存器中;在指令译码的时候,我们需要另外一个译码器,把指令解析成对应的控制信号、内存地址和数据;到了指令执行的时候,我们需要的则是一个完成计算工作的ALU。这些都是一个一个独立的组合逻辑电路,我们可以把他们看成一个团队里面的产品、后端、前端,共同来完成任务。

深入计算机组成原理(二十)面向流水线的指令设计(上):一心多用的现代CPU_第3张图片

这样一来,我们就不用把时钟周期设置成整条指令执行的时间,而是拆分成完成这样的一个一个小步骤需要的时间。同时,每一个阶段的电路在完成对应的任务之后,也不需要等待整个指令执行完成,而是可以直接执行下一条指令的对应阶段。

这就好像我们的后端程序员不需要等待功能上线,就会从产品手中拿到下一个需求,开始开发API。这样的协作模式,就是我们所说的指令流水线。这里面每一个独立的步骤,我们就称之为流水线阶段或者流水线级(Pipeline Stage)。

如果我们把一个指令拆分成“取指令- 指令译码 - 执行指令”这样三个部分,那这就是一个三级的流水线。如果我们进一步把“执行指令”拆分成“ALU计算(指令执行) - 内存访问 - 数据写回”,那么它就会变成一个五级的流水线。

五级的流水线,就表示我们在同一个时钟周期里面,同时运行五条指令的不同阶段。这个时候,虽然执行一条指令的时钟周期变成了5,但是我们可以把CPU的主频提的更高了。我们不需要确保最复杂的那条指令在时钟周期里面执行完成,而只要保障一个最复杂的流水线级的操作,在一个时钟周期内完成就好了

如果某一个操作步骤的时间太长,我们就可以考虑把这个步骤,拆分成更多的步骤,让所有步骤需要执行的时间尽量都差不多一样。这样,也就可以解决我们在单指令周期处理器中遇到的,性能瓶颈来自于最复杂的指令的问题。像我们的ARM或者Intel CPU,流水线级数都已经到了14级。

虽然我们不能通过流水线来减少单条指令执行的“延时”这个性能指标,但是,通过同时在执行多条指令的不同阶段,我们提升了CPU的“吞吐率”。在外部看来,我们的CPU好像是“一心多用”,在同一时间,同时执行五条不同指令的不同阶段。在CPU内部,其实它就像生产线一样,不同分工的组件不断处理上游传递下来的内容,而不需要等待单件商品生产完毕之后,再启动下一件商品的生产过程。

超长流水线的性能瓶颈

既然流水线可以增加我们的吞吐率,你可能就要问了,为什么我们不把流水线级数做的更深呢?为什么不做成20级、40级呢?这个其实有很多原因,我们在后面会详细介绍。这里,我先讲一个最基本的原因,就是增加流水线深度,其实是有性能成本的。

我们用来同步时钟周期的,不再是指令级别的,而是流水线阶段级别的。每一级流水线对应的输出,都要放到流水线按寄存器(Pipeline Register)里面,然后在下一个时钟周期,交给下一个流水线级去处理。所以,每增加一级的柳树西安,就要多一级写入到流水线寄存器的操作。虽然流水线寄存器非常快,比如只有20皮秒。

深入计算机组成原理(二十)面向流水线的指令设计(上):一心多用的现代CPU_第4张图片
但是,如果我们不断加深流水线,这些操作占整个指令的执行时间的比例就会不断增加。最后,我们的性能瓶颈就会出现在这些overhead上。如果我们指令的执行有3纳秒,也就是3000皮秒。我们需要20级的流水线,那流水线寄存器的写入就需要花费400皮秒,占了超过10%。这也就意味着,单纯的增加流水线级数,不仅不能增加性能,反而会有更多的overhead的开销。所以,设计合理的流水线级数也是现代CPU中非常重要的一点。

总结延伸

讲到这里,相信你已经能够理解,为什么我们的CPU需要流水线设计,也能把每一个刘先睡阶段在干什么,和上一讲的整个CPU的数据通路的连接过程对上了。

可以看到,为了能够不浪费CPU的性能,我们通过把指令的执行过程,切分成一个一个流水线级,来提升CPU的吞吐率。而我们本身的CPU的设计,又是由一个个独立的组合逻辑电路串接起来形成的,天然能够适合这样采用流水线“专业分工”的工作方式。

因为每一级的overhead,一味的增加流水线深度,并不能无限的提升性能。同样的,因为指令的执行不再是顺序的一条条执行,而是在上一条执行到一半的时候,下一条就可能已经开启了。所以也为我们的程序带来了很多的挑战。这些挑战和对应的解决方案,在下几讲再说。

思考

我们在前面讲过,一个CPU的时钟周期,可以认为是完成一条简单指令的时间。在这一讲之后,你觉得这句话正确吗?为什么?在了解了CPU的流水线设计之后,你是怎么理解这句话的?

你可能感兴趣的:(计算机组成原理,笔记)