summaryrefslogtreecommitdiff
path: root/gcc/ada/sem_prag.adb
diff options
context:
space:
mode:
authorcharlet <charlet@138bc75d-0d04-0410-961f-82ee72b054a4>2012-12-05 11:17:09 +0000
committercharlet <charlet@138bc75d-0d04-0410-961f-82ee72b054a4>2012-12-05 11:17:09 +0000
commit9f727ad2f2a5990d330e8848e12156e522827726 (patch)
treed1adb2b24cd145dae1facb856252c6df719f5c7f /gcc/ada/sem_prag.adb
parent8ae29f8f6a26064cbfcd01f7142e6601100c81c4 (diff)
downloadgcc-9f727ad2f2a5990d330e8848e12156e522827726.tar.gz
2012-12-05 Hristian Kirtchev <kirtchev@adacore.com>
* sem_prag.adb (Check_Loop_Invariant_Variant_Placement): When pragma Loop_[In]variant does not appear immediately within the statements of a loop, it must appear in a chain of nested blocks. 2012-12-05 Thomas Quinot <quinot@adacore.com> * sem_ch13.adb: Minor reformatting. Remove redundant assertion. git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@194213 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc/ada/sem_prag.adb')
-rw-r--r--gcc/ada/sem_prag.adb96
1 files changed, 74 insertions, 22 deletions
diff --git a/gcc/ada/sem_prag.adb b/gcc/ada/sem_prag.adb
index ddd84822ce1..be5afe028a6 100644
--- a/gcc/ada/sem_prag.adb
+++ b/gcc/ada/sem_prag.adb
@@ -620,7 +620,7 @@ package body Sem_Prag is
procedure Check_Loop_Invariant_Variant_Placement;
-- Verify whether pragma Loop_Invariant or pragma Loop_Variant appear
- -- immediately within the statements of the related loop.
+ -- immediately within a construct restricted to loops.
procedure Check_Is_In_Decl_Part_Or_Package_Spec;
-- Check that pragma appears in a declarative part, or in a package
@@ -1921,37 +1921,89 @@ package body Sem_Prag is
--------------------------------------------
procedure Check_Loop_Invariant_Variant_Placement is
- Loop_Stmt : Node_Id;
+ procedure Placement_Error (Constr : Node_Id);
+ -- Node Constr denotes the last loop restricted construct before we
+ -- encountered an illegal relation between enclosing constructs. Emit
+ -- an error depending on what Constr was.
+
+ ---------------------
+ -- Placement_Error --
+ ---------------------
+
+ procedure Placement_Error (Constr : Node_Id) is
+ begin
+ if Nkind (Constr) = N_Pragma then
+ Error_Pragma
+ ("pragma % must appear immediately within the statements " &
+ "of a loop");
+ else
+ Error_Pragma_Arg
+ ("block containing pragma % must appear immediately within " &
+ "the statements of a loop", Constr);
+ end if;
+ end Placement_Error;
+
+ -- Local declarations
+
+ Prev : Node_Id;
+ Stmt : Node_Id;
+
+ -- Start of processing for Check_Loop_Invariant_Variant_Placement
begin
- -- Locate the enclosing loop statement (if any)
+ Prev := N;
+ Stmt := Parent (N);
+ while Present (Stmt) loop
- Loop_Stmt := N;
- while Present (Loop_Stmt) loop
- if Nkind (Loop_Stmt) = N_Loop_Statement then
- exit;
+ -- The pragma or previous block must appear immediately within the
+ -- current block's declarative or statement part.
+
+ if Nkind (Stmt) = N_Block_Statement then
+ if (No (Declarations (Stmt))
+ or else List_Containing (Prev) /= Declarations (Stmt))
+ and then
+ List_Containing (Prev) /=
+ Statements (Handled_Statement_Sequence (Stmt))
+ then
+ Placement_Error (Prev);
+ return;
- -- Prevent the search from going too far
+ -- Keep inspecting the parents because we are now within a
+ -- chain of nested blocks.
+
+ else
+ Prev := Stmt;
+ Stmt := Parent (Stmt);
+ end if;
+
+ -- The pragma or previous block must appear immediately within the
+ -- statements of the loop.
+
+ elsif Nkind (Stmt) = N_Loop_Statement then
+ if List_Containing (Prev) /= Statements (Stmt) then
+ Placement_Error (Prev);
+ end if;
+
+ -- Stop the traversal because we reached the innermost loop
+ -- regardless of whether we encountered an error or not.
- elsif Nkind_In (Loop_Stmt, N_Entry_Body,
- N_Package_Body,
- N_Package_Declaration,
- N_Protected_Body,
- N_Subprogram_Body,
- N_Task_Body)
- then
- Error_Pragma ("pragma % must appear inside a loop statement");
return;
+ -- Ignore a handled statement sequence. Note that this node may
+ -- be related to a subprogram body in which case we will emit an
+ -- error on the next iteration of the search.
+
+ elsif Nkind (Stmt) = N_Handled_Sequence_Of_Statements then
+ Stmt := Parent (Stmt);
+
+ -- Any other statement breaks the chain from the pragma to the
+ -- loop.
+
else
- Loop_Stmt := Parent (Loop_Stmt);
+ Placement_Error (Prev);
+ return;
end if;
end loop;
-
- if List_Containing (N) /= Statements (Loop_Stmt) then
- Error_Pragma
- ("pragma % must occur immediately in the statements of a loop");
- end if;
end Check_Loop_Invariant_Variant_Placement;
-------------------------------------------