編譯原理課程設(shè)計-- pl0語言的擴充_第1頁
已閱讀1頁,還剩28頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)

文檔簡介

1、<p>  《編譯原理課程設(shè)計》</p><p>  項目名稱 PL/0語言的擴充 </p><p>  學(xué) 院 計算機學(xué)院 </p><p><b>  專業(yè)班級 </b></p><p>  學(xué) 號 </p>

2、<p>  姓 名 </p><p>  指導(dǎo)教師 </p><p>  2013 年 1 月 3 日</p><p>  一. 課程設(shè)計目的0</p><p>  二. 課程設(shè)計要求0</p><p><b>

3、;  基本內(nèi)容:0</b></p><p><b>  選做內(nèi)容:0</b></p><p><b>  已實現(xiàn)的功能:0</b></p><p>  三. 編譯環(huán)境與工具0</p><p>  四. 結(jié)構(gòu)設(shè)計說明0</p><p>  1.PL/0編譯

4、程序的結(jié)構(gòu)圖0</p><p>  2.PL/0編譯程序的過程或函數(shù)的功能表1</p><p>  3.PL/0編譯程序的總體流程圖1</p><p>  4.PL/0編譯程序的中間代碼2</p><p>  5.PL\0的編譯程序的過程和函數(shù)的功能3</p><p>  五. 課程設(shè)計的設(shè)計與步驟3<

5、;/p><p>  1.擴充賦值運算:+= -= *= /= ++ 和- -5</p><p>  2.增加Pascal的FOR語句15</p><p><b>  3.一維數(shù)組17</b></p><p>  六. 調(diào)試測試19</p><p>  1.+= -= *= /= ++ --

6、的測試19</p><p>  2.FOR語句的測試21</p><p>  3.數(shù)組的調(diào)試22</p><p><b>  4.綜合調(diào)試23</b></p><p>  七. 課程設(shè)計總結(jié)25</p><p><b>  【參考文獻】26</b></p&g

7、t;<p><b>  課程設(shè)計目的</b></p><p>  在分析理解一個教學(xué)型編譯程序(如PL/0)的基礎(chǔ)上,對其詞法分析程序、語法分析程序和語義處理程序進行部分修改擴充。達到進一步了解程序編譯過程的基本原理和基本實現(xiàn)方法的目的。</p><p><b>  課程設(shè)計要求</b></p><p>&l

8、t;b>  基本內(nèi)容:</b></p><p> ?。?)擴充賦值運算:*= 和 /=</p><p> ?。?)擴充語句(Pascal的FOR語句):</p><p> ?、貴OR <變量>:=<表達式> TO <表達式> DO <語句></p><p> ?、贔OR <

9、;變量>:=<表達式> DOWNTO <表達式> DO <語句></p><p>  其中,語句①的循環(huán)變量的步長為2,語句②的循環(huán)變量的步長為-2。</p><p> ?。?)增加運算:++ 和 --。</p><p><b>  選做內(nèi)容:</b></p><p>  (1)

10、增加類型:① 字符類型;② 實數(shù)類型。</p><p> ?。?)擴充函數(shù):① 有返回值和返回語句;② 有參數(shù)函數(shù)。</p><p>  (3)增加一維數(shù)組類型(可增加指令)。</p><p> ?。?)其他典型語言設(shè)施。</p><p><b>  已實現(xiàn)的功能:</b></p><p> ?。?/p>

11、1)擴充賦值運算:*=和 /= (另外也擴充了+= 和 -=)</p><p> ?。?)擴充語句(Pascal的FOR語句):</p><p>  ①FOR <變量>:=<表達式> TO <表達式> DO <語句></p><p> ?、贔OR <變量>:=<表達式> DOWNTO <表

12、達式> DO <語句></p><p> ?。?)增加運算:++ 和 --。</p><p>  (4)增加一維數(shù)組類型(可增加指令)。</p><p><b>  編譯環(huán)境與工具</b></p><p>  編譯環(huán)境:Microsoft Windows 7 (64bit)</p><

13、;p>  編譯工具:Microsoft Visual C++ 6.0</p><p><b>  結(jié)構(gòu)設(shè)計說明</b></p><p>  1.PL/0編譯程序的結(jié)構(gòu)圖</p><p>  2.PL/0編譯程序的過程或函數(shù)的功能表</p><p>  表4-1 PL/0編譯程序的過程或函數(shù)的功能表</p&g

