编译器的最后一部分,做了5个星期总算全部做完了。
这个Location class 蛮重要的 记录了在当前这个Scope 中 base 的位置和offset多少位。 比如说进入一个新的function scope里,可能有许多的变量之类的 我们需要知道变量在多少个offset。
data:image/s3,"s3://crabby-images/89600/89600c66b189251daac243398ef52266314428d9" alt=""
dispatchTable这个HashMap 包含的是ClassName, _从parent到当前所有defined的方法__
这个部分很Tricky,我们最后有一个case卡了很久没过去就是在这个地方。
data:image/s3,"s3://crabby-images/9a1d6/9a1d694eda9ddf1f5432e4f795ec34f257a42625" alt=""
这个是classDispatchTable fill的过程:
首先需要往MIPS 代码里output出每个类的DispatchTables.
然后还要往HashMap里填充。这里的代码也许有点重复多余 但是由于时间限制没时间去优化。。。
大概idea就是每到一个class,从低往上一路追到Parent,把父类装入一个orderList里
然后从后往前travel这个list,就有点像是从ancestor往子孙类走。然后一路把父类的method纳入子类的dispatchTable里!<className, <methodName, count>>
data:image/s3,"s3://crabby-images/b56a5/b56a533507ca8931149ce5913ceb5295e4d829b5" alt=""
这里有一个Tricky部分要注意的就是如果parent定义了一个方法,当前类也定义了同一个名字的方法,是需要改写的!!!
data:image/s3,"s3://crabby-images/04a5a/04a5aa7a7c38f2bee801ac37debdcb7832509793" alt=""
初始化Object。这个部分负责把一些初始值绑定给变量。
比如a= 2+3 我们在初始化阶段就要去把2+3 算出来 赋值给a。
然后如果有的变量是不需要的话,我们可以在这个部分执行辣鸡回收。
data:image/s3,"s3://crabby-images/ecdca/ecdca86bc90575cdbd865341384f2cde2720fc12" alt=""
data:image/s3,"s3://crabby-images/071d0/071d0c3514985b0c763ebc86d8f9442572fa5442" alt=""
这个部分负责给方法函数生成MIPS 代码:
//setCurrentClass 应该是不需要的。。
遍历所有的classes,然后创建一层一层的scope, one scope for each class.
一个scope 其实就是放置一个新的hashMap到stack顶部。hashmap里包含了这个class里所有的变量名, 以及在object的哪个位置。【object前几个位置是用来放classTag, classSize, dispatchTable等信息】
然后对于class里的每个local Defined的方法,我们生成MIPS代码。【为什么不需要生成父类method的代码?因为我们可以refer back to 之前生成好的代码】
data:image/s3,"s3://crabby-images/dbf60/dbf6068b4eaeba467c684999aaecf99db374178a" alt=""
生成好一个class,就可以pop出一个class的scope
data:image/s3,"s3://crabby-images/23282/232827010e2bbddd7673af0c2e09eae82cb4c3cd" alt=""
Assign:
这里会用到之前用到过的Location class. 赋值过程就是把expr1+expr2比如计算出来放在a0 register里的值store到变量define的地方。我们用location base+offset找到那个地方。
data:image/s3,"s3://crabby-images/446cb/446cb8f553a5ce28bc41a5ec9a72d673c1b014c3" alt=""
static Dispatch:
data:image/s3,"s3://crabby-images/80003/800034c327c968c1339c3e5ed864fe01961b4663" alt=""
dispatch:
最下面if else可以替换成一行。我们其实只要用expr.get_type()就行了 它会准确告诉我们当前Type是什么。这个部分我卡了超久,主要是整个structure没想好。。
setCurclass只要在最外层调用的时候才使用到,如果一个Main class里new 一个class A.然后调用A.foo() 那么A是不知道自己的curclass是多少的,以为自己是Main。然后去Dispatch<main>里找 foo() 这个方法。expr.get_type()会知道当前type是多少。
data:image/s3,"s3://crabby-images/c05b4/c05b41aa982ba91d628330b9c8c17bcec38312a1" alt=""
Cool里面的Block是当有多行代码的时候使用的:例如
{
{
a=1;
b=2;
}
}
分别给每个expr 生成代码
data:image/s3,"s3://crabby-images/33294/33294094b4e4f9ed7190485826a31ed90a22dde7" alt=""
***let
data:image/s3,"s3://crabby-images/eac58/eac5802f1e84419e7d92f41ac3c9c4ed35412119" alt=""
例子: 加法
expr1 + expr2 我们得分别给expr1和expr2生成MIPS代码 然后加起来。这里会用到一些register。比如expr1 结果会在a0,我们要先存到s1里 然后再去执行expr2存到a0.
最后a0 = a0+s1.
data:image/s3,"s3://crabby-images/aeb80/aeb8058cbfca39632886841a118c94e931d74745" alt=""
很难的一个test case.. 主要在于他这里的一个dispatch case
a.m()
data:image/s3,"s3://crabby-images/a079f/a079ffc56db372bf5d5d9a9f529bfa47231e912d" alt=""
(new B)@A.foo() 这个是static dispatch
意思就是B虽然是一个类型B的东西,并且继承了类型A。但是他就是想用父类A里的method foo不用继承后改写的!
有点像java里面的super。
data:image/s3,"s3://crabby-images/b1f69/b1f696af3e3fa38f1fd0d454fc15c770772c74a1" alt=""
外面一开始有会有一个x 变量,但是function内部也有一个x。
但是由于我们会有一个scope for foo() 所以 会先看看这个scope有没有x,找到location。
data:image/s3,"s3://crabby-images/52af4/52af47a88e816d1647a882dfd2a44107f68b60b5" alt=""
这个case的话,先在foo() scope里找有没有x,没有的话去parent scope里找x
data:image/s3,"s3://crabby-images/95159/95159b0e7188a7fc1785725b4bbf966cb26eb564" alt=""
网友评论