diff options
author | Tristan Gingold <tgingold@free.fr> | 2015-06-05 22:04:42 +0200 |
---|---|---|
committer | Tristan Gingold <tgingold@free.fr> | 2015-06-05 22:04:42 +0200 |
commit | 463e00e93d8b2507519310789ea9e4fc668cc4ac (patch) | |
tree | 3292cbb3bd7ef34156c1768b7719e828078eeeab /src | |
parent | b250273b27b71e2096f05fe4669dae42d83f4e26 (diff) | |
download | ghdl-463e00e93d8b2507519310789ea9e4fc668cc4ac.tar.gz ghdl-463e00e93d8b2507519310789ea9e4fc668cc4ac.tar.bz2 ghdl-463e00e93d8b2507519310789ea9e4fc668cc4ac.zip |
Rework procedure calls, now use a record to pass parameters.
Diffstat (limited to 'src')
-rw-r--r-- | src/grt/grt-files.adb | 12 | ||||
-rw-r--r-- | src/grt/grt-files.ads | 12 | ||||
-rw-r--r-- | src/grt/grt-lib.adb | 12 | ||||
-rw-r--r-- | src/grt/grt-lib.ads | 12 | ||||
-rw-r--r-- | src/vhdl/translate/trans-chap2.adb | 154 | ||||
-rw-r--r-- | src/vhdl/translate/trans-chap4.adb | 27 | ||||
-rw-r--r-- | src/vhdl/translate/trans-chap4.ads | 3 | ||||
-rw-r--r-- | src/vhdl/translate/trans-chap6.adb | 35 | ||||
-rw-r--r-- | src/vhdl/translate/trans-chap8.adb | 388 | ||||
-rw-r--r-- | src/vhdl/translate/trans-chap9.adb | 3 | ||||
-rw-r--r-- | src/vhdl/translate/trans.ads | 20 |
11 files changed, 311 insertions, 367 deletions
diff --git a/src/grt/grt-files.adb b/src/grt/grt-files.adb index 14dde9702..46d3cedac 100644 --- a/src/grt/grt-files.adb +++ b/src/grt/grt-files.adb @@ -384,16 +384,15 @@ package body Grt.Files is end Ghdl_Text_Read_Length; procedure Ghdl_Untruncated_Text_Read - (Res : Ghdl_Untruncated_Text_Read_Result_Acc; - File : Ghdl_File_Index; - Str : Std_String_Ptr) + (Params : Ghdl_Untruncated_Text_Read_Params_Acc) is + Str : constant Std_String_Ptr := Params.Str; Stream : C_Files; Len : int; Idx : Ghdl_Index_Type; begin - Stream := Get_File (File); - Check_File_Mode (File, True); + Stream := Get_File (Params.File); + Check_File_Mode (Params.File, True); Len := int (Str.Bounds.Dim_1.Length); if fgets (Str.Base (0)'Address, Len, Stream) = Null_Address then Internal_Error ("ghdl_untruncated_text_read: end of file"); @@ -405,7 +404,7 @@ package body Grt.Files is exit; end if; end loop; - Res.Len := Std_Integer (Idx); + Params.Len := Std_Integer (Idx); end Ghdl_Untruncated_Text_Read; procedure File_Close (File : Ghdl_File_Index; Is_Text : Boolean) @@ -447,4 +446,3 @@ package body Grt.Files is fflush (Stream); end Ghdl_File_Flush; end Grt.Files; - diff --git a/src/grt/grt-files.ads b/src/grt/grt-files.ads index 14f998468..3fadc981e 100644 --- a/src/grt/grt-files.ads +++ b/src/grt/grt-files.ads @@ -75,17 +75,17 @@ package Grt.Files is function Ghdl_Text_Read_Length (File : Ghdl_File_Index; Str : Std_String_Ptr) return Std_Integer; - type Ghdl_Untruncated_Text_Read_Result is record + type Ghdl_Untruncated_Text_Read_Params is record + File : Ghdl_File_Index; + Str : Std_String_Ptr; Len : Std_Integer; end record; - type Ghdl_Untruncated_Text_Read_Result_Acc is - access Ghdl_Untruncated_Text_Read_Result; + type Ghdl_Untruncated_Text_Read_Params_Acc is + access Ghdl_Untruncated_Text_Read_Params; procedure Ghdl_Untruncated_Text_Read - (Res : Ghdl_Untruncated_Text_Read_Result_Acc; - File : Ghdl_File_Index; - Str : Std_String_Ptr); + (Params : Ghdl_Untruncated_Text_Read_Params_Acc); procedure Ghdl_Text_File_Close (File : Ghdl_File_Index); procedure Ghdl_File_Close (File : Ghdl_File_Index); diff --git a/src/grt/grt-lib.adb b/src/grt/grt-lib.adb index d2b095c67..b4505adb6 100644 --- a/src/grt/grt-lib.adb +++ b/src/grt/grt-lib.adb @@ -272,25 +272,25 @@ package body Grt.Lib is end Ghdl_Get_Resolution_Limit; procedure Ghdl_Control_Simulation - (Stop : Ghdl_B1; Has_Status : Ghdl_B1; Status : Std_Integer) is + (Params : Ghdl_Control_Simulation_Params_Ptr) is begin Report_H; -- Report_C (Grt.Options.Progname); Report_C ("simulation "); - if Stop then + if Params.Stop then Report_C ("stopped"); else Report_C ("finished"); end if; Report_C (" @"); Report_Now_C; - if Has_Status then + if Params.Has_Status then Report_C (" with status "); - Report_C (Integer (Status)); + Report_C (Integer (Params.Status)); end if; Report_E (""); - if Has_Status then - Exit_Status := Integer (Status); + if Params.Has_Status then + Exit_Status := Integer (Params.Status); end if; Exit_Simulation; end Ghdl_Control_Simulation; diff --git a/src/grt/grt-lib.ads b/src/grt/grt-lib.ads index 50be6a7a6..dcd2c55b7 100644 --- a/src/grt/grt-lib.ads +++ b/src/grt/grt-lib.ads @@ -94,8 +94,18 @@ package Grt.Lib is ); function Ghdl_Get_Resolution_Limit return Std_Time; + + type Ghdl_Control_Simulation_Params is record + Stop : Ghdl_B1; + Has_Status : Ghdl_B1; + Status : Std_Integer; + end record; + + type Ghdl_Control_Simulation_Params_Ptr is access + Ghdl_Control_Simulation_Params; + procedure Ghdl_Control_Simulation - (Stop : Ghdl_B1; Has_Status : Ghdl_B1; Status : Std_Integer); + (Params : Ghdl_Control_Simulation_Params_Ptr); private pragma Export (C, Ghdl_Memcpy, "__ghdl_memcpy"); diff --git a/src/vhdl/translate/trans-chap2.adb b/src/vhdl/translate/trans-chap2.adb index e0c19c606..a43179e78 100644 --- a/src/vhdl/translate/trans-chap2.adb +++ b/src/vhdl/translate/trans-chap2.adb @@ -38,8 +38,8 @@ package body Trans.Chap2 is procedure Elab_Package (Spec : Iir_Package_Declaration); - type Name_String_Xlat_Array is array (Name_Id range <>) of - String (1 .. 4); + type Name_String_Xlat_Array is array (Name_Id range <>) of String (1 .. 4); + Operator_String_Xlat : constant Name_String_Xlat_Array (Std_Names.Name_Id_Operators) := (Std_Names.Name_Op_Equality => "OPEq", @@ -66,11 +66,10 @@ package body Trans.Chap2 is -- overload number if any. procedure Push_Subprg_Identifier (Spec : Iir; Mark : out Id_Mark_Type) is - Id : Name_Id; + Id : constant Name_Id := Get_Identifier (Spec); begin -- FIXME: name_shift_operators, name_logical_operators, -- name_word_operators, name_mod, name_rem - Id := Get_Identifier (Spec); if Id in Std_Names.Name_Id_Operators then Push_Identifier_Prefix (Mark, Operator_String_Xlat (Id), Get_Overload_Number (Spec)); @@ -109,7 +108,6 @@ package body Trans.Chap2 is end loop; end Elab_Subprogram_Interfaces; - -- Return the type of a subprogram interface. -- Return O_Tnode_Null if the parameter is passed through the -- interface record. @@ -141,21 +139,19 @@ package body Trans.Chap2 is procedure Translate_Subprogram_Declaration (Spec : Iir) is - Info : constant Subprg_Info_Acc := Get_Info (Spec); - Is_Func : constant Boolean := + Info : constant Subprg_Info_Acc := Get_Info (Spec); + Is_Func : constant Boolean := Get_Kind (Spec) = Iir_Kind_Function_Declaration; - Inter : Iir; - Inter_Type : Iir; - Arg_Info : Ortho_Info_Acc; - Tinfo : Type_Info_Acc; - Interface_List : O_Inter_List; - Has_Result_Record : Boolean; - El_List : O_Element_List; - Mark : Id_Mark_Type; - Rtype : Iir; - Id : O_Ident; - Storage : O_Storage; - Foreign : Foreign_Info_Type := Foreign_Bad; + Inter : Iir; + Arg_Info : Ortho_Info_Acc; + Tinfo : Type_Info_Acc; + Interface_List : O_Inter_List; + El_List : O_Element_List; + Mark : Id_Mark_Type; + Rtype : Iir; + Id : O_Ident; + Storage : O_Storage; + Foreign : Foreign_Info_Type := Foreign_Bad; begin -- Set the identifier prefix with the subprogram identifier and -- overload number if any. @@ -210,54 +206,37 @@ package body Trans.Chap2 is -- For parameters passed via copy and that needs a copy-out, -- gather them in a record. An access to the record is then -- passed to the procedure. - Has_Result_Record := False; Inter := Get_Interface_Declaration_Chain (Spec); - while Inter /= Null_Iir loop - Arg_Info := Add_Info (Inter, Kind_Interface); - Inter_Type := Get_Type (Inter); - Tinfo := Get_Info (Inter_Type); - if Get_Kind (Inter) = Iir_Kind_Interface_Variable_Declaration - and then Get_Mode (Inter) in Iir_Out_Modes - and then Tinfo.Type_Mode not in Type_Mode_By_Ref - and then Tinfo.Type_Mode /= Type_Mode_File - then - -- This interface is done via the result record. - -- Note: file passed through variables are vhdl87 files, - -- which are initialized at elaboration and thus - -- behave like an IN parameter. - if not Has_Result_Record then - -- Create the record. - Start_Record_Type (El_List); - Has_Result_Record := True; - end if; - -- Add a field to the record. + if Inter /= Null_Iir then + Start_Record_Type (El_List); + while Inter /= Null_Iir loop + Arg_Info := Add_Info (Inter, Kind_Interface); New_Record_Field (El_List, Arg_Info.Interface_Field, Create_Identifier_Without_Prefix (Inter), - Tinfo.Ortho_Type (Mode_Value)); - else - Arg_Info.Interface_Field := O_Fnode_Null; - end if; - Inter := Get_Chain (Inter); - end loop; - if Has_Result_Record then + Translate_Interface_Type (Inter)); + Inter := Get_Chain (Inter); + end loop; -- Declare the record type and an access to the record. - Finish_Record_Type (El_List, Info.Res_Record_Type); - New_Type_Decl (Create_Identifier ("RESTYPE"), - Info.Res_Record_Type); - Info.Res_Record_Ptr := New_Access_Type (Info.Res_Record_Type); - New_Type_Decl (Create_Identifier ("RESPTR"), - Info.Res_Record_Ptr); + Finish_Record_Type (El_List, Info.Subprg_Params_Type); + New_Type_Decl (Create_Identifier ("PARAMSTYPE"), + Info.Subprg_Params_Type); + Info.Subprg_Params_Ptr := + New_Access_Type (Info.Subprg_Params_Type); + New_Type_Decl (Create_Identifier ("PARAMSPTR"), + Info.Subprg_Params_Ptr); else - Info.Res_Interface := O_Dnode_Null; + Info.Subprg_Params_Type := O_Tnode_Null; + Info.Subprg_Params_Ptr := O_Tnode_Null; end if; Start_Procedure_Decl (Interface_List, Id, Storage); - if Has_Result_Record then - -- Add the record parameter. + if Info.Subprg_Params_Type /= O_Tnode_Null then New_Interface_Decl (Interface_List, Info.Res_Interface, - Get_Identifier ("RESULT"), - Info.Res_Record_Ptr); + Get_Identifier ("PARAMS"), + Info.Subprg_Params_Ptr); + else + Info.Res_Interface := O_Dnode_Null; end if; end if; @@ -267,27 +246,21 @@ package body Trans.Chap2 is end if; -- Translate interfaces. - Inter := Get_Interface_Declaration_Chain (Spec); - while Inter /= Null_Iir loop - if Is_Func then + if Is_Func then + Inter := Get_Interface_Declaration_Chain (Spec); + while Inter /= Null_Iir loop -- Create the info. Arg_Info := Add_Info (Inter, Kind_Interface); Arg_Info.Interface_Field := O_Fnode_Null; - else - -- The info was already created (just above) - Arg_Info := Get_Info (Inter); - end if; - if Arg_Info.Interface_Field = O_Fnode_Null then - -- Not via the RESULT parameter. Arg_Info.Interface_Type := Translate_Interface_Type (Inter); New_Interface_Decl (Interface_List, Arg_Info.Interface_Node, Create_Identifier_Without_Prefix (Inter), Arg_Info.Interface_Type); - end if; - Inter := Get_Chain (Inter); - end loop; + Inter := Get_Chain (Inter); + end loop; + end if; Finish_Subprogram_Decl (Interface_List, Info.Ortho_Func); -- Call the hook for foreign subprograms. @@ -411,10 +384,10 @@ package body Trans.Chap2 is Push_Instance_Factory (Info.Subprg_Frame_Scope'Access); Add_Subprg_Instance_Field (Upframe_Field); - if Info.Res_Record_Ptr /= O_Tnode_Null then - Info.Res_Record_Var := + if Info.Subprg_Params_Ptr /= O_Tnode_Null then + Info.Subprg_Params_Var := Create_Var (Create_Var_Identifier ("RESULT"), - Info.Res_Record_Ptr); + Info.Subprg_Params_Ptr); end if; -- Create fields for parameters. @@ -501,14 +474,14 @@ package body Trans.Chap2 is Subprgs.Set_Subprg_Instance_Field (Frame_Ptr, Upframe_Field, Info.Subprg_Instance); - if Info.Res_Record_Type /= O_Tnode_Null then + if Info.Subprg_Params_Type /= O_Tnode_Null then -- Initialize the RESULT field - New_Assign_Stmt (Get_Var (Info.Res_Record_Var), + New_Assign_Stmt (Get_Var (Info.Subprg_Params_Var), New_Obj_Value (Info.Res_Interface)); -- Do not reference the RESULT field in the subprogram body, -- directly reference the RESULT parameter. -- FIXME: has a flag (see below for parameters). - Info.Res_Record_Var := Null_Var; + Info.Subprg_Params_Var := Null_Var; end if; -- Copy parameters to FRAME. @@ -535,31 +508,6 @@ package body Trans.Chap2 is end; end if; - -- Init out parameters passed by value/copy. - declare - Inter : Iir; - Inter_Type : Iir; - Type_Info : Type_Info_Acc; - begin - Inter := Get_Interface_Declaration_Chain (Spec); - while Inter /= Null_Iir loop - if Get_Kind (Inter) = Iir_Kind_Interface_Variable_Declaration - and then Get_Mode (Inter) = Iir_Out_Mode - then - Inter_Type := Get_Type (Inter); - Type_Info := Get_Info (Inter_Type); - if (Type_Info.Type_Mode in Type_Mode_By_Value - or Type_Info.Type_Mode in Type_Mode_By_Copy) - and then Type_Info.Type_Mode /= Type_Mode_File - then - Chap4.Init_Object - (Chap6.Translate_Name (Inter), Inter_Type); - end if; - end if; - Inter := Get_Chain (Inter); - end loop; - end; - Chap4.Elab_Declaration_Chain (Subprg, Final); -- If finalization is required, create a dummy loop around the @@ -922,9 +870,9 @@ package body Trans.Chap2 is Use_Stack2 => Src.Use_Stack2, Ortho_Func => Src.Ortho_Func, Res_Interface => Src.Res_Interface, - Res_Record_Var => Instantiate_Var (Src.Res_Record_Var), - Res_Record_Type => Src.Res_Record_Type, - Res_Record_Ptr => Src.Res_Record_Ptr, + Subprg_Params_Var => Instantiate_Var (Src.Subprg_Params_Var), + Subprg_Params_Type => Src.Subprg_Params_Type, + Subprg_Params_Ptr => Src.Subprg_Params_Ptr, Subprg_Frame_Scope => Dest.Subprg_Frame_Scope, Subprg_Instance => Instantiate_Subprg_Instance (Src.Subprg_Instance), diff --git a/src/vhdl/translate/trans-chap4.adb b/src/vhdl/translate/trans-chap4.adb index 2e330338e..d9de806eb 100644 --- a/src/vhdl/translate/trans-chap4.adb +++ b/src/vhdl/translate/trans-chap4.adb @@ -358,19 +358,28 @@ package body Trans.Chap4 is New_Procedure_Call (Assoc); end Fini_Protected_Object; - procedure Init_Object (Obj : Mnode; Obj_Type : Iir) + function Get_Scalar_Initial_Value (Atype : Iir) return O_Enode is - Tinfo : Type_Info_Acc; + Tinfo : constant Type_Info_Acc := Get_Info (Atype); begin - Tinfo := Get_Type_Info (Obj); case Tinfo.Type_Mode is when Type_Mode_Scalar => - New_Assign_Stmt - (M2Lv (Obj), Chap14.Translate_Left_Type_Attribute (Obj_Type)); + return Chap14.Translate_Left_Type_Attribute (Atype); when Type_Mode_Acc => - New_Assign_Stmt - (M2Lv (Obj), - New_Lit (New_Null_Access (Tinfo.Ortho_Type (Mode_Value)))); + return New_Lit (New_Null_Access (Tinfo.Ortho_Type (Mode_Value))); + when others => + Error_Kind ("get_scalar_initial_value", Atype); + end case; + end Get_Scalar_Initial_Value; + + procedure Init_Object (Obj : Mnode; Obj_Type : Iir) + is + Tinfo : constant Type_Info_Acc := Get_Type_Info (Obj); + begin + case Tinfo.Type_Mode is + when Type_Mode_Scalar + | Type_Mode_Acc => + New_Assign_Stmt (M2Lv (Obj), Get_Scalar_Initial_Value (Obj_Type)); when Type_Mode_Fat_Acc => declare Dinfo : Type_Info_Acc; @@ -814,7 +823,7 @@ package body Trans.Chap4 is if Data.Has_Val then Init_Val := M2E (Data.Val); else - Init_Val := Chap14.Translate_Left_Type_Attribute (Targ_Type); + Init_Val := Get_Scalar_Initial_Value (Targ_Type); end if; Start_Association (Assoc, Create_Subprg); diff --git a/src/vhdl/translate/trans-chap4.ads b/src/vhdl/translate/trans-chap4.ads index 129942437..6f9b8aefc 100644 --- a/src/vhdl/translate/trans-chap4.ads +++ b/src/vhdl/translate/trans-chap4.ads @@ -87,6 +87,9 @@ package Trans.Chap4 is -- Allocate the storage for OBJ, if necessary. procedure Elab_Object_Storage (Obj : Iir); + -- For a scalar or access type ATYPE, return the default initial value. + function Get_Scalar_Initial_Value (Atype : Iir) return O_Enode; + -- Initialize NAME/OBJ with VALUE. procedure Elab_Object_Init (Name : Mnode; Obj : Iir; Value : Iir); diff --git a/src/vhdl/translate/trans-chap6.adb b/src/vhdl/translate/trans-chap6.adb index 4b89cecc2..96e7b394f 100644 --- a/src/vhdl/translate/trans-chap6.adb +++ b/src/vhdl/translate/trans-chap6.adb @@ -771,18 +771,17 @@ package body Trans.Chap6 is Linter : O_Lnode; begin if Info.Interface_Node = O_Dnode_Null then - -- The parameter is passed via a field of the RESULT + -- The parameter is passed via a field of the PARAMS -- record parameter. - if Subprg_Info.Res_Record_Var = Null_Var then + if Subprg_Info.Subprg_Params_Var = Null_Var then + -- Direct access to the parameter. Linter := New_Obj (Subprg_Info.Res_Interface); else - -- Unnesting case. - Linter := Get_Var (Subprg_Info.Res_Record_Var); + -- Unnesting case: upscope access. + Linter := Get_Var (Subprg_Info.Subprg_Params_Var); end if; - return Lv2M (New_Selected_Element - (New_Acc_Value (Linter), - Info.Interface_Field), - Type_Info, Kind); + Linter := New_Selected_Element + (New_Acc_Value (Linter), Info.Interface_Field); else -- Unnesting case: the parameter was copied in the -- subprogram frame so that nested subprograms can @@ -790,17 +789,17 @@ package body Trans.Chap6 is Linter := New_Selected_Element (Get_Instance_Ref (Subprg_Info.Subprg_Frame_Scope), Info.Interface_Field); - case Type_Info.Type_Mode is - when Type_Mode_Unknown => - raise Internal_Error; - when Type_Mode_By_Value => - return Lv2M (Linter, Type_Info, Kind); - when Type_Mode_By_Copy - | Type_Mode_By_Ref => - -- Parameter is passed by reference. - return Lp2M (Linter, Type_Info, Kind); - end case; end if; + case Type_Info.Type_Mode is + when Type_Mode_Unknown => + raise Internal_Error; + when Type_Mode_By_Value => + return Lv2M (Linter, Type_Info, Kind); + when Type_Mode_By_Copy + | Type_Mode_By_Ref => + -- Parameter is passed by reference. + return Lp2M (Linter, Type_Info, Kind); + end case; end; end if; when others => diff --git a/src/vhdl/translate/trans-chap8.adb b/src/vhdl/translate/trans-chap8.adb index 283ffbcdb..8a3711ee2 100644 --- a/src/vhdl/translate/trans-chap8.adb +++ b/src/vhdl/translate/trans-chap8.adb @@ -1608,11 +1608,11 @@ package body Trans.Chap8 is end case; end Translate_Implicit_Procedure_Call; - function Do_Conversion (Conv : Iir; Expr : Iir; Src : Mnode) return O_Enode - is + function Do_Conversion (Conv : Iir; Expr : Iir; Src : O_Enode) + return O_Enode is begin if Conv = Null_Iir then - return M2E (Src); + return Src; -- case Get_Type_Info (Dest).Type_Mode is -- when Type_Mode_Thin => -- New_Assign_Stmt (M2Lv (Dest), M2E (Src)); @@ -1647,7 +1647,7 @@ package body Trans.Chap8 is Subprgs.Add_Subprg_Instance_Assoc (Constr, Conv_Info.Subprg_Instance); - New_Association (Constr, M2E (Src)); + New_Association (Constr, Src); if Conv_Info.Res_Interface /= O_Dnode_Null then -- Composite result. @@ -1660,28 +1660,65 @@ package body Trans.Chap8 is end; when Iir_Kind_Type_Conversion => return Chap7.Translate_Type_Conversion - (M2E (Src), Get_Type (Expr), - Get_Type (Conv), Null_Iir); + (Src, Get_Type (Expr), Get_Type (Conv), Null_Iir); when others => Error_Kind ("do_conversion", Conv); end case; end if; end Do_Conversion; + -- Translate the formal name FORMAL_NAME of an individual association but + -- replace the interface name by INTER_VAR. FORMAL_INFO is the info of + -- the interface. This is used to access to a sub-element of the variable + -- representing the whole actual. + function Translate_Individual_Association_Formal + (Formal_Name : Iir; + Formal_Info : Ortho_Info_Acc; + Inter_Var : Mnode) + return Mnode + is + Prev_Node : O_Dnode; + Prev_Field : O_Fnode; + Res : Mnode; + begin + -- Change the formal variable so that it is the local variable + -- that will be passed to the subprogram. + Prev_Node := Formal_Info.Interface_Node; + Prev_Field := Formal_Info.Interface_Field; + + -- We need a pointer since the interface is by reference. + Formal_Info.Interface_Node := M2Dp (Inter_Var); + Formal_Info.Interface_Field := O_Fnode_Null; + + Res := Chap6.Translate_Name (Formal_Name); + + Formal_Info.Interface_Node := Prev_Node; + Formal_Info.Interface_Field := Prev_Field; + + return Res; + end Translate_Individual_Association_Formal; + function Translate_Subprogram_Call (Imp : Iir; Assoc_Chain : Iir; Obj : Iir) return O_Enode is Is_Procedure : constant Boolean := Get_Kind (Imp) = Iir_Kind_Procedure_Declaration; Is_Function : constant Boolean := not Is_Procedure; + Info : constant Subprg_Info_Acc := Get_Info (Imp); + type Mnode_Array is array (Natural range <>) of Mnode; type O_Enode_Array is array (Natural range <>) of O_Enode; Nbr_Assoc : constant Natural := Iir_Chains.Get_Chain_Length (Assoc_Chain); + + -- References to the formals (for copy-out), and variables for whole + -- actual of individual associations. Params : Mnode_Array (0 .. Nbr_Assoc - 1); + + -- The values of actuals. E_Params : O_Enode_Array (0 .. Nbr_Assoc - 1); - Info : constant Subprg_Info_Acc := Get_Info (Imp); - Copy_Out : O_Dnode; + + Params_Var : O_Dnode; Res : Mnode; El : Iir; Pos : Natural; @@ -1689,17 +1726,17 @@ package body Trans.Chap8 is Act : Iir; Actual_Type : Iir; Formal : Iir; + Mode : Iir_Mode; Base_Formal : Iir; Formal_Type : Iir; Ftype_Info : Type_Info_Acc; - Ftype_Binfo : Type_Info_Acc; Formal_Info : Ortho_Info_Acc; Val : O_Enode; Param : Mnode; + Param_Type : Iir; Last_Individual : Natural; Ptr : O_Lnode; In_Conv : Iir; - In_Expr : Iir; Out_Conv : Iir; Out_Expr : Iir; Formal_Object_Kind : Object_Kind_Type; @@ -1724,12 +1761,11 @@ package body Trans.Chap8 is end; end if; - -- Create an in-out result record for in-out arguments passed by - -- value. - if Is_Procedure and then Info.Res_Record_Type /= O_Tnode_Null then - Copy_Out := Create_Temp (Info.Res_Record_Type); + -- Create the variable containing the parameters (only for procedures). + if Is_Procedure and then Info.Subprg_Params_Type /= O_Tnode_Null then + Params_Var := Create_Temp (Info.Subprg_Params_Type); else - Copy_Out := O_Dnode_Null; + Params_Var := O_Dnode_Null; end if; -- Evaluate in-out parameters and parameters passed by ref, since @@ -1742,145 +1778,138 @@ package body Trans.Chap8 is Params (Pos) := Mnode_Null; E_Params (Pos) := O_Enode_Null; - Formal := Get_Formal (El); - if Get_Kind (Formal) in Iir_Kinds_Denoting_Name then - Formal := Get_Named_Entity (Formal); - end if; + Formal := Strip_Denoting_Name (Get_Formal (El)); Base_Formal := Get_Association_Interface (El); Formal_Type := Get_Type (Formal); Formal_Info := Get_Info (Base_Formal); + Ftype_Info := Get_Info (Formal_Type); + if Get_Kind (Base_Formal) = Iir_Kind_Interface_Signal_Declaration then Formal_Object_Kind := Mode_Signal; else Formal_Object_Kind := Mode_Value; end if; - Ftype_Info := Get_Info (Formal_Type); - Ftype_Binfo := Get_Info (Get_Base_Type (Formal_Type)); case Get_Kind (El) is when Iir_Kind_Association_Element_Open => Act := Get_Default_Value (Formal); In_Conv := Null_Iir; - Out_Conv := Null_Iir; when Iir_Kind_Association_Element_By_Expression => Act := Get_Actual (El); In_Conv := Get_In_Conversion (El); - Out_Conv := Get_Out_Conversion (El); when Iir_Kind_Association_Element_By_Individual => Actual_Type := Get_Actual_Type (El); - -- A non-composite type cannot be associated by element. - pragma Assert (Formal_Info.Interface_Field = O_Fnode_Null); - if Ftype_Info.Type_Mode = Type_Mode_Fat_Array then + -- Create the constraints and then the object. Chap3.Create_Array_Subtype (Actual_Type, True); Bounds := Chap3.Get_Array_Type_Bounds (Actual_Type); Param := Create_Temp (Ftype_Info, Formal_Object_Kind); Chap3.Translate_Object_Allocation (Param, Alloc_Stack, Formal_Type, Bounds); else + -- Create the object. Param := Create_Temp (Ftype_Info, Formal_Object_Kind); Chap4.Allocate_Complex_Object (Formal_Type, Alloc_Stack, Param); end if; + + -- Save the object as it will be used by the following + -- associations. Last_Individual := Pos; Params (Pos) := Param; + + if Formal_Info.Interface_Field /= O_Fnode_Null then + -- Set the PARAMS field. + Ptr := New_Selected_Element + (New_Obj (Params_Var), Formal_Info.Interface_Field); + New_Assign_Stmt (Ptr, M2E (Param)); + end if; + goto Continue; when others => Error_Kind ("translate_procedure_call", El); end case; Actual_Type := Get_Type (Act); - if Formal_Info.Interface_Field /= O_Fnode_Null then - -- Copy-out argument. - -- This is not a composite type. - Param := Chap6.Translate_Name (Act); - pragma Assert (Get_Object_Kind (Param) = Mode_Value); - Params (Pos) := Stabilize (Param); - if In_Conv /= Null_Iir - or else Get_Mode (Formal) = Iir_Inout_Mode - then - -- Arguments may be assigned if there is an in conversion. - Ptr := New_Selected_Element - (New_Obj (Copy_Out), Formal_Info.Interface_Field); - Param := Lv2M (Ptr, Ftype_Info, Mode_Value); - if In_Conv /= Null_Iir then - In_Expr := In_Conv; + -- Evaluate the actual. + Param_Type := Actual_Type; + case Get_Kind (Base_Formal) is + when Iir_Kind_Interface_Constant_Declaration + | Iir_Kind_Interface_File_Declaration => + -- No conversion here. + pragma Assert (In_Conv = Null_Iir); + Val := Chap7.Translate_Expression (Act, Formal_Type); + Param_Type := Formal_Type; + when Iir_Kind_Interface_Signal_Declaration => + -- No conversion. + Param := Chap6.Translate_Name (Act); + Val := M2E (Param); + when Iir_Kind_Interface_Variable_Declaration => + Mode := Get_Mode (Base_Formal); + if Mode = Iir_In_Mode then + Val := Chap7.Translate_Expression (Act); else - In_Expr := Act; - end if; - Chap7.Translate_Assign - (Param, - Do_Conversion (In_Conv, Act, Params (Pos)), - In_Expr, - Formal_Type, El); - end if; - elsif Ftype_Binfo.Type_Mode not in Type_Mode_By_Value then - -- Passed by reference. - case Get_Kind (Base_Formal) is - when Iir_Kind_Interface_Constant_Declaration - | Iir_Kind_Interface_File_Declaration => - -- No conversion here. - E_Params (Pos) := - Chap7.Translate_Expression (Act, Formal_Type); - when Iir_Kind_Interface_Variable_Declaration - | Iir_Kind_Interface_Signal_Declaration => Param := Chap6.Translate_Name (Act); - -- Atype may not have been set (eg: slice). - if Base_Formal /= Formal then + if Base_Formal /= Formal + or else Ftype_Info.Type_Mode in Type_Mode_By_Value + then + -- For out/inout, we need to keep the reference for the + -- copy-out. Stabilize (Param); Params (Pos) := Param; end if; - E_Params (Pos) := M2E (Param); - if Formal_Type /= Actual_Type then - -- Implicit array conversion or subtype check. - E_Params (Pos) := Chap7.Translate_Implicit_Conv - (E_Params (Pos), Actual_Type, Formal_Type, - Get_Object_Kind (Param), Act); + if In_Conv = Null_Iir + and then Mode = Iir_Out_Mode + and then Ftype_Info.Type_Mode in Type_Mode_Thin + and then Ftype_Info.Type_Mode /= Type_Mode_File + then + -- Scalar OUT interface. Just give an initial value. + -- FIXME: individual association ?? + Val := Chap4.Get_Scalar_Initial_Value (Formal_Type); + Param_Type := Formal_Type; + else + Val := M2E (Param); end if; - when others => - Error_Kind ("translate_procedure_call(2)", Formal); - end case; + end if; + if In_Conv /= Null_Iir then + Val := Do_Conversion (In_Conv, Act, Val); + Act := In_Conv; + Param_Type := Get_Type (In_Conv); + end if; + when others => + Error_Kind ("translate_procedure_call(2)", Formal); + end case; + + -- Implicit conversion to formal type. + if Param_Type /= Formal_Type then + -- Implicit array conversion or subtype check. + Val := Chap7.Translate_Implicit_Conv + (Val, Param_Type, Formal_Type, Formal_Object_Kind, Act); + end if; + if Get_Kind (Base_Formal) /= Iir_Kind_Interface_Signal_Declaration + then + Val := Chap3.Maybe_Insert_Scalar_Check (Val, Act, Formal_Type); end if; + + -- Assign actual, if needed. if Base_Formal /= Formal then - -- Individual association. - if Ftype_Binfo.Type_Mode not in Type_Mode_By_Value then - -- Not by-value actual already translated. - Val := E_Params (Pos); - else - -- By value association. - Act := Get_Actual (El); - if Get_Kind (Base_Formal) - = Iir_Kind_Interface_Constant_Declaration - then - Val := Chap7.Translate_Expression (Act, Formal_Type); - else - Params (Pos) := Chap6.Translate_Name (Act); - -- Since signals are passed by reference, they are not - -- copied back, so do not stabilize them (furthermore, - -- it is not possible to stabilize them). - if Formal_Object_Kind = Mode_Value then - Params (Pos) := Stabilize (Params (Pos)); - end if; - Val := M2E (Params (Pos)); - end if; - end if; - -- Assign formal. - -- Change the formal variable so that it is the local variable - -- that will be passed to the subprogram. - declare - Prev_Node : O_Dnode; - begin - Prev_Node := Formal_Info.Interface_Node; - -- We need a pointer since the interface is by reference. - Formal_Info.Interface_Node := - M2Dp (Params (Last_Individual)); - Param := Chap6.Translate_Name (Formal); - Formal_Info.Interface_Node := Prev_Node; - end; - Chap7.Translate_Assign (Param, Val, Act, Formal_Type, El); + -- Individual association: assign the individual actual to the + -- whole actual. + Param := Translate_Individual_Association_Formal + (Formal, Formal_Info, Params (Last_Individual)); + Chap7.Translate_Assign + (Param, Val, Act, Formal_Type, El); + elsif Formal_Info.Interface_Field /= O_Fnode_Null then + -- Set the PARAMS field. + Ptr := New_Selected_Element + (New_Obj (Params_Var), Formal_Info.Interface_Field); + New_Assign_Stmt (Ptr, Val); + else + E_Params (Pos) := Val; end if; + << Continue >> null; El := Get_Chain (El); Pos := Pos + 1; @@ -1894,14 +1923,17 @@ package body Trans.Chap8 is New_Association (Constr, M2E (Res)); end if; - if Copy_Out /= O_Dnode_Null then - New_Association - (Constr, New_Address (New_Obj (Copy_Out), Info.Res_Record_Ptr)); + if Params_Var /= O_Dnode_Null then + -- Parameters record (for procedures). + New_Association (Constr, New_Address (New_Obj (Params_Var), + Info.Subprg_Params_Ptr)); end if; if Obj /= Null_Iir then + -- Protected object. New_Association (Constr, M2E (Chap6.Translate_Name (Obj))); else + -- Instance. Subprgs.Add_Subprg_Instance_Assoc (Constr, Info.Subprg_Instance); end if; @@ -1909,64 +1941,17 @@ package body Trans.Chap8 is El := Assoc_Chain; Pos := 0; while El /= Null_Iir loop - Formal := Get_Formal (El); - if Get_Kind (Formal) in Iir_Kinds_Denoting_Name then - Formal := Get_Named_Entity (Formal); - end if; + Formal := Strip_Denoting_Name (Get_Formal (El)); Base_Formal := Get_Association_Interface (El); Formal_Info := Get_Info (Base_Formal); - Formal_Type := Get_Type (Formal); - Ftype_Info := Get_Info (Formal_Type); - if Get_Kind (El) = Iir_Kind_Association_Element_By_Individual then - Last_Individual := Pos; - New_Association (Constr, M2E (Params (Pos))); - elsif Base_Formal /= Formal then - -- Individual association. - null; - elsif Formal_Info.Interface_Field = O_Fnode_Null then - if Ftype_Info.Type_Mode in Type_Mode_By_Value then - -- Parameter passed by value. - if E_Params (Pos) /= O_Enode_Null then - Val := E_Params (Pos); - raise Internal_Error; - else - case Get_Kind (El) is - when Iir_Kind_Association_Element_Open => - Act := Get_Default_Value (Formal); - In_Conv := Null_Iir; - when Iir_Kind_Association_Element_By_Expression => - Act := Get_Actual (El); - In_Conv := Get_In_Conversion (El); - when others => - Error_Kind ("translate_procedure_call(2)", El); - end case; - case Get_Kind (Formal) is - when Iir_Kind_Interface_Signal_Declaration => - Param := Chap6.Translate_Name (Act); - -- This is a scalar. - Val := M2E (Param); - when others => - if In_Conv = Null_Iir then - Val := Chap7.Translate_Expression - (Act, Formal_Type); - Val := Chap3.Maybe_Insert_Scalar_Check - (Val, Act, Formal_Type); - else - Actual_Type := Get_Type (Act); - Val := Do_Conversion - (In_Conv, - Act, - E2M (Chap7.Translate_Expression (Act, - Actual_Type), - Get_Info (Actual_Type), - Mode_Value)); - end if; - end case; - end if; - New_Association (Constr, Val); - else - -- Parameter passed by ref, which was already computed. + if Formal_Info.Interface_Field = O_Fnode_Null then + -- Not a PARAMS field. + if Get_Kind (El) = Iir_Kind_Association_Element_By_Individual then + -- Pass the whole data for an individual association. + New_Association (Constr, M2E (Params (Pos))); + elsif Base_Formal = Formal then + -- Whole association. New_Association (Constr, E_Params (Pos)); end if; end if; @@ -1974,6 +1959,7 @@ package body Trans.Chap8 is Pos := Pos + 1; end loop; + -- Subprogram call. if Is_Procedure then New_Procedure_Call (Constr); else @@ -1990,49 +1976,43 @@ package body Trans.Chap8 is El := Assoc_Chain; Pos := 0; while El /= Null_Iir loop - Formal := Get_Formal (El); - Base_Formal := Get_Association_Interface (El); - Formal_Type := Get_Type (Formal); - Ftype_Info := Get_Info (Formal_Type); - Formal_Info := Get_Info (Base_Formal); - if Get_Kind (Base_Formal) = Iir_Kind_Interface_Variable_Declaration - and then Get_Mode (Base_Formal) in Iir_Out_Modes - and then Params (Pos) /= Mnode_Null - then - if Formal_Info.Interface_Field /= O_Fnode_Null then - -- OUT parameters. - Out_Conv := Get_Out_Conversion (El); - if Out_Conv = Null_Iir then - Out_Expr := Formal; - else - Out_Expr := Out_Conv; - end if; + if Get_Kind (El) = Iir_Kind_Association_Element_By_Individual then + Last_Individual := Pos; + elsif Params (Pos) /= Mnode_Null then + Formal := Strip_Denoting_Name (Get_Formal (El)); + Base_Formal := Get_Association_Interface (El); + + pragma Assert (Get_Kind (Base_Formal) + = Iir_Kind_Interface_Variable_Declaration); + pragma Assert (Get_Mode (Base_Formal) in Iir_Out_Modes); + + Formal_Type := Get_Type (Formal); + Ftype_Info := Get_Info (Formal_Type); + Formal_Info := Get_Info (Base_Formal); + + -- Extract the value + if Base_Formal /= Formal then + -- By individual, copy back. + Param := Translate_Individual_Association_Formal + (Formal, Formal_Info, Params (Last_Individual)); + else + pragma Assert (Formal_Info.Interface_Field /= O_Fnode_Null); Ptr := New_Selected_Element - (New_Obj (Copy_Out), Formal_Info.Interface_Field); + (New_Obj (Params_Var), Formal_Info.Interface_Field); Param := Lv2M (Ptr, Ftype_Info, Mode_Value); - Chap7.Translate_Assign (Params (Pos), - Do_Conversion (Out_Conv, Formal, - Param), - Out_Expr, - Get_Type (Get_Actual (El)), El); - elsif Base_Formal /= Formal then - -- By individual. - -- Copy back. - Act := Get_Actual (El); - declare - Prev_Node : O_Dnode; - begin - Prev_Node := Formal_Info.Interface_Node; - -- We need a pointer since the interface is by reference. - Formal_Info.Interface_Node := - M2Dp (Params (Last_Individual)); - Val := Chap7.Translate_Expression - (Formal, Get_Type (Act)); - Formal_Info.Interface_Node := Prev_Node; - end; - Chap7.Translate_Assign - (Params (Pos), Val, Formal, Get_Type (Act), El); end if; + + Out_Conv := Get_Out_Conversion (El); + if Out_Conv = Null_Iir then + Out_Expr := Formal; + Val := M2E (Param); + else + Out_Expr := Out_Conv; + Val := Do_Conversion (Out_Conv, Formal, M2E (Param)); + end if; + + Chap7.Translate_Assign + (Params (Pos), Val, Out_Expr, Get_Type (Get_Actual (El)), El); end if; El := Get_Chain (El); Pos := Pos + 1; diff --git a/src/vhdl/translate/trans-chap9.adb b/src/vhdl/translate/trans-chap9.adb index 146bb818a..86faf6a3d 100644 --- a/src/vhdl/translate/trans-chap9.adb +++ b/src/vhdl/translate/trans-chap9.adb @@ -35,7 +35,6 @@ with Trans.Chap5; with Trans.Chap6; with Trans.Chap7; with Trans.Chap8; -with Trans.Chap14; with Trans.Rtis; with Translation; use Translation; with Trans_Decls; use Trans_Decls; @@ -1826,7 +1825,7 @@ package body Trans.Chap9 is if Data.Has_Val then Init_Val := M2E (Data.Val); else - Init_Val := Chap14.Translate_Left_Type_Attribute (Targ_Type); + Init_Val := Chap4.Get_Scalar_Initial_Value (Targ_Type); end if; New_Association (Assoc, New_Convert_Ov (Init_Val, Conv)); New_Procedure_Call (Assoc); diff --git a/src/vhdl/translate/trans.ads b/src/vhdl/translate/trans.ads index a97dcf706..4e778de1b 100644 --- a/src/vhdl/translate/trans.ads +++ b/src/vhdl/translate/trans.ads @@ -1104,19 +1104,17 @@ package Trans is -- given via an (hidden to the user) interface. Furthermore, -- the function is translated into a procedure. -- For a procedure: - -- If there are copy-out interfaces, they are gathered in a - -- record and a pointer to the record is passed to the - -- procedure. RES_INTERFACE is the interface for this pointer. + -- Interface for parameters. Res_Interface : O_Dnode := O_Dnode_Null; - -- Field in the frame for a pointer to the RESULT structure. - Res_Record_Var : Var_Type := Null_Var; + -- Field in the frame for a pointer to the PARAMS structure. This + -- is necessary when nested subprograms need to access to + -- paramters. of this subprogram. + Subprg_Params_Var : Var_Type := Null_Var; - -- For a procedure: record containing inout/out scalar parameters. - -- Type definition for the record. - Res_Record_Type : O_Tnode := O_Tnode_Null; - -- Type definition for access to the record. - Res_Record_Ptr : O_Tnode := O_Tnode_Null; + -- For a procedure, record containing the parameters. + Subprg_Params_Type : O_Tnode := O_Tnode_Null; + Subprg_Params_Ptr : O_Tnode := O_Tnode_Null; -- Access to the declarations within this subprogram. Subprg_Frame_Scope : aliased Var_Scope_Type; @@ -1168,7 +1166,7 @@ package Trans is -- Node: null, Field: null: not possible -- Node: null, Field: not null: field in RESULT record Interface_Node : O_Dnode := O_Dnode_Null; - -- Field of the result record for copy-out arguments of procedure. + -- Field of the PARAMS record for arguments of procedure. -- In that case, Interface_Node must be null. Interface_Field : O_Fnode; -- Type of the interface. |