14、t;<p>  3.PL/0編譯程序的總體流程圖</p><p>  圖4-2 PL/0編譯程序的總體流程圖</p><p>  4.PL/0編譯程序的中間代碼</p><p>  對PL/0編譯程序的目標代碼的指令格式描述如下:</p><p>  f l a</p>

15、<p>  其中f代表功能碼,l表示層次差,a的含意對不同的指令有所區(qū)別,見下面對每條指令的解釋說明:</p><p>  表4-2 PL/0編譯程序的目標指令</p><p>  5.PL\0的編譯程序的過程和函數(shù)的功能</p><p><b>  Pl0:主程序</b></p><p>  Error:出

16、錯處理,打印出錯位置和錯誤編碼</p><p>  Getsym:詞法分析,讀取一個單詞</p><p>  Getch:漏掉空格,讀取一個字符</p><p>  Gen:生成目標代碼,并送入目標程序區(qū)</p><p>  Test:測試當前單詞符號是否合法</p><p>  Block:分程序分析處理過程<

17、/p><p>  Enter:登陸名字表</p><p>  Position:查找標識符在名字表中的位置</p><p>  Constdeclaration:常量定義處理</p><p>  Vardeclaration:變量說明處理</p><p>  Listcode:列出目標代碼清單</p><

18、;p>  Statement:語句部分處理</p><p>  Expression:表達式處理</p><p><b>  Term:項處理</b></p><p>  Factor:因子處理</p><p>  Condition:條件處理</p><p>  Interpret:對目標

19、代碼的解釋執(zhí)行程序</p><p>  Base:通過靜態(tài)鏈求出數(shù)據(jù)區(qū)的基地址</p><p>  課程設(shè)計的設(shè)計與步驟</p><p>  本次課程設(shè)計是在實驗的基礎(chǔ)上,在實驗中,已經(jīng)實現(xiàn)對PL/0作以下修改擴充:</p><p>  (1)增加單詞:保留字 ELSE,F(xiàn)OR,STEP,UNTIL,RETURN </p>&l

20、t;p>  運算符 +=,-=,++,--,∧,∨,┓</p><p> ?。?)修改單詞:不等號# 改為 <></p><p> ?。?)增加條件語句的ELSE子句</p><p>  在課程設(shè)計中,因為關(guān)鍵字增加到了18個,所以令 # define norw 18。SYMBOL擴展為48個值。在pl0.h中做了如下更改,具體如下(粗體部分為課程

21、設(shè)計更改部分):</p><p>  enum symbol{</p><p>  nul, ident, number, plus, minus,</p><p>  times, slash, oddsym, eql, neq,</p><p>  lss, leq, g

22、tr, geq, lparen,</p><p>  rparen, comma, semicolon,period, becomes,</p><p>  beginsym, endsym, ifsym, thensym, whilesym,</p><p>  writesym, readsym, dosym, callsym

23、, constsym,</p><p>  varsym, procsym, elsesym, forsym, stepsym, untilsym, returnsym,</p><p>  plusbecomes, dplus, minusbecomes, dminus, and, or, not,lepa,ripa,</p><p>  slashbecom

24、es,timesbecomes,</p><p><b>  };</b></p><p>  #define symnum 48 </p><p>  符號和關(guān)鍵字擴展(粗體部分為課程設(shè)計更改部分):</p><p>  ssym['+']=plus;</p><p>  ssy

25、m['-']=minus;</p><p>  ssym['*']=times;</p><p>  ssym['/']=slash;</p><p>  ssym['(']=lparen;</p><p>  ssym[')']=rparen;</p>

26、;<p>  ssym['=']=eql;</p><p>  ssym[',']=comma;</p><p>  ssym['.']=period;</p><p>  //ssym['#']=neq; 去掉#,改為<></p><p>  ssym

27、[';']=semicolon;</p><p>  ssym['!']=not; //非</p><p>  ssym['[']=lepa;//一維數(shù)組的左括號[</p><p>  ssym[']']=ripa;//一維數(shù)組的右括號]</p><p>  /*設(shè)置保留字名字

