diff options
Diffstat (limited to 'src/synth/synth-context.adb')
-rw-r--r-- | src/synth/synth-context.adb | 182 |
1 files changed, 142 insertions, 40 deletions
diff --git a/src/synth/synth-context.adb b/src/synth/synth-context.adb index f89a708b1..7681d8f3b 100644 --- a/src/synth/synth-context.adb +++ b/src/synth/synth-context.adb @@ -234,6 +234,123 @@ package body Synth.Context is return Val.Typ; end Get_Value_Type; + function Vec2net (Val : Value_Acc) return Net is + begin + if Val.Typ.Vbound.Len <= 32 then + declare + Len : constant Iir_Index32 := Iir_Index32 (Val.Typ.Vbound.Len); + R_Val, R_Zx : Uns32; + V, Zx : Uns32; + begin + R_Val := 0; + R_Zx := 0; + for I in 1 .. Len loop + To_Logic (Val.Arr.V (I).Scal, Val.Typ.Vec_El, V, Zx); + R_Val := R_Val or Shift_Left (V, Natural (Len - I)); + R_Zx := R_Zx or Shift_Left (Zx, Natural (Len - I)); + end loop; + if R_Zx = 0 then + return Build_Const_UB32 (Build_Context, R_Val, Uns32 (Len)); + else + return Build_Const_UL32 + (Build_Context, R_Val, R_Zx, Uns32 (Len)); + end if; + end; + else + -- Need Uconst64 / UconstBig + raise Internal_Error; + end if; + end Vec2net; + + pragma Unreferenced (Vec2net); + + type Logic_32 is record + Val : Uns32; -- AKA aval + Zx : Uns32; -- AKA bval + end record; + + type Digit_Index is new Natural; + type Logvec_Array is array (Digit_Index range <>) of Logic_32; + + -- type Logvec_Array_Acc is access Logvec_Array; + + procedure Value2net (Val : Value_Acc; + Vec : in out Logvec_Array; + Off : in out Uns32; + Has_Zx : in out Boolean) is + begin + case Val.Typ.Kind is + when Type_Bit => + declare + Idx : constant Digit_Index := Digit_Index (Off / 32); + Pos : constant Natural := Natural (Off mod 32); + Va : Uns32; + Zx : Uns32; + begin + if Val.Typ = Logic_Type then + From_Std_Logic (Val.Scal, Va, Zx); + Has_Zx := Has_Zx or Zx /= 0; + else + Va := Uns32 (Val.Scal); + Zx := 0; + end if; + Va := Shift_Left (Va, Pos); + Zx := Shift_Left (Zx, Pos); + Vec (Idx).Val := Vec (Idx).Val or Va; + Vec (Idx).Zx := Vec (Idx).Zx or Zx; + Off := Off + 1; + end; + when Type_Vector => + -- TODO: optimize off mod 32 = 0. + for I in reverse Val.Arr.V'Range loop + Value2net (Val.Arr.V (I), Vec, Off, Has_Zx); + end loop; + when Type_Array => + for I in reverse Val.Arr.V'Range loop + Value2net (Val.Arr.V (I), Vec, Off, Has_Zx); + end loop; + when others => + raise Internal_Error; + end case; + end Value2net; + + procedure Value2net + (Val : Value_Acc; W : Width; Vec : in out Logvec_Array; Res : out Net) + is + Off : Uns32; + Has_Zx : Boolean; + Inst : Instance; + begin + Has_Zx := False; + Off := 0; + Value2net (Val, Vec, Off, Has_Zx); + if W <= 32 then + -- 32 bit result. + if not Has_Zx then + Res := Build_Const_UB32 (Build_Context, Vec (0).Val, W); + else + Res := Build_Const_UL32 + (Build_Context, Vec (0).Val, Vec (0).Zx, W); + end if; + return; + else + if not Has_Zx then + Inst := Build_Const_Bit (Build_Context, W); + for I in Vec'Range loop + Set_Param_Uns32 (Inst, Param_Idx (I), Vec (I).Val); + end loop; + Res := Get_Output (Inst, 0); + else + Inst := Build_Const_Log (Build_Context, W); + for I in Vec'Range loop + Set_Param_Uns32 (Inst, Param_Idx (2 * I), Vec (I).Val); + Set_Param_Uns32 (Inst, Param_Idx (2 * I + 1), Vec (I).Zx); + end loop; + Res := Get_Output (Inst, 0); + end if; + end if; + end Value2net; + function Get_Net (Val : Value_Acc) return Net is begin case Val.Kind is @@ -250,52 +367,37 @@ package body Synth.Context is I1 => Get_Net (Val.M_T)); end; when Value_Discrete => - declare - Va : Uns32; - Zx : Uns32; - begin - if Val.Typ = Logic_Type then - From_Std_Logic (Val.Scal, Va, Zx); - if Zx = 0 then - return Build_Const_UB32 (Build_Context, Va, 1); - else - return Build_Const_UL32 (Build_Context, Va, Zx, 1); - end if; - elsif Val.Typ = Boolean_Type then - From_Bit (Val.Scal, Va); - return Build_Const_UB32 (Build_Context, Va, 1); - else - return Build_Const_UB32 - (Build_Context, Uns32 (Val.Scal), Val.Typ.Drange.W); - end if; - end; - when Value_Array => - if Val.Typ.Vbound.Len <= 32 then + if Val.Typ.Kind = Type_Bit then declare - Len : constant Iir_Index32 := - Iir_Index32 (Val.Typ.Vbound.Len); - R_Val, R_Zx : Uns32; - V, Zx : Uns32; + V : Logvec_Array (0 .. 0) := (0 => (0, 0)); + Res : Net; begin - R_Val := 0; - R_Zx := 0; - for I in 1 .. Len loop - To_Logic (Val.Arr.V (I).Scal, Val.Typ.Vec_El, V, Zx); - R_Val := R_Val or Shift_Left (V, Natural (Len - I)); - R_Zx := R_Zx or Shift_Left (Zx, Natural (Len - I)); - end loop; - if R_Zx = 0 then - return Build_Const_UB32 - (Build_Context, R_Val, Uns32 (Len)); - else - return Build_Const_UL32 - (Build_Context, R_Val, R_Zx, Uns32 (Len)); - end if; + Value2net (Val, 1, V, Res); + return Res; end; + elsif Val.Typ.Drange.W <= 32 then + return Build_Const_UB32 + (Build_Context, Uns32 (Val.Scal), Val.Typ.Drange.W); else - -- Need Uconst64 / UconstBig raise Internal_Error; end if; + when Value_Array => + declare + W : constant Width := Get_Type_Width (Val.Typ); + Nd : constant Digit_Index := Digit_Index ((W + 31) / 32); + Res : Net; + begin + if Nd > 64 then + raise Internal_Error; + else + declare + Vec : Logvec_Array (0 .. Nd - 1) := (others => (0, 0)); + begin + Value2net (Val, W, Vec, Res); + return Res; + end; + end if; + end; when others => raise Internal_Error; end case; |