2013-11-13 Jan-Benedict Glaw <jbglaw@lug-owl.de>
[official-gcc.git] / gcc / ada / mlib.adb
blob4c4d375f3243bc547bcc8538b2ce335322532866
1 ------------------------------------------------------------------------------
2 -- --
3 -- GNAT COMPILER COMPONENTS --
4 -- --
5 -- M L I B --
6 -- --
7 -- B o d y --
8 -- --
9 -- Copyright (C) 1999-2009, 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 Ada.Characters.Handling; use Ada.Characters.Handling;
27 with Interfaces.C.Strings;
28 with System;
30 with Hostparm;
31 with Opt;
32 with Output; use Output;
34 with MLib.Utl; use MLib.Utl;
36 with Prj.Com;
38 with GNAT.Directory_Operations; use GNAT.Directory_Operations;
40 package body MLib is
42 -------------------
43 -- Build_Library --
44 -------------------
46 procedure Build_Library
47 (Ofiles : Argument_List;
48 Output_File : String;
49 Output_Dir : String)
51 begin
52 if Opt.Verbose_Mode and not Opt.Quiet_Output then
53 Write_Line ("building a library...");
54 Write_Str (" make ");
55 Write_Line (Output_File);
56 end if;
58 Ar (Output_Dir &
59 "lib" & Output_File & ".a", Objects => Ofiles);
60 end Build_Library;
62 ------------------------
63 -- Check_Library_Name --
64 ------------------------
66 procedure Check_Library_Name (Name : String) is
67 begin
68 if Name'Length = 0 then
69 Prj.Com.Fail ("library name cannot be empty");
70 end if;
72 if Name'Length > Max_Characters_In_Library_Name then
73 Prj.Com.Fail ("illegal library name """
74 & Name
75 & """: too long");
76 end if;
78 if not Is_Letter (Name (Name'First)) then
79 Prj.Com.Fail ("illegal library name """
80 & Name
81 & """: should start with a letter");
82 end if;
84 for Index in Name'Range loop
85 if not Is_Alphanumeric (Name (Index)) then
86 Prj.Com.Fail ("illegal library name """
87 & Name
88 & """: should include only letters and digits");
89 end if;
90 end loop;
91 end Check_Library_Name;
93 --------------------
94 -- Copy_ALI_Files --
95 --------------------
97 procedure Copy_ALI_Files
98 (Files : Argument_List;
99 To : Path_Name_Type;
100 Interfaces : String_List)
102 Success : Boolean := False;
103 To_Dir : constant String := Get_Name_String (To);
104 Is_Interface : Boolean := False;
106 procedure Verbose_Copy (Index : Positive);
107 -- In verbose mode, output a message that the indexed file is copied
108 -- to the destination directory.
110 ------------------
111 -- Verbose_Copy --
112 ------------------
114 procedure Verbose_Copy (Index : Positive) is
115 begin
116 if Opt.Verbose_Mode then
117 Write_Str ("Copying """);
118 Write_Str (Files (Index).all);
119 Write_Str (""" to """);
120 Write_Str (To_Dir);
121 Write_Line ("""");
122 end if;
123 end Verbose_Copy;
125 -- Start of processing for Copy_ALI_Files
127 begin
128 if Interfaces'Length = 0 then
130 -- If there are no Interfaces, copy all the ALI files as is
132 for Index in Files'Range loop
133 Verbose_Copy (Index);
134 Set_Writable
135 (To_Dir &
136 Directory_Separator &
137 Base_Name (Files (Index).all));
138 Copy_File
139 (Files (Index).all,
140 To_Dir,
141 Success,
142 Mode => Overwrite,
143 Preserve => Preserve);
145 exit when not Success;
146 end loop;
148 else
149 -- Copy only the interface ALI file, and put the special indicator
150 -- "SL" on the P line.
152 for Index in Files'Range loop
154 declare
155 File_Name : String := Base_Name (Files (Index).all);
157 begin
158 Canonical_Case_File_Name (File_Name);
160 -- Check if this is one of the interface ALIs
162 Is_Interface := False;
164 for Index in Interfaces'Range loop
165 if File_Name = Interfaces (Index).all then
166 Is_Interface := True;
167 exit;
168 end if;
169 end loop;
171 -- If it is an interface ALI, copy line by line. Insert
172 -- the interface indication at the end of the P line.
173 -- Do not copy ALI files that are not Interfaces.
175 if Is_Interface then
176 Success := False;
177 Verbose_Copy (Index);
178 Set_Writable
179 (To_Dir &
180 Directory_Separator &
181 Base_Name (Files (Index).all));
183 declare
184 FD : File_Descriptor;
185 Len : Integer;
186 Actual_Len : Integer;
187 S : String_Access;
188 Curr : Natural;
189 P_Line_Found : Boolean;
190 Status : Boolean;
192 begin
193 -- Open the file
195 Name_Len := Files (Index)'Length;
196 Name_Buffer (1 .. Name_Len) := Files (Index).all;
197 Name_Len := Name_Len + 1;
198 Name_Buffer (Name_Len) := ASCII.NUL;
200 FD := Open_Read (Name_Buffer'Address, Binary);
202 if FD /= Invalid_FD then
203 Len := Integer (File_Length (FD));
205 -- ??? Why "+3" here
207 S := new String (1 .. Len + 3);
209 -- Read the file. Note that the loop is not necessary
210 -- since the whole file is read at once except on VMS.
212 Curr := S'First;
213 while Curr <= Len loop
214 Actual_Len := Read (FD, S (Curr)'Address, Len);
216 -- Exit if we could not read for some reason
218 exit when Actual_Len = 0;
220 Curr := Curr + Actual_Len;
221 end loop;
223 -- We are done with the input file, so we close it
224 -- ignoring any bad status.
226 Close (FD, Status);
228 P_Line_Found := False;
230 -- Look for the P line. When found, add marker SL
231 -- at the beginning of the P line.
233 for Index in 1 .. Len - 3 loop
234 if (S (Index) = ASCII.LF
235 or else
236 S (Index) = ASCII.CR)
237 and then S (Index + 1) = 'P'
238 then
239 S (Index + 5 .. Len + 3) := S (Index + 2 .. Len);
240 S (Index + 2 .. Index + 4) := " SL";
241 P_Line_Found := True;
242 exit;
243 end if;
244 end loop;
246 if P_Line_Found then
248 -- Create new modified ALI file
250 Name_Len := To_Dir'Length;
251 Name_Buffer (1 .. Name_Len) := To_Dir;
252 Name_Len := Name_Len + 1;
253 Name_Buffer (Name_Len) := Directory_Separator;
254 Name_Buffer
255 (Name_Len + 1 .. Name_Len + File_Name'Length) :=
256 File_Name;
257 Name_Len := Name_Len + File_Name'Length + 1;
258 Name_Buffer (Name_Len) := ASCII.NUL;
260 FD := Create_File (Name_Buffer'Address, Binary);
262 -- Write the modified text and close the newly
263 -- created file.
265 if FD /= Invalid_FD then
266 Actual_Len := Write (FD, S (1)'Address, Len + 3);
268 Close (FD, Status);
270 -- Set Success to True only if the newly
271 -- created file has been correctly written.
273 Success := Status and then Actual_Len = Len + 3;
275 if Success then
277 -- Set_Read_Only is used here, rather than
278 -- Set_Non_Writable, so that gprbuild can
279 -- he compiled with older compilers.
281 Set_Read_Only
282 (Name_Buffer (1 .. Name_Len - 1));
283 end if;
284 end if;
285 end if;
286 end if;
287 end;
289 -- This is not an interface ALI
291 else
292 Success := True;
293 end if;
294 end;
296 if not Success then
297 Prj.Com.Fail ("could not copy ALI files to library dir");
298 end if;
299 end loop;
300 end if;
301 end Copy_ALI_Files;
303 ----------------------
304 -- Create_Sym_Links --
305 ----------------------
307 procedure Create_Sym_Links
308 (Lib_Path : String;
309 Lib_Version : String;
310 Lib_Dir : String;
311 Maj_Version : String)
313 function Symlink
314 (Oldpath : System.Address;
315 Newpath : System.Address) return Integer;
316 pragma Import (C, Symlink, "__gnat_symlink");
318 Version_Path : String_Access;
320 Success : Boolean;
321 Result : Integer;
322 pragma Unreferenced (Success, Result);
324 begin
325 Version_Path := new String (1 .. Lib_Version'Length + 1);
326 Version_Path (1 .. Lib_Version'Length) := Lib_Version;
327 Version_Path (Version_Path'Last) := ASCII.NUL;
329 if Maj_Version'Length = 0 then
330 declare
331 Newpath : String (1 .. Lib_Path'Length + 1);
332 begin
333 Newpath (1 .. Lib_Path'Length) := Lib_Path;
334 Newpath (Newpath'Last) := ASCII.NUL;
335 Delete_File (Lib_Path, Success);
336 Result := Symlink (Version_Path (1)'Address, Newpath'Address);
337 end;
339 else
340 declare
341 Newpath1 : String (1 .. Lib_Path'Length + 1);
342 Maj_Path : constant String :=
343 Lib_Dir & Directory_Separator & Maj_Version;
344 Newpath2 : String (1 .. Maj_Path'Length + 1);
345 Maj_Ver : String (1 .. Maj_Version'Length + 1);
347 begin
348 Newpath1 (1 .. Lib_Path'Length) := Lib_Path;
349 Newpath1 (Newpath1'Last) := ASCII.NUL;
351 Newpath2 (1 .. Maj_Path'Length) := Maj_Path;
352 Newpath2 (Newpath2'Last) := ASCII.NUL;
354 Maj_Ver (1 .. Maj_Version'Length) := Maj_Version;
355 Maj_Ver (Maj_Ver'Last) := ASCII.NUL;
357 Delete_File (Maj_Path, Success);
359 Result := Symlink (Version_Path (1)'Address, Newpath2'Address);
361 Delete_File (Lib_Path, Success);
363 Result := Symlink (Maj_Ver'Address, Newpath1'Address);
364 end;
365 end if;
366 end Create_Sym_Links;
368 --------------------------------
369 -- Linker_Library_Path_Option --
370 --------------------------------
372 function Linker_Library_Path_Option return String_Access is
374 Run_Path_Option_Ptr : Interfaces.C.Strings.chars_ptr;
375 pragma Import (C, Run_Path_Option_Ptr, "__gnat_run_path_option");
376 -- Pointer to string representing the native linker option which
377 -- specifies the path where the dynamic loader should find shared
378 -- libraries. Equal to null string if this system doesn't support it.
380 S : constant String := Interfaces.C.Strings.Value (Run_Path_Option_Ptr);
382 begin
383 if S'Length = 0 then
384 return null;
385 else
386 return new String'(S);
387 end if;
388 end Linker_Library_Path_Option;
390 -------------------
391 -- Major_Id_Name --
392 -------------------
394 function Major_Id_Name
395 (Lib_Filename : String;
396 Lib_Version : String)
397 return String
399 Maj_Version : constant String := Lib_Version;
400 Last_Maj : Positive;
401 Last : Positive;
402 Ok_Maj : Boolean := False;
404 begin
405 Last_Maj := Maj_Version'Last;
406 while Last_Maj > Maj_Version'First loop
407 if Maj_Version (Last_Maj) in '0' .. '9' then
408 Last_Maj := Last_Maj - 1;
410 else
411 Ok_Maj := Last_Maj /= Maj_Version'Last and then
412 Maj_Version (Last_Maj) = '.';
414 if Ok_Maj then
415 Last_Maj := Last_Maj - 1;
416 end if;
418 exit;
419 end if;
420 end loop;
422 if Ok_Maj then
423 Last := Last_Maj;
424 while Last > Maj_Version'First loop
425 if Maj_Version (Last) in '0' .. '9' then
426 Last := Last - 1;
428 else
429 Ok_Maj := Last /= Last_Maj and then
430 Maj_Version (Last) = '.';
432 if Ok_Maj then
433 Last := Last - 1;
434 Ok_Maj :=
435 Maj_Version (Maj_Version'First .. Last) = Lib_Filename;
436 end if;
438 exit;
439 end if;
440 end loop;
441 end if;
443 if Ok_Maj then
444 return Maj_Version (Maj_Version'First .. Last_Maj);
445 else
446 return "";
447 end if;
448 end Major_Id_Name;
450 -------------------------------
451 -- Separate_Run_Path_Options --
452 -------------------------------
454 function Separate_Run_Path_Options return Boolean is
455 Separate_Paths : Boolean;
456 for Separate_Paths'Size use Character'Size;
457 pragma Import (C, Separate_Paths, "__gnat_separate_run_path_options");
458 begin
459 return Separate_Paths;
460 end Separate_Run_Path_Options;
462 -- Package elaboration
464 begin
465 -- Copy_Attributes always fails on VMS
467 if Hostparm.OpenVMS then
468 Preserve := None;
469 end if;
470 end MLib;