⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 lab.c

📁 早期freebsd实现
💻 C
字号:
/*- * Copyright (c) 1980, 1993 *	The Regents of the University of California.  All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright *    notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright *    notice, this list of conditions and the following disclaimer in the *    documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software *    must display the following acknowledgement: *	This product includes software developed by the University of *	California, Berkeley and its contributors. * 4. Neither the name of the University nor the names of its contributors *    may be used to endorse or promote products derived from this software *    without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */#ifndef lintstatic char sccsid[] = "@(#)lab.c	8.1 (Berkeley) 6/6/93";#endif /* not lint */#include "whoami.h"#include "0.h"#include "tree.h"#include "opcode.h"#include "objfmt.h"#ifdef PC#   include	"pc.h"#   include	<pcc.h>#endif PC#include "tree_ty.h"/* * Label enters the definitions * of the label declaration part * into the namelist. */label(r, l)	struct tnode *r;	int l;{    static bool	label_order = FALSE;    static bool	label_seen = FALSE;#ifdef PC	char	extname[ BUFSIZ ];#endif PC#ifndef PI0	register struct tnode *ll;	register struct nl *p, *lp;	lp = NIL;#else	send(REVLAB, r);#endif	if ( ! progseen ) {	    level1();	}	line = l;#ifndef PI1	if (parts[ cbn ] & (CPRT|TPRT|VPRT|RPRT)){	    if ( opt( 's' ) ) {		standard();		error("Label declarations should precede const, type, var and routine declarations");	    } else {		if ( !label_order ) {		    label_order = TRUE;		    warning();		    error("Label declarations should precede const, type, var and routine declarations");		}	    }	}	if (parts[ cbn ] & LPRT) {	    if ( opt( 's' ) ) {		standard();		error("All labels should be declared in one label part");	    } else {		if ( !label_seen ) {		    label_seen = TRUE;		    warning();		    error("All labels should be declared in one label part");		}	    }	}	parts[ cbn ] |= LPRT;#endif#ifndef PI0	for (ll = r; ll != TR_NIL; ll = ll->list_node.next) {		l = (int) getlab();		p = enter(defnl((char *) ll->list_node.list, LABEL, NLNIL,				(int) l));		/*		 * Get the label for the eventual target		 */		p->value[1] = (int) getlab();		p->chain = lp;		p->nl_flags |= (NFORWD|NMOD);		p->value[NL_GOLEV] = NOTYET;		p->value[NL_ENTLOC] = l;		lp = p;#		ifdef OBJ		    /*		     * This operator is between		     * the bodies of two procedures		     * and provides a target for		     * gotos for this label via TRA.		     */		    (void) putlab((char *) l);		    (void) put(2, O_GOTO | cbn<<8, (long)p->value[1]);#		endif OBJ#		ifdef PC		    /*		     *	labels have to be .globl otherwise /lib/c2 may		     *	throw them away if they aren't used in the function		     *	which defines them.		     */		    extlabname( extname , p -> symbol , cbn );		    putprintf("	.globl	%s", 0, (int) extname);		    if ( cbn == 1 ) {			stabglabel( extname , line );		    }#		endif PC	}	gotos[cbn] = lp;#	ifdef PTREE	    {		pPointer	Labels = LabelDCopy( r );		pDEF( PorFHeader[ nesting ] ).PorFLabels = Labels;	    }#	endif PTREE#endif}#ifndef PI0/* * Gotoop is called when * we get a statement "goto label" * and generates the needed tra. */gotoop(s)	char *s;{	register struct nl *p;#ifdef PC	char	extname[ BUFSIZ ];#endif PC	gocnt++;	p = lookup(s);	if (p == NIL)		return;#	ifdef OBJ	    (void) put(2, O_TRA4, (long)p->value[NL_ENTLOC]);#	endif OBJ#	ifdef PC	    if ( cbn == bn ) {		    /*		     *	local goto.		     */		extlabname( extname , p -> symbol , bn );		    /*		     * this is a funny jump because it's to a label that		     * has been declared global.		     * Although this branch is within this module		     * the assembler will complain that the destination		     * is a global symbol.		     * The complaint arises because the assembler		     * doesn't change relative jumps into absolute jumps.		     * and this  may cause a branch displacement overflow		     * when the module is subsequently linked with		     * the rest of the program.		     */#		if defined(vax) || defined(tahoe)		    putprintf("	jmp	%s", 0, (int) extname);#		endif vax || tahoe#		ifdef mc68000		    putprintf("	jra	%s", 0, (int) extname);#		endif mc68000	    } else {		    /*		     *	Non-local goto.		     *		     *  Close all active files between top of stack and		     *  frame at the destination level.	Then call longjmp		     *	to unwind the stack to the destination level.		     *		     *	For nested routines the end of the frame		     *	is calculated as:		     *	    __disply[bn].fp + sizeof(local frame)		     *	(adjusted by (sizeof int) to get just past the end).		     *	The size of the local frame is dumped out by		     *	the second pass as an assembler constant.		     *	The main routine may not be compiled in this		     *	module, so its size may not be available.		     * 	However all of its variables will be globally		     *	declared, so only the known runtime temporaries		     *	will be in its stack frame.		     */		parts[ bn ] |= NONLOCALGOTO;		putleaf( PCC_ICON , 0 , 0 , PCCM_ADDTYPE( PCCTM_FTN | PCCT_INT , PCCTM_PTR )			, "_PCLOSE" );		if ( bn > 1 ) {		    p = lookup( enclosing[ bn - 1 ] );		    sprintf( extname, "%s%d+%d",			FRAME_SIZE_LABEL, p -> value[NL_ENTLOC], sizeof(int));		    p = lookup(s);		    putLV( extname , bn , 0 , NNLOCAL , PCCTM_PTR | PCCT_CHAR );		} else {		    putLV((char *) 0 , bn , -( DPOFF1 + sizeof( int ) ) , LOCALVAR ,			PCCTM_PTR | PCCT_CHAR );		}		putop( PCC_CALL , PCCT_INT );		putdot( filename , line );		putleaf( PCC_ICON , 0 , 0 , PCCM_ADDTYPE( PCCTM_FTN | PCCT_INT , PCCTM_PTR )			, "_longjmp" );		putLV((char *) 0 , bn , GOTOENVOFFSET , NLOCAL , PCCTM_PTR|PCCT_STRTY );		extlabname( extname , p -> symbol , bn );		putLV( extname , 0 , 0 , NGLOBAL , PCCTM_PTR|PCCT_STRTY );		putop( PCC_CM , PCCT_INT );		putop( PCC_CALL , PCCT_INT );		putdot( filename , line );	    }#	endif PC	if (bn == cbn)		if (p->nl_flags & NFORWD) {			if (p->value[NL_GOLEV] == NOTYET) {				p->value[NL_GOLEV] = level;				p->value[NL_GOLINE] = line;			}		} else			if (p->value[NL_GOLEV] == DEAD) {				recovered();				error("Goto %s is into a structured statement", p->symbol);			}}/* * Labeled is called when a label * definition is encountered, and * marks that it has been found and * patches the associated GOTO generated * by gotoop. */labeled(s)	char *s;{	register struct nl *p;#ifdef PC	char	extname[ BUFSIZ ];#endif PC	p = lookup(s);	if (p == NIL)		return;	if (bn != cbn) {		error("Label %s not defined in correct block", s);		return;	}	if ((p->nl_flags & NFORWD) == 0) {		error("Label %s redefined", s);		return;	}	p->nl_flags &= ~NFORWD;#	ifdef OBJ	    patch4((PTR_DCL) p->value[NL_ENTLOC]);#	endif OBJ#	ifdef PC	    extlabname( extname , p -> symbol , bn );	    putprintf( "%s:" , 0 , (int) extname );#	endif PC	if (p->value[NL_GOLEV] != NOTYET)		if (p->value[NL_GOLEV] < level) {			recovered();			error("Goto %s from line %d is into a structured statement", s, (char *) p->value[NL_GOLINE]);		}	p->value[NL_GOLEV] = level;}#endif#ifdef PC    /*     *	construct the long name of a label based on it's static nesting.     *	into a caller-supplied buffer (that should be about BUFSIZ big).     */extlabname( buffer , name , level )    char	buffer[];    char	*name;    int		level;{    char	*starthere;    int		i;    starthere = &buffer[0];    for ( i = 1 ; i < level ; i++ ) {	sprintf( starthere , EXTFORMAT , enclosing[ i ] );	starthere += strlen( enclosing[ i ] ) + 1;    }    sprintf( starthere , EXTFORMAT , "" );    starthere += 1;    sprintf( starthere , LABELFORMAT , name );    starthere += strlen( name ) + 1;    if ( starthere >= &buffer[ BUFSIZ ] ) {	panic( "extlabname" );    }}#endif PC

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -