1756Speter /* Copyright (c) 1979 Regents of the University of California */ 2756Speter 3*1430Speter static char sccsid[] = "@(#)lab.c 1.5 10/14/80"; 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); 75756Speter put2(O_GOTO | cbn<<8+INDX, p->value[1]); 76756Speter # endif OBJ 77756Speter # ifdef PC 78756Speter /* 79756Speter * labels have to be .globl otherwise /lib/c2 may 80756Speter * throw them away if they aren't used in the function 81756Speter * which defines them. 82756Speter */ 83*1430Speter { 84*1430Speter char extname[ BUFSIZ ]; 85*1430Speter char *starthere; 86*1430Speter int i; 87*1430Speter 88*1430Speter starthere = &extname[0]; 89*1430Speter for ( i = 1 ; i < cbn ; i++ ) { 90*1430Speter sprintf( starthere , EXTFORMAT , enclosing[ i ] ); 91*1430Speter starthere += strlen( enclosing[ i ] ) + 1; 92*1430Speter } 93*1430Speter sprintf( starthere , EXTFORMAT , p -> symbol ); 94*1430Speter starthere += strlen( p -> symbol ) + 1; 95*1430Speter if ( starthere >= &extname[ BUFSIZ ] ) { 96*1430Speter panic( "lab decl namelength" ); 97*1430Speter } 98*1430Speter putprintf( " .globl " , 1 ); 99*1430Speter putprintf( NAMEFORMAT , 0 , extname ); 100*1430Speter if ( cbn == 1 ) { 101*1430Speter stabglabel( extname , line ); 102*1430Speter } 103756Speter } 104756Speter # endif PC 105756Speter } 106756Speter gotos[cbn] = lp; 107756Speter # ifdef PTREE 108756Speter { 109756Speter pPointer Labels = LabelDCopy( r ); 110756Speter 111756Speter pDEF( PorFHeader[ nesting ] ).PorFLabels = Labels; 112756Speter } 113756Speter # endif PTREE 114756Speter #endif 115756Speter } 116756Speter 117756Speter #ifndef PI0 118756Speter /* 119756Speter * Gotoop is called when 120756Speter * we get a statement "goto label" 121756Speter * and generates the needed tra. 122756Speter */ 123756Speter gotoop(s) 124756Speter char *s; 125756Speter { 126756Speter register struct nl *p; 127756Speter 128756Speter gocnt++; 129756Speter p = lookup(s); 130756Speter if (p == NIL) 131756Speter return (NIL); 132756Speter # ifdef OBJ 133756Speter put2(O_TRA4, p->entloc); 134756Speter # endif OBJ 135756Speter # ifdef PC 136756Speter if ( cbn != bn ) { 137756Speter /* 138756Speter * call goto to unwind the stack to the destination level 139756Speter */ 140756Speter putleaf( P2ICON , 0 , 0 , ADDTYPE( P2FTN | P2INT , P2PTR ) 141756Speter , "_GOTO" ); 142756Speter putLV( DISPLAYNAME , 0 , bn * sizeof( struct dispsave ) 143756Speter , P2PTR | P2INT ); 144756Speter putop( P2CALL , P2INT ); 145756Speter putdot( filename , line ); 146756Speter } 147*1430Speter { 148*1430Speter char extname[ BUFSIZ ]; 149*1430Speter char *starthere; 150*1430Speter int i; 151*1430Speter 152*1430Speter starthere = &extname[0]; 153*1430Speter for ( i = 1 ; i < bn ; i++ ) { 154*1430Speter sprintf( starthere , EXTFORMAT , enclosing[ i ] ); 155*1430Speter starthere += strlen( enclosing[ i ] ) + 1; 156*1430Speter } 157*1430Speter sprintf( starthere , EXTFORMAT , p -> symbol ); 158*1430Speter starthere += strlen( p -> symbol ) + 1; 159*1430Speter if ( starthere >= &extname[ BUFSIZ ] ) { 160*1430Speter panic( "goto namelength" ); 161*1430Speter } 162*1430Speter putprintf( " jbr " , 1 ); 163*1430Speter putprintf( NAMEFORMAT , 0 , extname ); 164756Speter } 165756Speter # endif PC 166756Speter if (bn == cbn) 167756Speter if (p->nl_flags & NFORWD) { 168756Speter if (p->value[NL_GOLEV] == NOTYET) { 169756Speter p->value[NL_GOLEV] = level; 170756Speter p->value[NL_GOLINE] = line; 171756Speter } 172756Speter } else 173756Speter if (p->value[NL_GOLEV] == DEAD) { 174756Speter recovered(); 175756Speter error("Goto %s is into a structured statement", p->symbol); 176756Speter } 177756Speter } 178756Speter 179756Speter /* 180756Speter * Labeled is called when a label 181756Speter * definition is encountered, and 182756Speter * marks that it has been found and 183756Speter * patches the associated GOTO generated 184756Speter * by gotoop. 185756Speter */ 186756Speter labeled(s) 187756Speter char *s; 188756Speter { 189756Speter register struct nl *p; 190756Speter 191756Speter p = lookup(s); 192756Speter if (p == NIL) 193756Speter return (NIL); 194756Speter if (bn != cbn) { 195756Speter error("Label %s not defined in correct block", s); 196756Speter return; 197756Speter } 198756Speter if ((p->nl_flags & NFORWD) == 0) { 199756Speter error("Label %s redefined", s); 200756Speter return; 201756Speter } 202756Speter p->nl_flags &= ~NFORWD; 203756Speter # ifdef OBJ 204756Speter patch4(p->entloc); 205756Speter # endif OBJ 206756Speter # ifdef PC 207*1430Speter { 208*1430Speter char extname[ BUFSIZ ]; 209*1430Speter char *starthere; 210*1430Speter int i; 211*1430Speter 212*1430Speter starthere = &extname[0]; 213*1430Speter for ( i = 1 ; i < bn ; i++ ) { 214*1430Speter sprintf( starthere , EXTFORMAT , enclosing[ i ] ); 215*1430Speter starthere += strlen( enclosing[ i ] ) + 1; 216*1430Speter } 217*1430Speter sprintf( starthere , EXTFORMAT , p -> symbol ); 218*1430Speter starthere += strlen( p -> symbol ) + 1; 219*1430Speter if ( starthere >= &extname[ BUFSIZ ] ) { 220*1430Speter panic( "labeled namelength" ); 221*1430Speter } 222*1430Speter putprintf( NAMEFORMAT , 1 , extname ); 223*1430Speter putprintf( ":" , 0 ); 224756Speter } 225756Speter # endif PC 226756Speter if (p->value[NL_GOLEV] != NOTYET) 227756Speter if (p->value[NL_GOLEV] < level) { 228756Speter recovered(); 229756Speter error("Goto %s from line %d is into a structured statement", s, p->value[NL_GOLINE]); 230756Speter } 231756Speter p->value[NL_GOLEV] = level; 232756Speter } 233756Speter #endif 234