广工编译原理课程设计_第1页
广工编译原理课程设计_第2页
广工编译原理课程设计_第3页
已阅读5页,还剩15页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、课程设计课程名称题目名称编译原理课程设计学生学院专业班级2011级计算机科学与技术 7班学 号 3111006002学生姓名谢佳旭2013年1月08日目录一、课程设计目的 5二、课程设计要求 51、设计内容 5三、课程设计环境与工具 6四、结构设计说明 51、PL/0 语言编译器 52、总流程图 63、各功能模块描述 6五、主要成分描述 7六、测试用例 91、测试 *= 和 /= 的运算功能 92、测试 FOR 语句功能 9七、开发过程和完成情况 91、扩充赋值运算: *= 和 /= 92、扩充 FOR 语句( Pascal )103、添加单行注释 12八、心得体会 14一、课程设计目的在分析

2、理解一个教学型编译程序(如PL/O )的基础上,对其词法分析程序、语法分析程序和语义处理程序进行部分修改扩充。达到进 一步了解程序编译过程的基本原理和基本实现方法的目的。二、课程设计要求1.课程设计基本内容(成绩范围:“中”、“与格”或“不与格”)(1 )扩充赋值运算:*=和/=扩充语句(Pascal的FOR语句):FOR 变量 := 表达式STEP表达式 UNTILv 表达式Do 语 句选做内容(成绩评定范围扩大到:“优”和“良”)(1) 增加类型: 字符类型; 实数类型。(2) 增加 注释;注释由/*和*/包含;(3) 扩充函数: 有返回值和返回语句;有参数函数。(4) 增加一维数组类型(

3、可增加指令)。(5) 其他典型语言设施。三、课程设计环境与工具(1 )运行平台:WIN7旗舰SP1(2) 实现工具:C+Builder 6.0(3) 教学型编译程序:PL/0四、结构设计说明和各功能模块描述(1) PL/0语言编译器PL/0语言可看成是PASCAL语言的子集,它的编译程序是一个编 译解释执行系统。PL/0的目标程序为假想栈式计算机的汇编语言,与具体计算机无关。(2) PL/O编译程序的过程或函数的功能表1、 void Error(int n):错误处理 打印出错位置和错误编码2、void GetCh():漏掉空格,读取一个字符3、void GetSym():词法分析,读取一个单

4、词4、 void GEN(FCT X, int Y, int Z):生成目标代码,并送入目标程序区5、void TEST(SYMSET S1, SYMSET S2, int N):测试当前单词符号是否合法:登录名6、void ENTER(OBJECTS K, int LEV, int &TX, int &DX)字表7、int POSITION(ALFA ID, int TX) :查找标识符在名字表中的位 置8 、void ConstDeclaration(int LEV,int &TX,int &DX):常量说明处理9、 void VarDeclaration(

5、int LEV,int &TX,int &DX):变量说明处理10 、void CharDeclaration(int LEV,int &TX,int &DX):字符说明处理11 、 void RealDeclaration(int LEV,int &TX,int &DX):实数说明处理12 、 void ListCode(int CX0) :列出目标代码清单13 、 void FACTOR(SYMSET FSYS, int LEV, int &TX):因子处理14 、 void TERM(SYMSET FSYS, int LEV, in

6、t &TX):项处理15 、 void EXPRESSION(SYMSET FSYS, int LEV, int &TX):表达式处理16 、 void CONDITION(SYMSET FSYS,int LEV,int &TX) :条件 处理17 、void STATEMENT(SYMSET FSYS,int LEV,int &TX):语句部分处理18 、 void Block(int LEV, int TX, SYMSET FSYS):分程序分析处理过程19 、 int BASE(int L,int B,int S):通过静态链求出数据区的基地址20、voi

7、d Interpret() :对目标代码的解释执行程序21、 void_fastcallTForm1:Butto nRun Click(TObject*Sender):进行编译,RUN(3) PL/0编译程序的总流程图五、主要成分描述1、符号表在编译程序中符号表用来存放语言程序中出现的有关标识符的属性信 息,符号表中所登记的信息在编译的不同阶段都要用到。在语义分析中,符号表所登记的内容将用于语义检查(如检查一个名 字的使用和原先的说明是否一致)和产生中间代码。在目标代码生成阶段,当对符号名进行地址分配时,符号表是地址分 配的依据。对一个多遍扫描的编译程序,不同遍所用的符号表也往往各有 不同。因

