From e3a82f2e6894155cc030680332f31db6f79aba28 Mon Sep 17 00:00:00 2001 From: Tristan Gingold Date: Wed, 28 Apr 2021 17:37:26 +0200 Subject: synth: file renaming for decls, expr, insts and stmts. --- src/synth/synth-decls.adb | 1227 ------------ src/synth/synth-decls.ads | 79 - src/synth/synth-expr.adb | 2572 ------------------------- src/synth/synth-expr.ads | 152 -- src/synth/synth-insts.adb | 1751 ----------------- src/synth/synth-insts.ads | 47 - src/synth/synth-static_oper.adb | 2 +- src/synth/synth-stmts.adb | 3853 ------------------------------------- src/synth/synth-stmts.ads | 167 -- src/synth/synth-vhdl_aggr.adb | 6 +- src/synth/synth-vhdl_context.adb | 2 +- src/synth/synth-vhdl_decls.adb | 1227 ++++++++++++ src/synth/synth-vhdl_decls.ads | 79 + src/synth/synth-vhdl_expr.adb | 2572 +++++++++++++++++++++++++ src/synth/synth-vhdl_expr.ads | 152 ++ src/synth/synth-vhdl_files.adb | 2 +- src/synth/synth-vhdl_insts.adb | 1752 +++++++++++++++++ src/synth/synth-vhdl_insts.ads | 47 + src/synth/synth-vhdl_oper.adb | 4 +- src/synth/synth-vhdl_stmts.adb | 3856 ++++++++++++++++++++++++++++++++++++++ src/synth/synth-vhdl_stmts.ads | 167 ++ src/synth/synthesis.adb | 2 +- 22 files changed, 9861 insertions(+), 9857 deletions(-) delete mode 100644 src/synth/synth-decls.adb delete mode 100644 src/synth/synth-decls.ads delete mode 100644 src/synth/synth-expr.adb delete mode 100644 src/synth/synth-expr.ads delete mode 100644 src/synth/synth-insts.adb delete mode 100644 src/synth/synth-insts.ads delete mode 100644 src/synth/synth-stmts.adb delete mode 100644 src/synth/synth-stmts.ads create mode 100644 src/synth/synth-vhdl_decls.adb create mode 100644 src/synth/synth-vhdl_decls.ads create mode 100644 src/synth/synth-vhdl_expr.adb create mode 100644 src/synth/synth-vhdl_expr.ads create mode 100644 src/synth/synth-vhdl_insts.adb create mode 100644 src/synth/synth-vhdl_insts.ads create mode 100644 src/synth/synth-vhdl_stmts.adb create mode 100644 src/synth/synth-vhdl_stmts.ads diff --git a/src/synth/synth-decls.adb b/src/synth/synth-decls.adb deleted file mode 100644 index a8f92c1f9..000000000 --- a/src/synth/synth-decls.adb +++ /dev/null @@ -1,1227 +0,0 @@ --- Create declarations for synthesis. --- Copyright (C) 2017 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with Types; use Types; -with Mutils; use Mutils; -with Std_Names; - -with Netlists.Builders; use Netlists.Builders; -with Netlists.Folds; use Netlists.Folds; -with Netlists.Utils; use Netlists.Utils; -with Netlists.Gates; - -with Vhdl.Errors; -with Vhdl.Utils; use Vhdl.Utils; -with Vhdl.Std_Package; -with Vhdl.Ieee.Std_Logic_1164; - -with Synth.Vhdl_Environment; use Synth.Vhdl_Environment.Env; -with Synth.Expr; use Synth.Expr; -with Synth.Stmts; -with Synth.Source; use Synth.Source; -with Synth.Errors; use Synth.Errors; -with Synth.Vhdl_Files; -with Synth.Values; use Synth.Values; - -package body Synth.Decls is - procedure Create_Var_Wire - (Syn_Inst : Synth_Instance_Acc; Decl : Iir; Init : Valtyp) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Vt : constant Valtyp := Get_Value (Syn_Inst, Decl); - Value : Net; - Ival : Net; - W : Width; - Name : Sname; - begin - case Vt.Val.Kind is - when Value_Wire => - -- FIXME: get the width directly from the wire ? - W := Get_Type_Width (Vt.Typ); - Name := New_Sname_User (Get_Identifier (Decl), - Get_Sname (Syn_Inst)); - if Init /= No_Valtyp then - Ival := Get_Net (Ctxt, Init); - pragma Assert (Get_Width (Ival) = W); - Value := Build_Isignal (Ctxt, Name, Ival); - else - Value := Build_Signal (Ctxt, Name, W); - end if; - Set_Location (Value, Decl); - Set_Wire_Gate (Vt.Val.W, Value); - when others => - raise Internal_Error; - end case; - end Create_Var_Wire; - - function Type_To_Param_Type (Atype : Node) return Param_Type - is - use Vhdl.Std_Package; - Btype : constant Node := Get_Base_Type (Atype); - begin - if Btype = String_Type_Definition then - return Param_Pval_String; - elsif Btype = Time_Type_Definition then - return Param_Pval_Time_Ps; - else - case Get_Kind (Btype) is - when Iir_Kind_Integer_Type_Definition => - return Param_Pval_Integer; - when Iir_Kind_Floating_Type_Definition => - return Param_Pval_Real; - when others => - return Param_Pval_Vector; - end case; - end if; - end Type_To_Param_Type; - - function Memtyp_To_Pval (Mt : Memtyp) return Pval - is - Len : constant Uns32 := (Mt.Typ.W + 31) / 32; - Vec : Logvec_Array_Acc; - Off : Uns32; - Has_Zx : Boolean; - Pv : Pval; - begin - if Len = 0 then - return Create_Pval2 (0); - end if; - - Vec := new Logvec_Array'(0 .. Digit_Index (Len - 1) => (0, 0)); - Off := 0; - Has_Zx := False; - Value2logvec (Mt, 0, Mt.Typ.W, Vec.all, Off, Has_Zx); - pragma Assert (Off = Mt.Typ.W); - if Has_Zx then - Pv := Create_Pval4 (Mt.Typ.W); - else - Pv := Create_Pval2 (Mt.Typ.W); - end if; - for I in 0 .. Len - 1 loop - Write_Pval (Pv, I, Vec (Digit_Index (I))); - end loop; - Free_Logvec_Array (Vec); - return Pv; - end Memtyp_To_Pval; - - procedure Synth_Subtype_Indication_If_Anonymous - (Syn_Inst : Synth_Instance_Acc; Atype : Node) is - begin - if Get_Type_Declarator (Atype) = Null_Node then - Synth_Subtype_Indication (Syn_Inst, Atype); - end if; - end Synth_Subtype_Indication_If_Anonymous; - - function Synth_Subtype_Indication_If_Anonymous - (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc is - begin - if Get_Type_Declarator (Atype) = Null_Node then - return Synth_Subtype_Indication (Syn_Inst, Atype); - else - return Get_Subtype_Object (Syn_Inst, Atype); - end if; - end Synth_Subtype_Indication_If_Anonymous; - - function Synth_Array_Type_Definition - (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc - is - El_Type : constant Node := Get_Element_Subtype (Def); - Ndims : constant Natural := Get_Nbr_Dimensions (Def); - El_Typ : Type_Acc; - Typ : Type_Acc; - begin - Synth_Subtype_Indication_If_Anonymous (Syn_Inst, El_Type); - El_Typ := Get_Subtype_Object (Syn_Inst, El_Type); - - if El_Typ.Kind in Type_Nets and then Ndims = 1 then - Typ := Create_Unbounded_Vector (El_Typ); - else - Typ := Create_Unbounded_Array (Dim_Type (Ndims), El_Typ); - end if; - return Typ; - end Synth_Array_Type_Definition; - - function Synth_Record_Type_Definition - (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc - is - El_List : constant Node_Flist := Get_Elements_Declaration_List (Def); - Rec_Els : Rec_El_Array_Acc; - El : Node; - El_Type : Node; - El_Typ : Type_Acc; - begin - Rec_Els := Create_Rec_El_Array - (Iir_Index32 (Get_Nbr_Elements (El_List))); - - for I in Flist_First .. Flist_Last (El_List) loop - El := Get_Nth_Element (El_List, I); - El_Type := Get_Type (El); - El_Typ := Synth_Subtype_Indication_If_Anonymous (Syn_Inst, El_Type); - Rec_Els.E (Iir_Index32 (I + 1)).Typ := El_Typ; - end loop; - - if not Is_Fully_Constrained_Type (Def) then - return Create_Unbounded_Record (Rec_Els); - else - return Create_Record_Type (Rec_Els); - end if; - end Synth_Record_Type_Definition; - - function Synth_Access_Type_Definition - (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc - is - Des_Type : constant Node := Get_Designated_Type (Def); - Des_Typ : Type_Acc; - Typ : Type_Acc; - begin - Synth_Subtype_Indication_If_Anonymous (Syn_Inst, Des_Type); - Des_Typ := Get_Subtype_Object (Syn_Inst, Des_Type); - - Typ := Create_Access_Type (Des_Typ); - return Typ; - end Synth_Access_Type_Definition; - - function Synth_File_Type_Definition - (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc - is - File_Type : constant Node := Get_Type (Get_File_Type_Mark (Def)); - File_Typ : Type_Acc; - Typ : Type_Acc; - Sig : String_Acc; - begin - File_Typ := Get_Subtype_Object (Syn_Inst, File_Type); - - if Get_Text_File_Flag (Def) - or else - Get_Kind (File_Type) in Iir_Kinds_Scalar_Type_And_Subtype_Definition - then - Sig := null; - else - declare - Sig_Str : String (1 .. Get_File_Signature_Length (File_Type) + 2); - Off : Natural := Sig_Str'First; - begin - Get_File_Signature (File_Type, Sig_Str, Off); - Sig_Str (Off + 0) := '.'; - Sig_Str (Off + 1) := ASCII.NUL; - Sig := new String'(Sig_Str); - end; - end if; - - Typ := Create_File_Type (File_Typ); - Typ.File_Signature := Sig; - - return Typ; - end Synth_File_Type_Definition; - - function Scalar_Size_To_Size (Def : Node) return Size_Type is - begin - case Get_Scalar_Size (Def) is - when Scalar_8 => - return 1; - when Scalar_16 => - return 2; - when Scalar_32 => - return 4; - when Scalar_64 => - return 8; - end case; - end Scalar_Size_To_Size; - - procedure Synth_Type_Definition (Syn_Inst : Synth_Instance_Acc; Def : Node) - is - Typ : Type_Acc; - begin - case Get_Kind (Def) is - when Iir_Kind_Enumeration_Type_Definition => - if Def = Vhdl.Ieee.Std_Logic_1164.Std_Ulogic_Type - or else Def = Vhdl.Ieee.Std_Logic_1164.Std_Logic_Type - then - Typ := Logic_Type; - elsif Def = Vhdl.Std_Package.Boolean_Type_Definition then - Typ := Boolean_Type; - elsif Def = Vhdl.Std_Package.Bit_Type_Definition then - Typ := Bit_Type; - else - declare - Nbr_El : constant Natural := - Get_Nbr_Elements (Get_Enumeration_Literal_List (Def)); - Rng : Discrete_Range_Type; - W : Width; - begin - W := Uns32 (Clog2 (Uns64 (Nbr_El))); - Rng := (Dir => Dir_To, - Is_Signed => False, - Left => 0, - Right => Int64 (Nbr_El - 1)); - Typ := Create_Discrete_Type - (Rng, Scalar_Size_To_Size (Def), W); - end; - end if; - when Iir_Kind_Array_Type_Definition => - Typ := Synth_Array_Type_Definition (Syn_Inst, Def); - when Iir_Kind_Access_Type_Definition => - Typ := Synth_Access_Type_Definition (Syn_Inst, Def); - when Iir_Kind_File_Type_Definition => - Typ := Synth_File_Type_Definition (Syn_Inst, Def); - when Iir_Kind_Record_Type_Definition => - Typ := Synth_Record_Type_Definition (Syn_Inst, Def); - when Iir_Kind_Protected_Type_Declaration => - Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Def)); - when others => - Vhdl.Errors.Error_Kind ("synth_type_definition", Def); - end case; - if Typ /= null then - Create_Subtype_Object (Syn_Inst, Def, Typ); - end if; - end Synth_Type_Definition; - - procedure Synth_Anonymous_Type_Definition - (Syn_Inst : Synth_Instance_Acc; Def : Node; St : Node) - is - Typ : Type_Acc; - begin - case Get_Kind (Def) is - when Iir_Kind_Integer_Type_Definition - | Iir_Kind_Physical_Type_Definition => - declare - Cst : constant Node := Get_Range_Constraint (St); - L, R : Int64; - Rng : Discrete_Range_Type; - W : Width; - begin - L := Get_Value (Get_Left_Limit (Cst)); - R := Get_Value (Get_Right_Limit (Cst)); - Rng := Build_Discrete_Range_Type (L, R, Get_Direction (Cst)); - W := Discrete_Range_Width (Rng); - Typ := Create_Discrete_Type - (Rng, Scalar_Size_To_Size (Def), W); - end; - when Iir_Kind_Floating_Type_Definition => - declare - Cst : constant Node := Get_Range_Constraint (St); - L, R : Fp64; - Rng : Float_Range_Type; - begin - L := Get_Fp_Value (Get_Left_Limit (Cst)); - R := Get_Fp_Value (Get_Right_Limit (Cst)); - Rng := (Get_Direction (Cst), L, R); - Typ := Create_Float_Type (Rng); - end; - when Iir_Kind_Array_Type_Definition => - Typ := Synth_Array_Type_Definition (Syn_Inst, Def); - when others => - Vhdl.Errors.Error_Kind ("synth_anonymous_type_definition", Def); - end case; - Create_Subtype_Object (Syn_Inst, Def, Typ); - end Synth_Anonymous_Type_Definition; - - function Synth_Discrete_Range_Constraint - (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Discrete_Range_Type - is - Res : Discrete_Range_Type; - begin - Synth_Discrete_Range (Syn_Inst, Rng, Res); - return Res; - end Synth_Discrete_Range_Constraint; - - function Synth_Float_Range_Constraint - (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Float_Range_Type is - begin - case Get_Kind (Rng) is - when Iir_Kind_Range_Expression => - -- FIXME: check range. - return Synth_Float_Range_Expression (Syn_Inst, Rng); - when others => - Vhdl.Errors.Error_Kind ("synth_float_range_constraint", Rng); - end case; - end Synth_Float_Range_Constraint; - - function Has_Element_Subtype_Indication (Atype : Node) return Boolean is - begin - return Get_Array_Element_Constraint (Atype) /= Null_Node - or else - (Get_Resolution_Indication (Atype) /= Null_Node - and then - (Get_Kind (Get_Resolution_Indication (Atype)) - = Iir_Kind_Array_Element_Resolution)); - end Has_Element_Subtype_Indication; - - function Synth_Array_Subtype_Indication - (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc - is - El_Type : constant Node := Get_Element_Subtype (Atype); - St_Indexes : constant Node_Flist := Get_Index_Subtype_List (Atype); - Ptype : Node; - St_El : Node; - Btyp : Type_Acc; - Etyp : Type_Acc; - Bnds : Bound_Array_Acc; - begin - -- VHDL08 - if Has_Element_Subtype_Indication (Atype) then - -- This subtype has created a new anonymous subtype for the - -- element. - Synth_Subtype_Indication (Syn_Inst, El_Type); - end if; - - if not Get_Index_Constraint_Flag (Atype) then - Ptype := Get_Type (Get_Subtype_Type_Mark (Atype)); - if Get_Element_Subtype (Ptype) = Get_Element_Subtype (Atype) then - -- That's an alias. - -- FIXME: maybe a resolution function was added? - -- FIXME: also handle resolution added in element subtype. - return Get_Subtype_Object (Syn_Inst, Ptype); - end if; - end if; - - Btyp := Get_Subtype_Object (Syn_Inst, Get_Base_Type (Atype)); - case Btyp.Kind is - when Type_Unbounded_Vector => - if Get_Index_Constraint_Flag (Atype) then - St_El := Get_Index_Type (St_Indexes, 0); - return Create_Vector_Type - (Synth_Bounds_From_Range (Syn_Inst, St_El), Btyp.Uvec_El); - else - -- An alias. - -- Handle vhdl08 definition of std_logic_vector from - -- std_ulogic_vector. - return Btyp; - end if; - when Type_Unbounded_Array => - -- FIXME: partially constrained arrays, subtype in indexes... - Etyp := Get_Subtype_Object (Syn_Inst, El_Type); - if Get_Index_Constraint_Flag (Atype) then - Bnds := Create_Bound_Array - (Dim_Type (Get_Nbr_Elements (St_Indexes))); - for I in Flist_First .. Flist_Last (St_Indexes) loop - St_El := Get_Index_Type (St_Indexes, I); - Bnds.D (Dim_Type (I + 1)) := - Synth_Bounds_From_Range (Syn_Inst, St_El); - end loop; - return Create_Array_Type (Bnds, Etyp); - else - raise Internal_Error; - end if; - when others => - raise Internal_Error; - end case; - end Synth_Array_Subtype_Indication; - - function Synth_Subtype_Indication - (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc is - begin - -- TODO: handle aliases directly. - case Get_Kind (Atype) is - when Iir_Kind_Array_Subtype_Definition => - return Synth_Array_Subtype_Indication (Syn_Inst, Atype); - when Iir_Kind_Record_Subtype_Definition => - return Synth_Record_Type_Definition (Syn_Inst, Atype); - when Iir_Kind_Integer_Subtype_Definition - | Iir_Kind_Physical_Subtype_Definition - | Iir_Kind_Enumeration_Subtype_Definition => - declare - Btype : constant Type_Acc := - Get_Subtype_Object (Syn_Inst, Get_Base_Type (Atype)); - Rng : Discrete_Range_Type; - W : Width; - begin - if Btype.Kind in Type_Nets then - -- A subtype of a bit/logic type is still a bit/logic. - -- FIXME: bounds. - return Btype; - else - Rng := Synth_Discrete_Range_Constraint - (Syn_Inst, Get_Range_Constraint (Atype)); - W := Discrete_Range_Width (Rng); - return Create_Discrete_Type (Rng, Btype.Sz, W); - end if; - end; - when Iir_Kind_Floating_Subtype_Definition => - declare - Rng : Float_Range_Type; - begin - Rng := Synth_Float_Range_Constraint - (Syn_Inst, Get_Range_Constraint (Atype)); - return Create_Float_Type (Rng); - end; - when others => - Vhdl.Errors.Error_Kind ("synth_subtype_indication", Atype); - end case; - end Synth_Subtype_Indication; - - procedure Synth_Subtype_Indication - (Syn_Inst : Synth_Instance_Acc; Atype : Node) - is - Typ : Type_Acc; - begin - Typ := Synth_Subtype_Indication (Syn_Inst, Atype); - Create_Subtype_Object (Syn_Inst, Atype, Typ); - end Synth_Subtype_Indication; - - function Get_Declaration_Type (Decl : Node) return Node - is - Ind : constant Node := Get_Subtype_Indication (Decl); - Atype : Node; - begin - if Get_Is_Ref (Decl) or else Ind = Null_Iir then - -- A secondary declaration in a list. - return Null_Node; - end if; - Atype := Ind; - loop - case Get_Kind (Atype) is - when Iir_Kinds_Denoting_Name => - Atype := Get_Named_Entity (Atype); - when Iir_Kind_Subtype_Declaration - | Iir_Kind_Type_Declaration => - -- Type already declared, so already handled. - return Null_Node; - when Iir_Kind_Array_Subtype_Definition - | Iir_Kind_Record_Subtype_Definition - | Iir_Kind_Integer_Subtype_Definition - | Iir_Kind_Floating_Subtype_Definition - | Iir_Kind_Physical_Subtype_Definition - | Iir_Kind_Enumeration_Subtype_Definition => - return Atype; - when others => - Vhdl.Errors.Error_Kind ("get_declaration_type", Atype); - end case; - end loop; - end Get_Declaration_Type; - - procedure Synth_Declaration_Type - (Syn_Inst : Synth_Instance_Acc; Decl : Node) - is - Atype : constant Node := Get_Declaration_Type (Decl); - begin - if Atype = Null_Node then - return; - end if; - Synth_Subtype_Indication (Syn_Inst, Atype); - end Synth_Declaration_Type; - - procedure Synth_Constant_Declaration (Syn_Inst : Synth_Instance_Acc; - Decl : Node; - Is_Subprg : Boolean; - Last_Type : in out Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Deferred_Decl : constant Node := Get_Deferred_Declaration (Decl); - First_Decl : Node; - Decl_Type : Node; - Val : Valtyp; - Cst : Valtyp; - Obj_Type : Type_Acc; - begin - Synth_Declaration_Type (Syn_Inst, Decl); - if Deferred_Decl = Null_Node - or else Get_Deferred_Declaration_Flag (Decl) - then - -- Create the object (except for full declaration of a - -- deferred constant). - Create_Object (Syn_Inst, Decl, No_Valtyp); - end if; - -- Initialize the value (except for a deferred declaration). - if Get_Deferred_Declaration_Flag (Decl) then - return; - end if; - if Deferred_Decl = Null_Node then - -- A normal constant declaration - First_Decl := Decl; - else - -- The full declaration of a deferred constant. - First_Decl := Deferred_Decl; - end if; - pragma Assert (First_Decl /= Null_Node); - - -- Use the type of the declaration. The type of the constant may - -- be derived from the value. - -- FIXME: what about multiple declarations ? - Decl_Type := Get_Subtype_Indication (Decl); - if Decl_Type = Null_Node then - Decl_Type := Last_Type; - else - if Get_Kind (Decl_Type) in Iir_Kinds_Denoting_Name then - -- Type mark. - Decl_Type := Get_Type (Get_Named_Entity (Decl_Type)); - end if; - Last_Type := Decl_Type; - end if; - Obj_Type := Get_Subtype_Object (Syn_Inst, Decl_Type); - Val := Synth_Expression_With_Type - (Syn_Inst, Get_Default_Value (Decl), Obj_Type); - if Val = No_Valtyp then - Set_Error (Syn_Inst); - return; - end if; - Val := Synth_Subtype_Conversion (Ctxt, Val, Obj_Type, True, Decl); - -- For constant functions, the value must be constant. - pragma Assert (not Get_Instance_Const (Syn_Inst) - or else Is_Static (Val.Val)); - case Val.Val.Kind is - when Value_Const - | Value_Alias => - Cst := Val; - when others => - if Is_Static (Val.Val) then - Cst := Create_Value_Const (Val, Decl); - else - if not Is_Subprg then - Error_Msg_Synth - (+Decl, "signals cannot be used in default value " - & "of this constant"); - end if; - Cst := Val; - end if; - end case; - Create_Object_Force (Syn_Inst, First_Decl, Cst); - end Synth_Constant_Declaration; - - procedure Synth_Attribute_Object (Syn_Inst : Synth_Instance_Acc; - Attr_Value : Node; - Attr_Decl : Node; - Val : Valtyp) - is - Obj : constant Node := Get_Designated_Entity (Attr_Value); - Id : constant Name_Id := Get_Identifier (Attr_Decl); - Inst : Instance; - V : Valtyp; - Ptype : Param_Type; - Pv : Pval; - begin - if Id = Std_Names.Name_Foreign then - -- Not for synthesis. - return; - end if; - - case Get_Kind (Obj) is - when Iir_Kind_Signal_Declaration - | Iir_Kind_Variable_Declaration - | Iir_Kind_Interface_Signal_Declaration => - V := Get_Value (Syn_Inst, Obj); - pragma Assert (V.Val.Kind = Value_Wire); - Inst := Get_Net_Parent (Get_Wire_Gate (V.Val.W)); - when Iir_Kind_Component_Instantiation_Statement => - -- TODO - return; - when others => - -- TODO: components ? - -- TODO: Interface_Signal ? But no instance for them. - Warning_Msg_Synth - (+Attr_Value, "attribute %i for %n is not kept in the netlist", - (+Attr_Decl, +Obj)); - return; - end case; - - Ptype := Type_To_Param_Type (Get_Type (Attr_Decl)); - Pv := Memtyp_To_Pval (Get_Memtyp (Val)); - - Set_Attribute (Inst, Id, Ptype, Pv); - end Synth_Attribute_Object; - - procedure Synth_Attribute_Specification - (Syn_Inst : Synth_Instance_Acc; Spec : Node) - is - Attr_Decl : constant Node := - Get_Named_Entity (Get_Attribute_Designator (Spec)); - Value : Node; - Val : Valtyp; - Val_Type : Type_Acc; - begin - Val_Type := Get_Subtype_Object (Syn_Inst, Get_Type (Attr_Decl)); - Value := Get_Attribute_Value_Spec_Chain (Spec); - while Value /= Null_Iir loop - -- 2. The expression is evaluated to determine the value - -- of the attribute. - -- It is an error if the value of the expression does not - -- belong to the subtype of the attribute; if the - -- attribute is of an array type, then an implicit - -- subtype conversion is first performed on the value, - -- unless the attribute's subtype indication denotes an - -- unconstrained array type. - Val := Synth_Expression_With_Type - (Syn_Inst, Get_Expression (Spec), Val_Type); - -- Check_Constraints (Instance, Val, Attr_Type, Decl); - - -- 3. A new instance of the designated attribute is created - -- and associated with each of the affected items. - -- - -- 4. Each new attribute instance is assigned the value of - -- the expression. - Create_Object (Syn_Inst, Value, Val); - -- Unshare (Val, Instance_Pool); - - if not Get_Instance_Const (Syn_Inst) then - Synth_Attribute_Object (Syn_Inst, Value, Attr_Decl, Val); - end if; - - Value := Get_Spec_Chain (Value); - end loop; - end Synth_Attribute_Specification; - - procedure Synth_Subprogram_Declaration - (Syn_Inst : Synth_Instance_Acc; Subprg : Node) - is - Inter : Node; - begin - if Is_Second_Subprogram_Specification (Subprg) then - -- Already handled. - return; - end if; - - Inter := Get_Interface_Declaration_Chain (Subprg); - while Inter /= Null_Node loop - Synth_Declaration_Type (Syn_Inst, Inter); - Inter := Get_Chain (Inter); - end loop; - end Synth_Subprogram_Declaration; - - procedure Synth_Convertible_Declarations (Syn_Inst : Synth_Instance_Acc) - is - use Vhdl.Std_Package; - begin - Create_Subtype_Object - (Syn_Inst, Convertible_Integer_Type_Definition, - Get_Subtype_Object (Syn_Inst, Universal_Integer_Type_Definition)); - Create_Subtype_Object - (Syn_Inst, Convertible_Real_Type_Definition, - Get_Subtype_Object (Syn_Inst, Universal_Real_Type_Definition)); - end Synth_Convertible_Declarations; - - function Create_Package_Instance (Parent_Inst : Synth_Instance_Acc; - Pkg : Node) - return Synth_Instance_Acc - is - Syn_Inst : Synth_Instance_Acc; - begin - Syn_Inst := Make_Instance (Parent_Inst, Pkg); - if Get_Kind (Get_Parent (Pkg)) = Iir_Kind_Design_Unit then - -- Global package. - Create_Package_Object (Parent_Inst, Pkg, Syn_Inst, True); - else - -- Local package: check elaboration order. - Create_Package_Object (Parent_Inst, Pkg, Syn_Inst, False); - end if; - return Syn_Inst; - end Create_Package_Instance; - - procedure Synth_Package_Declaration - (Parent_Inst : Synth_Instance_Acc; Pkg : Node) - is - Syn_Inst : Synth_Instance_Acc; - begin - if Is_Uninstantiated_Package (Pkg) then - -- Nothing to do (yet) for uninstantiated packages. - return; - end if; - - Syn_Inst := Create_Package_Instance (Parent_Inst, Pkg); - - Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Pkg)); - if Pkg = Vhdl.Std_Package.Standard_Package then - Synth_Convertible_Declarations (Syn_Inst); - end if; - end Synth_Package_Declaration; - - procedure Synth_Package_Body - (Parent_Inst : Synth_Instance_Acc; Pkg : Node; Bod : Node) - is - Pkg_Inst : Synth_Instance_Acc; - begin - if Is_Uninstantiated_Package (Pkg) then - -- Nothing to do (yet) for uninstantiated packages. - return; - end if; - - Pkg_Inst := Get_Package_Object (Parent_Inst, Pkg); - - Synth_Declarations (Pkg_Inst, Get_Declaration_Chain (Bod)); - end Synth_Package_Body; - - procedure Synth_Generics_Association (Sub_Inst : Synth_Instance_Acc; - Syn_Inst : Synth_Instance_Acc; - Inter_Chain : Node; - Assoc_Chain : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Inter : Node; - Inter_Type : Type_Acc; - Assoc : Node; - Assoc_Inter : Node; - Actual : Node; - Val : Valtyp; - begin - Assoc := Assoc_Chain; - Assoc_Inter := Inter_Chain; - while Is_Valid (Assoc) loop - Inter := Get_Association_Interface (Assoc, Assoc_Inter); - case Iir_Kinds_Interface_Declaration (Get_Kind (Inter)) is - when Iir_Kind_Interface_Constant_Declaration => - Synth_Declaration_Type (Sub_Inst, Inter); - Inter_Type := Get_Subtype_Object (Sub_Inst, Get_Type (Inter)); - - case Get_Kind (Assoc) is - when Iir_Kind_Association_Element_Open => - Actual := Get_Default_Value (Inter); - Val := Synth_Expression_With_Type - (Sub_Inst, Actual, Inter_Type); - when Iir_Kind_Association_Element_By_Expression => - Actual := Get_Actual (Assoc); - Val := Synth_Expression_With_Type - (Syn_Inst, Actual, Inter_Type); - when others => - raise Internal_Error; - end case; - - Val := Synth_Subtype_Conversion - (Ctxt, Val, Inter_Type, True, Assoc); - - if Val = No_Valtyp then - Set_Error (Sub_Inst); - elsif not Is_Static (Val.Val) then - Error_Msg_Synth - (+Assoc, "value of generic %i must be static", +Inter); - Val := No_Valtyp; - Set_Error (Sub_Inst); - end if; - - Create_Object (Sub_Inst, Inter, Val); - - when Iir_Kind_Interface_Package_Declaration => - declare - Actual : constant Iir := - Strip_Denoting_Name (Get_Actual (Assoc)); - Pkg_Inst : Synth_Instance_Acc; - begin - Pkg_Inst := Get_Package_Object (Sub_Inst, Actual); - Create_Package_Interface (Sub_Inst, Inter, Pkg_Inst); - end; - - when Iir_Kind_Interface_Variable_Declaration - | Iir_Kind_Interface_File_Declaration - | Iir_Kind_Interface_Signal_Declaration - | Iir_Kind_Interface_Quantity_Declaration - | Iir_Kind_Interface_Terminal_Declaration => - raise Internal_Error; - - when Iir_Kinds_Interface_Subprogram_Declaration - | Iir_Kind_Interface_Type_Declaration => - raise Internal_Error; - end case; - - Next_Association_Interface (Assoc, Assoc_Inter); - end loop; - end Synth_Generics_Association; - - procedure Synth_Package_Instantiation - (Parent_Inst : Synth_Instance_Acc; Pkg : Node) - is - Bod : constant Node := Get_Instance_Package_Body (Pkg); - Sub_Inst : Synth_Instance_Acc; - begin - Sub_Inst := Create_Package_Instance (Parent_Inst, Pkg); - - Synth_Generics_Association - (Sub_Inst, Parent_Inst, - Get_Generic_Chain (Pkg), Get_Generic_Map_Aspect_Chain (Pkg)); - - Synth_Declarations (Sub_Inst, Get_Declaration_Chain (Pkg)); - - if Bod /= Null_Node then - -- Macro expanded package instantiation. - raise Internal_Error; - else - -- Shared body - declare - Uninst : constant Node := Get_Uninstantiated_Package_Decl (Pkg); - Uninst_Bod : constant Node := Get_Package_Body (Uninst); - begin - Set_Uninstantiated_Scope (Sub_Inst, Uninst); - -- Synth declarations of (optional) body. - if Uninst_Bod /= Null_Node then - Synth_Declarations - (Sub_Inst, Get_Declaration_Chain (Uninst_Bod)); - end if; - end; - end if; - end Synth_Package_Instantiation; - - procedure Synth_Variable_Declaration (Syn_Inst : Synth_Instance_Acc; - Decl : Node; - Is_Subprg : Boolean) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Def : constant Node := Get_Default_Value (Decl); - Decl_Type : constant Node := Get_Type (Decl); - Init : Valtyp; - Obj_Typ : Type_Acc; - Wid : Wire_Id; - begin - Synth_Declaration_Type (Syn_Inst, Decl); - if Get_Kind (Decl_Type) = Iir_Kind_Protected_Type_Declaration then - Error_Msg_Synth - (+Decl, "protected type variable is not synthesizable"); - Set_Error (Syn_Inst); - Create_Object (Syn_Inst, Decl, No_Valtyp); - return; - end if; - - Obj_Typ := Get_Subtype_Object (Syn_Inst, Decl_Type); - if not Obj_Typ.Is_Synth - and then not Get_Instance_Const (Syn_Inst) - then - Error_Msg_Synth - (+Decl, "variable with access type is not synthesizable"); - -- FIXME: use a poison value ? - Create_Object (Syn_Inst, Decl, Create_Value_Default (Obj_Typ)); - else - if Is_Valid (Def) then - Init := Synth_Expression_With_Type (Syn_Inst, Def, Obj_Typ); - Init := Synth_Subtype_Conversion - (Ctxt, Init, Obj_Typ, False, Decl); - if not Is_Subprg - and then not Is_Static (Init.Val) - then - Error_Msg_Synth - (+Decl, "signals cannot be used in default value of " - & "this variable"); - end if; - else - Init := Create_Value_Default (Obj_Typ); - end if; - if Get_Instance_Const (Syn_Inst) then - Init := Strip_Alias_Const (Init); - Init := Unshare (Init, Current_Pool); - Create_Object (Syn_Inst, Decl, Init); - else - Create_Wire_Object (Syn_Inst, Wire_Variable, Decl); - Create_Var_Wire (Syn_Inst, Decl, Init); - Wid := Get_Value (Syn_Inst, Decl).Val.W; - if Is_Subprg then - if Is_Static (Init.Val) then - Phi_Assign_Static (Wid, Get_Memtyp (Init)); - else - Phi_Assign_Net (Ctxt, Wid, Get_Net (Ctxt, Init), 0); - end if; - end if; - end if; - end if; - end Synth_Variable_Declaration; - - procedure Synth_Signal_Declaration (Syn_Inst : Synth_Instance_Acc; - Decl : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Def : constant Iir := Get_Default_Value (Decl); - -- Slot : constant Object_Slot_Type := Get_Info (Decl).Slot; - Init : Valtyp; - Obj_Typ : Type_Acc; - begin - Synth_Declaration_Type (Syn_Inst, Decl); - if Get_Kind (Get_Parent (Decl)) = Iir_Kind_Package_Declaration then - Error_Msg_Synth (+Decl, "signals in packages are not supported"); - -- Avoid elaboration error. - Create_Object (Syn_Inst, Decl, No_Valtyp); - return; - end if; - - Create_Wire_Object (Syn_Inst, Wire_Signal, Decl); - if Is_Valid (Def) then - Obj_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Decl)); - Init := Synth_Expression_With_Type (Syn_Inst, Def, Obj_Typ); - Init := Synth_Subtype_Conversion (Ctxt, Init, Obj_Typ, False, Decl); - if not Is_Static (Init.Val) then - Error_Msg_Synth (+Decl, "signals cannot be used in default value " - & "of a signal"); - end if; - else - Init := No_Valtyp; - end if; - Create_Var_Wire (Syn_Inst, Decl, Init); - end Synth_Signal_Declaration; - - procedure Synth_Object_Alias_Declaration - (Syn_Inst : Synth_Instance_Acc; Decl : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Atype : constant Node := Get_Declaration_Type (Decl); - Off : Value_Offsets; - Dyn : Stmts.Dyn_Name; - Res : Valtyp; - Obj_Typ : Type_Acc; - Base : Valtyp; - Typ : Type_Acc; - begin - -- Subtype indication may not be present. - if Atype /= Null_Node then - Synth_Subtype_Indication (Syn_Inst, Atype); - Obj_Typ := Get_Subtype_Object (Syn_Inst, Atype); - else - Obj_Typ := null; - end if; - - Stmts.Synth_Assignment_Prefix (Syn_Inst, Get_Name (Decl), - Base, Typ, Off, Dyn); - pragma Assert (Dyn.Voff = No_Net); - if Base.Val.Kind = Value_Net then - -- Object is a net if it is not writable. Extract the - -- bits for the alias. - Res := Create_Value_Net - (Build2_Extract (Ctxt, Base.Val.N, Off.Net_Off, Typ.W), - Typ); - else - Res := Create_Value_Alias (Base, Off, Typ); - end if; - if Obj_Typ /= null then - Res := Synth_Subtype_Conversion (Ctxt, Res, Obj_Typ, True, Decl); - end if; - Create_Object (Syn_Inst, Decl, Res); - end Synth_Object_Alias_Declaration; - - procedure Synth_Declaration (Syn_Inst : Synth_Instance_Acc; - Decl : Node; - Is_Subprg : Boolean; - Last_Type : in out Node) is - begin - case Get_Kind (Decl) is - when Iir_Kind_Variable_Declaration => - Synth_Variable_Declaration (Syn_Inst, Decl, Is_Subprg); - when Iir_Kind_Interface_Variable_Declaration => - -- Ignore default value. - Create_Wire_Object (Syn_Inst, Wire_Variable, Decl); - Create_Var_Wire (Syn_Inst, Decl, No_Valtyp); - when Iir_Kind_Constant_Declaration => - Synth_Constant_Declaration (Syn_Inst, Decl, Is_Subprg, Last_Type); - when Iir_Kind_Signal_Declaration => - pragma Assert (not Is_Subprg); - Synth_Signal_Declaration (Syn_Inst, Decl); - when Iir_Kind_Object_Alias_Declaration => - Synth_Object_Alias_Declaration (Syn_Inst, Decl); - when Iir_Kind_Anonymous_Signal_Declaration => - -- Anonymous signals created by inertial associations are - -- simply ignored. - null; - when Iir_Kind_Procedure_Declaration - | Iir_Kind_Function_Declaration => - Synth_Subprogram_Declaration (Syn_Inst, Decl); - when Iir_Kind_Procedure_Body - | Iir_Kind_Function_Body => - null; - when Iir_Kind_Non_Object_Alias_Declaration => - null; - when Iir_Kind_Attribute_Declaration => - -- Nothing to do: the type is a type_mark, not a subtype - -- indication. - null; - when Iir_Kind_Attribute_Specification => - Synth_Attribute_Specification (Syn_Inst, Decl); - when Iir_Kind_Type_Declaration => - Synth_Type_Definition (Syn_Inst, Get_Type_Definition (Decl)); - when Iir_Kind_Anonymous_Type_Declaration => - Synth_Anonymous_Type_Definition - (Syn_Inst, Get_Type_Definition (Decl), - Get_Subtype_Definition (Decl)); - when Iir_Kind_Subtype_Declaration => - Synth_Declaration_Type (Syn_Inst, Decl); - when Iir_Kind_Component_Declaration => - null; - when Iir_Kind_File_Declaration => - declare - F : File_Index; - Res : Valtyp; - Obj_Typ : Type_Acc; - begin - F := Synth.Vhdl_Files.Elaborate_File_Declaration - (Syn_Inst, Decl); - Obj_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Decl)); - Res := Create_Value_File (Obj_Typ, F); - Create_Object (Syn_Inst, Decl, Res); - end; - when Iir_Kind_Protected_Type_Body => - null; - when Iir_Kind_Psl_Default_Clock => - -- Ignored; directly used by PSL directives. - null; - when Iir_Kind_Use_Clause => - null; - when Iir_Kind_Configuration_Specification => - null; - when Iir_Kind_Signal_Attribute_Declaration => - -- Not supported by synthesis. - null; - when others => - Vhdl.Errors.Error_Kind ("synth_declaration", Decl); - end case; - end Synth_Declaration; - - procedure Synth_Declarations (Syn_Inst : Synth_Instance_Acc; - Decls : Iir; - Is_Subprg : Boolean := False) - is - Decl : Node; - Last_Type : Node; - begin - Last_Type := Null_Node; - Decl := Decls; - while Is_Valid (Decl) loop - Synth_Declaration (Syn_Inst, Decl, Is_Subprg, Last_Type); - - exit when Is_Error (Syn_Inst); - - Decl := Get_Chain (Decl); - end loop; - end Synth_Declarations; - - procedure Finalize_Signal (Syn_Inst : Synth_Instance_Acc; Decl : Node) - is - use Netlists.Gates; - Vt : Valtyp; - Gate_Net : Net; - Gate : Instance; - Drv : Net; - Def_Val : Net; - begin - Vt := Get_Value (Syn_Inst, Decl); - if Vt = No_Valtyp then - pragma Assert (Is_Error (Syn_Inst)); - return; - end if; - if Vt.Val.Kind = Value_Net then - -- Could be a net for in ports. - return; - end if; - - Finalize_Assignment (Get_Build (Syn_Inst), Vt.Val.W); - - Gate_Net := Get_Wire_Gate (Vt.Val.W); - Gate := Get_Net_Parent (Gate_Net); - case Get_Id (Gate) is - when Id_Signal - | Id_Output - | Id_Inout => - Drv := Get_Input_Net (Gate, 0); - Def_Val := No_Net; - when Id_Isignal - | Id_Ioutput - | Id_Iinout => - Drv := Get_Input_Net (Gate, 0); - Def_Val := Get_Input_Net (Gate, 1); - when others => - -- Todo: output ? - raise Internal_Error; - end case; - if Drv = No_Net then - if Is_Connected (Get_Output (Gate, 0)) then - -- No warning if the signal is not used. - -- TODO: maybe simply remove it. - if Def_Val = No_Net then - Warning_Msg_Synth - (+Decl, "%n is never assigned and has no default value", - (1 => +Decl)); - else - Warning_Msg_Synth (+Decl, "%n is never assigned", (1 => +Decl)); - end if; - end if; - if Def_Val = No_Net then - Def_Val := Build_Const_X (Get_Build (Syn_Inst), - Get_Width (Gate_Net)); - end if; - Connect (Get_Input (Gate, 0), Def_Val); - end if; - - Free_Wire (Vt.Val.W); - end Finalize_Signal; - - procedure Finalize_Declaration - (Syn_Inst : Synth_Instance_Acc; Decl : Node; Is_Subprg : Boolean) is - begin - case Get_Kind (Decl) is - when Iir_Kind_Variable_Declaration - | Iir_Kind_Interface_Variable_Declaration => - if not Get_Instance_Const (Syn_Inst) then - declare - Vt : constant Valtyp := Get_Value (Syn_Inst, Decl); - begin - if Vt /= No_Valtyp - and then Vt.Val.Kind = Value_Wire - then - Finalize_Assignment (Get_Build (Syn_Inst), Vt.Val.W); - Free_Wire (Vt.Val.W); - end if; - end; - end if; - when Iir_Kind_Constant_Declaration => - null; - when Iir_Kind_Signal_Declaration - | Iir_Kind_Interface_Signal_Declaration => - pragma Assert (not Is_Subprg); - Finalize_Signal (Syn_Inst, Decl); - when Iir_Kind_Anonymous_Signal_Declaration => - null; - when Iir_Kind_Object_Alias_Declaration => - null; - when Iir_Kind_Procedure_Declaration - | Iir_Kind_Function_Declaration => - null; - when Iir_Kind_Procedure_Body - | Iir_Kind_Function_Body => - null; - when Iir_Kind_Non_Object_Alias_Declaration => - null; - when Iir_Kind_Attribute_Declaration => - null; - when Iir_Kind_Attribute_Specification => - null; - when Iir_Kind_Type_Declaration => - null; - when Iir_Kind_Anonymous_Type_Declaration => - null; - when Iir_Kind_Subtype_Declaration => - null; - when Iir_Kind_Component_Declaration => - null; - when Iir_Kind_File_Declaration => - null; - when Iir_Kind_Configuration_Specification => - null; - when Iir_Kind_Psl_Default_Clock => - -- Ignored; directly used by PSL directives. - null; - when Iir_Kind_Signal_Attribute_Declaration => - -- Not supported by synthesis. - null; - when others => - Vhdl.Errors.Error_Kind ("finalize_declaration", Decl); - end case; - end Finalize_Declaration; - - procedure Finalize_Declarations (Syn_Inst : Synth_Instance_Acc; - Decls : Iir; - Is_Subprg : Boolean := False) - is - Decl : Iir; - begin - Decl := Decls; - while Is_Valid (Decl) loop - Finalize_Declaration (Syn_Inst, Decl, Is_Subprg); - - Decl := Get_Chain (Decl); - end loop; - end Finalize_Declarations; -end Synth.Decls; diff --git a/src/synth/synth-decls.ads b/src/synth/synth-decls.ads deleted file mode 100644 index d227bdbe1..000000000 --- a/src/synth/synth-decls.ads +++ /dev/null @@ -1,79 +0,0 @@ --- Create declarations for synthesis. --- Copyright (C) 2017 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with Vhdl.Nodes; use Vhdl.Nodes; - -with Netlists; use Netlists; -with Synth.Vhdl_Context; use Synth.Vhdl_Context; -with Synth.Objtypes; use Synth.Objtypes; - -package Synth.Decls is - -- Return the Param_Type for ATYPE. - function Type_To_Param_Type (Atype : Node) return Param_Type; - - -- Convert MT to a Pval. - function Memtyp_To_Pval (Mt : Memtyp) return Pval; - - -- Get the type of DECL iff it is standalone (not an already existing - -- subtype). - function Get_Declaration_Type (Decl : Node) return Node; - - -- True if the element subtype indication of ATYPE needs to be created. - function Has_Element_Subtype_Indication (Atype : Node) return Boolean; - - function Synth_Array_Subtype_Indication - (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc; - - procedure Synth_Subtype_Indication - (Syn_Inst : Synth_Instance_Acc; Atype : Node); - function Synth_Subtype_Indication - (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc; - - -- Elaborate the type of DECL. - procedure Synth_Declaration_Type - (Syn_Inst : Synth_Instance_Acc; Decl : Node); - - procedure Synth_Declaration (Syn_Inst : Synth_Instance_Acc; - Decl : Node; - Is_Subprg : Boolean; - Last_Type : in out Node); - - procedure Synth_Declarations (Syn_Inst : Synth_Instance_Acc; - Decls : Iir; - Is_Subprg : Boolean := False); - - procedure Finalize_Declaration (Syn_Inst : Synth_Instance_Acc; - Decl : Iir; - Is_Subprg : Boolean); - procedure Finalize_Declarations (Syn_Inst : Synth_Instance_Acc; - Decls : Iir; - Is_Subprg : Boolean := False); - - procedure Synth_Package_Declaration - (Parent_Inst : Synth_Instance_Acc; Pkg : Node); - procedure Synth_Package_Body - (Parent_Inst : Synth_Instance_Acc; Pkg : Node; Bod : Node); - - procedure Synth_Generics_Association (Sub_Inst : Synth_Instance_Acc; - Syn_Inst : Synth_Instance_Acc; - Inter_Chain : Node; - Assoc_Chain : Node); - - procedure Synth_Package_Instantiation - (Parent_Inst : Synth_Instance_Acc; Pkg : Node); -end Synth.Decls; diff --git a/src/synth/synth-expr.adb b/src/synth/synth-expr.adb deleted file mode 100644 index d05c0d089..000000000 --- a/src/synth/synth-expr.adb +++ /dev/null @@ -1,2572 +0,0 @@ --- Expressions synthesis. --- Copyright (C) 2017 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with Types_Utils; use Types_Utils; -with Name_Table; -with Std_Names; -with Str_Table; -with Mutils; use Mutils; -with Errorout; use Errorout; - -with Vhdl.Types; -with Vhdl.Ieee.Std_Logic_1164; use Vhdl.Ieee.Std_Logic_1164; -with Vhdl.Std_Package; -with Vhdl.Errors; use Vhdl.Errors; -with Vhdl.Utils; use Vhdl.Utils; -with Vhdl.Evaluation; use Vhdl.Evaluation; -with Vhdl.Annotations; use Vhdl.Annotations; - -with PSL.Nodes; -with PSL.Errors; - -with Netlists.Gates; use Netlists.Gates; -with Netlists.Folds; use Netlists.Folds; -with Netlists.Utils; use Netlists.Utils; -with Netlists.Locations; - -with Synth.Memtype; use Synth.Memtype; -with Synth.Errors; use Synth.Errors; -with Synth.Vhdl_Environment; -with Synth.Decls; -with Synth.Stmts; use Synth.Stmts; -with Synth.Vhdl_Oper; use Synth.Vhdl_Oper; -with Synth.Vhdl_Heap; use Synth.Vhdl_Heap; -with Synth.Debugger; -with Synth.Vhdl_Aggr; - -with Grt.Types; -with Grt.To_Strings; - -package body Synth.Expr is - function Synth_Name (Syn_Inst : Synth_Instance_Acc; Name : Node) - return Valtyp; - - procedure Set_Location (N : Net; Loc : Node) - renames Synth.Source.Set_Location; - - function Get_Value_Memtyp (V : Valtyp) return Memtyp is - begin - case V.Val.Kind is - when Value_Memory => - return (V.Typ, V.Val.Mem); - when Value_Const => - return Get_Memtyp (V); - when Value_Wire => - return Synth.Vhdl_Environment.Env.Get_Static_Wire (V.Val.W); - when Value_Alias => - declare - Res : Memtyp; - begin - Res := Get_Value_Memtyp ((V.Val.A_Typ, V.Val.A_Obj)); - return (V.Typ, Res.Mem + V.Val.A_Off.Mem_Off); - end; - when others => - raise Internal_Error; - end case; - end Get_Value_Memtyp; - - function Get_Static_Discrete (V : Valtyp) return Int64 is - begin - case V.Val.Kind is - when Value_Memory => - return Read_Discrete (V); - when Value_Const => - return Read_Discrete (Get_Memtyp (V)); - when Value_Wire => - return Read_Discrete - (Synth.Vhdl_Environment.Env.Get_Static_Wire (V.Val.W)); - when others => - raise Internal_Error; - end case; - end Get_Static_Discrete; - - function Is_Positive (V : Valtyp) return Boolean - is - N : Net; - Inst : Instance; - begin - pragma Assert (V.Typ.Kind = Type_Discrete); - case V.Val.Kind is - when Value_Const - | Value_Memory => - return Read_Discrete (Get_Memtyp (V)) >= 0; - when Value_Net => - N := V.Val.N; - when Value_Wire => - if Synth.Vhdl_Environment.Env.Is_Static_Wire (V.Val.W) then - return Read_Discrete - (Synth.Vhdl_Environment.Env.Get_Static_Wire (V.Val.W)) >= 0; - else - return False; - end if; - when others => - raise Internal_Error; - end case; - Inst := Get_Net_Parent (N); - case Get_Id (Inst) is - when Id_Uextend - | Id_Const_UB32 => - return True; - when others => - -- Be conservative. - return False; - end case; - end Is_Positive; - - procedure From_Std_Logic (Enum : Int64; Val : out Uns32; Zx : out Uns32) is - begin - case Enum is - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_0_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_L_Pos => - Val := 0; - Zx := 0; - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_1_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_H_Pos => - Val := 1; - Zx := 0; - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_U_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_X_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_D_Pos => - Val := 1; - Zx := 1; - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_Z_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_W_Pos => - Val := 0; - Zx := 1; - when others => - -- Only 9 values. - raise Internal_Error; - end case; - end From_Std_Logic; - - procedure From_Bit (Enum : Int64; Val : out Uns32) is - begin - if Enum = 0 then - Val := 0; - elsif Enum = 1 then - Val := 1; - else - raise Internal_Error; - end if; - end From_Bit; - - procedure To_Logic - (Enum : Int64; Etype : Type_Acc; Val : out Uns32; Zx : out Uns32) is - begin - if Etype = Logic_Type then - pragma Assert (Etype.Kind = Type_Logic); - From_Std_Logic (Enum, Val, Zx); - elsif Etype = Boolean_Type or Etype = Bit_Type then - pragma Assert (Etype.Kind = Type_Bit); - From_Bit (Enum, Val); - Zx := 0; - else - raise Internal_Error; - end if; - end To_Logic; - - procedure Uns2logvec (Val : Uns64; - W : Width; - Vec : in out Logvec_Array; - Off : in out Uns32) is - begin - if W = 0 then - return; - end if; - - for I in 0 .. W - 1 loop - declare - B : constant Uns32 := Uns32 (Shift_Right (Val, Natural (I)) and 1); - Idx : constant Digit_Index := Digit_Index (Off / 32); - Pos : constant Natural := Natural (Off mod 32); - begin - Vec (Idx).Val := Vec (Idx).Val or Shift_Left (B, Pos); - end; - Off := Off + 1; - end loop; - end Uns2logvec; - - procedure Bit2logvec (Val : Uns32; - Vec : in out Logvec_Array; - Off : in out Uns32) - is - pragma Assert (Val <= 1); - Idx : constant Digit_Index := Digit_Index (Off / 32); - Pos : constant Natural := Natural (Off mod 32); - Va : Uns32; - begin - Va := Shift_Left (Val, Pos); - Vec (Idx).Val := Vec (Idx).Val or Va; - Vec (Idx).Zx := 0; - Off := Off + 1; - end Bit2logvec; - - procedure Logic2logvec (Val : Int64; - Vec : in out Logvec_Array; - Off : in out Uns32; - Has_Zx : in out Boolean) - is - pragma Assert (Val <= 8); - Idx : constant Digit_Index := Digit_Index (Off / 32); - Pos : constant Natural := Natural (Off mod 32); - Va : Uns32; - Zx : Uns32; - begin - From_Std_Logic (Val, Va, Zx); - Has_Zx := Has_Zx or Zx /= 0; - 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 Logic2logvec; - - procedure Value2logvec (Mem : Memory_Ptr; - Typ : Type_Acc; - Off : in out Uns32; - W : in out Width; - Vec : in out Logvec_Array; - Vec_Off : in out Uns32; - Has_Zx : in out Boolean) is - begin - if Off >= Typ.W then - -- Offset not yet reached. - Off := Off - Typ.W; - return; - end if; - if W = 0 then - return; - end if; - - case Typ.Kind is - when Type_Bit => - -- Scalar bits cannot be cut. - pragma Assert (Off = 0 and W >= Typ.W); - Bit2logvec (Uns32 (Read_U8 (Mem)), Vec, Vec_Off); - W := W - Typ.W; - when Type_Logic => - -- Scalar bits cannot be cut. - pragma Assert (Off = 0 and W >= Typ.W); - Logic2logvec (Int64 (Read_U8 (Mem)), Vec, Vec_Off, Has_Zx); - W := W - Typ.W; - when Type_Discrete => - -- Scalar bits cannot be cut. - pragma Assert (Off = 0 and W >= Typ.W); - Uns2logvec (To_Uns64 (Read_Discrete (Memtyp'(Typ, Mem))), - Typ.W, Vec, Vec_Off); - W := W - Typ.W; - when Type_Float => - -- Fp64 is for sure 64 bits. Assume the endianness of floats is - -- the same as integers endianness. - -- Scalar bits cannot be cut. - pragma Assert (Off = 0 and W >= Typ.W); - Uns2logvec (To_Uns64 (Read_Fp64 (Mem)), 64, Vec, Vec_Off); - W := W - Typ.W; - when Type_Vector => - declare - Vlen : Uns32; - begin - Vlen := Uns32 (Vec_Length (Typ)); - pragma Assert (Off < Vlen); - pragma Assert (Vlen > 0); - - if Vlen > Off + W then - Vlen := Off + W; - end if; - case Typ.Vec_El.Kind is - when Type_Bit => - -- TODO: optimize off mod 32 = 0. - for I in reverse Off + 1 .. Vlen loop - Bit2logvec (Uns32 (Read_U8 (Mem + Size_Type (I - 1))), - Vec, Vec_Off); - end loop; - when Type_Logic => - for I in reverse Off + 1 .. Vlen loop - Logic2logvec - (Int64 (Read_U8 (Mem + Size_Type (I - 1))), - Vec, Vec_Off, Has_Zx); - end loop; - when others => - raise Internal_Error; - end case; - W := W - (Vlen - Off); - Off := 0; - end; - when Type_Array => - declare - Alen : constant Iir_Index32 := Get_Array_Flat_Length (Typ); - El_Typ : constant Type_Acc := Typ.Arr_El; - begin - for I in reverse 1 .. Alen loop - Value2logvec (Mem + Size_Type (I - 1) * El_Typ.Sz, El_Typ, - Off, W, Vec, Vec_Off, Has_Zx); - exit when W = 0; - end loop; - end; - when Type_Record => - for I in Typ.Rec.E'Range loop - Value2logvec (Mem + Typ.Rec.E (I).Moff, Typ.Rec.E (I).Typ, - Off, W, Vec, Vec_Off, Has_Zx); - exit when W = 0; - end loop; - when others => - raise Internal_Error; - end case; - end Value2logvec; - - procedure Value2logvec (Val : Memtyp; - Off : Uns32; - W : Width; - Vec : in out Logvec_Array; - Vec_Off : in out Uns32; - Has_Zx : in out Boolean) - is - Off1 : Uns32; - W1 : Width; - begin - Off1 := Off; - W1 := W; - Value2logvec (Val.Mem, Val.Typ, Off1, W1, Vec, Vec_Off, Has_Zx); - pragma Assert (Off1 = 0); - pragma Assert (W1 = 0); - end Value2logvec; - - -- Resize for a discrete value. - function Synth_Resize - (Ctxt : Context_Acc; Val : Valtyp; W : Width; Loc : Node) return Net - is - Wn : constant Width := Val.Typ.W; - N : Net; - Res : Net; - V : Int64; - begin - if Is_Static (Val.Val) - and then Wn /= W - then - -- Optimization: resize directly. - V := Read_Discrete (Val); - if Val.Typ.Drange.Is_Signed then - Res := Build2_Const_Int (Ctxt, V, W); - else - Res := Build2_Const_Uns (Ctxt, To_Uns64 (V), W); - end if; - Set_Location (Res, Loc); - return Res; - end if; - - N := Get_Net (Ctxt, Val); - if Wn > W then - return Build2_Trunc (Ctxt, Id_Utrunc, N, W, Get_Location (Loc)); - elsif Wn < W then - if Val.Typ.Drange.Is_Signed then - Res := Build_Extend (Ctxt, Id_Sextend, N, W); - else - Res := Build_Extend (Ctxt, Id_Uextend, N, W); - end if; - Set_Location (Res, Loc); - return Res; - else - return N; - end if; - end Synth_Resize; - - procedure Concat_Array (Ctxt : Context_Acc; Arr : in out Net_Array) - is - Last : Int32; - Idx, New_Idx : Int32; - begin - Last := Arr'Last; - while Last > Arr'First loop - Idx := Arr'First; - New_Idx := Arr'First - 1; - while Idx <= Last loop - -- Gather at most 4 nets. - New_Idx := New_Idx + 1; - - if Idx = Last then - Arr (New_Idx) := Arr (Idx); - Idx := Idx + 1; - elsif Idx + 1 = Last then - Arr (New_Idx) := Build_Concat2 - (Ctxt, Arr (Idx), Arr (Idx + 1)); - Idx := Idx + 2; - elsif Idx + 2 = Last then - Arr (New_Idx) := Build_Concat3 - (Ctxt, Arr (Idx), Arr (Idx + 1), Arr (Idx + 2)); - Idx := Idx + 3; - else - Arr (New_Idx) := Build_Concat4 - (Ctxt, - Arr (Idx), Arr (Idx + 1), Arr (Idx + 2), Arr (Idx + 3)); - Idx := Idx + 4; - end if; - end loop; - Last := New_Idx; - end loop; - end Concat_Array; - - procedure Concat_Array - (Ctxt : Context_Acc; Arr : in out Net_Array; N : out Net) is - begin - Concat_Array (Ctxt, Arr); - N := Arr (Arr'First); - end Concat_Array; - - function Build_Discrete_Range_Type - (L : Int64; R : Int64; Dir : Direction_Type) return Discrete_Range_Type is - begin - return (Dir => Dir, - Left => L, - Right => R, - Is_Signed => L < 0 or R < 0); - end Build_Discrete_Range_Type; - - function Synth_Discrete_Range_Expression - (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Discrete_Range_Type - is - L, R : Valtyp; - Lval, Rval : Int64; - begin - -- Static values. - L := Synth_Expression_With_Basetype (Syn_Inst, Get_Left_Limit (Rng)); - R := Synth_Expression_With_Basetype (Syn_Inst, Get_Right_Limit (Rng)); - Strip_Const (L); - Strip_Const (R); - - if not (Is_Static (L.Val) and Is_Static (R.Val)) then - Error_Msg_Synth (+Rng, "limits of range are not constant"); - Set_Error (Syn_Inst); - return (Dir => Get_Direction (Rng), - Left => 0, - Right => 0, - Is_Signed => False); - end if; - - Lval := Read_Discrete (L); - Rval := Read_Discrete (R); - return Build_Discrete_Range_Type (Lval, Rval, Get_Direction (Rng)); - end Synth_Discrete_Range_Expression; - - function Synth_Float_Range_Expression - (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Float_Range_Type - is - L, R : Valtyp; - begin - -- Static values (so no enable). - L := Synth_Expression (Syn_Inst, Get_Left_Limit (Rng)); - R := Synth_Expression (Syn_Inst, Get_Right_Limit (Rng)); - return (Get_Direction (Rng), Read_Fp64 (L), Read_Fp64 (R)); - end Synth_Float_Range_Expression; - - -- Return the type of EXPR without evaluating it. - function Synth_Type_Of_Object (Syn_Inst : Synth_Instance_Acc; Expr : Node) - return Type_Acc is - begin - case Get_Kind (Expr) is - when Iir_Kinds_Object_Declaration => - declare - Val : constant Valtyp := Get_Value (Syn_Inst, Expr); - begin - return Val.Typ; - end; - when Iir_Kind_Simple_Name => - return Synth_Type_Of_Object (Syn_Inst, Get_Named_Entity (Expr)); - when Iir_Kind_Slice_Name => - declare - Pfx_Typ : Type_Acc; - Pfx_Bnd : Bound_Type; - El_Typ : Type_Acc; - Res_Bnd : Bound_Type; - Sl_Voff : Net; - Sl_Off : Value_Offsets; - begin - Pfx_Typ := Synth_Type_Of_Object (Syn_Inst, Get_Prefix (Expr)); - Get_Onedimensional_Array_Bounds (Pfx_Typ, Pfx_Bnd, El_Typ); - Synth_Slice_Suffix (Syn_Inst, Expr, Pfx_Bnd, El_Typ, - Res_Bnd, Sl_Voff, Sl_Off); - - if Sl_Voff /= No_Net then - raise Internal_Error; - end if; - return Create_Onedimensional_Array_Subtype (Pfx_Typ, Res_Bnd); - end; - when Iir_Kind_Indexed_Name => - declare - Pfx_Typ : Type_Acc; - begin - Pfx_Typ := Synth_Type_Of_Object (Syn_Inst, Get_Prefix (Expr)); - return Get_Array_Element (Pfx_Typ); - end; - when Iir_Kind_Selected_Element => - declare - Idx : constant Iir_Index32 := - Get_Element_Position (Get_Named_Entity (Expr)); - Pfx_Typ : Type_Acc; - begin - Pfx_Typ := Synth_Type_Of_Object (Syn_Inst, Get_Prefix (Expr)); - return Pfx_Typ.Rec.E (Idx + 1).Typ; - end; - - when Iir_Kind_Implicit_Dereference - | Iir_Kind_Dereference => - declare - Val : Valtyp; - Res : Valtyp; - begin - -- Maybe do not dereference it if its type is known ? - Val := Synth_Expression (Syn_Inst, Get_Prefix (Expr)); - Res := Vhdl_Heap.Synth_Dereference (Read_Access (Val)); - return Res.Typ; - end; - - when Iir_Kind_String_Literal8 => - -- TODO: the value should be computed (once) and its type - -- returned. - return Synth.Decls.Synth_Subtype_Indication - (Syn_Inst, Get_Type (Expr)); - - when others => - Vhdl.Errors.Error_Kind ("synth_type_of_object", Expr); - end case; - return null; - end Synth_Type_Of_Object; - - function Synth_Array_Attribute (Syn_Inst : Synth_Instance_Acc; Attr : Node) - return Bound_Type - is - Prefix_Name : constant Iir := Get_Prefix (Attr); - Prefix : constant Iir := Strip_Denoting_Name (Prefix_Name); - Dim : constant Natural := - Vhdl.Evaluation.Eval_Attribute_Parameter_Or_1 (Attr); - Typ : Type_Acc; - Val : Valtyp; - begin - -- Prefix is an array object or an array subtype. - if Get_Kind (Prefix) = Iir_Kind_Subtype_Declaration then - -- TODO: does this cover all the cases ? - Typ := Get_Subtype_Object (Syn_Inst, Get_Subtype_Indication (Prefix)); - else - Val := Synth_Expression_With_Basetype (Syn_Inst, Prefix_Name); - Typ := Val.Typ; - end if; - - return Get_Array_Bound (Typ, Dim_Type (Dim)); - end Synth_Array_Attribute; - - procedure Synth_Discrete_Range (Syn_Inst : Synth_Instance_Acc; - Bound : Node; - Rng : out Discrete_Range_Type) is - begin - case Get_Kind (Bound) is - when Iir_Kind_Range_Expression => - Rng := Synth_Discrete_Range_Expression (Syn_Inst, Bound); - when Iir_Kind_Integer_Subtype_Definition - | Iir_Kind_Enumeration_Subtype_Definition => - if Get_Type_Declarator (Bound) /= Null_Node then - declare - Typ : Type_Acc; - begin - -- This is a named subtype, so it has been evaluated. - Typ := Get_Subtype_Object (Syn_Inst, Bound); - Rng := Typ.Drange; - end; - else - Synth_Discrete_Range - (Syn_Inst, Get_Range_Constraint (Bound), Rng); - end if; - when Iir_Kind_Range_Array_Attribute => - declare - B : Bound_Type; - begin - B := Synth_Array_Attribute (Syn_Inst, Bound); - Rng := Build_Discrete_Range_Type - (Int64 (B.Left), Int64 (B.Right), B.Dir); - end; - when Iir_Kind_Reverse_Range_Array_Attribute => - declare - B : Bound_Type; - T : Int32; - begin - B := Synth_Array_Attribute (Syn_Inst, Bound); - -- Reverse - case B.Dir is - when Dir_To => - B.Dir := Dir_Downto; - when Dir_Downto => - B.Dir := Dir_To; - end case; - T := B.Right; - B.Right := B.Left; - B.Left := T; - - Rng := Build_Discrete_Range_Type - (Int64 (B.Left), Int64 (B.Right), B.Dir); - end; - when Iir_Kinds_Denoting_Name => - -- A discrete subtype name. - Synth_Discrete_Range - (Syn_Inst, Get_Subtype_Indication (Get_Named_Entity (Bound)), - Rng); - when others => - Error_Kind ("synth_discrete_range", Bound); - end case; - end Synth_Discrete_Range; - - function Synth_Array_Bounds (Syn_Inst : Synth_Instance_Acc; - Atype : Node; - Dim : Dim_Type) return Bound_Type - is - Info : constant Sim_Info_Acc := Get_Info (Atype); - begin - if Info = null then - pragma Assert (Get_Type_Declarator (Atype) = Null_Node); - declare - Index_Type : constant Node := - Get_Index_Type (Atype, Natural (Dim - 1)); - begin - return Synth_Bounds_From_Range (Syn_Inst, Index_Type); - end; - else - declare - Bnds : constant Type_Acc := Get_Subtype_Object (Syn_Inst, Atype); - begin - case Bnds.Kind is - when Type_Vector => - pragma Assert (Dim = 1); - return Bnds.Vbound; - when Type_Array => - return Bnds.Abounds.D (Dim); - when others => - raise Internal_Error; - end case; - end; - end if; - end Synth_Array_Bounds; - - function Synth_Bounds_From_Range (Syn_Inst : Synth_Instance_Acc; - Atype : Node) return Bound_Type - is - Rng : Discrete_Range_Type; - begin - Synth_Discrete_Range (Syn_Inst, Atype, Rng); - return (Dir => Rng.Dir, - Left => Int32 (Rng.Left), Right => Int32 (Rng.Right), - Len => Get_Range_Length (Rng)); - end Synth_Bounds_From_Range; - - function Synth_Bounds_From_Length (Atype : Node; Len : Int32) - return Bound_Type - is - Rng : constant Node := Get_Range_Constraint (Atype); - Limit : Int32; - begin - Limit := Int32 (Eval_Pos (Get_Left_Limit (Rng))); - case Get_Direction (Rng) is - when Dir_To => - return (Dir => Dir_To, - Left => Limit, - Right => Limit + Len - 1, - Len => Uns32 (Len)); - when Dir_Downto => - return (Dir => Dir_Downto, - Left => Limit, - Right => Limit - Len + 1, - Len => Uns32 (Len)); - end case; - end Synth_Bounds_From_Length; - - function Synth_Simple_Aggregate (Syn_Inst : Synth_Instance_Acc; - Aggr : Node) return Valtyp - is - Aggr_Type : constant Node := Get_Type (Aggr); - pragma Assert (Get_Nbr_Dimensions (Aggr_Type) = 1); - El_Type : constant Node := Get_Element_Subtype (Aggr_Type); - El_Typ : constant Type_Acc := Get_Subtype_Object (Syn_Inst, El_Type); - Els : constant Iir_Flist := Get_Simple_Aggregate_List (Aggr); - Last : constant Natural := Flist_Last (Els); - Bnd : Bound_Type; - Bnds : Bound_Array_Acc; - Res_Type : Type_Acc; - Val : Valtyp; - Res : Valtyp; - begin - -- Allocate the result. - Bnd := Synth_Array_Bounds (Syn_Inst, Aggr_Type, 1); - pragma Assert (Bnd.Len = Uns32 (Last + 1)); - - if El_Typ.Kind in Type_Nets then - Res_Type := Create_Vector_Type (Bnd, El_Typ); - else - Bnds := Create_Bound_Array (1); - Bnds.D (1) := Bnd; - Res_Type := Create_Array_Type (Bnds, El_Typ); - end if; - - Res := Create_Value_Memory (Res_Type); - - for I in Flist_First .. Last loop - -- Elements are supposed to be static, so no need for enable. - Val := Synth_Expression_With_Type - (Syn_Inst, Get_Nth_Element (Els, I), El_Typ); - pragma Assert (Is_Static (Val.Val)); - Write_Value (Res.Val.Mem + Size_Type (I) * El_Typ.Sz, Val); - end loop; - - return Res; - end Synth_Simple_Aggregate; - - -- Change the bounds of VAL. - function Reshape_Value (Val : Valtyp; Ntype : Type_Acc) return Valtyp is - begin - case Val.Val.Kind is - when Value_Wire => - return Create_Value_Wire (Val.Val.W, Ntype); - when Value_Net => - return Create_Value_Net (Val.Val.N, Ntype); - when Value_Alias => - return Create_Value_Alias - ((Val.Val.A_Typ, Val.Val.A_Obj), Val.Val.A_Off, Ntype); - when Value_Const => - return Reshape_Value ((Val.Typ, Val.Val.C_Val), Ntype); - when Value_Memory => - return (Ntype, Val.Val); - when others => - raise Internal_Error; - end case; - end Reshape_Value; - - function Synth_Subtype_Conversion (Ctxt : Context_Acc; - Vt : Valtyp; - Dtype : Type_Acc; - Bounds : Boolean; - Loc : Source.Syn_Src) - return Valtyp - is - Vtype : constant Type_Acc := Vt.Typ; - begin - if Vt = No_Valtyp then - -- Propagate error. - return No_Valtyp; - end if; - if Dtype = Vtype then - return Vt; - end if; - - case Dtype.Kind is - when Type_Bit => - pragma Assert (Vtype.Kind = Type_Bit); - return Vt; - when Type_Logic => - pragma Assert (Vtype.Kind = Type_Logic); - return Vt; - when Type_Discrete => - pragma Assert (Vtype.Kind in Type_All_Discrete); - case Vt.Val.Kind is - when Value_Net - | Value_Wire - | Value_Alias => - if Vtype.W /= Dtype.W then - -- Truncate. - -- TODO: check overflow. - declare - N : Net; - begin - if Is_Static_Val (Vt.Val) then - return Create_Value_Discrete - (Get_Static_Discrete (Vt), Dtype); - end if; - - N := Get_Net (Ctxt, Vt); - if Vtype.Drange.Is_Signed then - N := Build2_Sresize - (Ctxt, N, Dtype.W, Get_Location (Loc)); - else - N := Build2_Uresize - (Ctxt, N, Dtype.W, Get_Location (Loc)); - end if; - return Create_Value_Net (N, Dtype); - end; - else - return Vt; - end if; - when Value_Const => - return Synth_Subtype_Conversion - (Ctxt, (Vt.Typ, Vt.Val.C_Val), Dtype, Bounds, Loc); - when Value_Memory => - -- Check for overflow. - declare - Val : constant Int64 := Read_Discrete (Vt); - begin - if not In_Range (Dtype.Drange, Val) then - Error_Msg_Synth (+Loc, "value out of range"); - return No_Valtyp; - end if; - return Create_Value_Discrete (Val, Dtype); - end; - when others => - raise Internal_Error; - end case; - when Type_Float => - pragma Assert (Vtype.Kind = Type_Float); - -- TODO: check range - return Vt; - when Type_Vector => - pragma Assert (Vtype.Kind = Type_Vector - or Vtype.Kind = Type_Slice); - if Dtype.W /= Vtype.W then - Error_Msg_Synth - (+Loc, "mismatching vector length; got %v, expect %v", - (Errorout."+" (Vtype.W), +Dtype.W)); - return No_Valtyp; - end if; - if Bounds then - return Reshape_Value (Vt, Dtype); - else - return Vt; - end if; - when Type_Slice => - -- TODO: check width - return Vt; - when Type_Array => - pragma Assert (Vtype.Kind = Type_Array); - -- Check bounds. - for I in Vtype.Abounds.D'Range loop - if Vtype.Abounds.D (I).Len /= Dtype.Abounds.D (I).Len then - Error_Msg_Synth (+Loc, "mismatching array bounds"); - return No_Valtyp; - end if; - end loop; - -- TODO: check element. - if Bounds then - return Reshape_Value (Vt, Dtype); - else - return Vt; - end if; - when Type_Unbounded_Array => - pragma Assert (Vtype.Kind = Type_Array); - return Vt; - when Type_Unbounded_Vector => - pragma Assert (Vtype.Kind = Type_Vector - or else Vtype.Kind = Type_Slice); - return Vt; - when Type_Record => - pragma Assert (Vtype.Kind = Type_Record); - -- TODO: handle elements. - return Vt; - when Type_Unbounded_Record => - pragma Assert (Vtype.Kind = Type_Record); - return Vt; - when Type_Access => - return Vt; - when Type_File - | Type_Protected => - -- No conversion expected. - -- As the subtype is identical, it is already handled by the - -- above check. - raise Internal_Error; - end case; - end Synth_Subtype_Conversion; - - function Synth_Value_Attribute (Syn_Inst : Synth_Instance_Acc; Attr : Node) - return Valtyp - is - Param : constant Node := Get_Parameter (Attr); - Etype : constant Node := Get_Type (Attr); - Btype : constant Node := Get_Base_Type (Etype); - V : Valtyp; - Dtype : Type_Acc; - begin - -- The value is supposed to be static. - V := Synth_Expression (Syn_Inst, Param); - if V = No_Valtyp then - return No_Valtyp; - end if; - - Dtype := Get_Subtype_Object (Syn_Inst, Etype); - if not Is_Static (V.Val) then - Error_Msg_Synth (+Attr, "parameter of 'value must be static"); - return No_Valtyp; - end if; - - declare - Str : constant String := Value_To_String (V); - Res_N : Node; - Val : Int64; - begin - case Get_Kind (Btype) is - when Iir_Kind_Enumeration_Type_Definition => - Res_N := Eval_Value_Attribute (Str, Etype, Attr); - Val := Int64 (Get_Enum_Pos (Res_N)); - Free_Iir (Res_N); - when Iir_Kind_Integer_Type_Definition => - Val := Int64'Value (Str); - when others => - Error_Msg_Synth (+Attr, "unhandled type for 'value"); - return No_Valtyp; - end case; - return Create_Value_Discrete (Val, Dtype); - end; - end Synth_Value_Attribute; - - function Synth_Image_Attribute_Str (Val : Valtyp; Expr_Type : Iir) - return String - is - use Grt.Types; - begin - case Get_Kind (Expr_Type) is - when Iir_Kind_Floating_Type_Definition - | Iir_Kind_Floating_Subtype_Definition => - declare - Str : String (1 .. 24); - Last : Natural; - begin - Grt.To_Strings.To_String - (Str, Last, Ghdl_F64 (Read_Fp64 (Val))); - return Str (Str'First .. Last); - end; - when Iir_Kind_Integer_Type_Definition - | Iir_Kind_Integer_Subtype_Definition => - declare - Str : String (1 .. 21); - First : Natural; - begin - Grt.To_Strings.To_String - (Str, First, Ghdl_I64 (Read_Discrete (Val))); - return Str (First .. Str'Last); - end; - when Iir_Kind_Enumeration_Type_Definition - | Iir_Kind_Enumeration_Subtype_Definition => - declare - Lits : constant Iir_Flist := - Get_Enumeration_Literal_List (Get_Base_Type (Expr_Type)); - begin - return Name_Table.Image - (Get_Identifier - (Get_Nth_Element (Lits, Natural (Read_Discrete (Val))))); - end; - when Iir_Kind_Physical_Type_Definition - | Iir_Kind_Physical_Subtype_Definition => - declare - Str : String (1 .. 21); - First : Natural; - Id : constant Name_Id := - Get_Identifier (Get_Primary_Unit (Get_Base_Type (Expr_Type))); - begin - Grt.To_Strings.To_String - (Str, First, Ghdl_I64 (Read_Discrete (Val))); - return Str (First .. Str'Last) & ' ' & Name_Table.Image (Id); - end; - when others => - Error_Kind ("execute_image_attribute", Expr_Type); - end case; - end Synth_Image_Attribute_Str; - - function String_To_Valtyp (Str : String; Styp : Type_Acc) return Valtyp - is - Len : constant Natural := Str'Length; - Bnd : Bound_Array_Acc; - Typ : Type_Acc; - Res : Valtyp; - begin - Bnd := Create_Bound_Array (1); - Bnd.D (1) := (Dir => Dir_To, Left => 1, Right => Int32 (Len), - Len => Width (Len)); - Typ := Create_Array_Type (Bnd, Styp.Uarr_El); - - Res := Create_Value_Memory (Typ); - for I in Str'Range loop - Write_U8 (Res.Val.Mem + Size_Type (I - Str'First), - Character'Pos (Str (I))); - end loop; - return Res; - end String_To_Valtyp; - - function Synth_Image_Attribute (Syn_Inst : Synth_Instance_Acc; Attr : Node) - return Valtyp - is - Param : constant Node := Get_Parameter (Attr); - Etype : constant Node := Get_Type (Attr); - V : Valtyp; - Dtype : Type_Acc; - begin - -- The parameter is expected to be static. - V := Synth_Expression (Syn_Inst, Param); - if V = No_Valtyp then - return No_Valtyp; - end if; - Dtype := Get_Subtype_Object (Syn_Inst, Etype); - if not Is_Static (V.Val) then - Error_Msg_Synth (+Attr, "parameter of 'image must be static"); - return No_Valtyp; - end if; - - Strip_Const (V); - return String_To_Valtyp - (Synth_Image_Attribute_Str (V, Get_Type (Param)), Dtype); - end Synth_Image_Attribute; - - function Synth_Instance_Name_Attribute - (Syn_Inst : Synth_Instance_Acc; Attr : Node) return Valtyp - is - Atype : constant Node := Get_Type (Attr); - Atyp : constant Type_Acc := Get_Subtype_Object (Syn_Inst, Atype); - Name : constant Path_Instance_Name_Type := - Get_Path_Instance_Name_Suffix (Attr); - begin - -- Return a truncated name, as the prefix is not completly known. - return String_To_Valtyp (Name.Suffix, Atyp); - end Synth_Instance_Name_Attribute; - - function Synth_Name (Syn_Inst : Synth_Instance_Acc; Name : Node) - return Valtyp is - begin - case Get_Kind (Name) is - when Iir_Kind_Simple_Name - | Iir_Kind_Selected_Name => - return Synth_Name (Syn_Inst, Get_Named_Entity (Name)); - when Iir_Kind_Interface_Signal_Declaration - | Iir_Kind_Variable_Declaration - | Iir_Kind_Interface_Variable_Declaration - | Iir_Kind_Signal_Declaration - | Iir_Kind_Interface_Constant_Declaration - | Iir_Kind_Constant_Declaration - | Iir_Kind_Iterator_Declaration - | Iir_Kind_Object_Alias_Declaration - | Iir_Kind_File_Declaration - | Iir_Kind_Interface_File_Declaration => - return Get_Value (Syn_Inst, Name); - when Iir_Kind_Enumeration_Literal => - declare - Typ : constant Type_Acc := - Get_Subtype_Object (Syn_Inst, Get_Type (Name)); - Res : Valtyp; - begin - Res := Create_Value_Memory (Typ); - Write_Discrete (Res, Int64 (Get_Enum_Pos (Name))); - return Res; - end; - when Iir_Kind_Unit_Declaration => - declare - Typ : constant Type_Acc := - Get_Subtype_Object (Syn_Inst, Get_Type (Name)); - begin - return Create_Value_Discrete - (Vhdl.Evaluation.Get_Physical_Value (Name), Typ); - end; - when Iir_Kind_Implicit_Dereference - | Iir_Kind_Dereference => - declare - Val : Valtyp; - begin - Val := Synth_Expression (Syn_Inst, Get_Prefix (Name)); - return Vhdl_Heap.Synth_Dereference (Read_Access (Val)); - end; - when others => - Error_Kind ("synth_name", Name); - end case; - end Synth_Name; - - -- Convert index IDX in PFX to an offset. - -- SYN_INST and LOC are used in case of error. - function Index_To_Offset - (Syn_Inst : Synth_Instance_Acc; Bnd : Bound_Type; Idx : Int64; Loc : Node) - return Value_Offsets - is - Res : Value_Offsets; - begin - if not In_Bounds (Bnd, Int32 (Idx)) then - Error_Msg_Synth (+Loc, "index not within bounds"); - Synth.Debugger.Debug_Error (Syn_Inst, Loc); - return (0, 0); - end if; - - -- The offset is from the LSB (bit 0). Bit 0 is the rightmost one. - case Bnd.Dir is - when Dir_To => - Res.Net_Off := Uns32 (Bnd.Right - Int32 (Idx)); - Res.Mem_Off := Size_Type (Int32 (Idx) - Bnd.Left); - when Dir_Downto => - Res.Net_Off := Uns32 (Int32 (Idx) - Bnd.Right); - Res.Mem_Off := Size_Type (Bnd.Left - Int32 (Idx)); - end case; - - return Res; - end Index_To_Offset; - - function Dyn_Index_To_Offset - (Ctxt : Context_Acc; Bnd : Bound_Type; Idx_Val : Valtyp; Loc : Node) - return Net - is - Idx2 : Net; - Off : Net; - Right : Net; - Wbounds : Width; - begin - Wbounds := Clog2 (Bnd.Len); - Idx2 := Synth_Resize (Ctxt, Idx_Val, Wbounds, Loc); - - if Bnd.Right = 0 and then Bnd.Dir = Dir_Downto then - -- Simple case without adjustments. - return Idx2; - end if; - - Right := Build_Const_UB32 (Ctxt, To_Uns32 (Bnd.Right), Wbounds); - Set_Location (Right, Loc); - - case Bnd.Dir is - when Dir_To => - -- L <= I <= R --> off = R - I - Off := Build_Dyadic (Ctxt, Id_Sub, Right, Idx2); - when Dir_Downto => - -- L >= I >= R --> off = I - R - Off := Build_Dyadic (Ctxt, Id_Sub, Idx2, Right); - end case; - Set_Location (Off, Loc); - return Off; - end Dyn_Index_To_Offset; - - -- Return the bounds of a one dimensional array/vector type and the - -- width of the element. - procedure Get_Onedimensional_Array_Bounds - (Typ : Type_Acc; Bnd : out Bound_Type; El_Typ : out Type_Acc) is - begin - case Typ.Kind is - when Type_Vector => - El_Typ := Typ.Vec_El; - Bnd := Typ.Vbound; - when Type_Array => - El_Typ := Typ.Arr_El; - Bnd := Typ.Abounds.D (1); - when others => - raise Internal_Error; - end case; - end Get_Onedimensional_Array_Bounds; - - function Create_Onedimensional_Array_Subtype - (Btyp : Type_Acc; Bnd : Bound_Type) return Type_Acc - is - Res : Type_Acc; - Bnds : Bound_Array_Acc; - begin - case Btyp.Kind is - when Type_Vector => - Res := Create_Vector_Type (Bnd, Btyp.Vec_El); - when Type_Unbounded_Vector => - Res := Create_Vector_Type (Bnd, Btyp.Uvec_El); - when Type_Array => - pragma Assert (Btyp.Abounds.Ndim = 1); - Bnds := Create_Bound_Array (1); - Bnds.D (1) := Bnd; - Res := Create_Array_Type (Bnds, Btyp.Arr_El); - when Type_Unbounded_Array => - pragma Assert (Btyp.Uarr_Ndim = 1); - Bnds := Create_Bound_Array (1); - Bnds.D (1) := Bnd; - Res := Create_Array_Type (Bnds, Btyp.Uarr_El); - when others => - raise Internal_Error; - end case; - return Res; - end Create_Onedimensional_Array_Subtype; - - procedure Synth_Indexed_Name (Syn_Inst : Synth_Instance_Acc; - Name : Node; - Pfx_Type : Type_Acc; - Voff : out Net; - Off : out Value_Offsets) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Indexes : constant Iir_Flist := Get_Index_List (Name); - El_Typ : constant Type_Acc := Get_Array_Element (Pfx_Type); - Idx_Expr : Node; - Idx_Val : Valtyp; - Bnd : Bound_Type; - Stride : Uns32; - Ivoff : Net; - Idx_Off : Value_Offsets; - begin - Voff := No_Net; - Off := (0, 0); - - Stride := 1; - for I in reverse Flist_First .. Flist_Last (Indexes) loop - Idx_Expr := Get_Nth_Element (Indexes, I); - - -- Use the base type as the subtype of the index is not synth-ed. - Idx_Val := Synth_Expression_With_Basetype (Syn_Inst, Idx_Expr); - if Idx_Val = No_Valtyp then - -- Propagate errorc - Voff := No_Net; - Off := (0, 0); - return; - end if; - - Strip_Const (Idx_Val); - - Bnd := Get_Array_Bound (Pfx_Type, Dim_Type (I + 1)); - - if Is_Static_Val (Idx_Val.Val) then - Idx_Off := Index_To_Offset (Syn_Inst, Bnd, - Get_Static_Discrete (Idx_Val), Name); - Off.Net_Off := Off.Net_Off + Idx_Off.Net_Off * Stride * El_Typ.W; - Off.Mem_Off := Off.Mem_Off - + Idx_Off.Mem_Off * Size_Type (Stride) * El_Typ.Sz; - else - Ivoff := Dyn_Index_To_Offset (Ctxt, Bnd, Idx_Val, Name); - Ivoff := Build_Memidx - (Get_Build (Syn_Inst), Ivoff, El_Typ.W * Stride, - Bnd.Len - 1, - Width (Clog2 (Uns64 (Stride * Bnd.Len)))); - Set_Location (Ivoff, Idx_Expr); - - if Voff = No_Net then - Voff := Ivoff; - else - Voff := Build_Addidx (Get_Build (Syn_Inst), Ivoff, Voff); - Set_Location (Voff, Idx_Expr); - end if; - end if; - - Stride := Stride * Bnd.Len; - end loop; - end Synth_Indexed_Name; - - function Is_Static (N : Net) return Boolean is - begin - case Get_Id (Get_Module (Get_Net_Parent (N))) is - when Id_Const_UB32 => - return True; - when others => - return False; - end case; - end Is_Static; - - function Get_Const (N : Net) return Int32 - is - Inst : constant Instance := Get_Net_Parent (N); - begin - case Get_Id (Get_Module (Inst)) is - when Id_Const_UB32 => - return To_Int32 (Get_Param_Uns32 (Inst, 0)); - when others => - raise Internal_Error; - end case; - end Get_Const; - - -- Decompose VAL as FACTOR * INP + ADDEND (where only INP is non-static). - procedure Decompose_Mul_Add (Val : Net; - Inp : out Net; - Factor : out Int32; - Addend : out Int32) - is - Inst : Instance; - Val_I0, Val_I1 : Net; - begin - Factor := 1; - Addend := 0; - Inp := Val; - - loop - Inst := Get_Net_Parent (Inp); - case Get_Id (Get_Module (Inst)) is - when Id_Add => - Val_I0 := Get_Input_Net (Inst, 0); - Val_I1 := Get_Input_Net (Inst, 1); - if Is_Static (Val_I0) then - Addend := Addend + Get_Const (Val_I0) * Factor; - Inp := Val_I1; - elsif Is_Static (Val_I1) then - Addend := Addend + Get_Const (Val_I1) * Factor; - Inp := Val_I0; - else - -- It's an addition, but without any constant value. - return; - end if; - when Id_Sub => - Val_I0 := Get_Input_Net (Inst, 0); - Val_I1 := Get_Input_Net (Inst, 1); - if Is_Static (Val_I1) then - Addend := Addend - Get_Const (Val_I1) * Factor; - Inp := Val_I0; - elsif Is_Static (Val_I0) then - Addend := Addend + Get_Const (Val_I0) * Factor; - Factor := -Factor; - Inp := Val_I1; - else - -- It's a substraction, but without any constant value. - return; - end if; - when Id_Smul => - Val_I0 := Get_Input_Net (Inst, 0); - Val_I1 := Get_Input_Net (Inst, 1); - if Is_Static (Val_I0) then - Factor := Factor * Get_Const (Val_I0); - Inp := Val_I1; - elsif Is_Static (Val_I1) then - Factor := Factor * Get_Const (Val_I1); - Inp := Val_I0; - else - -- A mul but without any constant value. - return; - end if; - when Id_Utrunc - | Id_Uextend => - Inp := Get_Input_Net (Inst, 0); - when others => - -- Cannot decompose it. - return; - end case; - end loop; - end Decompose_Mul_Add; - - -- Identify LEFT to/downto RIGHT as: - -- INP * STEP + WIDTH - 1 + OFF to/downto INP * STEP + OFF - procedure Synth_Extract_Dyn_Suffix (Ctxt : Context_Acc; - Loc : Node; - Pfx_Bnd : Bound_Type; - Left : Net; - Right : Net; - Inp : out Net; - Step : out Uns32; - Off : out Uns32; - Width : out Uns32) - is - L_Inp, R_Inp : Net; - L_Fac, R_Fac : Int32; - L_Add, R_Add : Int32; - begin - Inp := No_Net; - Step := 0; - Off := 0; - Width := 0; - - if Left = Right then - L_Inp := Left; - R_Inp := Right; - L_Fac := 1; - R_Fac := 1; - L_Add := 0; - R_Add := 0; - else - Decompose_Mul_Add (Left, L_Inp, L_Fac, L_Add); - Decompose_Mul_Add (Right, R_Inp, R_Fac, R_Add); - end if; - - if not Same_Net (L_Inp, R_Inp) then - Error_Msg_Synth - (+Loc, "cannot extract same variable part for dynamic slice"); - return; - end if; - Inp := L_Inp; - - if L_Fac /= R_Fac then - Error_Msg_Synth - (+Loc, "cannot extract same constant factor for dynamic slice"); - return; - end if; - if L_Fac < 0 then - Step := Uns32 (-L_Fac); - Inp := Build_Monadic (Ctxt, Id_Neg, Inp); - Set_Location (Inp, Loc); - else - Step := Uns32 (L_Fac); - end if; - - case Pfx_Bnd.Dir is - when Dir_To => - Width := Uns32 (R_Add - L_Add + 1); - Off := Uns32 (L_Add - Pfx_Bnd.Left); - when Dir_Downto => - Width := Uns32 (L_Add - R_Add + 1); - if R_Add >= Pfx_Bnd.Right then - Off := Uns32 (R_Add - Pfx_Bnd.Right); - else - -- Handle biased values. - declare - Bias : constant Uns32 := - (Uns32 (Pfx_Bnd.Right - R_Add) + Step - 1) / Step; - Bias_Net : Net; - begin - -- Add bias to INP and adjust the offset. - Bias_Net := Build2_Const_Uns - (Ctxt, Uns64 (Bias), Get_Width (Inp)); - Inp := Build_Dyadic (Ctxt, Id_Add, Inp, Bias_Net); - Set_Location (Inp, Loc); - Off := Uns32 (Int32 (Bias * Step) + R_Add - Pfx_Bnd.Right); - end; - end if; - end case; - end Synth_Extract_Dyn_Suffix; - - procedure Synth_Slice_Const_Suffix (Syn_Inst: Synth_Instance_Acc; - Expr : Node; - Name : Node; - Pfx_Bnd : Bound_Type; - L, R : Int64; - Dir : Direction_Type; - El_Typ : Type_Acc; - Res_Bnd : out Bound_Type; - Off : out Value_Offsets) - is - Is_Null : Boolean; - Len : Uns32; - begin - if Pfx_Bnd.Dir /= Dir then - Error_Msg_Synth (+Name, "direction mismatch in slice"); - Off := (0, 0); - if Dir = Dir_To then - Res_Bnd := (Dir => Dir_To, Left => 1, Right => 0, Len => 0); - else - Res_Bnd := (Dir => Dir_Downto, Left => 0, Right => 1, Len => 0); - end if; - return; - end if; - - -- Might be a null slice. - case Pfx_Bnd.Dir is - when Dir_To => - Is_Null := L > R; - when Dir_Downto => - Is_Null := L < R; - end case; - if Is_Null then - Len := 0; - Off := (0, 0); - else - if not In_Bounds (Pfx_Bnd, Int32 (L)) - or else not In_Bounds (Pfx_Bnd, Int32 (R)) - then - Error_Msg_Synth (+Name, "index not within bounds"); - Synth.Debugger.Debug_Error (Syn_Inst, Expr); - Off := (0, 0); - return; - end if; - - case Pfx_Bnd.Dir is - when Dir_To => - Len := Uns32 (R - L + 1); - Off.Net_Off := Uns32 (Pfx_Bnd.Right - Int32 (R)) * El_Typ.W; - Off.Mem_Off := Size_Type (Int32 (L) - Pfx_Bnd.Left) * El_Typ.Sz; - when Dir_Downto => - Len := Uns32 (L - R + 1); - Off.Net_Off := Uns32 (Int32 (R) - Pfx_Bnd.Right) * El_Typ.W; - Off.Mem_Off := Size_Type (Pfx_Bnd.Left - Int32 (L)) * El_Typ.Sz; - end case; - end if; - Res_Bnd := (Dir => Pfx_Bnd.Dir, - Len => Len, - Left => Int32 (L), - Right => Int32 (R)); - end Synth_Slice_Const_Suffix; - - procedure Synth_Slice_Suffix (Syn_Inst : Synth_Instance_Acc; - Name : Node; - Pfx_Bnd : Bound_Type; - El_Typ : Type_Acc; - Res_Bnd : out Bound_Type; - Inp : out Net; - Off : out Value_Offsets) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Expr : constant Node := Get_Suffix (Name); - Left, Right : Valtyp; - Dir : Direction_Type; - Step : Uns32; - Max : Uns32; - Inp_W : Width; - begin - Off := (0, 0); - Inp := No_Net; - - case Get_Kind (Expr) is - when Iir_Kind_Range_Expression => - -- As the range may be dynamic, cannot use synth_discrete_range. - Left := Synth_Expression_With_Basetype - (Syn_Inst, Get_Left_Limit (Expr)); - Right := Synth_Expression_With_Basetype - (Syn_Inst, Get_Right_Limit (Expr)); - Dir := Get_Direction (Expr); - - when Iir_Kind_Range_Array_Attribute - | Iir_Kind_Reverse_Range_Array_Attribute - | Iir_Kinds_Denoting_Name => - declare - Rng : Discrete_Range_Type; - begin - Synth_Discrete_Range (Syn_Inst, Expr, Rng); - Synth_Slice_Const_Suffix (Syn_Inst, Expr, - Name, Pfx_Bnd, - Rng.Left, Rng.Right, Rng.Dir, - El_Typ, Res_Bnd, Off); - return; - end; - when others => - Error_Msg_Synth - (+Expr, "only range expression supported for slices"); - Res_Bnd := (Dir => Dir_To, Left => 1, Right => 0, Len => 0); - return; - end case; - - if Is_Static_Val (Left.Val) and then Is_Static_Val (Right.Val) then - Synth_Slice_Const_Suffix (Syn_Inst, Expr, - Name, Pfx_Bnd, - Get_Static_Discrete (Left), - Get_Static_Discrete (Right), - Dir, - El_Typ, Res_Bnd, Off); - else - if Pfx_Bnd.Dir /= Dir then - Error_Msg_Synth (+Name, "direction mismatch in slice"); - if Dir = Dir_To then - Res_Bnd := (Dir => Dir_To, Left => 1, Right => 0, Len => 0); - else - Res_Bnd := (Dir => Dir_Downto, Left => 0, Right => 1, Len => 0); - end if; - return; - end if; - - if Is_Static (Left.Val) or else Is_Static (Right.Val) then - Error_Msg_Synth - (+Name, "left and right bounds of a slice must be " - & "either constant or dynamic"); - return; - end if; - - Synth_Extract_Dyn_Suffix - (Ctxt, Name, Pfx_Bnd, Get_Net (Ctxt, Left), Get_Net (Ctxt, Right), - Inp, Step, Off.Net_Off, Res_Bnd.Len); - if Inp = No_Net then - return; - end if; - Inp_W := Get_Width (Inp); - -- FIXME: convert range to offset. - -- Extract max from the range. - -- example: len=128 wd=8 step=8 => max=16 - -- len=8 wd=4 step=1 => max=4 - -- max so that max*step+wd <= len - off - -- max <= (len - off - wd) / step - Max := (Pfx_Bnd.Len - Off.Net_Off - Res_Bnd.Len) / Step; - if Clog2 (Uns64 (Max)) > Natural (Inp_W) then - -- The width of Inp limits the max. - Max := 2**Natural (Inp_W) - 1; - end if; - Inp := Build_Memidx - (Ctxt, Inp, Step * El_Typ.W, Max, - Inp_W + Width (Clog2 (Uns64 (Step * El_Typ.W)))); - Set_Location (Inp, Name); - end if; - end Synth_Slice_Suffix; - - -- Match: clk_signal_name'event - -- and return clk_signal_name. - function Extract_Event_Expr_Prefix (Expr : Node) return Node is - begin - if Get_Kind (Expr) = Iir_Kind_Event_Attribute then - return Get_Prefix (Expr); - else - return Null_Node; - end if; - end Extract_Event_Expr_Prefix; - - function Is_Same_Clock (Syn_Inst : Synth_Instance_Acc; - Left, Right : Node; - Clk_Left : Net) return Boolean - is - N : Net; - begin - -- Handle directly the common case (when clock is a simple name). - if Get_Kind (Left) = Iir_Kind_Simple_Name - and then Get_Kind (Right) = Iir_Kind_Simple_Name - then - return Get_Named_Entity (Left) = Get_Named_Entity (Right); - end if; - - N := Get_Net (Get_Build (Syn_Inst), Synth_Expression (Syn_Inst, Right)); - - return Same_Net (Clk_Left, N); - end Is_Same_Clock; - - -- Match: clk_signal_name = '1' | clk_signal_name = '0' - function Extract_Clock_Level - (Syn_Inst : Synth_Instance_Acc; Expr : Node; Prefix : Node) return Net - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Clk : Net; - Imp : Node; - Left, Right : Node; - Lit : Valtyp; - Lit_Type : Node; - Posedge : Boolean; - Res : Net; - begin - Clk := Get_Net (Ctxt, Synth_Expression (Syn_Inst, Prefix)); - if Get_Kind (Expr) /= Iir_Kind_Equality_Operator then - Error_Msg_Synth (+Expr, "ill-formed clock-level, '=' expected"); - Res := Build_Posedge (Ctxt, Clk); - Set_Location (Res, Expr); - return Res; - end if; - Imp := Get_Implementation (Expr); - if Get_Implicit_Definition (Imp) /= Iir_Predefined_Enum_Equality then - Error_Msg_Synth (+Expr, "ill-formed clock-level, '=' expected"); - Res := Build_Posedge (Ctxt, Clk); - Set_Location (Res, Expr); - return Res; - end if; - - Left := Get_Left (Expr); - if not Is_Same_Clock (Syn_Inst, Prefix, Left, Clk) then - Error_Msg_Synth (+Left, "clock signal name doesn't match"); - end if; - - Right := Get_Right (Expr); - Lit_Type := Get_Base_Type (Get_Type (Right)); - Lit := Synth_Expression (Syn_Inst, Right); - if Lit.Val.Kind /= Value_Memory then - Error_Msg_Synth (+Right, "clock-level is not a constant"); - Posedge := True; - else - if Lit_Type = Vhdl.Ieee.Std_Logic_1164.Std_Ulogic_Type then - case Read_U8 (Lit.Val.Mem) is - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_0_Pos => - Posedge := False; - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_1_Pos => - Posedge := True; - when others => - Error_Msg_Synth - (+Right, "clock-level must be either '0' or '1'"); - Posedge := True; - end case; - else - pragma Assert (Lit_Type = Vhdl.Std_Package.Bit_Type_Definition); - case Read_U8 (Lit.Val.Mem) is - when 0 => - Posedge := False; - when 1 => - Posedge := True; - when others => - raise Internal_Error; - end case; - end if; - end if; - if Posedge then - Res := Build_Posedge (Ctxt, Clk); - else - Res := Build_Negedge (Ctxt, Clk); - end if; - Set_Location (Res, Expr); - return Res; - end Extract_Clock_Level; - - -- Try to match: clk'event and clk = X - -- or: clk = X and clk'event - -- where X is '0' or '1'. - function Synth_Clock_Edge - (Syn_Inst : Synth_Instance_Acc; Left, Right : Node) return Net - is - Prefix : Node; - begin - -- Try with left. - Prefix := Extract_Event_Expr_Prefix (Left); - if Is_Valid (Prefix) then - return Extract_Clock_Level (Syn_Inst, Right, Prefix); - end if; - - -- Try with right. - Prefix := Extract_Event_Expr_Prefix (Right); - if Is_Valid (Prefix) then - return Extract_Clock_Level (Syn_Inst, Left, Prefix); - end if; - - return No_Net; - end Synth_Clock_Edge; - - function Synth_Type_Conversion - (Syn_Inst : Synth_Instance_Acc; Conv : Node) return Valtyp - is - Expr : constant Node := Get_Expression (Conv); - Conv_Type : constant Node := Get_Type (Conv); - Conv_Typ : constant Type_Acc := Get_Subtype_Object (Syn_Inst, Conv_Type); - Val : Valtyp; - begin - Val := Synth_Expression_With_Basetype (Syn_Inst, Expr); - if Val = No_Valtyp then - return No_Valtyp; - end if; - Strip_Const (Val); - case Get_Kind (Conv_Type) is - when Iir_Kind_Integer_Subtype_Definition => - if Val.Typ.Kind = Type_Discrete then - -- Int to int. - return Val; - elsif Val.Typ.Kind = Type_Float then - return Create_Value_Discrete - (Int64 (Read_Fp64 (Val)), Conv_Typ); - else - Error_Msg_Synth (+Conv, "unhandled type conversion (to int)"); - return No_Valtyp; - end if; - when Iir_Kind_Floating_Subtype_Definition => - if Is_Static (Val.Val) then - return Create_Value_Float - (Fp64 (Read_Discrete (Val)), Conv_Typ); - else - Error_Msg_Synth (+Conv, "unhandled type conversion (to float)"); - return No_Valtyp; - end if; - when Iir_Kind_Array_Type_Definition - | Iir_Kind_Array_Subtype_Definition => - case Conv_Typ.Kind is - when Type_Vector - | Type_Unbounded_Vector => - return Val; - when others => - Error_Msg_Synth - (+Conv, "unhandled type conversion (to array)"); - return No_Valtyp; - end case; - when Iir_Kind_Enumeration_Type_Definition - | Iir_Kind_Enumeration_Subtype_Definition => - pragma Assert (Get_Base_Type (Get_Type (Expr)) - = Get_Base_Type (Conv_Type)); - return Val; - when others => - Error_Msg_Synth (+Conv, "unhandled type conversion"); - return No_Valtyp; - end case; - end Synth_Type_Conversion; - - function Error_Ieee_Operator (Imp : Node; Loc : Node) return Boolean - is - use Std_Names; - Parent : constant Iir := Get_Parent (Imp); - begin - if Get_Kind (Parent) = Iir_Kind_Package_Declaration - and then (Get_Identifier - (Get_Library (Get_Design_File (Get_Design_Unit (Parent)))) - = Name_Ieee) - then - case Get_Identifier (Parent) is - when Name_Std_Logic_1164 - | Name_Std_Logic_Arith - | Name_Std_Logic_Signed - | Name_Std_Logic_Unsigned - | Name_Std_Logic_Misc - | Name_Numeric_Std - | Name_Numeric_Bit - | Name_Math_Real => - Error_Msg_Synth - (+Loc, "unhandled predefined IEEE operator %i", +Imp); - Error_Msg_Synth - (+Imp, " declared here"); - return True; - when others => - -- ieee 2008 packages are handled like regular packages. - null; - end case; - end if; - - return False; - end Error_Ieee_Operator; - - function Synth_String_Literal - (Syn_Inst : Synth_Instance_Acc; Str : Node; Str_Typ : Type_Acc) - return Valtyp - is - pragma Assert (Get_Kind (Str) = Iir_Kind_String_Literal8); - Id : constant String8_Id := Get_String8_Id (Str); - - Str_Type : constant Node := Get_Type (Str); - El_Type : Type_Acc; - Bounds : Bound_Type; - Bnds : Bound_Array_Acc; - Res_Type : Type_Acc; - Res : Valtyp; - Pos : Nat8; - begin - case Str_Typ.Kind is - when Type_Vector => - Bounds := Str_Typ.Vbound; - when Type_Array => - Bounds := Str_Typ.Abounds.D (1); - when Type_Unbounded_Vector - | Type_Unbounded_Array => - Bounds := Synth_Bounds_From_Length - (Get_Index_Type (Str_Type, 0), Get_String_Length (Str)); - when others => - raise Internal_Error; - end case; - - El_Type := Get_Subtype_Object (Syn_Inst, Get_Element_Subtype (Str_Type)); - if El_Type.Kind in Type_Nets then - Res_Type := Create_Vector_Type (Bounds, El_Type); - else - Bnds := Create_Bound_Array (1); - Bnds.D (1) := Bounds; - Res_Type := Create_Array_Type (Bnds, El_Type); - end if; - Res := Create_Value_Memory (Res_Type); - - -- Only U8 are handled. - pragma Assert (El_Type.Sz = 1); - - -- From left to right. - for I in 1 .. Bounds.Len loop - -- FIXME: use literal from type ?? - Pos := Str_Table.Element_String8 (Id, Pos32 (I)); - Write_U8 (Res.Val.Mem + Size_Type (I - 1), Nat8'Pos (Pos)); - end loop; - - return Res; - end Synth_String_Literal; - - -- Return the left bound if the direction of the range is LEFT_DIR. - function Synth_Low_High_Type_Attribute - (Syn_Inst : Synth_Instance_Acc; Expr : Node; Left_Dir : Direction_Type) - return Valtyp - is - Typ : Type_Acc; - R : Int64; - begin - Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Get_Prefix (Expr))); - pragma Assert (Typ.Kind = Type_Discrete); - if Typ.Drange.Dir = Left_Dir then - R := Typ.Drange.Left; - else - R := Typ.Drange.Right; - end if; - return Create_Value_Discrete (R, Typ); - end Synth_Low_High_Type_Attribute; - - function Synth_PSL_Expression - (Syn_Inst : Synth_Instance_Acc; Expr : PSL.Types.PSL_Node) return Net - is - use PSL.Types; - use PSL.Nodes; - - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Loc : constant Location_Type := Get_Location (Expr); - Res : Net; - begin - case Get_Kind (Expr) is - when N_HDL_Bool => - declare - E : constant Vhdl.Types.Vhdl_Node := Get_HDL_Node (Expr); - begin - return Get_Net (Ctxt, Synth_Expression (Syn_Inst, E)); - end; - when N_Not_Bool => - pragma Assert (Loc /= No_Location); - Res := Build_Monadic - (Ctxt, Id_Not, - Synth_PSL_Expression (Syn_Inst, Get_Boolean (Expr))); - when N_And_Bool => - pragma Assert (Loc /= No_Location); - declare - L : constant PSL_Node := Get_Left (Expr); - R : constant PSL_Node := Get_Right (Expr); - Edge : Net; - begin - -- Handle edge (as it can be in default clock). - if Get_Kind (L) in N_HDLs and then Get_Kind (R) in N_HDLs then - Edge := Synth_Clock_Edge - (Syn_Inst, Get_HDL_Node (L), Get_HDL_Node (R)); - if Edge /= No_Net then - return Edge; - end if; - end if; - if Get_Kind (R) = N_EOS then - -- It is never EOS! - Res := Build_Const_UB32 (Ctxt, 0, 1); - else - Res := Build_Dyadic (Ctxt, Id_And, - Synth_PSL_Expression (Syn_Inst, L), - Synth_PSL_Expression (Syn_Inst, R)); - end if; - end; - when N_Or_Bool => - pragma Assert (Loc /= No_Location); - Res := Build_Dyadic - (Ctxt, Id_Or, - Synth_PSL_Expression (Syn_Inst, Get_Left (Expr)), - Synth_PSL_Expression (Syn_Inst, Get_Right (Expr))); - when N_True => - Res := Build_Const_UB32 (Ctxt, 1, 1); - when N_False - | N_EOS => - Res := Build_Const_UB32 (Ctxt, 0, 1); - when others => - PSL.Errors.Error_Kind ("synth_psl_expr", Expr); - return No_Net; - end case; - Netlists.Locations.Set_Location (Get_Net_Parent (Res), Loc); - return Res; - end Synth_PSL_Expression; - - function Synth_Psl_Function_Clock - (Syn_Inst : Synth_Instance_Acc; Call : Node; Ctxt : Context_Acc) - return Net - is - Clock : Node; - Clk : Valtyp; - Clk_Net : Net; - begin - Clock := Get_Clock_Expression (Call); - if Clock /= Null_Node then - Clk := Synth_Expression (Syn_Inst, Clock); - Clk_Net := Get_Net (Ctxt, Clk); - else - Clock := Get_Default_Clock (Call); - pragma Assert (Clock /= Null_Node); - Clk_Net := Synth_PSL_Expression (Syn_Inst, Get_Psl_Boolean (Clock)); - end if; - return Clk_Net; - end Synth_Psl_Function_Clock; - - function Synth_Psl_Prev (Syn_Inst : Synth_Instance_Acc; Call : Node) - return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Count : constant Node := Get_Count_Expression (Call); - Count_Val : Valtyp; - Dff : Net; - Expr : Valtyp; - Clk_Net : Net; - Num : Int64; - begin - Expr := Synth_Expression_With_Basetype (Syn_Inst, Get_Expression (Call)); - - Clk_Net := Synth_Psl_Function_Clock (Syn_Inst, Call, Ctxt); - - if Count /= Null_Node then - Count_Val := Synth_Expression (Syn_Inst, Count); - Num := Read_Discrete (Count_Val); - pragma Assert (Num >= 1); - else - Num := 1; - end if; - - Dff := Get_Net (Ctxt, Expr); - for I in 1 .. Num loop - Dff := Build_Dff (Ctxt, Clk_Net, Dff); - Set_Location (Dff, Call); - end loop; - - return Create_Value_Net (Dff, Expr.Typ); - end Synth_Psl_Prev; - - function Synth_Psl_Stable (Syn_Inst : Synth_Instance_Acc; Call : Node) - return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - DffCurr : Net; - Dff : Net; - Expr : Valtyp; - Clk_Net : Net; - Res : Net; - begin - Expr := Synth_Expression_With_Basetype (Syn_Inst, Get_Expression (Call)); - - Clk_Net := Synth_Psl_Function_Clock (Syn_Inst, Call, Ctxt); - - DffCurr := Get_Net (Ctxt, Expr); - Set_Location (DffCurr, Call); - Dff := Build_Dff (Ctxt, Clk_Net, DffCurr); - Set_Location (Dff, Call); - - Res := Build_Compare(Ctxt, Id_Eq, DffCurr, Dff); - Set_Location (Res, Call); - - return Create_Value_Net (Res, Boolean_Type); - - end Synth_Psl_Stable; - - function Synth_Psl_Rose (Syn_Inst : Synth_Instance_Acc; Call : Node) - return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - DffCurr : Net; - Dff : Net; - NotDff : Net; - Clk_Net : Net; - Expr : Valtyp; - Res : Net; - begin - Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); - - Clk_Net := Synth_Psl_Function_Clock (Syn_Inst, Call, Ctxt); - - DffCurr := Get_Net (Ctxt, Expr); - Set_Location (DffCurr, Call); - Dff := Build_Dff (Ctxt, Clk_Net, DffCurr); - Set_Location (Dff, Call); - - NotDff := Build_Monadic (Ctxt, Id_Not, Dff); - Set_Location (NotDff, Call); - - Res := Build_Dyadic (Ctxt, Id_And, - NotDff, DffCurr); - Set_Location (Res, Call); - - return Create_Value_Net (Res, Boolean_Type); - - end Synth_Psl_Rose; - - function Synth_Psl_Fell (Syn_Inst : Synth_Instance_Acc; Call : Node) - return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - DffCurr : Net; - NotDffCurr : Net; - Dff : Net; - Clk_Net : Net; - Expr : Valtyp; - Res : Net; - begin - Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); - - Clk_Net := Synth_Psl_Function_Clock(Syn_Inst, Call, Ctxt); - - DffCurr := Get_Net (Ctxt, Expr); - Set_Location (DffCurr, Call); - Dff := Build_Dff (Ctxt, Clk_Net, DffCurr); - Set_Location (Dff, Call); - - NotDffCurr := Build_Monadic (Ctxt, Id_Not, DffCurr); - Set_Location (NotDffCurr, Call); - - Res := Build_Dyadic (Ctxt, Id_And, Dff, NotDffCurr); - Set_Location (Res, Call); - - return Create_Value_Net (Res, Boolean_Type); - - end Synth_Psl_Fell; - - function Synth_Onehot0 (Ctxt : Context_Acc; DffCurr : Net; Call : Node; - Vlen : Uns32) - return Net - is - DffZero : Net; - DffOne : Net; - DffOneHot0 : Net; - Res : Net; - begin - -- Create a constant vector of 0 for comparing - DffZero := Build2_Const_Uns(Ctxt, 0, Vlen); - - -- Create vector of value 1 for subtraction - DffOne := Build2_Const_Uns(Ctxt, 1, Vlen); - - -- Subtraction -> v - 1 - DffOneHot0 := Build_Dyadic (Ctxt, Id_Sub, DffCurr, DffOne); - Set_Location (DffOneHot0, Call); - - -- Binary And -> v & (v - 1) - DffOneHot0 := Build_Dyadic (Ctxt, Id_And, DffCurr, DffOneHot0); - Set_Location (DffOneHot0, Call); - - -- Compare with 0 -> (v & (v - 1)) == 0 - Res := Build_Compare (Ctxt, Id_Eq, DffOneHot0, DffZero); - Set_Location (Res, Call); - - return Res; - end Synth_Onehot0; - - function Synth_Psl_Onehot (Syn_Inst : Synth_Instance_Acc; Call : Node) - return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Expr : Valtyp; - DffCurr : Net; - DffCurrIsNotZero : Net; - DffOneHot0 : Net; - Res : Net; - Vlen : Uns32; - begin - -- Get parameter & its length - Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); - Vlen := Expr.Typ.W; - - -- First get net of parameter - DffCurr := Get_Net (Ctxt, Expr); - Set_Location (DffCurr, Call); - - -- Compare parameter with 0 -> v != 0 - DffCurrIsNotZero := Build_Compare (Ctxt, Id_Ne, DffCurr, - Build2_Const_Uns(Ctxt, 0, Vlen)); - Set_Location (DffCurrIsNotZero, Call); - - -- Synth onehot0 - DffOneHot0 := Synth_Onehot0 (Ctxt, DffCurr, Call, Vlen); - Set_Location (DffOneHot0, Call); - - -- Final Binary And -> (v != 0) & ((v & (v - 1)) == 0) - Res := Build_Dyadic (Ctxt, Id_And, DffOneHot0, DffCurrIsNotZero); - Set_Location (Res, Call); - - return Create_Value_Net (Res, Boolean_Type); - end Synth_Psl_Onehot; - - function Synth_Psl_Onehot0 (Syn_Inst : Synth_Instance_Acc; Call : Node) - return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Expr : Valtyp; - Vlen : Uns32; - DffCurr : Net; - Res : Net; - begin - -- Get parameter & its length - Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); - Vlen := Expr.Typ.W; - - -- First get net of parameter - DffCurr := Get_Net (Ctxt, Expr); - Set_Location (DffCurr, Call); - - -- Synth onehot0 - Res := Synth_Onehot0 (Ctxt, DffCurr, Call, Vlen); - - return Create_Value_Net (Res, Boolean_Type); - end Synth_Psl_Onehot0; - - subtype And_Or_Module_Id is Module_Id range Id_And .. Id_Or; - - function Synth_Short_Circuit (Syn_Inst : Synth_Instance_Acc; - Id : And_Or_Module_Id; - Left_Expr : Node; - Right_Expr : Node; - Typ : Type_Acc; - Expr : Node) return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Left : Valtyp; - Right : Valtyp; - Val : Int64; - N : Net; - begin - -- The short-circuit value. - case Id is - when Id_And => - Val := 0; - when Id_Or => - Val := 1; - end case; - - Left := Synth_Expression_With_Type (Syn_Inst, Left_Expr, Typ); - if Left = No_Valtyp then - -- Propagate error. - return No_Valtyp; - end if; - if Is_Static_Val (Left.Val) - and then Get_Static_Discrete (Left) = Val - then - -- Short-circuit when the left operand determines the result. - return Create_Value_Discrete (Val, Boolean_Type); - end if; - - Strip_Const (Left); - Right := Synth_Expression_With_Type (Syn_Inst, Right_Expr, Typ); - if Right = No_Valtyp then - -- Propagate error. - return No_Valtyp; - end if; - Strip_Const (Right); - - if Is_Static_Val (Right.Val) - and then Get_Static_Discrete (Right) = Val - then - -- If the right operand can determine the result, return it. - return Create_Value_Discrete (Val, Boolean_Type); - end if; - - -- Return a static value if both operands are static. - -- Note: we know the value of left if it is not constant. - if Is_Static_Val (Left.Val) and then Is_Static_Val (Right.Val) then - Val := Get_Static_Discrete (Right); - return Create_Value_Discrete (Val, Boolean_Type); - end if; - - -- Non-static result. - N := Build_Dyadic (Ctxt, Id, - Get_Net (Ctxt, Left), Get_Net (Ctxt, Right)); - Set_Location (N, Expr); - return Create_Value_Net (N, Boolean_Type); - end Synth_Short_Circuit; - - function Synth_Expression_With_Type (Syn_Inst : Synth_Instance_Acc; - Expr : Node; - Expr_Type : Type_Acc) return Valtyp is - begin - case Get_Kind (Expr) is - when Iir_Kinds_Dyadic_Operator => - declare - Imp : constant Node := Get_Implementation (Expr); - Def : constant Iir_Predefined_Functions := - Get_Implicit_Definition (Imp); - Edge : Net; - begin - -- Match clock-edge - if Def = Iir_Predefined_Boolean_And then - Edge := Synth_Clock_Edge (Syn_Inst, - Get_Left (Expr), Get_Right (Expr)); - if Edge /= No_Net then - return Create_Value_Net (Edge, Boolean_Type); - end if; - end if; - - -- Specially handle short-circuit operators. - case Def is - when Iir_Predefined_Boolean_And => - return Synth_Short_Circuit - (Syn_Inst, Id_And, Get_Left (Expr), Get_Right (Expr), - Boolean_Type, Expr); - when Iir_Predefined_Boolean_Or => - return Synth_Short_Circuit - (Syn_Inst, Id_Or, Get_Left (Expr), Get_Right (Expr), - Boolean_Type, Expr); - when Iir_Predefined_Bit_And => - return Synth_Short_Circuit - (Syn_Inst, Id_And, Get_Left (Expr), Get_Right (Expr), - Bit_Type, Expr); - when Iir_Predefined_Bit_Or => - return Synth_Short_Circuit - (Syn_Inst, Id_Or, Get_Left (Expr), Get_Right (Expr), - Bit_Type, Expr); - when Iir_Predefined_None => - if Error_Ieee_Operator (Imp, Expr) then - return No_Valtyp; - else - return Synth_User_Operator - (Syn_Inst, Get_Left (Expr), Get_Right (Expr), Expr); - end if; - when others => - return Synth_Dyadic_Operation - (Syn_Inst, Imp, - Get_Left (Expr), Get_Right (Expr), Expr); - end case; - end; - when Iir_Kinds_Monadic_Operator => - declare - Imp : constant Node := Get_Implementation (Expr); - Def : constant Iir_Predefined_Functions := - Get_Implicit_Definition (Imp); - begin - if Def = Iir_Predefined_None then - if Error_Ieee_Operator (Imp, Expr) then - return No_Valtyp; - else - return Synth_User_Operator - (Syn_Inst, Get_Operand (Expr), Null_Node, Expr); - end if; - else - return Synth_Monadic_Operation - (Syn_Inst, Imp, Get_Operand (Expr), Expr); - end if; - end; - when Iir_Kind_Simple_Name - | Iir_Kind_Selected_Name - | Iir_Kind_Interface_Signal_Declaration -- For PSL. - | Iir_Kind_Signal_Declaration -- For PSL. - | Iir_Kind_Implicit_Dereference - | Iir_Kind_Dereference => - declare - Res : Valtyp; - begin - Res := Synth_Name (Syn_Inst, Expr); - if Res.Typ.W = 0 and then Res.Val.Kind /= Value_Memory then - -- This is a null object. As nothing can be done about it, - -- returns 0. - return Create_Value_Memtyp (Create_Memory_Zero (Res.Typ)); - end if; - return Res; - end; - when Iir_Kind_Reference_Name => - -- Only used for anonymous signals in internal association. - return Synth_Expression_With_Type - (Syn_Inst, Get_Named_Entity (Expr), Expr_Type); - when Iir_Kind_Anonymous_Signal_Declaration => - return Synth_Expression_With_Type - (Syn_Inst, Get_Expression (Expr), Expr_Type); - when Iir_Kind_Indexed_Name - | Iir_Kind_Slice_Name => - declare - Base : Valtyp; - Typ : Type_Acc; - Off : Value_Offsets; - Res : Valtyp; - - Dyn : Dyn_Name; - begin - Synth_Assignment_Prefix (Syn_Inst, Expr, Base, Typ, Off, Dyn); - if Dyn.Voff = No_Net and then Is_Static (Base.Val) then - Res := Create_Value_Memory (Typ); - Copy_Memory - (Res.Val.Mem, Base.Val.Mem + Off.Mem_Off, Typ.Sz); - return Res; - end if; - return Synth_Read_Memory - (Syn_Inst, Base, Typ, Off.Net_Off, Dyn, Expr); - end; - when Iir_Kind_Selected_Element => - declare - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Idx : constant Iir_Index32 := - Get_Element_Position (Get_Named_Entity (Expr)); - Pfx : constant Node := Get_Prefix (Expr); - Res_Typ : Type_Acc; - N : Net; - Val : Valtyp; - Res : Valtyp; - begin - Val := Synth_Expression (Syn_Inst, Pfx); - Strip_Const (Val); - Res_Typ := Val.Typ.Rec.E (Idx + 1).Typ; - if Res_Typ.W = 0 and then Val.Val.Kind /= Value_Memory then - -- This is a null object. As nothing can be done about it, - -- returns 0. - return Create_Value_Memtyp (Create_Memory_Zero (Res_Typ)); - elsif Is_Static (Val.Val) then - Res := Create_Value_Memory (Res_Typ); - Copy_Memory - (Res.Val.Mem, Val.Val.Mem + Val.Typ.Rec.E (Idx + 1).Moff, - Res_Typ.Sz); - return Res; - else - N := Build_Extract - (Ctxt, Get_Net (Ctxt, Val), - Val.Typ.Rec.E (Idx + 1).Boff, Get_Type_Width (Res_Typ)); - Set_Location (N, Expr); - return Create_Value_Net (N, Res_Typ); - end if; - end; - when Iir_Kind_Character_Literal => - return Synth_Expression_With_Type - (Syn_Inst, Get_Named_Entity (Expr), Expr_Type); - when Iir_Kind_Integer_Literal => - declare - Res : Valtyp; - begin - Res := Create_Value_Memory (Expr_Type); - Write_Discrete (Res, Get_Value (Expr)); - return Res; - end; - when Iir_Kind_Floating_Point_Literal => - return Create_Value_Float (Get_Fp_Value (Expr), Expr_Type); - when Iir_Kind_Physical_Int_Literal - | Iir_Kind_Physical_Fp_Literal => - return Create_Value_Discrete - (Get_Physical_Value (Expr), Expr_Type); - when Iir_Kind_String_Literal8 => - return Synth_String_Literal (Syn_Inst, Expr, Expr_Type); - when Iir_Kind_Enumeration_Literal => - return Synth_Name (Syn_Inst, Expr); - when Iir_Kind_Type_Conversion => - return Synth_Type_Conversion (Syn_Inst, Expr); - when Iir_Kind_Qualified_Expression => - return Synth_Expression_With_Type - (Syn_Inst, Get_Expression (Expr), - Get_Subtype_Object (Syn_Inst, Get_Type (Get_Type_Mark (Expr)))); - when Iir_Kind_Function_Call => - declare - Imp : constant Node := Get_Implementation (Expr); - begin - case Get_Implicit_Definition (Imp) is - when Iir_Predefined_Pure_Functions - | Iir_Predefined_Ieee_Numeric_Std_Binary_Operators => - return Synth_Operator_Function_Call (Syn_Inst, Expr); - when Iir_Predefined_None => - return Synth_User_Function_Call (Syn_Inst, Expr); - when others => - return Synth_Predefined_Function_Call (Syn_Inst, Expr); - end case; - end; - when Iir_Kind_Aggregate => - return Synth.Vhdl_Aggr.Synth_Aggregate (Syn_Inst, Expr, Expr_Type); - when Iir_Kind_Simple_Aggregate => - return Synth_Simple_Aggregate (Syn_Inst, Expr); - when Iir_Kind_Parenthesis_Expression => - return Synth_Expression_With_Type - (Syn_Inst, Get_Expression (Expr), Expr_Type); - when Iir_Kind_Left_Array_Attribute => - declare - B : Bound_Type; - begin - B := Synth_Array_Attribute (Syn_Inst, Expr); - return Create_Value_Discrete (Int64 (B.Left), Expr_Type); - end; - when Iir_Kind_Right_Array_Attribute => - declare - B : Bound_Type; - begin - B := Synth_Array_Attribute (Syn_Inst, Expr); - return Create_Value_Discrete (Int64 (B.Right), Expr_Type); - end; - when Iir_Kind_High_Array_Attribute => - declare - B : Bound_Type; - V : Int32; - begin - B := Synth_Array_Attribute (Syn_Inst, Expr); - case B.Dir is - when Dir_To => - V := B.Right; - when Dir_Downto => - V := B.Left; - end case; - return Create_Value_Discrete (Int64 (V), Expr_Type); - end; - when Iir_Kind_Low_Array_Attribute => - declare - B : Bound_Type; - V : Int32; - begin - B := Synth_Array_Attribute (Syn_Inst, Expr); - case B.Dir is - when Dir_To => - V := B.Left; - when Dir_Downto => - V := B.Right; - end case; - return Create_Value_Discrete (Int64 (V), Expr_Type); - end; - when Iir_Kind_Length_Array_Attribute => - declare - B : Bound_Type; - begin - B := Synth_Array_Attribute (Syn_Inst, Expr); - return Create_Value_Discrete (Int64 (B.Len), Expr_Type); - end; - when Iir_Kind_Ascending_Array_Attribute => - declare - B : Bound_Type; - V : Int64; - begin - B := Synth_Array_Attribute (Syn_Inst, Expr); - case B.Dir is - when Dir_To => - V := 1; - when Dir_Downto => - V := 0; - end case; - return Create_Value_Discrete (V, Expr_Type); - end; - - when Iir_Kind_Pos_Attribute - | Iir_Kind_Val_Attribute => - declare - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Param : constant Node := Get_Parameter (Expr); - V : Valtyp; - Dtype : Type_Acc; - begin - V := Synth_Expression (Syn_Inst, Param); - Dtype := Get_Subtype_Object (Syn_Inst, Get_Type (Expr)); - -- FIXME: to be generalized. Not always as simple as a - -- subtype conversion. - return Synth_Subtype_Conversion (Ctxt, V, Dtype, False, Expr); - end; - when Iir_Kind_Low_Type_Attribute => - return Synth_Low_High_Type_Attribute (Syn_Inst, Expr, Dir_To); - when Iir_Kind_High_Type_Attribute => - return Synth_Low_High_Type_Attribute (Syn_Inst, Expr, Dir_Downto); - when Iir_Kind_Value_Attribute => - return Synth_Value_Attribute (Syn_Inst, Expr); - when Iir_Kind_Image_Attribute => - return Synth_Image_Attribute (Syn_Inst, Expr); - when Iir_Kind_Instance_Name_Attribute => - return Synth_Instance_Name_Attribute (Syn_Inst, Expr); - when Iir_Kind_Null_Literal => - return Create_Value_Access (Null_Heap_Index, Expr_Type); - when Iir_Kind_Allocator_By_Subtype => - declare - T : Type_Acc; - Acc : Heap_Index; - begin - T := Synth.Decls.Synth_Subtype_Indication - (Syn_Inst, Get_Subtype_Indication (Expr)); - Acc := Allocate_By_Type (T); - return Create_Value_Access (Acc, Expr_Type); - end; - when Iir_Kind_Allocator_By_Expression => - declare - V : Valtyp; - Acc : Heap_Index; - begin - V := Synth_Expression_With_Type - (Syn_Inst, Get_Expression (Expr), Expr_Type.Acc_Acc); - Acc := Allocate_By_Value (V); - return Create_Value_Access (Acc, Expr_Type); - end; - when Iir_Kind_Stable_Attribute => - Error_Msg_Synth (+Expr, "signal attribute not supported"); - return No_Valtyp; - when Iir_Kind_Psl_Prev => - return Synth_Psl_Prev (Syn_Inst, Expr); - when Iir_Kind_Psl_Stable => - return Synth_Psl_Stable (Syn_Inst, Expr); - when Iir_Kind_Psl_Rose => - return Synth_Psl_Rose(Syn_Inst, Expr); - when Iir_Kind_Psl_Fell => - return Synth_Psl_Fell(Syn_Inst, Expr); - when Iir_Kind_Psl_Onehot => - return Synth_Psl_Onehot(Syn_Inst, Expr); - when Iir_Kind_Psl_Onehot0 => - return Synth_Psl_Onehot0(Syn_Inst, Expr); - when Iir_Kind_Overflow_Literal => - Error_Msg_Synth (+Expr, "out of bound expression"); - return No_Valtyp; - when others => - Error_Kind ("synth_expression_with_type", Expr); - end case; - end Synth_Expression_With_Type; - - function Synth_Expression (Syn_Inst : Synth_Instance_Acc; Expr : Node) - return Valtyp - is - Etype : Node; - begin - Etype := Get_Type (Expr); - - case Get_Kind (Expr) is - when Iir_Kind_High_Array_Attribute - | Iir_Kind_Low_Array_Attribute - | Iir_Kind_Integer_Literal => - -- The type of this attribute is the type of the index, which is - -- not synthesized as atype (only as an index). - -- For integer_literal, the type is not really needed, and it - -- may be created by static evaluation of an array attribute. - Etype := Get_Base_Type (Etype); - when others => - null; - end case; - - return Synth_Expression_With_Type - (Syn_Inst, Expr, Get_Subtype_Object (Syn_Inst, Etype)); - end Synth_Expression; - - function Synth_Expression_With_Basetype - (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp - is - Basetype : Type_Acc; - begin - Basetype := Get_Subtype_Object - (Syn_Inst, Get_Base_Type (Get_Type (Expr))); - return Synth_Expression_With_Type (Syn_Inst, Expr, Basetype); - end Synth_Expression_With_Basetype; -end Synth.Expr; diff --git a/src/synth/synth-expr.ads b/src/synth/synth-expr.ads deleted file mode 100644 index 8dac335c4..000000000 --- a/src/synth/synth-expr.ads +++ /dev/null @@ -1,152 +0,0 @@ --- Expressions synthesis. --- Copyright (C) 2017 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with Ada.Unchecked_Deallocation; - -with Types; use Types; - -with PSL.Types; -with Vhdl.Nodes; use Vhdl.Nodes; - -with Netlists; use Netlists; -with Netlists.Builders; use Netlists.Builders; - -with Synth.Source; -with Synth.Objtypes; use Synth.Objtypes; -with Synth.Values; use Synth.Values; -with Synth.Vhdl_Context; use Synth.Vhdl_Context; - -package Synth.Expr is - -- Perform a subtype conversion. Check constraints. - function Synth_Subtype_Conversion (Ctxt : Context_Acc; - Vt : Valtyp; - Dtype : Type_Acc; - Bounds : Boolean; - Loc : Source.Syn_Src) - return Valtyp; - - -- For a static value V, return the value. - function Get_Static_Discrete (V : Valtyp) return Int64; - - -- Return the memory (as a memtyp) of static value V. - function Get_Value_Memtyp (V : Valtyp) return Memtyp; - - -- Return True only if discrete value V is known to be positive or 0. - -- False means either not positive or unknown. - function Is_Positive (V : Valtyp) return Boolean; - - -- Return the bounds of a one dimensional array/vector type and the - -- width of the element. - procedure Get_Onedimensional_Array_Bounds - (Typ : Type_Acc; Bnd : out Bound_Type; El_Typ : out Type_Acc); - - -- Create an array subtype from bound BND. - function Create_Onedimensional_Array_Subtype - (Btyp : Type_Acc; Bnd : Bound_Type) return Type_Acc; - - procedure From_Std_Logic (Enum : Int64; Val : out Uns32; Zx : out Uns32); - procedure From_Bit (Enum : Int64; Val : out Uns32); - procedure To_Logic - (Enum : Int64; Etype : Type_Acc; Val : out Uns32; Zx : out Uns32); - - -- Try to match: clk'event and clk = X - -- or: clk = X and clk'event - -- where X is '0' or '1'. - function Synth_Clock_Edge - (Syn_Inst : Synth_Instance_Acc; Left, Right : Node) return Net; - - procedure Concat_Array - (Ctxt : Context_Acc; Arr : in out Net_Array; N : out Net); - - -- Synthesize EXPR. The expression must be self-constrained. - -- If EN is not No_Net, the execution is controlled by EN. This is used - -- for assertions and checks. - function Synth_Expression - (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp; - - -- Same as Synth_Expression, but the expression may be constrained by - -- EXPR_TYPE. - function Synth_Expression_With_Type (Syn_Inst : Synth_Instance_Acc; - Expr : Node; - Expr_Type : Type_Acc) return Valtyp; - - -- Use base type of EXPR to synthesize EXPR. Useful when the type of - -- EXPR is defined by itself or a range. - function Synth_Expression_With_Basetype (Syn_Inst : Synth_Instance_Acc; - Expr : Node) return Valtyp; - - function Synth_PSL_Expression - (Syn_Inst : Synth_Instance_Acc; Expr : PSL.Types.PSL_Node) return Net; - - function Synth_Bounds_From_Range (Syn_Inst : Synth_Instance_Acc; - Atype : Node) return Bound_Type; - - function Synth_Array_Bounds (Syn_Inst : Synth_Instance_Acc; - Atype : Node; - Dim : Dim_Type) return Bound_Type; - - function Build_Discrete_Range_Type - (L : Int64; R : Int64; Dir : Direction_Type) return Discrete_Range_Type; - function Synth_Discrete_Range_Expression - (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Discrete_Range_Type; - function Synth_Float_Range_Expression - (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Float_Range_Type; - - procedure Synth_Discrete_Range (Syn_Inst : Synth_Instance_Acc; - Bound : Node; - Rng : out Discrete_Range_Type); - - procedure Synth_Slice_Suffix (Syn_Inst : Synth_Instance_Acc; - Name : Node; - Pfx_Bnd : Bound_Type; - El_Typ : Type_Acc; - Res_Bnd : out Bound_Type; - Inp : out Net; - Off : out Value_Offsets); - - -- If VOFF is No_Net then OFF is valid, if VOFF is not No_Net then - -- OFF is 0. - procedure Synth_Indexed_Name (Syn_Inst : Synth_Instance_Acc; - Name : Node; - Pfx_Type : Type_Acc; - Voff : out Net; - Off : out Value_Offsets); - - -- Return the type of EXPR (an object) without evaluating it (except when - -- needed, like bounds of a slice). - function Synth_Type_Of_Object (Syn_Inst : Synth_Instance_Acc; Expr : Node) - return Type_Acc; - - -- Conversion to logic vector. - - 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 Free_Logvec_Array is new Ada.Unchecked_Deallocation - (Logvec_Array, Logvec_Array_Acc); - - -- Convert W bits from OFF of VAL to a Logvec_Array. - -- OFF and W are offset and width in bit representation. - procedure Value2logvec (Val : Memtyp; - Off : Uns32; - W : Width; - Vec : in out Logvec_Array; - Vec_Off : in out Uns32; - Has_Zx : in out Boolean); -end Synth.Expr; diff --git a/src/synth/synth-insts.adb b/src/synth/synth-insts.adb deleted file mode 100644 index ac37f8b0a..000000000 --- a/src/synth/synth-insts.adb +++ /dev/null @@ -1,1751 +0,0 @@ --- Instantiation synthesis. --- Copyright (C) 2019 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with GNAT.SHA1; - -with Types; use Types; -with Types_Utils; use Types_Utils; -with Files_Map; -with Name_Table; -with Libraries; -with Hash; use Hash; -with Dyn_Tables; -with Interning; -with Synthesis; use Synthesis; - -with Grt.Algos; - -with Netlists; use Netlists; -with Netlists.Builders; use Netlists.Builders; -with Netlists.Cleanup; -with Netlists.Memories; -with Netlists.Expands; -with Netlists.Concats; -with Netlists.Folds; - -with Vhdl.Utils; use Vhdl.Utils; -with Vhdl.Errors; -with Vhdl.Ieee.Math_Real; - -with Synth.Memtype; use Synth.Memtype; -with Synth.Objtypes; use Synth.Objtypes; -with Synth.Values; use Synth.Values; -with Synth.Vhdl_Environment; use Synth.Vhdl_Environment.Env; -with Synth.Stmts; use Synth.Stmts; -with Synth.Decls; use Synth.Decls; -with Synth.Expr; use Synth.Expr; -with Synth.Source; use Synth.Source; -with Synth.Debugger; -with Synth.Vhdl_Files; -with Synth.Errors; - -package body Synth.Insts is - Root_Instance : Synth_Instance_Acc; - - function Mode_To_Port_Kind (Mode : Iir_Mode) return Port_Kind is - begin - case Mode is - when Iir_In_Mode => - return Port_In; - when Iir_Buffer_Mode - | Iir_Out_Mode => - return Port_Out; - when Iir_Inout_Mode => - return Port_Inout; - when Iir_Linkage_Mode - | Iir_Unknown_Mode => - raise Synth_Error; - end case; - end Mode_To_Port_Kind; - - -- Parameters that define an instance. - type Inst_Params is record - -- Declaration: either the entity or the component. - Decl : Node; - -- Implementation: the architecture or Null_Node for black boxes. - Arch : Node; - -- Configuration (Null_Node for black boxes). - Config : Node; - -- Values of generics. - Syn_Inst : Synth_Instance_Acc; - -- Encoding if the instance name. - Encoding : Name_Encoding; - end record; - - type Inst_Object is record - Decl : Node; - Arch : Node; - Config : Node; - Syn_Inst : Synth_Instance_Acc; - M : Module; - -- Encoding if the instance name. - Encoding : Name_Encoding; - end record; - - function Hash (Params : Inst_Params) return Hash_Value_Type - is - Res : Hash_Value_Type; - begin - Res := Hash_Value_Type (Params.Decl); - Res := Res xor Hash_Value_Type (Params.Arch); - Res := Res xor Hash_Value_Type (Params.Config); - -- TODO: hash generics - return Res; - end Hash; - - function Equal (Obj : Inst_Object; Params : Inst_Params) return Boolean - is - Inter : Node; - begin - if Obj.Decl /= Params.Decl - or else Obj.Arch /= Params.Arch - or else Obj.Config /= Params.Config - then - return False; - end if; - Inter := Get_Generic_Chain (Params.Decl); - while Inter /= Null_Node loop - if not Is_Equal (Get_Value (Obj.Syn_Inst, Inter), - Get_Value (Params.Syn_Inst, Inter)) - then - return False; - end if; - Inter := Get_Chain (Inter); - end loop; - - Inter := Get_Port_Chain (Params.Decl); - while Inter /= Null_Node loop - if not Is_Fully_Constrained_Type (Get_Type (Inter)) then - if not Are_Types_Equal (Get_Value (Obj.Syn_Inst, Inter).Typ, - Get_Value (Params.Syn_Inst, Inter).Typ) - then - return False; - end if; - end if; - Inter := Get_Chain (Inter); - end loop; - - return True; - end Equal; - - procedure Hash_Uns64 (C : in out GNAT.SHA1.Context; Val : Uns64) - is - V : Uns64; - S : String (1 .. 8); - begin - -- Store to S using little endianness. - V := Val; - for I in S'Range loop - S (I) := Character'Val (V and 16#ff#); - V := Shift_Right (V, 8); - end loop; - - GNAT.SHA1.Update (C, S); - end Hash_Uns64; - - procedure Hash_Memory (C : in out GNAT.SHA1.Context; - M : Memory_Ptr; - Typ : Type_Acc) - is - S : String (1 .. Natural (Typ.Sz)); - for S'Address use M (0)'Address; - pragma Import (Ada, S); - begin - GNAT.SHA1.Update (C, S); - end Hash_Memory; - - procedure Hash_Bound (C : in out GNAT.SHA1.Context; B : Bound_Type) is - begin - Hash_Uns64 (C, Direction_Type'Pos (B.Dir)); - Hash_Uns64 (C, To_Uns64 (Int64 (B.Left))); - Hash_Uns64 (C, To_Uns64 (Int64 (B.Right))); - end Hash_Bound; - - procedure Hash_Bounds (C : in out GNAT.SHA1.Context; Typ : Type_Acc) is - begin - case Typ.Kind is - when Type_Vector => - Hash_Bound (C, Typ.Vbound); - when Type_Array => - for I in Typ.Abounds.D'Range loop - Hash_Bound (C, Typ.Abounds.D (I)); - end loop; - when others => - raise Internal_Error; - end case; - end Hash_Bounds; - - procedure Hash_Const (C : in out GNAT.SHA1.Context; - Val : Value_Acc; - Typ : Type_Acc) is - begin - case Val.Kind is - when Value_Memory => - Hash_Memory (C, Val.Mem, Typ); - when Value_Const => - Hash_Const (C, Val.C_Val, Typ); - when Value_Alias => - if Val.A_Off /= (0, 0) then - raise Internal_Error; - end if; - Hash_Const (C, Val.A_Obj, Typ); - when Value_Net - | Value_Wire - | Value_File => - raise Internal_Error; - end case; - end Hash_Const; - - function Get_Source_Identifier (Decl : Node) return Name_Id - is - use Files_Map; - use Name_Table; - Loc : constant Location_Type := Get_Location (Decl); - Len : constant Natural := Get_Name_Length (Get_Identifier (Decl)); - subtype Ident_Str is String (1 .. Len); - File : Source_File_Entry; - Pos : Source_Ptr; - Buf : File_Buffer_Acc; - begin - Location_To_File_Pos (Loc, File, Pos); - Buf := Get_File_Source (File); - return Get_Identifier - (Ident_Str (Buf (Pos .. Pos + Source_Ptr (Len - 1)))); - end Get_Source_Identifier; - - function Create_Module_Name (Params : Inst_Params) return Sname - is - use GNAT.SHA1; - Decl : constant Node := Params.Decl; - Id : constant Name_Id := Get_Identifier (Decl); - Generics : constant Node := Get_Generic_Chain (Decl); - Ports : constant Node := Get_Port_Chain (Decl); - Ctxt : GNAT.SHA1.Context; - Has_Hash : Boolean; - - -- Create a buffer, store the entity name. - -- For each generic: - -- * write the value for integers. - -- * write the identifier for enumerated type with only non-extended - -- identifiers. - -- * hash all other values - -- Append the hash if any. - use Name_Table; - Id_Len : constant Natural := Get_Name_Length (Id); - Str_Len : constant Natural := Id_Len + 512; - - -- True in practice (and used to set the length of STR, but doesn't work - -- anymore with gcc/gnat 11. - -- pragma Assert (GNAT.SHA1.Hash_Length = 20); - Str : String (1 .. Str_Len + 41); - Len : Natural; - - Gen_Decl : Node; - Gen : Valtyp; - begin - Len := Id_Len; - Str (1 .. Len) := Get_Name_Ptr (Id) (1 .. Len); - - Has_Hash := False; - - case Params.Encoding is - when Name_Hash => - Ctxt := GNAT.SHA1.Initial_Context; - - Gen_Decl := Generics; - while Gen_Decl /= Null_Node loop - Gen := Get_Value (Params.Syn_Inst, Gen_Decl); - Strip_Const (Gen); - case Gen.Typ.Kind is - when Type_Discrete => - declare - S : constant String := - Uns64'Image (To_Uns64 (Read_Discrete (Gen))); - begin - if Len + S'Length > Str_Len then - Has_Hash := True; - Hash_Const (Ctxt, Gen.Val, Gen.Typ); - else - Str (Len + 1 .. Len + S'Length) := S; - pragma Assert (Str (Len + 1) = ' '); - Str (Len + 1) := '_'; -- Overwrite the space. - Len := Len + S'Length; - end if; - end; - when others => - Has_Hash := True; - Hash_Const (Ctxt, Gen.Val, Gen.Typ); - end case; - Gen_Decl := Get_Chain (Gen_Decl); - end loop; - - declare - Port_Decl : Node; - Port_Typ : Type_Acc; - begin - Port_Decl := Ports; - while Port_Decl /= Null_Node loop - if not Is_Fully_Constrained_Type (Get_Type (Port_Decl)) then - Port_Typ := Get_Value (Params.Syn_Inst, Port_Decl).Typ; - Has_Hash := True; - Hash_Bounds (Ctxt, Port_Typ); - end if; - Port_Decl := Get_Chain (Port_Decl); - end loop; - end; - if not Has_Hash and then Generics = Null_Node then - -- Simple case: same name. - -- TODO: what about two entities with the same identifier but - -- declared in two different libraries ? - -- TODO: what about extended identifiers ? - return New_Sname_User (Id, No_Sname); - end if; - - if Has_Hash then - Str (Len + 1) := '_'; - Len := Len + 1; - Str (Len + 1 .. Len + 40) := GNAT.SHA1.Digest (Ctxt); - Len := Len + 40; - end if; - - when Name_Asis - | Name_Parameters => - return New_Sname_User (Get_Source_Identifier (Decl), No_Sname); - - when Name_Index => - -- TODO. - raise Internal_Error; - end case; - - - return New_Sname_User (Get_Identifier (Str (1 .. Len)), No_Sname); - end Create_Module_Name; - - -- Create the name of an interface. - function Get_Encoded_Name_Id (Decl : Node; Enc : Name_Encoding) - return Name_Id is - begin - case Enc is - when Name_Asis - | Name_Parameters => - return Get_Source_Identifier (Decl); - when others => - return Get_Identifier (Decl); - end case; - end Get_Encoded_Name_Id; - - -- Create the name of an interface. - function Create_Inter_Name (Decl : Node; Enc : Name_Encoding) - return Sname is - begin - return New_Sname_User (Get_Encoded_Name_Id (Decl, Enc), No_Sname); - end Create_Inter_Name; - - procedure Copy_Object_Subtype (Syn_Inst : Synth_Instance_Acc; - Inter_Type : Node; - Proto_Inst : Synth_Instance_Acc) - is - Inter_Typ : Type_Acc; - begin - case Get_Kind (Inter_Type) is - when Iir_Kind_Array_Subtype_Definition => - if Synth.Decls.Has_Element_Subtype_Indication (Inter_Type) then - Copy_Object_Subtype - (Syn_Inst, Get_Element_Subtype (Inter_Type), Proto_Inst); - end if; - when others => - null; - end case; - Inter_Typ := Get_Subtype_Object (Proto_Inst, Inter_Type); - Create_Subtype_Object (Syn_Inst, Inter_Type, Inter_Typ); - end Copy_Object_Subtype; - - procedure Build_Object_Subtype (Syn_Inst : Synth_Instance_Acc; - Inter : Node; - Proto_Inst : Synth_Instance_Acc) is - begin - if Get_Declaration_Type (Inter) /= Null_Node then - Copy_Object_Subtype (Syn_Inst, Get_Type (Inter), Proto_Inst); - end if; - end Build_Object_Subtype; - - -- Return the number of ports for a type. A record type create one - -- port per immediate subelement. Sub-records are not expanded. - function Count_Nbr_Ports (Typ : Type_Acc) return Port_Nbr is - begin - case Typ.Kind is - when Type_Bit - | Type_Logic - | Type_Discrete - | Type_Float - | Type_Vector - | Type_Unbounded_Vector - | Type_Array - | Type_Unbounded_Array => - return 1; - when Type_Record - | Type_Unbounded_Record => - return Port_Nbr (Typ.Rec.Len); - when Type_Slice - | Type_Access - | Type_File - | Type_Protected => - raise Internal_Error; - end case; - end Count_Nbr_Ports; - - procedure Build_Ports_Desc (Descs : in out Port_Desc_Array; - Idx : in out Port_Nbr; - Pkind : Port_Kind; - Encoding : Name_Encoding; - Typ : Type_Acc; - Inter : Node) - is - Port_Sname : Sname; - begin - Port_Sname := Create_Inter_Name (Inter, Encoding); - - case Typ.Kind is - when Type_Bit - | Type_Logic - | Type_Discrete - | Type_Float - | Type_Vector - | Type_Unbounded_Vector - | Type_Array - | Type_Unbounded_Array => - Idx := Idx + 1; - Descs (Idx) := (Name => Port_Sname, - Is_Inout => Pkind = Port_Inout, - W => Get_Type_Width (Typ)); - when Type_Record - | Type_Unbounded_Record => - declare - Els : constant Node_Flist := Get_Elements_Declaration_List - (Get_Type (Inter)); - El : Node; - begin - for I in Typ.Rec.E'Range loop - El := Get_Nth_Element (Els, Natural (I - 1)); - Idx := Idx + 1; - Descs (Idx) := - (Name => New_Sname_User - (Get_Encoded_Name_Id (El, Encoding), Port_Sname), - Is_Inout => Pkind = Port_Inout, - W => Get_Type_Width (Typ.Rec.E (I).Typ)); - end loop; - end; - when Type_Slice - | Type_Access - | Type_File - | Type_Protected => - raise Internal_Error; - end case; - end Build_Ports_Desc; - - function Build (Params : Inst_Params) return Inst_Object - is - Decl : constant Node := Params.Decl; - Arch : constant Node := Params.Arch; - Imp : Node; - Syn_Inst : Synth_Instance_Acc; - Inter : Node; - Inter_Typ : Type_Acc; - Nbr_Inputs : Port_Nbr; - Nbr_Outputs : Port_Nbr; - Nbr_Params : Param_Nbr; - Cur_Module : Module; - Val : Valtyp; - Id : Module_Id; - begin - if Get_Kind (Params.Decl) = Iir_Kind_Component_Declaration then - pragma Assert (Params.Arch = Null_Node); - pragma Assert (Params.Config = Null_Node); - Imp := Params.Decl; - else - pragma Assert - (Get_Kind (Params.Config) = Iir_Kind_Block_Configuration); - Imp := Params.Arch; - end if; - - -- Create the instance. - Syn_Inst := Make_Instance (Root_Instance, Imp, No_Sname); - - -- Copy values for generics. - Inter := Get_Generic_Chain (Decl); - Nbr_Params := 0; - while Inter /= Null_Node loop - -- Bounds or range of the type. - Build_Object_Subtype (Syn_Inst, Inter, Params.Syn_Inst); - - -- Object. - Create_Object (Syn_Inst, Inter, Get_Value (Params.Syn_Inst, Inter)); - Nbr_Params := Nbr_Params + 1; - Inter := Get_Chain (Inter); - end loop; - - -- Allocate values and count inputs and outputs - Inter := Get_Port_Chain (Decl); - Nbr_Inputs := 0; - Nbr_Outputs := 0; - while Is_Valid (Inter) loop - -- Copy the type from PARAMS if needed. The subtype indication of - -- the port may reference objects that aren't anymore reachable - -- (particularly if it is a port of a component). So the subtype - -- cannot be regularly elaborated. - -- Also, for unconstrained subtypes, we need the constraint. - Build_Object_Subtype (Syn_Inst, Inter, Params.Syn_Inst); - Inter_Typ := Get_Value (Params.Syn_Inst, Inter).Typ; - - case Mode_To_Port_Kind (Get_Mode (Inter)) is - when Port_In => - Val := Create_Value_Net (No_Net, Inter_Typ); - Nbr_Inputs := Nbr_Inputs + Count_Nbr_Ports (Inter_Typ); - when Port_Out - | Port_Inout => - Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); - Nbr_Outputs := Nbr_Outputs + Count_Nbr_Ports (Inter_Typ); - end case; - Create_Object (Syn_Inst, Inter, Val); - Inter := Get_Chain (Inter); - end loop; - - -- Declare module. - -- Build it now because it may be referenced for instantiations before - -- being synthetized. - if Params.Encoding = Name_Parameters - and then Nbr_Params > 0 - then - Id := Id_User_Parameters; - else - Id := Id_User_None; - Nbr_Params := 0; - end if; - Cur_Module := New_User_Module (Get_Top_Module (Root_Instance), - Create_Module_Name (Params), Id, - Nbr_Inputs, Nbr_Outputs, Nbr_Params); - - if Id = Id_User_Parameters then - declare - Descs : Param_Desc_Array (1 .. Nbr_Params); - Ptype : Param_Type; - begin - Inter := Get_Generic_Chain (Decl); - Nbr_Params := 0; - while Inter /= Null_Node loop - -- Bounds or range of the type. - Ptype := Type_To_Param_Type (Get_Type (Inter)); - Nbr_Params := Nbr_Params + 1; - Descs (Nbr_Params) := - (Name => Create_Inter_Name (Inter, Params.Encoding), - Typ => Ptype); - Inter := Get_Chain (Inter); - end loop; - Set_Params_Desc (Cur_Module, Descs); - end; - end if; - - -- Add ports to module. - declare - Inports : Port_Desc_Array (1 .. Nbr_Inputs); - Outports : Port_Desc_Array (1 .. Nbr_Outputs); - Pkind : Port_Kind; - Vt : Valtyp; - begin - Inter := Get_Port_Chain (Decl); - Nbr_Inputs := 0; - Nbr_Outputs := 0; - while Is_Valid (Inter) loop - Pkind := Mode_To_Port_Kind (Get_Mode (Inter)); - Vt := Get_Value (Syn_Inst, Inter); - - case Pkind is - when Port_In => - Build_Ports_Desc (Inports, Nbr_Inputs, - Pkind, Params.Encoding, - Vt.Typ, Inter); - when Port_Out - | Port_Inout => - Build_Ports_Desc (Outports, Nbr_Outputs, - Pkind, Params.Encoding, - Vt.Typ, Inter); - end case; - Inter := Get_Chain (Inter); - end loop; - pragma Assert (Nbr_Inputs = Inports'Last); - pragma Assert (Nbr_Outputs = Outports'Last); - Set_Ports_Desc (Cur_Module, Inports, Outports); - end; - - return Inst_Object'(Decl => Decl, - Arch => Arch, - Config => Params.Config, - Syn_Inst => Syn_Inst, - M => Cur_Module, - Encoding => Params.Encoding); - end Build; - - package Insts_Interning is new Interning - (Params_Type => Inst_Params, - Object_Type => Inst_Object, - Hash => Hash, - Build => Build, - Equal => Equal); - - procedure Synth_Individual_Prefix (Syn_Inst : Synth_Instance_Acc; - Inter_Inst : Synth_Instance_Acc; - Formal : Node; - Off : out Uns32; - Typ : out Type_Acc) is - begin - case Get_Kind (Formal) is - when Iir_Kind_Interface_Signal_Declaration => - Off := 0; - Typ := Get_Subtype_Object (Inter_Inst, Get_Type (Formal)); - when Iir_Kind_Simple_Name => - Synth_Individual_Prefix - (Syn_Inst, Inter_Inst, Get_Named_Entity (Formal), Off, Typ); - when Iir_Kind_Selected_Element => - declare - Idx : constant Iir_Index32 := - Get_Element_Position (Get_Named_Entity (Formal)); - begin - Synth_Individual_Prefix - (Syn_Inst, Inter_Inst, Get_Prefix (Formal), Off, Typ); - Off := Off + Typ.Rec.E (Idx + 1).Boff; - Typ := Typ.Rec.E (Idx + 1).Typ; - end; - when Iir_Kind_Indexed_Name => - declare - Voff : Net; - Arr_Off : Value_Offsets; - begin - Synth_Individual_Prefix - (Syn_Inst, Inter_Inst, Get_Prefix (Formal), Off, Typ); - Synth_Indexed_Name (Syn_Inst, Formal, Typ, Voff, Arr_Off); - if Voff /= No_Net then - raise Internal_Error; - end if; - Off := Off + Arr_Off.Net_Off; - Typ := Get_Array_Element (Typ); - end; - when Iir_Kind_Slice_Name => - declare - Pfx_Bnd : Bound_Type; - El_Typ : Type_Acc; - Res_Bnd : Bound_Type; - Sl_Voff : Net; - Sl_Off : Value_Offsets; - begin - Synth_Individual_Prefix - (Syn_Inst, Inter_Inst, Get_Prefix (Formal), Off, Typ); - - Get_Onedimensional_Array_Bounds (Typ, Pfx_Bnd, El_Typ); - Synth_Slice_Suffix (Syn_Inst, Formal, Pfx_Bnd, El_Typ, - Res_Bnd, Sl_Voff, Sl_Off); - if Sl_Voff /= No_Net then - raise Internal_Error; - end if; - Off := Off + Sl_Off.Net_Off; - Typ := Create_Onedimensional_Array_Subtype (Typ, Res_Bnd); - end; - when others => - Vhdl.Errors.Error_Kind ("synth_individual_prefix", Formal); - end case; - end Synth_Individual_Prefix; - - type Value_Offset_Record is record - Off : Uns32; - Val : Valtyp; - end record; - - package Value_Offset_Tables is new Dyn_Tables - (Table_Component_Type => Value_Offset_Record, - Table_Index_Type => Natural, - Table_Low_Bound => 1); - - procedure Sort_Value_Offset (Els : Value_Offset_Tables.Instance) - is - function Lt (Op1, Op2 : Natural) return Boolean is - begin - return Els.Table (Op1).Off < Els.Table (Op2).Off; - end Lt; - - procedure Swap (From : Natural; To : Natural) - is - T : constant Value_Offset_Record := Els.Table (From); - begin - Els.Table (From) := Els.Table (To); - Els.Table (To) := T; - end Swap; - - procedure Heap_Sort is new Grt.Algos.Heap_Sort (Lt => Lt, Swap => Swap); - begin - Heap_Sort (Value_Offset_Tables.Last (Els)); - end Sort_Value_Offset; - - function Synth_Individual_Input_Assoc (Syn_Inst : Synth_Instance_Acc; - Assoc : Node; - Inter_Inst : Synth_Instance_Acc) - return Net - is - use Netlists.Concats; - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Iassoc : Node; - V : Valtyp; - Off : Uns32; - Typ : Type_Acc; - Els : Value_Offset_Tables.Instance; - Concat : Concat_Type; - N_Off : Uns32; - N : Net; - begin - Value_Offset_Tables.Init (Els, 16); - - Iassoc := Get_Chain (Assoc); - while Iassoc /= Null_Node - and then not Get_Whole_Association_Flag (Iassoc) - loop - -- For each individual assoc: - -- 1. compute type and offset - Synth_Individual_Prefix - (Syn_Inst, Inter_Inst, Get_Formal (Iassoc), Off, Typ); - - -- 2. synth expression - V := Synth_Expression_With_Type (Syn_Inst, Get_Actual (Iassoc), Typ); - - -- 3. save in a table - Value_Offset_Tables.Append (Els, (Off, V)); - - Iassoc := Get_Chain (Iassoc); - end loop; - - -- Then: - -- 1. sort table by offset - Sort_Value_Offset (Els); - - -- 2. concat - N_Off := 0; - for I in Value_Offset_Tables.First .. Value_Offset_Tables.Last (Els) - loop - pragma Assert (N_Off = Els.Table (I).Off); - V := Els.Table (I).Val; - N_Off := N_Off + V.Typ.W; - Append (Concat, Get_Net (Ctxt, V)); - end loop; - Value_Offset_Tables.Free (Els); - - -- 3. connect - Build (Ctxt, Concat, N); - return N; - end Synth_Individual_Input_Assoc; - - function Synth_Input_Assoc (Syn_Inst : Synth_Instance_Acc; - Assoc : Node; - Inter_Inst : Synth_Instance_Acc; - Inter : Node; - Inter_Typ : Type_Acc) - return Net - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Actual : Node; - Act_Inst : Synth_Instance_Acc; - Act : Valtyp; - begin - case Iir_Kinds_Association_Element_Parameters (Get_Kind (Assoc)) is - when Iir_Kind_Association_Element_Open => - Actual := Get_Default_Value (Inter); - Act_Inst := Inter_Inst; - when Iir_Kind_Association_Element_By_Expression => - Actual := Get_Actual (Assoc); - if Get_Kind (Actual) = Iir_Kind_Reference_Name then - -- Skip inserted anonymous signal declaration. - -- FIXME: simply do not insert it ? - Actual := Get_Named_Entity (Actual); - pragma Assert - (Get_Kind (Actual) = Iir_Kind_Anonymous_Signal_Declaration); - Actual := Get_Expression (Actual); - end if; - Act_Inst := Syn_Inst; - when Iir_Kind_Association_Element_By_Individual => - return Synth_Individual_Input_Assoc (Syn_Inst, Assoc, Inter_Inst); - end case; - - Act := Synth_Expression_With_Type (Act_Inst, Actual, Inter_Typ); - Act := Synth_Subtype_Conversion (Ctxt, Act, Inter_Typ, False, Assoc); - if Act = No_Valtyp then - return No_Net; - end if; - return Get_Net (Ctxt, Act); - end Synth_Input_Assoc; - - procedure Synth_Individual_Output_Assoc (Outp : Net; - Syn_Inst : Synth_Instance_Acc; - Assoc : Node; - Inter_Inst : Synth_Instance_Acc) - is - Iassoc : Node; - V : Valtyp; - Off : Uns32; - Typ : Type_Acc; - O : Net; - Port : Net; - begin - Port := Builders.Build_Port (Get_Build (Syn_Inst), Outp); - Set_Location (Port, Assoc); - - Iassoc := Get_Chain (Assoc); - while Iassoc /= Null_Node - and then not Get_Whole_Association_Flag (Iassoc) - loop - -- For each individual assoc: - -- 1. compute type and offset - Synth_Individual_Prefix - (Syn_Inst, Inter_Inst, Get_Formal (Iassoc), Off, Typ); - - -- 2. Extract the value. - O := Build_Extract (Get_Build (Syn_Inst), Port, Off, Typ.W); - V := Create_Value_Net (O, Typ); - - -- 3. Assign. - Synth_Assignment (Syn_Inst, Get_Actual (Iassoc), V, Iassoc); - - Iassoc := Get_Chain (Iassoc); - end loop; - end Synth_Individual_Output_Assoc; - - procedure Synth_Output_Assoc (Outp : Net; - Syn_Inst : Synth_Instance_Acc; - Assoc : Node; - Inter_Inst : Synth_Instance_Acc; - Inter : Node) - is - Actual : Node; - Formal_Typ : Type_Acc; - Port : Net; - O : Valtyp; - begin - case Get_Kind (Assoc) is - when Iir_Kind_Association_Element_Open => - -- Not connected. - return; - when Iir_Kind_Association_Element_By_Expression => - Actual := Get_Actual (Assoc); - when others => - Synth_Individual_Output_Assoc - (Outp, Syn_Inst, Assoc, Inter_Inst); - return; - end case; - - Formal_Typ := Get_Value (Inter_Inst, Inter).Typ; - - -- Create a port gate (so that is has a name). - Port := Builders.Build_Port (Get_Build (Syn_Inst), Outp); - Set_Location (Port, Assoc); - O := Create_Value_Net (Port, Formal_Typ); - -- Assign the port output to the actual (a net). - Synth_Assignment (Syn_Inst, Actual, O, Assoc); - end Synth_Output_Assoc; - - procedure Inst_Input_Connect (Syn_Inst : Synth_Instance_Acc; - Inst : Instance; - Port : in out Port_Idx; - Inter_Typ : Type_Acc; - N : Net) is - begin - case Inter_Typ.Kind is - when Type_Bit - | Type_Logic - | Type_Discrete - | Type_Float - | Type_Vector - | Type_Unbounded_Vector - | Type_Array - | Type_Unbounded_Array => - if N /= No_Net then - Connect (Get_Input (Inst, Port), N); - end if; - Port := Port + 1; - when Type_Record - | Type_Unbounded_Record => - for I in Inter_Typ.Rec.E'Range loop - if N /= No_Net then - Connect (Get_Input (Inst, Port), - Build_Extract (Get_Build (Syn_Inst), N, - Inter_Typ.Rec.E (I).Boff, - Inter_Typ.Rec.E (I).Typ.W)); - end if; - Port := Port + 1; - end loop; - when Type_Slice - | Type_Access - | Type_File - | Type_Protected => - raise Internal_Error; - end case; - end Inst_Input_Connect; - - procedure Inst_Output_Connect (Syn_Inst : Synth_Instance_Acc; - Inst : Instance; - Idx : in out Port_Idx; - Inter_Typ : Type_Acc; - N : out Net) is - begin - case Inter_Typ.Kind is - when Type_Bit - | Type_Logic - | Type_Discrete - | Type_Float - | Type_Vector - | Type_Unbounded_Vector - | Type_Array - | Type_Unbounded_Array => - N := Get_Output (Inst, Idx); - Idx := Idx + 1; - when Type_Record - | Type_Unbounded_Record => - declare - Nets : Net_Array (1 .. Nat32 (Inter_Typ.Rec.Len)); - begin - for I in Inter_Typ.Rec.E'Range loop - Nets (Nat32 (I)) := Get_Output (Inst, Idx); - Idx := Idx + 1; - end loop; - N := Folds.Build2_Concat (Get_Build (Syn_Inst), Nets); - end; - when Type_Slice - | Type_Access - | Type_File - | Type_Protected => - raise Internal_Error; - end case; - end Inst_Output_Connect; - - -- Subprogram used for instantiation (direct or by component). - -- PORTS_ASSOC belong to SYN_INST. - procedure Synth_Instantiate_Module (Syn_Inst : Synth_Instance_Acc; - Inst : Instance; - Inst_Obj : Inst_Object; - Ports_Assoc : Node) - is - -- Instantiate the module - -- Elaborate ports + map aspect for the inputs (component then entity) - -- Elaborate ports + map aspect for the outputs (entity then component) - - Assoc : Node; - Assoc_Inter : Node; - Inter : Node; - Inter_Typ : Type_Acc; - Nbr_Inputs : Port_Nbr; - Nbr_Outputs : Port_Nbr; - N : Net; - begin - Assoc := Ports_Assoc; - Assoc_Inter := Get_Port_Chain (Inst_Obj.Decl); - Nbr_Inputs := 0; - Nbr_Outputs := 0; - while Is_Valid (Assoc) loop - if Get_Whole_Association_Flag (Assoc) then - Inter := Get_Association_Interface (Assoc, Assoc_Inter); - Inter_Typ := Get_Subtype_Object - (Inst_Obj.Syn_Inst, Get_Type (Inter)); - - case Mode_To_Port_Kind (Get_Mode (Inter)) is - when Port_In => - -- Connect the net to the input. - N := Synth_Input_Assoc - (Syn_Inst, Assoc, Inst_Obj.Syn_Inst, Inter, Inter_Typ); - Inst_Input_Connect - (Syn_Inst, Inst, Nbr_Inputs, Inter_Typ, N); - when Port_Out - | Port_Inout => - Inst_Output_Connect - (Syn_Inst, Inst, Nbr_Outputs, Inter_Typ, N); - Synth_Output_Assoc - (N, Syn_Inst, Assoc, Inst_Obj.Syn_Inst, Inter); - end case; - end if; - Next_Association_Interface (Assoc, Assoc_Inter); - end loop; - - if Inst_Obj.Encoding = Name_Parameters then - -- Copy values of the generics to module parameters. - declare - Inter : Node; - Vt : Valtyp; - Pv : Pval; - Idx : Param_Idx; - begin - Idx := 0; - Inter := Get_Generic_Chain (Inst_Obj.Decl); - while Inter /= Null_Node loop - Vt := Get_Value (Inst_Obj.Syn_Inst, Inter); - if Vt /= No_Valtyp then - -- Avoid errors - Pv := Memtyp_To_Pval (Get_Memtyp (Vt)); - Set_Param_Pval (Inst, Idx, Pv); - end if; - Inter := Get_Chain (Inter); - Idx := Idx + 1; - end loop; - end; - end if; - end Synth_Instantiate_Module; - - function Synth_Port_Association_Type (Sub_Inst : Synth_Instance_Acc; - Syn_Inst : Synth_Instance_Acc; - Inter : Node; - Assoc : Node) return Type_Acc is - begin - if not Is_Fully_Constrained_Type (Get_Type (Inter)) then - -- TODO - -- Find the association for this interface - -- * if individual assoc: get type - -- * if whole assoc: get type from object. - if Assoc = Null_Node then - raise Internal_Error; - end if; - case Get_Kind (Assoc) is - when Iir_Kind_Association_Element_By_Expression => - return Synth_Type_Of_Object (Syn_Inst, Get_Actual (Assoc)); - when others => - raise Internal_Error; - end case; - else - Synth_Declaration_Type (Sub_Inst, Inter); - return Get_Subtype_Object (Sub_Inst, Get_Type (Inter)); - end if; - end Synth_Port_Association_Type; - - procedure Synth_Ports_Association_Type (Sub_Inst : Synth_Instance_Acc; - Syn_Inst : Synth_Instance_Acc; - Inter_Chain : Node; - Assoc_Chain : Node) - is - Inter : Node; - Assoc : Node; - Assoc_Inter : Node; - Val : Valtyp; - Inter_Typ : Type_Acc; - begin - Assoc := Assoc_Chain; - Assoc_Inter := Inter_Chain; - while Is_Valid (Assoc) loop - Inter := Get_Association_Interface (Assoc, Assoc_Inter); - if Get_Whole_Association_Flag (Assoc) then - Inter_Typ := Synth_Port_Association_Type - (Sub_Inst, Syn_Inst, Inter, Assoc); - case Mode_To_Port_Kind (Get_Mode (Inter)) is - when Port_In => - Val := Create_Value_Net (No_Net, Inter_Typ); - when Port_Out - | Port_Inout => - Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); - end case; - Create_Object (Sub_Inst, Inter, Val); - end if; - Next_Association_Interface (Assoc, Assoc_Inter); - end loop; - end Synth_Ports_Association_Type; - - procedure Synth_Direct_Instantiation_Statement - (Syn_Inst : Synth_Instance_Acc; - Stmt : Node; - Ent : Node; - Arch : Node; - Config : Node) - is - Sub_Inst : Synth_Instance_Acc; - Inst_Obj : Inst_Object; - Inst : Instance; - Enc : Name_Encoding; - begin - -- Elaborate generic + map aspect - Sub_Inst := Make_Instance - (Syn_Inst, Ent, New_Sname_User (Get_Identifier (Ent), No_Sname)); - - Synth_Generics_Association (Sub_Inst, Syn_Inst, - Get_Generic_Chain (Ent), - Get_Generic_Map_Aspect_Chain (Stmt)); - - -- Elaborate port types. - Synth_Ports_Association_Type (Sub_Inst, Syn_Inst, - Get_Port_Chain (Ent), - Get_Port_Map_Aspect_Chain (Stmt)); - - if Is_Error (Sub_Inst) then - -- TODO: Free it? - return; - end if; - - if Arch /= Null_Node then - -- For whiteboxes: append parameters or/and hash. - Enc := Name_Hash; - else - -- For blackboxes: define the parameters. - Enc := Name_Parameters; - end if; - - -- Search if corresponding module has already been used. - -- If not create a new module - -- * create a name from the generics and the library - -- * create inputs/outputs - -- * add it to the list of module to be synthesized. - Inst_Obj := Insts_Interning.Get ((Decl => Ent, - Arch => Arch, - Config => Config, - Syn_Inst => Sub_Inst, - Encoding => Enc)); - - -- TODO: free sub_inst. - - Inst := New_Instance - (Get_Instance_Module (Syn_Inst), - Inst_Obj.M, - New_Sname_User (Get_Identifier (Stmt), Get_Sname (Syn_Inst))); - Set_Location (Inst, Stmt); - - Push_Phi; - - Synth_Instantiate_Module - (Syn_Inst, Inst, Inst_Obj, Get_Port_Map_Aspect_Chain (Stmt)); - - Pop_And_Merge_Phi (Get_Build (Syn_Inst), Get_Location (Stmt)); - end Synth_Direct_Instantiation_Statement; - - procedure Synth_Design_Instantiation_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Aspect : constant Iir := Get_Instantiated_Unit (Stmt); - Arch : Node; - Ent : Node; - Config : Node; - begin - -- Load configured entity + architecture - case Iir_Kinds_Entity_Aspect (Get_Kind (Aspect)) is - when Iir_Kind_Entity_Aspect_Entity => - Arch := Get_Architecture (Aspect); - if Arch = Null_Node then - Arch := Libraries.Get_Latest_Architecture (Get_Entity (Aspect)); - else - Arch := Strip_Denoting_Name (Arch); - end if; - Config := Get_Library_Unit - (Get_Default_Configuration_Declaration (Arch)); - when Iir_Kind_Entity_Aspect_Configuration => - Config := Get_Configuration (Aspect); - Arch := Get_Block_Specification (Get_Block_Configuration (Config)); - when Iir_Kind_Entity_Aspect_Open => - return; - end case; - Config := Get_Block_Configuration (Config); - Ent := Get_Entity (Arch); - - Synth_Direct_Instantiation_Statement - (Syn_Inst, Stmt, Ent, Arch, Config); - end Synth_Design_Instantiation_Statement; - - procedure Synth_Blackbox_Instantiation_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Comp : constant Node := - Get_Named_Entity (Get_Instantiated_Unit (Stmt)); - begin - Synth_Direct_Instantiation_Statement - (Syn_Inst, Stmt, Comp, Null_Node, Null_Node); - end Synth_Blackbox_Instantiation_Statement; - - procedure Create_Component_Wire (Ctxt : Context_Acc; - Inter : Node; - Val : Valtyp; - Pfx_Name : Sname; - Loc : Source.Syn_Src) - is - Value : Net; - W : Width; - begin - case Val.Val.Kind is - when Value_Wire => - -- Create a gate for the output, so that it could be read. - Val.Val.W := Alloc_Wire (Wire_Output, (Inter, Bit_Type)); - W := Get_Type_Width (Val.Typ); - Value := Build_Signal - (Ctxt, New_Internal_Name (Ctxt, Pfx_Name), W); - Set_Location (Value, Loc); - Set_Wire_Gate (Val.Val.W, Value); - when others => - raise Internal_Error; - end case; - end Create_Component_Wire; - - procedure Synth_Component_Instantiation_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Component : constant Node := - Get_Named_Entity (Get_Instantiated_Unit (Stmt)); - Config : constant Node := Get_Component_Configuration (Stmt); - Bind : constant Node := Get_Binding_Indication (Config); - Aspect : constant Node := Get_Entity_Aspect (Bind); - Comp_Inst : Synth_Instance_Acc; - - Ent : Node; - Arch : Node; - Sub_Config : Node; - Sub_Inst : Synth_Instance_Acc; - Inst_Obj : Inst_Object; - Inst : Instance; - Inst_Name : Sname; - begin - pragma Assert (Get_Component_Configuration (Stmt) /= Null_Node); - pragma Assert (Get_Kind (Aspect) = Iir_Kind_Entity_Aspect_Entity); - - Push_Phi; - - Inst_Name := New_Sname_User (Get_Identifier (Stmt), - Get_Sname (Syn_Inst)); - - -- Create the sub-instance for the component - -- Elaborate generic + map aspect - Comp_Inst := Make_Instance - (Syn_Inst, Component, - New_Sname_User (Get_Identifier (Component), No_Sname)); - - Synth_Generics_Association (Comp_Inst, Syn_Inst, - Get_Generic_Chain (Component), - Get_Generic_Map_Aspect_Chain (Stmt)); - - -- Create objects for the inputs and the outputs of the component, - -- assign inputs (that's nets) and create wires for outputs. - declare - Assoc : Node; - Assoc_Inter : Node; - Inter : Node; - Inter_Typ : Type_Acc; - Val : Valtyp; - N : Net; - begin - Assoc := Get_Port_Map_Aspect_Chain (Stmt); - Assoc_Inter := Get_Port_Chain (Component); - while Is_Valid (Assoc) loop - if Get_Whole_Association_Flag (Assoc) then - Inter := Get_Association_Interface (Assoc, Assoc_Inter); - - Inter_Typ := Synth_Port_Association_Type - (Comp_Inst, Syn_Inst, Inter, Assoc); - - case Mode_To_Port_Kind (Get_Mode (Inter)) is - when Port_In => - N := Synth_Input_Assoc - (Syn_Inst, Assoc, Comp_Inst, Inter, Inter_Typ); - Val := Create_Value_Net (N, Inter_Typ); - when Port_Out - | Port_Inout => - Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); - Create_Component_Wire - (Get_Build (Syn_Inst), Assoc_Inter, Val, Inst_Name, - Assoc); - end case; - Create_Object (Comp_Inst, Assoc_Inter, Val); - end if; - Next_Association_Interface (Assoc, Assoc_Inter); - end loop; - end; - - -- Extract entity/architecture instantiated by the component. - case Get_Kind (Aspect) is - when Iir_Kind_Entity_Aspect_Entity => - Ent := Get_Entity (Aspect); - Arch := Get_Architecture (Aspect); - when others => - Vhdl.Errors.Error_Kind - ("Synth_Component_Instantiation_Statement(2)", Aspect); - end case; - - if Get_Kind (Ent) = Iir_Kind_Foreign_Module then - -- TODO. - raise Internal_Error; - end if; - - if Arch = Null_Node then - Arch := Libraries.Get_Latest_Architecture (Ent); - else - Arch := Get_Named_Entity (Arch); - end if; - Sub_Config := Get_Library_Unit - (Get_Default_Configuration_Declaration (Arch)); - Sub_Config := Get_Block_Configuration (Sub_Config); - - -- Elaborate generic + map aspect for the entity instance. - Sub_Inst := Make_Instance - (Comp_Inst, Ent, New_Sname_User (Get_Identifier (Ent), No_Sname)); - Synth_Generics_Association (Sub_Inst, Comp_Inst, - Get_Generic_Chain (Ent), - Get_Generic_Map_Aspect_Chain (Bind)); - - Synth_Ports_Association_Type (Sub_Inst, Comp_Inst, - Get_Port_Chain (Ent), - Get_Port_Map_Aspect_Chain (Bind)); - - -- Search if corresponding module has already been used. - -- If not create a new module - -- * create a name from the generics and the library - -- * create inputs/outputs - -- * add it to the list of module to be synthesized. - Inst_Obj := Insts_Interning.Get ((Decl => Ent, - Arch => Arch, - Config => Sub_Config, - Syn_Inst => Sub_Inst, - Encoding => Name_Hash)); - - -- TODO: free sub_inst. - - Inst := New_Instance (Get_Instance_Module (Syn_Inst), - Inst_Obj.M, Inst_Name); - Set_Location (Inst, Stmt); - - Synth_Instantiate_Module - (Comp_Inst, Inst, Inst_Obj, Get_Port_Map_Aspect_Chain (Bind)); - - -- Connect out from component to instance. - -- Instantiate the module - -- Elaborate ports + map aspect for the inputs (component then entity) - -- Elaborate ports + map aspect for the outputs (entity then component) - declare - Assoc : Node; - Assoc_Inter : Node; - Inter : Node; - Port : Net; - O : Valtyp; - Nbr_Outputs : Port_Nbr; - begin - Assoc := Get_Port_Map_Aspect_Chain (Stmt); - Assoc_Inter := Get_Port_Chain (Component); - Nbr_Outputs := 0; - while Is_Valid (Assoc) loop - if Get_Whole_Association_Flag (Assoc) then - Inter := Get_Association_Interface (Assoc, Assoc_Inter); - - if Mode_To_Port_Kind (Get_Mode (Inter)) = Port_Out then - O := Get_Value (Comp_Inst, Inter); - Port := Get_Net (Ctxt, O); - Synth_Output_Assoc (Port, Syn_Inst, Assoc, Comp_Inst, Inter); - Nbr_Outputs := Nbr_Outputs + 1; - end if; - end if; - Next_Association_Interface (Assoc, Assoc_Inter); - end loop; - end; - - Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); - - Finalize_Declarations (Comp_Inst, Get_Port_Chain (Component)); - end Synth_Component_Instantiation_Statement; - - procedure Synth_Dependencies (Parent_Inst : Synth_Instance_Acc; Unit : Node) - is - Dep_List : constant Node_List := Get_Dependence_List (Unit); - Dep_It : List_Iterator; - Dep : Node; - Dep_Unit : Node; - begin - Dep_It := List_Iterate (Dep_List); - while Is_Valid (Dep_It) loop - Dep := Get_Element (Dep_It); - if Get_Kind (Dep) = Iir_Kind_Design_Unit - and then not Get_Elab_Flag (Dep) - then - Set_Elab_Flag (Dep, True); - Synth_Dependencies (Parent_Inst, Dep); - Dep_Unit := Get_Library_Unit (Dep); - case Iir_Kinds_Library_Unit (Get_Kind (Dep_Unit)) is - when Iir_Kind_Entity_Declaration => - null; - when Iir_Kind_Configuration_Declaration => - null; - when Iir_Kind_Context_Declaration => - null; - when Iir_Kind_Package_Declaration => - declare - Bod : constant Node := Get_Package_Body (Dep_Unit); - Bod_Unit : Node; - begin - Synth_Package_Declaration (Parent_Inst, Dep_Unit); - -- Do not try to elaborate math_real body: there are - -- functions with loop. Currently, try create signals, - -- which is not possible during package elaboration. - if Bod /= Null_Node - and then Dep_Unit /= Vhdl.Ieee.Math_Real.Math_Real_Pkg - then - Bod_Unit := Get_Design_Unit (Bod); - Synth_Dependencies (Parent_Inst, Bod_Unit); - Synth_Package_Body (Parent_Inst, Dep_Unit, Bod); - end if; - end; - when Iir_Kind_Package_Instantiation_Declaration => - Synth_Package_Instantiation (Parent_Inst, Dep_Unit); - when Iir_Kind_Package_Body => - null; - when Iir_Kind_Architecture_Body => - null; - when Iir_Kinds_Verification_Unit => - null; - end case; - end if; - Next (Dep_It); - end loop; - end Synth_Dependencies; - - procedure Synth_Top_Entity (Global_Instance : Synth_Instance_Acc; - Arch : Node; - Config : Node; - Encoding : Name_Encoding; - Inst : out Synth_Instance_Acc) - is - Entity : constant Node := Get_Entity (Arch); - Syn_Inst : Synth_Instance_Acc; - Inter : Node; - Inter_Typ : Type_Acc; - Inst_Obj : Inst_Object; - Val : Valtyp; - begin - Root_Instance := Global_Instance; - - Insts_Interning.Init; - - if Flags.Flag_Debug_Init then - Synth.Debugger.Debug_Init (Arch); - end if; - - -- Dependencies first. - Synth_Dependencies (Global_Instance, Get_Design_Unit (Entity)); - Synth_Dependencies (Global_Instance, Get_Design_Unit (Arch)); - - Syn_Inst := Make_Instance - (Global_Instance, Arch, - New_Sname_User (Get_Identifier (Entity), No_Sname)); - - -- Compute generics. - Inter := Get_Generic_Chain (Entity); - while Is_Valid (Inter) loop - Synth_Declaration_Type (Syn_Inst, Inter); - declare - Val : Valtyp; - Inter_Typ : Type_Acc; - begin - Inter_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Inter)); - Val := Synth_Expression_With_Type - (Syn_Inst, Get_Default_Value (Inter), Inter_Typ); - pragma Assert (Is_Static (Val.Val)); - Create_Object (Syn_Inst, Inter, Val); - end; - Inter := Get_Chain (Inter); - end loop; - - -- Elaborate port types. - -- FIXME: what about unconstrained ports ? Get the type from the - -- association. - Inter := Get_Port_Chain (Entity); - while Is_Valid (Inter) loop - if not Is_Fully_Constrained_Type (Get_Type (Inter)) then - -- TODO - raise Internal_Error; - end if; - Synth_Declaration_Type (Syn_Inst, Inter); - Inter_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Inter)); - case Mode_To_Port_Kind (Get_Mode (Inter)) is - when Port_In => - Val := Create_Value_Net (No_Net, Inter_Typ); - when Port_Out - | Port_Inout => - Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); - end case; - Create_Object (Syn_Inst, Inter, Val); - Inter := Get_Chain (Inter); - end loop; - - -- Search if corresponding module has already been used. - -- If not create a new module - -- * create a name from the generics and the library - -- * create inputs/outputs - -- * add it to the list of module to be synthesized. - Inst_Obj := Insts_Interning.Get - ((Decl => Entity, - Arch => Arch, - Config => Get_Block_Configuration (Config), - Syn_Inst => Syn_Inst, - Encoding => Encoding)); - Inst := Inst_Obj.Syn_Inst; - end Synth_Top_Entity; - - procedure Create_Input_Wire (Syn_Inst : Synth_Instance_Acc; - Self_Inst : Instance; - Idx : in out Port_Idx; - Val : Valtyp) is - begin - pragma Assert (Val.Val.Kind = Value_Net); - Inst_Output_Connect (Syn_Inst, Self_Inst, Idx, Val.Typ, Val.Val.N); - end Create_Input_Wire; - - procedure Create_Output_Wire (Syn_Inst : Synth_Instance_Acc; - Self_Inst : Instance; - Inter : Node; - Idx : in out Port_Idx; - Val : Valtyp) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Default : constant Node := Get_Default_Value (Inter); - Desc : constant Port_Desc := - Get_Output_Desc (Get_Module (Self_Inst), Idx); - Inter_Typ : Type_Acc; - Value : Net; - Vout : Net; - Init : Valtyp; - Init_Net : Net; - begin - pragma Assert (Val.Val.Kind = Value_Wire); - - -- Create a gate for the output, so that it could be read. - Val.Val.W := Alloc_Wire (Wire_Output, (Inter, Val.Typ)); - -- pragma Assert (Desc.W = Get_Type_Width (Val.Typ)); - - if Default /= Null_Node then - Inter_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Inter)); - Init := Synth_Expression_With_Type (Syn_Inst, Default, Inter_Typ); - Init := Synth_Subtype_Conversion - (Ctxt, Init, Inter_Typ, False, Inter); - Init_Net := Get_Net (Ctxt, Init); - else - Init_Net := No_Net; - end if; - - if Desc.Is_Inout then - declare - Io_Inst : Instance; - begin - if Init_Net /= No_Net then - Io_Inst := Builders.Build_Iinout (Ctxt, Val.Typ.W); - Connect (Get_Input (Io_Inst, 1), Init_Net); - else - Io_Inst := Builders.Build_Inout (Ctxt, Val.Typ.W); - end if; - -- Connect port1 of gate inout to the pin. - Vout := Get_Output (Io_Inst, 1); - -- And port0 of the gate will be use to read from the pin. - Value := Get_Output (Io_Inst, 0); - end; - else - if Init_Net /= No_Net then - Value := Builders.Build_Ioutput (Ctxt, Init_Net); - else - Value := Builders.Build_Output (Ctxt, Val.Typ.W); - end if; - Vout := Value; - end if; - Set_Location (Value, Inter); - Set_Wire_Gate (Val.Val.W, Value); - - Inst_Input_Connect (Syn_Inst, Self_Inst, Idx, Val.Typ, Vout); - end Create_Output_Wire; - - procedure Apply_Block_Configuration (Cfg : Node; Blk : Node) - is - Item : Node; - begin - -- Be sure CFG applies to BLK. - pragma Assert (Get_Block_From_Block_Specification - (Get_Block_Specification (Cfg)) = Blk); - - -- Clear_Instantiation_Configuration (Blk); - - Item := Get_Configuration_Item_Chain (Cfg); - while Item /= Null_Node loop - case Get_Kind (Item) is - when Iir_Kind_Component_Configuration => - declare - List : constant Iir_Flist := - Get_Instantiation_List (Item); - El : Node; - Inst : Node; - begin - for I in Flist_First .. Flist_Last (List) loop - El := Get_Nth_Element (List, I); - Inst := Get_Named_Entity (El); - pragma Assert - (Get_Kind (Inst) - = Iir_Kind_Component_Instantiation_Statement); - pragma Assert - (Get_Component_Configuration (Inst) = Null_Node); - Set_Component_Configuration (Inst, Item); - end loop; - end; - when Iir_Kind_Block_Configuration => - declare - Sub_Blk : constant Node := Get_Block_From_Block_Specification - (Get_Block_Specification (Item)); - begin - case Get_Kind (Sub_Blk) is - when Iir_Kind_Generate_Statement_Body => - -- Linked chain. - Set_Prev_Block_Configuration - (Item, Get_Generate_Block_Configuration (Sub_Blk)); - Set_Generate_Block_Configuration (Sub_Blk, Item); - when Iir_Kind_Block_Statement => - Set_Block_Block_Configuration (Sub_Blk, Item); - when others => - Vhdl.Errors.Error_Kind - ("apply_block_configuration(blk)", Sub_Blk); - end case; - end; - when others => - Vhdl.Errors.Error_Kind ("apply_block_configuration", Item); - end case; - Item := Get_Chain (Item); - end loop; - end Apply_Block_Configuration; - - procedure Synth_Verification_Units - (Syn_Inst : Synth_Instance_Acc; Parent : Node) - is - Unit : Node; - begin - Unit := Get_Bound_Vunit_Chain (Parent); - while Unit /= Null_Node loop - Synth_Verification_Unit (Syn_Inst, Unit); - Unit := Get_Bound_Vunit_Chain (Unit); - end loop; - end Synth_Verification_Units; - - procedure Synth_Instance (Inst : Inst_Object) - is - Entity : constant Node := Inst.Decl; - Arch : constant Node := Inst.Arch; - Syn_Inst : constant Synth_Instance_Acc := Inst.Syn_Inst; - Self_Inst : Instance; - Inter : Node; - Vt : Valtyp; - Nbr_Inputs : Port_Nbr; - Nbr_Outputs : Port_Nbr; - begin - if Arch = Null_Node then - -- Black box. - return; - end if; - - if Flag_Verbose then - Errors.Info_Msg_Synth (+Entity, "synthesizing %n", (1 => +Entity)); - end if; - - -- Save the current architecture, so that files can be open using a - -- path relative to the architecture filename. - Synth.Vhdl_Files.Set_Design_Unit (Arch); - - Synth_Dependencies (Root_Instance, Get_Design_Unit (Arch)); - - Set_Instance_Module (Syn_Inst, Inst.M); - Self_Inst := Get_Self_Instance (Inst.M); - Set_Location (Self_Inst, Entity); - - -- Create wires for inputs and outputs. - Inter := Get_Port_Chain (Entity); - Nbr_Inputs := 0; - Nbr_Outputs := 0; - while Is_Valid (Inter) loop - Vt := Get_Value (Syn_Inst, Inter); - case Mode_To_Port_Kind (Get_Mode (Inter)) is - when Port_In => - Create_Input_Wire (Syn_Inst, Self_Inst, Nbr_Inputs, Vt); - when Port_Out - | Port_Inout => - Create_Output_Wire - (Syn_Inst, Self_Inst, Inter, Nbr_Outputs, Vt); - end case; - Inter := Get_Chain (Inter); - end loop; - - -- Apply configuration. - -- FIXME: what about inner block configuration ? - pragma Assert (Get_Kind (Inst.Config) = Iir_Kind_Block_Configuration); - Apply_Block_Configuration (Inst.Config, Arch); - - Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Entity)); - if not Is_Error (Syn_Inst) then - Synth_Concurrent_Statements - (Syn_Inst, Get_Concurrent_Statement_Chain (Entity)); - end if; - - if not Is_Error (Syn_Inst) then - Synth_Attribute_Values (Syn_Inst, Entity); - end if; - - if not Is_Error (Syn_Inst) then - Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Arch)); - end if; - if not Is_Error (Syn_Inst) then - Synth_Concurrent_Statements - (Syn_Inst, Get_Concurrent_Statement_Chain (Arch)); - end if; - - if not Is_Error (Syn_Inst) then - Synth_Attribute_Values (Syn_Inst, Arch); - end if; - - if not Is_Error (Syn_Inst) then - Synth_Verification_Units (Syn_Inst, Entity); - end if; - if not Is_Error (Syn_Inst) then - Synth_Verification_Units (Syn_Inst, Arch); - end if; - - Finalize_Declarations (Syn_Inst, Get_Declaration_Chain (Arch)); - Finalize_Declarations (Syn_Inst, Get_Port_Chain (Entity)); - - Finalize_Wires; - - -- Remove unused gates. This is not only an optimization but also - -- a correctness point: there might be some unsynthesizable gates, like - -- the one created for 'rising_egde (clk) and not rst'. - if not Synth.Flags.Flag_Debug_Nocleanup then - -- Netlists.Cleanup.Remove_Unconnected_Instances (Inst.M); - Netlists.Cleanup.Mark_And_Sweep (Inst.M); - Netlists.Cleanup.Remove_Output_Gates (Inst.M); - end if; - - if not Synth.Flags.Flag_Debug_Nomemory2 then - Netlists.Memories.Extract_Memories2 (Get_Build (Syn_Inst), Inst.M); - -- Remove remaining clock edge gates. - Netlists.Cleanup.Mark_And_Sweep (Inst.M); - end if; - - if not Synth.Flags.Flag_Debug_Noexpand then - Netlists.Expands.Expand_Gates (Get_Build (Syn_Inst), Inst.M); - end if; - end Synth_Instance; - - procedure Synth_All_Instances - is - use Insts_Interning; - Idx : Index_Type; - begin - Idx := First_Index; - while Idx <= Last_Index loop - Synth_Instance (Get_By_Index (Idx)); - Idx := Idx + 1; - end loop; - end Synth_All_Instances; -end Synth.Insts; diff --git a/src/synth/synth-insts.ads b/src/synth/synth-insts.ads deleted file mode 100644 index f0ac690e6..000000000 --- a/src/synth/synth-insts.ads +++ /dev/null @@ -1,47 +0,0 @@ --- Instantiation synthesis. --- Copyright (C) 2019 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with Vhdl.Nodes; use Vhdl.Nodes; - -with Synth.Vhdl_Context; use Synth.Vhdl_Context; -with Synth.Flags; use Synth.Flags; - -package Synth.Insts is - -- Create the declaration of the top entity. - procedure Synth_Top_Entity (Global_Instance : Synth_Instance_Acc; - Arch : Node; - Config : Node; - Encoding : Name_Encoding; - Inst : out Synth_Instance_Acc); - - -- Synthesize the top entity and all the sub-modules. - procedure Synth_All_Instances; - - -- Apply block configuration CFG to BLK. - -- Must be done before synthesis of BLK. - -- The synthesis of BLK will clear all configuration of it. - procedure Apply_Block_Configuration (Cfg : Node; Blk : Node); - - procedure Synth_Design_Instantiation_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node); - procedure Synth_Blackbox_Instantiation_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node); - - procedure Synth_Component_Instantiation_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node); -end Synth.Insts; diff --git a/src/synth/synth-static_oper.adb b/src/synth/synth-static_oper.adb index 5c9db02e9..2b9b5ffab 100644 --- a/src/synth/synth-static_oper.adb +++ b/src/synth/synth-static_oper.adb @@ -29,7 +29,7 @@ with Netlists; use Netlists; with Synth.Memtype; use Synth.Memtype; with Synth.Errors; use Synth.Errors; with Synth.Source; use Synth.Source; -with Synth.Expr; use Synth.Expr; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; with Synth.Vhdl_Oper; with Synth.Ieee.Std_Logic_1164; use Synth.Ieee.Std_Logic_1164; with Synth.Ieee.Numeric_Std; use Synth.Ieee.Numeric_Std; diff --git a/src/synth/synth-stmts.adb b/src/synth/synth-stmts.adb deleted file mode 100644 index 8f33e3421..000000000 --- a/src/synth/synth-stmts.adb +++ /dev/null @@ -1,3853 +0,0 @@ --- Statements synthesis. --- Copyright (C) 2017 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with Ada.Unchecked_Deallocation; - -with Grt.Types; use Grt.Types; -with Grt.Algos; -with Grt.Severity; use Grt.Severity; -with Areapools; -with Name_Table; -with Std_Names; -with Errorout; use Errorout; -with Files_Map; -with Simple_IO; - -with Vhdl.Errors; use Vhdl.Errors; -with Vhdl.Sem_Expr; -with Vhdl.Sem_Inst; -with Vhdl.Utils; use Vhdl.Utils; -with Vhdl.Std_Package; -with Vhdl.Evaluation; -with Vhdl.Ieee.Std_Logic_1164; - -with PSL.Types; -with PSL.NFAs; - -with Synth.Memtype; use Synth.Memtype; -with Synth.Errors; use Synth.Errors; -with Synth.Decls; use Synth.Decls; -with Synth.Expr; use Synth.Expr; -with Synth.Insts; use Synth.Insts; -with Synth.Source; -with Synth.Vhdl_Static_Proc; -with Synth.Vhdl_Heap; -with Synth.Flags; -with Synth.Debugger; - -with Netlists.Builders; use Netlists.Builders; -with Netlists.Folds; use Netlists.Folds; -with Netlists.Gates; use Netlists.Gates; -with Netlists.Utils; use Netlists.Utils; -with Netlists.Locations; use Netlists.Locations; - -package body Synth.Stmts is - procedure Synth_Sequential_Statements - (C : in out Seq_Context; Stmts : Node); - - procedure Set_Location (N : Net; Loc : Node) - renames Synth.Source.Set_Location; - - function Synth_Waveform (Syn_Inst : Synth_Instance_Acc; - Wf : Node; - Targ_Type : Type_Acc) return Valtyp - is - Res : Valtyp; - begin - if Get_Kind (Wf) = Iir_Kind_Unaffected_Waveform then - -- TODO - raise Internal_Error; - end if; - if Get_Chain (Wf) /= Null_Node then - -- Warning. - null; - end if; - if Get_Time (Wf) /= Null_Node then - -- Warning - null; - end if; - if Targ_Type = null then - return Synth_Expression (Syn_Inst, Get_We_Value (Wf)); - else - Res := Synth_Expression_With_Type - (Syn_Inst, Get_We_Value (Wf), Targ_Type); - Res := Synth_Subtype_Conversion - (Get_Build (Syn_Inst), Res, Targ_Type, False, Wf); - return Res; - end if; - end Synth_Waveform; - - procedure Synth_Assignment_Prefix (Syn_Inst : Synth_Instance_Acc; - Pfx : Node; - Dest_Base : out Valtyp; - Dest_Typ : out Type_Acc; - Dest_Off : out Value_Offsets; - Dest_Dyn : out Dyn_Name) is - begin - case Get_Kind (Pfx) is - when Iir_Kind_Simple_Name => - Synth_Assignment_Prefix (Syn_Inst, Get_Named_Entity (Pfx), - Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); - when Iir_Kind_Interface_Signal_Declaration - | Iir_Kind_Variable_Declaration - | Iir_Kind_Interface_Variable_Declaration - | Iir_Kind_Signal_Declaration - | Iir_Kind_Anonymous_Signal_Declaration - | Iir_Kind_Interface_Constant_Declaration - | Iir_Kind_Constant_Declaration - | Iir_Kind_File_Declaration - | Iir_Kind_Interface_File_Declaration - | Iir_Kind_Object_Alias_Declaration => - declare - Targ : constant Valtyp := Get_Value (Syn_Inst, Pfx); - begin - Dest_Dyn := No_Dyn_Name; - Dest_Typ := Targ.Typ; - - if Targ.Val.Kind = Value_Alias then - -- Replace alias by the aliased name. - Dest_Base := (Targ.Val.A_Typ, Targ.Val.A_Obj); - Dest_Off := Targ.Val.A_Off; - else - Dest_Base := Targ; - Dest_Off := (0, 0); - end if; - end; - when Iir_Kind_Function_Call => - Dest_Base := Synth_Expression (Syn_Inst, Pfx); - Dest_Typ := Dest_Base.Typ; - Dest_Off := (0, 0); - Dest_Dyn := No_Dyn_Name; - - when Iir_Kind_Indexed_Name => - declare - Voff : Net; - Off : Value_Offsets; - begin - Synth_Assignment_Prefix - (Syn_Inst, Get_Prefix (Pfx), - Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); - Strip_Const (Dest_Base); - Synth_Indexed_Name (Syn_Inst, Pfx, Dest_Typ, Voff, Off); - - if Voff = No_Net then - -- Static index. - Dest_Off := Dest_Off + Off; - else - -- Dynamic index. - if Dest_Dyn.Voff = No_Net then - -- The first one. - Dest_Dyn := (Pfx_Off => Dest_Off, - Pfx_Typ => Dest_Typ, - Voff => Voff); - Dest_Off := Off; - else - -- Nested one. - -- FIXME - Dest_Off := Dest_Off + Off; - -- if Dest_Off /= (0, 0) then - -- Error_Msg_Synth (+Pfx, "nested memory not supported"); - -- end if; - - Dest_Dyn.Voff := Build_Addidx - (Get_Build (Syn_Inst), Dest_Dyn.Voff, Voff); - end if; - end if; - - Dest_Typ := Get_Array_Element (Dest_Typ); - end; - - when Iir_Kind_Selected_Element => - declare - Idx : constant Iir_Index32 := - Get_Element_Position (Get_Named_Entity (Pfx)); - begin - Synth_Assignment_Prefix - (Syn_Inst, Get_Prefix (Pfx), - Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); - Dest_Off.Net_Off := - Dest_Off.Net_Off + Dest_Typ.Rec.E (Idx + 1).Boff; - Dest_Off.Mem_Off := - Dest_Off.Mem_Off + Dest_Typ.Rec.E (Idx + 1).Moff; - - Dest_Typ := Dest_Typ.Rec.E (Idx + 1).Typ; - end; - - when Iir_Kind_Slice_Name => - declare - Pfx_Bnd : Bound_Type; - El_Typ : Type_Acc; - Res_Bnd : Bound_Type; - Sl_Voff : Net; - Sl_Off : Value_Offsets; - begin - Synth_Assignment_Prefix - (Syn_Inst, Get_Prefix (Pfx), - Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); - Strip_Const (Dest_Base); - - Get_Onedimensional_Array_Bounds (Dest_Typ, Pfx_Bnd, El_Typ); - Synth_Slice_Suffix (Syn_Inst, Pfx, Pfx_Bnd, El_Typ, - Res_Bnd, Sl_Voff, Sl_Off); - - - if Sl_Voff = No_Net then - -- Fixed slice. - Dest_Typ := Create_Onedimensional_Array_Subtype - (Dest_Typ, Res_Bnd); - Dest_Off.Net_Off := Dest_Off.Net_Off + Sl_Off.Net_Off; - Dest_Off.Mem_Off := Dest_Off.Mem_Off + Sl_Off.Mem_Off; - else - -- Variable slice. - if Dest_Dyn.Voff = No_Net then - -- First one. - Dest_Dyn := (Pfx_Off => Dest_Off, - Pfx_Typ => Dest_Typ, - Voff => Sl_Voff); - Dest_Off := Sl_Off; - else - -- Nested. - if Dest_Off /= (0, 0) then - Error_Msg_Synth (+Pfx, "nested memory not supported"); - end if; - - Dest_Dyn.Voff := Build_Addidx - (Get_Build (Syn_Inst), Dest_Dyn.Voff, Sl_Voff); - end if; - Dest_Typ := Create_Slice_Type (Res_Bnd.Len, El_Typ); - end if; - end; - - when Iir_Kind_Implicit_Dereference - | Iir_Kind_Dereference => - Synth_Assignment_Prefix - (Syn_Inst, Get_Prefix (Pfx), - Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); - if Dest_Off /= (0, 0) and then Dest_Dyn.Voff /= No_Net then - raise Internal_Error; - end if; - Dest_Base := Vhdl_Heap.Synth_Dereference (Read_Access (Dest_Base)); - Dest_Typ := Dest_Base.Typ; - - when others => - Error_Kind ("synth_assignment_prefix", Pfx); - end case; - end Synth_Assignment_Prefix; - - type Target_Kind is - ( - -- The target is an object or a static part of it. - Target_Simple, - - -- The target is an aggregate. - Target_Aggregate, - - -- The assignment is dynamically indexed. - Target_Memory - ); - - type Target_Info (Kind : Target_Kind := Target_Simple) is record - -- In all cases, the type of the target is known or computed. - Targ_Type : Type_Acc; - - case Kind is - when Target_Simple => - -- For a simple target, the destination is known. - Obj : Valtyp; - Off : Value_Offsets; - when Target_Aggregate => - -- For an aggregate: the type is computed and the details will - -- be handled at the assignment. - Aggr : Node; - when Target_Memory => - -- For a memory: the destination is known. - Mem_Obj : Valtyp; - -- The dynamic offset. - Mem_Dyn : Dyn_Name; - -- Offset of the data to be accessed from the memory. - Mem_Doff : Uns32; - end case; - end record; - - type Target_Info_Array is array (Natural range <>) of Target_Info; - - function Synth_Aggregate_Target_Type (Syn_Inst : Synth_Instance_Acc; - Target : Node) return Type_Acc - is - Targ_Type : constant Node := Get_Type (Target); - Base_Type : constant Node := Get_Base_Type (Targ_Type); - Base_Typ : Type_Acc; - Bnd : Bound_Type; - Len : Uns32; - Res : Type_Acc; - begin - Base_Typ := Get_Subtype_Object (Syn_Inst, Base_Type); - -- It's a basetype, so not bounded. - pragma Assert (Base_Typ.Kind = Type_Unbounded_Vector); - - if Is_Fully_Constrained_Type (Targ_Type) then - -- If the aggregate subtype is known, just use it. - Bnd := Expr.Synth_Array_Bounds (Syn_Inst, Targ_Type, 1); - else - -- Ok, so the subtype of the aggregate is not known, in general - -- because the length of an element is not known. That's with - -- vhdl-2008. - Len := 0; - declare - Choice : Node; - El : Node; - El_Typ : Type_Acc; - begin - Choice := Get_Association_Choices_Chain (Target); - while Choice /= Null_Node loop - pragma Assert (Get_Kind (Choice) = Iir_Kind_Choice_By_None); - El := Get_Associated_Expr (Choice); - El_Typ := Synth_Type_Of_Object (Syn_Inst, El); - Bnd := Get_Array_Bound (El_Typ, 1); - Len := Len + Bnd.Len; - Choice := Get_Chain (Choice); - end loop; - end; - - -- Compute the range. - declare - Idx_Type : constant Node := Get_Index_Type (Base_Type, 0); - Idx_Typ : Type_Acc; - begin - Idx_Typ := Get_Subtype_Object (Syn_Inst, Idx_Type); - Bnd := (Dir => Idx_Typ.Drange.Dir, - Left => Int32 (Idx_Typ.Drange.Left), - Right => 0, - Len => Len); - case Bnd.Dir is - when Dir_To => - Bnd.Right := Bnd.Left + Int32 (Len); - when Dir_Downto => - Bnd.Right := Bnd.Left - Int32 (Len); - end case; - end; - end if; - - -- Compute the type. - case Base_Typ.Kind is - when Type_Unbounded_Vector => - Res := Create_Vector_Type (Bnd, Base_Typ.Uvec_El); - when others => - raise Internal_Error; - end case; - return Res; - end Synth_Aggregate_Target_Type; - - function Synth_Target (Syn_Inst : Synth_Instance_Acc; - Target : Node) return Target_Info is - begin - case Get_Kind (Target) is - when Iir_Kind_Aggregate => - return Target_Info'(Kind => Target_Aggregate, - Targ_Type => Synth_Aggregate_Target_Type - (Syn_Inst, Target), - Aggr => Target); - when Iir_Kind_Simple_Name - | Iir_Kind_Selected_Element - | Iir_Kind_Interface_Signal_Declaration - | Iir_Kind_Variable_Declaration - | Iir_Kind_Signal_Declaration - | Iir_Kind_Anonymous_Signal_Declaration - | Iir_Kind_Indexed_Name - | Iir_Kind_Slice_Name - | Iir_Kind_Dereference => - declare - Base : Valtyp; - Typ : Type_Acc; - Off : Value_Offsets; - - Dyn : Dyn_Name; - begin - Synth_Assignment_Prefix (Syn_Inst, Target, Base, Typ, Off, Dyn); - if Dyn.Voff = No_Net then - -- FIXME: check index. - return Target_Info'(Kind => Target_Simple, - Targ_Type => Typ, - Obj => Base, - Off => Off); - else - return Target_Info'(Kind => Target_Memory, - Targ_Type => Typ, - Mem_Obj => Base, - Mem_Dyn => Dyn, - Mem_Doff => Off.Net_Off); - end if; - end; - when others => - Error_Kind ("synth_target", Target); - end case; - end Synth_Target; - - procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; - Target : Target_Info; - Val : Valtyp; - Loc : Node); - - -- Extract a part of VAL from a target aggregate at offset OFF (offset - -- in the array). - function Aggregate_Extract (Ctxt : Context_Acc; - Val : Valtyp; - Off : Uns32; - Typ : Type_Acc; - Loc : Node) return Valtyp - is - El_Typ : constant Type_Acc := Get_Array_Element (Val.Typ); - begin - case Val.Val.Kind is - when Value_Net - | Value_Wire => - declare - N : Net; - begin - N := Build2_Extract - (Ctxt, Get_Net (Ctxt, Val), Off * El_Typ.W, Typ.W); - Set_Location (N, Loc); - return Create_Value_Net (N, Typ); - end; - when Value_Memory => - declare - Res : Valtyp; - begin - Res := Create_Value_Memory (Typ); - -- Need to reverse offsets. - Copy_Memory - (Res.Val.Mem, - Val.Val.Mem + (Val.Typ.Sz - Size_Type (Off + 1) * El_Typ.Sz), - Typ.Sz); - return Res; - end; - when others => - raise Internal_Error; - end case; - end Aggregate_Extract; - - procedure Synth_Assignment_Aggregate (Syn_Inst : Synth_Instance_Acc; - Target : Node; - Target_Typ : Type_Acc; - Val : Valtyp; - Loc : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Targ_Bnd : constant Bound_Type := Get_Array_Bound (Target_Typ, 1); - Choice : Node; - Assoc : Node; - Pos : Uns32; - Targ_Info : Target_Info; - begin - Choice := Get_Association_Choices_Chain (Target); - Pos := Targ_Bnd.Len; - while Is_Valid (Choice) loop - Assoc := Get_Associated_Expr (Choice); - case Get_Kind (Choice) is - when Iir_Kind_Choice_By_None => - Targ_Info := Synth_Target (Syn_Inst, Assoc); - if Get_Element_Type_Flag (Choice) then - Pos := Pos - 1; - else - Pos := Pos - Get_Array_Bound (Targ_Info.Targ_Type, 1).Len; - end if; - Synth_Assignment - (Syn_Inst, Targ_Info, - Aggregate_Extract (Ctxt, Val, Pos, - Targ_Info.Targ_Type, Assoc), - Loc); - when others => - Error_Kind ("synth_assignment_aggregate", Choice); - end case; - Choice := Get_Chain (Choice); - end loop; - end Synth_Assignment_Aggregate; - - procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; - Target : Target_Info; - Val : Valtyp; - Loc : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - V : Valtyp; - begin - V := Synth_Subtype_Conversion (Ctxt, Val, Target.Targ_Type, False, Loc); - pragma Unreferenced (Val); - if V = No_Valtyp then - -- In case of error. - return; - end if; - - case Target.Kind is - when Target_Aggregate => - Synth_Assignment_Aggregate - (Syn_Inst, Target.Aggr, Target.Targ_Type, V, Loc); - when Target_Simple => - if V.Typ.Sz = 0 then - -- If there is nothing to assign (like a null slice), - -- return now. - return; - end if; - - if Target.Obj.Val.Kind = Value_Wire then - if Is_Static (V.Val) - and then V.Typ.Sz = Target.Obj.Typ.Sz - then - pragma Assert (Target.Off = (0, 0)); - Phi_Assign_Static - (Target.Obj.Val.W, Unshare (Get_Memtyp (V))); - else - if V.Typ.W = 0 then - -- Forget about null wires. - return; - end if; - Phi_Assign_Net (Ctxt, Target.Obj.Val.W, - Get_Net (Ctxt, V), Target.Off.Net_Off); - end if; - else - if not Is_Static (V.Val) then - -- Maybe the error message is too cryptic ? - Error_Msg_Synth - (+Loc, "cannot assign a net to a static value"); - else - Strip_Const (V); - Copy_Memory (Target.Obj.Val.Mem + Target.Off.Mem_Off, - V.Val.Mem, V.Typ.Sz); - end if; - end if; - when Target_Memory => - declare - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - N : Net; - begin - N := Get_Current_Assign_Value - (Ctxt, Target.Mem_Obj.Val.W, - Target.Mem_Dyn.Pfx_Off.Net_Off, Target.Mem_Dyn.Pfx_Typ.W); - N := Build_Dyn_Insert (Ctxt, N, Get_Net (Ctxt, V), - Target.Mem_Dyn.Voff, Target.Mem_Doff); - Set_Location (N, Loc); - Phi_Assign_Net (Ctxt, Target.Mem_Obj.Val.W, N, - Target.Mem_Dyn.Pfx_Off.Net_Off); - end; - end case; - end Synth_Assignment; - - procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; - Target : Node; - Val : Valtyp; - Loc : Node) - is - Info : Target_Info; - begin - Info := Synth_Target (Syn_Inst, Target); - Synth_Assignment (Syn_Inst, Info, Val, Loc); - end Synth_Assignment; - - function Synth_Read_Memory (Syn_Inst : Synth_Instance_Acc; - Obj : Valtyp; - Res_Typ : Type_Acc; - Off : Uns32; - Dyn : Dyn_Name; - Loc : Node) return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - N : Net; - begin - N := Get_Net (Ctxt, Obj); - if Dyn.Voff /= No_Net then - Synth.Source.Set_Location_Maybe (N, Loc); - if Dyn.Pfx_Off.Net_Off /= 0 then - N := Build2_Extract (Ctxt, N, Dyn.Pfx_Off.Net_Off, Dyn.Pfx_Typ.W); - end if; - if Res_Typ.W /= 0 then - -- Do not try to extract if the net is null. - N := Build_Dyn_Extract (Ctxt, N, Dyn.Voff, Off, Res_Typ.W); - end if; - else - pragma Assert (not Is_Static (Obj.Val)); - N := Build2_Extract (Ctxt, N, Off, Res_Typ.W); - end if; - Set_Location (N, Loc); - return Create_Value_Net (N, Res_Typ); - end Synth_Read_Memory; - - function Synth_Read (Syn_Inst : Synth_Instance_Acc; - Targ : Target_Info; - Loc : Node) return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - N : Net; - begin - case Targ.Kind is - when Target_Simple => - N := Build2_Extract (Ctxt, Get_Net (Ctxt, Targ.Obj), - Targ.Off.Net_Off, Targ.Targ_Type.W); - return Create_Value_Net (N, Targ.Targ_Type); - when Target_Aggregate => - raise Internal_Error; - when Target_Memory => - return Synth_Read_Memory (Syn_Inst, Targ.Mem_Obj, Targ.Targ_Type, - 0, Targ.Mem_Dyn, Loc); - end case; - end Synth_Read; - - -- Concurrent or sequential simple signal assignment - procedure Synth_Simple_Signal_Assignment - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Targ : Target_Info; - Val : Valtyp; - begin - Targ := Synth_Target (Syn_Inst, Get_Target (Stmt)); - Val := Synth_Waveform - (Syn_Inst, Get_Waveform_Chain (Stmt), Targ.Targ_Type); - Synth_Assignment (Syn_Inst, Targ, Val, Stmt); - end Synth_Simple_Signal_Assignment; - - procedure Synth_Conditional_Signal_Assignment - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Targ : Target_Info; - Cond : Node; - Cwf : Node; - Inp : Input; - Val, Cond_Val : Valtyp; - Cond_Net : Net; - First, Last : Net; - V : Net; - begin - Targ := Synth_Target (Syn_Inst, Get_Target (Stmt)); - Last := No_Net; - Cwf := Get_Conditional_Waveform_Chain (Stmt); - Cond := Null_Node; - while Cwf /= Null_Node loop - Val := Synth_Waveform - (Syn_Inst, Get_Waveform_Chain (Cwf), Targ.Targ_Type); - if Val = No_Valtyp then - -- Mark the error, but try to continue. - Set_Error (Syn_Inst); - else - V := Get_Net (Ctxt, Val); - Cond := Get_Condition (Cwf); - if Cond /= Null_Node then - Cond_Val := Synth_Expression (Syn_Inst, Cond); - if Cond_Val = No_Valtyp then - Cond_Net := Build_Const_UB32 (Ctxt, 0, 1); - else - Cond_Net := Get_Net (Ctxt, Cond_Val); - end if; - - V := Build_Mux2 (Ctxt, Cond_Net, No_Net, V); - Set_Location (V, Cwf); - end if; - - if Last /= No_Net then - Inp := Get_Input (Get_Net_Parent (Last), 1); - Connect (Inp, V); - else - First := V; - end if; - Last := V; - end if; - Cwf := Get_Chain (Cwf); - end loop; - if Cond /= Null_Node then - pragma Assert (Last /= No_Net); - Inp := Get_Input (Get_Net_Parent (Last), 1); - if Get_Driver (Inp) = No_Net then - -- No else. - Val := Synth_Read (Syn_Inst, Targ, Stmt); - Connect (Inp, Get_Net (Ctxt, Val)); - end if; - end if; - Val := Create_Value_Net (First, Targ.Targ_Type); - Synth_Assignment (Syn_Inst, Targ, Val, Stmt); - end Synth_Conditional_Signal_Assignment; - - procedure Synth_Variable_Assignment (C : Seq_Context; Stmt : Node) - is - Targ : Target_Info; - Val : Valtyp; - begin - Targ := Synth_Target (C.Inst, Get_Target (Stmt)); - Val := Synth_Expression_With_Type - (C.Inst, Get_Expression (Stmt), Targ.Targ_Type); - if Val = No_Valtyp then - Set_Error (C.Inst); - return; - end if; - Synth_Assignment (C.Inst, Targ, Val, Stmt); - end Synth_Variable_Assignment; - - procedure Synth_Conditional_Variable_Assignment - (C : Seq_Context; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (C.Inst); - Target : constant Node := Get_Target (Stmt); - Targ_Type : Type_Acc; - Cond : Node; - Ce : Node; - Val, Cond_Val : Valtyp; - V : Net; - First, Last : Net; - begin - Targ_Type := Get_Subtype_Object (C.Inst, Get_Type (Target)); - Last := No_Net; - Ce := Get_Conditional_Expression_Chain (Stmt); - while Ce /= Null_Node loop - Val := Synth_Expression_With_Type - (C.Inst, Get_Expression (Ce), Targ_Type); - V := Get_Net (Ctxt, Val); - Cond := Get_Condition (Ce); - if Cond /= Null_Node then - Cond_Val := Synth_Expression (C.Inst, Cond); - V := Build_Mux2 (Ctxt, Get_Net (Ctxt, Cond_Val), No_Net, V); - Set_Location (V, Ce); - end if; - - if Last /= No_Net then - Connect (Get_Input (Get_Net_Parent (Last), 1), V); - else - First := V; - end if; - Last := V; - Ce := Get_Chain (Ce); - end loop; - Val := Create_Value_Net (First, Targ_Type); - Synth_Assignment (C.Inst, Target, Val, Stmt); - end Synth_Conditional_Variable_Assignment; - - procedure Synth_If_Statement (C : in out Seq_Context; Stmt : Node) - is - Cond : constant Node := Get_Condition (Stmt); - Els : constant Node := Get_Else_Clause (Stmt); - Ctxt : constant Context_Acc := Get_Build (C.Inst); - Cond_Val : Valtyp; - Cond_Net : Net; - Phi_True : Phi_Type; - Phi_False : Phi_Type; - begin - Cond_Val := Synth_Expression (C.Inst, Cond); - if Cond_Val = No_Valtyp then - Set_Error (C.Inst); - return; - end if; - if Is_Static_Val (Cond_Val.Val) then - Strip_Const (Cond_Val); - if Read_Discrete (Get_Value_Memtyp (Cond_Val)) = 1 then - -- True. - Synth_Sequential_Statements - (C, Get_Sequential_Statement_Chain (Stmt)); - else - pragma Assert (Read_Discrete (Get_Value_Memtyp (Cond_Val)) = 0); - if Is_Valid (Els) then - -- Else part - if Is_Null (Get_Condition (Els)) then - -- Final else part. - Synth_Sequential_Statements - (C, Get_Sequential_Statement_Chain (Els)); - else - -- Elsif. Handled as a nested if. - Synth_If_Statement (C, Els); - end if; - end if; - end if; - else - -- The statements for the 'then' part. - Push_Phi; - Synth_Sequential_Statements - (C, Get_Sequential_Statement_Chain (Stmt)); - Pop_Phi (Phi_True); - - Push_Phi; - - if Is_Valid (Els) then - if Is_Null (Get_Condition (Els)) then - -- Final else part. - Synth_Sequential_Statements - (C, Get_Sequential_Statement_Chain (Els)); - else - -- Elsif. Handled as a nested if. - Synth_If_Statement (C, Els); - end if; - end if; - - Pop_Phi (Phi_False); - - Cond_Net := Get_Net (Ctxt, Cond_Val); - Merge_Phis (Ctxt, Cond_Net, Phi_True, Phi_False, Get_Location (Stmt)); - end if; - end Synth_If_Statement; - - type Alternative_Index is new Int32; - - -- Only keep '0' and '1' in choices for std_logic. - function Ignore_Choice_Logic (V : Ghdl_U8; Loc : Node) return Boolean is - begin - case V is - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_0_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_1_Pos => - return False; - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_L_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_H_Pos => - Warning_Msg_Synth - (+Loc, "choice with 'L' or 'H' value is ignored"); - return True; - when Vhdl.Ieee.Std_Logic_1164.Std_Logic_U_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_X_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_D_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_Z_Pos - | Vhdl.Ieee.Std_Logic_1164.Std_Logic_W_Pos => - Warning_Msg_Synth (+Loc, "choice with meta-value is ignored"); - return True; - when others => - -- Only 9 values. - raise Internal_Error; - end case; - end Ignore_Choice_Logic; - - function Ignore_Choice_Expression (V : Valtyp; Loc : Node) return Boolean is - begin - case V.Typ.Kind is - when Type_Bit => - return False; - when Type_Logic => - if V.Typ = Logic_Type then - return Ignore_Choice_Logic (Read_U8 (V.Val.Mem), Loc); - else - return False; - end if; - when Type_Discrete => - return False; - when Type_Vector => - if V.Typ.Vec_El = Logic_Type then - for I in 1 .. Size_Type (V.Typ.Vbound.Len) loop - if Ignore_Choice_Logic (Read_U8 (V.Val.Mem + (I - 1)), Loc) - then - return True; - end if; - end loop; - return False; - else - return False; - end if; - when Type_Array => - return False; - when others => - raise Internal_Error; - end case; - end Ignore_Choice_Expression; - - -- Create the condition for choices of CHOICE chain belonging to the same - -- alternative. Update CHOICE to the next alternative. - procedure Synth_Choice (Syn_Inst : Synth_Instance_Acc; - Sel : Net; - Choice_Typ : Type_Acc; - Nets : in out Net_Array; - Other_Choice : in out Nat32; - Choice_Idx : in out Nat32; - Choice : in out Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Cond : Net; - Res : Net; - begin - Res := No_Net; - loop - case Iir_Kinds_Case_Choice (Get_Kind (Choice)) is - when Iir_Kind_Choice_By_Expression => - declare - V : Valtyp; - begin - V := Synth_Expression_With_Basetype - (Syn_Inst, Get_Choice_Expression (Choice)); - V := Synth_Subtype_Conversion - (Ctxt, V, Choice_Typ, False, Choice); - if Ignore_Choice_Expression (V, Choice) then - Cond := No_Net; - else - Cond := Build_Compare - (Ctxt, Id_Eq, Sel, Get_Net (Ctxt, V)); - Set_Location (Cond, Choice); - end if; - end; - - when Iir_Kind_Choice_By_Range => - declare - Rng : Discrete_Range_Type; - Cmp_L, Cmp_R : Module_Id; - L, R : Net; - begin - Synth_Discrete_Range - (Syn_Inst, Get_Choice_Range (Choice), Rng); - - if Rng.Is_Signed then - case Rng.Dir is - when Dir_To => - Cmp_L := Id_Sge; - Cmp_R := Id_Sle; - when Dir_Downto => - Cmp_L := Id_Sle; - Cmp_R := Id_Sge; - end case; - L := Build2_Const_Int (Ctxt, Rng.Left, Choice_Typ.W); - R := Build2_Const_Int (Ctxt, Rng.Right, Choice_Typ.W); - else - case Rng.Dir is - when Dir_To => - Cmp_L := Id_Uge; - Cmp_R := Id_Ule; - when Dir_Downto => - Cmp_L := Id_Ule; - Cmp_R := Id_Uge; - end case; - L := Build2_Const_Uns - (Ctxt, Uns64 (Rng.Left), Choice_Typ.W); - R := Build2_Const_Uns - (Ctxt, Uns64 (Rng.Right), Choice_Typ.W); - end if; - - L := Build_Compare (Ctxt, Cmp_L, Sel, L); - Set_Location (L, Choice); - - R := Build_Compare (Ctxt, Cmp_R, Sel, R); - Set_Location (R, Choice); - - Cond := Build_Dyadic (Ctxt, Id_And, L, R); - Set_Location (Cond, Choice); - end; - - when Iir_Kind_Choice_By_Others => - -- Last and only one. - pragma Assert (Res = No_Net); - Other_Choice := Choice_Idx + 1; - pragma Assert (Get_Chain (Choice) = Null_Node); - Choice := Null_Node; - return; - end case; - - if not Get_Same_Alternative_Flag (Choice) then - -- First choice. - Choice_Idx := Choice_Idx + 1; - Res := Cond; - else - if Cond = No_Net then - -- No new condition. - null; - else - if Res /= No_Net then - Res := Build_Dyadic (Ctxt, Id_Or, Res, Cond); - Set_Location (Res, Choice); - else - Res := Cond; - end if; - end if; - end if; - - Choice := Get_Chain (Choice); - exit when Choice = Null_Node - or else not Get_Same_Alternative_Flag (Choice); - end loop; - if Res = No_Net then - Res := Build_Const_UB32 (Ctxt, 0, 1); - end if; - Nets (Choice_Idx) := Res; - end Synth_Choice; - - type Alternative_Data_Type is record - Asgns : Seq_Assign; - Val : Net; - end record; - type Alternative_Data_Array is - array (Alternative_Index range <>) of Alternative_Data_Type; - type Alternative_Data_Acc is access Alternative_Data_Array; - procedure Free_Alternative_Data_Array is new Ada.Unchecked_Deallocation - (Alternative_Data_Array, Alternative_Data_Acc); - - type Wire_Id_Array is array (Natural range <>) of Wire_Id; - type Wire_Id_Array_Acc is access Wire_Id_Array; - procedure Free_Wire_Id_Array is new Ada.Unchecked_Deallocation - (Wire_Id_Array, Wire_Id_Array_Acc); - - procedure Sort_Wire_Id_Array (Arr : in out Wire_Id_Array) - is - function Lt (Op1, Op2 : Natural) return Boolean is - begin - return Is_Lt (Arr (Op1), Arr (Op2)); - end Lt; - - procedure Swap (From : Natural; To : Natural) - is - T : Wire_Id; - begin - T := Arr (From); - Arr (From) := Arr (To); - Arr (To) := T; - end Swap; - - procedure Wid_Heap_Sort is - new Grt.Algos.Heap_Sort (Lt => Lt, Swap => Swap); - begin - Wid_Heap_Sort (Arr'Length); - end Sort_Wire_Id_Array; - - -- Count the number of wires used in all the alternatives. - function Count_Wires_In_Alternatives (Alts : Alternative_Data_Array) - return Natural - is - Res : Natural; - Asgn : Seq_Assign; - W : Wire_Id; - begin - Res := 0; - for I in Alts'Range loop - Asgn := Alts (I).Asgns; - while Asgn /= No_Seq_Assign loop - W := Get_Wire_Id (Asgn); - if not Get_Wire_Mark (W) then - Res := Res + 1; - Set_Wire_Mark (W, True); - end if; - Asgn := Get_Assign_Chain (Asgn); - end loop; - end loop; - return Res; - end Count_Wires_In_Alternatives; - - -- Fill ARR from wire_id of ALTS. - procedure Fill_Wire_Id_Array (Arr : out Wire_Id_Array; - Alts : Alternative_Data_Array) - is - Idx : Natural; - Asgn : Seq_Assign; - W : Wire_Id; - begin - Idx := Arr'First; - for I in Alts'Range loop - Asgn := Alts (I).Asgns; - while Asgn /= No_Seq_Assign loop - W := Get_Wire_Id (Asgn); - if Get_Wire_Mark (W) then - Arr (Idx) := W; - Idx := Idx + 1; - Set_Wire_Mark (W, False); - end if; - Asgn := Get_Assign_Chain (Asgn); - end loop; - end loop; - pragma Assert (Idx = Arr'Last + 1); - end Fill_Wire_Id_Array; - - type Seq_Assign_Value_Array_Acc is access Seq_Assign_Value_Array; - procedure Free_Seq_Assign_Value_Array is new Ada.Unchecked_Deallocation - (Seq_Assign_Value_Array, Seq_Assign_Value_Array_Acc); - - function Is_Assign_Value_Array_Static - (Wid : Wire_Id; Arr : Seq_Assign_Value_Array) return Memtyp - is - Res : Memtyp; - Prev_Val : Memtyp; - begin - Prev_Val := Null_Memtyp; - for I in Arr'Range loop - case Arr (I).Is_Static is - when False => - -- A value is not static. - return Null_Memtyp; - when Unknown => - if Prev_Val = Null_Memtyp then - -- First use of previous value. - if not Is_Static_Wire (Wid) then - -- The previous value is not static. - return Null_Memtyp; - end if; - Prev_Val := Get_Static_Wire (Wid); - if Res /= Null_Memtyp then - -- There is already a result. - if not Is_Equal (Res, Prev_Val) then - -- The previous value is different from the result. - return Null_Memtyp; - end if; - else - Res := Prev_Val; - end if; - end if; - when True => - if Res = Null_Memtyp then - -- First value. Keep it. - Res := Arr (I).Val; - else - if not Is_Equal (Res, Arr (I).Val) then - -- Value is different. - return Null_Memtyp; - end if; - end if; - end case; - end loop; - return Res; - end Is_Assign_Value_Array_Static; - - procedure Synth_Case_Statement_Dynamic - (C : in out Seq_Context; Stmt : Node; Sel : Valtyp) - is - use Vhdl.Sem_Expr; - Ctxt : constant Context_Acc := Get_Build (C.Inst); - - Choices : constant Node := Get_Case_Statement_Alternative_Chain (Stmt); - - Case_Info : Choice_Info_Type; - - -- Array of alternatives - Alts : Alternative_Data_Acc; - Alt_Idx : Alternative_Index; - Others_Alt_Idx : Alternative_Index; - - Nbr_Choices : Nat32; - - Pasgns : Seq_Assign_Value_Array_Acc; - Nets : Net_Array_Acc; - - Nbr_Wires : Natural; - Wires : Wire_Id_Array_Acc; - - Sel_Net : Net; - begin - -- Strategies to synthesize a case statement. Assume the selector is - -- a net of W bits - -- - a large mux, with 2**W inputs - -- - if the number of choices is dense - -- - if W is small - -- - a onehot mux. Each choice is converted to an single bit condition - -- by adding a comparison operator (equal for single choice, - -- inequalities for ranges, or for multiple choices). Only one of - -- these conditions is true (plus 'others'). - -- - if the number of choices is sparse - -- - large range choices - -- - a tree of mux/mux2 - -- - large number of choices, densily grouped but sparsed compared - -- to 2**W (eg: a partially filled memory) - -- - divide and conquier - - -- Count choices and alternatives. - Count_Choices (Case_Info, Choices); - --Fill_Choices_Array (Case_Info, Choices); - - -- Allocate structures. - -- Because there is no 1-1 link between choices and alternatives, - -- create an array for the choices and an array for the alternatives. - Alts := new Alternative_Data_Array - (1 .. Alternative_Index (Case_Info.Nbr_Alternatives)); - - -- Compute number of non-default alternatives. - Nbr_Choices := Nat32 (Case_Info.Nbr_Alternatives); - if Case_Info.Others_Choice /= Null_Node then - Nbr_Choices := Nbr_Choices - 1; - end if; - - Nets := new Net_Array (1 .. Int32 (Alts'Last)); - - Sel_Net := Get_Net (Ctxt, Sel); - - -- Synth statements and keep list of assignments. - -- Also synth choices. - declare - Choice : Node; - Choice_Idx, Other_Choice : Nat32; - Phi : Phi_Type; - begin - Alt_Idx := 0; - Choice_Idx := 0; - Other_Choice := 0; - - Choice := Choices; - while Is_Valid (Choice) loop - -- Must be a choice for a new alternative. - pragma Assert (not Get_Same_Alternative_Flag (Choice)); - - -- A new sequence of statements. - Alt_Idx := Alt_Idx + 1; - - Push_Phi; - Synth_Sequential_Statements (C, Get_Associated_Chain (Choice)); - Pop_Phi (Phi); - Alts (Alt_Idx).Asgns := Sort_Phi (Phi); - - Synth_Choice (C.Inst, Sel_Net, Sel.Typ, - Nets.all, Other_Choice, Choice_Idx, Choice); - end loop; - pragma Assert (Choice_Idx = Nbr_Choices); - Others_Alt_Idx := Alternative_Index (Other_Choice); - end; - - -- Create the one-hot vector. - if Nbr_Choices = 0 then - Sel_Net := No_Net; - else - Sel_Net := Build2_Concat (Ctxt, Nets (1 .. Nbr_Choices)); - end if; - - -- Create list of wire_id, sort it. - Nbr_Wires := Count_Wires_In_Alternatives (Alts.all); - Wires := new Wire_Id_Array (1 .. Nbr_Wires); - Fill_Wire_Id_Array (Wires.all, Alts.all); - Sort_Wire_Id_Array (Wires.all); - - -- Associate each choice with the assign node - -- For each wire_id: - -- Build mux2/mux4 tree (group by 4) - Pasgns := new Seq_Assign_Value_Array (1 .. Int32 (Alts'Last)); - - -- For each wire, compute the result. - for I in Wires'Range loop - declare - Wi : constant Wire_Id := Wires (I); - Last_Val : Net; - Res_Inst : Instance; - Res : Net; - Default : Net; - Min_Off, Off : Uns32; - Wd : Width; - List : Partial_Assign_List; - Sval : Memtyp; - begin - -- Extract the value for each branch. - for I in Alts'Range loop - -- If there is an assignment to Wi in Alt, it will define the - -- value. - if Get_Wire_Id (Alts (I).Asgns) = Wi then - Pasgns (Int32 (I)) := - Get_Seq_Assign_Value (Alts (I).Asgns); - Alts (I).Asgns := Get_Assign_Chain (Alts (I).Asgns); - else - Pasgns (Int32 (I)) := (Is_Static => Unknown); - end if; - end loop; - - -- If: - -- 1) All present values in PASGNS are static - -- 2) There is no missing values *or* the previous value is - -- static. - -- 3) The default value is unused *or* it is static - -- 4) All the values are equal. - -- then assign directly. - Sval := Is_Assign_Value_Array_Static (Wi, Pasgns.all); - if Sval /= Null_Memtyp then - -- Use static assignment. - Phi_Assign_Static (Wi, Sval); - else - -- Compute the final value for each partial part of the wire. - Partial_Assign_Init (List); - Min_Off := 0; - loop - Off := Min_Off; - - -- Extract value of partial assignments to NETS. - Extract_Merge_Partial_Assigns - (Ctxt, Pasgns.all, Nets.all, Off, Wd); - exit when Off = Uns32'Last and Wd = Width'Last; - - -- If a branch has no value, use the value before the case. - -- Also do it for the default value! - Last_Val := No_Net; - for I in Nets'Range loop - if Nets (I) = No_Net then - if Last_Val = No_Net then - Last_Val := Get_Current_Assign_Value - (Ctxt, Wi, Off, Wd); - end if; - Nets (I) := Last_Val; - end if; - end loop; - - -- Extract default value (for missing alternative). - if Others_Alt_Idx /= 0 then - Default := Nets (Int32 (Others_Alt_Idx)); - else - Default := Build_Const_X (Ctxt, Wd); - end if; - - if Nbr_Choices = 0 then - Res := Default; - else - Res := Build_Pmux (Ctxt, Sel_Net, Default); - Res_Inst := Get_Net_Parent (Res); - Set_Location (Res_Inst, Get_Location (Stmt)); - - for I in 1 .. Nbr_Choices loop - Connect - (Get_Input (Res_Inst, Port_Nbr (2 + I - Nets'First)), - Nets (I)); - end loop; - end if; - - Partial_Assign_Append (List, New_Partial_Assign (Res, Off)); - Min_Off := Off + Wd; - end loop; - - Merge_Partial_Assigns (Ctxt, Wi, List); - end if; - end; - end loop; - - -- free. - Free_Wire_Id_Array (Wires); - Free_Alternative_Data_Array (Alts); - Free_Seq_Assign_Value_Array (Pasgns); - Free_Net_Array (Nets); - end Synth_Case_Statement_Dynamic; - - procedure Synth_Case_Statement_Static_Array - (C : in out Seq_Context; Stmt : Node; Sel : Valtyp) - is - Choices : constant Node := Get_Case_Statement_Alternative_Chain (Stmt); - Choice : Node; - Stmts : Node; - Sel_Expr : Node; - Sel_Val : Valtyp; - begin - -- Synth statements, extract choice value. - Stmts := Null_Node; - Choice := Choices; - loop - pragma Assert (Is_Valid (Choice)); - if not Get_Same_Alternative_Flag (Choice) then - Stmts := Get_Associated_Chain (Choice); - end if; - - case Get_Kind (Choice) is - when Iir_Kind_Choice_By_Expression => - Sel_Expr := Get_Choice_Expression (Choice); - Sel_Val := Synth_Expression_With_Basetype (C.Inst, Sel_Expr); - if Is_Equal (Sel_Val, Sel) then - Synth_Sequential_Statements (C, Stmts); - exit; - end if; - when Iir_Kind_Choice_By_Others => - Synth_Sequential_Statements (C, Stmts); - exit; - when others => - raise Internal_Error; - end case; - Choice := Get_Chain (Choice); - end loop; - end Synth_Case_Statement_Static_Array; - - procedure Synth_Case_Statement_Static_Scalar - (C : in out Seq_Context; Stmt : Node; Sel : Int64) - is - Choices : constant Node := Get_Case_Statement_Alternative_Chain (Stmt); - Choice : Node; - Stmts : Node; - Sel_Expr : Node; - begin - -- Synth statements, extract choice value. - Stmts := Null_Node; - Choice := Choices; - loop - pragma Assert (Is_Valid (Choice)); - if not Get_Same_Alternative_Flag (Choice) then - Stmts := Get_Associated_Chain (Choice); - end if; - - case Get_Kind (Choice) is - when Iir_Kind_Choice_By_Expression => - Sel_Expr := Get_Choice_Expression (Choice); - if Vhdl.Evaluation.Eval_Pos (Sel_Expr) = Sel then - Synth_Sequential_Statements (C, Stmts); - exit; - end if; - when Iir_Kind_Choice_By_Others => - Synth_Sequential_Statements (C, Stmts); - exit; - when Iir_Kind_Choice_By_Range => - declare - Bnd : Discrete_Range_Type; - Is_In : Boolean; - begin - Synth_Discrete_Range - (C.Inst, Get_Choice_Range (Choice), Bnd); - case Bnd.Dir is - when Dir_To => - Is_In := Sel >= Bnd.Left and Sel <= Bnd.Right; - when Dir_Downto => - Is_In := Sel <= Bnd.Left and Sel >= Bnd.Right; - end case; - if Is_In then - Synth_Sequential_Statements (C, Stmts); - exit; - end if; - end; - when others => - raise Internal_Error; - end case; - Choice := Get_Chain (Choice); - end loop; - end Synth_Case_Statement_Static_Scalar; - - procedure Synth_Case_Statement (C : in out Seq_Context; Stmt : Node) - is - Expr : constant Node := Get_Expression (Stmt); - Sel : Valtyp; - begin - Sel := Synth_Expression_With_Basetype (C.Inst, Expr); - Strip_Const (Sel); - if Is_Static (Sel.Val) then - case Sel.Typ.Kind is - when Type_Bit - | Type_Logic - | Type_Discrete => - Synth_Case_Statement_Static_Scalar (C, Stmt, - Read_Discrete (Sel)); - when Type_Vector - | Type_Array => - Synth_Case_Statement_Static_Array (C, Stmt, Sel); - when others => - raise Internal_Error; - end case; - else - Synth_Case_Statement_Dynamic (C, Stmt, Sel); - end if; - end Synth_Case_Statement; - - procedure Synth_Selected_Signal_Assignment - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - use Vhdl.Sem_Expr; - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - - Expr : constant Node := Get_Expression (Stmt); - Choices : constant Node := Get_Selected_Waveform_Chain (Stmt); - - Targ : Target_Info; - Targ_Type : Type_Acc; - - Case_Info : Choice_Info_Type; - - -- Array of alternatives - Alts : Alternative_Data_Acc; - Alt_Idx : Alternative_Index; - Others_Alt_Idx : Alternative_Index; - - -- Array of choices. Contains tuple of (Value, Alternative). - Nbr_Choices : Nat32; - - Nets : Net_Array_Acc; - - - Sel : Valtyp; - Sel_Net : Net; - begin - Targ := Synth_Target (Syn_Inst, Get_Target (Stmt)); - Targ_Type := Targ.Targ_Type; - - -- Create a net for the expression. - Sel := Synth_Expression_With_Basetype (Syn_Inst, Expr); - Sel_Net := Get_Net (Ctxt, Sel); - - -- Count choices and alternatives. - Count_Choices (Case_Info, Choices); - -- Fill_Choices_Array (Case_Info, Choices); - - -- Allocate structures. - -- Because there is no 1-1 link between choices and alternatives, - -- create an array for the choices and an array for the alternatives. - Alts := new Alternative_Data_Array - (1 .. Alternative_Index (Case_Info.Nbr_Alternatives)); - - -- Compute number of non-default alternatives. - Nbr_Choices := Nat32 (Case_Info.Nbr_Alternatives); - if Case_Info.Others_Choice /= Null_Node then - Nbr_Choices := Nbr_Choices - 1; - end if; - - Nets := new Net_Array (1 .. Nbr_Choices); - - -- Synth statements, extract choice value. - declare - Choice, Wf : Node; - Val : Valtyp; - Choice_Idx, Other_Choice : Nat32; - begin - Alt_Idx := 0; - Choice_Idx := 0; - Other_Choice := 0; - - Choice := Choices; - while Is_Valid (Choice) loop - pragma Assert (not Get_Same_Alternative_Flag (Choice)); - - Wf := Get_Associated_Chain (Choice); - Val := Synth_Waveform (Syn_Inst, Wf, Targ_Type); - - Alt_Idx := Alt_Idx + 1; - Alts (Alt_Idx).Val := Get_Net (Ctxt, Val); - - Synth_Choice (Syn_Inst, Sel_Net, Sel.Typ, - Nets.all, Other_Choice, Choice_Idx, Choice); - end loop; - pragma Assert (Choice_Idx = Nbr_Choices); - Others_Alt_Idx := Alternative_Index (Other_Choice); - end; - - -- Create the one-hot vector. - if Nbr_Choices = 0 then - Sel_Net := No_Net; - else - Sel_Net := Build2_Concat (Ctxt, Nets (1 .. Nbr_Choices)); - end if; - - declare - Res : Net; - Res_Inst : Instance; - Default : Net; - begin - -- Extract default value (for missing alternative). - if Others_Alt_Idx /= 0 then - Default := Alts (Others_Alt_Idx).Val; - else - Default := Build_Const_X (Ctxt, Targ_Type.W); - end if; - - if Nbr_Choices = 0 then - Res := Default; - else - Res := Build_Pmux (Ctxt, Sel_Net, Default); - Res_Inst := Get_Net_Parent (Res); - Set_Location (Res_Inst, Get_Location (Stmt)); - - for I in 1 .. Nbr_Choices loop - Connect - (Get_Input (Res_Inst, Port_Nbr (2 + I - Nets'First)), - Alts (Alternative_Index (I)).Val); - end loop; - end if; - - Synth_Assignment - (Syn_Inst, Targ, Create_Value_Net (Res, Targ_Type), Stmt); - end; - - -- free. - Free_Alternative_Data_Array (Alts); - Free_Net_Array (Nets); - end Synth_Selected_Signal_Assignment; - - function Synth_Label (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - return Sname - is - Label : constant Name_Id := Get_Label (Stmt); - begin - if Label = Null_Identifier then - return No_Sname; - else - return New_Sname_User (Label, Get_Sname (Syn_Inst)); - end if; - end Synth_Label; - - function Is_Copyback_Interface (Inter : Node) return Boolean is - begin - case Iir_Parameter_Modes (Get_Mode (Inter)) is - when Iir_In_Mode => - return False; - when Iir_Out_Mode | Iir_Inout_Mode => - return Get_Kind (Inter) = Iir_Kind_Interface_Variable_Declaration; - end case; - end Is_Copyback_Interface; - - type Association_Iterator_Kind is - (Association_Function, - Association_Operator); - - type Association_Iterator_Init - (Kind : Association_Iterator_Kind := Association_Function) is - record - Inter_Chain : Node; - case Kind is - when Association_Function => - Assoc_Chain : Node; - when Association_Operator => - Left : Node; - Right : Node; - end case; - end record; - - function Association_Iterator_Build (Inter_Chain : Node; Assoc_Chain : Node) - return Association_Iterator_Init is - begin - return Association_Iterator_Init'(Kind => Association_Function, - Inter_Chain => Inter_Chain, - Assoc_Chain => Assoc_Chain); - end Association_Iterator_Build; - - function Association_Iterator_Build - (Inter_Chain : Node; Left : Node; Right : Node) - return Association_Iterator_Init is - begin - return Association_Iterator_Init'(Kind => Association_Operator, - Inter_Chain => Inter_Chain, - Left => Left, - Right => Right); - end Association_Iterator_Build; - - function Count_Associations (Init : Association_Iterator_Init) - return Natural - is - Assoc : Node; - Assoc_Inter : Node; - Inter : Node; - Nbr_Inout : Natural; - begin - case Init.Kind is - when Association_Function => - Nbr_Inout := 0; - - Assoc := Init.Assoc_Chain; - Assoc_Inter := Init.Inter_Chain; - while Is_Valid (Assoc) loop - Inter := Get_Association_Interface (Assoc, Assoc_Inter); - - if Is_Copyback_Interface (Inter) then - Nbr_Inout := Nbr_Inout + 1; - end if; - - Next_Association_Interface (Assoc, Assoc_Inter); - end loop; - - return Nbr_Inout; - when Association_Operator => - return 0; - end case; - end Count_Associations; - - type Association_Iterator - (Kind : Association_Iterator_Kind := Association_Function) is - record - Inter : Node; - case Kind is - when Association_Function => - First_Named_Assoc : Node; - Next_Assoc : Node; - when Association_Operator => - Op1 : Node; - Op2 : Node; - end case; - end record; - - procedure Association_Iterate_Init (Iterator : out Association_Iterator; - Init : Association_Iterator_Init) is - begin - case Init.Kind is - when Association_Function => - Iterator := (Kind => Association_Function, - Inter => Init.Inter_Chain, - First_Named_Assoc => Null_Node, - Next_Assoc => Init.Assoc_Chain); - when Association_Operator => - Iterator := (Kind => Association_Operator, - Inter => Init.Inter_Chain, - Op1 => Init.Left, - Op2 => Init.Right); - end case; - end Association_Iterate_Init; - - -- Return the next association. - -- ASSOC can be: - -- * an Iir_Kind_Association_By_XXX node (normal case) - -- * Null_Iir if INTER is not associated (and has a default value). - -- * an expression (for operator association). - procedure Association_Iterate_Next (Iterator : in out Association_Iterator; - Inter : out Node; - Assoc : out Node) - is - Formal : Node; - begin - Inter := Iterator.Inter; - if Inter = Null_Node then - -- End of iterator. - Assoc := Null_Node; - return; - else - -- Advance to the next interface for the next call. - Iterator.Inter := Get_Chain (Iterator.Inter); - end if; - - case Iterator.Kind is - when Association_Function => - if Iterator.First_Named_Assoc = Null_Node then - Assoc := Iterator.Next_Assoc; - if Assoc = Null_Node then - -- No more association: open association. - return; - end if; - Formal := Get_Formal (Assoc); - if Formal = Null_Node then - -- Association by position. - -- Update for the next call. - Iterator.Next_Assoc := Get_Chain (Assoc); - return; - end if; - Iterator.First_Named_Assoc := Assoc; - end if; - - -- Search by name. - Assoc := Iterator.First_Named_Assoc; - while Assoc /= Null_Node loop - Formal := Get_Formal (Assoc); - pragma Assert (Formal /= Null_Node); - Formal := Get_Interface_Of_Formal (Formal); - if Formal = Inter then - -- Found. - -- Optimize in case assocs are in order. - if Assoc = Iterator.First_Named_Assoc then - Iterator.First_Named_Assoc := Get_Chain (Assoc); - end if; - return; - end if; - Assoc := Get_Chain (Assoc); - end loop; - - -- Not found: open association. - return; - - when Association_Operator => - Assoc := Iterator.Op1; - Iterator.Op1 := Iterator.Op2; - Iterator.Op2 := Null_Node; - end case; - end Association_Iterate_Next; - - procedure Synth_Subprogram_Association (Subprg_Inst : Synth_Instance_Acc; - Caller_Inst : Synth_Instance_Acc; - Init : Association_Iterator_Init; - Infos : out Target_Info_Array) - is - pragma Assert (Infos'First = 1); - Ctxt : constant Context_Acc := Get_Build (Caller_Inst); - Inter : Node; - Inter_Type : Type_Acc; - Assoc : Node; - Actual : Node; - Val : Valtyp; - Nbr_Inout : Natural; - Iterator : Association_Iterator; - Info : Target_Info; - begin - Set_Instance_Const (Subprg_Inst, True); - - Nbr_Inout := 0; - - -- Process in INTER order. - Association_Iterate_Init (Iterator, Init); - loop - Association_Iterate_Next (Iterator, Inter, Assoc); - exit when Inter = Null_Node; - - Inter_Type := Get_Subtype_Object (Caller_Inst, Get_Type (Inter)); - - case Iir_Parameter_Modes (Get_Mode (Inter)) is - when Iir_In_Mode => - if Assoc = Null_Node - or else Get_Kind (Assoc) = Iir_Kind_Association_Element_Open - then - Actual := Get_Default_Value (Inter); - Val := Synth_Expression_With_Type - (Subprg_Inst, Actual, Inter_Type); - else - if Get_Kind (Assoc) = - Iir_Kind_Association_Element_By_Expression - then - Actual := Get_Actual (Assoc); - else - Actual := Assoc; - end if; - Val := Synth_Expression_With_Type - (Caller_Inst, Actual, Inter_Type); - end if; - when Iir_Out_Mode | Iir_Inout_Mode => - Actual := Get_Actual (Assoc); - Info := Synth_Target (Caller_Inst, Actual); - - case Iir_Kinds_Interface_Object_Declaration (Get_Kind (Inter)) - is - when Iir_Kind_Interface_Constant_Declaration => - raise Internal_Error; - when Iir_Kind_Interface_Variable_Declaration => - -- Always pass by value. - Nbr_Inout := Nbr_Inout + 1; - Infos (Nbr_Inout) := Info; - if Info.Kind /= Target_Memory - and then Is_Static (Info.Obj.Val) - then - Val := Create_Value_Memory (Info.Targ_Type); - Copy_Memory (Val.Val.Mem, - Info.Obj.Val.Mem + Info.Off.Mem_Off, - Info.Targ_Type.Sz); - else - Val := Synth_Read (Caller_Inst, Info, Assoc); - end if; - when Iir_Kind_Interface_Signal_Declaration => - -- Always pass by reference (use an alias). - if Info.Kind = Target_Memory then - raise Internal_Error; - end if; - Val := Create_Value_Alias - (Info.Obj, Info.Off, Info.Targ_Type); - when Iir_Kind_Interface_File_Declaration => - Val := Info.Obj; - when Iir_Kind_Interface_Quantity_Declaration => - raise Internal_Error; - end case; - end case; - - if Val = No_Valtyp then - Set_Error (Subprg_Inst); - return; - end if; - - -- FIXME: conversion only for constants, reshape for all. - Val := Synth_Subtype_Conversion (Ctxt, Val, Inter_Type, True, Assoc); - - if Get_Instance_Const (Subprg_Inst) and then not Is_Static (Val.Val) - then - Set_Instance_Const (Subprg_Inst, False); - end if; - - case Iir_Kinds_Interface_Object_Declaration (Get_Kind (Inter)) is - when Iir_Kind_Interface_Constant_Declaration => - -- Pass by reference. - Create_Object (Subprg_Inst, Inter, Val); - when Iir_Kind_Interface_Variable_Declaration => - -- Arguments are passed by copy. - if Is_Static (Val.Val) or else Get_Mode (Inter) = Iir_In_Mode - then - Val := Unshare (Val, Current_Pool); - else - -- Will be changed to a wire. - null; - end if; - Create_Object (Subprg_Inst, Inter, Val); - when Iir_Kind_Interface_Signal_Declaration => - Create_Object (Subprg_Inst, Inter, Val); - when Iir_Kind_Interface_File_Declaration => - Create_Object (Subprg_Inst, Inter, Val); - when Iir_Kind_Interface_Quantity_Declaration => - raise Internal_Error; - end case; - end loop; - end Synth_Subprogram_Association; - - procedure Synth_Subprogram_Association (Subprg_Inst : Synth_Instance_Acc; - Caller_Inst : Synth_Instance_Acc; - Inter_Chain : Node; - Assoc_Chain : Node) - is - Infos : Target_Info_Array (1 .. 0); - pragma Unreferenced (Infos); - Init : Association_Iterator_Init; - begin - Init := Association_Iterator_Build (Inter_Chain, Assoc_Chain); - Synth_Subprogram_Association (Subprg_Inst, Caller_Inst, Init, Infos); - end Synth_Subprogram_Association; - - -- Create wires for out and inout interface variables. - procedure Synth_Subprogram_Association_Wires - (Subprg_Inst : Synth_Instance_Acc; Init : Association_Iterator_Init) - is - Ctxt : constant Context_Acc := Get_Build (Subprg_Inst); - Inter : Node; - Assoc : Node; - Val : Valtyp; - Iterator : Association_Iterator; - Wire : Wire_Id; - begin - -- Process in INTER order. - Association_Iterate_Init (Iterator, Init); - loop - Association_Iterate_Next (Iterator, Inter, Assoc); - exit when Inter = Null_Node; - - if Get_Mode (Inter) in Iir_Out_Modes - and then Get_Kind (Inter) = Iir_Kind_Interface_Variable_Declaration - then - Val := Get_Value (Subprg_Inst, Inter); - -- Arguments are passed by copy. - Wire := Alloc_Wire (Wire_Variable, (Inter, Val.Typ)); - Set_Wire_Gate (Wire, Get_Net (Ctxt, Val)); - - Val := Create_Value_Wire (Wire, Val.Typ); - Create_Object_Force (Subprg_Inst, Inter, No_Valtyp); - Create_Object_Force (Subprg_Inst, Inter, Val); - end if; - end loop; - end Synth_Subprogram_Association_Wires; - - procedure Synth_Subprogram_Back_Association - (Subprg_Inst : Synth_Instance_Acc; - Caller_Inst : Synth_Instance_Acc; - Init : Association_Iterator_Init; - Infos : Target_Info_Array) - is - pragma Assert (Infos'First = 1); - Inter : Node; - Assoc : Node; - Assoc_Inter : Node; - Val : Valtyp; - Nbr_Inout : Natural; - begin - Nbr_Inout := 0; - pragma Assert (Init.Kind = Association_Function); - Assoc := Init.Assoc_Chain; - Assoc_Inter := Init.Inter_Chain; - while Is_Valid (Assoc) loop - Inter := Get_Association_Interface (Assoc, Assoc_Inter); - - if Is_Copyback_Interface (Inter) then - if not Get_Whole_Association_Flag (Assoc) then - raise Internal_Error; - end if; - Nbr_Inout := Nbr_Inout + 1; - Val := Get_Value (Subprg_Inst, Inter); - Synth_Assignment (Caller_Inst, Infos (Nbr_Inout), Val, Assoc); - - -- Free wire used for out/inout interface variables. - if Val.Val.Kind = Value_Wire then - Phi_Discard_Wires (Val.Val.W, No_Wire_Id); - Free_Wire (Val.Val.W); - end if; - end if; - - Next_Association_Interface (Assoc, Assoc_Inter); - end loop; - pragma Assert (Nbr_Inout = Infos'Last); - end Synth_Subprogram_Back_Association; - - function Build_Control_Signal (Syn_Inst : Synth_Instance_Acc; - W : Width; - Loc : Source.Syn_Src) return Net - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Res : Net; - begin - Res := Build_Signal (Ctxt, New_Internal_Name (Ctxt), W); - Set_Location (Res, Loc); - return Res; - end Build_Control_Signal; - - function Synth_Dynamic_Subprogram_Call (Syn_Inst : Synth_Instance_Acc; - Sub_Inst : Synth_Instance_Acc; - Call : Node; - Init : Association_Iterator_Init; - Infos : Target_Info_Array) - return Valtyp - is - Imp : constant Node := Get_Implementation (Call); - Is_Func : constant Boolean := Is_Function_Declaration (Imp); - Bod : constant Node := Vhdl.Sem_Inst.Get_Subprogram_Body_Origin (Imp); - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Res : Valtyp; - C : Seq_Context (Mode_Dynamic); - Wire_Mark : Wire_Id; - Subprg_Phi : Phi_Type; - begin - Mark (Wire_Mark); - C := (Mode => Mode_Dynamic, - Inst => Sub_Inst, - Cur_Loop => null, - W_En => No_Wire_Id, - W_Ret => No_Wire_Id, - W_Val => No_Wire_Id, - Ret_Init => No_Net, - Ret_Value => No_Valtyp, - Ret_Typ => null, - Nbr_Ret => 0); - - C.W_En := Alloc_Wire (Wire_Variable, (Imp, Bit_Type)); - C.W_Ret := Alloc_Wire (Wire_Variable, (Imp, Bit_Type)); - - if Is_Func then - C.W_Val := Alloc_Wire (Wire_Variable, (Imp, null)); - end if; - - -- Create a phi so that all assignments are gathered. - Push_Phi; - - Synth_Subprogram_Association_Wires (Sub_Inst, Init); - - if Is_Func then - -- Set a default value for the return. - C.Ret_Typ := Get_Subtype_Object (Syn_Inst, Get_Return_Type (Imp)); - - Set_Wire_Gate (C.W_Val, - Build_Control_Signal (Sub_Inst, C.Ret_Typ.W, Imp)); - C.Ret_Init := Build_Const_X (Ctxt, C.Ret_Typ.W); - Phi_Assign_Net (Ctxt, C.W_Val, C.Ret_Init, 0); - end if; - - Set_Wire_Gate - (C.W_En, Build_Control_Signal (Sub_Inst, 1, Imp)); - Phi_Assign_Static (C.W_En, Bit1); - - Set_Wire_Gate - (C.W_Ret, Build_Control_Signal (Sub_Inst, 1, Imp)); - Phi_Assign_Static (C.W_Ret, Bit1); - - Decls.Synth_Declarations (C.Inst, Get_Declaration_Chain (Bod), True); - if not Is_Error (C.Inst) then - Synth_Sequential_Statements (C, Get_Sequential_Statement_Chain (Bod)); - end if; - - if Is_Error (C.Inst) then - Res := No_Valtyp; - else - if Is_Func then - if C.Nbr_Ret = 0 then - raise Internal_Error; - elsif C.Nbr_Ret = 1 and then Is_Static (C.Ret_Value.Val) then - Res := C.Ret_Value; - else - Res := Create_Value_Net - (Get_Current_Value (Ctxt, C.W_Val), C.Ret_Value.Typ); - end if; - else - Res := No_Valtyp; - Synth_Subprogram_Back_Association (C.Inst, Syn_Inst, Init, Infos); - end if; - end if; - - Pop_Phi (Subprg_Phi); - - Decls.Finalize_Declarations (C.Inst, Get_Declaration_Chain (Bod), True); - pragma Unreferenced (Infos); - - -- Propagate assignments. - -- Wires that have been created for this subprogram will be destroyed. - -- But assignment for outer wires (passed through parameters) have - -- to be kept. We cannot merge phi because this won't be allowed for - -- local wires. - Propagate_Phi_Until_Mark (Ctxt, Subprg_Phi, Wire_Mark); - - -- Free wires. - -- These wires are currently unassigned because they were created - -- within the Phi. - Free_Wire (C.W_En); - Free_Wire (C.W_Ret); - if Is_Func then - Free_Wire (C.W_Val); - end if; - - Release (Wire_Mark); - - return Res; - end Synth_Dynamic_Subprogram_Call; - - function Synth_Static_Subprogram_Call (Syn_Inst : Synth_Instance_Acc; - Sub_Inst : Synth_Instance_Acc; - Call : Node; - Bod : Node; - Init : Association_Iterator_Init; - Infos : Target_Info_Array) - return Valtyp - is - Imp : constant Node := Get_Implementation (Call); - Is_Func : constant Boolean := Is_Function_Declaration (Imp); - Res : Valtyp; - C : Seq_Context (Mode_Static); - begin - C := (Mode_Static, - Inst => Sub_Inst, - Cur_Loop => null, - S_En => True, - Ret_Value => No_Valtyp, - Ret_Typ => null, - Nbr_Ret => 0); - - if Is_Func then - -- Set a default value for the return. - C.Ret_Typ := Get_Subtype_Object (Syn_Inst, Get_Return_Type (Imp)); - end if; - - Synth_Declarations (C.Inst, Get_Declaration_Chain (Bod), True); - - if not Is_Error (C.Inst) then - Synth_Sequential_Statements (C, Get_Sequential_Statement_Chain (Bod)); - end if; - - if Is_Error (C.Inst) then - Res := No_Valtyp; - else - if Is_Func then - if C.Nbr_Ret = 0 then - Error_Msg_Synth - (+Call, "function call completed without a return statement"); - Res := No_Valtyp; - else - pragma Assert (C.Nbr_Ret = 1); - pragma Assert (Is_Static (C.Ret_Value.Val)); - Res := C.Ret_Value; - end if; - else - Res := No_Valtyp; - Synth_Subprogram_Back_Association (C.Inst, Syn_Inst, Init, Infos); - end if; - end if; - - Decls.Finalize_Declarations (C.Inst, Get_Declaration_Chain (Bod), True); - pragma Unreferenced (Infos); - - return Res; - end Synth_Static_Subprogram_Call; - - function Synth_Subprogram_Call (Syn_Inst : Synth_Instance_Acc; - Call : Node; - Init : Association_Iterator_Init) - return Valtyp - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Imp : constant Node := Get_Implementation (Call); - Is_Func : constant Boolean := Is_Function_Declaration (Imp); - Bod : constant Node := Vhdl.Sem_Inst.Get_Subprogram_Body_Origin (Imp); - Nbr_Inout : constant Natural := Count_Associations (Init); - Infos : Target_Info_Array (1 .. Nbr_Inout); - Area_Mark : Areapools.Mark_Type; - Res : Valtyp; - Sub_Inst : Synth_Instance_Acc; - Up_Inst : Synth_Instance_Acc; - begin - Areapools.Mark (Area_Mark, Instance_Pool.all); - - Up_Inst := Get_Instance_By_Scope (Syn_Inst, Get_Parent_Scope (Imp)); - Sub_Inst := Make_Instance (Up_Inst, Bod, New_Internal_Name (Ctxt)); - Set_Instance_Base (Sub_Inst, Syn_Inst); - - Synth_Subprogram_Association (Sub_Inst, Syn_Inst, Init, Infos); - - if Is_Error (Sub_Inst) then - Res := No_Valtyp; - else - if not Is_Func then - if Get_Purity_State (Imp) /= Pure then - Set_Instance_Const (Sub_Inst, False); - end if; - end if; - - if Get_Instance_Const (Sub_Inst) then - Res := Synth_Static_Subprogram_Call - (Syn_Inst, Sub_Inst, Call, Bod, Init, Infos); - else - Res := Synth_Dynamic_Subprogram_Call - (Syn_Inst, Sub_Inst, Call, Init, Infos); - end if; - end if; - - -- Propagate error. - if Is_Error (Sub_Inst) then - Set_Error (Syn_Inst); - end if; - - if Debugger.Flag_Need_Debug then - Debugger.Debug_Leave (Sub_Inst); - end if; - - Free_Instance (Sub_Inst); - Areapools.Release (Area_Mark, Instance_Pool.all); - - return Res; - end Synth_Subprogram_Call; - - function Synth_Subprogram_Call - (Syn_Inst : Synth_Instance_Acc; Call : Node) return Valtyp - is - Imp : constant Node := Get_Implementation (Call); - Assoc_Chain : constant Node := Get_Parameter_Association_Chain (Call); - Inter_Chain : constant Node := Get_Interface_Declaration_Chain (Imp); - Init : Association_Iterator_Init; - begin - Init := Association_Iterator_Build (Inter_Chain, Assoc_Chain); - return Synth_Subprogram_Call (Syn_Inst, Call, Init); - end Synth_Subprogram_Call; - - function Synth_User_Operator (Syn_Inst : Synth_Instance_Acc; - Left_Expr : Node; - Right_Expr : Node; - Expr : Node) return Valtyp - is - Imp : constant Node := Get_Implementation (Expr); - Inter_Chain : constant Node := Get_Interface_Declaration_Chain (Imp); - Init : Association_Iterator_Init; - begin - Init := Association_Iterator_Build (Inter_Chain, Left_Expr, Right_Expr); - return Synth_Subprogram_Call (Syn_Inst, Expr, Init); - end Synth_User_Operator; - - procedure Synth_Implicit_Procedure_Call - (Syn_Inst : Synth_Instance_Acc; Call : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Imp : constant Node := Get_Implementation (Call); - Assoc_Chain : constant Node := Get_Parameter_Association_Chain (Call); - Inter_Chain : constant Node := Get_Interface_Declaration_Chain (Imp); - Init : constant Association_Iterator_Init := - Association_Iterator_Build (Inter_Chain, Assoc_Chain); - Nbr_Inout : constant Natural := Count_Associations (Init); - Infos : Target_Info_Array (1 .. Nbr_Inout); - Area_Mark : Areapools.Mark_Type; - Sub_Inst : Synth_Instance_Acc; - begin - Areapools.Mark (Area_Mark, Instance_Pool.all); - Sub_Inst := Make_Instance (Syn_Inst, Imp, New_Internal_Name (Ctxt)); - - Synth_Subprogram_Association (Sub_Inst, Syn_Inst, Init, Infos); - - Synth.Vhdl_Static_Proc.Synth_Static_Procedure (Sub_Inst, Imp, Call); - - Synth_Subprogram_Back_Association (Sub_Inst, Syn_Inst, Init, Infos); - - Free_Instance (Sub_Inst); - Areapools.Release (Area_Mark, Instance_Pool.all); - end Synth_Implicit_Procedure_Call; - - procedure Synth_Procedure_Call - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Call : constant Node := Get_Procedure_Call (Stmt); - Imp : constant Node := Get_Implementation (Call); - Res : Valtyp; - begin - case Get_Implicit_Definition (Imp) is - when Iir_Predefined_None => - if Get_Foreign_Flag (Imp) then - Error_Msg_Synth - (+Stmt, "call to foreign %n is not supported", +Imp); - else - Res := Synth_Subprogram_Call (Syn_Inst, Call); - pragma Assert (Res = No_Valtyp); - end if; - when others => - Synth_Implicit_Procedure_Call (Syn_Inst, Call); - end case; - end Synth_Procedure_Call; - - procedure Update_Index (Rng : Discrete_Range_Type; V : in out Valtyp) - is - T : Int64; - begin - T := Read_Discrete (V); - case Rng.Dir is - when Dir_To => - T := T + 1; - when Dir_Downto => - T := T - 1; - end case; - Write_Discrete (V, T); - end Update_Index; - - -- Return True iff WID is a static wire and its value is V. - function Is_Static_Bit (Wid : Wire_Id; V : Ghdl_U8) return Boolean - is - M : Memtyp; - begin - if not Is_Static_Wire (Wid) then - return False; - end if; - M := Get_Static_Wire (Wid); - return Read_U8 (M) = V; - end Is_Static_Bit; - - function Is_Static_Bit0 (Wid : Wire_Id) return Boolean is - begin - return Is_Static_Bit (Wid, 0); - end Is_Static_Bit0; - - function Is_Static_Bit1 (Wid : Wire_Id) return Boolean is - begin - return Is_Static_Bit (Wid, 1); - end Is_Static_Bit1; - - pragma Inline (Is_Static_Bit0); - pragma Inline (Is_Static_Bit1); - - procedure Loop_Control_Init (C : Seq_Context; Stmt : Node) - is - Lc : constant Loop_Context_Acc := C.Cur_Loop; - begin - -- We might create new wires that will be destroy at the end of the - -- loop. Use mark and sweep to control their lifetime. - Mark (C.Cur_Loop.Wire_Mark); - - if Lc.Prev_Loop /= null and then Lc.Prev_Loop.Need_Quit then - -- An exit or next statement that targets an outer loop may suspend - -- the execution of this loop. - Lc.W_Quit := Alloc_Wire (Wire_Variable, (Lc.Loop_Stmt, Bit_Type)); - Set_Wire_Gate (Lc.W_Quit, Build_Control_Signal (C.Inst, 1, Stmt)); - Phi_Assign_Static (Lc.W_Quit, Bit1); - end if; - - if Get_Exit_Flag (Stmt) or else Get_Next_Flag (Stmt) then - -- There is an exit or next statement that target this loop. - -- We need to save W_En, as if the execution is suspended due to - -- exit or next, it will resume at the end of the loop. - if Is_Static_Wire (C.W_En) then - pragma Assert (Is_Static_Bit1 (C.W_En)); - Lc.Saved_En := No_Net; - else - Lc.Saved_En := Get_Current_Value (null, C.W_En); - end if; - -- Subloops may be suspended if there is an exit or a next statement - -- for this loop within subloops. - Lc.Need_Quit := True; - end if; - - if Get_Exit_Flag (Stmt) then - -- There is an exit statement for this loop. Create the wire. - Lc.W_Exit := Alloc_Wire (Wire_Variable, (Lc.Loop_Stmt, Bit_Type)); - Set_Wire_Gate (Lc.W_Exit, Build_Control_Signal (C.Inst, 1, Stmt)); - Phi_Assign_Static (Lc.W_Exit, Bit1); - end if; - end Loop_Control_Init; - - procedure Loop_Control_And_Start (Is_Net : out Boolean; - S : out Boolean; - N : out Net; - En : Net) is - begin - if En = No_Net then - Is_Net := False; - N := No_Net; - S := True; - else - Is_Net := True; - N := En; - S := True; - end if; - end Loop_Control_And_Start; - - procedure Loop_Control_And (C : Seq_Context; - Is_Net : in out Boolean; - S : in out Boolean; - N : in out Net; - R : Wire_Id) - is - Res : Net; - begin - if R = No_Wire_Id or else Is_Static_Bit1 (R) then - -- No change. - return; - end if; - - if Is_Static_Bit0 (R) then - -- Stays 0. - Is_Net := False; - S := False; - N := No_Net; - return; - end if; - - if not Is_Net and then not S then - -- Was 0, remains 0. - return; - end if; - - pragma Assert (Is_Net or else S); - - -- Optimize common cases. - Res := Get_Current_Value (null, R); - - if Is_Net then - N := Build_Dyadic (Get_Build (C.Inst), Id_And, N, Res); - Set_Location (N, C.Cur_Loop.Loop_Stmt); - else - N := Res; - end if; - - Is_Net := True; - end Loop_Control_And; - - procedure Loop_Control_And_Assign (C : Seq_Context; - Is_Net : Boolean; - S : Boolean; - N : Net; - W : Wire_Id) is - begin - if Is_Net then - Phi_Assign_Net (Get_Build (C.Inst), W, N, 0); - else - if S then - Phi_Assign_Static (W, Bit1); - else - Phi_Assign_Static (W, Bit0); - end if; - end if; - end Loop_Control_And_Assign; - - procedure Loop_Control_Update (C : Seq_Context) - is - Lc : constant Loop_Context_Acc := C.Cur_Loop; - N : Net; - S : Boolean; - Is_Net : Boolean; - begin - if not Lc.Need_Quit then - -- No next/exit statement for this loop. So no control. - return; - end if; - - -- Execution continue iff: - -- 1. Loop was enabled (Lc.Saved_En) - Loop_Control_And_Start (Is_Net, S, N, Lc.Saved_En); - - -- 2. No return (C.W_Ret) - Loop_Control_And (C, Is_Net, S, N, C.W_Ret); - - -- 3. No exit. - Loop_Control_And (C, Is_Net, S, N, Lc.W_Exit); - - -- 4. No quit. - Loop_Control_And (C, Is_Net, S, N, Lc.W_Quit); - - Loop_Control_And_Assign (C, Is_Net, S, N, C.W_En); - end Loop_Control_Update; - - procedure Loop_Control_Finish (C : Seq_Context) - is - Lc : constant Loop_Context_Acc := C.Cur_Loop; - N : Net; - S : Boolean; - Is_Net : Boolean; - begin - -- Execution continue after this loop iff: - -- 1. Loop was enabled (Lc.Saved_En) - Loop_Control_And_Start (Is_Net, S, N, Lc.Saved_En); - - -- 2. No return (C.W_Ret) - Loop_Control_And (C, Is_Net, S, N, C.W_Ret); - - -- 3. No quit (C.W_Quit) - Loop_Control_And (C, Is_Net, S, N, Lc.W_Quit); - - Phi_Discard_Wires (Lc.W_Quit, Lc.W_Exit); - - if Lc.W_Quit /= No_Wire_Id then - Free_Wire (Lc.W_Quit); - end if; - - if Lc.W_Exit /= No_Wire_Id then - Free_Wire (Lc.W_Exit); - end if; - - Release (C.Cur_Loop.Wire_Mark); - - Loop_Control_And_Assign (C, Is_Net, S, N, C.W_En); - end Loop_Control_Finish; - - procedure Synth_Dynamic_Exit_Next_Statement - (C : in out Seq_Context; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (C.Inst); - Cond : constant Node := Get_Condition (Stmt); - Is_Exit : constant Boolean := Get_Kind (Stmt) = Iir_Kind_Exit_Statement; - Static_Cond : Boolean; - Loop_Label : Node; - Lc : Loop_Context_Acc; - Cond_Val : Valtyp; - Phi_True : Phi_Type; - Phi_False : Phi_Type; - begin - if Cond /= Null_Node then - Cond_Val := Synth_Expression (C.Inst, Cond); - Static_Cond := Is_Static_Val (Cond_Val.Val); - if Static_Cond then - if Get_Static_Discrete (Cond_Val) = 0 then - -- Not executed. - return; - end if; - else - -- Create a branch for the True case. - Push_Phi; - end if; - end if; - - -- Execution is suspended for the current sequence of statements. - Phi_Assign_Static (C.W_En, Bit0); - - Lc := C.Cur_Loop; - - -- Compute the loop statement indicated by the exit/next statement. - Loop_Label := Get_Loop_Label (Stmt); - if Loop_Label = Null_Node then - Loop_Label := Lc.Loop_Stmt; - else - Loop_Label := Get_Named_Entity (Loop_Label); - end if; - - -- Update the W_Exit and W_Quit flags for the loops. All the loops - -- until the label are canceled. - loop - if Lc.Loop_Stmt = Loop_Label then - -- Final loop. - if Is_Exit then - Phi_Assign_Static (Lc.W_Exit, Bit0); - end if; - exit; - else - Phi_Assign_Static (Lc.W_Quit, Bit0); - end if; - Lc := Lc.Prev_Loop; - end loop; - - if Cond /= Null_Node and not Static_Cond then - Pop_Phi (Phi_True); - - -- If the condition is false, do nothing. - Push_Phi; - Pop_Phi (Phi_False); - - Merge_Phis (Ctxt, Get_Net (Ctxt, Cond_Val), Phi_True, Phi_False, - Get_Location (Stmt)); - end if; - end Synth_Dynamic_Exit_Next_Statement; - - procedure Synth_Static_Exit_Next_Statement - (C : in out Seq_Context; Stmt : Node) - is - Cond : constant Node := Get_Condition (Stmt); - Is_Exit : constant Boolean := Get_Kind (Stmt) = Iir_Kind_Exit_Statement; - Loop_Label : Node; - Lc : Loop_Context_Acc; - Cond_Val : Valtyp; - begin - if Cond /= Null_Node then - Cond_Val := Synth_Expression (C.Inst, Cond); - if Cond_Val = No_Valtyp then - Set_Error (C.Inst); - return; - end if; - pragma Assert (Is_Static_Val (Cond_Val.Val)); - if Get_Static_Discrete (Cond_Val) = 0 then - -- Not executed. - return; - end if; - end if; - - -- Execution is suspended. - C.S_En := False; - - Lc := C.Cur_Loop; - - Loop_Label := Get_Loop_Label (Stmt); - if Loop_Label = Null_Node then - Loop_Label := Lc.Loop_Stmt; - else - Loop_Label := Get_Named_Entity (Loop_Label); - end if; - - loop - if Lc.Loop_Stmt = Loop_Label then - if Is_Exit then - Lc.S_Exit := True; - end if; - exit; - else - Lc.S_Quit := True; - end if; - Lc := Lc.Prev_Loop; - end loop; - end Synth_Static_Exit_Next_Statement; - - procedure Init_For_Loop_Statement (C : in out Seq_Context; - Stmt : Node; - Val : out Valtyp) - is - Iterator : constant Node := Get_Parameter_Specification (Stmt); - It_Type : constant Node := Get_Declaration_Type (Iterator); - It_Rng : Type_Acc; - begin - if It_Type /= Null_Node then - Synth_Subtype_Indication (C.Inst, It_Type); - end if; - - -- Initial value. - It_Rng := Get_Subtype_Object (C.Inst, Get_Type (Iterator)); - Val := Create_Value_Discrete (It_Rng.Drange.Left, It_Rng); - Create_Object (C.Inst, Iterator, Val); - end Init_For_Loop_Statement; - - procedure Finish_For_Loop_Statement (C : in out Seq_Context; - Stmt : Node) - is - Iterator : constant Node := Get_Parameter_Specification (Stmt); - It_Type : constant Node := Get_Declaration_Type (Iterator); - begin - Destroy_Object (C.Inst, Iterator); - if It_Type /= Null_Node then - Destroy_Object (C.Inst, It_Type); - end if; - end Finish_For_Loop_Statement; - - procedure Synth_Dynamic_For_Loop_Statement - (C : in out Seq_Context; Stmt : Node) - is - Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); - Val : Valtyp; - Lc : aliased Loop_Context (Mode_Dynamic); - begin - Lc := (Mode => Mode_Dynamic, - Prev_Loop => C.Cur_Loop, - Loop_Stmt => Stmt, - Need_Quit => False, - Saved_En => No_Net, - W_Exit => No_Wire_Id, - W_Quit => No_Wire_Id, - Wire_Mark => No_Wire_Id); - C.Cur_Loop := Lc'Unrestricted_Access; - - Loop_Control_Init (C, Stmt); - - Init_For_Loop_Statement (C, Stmt, Val); - - while In_Range (Val.Typ.Drange, Read_Discrete (Val)) loop - Synth_Sequential_Statements (C, Stmts); - - Update_Index (Val.Typ.Drange, Val); - Loop_Control_Update (C); - - -- Constant exit. - exit when Is_Static_Bit0 (C.W_En); - - -- FIXME: dynamic exits. - end loop; - Loop_Control_Finish (C); - - Finish_For_Loop_Statement (C, Stmt); - - C.Cur_Loop := Lc.Prev_Loop; - end Synth_Dynamic_For_Loop_Statement; - - procedure Synth_Static_For_Loop_Statement - (C : in out Seq_Context; Stmt : Node) - is - Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); - Val : Valtyp; - Lc : aliased Loop_Context (Mode_Static); - begin - Lc := (Mode_Static, - Prev_Loop => C.Cur_Loop, - Loop_Stmt => Stmt, - S_Exit => False, - S_Quit => False); - C.Cur_Loop := Lc'Unrestricted_Access; - - Init_For_Loop_Statement (C, Stmt, Val); - - while In_Range (Val.Typ.Drange, Read_Discrete (Val)) loop - Synth_Sequential_Statements (C, Stmts); - C.S_En := True; - - Update_Index (Val.Typ.Drange, Val); - - exit when Lc.S_Exit or Lc.S_Quit or C.Nbr_Ret > 0; - end loop; - - Finish_For_Loop_Statement (C, Stmt); - - C.Cur_Loop := Lc.Prev_Loop; - end Synth_Static_For_Loop_Statement; - - procedure Synth_Dynamic_While_Loop_Statement - (C : in out Seq_Context; Stmt : Node) - is - Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); - Cond : constant Node := Get_Condition (Stmt); - Val : Valtyp; - Lc : aliased Loop_Context (Mode_Dynamic); - Iter_Nbr : Natural; - begin - Lc := (Mode => Mode_Dynamic, - Prev_Loop => C.Cur_Loop, - Loop_Stmt => Stmt, - Need_Quit => False, - Saved_En => No_Net, - W_Exit => No_Wire_Id, - W_Quit => No_Wire_Id, - Wire_Mark => No_Wire_Id); - C.Cur_Loop := Lc'Unrestricted_Access; - - Iter_Nbr := 0; - - Loop_Control_Init (C, Stmt); - - loop - if Cond /= Null_Node then - Val := Synth_Expression_With_Type (C.Inst, Cond, Boolean_Type); - if not Is_Static (Val.Val) then - Error_Msg_Synth (+Cond, "loop condition must be static"); - exit; - end if; - exit when Read_Discrete (Val) = 0; - end if; - - Synth_Sequential_Statements (C, Stmts); - - Loop_Control_Update (C); - - -- Exit from the loop if W_Exit/W_Ret/W_Quit = 0 - exit when Lc.W_Exit /= No_Wire_Id and then Is_Static_Bit0 (Lc.W_Exit); - exit when C.W_Ret /= No_Wire_Id and then Is_Static_Bit0 (C.W_Ret); - exit when Lc.W_Quit /= No_Wire_Id and then Is_Static_Bit0 (Lc.W_Quit); - - Iter_Nbr := Iter_Nbr + 1; - if Iter_Nbr > Flags.Flag_Max_Loop and Flags.Flag_Max_Loop /= 0 then - Error_Msg_Synth - (+Stmt, "maximum number of iterations (%v) reached", - +Uns32 (Flags.Flag_Max_Loop)); - exit; - end if; - end loop; - Loop_Control_Finish (C); - - C.Cur_Loop := Lc.Prev_Loop; - end Synth_Dynamic_While_Loop_Statement; - - procedure Synth_Static_While_Loop_Statement - (C : in out Seq_Context; Stmt : Node) - is - Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); - Cond : constant Node := Get_Condition (Stmt); - Val : Valtyp; - Lc : aliased Loop_Context (Mode_Static); - begin - Lc := (Mode => Mode_Static, - Prev_Loop => C.Cur_Loop, - Loop_Stmt => Stmt, - S_Exit => False, - S_Quit => False); - C.Cur_Loop := Lc'Unrestricted_Access; - - loop - if Cond /= Null_Node then - Val := Synth_Expression_With_Type (C.Inst, Cond, Boolean_Type); - pragma Assert (Is_Static (Val.Val)); - exit when Read_Discrete (Val) = 0; - end if; - - Synth_Sequential_Statements (C, Stmts); - C.S_En := True; - - -- Exit from the loop if S_Exit/S_Quit - exit when Lc.S_Exit or Lc.S_Quit or C.Nbr_Ret > 0; - end loop; - - C.Cur_Loop := Lc.Prev_Loop; - end Synth_Static_While_Loop_Statement; - - procedure Synth_Return_Statement (C : in out Seq_Context; Stmt : Node) - is - Is_Dyn : constant Boolean := not Get_Instance_Const (C.Inst); - Ctxt : constant Context_Acc := Get_Build (C.Inst); - Val : Valtyp; - Expr : constant Node := Get_Expression (Stmt); - begin - if Expr /= Null_Node then - -- Return in function. - Val := Synth_Expression_With_Type (C.Inst, Expr, C.Ret_Typ); - if Val = No_Valtyp then - Set_Error (C.Inst); - return; - end if; - - Val := Synth_Subtype_Conversion (Ctxt, Val, C.Ret_Typ, True, Stmt); - - if C.Nbr_Ret = 0 then - C.Ret_Value := Val; - if not Is_Bounded_Type (C.Ret_Typ) then - -- The function was declared with an unconstrained return type. - -- Now that a value has been returned, we know the subtype of - -- the returned values. So adjust it. - -- All the returned values must have the same length. - C.Ret_Typ := Val.Typ; - if Is_Dyn then - Set_Width (Get_Wire_Gate (C.W_Val), C.Ret_Typ.W); - Set_Width (C.Ret_Init, C.Ret_Typ.W); - end if; - end if; - end if; - if Is_Dyn then - Phi_Assign_Net (Ctxt, C.W_Val, Get_Net (Ctxt, Val), 0); - end if; - end if; - - if Is_Dyn then - -- The subprogram has returned. Do not execute further statements. - Phi_Assign_Static (C.W_En, Bit0); - - if C.W_Ret /= No_Wire_Id then - Phi_Assign_Static (C.W_Ret, Bit0); - end if; - end if; - - C.Nbr_Ret := C.Nbr_Ret + 1; - end Synth_Return_Statement; - - procedure Synth_Static_Report (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - use Simple_IO; - - Is_Report : constant Boolean := - Get_Kind (Stmt) = Iir_Kind_Report_Statement; - Rep_Expr : constant Node := Get_Report_Expression (Stmt); - Sev_Expr : constant Node := Get_Severity_Expression (Stmt); - Rep : Valtyp; - Sev : Valtyp; - Sev_V : Natural; - begin - if Rep_Expr /= Null_Node then - Rep := Synth_Expression_With_Basetype (Syn_Inst, Rep_Expr); - if Rep = No_Valtyp then - Set_Error (Syn_Inst); - return; - end if; - Strip_Const (Rep); - end if; - if Sev_Expr /= Null_Node then - Sev := Synth_Expression (Syn_Inst, Sev_Expr); - if Sev = No_Valtyp then - Set_Error (Syn_Inst); - return; - end if; - Strip_Const (Sev); - end if; - - Put_Err (Disp_Location (Stmt)); - Put_Err (":("); - if Is_Report then - Put_Err ("report"); - else - Put_Err ("assertion"); - end if; - Put_Err (' '); - if Sev = No_Valtyp then - if Is_Report then - Sev_V := 0; - else - Sev_V := 2; - end if; - else - Sev_V := Natural (Read_Discrete (Sev)); - end if; - case Sev_V is - when Note_Severity => - Put_Err ("note"); - when Warning_Severity => - Put_Err ("warning"); - when Error_Severity => - Put_Err ("error"); - when Failure_Severity => - Put_Err ("failure"); - when others => - Put_Err ("??"); - end case; - Put_Err ("): "); - - if Rep = No_Valtyp then - Put_Line_Err ("assertion failure"); - else - Put_Line_Err (Value_To_String (Rep)); - end if; - - if Sev_V >= Flags.Severity_Level then - Error_Msg_Synth (+Stmt, "error due to assertion failure"); - end if; - end Synth_Static_Report; - - procedure Synth_Static_Report_Statement (C : Seq_Context; Stmt : Node) is - begin - Synth_Static_Report (C.Inst, Stmt); - end Synth_Static_Report_Statement; - - procedure Synth_Static_Assertion_Statement (C : Seq_Context; Stmt : Node) - is - Cond : Valtyp; - begin - Cond := Synth_Expression (C.Inst, Get_Assertion_Condition (Stmt)); - if Cond = No_Valtyp then - Set_Error (C.Inst); - return; - end if; - pragma Assert (Is_Static (Cond.Val)); - Strip_Const (Cond); - if Read_Discrete (Cond) = 1 then - return; - end if; - Synth_Static_Report (C.Inst, Stmt); - end Synth_Static_Assertion_Statement; - - procedure Synth_Dynamic_Assertion_Statement (C : Seq_Context; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (C.Inst); - Loc : constant Location_Type := Get_Location (Stmt); - Cond : Valtyp; - N : Net; - En : Net; - Inst : Instance; - begin - if not Flags.Flag_Formal then - return; - end if; - - Cond := Synth_Expression (C.Inst, Get_Assertion_Condition (Stmt)); - if Cond = No_Valtyp then - Set_Error (C.Inst); - return; - end if; - N := Get_Net (Ctxt, Cond); - En := Phi_Enable (Ctxt, (Stmt, Bit_Type), Bit0, Bit1, - Get_Location (Stmt)); - if En /= No_Net then - -- Build: En -> Cond - N := Build2_Imp (Ctxt, En, N, Loc); - end if; - Inst := Build_Assert (Ctxt, Synth_Label (C.Inst, Stmt), N); - Set_Location (Inst, Loc); - end Synth_Dynamic_Assertion_Statement; - - procedure Synth_Sequential_Statements - (C : in out Seq_Context; Stmts : Node) - is - Is_Dyn : constant Boolean := not Get_Instance_Const (C.Inst); - Ctxt : constant Context_Acc := Get_Build (C.Inst); - Stmt : Node; - Phi_T, Phi_F : Phi_Type; - Has_Phi : Boolean; - begin - Stmt := Stmts; - while Is_Valid (Stmt) loop - if Is_Dyn then - pragma Assert (not Is_Static_Bit0 (C.W_En)); - Has_Phi := not Is_Static_Bit1 (C.W_En); - if Has_Phi then - Push_Phi; - end if; - end if; - - if Flags.Flag_Trace_Statements then - declare - Name : Name_Id; - Line : Natural; - Col : Natural; - begin - Files_Map.Location_To_Position - (Get_Location (Stmt), Name, Line, Col); - Simple_IO.Put_Line ("Execute statement at " - & Name_Table.Image (Name) - & Natural'Image (Line)); - end; - end if; - if Synth.Debugger.Flag_Need_Debug then - Synth.Debugger.Debug_Break (C.Inst, Stmt); - end if; - - case Get_Kind (Stmt) is - when Iir_Kind_If_Statement => - Synth_If_Statement (C, Stmt); - when Iir_Kind_Simple_Signal_Assignment_Statement => - Synth_Simple_Signal_Assignment (C.Inst, Stmt); - when Iir_Kind_Conditional_Signal_Assignment_Statement => - Synth_Conditional_Signal_Assignment (C.Inst, Stmt); - when Iir_Kind_Variable_Assignment_Statement => - Synth_Variable_Assignment (C, Stmt); - when Iir_Kind_Conditional_Variable_Assignment_Statement => - Synth_Conditional_Variable_Assignment (C, Stmt); - when Iir_Kind_Case_Statement => - Synth_Case_Statement (C, Stmt); - when Iir_Kind_For_Loop_Statement => - if Is_Dyn then - Synth_Dynamic_For_Loop_Statement (C, Stmt); - else - Synth_Static_For_Loop_Statement (C, Stmt); - end if; - when Iir_Kind_While_Loop_Statement => - if Is_Dyn then - Synth_Dynamic_While_Loop_Statement (C, Stmt); - else - Synth_Static_While_Loop_Statement (C, Stmt); - end if; - when Iir_Kind_Null_Statement => - -- Easy - null; - when Iir_Kind_Return_Statement => - Synth_Return_Statement (C, Stmt); - when Iir_Kind_Procedure_Call_Statement => - Synth_Procedure_Call (C.Inst, Stmt); - when Iir_Kind_Report_Statement => - if not Is_Dyn then - Synth_Static_Report_Statement (C, Stmt); - end if; - when Iir_Kind_Assertion_Statement => - if not Is_Dyn then - Synth_Static_Assertion_Statement (C, Stmt); - else - Synth_Dynamic_Assertion_Statement (C, Stmt); - end if; - when Iir_Kind_Exit_Statement - | Iir_Kind_Next_Statement => - if Is_Dyn then - Synth_Dynamic_Exit_Next_Statement (C, Stmt); - else - Synth_Static_Exit_Next_Statement (C, Stmt); - end if; - when others => - Error_Kind ("synth_sequential_statements", Stmt); - end case; - if Is_Dyn then - if Has_Phi then - Pop_Phi (Phi_T); - Push_Phi; - Pop_Phi (Phi_F); - Merge_Phis (Ctxt, Get_Current_Value (Ctxt, C.W_En), - Phi_T, Phi_F, Get_Location (Stmt)); - end if; - if Is_Static_Bit0 (C.W_En) then - -- Not more execution. - return; - end if; - else - if not C.S_En or C.Nbr_Ret /= 0 then - return; - end if; - end if; - Stmt := Get_Chain (Stmt); - end loop; - end Synth_Sequential_Statements; - - Proc_Pool : aliased Areapools.Areapool; - - -- Synthesis of statements of a non-sensitized process. - procedure Synth_Process_Sequential_Statements - (C : in out Seq_Context; Proc : Node) - is - Ctxt : constant Context_Acc := Get_Build (C.Inst); - Stmt : Node; - Cond : Node; - Cond_Val : Valtyp; - Phi_True : Phi_Type; - Phi_False : Phi_Type; - begin - Stmt := Get_Sequential_Statement_Chain (Proc); - - -- The first statement must be a wait statement. - if Get_Kind (Stmt) /= Iir_Kind_Wait_Statement then - Error_Msg_Synth (+Stmt, "expect wait as the first statement"); - return; - end if; - - -- Handle the condition as an if. - Cond := Get_Condition_Clause (Stmt); - if Cond = Null_Node then - Error_Msg_Synth (+Stmt, "expect wait condition"); - return; - end if; - Cond_Val := Synth_Expression (C.Inst, Cond); - - Push_Phi; - Synth_Sequential_Statements (C, Get_Chain (Stmt)); - Pop_Phi (Phi_True); - Push_Phi; - Pop_Phi (Phi_False); - - Merge_Phis (Ctxt, Get_Net (Ctxt, Cond_Val), Phi_True, Phi_False, - Get_Location (Stmt)); - end Synth_Process_Sequential_Statements; - - procedure Synth_Process_Statement - (Syn_Inst : Synth_Instance_Acc; Proc : Node) - is - use Areapools; - Label : constant Name_Id := Get_Identifier (Proc); - Decls_Chain : constant Node := Get_Declaration_Chain (Proc); - Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - M : Areapools.Mark_Type; - C_Sname : Sname; - C : Seq_Context (Mode_Dynamic); - begin - if Label = Null_Identifier then - C_Sname := New_Internal_Name (Ctxt, Get_Sname (Syn_Inst)); - else - C_Sname := New_Sname_User (Label, Get_Sname (Syn_Inst)); - end if; - C := (Mode => Mode_Dynamic, - Inst => Make_Instance (Syn_Inst, Proc, C_Sname), - Cur_Loop => null, - W_En => Alloc_Wire (Wire_Variable, (Proc, Bit_Type)), - W_Ret => No_Wire_Id, - W_Val => No_Wire_Id, - Ret_Init => No_Net, - Ret_Value => No_Valtyp, - Ret_Typ => null, - Nbr_Ret => 0); - - Mark (M, Proc_Pool); - Instance_Pool := Proc_Pool'Access; - - Push_Phi; - - Synth_Declarations (C.Inst, Decls_Chain); - - Set_Wire_Gate (C.W_En, Build_Control_Signal (Syn_Inst, 1, Proc)); - Phi_Assign_Static (C.W_En, Bit1); - - if not Is_Error (C.Inst) then - case Iir_Kinds_Process_Statement (Get_Kind (Proc)) is - when Iir_Kind_Sensitized_Process_Statement => - Synth_Sequential_Statements - (C, Get_Sequential_Statement_Chain (Proc)); - -- FIXME: check sensitivity list. - when Iir_Kind_Process_Statement => - Synth_Process_Sequential_Statements (C, Proc); - end case; - end if; - - Pop_And_Merge_Phi (Ctxt, Get_Location (Proc)); - - Finalize_Declarations (C.Inst, Decls_Chain); - - Free_Instance (C.Inst); - Release (M, Proc_Pool); - Instance_Pool := Prev_Instance_Pool; - - Finalize_Assignment (Ctxt, C.W_En); - Free_Wire (C.W_En); - end Synth_Process_Statement; - - function Synth_User_Function_Call - (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp is - begin - -- Is it a call to an ieee function ? - declare - Imp : constant Node := Get_Implementation (Expr); - Pkg : constant Node := Get_Parent (Imp); - Unit : Node; - Lib : Node; - begin - if Get_Kind (Pkg) = Iir_Kind_Package_Declaration - and then not Is_Uninstantiated_Package (Pkg) - then - Unit := Get_Parent (Pkg); - if Get_Kind (Unit) = Iir_Kind_Design_Unit then - Lib := Get_Library (Get_Design_File (Unit)); - if Get_Identifier (Lib) = Std_Names.Name_Ieee then - Error_Msg_Synth - (+Expr, "unhandled call to ieee function %i", +Imp); - Set_Error (Syn_Inst); - return No_Valtyp; - end if; - end if; - end if; - end; - - return Synth_Subprogram_Call (Syn_Inst, Expr); - end Synth_User_Function_Call; - - procedure Synth_Concurrent_Assertion_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Cond : constant Node := Get_Assertion_Condition (Stmt); - Val : Valtyp; - Inst : Instance; - begin - Val := Synth_Expression (Syn_Inst, Cond); - if Val = No_Valtyp then - Set_Error (Syn_Inst); - return; - end if; - if Is_Static (Val.Val) then - if Read_Discrete (Val) /= 1 then - Synth_Static_Report (Syn_Inst, Stmt); - end if; - return; - end if; - - if not Flags.Flag_Formal then - -- Ignore the net. - return; - end if; - - Inst := Build_Assert - (Ctxt, Synth_Label (Syn_Inst, Stmt), Get_Net (Ctxt, Val)); - Set_Location (Inst, Get_Location (Stmt)); - end Synth_Concurrent_Assertion_Statement; - - procedure Synth_Block_Statement (Syn_Inst : Synth_Instance_Acc; Blk : Node) - is - use Areapools; - Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; - Blk_Inst : Synth_Instance_Acc; - Blk_Sname : Sname; - M : Areapools.Mark_Type; - begin - -- No support for guard or header. - if Get_Block_Header (Blk) /= Null_Node - or else Get_Guard_Decl (Blk) /= Null_Node - then - raise Internal_Error; - end if; - - Apply_Block_Configuration - (Get_Block_Block_Configuration (Blk), Blk); - - Blk_Sname := New_Sname_User (Get_Identifier (Blk), Get_Sname (Syn_Inst)); - Blk_Inst := Make_Instance (Syn_Inst, Blk, Blk_Sname); - Mark (M, Proc_Pool); - Instance_Pool := Proc_Pool'Access; - - Synth_Declarations (Blk_Inst, Get_Declaration_Chain (Blk)); - Synth_Concurrent_Statements - (Blk_Inst, Get_Concurrent_Statement_Chain (Blk)); - - Synth_Attribute_Values (Blk_Inst, Blk); - - Finalize_Declarations (Blk_Inst, Get_Declaration_Chain (Blk)); - - Free_Instance (Blk_Inst); - Release (M, Proc_Pool); - Instance_Pool := Prev_Instance_Pool; - end Synth_Block_Statement; - - function Synth_Psl_NFA (Syn_Inst : Synth_Instance_Acc; - NFA : PSL.Types.PSL_NFA; - Nbr_States : Int32; - States : Net; - Loc : Source.Syn_Src) return Net - is - use PSL.NFAs; - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - S : NFA_State; - S_Num : Int32; - D_Num : Int32; - I : Net; - Cond : Net; - E : NFA_Edge; - D_Arr : Net_Array_Acc; - Res : Net; - begin - D_Arr := new Net_Array'(0 .. Nbr_States - 1 => No_Net); - - -- For each state: - S := Get_First_State (NFA); - while S /= No_State loop - S_Num := Get_State_Label (S); - I := Build_Extract_Bit (Ctxt, States, Uns32 (S_Num)); - Set_Location (I, Loc); - - -- For each edge: - E := Get_First_Src_Edge (S); - while E /= No_Edge loop - -- Edge condition. - Cond := Build_Dyadic - (Ctxt, Id_And, - I, Synth_PSL_Expression (Syn_Inst, Get_Edge_Expr (E))); - Set_Location (Cond, Loc); - - -- TODO: if EOS is present, then this is a live state. - - -- Reverse order for final concatenation. - D_Num := Nbr_States - 1 - Get_State_Label (Get_Edge_Dest (E)); - if D_Arr (D_Num) /= No_Net then - Cond := Build_Dyadic (Ctxt, Id_Or, D_Arr (D_Num), Cond); - Set_Location (Cond, Loc); - end if; - D_Arr (D_Num) := Cond; - - E := Get_Next_Src_Edge (E); - end loop; - - S := Get_Next_State (S); - end loop; - - if D_Arr (Nbr_States - 1) = No_Net then - D_Arr (Nbr_States - 1) := Build_Const_UB32 (Ctxt, 0, 1); - end if; - - Concat_Array (Ctxt, D_Arr.all, Res); - Free_Net_Array (D_Arr); - - return Res; - end Synth_Psl_NFA; - - procedure Synth_Psl_Dff (Syn_Inst : Synth_Instance_Acc; - Stmt : Node; - Next_States : out Net) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Nbr_States : constant Int32 := Get_PSL_Nbr_States (Stmt); - States : Net; - Init : Net; - Clk : Net; - Clk_Inst : Instance; - begin - -- create init net, clock net - Init := Build_Const_UB32 (Ctxt, 1, Uns32 (Nbr_States)); - Set_Location (Init, Stmt); - Clk := Synth_PSL_Expression (Syn_Inst, Get_PSL_Clock (Stmt)); - - -- Check the clock is an edge and extract it. - Clk_Inst := Get_Net_Parent (Clk); - if Get_Id (Clk_Inst) not in Edge_Module_Id then - Error_Msg_Synth (+Stmt, "clock is not an edge"); - Next_States := No_Net; - return; - end if; - - -- build idff - States := Build_Idff (Ctxt, Clk, No_Net, Init); - Set_Location (States, Stmt); - - -- create update nets - -- For each state: if set, evaluate all outgoing edges. - Next_States := - Synth_Psl_NFA (Syn_Inst, Get_PSL_NFA (Stmt), Nbr_States, States, Stmt); - Connect (Get_Input (Get_Net_Parent (States), 1), Next_States); - end Synth_Psl_Dff; - - function Synth_Psl_Final - (Syn_Inst : Synth_Instance_Acc; Stmt : Node; Next_States : Net) return Net - is - use PSL.Types; - use PSL.NFAs; - NFA : constant PSL_NFA := Get_PSL_NFA (Stmt); - Res : Net; - begin - Res := Build_Extract_Bit - (Get_Build (Syn_Inst), Next_States, - Uns32 (Get_State_Label (Get_Final_State (NFA)))); - Set_Location (Res, Stmt); - return Res; - end Synth_Psl_Final; - - function Synth_Psl_Not_Final - (Syn_Inst : Synth_Instance_Acc; Stmt : Node; Next_States : Net) - return Net - is - Res : Net; - begin - Res := Build_Monadic (Get_Build (Syn_Inst), Id_Not, - Synth_Psl_Final (Syn_Inst, Stmt, Next_States)); - Set_Location (Res, Stmt); - return Res; - end Synth_Psl_Not_Final; - - procedure Synth_Psl_Restrict_Directive - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Next_States : Net; - Res : Net; - Inst : Instance; - begin - if not Flags.Flag_Formal then - return; - end if; - - -- Build assume gate. - -- Note: for synthesis, we assume the next state will be correct. - -- (If we assume on States, then the first cycle is ignored). - Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); - if Next_States /= No_Net then - -- The restriction holds as long as there is a 1 in the NFA state. - Res := Build_Reduce (Ctxt, Id_Red_Or, Next_States); - Set_Location (Res, Stmt); - Inst := Build_Assume (Ctxt, Synth_Label (Syn_Inst, Stmt), Res); - Set_Location (Inst, Get_Location (Stmt)); - end if; - end Synth_Psl_Restrict_Directive; - - procedure Synth_Psl_Cover_Directive - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Next_States : Net; - Res : Net; - Inst : Instance; - begin - if not Flags.Flag_Formal then - return; - end if; - - -- Build cover gate. - -- Note: for synthesis, we assume the next state will be correct. - -- (If we assume on States, then the first cycle is ignored). - Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); - if Next_States /= No_Net then - -- The sequence is covered as soon as the final state is reached. - Res := Synth_Psl_Final (Syn_Inst, Stmt, Next_States); - Inst := Build_Cover - (Get_Build (Syn_Inst), Synth_Label (Syn_Inst, Stmt), Res); - Set_Location (Inst, Get_Location (Stmt)); - end if; - end Synth_Psl_Cover_Directive; - - procedure Synth_Psl_Assume_Directive - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - Next_States : Net; - Inst : Instance; - begin - if not Flags.Flag_Formal then - return; - end if; - - -- Build assume gate. - -- Note: for synthesis, we assume the next state will be correct. - -- (If we assume on States, then the first cycle is ignored). - Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); - if Next_States /= No_Net then - Inst := Build_Assume - (Ctxt, Synth_Label (Syn_Inst, Stmt), - Synth_Psl_Not_Final (Syn_Inst, Stmt, Next_States)); - Set_Location (Inst, Get_Location (Stmt)); - end if; - end Synth_Psl_Assume_Directive; - - procedure Synth_Psl_Assert_Directive - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - use PSL.Types; - use PSL.NFAs; - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - NFA : constant PSL_NFA := Get_PSL_NFA (Stmt); - Active : NFA_State; - Next_States : Net; - Inst : Instance; - Lab : Sname; - begin - if not Flags.Flag_Formal then - return; - end if; - - -- Build assert gate. - -- Note: for synthesis, we assume the next state will be correct. - -- (If we assert on States, then the first cycle is ignored). - Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); - if Next_States = No_Net then - return; - end if; - Lab := Synth_Label (Syn_Inst, Stmt); - - Inst := Build_Assert - (Ctxt, Lab, Synth_Psl_Not_Final (Syn_Inst, Stmt, Next_States)); - Set_Location (Inst, Get_Location (Stmt)); - - -- Also add a cover gate to cover assertion activation. - if Flags.Flag_Assert_Cover then - Active := Get_Active_State (NFA); - if Active /= No_State then - if Lab /= No_Sname then - Lab := New_Sname_User (Std_Names.Name_Cover, Lab); - end if; - Inst := Build_Assert_Cover - (Get_Build (Syn_Inst), Lab, - Build_Extract_Bit (Get_Build (Syn_Inst), Next_States, - Uns32 (Get_State_Label (Active)))); - Set_Location (Inst, Get_Location (Stmt)); - end if; - end if; - end Synth_Psl_Assert_Directive; - - procedure Synth_Generate_Statement_Body - (Syn_Inst : Synth_Instance_Acc; - Bod : Node; - Name : Sname; - Iterator : Node := Null_Node; - Iterator_Val : Valtyp := No_Valtyp) - is - use Areapools; - Decls_Chain : constant Node := Get_Declaration_Chain (Bod); - Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; - Bod_Inst : Synth_Instance_Acc; - M : Areapools.Mark_Type; - begin - Bod_Inst := Make_Instance (Syn_Inst, Bod, Name); - Mark (M, Proc_Pool); - Instance_Pool := Proc_Pool'Access; - - if Iterator /= Null_Node then - -- Add the iterator (for for-generate). - Create_Object (Bod_Inst, Iterator, Iterator_Val); - end if; - - Synth_Declarations (Bod_Inst, Decls_Chain); - - Synth_Concurrent_Statements - (Bod_Inst, Get_Concurrent_Statement_Chain (Bod)); - - Synth_Attribute_Values (Bod_Inst, Bod); - - Finalize_Declarations (Bod_Inst, Decls_Chain); - - Free_Instance (Bod_Inst); - Release (M, Proc_Pool); - Instance_Pool := Prev_Instance_Pool; - end Synth_Generate_Statement_Body; - - procedure Synth_If_Generate_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Gen : Node; - Bod : Node; - Icond : Node; - Cond : Valtyp; - Name : Sname; - begin - Gen := Stmt; - Name := New_Sname_User (Get_Identifier (Stmt), Get_Sname (Syn_Inst)); - loop - Icond := Get_Condition (Gen); - if Icond /= Null_Node then - Cond := Synth_Expression (Syn_Inst, Icond); - Strip_Const (Cond); - else - -- It is the else generate. - Cond := No_Valtyp; - end if; - if Cond = No_Valtyp or else Read_Discrete (Cond) = 1 then - Bod := Get_Generate_Statement_Body (Gen); - Apply_Block_Configuration - (Get_Generate_Block_Configuration (Bod), Bod); - Synth_Generate_Statement_Body (Syn_Inst, Bod, Name); - exit; - end if; - Gen := Get_Generate_Else_Clause (Gen); - exit when Gen = Null_Node; - end loop; - end Synth_If_Generate_Statement; - - procedure Synth_For_Generate_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Iterator : constant Node := Get_Parameter_Specification (Stmt); - Bod : constant Node := Get_Generate_Statement_Body (Stmt); - Configs : constant Node := Get_Generate_Block_Configuration (Bod); - It_Type : constant Node := Get_Declaration_Type (Iterator); - Config : Node; - It_Rng : Type_Acc; - Val : Valtyp; - Name : Sname; - Lname : Sname; - begin - if It_Type /= Null_Node then - Synth_Subtype_Indication (Syn_Inst, It_Type); - end if; - - -- Initial value. - It_Rng := Get_Subtype_Object (Syn_Inst, Get_Type (Iterator)); - Val := Create_Value_Discrete (It_Rng.Drange.Left, It_Rng); - - Name := New_Sname_User (Get_Identifier (Stmt), Get_Sname (Syn_Inst)); - - while In_Range (It_Rng.Drange, Read_Discrete (Val)) loop - -- Find and apply the config block. - declare - Spec : Node; - begin - Config := Configs; - while Config /= Null_Node loop - Spec := Get_Block_Specification (Config); - case Get_Kind (Spec) is - when Iir_Kind_Simple_Name => - exit; - when others => - Error_Kind ("synth_for_generate_statement", Spec); - end case; - Config := Get_Prev_Block_Configuration (Config); - end loop; - if Config = Null_Node then - raise Internal_Error; - end if; - Apply_Block_Configuration (Config, Bod); - end; - - -- FIXME: get position ? - Lname := New_Sname_Version (Uns32 (Read_Discrete (Val)), Name); - - Synth_Generate_Statement_Body (Syn_Inst, Bod, Lname, Iterator, Val); - Update_Index (It_Rng.Drange, Val); - end loop; - end Synth_For_Generate_Statement; - - procedure Synth_Concurrent_Statement - (Syn_Inst : Synth_Instance_Acc; Stmt : Node) - is - Ctxt : constant Context_Acc := Get_Build (Syn_Inst); - begin - case Get_Kind (Stmt) is - when Iir_Kind_Concurrent_Simple_Signal_Assignment => - Push_Phi; - Synth_Simple_Signal_Assignment (Syn_Inst, Stmt); - Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); - when Iir_Kind_Concurrent_Conditional_Signal_Assignment => - Push_Phi; - Synth_Conditional_Signal_Assignment (Syn_Inst, Stmt); - Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); - when Iir_Kind_Concurrent_Selected_Signal_Assignment => - Push_Phi; - Synth_Selected_Signal_Assignment (Syn_Inst, Stmt); - Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); - when Iir_Kind_Concurrent_Procedure_Call_Statement => - Push_Phi; - Synth_Procedure_Call (Syn_Inst, Stmt); - Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); - when Iir_Kinds_Process_Statement => - Synth_Process_Statement (Syn_Inst, Stmt); - when Iir_Kind_If_Generate_Statement => - Synth_If_Generate_Statement (Syn_Inst, Stmt); - when Iir_Kind_For_Generate_Statement => - Synth_For_Generate_Statement (Syn_Inst, Stmt); - when Iir_Kind_Component_Instantiation_Statement => - if Is_Component_Instantiation (Stmt) then - declare - Comp_Config : constant Node := - Get_Component_Configuration (Stmt); - begin - if Get_Binding_Indication (Comp_Config) = Null_Node then - -- Not bound. - Synth_Blackbox_Instantiation_Statement (Syn_Inst, Stmt); - else - Synth_Component_Instantiation_Statement (Syn_Inst, Stmt); - end if; - end; - -- Un-apply configuration. - Set_Component_Configuration (Stmt, Null_Node); - else - Synth_Design_Instantiation_Statement (Syn_Inst, Stmt); - end if; - when Iir_Kind_Block_Statement => - Synth_Block_Statement (Syn_Inst, Stmt); - when Iir_Kind_Psl_Default_Clock => - null; - when Iir_Kind_Psl_Restrict_Directive => - Synth_Psl_Restrict_Directive (Syn_Inst, Stmt); - when Iir_Kind_Psl_Assume_Directive => - if Flags.Flag_Assume_As_Assert then - Synth_Psl_Assert_Directive (Syn_Inst, Stmt); - else - Synth_Psl_Assume_Directive (Syn_Inst, Stmt); - end if; - when Iir_Kind_Psl_Cover_Directive => - Synth_Psl_Cover_Directive (Syn_Inst, Stmt); - when Iir_Kind_Psl_Assert_Directive => - if Flags.Flag_Assert_As_Assume then - Synth_Psl_Assume_Directive (Syn_Inst, Stmt); - else - Synth_Psl_Assert_Directive (Syn_Inst, Stmt); - end if; - when Iir_Kind_Concurrent_Assertion_Statement => - -- Passive statement. - Synth_Concurrent_Assertion_Statement (Syn_Inst, Stmt); - when others => - Error_Kind ("synth_concurrent_statement", Stmt); - end case; - end Synth_Concurrent_Statement; - - procedure Synth_Concurrent_Statements - (Syn_Inst : Synth_Instance_Acc; Stmts : Node) - is - Stmt : Node; - begin - Stmt := Stmts; - while Is_Valid (Stmt) loop - Synth_Concurrent_Statement (Syn_Inst, Stmt); - Stmt := Get_Chain (Stmt); - end loop; - end Synth_Concurrent_Statements; - - -- For allconst/allseq/... - procedure Synth_Attribute_Formal (Syn_Inst : Synth_Instance_Acc; - Val : Node; - Id : Formal_Module_Id) - is - Spec : constant Node := Get_Attribute_Specification (Val); - Sig : constant Node := Get_Designated_Entity (Val); - V : Valtyp; - begin - -- The type must be boolean - if (Get_Base_Type (Get_Type (Val)) /= - Vhdl.Std_Package.Boolean_Type_Definition) - then - Error_Msg_Synth (+Val, "type of attribute %i must be boolean", - (1 => +Get_Attribute_Designator (Spec))); - return; - end if; - - -- The designated entity must be a signal. - if Get_Kind (Sig) /= Iir_Kind_Signal_Declaration then - Error_Msg_Synth (+Val, "attribute %i only applies to signals", - (1 => +Get_Attribute_Designator (Spec))); - return; - end if; - - -- The value must be true - V := Synth_Expression_With_Type - (Syn_Inst, Get_Expression (Spec), Boolean_Type); - if Read_Discrete (V) /= 1 then - return; - end if; - - declare - Off : Value_Offsets; - Dyn : Dyn_Name; - N : Net; - Base : Valtyp; - Typ : Type_Acc; - begin - Synth_Assignment_Prefix (Syn_Inst, Sig, Base, Typ, Off, Dyn); - pragma Assert (Off = (0, 0)); - pragma Assert (Dyn.Voff = No_Net); - pragma Assert (Base.Val.Kind = Value_Wire); - pragma Assert (Base.Typ = Typ); - - N := Build_Formal_Input (Get_Build (Syn_Inst), Id, Typ.W); - Set_Location (N, Val); - Add_Conc_Assign (Base.Val.W, N, 0); - end; - end Synth_Attribute_Formal; - - procedure Synth_Attribute_Values - (Syn_Inst : Synth_Instance_Acc; Unit : Node) - is - use Std_Names; - - Val : Node; - Spec : Node; - Id : Name_Id; - begin - Val := Get_Attribute_Value_Chain (Unit); - while Val /= Null_Node loop - Spec := Get_Attribute_Specification (Val); - Id := Get_Identifier (Get_Attribute_Designator (Spec)); - case Id is - when Name_Allconst => - Synth_Attribute_Formal (Syn_Inst, Val, Id_Allconst); - when Name_Allseq => - Synth_Attribute_Formal (Syn_Inst, Val, Id_Allseq); - when Name_Anyconst => - Synth_Attribute_Formal (Syn_Inst, Val, Id_Anyconst); - when Name_Anyseq => - Synth_Attribute_Formal (Syn_Inst, Val, Id_Anyseq); - when Name_Loc => - -- Applies to nets/ports. - null; - when others => - Warning_Msg_Synth (+Spec, "unhandled attribute %i", (1 => +Id)); - end case; - Val := Get_Value_Chain (Val); - end loop; - end Synth_Attribute_Values; - - procedure Synth_Verification_Unit - (Syn_Inst : Synth_Instance_Acc; Unit : Node) - is - use Areapools; - Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; - Unit_Inst : Synth_Instance_Acc; - Unit_Sname : Sname; - M : Areapools.Mark_Type; - Item : Node; - Last_Type : Node; - begin - Unit_Sname := New_Sname_User (Get_Identifier (Unit), - Get_Sname (Syn_Inst)); - Unit_Inst := Make_Instance (Syn_Inst, Unit, Unit_Sname); - Mark (M, Proc_Pool); - Instance_Pool := Proc_Pool'Access; - - Apply_Block_Configuration - (Get_Verification_Block_Configuration (Unit), Unit); - - Last_Type := Null_Node; - Item := Get_Vunit_Item_Chain (Unit); - while Item /= Null_Node loop - case Get_Kind (Item) is - when Iir_Kind_Psl_Default_Clock => - null; - when Iir_Kind_Psl_Assert_Directive => - Synth_Psl_Assert_Directive (Unit_Inst, Item); - when Iir_Kind_Psl_Assume_Directive => - Synth_Psl_Assume_Directive (Unit_Inst, Item); - when Iir_Kind_Psl_Restrict_Directive => - Synth_Psl_Restrict_Directive (Unit_Inst, Item); - when Iir_Kind_Psl_Cover_Directive => - Synth_Psl_Cover_Directive (Unit_Inst, Item); - when Iir_Kind_Signal_Declaration - | Iir_Kind_Function_Declaration - | Iir_Kind_Procedure_Declaration - | Iir_Kind_Function_Body - | Iir_Kind_Procedure_Body - | Iir_Kind_Attribute_Declaration - | Iir_Kind_Attribute_Specification => - Synth_Declaration (Unit_Inst, Item, False, Last_Type); - when Iir_Kinds_Concurrent_Signal_Assignment - | Iir_Kinds_Process_Statement - | Iir_Kinds_Generate_Statement - | Iir_Kind_Block_Statement - | Iir_Kind_Concurrent_Procedure_Call_Statement - | Iir_Kind_Component_Instantiation_Statement => - Synth_Concurrent_Statement (Unit_Inst, Item); - when others => - Error_Kind ("synth_verification_unit", Item); - end case; - Item := Get_Chain (Item); - end loop; - - Synth_Attribute_Values (Unit_Inst, Unit); - - -- Finalize - Item := Get_Vunit_Item_Chain (Unit); - while Item /= Null_Node loop - case Get_Kind (Item) is - when Iir_Kind_Psl_Default_Clock - | Iir_Kind_Psl_Assert_Directive - | Iir_Kind_Psl_Assume_Directive - | Iir_Kind_Psl_Restrict_Directive - | Iir_Kind_Psl_Cover_Directive => - null; - when Iir_Kinds_Concurrent_Signal_Assignment - | Iir_Kinds_Process_Statement - | Iir_Kinds_Generate_Statement - | Iir_Kind_Block_Statement - | Iir_Kind_Concurrent_Procedure_Call_Statement - | Iir_Kind_Component_Instantiation_Statement => - null; - when Iir_Kind_Signal_Declaration - | Iir_Kind_Function_Declaration - | Iir_Kind_Procedure_Declaration - | Iir_Kind_Function_Body - | Iir_Kind_Procedure_Body - | Iir_Kind_Attribute_Declaration - | Iir_Kind_Attribute_Specification => - Finalize_Declaration (Unit_Inst, Item, False); - when others => - Error_Kind ("synth_verification_unit(2)", Item); - end case; - Item := Get_Chain (Item); - end loop; - - Free_Instance (Unit_Inst); - Release (M, Proc_Pool); - Instance_Pool := Prev_Instance_Pool; - end Synth_Verification_Unit; -end Synth.Stmts; diff --git a/src/synth/synth-stmts.ads b/src/synth/synth-stmts.ads deleted file mode 100644 index 2009b1d4f..000000000 --- a/src/synth/synth-stmts.ads +++ /dev/null @@ -1,167 +0,0 @@ --- Statements synthesis. --- Copyright (C) 2017 Tristan Gingold --- --- This file is part of GHDL. --- --- This program is free software: you can redistribute it and/or modify --- it under the terms of the GNU General Public License as published by --- the Free Software Foundation, either version 2 of the License, or --- (at your option) any later version. --- --- This program is distributed in the hope that it will be useful, --- but WITHOUT ANY WARRANTY; without even the implied warranty of --- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the --- GNU General Public License for more details. --- --- You should have received a copy of the GNU General Public License --- along with this program. If not, see . - -with Types; use Types; -with Vhdl.Nodes; use Vhdl.Nodes; - -with Netlists; use Netlists; - -with Synth.Objtypes; use Synth.Objtypes; -with Synth.Values; use Synth.Values; -with Synth.Vhdl_Context; use Synth.Vhdl_Context; -with Synth.Vhdl_Environment; use Synth.Vhdl_Environment.Env; - -package Synth.Stmts is - procedure Synth_Subprogram_Association (Subprg_Inst : Synth_Instance_Acc; - Caller_Inst : Synth_Instance_Acc; - Inter_Chain : Node; - Assoc_Chain : Node); - - -- Dynamic index for Synth_Assignment_Prefix. - -- As dynamic is about dynamic (!) index, the index is a net. - type Dyn_Name is record - -- Start and type of the indexed part, which can be a part of the - -- base name. - Pfx_Off : Value_Offsets; - Pfx_Typ : Type_Acc; - - -- Variable offset. - Voff : Net; - end record; - - No_Dyn_Name : constant Dyn_Name := (Pfx_Off => No_Value_Offsets, - Pfx_Typ => null, - Voff => No_Net); - - -- Transform PFX into DEST_*. - -- DEST_BASE is the base object (with its own typ). Can be the result, - -- a net or an object larger than the result. - -- DEST_TYP is the type of the result. - -- DEST_OFF is the offset, within DEST_DYN. - -- DEST_DYN is set (Voff field set) when there is a non-static index. - procedure Synth_Assignment_Prefix (Syn_Inst : Synth_Instance_Acc; - Pfx : Node; - Dest_Base : out Valtyp; - Dest_Typ : out Type_Acc; - Dest_Off : out Value_Offsets; - Dest_Dyn : out Dyn_Name); - - procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; - Target : Node; - Val : Valtyp; - Loc : Node); - - function Synth_Read_Memory (Syn_Inst : Synth_Instance_Acc; - Obj : Valtyp; - Res_Typ : Type_Acc; - Off : Uns32; - Dyn : Dyn_Name; - Loc : Node) return Valtyp; - - function Synth_User_Function_Call - (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp; - - -- Operation implemented by a user function. - function Synth_User_Operator (Syn_Inst : Synth_Instance_Acc; - Left_Expr : Node; - Right_Expr : Node; - Expr : Node) return Valtyp; - - -- Generate netlists for concurrent statements STMTS. - procedure Synth_Concurrent_Statements - (Syn_Inst : Synth_Instance_Acc; Stmts : Node); - - -- Apply attributes of UNIT. - procedure Synth_Attribute_Values - (Syn_Inst : Synth_Instance_Acc; Unit : Node); - - procedure Synth_Verification_Unit - (Syn_Inst : Synth_Instance_Acc; Unit : Node); - - -- For iterators. - procedure Update_Index (Rng : Discrete_Range_Type; V : in out Valtyp); - -private - -- There are 2 execution mode: - -- * static: it is like simulation, all the inputs are known, neither - -- gates nor signals are generated. This mode is used during - -- elaboration and when all inputs of a subprogram are known. - -- * dynamic: inputs can be wires so gates are generated. But many types - -- (like file or access) cannot be handled. - type Mode_Type is (Mode_Static, Mode_Dynamic); - - type Loop_Context (Mode : Mode_Type); - type Loop_Context_Acc is access all Loop_Context; - - type Loop_Context (Mode : Mode_Type) is record - Prev_Loop : Loop_Context_Acc; - Loop_Stmt : Node; - - case Mode is - when Mode_Dynamic => - -- Set when this loop has next/exit statements for itself. - -- Set to true so that inner loops have to declare W_Quit. - Need_Quit : Boolean; - - -- Value of W_En at the entry of the loop. - Saved_En : Net; - - -- Set to 0 in case of exit for the loop. - -- Set to 0 in case of exit/next for outer loop. - -- Initialized to 1. - W_Exit : Wire_Id; - - -- Set to 0 if this loop has to be quited because of an - -- exit/next for an outer loop. Initialized to 1. - W_Quit : Wire_Id; - - -- Mark to release wires. - Wire_Mark : Wire_Id; - when Mode_Static => - S_Exit : Boolean; - S_Quit : Boolean; - end case; - end record; - - -- Context for sequential statements. - type Seq_Context (Mode : Mode_Type) is record - Inst : Synth_Instance_Acc; - - Cur_Loop : Loop_Context_Acc; - - Ret_Value : Valtyp; - Ret_Typ : Type_Acc; - Nbr_Ret : Int32; - - case Mode is - when Mode_Dynamic => - -- Enable execution. For loop controls. - W_En : Wire_Id; - - W_Ret : Wire_Id; - - -- Return value. - W_Val : Wire_Id; - - Ret_Init : Net; - - when Mode_Static => - S_En : Boolean; - end case; - end record; -end Synth.Stmts; diff --git a/src/synth/synth-vhdl_aggr.adb b/src/synth/synth-vhdl_aggr.adb index 250ebb1aa..fe7e95058 100644 --- a/src/synth/synth-vhdl_aggr.adb +++ b/src/synth/synth-vhdl_aggr.adb @@ -28,9 +28,9 @@ with Vhdl.Utils; use Vhdl.Utils; with Synth.Memtype; use Synth.Memtype; with Synth.Errors; use Synth.Errors; -with Synth.Expr; use Synth.Expr; -with Synth.Stmts; use Synth.Stmts; -with Synth.Decls; use Synth.Decls; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; +with Synth.Vhdl_Stmts; use Synth.Vhdl_Stmts; +with Synth.Vhdl_Decls; use Synth.Vhdl_Decls; package body Synth.Vhdl_Aggr is type Stride_Array is array (Dim_Type range <>) of Nat32; diff --git a/src/synth/synth-vhdl_context.adb b/src/synth/synth-vhdl_context.adb index 0ef9b417e..bb3d7b98c 100644 --- a/src/synth/synth-vhdl_context.adb +++ b/src/synth/synth-vhdl_context.adb @@ -26,7 +26,7 @@ with Vhdl.Utils; with Netlists.Folds; use Netlists.Folds; -with Synth.Expr; use Synth.Expr; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; with Netlists.Locations; package body Synth.Vhdl_Context is diff --git a/src/synth/synth-vhdl_decls.adb b/src/synth/synth-vhdl_decls.adb new file mode 100644 index 000000000..7507d21b0 --- /dev/null +++ b/src/synth/synth-vhdl_decls.adb @@ -0,0 +1,1227 @@ +-- Create declarations for synthesis. +-- Copyright (C) 2017 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with Types; use Types; +with Mutils; use Mutils; +with Std_Names; + +with Netlists.Builders; use Netlists.Builders; +with Netlists.Folds; use Netlists.Folds; +with Netlists.Utils; use Netlists.Utils; +with Netlists.Gates; + +with Vhdl.Errors; +with Vhdl.Utils; use Vhdl.Utils; +with Vhdl.Std_Package; +with Vhdl.Ieee.Std_Logic_1164; + +with Synth.Vhdl_Environment; use Synth.Vhdl_Environment.Env; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; +with Synth.Vhdl_Stmts; +with Synth.Source; use Synth.Source; +with Synth.Errors; use Synth.Errors; +with Synth.Vhdl_Files; +with Synth.Values; use Synth.Values; + +package body Synth.Vhdl_Decls is + procedure Create_Var_Wire + (Syn_Inst : Synth_Instance_Acc; Decl : Iir; Init : Valtyp) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Vt : constant Valtyp := Get_Value (Syn_Inst, Decl); + Value : Net; + Ival : Net; + W : Width; + Name : Sname; + begin + case Vt.Val.Kind is + when Value_Wire => + -- FIXME: get the width directly from the wire ? + W := Get_Type_Width (Vt.Typ); + Name := New_Sname_User (Get_Identifier (Decl), + Get_Sname (Syn_Inst)); + if Init /= No_Valtyp then + Ival := Get_Net (Ctxt, Init); + pragma Assert (Get_Width (Ival) = W); + Value := Build_Isignal (Ctxt, Name, Ival); + else + Value := Build_Signal (Ctxt, Name, W); + end if; + Set_Location (Value, Decl); + Set_Wire_Gate (Vt.Val.W, Value); + when others => + raise Internal_Error; + end case; + end Create_Var_Wire; + + function Type_To_Param_Type (Atype : Node) return Param_Type + is + use Vhdl.Std_Package; + Btype : constant Node := Get_Base_Type (Atype); + begin + if Btype = String_Type_Definition then + return Param_Pval_String; + elsif Btype = Time_Type_Definition then + return Param_Pval_Time_Ps; + else + case Get_Kind (Btype) is + when Iir_Kind_Integer_Type_Definition => + return Param_Pval_Integer; + when Iir_Kind_Floating_Type_Definition => + return Param_Pval_Real; + when others => + return Param_Pval_Vector; + end case; + end if; + end Type_To_Param_Type; + + function Memtyp_To_Pval (Mt : Memtyp) return Pval + is + Len : constant Uns32 := (Mt.Typ.W + 31) / 32; + Vec : Logvec_Array_Acc; + Off : Uns32; + Has_Zx : Boolean; + Pv : Pval; + begin + if Len = 0 then + return Create_Pval2 (0); + end if; + + Vec := new Logvec_Array'(0 .. Digit_Index (Len - 1) => (0, 0)); + Off := 0; + Has_Zx := False; + Value2logvec (Mt, 0, Mt.Typ.W, Vec.all, Off, Has_Zx); + pragma Assert (Off = Mt.Typ.W); + if Has_Zx then + Pv := Create_Pval4 (Mt.Typ.W); + else + Pv := Create_Pval2 (Mt.Typ.W); + end if; + for I in 0 .. Len - 1 loop + Write_Pval (Pv, I, Vec (Digit_Index (I))); + end loop; + Free_Logvec_Array (Vec); + return Pv; + end Memtyp_To_Pval; + + procedure Synth_Subtype_Indication_If_Anonymous + (Syn_Inst : Synth_Instance_Acc; Atype : Node) is + begin + if Get_Type_Declarator (Atype) = Null_Node then + Synth_Subtype_Indication (Syn_Inst, Atype); + end if; + end Synth_Subtype_Indication_If_Anonymous; + + function Synth_Subtype_Indication_If_Anonymous + (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc is + begin + if Get_Type_Declarator (Atype) = Null_Node then + return Synth_Subtype_Indication (Syn_Inst, Atype); + else + return Get_Subtype_Object (Syn_Inst, Atype); + end if; + end Synth_Subtype_Indication_If_Anonymous; + + function Synth_Array_Type_Definition + (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc + is + El_Type : constant Node := Get_Element_Subtype (Def); + Ndims : constant Natural := Get_Nbr_Dimensions (Def); + El_Typ : Type_Acc; + Typ : Type_Acc; + begin + Synth_Subtype_Indication_If_Anonymous (Syn_Inst, El_Type); + El_Typ := Get_Subtype_Object (Syn_Inst, El_Type); + + if El_Typ.Kind in Type_Nets and then Ndims = 1 then + Typ := Create_Unbounded_Vector (El_Typ); + else + Typ := Create_Unbounded_Array (Dim_Type (Ndims), El_Typ); + end if; + return Typ; + end Synth_Array_Type_Definition; + + function Synth_Record_Type_Definition + (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc + is + El_List : constant Node_Flist := Get_Elements_Declaration_List (Def); + Rec_Els : Rec_El_Array_Acc; + El : Node; + El_Type : Node; + El_Typ : Type_Acc; + begin + Rec_Els := Create_Rec_El_Array + (Iir_Index32 (Get_Nbr_Elements (El_List))); + + for I in Flist_First .. Flist_Last (El_List) loop + El := Get_Nth_Element (El_List, I); + El_Type := Get_Type (El); + El_Typ := Synth_Subtype_Indication_If_Anonymous (Syn_Inst, El_Type); + Rec_Els.E (Iir_Index32 (I + 1)).Typ := El_Typ; + end loop; + + if not Is_Fully_Constrained_Type (Def) then + return Create_Unbounded_Record (Rec_Els); + else + return Create_Record_Type (Rec_Els); + end if; + end Synth_Record_Type_Definition; + + function Synth_Access_Type_Definition + (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc + is + Des_Type : constant Node := Get_Designated_Type (Def); + Des_Typ : Type_Acc; + Typ : Type_Acc; + begin + Synth_Subtype_Indication_If_Anonymous (Syn_Inst, Des_Type); + Des_Typ := Get_Subtype_Object (Syn_Inst, Des_Type); + + Typ := Create_Access_Type (Des_Typ); + return Typ; + end Synth_Access_Type_Definition; + + function Synth_File_Type_Definition + (Syn_Inst : Synth_Instance_Acc; Def : Node) return Type_Acc + is + File_Type : constant Node := Get_Type (Get_File_Type_Mark (Def)); + File_Typ : Type_Acc; + Typ : Type_Acc; + Sig : String_Acc; + begin + File_Typ := Get_Subtype_Object (Syn_Inst, File_Type); + + if Get_Text_File_Flag (Def) + or else + Get_Kind (File_Type) in Iir_Kinds_Scalar_Type_And_Subtype_Definition + then + Sig := null; + else + declare + Sig_Str : String (1 .. Get_File_Signature_Length (File_Type) + 2); + Off : Natural := Sig_Str'First; + begin + Get_File_Signature (File_Type, Sig_Str, Off); + Sig_Str (Off + 0) := '.'; + Sig_Str (Off + 1) := ASCII.NUL; + Sig := new String'(Sig_Str); + end; + end if; + + Typ := Create_File_Type (File_Typ); + Typ.File_Signature := Sig; + + return Typ; + end Synth_File_Type_Definition; + + function Scalar_Size_To_Size (Def : Node) return Size_Type is + begin + case Get_Scalar_Size (Def) is + when Scalar_8 => + return 1; + when Scalar_16 => + return 2; + when Scalar_32 => + return 4; + when Scalar_64 => + return 8; + end case; + end Scalar_Size_To_Size; + + procedure Synth_Type_Definition (Syn_Inst : Synth_Instance_Acc; Def : Node) + is + Typ : Type_Acc; + begin + case Get_Kind (Def) is + when Iir_Kind_Enumeration_Type_Definition => + if Def = Vhdl.Ieee.Std_Logic_1164.Std_Ulogic_Type + or else Def = Vhdl.Ieee.Std_Logic_1164.Std_Logic_Type + then + Typ := Logic_Type; + elsif Def = Vhdl.Std_Package.Boolean_Type_Definition then + Typ := Boolean_Type; + elsif Def = Vhdl.Std_Package.Bit_Type_Definition then + Typ := Bit_Type; + else + declare + Nbr_El : constant Natural := + Get_Nbr_Elements (Get_Enumeration_Literal_List (Def)); + Rng : Discrete_Range_Type; + W : Width; + begin + W := Uns32 (Clog2 (Uns64 (Nbr_El))); + Rng := (Dir => Dir_To, + Is_Signed => False, + Left => 0, + Right => Int64 (Nbr_El - 1)); + Typ := Create_Discrete_Type + (Rng, Scalar_Size_To_Size (Def), W); + end; + end if; + when Iir_Kind_Array_Type_Definition => + Typ := Synth_Array_Type_Definition (Syn_Inst, Def); + when Iir_Kind_Access_Type_Definition => + Typ := Synth_Access_Type_Definition (Syn_Inst, Def); + when Iir_Kind_File_Type_Definition => + Typ := Synth_File_Type_Definition (Syn_Inst, Def); + when Iir_Kind_Record_Type_Definition => + Typ := Synth_Record_Type_Definition (Syn_Inst, Def); + when Iir_Kind_Protected_Type_Declaration => + Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Def)); + when others => + Vhdl.Errors.Error_Kind ("synth_type_definition", Def); + end case; + if Typ /= null then + Create_Subtype_Object (Syn_Inst, Def, Typ); + end if; + end Synth_Type_Definition; + + procedure Synth_Anonymous_Type_Definition + (Syn_Inst : Synth_Instance_Acc; Def : Node; St : Node) + is + Typ : Type_Acc; + begin + case Get_Kind (Def) is + when Iir_Kind_Integer_Type_Definition + | Iir_Kind_Physical_Type_Definition => + declare + Cst : constant Node := Get_Range_Constraint (St); + L, R : Int64; + Rng : Discrete_Range_Type; + W : Width; + begin + L := Get_Value (Get_Left_Limit (Cst)); + R := Get_Value (Get_Right_Limit (Cst)); + Rng := Build_Discrete_Range_Type (L, R, Get_Direction (Cst)); + W := Discrete_Range_Width (Rng); + Typ := Create_Discrete_Type + (Rng, Scalar_Size_To_Size (Def), W); + end; + when Iir_Kind_Floating_Type_Definition => + declare + Cst : constant Node := Get_Range_Constraint (St); + L, R : Fp64; + Rng : Float_Range_Type; + begin + L := Get_Fp_Value (Get_Left_Limit (Cst)); + R := Get_Fp_Value (Get_Right_Limit (Cst)); + Rng := (Get_Direction (Cst), L, R); + Typ := Create_Float_Type (Rng); + end; + when Iir_Kind_Array_Type_Definition => + Typ := Synth_Array_Type_Definition (Syn_Inst, Def); + when others => + Vhdl.Errors.Error_Kind ("synth_anonymous_type_definition", Def); + end case; + Create_Subtype_Object (Syn_Inst, Def, Typ); + end Synth_Anonymous_Type_Definition; + + function Synth_Discrete_Range_Constraint + (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Discrete_Range_Type + is + Res : Discrete_Range_Type; + begin + Synth_Discrete_Range (Syn_Inst, Rng, Res); + return Res; + end Synth_Discrete_Range_Constraint; + + function Synth_Float_Range_Constraint + (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Float_Range_Type is + begin + case Get_Kind (Rng) is + when Iir_Kind_Range_Expression => + -- FIXME: check range. + return Synth_Float_Range_Expression (Syn_Inst, Rng); + when others => + Vhdl.Errors.Error_Kind ("synth_float_range_constraint", Rng); + end case; + end Synth_Float_Range_Constraint; + + function Has_Element_Subtype_Indication (Atype : Node) return Boolean is + begin + return Get_Array_Element_Constraint (Atype) /= Null_Node + or else + (Get_Resolution_Indication (Atype) /= Null_Node + and then + (Get_Kind (Get_Resolution_Indication (Atype)) + = Iir_Kind_Array_Element_Resolution)); + end Has_Element_Subtype_Indication; + + function Synth_Array_Subtype_Indication + (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc + is + El_Type : constant Node := Get_Element_Subtype (Atype); + St_Indexes : constant Node_Flist := Get_Index_Subtype_List (Atype); + Ptype : Node; + St_El : Node; + Btyp : Type_Acc; + Etyp : Type_Acc; + Bnds : Bound_Array_Acc; + begin + -- VHDL08 + if Has_Element_Subtype_Indication (Atype) then + -- This subtype has created a new anonymous subtype for the + -- element. + Synth_Subtype_Indication (Syn_Inst, El_Type); + end if; + + if not Get_Index_Constraint_Flag (Atype) then + Ptype := Get_Type (Get_Subtype_Type_Mark (Atype)); + if Get_Element_Subtype (Ptype) = Get_Element_Subtype (Atype) then + -- That's an alias. + -- FIXME: maybe a resolution function was added? + -- FIXME: also handle resolution added in element subtype. + return Get_Subtype_Object (Syn_Inst, Ptype); + end if; + end if; + + Btyp := Get_Subtype_Object (Syn_Inst, Get_Base_Type (Atype)); + case Btyp.Kind is + when Type_Unbounded_Vector => + if Get_Index_Constraint_Flag (Atype) then + St_El := Get_Index_Type (St_Indexes, 0); + return Create_Vector_Type + (Synth_Bounds_From_Range (Syn_Inst, St_El), Btyp.Uvec_El); + else + -- An alias. + -- Handle vhdl08 definition of std_logic_vector from + -- std_ulogic_vector. + return Btyp; + end if; + when Type_Unbounded_Array => + -- FIXME: partially constrained arrays, subtype in indexes... + Etyp := Get_Subtype_Object (Syn_Inst, El_Type); + if Get_Index_Constraint_Flag (Atype) then + Bnds := Create_Bound_Array + (Dim_Type (Get_Nbr_Elements (St_Indexes))); + for I in Flist_First .. Flist_Last (St_Indexes) loop + St_El := Get_Index_Type (St_Indexes, I); + Bnds.D (Dim_Type (I + 1)) := + Synth_Bounds_From_Range (Syn_Inst, St_El); + end loop; + return Create_Array_Type (Bnds, Etyp); + else + raise Internal_Error; + end if; + when others => + raise Internal_Error; + end case; + end Synth_Array_Subtype_Indication; + + function Synth_Subtype_Indication + (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc is + begin + -- TODO: handle aliases directly. + case Get_Kind (Atype) is + when Iir_Kind_Array_Subtype_Definition => + return Synth_Array_Subtype_Indication (Syn_Inst, Atype); + when Iir_Kind_Record_Subtype_Definition => + return Synth_Record_Type_Definition (Syn_Inst, Atype); + when Iir_Kind_Integer_Subtype_Definition + | Iir_Kind_Physical_Subtype_Definition + | Iir_Kind_Enumeration_Subtype_Definition => + declare + Btype : constant Type_Acc := + Get_Subtype_Object (Syn_Inst, Get_Base_Type (Atype)); + Rng : Discrete_Range_Type; + W : Width; + begin + if Btype.Kind in Type_Nets then + -- A subtype of a bit/logic type is still a bit/logic. + -- FIXME: bounds. + return Btype; + else + Rng := Synth_Discrete_Range_Constraint + (Syn_Inst, Get_Range_Constraint (Atype)); + W := Discrete_Range_Width (Rng); + return Create_Discrete_Type (Rng, Btype.Sz, W); + end if; + end; + when Iir_Kind_Floating_Subtype_Definition => + declare + Rng : Float_Range_Type; + begin + Rng := Synth_Float_Range_Constraint + (Syn_Inst, Get_Range_Constraint (Atype)); + return Create_Float_Type (Rng); + end; + when others => + Vhdl.Errors.Error_Kind ("synth_subtype_indication", Atype); + end case; + end Synth_Subtype_Indication; + + procedure Synth_Subtype_Indication + (Syn_Inst : Synth_Instance_Acc; Atype : Node) + is + Typ : Type_Acc; + begin + Typ := Synth_Subtype_Indication (Syn_Inst, Atype); + Create_Subtype_Object (Syn_Inst, Atype, Typ); + end Synth_Subtype_Indication; + + function Get_Declaration_Type (Decl : Node) return Node + is + Ind : constant Node := Get_Subtype_Indication (Decl); + Atype : Node; + begin + if Get_Is_Ref (Decl) or else Ind = Null_Iir then + -- A secondary declaration in a list. + return Null_Node; + end if; + Atype := Ind; + loop + case Get_Kind (Atype) is + when Iir_Kinds_Denoting_Name => + Atype := Get_Named_Entity (Atype); + when Iir_Kind_Subtype_Declaration + | Iir_Kind_Type_Declaration => + -- Type already declared, so already handled. + return Null_Node; + when Iir_Kind_Array_Subtype_Definition + | Iir_Kind_Record_Subtype_Definition + | Iir_Kind_Integer_Subtype_Definition + | Iir_Kind_Floating_Subtype_Definition + | Iir_Kind_Physical_Subtype_Definition + | Iir_Kind_Enumeration_Subtype_Definition => + return Atype; + when others => + Vhdl.Errors.Error_Kind ("get_declaration_type", Atype); + end case; + end loop; + end Get_Declaration_Type; + + procedure Synth_Declaration_Type + (Syn_Inst : Synth_Instance_Acc; Decl : Node) + is + Atype : constant Node := Get_Declaration_Type (Decl); + begin + if Atype = Null_Node then + return; + end if; + Synth_Subtype_Indication (Syn_Inst, Atype); + end Synth_Declaration_Type; + + procedure Synth_Constant_Declaration (Syn_Inst : Synth_Instance_Acc; + Decl : Node; + Is_Subprg : Boolean; + Last_Type : in out Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Deferred_Decl : constant Node := Get_Deferred_Declaration (Decl); + First_Decl : Node; + Decl_Type : Node; + Val : Valtyp; + Cst : Valtyp; + Obj_Type : Type_Acc; + begin + Synth_Declaration_Type (Syn_Inst, Decl); + if Deferred_Decl = Null_Node + or else Get_Deferred_Declaration_Flag (Decl) + then + -- Create the object (except for full declaration of a + -- deferred constant). + Create_Object (Syn_Inst, Decl, No_Valtyp); + end if; + -- Initialize the value (except for a deferred declaration). + if Get_Deferred_Declaration_Flag (Decl) then + return; + end if; + if Deferred_Decl = Null_Node then + -- A normal constant declaration + First_Decl := Decl; + else + -- The full declaration of a deferred constant. + First_Decl := Deferred_Decl; + end if; + pragma Assert (First_Decl /= Null_Node); + + -- Use the type of the declaration. The type of the constant may + -- be derived from the value. + -- FIXME: what about multiple declarations ? + Decl_Type := Get_Subtype_Indication (Decl); + if Decl_Type = Null_Node then + Decl_Type := Last_Type; + else + if Get_Kind (Decl_Type) in Iir_Kinds_Denoting_Name then + -- Type mark. + Decl_Type := Get_Type (Get_Named_Entity (Decl_Type)); + end if; + Last_Type := Decl_Type; + end if; + Obj_Type := Get_Subtype_Object (Syn_Inst, Decl_Type); + Val := Synth_Expression_With_Type + (Syn_Inst, Get_Default_Value (Decl), Obj_Type); + if Val = No_Valtyp then + Set_Error (Syn_Inst); + return; + end if; + Val := Synth_Subtype_Conversion (Ctxt, Val, Obj_Type, True, Decl); + -- For constant functions, the value must be constant. + pragma Assert (not Get_Instance_Const (Syn_Inst) + or else Is_Static (Val.Val)); + case Val.Val.Kind is + when Value_Const + | Value_Alias => + Cst := Val; + when others => + if Is_Static (Val.Val) then + Cst := Create_Value_Const (Val, Decl); + else + if not Is_Subprg then + Error_Msg_Synth + (+Decl, "signals cannot be used in default value " + & "of this constant"); + end if; + Cst := Val; + end if; + end case; + Create_Object_Force (Syn_Inst, First_Decl, Cst); + end Synth_Constant_Declaration; + + procedure Synth_Attribute_Object (Syn_Inst : Synth_Instance_Acc; + Attr_Value : Node; + Attr_Decl : Node; + Val : Valtyp) + is + Obj : constant Node := Get_Designated_Entity (Attr_Value); + Id : constant Name_Id := Get_Identifier (Attr_Decl); + Inst : Instance; + V : Valtyp; + Ptype : Param_Type; + Pv : Pval; + begin + if Id = Std_Names.Name_Foreign then + -- Not for synthesis. + return; + end if; + + case Get_Kind (Obj) is + when Iir_Kind_Signal_Declaration + | Iir_Kind_Variable_Declaration + | Iir_Kind_Interface_Signal_Declaration => + V := Get_Value (Syn_Inst, Obj); + pragma Assert (V.Val.Kind = Value_Wire); + Inst := Get_Net_Parent (Get_Wire_Gate (V.Val.W)); + when Iir_Kind_Component_Instantiation_Statement => + -- TODO + return; + when others => + -- TODO: components ? + -- TODO: Interface_Signal ? But no instance for them. + Warning_Msg_Synth + (+Attr_Value, "attribute %i for %n is not kept in the netlist", + (+Attr_Decl, +Obj)); + return; + end case; + + Ptype := Type_To_Param_Type (Get_Type (Attr_Decl)); + Pv := Memtyp_To_Pval (Get_Memtyp (Val)); + + Set_Attribute (Inst, Id, Ptype, Pv); + end Synth_Attribute_Object; + + procedure Synth_Attribute_Specification + (Syn_Inst : Synth_Instance_Acc; Spec : Node) + is + Attr_Decl : constant Node := + Get_Named_Entity (Get_Attribute_Designator (Spec)); + Value : Node; + Val : Valtyp; + Val_Type : Type_Acc; + begin + Val_Type := Get_Subtype_Object (Syn_Inst, Get_Type (Attr_Decl)); + Value := Get_Attribute_Value_Spec_Chain (Spec); + while Value /= Null_Iir loop + -- 2. The expression is evaluated to determine the value + -- of the attribute. + -- It is an error if the value of the expression does not + -- belong to the subtype of the attribute; if the + -- attribute is of an array type, then an implicit + -- subtype conversion is first performed on the value, + -- unless the attribute's subtype indication denotes an + -- unconstrained array type. + Val := Synth_Expression_With_Type + (Syn_Inst, Get_Expression (Spec), Val_Type); + -- Check_Constraints (Instance, Val, Attr_Type, Decl); + + -- 3. A new instance of the designated attribute is created + -- and associated with each of the affected items. + -- + -- 4. Each new attribute instance is assigned the value of + -- the expression. + Create_Object (Syn_Inst, Value, Val); + -- Unshare (Val, Instance_Pool); + + if not Get_Instance_Const (Syn_Inst) then + Synth_Attribute_Object (Syn_Inst, Value, Attr_Decl, Val); + end if; + + Value := Get_Spec_Chain (Value); + end loop; + end Synth_Attribute_Specification; + + procedure Synth_Subprogram_Declaration + (Syn_Inst : Synth_Instance_Acc; Subprg : Node) + is + Inter : Node; + begin + if Is_Second_Subprogram_Specification (Subprg) then + -- Already handled. + return; + end if; + + Inter := Get_Interface_Declaration_Chain (Subprg); + while Inter /= Null_Node loop + Synth_Declaration_Type (Syn_Inst, Inter); + Inter := Get_Chain (Inter); + end loop; + end Synth_Subprogram_Declaration; + + procedure Synth_Convertible_Declarations (Syn_Inst : Synth_Instance_Acc) + is + use Vhdl.Std_Package; + begin + Create_Subtype_Object + (Syn_Inst, Convertible_Integer_Type_Definition, + Get_Subtype_Object (Syn_Inst, Universal_Integer_Type_Definition)); + Create_Subtype_Object + (Syn_Inst, Convertible_Real_Type_Definition, + Get_Subtype_Object (Syn_Inst, Universal_Real_Type_Definition)); + end Synth_Convertible_Declarations; + + function Create_Package_Instance (Parent_Inst : Synth_Instance_Acc; + Pkg : Node) + return Synth_Instance_Acc + is + Syn_Inst : Synth_Instance_Acc; + begin + Syn_Inst := Make_Instance (Parent_Inst, Pkg); + if Get_Kind (Get_Parent (Pkg)) = Iir_Kind_Design_Unit then + -- Global package. + Create_Package_Object (Parent_Inst, Pkg, Syn_Inst, True); + else + -- Local package: check elaboration order. + Create_Package_Object (Parent_Inst, Pkg, Syn_Inst, False); + end if; + return Syn_Inst; + end Create_Package_Instance; + + procedure Synth_Package_Declaration + (Parent_Inst : Synth_Instance_Acc; Pkg : Node) + is + Syn_Inst : Synth_Instance_Acc; + begin + if Is_Uninstantiated_Package (Pkg) then + -- Nothing to do (yet) for uninstantiated packages. + return; + end if; + + Syn_Inst := Create_Package_Instance (Parent_Inst, Pkg); + + Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Pkg)); + if Pkg = Vhdl.Std_Package.Standard_Package then + Synth_Convertible_Declarations (Syn_Inst); + end if; + end Synth_Package_Declaration; + + procedure Synth_Package_Body + (Parent_Inst : Synth_Instance_Acc; Pkg : Node; Bod : Node) + is + Pkg_Inst : Synth_Instance_Acc; + begin + if Is_Uninstantiated_Package (Pkg) then + -- Nothing to do (yet) for uninstantiated packages. + return; + end if; + + Pkg_Inst := Get_Package_Object (Parent_Inst, Pkg); + + Synth_Declarations (Pkg_Inst, Get_Declaration_Chain (Bod)); + end Synth_Package_Body; + + procedure Synth_Generics_Association (Sub_Inst : Synth_Instance_Acc; + Syn_Inst : Synth_Instance_Acc; + Inter_Chain : Node; + Assoc_Chain : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Inter : Node; + Inter_Type : Type_Acc; + Assoc : Node; + Assoc_Inter : Node; + Actual : Node; + Val : Valtyp; + begin + Assoc := Assoc_Chain; + Assoc_Inter := Inter_Chain; + while Is_Valid (Assoc) loop + Inter := Get_Association_Interface (Assoc, Assoc_Inter); + case Iir_Kinds_Interface_Declaration (Get_Kind (Inter)) is + when Iir_Kind_Interface_Constant_Declaration => + Synth_Declaration_Type (Sub_Inst, Inter); + Inter_Type := Get_Subtype_Object (Sub_Inst, Get_Type (Inter)); + + case Get_Kind (Assoc) is + when Iir_Kind_Association_Element_Open => + Actual := Get_Default_Value (Inter); + Val := Synth_Expression_With_Type + (Sub_Inst, Actual, Inter_Type); + when Iir_Kind_Association_Element_By_Expression => + Actual := Get_Actual (Assoc); + Val := Synth_Expression_With_Type + (Syn_Inst, Actual, Inter_Type); + when others => + raise Internal_Error; + end case; + + Val := Synth_Subtype_Conversion + (Ctxt, Val, Inter_Type, True, Assoc); + + if Val = No_Valtyp then + Set_Error (Sub_Inst); + elsif not Is_Static (Val.Val) then + Error_Msg_Synth + (+Assoc, "value of generic %i must be static", +Inter); + Val := No_Valtyp; + Set_Error (Sub_Inst); + end if; + + Create_Object (Sub_Inst, Inter, Val); + + when Iir_Kind_Interface_Package_Declaration => + declare + Actual : constant Iir := + Strip_Denoting_Name (Get_Actual (Assoc)); + Pkg_Inst : Synth_Instance_Acc; + begin + Pkg_Inst := Get_Package_Object (Sub_Inst, Actual); + Create_Package_Interface (Sub_Inst, Inter, Pkg_Inst); + end; + + when Iir_Kind_Interface_Variable_Declaration + | Iir_Kind_Interface_File_Declaration + | Iir_Kind_Interface_Signal_Declaration + | Iir_Kind_Interface_Quantity_Declaration + | Iir_Kind_Interface_Terminal_Declaration => + raise Internal_Error; + + when Iir_Kinds_Interface_Subprogram_Declaration + | Iir_Kind_Interface_Type_Declaration => + raise Internal_Error; + end case; + + Next_Association_Interface (Assoc, Assoc_Inter); + end loop; + end Synth_Generics_Association; + + procedure Synth_Package_Instantiation + (Parent_Inst : Synth_Instance_Acc; Pkg : Node) + is + Bod : constant Node := Get_Instance_Package_Body (Pkg); + Sub_Inst : Synth_Instance_Acc; + begin + Sub_Inst := Create_Package_Instance (Parent_Inst, Pkg); + + Synth_Generics_Association + (Sub_Inst, Parent_Inst, + Get_Generic_Chain (Pkg), Get_Generic_Map_Aspect_Chain (Pkg)); + + Synth_Declarations (Sub_Inst, Get_Declaration_Chain (Pkg)); + + if Bod /= Null_Node then + -- Macro expanded package instantiation. + raise Internal_Error; + else + -- Shared body + declare + Uninst : constant Node := Get_Uninstantiated_Package_Decl (Pkg); + Uninst_Bod : constant Node := Get_Package_Body (Uninst); + begin + Set_Uninstantiated_Scope (Sub_Inst, Uninst); + -- Synth declarations of (optional) body. + if Uninst_Bod /= Null_Node then + Synth_Declarations + (Sub_Inst, Get_Declaration_Chain (Uninst_Bod)); + end if; + end; + end if; + end Synth_Package_Instantiation; + + procedure Synth_Variable_Declaration (Syn_Inst : Synth_Instance_Acc; + Decl : Node; + Is_Subprg : Boolean) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Def : constant Node := Get_Default_Value (Decl); + Decl_Type : constant Node := Get_Type (Decl); + Init : Valtyp; + Obj_Typ : Type_Acc; + Wid : Wire_Id; + begin + Synth_Declaration_Type (Syn_Inst, Decl); + if Get_Kind (Decl_Type) = Iir_Kind_Protected_Type_Declaration then + Error_Msg_Synth + (+Decl, "protected type variable is not synthesizable"); + Set_Error (Syn_Inst); + Create_Object (Syn_Inst, Decl, No_Valtyp); + return; + end if; + + Obj_Typ := Get_Subtype_Object (Syn_Inst, Decl_Type); + if not Obj_Typ.Is_Synth + and then not Get_Instance_Const (Syn_Inst) + then + Error_Msg_Synth + (+Decl, "variable with access type is not synthesizable"); + -- FIXME: use a poison value ? + Create_Object (Syn_Inst, Decl, Create_Value_Default (Obj_Typ)); + else + if Is_Valid (Def) then + Init := Synth_Expression_With_Type (Syn_Inst, Def, Obj_Typ); + Init := Synth_Subtype_Conversion + (Ctxt, Init, Obj_Typ, False, Decl); + if not Is_Subprg + and then not Is_Static (Init.Val) + then + Error_Msg_Synth + (+Decl, "signals cannot be used in default value of " + & "this variable"); + end if; + else + Init := Create_Value_Default (Obj_Typ); + end if; + if Get_Instance_Const (Syn_Inst) then + Init := Strip_Alias_Const (Init); + Init := Unshare (Init, Current_Pool); + Create_Object (Syn_Inst, Decl, Init); + else + Create_Wire_Object (Syn_Inst, Wire_Variable, Decl); + Create_Var_Wire (Syn_Inst, Decl, Init); + Wid := Get_Value (Syn_Inst, Decl).Val.W; + if Is_Subprg then + if Is_Static (Init.Val) then + Phi_Assign_Static (Wid, Get_Memtyp (Init)); + else + Phi_Assign_Net (Ctxt, Wid, Get_Net (Ctxt, Init), 0); + end if; + end if; + end if; + end if; + end Synth_Variable_Declaration; + + procedure Synth_Signal_Declaration (Syn_Inst : Synth_Instance_Acc; + Decl : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Def : constant Iir := Get_Default_Value (Decl); + -- Slot : constant Object_Slot_Type := Get_Info (Decl).Slot; + Init : Valtyp; + Obj_Typ : Type_Acc; + begin + Synth_Declaration_Type (Syn_Inst, Decl); + if Get_Kind (Get_Parent (Decl)) = Iir_Kind_Package_Declaration then + Error_Msg_Synth (+Decl, "signals in packages are not supported"); + -- Avoid elaboration error. + Create_Object (Syn_Inst, Decl, No_Valtyp); + return; + end if; + + Create_Wire_Object (Syn_Inst, Wire_Signal, Decl); + if Is_Valid (Def) then + Obj_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Decl)); + Init := Synth_Expression_With_Type (Syn_Inst, Def, Obj_Typ); + Init := Synth_Subtype_Conversion (Ctxt, Init, Obj_Typ, False, Decl); + if not Is_Static (Init.Val) then + Error_Msg_Synth (+Decl, "signals cannot be used in default value " + & "of a signal"); + end if; + else + Init := No_Valtyp; + end if; + Create_Var_Wire (Syn_Inst, Decl, Init); + end Synth_Signal_Declaration; + + procedure Synth_Object_Alias_Declaration + (Syn_Inst : Synth_Instance_Acc; Decl : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Atype : constant Node := Get_Declaration_Type (Decl); + Off : Value_Offsets; + Dyn : Vhdl_Stmts.Dyn_Name; + Res : Valtyp; + Obj_Typ : Type_Acc; + Base : Valtyp; + Typ : Type_Acc; + begin + -- Subtype indication may not be present. + if Atype /= Null_Node then + Synth_Subtype_Indication (Syn_Inst, Atype); + Obj_Typ := Get_Subtype_Object (Syn_Inst, Atype); + else + Obj_Typ := null; + end if; + + Vhdl_Stmts.Synth_Assignment_Prefix (Syn_Inst, Get_Name (Decl), + Base, Typ, Off, Dyn); + pragma Assert (Dyn.Voff = No_Net); + if Base.Val.Kind = Value_Net then + -- Object is a net if it is not writable. Extract the + -- bits for the alias. + Res := Create_Value_Net + (Build2_Extract (Ctxt, Base.Val.N, Off.Net_Off, Typ.W), + Typ); + else + Res := Create_Value_Alias (Base, Off, Typ); + end if; + if Obj_Typ /= null then + Res := Synth_Subtype_Conversion (Ctxt, Res, Obj_Typ, True, Decl); + end if; + Create_Object (Syn_Inst, Decl, Res); + end Synth_Object_Alias_Declaration; + + procedure Synth_Declaration (Syn_Inst : Synth_Instance_Acc; + Decl : Node; + Is_Subprg : Boolean; + Last_Type : in out Node) is + begin + case Get_Kind (Decl) is + when Iir_Kind_Variable_Declaration => + Synth_Variable_Declaration (Syn_Inst, Decl, Is_Subprg); + when Iir_Kind_Interface_Variable_Declaration => + -- Ignore default value. + Create_Wire_Object (Syn_Inst, Wire_Variable, Decl); + Create_Var_Wire (Syn_Inst, Decl, No_Valtyp); + when Iir_Kind_Constant_Declaration => + Synth_Constant_Declaration (Syn_Inst, Decl, Is_Subprg, Last_Type); + when Iir_Kind_Signal_Declaration => + pragma Assert (not Is_Subprg); + Synth_Signal_Declaration (Syn_Inst, Decl); + when Iir_Kind_Object_Alias_Declaration => + Synth_Object_Alias_Declaration (Syn_Inst, Decl); + when Iir_Kind_Anonymous_Signal_Declaration => + -- Anonymous signals created by inertial associations are + -- simply ignored. + null; + when Iir_Kind_Procedure_Declaration + | Iir_Kind_Function_Declaration => + Synth_Subprogram_Declaration (Syn_Inst, Decl); + when Iir_Kind_Procedure_Body + | Iir_Kind_Function_Body => + null; + when Iir_Kind_Non_Object_Alias_Declaration => + null; + when Iir_Kind_Attribute_Declaration => + -- Nothing to do: the type is a type_mark, not a subtype + -- indication. + null; + when Iir_Kind_Attribute_Specification => + Synth_Attribute_Specification (Syn_Inst, Decl); + when Iir_Kind_Type_Declaration => + Synth_Type_Definition (Syn_Inst, Get_Type_Definition (Decl)); + when Iir_Kind_Anonymous_Type_Declaration => + Synth_Anonymous_Type_Definition + (Syn_Inst, Get_Type_Definition (Decl), + Get_Subtype_Definition (Decl)); + when Iir_Kind_Subtype_Declaration => + Synth_Declaration_Type (Syn_Inst, Decl); + when Iir_Kind_Component_Declaration => + null; + when Iir_Kind_File_Declaration => + declare + F : File_Index; + Res : Valtyp; + Obj_Typ : Type_Acc; + begin + F := Synth.Vhdl_Files.Elaborate_File_Declaration + (Syn_Inst, Decl); + Obj_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Decl)); + Res := Create_Value_File (Obj_Typ, F); + Create_Object (Syn_Inst, Decl, Res); + end; + when Iir_Kind_Protected_Type_Body => + null; + when Iir_Kind_Psl_Default_Clock => + -- Ignored; directly used by PSL directives. + null; + when Iir_Kind_Use_Clause => + null; + when Iir_Kind_Configuration_Specification => + null; + when Iir_Kind_Signal_Attribute_Declaration => + -- Not supported by synthesis. + null; + when others => + Vhdl.Errors.Error_Kind ("synth_declaration", Decl); + end case; + end Synth_Declaration; + + procedure Synth_Declarations (Syn_Inst : Synth_Instance_Acc; + Decls : Iir; + Is_Subprg : Boolean := False) + is + Decl : Node; + Last_Type : Node; + begin + Last_Type := Null_Node; + Decl := Decls; + while Is_Valid (Decl) loop + Synth_Declaration (Syn_Inst, Decl, Is_Subprg, Last_Type); + + exit when Is_Error (Syn_Inst); + + Decl := Get_Chain (Decl); + end loop; + end Synth_Declarations; + + procedure Finalize_Signal (Syn_Inst : Synth_Instance_Acc; Decl : Node) + is + use Netlists.Gates; + Vt : Valtyp; + Gate_Net : Net; + Gate : Instance; + Drv : Net; + Def_Val : Net; + begin + Vt := Get_Value (Syn_Inst, Decl); + if Vt = No_Valtyp then + pragma Assert (Is_Error (Syn_Inst)); + return; + end if; + if Vt.Val.Kind = Value_Net then + -- Could be a net for in ports. + return; + end if; + + Finalize_Assignment (Get_Build (Syn_Inst), Vt.Val.W); + + Gate_Net := Get_Wire_Gate (Vt.Val.W); + Gate := Get_Net_Parent (Gate_Net); + case Get_Id (Gate) is + when Id_Signal + | Id_Output + | Id_Inout => + Drv := Get_Input_Net (Gate, 0); + Def_Val := No_Net; + when Id_Isignal + | Id_Ioutput + | Id_Iinout => + Drv := Get_Input_Net (Gate, 0); + Def_Val := Get_Input_Net (Gate, 1); + when others => + -- Todo: output ? + raise Internal_Error; + end case; + if Drv = No_Net then + if Is_Connected (Get_Output (Gate, 0)) then + -- No warning if the signal is not used. + -- TODO: maybe simply remove it. + if Def_Val = No_Net then + Warning_Msg_Synth + (+Decl, "%n is never assigned and has no default value", + (1 => +Decl)); + else + Warning_Msg_Synth (+Decl, "%n is never assigned", (1 => +Decl)); + end if; + end if; + if Def_Val = No_Net then + Def_Val := Build_Const_X (Get_Build (Syn_Inst), + Get_Width (Gate_Net)); + end if; + Connect (Get_Input (Gate, 0), Def_Val); + end if; + + Free_Wire (Vt.Val.W); + end Finalize_Signal; + + procedure Finalize_Declaration + (Syn_Inst : Synth_Instance_Acc; Decl : Node; Is_Subprg : Boolean) is + begin + case Get_Kind (Decl) is + when Iir_Kind_Variable_Declaration + | Iir_Kind_Interface_Variable_Declaration => + if not Get_Instance_Const (Syn_Inst) then + declare + Vt : constant Valtyp := Get_Value (Syn_Inst, Decl); + begin + if Vt /= No_Valtyp + and then Vt.Val.Kind = Value_Wire + then + Finalize_Assignment (Get_Build (Syn_Inst), Vt.Val.W); + Free_Wire (Vt.Val.W); + end if; + end; + end if; + when Iir_Kind_Constant_Declaration => + null; + when Iir_Kind_Signal_Declaration + | Iir_Kind_Interface_Signal_Declaration => + pragma Assert (not Is_Subprg); + Finalize_Signal (Syn_Inst, Decl); + when Iir_Kind_Anonymous_Signal_Declaration => + null; + when Iir_Kind_Object_Alias_Declaration => + null; + when Iir_Kind_Procedure_Declaration + | Iir_Kind_Function_Declaration => + null; + when Iir_Kind_Procedure_Body + | Iir_Kind_Function_Body => + null; + when Iir_Kind_Non_Object_Alias_Declaration => + null; + when Iir_Kind_Attribute_Declaration => + null; + when Iir_Kind_Attribute_Specification => + null; + when Iir_Kind_Type_Declaration => + null; + when Iir_Kind_Anonymous_Type_Declaration => + null; + when Iir_Kind_Subtype_Declaration => + null; + when Iir_Kind_Component_Declaration => + null; + when Iir_Kind_File_Declaration => + null; + when Iir_Kind_Configuration_Specification => + null; + when Iir_Kind_Psl_Default_Clock => + -- Ignored; directly used by PSL directives. + null; + when Iir_Kind_Signal_Attribute_Declaration => + -- Not supported by synthesis. + null; + when others => + Vhdl.Errors.Error_Kind ("finalize_declaration", Decl); + end case; + end Finalize_Declaration; + + procedure Finalize_Declarations (Syn_Inst : Synth_Instance_Acc; + Decls : Iir; + Is_Subprg : Boolean := False) + is + Decl : Iir; + begin + Decl := Decls; + while Is_Valid (Decl) loop + Finalize_Declaration (Syn_Inst, Decl, Is_Subprg); + + Decl := Get_Chain (Decl); + end loop; + end Finalize_Declarations; +end Synth.Vhdl_Decls; diff --git a/src/synth/synth-vhdl_decls.ads b/src/synth/synth-vhdl_decls.ads new file mode 100644 index 000000000..fa1569430 --- /dev/null +++ b/src/synth/synth-vhdl_decls.ads @@ -0,0 +1,79 @@ +-- Create declarations for synthesis. +-- Copyright (C) 2017 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with Vhdl.Nodes; use Vhdl.Nodes; + +with Netlists; use Netlists; +with Synth.Vhdl_Context; use Synth.Vhdl_Context; +with Synth.Objtypes; use Synth.Objtypes; + +package Synth.Vhdl_Decls is + -- Return the Param_Type for ATYPE. + function Type_To_Param_Type (Atype : Node) return Param_Type; + + -- Convert MT to a Pval. + function Memtyp_To_Pval (Mt : Memtyp) return Pval; + + -- Get the type of DECL iff it is standalone (not an already existing + -- subtype). + function Get_Declaration_Type (Decl : Node) return Node; + + -- True if the element subtype indication of ATYPE needs to be created. + function Has_Element_Subtype_Indication (Atype : Node) return Boolean; + + function Synth_Array_Subtype_Indication + (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc; + + procedure Synth_Subtype_Indication + (Syn_Inst : Synth_Instance_Acc; Atype : Node); + function Synth_Subtype_Indication + (Syn_Inst : Synth_Instance_Acc; Atype : Node) return Type_Acc; + + -- Elaborate the type of DECL. + procedure Synth_Declaration_Type + (Syn_Inst : Synth_Instance_Acc; Decl : Node); + + procedure Synth_Declaration (Syn_Inst : Synth_Instance_Acc; + Decl : Node; + Is_Subprg : Boolean; + Last_Type : in out Node); + + procedure Synth_Declarations (Syn_Inst : Synth_Instance_Acc; + Decls : Iir; + Is_Subprg : Boolean := False); + + procedure Finalize_Declaration (Syn_Inst : Synth_Instance_Acc; + Decl : Iir; + Is_Subprg : Boolean); + procedure Finalize_Declarations (Syn_Inst : Synth_Instance_Acc; + Decls : Iir; + Is_Subprg : Boolean := False); + + procedure Synth_Package_Declaration + (Parent_Inst : Synth_Instance_Acc; Pkg : Node); + procedure Synth_Package_Body + (Parent_Inst : Synth_Instance_Acc; Pkg : Node; Bod : Node); + + procedure Synth_Generics_Association (Sub_Inst : Synth_Instance_Acc; + Syn_Inst : Synth_Instance_Acc; + Inter_Chain : Node; + Assoc_Chain : Node); + + procedure Synth_Package_Instantiation + (Parent_Inst : Synth_Instance_Acc; Pkg : Node); +end Synth.Vhdl_Decls; diff --git a/src/synth/synth-vhdl_expr.adb b/src/synth/synth-vhdl_expr.adb new file mode 100644 index 000000000..9b2072865 --- /dev/null +++ b/src/synth/synth-vhdl_expr.adb @@ -0,0 +1,2572 @@ +-- Expressions synthesis. +-- Copyright (C) 2017 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with Types_Utils; use Types_Utils; +with Name_Table; +with Std_Names; +with Str_Table; +with Mutils; use Mutils; +with Errorout; use Errorout; + +with Vhdl.Types; +with Vhdl.Ieee.Std_Logic_1164; use Vhdl.Ieee.Std_Logic_1164; +with Vhdl.Std_Package; +with Vhdl.Errors; use Vhdl.Errors; +with Vhdl.Utils; use Vhdl.Utils; +with Vhdl.Evaluation; use Vhdl.Evaluation; +with Vhdl.Annotations; use Vhdl.Annotations; + +with PSL.Nodes; +with PSL.Errors; + +with Netlists.Gates; use Netlists.Gates; +with Netlists.Folds; use Netlists.Folds; +with Netlists.Utils; use Netlists.Utils; +with Netlists.Locations; + +with Synth.Memtype; use Synth.Memtype; +with Synth.Errors; use Synth.Errors; +with Synth.Vhdl_Environment; +with Synth.Vhdl_Decls; +with Synth.Vhdl_Stmts; use Synth.Vhdl_Stmts; +with Synth.Vhdl_Oper; use Synth.Vhdl_Oper; +with Synth.Vhdl_Heap; use Synth.Vhdl_Heap; +with Synth.Debugger; +with Synth.Vhdl_Aggr; + +with Grt.Types; +with Grt.To_Strings; + +package body Synth.Vhdl_Expr is + function Synth_Name (Syn_Inst : Synth_Instance_Acc; Name : Node) + return Valtyp; + + procedure Set_Location (N : Net; Loc : Node) + renames Synth.Source.Set_Location; + + function Get_Value_Memtyp (V : Valtyp) return Memtyp is + begin + case V.Val.Kind is + when Value_Memory => + return (V.Typ, V.Val.Mem); + when Value_Const => + return Get_Memtyp (V); + when Value_Wire => + return Synth.Vhdl_Environment.Env.Get_Static_Wire (V.Val.W); + when Value_Alias => + declare + Res : Memtyp; + begin + Res := Get_Value_Memtyp ((V.Val.A_Typ, V.Val.A_Obj)); + return (V.Typ, Res.Mem + V.Val.A_Off.Mem_Off); + end; + when others => + raise Internal_Error; + end case; + end Get_Value_Memtyp; + + function Get_Static_Discrete (V : Valtyp) return Int64 is + begin + case V.Val.Kind is + when Value_Memory => + return Read_Discrete (V); + when Value_Const => + return Read_Discrete (Get_Memtyp (V)); + when Value_Wire => + return Read_Discrete + (Synth.Vhdl_Environment.Env.Get_Static_Wire (V.Val.W)); + when others => + raise Internal_Error; + end case; + end Get_Static_Discrete; + + function Is_Positive (V : Valtyp) return Boolean + is + N : Net; + Inst : Instance; + begin + pragma Assert (V.Typ.Kind = Type_Discrete); + case V.Val.Kind is + when Value_Const + | Value_Memory => + return Read_Discrete (Get_Memtyp (V)) >= 0; + when Value_Net => + N := V.Val.N; + when Value_Wire => + if Synth.Vhdl_Environment.Env.Is_Static_Wire (V.Val.W) then + return Read_Discrete + (Synth.Vhdl_Environment.Env.Get_Static_Wire (V.Val.W)) >= 0; + else + return False; + end if; + when others => + raise Internal_Error; + end case; + Inst := Get_Net_Parent (N); + case Get_Id (Inst) is + when Id_Uextend + | Id_Const_UB32 => + return True; + when others => + -- Be conservative. + return False; + end case; + end Is_Positive; + + procedure From_Std_Logic (Enum : Int64; Val : out Uns32; Zx : out Uns32) is + begin + case Enum is + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_0_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_L_Pos => + Val := 0; + Zx := 0; + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_1_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_H_Pos => + Val := 1; + Zx := 0; + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_U_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_X_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_D_Pos => + Val := 1; + Zx := 1; + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_Z_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_W_Pos => + Val := 0; + Zx := 1; + when others => + -- Only 9 values. + raise Internal_Error; + end case; + end From_Std_Logic; + + procedure From_Bit (Enum : Int64; Val : out Uns32) is + begin + if Enum = 0 then + Val := 0; + elsif Enum = 1 then + Val := 1; + else + raise Internal_Error; + end if; + end From_Bit; + + procedure To_Logic + (Enum : Int64; Etype : Type_Acc; Val : out Uns32; Zx : out Uns32) is + begin + if Etype = Logic_Type then + pragma Assert (Etype.Kind = Type_Logic); + From_Std_Logic (Enum, Val, Zx); + elsif Etype = Boolean_Type or Etype = Bit_Type then + pragma Assert (Etype.Kind = Type_Bit); + From_Bit (Enum, Val); + Zx := 0; + else + raise Internal_Error; + end if; + end To_Logic; + + procedure Uns2logvec (Val : Uns64; + W : Width; + Vec : in out Logvec_Array; + Off : in out Uns32) is + begin + if W = 0 then + return; + end if; + + for I in 0 .. W - 1 loop + declare + B : constant Uns32 := Uns32 (Shift_Right (Val, Natural (I)) and 1); + Idx : constant Digit_Index := Digit_Index (Off / 32); + Pos : constant Natural := Natural (Off mod 32); + begin + Vec (Idx).Val := Vec (Idx).Val or Shift_Left (B, Pos); + end; + Off := Off + 1; + end loop; + end Uns2logvec; + + procedure Bit2logvec (Val : Uns32; + Vec : in out Logvec_Array; + Off : in out Uns32) + is + pragma Assert (Val <= 1); + Idx : constant Digit_Index := Digit_Index (Off / 32); + Pos : constant Natural := Natural (Off mod 32); + Va : Uns32; + begin + Va := Shift_Left (Val, Pos); + Vec (Idx).Val := Vec (Idx).Val or Va; + Vec (Idx).Zx := 0; + Off := Off + 1; + end Bit2logvec; + + procedure Logic2logvec (Val : Int64; + Vec : in out Logvec_Array; + Off : in out Uns32; + Has_Zx : in out Boolean) + is + pragma Assert (Val <= 8); + Idx : constant Digit_Index := Digit_Index (Off / 32); + Pos : constant Natural := Natural (Off mod 32); + Va : Uns32; + Zx : Uns32; + begin + From_Std_Logic (Val, Va, Zx); + Has_Zx := Has_Zx or Zx /= 0; + 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 Logic2logvec; + + procedure Value2logvec (Mem : Memory_Ptr; + Typ : Type_Acc; + Off : in out Uns32; + W : in out Width; + Vec : in out Logvec_Array; + Vec_Off : in out Uns32; + Has_Zx : in out Boolean) is + begin + if Off >= Typ.W then + -- Offset not yet reached. + Off := Off - Typ.W; + return; + end if; + if W = 0 then + return; + end if; + + case Typ.Kind is + when Type_Bit => + -- Scalar bits cannot be cut. + pragma Assert (Off = 0 and W >= Typ.W); + Bit2logvec (Uns32 (Read_U8 (Mem)), Vec, Vec_Off); + W := W - Typ.W; + when Type_Logic => + -- Scalar bits cannot be cut. + pragma Assert (Off = 0 and W >= Typ.W); + Logic2logvec (Int64 (Read_U8 (Mem)), Vec, Vec_Off, Has_Zx); + W := W - Typ.W; + when Type_Discrete => + -- Scalar bits cannot be cut. + pragma Assert (Off = 0 and W >= Typ.W); + Uns2logvec (To_Uns64 (Read_Discrete (Memtyp'(Typ, Mem))), + Typ.W, Vec, Vec_Off); + W := W - Typ.W; + when Type_Float => + -- Fp64 is for sure 64 bits. Assume the endianness of floats is + -- the same as integers endianness. + -- Scalar bits cannot be cut. + pragma Assert (Off = 0 and W >= Typ.W); + Uns2logvec (To_Uns64 (Read_Fp64 (Mem)), 64, Vec, Vec_Off); + W := W - Typ.W; + when Type_Vector => + declare + Vlen : Uns32; + begin + Vlen := Uns32 (Vec_Length (Typ)); + pragma Assert (Off < Vlen); + pragma Assert (Vlen > 0); + + if Vlen > Off + W then + Vlen := Off + W; + end if; + case Typ.Vec_El.Kind is + when Type_Bit => + -- TODO: optimize off mod 32 = 0. + for I in reverse Off + 1 .. Vlen loop + Bit2logvec (Uns32 (Read_U8 (Mem + Size_Type (I - 1))), + Vec, Vec_Off); + end loop; + when Type_Logic => + for I in reverse Off + 1 .. Vlen loop + Logic2logvec + (Int64 (Read_U8 (Mem + Size_Type (I - 1))), + Vec, Vec_Off, Has_Zx); + end loop; + when others => + raise Internal_Error; + end case; + W := W - (Vlen - Off); + Off := 0; + end; + when Type_Array => + declare + Alen : constant Iir_Index32 := Get_Array_Flat_Length (Typ); + El_Typ : constant Type_Acc := Typ.Arr_El; + begin + for I in reverse 1 .. Alen loop + Value2logvec (Mem + Size_Type (I - 1) * El_Typ.Sz, El_Typ, + Off, W, Vec, Vec_Off, Has_Zx); + exit when W = 0; + end loop; + end; + when Type_Record => + for I in Typ.Rec.E'Range loop + Value2logvec (Mem + Typ.Rec.E (I).Moff, Typ.Rec.E (I).Typ, + Off, W, Vec, Vec_Off, Has_Zx); + exit when W = 0; + end loop; + when others => + raise Internal_Error; + end case; + end Value2logvec; + + procedure Value2logvec (Val : Memtyp; + Off : Uns32; + W : Width; + Vec : in out Logvec_Array; + Vec_Off : in out Uns32; + Has_Zx : in out Boolean) + is + Off1 : Uns32; + W1 : Width; + begin + Off1 := Off; + W1 := W; + Value2logvec (Val.Mem, Val.Typ, Off1, W1, Vec, Vec_Off, Has_Zx); + pragma Assert (Off1 = 0); + pragma Assert (W1 = 0); + end Value2logvec; + + -- Resize for a discrete value. + function Synth_Resize + (Ctxt : Context_Acc; Val : Valtyp; W : Width; Loc : Node) return Net + is + Wn : constant Width := Val.Typ.W; + N : Net; + Res : Net; + V : Int64; + begin + if Is_Static (Val.Val) + and then Wn /= W + then + -- Optimization: resize directly. + V := Read_Discrete (Val); + if Val.Typ.Drange.Is_Signed then + Res := Build2_Const_Int (Ctxt, V, W); + else + Res := Build2_Const_Uns (Ctxt, To_Uns64 (V), W); + end if; + Set_Location (Res, Loc); + return Res; + end if; + + N := Get_Net (Ctxt, Val); + if Wn > W then + return Build2_Trunc (Ctxt, Id_Utrunc, N, W, Get_Location (Loc)); + elsif Wn < W then + if Val.Typ.Drange.Is_Signed then + Res := Build_Extend (Ctxt, Id_Sextend, N, W); + else + Res := Build_Extend (Ctxt, Id_Uextend, N, W); + end if; + Set_Location (Res, Loc); + return Res; + else + return N; + end if; + end Synth_Resize; + + procedure Concat_Array (Ctxt : Context_Acc; Arr : in out Net_Array) + is + Last : Int32; + Idx, New_Idx : Int32; + begin + Last := Arr'Last; + while Last > Arr'First loop + Idx := Arr'First; + New_Idx := Arr'First - 1; + while Idx <= Last loop + -- Gather at most 4 nets. + New_Idx := New_Idx + 1; + + if Idx = Last then + Arr (New_Idx) := Arr (Idx); + Idx := Idx + 1; + elsif Idx + 1 = Last then + Arr (New_Idx) := Build_Concat2 + (Ctxt, Arr (Idx), Arr (Idx + 1)); + Idx := Idx + 2; + elsif Idx + 2 = Last then + Arr (New_Idx) := Build_Concat3 + (Ctxt, Arr (Idx), Arr (Idx + 1), Arr (Idx + 2)); + Idx := Idx + 3; + else + Arr (New_Idx) := Build_Concat4 + (Ctxt, + Arr (Idx), Arr (Idx + 1), Arr (Idx + 2), Arr (Idx + 3)); + Idx := Idx + 4; + end if; + end loop; + Last := New_Idx; + end loop; + end Concat_Array; + + procedure Concat_Array + (Ctxt : Context_Acc; Arr : in out Net_Array; N : out Net) is + begin + Concat_Array (Ctxt, Arr); + N := Arr (Arr'First); + end Concat_Array; + + function Build_Discrete_Range_Type + (L : Int64; R : Int64; Dir : Direction_Type) return Discrete_Range_Type is + begin + return (Dir => Dir, + Left => L, + Right => R, + Is_Signed => L < 0 or R < 0); + end Build_Discrete_Range_Type; + + function Synth_Discrete_Range_Expression + (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Discrete_Range_Type + is + L, R : Valtyp; + Lval, Rval : Int64; + begin + -- Static values. + L := Synth_Expression_With_Basetype (Syn_Inst, Get_Left_Limit (Rng)); + R := Synth_Expression_With_Basetype (Syn_Inst, Get_Right_Limit (Rng)); + Strip_Const (L); + Strip_Const (R); + + if not (Is_Static (L.Val) and Is_Static (R.Val)) then + Error_Msg_Synth (+Rng, "limits of range are not constant"); + Set_Error (Syn_Inst); + return (Dir => Get_Direction (Rng), + Left => 0, + Right => 0, + Is_Signed => False); + end if; + + Lval := Read_Discrete (L); + Rval := Read_Discrete (R); + return Build_Discrete_Range_Type (Lval, Rval, Get_Direction (Rng)); + end Synth_Discrete_Range_Expression; + + function Synth_Float_Range_Expression + (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Float_Range_Type + is + L, R : Valtyp; + begin + -- Static values (so no enable). + L := Synth_Expression (Syn_Inst, Get_Left_Limit (Rng)); + R := Synth_Expression (Syn_Inst, Get_Right_Limit (Rng)); + return (Get_Direction (Rng), Read_Fp64 (L), Read_Fp64 (R)); + end Synth_Float_Range_Expression; + + -- Return the type of EXPR without evaluating it. + function Synth_Type_Of_Object (Syn_Inst : Synth_Instance_Acc; Expr : Node) + return Type_Acc is + begin + case Get_Kind (Expr) is + when Iir_Kinds_Object_Declaration => + declare + Val : constant Valtyp := Get_Value (Syn_Inst, Expr); + begin + return Val.Typ; + end; + when Iir_Kind_Simple_Name => + return Synth_Type_Of_Object (Syn_Inst, Get_Named_Entity (Expr)); + when Iir_Kind_Slice_Name => + declare + Pfx_Typ : Type_Acc; + Pfx_Bnd : Bound_Type; + El_Typ : Type_Acc; + Res_Bnd : Bound_Type; + Sl_Voff : Net; + Sl_Off : Value_Offsets; + begin + Pfx_Typ := Synth_Type_Of_Object (Syn_Inst, Get_Prefix (Expr)); + Get_Onedimensional_Array_Bounds (Pfx_Typ, Pfx_Bnd, El_Typ); + Synth_Slice_Suffix (Syn_Inst, Expr, Pfx_Bnd, El_Typ, + Res_Bnd, Sl_Voff, Sl_Off); + + if Sl_Voff /= No_Net then + raise Internal_Error; + end if; + return Create_Onedimensional_Array_Subtype (Pfx_Typ, Res_Bnd); + end; + when Iir_Kind_Indexed_Name => + declare + Pfx_Typ : Type_Acc; + begin + Pfx_Typ := Synth_Type_Of_Object (Syn_Inst, Get_Prefix (Expr)); + return Get_Array_Element (Pfx_Typ); + end; + when Iir_Kind_Selected_Element => + declare + Idx : constant Iir_Index32 := + Get_Element_Position (Get_Named_Entity (Expr)); + Pfx_Typ : Type_Acc; + begin + Pfx_Typ := Synth_Type_Of_Object (Syn_Inst, Get_Prefix (Expr)); + return Pfx_Typ.Rec.E (Idx + 1).Typ; + end; + + when Iir_Kind_Implicit_Dereference + | Iir_Kind_Dereference => + declare + Val : Valtyp; + Res : Valtyp; + begin + -- Maybe do not dereference it if its type is known ? + Val := Synth_Expression (Syn_Inst, Get_Prefix (Expr)); + Res := Vhdl_Heap.Synth_Dereference (Read_Access (Val)); + return Res.Typ; + end; + + when Iir_Kind_String_Literal8 => + -- TODO: the value should be computed (once) and its type + -- returned. + return Synth.Vhdl_Decls.Synth_Subtype_Indication + (Syn_Inst, Get_Type (Expr)); + + when others => + Vhdl.Errors.Error_Kind ("synth_type_of_object", Expr); + end case; + return null; + end Synth_Type_Of_Object; + + function Synth_Array_Attribute (Syn_Inst : Synth_Instance_Acc; Attr : Node) + return Bound_Type + is + Prefix_Name : constant Iir := Get_Prefix (Attr); + Prefix : constant Iir := Strip_Denoting_Name (Prefix_Name); + Dim : constant Natural := + Vhdl.Evaluation.Eval_Attribute_Parameter_Or_1 (Attr); + Typ : Type_Acc; + Val : Valtyp; + begin + -- Prefix is an array object or an array subtype. + if Get_Kind (Prefix) = Iir_Kind_Subtype_Declaration then + -- TODO: does this cover all the cases ? + Typ := Get_Subtype_Object (Syn_Inst, Get_Subtype_Indication (Prefix)); + else + Val := Synth_Expression_With_Basetype (Syn_Inst, Prefix_Name); + Typ := Val.Typ; + end if; + + return Get_Array_Bound (Typ, Dim_Type (Dim)); + end Synth_Array_Attribute; + + procedure Synth_Discrete_Range (Syn_Inst : Synth_Instance_Acc; + Bound : Node; + Rng : out Discrete_Range_Type) is + begin + case Get_Kind (Bound) is + when Iir_Kind_Range_Expression => + Rng := Synth_Discrete_Range_Expression (Syn_Inst, Bound); + when Iir_Kind_Integer_Subtype_Definition + | Iir_Kind_Enumeration_Subtype_Definition => + if Get_Type_Declarator (Bound) /= Null_Node then + declare + Typ : Type_Acc; + begin + -- This is a named subtype, so it has been evaluated. + Typ := Get_Subtype_Object (Syn_Inst, Bound); + Rng := Typ.Drange; + end; + else + Synth_Discrete_Range + (Syn_Inst, Get_Range_Constraint (Bound), Rng); + end if; + when Iir_Kind_Range_Array_Attribute => + declare + B : Bound_Type; + begin + B := Synth_Array_Attribute (Syn_Inst, Bound); + Rng := Build_Discrete_Range_Type + (Int64 (B.Left), Int64 (B.Right), B.Dir); + end; + when Iir_Kind_Reverse_Range_Array_Attribute => + declare + B : Bound_Type; + T : Int32; + begin + B := Synth_Array_Attribute (Syn_Inst, Bound); + -- Reverse + case B.Dir is + when Dir_To => + B.Dir := Dir_Downto; + when Dir_Downto => + B.Dir := Dir_To; + end case; + T := B.Right; + B.Right := B.Left; + B.Left := T; + + Rng := Build_Discrete_Range_Type + (Int64 (B.Left), Int64 (B.Right), B.Dir); + end; + when Iir_Kinds_Denoting_Name => + -- A discrete subtype name. + Synth_Discrete_Range + (Syn_Inst, Get_Subtype_Indication (Get_Named_Entity (Bound)), + Rng); + when others => + Error_Kind ("synth_discrete_range", Bound); + end case; + end Synth_Discrete_Range; + + function Synth_Array_Bounds (Syn_Inst : Synth_Instance_Acc; + Atype : Node; + Dim : Dim_Type) return Bound_Type + is + Info : constant Sim_Info_Acc := Get_Info (Atype); + begin + if Info = null then + pragma Assert (Get_Type_Declarator (Atype) = Null_Node); + declare + Index_Type : constant Node := + Get_Index_Type (Atype, Natural (Dim - 1)); + begin + return Synth_Bounds_From_Range (Syn_Inst, Index_Type); + end; + else + declare + Bnds : constant Type_Acc := Get_Subtype_Object (Syn_Inst, Atype); + begin + case Bnds.Kind is + when Type_Vector => + pragma Assert (Dim = 1); + return Bnds.Vbound; + when Type_Array => + return Bnds.Abounds.D (Dim); + when others => + raise Internal_Error; + end case; + end; + end if; + end Synth_Array_Bounds; + + function Synth_Bounds_From_Range (Syn_Inst : Synth_Instance_Acc; + Atype : Node) return Bound_Type + is + Rng : Discrete_Range_Type; + begin + Synth_Discrete_Range (Syn_Inst, Atype, Rng); + return (Dir => Rng.Dir, + Left => Int32 (Rng.Left), Right => Int32 (Rng.Right), + Len => Get_Range_Length (Rng)); + end Synth_Bounds_From_Range; + + function Synth_Bounds_From_Length (Atype : Node; Len : Int32) + return Bound_Type + is + Rng : constant Node := Get_Range_Constraint (Atype); + Limit : Int32; + begin + Limit := Int32 (Eval_Pos (Get_Left_Limit (Rng))); + case Get_Direction (Rng) is + when Dir_To => + return (Dir => Dir_To, + Left => Limit, + Right => Limit + Len - 1, + Len => Uns32 (Len)); + when Dir_Downto => + return (Dir => Dir_Downto, + Left => Limit, + Right => Limit - Len + 1, + Len => Uns32 (Len)); + end case; + end Synth_Bounds_From_Length; + + function Synth_Simple_Aggregate (Syn_Inst : Synth_Instance_Acc; + Aggr : Node) return Valtyp + is + Aggr_Type : constant Node := Get_Type (Aggr); + pragma Assert (Get_Nbr_Dimensions (Aggr_Type) = 1); + El_Type : constant Node := Get_Element_Subtype (Aggr_Type); + El_Typ : constant Type_Acc := Get_Subtype_Object (Syn_Inst, El_Type); + Els : constant Iir_Flist := Get_Simple_Aggregate_List (Aggr); + Last : constant Natural := Flist_Last (Els); + Bnd : Bound_Type; + Bnds : Bound_Array_Acc; + Res_Type : Type_Acc; + Val : Valtyp; + Res : Valtyp; + begin + -- Allocate the result. + Bnd := Synth_Array_Bounds (Syn_Inst, Aggr_Type, 1); + pragma Assert (Bnd.Len = Uns32 (Last + 1)); + + if El_Typ.Kind in Type_Nets then + Res_Type := Create_Vector_Type (Bnd, El_Typ); + else + Bnds := Create_Bound_Array (1); + Bnds.D (1) := Bnd; + Res_Type := Create_Array_Type (Bnds, El_Typ); + end if; + + Res := Create_Value_Memory (Res_Type); + + for I in Flist_First .. Last loop + -- Elements are supposed to be static, so no need for enable. + Val := Synth_Expression_With_Type + (Syn_Inst, Get_Nth_Element (Els, I), El_Typ); + pragma Assert (Is_Static (Val.Val)); + Write_Value (Res.Val.Mem + Size_Type (I) * El_Typ.Sz, Val); + end loop; + + return Res; + end Synth_Simple_Aggregate; + + -- Change the bounds of VAL. + function Reshape_Value (Val : Valtyp; Ntype : Type_Acc) return Valtyp is + begin + case Val.Val.Kind is + when Value_Wire => + return Create_Value_Wire (Val.Val.W, Ntype); + when Value_Net => + return Create_Value_Net (Val.Val.N, Ntype); + when Value_Alias => + return Create_Value_Alias + ((Val.Val.A_Typ, Val.Val.A_Obj), Val.Val.A_Off, Ntype); + when Value_Const => + return Reshape_Value ((Val.Typ, Val.Val.C_Val), Ntype); + when Value_Memory => + return (Ntype, Val.Val); + when others => + raise Internal_Error; + end case; + end Reshape_Value; + + function Synth_Subtype_Conversion (Ctxt : Context_Acc; + Vt : Valtyp; + Dtype : Type_Acc; + Bounds : Boolean; + Loc : Source.Syn_Src) + return Valtyp + is + Vtype : constant Type_Acc := Vt.Typ; + begin + if Vt = No_Valtyp then + -- Propagate error. + return No_Valtyp; + end if; + if Dtype = Vtype then + return Vt; + end if; + + case Dtype.Kind is + when Type_Bit => + pragma Assert (Vtype.Kind = Type_Bit); + return Vt; + when Type_Logic => + pragma Assert (Vtype.Kind = Type_Logic); + return Vt; + when Type_Discrete => + pragma Assert (Vtype.Kind in Type_All_Discrete); + case Vt.Val.Kind is + when Value_Net + | Value_Wire + | Value_Alias => + if Vtype.W /= Dtype.W then + -- Truncate. + -- TODO: check overflow. + declare + N : Net; + begin + if Is_Static_Val (Vt.Val) then + return Create_Value_Discrete + (Get_Static_Discrete (Vt), Dtype); + end if; + + N := Get_Net (Ctxt, Vt); + if Vtype.Drange.Is_Signed then + N := Build2_Sresize + (Ctxt, N, Dtype.W, Get_Location (Loc)); + else + N := Build2_Uresize + (Ctxt, N, Dtype.W, Get_Location (Loc)); + end if; + return Create_Value_Net (N, Dtype); + end; + else + return Vt; + end if; + when Value_Const => + return Synth_Subtype_Conversion + (Ctxt, (Vt.Typ, Vt.Val.C_Val), Dtype, Bounds, Loc); + when Value_Memory => + -- Check for overflow. + declare + Val : constant Int64 := Read_Discrete (Vt); + begin + if not In_Range (Dtype.Drange, Val) then + Error_Msg_Synth (+Loc, "value out of range"); + return No_Valtyp; + end if; + return Create_Value_Discrete (Val, Dtype); + end; + when others => + raise Internal_Error; + end case; + when Type_Float => + pragma Assert (Vtype.Kind = Type_Float); + -- TODO: check range + return Vt; + when Type_Vector => + pragma Assert (Vtype.Kind = Type_Vector + or Vtype.Kind = Type_Slice); + if Dtype.W /= Vtype.W then + Error_Msg_Synth + (+Loc, "mismatching vector length; got %v, expect %v", + (Errorout."+" (Vtype.W), +Dtype.W)); + return No_Valtyp; + end if; + if Bounds then + return Reshape_Value (Vt, Dtype); + else + return Vt; + end if; + when Type_Slice => + -- TODO: check width + return Vt; + when Type_Array => + pragma Assert (Vtype.Kind = Type_Array); + -- Check bounds. + for I in Vtype.Abounds.D'Range loop + if Vtype.Abounds.D (I).Len /= Dtype.Abounds.D (I).Len then + Error_Msg_Synth (+Loc, "mismatching array bounds"); + return No_Valtyp; + end if; + end loop; + -- TODO: check element. + if Bounds then + return Reshape_Value (Vt, Dtype); + else + return Vt; + end if; + when Type_Unbounded_Array => + pragma Assert (Vtype.Kind = Type_Array); + return Vt; + when Type_Unbounded_Vector => + pragma Assert (Vtype.Kind = Type_Vector + or else Vtype.Kind = Type_Slice); + return Vt; + when Type_Record => + pragma Assert (Vtype.Kind = Type_Record); + -- TODO: handle elements. + return Vt; + when Type_Unbounded_Record => + pragma Assert (Vtype.Kind = Type_Record); + return Vt; + when Type_Access => + return Vt; + when Type_File + | Type_Protected => + -- No conversion expected. + -- As the subtype is identical, it is already handled by the + -- above check. + raise Internal_Error; + end case; + end Synth_Subtype_Conversion; + + function Synth_Value_Attribute (Syn_Inst : Synth_Instance_Acc; Attr : Node) + return Valtyp + is + Param : constant Node := Get_Parameter (Attr); + Etype : constant Node := Get_Type (Attr); + Btype : constant Node := Get_Base_Type (Etype); + V : Valtyp; + Dtype : Type_Acc; + begin + -- The value is supposed to be static. + V := Synth_Expression (Syn_Inst, Param); + if V = No_Valtyp then + return No_Valtyp; + end if; + + Dtype := Get_Subtype_Object (Syn_Inst, Etype); + if not Is_Static (V.Val) then + Error_Msg_Synth (+Attr, "parameter of 'value must be static"); + return No_Valtyp; + end if; + + declare + Str : constant String := Value_To_String (V); + Res_N : Node; + Val : Int64; + begin + case Get_Kind (Btype) is + when Iir_Kind_Enumeration_Type_Definition => + Res_N := Eval_Value_Attribute (Str, Etype, Attr); + Val := Int64 (Get_Enum_Pos (Res_N)); + Free_Iir (Res_N); + when Iir_Kind_Integer_Type_Definition => + Val := Int64'Value (Str); + when others => + Error_Msg_Synth (+Attr, "unhandled type for 'value"); + return No_Valtyp; + end case; + return Create_Value_Discrete (Val, Dtype); + end; + end Synth_Value_Attribute; + + function Synth_Image_Attribute_Str (Val : Valtyp; Expr_Type : Iir) + return String + is + use Grt.Types; + begin + case Get_Kind (Expr_Type) is + when Iir_Kind_Floating_Type_Definition + | Iir_Kind_Floating_Subtype_Definition => + declare + Str : String (1 .. 24); + Last : Natural; + begin + Grt.To_Strings.To_String + (Str, Last, Ghdl_F64 (Read_Fp64 (Val))); + return Str (Str'First .. Last); + end; + when Iir_Kind_Integer_Type_Definition + | Iir_Kind_Integer_Subtype_Definition => + declare + Str : String (1 .. 21); + First : Natural; + begin + Grt.To_Strings.To_String + (Str, First, Ghdl_I64 (Read_Discrete (Val))); + return Str (First .. Str'Last); + end; + when Iir_Kind_Enumeration_Type_Definition + | Iir_Kind_Enumeration_Subtype_Definition => + declare + Lits : constant Iir_Flist := + Get_Enumeration_Literal_List (Get_Base_Type (Expr_Type)); + begin + return Name_Table.Image + (Get_Identifier + (Get_Nth_Element (Lits, Natural (Read_Discrete (Val))))); + end; + when Iir_Kind_Physical_Type_Definition + | Iir_Kind_Physical_Subtype_Definition => + declare + Str : String (1 .. 21); + First : Natural; + Id : constant Name_Id := + Get_Identifier (Get_Primary_Unit (Get_Base_Type (Expr_Type))); + begin + Grt.To_Strings.To_String + (Str, First, Ghdl_I64 (Read_Discrete (Val))); + return Str (First .. Str'Last) & ' ' & Name_Table.Image (Id); + end; + when others => + Error_Kind ("execute_image_attribute", Expr_Type); + end case; + end Synth_Image_Attribute_Str; + + function String_To_Valtyp (Str : String; Styp : Type_Acc) return Valtyp + is + Len : constant Natural := Str'Length; + Bnd : Bound_Array_Acc; + Typ : Type_Acc; + Res : Valtyp; + begin + Bnd := Create_Bound_Array (1); + Bnd.D (1) := (Dir => Dir_To, Left => 1, Right => Int32 (Len), + Len => Width (Len)); + Typ := Create_Array_Type (Bnd, Styp.Uarr_El); + + Res := Create_Value_Memory (Typ); + for I in Str'Range loop + Write_U8 (Res.Val.Mem + Size_Type (I - Str'First), + Character'Pos (Str (I))); + end loop; + return Res; + end String_To_Valtyp; + + function Synth_Image_Attribute (Syn_Inst : Synth_Instance_Acc; Attr : Node) + return Valtyp + is + Param : constant Node := Get_Parameter (Attr); + Etype : constant Node := Get_Type (Attr); + V : Valtyp; + Dtype : Type_Acc; + begin + -- The parameter is expected to be static. + V := Synth_Expression (Syn_Inst, Param); + if V = No_Valtyp then + return No_Valtyp; + end if; + Dtype := Get_Subtype_Object (Syn_Inst, Etype); + if not Is_Static (V.Val) then + Error_Msg_Synth (+Attr, "parameter of 'image must be static"); + return No_Valtyp; + end if; + + Strip_Const (V); + return String_To_Valtyp + (Synth_Image_Attribute_Str (V, Get_Type (Param)), Dtype); + end Synth_Image_Attribute; + + function Synth_Instance_Name_Attribute + (Syn_Inst : Synth_Instance_Acc; Attr : Node) return Valtyp + is + Atype : constant Node := Get_Type (Attr); + Atyp : constant Type_Acc := Get_Subtype_Object (Syn_Inst, Atype); + Name : constant Path_Instance_Name_Type := + Get_Path_Instance_Name_Suffix (Attr); + begin + -- Return a truncated name, as the prefix is not completly known. + return String_To_Valtyp (Name.Suffix, Atyp); + end Synth_Instance_Name_Attribute; + + function Synth_Name (Syn_Inst : Synth_Instance_Acc; Name : Node) + return Valtyp is + begin + case Get_Kind (Name) is + when Iir_Kind_Simple_Name + | Iir_Kind_Selected_Name => + return Synth_Name (Syn_Inst, Get_Named_Entity (Name)); + when Iir_Kind_Interface_Signal_Declaration + | Iir_Kind_Variable_Declaration + | Iir_Kind_Interface_Variable_Declaration + | Iir_Kind_Signal_Declaration + | Iir_Kind_Interface_Constant_Declaration + | Iir_Kind_Constant_Declaration + | Iir_Kind_Iterator_Declaration + | Iir_Kind_Object_Alias_Declaration + | Iir_Kind_File_Declaration + | Iir_Kind_Interface_File_Declaration => + return Get_Value (Syn_Inst, Name); + when Iir_Kind_Enumeration_Literal => + declare + Typ : constant Type_Acc := + Get_Subtype_Object (Syn_Inst, Get_Type (Name)); + Res : Valtyp; + begin + Res := Create_Value_Memory (Typ); + Write_Discrete (Res, Int64 (Get_Enum_Pos (Name))); + return Res; + end; + when Iir_Kind_Unit_Declaration => + declare + Typ : constant Type_Acc := + Get_Subtype_Object (Syn_Inst, Get_Type (Name)); + begin + return Create_Value_Discrete + (Vhdl.Evaluation.Get_Physical_Value (Name), Typ); + end; + when Iir_Kind_Implicit_Dereference + | Iir_Kind_Dereference => + declare + Val : Valtyp; + begin + Val := Synth_Expression (Syn_Inst, Get_Prefix (Name)); + return Vhdl_Heap.Synth_Dereference (Read_Access (Val)); + end; + when others => + Error_Kind ("synth_name", Name); + end case; + end Synth_Name; + + -- Convert index IDX in PFX to an offset. + -- SYN_INST and LOC are used in case of error. + function Index_To_Offset + (Syn_Inst : Synth_Instance_Acc; Bnd : Bound_Type; Idx : Int64; Loc : Node) + return Value_Offsets + is + Res : Value_Offsets; + begin + if not In_Bounds (Bnd, Int32 (Idx)) then + Error_Msg_Synth (+Loc, "index not within bounds"); + Synth.Debugger.Debug_Error (Syn_Inst, Loc); + return (0, 0); + end if; + + -- The offset is from the LSB (bit 0). Bit 0 is the rightmost one. + case Bnd.Dir is + when Dir_To => + Res.Net_Off := Uns32 (Bnd.Right - Int32 (Idx)); + Res.Mem_Off := Size_Type (Int32 (Idx) - Bnd.Left); + when Dir_Downto => + Res.Net_Off := Uns32 (Int32 (Idx) - Bnd.Right); + Res.Mem_Off := Size_Type (Bnd.Left - Int32 (Idx)); + end case; + + return Res; + end Index_To_Offset; + + function Dyn_Index_To_Offset + (Ctxt : Context_Acc; Bnd : Bound_Type; Idx_Val : Valtyp; Loc : Node) + return Net + is + Idx2 : Net; + Off : Net; + Right : Net; + Wbounds : Width; + begin + Wbounds := Clog2 (Bnd.Len); + Idx2 := Synth_Resize (Ctxt, Idx_Val, Wbounds, Loc); + + if Bnd.Right = 0 and then Bnd.Dir = Dir_Downto then + -- Simple case without adjustments. + return Idx2; + end if; + + Right := Build_Const_UB32 (Ctxt, To_Uns32 (Bnd.Right), Wbounds); + Set_Location (Right, Loc); + + case Bnd.Dir is + when Dir_To => + -- L <= I <= R --> off = R - I + Off := Build_Dyadic (Ctxt, Id_Sub, Right, Idx2); + when Dir_Downto => + -- L >= I >= R --> off = I - R + Off := Build_Dyadic (Ctxt, Id_Sub, Idx2, Right); + end case; + Set_Location (Off, Loc); + return Off; + end Dyn_Index_To_Offset; + + -- Return the bounds of a one dimensional array/vector type and the + -- width of the element. + procedure Get_Onedimensional_Array_Bounds + (Typ : Type_Acc; Bnd : out Bound_Type; El_Typ : out Type_Acc) is + begin + case Typ.Kind is + when Type_Vector => + El_Typ := Typ.Vec_El; + Bnd := Typ.Vbound; + when Type_Array => + El_Typ := Typ.Arr_El; + Bnd := Typ.Abounds.D (1); + when others => + raise Internal_Error; + end case; + end Get_Onedimensional_Array_Bounds; + + function Create_Onedimensional_Array_Subtype + (Btyp : Type_Acc; Bnd : Bound_Type) return Type_Acc + is + Res : Type_Acc; + Bnds : Bound_Array_Acc; + begin + case Btyp.Kind is + when Type_Vector => + Res := Create_Vector_Type (Bnd, Btyp.Vec_El); + when Type_Unbounded_Vector => + Res := Create_Vector_Type (Bnd, Btyp.Uvec_El); + when Type_Array => + pragma Assert (Btyp.Abounds.Ndim = 1); + Bnds := Create_Bound_Array (1); + Bnds.D (1) := Bnd; + Res := Create_Array_Type (Bnds, Btyp.Arr_El); + when Type_Unbounded_Array => + pragma Assert (Btyp.Uarr_Ndim = 1); + Bnds := Create_Bound_Array (1); + Bnds.D (1) := Bnd; + Res := Create_Array_Type (Bnds, Btyp.Uarr_El); + when others => + raise Internal_Error; + end case; + return Res; + end Create_Onedimensional_Array_Subtype; + + procedure Synth_Indexed_Name (Syn_Inst : Synth_Instance_Acc; + Name : Node; + Pfx_Type : Type_Acc; + Voff : out Net; + Off : out Value_Offsets) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Indexes : constant Iir_Flist := Get_Index_List (Name); + El_Typ : constant Type_Acc := Get_Array_Element (Pfx_Type); + Idx_Expr : Node; + Idx_Val : Valtyp; + Bnd : Bound_Type; + Stride : Uns32; + Ivoff : Net; + Idx_Off : Value_Offsets; + begin + Voff := No_Net; + Off := (0, 0); + + Stride := 1; + for I in reverse Flist_First .. Flist_Last (Indexes) loop + Idx_Expr := Get_Nth_Element (Indexes, I); + + -- Use the base type as the subtype of the index is not synth-ed. + Idx_Val := Synth_Expression_With_Basetype (Syn_Inst, Idx_Expr); + if Idx_Val = No_Valtyp then + -- Propagate errorc + Voff := No_Net; + Off := (0, 0); + return; + end if; + + Strip_Const (Idx_Val); + + Bnd := Get_Array_Bound (Pfx_Type, Dim_Type (I + 1)); + + if Is_Static_Val (Idx_Val.Val) then + Idx_Off := Index_To_Offset (Syn_Inst, Bnd, + Get_Static_Discrete (Idx_Val), Name); + Off.Net_Off := Off.Net_Off + Idx_Off.Net_Off * Stride * El_Typ.W; + Off.Mem_Off := Off.Mem_Off + + Idx_Off.Mem_Off * Size_Type (Stride) * El_Typ.Sz; + else + Ivoff := Dyn_Index_To_Offset (Ctxt, Bnd, Idx_Val, Name); + Ivoff := Build_Memidx + (Get_Build (Syn_Inst), Ivoff, El_Typ.W * Stride, + Bnd.Len - 1, + Width (Clog2 (Uns64 (Stride * Bnd.Len)))); + Set_Location (Ivoff, Idx_Expr); + + if Voff = No_Net then + Voff := Ivoff; + else + Voff := Build_Addidx (Get_Build (Syn_Inst), Ivoff, Voff); + Set_Location (Voff, Idx_Expr); + end if; + end if; + + Stride := Stride * Bnd.Len; + end loop; + end Synth_Indexed_Name; + + function Is_Static (N : Net) return Boolean is + begin + case Get_Id (Get_Module (Get_Net_Parent (N))) is + when Id_Const_UB32 => + return True; + when others => + return False; + end case; + end Is_Static; + + function Get_Const (N : Net) return Int32 + is + Inst : constant Instance := Get_Net_Parent (N); + begin + case Get_Id (Get_Module (Inst)) is + when Id_Const_UB32 => + return To_Int32 (Get_Param_Uns32 (Inst, 0)); + when others => + raise Internal_Error; + end case; + end Get_Const; + + -- Decompose VAL as FACTOR * INP + ADDEND (where only INP is non-static). + procedure Decompose_Mul_Add (Val : Net; + Inp : out Net; + Factor : out Int32; + Addend : out Int32) + is + Inst : Instance; + Val_I0, Val_I1 : Net; + begin + Factor := 1; + Addend := 0; + Inp := Val; + + loop + Inst := Get_Net_Parent (Inp); + case Get_Id (Get_Module (Inst)) is + when Id_Add => + Val_I0 := Get_Input_Net (Inst, 0); + Val_I1 := Get_Input_Net (Inst, 1); + if Is_Static (Val_I0) then + Addend := Addend + Get_Const (Val_I0) * Factor; + Inp := Val_I1; + elsif Is_Static (Val_I1) then + Addend := Addend + Get_Const (Val_I1) * Factor; + Inp := Val_I0; + else + -- It's an addition, but without any constant value. + return; + end if; + when Id_Sub => + Val_I0 := Get_Input_Net (Inst, 0); + Val_I1 := Get_Input_Net (Inst, 1); + if Is_Static (Val_I1) then + Addend := Addend - Get_Const (Val_I1) * Factor; + Inp := Val_I0; + elsif Is_Static (Val_I0) then + Addend := Addend + Get_Const (Val_I0) * Factor; + Factor := -Factor; + Inp := Val_I1; + else + -- It's a substraction, but without any constant value. + return; + end if; + when Id_Smul => + Val_I0 := Get_Input_Net (Inst, 0); + Val_I1 := Get_Input_Net (Inst, 1); + if Is_Static (Val_I0) then + Factor := Factor * Get_Const (Val_I0); + Inp := Val_I1; + elsif Is_Static (Val_I1) then + Factor := Factor * Get_Const (Val_I1); + Inp := Val_I0; + else + -- A mul but without any constant value. + return; + end if; + when Id_Utrunc + | Id_Uextend => + Inp := Get_Input_Net (Inst, 0); + when others => + -- Cannot decompose it. + return; + end case; + end loop; + end Decompose_Mul_Add; + + -- Identify LEFT to/downto RIGHT as: + -- INP * STEP + WIDTH - 1 + OFF to/downto INP * STEP + OFF + procedure Synth_Extract_Dyn_Suffix (Ctxt : Context_Acc; + Loc : Node; + Pfx_Bnd : Bound_Type; + Left : Net; + Right : Net; + Inp : out Net; + Step : out Uns32; + Off : out Uns32; + Width : out Uns32) + is + L_Inp, R_Inp : Net; + L_Fac, R_Fac : Int32; + L_Add, R_Add : Int32; + begin + Inp := No_Net; + Step := 0; + Off := 0; + Width := 0; + + if Left = Right then + L_Inp := Left; + R_Inp := Right; + L_Fac := 1; + R_Fac := 1; + L_Add := 0; + R_Add := 0; + else + Decompose_Mul_Add (Left, L_Inp, L_Fac, L_Add); + Decompose_Mul_Add (Right, R_Inp, R_Fac, R_Add); + end if; + + if not Same_Net (L_Inp, R_Inp) then + Error_Msg_Synth + (+Loc, "cannot extract same variable part for dynamic slice"); + return; + end if; + Inp := L_Inp; + + if L_Fac /= R_Fac then + Error_Msg_Synth + (+Loc, "cannot extract same constant factor for dynamic slice"); + return; + end if; + if L_Fac < 0 then + Step := Uns32 (-L_Fac); + Inp := Build_Monadic (Ctxt, Id_Neg, Inp); + Set_Location (Inp, Loc); + else + Step := Uns32 (L_Fac); + end if; + + case Pfx_Bnd.Dir is + when Dir_To => + Width := Uns32 (R_Add - L_Add + 1); + Off := Uns32 (L_Add - Pfx_Bnd.Left); + when Dir_Downto => + Width := Uns32 (L_Add - R_Add + 1); + if R_Add >= Pfx_Bnd.Right then + Off := Uns32 (R_Add - Pfx_Bnd.Right); + else + -- Handle biased values. + declare + Bias : constant Uns32 := + (Uns32 (Pfx_Bnd.Right - R_Add) + Step - 1) / Step; + Bias_Net : Net; + begin + -- Add bias to INP and adjust the offset. + Bias_Net := Build2_Const_Uns + (Ctxt, Uns64 (Bias), Get_Width (Inp)); + Inp := Build_Dyadic (Ctxt, Id_Add, Inp, Bias_Net); + Set_Location (Inp, Loc); + Off := Uns32 (Int32 (Bias * Step) + R_Add - Pfx_Bnd.Right); + end; + end if; + end case; + end Synth_Extract_Dyn_Suffix; + + procedure Synth_Slice_Const_Suffix (Syn_Inst: Synth_Instance_Acc; + Expr : Node; + Name : Node; + Pfx_Bnd : Bound_Type; + L, R : Int64; + Dir : Direction_Type; + El_Typ : Type_Acc; + Res_Bnd : out Bound_Type; + Off : out Value_Offsets) + is + Is_Null : Boolean; + Len : Uns32; + begin + if Pfx_Bnd.Dir /= Dir then + Error_Msg_Synth (+Name, "direction mismatch in slice"); + Off := (0, 0); + if Dir = Dir_To then + Res_Bnd := (Dir => Dir_To, Left => 1, Right => 0, Len => 0); + else + Res_Bnd := (Dir => Dir_Downto, Left => 0, Right => 1, Len => 0); + end if; + return; + end if; + + -- Might be a null slice. + case Pfx_Bnd.Dir is + when Dir_To => + Is_Null := L > R; + when Dir_Downto => + Is_Null := L < R; + end case; + if Is_Null then + Len := 0; + Off := (0, 0); + else + if not In_Bounds (Pfx_Bnd, Int32 (L)) + or else not In_Bounds (Pfx_Bnd, Int32 (R)) + then + Error_Msg_Synth (+Name, "index not within bounds"); + Synth.Debugger.Debug_Error (Syn_Inst, Expr); + Off := (0, 0); + return; + end if; + + case Pfx_Bnd.Dir is + when Dir_To => + Len := Uns32 (R - L + 1); + Off.Net_Off := Uns32 (Pfx_Bnd.Right - Int32 (R)) * El_Typ.W; + Off.Mem_Off := Size_Type (Int32 (L) - Pfx_Bnd.Left) * El_Typ.Sz; + when Dir_Downto => + Len := Uns32 (L - R + 1); + Off.Net_Off := Uns32 (Int32 (R) - Pfx_Bnd.Right) * El_Typ.W; + Off.Mem_Off := Size_Type (Pfx_Bnd.Left - Int32 (L)) * El_Typ.Sz; + end case; + end if; + Res_Bnd := (Dir => Pfx_Bnd.Dir, + Len => Len, + Left => Int32 (L), + Right => Int32 (R)); + end Synth_Slice_Const_Suffix; + + procedure Synth_Slice_Suffix (Syn_Inst : Synth_Instance_Acc; + Name : Node; + Pfx_Bnd : Bound_Type; + El_Typ : Type_Acc; + Res_Bnd : out Bound_Type; + Inp : out Net; + Off : out Value_Offsets) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Expr : constant Node := Get_Suffix (Name); + Left, Right : Valtyp; + Dir : Direction_Type; + Step : Uns32; + Max : Uns32; + Inp_W : Width; + begin + Off := (0, 0); + Inp := No_Net; + + case Get_Kind (Expr) is + when Iir_Kind_Range_Expression => + -- As the range may be dynamic, cannot use synth_discrete_range. + Left := Synth_Expression_With_Basetype + (Syn_Inst, Get_Left_Limit (Expr)); + Right := Synth_Expression_With_Basetype + (Syn_Inst, Get_Right_Limit (Expr)); + Dir := Get_Direction (Expr); + + when Iir_Kind_Range_Array_Attribute + | Iir_Kind_Reverse_Range_Array_Attribute + | Iir_Kinds_Denoting_Name => + declare + Rng : Discrete_Range_Type; + begin + Synth_Discrete_Range (Syn_Inst, Expr, Rng); + Synth_Slice_Const_Suffix (Syn_Inst, Expr, + Name, Pfx_Bnd, + Rng.Left, Rng.Right, Rng.Dir, + El_Typ, Res_Bnd, Off); + return; + end; + when others => + Error_Msg_Synth + (+Expr, "only range expression supported for slices"); + Res_Bnd := (Dir => Dir_To, Left => 1, Right => 0, Len => 0); + return; + end case; + + if Is_Static_Val (Left.Val) and then Is_Static_Val (Right.Val) then + Synth_Slice_Const_Suffix (Syn_Inst, Expr, + Name, Pfx_Bnd, + Get_Static_Discrete (Left), + Get_Static_Discrete (Right), + Dir, + El_Typ, Res_Bnd, Off); + else + if Pfx_Bnd.Dir /= Dir then + Error_Msg_Synth (+Name, "direction mismatch in slice"); + if Dir = Dir_To then + Res_Bnd := (Dir => Dir_To, Left => 1, Right => 0, Len => 0); + else + Res_Bnd := (Dir => Dir_Downto, Left => 0, Right => 1, Len => 0); + end if; + return; + end if; + + if Is_Static (Left.Val) or else Is_Static (Right.Val) then + Error_Msg_Synth + (+Name, "left and right bounds of a slice must be " + & "either constant or dynamic"); + return; + end if; + + Synth_Extract_Dyn_Suffix + (Ctxt, Name, Pfx_Bnd, Get_Net (Ctxt, Left), Get_Net (Ctxt, Right), + Inp, Step, Off.Net_Off, Res_Bnd.Len); + if Inp = No_Net then + return; + end if; + Inp_W := Get_Width (Inp); + -- FIXME: convert range to offset. + -- Extract max from the range. + -- example: len=128 wd=8 step=8 => max=16 + -- len=8 wd=4 step=1 => max=4 + -- max so that max*step+wd <= len - off + -- max <= (len - off - wd) / step + Max := (Pfx_Bnd.Len - Off.Net_Off - Res_Bnd.Len) / Step; + if Clog2 (Uns64 (Max)) > Natural (Inp_W) then + -- The width of Inp limits the max. + Max := 2**Natural (Inp_W) - 1; + end if; + Inp := Build_Memidx + (Ctxt, Inp, Step * El_Typ.W, Max, + Inp_W + Width (Clog2 (Uns64 (Step * El_Typ.W)))); + Set_Location (Inp, Name); + end if; + end Synth_Slice_Suffix; + + -- Match: clk_signal_name'event + -- and return clk_signal_name. + function Extract_Event_Expr_Prefix (Expr : Node) return Node is + begin + if Get_Kind (Expr) = Iir_Kind_Event_Attribute then + return Get_Prefix (Expr); + else + return Null_Node; + end if; + end Extract_Event_Expr_Prefix; + + function Is_Same_Clock (Syn_Inst : Synth_Instance_Acc; + Left, Right : Node; + Clk_Left : Net) return Boolean + is + N : Net; + begin + -- Handle directly the common case (when clock is a simple name). + if Get_Kind (Left) = Iir_Kind_Simple_Name + and then Get_Kind (Right) = Iir_Kind_Simple_Name + then + return Get_Named_Entity (Left) = Get_Named_Entity (Right); + end if; + + N := Get_Net (Get_Build (Syn_Inst), Synth_Expression (Syn_Inst, Right)); + + return Same_Net (Clk_Left, N); + end Is_Same_Clock; + + -- Match: clk_signal_name = '1' | clk_signal_name = '0' + function Extract_Clock_Level + (Syn_Inst : Synth_Instance_Acc; Expr : Node; Prefix : Node) return Net + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Clk : Net; + Imp : Node; + Left, Right : Node; + Lit : Valtyp; + Lit_Type : Node; + Posedge : Boolean; + Res : Net; + begin + Clk := Get_Net (Ctxt, Synth_Expression (Syn_Inst, Prefix)); + if Get_Kind (Expr) /= Iir_Kind_Equality_Operator then + Error_Msg_Synth (+Expr, "ill-formed clock-level, '=' expected"); + Res := Build_Posedge (Ctxt, Clk); + Set_Location (Res, Expr); + return Res; + end if; + Imp := Get_Implementation (Expr); + if Get_Implicit_Definition (Imp) /= Iir_Predefined_Enum_Equality then + Error_Msg_Synth (+Expr, "ill-formed clock-level, '=' expected"); + Res := Build_Posedge (Ctxt, Clk); + Set_Location (Res, Expr); + return Res; + end if; + + Left := Get_Left (Expr); + if not Is_Same_Clock (Syn_Inst, Prefix, Left, Clk) then + Error_Msg_Synth (+Left, "clock signal name doesn't match"); + end if; + + Right := Get_Right (Expr); + Lit_Type := Get_Base_Type (Get_Type (Right)); + Lit := Synth_Expression (Syn_Inst, Right); + if Lit.Val.Kind /= Value_Memory then + Error_Msg_Synth (+Right, "clock-level is not a constant"); + Posedge := True; + else + if Lit_Type = Vhdl.Ieee.Std_Logic_1164.Std_Ulogic_Type then + case Read_U8 (Lit.Val.Mem) is + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_0_Pos => + Posedge := False; + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_1_Pos => + Posedge := True; + when others => + Error_Msg_Synth + (+Right, "clock-level must be either '0' or '1'"); + Posedge := True; + end case; + else + pragma Assert (Lit_Type = Vhdl.Std_Package.Bit_Type_Definition); + case Read_U8 (Lit.Val.Mem) is + when 0 => + Posedge := False; + when 1 => + Posedge := True; + when others => + raise Internal_Error; + end case; + end if; + end if; + if Posedge then + Res := Build_Posedge (Ctxt, Clk); + else + Res := Build_Negedge (Ctxt, Clk); + end if; + Set_Location (Res, Expr); + return Res; + end Extract_Clock_Level; + + -- Try to match: clk'event and clk = X + -- or: clk = X and clk'event + -- where X is '0' or '1'. + function Synth_Clock_Edge + (Syn_Inst : Synth_Instance_Acc; Left, Right : Node) return Net + is + Prefix : Node; + begin + -- Try with left. + Prefix := Extract_Event_Expr_Prefix (Left); + if Is_Valid (Prefix) then + return Extract_Clock_Level (Syn_Inst, Right, Prefix); + end if; + + -- Try with right. + Prefix := Extract_Event_Expr_Prefix (Right); + if Is_Valid (Prefix) then + return Extract_Clock_Level (Syn_Inst, Left, Prefix); + end if; + + return No_Net; + end Synth_Clock_Edge; + + function Synth_Type_Conversion + (Syn_Inst : Synth_Instance_Acc; Conv : Node) return Valtyp + is + Expr : constant Node := Get_Expression (Conv); + Conv_Type : constant Node := Get_Type (Conv); + Conv_Typ : constant Type_Acc := Get_Subtype_Object (Syn_Inst, Conv_Type); + Val : Valtyp; + begin + Val := Synth_Expression_With_Basetype (Syn_Inst, Expr); + if Val = No_Valtyp then + return No_Valtyp; + end if; + Strip_Const (Val); + case Get_Kind (Conv_Type) is + when Iir_Kind_Integer_Subtype_Definition => + if Val.Typ.Kind = Type_Discrete then + -- Int to int. + return Val; + elsif Val.Typ.Kind = Type_Float then + return Create_Value_Discrete + (Int64 (Read_Fp64 (Val)), Conv_Typ); + else + Error_Msg_Synth (+Conv, "unhandled type conversion (to int)"); + return No_Valtyp; + end if; + when Iir_Kind_Floating_Subtype_Definition => + if Is_Static (Val.Val) then + return Create_Value_Float + (Fp64 (Read_Discrete (Val)), Conv_Typ); + else + Error_Msg_Synth (+Conv, "unhandled type conversion (to float)"); + return No_Valtyp; + end if; + when Iir_Kind_Array_Type_Definition + | Iir_Kind_Array_Subtype_Definition => + case Conv_Typ.Kind is + when Type_Vector + | Type_Unbounded_Vector => + return Val; + when others => + Error_Msg_Synth + (+Conv, "unhandled type conversion (to array)"); + return No_Valtyp; + end case; + when Iir_Kind_Enumeration_Type_Definition + | Iir_Kind_Enumeration_Subtype_Definition => + pragma Assert (Get_Base_Type (Get_Type (Expr)) + = Get_Base_Type (Conv_Type)); + return Val; + when others => + Error_Msg_Synth (+Conv, "unhandled type conversion"); + return No_Valtyp; + end case; + end Synth_Type_Conversion; + + function Error_Ieee_Operator (Imp : Node; Loc : Node) return Boolean + is + use Std_Names; + Parent : constant Iir := Get_Parent (Imp); + begin + if Get_Kind (Parent) = Iir_Kind_Package_Declaration + and then (Get_Identifier + (Get_Library (Get_Design_File (Get_Design_Unit (Parent)))) + = Name_Ieee) + then + case Get_Identifier (Parent) is + when Name_Std_Logic_1164 + | Name_Std_Logic_Arith + | Name_Std_Logic_Signed + | Name_Std_Logic_Unsigned + | Name_Std_Logic_Misc + | Name_Numeric_Std + | Name_Numeric_Bit + | Name_Math_Real => + Error_Msg_Synth + (+Loc, "unhandled predefined IEEE operator %i", +Imp); + Error_Msg_Synth + (+Imp, " declared here"); + return True; + when others => + -- ieee 2008 packages are handled like regular packages. + null; + end case; + end if; + + return False; + end Error_Ieee_Operator; + + function Synth_String_Literal + (Syn_Inst : Synth_Instance_Acc; Str : Node; Str_Typ : Type_Acc) + return Valtyp + is + pragma Assert (Get_Kind (Str) = Iir_Kind_String_Literal8); + Id : constant String8_Id := Get_String8_Id (Str); + + Str_Type : constant Node := Get_Type (Str); + El_Type : Type_Acc; + Bounds : Bound_Type; + Bnds : Bound_Array_Acc; + Res_Type : Type_Acc; + Res : Valtyp; + Pos : Nat8; + begin + case Str_Typ.Kind is + when Type_Vector => + Bounds := Str_Typ.Vbound; + when Type_Array => + Bounds := Str_Typ.Abounds.D (1); + when Type_Unbounded_Vector + | Type_Unbounded_Array => + Bounds := Synth_Bounds_From_Length + (Get_Index_Type (Str_Type, 0), Get_String_Length (Str)); + when others => + raise Internal_Error; + end case; + + El_Type := Get_Subtype_Object (Syn_Inst, Get_Element_Subtype (Str_Type)); + if El_Type.Kind in Type_Nets then + Res_Type := Create_Vector_Type (Bounds, El_Type); + else + Bnds := Create_Bound_Array (1); + Bnds.D (1) := Bounds; + Res_Type := Create_Array_Type (Bnds, El_Type); + end if; + Res := Create_Value_Memory (Res_Type); + + -- Only U8 are handled. + pragma Assert (El_Type.Sz = 1); + + -- From left to right. + for I in 1 .. Bounds.Len loop + -- FIXME: use literal from type ?? + Pos := Str_Table.Element_String8 (Id, Pos32 (I)); + Write_U8 (Res.Val.Mem + Size_Type (I - 1), Nat8'Pos (Pos)); + end loop; + + return Res; + end Synth_String_Literal; + + -- Return the left bound if the direction of the range is LEFT_DIR. + function Synth_Low_High_Type_Attribute + (Syn_Inst : Synth_Instance_Acc; Expr : Node; Left_Dir : Direction_Type) + return Valtyp + is + Typ : Type_Acc; + R : Int64; + begin + Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Get_Prefix (Expr))); + pragma Assert (Typ.Kind = Type_Discrete); + if Typ.Drange.Dir = Left_Dir then + R := Typ.Drange.Left; + else + R := Typ.Drange.Right; + end if; + return Create_Value_Discrete (R, Typ); + end Synth_Low_High_Type_Attribute; + + function Synth_PSL_Expression + (Syn_Inst : Synth_Instance_Acc; Expr : PSL.Types.PSL_Node) return Net + is + use PSL.Types; + use PSL.Nodes; + + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Loc : constant Location_Type := Get_Location (Expr); + Res : Net; + begin + case Get_Kind (Expr) is + when N_HDL_Bool => + declare + E : constant Vhdl.Types.Vhdl_Node := Get_HDL_Node (Expr); + begin + return Get_Net (Ctxt, Synth_Expression (Syn_Inst, E)); + end; + when N_Not_Bool => + pragma Assert (Loc /= No_Location); + Res := Build_Monadic + (Ctxt, Id_Not, + Synth_PSL_Expression (Syn_Inst, Get_Boolean (Expr))); + when N_And_Bool => + pragma Assert (Loc /= No_Location); + declare + L : constant PSL_Node := Get_Left (Expr); + R : constant PSL_Node := Get_Right (Expr); + Edge : Net; + begin + -- Handle edge (as it can be in default clock). + if Get_Kind (L) in N_HDLs and then Get_Kind (R) in N_HDLs then + Edge := Synth_Clock_Edge + (Syn_Inst, Get_HDL_Node (L), Get_HDL_Node (R)); + if Edge /= No_Net then + return Edge; + end if; + end if; + if Get_Kind (R) = N_EOS then + -- It is never EOS! + Res := Build_Const_UB32 (Ctxt, 0, 1); + else + Res := Build_Dyadic (Ctxt, Id_And, + Synth_PSL_Expression (Syn_Inst, L), + Synth_PSL_Expression (Syn_Inst, R)); + end if; + end; + when N_Or_Bool => + pragma Assert (Loc /= No_Location); + Res := Build_Dyadic + (Ctxt, Id_Or, + Synth_PSL_Expression (Syn_Inst, Get_Left (Expr)), + Synth_PSL_Expression (Syn_Inst, Get_Right (Expr))); + when N_True => + Res := Build_Const_UB32 (Ctxt, 1, 1); + when N_False + | N_EOS => + Res := Build_Const_UB32 (Ctxt, 0, 1); + when others => + PSL.Errors.Error_Kind ("synth_psl_expr", Expr); + return No_Net; + end case; + Netlists.Locations.Set_Location (Get_Net_Parent (Res), Loc); + return Res; + end Synth_PSL_Expression; + + function Synth_Psl_Function_Clock + (Syn_Inst : Synth_Instance_Acc; Call : Node; Ctxt : Context_Acc) + return Net + is + Clock : Node; + Clk : Valtyp; + Clk_Net : Net; + begin + Clock := Get_Clock_Expression (Call); + if Clock /= Null_Node then + Clk := Synth_Expression (Syn_Inst, Clock); + Clk_Net := Get_Net (Ctxt, Clk); + else + Clock := Get_Default_Clock (Call); + pragma Assert (Clock /= Null_Node); + Clk_Net := Synth_PSL_Expression (Syn_Inst, Get_Psl_Boolean (Clock)); + end if; + return Clk_Net; + end Synth_Psl_Function_Clock; + + function Synth_Psl_Prev (Syn_Inst : Synth_Instance_Acc; Call : Node) + return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Count : constant Node := Get_Count_Expression (Call); + Count_Val : Valtyp; + Dff : Net; + Expr : Valtyp; + Clk_Net : Net; + Num : Int64; + begin + Expr := Synth_Expression_With_Basetype (Syn_Inst, Get_Expression (Call)); + + Clk_Net := Synth_Psl_Function_Clock (Syn_Inst, Call, Ctxt); + + if Count /= Null_Node then + Count_Val := Synth_Expression (Syn_Inst, Count); + Num := Read_Discrete (Count_Val); + pragma Assert (Num >= 1); + else + Num := 1; + end if; + + Dff := Get_Net (Ctxt, Expr); + for I in 1 .. Num loop + Dff := Build_Dff (Ctxt, Clk_Net, Dff); + Set_Location (Dff, Call); + end loop; + + return Create_Value_Net (Dff, Expr.Typ); + end Synth_Psl_Prev; + + function Synth_Psl_Stable (Syn_Inst : Synth_Instance_Acc; Call : Node) + return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + DffCurr : Net; + Dff : Net; + Expr : Valtyp; + Clk_Net : Net; + Res : Net; + begin + Expr := Synth_Expression_With_Basetype (Syn_Inst, Get_Expression (Call)); + + Clk_Net := Synth_Psl_Function_Clock (Syn_Inst, Call, Ctxt); + + DffCurr := Get_Net (Ctxt, Expr); + Set_Location (DffCurr, Call); + Dff := Build_Dff (Ctxt, Clk_Net, DffCurr); + Set_Location (Dff, Call); + + Res := Build_Compare(Ctxt, Id_Eq, DffCurr, Dff); + Set_Location (Res, Call); + + return Create_Value_Net (Res, Boolean_Type); + + end Synth_Psl_Stable; + + function Synth_Psl_Rose (Syn_Inst : Synth_Instance_Acc; Call : Node) + return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + DffCurr : Net; + Dff : Net; + NotDff : Net; + Clk_Net : Net; + Expr : Valtyp; + Res : Net; + begin + Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); + + Clk_Net := Synth_Psl_Function_Clock (Syn_Inst, Call, Ctxt); + + DffCurr := Get_Net (Ctxt, Expr); + Set_Location (DffCurr, Call); + Dff := Build_Dff (Ctxt, Clk_Net, DffCurr); + Set_Location (Dff, Call); + + NotDff := Build_Monadic (Ctxt, Id_Not, Dff); + Set_Location (NotDff, Call); + + Res := Build_Dyadic (Ctxt, Id_And, + NotDff, DffCurr); + Set_Location (Res, Call); + + return Create_Value_Net (Res, Boolean_Type); + + end Synth_Psl_Rose; + + function Synth_Psl_Fell (Syn_Inst : Synth_Instance_Acc; Call : Node) + return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + DffCurr : Net; + NotDffCurr : Net; + Dff : Net; + Clk_Net : Net; + Expr : Valtyp; + Res : Net; + begin + Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); + + Clk_Net := Synth_Psl_Function_Clock(Syn_Inst, Call, Ctxt); + + DffCurr := Get_Net (Ctxt, Expr); + Set_Location (DffCurr, Call); + Dff := Build_Dff (Ctxt, Clk_Net, DffCurr); + Set_Location (Dff, Call); + + NotDffCurr := Build_Monadic (Ctxt, Id_Not, DffCurr); + Set_Location (NotDffCurr, Call); + + Res := Build_Dyadic (Ctxt, Id_And, Dff, NotDffCurr); + Set_Location (Res, Call); + + return Create_Value_Net (Res, Boolean_Type); + + end Synth_Psl_Fell; + + function Synth_Onehot0 (Ctxt : Context_Acc; DffCurr : Net; Call : Node; + Vlen : Uns32) + return Net + is + DffZero : Net; + DffOne : Net; + DffOneHot0 : Net; + Res : Net; + begin + -- Create a constant vector of 0 for comparing + DffZero := Build2_Const_Uns(Ctxt, 0, Vlen); + + -- Create vector of value 1 for subtraction + DffOne := Build2_Const_Uns(Ctxt, 1, Vlen); + + -- Subtraction -> v - 1 + DffOneHot0 := Build_Dyadic (Ctxt, Id_Sub, DffCurr, DffOne); + Set_Location (DffOneHot0, Call); + + -- Binary And -> v & (v - 1) + DffOneHot0 := Build_Dyadic (Ctxt, Id_And, DffCurr, DffOneHot0); + Set_Location (DffOneHot0, Call); + + -- Compare with 0 -> (v & (v - 1)) == 0 + Res := Build_Compare (Ctxt, Id_Eq, DffOneHot0, DffZero); + Set_Location (Res, Call); + + return Res; + end Synth_Onehot0; + + function Synth_Psl_Onehot (Syn_Inst : Synth_Instance_Acc; Call : Node) + return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Expr : Valtyp; + DffCurr : Net; + DffCurrIsNotZero : Net; + DffOneHot0 : Net; + Res : Net; + Vlen : Uns32; + begin + -- Get parameter & its length + Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); + Vlen := Expr.Typ.W; + + -- First get net of parameter + DffCurr := Get_Net (Ctxt, Expr); + Set_Location (DffCurr, Call); + + -- Compare parameter with 0 -> v != 0 + DffCurrIsNotZero := Build_Compare (Ctxt, Id_Ne, DffCurr, + Build2_Const_Uns(Ctxt, 0, Vlen)); + Set_Location (DffCurrIsNotZero, Call); + + -- Synth onehot0 + DffOneHot0 := Synth_Onehot0 (Ctxt, DffCurr, Call, Vlen); + Set_Location (DffOneHot0, Call); + + -- Final Binary And -> (v != 0) & ((v & (v - 1)) == 0) + Res := Build_Dyadic (Ctxt, Id_And, DffOneHot0, DffCurrIsNotZero); + Set_Location (Res, Call); + + return Create_Value_Net (Res, Boolean_Type); + end Synth_Psl_Onehot; + + function Synth_Psl_Onehot0 (Syn_Inst : Synth_Instance_Acc; Call : Node) + return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Expr : Valtyp; + Vlen : Uns32; + DffCurr : Net; + Res : Net; + begin + -- Get parameter & its length + Expr := Synth_Expression (Syn_Inst, Get_Expression (Call)); + Vlen := Expr.Typ.W; + + -- First get net of parameter + DffCurr := Get_Net (Ctxt, Expr); + Set_Location (DffCurr, Call); + + -- Synth onehot0 + Res := Synth_Onehot0 (Ctxt, DffCurr, Call, Vlen); + + return Create_Value_Net (Res, Boolean_Type); + end Synth_Psl_Onehot0; + + subtype And_Or_Module_Id is Module_Id range Id_And .. Id_Or; + + function Synth_Short_Circuit (Syn_Inst : Synth_Instance_Acc; + Id : And_Or_Module_Id; + Left_Expr : Node; + Right_Expr : Node; + Typ : Type_Acc; + Expr : Node) return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Left : Valtyp; + Right : Valtyp; + Val : Int64; + N : Net; + begin + -- The short-circuit value. + case Id is + when Id_And => + Val := 0; + when Id_Or => + Val := 1; + end case; + + Left := Synth_Expression_With_Type (Syn_Inst, Left_Expr, Typ); + if Left = No_Valtyp then + -- Propagate error. + return No_Valtyp; + end if; + if Is_Static_Val (Left.Val) + and then Get_Static_Discrete (Left) = Val + then + -- Short-circuit when the left operand determines the result. + return Create_Value_Discrete (Val, Boolean_Type); + end if; + + Strip_Const (Left); + Right := Synth_Expression_With_Type (Syn_Inst, Right_Expr, Typ); + if Right = No_Valtyp then + -- Propagate error. + return No_Valtyp; + end if; + Strip_Const (Right); + + if Is_Static_Val (Right.Val) + and then Get_Static_Discrete (Right) = Val + then + -- If the right operand can determine the result, return it. + return Create_Value_Discrete (Val, Boolean_Type); + end if; + + -- Return a static value if both operands are static. + -- Note: we know the value of left if it is not constant. + if Is_Static_Val (Left.Val) and then Is_Static_Val (Right.Val) then + Val := Get_Static_Discrete (Right); + return Create_Value_Discrete (Val, Boolean_Type); + end if; + + -- Non-static result. + N := Build_Dyadic (Ctxt, Id, + Get_Net (Ctxt, Left), Get_Net (Ctxt, Right)); + Set_Location (N, Expr); + return Create_Value_Net (N, Boolean_Type); + end Synth_Short_Circuit; + + function Synth_Expression_With_Type (Syn_Inst : Synth_Instance_Acc; + Expr : Node; + Expr_Type : Type_Acc) return Valtyp is + begin + case Get_Kind (Expr) is + when Iir_Kinds_Dyadic_Operator => + declare + Imp : constant Node := Get_Implementation (Expr); + Def : constant Iir_Predefined_Functions := + Get_Implicit_Definition (Imp); + Edge : Net; + begin + -- Match clock-edge + if Def = Iir_Predefined_Boolean_And then + Edge := Synth_Clock_Edge (Syn_Inst, + Get_Left (Expr), Get_Right (Expr)); + if Edge /= No_Net then + return Create_Value_Net (Edge, Boolean_Type); + end if; + end if; + + -- Specially handle short-circuit operators. + case Def is + when Iir_Predefined_Boolean_And => + return Synth_Short_Circuit + (Syn_Inst, Id_And, Get_Left (Expr), Get_Right (Expr), + Boolean_Type, Expr); + when Iir_Predefined_Boolean_Or => + return Synth_Short_Circuit + (Syn_Inst, Id_Or, Get_Left (Expr), Get_Right (Expr), + Boolean_Type, Expr); + when Iir_Predefined_Bit_And => + return Synth_Short_Circuit + (Syn_Inst, Id_And, Get_Left (Expr), Get_Right (Expr), + Bit_Type, Expr); + when Iir_Predefined_Bit_Or => + return Synth_Short_Circuit + (Syn_Inst, Id_Or, Get_Left (Expr), Get_Right (Expr), + Bit_Type, Expr); + when Iir_Predefined_None => + if Error_Ieee_Operator (Imp, Expr) then + return No_Valtyp; + else + return Synth_User_Operator + (Syn_Inst, Get_Left (Expr), Get_Right (Expr), Expr); + end if; + when others => + return Synth_Dyadic_Operation + (Syn_Inst, Imp, + Get_Left (Expr), Get_Right (Expr), Expr); + end case; + end; + when Iir_Kinds_Monadic_Operator => + declare + Imp : constant Node := Get_Implementation (Expr); + Def : constant Iir_Predefined_Functions := + Get_Implicit_Definition (Imp); + begin + if Def = Iir_Predefined_None then + if Error_Ieee_Operator (Imp, Expr) then + return No_Valtyp; + else + return Synth_User_Operator + (Syn_Inst, Get_Operand (Expr), Null_Node, Expr); + end if; + else + return Synth_Monadic_Operation + (Syn_Inst, Imp, Get_Operand (Expr), Expr); + end if; + end; + when Iir_Kind_Simple_Name + | Iir_Kind_Selected_Name + | Iir_Kind_Interface_Signal_Declaration -- For PSL. + | Iir_Kind_Signal_Declaration -- For PSL. + | Iir_Kind_Implicit_Dereference + | Iir_Kind_Dereference => + declare + Res : Valtyp; + begin + Res := Synth_Name (Syn_Inst, Expr); + if Res.Typ.W = 0 and then Res.Val.Kind /= Value_Memory then + -- This is a null object. As nothing can be done about it, + -- returns 0. + return Create_Value_Memtyp (Create_Memory_Zero (Res.Typ)); + end if; + return Res; + end; + when Iir_Kind_Reference_Name => + -- Only used for anonymous signals in internal association. + return Synth_Expression_With_Type + (Syn_Inst, Get_Named_Entity (Expr), Expr_Type); + when Iir_Kind_Anonymous_Signal_Declaration => + return Synth_Expression_With_Type + (Syn_Inst, Get_Expression (Expr), Expr_Type); + when Iir_Kind_Indexed_Name + | Iir_Kind_Slice_Name => + declare + Base : Valtyp; + Typ : Type_Acc; + Off : Value_Offsets; + Res : Valtyp; + + Dyn : Dyn_Name; + begin + Synth_Assignment_Prefix (Syn_Inst, Expr, Base, Typ, Off, Dyn); + if Dyn.Voff = No_Net and then Is_Static (Base.Val) then + Res := Create_Value_Memory (Typ); + Copy_Memory + (Res.Val.Mem, Base.Val.Mem + Off.Mem_Off, Typ.Sz); + return Res; + end if; + return Synth_Read_Memory + (Syn_Inst, Base, Typ, Off.Net_Off, Dyn, Expr); + end; + when Iir_Kind_Selected_Element => + declare + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Idx : constant Iir_Index32 := + Get_Element_Position (Get_Named_Entity (Expr)); + Pfx : constant Node := Get_Prefix (Expr); + Res_Typ : Type_Acc; + N : Net; + Val : Valtyp; + Res : Valtyp; + begin + Val := Synth_Expression (Syn_Inst, Pfx); + Strip_Const (Val); + Res_Typ := Val.Typ.Rec.E (Idx + 1).Typ; + if Res_Typ.W = 0 and then Val.Val.Kind /= Value_Memory then + -- This is a null object. As nothing can be done about it, + -- returns 0. + return Create_Value_Memtyp (Create_Memory_Zero (Res_Typ)); + elsif Is_Static (Val.Val) then + Res := Create_Value_Memory (Res_Typ); + Copy_Memory + (Res.Val.Mem, Val.Val.Mem + Val.Typ.Rec.E (Idx + 1).Moff, + Res_Typ.Sz); + return Res; + else + N := Build_Extract + (Ctxt, Get_Net (Ctxt, Val), + Val.Typ.Rec.E (Idx + 1).Boff, Get_Type_Width (Res_Typ)); + Set_Location (N, Expr); + return Create_Value_Net (N, Res_Typ); + end if; + end; + when Iir_Kind_Character_Literal => + return Synth_Expression_With_Type + (Syn_Inst, Get_Named_Entity (Expr), Expr_Type); + when Iir_Kind_Integer_Literal => + declare + Res : Valtyp; + begin + Res := Create_Value_Memory (Expr_Type); + Write_Discrete (Res, Get_Value (Expr)); + return Res; + end; + when Iir_Kind_Floating_Point_Literal => + return Create_Value_Float (Get_Fp_Value (Expr), Expr_Type); + when Iir_Kind_Physical_Int_Literal + | Iir_Kind_Physical_Fp_Literal => + return Create_Value_Discrete + (Get_Physical_Value (Expr), Expr_Type); + when Iir_Kind_String_Literal8 => + return Synth_String_Literal (Syn_Inst, Expr, Expr_Type); + when Iir_Kind_Enumeration_Literal => + return Synth_Name (Syn_Inst, Expr); + when Iir_Kind_Type_Conversion => + return Synth_Type_Conversion (Syn_Inst, Expr); + when Iir_Kind_Qualified_Expression => + return Synth_Expression_With_Type + (Syn_Inst, Get_Expression (Expr), + Get_Subtype_Object (Syn_Inst, Get_Type (Get_Type_Mark (Expr)))); + when Iir_Kind_Function_Call => + declare + Imp : constant Node := Get_Implementation (Expr); + begin + case Get_Implicit_Definition (Imp) is + when Iir_Predefined_Pure_Functions + | Iir_Predefined_Ieee_Numeric_Std_Binary_Operators => + return Synth_Operator_Function_Call (Syn_Inst, Expr); + when Iir_Predefined_None => + return Synth_User_Function_Call (Syn_Inst, Expr); + when others => + return Synth_Predefined_Function_Call (Syn_Inst, Expr); + end case; + end; + when Iir_Kind_Aggregate => + return Synth.Vhdl_Aggr.Synth_Aggregate (Syn_Inst, Expr, Expr_Type); + when Iir_Kind_Simple_Aggregate => + return Synth_Simple_Aggregate (Syn_Inst, Expr); + when Iir_Kind_Parenthesis_Expression => + return Synth_Expression_With_Type + (Syn_Inst, Get_Expression (Expr), Expr_Type); + when Iir_Kind_Left_Array_Attribute => + declare + B : Bound_Type; + begin + B := Synth_Array_Attribute (Syn_Inst, Expr); + return Create_Value_Discrete (Int64 (B.Left), Expr_Type); + end; + when Iir_Kind_Right_Array_Attribute => + declare + B : Bound_Type; + begin + B := Synth_Array_Attribute (Syn_Inst, Expr); + return Create_Value_Discrete (Int64 (B.Right), Expr_Type); + end; + when Iir_Kind_High_Array_Attribute => + declare + B : Bound_Type; + V : Int32; + begin + B := Synth_Array_Attribute (Syn_Inst, Expr); + case B.Dir is + when Dir_To => + V := B.Right; + when Dir_Downto => + V := B.Left; + end case; + return Create_Value_Discrete (Int64 (V), Expr_Type); + end; + when Iir_Kind_Low_Array_Attribute => + declare + B : Bound_Type; + V : Int32; + begin + B := Synth_Array_Attribute (Syn_Inst, Expr); + case B.Dir is + when Dir_To => + V := B.Left; + when Dir_Downto => + V := B.Right; + end case; + return Create_Value_Discrete (Int64 (V), Expr_Type); + end; + when Iir_Kind_Length_Array_Attribute => + declare + B : Bound_Type; + begin + B := Synth_Array_Attribute (Syn_Inst, Expr); + return Create_Value_Discrete (Int64 (B.Len), Expr_Type); + end; + when Iir_Kind_Ascending_Array_Attribute => + declare + B : Bound_Type; + V : Int64; + begin + B := Synth_Array_Attribute (Syn_Inst, Expr); + case B.Dir is + when Dir_To => + V := 1; + when Dir_Downto => + V := 0; + end case; + return Create_Value_Discrete (V, Expr_Type); + end; + + when Iir_Kind_Pos_Attribute + | Iir_Kind_Val_Attribute => + declare + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Param : constant Node := Get_Parameter (Expr); + V : Valtyp; + Dtype : Type_Acc; + begin + V := Synth_Expression (Syn_Inst, Param); + Dtype := Get_Subtype_Object (Syn_Inst, Get_Type (Expr)); + -- FIXME: to be generalized. Not always as simple as a + -- subtype conversion. + return Synth_Subtype_Conversion (Ctxt, V, Dtype, False, Expr); + end; + when Iir_Kind_Low_Type_Attribute => + return Synth_Low_High_Type_Attribute (Syn_Inst, Expr, Dir_To); + when Iir_Kind_High_Type_Attribute => + return Synth_Low_High_Type_Attribute (Syn_Inst, Expr, Dir_Downto); + when Iir_Kind_Value_Attribute => + return Synth_Value_Attribute (Syn_Inst, Expr); + when Iir_Kind_Image_Attribute => + return Synth_Image_Attribute (Syn_Inst, Expr); + when Iir_Kind_Instance_Name_Attribute => + return Synth_Instance_Name_Attribute (Syn_Inst, Expr); + when Iir_Kind_Null_Literal => + return Create_Value_Access (Null_Heap_Index, Expr_Type); + when Iir_Kind_Allocator_By_Subtype => + declare + T : Type_Acc; + Acc : Heap_Index; + begin + T := Synth.Vhdl_Decls.Synth_Subtype_Indication + (Syn_Inst, Get_Subtype_Indication (Expr)); + Acc := Allocate_By_Type (T); + return Create_Value_Access (Acc, Expr_Type); + end; + when Iir_Kind_Allocator_By_Expression => + declare + V : Valtyp; + Acc : Heap_Index; + begin + V := Synth_Expression_With_Type + (Syn_Inst, Get_Expression (Expr), Expr_Type.Acc_Acc); + Acc := Allocate_By_Value (V); + return Create_Value_Access (Acc, Expr_Type); + end; + when Iir_Kind_Stable_Attribute => + Error_Msg_Synth (+Expr, "signal attribute not supported"); + return No_Valtyp; + when Iir_Kind_Psl_Prev => + return Synth_Psl_Prev (Syn_Inst, Expr); + when Iir_Kind_Psl_Stable => + return Synth_Psl_Stable (Syn_Inst, Expr); + when Iir_Kind_Psl_Rose => + return Synth_Psl_Rose(Syn_Inst, Expr); + when Iir_Kind_Psl_Fell => + return Synth_Psl_Fell(Syn_Inst, Expr); + when Iir_Kind_Psl_Onehot => + return Synth_Psl_Onehot(Syn_Inst, Expr); + when Iir_Kind_Psl_Onehot0 => + return Synth_Psl_Onehot0(Syn_Inst, Expr); + when Iir_Kind_Overflow_Literal => + Error_Msg_Synth (+Expr, "out of bound expression"); + return No_Valtyp; + when others => + Error_Kind ("synth_expression_with_type", Expr); + end case; + end Synth_Expression_With_Type; + + function Synth_Expression (Syn_Inst : Synth_Instance_Acc; Expr : Node) + return Valtyp + is + Etype : Node; + begin + Etype := Get_Type (Expr); + + case Get_Kind (Expr) is + when Iir_Kind_High_Array_Attribute + | Iir_Kind_Low_Array_Attribute + | Iir_Kind_Integer_Literal => + -- The type of this attribute is the type of the index, which is + -- not synthesized as atype (only as an index). + -- For integer_literal, the type is not really needed, and it + -- may be created by static evaluation of an array attribute. + Etype := Get_Base_Type (Etype); + when others => + null; + end case; + + return Synth_Expression_With_Type + (Syn_Inst, Expr, Get_Subtype_Object (Syn_Inst, Etype)); + end Synth_Expression; + + function Synth_Expression_With_Basetype + (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp + is + Basetype : Type_Acc; + begin + Basetype := Get_Subtype_Object + (Syn_Inst, Get_Base_Type (Get_Type (Expr))); + return Synth_Expression_With_Type (Syn_Inst, Expr, Basetype); + end Synth_Expression_With_Basetype; +end Synth.Vhdl_Expr; diff --git a/src/synth/synth-vhdl_expr.ads b/src/synth/synth-vhdl_expr.ads new file mode 100644 index 000000000..c6726732e --- /dev/null +++ b/src/synth/synth-vhdl_expr.ads @@ -0,0 +1,152 @@ +-- Expressions synthesis. +-- Copyright (C) 2017 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with Ada.Unchecked_Deallocation; + +with Types; use Types; + +with PSL.Types; +with Vhdl.Nodes; use Vhdl.Nodes; + +with Netlists; use Netlists; +with Netlists.Builders; use Netlists.Builders; + +with Synth.Source; +with Synth.Objtypes; use Synth.Objtypes; +with Synth.Values; use Synth.Values; +with Synth.Vhdl_Context; use Synth.Vhdl_Context; + +package Synth.Vhdl_Expr is + -- Perform a subtype conversion. Check constraints. + function Synth_Subtype_Conversion (Ctxt : Context_Acc; + Vt : Valtyp; + Dtype : Type_Acc; + Bounds : Boolean; + Loc : Source.Syn_Src) + return Valtyp; + + -- For a static value V, return the value. + function Get_Static_Discrete (V : Valtyp) return Int64; + + -- Return the memory (as a memtyp) of static value V. + function Get_Value_Memtyp (V : Valtyp) return Memtyp; + + -- Return True only if discrete value V is known to be positive or 0. + -- False means either not positive or unknown. + function Is_Positive (V : Valtyp) return Boolean; + + -- Return the bounds of a one dimensional array/vector type and the + -- width of the element. + procedure Get_Onedimensional_Array_Bounds + (Typ : Type_Acc; Bnd : out Bound_Type; El_Typ : out Type_Acc); + + -- Create an array subtype from bound BND. + function Create_Onedimensional_Array_Subtype + (Btyp : Type_Acc; Bnd : Bound_Type) return Type_Acc; + + procedure From_Std_Logic (Enum : Int64; Val : out Uns32; Zx : out Uns32); + procedure From_Bit (Enum : Int64; Val : out Uns32); + procedure To_Logic + (Enum : Int64; Etype : Type_Acc; Val : out Uns32; Zx : out Uns32); + + -- Try to match: clk'event and clk = X + -- or: clk = X and clk'event + -- where X is '0' or '1'. + function Synth_Clock_Edge + (Syn_Inst : Synth_Instance_Acc; Left, Right : Node) return Net; + + procedure Concat_Array + (Ctxt : Context_Acc; Arr : in out Net_Array; N : out Net); + + -- Synthesize EXPR. The expression must be self-constrained. + -- If EN is not No_Net, the execution is controlled by EN. This is used + -- for assertions and checks. + function Synth_Expression + (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp; + + -- Same as Synth_Expression, but the expression may be constrained by + -- EXPR_TYPE. + function Synth_Expression_With_Type (Syn_Inst : Synth_Instance_Acc; + Expr : Node; + Expr_Type : Type_Acc) return Valtyp; + + -- Use base type of EXPR to synthesize EXPR. Useful when the type of + -- EXPR is defined by itself or a range. + function Synth_Expression_With_Basetype (Syn_Inst : Synth_Instance_Acc; + Expr : Node) return Valtyp; + + function Synth_PSL_Expression + (Syn_Inst : Synth_Instance_Acc; Expr : PSL.Types.PSL_Node) return Net; + + function Synth_Bounds_From_Range (Syn_Inst : Synth_Instance_Acc; + Atype : Node) return Bound_Type; + + function Synth_Array_Bounds (Syn_Inst : Synth_Instance_Acc; + Atype : Node; + Dim : Dim_Type) return Bound_Type; + + function Build_Discrete_Range_Type + (L : Int64; R : Int64; Dir : Direction_Type) return Discrete_Range_Type; + function Synth_Discrete_Range_Expression + (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Discrete_Range_Type; + function Synth_Float_Range_Expression + (Syn_Inst : Synth_Instance_Acc; Rng : Node) return Float_Range_Type; + + procedure Synth_Discrete_Range (Syn_Inst : Synth_Instance_Acc; + Bound : Node; + Rng : out Discrete_Range_Type); + + procedure Synth_Slice_Suffix (Syn_Inst : Synth_Instance_Acc; + Name : Node; + Pfx_Bnd : Bound_Type; + El_Typ : Type_Acc; + Res_Bnd : out Bound_Type; + Inp : out Net; + Off : out Value_Offsets); + + -- If VOFF is No_Net then OFF is valid, if VOFF is not No_Net then + -- OFF is 0. + procedure Synth_Indexed_Name (Syn_Inst : Synth_Instance_Acc; + Name : Node; + Pfx_Type : Type_Acc; + Voff : out Net; + Off : out Value_Offsets); + + -- Return the type of EXPR (an object) without evaluating it (except when + -- needed, like bounds of a slice). + function Synth_Type_Of_Object (Syn_Inst : Synth_Instance_Acc; Expr : Node) + return Type_Acc; + + -- Conversion to logic vector. + + 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 Free_Logvec_Array is new Ada.Unchecked_Deallocation + (Logvec_Array, Logvec_Array_Acc); + + -- Convert W bits from OFF of VAL to a Logvec_Array. + -- OFF and W are offset and width in bit representation. + procedure Value2logvec (Val : Memtyp; + Off : Uns32; + W : Width; + Vec : in out Logvec_Array; + Vec_Off : in out Uns32; + Has_Zx : in out Boolean); +end Synth.Vhdl_Expr; diff --git a/src/synth/synth-vhdl_files.adb b/src/synth/synth-vhdl_files.adb index 180062e01..2300ff9f9 100644 --- a/src/synth/synth-vhdl_files.adb +++ b/src/synth/synth-vhdl_files.adb @@ -26,7 +26,7 @@ with Grt.Stdio; with Synth.Memtype; use Synth.Memtype; with Synth.Objtypes; use Synth.Objtypes; -with Synth.Expr; use Synth.Expr; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; with Synth.Errors; use Synth.Errors; package body Synth.Vhdl_Files is diff --git a/src/synth/synth-vhdl_insts.adb b/src/synth/synth-vhdl_insts.adb new file mode 100644 index 000000000..679b63312 --- /dev/null +++ b/src/synth/synth-vhdl_insts.adb @@ -0,0 +1,1752 @@ +-- Instantiation synthesis. +-- Copyright (C) 2019 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with GNAT.SHA1; + +with Types; use Types; +with Types_Utils; use Types_Utils; +with Files_Map; +with Name_Table; +with Libraries; +with Hash; use Hash; +with Dyn_Tables; +with Interning; +with Synthesis; use Synthesis; + +with Grt.Algos; + +with Netlists; use Netlists; +with Netlists.Builders; use Netlists.Builders; +with Netlists.Cleanup; +with Netlists.Memories; +with Netlists.Expands; +with Netlists.Concats; +with Netlists.Folds; + +with Vhdl.Utils; use Vhdl.Utils; +with Vhdl.Errors; +with Vhdl.Ieee.Math_Real; + +with Synth.Memtype; use Synth.Memtype; +with Synth.Objtypes; use Synth.Objtypes; +with Synth.Values; use Synth.Values; +with Synth.Vhdl_Environment; use Synth.Vhdl_Environment.Env; +with Synth.Vhdl_Stmts; use Synth.Vhdl_Stmts; +with Synth.Vhdl_Decls; use Synth.Vhdl_Decls; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; +with Synth.Source; use Synth.Source; +with Synth.Debugger; +with Synth.Vhdl_Files; +with Synth.Errors; + +package body Synth.Vhdl_Insts is + Root_Instance : Synth_Instance_Acc; + + function Mode_To_Port_Kind (Mode : Iir_Mode) return Port_Kind is + begin + case Mode is + when Iir_In_Mode => + return Port_In; + when Iir_Buffer_Mode + | Iir_Out_Mode => + return Port_Out; + when Iir_Inout_Mode => + return Port_Inout; + when Iir_Linkage_Mode + | Iir_Unknown_Mode => + raise Synth_Error; + end case; + end Mode_To_Port_Kind; + + -- Parameters that define an instance. + type Inst_Params is record + -- Declaration: either the entity or the component. + Decl : Node; + -- Implementation: the architecture or Null_Node for black boxes. + Arch : Node; + -- Configuration (Null_Node for black boxes). + Config : Node; + -- Values of generics. + Syn_Inst : Synth_Instance_Acc; + -- Encoding if the instance name. + Encoding : Name_Encoding; + end record; + + type Inst_Object is record + Decl : Node; + Arch : Node; + Config : Node; + Syn_Inst : Synth_Instance_Acc; + M : Module; + -- Encoding if the instance name. + Encoding : Name_Encoding; + end record; + + function Hash (Params : Inst_Params) return Hash_Value_Type + is + Res : Hash_Value_Type; + begin + Res := Hash_Value_Type (Params.Decl); + Res := Res xor Hash_Value_Type (Params.Arch); + Res := Res xor Hash_Value_Type (Params.Config); + -- TODO: hash generics + return Res; + end Hash; + + function Equal (Obj : Inst_Object; Params : Inst_Params) return Boolean + is + Inter : Node; + begin + if Obj.Decl /= Params.Decl + or else Obj.Arch /= Params.Arch + or else Obj.Config /= Params.Config + then + return False; + end if; + Inter := Get_Generic_Chain (Params.Decl); + while Inter /= Null_Node loop + if not Is_Equal (Get_Value (Obj.Syn_Inst, Inter), + Get_Value (Params.Syn_Inst, Inter)) + then + return False; + end if; + Inter := Get_Chain (Inter); + end loop; + + Inter := Get_Port_Chain (Params.Decl); + while Inter /= Null_Node loop + if not Is_Fully_Constrained_Type (Get_Type (Inter)) then + if not Are_Types_Equal (Get_Value (Obj.Syn_Inst, Inter).Typ, + Get_Value (Params.Syn_Inst, Inter).Typ) + then + return False; + end if; + end if; + Inter := Get_Chain (Inter); + end loop; + + return True; + end Equal; + + procedure Hash_Uns64 (C : in out GNAT.SHA1.Context; Val : Uns64) + is + V : Uns64; + S : String (1 .. 8); + begin + -- Store to S using little endianness. + V := Val; + for I in S'Range loop + S (I) := Character'Val (V and 16#ff#); + V := Shift_Right (V, 8); + end loop; + + GNAT.SHA1.Update (C, S); + end Hash_Uns64; + + procedure Hash_Memory (C : in out GNAT.SHA1.Context; + M : Memory_Ptr; + Typ : Type_Acc) + is + S : String (1 .. Natural (Typ.Sz)); + for S'Address use M (0)'Address; + pragma Import (Ada, S); + begin + GNAT.SHA1.Update (C, S); + end Hash_Memory; + + procedure Hash_Bound (C : in out GNAT.SHA1.Context; B : Bound_Type) is + begin + Hash_Uns64 (C, Direction_Type'Pos (B.Dir)); + Hash_Uns64 (C, To_Uns64 (Int64 (B.Left))); + Hash_Uns64 (C, To_Uns64 (Int64 (B.Right))); + end Hash_Bound; + + procedure Hash_Bounds (C : in out GNAT.SHA1.Context; Typ : Type_Acc) is + begin + case Typ.Kind is + when Type_Vector => + Hash_Bound (C, Typ.Vbound); + when Type_Array => + for I in Typ.Abounds.D'Range loop + Hash_Bound (C, Typ.Abounds.D (I)); + end loop; + when others => + raise Internal_Error; + end case; + end Hash_Bounds; + + procedure Hash_Const (C : in out GNAT.SHA1.Context; + Val : Value_Acc; + Typ : Type_Acc) is + begin + case Val.Kind is + when Value_Memory => + Hash_Memory (C, Val.Mem, Typ); + when Value_Const => + Hash_Const (C, Val.C_Val, Typ); + when Value_Alias => + if Val.A_Off /= (0, 0) then + raise Internal_Error; + end if; + Hash_Const (C, Val.A_Obj, Typ); + when Value_Net + | Value_Wire + | Value_File => + raise Internal_Error; + end case; + end Hash_Const; + + function Get_Source_Identifier (Decl : Node) return Name_Id + is + use Files_Map; + use Name_Table; + Loc : constant Location_Type := Get_Location (Decl); + Len : constant Natural := Get_Name_Length (Get_Identifier (Decl)); + subtype Ident_Str is String (1 .. Len); + File : Source_File_Entry; + Pos : Source_Ptr; + Buf : File_Buffer_Acc; + begin + Location_To_File_Pos (Loc, File, Pos); + Buf := Get_File_Source (File); + return Get_Identifier + (Ident_Str (Buf (Pos .. Pos + Source_Ptr (Len - 1)))); + end Get_Source_Identifier; + + function Create_Module_Name (Params : Inst_Params) return Sname + is + use GNAT.SHA1; + Decl : constant Node := Params.Decl; + Id : constant Name_Id := Get_Identifier (Decl); + Generics : constant Node := Get_Generic_Chain (Decl); + Ports : constant Node := Get_Port_Chain (Decl); + Ctxt : GNAT.SHA1.Context; + Has_Hash : Boolean; + + -- Create a buffer, store the entity name. + -- For each generic: + -- * write the value for integers. + -- * write the identifier for enumerated type with only non-extended + -- identifiers. + -- * hash all other values + -- Append the hash if any. + use Name_Table; + Id_Len : constant Natural := Get_Name_Length (Id); + Str_Len : constant Natural := Id_Len + 512; + + -- True in practice (and used to set the length of STR, but doesn't work + -- anymore with gcc/gnat 11. + -- pragma Assert (GNAT.SHA1.Hash_Length = 20); + Str : String (1 .. Str_Len + 41); + Len : Natural; + + Gen_Decl : Node; + Gen : Valtyp; + begin + Len := Id_Len; + Str (1 .. Len) := Get_Name_Ptr (Id) (1 .. Len); + + Has_Hash := False; + + case Params.Encoding is + when Name_Hash => + Ctxt := GNAT.SHA1.Initial_Context; + + Gen_Decl := Generics; + while Gen_Decl /= Null_Node loop + Gen := Get_Value (Params.Syn_Inst, Gen_Decl); + Strip_Const (Gen); + case Gen.Typ.Kind is + when Type_Discrete => + declare + S : constant String := + Uns64'Image (To_Uns64 (Read_Discrete (Gen))); + begin + if Len + S'Length > Str_Len then + Has_Hash := True; + Hash_Const (Ctxt, Gen.Val, Gen.Typ); + else + Str (Len + 1 .. Len + S'Length) := S; + pragma Assert (Str (Len + 1) = ' '); + Str (Len + 1) := '_'; -- Overwrite the space. + Len := Len + S'Length; + end if; + end; + when others => + Has_Hash := True; + Hash_Const (Ctxt, Gen.Val, Gen.Typ); + end case; + Gen_Decl := Get_Chain (Gen_Decl); + end loop; + + declare + Port_Decl : Node; + Port_Typ : Type_Acc; + begin + Port_Decl := Ports; + while Port_Decl /= Null_Node loop + if not Is_Fully_Constrained_Type (Get_Type (Port_Decl)) then + Port_Typ := Get_Value (Params.Syn_Inst, Port_Decl).Typ; + Has_Hash := True; + Hash_Bounds (Ctxt, Port_Typ); + end if; + Port_Decl := Get_Chain (Port_Decl); + end loop; + end; + if not Has_Hash and then Generics = Null_Node then + -- Simple case: same name. + -- TODO: what about two entities with the same identifier but + -- declared in two different libraries ? + -- TODO: what about extended identifiers ? + return New_Sname_User (Id, No_Sname); + end if; + + if Has_Hash then + Str (Len + 1) := '_'; + Len := Len + 1; + Str (Len + 1 .. Len + 40) := GNAT.SHA1.Digest (Ctxt); + Len := Len + 40; + end if; + + when Name_Asis + | Name_Parameters => + return New_Sname_User (Get_Source_Identifier (Decl), No_Sname); + + when Name_Index => + -- TODO. + raise Internal_Error; + end case; + + + return New_Sname_User (Get_Identifier (Str (1 .. Len)), No_Sname); + end Create_Module_Name; + + -- Create the name of an interface. + function Get_Encoded_Name_Id (Decl : Node; Enc : Name_Encoding) + return Name_Id is + begin + case Enc is + when Name_Asis + | Name_Parameters => + return Get_Source_Identifier (Decl); + when others => + return Get_Identifier (Decl); + end case; + end Get_Encoded_Name_Id; + + -- Create the name of an interface. + function Create_Inter_Name (Decl : Node; Enc : Name_Encoding) + return Sname is + begin + return New_Sname_User (Get_Encoded_Name_Id (Decl, Enc), No_Sname); + end Create_Inter_Name; + + procedure Copy_Object_Subtype (Syn_Inst : Synth_Instance_Acc; + Inter_Type : Node; + Proto_Inst : Synth_Instance_Acc) + is + Inter_Typ : Type_Acc; + begin + case Get_Kind (Inter_Type) is + when Iir_Kind_Array_Subtype_Definition => + if Synth.Vhdl_Decls.Has_Element_Subtype_Indication (Inter_Type) + then + Copy_Object_Subtype + (Syn_Inst, Get_Element_Subtype (Inter_Type), Proto_Inst); + end if; + when others => + null; + end case; + Inter_Typ := Get_Subtype_Object (Proto_Inst, Inter_Type); + Create_Subtype_Object (Syn_Inst, Inter_Type, Inter_Typ); + end Copy_Object_Subtype; + + procedure Build_Object_Subtype (Syn_Inst : Synth_Instance_Acc; + Inter : Node; + Proto_Inst : Synth_Instance_Acc) is + begin + if Get_Declaration_Type (Inter) /= Null_Node then + Copy_Object_Subtype (Syn_Inst, Get_Type (Inter), Proto_Inst); + end if; + end Build_Object_Subtype; + + -- Return the number of ports for a type. A record type create one + -- port per immediate subelement. Sub-records are not expanded. + function Count_Nbr_Ports (Typ : Type_Acc) return Port_Nbr is + begin + case Typ.Kind is + when Type_Bit + | Type_Logic + | Type_Discrete + | Type_Float + | Type_Vector + | Type_Unbounded_Vector + | Type_Array + | Type_Unbounded_Array => + return 1; + when Type_Record + | Type_Unbounded_Record => + return Port_Nbr (Typ.Rec.Len); + when Type_Slice + | Type_Access + | Type_File + | Type_Protected => + raise Internal_Error; + end case; + end Count_Nbr_Ports; + + procedure Build_Ports_Desc (Descs : in out Port_Desc_Array; + Idx : in out Port_Nbr; + Pkind : Port_Kind; + Encoding : Name_Encoding; + Typ : Type_Acc; + Inter : Node) + is + Port_Sname : Sname; + begin + Port_Sname := Create_Inter_Name (Inter, Encoding); + + case Typ.Kind is + when Type_Bit + | Type_Logic + | Type_Discrete + | Type_Float + | Type_Vector + | Type_Unbounded_Vector + | Type_Array + | Type_Unbounded_Array => + Idx := Idx + 1; + Descs (Idx) := (Name => Port_Sname, + Is_Inout => Pkind = Port_Inout, + W => Get_Type_Width (Typ)); + when Type_Record + | Type_Unbounded_Record => + declare + Els : constant Node_Flist := Get_Elements_Declaration_List + (Get_Type (Inter)); + El : Node; + begin + for I in Typ.Rec.E'Range loop + El := Get_Nth_Element (Els, Natural (I - 1)); + Idx := Idx + 1; + Descs (Idx) := + (Name => New_Sname_User + (Get_Encoded_Name_Id (El, Encoding), Port_Sname), + Is_Inout => Pkind = Port_Inout, + W => Get_Type_Width (Typ.Rec.E (I).Typ)); + end loop; + end; + when Type_Slice + | Type_Access + | Type_File + | Type_Protected => + raise Internal_Error; + end case; + end Build_Ports_Desc; + + function Build (Params : Inst_Params) return Inst_Object + is + Decl : constant Node := Params.Decl; + Arch : constant Node := Params.Arch; + Imp : Node; + Syn_Inst : Synth_Instance_Acc; + Inter : Node; + Inter_Typ : Type_Acc; + Nbr_Inputs : Port_Nbr; + Nbr_Outputs : Port_Nbr; + Nbr_Params : Param_Nbr; + Cur_Module : Module; + Val : Valtyp; + Id : Module_Id; + begin + if Get_Kind (Params.Decl) = Iir_Kind_Component_Declaration then + pragma Assert (Params.Arch = Null_Node); + pragma Assert (Params.Config = Null_Node); + Imp := Params.Decl; + else + pragma Assert + (Get_Kind (Params.Config) = Iir_Kind_Block_Configuration); + Imp := Params.Arch; + end if; + + -- Create the instance. + Syn_Inst := Make_Instance (Root_Instance, Imp, No_Sname); + + -- Copy values for generics. + Inter := Get_Generic_Chain (Decl); + Nbr_Params := 0; + while Inter /= Null_Node loop + -- Bounds or range of the type. + Build_Object_Subtype (Syn_Inst, Inter, Params.Syn_Inst); + + -- Object. + Create_Object (Syn_Inst, Inter, Get_Value (Params.Syn_Inst, Inter)); + Nbr_Params := Nbr_Params + 1; + Inter := Get_Chain (Inter); + end loop; + + -- Allocate values and count inputs and outputs + Inter := Get_Port_Chain (Decl); + Nbr_Inputs := 0; + Nbr_Outputs := 0; + while Is_Valid (Inter) loop + -- Copy the type from PARAMS if needed. The subtype indication of + -- the port may reference objects that aren't anymore reachable + -- (particularly if it is a port of a component). So the subtype + -- cannot be regularly elaborated. + -- Also, for unconstrained subtypes, we need the constraint. + Build_Object_Subtype (Syn_Inst, Inter, Params.Syn_Inst); + Inter_Typ := Get_Value (Params.Syn_Inst, Inter).Typ; + + case Mode_To_Port_Kind (Get_Mode (Inter)) is + when Port_In => + Val := Create_Value_Net (No_Net, Inter_Typ); + Nbr_Inputs := Nbr_Inputs + Count_Nbr_Ports (Inter_Typ); + when Port_Out + | Port_Inout => + Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); + Nbr_Outputs := Nbr_Outputs + Count_Nbr_Ports (Inter_Typ); + end case; + Create_Object (Syn_Inst, Inter, Val); + Inter := Get_Chain (Inter); + end loop; + + -- Declare module. + -- Build it now because it may be referenced for instantiations before + -- being synthetized. + if Params.Encoding = Name_Parameters + and then Nbr_Params > 0 + then + Id := Id_User_Parameters; + else + Id := Id_User_None; + Nbr_Params := 0; + end if; + Cur_Module := New_User_Module (Get_Top_Module (Root_Instance), + Create_Module_Name (Params), Id, + Nbr_Inputs, Nbr_Outputs, Nbr_Params); + + if Id = Id_User_Parameters then + declare + Descs : Param_Desc_Array (1 .. Nbr_Params); + Ptype : Param_Type; + begin + Inter := Get_Generic_Chain (Decl); + Nbr_Params := 0; + while Inter /= Null_Node loop + -- Bounds or range of the type. + Ptype := Type_To_Param_Type (Get_Type (Inter)); + Nbr_Params := Nbr_Params + 1; + Descs (Nbr_Params) := + (Name => Create_Inter_Name (Inter, Params.Encoding), + Typ => Ptype); + Inter := Get_Chain (Inter); + end loop; + Set_Params_Desc (Cur_Module, Descs); + end; + end if; + + -- Add ports to module. + declare + Inports : Port_Desc_Array (1 .. Nbr_Inputs); + Outports : Port_Desc_Array (1 .. Nbr_Outputs); + Pkind : Port_Kind; + Vt : Valtyp; + begin + Inter := Get_Port_Chain (Decl); + Nbr_Inputs := 0; + Nbr_Outputs := 0; + while Is_Valid (Inter) loop + Pkind := Mode_To_Port_Kind (Get_Mode (Inter)); + Vt := Get_Value (Syn_Inst, Inter); + + case Pkind is + when Port_In => + Build_Ports_Desc (Inports, Nbr_Inputs, + Pkind, Params.Encoding, + Vt.Typ, Inter); + when Port_Out + | Port_Inout => + Build_Ports_Desc (Outports, Nbr_Outputs, + Pkind, Params.Encoding, + Vt.Typ, Inter); + end case; + Inter := Get_Chain (Inter); + end loop; + pragma Assert (Nbr_Inputs = Inports'Last); + pragma Assert (Nbr_Outputs = Outports'Last); + Set_Ports_Desc (Cur_Module, Inports, Outports); + end; + + return Inst_Object'(Decl => Decl, + Arch => Arch, + Config => Params.Config, + Syn_Inst => Syn_Inst, + M => Cur_Module, + Encoding => Params.Encoding); + end Build; + + package Insts_Interning is new Interning + (Params_Type => Inst_Params, + Object_Type => Inst_Object, + Hash => Hash, + Build => Build, + Equal => Equal); + + procedure Synth_Individual_Prefix (Syn_Inst : Synth_Instance_Acc; + Inter_Inst : Synth_Instance_Acc; + Formal : Node; + Off : out Uns32; + Typ : out Type_Acc) is + begin + case Get_Kind (Formal) is + when Iir_Kind_Interface_Signal_Declaration => + Off := 0; + Typ := Get_Subtype_Object (Inter_Inst, Get_Type (Formal)); + when Iir_Kind_Simple_Name => + Synth_Individual_Prefix + (Syn_Inst, Inter_Inst, Get_Named_Entity (Formal), Off, Typ); + when Iir_Kind_Selected_Element => + declare + Idx : constant Iir_Index32 := + Get_Element_Position (Get_Named_Entity (Formal)); + begin + Synth_Individual_Prefix + (Syn_Inst, Inter_Inst, Get_Prefix (Formal), Off, Typ); + Off := Off + Typ.Rec.E (Idx + 1).Boff; + Typ := Typ.Rec.E (Idx + 1).Typ; + end; + when Iir_Kind_Indexed_Name => + declare + Voff : Net; + Arr_Off : Value_Offsets; + begin + Synth_Individual_Prefix + (Syn_Inst, Inter_Inst, Get_Prefix (Formal), Off, Typ); + Synth_Indexed_Name (Syn_Inst, Formal, Typ, Voff, Arr_Off); + if Voff /= No_Net then + raise Internal_Error; + end if; + Off := Off + Arr_Off.Net_Off; + Typ := Get_Array_Element (Typ); + end; + when Iir_Kind_Slice_Name => + declare + Pfx_Bnd : Bound_Type; + El_Typ : Type_Acc; + Res_Bnd : Bound_Type; + Sl_Voff : Net; + Sl_Off : Value_Offsets; + begin + Synth_Individual_Prefix + (Syn_Inst, Inter_Inst, Get_Prefix (Formal), Off, Typ); + + Get_Onedimensional_Array_Bounds (Typ, Pfx_Bnd, El_Typ); + Synth_Slice_Suffix (Syn_Inst, Formal, Pfx_Bnd, El_Typ, + Res_Bnd, Sl_Voff, Sl_Off); + if Sl_Voff /= No_Net then + raise Internal_Error; + end if; + Off := Off + Sl_Off.Net_Off; + Typ := Create_Onedimensional_Array_Subtype (Typ, Res_Bnd); + end; + when others => + Vhdl.Errors.Error_Kind ("synth_individual_prefix", Formal); + end case; + end Synth_Individual_Prefix; + + type Value_Offset_Record is record + Off : Uns32; + Val : Valtyp; + end record; + + package Value_Offset_Tables is new Dyn_Tables + (Table_Component_Type => Value_Offset_Record, + Table_Index_Type => Natural, + Table_Low_Bound => 1); + + procedure Sort_Value_Offset (Els : Value_Offset_Tables.Instance) + is + function Lt (Op1, Op2 : Natural) return Boolean is + begin + return Els.Table (Op1).Off < Els.Table (Op2).Off; + end Lt; + + procedure Swap (From : Natural; To : Natural) + is + T : constant Value_Offset_Record := Els.Table (From); + begin + Els.Table (From) := Els.Table (To); + Els.Table (To) := T; + end Swap; + + procedure Heap_Sort is new Grt.Algos.Heap_Sort (Lt => Lt, Swap => Swap); + begin + Heap_Sort (Value_Offset_Tables.Last (Els)); + end Sort_Value_Offset; + + function Synth_Individual_Input_Assoc (Syn_Inst : Synth_Instance_Acc; + Assoc : Node; + Inter_Inst : Synth_Instance_Acc) + return Net + is + use Netlists.Concats; + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Iassoc : Node; + V : Valtyp; + Off : Uns32; + Typ : Type_Acc; + Els : Value_Offset_Tables.Instance; + Concat : Concat_Type; + N_Off : Uns32; + N : Net; + begin + Value_Offset_Tables.Init (Els, 16); + + Iassoc := Get_Chain (Assoc); + while Iassoc /= Null_Node + and then not Get_Whole_Association_Flag (Iassoc) + loop + -- For each individual assoc: + -- 1. compute type and offset + Synth_Individual_Prefix + (Syn_Inst, Inter_Inst, Get_Formal (Iassoc), Off, Typ); + + -- 2. synth expression + V := Synth_Expression_With_Type (Syn_Inst, Get_Actual (Iassoc), Typ); + + -- 3. save in a table + Value_Offset_Tables.Append (Els, (Off, V)); + + Iassoc := Get_Chain (Iassoc); + end loop; + + -- Then: + -- 1. sort table by offset + Sort_Value_Offset (Els); + + -- 2. concat + N_Off := 0; + for I in Value_Offset_Tables.First .. Value_Offset_Tables.Last (Els) + loop + pragma Assert (N_Off = Els.Table (I).Off); + V := Els.Table (I).Val; + N_Off := N_Off + V.Typ.W; + Append (Concat, Get_Net (Ctxt, V)); + end loop; + Value_Offset_Tables.Free (Els); + + -- 3. connect + Build (Ctxt, Concat, N); + return N; + end Synth_Individual_Input_Assoc; + + function Synth_Input_Assoc (Syn_Inst : Synth_Instance_Acc; + Assoc : Node; + Inter_Inst : Synth_Instance_Acc; + Inter : Node; + Inter_Typ : Type_Acc) + return Net + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Actual : Node; + Act_Inst : Synth_Instance_Acc; + Act : Valtyp; + begin + case Iir_Kinds_Association_Element_Parameters (Get_Kind (Assoc)) is + when Iir_Kind_Association_Element_Open => + Actual := Get_Default_Value (Inter); + Act_Inst := Inter_Inst; + when Iir_Kind_Association_Element_By_Expression => + Actual := Get_Actual (Assoc); + if Get_Kind (Actual) = Iir_Kind_Reference_Name then + -- Skip inserted anonymous signal declaration. + -- FIXME: simply do not insert it ? + Actual := Get_Named_Entity (Actual); + pragma Assert + (Get_Kind (Actual) = Iir_Kind_Anonymous_Signal_Declaration); + Actual := Get_Expression (Actual); + end if; + Act_Inst := Syn_Inst; + when Iir_Kind_Association_Element_By_Individual => + return Synth_Individual_Input_Assoc (Syn_Inst, Assoc, Inter_Inst); + end case; + + Act := Synth_Expression_With_Type (Act_Inst, Actual, Inter_Typ); + Act := Synth_Subtype_Conversion (Ctxt, Act, Inter_Typ, False, Assoc); + if Act = No_Valtyp then + return No_Net; + end if; + return Get_Net (Ctxt, Act); + end Synth_Input_Assoc; + + procedure Synth_Individual_Output_Assoc (Outp : Net; + Syn_Inst : Synth_Instance_Acc; + Assoc : Node; + Inter_Inst : Synth_Instance_Acc) + is + Iassoc : Node; + V : Valtyp; + Off : Uns32; + Typ : Type_Acc; + O : Net; + Port : Net; + begin + Port := Builders.Build_Port (Get_Build (Syn_Inst), Outp); + Set_Location (Port, Assoc); + + Iassoc := Get_Chain (Assoc); + while Iassoc /= Null_Node + and then not Get_Whole_Association_Flag (Iassoc) + loop + -- For each individual assoc: + -- 1. compute type and offset + Synth_Individual_Prefix + (Syn_Inst, Inter_Inst, Get_Formal (Iassoc), Off, Typ); + + -- 2. Extract the value. + O := Build_Extract (Get_Build (Syn_Inst), Port, Off, Typ.W); + V := Create_Value_Net (O, Typ); + + -- 3. Assign. + Synth_Assignment (Syn_Inst, Get_Actual (Iassoc), V, Iassoc); + + Iassoc := Get_Chain (Iassoc); + end loop; + end Synth_Individual_Output_Assoc; + + procedure Synth_Output_Assoc (Outp : Net; + Syn_Inst : Synth_Instance_Acc; + Assoc : Node; + Inter_Inst : Synth_Instance_Acc; + Inter : Node) + is + Actual : Node; + Formal_Typ : Type_Acc; + Port : Net; + O : Valtyp; + begin + case Get_Kind (Assoc) is + when Iir_Kind_Association_Element_Open => + -- Not connected. + return; + when Iir_Kind_Association_Element_By_Expression => + Actual := Get_Actual (Assoc); + when others => + Synth_Individual_Output_Assoc + (Outp, Syn_Inst, Assoc, Inter_Inst); + return; + end case; + + Formal_Typ := Get_Value (Inter_Inst, Inter).Typ; + + -- Create a port gate (so that is has a name). + Port := Builders.Build_Port (Get_Build (Syn_Inst), Outp); + Set_Location (Port, Assoc); + O := Create_Value_Net (Port, Formal_Typ); + -- Assign the port output to the actual (a net). + Synth_Assignment (Syn_Inst, Actual, O, Assoc); + end Synth_Output_Assoc; + + procedure Inst_Input_Connect (Syn_Inst : Synth_Instance_Acc; + Inst : Instance; + Port : in out Port_Idx; + Inter_Typ : Type_Acc; + N : Net) is + begin + case Inter_Typ.Kind is + when Type_Bit + | Type_Logic + | Type_Discrete + | Type_Float + | Type_Vector + | Type_Unbounded_Vector + | Type_Array + | Type_Unbounded_Array => + if N /= No_Net then + Connect (Get_Input (Inst, Port), N); + end if; + Port := Port + 1; + when Type_Record + | Type_Unbounded_Record => + for I in Inter_Typ.Rec.E'Range loop + if N /= No_Net then + Connect (Get_Input (Inst, Port), + Build_Extract (Get_Build (Syn_Inst), N, + Inter_Typ.Rec.E (I).Boff, + Inter_Typ.Rec.E (I).Typ.W)); + end if; + Port := Port + 1; + end loop; + when Type_Slice + | Type_Access + | Type_File + | Type_Protected => + raise Internal_Error; + end case; + end Inst_Input_Connect; + + procedure Inst_Output_Connect (Syn_Inst : Synth_Instance_Acc; + Inst : Instance; + Idx : in out Port_Idx; + Inter_Typ : Type_Acc; + N : out Net) is + begin + case Inter_Typ.Kind is + when Type_Bit + | Type_Logic + | Type_Discrete + | Type_Float + | Type_Vector + | Type_Unbounded_Vector + | Type_Array + | Type_Unbounded_Array => + N := Get_Output (Inst, Idx); + Idx := Idx + 1; + when Type_Record + | Type_Unbounded_Record => + declare + Nets : Net_Array (1 .. Nat32 (Inter_Typ.Rec.Len)); + begin + for I in Inter_Typ.Rec.E'Range loop + Nets (Nat32 (I)) := Get_Output (Inst, Idx); + Idx := Idx + 1; + end loop; + N := Folds.Build2_Concat (Get_Build (Syn_Inst), Nets); + end; + when Type_Slice + | Type_Access + | Type_File + | Type_Protected => + raise Internal_Error; + end case; + end Inst_Output_Connect; + + -- Subprogram used for instantiation (direct or by component). + -- PORTS_ASSOC belong to SYN_INST. + procedure Synth_Instantiate_Module (Syn_Inst : Synth_Instance_Acc; + Inst : Instance; + Inst_Obj : Inst_Object; + Ports_Assoc : Node) + is + -- Instantiate the module + -- Elaborate ports + map aspect for the inputs (component then entity) + -- Elaborate ports + map aspect for the outputs (entity then component) + + Assoc : Node; + Assoc_Inter : Node; + Inter : Node; + Inter_Typ : Type_Acc; + Nbr_Inputs : Port_Nbr; + Nbr_Outputs : Port_Nbr; + N : Net; + begin + Assoc := Ports_Assoc; + Assoc_Inter := Get_Port_Chain (Inst_Obj.Decl); + Nbr_Inputs := 0; + Nbr_Outputs := 0; + while Is_Valid (Assoc) loop + if Get_Whole_Association_Flag (Assoc) then + Inter := Get_Association_Interface (Assoc, Assoc_Inter); + Inter_Typ := Get_Subtype_Object + (Inst_Obj.Syn_Inst, Get_Type (Inter)); + + case Mode_To_Port_Kind (Get_Mode (Inter)) is + when Port_In => + -- Connect the net to the input. + N := Synth_Input_Assoc + (Syn_Inst, Assoc, Inst_Obj.Syn_Inst, Inter, Inter_Typ); + Inst_Input_Connect + (Syn_Inst, Inst, Nbr_Inputs, Inter_Typ, N); + when Port_Out + | Port_Inout => + Inst_Output_Connect + (Syn_Inst, Inst, Nbr_Outputs, Inter_Typ, N); + Synth_Output_Assoc + (N, Syn_Inst, Assoc, Inst_Obj.Syn_Inst, Inter); + end case; + end if; + Next_Association_Interface (Assoc, Assoc_Inter); + end loop; + + if Inst_Obj.Encoding = Name_Parameters then + -- Copy values of the generics to module parameters. + declare + Inter : Node; + Vt : Valtyp; + Pv : Pval; + Idx : Param_Idx; + begin + Idx := 0; + Inter := Get_Generic_Chain (Inst_Obj.Decl); + while Inter /= Null_Node loop + Vt := Get_Value (Inst_Obj.Syn_Inst, Inter); + if Vt /= No_Valtyp then + -- Avoid errors + Pv := Memtyp_To_Pval (Get_Memtyp (Vt)); + Set_Param_Pval (Inst, Idx, Pv); + end if; + Inter := Get_Chain (Inter); + Idx := Idx + 1; + end loop; + end; + end if; + end Synth_Instantiate_Module; + + function Synth_Port_Association_Type (Sub_Inst : Synth_Instance_Acc; + Syn_Inst : Synth_Instance_Acc; + Inter : Node; + Assoc : Node) return Type_Acc is + begin + if not Is_Fully_Constrained_Type (Get_Type (Inter)) then + -- TODO + -- Find the association for this interface + -- * if individual assoc: get type + -- * if whole assoc: get type from object. + if Assoc = Null_Node then + raise Internal_Error; + end if; + case Get_Kind (Assoc) is + when Iir_Kind_Association_Element_By_Expression => + return Synth_Type_Of_Object (Syn_Inst, Get_Actual (Assoc)); + when others => + raise Internal_Error; + end case; + else + Synth_Declaration_Type (Sub_Inst, Inter); + return Get_Subtype_Object (Sub_Inst, Get_Type (Inter)); + end if; + end Synth_Port_Association_Type; + + procedure Synth_Ports_Association_Type (Sub_Inst : Synth_Instance_Acc; + Syn_Inst : Synth_Instance_Acc; + Inter_Chain : Node; + Assoc_Chain : Node) + is + Inter : Node; + Assoc : Node; + Assoc_Inter : Node; + Val : Valtyp; + Inter_Typ : Type_Acc; + begin + Assoc := Assoc_Chain; + Assoc_Inter := Inter_Chain; + while Is_Valid (Assoc) loop + Inter := Get_Association_Interface (Assoc, Assoc_Inter); + if Get_Whole_Association_Flag (Assoc) then + Inter_Typ := Synth_Port_Association_Type + (Sub_Inst, Syn_Inst, Inter, Assoc); + case Mode_To_Port_Kind (Get_Mode (Inter)) is + when Port_In => + Val := Create_Value_Net (No_Net, Inter_Typ); + when Port_Out + | Port_Inout => + Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); + end case; + Create_Object (Sub_Inst, Inter, Val); + end if; + Next_Association_Interface (Assoc, Assoc_Inter); + end loop; + end Synth_Ports_Association_Type; + + procedure Synth_Direct_Instantiation_Statement + (Syn_Inst : Synth_Instance_Acc; + Stmt : Node; + Ent : Node; + Arch : Node; + Config : Node) + is + Sub_Inst : Synth_Instance_Acc; + Inst_Obj : Inst_Object; + Inst : Instance; + Enc : Name_Encoding; + begin + -- Elaborate generic + map aspect + Sub_Inst := Make_Instance + (Syn_Inst, Ent, New_Sname_User (Get_Identifier (Ent), No_Sname)); + + Synth_Generics_Association (Sub_Inst, Syn_Inst, + Get_Generic_Chain (Ent), + Get_Generic_Map_Aspect_Chain (Stmt)); + + -- Elaborate port types. + Synth_Ports_Association_Type (Sub_Inst, Syn_Inst, + Get_Port_Chain (Ent), + Get_Port_Map_Aspect_Chain (Stmt)); + + if Is_Error (Sub_Inst) then + -- TODO: Free it? + return; + end if; + + if Arch /= Null_Node then + -- For whiteboxes: append parameters or/and hash. + Enc := Name_Hash; + else + -- For blackboxes: define the parameters. + Enc := Name_Parameters; + end if; + + -- Search if corresponding module has already been used. + -- If not create a new module + -- * create a name from the generics and the library + -- * create inputs/outputs + -- * add it to the list of module to be synthesized. + Inst_Obj := Insts_Interning.Get ((Decl => Ent, + Arch => Arch, + Config => Config, + Syn_Inst => Sub_Inst, + Encoding => Enc)); + + -- TODO: free sub_inst. + + Inst := New_Instance + (Get_Instance_Module (Syn_Inst), + Inst_Obj.M, + New_Sname_User (Get_Identifier (Stmt), Get_Sname (Syn_Inst))); + Set_Location (Inst, Stmt); + + Push_Phi; + + Synth_Instantiate_Module + (Syn_Inst, Inst, Inst_Obj, Get_Port_Map_Aspect_Chain (Stmt)); + + Pop_And_Merge_Phi (Get_Build (Syn_Inst), Get_Location (Stmt)); + end Synth_Direct_Instantiation_Statement; + + procedure Synth_Design_Instantiation_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Aspect : constant Iir := Get_Instantiated_Unit (Stmt); + Arch : Node; + Ent : Node; + Config : Node; + begin + -- Load configured entity + architecture + case Iir_Kinds_Entity_Aspect (Get_Kind (Aspect)) is + when Iir_Kind_Entity_Aspect_Entity => + Arch := Get_Architecture (Aspect); + if Arch = Null_Node then + Arch := Libraries.Get_Latest_Architecture (Get_Entity (Aspect)); + else + Arch := Strip_Denoting_Name (Arch); + end if; + Config := Get_Library_Unit + (Get_Default_Configuration_Declaration (Arch)); + when Iir_Kind_Entity_Aspect_Configuration => + Config := Get_Configuration (Aspect); + Arch := Get_Block_Specification (Get_Block_Configuration (Config)); + when Iir_Kind_Entity_Aspect_Open => + return; + end case; + Config := Get_Block_Configuration (Config); + Ent := Get_Entity (Arch); + + Synth_Direct_Instantiation_Statement + (Syn_Inst, Stmt, Ent, Arch, Config); + end Synth_Design_Instantiation_Statement; + + procedure Synth_Blackbox_Instantiation_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Comp : constant Node := + Get_Named_Entity (Get_Instantiated_Unit (Stmt)); + begin + Synth_Direct_Instantiation_Statement + (Syn_Inst, Stmt, Comp, Null_Node, Null_Node); + end Synth_Blackbox_Instantiation_Statement; + + procedure Create_Component_Wire (Ctxt : Context_Acc; + Inter : Node; + Val : Valtyp; + Pfx_Name : Sname; + Loc : Source.Syn_Src) + is + Value : Net; + W : Width; + begin + case Val.Val.Kind is + when Value_Wire => + -- Create a gate for the output, so that it could be read. + Val.Val.W := Alloc_Wire (Wire_Output, (Inter, Bit_Type)); + W := Get_Type_Width (Val.Typ); + Value := Build_Signal + (Ctxt, New_Internal_Name (Ctxt, Pfx_Name), W); + Set_Location (Value, Loc); + Set_Wire_Gate (Val.Val.W, Value); + when others => + raise Internal_Error; + end case; + end Create_Component_Wire; + + procedure Synth_Component_Instantiation_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Component : constant Node := + Get_Named_Entity (Get_Instantiated_Unit (Stmt)); + Config : constant Node := Get_Component_Configuration (Stmt); + Bind : constant Node := Get_Binding_Indication (Config); + Aspect : constant Node := Get_Entity_Aspect (Bind); + Comp_Inst : Synth_Instance_Acc; + + Ent : Node; + Arch : Node; + Sub_Config : Node; + Sub_Inst : Synth_Instance_Acc; + Inst_Obj : Inst_Object; + Inst : Instance; + Inst_Name : Sname; + begin + pragma Assert (Get_Component_Configuration (Stmt) /= Null_Node); + pragma Assert (Get_Kind (Aspect) = Iir_Kind_Entity_Aspect_Entity); + + Push_Phi; + + Inst_Name := New_Sname_User (Get_Identifier (Stmt), + Get_Sname (Syn_Inst)); + + -- Create the sub-instance for the component + -- Elaborate generic + map aspect + Comp_Inst := Make_Instance + (Syn_Inst, Component, + New_Sname_User (Get_Identifier (Component), No_Sname)); + + Synth_Generics_Association (Comp_Inst, Syn_Inst, + Get_Generic_Chain (Component), + Get_Generic_Map_Aspect_Chain (Stmt)); + + -- Create objects for the inputs and the outputs of the component, + -- assign inputs (that's nets) and create wires for outputs. + declare + Assoc : Node; + Assoc_Inter : Node; + Inter : Node; + Inter_Typ : Type_Acc; + Val : Valtyp; + N : Net; + begin + Assoc := Get_Port_Map_Aspect_Chain (Stmt); + Assoc_Inter := Get_Port_Chain (Component); + while Is_Valid (Assoc) loop + if Get_Whole_Association_Flag (Assoc) then + Inter := Get_Association_Interface (Assoc, Assoc_Inter); + + Inter_Typ := Synth_Port_Association_Type + (Comp_Inst, Syn_Inst, Inter, Assoc); + + case Mode_To_Port_Kind (Get_Mode (Inter)) is + when Port_In => + N := Synth_Input_Assoc + (Syn_Inst, Assoc, Comp_Inst, Inter, Inter_Typ); + Val := Create_Value_Net (N, Inter_Typ); + when Port_Out + | Port_Inout => + Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); + Create_Component_Wire + (Get_Build (Syn_Inst), Assoc_Inter, Val, Inst_Name, + Assoc); + end case; + Create_Object (Comp_Inst, Assoc_Inter, Val); + end if; + Next_Association_Interface (Assoc, Assoc_Inter); + end loop; + end; + + -- Extract entity/architecture instantiated by the component. + case Get_Kind (Aspect) is + when Iir_Kind_Entity_Aspect_Entity => + Ent := Get_Entity (Aspect); + Arch := Get_Architecture (Aspect); + when others => + Vhdl.Errors.Error_Kind + ("Synth_Component_Instantiation_Statement(2)", Aspect); + end case; + + if Get_Kind (Ent) = Iir_Kind_Foreign_Module then + -- TODO. + raise Internal_Error; + end if; + + if Arch = Null_Node then + Arch := Libraries.Get_Latest_Architecture (Ent); + else + Arch := Get_Named_Entity (Arch); + end if; + Sub_Config := Get_Library_Unit + (Get_Default_Configuration_Declaration (Arch)); + Sub_Config := Get_Block_Configuration (Sub_Config); + + -- Elaborate generic + map aspect for the entity instance. + Sub_Inst := Make_Instance + (Comp_Inst, Ent, New_Sname_User (Get_Identifier (Ent), No_Sname)); + Synth_Generics_Association (Sub_Inst, Comp_Inst, + Get_Generic_Chain (Ent), + Get_Generic_Map_Aspect_Chain (Bind)); + + Synth_Ports_Association_Type (Sub_Inst, Comp_Inst, + Get_Port_Chain (Ent), + Get_Port_Map_Aspect_Chain (Bind)); + + -- Search if corresponding module has already been used. + -- If not create a new module + -- * create a name from the generics and the library + -- * create inputs/outputs + -- * add it to the list of module to be synthesized. + Inst_Obj := Insts_Interning.Get ((Decl => Ent, + Arch => Arch, + Config => Sub_Config, + Syn_Inst => Sub_Inst, + Encoding => Name_Hash)); + + -- TODO: free sub_inst. + + Inst := New_Instance (Get_Instance_Module (Syn_Inst), + Inst_Obj.M, Inst_Name); + Set_Location (Inst, Stmt); + + Synth_Instantiate_Module + (Comp_Inst, Inst, Inst_Obj, Get_Port_Map_Aspect_Chain (Bind)); + + -- Connect out from component to instance. + -- Instantiate the module + -- Elaborate ports + map aspect for the inputs (component then entity) + -- Elaborate ports + map aspect for the outputs (entity then component) + declare + Assoc : Node; + Assoc_Inter : Node; + Inter : Node; + Port : Net; + O : Valtyp; + Nbr_Outputs : Port_Nbr; + begin + Assoc := Get_Port_Map_Aspect_Chain (Stmt); + Assoc_Inter := Get_Port_Chain (Component); + Nbr_Outputs := 0; + while Is_Valid (Assoc) loop + if Get_Whole_Association_Flag (Assoc) then + Inter := Get_Association_Interface (Assoc, Assoc_Inter); + + if Mode_To_Port_Kind (Get_Mode (Inter)) = Port_Out then + O := Get_Value (Comp_Inst, Inter); + Port := Get_Net (Ctxt, O); + Synth_Output_Assoc (Port, Syn_Inst, Assoc, Comp_Inst, Inter); + Nbr_Outputs := Nbr_Outputs + 1; + end if; + end if; + Next_Association_Interface (Assoc, Assoc_Inter); + end loop; + end; + + Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); + + Finalize_Declarations (Comp_Inst, Get_Port_Chain (Component)); + end Synth_Component_Instantiation_Statement; + + procedure Synth_Dependencies (Parent_Inst : Synth_Instance_Acc; Unit : Node) + is + Dep_List : constant Node_List := Get_Dependence_List (Unit); + Dep_It : List_Iterator; + Dep : Node; + Dep_Unit : Node; + begin + Dep_It := List_Iterate (Dep_List); + while Is_Valid (Dep_It) loop + Dep := Get_Element (Dep_It); + if Get_Kind (Dep) = Iir_Kind_Design_Unit + and then not Get_Elab_Flag (Dep) + then + Set_Elab_Flag (Dep, True); + Synth_Dependencies (Parent_Inst, Dep); + Dep_Unit := Get_Library_Unit (Dep); + case Iir_Kinds_Library_Unit (Get_Kind (Dep_Unit)) is + when Iir_Kind_Entity_Declaration => + null; + when Iir_Kind_Configuration_Declaration => + null; + when Iir_Kind_Context_Declaration => + null; + when Iir_Kind_Package_Declaration => + declare + Bod : constant Node := Get_Package_Body (Dep_Unit); + Bod_Unit : Node; + begin + Synth_Package_Declaration (Parent_Inst, Dep_Unit); + -- Do not try to elaborate math_real body: there are + -- functions with loop. Currently, try create signals, + -- which is not possible during package elaboration. + if Bod /= Null_Node + and then Dep_Unit /= Vhdl.Ieee.Math_Real.Math_Real_Pkg + then + Bod_Unit := Get_Design_Unit (Bod); + Synth_Dependencies (Parent_Inst, Bod_Unit); + Synth_Package_Body (Parent_Inst, Dep_Unit, Bod); + end if; + end; + when Iir_Kind_Package_Instantiation_Declaration => + Synth_Package_Instantiation (Parent_Inst, Dep_Unit); + when Iir_Kind_Package_Body => + null; + when Iir_Kind_Architecture_Body => + null; + when Iir_Kinds_Verification_Unit => + null; + end case; + end if; + Next (Dep_It); + end loop; + end Synth_Dependencies; + + procedure Synth_Top_Entity (Global_Instance : Synth_Instance_Acc; + Arch : Node; + Config : Node; + Encoding : Name_Encoding; + Inst : out Synth_Instance_Acc) + is + Entity : constant Node := Get_Entity (Arch); + Syn_Inst : Synth_Instance_Acc; + Inter : Node; + Inter_Typ : Type_Acc; + Inst_Obj : Inst_Object; + Val : Valtyp; + begin + Root_Instance := Global_Instance; + + Insts_Interning.Init; + + if Flags.Flag_Debug_Init then + Synth.Debugger.Debug_Init (Arch); + end if; + + -- Dependencies first. + Synth_Dependencies (Global_Instance, Get_Design_Unit (Entity)); + Synth_Dependencies (Global_Instance, Get_Design_Unit (Arch)); + + Syn_Inst := Make_Instance + (Global_Instance, Arch, + New_Sname_User (Get_Identifier (Entity), No_Sname)); + + -- Compute generics. + Inter := Get_Generic_Chain (Entity); + while Is_Valid (Inter) loop + Synth_Declaration_Type (Syn_Inst, Inter); + declare + Val : Valtyp; + Inter_Typ : Type_Acc; + begin + Inter_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Inter)); + Val := Synth_Expression_With_Type + (Syn_Inst, Get_Default_Value (Inter), Inter_Typ); + pragma Assert (Is_Static (Val.Val)); + Create_Object (Syn_Inst, Inter, Val); + end; + Inter := Get_Chain (Inter); + end loop; + + -- Elaborate port types. + -- FIXME: what about unconstrained ports ? Get the type from the + -- association. + Inter := Get_Port_Chain (Entity); + while Is_Valid (Inter) loop + if not Is_Fully_Constrained_Type (Get_Type (Inter)) then + -- TODO + raise Internal_Error; + end if; + Synth_Declaration_Type (Syn_Inst, Inter); + Inter_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Inter)); + case Mode_To_Port_Kind (Get_Mode (Inter)) is + when Port_In => + Val := Create_Value_Net (No_Net, Inter_Typ); + when Port_Out + | Port_Inout => + Val := Create_Value_Wire (No_Wire_Id, Inter_Typ); + end case; + Create_Object (Syn_Inst, Inter, Val); + Inter := Get_Chain (Inter); + end loop; + + -- Search if corresponding module has already been used. + -- If not create a new module + -- * create a name from the generics and the library + -- * create inputs/outputs + -- * add it to the list of module to be synthesized. + Inst_Obj := Insts_Interning.Get + ((Decl => Entity, + Arch => Arch, + Config => Get_Block_Configuration (Config), + Syn_Inst => Syn_Inst, + Encoding => Encoding)); + Inst := Inst_Obj.Syn_Inst; + end Synth_Top_Entity; + + procedure Create_Input_Wire (Syn_Inst : Synth_Instance_Acc; + Self_Inst : Instance; + Idx : in out Port_Idx; + Val : Valtyp) is + begin + pragma Assert (Val.Val.Kind = Value_Net); + Inst_Output_Connect (Syn_Inst, Self_Inst, Idx, Val.Typ, Val.Val.N); + end Create_Input_Wire; + + procedure Create_Output_Wire (Syn_Inst : Synth_Instance_Acc; + Self_Inst : Instance; + Inter : Node; + Idx : in out Port_Idx; + Val : Valtyp) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Default : constant Node := Get_Default_Value (Inter); + Desc : constant Port_Desc := + Get_Output_Desc (Get_Module (Self_Inst), Idx); + Inter_Typ : Type_Acc; + Value : Net; + Vout : Net; + Init : Valtyp; + Init_Net : Net; + begin + pragma Assert (Val.Val.Kind = Value_Wire); + + -- Create a gate for the output, so that it could be read. + Val.Val.W := Alloc_Wire (Wire_Output, (Inter, Val.Typ)); + -- pragma Assert (Desc.W = Get_Type_Width (Val.Typ)); + + if Default /= Null_Node then + Inter_Typ := Get_Subtype_Object (Syn_Inst, Get_Type (Inter)); + Init := Synth_Expression_With_Type (Syn_Inst, Default, Inter_Typ); + Init := Synth_Subtype_Conversion + (Ctxt, Init, Inter_Typ, False, Inter); + Init_Net := Get_Net (Ctxt, Init); + else + Init_Net := No_Net; + end if; + + if Desc.Is_Inout then + declare + Io_Inst : Instance; + begin + if Init_Net /= No_Net then + Io_Inst := Builders.Build_Iinout (Ctxt, Val.Typ.W); + Connect (Get_Input (Io_Inst, 1), Init_Net); + else + Io_Inst := Builders.Build_Inout (Ctxt, Val.Typ.W); + end if; + -- Connect port1 of gate inout to the pin. + Vout := Get_Output (Io_Inst, 1); + -- And port0 of the gate will be use to read from the pin. + Value := Get_Output (Io_Inst, 0); + end; + else + if Init_Net /= No_Net then + Value := Builders.Build_Ioutput (Ctxt, Init_Net); + else + Value := Builders.Build_Output (Ctxt, Val.Typ.W); + end if; + Vout := Value; + end if; + Set_Location (Value, Inter); + Set_Wire_Gate (Val.Val.W, Value); + + Inst_Input_Connect (Syn_Inst, Self_Inst, Idx, Val.Typ, Vout); + end Create_Output_Wire; + + procedure Apply_Block_Configuration (Cfg : Node; Blk : Node) + is + Item : Node; + begin + -- Be sure CFG applies to BLK. + pragma Assert (Get_Block_From_Block_Specification + (Get_Block_Specification (Cfg)) = Blk); + + -- Clear_Instantiation_Configuration (Blk); + + Item := Get_Configuration_Item_Chain (Cfg); + while Item /= Null_Node loop + case Get_Kind (Item) is + when Iir_Kind_Component_Configuration => + declare + List : constant Iir_Flist := + Get_Instantiation_List (Item); + El : Node; + Inst : Node; + begin + for I in Flist_First .. Flist_Last (List) loop + El := Get_Nth_Element (List, I); + Inst := Get_Named_Entity (El); + pragma Assert + (Get_Kind (Inst) + = Iir_Kind_Component_Instantiation_Statement); + pragma Assert + (Get_Component_Configuration (Inst) = Null_Node); + Set_Component_Configuration (Inst, Item); + end loop; + end; + when Iir_Kind_Block_Configuration => + declare + Sub_Blk : constant Node := Get_Block_From_Block_Specification + (Get_Block_Specification (Item)); + begin + case Get_Kind (Sub_Blk) is + when Iir_Kind_Generate_Statement_Body => + -- Linked chain. + Set_Prev_Block_Configuration + (Item, Get_Generate_Block_Configuration (Sub_Blk)); + Set_Generate_Block_Configuration (Sub_Blk, Item); + when Iir_Kind_Block_Statement => + Set_Block_Block_Configuration (Sub_Blk, Item); + when others => + Vhdl.Errors.Error_Kind + ("apply_block_configuration(blk)", Sub_Blk); + end case; + end; + when others => + Vhdl.Errors.Error_Kind ("apply_block_configuration", Item); + end case; + Item := Get_Chain (Item); + end loop; + end Apply_Block_Configuration; + + procedure Synth_Verification_Units + (Syn_Inst : Synth_Instance_Acc; Parent : Node) + is + Unit : Node; + begin + Unit := Get_Bound_Vunit_Chain (Parent); + while Unit /= Null_Node loop + Synth_Verification_Unit (Syn_Inst, Unit); + Unit := Get_Bound_Vunit_Chain (Unit); + end loop; + end Synth_Verification_Units; + + procedure Synth_Instance (Inst : Inst_Object) + is + Entity : constant Node := Inst.Decl; + Arch : constant Node := Inst.Arch; + Syn_Inst : constant Synth_Instance_Acc := Inst.Syn_Inst; + Self_Inst : Instance; + Inter : Node; + Vt : Valtyp; + Nbr_Inputs : Port_Nbr; + Nbr_Outputs : Port_Nbr; + begin + if Arch = Null_Node then + -- Black box. + return; + end if; + + if Flag_Verbose then + Errors.Info_Msg_Synth (+Entity, "synthesizing %n", (1 => +Entity)); + end if; + + -- Save the current architecture, so that files can be open using a + -- path relative to the architecture filename. + Synth.Vhdl_Files.Set_Design_Unit (Arch); + + Synth_Dependencies (Root_Instance, Get_Design_Unit (Arch)); + + Set_Instance_Module (Syn_Inst, Inst.M); + Self_Inst := Get_Self_Instance (Inst.M); + Set_Location (Self_Inst, Entity); + + -- Create wires for inputs and outputs. + Inter := Get_Port_Chain (Entity); + Nbr_Inputs := 0; + Nbr_Outputs := 0; + while Is_Valid (Inter) loop + Vt := Get_Value (Syn_Inst, Inter); + case Mode_To_Port_Kind (Get_Mode (Inter)) is + when Port_In => + Create_Input_Wire (Syn_Inst, Self_Inst, Nbr_Inputs, Vt); + when Port_Out + | Port_Inout => + Create_Output_Wire + (Syn_Inst, Self_Inst, Inter, Nbr_Outputs, Vt); + end case; + Inter := Get_Chain (Inter); + end loop; + + -- Apply configuration. + -- FIXME: what about inner block configuration ? + pragma Assert (Get_Kind (Inst.Config) = Iir_Kind_Block_Configuration); + Apply_Block_Configuration (Inst.Config, Arch); + + Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Entity)); + if not Is_Error (Syn_Inst) then + Synth_Concurrent_Statements + (Syn_Inst, Get_Concurrent_Statement_Chain (Entity)); + end if; + + if not Is_Error (Syn_Inst) then + Synth_Attribute_Values (Syn_Inst, Entity); + end if; + + if not Is_Error (Syn_Inst) then + Synth_Declarations (Syn_Inst, Get_Declaration_Chain (Arch)); + end if; + if not Is_Error (Syn_Inst) then + Synth_Concurrent_Statements + (Syn_Inst, Get_Concurrent_Statement_Chain (Arch)); + end if; + + if not Is_Error (Syn_Inst) then + Synth_Attribute_Values (Syn_Inst, Arch); + end if; + + if not Is_Error (Syn_Inst) then + Synth_Verification_Units (Syn_Inst, Entity); + end if; + if not Is_Error (Syn_Inst) then + Synth_Verification_Units (Syn_Inst, Arch); + end if; + + Finalize_Declarations (Syn_Inst, Get_Declaration_Chain (Arch)); + Finalize_Declarations (Syn_Inst, Get_Port_Chain (Entity)); + + Finalize_Wires; + + -- Remove unused gates. This is not only an optimization but also + -- a correctness point: there might be some unsynthesizable gates, like + -- the one created for 'rising_egde (clk) and not rst'. + if not Synth.Flags.Flag_Debug_Nocleanup then + -- Netlists.Cleanup.Remove_Unconnected_Instances (Inst.M); + Netlists.Cleanup.Mark_And_Sweep (Inst.M); + Netlists.Cleanup.Remove_Output_Gates (Inst.M); + end if; + + if not Synth.Flags.Flag_Debug_Nomemory2 then + Netlists.Memories.Extract_Memories2 (Get_Build (Syn_Inst), Inst.M); + -- Remove remaining clock edge gates. + Netlists.Cleanup.Mark_And_Sweep (Inst.M); + end if; + + if not Synth.Flags.Flag_Debug_Noexpand then + Netlists.Expands.Expand_Gates (Get_Build (Syn_Inst), Inst.M); + end if; + end Synth_Instance; + + procedure Synth_All_Instances + is + use Insts_Interning; + Idx : Index_Type; + begin + Idx := First_Index; + while Idx <= Last_Index loop + Synth_Instance (Get_By_Index (Idx)); + Idx := Idx + 1; + end loop; + end Synth_All_Instances; +end Synth.Vhdl_Insts; diff --git a/src/synth/synth-vhdl_insts.ads b/src/synth/synth-vhdl_insts.ads new file mode 100644 index 000000000..980b4ca8b --- /dev/null +++ b/src/synth/synth-vhdl_insts.ads @@ -0,0 +1,47 @@ +-- Instantiation synthesis. +-- Copyright (C) 2019 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with Vhdl.Nodes; use Vhdl.Nodes; + +with Synth.Vhdl_Context; use Synth.Vhdl_Context; +with Synth.Flags; use Synth.Flags; + +package Synth.Vhdl_Insts is + -- Create the declaration of the top entity. + procedure Synth_Top_Entity (Global_Instance : Synth_Instance_Acc; + Arch : Node; + Config : Node; + Encoding : Name_Encoding; + Inst : out Synth_Instance_Acc); + + -- Synthesize the top entity and all the sub-modules. + procedure Synth_All_Instances; + + -- Apply block configuration CFG to BLK. + -- Must be done before synthesis of BLK. + -- The synthesis of BLK will clear all configuration of it. + procedure Apply_Block_Configuration (Cfg : Node; Blk : Node); + + procedure Synth_Design_Instantiation_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node); + procedure Synth_Blackbox_Instantiation_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node); + + procedure Synth_Component_Instantiation_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node); +end Synth.Vhdl_Insts; diff --git a/src/synth/synth-vhdl_oper.adb b/src/synth/synth-vhdl_oper.adb index 2c3252a83..fadd500b5 100644 --- a/src/synth/synth-vhdl_oper.adb +++ b/src/synth/synth-vhdl_oper.adb @@ -34,8 +34,8 @@ with Netlists.Utils; with Synth.Memtype; use Synth.Memtype; with Synth.Errors; use Synth.Errors; -with Synth.Stmts; use Synth.Stmts; -with Synth.Expr; use Synth.Expr; +with Synth.Vhdl_Stmts; use Synth.Vhdl_Stmts; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; with Synth.Source; with Synth.Static_Oper; use Synth.Static_Oper; diff --git a/src/synth/synth-vhdl_stmts.adb b/src/synth/synth-vhdl_stmts.adb new file mode 100644 index 000000000..0f2694e06 --- /dev/null +++ b/src/synth/synth-vhdl_stmts.adb @@ -0,0 +1,3856 @@ +-- Statements synthesis. +-- Copyright (C) 2017 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with Ada.Unchecked_Deallocation; + +with Grt.Types; use Grt.Types; +with Grt.Algos; +with Grt.Severity; use Grt.Severity; +with Areapools; +with Name_Table; +with Std_Names; +with Errorout; use Errorout; +with Files_Map; +with Simple_IO; + +with Vhdl.Errors; use Vhdl.Errors; +with Vhdl.Sem_Expr; +with Vhdl.Sem_Inst; +with Vhdl.Utils; use Vhdl.Utils; +with Vhdl.Std_Package; +with Vhdl.Evaluation; +with Vhdl.Ieee.Std_Logic_1164; + +with PSL.Types; +with PSL.NFAs; + +with Synth.Memtype; use Synth.Memtype; +with Synth.Errors; use Synth.Errors; +with Synth.Vhdl_Decls; use Synth.Vhdl_Decls; +with Synth.Vhdl_Expr; use Synth.Vhdl_Expr; +with Synth.Vhdl_Insts; use Synth.Vhdl_Insts; +with Synth.Source; +with Synth.Vhdl_Static_Proc; +with Synth.Vhdl_Heap; +with Synth.Flags; +with Synth.Debugger; + +with Netlists.Builders; use Netlists.Builders; +with Netlists.Folds; use Netlists.Folds; +with Netlists.Gates; use Netlists.Gates; +with Netlists.Utils; use Netlists.Utils; +with Netlists.Locations; use Netlists.Locations; + +package body Synth.Vhdl_Stmts is + procedure Synth_Sequential_Statements + (C : in out Seq_Context; Stmts : Node); + + procedure Set_Location (N : Net; Loc : Node) + renames Synth.Source.Set_Location; + + function Synth_Waveform (Syn_Inst : Synth_Instance_Acc; + Wf : Node; + Targ_Type : Type_Acc) return Valtyp + is + Res : Valtyp; + begin + if Get_Kind (Wf) = Iir_Kind_Unaffected_Waveform then + -- TODO + raise Internal_Error; + end if; + if Get_Chain (Wf) /= Null_Node then + -- Warning. + null; + end if; + if Get_Time (Wf) /= Null_Node then + -- Warning + null; + end if; + if Targ_Type = null then + return Synth_Expression (Syn_Inst, Get_We_Value (Wf)); + else + Res := Synth_Expression_With_Type + (Syn_Inst, Get_We_Value (Wf), Targ_Type); + Res := Synth_Subtype_Conversion + (Get_Build (Syn_Inst), Res, Targ_Type, False, Wf); + return Res; + end if; + end Synth_Waveform; + + procedure Synth_Assignment_Prefix (Syn_Inst : Synth_Instance_Acc; + Pfx : Node; + Dest_Base : out Valtyp; + Dest_Typ : out Type_Acc; + Dest_Off : out Value_Offsets; + Dest_Dyn : out Dyn_Name) is + begin + case Get_Kind (Pfx) is + when Iir_Kind_Simple_Name => + Synth_Assignment_Prefix (Syn_Inst, Get_Named_Entity (Pfx), + Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); + when Iir_Kind_Interface_Signal_Declaration + | Iir_Kind_Variable_Declaration + | Iir_Kind_Interface_Variable_Declaration + | Iir_Kind_Signal_Declaration + | Iir_Kind_Anonymous_Signal_Declaration + | Iir_Kind_Interface_Constant_Declaration + | Iir_Kind_Constant_Declaration + | Iir_Kind_File_Declaration + | Iir_Kind_Interface_File_Declaration + | Iir_Kind_Object_Alias_Declaration => + declare + Targ : constant Valtyp := Get_Value (Syn_Inst, Pfx); + begin + Dest_Dyn := No_Dyn_Name; + Dest_Typ := Targ.Typ; + + if Targ.Val.Kind = Value_Alias then + -- Replace alias by the aliased name. + Dest_Base := (Targ.Val.A_Typ, Targ.Val.A_Obj); + Dest_Off := Targ.Val.A_Off; + else + Dest_Base := Targ; + Dest_Off := (0, 0); + end if; + end; + when Iir_Kind_Function_Call => + Dest_Base := Synth_Expression (Syn_Inst, Pfx); + Dest_Typ := Dest_Base.Typ; + Dest_Off := (0, 0); + Dest_Dyn := No_Dyn_Name; + + when Iir_Kind_Indexed_Name => + declare + Voff : Net; + Off : Value_Offsets; + begin + Synth_Assignment_Prefix + (Syn_Inst, Get_Prefix (Pfx), + Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); + Strip_Const (Dest_Base); + Synth_Indexed_Name (Syn_Inst, Pfx, Dest_Typ, Voff, Off); + + if Voff = No_Net then + -- Static index. + Dest_Off := Dest_Off + Off; + else + -- Dynamic index. + if Dest_Dyn.Voff = No_Net then + -- The first one. + Dest_Dyn := (Pfx_Off => Dest_Off, + Pfx_Typ => Dest_Typ, + Voff => Voff); + Dest_Off := Off; + else + -- Nested one. + -- FIXME + Dest_Off := Dest_Off + Off; + -- if Dest_Off /= (0, 0) then + -- Error_Msg_Synth (+Pfx, "nested memory not supported"); + -- end if; + + Dest_Dyn.Voff := Build_Addidx + (Get_Build (Syn_Inst), Dest_Dyn.Voff, Voff); + end if; + end if; + + Dest_Typ := Get_Array_Element (Dest_Typ); + end; + + when Iir_Kind_Selected_Element => + declare + Idx : constant Iir_Index32 := + Get_Element_Position (Get_Named_Entity (Pfx)); + begin + Synth_Assignment_Prefix + (Syn_Inst, Get_Prefix (Pfx), + Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); + Dest_Off.Net_Off := + Dest_Off.Net_Off + Dest_Typ.Rec.E (Idx + 1).Boff; + Dest_Off.Mem_Off := + Dest_Off.Mem_Off + Dest_Typ.Rec.E (Idx + 1).Moff; + + Dest_Typ := Dest_Typ.Rec.E (Idx + 1).Typ; + end; + + when Iir_Kind_Slice_Name => + declare + Pfx_Bnd : Bound_Type; + El_Typ : Type_Acc; + Res_Bnd : Bound_Type; + Sl_Voff : Net; + Sl_Off : Value_Offsets; + begin + Synth_Assignment_Prefix + (Syn_Inst, Get_Prefix (Pfx), + Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); + Strip_Const (Dest_Base); + + Get_Onedimensional_Array_Bounds (Dest_Typ, Pfx_Bnd, El_Typ); + Synth_Slice_Suffix (Syn_Inst, Pfx, Pfx_Bnd, El_Typ, + Res_Bnd, Sl_Voff, Sl_Off); + + + if Sl_Voff = No_Net then + -- Fixed slice. + Dest_Typ := Create_Onedimensional_Array_Subtype + (Dest_Typ, Res_Bnd); + Dest_Off.Net_Off := Dest_Off.Net_Off + Sl_Off.Net_Off; + Dest_Off.Mem_Off := Dest_Off.Mem_Off + Sl_Off.Mem_Off; + else + -- Variable slice. + if Dest_Dyn.Voff = No_Net then + -- First one. + Dest_Dyn := (Pfx_Off => Dest_Off, + Pfx_Typ => Dest_Typ, + Voff => Sl_Voff); + Dest_Off := Sl_Off; + else + -- Nested. + if Dest_Off /= (0, 0) then + Error_Msg_Synth (+Pfx, "nested memory not supported"); + end if; + + Dest_Dyn.Voff := Build_Addidx + (Get_Build (Syn_Inst), Dest_Dyn.Voff, Sl_Voff); + end if; + Dest_Typ := Create_Slice_Type (Res_Bnd.Len, El_Typ); + end if; + end; + + when Iir_Kind_Implicit_Dereference + | Iir_Kind_Dereference => + Synth_Assignment_Prefix + (Syn_Inst, Get_Prefix (Pfx), + Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); + if Dest_Off /= (0, 0) and then Dest_Dyn.Voff /= No_Net then + raise Internal_Error; + end if; + Dest_Base := Vhdl_Heap.Synth_Dereference (Read_Access (Dest_Base)); + Dest_Typ := Dest_Base.Typ; + + when others => + Error_Kind ("synth_assignment_prefix", Pfx); + end case; + end Synth_Assignment_Prefix; + + type Target_Kind is + ( + -- The target is an object or a static part of it. + Target_Simple, + + -- The target is an aggregate. + Target_Aggregate, + + -- The assignment is dynamically indexed. + Target_Memory + ); + + type Target_Info (Kind : Target_Kind := Target_Simple) is record + -- In all cases, the type of the target is known or computed. + Targ_Type : Type_Acc; + + case Kind is + when Target_Simple => + -- For a simple target, the destination is known. + Obj : Valtyp; + Off : Value_Offsets; + when Target_Aggregate => + -- For an aggregate: the type is computed and the details will + -- be handled at the assignment. + Aggr : Node; + when Target_Memory => + -- For a memory: the destination is known. + Mem_Obj : Valtyp; + -- The dynamic offset. + Mem_Dyn : Dyn_Name; + -- Offset of the data to be accessed from the memory. + Mem_Doff : Uns32; + end case; + end record; + + type Target_Info_Array is array (Natural range <>) of Target_Info; + + function Synth_Aggregate_Target_Type (Syn_Inst : Synth_Instance_Acc; + Target : Node) return Type_Acc + is + Targ_Type : constant Node := Get_Type (Target); + Base_Type : constant Node := Get_Base_Type (Targ_Type); + Base_Typ : Type_Acc; + Bnd : Bound_Type; + Len : Uns32; + Res : Type_Acc; + begin + Base_Typ := Get_Subtype_Object (Syn_Inst, Base_Type); + -- It's a basetype, so not bounded. + pragma Assert (Base_Typ.Kind = Type_Unbounded_Vector); + + if Is_Fully_Constrained_Type (Targ_Type) then + -- If the aggregate subtype is known, just use it. + Bnd := Vhdl_Expr.Synth_Array_Bounds (Syn_Inst, Targ_Type, 1); + else + -- Ok, so the subtype of the aggregate is not known, in general + -- because the length of an element is not known. That's with + -- vhdl-2008. + Len := 0; + declare + Choice : Node; + El : Node; + El_Typ : Type_Acc; + begin + Choice := Get_Association_Choices_Chain (Target); + while Choice /= Null_Node loop + pragma Assert (Get_Kind (Choice) = Iir_Kind_Choice_By_None); + El := Get_Associated_Expr (Choice); + El_Typ := Synth_Type_Of_Object (Syn_Inst, El); + Bnd := Get_Array_Bound (El_Typ, 1); + Len := Len + Bnd.Len; + Choice := Get_Chain (Choice); + end loop; + end; + + -- Compute the range. + declare + Idx_Type : constant Node := Get_Index_Type (Base_Type, 0); + Idx_Typ : Type_Acc; + begin + Idx_Typ := Get_Subtype_Object (Syn_Inst, Idx_Type); + Bnd := (Dir => Idx_Typ.Drange.Dir, + Left => Int32 (Idx_Typ.Drange.Left), + Right => 0, + Len => Len); + case Bnd.Dir is + when Dir_To => + Bnd.Right := Bnd.Left + Int32 (Len); + when Dir_Downto => + Bnd.Right := Bnd.Left - Int32 (Len); + end case; + end; + end if; + + -- Compute the type. + case Base_Typ.Kind is + when Type_Unbounded_Vector => + Res := Create_Vector_Type (Bnd, Base_Typ.Uvec_El); + when others => + raise Internal_Error; + end case; + return Res; + end Synth_Aggregate_Target_Type; + + function Synth_Target (Syn_Inst : Synth_Instance_Acc; + Target : Node) return Target_Info is + begin + case Get_Kind (Target) is + when Iir_Kind_Aggregate => + return Target_Info'(Kind => Target_Aggregate, + Targ_Type => Synth_Aggregate_Target_Type + (Syn_Inst, Target), + Aggr => Target); + when Iir_Kind_Simple_Name + | Iir_Kind_Selected_Element + | Iir_Kind_Interface_Signal_Declaration + | Iir_Kind_Variable_Declaration + | Iir_Kind_Signal_Declaration + | Iir_Kind_Anonymous_Signal_Declaration + | Iir_Kind_Indexed_Name + | Iir_Kind_Slice_Name + | Iir_Kind_Dereference => + declare + Base : Valtyp; + Typ : Type_Acc; + Off : Value_Offsets; + + Dyn : Dyn_Name; + begin + Synth_Assignment_Prefix (Syn_Inst, Target, Base, Typ, Off, Dyn); + if Dyn.Voff = No_Net then + -- FIXME: check index. + return Target_Info'(Kind => Target_Simple, + Targ_Type => Typ, + Obj => Base, + Off => Off); + else + return Target_Info'(Kind => Target_Memory, + Targ_Type => Typ, + Mem_Obj => Base, + Mem_Dyn => Dyn, + Mem_Doff => Off.Net_Off); + end if; + end; + when others => + Error_Kind ("synth_target", Target); + end case; + end Synth_Target; + + procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; + Target : Target_Info; + Val : Valtyp; + Loc : Node); + + -- Extract a part of VAL from a target aggregate at offset OFF (offset + -- in the array). + function Aggregate_Extract (Ctxt : Context_Acc; + Val : Valtyp; + Off : Uns32; + Typ : Type_Acc; + Loc : Node) return Valtyp + is + El_Typ : constant Type_Acc := Get_Array_Element (Val.Typ); + begin + case Val.Val.Kind is + when Value_Net + | Value_Wire => + declare + N : Net; + begin + N := Build2_Extract + (Ctxt, Get_Net (Ctxt, Val), Off * El_Typ.W, Typ.W); + Set_Location (N, Loc); + return Create_Value_Net (N, Typ); + end; + when Value_Memory => + declare + Res : Valtyp; + begin + Res := Create_Value_Memory (Typ); + -- Need to reverse offsets. + Copy_Memory + (Res.Val.Mem, + Val.Val.Mem + (Val.Typ.Sz - Size_Type (Off + 1) * El_Typ.Sz), + Typ.Sz); + return Res; + end; + when others => + raise Internal_Error; + end case; + end Aggregate_Extract; + + procedure Synth_Assignment_Aggregate (Syn_Inst : Synth_Instance_Acc; + Target : Node; + Target_Typ : Type_Acc; + Val : Valtyp; + Loc : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Targ_Bnd : constant Bound_Type := Get_Array_Bound (Target_Typ, 1); + Choice : Node; + Assoc : Node; + Pos : Uns32; + Targ_Info : Target_Info; + begin + Choice := Get_Association_Choices_Chain (Target); + Pos := Targ_Bnd.Len; + while Is_Valid (Choice) loop + Assoc := Get_Associated_Expr (Choice); + case Get_Kind (Choice) is + when Iir_Kind_Choice_By_None => + Targ_Info := Synth_Target (Syn_Inst, Assoc); + if Get_Element_Type_Flag (Choice) then + Pos := Pos - 1; + else + Pos := Pos - Get_Array_Bound (Targ_Info.Targ_Type, 1).Len; + end if; + Synth_Assignment + (Syn_Inst, Targ_Info, + Aggregate_Extract (Ctxt, Val, Pos, + Targ_Info.Targ_Type, Assoc), + Loc); + when others => + Error_Kind ("synth_assignment_aggregate", Choice); + end case; + Choice := Get_Chain (Choice); + end loop; + end Synth_Assignment_Aggregate; + + procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; + Target : Target_Info; + Val : Valtyp; + Loc : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + V : Valtyp; + begin + V := Synth_Subtype_Conversion (Ctxt, Val, Target.Targ_Type, False, Loc); + pragma Unreferenced (Val); + if V = No_Valtyp then + -- In case of error. + return; + end if; + + case Target.Kind is + when Target_Aggregate => + Synth_Assignment_Aggregate + (Syn_Inst, Target.Aggr, Target.Targ_Type, V, Loc); + when Target_Simple => + if V.Typ.Sz = 0 then + -- If there is nothing to assign (like a null slice), + -- return now. + return; + end if; + + if Target.Obj.Val.Kind = Value_Wire then + if Is_Static (V.Val) + and then V.Typ.Sz = Target.Obj.Typ.Sz + then + pragma Assert (Target.Off = (0, 0)); + Phi_Assign_Static + (Target.Obj.Val.W, Unshare (Get_Memtyp (V))); + else + if V.Typ.W = 0 then + -- Forget about null wires. + return; + end if; + Phi_Assign_Net (Ctxt, Target.Obj.Val.W, + Get_Net (Ctxt, V), Target.Off.Net_Off); + end if; + else + if not Is_Static (V.Val) then + -- Maybe the error message is too cryptic ? + Error_Msg_Synth + (+Loc, "cannot assign a net to a static value"); + else + Strip_Const (V); + Copy_Memory (Target.Obj.Val.Mem + Target.Off.Mem_Off, + V.Val.Mem, V.Typ.Sz); + end if; + end if; + when Target_Memory => + declare + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + N : Net; + begin + N := Get_Current_Assign_Value + (Ctxt, Target.Mem_Obj.Val.W, + Target.Mem_Dyn.Pfx_Off.Net_Off, Target.Mem_Dyn.Pfx_Typ.W); + N := Build_Dyn_Insert (Ctxt, N, Get_Net (Ctxt, V), + Target.Mem_Dyn.Voff, Target.Mem_Doff); + Set_Location (N, Loc); + Phi_Assign_Net (Ctxt, Target.Mem_Obj.Val.W, N, + Target.Mem_Dyn.Pfx_Off.Net_Off); + end; + end case; + end Synth_Assignment; + + procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; + Target : Node; + Val : Valtyp; + Loc : Node) + is + Info : Target_Info; + begin + Info := Synth_Target (Syn_Inst, Target); + Synth_Assignment (Syn_Inst, Info, Val, Loc); + end Synth_Assignment; + + function Synth_Read_Memory (Syn_Inst : Synth_Instance_Acc; + Obj : Valtyp; + Res_Typ : Type_Acc; + Off : Uns32; + Dyn : Dyn_Name; + Loc : Node) return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + N : Net; + begin + N := Get_Net (Ctxt, Obj); + if Dyn.Voff /= No_Net then + Synth.Source.Set_Location_Maybe (N, Loc); + if Dyn.Pfx_Off.Net_Off /= 0 then + N := Build2_Extract (Ctxt, N, Dyn.Pfx_Off.Net_Off, Dyn.Pfx_Typ.W); + end if; + if Res_Typ.W /= 0 then + -- Do not try to extract if the net is null. + N := Build_Dyn_Extract (Ctxt, N, Dyn.Voff, Off, Res_Typ.W); + end if; + else + pragma Assert (not Is_Static (Obj.Val)); + N := Build2_Extract (Ctxt, N, Off, Res_Typ.W); + end if; + Set_Location (N, Loc); + return Create_Value_Net (N, Res_Typ); + end Synth_Read_Memory; + + function Synth_Read (Syn_Inst : Synth_Instance_Acc; + Targ : Target_Info; + Loc : Node) return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + N : Net; + begin + case Targ.Kind is + when Target_Simple => + N := Build2_Extract (Ctxt, Get_Net (Ctxt, Targ.Obj), + Targ.Off.Net_Off, Targ.Targ_Type.W); + return Create_Value_Net (N, Targ.Targ_Type); + when Target_Aggregate => + raise Internal_Error; + when Target_Memory => + return Synth_Read_Memory (Syn_Inst, Targ.Mem_Obj, Targ.Targ_Type, + 0, Targ.Mem_Dyn, Loc); + end case; + end Synth_Read; + + -- Concurrent or sequential simple signal assignment + procedure Synth_Simple_Signal_Assignment + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Targ : Target_Info; + Val : Valtyp; + begin + Targ := Synth_Target (Syn_Inst, Get_Target (Stmt)); + Val := Synth_Waveform + (Syn_Inst, Get_Waveform_Chain (Stmt), Targ.Targ_Type); + Synth_Assignment (Syn_Inst, Targ, Val, Stmt); + end Synth_Simple_Signal_Assignment; + + procedure Synth_Conditional_Signal_Assignment + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Targ : Target_Info; + Cond : Node; + Cwf : Node; + Inp : Input; + Val, Cond_Val : Valtyp; + Cond_Net : Net; + First, Last : Net; + V : Net; + begin + Targ := Synth_Target (Syn_Inst, Get_Target (Stmt)); + Last := No_Net; + Cwf := Get_Conditional_Waveform_Chain (Stmt); + Cond := Null_Node; + while Cwf /= Null_Node loop + Val := Synth_Waveform + (Syn_Inst, Get_Waveform_Chain (Cwf), Targ.Targ_Type); + if Val = No_Valtyp then + -- Mark the error, but try to continue. + Set_Error (Syn_Inst); + else + V := Get_Net (Ctxt, Val); + Cond := Get_Condition (Cwf); + if Cond /= Null_Node then + Cond_Val := Synth_Expression (Syn_Inst, Cond); + if Cond_Val = No_Valtyp then + Cond_Net := Build_Const_UB32 (Ctxt, 0, 1); + else + Cond_Net := Get_Net (Ctxt, Cond_Val); + end if; + + V := Build_Mux2 (Ctxt, Cond_Net, No_Net, V); + Set_Location (V, Cwf); + end if; + + if Last /= No_Net then + Inp := Get_Input (Get_Net_Parent (Last), 1); + Connect (Inp, V); + else + First := V; + end if; + Last := V; + end if; + Cwf := Get_Chain (Cwf); + end loop; + if Cond /= Null_Node then + pragma Assert (Last /= No_Net); + Inp := Get_Input (Get_Net_Parent (Last), 1); + if Get_Driver (Inp) = No_Net then + -- No else. + Val := Synth_Read (Syn_Inst, Targ, Stmt); + Connect (Inp, Get_Net (Ctxt, Val)); + end if; + end if; + Val := Create_Value_Net (First, Targ.Targ_Type); + Synth_Assignment (Syn_Inst, Targ, Val, Stmt); + end Synth_Conditional_Signal_Assignment; + + procedure Synth_Variable_Assignment (C : Seq_Context; Stmt : Node) + is + Targ : Target_Info; + Val : Valtyp; + begin + Targ := Synth_Target (C.Inst, Get_Target (Stmt)); + Val := Synth_Expression_With_Type + (C.Inst, Get_Expression (Stmt), Targ.Targ_Type); + if Val = No_Valtyp then + Set_Error (C.Inst); + return; + end if; + Synth_Assignment (C.Inst, Targ, Val, Stmt); + end Synth_Variable_Assignment; + + procedure Synth_Conditional_Variable_Assignment + (C : Seq_Context; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (C.Inst); + Target : constant Node := Get_Target (Stmt); + Targ_Type : Type_Acc; + Cond : Node; + Ce : Node; + Val, Cond_Val : Valtyp; + V : Net; + First, Last : Net; + begin + Targ_Type := Get_Subtype_Object (C.Inst, Get_Type (Target)); + Last := No_Net; + Ce := Get_Conditional_Expression_Chain (Stmt); + while Ce /= Null_Node loop + Val := Synth_Expression_With_Type + (C.Inst, Get_Expression (Ce), Targ_Type); + V := Get_Net (Ctxt, Val); + Cond := Get_Condition (Ce); + if Cond /= Null_Node then + Cond_Val := Synth_Expression (C.Inst, Cond); + V := Build_Mux2 (Ctxt, Get_Net (Ctxt, Cond_Val), No_Net, V); + Set_Location (V, Ce); + end if; + + if Last /= No_Net then + Connect (Get_Input (Get_Net_Parent (Last), 1), V); + else + First := V; + end if; + Last := V; + Ce := Get_Chain (Ce); + end loop; + Val := Create_Value_Net (First, Targ_Type); + Synth_Assignment (C.Inst, Target, Val, Stmt); + end Synth_Conditional_Variable_Assignment; + + procedure Synth_If_Statement (C : in out Seq_Context; Stmt : Node) + is + Cond : constant Node := Get_Condition (Stmt); + Els : constant Node := Get_Else_Clause (Stmt); + Ctxt : constant Context_Acc := Get_Build (C.Inst); + Cond_Val : Valtyp; + Cond_Net : Net; + Phi_True : Phi_Type; + Phi_False : Phi_Type; + begin + Cond_Val := Synth_Expression (C.Inst, Cond); + if Cond_Val = No_Valtyp then + Set_Error (C.Inst); + return; + end if; + if Is_Static_Val (Cond_Val.Val) then + Strip_Const (Cond_Val); + if Read_Discrete (Get_Value_Memtyp (Cond_Val)) = 1 then + -- True. + Synth_Sequential_Statements + (C, Get_Sequential_Statement_Chain (Stmt)); + else + pragma Assert (Read_Discrete (Get_Value_Memtyp (Cond_Val)) = 0); + if Is_Valid (Els) then + -- Else part + if Is_Null (Get_Condition (Els)) then + -- Final else part. + Synth_Sequential_Statements + (C, Get_Sequential_Statement_Chain (Els)); + else + -- Elsif. Handled as a nested if. + Synth_If_Statement (C, Els); + end if; + end if; + end if; + else + -- The statements for the 'then' part. + Push_Phi; + Synth_Sequential_Statements + (C, Get_Sequential_Statement_Chain (Stmt)); + Pop_Phi (Phi_True); + + Push_Phi; + + if Is_Valid (Els) then + if Is_Null (Get_Condition (Els)) then + -- Final else part. + Synth_Sequential_Statements + (C, Get_Sequential_Statement_Chain (Els)); + else + -- Elsif. Handled as a nested if. + Synth_If_Statement (C, Els); + end if; + end if; + + Pop_Phi (Phi_False); + + Cond_Net := Get_Net (Ctxt, Cond_Val); + Merge_Phis (Ctxt, Cond_Net, Phi_True, Phi_False, Get_Location (Stmt)); + end if; + end Synth_If_Statement; + + type Alternative_Index is new Int32; + + -- Only keep '0' and '1' in choices for std_logic. + function Ignore_Choice_Logic (V : Ghdl_U8; Loc : Node) return Boolean is + begin + case V is + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_0_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_1_Pos => + return False; + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_L_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_H_Pos => + Warning_Msg_Synth + (+Loc, "choice with 'L' or 'H' value is ignored"); + return True; + when Vhdl.Ieee.Std_Logic_1164.Std_Logic_U_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_X_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_D_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_Z_Pos + | Vhdl.Ieee.Std_Logic_1164.Std_Logic_W_Pos => + Warning_Msg_Synth (+Loc, "choice with meta-value is ignored"); + return True; + when others => + -- Only 9 values. + raise Internal_Error; + end case; + end Ignore_Choice_Logic; + + function Ignore_Choice_Expression (V : Valtyp; Loc : Node) return Boolean is + begin + case V.Typ.Kind is + when Type_Bit => + return False; + when Type_Logic => + if V.Typ = Logic_Type then + return Ignore_Choice_Logic (Read_U8 (V.Val.Mem), Loc); + else + return False; + end if; + when Type_Discrete => + return False; + when Type_Vector => + if V.Typ.Vec_El = Logic_Type then + for I in 1 .. Size_Type (V.Typ.Vbound.Len) loop + if Ignore_Choice_Logic (Read_U8 (V.Val.Mem + (I - 1)), Loc) + then + return True; + end if; + end loop; + return False; + else + return False; + end if; + when Type_Array => + return False; + when others => + raise Internal_Error; + end case; + end Ignore_Choice_Expression; + + -- Create the condition for choices of CHOICE chain belonging to the same + -- alternative. Update CHOICE to the next alternative. + procedure Synth_Choice (Syn_Inst : Synth_Instance_Acc; + Sel : Net; + Choice_Typ : Type_Acc; + Nets : in out Net_Array; + Other_Choice : in out Nat32; + Choice_Idx : in out Nat32; + Choice : in out Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Cond : Net; + Res : Net; + begin + Res := No_Net; + loop + case Iir_Kinds_Case_Choice (Get_Kind (Choice)) is + when Iir_Kind_Choice_By_Expression => + declare + V : Valtyp; + begin + V := Synth_Expression_With_Basetype + (Syn_Inst, Get_Choice_Expression (Choice)); + V := Synth_Subtype_Conversion + (Ctxt, V, Choice_Typ, False, Choice); + if Ignore_Choice_Expression (V, Choice) then + Cond := No_Net; + else + Cond := Build_Compare + (Ctxt, Id_Eq, Sel, Get_Net (Ctxt, V)); + Set_Location (Cond, Choice); + end if; + end; + + when Iir_Kind_Choice_By_Range => + declare + Rng : Discrete_Range_Type; + Cmp_L, Cmp_R : Module_Id; + L, R : Net; + begin + Synth_Discrete_Range + (Syn_Inst, Get_Choice_Range (Choice), Rng); + + if Rng.Is_Signed then + case Rng.Dir is + when Dir_To => + Cmp_L := Id_Sge; + Cmp_R := Id_Sle; + when Dir_Downto => + Cmp_L := Id_Sle; + Cmp_R := Id_Sge; + end case; + L := Build2_Const_Int (Ctxt, Rng.Left, Choice_Typ.W); + R := Build2_Const_Int (Ctxt, Rng.Right, Choice_Typ.W); + else + case Rng.Dir is + when Dir_To => + Cmp_L := Id_Uge; + Cmp_R := Id_Ule; + when Dir_Downto => + Cmp_L := Id_Ule; + Cmp_R := Id_Uge; + end case; + L := Build2_Const_Uns + (Ctxt, Uns64 (Rng.Left), Choice_Typ.W); + R := Build2_Const_Uns + (Ctxt, Uns64 (Rng.Right), Choice_Typ.W); + end if; + + L := Build_Compare (Ctxt, Cmp_L, Sel, L); + Set_Location (L, Choice); + + R := Build_Compare (Ctxt, Cmp_R, Sel, R); + Set_Location (R, Choice); + + Cond := Build_Dyadic (Ctxt, Id_And, L, R); + Set_Location (Cond, Choice); + end; + + when Iir_Kind_Choice_By_Others => + -- Last and only one. + pragma Assert (Res = No_Net); + Other_Choice := Choice_Idx + 1; + pragma Assert (Get_Chain (Choice) = Null_Node); + Choice := Null_Node; + return; + end case; + + if not Get_Same_Alternative_Flag (Choice) then + -- First choice. + Choice_Idx := Choice_Idx + 1; + Res := Cond; + else + if Cond = No_Net then + -- No new condition. + null; + else + if Res /= No_Net then + Res := Build_Dyadic (Ctxt, Id_Or, Res, Cond); + Set_Location (Res, Choice); + else + Res := Cond; + end if; + end if; + end if; + + Choice := Get_Chain (Choice); + exit when Choice = Null_Node + or else not Get_Same_Alternative_Flag (Choice); + end loop; + if Res = No_Net then + Res := Build_Const_UB32 (Ctxt, 0, 1); + end if; + Nets (Choice_Idx) := Res; + end Synth_Choice; + + type Alternative_Data_Type is record + Asgns : Seq_Assign; + Val : Net; + end record; + type Alternative_Data_Array is + array (Alternative_Index range <>) of Alternative_Data_Type; + type Alternative_Data_Acc is access Alternative_Data_Array; + procedure Free_Alternative_Data_Array is new Ada.Unchecked_Deallocation + (Alternative_Data_Array, Alternative_Data_Acc); + + type Wire_Id_Array is array (Natural range <>) of Wire_Id; + type Wire_Id_Array_Acc is access Wire_Id_Array; + procedure Free_Wire_Id_Array is new Ada.Unchecked_Deallocation + (Wire_Id_Array, Wire_Id_Array_Acc); + + procedure Sort_Wire_Id_Array (Arr : in out Wire_Id_Array) + is + function Lt (Op1, Op2 : Natural) return Boolean is + begin + return Is_Lt (Arr (Op1), Arr (Op2)); + end Lt; + + procedure Swap (From : Natural; To : Natural) + is + T : Wire_Id; + begin + T := Arr (From); + Arr (From) := Arr (To); + Arr (To) := T; + end Swap; + + procedure Wid_Heap_Sort is + new Grt.Algos.Heap_Sort (Lt => Lt, Swap => Swap); + begin + Wid_Heap_Sort (Arr'Length); + end Sort_Wire_Id_Array; + + -- Count the number of wires used in all the alternatives. + function Count_Wires_In_Alternatives (Alts : Alternative_Data_Array) + return Natural + is + Res : Natural; + Asgn : Seq_Assign; + W : Wire_Id; + begin + Res := 0; + for I in Alts'Range loop + Asgn := Alts (I).Asgns; + while Asgn /= No_Seq_Assign loop + W := Get_Wire_Id (Asgn); + if not Get_Wire_Mark (W) then + Res := Res + 1; + Set_Wire_Mark (W, True); + end if; + Asgn := Get_Assign_Chain (Asgn); + end loop; + end loop; + return Res; + end Count_Wires_In_Alternatives; + + -- Fill ARR from wire_id of ALTS. + procedure Fill_Wire_Id_Array (Arr : out Wire_Id_Array; + Alts : Alternative_Data_Array) + is + Idx : Natural; + Asgn : Seq_Assign; + W : Wire_Id; + begin + Idx := Arr'First; + for I in Alts'Range loop + Asgn := Alts (I).Asgns; + while Asgn /= No_Seq_Assign loop + W := Get_Wire_Id (Asgn); + if Get_Wire_Mark (W) then + Arr (Idx) := W; + Idx := Idx + 1; + Set_Wire_Mark (W, False); + end if; + Asgn := Get_Assign_Chain (Asgn); + end loop; + end loop; + pragma Assert (Idx = Arr'Last + 1); + end Fill_Wire_Id_Array; + + type Seq_Assign_Value_Array_Acc is access Seq_Assign_Value_Array; + procedure Free_Seq_Assign_Value_Array is new Ada.Unchecked_Deallocation + (Seq_Assign_Value_Array, Seq_Assign_Value_Array_Acc); + + function Is_Assign_Value_Array_Static + (Wid : Wire_Id; Arr : Seq_Assign_Value_Array) return Memtyp + is + Res : Memtyp; + Prev_Val : Memtyp; + begin + Prev_Val := Null_Memtyp; + for I in Arr'Range loop + case Arr (I).Is_Static is + when False => + -- A value is not static. + return Null_Memtyp; + when Unknown => + if Prev_Val = Null_Memtyp then + -- First use of previous value. + if not Is_Static_Wire (Wid) then + -- The previous value is not static. + return Null_Memtyp; + end if; + Prev_Val := Get_Static_Wire (Wid); + if Res /= Null_Memtyp then + -- There is already a result. + if not Is_Equal (Res, Prev_Val) then + -- The previous value is different from the result. + return Null_Memtyp; + end if; + else + Res := Prev_Val; + end if; + end if; + when True => + if Res = Null_Memtyp then + -- First value. Keep it. + Res := Arr (I).Val; + else + if not Is_Equal (Res, Arr (I).Val) then + -- Value is different. + return Null_Memtyp; + end if; + end if; + end case; + end loop; + return Res; + end Is_Assign_Value_Array_Static; + + procedure Synth_Case_Statement_Dynamic + (C : in out Seq_Context; Stmt : Node; Sel : Valtyp) + is + use Vhdl.Sem_Expr; + Ctxt : constant Context_Acc := Get_Build (C.Inst); + + Choices : constant Node := Get_Case_Statement_Alternative_Chain (Stmt); + + Case_Info : Choice_Info_Type; + + -- Array of alternatives + Alts : Alternative_Data_Acc; + Alt_Idx : Alternative_Index; + Others_Alt_Idx : Alternative_Index; + + Nbr_Choices : Nat32; + + Pasgns : Seq_Assign_Value_Array_Acc; + Nets : Net_Array_Acc; + + Nbr_Wires : Natural; + Wires : Wire_Id_Array_Acc; + + Sel_Net : Net; + begin + -- Strategies to synthesize a case statement. Assume the selector is + -- a net of W bits + -- - a large mux, with 2**W inputs + -- - if the number of choices is dense + -- - if W is small + -- - a onehot mux. Each choice is converted to an single bit condition + -- by adding a comparison operator (equal for single choice, + -- inequalities for ranges, or for multiple choices). Only one of + -- these conditions is true (plus 'others'). + -- - if the number of choices is sparse + -- - large range choices + -- - a tree of mux/mux2 + -- - large number of choices, densily grouped but sparsed compared + -- to 2**W (eg: a partially filled memory) + -- - divide and conquier + + -- Count choices and alternatives. + Count_Choices (Case_Info, Choices); + --Fill_Choices_Array (Case_Info, Choices); + + -- Allocate structures. + -- Because there is no 1-1 link between choices and alternatives, + -- create an array for the choices and an array for the alternatives. + Alts := new Alternative_Data_Array + (1 .. Alternative_Index (Case_Info.Nbr_Alternatives)); + + -- Compute number of non-default alternatives. + Nbr_Choices := Nat32 (Case_Info.Nbr_Alternatives); + if Case_Info.Others_Choice /= Null_Node then + Nbr_Choices := Nbr_Choices - 1; + end if; + + Nets := new Net_Array (1 .. Int32 (Alts'Last)); + + Sel_Net := Get_Net (Ctxt, Sel); + + -- Synth statements and keep list of assignments. + -- Also synth choices. + declare + Choice : Node; + Choice_Idx, Other_Choice : Nat32; + Phi : Phi_Type; + begin + Alt_Idx := 0; + Choice_Idx := 0; + Other_Choice := 0; + + Choice := Choices; + while Is_Valid (Choice) loop + -- Must be a choice for a new alternative. + pragma Assert (not Get_Same_Alternative_Flag (Choice)); + + -- A new sequence of statements. + Alt_Idx := Alt_Idx + 1; + + Push_Phi; + Synth_Sequential_Statements (C, Get_Associated_Chain (Choice)); + Pop_Phi (Phi); + Alts (Alt_Idx).Asgns := Sort_Phi (Phi); + + Synth_Choice (C.Inst, Sel_Net, Sel.Typ, + Nets.all, Other_Choice, Choice_Idx, Choice); + end loop; + pragma Assert (Choice_Idx = Nbr_Choices); + Others_Alt_Idx := Alternative_Index (Other_Choice); + end; + + -- Create the one-hot vector. + if Nbr_Choices = 0 then + Sel_Net := No_Net; + else + Sel_Net := Build2_Concat (Ctxt, Nets (1 .. Nbr_Choices)); + end if; + + -- Create list of wire_id, sort it. + Nbr_Wires := Count_Wires_In_Alternatives (Alts.all); + Wires := new Wire_Id_Array (1 .. Nbr_Wires); + Fill_Wire_Id_Array (Wires.all, Alts.all); + Sort_Wire_Id_Array (Wires.all); + + -- Associate each choice with the assign node + -- For each wire_id: + -- Build mux2/mux4 tree (group by 4) + Pasgns := new Seq_Assign_Value_Array (1 .. Int32 (Alts'Last)); + + -- For each wire, compute the result. + for I in Wires'Range loop + declare + Wi : constant Wire_Id := Wires (I); + Last_Val : Net; + Res_Inst : Instance; + Res : Net; + Default : Net; + Min_Off, Off : Uns32; + Wd : Width; + List : Partial_Assign_List; + Sval : Memtyp; + begin + -- Extract the value for each branch. + for I in Alts'Range loop + -- If there is an assignment to Wi in Alt, it will define the + -- value. + if Get_Wire_Id (Alts (I).Asgns) = Wi then + Pasgns (Int32 (I)) := + Get_Seq_Assign_Value (Alts (I).Asgns); + Alts (I).Asgns := Get_Assign_Chain (Alts (I).Asgns); + else + Pasgns (Int32 (I)) := (Is_Static => Unknown); + end if; + end loop; + + -- If: + -- 1) All present values in PASGNS are static + -- 2) There is no missing values *or* the previous value is + -- static. + -- 3) The default value is unused *or* it is static + -- 4) All the values are equal. + -- then assign directly. + Sval := Is_Assign_Value_Array_Static (Wi, Pasgns.all); + if Sval /= Null_Memtyp then + -- Use static assignment. + Phi_Assign_Static (Wi, Sval); + else + -- Compute the final value for each partial part of the wire. + Partial_Assign_Init (List); + Min_Off := 0; + loop + Off := Min_Off; + + -- Extract value of partial assignments to NETS. + Extract_Merge_Partial_Assigns + (Ctxt, Pasgns.all, Nets.all, Off, Wd); + exit when Off = Uns32'Last and Wd = Width'Last; + + -- If a branch has no value, use the value before the case. + -- Also do it for the default value! + Last_Val := No_Net; + for I in Nets'Range loop + if Nets (I) = No_Net then + if Last_Val = No_Net then + Last_Val := Get_Current_Assign_Value + (Ctxt, Wi, Off, Wd); + end if; + Nets (I) := Last_Val; + end if; + end loop; + + -- Extract default value (for missing alternative). + if Others_Alt_Idx /= 0 then + Default := Nets (Int32 (Others_Alt_Idx)); + else + Default := Build_Const_X (Ctxt, Wd); + end if; + + if Nbr_Choices = 0 then + Res := Default; + else + Res := Build_Pmux (Ctxt, Sel_Net, Default); + Res_Inst := Get_Net_Parent (Res); + Set_Location (Res_Inst, Get_Location (Stmt)); + + for I in 1 .. Nbr_Choices loop + Connect + (Get_Input (Res_Inst, Port_Nbr (2 + I - Nets'First)), + Nets (I)); + end loop; + end if; + + Partial_Assign_Append (List, New_Partial_Assign (Res, Off)); + Min_Off := Off + Wd; + end loop; + + Merge_Partial_Assigns (Ctxt, Wi, List); + end if; + end; + end loop; + + -- free. + Free_Wire_Id_Array (Wires); + Free_Alternative_Data_Array (Alts); + Free_Seq_Assign_Value_Array (Pasgns); + Free_Net_Array (Nets); + end Synth_Case_Statement_Dynamic; + + procedure Synth_Case_Statement_Static_Array + (C : in out Seq_Context; Stmt : Node; Sel : Valtyp) + is + Choices : constant Node := Get_Case_Statement_Alternative_Chain (Stmt); + Choice : Node; + Stmts : Node; + Sel_Expr : Node; + Sel_Val : Valtyp; + begin + -- Synth statements, extract choice value. + Stmts := Null_Node; + Choice := Choices; + loop + pragma Assert (Is_Valid (Choice)); + if not Get_Same_Alternative_Flag (Choice) then + Stmts := Get_Associated_Chain (Choice); + end if; + + case Get_Kind (Choice) is + when Iir_Kind_Choice_By_Expression => + Sel_Expr := Get_Choice_Expression (Choice); + Sel_Val := Synth_Expression_With_Basetype (C.Inst, Sel_Expr); + if Is_Equal (Sel_Val, Sel) then + Synth_Sequential_Statements (C, Stmts); + exit; + end if; + when Iir_Kind_Choice_By_Others => + Synth_Sequential_Statements (C, Stmts); + exit; + when others => + raise Internal_Error; + end case; + Choice := Get_Chain (Choice); + end loop; + end Synth_Case_Statement_Static_Array; + + procedure Synth_Case_Statement_Static_Scalar + (C : in out Seq_Context; Stmt : Node; Sel : Int64) + is + Choices : constant Node := Get_Case_Statement_Alternative_Chain (Stmt); + Choice : Node; + Stmts : Node; + Sel_Expr : Node; + begin + -- Synth statements, extract choice value. + Stmts := Null_Node; + Choice := Choices; + loop + pragma Assert (Is_Valid (Choice)); + if not Get_Same_Alternative_Flag (Choice) then + Stmts := Get_Associated_Chain (Choice); + end if; + + case Get_Kind (Choice) is + when Iir_Kind_Choice_By_Expression => + Sel_Expr := Get_Choice_Expression (Choice); + if Vhdl.Evaluation.Eval_Pos (Sel_Expr) = Sel then + Synth_Sequential_Statements (C, Stmts); + exit; + end if; + when Iir_Kind_Choice_By_Others => + Synth_Sequential_Statements (C, Stmts); + exit; + when Iir_Kind_Choice_By_Range => + declare + Bnd : Discrete_Range_Type; + Is_In : Boolean; + begin + Synth_Discrete_Range + (C.Inst, Get_Choice_Range (Choice), Bnd); + case Bnd.Dir is + when Dir_To => + Is_In := Sel >= Bnd.Left and Sel <= Bnd.Right; + when Dir_Downto => + Is_In := Sel <= Bnd.Left and Sel >= Bnd.Right; + end case; + if Is_In then + Synth_Sequential_Statements (C, Stmts); + exit; + end if; + end; + when others => + raise Internal_Error; + end case; + Choice := Get_Chain (Choice); + end loop; + end Synth_Case_Statement_Static_Scalar; + + procedure Synth_Case_Statement (C : in out Seq_Context; Stmt : Node) + is + Expr : constant Node := Get_Expression (Stmt); + Sel : Valtyp; + begin + Sel := Synth_Expression_With_Basetype (C.Inst, Expr); + Strip_Const (Sel); + if Is_Static (Sel.Val) then + case Sel.Typ.Kind is + when Type_Bit + | Type_Logic + | Type_Discrete => + Synth_Case_Statement_Static_Scalar (C, Stmt, + Read_Discrete (Sel)); + when Type_Vector + | Type_Array => + Synth_Case_Statement_Static_Array (C, Stmt, Sel); + when others => + raise Internal_Error; + end case; + else + Synth_Case_Statement_Dynamic (C, Stmt, Sel); + end if; + end Synth_Case_Statement; + + procedure Synth_Selected_Signal_Assignment + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + use Vhdl.Sem_Expr; + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + + Expr : constant Node := Get_Expression (Stmt); + Choices : constant Node := Get_Selected_Waveform_Chain (Stmt); + + Targ : Target_Info; + Targ_Type : Type_Acc; + + Case_Info : Choice_Info_Type; + + -- Array of alternatives + Alts : Alternative_Data_Acc; + Alt_Idx : Alternative_Index; + Others_Alt_Idx : Alternative_Index; + + -- Array of choices. Contains tuple of (Value, Alternative). + Nbr_Choices : Nat32; + + Nets : Net_Array_Acc; + + + Sel : Valtyp; + Sel_Net : Net; + begin + Targ := Synth_Target (Syn_Inst, Get_Target (Stmt)); + Targ_Type := Targ.Targ_Type; + + -- Create a net for the expression. + Sel := Synth_Expression_With_Basetype (Syn_Inst, Expr); + Sel_Net := Get_Net (Ctxt, Sel); + + -- Count choices and alternatives. + Count_Choices (Case_Info, Choices); + -- Fill_Choices_Array (Case_Info, Choices); + + -- Allocate structures. + -- Because there is no 1-1 link between choices and alternatives, + -- create an array for the choices and an array for the alternatives. + Alts := new Alternative_Data_Array + (1 .. Alternative_Index (Case_Info.Nbr_Alternatives)); + + -- Compute number of non-default alternatives. + Nbr_Choices := Nat32 (Case_Info.Nbr_Alternatives); + if Case_Info.Others_Choice /= Null_Node then + Nbr_Choices := Nbr_Choices - 1; + end if; + + Nets := new Net_Array (1 .. Nbr_Choices); + + -- Synth statements, extract choice value. + declare + Choice, Wf : Node; + Val : Valtyp; + Choice_Idx, Other_Choice : Nat32; + begin + Alt_Idx := 0; + Choice_Idx := 0; + Other_Choice := 0; + + Choice := Choices; + while Is_Valid (Choice) loop + pragma Assert (not Get_Same_Alternative_Flag (Choice)); + + Wf := Get_Associated_Chain (Choice); + Val := Synth_Waveform (Syn_Inst, Wf, Targ_Type); + + Alt_Idx := Alt_Idx + 1; + Alts (Alt_Idx).Val := Get_Net (Ctxt, Val); + + Synth_Choice (Syn_Inst, Sel_Net, Sel.Typ, + Nets.all, Other_Choice, Choice_Idx, Choice); + end loop; + pragma Assert (Choice_Idx = Nbr_Choices); + Others_Alt_Idx := Alternative_Index (Other_Choice); + end; + + -- Create the one-hot vector. + if Nbr_Choices = 0 then + Sel_Net := No_Net; + else + Sel_Net := Build2_Concat (Ctxt, Nets (1 .. Nbr_Choices)); + end if; + + declare + Res : Net; + Res_Inst : Instance; + Default : Net; + begin + -- Extract default value (for missing alternative). + if Others_Alt_Idx /= 0 then + Default := Alts (Others_Alt_Idx).Val; + else + Default := Build_Const_X (Ctxt, Targ_Type.W); + end if; + + if Nbr_Choices = 0 then + Res := Default; + else + Res := Build_Pmux (Ctxt, Sel_Net, Default); + Res_Inst := Get_Net_Parent (Res); + Set_Location (Res_Inst, Get_Location (Stmt)); + + for I in 1 .. Nbr_Choices loop + Connect + (Get_Input (Res_Inst, Port_Nbr (2 + I - Nets'First)), + Alts (Alternative_Index (I)).Val); + end loop; + end if; + + Synth_Assignment + (Syn_Inst, Targ, Create_Value_Net (Res, Targ_Type), Stmt); + end; + + -- free. + Free_Alternative_Data_Array (Alts); + Free_Net_Array (Nets); + end Synth_Selected_Signal_Assignment; + + function Synth_Label (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + return Sname + is + Label : constant Name_Id := Get_Label (Stmt); + begin + if Label = Null_Identifier then + return No_Sname; + else + return New_Sname_User (Label, Get_Sname (Syn_Inst)); + end if; + end Synth_Label; + + function Is_Copyback_Interface (Inter : Node) return Boolean is + begin + case Iir_Parameter_Modes (Get_Mode (Inter)) is + when Iir_In_Mode => + return False; + when Iir_Out_Mode | Iir_Inout_Mode => + return Get_Kind (Inter) = Iir_Kind_Interface_Variable_Declaration; + end case; + end Is_Copyback_Interface; + + type Association_Iterator_Kind is + (Association_Function, + Association_Operator); + + type Association_Iterator_Init + (Kind : Association_Iterator_Kind := Association_Function) is + record + Inter_Chain : Node; + case Kind is + when Association_Function => + Assoc_Chain : Node; + when Association_Operator => + Left : Node; + Right : Node; + end case; + end record; + + function Association_Iterator_Build (Inter_Chain : Node; Assoc_Chain : Node) + return Association_Iterator_Init is + begin + return Association_Iterator_Init'(Kind => Association_Function, + Inter_Chain => Inter_Chain, + Assoc_Chain => Assoc_Chain); + end Association_Iterator_Build; + + function Association_Iterator_Build + (Inter_Chain : Node; Left : Node; Right : Node) + return Association_Iterator_Init is + begin + return Association_Iterator_Init'(Kind => Association_Operator, + Inter_Chain => Inter_Chain, + Left => Left, + Right => Right); + end Association_Iterator_Build; + + function Count_Associations (Init : Association_Iterator_Init) + return Natural + is + Assoc : Node; + Assoc_Inter : Node; + Inter : Node; + Nbr_Inout : Natural; + begin + case Init.Kind is + when Association_Function => + Nbr_Inout := 0; + + Assoc := Init.Assoc_Chain; + Assoc_Inter := Init.Inter_Chain; + while Is_Valid (Assoc) loop + Inter := Get_Association_Interface (Assoc, Assoc_Inter); + + if Is_Copyback_Interface (Inter) then + Nbr_Inout := Nbr_Inout + 1; + end if; + + Next_Association_Interface (Assoc, Assoc_Inter); + end loop; + + return Nbr_Inout; + when Association_Operator => + return 0; + end case; + end Count_Associations; + + type Association_Iterator + (Kind : Association_Iterator_Kind := Association_Function) is + record + Inter : Node; + case Kind is + when Association_Function => + First_Named_Assoc : Node; + Next_Assoc : Node; + when Association_Operator => + Op1 : Node; + Op2 : Node; + end case; + end record; + + procedure Association_Iterate_Init (Iterator : out Association_Iterator; + Init : Association_Iterator_Init) is + begin + case Init.Kind is + when Association_Function => + Iterator := (Kind => Association_Function, + Inter => Init.Inter_Chain, + First_Named_Assoc => Null_Node, + Next_Assoc => Init.Assoc_Chain); + when Association_Operator => + Iterator := (Kind => Association_Operator, + Inter => Init.Inter_Chain, + Op1 => Init.Left, + Op2 => Init.Right); + end case; + end Association_Iterate_Init; + + -- Return the next association. + -- ASSOC can be: + -- * an Iir_Kind_Association_By_XXX node (normal case) + -- * Null_Iir if INTER is not associated (and has a default value). + -- * an expression (for operator association). + procedure Association_Iterate_Next (Iterator : in out Association_Iterator; + Inter : out Node; + Assoc : out Node) + is + Formal : Node; + begin + Inter := Iterator.Inter; + if Inter = Null_Node then + -- End of iterator. + Assoc := Null_Node; + return; + else + -- Advance to the next interface for the next call. + Iterator.Inter := Get_Chain (Iterator.Inter); + end if; + + case Iterator.Kind is + when Association_Function => + if Iterator.First_Named_Assoc = Null_Node then + Assoc := Iterator.Next_Assoc; + if Assoc = Null_Node then + -- No more association: open association. + return; + end if; + Formal := Get_Formal (Assoc); + if Formal = Null_Node then + -- Association by position. + -- Update for the next call. + Iterator.Next_Assoc := Get_Chain (Assoc); + return; + end if; + Iterator.First_Named_Assoc := Assoc; + end if; + + -- Search by name. + Assoc := Iterator.First_Named_Assoc; + while Assoc /= Null_Node loop + Formal := Get_Formal (Assoc); + pragma Assert (Formal /= Null_Node); + Formal := Get_Interface_Of_Formal (Formal); + if Formal = Inter then + -- Found. + -- Optimize in case assocs are in order. + if Assoc = Iterator.First_Named_Assoc then + Iterator.First_Named_Assoc := Get_Chain (Assoc); + end if; + return; + end if; + Assoc := Get_Chain (Assoc); + end loop; + + -- Not found: open association. + return; + + when Association_Operator => + Assoc := Iterator.Op1; + Iterator.Op1 := Iterator.Op2; + Iterator.Op2 := Null_Node; + end case; + end Association_Iterate_Next; + + procedure Synth_Subprogram_Association (Subprg_Inst : Synth_Instance_Acc; + Caller_Inst : Synth_Instance_Acc; + Init : Association_Iterator_Init; + Infos : out Target_Info_Array) + is + pragma Assert (Infos'First = 1); + Ctxt : constant Context_Acc := Get_Build (Caller_Inst); + Inter : Node; + Inter_Type : Type_Acc; + Assoc : Node; + Actual : Node; + Val : Valtyp; + Nbr_Inout : Natural; + Iterator : Association_Iterator; + Info : Target_Info; + begin + Set_Instance_Const (Subprg_Inst, True); + + Nbr_Inout := 0; + + -- Process in INTER order. + Association_Iterate_Init (Iterator, Init); + loop + Association_Iterate_Next (Iterator, Inter, Assoc); + exit when Inter = Null_Node; + + Inter_Type := Get_Subtype_Object (Caller_Inst, Get_Type (Inter)); + + case Iir_Parameter_Modes (Get_Mode (Inter)) is + when Iir_In_Mode => + if Assoc = Null_Node + or else Get_Kind (Assoc) = Iir_Kind_Association_Element_Open + then + Actual := Get_Default_Value (Inter); + Val := Synth_Expression_With_Type + (Subprg_Inst, Actual, Inter_Type); + else + if Get_Kind (Assoc) = + Iir_Kind_Association_Element_By_Expression + then + Actual := Get_Actual (Assoc); + else + Actual := Assoc; + end if; + Val := Synth_Expression_With_Type + (Caller_Inst, Actual, Inter_Type); + end if; + when Iir_Out_Mode | Iir_Inout_Mode => + Actual := Get_Actual (Assoc); + Info := Synth_Target (Caller_Inst, Actual); + + case Iir_Kinds_Interface_Object_Declaration (Get_Kind (Inter)) + is + when Iir_Kind_Interface_Constant_Declaration => + raise Internal_Error; + when Iir_Kind_Interface_Variable_Declaration => + -- Always pass by value. + Nbr_Inout := Nbr_Inout + 1; + Infos (Nbr_Inout) := Info; + if Info.Kind /= Target_Memory + and then Is_Static (Info.Obj.Val) + then + Val := Create_Value_Memory (Info.Targ_Type); + Copy_Memory (Val.Val.Mem, + Info.Obj.Val.Mem + Info.Off.Mem_Off, + Info.Targ_Type.Sz); + else + Val := Synth_Read (Caller_Inst, Info, Assoc); + end if; + when Iir_Kind_Interface_Signal_Declaration => + -- Always pass by reference (use an alias). + if Info.Kind = Target_Memory then + raise Internal_Error; + end if; + Val := Create_Value_Alias + (Info.Obj, Info.Off, Info.Targ_Type); + when Iir_Kind_Interface_File_Declaration => + Val := Info.Obj; + when Iir_Kind_Interface_Quantity_Declaration => + raise Internal_Error; + end case; + end case; + + if Val = No_Valtyp then + Set_Error (Subprg_Inst); + return; + end if; + + -- FIXME: conversion only for constants, reshape for all. + Val := Synth_Subtype_Conversion (Ctxt, Val, Inter_Type, True, Assoc); + + if Get_Instance_Const (Subprg_Inst) and then not Is_Static (Val.Val) + then + Set_Instance_Const (Subprg_Inst, False); + end if; + + case Iir_Kinds_Interface_Object_Declaration (Get_Kind (Inter)) is + when Iir_Kind_Interface_Constant_Declaration => + -- Pass by reference. + Create_Object (Subprg_Inst, Inter, Val); + when Iir_Kind_Interface_Variable_Declaration => + -- Arguments are passed by copy. + if Is_Static (Val.Val) or else Get_Mode (Inter) = Iir_In_Mode + then + Val := Unshare (Val, Current_Pool); + else + -- Will be changed to a wire. + null; + end if; + Create_Object (Subprg_Inst, Inter, Val); + when Iir_Kind_Interface_Signal_Declaration => + Create_Object (Subprg_Inst, Inter, Val); + when Iir_Kind_Interface_File_Declaration => + Create_Object (Subprg_Inst, Inter, Val); + when Iir_Kind_Interface_Quantity_Declaration => + raise Internal_Error; + end case; + end loop; + end Synth_Subprogram_Association; + + procedure Synth_Subprogram_Association (Subprg_Inst : Synth_Instance_Acc; + Caller_Inst : Synth_Instance_Acc; + Inter_Chain : Node; + Assoc_Chain : Node) + is + Infos : Target_Info_Array (1 .. 0); + pragma Unreferenced (Infos); + Init : Association_Iterator_Init; + begin + Init := Association_Iterator_Build (Inter_Chain, Assoc_Chain); + Synth_Subprogram_Association (Subprg_Inst, Caller_Inst, Init, Infos); + end Synth_Subprogram_Association; + + -- Create wires for out and inout interface variables. + procedure Synth_Subprogram_Association_Wires + (Subprg_Inst : Synth_Instance_Acc; Init : Association_Iterator_Init) + is + Ctxt : constant Context_Acc := Get_Build (Subprg_Inst); + Inter : Node; + Assoc : Node; + Val : Valtyp; + Iterator : Association_Iterator; + Wire : Wire_Id; + begin + -- Process in INTER order. + Association_Iterate_Init (Iterator, Init); + loop + Association_Iterate_Next (Iterator, Inter, Assoc); + exit when Inter = Null_Node; + + if Get_Mode (Inter) in Iir_Out_Modes + and then Get_Kind (Inter) = Iir_Kind_Interface_Variable_Declaration + then + Val := Get_Value (Subprg_Inst, Inter); + -- Arguments are passed by copy. + Wire := Alloc_Wire (Wire_Variable, (Inter, Val.Typ)); + Set_Wire_Gate (Wire, Get_Net (Ctxt, Val)); + + Val := Create_Value_Wire (Wire, Val.Typ); + Create_Object_Force (Subprg_Inst, Inter, No_Valtyp); + Create_Object_Force (Subprg_Inst, Inter, Val); + end if; + end loop; + end Synth_Subprogram_Association_Wires; + + procedure Synth_Subprogram_Back_Association + (Subprg_Inst : Synth_Instance_Acc; + Caller_Inst : Synth_Instance_Acc; + Init : Association_Iterator_Init; + Infos : Target_Info_Array) + is + pragma Assert (Infos'First = 1); + Inter : Node; + Assoc : Node; + Assoc_Inter : Node; + Val : Valtyp; + Nbr_Inout : Natural; + begin + Nbr_Inout := 0; + pragma Assert (Init.Kind = Association_Function); + Assoc := Init.Assoc_Chain; + Assoc_Inter := Init.Inter_Chain; + while Is_Valid (Assoc) loop + Inter := Get_Association_Interface (Assoc, Assoc_Inter); + + if Is_Copyback_Interface (Inter) then + if not Get_Whole_Association_Flag (Assoc) then + raise Internal_Error; + end if; + Nbr_Inout := Nbr_Inout + 1; + Val := Get_Value (Subprg_Inst, Inter); + Synth_Assignment (Caller_Inst, Infos (Nbr_Inout), Val, Assoc); + + -- Free wire used for out/inout interface variables. + if Val.Val.Kind = Value_Wire then + Phi_Discard_Wires (Val.Val.W, No_Wire_Id); + Free_Wire (Val.Val.W); + end if; + end if; + + Next_Association_Interface (Assoc, Assoc_Inter); + end loop; + pragma Assert (Nbr_Inout = Infos'Last); + end Synth_Subprogram_Back_Association; + + function Build_Control_Signal (Syn_Inst : Synth_Instance_Acc; + W : Width; + Loc : Source.Syn_Src) return Net + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Res : Net; + begin + Res := Build_Signal (Ctxt, New_Internal_Name (Ctxt), W); + Set_Location (Res, Loc); + return Res; + end Build_Control_Signal; + + function Synth_Dynamic_Subprogram_Call (Syn_Inst : Synth_Instance_Acc; + Sub_Inst : Synth_Instance_Acc; + Call : Node; + Init : Association_Iterator_Init; + Infos : Target_Info_Array) + return Valtyp + is + Imp : constant Node := Get_Implementation (Call); + Is_Func : constant Boolean := Is_Function_Declaration (Imp); + Bod : constant Node := Vhdl.Sem_Inst.Get_Subprogram_Body_Origin (Imp); + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Res : Valtyp; + C : Seq_Context (Mode_Dynamic); + Wire_Mark : Wire_Id; + Subprg_Phi : Phi_Type; + begin + Mark (Wire_Mark); + C := (Mode => Mode_Dynamic, + Inst => Sub_Inst, + Cur_Loop => null, + W_En => No_Wire_Id, + W_Ret => No_Wire_Id, + W_Val => No_Wire_Id, + Ret_Init => No_Net, + Ret_Value => No_Valtyp, + Ret_Typ => null, + Nbr_Ret => 0); + + C.W_En := Alloc_Wire (Wire_Variable, (Imp, Bit_Type)); + C.W_Ret := Alloc_Wire (Wire_Variable, (Imp, Bit_Type)); + + if Is_Func then + C.W_Val := Alloc_Wire (Wire_Variable, (Imp, null)); + end if; + + -- Create a phi so that all assignments are gathered. + Push_Phi; + + Synth_Subprogram_Association_Wires (Sub_Inst, Init); + + if Is_Func then + -- Set a default value for the return. + C.Ret_Typ := Get_Subtype_Object (Syn_Inst, Get_Return_Type (Imp)); + + Set_Wire_Gate (C.W_Val, + Build_Control_Signal (Sub_Inst, C.Ret_Typ.W, Imp)); + C.Ret_Init := Build_Const_X (Ctxt, C.Ret_Typ.W); + Phi_Assign_Net (Ctxt, C.W_Val, C.Ret_Init, 0); + end if; + + Set_Wire_Gate + (C.W_En, Build_Control_Signal (Sub_Inst, 1, Imp)); + Phi_Assign_Static (C.W_En, Bit1); + + Set_Wire_Gate + (C.W_Ret, Build_Control_Signal (Sub_Inst, 1, Imp)); + Phi_Assign_Static (C.W_Ret, Bit1); + + Vhdl_Decls.Synth_Declarations + (C.Inst, Get_Declaration_Chain (Bod), True); + if not Is_Error (C.Inst) then + Synth_Sequential_Statements (C, Get_Sequential_Statement_Chain (Bod)); + end if; + + if Is_Error (C.Inst) then + Res := No_Valtyp; + else + if Is_Func then + if C.Nbr_Ret = 0 then + raise Internal_Error; + elsif C.Nbr_Ret = 1 and then Is_Static (C.Ret_Value.Val) then + Res := C.Ret_Value; + else + Res := Create_Value_Net + (Get_Current_Value (Ctxt, C.W_Val), C.Ret_Value.Typ); + end if; + else + Res := No_Valtyp; + Synth_Subprogram_Back_Association (C.Inst, Syn_Inst, Init, Infos); + end if; + end if; + + Pop_Phi (Subprg_Phi); + + Vhdl_Decls.Finalize_Declarations + (C.Inst, Get_Declaration_Chain (Bod), True); + pragma Unreferenced (Infos); + + -- Propagate assignments. + -- Wires that have been created for this subprogram will be destroyed. + -- But assignment for outer wires (passed through parameters) have + -- to be kept. We cannot merge phi because this won't be allowed for + -- local wires. + Propagate_Phi_Until_Mark (Ctxt, Subprg_Phi, Wire_Mark); + + -- Free wires. + -- These wires are currently unassigned because they were created + -- within the Phi. + Free_Wire (C.W_En); + Free_Wire (C.W_Ret); + if Is_Func then + Free_Wire (C.W_Val); + end if; + + Release (Wire_Mark); + + return Res; + end Synth_Dynamic_Subprogram_Call; + + function Synth_Static_Subprogram_Call (Syn_Inst : Synth_Instance_Acc; + Sub_Inst : Synth_Instance_Acc; + Call : Node; + Bod : Node; + Init : Association_Iterator_Init; + Infos : Target_Info_Array) + return Valtyp + is + Imp : constant Node := Get_Implementation (Call); + Is_Func : constant Boolean := Is_Function_Declaration (Imp); + Res : Valtyp; + C : Seq_Context (Mode_Static); + begin + C := (Mode_Static, + Inst => Sub_Inst, + Cur_Loop => null, + S_En => True, + Ret_Value => No_Valtyp, + Ret_Typ => null, + Nbr_Ret => 0); + + if Is_Func then + -- Set a default value for the return. + C.Ret_Typ := Get_Subtype_Object (Syn_Inst, Get_Return_Type (Imp)); + end if; + + Synth_Declarations (C.Inst, Get_Declaration_Chain (Bod), True); + + if not Is_Error (C.Inst) then + Synth_Sequential_Statements (C, Get_Sequential_Statement_Chain (Bod)); + end if; + + if Is_Error (C.Inst) then + Res := No_Valtyp; + else + if Is_Func then + if C.Nbr_Ret = 0 then + Error_Msg_Synth + (+Call, "function call completed without a return statement"); + Res := No_Valtyp; + else + pragma Assert (C.Nbr_Ret = 1); + pragma Assert (Is_Static (C.Ret_Value.Val)); + Res := C.Ret_Value; + end if; + else + Res := No_Valtyp; + Synth_Subprogram_Back_Association (C.Inst, Syn_Inst, Init, Infos); + end if; + end if; + + Vhdl_Decls.Finalize_Declarations + (C.Inst, Get_Declaration_Chain (Bod), True); + pragma Unreferenced (Infos); + + return Res; + end Synth_Static_Subprogram_Call; + + function Synth_Subprogram_Call (Syn_Inst : Synth_Instance_Acc; + Call : Node; + Init : Association_Iterator_Init) + return Valtyp + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Imp : constant Node := Get_Implementation (Call); + Is_Func : constant Boolean := Is_Function_Declaration (Imp); + Bod : constant Node := Vhdl.Sem_Inst.Get_Subprogram_Body_Origin (Imp); + Nbr_Inout : constant Natural := Count_Associations (Init); + Infos : Target_Info_Array (1 .. Nbr_Inout); + Area_Mark : Areapools.Mark_Type; + Res : Valtyp; + Sub_Inst : Synth_Instance_Acc; + Up_Inst : Synth_Instance_Acc; + begin + Areapools.Mark (Area_Mark, Instance_Pool.all); + + Up_Inst := Get_Instance_By_Scope (Syn_Inst, Get_Parent_Scope (Imp)); + Sub_Inst := Make_Instance (Up_Inst, Bod, New_Internal_Name (Ctxt)); + Set_Instance_Base (Sub_Inst, Syn_Inst); + + Synth_Subprogram_Association (Sub_Inst, Syn_Inst, Init, Infos); + + if Is_Error (Sub_Inst) then + Res := No_Valtyp; + else + if not Is_Func then + if Get_Purity_State (Imp) /= Pure then + Set_Instance_Const (Sub_Inst, False); + end if; + end if; + + if Get_Instance_Const (Sub_Inst) then + Res := Synth_Static_Subprogram_Call + (Syn_Inst, Sub_Inst, Call, Bod, Init, Infos); + else + Res := Synth_Dynamic_Subprogram_Call + (Syn_Inst, Sub_Inst, Call, Init, Infos); + end if; + end if; + + -- Propagate error. + if Is_Error (Sub_Inst) then + Set_Error (Syn_Inst); + end if; + + if Debugger.Flag_Need_Debug then + Debugger.Debug_Leave (Sub_Inst); + end if; + + Free_Instance (Sub_Inst); + Areapools.Release (Area_Mark, Instance_Pool.all); + + return Res; + end Synth_Subprogram_Call; + + function Synth_Subprogram_Call + (Syn_Inst : Synth_Instance_Acc; Call : Node) return Valtyp + is + Imp : constant Node := Get_Implementation (Call); + Assoc_Chain : constant Node := Get_Parameter_Association_Chain (Call); + Inter_Chain : constant Node := Get_Interface_Declaration_Chain (Imp); + Init : Association_Iterator_Init; + begin + Init := Association_Iterator_Build (Inter_Chain, Assoc_Chain); + return Synth_Subprogram_Call (Syn_Inst, Call, Init); + end Synth_Subprogram_Call; + + function Synth_User_Operator (Syn_Inst : Synth_Instance_Acc; + Left_Expr : Node; + Right_Expr : Node; + Expr : Node) return Valtyp + is + Imp : constant Node := Get_Implementation (Expr); + Inter_Chain : constant Node := Get_Interface_Declaration_Chain (Imp); + Init : Association_Iterator_Init; + begin + Init := Association_Iterator_Build (Inter_Chain, Left_Expr, Right_Expr); + return Synth_Subprogram_Call (Syn_Inst, Expr, Init); + end Synth_User_Operator; + + procedure Synth_Implicit_Procedure_Call + (Syn_Inst : Synth_Instance_Acc; Call : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Imp : constant Node := Get_Implementation (Call); + Assoc_Chain : constant Node := Get_Parameter_Association_Chain (Call); + Inter_Chain : constant Node := Get_Interface_Declaration_Chain (Imp); + Init : constant Association_Iterator_Init := + Association_Iterator_Build (Inter_Chain, Assoc_Chain); + Nbr_Inout : constant Natural := Count_Associations (Init); + Infos : Target_Info_Array (1 .. Nbr_Inout); + Area_Mark : Areapools.Mark_Type; + Sub_Inst : Synth_Instance_Acc; + begin + Areapools.Mark (Area_Mark, Instance_Pool.all); + Sub_Inst := Make_Instance (Syn_Inst, Imp, New_Internal_Name (Ctxt)); + + Synth_Subprogram_Association (Sub_Inst, Syn_Inst, Init, Infos); + + Synth.Vhdl_Static_Proc.Synth_Static_Procedure (Sub_Inst, Imp, Call); + + Synth_Subprogram_Back_Association (Sub_Inst, Syn_Inst, Init, Infos); + + Free_Instance (Sub_Inst); + Areapools.Release (Area_Mark, Instance_Pool.all); + end Synth_Implicit_Procedure_Call; + + procedure Synth_Procedure_Call + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Call : constant Node := Get_Procedure_Call (Stmt); + Imp : constant Node := Get_Implementation (Call); + Res : Valtyp; + begin + case Get_Implicit_Definition (Imp) is + when Iir_Predefined_None => + if Get_Foreign_Flag (Imp) then + Error_Msg_Synth + (+Stmt, "call to foreign %n is not supported", +Imp); + else + Res := Synth_Subprogram_Call (Syn_Inst, Call); + pragma Assert (Res = No_Valtyp); + end if; + when others => + Synth_Implicit_Procedure_Call (Syn_Inst, Call); + end case; + end Synth_Procedure_Call; + + procedure Update_Index (Rng : Discrete_Range_Type; V : in out Valtyp) + is + T : Int64; + begin + T := Read_Discrete (V); + case Rng.Dir is + when Dir_To => + T := T + 1; + when Dir_Downto => + T := T - 1; + end case; + Write_Discrete (V, T); + end Update_Index; + + -- Return True iff WID is a static wire and its value is V. + function Is_Static_Bit (Wid : Wire_Id; V : Ghdl_U8) return Boolean + is + M : Memtyp; + begin + if not Is_Static_Wire (Wid) then + return False; + end if; + M := Get_Static_Wire (Wid); + return Read_U8 (M) = V; + end Is_Static_Bit; + + function Is_Static_Bit0 (Wid : Wire_Id) return Boolean is + begin + return Is_Static_Bit (Wid, 0); + end Is_Static_Bit0; + + function Is_Static_Bit1 (Wid : Wire_Id) return Boolean is + begin + return Is_Static_Bit (Wid, 1); + end Is_Static_Bit1; + + pragma Inline (Is_Static_Bit0); + pragma Inline (Is_Static_Bit1); + + procedure Loop_Control_Init (C : Seq_Context; Stmt : Node) + is + Lc : constant Loop_Context_Acc := C.Cur_Loop; + begin + -- We might create new wires that will be destroy at the end of the + -- loop. Use mark and sweep to control their lifetime. + Mark (C.Cur_Loop.Wire_Mark); + + if Lc.Prev_Loop /= null and then Lc.Prev_Loop.Need_Quit then + -- An exit or next statement that targets an outer loop may suspend + -- the execution of this loop. + Lc.W_Quit := Alloc_Wire (Wire_Variable, (Lc.Loop_Stmt, Bit_Type)); + Set_Wire_Gate (Lc.W_Quit, Build_Control_Signal (C.Inst, 1, Stmt)); + Phi_Assign_Static (Lc.W_Quit, Bit1); + end if; + + if Get_Exit_Flag (Stmt) or else Get_Next_Flag (Stmt) then + -- There is an exit or next statement that target this loop. + -- We need to save W_En, as if the execution is suspended due to + -- exit or next, it will resume at the end of the loop. + if Is_Static_Wire (C.W_En) then + pragma Assert (Is_Static_Bit1 (C.W_En)); + Lc.Saved_En := No_Net; + else + Lc.Saved_En := Get_Current_Value (null, C.W_En); + end if; + -- Subloops may be suspended if there is an exit or a next statement + -- for this loop within subloops. + Lc.Need_Quit := True; + end if; + + if Get_Exit_Flag (Stmt) then + -- There is an exit statement for this loop. Create the wire. + Lc.W_Exit := Alloc_Wire (Wire_Variable, (Lc.Loop_Stmt, Bit_Type)); + Set_Wire_Gate (Lc.W_Exit, Build_Control_Signal (C.Inst, 1, Stmt)); + Phi_Assign_Static (Lc.W_Exit, Bit1); + end if; + end Loop_Control_Init; + + procedure Loop_Control_And_Start (Is_Net : out Boolean; + S : out Boolean; + N : out Net; + En : Net) is + begin + if En = No_Net then + Is_Net := False; + N := No_Net; + S := True; + else + Is_Net := True; + N := En; + S := True; + end if; + end Loop_Control_And_Start; + + procedure Loop_Control_And (C : Seq_Context; + Is_Net : in out Boolean; + S : in out Boolean; + N : in out Net; + R : Wire_Id) + is + Res : Net; + begin + if R = No_Wire_Id or else Is_Static_Bit1 (R) then + -- No change. + return; + end if; + + if Is_Static_Bit0 (R) then + -- Stays 0. + Is_Net := False; + S := False; + N := No_Net; + return; + end if; + + if not Is_Net and then not S then + -- Was 0, remains 0. + return; + end if; + + pragma Assert (Is_Net or else S); + + -- Optimize common cases. + Res := Get_Current_Value (null, R); + + if Is_Net then + N := Build_Dyadic (Get_Build (C.Inst), Id_And, N, Res); + Set_Location (N, C.Cur_Loop.Loop_Stmt); + else + N := Res; + end if; + + Is_Net := True; + end Loop_Control_And; + + procedure Loop_Control_And_Assign (C : Seq_Context; + Is_Net : Boolean; + S : Boolean; + N : Net; + W : Wire_Id) is + begin + if Is_Net then + Phi_Assign_Net (Get_Build (C.Inst), W, N, 0); + else + if S then + Phi_Assign_Static (W, Bit1); + else + Phi_Assign_Static (W, Bit0); + end if; + end if; + end Loop_Control_And_Assign; + + procedure Loop_Control_Update (C : Seq_Context) + is + Lc : constant Loop_Context_Acc := C.Cur_Loop; + N : Net; + S : Boolean; + Is_Net : Boolean; + begin + if not Lc.Need_Quit then + -- No next/exit statement for this loop. So no control. + return; + end if; + + -- Execution continue iff: + -- 1. Loop was enabled (Lc.Saved_En) + Loop_Control_And_Start (Is_Net, S, N, Lc.Saved_En); + + -- 2. No return (C.W_Ret) + Loop_Control_And (C, Is_Net, S, N, C.W_Ret); + + -- 3. No exit. + Loop_Control_And (C, Is_Net, S, N, Lc.W_Exit); + + -- 4. No quit. + Loop_Control_And (C, Is_Net, S, N, Lc.W_Quit); + + Loop_Control_And_Assign (C, Is_Net, S, N, C.W_En); + end Loop_Control_Update; + + procedure Loop_Control_Finish (C : Seq_Context) + is + Lc : constant Loop_Context_Acc := C.Cur_Loop; + N : Net; + S : Boolean; + Is_Net : Boolean; + begin + -- Execution continue after this loop iff: + -- 1. Loop was enabled (Lc.Saved_En) + Loop_Control_And_Start (Is_Net, S, N, Lc.Saved_En); + + -- 2. No return (C.W_Ret) + Loop_Control_And (C, Is_Net, S, N, C.W_Ret); + + -- 3. No quit (C.W_Quit) + Loop_Control_And (C, Is_Net, S, N, Lc.W_Quit); + + Phi_Discard_Wires (Lc.W_Quit, Lc.W_Exit); + + if Lc.W_Quit /= No_Wire_Id then + Free_Wire (Lc.W_Quit); + end if; + + if Lc.W_Exit /= No_Wire_Id then + Free_Wire (Lc.W_Exit); + end if; + + Release (C.Cur_Loop.Wire_Mark); + + Loop_Control_And_Assign (C, Is_Net, S, N, C.W_En); + end Loop_Control_Finish; + + procedure Synth_Dynamic_Exit_Next_Statement + (C : in out Seq_Context; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (C.Inst); + Cond : constant Node := Get_Condition (Stmt); + Is_Exit : constant Boolean := Get_Kind (Stmt) = Iir_Kind_Exit_Statement; + Static_Cond : Boolean; + Loop_Label : Node; + Lc : Loop_Context_Acc; + Cond_Val : Valtyp; + Phi_True : Phi_Type; + Phi_False : Phi_Type; + begin + if Cond /= Null_Node then + Cond_Val := Synth_Expression (C.Inst, Cond); + Static_Cond := Is_Static_Val (Cond_Val.Val); + if Static_Cond then + if Get_Static_Discrete (Cond_Val) = 0 then + -- Not executed. + return; + end if; + else + -- Create a branch for the True case. + Push_Phi; + end if; + end if; + + -- Execution is suspended for the current sequence of statements. + Phi_Assign_Static (C.W_En, Bit0); + + Lc := C.Cur_Loop; + + -- Compute the loop statement indicated by the exit/next statement. + Loop_Label := Get_Loop_Label (Stmt); + if Loop_Label = Null_Node then + Loop_Label := Lc.Loop_Stmt; + else + Loop_Label := Get_Named_Entity (Loop_Label); + end if; + + -- Update the W_Exit and W_Quit flags for the loops. All the loops + -- until the label are canceled. + loop + if Lc.Loop_Stmt = Loop_Label then + -- Final loop. + if Is_Exit then + Phi_Assign_Static (Lc.W_Exit, Bit0); + end if; + exit; + else + Phi_Assign_Static (Lc.W_Quit, Bit0); + end if; + Lc := Lc.Prev_Loop; + end loop; + + if Cond /= Null_Node and not Static_Cond then + Pop_Phi (Phi_True); + + -- If the condition is false, do nothing. + Push_Phi; + Pop_Phi (Phi_False); + + Merge_Phis (Ctxt, Get_Net (Ctxt, Cond_Val), Phi_True, Phi_False, + Get_Location (Stmt)); + end if; + end Synth_Dynamic_Exit_Next_Statement; + + procedure Synth_Static_Exit_Next_Statement + (C : in out Seq_Context; Stmt : Node) + is + Cond : constant Node := Get_Condition (Stmt); + Is_Exit : constant Boolean := Get_Kind (Stmt) = Iir_Kind_Exit_Statement; + Loop_Label : Node; + Lc : Loop_Context_Acc; + Cond_Val : Valtyp; + begin + if Cond /= Null_Node then + Cond_Val := Synth_Expression (C.Inst, Cond); + if Cond_Val = No_Valtyp then + Set_Error (C.Inst); + return; + end if; + pragma Assert (Is_Static_Val (Cond_Val.Val)); + if Get_Static_Discrete (Cond_Val) = 0 then + -- Not executed. + return; + end if; + end if; + + -- Execution is suspended. + C.S_En := False; + + Lc := C.Cur_Loop; + + Loop_Label := Get_Loop_Label (Stmt); + if Loop_Label = Null_Node then + Loop_Label := Lc.Loop_Stmt; + else + Loop_Label := Get_Named_Entity (Loop_Label); + end if; + + loop + if Lc.Loop_Stmt = Loop_Label then + if Is_Exit then + Lc.S_Exit := True; + end if; + exit; + else + Lc.S_Quit := True; + end if; + Lc := Lc.Prev_Loop; + end loop; + end Synth_Static_Exit_Next_Statement; + + procedure Init_For_Loop_Statement (C : in out Seq_Context; + Stmt : Node; + Val : out Valtyp) + is + Iterator : constant Node := Get_Parameter_Specification (Stmt); + It_Type : constant Node := Get_Declaration_Type (Iterator); + It_Rng : Type_Acc; + begin + if It_Type /= Null_Node then + Synth_Subtype_Indication (C.Inst, It_Type); + end if; + + -- Initial value. + It_Rng := Get_Subtype_Object (C.Inst, Get_Type (Iterator)); + Val := Create_Value_Discrete (It_Rng.Drange.Left, It_Rng); + Create_Object (C.Inst, Iterator, Val); + end Init_For_Loop_Statement; + + procedure Finish_For_Loop_Statement (C : in out Seq_Context; + Stmt : Node) + is + Iterator : constant Node := Get_Parameter_Specification (Stmt); + It_Type : constant Node := Get_Declaration_Type (Iterator); + begin + Destroy_Object (C.Inst, Iterator); + if It_Type /= Null_Node then + Destroy_Object (C.Inst, It_Type); + end if; + end Finish_For_Loop_Statement; + + procedure Synth_Dynamic_For_Loop_Statement + (C : in out Seq_Context; Stmt : Node) + is + Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); + Val : Valtyp; + Lc : aliased Loop_Context (Mode_Dynamic); + begin + Lc := (Mode => Mode_Dynamic, + Prev_Loop => C.Cur_Loop, + Loop_Stmt => Stmt, + Need_Quit => False, + Saved_En => No_Net, + W_Exit => No_Wire_Id, + W_Quit => No_Wire_Id, + Wire_Mark => No_Wire_Id); + C.Cur_Loop := Lc'Unrestricted_Access; + + Loop_Control_Init (C, Stmt); + + Init_For_Loop_Statement (C, Stmt, Val); + + while In_Range (Val.Typ.Drange, Read_Discrete (Val)) loop + Synth_Sequential_Statements (C, Stmts); + + Update_Index (Val.Typ.Drange, Val); + Loop_Control_Update (C); + + -- Constant exit. + exit when Is_Static_Bit0 (C.W_En); + + -- FIXME: dynamic exits. + end loop; + Loop_Control_Finish (C); + + Finish_For_Loop_Statement (C, Stmt); + + C.Cur_Loop := Lc.Prev_Loop; + end Synth_Dynamic_For_Loop_Statement; + + procedure Synth_Static_For_Loop_Statement + (C : in out Seq_Context; Stmt : Node) + is + Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); + Val : Valtyp; + Lc : aliased Loop_Context (Mode_Static); + begin + Lc := (Mode_Static, + Prev_Loop => C.Cur_Loop, + Loop_Stmt => Stmt, + S_Exit => False, + S_Quit => False); + C.Cur_Loop := Lc'Unrestricted_Access; + + Init_For_Loop_Statement (C, Stmt, Val); + + while In_Range (Val.Typ.Drange, Read_Discrete (Val)) loop + Synth_Sequential_Statements (C, Stmts); + C.S_En := True; + + Update_Index (Val.Typ.Drange, Val); + + exit when Lc.S_Exit or Lc.S_Quit or C.Nbr_Ret > 0; + end loop; + + Finish_For_Loop_Statement (C, Stmt); + + C.Cur_Loop := Lc.Prev_Loop; + end Synth_Static_For_Loop_Statement; + + procedure Synth_Dynamic_While_Loop_Statement + (C : in out Seq_Context; Stmt : Node) + is + Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); + Cond : constant Node := Get_Condition (Stmt); + Val : Valtyp; + Lc : aliased Loop_Context (Mode_Dynamic); + Iter_Nbr : Natural; + begin + Lc := (Mode => Mode_Dynamic, + Prev_Loop => C.Cur_Loop, + Loop_Stmt => Stmt, + Need_Quit => False, + Saved_En => No_Net, + W_Exit => No_Wire_Id, + W_Quit => No_Wire_Id, + Wire_Mark => No_Wire_Id); + C.Cur_Loop := Lc'Unrestricted_Access; + + Iter_Nbr := 0; + + Loop_Control_Init (C, Stmt); + + loop + if Cond /= Null_Node then + Val := Synth_Expression_With_Type (C.Inst, Cond, Boolean_Type); + if not Is_Static (Val.Val) then + Error_Msg_Synth (+Cond, "loop condition must be static"); + exit; + end if; + exit when Read_Discrete (Val) = 0; + end if; + + Synth_Sequential_Statements (C, Stmts); + + Loop_Control_Update (C); + + -- Exit from the loop if W_Exit/W_Ret/W_Quit = 0 + exit when Lc.W_Exit /= No_Wire_Id and then Is_Static_Bit0 (Lc.W_Exit); + exit when C.W_Ret /= No_Wire_Id and then Is_Static_Bit0 (C.W_Ret); + exit when Lc.W_Quit /= No_Wire_Id and then Is_Static_Bit0 (Lc.W_Quit); + + Iter_Nbr := Iter_Nbr + 1; + if Iter_Nbr > Flags.Flag_Max_Loop and Flags.Flag_Max_Loop /= 0 then + Error_Msg_Synth + (+Stmt, "maximum number of iterations (%v) reached", + +Uns32 (Flags.Flag_Max_Loop)); + exit; + end if; + end loop; + Loop_Control_Finish (C); + + C.Cur_Loop := Lc.Prev_Loop; + end Synth_Dynamic_While_Loop_Statement; + + procedure Synth_Static_While_Loop_Statement + (C : in out Seq_Context; Stmt : Node) + is + Stmts : constant Node := Get_Sequential_Statement_Chain (Stmt); + Cond : constant Node := Get_Condition (Stmt); + Val : Valtyp; + Lc : aliased Loop_Context (Mode_Static); + begin + Lc := (Mode => Mode_Static, + Prev_Loop => C.Cur_Loop, + Loop_Stmt => Stmt, + S_Exit => False, + S_Quit => False); + C.Cur_Loop := Lc'Unrestricted_Access; + + loop + if Cond /= Null_Node then + Val := Synth_Expression_With_Type (C.Inst, Cond, Boolean_Type); + pragma Assert (Is_Static (Val.Val)); + exit when Read_Discrete (Val) = 0; + end if; + + Synth_Sequential_Statements (C, Stmts); + C.S_En := True; + + -- Exit from the loop if S_Exit/S_Quit + exit when Lc.S_Exit or Lc.S_Quit or C.Nbr_Ret > 0; + end loop; + + C.Cur_Loop := Lc.Prev_Loop; + end Synth_Static_While_Loop_Statement; + + procedure Synth_Return_Statement (C : in out Seq_Context; Stmt : Node) + is + Is_Dyn : constant Boolean := not Get_Instance_Const (C.Inst); + Ctxt : constant Context_Acc := Get_Build (C.Inst); + Val : Valtyp; + Expr : constant Node := Get_Expression (Stmt); + begin + if Expr /= Null_Node then + -- Return in function. + Val := Synth_Expression_With_Type (C.Inst, Expr, C.Ret_Typ); + if Val = No_Valtyp then + Set_Error (C.Inst); + return; + end if; + + Val := Synth_Subtype_Conversion (Ctxt, Val, C.Ret_Typ, True, Stmt); + + if C.Nbr_Ret = 0 then + C.Ret_Value := Val; + if not Is_Bounded_Type (C.Ret_Typ) then + -- The function was declared with an unconstrained return type. + -- Now that a value has been returned, we know the subtype of + -- the returned values. So adjust it. + -- All the returned values must have the same length. + C.Ret_Typ := Val.Typ; + if Is_Dyn then + Set_Width (Get_Wire_Gate (C.W_Val), C.Ret_Typ.W); + Set_Width (C.Ret_Init, C.Ret_Typ.W); + end if; + end if; + end if; + if Is_Dyn then + Phi_Assign_Net (Ctxt, C.W_Val, Get_Net (Ctxt, Val), 0); + end if; + end if; + + if Is_Dyn then + -- The subprogram has returned. Do not execute further statements. + Phi_Assign_Static (C.W_En, Bit0); + + if C.W_Ret /= No_Wire_Id then + Phi_Assign_Static (C.W_Ret, Bit0); + end if; + end if; + + C.Nbr_Ret := C.Nbr_Ret + 1; + end Synth_Return_Statement; + + procedure Synth_Static_Report (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + use Simple_IO; + + Is_Report : constant Boolean := + Get_Kind (Stmt) = Iir_Kind_Report_Statement; + Rep_Expr : constant Node := Get_Report_Expression (Stmt); + Sev_Expr : constant Node := Get_Severity_Expression (Stmt); + Rep : Valtyp; + Sev : Valtyp; + Sev_V : Natural; + begin + if Rep_Expr /= Null_Node then + Rep := Synth_Expression_With_Basetype (Syn_Inst, Rep_Expr); + if Rep = No_Valtyp then + Set_Error (Syn_Inst); + return; + end if; + Strip_Const (Rep); + end if; + if Sev_Expr /= Null_Node then + Sev := Synth_Expression (Syn_Inst, Sev_Expr); + if Sev = No_Valtyp then + Set_Error (Syn_Inst); + return; + end if; + Strip_Const (Sev); + end if; + + Put_Err (Disp_Location (Stmt)); + Put_Err (":("); + if Is_Report then + Put_Err ("report"); + else + Put_Err ("assertion"); + end if; + Put_Err (' '); + if Sev = No_Valtyp then + if Is_Report then + Sev_V := 0; + else + Sev_V := 2; + end if; + else + Sev_V := Natural (Read_Discrete (Sev)); + end if; + case Sev_V is + when Note_Severity => + Put_Err ("note"); + when Warning_Severity => + Put_Err ("warning"); + when Error_Severity => + Put_Err ("error"); + when Failure_Severity => + Put_Err ("failure"); + when others => + Put_Err ("??"); + end case; + Put_Err ("): "); + + if Rep = No_Valtyp then + Put_Line_Err ("assertion failure"); + else + Put_Line_Err (Value_To_String (Rep)); + end if; + + if Sev_V >= Flags.Severity_Level then + Error_Msg_Synth (+Stmt, "error due to assertion failure"); + end if; + end Synth_Static_Report; + + procedure Synth_Static_Report_Statement (C : Seq_Context; Stmt : Node) is + begin + Synth_Static_Report (C.Inst, Stmt); + end Synth_Static_Report_Statement; + + procedure Synth_Static_Assertion_Statement (C : Seq_Context; Stmt : Node) + is + Cond : Valtyp; + begin + Cond := Synth_Expression (C.Inst, Get_Assertion_Condition (Stmt)); + if Cond = No_Valtyp then + Set_Error (C.Inst); + return; + end if; + pragma Assert (Is_Static (Cond.Val)); + Strip_Const (Cond); + if Read_Discrete (Cond) = 1 then + return; + end if; + Synth_Static_Report (C.Inst, Stmt); + end Synth_Static_Assertion_Statement; + + procedure Synth_Dynamic_Assertion_Statement (C : Seq_Context; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (C.Inst); + Loc : constant Location_Type := Get_Location (Stmt); + Cond : Valtyp; + N : Net; + En : Net; + Inst : Instance; + begin + if not Flags.Flag_Formal then + return; + end if; + + Cond := Synth_Expression (C.Inst, Get_Assertion_Condition (Stmt)); + if Cond = No_Valtyp then + Set_Error (C.Inst); + return; + end if; + N := Get_Net (Ctxt, Cond); + En := Phi_Enable (Ctxt, (Stmt, Bit_Type), Bit0, Bit1, + Get_Location (Stmt)); + if En /= No_Net then + -- Build: En -> Cond + N := Build2_Imp (Ctxt, En, N, Loc); + end if; + Inst := Build_Assert (Ctxt, Synth_Label (C.Inst, Stmt), N); + Set_Location (Inst, Loc); + end Synth_Dynamic_Assertion_Statement; + + procedure Synth_Sequential_Statements + (C : in out Seq_Context; Stmts : Node) + is + Is_Dyn : constant Boolean := not Get_Instance_Const (C.Inst); + Ctxt : constant Context_Acc := Get_Build (C.Inst); + Stmt : Node; + Phi_T, Phi_F : Phi_Type; + Has_Phi : Boolean; + begin + Stmt := Stmts; + while Is_Valid (Stmt) loop + if Is_Dyn then + pragma Assert (not Is_Static_Bit0 (C.W_En)); + Has_Phi := not Is_Static_Bit1 (C.W_En); + if Has_Phi then + Push_Phi; + end if; + end if; + + if Flags.Flag_Trace_Statements then + declare + Name : Name_Id; + Line : Natural; + Col : Natural; + begin + Files_Map.Location_To_Position + (Get_Location (Stmt), Name, Line, Col); + Simple_IO.Put_Line ("Execute statement at " + & Name_Table.Image (Name) + & Natural'Image (Line)); + end; + end if; + if Synth.Debugger.Flag_Need_Debug then + Synth.Debugger.Debug_Break (C.Inst, Stmt); + end if; + + case Get_Kind (Stmt) is + when Iir_Kind_If_Statement => + Synth_If_Statement (C, Stmt); + when Iir_Kind_Simple_Signal_Assignment_Statement => + Synth_Simple_Signal_Assignment (C.Inst, Stmt); + when Iir_Kind_Conditional_Signal_Assignment_Statement => + Synth_Conditional_Signal_Assignment (C.Inst, Stmt); + when Iir_Kind_Variable_Assignment_Statement => + Synth_Variable_Assignment (C, Stmt); + when Iir_Kind_Conditional_Variable_Assignment_Statement => + Synth_Conditional_Variable_Assignment (C, Stmt); + when Iir_Kind_Case_Statement => + Synth_Case_Statement (C, Stmt); + when Iir_Kind_For_Loop_Statement => + if Is_Dyn then + Synth_Dynamic_For_Loop_Statement (C, Stmt); + else + Synth_Static_For_Loop_Statement (C, Stmt); + end if; + when Iir_Kind_While_Loop_Statement => + if Is_Dyn then + Synth_Dynamic_While_Loop_Statement (C, Stmt); + else + Synth_Static_While_Loop_Statement (C, Stmt); + end if; + when Iir_Kind_Null_Statement => + -- Easy + null; + when Iir_Kind_Return_Statement => + Synth_Return_Statement (C, Stmt); + when Iir_Kind_Procedure_Call_Statement => + Synth_Procedure_Call (C.Inst, Stmt); + when Iir_Kind_Report_Statement => + if not Is_Dyn then + Synth_Static_Report_Statement (C, Stmt); + end if; + when Iir_Kind_Assertion_Statement => + if not Is_Dyn then + Synth_Static_Assertion_Statement (C, Stmt); + else + Synth_Dynamic_Assertion_Statement (C, Stmt); + end if; + when Iir_Kind_Exit_Statement + | Iir_Kind_Next_Statement => + if Is_Dyn then + Synth_Dynamic_Exit_Next_Statement (C, Stmt); + else + Synth_Static_Exit_Next_Statement (C, Stmt); + end if; + when others => + Error_Kind ("synth_sequential_statements", Stmt); + end case; + if Is_Dyn then + if Has_Phi then + Pop_Phi (Phi_T); + Push_Phi; + Pop_Phi (Phi_F); + Merge_Phis (Ctxt, Get_Current_Value (Ctxt, C.W_En), + Phi_T, Phi_F, Get_Location (Stmt)); + end if; + if Is_Static_Bit0 (C.W_En) then + -- Not more execution. + return; + end if; + else + if not C.S_En or C.Nbr_Ret /= 0 then + return; + end if; + end if; + Stmt := Get_Chain (Stmt); + end loop; + end Synth_Sequential_Statements; + + Proc_Pool : aliased Areapools.Areapool; + + -- Synthesis of statements of a non-sensitized process. + procedure Synth_Process_Sequential_Statements + (C : in out Seq_Context; Proc : Node) + is + Ctxt : constant Context_Acc := Get_Build (C.Inst); + Stmt : Node; + Cond : Node; + Cond_Val : Valtyp; + Phi_True : Phi_Type; + Phi_False : Phi_Type; + begin + Stmt := Get_Sequential_Statement_Chain (Proc); + + -- The first statement must be a wait statement. + if Get_Kind (Stmt) /= Iir_Kind_Wait_Statement then + Error_Msg_Synth (+Stmt, "expect wait as the first statement"); + return; + end if; + + -- Handle the condition as an if. + Cond := Get_Condition_Clause (Stmt); + if Cond = Null_Node then + Error_Msg_Synth (+Stmt, "expect wait condition"); + return; + end if; + Cond_Val := Synth_Expression (C.Inst, Cond); + + Push_Phi; + Synth_Sequential_Statements (C, Get_Chain (Stmt)); + Pop_Phi (Phi_True); + Push_Phi; + Pop_Phi (Phi_False); + + Merge_Phis (Ctxt, Get_Net (Ctxt, Cond_Val), Phi_True, Phi_False, + Get_Location (Stmt)); + end Synth_Process_Sequential_Statements; + + procedure Synth_Process_Statement + (Syn_Inst : Synth_Instance_Acc; Proc : Node) + is + use Areapools; + Label : constant Name_Id := Get_Identifier (Proc); + Decls_Chain : constant Node := Get_Declaration_Chain (Proc); + Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + M : Areapools.Mark_Type; + C_Sname : Sname; + C : Seq_Context (Mode_Dynamic); + begin + if Label = Null_Identifier then + C_Sname := New_Internal_Name (Ctxt, Get_Sname (Syn_Inst)); + else + C_Sname := New_Sname_User (Label, Get_Sname (Syn_Inst)); + end if; + C := (Mode => Mode_Dynamic, + Inst => Make_Instance (Syn_Inst, Proc, C_Sname), + Cur_Loop => null, + W_En => Alloc_Wire (Wire_Variable, (Proc, Bit_Type)), + W_Ret => No_Wire_Id, + W_Val => No_Wire_Id, + Ret_Init => No_Net, + Ret_Value => No_Valtyp, + Ret_Typ => null, + Nbr_Ret => 0); + + Mark (M, Proc_Pool); + Instance_Pool := Proc_Pool'Access; + + Push_Phi; + + Synth_Declarations (C.Inst, Decls_Chain); + + Set_Wire_Gate (C.W_En, Build_Control_Signal (Syn_Inst, 1, Proc)); + Phi_Assign_Static (C.W_En, Bit1); + + if not Is_Error (C.Inst) then + case Iir_Kinds_Process_Statement (Get_Kind (Proc)) is + when Iir_Kind_Sensitized_Process_Statement => + Synth_Sequential_Statements + (C, Get_Sequential_Statement_Chain (Proc)); + -- FIXME: check sensitivity list. + when Iir_Kind_Process_Statement => + Synth_Process_Sequential_Statements (C, Proc); + end case; + end if; + + Pop_And_Merge_Phi (Ctxt, Get_Location (Proc)); + + Finalize_Declarations (C.Inst, Decls_Chain); + + Free_Instance (C.Inst); + Release (M, Proc_Pool); + Instance_Pool := Prev_Instance_Pool; + + Finalize_Assignment (Ctxt, C.W_En); + Free_Wire (C.W_En); + end Synth_Process_Statement; + + function Synth_User_Function_Call + (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp is + begin + -- Is it a call to an ieee function ? + declare + Imp : constant Node := Get_Implementation (Expr); + Pkg : constant Node := Get_Parent (Imp); + Unit : Node; + Lib : Node; + begin + if Get_Kind (Pkg) = Iir_Kind_Package_Declaration + and then not Is_Uninstantiated_Package (Pkg) + then + Unit := Get_Parent (Pkg); + if Get_Kind (Unit) = Iir_Kind_Design_Unit then + Lib := Get_Library (Get_Design_File (Unit)); + if Get_Identifier (Lib) = Std_Names.Name_Ieee then + Error_Msg_Synth + (+Expr, "unhandled call to ieee function %i", +Imp); + Set_Error (Syn_Inst); + return No_Valtyp; + end if; + end if; + end if; + end; + + return Synth_Subprogram_Call (Syn_Inst, Expr); + end Synth_User_Function_Call; + + procedure Synth_Concurrent_Assertion_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Cond : constant Node := Get_Assertion_Condition (Stmt); + Val : Valtyp; + Inst : Instance; + begin + Val := Synth_Expression (Syn_Inst, Cond); + if Val = No_Valtyp then + Set_Error (Syn_Inst); + return; + end if; + if Is_Static (Val.Val) then + if Read_Discrete (Val) /= 1 then + Synth_Static_Report (Syn_Inst, Stmt); + end if; + return; + end if; + + if not Flags.Flag_Formal then + -- Ignore the net. + return; + end if; + + Inst := Build_Assert + (Ctxt, Synth_Label (Syn_Inst, Stmt), Get_Net (Ctxt, Val)); + Set_Location (Inst, Get_Location (Stmt)); + end Synth_Concurrent_Assertion_Statement; + + procedure Synth_Block_Statement (Syn_Inst : Synth_Instance_Acc; Blk : Node) + is + use Areapools; + Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; + Blk_Inst : Synth_Instance_Acc; + Blk_Sname : Sname; + M : Areapools.Mark_Type; + begin + -- No support for guard or header. + if Get_Block_Header (Blk) /= Null_Node + or else Get_Guard_Decl (Blk) /= Null_Node + then + raise Internal_Error; + end if; + + Apply_Block_Configuration + (Get_Block_Block_Configuration (Blk), Blk); + + Blk_Sname := New_Sname_User (Get_Identifier (Blk), Get_Sname (Syn_Inst)); + Blk_Inst := Make_Instance (Syn_Inst, Blk, Blk_Sname); + Mark (M, Proc_Pool); + Instance_Pool := Proc_Pool'Access; + + Synth_Declarations (Blk_Inst, Get_Declaration_Chain (Blk)); + Synth_Concurrent_Statements + (Blk_Inst, Get_Concurrent_Statement_Chain (Blk)); + + Synth_Attribute_Values (Blk_Inst, Blk); + + Finalize_Declarations (Blk_Inst, Get_Declaration_Chain (Blk)); + + Free_Instance (Blk_Inst); + Release (M, Proc_Pool); + Instance_Pool := Prev_Instance_Pool; + end Synth_Block_Statement; + + function Synth_Psl_NFA (Syn_Inst : Synth_Instance_Acc; + NFA : PSL.Types.PSL_NFA; + Nbr_States : Int32; + States : Net; + Loc : Source.Syn_Src) return Net + is + use PSL.NFAs; + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + S : NFA_State; + S_Num : Int32; + D_Num : Int32; + I : Net; + Cond : Net; + E : NFA_Edge; + D_Arr : Net_Array_Acc; + Res : Net; + begin + D_Arr := new Net_Array'(0 .. Nbr_States - 1 => No_Net); + + -- For each state: + S := Get_First_State (NFA); + while S /= No_State loop + S_Num := Get_State_Label (S); + I := Build_Extract_Bit (Ctxt, States, Uns32 (S_Num)); + Set_Location (I, Loc); + + -- For each edge: + E := Get_First_Src_Edge (S); + while E /= No_Edge loop + -- Edge condition. + Cond := Build_Dyadic + (Ctxt, Id_And, + I, Synth_PSL_Expression (Syn_Inst, Get_Edge_Expr (E))); + Set_Location (Cond, Loc); + + -- TODO: if EOS is present, then this is a live state. + + -- Reverse order for final concatenation. + D_Num := Nbr_States - 1 - Get_State_Label (Get_Edge_Dest (E)); + if D_Arr (D_Num) /= No_Net then + Cond := Build_Dyadic (Ctxt, Id_Or, D_Arr (D_Num), Cond); + Set_Location (Cond, Loc); + end if; + D_Arr (D_Num) := Cond; + + E := Get_Next_Src_Edge (E); + end loop; + + S := Get_Next_State (S); + end loop; + + if D_Arr (Nbr_States - 1) = No_Net then + D_Arr (Nbr_States - 1) := Build_Const_UB32 (Ctxt, 0, 1); + end if; + + Concat_Array (Ctxt, D_Arr.all, Res); + Free_Net_Array (D_Arr); + + return Res; + end Synth_Psl_NFA; + + procedure Synth_Psl_Dff (Syn_Inst : Synth_Instance_Acc; + Stmt : Node; + Next_States : out Net) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Nbr_States : constant Int32 := Get_PSL_Nbr_States (Stmt); + States : Net; + Init : Net; + Clk : Net; + Clk_Inst : Instance; + begin + -- create init net, clock net + Init := Build_Const_UB32 (Ctxt, 1, Uns32 (Nbr_States)); + Set_Location (Init, Stmt); + Clk := Synth_PSL_Expression (Syn_Inst, Get_PSL_Clock (Stmt)); + + -- Check the clock is an edge and extract it. + Clk_Inst := Get_Net_Parent (Clk); + if Get_Id (Clk_Inst) not in Edge_Module_Id then + Error_Msg_Synth (+Stmt, "clock is not an edge"); + Next_States := No_Net; + return; + end if; + + -- build idff + States := Build_Idff (Ctxt, Clk, No_Net, Init); + Set_Location (States, Stmt); + + -- create update nets + -- For each state: if set, evaluate all outgoing edges. + Next_States := + Synth_Psl_NFA (Syn_Inst, Get_PSL_NFA (Stmt), Nbr_States, States, Stmt); + Connect (Get_Input (Get_Net_Parent (States), 1), Next_States); + end Synth_Psl_Dff; + + function Synth_Psl_Final + (Syn_Inst : Synth_Instance_Acc; Stmt : Node; Next_States : Net) return Net + is + use PSL.Types; + use PSL.NFAs; + NFA : constant PSL_NFA := Get_PSL_NFA (Stmt); + Res : Net; + begin + Res := Build_Extract_Bit + (Get_Build (Syn_Inst), Next_States, + Uns32 (Get_State_Label (Get_Final_State (NFA)))); + Set_Location (Res, Stmt); + return Res; + end Synth_Psl_Final; + + function Synth_Psl_Not_Final + (Syn_Inst : Synth_Instance_Acc; Stmt : Node; Next_States : Net) + return Net + is + Res : Net; + begin + Res := Build_Monadic (Get_Build (Syn_Inst), Id_Not, + Synth_Psl_Final (Syn_Inst, Stmt, Next_States)); + Set_Location (Res, Stmt); + return Res; + end Synth_Psl_Not_Final; + + procedure Synth_Psl_Restrict_Directive + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Next_States : Net; + Res : Net; + Inst : Instance; + begin + if not Flags.Flag_Formal then + return; + end if; + + -- Build assume gate. + -- Note: for synthesis, we assume the next state will be correct. + -- (If we assume on States, then the first cycle is ignored). + Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); + if Next_States /= No_Net then + -- The restriction holds as long as there is a 1 in the NFA state. + Res := Build_Reduce (Ctxt, Id_Red_Or, Next_States); + Set_Location (Res, Stmt); + Inst := Build_Assume (Ctxt, Synth_Label (Syn_Inst, Stmt), Res); + Set_Location (Inst, Get_Location (Stmt)); + end if; + end Synth_Psl_Restrict_Directive; + + procedure Synth_Psl_Cover_Directive + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Next_States : Net; + Res : Net; + Inst : Instance; + begin + if not Flags.Flag_Formal then + return; + end if; + + -- Build cover gate. + -- Note: for synthesis, we assume the next state will be correct. + -- (If we assume on States, then the first cycle is ignored). + Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); + if Next_States /= No_Net then + -- The sequence is covered as soon as the final state is reached. + Res := Synth_Psl_Final (Syn_Inst, Stmt, Next_States); + Inst := Build_Cover + (Get_Build (Syn_Inst), Synth_Label (Syn_Inst, Stmt), Res); + Set_Location (Inst, Get_Location (Stmt)); + end if; + end Synth_Psl_Cover_Directive; + + procedure Synth_Psl_Assume_Directive + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + Next_States : Net; + Inst : Instance; + begin + if not Flags.Flag_Formal then + return; + end if; + + -- Build assume gate. + -- Note: for synthesis, we assume the next state will be correct. + -- (If we assume on States, then the first cycle is ignored). + Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); + if Next_States /= No_Net then + Inst := Build_Assume + (Ctxt, Synth_Label (Syn_Inst, Stmt), + Synth_Psl_Not_Final (Syn_Inst, Stmt, Next_States)); + Set_Location (Inst, Get_Location (Stmt)); + end if; + end Synth_Psl_Assume_Directive; + + procedure Synth_Psl_Assert_Directive + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + use PSL.Types; + use PSL.NFAs; + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + NFA : constant PSL_NFA := Get_PSL_NFA (Stmt); + Active : NFA_State; + Next_States : Net; + Inst : Instance; + Lab : Sname; + begin + if not Flags.Flag_Formal then + return; + end if; + + -- Build assert gate. + -- Note: for synthesis, we assume the next state will be correct. + -- (If we assert on States, then the first cycle is ignored). + Synth_Psl_Dff (Syn_Inst, Stmt, Next_States); + if Next_States = No_Net then + return; + end if; + Lab := Synth_Label (Syn_Inst, Stmt); + + Inst := Build_Assert + (Ctxt, Lab, Synth_Psl_Not_Final (Syn_Inst, Stmt, Next_States)); + Set_Location (Inst, Get_Location (Stmt)); + + -- Also add a cover gate to cover assertion activation. + if Flags.Flag_Assert_Cover then + Active := Get_Active_State (NFA); + if Active /= No_State then + if Lab /= No_Sname then + Lab := New_Sname_User (Std_Names.Name_Cover, Lab); + end if; + Inst := Build_Assert_Cover + (Get_Build (Syn_Inst), Lab, + Build_Extract_Bit (Get_Build (Syn_Inst), Next_States, + Uns32 (Get_State_Label (Active)))); + Set_Location (Inst, Get_Location (Stmt)); + end if; + end if; + end Synth_Psl_Assert_Directive; + + procedure Synth_Generate_Statement_Body + (Syn_Inst : Synth_Instance_Acc; + Bod : Node; + Name : Sname; + Iterator : Node := Null_Node; + Iterator_Val : Valtyp := No_Valtyp) + is + use Areapools; + Decls_Chain : constant Node := Get_Declaration_Chain (Bod); + Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; + Bod_Inst : Synth_Instance_Acc; + M : Areapools.Mark_Type; + begin + Bod_Inst := Make_Instance (Syn_Inst, Bod, Name); + Mark (M, Proc_Pool); + Instance_Pool := Proc_Pool'Access; + + if Iterator /= Null_Node then + -- Add the iterator (for for-generate). + Create_Object (Bod_Inst, Iterator, Iterator_Val); + end if; + + Synth_Declarations (Bod_Inst, Decls_Chain); + + Synth_Concurrent_Statements + (Bod_Inst, Get_Concurrent_Statement_Chain (Bod)); + + Synth_Attribute_Values (Bod_Inst, Bod); + + Finalize_Declarations (Bod_Inst, Decls_Chain); + + Free_Instance (Bod_Inst); + Release (M, Proc_Pool); + Instance_Pool := Prev_Instance_Pool; + end Synth_Generate_Statement_Body; + + procedure Synth_If_Generate_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Gen : Node; + Bod : Node; + Icond : Node; + Cond : Valtyp; + Name : Sname; + begin + Gen := Stmt; + Name := New_Sname_User (Get_Identifier (Stmt), Get_Sname (Syn_Inst)); + loop + Icond := Get_Condition (Gen); + if Icond /= Null_Node then + Cond := Synth_Expression (Syn_Inst, Icond); + Strip_Const (Cond); + else + -- It is the else generate. + Cond := No_Valtyp; + end if; + if Cond = No_Valtyp or else Read_Discrete (Cond) = 1 then + Bod := Get_Generate_Statement_Body (Gen); + Apply_Block_Configuration + (Get_Generate_Block_Configuration (Bod), Bod); + Synth_Generate_Statement_Body (Syn_Inst, Bod, Name); + exit; + end if; + Gen := Get_Generate_Else_Clause (Gen); + exit when Gen = Null_Node; + end loop; + end Synth_If_Generate_Statement; + + procedure Synth_For_Generate_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Iterator : constant Node := Get_Parameter_Specification (Stmt); + Bod : constant Node := Get_Generate_Statement_Body (Stmt); + Configs : constant Node := Get_Generate_Block_Configuration (Bod); + It_Type : constant Node := Get_Declaration_Type (Iterator); + Config : Node; + It_Rng : Type_Acc; + Val : Valtyp; + Name : Sname; + Lname : Sname; + begin + if It_Type /= Null_Node then + Synth_Subtype_Indication (Syn_Inst, It_Type); + end if; + + -- Initial value. + It_Rng := Get_Subtype_Object (Syn_Inst, Get_Type (Iterator)); + Val := Create_Value_Discrete (It_Rng.Drange.Left, It_Rng); + + Name := New_Sname_User (Get_Identifier (Stmt), Get_Sname (Syn_Inst)); + + while In_Range (It_Rng.Drange, Read_Discrete (Val)) loop + -- Find and apply the config block. + declare + Spec : Node; + begin + Config := Configs; + while Config /= Null_Node loop + Spec := Get_Block_Specification (Config); + case Get_Kind (Spec) is + when Iir_Kind_Simple_Name => + exit; + when others => + Error_Kind ("synth_for_generate_statement", Spec); + end case; + Config := Get_Prev_Block_Configuration (Config); + end loop; + if Config = Null_Node then + raise Internal_Error; + end if; + Apply_Block_Configuration (Config, Bod); + end; + + -- FIXME: get position ? + Lname := New_Sname_Version (Uns32 (Read_Discrete (Val)), Name); + + Synth_Generate_Statement_Body (Syn_Inst, Bod, Lname, Iterator, Val); + Update_Index (It_Rng.Drange, Val); + end loop; + end Synth_For_Generate_Statement; + + procedure Synth_Concurrent_Statement + (Syn_Inst : Synth_Instance_Acc; Stmt : Node) + is + Ctxt : constant Context_Acc := Get_Build (Syn_Inst); + begin + case Get_Kind (Stmt) is + when Iir_Kind_Concurrent_Simple_Signal_Assignment => + Push_Phi; + Synth_Simple_Signal_Assignment (Syn_Inst, Stmt); + Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); + when Iir_Kind_Concurrent_Conditional_Signal_Assignment => + Push_Phi; + Synth_Conditional_Signal_Assignment (Syn_Inst, Stmt); + Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); + when Iir_Kind_Concurrent_Selected_Signal_Assignment => + Push_Phi; + Synth_Selected_Signal_Assignment (Syn_Inst, Stmt); + Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); + when Iir_Kind_Concurrent_Procedure_Call_Statement => + Push_Phi; + Synth_Procedure_Call (Syn_Inst, Stmt); + Pop_And_Merge_Phi (Ctxt, Get_Location (Stmt)); + when Iir_Kinds_Process_Statement => + Synth_Process_Statement (Syn_Inst, Stmt); + when Iir_Kind_If_Generate_Statement => + Synth_If_Generate_Statement (Syn_Inst, Stmt); + when Iir_Kind_For_Generate_Statement => + Synth_For_Generate_Statement (Syn_Inst, Stmt); + when Iir_Kind_Component_Instantiation_Statement => + if Is_Component_Instantiation (Stmt) then + declare + Comp_Config : constant Node := + Get_Component_Configuration (Stmt); + begin + if Get_Binding_Indication (Comp_Config) = Null_Node then + -- Not bound. + Synth_Blackbox_Instantiation_Statement (Syn_Inst, Stmt); + else + Synth_Component_Instantiation_Statement (Syn_Inst, Stmt); + end if; + end; + -- Un-apply configuration. + Set_Component_Configuration (Stmt, Null_Node); + else + Synth_Design_Instantiation_Statement (Syn_Inst, Stmt); + end if; + when Iir_Kind_Block_Statement => + Synth_Block_Statement (Syn_Inst, Stmt); + when Iir_Kind_Psl_Default_Clock => + null; + when Iir_Kind_Psl_Restrict_Directive => + Synth_Psl_Restrict_Directive (Syn_Inst, Stmt); + when Iir_Kind_Psl_Assume_Directive => + if Flags.Flag_Assume_As_Assert then + Synth_Psl_Assert_Directive (Syn_Inst, Stmt); + else + Synth_Psl_Assume_Directive (Syn_Inst, Stmt); + end if; + when Iir_Kind_Psl_Cover_Directive => + Synth_Psl_Cover_Directive (Syn_Inst, Stmt); + when Iir_Kind_Psl_Assert_Directive => + if Flags.Flag_Assert_As_Assume then + Synth_Psl_Assume_Directive (Syn_Inst, Stmt); + else + Synth_Psl_Assert_Directive (Syn_Inst, Stmt); + end if; + when Iir_Kind_Concurrent_Assertion_Statement => + -- Passive statement. + Synth_Concurrent_Assertion_Statement (Syn_Inst, Stmt); + when others => + Error_Kind ("synth_concurrent_statement", Stmt); + end case; + end Synth_Concurrent_Statement; + + procedure Synth_Concurrent_Statements + (Syn_Inst : Synth_Instance_Acc; Stmts : Node) + is + Stmt : Node; + begin + Stmt := Stmts; + while Is_Valid (Stmt) loop + Synth_Concurrent_Statement (Syn_Inst, Stmt); + Stmt := Get_Chain (Stmt); + end loop; + end Synth_Concurrent_Statements; + + -- For allconst/allseq/... + procedure Synth_Attribute_Formal (Syn_Inst : Synth_Instance_Acc; + Val : Node; + Id : Formal_Module_Id) + is + Spec : constant Node := Get_Attribute_Specification (Val); + Sig : constant Node := Get_Designated_Entity (Val); + V : Valtyp; + begin + -- The type must be boolean + if (Get_Base_Type (Get_Type (Val)) /= + Vhdl.Std_Package.Boolean_Type_Definition) + then + Error_Msg_Synth (+Val, "type of attribute %i must be boolean", + (1 => +Get_Attribute_Designator (Spec))); + return; + end if; + + -- The designated entity must be a signal. + if Get_Kind (Sig) /= Iir_Kind_Signal_Declaration then + Error_Msg_Synth (+Val, "attribute %i only applies to signals", + (1 => +Get_Attribute_Designator (Spec))); + return; + end if; + + -- The value must be true + V := Synth_Expression_With_Type + (Syn_Inst, Get_Expression (Spec), Boolean_Type); + if Read_Discrete (V) /= 1 then + return; + end if; + + declare + Off : Value_Offsets; + Dyn : Dyn_Name; + N : Net; + Base : Valtyp; + Typ : Type_Acc; + begin + Synth_Assignment_Prefix (Syn_Inst, Sig, Base, Typ, Off, Dyn); + pragma Assert (Off = (0, 0)); + pragma Assert (Dyn.Voff = No_Net); + pragma Assert (Base.Val.Kind = Value_Wire); + pragma Assert (Base.Typ = Typ); + + N := Build_Formal_Input (Get_Build (Syn_Inst), Id, Typ.W); + Set_Location (N, Val); + Add_Conc_Assign (Base.Val.W, N, 0); + end; + end Synth_Attribute_Formal; + + procedure Synth_Attribute_Values + (Syn_Inst : Synth_Instance_Acc; Unit : Node) + is + use Std_Names; + + Val : Node; + Spec : Node; + Id : Name_Id; + begin + Val := Get_Attribute_Value_Chain (Unit); + while Val /= Null_Node loop + Spec := Get_Attribute_Specification (Val); + Id := Get_Identifier (Get_Attribute_Designator (Spec)); + case Id is + when Name_Allconst => + Synth_Attribute_Formal (Syn_Inst, Val, Id_Allconst); + when Name_Allseq => + Synth_Attribute_Formal (Syn_Inst, Val, Id_Allseq); + when Name_Anyconst => + Synth_Attribute_Formal (Syn_Inst, Val, Id_Anyconst); + when Name_Anyseq => + Synth_Attribute_Formal (Syn_Inst, Val, Id_Anyseq); + when Name_Loc => + -- Applies to nets/ports. + null; + when others => + Warning_Msg_Synth (+Spec, "unhandled attribute %i", (1 => +Id)); + end case; + Val := Get_Value_Chain (Val); + end loop; + end Synth_Attribute_Values; + + procedure Synth_Verification_Unit + (Syn_Inst : Synth_Instance_Acc; Unit : Node) + is + use Areapools; + Prev_Instance_Pool : constant Areapool_Acc := Instance_Pool; + Unit_Inst : Synth_Instance_Acc; + Unit_Sname : Sname; + M : Areapools.Mark_Type; + Item : Node; + Last_Type : Node; + begin + Unit_Sname := New_Sname_User (Get_Identifier (Unit), + Get_Sname (Syn_Inst)); + Unit_Inst := Make_Instance (Syn_Inst, Unit, Unit_Sname); + Mark (M, Proc_Pool); + Instance_Pool := Proc_Pool'Access; + + Apply_Block_Configuration + (Get_Verification_Block_Configuration (Unit), Unit); + + Last_Type := Null_Node; + Item := Get_Vunit_Item_Chain (Unit); + while Item /= Null_Node loop + case Get_Kind (Item) is + when Iir_Kind_Psl_Default_Clock => + null; + when Iir_Kind_Psl_Assert_Directive => + Synth_Psl_Assert_Directive (Unit_Inst, Item); + when Iir_Kind_Psl_Assume_Directive => + Synth_Psl_Assume_Directive (Unit_Inst, Item); + when Iir_Kind_Psl_Restrict_Directive => + Synth_Psl_Restrict_Directive (Unit_Inst, Item); + when Iir_Kind_Psl_Cover_Directive => + Synth_Psl_Cover_Directive (Unit_Inst, Item); + when Iir_Kind_Signal_Declaration + | Iir_Kind_Function_Declaration + | Iir_Kind_Procedure_Declaration + | Iir_Kind_Function_Body + | Iir_Kind_Procedure_Body + | Iir_Kind_Attribute_Declaration + | Iir_Kind_Attribute_Specification => + Synth_Declaration (Unit_Inst, Item, False, Last_Type); + when Iir_Kinds_Concurrent_Signal_Assignment + | Iir_Kinds_Process_Statement + | Iir_Kinds_Generate_Statement + | Iir_Kind_Block_Statement + | Iir_Kind_Concurrent_Procedure_Call_Statement + | Iir_Kind_Component_Instantiation_Statement => + Synth_Concurrent_Statement (Unit_Inst, Item); + when others => + Error_Kind ("synth_verification_unit", Item); + end case; + Item := Get_Chain (Item); + end loop; + + Synth_Attribute_Values (Unit_Inst, Unit); + + -- Finalize + Item := Get_Vunit_Item_Chain (Unit); + while Item /= Null_Node loop + case Get_Kind (Item) is + when Iir_Kind_Psl_Default_Clock + | Iir_Kind_Psl_Assert_Directive + | Iir_Kind_Psl_Assume_Directive + | Iir_Kind_Psl_Restrict_Directive + | Iir_Kind_Psl_Cover_Directive => + null; + when Iir_Kinds_Concurrent_Signal_Assignment + | Iir_Kinds_Process_Statement + | Iir_Kinds_Generate_Statement + | Iir_Kind_Block_Statement + | Iir_Kind_Concurrent_Procedure_Call_Statement + | Iir_Kind_Component_Instantiation_Statement => + null; + when Iir_Kind_Signal_Declaration + | Iir_Kind_Function_Declaration + | Iir_Kind_Procedure_Declaration + | Iir_Kind_Function_Body + | Iir_Kind_Procedure_Body + | Iir_Kind_Attribute_Declaration + | Iir_Kind_Attribute_Specification => + Finalize_Declaration (Unit_Inst, Item, False); + when others => + Error_Kind ("synth_verification_unit(2)", Item); + end case; + Item := Get_Chain (Item); + end loop; + + Free_Instance (Unit_Inst); + Release (M, Proc_Pool); + Instance_Pool := Prev_Instance_Pool; + end Synth_Verification_Unit; +end Synth.Vhdl_Stmts; diff --git a/src/synth/synth-vhdl_stmts.ads b/src/synth/synth-vhdl_stmts.ads new file mode 100644 index 000000000..9621a7c9f --- /dev/null +++ b/src/synth/synth-vhdl_stmts.ads @@ -0,0 +1,167 @@ +-- Statements synthesis. +-- Copyright (C) 2017 Tristan Gingold +-- +-- This file is part of GHDL. +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 2 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see . + +with Types; use Types; +with Vhdl.Nodes; use Vhdl.Nodes; + +with Netlists; use Netlists; + +with Synth.Objtypes; use Synth.Objtypes; +with Synth.Values; use Synth.Values; +with Synth.Vhdl_Context; use Synth.Vhdl_Context; +with Synth.Vhdl_Environment; use Synth.Vhdl_Environment.Env; + +package Synth.Vhdl_Stmts is + procedure Synth_Subprogram_Association (Subprg_Inst : Synth_Instance_Acc; + Caller_Inst : Synth_Instance_Acc; + Inter_Chain : Node; + Assoc_Chain : Node); + + -- Dynamic index for Synth_Assignment_Prefix. + -- As dynamic is about dynamic (!) index, the index is a net. + type Dyn_Name is record + -- Start and type of the indexed part, which can be a part of the + -- base name. + Pfx_Off : Value_Offsets; + Pfx_Typ : Type_Acc; + + -- Variable offset. + Voff : Net; + end record; + + No_Dyn_Name : constant Dyn_Name := (Pfx_Off => No_Value_Offsets, + Pfx_Typ => null, + Voff => No_Net); + + -- Transform PFX into DEST_*. + -- DEST_BASE is the base object (with its own typ). Can be the result, + -- a net or an object larger than the result. + -- DEST_TYP is the type of the result. + -- DEST_OFF is the offset, within DEST_DYN. + -- DEST_DYN is set (Voff field set) when there is a non-static index. + procedure Synth_Assignment_Prefix (Syn_Inst : Synth_Instance_Acc; + Pfx : Node; + Dest_Base : out Valtyp; + Dest_Typ : out Type_Acc; + Dest_Off : out Value_Offsets; + Dest_Dyn : out Dyn_Name); + + procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc; + Target : Node; + Val : Valtyp; + Loc : Node); + + function Synth_Read_Memory (Syn_Inst : Synth_Instance_Acc; + Obj : Valtyp; + Res_Typ : Type_Acc; + Off : Uns32; + Dyn : Dyn_Name; + Loc : Node) return Valtyp; + + function Synth_User_Function_Call + (Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp; + + -- Operation implemented by a user function. + function Synth_User_Operator (Syn_Inst : Synth_Instance_Acc; + Left_Expr : Node; + Right_Expr : Node; + Expr : Node) return Valtyp; + + -- Generate netlists for concurrent statements STMTS. + procedure Synth_Concurrent_Statements + (Syn_Inst : Synth_Instance_Acc; Stmts : Node); + + -- Apply attributes of UNIT. + procedure Synth_Attribute_Values + (Syn_Inst : Synth_Instance_Acc; Unit : Node); + + procedure Synth_Verification_Unit + (Syn_Inst : Synth_Instance_Acc; Unit : Node); + + -- For iterators. + procedure Update_Index (Rng : Discrete_Range_Type; V : in out Valtyp); + +private + -- There are 2 execution mode: + -- * static: it is like simulation, all the inputs are known, neither + -- gates nor signals are generated. This mode is used during + -- elaboration and when all inputs of a subprogram are known. + -- * dynamic: inputs can be wires so gates are generated. But many types + -- (like file or access) cannot be handled. + type Mode_Type is (Mode_Static, Mode_Dynamic); + + type Loop_Context (Mode : Mode_Type); + type Loop_Context_Acc is access all Loop_Context; + + type Loop_Context (Mode : Mode_Type) is record + Prev_Loop : Loop_Context_Acc; + Loop_Stmt : Node; + + case Mode is + when Mode_Dynamic => + -- Set when this loop has next/exit statements for itself. + -- Set to true so that inner loops have to declare W_Quit. + Need_Quit : Boolean; + + -- Value of W_En at the entry of the loop. + Saved_En : Net; + + -- Set to 0 in case of exit for the loop. + -- Set to 0 in case of exit/next for outer loop. + -- Initialized to 1. + W_Exit : Wire_Id; + + -- Set to 0 if this loop has to be quited because of an + -- exit/next for an outer loop. Initialized to 1. + W_Quit : Wire_Id; + + -- Mark to release wires. + Wire_Mark : Wire_Id; + when Mode_Static => + S_Exit : Boolean; + S_Quit : Boolean; + end case; + end record; + + -- Context for sequential statements. + type Seq_Context (Mode : Mode_Type) is record + Inst : Synth_Instance_Acc; + + Cur_Loop : Loop_Context_Acc; + + Ret_Value : Valtyp; + Ret_Typ : Type_Acc; + Nbr_Ret : Int32; + + case Mode is + when Mode_Dynamic => + -- Enable execution. For loop controls. + W_En : Wire_Id; + + W_Ret : Wire_Id; + + -- Return value. + W_Val : Wire_Id; + + Ret_Init : Net; + + when Mode_Static => + S_En : Boolean; + end case; + end record; +end Synth.Vhdl_Stmts; diff --git a/src/synth/synthesis.adb b/src/synth/synthesis.adb index 6e3dabfc0..131e6ba04 100644 --- a/src/synth/synthesis.adb +++ b/src/synth/synthesis.adb @@ -20,7 +20,7 @@ with Errorout; use Errorout; with Vhdl.Errors; use Vhdl.Errors; with Synth.Objtypes; -with Synth.Insts; use Synth.Insts; +with Synth.Vhdl_Insts; use Synth.Vhdl_Insts; with Synth.Values.Debug; pragma Unreferenced (Synth.Values.Debug); -- cgit v1.2.3