2006-10-31 Vincent Celier <celier@adacore.com> * gnatls.adb: Take into account GPR_PROJECT_PATH, when it is defined, instead of ADA_PROJECT_PATH, for the project path. (Gnatls): When displaying the project path directories, use host dir specs. * prj-ext.adb (Prj.Ext elaboration): On VMS, only expand relative path names in the project path, as absolute paths may correspond to multi-valued VMS logical names. From-SVN: r118278
299 lines
10 KiB
Ada
299 lines
10 KiB
Ada
------------------------------------------------------------------------------
|
|
-- --
|
|
-- GNAT COMPILER COMPONENTS --
|
|
-- --
|
|
-- P R J . E X T --
|
|
-- --
|
|
-- B o d y --
|
|
-- --
|
|
-- Copyright (C) 2000-2006, 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, 51 Franklin Street, Fifth Floor, --
|
|
-- Boston, MA 02110-1301, USA. --
|
|
-- --
|
|
-- GNAT was originally developed by the GNAT team at New York University. --
|
|
-- Extensive contributions were provided by Ada Core Technologies Inc. --
|
|
-- --
|
|
------------------------------------------------------------------------------
|
|
|
|
with Hostparm;
|
|
with Namet; use Namet;
|
|
with Output; use Output;
|
|
with Osint; use Osint;
|
|
with Sdefault;
|
|
|
|
with GNAT.HTable;
|
|
|
|
package body Prj.Ext is
|
|
|
|
Gpr_Project_Path : constant String := "GPR_PROJECT_PATH";
|
|
Ada_Project_Path : constant String := "ADA_PROJECT_PATH";
|
|
-- Name of the env. variables that contain path name(s) of directories
|
|
-- where project files may reside. GPR_PROJECT_PATH has precedence over
|
|
-- ADA_PROJECT_PATH.
|
|
|
|
Gpr_Prj_Path : constant String_Access := Getenv (Gpr_Project_Path);
|
|
Ada_Prj_Path : constant String_Access := Getenv (Ada_Project_Path);
|
|
-- The path name(s) of directories where project files may reside.
|
|
-- May be empty.
|
|
|
|
No_Project_Default_Dir : constant String := "-";
|
|
|
|
Current_Project_Path : String_Access;
|
|
-- The project path. Initialized during elaboration of package Contains at
|
|
-- least the current working directory.
|
|
|
|
package Htable is new GNAT.HTable.Simple_HTable
|
|
(Header_Num => Header_Num,
|
|
Element => Name_Id,
|
|
No_Element => No_Name,
|
|
Key => Name_Id,
|
|
Hash => Hash,
|
|
Equal => "=");
|
|
-- External references are stored in this hash table, either by procedure
|
|
-- Add (directly or through a call to function Check) or by function
|
|
-- Value_Of when an environment variable is found non empty. Value_Of
|
|
-- first for external reference in this table, before checking the
|
|
-- environment. Htable is emptied (reset) by procedure Reset.
|
|
|
|
---------
|
|
-- Add --
|
|
---------
|
|
|
|
procedure Add
|
|
(External_Name : String;
|
|
Value : String)
|
|
is
|
|
The_Key : Name_Id;
|
|
The_Value : Name_Id;
|
|
begin
|
|
Name_Len := Value'Length;
|
|
Name_Buffer (1 .. Name_Len) := Value;
|
|
The_Value := Name_Find;
|
|
Name_Len := External_Name'Length;
|
|
Name_Buffer (1 .. Name_Len) := External_Name;
|
|
Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len));
|
|
The_Key := Name_Find;
|
|
Htable.Set (The_Key, The_Value);
|
|
end Add;
|
|
|
|
-----------
|
|
-- Check --
|
|
-----------
|
|
|
|
function Check (Declaration : String) return Boolean is
|
|
begin
|
|
for Equal_Pos in Declaration'Range loop
|
|
if Declaration (Equal_Pos) = '=' then
|
|
exit when Equal_Pos = Declaration'First;
|
|
exit when Equal_Pos = Declaration'Last;
|
|
Add
|
|
(External_Name =>
|
|
Declaration (Declaration'First .. Equal_Pos - 1),
|
|
Value =>
|
|
Declaration (Equal_Pos + 1 .. Declaration'Last));
|
|
return True;
|
|
end if;
|
|
end loop;
|
|
|
|
return False;
|
|
end Check;
|
|
|
|
------------------
|
|
-- Project_Path --
|
|
------------------
|
|
|
|
function Project_Path return String is
|
|
begin
|
|
return Current_Project_Path.all;
|
|
end Project_Path;
|
|
|
|
-----------
|
|
-- Reset --
|
|
-----------
|
|
|
|
procedure Reset is
|
|
begin
|
|
Htable.Reset;
|
|
end Reset;
|
|
|
|
----------------------
|
|
-- Set_Project_Path --
|
|
----------------------
|
|
|
|
procedure Set_Project_Path (New_Path : String) is
|
|
begin
|
|
Free (Current_Project_Path);
|
|
Current_Project_Path := new String'(New_Path);
|
|
end Set_Project_Path;
|
|
|
|
--------------
|
|
-- Value_Of --
|
|
--------------
|
|
|
|
function Value_Of
|
|
(External_Name : Name_Id;
|
|
With_Default : Name_Id := No_Name)
|
|
return Name_Id
|
|
is
|
|
The_Value : Name_Id;
|
|
Name : String := Get_Name_String (External_Name);
|
|
|
|
begin
|
|
Canonical_Case_File_Name (Name);
|
|
Name_Len := Name'Length;
|
|
Name_Buffer (1 .. Name_Len) := Name;
|
|
The_Value := Htable.Get (Name_Find);
|
|
|
|
if The_Value /= No_Name then
|
|
return The_Value;
|
|
end if;
|
|
|
|
-- Find if it is an environment, if it is, put value in the hash table
|
|
|
|
declare
|
|
Env_Value : String_Access := Getenv (Name);
|
|
|
|
begin
|
|
if Env_Value /= null and then Env_Value'Length > 0 then
|
|
Name_Len := Env_Value'Length;
|
|
Name_Buffer (1 .. Name_Len) := Env_Value.all;
|
|
The_Value := Name_Find;
|
|
Htable.Set (External_Name, The_Value);
|
|
Free (Env_Value);
|
|
return The_Value;
|
|
|
|
else
|
|
Free (Env_Value);
|
|
return With_Default;
|
|
end if;
|
|
end;
|
|
end Value_Of;
|
|
|
|
begin
|
|
-- Initialize Current_Project_Path during package elaboration
|
|
|
|
declare
|
|
Add_Default_Dir : Boolean := True;
|
|
First : Positive;
|
|
Last : Positive;
|
|
New_Len : Positive;
|
|
New_Last : Positive;
|
|
Prj_Path : String_Access := Gpr_Prj_Path;
|
|
|
|
begin
|
|
if Gpr_Prj_Path.all /= "" then
|
|
|
|
-- Warn if both environment variables are defined
|
|
|
|
if Ada_Prj_Path.all /= "" then
|
|
Write_Line ("Warning: ADA_PROJECT_PATH is not taken into account");
|
|
Write_Line (" when GPR_PROJECT_PATH is defined");
|
|
end if;
|
|
|
|
else
|
|
Prj_Path := Ada_Prj_Path;
|
|
end if;
|
|
|
|
-- The current directory is always first
|
|
|
|
Name_Len := 1;
|
|
Name_Buffer (Name_Len) := '.';
|
|
|
|
-- If environment variable is defined and not empty, add its content
|
|
|
|
if Prj_Path.all /= "" then
|
|
Name_Len := Name_Len + 1;
|
|
Name_Buffer (Name_Len) := Path_Separator;
|
|
|
|
Add_Str_To_Name_Buffer (Prj_Path.all);
|
|
|
|
-- Scan the directory path to see if "-" is one of the directories.
|
|
-- Remove each occurence of "-" and set Add_Default_Dir to False.
|
|
-- Also resolve relative paths and symbolic links.
|
|
|
|
First := 3;
|
|
loop
|
|
while First <= Name_Len
|
|
and then (Name_Buffer (First) = Path_Separator)
|
|
loop
|
|
First := First + 1;
|
|
end loop;
|
|
|
|
exit when First > Name_Len;
|
|
|
|
Last := First;
|
|
|
|
while Last < Name_Len
|
|
and then Name_Buffer (Last + 1) /= Path_Separator
|
|
loop
|
|
Last := Last + 1;
|
|
end loop;
|
|
|
|
-- If the directory is "-", set Add_Default_Dir to False and
|
|
-- remove from path.
|
|
|
|
if Name_Buffer (First .. Last) = No_Project_Default_Dir then
|
|
Add_Default_Dir := False;
|
|
|
|
for J in Last + 1 .. Name_Len loop
|
|
Name_Buffer (J - No_Project_Default_Dir'Length - 1) :=
|
|
Name_Buffer (J);
|
|
end loop;
|
|
|
|
Name_Len := Name_Len - No_Project_Default_Dir'Length - 1;
|
|
|
|
elsif not Hostparm.OpenVMS
|
|
or else not Is_Absolute_Path (Name_Buffer (First .. Last))
|
|
then
|
|
-- On VMS, only expand relative path names, as absolute paths
|
|
-- may correspond to multi-valued VMS logical names.
|
|
|
|
declare
|
|
New_Dir : constant String :=
|
|
Normalize_Pathname (Name_Buffer (First .. Last));
|
|
|
|
begin
|
|
-- If the absolute path was resolved and is different from
|
|
-- the original, replace original with the resolved path.
|
|
|
|
if New_Dir /= Name_Buffer (First .. Last)
|
|
and then New_Dir'Length /= 0
|
|
then
|
|
New_Len := Name_Len + New_Dir'Length - (Last - First + 1);
|
|
New_Last := First + New_Dir'Length - 1;
|
|
Name_Buffer (New_Last + 1 .. New_Len) :=
|
|
Name_Buffer (Last + 1 .. Name_Len);
|
|
Name_Buffer (First .. New_Last) := New_Dir;
|
|
Name_Len := New_Len;
|
|
Last := New_Last;
|
|
end if;
|
|
end;
|
|
end if;
|
|
|
|
First := Last + 1;
|
|
end loop;
|
|
end if;
|
|
|
|
-- Set the initial value of Current_Project_Path
|
|
|
|
if Add_Default_Dir then
|
|
Current_Project_Path :=
|
|
new String'(Name_Buffer (1 .. Name_Len) & Path_Separator &
|
|
Sdefault.Search_Dir_Prefix.all & ".." &
|
|
Directory_Separator & ".." & Directory_Separator &
|
|
".." & Directory_Separator & "gnat");
|
|
else
|
|
Current_Project_Path := new String'(Name_Buffer (1 .. Name_Len));
|
|
end if;
|
|
end;
|
|
end Prj.Ext;
|