diff options
author | Eric Botcazou <ebotcazou@adacore.com> | 2022-12-02 10:55:49 +0100 |
---|---|---|
committer | Marc Poulhiès <poulhies@adacore.com> | 2022-12-06 14:58:48 +0100 |
commit | ea588d41f39428b0c3b02f016353dceb1aaaaa39 (patch) | |
tree | ce91f5abdcd15583a7e4e5f4ed70e44022c10d07 /gcc/ada/exp_ch4.adb | |
parent | a444c05623faa2b6bd1bbc8f7908b8ea3d83b475 (diff) | |
download | gcc-ea588d41f39428b0c3b02f016353dceb1aaaaa39.zip gcc-ea588d41f39428b0c3b02f016353dceb1aaaaa39.tar.gz gcc-ea588d41f39428b0c3b02f016353dceb1aaaaa39.tar.bz2 |
ada: Elide the copy in extended returns for nonlimited by-reference types
This implements elision of the copy operation for extended return statements
in the case of nonlimited by-reference types (the copy operation is already
elided for limited types by the front-end and nonlimited non-by-reference
types by the code generator), which comprise controlled and tagged types.
The implementation partly reuses the machinery implemented for limited types
(the build-in-place machinery) to allocate the return object directly on the
primary or the secondary stack, depending on whether the result type of the
function is constrained or not.
This requires further special-casing for the allocators generated by this
machinery as well as an adjustment to the implementation of a specific case
of string concatenation.
gcc/ada/
* einfo.ads (Actual_Subtype): Document additional usage.
* exp_aggr.adb (Expand_Array_Aggregate): Replace test on
Is_Build_In_Place_Return_Object with Is_Special_Return_Object.
* exp_ch3.adb (Expand_N_Object_Declaration): Factor out parts of the
processing done for build-in-place return objects and reuse them to
implement a similar processing for specific return objects.
* exp_ch4.adb (Expand_Allocator_Expression): Do not generate a tag
assignment or an adjustment if the allocator was made for a special
return object.
(Expand_Concatenate): If the result is allocated on the secondary
stack, use an unconstrained allocation.
* exp_ch6.ads (Apply_CW_Accessibility_Check): New declaration.
(Is_By_Reference_Return_Object): Likewise.
(Is_Secondary_Stack_Return_Object): Likewise.
(Is_Special_Return_Object): Likewise.
* exp_ch6.adb (Expand_Ctrl_Function_Call): Do not bail out for the
expression in the declaration of a special return object.
(Expand_N_Extended_Return_Statement): Add missing guard and move
the class-wide accessibility check to Expand_N_Object_Declaration.
(Expand_Simple_Function_Return): Delete obsolete commentary.
Skip the special processing for types that require finalization or
are returned on the secondary stack if the return originally comes
from an extended return statement. Add missing Constant_Present.
(Is_By_Reference_Return_Object): New predicate.
(Is_Secondary_Stack_Return_Object): Likewise.
(Is_Special_Return_Object): Likewise.
* exp_util.adb (Is_Related_To_Func_Return): Also return true if the
parent of the expression is the renaming declaration generated for
the expansion of a return object.
* gen_il-fields.ads (Opt_Field_Enum): Replace Alloc_For_BIP_Return
with For_Special_Return_Object.
* gen_il-gen-gen_nodes.adb (N_Allocator): Likewise.
* gen_il-internals.adb (Image): Remove Alloc_For_BIP_Return.
* sem_ch3.adb (Check_Return_Subtype_Indication): New procedure
moved from sem_ch6.adb.
(Analyze_Object_Declaration): Call it on a return object.
* sem_ch4.adb: Add with and use clauses for Rtsfind.
(Analyze_Allocator): Test For_Special_Return_Object to skip checks
for allocators made for special return objects.
Do not report restriction violations for the return stack pool.
* sem_ch5.adb (Analyze_Assignment.Set_Assignment_Type): Return the
Actual_Subtype for return objects that live on the secondary stack.
* sem_ch6.adb (Check_Return_Subtype_Indication): Move procedure to
sem_ch3.adb.
(Analyze_Function_Return): Do not call above procedure.
* sem_res.adb (Resolve_Allocator): Replace Alloc_For_BIP_Return
with For_Special_Return_Object.
* sinfo.ads: Likewise.
* treepr.adb (Image): Remove Alloc_For_BIP_Return.
* gcc-interface/trans.cc (gnat_to_gnu): Do not convert to the result
type in the unconstrained array type case if the parent is a simple
return statement.
Diffstat (limited to 'gcc/ada/exp_ch4.adb')
-rw-r--r-- | gcc/ada/exp_ch4.adb | 126 |
1 files changed, 73 insertions, 53 deletions
diff --git a/gcc/ada/exp_ch4.adb b/gcc/ada/exp_ch4.adb index 7edef4c..a8980a6 100644 --- a/gcc/ada/exp_ch4.adb +++ b/gcc/ada/exp_ch4.adb @@ -898,6 +898,11 @@ package body Exp_Ch4 is (Directly_Designated_Type (Etype (N)))); null; + -- Likewise if the allocator is made for a special return object + + elsif For_Special_Return_Object (N) then + null; + elsif Is_Tagged_Type (T) and then not Is_Class_Wide_Type (T) then TagT := T; TagR := @@ -946,19 +951,18 @@ package body Exp_Ch4 is -- Adjust procedure, and the object is built in place. In Ada 95, the -- object can be limited but not inherently limited if this allocator -- came from a return statement (we're allocating the result on the - -- secondary stack). In that case, the object will be moved, so we do - -- want to Adjust. However, if it's a nonlimited build-in-place - -- function call, Adjust is not wanted. - -- - -- Needs_Finalization (DesigT) can differ from Needs_Finalization (T) + -- secondary stack); in that case, the object will be moved, so we do + -- want to Adjust. But the call is always skipped if the allocator is + -- made for a special return object because it's generated elsewhere. + + -- Needs_Finalization (DesigT) may differ from Needs_Finalization (T) -- if one of the two types is class-wide, and the other is not. if Needs_Finalization (DesigT) and then Needs_Finalization (T) and then not Aggr_In_Place and then not Is_Limited_View (T) - and then not Alloc_For_BIP_Return (N) - and then not Is_Build_In_Place_Function_Call (Expression (N)) + and then not For_Special_Return_Object (N) then -- An unchecked conversion is needed in the classwide case because -- the designated type can be an ancestor of the subtype mark of @@ -2724,6 +2728,7 @@ package body Exp_Ch4 is Len : Unat; J : Nat; Clen : Node_Id; + Decl : Node_Id; Set : Boolean; -- Start of processing for Expand_Concatenate @@ -3250,10 +3255,32 @@ package body Exp_Ch4 is Set_Is_Internal (Ent); Set_Debug_Info_Needed (Ent); + -- If the bound is statically known to be out of range, we do not want + -- to abort, we want a warning and a constraint error at run time. Note + -- that we have arranged that the result will not be treated as a static + -- constant, so we won't get an illegality during the insertion. We also + -- enable all checks (in particular range checks) in case the bounds of + -- Subtyp_Ind are out of range. + + Decl := + Make_Object_Declaration (Loc, + Defining_Identifier => Ent, + Object_Definition => Subtyp_Ind); + Insert_Action (Cnode, Decl); + + -- If the result of the concatenation appears as the initializing + -- expression of an object declaration, we can just rename the + -- result, rather than copying it. + + Set_OK_To_Rename (Ent); + -- If we are concatenating strings and the current scope already uses - -- the secondary stack, allocate the resulting string also on the - -- secondary stack to avoid putting too much pressure on the primary - -- stack. + -- the secondary stack, allocate the result also on the secondary stack + -- to avoid putting too much pressure on the primary stack. + + -- We use an unconstrained allocation, i.e. we also allocate the bounds, + -- so that the result can be renamed in all contexts. + -- Don't do this if -gnatd.h is set, as this will break the wrapping of -- Cnode in an Expression_With_Actions, see Expand_N_Op_Concat. @@ -3263,84 +3290,77 @@ package body Exp_Ch4 is and then not Debug_Flag_Dot_H then -- Generate: - -- subtype Axx is ...; - -- type Ayy is access Axx; - -- Rxx : Ayy := new <subtype> [storage_pool = ss_pool]; - -- Sxx : <subtype> renames Rxx.all; + -- subtype Axx is String (<low-bound> .. <high-bound>) + -- type Ayy is access String; + -- Rxx : Ayy := new <Axx> [storage_pool = ss_pool]; + -- Sxx : String renames Rxx.all; declare - Alloc : Node_Id; ConstrT : constant Entity_Id := Make_Temporary (Loc, 'A'); Acc_Typ : constant Entity_Id := Make_Temporary (Loc, 'A'); + + Alloc : Node_Id; + Deref : Node_Id; Temp : Entity_Id; begin - Insert_Action (Cnode, + Insert_Action (Decl, Make_Subtype_Declaration (Loc, Defining_Identifier => ConstrT, Subtype_Indication => Subtyp_Ind), Suppress => All_Checks); - Freeze_Itype (ConstrT, Cnode); - Insert_Action (Cnode, + Freeze_Itype (ConstrT, Decl); + + Insert_Action (Decl, Make_Full_Type_Declaration (Loc, Defining_Identifier => Acc_Typ, Type_Definition => Make_Access_To_Object_Definition (Loc, - Subtype_Indication => New_Occurrence_Of (ConstrT, Loc))), + Subtype_Indication => New_Occurrence_Of (Atyp, Loc))), Suppress => All_Checks); + + Mutate_Ekind (Acc_Typ, E_Access_Type); + Set_Associated_Storage_Pool (Acc_Typ, RTE (RE_SS_Pool)); + Alloc := Make_Allocator (Loc, Expression => New_Occurrence_Of (ConstrT, Loc)); - -- Allocate on the secondary stack. This is currently done - -- only for type String, which normally doesn't have default - -- initialization, but we need to Set_No_Initialization in case - -- of Initialize_Scalars or Normalize_Scalars; otherwise, the - -- allocator will get transformed and will not use the secondary - -- stack. + -- This is currently done only for type String, which normally + -- doesn't have default initialization, but we need to set the + -- No_Initialization flag in case of either Initialize_Scalars + -- or Normalize_Scalars. - Set_Storage_Pool (Alloc, RTE (RE_SS_Pool)); - Set_Procedure_To_Call (Alloc, RTE (RE_SS_Allocate)); Set_No_Initialization (Alloc); Temp := Make_Temporary (Loc, 'R', Alloc); - Insert_Action (Cnode, + Insert_Action (Decl, Make_Object_Declaration (Loc, Defining_Identifier => Temp, Object_Definition => New_Occurrence_Of (Acc_Typ, Loc), Expression => Alloc), Suppress => All_Checks); - Insert_Action (Cnode, + Deref := + Make_Explicit_Dereference (Loc, + Prefix => New_Occurrence_Of (Temp, Loc)); + Set_Etype (Deref, Atyp); + + Rewrite (Decl, Make_Object_Renaming_Declaration (Loc, Defining_Identifier => Ent, - Subtype_Mark => New_Occurrence_Of (ConstrT, Loc), - Name => - Make_Explicit_Dereference (Loc, - Prefix => New_Occurrence_Of (Temp, Loc))), - Suppress => All_Checks); - end; - else - -- If the bound is statically known to be out of range, we do not - -- want to abort, we want a warning and a runtime constraint error. - -- Note that we have arranged that the result will not be treated as - -- a static constant, so we won't get an illegality during this - -- insertion. - -- We also enable checks (in particular range checks) in case the - -- bounds of Subtyp_Ind are out of range. - - Insert_Action (Cnode, - Make_Object_Declaration (Loc, - Defining_Identifier => Ent, - Object_Definition => Subtyp_Ind)); - end if; + Subtype_Mark => New_Occurrence_Of (Atyp, Loc), + Name => Deref)); - -- If the result of the concatenation appears as the initializing - -- expression of an object declaration, we can just rename the - -- result, rather than copying it. + -- We do not analyze this renaming declaration because this would + -- change the subtype of Ent back to a constrained string. - Set_OK_To_Rename (Ent); + Set_Etype (Ent, Atyp); + Set_Renamed_Object (Ent, Deref); + Set_Analyzed (Decl); + end; + end if; -- Catch the static out of range case now |