28、,按照字母順序,便于折半查找*/</p><p>  strcpy(&(word[0][0]),"begin");</p><p>  strcpy(&(word[1][0]),"call");</p><p>  strcpy(&(word[2][0]),"const");<

29、/p><p>  strcpy(&(word[3][0]),"do");</p><p>  strcpy(&(word[4][0]),"else");</p><p>  strcpy(&(word[5][0]),"end");</p><p>  strcpy

30、(&(word[6][0]),"for");</p><p>  strcpy(&(word[7][0]),"if");</p><p>  strcpy(&(word[8][0]),"odd");</p><p>  strcpy(&(word[9][0]),"p

31、rocedure");</p><p>  strcpy(&(word[10][0]),"read");</p><p>  strcpy(&(word[11][0]),"return");</p><p>  strcpy(&(word[12][0]),"step");&

32、lt;/p><p>  strcpy(&(word[13][0]),"then");</p><p>  strcpy(&(word[14][0]),"until");</p><p>  strcpy(&(word[15][0]),"var");</p><p>

33、;  strcpy(&(word[16][0]),"while");</p><p>  strcpy(&(word[17][0]),"write");</p><p>  /*設(shè)置保留字符號*/</p><p>  wsym[0]=beginsym;</p><p>  wsym[1]

34、=callsym;</p><p>  wsym[2]=constsym;</p><p>  wsym[3]=dosym;</p><p>  wsym[4]=elsesym;</p><p>  wsym[5]=endsym;</p><p>  wsym[6]=forsym;</p><p&g

35、t;  wsym[7]=ifsym;</p><p>  wsym[8]=oddsym;</p><p>  wsym[9]=procsym;</p><p>  wsym[10]=readsym;</p><p>  wsym[11]=returnsym; </p><p>  wsym[12]=stepsym;&l

36、t;/p><p>  wsym[13]=thensym;</p><p>  wsym[14]=untilsym;</p><p>  wsym[15]=varsym;</p><p>  wsym[16]=whilesym;</p><p>  wsym[17]=writesym;</p><p>

37、;  設(shè)置指令名稱(粗體部分為課程設(shè)計更改部分):</p><p>  strcpy(&(mnemonic[lit][0]),"lit");</p><p>  strcpy(&(mnemonic[opr][0]),"opr");</p><p>  strcpy(&(mnemonic[lod][0])

38、,"lod");</p><p>  strcpy(&(mnemonic[sto][0]),"sto");</p><p>  strcpy(&(mnemonic[cal][0]),"cal");</p><p>  strcpy(&(mnemonic[inte][0]),"

39、;int");</p><p>  strcpy(&(mnemonic[jmp][0]),"jmp");</p><p>  strcpy(&(mnemonic[jpc][0]),"jpc");</p><p>  strcpy(&(mnemonic[gar][0]),"gar&qu

40、ot;);</p><p>  strcpy(&(mnemonic[sar][0]),"sar");</p><p>  strcpy(&(mnemonic[shd][0]),"shd");</p><p>  strcpy(&(mnemonic[del][0]),"del");&l

41、t;/p><p>  strcpy(&(mnemonic[jud][0]),"jud");</p><p>  strcpy(&(mnemonic[tra][0]),"tra");</p><p>  1.擴充賦值運算:+= -= *= /= ++ 和- -</p><p>  1)在s

42、ymbol增加字符:</p><p>  Plusbecomes:+=</p><p>  Minusbecomes:-=</p><p>  Slashbecomes:*=</p><p>  Timesbecomes:/=</p><p>  2)對取字符getsym()函數(shù)進行擴充,具體如下(粗體部分為更改部分)

43、:</p><p>  int getsym()</p><p><b>  {</b></p><p>  ...//此處省略部分未修改過的代碼</p><p>  else if(ch=='+') //讀到加號</p><p><b>  {</b>&l

44、t;/p><p>  getchdo; //再讀一個字符</p><p>  if(ch=='=') //讀到等號,則與+號構(gòu)成加等</p><p><b>  {</b></p><p>  sym=plusbecomes;</p><p><b>  getchdo;&

45、lt;/b></p><p><b>  }</b></p><p>  else if(ch=='+') //讀到加號,則與加號構(gòu)成++</p><p><b>  {</b></p><p>  sym=dplus;</p><p>  getchd

