diff options
author | Justin Squirek <squirek@adacore.com> | 2020-11-16 09:08:51 -0500 |
---|---|---|
committer | Pierre-Marie de Rodat <derodat@adacore.com> | 2020-12-15 06:41:56 -0500 |
commit | a1023434a8dd3ce2281a726d30ef370caa425252 (patch) | |
tree | 96879e168c070a781a22b5670da318bf9bf47143 /gcc/ada/contracts.adb | |
parent | 43852482cafa73da6408120722dcbe7ff6fd3ded (diff) | |
download | gcc-a1023434a8dd3ce2281a726d30ef370caa425252.zip gcc-a1023434a8dd3ce2281a726d30ef370caa425252.tar.gz gcc-a1023434a8dd3ce2281a726d30ef370caa425252.tar.bz2 |
[Ada] Postcondition checks performed before finalization
gcc/ada/
* contracts.adb, contracts.ads (Build_Postconditions_Procedure):
Add declarations for Postcond_Enabled,
Result_Object_For_Postcondition, and
Return_Success_For_Postcond, and place all postconditions within
an if statement to control their execution for interactions when
cleanup actions get generated.
(Get_Postcond_Enabled): Created to fetch object declared to
handle new expansion of postconditions.
(Get_Result_Object_For_Postcond): Created to fetch object
declared to handle new expansion of postconditions.
(Get_Return_Success_For_Postcond): Created to fetch object
declared to handle new expansion of postconditions.
* einfo.adb, einfo.ads: Modify flag Stores_Attribute_Old_Prefix
to apply to constants, variables, and types.
* exp_ch6.adb (Add_Return): Add assignment to
Return_Success_For_Postcond.
(Expand_Non_Function_Return): Add assignment to
Return_Success_For_Postcond
(Expand_Simple_Function_Return): Add assignment to
Result_Object_For_Postcond and Return_Success_For_Postcond.
* exp_ch7.adb (Build_Finalization_Master): Mark finalization
masters which finalize types created store 'Old objects as
storing 'Old objects.
(Build_Finalizer): Created to generated a unified and special
expansion for finalization when postconditions are present.
(Build_Finalizer_Helper): Renamed Build_Finalizer and added
parameter to facilitate the creation of separate finalization
routines for 'Old objects and general objects.
(Create_Finalizer): Add condition for the insertion of the
finalizer spec to avoid malformed trees.
(Expand_Cleanup_Actions): Move _postconditions and related
declarations to the new declarative section. Fix the loop to
properly stop at the subprogram declaration for the
postconditions procedure and exclude its body from being moved
to the new list of declarations to avoid freezing issues.
* exp_prag.adb (Expand_Attributes): Mark temporary created to
store 'Old objects as storing a 'Old attribute.
* sem_ch6.adb (Find_What_Applies_To): Remove strange exception
to postconditions when traversing the scope stack.
* sem_prag.adb (Find_Related_Declaration_Or_Body): Use the newly
created Enclosing_HSS function to find the HSS for a potentially
nested statement.
* sem_util.adb, sem_util.ads (Declare_Indirect_Temp): Mark types
created to store 'Old objects as storing 'Old attributes.
(Enclosing_HSS): Created to find the enclosing handled sequence
of statements for a given statement.
* snames.ads-tmpl: Add multiple names to aid in the expansion of
finalization and to control the evaluation of postconditions.
Including _finalization_controller, a new routine to centralize
finalization actions and postcondition evaluation.
Diffstat (limited to 'gcc/ada/contracts.adb')
-rw-r--r-- | gcc/ada/contracts.adb | 241 |
1 files changed, 237 insertions, 4 deletions
diff --git a/gcc/ada/contracts.adb b/gcc/ada/contracts.adb index 7387ffe..29557ec 100644 --- a/gcc/ada/contracts.adb +++ b/gcc/ada/contracts.adb @@ -2198,12 +2198,24 @@ package body Contracts is Result : Entity_Id) is Loc : constant Source_Ptr := Sloc (Body_Decl); + Last_Decl : Node_Id; Params : List_Id := No_List; Proc_Bod : Node_Id; Proc_Decl : Node_Id; Proc_Id : Entity_Id; Proc_Spec : Node_Id; + -- Extra declarations needed to handle interactions between + -- postconditions and finalization. + + Postcond_Enabled_Decl : Node_Id; + Return_Success_Decl : Node_Id; + Result_Obj_Decl : Node_Id; + Result_Obj_Type_Decl : Node_Id; + Result_Obj_Type : Entity_Id; + + -- Start of processing for Build_Postconditions_Procedure + begin -- Nothing to do if there are no actions to check on exit @@ -2211,6 +2223,29 @@ package body Contracts is return; end if; + -- Otherwise, we generate the postcondition procedure and add + -- associated objects and conditions used to coordinate postcondition + -- evaluation with finalization. + + -- Generate: + -- + -- procedure _postconditions (Return_Exp : Result_Typ); + -- + -- -- Result_Obj_Type created when Result_Type is non-elementary + -- [type Result_Obj_Type is access all Result_Typ;] + -- + -- Result_Obj : Result_Obj_Type; + -- + -- Postcond_Enabled : Boolean := True; + -- Return_Success_For_Postcond : Boolean := False; + -- + -- procedure _postconditions (Return_Exp : Result_Typ) is + -- begin + -- if Postcond_Enabled and then Return_Success_For_Postcond then + -- [stmts]; + -- end if; + -- end; + Proc_Id := Make_Defining_Identifier (Loc, Name_uPostconditions); Set_Debug_Info_Needed (Proc_Id); Set_Postconditions_Proc (Subp_Id, Proc_Id); @@ -2248,12 +2283,14 @@ package body Contracts is -- body. This ensures that the body will not cause any premature -- freezing, as it may mention types: + -- Generate: + -- -- procedure Proc (Obj : Array_Typ) is -- procedure _postconditions is -- begin -- ... Obj ... -- end _postconditions; - + -- -- subtype T is Array_Typ (Obj'First (1) .. Obj'Last (1)); -- begin @@ -2265,12 +2302,121 @@ package body Contracts is Insert_Before_First_Source_Declaration (Proc_Decl, Declarations (Body_Decl)); Analyze (Proc_Decl); + Last_Decl := Proc_Decl; + + -- When Result is present (e.g. the postcondition checks apply to a + -- function) we make a local object to capture the result, so, if + -- needed, we can call the generated postconditions procedure during + -- finalization instead of at the point of return. + + -- Note: The placement of the following declarations before the + -- declaration of the body of the postconditions, but after the + -- declaration of the postconditions spec is deliberate and required + -- since other code within the expander expects them to be located + -- here. Perhaps when more space is available in the tree this will + -- no longer be necessary ??? + + if Present (Result) then + -- Elementary result types mean a copy is cheap and preferred over + -- using pointers. + + if Is_Elementary_Type (Etype (Result)) then + Result_Obj_Type := Etype (Result); + + -- Otherwise, we create a named access type to capture the result + + -- Generate: + -- + -- type Result_Obj_Type is access all [Result_Type]; + + else + Result_Obj_Type := Make_Temporary (Loc, 'R'); + + Result_Obj_Type_Decl := + Make_Full_Type_Declaration (Loc, + Defining_Identifier => Result_Obj_Type, + Type_Definition => + Make_Access_To_Object_Definition (Loc, + All_Present => True, + Subtype_Indication => New_Occurrence_Of + (Etype (Result), Loc))); + Insert_After_And_Analyze (Proc_Decl, Result_Obj_Type_Decl); + Last_Decl := Result_Obj_Type_Decl; + end if; + + -- Create the result obj declaration + + -- Generate: + -- + -- Result_Object_For_Postcond : Result_Obj_Type; + + Result_Obj_Decl := + Make_Object_Declaration (Loc, + Defining_Identifier => + Make_Defining_Identifier + (Loc, Name_uResult_Object_For_Postcond), + Object_Definition => + New_Occurrence_Of + (Result_Obj_Type, Loc)); + Set_No_Initialization (Result_Obj_Decl); + Insert_After_And_Analyze (Last_Decl, Result_Obj_Decl); + Last_Decl := Result_Obj_Decl; + end if; + + -- Build the Postcond_Enabled flag used to delay evaluation of + -- postconditions until finalization has been performed when cleanup + -- actions are present. + + -- Generate: + -- + -- Postcond_Enabled : Boolean := True; + + Postcond_Enabled_Decl := + Make_Object_Declaration (Loc, + Defining_Identifier => + Make_Defining_Identifier + (Loc, Name_uPostcond_Enabled), + Object_Definition => New_Occurrence_Of (Standard_Boolean, Loc), + Expression => New_Occurrence_Of (Standard_True, Loc)); + Insert_After_And_Analyze (Last_Decl, Postcond_Enabled_Decl); + Last_Decl := Postcond_Enabled_Decl; + + -- Create a flag to indicate that return has been reached + + -- This is necessary for deciding whether to execute _postconditions + -- during finalization. + + -- Generate: + -- + -- Return_Success_For_Postcond : Boolean := False; + + Return_Success_Decl := + Make_Object_Declaration (Loc, + Defining_Identifier => + Make_Defining_Identifier + (Loc, Name_uReturn_Success_For_Postcond), + Object_Definition => New_Occurrence_Of (Standard_Boolean, Loc), + Expression => New_Occurrence_Of (Standard_False, Loc)); + Insert_After_And_Analyze (Last_Decl, Return_Success_Decl); + Last_Decl := Return_Success_Decl; -- Set an explicit End_Label to override the sloc of the implicit -- RETURN statement, and prevent it from inheriting the sloc of one -- the postconditions: this would cause confusing debug info to be -- produced, interfering with coverage-analysis tools. + -- Also, wrap the postcondition checks in a conditional which can be + -- used to delay their evaluation when clean-up actions are present. + + -- Generate: + -- + -- procedure _postconditions is + -- begin + -- if Postcond_Enabled and then Return_Success_For_Postcond then + -- [Stmts]; + -- end if; + -- end; + Proc_Bod := Make_Subprogram_Body (Loc, Specification => @@ -2278,10 +2424,22 @@ package body Contracts is Declarations => Empty_List, Handled_Statement_Sequence => Make_Handled_Sequence_Of_Statements (Loc, - Statements => Stmts, - End_Label => Make_Identifier (Loc, Chars (Proc_Id)))); + End_Label => Make_Identifier (Loc, Chars (Proc_Id)), + Statements => New_List ( + Make_If_Statement (Loc, + Condition => + Make_And_Then (Loc, + Left_Opnd => + New_Occurrence_Of + (Defining_Identifier + (Postcond_Enabled_Decl), Loc), + Right_Opnd => + New_Occurrence_Of + (Defining_Identifier + (Return_Success_Decl), Loc)), + Then_Statements => Stmts)))); + Insert_After_And_Analyze (Last_Decl, Proc_Bod); - Insert_After_And_Analyze (Proc_Decl, Proc_Bod); end Build_Postconditions_Procedure; ---------------------------- @@ -3280,6 +3438,81 @@ package body Contracts is Freeze_Contracts; end Freeze_Previous_Contracts; + -------------------------- + -- Get_Postcond_Enabled -- + -------------------------- + + function Get_Postcond_Enabled (Subp : Entity_Id) return Node_Id is + Decl : Node_Id; + begin + Decl := + Next (Unit_Declaration_Node (Postconditions_Proc (Subp))); + while Present (Decl) loop + + if Nkind (Decl) = N_Object_Declaration + and then Chars (Defining_Identifier (Decl)) + = Name_uPostcond_Enabled + then + return Defining_Identifier (Decl); + end if; + + Next (Decl); + end loop; + + return Empty; + end Get_Postcond_Enabled; + + ------------------------------------ + -- Get_Result_Object_For_Postcond -- + ------------------------------------ + + function Get_Result_Object_For_Postcond + (Subp : Entity_Id) return Node_Id + is + Decl : Node_Id; + begin + Decl := + Next (Unit_Declaration_Node (Postconditions_Proc (Subp))); + while Present (Decl) loop + + if Nkind (Decl) = N_Object_Declaration + and then Chars (Defining_Identifier (Decl)) + = Name_uResult_Object_For_Postcond + then + return Defining_Identifier (Decl); + end if; + + Next (Decl); + end loop; + + return Empty; + end Get_Result_Object_For_Postcond; + + ------------------------------------- + -- Get_Return_Success_For_Postcond -- + ------------------------------------- + + function Get_Return_Success_For_Postcond (Subp : Entity_Id) return Node_Id + is + Decl : Node_Id; + begin + Decl := + Next (Unit_Declaration_Node (Postconditions_Proc (Subp))); + while Present (Decl) loop + + if Nkind (Decl) = N_Object_Declaration + and then Chars (Defining_Identifier (Decl)) + = Name_uReturn_Success_For_Postcond + then + return Defining_Identifier (Decl); + end if; + + Next (Decl); + end loop; + + return Empty; + end Get_Return_Success_For_Postcond; + --------------------------------- -- Inherit_Subprogram_Contract -- --------------------------------- |