358 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			358 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
 | 
						|
 * See the copyright notice in the ACK home directory, in the file "Copyright".
 | 
						|
 */
 | 
						|
/* $Header$ */
 | 
						|
/*	U S E R   O P T I O N - H A N D L I N G		*/
 | 
						|
 | 
						|
#include	"lint.h"
 | 
						|
#include	"botch_free.h"
 | 
						|
#include	<alloc.h>
 | 
						|
#include	"nofloat.h"
 | 
						|
#include	"nopp.h"
 | 
						|
#include	"idfsize.h"
 | 
						|
#include	"nobitfield.h"
 | 
						|
#include	"class.h"
 | 
						|
#include	"macro.h"
 | 
						|
#include	"idf.h"
 | 
						|
#include	"arith.h"
 | 
						|
#include	"sizes.h"
 | 
						|
#include	"align.h"
 | 
						|
#include	"use_tmp.h"
 | 
						|
#include	"dataflow.h"
 | 
						|
#include	"noRoption.h"
 | 
						|
 | 
						|
#ifndef NOPP
 | 
						|
extern char **inctable;
 | 
						|
extern int inc_pos;
 | 
						|
extern int inc_max;
 | 
						|
extern int inc_total;
 | 
						|
int do_dependencies = 0;
 | 
						|
char *dep_file = 0;
 | 
						|
 | 
						|
#endif NOPP
 | 
						|
 | 
						|
char options[128];			/* one for every char	*/
 | 
						|
#ifdef	LINT
 | 
						|
char loptions[128];			/* one for every char	*/
 | 
						|
#endif	LINT
 | 
						|
 | 
						|
extern int idfsize;
 | 
						|
 | 
						|
static int txt2int();
 | 
						|
 | 
						|
