PL0編譯程序講解_第1頁
PL0編譯程序講解_第2頁
PL0編譯程序講解_第3頁
PL0編譯程序講解_第4頁
PL0編譯程序講解_第5頁
已閱讀5頁,還剩55頁未讀 繼續(xù)免費(fèi)閱讀

下載本文檔

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

文檔簡介

1、第二章第二章 PL/0PL/0編譯程序的實(shí)現(xiàn)編譯程序的實(shí)現(xiàn)本章以本章以PL/0PL/0編譯程序編譯程序?yàn)閷?shí)例為實(shí)例, , 使大家對編譯程序的實(shí)現(xiàn)建立起整體概念,對編譯程序的構(gòu)造得到一些感性認(rèn)識和初步了解。1 PL/0PL/0語言語言2 PL/0PL/0處理機(jī)處理機(jī)假想棧式機(jī)假想棧式機(jī)3 PL/0PL/0編譯程序編譯程序4 4 符號表的一般形式討論符號表的一般形式討論5 5 棧式存儲管理的再討論棧式存儲管理的再討論1 PL/0PL/0語言語言PL/0PL/0功能簡單、結(jié)構(gòu)清晰、可讀性強(qiáng),而又具備了一功能簡單、結(jié)構(gòu)清晰、可讀性強(qiáng),而又具備了一般高級語言的必備部分,因而其編譯程序能充分體般高級語言的

2、必備部分,因而其編譯程序能充分體現(xiàn)一個(gè)高級語言編譯程序的基本技術(shù)和步驟。現(xiàn)一個(gè)高級語言編譯程序的基本技術(shù)和步驟。zPL/0PL/0語言:語言:PASCALPASCAL語言的語言的子集,用于教學(xué)子集,用于教學(xué)zPL/0PL/0程序示例程序示例zPL/0PL/0的的語法描述圖語法描述圖zPL/0PL/0語言語言的的EBNFEBNF表示表示 PL/0PL/0語言是語言是PASCALPASCAL語言的語言的子集子集z 過程可過程可嵌套定義,內(nèi)層嵌套定義,內(nèi)層可引用包圍它的外層定義的可引用包圍它的外層定義的標(biāo)識符標(biāo)識符, ,可遞歸調(diào)用可遞歸調(diào)用z 數(shù)據(jù)類型數(shù)據(jù)類型, ,只有整型只有整型z 數(shù)據(jù)結(jié)構(gòu)數(shù)據(jù)結(jié)

3、構(gòu) , ,只有簡變和常數(shù)只有簡變和常數(shù)z 標(biāo)識符的有效長度是標(biāo)識符的有效長度是1010z 語句種類:語句種類: begin/endbegin/end、ifif、whilewhile、賦值、賦值、read/writeread/write、callcall、constconst、varvar、procedureprocedurez 過程無參,最多可過程無參,最多可嵌套嵌套三層三層z 1313個(gè)保留字:個(gè)保留字:ifif、thenthen、whilewhile、dodo、readread、writewrite、callcall、beginbegin、endend、constconst、varvar、

4、procedureprocedure、oddoddz + +、- -、* *、/ /、= =、 、= 、=、( (、) ) PL/0PL/0程序示例程序示例 CONST A=10; CONST A=10; (* * 常量說明部分常量說明部分 * *) VAR B,C; VAR B,C; (* * 變量變量說明部分說明部分 * *) PROCEDURE PROCEDURE P; P; (* * 過程過程說明部分說明部分 * *) VAR D;VAR D;(* * P P的局部變量的局部變量說明部分說明部分 * *) PROCEDURE PROCEDURE Q; Q; (* * P P的局部過程的

5、局部過程說明部分說明部分 * *) VAR X;VAR X; BEGINBEGIN READ(X); READ(X); D:=X; D:=X; IF X#0 DO CALL P; IF X#0 DO CALL P; END; END; BEGINBEGIN CALL Q; WRITE(D); CALL Q; WRITE(D); END; END; BEGIN CALL P; END.BEGIN CALL P; END.遞歸計(jì)算 sum = 1! + 2 ! + . + n! var n, m, fact, sum; 遞規(guī)計(jì)算 fact = m! procedure factorial;begi

6、n if m 0 then begin fact := fact * m; m := m - 1; call factorial; end;end;begin 讀入n read(n); sum := 0; while n 0 do begin m := n; fact := 1; call factorial; sum := sum + fact; n := n - 1; end; 輸出n ! write(sum);end.constidentnumbervaridentprocedureident分程序分程序語句語句分程序分程序程序程序分程序分程序.語法圖語法圖identreadend語句語