8、为每遍所关心的信息各有差异。一张符号表的每一项(或称入口才包含两大栏(或称区段、字域) ,即 名字栏 (NAME) 信息栏 (INFORMATION ) 信息栏包含许多子栏和标志位, 用来记录相应名字和种种不同属性, 由于查填符号表一般是通过匹配名字来寮现的,因此,名字栏也称主栏。 主栏的内容称为关键字( key word )。2 、运行时存储组织和管理由于编译时目标程序运行的数据空间大小已经规定,所以存储组织属 于静态存储。源程序的标识符存放在 TABLE表中,目标代码存放在 CODE中,S是由解释程序定义的一维整型数组,是程序运行时的数据存储空间。解释程序还定义了 4个寄存器:1) P:程

9、序地址寄存器:指向下一条要执行的目标程序的地址(相当 目标程序CODE数组的下标)。2) I :指令寄存器:存放着当前正在解释的下一条目标指令。3) T:栈顶寄存器:由于每个过程当它被运行时,给它分配的数据空 间(下边称数据段)包括静态部分和动态部分,对于动态部分,要注意的 是,栈顶寄存器指出了当前栈中最新分配的单元。4) B:基址寄存器:指向每个过程被调用时,在数据区S中给它分配 的数据段的起始地址,也称基地址2.1中间代码表示对PL/0编译程序的目标代码的指令格式描述如下:其中f代表功能码,I表示层次差,a的含意对不同的指令有所区别,见下面对每条指令的解释说明:lit 0 a将常数值取到栈

10、顶,a为常数值Lod l a将变量值取到栈顶,a为偏移量,l为层差SSTEPl a将栈顶内容送入某变量单元中,a为偏移量,l为层差Cal l a调用过程,a为过程地址,l为层差Int 0 a在运行栈中为被调用的过程幵辟 a个单元的数据区jmp 0 a无条件跳转至a地址Jpc 0 a条件跳转,当栈顶布尔值非真则跳转至a地址,否则顺序执行opr 0 0过程调用结束后,返回调用点并退栈opr 0 1栈顶兀素取反opr 0 2次栈顶与栈顶相加,退两个栈兀素,结果值进栈opr 0 3次栈顶减去栈顶,退两个栈元素,结果值进栈opr 0 4次栈顶乘以栈顶,退两个栈元素,结果值进栈opr 0 5次栈顶除以栈顶

11、,退两个栈元素,结果值进栈opr 0 6栈顶元素的奇偶判断,结果值在栈顶opr 0 7opr 0 8次栈顶与栈顶是否相等,退两个栈兀素,结果值进栈opr 0 9次栈顶与栈顶是否不等,退两个栈兀素,结果值进栈opr 0 10次栈顶是否小于栈顶,退两个栈兀素,结果值进栈opr 0 11次栈顶是否大于等于栈顶,退两个栈兀素,结果值进栈opr 0 12次栈顶是否大于栈顶,退两个栈兀素,结果值进栈opr 0 13次栈顶是否小于等于栈顶,退两个栈兀素,结果值进栈opr 0 14栈顶值输出至屏幕opr 0 15屏幕输出换行opr 0 16从命令行读入一个输入置于栈顶2.2语法分析方法自顶向下的语法分析程序分

12、程序变量说明部分 语句/ I、 IVAR 标识符 ; 复合语句ABEGIN 语句 END读语句/ 、READ (标识符)A六、测试用例1. 测试*=和/=的运算功能,测试文件为P9101 .PLO=COMPILE PLO =0 PROGRAM EX01;0 VAR A.B.C;1 BEGIN2 A:=25;4 B:=17;6 C:=A"B;10 A/2;B*=2;19 C*-A;22 WRITE(A);25 WRITE(B);28 WRITE(C);31 END.-w-v-* RUN FLO *123496 EMO PT-0 *P9101目标代码 r显示“不显示RUN2.测试FOR语

13、句功能,测试文件为 P9104.PLOFOR 变量:= 表达式 STEPv表达式 DO 语句FOR 变量:= 表达式 UNTILv 表达式 DO 语句:=COMPILE PLO 0PROGRAMMAIN;0VAR A#B,CfD;1BEGIN2C:=l;4D:=1024;6FOR A:二1 TO a* PL/0 Compiler Demo *P9104目标代码'显乔“不显示16 C*-2;21 FOR B:= 10 DOWNTO 1 DORUN31 D/=2;36 WRITE(C);3Q WRITE(D);42 END+gx RUN PLO *-*- 1632 ITTd IT) PL

