ack/lang/a68s/aem/perqce.p
1988-10-04 10:56:50 +00:00

1078 lines
42 KiB
OpenEdge ABL

~>|sed -e '/ *$/s/ *$/~~~~/' -e '/~~~~/s///' >a68s1ce.pp
00100 (*CODE EMITTER*)
00110 (**************)
00111 Things needing attention
00112 OCVIMMPTR and OCVIMMLONG (see PARAM and EMITOP)
00120 (**)
00130 (*+01() (*$T-+) ()+01*)
00140 (*+02() (*$T-+) ()+02*)
00150 (*-05()
00160 PROCEDURE LOAD (WHERE:SBTTYP; SB:PSB); FORWARD;
00170 PROCEDURE EMITEND; FORWARD;
00180 PROCEDURE EMITX1 (OPCOD:POP;TYP1:OPDTYP;OPND1:ADDRINT); FORWARD;
00190 PROCEDURE EMITX2 (OPCOD:POP;TYP1:OPDTYP;OPND1:ADDRINT;TYP2:OPDTYP;OPND2:ADDRINT); FORWARD;
00200 FUNCTION GENLCLGBL (VAR OPCOD:POP; SB:PSB):OFFSETR; FORWARD;
00210 PROCEDURE FIXUPF(ALABL:LABL);FORWARD;
00220 FUNCTION FIXUPM: LABL; FORWARD;
00230 PROCEDURE UNSTKP1(TYP:OPDTYP; OPND:PSB); FORWARD;
00240 ()-05*)
00250 PROCEDURE EMITOP (OPCOD:POP); FORWARD;
00260 PROCEDURE GENDENOT (OPCOD:POP; SB:PSB); FORWARD;
00270 FUNCTION GETNEXTLABEL: LABL;
00280 BEGIN GETNEXTLABEL := NEXTLABEL; NEXTLABEL := NEXTLABEL+1 END;
00290 (**)
00300 (**)
00310 (*+32()
00320 (*-01() PROCEDURE HALT; VAR I,K: INTEGER; BEGIN I:=0;K := K DIV I END; ()-01*)
00330 PROCEDURE ASSERT (ASSERTION:BOOLEAN; REASON:ALFA);
00340 BEGIN
00350 IF NOT (ASSERTION) THEN
00360 BEGIN
00370 WRITELN(OUTPUT,' ASSERT FAILED ',REASON);
00380 (*+01() PUTSEG(OUTPUT); ()+01*)
00390 EMITEND;
00400 HALT
00410 END
00420 END;
00430 (**)
00440 ()+32*)
00450 (* PERQ CODE EMITTER *)
00460 (*********************)
00470 (*+05()
00480 PROCEDURE PARAM(TYP:OPDTYP; OPND:INTEGER; OPCOD: POP; ALIGN: INTEGER; FIRSTIME: BOOLEAN); FORWARD;
00490 PROCEDURE EMITOPRAND(TYP:OPDTYP;OPERAND:ADDRINT);
00500 VAR REC: RECORD CASE SEVERAL OF
00510 1: (INT:ADDRINT);
00520 2: (LEX:PLEX);
00530 3,4,5,6,7,8,9,10: ()
00540 END;
00550 I:INTEGER;
00560 BEGIN
00570 CASE TYP OF
00580 OCVIMMED: WRITE(LGO[ROUTNL^.RNLEVEL],' ',OPERAND:1);
00590 OCVFREF,OCVMEM,OCVFIM:
00600 WRITE(LGO[ROUTNL^.RNLEVEL],' L',OPERAND:1);
00610 OCVEXT: BEGIN
00620 REC.INT := OPERAND;
00630 WRITE(LGO[ROUTNL^.RNLEVEL], '_');
00640 FOR I := 1 TO 7 DO
00650 (*IF REC.LEX^.S10[I]<>' ' THEN WRITE(LGO[ROUTNL^.RNLEVEL], CHR(ORD(REC.LEX^.S10[I])+32));*)
00660 WRITE(LGO[ROUTNL^.RNLEVEL], REC.LEX^.S10[I]);
00670 END
00680 END;
00690 END;
00700 (**)
00710 PROCEDURE EMITXWORD(TYP:OPDTYP;OPERAND:ADDRINT);
00720 VAR REC: RECORD CASE SEVERAL OF
00730 1: (INT:ADDRINT);
00740 2: (LEX:PLEX);
00750 3,4,5,6,7,8,9,10: ()
00760 END;
00770 I:INTEGER;
00780 BEGIN
00790 (*+32() ASSERT(TYP<>OCVFIM, 'EMITXWORD '); ()+32*)
00800 IF TYP=OCVIMMED THEN WRITE(LGO[ROUTNL^.RNLEVEL],' int ')
00810 ELSE WRITE(LGO[ROUTNL^.RNLEVEL], ' ptrw ');
00820 EMITOPRAND(TYP,OPERAND);
00830 WRITELN(LGO[ROUTNL^.RNLEVEL]);
00840 END;
00850 (**)
00860 PROCEDURE EMITXPROC(TYP:OPDTYP;OPERAND:ADDRINT);
00870 VAR REC: RECORD CASE SEVERAL OF
00880 1: (INT:ADDRINT);
00890 2: (LEX:PLEX);
00900 3,4,5,6,7,8,9,10: ()
00910 END;
00920 I:INTEGER;
00930 BEGIN
00940 WRITE(LGO[ROUTNL^.RNLEVEL],' ptrf ');EMITOPRAND(TYP,OPERAND);
00950 WRITELN(LGO[ROUTNL^.RNLEVEL]);
00960 END;
00970 (**)
00980 PROCEDURE EMITALF(OPERAND: ALFA);
00990 VAR I: INTEGER;
01000 BEGIN
01010 IF DATASTATE=STARTDATA THEN
01020 BEGIN WRITELN(LGO[ROUTNL^.RNLEVEL], 'data'); DATASTATE := INDATA END;
01030 WRITE(LGO[ROUTNL^.RNLEVEL], ' byte '); FOR I := 1 TO 9 DO WRITE(LGO[ROUTNL^.RNLEVEL], ORD(OPERAND[I]):3, ',');
01040 WRITELN(LGO[ROUTNL^.RNLEVEL], ORD(OPERAND[10]):3);
01050 END;
01060 (**)
01070 (**)
01080 PROCEDURE EMITOP (* (OPCOD:POP) *);
01090 VAR I,COUNT:INTEGER; JUMPOVER:LABL;
01100 TEMP:INTEGER; OP:MNEMONICS;
01110 PARAMNOTUSED: BOOLEAN;
01120 BEGIN
01130 IF DATASTATE<>OUTDATA THEN
01140 BEGIN DATASTATE := OUTDATA; WRITELN(LGO[ROUTNL^.RNLEVEL], 'text') END;
01150 COUNT := 0; PARAMNOTUSED := TRUE;
01160 WHILE OPCOD <> 0 DO WITH CODETABLE[OPCOD] DO
01170 BEGIN
01180 IF INLINE THEN
01190 BEGIN
01200 IF PERQCOD='CI ' THEN
01210 IF OCV=OCVFIM THEN WRITE(LGO[ROUTNL^.RNLEVEL], ' cil ')
01220 ELSE IF (OCV=OCVMEM) OR (OCV=OCVFREF) OR (OCV=OCVEXT) THEN WRITE(LGO[ROUTNL^.RNLEVEL], ' lga ')
01230 ELSE WRITE(LGO[ROUTNL^.RNLEVEL], ' ci ')
01240 ELSE IF OPCOD<>PNOOP THEN
01242 BEGIN
01250 WRITE(LGO[ROUTNL^.RNLEVEL],' ');
01260 FOR i := 1 TO 8 DO
01270 WRITE(LGO[ROUTNL^.RNLEVEL],CHR(ORD(PERQCOD[I])+32*ORD(ORD(PERQCOD[I])>63)));
01280 END;
01290 CASE PARTYP OF
01300 WOP,ACP: (* OPERAND SUPPLIED BY CODETABLE *)
01310 WRITE(LGO[ROUTNL^.RNLEVEL], ' ', PARM:1);
01320 WNP,ANP: (*NEGATIVE OPERAND SUPPLIED BY CODETABLE*)
01330 WRITE(LGO[ROUTNL^.RNLEVEL], ' ', -PARM:1);
01340 OPX,ACX: (* OPERAND IS SUPPLIED BY CODE GENERATOR *)
01350 BEGIN EMITOPRAND(OCV, OPRAND+PARM); PARAMNOTUSED := FALSE END;
01360 ONX,ANX: (* NEGATIVE OPERAND SUPPLIED BY CODE GENERATOR*)
01370 BEGIN EMITOPRAND(OCV, -OPRAND-PARM); PARAMNOTUSED := FALSE END;
01380 JMP: (* P-OP GENERATES ITS OWN LABELS FOR LOOPS ETC. *)
01390 BEGIN
01400 COUNT := PARM;
01410 JUMPOVER := GETNEXTLABEL;
01420 WRITE(LGO[ROUTNL^.RNLEVEL],' L',JUMPOVER:1);
01430 END;
01440 NON: (* NO OPERAND *);
01450 GBX: (* GLOBAL LABEL EXPECTED *)
01460 BEGIN WRITE(LGO[ROUTNL^.RNLEVEL],' L',OPRAND:1); PARAMNOTUSED := FALSE END;
01470 LCX: (* INSTRUCTION LABEL EXPECTED *)
01480 BEGIN WRITE(LGO[ROUTNL^.RNLEVEL],' L',OPRAND:1); PARAMNOTUSED := FALSE END;
01490 MOR: (* LONG OPERAND FOLLOWS IN NEXT OPCOD *)
01500 BEGIN OPCOD := NEXT;
01510 WRITE(LGO[ROUTNL^.RNLEVEL], CODETABLE[OPCOD].PERQCOD);
01520 END;
01530 END; (* OF CASE *)
01540 IF PARTYP>=ACP THEN BEGIN ADJUSTSP := ADJUSTSP+SZWORD; PARAMNOTUSED := FALSE END;
01550 IF OPCOD<>PNOOP THEN WRITELN(LGO[ROUTNL^.RNLEVEL]);
01560 IF (PERQCOD[1]=' ') AND (REGSINUSE.ECOUNT<>0) THEN EMITOP(PDISCARD);
01570 OPCOD := CODETABLE[OPCOD].NEXT;
01572 IF COUNT = 1 THEN WRITELN(LGO[ROUTNL^.RNLEVEL],' L',JUMPOVER: 1,':');
01574 COUNT := COUNT-1;
01580 END
01590 ELSE
01600 BEGIN
01610 IF PARAMNOTUSED THEN PARAM(OCVNONE, 0, OPCOD, 0, FALSE);
01620 WRITE(LGO[ROUTNL^.RNLEVEL],' ','call _',ROUTINE); WRITELN(LGO[ROUTNL^.RNLEVEL]) ;
01630 OPCOD := 0;
01640 (*+32() ASSERT((RTSTKDEPTH+ADJUSTSP) MOD 4 = 0, 'EMITOP - A'); ()+32*)
01650 IF COUNT = 1 THEN WRITELN(LGO[ROUTNL^.RNLEVEL],' L',JUMPOVER: 1,':');
01700 COUNT := COUNT-1;
01702 IF ADJUSTSP<>0 THEN EMITX1(PASP, OCVIMMED, ADJUSTSP);
01710 END;
01750 END;
01760 END;
01770 (**)
01780 PROCEDURE FIXUPF (*+05() (ALABL:LABL) ()+05*);
01790 BEGIN
01800 IF DATASTATE=STARTDATA THEN
01810 BEGIN WRITELN(LGO[ROUTNL^.RNLEVEL], 'data'); WRITELN(LGO[ROUTNL^.RNLEVEL], 'align4'); DATASTATE := INDATA END
01820 ELSE IF DATASTATE=ENDDATA THEN
01830 BEGIN WRITELN(LGO[ROUTNL^.RNLEVEL], 'text'); DATASTATE := OUTDATA END;
01840 WRITELN(LGO[ROUTNL^.RNLEVEL],'L',ALABL:1,':');
01850 END;
01860 (**)
01870 FUNCTION FIXUPM:LABL;
01880 VAR L:LABL;
01890 BEGIN
01900 IF DATASTATE=STARTDATA THEN
01910 BEGIN WRITELN(LGO[ROUTNL^.RNLEVEL], 'data'); WRITELN(LGO[ROUTNL^.RNLEVEL], 'align4'); DATASTATE := INDATA END
01920 ELSE IF DATASTATE=ENDDATA THEN
01930 BEGIN WRITELN(LGO[ROUTNL^.RNLEVEL], 'text'); DATASTATE := OUTDATA END;
01940 L := GETNEXTLABEL;
01950 FIXUPM := L;
01960 WRITELN(LGO[ROUTNL^.RNLEVEL],'L',L:1,':');
01970 END;
01980 (**)
01990 PROCEDURE FIXUPFIM(ALABL:LABL;VALUE:A68INT);
02000 BEGIN
02010 WRITELN(LGO[ROUTNL^.RNLEVEL], ' constant L', ALABL:1, ' ', VALUE: 1);
02020 END;
02030 (**)
02040 PROCEDURE FIXLABL(OLDLABL,NEWLABL:LABL; KNOWN:BOOLEAN);
02050 VAR JUMPOVER: LABL;
02060 BEGIN
02070 JUMPOVER := GETNEXTLABEL;
02080 WRITELN(LGO[ROUTNL^.RNLEVEL], ' jump L', JUMPOVER:1);
02090 WRITELN(LGO[ROUTNL^.RNLEVEL], 'L',OLDLABL:1, ':');
02100 WRITELN(LGO[ROUTNL^.RNLEVEL], ' jump L', NEWLABL:1);
02110 WRITELN(LGO[ROUTNL^.RNLEVEL], 'L',JUMPOVER:1, ':');
02120 END;
02130 FUNCTION NORMAL(SB: PSB): SBTTYP;
02140 (*RETURNS THE SBTTYP IN WHICH VALUES OF MODE SB^.SBMODE SHOULD BE STORED DURING BALANCES*)
02150 BEGIN WITH SB^ DO WITH SBMODE^.MDV DO
02160 IF SBTYP=SBTDL THEN NORMAL := SBTDL
02170 ELSE IF SBUNION IN SBINF THEN NORMAL := SBTSTKN
02180 ELSE IF SBNAKED IN SBINF THEN NORMAL := SBTFPR0
02190 ELSE IF MDPILE THEN NORMAL := SBTE
02200 ELSE CASE MDLEN OF
02210 0: NORMAL := SBTVOID;
02220 2: NORMAL := SBTE;
02230 4: NORMAL := SBTFPR0;
02240 END;
02250 END;
02260 (**)
02270 FUNCTION LENOF(SB: PSB): INTEGER;
02280 BEGIN
02290 WITH SB^,SBMODE^.MDV DO
02300 IF SBUNION IN SBINF THEN LENOF := SBLEN
02310 ELSE IF SBNAKED IN SBINF THEN LENOF := SZNAKED
02320 ELSE IF MDPILE THEN LENOF := SZADDR
02330 ELSE LENOF := MDLEN;
02340 END;
02350 (**)
02360 PROCEDURE LOADSTK(SB: PSB);
02370 BEGIN
02380 (*+21() WRITELN(OUTPUT, 'LOADSTK ', ORD(SB)); ()+21*)
02390 IF NOT(SB^.SBTYP IN [SBTSTKN,SBTDL]) THEN
02400 CASE LENOF(SB) OF
02410 0: LOAD(SBTVOID, SB);
02420 2: LOAD(SBTSTK, SB);
02430 4: LOAD(SBTSTK4, SB);
02440 END;
02450 END;
02460 (**)
02470 PROCEDURE TWIST;
02480 VAR TEMPPTR : PSB;
02490 NORM: SBTTYP;
02500 BEGIN
02510 (*+21() WRITELN(OUTPUT, 'TWIST'); ()+21*)
02520 IF [RTSTACK^.SBRTSTK^.SBTYP , RTSTACK^.SBTYP] * [SBTVOID..SBTDEN] = [] THEN
02530 (*NEITHER SB IS A FAKE*)
02540 BEGIN
02550 IF RTSTACK^.SBTYP IN [SBTSTK..SBTDL] THEN
02560 LOAD(NORMAL(RTSTACK),RTSTACK); (*GET IT INTO REGISTER 3*)
02570 TEMPPTR := RTSTACK^.SBRTSTK;
02580 RTSTACK^.SBRTSTK := TEMPPTR^.SBRTSTK;
02590 TEMPPTR^.SBRTSTK := RTSTACK;
02600 RTSTACK := TEMPPTR;
02610 IF RTSTACK^.SBTYP IN [SBTSTK..SBTDL] THEN
02620 BEGIN
02630 NORM := NORMAL(RTSTACK);
02640 IF NORM IN [SBTFPR0..SBTFPR3] THEN IF NORM IN REGSINUSE.FPR THEN NORM := SBTFPR1;
02650 LOAD(NORM,RTSTACK) (*GET IT INTO A REGISTER TOO*)
02660 END
02670 ELSE IF (RTSTACK^.SBTYP IN [SBTE,SBTER0]) AND (RTSTACK^.SBRTSTK^.SBTYP IN [SBTE,SBTER0]) THEN
02680 EMITOP(PSWAP)
02690 END
02700 ELSE BEGIN
02710 TEMPPTR := RTSTACK^.SBRTSTK;
02720 RTSTACK^.SBRTSTK := TEMPPTR^.SBRTSTK;
02730 TEMPPTR^.SBRTSTK := RTSTACK;
02740 RTSTACK := TEMPPTR
02750 END
02760 END;
02770 (**)
02780 PROCEDURE HOIST(HOISTLEN, LEN:INTEGER; ALIGN: BOOLEAN);
02782 (*HOISTLEN IS AMOUNT ALREADY STACKED; LEN IS TOTAL AMOUNT TO BE STACKED*)
02790 BEGIN
02800 IF ((RTSTKDEPTH-HOISTLEN+LEN) MOD 4 = 0) = ALIGN THEN
02810 BEGIN
02820 IF HOISTLEN=0 THEN EMITOP(PALIGN)
02830 ELSE
02840 BEGIN
02850 HOISTLEN := HOISTLEN-RTSTKDEPTH; CLEAR(RTSTACK); HOISTLEN := HOISTLEN+RTSTKDEPTH;
02860 EMITX1(PHEAVE, OCVIMMED, HOISTLEN);
02870 END;
02880 ADJUSTSP := ADJUSTSP+SZWORD;
02890 END;
02900 END;
02910 (**)
02920 PROCEDURE PROC1OP(OPCOD:POP;TYP:OPDTYP;OPND:INTEGER;NOTINL:BOOLEAN;ALIGN:INTEGER);
02930 VAR SB, SB1: PSB;
02940 HOISTLEN,LEN: INTEGER;
02950 BEGIN
02960 SB:=ASPTR(OPND);
02962 SB^.SBINF := SB^.SBINF-[SBSTKDELAY];
02970 IF RTSTACK<>SB THEN TWIST;
02972 WITH CODETABLE[OPCOD] DO
02974 BEGIN
02980 IF NOTINL THEN WITH SB^ DO
02990 BEGIN
03000 IF SBSTKDELAY IN SBRTSTK^.SBINF THEN LOADSTK(SBRTSTK)
03010 ELSE CLEAR(SBRTSTK);
03020 IF TYP=OCVSBS THEN
03030 BEGIN
03040 HOISTLEN := SUBSTLEN([SBTSTK..SBTDL]);
03050 LEN := HOISTLEN+LENOF(SB)*ORD(NOT(SB^.SBTYP IN [SBTSTK..SBTDL]));
03060 END
03070 ELSE BEGIN
03080 LEN := LENOF(SB)*ORD(P1 IN [SBTSTK..SBTDL,SBTPR1,SBTPR2]);
03090 HOISTLEN := SBLEN*ORD(SB^.SBTYP IN [SBTSTK..SBTDL]);
03100 END;
03110 HOIST(HOISTLEN, LEN, NOT ODD(ALIGN+APARAMS));
03120 END;
03150 REPEAT
03151 IF PR IN (REGSINUSE.FPR-[P1]) THEN
03152 BEGIN SB1 := RTSTACK;
03153 WHILE NOT(SB1^.SBTYP IN (REGSINUSE.FPR-[P1])) DO SB1 := SB1^.SBRTSTK;
03155 CLEAR(SB1);
03156 END;
03157 LOAD(P1, SB);
03158 UNTIL P1 IN [SBTSTKN,SBTPR1,SBTPR2,SBTXN,SB^.SBTYP]; (*ESTACK MAY HAVE OVERFLOWED*)
03160 UNSTKP1(TYP,SB);
03180 END;
03190 OCV := OCVNONE; (*FOR 1ST CALL OF PARAM*)
03200 END;
03210 (**)
03220 PROCEDURE PROC2OP(OPCOD:POP;TYP1:OPDTYP;OPND1:INTEGER;TYP2:OPDTYP;OPND2:INTEGER;NOTINL:BOOLEAN;ALIGN:INTEGER);
03230 VAR SB1, SB2, SB3: PSB;
03240 HOISTLEN,LEN1,LEN2: INTEGER;
03250 BEGIN
03260 SB1:=ASPTR(OPND1);
03262 SB1^.SBINF := SB1^.SBINF-[SBSTKDELAY];
03270 SB2:=ASPTR(OPND2);
03271 SB2^.SBINF := SB2^.SBINF-[SBSTKDELAY];
03272 WITH CODETABLE[OPCOD] DO
03274 BEGIN
03280 IF NOTINL THEN WITH RTSTACK^.SBRTSTK^ DO
03290 BEGIN
03300 IF SBSTKDELAY IN SBRTSTK^.SBINF THEN LOADSTK(SBRTSTK)
03310 ELSE CLEAR(SBRTSTK);
03312 IF TYP1=OCVSBS THEN
03314 HOIST(SUBSTLEN([SBTSTK..SBTDL]), SUBSTLEN([SBTID..SBTFPR1]), ODD(ALIGN+APARAMS))
03316 ELSE
03318 BEGIN
03320 LEN1 := LENOF(SB1)*ORD(P1 IN [SBTSTK..SBTDL,SBTPR1,SBTPR2]);
03322 LEN2 := LENOF(SB2)*ORD(P2 IN [SBTSTK..SBTDL,SBTPR1,SBTPR2]);
03330 HOISTLEN := SB1^.SBLEN*ORD(SB1^.SBTYP IN [SBTSTK..SBTDL])
03340 +SB2^.SBLEN*ORD(SB2^.SBTYP IN [SBTSTK..SBTDL]);
03350 HOIST(HOISTLEN, LEN1+LEN2, ODD(ALIGN+APARAMS));
03352 END;
03360 END;
03370 IF RTSTACK<>SB2 THEN TWIST;
03400 IF (SB2^.SBTYP IN [SBTSTK..SBTSTKN,SBTVAR]) OR ((P1 IN REGSINUSE.FPR) AND (P1<>SB1^.SBTYP)) THEN
03410 LOAD(P2,SB2);
03412 REPEAT
03413 IF PR IN (REGSINUSE.FPR-[P1,P2]) THEN
03414 BEGIN SB3 := RTSTACK;
03415 WHILE NOT(SB3^.SBTYP IN (REGSINUSE.FPR-[P1,P2])) DO SB3 := SB3^.SBRTSTK;
03416 CLEAR(SB3);
03418 END;
03420 LOAD(P1, SB1);
03430 LOAD(P2, SB2);
03432 UNTIL (P1 IN [SBTSTKN,SBTPR1,SBTPR2,SBTXN,SB1^.SBTYP]) AND
03434 (P2 IN [SBTSTKN,SBTPR1,SBTPR2,SBTXN,SB2^.SBTYP]); (*ESTACK MAY HAVE OVERFLOWED*)
03440 UNSTKP1(TYP2,SB2);
03450 UNSTKP1(TYP1,SB1);
03470 END;
03480 OCV := OCVNONE; (*FOR 1ST CALL OF PARAM*)
03490 END;
03500 (**)
03510 PROCEDURE FILL (WHERE:SBTTYP; SB:PSB);
03520 BEGIN
03530 WITH SB^ DO WITH REGSINUSE DO
03540 BEGIN
03550 IF SBTYP IN [SBTER0..SBTFPR3] THEN FPR := FPR-[SBTYP];
03560 IF SBTYP IN [SBTE,SBTER0] THEN ECOUNT := ECOUNT-1
03570 ELSE IF SBTYP IN [SBTSTK..SBTDL] THEN RTSTKDEPTH := RTSTKDEPTH-SBLEN;
03572 SBTYP:=WHERE;
03580 IF NOT(WHERE IN [SBTSTKN,SBTDL,SBTXN]) THEN SBLEN := LENARRAY[WHERE];
03590 IF WHERE IN [SBTSTK..SBTDL] THEN
03600 BEGIN
03610 RTSTKDEPTH := RTSTKDEPTH+SBLEN;
03620 WITH ROUTNL^ DO
03630 IF RTSTKDEPTH>RNLENSTK THEN RNLENSTK:=RTSTKDEPTH
03640 END
03650 ELSE
03654 BEGIN
03660 IF WHERE IN [SBTE,SBTER0] THEN
03662 BEGIN ECOUNT := ECOUNT+1; IF ECOUNT>=6 THEN CLEAR(RTSTACK) END;
03670 IF WHERE IN [SBTER0..SBTFPR3] THEN FPR := FPR+[WHERE];
03674 END;
03690 END
03700 END;
03710 (**)
03720 FUNCTION SETINLINE (OPCOD:POP):BOOLEAN;
03730 VAR INL:BOOLEAN;
03740 BEGIN
03750 APARAMS := 0;
03760 OCV := OCVNONE; (*FOR 1ST CALL OF PARAM*)
03770 REPEAT WITH CODETABLE[OPCOD] DO
03780 BEGIN
03790 APARAMS := APARAMS+ORD(PARTYP IN [ACP,ANP]); (*NUMBER OF SECRET PARAMETERS*)
03800 INL := INLINE;
03810 OPCOD := NEXT
03820 END
03830 UNTIL NOT(INL) OR (OPCOD=0);
03840 SETINLINE := INL
03850 END;
03860 (**)
03870 (**)
03880 PROCEDURE LOAD (*+05() (WHERE:SBTTYP; SB:PSB) ()+05*);
03890 (*EMITS CODE TO MOVE SB TO WHERE: CALLS FILL TO RECORD THE MOVE*)
03900 VAR TEMPOP: POP;
03910 TOFFSET: INTEGER;
03920 TEMPTYP: SBTTYP;
03930 OCVFIX: OPDTYP;
03940 TWISTED: BOOLEAN;
03950 TYPS: SET OF SBTTYP;
03960 SB1, SB2: PSB;
03970 SAVE, EC:INTEGER;
03980 BEGIN
03990 (*+21() WRITELN(OUTPUT, 'LOAD ',ORD(SB):5,ORD(SB^.SBTYP):3,' TO ', ORD(WHERE):3, SB=RTSTACK); ()+21*)
04000 WITH SB^ DO
04010 BEGIN
04012 SBINF := SBINF-[SBSTKDELAY];
04020 IF SBRTSTK<>NIL THEN
04030 IF SBSTKDELAY IN SBRTSTK^.SBINF THEN
04040 LOADSTK(SBRTSTK);
04050 IF (WHERE IN [SBTSTK..SBTDL]) THEN CLEAR(SBRTSTK);
04060 TWISTED := FALSE;
04070 IF WHERE IN [SBTSTKN,SBTPR1,SBTPR2] THEN
04080 LOADSTK(SB)
04090 ELSE IF WHERE=SBTXN THEN LOAD(NORMAL(SB),SB)
04100 ELSE
04110 IF WHERE <> SBTVOID THEN
04120 BEGIN
04140 IF WHERE IN [SBTER0..SBTFPR3] THEN
04150 IF (WHERE IN REGSINUSE.FPR) AND (WHERE<>SBTYP) THEN
04160 BEGIN
04170 SB1 := RTSTACK;
04180 WHILE NOT(SB1^.SBTYP IN REGSINUSE.FPR) DO SB1 := SB1^.SBRTSTK;
04190 LOADSTK(SB1);
04200 END;
04240 TYPS := [WHERE, RTSTACK^.SBTYP];
04250 IF (RTSTACK<>SB) THEN
04260 IF (TYPS <= [SBTSTK..SBTDL]) AND NOT(SBTYP IN [SBTSTK..SBTDL]) OR (TYPS<=[SBTE,SBTER0]) THEN
04270 BEGIN TWISTED:=TRUE; TWIST;
04280 (*+32() ASSERT (RTSTACK =SB,'LOAD-B '); ()+32*)
04290 END;
04310 TEMPOP := POPARRAY[WHERE,SBTYP];
04320 (*+32() ASSERT(TEMPOP<>PNONE, 'LOAD-C '); ()+32*)
04330 IF (TEMPOP<>PNOOP) OR (SBTYP=SBTSTKR0) THEN
04340 CASE SBTYP OF
04350 SBTRPROC,SBTPROC,SBTVAR: BEGIN
04360 SAVE := ADJUSTSP; ADJUSTSP := 0;
04370 RTSTKDEPTH := RTSTKDEPTH+SAVE;
04380 IF WHERE <> SBTE THEN BEGIN LOAD(SBTE,SB); LOAD(WHERE,SB) END
04390 ELSE BEGIN TOFFSET:=GENLCLGBL(TEMPOP,SB);
04400 IF SBTYP=SBTVAR THEN
04410 EMITX2(TEMPOP,OCVIMMED,SBLOCRG,OCVIMMED,TOFFSET)
04420 ELSE BEGIN (*SBTPROC OR SBTRPROC*)
04430 IF SBTYP=SBTPROC THEN OCVFIX := OCVMEM
04440 ELSE (* SBTRPROC *) OCVFIX := OCVFREF;
04450 EMITX2(TEMPOP,OCVFIX,SBXPTR,OCVIMMED,TOFFSET);
04460 END;
04470 END;
04480 RTSTKDEPTH := RTSTKDEPTH-SAVE;
04490 ADJUSTSP := SAVE;
04500 END;
04510 (**)
04520 SBTID,SBTIDV: BEGIN TOFFSET:=GENLCLGBL(TEMPOP,SB);
04530 EMITX1(TEMPOP,OCVIMMED,TOFFSET) END;
04540 SBTLIT: EMITX1(TEMPOP, OCVIMMED, SBVALUE);
04550 SBTDEN: GENDENOT(TEMPOP,SB);
04560 SBTPR1,SBTPR2,
04570 SBTSTK,SBTSTK4,SBTDL,SBTER0: EMITOP(TEMPOP);
04580 SBTE: WITH REGSINUSE DO
04600 BEGIN
04610 (*ATTEMPT TO STACK E MUST FORCE STACKING OF ALL E'S ABOVE IT;
04612 THESE ARE THE EXTRAS*)
04620 SB1 := RTSTACK; EEXTRA := 0; EC := ECOUNT; TEMPOP := TEMPOP+ORD(EC=2)+ORD(EC>2);
04630 REPEAT WITH SB1^ DO (*PREVENT CLEAR IF TEMPOP IS AN OCODE*)
04632 BEGIN
04634 IF SBTYP=SBTE THEN
04636 BEGIN FILL(SBTSTK, SB1); EEXTRA := EEXTRA+1 END
04637 ELSE IF SBTYP=SBTER0 THEN
04638 BEGIN FILL(SBTSTKR0, SB1); EEXTRA := 0 END
04639 ELSE IF SBTYP IN [SBTFPR0,SBTFPR1] THEN EEXTRA := 0;
04640 SB2 := SB1; SB1 := SBRTSTK;
04642 END
04644 UNTIL SB2=SB;
04650 EMITX1(TEMPOP, OCVIMMED, ECOUNT);
04660 EEXTRA := EC-EEXTRA;
04661 (*NO. OF E'S OR ER0'S ABOVE FIRST FPR, OR ABOVE & INCL. FIRST ER0*)
04662 END;
04670 SBTSTKR0,SBTFPR0,SBTFPR1: WITH REGSINUSE DO
04680 BEGIN
04690 IF EEXTRA<>0 THEN
04700 BEGIN
04710 EMITX1(PSTKTOE+ORD(EEXTRA=2)+ORD(EEXTRA>2), OCVIMMED, EEXTRA);
04720 SB1 := RTSTACK;
04722 WHILE EEXTRA>0 DO WITH SB1^ DO
04723 BEGIN
04724 IF SBTYP=SBTSTK THEN
04725 BEGIN FILL(SBTE, SB1); EEXTRA := EEXTRA-1 END
04726 ELSE IF SBTYP=SBTSTKR0 THEN
04727 BEGIN FILL(SBTER0, SB1); EEXTRA := EEXTRA-1 END;
04728 SB1 := SBRTSTK;
04729 END;
04730 END;
04740 EMITOP(TEMPOP);
04750 END;
04760 END;
04770 FILL(WHERE,SB);
04780 END;
04790 IF TWISTED THEN TWIST;
04800 END;
04810 END;
04820 (**)
04830 PROCEDURE PARAM (*(TYP:OPDTYP; OPND:INTEGER; OPCOD: POP; ALIGN: INTEGER; FIRSTIME: BOOLEAN)*);
04840 VAR TEMPOP:POP;
04850 OPERANDUSED, INL: BOOLEAN;
04860 BEGIN
04870 IF OCV<>OCVNONE THEN
04880 BEGIN
04890 TEMPOP := PPUSHIM;
04900 EMITOP(TEMPOP) ; ADJUSTSP := ADJUSTSP+SZWORD;
04910 END;
04920 IF FIRSTIME AND (((RTSTKDEPTH+ADJUSTSP) MOD 4 = 0) = ODD(ALIGN+APARAMS)) THEN
04930 BEGIN EMITOP(PALIGN); ADJUSTSP := ADJUSTSP+SZWORD END;
04940 OPRAND:=OPND; OCV := TYP;
04950 END;
04960 (**)
04970 ()+05*)
04980 (**)
04990 (*+01() (*+31() (*$T+ +) ()+31+) ()+01*)
05000 (*+05() (*+31() (*$T+ +) ()+31+) ()+05*)
05010 (**)
05020 (**)
05030 (**)
05040 PROCEDURE CLEAR (SB:PSB);
05050 (*ENSURES THAT NOTHING ON RTSTACK FROM SB DOWNWARDS IS IN A REGISTER*)
05060 LABEL 9;
05070 VAR TEMPPTR: PSB;
05080 BEGIN
05090 (*INVARIANT: IF SBTYP IN [SBTSTK..SBTSTKN], NOTHING BELOW SB IS IN A REGISTER*)
05100 TEMPPTR:=SB;
05110 WHILE TEMPPTR<>NIL DO WITH TEMPPTR^ DO
05120 IF SBTYP>SBTSTKN THEN
05130 BEGIN LOADSTK(TEMPPTR); GOTO 9 END
05140 ELSE IF SBTYP>=SBTSTK THEN GOTO 9 (*BECAUSE OF INVARIANT*)
05150 ELSE TEMPPTR := SBRTSTK;
05160 9:
05170 END;
05180 (**)
05190 (*-23()
05200 ()-23*)
05210 PROCEDURE UNSTKP1 (*+05() (TYP:OPDTYP; OPND:PSB) ()+05*);
05220 BEGIN
05230 IF TYP = OCVSBS THEN
05240 (*ASSERT: OPND = RTSTACK*)
05250 REPEAT
05260 OPND := RTSTACK;
05270 UNSTACKSB;
05280 IF OPND^.SBTYP IN [SBTSTK..SBTDL] THEN ADJUSTSP := ADJUSTSP+OPND^.SBLEN;
05290 OPND^.SBTYP := SBTVOID;
05300 UNTIL OPND=SRSTK[SRSUBP+1].SB
05310 ELSE IF TYP <> OCVSBP THEN
05320 BEGIN UNSTACKSB;
05330 IF OPND^.SBTYP IN [SBTSTK..SBTDL] THEN ADJUSTSP := ADJUSTSP+OPND^.SBLEN;
05340 OPND^.SBTYP:=SBTVOID;
05350 END
05360 (*+02() ELSE (*TYP=OCVSBP*) ADJUSTSP := ADJUSTSP-LENOF(OPND); ()+02*)
05370 END;
05380 (**)
05390 (*-23()
05400 ()-23*)
05410 (**)
05420 PROCEDURE EMITX0(OPCOD: POP);
05430 BEGIN IF NOT SETINLINE(OPCOD) THEN BEGIN ADJUSTSP := 0; CLEAR(RTSTACK) END;
05440 (*+05() PARAM(OCVNONE,0,OPCOD,0,NOT SETINLINE(OPCOD)); ()+05*)
05450 EMITOP(OPCOD);
05460 END;
05470 (**)
05480 (**)
05490 PROCEDURE EMITX1 (*+05() (OPCOD:POP; TYP1:OPDTYP;OPND1:ADDRINT) ()+05*);
05500 VAR SB1:PSB; NOTINL:BOOLEAN;
05510 BEGIN
05520 (*-24()(*+23() WRITELN(LGO[ROUTNL^.RNLEVEL]); ()+23*) ()-24*)
05530 IF TYP1 = OCVRES THEN
05540 BEGIN
05550 SB1 := ASPTR(OPND1);
05560 EMITX0 (OPCOD);
05570 (*+32() ASSERT(CODETABLE[OPCOD].PR<>SBTVOID,'EMITX1-A ');
05580 ASSERT(SB1^.SBTYP=SBTVOID,'EMITX1-B '); ()+32*)
05590 FILL(CODETABLE[OPCOD].PR,SB1);
05600 SB1^.SBRTSTK:=RTSTACK; RTSTACK:=SB1;
05610 END
05620 ELSE
05630 BEGIN
05640 NOTINL := NOT(SETINLINE(OPCOD));
05650 IF NOTINL THEN ADJUSTSP := 0;
05660 IF TYP1 >= OCVSB THEN
05670 PROC1OP(OPCOD,TYP1,OPND1,NOTINL(*+05(),1()+05*))
05680 ELSE
05690 BEGIN
05700 IF NOTINL THEN CLEAR(RTSTACK);
05710 (*+01() NEXTREG := 0; ()+01*)
05720 PARAM(TYP1,OPND1,OPCOD(*+05(),1,NOTINL()+05*));
05730 END;
05740 EMITOP(OPCOD)
05750 END
05760 END;
05770 (**)
05780 (**)
05790 PROCEDURE EMITX2 (*+05() (OPCOD:POP; TYP1:OPDTYP;OPND1:ADDRINT;
05800 TYP2:OPDTYP; OPND2:ADDRINT) ()+05*);
05810 VAR SB2:PSB; NOTINL:BOOLEAN;
05820 BEGIN
05830 (*+23() WRITELN(LGO[ROUTNL^.RNLEVEL]); ()+23*)
05840 IF TYP2 = OCVRES THEN
05850 BEGIN
05860 SB2 := ASPTR(OPND2);
05870 EMITX1 (OPCOD, TYP1,OPND1);
05880 (*+32() ASSERT(CODETABLE[OPCOD].PR<>SBTVOID,'EMITX2-A ');
05890 ASSERT(SB2^.SBTYP=SBTVOID,'EMITX2-B '); ()+32*)
05900 FILL(CODETABLE[OPCOD].PR,SB2);
05910 SB2^.SBRTSTK:=RTSTACK; RTSTACK:=SB2;
05920 END
05930 ELSE
05940 BEGIN
05950 NOTINL := NOT(SETINLINE(OPCOD));
05960 IF NOTINL THEN ADJUSTSP := 0;
05970 IF TYP1 >= OCVSB THEN
05980 IF TYP2 >= OCVSB THEN PROC2OP(OPCOD,TYP1,OPND1,TYP2,OPND2,NOTINL(*+05(),2()+05*))
05990 ELSE BEGIN PROC1OP(OPCOD,TYP1,OPND1,NOTINL(*+05(),2()+05*));
06000 PARAM(TYP2,OPND2,OPCOD(*+05(),1,FALSE()+05*)) END
06010 ELSE
06020 BEGIN
06030 IF NOTINL THEN CLEAR(RTSTACK);
06040 (*+01() NEXTREG:=0; ()+01*)
06050 PARAM(TYP1,OPND1,OPCOD(*+05(),2,NOTINL()+05*));
06060 PARAM(TYP2,OPND2,OPCOD(*+05(),1,FALSE()+05*))
06070 END;
06080 EMITOP(OPCOD)
06090 END
06100 END;
06110 (**)
06120 (**)
06130 PROCEDURE EMITX3 (OPCOD:POP; TYP1:OPDTYP;OPND1:ADDRINT; TYP2:OPDTYP;OPND2:ADDRINT;
06140 TYP3:OPDTYP; OPND3:ADDRINT);
06150 VAR SB3:PSB; NOTINL:BOOLEAN;
06160 BEGIN
06170 (*+23() WRITELN(LGO[ROUTNL^.RNLEVEL]); ()+23*)
06180 IF TYP3 = OCVRES THEN
06190 BEGIN
06200 SB3 := ASPTR(OPND3);
06210 EMITX2 (OPCOD, TYP1,OPND1, TYP2,OPND2);
06220 (*+32() ASSERT(CODETABLE[OPCOD].PR<>SBTVOID,'EMITX3-A ');
06230 ASSERT(SB3^.SBTYP=SBTVOID,'EMITX3-B '); ()+32*)
06240 FILL(CODETABLE[OPCOD].PR,SB3);
06250 SB3^.SBRTSTK:=RTSTACK; RTSTACK:=SB3;
06260 END
06270 ELSE
06280 BEGIN
06290 NOTINL := NOT(SETINLINE(OPCOD));
06300 IF NOTINL THEN ADJUSTSP := 0;
06310 IF TYP1 >= OCVSB THEN
06320 IF TYP2 >= OCVSB THEN PROC2OP(OPCOD,TYP1,OPND1,TYP2,OPND2,NOTINL(*+05(),3()+05*))
06330 ELSE BEGIN PROC1OP(OPCOD,TYP1,OPND1,NOTINL(*+05(),3()+05*));
06340 PARAM(TYP2,OPND2,OPCOD(*+05(),2,FALSE()+05*)) END
06350 ELSE
06360 BEGIN
06370 IF NOTINL THEN CLEAR(RTSTACK);
06380 (*+01() NEXTREG:=0; ()+01*)
06390 PARAM(TYP1,OPND1,OPCOD(*+05(),3,NOTINL()+05*));
06400 PARAM(TYP2,OPND2,OPCOD(*+05(),2,FALSE()+05*))
06410 END;
06420 PARAM(TYP3,OPND3,OPCOD(*+05(),1,FALSE()+05*));
06430 EMITOP(OPCOD)
06440 END
06450 END;
06460 (**)
06470 (**)
06480 PROCEDURE EMITX4 (OPCOD:POP; TYP1:OPDTYP;OPND1:ADDRINT; TYP2:OPDTYP;OPND2:ADDRINT;
06490 TYP3:OPDTYP; OPND3:ADDRINT; TYP4:OPDTYP;OPND4:ADDRINT);
06500 VAR SB4:PSB; NOTINL:BOOLEAN;
06510 BEGIN
06520 (*+23() WRITELN(LGO[ROUTNL^.RNLEVEL]); ()+23*)
06530 IF TYP4 = OCVRES THEN
06540 BEGIN
06550 SB4 := ASPTR(OPND4);
06560 EMITX3 (OPCOD, TYP1,OPND1, TYP2,OPND2, TYP3,OPND3);
06570 (*+32() ASSERT(CODETABLE[OPCOD].PR<>SBTVOID,'EMITX4-A ');
06580 ASSERT(SB4^.SBTYP=SBTVOID,'EMITX4-B '); ()+32*)
06590 FILL(CODETABLE[OPCOD].PR,SB4);
06600 SB4^.SBRTSTK:=RTSTACK; RTSTACK:=SB4;
06610 END
06620 ELSE
06630 BEGIN
06640 NOTINL := NOT(SETINLINE(OPCOD));
06650 IF NOTINL THEN ADJUSTSP := 0;
06660 IF TYP1 >= OCVSB THEN
06670 IF TYP2 >= OCVSB THEN PROC2OP(OPCOD,TYP1,OPND1,TYP2,OPND2,NOTINL(*+05(),4()+05*))
06680 ELSE BEGIN PROC1OP(OPCOD,TYP1,OPND1,NOTINL(*+05(),4()+05*));
06690 PARAM(TYP2,OPND2,OPCOD(*+05(),3,FALSE()+05*)) END
06700 ELSE
06710 BEGIN
06720 IF NOTINL THEN CLEAR(RTSTACK);
06730 (*+01() NEXTREG:=0; ()+01*)
06740 PARAM(TYP1,OPND1,OPCOD(*+05(),4,NOTINL()+05*));
06750 PARAM(TYP2,OPND2,OPCOD(*+05(),3,FALSE()+05*))
06760 END;
06770 PARAM(TYP3,OPND3,OPCOD(*+05(),2,FALSE()+05*));
06780 PARAM(TYP4,OPND4,OPCOD(*+05(),1,FALSE()+05*));
06790 EMITOP(OPCOD)
06800 END
06810 END;
06820 (**)
06830 (**)
06840 PROCEDURE EMITX5 (OPCOD:POP; TYP1:OPDTYP;OPND1:ADDRINT; TYP2:OPDTYP;OPND2:ADDRINT;
06850 TYP3:OPDTYP;OPND3:ADDRINT;TYP4:OPDTYP;OPND4:ADDRINT;TYP5:OPDTYP;OPND5:ADDRINT);
06860 VAR SB5:PSB; NOTINL:BOOLEAN;
06870 BEGIN
06880 (*+23() WRITELN(LGO[ROUTNL^.RNLEVEL]); ()+23*)
06890 IF TYP5 = OCVRES THEN
06900 BEGIN
06910 SB5 := ASPTR(OPND5);
06920 EMITX4 (OPCOD, TYP1,OPND1, TYP2,OPND2, TYP3,OPND3,TYP4,OPND4);
06930 (*+32() ASSERT(CODETABLE[OPCOD].PR<>SBTVOID,'EMITX5-A ');
06940 ASSERT(SB5^.SBTYP=SBTVOID,'EMITX5-B '); ()+32*)
06950 FILL(CODETABLE[OPCOD].PR,SB5);
06960 SB5^.SBRTSTK:=RTSTACK; RTSTACK:=SB5;
06970 END
06980 ELSE
06990 BEGIN
07000 NOTINL := NOT(SETINLINE(OPCOD));
07010 IF NOTINL THEN ADJUSTSP := 0;
07020 IF TYP1 >= OCVSB THEN
07030 IF TYP2 >= OCVSB THEN PROC2OP(OPCOD,TYP1,OPND1,TYP2,OPND2,NOTINL(*+05(),5()+05*))
07040 ELSE BEGIN PROC1OP(OPCOD,TYP1,OPND1,NOTINL(*+05(),5()+05*));
07050 PARAM(TYP2,OPND2,OPCOD(*+05(),4,FALSE()+05*)) END
07060 ELSE
07070 BEGIN
07080 IF NOTINL THEN CLEAR(RTSTACK);
07090 (*+01() NEXTREG:=0; ()+01*)
07100 PARAM(TYP1,OPND1,OPCOD(*+05(),5,NOTINL()+05*));
07110 PARAM(TYP2,OPND2,OPCOD(*+05(),4,FALSE()+05*))
07120 END;
07130 PARAM(TYP3,OPND3,OPCOD(*+05(),3,FALSE()+05*));
07140 PARAM(TYP4,OPND4,OPCOD(*+05(),2,FALSE()+05*));
07150 PARAM(TYP5,OPND5,OPCOD(*+05(),1,FALSE()+05*));
07160 EMITOP(OPCOD)
07170 END
07180 END;
07190 (**)
07200 (**)
07210 (*-23()
07220 ()-23*) (* MORE PERQ DEPENDENT ROUTINES *)
07230 (**) (********************************)
07240 (*+05()
07250 PROCEDURE EMITBEG;
07260 VAR TEMP : PLEX;
07270 S: ARGSTRING;
07280 I,J: INTEGER;
07290 PROCEDURE NAMEFILE(S: ARGSTRING; SU, SL: INTEGER; VAR F: ANYFILE); EXTERN;
07300 FUNCTION GETARG(VAR S: ARGSTRING; SU, SL: INTEGER; I: INTEGER): BOOLEAN; EXTERN;
07310 BEGIN
07320 NEXTLABEL := 1;
07330 DATASTATE := ENDDATA;(* ??? *)
07340 ADJUSTSP := 0;
07350 WITH REGSINUSE DO
07360 BEGIN
07370 ECOUNT := 0;
07380 EEXTRA := 0;
07390 FPR := [];
07400 END;
07410 IF GETARG(S, 50, 1, 2) THEN
07420 BEGIN
07430 J := 1; WHILE S[J]<>CHR(0) DO J := J+1; S[J+1] := CHR(0);
07440 FOR I := 0 TO LASTRNLEVEL DO
07450 BEGIN S[J] := CHR(I+ORD('0')); NAMEFILE(S, 50, 1, LGO[I]); REWRITE(LGO[I]) END;
07460 END;
07470 (*+33() WRITELN(LGO[ROUTNL^.RNLEVEL], 'stab "a68",8#44,0,0,_AL68_'); ()+33*)
07480 WRITELN(LGO[ROUTNL^.RNLEVEL],'global _AL68_');
07490 WRITELN(LGO[ROUTNL^.RNLEVEL],'function _AL68_');
07492 ROUTNL^.RNADDRESS := GETNEXTLABEL;
07494 WRITELN(LGO[ROUTNL^.RNLEVEL],'function L', ROUTNL^.RNADDRESS:1);
07500 ROUTNL^.RNPROCBLK := GETNEXTLABEL;
07502 WRITELN(LGO[ROUTNL^.RNLEVEL], 'data');
07504 WRITELN(LGO[ROUTNL^.RNLEVEL], 'int 1,1,1,1');
07506 (*so that no dblock has address < maxsize of undressed value*)
07510 EMITX1(PASP, OCVFIM, ROUTNL^.RNPROCBLK);
07520 EMITX0(PPBEGIN+1);
07530 EMITX0(PPBEGIN);
07540 END;
07550 (**)
07560 (**)
07570 FUNCTION EMITRTNHEAD: LABL;
07580 VAR L: LABL;
07590 BEGIN
07600 L := GETNEXTLABEL;
07610 (*+33() WRITELN(LGO[ROUTNL^.RNLEVEL], 'stab "a68",8#44,0,0,L', L:1); ()+33*)
07620 WRITELN(LGO[ROUTNL^.RNLEVEL], 'function L', L:1);
07630 EMITRTNHEAD := L;
07640 END;
07650 (**)
07660 (**)
07670 PROCEDURE EMITEND;
07680 BEGIN
07690 WITH ROUTNL^ DO IF (RNLENIDS MOD 4) = 0 THEN RNLENIDS := RNLENIDS+SZWORD;
07700 FIXUPFIM(ROUTNL^.RNPROCBLK, -(ROUTNL^.RNLENIDS+SIZIBTOP+SZWORD+FIRSTIBOFFSET));
07710 RTSTKDEPTH := 0;
07720 EMITX0(PPEND);
07730 WRITELN(LGO[ROUTNL^.RNLEVEL],' return');
07740 END;
07750 ()+05*)
07760 (**)
07770 (*+05()
07780 PROCEDURE GENDENOT (* (OPCOD: POP; SB: PSB) *) ;
07790 VAR I: INTEGER;
07800 ALABL: LABL;
07810 THING: OBJECT;
07820 BEGIN WITH SB^ DO
07830 WITH SBLEX^ (*A LEXEME*) DO
07840 IF SBLEX=LEXFALSE THEN
07850 EMITX1(OPCOD, OCVIMMED, 0)
07860 ELSE IF SBLEX=LEXTRUE THEN
07870 EMITX1(OPCOD, OCVIMMED, TRUEVALUE)
07880 ELSE IF ((SBMODE=MDINT) OR (SBMODE=MDBITS) OR (SBMODE=MDCHAR))
07890 AND (LXTOKEN=TKDENOT) THEN
07900 EMITX1(OPCOD, OCVIMMED, LXDENRP)
07910 ELSE
07920 BEGIN
07930 IF LXV.LXPYPTR=0 THEN
07940 BEGIN
07950 DATASTATE := STARTDATA; ALABL := FIXUPM;
07960 LXV.LXPYPTR := ALABL;
07970 IF SBMODE^.MDV.MDPILE THEN WITH THING DO
07980 BEGIN
07990 FIRSTWORD := 0; PCOUNT := 255;
08000 EMITXWORD(OCVIMMED, FIRSTWORD);
08010 EMITXWORD(OCVIMMED, 0);
08012 EMITXWORD(OCVIMMED, 0);
08014 EMITXWORD(OCVIMMED, 0);
08020 EMITXWORD(OCVIMMED, LXDENRP);
08030 FOR I := 3 TO LXCOUNT DO
08040 EMITXWORD(OCVIMMED, INTEGERS[I])
08050 END
08060 ELSE IF SBMODE^.MDV.MDID IN [MDIDCHAN,MDIDPASC] THEN
08070 BEGIN
08080 EMITXWORD(OCVIMMED, -2);
08090 EMITXPROC(OCVEXT, ORD(SBLEX));
08100 END
08110 ELSE
08120 BEGIN
08130 EMITXWORD(OCVIMMED,INTEGERS[2]);
08140 EMITXWORD(OCVIMMED,INTEGERS[3]);
08150 END;
08160 END;
08170 EMITX1(OPCOD+1, OCVMEM, LXV.LXPYPTR)
08180 END;
08190 END;
08200 (**)
08210 PROCEDURE GENDP(M: MODE);
08220 (*FUNCTION: RETURNS THE ADDRESS OF THE DBLOCK FOR MODE M, OR THE VALULENGTH,
08230 IN GLOBAL VARIABLE GENDPVAL, WITH CORRESPONDING OPDTYP IN GENDPOCV.
08240 *)
08250 VAR OFFSET: 0..127;
08260 PROCEDURE DBLOCK(M: MODE);
08270 VAR I, J: INTEGER;
08280 BEGIN WITH M^ DO
08290 FOR I := 0 TO MDV.MDCNT-1 DO
08300 WITH MDSTRFLDS[I] DO WITH MDSTRFMD^.MDV DO
08310 IF MDDRESSED THEN
08320 BEGIN EMITXWORD(OCVIMMED, OFFSET); OFFSET := OFFSET+MDLEN END
08330 ELSE IF MDID=MDIDSTRUCT THEN
08340 DBLOCK(MDSTRFMD)
08350 ELSE OFFSET := OFFSET+MDLEN
08360 END;
08370 PROCEDURE DBLOCKM(M: MODE);
08380 VAR I: INTEGER; X: XTYPE;
08390 BEGIN WITH M^ DO
08400 FOR I := 0 TO MDV.MDCNT-1 DO
08410 WITH MDSTRFLDS[I] DO
08420 BEGIN X := TX(MDSTRFMD);
08430 IF X=12 THEN DBLOCKM(MDSTRFMD)
08440 ELSE EMITXWORD(OCVIMMED, X+1)
08450 END
08460 END;
08470 BEGIN WITH M^ DO
08480 IF MDV.MDID=MDIDROW THEN GENDP(MDPRRMD)
08490 ELSE IF MDV.MDID=MDIDSTRUCT THEN
08500 BEGIN
08510 IF MDSTRSDB=0 THEN (*DBLOCK MUST BE CREATED*)
08520 BEGIN
08530 DATASTATE := STARTDATA; MDSTRSDB := FIXUPM;
08540 EMITXWORD(OCVIMMED, MDV.MDLEN);
08550 OFFSET := 0; DBLOCK(M);
08560 EMITXWORD(OCVIMMED, -1);
08570 DBLOCKM(M);
08580 END;
08590 GENDPOCV := OCVMEM; GENDPVAL :=MDSTRSDB
08600 END
08610 ELSE IF MDV.MDDRESSED THEN
08620 BEGIN GENDPVAL:=0; GENDPOCV:=OCVIMMED END
08630 ELSE
08640 BEGIN GENDPVAL := MDV.MDLEN; GENDPOCV := OCVIMMED END;
08650 END;
08660 (**)
08670 (**)
08680 ()+05*)
08690 (**)
08700 (**)
08710 (*-01() (*-02() (*-05()
08720 (*MODEL EMITBEG AND EMITEND FOR THOSE WHO HAVE NOT WRITTEN THEIR OWN YET*)
08730 PROCEDURE EMITBEG;
08740 BEGIN
08750 NEXTLABEL := 1;
08760 REWRITE(LGO);
08770 (*NOW INITIALIZE YOUR CODE BUFFER, OR WHATEVER, AND EMIT INIAL CODE*)
08780 END;
08790 (**)
08800 (**)
08810 PROCEDURE EMITEND;
08820 BEGIN
08830 (*EMIT YOUR FINAL CODE*)
08840 (*FLUSH YOUR CODE BUFFER, OR WHATEVER*)
08850 END;
08860 ()-05*) ()-02*) ()-01*)
08870 (**)
08880 (*-02() (*-05()
08890 (**)
08900 PROCEDURE GENDP(M: MODE);
08910 (*FUNCTION: RETURNS THE ADDRESS OF THE DBLOCK FOR MODE M, OR THE VALULENGTH,
08920 IN GLOBAL VARIABLE GENDPVAL, WITH CORRESPONDING OPDTYP IN GENDPOCV.
08930 *)
08940 VAR JUMPOVER: LABL;
08950 OFFSET: 0..127;
08960 PROCEDURE DBLOCK(M: MODE);
08970 VAR I, J: INTEGER;
08980 BEGIN WITH M^ DO
08990 FOR I := 0 TO MDV.MDCNT-1 DO
09000 WITH MDSTRFLDS[I] DO WITH MDSTRFMD^.MDV DO
09010 IF MDDRESSED THEN
09020 BEGIN EMITXWORD(OCVIMMED, OFFSET); OFFSET := OFFSET+MDLEN END
09030 ELSE IF MDID=MDIDSTRUCT THEN
09040 DBLOCK(MDSTRFMD)
09050 ELSE OFFSET := OFFSET+MDLEN
09060 END;
09070 PROCEDURE DBLOCKM(M: MODE);
09080 VAR I: INTEGER; X: XTYPE;
09090 BEGIN WITH M^ DO
09100 FOR I := 0 TO MDV.MDCNT-1 DO
09110 WITH MDSTRFLDS[I] DO
09120 BEGIN X := TX(MDSTRFMD);
09130 IF X=12 THEN DBLOCKM(MDSTRFMD)
09140 ELSE EMITXWORD(OCVIMMED, X+1)
09150 END
09160 END;
09170 BEGIN WITH M^ DO
09180 IF MDV.MDID=MDIDROW THEN GENDP(MDPRRMD)
09190 ELSE IF MDV.MDID=MDIDSTRUCT THEN
09200 BEGIN
09210 IF MDSTRSDB=0 THEN (*DBLOCK MUST BE CREATED*)
09220 BEGIN
09230 JUMPOVER := GETNEXTLABEL; EMITX1(PJMP, OCVFREF, JUMPOVER);
09240 MDSTRSDB := FIXUPM;
09250 EMITXWORD(OCVIMMED, MDV.MDLEN);
09260 OFFSET := 0; DBLOCK(M);
09270 EMITXWORD(OCVIMMED, -1);
09280 DBLOCKM(M);
09290 FIXUPF(JUMPOVER)
09300 END;
09310 GENDPOCV := OCVMEM; GENDPVAL :=MDSTRSDB
09320 END
09330 ELSE IF MDV.MDDRESSED THEN
09340 BEGIN GENDPVAL:=0; GENDPOCV:=OCVIMMED END
09350 ELSE
09360 BEGIN GENDPVAL := MDV.MDLEN; GENDPOCV := OCVIMMED END
09370 END;
09380 (**)
09390 ()-05*) ()-02*)
09400 (**)
09410 FUNCTION GETCASE(M: MODE; OLST: OLSTTYP; SB: PSB): STATE;
09420 (*FUNCTION: COMPUTES AN ADDITION TO SOME OPCOD.
09430 THE SB HERE AND IN RELATED PLACES IS A TEMPORARY KLUDGE ??????
09440 *)
09450 VAR WHICH: STATE;
09460 WEAKREF: BOOLEAN;
09470 BEGIN WITH M^ DO
09480 BEGIN
09490 IF SB<>NIL THEN WEAKREF:=(SBWEAKREF IN SB^.SBINF) ELSE WEAKREF:=FALSE;
09500 IF NOT MDV.MDPILE THEN
09510 IF MDV.MDLEN=SZINT THEN WHICH := 0 ELSE WHICH := 1
09520 ELSE IF WEAKREF THEN WHICH:=2
09530 ELSE IF MDV.MDID=MDIDROW THEN WHICH:=3
09540 ELSE IF MDV.MDDRESSED THEN WHICH:=4
09550 ELSE WHICH:=5;
09560 NEEDDP := OLST[WHICH].DP;
09570 GETCASE := OLST[WHICH].OVAL
09580 END
09590 END;
09600 (**)
09610 (**)
09620 PROCEDURE GENOP(VAR OPCOD: POP; M: MODE; VAR OLIST: OLSTTYP; SB: PSB);
09630 (*USES GETCASE TO MODIFY OPCOD AND DOES GENDP IF NECESSARY*)
09640 BEGIN
09650 OPCOD := OPCOD+GETCASE(M, OLIST, SB);
09660 IF NEEDDP THEN
09670 BEGIN
09680 IF SB<>NIL THEN
09690 IF SBWEAKREF IN SB^.SBINF THEN M := M^.MDPRRMD;
09700 GENDP(M);
09710 END
09720 ELSE BEGIN GENDPOCV:=OCVNONE; GENDPVAL:=0 END
09730 END;
09740 (**)
09750 (**)
09760 FUNCTION GENLCLGBL (*+05() (VAR OPCOD: POP; SB: PSB):INTEGER ()+05*) ;
09770 VAR I,X: INTEGER;
09780 VP : SBTTYP;
09790 BEGIN WITH SB^ DO
09800 BEGIN
09810 (*-05() GENLCLGBL:=SBOFFSET; ()-05*)
09820 (*+05() GENLCLGBL:=-SBOFFSET; ()+05*)
09830 IF (SBLEVEL = 0) (*+05() AND (SBLEVEL<>ROUTNL^.RNLEVEL) ()+05*) THEN (*GLOBAL*)
09840 BEGIN X:=1; (*-05() GENLCLGBL:=SBOFFSET+FIRSTIBOFFSET; ()-05*)
09850 END
09860 ELSE IF SBLEVEL = ROUTNL^.RNLEVEL THEN (*LOCAL*) X := 0
09870 ELSE (*INTERMEDIATE*) BEGIN
09880 (*-02() EMITX0(PENVCHAIN);
09890 FOR I:=1 TO ROUTNL^.RNLEVEL-SBLEVEL-1 DO
09900 BEGIN
09910 EMITX0(PENVCHAIN+1);
09920 END;
09930 ()-02*)
09940 (*+02() EMITX1(PENVCHAIN,OCVIMMED,ROUTNL^.RNLEVEL-SBLEVEL); ()+02*)
09950 X := 2 END;
09960 OPCOD := OPCOD+X;
09970 END
09980 END;
09990 (**)
10000 (**)
10010 (*-05()
10020 PROCEDURE GENDENOT (* (OPCOD: POP; SB: PSB) *) ;
10030 VAR THING: OBJECT; I: INTEGER;
10040 JUMPOVER: LABL;
10050 BEGIN WITH SB^ DO
10060 WITH SBLEX^ (*A LEXEME*) DO
10070 IF SBMODE^.MDV.MDID IN [MDIDCHAN,MDIDPASC] THEN
10080 EMITX1(OPCOD, OCVEXT, ORD(SBLEX))
10090 ELSE IF SBLEX=LEXFALSE THEN
10100 EMITX1(OPCOD, OCVIMMED, 0)
10110 ELSE IF ((LXDENMD=MDINT) OR (LXDENMD=MDBITS) OR (LXDENMD=MDCHAR))
10120 (*+01() AND (LXDENRP<400000B) ()+01*) AND (LXTOKEN=TKDENOT) THEN
10130 EMITX1(OPCOD, OCVIMMED, LXDENRP)
10140 ELSE
10150 BEGIN
10160 IF LXV.LXPYPTR=0 THEN
10170 BEGIN
10180 JUMPOVER := GETNEXTLABEL; EMITX1(PJMP, OCVFREF, JUMPOVER);
10190 LXV.LXPYPTR := FIXUPM;
10200 IF SBLEX=LEXTRUE THEN
10210 EMITXWORD(OCVIMMED, TRUEVALUE)
10220 ELSE IF LXDENMD^.MDV.MDPILE THEN WITH THING DO
10230 BEGIN
10240 FIRSTWORD := 0; PCOUNT := 255;
10250 LENGTH := (*-04() LXDENRP; ()-04*)(*+04() SHRINK(LXDENRP); ()+04*)
10260 EMITXWORD(OCVIMMED, FIRSTWORD);
10270 FOR I := 3 TO LXCOUNT DO
10280 EMITXWORD(OCVIMMED, INTEGERS[I])
10290 END
10300 ELSE EMITXWORD(OCVIMMED, LXDENRP);
10310 FIXUPF(JUMPOVER)
10320 END;
10330 IF LXTOKEN=TKDENOT THEN (*NOT LEXTRUE*)
10340 IF LXDENMD^.MDV.MDPILE THEN OPCOD := OPCOD-1;
10350 EMITX1(OPCOD+1, OCVMEM, LXV.LXPYPTR)
10360 END
10370 END;
10380 ()-05*)
~>
####S