46、o; </p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><p>  sym=plus; //那就是一個單獨的加號</p><p><b>  }</b

47、></p><p><b>  }</b></p><p>  else if(ch=='-') //讀到減號</p><p><b>  {</b></p><p><b>  getchdo;</b></p><p>  if

48、(ch=='=') //讀到等號,則與減號構(gòu)成減等</p><p><b>  {</b></p><p>  sym=minusbecomes;</p><p><b>  getchdo;</b></p><p><b>  }</b></p>

49、;<p>  else if(ch=='-') //讀到減號,則與減號構(gòu)成--</p><p><b>  {</b></p><p>  sym=dminus;</p><p><b>  getchdo;</b></p><p><b>  }</

50、b></p><p><b>  else</b></p><p><b>  {</b></p><p>  sym=minus; //那就是一個單獨的減號</p><p><b>  }</b></p><p><b>  }<

51、/b></p><p>  else if(ch=='*') </p><p><b>  {</b></p><p><b>  getchdo;</b></p><p>  if(ch=='=') //讀到乘號,則與減號構(gòu)成乘等</p>&

52、lt;p><b>  {</b></p><p>  sym=timesbecomes;</p><p><b>  getchdo;</b></p><p><b>  }</b></p><p><b>  else</b></p>

53、<p><b>  {</b></p><p>  sym=times; //那就是一個單獨的乘</p><p><b>  }</b></p><p><b>  }</b></p><p>  else if(ch=='/') </p>

54、;<p><b>  {</b></p><p><b>  getchdo;</b></p><p>  if(ch=='=') //讀到乘號,則與減號構(gòu)成除等</p><p><b>  {</b></p><p>  sym=slashb

55、ecomes;</p><p><b>  getchdo;</b></p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><p>  sym=slas

56、h; //那就是一個單獨的除</p><p><b>  }</b></p><p><b>  }</b></p><p>  ...//此處省略部分未修改過的代碼</p><p><b>  }</b></p><p>  3)對statemen函數(shù)

57、進行擴展</p><p>  int statement(bool* fsys,int * ptx,int lev)</p><p><b>  {</b></p><p>  ...//此處省略部分未修改過的代碼</p><p>  if(sym==ident)</p><p><b>

58、;  {</b></p><p>  i=position(id,*ptx);</p><p><b>  if(i==0)</b></p><p><b>  {</b></p><p>  error(11);</p><p><b>  }<

59、/b></p><p><b>  else</b></p><p><b>  {</b></p><p>  if(table[i].kind!=variable&&table[i].kind!=array)</p><p><b>  {</b>&l

60、t;/p><p>  error(12);</p><p><b>  i=0;</b></p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p>

61、<p><b>  getsymdo;</b></p><p>  if(sym==lepa)//如果是數(shù)組</p><p><b>  {</b></p><p>  getsymdo; //處理[]里的表達式</p><p>  memcpy(nxtlev,fsys,sizeof(b

62、ool)* symnum);</p><p>  nxtlev[ripa]=true;</p><p>  expressiondo(nxtlev,ptx,lev);</p><p>  gendo(tra,0,table[i].size);//生成將數(shù)組下標范圍入棧指令</p><p>  gendo(jud,0,0);//判斷下標合法性&

63、lt;/p><p>  if(sym!=ripa)</p><p><b>  {</b></p><p>  error(26);</p><p><b>  }</b></p><p><b>  getsymdo;</b></p><

64、;p><b>  }</b></p><p>  if(sym==becomes)//賦值:=</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p><b>  flag=1;</b><

65、/p><p><b>  }</b></p><p>  else if(sym==plusbecomes)//加賦值+=</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p><b>  

66、flag=2;</b></p><p><b>  }</b></p><p>  else if(sym==minusbecomes)//減賦值-=</p><p><b>  {</b></p><p><b>  getsymdo;</b></p>

67、<p><b>  flag=3;</b></p><p><b>  }</b></p><p>  else if(sym==dplus)//++</p><p><b>  {</b></p><p><b>  getsymdo;</b&g

68、t;</p><p><b>  flag=4;</b></p><p><b>  }</b></p><p>  else if(sym==dminus)//--</p><p><b>  {</b></p><p><b>  gets

69、ymdo;</b></p><p><b>  flag=5;</b></p><p><b>  }</b></p><p>  else if(sym==timesbecomes)//*=</p><p><b>  {</b></p><p