do_option(text)
 | 
						|
	char *text;
 | 
						|
{
 | 
						|
	register char opt;
 | 
						|
 | 
						|
next_option:			/* to allow combined one-char options */
 | 
						|
	switch (opt = *text++)	{
 | 
						|
 | 
						|
	case 0:			/* to end the goto next_option loop */
 | 
						|
		break;
 | 
						|
 | 
						|
	default:
 | 
						|
#ifndef	LINT
 | 
						|
		fatal("illegal option: %c", opt);
 | 
						|
#else	LINT
 | 
						|
		warning("illegal option: %c", opt);
 | 
						|
#endif	LINT
 | 
						|
		break;
 | 
						|
 | 
						|
	case '-':
 | 
						|
		options[*text++] = 1;	/* flags, debug options etc.	*/
 | 
						|
		goto next_option;
 | 
						|
 | 
						|
#ifndef	LINT
 | 
						|
#ifdef	DATAFLOW
 | 
						|
	case 'd':
 | 
						|
#endif	DATAFLOW
 | 
						|
	case 'p':			/* procentry/procexit */
 | 
						|
	case 'L' :			/* no fil/lin */
 | 
						|
	case 'n':			/* use no registers */
 | 
						|
	case 'w':			/* no warnings will be given */
 | 
						|
		options[opt] = 1;
 | 
						|
		goto next_option;
 | 
						|
#endif	LINT
 | 
						|
 | 
						|
#ifdef	LINT
 | 
						|
	case 'h':	/* heuristic tests */
 | 
						|
	case 'v':	/* no complaints about unused arguments */
 | 
						|
	case 'a':	/* check long->int int->long conversions */
 | 
						|
	case 'b':	/* don't report unreachable break-statements */
 | 
						|
	case 'x':	/* complain about unused extern declared variables */
 | 
						|
	case 'u':	/* no "used but not defined"; for pass 2 */
 | 
						|
	case 'L':	/* lintlibrary */
 | 
						|
		loptions[opt] = 1;
 | 
						|
		goto next_option;
 | 
						|
#endif	LINT
 | 
						|
 | 
						|
#ifndef LINT
 | 
						|
#ifndef NOPP
 | 
						|
	case 'A' :      /* Amake dependency generation */
 | 
						|
		do_dependencies = 1;
 | 
						|
		if (*text) {
 | 
						|
			dep_file = text;
 | 
						|
		}
 | 
						|
		break;
 | 
						|
	case 'i':
 | 
						|
	case 'm':
 | 
						|
		options[opt] = 1;
 | 
						|
		break;
 | 
						|
#endif NOPP
 | 
						|
#endif LINT
 | 
						|
		
 | 
						|
	case 'R':			/* strict version */
 | 
						|
#ifndef	NOROPTION
 | 
						|
		options[opt] = 1;
 | 
						|
#else	NOROPTION
 | 
						|
		warning("-R option not implemented");
 | 
						|
#endif	NOROPTION
 | 
						|
		goto next_option;
 | 
						|
 | 
						|
#ifdef	___XXX___
 | 
						|
deleted, is now a debug-flag
 | 
						|
	case 'C' :	/* E option + comment output		*/
 | 
						|
#ifndef	NOPP
 | 
						|
		options['E'] = 1;
 | 
						|
		warning("-C: comment is not output");
 | 
						|
#else NOPP
 | 
						|
		warning("-C option ignored");
 | 
						|
#endif	NOPP
 | 
						|
		break;
 | 
						|
#endif	___XXX___
 | 
						|
 | 
						|
	case 'D' :	{	/* -Dname :	predefine name		*/
 | 
						|
#ifndef NOPP
 | 
						|
		register char *cp = text, *name, *mactext;
 | 
						|
 | 
						|
		if (class(*cp) != STIDF)	{
 | 
						|
			error("identifier missing in -D%s", text);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		name = cp;
 | 
						|
 | 
						|
		while (*cp && in_idf(*cp)) {
 | 
						|
			++cp;
 | 
						|
		}
 | 
						|
 | 
						|
		if (!*cp) {			/* -Dname */
 | 
						|
			mactext = "1";
 | 
						|
		}
 | 
						|
		else
 | 
						|
		if (*cp == '=')	{		/* -Dname=text	*/
 | 
						|
			*cp++ = '\0';		/* end of name	*/
 | 
						|
			mactext = cp;
 | 
						|
		}
 | 
						|
		else	{			/* -Dname?? */
 | 
						|
			error("malformed option -D%s", text);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		macro_def(str2idf(name), mactext, -1, strlen(mactext),
 | 
						|
			NOFLAG);
 | 
						|
#else NOPP
 | 
						|
		warning("-D option ignored");
 | 
						|
#endif NOPP
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
#ifdef ___XXX___
 | 
						|
deleted, is now a debug-flag
 | 
						|
	case 'E' :	/* run preprocessor only, with #<int>	*/
 | 
						|
#ifndef NOPP
 | 
						|
		options['E'] = 1;
 | 
						|
#else NOPP
 | 
						|
		warning("-E option ignored");
 | 
						|
#endif NOPP
 | 
						|
		break;
 | 
						|
#endif ___XXX___
 | 
						|
 | 
						|
	case 'I' :	/* -Ipath : insert "path" into include list	*/
 | 
						|
#ifndef NOPP
 | 
						|
		if (*text)	{
 | 
						|
			int i;
 | 
						|
			register char *new = text;
 | 
						|
			
 | 
						|
			if (++inc_total > inc_max) {
 | 
						|
				inctable = (char **)
 | 
						|
				  Realloc(inctable,(inc_max+=10)*sizeof(char *));
 | 
						|
			}
 | 
						|
				
 | 
						|
			for (i = inc_pos++; i < inc_total; i++) {
 | 
						|
				char *tmp = inctable[i];
 | 
						|
				
 | 
						|
				inctable[i] = new;
 | 
						|
				new = tmp;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else inctable[inc_pos] = 0;
 | 
						|
#else NOPP
 | 
						|
		warning("-I option ignored");
 | 
						|
#endif NOPP
 | 
						|
		break;
 | 
						|
 | 
						|
	case 'M':	/* maximum identifier length */
 | 
						|
		idfsize = txt2int(&text);
 | 
						|
		if (*text || idfsize <= 0)
 | 
						|
			fatal("malformed -M option");
 | 
						|
		if (idfsize > IDFSIZE)
 | 
						|
			fatal("maximum identifier length is %d", IDFSIZE);
 | 
						|
		break;
 | 
						|
 | 
						|
#ifdef ___XXX___
 | 
						|
deleted, is now a debug-flag
 | 
						|
	case 'P' :	/* run preprocessor stand-alone, without #'s	*/
 | 
						|
#ifndef NOPP
 | 
						|
		options['E'] = 1;
 | 
						|
		options['P'] = 1;
 | 
						|
#else NOPP
 | 
						|
		warning("-P option ignored");
 | 
						|
#endif NOPP
 | 
						|
		break;
 | 
						|
#endif ___XXX___
 | 
						|
 | 
						|
#ifdef	LINT
 | 
						|
	case 'S' : {		/* -Sint :	static scope number for lint */
 | 
						|
		extern int stat_number;
 | 
						|
		stat_number = txt2int(&text);
 | 
						|
		break;
 | 
						|
	}
 | 
						|
#endif	LINT
 | 
						|
 | 
						|
	case 'T' : {
 | 
						|
#ifdef USE_TMP
 | 
						|
		extern char *C_tmpdir;
 | 
						|
		if (*text)
 | 
						|
			C_tmpdir = text;
 | 
						|
		else
 | 
						|
			C_tmpdir = ".";
 | 
						|
#else USE_TMP
 | 
						|
		warning("-T option ignored");
 | 
						|
#endif USE_TMP
 | 
						|
		break;
 | 
						|
	}
 | 
						|
		
 | 
						|
	case 'U' :	{	/* -Uname :	undefine predefined	*/
 | 
						|
#ifndef NOPP
 | 
						|
		register struct idf *idef;
 | 
						|
 | 
						|
		if (*text)	{
 | 
						|
			if ((idef = str2idf(text))->id_macro) {
 | 
						|
				free_macro(idef->id_macro);
 | 
						|
				idef->id_macro = (struct macro *) 0;
 | 
						|
			}
 | 
						|
		}
 | 
						|
#else NOPP
 | 
						|
		warning("-U option ignored");
 | 
						|
#endif NOPP
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
#ifndef	LINT
 | 
						|
	case 'V' :	/* set object sizes and alignment requirements	*/
 | 
						|
#ifdef NOCROSS
 | 
						|
		warning("-V option ignored");
 | 
						|
		break;
 | 
						|
#else NOCROSS
 | 
						|
	{
 | 
						|
		register arith sz, algn;
 | 
						|
		char c;
 | 
						|
 | 
						|
		while (c = *text++)	{
 | 
						|
			sz = txt2int(&text);
 | 
						|
			algn = 0;
 | 
						|
			if (*text == '.')	{
 | 
						|
				text++;
 | 
						|
				algn = txt2int(&text);
 | 
						|
			}
 | 
						|
			switch (c)	{
 | 
						|
			case 's':	/* short	*/
 | 
						|
				if (sz != (arith)0)
 | 
						|
					short_size = sz;
 | 
						|
				if (algn != 0)
 | 
						|
					short_align = algn;
 | 
						|
				break;
 | 
						|
			case 'w':	/* word		*/
 | 
						|
				if (sz != (arith)0)
 | 
						|
					dword_size = (word_size = sz) << 1;
 | 
						|
				if (algn != 0)
 | 
						|
					word_align = algn;
 | 
						|
				break;
 | 
						|
			case 'i':	/* int		*/
 | 
						|
				if (sz != (arith)0)
 | 
						|
					int_size = sz;
 | 
						|
				if (algn != 0)
 | 
						|
					int_align = algn;
 | 
						|
				break;
 | 
						|
			case 'l':	/* long		*/
 | 
						|
				if (sz != (arith)0)
 | 
						|
					long_size = sz;
 | 
						|
				if (algn != 0)
 | 
						|
					long_align = algn;
 | 
						|
				break;
 | 
						|
			case 'f':	/* float	*/
 | 
						|
#ifndef NOFLOAT
 | 
						|
				if (sz != (arith)0)
 | 
						|
					float_size = sz;
 | 
						|
				if (algn != 0)
 | 
						|
					float_align = algn;
 | 
						|
#endif NOFLOAT
 | 
						|
				break;
 | 
						|
			case 'd':	/* double	*/
 | 
						|
#ifndef NOFLOAT
 | 
						|
				if (sz != (arith)0)
 | 
						|
					double_size = sz;
 | 
						|
				if (algn != 0)
 | 
						|
					double_align = algn;
 | 
						|
#endif NOFLOAT
 | 
						|
				break;
 | 
						|
			case 'p':	/* pointer	*/
 | 
						|
				if (sz != (arith)0)
 | 
						|
					pointer_size = sz;
 | 
						|
				if (algn != 0)
 | 
						|
					pointer_align = algn;
 | 
						|
				break;
 | 
						|
			case 'r':	/* adjust bitfields right	*/
 | 
						|
#ifndef NOBITFIELD
 | 
						|
				options['r'] = 1;
 | 
						|
#else NOBITFIELD
 | 
						|
				warning("bitfields are not implemented");
 | 
						|
#endif NOBITFIELD
 | 
						|
				break;
 | 
						|
			case 'S':	/* initial struct alignment	*/
 | 
						|
				if (sz != (arith)0)
 | 
						|
					struct_align = sz;
 | 
						|
				break;
 | 
						|
			case 'U':	/* initial union alignment	*/
 | 
						|
				if (sz != (arith)0)
 | 
						|
					union_align = sz;
 | 
						|
				break;
 | 
						|
			default:
 | 
						|
				error("-V: bad type indicator %c\n", c);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		break;
 | 
						|
	}
 | 
						|
#endif NOCROSS
 | 
						|
#endif	LINT
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
txt2int(tp)
 | 
						|
	register char **tp;
 | 
						|
{
 | 
						|
	/*	the integer pointed to by *tp is read, while increasing
 | 
						|
		*tp; the resulting value is yielded.
 | 
						|
	*/
 | 
						|
	register int val = 0, ch;
 | 
						|
	
 | 
						|
	while (ch = **tp, ch >= '0' && ch <= '9')	{
 | 
						|
		val = val * 10 + ch - '0';
 | 
						|
		(*tp)++;
 | 
						|
	}
 | 
						|
	return val;
 | 
						|
}
 |