aboutsummaryrefslogtreecommitdiffstats
path: root/src/synth/synth-expr.adb
diff options
context:
space:
mode:
Diffstat (limited to 'src/synth/synth-expr.adb')
-rw-r--r--src/synth/synth-expr.adb143
1 files changed, 98 insertions, 45 deletions
diff --git a/src/synth/synth-expr.adb b/src/synth/synth-expr.adb
index cda931798..c3f1f2589 100644
--- a/src/synth/synth-expr.adb
+++ b/src/synth/synth-expr.adb
@@ -705,41 +705,95 @@ package body Synth.Expr is
return Res;
end Create_Bounds_From_Length;
- -- Implicit conversion of literals.
- function Synth_Implicit_Conv (Syn_Inst : Synth_Instance_Acc;
- Val : Value_Acc;
- Vtype : Node;
- Rtype : Node) return Value_Acc
+ function Synth_Subtype_Conversion
+ (Val : Value_Acc; Dtype : Type_Acc; Loc : Source.Syn_Src)
+ return Value_Acc
is
- Vtyp : Type_Acc;
- Rtyp : Type_Acc;
+ Vtype : constant Type_Acc := Val.Typ;
begin
- if Rtype = Vtype or else Val.Kind /= Value_Discrete then
- return Val;
- end if;
- if Vtype /= Vhdl.Std_Package.Convertible_Integer_Type_Definition then
- raise Internal_Error;
- end if;
- Vtyp := Val.Typ;
- Rtyp := Get_Value_Type (Syn_Inst, Rtype);
- if Rtyp.Drange.W < Vtyp.Drange.W then
- -- TODO: check bounds.
- return Create_Value_Discrete (Val.Scal, Rtyp);
- else
- pragma Assert (Vtyp.Drange.W = Rtyp.Drange.W);
- return Val;
- end if;
- end Synth_Implicit_Conv;
+ case Dtype.Kind is
+ when Type_Bit =>
+ pragma Assert (Vtype.Kind = Type_Bit);
+ return Val;
+ when Type_Discrete =>
+ pragma Assert (Vtype.Kind = Type_Discrete);
+ declare
+ Vrng : Discrete_Range_Type renames Vtype.Drange;
+ Drng : Discrete_Range_Type renames Dtype.Drange;
+ N : Net;
+ begin
+ if Vrng.W > Drng.W then
+ -- Truncate.
+ -- TODO: check overflow.
+ case Val.Kind is
+ when Value_Net
+ | Value_Wire =>
+ N := Get_Net (Val);
+ N := Build_Trunc (Build_Context, Id_Utrunc, N, Drng.W);
+ Set_Location (N, Loc);
+ return Create_Value_Net (N, Dtype);
+ when others =>
+ raise Internal_Error;
+ end case;
+ elsif Vrng.W < Drng.W then
+ -- Extend.
+ case Val.Kind is
+ when Value_Discrete =>
+ return Create_Value_Discrete (Val.Scal, Dtype);
+ when Value_Net
+ | Value_Wire =>
+ N := Get_Net (Val);
+ if Vrng.Is_Signed then
+ N := Build_Extend
+ (Build_Context, Id_Sextend, N, Drng.W);
+ else
+ N := Build_Extend
+ (Build_Context, Id_Uextend, N, Drng.W);
+ end if;
+ Set_Location (N, Loc);
+ return Create_Value_Net (N, Dtype);
+ when others =>
+ raise Internal_Error;
+ end case;
+ else
+ -- TODO: check overflow if sign differ.
+ return Val;
+ end if;
+ end;
+ when Type_Float =>
+ pragma Assert (Vtype.Kind = Type_Float);
+ -- TODO: check range
+ return Val;
+ when Type_Vector =>
+ -- TODO: check width
+ return Val;
+ when Type_Array =>
+ -- TODO: check bounds, handle elements
+ return Val;
+ when Type_Unbounded_Array =>
+ pragma Assert (Vtype.Kind = Type_Vector
+ or else Vtype.Kind = Type_Array);
+ return Val;
+ when Type_Record =>
+ -- TODO: handle elements.
+ return Val;
+ end case;
+ end Synth_Subtype_Conversion;
+ -- Implicit conversion of literals.
function Synth_Dyadic_Operation (Syn_Inst : Synth_Instance_Acc;
- Def : Iir_Predefined_Functions;
+ Imp : Node;
Left_Expr : Node;
Right_Expr : Node;
Expr : Node) return Value_Acc
is
+ Def : constant Iir_Predefined_Functions :=
+ Get_Implicit_Definition (Imp);
+ Inter_Chain : constant Node :=
+ Get_Interface_Declaration_Chain (Imp);
Expr_Type : constant Node := Get_Type (Expr);
- Ltype : constant Node := Get_Type (Left_Expr);
- Rtype : constant Node := Get_Type (Right_Expr);
+ Left_Type : constant Node := Get_Type (Inter_Chain);
+ Right_Type : constant Node := Get_Type (Get_Chain (Inter_Chain));
Left : Value_Acc;
Right : Value_Acc;
@@ -768,7 +822,7 @@ package body Synth.Expr is
is
N : Net;
begin
- N := Synth_Uresize (Right, Rtype, Get_Width (Left));
+ N := Synth_Uresize (Right, Right_Type, Get_Width (Left));
Set_Location (N, Expr);
N := Build_Compare (Build_Context, Id, Get_Net (Left), N);
Set_Location (N, Expr);
@@ -788,15 +842,10 @@ package body Synth.Expr is
function Synth_Int_Dyadic (Id : Dyadic_Module_Id) return Value_Acc
is
Etype : constant Type_Acc := Get_Value_Type (Syn_Inst, Expr_Type);
- L, R : Net;
N : Net;
begin
- Left := Synth_Implicit_Conv (Syn_Inst, Left, Ltype, Expr_Type);
- Right := Synth_Implicit_Conv (Syn_Inst, Right, Rtype, Expr_Type);
-
- L := Synth_Resize (Left, Etype.Drange.W, Expr);
- R := Synth_Resize (Right, Etype.Drange.W, Expr);
- N := Build_Dyadic (Build_Context, Id, L, R);
+ N := Build_Dyadic
+ (Build_Context, Id, Get_Net (Left), Get_Net (Right));
Set_Location (N, Expr);
return Create_Value_Net (N, Etype);
end Synth_Int_Dyadic;
@@ -849,15 +898,15 @@ package body Synth.Expr is
R1 : Net;
N : Net;
begin
- R1 := Synth_Uresize (Right, Rtype, Get_Width (Left));
+ R1 := Synth_Uresize (Right, Right_Type, Get_Width (Left));
Set_Location (R1, Expr);
N := Build_Dyadic (Build_Context, Id, L, R1);
Set_Location (N, Expr);
return Create_Value_Net (N, Create_Res_Bound (Left, L));
end Synth_Dyadic_Uns_Nat;
begin
- Left := Synth_Expression (Syn_Inst, Left_Expr);
- Right := Synth_Expression (Syn_Inst, Right_Expr);
+ Left := Synth_Expression_With_Type (Syn_Inst, Left_Expr, Left_Type);
+ Right := Synth_Expression_With_Type (Syn_Inst, Right_Expr, Right_Type);
case Def is
when Iir_Predefined_Error =>
@@ -892,8 +941,8 @@ package body Synth.Expr is
return Synth_Vec_Dyadic (Id_Xor);
when Iir_Predefined_Enum_Equality =>
- if Is_Bit_Type (Ltype) then
- pragma Assert (Is_Bit_Type (Rtype));
+ if Is_Bit_Type (Left_Type) then
+ pragma Assert (Is_Bit_Type (Right_Type));
if Is_Const (Left) then
return Synth_Bit_Eq_Const (Left, Right, Expr);
elsif Is_Const (Right) then
@@ -909,14 +958,14 @@ package body Synth.Expr is
when Iir_Predefined_Array_Equality =>
-- TODO: check size, handle non-vector.
- if Is_Vector_Type (Ltype) then
+ if Is_Vector_Type (Left_Type) then
return Synth_Compare (Id_Eq);
else
raise Internal_Error;
end if;
when Iir_Predefined_Array_Inequality =>
-- TODO: check size, handle non-vector.
- if Is_Vector_Type (Ltype) then
+ if Is_Vector_Type (Left_Type) then
return Synth_Compare (Id_Ne);
else
raise Internal_Error;
@@ -924,7 +973,7 @@ package body Synth.Expr is
when Iir_Predefined_Array_Greater =>
-- TODO: check size, non-vector.
-- TODO: that's certainly not the correct operator.
- if Is_Vector_Type (Ltype) then
+ if Is_Vector_Type (Left_Type) then
return Synth_Compare (Id_Ugt);
else
raise Internal_Error;
@@ -1979,7 +2028,9 @@ package body Synth.Expr is
function Synth_Expression_With_Type
(Syn_Inst : Synth_Instance_Acc; Expr : Node; Expr_Type : Node)
- return Value_Acc is
+ return Value_Acc
+ is
+ Res : Value_Acc;
begin
case Get_Kind (Expr) is
when Iir_Kinds_Dyadic_Operator =>
@@ -2002,7 +2053,7 @@ package body Synth.Expr is
or else Def in Iir_Predefined_IEEE_Explicit
then
return Synth_Dyadic_Operation
- (Syn_Inst, Def, Get_Left (Expr), Get_Right (Expr), Expr);
+ (Syn_Inst, Imp, Get_Left (Expr), Get_Right (Expr), Expr);
else
Error_Unknown_Operator (Imp, Expr);
raise Internal_Error;
@@ -2026,7 +2077,9 @@ package body Synth.Expr is
end;
when Iir_Kind_Simple_Name
| Iir_Kind_Interface_Signal_Declaration => -- For PSL...
- return Synth_Name (Syn_Inst, Expr);
+ Res := Synth_Name (Syn_Inst, Expr);
+ return Synth_Subtype_Conversion
+ (Res, Get_Value_Type (Syn_Inst, Expr_Type), Expr);
when Iir_Kind_Reference_Name =>
return Synth_Name (Syn_Inst, Get_Named_Entity (Expr));
when Iir_Kind_Indexed_Name =>