1756Speter /* Copyright (c) 1979 Regents of the University of California */ 2756Speter 3*835Speter static char sccsid[] = "@(#)lab.c 1.3 09/02/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 36*835Speter if (parts[ cbn ] & (CPRT|TPRT|VPRT|RPRT)){ 37*835Speter if ( opt( 's' ) ) { 38756Speter standard(); 39*835Speter } else { 40*835Speter warning(); 41*835Speter } 42*835Speter error("Label declarations should precede const, type, var and routine declarations"); 43756Speter } 44*835Speter if (parts[ cbn ] & LPRT) { 45*835Speter if ( opt( 's' ) ) { 46756Speter standard(); 47*835Speter } else { 48*835Speter warning(); 49*835Speter } 50*835Speter error("All labels should be declared in one label part"); 51756Speter } 52*835Speter 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 */ 83756Speter if (cbn == 1) { 84756Speter /* 85756Speter * stab the label for separate compilation. 86756Speter * make label number = label name. 87756Speter */ 88826Speter stabglab( p -> symbol , line ); 89756Speter p -> value[1] = atol( p -> symbol ); 90756Speter putprintf( " .globl " , 1 ); 91756Speter putprintf( PREFIXFORMAT , 0 , PLABELPREFIX 92756Speter , p -> value[1] ); 93756Speter } else { 94756Speter putprintf( " .globl " , 1 ); 95756Speter putprintf( PREFIXFORMAT , 0 , GLABELPREFIX 96756Speter , p -> value[1] ); 97756Speter } 98756Speter # endif PC 99756Speter } 100756Speter gotos[cbn] = lp; 101756Speter # ifdef PTREE 102756Speter { 103756Speter pPointer Labels = LabelDCopy( r ); 104756Speter 105756Speter pDEF( PorFHeader[ nesting ] ).PorFLabels = Labels; 106756Speter } 107756Speter # endif PTREE 108756Speter #endif 109756Speter } 110756Speter 111756Speter #ifndef PI0 112756Speter /* 113756Speter * Gotoop is called when 114756Speter * we get a statement "goto label" 115756Speter * and generates the needed tra. 116756Speter */ 117756Speter gotoop(s) 118756Speter char *s; 119756Speter { 120756Speter register struct nl *p; 121756Speter 122756Speter gocnt++; 123756Speter p = lookup(s); 124756Speter if (p == NIL) 125756Speter return (NIL); 126756Speter # ifdef OBJ 127756Speter put2(O_TRA4, p->entloc); 128756Speter # endif OBJ 129756Speter # ifdef PC 130756Speter if ( cbn != bn ) { 131756Speter /* 132756Speter * call goto to unwind the stack to the destination level 133756Speter */ 134756Speter putleaf( P2ICON , 0 , 0 , ADDTYPE( P2FTN | P2INT , P2PTR ) 135756Speter , "_GOTO" ); 136756Speter putLV( DISPLAYNAME , 0 , bn * sizeof( struct dispsave ) 137756Speter , P2PTR | P2INT ); 138756Speter putop( P2CALL , P2INT ); 139756Speter putdot( filename , line ); 140756Speter } 141756Speter if ( bn <= 1 ) { 142756Speter printjbr( PLABELPREFIX , p -> value[1] ); 143756Speter } else { 144756Speter printjbr( GLABELPREFIX , p -> value[1] ); 145756Speter } 146756Speter # endif PC 147756Speter if (bn == cbn) 148756Speter if (p->nl_flags & NFORWD) { 149756Speter if (p->value[NL_GOLEV] == NOTYET) { 150756Speter p->value[NL_GOLEV] = level; 151756Speter p->value[NL_GOLINE] = line; 152756Speter } 153756Speter } else 154756Speter if (p->value[NL_GOLEV] == DEAD) { 155756Speter recovered(); 156756Speter error("Goto %s is into a structured statement", p->symbol); 157756Speter } 158756Speter } 159756Speter 160756Speter /* 161756Speter * Labeled is called when a label 162756Speter * definition is encountered, and 163756Speter * marks that it has been found and 164756Speter * patches the associated GOTO generated 165756Speter * by gotoop. 166756Speter */ 167756Speter labeled(s) 168756Speter char *s; 169756Speter { 170756Speter register struct nl *p; 171756Speter 172756Speter p = lookup(s); 173756Speter if (p == NIL) 174756Speter return (NIL); 175756Speter if (bn != cbn) { 176756Speter error("Label %s not defined in correct block", s); 177756Speter return; 178756Speter } 179756Speter if ((p->nl_flags & NFORWD) == 0) { 180756Speter error("Label %s redefined", s); 181756Speter return; 182756Speter } 183756Speter p->nl_flags &= ~NFORWD; 184756Speter # ifdef OBJ 185756Speter patch4(p->entloc); 186756Speter # endif OBJ 187756Speter # ifdef PC 188756Speter if ( bn <= 1 ) { 189756Speter putprintf( PREFIXFORMAT , 1 , PLABELPREFIX , p -> value[1] ); 190756Speter } else { 191756Speter putprintf( PREFIXFORMAT , 1 , GLABELPREFIX , p -> value[1] ); 192756Speter } 193756Speter putprintf( ":" , 0 ); 194756Speter # endif PC 195756Speter if (p->value[NL_GOLEV] != NOTYET) 196756Speter if (p->value[NL_GOLEV] < level) { 197756Speter recovered(); 198756Speter error("Goto %s from line %d is into a structured statement", s, p->value[NL_GOLINE]); 199756Speter } 200756Speter p->value[NL_GOLEV] = level; 201756Speter } 202756Speter #endif 203