annotate gcc/ada/par_sco.adb @ 131:84e7813d76e9

gcc-8.2
author mir3636
date Thu, 25 Oct 2018 07:37:49 +0900
parents 04ced10e8804
children 1830386684a0
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 ------------------------------------------------------------------------------
kono
parents:
diff changeset
2 -- --
kono
parents:
diff changeset
3 -- GNAT COMPILER COMPONENTS --
kono
parents:
diff changeset
4 -- --
kono
parents:
diff changeset
5 -- P A R _ S C O --
kono
parents:
diff changeset
6 -- --
kono
parents:
diff changeset
7 -- B o d y --
kono
parents:
diff changeset
8 -- --
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
9 -- Copyright (C) 2009-2018, Free Software Foundation, Inc. --
111
kono
parents:
diff changeset
10 -- --
kono
parents:
diff changeset
11 -- GNAT is free software; you can redistribute it and/or modify it under --
kono
parents:
diff changeset
12 -- terms of the GNU General Public License as published by the Free Soft- --
kono
parents:
diff changeset
13 -- ware Foundation; either version 3, or (at your option) any later ver- --
kono
parents:
diff changeset
14 -- sion. GNAT is distributed in the hope that it will be useful, but WITH- --
kono
parents:
diff changeset
15 -- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY --
kono
parents:
diff changeset
16 -- or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License --
kono
parents:
diff changeset
17 -- for more details. You should have received a copy of the GNU General --
kono
parents:
diff changeset
18 -- Public License distributed with GNAT; see file COPYING3. If not, go to --
kono
parents:
diff changeset
19 -- http://www.gnu.org/licenses for a complete copy of the license. --
kono
parents:
diff changeset
20 -- --
kono
parents:
diff changeset
21 -- GNAT was originally developed by the GNAT team at New York University. --
kono
parents:
diff changeset
22 -- Extensive contributions were provided by Ada Core Technologies Inc. --
kono
parents:
diff changeset
23 -- --
kono
parents:
diff changeset
24 ------------------------------------------------------------------------------
kono
parents:
diff changeset
25
kono
parents:
diff changeset
26 with Aspects; use Aspects;
kono
parents:
diff changeset
27 with Atree; use Atree;
kono
parents:
diff changeset
28 with Debug; use Debug;
kono
parents:
diff changeset
29 with Errout; use Errout;
kono
parents:
diff changeset
30 with Lib; use Lib;
kono
parents:
diff changeset
31 with Lib.Util; use Lib.Util;
kono
parents:
diff changeset
32 with Namet; use Namet;
kono
parents:
diff changeset
33 with Nlists; use Nlists;
kono
parents:
diff changeset
34 with Opt; use Opt;
kono
parents:
diff changeset
35 with Output; use Output;
kono
parents:
diff changeset
36 with Put_SCOs;
kono
parents:
diff changeset
37 with SCOs; use SCOs;
kono
parents:
diff changeset
38 with Sem; use Sem;
kono
parents:
diff changeset
39 with Sem_Util; use Sem_Util;
kono
parents:
diff changeset
40 with Sinfo; use Sinfo;
kono
parents:
diff changeset
41 with Sinput; use Sinput;
kono
parents:
diff changeset
42 with Snames; use Snames;
kono
parents:
diff changeset
43 with Table;
kono
parents:
diff changeset
44
kono
parents:
diff changeset
45 with GNAT.HTable; use GNAT.HTable;
kono
parents:
diff changeset
46 with GNAT.Heap_Sort_G;
kono
parents:
diff changeset
47
kono
parents:
diff changeset
48 package body Par_SCO is
kono
parents:
diff changeset
49
kono
parents:
diff changeset
50 --------------------------
kono
parents:
diff changeset
51 -- First-pass SCO table --
kono
parents:
diff changeset
52 --------------------------
kono
parents:
diff changeset
53
kono
parents:
diff changeset
54 -- The Short_Circuit_And_Or pragma enables one to use AND and OR operators
kono
parents:
diff changeset
55 -- in source code while the ones used with booleans will be interpreted as
kono
parents:
diff changeset
56 -- their short circuit alternatives (AND THEN and OR ELSE). Thus, the true
kono
parents:
diff changeset
57 -- meaning of these operators is known only after the semantic analysis.
kono
parents:
diff changeset
58
kono
parents:
diff changeset
59 -- However, decision SCOs include short circuit operators only. The SCO
kono
parents:
diff changeset
60 -- information generation pass must be done before expansion, hence before
kono
parents:
diff changeset
61 -- the semantic analysis. Because of this, the SCO information generation
kono
parents:
diff changeset
62 -- is done in two passes.
kono
parents:
diff changeset
63
kono
parents:
diff changeset
64 -- The first one (SCO_Record_Raw, before semantic analysis) completes the
kono
parents:
diff changeset
65 -- SCO_Raw_Table assuming all AND/OR operators are short circuit ones.
kono
parents:
diff changeset
66 -- Then, the semantic analysis determines which operators are promoted to
kono
parents:
diff changeset
67 -- short circuit ones. Finally, the second pass (SCO_Record_Filtered)
kono
parents:
diff changeset
68 -- translates the SCO_Raw_Table to SCO_Table, taking care of removing the
kono
parents:
diff changeset
69 -- remaining AND/OR operators and of adjusting decisions accordingly
kono
parents:
diff changeset
70 -- (splitting decisions, removing empty ones, etc.).
kono
parents:
diff changeset
71
kono
parents:
diff changeset
72 type SCO_Generation_State_Type is (None, Raw, Filtered);
kono
parents:
diff changeset
73 SCO_Generation_State : SCO_Generation_State_Type := None;
kono
parents:
diff changeset
74 -- Keep track of the SCO generation state: this will prevent us from
kono
parents:
diff changeset
75 -- running some steps multiple times (the second pass has to be started
kono
parents:
diff changeset
76 -- from multiple places).
kono
parents:
diff changeset
77
kono
parents:
diff changeset
78 package SCO_Raw_Table is new Table.Table
kono
parents:
diff changeset
79 (Table_Component_Type => SCO_Table_Entry,
kono
parents:
diff changeset
80 Table_Index_Type => Nat,
kono
parents:
diff changeset
81 Table_Low_Bound => 1,
kono
parents:
diff changeset
82 Table_Initial => 500,
kono
parents:
diff changeset
83 Table_Increment => 300,
kono
parents:
diff changeset
84 Table_Name => "Raw_Table");
kono
parents:
diff changeset
85
kono
parents:
diff changeset
86 -----------------------
kono
parents:
diff changeset
87 -- Unit Number Table --
kono
parents:
diff changeset
88 -----------------------
kono
parents:
diff changeset
89
kono
parents:
diff changeset
90 -- This table parallels the SCO_Unit_Table, keeping track of the unit
kono
parents:
diff changeset
91 -- numbers corresponding to the entries made in this table, so that before
kono
parents:
diff changeset
92 -- writing out the SCO information to the ALI file, we can fill in the
kono
parents:
diff changeset
93 -- proper dependency numbers and file names.
kono
parents:
diff changeset
94
kono
parents:
diff changeset
95 -- Note that the zero'th entry is here for convenience in sorting the
kono
parents:
diff changeset
96 -- table, the real lower bound is 1.
kono
parents:
diff changeset
97
kono
parents:
diff changeset
98 package SCO_Unit_Number_Table is new Table.Table
kono
parents:
diff changeset
99 (Table_Component_Type => Unit_Number_Type,
kono
parents:
diff changeset
100 Table_Index_Type => SCO_Unit_Index,
kono
parents:
diff changeset
101 Table_Low_Bound => 0, -- see note above on sort
kono
parents:
diff changeset
102 Table_Initial => 20,
kono
parents:
diff changeset
103 Table_Increment => 200,
kono
parents:
diff changeset
104 Table_Name => "SCO_Unit_Number_Entry");
kono
parents:
diff changeset
105
kono
parents:
diff changeset
106 ------------------------------------------
kono
parents:
diff changeset
107 -- Condition/Operator/Pragma Hash Table --
kono
parents:
diff changeset
108 ------------------------------------------
kono
parents:
diff changeset
109
kono
parents:
diff changeset
110 -- We need to be able to get to conditions quickly for handling the calls
kono
parents:
diff changeset
111 -- to Set_SCO_Condition efficiently, and similarly to get to pragmas to
kono
parents:
diff changeset
112 -- handle calls to Set_SCO_Pragma_Enabled (the same holds for operators and
kono
parents:
diff changeset
113 -- Set_SCO_Logical_Operator). For this purpose we identify the conditions,
kono
parents:
diff changeset
114 -- operators and pragmas in the table by their starting sloc, and use this
kono
parents:
diff changeset
115 -- hash table to map from these sloc values to SCO_Table indexes.
kono
parents:
diff changeset
116
kono
parents:
diff changeset
117 type Header_Num is new Integer range 0 .. 996;
kono
parents:
diff changeset
118 -- Type for hash table headers
kono
parents:
diff changeset
119
kono
parents:
diff changeset
120 function Hash (F : Source_Ptr) return Header_Num;
kono
parents:
diff changeset
121 -- Function to Hash source pointer value
kono
parents:
diff changeset
122
kono
parents:
diff changeset
123 function Equal (F1 : Source_Ptr; F2 : Source_Ptr) return Boolean;
kono
parents:
diff changeset
124 -- Function to test two keys for equality
kono
parents:
diff changeset
125
kono
parents:
diff changeset
126 function "<" (S1 : Source_Location; S2 : Source_Location) return Boolean;
kono
parents:
diff changeset
127 -- Function to test for source locations order
kono
parents:
diff changeset
128
kono
parents:
diff changeset
129 package SCO_Raw_Hash_Table is new Simple_HTable
kono
parents:
diff changeset
130 (Header_Num, Int, 0, Source_Ptr, Hash, Equal);
kono
parents:
diff changeset
131 -- The actual hash table
kono
parents:
diff changeset
132
kono
parents:
diff changeset
133 --------------------------
kono
parents:
diff changeset
134 -- Internal Subprograms --
kono
parents:
diff changeset
135 --------------------------
kono
parents:
diff changeset
136
kono
parents:
diff changeset
137 function Has_Decision (N : Node_Id) return Boolean;
kono
parents:
diff changeset
138 -- N is the node for a subexpression. Returns True if the subexpression
kono
parents:
diff changeset
139 -- contains a nested decision (i.e. either is a logical operator, or
kono
parents:
diff changeset
140 -- contains a logical operator in its subtree).
kono
parents:
diff changeset
141 --
kono
parents:
diff changeset
142 -- This must be used in the first pass (SCO_Record_Raw) only: here AND/OR
kono
parents:
diff changeset
143 -- operators are considered as short circuit, just in case the
kono
parents:
diff changeset
144 -- Short_Circuit_And_Or pragma is used: only real short circuit operations
kono
parents:
diff changeset
145 -- will be kept in the secord pass.
kono
parents:
diff changeset
146
kono
parents:
diff changeset
147 type Tristate is (False, True, Unknown);
kono
parents:
diff changeset
148
kono
parents:
diff changeset
149 function Is_Logical_Operator (N : Node_Id) return Tristate;
kono
parents:
diff changeset
150 -- N is the node for a subexpression. This procedure determines whether N
kono
parents:
diff changeset
151 -- is a logical operator: True for short circuit conditions, Unknown for OR
kono
parents:
diff changeset
152 -- and AND (the Short_Circuit_And_Or pragma may be used) and False
kono
parents:
diff changeset
153 -- otherwise. Note that in cases where True is returned, callers assume
kono
parents:
diff changeset
154 -- Nkind (N) in N_Op.
kono
parents:
diff changeset
155
kono
parents:
diff changeset
156 function To_Source_Location (S : Source_Ptr) return Source_Location;
kono
parents:
diff changeset
157 -- Converts Source_Ptr value to Source_Location (line/col) format
kono
parents:
diff changeset
158
kono
parents:
diff changeset
159 procedure Process_Decisions
kono
parents:
diff changeset
160 (N : Node_Id;
kono
parents:
diff changeset
161 T : Character;
kono
parents:
diff changeset
162 Pragma_Sloc : Source_Ptr);
kono
parents:
diff changeset
163 -- If N is Empty, has no effect. Otherwise scans the tree for the node N,
kono
parents:
diff changeset
164 -- to output any decisions it contains. T is one of IEGPWX (for context of
kono
parents:
diff changeset
165 -- expression: if/exit when/entry guard/pragma/while/expression). If T is
kono
parents:
diff changeset
166 -- other than X, the node N is the if expression involved, and a decision
kono
parents:
diff changeset
167 -- is always present (at the very least a simple decision is present at the
kono
parents:
diff changeset
168 -- top level).
kono
parents:
diff changeset
169
kono
parents:
diff changeset
170 procedure Process_Decisions
kono
parents:
diff changeset
171 (L : List_Id;
kono
parents:
diff changeset
172 T : Character;
kono
parents:
diff changeset
173 Pragma_Sloc : Source_Ptr);
kono
parents:
diff changeset
174 -- Calls above procedure for each element of the list L
kono
parents:
diff changeset
175
kono
parents:
diff changeset
176 procedure Set_Raw_Table_Entry
kono
parents:
diff changeset
177 (C1 : Character;
kono
parents:
diff changeset
178 C2 : Character;
kono
parents:
diff changeset
179 From : Source_Ptr;
kono
parents:
diff changeset
180 To : Source_Ptr;
kono
parents:
diff changeset
181 Last : Boolean;
kono
parents:
diff changeset
182 Pragma_Sloc : Source_Ptr := No_Location;
kono
parents:
diff changeset
183 Pragma_Aspect_Name : Name_Id := No_Name);
kono
parents:
diff changeset
184 -- Append an entry to SCO_Raw_Table with fields set as per arguments
kono
parents:
diff changeset
185
kono
parents:
diff changeset
186 type Dominant_Info is record
kono
parents:
diff changeset
187 K : Character;
kono
parents:
diff changeset
188 -- F/T/S/E for a valid dominance marker, or ' ' for no dominant
kono
parents:
diff changeset
189
kono
parents:
diff changeset
190 N : Node_Id;
kono
parents:
diff changeset
191 -- Node providing the Sloc(s) for the dominance marker
kono
parents:
diff changeset
192 end record;
kono
parents:
diff changeset
193 No_Dominant : constant Dominant_Info := (' ', Empty);
kono
parents:
diff changeset
194
kono
parents:
diff changeset
195 procedure Record_Instance (Id : Instance_Id; Inst_Sloc : Source_Ptr);
kono
parents:
diff changeset
196 -- Add one entry from the instance table to the corresponding SCO table
kono
parents:
diff changeset
197
kono
parents:
diff changeset
198 procedure Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
199 (L : List_Id;
kono
parents:
diff changeset
200 D : Dominant_Info := No_Dominant;
kono
parents:
diff changeset
201 P : Node_Id := Empty);
kono
parents:
diff changeset
202 -- Process L, a list of statements or declarations dominated by D. If P is
kono
parents:
diff changeset
203 -- present, it is processed as though it had been prepended to L.
kono
parents:
diff changeset
204
kono
parents:
diff changeset
205 function Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
206 (L : List_Id;
kono
parents:
diff changeset
207 D : Dominant_Info := No_Dominant;
kono
parents:
diff changeset
208 P : Node_Id := Empty) return Dominant_Info;
kono
parents:
diff changeset
209 -- Same as above, and returns dominant information corresponding to the
kono
parents:
diff changeset
210 -- last node with SCO in L.
kono
parents:
diff changeset
211
kono
parents:
diff changeset
212 -- The following Traverse_* routines perform appropriate calls to
kono
parents:
diff changeset
213 -- Traverse_Declarations_Or_Statements to traverse specific node kinds.
kono
parents:
diff changeset
214 -- Parameter D, when present, indicates the dominant of the first
kono
parents:
diff changeset
215 -- declaration or statement within N.
kono
parents:
diff changeset
216
kono
parents:
diff changeset
217 -- Why is Traverse_Sync_Definition commented specifically, whereas
kono
parents:
diff changeset
218 -- the others are not???
kono
parents:
diff changeset
219
kono
parents:
diff changeset
220 procedure Traverse_Generic_Package_Declaration (N : Node_Id);
kono
parents:
diff changeset
221
kono
parents:
diff changeset
222 procedure Traverse_Handled_Statement_Sequence
kono
parents:
diff changeset
223 (N : Node_Id;
kono
parents:
diff changeset
224 D : Dominant_Info := No_Dominant);
kono
parents:
diff changeset
225
kono
parents:
diff changeset
226 procedure Traverse_Package_Body (N : Node_Id);
kono
parents:
diff changeset
227
kono
parents:
diff changeset
228 procedure Traverse_Package_Declaration
kono
parents:
diff changeset
229 (N : Node_Id;
kono
parents:
diff changeset
230 D : Dominant_Info := No_Dominant);
kono
parents:
diff changeset
231
kono
parents:
diff changeset
232 procedure Traverse_Subprogram_Or_Task_Body
kono
parents:
diff changeset
233 (N : Node_Id;
kono
parents:
diff changeset
234 D : Dominant_Info := No_Dominant);
kono
parents:
diff changeset
235
kono
parents:
diff changeset
236 procedure Traverse_Sync_Definition (N : Node_Id);
kono
parents:
diff changeset
237 -- Traverse a protected definition or task definition
kono
parents:
diff changeset
238
kono
parents:
diff changeset
239 -- Note regarding traversals: In a few cases where an Alternatives list is
kono
parents:
diff changeset
240 -- involved, pragmas such as "pragma Page" may show up before the first
kono
parents:
diff changeset
241 -- alternative. We skip them because we're out of statement or declaration
kono
parents:
diff changeset
242 -- context, so these can't be pragmas of interest for SCO purposes, and
kono
parents:
diff changeset
243 -- the regular alternative processing typically involves attribute queries
kono
parents:
diff changeset
244 -- which aren't valid for a pragma.
kono
parents:
diff changeset
245
kono
parents:
diff changeset
246 procedure Write_SCOs_To_ALI_File is new Put_SCOs;
kono
parents:
diff changeset
247 -- Write SCO information to the ALI file using routines in Lib.Util
kono
parents:
diff changeset
248
kono
parents:
diff changeset
249 ----------
kono
parents:
diff changeset
250 -- dsco --
kono
parents:
diff changeset
251 ----------
kono
parents:
diff changeset
252
kono
parents:
diff changeset
253 procedure dsco is
kono
parents:
diff changeset
254 procedure Dump_Entry (Index : Nat; T : SCO_Table_Entry);
kono
parents:
diff changeset
255 -- Dump a SCO table entry
kono
parents:
diff changeset
256
kono
parents:
diff changeset
257 ----------------
kono
parents:
diff changeset
258 -- Dump_Entry --
kono
parents:
diff changeset
259 ----------------
kono
parents:
diff changeset
260
kono
parents:
diff changeset
261 procedure Dump_Entry (Index : Nat; T : SCO_Table_Entry) is
kono
parents:
diff changeset
262 begin
kono
parents:
diff changeset
263 Write_Str (" ");
kono
parents:
diff changeset
264 Write_Int (Index);
kono
parents:
diff changeset
265 Write_Char ('.');
kono
parents:
diff changeset
266
kono
parents:
diff changeset
267 if T.C1 /= ' ' then
kono
parents:
diff changeset
268 Write_Str (" C1 = '");
kono
parents:
diff changeset
269 Write_Char (T.C1);
kono
parents:
diff changeset
270 Write_Char (''');
kono
parents:
diff changeset
271 end if;
kono
parents:
diff changeset
272
kono
parents:
diff changeset
273 if T.C2 /= ' ' then
kono
parents:
diff changeset
274 Write_Str (" C2 = '");
kono
parents:
diff changeset
275 Write_Char (T.C2);
kono
parents:
diff changeset
276 Write_Char (''');
kono
parents:
diff changeset
277 end if;
kono
parents:
diff changeset
278
kono
parents:
diff changeset
279 if T.From /= No_Source_Location then
kono
parents:
diff changeset
280 Write_Str (" From = ");
kono
parents:
diff changeset
281 Write_Int (Int (T.From.Line));
kono
parents:
diff changeset
282 Write_Char (':');
kono
parents:
diff changeset
283 Write_Int (Int (T.From.Col));
kono
parents:
diff changeset
284 end if;
kono
parents:
diff changeset
285
kono
parents:
diff changeset
286 if T.To /= No_Source_Location then
kono
parents:
diff changeset
287 Write_Str (" To = ");
kono
parents:
diff changeset
288 Write_Int (Int (T.To.Line));
kono
parents:
diff changeset
289 Write_Char (':');
kono
parents:
diff changeset
290 Write_Int (Int (T.To.Col));
kono
parents:
diff changeset
291 end if;
kono
parents:
diff changeset
292
kono
parents:
diff changeset
293 if T.Last then
kono
parents:
diff changeset
294 Write_Str (" True");
kono
parents:
diff changeset
295 else
kono
parents:
diff changeset
296 Write_Str (" False");
kono
parents:
diff changeset
297 end if;
kono
parents:
diff changeset
298
kono
parents:
diff changeset
299 Write_Eol;
kono
parents:
diff changeset
300 end Dump_Entry;
kono
parents:
diff changeset
301
kono
parents:
diff changeset
302 -- Start of processing for dsco
kono
parents:
diff changeset
303
kono
parents:
diff changeset
304 begin
kono
parents:
diff changeset
305 -- Dump SCO unit table
kono
parents:
diff changeset
306
kono
parents:
diff changeset
307 Write_Line ("SCO Unit Table");
kono
parents:
diff changeset
308 Write_Line ("--------------");
kono
parents:
diff changeset
309
kono
parents:
diff changeset
310 for Index in 1 .. SCO_Unit_Table.Last loop
kono
parents:
diff changeset
311 declare
kono
parents:
diff changeset
312 UTE : SCO_Unit_Table_Entry renames SCO_Unit_Table.Table (Index);
kono
parents:
diff changeset
313
kono
parents:
diff changeset
314 begin
kono
parents:
diff changeset
315 Write_Str (" ");
kono
parents:
diff changeset
316 Write_Int (Int (Index));
kono
parents:
diff changeset
317 Write_Str (" Dep_Num = ");
kono
parents:
diff changeset
318 Write_Int (Int (UTE.Dep_Num));
kono
parents:
diff changeset
319 Write_Str (" From = ");
kono
parents:
diff changeset
320 Write_Int (Int (UTE.From));
kono
parents:
diff changeset
321 Write_Str (" To = ");
kono
parents:
diff changeset
322 Write_Int (Int (UTE.To));
kono
parents:
diff changeset
323
kono
parents:
diff changeset
324 Write_Str (" File_Name = """);
kono
parents:
diff changeset
325
kono
parents:
diff changeset
326 if UTE.File_Name /= null then
kono
parents:
diff changeset
327 Write_Str (UTE.File_Name.all);
kono
parents:
diff changeset
328 end if;
kono
parents:
diff changeset
329
kono
parents:
diff changeset
330 Write_Char ('"');
kono
parents:
diff changeset
331 Write_Eol;
kono
parents:
diff changeset
332 end;
kono
parents:
diff changeset
333 end loop;
kono
parents:
diff changeset
334
kono
parents:
diff changeset
335 -- Dump SCO Unit number table if it contains any entries
kono
parents:
diff changeset
336
kono
parents:
diff changeset
337 if SCO_Unit_Number_Table.Last >= 1 then
kono
parents:
diff changeset
338 Write_Eol;
kono
parents:
diff changeset
339 Write_Line ("SCO Unit Number Table");
kono
parents:
diff changeset
340 Write_Line ("---------------------");
kono
parents:
diff changeset
341
kono
parents:
diff changeset
342 for Index in 1 .. SCO_Unit_Number_Table.Last loop
kono
parents:
diff changeset
343 Write_Str (" ");
kono
parents:
diff changeset
344 Write_Int (Int (Index));
kono
parents:
diff changeset
345 Write_Str (". Unit_Number = ");
kono
parents:
diff changeset
346 Write_Int (Int (SCO_Unit_Number_Table.Table (Index)));
kono
parents:
diff changeset
347 Write_Eol;
kono
parents:
diff changeset
348 end loop;
kono
parents:
diff changeset
349 end if;
kono
parents:
diff changeset
350
kono
parents:
diff changeset
351 -- Dump SCO raw-table
kono
parents:
diff changeset
352
kono
parents:
diff changeset
353 Write_Eol;
kono
parents:
diff changeset
354 Write_Line ("SCO Raw Table");
kono
parents:
diff changeset
355 Write_Line ("---------");
kono
parents:
diff changeset
356
kono
parents:
diff changeset
357 if SCO_Generation_State = Filtered then
kono
parents:
diff changeset
358 Write_Line ("Empty (free'd after second pass)");
kono
parents:
diff changeset
359 else
kono
parents:
diff changeset
360 for Index in 1 .. SCO_Raw_Table.Last loop
kono
parents:
diff changeset
361 Dump_Entry (Index, SCO_Raw_Table.Table (Index));
kono
parents:
diff changeset
362 end loop;
kono
parents:
diff changeset
363 end if;
kono
parents:
diff changeset
364
kono
parents:
diff changeset
365 -- Dump SCO table itself
kono
parents:
diff changeset
366
kono
parents:
diff changeset
367 Write_Eol;
kono
parents:
diff changeset
368 Write_Line ("SCO Filtered Table");
kono
parents:
diff changeset
369 Write_Line ("---------");
kono
parents:
diff changeset
370
kono
parents:
diff changeset
371 for Index in 1 .. SCO_Table.Last loop
kono
parents:
diff changeset
372 Dump_Entry (Index, SCO_Table.Table (Index));
kono
parents:
diff changeset
373 end loop;
kono
parents:
diff changeset
374 end dsco;
kono
parents:
diff changeset
375
kono
parents:
diff changeset
376 -----------
kono
parents:
diff changeset
377 -- Equal --
kono
parents:
diff changeset
378 -----------
kono
parents:
diff changeset
379
kono
parents:
diff changeset
380 function Equal (F1 : Source_Ptr; F2 : Source_Ptr) return Boolean is
kono
parents:
diff changeset
381 begin
kono
parents:
diff changeset
382 return F1 = F2;
kono
parents:
diff changeset
383 end Equal;
kono
parents:
diff changeset
384
kono
parents:
diff changeset
385 -------
kono
parents:
diff changeset
386 -- < --
kono
parents:
diff changeset
387 -------
kono
parents:
diff changeset
388
kono
parents:
diff changeset
389 function "<" (S1 : Source_Location; S2 : Source_Location) return Boolean is
kono
parents:
diff changeset
390 begin
kono
parents:
diff changeset
391 return S1.Line < S2.Line
kono
parents:
diff changeset
392 or else (S1.Line = S2.Line and then S1.Col < S2.Col);
kono
parents:
diff changeset
393 end "<";
kono
parents:
diff changeset
394
kono
parents:
diff changeset
395 ------------------
kono
parents:
diff changeset
396 -- Has_Decision --
kono
parents:
diff changeset
397 ------------------
kono
parents:
diff changeset
398
kono
parents:
diff changeset
399 function Has_Decision (N : Node_Id) return Boolean is
kono
parents:
diff changeset
400 function Check_Node (N : Node_Id) return Traverse_Result;
kono
parents:
diff changeset
401 -- Determine if Nkind (N) indicates the presence of a decision (i.e. N
kono
parents:
diff changeset
402 -- is a logical operator, which is a decision in itself, or an
kono
parents:
diff changeset
403 -- IF-expression whose Condition attribute is a decision).
kono
parents:
diff changeset
404
kono
parents:
diff changeset
405 ----------------
kono
parents:
diff changeset
406 -- Check_Node --
kono
parents:
diff changeset
407 ----------------
kono
parents:
diff changeset
408
kono
parents:
diff changeset
409 function Check_Node (N : Node_Id) return Traverse_Result is
kono
parents:
diff changeset
410 begin
kono
parents:
diff changeset
411 -- If we are not sure this is a logical operator (AND and OR may be
kono
parents:
diff changeset
412 -- turned into logical operators with the Short_Circuit_And_Or
kono
parents:
diff changeset
413 -- pragma), assume it is. Putative decisions will be discarded if
kono
parents:
diff changeset
414 -- needed in the secord pass.
kono
parents:
diff changeset
415
kono
parents:
diff changeset
416 if Is_Logical_Operator (N) /= False
kono
parents:
diff changeset
417 or else Nkind (N) = N_If_Expression
kono
parents:
diff changeset
418 then
kono
parents:
diff changeset
419 return Abandon;
kono
parents:
diff changeset
420 else
kono
parents:
diff changeset
421 return OK;
kono
parents:
diff changeset
422 end if;
kono
parents:
diff changeset
423 end Check_Node;
kono
parents:
diff changeset
424
kono
parents:
diff changeset
425 function Traverse is new Traverse_Func (Check_Node);
kono
parents:
diff changeset
426
kono
parents:
diff changeset
427 -- Start of processing for Has_Decision
kono
parents:
diff changeset
428
kono
parents:
diff changeset
429 begin
kono
parents:
diff changeset
430 return Traverse (N) = Abandon;
kono
parents:
diff changeset
431 end Has_Decision;
kono
parents:
diff changeset
432
kono
parents:
diff changeset
433 ----------
kono
parents:
diff changeset
434 -- Hash --
kono
parents:
diff changeset
435 ----------
kono
parents:
diff changeset
436
kono
parents:
diff changeset
437 function Hash (F : Source_Ptr) return Header_Num is
kono
parents:
diff changeset
438 begin
kono
parents:
diff changeset
439 return Header_Num (Nat (F) mod 997);
kono
parents:
diff changeset
440 end Hash;
kono
parents:
diff changeset
441
kono
parents:
diff changeset
442 ----------------
kono
parents:
diff changeset
443 -- Initialize --
kono
parents:
diff changeset
444 ----------------
kono
parents:
diff changeset
445
kono
parents:
diff changeset
446 procedure Initialize is
kono
parents:
diff changeset
447 begin
kono
parents:
diff changeset
448 SCO_Unit_Number_Table.Init;
kono
parents:
diff changeset
449
kono
parents:
diff changeset
450 -- The SCO_Unit_Number_Table entry with index 0 is intentionally set
kono
parents:
diff changeset
451 -- aside to be used as temporary for sorting.
kono
parents:
diff changeset
452
kono
parents:
diff changeset
453 SCO_Unit_Number_Table.Increment_Last;
kono
parents:
diff changeset
454 end Initialize;
kono
parents:
diff changeset
455
kono
parents:
diff changeset
456 -------------------------
kono
parents:
diff changeset
457 -- Is_Logical_Operator --
kono
parents:
diff changeset
458 -------------------------
kono
parents:
diff changeset
459
kono
parents:
diff changeset
460 function Is_Logical_Operator (N : Node_Id) return Tristate is
kono
parents:
diff changeset
461 begin
kono
parents:
diff changeset
462 if Nkind_In (N, N_And_Then, N_Op_Not, N_Or_Else) then
kono
parents:
diff changeset
463 return True;
kono
parents:
diff changeset
464 elsif Nkind_In (N, N_Op_And, N_Op_Or) then
kono
parents:
diff changeset
465 return Unknown;
kono
parents:
diff changeset
466 else
kono
parents:
diff changeset
467 return False;
kono
parents:
diff changeset
468 end if;
kono
parents:
diff changeset
469 end Is_Logical_Operator;
kono
parents:
diff changeset
470
kono
parents:
diff changeset
471 -----------------------
kono
parents:
diff changeset
472 -- Process_Decisions --
kono
parents:
diff changeset
473 -----------------------
kono
parents:
diff changeset
474
kono
parents:
diff changeset
475 -- Version taking a list
kono
parents:
diff changeset
476
kono
parents:
diff changeset
477 procedure Process_Decisions
kono
parents:
diff changeset
478 (L : List_Id;
kono
parents:
diff changeset
479 T : Character;
kono
parents:
diff changeset
480 Pragma_Sloc : Source_Ptr)
kono
parents:
diff changeset
481 is
kono
parents:
diff changeset
482 N : Node_Id;
kono
parents:
diff changeset
483
kono
parents:
diff changeset
484 begin
kono
parents:
diff changeset
485 if L /= No_List then
kono
parents:
diff changeset
486 N := First (L);
kono
parents:
diff changeset
487 while Present (N) loop
kono
parents:
diff changeset
488 Process_Decisions (N, T, Pragma_Sloc);
kono
parents:
diff changeset
489 Next (N);
kono
parents:
diff changeset
490 end loop;
kono
parents:
diff changeset
491 end if;
kono
parents:
diff changeset
492 end Process_Decisions;
kono
parents:
diff changeset
493
kono
parents:
diff changeset
494 -- Version taking a node
kono
parents:
diff changeset
495
kono
parents:
diff changeset
496 Current_Pragma_Sloc : Source_Ptr := No_Location;
kono
parents:
diff changeset
497 -- While processing a pragma, this is set to the sloc of the N_Pragma node
kono
parents:
diff changeset
498
kono
parents:
diff changeset
499 procedure Process_Decisions
kono
parents:
diff changeset
500 (N : Node_Id;
kono
parents:
diff changeset
501 T : Character;
kono
parents:
diff changeset
502 Pragma_Sloc : Source_Ptr)
kono
parents:
diff changeset
503 is
kono
parents:
diff changeset
504 Mark : Nat;
kono
parents:
diff changeset
505 -- This is used to mark the location of a decision sequence in the SCO
kono
parents:
diff changeset
506 -- table. We use it for backing out a simple decision in an expression
kono
parents:
diff changeset
507 -- context that contains only NOT operators.
kono
parents:
diff changeset
508
kono
parents:
diff changeset
509 Mark_Hash : Nat;
kono
parents:
diff changeset
510 -- Likewise for the putative SCO_Raw_Hash_Table entries: see below
kono
parents:
diff changeset
511
kono
parents:
diff changeset
512 type Hash_Entry is record
kono
parents:
diff changeset
513 Sloc : Source_Ptr;
kono
parents:
diff changeset
514 SCO_Index : Nat;
kono
parents:
diff changeset
515 end record;
kono
parents:
diff changeset
516 -- We must register all conditions/pragmas in SCO_Raw_Hash_Table.
kono
parents:
diff changeset
517 -- However we cannot register them in the same time we are adding the
kono
parents:
diff changeset
518 -- corresponding SCO entries to the raw table since we may discard them
kono
parents:
diff changeset
519 -- later on. So instead we put all putative conditions into Hash_Entries
kono
parents:
diff changeset
520 -- (see below) and register them once we are sure we keep them.
kono
parents:
diff changeset
521 --
kono
parents:
diff changeset
522 -- This data structure holds the conditions/pragmas to register in
kono
parents:
diff changeset
523 -- SCO_Raw_Hash_Table.
kono
parents:
diff changeset
524
kono
parents:
diff changeset
525 package Hash_Entries is new Table.Table
kono
parents:
diff changeset
526 (Table_Component_Type => Hash_Entry,
kono
parents:
diff changeset
527 Table_Index_Type => Nat,
kono
parents:
diff changeset
528 Table_Low_Bound => 1,
kono
parents:
diff changeset
529 Table_Initial => 10,
kono
parents:
diff changeset
530 Table_Increment => 10,
kono
parents:
diff changeset
531 Table_Name => "Hash_Entries");
kono
parents:
diff changeset
532 -- Hold temporarily (i.e. free'd before returning) the Hash_Entry before
kono
parents:
diff changeset
533 -- they are registered in SCO_Raw_Hash_Table.
kono
parents:
diff changeset
534
kono
parents:
diff changeset
535 X_Not_Decision : Boolean;
kono
parents:
diff changeset
536 -- This flag keeps track of whether a decision sequence in the SCO table
kono
parents:
diff changeset
537 -- contains only NOT operators, and is for an expression context (T=X).
kono
parents:
diff changeset
538 -- The flag will be set False if T is other than X, or if an operator
kono
parents:
diff changeset
539 -- other than NOT is in the sequence.
kono
parents:
diff changeset
540
kono
parents:
diff changeset
541 procedure Output_Decision_Operand (N : Node_Id);
kono
parents:
diff changeset
542 -- The node N is the top level logical operator of a decision, or it is
kono
parents:
diff changeset
543 -- one of the operands of a logical operator belonging to a single
kono
parents:
diff changeset
544 -- complex decision. This routine outputs the sequence of table entries
kono
parents:
diff changeset
545 -- corresponding to the node. Note that we do not process the sub-
kono
parents:
diff changeset
546 -- operands to look for further decisions, that processing is done in
kono
parents:
diff changeset
547 -- Process_Decision_Operand, because we can't get decisions mixed up in
kono
parents:
diff changeset
548 -- the global table. Call has no effect if N is Empty.
kono
parents:
diff changeset
549
kono
parents:
diff changeset
550 procedure Output_Element (N : Node_Id);
kono
parents:
diff changeset
551 -- Node N is an operand of a logical operator that is not itself a
kono
parents:
diff changeset
552 -- logical operator, or it is a simple decision. This routine outputs
kono
parents:
diff changeset
553 -- the table entry for the element, with C1 set to ' '. Last is set
kono
parents:
diff changeset
554 -- False, and an entry is made in the condition hash table.
kono
parents:
diff changeset
555
kono
parents:
diff changeset
556 procedure Output_Header (T : Character);
kono
parents:
diff changeset
557 -- Outputs a decision header node. T is I/W/E/P for IF/WHILE/EXIT WHEN/
kono
parents:
diff changeset
558 -- PRAGMA, and 'X' for the expression case.
kono
parents:
diff changeset
559
kono
parents:
diff changeset
560 procedure Process_Decision_Operand (N : Node_Id);
kono
parents:
diff changeset
561 -- This is called on node N, the top level node of a decision, or on one
kono
parents:
diff changeset
562 -- of its operands or suboperands after generating the full output for
kono
parents:
diff changeset
563 -- the complex decision. It process the suboperands of the decision
kono
parents:
diff changeset
564 -- looking for nested decisions.
kono
parents:
diff changeset
565
kono
parents:
diff changeset
566 function Process_Node (N : Node_Id) return Traverse_Result;
kono
parents:
diff changeset
567 -- Processes one node in the traversal, looking for logical operators,
kono
parents:
diff changeset
568 -- and if one is found, outputs the appropriate table entries.
kono
parents:
diff changeset
569
kono
parents:
diff changeset
570 -----------------------------
kono
parents:
diff changeset
571 -- Output_Decision_Operand --
kono
parents:
diff changeset
572 -----------------------------
kono
parents:
diff changeset
573
kono
parents:
diff changeset
574 procedure Output_Decision_Operand (N : Node_Id) is
kono
parents:
diff changeset
575 C1 : Character;
kono
parents:
diff changeset
576 C2 : Character;
kono
parents:
diff changeset
577 -- C1 holds a character that identifies the operation while C2
kono
parents:
diff changeset
578 -- indicates whether we are sure (' ') or not ('?') this operation
kono
parents:
diff changeset
579 -- belongs to the decision. '?' entries will be filtered out in the
kono
parents:
diff changeset
580 -- second (SCO_Record_Filtered) pass.
kono
parents:
diff changeset
581
kono
parents:
diff changeset
582 L : Node_Id;
kono
parents:
diff changeset
583 T : Tristate;
kono
parents:
diff changeset
584
kono
parents:
diff changeset
585 begin
kono
parents:
diff changeset
586 if No (N) then
kono
parents:
diff changeset
587 return;
kono
parents:
diff changeset
588 end if;
kono
parents:
diff changeset
589
kono
parents:
diff changeset
590 T := Is_Logical_Operator (N);
kono
parents:
diff changeset
591
kono
parents:
diff changeset
592 -- Logical operator
kono
parents:
diff changeset
593
kono
parents:
diff changeset
594 if T /= False then
kono
parents:
diff changeset
595 if Nkind (N) = N_Op_Not then
kono
parents:
diff changeset
596 C1 := '!';
kono
parents:
diff changeset
597 L := Empty;
kono
parents:
diff changeset
598
kono
parents:
diff changeset
599 else
kono
parents:
diff changeset
600 L := Left_Opnd (N);
kono
parents:
diff changeset
601
kono
parents:
diff changeset
602 if Nkind_In (N, N_Op_Or, N_Or_Else) then
kono
parents:
diff changeset
603 C1 := '|';
kono
parents:
diff changeset
604 else pragma Assert (Nkind_In (N, N_Op_And, N_And_Then));
kono
parents:
diff changeset
605 C1 := '&';
kono
parents:
diff changeset
606 end if;
kono
parents:
diff changeset
607 end if;
kono
parents:
diff changeset
608
kono
parents:
diff changeset
609 if T = True then
kono
parents:
diff changeset
610 C2 := ' ';
kono
parents:
diff changeset
611 else
kono
parents:
diff changeset
612 C2 := '?';
kono
parents:
diff changeset
613 end if;
kono
parents:
diff changeset
614
kono
parents:
diff changeset
615 Set_Raw_Table_Entry
kono
parents:
diff changeset
616 (C1 => C1,
kono
parents:
diff changeset
617 C2 => C2,
kono
parents:
diff changeset
618 From => Sloc (N),
kono
parents:
diff changeset
619 To => No_Location,
kono
parents:
diff changeset
620 Last => False);
kono
parents:
diff changeset
621
kono
parents:
diff changeset
622 Hash_Entries.Append ((Sloc (N), SCO_Raw_Table.Last));
kono
parents:
diff changeset
623
kono
parents:
diff changeset
624 Output_Decision_Operand (L);
kono
parents:
diff changeset
625 Output_Decision_Operand (Right_Opnd (N));
kono
parents:
diff changeset
626
kono
parents:
diff changeset
627 -- Not a logical operator
kono
parents:
diff changeset
628
kono
parents:
diff changeset
629 else
kono
parents:
diff changeset
630 Output_Element (N);
kono
parents:
diff changeset
631 end if;
kono
parents:
diff changeset
632 end Output_Decision_Operand;
kono
parents:
diff changeset
633
kono
parents:
diff changeset
634 --------------------
kono
parents:
diff changeset
635 -- Output_Element --
kono
parents:
diff changeset
636 --------------------
kono
parents:
diff changeset
637
kono
parents:
diff changeset
638 procedure Output_Element (N : Node_Id) is
kono
parents:
diff changeset
639 FSloc : Source_Ptr;
kono
parents:
diff changeset
640 LSloc : Source_Ptr;
kono
parents:
diff changeset
641 begin
kono
parents:
diff changeset
642 Sloc_Range (N, FSloc, LSloc);
kono
parents:
diff changeset
643 Set_Raw_Table_Entry
kono
parents:
diff changeset
644 (C1 => ' ',
kono
parents:
diff changeset
645 C2 => 'c',
kono
parents:
diff changeset
646 From => FSloc,
kono
parents:
diff changeset
647 To => LSloc,
kono
parents:
diff changeset
648 Last => False);
kono
parents:
diff changeset
649 Hash_Entries.Append ((FSloc, SCO_Raw_Table.Last));
kono
parents:
diff changeset
650 end Output_Element;
kono
parents:
diff changeset
651
kono
parents:
diff changeset
652 -------------------
kono
parents:
diff changeset
653 -- Output_Header --
kono
parents:
diff changeset
654 -------------------
kono
parents:
diff changeset
655
kono
parents:
diff changeset
656 procedure Output_Header (T : Character) is
kono
parents:
diff changeset
657 Loc : Source_Ptr := No_Location;
kono
parents:
diff changeset
658 -- Node whose Sloc is used for the decision
kono
parents:
diff changeset
659
kono
parents:
diff changeset
660 Nam : Name_Id := No_Name;
kono
parents:
diff changeset
661 -- For the case of an aspect, aspect name
kono
parents:
diff changeset
662
kono
parents:
diff changeset
663 begin
kono
parents:
diff changeset
664 case T is
kono
parents:
diff changeset
665 when 'I' | 'E' | 'W' | 'a' | 'A' =>
kono
parents:
diff changeset
666
kono
parents:
diff changeset
667 -- For IF, EXIT, WHILE, or aspects, the token SLOC is that of
kono
parents:
diff changeset
668 -- the parent of the expression.
kono
parents:
diff changeset
669
kono
parents:
diff changeset
670 Loc := Sloc (Parent (N));
kono
parents:
diff changeset
671
kono
parents:
diff changeset
672 if T = 'a' or else T = 'A' then
kono
parents:
diff changeset
673 Nam := Chars (Identifier (Parent (N)));
kono
parents:
diff changeset
674 end if;
kono
parents:
diff changeset
675
kono
parents:
diff changeset
676 when 'G' | 'P' =>
kono
parents:
diff changeset
677
kono
parents:
diff changeset
678 -- For entry guard, the token sloc is from the N_Entry_Body.
kono
parents:
diff changeset
679 -- For PRAGMA, we must get the location from the pragma node.
kono
parents:
diff changeset
680 -- Argument N is the pragma argument, and we have to go up
kono
parents:
diff changeset
681 -- two levels (through the pragma argument association) to
kono
parents:
diff changeset
682 -- get to the pragma node itself. For the guard on a select
kono
parents:
diff changeset
683 -- alternative, we do not have access to the token location for
kono
parents:
diff changeset
684 -- the WHEN, so we use the first sloc of the condition itself
kono
parents:
diff changeset
685 -- (note: we use First_Sloc, not Sloc, because this is what is
kono
parents:
diff changeset
686 -- referenced by dominance markers).
kono
parents:
diff changeset
687
kono
parents:
diff changeset
688 -- Doesn't this requirement of using First_Sloc need to be
kono
parents:
diff changeset
689 -- documented in the spec ???
kono
parents:
diff changeset
690
kono
parents:
diff changeset
691 if Nkind_In (Parent (N), N_Accept_Alternative,
kono
parents:
diff changeset
692 N_Delay_Alternative,
kono
parents:
diff changeset
693 N_Terminate_Alternative)
kono
parents:
diff changeset
694 then
kono
parents:
diff changeset
695 Loc := First_Sloc (N);
kono
parents:
diff changeset
696 else
kono
parents:
diff changeset
697 Loc := Sloc (Parent (Parent (N)));
kono
parents:
diff changeset
698 end if;
kono
parents:
diff changeset
699
kono
parents:
diff changeset
700 when 'X' =>
kono
parents:
diff changeset
701
kono
parents:
diff changeset
702 -- For an expression, no Sloc
kono
parents:
diff changeset
703
kono
parents:
diff changeset
704 null;
kono
parents:
diff changeset
705
kono
parents:
diff changeset
706 -- No other possibilities
kono
parents:
diff changeset
707
kono
parents:
diff changeset
708 when others =>
kono
parents:
diff changeset
709 raise Program_Error;
kono
parents:
diff changeset
710 end case;
kono
parents:
diff changeset
711
kono
parents:
diff changeset
712 Set_Raw_Table_Entry
kono
parents:
diff changeset
713 (C1 => T,
kono
parents:
diff changeset
714 C2 => ' ',
kono
parents:
diff changeset
715 From => Loc,
kono
parents:
diff changeset
716 To => No_Location,
kono
parents:
diff changeset
717 Last => False,
kono
parents:
diff changeset
718 Pragma_Sloc => Pragma_Sloc,
kono
parents:
diff changeset
719 Pragma_Aspect_Name => Nam);
kono
parents:
diff changeset
720
kono
parents:
diff changeset
721 -- For an aspect specification, which will be rewritten into a
kono
parents:
diff changeset
722 -- pragma, enter a hash table entry now.
kono
parents:
diff changeset
723
kono
parents:
diff changeset
724 if T = 'a' then
kono
parents:
diff changeset
725 Hash_Entries.Append ((Loc, SCO_Raw_Table.Last));
kono
parents:
diff changeset
726 end if;
kono
parents:
diff changeset
727 end Output_Header;
kono
parents:
diff changeset
728
kono
parents:
diff changeset
729 ------------------------------
kono
parents:
diff changeset
730 -- Process_Decision_Operand --
kono
parents:
diff changeset
731 ------------------------------
kono
parents:
diff changeset
732
kono
parents:
diff changeset
733 procedure Process_Decision_Operand (N : Node_Id) is
kono
parents:
diff changeset
734 begin
kono
parents:
diff changeset
735 if Is_Logical_Operator (N) /= False then
kono
parents:
diff changeset
736 if Nkind (N) /= N_Op_Not then
kono
parents:
diff changeset
737 Process_Decision_Operand (Left_Opnd (N));
kono
parents:
diff changeset
738 X_Not_Decision := False;
kono
parents:
diff changeset
739 end if;
kono
parents:
diff changeset
740
kono
parents:
diff changeset
741 Process_Decision_Operand (Right_Opnd (N));
kono
parents:
diff changeset
742
kono
parents:
diff changeset
743 else
kono
parents:
diff changeset
744 Process_Decisions (N, 'X', Pragma_Sloc);
kono
parents:
diff changeset
745 end if;
kono
parents:
diff changeset
746 end Process_Decision_Operand;
kono
parents:
diff changeset
747
kono
parents:
diff changeset
748 ------------------
kono
parents:
diff changeset
749 -- Process_Node --
kono
parents:
diff changeset
750 ------------------
kono
parents:
diff changeset
751
kono
parents:
diff changeset
752 function Process_Node (N : Node_Id) return Traverse_Result is
kono
parents:
diff changeset
753 begin
kono
parents:
diff changeset
754 case Nkind (N) is
kono
parents:
diff changeset
755
kono
parents:
diff changeset
756 -- Logical operators, output table entries and then process
kono
parents:
diff changeset
757 -- operands recursively to deal with nested conditions.
kono
parents:
diff changeset
758
kono
parents:
diff changeset
759 when N_And_Then
kono
parents:
diff changeset
760 | N_Op_And
kono
parents:
diff changeset
761 | N_Op_Not
kono
parents:
diff changeset
762 | N_Op_Or
kono
parents:
diff changeset
763 | N_Or_Else
kono
parents:
diff changeset
764 =>
kono
parents:
diff changeset
765 declare
kono
parents:
diff changeset
766 T : Character;
kono
parents:
diff changeset
767
kono
parents:
diff changeset
768 begin
kono
parents:
diff changeset
769 -- If outer level, then type comes from call, otherwise it
kono
parents:
diff changeset
770 -- is more deeply nested and counts as X for expression.
kono
parents:
diff changeset
771
kono
parents:
diff changeset
772 if N = Process_Decisions.N then
kono
parents:
diff changeset
773 T := Process_Decisions.T;
kono
parents:
diff changeset
774 else
kono
parents:
diff changeset
775 T := 'X';
kono
parents:
diff changeset
776 end if;
kono
parents:
diff changeset
777
kono
parents:
diff changeset
778 -- Output header for sequence
kono
parents:
diff changeset
779
kono
parents:
diff changeset
780 X_Not_Decision := T = 'X' and then Nkind (N) = N_Op_Not;
kono
parents:
diff changeset
781 Mark := SCO_Raw_Table.Last;
kono
parents:
diff changeset
782 Mark_Hash := Hash_Entries.Last;
kono
parents:
diff changeset
783 Output_Header (T);
kono
parents:
diff changeset
784
kono
parents:
diff changeset
785 -- Output the decision
kono
parents:
diff changeset
786
kono
parents:
diff changeset
787 Output_Decision_Operand (N);
kono
parents:
diff changeset
788
kono
parents:
diff changeset
789 -- If the decision was in an expression context (T = 'X')
kono
parents:
diff changeset
790 -- and contained only NOT operators, then we don't output
kono
parents:
diff changeset
791 -- it, so delete it.
kono
parents:
diff changeset
792
kono
parents:
diff changeset
793 if X_Not_Decision then
kono
parents:
diff changeset
794 SCO_Raw_Table.Set_Last (Mark);
kono
parents:
diff changeset
795 Hash_Entries.Set_Last (Mark_Hash);
kono
parents:
diff changeset
796
kono
parents:
diff changeset
797 -- Otherwise, set Last in last table entry to mark end
kono
parents:
diff changeset
798
kono
parents:
diff changeset
799 else
kono
parents:
diff changeset
800 SCO_Raw_Table.Table (SCO_Raw_Table.Last).Last := True;
kono
parents:
diff changeset
801 end if;
kono
parents:
diff changeset
802
kono
parents:
diff changeset
803 -- Process any embedded decisions
kono
parents:
diff changeset
804
kono
parents:
diff changeset
805 Process_Decision_Operand (N);
kono
parents:
diff changeset
806 return Skip;
kono
parents:
diff changeset
807 end;
kono
parents:
diff changeset
808
kono
parents:
diff changeset
809 -- Case expression
kono
parents:
diff changeset
810
kono
parents:
diff changeset
811 -- Really hard to believe this is correct given the special
kono
parents:
diff changeset
812 -- handling for if expressions below ???
kono
parents:
diff changeset
813
kono
parents:
diff changeset
814 when N_Case_Expression =>
kono
parents:
diff changeset
815 return OK; -- ???
kono
parents:
diff changeset
816
kono
parents:
diff changeset
817 -- If expression, processed like an if statement
kono
parents:
diff changeset
818
kono
parents:
diff changeset
819 when N_If_Expression =>
kono
parents:
diff changeset
820 declare
kono
parents:
diff changeset
821 Cond : constant Node_Id := First (Expressions (N));
kono
parents:
diff changeset
822 Thnx : constant Node_Id := Next (Cond);
kono
parents:
diff changeset
823 Elsx : constant Node_Id := Next (Thnx);
kono
parents:
diff changeset
824
kono
parents:
diff changeset
825 begin
kono
parents:
diff changeset
826 Process_Decisions (Cond, 'I', Pragma_Sloc);
kono
parents:
diff changeset
827 Process_Decisions (Thnx, 'X', Pragma_Sloc);
kono
parents:
diff changeset
828 Process_Decisions (Elsx, 'X', Pragma_Sloc);
kono
parents:
diff changeset
829 return Skip;
kono
parents:
diff changeset
830 end;
kono
parents:
diff changeset
831
kono
parents:
diff changeset
832 -- All other cases, continue scan
kono
parents:
diff changeset
833
kono
parents:
diff changeset
834 when others =>
kono
parents:
diff changeset
835 return OK;
kono
parents:
diff changeset
836 end case;
kono
parents:
diff changeset
837 end Process_Node;
kono
parents:
diff changeset
838
kono
parents:
diff changeset
839 procedure Traverse is new Traverse_Proc (Process_Node);
kono
parents:
diff changeset
840
kono
parents:
diff changeset
841 -- Start of processing for Process_Decisions
kono
parents:
diff changeset
842
kono
parents:
diff changeset
843 begin
kono
parents:
diff changeset
844 if No (N) then
kono
parents:
diff changeset
845 return;
kono
parents:
diff changeset
846 end if;
kono
parents:
diff changeset
847
kono
parents:
diff changeset
848 Hash_Entries.Init;
kono
parents:
diff changeset
849
kono
parents:
diff changeset
850 -- See if we have simple decision at outer level and if so then
kono
parents:
diff changeset
851 -- generate the decision entry for this simple decision. A simple
kono
parents:
diff changeset
852 -- decision is a boolean expression (which is not a logical operator
kono
parents:
diff changeset
853 -- or short circuit form) appearing as the operand of an IF, WHILE,
kono
parents:
diff changeset
854 -- EXIT WHEN, or special PRAGMA construct.
kono
parents:
diff changeset
855
kono
parents:
diff changeset
856 if T /= 'X' and then Is_Logical_Operator (N) = False then
kono
parents:
diff changeset
857 Output_Header (T);
kono
parents:
diff changeset
858 Output_Element (N);
kono
parents:
diff changeset
859
kono
parents:
diff changeset
860 -- Change Last in last table entry to True to mark end of
kono
parents:
diff changeset
861 -- sequence, which is this case is only one element long.
kono
parents:
diff changeset
862
kono
parents:
diff changeset
863 SCO_Raw_Table.Table (SCO_Raw_Table.Last).Last := True;
kono
parents:
diff changeset
864 end if;
kono
parents:
diff changeset
865
kono
parents:
diff changeset
866 Traverse (N);
kono
parents:
diff changeset
867
kono
parents:
diff changeset
868 -- Now we have the definitive set of SCO entries, register them in the
kono
parents:
diff changeset
869 -- corresponding hash table.
kono
parents:
diff changeset
870
kono
parents:
diff changeset
871 for J in 1 .. Hash_Entries.Last loop
kono
parents:
diff changeset
872 SCO_Raw_Hash_Table.Set
kono
parents:
diff changeset
873 (Hash_Entries.Table (J).Sloc,
kono
parents:
diff changeset
874 Hash_Entries.Table (J).SCO_Index);
kono
parents:
diff changeset
875 end loop;
kono
parents:
diff changeset
876
kono
parents:
diff changeset
877 Hash_Entries.Free;
kono
parents:
diff changeset
878 end Process_Decisions;
kono
parents:
diff changeset
879
kono
parents:
diff changeset
880 -----------
kono
parents:
diff changeset
881 -- pscos --
kono
parents:
diff changeset
882 -----------
kono
parents:
diff changeset
883
kono
parents:
diff changeset
884 procedure pscos is
kono
parents:
diff changeset
885 procedure Write_Info_Char (C : Character) renames Write_Char;
kono
parents:
diff changeset
886 -- Write one character;
kono
parents:
diff changeset
887
kono
parents:
diff changeset
888 procedure Write_Info_Initiate (Key : Character) renames Write_Char;
kono
parents:
diff changeset
889 -- Start new one and write one character;
kono
parents:
diff changeset
890
kono
parents:
diff changeset
891 procedure Write_Info_Nat (N : Nat);
kono
parents:
diff changeset
892 -- Write value of N
kono
parents:
diff changeset
893
kono
parents:
diff changeset
894 procedure Write_Info_Terminate renames Write_Eol;
kono
parents:
diff changeset
895 -- Terminate current line
kono
parents:
diff changeset
896
kono
parents:
diff changeset
897 --------------------
kono
parents:
diff changeset
898 -- Write_Info_Nat --
kono
parents:
diff changeset
899 --------------------
kono
parents:
diff changeset
900
kono
parents:
diff changeset
901 procedure Write_Info_Nat (N : Nat) is
kono
parents:
diff changeset
902 begin
kono
parents:
diff changeset
903 Write_Int (N);
kono
parents:
diff changeset
904 end Write_Info_Nat;
kono
parents:
diff changeset
905
kono
parents:
diff changeset
906 procedure Debug_Put_SCOs is new Put_SCOs;
kono
parents:
diff changeset
907
kono
parents:
diff changeset
908 -- Start of processing for pscos
kono
parents:
diff changeset
909
kono
parents:
diff changeset
910 begin
kono
parents:
diff changeset
911 Debug_Put_SCOs;
kono
parents:
diff changeset
912 end pscos;
kono
parents:
diff changeset
913
kono
parents:
diff changeset
914 ---------------------
kono
parents:
diff changeset
915 -- Record_Instance --
kono
parents:
diff changeset
916 ---------------------
kono
parents:
diff changeset
917
kono
parents:
diff changeset
918 procedure Record_Instance (Id : Instance_Id; Inst_Sloc : Source_Ptr) is
kono
parents:
diff changeset
919 Inst_Src : constant Source_File_Index :=
kono
parents:
diff changeset
920 Get_Source_File_Index (Inst_Sloc);
kono
parents:
diff changeset
921 begin
kono
parents:
diff changeset
922 SCO_Instance_Table.Append
kono
parents:
diff changeset
923 ((Inst_Dep_Num => Dependency_Num (Unit (Inst_Src)),
kono
parents:
diff changeset
924 Inst_Loc => To_Source_Location (Inst_Sloc),
kono
parents:
diff changeset
925 Enclosing_Instance => SCO_Instance_Index (Instance (Inst_Src))));
kono
parents:
diff changeset
926
kono
parents:
diff changeset
927 pragma Assert
kono
parents:
diff changeset
928 (SCO_Instance_Table.Last = SCO_Instance_Index (Id));
kono
parents:
diff changeset
929 end Record_Instance;
kono
parents:
diff changeset
930
kono
parents:
diff changeset
931 ----------------
kono
parents:
diff changeset
932 -- SCO_Output --
kono
parents:
diff changeset
933 ----------------
kono
parents:
diff changeset
934
kono
parents:
diff changeset
935 procedure SCO_Output is
kono
parents:
diff changeset
936 procedure Populate_SCO_Instance_Table is
kono
parents:
diff changeset
937 new Sinput.Iterate_On_Instances (Record_Instance);
kono
parents:
diff changeset
938
kono
parents:
diff changeset
939 begin
kono
parents:
diff changeset
940 pragma Assert (SCO_Generation_State = Filtered);
kono
parents:
diff changeset
941
kono
parents:
diff changeset
942 if Debug_Flag_Dot_OO then
kono
parents:
diff changeset
943 dsco;
kono
parents:
diff changeset
944 end if;
kono
parents:
diff changeset
945
kono
parents:
diff changeset
946 Populate_SCO_Instance_Table;
kono
parents:
diff changeset
947
kono
parents:
diff changeset
948 -- Sort the unit tables based on dependency numbers
kono
parents:
diff changeset
949
kono
parents:
diff changeset
950 Unit_Table_Sort : declare
kono
parents:
diff changeset
951 function Lt (Op1 : Natural; Op2 : Natural) return Boolean;
kono
parents:
diff changeset
952 -- Comparison routine for sort call
kono
parents:
diff changeset
953
kono
parents:
diff changeset
954 procedure Move (From : Natural; To : Natural);
kono
parents:
diff changeset
955 -- Move routine for sort call
kono
parents:
diff changeset
956
kono
parents:
diff changeset
957 --------
kono
parents:
diff changeset
958 -- Lt --
kono
parents:
diff changeset
959 --------
kono
parents:
diff changeset
960
kono
parents:
diff changeset
961 function Lt (Op1 : Natural; Op2 : Natural) return Boolean is
kono
parents:
diff changeset
962 begin
kono
parents:
diff changeset
963 return
kono
parents:
diff changeset
964 Dependency_Num
kono
parents:
diff changeset
965 (SCO_Unit_Number_Table.Table (SCO_Unit_Index (Op1)))
kono
parents:
diff changeset
966 <
kono
parents:
diff changeset
967 Dependency_Num
kono
parents:
diff changeset
968 (SCO_Unit_Number_Table.Table (SCO_Unit_Index (Op2)));
kono
parents:
diff changeset
969 end Lt;
kono
parents:
diff changeset
970
kono
parents:
diff changeset
971 ----------
kono
parents:
diff changeset
972 -- Move --
kono
parents:
diff changeset
973 ----------
kono
parents:
diff changeset
974
kono
parents:
diff changeset
975 procedure Move (From : Natural; To : Natural) is
kono
parents:
diff changeset
976 begin
kono
parents:
diff changeset
977 SCO_Unit_Table.Table (SCO_Unit_Index (To)) :=
kono
parents:
diff changeset
978 SCO_Unit_Table.Table (SCO_Unit_Index (From));
kono
parents:
diff changeset
979 SCO_Unit_Number_Table.Table (SCO_Unit_Index (To)) :=
kono
parents:
diff changeset
980 SCO_Unit_Number_Table.Table (SCO_Unit_Index (From));
kono
parents:
diff changeset
981 end Move;
kono
parents:
diff changeset
982
kono
parents:
diff changeset
983 package Sorting is new GNAT.Heap_Sort_G (Move, Lt);
kono
parents:
diff changeset
984
kono
parents:
diff changeset
985 -- Start of processing for Unit_Table_Sort
kono
parents:
diff changeset
986
kono
parents:
diff changeset
987 begin
kono
parents:
diff changeset
988 Sorting.Sort (Integer (SCO_Unit_Table.Last));
kono
parents:
diff changeset
989 end Unit_Table_Sort;
kono
parents:
diff changeset
990
kono
parents:
diff changeset
991 -- Loop through entries in the unit table to set file name and
kono
parents:
diff changeset
992 -- dependency number entries.
kono
parents:
diff changeset
993
kono
parents:
diff changeset
994 for J in 1 .. SCO_Unit_Table.Last loop
kono
parents:
diff changeset
995 declare
kono
parents:
diff changeset
996 U : constant Unit_Number_Type := SCO_Unit_Number_Table.Table (J);
kono
parents:
diff changeset
997 UTE : SCO_Unit_Table_Entry renames SCO_Unit_Table.Table (J);
kono
parents:
diff changeset
998
kono
parents:
diff changeset
999 begin
kono
parents:
diff changeset
1000 Get_Name_String (Reference_Name (Source_Index (U)));
kono
parents:
diff changeset
1001 UTE.File_Name := new String'(Name_Buffer (1 .. Name_Len));
kono
parents:
diff changeset
1002 UTE.Dep_Num := Dependency_Num (U);
kono
parents:
diff changeset
1003 end;
kono
parents:
diff changeset
1004 end loop;
kono
parents:
diff changeset
1005
kono
parents:
diff changeset
1006 -- Now the tables are all setup for output to the ALI file
kono
parents:
diff changeset
1007
kono
parents:
diff changeset
1008 Write_SCOs_To_ALI_File;
kono
parents:
diff changeset
1009 end SCO_Output;
kono
parents:
diff changeset
1010
kono
parents:
diff changeset
1011 -------------------------
kono
parents:
diff changeset
1012 -- SCO_Pragma_Disabled --
kono
parents:
diff changeset
1013 -------------------------
kono
parents:
diff changeset
1014
kono
parents:
diff changeset
1015 function SCO_Pragma_Disabled (Loc : Source_Ptr) return Boolean is
kono
parents:
diff changeset
1016 Index : Nat;
kono
parents:
diff changeset
1017
kono
parents:
diff changeset
1018 begin
kono
parents:
diff changeset
1019 if Loc = No_Location then
kono
parents:
diff changeset
1020 return False;
kono
parents:
diff changeset
1021 end if;
kono
parents:
diff changeset
1022
kono
parents:
diff changeset
1023 Index := SCO_Raw_Hash_Table.Get (Loc);
kono
parents:
diff changeset
1024
kono
parents:
diff changeset
1025 -- The test here for zero is to deal with possible previous errors, and
kono
parents:
diff changeset
1026 -- for the case of pragma statement SCOs, for which we always set the
kono
parents:
diff changeset
1027 -- Pragma_Sloc even if the particular pragma cannot be specifically
kono
parents:
diff changeset
1028 -- disabled.
kono
parents:
diff changeset
1029
kono
parents:
diff changeset
1030 if Index /= 0 then
kono
parents:
diff changeset
1031 declare
kono
parents:
diff changeset
1032 T : SCO_Table_Entry renames SCO_Raw_Table.Table (Index);
kono
parents:
diff changeset
1033
kono
parents:
diff changeset
1034 begin
kono
parents:
diff changeset
1035 case T.C1 is
kono
parents:
diff changeset
1036 when 'S' =>
kono
parents:
diff changeset
1037 -- Pragma statement
kono
parents:
diff changeset
1038
kono
parents:
diff changeset
1039 return T.C2 = 'p';
kono
parents:
diff changeset
1040
kono
parents:
diff changeset
1041 when 'A' =>
kono
parents:
diff changeset
1042 -- Aspect decision (enabled)
kono
parents:
diff changeset
1043
kono
parents:
diff changeset
1044 return False;
kono
parents:
diff changeset
1045
kono
parents:
diff changeset
1046 when 'a' =>
kono
parents:
diff changeset
1047 -- Aspect decision (not enabled)
kono
parents:
diff changeset
1048
kono
parents:
diff changeset
1049 return True;
kono
parents:
diff changeset
1050
kono
parents:
diff changeset
1051 when ASCII.NUL =>
kono
parents:
diff changeset
1052 -- Nullified disabled SCO
kono
parents:
diff changeset
1053
kono
parents:
diff changeset
1054 return True;
kono
parents:
diff changeset
1055
kono
parents:
diff changeset
1056 when others =>
kono
parents:
diff changeset
1057 raise Program_Error;
kono
parents:
diff changeset
1058 end case;
kono
parents:
diff changeset
1059 end;
kono
parents:
diff changeset
1060
kono
parents:
diff changeset
1061 else
kono
parents:
diff changeset
1062 return False;
kono
parents:
diff changeset
1063 end if;
kono
parents:
diff changeset
1064 end SCO_Pragma_Disabled;
kono
parents:
diff changeset
1065
kono
parents:
diff changeset
1066 --------------------
kono
parents:
diff changeset
1067 -- SCO_Record_Raw --
kono
parents:
diff changeset
1068 --------------------
kono
parents:
diff changeset
1069
kono
parents:
diff changeset
1070 procedure SCO_Record_Raw (U : Unit_Number_Type) is
kono
parents:
diff changeset
1071 procedure Traverse_Aux_Decls (N : Node_Id);
kono
parents:
diff changeset
1072 -- Traverse the Aux_Decls_Node of compilation unit N
kono
parents:
diff changeset
1073
kono
parents:
diff changeset
1074 ------------------------
kono
parents:
diff changeset
1075 -- Traverse_Aux_Decls --
kono
parents:
diff changeset
1076 ------------------------
kono
parents:
diff changeset
1077
kono
parents:
diff changeset
1078 procedure Traverse_Aux_Decls (N : Node_Id) is
kono
parents:
diff changeset
1079 ADN : constant Node_Id := Aux_Decls_Node (N);
kono
parents:
diff changeset
1080
kono
parents:
diff changeset
1081 begin
kono
parents:
diff changeset
1082 Traverse_Declarations_Or_Statements (Config_Pragmas (ADN));
kono
parents:
diff changeset
1083 Traverse_Declarations_Or_Statements (Pragmas_After (ADN));
kono
parents:
diff changeset
1084
kono
parents:
diff changeset
1085 -- Declarations and Actions do not correspond to source constructs,
kono
parents:
diff changeset
1086 -- they contain only nodes from expansion, so at this point they
kono
parents:
diff changeset
1087 -- should still be empty:
kono
parents:
diff changeset
1088
kono
parents:
diff changeset
1089 pragma Assert (No (Declarations (ADN)));
kono
parents:
diff changeset
1090 pragma Assert (No (Actions (ADN)));
kono
parents:
diff changeset
1091 end Traverse_Aux_Decls;
kono
parents:
diff changeset
1092
kono
parents:
diff changeset
1093 -- Local variables
kono
parents:
diff changeset
1094
kono
parents:
diff changeset
1095 From : Nat;
kono
parents:
diff changeset
1096 Lu : Node_Id;
kono
parents:
diff changeset
1097
kono
parents:
diff changeset
1098 -- Start of processing for SCO_Record_Raw
kono
parents:
diff changeset
1099
kono
parents:
diff changeset
1100 begin
kono
parents:
diff changeset
1101 -- It is legitimate to run this pass multiple times (once per unit) so
kono
parents:
diff changeset
1102 -- run it even if it was already run before.
kono
parents:
diff changeset
1103
kono
parents:
diff changeset
1104 pragma Assert (SCO_Generation_State in None .. Raw);
kono
parents:
diff changeset
1105 SCO_Generation_State := Raw;
kono
parents:
diff changeset
1106
kono
parents:
diff changeset
1107 -- Ignore call if not generating code and generating SCO's
kono
parents:
diff changeset
1108
kono
parents:
diff changeset
1109 if not (Generate_SCO and then Operating_Mode = Generate_Code) then
kono
parents:
diff changeset
1110 return;
kono
parents:
diff changeset
1111 end if;
kono
parents:
diff changeset
1112
kono
parents:
diff changeset
1113 -- Ignore call if this unit already recorded
kono
parents:
diff changeset
1114
kono
parents:
diff changeset
1115 for J in 1 .. SCO_Unit_Number_Table.Last loop
kono
parents:
diff changeset
1116 if U = SCO_Unit_Number_Table.Table (J) then
kono
parents:
diff changeset
1117 return;
kono
parents:
diff changeset
1118 end if;
kono
parents:
diff changeset
1119 end loop;
kono
parents:
diff changeset
1120
kono
parents:
diff changeset
1121 -- Otherwise record starting entry
kono
parents:
diff changeset
1122
kono
parents:
diff changeset
1123 From := SCO_Raw_Table.Last + 1;
kono
parents:
diff changeset
1124
kono
parents:
diff changeset
1125 -- Get Unit (checking case of subunit)
kono
parents:
diff changeset
1126
kono
parents:
diff changeset
1127 Lu := Unit (Cunit (U));
kono
parents:
diff changeset
1128
kono
parents:
diff changeset
1129 if Nkind (Lu) = N_Subunit then
kono
parents:
diff changeset
1130 Lu := Proper_Body (Lu);
kono
parents:
diff changeset
1131 end if;
kono
parents:
diff changeset
1132
kono
parents:
diff changeset
1133 -- Traverse the unit
kono
parents:
diff changeset
1134
kono
parents:
diff changeset
1135 Traverse_Aux_Decls (Cunit (U));
kono
parents:
diff changeset
1136
kono
parents:
diff changeset
1137 case Nkind (Lu) is
kono
parents:
diff changeset
1138 when N_Generic_Instantiation
kono
parents:
diff changeset
1139 | N_Generic_Package_Declaration
kono
parents:
diff changeset
1140 | N_Package_Body
kono
parents:
diff changeset
1141 | N_Package_Declaration
kono
parents:
diff changeset
1142 | N_Protected_Body
kono
parents:
diff changeset
1143 | N_Subprogram_Body
kono
parents:
diff changeset
1144 | N_Subprogram_Declaration
kono
parents:
diff changeset
1145 | N_Task_Body
kono
parents:
diff changeset
1146 =>
kono
parents:
diff changeset
1147 Traverse_Declarations_Or_Statements (L => No_List, P => Lu);
kono
parents:
diff changeset
1148
kono
parents:
diff changeset
1149 -- All other cases of compilation units (e.g. renamings), generate no
kono
parents:
diff changeset
1150 -- SCO information.
kono
parents:
diff changeset
1151
kono
parents:
diff changeset
1152 when others =>
kono
parents:
diff changeset
1153 null;
kono
parents:
diff changeset
1154 end case;
kono
parents:
diff changeset
1155
kono
parents:
diff changeset
1156 -- Make entry for new unit in unit tables, we will fill in the file
kono
parents:
diff changeset
1157 -- name and dependency numbers later.
kono
parents:
diff changeset
1158
kono
parents:
diff changeset
1159 SCO_Unit_Table.Append (
kono
parents:
diff changeset
1160 (Dep_Num => 0,
kono
parents:
diff changeset
1161 File_Name => null,
kono
parents:
diff changeset
1162 File_Index => Get_Source_File_Index (Sloc (Lu)),
kono
parents:
diff changeset
1163 From => From,
kono
parents:
diff changeset
1164 To => SCO_Raw_Table.Last));
kono
parents:
diff changeset
1165
kono
parents:
diff changeset
1166 SCO_Unit_Number_Table.Append (U);
kono
parents:
diff changeset
1167 end SCO_Record_Raw;
kono
parents:
diff changeset
1168
kono
parents:
diff changeset
1169 -----------------------
kono
parents:
diff changeset
1170 -- Set_SCO_Condition --
kono
parents:
diff changeset
1171 -----------------------
kono
parents:
diff changeset
1172
kono
parents:
diff changeset
1173 procedure Set_SCO_Condition (Cond : Node_Id; Val : Boolean) is
kono
parents:
diff changeset
1174
kono
parents:
diff changeset
1175 -- SCO annotations are not processed after the filtering pass
kono
parents:
diff changeset
1176
kono
parents:
diff changeset
1177 pragma Assert (not Generate_SCO or else SCO_Generation_State = Raw);
kono
parents:
diff changeset
1178
kono
parents:
diff changeset
1179 Constant_Condition_Code : constant array (Boolean) of Character :=
kono
parents:
diff changeset
1180 (False => 'f', True => 't');
kono
parents:
diff changeset
1181
kono
parents:
diff changeset
1182 Orig : constant Node_Id := Original_Node (Cond);
kono
parents:
diff changeset
1183 Dummy : Source_Ptr;
kono
parents:
diff changeset
1184 Index : Nat;
kono
parents:
diff changeset
1185 Start : Source_Ptr;
kono
parents:
diff changeset
1186
kono
parents:
diff changeset
1187 begin
kono
parents:
diff changeset
1188 Sloc_Range (Orig, Start, Dummy);
kono
parents:
diff changeset
1189 Index := SCO_Raw_Hash_Table.Get (Start);
kono
parents:
diff changeset
1190
kono
parents:
diff changeset
1191 -- Index can be zero for boolean expressions that do not have SCOs
kono
parents:
diff changeset
1192 -- (simple decisions outside of a control flow structure), or in case
kono
parents:
diff changeset
1193 -- of a previous error.
kono
parents:
diff changeset
1194
kono
parents:
diff changeset
1195 if Index = 0 then
kono
parents:
diff changeset
1196 return;
kono
parents:
diff changeset
1197
kono
parents:
diff changeset
1198 else
kono
parents:
diff changeset
1199 pragma Assert (SCO_Raw_Table.Table (Index).C1 = ' ');
kono
parents:
diff changeset
1200 SCO_Raw_Table.Table (Index).C2 := Constant_Condition_Code (Val);
kono
parents:
diff changeset
1201 end if;
kono
parents:
diff changeset
1202 end Set_SCO_Condition;
kono
parents:
diff changeset
1203
kono
parents:
diff changeset
1204 ------------------------------
kono
parents:
diff changeset
1205 -- Set_SCO_Logical_Operator --
kono
parents:
diff changeset
1206 ------------------------------
kono
parents:
diff changeset
1207
kono
parents:
diff changeset
1208 procedure Set_SCO_Logical_Operator (Op : Node_Id) is
kono
parents:
diff changeset
1209
kono
parents:
diff changeset
1210 -- SCO annotations are not processed after the filtering pass
kono
parents:
diff changeset
1211
kono
parents:
diff changeset
1212 pragma Assert (not Generate_SCO or else SCO_Generation_State = Raw);
kono
parents:
diff changeset
1213
kono
parents:
diff changeset
1214 Orig : constant Node_Id := Original_Node (Op);
kono
parents:
diff changeset
1215 Orig_Sloc : constant Source_Ptr := Sloc (Orig);
kono
parents:
diff changeset
1216 Index : constant Nat := SCO_Raw_Hash_Table.Get (Orig_Sloc);
kono
parents:
diff changeset
1217
kono
parents:
diff changeset
1218 begin
kono
parents:
diff changeset
1219 -- All (putative) logical operators are supposed to have their own entry
kono
parents:
diff changeset
1220 -- in the SCOs table. However, the semantic analysis may invoke this
kono
parents:
diff changeset
1221 -- subprogram with nodes that are out of the SCO generation scope.
kono
parents:
diff changeset
1222
kono
parents:
diff changeset
1223 if Index /= 0 then
kono
parents:
diff changeset
1224 SCO_Raw_Table.Table (Index).C2 := ' ';
kono
parents:
diff changeset
1225 end if;
kono
parents:
diff changeset
1226 end Set_SCO_Logical_Operator;
kono
parents:
diff changeset
1227
kono
parents:
diff changeset
1228 ----------------------------
kono
parents:
diff changeset
1229 -- Set_SCO_Pragma_Enabled --
kono
parents:
diff changeset
1230 ----------------------------
kono
parents:
diff changeset
1231
kono
parents:
diff changeset
1232 procedure Set_SCO_Pragma_Enabled (Loc : Source_Ptr) is
kono
parents:
diff changeset
1233
kono
parents:
diff changeset
1234 -- SCO annotations are not processed after the filtering pass
kono
parents:
diff changeset
1235
kono
parents:
diff changeset
1236 pragma Assert (not Generate_SCO or else SCO_Generation_State = Raw);
kono
parents:
diff changeset
1237
kono
parents:
diff changeset
1238 Index : Nat;
kono
parents:
diff changeset
1239
kono
parents:
diff changeset
1240 begin
kono
parents:
diff changeset
1241 -- Nothing to do if not generating SCO, or if we're not processing the
kono
parents:
diff changeset
1242 -- original source occurrence of the pragma.
kono
parents:
diff changeset
1243
kono
parents:
diff changeset
1244 if not (Generate_SCO
kono
parents:
diff changeset
1245 and then In_Extended_Main_Source_Unit (Loc)
kono
parents:
diff changeset
1246 and then not (In_Instance or In_Inlined_Body))
kono
parents:
diff changeset
1247 then
kono
parents:
diff changeset
1248 return;
kono
parents:
diff changeset
1249 end if;
kono
parents:
diff changeset
1250
kono
parents:
diff changeset
1251 -- Note: the reason we use the Sloc value as the key is that in the
kono
parents:
diff changeset
1252 -- generic case, the call to this procedure is made on a copy of the
kono
parents:
diff changeset
1253 -- original node, so we can't use the Node_Id value.
kono
parents:
diff changeset
1254
kono
parents:
diff changeset
1255 Index := SCO_Raw_Hash_Table.Get (Loc);
kono
parents:
diff changeset
1256
kono
parents:
diff changeset
1257 -- A zero index here indicates that semantic analysis found an
kono
parents:
diff changeset
1258 -- activated pragma at Loc which does not have a corresponding pragma
kono
parents:
diff changeset
1259 -- or aspect at the syntax level. This may occur in legitimate cases
kono
parents:
diff changeset
1260 -- because of expanded code (such are Pre/Post conditions generated for
kono
parents:
diff changeset
1261 -- formal parameter validity checks), or as a consequence of a previous
kono
parents:
diff changeset
1262 -- error.
kono
parents:
diff changeset
1263
kono
parents:
diff changeset
1264 if Index = 0 then
kono
parents:
diff changeset
1265 return;
kono
parents:
diff changeset
1266
kono
parents:
diff changeset
1267 else
kono
parents:
diff changeset
1268 declare
kono
parents:
diff changeset
1269 T : SCO_Table_Entry renames SCO_Raw_Table.Table (Index);
kono
parents:
diff changeset
1270
kono
parents:
diff changeset
1271 begin
kono
parents:
diff changeset
1272 -- Note: may be called multiple times for the same sloc, so
kono
parents:
diff changeset
1273 -- account for the fact that the entry may already have been
kono
parents:
diff changeset
1274 -- marked enabled.
kono
parents:
diff changeset
1275
kono
parents:
diff changeset
1276 case T.C1 is
kono
parents:
diff changeset
1277 -- Aspect (decision SCO)
kono
parents:
diff changeset
1278
kono
parents:
diff changeset
1279 when 'a' =>
kono
parents:
diff changeset
1280 T.C1 := 'A';
kono
parents:
diff changeset
1281
kono
parents:
diff changeset
1282 when 'A' =>
kono
parents:
diff changeset
1283 null;
kono
parents:
diff changeset
1284
kono
parents:
diff changeset
1285 -- Pragma (statement SCO)
kono
parents:
diff changeset
1286
kono
parents:
diff changeset
1287 when 'S' =>
kono
parents:
diff changeset
1288 pragma Assert (T.C2 = 'p' or else T.C2 = 'P');
kono
parents:
diff changeset
1289 T.C2 := 'P';
kono
parents:
diff changeset
1290
kono
parents:
diff changeset
1291 when others =>
kono
parents:
diff changeset
1292 raise Program_Error;
kono
parents:
diff changeset
1293 end case;
kono
parents:
diff changeset
1294 end;
kono
parents:
diff changeset
1295 end if;
kono
parents:
diff changeset
1296 end Set_SCO_Pragma_Enabled;
kono
parents:
diff changeset
1297
kono
parents:
diff changeset
1298 -------------------------
kono
parents:
diff changeset
1299 -- Set_Raw_Table_Entry --
kono
parents:
diff changeset
1300 -------------------------
kono
parents:
diff changeset
1301
kono
parents:
diff changeset
1302 procedure Set_Raw_Table_Entry
kono
parents:
diff changeset
1303 (C1 : Character;
kono
parents:
diff changeset
1304 C2 : Character;
kono
parents:
diff changeset
1305 From : Source_Ptr;
kono
parents:
diff changeset
1306 To : Source_Ptr;
kono
parents:
diff changeset
1307 Last : Boolean;
kono
parents:
diff changeset
1308 Pragma_Sloc : Source_Ptr := No_Location;
kono
parents:
diff changeset
1309 Pragma_Aspect_Name : Name_Id := No_Name)
kono
parents:
diff changeset
1310 is
kono
parents:
diff changeset
1311 pragma Assert (SCO_Generation_State = Raw);
kono
parents:
diff changeset
1312 begin
kono
parents:
diff changeset
1313 SCO_Raw_Table.Append
kono
parents:
diff changeset
1314 ((C1 => C1,
kono
parents:
diff changeset
1315 C2 => C2,
kono
parents:
diff changeset
1316 From => To_Source_Location (From),
kono
parents:
diff changeset
1317 To => To_Source_Location (To),
kono
parents:
diff changeset
1318 Last => Last,
kono
parents:
diff changeset
1319 Pragma_Sloc => Pragma_Sloc,
kono
parents:
diff changeset
1320 Pragma_Aspect_Name => Pragma_Aspect_Name));
kono
parents:
diff changeset
1321 end Set_Raw_Table_Entry;
kono
parents:
diff changeset
1322
kono
parents:
diff changeset
1323 ------------------------
kono
parents:
diff changeset
1324 -- To_Source_Location --
kono
parents:
diff changeset
1325 ------------------------
kono
parents:
diff changeset
1326
kono
parents:
diff changeset
1327 function To_Source_Location (S : Source_Ptr) return Source_Location is
kono
parents:
diff changeset
1328 begin
kono
parents:
diff changeset
1329 if S = No_Location then
kono
parents:
diff changeset
1330 return No_Source_Location;
kono
parents:
diff changeset
1331 else
kono
parents:
diff changeset
1332 return
kono
parents:
diff changeset
1333 (Line => Get_Logical_Line_Number (S),
kono
parents:
diff changeset
1334 Col => Get_Column_Number (S));
kono
parents:
diff changeset
1335 end if;
kono
parents:
diff changeset
1336 end To_Source_Location;
kono
parents:
diff changeset
1337
kono
parents:
diff changeset
1338 -----------------------------------------
kono
parents:
diff changeset
1339 -- Traverse_Declarations_Or_Statements --
kono
parents:
diff changeset
1340 -----------------------------------------
kono
parents:
diff changeset
1341
kono
parents:
diff changeset
1342 -- Tables used by Traverse_Declarations_Or_Statements for temporarily
kono
parents:
diff changeset
1343 -- holding statement and decision entries. These are declared globally
kono
parents:
diff changeset
1344 -- since they are shared by recursive calls to this procedure.
kono
parents:
diff changeset
1345
kono
parents:
diff changeset
1346 type SC_Entry is record
kono
parents:
diff changeset
1347 N : Node_Id;
kono
parents:
diff changeset
1348 From : Source_Ptr;
kono
parents:
diff changeset
1349 To : Source_Ptr;
kono
parents:
diff changeset
1350 Typ : Character;
kono
parents:
diff changeset
1351 end record;
kono
parents:
diff changeset
1352 -- Used to store a single entry in the following table, From:To represents
kono
parents:
diff changeset
1353 -- the range of entries in the CS line entry, and typ is the type, with
kono
parents:
diff changeset
1354 -- space meaning that no type letter will accompany the entry.
kono
parents:
diff changeset
1355
kono
parents:
diff changeset
1356 package SC is new Table.Table
kono
parents:
diff changeset
1357 (Table_Component_Type => SC_Entry,
kono
parents:
diff changeset
1358 Table_Index_Type => Nat,
kono
parents:
diff changeset
1359 Table_Low_Bound => 1,
kono
parents:
diff changeset
1360 Table_Initial => 1000,
kono
parents:
diff changeset
1361 Table_Increment => 200,
kono
parents:
diff changeset
1362 Table_Name => "SCO_SC");
kono
parents:
diff changeset
1363 -- Used to store statement components for a CS entry to be output as a
kono
parents:
diff changeset
1364 -- result of the call to this procedure. SC.Last is the last entry stored,
kono
parents:
diff changeset
1365 -- so the current statement sequence is represented by SC_Array (SC_First
kono
parents:
diff changeset
1366 -- .. SC.Last), where SC_First is saved on entry to each recursive call to
kono
parents:
diff changeset
1367 -- the routine.
kono
parents:
diff changeset
1368 --
kono
parents:
diff changeset
1369 -- Extend_Statement_Sequence adds an entry to this array, and then
kono
parents:
diff changeset
1370 -- Set_Statement_Entry clears the entries starting with SC_First, copying
kono
parents:
diff changeset
1371 -- these entries to the main SCO output table. The reason that we do the
kono
parents:
diff changeset
1372 -- temporary caching of results in this array is that we want the SCO table
kono
parents:
diff changeset
1373 -- entries for a given CS line to be contiguous, and the processing may
kono
parents:
diff changeset
1374 -- output intermediate entries such as decision entries.
kono
parents:
diff changeset
1375
kono
parents:
diff changeset
1376 type SD_Entry is record
kono
parents:
diff changeset
1377 Nod : Node_Id;
kono
parents:
diff changeset
1378 Lst : List_Id;
kono
parents:
diff changeset
1379 Typ : Character;
kono
parents:
diff changeset
1380 Plo : Source_Ptr;
kono
parents:
diff changeset
1381 end record;
kono
parents:
diff changeset
1382 -- Used to store a single entry in the following table. Nod is the node to
kono
parents:
diff changeset
1383 -- be searched for decisions for the case of Process_Decisions_Defer with a
kono
parents:
diff changeset
1384 -- node argument (with Lst set to No_List. Lst is the list to be searched
kono
parents:
diff changeset
1385 -- for decisions for the case of Process_Decisions_Defer with a List
kono
parents:
diff changeset
1386 -- argument (in which case Nod is set to Empty). Plo is the sloc of the
kono
parents:
diff changeset
1387 -- enclosing pragma, if any.
kono
parents:
diff changeset
1388
kono
parents:
diff changeset
1389 package SD is new Table.Table
kono
parents:
diff changeset
1390 (Table_Component_Type => SD_Entry,
kono
parents:
diff changeset
1391 Table_Index_Type => Nat,
kono
parents:
diff changeset
1392 Table_Low_Bound => 1,
kono
parents:
diff changeset
1393 Table_Initial => 1000,
kono
parents:
diff changeset
1394 Table_Increment => 200,
kono
parents:
diff changeset
1395 Table_Name => "SCO_SD");
kono
parents:
diff changeset
1396 -- Used to store possible decision information. Instead of calling the
kono
parents:
diff changeset
1397 -- Process_Decisions procedures directly, we call Process_Decisions_Defer,
kono
parents:
diff changeset
1398 -- which simply stores the arguments in this table. Then when we clear
kono
parents:
diff changeset
1399 -- out a statement sequence using Set_Statement_Entry, after generating
kono
parents:
diff changeset
1400 -- the CS lines for the statements, the entries in this table result in
kono
parents:
diff changeset
1401 -- calls to Process_Decision. The reason for doing things this way is to
kono
parents:
diff changeset
1402 -- ensure that decisions are output after the CS line for the statements
kono
parents:
diff changeset
1403 -- in which the decisions occur.
kono
parents:
diff changeset
1404
kono
parents:
diff changeset
1405 procedure Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
1406 (L : List_Id;
kono
parents:
diff changeset
1407 D : Dominant_Info := No_Dominant;
kono
parents:
diff changeset
1408 P : Node_Id := Empty)
kono
parents:
diff changeset
1409 is
kono
parents:
diff changeset
1410 Discard_Dom : Dominant_Info;
kono
parents:
diff changeset
1411 pragma Warnings (Off, Discard_Dom);
kono
parents:
diff changeset
1412 begin
kono
parents:
diff changeset
1413 Discard_Dom := Traverse_Declarations_Or_Statements (L, D, P);
kono
parents:
diff changeset
1414 end Traverse_Declarations_Or_Statements;
kono
parents:
diff changeset
1415
kono
parents:
diff changeset
1416 function Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
1417 (L : List_Id;
kono
parents:
diff changeset
1418 D : Dominant_Info := No_Dominant;
kono
parents:
diff changeset
1419 P : Node_Id := Empty) return Dominant_Info
kono
parents:
diff changeset
1420 is
kono
parents:
diff changeset
1421 Current_Dominant : Dominant_Info := D;
kono
parents:
diff changeset
1422 -- Dominance information for the current basic block
kono
parents:
diff changeset
1423
kono
parents:
diff changeset
1424 Current_Test : Node_Id;
kono
parents:
diff changeset
1425 -- Conditional node (N_If_Statement or N_Elsiif being processed
kono
parents:
diff changeset
1426
kono
parents:
diff changeset
1427 N : Node_Id;
kono
parents:
diff changeset
1428
kono
parents:
diff changeset
1429 SC_First : constant Nat := SC.Last + 1;
kono
parents:
diff changeset
1430 SD_First : constant Nat := SD.Last + 1;
kono
parents:
diff changeset
1431 -- Record first entries used in SC/SD at this recursive level
kono
parents:
diff changeset
1432
kono
parents:
diff changeset
1433 procedure Extend_Statement_Sequence (N : Node_Id; Typ : Character);
kono
parents:
diff changeset
1434 -- Extend the current statement sequence to encompass the node N. Typ is
kono
parents:
diff changeset
1435 -- the letter that identifies the type of statement/declaration that is
kono
parents:
diff changeset
1436 -- being added to the sequence.
kono
parents:
diff changeset
1437
kono
parents:
diff changeset
1438 procedure Process_Decisions_Defer (N : Node_Id; T : Character);
kono
parents:
diff changeset
1439 pragma Inline (Process_Decisions_Defer);
kono
parents:
diff changeset
1440 -- This routine is logically the same as Process_Decisions, except that
kono
parents:
diff changeset
1441 -- the arguments are saved in the SD table for later processing when
kono
parents:
diff changeset
1442 -- Set_Statement_Entry is called, which goes through the saved entries
kono
parents:
diff changeset
1443 -- making the corresponding calls to Process_Decision. Note: the
kono
parents:
diff changeset
1444 -- enclosing statement must have already been added to the current
kono
parents:
diff changeset
1445 -- statement sequence, so that nested decisions are properly
kono
parents:
diff changeset
1446 -- identified as such.
kono
parents:
diff changeset
1447
kono
parents:
diff changeset
1448 procedure Process_Decisions_Defer (L : List_Id; T : Character);
kono
parents:
diff changeset
1449 pragma Inline (Process_Decisions_Defer);
kono
parents:
diff changeset
1450 -- Same case for list arguments, deferred call to Process_Decisions
kono
parents:
diff changeset
1451
kono
parents:
diff changeset
1452 procedure Set_Statement_Entry;
kono
parents:
diff changeset
1453 -- Output CS entries for all statements saved in table SC, and end the
kono
parents:
diff changeset
1454 -- current CS sequence. Then output entries for all decisions nested in
kono
parents:
diff changeset
1455 -- these statements, which have been deferred so far.
kono
parents:
diff changeset
1456
kono
parents:
diff changeset
1457 procedure Traverse_One (N : Node_Id);
kono
parents:
diff changeset
1458 -- Traverse one declaration or statement
kono
parents:
diff changeset
1459
kono
parents:
diff changeset
1460 procedure Traverse_Aspects (N : Node_Id);
kono
parents:
diff changeset
1461 -- Helper for Traverse_One: traverse N's aspect specifications
kono
parents:
diff changeset
1462
kono
parents:
diff changeset
1463 procedure Traverse_Degenerate_Subprogram (N : Node_Id);
kono
parents:
diff changeset
1464 -- Common code to handle null procedures and expression functions. Emit
kono
parents:
diff changeset
1465 -- a SCO of the given Kind and N outside of the dominance flow.
kono
parents:
diff changeset
1466
kono
parents:
diff changeset
1467 -------------------------------
kono
parents:
diff changeset
1468 -- Extend_Statement_Sequence --
kono
parents:
diff changeset
1469 -------------------------------
kono
parents:
diff changeset
1470
kono
parents:
diff changeset
1471 procedure Extend_Statement_Sequence (N : Node_Id; Typ : Character) is
kono
parents:
diff changeset
1472 Dummy : Source_Ptr;
kono
parents:
diff changeset
1473 F : Source_Ptr;
kono
parents:
diff changeset
1474 T : Source_Ptr;
kono
parents:
diff changeset
1475 To_Node : Node_Id := Empty;
kono
parents:
diff changeset
1476
kono
parents:
diff changeset
1477 begin
kono
parents:
diff changeset
1478 Sloc_Range (N, F, T);
kono
parents:
diff changeset
1479
kono
parents:
diff changeset
1480 case Nkind (N) is
kono
parents:
diff changeset
1481 when N_Accept_Statement =>
kono
parents:
diff changeset
1482 if Present (Parameter_Specifications (N)) then
kono
parents:
diff changeset
1483 To_Node := Last (Parameter_Specifications (N));
kono
parents:
diff changeset
1484 elsif Present (Entry_Index (N)) then
kono
parents:
diff changeset
1485 To_Node := Entry_Index (N);
kono
parents:
diff changeset
1486 else
kono
parents:
diff changeset
1487 To_Node := Entry_Direct_Name (N);
kono
parents:
diff changeset
1488 end if;
kono
parents:
diff changeset
1489
kono
parents:
diff changeset
1490 when N_Case_Statement =>
kono
parents:
diff changeset
1491 To_Node := Expression (N);
kono
parents:
diff changeset
1492
kono
parents:
diff changeset
1493 when N_Elsif_Part
kono
parents:
diff changeset
1494 | N_If_Statement
kono
parents:
diff changeset
1495 =>
kono
parents:
diff changeset
1496 To_Node := Condition (N);
kono
parents:
diff changeset
1497
kono
parents:
diff changeset
1498 when N_Extended_Return_Statement =>
kono
parents:
diff changeset
1499 To_Node := Last (Return_Object_Declarations (N));
kono
parents:
diff changeset
1500
kono
parents:
diff changeset
1501 when N_Loop_Statement =>
kono
parents:
diff changeset
1502 To_Node := Iteration_Scheme (N);
kono
parents:
diff changeset
1503
kono
parents:
diff changeset
1504 when N_Asynchronous_Select
kono
parents:
diff changeset
1505 | N_Conditional_Entry_Call
kono
parents:
diff changeset
1506 | N_Selective_Accept
kono
parents:
diff changeset
1507 | N_Single_Protected_Declaration
kono
parents:
diff changeset
1508 | N_Single_Task_Declaration
kono
parents:
diff changeset
1509 | N_Timed_Entry_Call
kono
parents:
diff changeset
1510 =>
kono
parents:
diff changeset
1511 T := F;
kono
parents:
diff changeset
1512
kono
parents:
diff changeset
1513 when N_Protected_Type_Declaration
kono
parents:
diff changeset
1514 | N_Task_Type_Declaration
kono
parents:
diff changeset
1515 =>
kono
parents:
diff changeset
1516 if Has_Aspects (N) then
kono
parents:
diff changeset
1517 To_Node := Last (Aspect_Specifications (N));
kono
parents:
diff changeset
1518
kono
parents:
diff changeset
1519 elsif Present (Discriminant_Specifications (N)) then
kono
parents:
diff changeset
1520 To_Node := Last (Discriminant_Specifications (N));
kono
parents:
diff changeset
1521
kono
parents:
diff changeset
1522 else
kono
parents:
diff changeset
1523 To_Node := Defining_Identifier (N);
kono
parents:
diff changeset
1524 end if;
kono
parents:
diff changeset
1525
kono
parents:
diff changeset
1526 when N_Subexpr =>
kono
parents:
diff changeset
1527 To_Node := N;
kono
parents:
diff changeset
1528
kono
parents:
diff changeset
1529 when others =>
kono
parents:
diff changeset
1530 null;
kono
parents:
diff changeset
1531 end case;
kono
parents:
diff changeset
1532
kono
parents:
diff changeset
1533 if Present (To_Node) then
kono
parents:
diff changeset
1534 Sloc_Range (To_Node, Dummy, T);
kono
parents:
diff changeset
1535 end if;
kono
parents:
diff changeset
1536
kono
parents:
diff changeset
1537 SC.Append ((N, F, T, Typ));
kono
parents:
diff changeset
1538 end Extend_Statement_Sequence;
kono
parents:
diff changeset
1539
kono
parents:
diff changeset
1540 -----------------------------
kono
parents:
diff changeset
1541 -- Process_Decisions_Defer --
kono
parents:
diff changeset
1542 -----------------------------
kono
parents:
diff changeset
1543
kono
parents:
diff changeset
1544 procedure Process_Decisions_Defer (N : Node_Id; T : Character) is
kono
parents:
diff changeset
1545 begin
kono
parents:
diff changeset
1546 SD.Append ((N, No_List, T, Current_Pragma_Sloc));
kono
parents:
diff changeset
1547 end Process_Decisions_Defer;
kono
parents:
diff changeset
1548
kono
parents:
diff changeset
1549 procedure Process_Decisions_Defer (L : List_Id; T : Character) is
kono
parents:
diff changeset
1550 begin
kono
parents:
diff changeset
1551 SD.Append ((Empty, L, T, Current_Pragma_Sloc));
kono
parents:
diff changeset
1552 end Process_Decisions_Defer;
kono
parents:
diff changeset
1553
kono
parents:
diff changeset
1554 -------------------------
kono
parents:
diff changeset
1555 -- Set_Statement_Entry --
kono
parents:
diff changeset
1556 -------------------------
kono
parents:
diff changeset
1557
kono
parents:
diff changeset
1558 procedure Set_Statement_Entry is
kono
parents:
diff changeset
1559 SC_Last : constant Int := SC.Last;
kono
parents:
diff changeset
1560 SD_Last : constant Int := SD.Last;
kono
parents:
diff changeset
1561
kono
parents:
diff changeset
1562 begin
kono
parents:
diff changeset
1563 -- Output statement entries from saved entries in SC table
kono
parents:
diff changeset
1564
kono
parents:
diff changeset
1565 for J in SC_First .. SC_Last loop
kono
parents:
diff changeset
1566 if J = SC_First then
kono
parents:
diff changeset
1567
kono
parents:
diff changeset
1568 if Current_Dominant /= No_Dominant then
kono
parents:
diff changeset
1569 declare
kono
parents:
diff changeset
1570 From : Source_Ptr;
kono
parents:
diff changeset
1571 To : Source_Ptr;
kono
parents:
diff changeset
1572
kono
parents:
diff changeset
1573 begin
kono
parents:
diff changeset
1574 Sloc_Range (Current_Dominant.N, From, To);
kono
parents:
diff changeset
1575
kono
parents:
diff changeset
1576 if Current_Dominant.K /= 'E' then
kono
parents:
diff changeset
1577 To := No_Location;
kono
parents:
diff changeset
1578 end if;
kono
parents:
diff changeset
1579
kono
parents:
diff changeset
1580 Set_Raw_Table_Entry
kono
parents:
diff changeset
1581 (C1 => '>',
kono
parents:
diff changeset
1582 C2 => Current_Dominant.K,
kono
parents:
diff changeset
1583 From => From,
kono
parents:
diff changeset
1584 To => To,
kono
parents:
diff changeset
1585 Last => False,
kono
parents:
diff changeset
1586 Pragma_Sloc => No_Location,
kono
parents:
diff changeset
1587 Pragma_Aspect_Name => No_Name);
kono
parents:
diff changeset
1588 end;
kono
parents:
diff changeset
1589 end if;
kono
parents:
diff changeset
1590 end if;
kono
parents:
diff changeset
1591
kono
parents:
diff changeset
1592 declare
kono
parents:
diff changeset
1593 SCE : SC_Entry renames SC.Table (J);
kono
parents:
diff changeset
1594 Pragma_Sloc : Source_Ptr := No_Location;
kono
parents:
diff changeset
1595 Pragma_Aspect_Name : Name_Id := No_Name;
kono
parents:
diff changeset
1596
kono
parents:
diff changeset
1597 begin
kono
parents:
diff changeset
1598 -- For the case of a statement SCO for a pragma controlled by
kono
parents:
diff changeset
1599 -- Set_SCO_Pragma_Enabled, set Pragma_Sloc so that the SCO (and
kono
parents:
diff changeset
1600 -- those of any nested decision) is emitted only if the pragma
kono
parents:
diff changeset
1601 -- is enabled.
kono
parents:
diff changeset
1602
kono
parents:
diff changeset
1603 if SCE.Typ = 'p' then
kono
parents:
diff changeset
1604 Pragma_Sloc := SCE.From;
kono
parents:
diff changeset
1605 SCO_Raw_Hash_Table.Set
kono
parents:
diff changeset
1606 (Pragma_Sloc, SCO_Raw_Table.Last + 1);
kono
parents:
diff changeset
1607 Pragma_Aspect_Name := Pragma_Name_Unmapped (SCE.N);
kono
parents:
diff changeset
1608 pragma Assert (Pragma_Aspect_Name /= No_Name);
kono
parents:
diff changeset
1609
kono
parents:
diff changeset
1610 elsif SCE.Typ = 'P' then
kono
parents:
diff changeset
1611 Pragma_Aspect_Name := Pragma_Name_Unmapped (SCE.N);
kono
parents:
diff changeset
1612 pragma Assert (Pragma_Aspect_Name /= No_Name);
kono
parents:
diff changeset
1613 end if;
kono
parents:
diff changeset
1614
kono
parents:
diff changeset
1615 Set_Raw_Table_Entry
kono
parents:
diff changeset
1616 (C1 => 'S',
kono
parents:
diff changeset
1617 C2 => SCE.Typ,
kono
parents:
diff changeset
1618 From => SCE.From,
kono
parents:
diff changeset
1619 To => SCE.To,
kono
parents:
diff changeset
1620 Last => (J = SC_Last),
kono
parents:
diff changeset
1621 Pragma_Sloc => Pragma_Sloc,
kono
parents:
diff changeset
1622 Pragma_Aspect_Name => Pragma_Aspect_Name);
kono
parents:
diff changeset
1623 end;
kono
parents:
diff changeset
1624 end loop;
kono
parents:
diff changeset
1625
kono
parents:
diff changeset
1626 -- Last statement of basic block, if present, becomes new current
kono
parents:
diff changeset
1627 -- dominant.
kono
parents:
diff changeset
1628
kono
parents:
diff changeset
1629 if SC_Last >= SC_First then
kono
parents:
diff changeset
1630 Current_Dominant := ('S', SC.Table (SC_Last).N);
kono
parents:
diff changeset
1631 end if;
kono
parents:
diff changeset
1632
kono
parents:
diff changeset
1633 -- Clear out used section of SC table
kono
parents:
diff changeset
1634
kono
parents:
diff changeset
1635 SC.Set_Last (SC_First - 1);
kono
parents:
diff changeset
1636
kono
parents:
diff changeset
1637 -- Output any embedded decisions
kono
parents:
diff changeset
1638
kono
parents:
diff changeset
1639 for J in SD_First .. SD_Last loop
kono
parents:
diff changeset
1640 declare
kono
parents:
diff changeset
1641 SDE : SD_Entry renames SD.Table (J);
kono
parents:
diff changeset
1642
kono
parents:
diff changeset
1643 begin
kono
parents:
diff changeset
1644 if Present (SDE.Nod) then
kono
parents:
diff changeset
1645 Process_Decisions (SDE.Nod, SDE.Typ, SDE.Plo);
kono
parents:
diff changeset
1646 else
kono
parents:
diff changeset
1647 Process_Decisions (SDE.Lst, SDE.Typ, SDE.Plo);
kono
parents:
diff changeset
1648 end if;
kono
parents:
diff changeset
1649 end;
kono
parents:
diff changeset
1650 end loop;
kono
parents:
diff changeset
1651
kono
parents:
diff changeset
1652 -- Clear out used section of SD table
kono
parents:
diff changeset
1653
kono
parents:
diff changeset
1654 SD.Set_Last (SD_First - 1);
kono
parents:
diff changeset
1655 end Set_Statement_Entry;
kono
parents:
diff changeset
1656
kono
parents:
diff changeset
1657 ----------------------
kono
parents:
diff changeset
1658 -- Traverse_Aspects --
kono
parents:
diff changeset
1659 ----------------------
kono
parents:
diff changeset
1660
kono
parents:
diff changeset
1661 procedure Traverse_Aspects (N : Node_Id) is
kono
parents:
diff changeset
1662 AE : Node_Id;
kono
parents:
diff changeset
1663 AN : Node_Id;
kono
parents:
diff changeset
1664 C1 : Character;
kono
parents:
diff changeset
1665
kono
parents:
diff changeset
1666 begin
kono
parents:
diff changeset
1667 AN := First (Aspect_Specifications (N));
kono
parents:
diff changeset
1668 while Present (AN) loop
kono
parents:
diff changeset
1669 AE := Expression (AN);
kono
parents:
diff changeset
1670
kono
parents:
diff changeset
1671 -- SCOs are generated before semantic analysis/expansion:
kono
parents:
diff changeset
1672 -- PPCs are not split yet.
kono
parents:
diff changeset
1673
kono
parents:
diff changeset
1674 pragma Assert (not Split_PPC (AN));
kono
parents:
diff changeset
1675
kono
parents:
diff changeset
1676 C1 := ASCII.NUL;
kono
parents:
diff changeset
1677
kono
parents:
diff changeset
1678 case Get_Aspect_Id (AN) is
kono
parents:
diff changeset
1679
kono
parents:
diff changeset
1680 -- Aspects rewritten into pragmas controlled by a Check_Policy:
kono
parents:
diff changeset
1681 -- Current_Pragma_Sloc must be set to the sloc of the aspect
kono
parents:
diff changeset
1682 -- specification. The corresponding pragma will have the same
kono
parents:
diff changeset
1683 -- sloc. Note that Invariant, Pre, and Post will be enabled if
kono
parents:
diff changeset
1684 -- the policy is Check; on the other hand, predicate aspects
kono
parents:
diff changeset
1685 -- will be enabled for Check and Ignore (when Add_Predicate
kono
parents:
diff changeset
1686 -- is called) because the actual checks occur in client units.
kono
parents:
diff changeset
1687 -- When the assertion policy for Predicate is Disable, the
kono
parents:
diff changeset
1688 -- SCO remains disabled, because Add_Predicate is never called.
kono
parents:
diff changeset
1689
kono
parents:
diff changeset
1690 -- Pre/post can have checks in client units too because of
kono
parents:
diff changeset
1691 -- inheritance, so should they receive the same treatment???
kono
parents:
diff changeset
1692
kono
parents:
diff changeset
1693 when Aspect_Dynamic_Predicate
kono
parents:
diff changeset
1694 | Aspect_Invariant
kono
parents:
diff changeset
1695 | Aspect_Post
kono
parents:
diff changeset
1696 | Aspect_Postcondition
kono
parents:
diff changeset
1697 | Aspect_Pre
kono
parents:
diff changeset
1698 | Aspect_Precondition
kono
parents:
diff changeset
1699 | Aspect_Predicate
kono
parents:
diff changeset
1700 | Aspect_Static_Predicate
kono
parents:
diff changeset
1701 | Aspect_Type_Invariant
kono
parents:
diff changeset
1702 =>
kono
parents:
diff changeset
1703 C1 := 'a';
kono
parents:
diff changeset
1704
kono
parents:
diff changeset
1705 -- Other aspects: just process any decision nested in the
kono
parents:
diff changeset
1706 -- aspect expression.
kono
parents:
diff changeset
1707
kono
parents:
diff changeset
1708 when others =>
kono
parents:
diff changeset
1709 if Has_Decision (AE) then
kono
parents:
diff changeset
1710 C1 := 'X';
kono
parents:
diff changeset
1711 end if;
kono
parents:
diff changeset
1712 end case;
kono
parents:
diff changeset
1713
kono
parents:
diff changeset
1714 if C1 /= ASCII.NUL then
kono
parents:
diff changeset
1715 pragma Assert (Current_Pragma_Sloc = No_Location);
kono
parents:
diff changeset
1716
kono
parents:
diff changeset
1717 if C1 = 'a' or else C1 = 'A' then
kono
parents:
diff changeset
1718 Current_Pragma_Sloc := Sloc (AN);
kono
parents:
diff changeset
1719 end if;
kono
parents:
diff changeset
1720
kono
parents:
diff changeset
1721 Process_Decisions_Defer (AE, C1);
kono
parents:
diff changeset
1722
kono
parents:
diff changeset
1723 Current_Pragma_Sloc := No_Location;
kono
parents:
diff changeset
1724 end if;
kono
parents:
diff changeset
1725
kono
parents:
diff changeset
1726 Next (AN);
kono
parents:
diff changeset
1727 end loop;
kono
parents:
diff changeset
1728 end Traverse_Aspects;
kono
parents:
diff changeset
1729
kono
parents:
diff changeset
1730 ------------------------------------
kono
parents:
diff changeset
1731 -- Traverse_Degenerate_Subprogram --
kono
parents:
diff changeset
1732 ------------------------------------
kono
parents:
diff changeset
1733
kono
parents:
diff changeset
1734 procedure Traverse_Degenerate_Subprogram (N : Node_Id) is
kono
parents:
diff changeset
1735 begin
kono
parents:
diff changeset
1736 -- Complete current sequence of statements
kono
parents:
diff changeset
1737
kono
parents:
diff changeset
1738 Set_Statement_Entry;
kono
parents:
diff changeset
1739
kono
parents:
diff changeset
1740 declare
kono
parents:
diff changeset
1741 Saved_Dominant : constant Dominant_Info := Current_Dominant;
kono
parents:
diff changeset
1742 -- Save last statement in current sequence as dominant
kono
parents:
diff changeset
1743
kono
parents:
diff changeset
1744 begin
kono
parents:
diff changeset
1745 -- Output statement SCO for degenerate subprogram body (null
kono
parents:
diff changeset
1746 -- statement or freestanding expression) outside of the dominance
kono
parents:
diff changeset
1747 -- chain.
kono
parents:
diff changeset
1748
kono
parents:
diff changeset
1749 Current_Dominant := No_Dominant;
kono
parents:
diff changeset
1750 Extend_Statement_Sequence (N, Typ => ' ');
kono
parents:
diff changeset
1751
kono
parents:
diff changeset
1752 -- For the case of an expression-function, collect decisions
kono
parents:
diff changeset
1753 -- embedded in the expression now.
kono
parents:
diff changeset
1754
kono
parents:
diff changeset
1755 if Nkind (N) in N_Subexpr then
kono
parents:
diff changeset
1756 Process_Decisions_Defer (N, 'X');
kono
parents:
diff changeset
1757 end if;
kono
parents:
diff changeset
1758
kono
parents:
diff changeset
1759 Set_Statement_Entry;
kono
parents:
diff changeset
1760
kono
parents:
diff changeset
1761 -- Restore current dominant information designating last statement
kono
parents:
diff changeset
1762 -- in previous sequence (i.e. make the dominance chain skip over
kono
parents:
diff changeset
1763 -- the degenerate body).
kono
parents:
diff changeset
1764
kono
parents:
diff changeset
1765 Current_Dominant := Saved_Dominant;
kono
parents:
diff changeset
1766 end;
kono
parents:
diff changeset
1767 end Traverse_Degenerate_Subprogram;
kono
parents:
diff changeset
1768
kono
parents:
diff changeset
1769 ------------------
kono
parents:
diff changeset
1770 -- Traverse_One --
kono
parents:
diff changeset
1771 ------------------
kono
parents:
diff changeset
1772
kono
parents:
diff changeset
1773 procedure Traverse_One (N : Node_Id) is
kono
parents:
diff changeset
1774 begin
kono
parents:
diff changeset
1775 -- Initialize or extend current statement sequence. Note that for
kono
parents:
diff changeset
1776 -- special cases such as IF and Case statements we will modify
kono
parents:
diff changeset
1777 -- the range to exclude internal statements that should not be
kono
parents:
diff changeset
1778 -- counted as part of the current statement sequence.
kono
parents:
diff changeset
1779
kono
parents:
diff changeset
1780 case Nkind (N) is
kono
parents:
diff changeset
1781
kono
parents:
diff changeset
1782 -- Package declaration
kono
parents:
diff changeset
1783
kono
parents:
diff changeset
1784 when N_Package_Declaration =>
kono
parents:
diff changeset
1785 Set_Statement_Entry;
kono
parents:
diff changeset
1786 Traverse_Package_Declaration (N, Current_Dominant);
kono
parents:
diff changeset
1787
kono
parents:
diff changeset
1788 -- Generic package declaration
kono
parents:
diff changeset
1789
kono
parents:
diff changeset
1790 when N_Generic_Package_Declaration =>
kono
parents:
diff changeset
1791 Set_Statement_Entry;
kono
parents:
diff changeset
1792 Traverse_Generic_Package_Declaration (N);
kono
parents:
diff changeset
1793
kono
parents:
diff changeset
1794 -- Package body
kono
parents:
diff changeset
1795
kono
parents:
diff changeset
1796 when N_Package_Body =>
kono
parents:
diff changeset
1797 Set_Statement_Entry;
kono
parents:
diff changeset
1798 Traverse_Package_Body (N);
kono
parents:
diff changeset
1799
kono
parents:
diff changeset
1800 -- Subprogram declaration or subprogram body stub
kono
parents:
diff changeset
1801
kono
parents:
diff changeset
1802 when N_Expression_Function
kono
parents:
diff changeset
1803 | N_Subprogram_Body_Stub
kono
parents:
diff changeset
1804 | N_Subprogram_Declaration
kono
parents:
diff changeset
1805 =>
kono
parents:
diff changeset
1806 declare
kono
parents:
diff changeset
1807 Spec : constant Node_Id := Specification (N);
kono
parents:
diff changeset
1808 begin
kono
parents:
diff changeset
1809 Process_Decisions_Defer
kono
parents:
diff changeset
1810 (Parameter_Specifications (Spec), 'X');
kono
parents:
diff changeset
1811
kono
parents:
diff changeset
1812 -- Case of a null procedure: generate SCO for fictitious
kono
parents:
diff changeset
1813 -- NULL statement located at the NULL keyword in the
kono
parents:
diff changeset
1814 -- procedure specification.
kono
parents:
diff changeset
1815
kono
parents:
diff changeset
1816 if Nkind (N) = N_Subprogram_Declaration
kono
parents:
diff changeset
1817 and then Nkind (Spec) = N_Procedure_Specification
kono
parents:
diff changeset
1818 and then Null_Present (Spec)
kono
parents:
diff changeset
1819 then
kono
parents:
diff changeset
1820 Traverse_Degenerate_Subprogram (Null_Statement (Spec));
kono
parents:
diff changeset
1821
kono
parents:
diff changeset
1822 -- Case of an expression function: generate a statement SCO
kono
parents:
diff changeset
1823 -- for the expression (and then decision SCOs for any nested
kono
parents:
diff changeset
1824 -- decisions).
kono
parents:
diff changeset
1825
kono
parents:
diff changeset
1826 elsif Nkind (N) = N_Expression_Function then
kono
parents:
diff changeset
1827 Traverse_Degenerate_Subprogram (Expression (N));
kono
parents:
diff changeset
1828 end if;
kono
parents:
diff changeset
1829 end;
kono
parents:
diff changeset
1830
kono
parents:
diff changeset
1831 -- Entry declaration
kono
parents:
diff changeset
1832
kono
parents:
diff changeset
1833 when N_Entry_Declaration =>
kono
parents:
diff changeset
1834 Process_Decisions_Defer (Parameter_Specifications (N), 'X');
kono
parents:
diff changeset
1835
kono
parents:
diff changeset
1836 -- Generic subprogram declaration
kono
parents:
diff changeset
1837
kono
parents:
diff changeset
1838 when N_Generic_Subprogram_Declaration =>
kono
parents:
diff changeset
1839 Process_Decisions_Defer
kono
parents:
diff changeset
1840 (Generic_Formal_Declarations (N), 'X');
kono
parents:
diff changeset
1841 Process_Decisions_Defer
kono
parents:
diff changeset
1842 (Parameter_Specifications (Specification (N)), 'X');
kono
parents:
diff changeset
1843
kono
parents:
diff changeset
1844 -- Task or subprogram body
kono
parents:
diff changeset
1845
kono
parents:
diff changeset
1846 when N_Subprogram_Body
kono
parents:
diff changeset
1847 | N_Task_Body
kono
parents:
diff changeset
1848 =>
kono
parents:
diff changeset
1849 Set_Statement_Entry;
kono
parents:
diff changeset
1850 Traverse_Subprogram_Or_Task_Body (N);
kono
parents:
diff changeset
1851
kono
parents:
diff changeset
1852 -- Entry body
kono
parents:
diff changeset
1853
kono
parents:
diff changeset
1854 when N_Entry_Body =>
kono
parents:
diff changeset
1855 declare
kono
parents:
diff changeset
1856 Cond : constant Node_Id :=
kono
parents:
diff changeset
1857 Condition (Entry_Body_Formal_Part (N));
kono
parents:
diff changeset
1858
kono
parents:
diff changeset
1859 Inner_Dominant : Dominant_Info := No_Dominant;
kono
parents:
diff changeset
1860
kono
parents:
diff changeset
1861 begin
kono
parents:
diff changeset
1862 Set_Statement_Entry;
kono
parents:
diff changeset
1863
kono
parents:
diff changeset
1864 if Present (Cond) then
kono
parents:
diff changeset
1865 Process_Decisions_Defer (Cond, 'G');
kono
parents:
diff changeset
1866
kono
parents:
diff changeset
1867 -- For an entry body with a barrier, the entry body
kono
parents:
diff changeset
1868 -- is dominanted by a True evaluation of the barrier.
kono
parents:
diff changeset
1869
kono
parents:
diff changeset
1870 Inner_Dominant := ('T', N);
kono
parents:
diff changeset
1871 end if;
kono
parents:
diff changeset
1872
kono
parents:
diff changeset
1873 Traverse_Subprogram_Or_Task_Body (N, Inner_Dominant);
kono
parents:
diff changeset
1874 end;
kono
parents:
diff changeset
1875
kono
parents:
diff changeset
1876 -- Protected body
kono
parents:
diff changeset
1877
kono
parents:
diff changeset
1878 when N_Protected_Body =>
kono
parents:
diff changeset
1879 Set_Statement_Entry;
kono
parents:
diff changeset
1880 Traverse_Declarations_Or_Statements (Declarations (N));
kono
parents:
diff changeset
1881
kono
parents:
diff changeset
1882 -- Exit statement, which is an exit statement in the SCO sense,
kono
parents:
diff changeset
1883 -- so it is included in the current statement sequence, but
kono
parents:
diff changeset
1884 -- then it terminates this sequence. We also have to process
kono
parents:
diff changeset
1885 -- any decisions in the exit statement expression.
kono
parents:
diff changeset
1886
kono
parents:
diff changeset
1887 when N_Exit_Statement =>
kono
parents:
diff changeset
1888 Extend_Statement_Sequence (N, 'E');
kono
parents:
diff changeset
1889 Process_Decisions_Defer (Condition (N), 'E');
kono
parents:
diff changeset
1890 Set_Statement_Entry;
kono
parents:
diff changeset
1891
kono
parents:
diff changeset
1892 -- If condition is present, then following statement is
kono
parents:
diff changeset
1893 -- only executed if the condition evaluates to False.
kono
parents:
diff changeset
1894
kono
parents:
diff changeset
1895 if Present (Condition (N)) then
kono
parents:
diff changeset
1896 Current_Dominant := ('F', N);
kono
parents:
diff changeset
1897 else
kono
parents:
diff changeset
1898 Current_Dominant := No_Dominant;
kono
parents:
diff changeset
1899 end if;
kono
parents:
diff changeset
1900
kono
parents:
diff changeset
1901 -- Label, which breaks the current statement sequence, but the
kono
parents:
diff changeset
1902 -- label itself is not included in the next statement sequence,
kono
parents:
diff changeset
1903 -- since it generates no code.
kono
parents:
diff changeset
1904
kono
parents:
diff changeset
1905 when N_Label =>
kono
parents:
diff changeset
1906 Set_Statement_Entry;
kono
parents:
diff changeset
1907 Current_Dominant := No_Dominant;
kono
parents:
diff changeset
1908
kono
parents:
diff changeset
1909 -- Block statement, which breaks the current statement sequence
kono
parents:
diff changeset
1910
kono
parents:
diff changeset
1911 when N_Block_Statement =>
kono
parents:
diff changeset
1912 Set_Statement_Entry;
kono
parents:
diff changeset
1913
kono
parents:
diff changeset
1914 -- The first statement in the handled sequence of statements
kono
parents:
diff changeset
1915 -- is dominated by the elaboration of the last declaration.
kono
parents:
diff changeset
1916
kono
parents:
diff changeset
1917 Current_Dominant := Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
1918 (L => Declarations (N),
kono
parents:
diff changeset
1919 D => Current_Dominant);
kono
parents:
diff changeset
1920
kono
parents:
diff changeset
1921 Traverse_Handled_Statement_Sequence
kono
parents:
diff changeset
1922 (N => Handled_Statement_Sequence (N),
kono
parents:
diff changeset
1923 D => Current_Dominant);
kono
parents:
diff changeset
1924
kono
parents:
diff changeset
1925 -- If statement, which breaks the current statement sequence,
kono
parents:
diff changeset
1926 -- but we include the condition in the current sequence.
kono
parents:
diff changeset
1927
kono
parents:
diff changeset
1928 when N_If_Statement =>
kono
parents:
diff changeset
1929 Current_Test := N;
kono
parents:
diff changeset
1930 Extend_Statement_Sequence (N, 'I');
kono
parents:
diff changeset
1931 Process_Decisions_Defer (Condition (N), 'I');
kono
parents:
diff changeset
1932 Set_Statement_Entry;
kono
parents:
diff changeset
1933
kono
parents:
diff changeset
1934 -- Now we traverse the statements in the THEN part
kono
parents:
diff changeset
1935
kono
parents:
diff changeset
1936 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
1937 (L => Then_Statements (N),
kono
parents:
diff changeset
1938 D => ('T', N));
kono
parents:
diff changeset
1939
kono
parents:
diff changeset
1940 -- Loop through ELSIF parts if present
kono
parents:
diff changeset
1941
kono
parents:
diff changeset
1942 if Present (Elsif_Parts (N)) then
kono
parents:
diff changeset
1943 declare
kono
parents:
diff changeset
1944 Saved_Dominant : constant Dominant_Info :=
kono
parents:
diff changeset
1945 Current_Dominant;
kono
parents:
diff changeset
1946
kono
parents:
diff changeset
1947 Elif : Node_Id := First (Elsif_Parts (N));
kono
parents:
diff changeset
1948
kono
parents:
diff changeset
1949 begin
kono
parents:
diff changeset
1950 while Present (Elif) loop
kono
parents:
diff changeset
1951
kono
parents:
diff changeset
1952 -- An Elsif is executed only if the previous test
kono
parents:
diff changeset
1953 -- got a FALSE outcome.
kono
parents:
diff changeset
1954
kono
parents:
diff changeset
1955 Current_Dominant := ('F', Current_Test);
kono
parents:
diff changeset
1956
kono
parents:
diff changeset
1957 -- Now update current test information
kono
parents:
diff changeset
1958
kono
parents:
diff changeset
1959 Current_Test := Elif;
kono
parents:
diff changeset
1960
kono
parents:
diff changeset
1961 -- We generate a statement sequence for the
kono
parents:
diff changeset
1962 -- construct "ELSIF condition", so that we have
kono
parents:
diff changeset
1963 -- a statement for the resulting decisions.
kono
parents:
diff changeset
1964
kono
parents:
diff changeset
1965 Extend_Statement_Sequence (Elif, 'I');
kono
parents:
diff changeset
1966 Process_Decisions_Defer (Condition (Elif), 'I');
kono
parents:
diff changeset
1967 Set_Statement_Entry;
kono
parents:
diff changeset
1968
kono
parents:
diff changeset
1969 -- An ELSIF part is never guaranteed to have
kono
parents:
diff changeset
1970 -- been executed, following statements are only
kono
parents:
diff changeset
1971 -- dominated by the initial IF statement.
kono
parents:
diff changeset
1972
kono
parents:
diff changeset
1973 Current_Dominant := Saved_Dominant;
kono
parents:
diff changeset
1974
kono
parents:
diff changeset
1975 -- Traverse the statements in the ELSIF
kono
parents:
diff changeset
1976
kono
parents:
diff changeset
1977 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
1978 (L => Then_Statements (Elif),
kono
parents:
diff changeset
1979 D => ('T', Elif));
kono
parents:
diff changeset
1980 Next (Elif);
kono
parents:
diff changeset
1981 end loop;
kono
parents:
diff changeset
1982 end;
kono
parents:
diff changeset
1983 end if;
kono
parents:
diff changeset
1984
kono
parents:
diff changeset
1985 -- Finally traverse the ELSE statements if present
kono
parents:
diff changeset
1986
kono
parents:
diff changeset
1987 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
1988 (L => Else_Statements (N),
kono
parents:
diff changeset
1989 D => ('F', Current_Test));
kono
parents:
diff changeset
1990
kono
parents:
diff changeset
1991 -- CASE statement, which breaks the current statement sequence,
kono
parents:
diff changeset
1992 -- but we include the expression in the current sequence.
kono
parents:
diff changeset
1993
kono
parents:
diff changeset
1994 when N_Case_Statement =>
kono
parents:
diff changeset
1995 Extend_Statement_Sequence (N, 'C');
kono
parents:
diff changeset
1996 Process_Decisions_Defer (Expression (N), 'X');
kono
parents:
diff changeset
1997 Set_Statement_Entry;
kono
parents:
diff changeset
1998
kono
parents:
diff changeset
1999 -- Process case branches, all of which are dominated by the
kono
parents:
diff changeset
2000 -- CASE statement.
kono
parents:
diff changeset
2001
kono
parents:
diff changeset
2002 declare
kono
parents:
diff changeset
2003 Alt : Node_Id;
kono
parents:
diff changeset
2004 begin
kono
parents:
diff changeset
2005 Alt := First_Non_Pragma (Alternatives (N));
kono
parents:
diff changeset
2006 while Present (Alt) loop
kono
parents:
diff changeset
2007 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2008 (L => Statements (Alt),
kono
parents:
diff changeset
2009 D => Current_Dominant);
kono
parents:
diff changeset
2010 Next (Alt);
kono
parents:
diff changeset
2011 end loop;
kono
parents:
diff changeset
2012 end;
kono
parents:
diff changeset
2013
kono
parents:
diff changeset
2014 -- ACCEPT statement
kono
parents:
diff changeset
2015
kono
parents:
diff changeset
2016 when N_Accept_Statement =>
kono
parents:
diff changeset
2017 Extend_Statement_Sequence (N, 'A');
kono
parents:
diff changeset
2018 Set_Statement_Entry;
kono
parents:
diff changeset
2019
kono
parents:
diff changeset
2020 -- Process sequence of statements, dominant is the ACCEPT
kono
parents:
diff changeset
2021 -- statement.
kono
parents:
diff changeset
2022
kono
parents:
diff changeset
2023 Traverse_Handled_Statement_Sequence
kono
parents:
diff changeset
2024 (N => Handled_Statement_Sequence (N),
kono
parents:
diff changeset
2025 D => Current_Dominant);
kono
parents:
diff changeset
2026
kono
parents:
diff changeset
2027 -- SELECT
kono
parents:
diff changeset
2028
kono
parents:
diff changeset
2029 when N_Selective_Accept =>
kono
parents:
diff changeset
2030 Extend_Statement_Sequence (N, 'S');
kono
parents:
diff changeset
2031 Set_Statement_Entry;
kono
parents:
diff changeset
2032
kono
parents:
diff changeset
2033 -- Process alternatives
kono
parents:
diff changeset
2034
kono
parents:
diff changeset
2035 declare
kono
parents:
diff changeset
2036 Alt : Node_Id;
kono
parents:
diff changeset
2037 Guard : Node_Id;
kono
parents:
diff changeset
2038 S_Dom : Dominant_Info;
kono
parents:
diff changeset
2039
kono
parents:
diff changeset
2040 begin
kono
parents:
diff changeset
2041 Alt := First (Select_Alternatives (N));
kono
parents:
diff changeset
2042 while Present (Alt) loop
kono
parents:
diff changeset
2043 S_Dom := Current_Dominant;
kono
parents:
diff changeset
2044 Guard := Condition (Alt);
kono
parents:
diff changeset
2045
kono
parents:
diff changeset
2046 if Present (Guard) then
kono
parents:
diff changeset
2047 Process_Decisions
kono
parents:
diff changeset
2048 (Guard,
kono
parents:
diff changeset
2049 'G',
kono
parents:
diff changeset
2050 Pragma_Sloc => No_Location);
kono
parents:
diff changeset
2051 Current_Dominant := ('T', Guard);
kono
parents:
diff changeset
2052 end if;
kono
parents:
diff changeset
2053
kono
parents:
diff changeset
2054 Traverse_One (Alt);
kono
parents:
diff changeset
2055
kono
parents:
diff changeset
2056 Current_Dominant := S_Dom;
kono
parents:
diff changeset
2057 Next (Alt);
kono
parents:
diff changeset
2058 end loop;
kono
parents:
diff changeset
2059 end;
kono
parents:
diff changeset
2060
kono
parents:
diff changeset
2061 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2062 (L => Else_Statements (N),
kono
parents:
diff changeset
2063 D => Current_Dominant);
kono
parents:
diff changeset
2064
kono
parents:
diff changeset
2065 when N_Conditional_Entry_Call
kono
parents:
diff changeset
2066 | N_Timed_Entry_Call
kono
parents:
diff changeset
2067 =>
kono
parents:
diff changeset
2068 Extend_Statement_Sequence (N, 'S');
kono
parents:
diff changeset
2069 Set_Statement_Entry;
kono
parents:
diff changeset
2070
kono
parents:
diff changeset
2071 -- Process alternatives
kono
parents:
diff changeset
2072
kono
parents:
diff changeset
2073 Traverse_One (Entry_Call_Alternative (N));
kono
parents:
diff changeset
2074
kono
parents:
diff changeset
2075 if Nkind (N) = N_Timed_Entry_Call then
kono
parents:
diff changeset
2076 Traverse_One (Delay_Alternative (N));
kono
parents:
diff changeset
2077 else
kono
parents:
diff changeset
2078 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2079 (L => Else_Statements (N),
kono
parents:
diff changeset
2080 D => Current_Dominant);
kono
parents:
diff changeset
2081 end if;
kono
parents:
diff changeset
2082
kono
parents:
diff changeset
2083 when N_Asynchronous_Select =>
kono
parents:
diff changeset
2084 Extend_Statement_Sequence (N, 'S');
kono
parents:
diff changeset
2085 Set_Statement_Entry;
kono
parents:
diff changeset
2086
kono
parents:
diff changeset
2087 Traverse_One (Triggering_Alternative (N));
kono
parents:
diff changeset
2088 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2089 (L => Statements (Abortable_Part (N)),
kono
parents:
diff changeset
2090 D => Current_Dominant);
kono
parents:
diff changeset
2091
kono
parents:
diff changeset
2092 when N_Accept_Alternative =>
kono
parents:
diff changeset
2093 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2094 (L => Statements (N),
kono
parents:
diff changeset
2095 D => Current_Dominant,
kono
parents:
diff changeset
2096 P => Accept_Statement (N));
kono
parents:
diff changeset
2097
kono
parents:
diff changeset
2098 when N_Entry_Call_Alternative =>
kono
parents:
diff changeset
2099 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2100 (L => Statements (N),
kono
parents:
diff changeset
2101 D => Current_Dominant,
kono
parents:
diff changeset
2102 P => Entry_Call_Statement (N));
kono
parents:
diff changeset
2103
kono
parents:
diff changeset
2104 when N_Delay_Alternative =>
kono
parents:
diff changeset
2105 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2106 (L => Statements (N),
kono
parents:
diff changeset
2107 D => Current_Dominant,
kono
parents:
diff changeset
2108 P => Delay_Statement (N));
kono
parents:
diff changeset
2109
kono
parents:
diff changeset
2110 when N_Triggering_Alternative =>
kono
parents:
diff changeset
2111 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2112 (L => Statements (N),
kono
parents:
diff changeset
2113 D => Current_Dominant,
kono
parents:
diff changeset
2114 P => Triggering_Statement (N));
kono
parents:
diff changeset
2115
kono
parents:
diff changeset
2116 when N_Terminate_Alternative =>
kono
parents:
diff changeset
2117
kono
parents:
diff changeset
2118 -- It is dubious to emit a statement SCO for a TERMINATE
kono
parents:
diff changeset
2119 -- alternative, since no code is actually executed if the
kono
parents:
diff changeset
2120 -- alternative is selected -- the tasking runtime call just
kono
parents:
diff changeset
2121 -- never returns???
kono
parents:
diff changeset
2122
kono
parents:
diff changeset
2123 Extend_Statement_Sequence (N, ' ');
kono
parents:
diff changeset
2124 Set_Statement_Entry;
kono
parents:
diff changeset
2125
kono
parents:
diff changeset
2126 -- Unconditional exit points, which are included in the current
kono
parents:
diff changeset
2127 -- statement sequence, but then terminate it
kono
parents:
diff changeset
2128
kono
parents:
diff changeset
2129 when N_Goto_Statement
kono
parents:
diff changeset
2130 | N_Raise_Statement
kono
parents:
diff changeset
2131 | N_Requeue_Statement
kono
parents:
diff changeset
2132 =>
kono
parents:
diff changeset
2133 Extend_Statement_Sequence (N, ' ');
kono
parents:
diff changeset
2134 Set_Statement_Entry;
kono
parents:
diff changeset
2135 Current_Dominant := No_Dominant;
kono
parents:
diff changeset
2136
kono
parents:
diff changeset
2137 -- Simple return statement. which is an exit point, but we
kono
parents:
diff changeset
2138 -- have to process the return expression for decisions.
kono
parents:
diff changeset
2139
kono
parents:
diff changeset
2140 when N_Simple_Return_Statement =>
kono
parents:
diff changeset
2141 Extend_Statement_Sequence (N, ' ');
kono
parents:
diff changeset
2142 Process_Decisions_Defer (Expression (N), 'X');
kono
parents:
diff changeset
2143 Set_Statement_Entry;
kono
parents:
diff changeset
2144 Current_Dominant := No_Dominant;
kono
parents:
diff changeset
2145
kono
parents:
diff changeset
2146 -- Extended return statement
kono
parents:
diff changeset
2147
kono
parents:
diff changeset
2148 when N_Extended_Return_Statement =>
kono
parents:
diff changeset
2149 Extend_Statement_Sequence (N, 'R');
kono
parents:
diff changeset
2150 Process_Decisions_Defer (Return_Object_Declarations (N), 'X');
kono
parents:
diff changeset
2151 Set_Statement_Entry;
kono
parents:
diff changeset
2152
kono
parents:
diff changeset
2153 Traverse_Handled_Statement_Sequence
kono
parents:
diff changeset
2154 (N => Handled_Statement_Sequence (N),
kono
parents:
diff changeset
2155 D => Current_Dominant);
kono
parents:
diff changeset
2156
kono
parents:
diff changeset
2157 Current_Dominant := No_Dominant;
kono
parents:
diff changeset
2158
kono
parents:
diff changeset
2159 -- Loop ends the current statement sequence, but we include
kono
parents:
diff changeset
2160 -- the iteration scheme if present in the current sequence.
kono
parents:
diff changeset
2161 -- But the body of the loop starts a new sequence, since it
kono
parents:
diff changeset
2162 -- may not be executed as part of the current sequence.
kono
parents:
diff changeset
2163
kono
parents:
diff changeset
2164 when N_Loop_Statement =>
kono
parents:
diff changeset
2165 declare
kono
parents:
diff changeset
2166 ISC : constant Node_Id := Iteration_Scheme (N);
kono
parents:
diff changeset
2167 Inner_Dominant : Dominant_Info := No_Dominant;
kono
parents:
diff changeset
2168
kono
parents:
diff changeset
2169 begin
kono
parents:
diff changeset
2170 if Present (ISC) then
kono
parents:
diff changeset
2171
kono
parents:
diff changeset
2172 -- If iteration scheme present, extend the current
kono
parents:
diff changeset
2173 -- statement sequence to include the iteration scheme
kono
parents:
diff changeset
2174 -- and process any decisions it contains.
kono
parents:
diff changeset
2175
kono
parents:
diff changeset
2176 -- While loop
kono
parents:
diff changeset
2177
kono
parents:
diff changeset
2178 if Present (Condition (ISC)) then
kono
parents:
diff changeset
2179 Extend_Statement_Sequence (N, 'W');
kono
parents:
diff changeset
2180 Process_Decisions_Defer (Condition (ISC), 'W');
kono
parents:
diff changeset
2181
kono
parents:
diff changeset
2182 -- Set more specific dominant for inner statements
kono
parents:
diff changeset
2183 -- (the control sloc for the decision is that of
kono
parents:
diff changeset
2184 -- the WHILE token).
kono
parents:
diff changeset
2185
kono
parents:
diff changeset
2186 Inner_Dominant := ('T', ISC);
kono
parents:
diff changeset
2187
kono
parents:
diff changeset
2188 -- For loop
kono
parents:
diff changeset
2189
kono
parents:
diff changeset
2190 else
kono
parents:
diff changeset
2191 Extend_Statement_Sequence (N, 'F');
kono
parents:
diff changeset
2192 Process_Decisions_Defer
kono
parents:
diff changeset
2193 (Loop_Parameter_Specification (ISC), 'X');
kono
parents:
diff changeset
2194 end if;
kono
parents:
diff changeset
2195 end if;
kono
parents:
diff changeset
2196
kono
parents:
diff changeset
2197 Set_Statement_Entry;
kono
parents:
diff changeset
2198
kono
parents:
diff changeset
2199 if Inner_Dominant = No_Dominant then
kono
parents:
diff changeset
2200 Inner_Dominant := Current_Dominant;
kono
parents:
diff changeset
2201 end if;
kono
parents:
diff changeset
2202
kono
parents:
diff changeset
2203 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2204 (L => Statements (N),
kono
parents:
diff changeset
2205 D => Inner_Dominant);
kono
parents:
diff changeset
2206 end;
kono
parents:
diff changeset
2207
kono
parents:
diff changeset
2208 -- Pragma
kono
parents:
diff changeset
2209
kono
parents:
diff changeset
2210 when N_Pragma =>
kono
parents:
diff changeset
2211
kono
parents:
diff changeset
2212 -- Record sloc of pragma (pragmas don't nest)
kono
parents:
diff changeset
2213
kono
parents:
diff changeset
2214 pragma Assert (Current_Pragma_Sloc = No_Location);
kono
parents:
diff changeset
2215 Current_Pragma_Sloc := Sloc (N);
kono
parents:
diff changeset
2216
kono
parents:
diff changeset
2217 -- Processing depends on the kind of pragma
kono
parents:
diff changeset
2218
kono
parents:
diff changeset
2219 declare
kono
parents:
diff changeset
2220 Nam : constant Name_Id := Pragma_Name_Unmapped (N);
kono
parents:
diff changeset
2221 Arg : Node_Id :=
kono
parents:
diff changeset
2222 First (Pragma_Argument_Associations (N));
kono
parents:
diff changeset
2223 Typ : Character;
kono
parents:
diff changeset
2224
kono
parents:
diff changeset
2225 begin
kono
parents:
diff changeset
2226 case Nam is
kono
parents:
diff changeset
2227 when Name_Assert
kono
parents:
diff changeset
2228 | Name_Assert_And_Cut
kono
parents:
diff changeset
2229 | Name_Assume
kono
parents:
diff changeset
2230 | Name_Check
kono
parents:
diff changeset
2231 | Name_Loop_Invariant
kono
parents:
diff changeset
2232 | Name_Postcondition
kono
parents:
diff changeset
2233 | Name_Precondition
kono
parents:
diff changeset
2234 =>
kono
parents:
diff changeset
2235 -- For Assert/Check/Precondition/Postcondition, we
kono
parents:
diff changeset
2236 -- must generate a P entry for the decision. Note
kono
parents:
diff changeset
2237 -- that this is done unconditionally at this stage.
kono
parents:
diff changeset
2238 -- Output for disabled pragmas is suppressed later
kono
parents:
diff changeset
2239 -- on when we output the decision line in Put_SCOs,
kono
parents:
diff changeset
2240 -- depending on setting by Set_SCO_Pragma_Enabled.
kono
parents:
diff changeset
2241
kono
parents:
diff changeset
2242 if Nam = Name_Check then
kono
parents:
diff changeset
2243 Next (Arg);
kono
parents:
diff changeset
2244 end if;
kono
parents:
diff changeset
2245
kono
parents:
diff changeset
2246 Process_Decisions_Defer (Expression (Arg), 'P');
kono
parents:
diff changeset
2247 Typ := 'p';
kono
parents:
diff changeset
2248
kono
parents:
diff changeset
2249 -- Pre/postconditions can be inherited so SCO should
kono
parents:
diff changeset
2250 -- never be deactivated???
kono
parents:
diff changeset
2251
kono
parents:
diff changeset
2252 when Name_Debug =>
kono
parents:
diff changeset
2253 if Present (Arg) and then Present (Next (Arg)) then
kono
parents:
diff changeset
2254
kono
parents:
diff changeset
2255 -- Case of a dyadic pragma Debug: first argument
kono
parents:
diff changeset
2256 -- is a P decision, any nested decision in the
kono
parents:
diff changeset
2257 -- second argument is an X decision.
kono
parents:
diff changeset
2258
kono
parents:
diff changeset
2259 Process_Decisions_Defer (Expression (Arg), 'P');
kono
parents:
diff changeset
2260 Next (Arg);
kono
parents:
diff changeset
2261 end if;
kono
parents:
diff changeset
2262
kono
parents:
diff changeset
2263 Process_Decisions_Defer (Expression (Arg), 'X');
kono
parents:
diff changeset
2264 Typ := 'p';
kono
parents:
diff changeset
2265
kono
parents:
diff changeset
2266 -- For all other pragmas, we generate decision entries
kono
parents:
diff changeset
2267 -- for any embedded expressions, and the pragma is
kono
parents:
diff changeset
2268 -- never disabled.
kono
parents:
diff changeset
2269
kono
parents:
diff changeset
2270 -- Should generate P decisions (not X) for assertion
kono
parents:
diff changeset
2271 -- related pragmas: [Type_]Invariant,
kono
parents:
diff changeset
2272 -- [{Static,Dynamic}_]Predicate???
kono
parents:
diff changeset
2273
kono
parents:
diff changeset
2274 when others =>
kono
parents:
diff changeset
2275 Process_Decisions_Defer (N, 'X');
kono
parents:
diff changeset
2276 Typ := 'P';
kono
parents:
diff changeset
2277 end case;
kono
parents:
diff changeset
2278
kono
parents:
diff changeset
2279 -- Add statement SCO
kono
parents:
diff changeset
2280
kono
parents:
diff changeset
2281 Extend_Statement_Sequence (N, Typ);
kono
parents:
diff changeset
2282
kono
parents:
diff changeset
2283 Current_Pragma_Sloc := No_Location;
kono
parents:
diff changeset
2284 end;
kono
parents:
diff changeset
2285
kono
parents:
diff changeset
2286 -- Object declaration. Ignored if Prev_Ids is set, since the
kono
parents:
diff changeset
2287 -- parser generates multiple instances of the whole declaration
kono
parents:
diff changeset
2288 -- if there is more than one identifier declared, and we only
kono
parents:
diff changeset
2289 -- want one entry in the SCOs, so we take the first, for which
kono
parents:
diff changeset
2290 -- Prev_Ids is False.
kono
parents:
diff changeset
2291
kono
parents:
diff changeset
2292 when N_Number_Declaration
kono
parents:
diff changeset
2293 | N_Object_Declaration
kono
parents:
diff changeset
2294 =>
kono
parents:
diff changeset
2295 if not Prev_Ids (N) then
kono
parents:
diff changeset
2296 Extend_Statement_Sequence (N, 'o');
kono
parents:
diff changeset
2297
kono
parents:
diff changeset
2298 if Has_Decision (N) then
kono
parents:
diff changeset
2299 Process_Decisions_Defer (N, 'X');
kono
parents:
diff changeset
2300 end if;
kono
parents:
diff changeset
2301 end if;
kono
parents:
diff changeset
2302
kono
parents:
diff changeset
2303 -- All other cases, which extend the current statement sequence
kono
parents:
diff changeset
2304 -- but do not terminate it, even if they have nested decisions.
kono
parents:
diff changeset
2305
kono
parents:
diff changeset
2306 when N_Protected_Type_Declaration
kono
parents:
diff changeset
2307 | N_Task_Type_Declaration
kono
parents:
diff changeset
2308 =>
kono
parents:
diff changeset
2309 Extend_Statement_Sequence (N, 't');
kono
parents:
diff changeset
2310 Process_Decisions_Defer (Discriminant_Specifications (N), 'X');
kono
parents:
diff changeset
2311 Set_Statement_Entry;
kono
parents:
diff changeset
2312
kono
parents:
diff changeset
2313 Traverse_Sync_Definition (N);
kono
parents:
diff changeset
2314
kono
parents:
diff changeset
2315 when N_Single_Protected_Declaration
kono
parents:
diff changeset
2316 | N_Single_Task_Declaration
kono
parents:
diff changeset
2317 =>
kono
parents:
diff changeset
2318 Extend_Statement_Sequence (N, 'o');
kono
parents:
diff changeset
2319 Set_Statement_Entry;
kono
parents:
diff changeset
2320
kono
parents:
diff changeset
2321 Traverse_Sync_Definition (N);
kono
parents:
diff changeset
2322
kono
parents:
diff changeset
2323 when others =>
kono
parents:
diff changeset
2324
kono
parents:
diff changeset
2325 -- Determine required type character code, or ASCII.NUL if
kono
parents:
diff changeset
2326 -- no SCO should be generated for this node.
kono
parents:
diff changeset
2327
kono
parents:
diff changeset
2328 declare
kono
parents:
diff changeset
2329 NK : constant Node_Kind := Nkind (N);
kono
parents:
diff changeset
2330 Typ : Character;
kono
parents:
diff changeset
2331
kono
parents:
diff changeset
2332 begin
kono
parents:
diff changeset
2333 case NK is
kono
parents:
diff changeset
2334 when N_Full_Type_Declaration
kono
parents:
diff changeset
2335 | N_Incomplete_Type_Declaration
kono
parents:
diff changeset
2336 | N_Private_Extension_Declaration
kono
parents:
diff changeset
2337 | N_Private_Type_Declaration
kono
parents:
diff changeset
2338 =>
kono
parents:
diff changeset
2339 Typ := 't';
kono
parents:
diff changeset
2340
kono
parents:
diff changeset
2341 when N_Subtype_Declaration =>
kono
parents:
diff changeset
2342 Typ := 's';
kono
parents:
diff changeset
2343
kono
parents:
diff changeset
2344 when N_Renaming_Declaration =>
kono
parents:
diff changeset
2345 Typ := 'r';
kono
parents:
diff changeset
2346
kono
parents:
diff changeset
2347 when N_Generic_Instantiation =>
kono
parents:
diff changeset
2348 Typ := 'i';
kono
parents:
diff changeset
2349
kono
parents:
diff changeset
2350 when N_Package_Body_Stub
kono
parents:
diff changeset
2351 | N_Protected_Body_Stub
kono
parents:
diff changeset
2352 | N_Representation_Clause
kono
parents:
diff changeset
2353 | N_Task_Body_Stub
kono
parents:
diff changeset
2354 | N_Use_Package_Clause
kono
parents:
diff changeset
2355 | N_Use_Type_Clause
kono
parents:
diff changeset
2356 =>
kono
parents:
diff changeset
2357 Typ := ASCII.NUL;
kono
parents:
diff changeset
2358
kono
parents:
diff changeset
2359 when N_Procedure_Call_Statement =>
kono
parents:
diff changeset
2360 Typ := ' ';
kono
parents:
diff changeset
2361
kono
parents:
diff changeset
2362 when others =>
kono
parents:
diff changeset
2363 if NK in N_Statement_Other_Than_Procedure_Call then
kono
parents:
diff changeset
2364 Typ := ' ';
kono
parents:
diff changeset
2365 else
kono
parents:
diff changeset
2366 Typ := 'd';
kono
parents:
diff changeset
2367 end if;
kono
parents:
diff changeset
2368 end case;
kono
parents:
diff changeset
2369
kono
parents:
diff changeset
2370 if Typ /= ASCII.NUL then
kono
parents:
diff changeset
2371 Extend_Statement_Sequence (N, Typ);
kono
parents:
diff changeset
2372 end if;
kono
parents:
diff changeset
2373 end;
kono
parents:
diff changeset
2374
kono
parents:
diff changeset
2375 -- Process any embedded decisions
kono
parents:
diff changeset
2376
kono
parents:
diff changeset
2377 if Has_Decision (N) then
kono
parents:
diff changeset
2378 Process_Decisions_Defer (N, 'X');
kono
parents:
diff changeset
2379 end if;
kono
parents:
diff changeset
2380 end case;
kono
parents:
diff changeset
2381
kono
parents:
diff changeset
2382 -- Process aspects if present
kono
parents:
diff changeset
2383
kono
parents:
diff changeset
2384 Traverse_Aspects (N);
kono
parents:
diff changeset
2385 end Traverse_One;
kono
parents:
diff changeset
2386
kono
parents:
diff changeset
2387 -- Start of processing for Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2388
kono
parents:
diff changeset
2389 begin
kono
parents:
diff changeset
2390 -- Process single prefixed node
kono
parents:
diff changeset
2391
kono
parents:
diff changeset
2392 if Present (P) then
kono
parents:
diff changeset
2393 Traverse_One (P);
kono
parents:
diff changeset
2394 end if;
kono
parents:
diff changeset
2395
kono
parents:
diff changeset
2396 -- Loop through statements or declarations
kono
parents:
diff changeset
2397
kono
parents:
diff changeset
2398 if Is_Non_Empty_List (L) then
kono
parents:
diff changeset
2399 N := First (L);
kono
parents:
diff changeset
2400 while Present (N) loop
kono
parents:
diff changeset
2401
kono
parents:
diff changeset
2402 -- Note: For separate bodies, we see the tree after Par.Labl has
kono
parents:
diff changeset
2403 -- introduced implicit labels, so we need to ignore those nodes.
kono
parents:
diff changeset
2404
kono
parents:
diff changeset
2405 if Nkind (N) /= N_Implicit_Label_Declaration then
kono
parents:
diff changeset
2406 Traverse_One (N);
kono
parents:
diff changeset
2407 end if;
kono
parents:
diff changeset
2408
kono
parents:
diff changeset
2409 Next (N);
kono
parents:
diff changeset
2410 end loop;
kono
parents:
diff changeset
2411
kono
parents:
diff changeset
2412 end if;
kono
parents:
diff changeset
2413
kono
parents:
diff changeset
2414 -- End sequence of statements and flush deferred decisions
kono
parents:
diff changeset
2415
kono
parents:
diff changeset
2416 if Present (P) or else Is_Non_Empty_List (L) then
kono
parents:
diff changeset
2417 Set_Statement_Entry;
kono
parents:
diff changeset
2418 end if;
kono
parents:
diff changeset
2419
kono
parents:
diff changeset
2420 return Current_Dominant;
kono
parents:
diff changeset
2421 end Traverse_Declarations_Or_Statements;
kono
parents:
diff changeset
2422
kono
parents:
diff changeset
2423 ------------------------------------------
kono
parents:
diff changeset
2424 -- Traverse_Generic_Package_Declaration --
kono
parents:
diff changeset
2425 ------------------------------------------
kono
parents:
diff changeset
2426
kono
parents:
diff changeset
2427 procedure Traverse_Generic_Package_Declaration (N : Node_Id) is
kono
parents:
diff changeset
2428 begin
kono
parents:
diff changeset
2429 Process_Decisions (Generic_Formal_Declarations (N), 'X', No_Location);
kono
parents:
diff changeset
2430 Traverse_Package_Declaration (N);
kono
parents:
diff changeset
2431 end Traverse_Generic_Package_Declaration;
kono
parents:
diff changeset
2432
kono
parents:
diff changeset
2433 -----------------------------------------
kono
parents:
diff changeset
2434 -- Traverse_Handled_Statement_Sequence --
kono
parents:
diff changeset
2435 -----------------------------------------
kono
parents:
diff changeset
2436
kono
parents:
diff changeset
2437 procedure Traverse_Handled_Statement_Sequence
kono
parents:
diff changeset
2438 (N : Node_Id;
kono
parents:
diff changeset
2439 D : Dominant_Info := No_Dominant)
kono
parents:
diff changeset
2440 is
kono
parents:
diff changeset
2441 Handler : Node_Id;
kono
parents:
diff changeset
2442
kono
parents:
diff changeset
2443 begin
kono
parents:
diff changeset
2444 -- For package bodies without a statement part, the parser adds an empty
kono
parents:
diff changeset
2445 -- one, to normalize the representation. The null statement therein,
kono
parents:
diff changeset
2446 -- which does not come from source, does not get a SCO.
kono
parents:
diff changeset
2447
kono
parents:
diff changeset
2448 if Present (N) and then Comes_From_Source (N) then
kono
parents:
diff changeset
2449 Traverse_Declarations_Or_Statements (Statements (N), D);
kono
parents:
diff changeset
2450
kono
parents:
diff changeset
2451 if Present (Exception_Handlers (N)) then
kono
parents:
diff changeset
2452 Handler := First_Non_Pragma (Exception_Handlers (N));
kono
parents:
diff changeset
2453 while Present (Handler) loop
kono
parents:
diff changeset
2454 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2455 (L => Statements (Handler),
kono
parents:
diff changeset
2456 D => ('E', Handler));
kono
parents:
diff changeset
2457 Next (Handler);
kono
parents:
diff changeset
2458 end loop;
kono
parents:
diff changeset
2459 end if;
kono
parents:
diff changeset
2460 end if;
kono
parents:
diff changeset
2461 end Traverse_Handled_Statement_Sequence;
kono
parents:
diff changeset
2462
kono
parents:
diff changeset
2463 ---------------------------
kono
parents:
diff changeset
2464 -- Traverse_Package_Body --
kono
parents:
diff changeset
2465 ---------------------------
kono
parents:
diff changeset
2466
kono
parents:
diff changeset
2467 procedure Traverse_Package_Body (N : Node_Id) is
kono
parents:
diff changeset
2468 Dom : Dominant_Info;
kono
parents:
diff changeset
2469 begin
kono
parents:
diff changeset
2470 -- The first statement in the handled sequence of statements is
kono
parents:
diff changeset
2471 -- dominated by the elaboration of the last declaration.
kono
parents:
diff changeset
2472
kono
parents:
diff changeset
2473 Dom := Traverse_Declarations_Or_Statements (Declarations (N));
kono
parents:
diff changeset
2474
kono
parents:
diff changeset
2475 Traverse_Handled_Statement_Sequence
kono
parents:
diff changeset
2476 (Handled_Statement_Sequence (N), Dom);
kono
parents:
diff changeset
2477 end Traverse_Package_Body;
kono
parents:
diff changeset
2478
kono
parents:
diff changeset
2479 ----------------------------------
kono
parents:
diff changeset
2480 -- Traverse_Package_Declaration --
kono
parents:
diff changeset
2481 ----------------------------------
kono
parents:
diff changeset
2482
kono
parents:
diff changeset
2483 procedure Traverse_Package_Declaration
kono
parents:
diff changeset
2484 (N : Node_Id;
kono
parents:
diff changeset
2485 D : Dominant_Info := No_Dominant)
kono
parents:
diff changeset
2486 is
kono
parents:
diff changeset
2487 Spec : constant Node_Id := Specification (N);
kono
parents:
diff changeset
2488 Dom : Dominant_Info;
kono
parents:
diff changeset
2489
kono
parents:
diff changeset
2490 begin
kono
parents:
diff changeset
2491 Dom :=
kono
parents:
diff changeset
2492 Traverse_Declarations_Or_Statements (Visible_Declarations (Spec), D);
kono
parents:
diff changeset
2493
kono
parents:
diff changeset
2494 -- First private declaration is dominated by last visible declaration
kono
parents:
diff changeset
2495
kono
parents:
diff changeset
2496 Traverse_Declarations_Or_Statements (Private_Declarations (Spec), Dom);
kono
parents:
diff changeset
2497 end Traverse_Package_Declaration;
kono
parents:
diff changeset
2498
kono
parents:
diff changeset
2499 ------------------------------
kono
parents:
diff changeset
2500 -- Traverse_Sync_Definition --
kono
parents:
diff changeset
2501 ------------------------------
kono
parents:
diff changeset
2502
kono
parents:
diff changeset
2503 procedure Traverse_Sync_Definition (N : Node_Id) is
kono
parents:
diff changeset
2504 Dom_Info : Dominant_Info := ('S', N);
kono
parents:
diff changeset
2505 -- The first declaration is dominated by the protected or task [type]
kono
parents:
diff changeset
2506 -- declaration.
kono
parents:
diff changeset
2507
kono
parents:
diff changeset
2508 Sync_Def : Node_Id;
kono
parents:
diff changeset
2509 -- N's protected or task definition
kono
parents:
diff changeset
2510
kono
parents:
diff changeset
2511 Priv_Decl : List_Id;
kono
parents:
diff changeset
2512 Vis_Decl : List_Id;
kono
parents:
diff changeset
2513 -- Sync_Def's Visible_Declarations and Private_Declarations
kono
parents:
diff changeset
2514
kono
parents:
diff changeset
2515 begin
kono
parents:
diff changeset
2516 case Nkind (N) is
kono
parents:
diff changeset
2517 when N_Protected_Type_Declaration
kono
parents:
diff changeset
2518 | N_Single_Protected_Declaration
kono
parents:
diff changeset
2519 =>
kono
parents:
diff changeset
2520 Sync_Def := Protected_Definition (N);
kono
parents:
diff changeset
2521
kono
parents:
diff changeset
2522 when N_Single_Task_Declaration
kono
parents:
diff changeset
2523 | N_Task_Type_Declaration
kono
parents:
diff changeset
2524 =>
kono
parents:
diff changeset
2525 Sync_Def := Task_Definition (N);
kono
parents:
diff changeset
2526
kono
parents:
diff changeset
2527 when others =>
kono
parents:
diff changeset
2528 raise Program_Error;
kono
parents:
diff changeset
2529 end case;
kono
parents:
diff changeset
2530
kono
parents:
diff changeset
2531 -- Sync_Def may be Empty at least for empty Task_Type_Declarations.
kono
parents:
diff changeset
2532 -- Querying Visible or Private_Declarations is invalid in this case.
kono
parents:
diff changeset
2533
kono
parents:
diff changeset
2534 if Present (Sync_Def) then
kono
parents:
diff changeset
2535 Vis_Decl := Visible_Declarations (Sync_Def);
kono
parents:
diff changeset
2536 Priv_Decl := Private_Declarations (Sync_Def);
kono
parents:
diff changeset
2537 else
kono
parents:
diff changeset
2538 Vis_Decl := No_List;
kono
parents:
diff changeset
2539 Priv_Decl := No_List;
kono
parents:
diff changeset
2540 end if;
kono
parents:
diff changeset
2541
kono
parents:
diff changeset
2542 Dom_Info := Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2543 (L => Vis_Decl,
kono
parents:
diff changeset
2544 D => Dom_Info);
kono
parents:
diff changeset
2545
kono
parents:
diff changeset
2546 -- If visible declarations are present, the first private declaration
kono
parents:
diff changeset
2547 -- is dominated by the last visible declaration.
kono
parents:
diff changeset
2548
kono
parents:
diff changeset
2549 Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2550 (L => Priv_Decl,
kono
parents:
diff changeset
2551 D => Dom_Info);
kono
parents:
diff changeset
2552 end Traverse_Sync_Definition;
kono
parents:
diff changeset
2553
kono
parents:
diff changeset
2554 --------------------------------------
kono
parents:
diff changeset
2555 -- Traverse_Subprogram_Or_Task_Body --
kono
parents:
diff changeset
2556 --------------------------------------
kono
parents:
diff changeset
2557
kono
parents:
diff changeset
2558 procedure Traverse_Subprogram_Or_Task_Body
kono
parents:
diff changeset
2559 (N : Node_Id;
kono
parents:
diff changeset
2560 D : Dominant_Info := No_Dominant)
kono
parents:
diff changeset
2561 is
kono
parents:
diff changeset
2562 Decls : constant List_Id := Declarations (N);
kono
parents:
diff changeset
2563 Dom_Info : Dominant_Info := D;
kono
parents:
diff changeset
2564
kono
parents:
diff changeset
2565 begin
kono
parents:
diff changeset
2566 -- If declarations are present, the first statement is dominated by the
kono
parents:
diff changeset
2567 -- last declaration.
kono
parents:
diff changeset
2568
kono
parents:
diff changeset
2569 Dom_Info := Traverse_Declarations_Or_Statements
kono
parents:
diff changeset
2570 (L => Decls, D => Dom_Info);
kono
parents:
diff changeset
2571
kono
parents:
diff changeset
2572 Traverse_Handled_Statement_Sequence
kono
parents:
diff changeset
2573 (N => Handled_Statement_Sequence (N),
kono
parents:
diff changeset
2574 D => Dom_Info);
kono
parents:
diff changeset
2575 end Traverse_Subprogram_Or_Task_Body;
kono
parents:
diff changeset
2576
kono
parents:
diff changeset
2577 -------------------------
kono
parents:
diff changeset
2578 -- SCO_Record_Filtered --
kono
parents:
diff changeset
2579 -------------------------
kono
parents:
diff changeset
2580
kono
parents:
diff changeset
2581 procedure SCO_Record_Filtered is
kono
parents:
diff changeset
2582 type Decision is record
kono
parents:
diff changeset
2583 Kind : Character;
kono
parents:
diff changeset
2584 -- Type of the SCO decision (see comments for SCO_Table_Entry.C1)
kono
parents:
diff changeset
2585
kono
parents:
diff changeset
2586 Sloc : Source_Location;
kono
parents:
diff changeset
2587
kono
parents:
diff changeset
2588 Top : Nat;
kono
parents:
diff changeset
2589 -- Index in the SCO_Raw_Table for the root operator/condition for the
kono
parents:
diff changeset
2590 -- expression that controls the decision.
kono
parents:
diff changeset
2591 end record;
kono
parents:
diff changeset
2592 -- Decision descriptor: used to gather information about a candidate
kono
parents:
diff changeset
2593 -- SCO decision.
kono
parents:
diff changeset
2594
kono
parents:
diff changeset
2595 package Pending_Decisions is new Table.Table
kono
parents:
diff changeset
2596 (Table_Component_Type => Decision,
kono
parents:
diff changeset
2597 Table_Index_Type => Nat,
kono
parents:
diff changeset
2598 Table_Low_Bound => 1,
kono
parents:
diff changeset
2599 Table_Initial => 1000,
kono
parents:
diff changeset
2600 Table_Increment => 200,
kono
parents:
diff changeset
2601 Table_Name => "Filter_Pending_Decisions");
kono
parents:
diff changeset
2602 -- Table used to hold decisions to process during the collection pass
kono
parents:
diff changeset
2603
kono
parents:
diff changeset
2604 procedure Add_Expression_Tree (Idx : in out Nat);
kono
parents:
diff changeset
2605 -- Add SCO raw table entries for the decision controlling expression
kono
parents:
diff changeset
2606 -- tree starting at Idx to the filtered SCO table.
kono
parents:
diff changeset
2607
kono
parents:
diff changeset
2608 procedure Collect_Decisions
kono
parents:
diff changeset
2609 (D : Decision;
kono
parents:
diff changeset
2610 Next : out Nat);
kono
parents:
diff changeset
2611 -- Collect decisions to add to the filtered SCO table starting at the
kono
parents:
diff changeset
2612 -- D decision (including it and its nested operators/conditions). Set
kono
parents:
diff changeset
2613 -- Next to the first node index passed the whole decision.
kono
parents:
diff changeset
2614
kono
parents:
diff changeset
2615 procedure Compute_Range
kono
parents:
diff changeset
2616 (Idx : in out Nat;
kono
parents:
diff changeset
2617 From : out Source_Location;
kono
parents:
diff changeset
2618 To : out Source_Location);
kono
parents:
diff changeset
2619 -- Compute the source location range for the expression tree starting at
kono
parents:
diff changeset
2620 -- Idx in the SCO raw table. Store its bounds in From and To.
kono
parents:
diff changeset
2621
kono
parents:
diff changeset
2622 function Is_Decision (Idx : Nat) return Boolean;
kono
parents:
diff changeset
2623 -- Return if the expression tree starting at Idx has adjacent nested
kono
parents:
diff changeset
2624 -- nodes that make a decision.
kono
parents:
diff changeset
2625
kono
parents:
diff changeset
2626 procedure Process_Pending_Decisions
kono
parents:
diff changeset
2627 (Original_Decision : SCO_Table_Entry);
kono
parents:
diff changeset
2628 -- Complete the filtered SCO table using collected decisions. Output
kono
parents:
diff changeset
2629 -- decisions inherit the pragma information from the original decision.
kono
parents:
diff changeset
2630
kono
parents:
diff changeset
2631 procedure Search_Nested_Decisions (Idx : in out Nat);
kono
parents:
diff changeset
2632 -- Collect decisions to add to the filtered SCO table starting at the
kono
parents:
diff changeset
2633 -- node at Idx in the SCO raw table. This node must not be part of an
kono
parents:
diff changeset
2634 -- already-processed decision. Set Idx to the first node index passed
kono
parents:
diff changeset
2635 -- the whole expression tree.
kono
parents:
diff changeset
2636
kono
parents:
diff changeset
2637 procedure Skip_Decision
kono
parents:
diff changeset
2638 (Idx : in out Nat;
kono
parents:
diff changeset
2639 Process_Nested_Decisions : Boolean);
kono
parents:
diff changeset
2640 -- Skip all the nodes that belong to the decision starting at Idx. If
kono
parents:
diff changeset
2641 -- Process_Nested_Decision, call Search_Nested_Decisions on the first
kono
parents:
diff changeset
2642 -- nested nodes that do not belong to the decision. Set Idx to the first
kono
parents:
diff changeset
2643 -- node index passed the whole expression tree.
kono
parents:
diff changeset
2644
kono
parents:
diff changeset
2645 -------------------------
kono
parents:
diff changeset
2646 -- Add_Expression_Tree --
kono
parents:
diff changeset
2647 -------------------------
kono
parents:
diff changeset
2648
kono
parents:
diff changeset
2649 procedure Add_Expression_Tree (Idx : in out Nat) is
kono
parents:
diff changeset
2650 Node_Idx : constant Nat := Idx;
kono
parents:
diff changeset
2651 T : SCO_Table_Entry renames SCO_Raw_Table.Table (Node_Idx);
kono
parents:
diff changeset
2652 From : Source_Location;
kono
parents:
diff changeset
2653 To : Source_Location;
kono
parents:
diff changeset
2654
kono
parents:
diff changeset
2655 begin
kono
parents:
diff changeset
2656 case T.C1 is
kono
parents:
diff changeset
2657 when ' ' =>
kono
parents:
diff changeset
2658
kono
parents:
diff changeset
2659 -- This is a single condition. Add an entry for it and move on
kono
parents:
diff changeset
2660
kono
parents:
diff changeset
2661 SCO_Table.Append (T);
kono
parents:
diff changeset
2662 Idx := Idx + 1;
kono
parents:
diff changeset
2663
kono
parents:
diff changeset
2664 when '!' =>
kono
parents:
diff changeset
2665
kono
parents:
diff changeset
2666 -- This is a NOT operator: add an entry for it and browse its
kono
parents:
diff changeset
2667 -- only child.
kono
parents:
diff changeset
2668
kono
parents:
diff changeset
2669 SCO_Table.Append (T);
kono
parents:
diff changeset
2670 Idx := Idx + 1;
kono
parents:
diff changeset
2671 Add_Expression_Tree (Idx);
kono
parents:
diff changeset
2672
kono
parents:
diff changeset
2673 when others =>
kono
parents:
diff changeset
2674
kono
parents:
diff changeset
2675 -- This must be an AND/OR/AND THEN/OR ELSE operator
kono
parents:
diff changeset
2676
kono
parents:
diff changeset
2677 if T.C2 = '?' then
kono
parents:
diff changeset
2678
kono
parents:
diff changeset
2679 -- This is not a short circuit operator: consider this one
kono
parents:
diff changeset
2680 -- and all its children as a single condition.
kono
parents:
diff changeset
2681
kono
parents:
diff changeset
2682 Compute_Range (Idx, From, To);
kono
parents:
diff changeset
2683 SCO_Table.Append
kono
parents:
diff changeset
2684 ((From => From,
kono
parents:
diff changeset
2685 To => To,
kono
parents:
diff changeset
2686 C1 => ' ',
kono
parents:
diff changeset
2687 C2 => 'c',
kono
parents:
diff changeset
2688 Last => False,
kono
parents:
diff changeset
2689 Pragma_Sloc => No_Location,
kono
parents:
diff changeset
2690 Pragma_Aspect_Name => No_Name));
kono
parents:
diff changeset
2691
kono
parents:
diff changeset
2692 else
kono
parents:
diff changeset
2693 -- This is a real short circuit operator: add an entry for
kono
parents:
diff changeset
2694 -- it and browse its children.
kono
parents:
diff changeset
2695
kono
parents:
diff changeset
2696 SCO_Table.Append (T);
kono
parents:
diff changeset
2697 Idx := Idx + 1;
kono
parents:
diff changeset
2698 Add_Expression_Tree (Idx);
kono
parents:
diff changeset
2699 Add_Expression_Tree (Idx);
kono
parents:
diff changeset
2700 end if;
kono
parents:
diff changeset
2701 end case;
kono
parents:
diff changeset
2702 end Add_Expression_Tree;
kono
parents:
diff changeset
2703
kono
parents:
diff changeset
2704 -----------------------
kono
parents:
diff changeset
2705 -- Collect_Decisions --
kono
parents:
diff changeset
2706 -----------------------
kono
parents:
diff changeset
2707
kono
parents:
diff changeset
2708 procedure Collect_Decisions
kono
parents:
diff changeset
2709 (D : Decision;
kono
parents:
diff changeset
2710 Next : out Nat)
kono
parents:
diff changeset
2711 is
kono
parents:
diff changeset
2712 Idx : Nat := D.Top;
kono
parents:
diff changeset
2713
kono
parents:
diff changeset
2714 begin
kono
parents:
diff changeset
2715 if D.Kind /= 'X' or else Is_Decision (D.Top) then
kono
parents:
diff changeset
2716 Pending_Decisions.Append (D);
kono
parents:
diff changeset
2717 end if;
kono
parents:
diff changeset
2718
kono
parents:
diff changeset
2719 Skip_Decision (Idx, True);
kono
parents:
diff changeset
2720 Next := Idx;
kono
parents:
diff changeset
2721 end Collect_Decisions;
kono
parents:
diff changeset
2722
kono
parents:
diff changeset
2723 -------------------
kono
parents:
diff changeset
2724 -- Compute_Range --
kono
parents:
diff changeset
2725 -------------------
kono
parents:
diff changeset
2726
kono
parents:
diff changeset
2727 procedure Compute_Range
kono
parents:
diff changeset
2728 (Idx : in out Nat;
kono
parents:
diff changeset
2729 From : out Source_Location;
kono
parents:
diff changeset
2730 To : out Source_Location)
kono
parents:
diff changeset
2731 is
kono
parents:
diff changeset
2732 Sloc_F : Source_Location := No_Source_Location;
kono
parents:
diff changeset
2733 Sloc_T : Source_Location := No_Source_Location;
kono
parents:
diff changeset
2734
kono
parents:
diff changeset
2735 procedure Process_One;
kono
parents:
diff changeset
2736 -- Process one node of the tree, and recurse over children. Update
kono
parents:
diff changeset
2737 -- Idx during the traversal.
kono
parents:
diff changeset
2738
kono
parents:
diff changeset
2739 -----------------
kono
parents:
diff changeset
2740 -- Process_One --
kono
parents:
diff changeset
2741 -----------------
kono
parents:
diff changeset
2742
kono
parents:
diff changeset
2743 procedure Process_One is
kono
parents:
diff changeset
2744 begin
kono
parents:
diff changeset
2745 if Sloc_F = No_Source_Location
kono
parents:
diff changeset
2746 or else
kono
parents:
diff changeset
2747 SCO_Raw_Table.Table (Idx).From < Sloc_F
kono
parents:
diff changeset
2748 then
kono
parents:
diff changeset
2749 Sloc_F := SCO_Raw_Table.Table (Idx).From;
kono
parents:
diff changeset
2750 end if;
kono
parents:
diff changeset
2751
kono
parents:
diff changeset
2752 if Sloc_T = No_Source_Location
kono
parents:
diff changeset
2753 or else
kono
parents:
diff changeset
2754 Sloc_T < SCO_Raw_Table.Table (Idx).To
kono
parents:
diff changeset
2755 then
kono
parents:
diff changeset
2756 Sloc_T := SCO_Raw_Table.Table (Idx).To;
kono
parents:
diff changeset
2757 end if;
kono
parents:
diff changeset
2758
kono
parents:
diff changeset
2759 if SCO_Raw_Table.Table (Idx).C1 = ' ' then
kono
parents:
diff changeset
2760
kono
parents:
diff changeset
2761 -- This is a condition: nothing special to do
kono
parents:
diff changeset
2762
kono
parents:
diff changeset
2763 Idx := Idx + 1;
kono
parents:
diff changeset
2764
kono
parents:
diff changeset
2765 elsif SCO_Raw_Table.Table (Idx).C1 = '!' then
kono
parents:
diff changeset
2766
kono
parents:
diff changeset
2767 -- The "not" operator has only one operand
kono
parents:
diff changeset
2768
kono
parents:
diff changeset
2769 Idx := Idx + 1;
kono
parents:
diff changeset
2770 Process_One;
kono
parents:
diff changeset
2771
kono
parents:
diff changeset
2772 else
kono
parents:
diff changeset
2773 -- This is an AND THEN or OR ELSE logical operator: follow the
kono
parents:
diff changeset
2774 -- left, then the right operands.
kono
parents:
diff changeset
2775
kono
parents:
diff changeset
2776 Idx := Idx + 1;
kono
parents:
diff changeset
2777
kono
parents:
diff changeset
2778 Process_One;
kono
parents:
diff changeset
2779 Process_One;
kono
parents:
diff changeset
2780 end if;
kono
parents:
diff changeset
2781 end Process_One;
kono
parents:
diff changeset
2782
kono
parents:
diff changeset
2783 -- Start of processing for Compute_Range
kono
parents:
diff changeset
2784
kono
parents:
diff changeset
2785 begin
kono
parents:
diff changeset
2786 Process_One;
kono
parents:
diff changeset
2787 From := Sloc_F;
kono
parents:
diff changeset
2788 To := Sloc_T;
kono
parents:
diff changeset
2789 end Compute_Range;
kono
parents:
diff changeset
2790
kono
parents:
diff changeset
2791 -----------------
kono
parents:
diff changeset
2792 -- Is_Decision --
kono
parents:
diff changeset
2793 -----------------
kono
parents:
diff changeset
2794
kono
parents:
diff changeset
2795 function Is_Decision (Idx : Nat) return Boolean is
kono
parents:
diff changeset
2796 Index : Nat := Idx;
kono
parents:
diff changeset
2797
kono
parents:
diff changeset
2798 begin
kono
parents:
diff changeset
2799 loop
kono
parents:
diff changeset
2800 declare
kono
parents:
diff changeset
2801 T : SCO_Table_Entry renames SCO_Raw_Table.Table (Index);
kono
parents:
diff changeset
2802
kono
parents:
diff changeset
2803 begin
kono
parents:
diff changeset
2804 case T.C1 is
kono
parents:
diff changeset
2805 when ' ' =>
kono
parents:
diff changeset
2806 return False;
kono
parents:
diff changeset
2807
kono
parents:
diff changeset
2808 when '!' =>
kono
parents:
diff changeset
2809
kono
parents:
diff changeset
2810 -- This is a decision iff the only operand of the NOT
kono
parents:
diff changeset
2811 -- operator could be a standalone decision.
kono
parents:
diff changeset
2812
kono
parents:
diff changeset
2813 Index := Idx + 1;
kono
parents:
diff changeset
2814
kono
parents:
diff changeset
2815 when others =>
kono
parents:
diff changeset
2816
kono
parents:
diff changeset
2817 -- This node is a logical operator (and thus could be a
kono
parents:
diff changeset
2818 -- standalone decision) iff it is a short circuit
kono
parents:
diff changeset
2819 -- operator.
kono
parents:
diff changeset
2820
kono
parents:
diff changeset
2821 return T.C2 /= '?';
kono
parents:
diff changeset
2822 end case;
kono
parents:
diff changeset
2823 end;
kono
parents:
diff changeset
2824 end loop;
kono
parents:
diff changeset
2825 end Is_Decision;
kono
parents:
diff changeset
2826
kono
parents:
diff changeset
2827 -------------------------------
kono
parents:
diff changeset
2828 -- Process_Pending_Decisions --
kono
parents:
diff changeset
2829 -------------------------------
kono
parents:
diff changeset
2830
kono
parents:
diff changeset
2831 procedure Process_Pending_Decisions
kono
parents:
diff changeset
2832 (Original_Decision : SCO_Table_Entry)
kono
parents:
diff changeset
2833 is
kono
parents:
diff changeset
2834 begin
kono
parents:
diff changeset
2835 for Index in 1 .. Pending_Decisions.Last loop
kono
parents:
diff changeset
2836 declare
kono
parents:
diff changeset
2837 D : Decision renames Pending_Decisions.Table (Index);
kono
parents:
diff changeset
2838 Idx : Nat := D.Top;
kono
parents:
diff changeset
2839
kono
parents:
diff changeset
2840 begin
kono
parents:
diff changeset
2841 -- Add a SCO table entry for the decision itself
kono
parents:
diff changeset
2842
kono
parents:
diff changeset
2843 pragma Assert (D.Kind /= ' ');
kono
parents:
diff changeset
2844
kono
parents:
diff changeset
2845 SCO_Table.Append
kono
parents:
diff changeset
2846 ((To => No_Source_Location,
kono
parents:
diff changeset
2847 From => D.Sloc,
kono
parents:
diff changeset
2848 C1 => D.Kind,
kono
parents:
diff changeset
2849 C2 => ' ',
kono
parents:
diff changeset
2850 Last => False,
kono
parents:
diff changeset
2851 Pragma_Sloc => Original_Decision.Pragma_Sloc,
kono
parents:
diff changeset
2852 Pragma_Aspect_Name =>
kono
parents:
diff changeset
2853 Original_Decision.Pragma_Aspect_Name));
kono
parents:
diff changeset
2854
kono
parents:
diff changeset
2855 -- Then add ones for its nested operators/operands. Do not
kono
parents:
diff changeset
2856 -- forget to tag its *last* entry as such.
kono
parents:
diff changeset
2857
kono
parents:
diff changeset
2858 Add_Expression_Tree (Idx);
kono
parents:
diff changeset
2859 SCO_Table.Table (SCO_Table.Last).Last := True;
kono
parents:
diff changeset
2860 end;
kono
parents:
diff changeset
2861 end loop;
kono
parents:
diff changeset
2862
kono
parents:
diff changeset
2863 -- Clear the pending decisions list
kono
parents:
diff changeset
2864 Pending_Decisions.Set_Last (0);
kono
parents:
diff changeset
2865 end Process_Pending_Decisions;
kono
parents:
diff changeset
2866
kono
parents:
diff changeset
2867 -----------------------------
kono
parents:
diff changeset
2868 -- Search_Nested_Decisions --
kono
parents:
diff changeset
2869 -----------------------------
kono
parents:
diff changeset
2870
kono
parents:
diff changeset
2871 procedure Search_Nested_Decisions (Idx : in out Nat) is
kono
parents:
diff changeset
2872 begin
kono
parents:
diff changeset
2873 loop
kono
parents:
diff changeset
2874 declare
kono
parents:
diff changeset
2875 T : SCO_Table_Entry renames SCO_Raw_Table.Table (Idx);
kono
parents:
diff changeset
2876
kono
parents:
diff changeset
2877 begin
kono
parents:
diff changeset
2878 case T.C1 is
kono
parents:
diff changeset
2879 when ' ' =>
kono
parents:
diff changeset
2880 Idx := Idx + 1;
kono
parents:
diff changeset
2881 exit;
kono
parents:
diff changeset
2882
kono
parents:
diff changeset
2883 when '!' =>
kono
parents:
diff changeset
2884 Collect_Decisions
kono
parents:
diff changeset
2885 ((Kind => 'X',
kono
parents:
diff changeset
2886 Sloc => T.From,
kono
parents:
diff changeset
2887 Top => Idx),
kono
parents:
diff changeset
2888 Idx);
kono
parents:
diff changeset
2889 exit;
kono
parents:
diff changeset
2890
kono
parents:
diff changeset
2891 when others =>
kono
parents:
diff changeset
2892 if T.C2 = '?' then
kono
parents:
diff changeset
2893
kono
parents:
diff changeset
2894 -- This is not a logical operator: start looking for
kono
parents:
diff changeset
2895 -- nested decisions from here. Recurse over the left
kono
parents:
diff changeset
2896 -- child and let the loop take care of the right one.
kono
parents:
diff changeset
2897
kono
parents:
diff changeset
2898 Idx := Idx + 1;
kono
parents:
diff changeset
2899 Search_Nested_Decisions (Idx);
kono
parents:
diff changeset
2900
kono
parents:
diff changeset
2901 else
kono
parents:
diff changeset
2902 -- We found a nested decision
kono
parents:
diff changeset
2903
kono
parents:
diff changeset
2904 Collect_Decisions
kono
parents:
diff changeset
2905 ((Kind => 'X',
kono
parents:
diff changeset
2906 Sloc => T.From,
kono
parents:
diff changeset
2907 Top => Idx),
kono
parents:
diff changeset
2908 Idx);
kono
parents:
diff changeset
2909 exit;
kono
parents:
diff changeset
2910 end if;
kono
parents:
diff changeset
2911 end case;
kono
parents:
diff changeset
2912 end;
kono
parents:
diff changeset
2913 end loop;
kono
parents:
diff changeset
2914 end Search_Nested_Decisions;
kono
parents:
diff changeset
2915
kono
parents:
diff changeset
2916 -------------------
kono
parents:
diff changeset
2917 -- Skip_Decision --
kono
parents:
diff changeset
2918 -------------------
kono
parents:
diff changeset
2919
kono
parents:
diff changeset
2920 procedure Skip_Decision
kono
parents:
diff changeset
2921 (Idx : in out Nat;
kono
parents:
diff changeset
2922 Process_Nested_Decisions : Boolean)
kono
parents:
diff changeset
2923 is
kono
parents:
diff changeset
2924 begin
kono
parents:
diff changeset
2925 loop
kono
parents:
diff changeset
2926 declare
kono
parents:
diff changeset
2927 T : SCO_Table_Entry renames SCO_Raw_Table.Table (Idx);
kono
parents:
diff changeset
2928
kono
parents:
diff changeset
2929 begin
kono
parents:
diff changeset
2930 Idx := Idx + 1;
kono
parents:
diff changeset
2931
kono
parents:
diff changeset
2932 case T.C1 is
kono
parents:
diff changeset
2933 when ' ' =>
kono
parents:
diff changeset
2934 exit;
kono
parents:
diff changeset
2935
kono
parents:
diff changeset
2936 when '!' =>
kono
parents:
diff changeset
2937
kono
parents:
diff changeset
2938 -- This NOT operator belongs to the outside decision:
kono
parents:
diff changeset
2939 -- just skip it.
kono
parents:
diff changeset
2940
kono
parents:
diff changeset
2941 null;
kono
parents:
diff changeset
2942
kono
parents:
diff changeset
2943 when others =>
kono
parents:
diff changeset
2944 if T.C2 = '?' and then Process_Nested_Decisions then
kono
parents:
diff changeset
2945
kono
parents:
diff changeset
2946 -- This is not a logical operator: start looking for
kono
parents:
diff changeset
2947 -- nested decisions from here. Recurse over the left
kono
parents:
diff changeset
2948 -- child and let the loop take care of the right one.
kono
parents:
diff changeset
2949
kono
parents:
diff changeset
2950 Search_Nested_Decisions (Idx);
kono
parents:
diff changeset
2951
kono
parents:
diff changeset
2952 else
kono
parents:
diff changeset
2953 -- This is a logical operator, so it belongs to the
kono
parents:
diff changeset
2954 -- outside decision: skip its left child, then let the
kono
parents:
diff changeset
2955 -- loop take care of the right one.
kono
parents:
diff changeset
2956
kono
parents:
diff changeset
2957 Skip_Decision (Idx, Process_Nested_Decisions);
kono
parents:
diff changeset
2958 end if;
kono
parents:
diff changeset
2959 end case;
kono
parents:
diff changeset
2960 end;
kono
parents:
diff changeset
2961 end loop;
kono
parents:
diff changeset
2962 end Skip_Decision;
kono
parents:
diff changeset
2963
kono
parents:
diff changeset
2964 -- Start of processing for SCO_Record_Filtered
kono
parents:
diff changeset
2965
kono
parents:
diff changeset
2966 begin
kono
parents:
diff changeset
2967 -- Filtering must happen only once: do nothing if it this pass was
kono
parents:
diff changeset
2968 -- already run.
kono
parents:
diff changeset
2969
kono
parents:
diff changeset
2970 if SCO_Generation_State = Filtered then
kono
parents:
diff changeset
2971 return;
kono
parents:
diff changeset
2972 else
kono
parents:
diff changeset
2973 pragma Assert (SCO_Generation_State = Raw);
kono
parents:
diff changeset
2974 SCO_Generation_State := Filtered;
kono
parents:
diff changeset
2975 end if;
kono
parents:
diff changeset
2976
kono
parents:
diff changeset
2977 -- Loop through all SCO entries under SCO units
kono
parents:
diff changeset
2978
kono
parents:
diff changeset
2979 for Unit_Idx in 1 .. SCO_Unit_Table.Last loop
kono
parents:
diff changeset
2980 declare
kono
parents:
diff changeset
2981 Unit : SCO_Unit_Table_Entry
kono
parents:
diff changeset
2982 renames SCO_Unit_Table.Table (Unit_Idx);
kono
parents:
diff changeset
2983
kono
parents:
diff changeset
2984 Idx : Nat := Unit.From;
kono
parents:
diff changeset
2985 -- Index of the current SCO raw table entry
kono
parents:
diff changeset
2986
kono
parents:
diff changeset
2987 New_From : constant Nat := SCO_Table.Last + 1;
kono
parents:
diff changeset
2988 -- After copying SCO enties of interest to the final table, we
kono
parents:
diff changeset
2989 -- will have to change the From/To indexes this unit targets.
kono
parents:
diff changeset
2990 -- This constant keeps track of the new From index.
kono
parents:
diff changeset
2991
kono
parents:
diff changeset
2992 begin
kono
parents:
diff changeset
2993 while Idx <= Unit.To loop
kono
parents:
diff changeset
2994 declare
kono
parents:
diff changeset
2995 T : SCO_Table_Entry renames SCO_Raw_Table.Table (Idx);
kono
parents:
diff changeset
2996
kono
parents:
diff changeset
2997 begin
kono
parents:
diff changeset
2998 case T.C1 is
kono
parents:
diff changeset
2999
kono
parents:
diff changeset
3000 -- Decision (of any kind, including pragmas and aspects)
kono
parents:
diff changeset
3001
kono
parents:
diff changeset
3002 when 'E' | 'G' | 'I' | 'W' | 'X' | 'P' | 'a' | 'A' =>
kono
parents:
diff changeset
3003 if SCO_Pragma_Disabled (T.Pragma_Sloc) then
kono
parents:
diff changeset
3004
kono
parents:
diff changeset
3005 -- Skip SCO entries for decisions in disabled
kono
parents:
diff changeset
3006 -- constructs (pragmas or aspects).
kono
parents:
diff changeset
3007
kono
parents:
diff changeset
3008 Idx := Idx + 1;
kono
parents:
diff changeset
3009 Skip_Decision (Idx, False);
kono
parents:
diff changeset
3010
kono
parents:
diff changeset
3011 else
kono
parents:
diff changeset
3012 Collect_Decisions
kono
parents:
diff changeset
3013 ((Kind => T.C1,
kono
parents:
diff changeset
3014 Sloc => T.From,
kono
parents:
diff changeset
3015 Top => Idx + 1),
kono
parents:
diff changeset
3016 Idx);
kono
parents:
diff changeset
3017 Process_Pending_Decisions (T);
kono
parents:
diff changeset
3018 end if;
kono
parents:
diff changeset
3019
kono
parents:
diff changeset
3020 -- There is no translation/filtering to do for other kind
kono
parents:
diff changeset
3021 -- of SCO items (statements, dominance markers, etc.).
kono
parents:
diff changeset
3022
kono
parents:
diff changeset
3023 when '|' | '&' | '!' | ' ' =>
kono
parents:
diff changeset
3024
kono
parents:
diff changeset
3025 -- SCO logical operators and conditions cannot exist
kono
parents:
diff changeset
3026 -- on their own: they must be inside a decision (such
kono
parents:
diff changeset
3027 -- entries must have been skipped by
kono
parents:
diff changeset
3028 -- Collect_Decisions).
kono
parents:
diff changeset
3029
kono
parents:
diff changeset
3030 raise Program_Error;
kono
parents:
diff changeset
3031
kono
parents:
diff changeset
3032 when others =>
kono
parents:
diff changeset
3033 SCO_Table.Append (T);
kono
parents:
diff changeset
3034 Idx := Idx + 1;
kono
parents:
diff changeset
3035 end case;
kono
parents:
diff changeset
3036 end;
kono
parents:
diff changeset
3037 end loop;
kono
parents:
diff changeset
3038
kono
parents:
diff changeset
3039 -- Now, update the SCO entry indexes in the unit entry
kono
parents:
diff changeset
3040
kono
parents:
diff changeset
3041 Unit.From := New_From;
kono
parents:
diff changeset
3042 Unit.To := SCO_Table.Last;
kono
parents:
diff changeset
3043 end;
kono
parents:
diff changeset
3044 end loop;
kono
parents:
diff changeset
3045
kono
parents:
diff changeset
3046 -- Then clear the raw table to free bytes
kono
parents:
diff changeset
3047
kono
parents:
diff changeset
3048 SCO_Raw_Table.Free;
kono
parents:
diff changeset
3049 end SCO_Record_Filtered;
kono
parents:
diff changeset
3050
kono
parents:
diff changeset
3051 end Par_SCO;