1756Speter /* Copyright (c) 1979 Regents of the University of California */ 2756Speter 3*7919Smckusick static char sccsid[] = "@(#)lab.c 1.12 08/26/82"; 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 { 233366Speter #ifdef PC 243366Speter char extname[ BUFSIZ ]; 253366Speter #endif PC 26756Speter #ifndef PI0 27756Speter register *ll; 28756Speter register struct nl *p, *lp; 29756Speter 30756Speter lp = NIL; 31756Speter #else 32756Speter send(REVLAB, r); 33756Speter #endif 34756Speter if ( ! progseen ) { 35756Speter level1(); 36756Speter } 37756Speter line = l; 38756Speter #ifndef PI1 39835Speter if (parts[ cbn ] & (CPRT|TPRT|VPRT|RPRT)){ 40835Speter if ( opt( 's' ) ) { 41756Speter standard(); 42835Speter } else { 43835Speter warning(); 44835Speter } 45835Speter error("Label declarations should precede const, type, var and routine declarations"); 46756Speter } 47835Speter if (parts[ cbn ] & LPRT) { 48835Speter if ( opt( 's' ) ) { 49756Speter standard(); 50835Speter } else { 51835Speter warning(); 52835Speter } 53835Speter error("All labels should be declared in one label part"); 54756Speter } 55835Speter parts[ cbn ] |= LPRT; 56756Speter #endif 57756Speter #ifndef PI0 58756Speter for (ll = r; ll != NIL; ll = ll[2]) { 59756Speter l = getlab(); 60756Speter p = enter(defnl(ll[1], LABEL, 0, l)); 61756Speter /* 62756Speter * Get the label for the eventual target 63756Speter */ 64756Speter p->value[1] = getlab(); 65756Speter p->chain = lp; 66756Speter p->nl_flags |= (NFORWD|NMOD); 67756Speter p->value[NL_GOLEV] = NOTYET; 68*7919Smckusick p->value[NL_ENTLOC] = l; 69756Speter lp = p; 70756Speter # ifdef OBJ 71756Speter /* 72756Speter * This operator is between 73756Speter * the bodies of two procedures 74756Speter * and provides a target for 75756Speter * gotos for this label via TRA. 76756Speter */ 77756Speter putlab(l); 783074Smckusic /* put(2, O_GOTO | cbn<<8+INDX, (long)p->value[1]); */ 793074Smckusic put(2, O_GOTO | cbn<<8, (long)p->value[1]); 80756Speter # endif OBJ 81756Speter # ifdef PC 82756Speter /* 83756Speter * labels have to be .globl otherwise /lib/c2 may 84756Speter * throw them away if they aren't used in the function 85756Speter * which defines them. 86756Speter */ 874880Speter extlabname( extname , p -> symbol , cbn ); 883366Speter putprintf( " .globl %s" , 0 , extname ); 893366Speter if ( cbn == 1 ) { 903366Speter stabglabel( extname , line ); 91756Speter } 92756Speter # endif PC 93756Speter } 94756Speter gotos[cbn] = lp; 95756Speter # ifdef PTREE 96756Speter { 97756Speter pPointer Labels = LabelDCopy( r ); 98756Speter 99756Speter pDEF( PorFHeader[ nesting ] ).PorFLabels = Labels; 100756Speter } 101756Speter # endif PTREE 102756Speter #endif 103756Speter } 104756Speter 105756Speter #ifndef PI0 106756Speter /* 107756Speter * Gotoop is called when 108756Speter * we get a statement "goto label" 109756Speter * and generates the needed tra. 110756Speter */ 111756Speter gotoop(s) 112756Speter char *s; 113756Speter { 114756Speter register struct nl *p; 1153366Speter #ifdef PC 1163366Speter char extname[ BUFSIZ ]; 1173366Speter #endif PC 118756Speter 119756Speter gocnt++; 120756Speter p = lookup(s); 121756Speter if (p == NIL) 122756Speter return (NIL); 123756Speter # ifdef OBJ 124*7919Smckusick put(2, O_TRA4, (long)p->value[NL_ENTLOC]); 125756Speter # endif OBJ 126756Speter # ifdef PC 127756Speter if ( cbn != bn ) { 128756Speter /* 129756Speter * call goto to unwind the stack to the destination level 130756Speter */ 131756Speter putleaf( P2ICON , 0 , 0 , ADDTYPE( P2FTN | P2INT , P2PTR ) 132756Speter , "_GOTO" ); 1333827Speter putLV( DISPLAYNAME , 0 , bn * sizeof( struct dispsave ) , 1343827Speter NGLOBAL , P2PTR | P2INT ); 135756Speter putop( P2CALL , P2INT ); 136756Speter putdot( filename , line ); 137756Speter } 1384880Speter extlabname( extname , p -> symbol , bn ); 1397449Speter /* 1407449Speter * this is a jmp because it's a jump to a global 1417449Speter * and the assembler doesn't change jbr's into jmp's 1427449Speter * if the destination is a global symbol. 1437449Speter */ 1447449Speter putprintf( " jmp %s" , 0 , extname ); 145756Speter # endif PC 146756Speter if (bn == cbn) 147756Speter if (p->nl_flags & NFORWD) { 148756Speter if (p->value[NL_GOLEV] == NOTYET) { 149756Speter p->value[NL_GOLEV] = level; 150756Speter p->value[NL_GOLINE] = line; 151756Speter } 152756Speter } else 153756Speter if (p->value[NL_GOLEV] == DEAD) { 154756Speter recovered(); 155756Speter error("Goto %s is into a structured statement", p->symbol); 156756Speter } 157756Speter } 158756Speter 159756Speter /* 160756Speter * Labeled is called when a label 161756Speter * definition is encountered, and 162756Speter * marks that it has been found and 163756Speter * patches the associated GOTO generated 164756Speter * by gotoop. 165756Speter */ 166756Speter labeled(s) 167756Speter char *s; 168756Speter { 169756Speter register struct nl *p; 1703366Speter #ifdef PC 1713366Speter char extname[ BUFSIZ ]; 1723366Speter #endif PC 173756Speter 174756Speter p = lookup(s); 175756Speter if (p == NIL) 176756Speter return (NIL); 177756Speter if (bn != cbn) { 178756Speter error("Label %s not defined in correct block", s); 179756Speter return; 180756Speter } 181756Speter if ((p->nl_flags & NFORWD) == 0) { 182756Speter error("Label %s redefined", s); 183756Speter return; 184756Speter } 185756Speter p->nl_flags &= ~NFORWD; 186756Speter # ifdef OBJ 187*7919Smckusick patch4(p->value[NL_ENTLOC]); 188756Speter # endif OBJ 189756Speter # ifdef PC 1904880Speter extlabname( extname , p -> symbol , bn ); 1913366Speter putprintf( "%s:" , 0 , extname ); 192756Speter # endif PC 193756Speter if (p->value[NL_GOLEV] != NOTYET) 194756Speter if (p->value[NL_GOLEV] < level) { 195756Speter recovered(); 196756Speter error("Goto %s from line %d is into a structured statement", s, p->value[NL_GOLINE]); 197756Speter } 198756Speter p->value[NL_GOLEV] = level; 199756Speter } 200756Speter #endif 2014880Speter 2024880Speter #ifdef PC 2034880Speter /* 2044880Speter * construct the long name of a label based on it's static nesting. 2054880Speter * into a caller-supplied buffer (that should be about BUFSIZ big). 2064880Speter */ 2074880Speter extlabname( buffer , name , level ) 2084880Speter char buffer[]; 2094880Speter char *name; 2104880Speter int level; 2114880Speter { 2124880Speter char *starthere; 2134880Speter int i; 2144880Speter 2154880Speter starthere = &buffer[0]; 2164880Speter for ( i = 1 ; i < level ; i++ ) { 2174880Speter sprintf( starthere , EXTFORMAT , enclosing[ i ] ); 2184880Speter starthere += strlen( enclosing[ i ] ) + 1; 2194880Speter } 2204880Speter sprintf( starthere , EXTFORMAT , "" ); 2214880Speter starthere += 1; 2224880Speter sprintf( starthere , LABELFORMAT , name ); 2234880Speter starthere += strlen( name ) + 1; 2244880Speter if ( starthere >= &buffer[ BUFSIZ ] ) { 2254880Speter panic( "extlabname" ); 2264880Speter } 2274880Speter } 2284880Speter #endif PC 229