1 ------------------------------------------------------------------------------
3 -- GNAT SYSTEM UTILITIES --
9 -- Copyright (C) 1992-2020, Free Software Foundation, Inc. --
11 -- GNAT is free software; you can redistribute it and/or modify it under --
12 -- terms of the GNU General Public License as published by the Free Soft- --
13 -- ware Foundation; either version 3, or (at your option) any later ver- --
14 -- sion. GNAT is distributed in the hope that it will be useful, but WITH- --
15 -- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY --
16 -- or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License --
17 -- for more details. You should have received a copy of the GNU General --
18 -- Public License distributed with GNAT; see file COPYING3. If not, go to --
19 -- http://www.gnu.org/licenses for a complete copy of the license. --
21 -- GNAT was originally developed by the GNAT team at New York University. --
22 -- Extensive contributions were provided by Ada Core Technologies Inc. --
24 ------------------------------------------------------------------------------
26 -- Program to construct C header file einfo.h (C version of einfo.ads spec)
27 -- for use by Gigi. This header file contains all definitions and access
28 -- functions, but does not contain set procedures, since Gigi is not allowed
29 -- to modify the GNAT tree.
33 -- einfo.ads spec of Einfo package
34 -- einfo.adb body of Einfo package
38 -- einfo.h corresponding C header file
40 -- Note: It is assumed that the input files have been compiled without errors
42 -- An optional argument allows the specification of an output file name to
43 -- override the default einfo.h file name for the generated output file.
45 -- Most, but not all of the functions in Einfo can be inlined in the C header.
46 -- They are the functions identified by pragma Inline in the spec. Functions
47 -- that cannot be inlined are simply defined in the header.
49 with Ada
.Command_Line
; use Ada
.Command_Line
;
50 with Ada
.Strings
.Unbounded
; use Ada
.Strings
.Unbounded
;
51 with Ada
.Strings
.Unbounded
.Text_IO
; use Ada
.Strings
.Unbounded
.Text_IO
;
52 with Ada
.Strings
.Maps
; use Ada
.Strings
.Maps
;
53 with Ada
.Strings
.Maps
.Constants
; use Ada
.Strings
.Maps
.Constants
;
54 with Ada
.Text_IO
; use Ada
.Text_IO
;
56 with GNAT
.Spitbol
; use GNAT
.Spitbol
;
57 with GNAT
.Spitbol
.Patterns
; use GNAT
.Spitbol
.Patterns
;
58 with GNAT
.Spitbol
.Table_Boolean
; use GNAT
.Spitbol
.Table_Boolean
;
64 package TB
renames GNAT
.Spitbol
.Table_Boolean
;
71 Expr
: VString
:= Nul
;
72 Filler
: VString
:= Nul
;
73 Fline
: VString
:= Nul
;
74 Formal
: VString
:= Nul
;
75 Formaltyp
: VString
:= Nul
;
77 Line
: VString
:= Nul
;
83 Name
: VString
:= Nul
;
84 NewS
: VString
:= Nul
;
85 Nextlin
: VString
:= Nul
;
86 OldS
: VString
:= Nul
;
88 Term
: VString
:= Nul
;
91 -- Used to read initial header from body
94 -- Used to read full text of both spec and body
97 -- Used to write output file
99 wsp
: constant Pattern
:= NSpan
(' ' & ASCII
.HT
);
100 Comment
: constant Pattern
:= wsp
& "--";
101 For_Rep
: constant Pattern
:= wsp
& "for";
102 Get_Func
: constant Pattern
:= wsp
* A
& "function" & wsp
103 & Break
(' ') * Name
;
104 Inline
: constant Pattern
:= wsp
& "pragma Inline (" & Break
(')') * Name
;
105 Get_Pack
: constant Pattern
:= wsp
& "package ";
106 Get_Enam
: constant Pattern
:= wsp
& Break
(',') * N
& ',';
107 Find_Fun
: constant Pattern
:= wsp
& "function";
108 F_Subtyp
: constant Pattern
:= wsp
* A
& "subtype " & Break
(' ') * N
;
109 G_Subtyp
: constant Pattern
:= wsp
& "subtype" & wsp
& Break
(' ') * NewS
110 & wsp
& "is" & wsp
& Break
(" ;") * OldS
111 & wsp
& ';' & wsp
& Rtab
(0);
112 F_Typ
: constant Pattern
:= wsp
* A
& "type " & Break
(' ') * N
&
114 Get_Nam
: constant Pattern
:= wsp
* A
& Break
(",)") * Nam
116 Get_Styp
: constant Pattern
:= wsp
* A
& "subtype " & Break
(' ') * N
;
117 Get_N1
: constant Pattern
:= wsp
& Break
(' ') * N1
;
118 Get_N2
: constant Pattern
:= wsp
& "-- " & Rest
* N2
;
119 Get_N3
: constant Pattern
:= wsp
& Break
(';') * N3
;
120 Get_FN
: constant Pattern
:= wsp
* C
& "function" & wsp
122 Is_Rturn
: constant Pattern
:= BreakX
('r') & "return";
123 Is_Begin
: constant Pattern
:= wsp
& "begin";
124 Get_Asrt
: constant Pattern
:= wsp
& "pragma Assert";
125 Semicoln
: constant Pattern
:= BreakX
(';');
126 Get_Cmnt
: constant Pattern
:= BreakX
('-') * A
& "--";
127 Get_Expr
: constant Pattern
:= wsp
& "return " & Break
(';') * Expr
;
128 Chek_End
: constant Pattern
:= wsp
& "end" & BreakX
(';') & ';';
129 Get_B0
: constant Pattern
:= BreakX
(' ') * A
& " or else " & Rest
* B
;
130 Get_B1
: constant Pattern
:= BreakX
(' ') * A
& " in " & Rest
* B
;
131 Get_B2
: constant Pattern
:= BreakX
(' ') * A
& " = " & Rest
* B
;
132 Get_B3
: constant Pattern
:= BreakX
(' ') * A
& " /= " & Rest
* B
;
133 To_Paren
: constant Pattern
:= wsp
* Filler
& '(';
134 Get_Fml
: constant Pattern
:= Break
(" :") * Formal
& wsp
& ':' & wsp
135 & BreakX
(" );") * Formaltyp
;
136 Nxt_Fml
: constant Pattern
:= wsp
& "; ";
137 Get_Rtn
: constant Pattern
:= wsp
& "return" & wsp
& BreakX
(" ;") * Rtn
;
138 Rem_Prn
: constant Pattern
:= wsp
& ')';
142 Lineno
: Natural := 0;
143 -- Line number in spec
148 Inlined
: TB
.Table
(200);
149 -- Inlined<N> = True for inlined function, False otherwise
151 Lastinlined
: Boolean;
154 pragma No_Return
(Badfunc
);
155 -- Signal bad function in body
157 function Getlin
return VString
;
158 -- Get non-comment line (comment lines skipped, also skips FOR rep clauses)
159 -- Fatal error (raises End_Error exception) if end of file encountered
161 procedure Must
(B
: Boolean);
162 -- Raises Err if the argument (a Match) call, returns False
164 procedure Sethead
(Line
: in out VString
; Term
: String);
165 -- Process function header into C
167 procedure Translate_Expr
(Expr
: in out VString
);
168 -- Translate expression from Ada to C
178 "Body for function " & FN
& " does not meet requirements");
186 function Getlin
return VString
is
191 Lin
:= Get_Line
(InF
);
192 Lineno
:= Lineno
+ 1;
195 and then not Match
(Lin
, Comment
)
196 and then not Match
(Lin
, For_Rep
)
207 procedure Must
(B
: Boolean) is
218 procedure Sethead
(Line
: in out VString
; Term
: String) is
222 Must
(Match
(Line
, Get_Func
, ""));
225 if Match
(Line
, To_Paren
, "") then
226 Args
:= Filler
& '(';
229 Must
(Match
(Line
, Get_Fml
, ""));
230 Append
(Args
, Formaltyp
& ' ' & Formal
);
231 exit when not Match
(Line
, Nxt_Fml
);
235 Match
(Line
, Rem_Prn
, "");
239 Must
(Match
(Line
, Get_Rtn
));
241 if Present
(Inlined
, Name
) then
242 Put_Line
(Ofile
, A
& "INLINE " & Rtn
& ' ' & Name
& Args
& Term
);
244 Put_Line
(Ofile
, A
& Rtn
& ' ' & Name
& Args
& Term
);
252 procedure Translate_Expr
(Expr
: in out VString
) is
256 Match
(Expr
, Get_B1
, M
);
257 Replace
(M
, "IN (" & A
& ", " & B
& ')');
258 Match
(Expr
, Get_B2
, M
);
259 Replace
(M
, A
& " == " & B
);
260 Match
(Expr
, Get_B3
, M
);
261 Replace
(M
, A
& " != " & B
);
264 -- Start of processing for XEinfo
267 -- First run CEinfo to check for errors. Note that CEinfo is also a
268 -- stand-alone program that can be run separately.
272 Anchored_Mode
:= True;
274 if Argument_Count
> 0 then
275 Create
(Ofile
, Out_File
, Argument
(1));
277 Create
(Ofile
, Out_File
, "einfo.h");
280 Open
(InB
, In_File
, "einfo.adb");
281 Open
(InF
, In_File
, "einfo.ads");
285 Line
:= Get_Line
(InF
);
286 Lineno
:= Lineno
+ 1;
291 "-- C Header File ");
292 Match
(Line
, "--", "/*");
293 Match
(Line
, Rtab
(2) * A
& "--", M
);
294 Replace
(M
, A
& "*/");
295 Put_Line
(Ofile
, Line
);
298 Put_Line
(Ofile
, "");
300 Put_Line
(Ofile
, "#ifdef __cplusplus");
301 Put_Line
(Ofile
, "extern ""C"" {");
302 Put_Line
(Ofile
, "#endif");
304 -- Find and record pragma Inlines
307 Line
:= Get_Line
(InF
);
308 exit when Match
(Line
, " -- END XEINFO INLINES");
310 if Match
(Line
, Inline
) then
311 Set
(Inlined
, Name
, True);
315 -- Skip to package line
317 Reset
(InF
, In_File
);
322 exit when Match
(Line
, Get_Pack
);
327 Must
(Match
(Line
, wsp
& "type Entity_Kind"));
329 -- Process entity kind code definitions
333 exit when not Match
(Line
, Get_Enam
);
334 Put_Line
(Ofile
, " #define " & Rpad
(N
, 32) & " " & V
);
338 Must
(Match
(Line
, wsp
& Rest
* N
));
339 Put_Line
(Ofile
, " #define " & Rpad
(N
, 32) & ' ' & V
);
342 Must
(Match
(Line
, wsp
& ");"));
343 Put_Line
(Ofile
, "");
345 -- Loop through subtype and type declarations
349 exit when Match
(Line
, Find_Fun
);
351 -- Case of a subtype declaration
353 if Match
(Line
, F_Subtyp
) then
355 -- Case of a subtype declaration that is an abbreviation of the
356 -- form subtype x is y, and if so generate the appropriate typedef
358 if Match
(Line
, G_Subtyp
) then
359 Put_Line
(Ofile
, A
& "typedef " & OldS
& ' ' & NewS
& ';');
361 -- Otherwise the subtype must be declaring a subrange of Entity_Id
364 Must
(Match
(Line
, Get_Styp
));
366 Must
(Match
(Line
, Get_N1
));
369 Line
:= Get_Line
(InF
);
370 Lineno
:= Lineno
+ 1;
371 exit when not Match
(Line
, Get_N2
);
374 Must
(Match
(Line
, Get_N3
));
375 Put_Line
(Ofile
, A
& "SUBTYPE (" & N
& ", Entity_Kind, ");
376 Put_Line
(Ofile
, A
& " " & N1
& ", " & N3
& ')');
377 Put_Line
(Ofile
, "");
380 -- Case of type declaration
382 elsif Match
(Line
, F_Typ
) then
384 -- Process type declaration (must be enumeration type)
387 Put_Line
(Ofile
, A
& "typedef char " & N
& ';');
391 Must
(Match
(Line
, Get_Nam
));
392 Put_Line
(Ofile
, A
& "#define " & Rpad
(Nam
, 25) & Ctr
);
394 exit when Term
/= ",";
397 Put_Line
(Ofile
, "");
399 -- Neither subtype nor type declaration
406 -- Process function declarations
408 -- Note: Lastinlined used to control blank lines
410 Put_Line
(Ofile
, "");
413 -- Loop through function declarations
415 while Match
(Line
, Get_FN
) loop
417 -- Non-inlined function
419 if not Present
(Inlined
, FN
) then
420 Put_Line
(Ofile
, "");
423 " #define " & FN
& " einfo__" & Translate
(FN
, Lower_Case_Map
));
428 if not Lastinlined
then
429 Put_Line
(Ofile
, "");
433 -- Merge here to output spec
436 Lastinlined
:= Get
(Inlined
, FN
);
440 Put_Line
(Ofile
, "");
442 -- Read body to find inlined functions
446 Open
(InF
, In_File
, "einfo.adb");
449 -- Loop through input lines to find bodies of inlined functions
451 while not End_Of_File
(InF
) loop
452 Fline
:= Get_Line
(InF
);
454 if Match
(Fline
, Get_FN
)
455 and then Get
(Inlined
, FN
)
457 -- Here we have an inlined function
459 if not Match
(Fline
, Is_Rturn
) then
466 if not Match
(Line
, Is_Begin
) then
470 -- Skip past pragma Asserts
474 exit when not Match
(Line
, Get_Asrt
);
476 -- Pragma assert found, get its continuation lines
479 exit when Match
(Line
, Semicoln
);
484 -- Process return statement
486 Match
(Line
, Get_Cmnt
, M
);
489 -- Get continuations of return statement
491 while not Match
(Line
, Semicoln
) loop
493 Match
(Nextlin
, wsp
, " ");
494 Append
(Line
, Nextlin
);
497 if not Match
(Line
, Get_Expr
) then
503 if not Match
(Line
, Chek_End
) then
507 -- Process expression
509 if Match
(Expr
, Get_B0
, M
) then
511 Saved_A
: VString
:= A
;
512 Saved_B
: VString
:= B
;
514 Translate_Expr
(Saved_A
);
515 Translate_Expr
(Saved_B
);
516 Replace
(M
, Saved_A
& " || " & Saved_B
);
519 Translate_Expr
(Expr
);
522 Put_Line
(Ofile
, "");
524 Put_Line
(Ofile
, C
& " { return " & Expr
& "; }");
528 Put_Line
(Ofile
, "");
530 Put_Line
(Ofile
, "#ifdef __cplusplus");
531 Put_Line
(Ofile
, "}");
532 Put_Line
(Ofile
, "#endif");
536 "/* End of einfo.h (C version of Einfo package specification) */");
543 Put_Line
(Standard_Error
, Lineno
& ". " & Line
);
544 Put_Line
(Standard_Error
, "**** fatal error ****");
548 Put_Line
(Standard_Error
, "unexpected end of file");
549 Put_Line
(Standard_Error
, "**** fatal error ****");