70、><b>  getsymdo;</b></p><p><b>  flag=6;</b></p><p><b>  }</b></p><p>  else if(sym==slashbecomes)// /=</p><p><b>  {</b&

71、gt;</p><p><b>  getsymdo;</b></p><p><b>  flag=7;</b></p><p><b>  }</b></p><p><b>  else</b></p><p><b&g

72、t;  {</b></p><p>  error(13);</p><p><b>  }</b></p><p><b>  if(i!=0)</b></p><p><b>  {</b></p><p>  if(flag==1) /

73、/:=</p><p><b>  {</b></p><p>  memcpy(nxtlev,fsys,sizeof(bool)* symnum);</p><p>  expressiondo(nxtlev,ptx,lev);</p><p><b>  }</b></p><

74、;p>  else if(flag==2)//+=</p><p><b>  {</b></p><p>  memcpy(nxtlev,fsys,sizeof(bool)* symnum);</p><p>  expressiondo(nxtlev,ptx,lev);</p><p>  gendo(lod,

75、lev-table[i].level,table[i].adr);//先取值到棧頂</p><p>  gendo(opr,lev-table[i].level,2);//執(zhí)行加法</p><p><b>  }</b></p><p>  else if(i!=0&&flag==3)//-=</p><p&

76、gt;<b>  {</b></p><p>  memcpy(nxtlev,fsys,sizeof(bool)* symnum);</p><p>  expressiondo(nxtlev,ptx,lev);</p><p>  gendo(lod,lev-table[i].level,table[i].adr);//先取值到棧頂</

77、p><p>  gendo(opr,lev-table[i].level,3);//執(zhí)行減法</p><p>  gendo(opr,lev-table[i].level,1);//取反</p><p><b>  }</b></p><p>  else if(i!=0&&flag==4) //++<

78、/p><p><b>  {</b></p><p>  if(table[i].kind==variable)</p><p><b>  {</b></p><p>  gendo(lod,lev-table[i].level,table[i].adr);//先取值到棧頂</p>&l

79、t;p>  gendo(lit,lev-table[i].level,1);//將值為1入棧</p><p>  gendo(opr,lev-table[i].level,2);//加法,即加1</p><p><b>  }</b></p><p>  if(table[i].kind==array)</p><p&

80、gt;<b>  {</b></p><p>  gendo(gar,lev-table[i].level,table[i].adr);</p><p>  gendo(lit,lev-table[i].level,1);</p><p>  gendo(opr,lev-table[i].level,2);</p><p&g

81、t;<b>  }</b></p><p><b>  }</b></p><p>  else if(i!=0&&flag==5)//--</p><p><b>  {</b></p><p>  if(table[i].kind==variable)<

82、;/p><p><b>  {</b></p><p>  gendo(lod,lev-table[i].level,table[i].adr);//先取值到棧頂</p><p>  gendo(lit,lev-table[i].level,1);//將值為1入棧</p><p>  gendo(opr,lev-table[

83、i].level,3);//減法,即-1</p><p><b>  }</b></p><p>  if(table[i].kind==array)</p><p><b>  {</b></p><p>  gendo(gar,lev-table[i].level,table[i].adr);&

84、lt;/p><p>  gendo(lit,lev-table[i].level,1);</p><p>  gendo(opr,lev-table[i].level,3);</p><p><b>  }</b></p><p><b>  }</b></p><p>  el

85、se if(i!=0&&flag==6)//*=</p><p><b>  {</b></p><p>  memcpy(nxtlev,fsys,sizeof(bool)* symnum);</p><p>  expressiondo(nxtlev,ptx,lev);</p><p>  gendo(

86、lod,lev-table[i].level,table[i].adr);//先取值到棧頂</p><p>  gendo(opr,lev-table[i].level,4);//乘法</p><p><b>  }</b></p><p>  else if(i!=0&&flag==7) // /=;</p>&

