497 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			497 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* C O N S T A N T   E X P R E S S I O N   H A N D L I N G */
 | |
| 
 | |
| #include    <stdlib.h>
 | |
| #include    <string.h>
 | |
| #include	"debug.h"
 | |
| #include	"target_sizes.h"
 | |
| 
 | |
| #include	<alloc.h>
 | |
| #include	<assert.h>
 | |
| #include	<em_arith.h>
 | |
| #include	<em_label.h>
 | |
| 
 | |
| #include	"LLlex.h"
 | |
| #include	"Lpars.h"
 | |
| #include	"const.h"
 | |
| #include	"node.h"
 | |
| #include	"required.h"
 | |
| #include	"type.h"
 | |
| 
 | |
| long mach_long_sign;	/* sign bit of the machine long */
 | |
| long full_mask[MAXSIZE+1];/* full_mask[1] == 0xFF, full_mask[2] == 0xFFFF, .. */
 | |
| arith max_int;		/* maximum integer on the target machine */
 | |
| arith min_int;		/* mimimum integer on the target machin */
 | |
| char *maxint_str;	/* string representation of maximum integer */
 | |
| arith wrd_bits;		/* number of bits in a word */
 | |
| arith max_intset;	/* largest value of set of integer */
 | |
| 
 | |
| overflow(expp)
 | |
| 	struct node *expp;
 | |
| {
 | |
| 	node_warning(expp, "overflow in constant expression");
 | |
| }
 | |
| 
 | |
| cstunary(expp)
 | |
| 	register struct node *expp;
 | |
| {
 | |
| 	/*	The unary operation in "expp" is performed on the constant
 | |
| 		expression below it, and the result restored in expp.
 | |
| 	*/
 | |
| 	register arith o1 = expp->nd_right->nd_INT;
 | |
| 
 | |
| 	switch( expp->nd_symb )	{
 | |
| 		/* Should not get here
 | |
| 		case '+':
 | |
| 		case '(':
 | |
| 			break;
 | |
| 		*/
 | |
| 
 | |
| 		case '-':
 | |
| 			o1 = -o1;
 | |
| 			break;
 | |
| 
 | |
| 		case NOT:
 | |
| 			o1 = !o1;
 | |
| 			break;
 | |
| 
 | |
| 		default:
 | |
| 			crash("(cstunary)");
 | |
| 	}
 | |
| 
 | |
| 	expp->nd_class = Value;
 | |
| 	expp->nd_token = expp->nd_right->nd_token;
 | |
| 	expp->nd_INT = o1;
 | |
| 	CutSize(expp);
 | |
| 	FreeNode(expp->nd_right);
 | |
| 	expp->nd_right = NULLNODE;
 | |
| }
 | |
| 
 | |
| cstbin(expp)
 | |
| 	register struct node *expp;
 | |
