annotate gcc/ada/sem_dim.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 -- S E M _ D I M --
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) 2011-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 Einfo; use Einfo;
kono
parents:
diff changeset
29 with Errout; use Errout;
kono
parents:
diff changeset
30 with Exp_Util; use Exp_Util;
kono
parents:
diff changeset
31 with Lib; use Lib;
kono
parents:
diff changeset
32 with Namet; use Namet;
kono
parents:
diff changeset
33 with Nlists; use Nlists;
kono
parents:
diff changeset
34 with Nmake; use Nmake;
kono
parents:
diff changeset
35 with Opt; use Opt;
kono
parents:
diff changeset
36 with Rtsfind; use Rtsfind;
kono
parents:
diff changeset
37 with Sem; use Sem;
kono
parents:
diff changeset
38 with Sem_Aux; use Sem_Aux;
kono
parents:
diff changeset
39 with Sem_Eval; use Sem_Eval;
kono
parents:
diff changeset
40 with Sem_Res; use Sem_Res;
kono
parents:
diff changeset
41 with Sem_Util; use Sem_Util;
kono
parents:
diff changeset
42 with Sinfo; use Sinfo;
kono
parents:
diff changeset
43 with Sinput; use Sinput;
kono
parents:
diff changeset
44 with Snames; use Snames;
kono
parents:
diff changeset
45 with Stand; use Stand;
kono
parents:
diff changeset
46 with Stringt; use Stringt;
kono
parents:
diff changeset
47 with Table;
kono
parents:
diff changeset
48 with Tbuild; use Tbuild;
kono
parents:
diff changeset
49 with Uintp; use Uintp;
kono
parents:
diff changeset
50 with Urealp; use Urealp;
kono
parents:
diff changeset
51
kono
parents:
diff changeset
52 with GNAT.HTable;
kono
parents:
diff changeset
53
kono
parents:
diff changeset
54 package body Sem_Dim is
kono
parents:
diff changeset
55
kono
parents:
diff changeset
56 -------------------------
kono
parents:
diff changeset
57 -- Rational Arithmetic --
kono
parents:
diff changeset
58 -------------------------
kono
parents:
diff changeset
59
kono
parents:
diff changeset
60 type Whole is new Int;
kono
parents:
diff changeset
61 subtype Positive_Whole is Whole range 1 .. Whole'Last;
kono
parents:
diff changeset
62
kono
parents:
diff changeset
63 type Rational is record
kono
parents:
diff changeset
64 Numerator : Whole;
kono
parents:
diff changeset
65 Denominator : Positive_Whole;
kono
parents:
diff changeset
66 end record;
kono
parents:
diff changeset
67
kono
parents:
diff changeset
68 Zero : constant Rational := Rational'(Numerator => 0,
kono
parents:
diff changeset
69 Denominator => 1);
kono
parents:
diff changeset
70
kono
parents:
diff changeset
71 No_Rational : constant Rational := Rational'(Numerator => 0,
kono
parents:
diff changeset
72 Denominator => 2);
kono
parents:
diff changeset
73 -- Used to indicate an expression that cannot be interpreted as a rational
kono
parents:
diff changeset
74 -- Returned value of the Create_Rational_From routine when parameter Expr
kono
parents:
diff changeset
75 -- is not a static representation of a rational.
kono
parents:
diff changeset
76
kono
parents:
diff changeset
77 -- Rational constructors
kono
parents:
diff changeset
78
kono
parents:
diff changeset
79 function "+" (Right : Whole) return Rational;
kono
parents:
diff changeset
80 function GCD (Left, Right : Whole) return Int;
kono
parents:
diff changeset
81 function Reduce (X : Rational) return Rational;
kono
parents:
diff changeset
82
kono
parents:
diff changeset
83 -- Unary operator for Rational
kono
parents:
diff changeset
84
kono
parents:
diff changeset
85 function "-" (Right : Rational) return Rational;
kono
parents:
diff changeset
86 function "abs" (Right : Rational) return Rational;
kono
parents:
diff changeset
87
kono
parents:
diff changeset
88 -- Rational operations for Rationals
kono
parents:
diff changeset
89
kono
parents:
diff changeset
90 function "+" (Left, Right : Rational) return Rational;
kono
parents:
diff changeset
91 function "-" (Left, Right : Rational) return Rational;
kono
parents:
diff changeset
92 function "*" (Left, Right : Rational) return Rational;
kono
parents:
diff changeset
93 function "/" (Left, Right : Rational) return Rational;
kono
parents:
diff changeset
94
kono
parents:
diff changeset
95 ------------------
kono
parents:
diff changeset
96 -- System Types --
kono
parents:
diff changeset
97 ------------------
kono
parents:
diff changeset
98
kono
parents:
diff changeset
99 Max_Number_Of_Dimensions : constant := 7;
kono
parents:
diff changeset
100 -- Maximum number of dimensions in a dimension system
kono
parents:
diff changeset
101
kono
parents:
diff changeset
102 High_Position_Bound : constant := Max_Number_Of_Dimensions;
kono
parents:
diff changeset
103 Invalid_Position : constant := 0;
kono
parents:
diff changeset
104 Low_Position_Bound : constant := 1;
kono
parents:
diff changeset
105
kono
parents:
diff changeset
106 subtype Dimension_Position is
kono
parents:
diff changeset
107 Nat range Invalid_Position .. High_Position_Bound;
kono
parents:
diff changeset
108
kono
parents:
diff changeset
109 type Name_Array is
kono
parents:
diff changeset
110 array (Dimension_Position range
kono
parents:
diff changeset
111 Low_Position_Bound .. High_Position_Bound) of Name_Id;
kono
parents:
diff changeset
112 -- Store the names of all units within a system
kono
parents:
diff changeset
113
kono
parents:
diff changeset
114 No_Names : constant Name_Array := (others => No_Name);
kono
parents:
diff changeset
115
kono
parents:
diff changeset
116 type Symbol_Array is
kono
parents:
diff changeset
117 array (Dimension_Position range
kono
parents:
diff changeset
118 Low_Position_Bound .. High_Position_Bound) of String_Id;
kono
parents:
diff changeset
119 -- Store the symbols of all units within a system
kono
parents:
diff changeset
120
kono
parents:
diff changeset
121 No_Symbols : constant Symbol_Array := (others => No_String);
kono
parents:
diff changeset
122
kono
parents:
diff changeset
123 -- The following record should be documented field by field
kono
parents:
diff changeset
124
kono
parents:
diff changeset
125 type System_Type is record
kono
parents:
diff changeset
126 Type_Decl : Node_Id;
kono
parents:
diff changeset
127 Unit_Names : Name_Array;
kono
parents:
diff changeset
128 Unit_Symbols : Symbol_Array;
kono
parents:
diff changeset
129 Dim_Symbols : Symbol_Array;
kono
parents:
diff changeset
130 Count : Dimension_Position;
kono
parents:
diff changeset
131 end record;
kono
parents:
diff changeset
132
kono
parents:
diff changeset
133 Null_System : constant System_Type :=
kono
parents:
diff changeset
134 (Empty, No_Names, No_Symbols, No_Symbols, Invalid_Position);
kono
parents:
diff changeset
135
kono
parents:
diff changeset
136 subtype System_Id is Nat;
kono
parents:
diff changeset
137
kono
parents:
diff changeset
138 -- The following table maps types to systems
kono
parents:
diff changeset
139
kono
parents:
diff changeset
140 package System_Table is new Table.Table (
kono
parents:
diff changeset
141 Table_Component_Type => System_Type,
kono
parents:
diff changeset
142 Table_Index_Type => System_Id,
kono
parents:
diff changeset
143 Table_Low_Bound => 1,
kono
parents:
diff changeset
144 Table_Initial => 5,
kono
parents:
diff changeset
145 Table_Increment => 5,
kono
parents:
diff changeset
146 Table_Name => "System_Table");
kono
parents:
diff changeset
147
kono
parents:
diff changeset
148 --------------------
kono
parents:
diff changeset
149 -- Dimension Type --
kono
parents:
diff changeset
150 --------------------
kono
parents:
diff changeset
151
kono
parents:
diff changeset
152 type Dimension_Type is
kono
parents:
diff changeset
153 array (Dimension_Position range
kono
parents:
diff changeset
154 Low_Position_Bound .. High_Position_Bound) of Rational;
kono
parents:
diff changeset
155
kono
parents:
diff changeset
156 Null_Dimension : constant Dimension_Type := (others => Zero);
kono
parents:
diff changeset
157
kono
parents:
diff changeset
158 type Dimension_Table_Range is range 0 .. 510;
kono
parents:
diff changeset
159 function Dimension_Table_Hash (Key : Node_Id) return Dimension_Table_Range;
kono
parents:
diff changeset
160
kono
parents:
diff changeset
161 -- The following table associates nodes with dimensions
kono
parents:
diff changeset
162
kono
parents:
diff changeset
163 package Dimension_Table is new
kono
parents:
diff changeset
164 GNAT.HTable.Simple_HTable
kono
parents:
diff changeset
165 (Header_Num => Dimension_Table_Range,
kono
parents:
diff changeset
166 Element => Dimension_Type,
kono
parents:
diff changeset
167 No_Element => Null_Dimension,
kono
parents:
diff changeset
168 Key => Node_Id,
kono
parents:
diff changeset
169 Hash => Dimension_Table_Hash,
kono
parents:
diff changeset
170 Equal => "=");
kono
parents:
diff changeset
171
kono
parents:
diff changeset
172 ------------------
kono
parents:
diff changeset
173 -- Symbol Types --
kono
parents:
diff changeset
174 ------------------
kono
parents:
diff changeset
175
kono
parents:
diff changeset
176 type Symbol_Table_Range is range 0 .. 510;
kono
parents:
diff changeset
177 function Symbol_Table_Hash (Key : Entity_Id) return Symbol_Table_Range;
kono
parents:
diff changeset
178
kono
parents:
diff changeset
179 -- Each subtype with a dimension has a symbolic representation of the
kono
parents:
diff changeset
180 -- related unit. This table establishes a relation between the subtype
kono
parents:
diff changeset
181 -- and the symbol.
kono
parents:
diff changeset
182
kono
parents:
diff changeset
183 package Symbol_Table is new
kono
parents:
diff changeset
184 GNAT.HTable.Simple_HTable
kono
parents:
diff changeset
185 (Header_Num => Symbol_Table_Range,
kono
parents:
diff changeset
186 Element => String_Id,
kono
parents:
diff changeset
187 No_Element => No_String,
kono
parents:
diff changeset
188 Key => Entity_Id,
kono
parents:
diff changeset
189 Hash => Symbol_Table_Hash,
kono
parents:
diff changeset
190 Equal => "=");
kono
parents:
diff changeset
191
kono
parents:
diff changeset
192 -- The following array enumerates all contexts which may contain or
kono
parents:
diff changeset
193 -- produce a dimension.
kono
parents:
diff changeset
194
kono
parents:
diff changeset
195 OK_For_Dimension : constant array (Node_Kind) of Boolean :=
kono
parents:
diff changeset
196 (N_Attribute_Reference => True,
kono
parents:
diff changeset
197 N_Case_Expression => True,
kono
parents:
diff changeset
198 N_Expanded_Name => True,
kono
parents:
diff changeset
199 N_Explicit_Dereference => True,
kono
parents:
diff changeset
200 N_Defining_Identifier => True,
kono
parents:
diff changeset
201 N_Function_Call => True,
kono
parents:
diff changeset
202 N_Identifier => True,
kono
parents:
diff changeset
203 N_If_Expression => True,
kono
parents:
diff changeset
204 N_Indexed_Component => True,
kono
parents:
diff changeset
205 N_Integer_Literal => True,
kono
parents:
diff changeset
206 N_Op_Abs => True,
kono
parents:
diff changeset
207 N_Op_Add => True,
kono
parents:
diff changeset
208 N_Op_Divide => True,
kono
parents:
diff changeset
209 N_Op_Expon => True,
kono
parents:
diff changeset
210 N_Op_Minus => True,
kono
parents:
diff changeset
211 N_Op_Mod => True,
kono
parents:
diff changeset
212 N_Op_Multiply => True,
kono
parents:
diff changeset
213 N_Op_Plus => True,
kono
parents:
diff changeset
214 N_Op_Rem => True,
kono
parents:
diff changeset
215 N_Op_Subtract => True,
kono
parents:
diff changeset
216 N_Qualified_Expression => True,
kono
parents:
diff changeset
217 N_Real_Literal => True,
kono
parents:
diff changeset
218 N_Selected_Component => True,
kono
parents:
diff changeset
219 N_Slice => True,
kono
parents:
diff changeset
220 N_Type_Conversion => True,
kono
parents:
diff changeset
221 N_Unchecked_Type_Conversion => True,
kono
parents:
diff changeset
222
kono
parents:
diff changeset
223 others => False);
kono
parents:
diff changeset
224
kono
parents:
diff changeset
225 -----------------------
kono
parents:
diff changeset
226 -- Local Subprograms --
kono
parents:
diff changeset
227 -----------------------
kono
parents:
diff changeset
228
kono
parents:
diff changeset
229 procedure Analyze_Dimension_Assignment_Statement (N : Node_Id);
kono
parents:
diff changeset
230 -- Subroutine of Analyze_Dimension for assignment statement. Check that the
kono
parents:
diff changeset
231 -- dimensions of the left-hand side and the right-hand side of N match.
kono
parents:
diff changeset
232
kono
parents:
diff changeset
233 procedure Analyze_Dimension_Binary_Op (N : Node_Id);
kono
parents:
diff changeset
234 -- Subroutine of Analyze_Dimension for binary operators. Check the
kono
parents:
diff changeset
235 -- dimensions of the right and the left operand permit the operation.
kono
parents:
diff changeset
236 -- Then, evaluate the resulting dimensions for each binary operator.
kono
parents:
diff changeset
237
kono
parents:
diff changeset
238 procedure Analyze_Dimension_Component_Declaration (N : Node_Id);
kono
parents:
diff changeset
239 -- Subroutine of Analyze_Dimension for component declaration. Check that
kono
parents:
diff changeset
240 -- the dimensions of the type of N and of the expression match.
kono
parents:
diff changeset
241
kono
parents:
diff changeset
242 procedure Analyze_Dimension_Extended_Return_Statement (N : Node_Id);
kono
parents:
diff changeset
243 -- Subroutine of Analyze_Dimension for extended return statement. Check
kono
parents:
diff changeset
244 -- that the dimensions of the returned type and of the returned object
kono
parents:
diff changeset
245 -- match.
kono
parents:
diff changeset
246
kono
parents:
diff changeset
247 procedure Analyze_Dimension_Has_Etype (N : Node_Id);
kono
parents:
diff changeset
248 -- Subroutine of Analyze_Dimension for a subset of N_Has_Etype denoted by
kono
parents:
diff changeset
249 -- the list below:
kono
parents:
diff changeset
250 -- N_Attribute_Reference
kono
parents:
diff changeset
251 -- N_Identifier
kono
parents:
diff changeset
252 -- N_Indexed_Component
kono
parents:
diff changeset
253 -- N_Qualified_Expression
kono
parents:
diff changeset
254 -- N_Selected_Component
kono
parents:
diff changeset
255 -- N_Slice
kono
parents:
diff changeset
256 -- N_Type_Conversion
kono
parents:
diff changeset
257 -- N_Unchecked_Type_Conversion
kono
parents:
diff changeset
258
kono
parents:
diff changeset
259 procedure Analyze_Dimension_Case_Expression (N : Node_Id);
kono
parents:
diff changeset
260 -- Verify that all alternatives have the same dimension
kono
parents:
diff changeset
261
kono
parents:
diff changeset
262 procedure Analyze_Dimension_If_Expression (N : Node_Id);
kono
parents:
diff changeset
263 -- Verify that all alternatives have the same dimension
kono
parents:
diff changeset
264
kono
parents:
diff changeset
265 procedure Analyze_Dimension_Number_Declaration (N : Node_Id);
kono
parents:
diff changeset
266 -- Procedure to analyze dimension of expression in a number declaration.
kono
parents:
diff changeset
267 -- This allows a named number to have nontrivial dimensions, while by
kono
parents:
diff changeset
268 -- default a named number is dimensionless.
kono
parents:
diff changeset
269
kono
parents:
diff changeset
270 procedure Analyze_Dimension_Object_Declaration (N : Node_Id);
kono
parents:
diff changeset
271 -- Subroutine of Analyze_Dimension for object declaration. Check that
kono
parents:
diff changeset
272 -- the dimensions of the object type and the dimensions of the expression
kono
parents:
diff changeset
273 -- (if expression is present) match. Note that when the expression is
kono
parents:
diff changeset
274 -- a literal, no error is returned. This special case allows object
kono
parents:
diff changeset
275 -- declaration such as: m : constant Length := 1.0;
kono
parents:
diff changeset
276
kono
parents:
diff changeset
277 procedure Analyze_Dimension_Object_Renaming_Declaration (N : Node_Id);
kono
parents:
diff changeset
278 -- Subroutine of Analyze_Dimension for object renaming declaration. Check
kono
parents:
diff changeset
279 -- the dimensions of the type and of the renamed object name of N match.
kono
parents:
diff changeset
280
kono
parents:
diff changeset
281 procedure Analyze_Dimension_Simple_Return_Statement (N : Node_Id);
kono
parents:
diff changeset
282 -- Subroutine of Analyze_Dimension for simple return statement
kono
parents:
diff changeset
283 -- Check that the dimensions of the returned type and of the returned
kono
parents:
diff changeset
284 -- expression match.
kono
parents:
diff changeset
285
kono
parents:
diff changeset
286 procedure Analyze_Dimension_Subtype_Declaration (N : Node_Id);
kono
parents:
diff changeset
287 -- Subroutine of Analyze_Dimension for subtype declaration. Propagate the
kono
parents:
diff changeset
288 -- dimensions from the parent type to the identifier of N. Note that if
kono
parents:
diff changeset
289 -- both the identifier and the parent type of N are not dimensionless,
kono
parents:
diff changeset
290 -- return an error.
kono
parents:
diff changeset
291
kono
parents:
diff changeset
292 procedure Analyze_Dimension_Type_Conversion (N : Node_Id);
kono
parents:
diff changeset
293 -- Type conversions handle conversions between literals and dimensioned
kono
parents:
diff changeset
294 -- types, from dimensioned types to their base type, and between different
kono
parents:
diff changeset
295 -- dimensioned systems. Dimensions of the conversion are obtained either
kono
parents:
diff changeset
296 -- from those of the expression, or from the target type, and dimensional
kono
parents:
diff changeset
297 -- consistency must be checked when converting between values belonging
kono
parents:
diff changeset
298 -- to different dimensioned systems.
kono
parents:
diff changeset
299
kono
parents:
diff changeset
300 procedure Analyze_Dimension_Unary_Op (N : Node_Id);
kono
parents:
diff changeset
301 -- Subroutine of Analyze_Dimension for unary operators. For Plus, Minus and
kono
parents:
diff changeset
302 -- Abs operators, propagate the dimensions from the operand to N.
kono
parents:
diff changeset
303
kono
parents:
diff changeset
304 function Create_Rational_From
kono
parents:
diff changeset
305 (Expr : Node_Id;
kono
parents:
diff changeset
306 Complain : Boolean) return Rational;
kono
parents:
diff changeset
307 -- Given an arbitrary expression Expr, return a valid rational if Expr can
kono
parents:
diff changeset
308 -- be interpreted as a rational. Otherwise return No_Rational and also an
kono
parents:
diff changeset
309 -- error message if Complain is set to True.
kono
parents:
diff changeset
310
kono
parents:
diff changeset
311 function Dimensions_Of (N : Node_Id) return Dimension_Type;
kono
parents:
diff changeset
312 -- Return the dimension vector of node N
kono
parents:
diff changeset
313
kono
parents:
diff changeset
314 function Dimensions_Msg_Of
kono
parents:
diff changeset
315 (N : Node_Id;
kono
parents:
diff changeset
316 Description_Needed : Boolean := False) return String;
kono
parents:
diff changeset
317 -- Given a node N, return the dimension symbols of N, preceded by "has
kono
parents:
diff changeset
318 -- dimension" if Description_Needed. if N is dimensionless, return "'[']",
kono
parents:
diff changeset
319 -- or "is dimensionless" if Description_Needed.
kono
parents:
diff changeset
320
kono
parents:
diff changeset
321 function Dimension_System_Root (T : Entity_Id) return Entity_Id;
kono
parents:
diff changeset
322 -- Given a type that has dimension information, return the type that is the
kono
parents:
diff changeset
323 -- root of its dimension system, e.g. Mks_Type. If T is not a dimensioned
kono
parents:
diff changeset
324 -- type, i.e. a standard numeric type, return Empty.
kono
parents:
diff changeset
325
kono
parents:
diff changeset
326 procedure Dim_Warning_For_Numeric_Literal (N : Node_Id; Typ : Entity_Id);
kono
parents:
diff changeset
327 -- Issue a warning on the given numeric literal N to indicate that the
kono
parents:
diff changeset
328 -- compiler made the assumption that the literal is not dimensionless
kono
parents:
diff changeset
329 -- but has the dimension of Typ.
kono
parents:
diff changeset
330
kono
parents:
diff changeset
331 procedure Eval_Op_Expon_With_Rational_Exponent
kono
parents:
diff changeset
332 (N : Node_Id;
kono
parents:
diff changeset
333 Exponent_Value : Rational);
kono
parents:
diff changeset
334 -- Evaluate the exponent it is a rational and the operand has a dimension
kono
parents:
diff changeset
335
kono
parents:
diff changeset
336 function Exists (Dim : Dimension_Type) return Boolean;
kono
parents:
diff changeset
337 -- Returns True iff Dim does not denote the null dimension
kono
parents:
diff changeset
338
kono
parents:
diff changeset
339 function Exists (Str : String_Id) return Boolean;
kono
parents:
diff changeset
340 -- Returns True iff Str does not denote No_String
kono
parents:
diff changeset
341
kono
parents:
diff changeset
342 function Exists (Sys : System_Type) return Boolean;
kono
parents:
diff changeset
343 -- Returns True iff Sys does not denote the null system
kono
parents:
diff changeset
344
kono
parents:
diff changeset
345 function From_Dim_To_Str_Of_Dim_Symbols
kono
parents:
diff changeset
346 (Dims : Dimension_Type;
kono
parents:
diff changeset
347 System : System_Type;
kono
parents:
diff changeset
348 In_Error_Msg : Boolean := False) return String_Id;
kono
parents:
diff changeset
349 -- Given a dimension vector and a dimension system, return the proper
kono
parents:
diff changeset
350 -- string of dimension symbols. If In_Error_Msg is True (i.e. the String_Id
kono
parents:
diff changeset
351 -- will be used to issue an error message) then this routine has a special
kono
parents:
diff changeset
352 -- handling for the insertion characters * or [ which must be preceded by
kono
parents:
diff changeset
353 -- a quote ' to be placed literally into the message.
kono
parents:
diff changeset
354
kono
parents:
diff changeset
355 function From_Dim_To_Str_Of_Unit_Symbols
kono
parents:
diff changeset
356 (Dims : Dimension_Type;
kono
parents:
diff changeset
357 System : System_Type) return String_Id;
kono
parents:
diff changeset
358 -- Given a dimension vector and a dimension system, return the proper
kono
parents:
diff changeset
359 -- string of unit symbols.
kono
parents:
diff changeset
360
kono
parents:
diff changeset
361 function Is_Dim_IO_Package_Entity (E : Entity_Id) return Boolean;
kono
parents:
diff changeset
362 -- Return True if E is the package entity of System.Dim.Float_IO or
kono
parents:
diff changeset
363 -- System.Dim.Integer_IO.
kono
parents:
diff changeset
364
kono
parents:
diff changeset
365 function Is_Invalid (Position : Dimension_Position) return Boolean;
kono
parents:
diff changeset
366 -- Return True if Pos denotes the invalid position
kono
parents:
diff changeset
367
kono
parents:
diff changeset
368 procedure Move_Dimensions (From : Node_Id; To : Node_Id);
kono
parents:
diff changeset
369 -- Copy dimension vector of From to To and delete dimension vector of From
kono
parents:
diff changeset
370
kono
parents:
diff changeset
371 procedure Remove_Dimensions (N : Node_Id);
kono
parents:
diff changeset
372 -- Remove the dimension vector of node N
kono
parents:
diff changeset
373
kono
parents:
diff changeset
374 procedure Set_Dimensions (N : Node_Id; Val : Dimension_Type);
kono
parents:
diff changeset
375 -- Associate a dimension vector with a node
kono
parents:
diff changeset
376
kono
parents:
diff changeset
377 procedure Set_Symbol (E : Entity_Id; Val : String_Id);
kono
parents:
diff changeset
378 -- Associate a symbol representation of a dimension vector with a subtype
kono
parents:
diff changeset
379
kono
parents:
diff changeset
380 function String_From_Numeric_Literal (N : Node_Id) return String_Id;
kono
parents:
diff changeset
381 -- Return the string that corresponds to the numeric litteral N as it
kono
parents:
diff changeset
382 -- appears in the source.
kono
parents:
diff changeset
383
kono
parents:
diff changeset
384 function Symbol_Of (E : Entity_Id) return String_Id;
kono
parents:
diff changeset
385 -- E denotes a subtype with a dimension. Return the symbol representation
kono
parents:
diff changeset
386 -- of the dimension vector.
kono
parents:
diff changeset
387
kono
parents:
diff changeset
388 function System_Of (E : Entity_Id) return System_Type;
kono
parents:
diff changeset
389 -- E denotes a type, return associated system of the type if it has one
kono
parents:
diff changeset
390
kono
parents:
diff changeset
391 ---------
kono
parents:
diff changeset
392 -- "+" --
kono
parents:
diff changeset
393 ---------
kono
parents:
diff changeset
394
kono
parents:
diff changeset
395 function "+" (Right : Whole) return Rational is
kono
parents:
diff changeset
396 begin
kono
parents:
diff changeset
397 return Rational'(Numerator => Right, Denominator => 1);
kono
parents:
diff changeset
398 end "+";
kono
parents:
diff changeset
399
kono
parents:
diff changeset
400 function "+" (Left, Right : Rational) return Rational is
kono
parents:
diff changeset
401 R : constant Rational :=
kono
parents:
diff changeset
402 Rational'(Numerator => Left.Numerator * Right.Denominator +
kono
parents:
diff changeset
403 Left.Denominator * Right.Numerator,
kono
parents:
diff changeset
404 Denominator => Left.Denominator * Right.Denominator);
kono
parents:
diff changeset
405 begin
kono
parents:
diff changeset
406 return Reduce (R);
kono
parents:
diff changeset
407 end "+";
kono
parents:
diff changeset
408
kono
parents:
diff changeset
409 ---------
kono
parents:
diff changeset
410 -- "-" --
kono
parents:
diff changeset
411 ---------
kono
parents:
diff changeset
412
kono
parents:
diff changeset
413 function "-" (Right : Rational) return Rational is
kono
parents:
diff changeset
414 begin
kono
parents:
diff changeset
415 return Rational'(Numerator => -Right.Numerator,
kono
parents:
diff changeset
416 Denominator => Right.Denominator);
kono
parents:
diff changeset
417 end "-";
kono
parents:
diff changeset
418
kono
parents:
diff changeset
419 function "-" (Left, Right : Rational) return Rational is
kono
parents:
diff changeset
420 R : constant Rational :=
kono
parents:
diff changeset
421 Rational'(Numerator => Left.Numerator * Right.Denominator -
kono
parents:
diff changeset
422 Left.Denominator * Right.Numerator,
kono
parents:
diff changeset
423 Denominator => Left.Denominator * Right.Denominator);
kono
parents:
diff changeset
424
kono
parents:
diff changeset
425 begin
kono
parents:
diff changeset
426 return Reduce (R);
kono
parents:
diff changeset
427 end "-";
kono
parents:
diff changeset
428
kono
parents:
diff changeset
429 ---------
kono
parents:
diff changeset
430 -- "*" --
kono
parents:
diff changeset
431 ---------
kono
parents:
diff changeset
432
kono
parents:
diff changeset
433 function "*" (Left, Right : Rational) return Rational is
kono
parents:
diff changeset
434 R : constant Rational :=
kono
parents:
diff changeset
435 Rational'(Numerator => Left.Numerator * Right.Numerator,
kono
parents:
diff changeset
436 Denominator => Left.Denominator * Right.Denominator);
kono
parents:
diff changeset
437 begin
kono
parents:
diff changeset
438 return Reduce (R);
kono
parents:
diff changeset
439 end "*";
kono
parents:
diff changeset
440
kono
parents:
diff changeset
441 ---------
kono
parents:
diff changeset
442 -- "/" --
kono
parents:
diff changeset
443 ---------
kono
parents:
diff changeset
444
kono
parents:
diff changeset
445 function "/" (Left, Right : Rational) return Rational is
kono
parents:
diff changeset
446 R : constant Rational := abs Right;
kono
parents:
diff changeset
447 L : Rational := Left;
kono
parents:
diff changeset
448
kono
parents:
diff changeset
449 begin
kono
parents:
diff changeset
450 if Right.Numerator < 0 then
kono
parents:
diff changeset
451 L.Numerator := Whole (-Integer (L.Numerator));
kono
parents:
diff changeset
452 end if;
kono
parents:
diff changeset
453
kono
parents:
diff changeset
454 return Reduce (Rational'(Numerator => L.Numerator * R.Denominator,
kono
parents:
diff changeset
455 Denominator => L.Denominator * R.Numerator));
kono
parents:
diff changeset
456 end "/";
kono
parents:
diff changeset
457
kono
parents:
diff changeset
458 -----------
kono
parents:
diff changeset
459 -- "abs" --
kono
parents:
diff changeset
460 -----------
kono
parents:
diff changeset
461
kono
parents:
diff changeset
462 function "abs" (Right : Rational) return Rational is
kono
parents:
diff changeset
463 begin
kono
parents:
diff changeset
464 return Rational'(Numerator => abs Right.Numerator,
kono
parents:
diff changeset
465 Denominator => Right.Denominator);
kono
parents:
diff changeset
466 end "abs";
kono
parents:
diff changeset
467
kono
parents:
diff changeset
468 ------------------------------
kono
parents:
diff changeset
469 -- Analyze_Aspect_Dimension --
kono
parents:
diff changeset
470 ------------------------------
kono
parents:
diff changeset
471
kono
parents:
diff changeset
472 -- with Dimension =>
kono
parents:
diff changeset
473 -- ([Symbol =>] SYMBOL, DIMENSION_VALUE {, DIMENSION_Value})
kono
parents:
diff changeset
474 --
kono
parents:
diff changeset
475 -- SYMBOL ::= STRING_LITERAL | CHARACTER_LITERAL
kono
parents:
diff changeset
476
kono
parents:
diff changeset
477 -- DIMENSION_VALUE ::=
kono
parents:
diff changeset
478 -- RATIONAL
kono
parents:
diff changeset
479 -- | others => RATIONAL
kono
parents:
diff changeset
480 -- | DISCRETE_CHOICE_LIST => RATIONAL
kono
parents:
diff changeset
481
kono
parents:
diff changeset
482 -- RATIONAL ::= [-] NUMERIC_LITERAL [/ NUMERIC_LITERAL]
kono
parents:
diff changeset
483
kono
parents:
diff changeset
484 -- Note that when the dimensioned type is an integer type, then any
kono
parents:
diff changeset
485 -- dimension value must be an integer literal.
kono
parents:
diff changeset
486
kono
parents:
diff changeset
487 procedure Analyze_Aspect_Dimension
kono
parents:
diff changeset
488 (N : Node_Id;
kono
parents:
diff changeset
489 Id : Entity_Id;
kono
parents:
diff changeset
490 Aggr : Node_Id)
kono
parents:
diff changeset
491 is
kono
parents:
diff changeset
492 Def_Id : constant Entity_Id := Defining_Identifier (N);
kono
parents:
diff changeset
493
kono
parents:
diff changeset
494 Processed : array (Dimension_Type'Range) of Boolean := (others => False);
kono
parents:
diff changeset
495 -- This array is used when processing ranges or Others_Choice as part of
kono
parents:
diff changeset
496 -- the dimension aggregate.
kono
parents:
diff changeset
497
kono
parents:
diff changeset
498 Dimensions : Dimension_Type := Null_Dimension;
kono
parents:
diff changeset
499
kono
parents:
diff changeset
500 procedure Extract_Power
kono
parents:
diff changeset
501 (Expr : Node_Id;
kono
parents:
diff changeset
502 Position : Dimension_Position);
kono
parents:
diff changeset
503 -- Given an expression with denotes a rational number, read the number
kono
parents:
diff changeset
504 -- and associate it with Position in Dimensions.
kono
parents:
diff changeset
505
kono
parents:
diff changeset
506 function Position_In_System
kono
parents:
diff changeset
507 (Id : Node_Id;
kono
parents:
diff changeset
508 System : System_Type) return Dimension_Position;
kono
parents:
diff changeset
509 -- Given an identifier which denotes a dimension, return the position of
kono
parents:
diff changeset
510 -- that dimension within System.
kono
parents:
diff changeset
511
kono
parents:
diff changeset
512 -------------------
kono
parents:
diff changeset
513 -- Extract_Power --
kono
parents:
diff changeset
514 -------------------
kono
parents:
diff changeset
515
kono
parents:
diff changeset
516 procedure Extract_Power
kono
parents:
diff changeset
517 (Expr : Node_Id;
kono
parents:
diff changeset
518 Position : Dimension_Position)
kono
parents:
diff changeset
519 is
kono
parents:
diff changeset
520 begin
kono
parents:
diff changeset
521 Dimensions (Position) := Create_Rational_From (Expr, True);
kono
parents:
diff changeset
522 Processed (Position) := True;
kono
parents:
diff changeset
523
kono
parents:
diff changeset
524 -- If the dimensioned root type is an integer type, it is not
kono
parents:
diff changeset
525 -- particularly useful, and fractional dimensions do not make
kono
parents:
diff changeset
526 -- much sense for such types, so previously we used to reject
kono
parents:
diff changeset
527 -- dimensions of integer types that were not integer literals.
kono
parents:
diff changeset
528 -- However, the manipulation of dimensions does not depend on
kono
parents:
diff changeset
529 -- the kind of root type, so we can accept this usage for rare
kono
parents:
diff changeset
530 -- cases where dimensions are specified for integer values.
kono
parents:
diff changeset
531
kono
parents:
diff changeset
532 end Extract_Power;
kono
parents:
diff changeset
533
kono
parents:
diff changeset
534 ------------------------
kono
parents:
diff changeset
535 -- Position_In_System --
kono
parents:
diff changeset
536 ------------------------
kono
parents:
diff changeset
537
kono
parents:
diff changeset
538 function Position_In_System
kono
parents:
diff changeset
539 (Id : Node_Id;
kono
parents:
diff changeset
540 System : System_Type) return Dimension_Position
kono
parents:
diff changeset
541 is
kono
parents:
diff changeset
542 Dimension_Name : constant Name_Id := Chars (Id);
kono
parents:
diff changeset
543
kono
parents:
diff changeset
544 begin
kono
parents:
diff changeset
545 for Position in System.Unit_Names'Range loop
kono
parents:
diff changeset
546 if Dimension_Name = System.Unit_Names (Position) then
kono
parents:
diff changeset
547 return Position;
kono
parents:
diff changeset
548 end if;
kono
parents:
diff changeset
549 end loop;
kono
parents:
diff changeset
550
kono
parents:
diff changeset
551 return Invalid_Position;
kono
parents:
diff changeset
552 end Position_In_System;
kono
parents:
diff changeset
553
kono
parents:
diff changeset
554 -- Local variables
kono
parents:
diff changeset
555
kono
parents:
diff changeset
556 Assoc : Node_Id;
kono
parents:
diff changeset
557 Choice : Node_Id;
kono
parents:
diff changeset
558 Expr : Node_Id;
kono
parents:
diff changeset
559 Num_Choices : Nat := 0;
kono
parents:
diff changeset
560 Num_Dimensions : Nat := 0;
kono
parents:
diff changeset
561 Others_Seen : Boolean := False;
kono
parents:
diff changeset
562 Position : Nat := 0;
kono
parents:
diff changeset
563 Sub_Ind : Node_Id;
kono
parents:
diff changeset
564 Symbol : String_Id := No_String;
kono
parents:
diff changeset
565 Symbol_Expr : Node_Id;
kono
parents:
diff changeset
566 System : System_Type;
kono
parents:
diff changeset
567 Typ : Entity_Id;
kono
parents:
diff changeset
568
kono
parents:
diff changeset
569 Errors_Count : Nat;
kono
parents:
diff changeset
570 -- Errors_Count is a count of errors detected by the compiler so far
kono
parents:
diff changeset
571 -- just before the extraction of symbol, names and values in the
kono
parents:
diff changeset
572 -- aggregate (Step 2).
kono
parents:
diff changeset
573 --
kono
parents:
diff changeset
574 -- At the end of the analysis, there is a check to verify that this
kono
parents:
diff changeset
575 -- count equals to Serious_Errors_Detected i.e. no erros have been
kono
parents:
diff changeset
576 -- encountered during the process. Otherwise the Dimension_Table is
kono
parents:
diff changeset
577 -- not filled.
kono
parents:
diff changeset
578
kono
parents:
diff changeset
579 -- Start of processing for Analyze_Aspect_Dimension
kono
parents:
diff changeset
580
kono
parents:
diff changeset
581 begin
kono
parents:
diff changeset
582 -- STEP 1: Legality of aspect
kono
parents:
diff changeset
583
kono
parents:
diff changeset
584 if Nkind (N) /= N_Subtype_Declaration then
kono
parents:
diff changeset
585 Error_Msg_NE ("aspect& must apply to subtype declaration", N, Id);
kono
parents:
diff changeset
586 return;
kono
parents:
diff changeset
587 end if;
kono
parents:
diff changeset
588
kono
parents:
diff changeset
589 Sub_Ind := Subtype_Indication (N);
kono
parents:
diff changeset
590 Typ := Etype (Sub_Ind);
kono
parents:
diff changeset
591 System := System_Of (Typ);
kono
parents:
diff changeset
592
kono
parents:
diff changeset
593 if Nkind (Sub_Ind) = N_Subtype_Indication then
kono
parents:
diff changeset
594 Error_Msg_NE
kono
parents:
diff changeset
595 ("constraint not allowed with aspect&", Constraint (Sub_Ind), Id);
kono
parents:
diff changeset
596 return;
kono
parents:
diff changeset
597 end if;
kono
parents:
diff changeset
598
kono
parents:
diff changeset
599 -- The dimension declarations are useless if the parent type does not
kono
parents:
diff changeset
600 -- declare a valid system.
kono
parents:
diff changeset
601
kono
parents:
diff changeset
602 if not Exists (System) then
kono
parents:
diff changeset
603 Error_Msg_NE
kono
parents:
diff changeset
604 ("parent type of& lacks dimension system", Sub_Ind, Def_Id);
kono
parents:
diff changeset
605 return;
kono
parents:
diff changeset
606 end if;
kono
parents:
diff changeset
607
kono
parents:
diff changeset
608 if Nkind (Aggr) /= N_Aggregate then
kono
parents:
diff changeset
609 Error_Msg_N ("aggregate expected", Aggr);
kono
parents:
diff changeset
610 return;
kono
parents:
diff changeset
611 end if;
kono
parents:
diff changeset
612
kono
parents:
diff changeset
613 -- STEP 2: Symbol, Names and values extraction
kono
parents:
diff changeset
614
kono
parents:
diff changeset
615 -- Get the number of errors detected by the compiler so far
kono
parents:
diff changeset
616
kono
parents:
diff changeset
617 Errors_Count := Serious_Errors_Detected;
kono
parents:
diff changeset
618
kono
parents:
diff changeset
619 -- STEP 2a: Symbol extraction
kono
parents:
diff changeset
620
kono
parents:
diff changeset
621 -- The first entry in the aggregate may be the symbolic representation
kono
parents:
diff changeset
622 -- of the quantity.
kono
parents:
diff changeset
623
kono
parents:
diff changeset
624 -- Positional symbol argument
kono
parents:
diff changeset
625
kono
parents:
diff changeset
626 Symbol_Expr := First (Expressions (Aggr));
kono
parents:
diff changeset
627
kono
parents:
diff changeset
628 -- Named symbol argument
kono
parents:
diff changeset
629
kono
parents:
diff changeset
630 if No (Symbol_Expr)
kono
parents:
diff changeset
631 or else not Nkind_In (Symbol_Expr, N_Character_Literal,
kono
parents:
diff changeset
632 N_String_Literal)
kono
parents:
diff changeset
633 then
kono
parents:
diff changeset
634 Symbol_Expr := Empty;
kono
parents:
diff changeset
635
kono
parents:
diff changeset
636 -- Component associations present
kono
parents:
diff changeset
637
kono
parents:
diff changeset
638 if Present (Component_Associations (Aggr)) then
kono
parents:
diff changeset
639 Assoc := First (Component_Associations (Aggr));
kono
parents:
diff changeset
640 Choice := First (Choices (Assoc));
kono
parents:
diff changeset
641
kono
parents:
diff changeset
642 if No (Next (Choice)) and then Nkind (Choice) = N_Identifier then
kono
parents:
diff changeset
643
kono
parents:
diff changeset
644 -- Symbol component association is present
kono
parents:
diff changeset
645
kono
parents:
diff changeset
646 if Chars (Choice) = Name_Symbol then
kono
parents:
diff changeset
647 Num_Choices := Num_Choices + 1;
kono
parents:
diff changeset
648 Symbol_Expr := Expression (Assoc);
kono
parents:
diff changeset
649
kono
parents:
diff changeset
650 -- Verify symbol expression is a string or a character
kono
parents:
diff changeset
651
kono
parents:
diff changeset
652 if not Nkind_In (Symbol_Expr, N_Character_Literal,
kono
parents:
diff changeset
653 N_String_Literal)
kono
parents:
diff changeset
654 then
kono
parents:
diff changeset
655 Symbol_Expr := Empty;
kono
parents:
diff changeset
656 Error_Msg_N
kono
parents:
diff changeset
657 ("symbol expression must be character or string",
kono
parents:
diff changeset
658 Symbol_Expr);
kono
parents:
diff changeset
659 end if;
kono
parents:
diff changeset
660
kono
parents:
diff changeset
661 -- Special error if no Symbol choice but expression is string
kono
parents:
diff changeset
662 -- or character.
kono
parents:
diff changeset
663
kono
parents:
diff changeset
664 elsif Nkind_In (Expression (Assoc), N_Character_Literal,
kono
parents:
diff changeset
665 N_String_Literal)
kono
parents:
diff changeset
666 then
kono
parents:
diff changeset
667 Num_Choices := Num_Choices + 1;
kono
parents:
diff changeset
668 Error_Msg_N
kono
parents:
diff changeset
669 ("optional component Symbol expected, found&", Choice);
kono
parents:
diff changeset
670 end if;
kono
parents:
diff changeset
671 end if;
kono
parents:
diff changeset
672 end if;
kono
parents:
diff changeset
673 end if;
kono
parents:
diff changeset
674
kono
parents:
diff changeset
675 -- STEP 2b: Names and values extraction
kono
parents:
diff changeset
676
kono
parents:
diff changeset
677 -- Positional elements
kono
parents:
diff changeset
678
kono
parents:
diff changeset
679 Expr := First (Expressions (Aggr));
kono
parents:
diff changeset
680
kono
parents:
diff changeset
681 -- Skip the symbol expression when present
kono
parents:
diff changeset
682
kono
parents:
diff changeset
683 if Present (Symbol_Expr) and then Num_Choices = 0 then
kono
parents:
diff changeset
684 Expr := Next (Expr);
kono
parents:
diff changeset
685 end if;
kono
parents:
diff changeset
686
kono
parents:
diff changeset
687 Position := Low_Position_Bound;
kono
parents:
diff changeset
688 while Present (Expr) loop
kono
parents:
diff changeset
689 if Position > High_Position_Bound then
kono
parents:
diff changeset
690 Error_Msg_N
kono
parents:
diff changeset
691 ("type& has more dimensions than system allows", Def_Id);
kono
parents:
diff changeset
692 exit;
kono
parents:
diff changeset
693 end if;
kono
parents:
diff changeset
694
kono
parents:
diff changeset
695 Extract_Power (Expr, Position);
kono
parents:
diff changeset
696
kono
parents:
diff changeset
697 Position := Position + 1;
kono
parents:
diff changeset
698 Num_Dimensions := Num_Dimensions + 1;
kono
parents:
diff changeset
699
kono
parents:
diff changeset
700 Next (Expr);
kono
parents:
diff changeset
701 end loop;
kono
parents:
diff changeset
702
kono
parents:
diff changeset
703 -- Named elements
kono
parents:
diff changeset
704
kono
parents:
diff changeset
705 Assoc := First (Component_Associations (Aggr));
kono
parents:
diff changeset
706
kono
parents:
diff changeset
707 -- Skip the symbol association when present
kono
parents:
diff changeset
708
kono
parents:
diff changeset
709 if Num_Choices = 1 then
kono
parents:
diff changeset
710 Next (Assoc);
kono
parents:
diff changeset
711 end if;
kono
parents:
diff changeset
712
kono
parents:
diff changeset
713 while Present (Assoc) loop
kono
parents:
diff changeset
714 Expr := Expression (Assoc);
kono
parents:
diff changeset
715
kono
parents:
diff changeset
716 Choice := First (Choices (Assoc));
kono
parents:
diff changeset
717 while Present (Choice) loop
kono
parents:
diff changeset
718
kono
parents:
diff changeset
719 -- Identifier case: NAME => EXPRESSION
kono
parents:
diff changeset
720
kono
parents:
diff changeset
721 if Nkind (Choice) = N_Identifier then
kono
parents:
diff changeset
722 Position := Position_In_System (Choice, System);
kono
parents:
diff changeset
723
kono
parents:
diff changeset
724 if Is_Invalid (Position) then
kono
parents:
diff changeset
725 Error_Msg_N ("dimension name& not part of system", Choice);
kono
parents:
diff changeset
726 else
kono
parents:
diff changeset
727 Extract_Power (Expr, Position);
kono
parents:
diff changeset
728 end if;
kono
parents:
diff changeset
729
kono
parents:
diff changeset
730 -- Range case: NAME .. NAME => EXPRESSION
kono
parents:
diff changeset
731
kono
parents:
diff changeset
732 elsif Nkind (Choice) = N_Range then
kono
parents:
diff changeset
733 declare
kono
parents:
diff changeset
734 Low : constant Node_Id := Low_Bound (Choice);
kono
parents:
diff changeset
735 High : constant Node_Id := High_Bound (Choice);
kono
parents:
diff changeset
736 Low_Pos : Dimension_Position;
kono
parents:
diff changeset
737 High_Pos : Dimension_Position;
kono
parents:
diff changeset
738
kono
parents:
diff changeset
739 begin
kono
parents:
diff changeset
740 if Nkind (Low) /= N_Identifier then
kono
parents:
diff changeset
741 Error_Msg_N ("bound must denote a dimension name", Low);
kono
parents:
diff changeset
742
kono
parents:
diff changeset
743 elsif Nkind (High) /= N_Identifier then
kono
parents:
diff changeset
744 Error_Msg_N ("bound must denote a dimension name", High);
kono
parents:
diff changeset
745
kono
parents:
diff changeset
746 else
kono
parents:
diff changeset
747 Low_Pos := Position_In_System (Low, System);
kono
parents:
diff changeset
748 High_Pos := Position_In_System (High, System);
kono
parents:
diff changeset
749
kono
parents:
diff changeset
750 if Is_Invalid (Low_Pos) then
kono
parents:
diff changeset
751 Error_Msg_N ("dimension name& not part of system",
kono
parents:
diff changeset
752 Low);
kono
parents:
diff changeset
753
kono
parents:
diff changeset
754 elsif Is_Invalid (High_Pos) then
kono
parents:
diff changeset
755 Error_Msg_N ("dimension name& not part of system",
kono
parents:
diff changeset
756 High);
kono
parents:
diff changeset
757
kono
parents:
diff changeset
758 elsif Low_Pos > High_Pos then
kono
parents:
diff changeset
759 Error_Msg_N ("expected low to high range", Choice);
kono
parents:
diff changeset
760
kono
parents:
diff changeset
761 else
kono
parents:
diff changeset
762 for Position in Low_Pos .. High_Pos loop
kono
parents:
diff changeset
763 Extract_Power (Expr, Position);
kono
parents:
diff changeset
764 end loop;
kono
parents:
diff changeset
765 end if;
kono
parents:
diff changeset
766 end if;
kono
parents:
diff changeset
767 end;
kono
parents:
diff changeset
768
kono
parents:
diff changeset
769 -- Others case: OTHERS => EXPRESSION
kono
parents:
diff changeset
770
kono
parents:
diff changeset
771 elsif Nkind (Choice) = N_Others_Choice then
kono
parents:
diff changeset
772 if Present (Next (Choice)) or else Present (Prev (Choice)) then
kono
parents:
diff changeset
773 Error_Msg_N
kono
parents:
diff changeset
774 ("OTHERS must appear alone in a choice list", Choice);
kono
parents:
diff changeset
775
kono
parents:
diff changeset
776 elsif Present (Next (Assoc)) then
kono
parents:
diff changeset
777 Error_Msg_N
kono
parents:
diff changeset
778 ("OTHERS must appear last in an aggregate", Choice);
kono
parents:
diff changeset
779
kono
parents:
diff changeset
780 elsif Others_Seen then
kono
parents:
diff changeset
781 Error_Msg_N ("multiple OTHERS not allowed", Choice);
kono
parents:
diff changeset
782
kono
parents:
diff changeset
783 else
kono
parents:
diff changeset
784 -- Fill the non-processed dimensions with the default value
kono
parents:
diff changeset
785 -- supplied by others.
kono
parents:
diff changeset
786
kono
parents:
diff changeset
787 for Position in Processed'Range loop
kono
parents:
diff changeset
788 if not Processed (Position) then
kono
parents:
diff changeset
789 Extract_Power (Expr, Position);
kono
parents:
diff changeset
790 end if;
kono
parents:
diff changeset
791 end loop;
kono
parents:
diff changeset
792 end if;
kono
parents:
diff changeset
793
kono
parents:
diff changeset
794 Others_Seen := True;
kono
parents:
diff changeset
795
kono
parents:
diff changeset
796 -- All other cases are illegal declarations of dimension names
kono
parents:
diff changeset
797
kono
parents:
diff changeset
798 else
kono
parents:
diff changeset
799 Error_Msg_NE ("wrong syntax for aspect&", Choice, Id);
kono
parents:
diff changeset
800 end if;
kono
parents:
diff changeset
801
kono
parents:
diff changeset
802 Num_Choices := Num_Choices + 1;
kono
parents:
diff changeset
803 Next (Choice);
kono
parents:
diff changeset
804 end loop;
kono
parents:
diff changeset
805
kono
parents:
diff changeset
806 Num_Dimensions := Num_Dimensions + 1;
kono
parents:
diff changeset
807 Next (Assoc);
kono
parents:
diff changeset
808 end loop;
kono
parents:
diff changeset
809
kono
parents:
diff changeset
810 -- STEP 3: Consistency of system and dimensions
kono
parents:
diff changeset
811
kono
parents:
diff changeset
812 if Present (First (Expressions (Aggr)))
kono
parents:
diff changeset
813 and then (First (Expressions (Aggr)) /= Symbol_Expr
kono
parents:
diff changeset
814 or else Present (Next (Symbol_Expr)))
kono
parents:
diff changeset
815 and then (Num_Choices > 1
kono
parents:
diff changeset
816 or else (Num_Choices = 1 and then not Others_Seen))
kono
parents:
diff changeset
817 then
kono
parents:
diff changeset
818 Error_Msg_N
kono
parents:
diff changeset
819 ("named associations cannot follow positional associations", Aggr);
kono
parents:
diff changeset
820 end if;
kono
parents:
diff changeset
821
kono
parents:
diff changeset
822 if Num_Dimensions > System.Count then
kono
parents:
diff changeset
823 Error_Msg_N ("type& has more dimensions than system allows", Def_Id);
kono
parents:
diff changeset
824
kono
parents:
diff changeset
825 elsif Num_Dimensions < System.Count and then not Others_Seen then
kono
parents:
diff changeset
826 Error_Msg_N ("type& has less dimensions than system allows", Def_Id);
kono
parents:
diff changeset
827 end if;
kono
parents:
diff changeset
828
kono
parents:
diff changeset
829 -- STEP 4: Dimension symbol extraction
kono
parents:
diff changeset
830
kono
parents:
diff changeset
831 if Present (Symbol_Expr) then
kono
parents:
diff changeset
832 if Nkind (Symbol_Expr) = N_Character_Literal then
kono
parents:
diff changeset
833 Start_String;
kono
parents:
diff changeset
834 Store_String_Char (UI_To_CC (Char_Literal_Value (Symbol_Expr)));
kono
parents:
diff changeset
835 Symbol := End_String;
kono
parents:
diff changeset
836
kono
parents:
diff changeset
837 else
kono
parents:
diff changeset
838 Symbol := Strval (Symbol_Expr);
kono
parents:
diff changeset
839 end if;
kono
parents:
diff changeset
840
kono
parents:
diff changeset
841 if String_Length (Symbol) = 0 then
kono
parents:
diff changeset
842 Error_Msg_N ("empty string not allowed here", Symbol_Expr);
kono
parents:
diff changeset
843 end if;
kono
parents:
diff changeset
844 end if;
kono
parents:
diff changeset
845
kono
parents:
diff changeset
846 -- STEP 5: Storage of extracted values
kono
parents:
diff changeset
847
kono
parents:
diff changeset
848 -- Check that no errors have been detected during the analysis
kono
parents:
diff changeset
849
kono
parents:
diff changeset
850 if Errors_Count = Serious_Errors_Detected then
kono
parents:
diff changeset
851
kono
parents:
diff changeset
852 -- Check for useless declaration
kono
parents:
diff changeset
853
kono
parents:
diff changeset
854 if Symbol = No_String and then not Exists (Dimensions) then
kono
parents:
diff changeset
855 Error_Msg_N ("useless dimension declaration", Aggr);
kono
parents:
diff changeset
856 end if;
kono
parents:
diff changeset
857
kono
parents:
diff changeset
858 if Symbol /= No_String then
kono
parents:
diff changeset
859 Set_Symbol (Def_Id, Symbol);
kono
parents:
diff changeset
860 end if;
kono
parents:
diff changeset
861
kono
parents:
diff changeset
862 if Exists (Dimensions) then
kono
parents:
diff changeset
863 Set_Dimensions (Def_Id, Dimensions);
kono
parents:
diff changeset
864 end if;
kono
parents:
diff changeset
865 end if;
kono
parents:
diff changeset
866 end Analyze_Aspect_Dimension;
kono
parents:
diff changeset
867
kono
parents:
diff changeset
868 -------------------------------------
kono
parents:
diff changeset
869 -- Analyze_Aspect_Dimension_System --
kono
parents:
diff changeset
870 -------------------------------------
kono
parents:
diff changeset
871
kono
parents:
diff changeset
872 -- with Dimension_System => (DIMENSION {, DIMENSION});
kono
parents:
diff changeset
873
kono
parents:
diff changeset
874 -- DIMENSION ::= (
kono
parents:
diff changeset
875 -- [Unit_Name =>] IDENTIFIER,
kono
parents:
diff changeset
876 -- [Unit_Symbol =>] SYMBOL,
kono
parents:
diff changeset
877 -- [Dim_Symbol =>] SYMBOL)
kono
parents:
diff changeset
878
kono
parents:
diff changeset
879 procedure Analyze_Aspect_Dimension_System
kono
parents:
diff changeset
880 (N : Node_Id;
kono
parents:
diff changeset
881 Id : Entity_Id;
kono
parents:
diff changeset
882 Aggr : Node_Id)
kono
parents:
diff changeset
883 is
kono
parents:
diff changeset
884 function Is_Derived_Numeric_Type (N : Node_Id) return Boolean;
kono
parents:
diff changeset
885 -- Determine whether type declaration N denotes a numeric derived type
kono
parents:
diff changeset
886
kono
parents:
diff changeset
887 -------------------------------
kono
parents:
diff changeset
888 -- Is_Derived_Numeric_Type --
kono
parents:
diff changeset
889 -------------------------------
kono
parents:
diff changeset
890
kono
parents:
diff changeset
891 function Is_Derived_Numeric_Type (N : Node_Id) return Boolean is
kono
parents:
diff changeset
892 begin
kono
parents:
diff changeset
893 return
kono
parents:
diff changeset
894 Nkind (N) = N_Full_Type_Declaration
kono
parents:
diff changeset
895 and then Nkind (Type_Definition (N)) = N_Derived_Type_Definition
kono
parents:
diff changeset
896 and then Is_Numeric_Type
kono
parents:
diff changeset
897 (Entity (Subtype_Indication (Type_Definition (N))));
kono
parents:
diff changeset
898 end Is_Derived_Numeric_Type;
kono
parents:
diff changeset
899
kono
parents:
diff changeset
900 -- Local variables
kono
parents:
diff changeset
901
kono
parents:
diff changeset
902 Assoc : Node_Id;
kono
parents:
diff changeset
903 Choice : Node_Id;
kono
parents:
diff changeset
904 Dim_Aggr : Node_Id;
kono
parents:
diff changeset
905 Dim_Symbol : Node_Id;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
906 Dim_Symbols : Symbol_Array := No_Symbols;
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
907 Dim_System : System_Type := Null_System;
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
908 Position : Dimension_Position := Invalid_Position;
111
kono
parents:
diff changeset
909 Unit_Name : Node_Id;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
910 Unit_Names : Name_Array := No_Names;
111
kono
parents:
diff changeset
911 Unit_Symbol : Node_Id;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
912 Unit_Symbols : Symbol_Array := No_Symbols;
111
kono
parents:
diff changeset
913
kono
parents:
diff changeset
914 Errors_Count : Nat;
kono
parents:
diff changeset
915 -- Errors_Count is a count of errors detected by the compiler so far
kono
parents:
diff changeset
916 -- just before the extraction of names and symbols in the aggregate
kono
parents:
diff changeset
917 -- (Step 3).
kono
parents:
diff changeset
918 --
kono
parents:
diff changeset
919 -- At the end of the analysis, there is a check to verify that this
kono
parents:
diff changeset
920 -- count equals Serious_Errors_Detected i.e. no errors have been
kono
parents:
diff changeset
921 -- encountered during the process. Otherwise the System_Table is
kono
parents:
diff changeset
922 -- not filled.
kono
parents:
diff changeset
923
kono
parents:
diff changeset
924 -- Start of processing for Analyze_Aspect_Dimension_System
kono
parents:
diff changeset
925
kono
parents:
diff changeset
926 begin
kono
parents:
diff changeset
927 -- STEP 1: Legality of aspect
kono
parents:
diff changeset
928
kono
parents:
diff changeset
929 if not Is_Derived_Numeric_Type (N) then
kono
parents:
diff changeset
930 Error_Msg_NE
kono
parents:
diff changeset
931 ("aspect& must apply to numeric derived type declaration", N, Id);
kono
parents:
diff changeset
932 return;
kono
parents:
diff changeset
933 end if;
kono
parents:
diff changeset
934
kono
parents:
diff changeset
935 if Nkind (Aggr) /= N_Aggregate then
kono
parents:
diff changeset
936 Error_Msg_N ("aggregate expected", Aggr);
kono
parents:
diff changeset
937 return;
kono
parents:
diff changeset
938 end if;
kono
parents:
diff changeset
939
kono
parents:
diff changeset
940 -- STEP 2: Structural verification of the dimension aggregate
kono
parents:
diff changeset
941
kono
parents:
diff changeset
942 if Present (Component_Associations (Aggr)) then
kono
parents:
diff changeset
943 Error_Msg_N ("expected positional aggregate", Aggr);
kono
parents:
diff changeset
944 return;
kono
parents:
diff changeset
945 end if;
kono
parents:
diff changeset
946
kono
parents:
diff changeset
947 -- STEP 3: Name and Symbol extraction
kono
parents:
diff changeset
948
kono
parents:
diff changeset
949 Dim_Aggr := First (Expressions (Aggr));
kono
parents:
diff changeset
950 Errors_Count := Serious_Errors_Detected;
kono
parents:
diff changeset
951 while Present (Dim_Aggr) loop
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
952 if Position = High_Position_Bound then
111
kono
parents:
diff changeset
953 Error_Msg_N ("too many dimensions in system", Aggr);
kono
parents:
diff changeset
954 exit;
kono
parents:
diff changeset
955 end if;
kono
parents:
diff changeset
956
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
957 Position := Position + 1;
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
958
111
kono
parents:
diff changeset
959 if Nkind (Dim_Aggr) /= N_Aggregate then
kono
parents:
diff changeset
960 Error_Msg_N ("aggregate expected", Dim_Aggr);
kono
parents:
diff changeset
961
kono
parents:
diff changeset
962 else
kono
parents:
diff changeset
963 if Present (Component_Associations (Dim_Aggr))
kono
parents:
diff changeset
964 and then Present (Expressions (Dim_Aggr))
kono
parents:
diff changeset
965 then
kono
parents:
diff changeset
966 Error_Msg_N
kono
parents:
diff changeset
967 ("mixed positional/named aggregate not allowed here",
kono
parents:
diff changeset
968 Dim_Aggr);
kono
parents:
diff changeset
969
kono
parents:
diff changeset
970 -- Verify each dimension aggregate has three arguments
kono
parents:
diff changeset
971
kono
parents:
diff changeset
972 elsif List_Length (Component_Associations (Dim_Aggr)) /= 3
kono
parents:
diff changeset
973 and then List_Length (Expressions (Dim_Aggr)) /= 3
kono
parents:
diff changeset
974 then
kono
parents:
diff changeset
975 Error_Msg_N
kono
parents:
diff changeset
976 ("three components expected in aggregate", Dim_Aggr);
kono
parents:
diff changeset
977
kono
parents:
diff changeset
978 else
kono
parents:
diff changeset
979 -- Named dimension aggregate
kono
parents:
diff changeset
980
kono
parents:
diff changeset
981 if Present (Component_Associations (Dim_Aggr)) then
kono
parents:
diff changeset
982
kono
parents:
diff changeset
983 -- Check first argument denotes the unit name
kono
parents:
diff changeset
984
kono
parents:
diff changeset
985 Assoc := First (Component_Associations (Dim_Aggr));
kono
parents:
diff changeset
986 Choice := First (Choices (Assoc));
kono
parents:
diff changeset
987 Unit_Name := Expression (Assoc);
kono
parents:
diff changeset
988
kono
parents:
diff changeset
989 if Present (Next (Choice))
kono
parents:
diff changeset
990 or else Nkind (Choice) /= N_Identifier
kono
parents:
diff changeset
991 then
kono
parents:
diff changeset
992 Error_Msg_NE ("wrong syntax for aspect&", Choice, Id);
kono
parents:
diff changeset
993
kono
parents:
diff changeset
994 elsif Chars (Choice) /= Name_Unit_Name then
kono
parents:
diff changeset
995 Error_Msg_N ("expected Unit_Name, found&", Choice);
kono
parents:
diff changeset
996 end if;
kono
parents:
diff changeset
997
kono
parents:
diff changeset
998 -- Check the second argument denotes the unit symbol
kono
parents:
diff changeset
999
kono
parents:
diff changeset
1000 Next (Assoc);
kono
parents:
diff changeset
1001 Choice := First (Choices (Assoc));
kono
parents:
diff changeset
1002 Unit_Symbol := Expression (Assoc);
kono
parents:
diff changeset
1003
kono
parents:
diff changeset
1004 if Present (Next (Choice))
kono
parents:
diff changeset
1005 or else Nkind (Choice) /= N_Identifier
kono
parents:
diff changeset
1006 then
kono
parents:
diff changeset
1007 Error_Msg_NE ("wrong syntax for aspect&", Choice, Id);
kono
parents:
diff changeset
1008
kono
parents:
diff changeset
1009 elsif Chars (Choice) /= Name_Unit_Symbol then
kono
parents:
diff changeset
1010 Error_Msg_N ("expected Unit_Symbol, found&", Choice);
kono
parents:
diff changeset
1011 end if;
kono
parents:
diff changeset
1012
kono
parents:
diff changeset
1013 -- Check the third argument denotes the dimension symbol
kono
parents:
diff changeset
1014
kono
parents:
diff changeset
1015 Next (Assoc);
kono
parents:
diff changeset
1016 Choice := First (Choices (Assoc));
kono
parents:
diff changeset
1017 Dim_Symbol := Expression (Assoc);
kono
parents:
diff changeset
1018
kono
parents:
diff changeset
1019 if Present (Next (Choice))
kono
parents:
diff changeset
1020 or else Nkind (Choice) /= N_Identifier
kono
parents:
diff changeset
1021 then
kono
parents:
diff changeset
1022 Error_Msg_NE ("wrong syntax for aspect&", Choice, Id);
kono
parents:
diff changeset
1023 elsif Chars (Choice) /= Name_Dim_Symbol then
kono
parents:
diff changeset
1024 Error_Msg_N ("expected Dim_Symbol, found&", Choice);
kono
parents:
diff changeset
1025 end if;
kono
parents:
diff changeset
1026
kono
parents:
diff changeset
1027 -- Positional dimension aggregate
kono
parents:
diff changeset
1028
kono
parents:
diff changeset
1029 else
kono
parents:
diff changeset
1030 Unit_Name := First (Expressions (Dim_Aggr));
kono
parents:
diff changeset
1031 Unit_Symbol := Next (Unit_Name);
kono
parents:
diff changeset
1032 Dim_Symbol := Next (Unit_Symbol);
kono
parents:
diff changeset
1033 end if;
kono
parents:
diff changeset
1034
kono
parents:
diff changeset
1035 -- Check the first argument for each dimension aggregate is
kono
parents:
diff changeset
1036 -- a name.
kono
parents:
diff changeset
1037
kono
parents:
diff changeset
1038 if Nkind (Unit_Name) = N_Identifier then
kono
parents:
diff changeset
1039 Unit_Names (Position) := Chars (Unit_Name);
kono
parents:
diff changeset
1040 else
kono
parents:
diff changeset
1041 Error_Msg_N ("expected unit name", Unit_Name);
kono
parents:
diff changeset
1042 end if;
kono
parents:
diff changeset
1043
kono
parents:
diff changeset
1044 -- Check the second argument for each dimension aggregate is
kono
parents:
diff changeset
1045 -- a string or a character.
kono
parents:
diff changeset
1046
kono
parents:
diff changeset
1047 if not Nkind_In (Unit_Symbol, N_String_Literal,
kono
parents:
diff changeset
1048 N_Character_Literal)
kono
parents:
diff changeset
1049 then
kono
parents:
diff changeset
1050 Error_Msg_N
kono
parents:
diff changeset
1051 ("expected unit symbol (string or character)",
kono
parents:
diff changeset
1052 Unit_Symbol);
kono
parents:
diff changeset
1053
kono
parents:
diff changeset
1054 else
kono
parents:
diff changeset
1055 -- String case
kono
parents:
diff changeset
1056
kono
parents:
diff changeset
1057 if Nkind (Unit_Symbol) = N_String_Literal then
kono
parents:
diff changeset
1058 Unit_Symbols (Position) := Strval (Unit_Symbol);
kono
parents:
diff changeset
1059
kono
parents:
diff changeset
1060 -- Character case
kono
parents:
diff changeset
1061
kono
parents:
diff changeset
1062 else
kono
parents:
diff changeset
1063 Start_String;
kono
parents:
diff changeset
1064 Store_String_Char
kono
parents:
diff changeset
1065 (UI_To_CC (Char_Literal_Value (Unit_Symbol)));
kono
parents:
diff changeset
1066 Unit_Symbols (Position) := End_String;
kono
parents:
diff changeset
1067 end if;
kono
parents:
diff changeset
1068
kono
parents:
diff changeset
1069 -- Verify that the string is not empty
kono
parents:
diff changeset
1070
kono
parents:
diff changeset
1071 if String_Length (Unit_Symbols (Position)) = 0 then
kono
parents:
diff changeset
1072 Error_Msg_N
kono
parents:
diff changeset
1073 ("empty string not allowed here", Unit_Symbol);
kono
parents:
diff changeset
1074 end if;
kono
parents:
diff changeset
1075 end if;
kono
parents:
diff changeset
1076
kono
parents:
diff changeset
1077 -- Check the third argument for each dimension aggregate is
kono
parents:
diff changeset
1078 -- a string or a character.
kono
parents:
diff changeset
1079
kono
parents:
diff changeset
1080 if not Nkind_In (Dim_Symbol, N_String_Literal,
kono
parents:
diff changeset
1081 N_Character_Literal)
kono
parents:
diff changeset
1082 then
kono
parents:
diff changeset
1083 Error_Msg_N
kono
parents:
diff changeset
1084 ("expected dimension symbol (string or character)",
kono
parents:
diff changeset
1085 Dim_Symbol);
kono
parents:
diff changeset
1086
kono
parents:
diff changeset
1087 else
kono
parents:
diff changeset
1088 -- String case
kono
parents:
diff changeset
1089
kono
parents:
diff changeset
1090 if Nkind (Dim_Symbol) = N_String_Literal then
kono
parents:
diff changeset
1091 Dim_Symbols (Position) := Strval (Dim_Symbol);
kono
parents:
diff changeset
1092
kono
parents:
diff changeset
1093 -- Character case
kono
parents:
diff changeset
1094
kono
parents:
diff changeset
1095 else
kono
parents:
diff changeset
1096 Start_String;
kono
parents:
diff changeset
1097 Store_String_Char
kono
parents:
diff changeset
1098 (UI_To_CC (Char_Literal_Value (Dim_Symbol)));
kono
parents:
diff changeset
1099 Dim_Symbols (Position) := End_String;
kono
parents:
diff changeset
1100 end if;
kono
parents:
diff changeset
1101
kono
parents:
diff changeset
1102 -- Verify that the string is not empty
kono
parents:
diff changeset
1103
kono
parents:
diff changeset
1104 if String_Length (Dim_Symbols (Position)) = 0 then
kono
parents:
diff changeset
1105 Error_Msg_N ("empty string not allowed here", Dim_Symbol);
kono
parents:
diff changeset
1106 end if;
kono
parents:
diff changeset
1107 end if;
kono
parents:
diff changeset
1108 end if;
kono
parents:
diff changeset
1109 end if;
kono
parents:
diff changeset
1110
kono
parents:
diff changeset
1111 Next (Dim_Aggr);
kono
parents:
diff changeset
1112 end loop;
kono
parents:
diff changeset
1113
kono
parents:
diff changeset
1114 -- STEP 4: Storage of extracted values
kono
parents:
diff changeset
1115
kono
parents:
diff changeset
1116 -- Check that no errors have been detected during the analysis
kono
parents:
diff changeset
1117
kono
parents:
diff changeset
1118 if Errors_Count = Serious_Errors_Detected then
kono
parents:
diff changeset
1119 Dim_System.Type_Decl := N;
kono
parents:
diff changeset
1120 Dim_System.Unit_Names := Unit_Names;
kono
parents:
diff changeset
1121 Dim_System.Unit_Symbols := Unit_Symbols;
kono
parents:
diff changeset
1122 Dim_System.Dim_Symbols := Dim_Symbols;
kono
parents:
diff changeset
1123 Dim_System.Count := Position;
kono
parents:
diff changeset
1124 System_Table.Append (Dim_System);
kono
parents:
diff changeset
1125 end if;
kono
parents:
diff changeset
1126 end Analyze_Aspect_Dimension_System;
kono
parents:
diff changeset
1127
kono
parents:
diff changeset
1128 -----------------------
kono
parents:
diff changeset
1129 -- Analyze_Dimension --
kono
parents:
diff changeset
1130 -----------------------
kono
parents:
diff changeset
1131
kono
parents:
diff changeset
1132 -- This dispatch routine propagates dimensions for each node
kono
parents:
diff changeset
1133
kono
parents:
diff changeset
1134 procedure Analyze_Dimension (N : Node_Id) is
kono
parents:
diff changeset
1135 begin
kono
parents:
diff changeset
1136 -- Aspect is an Ada 2012 feature. Note that there is no need to check
kono
parents:
diff changeset
1137 -- dimensions for nodes that don't come from source, except for subtype
kono
parents:
diff changeset
1138 -- declarations where the dimensions are inherited from the base type,
kono
parents:
diff changeset
1139 -- for explicit dereferences generated when expanding iterators, and
kono
parents:
diff changeset
1140 -- for object declarations generated for inlining.
kono
parents:
diff changeset
1141
kono
parents:
diff changeset
1142 if Ada_Version < Ada_2012 then
kono
parents:
diff changeset
1143 return;
kono
parents:
diff changeset
1144
kono
parents:
diff changeset
1145 elsif not Comes_From_Source (N) then
kono
parents:
diff changeset
1146 if Nkind_In (N, N_Explicit_Dereference,
kono
parents:
diff changeset
1147 N_Identifier,
kono
parents:
diff changeset
1148 N_Object_Declaration,
kono
parents:
diff changeset
1149 N_Subtype_Declaration)
kono
parents:
diff changeset
1150 then
kono
parents:
diff changeset
1151 null;
kono
parents:
diff changeset
1152 else
kono
parents:
diff changeset
1153 return;
kono
parents:
diff changeset
1154 end if;
kono
parents:
diff changeset
1155 end if;
kono
parents:
diff changeset
1156
kono
parents:
diff changeset
1157 case Nkind (N) is
kono
parents:
diff changeset
1158 when N_Assignment_Statement =>
kono
parents:
diff changeset
1159 Analyze_Dimension_Assignment_Statement (N);
kono
parents:
diff changeset
1160
kono
parents:
diff changeset
1161 when N_Binary_Op =>
kono
parents:
diff changeset
1162 Analyze_Dimension_Binary_Op (N);
kono
parents:
diff changeset
1163
kono
parents:
diff changeset
1164 when N_Case_Expression =>
kono
parents:
diff changeset
1165 Analyze_Dimension_Case_Expression (N);
kono
parents:
diff changeset
1166
kono
parents:
diff changeset
1167 when N_Component_Declaration =>
kono
parents:
diff changeset
1168 Analyze_Dimension_Component_Declaration (N);
kono
parents:
diff changeset
1169
kono
parents:
diff changeset
1170 when N_Extended_Return_Statement =>
kono
parents:
diff changeset
1171 Analyze_Dimension_Extended_Return_Statement (N);
kono
parents:
diff changeset
1172
kono
parents:
diff changeset
1173 when N_Attribute_Reference
kono
parents:
diff changeset
1174 | N_Expanded_Name
kono
parents:
diff changeset
1175 | N_Explicit_Dereference
kono
parents:
diff changeset
1176 | N_Function_Call
kono
parents:
diff changeset
1177 | N_Indexed_Component
kono
parents:
diff changeset
1178 | N_Qualified_Expression
kono
parents:
diff changeset
1179 | N_Selected_Component
kono
parents:
diff changeset
1180 | N_Slice
kono
parents:
diff changeset
1181 | N_Unchecked_Type_Conversion
kono
parents:
diff changeset
1182 =>
kono
parents:
diff changeset
1183 Analyze_Dimension_Has_Etype (N);
kono
parents:
diff changeset
1184
kono
parents:
diff changeset
1185 -- In the presence of a repaired syntax error, an identifier may be
kono
parents:
diff changeset
1186 -- introduced without a usable type.
kono
parents:
diff changeset
1187
kono
parents:
diff changeset
1188 when N_Identifier =>
kono
parents:
diff changeset
1189 if Present (Etype (N)) then
kono
parents:
diff changeset
1190 Analyze_Dimension_Has_Etype (N);
kono
parents:
diff changeset
1191 end if;
kono
parents:
diff changeset
1192
kono
parents:
diff changeset
1193 when N_If_Expression =>
kono
parents:
diff changeset
1194 Analyze_Dimension_If_Expression (N);
kono
parents:
diff changeset
1195
kono
parents:
diff changeset
1196 when N_Number_Declaration =>
kono
parents:
diff changeset
1197 Analyze_Dimension_Number_Declaration (N);
kono
parents:
diff changeset
1198
kono
parents:
diff changeset
1199 when N_Object_Declaration =>
kono
parents:
diff changeset
1200 Analyze_Dimension_Object_Declaration (N);
kono
parents:
diff changeset
1201
kono
parents:
diff changeset
1202 when N_Object_Renaming_Declaration =>
kono
parents:
diff changeset
1203 Analyze_Dimension_Object_Renaming_Declaration (N);
kono
parents:
diff changeset
1204
kono
parents:
diff changeset
1205 when N_Simple_Return_Statement =>
kono
parents:
diff changeset
1206 if not Comes_From_Extended_Return_Statement (N) then
kono
parents:
diff changeset
1207 Analyze_Dimension_Simple_Return_Statement (N);
kono
parents:
diff changeset
1208 end if;
kono
parents:
diff changeset
1209
kono
parents:
diff changeset
1210 when N_Subtype_Declaration =>
kono
parents:
diff changeset
1211 Analyze_Dimension_Subtype_Declaration (N);
kono
parents:
diff changeset
1212
kono
parents:
diff changeset
1213 when N_Type_Conversion =>
kono
parents:
diff changeset
1214 Analyze_Dimension_Type_Conversion (N);
kono
parents:
diff changeset
1215
kono
parents:
diff changeset
1216 when N_Unary_Op =>
kono
parents:
diff changeset
1217 Analyze_Dimension_Unary_Op (N);
kono
parents:
diff changeset
1218
kono
parents:
diff changeset
1219 when others =>
kono
parents:
diff changeset
1220 null;
kono
parents:
diff changeset
1221 end case;
kono
parents:
diff changeset
1222 end Analyze_Dimension;
kono
parents:
diff changeset
1223
kono
parents:
diff changeset
1224 ---------------------------------------
kono
parents:
diff changeset
1225 -- Analyze_Dimension_Array_Aggregate --
kono
parents:
diff changeset
1226 ---------------------------------------
kono
parents:
diff changeset
1227
kono
parents:
diff changeset
1228 procedure Analyze_Dimension_Array_Aggregate
kono
parents:
diff changeset
1229 (N : Node_Id;
kono
parents:
diff changeset
1230 Comp_Typ : Entity_Id)
kono
parents:
diff changeset
1231 is
kono
parents:
diff changeset
1232 Comp_Ass : constant List_Id := Component_Associations (N);
kono
parents:
diff changeset
1233 Dims_Of_Comp_Typ : constant Dimension_Type := Dimensions_Of (Comp_Typ);
kono
parents:
diff changeset
1234 Exps : constant List_Id := Expressions (N);
kono
parents:
diff changeset
1235
kono
parents:
diff changeset
1236 Comp : Node_Id;
kono
parents:
diff changeset
1237 Expr : Node_Id;
kono
parents:
diff changeset
1238
kono
parents:
diff changeset
1239 Error_Detected : Boolean := False;
kono
parents:
diff changeset
1240 -- This flag is used in order to indicate if an error has been detected
kono
parents:
diff changeset
1241 -- so far by the compiler in this routine.
kono
parents:
diff changeset
1242
kono
parents:
diff changeset
1243 begin
kono
parents:
diff changeset
1244 -- Aspect is an Ada 2012 feature. Nothing to do here if the component
kono
parents:
diff changeset
1245 -- base type is not a dimensioned type.
kono
parents:
diff changeset
1246
kono
parents:
diff changeset
1247 -- Note that here the original node must come from source since the
kono
parents:
diff changeset
1248 -- original array aggregate may not have been entirely decorated.
kono
parents:
diff changeset
1249
kono
parents:
diff changeset
1250 if Ada_Version < Ada_2012
kono
parents:
diff changeset
1251 or else not Comes_From_Source (Original_Node (N))
kono
parents:
diff changeset
1252 or else not Has_Dimension_System (Base_Type (Comp_Typ))
kono
parents:
diff changeset
1253 then
kono
parents:
diff changeset
1254 return;
kono
parents:
diff changeset
1255 end if;
kono
parents:
diff changeset
1256
kono
parents:
diff changeset
1257 -- Check whether there is any positional component association
kono
parents:
diff changeset
1258
kono
parents:
diff changeset
1259 if Is_Empty_List (Exps) then
kono
parents:
diff changeset
1260 Comp := First (Comp_Ass);
kono
parents:
diff changeset
1261 else
kono
parents:
diff changeset
1262 Comp := First (Exps);
kono
parents:
diff changeset
1263 end if;
kono
parents:
diff changeset
1264
kono
parents:
diff changeset
1265 while Present (Comp) loop
kono
parents:
diff changeset
1266
kono
parents:
diff changeset
1267 -- Get the expression from the component
kono
parents:
diff changeset
1268
kono
parents:
diff changeset
1269 if Nkind (Comp) = N_Component_Association then
kono
parents:
diff changeset
1270 Expr := Expression (Comp);
kono
parents:
diff changeset
1271 else
kono
parents:
diff changeset
1272 Expr := Comp;
kono
parents:
diff changeset
1273 end if;
kono
parents:
diff changeset
1274
kono
parents:
diff changeset
1275 -- Issue an error if the dimensions of the component type and the
kono
parents:
diff changeset
1276 -- dimensions of the component mismatch.
kono
parents:
diff changeset
1277
kono
parents:
diff changeset
1278 -- Note that we must ensure the expression has been fully analyzed
kono
parents:
diff changeset
1279 -- since it may not be decorated at this point. We also don't want to
kono
parents:
diff changeset
1280 -- issue the same error message multiple times on the same expression
kono
parents:
diff changeset
1281 -- (may happen when an aggregate is converted into a positional
kono
parents:
diff changeset
1282 -- aggregate). We also must verify that this is a scalar component,
kono
parents:
diff changeset
1283 -- and not a subaggregate of a multidimensional aggregate.
kono
parents:
diff changeset
1284
kono
parents:
diff changeset
1285 if Comes_From_Source (Original_Node (Expr))
kono
parents:
diff changeset
1286 and then Present (Etype (Expr))
kono
parents:
diff changeset
1287 and then Is_Numeric_Type (Etype (Expr))
kono
parents:
diff changeset
1288 and then Dimensions_Of (Expr) /= Dims_Of_Comp_Typ
kono
parents:
diff changeset
1289 and then Sloc (Comp) /= Sloc (Prev (Comp))
kono
parents:
diff changeset
1290 then
kono
parents:
diff changeset
1291 -- Check if an error has already been encountered so far
kono
parents:
diff changeset
1292
kono
parents:
diff changeset
1293 if not Error_Detected then
kono
parents:
diff changeset
1294 Error_Msg_N ("dimensions mismatch in array aggregate", N);
kono
parents:
diff changeset
1295 Error_Detected := True;
kono
parents:
diff changeset
1296 end if;
kono
parents:
diff changeset
1297
kono
parents:
diff changeset
1298 Error_Msg_N
kono
parents:
diff changeset
1299 ("\expected dimension " & Dimensions_Msg_Of (Comp_Typ)
kono
parents:
diff changeset
1300 & ", found " & Dimensions_Msg_Of (Expr), Expr);
kono
parents:
diff changeset
1301 end if;
kono
parents:
diff changeset
1302
kono
parents:
diff changeset
1303 -- Look at the named components right after the positional components
kono
parents:
diff changeset
1304
kono
parents:
diff changeset
1305 if not Present (Next (Comp))
kono
parents:
diff changeset
1306 and then List_Containing (Comp) = Exps
kono
parents:
diff changeset
1307 then
kono
parents:
diff changeset
1308 Comp := First (Comp_Ass);
kono
parents:
diff changeset
1309 else
kono
parents:
diff changeset
1310 Next (Comp);
kono
parents:
diff changeset
1311 end if;
kono
parents:
diff changeset
1312 end loop;
kono
parents:
diff changeset
1313 end Analyze_Dimension_Array_Aggregate;
kono
parents:
diff changeset
1314
kono
parents:
diff changeset
1315 --------------------------------------------
kono
parents:
diff changeset
1316 -- Analyze_Dimension_Assignment_Statement --
kono
parents:
diff changeset
1317 --------------------------------------------
kono
parents:
diff changeset
1318
kono
parents:
diff changeset
1319 procedure Analyze_Dimension_Assignment_Statement (N : Node_Id) is
kono
parents:
diff changeset
1320 Lhs : constant Node_Id := Name (N);
kono
parents:
diff changeset
1321 Dims_Of_Lhs : constant Dimension_Type := Dimensions_Of (Lhs);
kono
parents:
diff changeset
1322 Rhs : constant Node_Id := Expression (N);
kono
parents:
diff changeset
1323 Dims_Of_Rhs : constant Dimension_Type := Dimensions_Of (Rhs);
kono
parents:
diff changeset
1324
kono
parents:
diff changeset
1325 procedure Error_Dim_Msg_For_Assignment_Statement
kono
parents:
diff changeset
1326 (N : Node_Id;
kono
parents:
diff changeset
1327 Lhs : Node_Id;
kono
parents:
diff changeset
1328 Rhs : Node_Id);
kono
parents:
diff changeset
1329 -- Error using Error_Msg_N at node N. Output the dimensions of left
kono
parents:
diff changeset
1330 -- and right hand sides.
kono
parents:
diff changeset
1331
kono
parents:
diff changeset
1332 --------------------------------------------
kono
parents:
diff changeset
1333 -- Error_Dim_Msg_For_Assignment_Statement --
kono
parents:
diff changeset
1334 --------------------------------------------
kono
parents:
diff changeset
1335
kono
parents:
diff changeset
1336 procedure Error_Dim_Msg_For_Assignment_Statement
kono
parents:
diff changeset
1337 (N : Node_Id;
kono
parents:
diff changeset
1338 Lhs : Node_Id;
kono
parents:
diff changeset
1339 Rhs : Node_Id)
kono
parents:
diff changeset
1340 is
kono
parents:
diff changeset
1341 begin
kono
parents:
diff changeset
1342 Error_Msg_N ("dimensions mismatch in assignment", N);
kono
parents:
diff changeset
1343 Error_Msg_N ("\left-hand side " & Dimensions_Msg_Of (Lhs, True), N);
kono
parents:
diff changeset
1344 Error_Msg_N ("\right-hand side " & Dimensions_Msg_Of (Rhs, True), N);
kono
parents:
diff changeset
1345 end Error_Dim_Msg_For_Assignment_Statement;
kono
parents:
diff changeset
1346
kono
parents:
diff changeset
1347 -- Start of processing for Analyze_Dimension_Assignment
kono
parents:
diff changeset
1348
kono
parents:
diff changeset
1349 begin
kono
parents:
diff changeset
1350 if Dims_Of_Lhs /= Dims_Of_Rhs then
kono
parents:
diff changeset
1351 Error_Dim_Msg_For_Assignment_Statement (N, Lhs, Rhs);
kono
parents:
diff changeset
1352 end if;
kono
parents:
diff changeset
1353 end Analyze_Dimension_Assignment_Statement;
kono
parents:
diff changeset
1354
kono
parents:
diff changeset
1355 ---------------------------------
kono
parents:
diff changeset
1356 -- Analyze_Dimension_Binary_Op --
kono
parents:
diff changeset
1357 ---------------------------------
kono
parents:
diff changeset
1358
kono
parents:
diff changeset
1359 -- Check and propagate the dimensions for binary operators
kono
parents:
diff changeset
1360 -- Note that when the dimensions mismatch, no dimension is propagated to N.
kono
parents:
diff changeset
1361
kono
parents:
diff changeset
1362 procedure Analyze_Dimension_Binary_Op (N : Node_Id) is
kono
parents:
diff changeset
1363 N_Kind : constant Node_Kind := Nkind (N);
kono
parents:
diff changeset
1364
kono
parents:
diff changeset
1365 function Dimensions_Of_Operand (N : Node_Id) return Dimension_Type;
kono
parents:
diff changeset
1366 -- If the operand is a numeric literal that comes from a declared
kono
parents:
diff changeset
1367 -- constant, use the dimensions of the constant which were computed
kono
parents:
diff changeset
1368 -- from the expression of the constant declaration. Otherwise the
kono
parents:
diff changeset
1369 -- dimensions are those of the operand, or the type of the operand.
kono
parents:
diff changeset
1370 -- This takes care of node rewritings from validity checks, where the
kono
parents:
diff changeset
1371 -- dimensions of the operand itself may not be preserved, while the
kono
parents:
diff changeset
1372 -- type comes from context and must have dimension information.
kono
parents:
diff changeset
1373
kono
parents:
diff changeset
1374 procedure Error_Dim_Msg_For_Binary_Op (N, L, R : Node_Id);
kono
parents:
diff changeset
1375 -- Error using Error_Msg_NE and Error_Msg_N at node N. Output the
kono
parents:
diff changeset
1376 -- dimensions of both operands.
kono
parents:
diff changeset
1377
kono
parents:
diff changeset
1378 ---------------------------
kono
parents:
diff changeset
1379 -- Dimensions_Of_Operand --
kono
parents:
diff changeset
1380 ---------------------------
kono
parents:
diff changeset
1381
kono
parents:
diff changeset
1382 function Dimensions_Of_Operand (N : Node_Id) return Dimension_Type is
kono
parents:
diff changeset
1383 Dims : constant Dimension_Type := Dimensions_Of (N);
kono
parents:
diff changeset
1384
kono
parents:
diff changeset
1385 begin
kono
parents:
diff changeset
1386 if Exists (Dims) then
kono
parents:
diff changeset
1387 return Dims;
kono
parents:
diff changeset
1388
kono
parents:
diff changeset
1389 elsif Is_Entity_Name (N) then
kono
parents:
diff changeset
1390 return Dimensions_Of (Etype (Entity (N)));
kono
parents:
diff changeset
1391
kono
parents:
diff changeset
1392 elsif Nkind (N) = N_Real_Literal then
kono
parents:
diff changeset
1393
kono
parents:
diff changeset
1394 if Present (Original_Entity (N)) then
kono
parents:
diff changeset
1395 return Dimensions_Of (Original_Entity (N));
kono
parents:
diff changeset
1396
kono
parents:
diff changeset
1397 else
kono
parents:
diff changeset
1398 return Dimensions_Of (Etype (N));
kono
parents:
diff changeset
1399 end if;
kono
parents:
diff changeset
1400
kono
parents:
diff changeset
1401 -- Otherwise return the default dimensions
kono
parents:
diff changeset
1402
kono
parents:
diff changeset
1403 else
kono
parents:
diff changeset
1404 return Dims;
kono
parents:
diff changeset
1405 end if;
kono
parents:
diff changeset
1406 end Dimensions_Of_Operand;
kono
parents:
diff changeset
1407
kono
parents:
diff changeset
1408 ---------------------------------
kono
parents:
diff changeset
1409 -- Error_Dim_Msg_For_Binary_Op --
kono
parents:
diff changeset
1410 ---------------------------------
kono
parents:
diff changeset
1411
kono
parents:
diff changeset
1412 procedure Error_Dim_Msg_For_Binary_Op (N, L, R : Node_Id) is
kono
parents:
diff changeset
1413 begin
kono
parents:
diff changeset
1414 Error_Msg_NE
kono
parents:
diff changeset
1415 ("both operands for operation& must have same dimensions",
kono
parents:
diff changeset
1416 N, Entity (N));
kono
parents:
diff changeset
1417 Error_Msg_N ("\left operand " & Dimensions_Msg_Of (L, True), N);
kono
parents:
diff changeset
1418 Error_Msg_N ("\right operand " & Dimensions_Msg_Of (R, True), N);
kono
parents:
diff changeset
1419 end Error_Dim_Msg_For_Binary_Op;
kono
parents:
diff changeset
1420
kono
parents:
diff changeset
1421 -- Start of processing for Analyze_Dimension_Binary_Op
kono
parents:
diff changeset
1422
kono
parents:
diff changeset
1423 begin
kono
parents:
diff changeset
1424 -- If the node is already analyzed, do not examine the operands. At the
kono
parents:
diff changeset
1425 -- end of the analysis their dimensions have been removed, and the node
kono
parents:
diff changeset
1426 -- itself may have been rewritten.
kono
parents:
diff changeset
1427
kono
parents:
diff changeset
1428 if Analyzed (N) then
kono
parents:
diff changeset
1429 return;
kono
parents:
diff changeset
1430 end if;
kono
parents:
diff changeset
1431
kono
parents:
diff changeset
1432 if Nkind_In (N_Kind, N_Op_Add, N_Op_Expon, N_Op_Subtract)
kono
parents:
diff changeset
1433 or else N_Kind in N_Multiplying_Operator
kono
parents:
diff changeset
1434 or else N_Kind in N_Op_Compare
kono
parents:
diff changeset
1435 then
kono
parents:
diff changeset
1436 declare
kono
parents:
diff changeset
1437 L : constant Node_Id := Left_Opnd (N);
kono
parents:
diff changeset
1438 Dims_Of_L : constant Dimension_Type :=
kono
parents:
diff changeset
1439 Dimensions_Of_Operand (L);
kono
parents:
diff changeset
1440 L_Has_Dimensions : constant Boolean := Exists (Dims_Of_L);
kono
parents:
diff changeset
1441 R : constant Node_Id := Right_Opnd (N);
kono
parents:
diff changeset
1442 Dims_Of_R : constant Dimension_Type :=
kono
parents:
diff changeset
1443 Dimensions_Of_Operand (R);
kono
parents:
diff changeset
1444 R_Has_Dimensions : constant Boolean := Exists (Dims_Of_R);
kono
parents:
diff changeset
1445 Dims_Of_N : Dimension_Type := Null_Dimension;
kono
parents:
diff changeset
1446
kono
parents:
diff changeset
1447 begin
kono
parents:
diff changeset
1448 -- N_Op_Add, N_Op_Mod, N_Op_Rem or N_Op_Subtract case
kono
parents:
diff changeset
1449
kono
parents:
diff changeset
1450 if Nkind_In (N, N_Op_Add, N_Op_Mod, N_Op_Rem, N_Op_Subtract) then
kono
parents:
diff changeset
1451
kono
parents:
diff changeset
1452 -- Check both operands have same dimension
kono
parents:
diff changeset
1453
kono
parents:
diff changeset
1454 if Dims_Of_L /= Dims_Of_R then
kono
parents:
diff changeset
1455 Error_Dim_Msg_For_Binary_Op (N, L, R);
kono
parents:
diff changeset
1456 else
kono
parents:
diff changeset
1457 -- Check both operands are not dimensionless
kono
parents:
diff changeset
1458
kono
parents:
diff changeset
1459 if Exists (Dims_Of_L) then
kono
parents:
diff changeset
1460 Set_Dimensions (N, Dims_Of_L);
kono
parents:
diff changeset
1461 end if;
kono
parents:
diff changeset
1462 end if;
kono
parents:
diff changeset
1463
kono
parents:
diff changeset
1464 -- N_Op_Multiply or N_Op_Divide case
kono
parents:
diff changeset
1465
kono
parents:
diff changeset
1466 elsif Nkind_In (N_Kind, N_Op_Multiply, N_Op_Divide) then
kono
parents:
diff changeset
1467
kono
parents:
diff changeset
1468 -- Check at least one operand is not dimensionless
kono
parents:
diff changeset
1469
kono
parents:
diff changeset
1470 if L_Has_Dimensions or R_Has_Dimensions then
kono
parents:
diff changeset
1471
kono
parents:
diff changeset
1472 -- Multiplication case
kono
parents:
diff changeset
1473
kono
parents:
diff changeset
1474 -- Get both operands dimensions and add them
kono
parents:
diff changeset
1475
kono
parents:
diff changeset
1476 if N_Kind = N_Op_Multiply then
kono
parents:
diff changeset
1477 for Position in Dimension_Type'Range loop
kono
parents:
diff changeset
1478 Dims_Of_N (Position) :=
kono
parents:
diff changeset
1479 Dims_Of_L (Position) + Dims_Of_R (Position);
kono
parents:
diff changeset
1480 end loop;
kono
parents:
diff changeset
1481
kono
parents:
diff changeset
1482 -- Division case
kono
parents:
diff changeset
1483
kono
parents:
diff changeset
1484 -- Get both operands dimensions and subtract them
kono
parents:
diff changeset
1485
kono
parents:
diff changeset
1486 else
kono
parents:
diff changeset
1487 for Position in Dimension_Type'Range loop
kono
parents:
diff changeset
1488 Dims_Of_N (Position) :=
kono
parents:
diff changeset
1489 Dims_Of_L (Position) - Dims_Of_R (Position);
kono
parents:
diff changeset
1490 end loop;
kono
parents:
diff changeset
1491 end if;
kono
parents:
diff changeset
1492
kono
parents:
diff changeset
1493 if Exists (Dims_Of_N) then
kono
parents:
diff changeset
1494 Set_Dimensions (N, Dims_Of_N);
kono
parents:
diff changeset
1495 end if;
kono
parents:
diff changeset
1496 end if;
kono
parents:
diff changeset
1497
kono
parents:
diff changeset
1498 -- Exponentiation case
kono
parents:
diff changeset
1499
kono
parents:
diff changeset
1500 -- Note: a rational exponent is allowed for dimensioned operand
kono
parents:
diff changeset
1501
kono
parents:
diff changeset
1502 elsif N_Kind = N_Op_Expon then
kono
parents:
diff changeset
1503
kono
parents:
diff changeset
1504 -- Check the left operand is not dimensionless. Note that the
kono
parents:
diff changeset
1505 -- value of the exponent must be known compile time. Otherwise,
kono
parents:
diff changeset
1506 -- the exponentiation evaluation will return an error message.
kono
parents:
diff changeset
1507
kono
parents:
diff changeset
1508 if L_Has_Dimensions then
kono
parents:
diff changeset
1509 if not Compile_Time_Known_Value (R) then
kono
parents:
diff changeset
1510 Error_Msg_N
kono
parents:
diff changeset
1511 ("exponent of dimensioned operand must be "
kono
parents:
diff changeset
1512 & "known at compile time", N);
kono
parents:
diff changeset
1513 end if;
kono
parents:
diff changeset
1514
kono
parents:
diff changeset
1515 declare
kono
parents:
diff changeset
1516 Exponent_Value : Rational := Zero;
kono
parents:
diff changeset
1517
kono
parents:
diff changeset
1518 begin
kono
parents:
diff changeset
1519 -- Real operand case
kono
parents:
diff changeset
1520
kono
parents:
diff changeset
1521 if Is_Real_Type (Etype (L)) then
kono
parents:
diff changeset
1522
kono
parents:
diff changeset
1523 -- Define the exponent as a Rational number
kono
parents:
diff changeset
1524
kono
parents:
diff changeset
1525 Exponent_Value := Create_Rational_From (R, False);
kono
parents:
diff changeset
1526
kono
parents:
diff changeset
1527 -- Verify that the exponent cannot be interpreted
kono
parents:
diff changeset
1528 -- as a rational, otherwise interpret the exponent
kono
parents:
diff changeset
1529 -- as an integer.
kono
parents:
diff changeset
1530
kono
parents:
diff changeset
1531 if Exponent_Value = No_Rational then
kono
parents:
diff changeset
1532 Exponent_Value :=
kono
parents:
diff changeset
1533 +Whole (UI_To_Int (Expr_Value (R)));
kono
parents:
diff changeset
1534 end if;
kono
parents:
diff changeset
1535
kono
parents:
diff changeset
1536 -- Integer operand case.
kono
parents:
diff changeset
1537
kono
parents:
diff changeset
1538 -- For integer operand, the exponent cannot be
kono
parents:
diff changeset
1539 -- interpreted as a rational.
kono
parents:
diff changeset
1540
kono
parents:
diff changeset
1541 else
kono
parents:
diff changeset
1542 Exponent_Value := +Whole (UI_To_Int (Expr_Value (R)));
kono
parents:
diff changeset
1543 end if;
kono
parents:
diff changeset
1544
kono
parents:
diff changeset
1545 for Position in Dimension_Type'Range loop
kono
parents:
diff changeset
1546 Dims_Of_N (Position) :=
kono
parents:
diff changeset
1547 Dims_Of_L (Position) * Exponent_Value;
kono
parents:
diff changeset
1548 end loop;
kono
parents:
diff changeset
1549
kono
parents:
diff changeset
1550 if Exists (Dims_Of_N) then
kono
parents:
diff changeset
1551 Set_Dimensions (N, Dims_Of_N);
kono
parents:
diff changeset
1552 end if;
kono
parents:
diff changeset
1553 end;
kono
parents:
diff changeset
1554 end if;
kono
parents:
diff changeset
1555
kono
parents:
diff changeset
1556 -- Comparison cases
kono
parents:
diff changeset
1557
kono
parents:
diff changeset
1558 -- For relational operations, only dimension checking is
kono
parents:
diff changeset
1559 -- performed (no propagation). If one operand is the result
kono
parents:
diff changeset
1560 -- of constant folding the dimensions may have been lost
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1561 -- in a tree copy, so assume that preanalysis has verified
111
kono
parents:
diff changeset
1562 -- that dimensions are correct.
kono
parents:
diff changeset
1563
kono
parents:
diff changeset
1564 elsif N_Kind in N_Op_Compare then
kono
parents:
diff changeset
1565 if (L_Has_Dimensions or R_Has_Dimensions)
kono
parents:
diff changeset
1566 and then Dims_Of_L /= Dims_Of_R
kono
parents:
diff changeset
1567 then
kono
parents:
diff changeset
1568 if Nkind (L) = N_Real_Literal
kono
parents:
diff changeset
1569 and then not (Comes_From_Source (L))
kono
parents:
diff changeset
1570 and then Expander_Active
kono
parents:
diff changeset
1571 then
kono
parents:
diff changeset
1572 null;
kono
parents:
diff changeset
1573
kono
parents:
diff changeset
1574 elsif Nkind (R) = N_Real_Literal
kono
parents:
diff changeset
1575 and then not (Comes_From_Source (R))
kono
parents:
diff changeset
1576 and then Expander_Active
kono
parents:
diff changeset
1577 then
kono
parents:
diff changeset
1578 null;
kono
parents:
diff changeset
1579
kono
parents:
diff changeset
1580 -- Numeric literal case. Issue a warning to indicate the
kono
parents:
diff changeset
1581 -- literal is treated as if its dimension matches the type
kono
parents:
diff changeset
1582 -- dimension.
kono
parents:
diff changeset
1583
kono
parents:
diff changeset
1584 elsif Nkind_In (Original_Node (L), N_Integer_Literal,
kono
parents:
diff changeset
1585 N_Real_Literal)
kono
parents:
diff changeset
1586 then
kono
parents:
diff changeset
1587 Dim_Warning_For_Numeric_Literal (L, Etype (R));
kono
parents:
diff changeset
1588
kono
parents:
diff changeset
1589 elsif Nkind_In (Original_Node (R), N_Integer_Literal,
kono
parents:
diff changeset
1590 N_Real_Literal)
kono
parents:
diff changeset
1591 then
kono
parents:
diff changeset
1592 Dim_Warning_For_Numeric_Literal (R, Etype (L));
kono
parents:
diff changeset
1593
kono
parents:
diff changeset
1594 else
kono
parents:
diff changeset
1595 Error_Dim_Msg_For_Binary_Op (N, L, R);
kono
parents:
diff changeset
1596 end if;
kono
parents:
diff changeset
1597 end if;
kono
parents:
diff changeset
1598 end if;
kono
parents:
diff changeset
1599
kono
parents:
diff changeset
1600 -- If expander is active, remove dimension information from each
kono
parents:
diff changeset
1601 -- operand, as only dimensions of result are relevant.
kono
parents:
diff changeset
1602
kono
parents:
diff changeset
1603 if Expander_Active then
kono
parents:
diff changeset
1604 Remove_Dimensions (L);
kono
parents:
diff changeset
1605 Remove_Dimensions (R);
kono
parents:
diff changeset
1606 end if;
kono
parents:
diff changeset
1607 end;
kono
parents:
diff changeset
1608 end if;
kono
parents:
diff changeset
1609 end Analyze_Dimension_Binary_Op;
kono
parents:
diff changeset
1610
kono
parents:
diff changeset
1611 ----------------------------
kono
parents:
diff changeset
1612 -- Analyze_Dimension_Call --
kono
parents:
diff changeset
1613 ----------------------------
kono
parents:
diff changeset
1614
kono
parents:
diff changeset
1615 procedure Analyze_Dimension_Call (N : Node_Id; Nam : Entity_Id) is
kono
parents:
diff changeset
1616 Actuals : constant List_Id := Parameter_Associations (N);
kono
parents:
diff changeset
1617 Actual : Node_Id;
kono
parents:
diff changeset
1618 Dims_Of_Formal : Dimension_Type;
kono
parents:
diff changeset
1619 Formal : Node_Id;
kono
parents:
diff changeset
1620 Formal_Typ : Entity_Id;
kono
parents:
diff changeset
1621
kono
parents:
diff changeset
1622 Error_Detected : Boolean := False;
kono
parents:
diff changeset
1623 -- This flag is used in order to indicate if an error has been detected
kono
parents:
diff changeset
1624 -- so far by the compiler in this routine.
kono
parents:
diff changeset
1625
kono
parents:
diff changeset
1626 begin
kono
parents:
diff changeset
1627 -- Aspect is an Ada 2012 feature. Note that there is no need to check
kono
parents:
diff changeset
1628 -- dimensions for calls that don't come from source, or those that may
kono
parents:
diff changeset
1629 -- have semantic errors.
kono
parents:
diff changeset
1630
kono
parents:
diff changeset
1631 if Ada_Version < Ada_2012
kono
parents:
diff changeset
1632 or else not Comes_From_Source (N)
kono
parents:
diff changeset
1633 or else Error_Posted (N)
kono
parents:
diff changeset
1634 then
kono
parents:
diff changeset
1635 return;
kono
parents:
diff changeset
1636 end if;
kono
parents:
diff changeset
1637
kono
parents:
diff changeset
1638 -- Check the dimensions of the actuals, if any
kono
parents:
diff changeset
1639
kono
parents:
diff changeset
1640 if not Is_Empty_List (Actuals) then
kono
parents:
diff changeset
1641
kono
parents:
diff changeset
1642 -- Special processing for elementary functions
kono
parents:
diff changeset
1643
kono
parents:
diff changeset
1644 -- For Sqrt call, the resulting dimensions equal to half the
kono
parents:
diff changeset
1645 -- dimensions of the actual. For all other elementary calls, this
kono
parents:
diff changeset
1646 -- routine check that every actual is dimensionless.
kono
parents:
diff changeset
1647
kono
parents:
diff changeset
1648 if Nkind (N) = N_Function_Call then
kono
parents:
diff changeset
1649 Elementary_Function_Calls : declare
kono
parents:
diff changeset
1650 Dims_Of_Call : Dimension_Type;
kono
parents:
diff changeset
1651 Ent : Entity_Id := Nam;
kono
parents:
diff changeset
1652
kono
parents:
diff changeset
1653 function Is_Elementary_Function_Entity
kono
parents:
diff changeset
1654 (Sub_Id : Entity_Id) return Boolean;
kono
parents:
diff changeset
1655 -- Given Sub_Id, the original subprogram entity, return True
kono
parents:
diff changeset
1656 -- if call is to an elementary function (see Ada.Numerics.
kono
parents:
diff changeset
1657 -- Generic_Elementary_Functions).
kono
parents:
diff changeset
1658
kono
parents:
diff changeset
1659 -----------------------------------
kono
parents:
diff changeset
1660 -- Is_Elementary_Function_Entity --
kono
parents:
diff changeset
1661 -----------------------------------
kono
parents:
diff changeset
1662
kono
parents:
diff changeset
1663 function Is_Elementary_Function_Entity
kono
parents:
diff changeset
1664 (Sub_Id : Entity_Id) return Boolean
kono
parents:
diff changeset
1665 is
kono
parents:
diff changeset
1666 Loc : constant Source_Ptr := Sloc (Sub_Id);
kono
parents:
diff changeset
1667
kono
parents:
diff changeset
1668 begin
kono
parents:
diff changeset
1669 -- Is entity in Ada.Numerics.Generic_Elementary_Functions?
kono
parents:
diff changeset
1670
kono
parents:
diff changeset
1671 return
kono
parents:
diff changeset
1672 Loc > No_Location
kono
parents:
diff changeset
1673 and then
kono
parents:
diff changeset
1674 Is_RTU
kono
parents:
diff changeset
1675 (Cunit_Entity (Get_Source_Unit (Loc)),
kono
parents:
diff changeset
1676 Ada_Numerics_Generic_Elementary_Functions);
kono
parents:
diff changeset
1677 end Is_Elementary_Function_Entity;
kono
parents:
diff changeset
1678
kono
parents:
diff changeset
1679 -- Start of processing for Elementary_Function_Calls
kono
parents:
diff changeset
1680
kono
parents:
diff changeset
1681 begin
kono
parents:
diff changeset
1682 -- Get original subprogram entity following the renaming chain
kono
parents:
diff changeset
1683
kono
parents:
diff changeset
1684 if Present (Alias (Ent)) then
kono
parents:
diff changeset
1685 Ent := Alias (Ent);
kono
parents:
diff changeset
1686 end if;
kono
parents:
diff changeset
1687
kono
parents:
diff changeset
1688 -- Check the call is an Elementary function call
kono
parents:
diff changeset
1689
kono
parents:
diff changeset
1690 if Is_Elementary_Function_Entity (Ent) then
kono
parents:
diff changeset
1691
kono
parents:
diff changeset
1692 -- Sqrt function call case
kono
parents:
diff changeset
1693
kono
parents:
diff changeset
1694 if Chars (Ent) = Name_Sqrt then
kono
parents:
diff changeset
1695 Dims_Of_Call := Dimensions_Of (First_Actual (N));
kono
parents:
diff changeset
1696
kono
parents:
diff changeset
1697 -- Evaluates the resulting dimensions (i.e. half the
kono
parents:
diff changeset
1698 -- dimensions of the actual).
kono
parents:
diff changeset
1699
kono
parents:
diff changeset
1700 if Exists (Dims_Of_Call) then
kono
parents:
diff changeset
1701 for Position in Dims_Of_Call'Range loop
kono
parents:
diff changeset
1702 Dims_Of_Call (Position) :=
kono
parents:
diff changeset
1703 Dims_Of_Call (Position) *
kono
parents:
diff changeset
1704 Rational'(Numerator => 1, Denominator => 2);
kono
parents:
diff changeset
1705 end loop;
kono
parents:
diff changeset
1706
kono
parents:
diff changeset
1707 Set_Dimensions (N, Dims_Of_Call);
kono
parents:
diff changeset
1708 end if;
kono
parents:
diff changeset
1709
kono
parents:
diff changeset
1710 -- All other elementary functions case. Note that every
kono
parents:
diff changeset
1711 -- actual here should be dimensionless.
kono
parents:
diff changeset
1712
kono
parents:
diff changeset
1713 else
kono
parents:
diff changeset
1714 Actual := First_Actual (N);
kono
parents:
diff changeset
1715 while Present (Actual) loop
kono
parents:
diff changeset
1716 if Exists (Dimensions_Of (Actual)) then
kono
parents:
diff changeset
1717
kono
parents:
diff changeset
1718 -- Check if error has already been encountered
kono
parents:
diff changeset
1719
kono
parents:
diff changeset
1720 if not Error_Detected then
kono
parents:
diff changeset
1721 Error_Msg_NE
kono
parents:
diff changeset
1722 ("dimensions mismatch in call of&",
kono
parents:
diff changeset
1723 N, Name (N));
kono
parents:
diff changeset
1724 Error_Detected := True;
kono
parents:
diff changeset
1725 end if;
kono
parents:
diff changeset
1726
kono
parents:
diff changeset
1727 Error_Msg_N
kono
parents:
diff changeset
1728 ("\expected dimension '['], found "
kono
parents:
diff changeset
1729 & Dimensions_Msg_Of (Actual), Actual);
kono
parents:
diff changeset
1730 end if;
kono
parents:
diff changeset
1731
kono
parents:
diff changeset
1732 Next_Actual (Actual);
kono
parents:
diff changeset
1733 end loop;
kono
parents:
diff changeset
1734 end if;
kono
parents:
diff changeset
1735
kono
parents:
diff changeset
1736 -- Nothing more to do for elementary functions
kono
parents:
diff changeset
1737
kono
parents:
diff changeset
1738 return;
kono
parents:
diff changeset
1739 end if;
kono
parents:
diff changeset
1740 end Elementary_Function_Calls;
kono
parents:
diff changeset
1741 end if;
kono
parents:
diff changeset
1742
kono
parents:
diff changeset
1743 -- General case. Check, for each parameter, the dimensions of the
kono
parents:
diff changeset
1744 -- actual and its corresponding formal match. Otherwise, complain.
kono
parents:
diff changeset
1745
kono
parents:
diff changeset
1746 Actual := First_Actual (N);
kono
parents:
diff changeset
1747 Formal := First_Formal (Nam);
kono
parents:
diff changeset
1748 while Present (Formal) loop
kono
parents:
diff changeset
1749
kono
parents:
diff changeset
1750 -- A missing corresponding actual indicates that the analysis of
kono
parents:
diff changeset
1751 -- the call was aborted due to a previous error.
kono
parents:
diff changeset
1752
kono
parents:
diff changeset
1753 if No (Actual) then
kono
parents:
diff changeset
1754 Check_Error_Detected;
kono
parents:
diff changeset
1755 return;
kono
parents:
diff changeset
1756 end if;
kono
parents:
diff changeset
1757
kono
parents:
diff changeset
1758 Formal_Typ := Etype (Formal);
kono
parents:
diff changeset
1759 Dims_Of_Formal := Dimensions_Of (Formal_Typ);
kono
parents:
diff changeset
1760
kono
parents:
diff changeset
1761 -- If the formal is not dimensionless, check dimensions of formal
kono
parents:
diff changeset
1762 -- and actual match. Otherwise, complain.
kono
parents:
diff changeset
1763
kono
parents:
diff changeset
1764 if Exists (Dims_Of_Formal)
kono
parents:
diff changeset
1765 and then Dimensions_Of (Actual) /= Dims_Of_Formal
kono
parents:
diff changeset
1766 then
kono
parents:
diff changeset
1767 -- Check if an error has already been encountered so far
kono
parents:
diff changeset
1768
kono
parents:
diff changeset
1769 if not Error_Detected then
kono
parents:
diff changeset
1770 Error_Msg_NE ("dimensions mismatch in& call", N, Name (N));
kono
parents:
diff changeset
1771 Error_Detected := True;
kono
parents:
diff changeset
1772 end if;
kono
parents:
diff changeset
1773
kono
parents:
diff changeset
1774 Error_Msg_N
kono
parents:
diff changeset
1775 ("\expected dimension " & Dimensions_Msg_Of (Formal_Typ)
kono
parents:
diff changeset
1776 & ", found " & Dimensions_Msg_Of (Actual), Actual);
kono
parents:
diff changeset
1777 end if;
kono
parents:
diff changeset
1778
kono
parents:
diff changeset
1779 Next_Actual (Actual);
kono
parents:
diff changeset
1780 Next_Formal (Formal);
kono
parents:
diff changeset
1781 end loop;
kono
parents:
diff changeset
1782 end if;
kono
parents:
diff changeset
1783
kono
parents:
diff changeset
1784 -- For function calls, propagate the dimensions from the returned type
kono
parents:
diff changeset
1785
kono
parents:
diff changeset
1786 if Nkind (N) = N_Function_Call then
kono
parents:
diff changeset
1787 Analyze_Dimension_Has_Etype (N);
kono
parents:
diff changeset
1788 end if;
kono
parents:
diff changeset
1789 end Analyze_Dimension_Call;
kono
parents:
diff changeset
1790
kono
parents:
diff changeset
1791 ---------------------------------------
kono
parents:
diff changeset
1792 -- Analyze_Dimension_Case_Expression --
kono
parents:
diff changeset
1793 ---------------------------------------
kono
parents:
diff changeset
1794
kono
parents:
diff changeset
1795 procedure Analyze_Dimension_Case_Expression (N : Node_Id) is
kono
parents:
diff changeset
1796 Frst : constant Node_Id := First (Alternatives (N));
kono
parents:
diff changeset
1797 Frst_Expr : constant Node_Id := Expression (Frst);
kono
parents:
diff changeset
1798 Dims : constant Dimension_Type := Dimensions_Of (Frst_Expr);
kono
parents:
diff changeset
1799
kono
parents:
diff changeset
1800 Alt : Node_Id;
kono
parents:
diff changeset
1801
kono
parents:
diff changeset
1802 begin
kono
parents:
diff changeset
1803 Alt := Next (Frst);
kono
parents:
diff changeset
1804 while Present (Alt) loop
kono
parents:
diff changeset
1805 if Dimensions_Of (Expression (Alt)) /= Dims then
kono
parents:
diff changeset
1806 Error_Msg_N ("dimension mismatch in case expression", Alt);
kono
parents:
diff changeset
1807 exit;
kono
parents:
diff changeset
1808 end if;
kono
parents:
diff changeset
1809
kono
parents:
diff changeset
1810 Next (Alt);
kono
parents:
diff changeset
1811 end loop;
kono
parents:
diff changeset
1812
kono
parents:
diff changeset
1813 Copy_Dimensions (Frst_Expr, N);
kono
parents:
diff changeset
1814 end Analyze_Dimension_Case_Expression;
kono
parents:
diff changeset
1815
kono
parents:
diff changeset
1816 ---------------------------------------------
kono
parents:
diff changeset
1817 -- Analyze_Dimension_Component_Declaration --
kono
parents:
diff changeset
1818 ---------------------------------------------
kono
parents:
diff changeset
1819
kono
parents:
diff changeset
1820 procedure Analyze_Dimension_Component_Declaration (N : Node_Id) is
kono
parents:
diff changeset
1821 Expr : constant Node_Id := Expression (N);
kono
parents:
diff changeset
1822 Id : constant Entity_Id := Defining_Identifier (N);
kono
parents:
diff changeset
1823 Etyp : constant Entity_Id := Etype (Id);
kono
parents:
diff changeset
1824 Dims_Of_Etyp : constant Dimension_Type := Dimensions_Of (Etyp);
kono
parents:
diff changeset
1825 Dims_Of_Expr : Dimension_Type;
kono
parents:
diff changeset
1826
kono
parents:
diff changeset
1827 procedure Error_Dim_Msg_For_Component_Declaration
kono
parents:
diff changeset
1828 (N : Node_Id;
kono
parents:
diff changeset
1829 Etyp : Entity_Id;
kono
parents:
diff changeset
1830 Expr : Node_Id);
kono
parents:
diff changeset
1831 -- Error using Error_Msg_N at node N. Output the dimensions of the
kono
parents:
diff changeset
1832 -- type Etyp and the expression Expr of N.
kono
parents:
diff changeset
1833
kono
parents:
diff changeset
1834 ---------------------------------------------
kono
parents:
diff changeset
1835 -- Error_Dim_Msg_For_Component_Declaration --
kono
parents:
diff changeset
1836 ---------------------------------------------
kono
parents:
diff changeset
1837
kono
parents:
diff changeset
1838 procedure Error_Dim_Msg_For_Component_Declaration
kono
parents:
diff changeset
1839 (N : Node_Id;
kono
parents:
diff changeset
1840 Etyp : Entity_Id;
kono
parents:
diff changeset
1841 Expr : Node_Id) is
kono
parents:
diff changeset
1842 begin
kono
parents:
diff changeset
1843 Error_Msg_N ("dimensions mismatch in component declaration", N);
kono
parents:
diff changeset
1844 Error_Msg_N
kono
parents:
diff changeset
1845 ("\expected dimension " & Dimensions_Msg_Of (Etyp) & ", found "
kono
parents:
diff changeset
1846 & Dimensions_Msg_Of (Expr), Expr);
kono
parents:
diff changeset
1847 end Error_Dim_Msg_For_Component_Declaration;
kono
parents:
diff changeset
1848
kono
parents:
diff changeset
1849 -- Start of processing for Analyze_Dimension_Component_Declaration
kono
parents:
diff changeset
1850
kono
parents:
diff changeset
1851 begin
kono
parents:
diff changeset
1852 -- Expression is present
kono
parents:
diff changeset
1853
kono
parents:
diff changeset
1854 if Present (Expr) then
kono
parents:
diff changeset
1855 Dims_Of_Expr := Dimensions_Of (Expr);
kono
parents:
diff changeset
1856
kono
parents:
diff changeset
1857 -- Check dimensions match
kono
parents:
diff changeset
1858
kono
parents:
diff changeset
1859 if Dims_Of_Etyp /= Dims_Of_Expr then
kono
parents:
diff changeset
1860
kono
parents:
diff changeset
1861 -- Numeric literal case. Issue a warning if the object type is not
kono
parents:
diff changeset
1862 -- dimensionless to indicate the literal is treated as if its
kono
parents:
diff changeset
1863 -- dimension matches the type dimension.
kono
parents:
diff changeset
1864
kono
parents:
diff changeset
1865 if Nkind_In (Original_Node (Expr), N_Real_Literal,
kono
parents:
diff changeset
1866 N_Integer_Literal)
kono
parents:
diff changeset
1867 then
kono
parents:
diff changeset
1868 Dim_Warning_For_Numeric_Literal (Expr, Etyp);
kono
parents:
diff changeset
1869
kono
parents:
diff changeset
1870 -- Issue a dimension mismatch error for all other cases
kono
parents:
diff changeset
1871
kono
parents:
diff changeset
1872 else
kono
parents:
diff changeset
1873 Error_Dim_Msg_For_Component_Declaration (N, Etyp, Expr);
kono
parents:
diff changeset
1874 end if;
kono
parents:
diff changeset
1875 end if;
kono
parents:
diff changeset
1876 end if;
kono
parents:
diff changeset
1877 end Analyze_Dimension_Component_Declaration;
kono
parents:
diff changeset
1878
kono
parents:
diff changeset
1879 -------------------------------------------------
kono
parents:
diff changeset
1880 -- Analyze_Dimension_Extended_Return_Statement --
kono
parents:
diff changeset
1881 -------------------------------------------------
kono
parents:
diff changeset
1882
kono
parents:
diff changeset
1883 procedure Analyze_Dimension_Extended_Return_Statement (N : Node_Id) is
kono
parents:
diff changeset
1884 Return_Ent : constant Entity_Id := Return_Statement_Entity (N);
kono
parents:
diff changeset
1885 Return_Etyp : constant Entity_Id :=
kono
parents:
diff changeset
1886 Etype (Return_Applies_To (Return_Ent));
kono
parents:
diff changeset
1887 Return_Obj_Decls : constant List_Id := Return_Object_Declarations (N);
kono
parents:
diff changeset
1888 Return_Obj_Decl : Node_Id;
kono
parents:
diff changeset
1889 Return_Obj_Id : Entity_Id;
kono
parents:
diff changeset
1890 Return_Obj_Typ : Entity_Id;
kono
parents:
diff changeset
1891
kono
parents:
diff changeset
1892 procedure Error_Dim_Msg_For_Extended_Return_Statement
kono
parents:
diff changeset
1893 (N : Node_Id;
kono
parents:
diff changeset
1894 Return_Etyp : Entity_Id;
kono
parents:
diff changeset
1895 Return_Obj_Typ : Entity_Id);
kono
parents:
diff changeset
1896 -- Error using Error_Msg_N at node N. Output dimensions of the returned
kono
parents:
diff changeset
1897 -- type Return_Etyp and the returned object type Return_Obj_Typ of N.
kono
parents:
diff changeset
1898
kono
parents:
diff changeset
1899 -------------------------------------------------
kono
parents:
diff changeset
1900 -- Error_Dim_Msg_For_Extended_Return_Statement --
kono
parents:
diff changeset
1901 -------------------------------------------------
kono
parents:
diff changeset
1902
kono
parents:
diff changeset
1903 procedure Error_Dim_Msg_For_Extended_Return_Statement
kono
parents:
diff changeset
1904 (N : Node_Id;
kono
parents:
diff changeset
1905 Return_Etyp : Entity_Id;
kono
parents:
diff changeset
1906 Return_Obj_Typ : Entity_Id)
kono
parents:
diff changeset
1907 is
kono
parents:
diff changeset
1908 begin
kono
parents:
diff changeset
1909 Error_Msg_N ("dimensions mismatch in extended return statement", N);
kono
parents:
diff changeset
1910 Error_Msg_N
kono
parents:
diff changeset
1911 ("\expected dimension " & Dimensions_Msg_Of (Return_Etyp)
kono
parents:
diff changeset
1912 & ", found " & Dimensions_Msg_Of (Return_Obj_Typ), N);
kono
parents:
diff changeset
1913 end Error_Dim_Msg_For_Extended_Return_Statement;
kono
parents:
diff changeset
1914
kono
parents:
diff changeset
1915 -- Start of processing for Analyze_Dimension_Extended_Return_Statement
kono
parents:
diff changeset
1916
kono
parents:
diff changeset
1917 begin
kono
parents:
diff changeset
1918 if Present (Return_Obj_Decls) then
kono
parents:
diff changeset
1919 Return_Obj_Decl := First (Return_Obj_Decls);
kono
parents:
diff changeset
1920 while Present (Return_Obj_Decl) loop
kono
parents:
diff changeset
1921 if Nkind (Return_Obj_Decl) = N_Object_Declaration then
kono
parents:
diff changeset
1922 Return_Obj_Id := Defining_Identifier (Return_Obj_Decl);
kono
parents:
diff changeset
1923
kono
parents:
diff changeset
1924 if Is_Return_Object (Return_Obj_Id) then
kono
parents:
diff changeset
1925 Return_Obj_Typ := Etype (Return_Obj_Id);
kono
parents:
diff changeset
1926
kono
parents:
diff changeset
1927 -- Issue an error message if dimensions mismatch
kono
parents:
diff changeset
1928
kono
parents:
diff changeset
1929 if Dimensions_Of (Return_Etyp) /=
kono
parents:
diff changeset
1930 Dimensions_Of (Return_Obj_Typ)
kono
parents:
diff changeset
1931 then
kono
parents:
diff changeset
1932 Error_Dim_Msg_For_Extended_Return_Statement
kono
parents:
diff changeset
1933 (N, Return_Etyp, Return_Obj_Typ);
kono
parents:
diff changeset
1934 return;
kono
parents:
diff changeset
1935 end if;
kono
parents:
diff changeset
1936 end if;
kono
parents:
diff changeset
1937 end if;
kono
parents:
diff changeset
1938
kono
parents:
diff changeset
1939 Next (Return_Obj_Decl);
kono
parents:
diff changeset
1940 end loop;
kono
parents:
diff changeset
1941 end if;
kono
parents:
diff changeset
1942 end Analyze_Dimension_Extended_Return_Statement;
kono
parents:
diff changeset
1943
kono
parents:
diff changeset
1944 -----------------------------------------------------
kono
parents:
diff changeset
1945 -- Analyze_Dimension_Extension_Or_Record_Aggregate --
kono
parents:
diff changeset
1946 -----------------------------------------------------
kono
parents:
diff changeset
1947
kono
parents:
diff changeset
1948 procedure Analyze_Dimension_Extension_Or_Record_Aggregate (N : Node_Id) is
kono
parents:
diff changeset
1949 Comp : Node_Id;
kono
parents:
diff changeset
1950 Comp_Id : Entity_Id;
kono
parents:
diff changeset
1951 Comp_Typ : Entity_Id;
kono
parents:
diff changeset
1952 Expr : Node_Id;
kono
parents:
diff changeset
1953
kono
parents:
diff changeset
1954 Error_Detected : Boolean := False;
kono
parents:
diff changeset
1955 -- This flag is used in order to indicate if an error has been detected
kono
parents:
diff changeset
1956 -- so far by the compiler in this routine.
kono
parents:
diff changeset
1957
kono
parents:
diff changeset
1958 begin
kono
parents:
diff changeset
1959 -- Aspect is an Ada 2012 feature. Note that there is no need to check
kono
parents:
diff changeset
1960 -- dimensions for aggregates that don't come from source, or if we are
kono
parents:
diff changeset
1961 -- within an initialization procedure, whose expressions have been
kono
parents:
diff changeset
1962 -- checked at the point of record declaration.
kono
parents:
diff changeset
1963
kono
parents:
diff changeset
1964 if Ada_Version < Ada_2012
kono
parents:
diff changeset
1965 or else not Comes_From_Source (N)
kono
parents:
diff changeset
1966 or else Inside_Init_Proc
kono
parents:
diff changeset
1967 then
kono
parents:
diff changeset
1968 return;
kono
parents:
diff changeset
1969 end if;
kono
parents:
diff changeset
1970
kono
parents:
diff changeset
1971 Comp := First (Component_Associations (N));
kono
parents:
diff changeset
1972 while Present (Comp) loop
kono
parents:
diff changeset
1973 Comp_Id := Entity (First (Choices (Comp)));
kono
parents:
diff changeset
1974 Comp_Typ := Etype (Comp_Id);
kono
parents:
diff changeset
1975
kono
parents:
diff changeset
1976 -- Check the component type is either a dimensioned type or a
kono
parents:
diff changeset
1977 -- dimensioned subtype.
kono
parents:
diff changeset
1978
kono
parents:
diff changeset
1979 if Has_Dimension_System (Base_Type (Comp_Typ)) then
kono
parents:
diff changeset
1980 Expr := Expression (Comp);
kono
parents:
diff changeset
1981
kono
parents:
diff changeset
1982 -- A box-initialized component needs no checking.
kono
parents:
diff changeset
1983
kono
parents:
diff changeset
1984 if No (Expr) and then Box_Present (Comp) then
kono
parents:
diff changeset
1985 null;
kono
parents:
diff changeset
1986
kono
parents:
diff changeset
1987 -- Issue an error if the dimensions of the component type and the
kono
parents:
diff changeset
1988 -- dimensions of the component mismatch.
kono
parents:
diff changeset
1989
kono
parents:
diff changeset
1990 elsif Dimensions_Of (Expr) /= Dimensions_Of (Comp_Typ) then
kono
parents:
diff changeset
1991
kono
parents:
diff changeset
1992 -- Check if an error has already been encountered so far
kono
parents:
diff changeset
1993
kono
parents:
diff changeset
1994 if not Error_Detected then
kono
parents:
diff changeset
1995
kono
parents:
diff changeset
1996 -- Extension aggregate case
kono
parents:
diff changeset
1997
kono
parents:
diff changeset
1998 if Nkind (N) = N_Extension_Aggregate then
kono
parents:
diff changeset
1999 Error_Msg_N
kono
parents:
diff changeset
2000 ("dimensions mismatch in extension aggregate", N);
kono
parents:
diff changeset
2001
kono
parents:
diff changeset
2002 -- Record aggregate case
kono
parents:
diff changeset
2003
kono
parents:
diff changeset
2004 else
kono
parents:
diff changeset
2005 Error_Msg_N
kono
parents:
diff changeset
2006 ("dimensions mismatch in record aggregate", N);
kono
parents:
diff changeset
2007 end if;
kono
parents:
diff changeset
2008
kono
parents:
diff changeset
2009 Error_Detected := True;
kono
parents:
diff changeset
2010 end if;
kono
parents:
diff changeset
2011
kono
parents:
diff changeset
2012 Error_Msg_N
kono
parents:
diff changeset
2013 ("\expected dimension " & Dimensions_Msg_Of (Comp_Typ)
kono
parents:
diff changeset
2014 & ", found " & Dimensions_Msg_Of (Expr), Comp);
kono
parents:
diff changeset
2015 end if;
kono
parents:
diff changeset
2016 end if;
kono
parents:
diff changeset
2017
kono
parents:
diff changeset
2018 Next (Comp);
kono
parents:
diff changeset
2019 end loop;
kono
parents:
diff changeset
2020 end Analyze_Dimension_Extension_Or_Record_Aggregate;
kono
parents:
diff changeset
2021
kono
parents:
diff changeset
2022 -------------------------------
kono
parents:
diff changeset
2023 -- Analyze_Dimension_Formals --
kono
parents:
diff changeset
2024 -------------------------------
kono
parents:
diff changeset
2025
kono
parents:
diff changeset
2026 procedure Analyze_Dimension_Formals (N : Node_Id; Formals : List_Id) is
kono
parents:
diff changeset
2027 Dims_Of_Typ : Dimension_Type;
kono
parents:
diff changeset
2028 Formal : Node_Id;
kono
parents:
diff changeset
2029 Typ : Entity_Id;
kono
parents:
diff changeset
2030
kono
parents:
diff changeset
2031 begin
kono
parents:
diff changeset
2032 -- Aspect is an Ada 2012 feature. Note that there is no need to check
kono
parents:
diff changeset
2033 -- dimensions for sub specs that don't come from source.
kono
parents:
diff changeset
2034
kono
parents:
diff changeset
2035 if Ada_Version < Ada_2012 or else not Comes_From_Source (N) then
kono
parents:
diff changeset
2036 return;
kono
parents:
diff changeset
2037 end if;
kono
parents:
diff changeset
2038
kono
parents:
diff changeset
2039 Formal := First (Formals);
kono
parents:
diff changeset
2040 while Present (Formal) loop
kono
parents:
diff changeset
2041 Typ := Parameter_Type (Formal);
kono
parents:
diff changeset
2042 Dims_Of_Typ := Dimensions_Of (Typ);
kono
parents:
diff changeset
2043
kono
parents:
diff changeset
2044 if Exists (Dims_Of_Typ) then
kono
parents:
diff changeset
2045 declare
kono
parents:
diff changeset
2046 Expr : constant Node_Id := Expression (Formal);
kono
parents:
diff changeset
2047
kono
parents:
diff changeset
2048 begin
kono
parents:
diff changeset
2049 -- Issue a warning if Expr is a numeric literal and if its
kono
parents:
diff changeset
2050 -- dimensions differ with the dimensions of the formal type.
kono
parents:
diff changeset
2051
kono
parents:
diff changeset
2052 if Present (Expr)
kono
parents:
diff changeset
2053 and then Dims_Of_Typ /= Dimensions_Of (Expr)
kono
parents:
diff changeset
2054 and then Nkind_In (Original_Node (Expr), N_Real_Literal,
kono
parents:
diff changeset
2055 N_Integer_Literal)
kono
parents:
diff changeset
2056 then
kono
parents:
diff changeset
2057 Dim_Warning_For_Numeric_Literal (Expr, Etype (Typ));
kono
parents:
diff changeset
2058 end if;
kono
parents:
diff changeset
2059 end;
kono
parents:
diff changeset
2060 end if;
kono
parents:
diff changeset
2061
kono
parents:
diff changeset
2062 Next (Formal);
kono
parents:
diff changeset
2063 end loop;
kono
parents:
diff changeset
2064 end Analyze_Dimension_Formals;
kono
parents:
diff changeset
2065
kono
parents:
diff changeset
2066 ---------------------------------
kono
parents:
diff changeset
2067 -- Analyze_Dimension_Has_Etype --
kono
parents:
diff changeset
2068 ---------------------------------
kono
parents:
diff changeset
2069
kono
parents:
diff changeset
2070 procedure Analyze_Dimension_Has_Etype (N : Node_Id) is
kono
parents:
diff changeset
2071 Etyp : constant Entity_Id := Etype (N);
kono
parents:
diff changeset
2072 Dims_Of_Etyp : Dimension_Type := Dimensions_Of (Etyp);
kono
parents:
diff changeset
2073
kono
parents:
diff changeset
2074 begin
kono
parents:
diff changeset
2075 -- General case. Propagation of the dimensions from the type
kono
parents:
diff changeset
2076
kono
parents:
diff changeset
2077 if Exists (Dims_Of_Etyp) then
kono
parents:
diff changeset
2078 Set_Dimensions (N, Dims_Of_Etyp);
kono
parents:
diff changeset
2079
kono
parents:
diff changeset
2080 -- Identifier case. Propagate the dimensions from the entity for
kono
parents:
diff changeset
2081 -- identifier whose entity is a non-dimensionless constant.
kono
parents:
diff changeset
2082
kono
parents:
diff changeset
2083 elsif Nkind (N) = N_Identifier then
kono
parents:
diff changeset
2084 Analyze_Dimension_Identifier : declare
kono
parents:
diff changeset
2085 Id : constant Entity_Id := Entity (N);
kono
parents:
diff changeset
2086
kono
parents:
diff changeset
2087 begin
kono
parents:
diff changeset
2088 -- If Id is missing, abnormal tree, assume previous error
kono
parents:
diff changeset
2089
kono
parents:
diff changeset
2090 if No (Id) then
kono
parents:
diff changeset
2091 Check_Error_Detected;
kono
parents:
diff changeset
2092 return;
kono
parents:
diff changeset
2093
kono
parents:
diff changeset
2094 elsif Ekind_In (Id, E_Constant, E_Named_Real)
kono
parents:
diff changeset
2095 and then Exists (Dimensions_Of (Id))
kono
parents:
diff changeset
2096 then
kono
parents:
diff changeset
2097 Set_Dimensions (N, Dimensions_Of (Id));
kono
parents:
diff changeset
2098 end if;
kono
parents:
diff changeset
2099 end Analyze_Dimension_Identifier;
kono
parents:
diff changeset
2100
kono
parents:
diff changeset
2101 -- Attribute reference case. Propagate the dimensions from the prefix.
kono
parents:
diff changeset
2102
kono
parents:
diff changeset
2103 elsif Nkind (N) = N_Attribute_Reference
kono
parents:
diff changeset
2104 and then Has_Dimension_System (Base_Type (Etyp))
kono
parents:
diff changeset
2105 then
kono
parents:
diff changeset
2106 Dims_Of_Etyp := Dimensions_Of (Prefix (N));
kono
parents:
diff changeset
2107
kono
parents:
diff changeset
2108 -- Check the prefix is not dimensionless
kono
parents:
diff changeset
2109
kono
parents:
diff changeset
2110 if Exists (Dims_Of_Etyp) then
kono
parents:
diff changeset
2111 Set_Dimensions (N, Dims_Of_Etyp);
kono
parents:
diff changeset
2112 end if;
kono
parents:
diff changeset
2113 end if;
kono
parents:
diff changeset
2114
kono
parents:
diff changeset
2115 -- Remove dimensions from inner expressions, to prevent dimensions
kono
parents:
diff changeset
2116 -- table from growing uselessly.
kono
parents:
diff changeset
2117
kono
parents:
diff changeset
2118 case Nkind (N) is
kono
parents:
diff changeset
2119 when N_Attribute_Reference
kono
parents:
diff changeset
2120 | N_Indexed_Component
kono
parents:
diff changeset
2121 =>
kono
parents:
diff changeset
2122 declare
kono
parents:
diff changeset
2123 Exprs : constant List_Id := Expressions (N);
kono
parents:
diff changeset
2124 Expr : Node_Id;
kono
parents:
diff changeset
2125
kono
parents:
diff changeset
2126 begin
kono
parents:
diff changeset
2127 if Present (Exprs) then
kono
parents:
diff changeset
2128 Expr := First (Exprs);
kono
parents:
diff changeset
2129 while Present (Expr) loop
kono
parents:
diff changeset
2130 Remove_Dimensions (Expr);
kono
parents:
diff changeset
2131 Next (Expr);
kono
parents:
diff changeset
2132 end loop;
kono
parents:
diff changeset
2133 end if;
kono
parents:
diff changeset
2134 end;
kono
parents:
diff changeset
2135
kono
parents:
diff changeset
2136 when N_Qualified_Expression
kono
parents:
diff changeset
2137 | N_Type_Conversion
kono
parents:
diff changeset
2138 | N_Unchecked_Type_Conversion
kono
parents:
diff changeset
2139 =>
kono
parents:
diff changeset
2140 Remove_Dimensions (Expression (N));
kono
parents:
diff changeset
2141
kono
parents:
diff changeset
2142 when N_Selected_Component =>
kono
parents:
diff changeset
2143 Remove_Dimensions (Selector_Name (N));
kono
parents:
diff changeset
2144
kono
parents:
diff changeset
2145 when others =>
kono
parents:
diff changeset
2146 null;
kono
parents:
diff changeset
2147 end case;
kono
parents:
diff changeset
2148 end Analyze_Dimension_Has_Etype;
kono
parents:
diff changeset
2149
kono
parents:
diff changeset
2150 -------------------------------------
kono
parents:
diff changeset
2151 -- Analyze_Dimension_If_Expression --
kono
parents:
diff changeset
2152 -------------------------------------
kono
parents:
diff changeset
2153
kono
parents:
diff changeset
2154 procedure Analyze_Dimension_If_Expression (N : Node_Id) is
kono
parents:
diff changeset
2155 Then_Expr : constant Node_Id := Next (First (Expressions (N)));
kono
parents:
diff changeset
2156 Else_Expr : constant Node_Id := Next (Then_Expr);
kono
parents:
diff changeset
2157
kono
parents:
diff changeset
2158 begin
kono
parents:
diff changeset
2159 if Dimensions_Of (Then_Expr) /= Dimensions_Of (Else_Expr) then
kono
parents:
diff changeset
2160 Error_Msg_N ("dimensions mismatch in conditional expression", N);
kono
parents:
diff changeset
2161 else
kono
parents:
diff changeset
2162 Copy_Dimensions (Then_Expr, N);
kono
parents:
diff changeset
2163 end if;
kono
parents:
diff changeset
2164 end Analyze_Dimension_If_Expression;
kono
parents:
diff changeset
2165
kono
parents:
diff changeset
2166 ------------------------------------------
kono
parents:
diff changeset
2167 -- Analyze_Dimension_Number_Declaration --
kono
parents:
diff changeset
2168 ------------------------------------------
kono
parents:
diff changeset
2169
kono
parents:
diff changeset
2170 procedure Analyze_Dimension_Number_Declaration (N : Node_Id) is
kono
parents:
diff changeset
2171 Expr : constant Node_Id := Expression (N);
kono
parents:
diff changeset
2172 Id : constant Entity_Id := Defining_Identifier (N);
kono
parents:
diff changeset
2173 Dim_Of_Expr : constant Dimension_Type := Dimensions_Of (Expr);
kono
parents:
diff changeset
2174
kono
parents:
diff changeset
2175 begin
kono
parents:
diff changeset
2176 if Exists (Dim_Of_Expr) then
kono
parents:
diff changeset
2177 Set_Dimensions (Id, Dim_Of_Expr);
kono
parents:
diff changeset
2178 Set_Etype (Id, Etype (Expr));
kono
parents:
diff changeset
2179 end if;
kono
parents:
diff changeset
2180 end Analyze_Dimension_Number_Declaration;
kono
parents:
diff changeset
2181
kono
parents:
diff changeset
2182 ------------------------------------------
kono
parents:
diff changeset
2183 -- Analyze_Dimension_Object_Declaration --
kono
parents:
diff changeset
2184 ------------------------------------------
kono
parents:
diff changeset
2185
kono
parents:
diff changeset
2186 procedure Analyze_Dimension_Object_Declaration (N : Node_Id) is
kono
parents:
diff changeset
2187 Expr : constant Node_Id := Expression (N);
kono
parents:
diff changeset
2188 Id : constant Entity_Id := Defining_Identifier (N);
kono
parents:
diff changeset
2189 Etyp : constant Entity_Id := Etype (Id);
kono
parents:
diff changeset
2190 Dim_Of_Etyp : constant Dimension_Type := Dimensions_Of (Etyp);
kono
parents:
diff changeset
2191 Dim_Of_Expr : Dimension_Type;
kono
parents:
diff changeset
2192
kono
parents:
diff changeset
2193 procedure Error_Dim_Msg_For_Object_Declaration
kono
parents:
diff changeset
2194 (N : Node_Id;
kono
parents:
diff changeset
2195 Etyp : Entity_Id;
kono
parents:
diff changeset
2196 Expr : Node_Id);
kono
parents:
diff changeset
2197 -- Error using Error_Msg_N at node N. Output the dimensions of the
kono
parents:
diff changeset
2198 -- type Etyp and of the expression Expr.
kono
parents:
diff changeset
2199
kono
parents:
diff changeset
2200 ------------------------------------------
kono
parents:
diff changeset
2201 -- Error_Dim_Msg_For_Object_Declaration --
kono
parents:
diff changeset
2202 ------------------------------------------
kono
parents:
diff changeset
2203
kono
parents:
diff changeset
2204 procedure Error_Dim_Msg_For_Object_Declaration
kono
parents:
diff changeset
2205 (N : Node_Id;
kono
parents:
diff changeset
2206 Etyp : Entity_Id;
kono
parents:
diff changeset
2207 Expr : Node_Id) is
kono
parents:
diff changeset
2208 begin
kono
parents:
diff changeset
2209 Error_Msg_N ("dimensions mismatch in object declaration", N);
kono
parents:
diff changeset
2210 Error_Msg_N
kono
parents:
diff changeset
2211 ("\expected dimension " & Dimensions_Msg_Of (Etyp) & ", found "
kono
parents:
diff changeset
2212 & Dimensions_Msg_Of (Expr), Expr);
kono
parents:
diff changeset
2213 end Error_Dim_Msg_For_Object_Declaration;
kono
parents:
diff changeset
2214
kono
parents:
diff changeset
2215 -- Start of processing for Analyze_Dimension_Object_Declaration
kono
parents:
diff changeset
2216
kono
parents:
diff changeset
2217 begin
kono
parents:
diff changeset
2218 -- Expression is present
kono
parents:
diff changeset
2219
kono
parents:
diff changeset
2220 if Present (Expr) then
kono
parents:
diff changeset
2221 Dim_Of_Expr := Dimensions_Of (Expr);
kono
parents:
diff changeset
2222
kono
parents:
diff changeset
2223 -- Check dimensions match
kono
parents:
diff changeset
2224
kono
parents:
diff changeset
2225 if Dim_Of_Expr /= Dim_Of_Etyp then
kono
parents:
diff changeset
2226
kono
parents:
diff changeset
2227 -- Numeric literal case. Issue a warning if the object type is
kono
parents:
diff changeset
2228 -- not dimensionless to indicate the literal is treated as if
kono
parents:
diff changeset
2229 -- its dimension matches the type dimension.
kono
parents:
diff changeset
2230
kono
parents:
diff changeset
2231 if Nkind_In (Original_Node (Expr), N_Real_Literal,
kono
parents:
diff changeset
2232 N_Integer_Literal)
kono
parents:
diff changeset
2233 then
kono
parents:
diff changeset
2234 Dim_Warning_For_Numeric_Literal (Expr, Etyp);
kono
parents:
diff changeset
2235
kono
parents:
diff changeset
2236 -- Case of object is a constant whose type is a dimensioned type
kono
parents:
diff changeset
2237
kono
parents:
diff changeset
2238 elsif Constant_Present (N) and then not Exists (Dim_Of_Etyp) then
kono
parents:
diff changeset
2239
kono
parents:
diff changeset
2240 -- Propagate dimension from expression to object entity
kono
parents:
diff changeset
2241
kono
parents:
diff changeset
2242 Set_Dimensions (Id, Dim_Of_Expr);
kono
parents:
diff changeset
2243
kono
parents:
diff changeset
2244 -- Expression may have been constant-folded. If nominal type has
kono
parents:
diff changeset
2245 -- dimensions, verify that expression has same type.
kono
parents:
diff changeset
2246
kono
parents:
diff changeset
2247 elsif Exists (Dim_Of_Etyp) and then Etype (Expr) = Etyp then
kono
parents:
diff changeset
2248 null;
kono
parents:
diff changeset
2249
kono
parents:
diff changeset
2250 -- For all other cases, issue an error message
kono
parents:
diff changeset
2251
kono
parents:
diff changeset
2252 else
kono
parents:
diff changeset
2253 Error_Dim_Msg_For_Object_Declaration (N, Etyp, Expr);
kono
parents:
diff changeset
2254 end if;
kono
parents:
diff changeset
2255 end if;
kono
parents:
diff changeset
2256
kono
parents:
diff changeset
2257 -- Remove dimensions in expression after checking consistency with
kono
parents:
diff changeset
2258 -- given type.
kono
parents:
diff changeset
2259
kono
parents:
diff changeset
2260 Remove_Dimensions (Expr);
kono
parents:
diff changeset
2261 end if;
kono
parents:
diff changeset
2262 end Analyze_Dimension_Object_Declaration;
kono
parents:
diff changeset
2263
kono
parents:
diff changeset
2264 ---------------------------------------------------
kono
parents:
diff changeset
2265 -- Analyze_Dimension_Object_Renaming_Declaration --
kono
parents:
diff changeset
2266 ---------------------------------------------------
kono
parents:
diff changeset
2267
kono
parents:
diff changeset
2268 procedure Analyze_Dimension_Object_Renaming_Declaration (N : Node_Id) is
kono
parents:
diff changeset
2269 Renamed_Name : constant Node_Id := Name (N);
kono
parents:
diff changeset
2270 Sub_Mark : constant Node_Id := Subtype_Mark (N);
kono
parents:
diff changeset
2271
kono
parents:
diff changeset
2272 procedure Error_Dim_Msg_For_Object_Renaming_Declaration
kono
parents:
diff changeset
2273 (N : Node_Id;
kono
parents:
diff changeset
2274 Sub_Mark : Node_Id;
kono
parents:
diff changeset
2275 Renamed_Name : Node_Id);
kono
parents:
diff changeset
2276 -- Error using Error_Msg_N at node N. Output the dimensions of
kono
parents:
diff changeset
2277 -- Sub_Mark and of Renamed_Name.
kono
parents:
diff changeset
2278
kono
parents:
diff changeset
2279 ---------------------------------------------------
kono
parents:
diff changeset
2280 -- Error_Dim_Msg_For_Object_Renaming_Declaration --
kono
parents:
diff changeset
2281 ---------------------------------------------------
kono
parents:
diff changeset
2282
kono
parents:
diff changeset
2283 procedure Error_Dim_Msg_For_Object_Renaming_Declaration
kono
parents:
diff changeset
2284 (N : Node_Id;
kono
parents:
diff changeset
2285 Sub_Mark : Node_Id;
kono
parents:
diff changeset
2286 Renamed_Name : Node_Id) is
kono
parents:
diff changeset
2287 begin
kono
parents:
diff changeset
2288 Error_Msg_N ("dimensions mismatch in object renaming declaration", N);
kono
parents:
diff changeset
2289 Error_Msg_N
kono
parents:
diff changeset
2290 ("\expected dimension " & Dimensions_Msg_Of (Sub_Mark) & ", found "
kono
parents:
diff changeset
2291 & Dimensions_Msg_Of (Renamed_Name), Renamed_Name);
kono
parents:
diff changeset
2292 end Error_Dim_Msg_For_Object_Renaming_Declaration;
kono
parents:
diff changeset
2293
kono
parents:
diff changeset
2294 -- Start of processing for Analyze_Dimension_Object_Renaming_Declaration
kono
parents:
diff changeset
2295
kono
parents:
diff changeset
2296 begin
kono
parents:
diff changeset
2297 if Dimensions_Of (Renamed_Name) /= Dimensions_Of (Sub_Mark) then
kono
parents:
diff changeset
2298 Error_Dim_Msg_For_Object_Renaming_Declaration
kono
parents:
diff changeset
2299 (N, Sub_Mark, Renamed_Name);
kono
parents:
diff changeset
2300 end if;
kono
parents:
diff changeset
2301 end Analyze_Dimension_Object_Renaming_Declaration;
kono
parents:
diff changeset
2302
kono
parents:
diff changeset
2303 -----------------------------------------------
kono
parents:
diff changeset
2304 -- Analyze_Dimension_Simple_Return_Statement --
kono
parents:
diff changeset
2305 -----------------------------------------------
kono
parents:
diff changeset
2306
kono
parents:
diff changeset
2307 procedure Analyze_Dimension_Simple_Return_Statement (N : Node_Id) is
kono
parents:
diff changeset
2308 Expr : constant Node_Id := Expression (N);
kono
parents:
diff changeset
2309 Return_Ent : constant Entity_Id := Return_Statement_Entity (N);
kono
parents:
diff changeset
2310 Return_Etyp : constant Entity_Id :=
kono
parents:
diff changeset
2311 Etype (Return_Applies_To (Return_Ent));
kono
parents:
diff changeset
2312 Dims_Of_Return_Etyp : constant Dimension_Type :=
kono
parents:
diff changeset
2313 Dimensions_Of (Return_Etyp);
kono
parents:
diff changeset
2314
kono
parents:
diff changeset
2315 procedure Error_Dim_Msg_For_Simple_Return_Statement
kono
parents:
diff changeset
2316 (N : Node_Id;
kono
parents:
diff changeset
2317 Return_Etyp : Entity_Id;
kono
parents:
diff changeset
2318 Expr : Node_Id);
kono
parents:
diff changeset
2319 -- Error using Error_Msg_N at node N. Output the dimensions of the
kono
parents:
diff changeset
2320 -- returned type Return_Etyp and the returned expression Expr of N.
kono
parents:
diff changeset
2321
kono
parents:
diff changeset
2322 -----------------------------------------------
kono
parents:
diff changeset
2323 -- Error_Dim_Msg_For_Simple_Return_Statement --
kono
parents:
diff changeset
2324 -----------------------------------------------
kono
parents:
diff changeset
2325
kono
parents:
diff changeset
2326 procedure Error_Dim_Msg_For_Simple_Return_Statement
kono
parents:
diff changeset
2327 (N : Node_Id;
kono
parents:
diff changeset
2328 Return_Etyp : Entity_Id;
kono
parents:
diff changeset
2329 Expr : Node_Id)
kono
parents:
diff changeset
2330 is
kono
parents:
diff changeset
2331 begin
kono
parents:
diff changeset
2332 Error_Msg_N ("dimensions mismatch in return statement", N);
kono
parents:
diff changeset
2333 Error_Msg_N
kono
parents:
diff changeset
2334 ("\expected dimension " & Dimensions_Msg_Of (Return_Etyp)
kono
parents:
diff changeset
2335 & ", found " & Dimensions_Msg_Of (Expr), Expr);
kono
parents:
diff changeset
2336 end Error_Dim_Msg_For_Simple_Return_Statement;
kono
parents:
diff changeset
2337
kono
parents:
diff changeset
2338 -- Start of processing for Analyze_Dimension_Simple_Return_Statement
kono
parents:
diff changeset
2339
kono
parents:
diff changeset
2340 begin
kono
parents:
diff changeset
2341 if Dims_Of_Return_Etyp /= Dimensions_Of (Expr) then
kono
parents:
diff changeset
2342 Error_Dim_Msg_For_Simple_Return_Statement (N, Return_Etyp, Expr);
kono
parents:
diff changeset
2343 Remove_Dimensions (Expr);
kono
parents:
diff changeset
2344 end if;
kono
parents:
diff changeset
2345 end Analyze_Dimension_Simple_Return_Statement;
kono
parents:
diff changeset
2346
kono
parents:
diff changeset
2347 -------------------------------------------
kono
parents:
diff changeset
2348 -- Analyze_Dimension_Subtype_Declaration --
kono
parents:
diff changeset
2349 -------------------------------------------
kono
parents:
diff changeset
2350
kono
parents:
diff changeset
2351 procedure Analyze_Dimension_Subtype_Declaration (N : Node_Id) is
kono
parents:
diff changeset
2352 Id : constant Entity_Id := Defining_Identifier (N);
kono
parents:
diff changeset
2353 Dims_Of_Id : constant Dimension_Type := Dimensions_Of (Id);
kono
parents:
diff changeset
2354 Dims_Of_Etyp : Dimension_Type;
kono
parents:
diff changeset
2355 Etyp : Node_Id;
kono
parents:
diff changeset
2356
kono
parents:
diff changeset
2357 begin
kono
parents:
diff changeset
2358 -- No constraint case in subtype declaration
kono
parents:
diff changeset
2359
kono
parents:
diff changeset
2360 if Nkind (Subtype_Indication (N)) /= N_Subtype_Indication then
kono
parents:
diff changeset
2361 Etyp := Etype (Subtype_Indication (N));
kono
parents:
diff changeset
2362 Dims_Of_Etyp := Dimensions_Of (Etyp);
kono
parents:
diff changeset
2363
kono
parents:
diff changeset
2364 if Exists (Dims_Of_Etyp) then
kono
parents:
diff changeset
2365
kono
parents:
diff changeset
2366 -- If subtype already has a dimension (from Aspect_Dimension), it
kono
parents:
diff changeset
2367 -- cannot inherit different dimensions from its subtype.
kono
parents:
diff changeset
2368
kono
parents:
diff changeset
2369 if Exists (Dims_Of_Id) and then Dims_Of_Etyp /= Dims_Of_Id then
kono
parents:
diff changeset
2370 Error_Msg_NE
kono
parents:
diff changeset
2371 ("subtype& already " & Dimensions_Msg_Of (Id, True), N, Id);
kono
parents:
diff changeset
2372 else
kono
parents:
diff changeset
2373 Set_Dimensions (Id, Dims_Of_Etyp);
kono
parents:
diff changeset
2374 Set_Symbol (Id, Symbol_Of (Etyp));
kono
parents:
diff changeset
2375 end if;
kono
parents:
diff changeset
2376 end if;
kono
parents:
diff changeset
2377
kono
parents:
diff changeset
2378 -- Constraint present in subtype declaration
kono
parents:
diff changeset
2379
kono
parents:
diff changeset
2380 else
kono
parents:
diff changeset
2381 Etyp := Etype (Subtype_Mark (Subtype_Indication (N)));
kono
parents:
diff changeset
2382 Dims_Of_Etyp := Dimensions_Of (Etyp);
kono
parents:
diff changeset
2383
kono
parents:
diff changeset
2384 if Exists (Dims_Of_Etyp) then
kono
parents:
diff changeset
2385 Set_Dimensions (Id, Dims_Of_Etyp);
kono
parents:
diff changeset
2386 Set_Symbol (Id, Symbol_Of (Etyp));
kono
parents:
diff changeset
2387 end if;
kono
parents:
diff changeset
2388 end if;
kono
parents:
diff changeset
2389 end Analyze_Dimension_Subtype_Declaration;
kono
parents:
diff changeset
2390
kono
parents:
diff changeset
2391 ---------------------------------------
kono
parents:
diff changeset
2392 -- Analyze_Dimension_Type_Conversion --
kono
parents:
diff changeset
2393 ---------------------------------------
kono
parents:
diff changeset
2394
kono
parents:
diff changeset
2395 procedure Analyze_Dimension_Type_Conversion (N : Node_Id) is
kono
parents:
diff changeset
2396 Expr_Root : constant Entity_Id :=
kono
parents:
diff changeset
2397 Dimension_System_Root (Etype (Expression (N)));
kono
parents:
diff changeset
2398 Target_Root : constant Entity_Id :=
kono
parents:
diff changeset
2399 Dimension_System_Root (Etype (N));
kono
parents:
diff changeset
2400
kono
parents:
diff changeset
2401 begin
kono
parents:
diff changeset
2402 -- If the expression has dimensions and the target type has dimensions,
kono
parents:
diff changeset
2403 -- the conversion has the dimensions of the expression. Consistency is
kono
parents:
diff changeset
2404 -- checked below. Converting to a non-dimensioned type such as Float
kono
parents:
diff changeset
2405 -- ignores the dimensions of the expression.
kono
parents:
diff changeset
2406
kono
parents:
diff changeset
2407 if Exists (Dimensions_Of (Expression (N)))
kono
parents:
diff changeset
2408 and then Present (Target_Root)
kono
parents:
diff changeset
2409 then
kono
parents:
diff changeset
2410 Set_Dimensions (N, Dimensions_Of (Expression (N)));
kono
parents:
diff changeset
2411
kono
parents:
diff changeset
2412 -- Otherwise the dimensions are those of the target type.
kono
parents:
diff changeset
2413
kono
parents:
diff changeset
2414 else
kono
parents:
diff changeset
2415 Analyze_Dimension_Has_Etype (N);
kono
parents:
diff changeset
2416 end if;
kono
parents:
diff changeset
2417
kono
parents:
diff changeset
2418 -- A conversion between types in different dimension systems (e.g. MKS
kono
parents:
diff changeset
2419 -- and British units) must respect the dimensions of expression and
kono
parents:
diff changeset
2420 -- type, It is up to the user to provide proper conversion factors.
kono
parents:
diff changeset
2421
kono
parents:
diff changeset
2422 -- Upward conversions to root type of a dimensioned system are legal,
kono
parents:
diff changeset
2423 -- and correspond to "view conversions", i.e. preserve the dimensions
kono
parents:
diff changeset
2424 -- of the expression; otherwise conversion must be between types with
kono
parents:
diff changeset
2425 -- then same dimensions. Conversions to a non-dimensioned type such as
kono
parents:
diff changeset
2426 -- Float lose the dimensions of the expression.
kono
parents:
diff changeset
2427
kono
parents:
diff changeset
2428 if Present (Expr_Root)
kono
parents:
diff changeset
2429 and then Present (Target_Root)
kono
parents:
diff changeset
2430 and then Etype (N) /= Target_Root
kono
parents:
diff changeset
2431 and then Dimensions_Of (Expression (N)) /= Dimensions_Of (Etype (N))
kono
parents:
diff changeset
2432 then
kono
parents:
diff changeset
2433 Error_Msg_N ("dimensions mismatch in conversion", N);
kono
parents:
diff changeset
2434 Error_Msg_N
kono
parents:
diff changeset
2435 ("\expression " & Dimensions_Msg_Of (Expression (N), True), N);
kono
parents:
diff changeset
2436 Error_Msg_N
kono
parents:
diff changeset
2437 ("\target type " & Dimensions_Msg_Of (Etype (N), True), N);
kono
parents:
diff changeset
2438 end if;
kono
parents:
diff changeset
2439 end Analyze_Dimension_Type_Conversion;
kono
parents:
diff changeset
2440
kono
parents:
diff changeset
2441 --------------------------------
kono
parents:
diff changeset
2442 -- Analyze_Dimension_Unary_Op --
kono
parents:
diff changeset
2443 --------------------------------
kono
parents:
diff changeset
2444
kono
parents:
diff changeset
2445 procedure Analyze_Dimension_Unary_Op (N : Node_Id) is
kono
parents:
diff changeset
2446 begin
kono
parents:
diff changeset
2447 case Nkind (N) is
kono
parents:
diff changeset
2448
kono
parents:
diff changeset
2449 -- Propagate the dimension if the operand is not dimensionless
kono
parents:
diff changeset
2450
kono
parents:
diff changeset
2451 when N_Op_Abs
kono
parents:
diff changeset
2452 | N_Op_Minus
kono
parents:
diff changeset
2453 | N_Op_Plus
kono
parents:
diff changeset
2454 =>
kono
parents:
diff changeset
2455 declare
kono
parents:
diff changeset
2456 R : constant Node_Id := Right_Opnd (N);
kono
parents:
diff changeset
2457 begin
kono
parents:
diff changeset
2458 Move_Dimensions (R, N);
kono
parents:
diff changeset
2459 end;
kono
parents:
diff changeset
2460
kono
parents:
diff changeset
2461 when others =>
kono
parents:
diff changeset
2462 null;
kono
parents:
diff changeset
2463 end case;
kono
parents:
diff changeset
2464 end Analyze_Dimension_Unary_Op;
kono
parents:
diff changeset
2465
kono
parents:
diff changeset
2466 ---------------------------------
kono
parents:
diff changeset
2467 -- Check_Expression_Dimensions --
kono
parents:
diff changeset
2468 ---------------------------------
kono
parents:
diff changeset
2469
kono
parents:
diff changeset
2470 procedure Check_Expression_Dimensions
kono
parents:
diff changeset
2471 (Expr : Node_Id;
kono
parents:
diff changeset
2472 Typ : Entity_Id)
kono
parents:
diff changeset
2473 is
kono
parents:
diff changeset
2474 begin
kono
parents:
diff changeset
2475 if Is_Floating_Point_Type (Etype (Expr)) then
kono
parents:
diff changeset
2476 Analyze_Dimension (Expr);
kono
parents:
diff changeset
2477
kono
parents:
diff changeset
2478 if Dimensions_Of (Expr) /= Dimensions_Of (Typ) then
kono
parents:
diff changeset
2479 Error_Msg_N ("dimensions mismatch in array aggregate", Expr);
kono
parents:
diff changeset
2480 Error_Msg_N
kono
parents:
diff changeset
2481 ("\expected dimension " & Dimensions_Msg_Of (Typ)
kono
parents:
diff changeset
2482 & ", found " & Dimensions_Msg_Of (Expr), Expr);
kono
parents:
diff changeset
2483 end if;
kono
parents:
diff changeset
2484 end if;
kono
parents:
diff changeset
2485 end Check_Expression_Dimensions;
kono
parents:
diff changeset
2486
kono
parents:
diff changeset
2487 ---------------------
kono
parents:
diff changeset
2488 -- Copy_Dimensions --
kono
parents:
diff changeset
2489 ---------------------
kono
parents:
diff changeset
2490
kono
parents:
diff changeset
2491 procedure Copy_Dimensions (From : Node_Id; To : Node_Id) is
kono
parents:
diff changeset
2492 Dims_Of_From : constant Dimension_Type := Dimensions_Of (From);
kono
parents:
diff changeset
2493
kono
parents:
diff changeset
2494 begin
kono
parents:
diff changeset
2495 -- Ignore if not Ada 2012 or beyond
kono
parents:
diff changeset
2496
kono
parents:
diff changeset
2497 if Ada_Version < Ada_2012 then
kono
parents:
diff changeset
2498 return;
kono
parents:
diff changeset
2499
kono
parents:
diff changeset
2500 -- For Ada 2012, Copy the dimension of 'From to 'To'
kono
parents:
diff changeset
2501
kono
parents:
diff changeset
2502 elsif Exists (Dims_Of_From) then
kono
parents:
diff changeset
2503 Set_Dimensions (To, Dims_Of_From);
kono
parents:
diff changeset
2504 end if;
kono
parents:
diff changeset
2505 end Copy_Dimensions;
kono
parents:
diff changeset
2506
kono
parents:
diff changeset
2507 -----------------------------------
kono
parents:
diff changeset
2508 -- Copy_Dimensions_Of_Components --
kono
parents:
diff changeset
2509 -----------------------------------
kono
parents:
diff changeset
2510
kono
parents:
diff changeset
2511 procedure Copy_Dimensions_Of_Components (Rec : Entity_Id) is
kono
parents:
diff changeset
2512 C : Entity_Id;
kono
parents:
diff changeset
2513
kono
parents:
diff changeset
2514 begin
kono
parents:
diff changeset
2515 C := First_Component (Rec);
kono
parents:
diff changeset
2516 while Present (C) loop
kono
parents:
diff changeset
2517 if Nkind (Parent (C)) = N_Component_Declaration then
kono
parents:
diff changeset
2518 Copy_Dimensions
kono
parents:
diff changeset
2519 (Expression (Parent (Corresponding_Record_Component (C))),
kono
parents:
diff changeset
2520 Expression (Parent (C)));
kono
parents:
diff changeset
2521 end if;
kono
parents:
diff changeset
2522 Next_Component (C);
kono
parents:
diff changeset
2523 end loop;
kono
parents:
diff changeset
2524 end Copy_Dimensions_Of_Components;
kono
parents:
diff changeset
2525
kono
parents:
diff changeset
2526 --------------------------
kono
parents:
diff changeset
2527 -- Create_Rational_From --
kono
parents:
diff changeset
2528 --------------------------
kono
parents:
diff changeset
2529
kono
parents:
diff changeset
2530 -- RATIONAL ::= [-] NUMERAL [/ NUMERAL]
kono
parents:
diff changeset
2531
kono
parents:
diff changeset
2532 -- A rational number is a number that can be expressed as the quotient or
kono
parents:
diff changeset
2533 -- fraction a/b of two integers, where b is non-zero positive.
kono
parents:
diff changeset
2534
kono
parents:
diff changeset
2535 function Create_Rational_From
kono
parents:
diff changeset
2536 (Expr : Node_Id;
kono
parents:
diff changeset
2537 Complain : Boolean) return Rational
kono
parents:
diff changeset
2538 is
kono
parents:
diff changeset
2539 Or_Node_Of_Expr : constant Node_Id := Original_Node (Expr);
kono
parents:
diff changeset
2540 Result : Rational := No_Rational;
kono
parents:
diff changeset
2541
kono
parents:
diff changeset
2542 function Process_Minus (N : Node_Id) return Rational;
kono
parents:
diff changeset
2543 -- Create a rational from a N_Op_Minus node
kono
parents:
diff changeset
2544
kono
parents:
diff changeset
2545 function Process_Divide (N : Node_Id) return Rational;
kono
parents:
diff changeset
2546 -- Create a rational from a N_Op_Divide node
kono
parents:
diff changeset
2547
kono
parents:
diff changeset
2548 function Process_Literal (N : Node_Id) return Rational;
kono
parents:
diff changeset
2549 -- Create a rational from a N_Integer_Literal node
kono
parents:
diff changeset
2550
kono
parents:
diff changeset
2551 -------------------
kono
parents:
diff changeset
2552 -- Process_Minus --
kono
parents:
diff changeset
2553 -------------------
kono
parents:
diff changeset
2554
kono
parents:
diff changeset
2555 function Process_Minus (N : Node_Id) return Rational is
kono
parents:
diff changeset
2556 Right : constant Node_Id := Original_Node (Right_Opnd (N));
kono
parents:
diff changeset
2557 Result : Rational;
kono
parents:
diff changeset
2558
kono
parents:
diff changeset
2559 begin
kono
parents:
diff changeset
2560 -- Operand is an integer literal
kono
parents:
diff changeset
2561
kono
parents:
diff changeset
2562 if Nkind (Right) = N_Integer_Literal then
kono
parents:
diff changeset
2563 Result := -Process_Literal (Right);
kono
parents:
diff changeset
2564
kono
parents:
diff changeset
2565 -- Operand is a divide operator
kono
parents:
diff changeset
2566
kono
parents:
diff changeset
2567 elsif Nkind (Right) = N_Op_Divide then
kono
parents:
diff changeset
2568 Result := -Process_Divide (Right);
kono
parents:
diff changeset
2569
kono
parents:
diff changeset
2570 else
kono
parents:
diff changeset
2571 Result := No_Rational;
kono
parents:
diff changeset
2572 end if;
kono
parents:
diff changeset
2573
kono
parents:
diff changeset
2574 -- Provide minimal semantic information on dimension expressions,
kono
parents:
diff changeset
2575 -- even though they have no run-time existence. This is for use by
kono
parents:
diff changeset
2576 -- ASIS tools, in particular pretty-printing. If generating code
kono
parents:
diff changeset
2577 -- standard operator resolution will take place.
kono
parents:
diff changeset
2578
kono
parents:
diff changeset
2579 if ASIS_Mode then
kono
parents:
diff changeset
2580 Set_Entity (N, Standard_Op_Minus);
kono
parents:
diff changeset
2581 Set_Etype (N, Standard_Integer);
kono
parents:
diff changeset
2582 end if;
kono
parents:
diff changeset
2583
kono
parents:
diff changeset
2584 return Result;
kono
parents:
diff changeset
2585 end Process_Minus;
kono
parents:
diff changeset
2586
kono
parents:
diff changeset
2587 --------------------
kono
parents:
diff changeset
2588 -- Process_Divide --
kono
parents:
diff changeset
2589 --------------------
kono
parents:
diff changeset
2590
kono
parents:
diff changeset
2591 function Process_Divide (N : Node_Id) return Rational is
kono
parents:
diff changeset
2592 Left : constant Node_Id := Original_Node (Left_Opnd (N));
kono
parents:
diff changeset
2593 Right : constant Node_Id := Original_Node (Right_Opnd (N));
kono
parents:
diff changeset
2594 Left_Rat : Rational;
kono
parents:
diff changeset
2595 Result : Rational := No_Rational;
kono
parents:
diff changeset
2596 Right_Rat : Rational;
kono
parents:
diff changeset
2597
kono
parents:
diff changeset
2598 begin
kono
parents:
diff changeset
2599 -- Both left and right operands are integer literals
kono
parents:
diff changeset
2600
kono
parents:
diff changeset
2601 if Nkind (Left) = N_Integer_Literal
kono
parents:
diff changeset
2602 and then
kono
parents:
diff changeset
2603 Nkind (Right) = N_Integer_Literal
kono
parents:
diff changeset
2604 then
kono
parents:
diff changeset
2605 Left_Rat := Process_Literal (Left);
kono
parents:
diff changeset
2606 Right_Rat := Process_Literal (Right);
kono
parents:
diff changeset
2607 Result := Left_Rat / Right_Rat;
kono
parents:
diff changeset
2608 end if;
kono
parents:
diff changeset
2609
kono
parents:
diff changeset
2610 -- Provide minimal semantic information on dimension expressions,
kono
parents:
diff changeset
2611 -- even though they have no run-time existence. This is for use by
kono
parents:
diff changeset
2612 -- ASIS tools, in particular pretty-printing. If generating code
kono
parents:
diff changeset
2613 -- standard operator resolution will take place.
kono
parents:
diff changeset
2614
kono
parents:
diff changeset
2615 if ASIS_Mode then
kono
parents:
diff changeset
2616 Set_Entity (N, Standard_Op_Divide);
kono
parents:
diff changeset
2617 Set_Etype (N, Standard_Integer);
kono
parents:
diff changeset
2618 end if;
kono
parents:
diff changeset
2619
kono
parents:
diff changeset
2620 return Result;
kono
parents:
diff changeset
2621 end Process_Divide;
kono
parents:
diff changeset
2622
kono
parents:
diff changeset
2623 ---------------------
kono
parents:
diff changeset
2624 -- Process_Literal --
kono
parents:
diff changeset
2625 ---------------------
kono
parents:
diff changeset
2626
kono
parents:
diff changeset
2627 function Process_Literal (N : Node_Id) return Rational is
kono
parents:
diff changeset
2628 begin
kono
parents:
diff changeset
2629 return +Whole (UI_To_Int (Intval (N)));
kono
parents:
diff changeset
2630 end Process_Literal;
kono
parents:
diff changeset
2631
kono
parents:
diff changeset
2632 -- Start of processing for Create_Rational_From
kono
parents:
diff changeset
2633
kono
parents:
diff changeset
2634 begin
kono
parents:
diff changeset
2635 -- Check the expression is either a division of two integers or an
kono
parents:
diff changeset
2636 -- integer itself. Note that the check applies to the original node
kono
parents:
diff changeset
2637 -- since the node could have already been rewritten.
kono
parents:
diff changeset
2638
kono
parents:
diff changeset
2639 -- Integer literal case
kono
parents:
diff changeset
2640
kono
parents:
diff changeset
2641 if Nkind (Or_Node_Of_Expr) = N_Integer_Literal then
kono
parents:
diff changeset
2642 Result := Process_Literal (Or_Node_Of_Expr);
kono
parents:
diff changeset
2643
kono
parents:
diff changeset
2644 -- Divide operator case
kono
parents:
diff changeset
2645
kono
parents:
diff changeset
2646 elsif Nkind (Or_Node_Of_Expr) = N_Op_Divide then
kono
parents:
diff changeset
2647 Result := Process_Divide (Or_Node_Of_Expr);
kono
parents:
diff changeset
2648
kono
parents:
diff changeset
2649 -- Minus operator case
kono
parents:
diff changeset
2650
kono
parents:
diff changeset
2651 elsif Nkind (Or_Node_Of_Expr) = N_Op_Minus then
kono
parents:
diff changeset
2652 Result := Process_Minus (Or_Node_Of_Expr);
kono
parents:
diff changeset
2653 end if;
kono
parents:
diff changeset
2654
kono
parents:
diff changeset
2655 -- When Expr cannot be interpreted as a rational and Complain is true,
kono
parents:
diff changeset
2656 -- generate an error message.
kono
parents:
diff changeset
2657
kono
parents:
diff changeset
2658 if Complain and then Result = No_Rational then
kono
parents:
diff changeset
2659 Error_Msg_N ("rational expected", Expr);
kono
parents:
diff changeset
2660 end if;
kono
parents:
diff changeset
2661
kono
parents:
diff changeset
2662 return Result;
kono
parents:
diff changeset
2663 end Create_Rational_From;
kono
parents:
diff changeset
2664
kono
parents:
diff changeset
2665 -------------------
kono
parents:
diff changeset
2666 -- Dimensions_Of --
kono
parents:
diff changeset
2667 -------------------
kono
parents:
diff changeset
2668
kono
parents:
diff changeset
2669 function Dimensions_Of (N : Node_Id) return Dimension_Type is
kono
parents:
diff changeset
2670 begin
kono
parents:
diff changeset
2671 return Dimension_Table.Get (N);
kono
parents:
diff changeset
2672 end Dimensions_Of;
kono
parents:
diff changeset
2673
kono
parents:
diff changeset
2674 -----------------------
kono
parents:
diff changeset
2675 -- Dimensions_Msg_Of --
kono
parents:
diff changeset
2676 -----------------------
kono
parents:
diff changeset
2677
kono
parents:
diff changeset
2678 function Dimensions_Msg_Of
kono
parents:
diff changeset
2679 (N : Node_Id;
kono
parents:
diff changeset
2680 Description_Needed : Boolean := False) return String
kono
parents:
diff changeset
2681 is
kono
parents:
diff changeset
2682 Dims_Of_N : constant Dimension_Type := Dimensions_Of (N);
kono
parents:
diff changeset
2683 Dimensions_Msg : Name_Id;
kono
parents:
diff changeset
2684 System : System_Type;
kono
parents:
diff changeset
2685
kono
parents:
diff changeset
2686 begin
kono
parents:
diff changeset
2687 -- Initialization of Name_Buffer
kono
parents:
diff changeset
2688
kono
parents:
diff changeset
2689 Name_Len := 0;
kono
parents:
diff changeset
2690
kono
parents:
diff changeset
2691 -- N is not dimensionless
kono
parents:
diff changeset
2692
kono
parents:
diff changeset
2693 if Exists (Dims_Of_N) then
kono
parents:
diff changeset
2694 System := System_Of (Base_Type (Etype (N)));
kono
parents:
diff changeset
2695
kono
parents:
diff changeset
2696 -- When Description_Needed, add to string "has dimension " before the
kono
parents:
diff changeset
2697 -- actual dimension.
kono
parents:
diff changeset
2698
kono
parents:
diff changeset
2699 if Description_Needed then
kono
parents:
diff changeset
2700 Add_Str_To_Name_Buffer ("has dimension ");
kono
parents:
diff changeset
2701 end if;
kono
parents:
diff changeset
2702
kono
parents:
diff changeset
2703 Append
kono
parents:
diff changeset
2704 (Global_Name_Buffer,
kono
parents:
diff changeset
2705 From_Dim_To_Str_Of_Dim_Symbols (Dims_Of_N, System, True));
kono
parents:
diff changeset
2706
kono
parents:
diff changeset
2707 -- N is dimensionless
kono
parents:
diff changeset
2708
kono
parents:
diff changeset
2709 -- When Description_Needed, return "is dimensionless"
kono
parents:
diff changeset
2710
kono
parents:
diff changeset
2711 elsif Description_Needed then
kono
parents:
diff changeset
2712 Add_Str_To_Name_Buffer ("is dimensionless");
kono
parents:
diff changeset
2713
kono
parents:
diff changeset
2714 -- Otherwise, return "'[']"
kono
parents:
diff changeset
2715
kono
parents:
diff changeset
2716 else
kono
parents:
diff changeset
2717 Add_Str_To_Name_Buffer ("'[']");
kono
parents:
diff changeset
2718 end if;
kono
parents:
diff changeset
2719
kono
parents:
diff changeset
2720 Dimensions_Msg := Name_Find;
kono
parents:
diff changeset
2721 return Get_Name_String (Dimensions_Msg);
kono
parents:
diff changeset
2722 end Dimensions_Msg_Of;
kono
parents:
diff changeset
2723
kono
parents:
diff changeset
2724 --------------------------
kono
parents:
diff changeset
2725 -- Dimension_Table_Hash --
kono
parents:
diff changeset
2726 --------------------------
kono
parents:
diff changeset
2727
kono
parents:
diff changeset
2728 function Dimension_Table_Hash
kono
parents:
diff changeset
2729 (Key : Node_Id) return Dimension_Table_Range
kono
parents:
diff changeset
2730 is
kono
parents:
diff changeset
2731 begin
kono
parents:
diff changeset
2732 return Dimension_Table_Range (Key mod 511);
kono
parents:
diff changeset
2733 end Dimension_Table_Hash;
kono
parents:
diff changeset
2734
kono
parents:
diff changeset
2735 -------------------------------------
kono
parents:
diff changeset
2736 -- Dim_Warning_For_Numeric_Literal --
kono
parents:
diff changeset
2737 -------------------------------------
kono
parents:
diff changeset
2738
kono
parents:
diff changeset
2739 procedure Dim_Warning_For_Numeric_Literal (N : Node_Id; Typ : Entity_Id) is
kono
parents:
diff changeset
2740 begin
kono
parents:
diff changeset
2741 -- Consider the literal zero (integer 0 or real 0.0) to be of any
kono
parents:
diff changeset
2742 -- dimension.
kono
parents:
diff changeset
2743
kono
parents:
diff changeset
2744 case Nkind (Original_Node (N)) is
kono
parents:
diff changeset
2745 when N_Real_Literal =>
kono
parents:
diff changeset
2746 if Expr_Value_R (N) = Ureal_0 then
kono
parents:
diff changeset
2747 return;
kono
parents:
diff changeset
2748 end if;
kono
parents:
diff changeset
2749
kono
parents:
diff changeset
2750 when N_Integer_Literal =>
kono
parents:
diff changeset
2751 if Expr_Value (N) = Uint_0 then
kono
parents:
diff changeset
2752 return;
kono
parents:
diff changeset
2753 end if;
kono
parents:
diff changeset
2754
kono
parents:
diff changeset
2755 when others =>
kono
parents:
diff changeset
2756 null;
kono
parents:
diff changeset
2757 end case;
kono
parents:
diff changeset
2758
kono
parents:
diff changeset
2759 -- Initialize name buffer
kono
parents:
diff changeset
2760
kono
parents:
diff changeset
2761 Name_Len := 0;
kono
parents:
diff changeset
2762
kono
parents:
diff changeset
2763 Append (Global_Name_Buffer, String_From_Numeric_Literal (N));
kono
parents:
diff changeset
2764
kono
parents:
diff changeset
2765 -- Insert a blank between the literal and the symbol
kono
parents:
diff changeset
2766
kono
parents:
diff changeset
2767 Add_Str_To_Name_Buffer (" ");
kono
parents:
diff changeset
2768 Append (Global_Name_Buffer, Symbol_Of (Typ));
kono
parents:
diff changeset
2769
kono
parents:
diff changeset
2770 Error_Msg_Name_1 := Name_Find;
kono
parents:
diff changeset
2771 Error_Msg_N ("assumed to be%%??", N);
kono
parents:
diff changeset
2772 end Dim_Warning_For_Numeric_Literal;
kono
parents:
diff changeset
2773
kono
parents:
diff changeset
2774 ----------------------
kono
parents:
diff changeset
2775 -- Dimensions_Match --
kono
parents:
diff changeset
2776 ----------------------
kono
parents:
diff changeset
2777
kono
parents:
diff changeset
2778 function Dimensions_Match (T1 : Entity_Id; T2 : Entity_Id) return Boolean is
kono
parents:
diff changeset
2779 begin
kono
parents:
diff changeset
2780 return
kono
parents:
diff changeset
2781 not Has_Dimension_System (Base_Type (T1))
kono
parents:
diff changeset
2782 or else Dimensions_Of (T1) = Dimensions_Of (T2);
kono
parents:
diff changeset
2783 end Dimensions_Match;
kono
parents:
diff changeset
2784
kono
parents:
diff changeset
2785 ---------------------------
kono
parents:
diff changeset
2786 -- Dimension_System_Root --
kono
parents:
diff changeset
2787 ---------------------------
kono
parents:
diff changeset
2788
kono
parents:
diff changeset
2789 function Dimension_System_Root (T : Entity_Id) return Entity_Id is
kono
parents:
diff changeset
2790 Root : Entity_Id;
kono
parents:
diff changeset
2791
kono
parents:
diff changeset
2792 begin
kono
parents:
diff changeset
2793 Root := Base_Type (T);
kono
parents:
diff changeset
2794
kono
parents:
diff changeset
2795 if Has_Dimension_System (Root) then
kono
parents:
diff changeset
2796 return First_Subtype (Root); -- for example Dim_Mks
kono
parents:
diff changeset
2797
kono
parents:
diff changeset
2798 else
kono
parents:
diff changeset
2799 return Empty;
kono
parents:
diff changeset
2800 end if;
kono
parents:
diff changeset
2801 end Dimension_System_Root;
kono
parents:
diff changeset
2802
kono
parents:
diff changeset
2803 ----------------------------------------
kono
parents:
diff changeset
2804 -- Eval_Op_Expon_For_Dimensioned_Type --
kono
parents:
diff changeset
2805 ----------------------------------------
kono
parents:
diff changeset
2806
kono
parents:
diff changeset
2807 -- Evaluate the expon operator for real dimensioned type.
kono
parents:
diff changeset
2808
kono
parents:
diff changeset
2809 -- Note that if the exponent is an integer (denominator = 1) the node is
kono
parents:
diff changeset
2810 -- evaluated by the regular Eval_Op_Expon routine (see Sem_Eval).
kono
parents:
diff changeset
2811
kono
parents:
diff changeset
2812 procedure Eval_Op_Expon_For_Dimensioned_Type
kono
parents:
diff changeset
2813 (N : Node_Id;
kono
parents:
diff changeset
2814 Btyp : Entity_Id)
kono
parents:
diff changeset
2815 is
kono
parents:
diff changeset
2816 R : constant Node_Id := Right_Opnd (N);
kono
parents:
diff changeset
2817 R_Value : Rational := No_Rational;
kono
parents:
diff changeset
2818
kono
parents:
diff changeset
2819 begin
kono
parents:
diff changeset
2820 if Is_Real_Type (Btyp) then
kono
parents:
diff changeset
2821 R_Value := Create_Rational_From (R, False);
kono
parents:
diff changeset
2822 end if;
kono
parents:
diff changeset
2823
kono
parents:
diff changeset
2824 -- Check that the exponent is not an integer
kono
parents:
diff changeset
2825
kono
parents:
diff changeset
2826 if R_Value /= No_Rational and then R_Value.Denominator /= 1 then
kono
parents:
diff changeset
2827 Eval_Op_Expon_With_Rational_Exponent (N, R_Value);
kono
parents:
diff changeset
2828 else
kono
parents:
diff changeset
2829 Eval_Op_Expon (N);
kono
parents:
diff changeset
2830 end if;
kono
parents:
diff changeset
2831 end Eval_Op_Expon_For_Dimensioned_Type;
kono
parents:
diff changeset
2832
kono
parents:
diff changeset
2833 ------------------------------------------
kono
parents:
diff changeset
2834 -- Eval_Op_Expon_With_Rational_Exponent --
kono
parents:
diff changeset
2835 ------------------------------------------
kono
parents:
diff changeset
2836
kono
parents:
diff changeset
2837 -- For dimensioned operand in exponentiation, exponent is allowed to be a
kono
parents:
diff changeset
2838 -- Rational and not only an Integer like for dimensionless operands. For
kono
parents:
diff changeset
2839 -- that particular case, the left operand is rewritten as a function call
kono
parents:
diff changeset
2840 -- using the function Expon_LLF from s-llflex.ads.
kono
parents:
diff changeset
2841
kono
parents:
diff changeset
2842 procedure Eval_Op_Expon_With_Rational_Exponent
kono
parents:
diff changeset
2843 (N : Node_Id;
kono
parents:
diff changeset
2844 Exponent_Value : Rational)
kono
parents:
diff changeset
2845 is
kono
parents:
diff changeset
2846 Loc : constant Source_Ptr := Sloc (N);
kono
parents:
diff changeset
2847 Dims_Of_N : constant Dimension_Type := Dimensions_Of (N);
kono
parents:
diff changeset
2848 L : constant Node_Id := Left_Opnd (N);
kono
parents:
diff changeset
2849 Etyp_Of_L : constant Entity_Id := Etype (L);
kono
parents:
diff changeset
2850 Btyp_Of_L : constant Entity_Id := Base_Type (Etyp_Of_L);
kono
parents:
diff changeset
2851 Actual_1 : Node_Id;
kono
parents:
diff changeset
2852 Actual_2 : Node_Id;
kono
parents:
diff changeset
2853 Dim_Power : Rational;
kono
parents:
diff changeset
2854 List_Of_Dims : List_Id;
kono
parents:
diff changeset
2855 New_Aspect : Node_Id;
kono
parents:
diff changeset
2856 New_Aspects : List_Id;
kono
parents:
diff changeset
2857 New_Id : Entity_Id;
kono
parents:
diff changeset
2858 New_N : Node_Id;
kono
parents:
diff changeset
2859 New_Subtyp_Decl_For_L : Node_Id;
kono
parents:
diff changeset
2860 System : System_Type;
kono
parents:
diff changeset
2861
kono
parents:
diff changeset
2862 begin
kono
parents:
diff changeset
2863 -- Case when the operand is not dimensionless
kono
parents:
diff changeset
2864
kono
parents:
diff changeset
2865 if Exists (Dims_Of_N) then
kono
parents:
diff changeset
2866
kono
parents:
diff changeset
2867 -- Get the corresponding System_Type to know the exact number of
kono
parents:
diff changeset
2868 -- dimensions in the system.
kono
parents:
diff changeset
2869
kono
parents:
diff changeset
2870 System := System_Of (Btyp_Of_L);
kono
parents:
diff changeset
2871
kono
parents:
diff changeset
2872 -- Generation of a new subtype with the proper dimensions
kono
parents:
diff changeset
2873
kono
parents:
diff changeset
2874 -- In order to rewrite the operator as a type conversion, a new
kono
parents:
diff changeset
2875 -- dimensioned subtype with the resulting dimensions of the
kono
parents:
diff changeset
2876 -- exponentiation must be created.
kono
parents:
diff changeset
2877
kono
parents:
diff changeset
2878 -- Generate:
kono
parents:
diff changeset
2879
kono
parents:
diff changeset
2880 -- Btyp_Of_L : constant Entity_Id := Base_Type (Etyp_Of_L);
kono
parents:
diff changeset
2881 -- System : constant System_Id :=
kono
parents:
diff changeset
2882 -- Get_Dimension_System_Id (Btyp_Of_L);
kono
parents:
diff changeset
2883 -- Num_Of_Dims : constant Number_Of_Dimensions :=
kono
parents:
diff changeset
2884 -- Dimension_Systems.Table (System).Dimension_Count;
kono
parents:
diff changeset
2885
kono
parents:
diff changeset
2886 -- subtype T is Btyp_Of_L
kono
parents:
diff changeset
2887 -- with
kono
parents:
diff changeset
2888 -- Dimension => (
kono
parents:
diff changeset
2889 -- Dims_Of_N (1).Numerator / Dims_Of_N (1).Denominator,
kono
parents:
diff changeset
2890 -- Dims_Of_N (2).Numerator / Dims_Of_N (2).Denominator,
kono
parents:
diff changeset
2891 -- ...
kono
parents:
diff changeset
2892 -- Dims_Of_N (Num_Of_Dims).Numerator /
kono
parents:
diff changeset
2893 -- Dims_Of_N (Num_Of_Dims).Denominator);
kono
parents:
diff changeset
2894
kono
parents:
diff changeset
2895 -- Step 1: Generate the new aggregate for the aspect Dimension
kono
parents:
diff changeset
2896
kono
parents:
diff changeset
2897 New_Aspects := Empty_List;
kono
parents:
diff changeset
2898
kono
parents:
diff changeset
2899 List_Of_Dims := New_List;
kono
parents:
diff changeset
2900 for Position in Dims_Of_N'First .. System.Count loop
kono
parents:
diff changeset
2901 Dim_Power := Dims_Of_N (Position);
kono
parents:
diff changeset
2902 Append_To (List_Of_Dims,
kono
parents:
diff changeset
2903 Make_Op_Divide (Loc,
kono
parents:
diff changeset
2904 Left_Opnd =>
kono
parents:
diff changeset
2905 Make_Integer_Literal (Loc, Int (Dim_Power.Numerator)),
kono
parents:
diff changeset
2906 Right_Opnd =>
kono
parents:
diff changeset
2907 Make_Integer_Literal (Loc, Int (Dim_Power.Denominator))));
kono
parents:
diff changeset
2908 end loop;
kono
parents:
diff changeset
2909
kono
parents:
diff changeset
2910 -- Step 2: Create the new Aspect Specification for Aspect Dimension
kono
parents:
diff changeset
2911
kono
parents:
diff changeset
2912 New_Aspect :=
kono
parents:
diff changeset
2913 Make_Aspect_Specification (Loc,
kono
parents:
diff changeset
2914 Identifier => Make_Identifier (Loc, Name_Dimension),
kono
parents:
diff changeset
2915 Expression => Make_Aggregate (Loc, Expressions => List_Of_Dims));
kono
parents:
diff changeset
2916
kono
parents:
diff changeset
2917 -- Step 3: Make a temporary identifier for the new subtype
kono
parents:
diff changeset
2918
kono
parents:
diff changeset
2919 New_Id := Make_Temporary (Loc, 'T');
kono
parents:
diff changeset
2920 Set_Is_Internal (New_Id);
kono
parents:
diff changeset
2921
kono
parents:
diff changeset
2922 -- Step 4: Declaration of the new subtype
kono
parents:
diff changeset
2923
kono
parents:
diff changeset
2924 New_Subtyp_Decl_For_L :=
kono
parents:
diff changeset
2925 Make_Subtype_Declaration (Loc,
kono
parents:
diff changeset
2926 Defining_Identifier => New_Id,
kono
parents:
diff changeset
2927 Subtype_Indication => New_Occurrence_Of (Btyp_Of_L, Loc));
kono
parents:
diff changeset
2928
kono
parents:
diff changeset
2929 Append (New_Aspect, New_Aspects);
kono
parents:
diff changeset
2930 Set_Parent (New_Aspects, New_Subtyp_Decl_For_L);
kono
parents:
diff changeset
2931 Set_Aspect_Specifications (New_Subtyp_Decl_For_L, New_Aspects);
kono
parents:
diff changeset
2932
kono
parents:
diff changeset
2933 Analyze (New_Subtyp_Decl_For_L);
kono
parents:
diff changeset
2934
kono
parents:
diff changeset
2935 -- Case where the operand is dimensionless
kono
parents:
diff changeset
2936
kono
parents:
diff changeset
2937 else
kono
parents:
diff changeset
2938 New_Id := Btyp_Of_L;
kono
parents:
diff changeset
2939 end if;
kono
parents:
diff changeset
2940
kono
parents:
diff changeset
2941 -- Replacement of N by New_N
kono
parents:
diff changeset
2942
kono
parents:
diff changeset
2943 -- Generate:
kono
parents:
diff changeset
2944
kono
parents:
diff changeset
2945 -- Actual_1 := Long_Long_Float (L),
kono
parents:
diff changeset
2946
kono
parents:
diff changeset
2947 -- Actual_2 := Long_Long_Float (Exponent_Value.Numerator) /
kono
parents:
diff changeset
2948 -- Long_Long_Float (Exponent_Value.Denominator);
kono
parents:
diff changeset
2949
kono
parents:
diff changeset
2950 -- (T (Expon_LLF (Actual_1, Actual_2)));
kono
parents:
diff changeset
2951
kono
parents:
diff changeset
2952 -- where T is the subtype declared in step 1
kono
parents:
diff changeset
2953
kono
parents:
diff changeset
2954 -- The node is rewritten as a type conversion
kono
parents:
diff changeset
2955
kono
parents:
diff changeset
2956 -- Step 1: Creation of the two parameters of Expon_LLF function call
kono
parents:
diff changeset
2957
kono
parents:
diff changeset
2958 Actual_1 :=
kono
parents:
diff changeset
2959 Make_Type_Conversion (Loc,
kono
parents:
diff changeset
2960 Subtype_Mark => New_Occurrence_Of (Standard_Long_Long_Float, Loc),
kono
parents:
diff changeset
2961 Expression => Relocate_Node (L));
kono
parents:
diff changeset
2962
kono
parents:
diff changeset
2963 Actual_2 :=
kono
parents:
diff changeset
2964 Make_Op_Divide (Loc,
kono
parents:
diff changeset
2965 Left_Opnd =>
kono
parents:
diff changeset
2966 Make_Real_Literal (Loc,
kono
parents:
diff changeset
2967 UR_From_Uint (UI_From_Int (Int (Exponent_Value.Numerator)))),
kono
parents:
diff changeset
2968 Right_Opnd =>
kono
parents:
diff changeset
2969 Make_Real_Literal (Loc,
kono
parents:
diff changeset
2970 UR_From_Uint (UI_From_Int (Int (Exponent_Value.Denominator)))));
kono
parents:
diff changeset
2971
kono
parents:
diff changeset
2972 -- Step 2: Creation of New_N
kono
parents:
diff changeset
2973
kono
parents:
diff changeset
2974 New_N :=
kono
parents:
diff changeset
2975 Make_Type_Conversion (Loc,
kono
parents:
diff changeset
2976 Subtype_Mark => New_Occurrence_Of (New_Id, Loc),
kono
parents:
diff changeset
2977 Expression =>
kono
parents:
diff changeset
2978 Make_Function_Call (Loc,
kono
parents:
diff changeset
2979 Name => New_Occurrence_Of (RTE (RE_Expon_LLF), Loc),
kono
parents:
diff changeset
2980 Parameter_Associations => New_List (
kono
parents:
diff changeset
2981 Actual_1, Actual_2)));
kono
parents:
diff changeset
2982
kono
parents:
diff changeset
2983 -- Step 3: Rewrite N with the result
kono
parents:
diff changeset
2984
kono
parents:
diff changeset
2985 Rewrite (N, New_N);
kono
parents:
diff changeset
2986 Set_Etype (N, New_Id);
kono
parents:
diff changeset
2987 Analyze_And_Resolve (N, New_Id);
kono
parents:
diff changeset
2988 end Eval_Op_Expon_With_Rational_Exponent;
kono
parents:
diff changeset
2989
kono
parents:
diff changeset
2990 ------------
kono
parents:
diff changeset
2991 -- Exists --
kono
parents:
diff changeset
2992 ------------
kono
parents:
diff changeset
2993
kono
parents:
diff changeset
2994 function Exists (Dim : Dimension_Type) return Boolean is
kono
parents:
diff changeset
2995 begin
kono
parents:
diff changeset
2996 return Dim /= Null_Dimension;
kono
parents:
diff changeset
2997 end Exists;
kono
parents:
diff changeset
2998
kono
parents:
diff changeset
2999 function Exists (Str : String_Id) return Boolean is
kono
parents:
diff changeset
3000 begin
kono
parents:
diff changeset
3001 return Str /= No_String;
kono
parents:
diff changeset
3002 end Exists;
kono
parents:
diff changeset
3003
kono
parents:
diff changeset
3004 function Exists (Sys : System_Type) return Boolean is
kono
parents:
diff changeset
3005 begin
kono
parents:
diff changeset
3006 return Sys /= Null_System;
kono
parents:
diff changeset
3007 end Exists;
kono
parents:
diff changeset
3008
kono
parents:
diff changeset
3009 ---------------------------------
kono
parents:
diff changeset
3010 -- Expand_Put_Call_With_Symbol --
kono
parents:
diff changeset
3011 ---------------------------------
kono
parents:
diff changeset
3012
kono
parents:
diff changeset
3013 -- For procedure Put (resp. Put_Dim_Of) and function Image, defined in
kono
parents:
diff changeset
3014 -- System.Dim.Float_IO or System.Dim.Integer_IO, the default string
kono
parents:
diff changeset
3015 -- parameter is rewritten to include the unit symbol (or the dimension
kono
parents:
diff changeset
3016 -- symbols if not a defined quantity) in the output of a dimensioned
kono
parents:
diff changeset
3017 -- object. If a value is already supplied by the user for the parameter
kono
parents:
diff changeset
3018 -- Symbol, it is used as is.
kono
parents:
diff changeset
3019
kono
parents:
diff changeset
3020 -- Case 1. Item is dimensionless
kono
parents:
diff changeset
3021
kono
parents:
diff changeset
3022 -- * Put : Item appears without a suffix
kono
parents:
diff changeset
3023
kono
parents:
diff changeset
3024 -- * Put_Dim_Of : the output is []
kono
parents:
diff changeset
3025
kono
parents:
diff changeset
3026 -- Obj : Mks_Type := 2.6;
kono
parents:
diff changeset
3027 -- Put (Obj, 1, 1, 0);
kono
parents:
diff changeset
3028 -- Put_Dim_Of (Obj);
kono
parents:
diff changeset
3029
kono
parents:
diff changeset
3030 -- The corresponding outputs are:
kono
parents:
diff changeset
3031 -- $2.6
kono
parents:
diff changeset
3032 -- $[]
kono
parents:
diff changeset
3033
kono
parents:
diff changeset
3034 -- Case 2. Item has a dimension
kono
parents:
diff changeset
3035
kono
parents:
diff changeset
3036 -- * Put : If the type of Item is a dimensioned subtype whose
kono
parents:
diff changeset
3037 -- symbol is not empty, then the symbol appears as a
kono
parents:
diff changeset
3038 -- suffix. Otherwise, a new string is created and appears
kono
parents:
diff changeset
3039 -- as a suffix of Item. This string results in the
kono
parents:
diff changeset
3040 -- successive concatanations between each unit symbol
kono
parents:
diff changeset
3041 -- raised by its corresponding dimension power from the
kono
parents:
diff changeset
3042 -- dimensions of Item.
kono
parents:
diff changeset
3043
kono
parents:
diff changeset
3044 -- * Put_Dim_Of : The output is a new string resulting in the successive
kono
parents:
diff changeset
3045 -- concatanations between each dimension symbol raised by
kono
parents:
diff changeset
3046 -- its corresponding dimension power from the dimensions of
kono
parents:
diff changeset
3047 -- Item.
kono
parents:
diff changeset
3048
kono
parents:
diff changeset
3049 -- subtype Random is Mks_Type
kono
parents:
diff changeset
3050 -- with
kono
parents:
diff changeset
3051 -- Dimension => (
kono
parents:
diff changeset
3052 -- Meter => 3,
kono
parents:
diff changeset
3053 -- Candela => -1,
kono
parents:
diff changeset
3054 -- others => 0);
kono
parents:
diff changeset
3055
kono
parents:
diff changeset
3056 -- Obj : Random := 5.0;
kono
parents:
diff changeset
3057 -- Put (Obj);
kono
parents:
diff changeset
3058 -- Put_Dim_Of (Obj);
kono
parents:
diff changeset
3059
kono
parents:
diff changeset
3060 -- The corresponding outputs are:
kono
parents:
diff changeset
3061 -- $5.0 m**3.cd**(-1)
kono
parents:
diff changeset
3062 -- $[l**3.J**(-1)]
kono
parents:
diff changeset
3063
kono
parents:
diff changeset
3064 -- The function Image returns the string identical to that produced by
kono
parents:
diff changeset
3065 -- a call to Put whose first parameter is a string.
kono
parents:
diff changeset
3066
kono
parents:
diff changeset
3067 procedure Expand_Put_Call_With_Symbol (N : Node_Id) is
kono
parents:
diff changeset
3068 Actuals : constant List_Id := Parameter_Associations (N);
kono
parents:
diff changeset
3069 Loc : constant Source_Ptr := Sloc (N);
kono
parents:
diff changeset
3070 Name_Call : constant Node_Id := Name (N);
kono
parents:
diff changeset
3071 New_Actuals : constant List_Id := New_List;
kono
parents:
diff changeset
3072 Actual : Node_Id;
kono
parents:
diff changeset
3073 Dims_Of_Actual : Dimension_Type;
kono
parents:
diff changeset
3074 Etyp : Entity_Id;
kono
parents:
diff changeset
3075 New_Str_Lit : Node_Id := Empty;
kono
parents:
diff changeset
3076 Symbols : String_Id;
kono
parents:
diff changeset
3077
kono
parents:
diff changeset
3078 Is_Put_Dim_Of : Boolean := False;
kono
parents:
diff changeset
3079 -- This flag is used in order to differentiate routines Put and
kono
parents:
diff changeset
3080 -- Put_Dim_Of. Set to True if the procedure is one of the Put_Dim_Of
kono
parents:
diff changeset
3081 -- defined in System.Dim.Float_IO or System.Dim.Integer_IO.
kono
parents:
diff changeset
3082
kono
parents:
diff changeset
3083 function Has_Symbols return Boolean;
kono
parents:
diff changeset
3084 -- Return True if the current Put call already has a parameter
kono
parents:
diff changeset
3085 -- association for parameter "Symbols" with the correct string of
kono
parents:
diff changeset
3086 -- symbols.
kono
parents:
diff changeset
3087
kono
parents:
diff changeset
3088 function Is_Procedure_Put_Call return Boolean;
kono
parents:
diff changeset
3089 -- Return True if the current call is a call of an instantiation of a
kono
parents:
diff changeset
3090 -- procedure Put defined in the package System.Dim.Float_IO and
kono
parents:
diff changeset
3091 -- System.Dim.Integer_IO.
kono
parents:
diff changeset
3092
kono
parents:
diff changeset
3093 function Item_Actual return Node_Id;
kono
parents:
diff changeset
3094 -- Return the item actual parameter node in the output call
kono
parents:
diff changeset
3095
kono
parents:
diff changeset
3096 -----------------
kono
parents:
diff changeset
3097 -- Has_Symbols --
kono
parents:
diff changeset
3098 -----------------
kono
parents:
diff changeset
3099
kono
parents:
diff changeset
3100 function Has_Symbols return Boolean is
kono
parents:
diff changeset
3101 Actual : Node_Id;
kono
parents:
diff changeset
3102 Actual_Str : Node_Id;
kono
parents:
diff changeset
3103
kono
parents:
diff changeset
3104 begin
kono
parents:
diff changeset
3105 -- Look for a symbols parameter association in the list of actuals
kono
parents:
diff changeset
3106
kono
parents:
diff changeset
3107 Actual := First (Actuals);
kono
parents:
diff changeset
3108 while Present (Actual) loop
kono
parents:
diff changeset
3109
kono
parents:
diff changeset
3110 -- Positional parameter association case when the actual is a
kono
parents:
diff changeset
3111 -- string literal.
kono
parents:
diff changeset
3112
kono
parents:
diff changeset
3113 if Nkind (Actual) = N_String_Literal then
kono
parents:
diff changeset
3114 Actual_Str := Actual;
kono
parents:
diff changeset
3115
kono
parents:
diff changeset
3116 -- Named parameter association case when selector name is Symbol
kono
parents:
diff changeset
3117
kono
parents:
diff changeset
3118 elsif Nkind (Actual) = N_Parameter_Association
kono
parents:
diff changeset
3119 and then Chars (Selector_Name (Actual)) = Name_Symbol
kono
parents:
diff changeset
3120 then
kono
parents:
diff changeset
3121 Actual_Str := Explicit_Actual_Parameter (Actual);
kono
parents:
diff changeset
3122
kono
parents:
diff changeset
3123 -- Ignore all other cases
kono
parents:
diff changeset
3124
kono
parents:
diff changeset
3125 else
kono
parents:
diff changeset
3126 Actual_Str := Empty;
kono
parents:
diff changeset
3127 end if;
kono
parents:
diff changeset
3128
kono
parents:
diff changeset
3129 if Present (Actual_Str) then
kono
parents:
diff changeset
3130
kono
parents:
diff changeset
3131 -- Return True if the actual comes from source or if the string
kono
parents:
diff changeset
3132 -- of symbols doesn't have the default value (i.e. it is ""),
kono
parents:
diff changeset
3133 -- in which case it is used as suffix of the generated string.
kono
parents:
diff changeset
3134
kono
parents:
diff changeset
3135 if Comes_From_Source (Actual)
kono
parents:
diff changeset
3136 or else String_Length (Strval (Actual_Str)) /= 0
kono
parents:
diff changeset
3137 then
kono
parents:
diff changeset
3138 return True;
kono
parents:
diff changeset
3139
kono
parents:
diff changeset
3140 else
kono
parents:
diff changeset
3141 return False;
kono
parents:
diff changeset
3142 end if;
kono
parents:
diff changeset
3143 end if;
kono
parents:
diff changeset
3144
kono
parents:
diff changeset
3145 Next (Actual);
kono
parents:
diff changeset
3146 end loop;
kono
parents:
diff changeset
3147
kono
parents:
diff changeset
3148 -- At this point, the call has no parameter association. Look to the
kono
parents:
diff changeset
3149 -- last actual since the symbols parameter is the last one.
kono
parents:
diff changeset
3150
kono
parents:
diff changeset
3151 return Nkind (Last (Actuals)) = N_String_Literal;
kono
parents:
diff changeset
3152 end Has_Symbols;
kono
parents:
diff changeset
3153
kono
parents:
diff changeset
3154 ---------------------------
kono
parents:
diff changeset
3155 -- Is_Procedure_Put_Call --
kono
parents:
diff changeset
3156 ---------------------------
kono
parents:
diff changeset
3157
kono
parents:
diff changeset
3158 function Is_Procedure_Put_Call return Boolean is
kono
parents:
diff changeset
3159 Ent : Entity_Id;
kono
parents:
diff changeset
3160 Loc : Source_Ptr;
kono
parents:
diff changeset
3161
kono
parents:
diff changeset
3162 begin
kono
parents:
diff changeset
3163 -- There are three different Put (resp. Put_Dim_Of) routines in each
kono
parents:
diff changeset
3164 -- generic dim IO package. Verify the current procedure call is one
kono
parents:
diff changeset
3165 -- of them.
kono
parents:
diff changeset
3166
kono
parents:
diff changeset
3167 if Is_Entity_Name (Name_Call) then
kono
parents:
diff changeset
3168 Ent := Entity (Name_Call);
kono
parents:
diff changeset
3169
kono
parents:
diff changeset
3170 -- Get the original subprogram entity following the renaming chain
kono
parents:
diff changeset
3171
kono
parents:
diff changeset
3172 if Present (Alias (Ent)) then
kono
parents:
diff changeset
3173 Ent := Alias (Ent);
kono
parents:
diff changeset
3174 end if;
kono
parents:
diff changeset
3175
kono
parents:
diff changeset
3176 Loc := Sloc (Ent);
kono
parents:
diff changeset
3177
kono
parents:
diff changeset
3178 -- Check the name of the entity subprogram is Put (resp.
kono
parents:
diff changeset
3179 -- Put_Dim_Of) and verify this entity is located in either
kono
parents:
diff changeset
3180 -- System.Dim.Float_IO or System.Dim.Integer_IO.
kono
parents:
diff changeset
3181
kono
parents:
diff changeset
3182 if Loc > No_Location
kono
parents:
diff changeset
3183 and then Is_Dim_IO_Package_Entity
kono
parents:
diff changeset
3184 (Cunit_Entity (Get_Source_Unit (Loc)))
kono
parents:
diff changeset
3185 then
kono
parents:
diff changeset
3186 if Chars (Ent) = Name_Put_Dim_Of then
kono
parents:
diff changeset
3187 Is_Put_Dim_Of := True;
kono
parents:
diff changeset
3188 return True;
kono
parents:
diff changeset
3189
kono
parents:
diff changeset
3190 elsif Chars (Ent) = Name_Put
kono
parents:
diff changeset
3191 or else Chars (Ent) = Name_Image
kono
parents:
diff changeset
3192 then
kono
parents:
diff changeset
3193 return True;
kono
parents:
diff changeset
3194 end if;
kono
parents:
diff changeset
3195 end if;
kono
parents:
diff changeset
3196 end if;
kono
parents:
diff changeset
3197
kono
parents:
diff changeset
3198 return False;
kono
parents:
diff changeset
3199 end Is_Procedure_Put_Call;
kono
parents:
diff changeset
3200
kono
parents:
diff changeset
3201 -----------------
kono
parents:
diff changeset
3202 -- Item_Actual --
kono
parents:
diff changeset
3203 -----------------
kono
parents:
diff changeset
3204
kono
parents:
diff changeset
3205 function Item_Actual return Node_Id is
kono
parents:
diff changeset
3206 Actual : Node_Id;
kono
parents:
diff changeset
3207
kono
parents:
diff changeset
3208 begin
kono
parents:
diff changeset
3209 -- Look for the item actual as a parameter association
kono
parents:
diff changeset
3210
kono
parents:
diff changeset
3211 Actual := First (Actuals);
kono
parents:
diff changeset
3212 while Present (Actual) loop
kono
parents:
diff changeset
3213 if Nkind (Actual) = N_Parameter_Association
kono
parents:
diff changeset
3214 and then Chars (Selector_Name (Actual)) = Name_Item
kono
parents:
diff changeset
3215 then
kono
parents:
diff changeset
3216 return Explicit_Actual_Parameter (Actual);
kono
parents:
diff changeset
3217 end if;
kono
parents:
diff changeset
3218
kono
parents:
diff changeset
3219 Next (Actual);
kono
parents:
diff changeset
3220 end loop;
kono
parents:
diff changeset
3221
kono
parents:
diff changeset
3222 -- Case where the item has been defined without an association
kono
parents:
diff changeset
3223
kono
parents:
diff changeset
3224 Actual := First (Actuals);
kono
parents:
diff changeset
3225
kono
parents:
diff changeset
3226 -- Depending on the procedure Put, Item actual could be first or
kono
parents:
diff changeset
3227 -- second in the list of actuals.
kono
parents:
diff changeset
3228
kono
parents:
diff changeset
3229 if Has_Dimension_System (Base_Type (Etype (Actual))) then
kono
parents:
diff changeset
3230 return Actual;
kono
parents:
diff changeset
3231 else
kono
parents:
diff changeset
3232 return Next (Actual);
kono
parents:
diff changeset
3233 end if;
kono
parents:
diff changeset
3234 end Item_Actual;
kono
parents:
diff changeset
3235
kono
parents:
diff changeset
3236 -- Start of processing for Expand_Put_Call_With_Symbol
kono
parents:
diff changeset
3237
kono
parents:
diff changeset
3238 begin
kono
parents:
diff changeset
3239 if Is_Procedure_Put_Call and then not Has_Symbols then
kono
parents:
diff changeset
3240 Actual := Item_Actual;
kono
parents:
diff changeset
3241 Dims_Of_Actual := Dimensions_Of (Actual);
kono
parents:
diff changeset
3242 Etyp := Etype (Actual);
kono
parents:
diff changeset
3243
kono
parents:
diff changeset
3244 -- Put_Dim_Of case
kono
parents:
diff changeset
3245
kono
parents:
diff changeset
3246 if Is_Put_Dim_Of then
kono
parents:
diff changeset
3247
kono
parents:
diff changeset
3248 -- Check that the item is not dimensionless
kono
parents:
diff changeset
3249
kono
parents:
diff changeset
3250 -- Create the new String_Literal with the new String_Id generated
kono
parents:
diff changeset
3251 -- by the routine From_Dim_To_Str_Of_Dim_Symbols.
kono
parents:
diff changeset
3252
kono
parents:
diff changeset
3253 if Exists (Dims_Of_Actual) then
kono
parents:
diff changeset
3254 New_Str_Lit :=
kono
parents:
diff changeset
3255 Make_String_Literal (Loc,
kono
parents:
diff changeset
3256 From_Dim_To_Str_Of_Dim_Symbols
kono
parents:
diff changeset
3257 (Dims_Of_Actual, System_Of (Base_Type (Etyp))));
kono
parents:
diff changeset
3258
kono
parents:
diff changeset
3259 -- If dimensionless, the output is []
kono
parents:
diff changeset
3260
kono
parents:
diff changeset
3261 else
kono
parents:
diff changeset
3262 New_Str_Lit :=
kono
parents:
diff changeset
3263 Make_String_Literal (Loc, "[]");
kono
parents:
diff changeset
3264 end if;
kono
parents:
diff changeset
3265
kono
parents:
diff changeset
3266 -- Put case
kono
parents:
diff changeset
3267
kono
parents:
diff changeset
3268 else
kono
parents:
diff changeset
3269 -- Add the symbol as a suffix of the value if the subtype has a
kono
parents:
diff changeset
3270 -- unit symbol or if the parameter is not dimensionless.
kono
parents:
diff changeset
3271
kono
parents:
diff changeset
3272 if Exists (Symbol_Of (Etyp)) then
kono
parents:
diff changeset
3273 Symbols := Symbol_Of (Etyp);
kono
parents:
diff changeset
3274 else
kono
parents:
diff changeset
3275 Symbols := From_Dim_To_Str_Of_Unit_Symbols
kono
parents:
diff changeset
3276 (Dims_Of_Actual, System_Of (Base_Type (Etyp)));
kono
parents:
diff changeset
3277 end if;
kono
parents:
diff changeset
3278
kono
parents:
diff changeset
3279 -- Check Symbols exists
kono
parents:
diff changeset
3280
kono
parents:
diff changeset
3281 if Exists (Symbols) then
kono
parents:
diff changeset
3282 Start_String;
kono
parents:
diff changeset
3283
kono
parents:
diff changeset
3284 -- Put a space between the value and the dimension
kono
parents:
diff changeset
3285
kono
parents:
diff changeset
3286 Store_String_Char (' ');
kono
parents:
diff changeset
3287 Store_String_Chars (Symbols);
kono
parents:
diff changeset
3288 New_Str_Lit := Make_String_Literal (Loc, End_String);
kono
parents:
diff changeset
3289 end if;
kono
parents:
diff changeset
3290 end if;
kono
parents:
diff changeset
3291
kono
parents:
diff changeset
3292 if Present (New_Str_Lit) then
kono
parents:
diff changeset
3293
kono
parents:
diff changeset
3294 -- Insert all actuals in New_Actuals
kono
parents:
diff changeset
3295
kono
parents:
diff changeset
3296 Actual := First (Actuals);
kono
parents:
diff changeset
3297 while Present (Actual) loop
kono
parents:
diff changeset
3298
kono
parents:
diff changeset
3299 -- Copy every actuals in New_Actuals except the Symbols
kono
parents:
diff changeset
3300 -- parameter association.
kono
parents:
diff changeset
3301
kono
parents:
diff changeset
3302 if Nkind (Actual) = N_Parameter_Association
kono
parents:
diff changeset
3303 and then Chars (Selector_Name (Actual)) /= Name_Symbol
kono
parents:
diff changeset
3304 then
kono
parents:
diff changeset
3305 Append_To (New_Actuals,
kono
parents:
diff changeset
3306 Make_Parameter_Association (Loc,
kono
parents:
diff changeset
3307 Selector_Name => New_Copy (Selector_Name (Actual)),
kono
parents:
diff changeset
3308 Explicit_Actual_Parameter =>
kono
parents:
diff changeset
3309 New_Copy (Explicit_Actual_Parameter (Actual))));
kono
parents:
diff changeset
3310
kono
parents:
diff changeset
3311 elsif Nkind (Actual) /= N_Parameter_Association then
kono
parents:
diff changeset
3312 Append_To (New_Actuals, New_Copy (Actual));
kono
parents:
diff changeset
3313 end if;
kono
parents:
diff changeset
3314
kono
parents:
diff changeset
3315 Next (Actual);
kono
parents:
diff changeset
3316 end loop;
kono
parents:
diff changeset
3317
kono
parents:
diff changeset
3318 -- Create new Symbols param association and append to New_Actuals
kono
parents:
diff changeset
3319
kono
parents:
diff changeset
3320 Append_To (New_Actuals,
kono
parents:
diff changeset
3321 Make_Parameter_Association (Loc,
kono
parents:
diff changeset
3322 Selector_Name => Make_Identifier (Loc, Name_Symbol),
kono
parents:
diff changeset
3323 Explicit_Actual_Parameter => New_Str_Lit));
kono
parents:
diff changeset
3324
kono
parents:
diff changeset
3325 -- Rewrite and analyze the procedure call
kono
parents:
diff changeset
3326
kono
parents:
diff changeset
3327 if Chars (Name_Call) = Name_Image then
kono
parents:
diff changeset
3328 Rewrite (N,
kono
parents:
diff changeset
3329 Make_Function_Call (Loc,
kono
parents:
diff changeset
3330 Name => New_Copy (Name_Call),
kono
parents:
diff changeset
3331 Parameter_Associations => New_Actuals));
kono
parents:
diff changeset
3332 Analyze_And_Resolve (N);
kono
parents:
diff changeset
3333 else
kono
parents:
diff changeset
3334 Rewrite (N,
kono
parents:
diff changeset
3335 Make_Procedure_Call_Statement (Loc,
kono
parents:
diff changeset
3336 Name => New_Copy (Name_Call),
kono
parents:
diff changeset
3337 Parameter_Associations => New_Actuals));
kono
parents:
diff changeset
3338 Analyze (N);
kono
parents:
diff changeset
3339 end if;
kono
parents:
diff changeset
3340
kono
parents:
diff changeset
3341 end if;
kono
parents:
diff changeset
3342 end if;
kono
parents:
diff changeset
3343 end Expand_Put_Call_With_Symbol;
kono
parents:
diff changeset
3344
kono
parents:
diff changeset
3345 ------------------------------------
kono
parents:
diff changeset
3346 -- From_Dim_To_Str_Of_Dim_Symbols --
kono
parents:
diff changeset
3347 ------------------------------------
kono
parents:
diff changeset
3348
kono
parents:
diff changeset
3349 -- Given a dimension vector and the corresponding dimension system, create
kono
parents:
diff changeset
3350 -- a String_Id to output dimension symbols corresponding to the dimensions
kono
parents:
diff changeset
3351 -- Dims. If In_Error_Msg is True, there is a special handling for character
kono
parents:
diff changeset
3352 -- asterisk * which is an insertion character in error messages.
kono
parents:
diff changeset
3353
kono
parents:
diff changeset
3354 function From_Dim_To_Str_Of_Dim_Symbols
kono
parents:
diff changeset
3355 (Dims : Dimension_Type;
kono
parents:
diff changeset
3356 System : System_Type;
kono
parents:
diff changeset
3357 In_Error_Msg : Boolean := False) return String_Id
kono
parents:
diff changeset
3358 is
kono
parents:
diff changeset
3359 Dim_Power : Rational;
kono
parents:
diff changeset
3360 First_Dim : Boolean := True;
kono
parents:
diff changeset
3361
kono
parents:
diff changeset
3362 procedure Store_String_Oexpon;
kono
parents:
diff changeset
3363 -- Store the expon operator symbol "**" in the string. In error
kono
parents:
diff changeset
3364 -- messages, asterisk * is a special character and must be quoted
kono
parents:
diff changeset
3365 -- to be placed literally into the message.
kono
parents:
diff changeset
3366
kono
parents:
diff changeset
3367 -------------------------
kono
parents:
diff changeset
3368 -- Store_String_Oexpon --
kono
parents:
diff changeset
3369 -------------------------
kono
parents:
diff changeset
3370
kono
parents:
diff changeset
3371 procedure Store_String_Oexpon is
kono
parents:
diff changeset
3372 begin
kono
parents:
diff changeset
3373 if In_Error_Msg then
kono
parents:
diff changeset
3374 Store_String_Chars ("'*'*");
kono
parents:
diff changeset
3375 else
kono
parents:
diff changeset
3376 Store_String_Chars ("**");
kono
parents:
diff changeset
3377 end if;
kono
parents:
diff changeset
3378 end Store_String_Oexpon;
kono
parents:
diff changeset
3379
kono
parents:
diff changeset
3380 -- Start of processing for From_Dim_To_Str_Of_Dim_Symbols
kono
parents:
diff changeset
3381
kono
parents:
diff changeset
3382 begin
kono
parents:
diff changeset
3383 -- Initialization of the new String_Id
kono
parents:
diff changeset
3384
kono
parents:
diff changeset
3385 Start_String;
kono
parents:
diff changeset
3386
kono
parents:
diff changeset
3387 -- Store the dimension symbols inside boxes
kono
parents:
diff changeset
3388
kono
parents:
diff changeset
3389 if In_Error_Msg then
kono
parents:
diff changeset
3390 Store_String_Chars ("'[");
kono
parents:
diff changeset
3391 else
kono
parents:
diff changeset
3392 Store_String_Char ('[');
kono
parents:
diff changeset
3393 end if;
kono
parents:
diff changeset
3394
kono
parents:
diff changeset
3395 for Position in Dimension_Type'Range loop
kono
parents:
diff changeset
3396 Dim_Power := Dims (Position);
kono
parents:
diff changeset
3397 if Dim_Power /= Zero then
kono
parents:
diff changeset
3398
kono
parents:
diff changeset
3399 if First_Dim then
kono
parents:
diff changeset
3400 First_Dim := False;
kono
parents:
diff changeset
3401 else
kono
parents:
diff changeset
3402 Store_String_Char ('.');
kono
parents:
diff changeset
3403 end if;
kono
parents:
diff changeset
3404
kono
parents:
diff changeset
3405 Store_String_Chars (System.Dim_Symbols (Position));
kono
parents:
diff changeset
3406
kono
parents:
diff changeset
3407 -- Positive dimension case
kono
parents:
diff changeset
3408
kono
parents:
diff changeset
3409 if Dim_Power.Numerator > 0 then
kono
parents:
diff changeset
3410
kono
parents:
diff changeset
3411 -- Integer case
kono
parents:
diff changeset
3412
kono
parents:
diff changeset
3413 if Dim_Power.Denominator = 1 then
kono
parents:
diff changeset
3414 if Dim_Power.Numerator /= 1 then
kono
parents:
diff changeset
3415 Store_String_Oexpon;
kono
parents:
diff changeset
3416 Store_String_Int (Int (Dim_Power.Numerator));
kono
parents:
diff changeset
3417 end if;
kono
parents:
diff changeset
3418
kono
parents:
diff changeset
3419 -- Rational case when denominator /= 1
kono
parents:
diff changeset
3420
kono
parents:
diff changeset
3421 else
kono
parents:
diff changeset
3422 Store_String_Oexpon;
kono
parents:
diff changeset
3423 Store_String_Char ('(');
kono
parents:
diff changeset
3424 Store_String_Int (Int (Dim_Power.Numerator));
kono
parents:
diff changeset
3425 Store_String_Char ('/');
kono
parents:
diff changeset
3426 Store_String_Int (Int (Dim_Power.Denominator));
kono
parents:
diff changeset
3427 Store_String_Char (')');
kono
parents:
diff changeset
3428 end if;
kono
parents:
diff changeset
3429
kono
parents:
diff changeset
3430 -- Negative dimension case
kono
parents:
diff changeset
3431
kono
parents:
diff changeset
3432 else
kono
parents:
diff changeset
3433 Store_String_Oexpon;
kono
parents:
diff changeset
3434 Store_String_Char ('(');
kono
parents:
diff changeset
3435 Store_String_Char ('-');
kono
parents:
diff changeset
3436 Store_String_Int (Int (-Dim_Power.Numerator));
kono
parents:
diff changeset
3437
kono
parents:
diff changeset
3438 -- Integer case
kono
parents:
diff changeset
3439
kono
parents:
diff changeset
3440 if Dim_Power.Denominator = 1 then
kono
parents:
diff changeset
3441 Store_String_Char (')');
kono
parents:
diff changeset
3442
kono
parents:
diff changeset
3443 -- Rational case when denominator /= 1
kono
parents:
diff changeset
3444
kono
parents:
diff changeset
3445 else
kono
parents:
diff changeset
3446 Store_String_Char ('/');
kono
parents:
diff changeset
3447 Store_String_Int (Int (Dim_Power.Denominator));
kono
parents:
diff changeset
3448 Store_String_Char (')');
kono
parents:
diff changeset
3449 end if;
kono
parents:
diff changeset
3450 end if;
kono
parents:
diff changeset
3451 end if;
kono
parents:
diff changeset
3452 end loop;
kono
parents:
diff changeset
3453
kono
parents:
diff changeset
3454 if In_Error_Msg then
kono
parents:
diff changeset
3455 Store_String_Chars ("']");
kono
parents:
diff changeset
3456 else
kono
parents:
diff changeset
3457 Store_String_Char (']');
kono
parents:
diff changeset
3458 end if;
kono
parents:
diff changeset
3459
kono
parents:
diff changeset
3460 return End_String;
kono
parents:
diff changeset
3461 end From_Dim_To_Str_Of_Dim_Symbols;
kono
parents:
diff changeset
3462
kono
parents:
diff changeset
3463 -------------------------------------
kono
parents:
diff changeset
3464 -- From_Dim_To_Str_Of_Unit_Symbols --
kono
parents:
diff changeset
3465 -------------------------------------
kono
parents:
diff changeset
3466
kono
parents:
diff changeset
3467 -- Given a dimension vector and the corresponding dimension system,
kono
parents:
diff changeset
3468 -- create a String_Id to output the unit symbols corresponding to the
kono
parents:
diff changeset
3469 -- dimensions Dims.
kono
parents:
diff changeset
3470
kono
parents:
diff changeset
3471 function From_Dim_To_Str_Of_Unit_Symbols
kono
parents:
diff changeset
3472 (Dims : Dimension_Type;
kono
parents:
diff changeset
3473 System : System_Type) return String_Id
kono
parents:
diff changeset
3474 is
kono
parents:
diff changeset
3475 Dim_Power : Rational;
kono
parents:
diff changeset
3476 First_Dim : Boolean := True;
kono
parents:
diff changeset
3477
kono
parents:
diff changeset
3478 begin
kono
parents:
diff changeset
3479 -- Return No_String if dimensionless
kono
parents:
diff changeset
3480
kono
parents:
diff changeset
3481 if not Exists (Dims) then
kono
parents:
diff changeset
3482 return No_String;
kono
parents:
diff changeset
3483 end if;
kono
parents:
diff changeset
3484
kono
parents:
diff changeset
3485 -- Initialization of the new String_Id
kono
parents:
diff changeset
3486
kono
parents:
diff changeset
3487 Start_String;
kono
parents:
diff changeset
3488
kono
parents:
diff changeset
3489 for Position in Dimension_Type'Range loop
kono
parents:
diff changeset
3490 Dim_Power := Dims (Position);
kono
parents:
diff changeset
3491
kono
parents:
diff changeset
3492 if Dim_Power /= Zero then
kono
parents:
diff changeset
3493 if First_Dim then
kono
parents:
diff changeset
3494 First_Dim := False;
kono
parents:
diff changeset
3495 else
kono
parents:
diff changeset
3496 Store_String_Char ('.');
kono
parents:
diff changeset
3497 end if;
kono
parents:
diff changeset
3498
kono
parents:
diff changeset
3499 Store_String_Chars (System.Unit_Symbols (Position));
kono
parents:
diff changeset
3500
kono
parents:
diff changeset
3501 -- Positive dimension case
kono
parents:
diff changeset
3502
kono
parents:
diff changeset
3503 if Dim_Power.Numerator > 0 then
kono
parents:
diff changeset
3504
kono
parents:
diff changeset
3505 -- Integer case
kono
parents:
diff changeset
3506
kono
parents:
diff changeset
3507 if Dim_Power.Denominator = 1 then
kono
parents:
diff changeset
3508 if Dim_Power.Numerator /= 1 then
kono
parents:
diff changeset
3509 Store_String_Chars ("**");
kono
parents:
diff changeset
3510 Store_String_Int (Int (Dim_Power.Numerator));
kono
parents:
diff changeset
3511 end if;
kono
parents:
diff changeset
3512
kono
parents:
diff changeset
3513 -- Rational case when denominator /= 1
kono
parents:
diff changeset
3514
kono
parents:
diff changeset
3515 else
kono
parents:
diff changeset
3516 Store_String_Chars ("**");
kono
parents:
diff changeset
3517 Store_String_Char ('(');
kono
parents:
diff changeset
3518 Store_String_Int (Int (Dim_Power.Numerator));
kono
parents:
diff changeset
3519 Store_String_Char ('/');
kono
parents:
diff changeset
3520 Store_String_Int (Int (Dim_Power.Denominator));
kono
parents:
diff changeset
3521 Store_String_Char (')');
kono
parents:
diff changeset
3522 end if;
kono
parents:
diff changeset
3523
kono
parents:
diff changeset
3524 -- Negative dimension case
kono
parents:
diff changeset
3525
kono
parents:
diff changeset
3526 else
kono
parents:
diff changeset
3527 Store_String_Chars ("**");
kono
parents:
diff changeset
3528 Store_String_Char ('(');
kono
parents:
diff changeset
3529 Store_String_Char ('-');
kono
parents:
diff changeset
3530 Store_String_Int (Int (-Dim_Power.Numerator));
kono
parents:
diff changeset
3531
kono
parents:
diff changeset
3532 -- Integer case
kono
parents:
diff changeset
3533
kono
parents:
diff changeset
3534 if Dim_Power.Denominator = 1 then
kono
parents:
diff changeset
3535 Store_String_Char (')');
kono
parents:
diff changeset
3536
kono
parents:
diff changeset
3537 -- Rational case when denominator /= 1
kono
parents:
diff changeset
3538
kono
parents:
diff changeset
3539 else
kono
parents:
diff changeset
3540 Store_String_Char ('/');
kono
parents:
diff changeset
3541 Store_String_Int (Int (Dim_Power.Denominator));
kono
parents:
diff changeset
3542 Store_String_Char (')');
kono
parents:
diff changeset
3543 end if;
kono
parents:
diff changeset
3544 end if;
kono
parents:
diff changeset
3545 end if;
kono
parents:
diff changeset
3546 end loop;
kono
parents:
diff changeset
3547
kono
parents:
diff changeset
3548 return End_String;
kono
parents:
diff changeset
3549 end From_Dim_To_Str_Of_Unit_Symbols;
kono
parents:
diff changeset
3550
kono
parents:
diff changeset
3551 ---------
kono
parents:
diff changeset
3552 -- GCD --
kono
parents:
diff changeset
3553 ---------
kono
parents:
diff changeset
3554
kono
parents:
diff changeset
3555 function GCD (Left, Right : Whole) return Int is
kono
parents:
diff changeset
3556 L : Whole;
kono
parents:
diff changeset
3557 R : Whole;
kono
parents:
diff changeset
3558
kono
parents:
diff changeset
3559 begin
kono
parents:
diff changeset
3560 L := Left;
kono
parents:
diff changeset
3561 R := Right;
kono
parents:
diff changeset
3562 while R /= 0 loop
kono
parents:
diff changeset
3563 L := L mod R;
kono
parents:
diff changeset
3564
kono
parents:
diff changeset
3565 if L = 0 then
kono
parents:
diff changeset
3566 return Int (R);
kono
parents:
diff changeset
3567 end if;
kono
parents:
diff changeset
3568
kono
parents:
diff changeset
3569 R := R mod L;
kono
parents:
diff changeset
3570 end loop;
kono
parents:
diff changeset
3571
kono
parents:
diff changeset
3572 return Int (L);
kono
parents:
diff changeset
3573 end GCD;
kono
parents:
diff changeset
3574
kono
parents:
diff changeset
3575 --------------------------
kono
parents:
diff changeset
3576 -- Has_Dimension_System --
kono
parents:
diff changeset
3577 --------------------------
kono
parents:
diff changeset
3578
kono
parents:
diff changeset
3579 function Has_Dimension_System (Typ : Entity_Id) return Boolean is
kono
parents:
diff changeset
3580 begin
kono
parents:
diff changeset
3581 return Exists (System_Of (Typ));
kono
parents:
diff changeset
3582 end Has_Dimension_System;
kono
parents:
diff changeset
3583
kono
parents:
diff changeset
3584 ------------------------------
kono
parents:
diff changeset
3585 -- Is_Dim_IO_Package_Entity --
kono
parents:
diff changeset
3586 ------------------------------
kono
parents:
diff changeset
3587
kono
parents:
diff changeset
3588 function Is_Dim_IO_Package_Entity (E : Entity_Id) return Boolean is
kono
parents:
diff changeset
3589 begin
kono
parents:
diff changeset
3590 -- Check the package entity corresponds to System.Dim.Float_IO or
kono
parents:
diff changeset
3591 -- System.Dim.Integer_IO.
kono
parents:
diff changeset
3592
kono
parents:
diff changeset
3593 return
kono
parents:
diff changeset
3594 Is_RTU (E, System_Dim_Float_IO)
kono
parents:
diff changeset
3595 or else
kono
parents:
diff changeset
3596 Is_RTU (E, System_Dim_Integer_IO);
kono
parents:
diff changeset
3597 end Is_Dim_IO_Package_Entity;
kono
parents:
diff changeset
3598
kono
parents:
diff changeset
3599 -------------------------------------
kono
parents:
diff changeset
3600 -- Is_Dim_IO_Package_Instantiation --
kono
parents:
diff changeset
3601 -------------------------------------
kono
parents:
diff changeset
3602
kono
parents:
diff changeset
3603 function Is_Dim_IO_Package_Instantiation (N : Node_Id) return Boolean is
kono
parents:
diff changeset
3604 Gen_Id : constant Node_Id := Name (N);
kono
parents:
diff changeset
3605
kono
parents:
diff changeset
3606 begin
kono
parents:
diff changeset
3607 -- Check that the instantiated package is either System.Dim.Float_IO
kono
parents:
diff changeset
3608 -- or System.Dim.Integer_IO.
kono
parents:
diff changeset
3609
kono
parents:
diff changeset
3610 return
kono
parents:
diff changeset
3611 Is_Entity_Name (Gen_Id)
kono
parents:
diff changeset
3612 and then Is_Dim_IO_Package_Entity (Entity (Gen_Id));
kono
parents:
diff changeset
3613 end Is_Dim_IO_Package_Instantiation;
kono
parents:
diff changeset
3614
kono
parents:
diff changeset
3615 ----------------
kono
parents:
diff changeset
3616 -- Is_Invalid --
kono
parents:
diff changeset
3617 ----------------
kono
parents:
diff changeset
3618
kono
parents:
diff changeset
3619 function Is_Invalid (Position : Dimension_Position) return Boolean is
kono
parents:
diff changeset
3620 begin
kono
parents:
diff changeset
3621 return Position = Invalid_Position;
kono
parents:
diff changeset
3622 end Is_Invalid;
kono
parents:
diff changeset
3623
kono
parents:
diff changeset
3624 ---------------------
kono
parents:
diff changeset
3625 -- Move_Dimensions --
kono
parents:
diff changeset
3626 ---------------------
kono
parents:
diff changeset
3627
kono
parents:
diff changeset
3628 procedure Move_Dimensions (From, To : Node_Id) is
kono
parents:
diff changeset
3629 begin
kono
parents:
diff changeset
3630 if Ada_Version < Ada_2012 then
kono
parents:
diff changeset
3631 return;
kono
parents:
diff changeset
3632 end if;
kono
parents:
diff changeset
3633
kono
parents:
diff changeset
3634 -- Copy the dimension of 'From to 'To' and remove dimension of 'From'
kono
parents:
diff changeset
3635
kono
parents:
diff changeset
3636 Copy_Dimensions (From, To);
kono
parents:
diff changeset
3637 Remove_Dimensions (From);
kono
parents:
diff changeset
3638 end Move_Dimensions;
kono
parents:
diff changeset
3639
kono
parents:
diff changeset
3640 ---------------------------------------
kono
parents:
diff changeset
3641 -- New_Copy_Tree_And_Copy_Dimensions --
kono
parents:
diff changeset
3642 ---------------------------------------
kono
parents:
diff changeset
3643
kono
parents:
diff changeset
3644 function New_Copy_Tree_And_Copy_Dimensions
kono
parents:
diff changeset
3645 (Source : Node_Id;
kono
parents:
diff changeset
3646 Map : Elist_Id := No_Elist;
kono
parents:
diff changeset
3647 New_Sloc : Source_Ptr := No_Location;
kono
parents:
diff changeset
3648 New_Scope : Entity_Id := Empty) return Node_Id
kono
parents:
diff changeset
3649 is
kono
parents:
diff changeset
3650 New_Copy : constant Node_Id :=
kono
parents:
diff changeset
3651 New_Copy_Tree (Source, Map, New_Sloc, New_Scope);
kono
parents:
diff changeset
3652
kono
parents:
diff changeset
3653 begin
kono
parents:
diff changeset
3654 -- Move the dimensions of Source to New_Copy
kono
parents:
diff changeset
3655
kono
parents:
diff changeset
3656 Copy_Dimensions (Source, New_Copy);
kono
parents:
diff changeset
3657 return New_Copy;
kono
parents:
diff changeset
3658 end New_Copy_Tree_And_Copy_Dimensions;
kono
parents:
diff changeset
3659
kono
parents:
diff changeset
3660 ------------
kono
parents:
diff changeset
3661 -- Reduce --
kono
parents:
diff changeset
3662 ------------
kono
parents:
diff changeset
3663
kono
parents:
diff changeset
3664 function Reduce (X : Rational) return Rational is
kono
parents:
diff changeset
3665 begin
kono
parents:
diff changeset
3666 if X.Numerator = 0 then
kono
parents:
diff changeset
3667 return Zero;
kono
parents:
diff changeset
3668 end if;
kono
parents:
diff changeset
3669
kono
parents:
diff changeset
3670 declare
kono
parents:
diff changeset
3671 G : constant Int := GCD (X.Numerator, X.Denominator);
kono
parents:
diff changeset
3672 begin
kono
parents:
diff changeset
3673 return Rational'(Numerator => Whole (Int (X.Numerator) / G),
kono
parents:
diff changeset
3674 Denominator => Whole (Int (X.Denominator) / G));
kono
parents:
diff changeset
3675 end;
kono
parents:
diff changeset
3676 end Reduce;
kono
parents:
diff changeset
3677
kono
parents:
diff changeset
3678 -----------------------
kono
parents:
diff changeset
3679 -- Remove_Dimensions --
kono
parents:
diff changeset
3680 -----------------------
kono
parents:
diff changeset
3681
kono
parents:
diff changeset
3682 procedure Remove_Dimensions (N : Node_Id) is
kono
parents:
diff changeset
3683 Dims_Of_N : constant Dimension_Type := Dimensions_Of (N);
kono
parents:
diff changeset
3684 begin
kono
parents:
diff changeset
3685 if Exists (Dims_Of_N) then
kono
parents:
diff changeset
3686 Dimension_Table.Remove (N);
kono
parents:
diff changeset
3687 end if;
kono
parents:
diff changeset
3688 end Remove_Dimensions;
kono
parents:
diff changeset
3689
kono
parents:
diff changeset
3690 -----------------------------------
kono
parents:
diff changeset
3691 -- Remove_Dimension_In_Statement --
kono
parents:
diff changeset
3692 -----------------------------------
kono
parents:
diff changeset
3693
kono
parents:
diff changeset
3694 -- Removal of dimension in statement as part of the Analyze_Statements
kono
parents:
diff changeset
3695 -- routine (see package Sem_Ch5).
kono
parents:
diff changeset
3696
kono
parents:
diff changeset
3697 procedure Remove_Dimension_In_Statement (Stmt : Node_Id) is
kono
parents:
diff changeset
3698 begin
kono
parents:
diff changeset
3699 if Ada_Version < Ada_2012 then
kono
parents:
diff changeset
3700 return;
kono
parents:
diff changeset
3701 end if;
kono
parents:
diff changeset
3702
kono
parents:
diff changeset
3703 -- Remove dimension in parameter specifications for accept statement
kono
parents:
diff changeset
3704
kono
parents:
diff changeset
3705 if Nkind (Stmt) = N_Accept_Statement then
kono
parents:
diff changeset
3706 declare
kono
parents:
diff changeset
3707 Param : Node_Id := First (Parameter_Specifications (Stmt));
kono
parents:
diff changeset
3708 begin
kono
parents:
diff changeset
3709 while Present (Param) loop
kono
parents:
diff changeset
3710 Remove_Dimensions (Param);
kono
parents:
diff changeset
3711 Next (Param);
kono
parents:
diff changeset
3712 end loop;
kono
parents:
diff changeset
3713 end;
kono
parents:
diff changeset
3714
kono
parents:
diff changeset
3715 -- Remove dimension of name and expression in assignments
kono
parents:
diff changeset
3716
kono
parents:
diff changeset
3717 elsif Nkind (Stmt) = N_Assignment_Statement then
kono
parents:
diff changeset
3718 Remove_Dimensions (Expression (Stmt));
kono
parents:
diff changeset
3719 Remove_Dimensions (Name (Stmt));
kono
parents:
diff changeset
3720 end if;
kono
parents:
diff changeset
3721 end Remove_Dimension_In_Statement;
kono
parents:
diff changeset
3722
kono
parents:
diff changeset
3723 --------------------
kono
parents:
diff changeset
3724 -- Set_Dimensions --
kono
parents:
diff changeset
3725 --------------------
kono
parents:
diff changeset
3726
kono
parents:
diff changeset
3727 procedure Set_Dimensions (N : Node_Id; Val : Dimension_Type) is
kono
parents:
diff changeset
3728 begin
kono
parents:
diff changeset
3729 pragma Assert (OK_For_Dimension (Nkind (N)));
kono
parents:
diff changeset
3730 pragma Assert (Exists (Val));
kono
parents:
diff changeset
3731
kono
parents:
diff changeset
3732 Dimension_Table.Set (N, Val);
kono
parents:
diff changeset
3733 end Set_Dimensions;
kono
parents:
diff changeset
3734
kono
parents:
diff changeset
3735 ----------------
kono
parents:
diff changeset
3736 -- Set_Symbol --
kono
parents:
diff changeset
3737 ----------------
kono
parents:
diff changeset
3738
kono
parents:
diff changeset
3739 procedure Set_Symbol (E : Entity_Id; Val : String_Id) is
kono
parents:
diff changeset
3740 begin
kono
parents:
diff changeset
3741 Symbol_Table.Set (E, Val);
kono
parents:
diff changeset
3742 end Set_Symbol;
kono
parents:
diff changeset
3743
kono
parents:
diff changeset
3744 ---------------------------------
kono
parents:
diff changeset
3745 -- String_From_Numeric_Literal --
kono
parents:
diff changeset
3746 ---------------------------------
kono
parents:
diff changeset
3747
kono
parents:
diff changeset
3748 function String_From_Numeric_Literal (N : Node_Id) return String_Id is
kono
parents:
diff changeset
3749 Loc : constant Source_Ptr := Sloc (N);
kono
parents:
diff changeset
3750 Sbuffer : constant Source_Buffer_Ptr :=
kono
parents:
diff changeset
3751 Source_Text (Get_Source_File_Index (Loc));
kono
parents:
diff changeset
3752 Src_Ptr : Source_Ptr := Loc;
kono
parents:
diff changeset
3753
kono
parents:
diff changeset
3754 C : Character := Sbuffer (Src_Ptr);
kono
parents:
diff changeset
3755 -- Current source program character
kono
parents:
diff changeset
3756
kono
parents:
diff changeset
3757 function Belong_To_Numeric_Literal (C : Character) return Boolean;
kono
parents:
diff changeset
3758 -- Return True if C belongs to a numeric literal
kono
parents:
diff changeset
3759
kono
parents:
diff changeset
3760 -------------------------------
kono
parents:
diff changeset
3761 -- Belong_To_Numeric_Literal --
kono
parents:
diff changeset
3762 -------------------------------
kono
parents:
diff changeset
3763
kono
parents:
diff changeset
3764 function Belong_To_Numeric_Literal (C : Character) return Boolean is
kono
parents:
diff changeset
3765 begin
kono
parents:
diff changeset
3766 case C is
kono
parents:
diff changeset
3767 when '0' .. '9'
kono
parents:
diff changeset
3768 | '_' | '.' | 'e' | '#' | 'A' | 'B' | 'C' | 'D' | 'E' | 'F'
kono
parents:
diff changeset
3769 =>
kono
parents:
diff changeset
3770 return True;
kono
parents:
diff changeset
3771
kono
parents:
diff changeset
3772 -- Make sure '+' or '-' is part of an exponent.
kono
parents:
diff changeset
3773
kono
parents:
diff changeset
3774 when '+' | '-' =>
kono
parents:
diff changeset
3775 declare
kono
parents:
diff changeset
3776 Prev_C : constant Character := Sbuffer (Src_Ptr - 1);
kono
parents:
diff changeset
3777 begin
kono
parents:
diff changeset
3778 return Prev_C = 'e' or else Prev_C = 'E';
kono
parents:
diff changeset
3779 end;
kono
parents:
diff changeset
3780
kono
parents:
diff changeset
3781 -- All other character doesn't belong to a numeric literal
kono
parents:
diff changeset
3782
kono
parents:
diff changeset
3783 when others =>
kono
parents:
diff changeset
3784 return False;
kono
parents:
diff changeset
3785 end case;
kono
parents:
diff changeset
3786 end Belong_To_Numeric_Literal;
kono
parents:
diff changeset
3787
kono
parents:
diff changeset
3788 -- Start of processing for String_From_Numeric_Literal
kono
parents:
diff changeset
3789
kono
parents:
diff changeset
3790 begin
kono
parents:
diff changeset
3791 Start_String;
kono
parents:
diff changeset
3792 while Belong_To_Numeric_Literal (C) loop
kono
parents:
diff changeset
3793 Store_String_Char (C);
kono
parents:
diff changeset
3794 Src_Ptr := Src_Ptr + 1;
kono
parents:
diff changeset
3795 C := Sbuffer (Src_Ptr);
kono
parents:
diff changeset
3796 end loop;
kono
parents:
diff changeset
3797
kono
parents:
diff changeset
3798 return End_String;
kono
parents:
diff changeset
3799 end String_From_Numeric_Literal;
kono
parents:
diff changeset
3800
kono
parents:
diff changeset
3801 ---------------
kono
parents:
diff changeset
3802 -- Symbol_Of --
kono
parents:
diff changeset
3803 ---------------
kono
parents:
diff changeset
3804
kono
parents:
diff changeset
3805 function Symbol_Of (E : Entity_Id) return String_Id is
kono
parents:
diff changeset
3806 Subtype_Symbol : constant String_Id := Symbol_Table.Get (E);
kono
parents:
diff changeset
3807 begin
kono
parents:
diff changeset
3808 if Subtype_Symbol /= No_String then
kono
parents:
diff changeset
3809 return Subtype_Symbol;
kono
parents:
diff changeset
3810 else
kono
parents:
diff changeset
3811 return From_Dim_To_Str_Of_Unit_Symbols
kono
parents:
diff changeset
3812 (Dimensions_Of (E), System_Of (Base_Type (E)));
kono
parents:
diff changeset
3813 end if;
kono
parents:
diff changeset
3814 end Symbol_Of;
kono
parents:
diff changeset
3815
kono
parents:
diff changeset
3816 -----------------------
kono
parents:
diff changeset
3817 -- Symbol_Table_Hash --
kono
parents:
diff changeset
3818 -----------------------
kono
parents:
diff changeset
3819
kono
parents:
diff changeset
3820 function Symbol_Table_Hash (Key : Entity_Id) return Symbol_Table_Range is
kono
parents:
diff changeset
3821 begin
kono
parents:
diff changeset
3822 return Symbol_Table_Range (Key mod 511);
kono
parents:
diff changeset
3823 end Symbol_Table_Hash;
kono
parents:
diff changeset
3824
kono
parents:
diff changeset
3825 ---------------
kono
parents:
diff changeset
3826 -- System_Of --
kono
parents:
diff changeset
3827 ---------------
kono
parents:
diff changeset
3828
kono
parents:
diff changeset
3829 function System_Of (E : Entity_Id) return System_Type is
kono
parents:
diff changeset
3830 Type_Decl : constant Node_Id := Parent (E);
kono
parents:
diff changeset
3831
kono
parents:
diff changeset
3832 begin
kono
parents:
diff changeset
3833 -- Look for Type_Decl in System_Table
kono
parents:
diff changeset
3834
kono
parents:
diff changeset
3835 for Dim_Sys in 1 .. System_Table.Last loop
kono
parents:
diff changeset
3836 if Type_Decl = System_Table.Table (Dim_Sys).Type_Decl then
kono
parents:
diff changeset
3837 return System_Table.Table (Dim_Sys);
kono
parents:
diff changeset
3838 end if;
kono
parents:
diff changeset
3839 end loop;
kono
parents:
diff changeset
3840
kono
parents:
diff changeset
3841 return Null_System;
kono
parents:
diff changeset
3842 end System_Of;
kono
parents:
diff changeset
3843
kono
parents:
diff changeset
3844 end Sem_Dim;