87、lt;p><b>  {</b></p><p>  gendo(lod,lev-table[i].level,table[i].adr);//先取值到棧頂</p><p>  memcpy(nxtlev,fsys,sizeof(bool)* symnum);</p><p>  expressiondo(nxtlev,ptx,lev);

88、</p><p>  gendo(opr,lev-table[i].level,5);//除法</p><p><b>  }</b></p><p>  if(table[i].kind==array) //數(shù)組,根據(jù)偏移地址存入數(shù)組</p><p><b>  {</b></p>

89、<p>  gendo(sar,lev-table[i].level,table[i].adr);</p><p>  gendo(del,0,0);//賦值語句需將存入數(shù)組后棧頂為偏移地址出棧</p><p><b>  }</b></p><p>  else //變量</p><p><b>

90、  {</b></p><p>  gendo(sto,lev-table[i].level,table[i].adr); //根據(jù)變量地址存入變量</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b>

91、;</p><p><b>  }</b></p><p><b>  }</b></p><p>  else if(sym==dplus) //++變量</p><p><b>  {</b></p><p><b>  getsymdo;

92、</b></p><p>  if(sym==ident)//后面跟的是變量</p><p><b>  {</b></p><p>  i=position(id,*ptx);</p><p><b>  if(i==0)</b></p><p><b&g

93、t;  {</b></p><p>  error(11);</p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><p>  if(table[i].kind!

94、=variable&&table[i].kind!=array) //++后沒跟變量,出錯</p><p><b>  {</b></p><p>  error(12);i=0;</p><p><b>  }</b></p><p>  else //++后跟變量,處理生成中間代

95、碼</p><p><b>  {</b></p><p>  if(table[i].kind==variable)</p><p><b>  {</b></p><p>  gendo(lod,lev-table[i].level,table[i].adr);//先取值到棧頂</p>

96、;<p>  gendo(lit,0,1);//將值為1入棧</p><p>  gendo(opr,0,2);//加法,即+1,棧頂加次棧頂</p><p>  gendo(sto,lev-table[i].level,table[i].adr);//出棧取值到內(nèi)存</p><p><b>  getsymdo;</b><

97、/p><p><b>  }</b></p><p>  else if(table[i].kind==array) //后跟數(shù)組</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p>  if(sy

98、m==lepa)</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p>  memcpy(nxtlev,fsys,sizeof(bool)* symnum);</p><p>  nxtlev[ripa]=true;</p>&

99、lt;p>  expressiondo(nxtlev,ptx,lev);</p><p>  gendo(tra,0,table[i].size); //生成將數(shù)組下標范圍入棧指令</p><p>  gendo(jud,0,0); //判斷下標合法性</p><p>  if(sym!=ripa)</p><p><b> 

100、 {</b></p><p>  error(26); //缺少]</p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><p>  gendo(gar,lev-

101、table[i].level,table[i].adr);</p><p>  gendo(lit,0,1);</p><p>  gendo(opr,0,2);</p><p>  gendo(sar,lev-table[i].level,table[i].adr);</p><p>  gendo(del,0,0);</p>

102、<p><b>  getsymdo;</b></p><p><b>  }</b></p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></

103、p><p>  error(27);//缺少[</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><

104、;p><b>  }</b></p><p><b>  }</b></p><p>  else if(sym==dminus)//--變量</p><p><b>  {</b></p><p><b>  getsymdo;</b></p

105、><p>  if(sym==ident) //后面跟的是變量</p><p><b>  {</b></p><p>  i=position(id,*ptx);</p><p><b>  if(i==0)</b></p><p><b>  {</b>

106、</p><p>  error(11);</p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><p>  if(table[i].kind!=variable&&

107、amp;table[i].kind!=array) //--后沒跟變量,出錯</p><p><b>  {</b></p><p>  error(12);i=0;</p><p><b>  }</b></p><p>  else //--后跟變量,處理生成中間代碼</p>&l

108、t;p><b>  {</b></p><p>  if(table[i].kind==variable)//后跟變量</p><p><b>  {</b></p><p>  gendo(lod,lev-table[i].level,table[i].adr);//先取值到棧頂</p><p&

109、gt;  gendo(lit,0,1); //將值為1入棧</p><p>  gendo(opr,0,3);//加法,即-1,棧頂減次棧頂</p><p>  gendo(sto,lev-table[i].level,table[i].adr);//出棧取值到內(nèi)存</p><p><b>  getsymdo;</b></p>

110、<p><b>  }</b></p><p>  else if(table[i].kind==array)//后跟數(shù)組變量</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p>  if(sym==lepa