7、句表達(dá)式表達(dá)式:=begin語句語句語句語句)(ident,PL/0PL/0語言的語言的EBNFEBNF表示表示z BNFBNF(BACKUS-NAUR FORMBACKUS-NAUR FORM)與與EBNFEBNF的介紹的介紹BNFBNF是根據(jù)美國的是根據(jù)美國的John W.BackusJohn W.Backus與丹麥的與丹麥的Peter NaurPeter Naur來命名的,它是從來命名的,它是從語法上描述程序設(shè)計(jì)語言的語法上描述程序設(shè)計(jì)語言的元語言元語言。采用。采用BNFBNF就可說明哪些符號序列是對就可說明哪些符號序列是對于某給定語言于某給定語言在語法上在語法上有效的程序。有效的程序。

8、z 構(gòu)成構(gòu)成EBNFEBNF的元素:非終結(jié)符,終結(jié)符,開始符,規(guī)則的元素:非終結(jié)符,終結(jié)符,開始符,規(guī)則z EBNFEBNF的元符號:的元符號: 用左右尖括號括起來的內(nèi)容為用左右尖括號括起來的內(nèi)容為非終結(jié)符非終結(jié)符=或或 讀做讀做定義為定義為,的的左部由左部由右部右部定義定義 | | 讀做讀做或或 表示右部候選內(nèi)容表示右部候選內(nèi)容 表示花括號內(nèi)的內(nèi)容表示花括號內(nèi)的內(nèi)容可重復(fù)可重復(fù)任意次或限定次數(shù)任意次或限定次數(shù) 表示方括號內(nèi)的內(nèi)容為表示方括號內(nèi)的內(nèi)容為任選項(xiàng)任選項(xiàng) ( ) ( ) 表示圓括號內(nèi)的內(nèi)容表示圓括號內(nèi)的內(nèi)容優(yōu)先優(yōu)先PL/0PL/0語言文法的語言文法的EBNFEBNF表示表示程序-分程

9、序.分程序-常量說明部分-CONST常量定義部分,常量定義;無符號整數(shù)-數(shù)字?jǐn)?shù)字變量說明部分-VAR標(biāo)識符,標(biāo)識符;標(biāo)識符-字母字母|數(shù)字 B-C-B -先調(diào)用,后結(jié)束B區(qū)A區(qū)B區(qū)C區(qū)BT 二、運(yùn)行時(shí)數(shù)據(jù)的存儲與訪問-棧式存儲假設(shè)A、C同層,且A中嵌套B子程序的調(diào)用、執(zhí)行和返回l 過程被調(diào)用時(shí),子程序的每次調(diào)用都需在數(shù)據(jù)棧頂為其分配獨(dú)立的數(shù)據(jù)區(qū)l 子程序返回時(shí),需做兩件事情:一是代碼返回(需記住RA),二是數(shù)據(jù)區(qū)的同步恢復(fù)(DL)l 子程序運(yùn)行時(shí),要存取外層數(shù)據(jù)區(qū)中的存儲單元當(dāng)前B數(shù)據(jù)區(qū)須記住:返回地址RA動(dòng)態(tài)鏈DL記錄調(diào)用者數(shù)據(jù)區(qū)基地址 靜態(tài)鏈SL記錄定義該過程的直接外層過程數(shù)據(jù)區(qū)的基地址,

10、以便訪問外層數(shù)據(jù)運(yùn)行時(shí)數(shù)據(jù)棧運(yùn)行時(shí)數(shù)據(jù)棧S S的變化的變化 var m1,m2,m3;var m1,m2,m3; Procedure A; Procedure A; var a1; var a1; procedure B; procedure B; var b1,b2; var b1,b2; procedure C; procedure C; C C過程過程call B;call B; r1r1: : B B過程過程call C; call C; r2:r2: A A過程過程call B; call B; r3:r3: 主程序主程序Call A; Call A; r4:r4: B的臨時(shí)單元 b

11、2=120 b1=50 RA: r1 DL:115 SL:106 RA: r2 DL:110 SL:110 b2=25 b1=20 RA: r3 DL:106 SL:106 a1=15 RA: r4 DL:100 SL:100 m3=118 m2=472 m1=335 RA:0 DL:0 T B SL:0 100 三、PL/0機(jī)的指令系統(tǒng) f: 功能碼l: 層次差 (標(biāo)識符引用層減去定義層)a:根據(jù)不同的指令有所區(qū)別f l af l a指令格式:指令格式:所有運(yùn)算對棧頂?shù)膬蓚€(gè)或一個(gè)元素進(jìn)行,并用運(yùn)算結(jié)果代替原來的運(yùn)算對象。指指令令功功能能表表( 0) jmp 0 8 轉(zhuǎn)向轉(zhuǎn)向主程序入口主程序入

