File: Narrow.m3 Last Modified On Wed Jun 29 17:29:43 PDT 1994 By kalsow Modified On Sat Dec 8 00:54:20 1990 By muller
MODULE; IMPORT CG, CallExpr, Expr, ExprRep, Type, Error, TypeExpr; IMPORT Procedure, ObjectType, Reff, Null, M3RT, RefType; IMPORT Target, RunTyme, TInt; VAR Z: CallExpr.MethodList; PROCEDURE Narrow TypeOf (ce: CallExpr.T): Type.T = VAR t: Type.T; BEGIN IF TypeExpr.Split (ce.args[1], t) THEN RETURN t; ELSE RETURN Expr.TypeOf (ce.args[0]); END; END TypeOf; PROCEDURECheck (ce: CallExpr.T; <*UNUSED*> VAR cs: Expr.CheckState) = VAR dest: Type.T; src := Expr.TypeOf (ce.args[0]); BEGIN IF NOT TypeExpr.Split (ce.args[1], dest) THEN Error.Msg ("NARROW: second argument must be a type"); dest := src; END; IF NOT Type.IsAssignable (dest, src) THEN Error.Msg ("NARROW: types must be assignable"); ELSIF ObjectType.Is (dest) OR Type.IsSubtype (dest, Reff.T) THEN (* ok *) ELSE (* untraced ref type *) Error.Msg ("NARROW: must be a traced reference or object type"); END; ce.type := dest; END Check; PROCEDUREPrep (ce: CallExpr.T) = VAR t: Type.T; BEGIN EVAL TypeExpr.Split (ce.args[1], t); Type.Compile (t); Expr.Prep (ce.args[0]); Expr.Compile (ce.args[0]); ce.tmp := EmitCore (t, Expr.TypeOf (ce.args[0])); IF (ce.tmp = NIL) THEN (* capture the ref value *) ce.tmp := CG.Pop_temp (); END; END Prep; PROCEDURECompile (ce: CallExpr.T) = BEGIN (* all the work was done by Prep *) CG.Push (ce.tmp); CG.Free (ce.tmp); ce.tmp := NIL; END Compile; PROCEDUREEmit (tlhs, trhs: Type.T) = VAR tmp := EmitCore (tlhs, trhs); BEGIN IF (tmp # NIL) THEN (* reload the ref value on the stack *) CG.Push (tmp); CG.Free (tmp); END; END Emit; PROCEDUREEmitCore (tlhs, trhs: Type.T): CG.Val = VAR ok, tagged: CG.Label; ref: CG.Val; is_object := ObjectType.Is (tlhs); target: Type.T; align: INTEGER; lhs_info, info: Type.Info; proc: Procedure.T; BEGIN tlhs := Type.CheckInfo (tlhs, lhs_info); IF is_object THEN align := ObjectType.FieldAlignment (tlhs); CG.Boost_alignment (align); ELSIF RefType.Split (tlhs, target) THEN target := Type.CheckInfo (target, info); align := info.alignment; CG.Boost_alignment (align); END; (* test for the no-check cases... *) IF Type.IsSubtype (trhs, tlhs) THEN RETURN NIL; END; IF (NOT is_object) AND (NOT lhs_info.isTraced) THEN RETURN NIL; END; (* capture the right-hand side *) ref := CG.Pop (); ok := CG.Next_label (); (* check for ref = NIL *) CG.Push (ref); CG.Load_nil (); CG.If_compare (CG.Type.Addr, CG.Cmp.EQ, ok, CG.Maybe); IF NOT Type.IsEqual (tlhs, Null.T, NIL) THEN tagged := CG.Next_label (); CG.Push (ref); CG.Loophole (CG.Type.Addr, Target.Word.cg_type); CG.Load_integer (Target.Word.cg_type, TInt.One); CG.And (Target.Word.cg_type); CG.If_true (tagged, CG.Maybe); (* check for TYPECODE(ref) = TYPECODE(type) *) CG.Push (ref); CG.Ref_to_info (M3RT.RH_typecode_offset, M3RT.RH_typecode_size); Type.LoadInfo (tlhs, M3RT.TC_typecode); CG.If_compare (Target.Integer.cg_type, CG.Cmp.EQ, ok, CG.Always); IF is_object THEN (* finally, call the runtime to figure it out... *) proc := RunTyme.LookUpProc (RunTyme.Hook.CheckIsType); Procedure.StartCall (proc); IF Target.DefaultCall.args_left_to_right THEN CG.Push (ref); CG.Pop_param (CG.Type.Addr); Type.LoadInfo (tlhs, -1); CG.Pop_param (CG.Type.Addr); ELSE Type.LoadInfo (tlhs, -1); CG.Pop_param (CG.Type.Addr); CG.Push (ref); CG.Pop_param (CG.Type.Addr); END; Procedure.EmitCall (proc); CG.If_true (ok, CG.Always); END; CG.Abort (CG.RuntimeError.NarrowFailed); CG.Set_label (tagged); CG.Load_intt (M3RT.REFANY_typecode); Type.LoadInfo (tlhs, M3RT.TC_typecode); CG.If_compare (Target.Integer.cg_type, CG.Cmp.EQ, ok, CG.Always); END; CG.Abort (CG.RuntimeError.NarrowFailed); CG.Set_label (ok); RETURN ref; END EmitCore; PROCEDUREFold (ce: CallExpr.T): Expr.T = BEGIN RETURN Expr.ConstValue (ce.args[0]); END Fold; PROCEDURENoteWrites (ce: CallExpr.T) = BEGIN Expr.NoteWrite (ce.args[0]); END NoteWrites; PROCEDUREInitialize () = BEGIN Z := CallExpr.NewMethodList (2, 2, TRUE, FALSE, TRUE, NIL, TypeOf, CallExpr.NotAddressable, Check, Prep, Compile, CallExpr.NoLValue, CallExpr.NoLValue, CallExpr.NotBoolean, CallExpr.NotBoolean, Fold, CallExpr.NoBounds, CallExpr.IsNever, (* writable *) CallExpr.IsNever, (* designator *) NoteWrites); Procedure.Define ("NARROW", Z, TRUE); END Initialize; BEGIN END Narrow.