File: BitSize.m3 Last Modified On Wed Jun 29 17:00:26 PDT 1994 By kalsow Modified On Sat Dec 8 00:54:27 1990 By muller
MODULE; IMPORT CG, CallExpr, Expr, ExprRep, Type, Procedure, IntegerExpr; IMPORT TypeExpr, OpenArrayType, Error, Card, Target, TInt, Int; VAR Z: CallExpr.MethodList; PROCEDURE BitSize Check (ce: CallExpr.T; VAR cs: Expr.CheckState) = BEGIN DoCheck ("BITSIZE", ce, cs); END Check; PROCEDUREPrep (ce: CallExpr.T) = BEGIN DoPrep (ce.args[0]); END Prep; PROCEDURECompile (ce: CallExpr.T) = BEGIN DoCompile (ce.args[0], 1); END Compile; PROCEDUREFold (ce: CallExpr.T): Expr.T = BEGIN RETURN DoFold (ce.args[0], 1); END Fold; PROCEDUREDoCheck (name: TEXT; ce: CallExpr.T; <*UNUSED*> VAR cs: Expr.CheckState) = VAR t: Type.T; e := ce.args[0]; info: Type.Info; BEGIN IF Expr.IsDesignator (e) THEN (* ok *) t := Type.CheckInfo (Expr.TypeOf (e), info); ELSIF TypeExpr.Split (e, t) THEN IF OpenArrayType.Is (t) THEN Error.Txt (name, "argument cannot be an open array type"); END; ELSE Error.Txt (name, "argument must be a designator or type"); END; ce.type := Card.T; END DoCheck; PROCEDUREDoPrep (e: Expr.T) = VAR t: Type.T; info: Type.Info; BEGIN IF TypeExpr.Split (e, t) THEN RETURN; END; t := Type.CheckInfo (Expr.TypeOf (e), info); IF (info.class # Type.Class.OpenArray) THEN RETURN; END; (* ELSE, open array *) Expr.Prep (e); END DoPrep; PROCEDUREDoCompile (e: Expr.T; unit: INTEGER) = VAR t: Type.T; sz, align: INTEGER; t_array: CG.Val; info: Type.Info; BEGIN IF TypeExpr.Split (e, t) THEN Type.Compile (t); EVAL Type.CheckInfo (t, info); CG.Load_intt ((info.size + unit - 1) DIV unit); RETURN; END; t := Type.CheckInfo (Expr.TypeOf (e), info); IF (info.class # Type.Class.OpenArray) THEN CG.Load_intt ((info.size + unit - 1) DIV unit); RETURN; END; (* ELSE, open array *) align := info.alignment; Expr.Compile (e); t_array := CG.Pop (); FOR i := 0 TO OpenArrayType.OpenDepth (t) - 1 DO CG.Push (t_array); CG.Open_size (i); IF (i # 0) THEN CG.Multiply (Target.Integer.cg_type) END; END; CG.Free (t_array); sz := OpenArrayType.EltPack (t); IF (sz MOD unit) = 0 THEN CG.Load_intt (sz DIV unit); CG.Multiply (Target.Integer.cg_type); ELSE (* array elements aren't "unit"-aligned *) CG.Load_intt (sz); CG.Multiply (Target.Integer.cg_type); CG.Load_intt (unit - 1); CG.Add (Target.Integer.cg_type); CG.Load_intt (unit); CG.Div (Target.Integer.cg_type, CG.Sign.Positive, CG.Sign.Positive); END; END DoCompile; PROCEDUREDoFold (e: Expr.T; unit: INTEGER): Expr.T = VAR t: Type.T; size, a, b, c, d: Target.Int; info: Type.Info; BEGIN IF NOT TypeExpr.Split (e, t) THEN t := Type.CheckInfo (Expr.TypeOf (e), info); IF (info.class = Type.Class.OpenArray) THEN RETURN NIL END; END; t := Type.CheckInfo (t, info); IF TInt.FromInt (info.size, Target.Integer.bytes, size) AND TInt.FromInt (unit, Target.Integer.bytes, a) AND TInt.FromInt (unit - 1, Target.Integer.bytes, b) AND TInt.Add (size, b, c) AND TInt.Div (c, a, d) THEN RETURN IntegerExpr.New (Int.T, d); ELSE RETURN NIL; END; END DoFold; PROCEDUREInitialize () = BEGIN Z := CallExpr.NewMethodList (1, 1, TRUE, FALSE, FALSE, Card.T, NIL, CallExpr.NotAddressable, Check, Prep, Compile, CallExpr.NoLValue, CallExpr.NoLValue, CallExpr.NotBoolean, CallExpr.NotBoolean, Fold, CallExpr.NoBounds, CallExpr.IsNever, (* writable *) CallExpr.IsNever, (* designator *) CallExpr.NotWritable (* noteWriter *)); Procedure.Define ("BITSIZE", Z, TRUE); END Initialize; BEGIN END BitSize.