12、口( 1) jmp 0 2 轉(zhuǎn)向轉(zhuǎn)向過程過程p入口入口( 2) int 0 3 為過程為過程p開辟空間開辟空間( 3) lod 1 3( 4) lit 0 10( 5) opr 0 2( 6) sto 1 4( 7) opr 0 0 退棧并返回調(diào)用點(diǎn)退棧并返回調(diào)用點(diǎn)( 8) int 0 5( 9) opr 0 16 (10) sto 0 3(11) lod 0 3(12) lit 0 0(13) opr 0 9(14) jpc 0 24 條件不滿足轉(zhuǎn)條件不滿足轉(zhuǎn)24(15) cal 0 2 (16) lit 0 2(17) lod 0 4(18) opr 0 4(19) opr 0 14(20

13、) opr 0 15 換行換行(21) opr 0 16(22) sto 0 3(23) jmp 0 11(24) opr 0 0 SL 0DL 0RA 0變量變量b變量變量cRA 16SL 0DL 0運(yùn)行棧運(yùn)行棧c o n s t a = 1 0 ;v a r b , c ;p r o c e d u r e p ; begin c : = b + a ; end;begin r e a d ( b ) ; while b#0 do begin c a l l p ; w r i t e ( 2 * c ) ; r e a d ( b ) ; endend.SL:靜態(tài)鏈:靜態(tài)鏈DL:動(dòng)態(tài)鏈:

14、動(dòng)態(tài)鏈RA:返回地址:返回地址0演示執(zhí)行過程3 PL/0PL/0編譯程序的實(shí)現(xiàn)編譯程序的實(shí)現(xiàn)zPL/0PL/0編譯程序的總體設(shè)計(jì)編譯程序的總體設(shè)計(jì)zPL/0PL/0編譯程序詞法分析的設(shè)計(jì)與實(shí)現(xiàn)編譯程序詞法分析的設(shè)計(jì)與實(shí)現(xiàn)zPL/0PL/0編譯程序語法分析的設(shè)計(jì)與實(shí)現(xiàn)編譯程序語法分析的設(shè)計(jì)與實(shí)現(xiàn)zPL/0PL/0編譯程序語義分析的設(shè)計(jì)與實(shí)現(xiàn)編譯程序語義分析的設(shè)計(jì)與實(shí)現(xiàn)zPL/0PL/0編譯程序語法錯(cuò)誤處理的實(shí)現(xiàn)編譯程序語法錯(cuò)誤處理的實(shí)現(xiàn)zPL/0PL/0編譯程序代碼生成的實(shí)現(xiàn)編譯程序代碼生成的實(shí)現(xiàn)zpcodepcode代碼解釋器的設(shè)計(jì)與實(shí)現(xiàn)代碼解釋器的設(shè)計(jì)與實(shí)現(xiàn)3.1 PL/03.1 PL/0編

15、譯程序的總體設(shè)計(jì)編譯程序的總體設(shè)計(jì)z 單趟方式z 以語法、語義分析程序?yàn)楹诵?,詞法分析程序和代碼生成程序都作為一個(gè)過程,當(dāng)語法分析需要讀單詞時(shí)就調(diào)用詞法分析程序,而當(dāng)語法、語義分析正確,需要生成相應(yīng)的目標(biāo)代碼時(shí),則調(diào)用代碼生成程序。z 表格管理程序?qū)崿F(xiàn)變量,常量和過程標(biāo)識符的信息的登錄與查找。z 出錯(cuò)處理程序,對詞法和語法、語義分析遇到的錯(cuò)誤給出在源程序中出錯(cuò)的位置和與錯(cuò)誤 性質(zhì)有關(guān)的編號,并進(jìn)行錯(cuò)誤恢復(fù)。PL/0PL/0編譯程序編譯程序PL/0PL/0編譯程序編譯程序類類 p pcodecode解釋解釋程序程序類類 pcode代碼代碼PL/0源程序源程序輸入數(shù)據(jù)輸入數(shù)據(jù)輸出數(shù)據(jù)輸出數(shù)據(jù)PL/

16、0編譯程序的結(jié)構(gòu)框架 PL/0PL/0編譯程序的結(jié)構(gòu)編譯程序的結(jié)構(gòu)詞法分析程詞法分析程序序語法語義分析程序語法語義分析程序代碼生成程序代碼生成程序表格管理程序表格管理程序出錯(cuò)處理程序出錯(cuò)處理程序PL/0PL/0源程序源程序目標(biāo)程序目標(biāo)程序編譯系統(tǒng)總體流程圖編譯系統(tǒng)總體流程圖PL/0編譯程序語法、語義分析的核心3.2 PL/0編譯程序詞法分析的實(shí)現(xiàn)詞法分析函數(shù)詞法分析函數(shù)getsym()getsym()所識別的單詞:所識別的單詞:y保留字或關(guān)鍵字:如:保留字或關(guān)鍵字:如:BEGINBEGIN、 ENDEND、 IFIF、 THENTHEN等等y運(yùn)算符運(yùn)算符: 如:如:+ +、- -、* *、/

