| ------------------------------------------------------------------------------ |
| -- -- |
| -- GNAT COMPILER COMPONENTS -- |
| -- -- |
| -- P R E P -- |
| -- -- |
| -- B o d y -- |
| -- -- |
| -- Copyright (C) 2002-2022, 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 3, 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 COPYING3. If not, go to -- |
| -- http://www.gnu.org/licenses for a complete copy of the license. -- |
| -- -- |
| -- GNAT was originally developed by the GNAT team at New York University. -- |
| -- Extensive contributions were provided by Ada Core Technologies Inc. -- |
| -- -- |
| ------------------------------------------------------------------------------ |
| |
| with Csets; use Csets; |
| with Err_Vars; use Err_Vars; |
| with Opt; use Opt; |
| with Osint; use Osint; |
| with Output; use Output; |
| with Scans; use Scans; |
| with Snames; use Snames; |
| with Sinput; |
| with Stringt; use Stringt; |
| with Table; |
| with Uintp; use Uintp; |
| |
| with GNAT.Heap_Sort_G; |
| |
| package body Prep is |
| |
| use Symbol_Table; |
| |
| type Token_Name_Array is array (Token_Type) of Name_Id; |
| Token_Names : constant Token_Name_Array := |
| (Tok_Abort => Name_Abort, |
| Tok_Abs => Name_Abs, |
| Tok_Abstract => Name_Abstract, |
| Tok_Accept => Name_Accept, |
| Tok_Aliased => Name_Aliased, |
| Tok_All => Name_All, |
| Tok_Array => Name_Array, |
| Tok_And => Name_And, |
| Tok_At => Name_At, |
| Tok_Begin => Name_Begin, |
| Tok_Body => Name_Body, |
| Tok_Case => Name_Case, |
| Tok_Constant => Name_Constant, |
| Tok_Declare => Name_Declare, |
| Tok_Delay => Name_Delay, |
| Tok_Delta => Name_Delta, |
| Tok_Digits => Name_Digits, |
| Tok_Else => Name_Else, |
| Tok_Elsif => Name_Elsif, |
| Tok_End => Name_End, |
| Tok_Entry => Name_Entry, |
| Tok_Exception => Name_Exception, |
| Tok_Exit => Name_Exit, |
| Tok_For => Name_For, |
| Tok_Function => Name_Function, |
| Tok_Generic => Name_Generic, |
| Tok_Goto => Name_Goto, |
| Tok_If => Name_If, |
| Tok_Is => Name_Is, |
| Tok_Limited => Name_Limited, |
| Tok_Loop => Name_Loop, |
| Tok_Mod => Name_Mod, |
| Tok_New => Name_New, |
| Tok_Null => Name_Null, |
| Tok_Of => Name_Of, |
| Tok_Or => Name_Or, |
| Tok_Others => Name_Others, |
| Tok_Out => Name_Out, |
| Tok_Package => Name_Package, |
| Tok_Pragma => Name_Pragma, |
| Tok_Private => Name_Private, |
| Tok_Procedure => Name_Procedure, |
| Tok_Protected => Name_Protected, |
| Tok_Raise => Name_Raise, |
| Tok_Range => Name_Range, |
| Tok_Record => Name_Record, |
| Tok_Rem => Name_Rem, |
| Tok_Renames => Name_Renames, |
| Tok_Requeue => Name_Requeue, |
| Tok_Return => Name_Return, |
| Tok_Reverse => Name_Reverse, |
| Tok_Select => Name_Select, |
| Tok_Separate => Name_Separate, |
| Tok_Subtype => Name_Subtype, |
| Tok_Tagged => Name_Tagged, |
| Tok_Task => Name_Task, |
| Tok_Terminate => Name_Terminate, |
| Tok_Then => Name_Then, |
| Tok_Type => Name_Type, |
| Tok_Until => Name_Until, |
| Tok_Use => Name_Use, |
| Tok_When => Name_When, |
| Tok_While => Name_While, |
| Tok_With => Name_With, |
| Tok_Xor => Name_Xor, |
| others => No_Name); |
| |
| Already_Initialized : Boolean := False; |
| -- Used to avoid repetition of the part of the initialisation that needs |
| -- to be done only once. |
| |
| String_False : String_Id; |
| -- "false", as a string_id |
| |
| -------------- |
| -- Behavior -- |
| -------------- |
| |
| -- Accesses to procedure specified by procedure Initialize |
| |
| Error_Msg : Error_Msg_Proc; |
| -- Report an error |
| |
| Scan : Scan_Proc; |
| -- Scan one token |
| |
| Set_Ignore_Errors : Set_Ignore_Errors_Proc; |
| -- Indicate if error should be taken into account |
| |
| Put_Char : Put_Char_Proc; |
| -- Output one character |
| |
| New_EOL : New_EOL_Proc; |
| -- Output an end of line indication |
| |
| ------------------------------- |
| -- State of the Preprocessor -- |
| ------------------------------- |
| |
| type Pp_State is record |
| If_Ptr : Source_Ptr; |
| -- The location of the #if statement (used to flag #if with no |
| -- corresponding #end if, at the end). |
| |
| Else_Ptr : Source_Ptr; |
| -- The location of the #else statement (used to detect multiple #else's) |
| |
| Deleting : Boolean; |
| -- Set to True when the code should be deleted or commented out |
| |
| Match_Seen : Boolean; |
| -- Set to True when a condition in an #if or an #elsif is True. Also set |
| -- to True if Deleting at the previous level is True. Used to decide if |
| -- Deleting should be set to True in a following #elsif or #else. |
| |
| end record; |
| |
| type Pp_Depth is new Nat; |
| |
| Ground : constant Pp_Depth := 0; |
| |
| package Pp_States is new Table.Table |
| (Table_Component_Type => Pp_State, |
| Table_Index_Type => Pp_Depth, |
| Table_Low_Bound => 1, |
| Table_Initial => 10, |
| Table_Increment => 100, |
| Table_Name => "Prep.Pp_States"); |
| -- A stack of the states of the preprocessor, for nested #if |
| |
| type Operator is (None, Op_Or, Op_And); |
| |
| ----------------- |
| -- Subprograms -- |
| ----------------- |
| |
| function Deleting return Boolean; |
| -- Return True if code should be deleted or commented out |
| |
| function Expression |
| (Evaluate_It : Boolean; |
| Complemented : Boolean := False) return Boolean; |
| -- Evaluate a condition in an #if or an #elsif statement. If Evaluate_It |
| -- is False, the condition is effectively evaluated, otherwise, only the |
| -- syntax is checked. |
| |
| procedure Go_To_End_Of_Line; |
| -- Advance the scan pointer until we reach an end of line or the end of the |
| -- buffer. |
| |
| function Matching_Strings (S1, S2 : String_Id) return Boolean; |
| -- Returns True if the two string parameters are equal (case insensitive) |
| |
| --------------------------------------- |
| -- Change_Reserved_Keyword_To_Symbol -- |
| --------------------------------------- |
| |
| procedure Change_Reserved_Keyword_To_Symbol |
| (All_Keywords : Boolean := False) |
| is |
| New_Name : constant Name_Id := Token_Names (Token); |
| |
| begin |
| if New_Name /= No_Name then |
| case Token is |
| when Tok_And |
| | Tok_Else |
| | Tok_Elsif |
| | Tok_End |
| | Tok_If |
| | Tok_Or |
| | Tok_Then |
| => |
| if All_Keywords then |
| Token := Tok_Identifier; |
| Token_Name := New_Name; |
| end if; |
| |
| when others => |
| Token := Tok_Identifier; |
| Token_Name := New_Name; |
| end case; |
| end if; |
| end Change_Reserved_Keyword_To_Symbol; |
| |
| ------------------------------------------ |
| -- Check_Command_Line_Symbol_Definition -- |
| ------------------------------------------ |
| |
| procedure Check_Command_Line_Symbol_Definition |
| (Definition : String; |
| Data : out Symbol_Data) |
| is |
| Index : Natural := 0; |
| Result : Symbol_Data; |
| |
| begin |
| -- Look for the character '=' |
| |
| for J in Definition'Range loop |
| if Definition (J) = '=' then |
| Index := J; |
| exit; |
| end if; |
| end loop; |
| |
| -- If no character '=', then the value is True |
| |
| if Index = 0 then |
| |
| -- Put the symbol in the name buffer |
| |
| Name_Len := Definition'Length; |
| Name_Buffer (1 .. Name_Len) := Definition; |
| Result := True_Value; |
| |
| elsif Index = Definition'First then |
| Fail ("invalid symbol definition """ & Definition & """"); |
| |
| else |
| -- Put the symbol in the name buffer |
| |
| Name_Len := Index - Definition'First; |
| Name_Buffer (1 .. Name_Len) := |
| String'(Definition (Definition'First .. Index - 1)); |
| |
| -- Check the syntax of the value |
| |
| if Definition (Index + 1) /= '"' |
| or else Definition (Definition'Last) /= '"' |
| then |
| for J in Index + 1 .. Definition'Last loop |
| case Definition (J) is |
| when '_' | '.' | '0' .. '9' | 'a' .. 'z' | 'A' .. 'Z' => |
| null; |
| |
| when others => |
| Fail ("illegal value """ |
| & Definition (Index + 1 .. Definition'Last) |
| & """"); |
| end case; |
| end loop; |
| end if; |
| |
| -- Even if the value is a string, we still set Is_A_String to False, |
| -- to avoid adding additional quotes in the preprocessed sources when |
| -- replacing $<symbol>. |
| |
| Result.Is_A_String := False; |
| |
| -- Put the value in the result |
| |
| Start_String; |
| Store_String_Chars (Definition (Index + 1 .. Definition'Last)); |
| Result.Value := End_String; |
| end if; |
| |
| -- Now, check the syntax of the symbol (we don't allow accented or |
| -- wide characters). |
| |
| if Name_Buffer (1) not in 'a' .. 'z' |
| and then Name_Buffer (1) not in 'A' .. 'Z' |
| then |
| Fail ("symbol """ |
| & Name_Buffer (1 .. Name_Len) |
| & """ does not start with a letter"); |
| end if; |
| |
| for J in 2 .. Name_Len loop |
| case Name_Buffer (J) is |
| when 'a' .. 'z' | 'A' .. 'Z' | '0' .. '9' => |
| null; |
| |
| when '_' => |
| if J = Name_Len then |
| Fail ("symbol """ |
| & Name_Buffer (1 .. Name_Len) |
| & """ end with a '_'"); |
| |
| elsif Name_Buffer (J + 1) = '_' then |
| Fail ("symbol """ |
| & Name_Buffer (1 .. Name_Len) |
| & """ contains consecutive '_'"); |
| end if; |
| |
| when others => |
| Fail ("symbol """ |
| & Name_Buffer (1 .. Name_Len) |
| & """ contains illegal character(s)"); |
| end case; |
| end loop; |
| |
| Result.On_The_Command_Line := True; |
| |
| -- Put the symbol name in the result |
| |
| declare |
| Sym : constant String := Name_Buffer (1 .. Name_Len); |
| |
| begin |
| for Index in 1 .. Name_Len loop |
| Name_Buffer (Index) := Fold_Lower (Name_Buffer (Index)); |
| end loop; |
| |
| Result.Symbol := Name_Find; |
| Name_Len := Sym'Length; |
| Name_Buffer (1 .. Name_Len) := Sym; |
| Result.Original := Name_Find; |
| end; |
| |
| Data := Result; |
| end Check_Command_Line_Symbol_Definition; |
| |
| -------------- |
| -- Deleting -- |
| -------------- |
| |
| function Deleting return Boolean is |
| begin |
| -- Always return False when not inside an #if statement |
| |
| if Opt.No_Deletion or else Pp_States.Last = Ground then |
| return False; |
| else |
| return Pp_States.Table (Pp_States.Last).Deleting; |
| end if; |
| end Deleting; |
| |
| ---------------- |
| -- Expression -- |
| ---------------- |
| |
| function Expression |
| (Evaluate_It : Boolean; |
| Complemented : Boolean := False) return Boolean |
| is |
| Evaluation : Boolean := Evaluate_It; |
| -- Is set to False after an "or else" when left term is True and after |
| -- an "and then" when left term is False. |
| |
| Final_Result : Boolean := False; |
| |
| Current_Result : Boolean := False; |
| -- Value of a term |
| |
| Current_Operator : Operator := None; |
| Symbol1 : Symbol_Id; |
| Symbol2 : Symbol_Id; |
| Symbol_Name1 : Name_Id; |
| Symbol_Name2 : Name_Id; |
| Symbol_Pos1 : Source_Ptr; |
| Symbol_Pos2 : Source_Ptr; |
| Symbol_Value1 : String_Id; |
| Symbol_Value2 : String_Id; |
| |
| Relop : Token_Type; |
| |
| begin |
| -- Loop for each term |
| |
| loop |
| Change_Reserved_Keyword_To_Symbol; |
| |
| Current_Result := False; |
| |
| -- Scan current term, starting with Token |
| |
| case Token is |
| |
| -- Handle parenthesized expression |
| |
| when Tok_Left_Paren => |
| Scan.all; |
| Current_Result := Expression (Evaluation); |
| |
| if Token = Tok_Right_Paren then |
| Scan.all; |
| |
| else |
| Error_Msg -- CODEFIX |
| ("`)` expected", Token_Ptr); |
| end if; |
| |
| -- Handle not expression |
| |
| when Tok_Not => |
| Scan.all; |
| Current_Result := |
| not Expression (Evaluation, Complemented => True); |
| |
| -- Handle sequence starting with identifier |
| |
| when Tok_Identifier => |
| Symbol_Name1 := Token_Name; |
| Symbol_Pos1 := Token_Ptr; |
| Scan.all; |
| |
| if Token = Tok_Apostrophe then |
| |
| -- symbol'Defined |
| |
| Scan.all; |
| |
| if Token = Tok_Identifier |
| and then Token_Name = Name_Defined |
| then |
| Scan.all; |
| |
| else |
| Error_Msg ("identifier `Defined` expected", Token_Ptr); |
| end if; |
| |
| if Evaluation then |
| Current_Result := Index_Of (Symbol_Name1) /= No_Symbol; |
| end if; |
| |
| -- Handle relational operator |
| |
| elsif Token in Tok_Equal | Tok_Less | Tok_Less_Equal | |
| Tok_Greater | Tok_Greater_Equal |
| then |
| Relop := Token; |
| Scan.all; |
| Change_Reserved_Keyword_To_Symbol; |
| |
| if Token = Tok_Integer_Literal then |
| |
| -- symbol = integer |
| -- symbol < integer |
| -- symbol <= integer |
| -- symbol > integer |
| -- symbol >= integer |
| |
| declare |
| Value : constant Int := UI_To_Int (Int_Literal_Value); |
| Data : Symbol_Data; |
| |
| Symbol_Value : Int; |
| -- Value of symbol as Int |
| |
| begin |
| if Evaluation then |
| Symbol1 := Index_Of (Symbol_Name1); |
| |
| if Symbol1 = No_Symbol then |
| Error_Msg_Name_1 := Symbol_Name1; |
| Error_Msg ("unknown symbol %", Symbol_Pos1); |
| Symbol_Value1 := No_String; |
| |
| else |
| Data := Mapping.Table (Symbol1); |
| |
| if Data.Is_A_String then |
| Error_Msg_Name_1 := Symbol_Name1; |
| Error_Msg |
| ("symbol % value is not integer", |
| Symbol_Pos1); |
| |
| else |
| begin |
| String_To_Name_Buffer (Data.Value); |
| Symbol_Value := |
| Int'Value (Name_Buffer (1 .. Name_Len)); |
| |
| case Relop is |
| when Tok_Equal => |
| Current_Result := |
| Symbol_Value = Value; |
| |
| when Tok_Less => |
| Current_Result := |
| Symbol_Value < Value; |
| |
| when Tok_Less_Equal => |
| Current_Result := |
| Symbol_Value <= Value; |
| |
| when Tok_Greater => |
| Current_Result := |
| Symbol_Value > Value; |
| |
| when Tok_Greater_Equal => |
| Current_Result := |
| Symbol_Value >= Value; |
| |
| when others => |
| null; |
| end case; |
| |
| exception |
| when Constraint_Error => |
| Error_Msg_Name_1 := Symbol_Name1; |
| Error_Msg |
| ("symbol % value is not an integer", |
| Symbol_Pos1); |
| end; |
| end if; |
| end if; |
| end if; |
| |
| Scan.all; |
| end; |
| |
| -- Error if relational operator other than = if not numbers |
| |
| elsif Relop /= Tok_Equal then |
| Error_Msg ("number expected", Token_Ptr); |
| |
| -- Equality comparison of two strings |
| |
| elsif Token = Tok_Identifier then |
| |
| -- symbol = symbol |
| |
| Symbol_Name2 := Token_Name; |
| Symbol_Pos2 := Token_Ptr; |
| Scan.all; |
| |
| if Evaluation then |
| Symbol1 := Index_Of (Symbol_Name1); |
| |
| if Symbol1 = No_Symbol then |
| if Undefined_Symbols_Are_False then |
| Symbol_Value1 := String_False; |
| |
| else |
| Error_Msg_Name_1 := Symbol_Name1; |
| Error_Msg ("unknown symbol %", Symbol_Pos1); |
| Symbol_Value1 := No_String; |
| end if; |
| |
| else |
| Symbol_Value1 := |
| Mapping.Table (Symbol1).Value; |
| end if; |
| |
| Symbol2 := Index_Of (Symbol_Name2); |
| |
| if Symbol2 = No_Symbol then |
| if Undefined_Symbols_Are_False then |
| Symbol_Value2 := String_False; |
| |
| else |
| Error_Msg_Name_1 := Symbol_Name2; |
| Error_Msg ("unknown symbol %", Symbol_Pos2); |
| Symbol_Value2 := No_String; |
| end if; |
| |
| else |
| Symbol_Value2 := Mapping.Table (Symbol2).Value; |
| end if; |
| |
| if Symbol_Value1 /= No_String |
| and then |
| Symbol_Value2 /= No_String |
| then |
| Current_Result := |
| Matching_Strings (Symbol_Value1, Symbol_Value2); |
| end if; |
| end if; |
| |
| elsif Token = Tok_String_Literal then |
| |
| -- symbol = "value" |
| |
| if Evaluation then |
| Symbol1 := Index_Of (Symbol_Name1); |
| |
| if Symbol1 = No_Symbol then |
| if Undefined_Symbols_Are_False then |
| Symbol_Value1 := String_False; |
| |
| else |
| Error_Msg_Name_1 := Symbol_Name1; |
| Error_Msg ("unknown symbol %", Symbol_Pos1); |
| Symbol_Value1 := No_String; |
| end if; |
| |
| else |
| Symbol_Value1 := Mapping.Table (Symbol1).Value; |
| end if; |
| |
| if Symbol_Value1 /= No_String then |
| Current_Result := |
| Matching_Strings |
| (Symbol_Value1, |
| String_Literal_Id); |
| end if; |
| end if; |
| |
| Scan.all; |
| |
| else |
| Error_Msg |
| ("literal integer, symbol or literal string expected", |
| Token_Ptr); |
| end if; |
| |
| -- Handle True or False |
| |
| else |
| if Evaluation then |
| Symbol1 := Index_Of (Symbol_Name1); |
| |
| if Symbol1 = No_Symbol then |
| if Undefined_Symbols_Are_False then |
| Symbol_Value1 := String_False; |
| |
| else |
| Error_Msg_Name_1 := Symbol_Name1; |
| Error_Msg ("unknown symbol %", Symbol_Pos1); |
| Symbol_Value1 := No_String; |
| end if; |
| |
| else |
| Symbol_Value1 := Mapping.Table (Symbol1).Value; |
| end if; |
| |
| if Symbol_Value1 /= No_String then |
| String_To_Name_Buffer (Symbol_Value1); |
| |
| for Index in 1 .. Name_Len loop |
| Name_Buffer (Index) := |
| Fold_Lower (Name_Buffer (Index)); |
| end loop; |
| |
| if Name_Buffer (1 .. Name_Len) = "true" then |
| Current_Result := True; |
| |
| elsif Name_Buffer (1 .. Name_Len) = "false" then |
| Current_Result := False; |
| |
| else |
| Error_Msg_Name_1 := Symbol_Name1; |
| Error_Msg |
| ("value of symbol % is not True or False", |
| Symbol_Pos1); |
| end if; |
| end if; |
| end if; |
| end if; |
| |
| -- Unrecognized sequence |
| |
| when others => |
| Error_Msg ("`(`, NOT or symbol expected", Token_Ptr); |
| end case; |
| |
| -- Update the cumulative final result |
| |
| case Current_Operator is |
| when None => |
| Final_Result := Current_Result; |
| |
| when Op_Or => |
| Final_Result := Final_Result or Current_Result; |
| |
| when Op_And => |
| Final_Result := Final_Result and Current_Result; |
| end case; |
| |
| -- Handle AND |
| |
| if Token = Tok_And then |
| if Complemented then |
| Error_Msg |
| ("mixing NOT and AND is not allowed, parentheses are required", |
| Token_Ptr); |
| |
| elsif Current_Operator = Op_Or then |
| Error_Msg ("mixing OR and AND is not allowed", Token_Ptr); |
| end if; |
| |
| Current_Operator := Op_And; |
| Scan.all; |
| |
| if Token = Tok_Then then |
| Scan.all; |
| |
| if Final_Result = False then |
| Evaluation := False; |
| end if; |
| end if; |
| |
| -- Handle OR |
| |
| elsif Token = Tok_Or then |
| if Complemented then |
| Error_Msg |
| ("mixing NOT and OR is not allowed, parentheses are required", |
| Token_Ptr); |
| |
| elsif Current_Operator = Op_And then |
| Error_Msg ("mixing AND and OR is not allowed", Token_Ptr); |
| end if; |
| |
| Current_Operator := Op_Or; |
| Scan.all; |
| |
| if Token = Tok_Else then |
| Scan.all; |
| |
| if Final_Result then |
| Evaluation := False; |
| end if; |
| end if; |
| |
| -- No AND/OR operator, so exit from the loop through terms |
| |
| else |
| exit; |
| end if; |
| end loop; |
| |
| return Final_Result; |
| end Expression; |
| |
| ----------------------- |
| -- Go_To_End_Of_Line -- |
| ----------------------- |
| |
| procedure Go_To_End_Of_Line is |
| begin |
| -- Scan until we get an end of line or we reach the end of the buffer |
| |
| while Token not in Tok_End_Of_Line | Tok_EOF loop |
| Scan.all; |
| end loop; |
| end Go_To_End_Of_Line; |
| |
| -------------- |
| -- Index_Of -- |
| -------------- |
| |
| function Index_Of (Symbol : Name_Id) return Symbol_Id is |
| begin |
| if Mapping.Table /= null then |
| for J in Symbol_Id range 1 .. Symbol_Table.Last (Mapping) loop |
| if Mapping.Table (J).Symbol = Symbol then |
| return J; |
| end if; |
| end loop; |
| end if; |
| |
| return No_Symbol; |
| end Index_Of; |
| |
| ---------------- |
| -- Initialize -- |
| ---------------- |
| |
| procedure Initialize is |
| begin |
| if not Already_Initialized then |
| Start_String; |
| Store_String_Chars ("True"); |
| True_Value.Value := End_String; |
| |
| Start_String; |
| Store_String_Chars ("False"); |
| String_False := End_String; |
| |
| Already_Initialized := True; |
| end if; |
| end Initialize; |
| |
| ------------------ |
| -- List_Symbols -- |
| ------------------ |
| |
| procedure List_Symbols (Foreword : String) is |
| Order : array (0 .. Integer (Symbol_Table.Last (Mapping))) |
| of Symbol_Id; |
| -- After alphabetical sorting, this array stores the indexes of the |
| -- symbols in the order they are displayed. |
| |
| function Lt (Op1, Op2 : Natural) return Boolean; |
| -- Comparison routine for sort call |
| |
| procedure Move (From : Natural; To : Natural); |
| -- Move routine for sort call |
| |
| -------- |
| -- Lt -- |
| -------- |
| |
| function Lt (Op1, Op2 : Natural) return Boolean is |
| S1 : constant String := |
| Get_Name_String (Mapping.Table (Order (Op1)).Symbol); |
| S2 : constant String := |
| Get_Name_String (Mapping.Table (Order (Op2)).Symbol); |
| begin |
| return S1 < S2; |
| end Lt; |
| |
| ---------- |
| -- Move -- |
| ---------- |
| |
| procedure Move (From : Natural; To : Natural) is |
| begin |
| Order (To) := Order (From); |
| end Move; |
| |
| package Sort_Syms is new GNAT.Heap_Sort_G (Move, Lt); |
| |
| Max_L : Natural; |
| -- Maximum length of any symbol |
| |
| -- Start of processing for List_Symbols_Case |
| |
| begin |
| if Symbol_Table.Last (Mapping) = 0 then |
| return; |
| end if; |
| |
| if Foreword'Length > 0 then |
| Write_Eol; |
| Write_Line (Foreword); |
| |
| for J in Foreword'Range loop |
| Write_Char ('='); |
| end loop; |
| end if; |
| |
| -- Initialize the order |
| |
| for J in Order'Range loop |
| Order (J) := Symbol_Id (J); |
| end loop; |
| |
| -- Sort alphabetically |
| |
| Sort_Syms.Sort (Order'Last); |
| |
| Max_L := 7; |
| |
| for J in 1 .. Symbol_Table.Last (Mapping) loop |
| Get_Name_String (Mapping.Table (J).Original); |
| Max_L := Integer'Max (Max_L, Name_Len); |
| end loop; |
| |
| Write_Eol; |
| Write_Str ("Symbol"); |
| |
| for J in 1 .. Max_L - 5 loop |
| Write_Char (' '); |
| end loop; |
| |
| Write_Line ("Value"); |
| |
| Write_Str ("------"); |
| |
| for J in 1 .. Max_L - 5 loop |
| Write_Char (' '); |
| end loop; |
| |
| Write_Line ("------"); |
| |
| for J in 1 .. Order'Last loop |
| declare |
| Data : constant Symbol_Data := Mapping.Table (Order (J)); |
| |
| begin |
| Get_Name_String (Data.Original); |
| Write_Str (Name_Buffer (1 .. Name_Len)); |
| |
| for K in Name_Len .. Max_L loop |
| Write_Char (' '); |
| end loop; |
| |
| String_To_Name_Buffer (Data.Value); |
| |
| if Data.Is_A_String then |
| Write_Char ('"'); |
| |
| for J in 1 .. Name_Len loop |
| Write_Char (Name_Buffer (J)); |
| |
| if Name_Buffer (J) = '"' then |
| Write_Char ('"'); |
| end if; |
| end loop; |
| |
| Write_Char ('"'); |
| |
| else |
| Write_Str (Name_Buffer (1 .. Name_Len)); |
| end if; |
| end; |
| |
| Write_Eol; |
| end loop; |
| |
| Write_Eol; |
| end List_Symbols; |
| |
| ---------------------- |
| -- Matching_Strings -- |
| ---------------------- |
| |
| function Matching_Strings (S1, S2 : String_Id) return Boolean is |
| begin |
| String_To_Name_Buffer (S1); |
| |
| for Index in 1 .. Name_Len loop |
| Name_Buffer (Index) := Fold_Lower (Name_Buffer (Index)); |
| end loop; |
| |
| declare |
| String1 : constant String := Name_Buffer (1 .. Name_Len); |
| |
| begin |
| String_To_Name_Buffer (S2); |
| |
| for Index in 1 .. Name_Len loop |
| Name_Buffer (Index) := Fold_Lower (Name_Buffer (Index)); |
| end loop; |
| |
| return String1 = Name_Buffer (1 .. Name_Len); |
| end; |
| end Matching_Strings; |
| |
| -------------------- |
| -- Parse_Def_File -- |
| -------------------- |
| |
| -- This procedure REALLY needs some more comments ??? |
| |
| procedure Parse_Def_File is |
| Symbol : Symbol_Id; |
| Symbol_Name : Name_Id; |
| Original_Name : Name_Id; |
| Data : Symbol_Data; |
| Value_Start : Source_Ptr; |
| Value_End : Source_Ptr; |
| Ch : Character; |
| |
| use ASCII; |
| |
| begin |
| Def_Line_Loop : |
| loop |
| Scan.all; |
| |
| exit Def_Line_Loop when Token = Tok_EOF; |
| |
| if Token /= Tok_End_Of_Line then |
| Change_Reserved_Keyword_To_Symbol; |
| |
| if Token /= Tok_Identifier then |
| Error_Msg ("identifier expected", Token_Ptr); |
| goto Cleanup; |
| end if; |
| |
| Symbol_Name := Token_Name; |
| Name_Len := 0; |
| |
| for Ptr in Token_Ptr .. Scan_Ptr - 1 loop |
| Name_Len := Name_Len + 1; |
| Name_Buffer (Name_Len) := Sinput.Source (Ptr); |
| end loop; |
| |
| Original_Name := Name_Find; |
| Scan.all; |
| |
| if Token /= Tok_Colon_Equal then |
| Error_Msg -- CODEFIX |
| ("`:=` expected", Token_Ptr); |
| goto Cleanup; |
| end if; |
| |
| Scan.all; |
| |
| if Token = Tok_Integer_Literal then |
| declare |
| Ptr : Source_Ptr := Token_Ptr; |
| |
| begin |
| Start_String; |
| while Ptr < Scan_Ptr loop |
| Store_String_Char (Sinput.Source (Ptr)); |
| Ptr := Ptr + 1; |
| end loop; |
| |
| Data := (Symbol => Symbol_Name, |
| Original => Original_Name, |
| On_The_Command_Line => False, |
| Is_A_String => False, |
| Value => End_String); |
| end; |
| |
| Scan.all; |
| |
| if Token not in Tok_End_Of_Line | Tok_EOF then |
| Error_Msg ("extraneous text in definition", Token_Ptr); |
| goto Cleanup; |
| end if; |
| |
| elsif Token = Tok_String_Literal then |
| Data := (Symbol => Symbol_Name, |
| Original => Original_Name, |
| On_The_Command_Line => False, |
| Is_A_String => True, |
| Value => String_Literal_Id); |
| |
| Scan.all; |
| |
| if Token not in Tok_End_Of_Line | Tok_EOF then |
| Error_Msg ("extraneous text in definition", Token_Ptr); |
| goto Cleanup; |
| end if; |
| |
| elsif Token in Tok_End_Of_Line | Tok_EOF then |
| Data := (Symbol => Symbol_Name, |
| Original => Original_Name, |
| On_The_Command_Line => False, |
| Is_A_String => False, |
| Value => Null_String_Id); |
| |
| else |
| Value_Start := Token_Ptr; |
| Value_End := Token_Ptr - 1; |
| Scan_Ptr := Token_Ptr; |
| |
| Value_Chars_Loop : |
| loop |
| Ch := Sinput.Source (Scan_Ptr); |
| |
| case Ch is |
| when '_' | '.' | '0' .. '9' | 'a' .. 'z' | 'A' .. 'Z' => |
| Value_End := Scan_Ptr; |
| Scan_Ptr := Scan_Ptr + 1; |
| |
| when ' ' | HT | VT | CR | LF | FF => |
| exit Value_Chars_Loop; |
| |
| when others => |
| Error_Msg ("illegal character", Scan_Ptr); |
| goto Cleanup; |
| end case; |
| end loop Value_Chars_Loop; |
| |
| Scan.all; |
| |
| if Token not in Tok_End_Of_Line | Tok_EOF then |
| Error_Msg ("extraneous text in definition", Token_Ptr); |
| goto Cleanup; |
| end if; |
| |
| Start_String; |
| |
| while Value_Start <= Value_End loop |
| Store_String_Char (Sinput.Source (Value_Start)); |
| Value_Start := Value_Start + 1; |
| end loop; |
| |
| Data := (Symbol => Symbol_Name, |
| Original => Original_Name, |
| On_The_Command_Line => False, |
| Is_A_String => False, |
| Value => End_String); |
| end if; |
| |
| -- Now that we have the value, get the symbol index |
| |
| Symbol := Index_Of (Symbol_Name); |
| |
| if Symbol /= No_Symbol then |
| |
| -- If we already have an entry for this symbol, replace it |
| -- with the new value, except if the symbol was declared on |
| -- the command line. |
| |
| if Mapping.Table (Symbol).On_The_Command_Line then |
| goto Continue; |
| end if; |
| |
| else |
| -- As it is the first time we see this symbol, create a new |
| -- entry in the table. |
| |
| if Mapping.Table = null then |
| Symbol_Table.Init (Mapping); |
| end if; |
| |
| Symbol_Table.Increment_Last (Mapping); |
| Symbol := Symbol_Table.Last (Mapping); |
| end if; |
| |
| Mapping.Table (Symbol) := Data; |
| goto Continue; |
| |
| <<Cleanup>> |
| Set_Ignore_Errors (To => True); |
| |
| while Token not in Tok_End_Of_Line | Tok_EOF loop |
| Scan.all; |
| end loop; |
| |
| Set_Ignore_Errors (To => False); |
| |
| <<Continue>> |
| null; |
| end if; |
| end loop Def_Line_Loop; |
| end Parse_Def_File; |
| |
| ---------------- |
| -- Preprocess -- |
| ---------------- |
| |
| procedure Preprocess (Source_Modified : out Boolean) is |
| Start_Of_Processing : Source_Ptr; |
| Cond : Boolean; |
| Preprocessor_Line : Boolean := False; |
| No_Error_Found : Boolean := True; |
| Modified : Boolean := False; |
| |
| procedure Output (From, To : Source_Ptr); |
| -- Output the characters with indexes From .. To in the buffer to the |
| -- output file. |
| |
| procedure Output_Line (From, To : Source_Ptr); |
| -- Output a line or the end of a line from the buffer to the output |
| -- file, followed by an end of line terminator. Depending on the value |
| -- of Deleting and the switches, the line may be commented out, blank or |
| -- not output at all. |
| |
| ------------ |
| -- Output -- |
| ------------ |
| |
| procedure Output (From, To : Source_Ptr) is |
| begin |
| for J in From .. To loop |
| Put_Char (Sinput.Source (J)); |
| end loop; |
| end Output; |
| |
| ----------------- |
| -- Output_Line -- |
| ----------------- |
| |
| procedure Output_Line (From, To : Source_Ptr) is |
| begin |
| if Deleting or else Preprocessor_Line then |
| if Blank_Deleted_Lines then |
| New_EOL.all; |
| |
| elsif Comment_Deleted_Lines then |
| Put_Char ('-'); |
| Put_Char ('-'); |
| Put_Char ('!'); |
| |
| if From < To then |
| Put_Char (' '); |
| Output (From, To); |
| end if; |
| |
| New_EOL.all; |
| end if; |
| |
| else |
| Output (From, To); |
| New_EOL.all; |
| end if; |
| end Output_Line; |
| |
| -- Start of processing for Preprocess |
| |
| begin |
| Start_Of_Processing := Scan_Ptr; |
| |
| -- First a call to Scan, because Initialize_Scanner is not doing it |
| |
| Scan.all; |
| |
| Input_Line_Loop : loop |
| exit Input_Line_Loop when Token = Tok_EOF; |
| |
| Preprocessor_Line := False; |
| |
| if Token /= Tok_End_Of_Line then |
| |
| -- Preprocessor line |
| |
| if Token = Tok_Special and then Special_Character = '#' then |
| Modified := True; |
| Preprocessor_Line := True; |
| Scan.all; |
| |
| case Token is |
| |
| -- #if |
| |
| when Tok_If => |
| declare |
| If_Ptr : constant Source_Ptr := Token_Ptr; |
| |
| begin |
| Scan.all; |
| Cond := Expression (not Deleting); |
| |
| -- Check for an eventual "then" |
| |
| if Token = Tok_Then then |
| Scan.all; |
| end if; |
| |
| -- It is an error to have trailing characters after |
| -- the condition or "then". |
| |
| if Token not in Tok_End_Of_Line | Tok_EOF then |
| Error_Msg |
| ("extraneous text on preprocessor line", |
| Token_Ptr); |
| No_Error_Found := False; |
| Go_To_End_Of_Line; |
| end if; |
| |
| declare |
| -- Set the initial state of this new "#if". This |
| -- must be done before incrementing the Last of |
| -- the table, otherwise function Deleting does |
| -- not report the correct value. |
| |
| New_State : constant Pp_State := |
| (If_Ptr => If_Ptr, |
| Else_Ptr => 0, |
| Deleting => Deleting |
| or else not Cond, |
| Match_Seen => Deleting or else Cond); |
| |
| begin |
| Pp_States.Increment_Last; |
| Pp_States.Table (Pp_States.Last) := New_State; |
| end; |
| end; |
| |
| -- #elsif |
| |
| when Tok_Elsif => |
| Cond := False; |
| |
| if Pp_States.Last = 0 |
| or else Pp_States.Table (Pp_States.Last).Else_Ptr /= 0 |
| then |
| Error_Msg ("no IF for this ELSIF", Token_Ptr); |
| No_Error_Found := False; |
| |
| else |
| Cond := |
| not Pp_States.Table (Pp_States.Last).Match_Seen; |
| end if; |
| |
| Scan.all; |
| Cond := Expression (Cond); |
| |
| -- Check for an eventual "then" |
| |
| if Token = Tok_Then then |
| Scan.all; |
| end if; |
| |
| -- It is an error to have trailing characters after the |
| -- condition or "then". |
| |
| if Token not in Tok_End_Of_Line | Tok_EOF then |
| Error_Msg |
| ("extraneous text on preprocessor line", |
| Token_Ptr); |
| No_Error_Found := False; |
| |
| Go_To_End_Of_Line; |
| end if; |
| |
| -- Depending on the value of the condition, set the new |
| -- values of Deleting and Match_Seen. |
| |
| if Pp_States.Last > 0 then |
| if Pp_States.Table (Pp_States.Last).Match_Seen then |
| Pp_States.Table (Pp_States.Last).Deleting := True; |
| else |
| if Cond then |
| Pp_States.Table (Pp_States.Last).Match_Seen := |
| True; |
| Pp_States.Table (Pp_States.Last).Deleting := |
| False; |
| end if; |
| end if; |
| end if; |
| |
| -- #else |
| |
| when Tok_Else => |
| if Pp_States.Last = 0 then |
| Error_Msg ("no IF for this ELSE", Token_Ptr); |
| No_Error_Found := False; |
| |
| elsif |
| Pp_States.Table (Pp_States.Last).Else_Ptr /= 0 |
| then |
| Error_Msg -- CODEFIX |
| ("duplicate ELSE line", Token_Ptr); |
| No_Error_Found := False; |
| end if; |
| |
| -- Set the possibly new values of Deleting and Match_Seen |
| |
| if Pp_States.Last > 0 then |
| if Pp_States.Table (Pp_States.Last).Match_Seen then |
| Pp_States.Table (Pp_States.Last).Deleting := |
| True; |
| |
| else |
| Pp_States.Table (Pp_States.Last).Match_Seen := |
| True; |
| Pp_States.Table (Pp_States.Last).Deleting := |
| False; |
| end if; |
| |
| -- Set the Else_Ptr to check for illegal #elsif later |
| |
| Pp_States.Table (Pp_States.Last).Else_Ptr := |
| Token_Ptr; |
| end if; |
| |
| Scan.all; |
| |
| -- Error of character present after "#else" |
| |
| if Token not in Tok_End_Of_Line | Tok_EOF then |
| Error_Msg |
| ("extraneous text on preprocessor line", |
| Token_Ptr); |
| No_Error_Found := False; |
| Go_To_End_Of_Line; |
| end if; |
| |
| -- #end if; |
| |
| when Tok_End => |
| if Pp_States.Last = 0 then |
| Error_Msg ("no IF for this END", Token_Ptr); |
| No_Error_Found := False; |
| end if; |
| |
| Scan.all; |
| |
| -- Ignore all recoverable errors if Relaxed_RM_Semantics |
| |
| if Relaxed_RM_Semantics then |
| null; |
| |
| elsif Token /= Tok_If then |
| Error_Msg -- CODEFIX |
| ("IF expected", Token_Ptr); |
| No_Error_Found := False; |
| |
| else |
| Scan.all; |
| |
| if Token /= Tok_Semicolon then |
| Error_Msg -- CODEFIX |
| ("`;` Expected", Token_Ptr); |
| No_Error_Found := False; |
| |
| else |
| Scan.all; |
| |
| -- Error of character present after "#end if;" |
| |
| if Token not in Tok_End_Of_Line | Tok_EOF then |
| Error_Msg |
| ("extraneous text on preprocessor line", |
| Token_Ptr); |
| No_Error_Found := False; |
| end if; |
| end if; |
| end if; |
| |
| -- In case of one of the errors above, skip the tokens |
| -- until the end of line is reached. |
| |
| Go_To_End_Of_Line; |
| |
| -- Decrement the depth of the #if stack |
| |
| if Pp_States.Last > 0 then |
| Pp_States.Decrement_Last; |
| end if; |
| |
| -- Illegal preprocessor line |
| |
| when others => |
| if Pp_States.Last = 0 then |
| Error_Msg -- CODEFIX |
| ("IF expected", Token_Ptr); |
| No_Error_Found := False; |
| |
| elsif Relaxed_RM_Semantics |
| and then Get_Name_String (Token_Name) = "endif" |
| then |
| -- In relaxed mode, accept "endif" instead of |
| -- "end if". |
| |
| -- Decrement the depth of the #if stack |
| |
| if Pp_States.Last > 0 then |
| Pp_States.Decrement_Last; |
| end if; |
| elsif Pp_States.Table (Pp_States.Last).Else_Ptr = 0 then |
| Error_Msg |
| ("IF, ELSIF, ELSE, or `END IF` expected", |
| Token_Ptr); |
| No_Error_Found := False; |
| |
| else |
| Error_Msg ("IF or `END IF` expected", Token_Ptr); |
| No_Error_Found := False; |
| end if; |
| |
| -- Skip to the end of this illegal line |
| |
| Go_To_End_Of_Line; |
| end case; |
| |
| -- Not a preprocessor line |
| |
| else |
| -- Do not report errors for those lines, even if there are |
| -- Ada parsing errors. |
| |
| Set_Ignore_Errors (To => True); |
| |
| if Deleting then |
| Go_To_End_Of_Line; |
| |
| else |
| while Token not in Tok_End_Of_Line | Tok_EOF loop |
| if Token = Tok_Special |
| and then Special_Character = '$' |
| then |
| Modified := True; |
| |
| declare |
| Dollar_Ptr : constant Source_Ptr := Token_Ptr; |
| Symbol : Symbol_Id; |
| |
| begin |
| Scan.all; |
| Change_Reserved_Keyword_To_Symbol; |
| |
| if Token = Tok_Identifier |
| and then Token_Ptr = Dollar_Ptr + 1 |
| then |
| -- $symbol |
| |
| Symbol := Index_Of (Token_Name); |
| |
| -- If symbol exists, replace by its value |
| |
| if Symbol /= No_Symbol then |
| Output (Start_Of_Processing, Dollar_Ptr - 1); |
| Start_Of_Processing := Scan_Ptr; |
| String_To_Name_Buffer |
| (Mapping.Table (Symbol).Value); |
| |
| if Mapping.Table (Symbol).Is_A_String then |
| |
| -- Value is an Ada string |
| |
| Put_Char ('"'); |
| |
| for J in 1 .. Name_Len loop |
| Put_Char (Name_Buffer (J)); |
| |
| if Name_Buffer (J) = '"' then |
| Put_Char ('"'); |
| end if; |
| end loop; |
| |
| Put_Char ('"'); |
| |
| else |
| -- Value is a sequence of characters, not |
| -- an Ada string. |
| |
| for J in 1 .. Name_Len loop |
| Put_Char (Name_Buffer (J)); |
| end loop; |
| end if; |
| end if; |
| end if; |
| end; |
| end if; |
| |
| Scan.all; |
| end loop; |
| end if; |
| |
| Set_Ignore_Errors (To => False); |
| end if; |
| end if; |
| |
| pragma Assert (Token in Tok_End_Of_Line | Tok_EOF); |
| |
| -- At this point, the token is either end of line or EOF. The line to |
| -- possibly output stops just before the token. |
| |
| Output_Line (Start_Of_Processing, Token_Ptr - 1); |
| |
| -- If we are at the end of a line, the scan pointer is at the first |
| -- non-blank character (may not be the first character of the line), |
| -- so we have to deduct Start_Of_Processing from the token pointer. |
| |
| if Token = Tok_End_Of_Line then |
| if Sinput.Source (Token_Ptr) = ASCII.CR |
| and then Sinput.Source (Token_Ptr + 1) = ASCII.LF |
| then |
| Start_Of_Processing := Token_Ptr + 2; |
| else |
| Start_Of_Processing := Token_Ptr + 1; |
| end if; |
| end if; |
| |
| -- Now, scan the first token of the next line. If the token is EOF, |
| -- the scan pointer will not move, and the token will still be EOF. |
| |
| Set_Ignore_Errors (To => True); |
| Scan.all; |
| Set_Ignore_Errors (To => False); |
| end loop Input_Line_Loop; |
| |
| -- Report an error for any missing some "#end if;" |
| |
| for Level in reverse 1 .. Pp_States.Last loop |
| Error_Msg ("no `END IF` for this IF", Pp_States.Table (Level).If_Ptr); |
| No_Error_Found := False; |
| end loop; |
| |
| Source_Modified := No_Error_Found and Modified; |
| end Preprocess; |
| |
| ----------------- |
| -- Setup_Hooks -- |
| ----------------- |
| |
| procedure Setup_Hooks |
| (Error_Msg : Error_Msg_Proc; |
| Scan : Scan_Proc; |
| Set_Ignore_Errors : Set_Ignore_Errors_Proc; |
| Put_Char : Put_Char_Proc; |
| New_EOL : New_EOL_Proc) |
| is |
| begin |
| pragma Assert (Already_Initialized); |
| |
| Prep.Error_Msg := Error_Msg; |
| Prep.Scan := Scan; |
| Prep.Set_Ignore_Errors := Set_Ignore_Errors; |
| Prep.Put_Char := Put_Char; |
| Prep.New_EOL := New_EOL; |
| end Setup_Hooks; |
| |
| end Prep; |