北航2023年编译原理源代码
姓名:杜金阳
学号:21373191
完整SysY编译器代码见——SysY_Compiler_2023: 北航2023年编译原理源代码
编译器完整文法如下:
编译单元 CompUnit → {Decl} {FuncDef} MainFuncDef
声明 Decl → ConstDecl | VarDecl
常量声明 ConstDecl → 'const' BType ConstDef { ',' ConstDef } ';' // i
基本类型 BType → 'int'
常数定义 ConstDef → Ident { '[' ConstExp ']' } '=' ConstInitVal // b k
常量初值 ConstInitVal → ConstExp
| '{' [ ConstInitVal { ',' ConstInitVal } ] '}'
变量声明 VarDecl → BType VarDef { ',' VarDef } ';' // i
变量定义 VarDef → Ident { '[' ConstExp ']' } // b
| Ident { '[' ConstExp ']' } '=' InitVal // k
变量初值 InitVal → Exp | '{' [ InitVal { ',' InitVal } ] '}'
函数定义 FuncDef → FuncType Ident '(' [FuncFParams] ')' Block // b g j
主函数定义 MainFuncDef → 'int' 'main' '(' ')' Block // g j
函数类型 FuncType → 'void' | 'int'
函数形参表 FuncFParams → FuncFParam { ',' FuncFParam }
函数形参 FuncFParam → BType Ident ['[' ']' { '[' ConstExp ']' }] // b k
语句块 Block → '{' { BlockItem } '}'
语句块项 BlockItem → Decl | Stmt
语句 Stmt → LVal '=' Exp ';' | [Exp] ';' | Block // h i
| 'if' '(' Cond ')' Stmt [ 'else' Stmt ] // j
| 'for' '('[ForStmt] ';' [Cond] ';' [ForStmt] ')' Stmt
| 'break' ';' | 'continue' ';' // i m
| 'return' [Exp] ';' // f i
| LVal '=' 'getint''('')'';' // h i j
| 'printf''('FormatString{,Exp}')'';' // i j l
语句 ForStmt → LVal '=' Exp //h
表达式 Exp → AddExp 注:SysY 表达式是int 型表达式
条件表达式 Cond → LOrExp
左值表达式 LVal → Ident {'[' Exp ']'} // c k
基本表达式 PrimaryExp → '(' Exp ')' | LVal | Number
数值 Number → IntConst
一元表达式 UnaryExp → PrimaryExp | Ident '(' [FuncRParams] ')' // c d e j
| UnaryOp UnaryExp
单目运算符 UnaryOp → '+' | '−' | '!' 注:'!'仅出现在条件表达式中
函数实参表 FuncRParams → Exp { ',' Exp }
乘除模表达式 MulExp → UnaryExp | MulExp ('*' | '/' | '%') UnaryExp
加减表达式 AddExp → MulExp | AddExp ('+' | '−') MulExp
关系表达式 RelExp → AddExp | RelExp ('<' | '>' | '<=' | '>=') AddExp
相等性表达式 EqExp → RelExp | EqExp ('==' | '!=') RelExp
逻辑与表达式 LAndExp → EqExp | LAndExp '&&' EqExp
逻辑或表达式 LOrExp → LAndExp | LOrExp '||' LAndExp
常量表达式 ConstExp → AddExp 注:使用的Ident 必须是常量
格式字符串: <FormatString> → '"'{<Char>}'"' // a
本SysY编译器选择了采用Java语言进行编写,充分发挥了Java面向对象编程的特点,这样相比于C++更有利于大工程的设计,但也带来了代码冗长的问题,所以本编译器及其重视设计一个好的架构以减少不必要的重构,并使代码风格更加易读。
本编译器的总体设计结构参考了教程中的编译顺序,即词法分析、语法分析、错误处理、中间代码生成、生成mips代码、中端代码优化、后端代码优化几个部分,文件夹具体结构如下:
├─backend
│ ├─generation
│ │ ├─mips
│ │ │ └─asm
│ │ │ ├─datasegment
│ │ │ └─textsegment
│ │ │ ├─complex
│ │ │ ├─mipsinstr
│ │ │ └─structure
│ │ └─utils
│ └─simplify
│ └─method
├─frontend
│ ├─generation
│ │ ├─lexer
│ │ ├─semantic
│ │ │ ├─symtable
│ │ │ │ └─symbol
│ │ │ │ └─varsymbol
│ │ │ └─utils
│ │ └─syntax
│ │ ├─decl
│ │ ├─funcdef
│ │ ├─mainfuncdef
│ │ └─utils
│ └─simplify
│ └─method
├─iostream
│ ├─declare
│ └─structure
└─midend
├─generation
│ ├─llvm
│ ├─utils
│ │ └─irtype
│ └─value
│ ├─construction
│ │ ├─procedure
│ │ └─user
│ └─instr
│ ├─basis
│ └─optimizer
└─simplify
├─controller
│ └─datastruct
└─method
词法分析算是整个编译器设计较为简单的部分,承担的任务就是 通过扫描输入的源程序字符串,将其分割成一个个单词,同时记录这些单词的类别信息 。于此同时,词法分析还需要去除那些对于编译没用的符号,例如各种空白符和注释等。经过词法分析器的解析,我们就可以从词法分析器依次获取每个单词的信息,包括 单词值和单词类别 ,用于后续的编译。
对此,我在词法分析设计方面主要设计了三个类来辅助词法分析处理
LexicalAnalysis
(词法分析器)——用于保存保留字,并根据单词生成对应的类别码LexicalWordCheck
(词法分类检查)——用于分割语句形成对应的词法单词,并负责词法分析过程的错误处理SymToken
(标识符存储单位)——用于存储词法分析形成的信息,包括每个单词的类别码、值、所在行数等
具体而言,在Compiler
主类中从文件中读取完整的源码,然后按行输入LexicalAnalysis
进行分析,之后在LexicalWordCheck
中由if-else
分支语句进行类别判断,然后进入对应分支读取完整单词,得到一个SymToken
实例,不断重复上述操作直到读到源码末尾。
- 对于保留字采用
HashMap
进行查寻,保留字为键,类别码为值,当得到一个保留字单词就查询HashMap
获得类别码 - 对于注释空白符跳过不识别,即直接读到注释之后的第一个字符
- 对于行号记录,在
Compiler
每次循环前将行号加一
注:考虑到评测系统为
Linux
系统于是对于要跳过的空白符需要注意判断\r
的情况,当然也可以直接调用方法直接跳过空白符。
在词法分析部分,需要处理的词法分析错误只有一种—— 字符串中出现非法字符 。除下面三类以外,均为非法字符——
- 十进制编码为32,33,40-126的ASCII字符
- (编码92)出现当且仅当为'\n'
- <FormatChar>==>'%d'
判断使用正则表达式即可,注意要将%d和\n作为一个整体进行判断。
private boolean checkIllegalSym(String word) {
String legalSym = "\"([ !()*+,\\-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ\\[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~]|(%d)|(\\\\n))*\"";
return !word.matches(legalSym);
}
语法分析部分的主要任务是让我们按照课程组给定的文法构造相应的抽象语法树(AST
),并输出后序遍历AST
树的结果即可完成该部分任务。所以我采用在词法分析部分先进行一遍的扫描来获得词法分析分析出来的token
集合,这样虽然理论上并没有边读词法边分析文法效率高,但如果综合考虑由于文法的FIRST
集合的交集存在导致的词法预读现象,最终两种方式获得的分析效率其实相差无几,但预先分析出词法token
集合可以很好地减少编码出现错误的概率,同时也可以使整个编译过程更加结构清晰,架构完整,实现高内聚低耦合。
而具体到语法分析部分使用的方法,我采用了递归下降子程序的方法来构建AST
树,并通过设计Definer
类和Judge
类来避免了大量的重复造轮子的操作,具体操作细节请看下文。
在参加了编译大赛的同学那里了解到了一个神器——antlr4
,这是一个IDEA
的插件,通过它只要你给定设计好的词法和文法,再给定符合词法和文法的程序,你就可以直观的看到相应代码建立AST
树的结果,这将对我们更加深入理解文法分析的过程和之后的debug
工作做出巨大的贡献。
据说
antlr4
甚至可以直接帮你生成词法分析和语法分析程序,但这肯定是课程组不允许的,所以我就没有探索这部分的内容
下图为用antlr4
分析helloworld
程序构造的语法树(由于每年的文法可能有差异,可能会有细微的不同)。
由于我采用的是递归下降分析法,所以为了避免死循环的出现,一个非常重要的步骤就是去除文法中的左递归(左递归会导致死循环应该编译理论课已经讲过了),通过仔细分析课程组的文法,不难发现主要的左递归都出现在了Exp表达式那个部分中(如下图),所以我们要找到合适的方法改写这部分左递归。
关于如何修改左递归,课程组已经在教程和语法分析的专题讲座中给我们了一部分的思路,所以具体而言我采用了改写为BNF
范式并进行了少量修改以达到和课程组给定的AST
树结构相同的目的,下面以AddExp
和MulExp
举例。
原本存在左递归的文法是这样的。
AddExp : MulExp | AddExp ('+' | '−') MulExp
MulExp : UnaryExp | MulExp ('*' | '/' | '%') UnaryExp
所谓改写为BNF
范式,具体而言通过观察不难发现,AddExp
本质上是由若干个MulExp
组成,MulExp
本质上是由若干个UnaryExp
组成,所以改写后的文法是这样的。
AddExp : MulExp {('+' | '−') MulExp}
MulExp : UnaryExp {('*' | '/' | '%') UnaryExp}
这样我们就成功消除了左递归,但这就结束了吗,并不是,由于我们修改了文法,所以造成了我们的文法和课程组要求的AST
树出现了结构上的差异,所以我们在实际编写代码时还需要进行一些小的修改,比如在生成AddExp
时,可以在每次解析('+' | '−') MulExp
之前,先将之前已经解析出的若干个MulExp
合成一个AddExp
,最后再将最终的AddExp
加入调用该节点的父节点,具体到代码中实际就是一个不停更换父节点的过程,代码示例如下。
private static void AddExp(AstNode constExpNode) {
AstNode addExpNode = new AstNode("<AddExp>");
MulExp(addExpNode);
AstNode father = addExpNode;
while (getPreSym().equals("PLUS") || getPreSym().equals("MINU")) {
// Removed part unimportant code
father = extraAddExpNode;
if (getPreSym().equals("PLUS")) {
// Removed part unimportant code
MulExp(father);
} else if (getPreSym().equals("MINU")) {
// Removed part unimportant code
MulExp(father);
}
}
constExpNode.addChild(father);
}
这样左递归文法的问题就迎刃而解了。
在语法分析部分,我认为是编译过程一个非常重要的过程,它可以帮助我们将词法分析获得的token
集合进行进一步的分析,将其建立成结构化的语法树,以便于后续的编译。
而在实际编写代码的过程中,首先我建立了SyntaxAnalysis
(语法分析类)、AstNode
(抽象语法树节点类)两个类来负责语法树递归下降的调用和后序遍历的实现,并为建立AST
树打下基础。
具体到递归下降部分而言,我首先根据课程给定文法的开始符号CompUnit
的给定文法将建立了AstRecursion
类来模拟递归下降过程的开始,并根据起始文法(如下图)将CompUnit
分为了Decl
,FuncDef
和MainFuncDef
三个类,从而编写分别属于这几个部分不被共用的语法部分。
之后我又编写了Judge
类和Definer
类来定义了在递归下降过程中上述三个部分需要共用的语法成分的判断条件和具体操作。这样在结构比较清晰的情况下我们就很好地实现了递归下降结构的设计,之后只需要按照给定的文法定义相关结构即可,在此不再具体赘述。
在我的实际编码过程中,在处理Stmt
时,由于FIRST
集合存在交集,所以我们需要进行预读操作来将这两个成分分开,具体而言我定义了下面这个函数来获得相对于当前token
的位置前几个或后几个token
的具体值。
public static SymToken getNextSymToken(int pos) {
if (symPos + pos > symTokens.size() - 1 || symPos + pos < 0) {
return null;
}
return symTokens.get(symPos + pos);
}
这样我们就可以更加灵活的定义条件来避免FIRST集有交集的情况出现了。
在编写语法分析部分时,随着错误处理种类的不断增加,为了更好地提高错误处理的效率,所以我设计了**ErrorController
类并在类中提供了错误类别码来辅助我进行相关的错误不同种类的识别和精确处理**。而在语法分析方面,我们主要要处理三类错误——缺少分号、缺少右小括号和缺少右中括号,这三种错误处理要求如下表。(注意是报错行号是给定符号前一个非终结符所在的行号)
错误类型 | 错误类别码 | 解释 | 对应文法 |
---|---|---|---|
缺少分号 | i | 报错行号为分号前一个非终结符所在行号。 | <Stmt>,<ConstDecl>及<VarDecl>中的';’ |
缺少右小括号 | j | 报错行号为右小括号前一个非终结符所在行号。 | 函数调用(<UnaryExp>)、函数定义(<FuncDef>)及<Stmt>中的')’ |
缺少右中括号’]’ | k | 报错行号为右中括号前一个非终结符所在行号。 | 数组定义(<ConstDef>,<VarDef>,<FuncFParam>)和使用(<LVal>)中的']’ |
这三种错误处理还是比较简单的,只需要在文法读到相应的部分去申请一个ErrorController
对象并填入相应的错误类别码即可,还是比较简单的。
错误分析部分的主要任务是通过建立并管理符号表,从而实现对语义分析的控制,进而控制判断所建立的语法树中出现的语义错误。
语法错误部分已经在之前的词法分析和语法分析中处理完成
要想建立行之有效的语义分析模块,一个非常重要的先决条件就是建立符号表,本文设计的符号表建立在栈式符号表的基础上,通过下标模拟入栈退栈来实现了实际为“类树状结构”的栈式符号表。
符号表由Symbol
、StackSymbolTable
、SymbolTable
三部分组成。
本文创建了一个Symbol
这一符号类,用于存储符号的基本信息。然后由该类派生出VarSymbol
、ConstSymbol
、FuncSymbol
三个子类,用来存储不同类型符号的信息——
VarSymbol
:变量名,变量值类型(int),变量维度,变量各维度的长度,变量初始值。ConstSymbol
:常量名,常量值类型(int),常量维度,常量各维度的长度,常量初始值。FuncSymbol
:函数名,函数返回值类型(int/void),函数各形参类型,函数各形参维度。
通过符号表可以算出全局常量的具体值,与此同时全局变量的初始值可以在编译阶段计算出,因此对于非全局的常量和变量,其“变量初始值”一项为null
public class Symbol {
public enum SymType {
INT, VOID, CONST
}
private String symbolName;
private SymType symbolType;
private Integer symbolLevel;
}
- 这里设计了枚举类
SymType
来区分函数的类型(VOID
,INT
)和变量的类型(INT
,CONST
),由于有变量维度dim的存在,所以不需要额外设计ARRAY
和CONST_ARRAY
类型来区分变量类型和数组类型。
栈式符号表类StackSymbolTable
可以是一层栈式符号表,存储了属于当前作用域的所有符号。
public class StackSymbolTable {
private HashMap<String, Symbol> symbols;
}
symbols
的第一维存储的是这个作用域某个symbol
的name
,通过名字可以索引出这个对象。
符号表类SymbolTable
从全局角度管理了属于该程序的所有符号表,方便各层栈式符号表的入栈出栈及从全局的角度实现了各symbol
的插入删除需求。
public class SymbolTable {
private static HashMap<Integer, ArrayList<StackSymbolTable>> symbolTables;
private static boolean isGlobalArea;
private static int loopLevel;
private static FuncSymbol currentFunc;
private static int curLevel;
}
symbolTables
第一维存储的是某栈式符号表所在的维数信息,第二维是一个数组存储了在这一维所有的栈式符号表信息。isGlobalArea
记录了此时的ConstDecl
和VarDecl
是否为全局变量,从而决定是否适用symCalc
调用符号表算出对应的值。loopLevel
用于记录当前所处的循环维数,从而判断此时出现break
和continue
是否符合语义。currentFunc
用于记录当前正在解析的函数所对应的符号,用与检查函数中是否出现不匹配的return
语句。curLevel
用于记录当前symbolTable
所处的维数信息。
错误类型 | 错误类别码 | 解释 | 对应文法及出错符号 ( … 表示省略该条规则后续部分) |
---|---|---|---|
非法符号 | a | 格式字符串中出现非法字符报错行号为 <FormatString> 所在行数。 | <FormatString> → ‘“‘{<Char>}’” |
名字重定义 | b | 函数名或者变量名在当前作用域下重复定义。注意,变量一定是同一级作用域下才会判定出错,不同级作用域下,内层会覆盖外层定义。报错行号为 <Ident> 所在行数。 | <ConstDef>→<Ident> … <VarDef>→<Ident> … <Ident> … <FuncDef>→<FuncType><Ident> … <FuncFParam> → <BType> <Ident> … |
未定义的名字 | c | 使用了未定义的标识符报错行号为 <Ident> 所在行数。 | <LVal>→<Ident> … <UnaryExp>→<Ident> … |
函数参数个数不匹配 | d | 函数调用语句中,参数个数与函数定义中的参数个数不匹配。报错行号为函数调用语句的函数名所在行数。 | <UnaryExp>→<Ident>‘(’[<FuncRParams>]‘)’ |
函数参数类型不匹配 | e | 函数调用语句中,参数类型与函数定义中对应位置的参数类型不匹配。报错行号为函数调用语句的函数名所在行数。 | <UnaryExp>→<Ident>‘(’[<FuncRParams>]‘)’ |
无返回值的函数存在不匹配的return语句 | f | 报错行号为 ‘return’ 所在行号。 | <Stmt>→‘return’ {‘[’<Exp>’]’}‘;’ |
有返回值的函数缺少return语句 | g | 只需要考虑函数末尾是否存在return语句,无需考虑数据流。报错行号为函数结尾的’}’ 所在行号。 | <FuncDef> → <FuncType> <Ident> ‘(’ [<FuncFParams>] ‘)’ <Block> <MainFuncDef> → ‘int’ ‘main’ ‘(’ ‘)’ <Block> |
不能改变常量的值 | h | <LVal>为常量时,不能对其修改。报错行号为 <LVal> 所在行号。 | <Stmt>→<LVal>‘=’ <Exp>‘;’ <Stmt>→<LVal>‘=’ ‘getint’ ‘(’ ‘)’ ‘;’ |
缺少分号 | i | 报错行号为分号前一个非终结符所在行号。 | <Stmt>,<ConstDecl>及<VarDecl>中的’;’ |
缺少右小括号’)’ | j | 报错行号为右小括号前一个非终结符所在行号。 | 函数调用(<UnaryExp>)、函数定义(<FuncDef>)及<Stmt>中的’)’ |
缺少右中括号’]’ | k | 报错行号为右中括号前一个非终结符所在行号。 | 数组定义(<ConstDef>,<VarDef>,<FuncFParam>)和使用(<LVal>)中的’]’ |
printf中格式字符与表达式个数不匹配 | l | 报错行号为 ‘printf’ 所在行号。 | <Stmt> →‘printf’‘(’<FormatString>{,<Exp>}’)’‘;’ |
在非循环块中使用break和continue语句 | m | 报错行号为 ‘break’ 与 ’continue’ 所在行号。 | <Stmt>→‘break’‘;’ <Stmt>→‘continue’‘;’ |
- 在上文所示的表格中,a,i,j,k属于语法错误,已经在之前词法分析和语法分析部分完成了处理
和之前处理语法错误类似,各种语义错误也需要在ErrorController
中设置相关的错误处理函数,由于题目要求输出错误需要按照行号顺序依次输出,所以我们需要先用容器将相关的错误信息存储下来,之后再按照行号大小进行排序后统一输出,所以我重新建立了错误符号类ErrorToken
来存储每个错误的错误码及行号等信息,并在ErrorController
中统一输出。
public class ErrorToken {
private String errorCategoryCode;
private int lineNum;
}
errorCategoryCode
为当前的错误码,lineNum
为出现该错误时的行号
语义分析中有许多部分要求我们输出某个非终结符的行号,但在之前的设计中只有词法分析部分记录了所有终结符的行号信息,于此同时记录各个部分所处的行号范围对后续的debug
过程也有着巨大的帮助,所有我设计了范围类Span
来记录符号集中各符号所处的范围起始行数和终结行数信息。
public class Span {
private int startLine;
private int endLine;
}
在语义分析处理部分我写了四个工具类symChecker
,symDefiner
,symType
,symCalc
来辅助我建立符号表并输出错误信息。
下面给出各个工具类的部分代码
symChecker
类:检查类,用于对各个AstNode
进行错误检查并输出对应的Symbol
变量。
public void check(AstNode rootAst) throws IOException {
switch (rootAst.getGrammarType()) {
case "<CompUnit>":
checkCompUnitChecker(rootAst);
break;
case "<ConstDef>":
// Error b
checkConstDefChecker(rootAst);
break;
case "<VarDef>":
// Error b
checkVarDefChecker(rootAst);
break;
case "<Block>":
checkBlockChecker(rootAst);
break;
case "FORTK":
checkForStmtChecker(rootAst);
break;
case "BREAKTK":
// Error m
checkBreakStmtChecker(rootAst);
break;
case "CONTINUETK":
// Error m
checkContinueStmtChecker(rootAst);
break;
case "RETURNTK":
// Error f
checkReturnStmtChecker(rootAst);
break;
case "<LVal>":
// Error c
checkLValChecker(rootAst);
break;
case "<FuncDef>":
// Error b,g
checkFuncDefChecker(rootAst);
break;
case "<FuncFParam>":
// Error b
checkFuncFParamChecker(rootAst);
break;
case "<MainFuncDef>":
// Error b,g
checkMainFuncDefChecker(rootAst);
break;
case "ASSIGN":
// Error h
checkASSIGNChecker(rootAst);
break;
case "<UnaryExp>":
// Error b,c,d,e
checkUnaryExpChecker(rootAst);
break;
case "PRINTFTK":
// Error l
checkPRINTFTKChecker(rootAst);
break;
default:
SemanticAnalysis.preTraverse(rootAst);
break;
}
}
symDefiner
类:工具类,用于对AstNode
进行一些辅助操作,如给函数引入形式参数,求解某个Exp
对应的类型维数信息等。
public static void setParamInfo(AstNode astNode, FuncSymbol symbol) {
... ...
}
public static Integer getExpDim(AstNode astNode) {
switch (astNode.getGrammarType()) {
case "<Number>":
return getExpDimNumber(astNode);
case "<LVal>":
return getExpDimLVal(astNode);
case "<UnaryExp>":
return getExpDimUnaryExp(astNode);
case "<PrimaryExp>":
return getExpDimPrimaryExp(astNode);
default:
return getExpDim(astNode.getChildList().get(0));
}
}
symType
类:类别类,利用递归下降求解某变量或函数对应的参数类型。
public static Symbol.SymType getExpType(AstNode astNode) {
switch (astNode.getGrammarType()) {
case "<Number>":
return getExpTypeNumber(astNode);
case "<LVal>":
return getExpTypeLVal(astNode);
case "<UnaryExp>":
return getExpTypeUnaryExp(astNode);
case "<PrimaryExp>":
return getExpTypePrimaryExp(astNode);
default:
return getExpType(astNode.getChildList().get(0));
}
}
symCalc
类:计算类,书写了大量计算函数,用于计算可以立即被符号表计算出结果的信息,如全局变量区的VarDef
和ConstDef
的初始值。
public static int calc(AstNode astNode) {
switch (astNode.getGrammarType()) {
case "<Exp>", "<ConstExp>":
return calc(astNode.getChildList().get(0));
case "<AddExp>":
return calcAddExp(astNode);
case "<MulExp>":
return calcMulExp(astNode);
case "<UnaryExp>":
return calcUnaryExp(astNode);
case "<PrimaryExp>":
return calcPrimaryExp(astNode);
case "<Number>":
return calcNumber(astNode);
case "<LVal>":
return calcLValExp(astNode);
default:
return 0;
}
}
通过上面定义的这些工具类,我们就可以很有效的处理各种语义错误了。具体调用方法就是和语法分析过程类似,通过一个后续遍历让每个astNode
调用check
函数来按照顺序检查每个变量即可。
在这次作业中,相比下比较困难的部分时我们需要处理函数参数个数和维度匹配的问题。函数参数个数匹配比较简单,难的是函数维度的匹配(维度匹配包括维度个数的匹配、各维度长度的比较)。有些变量在定义时,纬度的长度不是简单的常数,而是用常量表达式(即ConstInitVal
)来表示长度。因此,为了能够进行匹配,我们需要进行在编译时将该长度计算出来。本文采用的方法是,给Exp
、MulExp
、AddExp
等类都编写一个evaluate
方法。在需要计算Exp
的值时,即调用Exp
的evaluate
方法,在这个方法中又调用AddExp
的evaluate
……以此类推,最终Exp
的evaluate
返回的就是整个表达式的值。
for (int i = 0; i < paramNum; i++) {
Symbol.SymType paramType = funcSymbol.getFparamTypes().get(i);
Symbol.SymType argType = SymTypeJudge.getExpType(childList.get(i));
Integer paramdim = funcSymbol.getFparamDims().get(i);
Integer argdim = SymDefiner.getExpDim(childList.get(i));
if (!paramType.equals(argType) || !paramdim.equals(argdim)) {
ErrorController.addError(new ErrorToken("e",rootAst.getSpan().getStartLine()));
}
}
中间代码生成主要分为两个重要部分,第一个是重新进行语义分析重建栈式符号表从而将各个部分转化为llvm_ir
的格式,另一个重要任务是利用llvm
中一切皆value的思想重新建立起一棵由不同种类的value组成的新语法树,最后生成中间代码其实就是对新语法树的一个后序遍历过程。
要想实现llvm_ir
的转化,我们首先需要清空之前错误处理时建立的符号表,并重新开始AST
的遍历,并将遍历到的各种不同成分的AstNode
进行处理即可,遍历及处理的代码如下:
//GenerationMain.java
public static void preTraverse(AstNode rootAst) {
if (rootAst.isLeaf()) {
return;
}
for (AstNode astNode : rootAst.getChildList()) {
GenerationMain.llvmGenIR.genIrAnalysis(astNode);
if (astNode.getGrammarType().matches(
"IFTK|FORTK|BREAKTK|CONTINUETK|RETURNTK|PRINTFTK|ASSIGN")) {
break;
}
}
}
//LLvmGenIr.java
public Value genIrAnalysis(AstNode rootAst) {
return switch (rootAst.getGrammarType()) {
case "<CompUnit>" -> genIrCompUnitChecker(rootAst);
//Definer.java
case "<ConstDef>" -> genIrConstDefChecker(rootAst);
case "<VarDef>" -> genIrVarDefChecker(rootAst);
case "<Block>" -> genIrBlockChecker(rootAst);
case "IFTK" -> genIrIfStmtChecker(rootAst);
case "FORTK" -> genIrForStmtChecker(rootAst);
case "BREAKTK" -> genIrBreakStmtChecker();
case "CONTINUETK" -> genIrContinueStmtChecker();
case "RETURNTK" -> genIrReturnStmtChecker(rootAst);
case "PRINTFTK" -> genIrPrintStmtChecker(rootAst);
case "<Exp>","<ConstExp>" -> genIrExpChecker(rootAst);
case "<PrimaryExp>" -> genIrPrimaryExpChecker(rootAst);
case "<Number>" -> genIrNumberCallChecker(rootAst);
case "<UnaryExp>" -> genIrUnaryExpChecker(rootAst);
case "<MulExp>" -> genIrMulExpChecker(rootAst);
case "<AddExp>" -> genIrAddExpChecker(rootAst);
case "<RelExp>" -> genIrRelExpChecker(rootAst);
case "<EqExp>" -> genIrEqExpChecker(rootAst);
//FuncDef.java
case "<FuncDef>" -> genIrFuncDefChecker(rootAst);
case "<FuncFParam>" -> genIrFuncFParamChecker(rootAst);
//MainFuncDef.java
case "<MainFuncDef>" -> genIrMainFuncDefChecker(rootAst);
//Lexer_part
case "ASSIGN" -> genIrAssignChecker(rootAst);
default -> {
GenerationMain.preTraverse(rootAst);
yield null;
}
};
}
这里需要注意,由于文法分析的限制,所以我们在判断stmt
的具体种类时不可避免的需要用到叶子节点,而叶子节点访问后需要回溯到其父节点进行对应的处理,这样就会造成二次访问,所以我们需要在其叶节点进行过一次访问后跳出循环。
由于llvm
中有着一切皆value
的思想,所以我们可以选择将该函数的返回值改为Value
,这样就可以在处理之后归于统一,从而维护更好的代码风格也更有利于管理。除此之外,在实际进行处理时,我发现有许多方法可以重复复用,所以我建立了LLvmGenUtils
类来处理这些可以重用的轮子。
在实际操作时,for
循环给我带来了不少的麻烦,主要相比于while
循环,for
循环的forstmt
和cond
均可以缺省,这也就导致在实际转化为llvm
时产生的BasicBlock
的数量会动态变化,可以说是一大难点了,需要重点注意,下面为我处理for
循环转化部分的代码。
private Value genIrForStmtChecker(AstNode sonAst) {
AstNode rootAst = sonAst.getParent();
AstNode forStmtVal1 = null;
AstNode condAst = null;
AstNode forStmtVal2 = null;
for (int i = 0; i < rootAst.getChildList().size(); i++) {
if (rootAst.getChildList().get(i).getGrammarType().equals("<ForStmt>")) {
if (i == 2) {
forStmtVal1 = rootAst.getChildList().get(i);
} else {
forStmtVal2 = rootAst.getChildList().get(i);
}
} else if (rootAst.getChildList().get(i).getGrammarType().equals("<Cond>")) {
condAst = rootAst.getChildList().get(i);
}
}
if (forStmtVal1 != null) {
genIrAnalysis(forStmtVal1);
}
SymbolTable.enterLoop();
BasicBlock condBlock = null;
if (condAst != null) {
condBlock = new BasicBlock(IrNameController.getBlockName());
}
BasicBlock currentLoopBlock = new BasicBlock(IrNameController.getBlockName());
BasicBlock followBlock = new BasicBlock(IrNameController.getBlockName());
IrNameController.pushLoop(new Loop(forStmtVal1, condBlock,
forStmtVal2, currentLoopBlock, followBlock));
if (condAst != null) {
new JumpInstr(condBlock);
IrNameController.setCurrentBlock(condBlock);
llvmGenUtils.genCondIr(condAst, currentLoopBlock, followBlock);
} else {
new JumpInstr(currentLoopBlock);
}
IrNameController.setCurrentBlock(currentLoopBlock);
genIrAnalysis(rootAst.getChildList().get(rootAst.getChildList().size() - 1));
if (forStmtVal2 != null) {
genIrAnalysis(forStmtVal2);
}
if (condAst != null) {
new JumpInstr(condBlock);
} else {
new JumpInstr(currentLoopBlock);
}
IrNameController.setCurrentBlock(followBlock);
IrNameController.popLoop();
SymbolTable.leaveLoop();
return null;
}
在实际生成llvm
中间代码的过程中,对虚拟寄存器的命名获得是一个非常重要的步骤,我们知道虚拟寄存器的命名有两种方式——使用数字递增顺序和使用字符串变量命名,前者难度较大且容易出错,可读性也较差,所以笔者选用了使用字符串变量命名的方法来获得虚拟寄存器的名称。
具体而言,笔者建立了IrNameController
类和IrPrefix
类来进行名称获取的管理。
IrNameController
类用于定义了一些静态方法来使每次调用时都可以动态的产生新的名字。IrPrefix
类按照虚拟寄存器的不同作用为这些命名加上对应的前缀使代码的可读性得以增强,是一个枚举类。
//IrNameController.java
private static HashMap<Function, Integer> blockNameIndexHashMap;
private static Integer paramNameIndex;
private static Integer stringLiteralNameIndex;
private static HashMap<Function, Integer> localVarNameIndexHashMap;
public static String getLocalVarName() {
int localVarNameIndex = localVarNameIndexHashMap.get(currentFunction);
localVarNameIndexHashMap.put(currentFunction, localVarNameIndex + 1);
return IrPrefix.LOCAL_VAR_NAME.toString() + localVarNameIndex;
}
public static String getBlockName() {
int blockNameIndex = blockNameIndexHashMap.get(currentFunction);
blockNameIndexHashMap.put(currentFunction, blockNameIndex + 1);
String funcName = currentFunction.getName().equals("@main") ? "main" :
currentFunction.getName().substring(3);
return funcName + "_" + IrPrefix.BB_NAME + blockNameIndex;
}
public static String getGlobalVarName(String varName) {
return IrPrefix.GLOBAL_VAR_NAME + varName;
}
public static String getParamName() {
return IrPrefix.PARAM_NAME.toString() + paramNameIndex++;
}
public static String getStringLiteralName() {
return IrPrefix.STRING_LITERAL_NAME.toString() + stringLiteralNameIndex++;
}
public static String getFuncName(String funcName) {
return (funcName.equals("main")) ? "@main" : IrPrefix.FUNC_NAME + funcName;
}
//IrPrefix.java
public enum IrPrefix {
BB_NAME("block_label_"),//基本块名
FUNC_NAME("@f_"),//函数名
GLOBAL_VAR_NAME("@g_"),//全局变量名
LOCAL_VAR_NAME("%v_"),//局部变量名
PARAM_NAME("%a_"),//参数名
STRING_LITERAL_NAME("@s_");//字符串字面量名
private final String prefix;
IrPrefix(String prefix) {
this.prefix = prefix;
}
@Override
public String toString() {
return prefix;
}
}
LLVM
中的核心观点就是“一切皆Value
”,所以我选择建立的value
类作为所有value
语法树的公共父节点,最终建立的继承结构如下图
具体而言,对于我们所生成的中间代码,可以理解为就是一个以module
为根节点的语法树,而module
中由字符串字面值列表、全局变量列表和函数列表三个部分组成,而这三个部分又可以依次向下展开,也就形成了树结构的基本样式。对于该结构而言,字符串列表和全局变量都比较简单,所以我这里重点介绍一下函数列表的结构。
Function
函数主要分为参数列表和基本块列表两个部分,参数列表较为容易实现,而基本块则是一个相对陌生的概念,所以我在正式展开介绍前先介绍一下基本块的概念。
所谓基本块,就是代码中可以连续执行的最大序列,简而言之就是对于一个基本块而言,程序的执行(控制流)只能从基本块的第一条语句进入,只能从基本块的最后一条语句离开,符合这样的条件的代码段就是一个基本块,在我们完成基本块的划分后,后续的中间代码优化都是建立在基本块的块内逻辑和块间逻辑上的,所以将一个函数划分成多个基本块是非常重要的。
public class Function extends User {
private final IrType returnType;
private final ArrayList<BasicBlock> basicBlocks;
private final ArrayList<Param> params;
}
那我们应该如何划分基本块呢,有一个非常重要的部分就是跳转语句,因为只有遇到跳转语句后才会使得我们的代码段不再连续,这样也就自然形成的基本块的划分。
而在BasicBlock
内部是由一系列指令Instr
和所属函数的指针组成的,所以我们在之前转化llvm_ir
的过程中其实本质就是不断将AST
中建立的各种函数变量和符号变量转化成一条条指令装入BasicBlock
的过程。
Instr
指令部分我分成了basis
和optimizer
两部分分别表示基本指令和优化指令,其中优化指令是后续代码优化需要用到的如phi等指令,而基本指令我分为了AllocaInstr
、BrInstr
、CalcInstr
、CallInstr
、GetEleInstr
、IcmpInstr
、JumpInstr
、LoadInstr
、RetInstr
、StoreInstr
、TruncInstr
、ZextInstr
等,之后按照不同指令需要的操作数不同各自重写相应的toString
方法即可。
值得一提的是,这里我们终于使用到了上述结构中定义的User
类,这个类是所有需要用到操作数的类的公共父类,在他其中定义了一个protected
属性的operands
操作数集合,并定义了各种操作数可以使用的方法,从而使其子类调用方法时更加便利。
public class User extends Value {
protected ArrayList<Value> operands;
}
为了更好的区分各种变量的类型,所以我们不可避免的需要建立类型类并加以管理,具体而言我建立了IrType
类并在其下方继承了四个子类ArrayType
,PointerType
,StructType
和VarType
分别表示数组类型、指针类型、结构类型和变量类型,其中结构类型是语法树中一些非叶子节点所属的类型,如BasicBlock
。建立好这些类型类之后,我们就可以在其中重写toString
方法,这样可以避免大量重复造轮子的需求。
在实际操作中,二维数组的处理毫无疑问是一大难点,仅仅它的toString
编写难度就异常巨大,你需要根据它定义初值的情况动态处理字符串,代码复杂程度还是很高的,具体体现在为变量初始化(Initial
)和取数(GetEleInstr
)两个部分,需要我们格外细心。
//Initial.java
@Override
public String toString() {
if (initValue == null || initValue.isEmpty()) {
return (type.isInt32()) ? type + " 0" : type + " zeroinitializer";
} else {
if (type.isInt32()) {
return type + " " + initValue.get(0);
} else {
if (space.size() == 1) {
return type + " [" + initValue.stream().map(number -> "i32 " + number)
.collect(Collectors.joining(", ")) + "]";
} else {
StringBuilder sb = new StringBuilder();
if (offset == 0) {
sb.append(type).append(" zeroinitializer");
} else {
sb.append(type).append(" [");
for (int i = 0; i < space.get(0); i++) {
if (i * space.get(1) < offset) {
sb.append("[").append(space.get(1)).append(" x i32]").append("[");
for (int j = 0; j < space.get(1); j++) {
sb.append("i32 ").append(initValue.get(i * space.get(1) + j));
if (j != space.get(1) - 1) {
sb.append(", ");
}
}
sb.append("]");
} else {
sb.append("[").append(space.get(1)).
append(" x i32] ").append("zeroinitializer");
}
if (i != space.get(0) - 1) {
sb.append(", ");
}
}
sb.append("]");
}
return sb.toString();
}
}
}
}
//GetEleInstr.java
@Override
public String toString() {
StringBuilder sb = new StringBuilder();
PointerType ptrType = (PointerType) (operands.get(0).getType());
IrType type = ptrType.getTarget();
sb.append(name).append(" = getelementptr inbounds ")
.append(type).append(", ")
.append(ptrType).append(" ")
.append(operands.get(0).getName())
.append((type.isArray() ? ", i32 0, i32 " : ", i32 "));
if (type.isArray() && ((ArrayType) type).getEleSpace().size() == 2) {
if (operands.get(1).getName().matches("[0-9]+")) {
ArrayList<Integer> spaces = ((ArrayType) type).getEleSpace();
Integer offset = Integer.parseInt(operands.get(1).getName());
sb.append(offset / spaces.get(1)).append(", i32 ").append(offset % spaces.get(1));
} else {
sb.append("0, i32 ").append(operands.get(1).getName());
}
} else {
sb.append(operands.get(1).getName());
}
return sb.toString();
}
当我们完成到这里后,最后一个步骤就是将所有继承了Value
类的节点全部重写toString
方法然后调用modele.toString()
就可以获得最终的中间代码了,module
类的toString
方法如下,其他类可以类似效仿:
@Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append(GetIntDeclare.getDeclare()).append("\n");
sb.append(PutIntDeclare.getDeclare()).append("\n");
sb.append(PutStrDeclare.getDeclare()).append("\n\n");
for (FormatString stringLiteral : stringLiterals) {
sb.append(stringLiteral).append("\n");
}
sb.append("\n");
for (GlobalVar globalVar : globalVars) {
sb.append(globalVar.toString()).append("\n");
}
sb.append("\n");
for (Function function : functions) {
sb.append(function.toString()).append("\n\n");
}
return sb.toString();
}
在进行代码优化之前,笔者先进行了mips
代码生成的编写,这样可以在中间代码相对简单时生成mips
,后续优化编写之后指令集变化也较小,这样可以得到更高的编译器编写效率。言归正传,我们要想由中间代码生成mips
汇编,我们需要遍历之前生成的Value
语法树,并以Instr
为单位将每条指令转化为几条汇编指令,转化的整体而言难度还是比较小的,主要的难点在于对栈指针的移动和内存管理。
汇编指令转化部分我将mips
汇编分为了.data
和.text
段,并分别用一个数组存储对应的Assembly
类,之后编写各类不同的指令继承Assembly
这个父类即可。
而具体到汇编指令的转化,我们需要在Instr
中编写generateAssembly()
方法并在各个子类中重写该方法,即可完成转化。
除此之外,我们还需要实现.word
和.asciiz
两个预处理指令,下方我以.word
指令举例,.asciiz
类似实现即可。
public class WordAsm extends DataAssembly {
private final ArrayList<Integer> initValue;
public WordAsm(String label, String value, ArrayList<Integer> initValue) {
super(label, value);
this.initValue = initValue;
}
@Override
public String toString() {
if (initValue == null) {
return label + ": .word " + value;
} else {
int offsetTot = Integer.parseInt(value) - initValue.size();
StringBuilder res = new StringBuilder(label + ": .word ");
for (int i = 0; i <= initValue.size() - 1; i++) {
res.append(initValue.get(i));
if (i != initValue.size() - 1) {
res.append(", ");
}
}
for (int i = 0; i < offsetTot; i++) {
res.append("0");
if (i != offsetTot - 1) {
res.append(", ");
}
}
return res.toString();
}
}
}
为了更好实现栈指针的移动和相关的取数操作,我编写了RegisterUtils
方法类,里面编写了包括移动栈指针,申请寄存器,取寄存器的值,取内存的值,取指针对应值等多种内存环境下的常用方法的封装,这样可以使代码复用性更强,适应高内聚低耦合的需求。
public class RegisterUtils {
public static Integer moveValueOffset(Value value) {
... ...
}
public static void allocReg(Value value, Register target) {
... ...
}
public static void reAllocReg(Value value, Register target) {
... ...
}
public static Register loadRegisterValue(Value operand, Register instead, Register reg) {
... ...
}
public static Register loadVariableValue(Value operand, Register reg, Register instead) {
... ...
}
public static Register loadPointerValue(Value operand, Register pointerReg, Register instead) {
... ...
}
public static Register loadMemoryOffset(Value operand, Register instead, Register target,
Register pointerReg, Register offsetReg) {
... ...
}
public static Register allocParamReg(Value para, Register paraReg,
int currentOffset, ArrayList<Register> allocatedRegs) {
... ...
}
public static Register allocParamMem(Value para, Register paraReg, int currentOffset,
ArrayList<Register> allocatedRegs, int paraNum) {
... ...
}
}
生成汇编还有一个非常重要的步骤是为每个虚拟寄存器分配对应的实际寄存器,这里需要我们创建Register
寄存器类和RegisterController
控制器类来控制寄存器的分配。
Register
类是一个枚举类,相比于直接使用字符串访问寄存器,使用枚举类的好处在于可以调用枚举类的ordinal()
方法,这个可以找到每个枚举常量的索引,就像数组索引一样,这样有利于我们动态访问所需的寄存器。
而RegisterController
类是一个建立了一组Value
和Register
的对应关系,从而及时更新各个Value
实际分配的寄存器,这样也就实现了对寄存器的动态管理的需求。
public class RegisterController {
private static HashMap<Value, Register> valueRegisterHashMap;
public Register getRegister(Value value) {
return (valueRegisterHashMap == null) ? null :
valueRegisterHashMap.get(value);
}
public void allocRegister(Value value, Register register) {
if (valueRegisterHashMap == null) {
return;
}
valueRegisterHashMap.put(value, register);
}
}
对于已经分配好寄存器的变量,我们可以直接将其翻译成对应的寄存器,而对于其他变量,我们只能将其放到内存中,使用的时候从内存中取,这就要涉及到内存管理策略了,而本编译器的内存管理策略如下。
调用新函数前,$sp
的值会设置为新函数的栈底,在当前函数的作用域中, $sp
的值不会随内存分配而变化。函数前3个形参保存在$a1, $a2, $a3
中,$a0
只用于IO,剩余的形参存入内存方便后续取用。在调用某个函数时,我们需要先保存当前已经分配的寄存器,然后将函数参数、$sp
和$ra
寄存器依次保存栈顶。最后,将$sp
的值加上目前的总偏移量,作为被调用函数的栈底。接下来,直接jump
到目标函数对应的label
即可。当函数返回时将存储于内存的变量恢复。
在函数内部使用到的局部变量依次存放,并在AssemblyUnit
记录每个局部变量的存放位置相对于栈底的偏移,以及当前栈顶到栈底的总偏移量。每定义一个需要分配空间的变量时,栈指针偏移,然后在AssemblyUnit
中记录好该变量到该偏移量的映射关系即可。
到此一个简单的、没有进行任何优化的编译器正式完工,后续可以开始开展优化工作了。
注:在编写llvm生成和mips生成时我编写了一台评测机,链接为——Charles2530/compiler_autotest: 北航2023年编译原理自动化测试 (github.com),可供参考。
全局变量局部化是指将函数出于全局区的变量转化到局部使用,由于全局变量(非数组)是可以在编译阶段计算出具体的值的,而在GVN有常数折叠的行为,所以将可以直接转化为常数的全局变量进行转化后可以更好的推进后续的优化。
具体而言,全局变量局部化主要对两种情况的全局变量进行了优化:
- 如果全局变量没有使用者,那么就可以局部化
- 如果全局变量只有一个使用者,那么就可以局部化
为了实现全局变量局部化,我们首先需要设计一个HashMap存储每个全局变量和调用它的函数列表,之后我们需要创建函数调用表,之后按照上述逻辑将符合条件的函数局部化即可,下面是局部化的代码逻辑。
/**
* localize 是局部化全局变量的函数
* 该函数的执行逻辑如下:
* 1. 如果全局变量没有使用者,那么就可以局部化
* 2. 如果全局变量只有一个使用者,那么就可以局部化
*/
private static boolean localize(GlobalVar globalVar) {
if (globalVarUsers.get(globalVar).isEmpty()) {
return true;
}
if (globalVarUsers.get(globalVar).size() == 1) {
Function target = globalVarUsers.get(globalVar).iterator().next();
if (!FunctionInlineUnit.getResponse(target).isEmpty()) {
return false;
}
BasicBlock entry = target.getBasicBlocks().get(0);
if (globalVar.getType().isInt32()) {
... ...
return true;
}
}
return false;
}
在生成中间代码后,我们首先需要一些不必要的跳转指令,虽然这部分并不会优化我们中间代码的效率,但这可以很好的简化中间代码,而且去除重复跳转语句后可以让我们在后续写优化时可以通过访问基本块的最后一条指令即可获得所需的跳转块,大大提高优化效率。具体而言主要分为删除重复的分支语句和删除死代码块两部分。
所谓删除基本块中重复的分支,对于每个基本块而言,仅需保留第一个分支即可,该分支以下的代码不会被执行,函数代码如下:
public static void deleteDuplicateBranch(BasicBlock basicBlock) {
for (int i = 0; i < basicBlock.getInstrArrayList().size() - 1; i++) {
Instr instr = basicBlock.getInstrArrayList().get(i);
if (instr instanceof BrInstr || instr instanceof JumpInstr
|| instr instanceof RetInstr) {
basicBlock.setInstrArrayList(new ArrayList<>(
basicBlock.getInstrArrayList().subList(0, i + 1)));
break;
}
}
}
删除死代码块而言,如果该基本块不可达,则该基本块为死代码块,应当被删除,函数代码如下:
public static void deleteDeadBlock(Function function) {
HashSet<BasicBlock> vis = new HashSet<>();
BasicBlock entry = function.getBasicBlocks().get(0);
dfs(entry, vis);
function.getBasicBlocks().removeIf(bb -> !vis.contains(bb) && bb.setDeleted(true));
}
这里通过深度优先搜索进行访问,并用vis数组进行记录即可。
Mem2Reg将一个非SSA的形式转化到含有phi函数的SSA形式,具体分为插入phi指令和变量重命名两个步骤。具体而言,在实现Mem2Reg之前,我们需要创建一些数据结构来辅助完成Mem2Reg的内容。
控制流程图(Control Flow Graph)简称CFG,记录了每个块之间的跳转关系,具体而言即记录了每个块的前序基本块和后继基本块。
要想构建控制流图,需要寻找函数各个块的直接跳转指令和间接跳转指令,并为期添加跳转的记录即可。具体函数逻辑如下:
/**
* buildControlFlowGraph 方法用于构建控制流图
* 这里求解出了IndBasicBlock和OutBasicBlock
*/
private static void buildControlFlowGraph(Function function) {
for (BasicBlock basicBlock : function.getBasicBlocks()) {
Instr lastInstr = basicBlock.getLastInstr();
if (lastInstr instanceof JumpInstr jumpInstr) {
ControlFlowGraph.addDoubleEdge(basicBlock, jumpInstr.getTarget());
} else if (lastInstr instanceof BrInstr brInstr) {
ControlFlowGraph.addDoubleEdge(basicBlock, brInstr.getThenBlock());
ControlFlowGraph.addDoubleEdge(basicBlock, brInstr.getElseBlock());
}
}
}
DominatorTree 是支配树,在这个阶段我们需要形成四个数据结构,即每个基本块的支配集合,支配边界集合,直接支配父节点和直接支配子节点集合。除此之外,我们还可以在这个过程中算出每个基本块所在支配树的深度,用于后续GCM优化提供服务。
这几个定义如下:
- 支配(dominate):如果CFG中从起始节点到基本块y的所有路径都经过了基本块x,我们说x支配y
- 严格支配(strict dominate):显然每个基本块都支配它自己。如果x支配y,且x不等于y,那么x严格支配y
- 直接支配者(immediate dominator, idom):严格支配n,且不严格支配任何严格支配 n 的节点的节点(直观理解就是所有严格支配n的节点中离n最近的那一个),我们称其为n的直接支配者
- 支配边界(Dominance Frontier):节点 n 的支配边界是 CFG 中刚好不被 n 支配到的节点集合
要想构建支配集合,对于这个函数而言,函数的逻辑是在for循环中找到所有不被 basicBlock支配的block,放入reachSet中,这样所有不在reachedSet中的Block就都是被basicBlock,支配的Block,然后将所有不在reachSet中的block放入domList中(包括basicBlock本身)
注:这里与教程中的支配集合定义有区别,应该理解为该支配集合里的节点都是被该节点支配的。这样省略了求不动点的过程,效率更高。
public void searchBlockDominateSet() {
BasicBlock entry = basicBlocks.get(0);
for (BasicBlock basicBlock : basicBlocks) {
HashSet<BasicBlock> reachedSet = new HashSet<>();
DominatorTree.dfsDominate(entry, basicBlock, reachedSet);
ArrayList<BasicBlock> domList = new ArrayList<>();
for (BasicBlock bb : basicBlocks) {
if (!reachedSet.contains(bb)) {
domList.add(bb);
}
}
DominatorTree.addBlockDominateSet(basicBlock, domList);
}
}
求解支配树换言之就是求解直接支配关系,即A直接支配B——A严格支配B,且不严格支配任何严格支配B的节点。所以我们需要借助之前生成的集合,判断该节点和他的支配集合是否为直接支配关系,如果是直接支配就直接生成直接支配父节点和直接支配子节点集合。
判断是否为直接支配关系的逻辑如下:
private static boolean isImmediateDominator(BasicBlock basicBlock, BasicBlock dominateBlock) {
boolean flag = basicBlock.getBlockDominateSet().contains(dominateBlock) &&
!dominateBlock.equals(basicBlock);
for (BasicBlock midBlock : basicBlock.getBlockDominateSet()) {
if (!midBlock.equals(dominateBlock) && !midBlock.equals(basicBlock)
&& midBlock.getBlockDominateSet().contains(dominateBlock)) {
flag = false;
break;
}
}
return flag;
}
求解支配边界是一个相对固定的算法,直接借助教程所给的伪代码即可。
上述伪代码是插入phi函数的伪代码,我们的迭代支配边界也将在这个过程被计算出来。
public void searchBlockDominanceFrontier() {
for (Map.Entry<BasicBlock, ArrayList<BasicBlock>> entry :
ControlFlowGraph.getFunctionOutBasicBlock(this).entrySet()) {
BasicBlock from = entry.getKey();
ArrayList<BasicBlock> outBasicBlocks = entry.getValue();
for (BasicBlock to : outBasicBlocks) {
BasicBlock runner = from;
while (!runner.getBlockDominateSet().contains(to)
|| runner.equals(to)) {
DominatorTree.addBlockDominanceFrontierEdge(runner, to);
runner = runner.getBlockDominateParent();
}
}
}
}
支配树深度集合计算相对比较简单,实际就是一个深度优先搜索的过程,代码如下:
public static void dfsDominateLevel(BasicBlock basicBlock, Integer depth) {
DominatorTree.addBlockDominateTreeDepth(basicBlock, depth);
for (BasicBlock child : basicBlock.getBlockDominateChildList()) {
DominatorTree.dfsDominateLevel(child, depth + 1);
}
}
在有了上述功能作为基础后,我们就可以开始我们Mem2Reg的过程了。这里主要分为Phi指令位置选择和变量重命名两个步骤。
这部分可以参考之前的伪代码,本算法首先将defBasicBlockArrayList中的基本块推入栈中,然后在while循环执行固定算法即可,这段代码形成的代码如下:
public static void insertPhi() {
HashSet<BasicBlock> f = new HashSet<>();
Stack<BasicBlock> w = new Stack<>();
Mem2RegUnit.defBasicBlockArrayList.forEach(w::push);
while (!w.isEmpty()) {
BasicBlock x = w.pop();
for (BasicBlock y : x.getBlockDominanceFrontier()) {
if (!f.contains(y)) {
f.add(y);
Instr phiInstr = new PhiInstr(IrNameController.getLocalVarName(
y.getBelongingFunc()), y.getBlockIndBasicBlock());
y.insertInstr(0, phiInstr);
useInstrArrayList.add(phiInstr);
defInstrArrayList.add(phiInstr);
if (!defBasicBlockArrayList.contains(y)) {
w.push(y);
}
}
}
}
}
这里f 为需要添加phi的基本块的集合,w 为定义变量的基本块的集合。
注:phi既是useInstr,又是defInstr
变量重命名主要依托于深度优先搜索,来实现对基本块的变量进行重命名,文档中的伪代码如下:
而变量重命名的主要代码如下:
public static void dfsVarRename(BasicBlock presentBlock) {
int cnt = removeUnnecessaryInstr(presentBlock);
for (BasicBlock basicBlock : presentBlock.getBlockOutBasicBlock()) {
Instr instr = basicBlock.getInstrArrayList().get(0);
if (instr instanceof PhiInstr phiInstr && useInstrArrayList.contains(phiInstr)) {
phiInstr.modifyValue(((stack.isEmpty()) ?
new Constant("0", new VarType(32), false) : stack.peek()), presentBlock);
}
}
presentBlock.getBlockDominateChildList().forEach(Mem2RegUnit::dfsVarRename);
for (int i = 0; i < cnt; i++) {
stack.pop();
}
}
在这里,cnt 记录遍历presentBlock的过程中,stack的push次数,而该函数的执行逻辑如下:
- 在移除了非必要的Instr后,遍历presentBlock的后继集合,将最新的define(stack.peek)填充进每个后继块的第一个phi指令中(有可能某个后继块没有和当前alloc指令相关的phi,需要进行特判(符合条件的Phi应该在useInstrList中))
- 对presentBlock支配的基本块使用dfsVarRename方法,实现DFS
- 将该次dfs时压入stack的数据全部弹出
程序包含的一些代码可能并不会被运行或者不会对结果产生影响,那么我们称这种代码为死代码。我们将不会被运行到的称为不可达代码,将不会对结果产生影响的代码成为无用代码。删除无用或不可达代码可以缩减IR代码,可使程序更小、编译更快、执行也更快。
其实之前的基本块简化也可以算是死代码删除,但那里删除的都是真的没有用的代码,对于优化没有帮助,所以这部分主要介绍删除不会对结果产生影响的代码。这里我主要定义了isDead()函数来判断某个指令是否可以被死代码优化。
public boolean isDead() {
return this.isValid() && !(this instanceof CallInstr) &&
!(this instanceof IoStreamGeneration) && this.getUseDefChain().isEmpty();
}
这里我们首先意识到IO指令和函数调用指令是肯定不能优化的,除此之外我们我们需要判断指令是否优先以及是否有其他指令使用了该指令,如果该指令无效或者没有其他指令使用该指令,这样我们就可以删除这条指令。
这里isValid 的指令有 alloca,alu,call,gep,io,getint,load,phi,zext,其中call指令调用的函数将指针作为形参、修改全局变量、调用了其他函数,因此不能直接删除,io中的 getint 指令获得的数字即使没有用到也应该完成io操作,也不能删除。
关于函数内联这个操作,其实正常是属于中端代码优化的,但我受到参考编译器的启发,发现这个操作其实可以在生成
AST
树后进行节点调整完成,这样可以很好的完成后续的操作,但在尝试后发现这个功能在Mem2Reg后修改效率更高,所以最终这部分还是放在了中端代码优化部分。
程序中的一次函数调用,除去函数内部执行造成的开销,会产生很多的额外开销,如保存和恢复现场,参数传递和跳转到被调用函数,以及完成执行后返回等,而对于非递归,且没有调用与递归相关函数的函数,我们可以将其内部的执行指令嵌入到调用它的函数中,这种优化,我们叫做函数内联。
函数内联时首先要判断这个函数是否可以被内联,评价标准是内联函数不能递归,不能调用其他函数。而函数内联分析主要分为以下几个步骤:
- 建立函数调用图
- 深度优先搜索函数调用图,判断是否有递归
- 内联函数并更新函数调用图
- 删除无用函数
- 重复1-4步骤,直到不动点稳定
而具体代码如下:
private static void inlineAnalysis() {
FunctionInlineUnit.fixedPoint = true;
while (FunctionInlineUnit.fixedPoint) {
FunctionInlineUnit.fixedPoint = false;
FunctionInlineUnit.buildFuncCallGraph();
module.getFunctions().forEach(Function::dfsCaller);
FunctionInlineUnit.inlineFunctionsList.forEach(Function::inlineFunction);
FunctionInlineUnit.inlineFunctionsList.clear();
FunctionInlineUnit.buildFuncCallGraph();
FunctionInlineUnit.removeUselessFunction();
}
}
在函数调用图构建时和上述全局变量局部化的函数调用图构建过程相同,其余部分也较为简单,这里我主要介绍内联函数那一步。
内联函数操作的执行逻辑如下:
- 通过遍历,获得所有调用 inlineFunc 的 callInstr
- 对于每一个 callInstr,将其替换为 inlineFunc 的所有基本块
- 去除调用关系
显而易见,这里最难的一步是将callInstr替换为基本块集合的操作,这里我们需要将函数调用指令所在的基本块分割成两半,之后将函数调用指令所在的基本块之后再建一个块,用于存放函数调用指令之后的指令,最后我们需要克隆函数调用指令所在的基本块,将克隆的基本块插入到函数调用指令所在的基本块之后即可。
public static void replaceFunctions(CallInstr callInstr) {
Function response = callInstr.getTarget();
BasicBlock basicBlock = callInstr.getBelongingBlock();
Function function = basicBlock.getBelongingFunc();
BasicBlock inlineBlock = new BasicBlock(IrNameController.getBlockName(function));
function.addBasicBlock(inlineBlock, function.getBasicBlocks().indexOf(basicBlock) + 1);
FunctionInlineUnit.splitBlock(callInstr, basicBlock, inlineBlock);
FunctionInlineUnit.cloneBlock(callInstr, basicBlock, inlineBlock, response);
}
除此之外需要强调,在函数内联完成后,由于内联函数属于块间优化,所以我们在执行后需要重建流图和支配树,再继续后续的操作。
GVN(Global Variable Numbering) 全局值编号:为全局的变量进行编号,实现全局的消除公共表达式。
要想实现这些操作,我们需要构建全局变量编号哈希表,之后在分析时我们可以将各个指令的hash值存入,如果在该hash表中有之前的记录就可以直接替换,这样就实现了全局的消除公共表达式。
由于常数相同值但计算表达式不同的情况的存在,所以我们需要先将可以合并的表达式进行合并,这样我们就可以获得更多的公共表达式。
进行常数折叠时我们需要根据每个calcInstr的两个操作数中常数的数量进行分类讨论:
-
如果没有常数,则可以将前后两个操作数相同且操作位减法、取模或除法时进行相应的操作。
private static Value simplifyNoConstant(CalcInstr calcInstr) { if (calcInstr.getOperands().get(0).equals(calcInstr.getOperands().get(1))) { if (calcInstr.getInstrType().matches("sub|srem")) { return new Constant("0", new VarType(32)); } else if (calcInstr.getInstrType().matches("sdiv")) { return new Constant("1", new VarType(32)); } } return null; }
-
如果有一个常数,我们需要按照常数所处的位置进行相应的优化。
private static Value simplifySingle(CalcInstr calcInstr) { if (calcInstr.getOperands().get(0) instanceof Constant constant) { int val = constant.getVal(); if (val == 0) { if (calcInstr.getInstrType().matches("add")) { return calcInstr.getOperands().get(1); } else if (calcInstr.getInstrType().matches("mul")) { return new Constant("0", new VarType(32)); } } else if (val == 1) { if (calcInstr.getInstrType().matches("mul")) { return calcInstr.getOperands().get(1); } } } else { int val = ((Constant) calcInstr.getOperands().get(1)).getVal(); if (val == 0) { if (calcInstr.getInstrType().matches("add|sub")) { return calcInstr.getOperands().get(0); } else if (calcInstr.getInstrType().matches("mul")) { return new Constant("0", new VarType(32)); } } else if (val == 1) { if (calcInstr.getInstrType().matches("mul|sdiv")) { return calcInstr.getOperands().get(0); } else if (calcInstr.getInstrType().matches("srem")) { return new Constant("0", new VarType(32)); } } } return null; }
-
如果有两个常数,我们可以直接提前进行计算。
private static Value simplifyDouble(CalcInstr calcInstr) { int op1 = ((Constant) calcInstr.getOperands().get(0)).getVal(); int op2 = ((Constant) calcInstr.getOperands().get(1)).getVal(); op2 = (calcInstr.getInstrType().matches("sdiv|srem") && op2 == 0) ? 1 : op2; return switch (calcInstr.getInstrType()) { case "add" -> new Constant(String.valueOf(op1 + op2), new VarType(32)); case "sub" -> new Constant(String.valueOf(op1 - op2), new VarType(32)); case "mul" -> new Constant(String.valueOf(op1 * op2), new VarType(32)); case "sdiv" -> new Constant(String.valueOf(op1 / op2), new VarType(32)); case "srem" -> new Constant(String.valueOf(op1 % op2), new VarType(32)); case "and" -> new Constant(String.valueOf(op1 & op2), new VarType(32)); case "or" -> new Constant(String.valueOf(op1 | op2), new VarType(32)); default -> null; }; }
这样我们就完成了常数折叠步骤。
遍历一遍Instr,找出所有可优化的alu, gep, icmp和func,如果GVN中已存在,直接用该值即可, 否则插入到GVN中,之后对其直接支配的子节点调用该函数实现递归,最后需要将该基本块插入GVN的instr全部删去, 避免影响兄弟子树的遍历。实现的代码如下:
public static void uniqueInstr(BasicBlock entry, HashMap<String, Instr> hashMap) {
ConstantFoldingController.foldingCalcInstr(entry);
HashSet<Instr> vis = new HashSet<>();
Iterator<Instr> iterator = entry.getInstrArrayList().iterator();
while (iterator.hasNext()) {
Instr instr = iterator.next();
if (instr instanceof CalcInstr || instr instanceof IcmpInstr ||
instr instanceof GetEleInstr || (instr instanceof CallInstr callInstr &&
((Function) callInstr.getOperands().get(0)).isImprovable())) {
String hash = instr.getGlobalVariableNumberingHash();
if (hashMap.containsKey(hash)) {
instr.replaceAllUse(hashMap.get(hash));
iterator.remove();
} else {
hashMap.put(hash, instr);
vis.add(instr);
}
}
}
entry.getBlockDominateChildList()
.forEach(child -> uniqueInstr(child, hashMap));
DebugDetailController.printGlobalVariableNumbering(entry.getBelongingFunc(), hashMap);
for (Instr instr : vis) {
hashMap.remove(instr.getGlobalVariableNumberingHash());
}
}
在上述优化过程中,有一个非常重要的部分是优化callInstr,但并不是每个调用函数指令都是可以被优化的,简而言之,只有函数满足参数没有指针类型, 每个指令没有读写全局变量,不能调用其他函数
GCM(Global Code Motion) 全局代码移动:根据Value之间的依赖关系,将代码的位置重新安排,从而使得一些不必要(不会影响结果)的代码尽可能少执行。
在实现GCM之前我们需要进行循环分析和副作用分析,除此之外我们还需要获得支配树的深度(这一步之前已经实现过了)。
所谓循环分析,也就是获得各个基本块所处的循环层数。而如何判断一个基本块进入循环了呢,这就需要用到我们理论课的知识了,即判断循环的依据是如果某个块支配了它的某个前序块,那么这个块就是循环的头这样我们将循环层数添加即可。
public void loopAnalysis() {
... ...
ArrayList<BasicBlock> latchBlocks = new ArrayList<>();
ArrayList<BasicBlock> posOrderBlocks =
DominatorTree.computeDominanceTreePostOrder(this);
for (BasicBlock basicBlock : posOrderBlocks) {
for (BasicBlock indBasicBlock : basicBlock.getBlockIndBasicBlock()) {
if (basicBlock.getBlockDominateSet().contains(indBasicBlock)) {
latchBlocks.add(indBasicBlock);
}
}
if (!latchBlocks.isEmpty()) {
LoopVal loop = new LoopVal(basicBlock, latchBlocks);
LoopAnalysisController.addLoopIntoGraph(latchBlocks, loop);
latchBlocks.clear();
}
}
LoopAnalysisController.addLoopSons(basicBlocks.get(0));
}
在副作用分析之前,我们需要构建函数调用图,然后借助这个图完成分析。之后我们需要看该函数调用的所有函数,如果它调用的函数本身存在副作用或者它调用的函数没有被访问过且该函数调用的函数存在副作用,则该函数存在副作用。主要代码逻辑如下:
private static boolean processAnalysis(Function function) {
boolean sideEffect = false;
addFunctionVisited(function, true);
if (getFunctionProcessed(function)) {
sideEffect = function.getSideEffect();
for (Function response : function.getResponses()) {
if (!getFunctionVisited(response) && !getFunctionProcessed(response)) {
processAnalysis(response);
}
}
} else {
for (Function response : function.getResponses()) {
if (response.getSideEffect() ||
(!getFunctionVisited(response) && !getFunctionProcessed(response)
&& processAnalysis(response))) {
sideEffect = true;
}
}
}
addFunctionVisited(function, false);
function.setSideEffect(sideEffect);
addFunctionProcessed(function, true);
return sideEffect;
}
在完成两个前序步骤后,我们就可以正式展开优化了。全局代码移动算法的功能是调度那些“浮动”的指令,把它们归位到一个个基本块中。这个过程必须保留已有的控制依赖和数据依赖。在这个大前提下,算法可以自由发挥,我们的目标就是尽可能把代码移动到循环外面,尽可能让代码的执行路径变少。
而如果我们想要调度指令,需要遵守以下几个流程:find_Pinned_Insts,schedule_Early,schedule_Late,select_block
有些指令是无法被灵活调度的,如phi,br/jump,ret等指令,这些指令我们叫做pinnedInst。pinnedInst受到控制依赖的牵制,无法被调度到其他基本块,也就是说这些指令和它们的基本块是绑定的。这一部分,我们需要先遍历指令将pinnedInst进行标记。具体代码如下:
public boolean isPinned() {
return !(this instanceof CalcInstr) && !(this instanceof IcmpInstr) &&
!(this instanceof ZextInstr) && !(this instanceof GetEleInstr) &&
!(this instanceof CallInstr callInstr && callInstr.isPure());
}
这一步骤我们会尽可能的把指令前移,确定每个指令能被调度到的最早的基本块,同时不影响指令间的依赖关系。这里我们可以直接参考教程提供的伪代码即可。
该函数对应的代码如下:
/**
* scheduleEarly 用于在GCM中实现尽可能的把指令前移,
* 确定每个指令能被调度到的最早的基本块,同时不影响指令间的依赖关系。
* 当我们把指令向前提时,限制它前移的是它的输入,
* 即每条指令最早要在它的所有输入定义后的位置。
* 该函数执行逻辑如下:
* 1.如果已经处理过了,或者是无法移动,那么就结束处理。
* 2.如果未处理,将这条指令从当前块移除,然后插入到入口块的最后一条指令之前。
* 3.遍历该指令用到的操作数,尝试前移。
*/
public static void scheduleEarly(Instr instr, Function function) {
if (!visited.contains(instr) && !instr.isPinned()) {
visited.add(instr);
BasicBlock root = function.getBasicBlocks().get(0);
instr.getBelongingBlock().getInstrArrayList().remove(instr);
root.addInstr(instr, root.getInstrArrayList().size() - 1);
GlobalCodeMovementUnit.addPath(instr, root);
instr.getOperands().forEach(v -> scheduleEarlyAnalysis(v, instr, function));
}
}
这部分我们会尽可能的把指令后移,确定每个指令能被调度到的最晚的基本块。每个指令也会被使用它们的指令限制,限制其不能无限向后移。这里我们同样可以直接参考教程提供的伪代码即可。
该函数对应的代码如下:
/**
* scheduleLate 用于在GCM中尽可能的把指令后移,
* 确定每个指令能被调度到的最晚的基本块。
* 每个指令也会被使用它们的指令限制,限制其不能无限向后移。
* 该函数执行逻辑如下:
* 1.如果已经处理过了,或者是无法移动,那么就结束处理。
* 2.如果未处理,遍历该指令的使用者,寻找LCA尝试后移。
* 3.如果该指令的使用者是Phi指令,那么遍历Phi指令的每个操作数,寻找LCA尝试后移。
*/
public static void scheduleLate(Instr instr) {
if (!visited.contains(instr) && !instr.isPinned()) {
visited.add(instr);
BasicBlock lcaBlock = null;
for (User user : instr.getUsers()) {
lcaBlock = scheduleLateAnalysis(user, instr, lcaBlock);
}
GlobalCodeMovementUnit.pickFinalPos(lcaBlock, instr);
BasicBlock bestBlock = instr.getBelongingBlock();
for (Instr instInst : bestBlock.getInstrArrayList()) {
if (!instInst.equals(instr) && !(instInst instanceof PhiInstr) &&
instInst.getOperands().contains(instr)) {
instr.getBelongingBlock().getInstrArrayList().remove(instr);
bestBlock.addInstr(instr, bestBlock.getInstrArrayList().indexOf(instInst));
GlobalCodeMovementUnit.addPath(instr, bestBlock);
break;
}
}
}
}
在确定每个指令可以被灵活调度的空间后,我们将进行最关键的一步,为指令选择它最终的基本块。这里选择的依据是循环深度尽可能浅且尽可能的靠前,这里循环分析就起作用了。这里对应代码如下:
/**
* pickFinalPos 用于在GCM中寻找指令最终所处的位置
* 该函数的执行逻辑如下:
* 1.如果该指令没有使用者,那么直接返回
* 2.如果该指令有使用者,那么遍历该指令的使用者,找到最佳位置
* 3.找到他们的LCA,如果他们是有共同祖先的,那么就将该指令插入到共同祖先的位置
* 4.如果共同祖先不是该指令的所在块,则尽量让循环深度变小
*/
private static void pickFinalPos(BasicBlock lcaBlock, Instr instr) {
BasicBlock posBlock = lcaBlock;
if (!instr.getUsers().isEmpty()) {
BasicBlock bestBlock = posBlock;
while ((posBlock.getBlockDominateParent() != null) &&
!posBlock.equals(instr.getBelongingBlock())) {
posBlock = posBlock.getBlockDominateParent();
if (posBlock.getLoopDepth() < bestBlock.getLoopDepth()) {
bestBlock = posBlock;
}
}
instr.getBelongingBlock().getInstrArrayList().remove(instr);
bestBlock.addInstr(instr, bestBlock.getInstrArrayList().size() - 1);
GlobalCodeMovementUnit.addPath(instr, bestBlock);
}
}
由于phi指令的存在,所以我们不能直接将phi指令转化为汇编,消PHI最简单的一个思路是在源基本块跳转之前将PHI指令拆成多条move指令,之后再将move指令转化为汇编即可完成相关工作。
要想将Phi指令转化为move指令,我们需要先将其转化为ParallelCopy指令并将其插入到跳转语句之前,而转化为ParallelCopy的方法就是将phiInstr的每个option都放到对应的ParallelCopy中,这里注意需要options按照前驱的顺序排列,然后删除Phi指令即可。这里可以参考课程教程伪代码如下:
而将ParallelCopy指令转化为Move指令的过程,我们首先需要将其直接转化为一个move数组的集合,之后需要通过添加临时变量的方法解决循环赋值以及共享寄存器等问题,这样就可以形成所需的move指令,相关伪代码如下:
而move指令的转汇编代码如下:
@Override
public void generateAssembly() {
super.generateAssembly();
Register dstReg = AssemblyUnit.getRegisterController().getRegister(to);
Register srcReg = AssemblyUnit.getRegisterController().getRegister(from);
if (dstReg != null && dstReg.equals(srcReg)) {
return;
}
dstReg = (dstReg == null) ? Register.K0 : dstReg;
srcReg = RegisterUtils.loadVariableValue(from, srcReg, dstReg);
if (!srcReg.equals(dstReg)) {
new MoveAsm(dstReg, srcReg);
}
RegisterUtils.memAllocReg(to, dstReg);
}
而这里在转move指令部分我们就需要用到分配寄存器,这些需要我们在后续使用图着色算法进行设计。
由于mips只有32个寄存器,所以如何进行寄存器分配是编译优化当中一个重要的问题。这里的基本思路是保留目前处于活跃状态的寄存器,如果当前寄存器全部分配完成,则需要依靠堆栈进行分配(不过这种情况基本不会出现)。
这里我先用枚举类Register来表示表示MIPS中的寄存器,代码如下:
public enum Register {
// MIPS registers
ZERO("$zero"),
V0("$v0"), V1("$v1"),
A0("$a0"), A1("$a1"), A2("$a2"), A3("$a3"),
T0("$t0"), T1("$t1"), T2("$t2"), T3("$t3"), T4("$t4"), T5("$t5"), T6("$t6"), T7("$t7"),
S0("$s0"), S1("$s1"), S2("$s2"), S3("$s3"), S4("$s4"), S5("$s5"), S6("$s6"), S7("$s7"),
T8("$t8"), T9("$t9"),
K0("$k0"), K1("$k1"), GP("$gp"), SP("$sp"), FP("$fp"), RA("$ra");
/**
* regName 是寄存器的名称,便于使用enum的值
*/
private final String regName;
Register(String regName) {
this.regName = regName;
}
/**
* regTransform 方法用于将int类型的index转换为Register类型,
* 类似于enum的valueOf方法
*/
public static Register regTransform(int index) {
return values()[index];
}
@Override
public String toString() {
return regName;
}
}
之后我们在寄存器分配控制器中装入所有可以使用的寄存器,然后完成相关分配工作即可,具体而言是我们需要设计var2reg和reg2var这两个实现value和寄存器的映射即可。
当我们需要分配寄存器给基本块时,首先遍历一边所有指令,记录每个变量在该基本块里最后一次被使用的位置,之后再遍历一遍所有指令,执行releaseReg方法,然后遍历其直接支配的节点,调用映射函数reflection并将该基本块定义的变量对应的寄存器释放,最后将“后继不再使用但是是从indBasicBlock传过来”的变量对应的寄存器映射恢复回来,即可完成申请寄存器操作。简要代码如下:
public static void blockAllocate(BasicBlock entry) {
HashMap<Value, Value> lastUseMap = new HashMap<>();
HashSet<Value> defined = new HashSet<>();
HashSet<Value> used = new HashSet<>();
entry.getInstrArrayList().forEach(instr -> instr.getOperands().forEach(
operand -> lastUseMap.put(operand, instr)));
entry.getInstrArrayList().forEach(instr -> releaseReg(entry, instr, lastUseMap,
var2reg, reg2var, defined, used));
entry.getBlockDominateChildList().forEach(RegisterAllocator::reflection);
for (Value value : defined) {
if (var2reg.containsKey(value)) {
reg2var.remove(var2reg.get(value));
}
}
for (Value value : used) {
if (var2reg.containsKey(value) && !defined.contains(value)) {
reg2var.put(var2reg.get(value), value);
}
}
}
而释放寄存器方面,我们分为两种情况讨论。如果该指令的某个operand是该基本块内的最后一次使用,并且该基本块的out中没有这个operand,那么我们可以暂时释放这个变量所占用的寄存器(释放reg2var,但不改变var2reg)。而如果该指令属于定义语句,并且不是创建数组的alloc指令,我们需要为该变量分配寄存器,代码逻辑如下:
private static void releaseReg(BasicBlock entry, Instr instr, HashMap<Value, Value> lastUseMap,
HashMap<Value, Register> var2reg,
HashMap<Register, Value> reg2var,
HashSet<Value> defined, HashSet<Value> used) {
if (!(instr instanceof PhiInstr)) {
for (Value operand : instr.getOperands()) {
if (lastUseMap.get(operand).equals(instr) && var2reg.containsKey(operand) &&
!entry.getOutBasicBlockHashSet().contains(operand)) {
reg2var.remove(var2reg.get(operand));
used.add(operand);
}
}
}
if (instr.isValid() && !(instr instanceof ZextInstr)) {
defined.add(instr);
Register reg = allocRegFor();
if (reg != null) {
if (reg2var.containsKey(reg)) {
var2reg.remove(reg2var.get(reg));
}
reg2var.put(reg, instr);
var2reg.put(instr, reg);
}
}
}
这样我们就完成了寄存器的分配,但还有一个重要的问题没有解决,哪些寄存器可以释放呢,这就需要活跃性分析决定哪些变量需要保留。
为了实现活跃性变量分析,首先我们需要生成Use-Def链,先使用后定义的变量放在use中,先定义后使用的变量放在def中,代码逻辑如下:
public void genUseDefAnalysis() {
for (Value operand : this.getOperands()) {
if (!this.getBelongingBlock().getDefBasicBlockHashSet().contains(operand) &&
(operand instanceof Instr || operand instanceof GlobalVar
|| operand instanceof Param)) {
this.getBelongingBlock().getUseBasicBlockHashSet().add(operand);
}
}
if (!this.getBelongingBlock().getUseBasicBlockHashSet().contains(this) && this.isValid()) {
this.getBelongingBlock().getDefBasicBlockHashSet().add(this);
}
}
之后需要根据之前获得的use和def的情况计算每个基本块的in集合和out集合,即根据后继的in,求出当前block的out,再根据公式in = (out - def) + use,求出当前基本块的in,如果in集合发生变化,则继续执行while循环,否则结束,代码逻辑如下:
public static void calculateInOut(Function function) {
ArrayList<BasicBlock> basicBlocks = function.getBasicBlocks();
boolean change = true;
while (change) {
change = false;
for (int i = basicBlocks.size() - 1; i >= 0; i--) {
BasicBlock basicBlock = basicBlocks.get(i);
HashSet<Value> out = new HashSet<>();
basicBlock.getBlockOutBasicBlock().forEach(
successor -> out.addAll(successor.getInBasicBlockHashSet()));
addOutBlockHashSet(basicBlock, out);
HashSet<Value> in = new HashSet<>(out);
in.removeAll(basicBlock.getDefBasicBlockHashSet());
in.addAll(basicBlock.getUseBasicBlockHashSet());
HashSet<Value> originIn = basicBlock.getInBasicBlockHashSet();
addInBlockHashSet(basicBlock, in);
if (!in.equals(originIn)) {
change = true;
}
}
}
}
由于在llvm中每个基本块的最后一定要有跳转语句,而mips如果最后没有跳转则可以自动顺延到下一个块,所以当llvm如果跳转的就是下一个块,则可以省略一条跳转语句,除此之外如果两个块的前序后继均为1则可以直接合并这两个块,这部分比较简单,在此不再详细说明。
至此编译器设计完整结束!!!
[1].Hyggge/Petrichor: Java 实现的 SysY - LLVM IR 编译器 (github.com)
[3].Chenrt-ggx/MipsCompiler: BUAA 1906 Mips Compiler (github.com)
[4].Thysrael/Pansy: 一个简单的编译 SysY 语言(C 语言子集)到 Mips 的编译器,采用 Java 实现。 (github.com)