17、/、:、:= =、# #、=、=等等y標(biāo)識符標(biāo)識符: 用戶定義的變量名、常數(shù)名、過程名用戶定義的變量名、常數(shù)名、過程名y常數(shù)常數(shù): 如:如:1010、2525、100100等整數(shù)等整數(shù)y界符界符: 如:如:,、. . 、; ; 、( ( 、)等等詞法分析過程詞法分析過程:getsym():getsym()框圖(框圖(P19P19圖圖2.52.5)在編譯程序中,單詞的表示方式在編譯程序中,單詞的表示方式:(:(sym, id/numsym, id/num)z enum symbol enum symbol nulnul, ,identident, ,numbernumber, ,plusplus

18、,varsymvarsym, ,procsymprocsym;z 當(dāng)識別出標(biāo)識符時(shí)先查當(dāng)識別出標(biāo)識符時(shí)先查保留字保留字表表z 保留字及內(nèi)部表示對應(yīng)表保留字及內(nèi)部表示對應(yīng)表: char wordnorwal; char wordnorwal; enum symble wsymnorw; enum symble wsymnorw;z 字符對應(yīng)的字符對應(yīng)的單詞表:單詞表: enum symble ssym256;enum symble ssym256; ssym+= ssym+=plusplus; ssym-=; ssym-=minusminus; ; z 詞法分析通過三個(gè)詞法分析通過三個(gè)全程量全程

19、量 symbol symbol symsym; char id; int ; char id; int numnum; ;將識別出的單詞信息將識別出的單詞信息傳遞傳遞給給語法分析語法分析程序。程序。ysymsym:存放單詞的類別:存放單詞的類別 如:如:initial:= 60initial:= 60;中各單詞對應(yīng)的類別為:;中各單詞對應(yīng)的類別為: initial initial identident, := , := becomesbecomes, 60 , 60 numbernumber, ; , ; semicolonsemicolonyidid:存放用戶標(biāo)識符:存放用戶標(biāo)識符, ,對對

20、initialinitial(sym-sym-identident,id-id-initialinitial)ynumnum:存放用戶定義的數(shù):存放用戶定義的數(shù) 對對6060(sym-sym-number,number,num-num-6060)用狀態(tài)轉(zhuǎn)換圖實(shí)現(xiàn)詞法分析程序的設(shè)計(jì)方法用狀態(tài)轉(zhuǎn)換圖實(shí)現(xiàn)詞法分析程序的設(shè)計(jì)方法狀態(tài)狀態(tài),對應(yīng)每個(gè)狀態(tài)編一段程序,對應(yīng)每個(gè)狀態(tài)編一段程序,每個(gè)狀態(tài)每個(gè)狀態(tài)調(diào)用調(diào)用取字符取字符程程序,根據(jù)當(dāng)前字符序,根據(jù)當(dāng)前字符轉(zhuǎn)到不同的狀態(tài),并做相應(yīng)操作。轉(zhuǎn)到不同的狀態(tài),并做相應(yīng)操作。表示表示終態(tài)終態(tài),已,已識識別出一個(gè)別出一個(gè)單詞單詞1 12 23 35 5141413

2197 78 86 64 41111空格空格字母字母字母數(shù)字字母數(shù)字非字母數(shù)字非字母數(shù)字?jǐn)?shù)字?jǐn)?shù)字?jǐn)?shù)字?jǐn)?shù)字非數(shù)字非數(shù)字:= = = =非非= =, + - ( 3.3 PL/03.3 PL/0編譯程序語法分析編譯程序語法分析z語法分析的設(shè)計(jì)與實(shí)現(xiàn)語法分析的設(shè)計(jì)與實(shí)現(xiàn)y 自頂向下的語法分析自頂向下的語法分析y 遞歸子程序法遞歸子程序法(遞歸下降分析器(遞歸下降分析器recursive-descent parser):對應(yīng)對應(yīng)每個(gè)非終結(jié)符每個(gè)非終結(jié)符(語法成分),(語法成分),編一個(gè)獨(dú)立的處理子程序。編一個(gè)獨(dú)立的處理子程序。 由由 開始,按規(guī)則右部開始,按規(guī)則右部(語法描述圖語

22、法描述圖箭頭箭頭方向方向)進(jìn)行分析進(jìn)行分析 遇到遇到非終結(jié)符非終結(jié)符,則,則調(diào)用調(diào)用相應(yīng)的相應(yīng)的處理過程處理過程 遇到遇到終結(jié)符終結(jié)符,則判斷當(dāng)前讀入的單詞是否與該終結(jié)符,則判斷當(dāng)前讀入的單詞是否與該終結(jié)符相匹相匹配配,若匹配,再讀取下一個(gè)單詞繼續(xù)分析。,若匹配,再讀取下一個(gè)單詞繼續(xù)分析。 程序程序 pl/0分程序分程序 block語句語句statement條件條件condition表達(dá)式表達(dá)式expression 項(xiàng)項(xiàng)term因子因子 factor語語法法調(diào)調(diào)用用關(guān)關(guān)系系圖圖VAR A;VAR A;BEGINBEGIN READ(A) READ(A)END.END. . . VARVAR ;

23、 A A BEGINBEGIN ENDEND READREAD ( ) A A 為文法的為文法的開始符號開始符號,以開,以開始符號作為根結(jié)始符號作為根結(jié)點(diǎn)存在一棵倒掛點(diǎn)存在一棵倒掛著的語法樹。著的語法樹。遞歸下降語法分遞歸下降語法分析過程隱含著對析過程隱含著對對語法樹的前序?qū)φZ法樹的前序遍歷遍歷表達(dá)式表達(dá)式=+|-+|-項(xiàng)項(xiàng) (+|-+|-)項(xiàng))項(xiàng) int expression(bool* fsys, int* ptx, int lev)if(sym=plus | sym=minus) getsymdo; termdo(nxtlev, ptx, lev); /處理項(xiàng) elsetermdo(nx

24、tlev, ptx, lev); / 處理項(xiàng) while (sym=plus | sym=minus)getsymdo; termdo(nxtlev, ptx, lev); / 處理項(xiàng) return 0;注意一致性:進(jìn)入每一語法單位處理程序之前,其第一個(gè)單詞已讀出,退出時(shí),應(yīng)讀出下一個(gè)語法單位的第一個(gè)單詞項(xiàng)項(xiàng)=因子因子 (* *|/|/)因子)因子 int term(bool* fsys, int* ptx, int lev)factordo(nxtlev, ptx, lev); /* 處理因子 */ while(sym=times | sym=slash)getsymdo; factordo

