diff options
author | Tristan Gingold <tgingold@free.fr> | 2022-09-24 06:43:53 +0200 |
---|---|---|
committer | Tristan Gingold <tgingold@free.fr> | 2022-09-25 11:43:25 +0200 |
commit | 9e789b43283c07e112c51cdf399eb8ba47eba5c5 (patch) | |
tree | a4c9ce09b9692c1c44b2fd3128385772d8f3d80f /src | |
parent | af3ca558ccec3cace3277b9d2bb12eeadf39559e (diff) | |
download | ghdl-9e789b43283c07e112c51cdf399eb8ba47eba5c5.tar.gz ghdl-9e789b43283c07e112c51cdf399eb8ba47eba5c5.tar.bz2 ghdl-9e789b43283c07e112c51cdf399eb8ba47eba5c5.zip |
synth-vhdl_stmts: support of individual paramater associations (WIP)
Diffstat (limited to 'src')
-rw-r--r-- | src/synth/elab-vhdl_objtypes.ads | 2 | ||||
-rw-r--r-- | src/synth/synth-vhdl_stmts.adb | 342 |
2 files changed, 238 insertions, 106 deletions
diff --git a/src/synth/elab-vhdl_objtypes.ads b/src/synth/elab-vhdl_objtypes.ads index 3b3547132..08da1c266 100644 --- a/src/synth/elab-vhdl_objtypes.ads +++ b/src/synth/elab-vhdl_objtypes.ads @@ -97,6 +97,8 @@ package Elab.Vhdl_Objtypes is Type_Array .. Type_Unbounded_Array; subtype Type_Vectors is Type_Kind range Type_Vector .. Type_Unbounded_Vector; + subtype Type_Composite is Type_Kind range + Type_Vector .. Type_Record; type Type_Type (Kind : Type_Kind); type Type_Acc is access Type_Type; diff --git a/src/synth/synth-vhdl_stmts.adb b/src/synth/synth-vhdl_stmts.adb index 0389bf3ae..889914943 100644 --- a/src/synth/synth-vhdl_stmts.adb +++ b/src/synth/synth-vhdl_stmts.adb @@ -115,7 +115,7 @@ package body Synth.Vhdl_Stmts is Dest_Dyn := No_Dyn_Name; Dest_Typ := Targ.Typ; - if Targ.Val.Kind = Value_Alias then + if Targ.Val /= null and then 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; @@ -163,7 +163,10 @@ package body Synth.Vhdl_Stmts is Synth_Assignment_Prefix (Syn_Inst, Inter_Inst, Get_Prefix (Pfx), Dest_Base, Dest_Typ, Dest_Off, Dest_Dyn); - Strip_Const (Dest_Base); + if Dest_Base.Val /= null then + -- For individual associations, only the typ can be set. + Strip_Const (Dest_Base); + end if; Synth_Indexed_Name (Syn_Inst, Pfx, Dest_Typ, El_Typ, Voff, Off, Err); @@ -358,6 +361,26 @@ package body Synth.Vhdl_Stmts is return Res; end Synth_Aggregate_Target_Type; + function To_Target_Info (Base : Valtyp; + Typ : Type_Acc; + Off : Value_Offsets; + Dyn : Dyn_Name) return Target_Info is + begin + 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 To_Target_Info; + function Synth_Target (Syn_Inst : Synth_Instance_Acc; Target : Node) return Target_Info is begin @@ -387,19 +410,7 @@ package body Synth.Vhdl_Stmts is begin Synth_Assignment_Prefix (Syn_Inst, 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; + return To_Target_Info (Base, Typ, Off, Dyn); end; when others => Error_Kind ("synth_target", Target); @@ -1823,32 +1834,6 @@ package body Synth.Vhdl_Stmts is end case; end record; - -- Find association for Iterator.Inter - procedure Association_Find_Assoc (Iterator : in out Association_Iterator) - is - Inter : constant Node := Iterator.Inter; - Formal : Node; - begin - -- Search by name. - Iterator.Assoc := Iterator.First_Named_Assoc; - while Iterator.Assoc /= Null_Node loop - Formal := Get_Formal (Iterator.Assoc); - pragma Assert (Formal /= Null_Node); - Formal := Get_Interface_Of_Formal (Formal); - -- Compare by identifier, as INTER can be the generic - -- interface, while FORMAL is the instantiated one. - if Get_Identifier (Formal) = Get_Identifier (Inter) then - -- Found. - -- Optimize in case assocs are in order. - if Iterator.Assoc = Iterator.First_Named_Assoc then - Iterator.First_Named_Assoc := Get_Chain (Iterator.Assoc); - end if; - return; - end if; - Iterator.Assoc := Get_Chain (Iterator.Assoc); - end loop; - end Association_Find_Assoc; - procedure Association_Iterate_Init (Iterator : out Association_Iterator; Init : Association_Iterator_Init) is begin @@ -1857,16 +1842,7 @@ package body Synth.Vhdl_Stmts is Iterator := (Kind => Association_Function, Inter => Init.Inter_Chain, First_Named_Assoc => Null_Node, - Assoc => Null_Node); - if Init.Assoc_Chain /= Null_Node - and then Get_Formal (Init.Assoc_Chain) /= Null_Node - then - -- The first assoc is a named association. - Iterator.First_Named_Assoc := Init.Assoc_Chain; - Association_Find_Assoc (Iterator); - else - Iterator.Assoc := Init.Assoc_Chain; - end if; + Assoc => Init.Assoc_Chain); when Association_Operator => Iterator := (Kind => Association_Operator, Inter => Init.Inter_Chain, @@ -1885,58 +1861,64 @@ package body Synth.Vhdl_Stmts is Inter : out Node; Assoc : out Node) is begin + -- Next interface. Inter := Iterator.Inter; + if Inter = Null_Node then -- End of iterator. Assoc := Null_Node; return; end if; + -- Advance to the next interface for the next call. + Iterator.Inter := Get_Chain (Iterator.Inter); + case Iterator.Kind is when Association_Function => - Assoc := Iterator.Assoc; - - -- Next individual association for the same interface. - if Assoc /= Null_Node then - if Get_Kind (Assoc) = Iir_Kind_Association_Element_By_Individual - then - Iterator.Assoc := Get_Chain (Assoc); - return; - end if; - if not Get_Whole_Association_Flag (Assoc) then - -- Still individual assoc. - Iterator.Assoc := Get_Chain (Assoc); - return; - end if; - end if; - - -- Advance to the next interface for the next call. - Iterator.Inter := Get_Chain (Iterator.Inter); - if Iterator.Inter = Null_Node then - -- Last one. - return; - end if; - if Iterator.First_Named_Assoc = Null_Node then + Assoc := Iterator.Assoc; -- Still using association by position. - if Iterator.Assoc = Null_Node then + if Assoc = Null_Node then -- No more associations, all open. return; end if; - Iterator.Assoc := Get_Chain (Iterator.Assoc); - if Iterator.Assoc = Null_Node - or else Get_Formal (Iterator.Assoc) = Null_Node - then - -- Still by position + if Get_Formal (Assoc) = Null_Node then + -- Still by position, update for the next call. + Iterator.Assoc := Get_Chain (Assoc); return; end if; - Iterator.First_Named_Assoc := Iterator.Assoc; + Iterator.First_Named_Assoc := Assoc; end if; - Association_Find_Assoc (Iterator); + + -- Search by name. + declare + Formal : Node; + begin + 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); + + -- Compare by identifier, as INTER can be the generic + -- interface, while FORMAL is the instantiated one. + if Get_Identifier (Formal) = Get_Identifier (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; + end; + + -- Not found: open association. + return; when Association_Operator => Assoc := Iterator.Op1; - Iterator.Inter := Get_Chain (Iterator.Inter); Iterator.Op1 := Iterator.Op2; Iterator.Op2 := Null_Node; end case; @@ -2165,9 +2147,151 @@ package body Synth.Vhdl_Stmts is when Iir_Kind_Interface_Quantity_Declaration => raise Internal_Error; end case; - end Synth_Subprogram_Association; + function Count_Individual_Associations (Inter : Node; + First_Assoc : Node) return Natural + is + Count : Natural; + Assoc : Node; + Formal : Node; + begin + -- 1. Count number of assocs + Count := 0; + Assoc := Get_Chain (First_Assoc); + Formal := Get_Formal (Assoc); + pragma Assert (Get_Interface_Of_Formal (Formal) = Inter); + loop + Count := Count + 1; + Assoc := Get_Chain (Assoc); + exit when Assoc = Null_Node; + Formal := Get_Formal (Assoc); + exit when Get_Interface_Of_Formal (Formal) /= Inter; + end loop; + return Count; + end Count_Individual_Associations; + + type Assoc_Record is record + Formal : Node; + Form_Off : Value_Offsets; + + Act_Base : Valtyp; + Act_Typ : Type_Acc; + Act_Off : Value_Offsets; + Act_Dyn : Dyn_Name; + end record; + + type Assoc_Array is array (Natural range <>) of Assoc_Record; + type Assoc_Array_Acc is access Assoc_Array; + procedure Free_Assoc_Array is new Ada.Unchecked_Deallocation + (Assoc_Array, Assoc_Array_Acc); + + function Synth_Individual_Association (Subprg_Inst : Synth_Instance_Acc; + Caller_Inst : Synth_Instance_Acc; + Inter : Node; + First_Assoc : Node) return Valtyp + is + Inter_Kind : constant Iir_Kinds_Interface_Object_Declaration := + Get_Kind (Inter); + Count : constant Natural := + Count_Individual_Associations (Inter, First_Assoc); + Assoc : Node; + Assocs : Assoc_Array_Acc; + Formal_Typ : Type_Acc; + Inter_Typ : Type_Acc; + Static : Boolean; + Res : Valtyp; + begin + -- 2. Build array formal-value + Assocs := new Assoc_Array (1 .. Count); + + -- 3. For each assoc: synth value + Inter_Typ := Get_Subtype_Object (Subprg_Inst, Get_Type (Inter)); + if Inter_Kind = Iir_Kind_Interface_Constant_Declaration then + raise Internal_Error; + else + Formal_Typ := Synth_Subtype_Indication + (Caller_Inst, Get_Actual_Type (First_Assoc)); + Formal_Typ := Unshare_Type_Instance (Formal_Typ, Inter_Typ); + + Create_Object (Subprg_Inst, Inter, (Formal_Typ, null)); + + Assoc := Get_Chain (First_Assoc); + Static := True; + for I in 1 .. Count loop + declare + Formal : Node; + Form_Base : Valtyp; + Form_Typ : Type_Acc; + Form_Off : Value_Offsets; + Dyn : Dyn_Name; + Act_Base : Valtyp; + Act_Typ : Type_Acc; + Act_Off : Value_Offsets; + Act_Dyn : Dyn_Name; + Cb_Val : Valtyp; + begin + Formal := Get_Formal (Assoc); + Synth_Assignment_Prefix + (Caller_Inst, Subprg_Inst, + Formal, Form_Base, Form_Typ, Form_Off, Dyn); + pragma Assert (Dyn = No_Dyn_Name); + pragma Assert (Form_Base = (Formal_Typ, null)); + Synth_Assignment_Prefix + (Caller_Inst, Subprg_Inst, + Get_Actual (Assoc), Act_Base, Act_Typ, Act_Off, Act_Dyn); + if Act_Typ.Kind in Type_Composite then + -- TODO: reshape + null; + end if; + Assocs (I) := (Formal => Formal, + Form_Off => Form_Off, + Act_Base => Act_Base, + Act_Typ => Act_Typ, + Act_Off => Act_Off, + Act_Dyn => Act_Dyn); + if Inter_Kind = Iir_Kind_Interface_Variable_Declaration + and then Get_Mode (Inter) /= Iir_In_Mode + then + Cb_Val := Info_To_Valtyp + (To_Target_Info (Act_Base, Act_Typ, Act_Off, Act_Dyn)); + Create_Object (Caller_Inst, Assoc, Cb_Val); + end if; + Static := Static and then Is_Static (Act_Base.Val); + end; + Assoc := Get_Chain (Assoc); + end loop; + end if; + + -- 4. If static: build mem, if in: build net, if out: build concat + if Static then + Res := Create_Value_Memory (Formal_Typ, Instance_Pool); + for I in Assocs'Range loop + declare + A : Assoc_Record renames Assocs (I); + begin + Copy_Memory (Get_Memory (Res) + A.Form_Off.Mem_Off, + Get_Memory (A.Act_Base) + A.Act_Off.Mem_Off, + A.Act_Typ.Sz); + end; + end loop; + declare + D : Destroy_Type; + begin + Destroy_Init (D, Subprg_Inst); + Destroy_Object (D, Inter); + Destroy_Finish (D); + end; + else + Res := No_Valtyp; + raise Internal_Error; + end if; + + Free_Assoc_Array (Assocs); + + return Res; + end Synth_Individual_Association; + procedure Synth_Subprogram_Associations (Subprg_Inst : Synth_Instance_Acc; Caller_Inst : Synth_Instance_Acc; Init : Association_Iterator_Init) @@ -2175,6 +2299,8 @@ package body Synth.Vhdl_Stmts is Inter : Node; Assoc : Node; Iterator : Association_Iterator; + Marker : Mark_Type; + Val : Valtyp; begin Set_Instance_Const (Subprg_Inst, True); @@ -2184,34 +2310,28 @@ package body Synth.Vhdl_Stmts is Association_Iterate_Next (Iterator, Inter, Assoc); exit when Inter = Null_Node; + Mark_Expr_Pool (Marker); + if Assoc /= Null_Node and then Get_Kind (Assoc) = Iir_Kind_Association_Element_By_Individual then - -- 1. Count number of assocs - -- 2. Build array formal-value - -- 3. For each assoc: synth value - -- 4. If static: build mem, if in: build net, if out: build concat - raise Internal_Error; + Val := Synth_Individual_Association + (Subprg_Inst, Caller_Inst, Inter, Assoc); else - declare - Marker : Mark_Type; - Val : Valtyp; - begin - Mark_Expr_Pool (Marker); - - Val := Synth_Subprogram_Association - (Subprg_Inst, Caller_Inst, Inter, Assoc); - if Val = No_Valtyp then - Set_Error (Subprg_Inst); - exit; - end if; + Val := Synth_Subprogram_Association + (Subprg_Inst, Caller_Inst, Inter, Assoc); + if Val /= No_Valtyp then Val := Unshare (Val, Instance_Pool); - Create_Object (Subprg_Inst, Inter, Val); - - Release_Expr_Pool (Marker); - end; + end if; + end if; + if Val = No_Valtyp then + Set_Error (Subprg_Inst); + exit; end if; + Create_Object (Subprg_Inst, Inter, Val); + + Release_Expr_Pool (Marker); end loop; end Synth_Subprogram_Associations; @@ -2264,14 +2384,17 @@ package body Synth.Vhdl_Stmts is Inter_Chain : Node; Assoc_Chain : Node) is + Marker : Mark_Type; Inter : Node; Assoc : Node; Assoc_Inter : Node; + Formal : Node; Val : Valtyp; Targ : Valtyp; W : Wire_Id; D : Destroy_Type; begin + Mark_Expr_Pool (Marker); Destroy_Init (D, Caller_Inst); Assoc := Assoc_Chain; Assoc_Inter := Inter_Chain; @@ -2282,8 +2405,13 @@ package body Synth.Vhdl_Stmts is and then Get_Kind (Assoc) /= Iir_Kind_Association_Element_By_Individual then + Formal := Get_Formal (Assoc); + if Formal = Null_Node then + Val := Get_Value (Subprg_Inst, Inter); + else + Val := Synth_Expression (Subprg_Inst, Formal); + end if; Targ := Get_Value (Caller_Inst, Assoc); - Val := Get_Value (Subprg_Inst, Inter); if Targ.Val.Kind = Value_Dyn_Alias then Synth_Assignment_Memory (Caller_Inst, Targ.Val.D_Obj, @@ -2295,6 +2423,8 @@ package body Synth.Vhdl_Stmts is (Caller_Inst, Targ, No_Value_Offsets, Val, Assoc); end if; + Release_Expr_Pool (Marker); + -- Free wire used for out/inout interface variables. if Val.Val.Kind = Value_Wire then W := Get_Value_Wire (Val.Val); |