/* $Header$ */ /* 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 "target_sizes.h" #include "idf.h" #include "arith.h" #include "type.h" #include "label.h" #include "expr.h" #include "sizes.h" #include "Lpars.h" #include "assert.h" long mach_long_sign; /* sign bit of the machine long */ int mach_long_size; /* size of long on this machine == sizeof(long) */ long full_mask[MAXSIZE];/* full_mask[1] == 0XFF, full_mask[2] == 0XFFFF, .. */ arith max_int; /* maximum integer on target machine */ arith max_unsigned; /* maximum unsigned on target machine */ cstbin(expp, oper, expr) register struct expr **expp, *expr; { /* The operation oper is performed on the constant expressions *expp(ld) and expr(ct), and the result restored in *expp. */ register arith o1 = (*expp)->VL_VALUE; register arith o2 = expr->VL_VALUE; int uns = (*expp)->ex_type->tp_unsigned; ASSERT(is_ld_cst(*expp) && is_cp_cst(expr)); switch (oper) { case '*': o1 *= o2; break; case '/': if (o2 == 0) { expr_error(expr, "division by 0"); break; } if (uns) { /* this is more of a problem than you might think on C compilers which do not have unsigned long. */ if (o2 & mach_long_sign) {/* o2 > max_long */ o1 = ! (o1 >= 0 || o1 < o2); /* this is the unsigned test o1 < o2 for o2 > max_long */ } else { /* o2 <= max_long */ long half, bit, hdiv, hrem, rem; half = (o1 >> 1) & ~mach_long_sign; bit = o1 & 01; /* now o1 == 2 * half + bit and half <= max_long and bit <= max_long */ hdiv = half / o2; hrem = half % o2; rem = 2 * hrem + bit; o1 = 2 * hdiv + (rem < 0 || rem >= o2); /* that is the unsigned compare rem >= o2 for o2 <= max_long */ } } else o1 /= o2; break; case '%': if (o2 == 0) { expr_error(expr, "modulo by 0"); break; } if (uns) { if (o2 & mach_long_sign) {/* o2 > max_long */ o1 = (o1 >= 0 || o1 < o2) ? o1 : o1 - o2; /* this is the unsigned test o1 < o2 for o2 > max_long */ } else { /* o2 <= max_long */ long half, bit, hrem, rem; half = (o1 >> 1) & ~mach_long_sign; bit = o1 & 01; /* now o1 == 2 * half + bit and half <= max_long and bit <= max_long */ hrem = half % o2; rem = 2 * hrem + bit; o1 = (rem < 0 || rem >= o2) ? rem - o2 : rem; } } else o1 %= o2; break; case '+': o1 += o2; break; case '-': o1 -= o2; break; case LEFT: o1 <<= o2; break; case RIGHT: if (o2 == 0) break; if (uns) { o1 >>= 1; o1 & = ~mach_long_sign; o1 >>= (o2-1); } else o1 >>= o2; break; case '<': { arith tmp = o1; o1 = o2; o2 = tmp; } /* Fall through */ case '>': if (uns) { o1 = (o1 & mach_long_sign ? (o2 & mach_long_sign ? o1 > o2 : 1) : (o2 & mach_long_sign ? 0 : o1 > o2) ); } else o1 = o1 > o2; break; case LESSEQ: { arith tmp = o1; o1 = o2; o2 = tmp; } /* Fall through */ case GREATEREQ: if (uns) { o1 = (o1 & mach_long_sign ? (o2 & mach_long_sign ? o1 >= o2 : 1) : (o2 & mach_long_sign ? 0 : o1 >= o2) ); } else o1 = o1 >= o2; break; case EQUAL: o1 = o1 == o2; break; case NOTEQUAL: o1 = o1 != o2; break; case '&': o1 &= o2; break; case '|': o1 |= o2; break; case '^': o1 ^= o2; break; } (*expp)->VL_VALUE = o1; cut_size(*expp); (*expp)->ex_flags |= expr->ex_flags; (*expp)->ex_flags &= ~EX_PARENS; free_expression(expr); } cut_size(expr) register struct expr *expr; { /* The constant value of the expression expr is made to conform to the size of the type of the expression. */ arith o1 = expr->VL_VALUE; int uns = expr->ex_type->tp_unsigned; int size = (int) expr->ex_type->tp_size; ASSERT(expr->ex_class == Value); if (uns) { if (o1 & ~full_mask[size]) expr_warning(expr, "overflow in unsigned constant expression"); o1 &= full_mask[size]; } else { int nbits = (int) (mach_long_size - size) * 8; long remainder = o1 & ~full_mask[size]; if (remainder != 0 && remainder != ~full_mask[size]) expr_warning(expr, "overflow in constant expression"); o1 <<= nbits; /* ??? */ o1 >>= nbits; } expr->VL_VALUE = o1; } init_cst() { register int i = 0; register arith bt = (arith)0; while (!(bt < 0)) { bt = (bt << 8) + 0377, i++; if (i == MAXSIZE) fatal("array full_mask too small for this machine"); full_mask[i] = bt; } mach_long_size = i; mach_long_sign = 1 << (mach_long_size * 8 - 1); if (long_size < mach_long_size) fatal("sizeof (long) insufficient on this machine"); max_int = full_mask[int_size] & ~(1 << (int_size * 8 - 1)); max_unsigned = full_mask[int_size]; }