25、(nxtlev, ptx, lev);return 0;因子因子=標(biāo)識符標(biāo)識符| |無符號整數(shù)無符號整數(shù)|(表達(dá)式表達(dá)式)int factor(bool* fsys, int* ptx, int lev)if(sym = ident)/* 因子為常量或變量 */ getsymdo; else if(sym = number) getsymdo; else if (sym = lparen)/* 因子為表達(dá)式 */ getsymdo; expressiondo(nxtlev, ptx, lev); if (sym = rparen) getsymdo; else error(22);/* 缺少右

26、括號 */ return 0;3.4 PL/03.4 PL/0語義分析的設(shè)計(jì)與實(shí)現(xiàn)語義分析的設(shè)計(jì)與實(shí)現(xiàn)z 說明部分的分析說明部分的分析與處理與處理z 表格管理表格管理z 過程體過程體( (語句)的分析語句)的分析與處理與處理說明部分的分析說明部分的分析與處理與處理-登錄符號表登錄符號表z 對每個(gè)過程(含主程序)對每個(gè)過程(含主程序)說明的對象說明的對象(變量變量,常量常量和和過程過程)造符號表造符號表z 登錄登錄標(biāo)識符的標(biāo)識符的屬性屬性。z 標(biāo)識符的屬性標(biāo)識符的屬性: :種類,所在種類,所在層次層次, ,值值和分配的和分配的相對位置相對位置。z 登錄信息由登錄信息由ENTERENTER過程完成

27、。過程完成。符號表結(jié)構(gòu)enum object constant, variable, procedur;struct tablestruct char nameal; enum object kind; int val, level, adr, size; tabletxmax; const a=35;/const a=35;/常量無層次常量無層次var a1,a2,a3;var a1,a2,a3;Procedure P;Procedure P; var b1,b2; var b1,b2; procedure P1; procedure P1; var c; var c; procedure

28、P2; procedure P2; var d; var d; 注意:在單趟編譯中,對于并列的函數(shù)(或分程序),其相注意:在單趟編譯中,對于并列的函數(shù)(或分程序),其相應(yīng)的符號表不會同時(shí)存在。應(yīng)的符號表不會同時(shí)存在。過程過程P2在在code的的入口入口 (0) jmp 0 0 CX (1) jmp 0 0 (2) jmp 0 0 (k) jmp 0 0 名字名字 類類 值值 層次層次 地址地址 大小大小: :=varvar, ;if(sym=varsym)/if(sym=varsym)/* * 收到變量聲明符號,開始處理變量聲明收到變量聲明符號,開始處理變量聲明 * */ / getsymdo

29、; getsymdo; dovardeclarationdo(&tx, lev, &dx); dovardeclarationdo(&tx, lev, &dx); while (sym = comma) while (sym = comma) getsymdo; getsymdo; vardeclarationdo(&tx, lev, &dx); vardeclarationdo(&tx, lev, &dx); if (sym = semicolon) if (sym = semicolon) getsymdo;getsymdo;

30、 else error(5);else error(5); while (sym = ident); while (sym = ident); 注意:注意:&tx&tx變量說明處理變量說明處理int vardeclaration(intint vardeclaration(int* * ptx,int lev,int ptx,int lev,int* * pdx) pdx)if (sym = ident)if (sym = ident) enter(variable, ptx, lev, pdx);/enter(variable, ptx, lev, pdx);/填寫名字表填寫