111、)</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p>  memcpy(nxtlev,fsys,sizeof(bool)* symnum);</p><p>  nxtlev[ripa]=true;</p><p>

112、;  expressiondo(nxtlev,ptx,lev);</p><p>  gendo(tra,0,table[i].size);//生成將數(shù)組下標范圍入棧指令</p><p>  gendo(jud,0,0);//判斷下標合法性</p><p>  if(sym!=ripa)</p><p><b>  {</b&

113、gt;</p><p>  error(26);//缺少]</p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><p>  gendo(gar,lev-table[i].l

114、evel,table[i].adr);</p><p>  gendo(lit,0,1);</p><p>  gendo(opr,0,3);</p><p>  gendo(sar,lev-table[i].level,table[i].adr);</p><p>  gendo(del,0,0);</p><p>

115、<b>  getsymdo;</b></p><p><b>  }</b></p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><

116、;p>  error(27);//缺少[</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p><

117、b>  }</b></p><p><b>  }</b></p><p>  ...//此處省略部分未修改過的代碼</p><p><b>  }</b></p><p>  2.增加Pascal的FOR語句</p><p>  本次課程設(shè)計中要求擴充的是

118、Pascal的FOR語句:</p><p> ?、貴OR <變量>:=<表達式> TO <表達式> DO <語句></p><p> ?、贔OR <變量>:=<表達式> DOWNTO <表達式> DO <語句></p><p><b>  ,流程圖如下:<

119、/b></p><p>  1)在symbol增加字符FORSYM。</p><p>  strcpy(KWORD[10],"for"); WSYM[ 10]= forsym;</p><p>  strcpy(KWORD[22],"to"); WSYM[22]=tosym;</p><p&

120、gt;  strcpy(KWORD[ 7],"downto"); WSYM[ 7]=downtosym;</p><p>  2)對statement函數(shù)進行擴充</p><p>  int statement(bool* fsys,int * ptx,int lev)</p><p><b>  {</b></p&

121、gt;<p>  ...//此處省略部分未修改過的代碼</p><p>  else if(sym== forsym)</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p>  if(sym==ident) // ident為變

122、量的符號</p><p><b>  {</b></p><p>  i=position(id,*ptx);</p><p><b>  if(i==0)</b></p><p>  error(11);</p><p><b>  else</b>&

123、lt;/p><p><b>  {</b></p><p>  if(table[i].kind!=variable) //賦值語句中,賦值號左部標識符屬性應(yīng)是變量</p><p><b>  {</b></p><p>  error(12);</p><p><b>

124、;  i=0;</b></p><p><b>  }</b></p><p><b>  else</b></p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p>

125、;  if(sym!=becomes) </p><p>  error(13);</p><p><b>  else </b></p><p><b>  getsymdo;</b></p><p>  memcpy(nxtlev,fsys,sizeof(bool)*symnum); </

126、p><p>  if(nxtlev[tosym]=true){} // tosym</p><p>  else nxtlev[downtosym]=true; // downtosym</p><p>  expressiondo(nxtlev,ptx,lev); //處理賦值語句右部的表達式</p><p>  gendo(sto,lev-t

127、able[i].level,table[i].adr); //保存初始值</p><p><b>  getsymdo;</b></p><p>  cx1=cx; //保存循環(huán)開始點</p><p>  memcpy(nxtlev,fsys,sizeof(bool)*symnum);</p><p>  if(sym

128、==dosym)</p><p><b>  {</b></p><p><b>  getsymdo;</b></p><p>  statement(fsys,ptx,lev); //循環(huán)體處理</p><p>  gendo(lod,lev-table[i].level,table[i].ad

129、r); //將循環(huán)變量取出放在棧頂</p><p>  gendo(opr,0,2); //循環(huán)變量加長</p><p>  gendo(sto,lev-table[i].level,table[i].adr); //將棧頂?shù)闹荡嫒胙h(huán)變量</p><p>  gendo(jmp,0,cx1);//無條件跳轉(zhuǎn)到循環(huán)開始點</p><p>  

