diff options
author | Tristan Gingold <tgingold@free.fr> | 2022-09-19 07:27:42 +0200 |
---|---|---|
committer | Tristan Gingold <tgingold@free.fr> | 2022-09-19 07:27:42 +0200 |
commit | 8d3dcfb5bf4feffd59eaf2802b824059b3d75070 (patch) | |
tree | f7a03711e2dc7abd6ad24a9a9e9e129ea1d3e085 /src | |
parent | f0900d17ff6ac00d3653e7aea5af166b603b155a (diff) | |
download | ghdl-8d3dcfb5bf4feffd59eaf2802b824059b3d75070.tar.gz ghdl-8d3dcfb5bf4feffd59eaf2802b824059b3d75070.tar.bz2 ghdl-8d3dcfb5bf4feffd59eaf2802b824059b3d75070.zip |
synth: rework subprogram associations (WIP)
Diffstat (limited to 'src')
-rw-r--r-- | src/synth/elab-vhdl_types.adb | 3 | ||||
-rw-r--r-- | src/synth/synth-vhdl_stmts.adb | 122 | ||||
-rw-r--r-- | src/vhdl/vhdl-annotations.adb | 4 |
3 files changed, 87 insertions, 42 deletions
diff --git a/src/synth/elab-vhdl_types.adb b/src/synth/elab-vhdl_types.adb index 9a8825493..b92c78452 100644 --- a/src/synth/elab-vhdl_types.adb +++ b/src/synth/elab-vhdl_types.adb @@ -655,6 +655,9 @@ package body Elab.Vhdl_Types is (Syn_Inst, Get_Designated_Type (Atype)); return Create_Access_Type (Acc_Typ); end; + when Iir_Kind_Record_Type_Definition + | Iir_Kind_Array_Type_Definition => + return Get_Subtype_Object (Syn_Inst, Atype); when others => Vhdl.Errors.Error_Kind ("synth_subtype_indication", Atype); end case; diff --git a/src/synth/synth-vhdl_stmts.adb b/src/synth/synth-vhdl_stmts.adb index ffa780625..8b2e4775f 100644 --- a/src/synth/synth-vhdl_stmts.adb +++ b/src/synth/synth-vhdl_stmts.adb @@ -1792,13 +1792,39 @@ package body Synth.Vhdl_Stmts is case Kind is when Association_Function => First_Named_Assoc : Node; - Next_Assoc : Node; + Assoc : Node; when Association_Operator => Op1 : Node; Op2 : Node; 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 @@ -1807,7 +1833,16 @@ package body Synth.Vhdl_Stmts is Iterator := (Kind => Association_Function, Inter => Init.Inter_Chain, First_Named_Assoc => Null_Node, - Next_Assoc => Init.Assoc_Chain); + 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; when Association_Operator => Iterator := (Kind => Association_Operator, Inter => Init.Inter_Chain, @@ -1821,64 +1856,63 @@ package body Synth.Vhdl_Stmts is -- * 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). + -- Associations are returned in the order of interfaces. procedure Association_Iterate_Next (Iterator : in out Association_Iterator; Inter : out Node; - Assoc : out Node) - is - Formal : Node; + Assoc : out Node) is 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. + 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; - Formal := Get_Formal (Assoc); - if Formal = Null_Node then - -- Association by position. - -- Update for the next call. - Iterator.Next_Assoc := Get_Chain (Assoc); + if not Get_Whole_Association_Flag (Assoc) then + -- Still individual assoc. + Iterator.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); - -- 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; + -- 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 + -- Still using association by position. + if Iterator.Assoc = Null_Node then + -- No more associations, all open. return; end if; - Assoc := Get_Chain (Assoc); - end loop; - - -- Not found: open association. - return; + Iterator.Assoc := Get_Chain (Iterator.Assoc); + if Iterator.Assoc = Null_Node + or else Get_Formal (Iterator.Assoc) = Null_Node + then + -- Still by position + return; + end if; + Iterator.First_Named_Assoc := Iterator.Assoc; + end if; + Association_Find_Assoc (Iterator); when Association_Operator => Assoc := Iterator.Op1; + Iterator.Inter := Get_Chain (Iterator.Inter); Iterator.Op1 := Iterator.Op2; Iterator.Op2 := Null_Node; end case; @@ -1979,6 +2013,12 @@ package body Synth.Vhdl_Stmts is Val := Synth_Expression_With_Type (Caller_Inst, Get_Default_Value (Inter), Inter_Typ); Val := Unshare (Val, Instance_Pool); + elsif (Get_Kind (Assoc) + = Iir_Kind_Association_Element_By_Individual) + then + Val.Typ := Synth_Subtype_Indication + (Caller_Inst, Get_Actual_Type (Assoc)); + Val := Create_Value_Memory (Val.Typ, Expr_Pool'Access); else Actual := Get_Actual (Assoc); Info := Synth_Target (Caller_Inst, Actual); @@ -2200,10 +2240,10 @@ package body Synth.Vhdl_Stmts is while Is_Valid (Assoc) loop Inter := Get_Association_Interface (Assoc, Assoc_Inter); - if Is_Copyback_Parameter (Inter) then - if not Get_Whole_Association_Flag (Assoc) then - raise Internal_Error; - end if; + if Is_Copyback_Parameter (Inter) + and then + Get_Kind (Assoc) /= Iir_Kind_Association_Element_By_Individual + then Targ := Get_Value (Caller_Inst, Assoc); Val := Get_Value (Subprg_Inst, Inter); if Targ.Val.Kind = Value_Dyn_Alias then diff --git a/src/vhdl/vhdl-annotations.adb b/src/vhdl/vhdl-annotations.adb index 6957ba4e3..194341730 100644 --- a/src/vhdl/vhdl-annotations.adb +++ b/src/vhdl/vhdl-annotations.adb @@ -885,7 +885,9 @@ package body Vhdl.Annotations is Assoc_Inter := Inter_Chain; while Assoc /= Null_Iir loop Inter := Get_Association_Interface (Assoc, Assoc_Inter); - if Is_Copyback_Parameter (Inter) then + if Get_Kind (Assoc) /= Iir_Kind_Association_Element_By_Individual + and then Is_Copyback_Parameter (Inter) + then Create_Object_Info (Block_Info, Assoc, Kind_Object); end if; Next_Association_Interface (Assoc, Assoc_Inter); |