31、名字表 getsymdo; getsymdo; else else error(4); error(4); / /* * var var后應(yīng)是標(biāo)識后應(yīng)是標(biāo)識符符 * */ / return 0; return 0; 過程過程ENTERENTER的實(shí)現(xiàn)的實(shí)現(xiàn)/ /* * 在名字表中加入一項(xiàng)在名字表中加入一項(xiàng) * * * * k: k: 名字種類名字種類const,var or procedureconst,var or procedure * * ptx: ptx: 名字表尾指針名字表尾指針 * * lev: lev: 名字所在的層次名字所在的層次, ,以后所有的以后所有的levlev都是這樣都

32、是這樣 * * pdx: pdx: 當(dāng)前應(yīng)分配變量的相對地址,分配后增加當(dāng)前應(yīng)分配變量的相對地址,分配后增加1 1 * */ /void enter(enum object k, intvoid enter(enum object k, int* * ptx,int lev, int ptx,int lev, int* * pdx) pdx)(* *ptx)+;ptx)+; strcpy(table( strcpy(table(* *ptx).name, id); ptx).name, id); / /* * 全局變量全局變量idid中已存有當(dāng)前名字的名字中已存有當(dāng)前名字的名字 * */ /

33、table( table(* *ptx).kind = k;ptx).kind = k; switch (k)switch (k) case constant:case constant: / /* * 常量名字常量名字 * */ / if (num amax) if (num amax) error(31); error(31);/ /* * 數(shù)越界數(shù)越界 * */ / num = 0; num = 0; table( table(* *ptx).val = num;ptx).val = num; break; break; case variable:case variable: / /*

34、 * 變量名字變量名字 * */ / table( table(* *ptx).level = lev;ptx).level = lev; table( table(* *ptx).adr = (ptx).adr = (* *pdx);pdx); ( (* *pdx)+;pdx)+; break; break; case procedur: case procedur: / /* *過程名字過程名字* */ / table( table(* *ptx).level = lev;ptx).level = lev; break; break; 過程體的處理變量引用的處理過程體的處理變量引用的處理y

35、對對語句進(jìn)行語句進(jìn)行語法語法分析分析y語義分析語義分析 當(dāng)遇到當(dāng)遇到標(biāo)識符的引用時(shí)標(biāo)識符的引用時(shí)就調(diào)用就調(diào)用POSITIONPOSITION函數(shù)函數(shù)查查TABLETABLE表表,看是否,看是否有有過過正確定義正確定義,若已有,則從表中,若已有,則從表中取相應(yīng)取相應(yīng)的有關(guān)的有關(guān)信息信息,供代碼的生成使用。,供代碼的生成使用。若無定義則錯(cuò)若無定義則錯(cuò)。y語義分析語義分析 TABLETABLE表表若已若已有有過過正確定義正確定義,檢查引用與說明的,檢查引用與說明的屬性是否一致,屬性是否一致,若不一致則錯(cuò)若不一致則錯(cuò)。y當(dāng)當(dāng)語法語義正確時(shí)語法語義正確時(shí),就,就生成生成相應(yīng)語句功能的相應(yīng)語句功能的目標(biāo)代

36、碼目標(biāo)代碼intint position( position(charchar* * id) id) intint i; i; strcpy(, id);strcpy(, id); i = tx + 1;i = tx + 1; whilewhile(strcmp(, id) != 0);(strcmp(, id) != 0); returnreturn i; i; / position/ position思考:在造表和查表過程中,如何保證每個(gè)過程的局部量不被它的外層引用?賦值賦值語句的處理語句的處理if

37、 (sym = ident) if (sym = ident) / /* * 準(zhǔn)備按照賦值語句處理準(zhǔn)備按照賦值語句處理 * */ / i = position(id, i = position(id, * *ptx);ptx); if (i = 0) error(11); if (i = 0) error(11);/ /* * 變量未找到變量未找到 * */ / elseif(tablei.kind != variable) elseif(tablei.kind != variable) error(12);error(12); / /* * 賦值語句格式錯(cuò)誤賦值語句格式錯(cuò)誤 * */ / i

38、 = 0; i = 0; else.else. gendo(sto,lev-tablei.level,tablei.adr); gendo(sto,lev-tablei.level,tablei.adr); . . 因子因子=標(biāo)識符標(biāo)識符| |無符號整數(shù)無符號整數(shù)|(表達(dá)式表達(dá)式)int factor(bool* fsys, int* ptx, int lev) /因子的語義分析if(sym = ident)/* 因子為常量或變量 */i = position(id, *ptx);/* 查找名字 */ if (i = 0) error(11); /* 標(biāo)識符未聲明 */ else switch