14、Q *七、幵发过程和完成情况1、扩充赋值运算:和/=思路:乘等和除等运算的实现方法近似,故把两者看成一个看待。采用临时变量存储这两个运算符,再分别对这两种情况处理。具体代码实现:if(SYM=ADDEQ | SYM=DIVEQ ) /遇到*= 和/=ADDOP=SYM;/ 临时存放运算符if(i!=0) / 开始处理GetSym();GEN(LOD,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR); /取值到栈顶EXPRESSION(FSYS,LEV,TX);if(ADDOP=ADDEQ)/ 遇到*= 时GEN(OPR,0,4); / 次栈顶乘以栈顶 ,退两个栈元素 ,结

15、果值进栈 .else / 遇到 /= 时GEN(OPR,0,5); / 次栈顶乘以栈顶 ,退两个栈元素 ,结果值进栈GEN(STO,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 出栈取值到内存2、扩充 FOR 语句( Pascal ) FOR 变量 := 表达式 STEPv表达式 DO 语句 FOR 变量 := 表达式 UNTILv 表达式 DO 语句其中,语句的循环变量的步长为2 ,语句的循环变量的步长为 -2。以下为新增代码段:/ 开始插入新增代码case FORSYM:GetSym();if(SYM != IDENT) Error(32);else i=POS

16、ITION(ID,TX);else ifif(i = 0) Error(11);(TABLEi.KIND!=VARIABLE) /*ASSIGNMENT STEPNON-VARIABLE*/Error(12); i=0; / 赋值语句中,赋值号左部标识 符属性应是变量/ 否则报错GetSym();if(SYM = BECOMES) GetSym();else Error(13); / 赋值语句左部标识符后应是赋值号 := 否则报错EXPRESSION(SymSetUnion(SymSetNew(TOSYM,DOWNTOSYM),FS YS),LEV,TX);/EXPRESSION(FSYS,L

17、EV,TX);if (i!=0)GEN(STO,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 保存初值if(SYM = TOSYM)/FOR .TO 组合,即递增CX1=CX; / 保存循环开始点GEN(LOD,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 将循环判断变量取出放到栈顶GetSym();EXPRESSION(SymSetUnion(SymSetNew(DOSYM),FSYS),LEV,TX);/* 判断循环变量条件,看是否跳出循环 */GEN(OPR,0,13); / 生成比较指令,看循环变量 是否依旧满足条件CX2=CX;

18、 / 保存循环结束点GEN(JPC,0,0); / 生成条件跳转指令,跳出循 环,跳出的地址未知GEN(LOD,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 将循环变量取出放在栈顶GEN(LIT,0,2); / 将步长取到栈顶GEN(OPR,0,2); / 循环变量加步长,实现逐级递 增GEN(STO,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 将栈顶的值存入循环变量if(SYM=DOSYM) / 处理循环体GetSym();STATEMENT(FSYS,LEV,TX);else Error(48);GEN(JMP,0,CX1);/ 无

19、条件跳转到循环开始点 CODECX2.A=CX; /* 回填循环结束点的地址,CX为ELSE后语句执行完的位置,它正是前面未 定的跳转地址 */else if(SYM=DOWNTOSYM)/FOR. DOWNTO.组合,即递减CX1=CX;/ 保存循环开始点GEN(LOD,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 将循环判断变量取出放到栈顶GetSym();EXPRESSION(SymSetUnion(SymSetNew(DOSYM),FSYS),LEV,TX);/* 判断循环变量条件,看是否跳出循环 */GEN(OPR,0,11);/ 生成比较指令 ,看循环变

20、量 是否依旧满足条件CX2=CX; / 保存循环结束点GEN(JPC,0,0);GEN(LOD,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 将循环变量取出放在栈顶GEN(LIT,0,2); / 将步长取到栈顶GEN(OPR,0,3); / 循环变量加步长,实现逐级 递减GEN(STO,LEV-TABLEi.vp.LEVEL,TABLEi.vp.ADR);/ 将栈顶的值存入循环变量if(SYM=DOSYM) / 处理循环体 GetSym();STATEMENT(FSYS,LEV,TX);/ 循环 体处理else Error(48);GEN(JMP,0,CX1);/ 无条件跳转到循环开始点CODECX2.A=CX; /* 回填循环结束点的地址,CX为ELSE后语句执行完的位置,它正是前面未定的跳转地址 */else Error(47);break;/ 结束插入新增代码3、添加单行注释在程序中,形如 /C:=1; 的代码段将会被忽略if(c

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论