xref: /csrg-svn/usr.bin/pascal/src/lab.c (revision 3074)
1756Speter /* Copyright (c) 1979 Regents of the University of California */
2756Speter 
3*3074Smckusic static char sccsid[] = "@(#)lab.c 1.7 03/08/81";
4756Speter 
5756Speter #include "whoami.h"
6756Speter #include "0.h"
7756Speter #include "tree.h"
8756Speter #include "opcode.h"
9756Speter #include "objfmt.h"
10756Speter #ifdef PC
11756Speter #   include	"pc.h"
12756Speter #   include	"pcops.h"
13756Speter #endif PC
14756Speter 
15756Speter /*
16756Speter  * Label enters the definitions
17756Speter  * of the label declaration part
18756Speter  * into the namelist.
19756Speter  */
20756Speter label(r, l)
21756Speter 	int *r, l;
22756Speter {
23756Speter #ifndef PI0
24756Speter 	register *ll;
25756Speter 	register struct nl *p, *lp;
26756Speter 
27756Speter 	lp = NIL;
28756Speter #else
29756Speter 	send(REVLAB, r);
30756Speter #endif
31756Speter 	if ( ! progseen ) {
32756Speter 	    level1();
33756Speter 	}
34756Speter 	line = l;
35756Speter #ifndef PI1
36835Speter 	if (parts[ cbn ] & (CPRT|TPRT|VPRT|RPRT)){
37835Speter 	    if ( opt( 's' ) ) {
38756Speter 		standard();
39835Speter 	    } else {
40835Speter 		warning();
41835Speter 	    }
42835Speter 	    error("Label declarations should precede const, type, var and routine declarations");
43756Speter 	}
44835Speter 	if (parts[ cbn ] & LPRT) {
45835Speter 	    if ( opt( 's' ) ) {
46756Speter 		standard();
47835Speter 	    } else {
48835Speter 		warning();
49835Speter 	    }
50835Speter 	    error("All labels should be declared in one label part");
51756Speter 	}
52835Speter 	parts[ cbn ] |= LPRT;
53756Speter #endif
54756Speter #ifndef PI0
55756Speter 	for (ll = r; ll != NIL; ll = ll[2]) {
56756Speter 		l = getlab();
57756Speter 		p = enter(defnl(ll[1], LABEL, 0, l));
58756Speter 		/*
59756Speter 		 * Get the label for the eventual target
60756Speter 		 */
61756Speter 		p->value[1] = getlab();
62756Speter 		p->chain = lp;
63756Speter 		p->nl_flags |= (NFORWD|NMOD);
64756Speter 		p->value[NL_GOLEV] = NOTYET;
65756Speter 		p->entloc = l;
66756Speter 		lp = p;
67756Speter #		ifdef OBJ
68756Speter 		    /*
69756Speter 		     * This operator is between
70756Speter 		     * the bodies of two procedures
71756Speter 		     * and provides a target for
72756Speter 		     * gotos for this label via TRA.
73756Speter 		     */
74756Speter 		    putlab(l);
75*3074Smckusic 		    /* put(2, O_GOTO | cbn<<8+INDX, (long)p->value[1]); */
76*3074Smckusic 		    put(2, O_GOTO | cbn<<8, (long)p->value[1]);
77756Speter #		endif OBJ
78756Speter #		ifdef PC
79756Speter 		    /*
80756Speter 		     *	labels have to be .globl otherwise /lib/c2 may
81756Speter 		     *	throw them away if they aren't used in the function
82756Speter 		     *	which defines them.
83756Speter 		     */
841430Speter 		    {
851430Speter 			char	extname[ BUFSIZ ];
861430Speter 			char	*starthere;
871430Speter 			int	i;
881430Speter 
891430Speter 			starthere = &extname[0];
901430Speter 			for ( i = 1 ; i < cbn ; i++ ) {
911430Speter 			    sprintf( starthere , EXTFORMAT , enclosing[ i ] );
921430Speter 			    starthere += strlen( enclosing[ i ] ) + 1;
931430Speter 			}
941430Speter 			sprintf( starthere , EXTFORMAT , p -> symbol );
951430Speter 			starthere += strlen( p -> symbol ) + 1;
961430Speter 			if ( starthere >= &extname[ BUFSIZ ] ) {
971430Speter 			    panic( "lab decl namelength" );
981430Speter 			}
991430Speter 			putprintf( "	.globl	" , 1 );
1001430Speter 			putprintf( NAMEFORMAT , 0 , extname );
1011430Speter 			if ( cbn == 1 ) {
1021430Speter 			    stabglabel( extname , line );
1031430Speter 			}
104756Speter 		    }
105756Speter #		endif PC
106756Speter 	}
107756Speter 	gotos[cbn] = lp;
108756Speter #	ifdef PTREE
109756Speter 	    {
110756Speter 		pPointer	Labels = LabelDCopy( r );
111756Speter 
112756Speter 		pDEF( PorFHeader[ nesting ] ).PorFLabels = Labels;
113756Speter 	    }
114756Speter #	endif PTREE
115756Speter #endif
116756Speter }
117756Speter 
118756Speter #ifndef PI0
119756Speter /*
120756Speter  * Gotoop is called when
121756Speter  * we get a statement "goto label"
122756Speter  * and generates the needed tra.
123756Speter  */
124756Speter gotoop(s)
125756Speter 	char *s;
126756Speter {
127756Speter 	register struct nl *p;
128756Speter 
129756Speter 	gocnt++;
130756Speter 	p = lookup(s);
131756Speter 	if (p == NIL)
132756Speter 		return (NIL);
133756Speter #	ifdef OBJ
134*3074Smckusic 	    put(2, O_TRA4, (long)p->entloc);
135756Speter #	endif OBJ
136756Speter #	ifdef PC
137756Speter 	    if ( cbn != bn ) {
138756Speter 		    /*
139756Speter 		     *	call goto to unwind the stack to the destination level
140756Speter 		     */
141756Speter 		putleaf( P2ICON , 0 , 0 , ADDTYPE( P2FTN | P2INT , P2PTR )
142756Speter 			, "_GOTO" );
143756Speter 		putLV( DISPLAYNAME , 0 , bn * sizeof( struct dispsave )
144756Speter 			, P2PTR | P2INT );
145756Speter 		putop( P2CALL , P2INT );
146756Speter 		putdot( filename , line );
147756Speter 	    }
1481430Speter 	    {
1491430Speter 		char	extname[ BUFSIZ ];
1501430Speter 		char	*starthere;
1511430Speter 		int	i;
1521430Speter 
1531430Speter 		starthere = &extname[0];
1541430Speter 		for ( i = 1 ; i < bn ; i++ ) {
1551430Speter 		    sprintf( starthere , EXTFORMAT , enclosing[ i ] );
1561430Speter 		    starthere += strlen( enclosing[ i ] ) + 1;
1571430Speter 		}
1581430Speter 		sprintf( starthere , EXTFORMAT , p -> symbol );
1591430Speter 		starthere += strlen( p -> symbol ) + 1;
1601430Speter 		if ( starthere >= &extname[ BUFSIZ ] ) {
1611430Speter 		    panic( "goto namelength" );
1621430Speter 		}
1631430Speter 		putprintf( "	jbr	" , 1 );
1641430Speter 		putprintf( NAMEFORMAT , 0 , extname );
165756Speter 	    }
166756Speter #	endif PC
167756Speter 	if (bn == cbn)
168756Speter 		if (p->nl_flags & NFORWD) {
169756Speter 			if (p->value[NL_GOLEV] == NOTYET) {
170756Speter 				p->value[NL_GOLEV] = level;
171756Speter 				p->value[NL_GOLINE] = line;
172756Speter 			}
173756Speter 		} else
174756Speter 			if (p->value[NL_GOLEV] == DEAD) {
175756Speter 				recovered();
176756Speter 				error("Goto %s is into a structured statement", p->symbol);
177756Speter 			}
178756Speter }
179756Speter 
180756Speter /*
181756Speter  * Labeled is called when a label
182756Speter  * definition is encountered, and
183756Speter  * marks that it has been found and
184756Speter  * patches the associated GOTO generated
185756Speter  * by gotoop.
186756Speter  */
187756Speter labeled(s)
188756Speter 	char *s;
189756Speter {
190756Speter 	register struct nl *p;
191756Speter 
192756Speter 	p = lookup(s);
193756Speter 	if (p == NIL)
194756Speter 		return (NIL);
195756Speter 	if (bn != cbn) {
196756Speter 		error("Label %s not defined in correct block", s);
197756Speter 		return;
198756Speter 	}
199756Speter 	if ((p->nl_flags & NFORWD) == 0) {
200756Speter 		error("Label %s redefined", s);
201756Speter 		return;
202756Speter 	}
203756Speter 	p->nl_flags &= ~NFORWD;
204756Speter #	ifdef OBJ
205756Speter 	    patch4(p->entloc);
206756Speter #	endif OBJ
207756Speter #	ifdef PC
2081430Speter 	    {
2091430Speter 		char	extname[ BUFSIZ ];
2101430Speter 		char	*starthere;
2111430Speter 		int	i;
2121430Speter 
2131430Speter 		starthere = &extname[0];
2141430Speter 		for ( i = 1 ; i < bn ; i++ ) {
2151430Speter 		    sprintf( starthere , EXTFORMAT , enclosing[ i ] );
2161430Speter 		    starthere += strlen( enclosing[ i ] ) + 1;
2171430Speter 		}
2181430Speter 		sprintf( starthere , EXTFORMAT , p -> symbol );
2191430Speter 		starthere += strlen( p -> symbol ) + 1;
2201430Speter 		if ( starthere >= &extname[ BUFSIZ ] ) {
2211430Speter 		    panic( "labeled namelength" );
2221430Speter 		}
2231430Speter 		putprintf( NAMEFORMAT , 1 , extname );
2241430Speter 		putprintf( ":" , 0 );
225756Speter 	    }
226756Speter #	endif PC
227756Speter 	if (p->value[NL_GOLEV] != NOTYET)
228756Speter 		if (p->value[NL_GOLEV] < level) {
229756Speter 			recovered();
230756Speter 			error("Goto %s from line %d is into a structured statement", s, p->value[NL_GOLINE]);
231756Speter 		}
232756Speter 	p->value[NL_GOLEV] = level;
233756Speter }
234756Speter #endif
235