39、 (tablei.kind) case constant:/* 名字為常量 */ break; case variable: /* 名字為變量 */ break; case procedur:/* 名字為過程 */ error(21); /* 不能為過程名 */3.5 編譯程序的錯(cuò)誤處理錯(cuò)誤處理的原則:盡可能準(zhǔn)確指出出錯(cuò)位置,錯(cuò)誤性質(zhì),錯(cuò)誤處理的原則:盡可能準(zhǔn)確指出出錯(cuò)位置,錯(cuò)誤性質(zhì),盡可能進(jìn)行校正。盡可能進(jìn)行校正。PL/0PL/0編譯程序?qū)φZ法錯(cuò)誤的處理:編譯程序?qū)φZ法錯(cuò)誤的處理:(1)(1)對于對于易于校正易于校正的錯(cuò)誤,如丟了逗號,分號等,指出出的錯(cuò)誤,如丟了逗號,分號等,指出出錯(cuò)位置,

40、錯(cuò)位置,加以校正加以校正,繼續(xù)進(jìn)行分析。,繼續(xù)進(jìn)行分析。(2)(2)對于對于難于校正難于校正的錯(cuò)誤,給出錯(cuò)誤的位置與性質(zhì),的錯(cuò)誤,給出錯(cuò)誤的位置與性質(zhì),跳過跳過后面的一些單詞后面的一些單詞,直到下一個(gè)可以進(jìn)行正常語法分析的直到下一個(gè)可以進(jìn)行正常語法分析的語法單位語法單位 z 在在進(jìn)入進(jìn)入某個(gè)某個(gè)語法單位語法單位時(shí),時(shí),調(diào)用調(diào)用TESTTEST, ,檢查當(dāng)前符號檢查當(dāng)前符號是否屬于該是否屬于該語法單位的開語法單位的開始符號集合始符號集合。若不屬于,。若不屬于,則則濾去濾去開始開始符號符號和和后跟后跟符符號號集合外集合外的所有符號。的所有符號。z 在在語法單位語法單位分析結(jié)束分析結(jié)束時(shí),時(shí),調(diào)用調(diào)

41、用TESTTEST, ,檢查當(dāng)前符號檢查當(dāng)前符號是否屬于調(diào)用該語法單位是否屬于調(diào)用該語法單位時(shí)應(yīng)有的時(shí)應(yīng)有的后跟后跟符號集合。符號集合。若不屬于,則若不屬于,則濾去濾去后跟后跟符符號和號和開始開始符號符號集合外集合外的所的所有符號。有符號。 TEST TEST TEST TESTz 開始符號集合 bool declbegsyssymnum,statbegsyssymnum,facbegsyssymnum; declbegsys: constsym,varsym,procsym; statbegsys: beginsym,callsym,ifsym,whilesym,readsym,writes

42、ym; facbegsys: ident,number,lparen;z 后跟符號集合fsys作為參數(shù): int test(bool* s1, bool* s2, int n); int block(int lev, int tx, bool *fsys); int statement(bool* fsys, int * ptx, int lev); int expression (bool* fsys, int * ptx, int lev); int term (bool* fsys, int * ptx, int lev); int factor (bool* fsys, int * p

43、tx, int lev);為symble的元素?cái)?shù)32后跟符號集TESTSYM在在S1中中?打印出錯(cuò)編號打印出錯(cuò)編號nS1:=S1+S2SYM在在S1中中?GETSYM返回返回YYNNTEST測試過程流程圖測試過程流程圖因子的處理過程因子的處理過程procedure factor(fsys:symset); var i:integer; begin 入口: test(facbegsys,fsys,24); while sym in facbegsys do begin if . 出口: test(fsys,facbegsys,23); end end;Facbegsysy 處理處理ident n

44、umber lparentestntest后跟符集是逐步補(bǔ)充的,需補(bǔ)充的內(nèi)容與當(dāng)前所處小環(huán)境有關(guān)。對調(diào)用expression(fsys); z 由于write語句的語法write(,); 所以在write語句中調(diào)用expression時(shí)后跟符為: expression(rparen,comma+fsys);zfactor的語法:factor=.|( exp ) 在factor中調(diào)用expression時(shí)后跟符 expression(rparen+fsys);3.6 代碼生成代碼生成是由過程代碼生成是由過程GENGEN完成。完成。GENGEN有有3 3個(gè)個(gè)參數(shù)參數(shù),分別代表目標(biāo)代碼的,分別代表目