130、code[cx2].a=cx; //回填循環(huán)結(jié)束點的地址</p><p><b>  }</b></p><p><b>  else </b></p><p>  error(29); //for語句中少了do</p><p><b>  }</b></p>&

131、lt;p><b>  }</b></p><p><b>  }</b></p><p><b>  else</b></p><p>  error(19); //for后面為賦值語句,賦值語句左部是變量,缺少變量</p><p><b>  }</b&

132、gt;</p><p>  ...// 此處省略部分未修改過的代碼</p><p><b>  }</b></p><p><b>  3.一維數(shù)組</b></p><p>  1)設(shè)置一維數(shù)組的左右括號:</p><p>  ssym['[']=lepa;

133、//一維數(shù)組的左括號[</p><p>  ssym[']']=ripa; //一維數(shù)組的右括號]</p><p><b>  2)增加指令:</b></p><p>  strcpy(&(mnemonic[gar][0]),"gar"); </p><p>  strcpy(

134、&(mnemonic[sar][0]),"sar");</p><p>  strcpy(&(mnemonic[shd][0]),"shd");</p><p>  strcpy(&(mnemonic[del][0]),"del");</p><p>  strcpy(&(m

135、nemonic[jud][0]),"jud");</p><p>  strcpy(&(mnemonic[tra][0]),"tra");</p><p><b>  enum fct{</b></p><p>  lit, opr, lod, sto, cal, inte, jmp,

136、 jpc, gar, sar, shd, del, jud, tra,</p><p><b>  };</b></p><p>  由于指令有原來的8條拓展到了14條,故進行如下修改</p><p>  #define fctnum 14</p><p>  增加的指令意義如下:</p><p>

137、;  gar:根據(jù)棧頂?shù)钠频刂窂臄?shù)組中取值到新的棧頂</p><p>  sar:根據(jù)次棧頂?shù)钠频刂钒褩m數(shù)闹荡嫒霐?shù)組</p><p>  shd:將棧頂?shù)闹迪乱频酱螚m?,棧頂出棧,即次棧頂成為棧?lt;/p><p><b>  del:出棧頂</b></p><p>  jud:判斷數(shù)組下標合法性</p>

138、<p>  tra:將數(shù)組的下標范圍入棧,gendo(tra,0,數(shù)組下標最大值);</p><p>  3)增加標識符類型屬性:</p><p>  enum object{</p><p><b>  constant,</b></p><p><b>  variable,</b>

139、;</p><p><b>  procedur,</b></p><p>  array, //數(shù)組</p><p><b>  };</b></p><p>  4)在block函數(shù)中添加如下代碼:</p><p>  for(i=tx0+1;i<=tx;i+

140、+)</p><p><b>  {</b></p><p>  switch(table[i].kind)</p><p><b>  {</b></p><p>  case constant: /*常量名字*/</p><p>

141、  ...//此處省略部分未修改過的代碼</p><p>  case variable: /*變量名字*/</p><p>  ...//此處省略部分未修改過的代碼</p><p>  case procedur: /*過程名字*/</p><p>  ...//此處省略部分未修改過的代碼<

142、;/p><p>  case array://數(shù)組變量</p><p>  printf("%d var-array %s ",i,table[i].name);</p><p>  printf("lev=%d addr=%d size=%d\n",table[i].level,table[i].adr,table[i].siz

143、e);</p><p>  fprintf(fas,"%d var-array %s",i,table[i].name);</p><p>  fprintf(fas,"lev=%d addr=%d size=%d\n",table[i].level,table[i].adr,table[i].size);</p><p>&

144、lt;b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  5)在enter()函數(shù)添加如下代碼:</p><p><b>  switch(k)</b></p><p>

145、;<b>  {</b></p><p>  case constant: /*常量名字*/</p><p>  ...//此處省略部分未修改過的代碼</p><p>  case variable: /*變量名字*/</p><p>  

146、...//此處省略部分未修改過的代碼</p><p>  case procedur: /*過程名字*/</p><p>  ...//此處省略部分未修改過的代碼</p><p>  case array: /*數(shù)組名字*/</p><p>  table[(*pt

147、x)].level=lev;</p><p>  table[(*ptx)].adr=(*pdx)-arraysize;</p><p>  table[(*ptx)].size=arraysize;</p><p><b>  break;</b></p><p><b>  }</b></

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論