2008-05-30 Vladimir Makarov <vmakarov@redhat.com>
[official-gcc.git] / gcc / ada / mlib-utl.adb
blobd743bb138e8bbfb26545b67821d5faed291c557c
1 ------------------------------------------------------------------------------
2 -- --
3 -- GNAT COMPILER COMPONENTS --
4 -- --
5 -- M L I B . U T L --
6 -- --
7 -- B o d y --
8 -- --
9 -- Copyright (C) 2002-2008, AdaCore --
10 -- --
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. --
20 -- --
21 -- GNAT was originally developed by the GNAT team at New York University. --
22 -- Extensive contributions were provided by Ada Core Technologies Inc. --
23 -- --
24 ------------------------------------------------------------------------------
26 with MLib.Fil; use MLib.Fil;
27 with MLib.Tgt; use MLib.Tgt;
28 with Opt;
29 with Osint;
30 with Output; use Output;
32 with Interfaces.C.Strings; use Interfaces.C.Strings;
34 with System;
36 package body MLib.Utl is
38 Gcc_Name : String_Access;
39 -- Default value of the "gcc" executable used in procedure Gcc
41 Gcc_Exec : String_Access;
42 -- The full path name of the "gcc" executable
44 Ar_Name : String_Access;
45 -- The name of the archive builder for the platform, set when procedure Ar
46 -- is called for the first time.
48 Ar_Exec : String_Access;
49 -- The full path name of the archive builder
51 Ar_Options : String_List_Access;
52 -- The minimum options used when invoking the archive builder
54 Ar_Append_Options : String_List_Access;
55 -- The options to be used when invoking the archive builder to add chunks
56 -- of object files, when building the archive in chunks.
58 Opt_Length : Natural := 0;
59 -- The max number of options for the Archive_Builder
61 Initial_Size : Natural := 0;
62 -- The minimum number of bytes for the invocation of the Archive Builder
63 -- (without name of the archive or object files).
65 Ranlib_Name : String_Access;
66 -- The name of the archive indexer for the platform, if there is one
68 Ranlib_Exec : String_Access := null;
69 -- The full path name of the archive indexer
71 Ranlib_Options : String_List_Access := null;
72 -- The options to be used when invoking the archive indexer, if any
74 --------
75 -- Ar --
76 --------
78 procedure Ar (Output_File : String; Objects : Argument_List) is
79 Full_Output_File : constant String :=
80 Ext_To (Output_File, Archive_Ext);
82 Arguments : Argument_List_Access;
83 Last_Arg : Natural := 0;
84 Success : Boolean;
85 Line_Length : Natural := 0;
87 Maximum_Size : Integer;
88 pragma Import (C, Maximum_Size, "__gnat_link_max");
89 -- Maximum number of bytes to put in an invocation of the
90 -- Archive_Builder.
92 Size : Integer;
93 -- The number of bytes for the invocation of the archive builder
95 Current_Object : Natural;
97 procedure Display;
98 -- Display an invocation of the Archive Builder
100 -------------
101 -- Display --
102 -------------
104 procedure Display is
105 begin
106 if not Opt.Quiet_Output then
107 Write_Str (Ar_Name.all);
108 Line_Length := Ar_Name'Length;
110 for J in 1 .. Last_Arg loop
112 -- Make sure the Output buffer does not overflow
114 if Line_Length + 1 + Arguments (J)'Length > Buffer_Max then
115 Write_Eol;
116 Line_Length := 0;
117 end if;
119 Write_Char (' ');
121 -- Only output the first object files when not in verbose mode
123 if (not Opt.Verbose_Mode) and then J = Opt_Length + 3 then
124 Write_Str ("...");
125 exit;
126 end if;
128 Write_Str (Arguments (J).all);
129 Line_Length := Line_Length + 1 + Arguments (J)'Length;
130 end loop;
132 Write_Eol;
133 end if;
135 end Display;
137 begin
138 if Ar_Exec = null then
139 Ar_Name := Osint.Program_Name (Archive_Builder, "gnatmake");
140 Ar_Exec := Locate_Exec_On_Path (Ar_Name.all);
142 if Ar_Exec = null then
143 Free (Ar_Name);
144 Ar_Name := new String'(Archive_Builder);
145 Ar_Exec := Locate_Exec_On_Path (Ar_Name.all);
146 end if;
148 if Ar_Exec = null then
149 Fail (Ar_Name.all, " not found in path");
151 elsif Opt.Verbose_Mode then
152 Write_Str ("found ");
153 Write_Line (Ar_Exec.all);
154 end if;
156 Ar_Options := Archive_Builder_Options;
158 Initial_Size := 0;
159 for J in Ar_Options'Range loop
160 Initial_Size := Initial_Size + Ar_Options (J)'Length + 1;
161 end loop;
163 Ar_Append_Options := Archive_Builder_Append_Options;
165 Opt_Length := Ar_Options'Length;
167 if Ar_Append_Options /= null then
168 Opt_Length := Natural'Max (Ar_Append_Options'Length, Opt_Length);
170 Size := 0;
171 for J in Ar_Append_Options'Range loop
172 Size := Size + Ar_Append_Options (J)'Length + 1;
173 end loop;
175 Initial_Size := Integer'Max (Initial_Size, Size);
176 end if;
178 -- ranlib
180 Ranlib_Name := Osint.Program_Name (Archive_Indexer, "gnatmake");
182 if Ranlib_Name'Length > 0 then
183 Ranlib_Exec := Locate_Exec_On_Path (Ranlib_Name.all);
185 if Ranlib_Exec = null then
186 Free (Ranlib_Name);
187 Ranlib_Name := new String'(Archive_Indexer);
188 Ranlib_Exec := Locate_Exec_On_Path (Ranlib_Name.all);
189 end if;
191 if Ranlib_Exec /= null and then Opt.Verbose_Mode then
192 Write_Str ("found ");
193 Write_Line (Ranlib_Exec.all);
194 end if;
195 end if;
197 Ranlib_Options := Archive_Indexer_Options;
198 end if;
200 Arguments :=
201 new String_List (1 .. 1 + Opt_Length + Objects'Length);
202 Arguments (1 .. Ar_Options'Length) := Ar_Options.all; -- "ar cr ..."
203 Arguments (Ar_Options'Length + 1) := new String'(Full_Output_File);
205 Delete_File (Full_Output_File);
207 Size := Initial_Size + Full_Output_File'Length + 1;
209 -- Check the full size of a call of the archive builder with all the
210 -- object files.
212 for J in Objects'Range loop
213 Size := Size + Objects (J)'Length + 1;
214 end loop;
216 -- If the size is not too large or if it is not possible to build the
217 -- archive in chunks, build the archive in a single invocation.
219 if Size <= Maximum_Size or else Ar_Append_Options = null then
220 Last_Arg := Ar_Options'Length + 1 + Objects'Length;
221 Arguments (Ar_Options'Length + 2 .. Last_Arg) := Objects;
223 Display;
225 Spawn (Ar_Exec.all, Arguments (1 .. Last_Arg), Success);
227 else
228 -- Build the archive in several invocation, making sure to not
229 -- go over the maximum size for each invocation.
231 Last_Arg := Ar_Options'Length + 1;
232 Current_Object := Objects'First;
233 Size := Initial_Size + Full_Output_File'Length + 1;
235 -- First invocation
237 while Current_Object <= Objects'Last loop
238 Size := Size + Objects (Current_Object)'Length + 1;
239 exit when Size > Maximum_Size;
240 Last_Arg := Last_Arg + 1;
241 Arguments (Last_Arg) := Objects (Current_Object);
242 Current_Object := Current_Object + 1;
243 end loop;
245 Display;
247 Spawn (Ar_Exec.all, Arguments (1 .. Last_Arg), Success);
249 Arguments (1 .. Ar_Append_Options'Length) := Ar_Append_Options.all;
250 Arguments
251 (Ar_Append_Options'Length + 1) := new String'(Full_Output_File);
253 -- Appending invocation(s)
255 Big_Loop : while Success and then Current_Object <= Objects'Last loop
256 Last_Arg := Ar_Append_Options'Length + 1;
257 Size := Initial_Size + Full_Output_File'Length + 1;
259 Inner_Loop : while Current_Object <= Objects'Last loop
260 Size := Size + Objects (Current_Object)'Length + 1;
261 exit Inner_Loop when Size > Maximum_Size;
262 Last_Arg := Last_Arg + 1;
263 Arguments (Last_Arg) := Objects (Current_Object);
264 Current_Object := Current_Object + 1;
265 end loop Inner_Loop;
267 Display;
269 Spawn (Ar_Exec.all, Arguments (1 .. Last_Arg), Success);
270 end loop Big_Loop;
271 end if;
273 if not Success then
274 Fail (Ar_Name.all, " execution error.");
275 end if;
277 -- If we have found ranlib, run it over the library
279 if Ranlib_Exec /= null then
280 if not Opt.Quiet_Output then
281 Write_Str (Ranlib_Name.all);
282 Write_Char (' ');
283 Write_Line (Arguments (Ar_Options'Length + 1).all);
284 end if;
286 Spawn
287 (Ranlib_Exec.all,
288 Ranlib_Options.all & (Arguments (Ar_Options'Length + 1)),
289 Success);
291 if not Success then
292 Fail (Ranlib_Name.all, " execution error.");
293 end if;
294 end if;
295 end Ar;
297 -----------------
298 -- Delete_File --
299 -----------------
301 procedure Delete_File (Filename : String) is
302 File : constant String := Filename & ASCII.NUL;
303 Success : Boolean;
305 begin
306 Delete_File (File'Address, Success);
308 if Opt.Verbose_Mode then
309 if Success then
310 Write_Str ("deleted ");
312 else
313 Write_Str ("could not delete ");
314 end if;
316 Write_Line (Filename);
317 end if;
318 end Delete_File;
320 ---------
321 -- Gcc --
322 ---------
324 procedure Gcc
325 (Output_File : String;
326 Objects : Argument_List;
327 Options : Argument_List;
328 Options_2 : Argument_List;
329 Driver_Name : Name_Id := No_Name)
331 Link_Bytes : Integer := 0;
332 -- Projected number of bytes for the linker command line
334 Link_Max : Integer;
335 pragma Import (C, Link_Max, "__gnat_link_max");
336 -- Maximum number of bytes on the command line supported by the OS
337 -- linker. Passed this limit the response file mechanism must be used
338 -- if supported.
340 Object_List_File_Supported : Boolean;
341 for Object_List_File_Supported'Size use Character'Size;
342 pragma Import
343 (C, Object_List_File_Supported, "__gnat_objlist_file_supported");
344 -- Predicate indicating whether the linker has an option whereby the
345 -- names of object files can be passed to the linker in a file.
347 Object_File_Option_Ptr : Interfaces.C.Strings.chars_ptr;
348 pragma Import (C, Object_File_Option_Ptr, "__gnat_object_file_option");
349 -- Pointer to a string representing the linker option which specifies
350 -- the response file.
352 Using_GNU_Linker : Boolean;
353 for Using_GNU_Linker'Size use Character'Size;
354 pragma Import (C, Using_GNU_Linker, "__gnat_using_gnu_linker");
355 -- Predicate indicating whether this target uses the GNU linker. In
356 -- this case we must output a GNU linker compatible response file.
358 Opening : aliased constant String := """";
359 Closing : aliased constant String := '"' & ASCII.LF;
360 -- Needed to quote object paths in object list files when GNU linker
361 -- is used.
363 Tname : String_Access;
364 Tname_FD : File_Descriptor := Invalid_FD;
365 -- Temporary file used by linker to pass list of object files on
366 -- certain systems with limitations on size of arguments.
368 Closing_Status : Boolean;
369 -- For call to Close
371 Arguments :
372 Argument_List
373 (1 .. 7 + Objects'Length + Options'Length + Options_2'Length);
375 A : Natural := 0;
376 Success : Boolean;
378 Out_Opt : constant String_Access := new String'("-o");
379 Out_V : constant String_Access := new String'(Output_File);
380 Lib_Dir : constant String_Access := new String'("-L" & Lib_Directory);
381 Lib_Opt : constant String_Access := new String'(Dynamic_Option);
383 Driver : String_Access;
385 type Object_Position is (First, Second, Last);
387 Position : Object_Position;
389 procedure Write_RF (A : System.Address; N : Integer);
390 -- Write a string to the response file and check if it was successful.
391 -- Fail the program if it was not successful (disk full).
393 --------------
394 -- Write_RF --
395 --------------
397 procedure Write_RF (A : System.Address; N : Integer) is
398 Status : Integer;
399 begin
400 Status := Write (Tname_FD, A, N);
402 if Status /= N then
403 Fail ("cannot generate response file to link library: disk full");
404 end if;
405 end Write_RF;
407 begin
408 if Driver_Name = No_Name then
409 if Gcc_Exec = null then
410 if Gcc_Name = null then
411 Gcc_Name := Osint.Program_Name ("gcc", "gnatmake");
412 end if;
414 Gcc_Exec := Locate_Exec_On_Path (Gcc_Name.all);
416 if Gcc_Exec = null then
417 Fail (Gcc_Name.all, " not found in path");
418 end if;
419 end if;
421 Driver := Gcc_Exec;
423 else
424 Driver := Locate_Exec_On_Path (Get_Name_String (Driver_Name));
426 if Driver = null then
427 Fail (Get_Name_String (Driver_Name), " not found in path");
428 end if;
429 end if;
431 Link_Bytes := 0;
433 if Lib_Opt'Length /= 0 then
434 A := A + 1;
435 Arguments (A) := Lib_Opt;
436 Link_Bytes := Link_Bytes + Lib_Opt'Length + 1;
437 end if;
439 A := A + 1;
440 Arguments (A) := Out_Opt;
441 Link_Bytes := Link_Bytes + Out_Opt'Length + 1;
443 A := A + 1;
444 Arguments (A) := Out_V;
445 Link_Bytes := Link_Bytes + Out_V'Length + 1;
447 A := A + 1;
448 Arguments (A) := Lib_Dir;
449 Link_Bytes := Link_Bytes + Lib_Dir'Length + 1;
451 A := A + Options'Length;
452 Arguments (A - Options'Length + 1 .. A) := Options;
454 for J in Options'Range loop
455 Link_Bytes := Link_Bytes + Options (J)'Length + 1;
456 end loop;
458 if not Opt.Quiet_Output then
459 Write_Str (Driver.all);
461 for J in 1 .. A loop
462 Write_Char (' ');
463 Write_Str (Arguments (J).all);
464 end loop;
466 -- Do not display all the object files if not in verbose mode, only
467 -- the first one.
469 Position := First;
470 for J in Objects'Range loop
471 if Opt.Verbose_Mode or else Position = First then
472 Write_Char (' ');
473 Write_Str (Objects (J).all);
474 Position := Second;
476 elsif Position = Second then
477 Write_Str (" ...");
478 Position := Last;
479 end if;
480 end loop;
482 for J in Options_2'Range loop
483 Write_Char (' ');
484 Write_Str (Options_2 (J).all);
485 end loop;
487 Write_Eol;
488 end if;
490 for J in Objects'Range loop
491 Link_Bytes := Link_Bytes + Objects (J)'Length + 1;
492 end loop;
494 for J in Options_2'Range loop
495 Link_Bytes := Link_Bytes + Options_2 (J)'Length + 1;
496 end loop;
498 if Object_List_File_Supported and then Link_Bytes > Link_Max then
499 -- Create a temporary file containing the object files, one object
500 -- file per line for maximal compatibility with linkers supporting
501 -- this option.
503 Create_Temp_File (Tname_FD, Tname);
505 -- If target is using the GNU linker we must add a special header
506 -- and footer in the response file.
508 -- The syntax is : INPUT (object1.o object2.o ... )
510 -- Because the GNU linker does not like name with characters such
511 -- as '!', we must put the object paths between double quotes.
513 if Using_GNU_Linker then
514 declare
515 GNU_Header : aliased constant String := "INPUT (";
517 begin
518 Write_RF (GNU_Header'Address, GNU_Header'Length);
519 end;
520 end if;
522 for J in Objects'Range loop
523 -- Opening quote for GNU linker
525 if Using_GNU_Linker then
526 Write_RF (Opening'Address, 1);
527 end if;
529 Write_RF
530 (Objects (J).all'Address, Objects (J).all'Length);
532 -- Closing quote for GNU linker
534 if Using_GNU_Linker then
535 Write_RF (Closing'Address, 2);
537 else
538 Write_RF (ASCII.LF'Address, 1);
539 end if;
540 end loop;
542 -- Handle GNU linker response file footer
544 if Using_GNU_Linker then
545 declare
546 GNU_Footer : aliased constant String := ")";
548 begin
549 Write_RF (GNU_Footer'Address, GNU_Footer'Length);
550 end;
551 end if;
553 Close (Tname_FD, Closing_Status);
555 if not Closing_Status then
556 Fail ("cannot generate response file to link library: disk full");
557 end if;
559 A := A + 1;
560 Arguments (A) :=
561 new String'(Value (Object_File_Option_Ptr) & Tname.all);
563 else
564 A := A + Objects'Length;
565 Arguments (A - Objects'Length + 1 .. A) := Objects;
566 end if;
568 A := A + Options_2'Length;
569 Arguments (A - Options_2'Length + 1 .. A) := Options_2;
571 Spawn (Driver.all, Arguments (1 .. A), Success);
573 if Tname /= null then
574 Delete_File (Tname.all, Closing_Status);
576 if not Closing_Status then
577 Write_Str ("warning: could not delete response file """);
578 Write_Str (Tname.all);
579 Write_Line (""" to link library");
580 end if;
581 end if;
583 if not Success then
584 if Driver_Name = No_Name then
585 Fail (Gcc_Name.all, " execution error");
586 else
587 Fail (Get_Name_String (Driver_Name), " execution error");
588 end if;
589 end if;
590 end Gcc;
592 -------------------
593 -- Lib_Directory --
594 -------------------
596 function Lib_Directory return String is
597 Libgnat : constant String := Tgt.Libgnat;
599 begin
600 Name_Len := Libgnat'Length;
601 Name_Buffer (1 .. Name_Len) := Libgnat;
602 Get_Name_String (Osint.Find_File (Name_Enter, Osint.Library));
604 -- Remove libgnat.a
606 return Name_Buffer (1 .. Name_Len - Libgnat'Length);
607 end Lib_Directory;
609 end MLib.Utl;