第六章 中间代码生成.docVIP

  1. 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
  2. 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  3. 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  4. 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  5. 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  6. 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  7. 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
第六章 中间代码生成 在编译器的分析-综合模型中,前端对源程序进行分析并产生中间表示,后端在此基础上生成目标代码。在理想情况下,和源语言相关的细节在前端分析中处理,而关于目标机器的细节则在后端处理。基于一个适当定义的中间表示形式,可以把针对源语言i的前端和针对目标机器j的后端组合起来,构造得到源语言i在目标机器j上的一个编译器。这种创建编译器组合的方法可以节约大量的工作量:只要写出m种前端和n种后端处理程序,就可以得到m×n种编译程序。 本章的内容处理中间代码表示、静态类型检查和中间代码生成。为简单起见,我们假设一个编译程序的前端处理按照图6.1所示方式进行组织,顺序地进行语法分析、静态检查和中间代码生成。有时候这几个过程也可以组合起来,在语法分析中一并完成。我们将使用第二章和第五章中的语法制导定义来描述类型检查和翻译过程。大部分的翻译方案可以基于第五章中给出的自顶向下或自底向上的技术来实现。所有的方案都可以通过生成并遍历抽象语法树来实现。 静态检查包括类型检查,保证运算符被作用于兼容的运算分量。静态检查还包括在语法分析之后进行的所有语法检查。例如,静态检查保证了C语言中的一条break指令必然位于一个while/for/switch语句之内。如果不存在这样的语句,静态检查将报告一个错误。 本章介绍的方法可以用于多种中间表示,包括抽象语法树和三地址代码。这两种中间表示方法都在本书的2.8节中介绍过。名为“三地址代码”的原因是这些指令的一般形式x=y op z具有三个地址:两个运算分量y和z,一个结果变量x。 在将给定源语言的一个程序翻译成特定的目标机器代码的过程中,一个编译器可能构造出一系列的中间表示,如图6.2所示。高层的表示接近于源语言,而低层的表示接近于目标机器。语法树是高层的表示,它刻画了源程序的自然的层次性结构,并且很适合进行诸如静态类型检查这样的处理。 低层的表示形式适合进行机器相关的处理任务,比如寄存器分配、指令选择等。通过选择不同的运算符,三地址代码既可以是高层的表示方式,也可以是低层的表示方式。从6.2.3节将可以看出,对表达式而言,语法树和三地址代码只是在表面上有所不同。对于循环语句,语法树表示了语句的各个组成部分,而三地址代码包含了标号和跳转指令,用来表示目标语言的控制流。 不同的编译程序对中间表示的选择和设计各有不同。中间表示可以是一种真正的语言,也可以是编译器的各个处理阶段共享的多个内部数据结构。C语言是一种程序设计语言。它具有很好的灵活性和通用性,可以很方便地把C程序编译成高效的机器代码,并且有很多C的编译器可用,因此C语言也常常被用作中间表示。早期的C++编译器的前端生成C代码,而把C编译器作为其后端。 6.1 语法树的变体 语法树中各个结点代表了源程序的构造;一个结点的所有子结点反映了该结点对应构造的有意义的组成成分。为表达式构建的无环有向图(Directed acyclic graph, 以后简称DAG)指出了表达式中的公共子表达式(多次出现的子表达式)。在本节我们将看到,可以用构造语法树的技术去构造DAG图。 6.1.1 表达式的有向无环图 和表达式的语法树类似,一个DAG图的叶子结点对应于原子运算分量,而内部结点对应于运算符。与语法树不同的是,如果DAG图中的一个结点N表示一个公共子表达式,则N可能有多个父结点。在语法树中,公共子表达式每出现一次,代表该公共子表达式的子树就会被复制一次。因此,DAG不仅更简洁地表示了表达式,而且可以为最终生成表达式的高效代码提供重要的信息。 例子6.1 :图6.3给出了下面的表达式的DAG图 a+a*(b-c)+(b-c)*d 叶子结点a在表达式中出现了两次,因此a有两个父结点。值得注意的是,结点“-”代表公共子表达式b-c的两次出现。该结点同样有两个父结点,表明该子表达式在子表达式a*(b-c)和(b-c)*d中两次被使用。尽管b和c在整个表达式中出现了两次,它们对应的结点都只有一个父结点,因为对它们的使用都出现在同样的公共子表达式b-c中 图6.4给出的SDD(语法制导定义)既可以用来构造语法树,也可以用来构造DAG图。它在例5.11中曾被用于构造语法树。在那时,函数lead和node每次被调用都会构造出一个新结点。要构造得到DAG图,这些函数就要在每次构造新结点之前首先检查是否已存在这样的结点。如果存在一个已被创建的结点,它们就返回这个结点。例如,在构造一个新结点Node(op,left,right)之前,我们首先检查是否已存在一个结点,该结点的标号为op,且其两个子结点为left和right。如果存在,Node函数返回这个已存在的结点;否则它创建一个新结点。 例子6.2: 图6.5给出了构造图6.3中所示DAG图的各个步骤。如上所述,函数node和

文档评论(0)

***** + 关注
实名认证
文档贡献者

本账号下所有文档分享可拿50%收益 欢迎分享

1亿VIP精品文档

相关文档