45、標(biāo)代碼的功能碼功能碼,層差層差和和位移量位移量。例如。例如 gen(opr,0,16); gen(opr,0,16); gen(sto, gen(sto,levlev- -levellevel,adr),adr) levlev:當(dāng)前:當(dāng)前處理的處理的過程過程層次層次 levellevel:被引用變量或過程所在:被引用變量或過程所在層次層次 CXCX:為目標(biāo)代碼:為目標(biāo)代碼codecode數(shù)組的下標(biāo)指針數(shù)組的下標(biāo)指針代碼結(jié)構(gòu)變換, 地址回填getsym;condition;if sym=thensym then getsym else error(16);cx1:= cx;gen(jpc,0,0

46、)statement( );codecx1.a:=cxIf c then sint main() . if(-1=block(0, 0, nxtlev) . . interpret(); / 調(diào)用解釋執(zhí)行程序 .int block(int lev, int tx, bool* fsys) dx=3; tx0=tx; tabletx.adr=cx; gen(jmp,0,0); /生成轉(zhuǎn)向過程體入口的指令 while (sym = procsym) getsymdo(); if(sym=ident) enter(procedur, &tx, lev, &dx); . if (-1

47、= block(lev+1, tx, nxtlev) . .3.7 PL/0分程序的處理子程序分程序的處理子程序-block初值為fsysperiod+declbegsys+statbegsys下標(biāo)指針下標(biāo)指針cx,tx和變量和變量dx的作用的作用CX:為目標(biāo)代碼code數(shù)組的下標(biāo)指針。實(shí)際上目標(biāo)代碼的順序是內(nèi)層過程的在前邊,主程序的目標(biāo)代碼在最后。tx :table表的下標(biāo)指針,是以值參數(shù)形式使用的。dx: 計(jì)算每個(gè)變量在運(yùn)行棧中相對本過程基地址的偏移量 ,放在table表中的adr域,生成目標(biāo)代碼時(shí)再放在code中的a域。過程體入口時(shí)的處理codetabletx0.adr.a=cx; /過

48、程入口地址填寫在code中tabletx0.adr=cx; /過程的入口填寫在table中tabletx0.size=dx; /過程占的空間填寫在table中 cx0=cx; /保留過程在code中的入口地址gen(int,0,dx); /生成過程入口指令3.8 類pcode解釋器z目標(biāo)代碼存放在數(shù)組CODE中(程序地址寄存器p)z解釋程序定義一個(gè)一維整型數(shù)組S作為運(yùn)行棧z棧頂寄存器(指針)t,基址寄存器(指針)b,指令寄存器i(當(dāng)前正在解釋的目標(biāo)指令)目標(biāo)代碼的解釋執(zhí)行目標(biāo)代碼的解釋執(zhí)行幾條幾條特殊指令特殊指令在在code中的中的位置位置和和功能功能yINT 0 AINT 0 A在在過程過程

49、目標(biāo)程序的目標(biāo)程序的入口處入口處,開辟開辟A A個(gè)單元的數(shù)據(jù)段。個(gè)單元的數(shù)據(jù)段。A A為為局局部變量部變量的的個(gè)數(shù)個(gè)數(shù)+ +3 3。yOPR 0 0OPR 0 0在在過程過程目標(biāo)程序的目標(biāo)程序的出口處出口處,釋放數(shù)據(jù)段釋放數(shù)據(jù)段(退棧),(退棧),恢復(fù)調(diào)恢復(fù)調(diào)用用該過程該過程前前正在運(yùn)行的過程正在運(yùn)行的過程的數(shù)據(jù)段的數(shù)據(jù)段基址寄存器基址寄存器B B和和棧頂棧頂寄存器寄存器T T的值,并將的值,并將返回地址返回地址送送到指令地址寄存器到指令地址寄存器P P中。中。yCAL L ACAL L A調(diào)用過程調(diào)用過程,填寫填寫靜態(tài)鏈靜態(tài)鏈、動(dòng)態(tài)鏈動(dòng)態(tài)鏈、返回地址返回地址,給出,給出被調(diào)被調(diào)用用過程過程的

50、的基地址基地址值,值,送送入基址寄存器入基址寄存器B B中,目標(biāo)程序的中,目標(biāo)程序的入口入口地址地址A A的值的值送送指令地址寄存器指令地址寄存器P P中,使指令從中,使指令從A A開始執(zhí)行開始執(zhí)行interpret三個(gè)寄存器賦初值三個(gè)寄存器賦初值t:=0; b:=1; p:=0;主程序的主程序的SL,DL,RA賦初值賦初值s1:=0; s2=0; s3=0;i:=codep;p:=p+1;P=0?返回返回解釋執(zhí)行的流程圖解釋執(zhí)行的流程圖執(zhí)行指令執(zhí)行指令iNY幾條特殊指令的解釋執(zhí)行過程過程出口出口opr 0 0opr 0 0 RA RA DL DL SL SLb. tMtbt=b-1;p=st+3;b=st+2Q調(diào)用過程調(diào)用過程-cal l acal l a st+1=base(l, s, b); 填寫靜態(tài)鏈 st+2=b; 填寫動(dòng)態(tài)鏈 st+3=p; 填寫返回地址 b=t+1; 被調(diào)用過程的基地址 p=a 過程入口地址a送p /t在int中設(shè)置i

溫馨提示

  • 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. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論