| {
 | |
| 	/*	The binary operation in "expp" is performed on the constant
 | |
| 		expressions below it, and the result restored in expp.
 | |
| 	*/
 | |
| 	register arith o1, o2;
 | |
| 	register char *s1, *s2;
 | |
| 	int str = expp->nd_left->nd_type->tp_fund & T_STRINGCONST;
 | |
| 
 | |
| 	if( str )	{
 | |
| 		o1 = o2 = 0;			/* so LINT won't complain */
 | |
| 		s1 = expp->nd_left->nd_STR;
 | |
| 		s2 = expp->nd_right->nd_STR;
 | |
| 	}
 | |
| 	else	{
 | |
| 		s1 = s2 = (char *) 0;		/* so LINT won't complain */
 | |
| 		o1 = expp->nd_left->nd_INT;
 | |
| 		o2 = expp->nd_right->nd_INT;
 | |
| 	}
 | |
| 
 | |
| 	assert(expp->nd_class == Boper);
 | |
| 	assert(expp->nd_left->nd_class == Value);
 | |
| 	assert(expp->nd_right->nd_class == Value);
 | |
| 
 | |
| 	switch( expp->nd_symb )	{
 | |
| 		case '+':
 | |
| 			if (o1 > 0 && o2 > 0) {
 | |
| 				if (max_int - o1 < o2) overflow(expp);
 | |
| 			}
 | |
| 			else if (o1 < 0 && o2 < 0) {
 | |
| 				if (min_int - o1 > o2) overflow(expp);
 | |
| 			}
 | |
| 			o1 += o2;
 | |
| 			break;
 | |
| 
 | |
| 		case '-':
 | |
| 			if ( o1 >= 0 && o2 < 0) {
 | |
| 				if (max_int + o2 < o1) overflow(expp);
 | |
| 			}
 | |
| 			else if (o1 < 0 && o2 >= 0) {
 | |
| 				if (min_int + o2 > o1) overflow(expp);
 | |
| 			}
 | |
| 			o1 -= o2;
 | |
| 			break;
 | |
| 
 | |
| 		case '*':
 | |
| 			if (o1 > 0 && o2 > 0) {
 | |
| 				if (max_int / o1 < o2) overflow(expp);
 | |
| 			}
 | |
| 			else if (o1 < 0 && o2 < 0) {
 | |
| 				if (o1 == min_int || o2 == min_int ||
 | |
| 				    max_int / (-o1) < (-o2)) overflow(expp);
 | |
| 			}
 | |
| 			else if (o1 > 0) {
 | |
| 				if (min_int / o1 > o2) overflow(expp);
 | |
| 			}
 | |
| 			else if (o2 > 0) {
 | |
| 				if (min_int / o2 > o1) overflow(expp);
 | |
| 			}
 | |
| 			o1 *= o2;
 | |
| 			break;
 | |
| 
 | |
| 		case DIV:
 | |
| 			if( o2 == 0 )	{
 | |
| 				node_error(expp, "division by 0");
 | |
| 				return;
 | |
| 			}
 | |
| 			else o1 /= o2;
 | |
| 			break;
 | |
| 
 | |
| 		case MOD:
 | |
| 			if( o2 == 0 )	{
 | |
| 				node_error(expp, "modulo by 0");
 | |
| 				return;
 | |
| 			}
 | |
| 			else
 | |
| 				o1 %= o2;
 | |
| 			break;
 | |
| 
 | |
| 		case OR:
 | |
| 			o1 = (o1 || o2);
 | |
| 			break;
 | |
| 
 | |
| 		case AND:
 | |
| 			o1 = (o1 && o2);
 | |
| 			break;
 | |
| 
 | |
| 		case '=':
 | |
| 			o1 = str ? !strcmp(s1, s2) : (o1 == o2);
 | |
| 			break;
 | |
| 
 | |
| 		case NOTEQUAL:
 | |
| 			o1 = str ? (strcmp(s1, s2) != 0) : (o1 != o2);
 | |
| 			break;
 | |
| 
 | |
| 		case LESSEQUAL:
 | |
| 			o1 = str ? (strcmp(s1, s2) <= 0) : (o1 <= o2);
 | |
| 			break;
 | |
| 
 | |
| 		case GREATEREQUAL:
 | |
| 			o1 = str ? (strcmp(s1, s2) >= 0) : (o1 >= o2);
 | |
| 			break;
 | |
| 
 | |
| 		case '<':
 | |
| 			o1 = str ? (strcmp(s1, s2) < 0) : (o1 < o2);
 | |
| 			break;
 | |
| 
 | |
| 		case '>':
 | |
| 			o1 = str ? (strcmp(s1, s2) > 0) : (o1 > o2);
 | |
| 			break;
 | |
| 
 | |
| 		/* case '/': */
 | |
| 		default:
 | |
| 			crash("(cstbin)");
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	expp->nd_class = Value;
 | |
| 	expp->nd_token = expp->nd_right->nd_token;
 | |
| 	/* STRING compare has a bool_type as result */
 | |
| 	if( expp->nd_type == bool_type ) expp->nd_symb = INTEGER;
 | |
| 	expp->nd_INT = o1;
 | |
| 	CutSize(expp);
 | |
| 	FreeNode(expp->nd_left);
 | |
| 	FreeNode(expp->nd_right);
 | |
| 	expp->nd_left = expp->nd_right = NULLNODE;
 | |
| }
 | |
| 
 | |
| cstset(expp)
 | |
| 	register struct node *expp;
 | |
| {
 | |
| 	register arith *set1, *set2;
 | |
| 	arith *resultset = (arith *) 0;
 | |
| 	int empty_result = 0;
 | |
| 	register int setsize, j;
 | |
| 
 | |
| 	assert(expp->nd_right->nd_class == Set);
 | |
| 	assert(expp->nd_symb == IN || expp->nd_left->nd_class == Set);
 | |
| 	set2 = expp->nd_right->nd_set;
 | |
| 	setsize = (unsigned) (expp->nd_right->nd_type->tp_size) / (unsigned) word_size;
 | |
| 
 | |
| 	if( expp->nd_symb == IN )	{
 | |
| 		arith i;
 | |
| 
 | |
| 		assert(expp->nd_left->nd_class == Value);
 | |
| 
 | |
| 		i = expp->nd_left->nd_INT;
 | |
| 		expp->nd_class = Value;
 | |
| 		expp->nd_symb = INTEGER;
 | |
| 
 | |
| 		expp->nd_INT = (i >= 0 && set2 && i < (setsize * wrd_bits) &&
 | |
| 				(set2[i/wrd_bits] & (1 << (i%wrd_bits))));
 | |
| 
 | |
| 		if( set2 ) free((char *) set2);
 | |
| 	}
 | |
| 	else	{
 | |
| 		set1 = expp->nd_left->nd_set;
 | |
| 		resultset = set1;
 | |
| 		expp->nd_left->nd_set = (arith *) 0;
 | |
| 		switch( expp->nd_symb )	{
 | |
| 		case '+':
 | |
| 			/* Set union
 | |
| 			*/
 | |
| 			if( !set1 )	{
 | |
| 				resultset = set2;
 | |
| 				expp->nd_right->nd_set = (arith *) 0;
 | |
| 				break;
 | |
| 			}
 | |
| 			if( set2 )
 | |
| 				for( j = 0; j < setsize; j++ )
 | |
| 					*set1++ |= *set2++;
 | |
| 			break;
 | |
| 
 | |
| 		case '-':
 | |
| 			/* Set difference
 | |
| 			*/
 | |
| 			if( !set1 || !set2 )	{
 | |
| 				/* The set from which something is substracted
 | |
| 				   is already empty, or the set that is
 | |
| 				   substracted is empty. In either case, the
 | |
| 				   result set is set1.
 | |
| 				*/
 | |
| 				break;
 | |
| 			}
 | |
| 			empty_result = 1;
 | |
| 			for( j = 0; j < setsize; j++ )
 | |
| 				if( *set1++ &= ~*set2++ ) empty_result = 0;
 | |
| 			break;
 | |
| 
 | |
| 		case '*':
 | |
| 			/* Set intersection
 | |
| 			*/
 | |
| 			if( !set1 )	{
 | |
| 				/* set1 is empty, and so is the result set
 | |
| 				*/
 | |
| 				break;
 | |
| 			}
 | |
| 			if( !set2 )	{
 | |
| 				/* set 2 is empty, so the result set must be
 | |
| 				   empty too.
 | |
| 				*/
 | |
| 				resultset = set2;
 | |
| 				expp->nd_right->nd_set = (arith *) 0;
 | |
| 				break;
 | |
| 			}
 | |
| 			empty_result = 1;
 | |
| 			for( j = 0; j < setsize; j++ )
 | |
| 				if( *set1++ &= *set2++ ) empty_result = 0;
 | |
| 			break;
 | |
| 
 | |
| 		case '=':
 | |
| 		case NOTEQUAL:
 | |
| 		case LESSEQUAL:
 | |
| 		case GREATEREQUAL:
 | |
| 			/* Constant set comparisons
 | |
| 			*/
 | |
| 			if( !setsize ) setsize++;	/* force comparison */
 | |
| 			expp->nd_left->nd_set = set1;	/* may be disposed of */
 | |
| 			for( j = 0; j < setsize; j++ )	{
 | |
| 				switch( expp->nd_symb )	{
 | |
| 				case '=':
 | |
| 				case NOTEQUAL:
 | |
| 					if( !set1 && !set2 )	{
 | |
| 						j = setsize;
 | |
| 						break;
 | |
| 					}
 | |
| 					if( !set1 || !set2 ) break;
 | |
| 					if( *set1++ != *set2++ ) break;
 | |
| 					continue;
 | |
| 				case LESSEQUAL:
 | |
| 					if( !set1 )	{
 | |
| 						j = setsize;
 | |
| 						break;
 | |
| 					}
 | |
| 					if( !set2 ) break;
 | |
| 					if( (*set2 | *set1++) != *set2 ) break;
 | |
| 					set2++;
 | |
| 					continue;
 | |
| 				case GREATEREQUAL:
 | |
| 					if( !set2 )	{
 | |
| 						j = setsize;
 | |
| 						break;
 | |
| 					}
 | |
| 					if( !set1 ) break;
 | |
| 					if( (*set1 | *set2++) != *set1 ) break;
 | |
| 					set1++;
 | |
| 					continue;
 | |
| 				}
 | |
| 				break;
 | |
| 			}
 | |
| 			if( j < setsize )
 | |
| 				expp->nd_INT = expp->nd_symb == NOTEQUAL;
 | |
| 			else
 | |
| 				expp->nd_INT = expp->nd_symb != NOTEQUAL;
 | |
| 			expp->nd_class = Value;
 | |
| 			expp->nd_symb = INTEGER;
 | |
| 			if( expp->nd_left->nd_set )
 | |
| 				free((char *) expp->nd_left->nd_set);
 | |
| 			if( expp->nd_right->nd_set )
 | |
| 				free((char *) expp->nd_right->nd_set);
 | |
| 			FreeNode(expp->nd_left);
 | |
| 			FreeNode(expp->nd_right);
 | |
| 			expp->nd_left = expp->nd_right = NULLNODE;
 | |
| 			return;
 | |
| 		default:
 | |
| 			crash("(cstset)");
 | |
| 		}
 | |
| 		if( expp->nd_right->nd_set )
 | |
| 			free((char *) expp->nd_right->nd_set);
 | |
| 		if( expp->nd_left->nd_set )
 | |
| 			free((char *) expp->nd_left->nd_set);
 | |
| 		if( empty_result )	{
 | |
| 			free((char *) resultset);
 | |
| 			resultset = (arith *) 0;
 | |
| 		}
 | |
| 		expp->nd_class = Set;
 | |
| 		expp->nd_set = resultset;
 | |
| 	}
 | |
| 	FreeNode(expp->nd_left);
 | |
| 	FreeNode(expp->nd_right);
 | |
| 	expp->nd_left = expp->nd_right = NULLNODE;
 | |
| }
 | |
| 
 | |
| cstcall(expp, req)
 | |
| 	register struct node *expp;
 | |
| {
 | |
| 	/*	a standard procedure call is found that can be evaluated
 | |
| 		compile time, so do so.
 | |
| 	*/
 | |
| 	register struct node *expr = NULLNODE;
 | |
| 
 | |
| 	assert(expp->nd_class == Call);
 | |
| 
 | |
| 	expr = expp->nd_right->nd_left;
 | |
| 
 | |
| 	expp->nd_class = Value;
 | |
| 	expp->nd_symb = INTEGER;
 | |
| 	switch( req )	{
 | |
| 	    case R_ABS:
 | |
| 		if( expr->nd_INT < 0 ) {
 | |
| 			if (expr->nd_INT <= min_int) {
 | |
| 				overflow(expr);
 | |
| 			}
 | |
| 			expp->nd_INT = - expr->nd_INT;
 | |
| 		}
 | |
| 		else expp->nd_INT = expr->nd_INT;
 | |
| 		CutSize(expp);
 | |
| 		break;
 | |
| 
 | |
| 	    case R_SQR:
 | |
| 		if (expr->nd_INT < 0) {
 | |
| 			if ( expr->nd_INT == min_int ||
 | |
| 			    max_int / expr->nd_INT > expr->nd_INT) {
 | |
| 				overflow(expr);
 | |
| 			}
 | |
| 		}
 | |
| 		else if (max_int / expr->nd_INT < expr->nd_INT) {
 | |
| 			overflow(expr);
 | |
| 		}
 | |
| 		expp->nd_INT = expr->nd_INT * expr->nd_INT;
 | |
| 		CutSize(expp);
 | |
| 		break;
 | |
| 
 | |
| 	    case R_ORD:
 | |
| 	    case R_CHR:
 | |
| 		expp->nd_INT = expr->nd_INT;
 | |
| 		CutSize(expp);
 | |
| 		break;
 | |
| 
 | |
| 	    case R_ODD:
 | |
| 		expp->nd_INT = (expr->nd_INT & 1);
 | |
| 		break;
 | |
| 
 | |
| 	    case R_SUCC:
 | |
| 		expp->nd_INT = expr->nd_INT + 1;
 | |
| 		if(	/* Check overflow of subranges or enumerations */
 | |
| 			(expp->nd_type->tp_fund & T_SUBRANGE &&
 | |
| 				expp->nd_INT > expp->nd_type->sub_ub
 | |
| 			)
 | |
| 		   ||
 | |
| 			( expp->nd_type->tp_fund & T_ENUMERATION &&
 | |
| 				expp->nd_INT >= expp->nd_type->enm_ncst
 | |
| 			)
 | |
| 		  )
 | |
| 			node_warning(expp, "\"succ\": no successor");
 | |
| 		else CutSize(expp);
 | |
| 		break;
 | |
| 
 | |
| 	    case R_PRED:
 | |
| 		expp->nd_INT = expr->nd_INT - 1;
 | |
| 		if(	/* Check with lowerbound of subranges or enumerations */
 | |
| 			(expp->nd_type->tp_fund & T_SUBRANGE &&
 | |
| 				expp->nd_INT < expp->nd_type->sub_lb
 | |
| 			)
 | |
| 		   ||
 | |
| 			( expp->nd_type->tp_fund & T_ENUMERATION &&
 | |
| 				expp->nd_INT < 0
 | |
| 			)
 | |
| 		  )
 | |
| 			node_warning(expp, "\"pred\": no predecessor");
 | |
| 		else CutSize(expp);
 | |
| 		break;
 | |
| 
 | |
| 	    default:
 | |
| 		crash("(cstcall)");
 | |
| 	}
 | |
| 	FreeNode(expp->nd_left);
 | |
| 	FreeNode(expp->nd_right);
 | |
| 	expp->nd_right = expp->nd_left = NULLNODE;
 | |
| }
 | |
| 
 | |
| CutSize(expr)
 | |
| 	register struct node *expr;
 | |
| {
 | |
| 	/* The constant value of the expression expr is made to conform
 | |
| 	 * to the size of the type of the expression
 | |
| 	 */
 | |
| 	register arith o1 = expr->nd_INT;
 | |
| 	register struct type *tp = BaseType(expr->nd_type);
 | |
| 	int size = tp->tp_size;
 | |
| 	long remainder = o1 & ~full_mask[size];
 | |
| 
 | |
| 	assert(expr->nd_class == Value);
 | |
| 
 | |
| 	if( tp->tp_fund & T_CHAR )	{
 | |
| 		if( o1 & (~full_mask[size] >> 1) ) 	{
 | |
| 			node_warning(expr, "overflow in character value");
 | |
| 			o1 &= 0177;
 | |
| 		}
 | |
| 	}
 | |
| 	else if( remainder != 0 && remainder != ~full_mask[size] ||
 | |
| 	    		(o1 & full_mask[size]) == 1 << (size * 8 - 1) )	{
 | |
| 		/* integers in [-maxint .. maxint] */
 | |
| 		int nbits = (int) (sizeof(long) - size) * 8;
 | |
| 
 | |
| 		/* overflow(expr); */
 | |
| 		/* sign bit of o1 in sign bit of mach_long */
 | |
| 		o1 <<= nbits;
 | |
| 		/* shift back to get sign extension */
 | |
| 		o1 >>= nbits;
 | |
| 	}
 | |
| 	expr->nd_INT = o1;
 | |
| }
 | |
| 
 | |
| InitCst()
 | |
| {
 | |
| 	extern char *long2str(), *Salloc();
 | |
| 	register int i = 0;
 | |
| 	register arith bt = (arith)0;
 | |
| 
 | |
| 	while( !(bt < 0) )	{
 | |
| 		bt = (bt << 8) + 0377;
 | |
| 		i++;
 | |
| 		if( i == MAXSIZE + 1 )
 | |
| 			fatal("array full_mask too small for this machine");
 | |
| 		full_mask[i] = bt;
 | |
| 	}
 | |
| 	mach_long_sign = 1L << (sizeof(long) * 8 - 1);
 | |
| 	if( int_size > sizeof(long) )
 | |
| 		fatal("sizeof (long) insufficient on this machine");
 | |
| 
 | |
| 	max_int = full_mask[int_size] & ~(1L << (int_size * 8 - 1));
 | |
| 	min_int = - max_int;
 | |
| 	maxint_str = long2str(max_int, 10);
 | |
| 	maxint_str = Salloc(maxint_str, (unsigned int) strlen(maxint_str));
 | |
| 	wrd_bits = 8 * (int) word_size;
 | |
| 	if( !max_intset ) max_intset = wrd_bits - 1;
 | |
| }
 |