* 5ataprop.adb, 5atpopsp.adb, 5ftaprop.adb, 5gmastop.adb, 5gtaprop.adb, 5htaprop.adb, 5itaprop.adb, 5lintman.adb, 5omastop.adb, 5oosinte.adb, 5otaprop.adb, 5staprop.adb, 5vinterr.adb, 5vtaprop.adb, 5vtpopde.adb, 5wintman.adb, 5wtaprop.adb, 5zinterr.adb, 5ztaprop.adb, 6vcstrea.adb, 7sintman.adb, 7staprop.adb, 9drpc.adb, ChangeLog, Makefile.in, a-except.adb, a-tags.ads, a-tasatt.adb, a-teioed.adb, a-textio.ads, a-witeio.ads, a-wtedit.adb, ali.ads, comperr.adb, cstand.adb, einfo.ads, errout.adb, exp_ch11.adb, exp_ch2.adb, exp_ch3.adb, exp_ch4.adb, exp_ch5.adb, exp_ch6.adb, exp_ch9.adb, exp_util.adb, exp_util.ads, fname-uf.adb, g-cgi.ads, g-exctra.ads, g-expect.ads, g-regist.adb, g-spipat.adb, gnatchop.adb, gnatlink.adb, gnatls.adb, gnatmain.adb, gnatmem.adb, init.c, make.adb, make.ads, mdlltool.adb, nlists.ads, osint.ads, par-ch3.adb, par-ch4.adb, par-ch5.adb, par-ch6.adb, par.adb, repinfo.adb, s-fatflt.ads, s-fatlfl.ads, s-fatllf.ads, s-fatsfl.ads, s-finimp.adb, s-finimp.ads, s-interr.adb, s-secsta.ads, s-shasto.ads, s-stalib.adb, s-stalib.ads, s-tarest.ads, s-tasdeb.adb, s-tassta.adb, s-tassta.ads, s-vaflop.ads, scans.ads, scn.adb, sem.ads, sem_aggr.adb, sem_attr.adb, sem_case.ads, sem_ch10.adb, sem_ch12.adb, sem_ch13.adb, sem_ch3.adb, sem_ch3.ads, sem_ch5.adb, sem_ch7.adb, sem_ch8.adb, sem_ch8.ads, sem_type.adb, sem_util.ads, sinfo.ads, sprint.adb, tbuild.ads, types.ads, utils.c, xeinfo.adb: Fix spelling errors. From-SVN: r48055
595 lines
18 KiB
Ada
595 lines
18 KiB
Ada
------------------------------------------------------------------------------
|
|
-- --
|
|
-- GNAT COMPILER COMPONENTS --
|
|
-- --
|
|
-- G N A T M A I N --
|
|
-- --
|
|
-- B o d y --
|
|
-- --
|
|
-- $Revision: 1.1 $
|
|
-- --
|
|
-- Copyright (C) 1992-2001 Free Software Foundation, Inc. --
|
|
-- --
|
|
-- GNAT is free software; you can redistribute it and/or modify it under --
|
|
-- terms of the GNU General Public License as published by the Free Soft- --
|
|
-- ware Foundation; either version 2, or (at your option) any later ver- --
|
|
-- sion. GNAT is distributed in the hope that it will be useful, but WITH- --
|
|
-- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY --
|
|
-- or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License --
|
|
-- for more details. You should have received a copy of the GNU General --
|
|
-- Public License distributed with GNAT; see file COPYING. If not, write --
|
|
-- to the Free Software Foundation, 59 Temple Place - Suite 330, Boston, --
|
|
-- MA 02111-1307, USA. --
|
|
-- --
|
|
-- GNAT was originally developed by the GNAT team at New York University. --
|
|
-- It is now maintained by Ada Core Technologies Inc (http://www.gnat.com). --
|
|
-- --
|
|
------------------------------------------------------------------------------
|
|
|
|
with Csets;
|
|
with GNAT.Case_Util;
|
|
with GNAT.OS_Lib; use GNAT.OS_Lib;
|
|
with Namet; use Namet;
|
|
with Opt;
|
|
with Osint; use Osint;
|
|
with Output; use Output;
|
|
with Prj; use Prj;
|
|
with Prj.Env;
|
|
with Prj.Ext; use Prj.Ext;
|
|
with Prj.Pars;
|
|
with Prj.Util; use Prj.Util;
|
|
with Snames; use Snames;
|
|
with Stringt; use Stringt;
|
|
with Table;
|
|
with Types; use Types;
|
|
|
|
procedure Gnatmain is
|
|
|
|
Ada_Include_Path : constant String := "ADA_INCLUDE_PATH";
|
|
Ada_Objects_Path : constant String := "ADA_OBJECTS_PATH";
|
|
|
|
type Tool_Type is (None, List, Xref, Find, Stub, Make, Comp, Bind, Link);
|
|
|
|
-- The tool that is going to be called
|
|
|
|
Tool : Tool_Type := None;
|
|
|
|
-- For each tool, Tool_Package_Names contains the name of the
|
|
-- corresponding package in the project file.
|
|
|
|
Tool_Package_Names : constant array (Tool_Type) of Name_Id :=
|
|
(None => No_Name,
|
|
List => Name_Gnatls,
|
|
Xref => Name_Cross_Reference,
|
|
Find => Name_Finder,
|
|
Stub => Name_Gnatstub,
|
|
Comp => No_Name,
|
|
Make => No_Name,
|
|
Bind => No_Name,
|
|
Link => No_Name);
|
|
|
|
-- For each tool, Tool_Names contains the name of the executable
|
|
-- to be spawned.
|
|
|
|
Gnatmake : constant String_Access := new String'("gnatmake");
|
|
|
|
Tool_Names : constant array (Tool_Type) of String_Access :=
|
|
(None => null,
|
|
List => new String'("gnatls"),
|
|
Xref => new String'("gnatxref"),
|
|
Find => new String'("gnatfind"),
|
|
Stub => new String'("gnatstub"),
|
|
Comp => Gnatmake,
|
|
Make => Gnatmake,
|
|
Bind => Gnatmake,
|
|
Link => Gnatmake);
|
|
|
|
Project_File : String_Access;
|
|
Project : Prj.Project_Id;
|
|
Current_Verbosity : Prj.Verbosity := Prj.Default;
|
|
|
|
-- This flag indicates a switch -p (for gnatxref and gnatfind) for
|
|
-- an old fashioned project file. -p cannot be used in conjonction
|
|
-- with -P.
|
|
|
|
Old_Project_File_Used : Boolean := False;
|
|
|
|
Next_Arg : Positive;
|
|
|
|
-- A table to keep the switches on the command line
|
|
|
|
package Saved_Switches is new Table.Table (
|
|
Table_Component_Type => String_Access,
|
|
Table_Index_Type => Integer,
|
|
Table_Low_Bound => 1,
|
|
Table_Initial => 20,
|
|
Table_Increment => 100,
|
|
Table_Name => "Gnatmain.Saved_Switches");
|
|
|
|
-- A table to keep the switches from the project file
|
|
|
|
package Switches is new Table.Table (
|
|
Table_Component_Type => String_Access,
|
|
Table_Index_Type => Integer,
|
|
Table_Low_Bound => 1,
|
|
Table_Initial => 20,
|
|
Table_Increment => 100,
|
|
Table_Name => "Gnatmain.Switches");
|
|
|
|
procedure Add_Switch (Argv : String; And_Save : Boolean);
|
|
-- Add a switch in one of the tables above
|
|
|
|
procedure Display (Program : String; Args : Argument_List);
|
|
-- Displays Program followed by the arguments in Args
|
|
|
|
function Index (Char : Character; Str : String) return Natural;
|
|
-- Returns the first occurrence of Char in Str.
|
|
-- Returns 0 if Char is not in Str.
|
|
|
|
procedure Scan_Arg (Argv : String; And_Save : Boolean);
|
|
-- Scan and process arguments. Argv is a single argument.
|
|
|
|
procedure Usage;
|
|
-- Output usage
|
|
|
|
----------------
|
|
-- Add_Switch --
|
|
----------------
|
|
|
|
procedure Add_Switch (Argv : String; And_Save : Boolean) is
|
|
begin
|
|
if And_Save then
|
|
Saved_Switches.Increment_Last;
|
|
Saved_Switches.Table (Saved_Switches.Last) := new String'(Argv);
|
|
|
|
else
|
|
Switches.Increment_Last;
|
|
Switches.Table (Switches.Last) := new String'(Argv);
|
|
end if;
|
|
end Add_Switch;
|
|
|
|
-------------
|
|
-- Display --
|
|
-------------
|
|
|
|
procedure Display (Program : String; Args : Argument_List) is
|
|
begin
|
|
if not Opt.Quiet_Output then
|
|
Write_Str (Program);
|
|
|
|
for J in Args'Range loop
|
|
Write_Str (" ");
|
|
Write_Str (Args (J).all);
|
|
end loop;
|
|
|
|
Write_Eol;
|
|
end if;
|
|
end Display;
|
|
|
|
-----------
|
|
-- Index --
|
|
-----------
|
|
|
|
function Index (Char : Character; Str : String) return Natural is
|
|
begin
|
|
for Index in Str'Range loop
|
|
if Str (Index) = Char then
|
|
return Index;
|
|
end if;
|
|
end loop;
|
|
|
|
return 0;
|
|
end Index;
|
|
|
|
--------------
|
|
-- Scan_Arg --
|
|
--------------
|
|
|
|
procedure Scan_Arg (Argv : String; And_Save : Boolean) is
|
|
begin
|
|
pragma Assert (Argv'First = 1);
|
|
|
|
if Argv'Length = 0 then
|
|
return;
|
|
end if;
|
|
|
|
if Argv (1) = Switch_Character or else Argv (1) = '-' then
|
|
|
|
if Argv'Length = 1 then
|
|
Fail ("switch character cannot be followed by a blank");
|
|
end if;
|
|
|
|
-- The two style project files (-p and -P) cannot be used together
|
|
|
|
if (Tool = Find or else Tool = Xref)
|
|
and then Argv (2) = 'p'
|
|
then
|
|
Old_Project_File_Used := True;
|
|
if Project_File /= null then
|
|
Fail ("-P and -p cannot be used together");
|
|
end if;
|
|
end if;
|
|
|
|
-- -q Be quiet: do not output tool command
|
|
|
|
if Argv (2 .. Argv'Last) = "q" then
|
|
Opt.Quiet_Output := True;
|
|
|
|
-- Only gnatstub and gnatmake have a -q switch
|
|
|
|
if Tool = Stub or else Tool_Names (Tool) = Gnatmake then
|
|
Add_Switch (Argv, And_Save);
|
|
end if;
|
|
|
|
-- gnatmake will take care of the project file related switches
|
|
|
|
elsif Tool_Names (Tool) = Gnatmake then
|
|
Add_Switch (Argv, And_Save);
|
|
|
|
-- -vPx Specify verbosity while parsing project files
|
|
|
|
elsif Argv'Length = 4 and then Argv (2 .. 3) = "vP" then
|
|
case Argv (4) is
|
|
when '0' =>
|
|
Current_Verbosity := Prj.Default;
|
|
when '1' =>
|
|
Current_Verbosity := Prj.Medium;
|
|
when '2' =>
|
|
Current_Verbosity := Prj.High;
|
|
when others =>
|
|
null;
|
|
end case;
|
|
|
|
-- -Pproject_file Specify project file to be used
|
|
|
|
elsif Argv'Length >= 3 and then Argv (2) = 'P' then
|
|
|
|
-- Only one -P switch can be used
|
|
|
|
if Project_File /= null then
|
|
Fail (Argv & ": second project file forbidden (first is """ &
|
|
Project_File.all & """)");
|
|
|
|
-- The two style project files (-p and -P) cannot be used together
|
|
|
|
elsif Old_Project_File_Used then
|
|
Fail ("-p and -P cannot be used together");
|
|
|
|
else
|
|
Project_File := new String'(Argv (3 .. Argv'Last));
|
|
end if;
|
|
|
|
-- -Xexternal=value Specify an external reference to be used
|
|
-- in project files
|
|
|
|
elsif Argv'Length >= 5 and then Argv (2) = 'X' then
|
|
declare
|
|
Equal_Pos : constant Natural :=
|
|
Index ('=', Argv (3 .. Argv'Last));
|
|
begin
|
|
if Equal_Pos >= 4 and then
|
|
Equal_Pos /= Argv'Last then
|
|
Add (External_Name => Argv (3 .. Equal_Pos - 1),
|
|
Value => Argv (Equal_Pos + 1 .. Argv'Last));
|
|
else
|
|
Fail (Argv & " is not a valid external assignment.");
|
|
end if;
|
|
end;
|
|
|
|
else
|
|
Add_Switch (Argv, And_Save);
|
|
end if;
|
|
|
|
else
|
|
Add_Switch (Argv, And_Save);
|
|
end if;
|
|
|
|
end Scan_Arg;
|
|
|
|
-----------
|
|
-- Usage --
|
|
-----------
|
|
|
|
procedure Usage is
|
|
begin
|
|
Write_Str ("Usage: ");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" list switches [list of object files]");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" xref switches file1 file2 ...");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" find switches pattern[:sourcefile[:line[:column]]] " &
|
|
"[file1 file2 ...]");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" stub switches filename [directory]");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" comp switches files");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" make switches [files]");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" bind switches files");
|
|
Write_Eol;
|
|
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" link switches files");
|
|
Write_Eol;
|
|
|
|
Write_Eol;
|
|
|
|
Write_Str ("switches interpreted by ");
|
|
Osint.Write_Program_Name;
|
|
Write_Str (" for List Xref and Find:");
|
|
Write_Eol;
|
|
|
|
Write_Str (" -q Be quiet: do not output tool command");
|
|
Write_Eol;
|
|
|
|
Write_Str (" -Pproj Use GNAT Project File proj");
|
|
Write_Eol;
|
|
|
|
Write_Str (" -vPx Specify verbosity when parsing " &
|
|
"GNAT Project Files");
|
|
Write_Eol;
|
|
|
|
Write_Str (" -Xnm=val Specify an external reference for " &
|
|
"GNAT Project Files");
|
|
Write_Eol;
|
|
|
|
Write_Eol;
|
|
|
|
Write_Str ("all other arguments are transmited to the tool");
|
|
Write_Eol;
|
|
|
|
Write_Eol;
|
|
|
|
end Usage;
|
|
|
|
begin
|
|
|
|
Osint.Initialize (Unspecified);
|
|
|
|
Namet.Initialize;
|
|
Csets.Initialize;
|
|
|
|
Snames.Initialize;
|
|
|
|
Prj.Initialize;
|
|
|
|
if Arg_Count = 1 then
|
|
Usage;
|
|
return;
|
|
end if;
|
|
|
|
-- Get the name of the tool
|
|
|
|
declare
|
|
Tool_Name : String (1 .. Len_Arg (1));
|
|
|
|
begin
|
|
Fill_Arg (Tool_Name'Address, 1);
|
|
GNAT.Case_Util.To_Lower (Tool_Name);
|
|
|
|
if Tool_Name = "list" then
|
|
Tool := List;
|
|
|
|
elsif Tool_Name = "xref" then
|
|
Tool := Xref;
|
|
|
|
elsif Tool_Name = "find" then
|
|
Tool := Find;
|
|
|
|
elsif Tool_Name = "stub" then
|
|
Tool := Stub;
|
|
|
|
elsif Tool_Name = "comp" then
|
|
Tool := Comp;
|
|
|
|
elsif Tool_Name = "make" then
|
|
Tool := Make;
|
|
|
|
elsif Tool_Name = "bind" then
|
|
Tool := Bind;
|
|
|
|
elsif Tool_Name = "link" then
|
|
Tool := Link;
|
|
|
|
else
|
|
Fail ("first argument needs to be ""list"", ""xref"", ""find""" &
|
|
", ""stub"", ""comp"", ""make"", ""bind"" or ""link""");
|
|
end if;
|
|
end;
|
|
|
|
Next_Arg := 2;
|
|
|
|
-- Get the command line switches that follow the name of the tool
|
|
|
|
Scan_Args : while Next_Arg < Arg_Count loop
|
|
declare
|
|
Next_Argv : String (1 .. Len_Arg (Next_Arg));
|
|
|
|
begin
|
|
Fill_Arg (Next_Argv'Address, Next_Arg);
|
|
Scan_Arg (Next_Argv, And_Save => True);
|
|
end;
|
|
|
|
Next_Arg := Next_Arg + 1;
|
|
end loop Scan_Args;
|
|
|
|
-- If a switch -P was specified, parse the project file.
|
|
-- Project_File is always null if we are going to invoke gnatmake,
|
|
-- that is when Tool is Comp, Make, Bind or Link.
|
|
|
|
if Project_File /= null then
|
|
|
|
Prj.Pars.Set_Verbosity (To => Current_Verbosity);
|
|
|
|
Prj.Pars.Parse
|
|
(Project => Project,
|
|
Project_File_Name => Project_File.all);
|
|
|
|
if Project = Prj.No_Project then
|
|
Fail ("""" & Project_File.all & """ processing failed");
|
|
end if;
|
|
|
|
-- Check if a package with the name of the tool is in the project file
|
|
-- and if there is one, get the switches, if any, and scan them.
|
|
|
|
declare
|
|
Data : Prj.Project_Data := Prj.Projects.Table (Project);
|
|
Pkg : Prj.Package_Id :=
|
|
Prj.Util.Value_Of
|
|
(Name => Tool_Package_Names (Tool),
|
|
In_Packages => Data.Decl.Packages);
|
|
Element : Package_Element;
|
|
Default_Switches_Array : Array_Element_Id;
|
|
Switches : Prj.Variable_Value;
|
|
Current : Prj.String_List_Id;
|
|
The_String : String_Element;
|
|
|
|
begin
|
|
if Pkg /= No_Package then
|
|
Element := Packages.Table (Pkg);
|
|
|
|
-- Packages Gnatls and Gnatstub have a single attribute Switches,
|
|
-- that is not an associative array.
|
|
|
|
if Tool = List or else Tool = Stub then
|
|
Switches :=
|
|
Prj.Util.Value_Of
|
|
(Variable_Name => Name_Switches,
|
|
In_Variables => Element.Decl.Attributes);
|
|
|
|
-- Packages Cross_Reference (for gnatxref) and Finder
|
|
-- (for gnatfind) have an attributed Default_Switches,
|
|
-- an associative array, indexed by the name of the
|
|
-- programming language.
|
|
else
|
|
Default_Switches_Array :=
|
|
Prj.Util.Value_Of
|
|
(Name => Name_Default_Switches,
|
|
In_Arrays => Packages.Table (Pkg).Decl.Arrays);
|
|
Switches := Prj.Util.Value_Of
|
|
(Index => Name_Ada,
|
|
In_Array => Default_Switches_Array);
|
|
|
|
end if;
|
|
|
|
-- If there are switches specified in the package of the
|
|
-- project file corresponding to the tool, scan them.
|
|
|
|
case Switches.Kind is
|
|
when Prj.Undefined =>
|
|
null;
|
|
|
|
when Prj.Single =>
|
|
if String_Length (Switches.Value) > 0 then
|
|
String_To_Name_Buffer (Switches.Value);
|
|
Scan_Arg
|
|
(Name_Buffer (1 .. Name_Len),
|
|
And_Save => False);
|
|
end if;
|
|
|
|
when Prj.List =>
|
|
Current := Switches.Values;
|
|
while Current /= Prj.Nil_String loop
|
|
The_String := String_Elements.Table (Current);
|
|
|
|
if String_Length (The_String.Value) > 0 then
|
|
String_To_Name_Buffer (The_String.Value);
|
|
Scan_Arg
|
|
(Name_Buffer (1 .. Name_Len),
|
|
And_Save => False);
|
|
end if;
|
|
|
|
Current := The_String.Next;
|
|
end loop;
|
|
end case;
|
|
end if;
|
|
end;
|
|
|
|
-- Set up the environment variables ADA_INCLUDE_PATH and
|
|
-- ADA_OBJECTS_PATH.
|
|
|
|
Setenv
|
|
(Name => Ada_Include_Path,
|
|
Value => Prj.Env.Ada_Include_Path (Project).all);
|
|
Setenv
|
|
(Name => Ada_Objects_Path,
|
|
Value => Prj.Env.Ada_Objects_Path
|
|
(Project, Including_Libraries => False).all);
|
|
|
|
end if;
|
|
|
|
-- Gather all the arguments, those from the project file first,
|
|
-- locate the tool and call it with the arguments.
|
|
|
|
declare
|
|
Args : Argument_List (1 .. Switches.Last + Saved_Switches.Last + 4);
|
|
Arg_Num : Natural := 0;
|
|
Tool_Path : String_Access;
|
|
Success : Boolean;
|
|
|
|
procedure Add (Arg : String_Access);
|
|
|
|
procedure Add (Arg : String_Access) is
|
|
begin
|
|
Arg_Num := Arg_Num + 1;
|
|
Args (Arg_Num) := Arg;
|
|
end Add;
|
|
|
|
begin
|
|
|
|
case Tool is
|
|
when Comp =>
|
|
Add (new String'("-u"));
|
|
Add (new String'("-f"));
|
|
|
|
when Bind =>
|
|
Add (new String'("-b"));
|
|
|
|
when Link =>
|
|
Add (new String'("-l"));
|
|
|
|
when others =>
|
|
null;
|
|
|
|
end case;
|
|
|
|
for Index in 1 .. Switches.Last loop
|
|
Arg_Num := Arg_Num + 1;
|
|
Args (Arg_Num) := Switches.Table (Index);
|
|
end loop;
|
|
|
|
for Index in 1 .. Saved_Switches.Last loop
|
|
Arg_Num := Arg_Num + 1;
|
|
Args (Arg_Num) := Saved_Switches.Table (Index);
|
|
end loop;
|
|
|
|
Tool_Path := GNAT.OS_Lib.Locate_Exec_On_Path (Tool_Names (Tool).all);
|
|
|
|
if Tool_Path = null then
|
|
Fail ("error, unable to locate " & Tool_Names (Tool).all);
|
|
end if;
|
|
|
|
Display (Tool_Names (Tool).all, Args (1 .. Arg_Num));
|
|
|
|
GNAT.OS_Lib.Spawn (Tool_Path.all, Args (1 .. Arg_Num), Success);
|
|
|
|
end;
|
|
|
|
end Gnatmain;
|