diff options
author | kenner <kenner@138bc75d-0d04-0410-961f-82ee72b054a4> | 2001-10-02 14:23:52 +0000 |
---|---|---|
committer | kenner <kenner@138bc75d-0d04-0410-961f-82ee72b054a4> | 2001-10-02 14:23:52 +0000 |
commit | 49d882a7d8c985758c04737e801f6028d5b7240f (patch) | |
tree | 0509e847916fc00cfe5c311617e039600afa9622 | |
parent | 83cce46b47d48de4c71b02a20f5bf36296a48568 (diff) | |
download | gcc-49d882a7d8c985758c04737e801f6028d5b7240f.tar.gz |
New Language: Ada
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@45956 138bc75d-0d04-0410-961f-82ee72b054a4
57 files changed, 38371 insertions, 0 deletions
diff --git a/gcc/ada/par-ch10.adb b/gcc/ada/par-ch10.adb new file mode 100644 index 00000000000..a4fa121ed16 --- /dev/null +++ b/gcc/ada/par-ch10.adb @@ -0,0 +1,1080 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 1 0 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.115 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +with Fname; use Fname; +with Fname.UF; use Fname.UF; +with Hostparm; use Hostparm; +with Uname; use Uname; + +separate (Par) +package body Ch10 is + + -- Local functions, used only in this chapter + + function P_Context_Clause return List_Id; + function P_Subunit return Node_Id; + + function Set_Location return Source_Ptr; + -- The current compilation unit starts with Token at Token_Ptr. This + -- function determines the corresponding source location for the start + -- of the unit, including any preceding comment lines. + + procedure Unit_Display + (Cunit : Node_Id; + Loc : Source_Ptr; + SR_Present : Boolean); + -- This procedure is used to generate a line of output for the a unit in + -- the source program. Cunit is the node for the compilation unit, and + -- Loc is the source location for the start of the unit in the source + -- file (which is not necessarily the Sloc of the Cunit node). This + -- output is written to the standard output file for use by gnatchop. + + procedure Unit_Location (Sind : Source_File_Index; Loc : Source_Ptr); + -- This routine has the same calling sequence as Unit_Display, but + -- it outputs only the line number and offset of the location, Loc, + -- using Cunit to obtain the proper source file index. + + ------------------------- + -- 10.1.1 Compilation -- + ------------------------- + + -- COMPILATION ::= {COMPILATION_UNIT} + + -- There is no specific parsing routine for a compilation, since we only + -- permit a single compilation in a source file, so there is no explicit + -- occurrence of compilations as such (our representation of a compilation + -- is a series of separate source files). + + ------------------------------ + -- 10.1.1 Compilation unit -- + ------------------------------ + + -- COMPILATION_UNIT ::= + -- CONTEXT_CLAUSE LIBRARY_ITEM + -- | CONTEXT_CLAUSE SUBUNIT + + -- LIBRARY_ITEM ::= + -- private LIBRARY_UNIT_DECLARATION + -- | LIBRARY_UNIT_BODY + -- | [private] LIBRARY_UNIT_RENAMING_DECLARATION + + -- LIBRARY_UNIT_DECLARATION ::= + -- SUBPROGRAM_DECLARATION | PACKAGE_DECLARATION + -- | GENERIC_DECLARATION | GENERIC_INSTANTIATION + + -- LIBRARY_UNIT_RENAMING_DECLARATION ::= + -- PACKAGE_RENAMING_DECLARATION + -- | GENERIC_RENAMING_DECLARATION + -- | SUBPROGRAM_RENAMING_DECLARATION + + -- LIBRARY_UNIT_BODY ::= SUBPROGRAM_BODY | PACKAGE_BODY + + -- Error recovery: cannot raise Error_Resync. If an error occurs, tokens + -- are skipped up to the next possible beginning of a compilation unit. + + -- Note: if only configuration pragmas are found, Empty is returned + + -- Note: in syntax-only mode, it is possible for P_Compilation_Unit + -- to return strange things that are not really compilation units. + -- This is done to help out gnatchop when it is faced with nonsense. + + function P_Compilation_Unit return Node_Id is + Scan_State : Saved_Scan_State; + Body_Node : Node_Id; + Specification_Node : Node_Id; + Unit_Node : Node_Id; + Comp_Unit_Node : Node_Id; + Name_Node : Node_Id; + Item : Node_Id; + Private_Sloc : Source_Ptr := No_Location; + Config_Pragmas : List_Id; + P : Node_Id; + SR_Present : Boolean; + + Cunit_Error_Flag : Boolean := False; + -- This flag is set True if we have to scan for a compilation unit + -- token. It is used to ensure clean termination in such cases by + -- not insisting on being at the end of file, and, in the sytax only + -- case by not scanning for additional compilation units. + + Cunit_Location : Source_Ptr; + -- Location of unit for unit identification output (List_Unit option) + + begin + Num_Library_Units := Num_Library_Units + 1; + + -- Set location of the compilation unit if unit list option set + -- and we are in syntax check only mode + + if List_Units and then Operating_Mode = Check_Syntax then + Cunit_Location := Set_Location; + else + Cunit_Location := No_Location; + end if; + + -- Deal with initial pragmas + + Config_Pragmas := No_List; + + -- If we have an initial Source_Reference pragma, then remember + -- the fact to generate an NR parameter in the output line. + + SR_Present := False; + + if Token = Tok_Pragma then + Save_Scan_State (Scan_State); + Item := P_Pragma; + + if Item = Error + or else Chars (Item) /= Name_Source_Reference + then + Restore_Scan_State (Scan_State); + + else + SR_Present := True; + + -- If first unit, record the file name for gnatchop use + + if Operating_Mode = Check_Syntax + and then List_Units + and then Num_Library_Units = 1 + then + Write_Str ("Source_Reference pragma for file """); + Write_Name (Full_Ref_Name (Current_Source_File)); + Write_Char ('"'); + Write_Eol; + end if; + + Config_Pragmas := New_List (Item); + end if; + end if; + + -- Scan out any configuration pragmas + + while Token = Tok_Pragma loop + Save_Scan_State (Scan_State); + Item := P_Pragma; + + if Item = Error + or else Chars (Item) > Last_Configuration_Pragma_Name + then + Restore_Scan_State (Scan_State); + exit; + end if; + + if Config_Pragmas = No_List then + Config_Pragmas := Empty_List; + + if Operating_Mode = Check_Syntax and then List_Units then + Write_Str ("Configuration pragmas at"); + Unit_Location (Current_Source_File, Cunit_Location); + Write_Eol; + end if; + end if; + + Append (Item, Config_Pragmas); + Cunit_Location := Set_Location; + end loop; + + -- Establish compilation unit node and scan context items + + Comp_Unit_Node := New_Node (N_Compilation_Unit, No_Location); + Set_Cunit (Current_Source_Unit, Comp_Unit_Node); + Set_Context_Items (Comp_Unit_Node, P_Context_Clause); + Set_Aux_Decls_Node + (Comp_Unit_Node, New_Node (N_Compilation_Unit_Aux, No_Location)); + + if Present (Config_Pragmas) then + + -- Check for case of only configuration pragmas present + + if Token = Tok_EOF + and then Is_Empty_List (Context_Items (Comp_Unit_Node)) + then + if Operating_Mode = Check_Syntax then + return Empty; + + else + Item := First (Config_Pragmas); + Error_Msg_N + ("cannot compile configuration pragmas with gcc", Item); + Error_Msg_N + ("use gnatchop -c to process configuration pragmas!", Item); + raise Unrecoverable_Error; + end if; + + -- Otherwise configuration pragmas are simply prepended to the + -- context of the current unit. + + else + Append_List (Context_Items (Comp_Unit_Node), Config_Pragmas); + Set_Context_Items (Comp_Unit_Node, Config_Pragmas); + end if; + end if; + + -- Check for PRIVATE. Note that for the moment we allow this in + -- Ada_83 mode, since we do not yet know if we are compiling a + -- predefined unit, and if we are then it would be allowed anyway. + + if Token = Tok_Private then + Private_Sloc := Token_Ptr; + Set_Keyword_Casing (Current_Source_File, Determine_Token_Casing); + if Style_Check then Style.Check_Indentation; end if; + + Save_Scan_State (Scan_State); -- at PRIVATE + Scan; -- past PRIVATE + + if Token = Tok_Separate then + Error_Msg_SP ("cannot have private subunits!"); + + elsif Token = Tok_Package then + Scan; -- past PACKAGE + + if Token = Tok_Body then + Restore_Scan_State (Scan_State); -- to PRIVATE + Error_Msg_SC ("cannot have private package body!"); + Scan; -- ignore PRIVATE + + else + Restore_Scan_State (Scan_State); -- to PRIVATE + Scan; -- past PRIVATE + Set_Private_Present (Comp_Unit_Node, True); + end if; + + elsif Token = Tok_Procedure + or else Token = Tok_Function + or else Token = Tok_Generic + then + Set_Private_Present (Comp_Unit_Node, True); + end if; + end if; + + -- Loop to find our way to a compilation unit token + + loop + exit when Token in Token_Class_Cunit and then Token /= Tok_With; + + exit when Bad_Spelling_Of (Tok_Package) + or else Bad_Spelling_Of (Tok_Function) + or else Bad_Spelling_Of (Tok_Generic) + or else Bad_Spelling_Of (Tok_Separate) + or else Bad_Spelling_Of (Tok_Procedure); + + -- Allow task and protected for nice error recovery purposes + + exit when Token = Tok_Task + or else Token = Tok_Protected; + + if Token = Tok_With then + Error_Msg_SC ("misplaced WITH"); + Append_List (P_Context_Clause, Context_Items (Comp_Unit_Node)); + + elsif Bad_Spelling_Of (Tok_With) then + Append_List (P_Context_Clause, Context_Items (Comp_Unit_Node)); + + else + Error_Msg_SC ("compilation unit expected"); + Cunit_Error_Flag := True; + Resync_Cunit; + + -- If we are at an end of file, then just quit, the above error + -- message was complaint enough. + + if Token = Tok_EOF then + return Error; + end if; + end if; + end loop; + + -- We have a compilation unit token, so that's a reasonable choice for + -- determining the standard casing convention used for keywords in case + -- it hasn't already been done on seeing a WITH or PRIVATE. + + Set_Keyword_Casing (Current_Source_File, Determine_Token_Casing); + if Style_Check then Style.Check_Indentation; end if; + + -- Remaining processing depends on particular type of compilation unit + + if Token = Tok_Package then + + -- A common error is to omit the body keyword after package. We can + -- often diagnose this early on (before getting loads of errors from + -- contained subprogram bodies), by knowing that that the file we + -- are compiling has a name that requires a body to be found. + + -- However, we do not do this check if we are operating in syntax + -- checking only mode, because in that case there may be multiple + -- units in the same file, and the file name is not a reliable guide. + + Save_Scan_State (Scan_State); + Scan; -- past Package keyword + + if Token /= Tok_Body + and then Operating_Mode /= Check_Syntax + and then + Get_Expected_Unit_Type + (File_Name (Current_Source_File)) = Expect_Body + then + Error_Msg_BC ("keyword BODY expected here [see file name]"); + Restore_Scan_State (Scan_State); + Set_Unit (Comp_Unit_Node, P_Package (Pf_Pbod)); + else + Restore_Scan_State (Scan_State); + Set_Unit (Comp_Unit_Node, P_Package (Pf_Decl_Gins_Pbod_Rnam)); + end if; + + elsif Token = Tok_Generic then + Set_Unit (Comp_Unit_Node, P_Generic); + + elsif Token = Tok_Separate then + Set_Unit (Comp_Unit_Node, P_Subunit); + + elsif Token = Tok_Procedure + or else Token = Tok_Function + then + Set_Unit (Comp_Unit_Node, P_Subprogram (Pf_Decl_Gins_Pbod_Rnam)); + + -- A little bit of an error recovery check here. If we just scanned + -- a subprogram declaration (as indicated by an SIS entry being + -- active), then if the following token is BEGIN or an identifier, + -- or a token which can reasonably start a declaration but cannot + -- start a compilation unit, then we assume that the semicolon in + -- the declaration should have been IS. + + if SIS_Entry_Active then + + if Token = Tok_Begin + or else Token = Tok_Identifier + or else Token in Token_Class_Deckn + then + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Sloc := SIS_Sloc; + Scope.Table (Scope.Last).Ecol := SIS_Ecol; + Scope.Table (Scope.Last).Lreq := False; + SIS_Entry_Active := False; + + -- If we had a missing semicolon in the declaration, then + -- change the message to from <missing ";"> to <missing "is"> + + if SIS_Missing_Semicolon_Message /= No_Error_Msg then + Change_Error_Text -- Replace: "missing "";"" " + (SIS_Missing_Semicolon_Message, "missing IS"); + + -- Otherwise we saved the semicolon position, so complain + + else + Error_Msg (""";"" should be IS", SIS_Semicolon_Sloc); + end if; + + Body_Node := Unit (Comp_Unit_Node); + Specification_Node := Specification (Body_Node); + Change_Node (Body_Node, N_Subprogram_Body); + Set_Specification (Body_Node, Specification_Node); + Parse_Decls_Begin_End (Body_Node); + Set_Unit (Comp_Unit_Node, Body_Node); + end if; + + -- If we scanned a subprogram body, make sure we did not have private + + elsif Private_Sloc /= No_Location + and then Nkind (Unit (Comp_Unit_Node)) /= N_Function_Instantiation + and then Nkind (Unit (Comp_Unit_Node)) /= N_Procedure_Instantiation + then + Error_Msg ("cannot have private subprogram body", Private_Sloc); + + -- P_Subprogram can yield an abstract subprogram, but this cannot + -- be a compilation unit. Treat as a subprogram declaration. + + elsif + Nkind (Unit (Comp_Unit_Node)) = N_Abstract_Subprogram_Declaration + then + Error_Msg_N + ("compilation unit cannot be abstract subprogram", + Unit (Comp_Unit_Node)); + + Unit_Node := + New_Node (N_Subprogram_Declaration, Sloc (Comp_Unit_Node)); + Set_Specification (Unit_Node, + Specification (Unit (Comp_Unit_Node))); + Set_Unit (Comp_Unit_Node, Unit_Node); + end if; + + -- Otherwise we have TASK. This is not really an acceptable token, + -- but we accept it to improve error recovery. + + elsif Token = Tok_Task then + Scan; -- Past TASK + + if Token = Tok_Type then + Error_Msg_SP + ("task type cannot be used as compilation unit"); + else + Error_Msg_SP + ("task declaration cannot be used as compilation unit"); + end if; + + -- If in check syntax mode, accept the task anyway. This is done + -- particularly to improve the behavior of GNATCHOP in this case. + + if Operating_Mode = Check_Syntax then + Set_Unit (Comp_Unit_Node, P_Task); + + -- If not in syntax only mode, treat this as horrible error + + else + Cunit_Error_Flag := True; + return Error; + end if; + + else pragma Assert (Token = Tok_Protected); + Scan; -- Past PROTECTED + + if Token = Tok_Type then + Error_Msg_SP + ("protected type cannot be used as compilation unit"); + else + Error_Msg_SP + ("protected declaration cannot be used as compilation unit"); + end if; + + -- If in check syntax mode, accept protected anyway. This is done + -- particularly to improve the behavior of GNATCHOP in this case. + + if Operating_Mode = Check_Syntax then + Set_Unit (Comp_Unit_Node, P_Protected); + + -- If not in syntax only mode, treat this as horrible error + + else + Cunit_Error_Flag := True; + return Error; + end if; + end if; + + -- Here is where locate the compilation unit entity. This is a little + -- tricky, since it is buried in various places. + + Unit_Node := Unit (Comp_Unit_Node); + + -- Another error from which it is hard to recover + + if Nkind (Unit_Node) = N_Subprogram_Body_Stub + or else Nkind (Unit_Node) = N_Package_Body_Stub + then + Cunit_Error_Flag := True; + return Error; + end if; + + -- Only try this if we got an OK unit! + + if Unit_Node /= Error then + if Nkind (Unit_Node) = N_Subunit then + Unit_Node := Proper_Body (Unit_Node); + end if; + + if Nkind (Unit_Node) in N_Generic_Declaration then + Unit_Node := Specification (Unit_Node); + end if; + + if Nkind (Unit_Node) = N_Package_Declaration + or else Nkind (Unit_Node) = N_Subprogram_Declaration + or else Nkind (Unit_Node) = N_Subprogram_Body + or else Nkind (Unit_Node) = N_Subprogram_Renaming_Declaration + then + Unit_Node := Specification (Unit_Node); + + elsif Nkind (Unit_Node) = N_Subprogram_Renaming_Declaration then + if Ada_83 then + Error_Msg_N + ("(Ada 83) library unit renaming not allowed", Unit_Node); + end if; + end if; + + if Nkind (Unit_Node) = N_Task_Body + or else Nkind (Unit_Node) = N_Protected_Body + or else Nkind (Unit_Node) = N_Task_Type_Declaration + or else Nkind (Unit_Node) = N_Protected_Type_Declaration + or else Nkind (Unit_Node) = N_Single_Task_Declaration + or else Nkind (Unit_Node) = N_Single_Protected_Declaration + then + Name_Node := Defining_Identifier (Unit_Node); + else + Name_Node := Defining_Unit_Name (Unit_Node); + end if; + + Set_Sloc (Comp_Unit_Node, Sloc (Name_Node)); + Set_Sloc (Aux_Decls_Node (Comp_Unit_Node), Sloc (Name_Node)); + + -- Set Entity field in file table. Easier now that we have name! + -- Note that this is also skipped if we had a bad unit + + if Nkind (Name_Node) = N_Defining_Program_Unit_Name then + Set_Cunit_Entity + (Current_Source_Unit, Defining_Identifier (Name_Node)); + else + Set_Cunit_Entity (Current_Source_Unit, Name_Node); + end if; + + Set_Unit_Name + (Current_Source_Unit, Get_Unit_Name (Unit (Comp_Unit_Node))); + + -- If we had a bad unit, make sure the fatal flag is set in the file + -- table entry, since this is surely a fatal error and also set our + -- flag to inhibit the requirement that we be at end of file. + + else + Cunit_Error_Flag := True; + Set_Fatal_Error (Current_Source_Unit); + end if; + + -- Clear away any missing semicolon indication, we are done with that + -- unit, so what's done is done, and we don't want anything hanging + -- around from the attempt to parse it! + + SIS_Entry_Active := False; + + -- Scan out pragmas after unit + + while Token = Tok_Pragma loop + Save_Scan_State (Scan_State); + + -- If we are in syntax scan mode allowing multiple units, then + -- start the next unit if we encounter a configuration pragma, + -- or a source reference pragma. We take care not to actually + -- scan the pragma in this case since we don't want it to take + -- effect for the current unit. + + if Operating_Mode = Check_Syntax then + Scan; -- past Pragma + + if Token = Tok_Identifier + and then + (Token_Name in + First_Pragma_Name .. Last_Configuration_Pragma_Name + or else Token_Name = Name_Source_Reference) + then + Restore_Scan_State (Scan_State); -- to Pragma + exit; + end if; + end if; + + -- Otherwise eat the pragma, it definitely belongs with the + -- current unit, and not with the following unit. + + Restore_Scan_State (Scan_State); -- to Pragma + P := P_Pragma; + + if No (Pragmas_After (Aux_Decls_Node (Comp_Unit_Node))) then + Set_Pragmas_After + (Aux_Decls_Node (Comp_Unit_Node), New_List); + end if; + + Append (P, Pragmas_After (Aux_Decls_Node (Comp_Unit_Node))); + end loop; + + -- Cancel effect of any outstanding pragma Warnings (Off) + + Set_Warnings_Mode_On (Scan_Ptr); + + -- Ada 83 error checks + + if Ada_83 then + + -- Check we did not with any child units + + Item := First (Context_Items (Comp_Unit_Node)); + + while Present (Item) loop + if Nkind (Item) = N_With_Clause + and then Nkind (Name (Item)) /= N_Identifier + then + Error_Msg_N ("(Ada 83) child units not allowed", Item); + end if; + + Next (Item); + end loop; + + -- Check that we did not have a PRIVATE keyword present + + if Private_Present (Comp_Unit_Node) then + Error_Msg + ("(Ada 83) private units not allowed", Private_Sloc); + end if; + end if; + + -- If no serious error, then output possible unit information line + -- for gnatchop if we are in syntax only, list units mode. + + if not Cunit_Error_Flag + and then List_Units + and then Operating_Mode = Check_Syntax + then + Unit_Display (Comp_Unit_Node, Cunit_Location, SR_Present); + end if; + + -- And now we should be at the end of file + + if Token /= Tok_EOF then + + -- If we already had to scan for a compilation unit, then don't + -- give any further error message, since it just sems to make + -- things worse, and we already gave a serious error message. + + if Cunit_Error_Flag then + null; + + -- If we are in check syntax mode, then we allow multiple units + -- so we just return with Token not set to Tok_EOF and no message. + + elsif Operating_Mode = Check_Syntax then + return Comp_Unit_Node; + + -- Otherwise we have an error. We suppress the error message + -- if we already had a fatal error, since this stops junk + -- cascaded messages in some situations. + + else + if not Fatal_Error (Current_Source_Unit) then + + if Token in Token_Class_Cunit then + Error_Msg_SC + ("end of file expected, " & + "file can have only one compilation unit"); + + else + Error_Msg_SC ("end of file expected"); + end if; + end if; + end if; + + -- Skip tokens to end of file, so that the -gnatl listing + -- will be complete in this situation, but no error checking + -- other than that provided at the token level. + + while Token /= Tok_EOF loop + Scan; + end loop; + + return Error; + + -- Normal return (we were at the end of file as expected) + + else + return Comp_Unit_Node; + end if; + + exception + + -- An error resync is a serious bomb, so indicate result unit no good + + when Error_Resync => + Set_Fatal_Error (Current_Source_Unit); + return Error; + + end P_Compilation_Unit; + + -------------------------- + -- 10.1.1 Library Item -- + -------------------------- + + -- Parsed by P_Compilation_Unit (10.1.1) + + -------------------------------------- + -- 10.1.1 Library Unit Declaration -- + -------------------------------------- + + -- Parsed by P_Compilation_Unit (10.1.1) + + ------------------------------------------------ + -- 10.1.1 Library Unit Renaming Declaration -- + ------------------------------------------------ + + -- Parsed by P_Compilation_Unit (10.1.1) + + ------------------------------- + -- 10.1.1 Library Unit Body -- + ------------------------------- + + -- Parsed by P_Compilation_Unit (10.1.1) + + ------------------------------ + -- 10.1.1 Parent Unit Name -- + ------------------------------ + + -- Parsed (as a name) by its parent construct + + ---------------------------- + -- 10.1.2 Context Clause -- + ---------------------------- + + -- CONTEXT_CLAUSE ::= {CONTEXT_ITEM} + + -- CONTEXT_ITEM ::= WITH_CLAUSE | USE_CLAUSE | WITH_TYPE_CLAUSE + + -- WITH_CLAUSE ::= + -- with library_unit_NAME {,library_unit_NAME}; + + -- WITH_TYPE_CLAUSE ::= + -- with type type_NAME is access; | with type type_NAME is tagged; + + -- Error recovery: Cannot raise Error_Resync + + function P_Context_Clause return List_Id is + Item_List : List_Id; + With_Node : Node_Id; + First_Flag : Boolean; + + begin + Item_List := New_List; + + -- Get keyword casing from WITH keyword in case not set yet + + if Token = Tok_With then + Set_Keyword_Casing (Current_Source_File, Determine_Token_Casing); + end if; + + -- Loop through context items + + loop + if Style_Check then Style.Check_Indentation; end if; + + -- Gather any pragmas appearing in the context clause + + P_Pragmas_Opt (Item_List); + + -- Processing for WITH clause + + if Token = Tok_With then + Scan; -- past WITH + + if Token = Tok_Type then + + -- WITH TYPE is an extension + + if not Extensions_Allowed then + Error_Msg_SP ("`WITH TYPE` is a non-standard extension"); + + if OpenVMS then + Error_Msg_SP + ("\unit must be compiled with " & + "'/'E'X'T'E'N'S'I'O'N'S'_'A'L'L'O'W'E'D qualifier"); + else + Error_Msg_SP + ("\unit must be compiled with -gnatX switch"); + end if; + end if; + + Scan; -- past TYPE + With_Node := New_Node (N_With_Type_Clause, Token_Ptr); + Append (With_Node, Item_List); + Set_Name (With_Node, P_Qualified_Simple_Name); + + T_Is; + + if Token = Tok_Tagged then + Set_Tagged_Present (With_Node); + Scan; + + elsif Token = Tok_Access then + Scan; + + else + Error_Msg_SC ("expect tagged or access qualifier"); + end if; + + TF_Semicolon; + + else + First_Flag := True; + + -- Loop through names in one with clause, generating a separate + -- N_With_Clause node for each nam encountered. + + loop + With_Node := New_Node (N_With_Clause, Token_Ptr); + Append (With_Node, Item_List); + + -- Note that we allow with'ing of child units, even in + -- Ada 83 mode, since presumably if this is not desired, + -- then the compilation of the child unit itself is the + -- place where such an "error" should be caught. + + Set_Name (With_Node, P_Qualified_Simple_Name); + Set_First_Name (With_Node, First_Flag); + First_Flag := False; + exit when Token /= Tok_Comma; + Scan; -- past comma + end loop; + + Set_Last_Name (With_Node, True); + TF_Semicolon; + end if; + + -- Processing for USE clause + + elsif Token = Tok_Use then + Append (P_Use_Clause, Item_List); + + -- Anything else is end of context clause + + else + exit; + end if; + end loop; + + return Item_List; + end P_Context_Clause; + + -------------------------- + -- 10.1.2 Context Item -- + -------------------------- + + -- Parsed by P_Context_Clause (10.1.2) + + ------------------------- + -- 10.1.2 With Clause -- + ------------------------- + + -- Parsed by P_Context_Clause (10.1.2) + + ----------------------- + -- 10.1.3 Body Stub -- + ----------------------- + + -- Subprogram stub parsed by P_Subprogram (6.1) + -- Package stub parsed by P_Package (7.1) + -- Task stub parsed by P_Task (9.1) + -- Protected stub parsed by P_Protected (9.4) + + ---------------------------------- + -- 10.1.3 Subprogram Body Stub -- + ---------------------------------- + + -- Parsed by P_Subprogram (6.1) + + ------------------------------- + -- 10.1.3 Package Body Stub -- + ------------------------------- + + -- Parsed by P_Package (7.1) + + ---------------------------- + -- 10.1.3 Task Body Stub -- + ---------------------------- + + -- Parsed by P_Task (9.1) + + --------------------------------- + -- 10.1.3 Protected Body Stub -- + --------------------------------- + + -- Parsed by P_Protected (9.4) + + --------------------- + -- 10.1.3 Subunit -- + --------------------- + + -- SUBUNIT ::= separate (PARENT_UNIT_NAME) PROPER_BODY + + -- PARENT_UNIT_NAME ::= NAME + + -- The caller has checked that the initial token is SEPARATE + + -- Error recovery: cannot raise Error_Resync + + function P_Subunit return Node_Id is + Subunit_Node : Node_Id; + Body_Node : Node_Id; + + begin + Subunit_Node := New_Node (N_Subunit, Token_Ptr); + Body_Node := Error; -- in case no good body found + Scan; -- past SEPARATE; + + T_Left_Paren; + Set_Name (Subunit_Node, P_Qualified_Simple_Name); + T_Right_Paren; + + if Token = Tok_Semicolon then + Error_Msg_SC ("unexpected semicolon ignored"); + Scan; + end if; + + if Token = Tok_Function or else Token = Tok_Procedure then + Body_Node := P_Subprogram (Pf_Pbod); + + elsif Token = Tok_Package then + Body_Node := P_Package (Pf_Pbod); + + elsif Token = Tok_Protected then + Scan; -- past PROTECTED + + if Token = Tok_Body then + Body_Node := P_Protected; + else + Error_Msg_AP ("BODY expected"); + return Error; + end if; + + elsif Token = Tok_Task then + Scan; -- past TASK + + if Token = Tok_Body then + Body_Node := P_Task; + else + Error_Msg_AP ("BODY expected"); + return Error; + end if; + + else + Error_Msg_SC ("proper body expected"); + return Error; + end if; + + Set_Proper_Body (Subunit_Node, Body_Node); + return Subunit_Node; + + end P_Subunit; + + ------------------ + -- Set_Location -- + ------------------ + + function Set_Location return Source_Ptr is + Physical : Boolean; + Loc : Source_Ptr; + Scan_State : Saved_Scan_State; + + begin + -- A special check. If the first token is pragma, and this is a + -- Source_Reference pragma, then do NOT eat previous comments, since + -- the Source_Reference pragma is required to be the first line in + -- the source file. + + if Token = Tok_Pragma then + Save_Scan_State (Scan_State); + Scan; -- past Pragma + + if Token = Tok_Identifier + and then Token_Name = Name_Source_Reference + then + Restore_Scan_State (Scan_State); + return Token_Ptr; + end if; + + Restore_Scan_State (Scan_State); + end if; + + -- Otherwise acquire previous comments and blank lines + + if Prev_Token = No_Token then + return Source_First (Current_Source_File); + + else + Loc := Prev_Token_Ptr; + loop + exit when Loc = Token_Ptr; + + if Source (Loc) in Line_Terminator then + Skip_Line_Terminators (Loc, Physical); + exit when Physical; + end if; + + Loc := Loc + 1; + end loop; + + return Loc; + end if; + end Set_Location; + + ------------------ + -- Unit_Display -- + ------------------ + + -- The format of the generated line, as expected by GNATCHOP is + + -- Unit {unit} line {line}, file offset {offs} [, SR], file name {file} + + -- where + + -- {unit} unit name with terminating (spec) or (body) + -- {line} starting line number + -- {offs} offset to start of text in file + -- {file} source file name + + -- The SR parameter is present only if a source reference pragma was + -- scanned for this unit. The significance is that gnatchop should not + -- attempt to add another one. + + procedure Unit_Display + (Cunit : Node_Id; + Loc : Source_Ptr; + SR_Present : Boolean) + is + Unum : constant Unit_Number_Type := Get_Cunit_Unit_Number (Cunit); + Sind : constant Source_File_Index := Source_Index (Unum); + Unam : constant Unit_Name_Type := Unit_Name (Unum); + + begin + if List_Units then + Write_Str ("Unit "); + Write_Unit_Name (Unit_Name (Unum)); + Unit_Location (Sind, Loc); + + if SR_Present then + Write_Str (", SR"); + end if; + + Write_Str (", file name "); + Write_Name (Get_File_Name (Unam, Nkind (Unit (Cunit)) = N_Subunit)); + Write_Eol; + end if; + end Unit_Display; + + ------------------- + -- Unit_Location -- + ------------------- + + procedure Unit_Location (Sind : Source_File_Index; Loc : Source_Ptr) is + Line : constant Logical_Line_Number := Get_Logical_Line_Number (Loc); + -- Should the above be the physical line number ??? + + begin + Write_Str (" line "); + Write_Int (Int (Line)); + + Write_Str (", file offset "); + Write_Int (Int (Loc) - Int (Source_First (Sind))); + end Unit_Location; + +end Ch10; diff --git a/gcc/ada/par-ch11.adb b/gcc/ada/par-ch11.adb new file mode 100644 index 00000000000..8b59c54ea13 --- /dev/null +++ b/gcc/ada/par-ch11.adb @@ -0,0 +1,246 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 1 1 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.22 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +with Sinfo.CN; use Sinfo.CN; + +separate (Par) +package body Ch11 is + + -- Local functions, used only in this chapter + + function P_Exception_Handler return Node_Id; + function P_Exception_Choice return Node_Id; + + --------------------------------- + -- 11.1 Exception Declaration -- + --------------------------------- + + -- Parsed by P_Identifier_Declaration (3.3.1) + + ------------------------------------------ + -- 11.2 Handled Sequence Of Statements -- + ------------------------------------------ + + -- HANDLED_SEQUENCE_OF_STATEMENTS ::= + -- SEQUENCE_OF_STATEMENTS + -- [exception + -- EXCEPTION_HANDLER + -- {EXCEPTION_HANDLER}] + + -- Error_Recovery : Cannot raise Error_Resync + + function P_Handled_Sequence_Of_Statements return Node_Id is + Handled_Stmt_Seq_Node : Node_Id; + + begin + Handled_Stmt_Seq_Node := + New_Node (N_Handled_Sequence_Of_Statements, Token_Ptr); + Set_Statements + (Handled_Stmt_Seq_Node, P_Sequence_Of_Statements (SS_Extm_Sreq)); + + if Token = Tok_Exception then + Scan; -- past EXCEPTION + Set_Exception_Handlers + (Handled_Stmt_Seq_Node, Parse_Exception_Handlers); + end if; + + return Handled_Stmt_Seq_Node; + end P_Handled_Sequence_Of_Statements; + + ----------------------------- + -- 11.2 Exception Handler -- + ----------------------------- + + -- EXCEPTION_HANDLER ::= + -- when [CHOICE_PARAMETER_SPECIFICATION :] + -- EXCEPTION_CHOICE {| EXCEPTION_CHOICE} => + -- SEQUENCE_OF_STATEMENTS + + -- CHOICE_PARAMETER_SPECIFICATION ::= DEFINING_IDENTIFIER + + -- Error recovery: cannot raise Error_Resync + + function P_Exception_Handler return Node_Id is + Scan_State : Saved_Scan_State; + Handler_Node : Node_Id; + Choice_Param_Node : Node_Id; + + begin + Handler_Node := New_Node (N_Exception_Handler, Token_Ptr); + T_When; + + -- Test for possible choice parameter present + + if Token = Tok_Identifier then + Choice_Param_Node := Token_Node; + Save_Scan_State (Scan_State); -- at identifier + Scan; -- past identifier + + if Token = Tok_Colon then + if Ada_83 then + Error_Msg_SP ("(Ada 83) choice parameter not allowed!"); + end if; + + Scan; -- past : + Change_Identifier_To_Defining_Identifier (Choice_Param_Node); + Set_Choice_Parameter (Handler_Node, Choice_Param_Node); + + elsif Token = Tok_Others then + Error_Msg_AP ("missing "":"""); + Change_Identifier_To_Defining_Identifier (Choice_Param_Node); + Set_Choice_Parameter (Handler_Node, Choice_Param_Node); + + else + Restore_Scan_State (Scan_State); -- to identifier + end if; + end if; + + -- Loop through exception choices + + Set_Exception_Choices (Handler_Node, New_List); + + loop + Append (P_Exception_Choice, Exception_Choices (Handler_Node)); + exit when Token /= Tok_Vertical_Bar; + Scan; -- past vertical bar + end loop; + + TF_Arrow; + Set_Statements (Handler_Node, P_Sequence_Of_Statements (SS_Sreq_Whtm)); + return Handler_Node; + end P_Exception_Handler; + + ------------------------------------------ + -- 11.2 Choice Parameter Specification -- + ------------------------------------------ + + -- Parsed by P_Exception_Handler (11.2) + + ---------------------------- + -- 11.2 Exception Choice -- + ---------------------------- + + -- EXCEPTION_CHOICE ::= exception_NAME | others + + -- Error recovery: cannot raise Error_Resync. If an error occurs, then the + -- scan pointer is advanced to the next arrow or vertical bar or semicolon. + + function P_Exception_Choice return Node_Id is + begin + + if Token = Tok_Others then + Scan; -- past OTHERS + return New_Node (N_Others_Choice, Prev_Token_Ptr); + + else + return P_Name; -- exception name + end if; + + exception + when Error_Resync => + Resync_Choice; + return Error; + end P_Exception_Choice; + + --------------------------- + -- 11.3 Raise Statement -- + --------------------------- + + -- RAISE_STATEMENT ::= raise [exception_NAME]; + + -- The caller has verified that the initial token is RAISE + + -- Error recovery: can raise Error_Resync + + function P_Raise_Statement return Node_Id is + Raise_Node : Node_Id; + + begin + Raise_Node := New_Node (N_Raise_Statement, Token_Ptr); + Scan; -- past RAISE + + if Token /= Tok_Semicolon then + Set_Name (Raise_Node, P_Name); + end if; + + TF_Semicolon; + return Raise_Node; + end P_Raise_Statement; + + ------------------------------ + -- Parse_Exception_Handlers -- + ------------------------------ + + -- This routine scans out a list of exception handlers appearing in a + -- construct as: + + -- exception + -- EXCEPTION_HANDLER {EXCEPTION_HANDLER} + + -- The caller has scanned out the EXCEPTION keyword + + -- Control returns after scanning the last exception handler, presumably + -- at the keyword END, but this is not checked in this routine. + + -- Error recovery: cannot raise Error_Resync + + function Parse_Exception_Handlers return List_Id is + Handler : Node_Id; + Handlers_List : List_Id; + Pragmas_List : List_Id; + + begin + Handlers_List := New_List; + P_Pragmas_Opt (Handlers_List); + + if Token = Tok_End then + Error_Msg_SC ("must have at least one exception handler!"); + + else + loop + Handler := P_Exception_Handler; + Pragmas_List := No_List; + Append (Handler, Handlers_List); + + -- Note: no need to check for pragmas here. Although the + -- syntax officially allows them in this position, they + -- will have been swallowed up as part of the statement + -- sequence of the handler we just scanned out. + + exit when Token /= Tok_When; + end loop; + end if; + + return Handlers_List; + end Parse_Exception_Handlers; + +end Ch11; diff --git a/gcc/ada/par-ch12.adb b/gcc/ada/par-ch12.adb new file mode 100644 index 00000000000..139243e67e2 --- /dev/null +++ b/gcc/ada/par-ch12.adb @@ -0,0 +1,882 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 1 2 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.46 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch12 is + + -- Local functions, used only in this chapter + + function P_Formal_Derived_Type_Definition return Node_Id; + function P_Formal_Discrete_Type_Definition return Node_Id; + function P_Formal_Fixed_Point_Definition return Node_Id; + function P_Formal_Floating_Point_Definition return Node_Id; + function P_Formal_Modular_Type_Definition return Node_Id; + function P_Formal_Package_Declaration return Node_Id; + function P_Formal_Private_Type_Definition return Node_Id; + function P_Formal_Signed_Integer_Type_Definition return Node_Id; + function P_Formal_Subprogram_Declaration return Node_Id; + function P_Formal_Type_Declaration return Node_Id; + function P_Formal_Type_Definition return Node_Id; + function P_Generic_Association return Node_Id; + + procedure P_Formal_Object_Declarations (Decls : List_Id); + -- Scans one or more formal object declarations and appends them to + -- Decls. Scans more than one declaration only in the case where the + -- source has a declaration with multiple defining identifiers. + + -------------------------------- + -- 12.1 Generic (also 8.5.5) -- + -------------------------------- + + -- This routine parses either one of the forms of a generic declaration + -- or a generic renaming declaration. + + -- GENERIC_DECLARATION ::= + -- GENERIC_SUBPROGRAM_DECLARATION | GENERIC_PACKAGE_DECLARATION + + -- GENERIC_SUBPROGRAM_DECLARATION ::= + -- GENERIC_FORMAL_PART SUBPROGRAM_SPECIFICATION; + + -- GENERIC_PACKAGE_DECLARATION ::= + -- GENERIC_FORMAL_PART PACKAGE_SPECIFICATION; + + -- GENERIC_FORMAL_PART ::= + -- generic {GENERIC_FORMAL_PARAMETER_DECLARATION | USE_CLAUSE} + + -- GENERIC_RENAMING_DECLARATION ::= + -- generic package DEFINING_PROGRAM_UNIT_NAME + -- renames generic_package_NAME + -- | generic procedure DEFINING_PROGRAM_UNIT_NAME + -- renames generic_procedure_NAME + -- | generic function DEFINING_PROGRAM_UNIT_NAME + -- renames generic_function_NAME + + -- GENERIC_FORMAL_PARAMETER_DECLARATION ::= + -- FORMAL_OBJECT_DECLARATION + -- | FORMAL_TYPE_DECLARATION + -- | FORMAL_SUBPROGRAM_DECLARATION + -- | FORMAL_PACKAGE_DECLARATION + + -- The caller has checked that the initial token is GENERIC + + -- Error recovery: can raise Error_Resync + + function P_Generic return Node_Id is + Gen_Sloc : constant Source_Ptr := Token_Ptr; + Gen_Decl : Node_Id; + Decl_Node : Node_Id; + Decls : List_Id; + Def_Unit : Node_Id; + Ren_Token : Token_Type; + Scan_State : Saved_Scan_State; + + begin + Scan; -- past GENERIC + + if Token = Tok_Private then + Error_Msg_SC ("PRIVATE goes before GENERIC, not after"); + Scan; -- past junk PRIVATE token + end if; + + Save_Scan_State (Scan_State); -- at token past GENERIC + + -- Check for generic renaming declaration case + + if Token = Tok_Package + or else Token = Tok_Function + or else Token = Tok_Procedure + then + Ren_Token := Token; + Scan; -- scan past PACKAGE, FUNCTION or PROCEDURE + + if Token = Tok_Identifier then + Def_Unit := P_Defining_Program_Unit_Name; + + Check_Misspelling_Of (Tok_Renames); + + if Token = Tok_Renames then + if Ren_Token = Tok_Package then + Decl_Node := New_Node + (N_Generic_Package_Renaming_Declaration, Gen_Sloc); + + elsif Ren_Token = Tok_Procedure then + Decl_Node := New_Node + (N_Generic_Procedure_Renaming_Declaration, Gen_Sloc); + + else -- Ren_Token = Tok_Function then + Decl_Node := New_Node + (N_Generic_Function_Renaming_Declaration, Gen_Sloc); + end if; + + Scan; -- past RENAMES + Set_Defining_Unit_Name (Decl_Node, Def_Unit); + Set_Name (Decl_Node, P_Name); + TF_Semicolon; + return Decl_Node; + end if; + end if; + end if; + + -- Fall through if this is *not* a generic renaming declaration + + Restore_Scan_State (Scan_State); + Decls := New_List; + + -- Loop through generic parameter declarations and use clauses + + Decl_Loop : loop + P_Pragmas_Opt (Decls); + Ignore (Tok_Private); + + if Token = Tok_Use then + Append (P_Use_Clause, Decls); + else + -- Parse a generic parameter declaration + + if Token = Tok_Identifier then + P_Formal_Object_Declarations (Decls); + + elsif Token = Tok_Type then + Append (P_Formal_Type_Declaration, Decls); + + elsif Token = Tok_With then + Scan; -- past WITH + + if Token = Tok_Package then + Append (P_Formal_Package_Declaration, Decls); + + elsif Token = Tok_Procedure or Token = Tok_Function then + Append (P_Formal_Subprogram_Declaration, Decls); + + else + Error_Msg_BC + ("FUNCTION, PROCEDURE or PACKAGE expected here"); + Resync_Past_Semicolon; + end if; + + elsif Token = Tok_Subtype then + Error_Msg_SC ("subtype declaration not allowed " & + "as generic parameter declaration!"); + Resync_Past_Semicolon; + + else + exit Decl_Loop; + end if; + end if; + + end loop Decl_Loop; + + -- Generic formal part is scanned, scan out subprogram or package spec + + if Token = Tok_Package then + Gen_Decl := New_Node (N_Generic_Package_Declaration, Gen_Sloc); + Set_Specification (Gen_Decl, P_Package (Pf_Spcn)); + else + Gen_Decl := New_Node (N_Generic_Subprogram_Declaration, Gen_Sloc); + Set_Specification (Gen_Decl, P_Subprogram_Specification); + TF_Semicolon; + end if; + + Set_Generic_Formal_Declarations (Gen_Decl, Decls); + return Gen_Decl; + end P_Generic; + + ------------------------------- + -- 12.1 Generic Declaration -- + ------------------------------- + + -- Parsed by P_Generic (12.1) + + ------------------------------------------ + -- 12.1 Generic Subprogram Declaration -- + ------------------------------------------ + + -- Parsed by P_Generic (12.1) + + --------------------------------------- + -- 12.1 Generic Package Declaration -- + --------------------------------------- + + -- Parsed by P_Generic (12.1) + + ------------------------------- + -- 12.1 Generic Formal Part -- + ------------------------------- + + -- Parsed by P_Generic (12.1) + + ------------------------------------------------- + -- 12.1 Generic Formal Parameter Declaration -- + ------------------------------------------------- + + -- Parsed by P_Generic (12.1) + + --------------------------------- + -- 12.3 Generic Instantiation -- + --------------------------------- + + -- Generic package instantiation parsed by P_Package (7.1) + -- Generic procedure instantiation parsed by P_Subprogram (6.1) + -- Generic function instantiation parsed by P_Subprogram (6.1) + + ------------------------------- + -- 12.3 Generic Actual Part -- + ------------------------------- + + -- GENERIC_ACTUAL_PART ::= + -- (GENERIC_ASSOCIATION {, GENERIC_ASSOCIATION}) + + -- Returns a list of generic associations, or Empty if none are present + + -- Error recovery: cannot raise Error_Resync + + function P_Generic_Actual_Part_Opt return List_Id is + Association_List : List_Id; + + begin + -- Figure out if a generic actual part operation is present. Clearly + -- there is no generic actual part if the current token is semicolon + + if Token = Tok_Semicolon then + return No_List; + + -- If we don't have a left paren, then we have an error, and the job + -- is to figure out whether a left paren or semicolon was intended. + -- We assume a missing left paren (and hence a generic actual part + -- present) if the current token is not on a new line, or if it is + -- indented from the subprogram token. Otherwise assume missing + -- semicolon (which will be diagnosed by caller) and no generic part + + elsif Token /= Tok_Left_Paren + and then Token_Is_At_Start_Of_Line + and then Start_Column <= Scope.Table (Scope.Last).Ecol + then + return No_List; + + -- Otherwise we have a generic actual part (either a left paren is + -- present, or we have decided that there must be a missing left paren) + + else + Association_List := New_List; + T_Left_Paren; + + loop + Append (P_Generic_Association, Association_List); + exit when not Comma_Present; + end loop; + + T_Right_Paren; + return Association_List; + end if; + + end P_Generic_Actual_Part_Opt; + + ------------------------------- + -- 12.3 Generic Association -- + ------------------------------- + + -- GENERIC_ASSOCIATION ::= + -- [generic_formal_parameter_SELECTOR_NAME =>] + -- EXPLICIT_GENERIC_ACTUAL_PARAMETER + + -- EXPLICIT_GENERIC_ACTUAL_PARAMETER ::= + -- EXPRESSION | variable_NAME | subprogram_NAME + -- | entry_NAME | SUBTYPE_MARK | package_instance_NAME + + -- Error recovery: cannot raise Error_Resync + + function P_Generic_Association return Node_Id is + Scan_State : Saved_Scan_State; + Param_Name_Node : Node_Id; + Generic_Assoc_Node : Node_Id; + + begin + Generic_Assoc_Node := New_Node (N_Generic_Association, Token_Ptr); + + if Token in Token_Class_Desig then + Param_Name_Node := Token_Node; + Save_Scan_State (Scan_State); -- at designator + Scan; -- past simple name or operator symbol + + if Token = Tok_Arrow then + Scan; -- past arrow + Set_Selector_Name (Generic_Assoc_Node, Param_Name_Node); + else + Restore_Scan_State (Scan_State); -- to designator + end if; + end if; + + Set_Explicit_Generic_Actual_Parameter (Generic_Assoc_Node, P_Expression); + return Generic_Assoc_Node; + end P_Generic_Association; + + --------------------------------------------- + -- 12.3 Explicit Generic Actual Parameter -- + --------------------------------------------- + + -- Parsed by P_Generic_Association (12.3) + + -------------------------------------- + -- 12.4 Formal Object Declarations -- + -------------------------------------- + + -- FORMAL_OBJECT_DECLARATION ::= + -- DEFINING_IDENTIFIER_LIST : + -- MODE SUBTYPE_MARK [:= DEFAULT_EXPRESSION]; + + -- The caller has checked that the initial token is an identifier + + -- Error recovery: cannot raise Error_Resync + + procedure P_Formal_Object_Declarations (Decls : List_Id) is + Decl_Node : Node_Id; + Scan_State : Saved_Scan_State; + Num_Idents : Nat; + Ident : Nat; + + Idents : array (Int range 1 .. 4096) of Entity_Id; + -- This array holds the list of defining identifiers. The upper bound + -- of 4096 is intended to be essentially infinite, and we do not even + -- bother to check for it being exceeded. + + begin + Idents (1) := P_Defining_Identifier; + Num_Idents := 1; + + while Comma_Present loop + Num_Idents := Num_Idents + 1; + Idents (Num_Idents) := P_Defining_Identifier; + end loop; + + T_Colon; + + -- If there are multiple identifiers, we repeatedly scan the + -- type and initialization expression information by resetting + -- the scan pointer (so that we get completely separate trees + -- for each occurrence). + + if Num_Idents > 1 then + Save_Scan_State (Scan_State); + end if; + + -- Loop through defining identifiers in list + + Ident := 1; + Ident_Loop : loop + Decl_Node := New_Node (N_Formal_Object_Declaration, Token_Ptr); + Set_Defining_Identifier (Decl_Node, Idents (Ident)); + P_Mode (Decl_Node); + Set_Subtype_Mark (Decl_Node, P_Subtype_Mark_Resync); + No_Constraint; + Set_Expression (Decl_Node, Init_Expr_Opt); + + if Ident > 1 then + Set_Prev_Ids (Decl_Node, True); + end if; + + if Ident < Num_Idents then + Set_More_Ids (Decl_Node, True); + end if; + + Append (Decl_Node, Decls); + + exit Ident_Loop when Ident = Num_Idents; + Ident := Ident + 1; + Restore_Scan_State (Scan_State); + end loop Ident_Loop; + + TF_Semicolon; + end P_Formal_Object_Declarations; + + ----------------------------------- + -- 12.5 Formal Type Declaration -- + ----------------------------------- + + -- FORMAL_TYPE_DECLARATION ::= + -- type DEFINING_IDENTIFIER [DISCRIMINANT_PART] + -- is FORMAL_TYPE_DEFINITION; + + -- The caller has checked that the initial token is TYPE + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Type_Declaration return Node_Id is + Decl_Node : Node_Id; + + begin + Decl_Node := New_Node (N_Formal_Type_Declaration, Token_Ptr); + Scan; -- past TYPE + Set_Defining_Identifier (Decl_Node, P_Defining_Identifier); + + if P_Unknown_Discriminant_Part_Opt then + Set_Unknown_Discriminants_Present (Decl_Node, True); + else + Set_Discriminant_Specifications + (Decl_Node, P_Known_Discriminant_Part_Opt); + end if; + + T_Is; + + Set_Formal_Type_Definition (Decl_Node, P_Formal_Type_Definition); + TF_Semicolon; + return Decl_Node; + end P_Formal_Type_Declaration; + + ---------------------------------- + -- 12.5 Formal Type Definition -- + ---------------------------------- + + -- FORMAL_TYPE_DEFINITION ::= + -- FORMAL_PRIVATE_TYPE_DEFINITION + -- | FORMAL_DERIVED_TYPE_DEFINITION + -- | FORMAL_DISCRETE_TYPE_DEFINITION + -- | FORMAL_SIGNED_INTEGER_TYPE_DEFINITION + -- | FORMAL_MODULAR_TYPE_DEFINITION + -- | FORMAL_FLOATING_POINT_DEFINITION + -- | FORMAL_ORDINARY_FIXED_POINT_DEFINITION + -- | FORMAL_DECIMAL_FIXED_POINT_DEFINITION + -- | FORMAL_ARRAY_TYPE_DEFINITION + -- | FORMAL_ACCESS_TYPE_DEFINITION + + -- FORMAL_ARRAY_TYPE_DEFINITION ::= ARRAY_TYPE_DEFINITION + + -- FORMAL_ACCESS_TYPE_DEFINITION ::= ACCESS_TYPE_DEFINITION + + function P_Formal_Type_Definition return Node_Id is + Scan_State : Saved_Scan_State; + + begin + if Token_Name = Name_Abstract then + Check_95_Keyword (Tok_Abstract, Tok_Tagged); + end if; + + if Token_Name = Name_Tagged then + Check_95_Keyword (Tok_Tagged, Tok_Private); + Check_95_Keyword (Tok_Tagged, Tok_Limited); + end if; + + case Token is + + -- Mostly we can tell what we have from the initial token. The one + -- exception is ABSTRACT, where we have to scan ahead to see if we + -- have a formal derived type or a formal private type definition. + + when Tok_Abstract => + Save_Scan_State (Scan_State); + Scan; -- past ABSTRACT + + if Token = Tok_New then + Restore_Scan_State (Scan_State); -- to ABSTRACT + return P_Formal_Derived_Type_Definition; + + else + Restore_Scan_State (Scan_State); -- to ABSTRACT + return P_Formal_Private_Type_Definition; + end if; + + when Tok_Private | Tok_Limited | Tok_Tagged => + return P_Formal_Private_Type_Definition; + + when Tok_New => + return P_Formal_Derived_Type_Definition; + + when Tok_Left_Paren => + return P_Formal_Discrete_Type_Definition; + + when Tok_Range => + return P_Formal_Signed_Integer_Type_Definition; + + when Tok_Mod => + return P_Formal_Modular_Type_Definition; + + when Tok_Digits => + return P_Formal_Floating_Point_Definition; + + when Tok_Delta => + return P_Formal_Fixed_Point_Definition; + + when Tok_Array => + return P_Array_Type_Definition; + + when Tok_Access => + return P_Access_Type_Definition; + + when Tok_Record => + Error_Msg_SC ("record not allowed in generic type definition!"); + Discard_Junk_Node (P_Record_Definition); + return Error; + + when others => + Error_Msg_BC ("expecting generic type definition here"); + Resync_Past_Semicolon; + return Error; + + end case; + end P_Formal_Type_Definition; + + -------------------------------------------- + -- 12.5.1 Formal Private Type Definition -- + -------------------------------------------- + + -- FORMAL_PRIVATE_TYPE_DEFINITION ::= + -- [[abstract] tagged] [limited] private + + -- The caller has checked the initial token is PRIVATE, ABSTRACT, + -- TAGGED or LIMITED + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Private_Type_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := New_Node (N_Formal_Private_Type_Definition, Token_Ptr); + + if Token = Tok_Abstract then + Scan; -- past ABSTRACT + + if Token_Name = Name_Tagged then + Check_95_Keyword (Tok_Tagged, Tok_Private); + Check_95_Keyword (Tok_Tagged, Tok_Limited); + end if; + + if Token /= Tok_Tagged then + Error_Msg_SP ("ABSTRACT must be followed by TAGGED"); + else + Set_Abstract_Present (Def_Node, True); + end if; + end if; + + if Token = Tok_Tagged then + Set_Tagged_Present (Def_Node, True); + Scan; -- past TAGGED + end if; + + if Token = Tok_Limited then + Set_Limited_Present (Def_Node, True); + Scan; -- past LIMITED + end if; + + Set_Sloc (Def_Node, Token_Ptr); + T_Private; + return Def_Node; + end P_Formal_Private_Type_Definition; + + -------------------------------------------- + -- 12.5.1 Formal Derived Type Definition -- + -------------------------------------------- + + -- FORMAL_DERIVED_TYPE_DEFINITION ::= + -- [abstract] new SUBTYPE_MARK [with private] + + -- The caller has checked the initial token(s) is/are NEW or ASTRACT NEW + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Derived_Type_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := New_Node (N_Formal_Derived_Type_Definition, Token_Ptr); + + if Token = Tok_Abstract then + Set_Abstract_Present (Def_Node); + Scan; -- past ABSTRACT + end if; + + Scan; -- past NEW; + Set_Subtype_Mark (Def_Node, P_Subtype_Mark); + No_Constraint; + + if Token = Tok_With then + Scan; -- past WITH + Set_Private_Present (Def_Node, True); + T_Private; + end if; + + return Def_Node; + end P_Formal_Derived_Type_Definition; + + --------------------------------------------- + -- 12.5.2 Formal Discrete Type Definition -- + --------------------------------------------- + + -- FORMAL_DISCRETE_TYPE_DEFINITION ::= (<>) + + -- The caller has checked the initial token is left paren + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Discrete_Type_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := New_Node (N_Formal_Discrete_Type_Definition, Token_Ptr); + Scan; -- past left paren + T_Box; + T_Right_Paren; + return Def_Node; + end P_Formal_Discrete_Type_Definition; + + --------------------------------------------------- + -- 12.5.2 Formal Signed Integer Type Definition -- + --------------------------------------------------- + + -- FORMAL_SIGNED_INTEGER_TYPE_DEFINITION ::= range <> + + -- The caller has checked the initial token is RANGE + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Signed_Integer_Type_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := + New_Node (N_Formal_Signed_Integer_Type_Definition, Token_Ptr); + Scan; -- past RANGE + T_Box; + return Def_Node; + end P_Formal_Signed_Integer_Type_Definition; + + -------------------------------------------- + -- 12.5.2 Formal Modular Type Definition -- + -------------------------------------------- + + -- FORMAL_MODULAR_TYPE_DEFINITION ::= mod <> + + -- The caller has checked the initial token is MOD + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Modular_Type_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := + New_Node (N_Formal_Modular_Type_Definition, Token_Ptr); + Scan; -- past MOD + T_Box; + return Def_Node; + end P_Formal_Modular_Type_Definition; + + ---------------------------------------------- + -- 12.5.2 Formal Floating Point Definition -- + ---------------------------------------------- + + -- FORMAL_FLOATING_POINT_DEFINITION ::= digits <> + + -- The caller has checked the initial token is DIGITS + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Floating_Point_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := + New_Node (N_Formal_Floating_Point_Definition, Token_Ptr); + Scan; -- past DIGITS + T_Box; + return Def_Node; + end P_Formal_Floating_Point_Definition; + + ------------------------------------------- + -- 12.5.2 Formal Fixed Point Definition -- + ------------------------------------------- + + -- This routine parses either a formal ordinary fixed point definition + -- or a formal decimal fixed point definition: + + -- FORMAL_ORDINARY_FIXED_POINT_DEFINITION ::= delta <> + + -- FORMAL_DECIMAL_FIXED_POINT_DEFINITION ::= delta <> digits <> + + -- The caller has checked the initial token is DELTA + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Fixed_Point_Definition return Node_Id is + Def_Node : Node_Id; + Delta_Sloc : Source_Ptr; + + begin + Delta_Sloc := Token_Ptr; + Scan; -- past DELTA + T_Box; + + if Token = Tok_Digits then + Def_Node := + New_Node (N_Formal_Decimal_Fixed_Point_Definition, Delta_Sloc); + Scan; -- past DIGITS + T_Box; + else + Def_Node := + New_Node (N_Formal_Ordinary_Fixed_Point_Definition, Delta_Sloc); + end if; + + return Def_Node; + end P_Formal_Fixed_Point_Definition; + + ---------------------------------------------------- + -- 12.5.2 Formal Ordinary Fixed Point Definition -- + ---------------------------------------------------- + + -- Parsed by P_Formal_Fixed_Point_Definition (12.5.2) + + --------------------------------------------------- + -- 12.5.2 Formal Decimal Fixed Point Definition -- + --------------------------------------------------- + + -- Parsed by P_Formal_Fixed_Point_Definition (12.5.2) + + ------------------------------------------ + -- 12.5.3 Formal Array Type Definition -- + ------------------------------------------ + + -- Parsed by P_Formal_Type_Definition (12.5) + + ------------------------------------------- + -- 12.5.4 Formal Access Type Definition -- + ------------------------------------------- + + -- Parsed by P_Formal_Type_Definition (12.5) + + ----------------------------------------- + -- 12.6 Formal Subprogram Declaration -- + ----------------------------------------- + + -- FORMAL_SUBPROGRAM_DECLARATION ::= + -- with SUBPROGRAM_SPECIFICATION [is SUBPROGRAM_DEFAULT]; + + -- SUBPROGRAM_DEFAULT ::= DEFAULT_NAME | <> + + -- DEFAULT_NAME ::= NAME + + -- The caller has checked that the initial tokens are WITH FUNCTION or + -- WITH PROCEDURE, and the initial WITH has been scanned out. + + -- Note: we separate this into two procedures because the name is allowed + -- to be an operator symbol for a function, but not for a procedure. + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Subprogram_Declaration return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := New_Node (N_Formal_Subprogram_Declaration, Prev_Token_Ptr); + Set_Specification (Def_Node, P_Subprogram_Specification); + + if Token = Tok_Is then + T_Is; -- past IS, skip extra IS or ";" + + if Token = Tok_Box then + Set_Box_Present (Def_Node, True); + Scan; -- past <> + + else + Set_Default_Name (Def_Node, P_Name); + end if; + + end if; + + T_Semicolon; + return Def_Node; + end P_Formal_Subprogram_Declaration; + + ------------------------------ + -- 12.6 Subprogram Default -- + ------------------------------ + + -- Parsed by P_Formal_Procedure_Declaration (12.6) + + ------------------------ + -- 12.6 Default Name -- + ------------------------ + + -- Parsed by P_Formal_Procedure_Declaration (12.6) + + -------------------------------------- + -- 12.7 Formal Package Declaration -- + -------------------------------------- + + -- FORMAL_PACKAGE_DECLARATION ::= + -- with package DEFINING_IDENTIFIER + -- is new generic_package_NAME FORMAL_PACKAGE_ACTUAL_PART; + + -- FORMAL_PACKAGE_ACTUAL_PART ::= + -- (<>) | [GENERIC_ACTUAL_PART] + + -- The caller has checked that the initial tokens are WITH PACKAGE, + -- and the initial WITH has been scanned out (so Token = Tok_Package). + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Package_Declaration return Node_Id is + Def_Node : Node_Id; + Scan_State : Saved_Scan_State; + + begin + Def_Node := New_Node (N_Formal_Package_Declaration, Prev_Token_Ptr); + Scan; -- past PACKAGE + Set_Defining_Identifier (Def_Node, P_Defining_Identifier); + T_Is; + T_New; + Set_Name (Def_Node, P_Qualified_Simple_Name); + + if Token = Tok_Left_Paren then + Save_Scan_State (Scan_State); -- at the left paren + Scan; -- past the left paren + + if Token = Tok_Box then + Set_Box_Present (Def_Node, True); + Scan; -- past box + T_Right_Paren; + + else + Restore_Scan_State (Scan_State); -- to the left paren + Set_Generic_Associations (Def_Node, P_Generic_Actual_Part_Opt); + end if; + end if; + + T_Semicolon; + return Def_Node; + end P_Formal_Package_Declaration; + + -------------------------------------- + -- 12.7 Formal Package Actual Part -- + -------------------------------------- + + -- Parsed by P_Formal_Package_Declaration (12.7) + +end Ch12; diff --git a/gcc/ada/par-ch13.adb b/gcc/ada/par-ch13.adb new file mode 100644 index 00000000000..03bd7bf1275 --- /dev/null +++ b/gcc/ada/par-ch13.adb @@ -0,0 +1,441 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 1 3 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.34 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch13 is + + -- Local functions, used only in this chapter + + function P_Component_Clause return Node_Id; + function P_Mod_Clause return Node_Id; + + -------------------------------------------- + -- 13.1 Representation Clause (also I.7) -- + -------------------------------------------- + + -- REPRESENTATION_CLAUSE ::= + -- ATTRIBUTE_DEFINITION_CLAUSE + -- | ENUMERATION_REPRESENTATION_CLAUSE + -- | RECORD_REPRESENTATION_CLAUSE + -- | AT_CLAUSE + + -- ATTRIBUTE_DEFINITION_CLAUSE ::= + -- for LOCAL_NAME'ATTRIBUTE_DESIGNATOR use EXPRESSION; + -- | for LOCAL_NAME'ATTRIBUTE_DESIGNATOR use NAME; + + -- Note: in Ada 83, the expression must be a simple expression + + -- AT_CLAUSE ::= for DIRECT_NAME use at EXPRESSION; + + -- Note: in Ada 83, the expression must be a simple expression + + -- ENUMERATION_REPRESENTATION_CLAUSE ::= + -- for first_subtype_LOCAL_NAME use ENUMERATION_AGGREGATE; + + -- ENUMERATION_AGGREGATE ::= ARRAY_AGGREGATE + + -- RECORD_REPRESENTATION_CLAUSE ::= + -- for first_subtype_LOCAL_NAME use + -- record [MOD_CLAUSE] + -- {COMPONENT_CLAUSE} + -- end record; + + -- Note: for now we allow only a direct name as the local name in the + -- above constructs. This probably needs changing later on ??? + + -- The caller has checked that the initial token is FOR + + -- Error recovery: cannot raise Error_Resync, if an error occurs, + -- the scan is repositioned past the next semicolon. + + function P_Representation_Clause return Node_Id is + For_Loc : Source_Ptr; + Name_Node : Node_Id; + Prefix_Node : Node_Id; + Attr_Name : Name_Id; + Identifier_Node : Node_Id; + Rep_Clause_Node : Node_Id; + Expr_Node : Node_Id; + Record_Items : List_Id; + + begin + For_Loc := Token_Ptr; + Scan; -- past FOR + + -- Note that the name in a representation clause is always a simple + -- name, even in the attribute case, see AI-300 which made this so! + + Identifier_Node := P_Identifier; + + -- Check case of qualified name to give good error message + + if Token = Tok_Dot then + Error_Msg_SC + ("representation clause requires simple name!"); + + loop + exit when Token /= Tok_Dot; + Scan; -- past dot + Discard_Junk_Node (P_Identifier); + end loop; + end if; + + -- Attribute Definition Clause + + if Token = Tok_Apostrophe then + + -- Allow local names of the form a'b'.... This enables + -- us to parse class-wide streams attributes correctly. + + Name_Node := Identifier_Node; + while Token = Tok_Apostrophe loop + + Scan; -- past apostrophe + + Identifier_Node := Token_Node; + Attr_Name := No_Name; + + if Token = Tok_Identifier then + Attr_Name := Token_Name; + + if not Is_Attribute_Name (Attr_Name) then + Signal_Bad_Attribute; + end if; + + if Style_Check then + Style.Check_Attribute_Name (False); + end if; + + -- Here for case of attribute designator is not an identifier + + else + if Token = Tok_Delta then + Attr_Name := Name_Delta; + + elsif Token = Tok_Digits then + Attr_Name := Name_Digits; + + elsif Token = Tok_Access then + Attr_Name := Name_Access; + + else + Error_Msg_AP ("attribute designator expected"); + raise Error_Resync; + end if; + + if Style_Check then + Style.Check_Attribute_Name (True); + end if; + end if; + + -- We come here with an OK attribute scanned, and the + -- corresponding Attribute identifier node stored in Ident_Node. + + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Attribute_Reference, Prev_Token_Ptr); + Set_Prefix (Name_Node, Prefix_Node); + Set_Attribute_Name (Name_Node, Attr_Name); + Scan; + end loop; + + Rep_Clause_Node := New_Node (N_Attribute_Definition_Clause, For_Loc); + Set_Name (Rep_Clause_Node, Prefix_Node); + Set_Chars (Rep_Clause_Node, Attr_Name); + T_Use; + + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Expression (Rep_Clause_Node, Expr_Node); + + else + TF_Use; + Rep_Clause_Node := Empty; + + -- AT follows USE (At Clause) + + if Token = Tok_At then + Scan; -- past AT + Rep_Clause_Node := New_Node (N_At_Clause, For_Loc); + Set_Identifier (Rep_Clause_Node, Identifier_Node); + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Expression (Rep_Clause_Node, Expr_Node); + + -- RECORD follows USE (Record Representation Clause) + + elsif Token = Tok_Record then + Record_Items := P_Pragmas_Opt; + Rep_Clause_Node := + New_Node (N_Record_Representation_Clause, For_Loc); + Set_Identifier (Rep_Clause_Node, Identifier_Node); + + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Record; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scan; -- past RECORD + Record_Items := P_Pragmas_Opt; + + -- Possible Mod Clause + + if Token = Tok_At then + Set_Mod_Clause (Rep_Clause_Node, P_Mod_Clause); + Set_Pragmas_Before (Mod_Clause (Rep_Clause_Node), Record_Items); + Record_Items := P_Pragmas_Opt; + end if; + + if No (Record_Items) then + Record_Items := New_List; + end if; + + Set_Component_Clauses (Rep_Clause_Node, Record_Items); + + -- Loop through component clauses + + loop + if Token not in Token_Class_Name then + exit when Check_End; + end if; + + Append (P_Component_Clause, Record_Items); + P_Pragmas_Opt (Record_Items); + end loop; + + -- Left paren follows USE (Enumeration Representation Clause) + + elsif Token = Tok_Left_Paren then + Rep_Clause_Node := + New_Node (N_Enumeration_Representation_Clause, For_Loc); + Set_Identifier (Rep_Clause_Node, Identifier_Node); + Set_Array_Aggregate (Rep_Clause_Node, P_Aggregate); + + -- Some other token follows FOR (invalid representation clause) + + else + Error_Msg_SC ("invalid representation clause"); + raise Error_Resync; + end if; + end if; + + TF_Semicolon; + return Rep_Clause_Node; + + exception + when Error_Resync => + Resync_Past_Semicolon; + return Error; + + end P_Representation_Clause; + + ---------------------- + -- 13.1 Local Name -- + ---------------------- + + -- Local name is always parsed by its parent. In the case of its use in + -- pragmas, the check for a local name is handled in Par.Prag and allows + -- all the possible forms of local name. For the uses in chapter 13, we + -- currently only allow a direct name, but this should probably change??? + + --------------------------- + -- 13.1 At Clause (I.7) -- + --------------------------- + + -- Parsed by P_Representation_Clause (13.1) + + --------------------------------------- + -- 13.3 Attribute Definition Clause -- + --------------------------------------- + + -- Parsed by P_Representation_Clause (13.1) + + --------------------------------------------- + -- 13.4 Enumeration Representation Clause -- + --------------------------------------------- + + -- Parsed by P_Representation_Clause (13.1) + + --------------------------------- + -- 13.4 Enumeration Aggregate -- + --------------------------------- + + -- Parsed by P_Representation_Clause (13.1) + + ------------------------------------------ + -- 13.5.1 Record Representation Clause -- + ------------------------------------------ + + -- Parsed by P_Representation_Clause (13.1) + + ------------------------------ + -- 13.5.1 Mod Clause (I.8) -- + ------------------------------ + + -- MOD_CLAUSE ::= at mod static_EXPRESSION; + + -- Note: in Ada 83, the expression must be a simple expression + + -- The caller has checked that the initial Token is AT + + -- Error recovery: cannot raise Error_Resync + + -- Note: the caller is responsible for setting the Pragmas_Before field + + function P_Mod_Clause return Node_Id is + Mod_Node : Node_Id; + Expr_Node : Node_Id; + + begin + Mod_Node := New_Node (N_Mod_Clause, Token_Ptr); + Scan; -- past AT + T_Mod; + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Expression (Mod_Node, Expr_Node); + TF_Semicolon; + return Mod_Node; + end P_Mod_Clause; + + ------------------------------ + -- 13.5.1 Component Clause -- + ------------------------------ + + -- COMPONENT_CLAUSE ::= + -- COMPONENT_CLAUSE_COMPONENT_NAME at POSITION + -- range FIRST_BIT .. LAST_BIT; + + -- COMPONENT_CLAUSE_COMPONENT_NAME ::= + -- component_DIRECT_NAME + -- | component_DIRECT_NAME'ATTRIBUTE_DESIGNATOR + -- | FIRST_SUBTYPE_DIRECT_NAME'ATTRIBUTE_DESIGNATOR + + -- POSITION ::= static_EXPRESSION + + -- Note: in Ada 83, the expression must be a simple expression + + -- FIRST_BIT ::= static_SIMPLE_EXPRESSION + -- LAST_BIT ::= static_SIMPLE_EXPRESSION + + -- Note: the AARM V2.0 grammar has an error at this point, it uses + -- EXPRESSION instead of SIMPLE_EXPRESSION for FIRST_BIT and LAST_BIT + + -- Error recovery: cannot raise Error_Resync + + function P_Component_Clause return Node_Id is + Component_Node : Node_Id; + Comp_Name : Node_Id; + Expr_Node : Node_Id; + + begin + Component_Node := New_Node (N_Component_Clause, Token_Ptr); + Comp_Name := P_Name; + + if Nkind (Comp_Name) = N_Identifier + or else Nkind (Comp_Name) = N_Attribute_Reference + then + Set_Component_Name (Component_Node, Comp_Name); + else + Error_Msg_N + ("component name must be direct name or attribute", Comp_Name); + Set_Component_Name (Component_Node, Error); + end if; + + Set_Sloc (Component_Node, Token_Ptr); + T_At; + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Position (Component_Node, Expr_Node); + T_Range; + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_First_Bit (Component_Node, Expr_Node); + T_Dot_Dot; + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Last_Bit (Component_Node, Expr_Node); + TF_Semicolon; + return Component_Node; + end P_Component_Clause; + + ---------------------- + -- 13.5.1 Position -- + ---------------------- + + -- Parsed by P_Component_Clause (13.5.1) + + ----------------------- + -- 13.5.1 First Bit -- + ----------------------- + + -- Parsed by P_Component_Clause (13.5.1) + + ---------------------- + -- 13.5.1 Last Bit -- + ---------------------- + + -- Parsed by P_Component_Clause (13.5.1) + + -------------------------- + -- 13.8 Code Statement -- + -------------------------- + + -- CODE_STATEMENT ::= QUALIFIED_EXPRESSION + + -- On entry the caller has scanned the SUBTYPE_MARK (passed in as the + -- single argument, and the scan points to the apostrophe. + + -- Error recovery: can raise Error_Resync + + function P_Code_Statement (Subtype_Mark : Node_Id) return Node_Id is + Node1 : Node_Id; + + begin + Scan; -- past apostrophe + + -- If left paren, then we have a possible code statement + + if Token = Tok_Left_Paren then + Node1 := New_Node (N_Code_Statement, Sloc (Subtype_Mark)); + Set_Expression (Node1, P_Qualified_Expression (Subtype_Mark)); + TF_Semicolon; + return Node1; + + -- Otherwise we have an illegal range attribute. Note that P_Name + -- ensures that Token = Tok_Range is the only possibility left here. + + else -- Token = Tok_Range + Error_Msg_SC ("RANGE attribute illegal here!"); + raise Error_Resync; + end if; + + end P_Code_Statement; + +end Ch13; diff --git a/gcc/ada/par-ch2.adb b/gcc/ada/par-ch2.adb new file mode 100644 index 00000000000..0eeacead811 --- /dev/null +++ b/gcc/ada/par-ch2.adb @@ -0,0 +1,405 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 2 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.35 $ -- +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch2 is + + -- Local functions, used only in this chapter + + function P_Pragma_Argument_Association return Node_Id; + + --------------------- + -- 2.3 Identifier -- + --------------------- + + -- IDENTIFIER ::= LETTER {[UNDERLINE] LETTER_OR_DIGIT} + + -- LETTER_OR_DIGIT ::= IDENTIFIER_LETTER | DIGIT + + -- An IDENTIFIER shall not be a reserved word + + -- Error recovery: can raise Error_Resync (cannot return Error) + + function P_Identifier return Node_Id is + Ident_Node : Node_Id; + + begin + -- All set if we do indeed have an identifier + + if Token = Tok_Identifier then + Ident_Node := Token_Node; + Scan; -- past Identifier + return Ident_Node; + + -- If we have a reserved identifier, manufacture an identifier with + -- a corresponding name after posting an appropriate error message + + elsif Is_Reserved_Identifier then + Scan_Reserved_Identifier (Force_Msg => False); + Ident_Node := Token_Node; + Scan; -- past the node + return Ident_Node; + + -- Otherwise we have junk that cannot be interpreted as an identifier + + else + T_Identifier; -- to give message + raise Error_Resync; + end if; + end P_Identifier; + + -------------------------- + -- 2.3 Letter Or Digit -- + -------------------------- + + -- Parsed by P_Identifier (2.3) + + -------------------------- + -- 2.4 Numeric Literal -- + -------------------------- + + -- NUMERIC_LITERAL ::= DECIMAL_LITERAL | BASED_LITERAL + + -- Numeric literal is returned by the scanner as either + -- Tok_Integer_Literal or Tok_Real_Literal + + ---------------------------- + -- 2.4.1 Decimal Literal -- + ---------------------------- + + -- DECIMAL_LITERAL ::= NUMERAL [.NUMERAL] [EXPONENT] + + -- Handled by scanner as part of numeric lIteral handing (see 2.4) + + -------------------- + -- 2.4.1 Numeral -- + -------------------- + + -- NUMERAL ::= DIGIT {[UNDERLINE] DIGIT} + + -- Handled by scanner as part of numeric literal handling (see 2.4) + + --------------------- + -- 2.4.1 Exponent -- + --------------------- + + -- EXPONENT ::= E [+] NUMERAL | E - NUMERAL + + -- Handled by scanner as part of numeric literal handling (see 2.4) + + -------------------------- + -- 2.4.2 Based Literal -- + -------------------------- + + -- BASED_LITERAL ::= + -- BASE # BASED_NUMERAL [.BASED_NUMERAL] # [EXPONENT] + + -- Handled by scanner as part of numeric literal handling (see 2.4) + + ----------------- + -- 2.4.2 Base -- + ----------------- + + -- BASE ::= NUMERAL + + -- Handled by scanner as part of numeric literal handling (see 2.4) + + -------------------------- + -- 2.4.2 Based Numeral -- + -------------------------- + + -- BASED_NUMERAL ::= + -- EXTENDED_DIGIT {[UNDERLINE] EXTENDED_DIGIT} + + -- Handled by scanner as part of numeric literal handling (see 2.4) + + --------------------------- + -- 2.4.2 Extended Digit -- + --------------------------- + + -- EXTENDED_DIGIT ::= DIGIT | A | B | C | D | E | F + + -- Handled by scanner as part of numeric literal handling (see 2.4) + + ---------------------------- + -- 2.5 Character Literal -- + ---------------------------- + + -- CHARACTER_LITERAL ::= ' GRAPHIC_CHARACTER ' + + -- Handled by the scanner and returned as Tok_Character_Literal + + ------------------------- + -- 2.6 String Literal -- + ------------------------- + + -- STRING LITERAL ::= "{STRING_ELEMENT}" + + -- Handled by the scanner and returned as Tok_Character_Literal + -- or if the string looks like an operator as Tok_Operator_Symbol. + + ------------------------- + -- 2.6 String Element -- + ------------------------- + + -- STRING_ELEMENT ::= "" | non-quotation_mark_GRAPHIC_CHARACTER + + -- A STRING_ELEMENT is either a pair of quotation marks ("), + -- or a single GRAPHIC_CHARACTER other than a quotation mark. + + -- Handled by scanner as part of string literal handling (see 2.4) + + ------------------ + -- 2.7 Comment -- + ------------------ + + -- A COMMENT starts with two adjacent hyphens and extends up to the + -- end of the line. A COMMENT may appear on any line of a program. + + -- Handled by the scanner which simply skips past encountered comments + + ----------------- + -- 2.8 Pragma -- + ----------------- + + -- PRAGMA ::= pragma IDENTIFIER + -- [(PRAGMA_ARGUMENT_ASSOCIATION {, PRAGMA_ARGUMENT_ASSOCIATION})]; + + -- The caller has checked that the initial token is PRAGMA + + -- Error recovery: cannot raise Error_Resync + + -- One special piece of processing is needed in this routine. As described + -- in the section on "Handling semicolon used in place of IS" in module + -- Parse, the parser detects the case of missing subprogram bodies to + -- allow recovery from this syntactic error. Pragma INTERFACE (and, for + -- Ada 95, pragma IMPORT) can appear in place of the body. The parser must + -- recognize the use of these two pragmas in this context, otherwise it + -- will think there are missing bodies, and try to change ; to IS, when + -- in fact the bodies ARE present, supplied by these pragmas. + + function P_Pragma return Node_Id is + + Interface_Check_Required : Boolean := False; + -- Set True if check of pragma INTERFACE is required + + Import_Check_Required : Boolean := False; + -- Set True if check of pragma IMPORT is required + + Arg_Count : Int := 0; + -- Number of argument associations processed + + Pragma_Node : Node_Id; + Pragma_Name : Name_Id; + Semicolon_Loc : Source_Ptr; + Ident_Node : Node_Id; + Assoc_Node : Node_Id; + + begin + Pragma_Node := New_Node (N_Pragma, Token_Ptr); + Scan; -- past PRAGMA + Pragma_Name := Token_Name; + + if Style_Check then + Style.Check_Pragma_Name; + end if; + + Ident_Node := P_Identifier; + Set_Chars (Pragma_Node, Pragma_Name); + Delete_Node (Ident_Node); + + -- See if special INTERFACE/IMPORT check is required + + if SIS_Entry_Active then + Interface_Check_Required := (Pragma_Name = Name_Interface); + Import_Check_Required := (Pragma_Name = Name_Import); + else + Interface_Check_Required := False; + Import_Check_Required := False; + end if; + + -- Scan arguments. We assume that arguments are present if there is + -- a left paren, or if a semicolon is missing and there is another + -- token on the same line as the pragma name. + + if Token = Tok_Left_Paren + or else (Token /= Tok_Semicolon + and then not Token_Is_At_Start_Of_Line) + then + Set_Pragma_Argument_Associations (Pragma_Node, New_List); + T_Left_Paren; + + loop + Arg_Count := Arg_Count + 1; + Assoc_Node := P_Pragma_Argument_Association; + + if Arg_Count = 2 + and then (Interface_Check_Required or else Import_Check_Required) + then + -- Here is where we cancel the SIS active status if this pragma + -- supplies a body for the currently active subprogram spec. + + if Nkind (Expression (Assoc_Node)) in N_Direct_Name + and then Chars (Expression (Assoc_Node)) = Chars (SIS_Labl) + then + SIS_Entry_Active := False; + end if; + end if; + + Append (Assoc_Node, Pragma_Argument_Associations (Pragma_Node)); + exit when Token /= Tok_Comma; + Scan; -- past comma + end loop; + + T_Right_Paren; + end if; + + Semicolon_Loc := Token_Ptr; + + if Token /= Tok_Semicolon then + T_Semicolon; + Resync_Past_Semicolon; + else + Scan; -- past semicolon + end if; + + if Is_Pragma_Name (Chars (Pragma_Node)) then + return Par.Prag (Pragma_Node, Semicolon_Loc); + + else + -- Unrecognized pragma, warning generated in Sem_Prag + + return Pragma_Node; + end if; + + exception + when Error_Resync => + Resync_Past_Semicolon; + return Error; + + end P_Pragma; + + -- This routine is called if a pragma is encountered in an inappropriate + -- position, the pragma is scanned out and control returns to continue. + + -- The caller has checked that the initial token is pragma + + -- Error recovery: cannot raise Error_Resync + + procedure P_Pragmas_Misplaced is + begin + while Token = Tok_Pragma loop + Error_Msg_SC ("pragma not allowed here"); + Discard_Junk_Node (P_Pragma); + end loop; + end P_Pragmas_Misplaced; + + -- This function is called to scan out an optional sequence of pragmas. + -- If no pragmas are found, then No_List is returned. + + -- Error recovery: Cannot raise Error_Resync + + function P_Pragmas_Opt return List_Id is + L : List_Id; + + begin + if Token = Tok_Pragma then + L := New_List; + P_Pragmas_Opt (L); + return L; + + else + return No_List; + end if; + end P_Pragmas_Opt; + + -- This procedure is called to scan out an optional sequence of pragmas. + -- Any pragmas found are appended to the list provided as an argument. + + -- Error recovery: Cannot raise Error_Resync + + procedure P_Pragmas_Opt (List : List_Id) is + P : Node_Id; + + begin + while Token = Tok_Pragma loop + P := P_Pragma; + + if Chars (P) = Name_Assert or else Chars (P) = Name_Debug then + Error_Msg_Name_1 := Chars (P); + Error_Msg_N + ("pragma% must be in declaration/statement context", P); + else + Append (P, List); + end if; + end loop; + end P_Pragmas_Opt; + + -------------------------------------- + -- 2.8 Pragma_Argument Association -- + -------------------------------------- + + -- PRAGMA_ARGUMENT_ASSOCIATION ::= + -- [pragma_argument_IDENTIFIER =>] NAME + -- | [pragma_argument_IDENTIFIER =>] EXPRESSION + + -- Error recovery: cannot raise Error_Resync + + function P_Pragma_Argument_Association return Node_Id is + Scan_State : Saved_Scan_State; + Pragma_Arg_Node : Node_Id; + Identifier_Node : Node_Id; + + begin + Pragma_Arg_Node := New_Node (N_Pragma_Argument_Association, Token_Ptr); + Set_Chars (Pragma_Arg_Node, No_Name); + + if Token = Tok_Identifier then + Identifier_Node := Token_Node; + Save_Scan_State (Scan_State); -- at Identifier + Scan; -- past Identifier + + if Token = Tok_Arrow then + Scan; -- past arrow + Set_Chars (Pragma_Arg_Node, Chars (Identifier_Node)); + Delete_Node (Identifier_Node); + else + Restore_Scan_State (Scan_State); -- to Identifier + end if; + end if; + + Set_Expression (Pragma_Arg_Node, P_Expression); + return Pragma_Arg_Node; + + end P_Pragma_Argument_Association; + +end Ch2; diff --git a/gcc/ada/par-ch3.adb b/gcc/ada/par-ch3.adb new file mode 100644 index 00000000000..937f02d0e7c --- /dev/null +++ b/gcc/ada/par-ch3.adb @@ -0,0 +1,3724 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 3 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.148 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +with Sinfo.CN; use Sinfo.CN; + +separate (Par) + +package body Ch3 is + + ----------------------- + -- Local Subprograms -- + ----------------------- + + function P_Component_List return Node_Id; + function P_Defining_Character_Literal return Node_Id; + function P_Delta_Constraint return Node_Id; + function P_Derived_Type_Def_Or_Private_Ext_Decl return Node_Id; + function P_Digits_Constraint return Node_Id; + function P_Discriminant_Association return Node_Id; + function P_Enumeration_Literal_Specification return Node_Id; + function P_Enumeration_Type_Definition return Node_Id; + function P_Fixed_Point_Definition return Node_Id; + function P_Floating_Point_Definition return Node_Id; + function P_Index_Or_Discriminant_Constraint return Node_Id; + function P_Real_Range_Specification_Opt return Node_Id; + function P_Subtype_Declaration return Node_Id; + function P_Type_Declaration return Node_Id; + function P_Modular_Type_Definition return Node_Id; + function P_Variant return Node_Id; + function P_Variant_Part return Node_Id; + + procedure P_Declarative_Items + (Decls : List_Id; + Done : out Boolean; + In_Spec : Boolean); + -- Scans out a single declarative item, or, in the case of a declaration + -- with a list of identifiers, a list of declarations, one for each of + -- the identifiers in the list. The declaration or declarations scanned + -- are appended to the given list. Done indicates whether or not there + -- may be additional declarative items to scan. If Done is True, then + -- a decision has been made that there are no more items to scan. If + -- Done is False, then there may be additional declarations to scan. + -- In_Spec is true if we are scanning a package declaration, and is used + -- to generate an appropriate message if a statement is encountered in + -- such a context. + + procedure P_Identifier_Declarations + (Decls : List_Id; + Done : out Boolean; + In_Spec : Boolean); + -- Scans out a set of declarations for an identifier or list of + -- identifiers, and appends them to the given list. The parameters have + -- the same significance as for P_Declarative_Items. + + procedure Statement_When_Declaration_Expected + (Decls : List_Id; + Done : out Boolean; + In_Spec : Boolean); + -- Called when a statement is found at a point where a declaration was + -- expected. The parameters are as described for P_Declarative_Items. + + procedure Set_Declaration_Expected; + -- Posts a "declaration expected" error messages at the start of the + -- current token, and if this is the first such message issued, saves + -- the message id in Missing_Begin_Msg, for possible later replacement. + + ------------------- + -- Init_Expr_Opt -- + ------------------- + + function Init_Expr_Opt (P : Boolean := False) return Node_Id is + begin + if Token = Tok_Colon_Equal + or else Token = Tok_Equal + or else Token = Tok_Colon + or else Token = Tok_Is + then + null; + + -- One other possibility. If we have a literal followed by a semicolon, + -- we assume that we have a missing colon-equal. + + elsif Token in Token_Class_Literal then + declare + Scan_State : Saved_Scan_State; + + begin + Save_Scan_State (Scan_State); + Scan; -- past literal or identifier + + if Token = Tok_Semicolon then + Restore_Scan_State (Scan_State); + else + Restore_Scan_State (Scan_State); + return Empty; + end if; + end; + + -- Otherwise we definitely have no initialization expression + + else + return Empty; + end if; + + -- Merge here if we have an initialization expression + + T_Colon_Equal; + + if P then + return P_Expression; + else + return P_Expression_No_Right_Paren; + end if; + end Init_Expr_Opt; + + ---------------------------- + -- 3.1 Basic Declaration -- + ---------------------------- + + -- Parsed by P_Basic_Declarative_Items (3.9) + + ------------------------------ + -- 3.1 Defining Identifier -- + ------------------------------ + + -- DEFINING_IDENTIFIER ::= IDENTIFIER + + -- Error recovery: can raise Error_Resync + + function P_Defining_Identifier return Node_Id is + Ident_Node : Node_Id; + + begin + -- Scan out the identifier. Note that this code is essentially identical + -- to P_Identifier, except that in the call to Scan_Reserved_Identifier + -- we set Force_Msg to True, since we want at least one message for each + -- separate declaration (but not use) of a reserved identifier. + + if Token = Tok_Identifier then + null; + + -- If we have a reserved identifier, manufacture an identifier with + -- a corresponding name after posting an appropriate error message + + elsif Is_Reserved_Identifier then + Scan_Reserved_Identifier (Force_Msg => True); + + -- Otherwise we have junk that cannot be interpreted as an identifier + + else + T_Identifier; -- to give message + raise Error_Resync; + end if; + + Ident_Node := Token_Node; + Scan; -- past the reserved identifier + + if Ident_Node /= Error then + Change_Identifier_To_Defining_Identifier (Ident_Node); + end if; + + return Ident_Node; + end P_Defining_Identifier; + + ----------------------------- + -- 3.2.1 Type Declaration -- + ----------------------------- + + -- TYPE_DECLARATION ::= + -- FULL_TYPE_DECLARATION + -- | INCOMPLETE_TYPE_DECLARATION + -- | PRIVATE_TYPE_DECLARATION + -- | PRIVATE_EXTENSION_DECLARATION + + -- FULL_TYPE_DECLARATION ::= + -- type DEFINING_IDENTIFIER [KNOWN_DISCRIMINANT_PART] is TYPE_DEFINITION; + -- | CONCURRENT_TYPE_DECLARATION + + -- INCOMPLETE_TYPE_DECLARATION ::= + -- type DEFINING_IDENTIFIER [DISCRIMINANT_PART]; + + -- PRIVATE_TYPE_DECLARATION ::= + -- type DEFINING_IDENTIFIER [DISCRIMINANT_PART] + -- is [abstract] [tagged] [limited] private; + + -- PRIVATE_EXTENSION_DECLARATION ::= + -- type DEFINING_IDENTIFIER [DISCRIMINANT_PART] is + -- [abstract] new ancestor_SUBTYPE_INDICATION with private; + + -- TYPE_DEFINITION ::= + -- ENUMERATION_TYPE_DEFINITION | INTEGER_TYPE_DEFINITION + -- | REAL_TYPE_DEFINITION | ARRAY_TYPE_DEFINITION + -- | RECORD_TYPE_DEFINITION | ACCESS_TYPE_DEFINITION + -- | DERIVED_TYPE_DEFINITION + + -- INTEGER_TYPE_DEFINITION ::= + -- SIGNED_INTEGER_TYPE_DEFINITION + -- MODULAR_TYPE_DEFINITION + + -- Error recovery: can raise Error_Resync + + -- Note: The processing for full type declaration, incomplete type + -- declaration, private type declaration and type definition is + -- included in this function. The processing for concurrent type + -- declarations is NOT here, but rather in chapter 9 (i.e. this + -- function handles only declarations starting with TYPE). + + function P_Type_Declaration return Node_Id is + Type_Loc : Source_Ptr; + Type_Start_Col : Column_Number; + Ident_Node : Node_Id; + Decl_Node : Node_Id; + Discr_List : List_Id; + Unknown_Dis : Boolean; + Discr_Sloc : Source_Ptr; + Abstract_Present : Boolean; + Abstract_Loc : Source_Ptr; + End_Labl : Node_Id; + + Typedef_Node : Node_Id; + -- Normally holds type definition, except in the case of a private + -- extension declaration, in which case it holds the declaration itself + + begin + Type_Loc := Token_Ptr; + Type_Start_Col := Start_Column; + T_Type; + Ident_Node := P_Defining_Identifier; + Discr_Sloc := Token_Ptr; + + if P_Unknown_Discriminant_Part_Opt then + Unknown_Dis := True; + Discr_List := No_List; + else + Unknown_Dis := False; + Discr_List := P_Known_Discriminant_Part_Opt; + end if; + + -- Incomplete type declaration. We complete the processing for this + -- case here and return the resulting incomplete type declaration node + + if Token = Tok_Semicolon then + Scan; -- past ; + Decl_Node := New_Node (N_Incomplete_Type_Declaration, Type_Loc); + Set_Defining_Identifier (Decl_Node, Ident_Node); + Set_Unknown_Discriminants_Present (Decl_Node, Unknown_Dis); + Set_Discriminant_Specifications (Decl_Node, Discr_List); + return Decl_Node; + + else + Decl_Node := Empty; + end if; + + -- Full type declaration or private type declaration, must have IS + + if Token = Tok_Equal then + TF_Is; + Scan; -- past = used in place of IS + + elsif Token = Tok_Renames then + Error_Msg_SC ("RENAMES should be IS"); + Scan; -- past RENAMES used in place of IS + + else + TF_Is; + end if; + + -- First an error check, if we have two identifiers in a row, a likely + -- possibility is that the first of the identifiers is an incorrectly + -- spelled keyword. + + if Token = Tok_Identifier then + declare + SS : Saved_Scan_State; + I2 : Boolean; + + begin + Save_Scan_State (SS); + Scan; -- past initial identifier + I2 := (Token = Tok_Identifier); + Restore_Scan_State (SS); + + if I2 + and then + (Bad_Spelling_Of (Tok_Abstract) or else + Bad_Spelling_Of (Tok_Access) or else + Bad_Spelling_Of (Tok_Aliased) or else + Bad_Spelling_Of (Tok_Constant)) + then + null; + end if; + end; + end if; + + -- Check for misuse of Ada 95 keyword abstract in Ada 83 mode + + if Token_Name = Name_Abstract then + Check_95_Keyword (Tok_Abstract, Tok_Tagged); + Check_95_Keyword (Tok_Abstract, Tok_New); + end if; + + -- Check cases of misuse of ABSTRACT + + if Token = Tok_Abstract then + Abstract_Present := True; + Abstract_Loc := Token_Ptr; + Scan; -- past ABSTRACT + + if Token = Tok_Limited + or else Token = Tok_Private + or else Token = Tok_Record + or else Token = Tok_Null + then + Error_Msg_AP ("TAGGED expected"); + end if; + + else + Abstract_Present := False; + Abstract_Loc := No_Location; + end if; + + -- Check for misuse of Ada 95 keyword Tagged + + if Token_Name = Name_Tagged then + Check_95_Keyword (Tok_Tagged, Tok_Private); + Check_95_Keyword (Tok_Tagged, Tok_Limited); + Check_95_Keyword (Tok_Tagged, Tok_Record); + end if; + + -- Special check for misuse of Aliased + + if Token = Tok_Aliased or else Token_Name = Name_Aliased then + Error_Msg_SC ("ALIASED not allowed in type definition"); + Scan; -- past ALIASED + end if; + + -- The following procesing deals with either a private type declaration + -- or a full type declaration. In the private type case, we build the + -- N_Private_Type_Declaration node, setting its Tagged_Present and + -- Limited_Present flags, on encountering the Private keyword, and + -- leave Typedef_Node set to Empty. For the full type declaration + -- case, Typedef_Node gets set to the type definition. + + Typedef_Node := Empty; + + -- Switch on token following the IS. The loop normally runs once. It + -- only runs more than once if an error is detected, to try again after + -- detecting and fixing up the error. + + loop + case Token is + + when Tok_Access => + Typedef_Node := P_Access_Type_Definition; + TF_Semicolon; + exit; + + when Tok_Array => + Typedef_Node := P_Array_Type_Definition; + TF_Semicolon; + exit; + + when Tok_Delta => + Typedef_Node := P_Fixed_Point_Definition; + TF_Semicolon; + exit; + + when Tok_Digits => + Typedef_Node := P_Floating_Point_Definition; + TF_Semicolon; + exit; + + when Tok_In => + Ignore (Tok_In); + + when Tok_Integer_Literal => + T_Range; + Typedef_Node := P_Signed_Integer_Type_Definition; + TF_Semicolon; + exit; + + when Tok_Null => + Typedef_Node := P_Record_Definition; + TF_Semicolon; + exit; + + when Tok_Left_Paren => + Typedef_Node := P_Enumeration_Type_Definition; + TF_Semicolon; + exit; + + when Tok_Mod => + Typedef_Node := P_Modular_Type_Definition; + TF_Semicolon; + exit; + + when Tok_New => + Typedef_Node := P_Derived_Type_Def_Or_Private_Ext_Decl; + TF_Semicolon; + exit; + + when Tok_Range => + Typedef_Node := P_Signed_Integer_Type_Definition; + TF_Semicolon; + exit; + + when Tok_Record => + Typedef_Node := P_Record_Definition; + + End_Labl := + Make_Identifier (Token_Ptr, + Chars => Chars (Ident_Node)); + Set_Comes_From_Source (End_Labl, False); + + Set_End_Label (Typedef_Node, End_Labl); + TF_Semicolon; + exit; + + when Tok_Tagged => + Scan; -- past TAGGED + + if Token = Tok_Abstract then + Error_Msg_SC ("ABSTRACT must come before TAGGED"); + Abstract_Present := True; + Abstract_Loc := Token_Ptr; + Scan; -- past ABSTRACT + end if; + + if Token = Tok_Limited then + Scan; -- past LIMITED + + -- TAGGED LIMITED PRIVATE case + + if Token = Tok_Private then + Decl_Node := + New_Node (N_Private_Type_Declaration, Type_Loc); + Set_Tagged_Present (Decl_Node, True); + Set_Limited_Present (Decl_Node, True); + Scan; -- past PRIVATE + + -- TAGGED LIMITED RECORD + + else + Typedef_Node := P_Record_Definition; + Set_Tagged_Present (Typedef_Node, True); + Set_Limited_Present (Typedef_Node, True); + end if; + + else + -- TAGGED PRIVATE + + if Token = Tok_Private then + Decl_Node := + New_Node (N_Private_Type_Declaration, Type_Loc); + Set_Tagged_Present (Decl_Node, True); + Scan; -- past PRIVATE + + -- TAGGED RECORD + + else + Typedef_Node := P_Record_Definition; + Set_Tagged_Present (Typedef_Node, True); + end if; + end if; + + TF_Semicolon; + exit; + + when Tok_Private => + Decl_Node := New_Node (N_Private_Type_Declaration, Type_Loc); + Scan; -- past PRIVATE + TF_Semicolon; + exit; + + when Tok_Limited => + Scan; -- past LIMITED + + loop + if Token = Tok_Tagged then + Error_Msg_SC ("TAGGED must come before LIMITED"); + Scan; -- past TAGGED + + elsif Token = Tok_Abstract then + Error_Msg_SC ("ABSTRACT must come before LIMITED"); + Scan; -- past ABSTRACT + + else + exit; + end if; + end loop; + + -- LIMITED RECORD or LIMITED NULL RECORD + + if Token = Tok_Record or else Token = Tok_Null then + if Ada_83 then + Error_Msg_SP + ("(Ada 83) limited record declaration not allowed!"); + end if; + + Typedef_Node := P_Record_Definition; + Set_Limited_Present (Typedef_Node, True); + + -- LIMITED PRIVATE is the only remaining possibility here + + else + Decl_Node := New_Node (N_Private_Type_Declaration, Type_Loc); + Set_Limited_Present (Decl_Node, True); + T_Private; -- past PRIVATE (or complain if not there!) + end if; + + TF_Semicolon; + exit; + + -- Here we have an identifier after the IS, which is certainly + -- wrong and which might be one of several different mistakes. + + when Tok_Identifier => + + -- First case, if identifier is on same line, then probably we + -- have something like "type X is Integer .." and the best + -- diagnosis is a missing NEW. Note: the missing new message + -- will be posted by P_Derived_Type_Def_Or_Private_Ext_Decl. + + if not Token_Is_At_Start_Of_Line then + Typedef_Node := P_Derived_Type_Def_Or_Private_Ext_Decl; + TF_Semicolon; + + -- If the identifier is at the start of the line, and is in the + -- same column as the type declaration itself then we consider + -- that we had a missing type definition on the previous line + + elsif Start_Column <= Type_Start_Col then + Error_Msg_AP ("type definition expected"); + Typedef_Node := Error; + + -- If the identifier is at the start of the line, and is in + -- a column to the right of the type declaration line, then we + -- may have something like: + + -- type x is + -- r : integer + + -- and the best diagnosis is a missing record keyword + + else + Typedef_Node := P_Record_Definition; + TF_Semicolon; + end if; + + exit; + + -- Anything else is an error + + when others => + if Bad_Spelling_Of (Tok_Access) + or else + Bad_Spelling_Of (Tok_Array) + or else + Bad_Spelling_Of (Tok_Delta) + or else + Bad_Spelling_Of (Tok_Digits) + or else + Bad_Spelling_Of (Tok_Limited) + or else + Bad_Spelling_Of (Tok_Private) + or else + Bad_Spelling_Of (Tok_Range) + or else + Bad_Spelling_Of (Tok_Record) + or else + Bad_Spelling_Of (Tok_Tagged) + then + null; + + else + Error_Msg_AP ("type definition expected"); + raise Error_Resync; + end if; + + end case; + end loop; + + -- For the private type declaration case, the private type declaration + -- node has been built, with the Tagged_Present and Limited_Present + -- flags set as needed, and Typedef_Node is left set to Empty. + + if No (Typedef_Node) then + Set_Unknown_Discriminants_Present (Decl_Node, Unknown_Dis); + Set_Abstract_Present (Decl_Node, Abstract_Present); + + -- For a private extension declaration, Typedef_Node contains the + -- N_Private_Extension_Declaration node, which we now complete. Note + -- that the private extension declaration, unlike a full type + -- declaration, does permit unknown discriminants. + + elsif Nkind (Typedef_Node) = N_Private_Extension_Declaration then + Decl_Node := Typedef_Node; + Set_Sloc (Decl_Node, Type_Loc); + Set_Unknown_Discriminants_Present (Decl_Node, Unknown_Dis); + Set_Abstract_Present (Typedef_Node, Abstract_Present); + + -- In the full type declaration case, Typedef_Node has the type + -- definition and here is where we build the full type declaration + -- node. This is also where we check for improper use of an unknown + -- discriminant part (not allowed for full type declaration). + + else + if Nkind (Typedef_Node) = N_Record_Definition + or else (Nkind (Typedef_Node) = N_Derived_Type_Definition + and then Present (Record_Extension_Part (Typedef_Node))) + then + Set_Abstract_Present (Typedef_Node, Abstract_Present); + + elsif Abstract_Present then + Error_Msg ("ABSTRACT not allowed here, ignored", Abstract_Loc); + end if; + + Decl_Node := New_Node (N_Full_Type_Declaration, Type_Loc); + Set_Type_Definition (Decl_Node, Typedef_Node); + + if Unknown_Dis then + Error_Msg + ("Full type declaration cannot have unknown discriminants", + Discr_Sloc); + end if; + end if; + + -- Remaining processing is common for all three cases + + Set_Defining_Identifier (Decl_Node, Ident_Node); + Set_Discriminant_Specifications (Decl_Node, Discr_List); + return Decl_Node; + + end P_Type_Declaration; + + ---------------------------------- + -- 3.2.1 Full Type Declaration -- + ---------------------------------- + + -- Parsed by P_Type_Declaration (3.2.1) + + ---------------------------- + -- 3.2.1 Type Definition -- + ---------------------------- + + -- Parsed by P_Type_Declaration (3.2.1) + + -------------------------------- + -- 3.2.2 Subtype Declaration -- + -------------------------------- + + -- SUBTYPE_DECLARATION ::= + -- subtype DEFINING_IDENTIFIER is SUBTYPE_INDICATION; + + -- The caller has checked that the initial token is SUBTYPE + + -- Error recovery: can raise Error_Resync + + function P_Subtype_Declaration return Node_Id is + Decl_Node : Node_Id; + + begin + Decl_Node := New_Node (N_Subtype_Declaration, Token_Ptr); + Scan; -- past SUBTYPE + Set_Defining_Identifier (Decl_Node, P_Defining_Identifier); + TF_Is; + + if Token = Tok_New then + Error_Msg_SC ("NEW ignored (only allowed in type declaration)"); + Scan; -- past NEW + end if; + + Set_Subtype_Indication (Decl_Node, P_Subtype_Indication); + TF_Semicolon; + return Decl_Node; + end P_Subtype_Declaration; + + ------------------------------- + -- 3.2.2 Subtype Indication -- + ------------------------------- + + -- SUBTYPE_INDICATION ::= SUBTYPE_MARK [CONSTRAINT] + + -- Error recovery: can raise Error_Resync + + function P_Subtype_Indication return Node_Id is + Type_Node : Node_Id; + + begin + if Token = Tok_Identifier or else Token = Tok_Operator_Symbol then + Type_Node := P_Subtype_Mark; + return P_Subtype_Indication (Type_Node); + + else + -- Check for error of using record definition and treat it nicely, + -- otherwise things are really messed up, so resynchronize. + + if Token = Tok_Record then + Error_Msg_SC ("anonymous record definitions are not permitted"); + Discard_Junk_Node (P_Record_Definition); + return Error; + + else + Error_Msg_AP ("subtype indication expected"); + raise Error_Resync; + end if; + end if; + end P_Subtype_Indication; + + -- The following function is identical except that it is called with + -- the subtype mark already scanned out, and it scans out the constraint + + -- Error recovery: can raise Error_Resync + + function P_Subtype_Indication (Subtype_Mark : Node_Id) return Node_Id is + Indic_Node : Node_Id; + Constr_Node : Node_Id; + + begin + Constr_Node := P_Constraint_Opt; + + if No (Constr_Node) then + return Subtype_Mark; + else + Indic_Node := New_Node (N_Subtype_Indication, Sloc (Subtype_Mark)); + Set_Subtype_Mark (Indic_Node, Check_Subtype_Mark (Subtype_Mark)); + Set_Constraint (Indic_Node, Constr_Node); + return Indic_Node; + end if; + + end P_Subtype_Indication; + + ------------------------- + -- 3.2.2 Subtype Mark -- + ------------------------- + + -- SUBTYPE_MARK ::= subtype_NAME; + + -- Note: The subtype mark which appears after an IN or NOT IN + -- operator is parsed by P_Range_Or_Subtype_Mark (3.5) + + -- Error recovery: cannot raise Error_Resync + + function P_Subtype_Mark return Node_Id is + begin + return P_Subtype_Mark_Resync; + + exception + when Error_Resync => + return Error; + end P_Subtype_Mark; + + -- This routine differs from P_Subtype_Mark in that it insists that an + -- identifier be present, and if it is not, it raises Error_Resync. + + -- Error recovery: can raise Error_Resync + + function P_Subtype_Mark_Resync return Node_Id is + Type_Node : Node_Id; + + begin + if Token = Tok_Access then + Error_Msg_SC ("anonymous access type definition not allowed here"); + Scan; -- past ACCESS + end if; + + if Token = Tok_Array then + Error_Msg_SC ("anonymous array definition not allowed here"); + Discard_Junk_Node (P_Array_Type_Definition); + return Empty; + + else + Type_Node := P_Qualified_Simple_Name_Resync; + + -- Check for a subtype mark attribute. The only valid possibilities + -- are 'CLASS and 'BASE. Anything else is a definite error. We may + -- as well catch it here. + + if Token = Tok_Apostrophe then + return P_Subtype_Mark_Attribute (Type_Node); + else + return Type_Node; + end if; + end if; + end P_Subtype_Mark_Resync; + + -- The following function is called to scan out a subtype mark attribute. + -- The caller has already scanned out the subtype mark, which is passed in + -- as the argument, and has checked that the current token is apostrophe. + + -- Only a special subclass of attributes, called type attributes + -- (see Snames package) are allowed in this syntactic position. + + -- Note: if the apostrophe is followed by other than an identifier, then + -- the input expression is returned unchanged, and the scan pointer is + -- left pointing to the apostrophe. + + -- Error recovery: can raise Error_Resync + + function P_Subtype_Mark_Attribute (Type_Node : Node_Id) return Node_Id is + Attr_Node : Node_Id := Empty; + Scan_State : Saved_Scan_State; + Prefix : Node_Id; + + begin + Prefix := Check_Subtype_Mark (Type_Node); + + if Prefix = Error then + raise Error_Resync; + end if; + + -- Loop through attributes appearing (more than one can appear as for + -- for example in X'Base'Class). We are at an apostrophe on entry to + -- this loop, and it runs once for each attribute parsed, with + -- Prefix being the current possible prefix if it is an attribute. + + loop + Save_Scan_State (Scan_State); -- at Apostrophe + Scan; -- past apostrophe + + if Token /= Tok_Identifier then + Restore_Scan_State (Scan_State); -- to apostrophe + return Prefix; -- no attribute after all + + elsif not Is_Type_Attribute_Name (Token_Name) then + Error_Msg_N + ("attribute & may not be used in a subtype mark", Token_Node); + raise Error_Resync; + + else + Attr_Node := + Make_Attribute_Reference (Prev_Token_Ptr, + Prefix => Prefix, + Attribute_Name => Token_Name); + Delete_Node (Token_Node); + Scan; -- past type attribute identifier + end if; + + exit when Token /= Tok_Apostrophe; + Prefix := Attr_Node; + end loop; + + -- Fall through here after scanning type attribute + + return Attr_Node; + end P_Subtype_Mark_Attribute; + + ----------------------- + -- 3.2.2 Constraint -- + ----------------------- + + -- CONSTRAINT ::= SCALAR_CONSTRAINT | COMPOSITE_CONSTRAINT + + -- SCALAR_CONSTRAINT ::= + -- RANGE_CONSTRAINT | DIGITS_CONSTRAINT | DELTA_CONSTRAINT + + -- COMPOSITE_CONSTRAINT ::= + -- INDEX_CONSTRAINT | DISCRIMINANT_CONSTRAINT + + -- If no constraint is present, this function returns Empty + + -- Error recovery: can raise Error_Resync + + function P_Constraint_Opt return Node_Id is + begin + if Token = Tok_Range + or else Bad_Spelling_Of (Tok_Range) + then + return P_Range_Constraint; + + elsif Token = Tok_Digits + or else Bad_Spelling_Of (Tok_Digits) + then + return P_Digits_Constraint; + + elsif Token = Tok_Delta + or else Bad_Spelling_Of (Tok_Delta) + then + return P_Delta_Constraint; + + elsif Token = Tok_Left_Paren then + return P_Index_Or_Discriminant_Constraint; + + elsif Token = Tok_In then + Ignore (Tok_In); + return P_Constraint_Opt; + + else + return Empty; + end if; + + end P_Constraint_Opt; + + ------------------------------ + -- 3.2.2 Scalar Constraint -- + ------------------------------ + + -- Parsed by P_Constraint_Opt (3.2.2) + + --------------------------------- + -- 3.2.2 Composite Constraint -- + --------------------------------- + + -- Parsed by P_Constraint_Opt (3.2.2) + + -------------------------------------------------------- + -- 3.3 Identifier Declarations (Also 7.4, 8.5, 11.1) -- + -------------------------------------------------------- + + -- This routine scans out a declaration starting with an identifier: + + -- OBJECT_DECLARATION ::= + -- DEFINING_IDENTIFIER_LIST : [constant] [aliased] + -- SUBTYPE_INDICATION [:= EXPRESSION]; + -- | DEFINING_IDENTIFIER_LIST : [constant] [aliased] + -- ARRAY_TYPE_DEFINITION [:= EXPRESSION]; + + -- NUMBER_DECLARATION ::= + -- DEFINING_IDENTIFIER_LIST : constant ::= static_EXPRESSION; + + -- OBJECT_RENAMING_DECLARATION ::= + -- DEFINING_IDENTIFIER : SUBTYPE_MARK renames object_NAME; + + -- EXCEPTION_RENAMING_DECLARATION ::= + -- DEFINING_IDENTIFIER : exception renames exception_NAME; + + -- EXCEPTION_DECLARATION ::= + -- DEFINING_IDENTIFIER_LIST : exception; + + -- Note that the ALIASED indication in an object declaration is + -- marked by a flag in the parent node. + + -- The caller has checked that the initial token is an identifier + + -- The value returned is a list of declarations, one for each identifier + -- in the list (as described in Sinfo, we always split up multiple + -- declarations into the equivalent sequence of single declarations + -- using the More_Ids and Prev_Ids flags to preserve the source). + + -- If the identifier turns out to be a probable statement rather than + -- an identifier, then the scan is left pointing to the identifier and + -- No_List is returned. + + -- Error recovery: can raise Error_Resync + + procedure P_Identifier_Declarations + (Decls : List_Id; + Done : out Boolean; + In_Spec : Boolean) + is + Decl_Node : Node_Id; + Type_Node : Node_Id; + Ident_Sloc : Source_Ptr; + Scan_State : Saved_Scan_State; + List_OK : Boolean := True; + Ident : Nat; + Init_Expr : Node_Id; + Init_Loc : Source_Ptr; + Con_Loc : Source_Ptr; + + Idents : array (Int range 1 .. 4096) of Entity_Id; + -- Used to save identifiers in the identifier list. The upper bound + -- of 4096 is expected to be infinite in practice, and we do not even + -- bother to check if this upper bound is exceeded. + + Num_Idents : Nat := 1; + -- Number of identifiers stored in Idents + + procedure No_List; + -- This procedure is called in renames cases to make sure that we do + -- not have more than one identifier. If we do have more than one + -- then an error message is issued (and the declaration is split into + -- multiple declarations) + + function Token_Is_Renames return Boolean; + -- Checks if current token is RENAMES, and if so, scans past it and + -- returns True, otherwise returns False. Includes checking for some + -- common error cases. + + procedure No_List is + begin + if Num_Idents > 1 then + Error_Msg ("identifier list not allowed for RENAMES", + Sloc (Idents (2))); + end if; + + List_OK := False; + end No_List; + + function Token_Is_Renames return Boolean is + At_Colon : Saved_Scan_State; + + begin + if Token = Tok_Colon then + Save_Scan_State (At_Colon); + Scan; -- past colon + Check_Misspelling_Of (Tok_Renames); + + if Token = Tok_Renames then + Error_Msg_SP ("extra "":"" ignored"); + Scan; -- past RENAMES + return True; + else + Restore_Scan_State (At_Colon); + return False; + end if; + + else + Check_Misspelling_Of (Tok_Renames); + + if Token = Tok_Renames then + Scan; -- past RENAMES + return True; + else + return False; + end if; + end if; + end Token_Is_Renames; + + -- Start of processing for P_Identifier_Declarations + + begin + Ident_Sloc := Token_Ptr; + Save_Scan_State (Scan_State); -- at first identifier + Idents (1) := P_Defining_Identifier; + + -- If we have a colon after the identifier, then we can assume that + -- this is in fact a valid identifier declaration and can steam ahead. + + if Token = Tok_Colon then + Scan; -- past colon + + -- If we have a comma, then scan out the list of identifiers + + elsif Token = Tok_Comma then + + while Comma_Present loop + Num_Idents := Num_Idents + 1; + Idents (Num_Idents) := P_Defining_Identifier; + end loop; + + Save_Scan_State (Scan_State); -- at colon + T_Colon; + + -- If we have identifier followed by := then we assume that what is + -- really meant is an assignment statement. The assignment statement + -- is scanned out and added to the list of declarations. An exception + -- occurs if the := is followed by the keyword constant, in which case + -- we assume it was meant to be a colon. + + elsif Token = Tok_Colon_Equal then + Scan; -- past := + + if Token = Tok_Constant then + Error_Msg_SP ("colon expected"); + + else + Restore_Scan_State (Scan_State); + Statement_When_Declaration_Expected (Decls, Done, In_Spec); + return; + end if; + + -- If we have an IS keyword, then assume the TYPE keyword was missing + + elsif Token = Tok_Is then + Restore_Scan_State (Scan_State); + Append_To (Decls, P_Type_Declaration); + Done := False; + return; + + -- Otherwise we have an error situation + + else + Restore_Scan_State (Scan_State); + + -- First case is possible misuse of PROTECTED in Ada 83 mode. If + -- so, fix the keyword and return to scan the protected declaration. + + if Token_Name = Name_Protected then + Check_95_Keyword (Tok_Protected, Tok_Identifier); + Check_95_Keyword (Tok_Protected, Tok_Type); + Check_95_Keyword (Tok_Protected, Tok_Body); + + if Token = Tok_Protected then + Done := False; + return; + end if; + + -- Check misspelling possibilities. If so, correct the misspelling + -- and return to scan out the resulting declaration. + + elsif Bad_Spelling_Of (Tok_Function) + or else Bad_Spelling_Of (Tok_Procedure) + or else Bad_Spelling_Of (Tok_Package) + or else Bad_Spelling_Of (Tok_Pragma) + or else Bad_Spelling_Of (Tok_Protected) + or else Bad_Spelling_Of (Tok_Generic) + or else Bad_Spelling_Of (Tok_Subtype) + or else Bad_Spelling_Of (Tok_Type) + or else Bad_Spelling_Of (Tok_Task) + or else Bad_Spelling_Of (Tok_Use) + or else Bad_Spelling_Of (Tok_For) + then + Done := False; + return; + + -- Otherwise we definitely have an ordinary identifier with a junk + -- token after it. Just complain that we expect a declaration, and + -- skip to a semicolon + + else + Set_Declaration_Expected; + Resync_Past_Semicolon; + Done := False; + return; + end if; + end if; + + -- Come here with an identifier list and colon scanned out. We now + -- build the nodes for the declarative items. One node is built for + -- each identifier in the list, with the type information being + -- repeated by rescanning the appropriate section of source. + + -- First an error check, if we have two identifiers in a row, a likely + -- possibility is that the first of the identifiers is an incorrectly + -- spelled keyword. + + if Token = Tok_Identifier then + declare + SS : Saved_Scan_State; + I2 : Boolean; + + begin + Save_Scan_State (SS); + Scan; -- past initial identifier + I2 := (Token = Tok_Identifier); + Restore_Scan_State (SS); + + if I2 + and then + (Bad_Spelling_Of (Tok_Access) or else + Bad_Spelling_Of (Tok_Aliased) or else + Bad_Spelling_Of (Tok_Constant)) + then + null; + end if; + end; + end if; + + -- Loop through identifiers + + Ident := 1; + Ident_Loop : loop + + -- Check for some cases of misused Ada 95 keywords + + if Token_Name = Name_Aliased then + Check_95_Keyword (Tok_Aliased, Tok_Array); + Check_95_Keyword (Tok_Aliased, Tok_Identifier); + Check_95_Keyword (Tok_Aliased, Tok_Constant); + end if; + + -- Constant cases + + if Token = Tok_Constant then + Con_Loc := Token_Ptr; + Scan; -- past CONSTANT + + -- Number declaration, initialization required + + Init_Expr := Init_Expr_Opt; + + if Present (Init_Expr) then + Decl_Node := New_Node (N_Number_Declaration, Ident_Sloc); + Set_Expression (Decl_Node, Init_Expr); + + -- Constant object declaration + + else + Decl_Node := New_Node (N_Object_Declaration, Ident_Sloc); + Set_Constant_Present (Decl_Node, True); + + if Token_Name = Name_Aliased then + Check_95_Keyword (Tok_Aliased, Tok_Array); + Check_95_Keyword (Tok_Aliased, Tok_Identifier); + end if; + + if Token = Tok_Aliased then + Error_Msg_SC ("ALIASED should be before CONSTANT"); + Scan; -- past ALIASED + Set_Aliased_Present (Decl_Node, True); + end if; + + if Token = Tok_Array then + Set_Object_Definition + (Decl_Node, P_Array_Type_Definition); + else + Set_Object_Definition (Decl_Node, P_Subtype_Indication); + end if; + + if Token = Tok_Renames then + Error_Msg + ("CONSTANT not permitted in renaming declaration", + Con_Loc); + Scan; -- Past renames + Discard_Junk_Node (P_Name); + end if; + end if; + + -- Exception cases + + elsif Token = Tok_Exception then + Scan; -- past EXCEPTION + + if Token_Is_Renames then + No_List; + Decl_Node := + New_Node (N_Exception_Renaming_Declaration, Ident_Sloc); + Set_Name (Decl_Node, P_Qualified_Simple_Name_Resync); + No_Constraint; + else + Decl_Node := New_Node (N_Exception_Declaration, Prev_Token_Ptr); + end if; + + -- Aliased case (note that an object definition is required) + + elsif Token = Tok_Aliased then + Scan; -- past ALIASED + Decl_Node := New_Node (N_Object_Declaration, Ident_Sloc); + Set_Aliased_Present (Decl_Node, True); + + if Token = Tok_Constant then + Scan; -- past CONSTANT + Set_Constant_Present (Decl_Node, True); + end if; + + if Token = Tok_Array then + Set_Object_Definition + (Decl_Node, P_Array_Type_Definition); + else + Set_Object_Definition (Decl_Node, P_Subtype_Indication); + end if; + + -- Array case + + elsif Token = Tok_Array then + Decl_Node := New_Node (N_Object_Declaration, Ident_Sloc); + Set_Object_Definition (Decl_Node, P_Array_Type_Definition); + + -- Subtype indication case + + else + Type_Node := P_Subtype_Mark; + + -- Object renaming declaration + + if Token_Is_Renames then + No_List; + Decl_Node := + New_Node (N_Object_Renaming_Declaration, Ident_Sloc); + Set_Subtype_Mark (Decl_Node, Type_Node); + Set_Name (Decl_Node, P_Name); + + -- Object declaration + + else + Decl_Node := New_Node (N_Object_Declaration, Ident_Sloc); + Set_Object_Definition + (Decl_Node, P_Subtype_Indication (Type_Node)); + + -- RENAMES at this point means that we had the combination of + -- a constraint on the Type_Node and renames, which is illegal + + if Token_Is_Renames then + Error_Msg_N + ("constraint not allowed in object renaming declaration", + Constraint (Object_Definition (Decl_Node))); + raise Error_Resync; + end if; + end if; + end if; + + -- Scan out initialization, allowed only for object declaration + + Init_Loc := Token_Ptr; + Init_Expr := Init_Expr_Opt; + + if Present (Init_Expr) then + if Nkind (Decl_Node) = N_Object_Declaration then + Set_Expression (Decl_Node, Init_Expr); + else + Error_Msg ("initialization not allowed here", Init_Loc); + end if; + end if; + + TF_Semicolon; + Set_Defining_Identifier (Decl_Node, Idents (Ident)); + + if List_OK then + if Ident < Num_Idents then + Set_More_Ids (Decl_Node, True); + end if; + + if Ident > 1 then + Set_Prev_Ids (Decl_Node, True); + end if; + end if; + + Append (Decl_Node, Decls); + exit Ident_Loop when Ident = Num_Idents; + Restore_Scan_State (Scan_State); + T_Colon; + Ident := Ident + 1; + end loop Ident_Loop; + + Done := False; + + end P_Identifier_Declarations; + + ------------------------------- + -- 3.3.1 Object Declaration -- + ------------------------------- + + -- OBJECT DECLARATION ::= + -- DEFINING_IDENTIFIER_LIST : [aliased] [constant] + -- SUBTYPE_INDICATION [:= EXPRESSION]; + -- | DEFINING_IDENTIFIER_LIST : [aliased] [constant] + -- ARRAY_TYPE_DEFINITION [:= EXPRESSION]; + -- | SINGLE_TASK_DECLARATION + -- | SINGLE_PROTECTED_DECLARATION + + -- Cases starting with TASK are parsed by P_Task (9.1) + -- Cases starting with PROTECTED are parsed by P_Protected (9.4) + -- All other cases are parsed by P_Identifier_Declarations (3.3) + + ------------------------------------- + -- 3.3.1 Defining Identifier List -- + ------------------------------------- + + -- DEFINING_IDENTIFIER_LIST ::= + -- DEFINING_IDENTIFIER {, DEFINING_IDENTIFIER} + + -- Always parsed by the construct in which it appears. See special + -- section on "Handling of Defining Identifier Lists" in this unit. + + ------------------------------- + -- 3.3.2 Number Declaration -- + ------------------------------- + + -- Parsed by P_Identifier_Declarations (3.3) + + ------------------------------------------------------------------------- + -- 3.4 Derived Type Definition or Private Extension Declaration (7.3) -- + ------------------------------------------------------------------------- + + -- DERIVED_TYPE_DEFINITION ::= + -- [abstract] new parent_SUBTYPE_INDICATION [RECORD_EXTENSION_PART] + + -- PRIVATE_EXTENSION_DECLARATION ::= + -- type DEFINING_IDENTIFIER [DISCRIMINANT_PART] is + -- [abstract] new ancestor_SUBTYPE_INDICATION with PRIVATE; + + -- RECORD_EXTENSION_PART ::= with RECORD_DEFINITION + + -- The caller has already scanned out the part up to the NEW, and Token + -- either contains Tok_New (or ought to, if it doesn't this procedure + -- will post an appropriate "NEW expected" message). + + -- Note: the caller is responsible for filling in the Sloc field of + -- the returned node in the private extension declaration case as + -- well as the stuff relating to the discriminant part. + + -- Error recovery: can raise Error_Resync; + + function P_Derived_Type_Def_Or_Private_Ext_Decl return Node_Id is + Typedef_Node : Node_Id; + Typedecl_Node : Node_Id; + + begin + Typedef_Node := New_Node (N_Derived_Type_Definition, Token_Ptr); + T_New; + + if Token = Tok_Abstract then + Error_Msg_SC ("ABSTRACT must come before NEW, not after"); + Scan; + end if; + + Set_Subtype_Indication (Typedef_Node, P_Subtype_Indication); + + -- Deal with record extension, note that we assume that a WITH is + -- missing in the case of "type X is new Y record ..." or in the + -- case of "type X is new Y null record". + + if Token = Tok_With + or else Token = Tok_Record + or else Token = Tok_Null + then + T_With; -- past WITH or give error message + + if Token = Tok_Limited then + Error_Msg_SC + ("LIMITED keyword not allowed in private extension"); + Scan; -- ignore LIMITED + end if; + + -- Private extension declaration + + if Token = Tok_Private then + Scan; -- past PRIVATE + + -- Throw away the type definition node and build the type + -- declaration node. Note the caller must set the Sloc, + -- Discriminant_Specifications, Unknown_Discriminants_Present, + -- and Defined_Identifier fields in the returned node. + + Typedecl_Node := + Make_Private_Extension_Declaration (No_Location, + Defining_Identifier => Empty, + Subtype_Indication => Subtype_Indication (Typedef_Node), + Abstract_Present => Abstract_Present (Typedef_Node)); + + Delete_Node (Typedef_Node); + return Typedecl_Node; + + -- Derived type definition with record extension part + + else + Set_Record_Extension_Part (Typedef_Node, P_Record_Definition); + return Typedef_Node; + end if; + + -- Derived type definition with no record extension part + + else + return Typedef_Node; + end if; + end P_Derived_Type_Def_Or_Private_Ext_Decl; + + --------------------------- + -- 3.5 Range Constraint -- + --------------------------- + + -- RANGE_CONSTRAINT ::= range RANGE + + -- The caller has checked that the initial token is RANGE + + -- Error recovery: cannot raise Error_Resync + + function P_Range_Constraint return Node_Id is + Range_Node : Node_Id; + + begin + Range_Node := New_Node (N_Range_Constraint, Token_Ptr); + Scan; -- past RANGE + Set_Range_Expression (Range_Node, P_Range); + return Range_Node; + end P_Range_Constraint; + + ---------------- + -- 3.5 Range -- + ---------------- + + -- RANGE ::= + -- RANGE_ATTRIBUTE_REFERENCE | SIMPLE_EXPRESSION .. SIMPLE_EXPRESSION + + -- Note: the range that appears in a membership test is parsed by + -- P_Range_Or_Subtype_Mark (3.5). + + -- Error recovery: cannot raise Error_Resync + + function P_Range return Node_Id is + Expr_Node : Node_Id; + Range_Node : Node_Id; + + begin + Expr_Node := P_Simple_Expression_Or_Range_Attribute; + + if Expr_Form = EF_Range_Attr then + return Expr_Node; + + elsif Token = Tok_Dot_Dot then + Range_Node := New_Node (N_Range, Token_Ptr); + Set_Low_Bound (Range_Node, Expr_Node); + Scan; -- past .. + Expr_Node := P_Expression; + Check_Simple_Expression (Expr_Node); + Set_High_Bound (Range_Node, Expr_Node); + return Range_Node; + + -- Anything else is an error + + else + T_Dot_Dot; -- force missing .. message + return Error; + end if; + end P_Range; + + ---------------------------------- + -- 3.5 P_Range_Or_Subtype_Mark -- + ---------------------------------- + + -- RANGE ::= + -- RANGE_ATTRIBUTE_REFERENCE + -- | SIMPLE_EXPRESSION .. SIMPLE_EXPRESSION + + -- This routine scans out the range or subtype mark that forms the right + -- operand of a membership test. + + -- Note: as documented in the Sinfo interface, although the syntax only + -- allows a subtype mark, we in fact allow any simple expression to be + -- returned from this routine. The semantics is responsible for issuing + -- an appropriate message complaining if the argument is not a name. + -- This simplifies the coding and error recovery processing in the + -- parser, and in any case it is preferable not to consider this a + -- syntax error and to continue with the semantic analysis. + + -- Error recovery: cannot raise Error_Resync + + function P_Range_Or_Subtype_Mark return Node_Id is + Expr_Node : Node_Id; + Range_Node : Node_Id; + + begin + Expr_Node := P_Simple_Expression_Or_Range_Attribute; + + if Expr_Form = EF_Range_Attr then + return Expr_Node; + + -- Simple_Expression .. Simple_Expression + + elsif Token = Tok_Dot_Dot then + Check_Simple_Expression (Expr_Node); + Range_Node := New_Node (N_Range, Token_Ptr); + Set_Low_Bound (Range_Node, Expr_Node); + Scan; -- past .. + Set_High_Bound (Range_Node, P_Simple_Expression); + return Range_Node; + + -- Case of subtype mark (optionally qualified simple name or an + -- attribute whose prefix is an optionally qualifed simple name) + + elsif Expr_Form = EF_Simple_Name + or else Nkind (Expr_Node) = N_Attribute_Reference + then + -- Check for error of range constraint after a subtype mark + + if Token = Tok_Range then + Error_Msg_SC + ("range constraint not allowed in membership test"); + Scan; -- past RANGE + raise Error_Resync; + + -- Check for error of DIGITS or DELTA after a subtype mark + + elsif Token = Tok_Digits or else Token = Tok_Delta then + Error_Msg_SC + ("accuracy definition not allowed in membership test"); + Scan; -- past DIGITS or DELTA + raise Error_Resync; + + elsif Token = Tok_Apostrophe then + return P_Subtype_Mark_Attribute (Expr_Node); + + else + return Expr_Node; + end if; + + -- At this stage, we have some junk following the expression. We + -- really can't tell what is wrong, might be a missing semicolon, + -- or a missing THEN, or whatever. Our caller will figure it out! + + else + return Expr_Node; + end if; + end P_Range_Or_Subtype_Mark; + + ---------------------------------------- + -- 3.5.1 Enumeration Type Definition -- + ---------------------------------------- + + -- ENUMERATION_TYPE_DEFINITION ::= + -- (ENUMERATION_LITERAL_SPECIFICATION + -- {, ENUMERATION_LITERAL_SPECIFICATION}) + + -- The caller has already scanned out the TYPE keyword + + -- Error recovery: can raise Error_Resync; + + function P_Enumeration_Type_Definition return Node_Id is + Typedef_Node : Node_Id; + + begin + Typedef_Node := New_Node (N_Enumeration_Type_Definition, Token_Ptr); + Set_Literals (Typedef_Node, New_List); + + T_Left_Paren; + + loop + Append (P_Enumeration_Literal_Specification, Literals (Typedef_Node)); + exit when not Comma_Present; + end loop; + + T_Right_Paren; + return Typedef_Node; + end P_Enumeration_Type_Definition; + + ---------------------------------------------- + -- 3.5.1 Enumeration Literal Specification -- + ---------------------------------------------- + + -- ENUMERATION_LITERAL_SPECIFICATION ::= + -- DEFINING_IDENTIFIER | DEFINING_CHARACTER_LITERAL + + -- Error recovery: can raise Error_Resync + + function P_Enumeration_Literal_Specification return Node_Id is + begin + if Token = Tok_Char_Literal then + return P_Defining_Character_Literal; + else + return P_Defining_Identifier; + end if; + end P_Enumeration_Literal_Specification; + + --------------------------------------- + -- 3.5.1 Defining_Character_Literal -- + --------------------------------------- + + -- DEFINING_CHARACTER_LITERAL ::= CHARACTER_LITERAL + + -- Error recovery: cannot raise Error_Resync + + -- The caller has checked that the current token is a character literal + + function P_Defining_Character_Literal return Node_Id is + Literal_Node : Node_Id; + + begin + Literal_Node := Token_Node; + Change_Character_Literal_To_Defining_Character_Literal (Literal_Node); + Scan; -- past character literal + return Literal_Node; + end P_Defining_Character_Literal; + + ------------------------------------ + -- 3.5.4 Integer Type Definition -- + ------------------------------------ + + -- Parsed by P_Type_Declaration (3.2.1) + + ------------------------------------------- + -- 3.5.4 Signed Integer Type Definition -- + ------------------------------------------- + + -- SIGNED_INTEGER_TYPE_DEFINITION ::= + -- range static_SIMPLE_EXPRESSION .. static_SIMPLE_EXPRESSION + + -- Normally the initial token on entry is RANGE, but in some + -- error conditions, the range token was missing and control is + -- passed with Token pointing to first token of the first expression. + + -- Error recovery: cannot raise Error_Resync + + function P_Signed_Integer_Type_Definition return Node_Id is + Typedef_Node : Node_Id; + Expr_Node : Node_Id; + + begin + Typedef_Node := New_Node (N_Signed_Integer_Type_Definition, Token_Ptr); + + if Token = Tok_Range then + Scan; -- past RANGE + end if; + + Expr_Node := P_Expression; + Check_Simple_Expression (Expr_Node); + Set_Low_Bound (Typedef_Node, Expr_Node); + T_Dot_Dot; + Expr_Node := P_Expression; + Check_Simple_Expression (Expr_Node); + Set_High_Bound (Typedef_Node, Expr_Node); + return Typedef_Node; + end P_Signed_Integer_Type_Definition; + + ------------------------------------ + -- 3.5.4 Modular Type Definition -- + ------------------------------------ + + -- MODULAR_TYPE_DEFINITION ::= mod static_EXPRESSION + + -- The caller has checked that the initial token is MOD + + -- Error recovery: cannot raise Error_Resync + + function P_Modular_Type_Definition return Node_Id is + Typedef_Node : Node_Id; + + begin + if Ada_83 then + Error_Msg_SC ("(Ada 83): modular types not allowed"); + end if; + + Typedef_Node := New_Node (N_Modular_Type_Definition, Token_Ptr); + Scan; -- past MOD + Set_Expression (Typedef_Node, P_Expression_No_Right_Paren); + + -- Handle mod L..R cleanly + + if Token = Tok_Dot_Dot then + Error_Msg_SC ("range not allowed for modular type"); + Scan; -- past .. + Set_Expression (Typedef_Node, P_Expression_No_Right_Paren); + end if; + + return Typedef_Node; + end P_Modular_Type_Definition; + + --------------------------------- + -- 3.5.6 Real Type Definition -- + --------------------------------- + + -- Parsed by P_Type_Declaration (3.2.1) + + -------------------------------------- + -- 3.5.7 Floating Point Definition -- + -------------------------------------- + + -- FLOATING_POINT_DEFINITION ::= + -- digits static_EXPRESSION [REAL_RANGE_SPECIFICATION] + + -- Note: In Ada-83, the EXPRESSION must be a SIMPLE_EXPRESSION + + -- The caller has checked that the initial token is DIGITS + + -- Error recovery: cannot raise Error_Resync + + function P_Floating_Point_Definition return Node_Id is + Digits_Loc : constant Source_Ptr := Token_Ptr; + Def_Node : Node_Id; + Expr_Node : Node_Id; + + begin + Scan; -- past DIGITS + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + + -- Handle decimal fixed-point defn with DIGITS/DELTA in wrong order + + if Token = Tok_Delta then + Error_Msg_SC ("DELTA must come before DIGITS"); + Def_Node := New_Node (N_Decimal_Fixed_Point_Definition, Digits_Loc); + Scan; -- past DELTA + Set_Delta_Expression (Def_Node, P_Expression_No_Right_Paren); + + -- OK floating-point definition + + else + Def_Node := New_Node (N_Floating_Point_Definition, Digits_Loc); + end if; + + Set_Digits_Expression (Def_Node, Expr_Node); + Set_Real_Range_Specification (Def_Node, P_Real_Range_Specification_Opt); + return Def_Node; + end P_Floating_Point_Definition; + + ------------------------------------- + -- 3.5.7 Real Range Specification -- + ------------------------------------- + + -- REAL_RANGE_SPECIFICATION ::= + -- range static_SIMPLE_EXPRESSION .. static_SIMPLE_EXPRESSION + + -- Error recovery: cannot raise Error_Resync + + function P_Real_Range_Specification_Opt return Node_Id is + Specification_Node : Node_Id; + Expr_Node : Node_Id; + + begin + if Token = Tok_Range then + Specification_Node := + New_Node (N_Real_Range_Specification, Token_Ptr); + Scan; -- past RANGE + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression (Expr_Node); + Set_Low_Bound (Specification_Node, Expr_Node); + T_Dot_Dot; + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression (Expr_Node); + Set_High_Bound (Specification_Node, Expr_Node); + return Specification_Node; + else + return Empty; + end if; + end P_Real_Range_Specification_Opt; + + ----------------------------------- + -- 3.5.9 Fixed Point Definition -- + ----------------------------------- + + -- FIXED_POINT_DEFINITION ::= + -- ORDINARY_FIXED_POINT_DEFINITION | DECIMAL_FIXED_POINT_DEFINITION + + -- ORDINARY_FIXED_POINT_DEFINITION ::= + -- delta static_EXPRESSION REAL_RANGE_SPECIFICATION + + -- DECIMAL_FIXED_POINT_DEFINITION ::= + -- delta static_EXPRESSION + -- digits static_EXPRESSION [REAL_RANGE_SPECIFICATION] + + -- The caller has checked that the initial token is DELTA + + -- Error recovery: cannot raise Error_Resync + + function P_Fixed_Point_Definition return Node_Id is + Delta_Node : Node_Id; + Delta_Loc : Source_Ptr; + Def_Node : Node_Id; + Expr_Node : Node_Id; + + begin + Delta_Loc := Token_Ptr; + Scan; -- past DELTA + Delta_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Delta_Node); + + if Token = Tok_Digits then + if Ada_83 then + Error_Msg_SC ("(Ada 83) decimal fixed type not allowed!"); + end if; + + Def_Node := New_Node (N_Decimal_Fixed_Point_Definition, Delta_Loc); + Scan; -- past DIGITS + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Digits_Expression (Def_Node, Expr_Node); + + else + Def_Node := New_Node (N_Ordinary_Fixed_Point_Definition, Delta_Loc); + + -- Range is required in ordinary fixed point case + + if Token /= Tok_Range then + Error_Msg_AP ("range must be given for fixed-point type"); + T_Range; + end if; + end if; + + Set_Delta_Expression (Def_Node, Delta_Node); + Set_Real_Range_Specification (Def_Node, P_Real_Range_Specification_Opt); + return Def_Node; + end P_Fixed_Point_Definition; + + -------------------------------------------- + -- 3.5.9 Ordinary Fixed Point Definition -- + -------------------------------------------- + + -- Parsed by P_Fixed_Point_Definition (3.5.9) + + ------------------------------------------- + -- 3.5.9 Decimal Fixed Point Definition -- + ------------------------------------------- + + -- Parsed by P_Decimal_Point_Definition (3.5.9) + + ------------------------------ + -- 3.5.9 Digits Constraint -- + ------------------------------ + + -- DIGITS_CONSTRAINT ::= + -- digits static_EXPRESSION [RANGE_CONSTRAINT] + + -- Note: in Ada 83, the EXPRESSION must be a SIMPLE_EXPRESSION + + -- The caller has checked that the initial token is DIGITS + + function P_Digits_Constraint return Node_Id is + Constraint_Node : Node_Id; + Expr_Node : Node_Id; + + begin + Constraint_Node := New_Node (N_Digits_Constraint, Token_Ptr); + Scan; -- past DIGITS + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Digits_Expression (Constraint_Node, Expr_Node); + + if Token = Tok_Range then + Set_Range_Constraint (Constraint_Node, P_Range_Constraint); + end if; + + return Constraint_Node; + end P_Digits_Constraint; + + ----------------------------- + -- 3.5.9 Delta Constraint -- + ----------------------------- + + -- DELTA CONSTRAINT ::= DELTA STATIC_EXPRESSION [RANGE_CONSTRAINT] + + -- Note: this is an obsolescent feature in Ada 95 (I.3) + + -- Note: in Ada 83, the EXPRESSION must be a SIMPLE_EXPRESSION + + -- The caller has checked that the initial token is DELTA + + -- Error recovery: cannot raise Error_Resync + + function P_Delta_Constraint return Node_Id is + Constraint_Node : Node_Id; + Expr_Node : Node_Id; + + begin + Constraint_Node := New_Node (N_Delta_Constraint, Token_Ptr); + Scan; -- past DELTA + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Set_Delta_Expression (Constraint_Node, Expr_Node); + + if Token = Tok_Range then + Set_Range_Constraint (Constraint_Node, P_Range_Constraint); + end if; + + return Constraint_Node; + end P_Delta_Constraint; + + -------------------------------- + -- 3.6 Array Type Definition -- + -------------------------------- + + -- ARRAY_TYPE_DEFINITION ::= + -- UNCONSTRAINED_ARRAY_DEFINITION | CONSTRAINED_ARRAY_DEFINITION + + -- UNCONSTRAINED_ARRAY_DEFINITION ::= + -- array (INDEX_SUBTYPE_DEFINITION {, INDEX_SUBTYPE_DEFINITION}) of + -- COMPONENT_DEFINITION + + -- INDEX_SUBTYPE_DEFINITION ::= SUBTYPE_MARK range <> + + -- CONSTRAINED_ARRAY_DEFINITION ::= + -- array (DISCRETE_SUBTYPE_DEFINITION {, DISCRETE_SUBTYPE_DEFINITION}) of + -- COMPONENT_DEFINITION + + -- DISCRETE_SUBTYPE_DEFINITION ::= + -- DISCRETE_SUBTYPE_INDICATION | RANGE + + -- COMPONENT_DEFINITION ::= [aliased] SUBTYPE_INDICATION + + -- The caller has checked that the initial token is ARRAY + + -- Error recovery: can raise Error_Resync + + function P_Array_Type_Definition return Node_Id is + Array_Loc : Source_Ptr; + Def_Node : Node_Id; + Subs_List : List_Id; + Scan_State : Saved_Scan_State; + + begin + Array_Loc := Token_Ptr; + Scan; -- past ARRAY + Subs_List := New_List; + T_Left_Paren; + + -- It's quite tricky to disentangle these two possibilities, so we do + -- a prescan to determine which case we have and then reset the scan. + -- The prescan skips past possible subtype mark tokens. + + Save_Scan_State (Scan_State); -- just after paren + + while Token in Token_Class_Desig or else + Token = Tok_Dot or else + Token = Tok_Apostrophe -- because of 'BASE, 'CLASS + loop + Scan; + end loop; + + -- If we end up on RANGE <> then we have the unconstrained case. We + -- will also allow the RANGE to be omitted, just to improve error + -- handling for a case like array (integer <>) of integer; + + Scan; -- past possible RANGE or <> + + if (Prev_Token = Tok_Range and then Token = Tok_Box) or else + Prev_Token = Tok_Box + then + Def_Node := New_Node (N_Unconstrained_Array_Definition, Array_Loc); + Restore_Scan_State (Scan_State); -- to first subtype mark + + loop + Append (P_Subtype_Mark_Resync, Subs_List); + T_Range; + T_Box; + exit when Token = Tok_Right_Paren or else Token = Tok_Of; + T_Comma; + end loop; + + Set_Subtype_Marks (Def_Node, Subs_List); + + else + Def_Node := New_Node (N_Constrained_Array_Definition, Array_Loc); + Restore_Scan_State (Scan_State); -- to first discrete range + + loop + Append (P_Discrete_Subtype_Definition, Subs_List); + exit when not Comma_Present; + end loop; + + Set_Discrete_Subtype_Definitions (Def_Node, Subs_List); + end if; + + T_Right_Paren; + T_Of; + + if Token = Tok_Aliased then + Set_Aliased_Present (Def_Node, True); + Scan; -- past ALIASED + end if; + + Set_Subtype_Indication (Def_Node, P_Subtype_Indication); + return Def_Node; + end P_Array_Type_Definition; + + ----------------------------------------- + -- 3.6 Unconstrained Array Definition -- + ----------------------------------------- + + -- Parsed by P_Array_Type_Definition (3.6) + + --------------------------------------- + -- 3.6 Constrained Array Definition -- + --------------------------------------- + + -- Parsed by P_Array_Type_Definition (3.6) + + -------------------------------------- + -- 3.6 Discrete Subtype Definition -- + -------------------------------------- + + -- DISCRETE_SUBTYPE_DEFINITION ::= + -- discrete_SUBTYPE_INDICATION | RANGE + + -- Note: the discrete subtype definition appearing in a constrained + -- array definition is parsed by P_Array_Type_Definition (3.6) + + -- Error recovery: cannot raise Error_Resync + + function P_Discrete_Subtype_Definition return Node_Id is + begin + + -- The syntax of a discrete subtype definition is identical to that + -- of a discrete range, so we simply share the same parsing code. + + return P_Discrete_Range; + end P_Discrete_Subtype_Definition; + + ------------------------------- + -- 3.6 Component Definition -- + ------------------------------- + + -- For the array case, parsed by P_Array_Type_Definition (3.6) + -- For the record case, parsed by P_Component_Declaration (3.8) + + ----------------------------- + -- 3.6.1 Index Constraint -- + ----------------------------- + + -- Parsed by P_Index_Or_Discriminant_Constraint (3.7.1) + + --------------------------- + -- 3.6.1 Discrete Range -- + --------------------------- + + -- DISCRETE_RANGE ::= discrete_SUBTYPE_INDICATION | RANGE + + -- The possible forms for a discrete range are: + + -- Subtype_Mark (SUBTYPE_INDICATION, 3.2.2) + -- Subtype_Mark range Range (SUBTYPE_INDICATION, 3.2.2) + -- Range_Attribute (RANGE, 3.5) + -- Simple_Expression .. Simple_Expression (RANGE, 3.5) + + -- Error recovery: cannot raise Error_Resync + + function P_Discrete_Range return Node_Id is + Expr_Node : Node_Id; + Range_Node : Node_Id; + + begin + Expr_Node := P_Simple_Expression_Or_Range_Attribute; + + if Expr_Form = EF_Range_Attr then + return Expr_Node; + + elsif Token = Tok_Range then + if Expr_Form /= EF_Simple_Name then + Error_Msg_SC ("range must be preceded by subtype mark"); + end if; + + return P_Subtype_Indication (Expr_Node); + + -- Check Expression .. Expression case + + elsif Token = Tok_Dot_Dot then + Range_Node := New_Node (N_Range, Token_Ptr); + Set_Low_Bound (Range_Node, Expr_Node); + Scan; -- past .. + Expr_Node := P_Expression; + Check_Simple_Expression (Expr_Node); + Set_High_Bound (Range_Node, Expr_Node); + return Range_Node; + + -- Otherwise we must have a subtype mark + + elsif Expr_Form = EF_Simple_Name then + return Expr_Node; + + -- If incorrect, complain that we expect .. + + else + T_Dot_Dot; + return Expr_Node; + end if; + end P_Discrete_Range; + + ---------------------------- + -- 3.7 Discriminant Part -- + ---------------------------- + + -- DISCRIMINANT_PART ::= + -- UNKNOWN_DISCRIMINANT_PART + -- | KNOWN_DISCRIMINANT_PART + + -- A discriminant part is parsed by P_Known_Discriminant_Part_Opt (3.7) + -- or P_Unknown_Discriminant_Part (3.7), since we know which we want. + + ------------------------------------ + -- 3.7 Unknown Discriminant Part -- + ------------------------------------ + + -- UNKNOWN_DISCRIMINANT_PART ::= (<>) + + -- If no unknown discriminant part is present, then False is returned, + -- otherwise the unknown discriminant is scanned out and True is returned. + + -- Error recovery: cannot raise Error_Resync + + function P_Unknown_Discriminant_Part_Opt return Boolean is + Scan_State : Saved_Scan_State; + + begin + if Token /= Tok_Left_Paren then + return False; + + else + Save_Scan_State (Scan_State); + Scan; -- past the left paren + + if Token = Tok_Box then + + if Ada_83 then + Error_Msg_SC ("(Ada 83) unknown discriminant not allowed!"); + end if; + + Scan; -- past the box + T_Right_Paren; -- must be followed by right paren + return True; + + else + Restore_Scan_State (Scan_State); + return False; + end if; + end if; + end P_Unknown_Discriminant_Part_Opt; + + ---------------------------------- + -- 3.7 Known Discriminant Part -- + ---------------------------------- + + -- KNOWN_DISCRIMINANT_PART ::= + -- (DISCRIMINANT_SPECIFICATION {; DISCRIMINANT_SPECIFICATION}) + + -- DISCRIMINANT_SPECIFICATION ::= + -- DEFINING_IDENTIFIER_LIST : SUBTYPE_MARK + -- [:= DEFAULT_EXPRESSION] + -- | DEFINING_IDENTIFIER_LIST : ACCESS_DEFINITION + -- [:= DEFAULT_EXPRESSION] + + -- If no known discriminant part is present, then No_List is returned + + -- Error recovery: cannot raise Error_Resync + + function P_Known_Discriminant_Part_Opt return List_Id is + Specification_Node : Node_Id; + Specification_List : List_Id; + Ident_Sloc : Source_Ptr; + Scan_State : Saved_Scan_State; + Num_Idents : Nat; + Ident : Nat; + + Idents : array (Int range 1 .. 4096) of Entity_Id; + -- This array holds the list of defining identifiers. The upper bound + -- of 4096 is intended to be essentially infinite, and we do not even + -- bother to check for it being exceeded. + + begin + if Token = Tok_Left_Paren then + Specification_List := New_List; + Scan; -- past ( + P_Pragmas_Misplaced; + + Specification_Loop : loop + + Ident_Sloc := Token_Ptr; + Idents (1) := P_Defining_Identifier; + Num_Idents := 1; + + while Comma_Present loop + Num_Idents := Num_Idents + 1; + Idents (Num_Idents) := P_Defining_Identifier; + end loop; + + T_Colon; + + -- If there are multiple identifiers, we repeatedly scan the + -- type and initialization expression information by resetting + -- the scan pointer (so that we get completely separate trees + -- for each occurrence). + + if Num_Idents > 1 then + Save_Scan_State (Scan_State); + end if; + + -- Loop through defining identifiers in list + + Ident := 1; + Ident_Loop : loop + Specification_Node := + New_Node (N_Discriminant_Specification, Ident_Sloc); + Set_Defining_Identifier (Specification_Node, Idents (Ident)); + + if Token = Tok_Access then + if Ada_83 then + Error_Msg_SC + ("(Ada 83) access discriminant not allowed!"); + end if; + + Set_Discriminant_Type + (Specification_Node, P_Access_Definition); + else + Set_Discriminant_Type + (Specification_Node, P_Subtype_Mark); + No_Constraint; + end if; + + Set_Expression + (Specification_Node, Init_Expr_Opt (True)); + + if Ident > 1 then + Set_Prev_Ids (Specification_Node, True); + end if; + + if Ident < Num_Idents then + Set_More_Ids (Specification_Node, True); + end if; + + Append (Specification_Node, Specification_List); + exit Ident_Loop when Ident = Num_Idents; + Ident := Ident + 1; + Restore_Scan_State (Scan_State); + end loop Ident_Loop; + + exit Specification_Loop when Token /= Tok_Semicolon; + Scan; -- past ; + P_Pragmas_Misplaced; + end loop Specification_Loop; + + T_Right_Paren; + return Specification_List; + + else + return No_List; + end if; + end P_Known_Discriminant_Part_Opt; + + ------------------------------------- + -- 3.7 DIscriminant Specification -- + ------------------------------------- + + -- Parsed by P_Known_Discriminant_Part_Opt (3.7) + + ----------------------------- + -- 3.7 Default Expression -- + ----------------------------- + + -- Always parsed (simply as an Expression) by the parent construct + + ------------------------------------ + -- 3.7.1 Discriminant Constraint -- + ------------------------------------ + + -- Parsed by P_Index_Or_Discriminant_Constraint (3.7.1) + + -------------------------------------------------------- + -- 3.7.1 Index or Discriminant Constraint (also 3.6) -- + -------------------------------------------------------- + + -- DISCRIMINANT_CONSTRAINT ::= + -- (DISCRIMINANT_ASSOCIATION {, DISCRIMINANT_ASSOCIATION}) + + -- DISCRIMINANT_ASSOCIATION ::= + -- [discriminant_SELECTOR_NAME {| discriminant_SELECTOR_NAME} =>] + -- EXPRESSION + + -- This routine parses either an index or a discriminant constraint. As + -- is clear from the above grammar, it is often possible to clearly + -- determine which of the two possibilities we have, but there are + -- cases (those in which we have a series of expressions of the same + -- syntactic form as subtype indications), where we cannot tell. Since + -- this means that in any case the semantic phase has to distinguish + -- between the two, there is not much point in the parser trying to + -- distinguish even those cases where the difference is clear. In any + -- case, if we have a situation like: + + -- (A => 123, 235 .. 500) + + -- it is not clear which of the two items is the wrong one, better to + -- let the semantic phase give a clear message. Consequently, this + -- routine in general returns a list of items which can be either + -- discrete ranges or discriminant associations. + + -- The caller has checked that the initial token is a left paren + + -- Error recovery: can raise Error_Resync + + function P_Index_Or_Discriminant_Constraint return Node_Id is + Scan_State : Saved_Scan_State; + Constr_Node : Node_Id; + Constr_List : List_Id; + Expr_Node : Node_Id; + Result_Node : Node_Id; + + begin + Result_Node := New_Node (N_Index_Or_Discriminant_Constraint, Token_Ptr); + Scan; -- past ( + Constr_List := New_List; + Set_Constraints (Result_Node, Constr_List); + + -- The two syntactic forms are a little mixed up, so what we are doing + -- here is looking at the first entry to determine which case we have + + -- A discriminant constraint is a list of discriminant associations, + -- which have one of the following possible forms: + + -- Expression + -- Id => Expression + -- Id | Id | .. | Id => Expression + + -- An index constraint is a list of discrete ranges which have one + -- of the following possible forms: + + -- Subtype_Mark + -- Subtype_Mark range Range + -- Range_Attribute + -- Simple_Expression .. Simple_Expression + + -- Loop through discriminants in list + + loop + -- Check cases of Id => Expression or Id | Id => Expression + + if Token = Tok_Identifier then + Save_Scan_State (Scan_State); -- at Id + Scan; -- past Id + + if Token = Tok_Arrow or else Token = Tok_Vertical_Bar then + Restore_Scan_State (Scan_State); -- to Id + Append (P_Discriminant_Association, Constr_List); + goto Loop_Continue; + else + Restore_Scan_State (Scan_State); -- to Id + end if; + end if; + + -- Otherwise scan out an expression and see what we have got + + Expr_Node := P_Expression_Or_Range_Attribute; + + if Expr_Form = EF_Range_Attr then + Append (Expr_Node, Constr_List); + + elsif Token = Tok_Range then + if Expr_Form /= EF_Simple_Name then + Error_Msg_SC ("subtype mark required before RANGE"); + end if; + + Append (P_Subtype_Indication (Expr_Node), Constr_List); + goto Loop_Continue; + + -- Check Simple_Expression .. Simple_Expression case + + elsif Token = Tok_Dot_Dot then + Check_Simple_Expression (Expr_Node); + Constr_Node := New_Node (N_Range, Token_Ptr); + Set_Low_Bound (Constr_Node, Expr_Node); + Scan; -- past .. + Expr_Node := P_Expression; + Check_Simple_Expression (Expr_Node); + Set_High_Bound (Constr_Node, Expr_Node); + Append (Constr_Node, Constr_List); + goto Loop_Continue; + + -- Case of an expression which could be either form + + else + Append (Expr_Node, Constr_List); + goto Loop_Continue; + end if; + + -- Here with a single entry scanned + + <<Loop_Continue>> + exit when not Comma_Present; + + end loop; + + T_Right_Paren; + return Result_Node; + + end P_Index_Or_Discriminant_Constraint; + + ------------------------------------- + -- 3.7.1 Discriminant Association -- + ------------------------------------- + + -- DISCRIMINANT_ASSOCIATION ::= + -- [discriminant_SELECTOR_NAME {| discriminant_SELECTOR_NAME} =>] + -- EXPRESSION + + -- This routine is used only when the name list is present and the caller + -- has already checked this (by scanning ahead and repositioning the + -- scan). + + -- Error_Recovery: cannot raise Error_Resync; + + function P_Discriminant_Association return Node_Id is + Discr_Node : Node_Id; + Names_List : List_Id; + Ident_Sloc : Source_Ptr; + + begin + Ident_Sloc := Token_Ptr; + Names_List := New_List; + + loop + Append (P_Identifier, Names_List); + exit when Token /= Tok_Vertical_Bar; + Scan; -- past | + end loop; + + Discr_Node := New_Node (N_Discriminant_Association, Ident_Sloc); + Set_Selector_Names (Discr_Node, Names_List); + TF_Arrow; + Set_Expression (Discr_Node, P_Expression); + return Discr_Node; + end P_Discriminant_Association; + + --------------------------------- + -- 3.8 Record Type Definition -- + --------------------------------- + + -- RECORD_TYPE_DEFINITION ::= + -- [[abstract] tagged] [limited] RECORD_DEFINITION + + -- There is no node in the tree for a record type definition. Instead + -- a record definition node appears, with possible Abstract_Present, + -- Tagged_Present, and Limited_Present flags set appropriately. + + ---------------------------- + -- 3.8 Record Definition -- + ---------------------------- + + -- RECORD_DEFINITION ::= + -- record + -- COMPONENT_LIST + -- end record + -- | null record + + -- Note: in the case where a record definition node is used to represent + -- a record type definition, the caller sets the Tagged_Present and + -- Limited_Present flags in the resulting N_Record_Definition node as + -- required. + + -- Note that the RECORD token at the start may be missing in certain + -- error situations, so this function is expected to post the error + + -- Error recovery: can raise Error_Resync + + function P_Record_Definition return Node_Id is + Rec_Node : Node_Id; + + begin + Rec_Node := New_Node (N_Record_Definition, Token_Ptr); + + -- Null record case + + if Token = Tok_Null then + Scan; -- past NULL + T_Record; + Set_Null_Present (Rec_Node, True); + + -- Case starting with RECORD keyword. Build scope stack entry. For the + -- column, we use the first non-blank character on the line, to deal + -- with situations such as: + + -- type X is record + -- ... + -- end record; + + -- which is not official RM indentation, but is not uncommon usage + + else + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Record; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Labl := Error; + Scope.Table (Scope.Last).Junk := (Token /= Tok_Record); + + T_Record; + + Set_Component_List (Rec_Node, P_Component_List); + + loop + exit when Check_End; + Discard_Junk_Node (P_Component_List); + end loop; + end if; + + return Rec_Node; + end P_Record_Definition; + + ------------------------- + -- 3.8 Component List -- + ------------------------- + + -- COMPONENT_LIST ::= + -- COMPONENT_ITEM {COMPONENT_ITEM} + -- | {COMPONENT_ITEM} VARIANT_PART + -- | null; + + -- Error recovery: cannot raise Error_Resync + + function P_Component_List return Node_Id is + Component_List_Node : Node_Id; + Decls_List : List_Id; + Scan_State : Saved_Scan_State; + + begin + Component_List_Node := New_Node (N_Component_List, Token_Ptr); + Decls_List := New_List; + + if Token = Tok_Null then + Scan; -- past NULL + TF_Semicolon; + P_Pragmas_Opt (Decls_List); + Set_Null_Present (Component_List_Node, True); + return Component_List_Node; + + else + P_Pragmas_Opt (Decls_List); + + if Token /= Tok_Case then + Component_Scan_Loop : loop + P_Component_Items (Decls_List); + P_Pragmas_Opt (Decls_List); + + exit Component_Scan_Loop when Token = Tok_End + or else Token = Tok_Case + or else Token = Tok_When; + + -- We are done if we do not have an identifier. However, if + -- we have a misspelled reserved identifier that is in a column + -- to the right of the record definition, we will treat it as + -- an identifier. It turns out to be too dangerous in practice + -- to accept such a mis-spelled identifier which does not have + -- this additional clue that confirms the incorrect spelling. + + if Token /= Tok_Identifier then + if Start_Column > Scope.Table (Scope.Last).Ecol + and then Is_Reserved_Identifier + then + Save_Scan_State (Scan_State); -- at reserved id + Scan; -- possible reserved id + + if Token = Tok_Comma or else Token = Tok_Colon then + Restore_Scan_State (Scan_State); + Scan_Reserved_Identifier (Force_Msg => True); + + -- Note reserved identifier used as field name after + -- all because not followed by colon or comma + + else + Restore_Scan_State (Scan_State); + exit Component_Scan_Loop; + end if; + + -- Non-identifier that definitely was not reserved id + + else + exit Component_Scan_Loop; + end if; + end if; + end loop Component_Scan_Loop; + end if; + + if Token = Tok_Case then + Set_Variant_Part (Component_List_Node, P_Variant_Part); + + -- Check for junk after variant part + + if Token = Tok_Identifier then + Save_Scan_State (Scan_State); + Scan; -- past identifier + + if Token = Tok_Colon then + Restore_Scan_State (Scan_State); + Error_Msg_SC ("component may not follow variant part"); + Discard_Junk_Node (P_Component_List); + + elsif Token = Tok_Case then + Restore_Scan_State (Scan_State); + Error_Msg_SC ("only one variant part allowed in a record"); + Discard_Junk_Node (P_Component_List); + + else + Restore_Scan_State (Scan_State); + end if; + end if; + end if; + end if; + + Set_Component_Items (Component_List_Node, Decls_List); + return Component_List_Node; + + end P_Component_List; + + ------------------------- + -- 3.8 Component Item -- + ------------------------- + + -- COMPONENT_ITEM ::= COMPONENT_DECLARATION | REPRESENTATION_CLAUSE + + -- COMPONENT_DECLARATION ::= + -- DEFINING_IDENTIFIER_LIST : COMPONENT_DEFINITION + -- [:= DEFAULT_EXPRESSION]; + + -- COMPONENT_DEFINITION ::= [aliased] SUBTYPE_INDICATION + + -- Error recovery: cannot raise Error_Resync, if an error occurs, + -- the scan is positioned past the following semicolon. + + -- Note: we do not yet allow representation clauses to appear as component + -- items, do we need to add this capability sometime in the future ??? + + procedure P_Component_Items (Decls : List_Id) is + Decl_Node : Node_Id; + Scan_State : Saved_Scan_State; + Num_Idents : Nat; + Ident : Nat; + Ident_Sloc : Source_Ptr; + + Idents : array (Int range 1 .. 4096) of Entity_Id; + -- This array holds the list of defining identifiers. The upper bound + -- of 4096 is intended to be essentially infinite, and we do not even + -- bother to check for it being exceeded. + + begin + if Token /= Tok_Identifier then + Error_Msg_SC ("component declaration expected"); + Resync_Past_Semicolon; + return; + end if; + + Ident_Sloc := Token_Ptr; + Idents (1) := P_Defining_Identifier; + Num_Idents := 1; + + while Comma_Present loop + Num_Idents := Num_Idents + 1; + Idents (Num_Idents) := P_Defining_Identifier; + end loop; + + T_Colon; + + -- If there are multiple identifiers, we repeatedly scan the + -- type and initialization expression information by resetting + -- the scan pointer (so that we get completely separate trees + -- for each occurrence). + + if Num_Idents > 1 then + Save_Scan_State (Scan_State); + end if; + + -- Loop through defining identifiers in list + + Ident := 1; + Ident_Loop : loop + + -- The following block is present to catch Error_Resync + -- which causes the parse to be reset past the semicolon + + begin + Decl_Node := New_Node (N_Component_Declaration, Ident_Sloc); + Set_Defining_Identifier (Decl_Node, Idents (Ident)); + + if Token = Tok_Constant then + Error_Msg_SC ("constant components are not permitted"); + Scan; + end if; + + if Token_Name = Name_Aliased then + Check_95_Keyword (Tok_Aliased, Tok_Identifier); + end if; + + if Token = Tok_Aliased then + Scan; -- past ALIASED + Set_Aliased_Present (Decl_Node, True); + end if; + + if Token = Tok_Array then + Error_Msg_SC ("anonymous arrays not allowed as components"); + raise Error_Resync; + end if; + + Set_Subtype_Indication (Decl_Node, P_Subtype_Indication); + Set_Expression (Decl_Node, Init_Expr_Opt); + + if Ident > 1 then + Set_Prev_Ids (Decl_Node, True); + end if; + + if Ident < Num_Idents then + Set_More_Ids (Decl_Node, True); + end if; + + Append (Decl_Node, Decls); + + exception + when Error_Resync => + if Token /= Tok_End then + Resync_Past_Semicolon; + end if; + end; + + exit Ident_Loop when Ident = Num_Idents; + Ident := Ident + 1; + Restore_Scan_State (Scan_State); + + end loop Ident_Loop; + + TF_Semicolon; + + end P_Component_Items; + + -------------------------------- + -- 3.8 Component Declaration -- + -------------------------------- + + -- Parsed by P_Component_Items (3.8) + + ------------------------- + -- 3.8.1 Variant Part -- + ------------------------- + + -- VARIANT_PART ::= + -- case discriminant_DIRECT_NAME is + -- VARIANT + -- {VARIANT} + -- end case; + + -- The caller has checked that the initial token is CASE + + -- Error recovery: cannot raise Error_Resync + + function P_Variant_Part return Node_Id is + Variant_Part_Node : Node_Id; + Variants_List : List_Id; + Case_Node : Node_Id; + Case_Sloc : Source_Ptr; + + begin + Variant_Part_Node := New_Node (N_Variant_Part, Token_Ptr); + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Case; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Ecol := Start_Column; + + Scan; -- past CASE + Case_Node := P_Expression; + Case_Sloc := Token_Ptr; + Set_Name (Variant_Part_Node, Case_Node); + + if Nkind (Case_Node) /= N_Identifier then + Set_Name (Variant_Part_Node, Error); + Error_Msg ("discriminant name expected", Sloc (Case_Node)); + end if; + + TF_Is; + Variants_List := New_List; + P_Pragmas_Opt (Variants_List); + + -- Test missing variant + + if Token = Tok_End then + Error_Msg_BC ("WHEN expected (must have at least one variant)"); + else + Append (P_Variant, Variants_List); + end if; + + -- Loop through variants, note that we allow if in place of when, + -- this error will be detected and handled in P_Variant. + + loop + P_Pragmas_Opt (Variants_List); + + if Token /= Tok_When + and then Token /= Tok_If + and then Token /= Tok_Others + then + exit when Check_End; + end if; + + Append (P_Variant, Variants_List); + end loop; + + Set_Variants (Variant_Part_Node, Variants_List); + return Variant_Part_Node; + + end P_Variant_Part; + + -------------------- + -- 3.8.1 Variant -- + -------------------- + + -- VARIANT ::= + -- when DISCRETE_CHOICE_LIST => + -- COMPONENT_LIST + + -- Error recovery: cannot raise Error_Resync + + -- The initial token on entry is either WHEN, IF or OTHERS + + function P_Variant return Node_Id is + Variant_Node : Node_Id; + + begin + -- Special check to recover nicely from use of IF in place of WHEN + + if Token = Tok_If then + T_When; + Scan; -- past IF + else + T_When; + end if; + + Variant_Node := New_Node (N_Variant, Prev_Token_Ptr); + Set_Discrete_Choices (Variant_Node, P_Discrete_Choice_List); + TF_Arrow; + Set_Component_List (Variant_Node, P_Component_List); + return Variant_Node; + end P_Variant; + + --------------------------------- + -- 3.8.1 Discrete Choice List -- + --------------------------------- + + -- DISCRETE_CHOICE_LIST ::= DISCRETE_CHOICE {| DISCRETE_CHOICE} + + -- DISCRETE_CHOICE ::= EXPRESSION | DISCRETE_RANGE | others + + -- Note: in Ada 83, the expression must be a simple expression + + -- Error recovery: cannot raise Error_Resync + + function P_Discrete_Choice_List return List_Id is + Choices : List_Id; + Expr_Node : Node_Id; + Choice_Node : Node_Id; + + begin + Choices := New_List; + + loop + if Token = Tok_Others then + Append (New_Node (N_Others_Choice, Token_Ptr), Choices); + Scan; -- past OTHERS + + else + begin + Expr_Node := No_Right_Paren (P_Expression_Or_Range_Attribute); + + if Token = Tok_Colon + and then Nkind (Expr_Node) = N_Identifier + then + Error_Msg_SP ("label not permitted in this context"); + Scan; -- past colon + + elsif Expr_Form = EF_Range_Attr then + Append (Expr_Node, Choices); + + elsif Token = Tok_Dot_Dot then + Check_Simple_Expression (Expr_Node); + Choice_Node := New_Node (N_Range, Token_Ptr); + Set_Low_Bound (Choice_Node, Expr_Node); + Scan; -- past .. + Expr_Node := P_Expression_No_Right_Paren; + Check_Simple_Expression (Expr_Node); + Set_High_Bound (Choice_Node, Expr_Node); + Append (Choice_Node, Choices); + + elsif Expr_Form = EF_Simple_Name then + if Token = Tok_Range then + Append (P_Subtype_Indication (Expr_Node), Choices); + + elsif Token in Token_Class_Consk then + Error_Msg_SC + ("the only constraint allowed here " & + "is a range constraint"); + Discard_Junk_Node (P_Constraint_Opt); + Append (Expr_Node, Choices); + + else + Append (Expr_Node, Choices); + end if; + + else + Check_Simple_Expression_In_Ada_83 (Expr_Node); + Append (Expr_Node, Choices); + end if; + + exception + when Error_Resync => + Resync_Choice; + return Error_List; + end; + end if; + + if Token = Tok_Comma then + Error_Msg_SC (""","" should be ""|"""); + else + exit when Token /= Tok_Vertical_Bar; + end if; + + Scan; -- past | or comma + end loop; + + return Choices; + end P_Discrete_Choice_List; + + ---------------------------- + -- 3.8.1 Discrete Choice -- + ---------------------------- + + -- Parsed by P_Discrete_Choice_List (3.8.1) + + ---------------------------------- + -- 3.9.1 Record Extension Part -- + ---------------------------------- + + -- RECORD_EXTENSION_PART ::= with RECORD_DEFINITION + + -- Parsed by P_Derived_Type_Def_Or_Private_Ext_Decl (3.4) + + ---------------------------------- + -- 3.10 Access Type Definition -- + ---------------------------------- + + -- ACCESS_TYPE_DEFINITION ::= + -- ACCESS_TO_OBJECT_DEFINITION + -- | ACCESS_TO_SUBPROGRAM_DEFINITION + + -- ACCESS_TO_OBJECT_DEFINITION ::= + -- access [GENERAL_ACCESS_MODIFIER] SUBTYPE_INDICATION + + -- GENERAL_ACCESS_MODIFIER ::= all | constant + + -- ACCESS_TO_SUBPROGRAM_DEFINITION + -- access [protected] procedure PARAMETER_PROFILE + -- | access [protected] function PARAMETER_AND_RESULT_PROFILE + + -- PARAMETER_PROFILE ::= [FORMAL_PART] + + -- PARAMETER_AND_RESULT_PROFILE ::= [FORMAL_PART] RETURN SUBTYPE_MARK + + -- The caller has checked that the initial token is ACCESS + + -- Error recovery: can raise Error_Resync + + function P_Access_Type_Definition return Node_Id is + Prot_Flag : Boolean; + Access_Loc : Source_Ptr; + Type_Def_Node : Node_Id; + + procedure Check_Junk_Subprogram_Name; + -- Used in access to subprogram definition cases to check for an + -- identifier or operator symbol that does not belong. + + procedure Check_Junk_Subprogram_Name is + Saved_State : Saved_Scan_State; + + begin + if Token = Tok_Identifier or else Token = Tok_Operator_Symbol then + Save_Scan_State (Saved_State); + Scan; -- past possible junk subprogram name + + if Token = Tok_Left_Paren or else Token = Tok_Semicolon then + Error_Msg_SP ("unexpected subprogram name ignored"); + return; + + else + Restore_Scan_State (Saved_State); + end if; + end if; + end Check_Junk_Subprogram_Name; + + -- Start of processing for P_Access_Type_Definition + + begin + Access_Loc := Token_Ptr; + Scan; -- past ACCESS + + if Token_Name = Name_Protected then + Check_95_Keyword (Tok_Protected, Tok_Procedure); + Check_95_Keyword (Tok_Protected, Tok_Function); + end if; + + Prot_Flag := (Token = Tok_Protected); + + if Prot_Flag then + Scan; -- past PROTECTED + if Token /= Tok_Procedure and then Token /= Tok_Function then + Error_Msg_SC ("FUNCTION or PROCEDURE expected"); + end if; + end if; + + if Token = Tok_Procedure then + if Ada_83 then + Error_Msg_SC ("(Ada 83) access to procedure not allowed!"); + end if; + + Type_Def_Node := New_Node (N_Access_Procedure_Definition, Access_Loc); + Scan; -- past PROCEDURE + Check_Junk_Subprogram_Name; + Set_Parameter_Specifications (Type_Def_Node, P_Parameter_Profile); + Set_Protected_Present (Type_Def_Node, Prot_Flag); + + elsif Token = Tok_Function then + if Ada_83 then + Error_Msg_SC ("(Ada 83) access to function not allowed!"); + end if; + + Type_Def_Node := New_Node (N_Access_Function_Definition, Access_Loc); + Scan; -- past FUNCTION + Check_Junk_Subprogram_Name; + Set_Parameter_Specifications (Type_Def_Node, P_Parameter_Profile); + Set_Protected_Present (Type_Def_Node, Prot_Flag); + TF_Return; + Set_Subtype_Mark (Type_Def_Node, P_Subtype_Mark); + No_Constraint; + + else + Type_Def_Node := + New_Node (N_Access_To_Object_Definition, Access_Loc); + + if Token = Tok_All or else Token = Tok_Constant then + if Ada_83 then + Error_Msg_SC ("(Ada 83) access modifier not allowed!"); + end if; + + if Token = Tok_All then + Set_All_Present (Type_Def_Node, True); + + else + Set_Constant_Present (Type_Def_Node, True); + end if; + + Scan; -- past ALL or CONSTANT + end if; + + Set_Subtype_Indication (Type_Def_Node, P_Subtype_Indication); + end if; + + return Type_Def_Node; + end P_Access_Type_Definition; + + --------------------------------------- + -- 3.10 Access To Object Definition -- + --------------------------------------- + + -- Parsed by P_Access_Type_Definition (3.10) + + ----------------------------------- + -- 3.10 General Access Modifier -- + ----------------------------------- + + -- Parsed by P_Access_Type_Definition (3.10) + + ------------------------------------------- + -- 3.10 Access To Subprogram Definition -- + ------------------------------------------- + + -- Parsed by P_Access_Type_Definition (3.10) + + ----------------------------- + -- 3.10 Access Definition -- + ----------------------------- + + -- ACCESS_DEFINITION ::= access SUBTYPE_MARK + + -- The caller has checked that the initial token is ACCESS + + -- Error recovery: cannot raise Error_Resync + + function P_Access_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := New_Node (N_Access_Definition, Token_Ptr); + Scan; -- past ACCESS + Set_Subtype_Mark (Def_Node, P_Subtype_Mark); + No_Constraint; + return Def_Node; + end P_Access_Definition; + + ----------------------------------------- + -- 3.10.1 Incomplete Type Declaration -- + ----------------------------------------- + + -- Parsed by P_Type_Declaration (3.2.1) + + ---------------------------- + -- 3.11 Declarative Part -- + ---------------------------- + + -- DECLARATIVE_PART ::= {DECLARATIVE_ITEM} + + -- Error recovery: cannot raise Error_Resync (because P_Declarative_Items + -- handles errors, and returns cleanly after an error has occurred) + + function P_Declarative_Part return List_Id is + Decls : List_Id; + Done : Boolean; + + begin + -- Indicate no bad declarations detected yet. This will be reset by + -- P_Declarative_Items if a bad declaration is discovered. + + Missing_Begin_Msg := No_Error_Msg; + + -- Get rid of active SIS entry from outer scope. This means we will + -- miss some nested cases, but it doesn't seem worth the effort. See + -- discussion in Par for further details + + SIS_Entry_Active := False; + Decls := New_List; + + -- Loop to scan out the declarations + + loop + P_Declarative_Items (Decls, Done, In_Spec => False); + exit when Done; + end loop; + + -- Get rid of active SIS entry which is left set only if we scanned a + -- procedure declaration and have not found the body. We could give + -- an error message, but that really would be usurping the role of + -- semantic analysis (this really is a missing body case). + + SIS_Entry_Active := False; + return Decls; + end P_Declarative_Part; + + ---------------------------- + -- 3.11 Declarative Item -- + ---------------------------- + + -- DECLARATIVE_ITEM ::= BASIC_DECLARATIVE_ITEM | BODY + + -- Can return Error if a junk declaration is found, or Empty if no + -- declaration is found (i.e. a token ending declarations, such as + -- BEGIN or END is encountered). + + -- Error recovery: cannot raise Error_Resync. If an error resync occurs, + -- then the scan is set past the next semicolon and Error is returned. + + procedure P_Declarative_Items + (Decls : List_Id; + Done : out Boolean; + In_Spec : Boolean) + is + Scan_State : Saved_Scan_State; + + begin + if Style_Check then Style.Check_Indentation; end if; + + case Token is + + when Tok_Function => + Check_Bad_Layout; + Append (P_Subprogram (Pf_Decl_Gins_Pbod_Rnam_Stub), Decls); + Done := False; + + when Tok_For => + Check_Bad_Layout; + + -- Check for loop (premature statement) + + Save_Scan_State (Scan_State); + Scan; -- past FOR + + if Token = Tok_Identifier then + Scan; -- past identifier + + if Token = Tok_In then + Restore_Scan_State (Scan_State); + Statement_When_Declaration_Expected (Decls, Done, In_Spec); + return; + end if; + end if; + + -- Not a loop, so must be rep clause + + Restore_Scan_State (Scan_State); + Append (P_Representation_Clause, Decls); + Done := False; + + when Tok_Generic => + Check_Bad_Layout; + Append (P_Generic, Decls); + Done := False; + + when Tok_Identifier => + Check_Bad_Layout; + P_Identifier_Declarations (Decls, Done, In_Spec); + + when Tok_Package => + Check_Bad_Layout; + Append (P_Package (Pf_Decl_Gins_Pbod_Rnam_Stub), Decls); + Done := False; + + when Tok_Pragma => + Append (P_Pragma, Decls); + Done := False; + + when Tok_Procedure => + Check_Bad_Layout; + Append (P_Subprogram (Pf_Decl_Gins_Pbod_Rnam_Stub), Decls); + Done := False; + + when Tok_Protected => + Check_Bad_Layout; + Scan; -- past PROTECTED + Append (P_Protected, Decls); + Done := False; + + when Tok_Subtype => + Check_Bad_Layout; + Append (P_Subtype_Declaration, Decls); + Done := False; + + when Tok_Task => + Check_Bad_Layout; + Scan; -- past TASK + Append (P_Task, Decls); + Done := False; + + when Tok_Type => + Check_Bad_Layout; + Append (P_Type_Declaration, Decls); + Done := False; + + when Tok_Use => + Check_Bad_Layout; + Append (P_Use_Clause, Decls); + Done := False; + + when Tok_With => + Check_Bad_Layout; + Error_Msg_SC ("WITH can only appear in context clause"); + raise Error_Resync; + + -- BEGIN terminates the scan of a sequence of declarations unless + -- there is a missing subprogram body, see section on handling + -- semicolon in place of IS. We only treat the begin as satisfying + -- the subprogram declaration if it falls in the expected column + -- or to its right. + + when Tok_Begin => + if SIS_Entry_Active and then Start_Column >= SIS_Ecol then + + -- Here we have the case where a BEGIN is encountered during + -- declarations in a declarative part, or at the outer level, + -- and there is a subprogram declaration outstanding for which + -- no body has been supplied. This is the case where we assume + -- that the semicolon in the subprogram declaration should + -- really have been is. The active SIS entry describes the + -- subprogram declaration. On return the declaration has been + -- modified to become a body. + + declare + Specification_Node : Node_Id; + Decl_Node : Node_Id; + Body_Node : Node_Id; + + begin + -- First issue the error message. If we had a missing + -- semicolon in the declaration, then change the message + -- to <missing "is"> + + if SIS_Missing_Semicolon_Message /= No_Error_Msg then + Change_Error_Text -- Replace: "missing "";"" " + (SIS_Missing_Semicolon_Message, "missing ""is"""); + + -- Otherwise we saved the semicolon position, so complain + + else + Error_Msg (""";"" should be IS", SIS_Semicolon_Sloc); + end if; + + -- The next job is to fix up any declarations that occurred + -- between the procedure header and the BEGIN. These got + -- chained to the outer declarative region (immediately + -- after the procedure declaration) and they should be + -- chained to the subprogram itself, which is a body + -- rather than a spec. + + Specification_Node := Specification (SIS_Declaration_Node); + Change_Node (SIS_Declaration_Node, N_Subprogram_Body); + Body_Node := SIS_Declaration_Node; + Set_Specification (Body_Node, Specification_Node); + Set_Declarations (Body_Node, New_List); + + loop + Decl_Node := Remove_Next (Body_Node); + exit when Decl_Node = Empty; + Append (Decl_Node, Declarations (Body_Node)); + end loop; + + -- Now make the scope table entry for the Begin-End and + -- scan it out + + Push_Scope_Stack; + Scope.Table (Scope.Last).Sloc := SIS_Sloc; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Ecol := SIS_Ecol; + Scope.Table (Scope.Last).Labl := SIS_Labl; + Scope.Table (Scope.Last).Lreq := False; + SIS_Entry_Active := False; + Scan; -- past BEGIN + Set_Handled_Statement_Sequence (Body_Node, + P_Handled_Sequence_Of_Statements); + End_Statements (Handled_Statement_Sequence (Body_Node)); + end; + + Done := False; + + else + Done := True; + end if; + + -- Normally an END terminates the scan for basic declarative + -- items. The one exception is END RECORD, which is probably + -- left over from some other junk. + + when Tok_End => + Save_Scan_State (Scan_State); -- at END + Scan; -- past END + + if Token = Tok_Record then + Error_Msg_SP ("no RECORD for this `end record`!"); + Scan; -- past RECORD + TF_Semicolon; + + else + Restore_Scan_State (Scan_State); -- to END + Done := True; + end if; + + -- The following tokens which can only be the start of a statement + -- are considered to end a declarative part (i.e. we have a missing + -- BEGIN situation). We are fairly conservative in making this + -- judgment, because it is a real mess to go into statement mode + -- prematurely in reponse to a junk declaration. + + when Tok_Abort | + Tok_Accept | + Tok_Declare | + Tok_Delay | + Tok_Exit | + Tok_Goto | + Tok_If | + Tok_Loop | + Tok_Null | + Tok_Requeue | + Tok_Select | + Tok_While => + + -- But before we decide that it's a statement, let's check for + -- a reserved word misused as an identifier. + + if Is_Reserved_Identifier then + Save_Scan_State (Scan_State); + Scan; -- past the token + + -- If reserved identifier not followed by colon or comma, then + -- this is most likely an assignment statement to the bad id. + + if Token /= Tok_Colon and then Token /= Tok_Comma then + Restore_Scan_State (Scan_State); + Statement_When_Declaration_Expected (Decls, Done, In_Spec); + return; + + -- Otherwise we have a declaration of the bad id + + else + Restore_Scan_State (Scan_State); + Scan_Reserved_Identifier (Force_Msg => True); + P_Identifier_Declarations (Decls, Done, In_Spec); + end if; + + -- If not reserved identifier, then it's definitely a statement + + else + Statement_When_Declaration_Expected (Decls, Done, In_Spec); + return; + end if; + + -- The token RETURN may well also signal a missing BEGIN situation, + -- however, we never let it end the declarative part, because it may + -- also be part of a half-baked function declaration. + + when Tok_Return => + Error_Msg_SC ("misplaced RETURN statement"); + raise Error_Resync; + + -- PRIVATE definitely terminates the declarations in a spec, + -- and is an error in a body. + + when Tok_Private => + if In_Spec then + Done := True; + else + Error_Msg_SC ("PRIVATE not allowed in body"); + Scan; -- past PRIVATE + end if; + + -- An end of file definitely terminates the declarations! + + when Tok_EOF => + Done := True; + + -- The remaining tokens do not end the scan, but cannot start a + -- valid declaration, so we signal an error and resynchronize. + -- But first check for misuse of a reserved identifier. + + when others => + + -- Here we check for a reserved identifier + + if Is_Reserved_Identifier then + Save_Scan_State (Scan_State); + Scan; -- past the token + + if Token /= Tok_Colon and then Token /= Tok_Comma then + Restore_Scan_State (Scan_State); + Set_Declaration_Expected; + raise Error_Resync; + else + Restore_Scan_State (Scan_State); + Scan_Reserved_Identifier (Force_Msg => True); + Check_Bad_Layout; + P_Identifier_Declarations (Decls, Done, In_Spec); + end if; + + else + Set_Declaration_Expected; + raise Error_Resync; + end if; + end case; + + -- To resynchronize after an error, we scan to the next semicolon and + -- return with Done = False, indicating that there may still be more + -- valid declarations to come. + + exception + when Error_Resync => + Resync_Past_Semicolon; + Done := False; + + end P_Declarative_Items; + + ---------------------------------- + -- 3.11 Basic Declarative Item -- + ---------------------------------- + + -- BASIC_DECLARATIVE_ITEM ::= + -- BASIC_DECLARATION | REPRESENTATION_CLAUSE | USE_CLAUSE + + -- Scan zero or more basic declarative items + + -- Error recovery: cannot raise Error_Resync. If an error is detected, then + -- the scan pointer is repositioned past the next semicolon, and the scan + -- for declarative items continues. + + function P_Basic_Declarative_Items return List_Id is + Decl : Node_Id; + Decls : List_Id; + Kind : Node_Kind; + Done : Boolean; + + begin + -- Get rid of active SIS entry from outer scope. This means we will + -- miss some nested cases, but it doesn't seem worth the effort. See + -- discussion in Par for further details + + SIS_Entry_Active := False; + + -- Loop to scan out declarations + + Decls := New_List; + + loop + P_Declarative_Items (Decls, Done, In_Spec => True); + exit when Done; + end loop; + + -- Get rid of active SIS entry. This is set only if we have scanned a + -- procedure declaration and have not found the body. We could give + -- an error message, but that really would be usurping the role of + -- semantic analysis (this really is a case of a missing body). + + SIS_Entry_Active := False; + + -- Test for assorted illegal declarations not diagnosed elsewhere. + + Decl := First (Decls); + + while Present (Decl) loop + Kind := Nkind (Decl); + + -- Test for body scanned, not acceptable as basic decl item + + if Kind = N_Subprogram_Body or else + Kind = N_Package_Body or else + Kind = N_Task_Body or else + Kind = N_Protected_Body + then + Error_Msg + ("proper body not allowed in package spec", Sloc (Decl)); + + -- Test for body stub scanned, not acceptable as basic decl item + + elsif Kind in N_Body_Stub then + Error_Msg + ("body stub not allowed in package spec", Sloc (Decl)); + + elsif Kind = N_Assignment_Statement then + Error_Msg + ("assignment statement not allowed in package spec", + Sloc (Decl)); + end if; + + Next (Decl); + end loop; + + return Decls; + end P_Basic_Declarative_Items; + + ---------------- + -- 3.11 Body -- + ---------------- + + -- For proper body, see below + -- For body stub, see 10.1.3 + + ----------------------- + -- 3.11 Proper Body -- + ----------------------- + + -- Subprogram body is parsed by P_Subprogram (6.1) + -- Package body is parsed by P_Package (7.1) + -- Task body is parsed by P_Task (9.1) + -- Protected body is parsed by P_Protected (9.4) + + ------------------------------ + -- Set_Declaration_Expected -- + ------------------------------ + + procedure Set_Declaration_Expected is + begin + Error_Msg_SC ("declaration expected"); + + if Missing_Begin_Msg = No_Error_Msg then + Missing_Begin_Msg := Get_Msg_Id; + end if; + end Set_Declaration_Expected; + + ---------------------- + -- Skip_Declaration -- + ---------------------- + + procedure Skip_Declaration (S : List_Id) is + Dummy_Done : Boolean; + + begin + P_Declarative_Items (S, Dummy_Done, False); + end Skip_Declaration; + + ----------------------------------------- + -- Statement_When_Declaration_Expected -- + ----------------------------------------- + + procedure Statement_When_Declaration_Expected + (Decls : List_Id; + Done : out Boolean; + In_Spec : Boolean) + is + begin + -- Case of second occurrence of statement in one declaration sequence + + if Missing_Begin_Msg /= No_Error_Msg then + + -- In the procedure spec case, just ignore it, we only give one + -- message for the first occurrence, since otherwise we may get + -- horrible cascading if BODY was missing in the header line. + + if In_Spec then + null; + + -- In the declarative part case, take a second statement as a sure + -- sign that we really have a missing BEGIN, and end the declarative + -- part now. Note that the caller will fix up the first message to + -- say "missing BEGIN" so that's how the error will be signalled. + + else + Done := True; + return; + end if; + + -- Case of first occurrence of unexpected statement + + else + -- If we are in a package spec, then give message of statement + -- not allowed in package spec. This message never gets changed. + + if In_Spec then + Error_Msg_SC ("statement not allowed in package spec"); + + -- If in declarative part, then we give the message complaining + -- about finding a statement when a declaration is expected. This + -- gets changed to a complaint about a missing BEGIN if we later + -- find that no BEGIN is present. + + else + Error_Msg_SC ("statement not allowed in declarative part"); + end if; + + -- Capture message Id. This is used for two purposes, first to + -- stop multiple messages, see test above, and second, to allow + -- the replacement of the message in the declarative part case. + + Missing_Begin_Msg := Get_Msg_Id; + end if; + + -- In all cases except the case in which we decided to terminate the + -- declaration sequence on a second error, we scan out the statement + -- and append it to the list of declarations (note that the semantics + -- can handle statements in a declaration list so if we proceed to + -- call the semantic phase, all will be (reasonably) well! + + Append_List_To (Decls, P_Sequence_Of_Statements (SS_Unco)); + + -- Done is set to False, since we want to continue the scan of + -- declarations, hoping that this statement was a temporary glitch. + -- If we indeed are now in the statement part (i.e. this was a missing + -- BEGIN, then it's not terrible, we will simply keep calling this + -- procedure to process the statements one by one, and then finally + -- hit the missing BEGIN, which will clean up the error message. + + Done := False; + + end Statement_When_Declaration_Expected; + +end Ch3; diff --git a/gcc/ada/par-ch4.adb b/gcc/ada/par-ch4.adb new file mode 100644 index 00000000000..30fba5619cc --- /dev/null +++ b/gcc/ada/par-ch4.adb @@ -0,0 +1,2298 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 4 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.91 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch4 is + + ----------------------- + -- Local Subprograms -- + ----------------------- + + function P_Aggregate_Or_Paren_Expr return Node_Id; + function P_Allocator return Node_Id; + function P_Record_Or_Array_Component_Association return Node_Id; + function P_Factor return Node_Id; + function P_Primary return Node_Id; + function P_Relation return Node_Id; + function P_Term return Node_Id; + + function P_Binary_Adding_Operator return Node_Kind; + function P_Logical_Operator return Node_Kind; + function P_Multiplying_Operator return Node_Kind; + function P_Relational_Operator return Node_Kind; + function P_Unary_Adding_Operator return Node_Kind; + + procedure Bad_Range_Attribute (Loc : Source_Ptr); + -- Called to place complaint about bad range attribute at the given + -- source location. Terminates by raising Error_Resync. + + function P_Range_Attribute_Reference + (Prefix_Node : Node_Id) + return Node_Id; + -- Scan a range attribute reference. The caller has scanned out the + -- prefix. The current token is known to be an apostrophe and the + -- following token is known to be RANGE. + + procedure Set_Op_Name (Node : Node_Id); + -- Procedure to set name field (Chars) in operator node + + ------------------------- + -- Bad_Range_Attribute -- + ------------------------- + + procedure Bad_Range_Attribute (Loc : Source_Ptr) is + begin + Error_Msg ("range attribute cannot be used in expression", Loc); + Resync_Expression; + end Bad_Range_Attribute; + + ------------------ + -- Set_Op_Name -- + ------------------ + + procedure Set_Op_Name (Node : Node_Id) is + type Name_Of_Type is array (N_Op) of Name_Id; + Name_Of : Name_Of_Type := Name_Of_Type'( + N_Op_And => Name_Op_And, + N_Op_Or => Name_Op_Or, + N_Op_Xor => Name_Op_Xor, + N_Op_Eq => Name_Op_Eq, + N_Op_Ne => Name_Op_Ne, + N_Op_Lt => Name_Op_Lt, + N_Op_Le => Name_Op_Le, + N_Op_Gt => Name_Op_Gt, + N_Op_Ge => Name_Op_Ge, + N_Op_Add => Name_Op_Add, + N_Op_Subtract => Name_Op_Subtract, + N_Op_Concat => Name_Op_Concat, + N_Op_Multiply => Name_Op_Multiply, + N_Op_Divide => Name_Op_Divide, + N_Op_Mod => Name_Op_Mod, + N_Op_Rem => Name_Op_Rem, + N_Op_Expon => Name_Op_Expon, + N_Op_Plus => Name_Op_Add, + N_Op_Minus => Name_Op_Subtract, + N_Op_Abs => Name_Op_Abs, + N_Op_Not => Name_Op_Not, + + -- We don't really need these shift operators, since they never + -- appear as operators in the source, but the path of least + -- resistance is to put them in (the aggregate must be complete) + + N_Op_Rotate_Left => Name_Rotate_Left, + N_Op_Rotate_Right => Name_Rotate_Right, + N_Op_Shift_Left => Name_Shift_Left, + N_Op_Shift_Right => Name_Shift_Right, + N_Op_Shift_Right_Arithmetic => Name_Shift_Right_Arithmetic); + + begin + if Nkind (Node) in N_Op then + Set_Chars (Node, Name_Of (Nkind (Node))); + end if; + end Set_Op_Name; + + -------------------------- + -- 4.1 Name (also 6.4) -- + -------------------------- + + -- NAME ::= + -- DIRECT_NAME | EXPLICIT_DEREFERENCE + -- | INDEXED_COMPONENT | SLICE + -- | SELECTED_COMPONENT | ATTRIBUTE + -- | TYPE_CONVERSION | FUNCTION_CALL + -- | CHARACTER_LITERAL + + -- DIRECT_NAME ::= IDENTIFIER | OPERATOR_SYMBOL + + -- PREFIX ::= NAME | IMPLICIT_DEREFERENCE + + -- EXPLICIT_DEREFERENCE ::= NAME . all + + -- IMPLICIT_DEREFERENCE ::= NAME + + -- INDEXED_COMPONENT ::= PREFIX (EXPRESSION {, EXPRESSION}) + + -- SLICE ::= PREFIX (DISCRETE_RANGE) + + -- SELECTED_COMPONENT ::= PREFIX . SELECTOR_NAME + + -- SELECTOR_NAME ::= IDENTIFIER | CHARACTER_LITERAL | OPERATOR_SYMBOL + + -- ATTRIBUTE_REFERENCE ::= PREFIX ' ATTRIBUTE_DESIGNATOR + + -- ATTRIBUTE_DESIGNATOR ::= + -- IDENTIFIER [(static_EXPRESSION)] + -- | access | delta | digits + + -- FUNCTION_CALL ::= + -- function_NAME + -- | function_PREFIX ACTUAL_PARAMETER_PART + + -- ACTUAL_PARAMETER_PART ::= + -- (PARAMETER_ASSOCIATION {,PARAMETER_ASSOCIATION}) + + -- PARAMETER_ASSOCIATION ::= + -- [formal_parameter_SELECTOR_NAME =>] EXPLICIT_ACTUAL_PARAMETER + + -- EXPLICIT_ACTUAL_PARAMETER ::= EXPRESSION | variable_NAME + + -- Note: syntactically a procedure call looks just like a function call, + -- so this routine is in practice used to scan out procedure calls as well. + + -- On return, Expr_Form is set to either EF_Name or EF_Simple_Name + + -- Error recovery: can raise Error_Resync + + -- Note: if on return Token = Tok_Apostrophe, then the apostrophe must be + -- followed by either a left paren (qualified expression case), or by + -- range (range attribute case). All other uses of apostrophe (i.e. all + -- other attributes) are handled in this routine. + + -- Error recovery: can raise Error_Resync + + function P_Name return Node_Id is + Scan_State : Saved_Scan_State; + Name_Node : Node_Id; + Prefix_Node : Node_Id; + Ident_Node : Node_Id; + Expr_Node : Node_Id; + Range_Node : Node_Id; + Arg_Node : Node_Id; + + Arg_List : List_Id := No_List; -- kill junk warning + Attr_Name : Name_Id := No_Name; -- kill junk warning + + begin + if Token not in Token_Class_Name then + Error_Msg_AP ("name expected"); + raise Error_Resync; + end if; + + -- Loop through designators in qualified name + + Name_Node := Token_Node; + + loop + Scan; -- past designator + exit when Token /= Tok_Dot; + Save_Scan_State (Scan_State); -- at dot + Scan; -- past dot + + -- If we do not have another designator after the dot, then join + -- the normal circuit to handle a dot extension (may be .all or + -- character literal case). Otherwise loop back to scan the next + -- designator. + + if Token not in Token_Class_Desig then + goto Scan_Name_Extension_Dot; + else + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Selected_Component, Prev_Token_Ptr); + Set_Prefix (Name_Node, Prefix_Node); + Set_Selector_Name (Name_Node, Token_Node); + end if; + end loop; + + -- We have now scanned out a qualified designator. If the last token is + -- an operator symbol, then we certainly do not have the Snam case, so + -- we can just use the normal name extension check circuit + + if Prev_Token = Tok_Operator_Symbol then + goto Scan_Name_Extension; + end if; + + -- We have scanned out a qualified simple name, check for name extension + -- Note that we know there is no dot here at this stage, so the only + -- possible cases of name extension are apostrophe and left paren. + + if Token = Tok_Apostrophe then + Save_Scan_State (Scan_State); -- at apostrophe + Scan; -- past apostrophe + + -- If left paren, then this might be a qualified expression, but we + -- are only in the business of scanning out names, so return with + -- Token backed up to point to the apostrophe. The treatment for + -- the range attribute is similar (we do not consider x'range to + -- be a name in this grammar). + + if Token = Tok_Left_Paren or else Token = Tok_Range then + Restore_Scan_State (Scan_State); -- to apostrophe + Expr_Form := EF_Simple_Name; + return Name_Node; + + -- Otherwise we have the case of a name extended by an attribute + + else + goto Scan_Name_Extension_Apostrophe; + end if; + + -- Check case of qualified simple name extended by a left parenthesis + + elsif Token = Tok_Left_Paren then + Scan; -- past left paren + goto Scan_Name_Extension_Left_Paren; + + -- Otherwise the qualified simple name is not extended, so return + + else + Expr_Form := EF_Simple_Name; + return Name_Node; + end if; + + -- Loop scanning past name extensions. A label is used for control + -- transfer for this loop for ease of interfacing with the finite state + -- machine in the parenthesis scanning circuit, and also to allow for + -- passing in control to the appropriate point from the above code. + + <<Scan_Name_Extension>> + + -- Character literal used as name cannot be extended. Also this + -- cannot be a call, since the name for a call must be a designator. + -- Return in these cases, or if there is no name extension + + if Token not in Token_Class_Namext + or else Prev_Token = Tok_Char_Literal + then + Expr_Form := EF_Name; + return Name_Node; + end if; + + -- Merge here when we know there is a name extension + + <<Scan_Name_Extension_OK>> + + if Token = Tok_Left_Paren then + Scan; -- past left paren + goto Scan_Name_Extension_Left_Paren; + + elsif Token = Tok_Apostrophe then + Save_Scan_State (Scan_State); -- at apostrophe + Scan; -- past apostrophe + goto Scan_Name_Extension_Apostrophe; + + else -- Token = Tok_Dot + Save_Scan_State (Scan_State); -- at dot + Scan; -- past dot + goto Scan_Name_Extension_Dot; + end if; + + -- Case of name extended by dot (selection), dot is already skipped + -- and the scan state at the point of the dot is saved in Scan_State. + + <<Scan_Name_Extension_Dot>> + + -- Explicit dereference case + + if Token = Tok_All then + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Explicit_Dereference, Token_Ptr); + Set_Prefix (Name_Node, Prefix_Node); + Scan; -- past ALL + goto Scan_Name_Extension; + + -- Selected component case + + elsif Token in Token_Class_Name then + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Selected_Component, Prev_Token_Ptr); + Set_Prefix (Name_Node, Prefix_Node); + Set_Selector_Name (Name_Node, Token_Node); + Scan; -- past selector + goto Scan_Name_Extension; + + -- Reserved identifier as selector + + elsif Is_Reserved_Identifier then + Scan_Reserved_Identifier (Force_Msg => False); + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Selected_Component, Prev_Token_Ptr); + Set_Prefix (Name_Node, Prefix_Node); + Set_Selector_Name (Name_Node, Token_Node); + Scan; -- past identifier used as selector + goto Scan_Name_Extension; + + -- If dot is at end of line and followed by nothing legal, + -- then assume end of name and quit (dot will be taken as + -- an erroneous form of some other punctuation by our caller). + + elsif Token_Is_At_Start_Of_Line then + Restore_Scan_State (Scan_State); + return Name_Node; + + -- Here if nothing legal after the dot + + else + Error_Msg_AP ("selector expected"); + raise Error_Resync; + end if; + + -- Here for an apostrophe as name extension. The scan position at the + -- apostrophe has already been saved, and the apostrophe scanned out. + + <<Scan_Name_Extension_Apostrophe>> + + Scan_Apostrophe : declare + function Apostrophe_Should_Be_Semicolon return Boolean; + -- Checks for case where apostrophe should probably be + -- a semicolon, and if so, gives appropriate message, + -- resets the scan pointer to the apostrophe, changes + -- the current token to Tok_Semicolon, and returns True. + -- Otherwise returns False. + + function Apostrophe_Should_Be_Semicolon return Boolean is + begin + if Token_Is_At_Start_Of_Line then + Restore_Scan_State (Scan_State); -- to apostrophe + Error_Msg_SC ("""''"" should be "";"""); + Token := Tok_Semicolon; + return True; + else + return False; + end if; + end Apostrophe_Should_Be_Semicolon; + + -- Start of processing for Scan_Apostrophe + + begin + -- If range attribute after apostrophe, then return with Token + -- pointing to the apostrophe. Note that in this case the prefix + -- need not be a simple name (cases like A.all'range). Similarly + -- if there is a left paren after the apostrophe, then we also + -- return with Token pointing to the apostrophe (this is the + -- qualified expression case). + + if Token = Tok_Range or else Token = Tok_Left_Paren then + Restore_Scan_State (Scan_State); -- to apostrophe + Expr_Form := EF_Name; + return Name_Node; + + -- Here for cases where attribute designator is an identifier + + elsif Token = Tok_Identifier then + Attr_Name := Token_Name; + + if not Is_Attribute_Name (Attr_Name) then + if Apostrophe_Should_Be_Semicolon then + Expr_Form := EF_Name; + return Name_Node; + else + Signal_Bad_Attribute; + end if; + end if; + + if Style_Check then + Style.Check_Attribute_Name (False); + end if; + + Delete_Node (Token_Node); + + -- Here for case of attribute designator is not an identifier + + else + if Token = Tok_Delta then + Attr_Name := Name_Delta; + + elsif Token = Tok_Digits then + Attr_Name := Name_Digits; + + elsif Token = Tok_Access then + Attr_Name := Name_Access; + + elsif Apostrophe_Should_Be_Semicolon then + Expr_Form := EF_Name; + return Name_Node; + + else + Error_Msg_AP ("attribute designator expected"); + raise Error_Resync; + end if; + + if Style_Check then + Style.Check_Attribute_Name (True); + end if; + end if; + + -- We come here with an OK attribute scanned, and the + -- corresponding Attribute identifier node stored in Ident_Node. + + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Attribute_Reference, Prev_Token_Ptr); + Scan; -- past attribute designator + Set_Prefix (Name_Node, Prefix_Node); + Set_Attribute_Name (Name_Node, Attr_Name); + + -- Scan attribute arguments/designator + + if Token = Tok_Left_Paren then + Set_Expressions (Name_Node, New_List); + Scan; -- past left paren + + loop + declare + Expr : constant Node_Id := P_Expression; + + begin + if Token = Tok_Arrow then + Error_Msg_SC + ("named parameters not permitted for attributes"); + Scan; -- past junk arrow + + else + Append (Expr, Expressions (Name_Node)); + exit when not Comma_Present; + end if; + end; + end loop; + + T_Right_Paren; + end if; + + goto Scan_Name_Extension; + end Scan_Apostrophe; + + -- Here for left parenthesis extending name (left paren skipped) + + <<Scan_Name_Extension_Left_Paren>> + + -- We now have to scan through a list of items, terminated by a + -- right parenthesis. The scan is handled by a finite state + -- machine. The possibilities are: + + -- (discrete_range) + + -- This is a slice. This case is handled in LP_State_Init. + + -- (expression, expression, ..) + + -- This is interpreted as an indexed component, i.e. as a + -- case of a name which can be extended in the normal manner. + -- This case is handled by LP_State_Name or LP_State_Expr. + + -- (..., identifier => expression , ...) + + -- If there is at least one occurence of identifier => (but + -- none of the other cases apply), then we have a call. + + -- Test for Id => case + + if Token = Tok_Identifier then + Save_Scan_State (Scan_State); -- at Id + Scan; -- past Id + + -- Test for => (allow := as an error substitute) + + if Token = Tok_Arrow or else Token = Tok_Colon_Equal then + Restore_Scan_State (Scan_State); -- to Id + Arg_List := New_List; + goto LP_State_Call; + + else + Restore_Scan_State (Scan_State); -- to Id + end if; + end if; + + -- Here we have an expression after all + + Expr_Node := P_Expression_Or_Range_Attribute; + + -- Check cases of discrete range for a slice + + -- First possibility: Range_Attribute_Reference + + if Expr_Form = EF_Range_Attr then + Range_Node := Expr_Node; + + -- Second possibility: Simple_expression .. Simple_expression + + elsif Token = Tok_Dot_Dot then + Check_Simple_Expression (Expr_Node); + Range_Node := New_Node (N_Range, Token_Ptr); + Set_Low_Bound (Range_Node, Expr_Node); + Scan; -- past .. + Expr_Node := P_Expression; + Check_Simple_Expression (Expr_Node); + Set_High_Bound (Range_Node, Expr_Node); + + -- Third possibility: Type_name range Range + + elsif Token = Tok_Range then + if Expr_Form /= EF_Simple_Name then + Error_Msg_SC ("subtype mark must precede RANGE"); + raise Error_Resync; + end if; + + Range_Node := P_Subtype_Indication (Expr_Node); + + -- Otherwise we just have an expression. It is true that we might + -- have a subtype mark without a range constraint but this case + -- is syntactically indistinguishable from the expression case. + + else + Arg_List := New_List; + goto LP_State_Expr; + end if; + + -- Fall through here with unmistakable Discrete range scanned, + -- which means that we definitely have the case of a slice. The + -- Discrete range is in Range_Node. + + if Token = Tok_Comma then + Error_Msg_SC ("slice cannot have more than one dimension"); + raise Error_Resync; + + elsif Token /= Tok_Right_Paren then + T_Right_Paren; + raise Error_Resync; + + else + Scan; -- past right paren + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Slice, Sloc (Prefix_Node)); + Set_Prefix (Name_Node, Prefix_Node); + Set_Discrete_Range (Name_Node, Range_Node); + + -- An operator node is legal as a prefix to other names, + -- but not for a slice. + + if Nkind (Prefix_Node) = N_Operator_Symbol then + Error_Msg_N ("illegal prefix for slice", Prefix_Node); + end if; + + -- If we have a name extension, go scan it + + if Token in Token_Class_Namext then + goto Scan_Name_Extension_OK; + + -- Otherwise return (a slice is a name, but is not a call) + + else + Expr_Form := EF_Name; + return Name_Node; + end if; + end if; + + -- In LP_State_Expr, we have scanned one or more expressions, and + -- so we have a call or an indexed component which is a name. On + -- entry we have the expression just scanned in Expr_Node and + -- Arg_List contains the list of expressions encountered so far + + <<LP_State_Expr>> + Append (Expr_Node, Arg_List); + + if Token = Tok_Arrow then + Error_Msg + ("expect identifier in parameter association", + Sloc (Expr_Node)); + Scan; -- past arrow. + + elsif not Comma_Present then + T_Right_Paren; + Prefix_Node := Name_Node; + Name_Node := New_Node (N_Indexed_Component, Sloc (Prefix_Node)); + Set_Prefix (Name_Node, Prefix_Node); + Set_Expressions (Name_Node, Arg_List); + goto Scan_Name_Extension; + end if; + + -- Comma present (and scanned out), test for identifier => case + -- Test for identifer => case + + if Token = Tok_Identifier then + Save_Scan_State (Scan_State); -- at Id + Scan; -- past Id + + -- Test for => (allow := as error substitute) + + if Token = Tok_Arrow or else Token = Tok_Colon_Equal then + Restore_Scan_State (Scan_State); -- to Id + goto LP_State_Call; + + -- Otherwise it's just an expression after all, so backup + + else + Restore_Scan_State (Scan_State); -- to Id + end if; + end if; + + -- Here we have an expression after all, so stay in this state + + Expr_Node := P_Expression; + goto LP_State_Expr; + + -- LP_State_Call corresponds to the situation in which at least + -- one instance of Id => Expression has been encountered, so we + -- know that we do not have a name, but rather a call. We enter + -- it with the scan pointer pointing to the next argument to scan, + -- and Arg_List containing the list of arguments scanned so far. + + <<LP_State_Call>> + + -- Test for case of Id => Expression (named parameter) + + if Token = Tok_Identifier then + Save_Scan_State (Scan_State); -- at Id + Ident_Node := Token_Node; + Scan; -- past Id + + -- Deal with => (allow := as erroneous substitute) + + if Token = Tok_Arrow or else Token = Tok_Colon_Equal then + Arg_Node := + New_Node (N_Parameter_Association, Prev_Token_Ptr); + Set_Selector_Name (Arg_Node, Ident_Node); + T_Arrow; + Set_Explicit_Actual_Parameter (Arg_Node, P_Expression); + Append (Arg_Node, Arg_List); + + -- If a comma follows, go back and scan next entry + + if Comma_Present then + goto LP_State_Call; + + -- Otherwise we have the end of a call + + else + Prefix_Node := Name_Node; + Name_Node := + New_Node (N_Function_Call, Sloc (Prefix_Node)); + Set_Name (Name_Node, Prefix_Node); + Set_Parameter_Associations (Name_Node, Arg_List); + T_Right_Paren; + + if Token in Token_Class_Namext then + goto Scan_Name_Extension_OK; + + -- This is a case of a call which cannot be a name + + else + Expr_Form := EF_Name; + return Name_Node; + end if; + end if; + + -- Not named parameter: Id started an expression after all + + else + Restore_Scan_State (Scan_State); -- to Id + end if; + end if; + + -- Here if entry did not start with Id => which means that it + -- is a positional parameter, which is not allowed, since we + -- have seen at least one named parameter already. + + Error_Msg_SC + ("positional parameter association " & + "not allowed after named one"); + + Expr_Node := P_Expression; + + -- Leaving the '>' in an association is not unusual, so suggest + -- a possible fix. + + if Nkind (Expr_Node) = N_Op_Eq then + Error_Msg_N ("\maybe `=>` was intended", Expr_Node); + end if; + + -- We go back to scanning out expressions, so that we do not get + -- multiple error messages when several positional parameters + -- follow a named parameter. + + goto LP_State_Expr; + + -- End of treatment for name extensions starting with left paren + + -- End of loop through name extensions + + end P_Name; + + -- This function parses a restricted form of Names which are either + -- designators, or designators preceded by a sequence of prefixes + -- that are direct names. + + -- Error recovery: cannot raise Error_Resync + + function P_Function_Name return Node_Id is + Designator_Node : Node_Id; + Prefix_Node : Node_Id; + Selector_Node : Node_Id; + Dot_Sloc : Source_Ptr := No_Location; + + begin + -- Prefix_Node is set to the gathered prefix so far, Empty means that + -- no prefix has been scanned. This allows us to build up the result + -- in the required right recursive manner. + + Prefix_Node := Empty; + + -- Loop through prefixes + + loop + Designator_Node := Token_Node; + + if Token not in Token_Class_Desig then + return P_Identifier; -- let P_Identifier issue the error message + + else -- Token in Token_Class_Desig + Scan; -- past designator + exit when Token /= Tok_Dot; + end if; + + -- Here at a dot, with token just before it in Designator_Node + + if No (Prefix_Node) then + Prefix_Node := Designator_Node; + else + Selector_Node := New_Node (N_Selected_Component, Dot_Sloc); + Set_Prefix (Selector_Node, Prefix_Node); + Set_Selector_Name (Selector_Node, Designator_Node); + Prefix_Node := Selector_Node; + end if; + + Dot_Sloc := Token_Ptr; + Scan; -- past dot + end loop; + + -- Fall out of the loop having just scanned a designator + + if No (Prefix_Node) then + return Designator_Node; + else + Selector_Node := New_Node (N_Selected_Component, Dot_Sloc); + Set_Prefix (Selector_Node, Prefix_Node); + Set_Selector_Name (Selector_Node, Designator_Node); + return Selector_Node; + end if; + + exception + when Error_Resync => + return Error; + + end P_Function_Name; + + -- This function parses a restricted form of Names which are either + -- identifiers, or identifiers preceded by a sequence of prefixes + -- that are direct names. + + -- Error recovery: cannot raise Error_Resync + + function P_Qualified_Simple_Name return Node_Id is + Designator_Node : Node_Id; + Prefix_Node : Node_Id; + Selector_Node : Node_Id; + Dot_Sloc : Source_Ptr := No_Location; + + begin + -- Prefix node is set to the gathered prefix so far, Empty means that + -- no prefix has been scanned. This allows us to build up the result + -- in the required right recursive manner. + + Prefix_Node := Empty; + + -- Loop through prefixes + + loop + Designator_Node := Token_Node; + + if Token = Tok_Identifier then + Scan; -- past identifier + exit when Token /= Tok_Dot; + + elsif Token not in Token_Class_Desig then + return P_Identifier; -- let P_Identifier issue the error message + + else + Scan; -- past designator + + if Token /= Tok_Dot then + Error_Msg_SP ("identifier expected"); + return Error; + end if; + end if; + + -- Here at a dot, with token just before it in Designator_Node + + if No (Prefix_Node) then + Prefix_Node := Designator_Node; + else + Selector_Node := New_Node (N_Selected_Component, Dot_Sloc); + Set_Prefix (Selector_Node, Prefix_Node); + Set_Selector_Name (Selector_Node, Designator_Node); + Prefix_Node := Selector_Node; + end if; + + Dot_Sloc := Token_Ptr; + Scan; -- past dot + end loop; + + -- Fall out of the loop having just scanned an identifier + + if No (Prefix_Node) then + return Designator_Node; + else + Selector_Node := New_Node (N_Selected_Component, Dot_Sloc); + Set_Prefix (Selector_Node, Prefix_Node); + Set_Selector_Name (Selector_Node, Designator_Node); + return Selector_Node; + end if; + + exception + when Error_Resync => + return Error; + + end P_Qualified_Simple_Name; + + -- This procedure differs from P_Qualified_Simple_Name only in that it + -- raises Error_Resync if any error is encountered. It only returns after + -- scanning a valid qualified simple name. + + -- Error recovery: can raise Error_Resync + + function P_Qualified_Simple_Name_Resync return Node_Id is + Designator_Node : Node_Id; + Prefix_Node : Node_Id; + Selector_Node : Node_Id; + Dot_Sloc : Source_Ptr := No_Location; + + begin + Prefix_Node := Empty; + + -- Loop through prefixes + + loop + Designator_Node := Token_Node; + + if Token = Tok_Identifier then + Scan; -- past identifier + exit when Token /= Tok_Dot; + + elsif Token not in Token_Class_Desig then + Discard_Junk_Node (P_Identifier); -- to issue the error message + raise Error_Resync; + + else + Scan; -- past designator + + if Token /= Tok_Dot then + Error_Msg_SP ("identifier expected"); + raise Error_Resync; + end if; + end if; + + -- Here at a dot, with token just before it in Designator_Node + + if No (Prefix_Node) then + Prefix_Node := Designator_Node; + else + Selector_Node := New_Node (N_Selected_Component, Dot_Sloc); + Set_Prefix (Selector_Node, Prefix_Node); + Set_Selector_Name (Selector_Node, Designator_Node); + Prefix_Node := Selector_Node; + end if; + + Dot_Sloc := Token_Ptr; + Scan; -- past period + end loop; + + -- Fall out of the loop having just scanned an identifier + + if No (Prefix_Node) then + return Designator_Node; + else + Selector_Node := New_Node (N_Selected_Component, Dot_Sloc); + Set_Prefix (Selector_Node, Prefix_Node); + Set_Selector_Name (Selector_Node, Designator_Node); + return Selector_Node; + end if; + + end P_Qualified_Simple_Name_Resync; + + ---------------------- + -- 4.1 Direct_Name -- + ---------------------- + + -- Parsed by P_Name and other functions in section 4.1 + + ----------------- + -- 4.1 Prefix -- + ----------------- + + -- Parsed by P_Name (4.1) + + ------------------------------- + -- 4.1 Explicit Dereference -- + ------------------------------- + + -- Parsed by P_Name (4.1) + + ------------------------------- + -- 4.1 Implicit_Dereference -- + ------------------------------- + + -- Parsed by P_Name (4.1) + + ---------------------------- + -- 4.1 Indexed Component -- + ---------------------------- + + -- Parsed by P_Name (4.1) + + ---------------- + -- 4.1 Slice -- + ---------------- + + -- Parsed by P_Name (4.1) + + ----------------------------- + -- 4.1 Selected_Component -- + ----------------------------- + + -- Parsed by P_Name (4.1) + + ------------------------ + -- 4.1 Selector Name -- + ------------------------ + + -- Parsed by P_Name (4.1) + + ------------------------------ + -- 4.1 Attribute Reference -- + ------------------------------ + + -- Parsed by P_Name (4.1) + + ------------------------------- + -- 4.1 Attribute Designator -- + ------------------------------- + + -- Parsed by P_Name (4.1) + + -------------------------------------- + -- 4.1.4 Range Attribute Reference -- + -------------------------------------- + + -- RANGE_ATTRIBUTE_REFERENCE ::= PREFIX ' RANGE_ATTRIBUTE_DESIGNATOR + + -- RANGE_ATTRIBUTE_DESIGNATOR ::= range [(static_EXPRESSION)] + + -- In the grammar, a RANGE attribute is simply a name, but its use is + -- highly restricted, so in the parser, we do not regard it as a name. + -- Instead, P_Name returns without scanning the 'RANGE part of the + -- attribute, and the caller uses the following function to construct + -- a range attribute in places where it is appropriate. + + -- Note that RANGE here is treated essentially as an identifier, + -- rather than a reserved word. + + -- The caller has parsed the prefix, i.e. a name, and Token points to + -- the apostrophe. The token after the apostrophe is known to be RANGE + -- at this point. The prefix node becomes the prefix of the attribute. + + -- Error_Recovery: Cannot raise Error_Resync + + function P_Range_Attribute_Reference + (Prefix_Node : Node_Id) + return Node_Id + is + Attr_Node : Node_Id; + + begin + Attr_Node := New_Node (N_Attribute_Reference, Token_Ptr); + Set_Prefix (Attr_Node, Prefix_Node); + Scan; -- past apostrophe + + if Style_Check then + Style.Check_Attribute_Name (True); + end if; + + Set_Attribute_Name (Attr_Node, Name_Range); + Scan; -- past RANGE + + if Token = Tok_Left_Paren then + Scan; -- past left paren + Set_Expressions (Attr_Node, New_List (P_Expression)); + T_Right_Paren; + end if; + + return Attr_Node; + end P_Range_Attribute_Reference; + + --------------------------------------- + -- 4.1.4 Range Attribute Designator -- + --------------------------------------- + + -- Parsed by P_Range_Attribute_Reference (4.4) + + -------------------- + -- 4.3 Aggregate -- + -------------------- + + -- AGGREGATE ::= RECORD_AGGREGATE | EXTENSION_AGGREGATE | ARRAY_AGGREGATE + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3), except in the case where + -- an aggregate is known to be required (code statement, extension + -- aggregate), in which cases this routine performs the necessary check + -- that we have an aggregate rather than a parenthesized expression + + -- Error recovery: can raise Error_Resync + + function P_Aggregate return Node_Id is + Aggr_Sloc : constant Source_Ptr := Token_Ptr; + Aggr_Node : constant Node_Id := P_Aggregate_Or_Paren_Expr; + + begin + if Nkind (Aggr_Node) /= N_Aggregate + and then + Nkind (Aggr_Node) /= N_Extension_Aggregate + then + Error_Msg + ("aggregate may not have single positional component", Aggr_Sloc); + return Error; + else + return Aggr_Node; + end if; + end P_Aggregate; + + ------------------------------------------------- + -- 4.3 Aggregate or Parenthesized Expresssion -- + ------------------------------------------------- + + -- This procedure parses out either an aggregate or a parenthesized + -- expression (these two constructs are closely related, since a + -- parenthesized expression looks like an aggregate with a single + -- positional component). + + -- AGGREGATE ::= + -- RECORD_AGGREGATE | EXTENSION_AGGREGATE | ARRAY_AGGREGATE + + -- RECORD_AGGREGATE ::= (RECORD_COMPONENT_ASSOCIATION_LIST) + + -- RECORD_COMPONENT_ASSOCIATION_LIST ::= + -- RECORD_COMPONENT_ASSOCIATION {, RECORD_COMPONENT_ASSOCIATION} + -- | null record + + -- RECORD_COMPONENT_ASSOCIATION ::= + -- [COMPONENT_CHOICE_LIST =>] EXPRESSION + + -- COMPONENT_CHOICE_LIST ::= + -- component_SELECTOR_NAME {| component_SELECTOR_NAME} + -- | others + + -- EXTENSION_AGGREGATE ::= + -- (ANCESTOR_PART with RECORD_COMPONENT_ASSOCIATION_LIST) + + -- ANCESTOR_PART ::= EXPRESSION | SUBTYPE_MARK + + -- ARRAY_AGGREGATE ::= + -- POSITIONAL_ARRAY_AGGREGATE | NAMED_ARRAY_AGGREGATE + + -- POSITIONAL_ARRAY_AGGREGATE ::= + -- (EXPRESSION, EXPRESSION {, EXPRESSION}) + -- | (EXPRESSION {, EXPRESSION}, others => EXPRESSION) + + -- NAMED_ARRAY_AGGREGATE ::= + -- (ARRAY_COMPONENT_ASSOCIATION {, ARRAY_COMPONENT_ASSOCIATION}) + + -- PRIMARY ::= (EXPRESSION); + + -- Error recovery: can raise Error_Resync + + function P_Aggregate_Or_Paren_Expr return Node_Id is + Aggregate_Node : Node_Id; + Expr_List : List_Id; + Assoc_List : List_Id; + Expr_Node : Node_Id; + Lparen_Sloc : Source_Ptr; + Scan_State : Saved_Scan_State; + + begin + Lparen_Sloc := Token_Ptr; + T_Left_Paren; + + -- Note: the mechanism used here of rescanning the initial expression + -- is distinctly unpleasant, but it saves a lot of fiddling in scanning + -- out the discrete choice list. + + -- Deal with expression and extension aggregate cases first + + if Token /= Tok_Others then + Save_Scan_State (Scan_State); -- at start of expression + + -- Deal with (NULL RECORD) case + + if Token = Tok_Null then + Scan; -- past NULL + + if Token = Tok_Record then + Aggregate_Node := New_Node (N_Aggregate, Lparen_Sloc); + Set_Null_Record_Present (Aggregate_Node, True); + Scan; -- past RECORD + T_Right_Paren; + return Aggregate_Node; + else + Restore_Scan_State (Scan_State); -- to NULL that must be expr + end if; + end if; + + Expr_Node := P_Expression_Or_Range_Attribute; + + -- Extension aggregate case + + if Token = Tok_With then + + if Nkind (Expr_Node) = N_Attribute_Reference + and then Attribute_Name (Expr_Node) = Name_Range + then + Bad_Range_Attribute (Sloc (Expr_Node)); + return Error; + end if; + + if Ada_83 then + Error_Msg_SC ("(Ada 83) extension aggregate not allowed"); + end if; + + Aggregate_Node := New_Node (N_Extension_Aggregate, Lparen_Sloc); + Set_Ancestor_Part (Aggregate_Node, Expr_Node); + Scan; -- past WITH + + -- Deal with WITH NULL RECORD case + + if Token = Tok_Null then + Save_Scan_State (Scan_State); -- at NULL + Scan; -- past NULL + + if Token = Tok_Record then + Scan; -- past RECORD + Set_Null_Record_Present (Aggregate_Node, True); + T_Right_Paren; + return Aggregate_Node; + + else + Restore_Scan_State (Scan_State); -- to NULL that must be expr + end if; + end if; + + if Token /= Tok_Others then + Save_Scan_State (Scan_State); + Expr_Node := P_Expression; + else + Expr_Node := Empty; + end if; + + -- Expression case + + elsif Token = Tok_Right_Paren or else Token in Token_Class_Eterm then + + if Nkind (Expr_Node) = N_Attribute_Reference + and then Attribute_Name (Expr_Node) = Name_Range + then + Bad_Range_Attribute (Sloc (Expr_Node)); + return Error; + end if; + + -- Bump paren count of expression, note that if the paren count + -- is already at the maximum, then we leave it alone. This will + -- cause some failures in pathalogical conformance tests, which + -- we do not shed a tear over! + + if Expr_Node /= Error then + if Paren_Count (Expr_Node) /= Paren_Count_Type'Last then + Set_Paren_Count (Expr_Node, Paren_Count (Expr_Node) + 1); + end if; + end if; + + T_Right_Paren; -- past right paren (error message if none) + return Expr_Node; + + -- Normal aggregate case + + else + Aggregate_Node := New_Node (N_Aggregate, Lparen_Sloc); + end if; + + -- Others case + + else + Aggregate_Node := New_Node (N_Aggregate, Lparen_Sloc); + Expr_Node := Empty; + end if; + + -- Prepare to scan list of component associations + + Expr_List := No_List; -- don't set yet, maybe all named entries + Assoc_List := No_List; -- don't set yet, maybe all positional entries + + -- This loop scans through component associations. On entry to the + -- loop, an expression has been scanned at the start of the current + -- association unless initial token was OTHERS, in which case + -- Expr_Node is set to Empty. + + loop + -- Deal with others association first. This is a named association + + if No (Expr_Node) then + if No (Assoc_List) then + Assoc_List := New_List; + end if; + + Append (P_Record_Or_Array_Component_Association, Assoc_List); + + -- Improper use of WITH + + elsif Token = Tok_With then + Error_Msg_SC ("WITH must be preceded by single expression in " & + "extension aggregate"); + raise Error_Resync; + + -- Assume positional case if comma, right paren, or literal or + -- identifier or OTHERS follows (the latter cases are missing + -- comma cases). Also assume positional if a semicolon follows, + -- which can happen if there are missing parens + + elsif Token = Tok_Comma + or else Token = Tok_Right_Paren + or else Token = Tok_Others + or else Token in Token_Class_Lit_Or_Name + or else Token = Tok_Semicolon + then + if Present (Assoc_List) then + Error_Msg_BC + ("""=>"" expected (positional association cannot follow " & + "named association)"); + end if; + + if No (Expr_List) then + Expr_List := New_List; + end if; + + Append (Expr_Node, Expr_List); + + -- Anything else is assumed to be a named association + + else + Restore_Scan_State (Scan_State); -- to start of expression + + if No (Assoc_List) then + Assoc_List := New_List; + end if; + + Append (P_Record_Or_Array_Component_Association, Assoc_List); + end if; + + exit when not Comma_Present; + + -- If we are at an expression terminator, something is seriously + -- wrong, so let's get out now, before we start eating up stuff + -- that doesn't belong to us! + + if Token in Token_Class_Eterm then + Error_Msg_AP ("expecting expression or component association"); + exit; + end if; + + -- Otherwise initiate for reentry to top of loop by scanning an + -- initial expression, unless the first token is OTHERS. + + if Token = Tok_Others then + Expr_Node := Empty; + else + Save_Scan_State (Scan_State); -- at start of expression + Expr_Node := P_Expression; + end if; + end loop; + + -- All component associations (positional and named) have been scanned + + T_Right_Paren; + Set_Expressions (Aggregate_Node, Expr_List); + Set_Component_Associations (Aggregate_Node, Assoc_List); + return Aggregate_Node; + end P_Aggregate_Or_Paren_Expr; + + ------------------------------------------------ + -- 4.3 Record or Array Component Association -- + ------------------------------------------------ + + -- RECORD_COMPONENT_ASSOCIATION ::= + -- [COMPONENT_CHOICE_LIST =>] EXPRESSION + + -- COMPONENT_CHOICE_LIST => + -- component_SELECTOR_NAME {| component_SELECTOR_NAME} + -- | others + + -- ARRAY_COMPONENT_ASSOCIATION ::= + -- DISCRETE_CHOICE_LIST => EXPRESSION + + -- Note: this routine only handles the named cases, including others. + -- Cases where the component choice list is not present have already + -- been handled directly. + + -- Error recovery: can raise Error_Resync + + function P_Record_Or_Array_Component_Association return Node_Id is + Assoc_Node : Node_Id; + + begin + Assoc_Node := New_Node (N_Component_Association, Token_Ptr); + Set_Choices (Assoc_Node, P_Discrete_Choice_List); + Set_Sloc (Assoc_Node, Token_Ptr); + TF_Arrow; + Set_Expression (Assoc_Node, P_Expression); + return Assoc_Node; + end P_Record_Or_Array_Component_Association; + + ----------------------------- + -- 4.3.1 Record Aggregate -- + ----------------------------- + + -- Case of enumeration aggregate is parsed by P_Aggregate (4.3) + -- All other cases are parsed by P_Aggregate_Or_Paren_Expr (4.3) + + ---------------------------------------------- + -- 4.3.1 Record Component Association List -- + ---------------------------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + ---------------------------------- + -- 4.3.1 Component Choice List -- + ---------------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + -------------------------------- + -- 4.3.1 Extension Aggregate -- + -------------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + -------------------------- + -- 4.3.1 Ancestor Part -- + -------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + ---------------------------- + -- 4.3.1 Array Aggregate -- + ---------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + --------------------------------------- + -- 4.3.1 Positional Array Aggregate -- + --------------------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + ---------------------------------- + -- 4.3.1 Named Array Aggregate -- + ---------------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + ---------------------------------------- + -- 4.3.1 Array Component Association -- + ---------------------------------------- + + -- Parsed by P_Aggregate_Or_Paren_Expr (4.3) + + --------------------- + -- 4.4 Expression -- + --------------------- + + -- EXPRESSION ::= + -- RELATION {and RELATION} | RELATION {and then RELATION} + -- | RELATION {or RELATION} | RELATION {or else RELATION} + -- | RELATION {xor RELATION} + + -- On return, Expr_Form indicates the categorization of the expression + -- EF_Range_Attr is not a possible value (if a range attribute is found, + -- an error message is given, and Error is returned). + + -- Error recovery: cannot raise Error_Resync + + function P_Expression return Node_Id is + Logical_Op : Node_Kind; + Prev_Logical_Op : Node_Kind; + Op_Location : Source_Ptr; + Node1 : Node_Id; + Node2 : Node_Id; + + begin + Node1 := P_Relation; + + if Token in Token_Class_Logop then + Prev_Logical_Op := N_Empty; + + loop + Op_Location := Token_Ptr; + Logical_Op := P_Logical_Operator; + + if Prev_Logical_Op /= N_Empty and then + Logical_Op /= Prev_Logical_Op + then + Error_Msg + ("mixed logical operators in expression", Op_Location); + Prev_Logical_Op := N_Empty; + else + Prev_Logical_Op := Logical_Op; + end if; + + Node2 := Node1; + Node1 := New_Node (Logical_Op, Op_Location); + Set_Left_Opnd (Node1, Node2); + Set_Right_Opnd (Node1, P_Relation); + Set_Op_Name (Node1); + exit when Token not in Token_Class_Logop; + end loop; + + Expr_Form := EF_Non_Simple; + end if; + + if Token = Tok_Apostrophe then + Bad_Range_Attribute (Token_Ptr); + return Error; + else + return Node1; + end if; + + end P_Expression; + + -- This function is identical to the normal P_Expression, except that it + -- checks that the expression scan did not stop on a right paren. It is + -- called in all contexts where a right parenthesis cannot legitimately + -- follow an expression. + + function P_Expression_No_Right_Paren return Node_Id is + begin + return No_Right_Paren (P_Expression); + end P_Expression_No_Right_Paren; + + ---------------------------------------- + -- 4.4 Expression_Or_Range_Attribute -- + ---------------------------------------- + + -- EXPRESSION ::= + -- RELATION {and RELATION} | RELATION {and then RELATION} + -- | RELATION {or RELATION} | RELATION {or else RELATION} + -- | RELATION {xor RELATION} + + -- RANGE_ATTRIBUTE_REFERENCE ::= PREFIX ' RANGE_ATTRIBUTE_DESIGNATOR + + -- RANGE_ATTRIBUTE_DESIGNATOR ::= range [(static_EXPRESSION)] + + -- On return, Expr_Form indicates the categorization of the expression + -- and EF_Range_Attr is one of the possibilities. + + -- Error recovery: cannot raise Error_Resync + + -- In the grammar, a RANGE attribute is simply a name, but its use is + -- highly restricted, so in the parser, we do not regard it as a name. + -- Instead, P_Name returns without scanning the 'RANGE part of the + -- attribute, and P_Expression_Or_Range_Attribute handles the range + -- attribute reference. In the normal case where a range attribute is + -- not allowed, an error message is issued by P_Expression. + + function P_Expression_Or_Range_Attribute return Node_Id is + Logical_Op : Node_Kind; + Prev_Logical_Op : Node_Kind; + Op_Location : Source_Ptr; + Node1 : Node_Id; + Node2 : Node_Id; + Attr_Node : Node_Id; + + begin + Node1 := P_Relation; + + if Token = Tok_Apostrophe then + Attr_Node := P_Range_Attribute_Reference (Node1); + Expr_Form := EF_Range_Attr; + return Attr_Node; + + elsif Token in Token_Class_Logop then + Prev_Logical_Op := N_Empty; + + loop + Op_Location := Token_Ptr; + Logical_Op := P_Logical_Operator; + + if Prev_Logical_Op /= N_Empty and then + Logical_Op /= Prev_Logical_Op + then + Error_Msg + ("mixed logical operators in expression", Op_Location); + Prev_Logical_Op := N_Empty; + else + Prev_Logical_Op := Logical_Op; + end if; + + Node2 := Node1; + Node1 := New_Node (Logical_Op, Op_Location); + Set_Left_Opnd (Node1, Node2); + Set_Right_Opnd (Node1, P_Relation); + Set_Op_Name (Node1); + exit when Token not in Token_Class_Logop; + end loop; + + Expr_Form := EF_Non_Simple; + end if; + + if Token = Tok_Apostrophe then + Bad_Range_Attribute (Token_Ptr); + return Error; + else + return Node1; + end if; + end P_Expression_Or_Range_Attribute; + + ------------------- + -- 4.4 Relation -- + ------------------- + + -- RELATION ::= + -- SIMPLE_EXPRESSION [RELATIONAL_OPERATOR SIMPLE_EXPRESSION] + -- | SIMPLE_EXPRESSION [not] in RANGE + -- | SIMPLE_EXPRESSION [not] in SUBTYPE_MARK + + -- On return, Expr_Form indicates the categorization of the expression + + -- Note: if Token = Tok_Apostrophe on return, then Expr_Form is set to + -- EF_Simple_Name and the following token is RANGE (range attribute case). + + -- Error recovery: cannot raise Error_Resync. If an error occurs within an + -- expression, then tokens are scanned until either a non-expression token, + -- a right paren (not matched by a left paren) or a comma, is encountered. + + function P_Relation return Node_Id is + Node1, Node2 : Node_Id; + Optok : Source_Ptr; + + begin + Node1 := P_Simple_Expression; + + if Token not in Token_Class_Relop then + return Node1; + + else + -- Here we have a relational operator following. If so then scan it + -- out. Note that the assignment symbol := is treated as a relational + -- operator to improve the error recovery when it is misused for =. + -- P_Relational_Operator also parses the IN and NOT IN operations. + + Optok := Token_Ptr; + Node2 := New_Node (P_Relational_Operator, Optok); + Set_Left_Opnd (Node2, Node1); + Set_Op_Name (Node2); + + -- Case of IN or NOT IN + + if Prev_Token = Tok_In then + Set_Right_Opnd (Node2, P_Range_Or_Subtype_Mark); + + -- Case of relational operator (= /= < <= > >=) + + else + Set_Right_Opnd (Node2, P_Simple_Expression); + end if; + + Expr_Form := EF_Non_Simple; + + if Token in Token_Class_Relop then + Error_Msg_SC ("unexpected relational operator"); + raise Error_Resync; + end if; + + return Node2; + end if; + + -- If any error occurs, then scan to the next expression terminator symbol + -- or comma or right paren at the outer (i.e. current) parentheses level. + -- The flags are set to indicate a normal simple expression. + + exception + when Error_Resync => + Resync_Expression; + Expr_Form := EF_Simple; + return Error; + end P_Relation; + + ---------------------------- + -- 4.4 Simple Expression -- + ---------------------------- + + -- SIMPLE_EXPRESSION ::= + -- [UNARY_ADDING_OPERATOR] TERM {BINARY_ADDING_OPERATOR TERM} + + -- On return, Expr_Form indicates the categorization of the expression + + -- Note: if Token = Tok_Apostrophe on return, then Expr_Form is set to + -- EF_Simple_Name and the following token is RANGE (range attribute case). + + -- Error recovery: cannot raise Error_Resync. If an error occurs within an + -- expression, then tokens are scanned until either a non-expression token, + -- a right paren (not matched by a left paren) or a comma, is encountered. + + -- Note: P_Simple_Expression is called only internally by higher level + -- expression routines. In cases in the grammar where a simple expression + -- is required, the approach is to scan an expression, and then post an + -- appropriate error message if the expression obtained is not simple. This + -- gives better error recovery and treatment. + + function P_Simple_Expression return Node_Id is + Scan_State : Saved_Scan_State; + Node1 : Node_Id; + Node2 : Node_Id; + Tokptr : Source_Ptr; + + begin + -- Check for cases starting with a name. There are two reasons for + -- special casing. First speed things up by catching a common case + -- without going through several routine layers. Second the caller must + -- be informed via Expr_Form when the simple expression is a name. + + if Token in Token_Class_Name then + Node1 := P_Name; + + -- Deal with apostrophe cases + + if Token = Tok_Apostrophe then + Save_Scan_State (Scan_State); -- at apostrophe + Scan; -- past apostrophe + + -- If qualified expression, scan it out and fall through + + if Token = Tok_Left_Paren then + Node1 := P_Qualified_Expression (Node1); + Expr_Form := EF_Simple; + + -- If range attribute, then we return with Token pointing to the + -- apostrophe. Note: avoid the normal error check on exit. We + -- know that the expression really is complete in this case! + + else -- Token = Tok_Range then + Restore_Scan_State (Scan_State); -- to apostrophe + Expr_Form := EF_Simple_Name; + return Node1; + end if; + end if; + + -- If an expression terminator follows, the previous processing + -- completely scanned out the expression (a common case), and + -- left Expr_Form set appropriately for returning to our caller. + + if Token in Token_Class_Sterm then + null; + + -- If we do not have an expression terminator, then complete the + -- scan of a simple expression. This code duplicates the code + -- found in P_Term and P_Factor. + + else + if Token = Tok_Double_Asterisk then + if Style_Check then Style.Check_Exponentiation_Operator; end if; + Node2 := New_Node (N_Op_Expon, Token_Ptr); + Scan; -- past ** + Set_Left_Opnd (Node2, Node1); + Set_Right_Opnd (Node2, P_Primary); + Set_Op_Name (Node2); + Node1 := Node2; + end if; + + loop + exit when Token not in Token_Class_Mulop; + Tokptr := Token_Ptr; + Node2 := New_Node (P_Multiplying_Operator, Tokptr); + if Style_Check then Style.Check_Binary_Operator; end if; + Scan; -- past operator + Set_Left_Opnd (Node2, Node1); + Set_Right_Opnd (Node2, P_Factor); + Set_Op_Name (Node2); + Node1 := Node2; + end loop; + + loop + exit when Token not in Token_Class_Binary_Addop; + Tokptr := Token_Ptr; + Node2 := New_Node (P_Binary_Adding_Operator, Tokptr); + if Style_Check then Style.Check_Binary_Operator; end if; + Scan; -- past operator + Set_Left_Opnd (Node2, Node1); + Set_Right_Opnd (Node2, P_Term); + Set_Op_Name (Node2); + Node1 := Node2; + end loop; + + Expr_Form := EF_Simple; + end if; + + -- Cases where simple expression does not start with a name + + else + -- Scan initial sign and initial Term + + if Token in Token_Class_Unary_Addop then + Tokptr := Token_Ptr; + Node1 := New_Node (P_Unary_Adding_Operator, Tokptr); + if Style_Check then Style.Check_Unary_Plus_Or_Minus; end if; + Scan; -- past operator + Set_Right_Opnd (Node1, P_Term); + Set_Op_Name (Node1); + else + Node1 := P_Term; + end if; + + -- Scan out sequence of terms separated by binary adding operators + + loop + exit when Token not in Token_Class_Binary_Addop; + Tokptr := Token_Ptr; + Node2 := New_Node (P_Binary_Adding_Operator, Tokptr); + Scan; -- past operator + Set_Left_Opnd (Node2, Node1); + Set_Right_Opnd (Node2, P_Term); + Set_Op_Name (Node2); + Node1 := Node2; + end loop; + + -- All done, we clearly do not have name or numeric literal so this + -- is a case of a simple expression which is some other possibility. + + Expr_Form := EF_Simple; + end if; + + -- Come here at end of simple expression, where we do a couple of + -- special checks to improve error recovery. + + -- Special test to improve error recovery. If the current token + -- is a period, then someone is trying to do selection on something + -- that is not a name, e.g. a qualified expression. + + if Token = Tok_Dot then + Error_Msg_SC ("prefix for selection is not a name"); + raise Error_Resync; + end if; + + -- Special test to improve error recovery: If the current token is + -- not the first token on a line (as determined by checking the + -- previous token position with the start of the current line), + -- then we insist that we have an appropriate terminating token. + -- Consider the following two examples: + + -- 1) if A nad B then ... + + -- 2) A := B + -- C := D + + -- In the first example, we would like to issue a binary operator + -- expected message and resynchronize to the then. In the second + -- example, we do not want to issue a binary operator message, so + -- that instead we will get the missing semicolon message. This + -- distinction is of course a heuristic which does not always work, + -- but in practice it is quite effective. + + -- Note: the one case in which we do not go through this circuit is + -- when we have scanned a range attribute and want to return with + -- Token pointing to the apostrophe. The apostrophe is not normally + -- an expression terminator, and is not in Token_Class_Sterm, but + -- in this special case we know that the expression is complete. + + if not Token_Is_At_Start_Of_Line + and then Token not in Token_Class_Sterm + then + Error_Msg_AP ("binary operator expected"); + raise Error_Resync; + else + return Node1; + end if; + + -- If any error occurs, then scan to next expression terminator symbol + -- or comma, right paren or vertical bar at the outer (i.e. current) paren + -- level. Expr_Form is set to indicate a normal simple expression. + + exception + when Error_Resync => + Resync_Expression; + Expr_Form := EF_Simple; + return Error; + + end P_Simple_Expression; + + ----------------------------------------------- + -- 4.4 Simple Expression or Range Attribute -- + ----------------------------------------------- + + -- SIMPLE_EXPRESSION ::= + -- [UNARY_ADDING_OPERATOR] TERM {BINARY_ADDING_OPERATOR TERM} + + -- RANGE_ATTRIBUTE_REFERENCE ::= PREFIX ' RANGE_ATTRIBUTE_DESIGNATOR + + -- RANGE_ATTRIBUTE_DESIGNATOR ::= range [(static_EXPRESSION)] + + -- Error recovery: cannot raise Error_Resync + + function P_Simple_Expression_Or_Range_Attribute return Node_Id is + Sexpr : Node_Id; + Attr_Node : Node_Id; + + begin + Sexpr := P_Simple_Expression; + + if Token = Tok_Apostrophe then + Attr_Node := P_Range_Attribute_Reference (Sexpr); + Expr_Form := EF_Range_Attr; + return Attr_Node; + + else + return Sexpr; + end if; + end P_Simple_Expression_Or_Range_Attribute; + + --------------- + -- 4.4 Term -- + --------------- + + -- TERM ::= FACTOR {MULTIPLYING_OPERATOR FACTOR} + + -- Error recovery: can raise Error_Resync + + function P_Term return Node_Id is + Node1, Node2 : Node_Id; + Tokptr : Source_Ptr; + + begin + Node1 := P_Factor; + + loop + exit when Token not in Token_Class_Mulop; + Tokptr := Token_Ptr; + Node2 := New_Node (P_Multiplying_Operator, Tokptr); + Scan; -- past operator + Set_Left_Opnd (Node2, Node1); + Set_Right_Opnd (Node2, P_Factor); + Set_Op_Name (Node2); + Node1 := Node2; + end loop; + + return Node1; + end P_Term; + + ----------------- + -- 4.4 Factor -- + ----------------- + + -- FACTOR ::= PRIMARY [** PRIMARY] | abs PRIMARY | not PRIMARY + + -- Error recovery: can raise Error_Resync + + function P_Factor return Node_Id is + Node1 : Node_Id; + Node2 : Node_Id; + + begin + if Token = Tok_Abs then + Node1 := New_Node (N_Op_Abs, Token_Ptr); + if Style_Check then Style.Check_Abs_Not; end if; + Scan; -- past ABS + Set_Right_Opnd (Node1, P_Primary); + Set_Op_Name (Node1); + return Node1; + + elsif Token = Tok_Not then + Node1 := New_Node (N_Op_Not, Token_Ptr); + if Style_Check then Style.Check_Abs_Not; end if; + Scan; -- past NOT + Set_Right_Opnd (Node1, P_Primary); + Set_Op_Name (Node1); + return Node1; + + else + Node1 := P_Primary; + + if Token = Tok_Double_Asterisk then + Node2 := New_Node (N_Op_Expon, Token_Ptr); + Scan; -- past ** + Set_Left_Opnd (Node2, Node1); + Set_Right_Opnd (Node2, P_Primary); + Set_Op_Name (Node2); + return Node2; + else + return Node1; + end if; + end if; + end P_Factor; + + ------------------ + -- 4.4 Primary -- + ------------------ + + -- PRIMARY ::= + -- NUMERIC_LITERAL | null + -- | STRING_LITERAL | AGGREGATE + -- | NAME | QUALIFIED_EXPRESSION + -- | ALLOCATOR | (EXPRESSION) + + -- Error recovery: can raise Error_Resync + + function P_Primary return Node_Id is + Scan_State : Saved_Scan_State; + Node1 : Node_Id; + + begin + -- The loop runs more than once only if misplaced pragmas are found + + loop + case Token is + + -- Name token can start a name, call or qualified expression, all + -- of which are acceptable possibilities for primary. Note also + -- that string literal is included in name (as operator symbol) + -- and type conversion is included in name (as indexed component). + + when Tok_Char_Literal | Tok_Operator_Symbol | Tok_Identifier => + Node1 := P_Name; + + -- All done unless apostrophe follows + + if Token /= Tok_Apostrophe then + return Node1; + + -- Apostrophe following means that we have either just parsed + -- the subtype mark of a qualified expression, or the prefix + -- or a range attribute. + + else -- Token = Tok_Apostrophe + Save_Scan_State (Scan_State); -- at apostrophe + Scan; -- past apostrophe + + -- If range attribute, then this is always an error, since + -- the only legitimate case (where the scanned expression is + -- a qualified simple name) is handled at the level of the + -- Simple_Expression processing. This case corresponds to a + -- usage such as 3 + A'Range, which is always illegal. + + if Token = Tok_Range then + Restore_Scan_State (Scan_State); -- to apostrophe + Bad_Range_Attribute (Token_Ptr); + return Error; + + -- If left paren, then we have a qualified expression. + -- Note that P_Name guarantees that in this case, where + -- Token = Tok_Apostrophe on return, the only two possible + -- tokens following the apostrophe are left paren and + -- RANGE, so we know we have a left paren here. + + else -- Token = Tok_Left_Paren + return P_Qualified_Expression (Node1); + + end if; + end if; + + -- Numeric or string literal + + when Tok_Integer_Literal | + Tok_Real_Literal | + Tok_String_Literal => + + Node1 := Token_Node; + Scan; -- past number + return Node1; + + -- Left paren, starts aggregate or parenthesized expression + + when Tok_Left_Paren => + return P_Aggregate_Or_Paren_Expr; + + -- Allocator + + when Tok_New => + return P_Allocator; + + -- Null + + when Tok_Null => + Scan; -- past NULL + return New_Node (N_Null, Prev_Token_Ptr); + + -- Pragma, not allowed here, so just skip past it + + when Tok_Pragma => + P_Pragmas_Misplaced; + + -- Anything else is illegal as the first token of a primary, but + -- we test for a reserved identifier so that it is treated nicely + + when others => + if Is_Reserved_Identifier then + return P_Identifier; + + elsif Prev_Token = Tok_Comma then + Error_Msg_SP ("extra "","" ignored"); + raise Error_Resync; + + else + Error_Msg_AP ("missing operand"); + raise Error_Resync; + end if; + + end case; + end loop; + end P_Primary; + + --------------------------- + -- 4.5 Logical Operator -- + --------------------------- + + -- LOGICAL_OPERATOR ::= and | or | xor + + -- Note: AND THEN and OR ELSE are also treated as logical operators + -- by the parser (even though they are not operators semantically) + + -- The value returned is the appropriate Node_Kind code for the operator + -- On return, Token points to the token following the scanned operator. + + -- The caller has checked that the first token is a legitimate logical + -- operator token (i.e. is either XOR, AND, OR). + + -- Error recovery: cannot raise Error_Resync + + function P_Logical_Operator return Node_Kind is + begin + if Token = Tok_And then + if Style_Check then Style.Check_Binary_Operator; end if; + Scan; -- past AND + + if Token = Tok_Then then + Scan; -- past THEN + return N_And_Then; + else + return N_Op_And; + end if; + + elsif Token = Tok_Or then + if Style_Check then Style.Check_Binary_Operator; end if; + Scan; -- past OR + + if Token = Tok_Else then + Scan; -- past ELSE + return N_Or_Else; + else + return N_Op_Or; + end if; + + else -- Token = Tok_Xor + if Style_Check then Style.Check_Binary_Operator; end if; + Scan; -- past XOR + return N_Op_Xor; + end if; + end P_Logical_Operator; + + ------------------------------ + -- 4.5 Relational Operator -- + ------------------------------ + + -- RELATIONAL_OPERATOR ::= = | /= | < | <= | > | >= + + -- The value returned is the appropriate Node_Kind code for the operator. + -- On return, Token points to the operator token, NOT past it. + + -- The caller has checked that the first token is a legitimate relational + -- operator token (i.e. is one of the operator tokens listed above). + + -- Error recovery: cannot raise Error_Resync + + function P_Relational_Operator return Node_Kind is + Op_Kind : Node_Kind; + Relop_Node : constant array (Token_Class_Relop) of Node_Kind := + (Tok_Less => N_Op_Lt, + Tok_Equal => N_Op_Eq, + Tok_Greater => N_Op_Gt, + Tok_Not_Equal => N_Op_Ne, + Tok_Greater_Equal => N_Op_Ge, + Tok_Less_Equal => N_Op_Le, + Tok_In => N_In, + Tok_Not => N_Not_In, + Tok_Box => N_Op_Ne); + + begin + if Token = Tok_Box then + Error_Msg_SC ("""<>"" should be ""/="""); + end if; + + Op_Kind := Relop_Node (Token); + if Style_Check then Style.Check_Binary_Operator; end if; + Scan; -- past operator token + + if Prev_Token = Tok_Not then + T_In; + end if; + + return Op_Kind; + end P_Relational_Operator; + + --------------------------------- + -- 4.5 Binary Adding Operator -- + --------------------------------- + + -- BINARY_ADDING_OPERATOR ::= + | - | & + + -- The value returned is the appropriate Node_Kind code for the operator. + -- On return, Token points to the operator token (NOT past it). + + -- The caller has checked that the first token is a legitimate adding + -- operator token (i.e. is one of the operator tokens listed above). + + -- Error recovery: cannot raise Error_Resync + + function P_Binary_Adding_Operator return Node_Kind is + Addop_Node : constant array (Token_Class_Binary_Addop) of Node_Kind := + (Tok_Ampersand => N_Op_Concat, + Tok_Minus => N_Op_Subtract, + Tok_Plus => N_Op_Add); + begin + return Addop_Node (Token); + end P_Binary_Adding_Operator; + + -------------------------------- + -- 4.5 Unary Adding Operator -- + -------------------------------- + + -- UNARY_ADDING_OPERATOR ::= + | - + + -- The value returned is the appropriate Node_Kind code for the operator. + -- On return, Token points to the operator token (NOT past it). + + -- The caller has checked that the first token is a legitimate adding + -- operator token (i.e. is one of the operator tokens listed above). + + -- Error recovery: cannot raise Error_Resync + + function P_Unary_Adding_Operator return Node_Kind is + Addop_Node : constant array (Token_Class_Unary_Addop) of Node_Kind := + (Tok_Minus => N_Op_Minus, + Tok_Plus => N_Op_Plus); + begin + return Addop_Node (Token); + end P_Unary_Adding_Operator; + + ------------------------------- + -- 4.5 Multiplying Operator -- + ------------------------------- + + -- MULTIPLYING_OPERATOR ::= * | / | mod | rem + + -- The value returned is the appropriate Node_Kind code for the operator. + -- On return, Token points to the operator token (NOT past it). + + -- The caller has checked that the first token is a legitimate multiplying + -- operator token (i.e. is one of the operator tokens listed above). + + -- Error recovery: cannot raise Error_Resync + + function P_Multiplying_Operator return Node_Kind is + Mulop_Node : constant array (Token_Class_Mulop) of Node_Kind := + (Tok_Asterisk => N_Op_Multiply, + Tok_Mod => N_Op_Mod, + Tok_Rem => N_Op_Rem, + Tok_Slash => N_Op_Divide); + begin + return Mulop_Node (Token); + end P_Multiplying_Operator; + + -------------------------------------- + -- 4.5 Highest Precedence Operator -- + -------------------------------------- + + -- Parsed by P_Factor (4.4) + + -- Note: this rule is not in fact used by the grammar at any point! + + -------------------------- + -- 4.6 Type Conversion -- + -------------------------- + + -- Parsed by P_Primary as a Name (4.1) + + ------------------------------- + -- 4.7 Qualified Expression -- + ------------------------------- + + -- QUALIFIED_EXPRESSION ::= + -- SUBTYPE_MARK ' (EXPRESSION) | SUBTYPE_MARK ' AGGREGATE + + -- The caller has scanned the name which is the Subtype_Mark parameter + -- and scanned past the single quote following the subtype mark. The + -- caller has not checked that this name is in fact appropriate for + -- a subtype mark name (i.e. it is a selected component or identifier). + + -- Error_Recovery: cannot raise Error_Resync + + function P_Qualified_Expression (Subtype_Mark : Node_Id) return Node_Id is + Qual_Node : Node_Id; + + begin + Qual_Node := New_Node (N_Qualified_Expression, Prev_Token_Ptr); + Set_Subtype_Mark (Qual_Node, Check_Subtype_Mark (Subtype_Mark)); + Set_Expression (Qual_Node, P_Aggregate_Or_Paren_Expr); + return Qual_Node; + end P_Qualified_Expression; + + -------------------- + -- 4.8 Allocator -- + -------------------- + + -- ALLOCATOR ::= + -- new SUBTYPE_INDICATION | new QUALIFIED_EXPRESSION + + -- The caller has checked that the initial token is NEW + + -- Error recovery: can raise Error_Resync + + function P_Allocator return Node_Id is + Alloc_Node : Node_Id; + Type_Node : Node_Id; + + begin + Alloc_Node := New_Node (N_Allocator, Token_Ptr); + T_New; + Type_Node := P_Subtype_Mark_Resync; + + if Token = Tok_Apostrophe then + Scan; -- past apostrophe + Set_Expression (Alloc_Node, P_Qualified_Expression (Type_Node)); + else + Set_Expression (Alloc_Node, P_Subtype_Indication (Type_Node)); + end if; + + return Alloc_Node; + end P_Allocator; + +end Ch4; diff --git a/gcc/ada/par-ch5.adb b/gcc/ada/par-ch5.adb new file mode 100644 index 00000000000..2ec56726e6b --- /dev/null +++ b/gcc/ada/par-ch5.adb @@ -0,0 +1,2184 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 5 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.95 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch5 is + + -- Local functions, used only in this chapter + + function P_Case_Statement return Node_Id; + function P_Case_Statement_Alternative return Node_Id; + function P_Condition return Node_Id; + function P_Exit_Statement return Node_Id; + function P_Goto_Statement return Node_Id; + function P_If_Statement return Node_Id; + function P_Label return Node_Id; + function P_Loop_Parameter_Specification return Node_Id; + function P_Null_Statement return Node_Id; + + function P_Assignment_Statement (LHS : Node_Id) return Node_Id; + -- Parse assignment statement. On entry, the caller has scanned the left + -- hand side (passed in as Lhs), and the colon-equal (or some symbol + -- taken to be an error equivalent such as equal). + + function P_Begin_Statement (Block_Name : Node_Id := Empty) return Node_Id; + -- Parse begin-end statement. If Block_Name is non-Empty on entry, it is + -- the N_Identifier node for the label on the block. If Block_Name is + -- Empty on entry (the default), then the block statement is unlabeled. + + function P_Declare_Statement (Block_Name : Node_Id := Empty) return Node_Id; + -- Parse declare block. If Block_Name is non-Empty on entry, it is + -- the N_Identifier node for the label on the block. If Block_Name is + -- Empty on entry (the default), then the block statement is unlabeled. + + function P_For_Statement (Loop_Name : Node_Id := Empty) return Node_Id; + -- Parse for statement. If Loop_Name is non-Empty on entry, it is + -- the N_Identifier node for the label on the loop. If Loop_Name is + -- Empty on entry (the default), then the for statement is unlabeled. + + function P_Loop_Statement (Loop_Name : Node_Id := Empty) return Node_Id; + -- Parse loop statement. If Loop_Name is non-Empty on entry, it is + -- the N_Identifier node for the label on the loop. If Loop_Name is + -- Empty on entry (the default), then the loop statement is unlabeled. + + function P_While_Statement (Loop_Name : Node_Id := Empty) return Node_Id; + -- Parse while statement. If Loop_Name is non-Empty on entry, it is + -- the N_Identifier node for the label on the loop. If Loop_Name is + -- Empty on entry (the default), then the while statement is unlabeled. + + function Set_Loop_Block_Name (L : Character) return Name_Id; + -- Given a letter 'L' for a loop or 'B' for a block, returns a name + -- of the form L_nn or B_nn where nn is a serial number obtained by + -- incrementing the variable Loop_Block_Count. + + procedure Then_Scan; + -- Scan past THEN token, testing for illegal junk after it + + --------------------------------- + -- 5.1 Sequence of Statements -- + --------------------------------- + + -- SEQUENCE_OF_STATEMENTS ::= STATEMENT {STATEMENT} + + -- STATEMENT ::= + -- {LABEL} SIMPLE_STATEMENT | {LABEL} COMPOUND_STATEMENT + + -- SIMPLE_STATEMENT ::= NULL_STATEMENT + -- | ASSIGNMENT_STATEMENT | EXIT_STATEMENT + -- | GOTO_STATEMENT | PROCEDURE_CALL_STATEMENT + -- | RETURN_STATEMENT | ENTRY_CALL_STATEMENT + -- | REQUEUE_STATEMENT | DELAY_STATEMENT + -- | ABORT_STATEMENT | RAISE_STATEMENT + -- | CODE_STATEMENT + + -- COMPOUND_STATEMENT ::= + -- IF_STATEMENT | CASE_STATEMENT + -- | LOOP_STATEMENT | BLOCK_STATEMENT + -- | ACCEPT_STATEMENT | SELECT_STATEMENT + + -- This procedure scans a sequence of statements. The caller sets SS_Flags + -- to indicate acceptable termination conditions for the sequence: + + -- SS_Flags.Eftm Terminate on ELSIF + -- SS_Flags.Eltm Terminate on ELSE + -- SS_Flags.Extm Terminate on EXCEPTION + -- SS_Flags.Ortm Terminate on OR + -- SS_Flags.Tatm Terminate on THEN ABORT (Token = ABORT on return) + -- SS_Flags.Whtm Terminate on WHEN + -- SS_Flags.Unco Unconditional terminate after scanning one statement + + -- In addition, the scan is always terminated by encountering END or the + -- end of file (EOF) condition. If one of the six above terminators is + -- encountered with the corresponding SS_Flags flag not set, then the + -- action taken is as follows: + + -- If the keyword occurs to the left of the expected column of the end + -- for the current sequence (as recorded in the current end context), + -- then it is assumed to belong to an outer context, and is considered + -- to terminate the sequence of statements. + + -- If the keyword occurs to the right of, or in the expected column of + -- the end for the current sequence, then an error message is output, + -- the keyword together with its associated context is skipped, and + -- the statement scan continues until another terminator is found. + + -- Note that the first action means that control can return to the caller + -- with Token set to a terminator other than one of those specified by the + -- SS parameter. The caller should treat such a case as equivalent to END. + + -- In addition, the flag SS_Flags.Sreq is set to True to indicate that at + -- least one real statement (other than a pragma) is required in the + -- statement sequence. During the processing of the sequence, this + -- flag is manipulated to indicate the current status of the requirement + -- for a statement. For example, it is turned off by the occurrence of a + -- statement, and back on by a label (which requires a following statement) + + -- Error recovery: cannot raise Error_Resync. If an error occurs during + -- parsing a statement, then the scan pointer is advanced past the next + -- semicolon and the parse continues. + + function P_Sequence_Of_Statements (SS_Flags : SS_Rec) return List_Id is + + Statement_Required : Boolean; + -- This flag indicates if a subsequent statement (other than a pragma) + -- is required. It is initialized from the Sreq flag, and modified as + -- statements are scanned (a statement turns it off, and a label turns + -- it back on again since a statement must follow a label). + + Declaration_Found : Boolean := False; + -- This flag is set True if a declaration is encountered, so that the + -- error message about declarations in the statement part is only + -- given once for a given sequence of statements. + + Scan_State_Label : Saved_Scan_State; + Scan_State : Saved_Scan_State; + + Statement_List : List_Id; + Block_Label : Name_Id; + Id_Node : Node_Id; + Name_Node : Node_Id; + + procedure Junk_Declaration; + -- Procedure called to handle error of declaration encountered in + -- statement sequence. + + procedure Test_Statement_Required; + -- Flag error if Statement_Required flag set + + procedure Junk_Declaration is + begin + if (not Declaration_Found) or All_Errors_Mode then + Error_Msg_SC ("declarations must come before BEGIN"); + Declaration_Found := True; + end if; + + Skip_Declaration (Statement_List); + end Junk_Declaration; + + procedure Test_Statement_Required is + begin + if Statement_Required then + Error_Msg_BC ("statement expected"); + end if; + end Test_Statement_Required; + + -- Start of processing for P_Sequence_Of_Statements + + begin + Statement_List := New_List; + Statement_Required := SS_Flags.Sreq; + + loop + while Token = Tok_Semicolon loop + Error_Msg_SC ("unexpected semicolon ignored"); + Scan; -- past junk semicolon + end loop; + + begin + if Style_Check then Style.Check_Indentation; end if; + + -- Deal with reserved identifier (in assignment or call) + + if Is_Reserved_Identifier then + Save_Scan_State (Scan_State); -- at possible bad identifier + Scan; -- and scan past it + + -- We have an reserved word which is spelled in identifier + -- style, so the question is whether it really is intended + -- to be an identifier. + + if + -- If followed by a semicolon, then it is an identifier, + -- with the exception of the cases tested for below. + + (Token = Tok_Semicolon + and then Prev_Token /= Tok_Return + and then Prev_Token /= Tok_Null + and then Prev_Token /= Tok_Raise + and then Prev_Token /= Tok_End + and then Prev_Token /= Tok_Exit) + + -- If followed by colon, colon-equal, or dot, then we + -- definitely have an identifier (could not be reserved) + + or else Token = Tok_Colon + or else Token = Tok_Colon_Equal + or else Token = Tok_Dot + + -- Left paren means we have an identifier except for those + -- reserved words that can legitimately be followed by a + -- left paren. + + or else + (Token = Tok_Left_Paren + and then Prev_Token /= Tok_Case + and then Prev_Token /= Tok_Delay + and then Prev_Token /= Tok_If + and then Prev_Token /= Tok_Elsif + and then Prev_Token /= Tok_Return + and then Prev_Token /= Tok_When + and then Prev_Token /= Tok_While + and then Prev_Token /= Tok_Separate) + then + -- Here we have an apparent reserved identifier and the + -- token past it is appropriate to this usage (and would + -- be a definite error if this is not an identifier). What + -- we do is to use P_Identifier to fix up the identifier, + -- and then fall into the normal processing. + + Restore_Scan_State (Scan_State); -- back to the ID + Scan_Reserved_Identifier (Force_Msg => False); + + -- Not a reserved identifier after all (or at least we can't + -- be sure that it is), so reset the scan and continue. + + else + Restore_Scan_State (Scan_State); -- back to the reserved word + end if; + end if; + + -- Now look to see what kind of statement we have + + case Token is + + -- Case of end or EOF + + when Tok_End | Tok_EOF => + + -- These tokens always terminate the statement sequence + + Test_Statement_Required; + exit; + + -- Case of ELSIF + + when Tok_Elsif => + + -- Terminate if Eftm set or if the ELSIF is to the left + -- of the expected column of the end for this sequence + + if SS_Flags.Eftm + or else Start_Column < Scope.Table (Scope.Last).Ecol + then + Test_Statement_Required; + exit; + + -- Otherwise complain and skip past ELSIF Condition then + + else + Error_Msg_SC ("ELSIF not allowed here"); + Scan; -- past ELSIF + Discard_Junk_Node (P_Expression_No_Right_Paren); + Then_Scan; + Statement_Required := False; + end if; + + -- Case of ELSE + + when Tok_Else => + + -- Terminate if Eltm set or if the else is to the left + -- of the expected column of the end for this sequence + + if SS_Flags.Eltm + or else Start_Column < Scope.Table (Scope.Last).Ecol + then + Test_Statement_Required; + exit; + + -- Otherwise complain and skip past else + + else + Error_Msg_SC ("ELSE not allowed here"); + Scan; -- past ELSE + Statement_Required := False; + end if; + + -- Case of exception + + when Tok_Exception => + Test_Statement_Required; + + -- If Extm not set and the exception is not to the left + -- of the expected column of the end for this sequence, then + -- we assume it belongs to the current sequence, even though + -- it is not permitted. + + if not SS_Flags.Extm and then + Start_Column >= Scope.Table (Scope.Last).Ecol + + then + Error_Msg_SC ("exception handler not permitted here"); + Scan; -- past EXCEPTION + Discard_Junk_List (Parse_Exception_Handlers); + end if; + + -- Always return, in the case where we scanned out handlers + -- that we did not expect, Parse_Exception_Handlers returned + -- with Token being either end or EOF, so we are OK + + exit; + + -- Case of OR + + when Tok_Or => + + -- Terminate if Ortm set or if the or is to the left + -- of the expected column of the end for this sequence + + if SS_Flags.Ortm + or else Start_Column < Scope.Table (Scope.Last).Ecol + then + Test_Statement_Required; + exit; + + -- Otherwise complain and skip past or + + else + Error_Msg_SC ("OR not allowed here"); + Scan; -- past or + Statement_Required := False; + end if; + + -- Case of THEN (deal also with THEN ABORT) + + when Tok_Then => + Save_Scan_State (Scan_State); -- at THEN + Scan; -- past THEN + + -- Terminate if THEN ABORT allowed (ATC case) + + exit when SS_Flags.Tatm and then Token = Tok_Abort; + + -- Otherwise we treat THEN as some kind of mess where we + -- did not see the associated IF, but we pick up assuming + -- it had been there! + + Restore_Scan_State (Scan_State); -- to THEN + Append_To (Statement_List, P_If_Statement); + Statement_Required := False; + + -- Case of WHEN (error because we are not in a case) + + when Tok_When | Tok_Others => + + -- Terminate if Whtm set or if the WHEN is to the left + -- of the expected column of the end for this sequence + + if SS_Flags.Whtm + or else Start_Column < Scope.Table (Scope.Last).Ecol + then + Test_Statement_Required; + exit; + + -- Otherwise complain and skip when Choice {| Choice} => + + else + Error_Msg_SC ("WHEN not allowed here"); + Scan; -- past when + Discard_Junk_List (P_Discrete_Choice_List); + TF_Arrow; + Statement_Required := False; + end if; + + -- Cases of statements starting with an identifier + + when Tok_Identifier => + Check_Bad_Layout; + + -- Save scan pointers and line number in case block label + + Id_Node := Token_Node; + Block_Label := Token_Name; + Save_Scan_State (Scan_State_Label); -- at possible label + Scan; -- past Id + + -- Check for common case of assignment, since it occurs + -- frequently, and we want to process it efficiently. + + if Token = Tok_Colon_Equal then + Scan; -- past the colon-equal + Append_To (Statement_List, + P_Assignment_Statement (Id_Node)); + Statement_Required := False; + + -- Check common case of procedure call, another case that + -- we want to speed up as much as possible. + + elsif Token = Tok_Semicolon then + Append_To (Statement_List, + P_Statement_Name (Id_Node)); + Scan; -- past semicolon + Statement_Required := False; + + -- Check for case of "go to" in place of "goto" + + elsif Token = Tok_Identifier + and then Block_Label = Name_Go + and then Token_Name = Name_To + then + Error_Msg_SP ("goto is one word"); + Append_To (Statement_List, P_Goto_Statement); + Statement_Required := False; + + -- Check common case of = used instead of :=, just so we + -- give a better error message for this special misuse. + + elsif Token = Tok_Equal then + T_Colon_Equal; -- give := expected message + Append_To (Statement_List, + P_Assignment_Statement (Id_Node)); + Statement_Required := False; + + -- Check case of loop label or block label + + elsif Token = Tok_Colon + or else (Token in Token_Class_Labeled_Stmt + and then not Token_Is_At_Start_Of_Line) + then + T_Colon; -- past colon (if there, or msg for missing one) + + -- Test for more than one label + + loop + exit when Token /= Tok_Identifier; + Save_Scan_State (Scan_State); -- at second Id + Scan; -- past Id + + if Token = Tok_Colon then + Error_Msg_SP + ("only one label allowed on block or loop"); + Scan; -- past colon on extra label + + -- Use the second label as the "real" label + + Scan_State_Label := Scan_State; + + -- We will set Error_name as the Block_Label since + -- we really don't know which of the labels might + -- be used at the end of the loop or block! + + Block_Label := Error_Name; + + -- If Id with no colon, then backup to point to the + -- Id and we will issue the message below when we try + -- to scan out the statement as some other form. + + else + Restore_Scan_State (Scan_State); -- to second Id + exit; + end if; + end loop; + + -- Loop_Statement (labeled Loop_Statement) + + if Token = Tok_Loop then + Append_To (Statement_List, + P_Loop_Statement (Id_Node)); + + -- While statement (labeled loop statement with WHILE) + + elsif Token = Tok_While then + Append_To (Statement_List, + P_While_Statement (Id_Node)); + + -- Declare statement (labeled block statement with + -- DECLARE part) + + elsif Token = Tok_Declare then + Append_To (Statement_List, + P_Declare_Statement (Id_Node)); + + -- Begin statement (labeled block statement with no + -- DECLARE part) + + elsif Token = Tok_Begin then + Append_To (Statement_List, + P_Begin_Statement (Id_Node)); + + -- For statement (labeled loop statement with FOR) + + elsif Token = Tok_For then + Append_To (Statement_List, + P_For_Statement (Id_Node)); + + -- Improper statement follows label. If we have an + -- expression token, then assume the colon was part + -- of a misplaced declaration. + + elsif Token not in Token_Class_Eterm then + Restore_Scan_State (Scan_State_Label); + Junk_Declaration; + + -- Otherwise complain we have inappropriate statement + + else + Error_Msg_AP + ("loop or block statement must follow label"); + end if; + + Statement_Required := False; + + -- Here we have an identifier followed by something + -- other than a colon, semicolon or assignment symbol. + -- The only valid possibility is a name extension symbol + + elsif Token in Token_Class_Namext then + Restore_Scan_State (Scan_State_Label); -- to Id + Name_Node := P_Name; + + -- Skip junk right parens in this context + + while Token = Tok_Right_Paren loop + Error_Msg_SC ("extra right paren"); + Scan; -- past ) + end loop; + + -- Check context following call + + if Token = Tok_Colon_Equal then + Scan; -- past colon equal + Append_To (Statement_List, + P_Assignment_Statement (Name_Node)); + Statement_Required := False; + + -- Check common case of = used instead of := + + elsif Token = Tok_Equal then + T_Colon_Equal; -- give := expected message + Append_To (Statement_List, + P_Assignment_Statement (Name_Node)); + Statement_Required := False; + + -- Check apostrophe cases + + elsif Token = Tok_Apostrophe then + Append_To (Statement_List, + P_Code_Statement (Name_Node)); + Statement_Required := False; + + -- The only other valid item after a name is ; which + -- means that the item we just scanned was a call. + + elsif Token = Tok_Semicolon then + Append_To (Statement_List, + P_Statement_Name (Name_Node)); + Scan; -- past semicolon + Statement_Required := False; + + -- Else we have a missing semicolon + + else + TF_Semicolon; + Statement_Required := False; + end if; + + -- If junk after identifier, check if identifier is an + -- instance of an incorrectly spelled keyword. If so, we + -- do nothing. The Bad_Spelling_Of will have reset Token + -- to the appropriate keyword, so the next time round the + -- loop we will process the modified token. Note that we + -- check for ELSIF before ELSE here. That's not accidental. + -- We don't want to identify a misspelling of ELSE as + -- ELSIF, and in particular we do not want to treat ELSEIF + -- as ELSE IF. + + else + Restore_Scan_State (Scan_State_Label); -- to identifier + + if Bad_Spelling_Of (Tok_Abort) + or else Bad_Spelling_Of (Tok_Accept) + or else Bad_Spelling_Of (Tok_Case) + or else Bad_Spelling_Of (Tok_Declare) + or else Bad_Spelling_Of (Tok_Delay) + or else Bad_Spelling_Of (Tok_Elsif) + or else Bad_Spelling_Of (Tok_Else) + or else Bad_Spelling_Of (Tok_End) + or else Bad_Spelling_Of (Tok_Exception) + or else Bad_Spelling_Of (Tok_Exit) + or else Bad_Spelling_Of (Tok_For) + or else Bad_Spelling_Of (Tok_Goto) + or else Bad_Spelling_Of (Tok_If) + or else Bad_Spelling_Of (Tok_Loop) + or else Bad_Spelling_Of (Tok_Or) + or else Bad_Spelling_Of (Tok_Pragma) + or else Bad_Spelling_Of (Tok_Raise) + or else Bad_Spelling_Of (Tok_Requeue) + or else Bad_Spelling_Of (Tok_Return) + or else Bad_Spelling_Of (Tok_Select) + or else Bad_Spelling_Of (Tok_When) + or else Bad_Spelling_Of (Tok_While) + then + null; + + -- If not a bad spelling, then we really have junk + + else + Scan; -- past identifier again + + -- If next token is first token on line, then we + -- consider that we were missing a semicolon after + -- the identifier, and process it as a procedure + -- call with no parameters. + + if Token_Is_At_Start_Of_Line then + Append_To (Statement_List, + P_Statement_Name (Id_Node)); + T_Semicolon; -- to give error message + Statement_Required := False; + + -- Otherwise we give a missing := message and + -- simply abandon the junk that is there now. + + else + T_Colon_Equal; -- give := expected message + raise Error_Resync; + end if; + + end if; + end if; + + -- Statement starting with operator symbol. This could be + -- a call, a name starting an assignment, or a qualified + -- expression. + + when Tok_Operator_Symbol => + Check_Bad_Layout; + Name_Node := P_Name; + + -- An attempt at a range attribute or a qualified expression + -- must be illegal here (a code statement cannot possibly + -- allow qualification by a function name). + + if Token = Tok_Apostrophe then + Error_Msg_SC ("apostrophe illegal here"); + raise Error_Resync; + end if; + + -- Scan possible assignment if we have a name + + if Expr_Form = EF_Name + and then Token = Tok_Colon_Equal + then + Scan; -- past colon equal + Append_To (Statement_List, + P_Assignment_Statement (Name_Node)); + else + Append_To (Statement_List, + P_Statement_Name (Name_Node)); + end if; + + TF_Semicolon; + Statement_Required := False; + + -- Label starting with << which must precede real statement + + when Tok_Less_Less => + Append_To (Statement_List, P_Label); + Statement_Required := True; + + -- Pragma appearing as a statement in a statement sequence + + when Tok_Pragma => + Check_Bad_Layout; + Append_To (Statement_List, P_Pragma); + + -- Abort_Statement + + when Tok_Abort => + Check_Bad_Layout; + Append_To (Statement_List, P_Abort_Statement); + Statement_Required := False; + + -- Accept_Statement + + when Tok_Accept => + Check_Bad_Layout; + Append_To (Statement_List, P_Accept_Statement); + Statement_Required := False; + + -- Begin_Statement (Block_Statement with no declare, no label) + + when Tok_Begin => + Check_Bad_Layout; + Append_To (Statement_List, P_Begin_Statement); + Statement_Required := False; + + -- Case_Statement + + when Tok_Case => + Check_Bad_Layout; + Append_To (Statement_List, P_Case_Statement); + Statement_Required := False; + + -- Block_Statement with DECLARE and no label + + when Tok_Declare => + Check_Bad_Layout; + Append_To (Statement_List, P_Declare_Statement); + Statement_Required := False; + + -- Delay_Statement + + when Tok_Delay => + Check_Bad_Layout; + Append_To (Statement_List, P_Delay_Statement); + Statement_Required := False; + + -- Exit_Statement + + when Tok_Exit => + Check_Bad_Layout; + Append_To (Statement_List, P_Exit_Statement); + Statement_Required := False; + + -- Loop_Statement with FOR and no label + + when Tok_For => + Check_Bad_Layout; + Append_To (Statement_List, P_For_Statement); + Statement_Required := False; + + -- Goto_Statement + + when Tok_Goto => + Check_Bad_Layout; + Append_To (Statement_List, P_Goto_Statement); + Statement_Required := False; + + -- If_Statement + + when Tok_If => + Check_Bad_Layout; + Append_To (Statement_List, P_If_Statement); + Statement_Required := False; + + -- Loop_Statement + + when Tok_Loop => + Check_Bad_Layout; + Append_To (Statement_List, P_Loop_Statement); + Statement_Required := False; + + -- Null_Statement + + when Tok_Null => + Check_Bad_Layout; + Append_To (Statement_List, P_Null_Statement); + Statement_Required := False; + + -- Raise_Statement + + when Tok_Raise => + Check_Bad_Layout; + Append_To (Statement_List, P_Raise_Statement); + Statement_Required := False; + + -- Requeue_Statement + + when Tok_Requeue => + Check_Bad_Layout; + Append_To (Statement_List, P_Requeue_Statement); + Statement_Required := False; + + -- Return_Statement + + when Tok_Return => + Check_Bad_Layout; + Append_To (Statement_List, P_Return_Statement); + Statement_Required := False; + + -- Select_Statement + + when Tok_Select => + Check_Bad_Layout; + Append_To (Statement_List, P_Select_Statement); + Statement_Required := False; + + -- While_Statement (Block_Statement with while and no loop) + + when Tok_While => + Check_Bad_Layout; + Append_To (Statement_List, P_While_Statement); + Statement_Required := False; + + -- Anything else is some kind of junk, signal an error message + -- and then raise Error_Resync, to merge with the normal + -- handling of a bad statement. + + when others => + + if Token in Token_Class_Declk then + Junk_Declaration; + + else + Error_Msg_BC ("statement expected"); + raise Error_Resync; + end if; + end case; + + -- On error resynchronization, skip past next semicolon, and, since + -- we are still in the statement loop, look for next statement. We + -- set Statement_Required False to avoid an unnecessary error message + -- complaining that no statement was found (i.e. we consider the + -- junk to satisfy the requirement for a statement being present). + + exception + when Error_Resync => + Resync_Past_Semicolon_Or_To_Loop_Or_Then; + Statement_Required := False; + end; + + exit when SS_Flags.Unco; + + end loop; + + return Statement_List; + + end P_Sequence_Of_Statements; + + -------------------- + -- 5.1 Statement -- + -------------------- + + -- Parsed by P_Sequence_Of_Statements (5.1), except for the case + -- of a statement of the form of a name, which is handled here. The + -- argument passed in is the tree for the name which has been scanned + -- The returned value is the corresponding statement form. + + -- This routine is also used by Par.Prag for processing the procedure + -- call that appears as the second argument of a pragma Assert. + + -- Error recovery: cannot raise Error_Resync + + function P_Statement_Name (Name_Node : Node_Id) return Node_Id is + Stmt_Node : Node_Id; + + begin + -- Case of Indexed component, which is a procedure call with arguments + + if Nkind (Name_Node) = N_Indexed_Component then + declare + Prefix_Node : Node_Id := Prefix (Name_Node); + Exprs_Node : List_Id := Expressions (Name_Node); + begin + Change_Node (Name_Node, N_Procedure_Call_Statement); + Set_Name (Name_Node, Prefix_Node); + Set_Parameter_Associations (Name_Node, Exprs_Node); + return Name_Node; + end; + + -- Case of function call node, which is a really a procedure call + + elsif Nkind (Name_Node) = N_Function_Call then + declare + Fname_Node : Node_Id := Name (Name_Node); + Params_List : List_Id := Parameter_Associations (Name_Node); + + begin + Change_Node (Name_Node, N_Procedure_Call_Statement); + Set_Name (Name_Node, Fname_Node); + Set_Parameter_Associations (Name_Node, Params_List); + return Name_Node; + end; + + -- Case of call to attribute that denotes a procedure. Here we + -- just leave the attribute reference unchanged. + + elsif Nkind (Name_Node) = N_Attribute_Reference + and then Is_Procedure_Attribute_Name (Attribute_Name (Name_Node)) + then + return Name_Node; + + -- All other cases of names are parameterless procedure calls + + else + Stmt_Node := + New_Node (N_Procedure_Call_Statement, Sloc (Name_Node)); + Set_Name (Stmt_Node, Name_Node); + return Stmt_Node; + end if; + + end P_Statement_Name; + + --------------------------- + -- 5.1 Simple Statement -- + --------------------------- + + -- Parsed by P_Sequence_Of_Statements (5.1) + + ----------------------------- + -- 5.1 Compound Statement -- + ----------------------------- + + -- Parsed by P_Sequence_Of_Statements (5.1) + + ------------------------- + -- 5.1 Null Statement -- + ------------------------- + + -- NULL_STATEMENT ::= null; + + -- The caller has already checked that the current token is null + + -- Error recovery: cannot raise Error_Resync + + function P_Null_Statement return Node_Id is + Null_Stmt_Node : Node_Id; + + begin + Null_Stmt_Node := New_Node (N_Null_Statement, Token_Ptr); + Scan; -- past NULL + TF_Semicolon; + return Null_Stmt_Node; + end P_Null_Statement; + + ---------------- + -- 5.1 Label -- + ---------------- + + -- LABEL ::= <<label_STATEMENT_IDENTIFIER>> + + -- STATEMENT_INDENTIFIER ::= DIRECT_NAME + + -- The IDENTIFIER of a STATEMENT_IDENTIFIER shall be an identifier + -- (not an OPERATOR_SYMBOL) + + -- The caller has already checked that the current token is << + + -- Error recovery: can raise Error_Resync + + function P_Label return Node_Id is + Label_Node : Node_Id; + + begin + Label_Node := New_Node (N_Label, Token_Ptr); + Scan; -- past << + Set_Identifier (Label_Node, P_Identifier); + T_Greater_Greater; + Append_Elmt (Label_Node, Label_List); + return Label_Node; + end P_Label; + + ------------------------------- + -- 5.1 Statement Identifier -- + ------------------------------- + + -- Statement label is parsed by P_Label (5.1) + + -- Loop label is parsed by P_Loop_Statement (5.5), P_For_Statement (5.5) + -- or P_While_Statement (5.5) + + -- Block label is parsed by P_Begin_Statement (5.6) or + -- P_Declare_Statement (5.6) + + ------------------------------- + -- 5.2 Assignment Statement -- + ------------------------------- + + -- ASSIGNMENT_STATEMENT ::= + -- variable_NAME := EXPRESSION; + + -- Error recovery: can raise Error_Resync + + function P_Assignment_Statement (LHS : Node_Id) return Node_Id is + Assign_Node : Node_Id; + + begin + Assign_Node := New_Node (N_Assignment_Statement, Prev_Token_Ptr); + Set_Name (Assign_Node, LHS); + Set_Expression (Assign_Node, P_Expression_No_Right_Paren); + TF_Semicolon; + return Assign_Node; + end P_Assignment_Statement; + + ----------------------- + -- 5.3 If Statement -- + ----------------------- + + -- IF_STATEMENT ::= + -- if CONDITION then + -- SEQUENCE_OF_STATEMENTS + -- {elsif CONDITION then + -- SEQUENCE_OF_STATEMENTS} + -- [else + -- SEQUENCE_OF_STATEMENTS] + -- end if; + + -- The caller has checked that the initial token is IF (or in the error + -- case of a mysterious THEN, the initial token may simply be THEN, in + -- which case, no condition (or IF) was scanned). + + -- Error recovery: can raise Error_Resync + + function P_If_Statement return Node_Id is + If_Node : Node_Id; + Elsif_Node : Node_Id; + Loc : Source_Ptr; + + procedure Add_Elsif_Part; + -- An internal procedure used to scan out a single ELSIF part. On entry + -- the ELSIF (or an ELSE which has been determined should be ELSIF) is + -- scanned out and is in Prev_Token. + + procedure Check_If_Column; + -- An internal procedure used to check that THEN, ELSE ELSE, or ELSIF + -- appear in the right place if column checking is enabled (i.e. if + -- they are the first token on the line, then they must appear in + -- the same column as the opening IF). + + procedure Check_Then_Column; + -- This procedure carries out the style checks for a THEN token + -- Note that the caller has set Loc to the Source_Ptr value for + -- the previous IF or ELSIF token. These checks apply only to a + -- THEN at the start of a line. + + function Else_Should_Be_Elsif return Boolean; + -- An internal routine used to do a special error recovery check when + -- an ELSE is encountered. It determines if the ELSE should be treated + -- as an ELSIF. A positive decision (TRUE returned, is made if the ELSE + -- is followed by a sequence of tokens, starting on the same line as + -- the ELSE, which are not expression terminators, followed by a THEN. + -- On entry, the ELSE has been scanned out. + + procedure Add_Elsif_Part is + begin + if No (Elsif_Parts (If_Node)) then + Set_Elsif_Parts (If_Node, New_List); + end if; + + Elsif_Node := New_Node (N_Elsif_Part, Prev_Token_Ptr); + Loc := Prev_Token_Ptr; + Set_Condition (Elsif_Node, P_Condition); + Check_Then_Column; + Then_Scan; + Set_Then_Statements + (Elsif_Node, P_Sequence_Of_Statements (SS_Eftm_Eltm_Sreq)); + Append (Elsif_Node, Elsif_Parts (If_Node)); + end Add_Elsif_Part; + + procedure Check_If_Column is + begin + if Style.RM_Column_Check and then Token_Is_At_Start_Of_Line + and then Start_Column /= Scope.Table (Scope.Last).Ecol + then + Error_Msg_Col := Scope.Table (Scope.Last).Ecol; + Error_Msg_SC ("(style) this token should be@"); + end if; + end Check_If_Column; + + procedure Check_Then_Column is + begin + if Token_Is_At_Start_Of_Line and then Token = Tok_Then then + Check_If_Column; + if Style_Check then Style.Check_Then (Loc); end if; + end if; + end Check_Then_Column; + + function Else_Should_Be_Elsif return Boolean is + Scan_State : Saved_Scan_State; + + begin + if Token_Is_At_Start_Of_Line then + return False; + + else + Save_Scan_State (Scan_State); + + loop + if Token in Token_Class_Eterm then + Restore_Scan_State (Scan_State); + return False; + else + Scan; -- past non-expression terminating token + + if Token = Tok_Then then + Restore_Scan_State (Scan_State); + return True; + end if; + end if; + end loop; + end if; + end Else_Should_Be_Elsif; + + -- Start of processing for P_If_Statement + + begin + If_Node := New_Node (N_If_Statement, Token_Ptr); + + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_If; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Labl := Error; + Scope.Table (Scope.Last).Node := If_Node; + + if Token = Tok_If then + Loc := Token_Ptr; + Scan; -- past IF + Set_Condition (If_Node, P_Condition); + + -- Deal with misuse of IF expression => used instead + -- of WHEN expression => + + if Token = Tok_Arrow then + Error_Msg_SC ("THEN expected"); + Scan; -- past the arrow + Pop_Scope_Stack; -- remove unneeded entry + raise Error_Resync; + end if; + + Check_Then_Column; + + else + Error_Msg_SC ("no IF for this THEN"); + Set_Condition (If_Node, Error); + end if; + + Then_Scan; + + Set_Then_Statements + (If_Node, P_Sequence_Of_Statements (SS_Eftm_Eltm_Sreq)); + + -- This loop scans out else and elsif parts + + loop + if Token = Tok_Elsif then + Check_If_Column; + + if Present (Else_Statements (If_Node)) then + Error_Msg_SP ("ELSIF cannot appear after ELSE"); + end if; + + Scan; -- past ELSIF + Add_Elsif_Part; + + elsif Token = Tok_Else then + Check_If_Column; + Scan; -- past ELSE + + if Else_Should_Be_Elsif then + Error_Msg_SP ("ELSE should be ELSIF"); + Add_Elsif_Part; + + else + -- Here we have an else that really is an else + + if Present (Else_Statements (If_Node)) then + Error_Msg_SP ("Only one ELSE part allowed"); + Append_List + (P_Sequence_Of_Statements (SS_Eftm_Eltm_Sreq), + Else_Statements (If_Node)); + else + Set_Else_Statements + (If_Node, P_Sequence_Of_Statements (SS_Eftm_Eltm_Sreq)); + end if; + end if; + + -- If anything other than ELSE or ELSIF, exit the loop. The token + -- had better be END (and in fact it had better be END IF), but + -- we will let End_Statements take care of checking that. + + else + exit; + end if; + end loop; + + End_Statements; + return If_Node; + + end P_If_Statement; + + -------------------- + -- 5.3 Condition -- + -------------------- + + -- CONDITION ::= boolean_EXPRESSION + + function P_Condition return Node_Id is + Cond : Node_Id; + + begin + Cond := P_Expression_No_Right_Paren; + + -- It is never possible for := to follow a condition, so if we get + -- a := we assume it is a mistyped equality. Note that we do not try + -- to reconstruct the tree correctly in this case, but we do at least + -- give an accurate error message. + + while Token = Tok_Colon_Equal loop + Error_Msg_SC (""":="" should be ""="""); + Scan; -- past junk := + Discard_Junk_Node (P_Expression_No_Right_Paren); + end loop; + + return Cond; + end P_Condition; + + ------------------------- + -- 5.4 Case Statement -- + ------------------------- + + -- CASE_STATEMENT ::= + -- case EXPRESSION is + -- CASE_STATEMENT_ALTERNATIVE + -- {CASE_STATEMENT_ALTERNATIVE} + -- end case; + + -- The caller has checked that the first token is CASE + + -- Can raise Error_Resync + + function P_Case_Statement return Node_Id is + Case_Node : Node_Id; + Alternatives_List : List_Id; + First_When_Loc : Source_Ptr; + + begin + Case_Node := New_Node (N_Case_Statement, Token_Ptr); + + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Case; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Labl := Error; + Scope.Table (Scope.Last).Node := Case_Node; + + Scan; -- past CASE + Set_Expression (Case_Node, P_Expression_No_Right_Paren); + TF_Is; + + -- Prepare to parse case statement alternatives + + Alternatives_List := New_List; + P_Pragmas_Opt (Alternatives_List); + First_When_Loc := Token_Ptr; + + -- Loop through case statement alternatives + + loop + -- If we have a WHEN or OTHERS, then that's fine keep going. Note + -- that it is a semantic check to ensure the proper use of OTHERS + + if Token = Tok_When or else Token = Tok_Others then + Append (P_Case_Statement_Alternative, Alternatives_List); + + -- If we have an END, then probably we are at the end of the case + -- but we only exit if Check_End thinks the END was reasonable. + + elsif Token = Tok_End then + exit when Check_End; + + -- Here if token is other than WHEN, OTHERS or END. We definitely + -- have an error, but the question is whether or not to get out of + -- the case statement. We don't want to get out early, or we will + -- get a slew of junk error messages for subsequent when tokens. + + -- If the token is not at the start of the line, or if it is indented + -- with respect to the current case statement, then the best guess is + -- that we are still supposed to be inside the case statement. We + -- complain about the missing WHEN, and discard the junk statements. + + elsif not Token_Is_At_Start_Of_Line + or else Start_Column > Scope.Table (Scope.Last).Ecol + then + Error_Msg_BC ("WHEN (case statement alternative) expected"); + + -- Here is a possibility for infinite looping if we don't make + -- progress. So try to process statements, otherwise exit + + declare + Error_Ptr : constant Source_Ptr := Scan_Ptr; + begin + Discard_Junk_List (P_Sequence_Of_Statements (SS_Whtm)); + exit when Scan_Ptr = Error_Ptr and then Check_End; + end; + + -- Here we have a junk token at the start of the line and it is + -- not indented. If Check_End thinks there is a missing END, then + -- we will get out of the case, otherwise we keep going. + + else + exit when Check_End; + end if; + end loop; + + -- Make sure we have at least one alternative + + if No (First_Non_Pragma (Alternatives_List)) then + Error_Msg + ("WHEN expected, must have at least one alternative in case", + First_When_Loc); + return Error; + + else + Set_Alternatives (Case_Node, Alternatives_List); + return Case_Node; + end if; + end P_Case_Statement; + + ------------------------------------- + -- 5.4 Case Statement Alternative -- + ------------------------------------- + + -- CASE_STATEMENT_ALTERNATIVE ::= + -- when DISCRETE_CHOICE_LIST => + -- SEQUENCE_OF_STATEMENTS + + -- The caller has checked that the initial token is WHEN or OTHERS + -- Error recovery: can raise Error_Resync + + function P_Case_Statement_Alternative return Node_Id is + Case_Alt_Node : Node_Id; + + begin + if Style_Check then Style.Check_Indentation; end if; + Case_Alt_Node := New_Node (N_Case_Statement_Alternative, Token_Ptr); + T_When; -- past WHEN (or give error in OTHERS case) + Set_Discrete_Choices (Case_Alt_Node, P_Discrete_Choice_List); + TF_Arrow; + Set_Statements (Case_Alt_Node, P_Sequence_Of_Statements (SS_Sreq_Whtm)); + return Case_Alt_Node; + end P_Case_Statement_Alternative; + + ------------------------- + -- 5.5 Loop Statement -- + ------------------------- + + -- LOOP_STATEMENT ::= + -- [LOOP_STATEMENT_IDENTIFIER:] + -- [ITERATION_SCHEME] loop + -- SEQUENCE_OF_STATEMENTS + -- end loop [loop_IDENTIFIER]; + + -- ITERATION_SCHEME ::= + -- while CONDITION + -- | for LOOP_PARAMETER_SPECIFICATION + + -- The parsing of loop statements is handled by one of three functions + -- P_Loop_Statement, P_For_Statement or P_While_Statement depending + -- on the initial keyword in the construct (excluding the identifier) + + -- P_Loop_Statement + + -- This function parses the case where no iteration scheme is present + + -- The caller has checked that the initial token is LOOP. The parameter + -- is the node identifiers for the loop label if any (or is set to Empty + -- if there is no loop label). + + -- Error recovery : cannot raise Error_Resync + + function P_Loop_Statement (Loop_Name : Node_Id := Empty) return Node_Id is + Loop_Node : Node_Id; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Labl := Loop_Name; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Etyp := E_Loop; + + Loop_Node := New_Node (N_Loop_Statement, Token_Ptr); + TF_Loop; + + if No (Loop_Name) then + Set_Has_Created_Identifier (Loop_Node, True); + Set_Identifier (Loop_Node, + Make_Identifier (Sloc (Loop_Node), Set_Loop_Block_Name ('L'))); + else + Set_Identifier (Loop_Node, Loop_Name); + end if; + + Append_Elmt (Loop_Node, Label_List); + + Set_Statements (Loop_Node, P_Sequence_Of_Statements (SS_Sreq)); + End_Statements (Loop_Node); + return Loop_Node; + end P_Loop_Statement; + + -- P_For_Statement + + -- This function parses a loop statement with a FOR iteration scheme + + -- The caller has checked that the initial token is FOR. The parameter + -- is the node identifier for the block label if any (or is set to Empty + -- if there is no block label). + + -- Note: the caller fills in the Identifier field if a label was present + + -- Error recovery: can raise Error_Resync + + function P_For_Statement (Loop_Name : Node_Id := Empty) return Node_Id is + Loop_Node : Node_Id; + Iter_Scheme_Node : Node_Id; + Loop_For_Flag : Boolean; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Labl := Loop_Name; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Etyp := E_Loop; + + Loop_For_Flag := (Prev_Token = Tok_Loop); + Scan; -- past FOR + Iter_Scheme_Node := New_Node (N_Iteration_Scheme, Token_Ptr); + Set_Loop_Parameter_Specification + (Iter_Scheme_Node, P_Loop_Parameter_Specification); + + -- The following is a special test so that a miswritten for loop such + -- as "loop for I in 1..10;" is handled nicely, without making an extra + -- entry in the scope stack. We don't bother to actually fix up the + -- tree in this case since it's not worth the effort. Instead we just + -- eat up the loop junk, leaving the entry for what now looks like an + -- unmodified loop intact. + + if Loop_For_Flag and then Token = Tok_Semicolon then + Error_Msg_SC ("LOOP belongs here, not before FOR"); + Pop_Scope_Stack; + return Error; + + -- Normal case + + else + Loop_Node := New_Node (N_Loop_Statement, Token_Ptr); + TF_Loop; + Set_Statements (Loop_Node, P_Sequence_Of_Statements (SS_Sreq)); + End_Statements (Loop_Node); + Set_Iteration_Scheme (Loop_Node, Iter_Scheme_Node); + + if No (Loop_Name) then + Set_Has_Created_Identifier (Loop_Node, True); + Set_Identifier (Loop_Node, + Make_Identifier (Sloc (Loop_Node), Set_Loop_Block_Name ('L'))); + else + Set_Identifier (Loop_Node, Loop_Name); + end if; + + Append_Elmt (Loop_Node, Label_List); + + return Loop_Node; + end if; + + end P_For_Statement; + + -- P_While_Statement + + -- This procedure scans a loop statement with a WHILE iteration scheme + + -- The caller has checked that the initial token is WHILE. The parameter + -- is the node identifier for the block label if any (or is set to Empty + -- if there is no block label). + + -- Error recovery: cannot raise Error_Resync + + function P_While_Statement (Loop_Name : Node_Id := Empty) return Node_Id is + Loop_Node : Node_Id; + Iter_Scheme_Node : Node_Id; + Loop_While_Flag : Boolean; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Labl := Loop_Name; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Etyp := E_Loop; + + Loop_While_Flag := (Prev_Token = Tok_Loop); + Iter_Scheme_Node := New_Node (N_Iteration_Scheme, Token_Ptr); + Scan; -- past WHILE + Set_Condition (Iter_Scheme_Node, P_Condition); + + -- The following is a special test so that a miswritten for loop such + -- as "loop while I > 10;" is handled nicely, without making an extra + -- entry in the scope stack. We don't bother to actually fix up the + -- tree in this case since it's not worth the effort. Instead we just + -- eat up the loop junk, leaving the entry for what now looks like an + -- unmodified loop intact. + + if Loop_While_Flag and then Token = Tok_Semicolon then + Error_Msg_SC ("LOOP belongs here, not before WHILE"); + Pop_Scope_Stack; + return Error; + + -- Normal case + + else + Loop_Node := New_Node (N_Loop_Statement, Token_Ptr); + TF_Loop; + Set_Statements (Loop_Node, P_Sequence_Of_Statements (SS_Sreq)); + End_Statements (Loop_Node); + Set_Iteration_Scheme (Loop_Node, Iter_Scheme_Node); + + if No (Loop_Name) then + Set_Has_Created_Identifier (Loop_Node, True); + Set_Identifier (Loop_Node, + Make_Identifier (Sloc (Loop_Node), Set_Loop_Block_Name ('L'))); + else + Set_Identifier (Loop_Node, Loop_Name); + end if; + + Append_Elmt (Loop_Node, Label_List); + + return Loop_Node; + end if; + + end P_While_Statement; + + --------------------------------------- + -- 5.5 Loop Parameter Specification -- + --------------------------------------- + + -- LOOP_PARAMETER_SPECIFICATION ::= + -- DEFINING_IDENTIFIER in [reverse] DISCRETE_SUBTYPE_DEFINITION + + -- Error recovery: cannot raise Error_Resync + + function P_Loop_Parameter_Specification return Node_Id is + Loop_Param_Specification_Node : Node_Id; + + ID_Node : Node_Id; + Scan_State : Saved_Scan_State; + + begin + Loop_Param_Specification_Node := + New_Node (N_Loop_Parameter_Specification, Token_Ptr); + + Save_Scan_State (Scan_State); + ID_Node := P_Defining_Identifier; + Set_Defining_Identifier (Loop_Param_Specification_Node, ID_Node); + + if Token = Tok_Left_Paren then + Error_Msg_SC ("subscripted loop parameter not allowed"); + Restore_Scan_State (Scan_State); + Discard_Junk_Node (P_Name); + + elsif Token = Tok_Dot then + Error_Msg_SC ("selected loop parameter not allowed"); + Restore_Scan_State (Scan_State); + Discard_Junk_Node (P_Name); + end if; + + T_In; + + if Token = Tok_Reverse then + Scan; -- past REVERSE + Set_Reverse_Present (Loop_Param_Specification_Node, True); + end if; + + Set_Discrete_Subtype_Definition + (Loop_Param_Specification_Node, P_Discrete_Subtype_Definition); + return Loop_Param_Specification_Node; + + exception + when Error_Resync => + return Error; + end P_Loop_Parameter_Specification; + + -------------------------- + -- 5.6 Block Statement -- + -------------------------- + + -- BLOCK_STATEMENT ::= + -- [block_STATEMENT_IDENTIFIER:] + -- [declare + -- DECLARATIVE_PART] + -- begin + -- HANDLED_SEQUENCE_OF_STATEMENTS + -- end [block_IDENTIFIER]; + + -- The parsing of block statements is handled by one of the two functions + -- P_Declare_Statement or P_Begin_Statement depending on whether or not + -- a declare section is present + + -- P_Declare_Statement + + -- This function parses a block statement with DECLARE present + + -- The caller has checked that the initial token is DECLARE. + + -- Error recovery: cannot raise Error_Resync + + function P_Declare_Statement + (Block_Name : Node_Id := Empty) + return Node_Id + is + Block_Node : Node_Id; + + begin + Block_Node := New_Node (N_Block_Statement, Token_Ptr); + + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Lreq := Present (Block_Name); + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Labl := Block_Name; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + + Scan; -- past DECLARE + + if No (Block_Name) then + Set_Has_Created_Identifier (Block_Node, True); + Set_Identifier (Block_Node, + Make_Identifier (Sloc (Block_Node), Set_Loop_Block_Name ('B'))); + else + Set_Identifier (Block_Node, Block_Name); + end if; + + Append_Elmt (Block_Node, Label_List); + Parse_Decls_Begin_End (Block_Node); + return Block_Node; + end P_Declare_Statement; + + -- P_Begin_Statement + + -- This function parses a block statement with no DECLARE present + + -- The caller has checked that the initial token is BEGIN + + -- Error recovery: cannot raise Error_Resync + + function P_Begin_Statement + (Block_Name : Node_Id := Empty) + return Node_Id + is + Block_Node : Node_Id; + + begin + Block_Node := New_Node (N_Block_Statement, Token_Ptr); + + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Lreq := Present (Block_Name); + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Labl := Block_Name; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + + if No (Block_Name) then + Set_Has_Created_Identifier (Block_Node, True); + Set_Identifier (Block_Node, + Make_Identifier (Sloc (Block_Node), Set_Loop_Block_Name ('B'))); + else + Set_Identifier (Block_Node, Block_Name); + end if; + + Append_Elmt (Block_Node, Label_List); + + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scan; -- past BEGIN + Set_Handled_Statement_Sequence + (Block_Node, P_Handled_Sequence_Of_Statements); + End_Statements (Handled_Statement_Sequence (Block_Node)); + return Block_Node; + end P_Begin_Statement; + + ------------------------- + -- 5.7 Exit Statement -- + ------------------------- + + -- EXIT_STATEMENT ::= + -- exit [loop_NAME] [when CONDITION]; + + -- The caller has checked that the initial token is EXIT + + -- Error recovery: can raise Error_Resync + + function P_Exit_Statement return Node_Id is + Exit_Node : Node_Id; + + function Missing_Semicolon_On_Exit return Boolean; + -- This function deals with the following specialized situation + -- + -- when 'x' => + -- exit [identifier] + -- when 'y' => + -- + -- This looks like a messed up EXIT WHEN, when in fact the problem + -- is a missing semicolon. It is called with Token pointing to the + -- WHEN token, and returns True if a semicolon is missing before + -- the WHEN as in the above example. + + function Missing_Semicolon_On_Exit return Boolean is + State : Saved_Scan_State; + + begin + if not Token_Is_At_Start_Of_Line then + return False; + + elsif Scope.Table (Scope.Last).Etyp /= E_Case then + return False; + + else + Save_Scan_State (State); + Scan; -- past WHEN + Scan; -- past token after WHEN + + if Token = Tok_Arrow then + Restore_Scan_State (State); + return True; + else + Restore_Scan_State (State); + return False; + end if; + end if; + end Missing_Semicolon_On_Exit; + + -- Start of processing for P_Exit_Statement + + begin + Exit_Node := New_Node (N_Exit_Statement, Token_Ptr); + Scan; -- past EXIT + + if Token = Tok_Identifier then + Set_Name (Exit_Node, P_Qualified_Simple_Name); + + elsif Style_Check then + -- This EXIT has no name, so check that + -- the innermost loop is unnamed too. + + Check_No_Exit_Name : + for J in reverse 1 .. Scope.Last loop + if Scope.Table (J).Etyp = E_Loop then + if Present (Scope.Table (J).Labl) then + + -- Innermost loop in fact had a name, style check fails + + Style.No_Exit_Name (Scope.Table (J).Labl); + end if; + + exit Check_No_Exit_Name; + end if; + end loop Check_No_Exit_Name; + end if; + + if Token = Tok_When and then not Missing_Semicolon_On_Exit then + Scan; -- past WHEN + Set_Condition (Exit_Node, P_Condition); + + -- Allow IF instead of WHEN, giving error message + + elsif Token = Tok_If then + T_When; + Scan; -- past IF used in place of WHEN + Set_Condition (Exit_Node, P_Expression_No_Right_Paren); + end if; + + TF_Semicolon; + return Exit_Node; + end P_Exit_Statement; + + ------------------------- + -- 5.8 Goto Statement -- + ------------------------- + + -- GOTO_STATEMENT ::= goto label_NAME; + + -- The caller has checked that the initial token is GOTO (or TO in the + -- error case where GO and TO were incorrectly separated). + + -- Error recovery: can raise Error_Resync + + function P_Goto_Statement return Node_Id is + Goto_Node : Node_Id; + + begin + Goto_Node := New_Node (N_Goto_Statement, Token_Ptr); + Scan; -- past GOTO (or TO) + Set_Name (Goto_Node, P_Qualified_Simple_Name_Resync); + No_Constraint; + TF_Semicolon; + return Goto_Node; + end P_Goto_Statement; + + --------------------------- + -- Parse_Decls_Begin_End -- + --------------------------- + + -- This function parses the construct: + + -- DECLARATIVE_PART + -- begin + -- HANDLED_SEQUENCE_OF_STATEMENTS + -- end [NAME]; + + -- The caller has built the scope stack entry, and created the node to + -- whose Declarations and Handled_Statement_Sequence fields are to be + -- set. On return these fields are filled in (except in the case of a + -- task body, where the handled statement sequence is optional, and may + -- thus be Empty), and the scan is positioned past the End sequence. + + -- If the BEGIN is missing, then the parent node is used to help construct + -- an appropriate missing BEGIN message. Possibilities for the parent are: + + -- N_Block_Statement declare block + -- N_Entry_Body entry body + -- N_Package_Body package body (begin part optional) + -- N_Subprogram_Body procedure or function body + -- N_Task_Body task body + + -- Note: in the case of a block statement, there is definitely a DECLARE + -- present (because a Begin statement without a DECLARE is handled by the + -- P_Begin_Statement procedure, which does not call Parse_Decls_Begin_End. + + -- Error recovery: cannot raise Error_Resync + + procedure Parse_Decls_Begin_End (Parent : Node_Id) is + Body_Decl : Node_Id; + Body_Sloc : Source_Ptr; + Decls : List_Id; + Decl : Node_Id; + Parent_Nkind : Node_Kind; + Spec_Node : Node_Id; + HSS : Node_Id; + + procedure Missing_Begin (Msg : String); + -- Called to post a missing begin message. In the normal case this is + -- posted at the start of the current token. A special case arises when + -- P_Declarative_Items has previously found a missing begin, in which + -- case we replace the original error message. + + procedure Set_Null_HSS (Parent : Node_Id); + -- Construct an empty handled statement sequence and install in Parent + -- Leaves HSS set to reference the newly constructed statement sequence. + + ------------------- + -- Missing_Begin -- + ------------------- + + procedure Missing_Begin (Msg : String) is + begin + if Missing_Begin_Msg = No_Error_Msg then + Error_Msg_BC (Msg); + else + Change_Error_Text (Missing_Begin_Msg, Msg); + + -- Purge any messages issued after than, since a missing begin + -- can cause a lot of havoc, and it is better not to dump these + -- cascaded messages on the user. + + Purge_Messages (Get_Location (Missing_Begin_Msg), Prev_Token_Ptr); + end if; + end Missing_Begin; + + ------------------ + -- Set_Null_HSS -- + ------------------ + + procedure Set_Null_HSS (Parent : Node_Id) is + Null_Stm : Node_Id; + + begin + Null_Stm := + Make_Null_Statement (Token_Ptr); + Set_Comes_From_Source (Null_Stm, False); + + HSS := + Make_Handled_Sequence_Of_Statements (Token_Ptr, + Statements => New_List (Null_Stm)); + Set_Comes_From_Source (HSS, False); + + Set_Handled_Statement_Sequence (Parent, HSS); + end Set_Null_HSS; + + -- Start of processing for Parse_Decls_Begin_End + + begin + Decls := P_Declarative_Part; + + -- Check for misplacement of later vs basic declarations in Ada 83 + + if Ada_83 then + Decl := First (Decls); + + -- Loop through sequence of basic declarative items + + Outer : while Present (Decl) loop + if Nkind (Decl) /= N_Subprogram_Body + and then Nkind (Decl) /= N_Package_Body + and then Nkind (Decl) /= N_Task_Body + and then Nkind (Decl) not in N_Body_Stub + then + Next (Decl); + + -- Once a body is encountered, we only allow later declarative + -- items. The inner loop checks the rest of the list. + + else + Body_Sloc := Sloc (Decl); + + Inner : while Present (Decl) loop + if Nkind (Decl) not in N_Later_Decl_Item + and then Nkind (Decl) /= N_Pragma + then + if Ada_83 then + Error_Msg_Sloc := Body_Sloc; + Error_Msg_N + ("(Ada 83) decl cannot appear after body#", Decl); + end if; + end if; + + Next (Decl); + end loop Inner; + end if; + end loop Outer; + end if; + + -- Here is where we deal with the case of IS used instead of semicolon. + -- Specifically, if the last declaration in the declarative part is a + -- subprogram body still marked as having a bad IS, then this is where + -- we decide that the IS should really have been a semicolon and that + -- the body should have been a declaration. Note that if the bad IS + -- had turned out to be OK (i.e. a decent begin/end was found for it), + -- then the Bad_Is_Detected flag would have been reset by now. + + Body_Decl := Last (Decls); + + if Present (Body_Decl) + and then Nkind (Body_Decl) = N_Subprogram_Body + and then Bad_Is_Detected (Body_Decl) + then + -- OK, we have the case of a bad IS, so we need to fix up the tree. + -- What we have now is a subprogram body with attached declarations + -- and a possible statement sequence. + + -- First step is to take the declarations that were part of the bogus + -- subprogram body and append them to the outer declaration chain. + -- In other words we append them past the body (which we will later + -- convert into a declaration). + + Append_List (Declarations (Body_Decl), Decls); + + -- Now take the handled statement sequence of the bogus body and + -- set it as the statement sequence for the outer construct. Note + -- that it may be empty (we specially allowed a missing BEGIN for + -- a subprogram body marked as having a bad IS -- see below). + + Set_Handled_Statement_Sequence (Parent, + Handled_Statement_Sequence (Body_Decl)); + + -- Next step is to convert the old body node to a declaration node + + Spec_Node := Specification (Body_Decl); + Change_Node (Body_Decl, N_Subprogram_Declaration); + Set_Specification (Body_Decl, Spec_Node); + + -- Final step is to put the declarations for the parent where + -- they belong, and then fall through the IF to scan out the + -- END statements. + + Set_Declarations (Parent, Decls); + + -- This is the normal case (i.e. any case except the bad IS case) + -- If we have a BEGIN, then scan out the sequence of statements, and + -- also reset the expected column for the END to match the BEGIN. + + else + Set_Declarations (Parent, Decls); + + if Token = Tok_Begin then + if Style_Check then Style.Check_Indentation; end if; + + Error_Msg_Col := Scope.Table (Scope.Last).Ecol; + + if Style.RM_Column_Check + and then Token_Is_At_Start_Of_Line + and then Start_Column /= Error_Msg_Col + then + Error_Msg_SC ("(style) BEGIN in wrong column, should be@"); + + else + Scope.Table (Scope.Last).Ecol := Start_Column; + end if; + + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scan; -- past BEGIN + Set_Handled_Statement_Sequence (Parent, + P_Handled_Sequence_Of_Statements); + + -- No BEGIN present + + else + Parent_Nkind := Nkind (Parent); + + -- A special check for the missing IS case. If we have a + -- subprogram body that was marked as having a suspicious + -- IS, and the current token is END, then we simply confirm + -- the suspicion, and do not require a BEGIN to be present + + if Parent_Nkind = N_Subprogram_Body + and then Token = Tok_End + and then Scope.Table (Scope.Last).Etyp = E_Suspicious_Is + then + Scope.Table (Scope.Last).Etyp := E_Bad_Is; + + -- Otherwise BEGIN is not required for a package body, so we + -- don't mind if it is missing, but we do construct a dummy + -- one (so that we have somewhere to set End_Label). + + -- However if we have something other than a BEGIN which + -- looks like it might be statements, then we signal a missing + -- BEGIN for these cases as well. We define "something which + -- looks like it might be statements" as a token other than + -- END, EOF, or a token which starts declarations. + + elsif Parent_Nkind = N_Package_Body + and then (Token = Tok_End + or else Token = Tok_EOF + or else Token in Token_Class_Declk) + then + Set_Null_HSS (Parent); + + -- These are cases in which a BEGIN is required and not present + + else + Set_Null_HSS (Parent); + + -- Prepare to issue error message + + Error_Msg_Sloc := Scope.Table (Scope.Last).Sloc; + Error_Msg_Node_1 := Scope.Table (Scope.Last).Labl; + + -- Now issue appropriate message + + if Parent_Nkind = N_Block_Statement then + Missing_Begin ("missing BEGIN for DECLARE#!"); + + elsif Parent_Nkind = N_Entry_Body then + Missing_Begin ("missing BEGIN for ENTRY#!"); + + elsif Parent_Nkind = N_Subprogram_Body then + if Nkind (Specification (Parent)) + = N_Function_Specification + then + Missing_Begin ("missing BEGIN for function&#!"); + else + Missing_Begin ("missing BEGIN for procedure&#!"); + end if; + + -- The case for package body arises only when + -- we have possible statement junk present. + + elsif Parent_Nkind = N_Package_Body then + Missing_Begin ("missing BEGIN for package body&#!"); + + else + pragma Assert (Parent_Nkind = N_Task_Body); + Missing_Begin ("missing BEGIN for task body&#!"); + end if; + + -- Here we pick up the statements after the BEGIN that + -- should have been present but was not. We don't insist + -- on statements being present if P_Declarative_Part had + -- already found a missing BEGIN, since it might have + -- swallowed a lone statement into the declarative part. + + if Missing_Begin_Msg /= No_Error_Msg + and then Token = Tok_End + then + null; + else + Set_Handled_Statement_Sequence (Parent, + P_Handled_Sequence_Of_Statements); + end if; + end if; + end if; + end if; + + -- Here with declarations and handled statement sequence scanned + + if Present (Handled_Statement_Sequence (Parent)) then + End_Statements (Handled_Statement_Sequence (Parent)); + else + End_Statements; + end if; + + -- We know that End_Statements removed an entry from the scope stack + -- (because it is required to do so under all circumstances). We can + -- therefore reference the entry it removed one past the stack top. + -- What we are interested in is whether it was a case of a bad IS. + + if Scope.Table (Scope.Last + 1).Etyp = E_Bad_Is then + Error_Msg ("IS should be "";""", Scope.Table (Scope.Last + 1).S_Is); + Set_Bad_Is_Detected (Parent, True); + end if; + + end Parse_Decls_Begin_End; + + ------------------------- + -- Set_Loop_Block_Name -- + ------------------------- + + function Set_Loop_Block_Name (L : Character) return Name_Id is + begin + Name_Buffer (1) := L; + Name_Buffer (2) := '_'; + Name_Len := 2; + Loop_Block_Count := Loop_Block_Count + 1; + Add_Nat_To_Name_Buffer (Loop_Block_Count); + return Name_Find; + end Set_Loop_Block_Name; + + --------------- + -- Then_Scan -- + --------------- + + procedure Then_Scan is + begin + TF_Then; + + while Token = Tok_Then loop + Error_Msg_SC ("redundant THEN"); + TF_Then; + end loop; + + if Token = Tok_And or else Token = Tok_Or then + Error_Msg_SC ("unexpected logical operator"); + Scan; + + if (Prev_Token = Tok_And and then Token = Tok_Then) + or else + (Prev_Token = Tok_Or and then Token = Tok_Else) + then + Scan; + end if; + + Discard_Junk_Node (P_Expression); + end if; + + if Token = Tok_Then then + Scan; + end if; + end Then_Scan; + +end Ch5; diff --git a/gcc/ada/par-ch6.adb b/gcc/ada/par-ch6.adb new file mode 100644 index 00000000000..d5d1d3daaa1 --- /dev/null +++ b/gcc/ada/par-ch6.adb @@ -0,0 +1,1165 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 6 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.81 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +with Sinfo.CN; use Sinfo.CN; + +separate (Par) +package body Ch6 is + + -- Local subprograms, used only in this chapter + + function P_Defining_Designator return Node_Id; + function P_Defining_Operator_Symbol return Node_Id; + + procedure Check_Junk_Semicolon_Before_Return; + -- Check for common error of junk semicolon before RETURN keyword of + -- function specification. If present, skip over it with appropriate + -- error message, leaving Scan_Ptr pointing to the RETURN after. This + -- routine also deals with a possibly misspelled version of Return. + + ---------------------------------------- + -- Check_Junk_Semicolon_Before_Return -- + ---------------------------------------- + + procedure Check_Junk_Semicolon_Before_Return is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Semicolon then + Save_Scan_State (Scan_State); + Scan; -- past the semicolon + + if Token = Tok_Return then + Restore_Scan_State (Scan_State); + Error_Msg_SC ("Unexpected semicolon ignored"); + Scan; -- rescan past junk semicolon + + else + Restore_Scan_State (Scan_State); + end if; + + elsif Bad_Spelling_Of (Tok_Return) then + null; + end if; + end Check_Junk_Semicolon_Before_Return; + + ----------------------------------------------------- + -- 6.1 Subprogram (Also 6.3, 8.5.4, 10.1.3, 12.3) -- + ----------------------------------------------------- + + -- This routine scans out a subprogram declaration, subprogram body, + -- subprogram renaming declaration or subprogram generic instantiation. + + -- SUBPROGRAM_DECLARATION ::= SUBPROGRAM_SPECIFICATION; + + -- ABSTRACT_SUBPROGRAM_DECLARATION ::= + -- SUBPROGRAM_SPECIFICATION is abstract; + + -- SUBPROGRAM_SPECIFICATION ::= + -- procedure DEFINING_PROGRAM_UNIT_NAME PARAMETER_PROFILE + -- | function DEFINING_DESIGNATOR PARAMETER_AND_RESULT_PROFILE + + -- PARAMETER_PROFILE ::= [FORMAL_PART] + + -- PARAMETER_AND_RESULT_PROFILE ::= [FORMAL_PART] return SUBTYPE_MARK + + -- SUBPROGRAM_BODY ::= + -- SUBPROGRAM_SPECIFICATION is + -- DECLARATIVE_PART + -- begin + -- HANDLED_SEQUENCE_OF_STATEMENTS + -- end [DESIGNATOR]; + + -- SUBPROGRAM_RENAMING_DECLARATION ::= + -- SUBPROGRAM_SPECIFICATION renames callable_entity_NAME; + + -- SUBPROGRAM_BODY_STUB ::= + -- SUBPROGRAM_SPECIFICATION is separate; + + -- GENERIC_INSTANTIATION ::= + -- procedure DEFINING_PROGRAM_UNIT_NAME is + -- new generic_procedure_NAME [GENERIC_ACTUAL_PART]; + -- | function DEFINING_DESIGNATOR is + -- new generic_function_NAME [GENERIC_ACTUAL_PART]; + + -- The value in Pf_Flags indicates which of these possible declarations + -- is acceptable to the caller: + + -- Pf_Flags.Decl Set if declaration OK + -- Pf_Flags.Gins Set if generic instantiation OK + -- Pf_Flags.Pbod Set if proper body OK + -- Pf_Flags.Rnam Set if renaming declaration OK + -- Pf_Flags.Stub Set if body stub OK + + -- If an inappropriate form is encountered, it is scanned out but an + -- error message indicating that it is appearing in an inappropriate + -- context is issued. The only possible values for Pf_Flags are those + -- defined as constants in the Par package. + + -- The caller has checked that the initial token is FUNCTION or PROCEDURE + + -- Error recovery: cannot raise Error_Resync + + function P_Subprogram (Pf_Flags : Pf_Rec) return Node_Id is + Specification_Node : Node_Id; + Name_Node : Node_Id; + Fpart_List : List_Id; + Fpart_Sloc : Source_Ptr; + Return_Node : Node_Id; + Inst_Node : Node_Id; + Body_Node : Node_Id; + Decl_Node : Node_Id; + Rename_Node : Node_Id; + Absdec_Node : Node_Id; + Stub_Node : Node_Id; + Fproc_Sloc : Source_Ptr; + Func : Boolean; + Scan_State : Saved_Scan_State; + + begin + -- Set up scope stack entry. Note that the Labl field will be set later + + SIS_Entry_Active := False; + SIS_Missing_Semicolon_Message := No_Error_Msg; + Push_Scope_Stack; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Lreq := False; + + Func := (Token = Tok_Function); + Fproc_Sloc := Token_Ptr; + Scan; -- past FUNCTION or PROCEDURE + Ignore (Tok_Type); + Ignore (Tok_Body); + + if Func then + Name_Node := P_Defining_Designator; + + if Nkind (Name_Node) = N_Defining_Operator_Symbol + and then Scope.Last = 1 + then + Error_Msg_SP ("operator symbol not allowed at library level"); + Name_Node := New_Entity (N_Defining_Identifier, Sloc (Name_Node)); + + -- Set name from file name, we need some junk name, and that's + -- as good as anything. This is only approximate, since we do + -- not do anything with non-standard name translations. + + Get_Name_String (File_Name (Current_Source_File)); + + for J in 1 .. Name_Len loop + if Name_Buffer (J) = '.' then + Name_Len := J - 1; + exit; + end if; + end loop; + + Set_Chars (Name_Node, Name_Find); + Set_Error_Posted (Name_Node); + end if; + + else + Name_Node := P_Defining_Program_Unit_Name; + end if; + + Scope.Table (Scope.Last).Labl := Name_Node; + + if Token = Tok_Colon then + Error_Msg_SC ("redundant colon ignored"); + Scan; -- past colon + end if; + + -- Deal with generic instantiation, the one case in which we do not + -- have a subprogram specification as part of whatever we are parsing + + if Token = Tok_Is then + Save_Scan_State (Scan_State); -- at the IS + T_Is; -- checks for redundant IS's + + if Token = Tok_New then + if not Pf_Flags.Gins then + Error_Msg_SC ("generic instantation not allowed here!"); + end if; + + Scan; -- past NEW + + if Func then + Inst_Node := New_Node (N_Function_Instantiation, Fproc_Sloc); + Set_Name (Inst_Node, P_Function_Name); + else + Inst_Node := New_Node (N_Procedure_Instantiation, Fproc_Sloc); + Set_Name (Inst_Node, P_Qualified_Simple_Name); + end if; + + Set_Defining_Unit_Name (Inst_Node, Name_Node); + Set_Generic_Associations (Inst_Node, P_Generic_Actual_Part_Opt); + TF_Semicolon; + Pop_Scope_Stack; -- Don't need scope stack entry in this case + return Inst_Node; + + else + Restore_Scan_State (Scan_State); -- to the IS + end if; + end if; + + -- If not a generic instantiation, then we definitely have a subprogram + -- specification (all possibilities at this stage include one here) + + Fpart_Sloc := Token_Ptr; + + Check_Misspelling_Of (Tok_Return); + + -- Scan formal part. First a special error check. If we have an + -- identifier here, then we have a definite error. If this identifier + -- is on the same line as the designator, then we assume it is the + -- first formal after a missing left parenthesis + + if Token = Tok_Identifier + and then not Token_Is_At_Start_Of_Line + then + T_Left_Paren; -- to generate message + Fpart_List := P_Formal_Part; + + -- Otherwise scan out an optional formal part in the usual manner + + else + Fpart_List := P_Parameter_Profile; + end if; + + -- We treat what we have as a function specification if FUNCTION was + -- used, or if a RETURN is present. This gives better error recovery + -- since later RETURN statements will be valid in either case. + + Check_Junk_Semicolon_Before_Return; + Return_Node := Error; + + if Token = Tok_Return then + if not Func then + Error_Msg ("PROCEDURE should be FUNCTION", Fproc_Sloc); + Func := True; + end if; + + Scan; -- past RETURN + Return_Node := P_Subtype_Mark; + No_Constraint; + + else + if Func then + Ignore (Tok_Right_Paren); + TF_Return; + end if; + end if; + + if Func then + Specification_Node := + New_Node (N_Function_Specification, Fproc_Sloc); + Set_Subtype_Mark (Specification_Node, Return_Node); + + else + Specification_Node := + New_Node (N_Procedure_Specification, Fproc_Sloc); + end if; + + Set_Defining_Unit_Name (Specification_Node, Name_Node); + Set_Parameter_Specifications (Specification_Node, Fpart_List); + + -- Error check: barriers not allowed on protected functions/procedures + + if Token = Tok_When then + if Func then + Error_Msg_SC ("barrier not allowed on function, only on entry"); + else + Error_Msg_SC ("barrier not allowed on procedure, only on entry"); + end if; + + Scan; -- past WHEN + Discard_Junk_Node (P_Expression); + end if; + + -- Deal with case of semicolon ending a subprogram declaration + + if Token = Tok_Semicolon then + if not Pf_Flags.Decl then + T_Is; + end if; + + Scan; -- past semicolon + + -- If semicolon is immediately followed by IS, then ignore the + -- semicolon, and go process the body. + + if Token = Tok_Is then + Error_Msg_SP ("unexpected semicolon ignored"); + T_Is; -- ignroe redundant IS's + goto Subprogram_Body; + + -- If BEGIN follows in an appropriate column, we immediately + -- commence the error action of assuming that the previous + -- subprogram declaration should have been a subprogram body, + -- i.e. that the terminating semicolon should have been IS. + + elsif Token = Tok_Begin + and then Start_Column >= Scope.Table (Scope.Last).Ecol + then + Error_Msg_SP (""";"" should be IS!"); + goto Subprogram_Body; + + else + goto Subprogram_Declaration; + end if; + + -- Case of not followed by semicolon + + else + -- Subprogram renaming declaration case + + Check_Misspelling_Of (Tok_Renames); + + if Token = Tok_Renames then + if not Pf_Flags.Rnam then + Error_Msg_SC ("renaming declaration not allowed here!"); + end if; + + Rename_Node := + New_Node (N_Subprogram_Renaming_Declaration, Token_Ptr); + Scan; -- past RENAMES + Set_Name (Rename_Node, P_Name); + Set_Specification (Rename_Node, Specification_Node); + TF_Semicolon; + Pop_Scope_Stack; + return Rename_Node; + + -- Case of IS following subprogram specification + + elsif Token = Tok_Is then + T_Is; -- ignore redundant Is's + + if Token_Name = Name_Abstract then + Check_95_Keyword (Tok_Abstract, Tok_Semicolon); + end if; + + -- Deal nicely with (now obsolete) use of <> in place of abstract + + if Token = Tok_Box then + Error_Msg_SC ("ABSTRACT expected"); + Token := Tok_Abstract; + end if; + + -- Abstract subprogram declaration case + + if Token = Tok_Abstract then + Absdec_Node := + New_Node (N_Abstract_Subprogram_Declaration, Token_Ptr); + Set_Specification (Absdec_Node, Specification_Node); + Pop_Scope_Stack; -- discard unneeded entry + Scan; -- past ABSTRACT + TF_Semicolon; + return Absdec_Node; + + -- Check for IS NEW with Formal_Part present and handle nicely + + elsif Token = Tok_New then + Error_Msg + ("formal part not allowed in instantiation", Fpart_Sloc); + Scan; -- past NEW + + if Func then + Inst_Node := New_Node (N_Function_Instantiation, Fproc_Sloc); + else + Inst_Node := + New_Node (N_Procedure_Instantiation, Fproc_Sloc); + end if; + + Set_Defining_Unit_Name (Inst_Node, Name_Node); + Set_Name (Inst_Node, P_Name); + Set_Generic_Associations (Inst_Node, P_Generic_Actual_Part_Opt); + TF_Semicolon; + Pop_Scope_Stack; -- Don't need scope stack entry in this case + return Inst_Node; + + else + goto Subprogram_Body; + end if; + + -- Here we have a missing IS or missing semicolon, we always guess + -- a missing semicolon, since we are pretty good at fixing up a + -- semicolon which should really be an IS + + else + Error_Msg_AP ("missing "";"""); + SIS_Missing_Semicolon_Message := Get_Msg_Id; + goto Subprogram_Declaration; + end if; + end if; + + -- Processing for subprogram body + + <<Subprogram_Body>> + if not Pf_Flags.Pbod then + Error_Msg_SP ("subprogram body not allowed here!"); + end if; + + -- Subprogram body stub case + + if Separate_Present then + if not Pf_Flags.Stub then + Error_Msg_SC ("body stub not allowed here!"); + end if; + + if Nkind (Name_Node) = N_Defining_Operator_Symbol then + Error_Msg + ("operator symbol cannot be used as subunit name", + Sloc (Name_Node)); + end if; + + Stub_Node := + New_Node (N_Subprogram_Body_Stub, Sloc (Specification_Node)); + Set_Specification (Stub_Node, Specification_Node); + Scan; -- past SEPARATE + Pop_Scope_Stack; + TF_Semicolon; + return Stub_Node; + + -- Subprogram body case + + else + -- Here is the test for a suspicious IS (i.e. one that looks + -- like it might more properly be a semicolon). See separate + -- section discussing use of IS instead of semicolon in + -- package Parse. + + if (Token in Token_Class_Declk + or else + Token = Tok_Identifier) + and then Start_Column <= Scope.Table (Scope.Last).Ecol + and then Scope.Last /= 1 + then + Scope.Table (Scope.Last).Etyp := E_Suspicious_Is; + Scope.Table (Scope.Last).S_Is := Prev_Token_Ptr; + end if; + + Body_Node := + New_Node (N_Subprogram_Body, Sloc (Specification_Node)); + Set_Specification (Body_Node, Specification_Node); + Parse_Decls_Begin_End (Body_Node); + return Body_Node; + end if; + + -- Processing for subprogram declaration + + <<Subprogram_Declaration>> + Decl_Node := + New_Node (N_Subprogram_Declaration, Sloc (Specification_Node)); + Set_Specification (Decl_Node, Specification_Node); + + -- If this is a context in which a subprogram body is permitted, + -- set active SIS entry in case (see section titled "Handling + -- Semicolon Used in Place of IS" in body of Parser package) + -- Note that SIS_Missing_Semicolon_Message is already set properly. + + if Pf_Flags.Pbod then + SIS_Labl := Scope.Table (Scope.Last).Labl; + SIS_Sloc := Scope.Table (Scope.Last).Sloc; + SIS_Ecol := Scope.Table (Scope.Last).Ecol; + SIS_Declaration_Node := Decl_Node; + SIS_Semicolon_Sloc := Prev_Token_Ptr; + SIS_Entry_Active := True; + end if; + + Pop_Scope_Stack; + return Decl_Node; + + end P_Subprogram; + + --------------------------------- + -- 6.1 Subprogram Declaration -- + --------------------------------- + + -- Parsed by P_Subprogram (6.1) + + ------------------------------------------ + -- 6.1 Abstract Subprogram Declaration -- + ------------------------------------------ + + -- Parsed by P_Subprogram (6.1) + + ----------------------------------- + -- 6.1 Subprogram Specification -- + ----------------------------------- + + -- SUBPROGRAM_SPECIFICATION ::= + -- procedure DEFINING_PROGRAM_UNIT_NAME PARAMETER_PROFILE + -- | function DEFINING_DESIGNATOR PARAMETER_AND_RESULT_PROFILE + + -- PARAMETER_PROFILE ::= [FORMAL_PART] + + -- PARAMETER_AND_RESULT_PROFILE ::= [FORMAL_PART] return SUBTYPE_MARK + + -- Subprogram specifications that appear in subprogram declarations + -- are parsed by P_Subprogram (6.1). This routine is used in other + -- contexts where subprogram specifications occur. + + -- Note: this routine does not affect the scope stack in any way + + -- Error recovery: can raise Error_Resync + + function P_Subprogram_Specification return Node_Id is + Specification_Node : Node_Id; + + begin + if Token = Tok_Function then + Specification_Node := New_Node (N_Function_Specification, Token_Ptr); + Scan; -- past FUNCTION + Ignore (Tok_Body); + Set_Defining_Unit_Name (Specification_Node, P_Defining_Designator); + Set_Parameter_Specifications + (Specification_Node, P_Parameter_Profile); + Check_Junk_Semicolon_Before_Return; + TF_Return; + Set_Subtype_Mark (Specification_Node, P_Subtype_Mark); + No_Constraint; + return Specification_Node; + + elsif Token = Tok_Procedure then + Specification_Node := New_Node (N_Procedure_Specification, Token_Ptr); + Scan; -- past PROCEDURE + Ignore (Tok_Body); + Set_Defining_Unit_Name + (Specification_Node, P_Defining_Program_Unit_Name); + Set_Parameter_Specifications + (Specification_Node, P_Parameter_Profile); + return Specification_Node; + + else + Error_Msg_SC ("subprogram specification expected"); + raise Error_Resync; + end if; + end P_Subprogram_Specification; + + --------------------- + -- 6.1 Designator -- + --------------------- + + -- DESIGNATOR ::= + -- [PARENT_UNIT_NAME .] IDENTIFIER | OPERATOR_SYMBOL + + -- The caller has checked that the initial token is an identifier, + -- operator symbol, or string literal. Note that we don't bother to + -- do much error diagnosis in this routine, since it is only used for + -- the label on END lines, and the routines in package Par.Endh will + -- check that the label is appropriate. + + -- Error recovery: cannot raise Error_Resync + + function P_Designator return Node_Id is + Ident_Node : Node_Id; + Name_Node : Node_Id; + Prefix_Node : Node_Id; + + function Real_Dot return Boolean; + -- Tests if a current token is an interesting period, i.e. is followed + -- by an identifier or operator symbol or string literal. If not, it is + -- probably just incorrect punctuation to be caught by our caller. Note + -- that the case of an operator symbol or string literal is also an + -- error, but that is an error that we catch here. If the result is + -- True, a real dot has been scanned and we are positioned past it, + -- if the result is False, the scan position is unchanged. + + function Real_Dot return Boolean is + Scan_State : Saved_Scan_State; + + begin + if Token /= Tok_Dot then + return False; + + else + Save_Scan_State (Scan_State); + Scan; -- past dot + + if Token = Tok_Identifier + or else Token = Tok_Operator_Symbol + or else Token = Tok_String_Literal + then + return True; + + else + Restore_Scan_State (Scan_State); + return False; + end if; + end if; + end Real_Dot; + + -- Start of processing for P_Designator + + begin + Ident_Node := Token_Node; + Scan; -- past initial token + + if Prev_Token = Tok_Operator_Symbol + or else Prev_Token = Tok_String_Literal + or else not Real_Dot + then + return Ident_Node; + + -- Child name case + + else + Prefix_Node := Ident_Node; + + -- Loop through child names, on entry to this loop, Prefix contains + -- the name scanned so far, and Ident_Node is the last identifier. + + loop + Name_Node := New_Node (N_Selected_Component, Prev_Token_Ptr); + Set_Prefix (Name_Node, Prefix_Node); + Ident_Node := P_Identifier; + Set_Selector_Name (Name_Node, Ident_Node); + Prefix_Node := Name_Node; + exit when not Real_Dot; + end loop; + + -- On exit from the loop, Ident_Node is the last identifier scanned, + -- i.e. the defining identifier, and Prefix_Node is a node for the + -- entire name, structured (incorrectly!) as a selected component. + + Name_Node := Prefix (Prefix_Node); + Change_Node (Prefix_Node, N_Designator); + Set_Name (Prefix_Node, Name_Node); + Set_Identifier (Prefix_Node, Ident_Node); + return Prefix_Node; + end if; + + exception + when Error_Resync => + while Token = Tok_Dot or else Token = Tok_Identifier loop + Scan; + end loop; + + return Error; + end P_Designator; + + ------------------------------ + -- 6.1 Defining Designator -- + ------------------------------ + + -- DEFINING_DESIGNATOR ::= + -- DEFINING_PROGRAM_UNIT_NAME | DEFINING_OPERATOR_SYMBOL + + -- Error recovery: cannot raise Error_Resync + + function P_Defining_Designator return Node_Id is + begin + if Token = Tok_Operator_Symbol then + return P_Defining_Operator_Symbol; + + elsif Token = Tok_String_Literal then + Error_Msg_SC ("invalid operator name"); + Scan; -- past junk string + return Error; + + else + return P_Defining_Program_Unit_Name; + end if; + end P_Defining_Designator; + + ------------------------------------- + -- 6.1 Defining Program Unit Name -- + ------------------------------------- + + -- DEFINING_PROGRAM_UNIT_NAME ::= + -- [PARENT_UNIT_NAME .] DEFINING_IDENTIFIER + + -- Note: PARENT_UNIT_NAME may be present only in 95 mode at the outer level + + -- Error recovery: cannot raise Error_Resync + + function P_Defining_Program_Unit_Name return Node_Id is + Ident_Node : Node_Id; + Name_Node : Node_Id; + Prefix_Node : Node_Id; + + begin + -- Set identifier casing if not already set and scan initial identifier + + if Token = Tok_Identifier + and then Identifier_Casing (Current_Source_File) = Unknown + then + Set_Identifier_Casing (Current_Source_File, Determine_Token_Casing); + end if; + + Ident_Node := P_Identifier; + Merge_Identifier (Ident_Node, Tok_Return); + + -- Normal case (not child library unit name) + + if Token /= Tok_Dot then + Change_Identifier_To_Defining_Identifier (Ident_Node); + return Ident_Node; + + -- Child library unit name case + + else + if Scope.Last > 1 then + Error_Msg_SP ("child unit allowed only at library level"); + raise Error_Resync; + + elsif Ada_83 then + Error_Msg_SP ("(Ada 83) child unit not allowed!"); + + end if; + + Prefix_Node := Ident_Node; + + -- Loop through child names, on entry to this loop, Prefix contains + -- the name scanned so far, and Ident_Node is the last identifier. + + loop + exit when Token /= Tok_Dot; + Name_Node := New_Node (N_Selected_Component, Token_Ptr); + Scan; -- past period + Set_Prefix (Name_Node, Prefix_Node); + Ident_Node := P_Identifier; + Set_Selector_Name (Name_Node, Ident_Node); + Prefix_Node := Name_Node; + end loop; + + -- On exit from the loop, Ident_Node is the last identifier scanned, + -- i.e. the defining identifier, and Prefix_Node is a node for the + -- entire name, structured (incorrectly!) as a selected component. + + Name_Node := Prefix (Prefix_Node); + Change_Node (Prefix_Node, N_Defining_Program_Unit_Name); + Set_Name (Prefix_Node, Name_Node); + Change_Identifier_To_Defining_Identifier (Ident_Node); + Set_Defining_Identifier (Prefix_Node, Ident_Node); + + -- All set with unit name parsed + + return Prefix_Node; + end if; + + exception + when Error_Resync => + while Token = Tok_Dot or else Token = Tok_Identifier loop + Scan; + end loop; + + return Error; + end P_Defining_Program_Unit_Name; + + -------------------------- + -- 6.1 Operator Symbol -- + -------------------------- + + -- OPERATOR_SYMBOL ::= STRING_LITERAL + + -- Operator symbol is returned by the scanner as Tok_Operator_Symbol + + ----------------------------------- + -- 6.1 Defining Operator Symbol -- + ----------------------------------- + + -- DEFINING_OPERATOR_SYMBOL ::= OPERATOR_SYMBOL + + -- The caller has checked that the initial symbol is an operator symbol + + function P_Defining_Operator_Symbol return Node_Id is + Op_Node : Node_Id; + + begin + Op_Node := Token_Node; + Change_Operator_Symbol_To_Defining_Operator_Symbol (Op_Node); + Scan; -- past operator symbol + return Op_Node; + end P_Defining_Operator_Symbol; + + ---------------------------- + -- 6.1 Parameter_Profile -- + ---------------------------- + + -- PARAMETER_PROFILE ::= [FORMAL_PART] + + -- Empty is returned if no formal part is present + + -- Error recovery: cannot raise Error_Resync + + function P_Parameter_Profile return List_Id is + begin + if Token = Tok_Left_Paren then + Scan; -- part left paren + return P_Formal_Part; + else + return No_List; + end if; + end P_Parameter_Profile; + + --------------------------------------- + -- 6.1 Parameter And Result Profile -- + --------------------------------------- + + -- Parsed by its parent construct, which uses P_Parameter_Profile to + -- parse the parameters, and P_Subtype_Mark to parse the return type. + + ---------------------- + -- 6.1 Formal part -- + ---------------------- + + -- FORMAL_PART ::= (PARAMETER_SPECIFICATION {; PARAMETER_SPECIFICATION}) + + -- PARAMETER_SPECIFICATION ::= + -- DEFINING_IDENTIFIER_LIST : MODE SUBTYPE_MARK + -- [:= DEFAULT_EXPRESSION] + -- | DEFINING_IDENTIFIER_LIST : ACCESS_DEFINITION + -- [:= DEFAULT_EXPRESSION] + + -- This scans the construct Formal_Part. The caller has already checked + -- that the initial token is a left parenthesis, and skipped past it, so + -- that on entry Token is the first token following the left parenthesis. + + -- Error recovery: cannot raise Error_Resync + + function P_Formal_Part return List_Id is + Specification_List : List_Id; + Specification_Node : Node_Id; + Scan_State : Saved_Scan_State; + Num_Idents : Nat; + Ident : Nat; + Ident_Sloc : Source_Ptr; + + Idents : array (Int range 1 .. 4096) of Entity_Id; + -- This array holds the list of defining identifiers. The upper bound + -- of 4096 is intended to be essentially infinite, and we do not even + -- bother to check for it being exceeded. + + begin + Specification_List := New_List; + + Specification_Loop : loop + begin + if Token = Tok_Pragma then + P_Pragmas_Misplaced; + end if; + + Ignore (Tok_Left_Paren); + Ident_Sloc := Token_Ptr; + Idents (1) := P_Defining_Identifier; + Num_Idents := 1; + + Ident_Loop : loop + exit Ident_Loop when Token = Tok_Colon; + + -- The only valid tokens are colon and comma, so if we have + -- neither do a bit of investigation to see which is the + -- better choice for insertion. + + if Token /= Tok_Comma then + + -- Assume colon if IN or OUT keyword found + + exit Ident_Loop when Token = Tok_In or else Token = Tok_Out; + + -- Otherwise scan ahead + + Save_Scan_State (Scan_State); + Look_Ahead : loop + + -- If we run into a semicolon, then assume that a + -- colon was missing, e.g. Parms (X Y; ...). Also + -- assume missing colon on EOF (a real disaster!) + -- and on a right paren, e.g. Parms (X Y), and also + -- on an assignment symbol, e.g. Parms (X Y := ..) + + if Token = Tok_Semicolon + or else Token = Tok_Right_Paren + or else Token = Tok_EOF + or else Token = Tok_Colon_Equal + then + Restore_Scan_State (Scan_State); + exit Ident_Loop; + + -- If we run into a colon, assume that we had a missing + -- comma, e.g. Parms (A B : ...). Also assume a missing + -- comma if we hit another comma, e.g. Parms (A B, C ..) + + elsif Token = Tok_Colon + or else Token = Tok_Comma + then + Restore_Scan_State (Scan_State); + exit Look_Ahead; + end if; + + Scan; + end loop Look_Ahead; + end if; + + -- Here if a comma is present, or to be assumed + + T_Comma; + Num_Idents := Num_Idents + 1; + Idents (Num_Idents) := P_Defining_Identifier; + end loop Ident_Loop; + + -- Fall through the loop on encountering a colon, or deciding + -- that there is a missing colon. + + T_Colon; + + -- If there are multiple identifiers, we repeatedly scan the + -- type and initialization expression information by resetting + -- the scan pointer (so that we get completely separate trees + -- for each occurrence). + + if Num_Idents > 1 then + Save_Scan_State (Scan_State); + end if; + + -- Loop through defining identifiers in list + + Ident := 1; + + Ident_List_Loop : loop + Specification_Node := + New_Node (N_Parameter_Specification, Ident_Sloc); + Set_Defining_Identifier (Specification_Node, Idents (Ident)); + + if Token = Tok_Access then + if Ada_83 then + Error_Msg_SC ("(Ada 83) access parameters not allowed"); + end if; + + Set_Parameter_Type + (Specification_Node, P_Access_Definition); + + else + P_Mode (Specification_Node); + + if Token = Tok_Procedure + or else + Token = Tok_Function + then + Error_Msg_SC ("formal subprogram parameter not allowed"); + Scan; + + if Token = Tok_Left_Paren then + Discard_Junk_List (P_Formal_Part); + end if; + + if Token = Tok_Return then + Scan; + Discard_Junk_Node (P_Subtype_Mark); + end if; + + Set_Parameter_Type (Specification_Node, Error); + + else + Set_Parameter_Type (Specification_Node, P_Subtype_Mark); + No_Constraint; + end if; + end if; + + Set_Expression (Specification_Node, Init_Expr_Opt (True)); + + if Ident > 1 then + Set_Prev_Ids (Specification_Node, True); + end if; + + if Ident < Num_Idents then + Set_More_Ids (Specification_Node, True); + end if; + + Append (Specification_Node, Specification_List); + exit Ident_List_Loop when Ident = Num_Idents; + Ident := Ident + 1; + Restore_Scan_State (Scan_State); + end loop Ident_List_Loop; + + exception + when Error_Resync => + Resync_Semicolon_List; + end; + + if Token = Tok_Semicolon then + Scan; -- past semicolon + + -- If we have RETURN or IS after the semicolon, then assume + -- that semicolon should have been a right parenthesis and exit + + if Token = Tok_Is or else Token = Tok_Return then + Error_Msg_SP ("expected "")"" in place of "";"""); + exit Specification_Loop; + end if; + + elsif Token = Tok_Right_Paren then + Scan; -- past right paren + exit Specification_Loop; + + -- Special check for common error of using comma instead of semicolon + + elsif Token = Tok_Comma then + T_Semicolon; + Scan; -- past comma + + -- Special check for omitted separator + + elsif Token = Tok_Identifier then + T_Semicolon; + + -- If nothing sensible, skip to next semicolon or right paren + + else + T_Semicolon; + Resync_Semicolon_List; + + if Token = Tok_Semicolon then + Scan; -- past semicolon + else + T_Right_Paren; + exit Specification_Loop; + end if; + end if; + end loop Specification_Loop; + + return Specification_List; + end P_Formal_Part; + + ---------------------------------- + -- 6.1 Parameter Specification -- + ---------------------------------- + + -- Parsed by P_Formal_Part (6.1) + + --------------- + -- 6.1 Mode -- + --------------- + + -- MODE ::= [in] | in out | out + + -- There is no explicit node in the tree for the Mode. Instead the + -- In_Present and Out_Present flags are set in the parent node to + -- record the presence of keywords specifying the mode. + + -- Error_Recovery: cannot raise Error_Resync + + procedure P_Mode (Node : Node_Id) is + begin + if Token = Tok_In then + Scan; -- past IN + Set_In_Present (Node, True); + end if; + + if Token = Tok_Out then + Scan; -- past OUT + Set_Out_Present (Node, True); + end if; + + if Token = Tok_In then + Error_Msg_SC ("IN must preceed OUT in parameter mode"); + Scan; -- past IN + Set_In_Present (Node, True); + end if; + end P_Mode; + + -------------------------- + -- 6.3 Subprogram Body -- + -------------------------- + + -- Parsed by P_Subprogram (6.1) + + ----------------------------------- + -- 6.4 Procedure Call Statement -- + ----------------------------------- + + -- Parsed by P_Sequence_Of_Statements (5.1) + + ------------------------ + -- 6.4 Function Call -- + ------------------------ + + -- Parsed by P_Call_Or_Name (4.1) + + -------------------------------- + -- 6.4 Actual Parameter Part -- + -------------------------------- + + -- Parsed by P_Call_Or_Name (4.1) + + -------------------------------- + -- 6.4 Parameter Association -- + -------------------------------- + + -- Parsed by P_Call_Or_Name (4.1) + + ------------------------------------ + -- 6.4 Explicit Actual Parameter -- + ------------------------------------ + + -- Parsed by P_Call_Or_Name (4.1) + + --------------------------- + -- 6.5 Return Statement -- + --------------------------- + + -- RETURN_STATEMENT ::= return [EXPRESSION]; + + -- The caller has checked that the initial token is RETURN + + -- Error recovery: can raise Error_Resync + + function P_Return_Statement return Node_Id is + Return_Node : Node_Id; + + begin + Return_Node := New_Node (N_Return_Statement, Token_Ptr); + + -- Sloc points to RETURN + -- Expression (Op3) + + Scan; -- past RETURN + + if Token /= Tok_Semicolon then + + -- If no semicolon, then scan an expression, except that + -- we avoid trying to scan an expression if we are at an + -- expression terminator since in that case the best error + -- message is probably that we have a missing semicolon. + + if Token not in Token_Class_Eterm then + Set_Expression (Return_Node, P_Expression_No_Right_Paren); + end if; + end if; + + TF_Semicolon; + return Return_Node; + end P_Return_Statement; + +end Ch6; diff --git a/gcc/ada/par-ch7.adb b/gcc/ada/par-ch7.adb new file mode 100644 index 00000000000..de632133f4b --- /dev/null +++ b/gcc/ada/par-ch7.adb @@ -0,0 +1,282 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 7 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.29 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch7 is + + --------------------------------------------- + -- 7.1 Package (also 8.5.3, 10.1.3, 12.3) -- + --------------------------------------------- + + -- This routine scans out a package declaration, package body, or a + -- renaming declaration or generic instantiation starting with PACKAGE + + -- PACKAGE_DECLARATION ::= PACKAGE_SPECIFICATION; + + -- PACKAGE_SPECIFICATION ::= + -- package DEFINING_PROGRAM_UNIT_NAME is + -- {BASIC_DECLARATIVE_ITEM} + -- [private + -- {BASIC_DECLARATIVE_ITEM}] + -- end [[PARENT_UNIT_NAME .] IDENTIFIER] + + -- PACKAGE_BODY ::= + -- package body DEFINING_PROGRAM_UNIT_NAME is + -- DECLARATIVE_PART + -- [begin + -- HANDLED_SEQUENCE_OF_STATEMENTS] + -- end [[PARENT_UNIT_NAME .] IDENTIFIER] + + -- PACKAGE_RENAMING_DECLARATION ::= + -- package DEFINING_IDENTIFIER renames package_NAME; + + -- PACKAGE_BODY_STUB ::= + -- package body DEFINING_IDENTIFIER is separate; + + -- The value in Pf_Flags indicates which of these possible declarations + -- is acceptable to the caller: + + -- Pf_Flags.Spcn Set if specification OK + -- Pf_Flags.Decl Set if declaration OK + -- Pf_Flags.Gins Set if generic instantiation OK + -- Pf_Flags.Pbod Set if proper body OK + -- Pf_Flags.Rnam Set if renaming declaration OK + -- Pf_Flags.Stub Set if body stub OK + + -- If an inappropriate form is encountered, it is scanned out but an + -- error message indicating that it is appearing in an inappropriate + -- context is issued. The only possible settings for Pf_Flags are those + -- defined as constants in package Par. + + -- Note: in all contexts where a package specification is required, there + -- is a terminating semicolon. This semicolon is scanned out in the case + -- where Pf_Flags is set to Pf_Spcn, even though it is not strictly part + -- of the package specification (it's just too much trouble, and really + -- quite unnecessary, to deal with scanning out an END where the semicolon + -- after the END is not considered to be part of the END. + + -- The caller has checked that the initial token is PACKAGE + + -- Error recovery: cannot raise Error_Resync + + function P_Package (Pf_Flags : Pf_Rec) return Node_Id is + Package_Node : Node_Id; + Specification_Node : Node_Id; + Name_Node : Node_Id; + Package_Sloc : Source_Ptr; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Lreq := False; + + Package_Sloc := Token_Ptr; + Scan; -- past PACKAGE + + if Token = Tok_Type then + Error_Msg_SC ("TYPE not allowed here"); + Scan; -- past TYPE + end if; + + -- Case of package body. Note that we demand a package body if that + -- is the only possibility (even if the BODY keyword is not present) + + if Token = Tok_Body or else Pf_Flags = Pf_Pbod then + if not Pf_Flags.Pbod then + Error_Msg_SC ("package body cannot appear here!"); + end if; + + T_Body; + Name_Node := P_Defining_Program_Unit_Name; + Scope.Table (Scope.Last).Labl := Name_Node; + TF_Is; + + if Separate_Present then + if not Pf_Flags.Stub then + Error_Msg_SC ("body stub cannot appear here!"); + end if; + + Scan; -- past SEPARATE + TF_Semicolon; + Pop_Scope_Stack; + + Package_Node := New_Node (N_Package_Body_Stub, Package_Sloc); + Set_Defining_Identifier (Package_Node, Name_Node); + + else + Package_Node := New_Node (N_Package_Body, Package_Sloc); + Set_Defining_Unit_Name (Package_Node, Name_Node); + Parse_Decls_Begin_End (Package_Node); + end if; + + return Package_Node; + + -- Cases other than Package_Body + + else + Name_Node := P_Defining_Program_Unit_Name; + Scope.Table (Scope.Last).Labl := Name_Node; + + -- Case of renaming declaration + + Check_Misspelling_Of (Tok_Renames); + + if Token = Tok_Renames then + if not Pf_Flags.Rnam then + Error_Msg_SC ("renaming declaration cannot appear here!"); + end if; + + Scan; -- past RENAMES; + + Package_Node := + New_Node (N_Package_Renaming_Declaration, Package_Sloc); + Set_Defining_Unit_Name (Package_Node, Name_Node); + Set_Name (Package_Node, P_Qualified_Simple_Name); + + No_Constraint; + TF_Semicolon; + Pop_Scope_Stack; + return Package_Node; + + else + TF_Is; + + -- Case of generic instantiation + + if Token = Tok_New then + if not Pf_Flags.Gins then + Error_Msg_SC + ("generic instantiation cannot appear here!"); + end if; + + Scan; -- past NEW + + Package_Node := + New_Node (N_Package_Instantiation, Package_Sloc); + Set_Defining_Unit_Name (Package_Node, Name_Node); + Set_Name (Package_Node, P_Qualified_Simple_Name); + Set_Generic_Associations + (Package_Node, P_Generic_Actual_Part_Opt); + TF_Semicolon; + Pop_Scope_Stack; + + -- Case of package declaration or package specification + + else + Specification_Node := + New_Node (N_Package_Specification, Package_Sloc); + + Set_Defining_Unit_Name (Specification_Node, Name_Node); + Set_Visible_Declarations + (Specification_Node, P_Basic_Declarative_Items); + + if Token = Tok_Private then + Error_Msg_Col := Scope.Table (Scope.Last).Ecol; + + if Style.RM_Column_Check then + if Token_Is_At_Start_Of_Line + and then Start_Column /= Error_Msg_Col + then + Error_Msg_SC + ("(style) PRIVATE in wrong column, should be@"); + end if; + end if; + + Scan; -- past PRIVATE + Set_Private_Declarations + (Specification_Node, P_Basic_Declarative_Items); + + -- Deal gracefully with multiple PRIVATE parts + + while Token = Tok_Private loop + Error_Msg_SC + ("only one private part allowed per package"); + Scan; -- past PRIVATE + Append_List (P_Basic_Declarative_Items, + Private_Declarations (Specification_Node)); + end loop; + end if; + + if Pf_Flags = Pf_Spcn then + Package_Node := Specification_Node; + else + Package_Node := + New_Node (N_Package_Declaration, Package_Sloc); + Set_Specification (Package_Node, Specification_Node); + end if; + + if Token = Tok_Begin then + Error_Msg_SC ("begin block not allowed in package spec"); + Scan; -- past BEGIN + Discard_Junk_List (P_Sequence_Of_Statements (SS_None)); + end if; + + End_Statements (Specification_Node); + end if; + + return Package_Node; + end if; + end if; + end P_Package; + + ------------------------------ + -- 7.1 Package Declaration -- + ------------------------------ + + -- Parsed by P_Package (7.1) + + -------------------------------- + -- 7.1 Package Specification -- + -------------------------------- + + -- Parsed by P_Package (7.1) + + ----------------------- + -- 7.1 Package Body -- + ----------------------- + + -- Parsed by P_Package (7.1) + + ----------------------------------- + -- 7.3 Private Type Declaration -- + ----------------------------------- + + -- Parsed by P_Type_Declaration (3.2.1) + + ---------------------------------------- + -- 7.3 Private Extension Declaration -- + ---------------------------------------- + + -- Parsed by P_Type_Declaration (3.2.1) + +end Ch7; diff --git a/gcc/ada/par-ch8.adb b/gcc/ada/par-ch8.adb new file mode 100644 index 00000000000..9d1b386280d --- /dev/null +++ b/gcc/ada/par-ch8.adb @@ -0,0 +1,175 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 8 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.16 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch8 is + + ----------------------- + -- Local Subprograms -- + ----------------------- + + function P_Use_Package_Clause return Node_Id; + function P_Use_Type_Clause return Node_Id; + + --------------------- + -- 8.4 Use Clause -- + --------------------- + + -- USE_CLAUSE ::= USE_PACKAGE_CLAUSE | USE_TYPE_CLAUSE + + -- The caller has checked that the initial token is USE + + -- Error recovery: cannot raise Error_Resync + + function P_Use_Clause return Node_Id is + begin + Scan; -- past USE + + if Token = Tok_Type then + return P_Use_Type_Clause; + + else + return P_Use_Package_Clause; + end if; + end P_Use_Clause; + + ----------------------------- + -- 8.4 Use Package Clause -- + ----------------------------- + + -- USE_PACKAGE_CLAUSE ::= use package_NAME {, package_NAME}; + + -- The caller has scanned out the USE keyword + + -- Error recovery: cannot raise Error_Resync + + function P_Use_Package_Clause return Node_Id is + Use_Node : Node_Id; + + begin + Use_Node := New_Node (N_Use_Package_Clause, Prev_Token_Ptr); + Set_Names (Use_Node, New_List); + + if Token = Tok_Package then + Error_Msg_SC ("PACKAGE should not appear here"); + Scan; -- past PACKAGE + end if; + + loop + Append (P_Qualified_Simple_Name, Names (Use_Node)); + exit when Token /= Tok_Comma; + Scan; -- past comma + end loop; + + TF_Semicolon; + return Use_Node; + end P_Use_Package_Clause; + + -------------------------- + -- 8.4 Use Type Clause -- + -------------------------- + + -- USE_TYPE_CLAUSE ::= use type SUBTYPE_MARK {, SUBTYPE_MARK}; + + -- The caller has checked that the initial token is USE, scanned it out + -- and that the current token is TYPE. + + -- Error recovery: cannot raise Error_Resync + + function P_Use_Type_Clause return Node_Id is + Use_Node : Node_Id; + + begin + Use_Node := New_Node (N_Use_Type_Clause, Prev_Token_Ptr); + Set_Subtype_Marks (Use_Node, New_List); + + if Ada_83 then + Error_Msg_SC ("(Ada 83) use type not allowed!"); + end if; + + Scan; -- past TYPE + + loop + Append (P_Subtype_Mark, Subtype_Marks (Use_Node)); + No_Constraint; + exit when Token /= Tok_Comma; + Scan; -- past comma + end loop; + + TF_Semicolon; + return Use_Node; + end P_Use_Type_Clause; + + ------------------------------- + -- 8.5 Renaming Declaration -- + ------------------------------- + + -- Object renaming declarations and exception renaming declarations + -- are parsed by P_Identifier_Declaration (3.3.1) + + -- Subprogram renaming declarations are parsed by P_Subprogram (6.1) + + -- Package renaming declarations are parsed by P_Package (7.1) + + -- Generic renaming declarations are parsed by P_Generic (12.1) + + ---------------------------------------- + -- 8.5.1 Object Renaming Declaration -- + ---------------------------------------- + + -- Parsed by P_Identifier_Declarations (3.3.1) + + ---------------------------------------- + -- 8.5.2 Exception Renaming Declaration -- + ---------------------------------------- + + -- Parsed by P_Identifier_Declarations (3.3.1) + + ----------------------------------------- + -- 8.5.3 Package Renaming Declaration -- + ----------------------------------------- + + -- Parsed by P_Package (7.1) + + -------------------------------------------- + -- 8.5.4 Subprogram Renaming Declaration -- + -------------------------------------------- + + -- Parsed by P_Subprogram (6.1) + + ----------------------------------------- + -- 8.5.2 Generic Renaming Declaration -- + ----------------------------------------- + + -- Parsed by P_Generic (12.1) + +end Ch8; diff --git a/gcc/ada/par-ch9.adb b/gcc/ada/par-ch9.adb new file mode 100644 index 00000000000..87d6be6ae5f --- /dev/null +++ b/gcc/ada/par-ch9.adb @@ -0,0 +1,1616 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . C H 9 -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.82 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +pragma Style_Checks (All_Checks); +-- Turn off subprogram body ordering check. Subprograms are in order +-- by RM section rather than alphabetical + +separate (Par) +package body Ch9 is + + -- Local subprograms, used only in this chapter + + function P_Accept_Alternative return Node_Id; + function P_Delay_Alternative return Node_Id; + function P_Delay_Relative_Statement return Node_Id; + function P_Delay_Until_Statement return Node_Id; + function P_Entry_Barrier return Node_Id; + function P_Entry_Body_Formal_Part return Node_Id; + function P_Entry_Declaration return Node_Id; + function P_Entry_Index_Specification return Node_Id; + function P_Protected_Definition return Node_Id; + function P_Protected_Operation_Declaration_Opt return Node_Id; + function P_Protected_Operation_Items return List_Id; + function P_Task_Definition return Node_Id; + function P_Task_Items return List_Id; + + ----------------------------- + -- 9.1 Task (also 10.1.3) -- + ----------------------------- + + -- TASK_TYPE_DECLARATION ::= + -- task type DEFINING_IDENTIFIER [KNOWN_DISCRIMINANT_PART] + -- [is TASK_DEFINITION]; + + -- SINGLE_TASK_DECLARATION ::= + -- task DEFINING_IDENTIFIER [is TASK_DEFINITION]; + + -- TASK_BODY ::= + -- task body DEFINING_IDENTIFIER is + -- DECLARATIVE_PART + -- begin + -- HANDLED_SEQUENCE_OF_STATEMENTS + -- end [task_IDENTIFIER] + + -- TASK_BODY_STUB ::= + -- task body DEFINING_IDENTIFIER is separate; + + -- This routine scans out a task declaration, task body, or task stub + + -- The caller has checked that the initial token is TASK and scanned + -- past it, so that Token is set to the token after TASK + + -- Error recovery: cannot raise Error_Resync + + function P_Task return Node_Id is + Name_Node : Node_Id; + Task_Node : Node_Id; + Task_Sloc : Source_Ptr; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Lreq := False; + Task_Sloc := Prev_Token_Ptr; + + if Token = Tok_Body then + Scan; -- past BODY + Name_Node := P_Defining_Identifier; + Scope.Table (Scope.Last).Labl := Name_Node; + + if Token = Tok_Left_Paren then + Error_Msg_SC ("discriminant part not allowed in task body"); + Discard_Junk_List (P_Known_Discriminant_Part_Opt); + end if; + + TF_Is; + + -- Task stub + + if Token = Tok_Separate then + Scan; -- past SEPARATE + Task_Node := New_Node (N_Task_Body_Stub, Task_Sloc); + Set_Defining_Identifier (Task_Node, Name_Node); + TF_Semicolon; + Pop_Scope_Stack; -- remove unused entry + + -- Task body + + else + Task_Node := New_Node (N_Task_Body, Task_Sloc); + Set_Defining_Identifier (Task_Node, Name_Node); + Parse_Decls_Begin_End (Task_Node); + end if; + + return Task_Node; + + -- Otherwise we must have a task declaration + + else + if Token = Tok_Type then + Scan; -- past TYPE + Task_Node := New_Node (N_Task_Type_Declaration, Task_Sloc); + Name_Node := P_Defining_Identifier; + Set_Defining_Identifier (Task_Node, Name_Node); + Scope.Table (Scope.Last).Labl := Name_Node; + Set_Discriminant_Specifications + (Task_Node, P_Known_Discriminant_Part_Opt); + + else + Task_Node := New_Node (N_Single_Task_Declaration, Task_Sloc); + Name_Node := P_Defining_Identifier; + Set_Defining_Identifier (Task_Node, Name_Node); + Scope.Table (Scope.Last).Labl := Name_Node; + + if Token = Tok_Left_Paren then + Error_Msg_SC ("discriminant part not allowed for single task"); + Discard_Junk_List (P_Known_Discriminant_Part_Opt); + end if; + + end if; + + -- Parse optional task definition. Note that P_Task_Definition scans + -- out the semicolon as well as the task definition itself. + + if Token = Tok_Semicolon then + + -- A little check, if the next token after semicolon is + -- Entry, then surely the semicolon should really be IS + + Scan; -- past semicolon + + if Token = Tok_Entry then + Error_Msg_SP (""";"" should be IS"); + Set_Task_Definition (Task_Node, P_Task_Definition); + else + Pop_Scope_Stack; -- Remove unused entry + end if; + else + TF_Is; -- must have IS if no semicolon + Set_Task_Definition (Task_Node, P_Task_Definition); + end if; + + return Task_Node; + end if; + end P_Task; + + -------------------------------- + -- 9.1 Task Type Declaration -- + -------------------------------- + + -- Parsed by P_Task (9.1) + + ---------------------------------- + -- 9.1 Single Task Declaration -- + ---------------------------------- + + -- Parsed by P_Task (9.1) + + -------------------------- + -- 9.1 Task Definition -- + -------------------------- + + -- TASK_DEFINITION ::= + -- {TASK_ITEM} + -- [private + -- {TASK_ITEM}] + -- end [task_IDENTIFIER]; + + -- The caller has already made the scope stack entry + + -- Note: there is a small deviation from official syntax here in that we + -- regard the semicolon after end as part of the Task_Definition, and in + -- the official syntax, it's part of the enclosing declaration. The reason + -- for this deviation is that otherwise the end processing would have to + -- be special cased, which would be a nuisance! + + -- Error recovery: cannot raise Error_Resync + + function P_Task_Definition return Node_Id is + Def_Node : Node_Id; + + begin + Def_Node := New_Node (N_Task_Definition, Token_Ptr); + Set_Visible_Declarations (Def_Node, P_Task_Items); + + if Token = Tok_Private then + Scan; -- past PRIVATE + Set_Private_Declarations (Def_Node, P_Task_Items); + + -- Deal gracefully with multiple PRIVATE parts + + while Token = Tok_Private loop + Error_Msg_SC ("Only one private part allowed per task"); + Scan; -- past PRIVATE + Append_List (P_Task_Items, Private_Declarations (Def_Node)); + end loop; + end if; + + End_Statements (Def_Node); + return Def_Node; + end P_Task_Definition; + + -------------------- + -- 9.1 Task Item -- + -------------------- + + -- TASK_ITEM ::= ENTRY_DECLARATION | REPRESENTATION_CLAUSE + + -- This subprogram scans a (possibly empty) list of task items and pragmas + + -- Error recovery: cannot raise Error_Resync + + -- Note: a pragma can also be returned in this position + + function P_Task_Items return List_Id is + Items : List_Id; + Item_Node : Node_Id; + Decl_Sloc : Source_Ptr; + + begin + -- Get rid of active SIS entry from outer scope. This means we will + -- miss some nested cases, but it doesn't seem worth the effort. See + -- discussion in Par for further details + + SIS_Entry_Active := False; + + -- Loop to scan out task items + + Items := New_List; + + Decl_Loop : loop + Decl_Sloc := Token_Ptr; + + if Token = Tok_Pragma then + Append (P_Pragma, Items); + + elsif Token = Tok_Entry then + Append (P_Entry_Declaration, Items); + + elsif Token = Tok_For then + -- Representation clause in task declaration. The only rep + -- clause which is legal in a protected is an address clause, + -- so that is what we try to scan out. + + Item_Node := P_Representation_Clause; + + if Nkind (Item_Node) = N_At_Clause then + Append (Item_Node, Items); + + elsif Nkind (Item_Node) = N_Attribute_Definition_Clause + and then Chars (Item_Node) = Name_Address + then + Append (Item_Node, Items); + + else + Error_Msg + ("the only representation clause " & + "allowed here is an address clause!", Decl_Sloc); + end if; + + elsif Token = Tok_Identifier + or else Token in Token_Class_Declk + then + Error_Msg_SC ("Illegal declaration in task definition"); + Resync_Past_Semicolon; + + else + exit Decl_Loop; + end if; + end loop Decl_Loop; + + return Items; + end P_Task_Items; + + -------------------- + -- 9.1 Task Body -- + -------------------- + + -- Parsed by P_Task (9.1) + + ---------------------------------- + -- 9.4 Protected (also 10.1.3) -- + ---------------------------------- + + -- PROTECTED_TYPE_DECLARATION ::= + -- protected type DEFINING_IDENTIFIER [KNOWN_DISCRIMINANT_PART] + -- is PROTECTED_DEFINITION; + + -- SINGLE_PROTECTED_DECLARATION ::= + -- protected DEFINING_IDENTIFIER is PROTECTED_DEFINITION; + + -- PROTECTED_BODY ::= + -- protected body DEFINING_IDENTIFIER is + -- {PROTECTED_OPERATION_ITEM} + -- end [protected_IDENTIFIER]; + + -- PROTECTED_BODY_STUB ::= + -- protected body DEFINING_IDENTIFIER is separate; + + -- This routine scans out a protected declaration, protected body + -- or a protected stub. + + -- The caller has checked that the initial token is PROTECTED and + -- scanned past it, so Token is set to the following token. + + -- Error recovery: cannot raise Error_Resync + + function P_Protected return Node_Id is + Name_Node : Node_Id; + Protected_Node : Node_Id; + Protected_Sloc : Source_Ptr; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Lreq := False; + Protected_Sloc := Prev_Token_Ptr; + + if Token = Tok_Body then + Scan; -- past BODY + Name_Node := P_Defining_Identifier; + Scope.Table (Scope.Last).Labl := Name_Node; + + if Token = Tok_Left_Paren then + Error_Msg_SC ("discriminant part not allowed in protected body"); + Discard_Junk_List (P_Known_Discriminant_Part_Opt); + end if; + + TF_Is; + + -- Protected stub + + if Token = Tok_Separate then + Scan; -- past SEPARATE + Protected_Node := New_Node (N_Protected_Body_Stub, Protected_Sloc); + Set_Defining_Identifier (Protected_Node, Name_Node); + TF_Semicolon; + Pop_Scope_Stack; -- remove unused entry + + -- Protected body + + else + Protected_Node := New_Node (N_Protected_Body, Protected_Sloc); + Set_Defining_Identifier (Protected_Node, Name_Node); + Set_Declarations (Protected_Node, P_Protected_Operation_Items); + End_Statements (Protected_Node); + end if; + + return Protected_Node; + + -- Otherwise we must have a protected declaration + + else + if Token = Tok_Type then + Scan; -- past TYPE + Protected_Node := + New_Node (N_Protected_Type_Declaration, Protected_Sloc); + Name_Node := P_Defining_Identifier; + Set_Defining_Identifier (Protected_Node, Name_Node); + Scope.Table (Scope.Last).Labl := Name_Node; + Set_Discriminant_Specifications + (Protected_Node, P_Known_Discriminant_Part_Opt); + + else + Protected_Node := + New_Node (N_Single_Protected_Declaration, Protected_Sloc); + Name_Node := P_Defining_Identifier; + Set_Defining_Identifier (Protected_Node, Name_Node); + + if Token = Tok_Left_Paren then + Error_Msg_SC + ("discriminant part not allowed for single protected"); + Discard_Junk_List (P_Known_Discriminant_Part_Opt); + end if; + + Scope.Table (Scope.Last).Labl := Name_Node; + end if; + + T_Is; + Set_Protected_Definition (Protected_Node, P_Protected_Definition); + return Protected_Node; + end if; + end P_Protected; + + ------------------------------------- + -- 9.4 Protected Type Declaration -- + ------------------------------------- + + -- Parsed by P_Protected (9.4) + + --------------------------------------- + -- 9.4 Single Protected Declaration -- + --------------------------------------- + + -- Parsed by P_Protected (9.4) + + ------------------------------- + -- 9.4 Protected Definition -- + ------------------------------- + + -- PROTECTED_DEFINITION ::= + -- {PROTECTED_OPERATION_DECLARATION} + -- [private + -- {PROTECTED_ELEMENT_DECLARATION}] + -- end [protected_IDENTIFIER] + + -- PROTECTED_ELEMENT_DECLARATION ::= + -- PROTECTED_OPERATION_DECLARATION + -- | COMPONENT_DECLARATION + + -- The caller has already established the scope stack entry + + -- Error recovery: cannot raise Error_Resync + + function P_Protected_Definition return Node_Id is + Def_Node : Node_Id; + Item_Node : Node_Id; + + begin + Def_Node := New_Node (N_Protected_Definition, Token_Ptr); + + -- Get rid of active SIS entry from outer scope. This means we will + -- miss some nested cases, but it doesn't seem worth the effort. See + -- discussion in Par for further details + + SIS_Entry_Active := False; + + -- Loop to scan visible declarations (protected operation declarations) + + Set_Visible_Declarations (Def_Node, New_List); + + loop + Item_Node := P_Protected_Operation_Declaration_Opt; + exit when No (Item_Node); + Append (Item_Node, Visible_Declarations (Def_Node)); + end loop; + + -- Deal with PRIVATE part (including graceful handling + -- of multiple PRIVATE parts). + + Private_Loop : while Token = Tok_Private loop + if No (Private_Declarations (Def_Node)) then + Set_Private_Declarations (Def_Node, New_List); + else + Error_Msg_SC ("duplicate private part"); + end if; + + Scan; -- past PRIVATE + + Declaration_Loop : loop + if Token = Tok_Identifier then + P_Component_Items (Private_Declarations (Def_Node)); + else + Item_Node := P_Protected_Operation_Declaration_Opt; + exit Declaration_Loop when No (Item_Node); + Append (Item_Node, Private_Declarations (Def_Node)); + end if; + end loop Declaration_Loop; + end loop Private_Loop; + + End_Statements (Def_Node); + return Def_Node; + end P_Protected_Definition; + + ------------------------------------------ + -- 9.4 Protected Operation Declaration -- + ------------------------------------------ + + -- PROTECTED_OPERATION_DECLARATION ::= + -- SUBPROGRAM_DECLARATION + -- | ENTRY_DECLARATION + -- | REPRESENTATION_CLAUSE + + -- Error recovery: cannot raise Error_Resync + + -- Note: a pragma can also be returned in this position + + -- We are not currently permitting representation clauses to appear as + -- protected operation declarations, do we have to rethink this??? + + function P_Protected_Operation_Declaration_Opt return Node_Id is + L : List_Id; + P : Source_Ptr; + + begin + -- This loop runs more than once only when a junk declaration + -- is skipped. + + loop + if Token = Tok_Pragma then + return P_Pragma; + + elsif Token = Tok_Entry then + return P_Entry_Declaration; + + elsif Token = Tok_Function or else Token = Tok_Procedure then + return P_Subprogram (Pf_Decl); + + elsif Token = Tok_Identifier then + L := New_List; + P := Token_Ptr; + Skip_Declaration (L); + + if Nkind (First (L)) = N_Object_Declaration then + Error_Msg + ("component must be declared in private part of " & + "protected type", P); + else + Error_Msg + ("illegal declaration in protected definition", P); + end if; + + elsif Token in Token_Class_Declk then + Error_Msg_SC ("illegal declaration in protected definition"); + Resync_Past_Semicolon; + + -- Return now to avoid cascaded messages if next declaration + -- is a valid component declaration. + + return Error; + + elsif Token = Tok_For then + Error_Msg_SC + ("representation clause not allowed in protected definition"); + Resync_Past_Semicolon; + + else + return Empty; + end if; + end loop; + end P_Protected_Operation_Declaration_Opt; + + ----------------------------------- + -- 9.4 Protected Operation Item -- + ----------------------------------- + + -- PROTECTED_OPERATION_ITEM ::= + -- SUBPROGRAM_DECLARATION + -- | SUBPROGRAM_BODY + -- | ENTRY_BODY + -- | REPRESENTATION_CLAUSE + + -- This procedure parses and returns a list of protected operation items + + -- We are not currently permitting representation clauses to appear + -- as protected operation items, do we have to rethink this??? + + function P_Protected_Operation_Items return List_Id is + Item_List : List_Id; + + begin + Item_List := New_List; + + loop + if Token = Tok_Entry or else Bad_Spelling_Of (Tok_Entry) then + Append (P_Entry_Body, Item_List); + + elsif Token = Tok_Function or else Bad_Spelling_Of (Tok_Function) + or else + Token = Tok_Procedure or else Bad_Spelling_Of (Tok_Procedure) + then + Append (P_Subprogram (Pf_Decl_Pbod), Item_List); + + elsif Token = Tok_Pragma or else Bad_Spelling_Of (Tok_Pragma) then + P_Pragmas_Opt (Item_List); + + elsif Token = Tok_Private or else Bad_Spelling_Of (Tok_Private) then + Error_Msg_SC ("PRIVATE not allowed in protected body"); + Scan; -- past PRIVATE + + elsif Token = Tok_Identifier then + Error_Msg_SC + ("all components must be declared in spec!"); + Resync_Past_Semicolon; + + elsif Token in Token_Class_Declk then + Error_Msg_SC ("this declaration not allowed in protected body"); + Resync_Past_Semicolon; + + else + exit; + end if; + end loop; + + return Item_List; + end P_Protected_Operation_Items; + + ------------------------------ + -- 9.5.2 Entry Declaration -- + ------------------------------ + + -- ENTRY_DECLARATION ::= + -- entry DEFINING_IDENTIFIER [(DISCRETE_SUBTYPE_DEFINITION)] + -- PARAMETER_PROFILE; + + -- The caller has checked that the initial token is ENTRY + + -- Error recovery: cannot raise Error_Resync + + function P_Entry_Declaration return Node_Id is + Decl_Node : Node_Id; + Scan_State : Saved_Scan_State; + + begin + Decl_Node := New_Node (N_Entry_Declaration, Token_Ptr); + Scan; -- past ENTRY + + Set_Defining_Identifier (Decl_Node, P_Defining_Identifier); + + -- If left paren, could be (Discrete_Subtype_Definition) or Formal_Part + + if Token = Tok_Left_Paren then + Scan; -- past ( + + -- If identifier after left paren, could still be either + + if Token = Tok_Identifier then + Save_Scan_State (Scan_State); -- at Id + Scan; -- past Id + + -- If comma or colon after Id, must be Formal_Part + + if Token = Tok_Comma or else Token = Tok_Colon then + Restore_Scan_State (Scan_State); -- to Id + Set_Parameter_Specifications (Decl_Node, P_Formal_Part); + + -- Else if Id wi no comma or colon, must be discrete subtype defn + + else + Restore_Scan_State (Scan_State); -- to Id + Set_Discrete_Subtype_Definition + (Decl_Node, P_Discrete_Subtype_Definition); + T_Right_Paren; + Set_Parameter_Specifications (Decl_Node, P_Parameter_Profile); + end if; + + -- If no Id, must be discrete subtype definition + + else + Set_Discrete_Subtype_Definition + (Decl_Node, P_Discrete_Subtype_Definition); + T_Right_Paren; + Set_Parameter_Specifications (Decl_Node, P_Parameter_Profile); + end if; + end if; + + -- Error recovery check for illegal return + + if Token = Tok_Return then + Error_Msg_SC ("entry cannot have return value!"); + Scan; + Discard_Junk_Node (P_Subtype_Indication); + end if; + + -- Error recovery check for improper use of entry barrier in spec + + if Token = Tok_When then + Error_Msg_SC ("barrier not allowed here (belongs in body)"); + Scan; -- past WHEN; + Discard_Junk_Node (P_Expression_No_Right_Paren); + end if; + + TF_Semicolon; + return Decl_Node; + end P_Entry_Declaration; + + ----------------------------- + -- 9.5.2 Accept Statement -- + ----------------------------- + + -- ACCEPT_STATEMENT ::= + -- accept entry_DIRECT_NAME + -- [(ENTRY_INDEX)] PARAMETER_PROFILE [do + -- HANDLED_SEQUENCE_OF_STATEMENTS + -- end [entry_IDENTIFIER]]; + + -- The caller has checked that the initial token is ACCEPT + + -- Error recovery: cannot raise Error_Resync. If an error occurs, the + -- scan is resynchronized past the next semicolon and control returns. + + function P_Accept_Statement return Node_Id is + Scan_State : Saved_Scan_State; + Accept_Node : Node_Id; + Hand_Seq : Node_Id; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Ecol := Start_Column; + + Accept_Node := New_Node (N_Accept_Statement, Token_Ptr); + Scan; -- past ACCEPT + Scope.Table (Scope.Last).Labl := Token_Node; + + Set_Entry_Direct_Name (Accept_Node, P_Identifier); + + -- Left paren could be (Entry_Index) or Formal_Part, determine which + + if Token = Tok_Left_Paren then + Save_Scan_State (Scan_State); -- at left paren + Scan; -- past left paren + + -- If first token after left paren not identifier, then Entry_Index + + if Token /= Tok_Identifier then + Set_Entry_Index (Accept_Node, P_Expression); + T_Right_Paren; + Set_Parameter_Specifications (Accept_Node, P_Parameter_Profile); + + -- First token after left paren is identifier, could be either case + + else -- Token = Tok_Identifier + Scan; -- past identifier + + -- If identifier followed by comma or colon, must be Formal_Part + + if Token = Tok_Comma or else Token = Tok_Colon then + Restore_Scan_State (Scan_State); -- to left paren + Set_Parameter_Specifications (Accept_Node, P_Parameter_Profile); + + -- If identifier not followed by comma/colon, must be entry index + + else + Restore_Scan_State (Scan_State); -- to left paren + Scan; -- past left paren (again!) + Set_Entry_Index (Accept_Node, P_Expression); + T_Right_Paren; + Set_Parameter_Specifications (Accept_Node, P_Parameter_Profile); + end if; + end if; + end if; + + -- Scan out DO if present + + if Token = Tok_Do then + Scope.Table (Scope.Last).Etyp := E_Name; + Scope.Table (Scope.Last).Lreq := False; + Scan; -- past DO + Hand_Seq := P_Handled_Sequence_Of_Statements; + Set_Handled_Statement_Sequence (Accept_Node, Hand_Seq); + End_Statements (Handled_Statement_Sequence (Accept_Node)); + + -- Exception handlers not allowed in Ada 95 node + + if Present (Exception_Handlers (Hand_Seq)) then + if Ada_83 then + Error_Msg_N + ("(Ada 83) exception handlers in accept not allowed", + First_Non_Pragma (Exception_Handlers (Hand_Seq))); + end if; + end if; + + else + Pop_Scope_Stack; -- discard unused entry + TF_Semicolon; + end if; + + return Accept_Node; + + -- If error, resynchronize past semicolon + + exception + when Error_Resync => + Resync_Past_Semicolon; + return Error; + + end P_Accept_Statement; + + ------------------------ + -- 9.5.2 Entry Index -- + ------------------------ + + -- Parsed by P_Expression (4.4) + + ----------------------- + -- 9.5.2 Entry Body -- + ----------------------- + + -- ENTRY_BODY ::= + -- entry DEFINING_IDENTIFIER ENTRY_BODY_FORMAL_PART ENTRY_BARRIER is + -- DECLARATIVE_PART + -- begin + -- HANDLED_SEQUENCE_OF_STATEMENTS + -- end [entry_IDENTIFIER]; + + -- The caller has checked that the initial token is ENTRY + + -- Error_Recovery: cannot raise Error_Resync + + function P_Entry_Body return Node_Id is + Entry_Node : Node_Id; + Formal_Part_Node : Node_Id; + Name_Node : Node_Id; + + begin + Push_Scope_Stack; + Entry_Node := New_Node (N_Entry_Body, Token_Ptr); + Scan; -- past ENTRY + + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Lreq := False; + Scope.Table (Scope.Last).Etyp := E_Name; + + Name_Node := P_Defining_Identifier; + Set_Defining_Identifier (Entry_Node, Name_Node); + Scope.Table (Scope.Last).Labl := Name_Node; + + Formal_Part_Node := P_Entry_Body_Formal_Part; + Set_Entry_Body_Formal_Part (Entry_Node, Formal_Part_Node); + + Set_Condition (Formal_Part_Node, P_Entry_Barrier); + Parse_Decls_Begin_End (Entry_Node); + return Entry_Node; + end P_Entry_Body; + + ----------------------------------- + -- 9.5.2 Entry Body Formal Part -- + ----------------------------------- + + -- ENTRY_BODY_FORMAL_PART ::= + -- [(ENTRY_INDEX_SPECIFICATION)] [PARAMETER_PART] + + -- Error_Recovery: cannot raise Error_Resync + + function P_Entry_Body_Formal_Part return Node_Id is + Fpart_Node : Node_Id; + Scan_State : Saved_Scan_State; + + begin + Fpart_Node := New_Node (N_Entry_Body_Formal_Part, Token_Ptr); + + -- See if entry index specification present, and if so parse it + + if Token = Tok_Left_Paren then + Save_Scan_State (Scan_State); -- at left paren + Scan; -- past left paren + + if Token = Tok_For then + Set_Entry_Index_Specification + (Fpart_Node, P_Entry_Index_Specification); + T_Right_Paren; + else + Restore_Scan_State (Scan_State); -- to left paren + end if; + + -- Check for (common?) case of left paren omitted before FOR. This + -- is a tricky case, because the corresponding missing left paren + -- can cause real havoc if a formal part is present which gets + -- treated as part of the discrete subtype definition of the + -- entry index specification, so just give error and resynchronize + + elsif Token = Tok_For then + T_Left_Paren; -- to give error message + Resync_To_When; + end if; + + Set_Parameter_Specifications (Fpart_Node, P_Parameter_Profile); + return Fpart_Node; + end P_Entry_Body_Formal_Part; + + -------------------------- + -- 9.5.2 Entry Barrier -- + -------------------------- + + -- ENTRY_BARRIER ::= when CONDITION + + -- Error_Recovery: cannot raise Error_Resync + + function P_Entry_Barrier return Node_Id is + Bnode : Node_Id; + + begin + if Token = Tok_When then + Scan; -- past WHEN; + Bnode := P_Expression_No_Right_Paren; + + if Token = Tok_Colon_Equal then + Error_Msg_SC (""":="" should be ""="""); + Scan; + Bnode := P_Expression_No_Right_Paren; + end if; + + else + T_When; -- to give error message + Bnode := Error; + end if; + + TF_Is; + return Bnode; + end P_Entry_Barrier; + + -------------------------------------- + -- 9.5.2 Entry Index Specification -- + -------------------------------------- + + -- ENTRY_INDEX_SPECIFICATION ::= + -- for DEFINING_IDENTIFIER in DISCRETE_SUBTYPE_DEFINITION + + -- Error recovery: can raise Error_Resync + + function P_Entry_Index_Specification return Node_Id is + Iterator_Node : Node_Id; + + begin + Iterator_Node := New_Node (N_Entry_Index_Specification, Token_Ptr); + T_For; -- past FOR + Set_Defining_Identifier (Iterator_Node, P_Defining_Identifier); + T_In; + Set_Discrete_Subtype_Definition + (Iterator_Node, P_Discrete_Subtype_Definition); + return Iterator_Node; + end P_Entry_Index_Specification; + + --------------------------------- + -- 9.5.3 Entry Call Statement -- + --------------------------------- + + -- Parsed by P_Name (4.1). Within a select, an entry call is parsed + -- by P_Select_Statement (9.7) + + ------------------------------ + -- 9.5.4 Requeue Statement -- + ------------------------------ + + -- REQUEUE_STATEMENT ::= requeue entry_NAME [with abort]; + + -- The caller has checked that the initial token is requeue + + -- Error recovery: can raise Error_Resync + + function P_Requeue_Statement return Node_Id is + Requeue_Node : Node_Id; + + begin + Requeue_Node := New_Node (N_Requeue_Statement, Token_Ptr); + Scan; -- past REQUEUE + Set_Name (Requeue_Node, P_Name); + + if Token = Tok_With then + Scan; -- past WITH + T_Abort; + Set_Abort_Present (Requeue_Node, True); + end if; + + TF_Semicolon; + return Requeue_Node; + end P_Requeue_Statement; + + -------------------------- + -- 9.6 Delay Statement -- + -------------------------- + + -- DELAY_STATEMENT ::= + -- DELAY_UNTIL_STATEMENT + -- | DELAY_RELATIVE_STATEMENT + + -- The caller has checked that the initial token is DELAY + + -- Error recovery: cannot raise Error_Resync + + function P_Delay_Statement return Node_Id is + begin + Scan; -- past DELAY + + -- The following check for delay until misused in Ada 83 doesn't catch + -- all cases, but it's good enough to catch most of them! + + if Token_Name = Name_Until then + Check_95_Keyword (Tok_Until, Tok_Left_Paren); + Check_95_Keyword (Tok_Until, Tok_Identifier); + end if; + + if Token = Tok_Until then + return P_Delay_Until_Statement; + else + return P_Delay_Relative_Statement; + end if; + end P_Delay_Statement; + + -------------------------------- + -- 9.6 Delay Until Statement -- + -------------------------------- + + -- DELAY_UNTIL_STATEMENT ::= delay until delay_EXPRESSION; + + -- The caller has checked that the initial token is DELAY, scanned it + -- out and checked that the current token is UNTIL + + -- Error recovery: cannot raise Error_Resync + + function P_Delay_Until_Statement return Node_Id is + Delay_Node : Node_Id; + + begin + Delay_Node := New_Node (N_Delay_Until_Statement, Prev_Token_Ptr); + Scan; -- past UNTIL + Set_Expression (Delay_Node, P_Expression_No_Right_Paren); + TF_Semicolon; + return Delay_Node; + end P_Delay_Until_Statement; + + ----------------------------------- + -- 9.6 Delay Relative Statement -- + ----------------------------------- + + -- DELAY_RELATIVE_STATEMENT ::= delay delay_EXPRESSION; + + -- The caller has checked that the initial token is DELAY, scanned it + -- out and determined that the current token is not UNTIL + + -- Error recovery: cannot raise Error_Resync + + function P_Delay_Relative_Statement return Node_Id is + Delay_Node : Node_Id; + + begin + Delay_Node := New_Node (N_Delay_Relative_Statement, Prev_Token_Ptr); + Set_Expression (Delay_Node, P_Expression_No_Right_Paren); + Check_Simple_Expression_In_Ada_83 (Expression (Delay_Node)); + TF_Semicolon; + return Delay_Node; + end P_Delay_Relative_Statement; + + --------------------------- + -- 9.7 Select Statement -- + --------------------------- + + -- SELECT_STATEMENT ::= + -- SELECTIVE_ACCEPT + -- | TIMED_ENTRY_CALL + -- | CONDITIONAL_ENTRY_CALL + -- | ASYNCHRONOUS_SELECT + + -- SELECTIVE_ACCEPT ::= + -- select + -- [GUARD] + -- SELECT_ALTERNATIVE + -- {or + -- [GUARD] + -- SELECT_ALTERNATIVE + -- [else + -- SEQUENCE_OF_STATEMENTS] + -- end select; + + -- GUARD ::= when CONDITION => + + -- Note: the guard preceding a select alternative is included as part + -- of the node generated for a selective accept alternative. + + -- SELECT_ALTERNATIVE ::= + -- ACCEPT_ALTERNATIVE + -- | DELAY_ALTERNATIVE + -- | TERMINATE_ALTERNATIVE + + -- TIMED_ENTRY_CALL ::= + -- select + -- ENTRY_CALL_ALTERNATIVE + -- or + -- DELAY_ALTERNATIVE + -- end select; + + -- CONDITIONAL_ENTRY_CALL ::= + -- select + -- ENTRY_CALL_ALTERNATIVE + -- else + -- SEQUENCE_OF_STATEMENTS + -- end select; + + -- ENTRY_CALL_ALTERNATIVE ::= + -- ENTRY_CALL_STATEMENT [SEQUENCE_OF_STATEMENTS] + + -- ASYNCHRONOUS_SELECT ::= + -- select + -- TRIGGERING_ALTERNATIVE + -- then abort + -- ABORTABLE_PART + -- end select; + + -- TRIGGERING_ALTERNATIVE ::= + -- TRIGGERING_STATEMENT [SEQUENCE_OF_STATEMENTS] + + -- TRIGGERING_STATEMENT ::= ENTRY_CALL_STATEMENT | DELAY_STATEMENT + + -- The caller has checked that the initial token is SELECT + + -- Error recovery: can raise Error_Resync + + function P_Select_Statement return Node_Id is + Select_Node : Node_Id; + Select_Sloc : Source_Ptr; + Stmnt_Sloc : Source_Ptr; + Ecall_Node : Node_Id; + Alternative : Node_Id; + Select_Pragmas : List_Id; + Alt_Pragmas : List_Id; + Statement_List : List_Id; + Alt_List : List_Id; + Cond_Expr : Node_Id; + Delay_Stmnt : Node_Id; + + begin + Push_Scope_Stack; + Scope.Table (Scope.Last).Etyp := E_Select; + Scope.Table (Scope.Last).Ecol := Start_Column; + Scope.Table (Scope.Last).Sloc := Token_Ptr; + Scope.Table (Scope.Last).Labl := Error; + + Select_Sloc := Token_Ptr; + Scan; -- past SELECT + Stmnt_Sloc := Token_Ptr; + Select_Pragmas := P_Pragmas_Opt; + + -- If first token after select is designator, then we have an entry + -- call, which must be the start of a conditional entry call, timed + -- entry call or asynchronous select + + if Token in Token_Class_Desig then + + -- Scan entry call statement + + begin + Ecall_Node := P_Name; + + -- ?? The following two clauses exactly parallel code in ch5 + -- and should be commoned sometime + + if Nkind (Ecall_Node) = N_Indexed_Component then + declare + Prefix_Node : Node_Id := Prefix (Ecall_Node); + Exprs_Node : List_Id := Expressions (Ecall_Node); + begin + Change_Node (Ecall_Node, N_Procedure_Call_Statement); + Set_Name (Ecall_Node, Prefix_Node); + Set_Parameter_Associations (Ecall_Node, Exprs_Node); + end; + + elsif Nkind (Ecall_Node) = N_Function_Call then + declare + Fname_Node : Node_Id := Name (Ecall_Node); + Params_List : List_Id := Parameter_Associations (Ecall_Node); + + begin + Change_Node (Ecall_Node, N_Procedure_Call_Statement); + Set_Name (Ecall_Node, Fname_Node); + Set_Parameter_Associations (Ecall_Node, Params_List); + end; + + elsif Nkind (Ecall_Node) = N_Identifier + or else Nkind (Ecall_Node) = N_Selected_Component + then + -- Case of a call to a parameterless entry. + + declare + C_Node : constant Node_Id := + New_Node (N_Procedure_Call_Statement, Stmnt_Sloc); + begin + Set_Name (C_Node, Ecall_Node); + Set_Parameter_Associations (C_Node, No_List); + Ecall_Node := C_Node; + end; + end if; + + TF_Semicolon; + + exception + when Error_Resync => + Resync_Past_Semicolon; + return Error; + end; + + Statement_List := P_Sequence_Of_Statements (SS_Eltm_Ortm_Tatm); + + -- OR follows, we have a timed entry call + + if Token = Tok_Or then + Scan; -- past OR + Alt_Pragmas := P_Pragmas_Opt; + + Select_Node := New_Node (N_Timed_Entry_Call, Select_Sloc); + Set_Entry_Call_Alternative (Select_Node, + Make_Entry_Call_Alternative (Stmnt_Sloc, + Entry_Call_Statement => Ecall_Node, + Pragmas_Before => Select_Pragmas, + Statements => Statement_List)); + + -- Only possibility is delay alternative. If we have anything + -- else, give message, and treat as conditional entry call. + + if Token /= Tok_Delay then + Error_Msg_SC + ("only allowed alternative in timed entry call is delay!"); + Discard_Junk_List (P_Sequence_Of_Statements (SS_Sreq)); + Set_Delay_Alternative (Select_Node, Error); + + else + Set_Delay_Alternative (Select_Node, P_Delay_Alternative); + Set_Pragmas_Before + (Delay_Alternative (Select_Node), Alt_Pragmas); + end if; + + -- ELSE follows, we have a conditional entry call + + elsif Token = Tok_Else then + Scan; -- past ELSE + Select_Node := New_Node (N_Conditional_Entry_Call, Select_Sloc); + + Set_Entry_Call_Alternative (Select_Node, + Make_Entry_Call_Alternative (Stmnt_Sloc, + Entry_Call_Statement => Ecall_Node, + Pragmas_Before => Select_Pragmas, + Statements => Statement_List)); + + Set_Else_Statements + (Select_Node, P_Sequence_Of_Statements (SS_Sreq)); + + -- Only remaining case is THEN ABORT (asynchronous select) + + elsif Token = Tok_Abort then + Select_Node := + Make_Asynchronous_Select (Select_Sloc, + Triggering_Alternative => + Make_Triggering_Alternative (Stmnt_Sloc, + Triggering_Statement => Ecall_Node, + Pragmas_Before => Select_Pragmas, + Statements => Statement_List), + Abortable_Part => P_Abortable_Part); + + -- Else error + + else + if Ada_83 then + Error_Msg_BC ("OR or ELSE expected"); + else + Error_Msg_BC ("OR or ELSE or THEN ABORT expected"); + end if; + + Select_Node := Error; + end if; + + End_Statements; + + -- Here we have a selective accept or an an asynchronous select (first + -- token after SELECT is other than a designator token). + + else + -- If we have delay with no guard, could be asynchronous select + + if Token = Tok_Delay then + Delay_Stmnt := P_Delay_Statement; + Statement_List := P_Sequence_Of_Statements (SS_Eltm_Ortm_Tatm); + + -- Asynchronous select + + if Token = Tok_Abort then + Select_Node := + Make_Asynchronous_Select (Select_Sloc, + Triggering_Alternative => + Make_Triggering_Alternative (Stmnt_Sloc, + Triggering_Statement => Delay_Stmnt, + Pragmas_Before => Select_Pragmas, + Statements => Statement_List), + Abortable_Part => P_Abortable_Part); + + End_Statements; + return Select_Node; + + -- Delay which was not an asynchronous select. Must be a selective + -- accept, and since at least one accept statement is required, + -- we must have at least one OR phrase present. + + else + Alt_List := New_List ( + Make_Delay_Alternative (Stmnt_Sloc, + Delay_Statement => Delay_Stmnt, + Pragmas_Before => Select_Pragmas, + Statements => Statement_List)); + T_Or; + Alt_Pragmas := P_Pragmas_Opt; + end if; + + -- If not a delay statement, then must be another possibility for + -- a selective accept alternative, or perhaps a guard is present + + else + Alt_List := New_List; + Alt_Pragmas := Select_Pragmas; + end if; + + Select_Node := New_Node (N_Selective_Accept, Select_Sloc); + Set_Select_Alternatives (Select_Node, Alt_List); + + -- Scan out selective accept alternatives. On entry to this loop, + -- we are just past a SELECT or OR token, and any pragmas that + -- immediately follow the SELECT or OR are in Alt_Pragmas. + + loop + if Token = Tok_When then + + if Present (Alt_Pragmas) then + Error_Msg_SC ("pragmas may not precede guard"); + end if; + + Scan; -- past WHEN + Cond_Expr := P_Expression_No_Right_Paren; + T_Arrow; + Alt_Pragmas := P_Pragmas_Opt; + + else + Cond_Expr := Empty; + end if; + + if Token = Tok_Accept then + Alternative := P_Accept_Alternative; + + -- Check for junk attempt at asynchronous select using + -- an Accept alternative as the triggering statement + + if Token = Tok_Abort + and then Is_Empty_List (Alt_List) + and then No (Cond_Expr) + then + Error_Msg + ("triggering statement must be entry call or delay", + Sloc (Alternative)); + Scan; -- past junk ABORT + Discard_Junk_List (P_Sequence_Of_Statements (SS_Sreq)); + End_Statements; + return Error; + end if; + + elsif Token = Tok_Delay then + Alternative := P_Delay_Alternative; + + elsif Token = Tok_Terminate then + Alternative := P_Terminate_Alternative; + + else + Error_Msg_SC + ("Select alternative (ACCEPT, ABORT, DELAY) expected"); + Alternative := Error; + + if Token = Tok_Semicolon then + Scan; -- past junk semicolon + end if; + end if; + + -- THEN ABORT at this stage is just junk + + if Token = Tok_Abort then + Error_Msg_SP ("misplaced `THEN ABORT`"); + Scan; -- past junk ABORT + Discard_Junk_List (P_Sequence_Of_Statements (SS_Sreq)); + End_Statements; + return Error; + + else + if Alternative /= Error then + Set_Condition (Alternative, Cond_Expr); + Set_Pragmas_Before (Alternative, Alt_Pragmas); + Append (Alternative, Alt_List); + end if; + + exit when Token /= Tok_Or; + end if; + + T_Or; + Alt_Pragmas := P_Pragmas_Opt; + end loop; + + if Token = Tok_Else then + Scan; -- past ELSE + Set_Else_Statements + (Select_Node, P_Sequence_Of_Statements (SS_Ortm_Sreq)); + + if Token = Tok_Or then + Error_Msg_SC ("select alternative cannot follow else part!"); + end if; + end if; + + End_Statements; + end if; + + return Select_Node; + end P_Select_Statement; + + ----------------------------- + -- 9.7.1 Selective Accept -- + ----------------------------- + + -- Parsed by P_Select_Statement (9.7) + + ------------------ + -- 9.7.1 Guard -- + ------------------ + + -- Parsed by P_Select_Statement (9.7) + + ------------------------------- + -- 9.7.1 Select Alternative -- + ------------------------------- + + -- SELECT_ALTERNATIVE ::= + -- ACCEPT_ALTERNATIVE + -- | DELAY_ALTERNATIVE + -- | TERMINATE_ALTERNATIVE + + -- Note: the guard preceding a select alternative is included as part + -- of the node generated for a selective accept alternative. + + -- Error recovery: cannot raise Error_Resync + + ------------------------------- + -- 9.7.1 Accept Alternative -- + ------------------------------- + + -- ACCEPT_ALTERNATIVE ::= + -- ACCEPT_STATEMENT [SEQUENCE_OF_STATEMENTS] + + -- Error_Recovery: Cannot raise Error_Resync + + -- Note: the caller is responsible for setting the Pragmas_Before + -- field of the returned N_Terminate_Alternative node. + + function P_Accept_Alternative return Node_Id is + Accept_Alt_Node : Node_Id; + + begin + Accept_Alt_Node := New_Node (N_Accept_Alternative, Token_Ptr); + Set_Accept_Statement (Accept_Alt_Node, P_Accept_Statement); + + -- Note: the reason that we accept THEN ABORT as a terminator for + -- the sequence of statements is for error recovery which allows + -- for misuse of an accept statement as a triggering statememt. + + Set_Statements + (Accept_Alt_Node, P_Sequence_Of_Statements (SS_Eltm_Ortm_Tatm)); + return Accept_Alt_Node; + end P_Accept_Alternative; + + ------------------------------ + -- 9.7.1 Delay Alternative -- + ------------------------------ + + -- DELAY_ALTERNATIVE ::= + -- DELAY_STATEMENT [SEQUENCE_OF_STATEMENTS] + + -- Error_Recovery: Cannot raise Error_Resync + + -- Note: the caller is responsible for setting the Pragmas_Before + -- field of the returned N_Terminate_Alternative node. + + function P_Delay_Alternative return Node_Id is + Delay_Alt_Node : Node_Id; + + begin + Delay_Alt_Node := New_Node (N_Delay_Alternative, Token_Ptr); + Set_Delay_Statement (Delay_Alt_Node, P_Delay_Statement); + + -- Note: the reason that we accept THEN ABORT as a terminator for + -- the sequence of statements is for error recovery which allows + -- for misuse of an accept statement as a triggering statememt. + + Set_Statements + (Delay_Alt_Node, P_Sequence_Of_Statements (SS_Eltm_Ortm_Tatm)); + return Delay_Alt_Node; + end P_Delay_Alternative; + + ---------------------------------- + -- 9.7.1 Terminate Alternative -- + ---------------------------------- + + -- TERMINATE_ALTERNATIVE ::= terminate; + + -- Error_Recovery: Cannot raise Error_Resync + + -- Note: the caller is responsible for setting the Pragmas_Before + -- field of the returned N_Terminate_Alternative node. + + function P_Terminate_Alternative return Node_Id is + Terminate_Alt_Node : Node_Id; + + begin + Terminate_Alt_Node := New_Node (N_Terminate_Alternative, Token_Ptr); + Scan; -- past TERMINATE + TF_Semicolon; + + -- For all other select alternatives, the sequence of statements + -- after the alternative statement will swallow up any pragmas + -- coming in this position. But the terminate alternative has no + -- sequence of statements, so the pragmas here must be treated + -- specially. + + Set_Pragmas_After (Terminate_Alt_Node, P_Pragmas_Opt); + return Terminate_Alt_Node; + end P_Terminate_Alternative; + + ----------------------------- + -- 9.7.2 Timed Entry Call -- + ----------------------------- + + -- Parsed by P_Select_Statement (9.7) + + ----------------------------------- + -- 9.7.2 Entry Call Alternative -- + ----------------------------------- + + -- Parsed by P_Select_Statement (9.7) + + ----------------------------------- + -- 9.7.3 Conditional Entry Call -- + ----------------------------------- + + -- Parsed by P_Select_Statement (9.7) + + -------------------------------- + -- 9.7.4 Asynchronous Select -- + -------------------------------- + + -- Parsed by P_Select_Statement (9.7) + + ----------------------------------- + -- 9.7.4 Triggering Alternative -- + ----------------------------------- + + -- Parsed by P_Select_Statement (9.7) + + --------------------------------- + -- 9.7.4 Triggering Statement -- + --------------------------------- + + -- Parsed by P_Select_Statement (9.7) + + --------------------------- + -- 9.7.4 Abortable Part -- + --------------------------- + + -- ABORTABLE_PART ::= SEQUENCE_OF_STATEMENTS + + -- The caller has verified that THEN ABORT is present, and Token is + -- pointing to the ABORT on entry (or if not, then we have an error) + + -- Error recovery: cannot raise Error_Resync + + function P_Abortable_Part return Node_Id is + Abortable_Part_Node : Node_Id; + + begin + Abortable_Part_Node := New_Node (N_Abortable_Part, Token_Ptr); + T_Abort; -- scan past ABORT + + if Ada_83 then + Error_Msg_SP ("(Ada 83) asynchronous select not allowed!"); + end if; + + Set_Statements (Abortable_Part_Node, P_Sequence_Of_Statements (SS_Sreq)); + return Abortable_Part_Node; + end P_Abortable_Part; + + -------------------------- + -- 9.8 Abort Statement -- + -------------------------- + + -- ABORT_STATEMENT ::= abort task_NAME {, task_NAME}; + + -- The caller has checked that the initial token is ABORT + + -- Error recovery: cannot raise Error_Resync + + function P_Abort_Statement return Node_Id is + Abort_Node : Node_Id; + + begin + Abort_Node := New_Node (N_Abort_Statement, Token_Ptr); + Scan; -- past ABORT + Set_Names (Abort_Node, New_List); + + loop + Append (P_Name, Names (Abort_Node)); + exit when Token /= Tok_Comma; + Scan; -- past comma + end loop; + + TF_Semicolon; + return Abort_Node; + end P_Abort_Statement; + +end Ch9; diff --git a/gcc/ada/par-endh.adb b/gcc/ada/par-endh.adb new file mode 100644 index 00000000000..fa5b8c20a1a --- /dev/null +++ b/gcc/ada/par-endh.adb @@ -0,0 +1,1191 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . E N D H -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.61 $ +-- -- +-- 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 Stringt; use Stringt; +with Uintp; use Uintp; + +with GNAT.Spelling_Checker; use GNAT.Spelling_Checker; + +separate (Par) +package body Endh is + + ---------------- + -- Local Data -- + ---------------- + + type End_Action_Type is ( + -- Type used to describe the result of the Pop_End_Context call + + Accept_As_Scanned, + -- Current end sequence is entirely c correct. In this case Token and + -- the scan pointer are left pointing past the end sequence (i.e. they + -- are unchanged from the values set on entry to Pop_End_Context). + + Insert_And_Accept, + -- Current end sequence is to be left in place to satisfy some outer + -- scope. Token and the scan pointer are set to point to the end + -- token, and should be left there. A message has been generated + -- indicating a missing end sequence. This status is also used for + -- the case when no end token is present. + + Skip_And_Accept, + -- The end sequence is incorrect (and an error message has been + -- posted), but it will still be accepted. In this case Token and + -- the scan pointer point back to the end token, and the caller + -- should skip past the end sequence before proceeding. + + Skip_And_Reject); + -- The end sequence is judged to belong to an unrecognized inner + -- scope. An appropriate message has been issued and the caller + -- should skip past the end sequence and then proceed as though + -- no end sequence had been encountered. + + End_Action : End_Action_Type; + -- The variable set by Pop_End_Context call showing which of the four + -- decisions described above is judged the best. + + End_Sloc : Source_Ptr; + -- Source location of END token + + End_OK : Boolean; + -- Set False if error is found in END line + + End_Column : Column_Number; + -- Column of END line + + End_Type : SS_End_Type; + -- Type of END expected. The special value E_Dummy is set to indicate that + -- no END token was present (so a missing END inserted message is needed) + + End_Labl : Node_Id; + -- Node_Id value for explicit name on END line, or for compiler supplied + -- name in the case where an optional name is not given. Empty if no name + -- appears. If non-empty, then it is either an N_Designator node for a + -- child unit or a node with a Chars field identifying the actual label. + + End_Labl_Present : Boolean; + -- Indicates that the value in End_Labl was for an explicit label. + + Syntax_OK : Boolean; + -- Set True if the entry is syntactically correct + + Token_OK : Boolean; + -- Set True if the keyword in the END sequence matches, or if neither + -- the END sequence nor the END stack entry has a keyword. + + Label_OK : Boolean; + -- Set True if both the END sequence and the END stack entry contained + -- labels (other than No_Name or Error_Name) and the labels matched. + -- This is a stronger condition than SYNTAX_OK, since it means that a + -- label was present, even in a case where it was optional. Note that + -- the case of no label required, and no label present does NOT set + -- Label_OK to True, it is True only if a positive label match is found. + + Column_OK : Boolean; + -- Column_OK is set True if the END sequence appears in the expected column + + Scan_State : Saved_Scan_State; + -- Save state at start of END sequence, in case we decide not to eat it up + + ----------------------- + -- Local Subprograms -- + ----------------------- + + procedure Evaluate_End_Entry (SS_Index : Int); + -- Compare scanned END entry (as recorded by a prior call to P_End_Scan) + -- with a specified entry in the scope stack (the single parameter is the + -- entry index in the scope stack). Note that Scan is not called. The above + -- variables xxx_OK are set to indicate the result of the evaluation. + + procedure Output_End_Deleted; + -- Output a message complaining that the current END structure does not + -- match anything and is being deleted. + + procedure Output_End_Expected (Ins : Boolean); + -- Output a message at the start of the current token which is always an + -- END, complaining that the END is not of the right form. The message + -- indicates the expected form. The information for the message is taken + -- from the top entry in the scope stack. The Ins parameter is True if + -- an end is being inserted, and false if an existing end is being + -- replaced. Note that in the case of a suspicious IS for the Ins case, + -- we do not output the message, but instead simply mark the scope stack + -- entry as being a case of a bad IS. + + procedure Output_End_Missing; + -- Output a message just before the current token, complaining that the + -- END is not of the right form. The message indicates the expected form. + -- The information for the message is taken from the top entry in the + -- scope stack. Note that in the case of a suspicious IS, we do not output + -- the message, but instead simply mark the scope stack entry as a bad IS. + + procedure Pop_End_Context; + -- Pop_End_Context is called after processing a construct, to pop the + -- top entry off the end stack. It decides on the appropriate action to + -- to take, signalling the result by setting End_Action as described in + -- the global variable section. + + function Same_Label (Label1, Label2 : Node_Id) return Boolean; + -- This function compares the two names associated with the given nodes. + -- If they are both simple (i.e. have Chars fields), then they have to + -- be the same name. Otherwise they must both be N_Selected_Component + -- nodes, referring to the same set of names, or Label1 is an N_Designator + -- referring to the same set of names as the N_Defining_Program_Unit_Name + -- in Label2. Any other combination returns False. This routine is used + -- to compare the End_Labl scanned from the End line with the saved label + -- value in the scope stack. + + --------------- + -- Check_End -- + --------------- + + function Check_End return Boolean is + Name_On_Separate_Line : Boolean; + -- Set True if the name on an END line is on a separate source line + -- from the END. This is highly suspicious, but is allowed. The point + -- is that we want to make sure that we don't just have a missing + -- semicolon misleading us into swallowing an identifier from the + -- following line. + + Name_Scan_State : Saved_Scan_State; + -- Save state at start of name if Name_On_Separate_Line is TRUE + + Span_Node : constant Node_Id := Scope.Table (Scope.Last).Node; + + begin + End_Labl_Present := False; + End_Labl := Empty; + + -- Our first task is to scan out the END sequence if one is present. + -- If none is present, signal by setting End_Type to E_Dummy. + + if Token /= Tok_End then + End_Type := E_Dummy; + + else + Save_Scan_State (Scan_State); -- at END + End_Sloc := Token_Ptr; + End_Column := Start_Column; + End_OK := True; + Scan; -- past END + + -- Set End_Span if expected. note that this will be useless + -- if we do not have the right ending keyword, but in this + -- case we have a malformed program anyway, and the setting + -- of End_Span will simply be unreliable in this case anyway. + + if Present (Span_Node) then + Set_End_Location (Span_Node, Token_Ptr); + end if; + + -- Cases of keywords where no label is allowed + + if Token = Tok_Case then + End_Type := E_Case; + Scan; -- past CASE + + elsif Token = Tok_If then + End_Type := E_If; + Scan; -- past IF + + elsif Token = Tok_Record then + End_Type := E_Record; + Scan; -- past RECORD + + elsif Token = Tok_Select then + End_Type := E_Select; + Scan; -- past SELECT + + -- Cases which do allow labels + + else + -- LOOP + + if Token = Tok_Loop then + Scan; -- past LOOP + End_Type := E_Loop; + + -- FOR or WHILE allowed (signalling error) to substitute for LOOP + -- if on the same line as the END + + elsif (Token = Tok_For or else Token = Tok_While) + and then not Token_Is_At_Start_Of_Line + then + Scan; -- past FOR or WHILE + End_Type := E_Loop; + End_OK := False; + + -- Cases with no keyword + + else + End_Type := E_Name; + end if; + + -- Now see if a name is present + + if Token = Tok_Identifier or else + Token = Tok_String_Literal or else + Token = Tok_Operator_Symbol + then + if Token_Is_At_Start_Of_Line then + Name_On_Separate_Line := True; + Save_Scan_State (Name_Scan_State); + else + Name_On_Separate_Line := False; + end if; + + End_Labl := P_Designator; + End_Labl_Present := True; + + -- We have now scanned out a name. Here is where we do a check + -- to catch the cases like: + -- + -- end loop + -- X := 3; + -- + -- where the missing semicolon might make us swallow up the X + -- as a bogus end label. In a situation like this, where the + -- apparent name is on a separate line, we accept it only if + -- it matches the label and is followed by a semicolon. + + if Name_On_Separate_Line then + if Token /= Tok_Semicolon or else + not Same_Label (End_Labl, Scope.Table (Scope.Last).Labl) + then + Restore_Scan_State (Name_Scan_State); + End_Labl := Empty; + End_Labl_Present := False; + end if; + end if; + + -- Here for case of name allowed, but no name present. We will + -- supply an implicit matching name, with source location set + -- to the scan location past the END token. + + else + End_Labl := Scope.Table (Scope.Last).Labl; + + if End_Labl > Empty_Or_Error then + + -- The task here is to construct a designator from the + -- opening label, with the components all marked as not + -- from source, and Is_End_Label set in the identifier + -- or operator symbol. The location for all components + -- is the curent token location. + + -- Case of child unit name + + if Nkind (End_Labl) = N_Defining_Program_Unit_Name then + declare + Eref : constant Node_Id := + Make_Identifier (Token_Ptr, + Chars => + Chars (Defining_Identifier (End_Labl))); + + function Copy_Name (N : Node_Id) return Node_Id; + -- Copies a selected component or identifier + + function Copy_Name (N : Node_Id) return Node_Id is + R : Node_Id; + + begin + if Nkind (N) = N_Selected_Component then + return + Make_Selected_Component (Token_Ptr, + Prefix => + Copy_Name (Prefix (N)), + Selector_Name => + Copy_Name (Selector_Name (N))); + + else + R := + Make_Identifier (Token_Ptr, + Chars => Chars (N)); + Set_Comes_From_Source (N, False); + return R; + end if; + end Copy_Name; + + begin + Set_Comes_From_Source (Eref, False); + + End_Labl := + Make_Designator (Token_Ptr, + Name => Copy_Name (Name (End_Labl)), + Identifier => Eref); + end; + + -- Simple identifier case + + elsif Nkind (End_Labl) = N_Defining_Identifier + or else Nkind (End_Labl) = N_Identifier + then + End_Labl := + Make_Identifier (Token_Ptr, + Chars => Chars (End_Labl)); + + elsif Nkind (End_Labl) = N_Defining_Operator_Symbol + or else Nkind (End_Labl) = N_Operator_Symbol + then + Get_Decoded_Name_String (Chars (End_Labl)); + + End_Labl := + Make_Operator_Symbol (Token_Ptr, + Chars => Chars (End_Labl), + Strval => String_From_Name_Buffer); + end if; + + Set_Comes_From_Source (End_Labl, False); + End_Labl_Present := False; + + -- Do style check for missing label + + if Style_Check + and then End_Type = E_Name + and then Present (Scope.Table (Scope.Last).Labl) + then + Style.No_End_Name (Scope.Table (Scope.Last).Labl); + end if; + end if; + end if; + end if; + + -- Except in case of END RECORD, semicolon must follow. For END + -- RECORD, a semicolon does follow, but it is part of a higher level + -- construct. In any case, a missing semicolon is not serious enough + -- to consider the END statement to be bad in the sense that we + -- are dealing with (i.e. to be suspicious that it is not in fact + -- the END statement we are looking for!) + + if End_Type /= E_Record then + if Token = Tok_Semicolon then + T_Semicolon; + + -- Semicolon is missing. If the missing semicolon is at the end + -- of the line, i.e. we are at the start of the line now, then + -- a missing semicolon gets flagged, but is not serious enough + -- to consider the END statement to be bad in the sense that we + -- are dealing with (i.e. to be suspicious that this END is not + -- the END statement we are looking for). + + -- Similarly, if we are at a colon, we flag it but a colon for + -- a semicolon is not serious enough to consider the END to be + -- incorrect. Same thing for a period in place of a semicolon. + + elsif Token_Is_At_Start_Of_Line + or else Token = Tok_Colon + or else Token = Tok_Dot + then + T_Semicolon; + + -- If the missing semicolon is not at the start of the line, + -- then we do consider the END line to be dubious in this sense. + + else + End_OK := False; + end if; + end if; + end if; + + -- Now we call the Pop_End_Context routine to get a recommendation + -- as to what should be done with the END sequence we have scanned. + + Pop_End_Context; + + -- Remaining action depends on End_Action set by Pop_End_Context + + case End_Action is + + -- Accept_As_Scanned. In this case, Pop_End_Context left Token + -- pointing past the last token of a syntactically correct END + + when Accept_As_Scanned => + + -- Syntactically correct included the possibility of a missing + -- semicolon. If we do have a missing semicolon, then we have + -- already given a message, but now we scan out possible rubbish + -- on the same line as the END + + while not Token_Is_At_Start_Of_Line + and then Prev_Token /= Tok_Record + and then Prev_Token /= Tok_Semicolon + and then Token /= Tok_End + and then Token /= Tok_EOF + loop + Scan; -- past junk + end loop; + + return True; + + -- Insert_And_Accept. In this case, Pop_End_Context has reset Token + -- to point to the start of the END sequence, and recommends that it + -- be left in place to satisfy an outer scope level END. This means + -- that we proceed as though an END were present, and leave the scan + -- pointer unchanged. + + when Insert_And_Accept => + return True; + + -- Skip_And_Accept. In this case, Pop_End_Context has reset Token + -- to point to the start of the END sequence. This END sequence is + -- syntactically incorrect, and an appropriate error message has + -- already been posted. Pop_End_Context recommends accepting the + -- END sequence as the one we want, so we skip past it and then + -- proceed as though an END were present. + + when Skip_And_Accept => + End_Skip; + return True; + + -- Skip_And_Reject. In this case, Pop_End_Context has reset Token + -- to point to the start of the END sequence. This END sequence is + -- syntactically incorrect, and an appropriate error message has + -- already been posted. Pop_End_Context recommends entirely ignoring + -- this END sequence, so we skip past it and then return False, since + -- as far as the caller is concerned, no END sequence is present. + + when Skip_And_Reject => + End_Skip; + return False; + end case; + end Check_End; + + -------------- + -- End Skip -- + -------------- + + -- This procedure skips past an END sequence. On entry Token contains + -- Tok_End, and we know that the END sequence is syntactically incorrect, + -- and that an appropriate error message has already been posted. The + -- mission is simply to position the scan pointer to be the best guess of + -- the position after the END sequence. We do not issue any additional + -- error messages while carrying this out. + + -- Error recovery: does not raise Error_Resync + + procedure End_Skip is + begin + Scan; -- past END + + -- If the scan past the END leaves us on the next line, that's probably + -- where we should quit the scan, since it is likely that what we have + -- is a missing semicolon. Consider the following: + + -- END + -- Process_Input; + + -- This will have looked like a syntactically valid END sequence to the + -- initial scan of the END, but subsequent checking will have determined + -- that the label Process_Input is not an appropriate label. The real + -- error is a missing semicolon after the END, and by leaving the scan + -- pointer just past the END, we will improve the error recovery. + + if Token_Is_At_Start_Of_Line then + return; + end if; + + -- If there is a semicolon after the END, scan it out and we are done + + if Token = Tok_Semicolon then + T_Semicolon; + return; + end if; + + -- Otherwise skip past a token after the END on the same line. Note + -- that we do not eat a token on the following line since it seems + -- very unlikely in any case that the END gets separated from its + -- token, and we do not want to swallow up a keyword that starts a + -- legitimate construct following the bad END. + + if not Token_Is_At_Start_Of_Line + and then + + -- Cases of normal tokens following an END + + (Token = Tok_Case or else + Token = Tok_For or else + Token = Tok_If or else + Token = Tok_Loop or else + Token = Tok_Record or else + Token = Tok_Select or else + + -- Cases of bogus keywords ending loops + + Token = Tok_For or else + Token = Tok_While or else + + -- Cases of operator symbol names without quotes + + Token = Tok_Abs or else + Token = Tok_And or else + Token = Tok_Mod or else + Token = Tok_Not or else + Token = Tok_Or or else + Token = Tok_Xor) + + then + Scan; -- past token after END + + -- If that leaves us on the next line, then we are done. This is the + -- same principle described above for the case of END at line end + + if Token_Is_At_Start_Of_Line then + return; + + -- If we just scanned out record, then we are done, since the + -- semicolon after END RECORD is not part of the END sequence + + elsif Prev_Token = Tok_Record then + return; + + -- If we have a semicolon, scan it out and we are done + + elsif Token = Tok_Semicolon then + T_Semicolon; + return; + end if; + end if; + + -- Check for a label present on the same line + + loop + if Token_Is_At_Start_Of_Line then + return; + end if; + + if Token /= Tok_Identifier + and then Token /= Tok_Operator_Symbol + and then Token /= Tok_String_Literal + then + exit; + end if; + + Scan; -- past identifier, operator symbol or string literal + + if Token_Is_At_Start_Of_Line then + return; + elsif Token = Tok_Dot then + Scan; -- past dot + end if; + end loop; + + -- Skip final semicolon + + if Token = Tok_Semicolon then + T_Semicolon; + + -- If we don't have a final semicolon, skip until we either encounter + -- an END token, or a semicolon or the start of the next line. This + -- allows general junk to follow the end line (normally it is hard to + -- think that anyone will put anything deliberate here, and remember + -- that we know there is a missing semicolon in any case). We also + -- quite on an EOF (or else we would get stuck in an infinite loop + -- if there is no line end at the end of the last line of the file) + + else + while Token /= Tok_End + and then Token /= Tok_EOF + and then Token /= Tok_Semicolon + and then not Token_Is_At_Start_Of_Line + loop + Scan; -- past junk token on same line + end loop; + end if; + + return; + end End_Skip; + + -------------------- + -- End Statements -- + -------------------- + + -- This procedure is called when END is required or expected to terminate + -- a sequence of statements. The caller has already made an appropriate + -- entry on the scope stack to describe the expected form of the END. + -- End_Statements should only be used in cases where the only appropriate + -- terminator is END. + + -- Error recovery: cannot raise Error_Resync; + + procedure End_Statements (Parent : Node_Id := Empty) is + begin + -- This loop runs more than once in the case where Check_End rejects + -- the END sequence, as indicated by Check_End returning False. + + loop + if Check_End then + if Present (Parent) then + Set_End_Label (Parent, End_Labl); + end if; + + return; + end if; + + -- Extra statements past the bogus END are discarded. This is not + -- ideal for maximum error recovery, but it's too much trouble to + -- find an appropriate place to put them! + + Discard_Junk_List (P_Sequence_Of_Statements (SS_None)); + end loop; + end End_Statements; + + ------------------------ + -- Evaluate End Entry -- + ------------------------ + + procedure Evaluate_End_Entry (SS_Index : Int) is + begin + Column_OK := (End_Column = Scope.Table (SS_Index).Ecol); + + Token_OK := (End_Type = Scope.Table (SS_Index).Etyp or else + (End_Type = E_Name and then + Scope.Table (SS_Index).Etyp >= E_Name)); + + Label_OK := End_Labl_Present + and then + (Same_Label (End_Labl, Scope.Table (SS_Index).Labl) + or else Scope.Table (SS_Index).Labl = Error); + + -- Compute setting of Syntax_OK. We definitely have a syntax error + -- if the Token does not match properly or if P_End_Scan detected + -- a syntax error such as a missing semicolon. + + if not Token_OK or not End_OK then + Syntax_OK := False; + + -- Final check is that label is OK. Certainly it is OK if there + -- was an exact match on the label (the END label = the stack label) + + elsif Label_OK then + Syntax_OK := True; + + -- Case of label present + + elsif End_Labl_Present then + + -- If probably misspelling, then complain, and pretend it is OK + + declare + Nam : constant Node_Or_Entity_Id := Scope.Table (SS_Index).Labl; + + begin + if Nkind (End_Labl) in N_Has_Chars + and then Nkind (Nam) in N_Has_Chars + and then Chars (End_Labl) > Error_Name + and then Chars (Nam) > Error_Name + then + Get_Name_String (Chars (End_Labl)); + Error_Msg_Name_1 := Chars (Nam); + + if Error_Msg_Name_1 > Error_Name then + declare + S : String (1 .. Name_Len) := Name_Buffer (1 .. Name_Len); + + begin + Get_Name_String (Error_Msg_Name_1); + + if Is_Bad_Spelling_Of + (Name_Buffer (1 .. Name_Len), S) + then + Error_Msg_N ("misspelling of %", End_Labl); + Syntax_OK := True; + return; + end if; + end; + end if; + end if; + end; + + Syntax_OK := False; + + -- Otherwise we have cases of no label on the END line. For the loop + -- case, this is acceptable only if the loop is unlabeled. + + elsif End_Type = E_Loop then + Syntax_OK := (Scope.Table (SS_Index).Labl = Empty); + + -- Cases where a label is definitely allowed on the END line + + elsif End_Type = E_Name then + Syntax_OK := (Scope.Table (SS_Index).Labl = Empty or else + not Scope.Table (SS_Index).Lreq); + + -- Otherwise we have cases which don't allow labels anyway, so we + -- certainly accept an END which does not have a label. + + else + Syntax_OK := True; + end if; + end Evaluate_End_Entry; + + ------------------------ + -- Output End Deleted -- + ------------------------ + + procedure Output_End_Deleted is + begin + + if End_Type = E_Loop then + Error_Msg_SC ("no LOOP for this `END LOOP`!"); + + elsif End_Type = E_Case then + Error_Msg_SC ("no CASE for this `END CASE`"); + + elsif End_Type = E_If then + Error_Msg_SC ("no IF for this `END IF`!"); + + elsif End_Type = E_Record then + Error_Msg_SC ("no RECORD for this `END RECORD`!"); + + elsif End_Type = E_Select then + Error_Msg_SC ("no SELECT for this `END SELECT`!"); + + else + Error_Msg_SC ("no BEGIN for this END!"); + end if; + end Output_End_Deleted; + + ------------------------- + -- Output End Expected -- + ------------------------- + + procedure Output_End_Expected (Ins : Boolean) is + End_Type : SS_End_Type; + + begin + -- Suppress message if this was a potentially junk entry (e.g. a + -- record entry where no record keyword was present. + + if Scope.Table (Scope.Last).Junk then + return; + end if; + + End_Type := Scope.Table (Scope.Last).Etyp; + Error_Msg_Col := Scope.Table (Scope.Last).Ecol; + Error_Msg_Node_1 := Scope.Table (Scope.Last).Labl; + Error_Msg_Sloc := Scope.Table (Scope.Last).Sloc; + + -- Suppress message if error was posted on opening label + + if Present (Error_Msg_Node_1) + and then Error_Posted (Error_Msg_Node_1) + then + return; + end if; + + if End_Type = E_Case then + Error_Msg_SC ("`END CASE;` expected@ for CASE#!"); + + elsif End_Type = E_If then + Error_Msg_SC ("`END IF;` expected@ for IF#!"); + + elsif End_Type = E_Loop then + if Error_Msg_Node_1 = Empty then + Error_Msg_SC + ("`END LOOP;` expected@ for LOOP#!"); + else + Error_Msg_SC ("`END LOOP &;` expected@!"); + end if; + + elsif End_Type = E_Record then + Error_Msg_SC + ("`END RECORD;` expected@ for RECORD#!"); + + elsif End_Type = E_Select then + Error_Msg_SC + ("`END SELECT;` expected@ for SELECT#!"); + + -- All remaining cases are cases with a name (we do not treat + -- the suspicious is cases specially for a replaced end, only + -- for an inserted end). + + elsif End_Type = E_Name or else (not Ins) then + if Error_Msg_Node_1 = Empty then + Error_Msg_SC ("`END;` expected@ for BEGIN#!"); + else + Error_Msg_SC ("`END &;` expected@!"); + end if; + + -- The other possibility is a missing END for a subprogram with a + -- suspicious IS (that probably should have been a semicolon). The + -- Missing IS confirms the suspicion! + + else -- End_Type = E_Suspicious_Is or E_Bad_Is + Scope.Table (Scope.Last).Etyp := E_Bad_Is; + end if; + end Output_End_Expected; + + ------------------------ + -- Output End Missing -- + ------------------------ + + procedure Output_End_Missing is + End_Type : SS_End_Type; + + begin + -- Suppress message if this was a potentially junk entry (e.g. a + -- record entry where no record keyword was present. + + if Scope.Table (Scope.Last).Junk then + return; + end if; + + End_Type := Scope.Table (Scope.Last).Etyp; + Error_Msg_Node_1 := Scope.Table (Scope.Last).Labl; + Error_Msg_Sloc := Scope.Table (Scope.Last).Sloc; + + if End_Type = E_Case then + Error_Msg_BC ("missing `END CASE;` for CASE#!"); + + elsif End_Type = E_If then + Error_Msg_BC ("missing `END IF;` for IF#!"); + + elsif End_Type = E_Loop then + if Error_Msg_Node_1 = Empty then + Error_Msg_BC ("missing `END LOOP;` for LOOP#!"); + else + Error_Msg_BC ("missing `END LOOP &;`!"); + end if; + + elsif End_Type = E_Record then + Error_Msg_SC + ("missing `END RECORD;` for RECORD#!"); + + elsif End_Type = E_Select then + Error_Msg_BC + ("missing `END SELECT;` for SELECT#!"); + + elsif End_Type = E_Name then + if Error_Msg_Node_1 = Empty then + Error_Msg_BC ("missing `END;` for BEGIN#!"); + else + Error_Msg_BC ("missing `END &;`!"); + end if; + + else -- End_Type = E_Suspicious_Is or E_Bad_Is + Scope.Table (Scope.Last).Etyp := E_Bad_Is; + end if; + end Output_End_Missing; + + --------------------- + -- Pop End Context -- + --------------------- + + procedure Pop_End_Context is + + Pretty_Good : Boolean; + -- This flag is set True if the END sequence is syntactically incorrect, + -- but is (from a heuristic point of view), pretty likely to be simply + -- a misspelling of the intended END. + + Outer_Match : Boolean; + -- This flag is set True if we decide that the current END sequence + -- belongs to some outer level entry in the scope stack, and thus + -- we will NOT eat it up in matching the current expected END. + + begin + -- If not at END, then output END expected message + + if End_Type = E_Dummy then + Output_End_Missing; + Pop_Scope_Stack; + End_Action := Insert_And_Accept; + return; + + -- Otherwise we do have an END present + + else + -- A special check. If we have END; followed by an end of file, + -- WITH or SEPARATE, then if we are not at the outer level, then + -- we have a sytax error. Consider the example: + + -- ... + -- declare + -- X : Integer; + -- begin + -- X := Father (A); + -- Process (X, X); + -- end; + -- with Package1; + -- ... + + -- Now the END; here is a syntactically correct closer for the + -- declare block, but if we eat it up, then we obviously have + -- a missing END for the outer context (since WITH can only appear + -- at the outer level. + + -- In this situation, we always reserve the END; for the outer level, + -- even if it is in the wrong column. This is because it's much more + -- useful to have the error message point to the DECLARE than to the + -- package header in this case. + + -- We also reserve an end with a name before the end of file if the + -- name is the one we expect at the outer level. + + if (Token = Tok_EOF or else + Token = Tok_With or else + Token = Tok_Separate) + and then End_Type >= E_Name + and then (not End_Labl_Present + or else Same_Label (End_Labl, Scope.Table (1).Labl)) + and then Scope.Last > 1 + then + Restore_Scan_State (Scan_State); -- to END + Output_End_Expected (Ins => True); + Pop_Scope_Stack; + End_Action := Insert_And_Accept; + return; + end if; + + -- Otherwise we go through the normal END evaluation procedure + + Evaluate_End_Entry (Scope.Last); + + -- If top entry in stack is syntactically correct, then we have + -- scanned it out and everything is fine. This is the required + -- action to properly process correct Ada programs. + + if Syntax_OK then + + -- Complain if checking columns and END is not in right column. + -- Right in this context means exactly right, or on the same + -- line as the opener. + + if Style.RM_Column_Check then + if End_Column /= Scope.Table (Scope.Last).Ecol + and then Current_Line_Start > Scope.Table (Scope.Last).Sloc + then + Error_Msg_Col := Scope.Table (Scope.Last).Ecol; + Error_Msg + ("(style) END in wrong column, should be@", End_Sloc); + end if; + end if; + + -- One final check. If the end had a label, check for an exact + -- duplicate of this end sequence, and if so, skip it with an + -- appropriate message. + + if End_Labl_Present and then Token = Tok_End then + declare + Scan_State : Saved_Scan_State; + End_Loc : constant Source_Ptr := Token_Ptr; + Nxt_Labl : Node_Id; + Dup_Found : Boolean := False; + + begin + Save_Scan_State (Scan_State); + + Scan; -- past END + + if Token = Tok_Identifier + or else Token = Tok_Operator_Symbol + then + Nxt_Labl := P_Designator; + + -- We only consider it an error if the label is a match + -- and would be wrong for the level one above us, and + -- the indentation is the same. + + if Token = Tok_Semicolon + and then Same_Label (End_Labl, Nxt_Labl) + and then End_Column = Start_Column + and then + (Scope.Last = 1 + or else + (No (Scope.Table (Scope.Last - 1).Labl) + or else + not Same_Label + (End_Labl, + Scope.Table (Scope.Last - 1).Labl))) + then + T_Semicolon; + Error_Msg ("duplicate end line ignored", End_Loc); + Dup_Found := True; + end if; + end if; + + if not Dup_Found then + Restore_Scan_State (Scan_State); + end if; + end; + end if; + + -- All OK, so return to caller indicating END is OK + + Pop_Scope_Stack; + End_Action := Accept_As_Scanned; + return; + end if; + + -- If that check failed, then we definitely have an error. The issue + -- is how to choose among three possible courses of action: + + -- 1. Ignore the current END text completely, scanning past it, + -- deciding that it belongs neither to the current context, + -- nor to any outer context. + + -- 2. Accept the current END text, scanning past it, and issuing + -- an error message that it does not have the right form. + + -- 3. Leave the current END text in place, NOT scanning past it, + -- issuing an error message indicating the END expected for the + -- current context. In this case, the END is available to match + -- some outer END context. + + -- From a correct functioning point of view, it does not make any + -- difference which of these three approaches we take, the program + -- will work correctly in any case. However, making an accurate + -- choice among these alternatives, i.e. choosing the one that + -- corresponds to what the programmer had in mind, does make a + -- significant difference in the quality of error recovery. + + Restore_Scan_State (Scan_State); -- to END + + -- First we see how good the current END entry is with respect to + -- what we expect. It is considered pretty good if the token is OK, + -- and either the label or the column matches. an END for RECORD is + -- always considered to be pretty good in the record case. This is + -- because not only does a record disallow a nested structure, but + -- also it is unlikely that such nesting could occur by accident. + + Pretty_Good := (Token_OK and (Column_OK or Label_OK)) + or else Scope.Table (Scope.Last).Etyp = E_Record; + + -- Next check, if there is a deeper entry in the stack which + -- has a very high probability of being acceptable, then insert + -- the END entry we want, leaving the higher level entry for later + + for J in reverse 1 .. Scope.Last - 1 loop + Evaluate_End_Entry (J); + + -- To even consider the deeper entry to be immediately acceptable, + -- it must be syntactically correct. Furthermore it must either + -- have a correct label, or the correct column. If the current + -- entry was a close match (Pretty_Good set), then we are even + -- more strict in accepting the outer level one: even if it has + -- the right label, it must have the right column as well. + + if Syntax_OK then + if Pretty_Good then + Outer_Match := Label_OK and Column_OK; + else + Outer_Match := Label_OK or Column_OK; + end if; + else + Outer_Match := False; + end if; + + -- If the outer entry does convincingly match the END text, then + -- back up the scan to the start of the END sequence, issue an + -- error message indicating the END we expected, and return with + -- Token pointing to the END (case 3 from above discussion). + + if Outer_Match then + Output_End_Missing; + Pop_Scope_Stack; + End_Action := Insert_And_Accept; + return; + end if; + end loop; + + -- Here we have a situation in which the current END entry is + -- syntactically incorrect, but there is no deeper entry in the + -- END stack which convincingly matches it. + + -- If the END text was judged to be a Pretty_Good match for the + -- expected token or if it appears left of the expected column, + -- then we will accept it as the one we want, scanning past it, even + -- though it is not completely right (we issue a message showing what + -- we expected it to be). This is action 2 from the discussion above. + -- There is one other special case to consider: the LOOP case. + -- Consider the example: + + -- Lbl: loop + -- null; + -- end loop; + + -- Here the column lines up with Lbl, so END LOOP is to the right, + -- but it is still acceptable. LOOP is the one case where alignment + -- practices vary substantially in practice. + + if Pretty_Good + or else End_Column <= Scope.Table (Scope.Last).Ecol + or else (End_Type = Scope.Table (Scope.Last).Etyp + and then End_Type = E_Loop) + then + Output_End_Expected (Ins => False); + Pop_Scope_Stack; + End_Action := Skip_And_Accept; + return; + + -- Here we have the case where the END is to the right of the + -- expected column and does not have a correct label to convince + -- us that it nevertheless belongs to the current scope. For this + -- we consider that it probably belongs not to the current context, + -- but to some inner context that was not properly recognized (due to + -- other syntax errors), and for which no proper scope stack entry + -- was made. The proper action in this case is to delete the END text + -- and return False to the caller as a signal to keep on looking for + -- an acceptable END. This is action 1 from the discussion above. + + else + Output_End_Deleted; + End_Action := Skip_And_Reject; + return; + end if; + end if; + end Pop_End_Context; + + ---------------- + -- Same_Label -- + ---------------- + + function Same_Label (Label1, Label2 : Node_Id) return Boolean is + begin + if Nkind (Label1) in N_Has_Chars + and then Nkind (Label2) in N_Has_Chars + then + return Chars (Label1) = Chars (Label2); + + elsif Nkind (Label1) = N_Selected_Component + and then Nkind (Label2) = N_Selected_Component + then + return Same_Label (Prefix (Label1), Prefix (Label2)) and then + Same_Label (Selector_Name (Label1), Selector_Name (Label2)); + + elsif Nkind (Label1) = N_Designator + and then Nkind (Label2) = N_Defining_Program_Unit_Name + then + return Same_Label (Name (Label1), Name (Label2)) and then + Same_Label (Identifier (Label1), Defining_Identifier (Label2)); + + else + return False; + end if; + end Same_Label; + +end Endh; diff --git a/gcc/ada/par-labl.adb b/gcc/ada/par-labl.adb new file mode 100644 index 00000000000..e43d3f3f01d --- /dev/null +++ b/gcc/ada/par-labl.adb @@ -0,0 +1,202 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . L A B L -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.18 $ -- +-- -- +-- Copyright (C) 1992-1998, 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). -- +-- -- +------------------------------------------------------------------------------ + +separate (Par) +procedure Labl is + Enclosing_Body_Or_Block : Node_Id; + -- Innermost enclosing body or block statement + + Label_Decl_Node : Node_Id; + -- Implicit label declaration node + + Defining_Ident_Node : Node_Id; + -- Defining identifier node for implicit label declaration + + Next_Label_Elmt : Elmt_Id; + -- Next element on label element list + + Label_Node : Node_Id; + -- Next label node to process + + function Find_Enclosing_Body_Or_Block (N : Node_Id) return Node_Id; + -- Find the innermost body or block that encloses N. + + function Find_Enclosing_Body (N : Node_Id) return Node_Id; + -- Find the innermost body that encloses N. + + procedure Check_Distinct_Labels; + -- Checks the rule in RM-5.1(11), which requires distinct identifiers + -- for all the labels in a given body. + + --------------------------- + -- Check_Distinct_Labels -- + --------------------------- + + procedure Check_Distinct_Labels is + Label_Id : constant Node_Id := Identifier (Label_Node); + + Enclosing_Body : constant Node_Id := + Find_Enclosing_Body (Enclosing_Body_Or_Block); + -- Innermost enclosing body + + Next_Other_Label_Elmt : Elmt_Id := First_Elmt (Label_List); + -- Next element on label element list + + Other_Label : Node_Id; + -- Next label node to process + + begin + -- Loop through all the labels, and if we find some other label + -- (i.e. not Label_Node) that has the same identifier, + -- and whose innermost enclosing body is the same, + -- then we have an error. + + -- Note that in the worst case, this is quadratic in the number + -- of labels. However, labels are not all that common, and this + -- is only called for explicit labels. + -- ???Nonetheless, the efficiency could be improved. For example, + -- call Labl for each body, rather than once per compilation. + + while Present (Next_Other_Label_Elmt) loop + Other_Label := Node (Next_Other_Label_Elmt); + + exit when Label_Node = Other_Label; + + if Chars (Label_Id) = Chars (Identifier (Other_Label)) + and then Enclosing_Body = Find_Enclosing_Body (Other_Label) + then + Error_Msg_Sloc := Sloc (Other_Label); + Error_Msg_N ("& conflicts with label#", Label_Id); + exit; + end if; + + Next_Elmt (Next_Other_Label_Elmt); + end loop; + end Check_Distinct_Labels; + + ------------------------- + -- Find_Enclosing_Body -- + ------------------------- + + function Find_Enclosing_Body (N : Node_Id) return Node_Id is + Result : Node_Id := N; + + begin + -- This is the same as Find_Enclosing_Body_Or_Block, except + -- that we skip block statements and accept statements, instead + -- of stopping at them. + + while Present (Result) + and then Nkind (Result) /= N_Entry_Body + and then Nkind (Result) /= N_Task_Body + and then Nkind (Result) /= N_Package_Body + and then Nkind (Result) /= N_Subprogram_Body + loop + Result := Parent (Result); + end loop; + + return Result; + end Find_Enclosing_Body; + + ---------------------------------- + -- Find_Enclosing_Body_Or_Block -- + ---------------------------------- + + function Find_Enclosing_Body_Or_Block (N : Node_Id) return Node_Id is + Result : Node_Id := Parent (N); + + begin + -- Climb up the parent chain until we find a body or block. + + while Present (Result) + and then Nkind (Result) /= N_Accept_Statement + and then Nkind (Result) /= N_Entry_Body + and then Nkind (Result) /= N_Task_Body + and then Nkind (Result) /= N_Package_Body + and then Nkind (Result) /= N_Subprogram_Body + and then Nkind (Result) /= N_Block_Statement + loop + Result := Parent (Result); + end loop; + + return Result; + end Find_Enclosing_Body_Or_Block; + +-- Start of processing for Par.Labl + +begin + Next_Label_Elmt := First_Elmt (Label_List); + + while Present (Next_Label_Elmt) loop + Label_Node := Node (Next_Label_Elmt); + + if not Comes_From_Source (Label_Node) then + goto Next_Label; + end if; + + -- Find the innermost enclosing body or block, which is where + -- we need to implicitly declare this label + + Enclosing_Body_Or_Block := Find_Enclosing_Body_Or_Block (Label_Node); + + -- If we didn't find a parent, then the label in question never got + -- hooked into a reasonable declarative part. This happens only in + -- error situations, and we simply ignore the entry (we aren't going + -- to get into the semantics in any case given the error). + + if Present (Enclosing_Body_Or_Block) then + Check_Distinct_Labels; + + -- Now create the implicit label declaration node and its + -- corresponding defining identifier. Note that the defining + -- occurrence of a label is the implicit label declaration that + -- we are creating. The label itself is an applied occurrence. + + Label_Decl_Node := + New_Node (N_Implicit_Label_Declaration, Sloc (Label_Node)); + Defining_Ident_Node := + New_Entity (N_Defining_Identifier, Sloc (Identifier (Label_Node))); + Set_Chars (Defining_Ident_Node, Chars (Identifier (Label_Node))); + Set_Defining_Identifier (Label_Decl_Node, Defining_Ident_Node); + Set_Label_Construct (Label_Decl_Node, Label_Node); + + -- Now attach the implicit label declaration to the appropriate + -- declarative region, creating a declaration list if none exists + + if not Present (Declarations (Enclosing_Body_Or_Block)) then + Set_Declarations (Enclosing_Body_Or_Block, New_List); + end if; + + Append (Label_Decl_Node, Declarations (Enclosing_Body_Or_Block)); + end if; + + <<Next_Label>> + Next_Elmt (Next_Label_Elmt); + end loop; + +end Labl; diff --git a/gcc/ada/par-load.adb b/gcc/ada/par-load.adb new file mode 100644 index 00000000000..39934caacb0 --- /dev/null +++ b/gcc/ada/par-load.adb @@ -0,0 +1,410 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . L O A D -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.60 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +-- The Par.Load procedure loads all units that are definitely required before +-- it makes any sense at all to proceed with semantic analysis, including +-- with'ed units, corresponding specs for bodies, parents of child specs, +-- and parents of subunits. All these units are loaded and pointers installed +-- in the tree as described in the spec of package Lib. + +with Fname; use Fname; +with Fname.UF; use Fname.UF; +with Lib.Load; use Lib.Load; +with Uname; use Uname; +with Namet; use Namet; +with Casing; use Casing; +with Opt; use Opt; +with Osint; use Osint; +with Sinput.L; use Sinput.L; +with Stylesw; use Stylesw; +with Validsw; use Validsw; + +separate (Par) +procedure Load is + + File_Name : File_Name_Type; + -- Name of file for current unit, derived from unit name + + Cur_Unum : Unit_Number_Type := Current_Source_Unit; + -- Unit number of unit that we just finished parsing. Note that we need + -- to capture this, because Source_Unit will change as we parse new + -- source files in the multiple main source file case. + + Curunit : constant Node_Id := Cunit (Cur_Unum); + -- Compilation unit node for current compilation unit + + Loc : Source_Ptr := Sloc (Curunit); + -- Source location for compilation unit node + + Save_Style_Check : Boolean; + Save_Style_Checks : Style_Check_Options; + -- Save style check so it can be restored later + + Save_Validity_Check : Boolean; + Save_Validity_Checks : Validity_Check_Options; + -- Save validity check so it can be restored later + + With_Cunit : Node_Id; + -- Compilation unit node for withed unit + + Context_Node : Node_Id; + -- Next node in context items list + + With_Node : Node_Id; + -- N_With_Clause node + + Spec_Name : Unit_Name_Type; + -- Unit name of required spec + + Body_Name : Unit_Name_Type; + -- Unit name of corresponding body + + Unum : Unit_Number_Type; + -- Unit number of loaded unit + + function Same_File_Name_Except_For_Case + (Expected_File_Name : File_Name_Type; + Actual_File_Name : File_Name_Type) + return Boolean; + -- Given an actual file name and an expected file name (the latter being + -- derived from the unit name), determine if they are the same except for + -- possibly different casing of letters. + + function Same_File_Name_Except_For_Case + (Expected_File_Name : File_Name_Type; + Actual_File_Name : File_Name_Type) + return Boolean + is + begin + Get_Name_String (Actual_File_Name); + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + + declare + Lower_Case_Actual_File_Name : String (1 .. Name_Len); + + begin + Lower_Case_Actual_File_Name := Name_Buffer (1 .. Name_Len); + Get_Name_String (Expected_File_Name); + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + return Lower_Case_Actual_File_Name = Name_Buffer (1 .. Name_Len); + end; + + end Same_File_Name_Except_For_Case; + +-- Start of processing for Load + +begin + -- Don't do any loads if we already had a fatal error + + if Fatal_Error (Cur_Unum) then + return; + end if; + + Save_Style_Check_Options (Save_Style_Checks); + Save_Style_Check := Opt.Style_Check; + + Save_Validity_Check_Options (Save_Validity_Checks); + Save_Validity_Check := Opt.Validity_Checks_On; + + -- If main unit, set Main_Unit_Entity (this will get overwritten if + -- the main unit has a separate spec, that happens later on in Load) + + if Cur_Unum = Main_Unit then + Main_Unit_Entity := Cunit_Entity (Main_Unit); + end if; + + -- If we have no unit name, things are seriously messed up by previous + -- errors, and we should not try to continue compilation. + + if Unit_Name (Cur_Unum) = No_Name then + raise Unrecoverable_Error; + end if; + + -- Next step, make sure that the unit name matches the file name + -- and issue a warning message if not. We only output this for the + -- main unit, since for other units it is more serious and is + -- caught in a separate test below. + + File_Name := + Get_File_Name + (Unit_Name (Cur_Unum), + Subunit => Nkind (Unit (Cunit (Cur_Unum))) = N_Subunit); + + if Cur_Unum = Main_Unit + and then File_Name /= Unit_File_Name (Cur_Unum) + and then (File_Names_Case_Sensitive + or not Same_File_Name_Except_For_Case + (File_Name, Unit_File_Name (Cur_Unum))) + then + Error_Msg_Name_1 := File_Name; + Error_Msg + ("?file name does not match unit name, should be{", Sloc (Curunit)); + end if; + + -- For units other than the main unit, the expected unit name is set and + -- must be the same as the actual unit name, or we are in big trouble, and + -- abandon the compilation since there are situations where this really + -- gets us into bad trouble (e.g. some subunit situations). + + if Cur_Unum /= Main_Unit + and then Expected_Unit (Cur_Unum) /= Unit_Name (Cur_Unum) + then + Loc := Error_Location (Cur_Unum); + Error_Msg_Name_1 := Unit_File_Name (Cur_Unum); + Get_Name_String (Error_Msg_Name_1); + + -- Check for predefined file case + + if Name_Len > 1 + and then Name_Buffer (2) = '-' + and then (Name_Buffer (1) = 'a' + or else + Name_Buffer (1) = 's' + or else + Name_Buffer (1) = 'i' + or else + Name_Buffer (1) = 'g') + then + -- In the predefined file case, we know the user did not construct + -- their own package, but we got the wrong one. This means that the + -- name supplied by the user crunched to something we recognized, + -- but then the file did not contain the unit expected. Most likely + -- this is due to a misspelling, e.g. + + -- with Ada.Calender; + + -- This crunches to a-calend, which indeed contains the unit + -- Ada.Calendar, and we can diagnose the misspelling. This is + -- a simple heuristic, but it catches many common cases of + -- misspelling of predefined unit names without needing a full + -- list of them. + + Error_Msg_Name_1 := Expected_Unit (Cur_Unum); + Error_Msg ("% is not a predefined library unit!", Loc); + Error_Msg_Name_1 := Unit_Name (Cur_Unum); + Error_Msg ("possible misspelling of %!", Loc); + + -- Non-predefined file name case + + else + Error_Msg ("file { does not contain expected unit!", Loc); + Error_Msg_Unit_1 := Expected_Unit (Cur_Unum); + Error_Msg ("expected unit $!", Loc); + Error_Msg_Unit_1 := Unit_Name (Cur_Unum); + Error_Msg ("found unit $!", Loc); + end if; + + raise Unrecoverable_Error; + end if; + + -- If current unit is a body, load its corresponding spec + + if Nkind (Unit (Curunit)) = N_Package_Body + or else Nkind (Unit (Curunit)) = N_Subprogram_Body + then + Spec_Name := Get_Spec_Name (Unit_Name (Cur_Unum)); + Unum := + Load_Unit + (Load_Name => Spec_Name, + Required => False, + Subunit => False, + Error_Node => Curunit, + Corr_Body => Cur_Unum); + + -- If we successfully load the unit, then set the spec pointer. Once + -- again note that if the loaded unit has a fatal error, Load will + -- have set our Fatal_Error flag to propagate this condition. + + if Unum /= No_Unit then + Set_Library_Unit (Curunit, Cunit (Unum)); + + -- If this is a separate spec for the main unit, then we reset + -- Main_Unit_Entity to point to the entity for this separate spec + + if Cur_Unum = Main_Unit then + Main_Unit_Entity := Cunit_Entity (Unum); + end if; + + -- If we don't find the spec, then if we have a subprogram body, we + -- are still OK, we just have a case of a body acting as its own spec + + elsif Nkind (Unit (Curunit)) = N_Subprogram_Body then + Set_Acts_As_Spec (Curunit, True); + Set_Library_Unit (Curunit, Curunit); + + -- Otherwise we do have an error, repeat the load request for the spec + -- with Required set True to generate an appropriate error message. + + else + Unum := + Load_Unit + (Load_Name => Spec_Name, + Required => True, + Subunit => False, + Error_Node => Curunit); + return; + end if; + + -- If current unit is a child unit spec, load its parent + + elsif Nkind (Unit (Curunit)) = N_Package_Declaration + or else Nkind (Unit (Curunit)) = N_Subprogram_Declaration + or else Nkind (Unit (Curunit)) in N_Generic_Declaration + or else Nkind (Unit (Curunit)) in N_Generic_Instantiation + or else Nkind (Unit (Curunit)) in N_Renaming_Declaration + then + -- Turn style and validity checks off for parent unit + + if not GNAT_Mode then + Reset_Style_Check_Options; + Reset_Validity_Check_Options; + end if; + + Spec_Name := Get_Parent_Spec_Name (Unit_Name (Cur_Unum)); + + if Spec_Name /= No_Name then + Unum := + Load_Unit + (Load_Name => Spec_Name, + Required => True, + Subunit => False, + Error_Node => Curunit); + + if Unum /= No_Unit then + Set_Parent_Spec (Unit (Curunit), Cunit (Unum)); + end if; + end if; + + -- If current unit is a subunit, then load its parent body + + elsif Nkind (Unit (Curunit)) = N_Subunit then + Body_Name := Get_Parent_Body_Name (Unit_Name (Cur_Unum)); + Unum := + Load_Unit + (Load_Name => Body_Name, + Required => True, + Subunit => True, + Error_Node => Name (Unit (Curunit))); + + if Unum /= No_Unit then + Set_Library_Unit (Curunit, Cunit (Unum)); + end if; + + end if; + + -- Now we load with'ed units, with style/validity checks turned off + + if not GNAT_Mode then + Reset_Style_Check_Options; + Reset_Validity_Check_Options; + end if; + + -- Loop through context items + + Context_Node := First (Context_Items (Curunit)); + while Present (Context_Node) loop + + if Nkind (Context_Node) = N_With_Clause then + With_Node := Context_Node; + Spec_Name := Get_Unit_Name (With_Node); + + Unum := + Load_Unit + (Load_Name => Spec_Name, + Required => False, + Subunit => False, + Error_Node => With_Node, + Renamings => True); + + -- If we find the unit, then set spec pointer in the N_With_Clause + -- to point to the compilation unit for the spec. Remember that + -- the Load routine itself sets our Fatal_Error flag if the loaded + -- unit gets a fatal error, so we don't need to worry about that. + + if Unum /= No_Unit then + Set_Library_Unit (With_Node, Cunit (Unum)); + + -- If the spec isn't found, then try finding the corresponding + -- body, since it is possible that we have a subprogram body + -- that is acting as a spec (since no spec is present). + + else + Body_Name := Get_Body_Name (Spec_Name); + Unum := + Load_Unit + (Load_Name => Body_Name, + Required => False, + Subunit => False, + Error_Node => With_Node, + Renamings => True); + + -- If we got a subprogram body, then mark that we are using + -- the body as a spec in the file table, and set the spec + -- pointer in the N_With_Clause to point to the body entity. + + if Unum /= No_Unit + and then Nkind (Unit (Cunit (Unum))) = N_Subprogram_Body + then + With_Cunit := Cunit (Unum); + Set_Library_Unit (With_Node, With_Cunit); + Set_Acts_As_Spec (With_Cunit, True); + Set_Library_Unit (With_Cunit, With_Cunit); + + -- If we couldn't find the body, or if it wasn't a body spec + -- then we are in trouble. We make one more call to Load to + -- require the spec. We know it will fail of course, the + -- purpose is to generate the required error message (we prefer + -- that this message refer to the missing spec, not the body) + + else + Unum := + Load_Unit + (Load_Name => Spec_Name, + Required => True, + Subunit => False, + Error_Node => With_Node, + Renamings => True); + + -- Here we create a dummy package unit for the missing unit + + Unum := Create_Dummy_Package_Unit (With_Node, Spec_Name); + Set_Library_Unit (With_Node, Cunit (Unum)); + end if; + end if; + end if; + + Next (Context_Node); + end loop; + + -- Restore style/validity check mode for main unit + + Set_Style_Check_Options (Save_Style_Checks); + Opt.Style_Check := Save_Style_Check; + Set_Validity_Check_Options (Save_Validity_Checks); + Opt.Validity_Checks_On := Save_Validity_Check; +end Load; diff --git a/gcc/ada/par-prag.adb b/gcc/ada/par-prag.adb new file mode 100644 index 00000000000..bfca40e8c18 --- /dev/null +++ b/gcc/ada/par-prag.adb @@ -0,0 +1,950 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . P R A G -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.149 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +-- Generally the parser checks the basic syntax of pragmas, but does not +-- do specialized syntax checks for individual pragmas, these are deferred +-- to semantic analysis time (see unit Sem_Prag). There are some pragmas +-- which require recognition and either partial or complete processing +-- during parsing, and this unit performs this required processing. + +with Fname.UF; use Fname.UF; +with Osint; use Osint; +with Stringt; use Stringt; +with Stylesw; use Stylesw; +with Uintp; use Uintp; +with Uname; use Uname; + +separate (Par) + +function Prag (Pragma_Node : Node_Id; Semi : Source_Ptr) return Node_Id is + Pragma_Name : constant Name_Id := Chars (Pragma_Node); + Pragma_Sloc : constant Source_Ptr := Sloc (Pragma_Node); + Arg_Count : Nat; + Arg_Node : Node_Id; + + ----------------------- + -- Local Subprograms -- + ----------------------- + + function Arg1 return Node_Id; + function Arg2 return Node_Id; + function Arg3 return Node_Id; + function Arg4 return Node_Id; + -- Obtain specified Pragma_Argument_Association. It is allowable to call + -- the routine for the argument one past the last present argument, but + -- that is the only case in which a non-present argument can be referenced. + + procedure Check_Arg_Count (Required : Int); + -- Check argument count for pragma = Required. + -- If not give error and raise Error_Resync. + + procedure Check_Arg_Is_String_Literal (Arg : Node_Id); + -- Check the expression of the specified argument to make sure that it + -- is a string literal. If not give error and raise Error_Resync. + + procedure Check_Arg_Is_On_Or_Off (Arg : Node_Id); + -- Check the expression of the specified argument to make sure that it + -- is an identifier which is either ON or OFF, and if not, then issue + -- an error message and raise Error_Resync. + + procedure Check_No_Identifier (Arg : Node_Id); + -- Checks that the given argument does not have an identifier. If an + -- identifier is present, then an error message is issued, and + -- Error_Resync is raised. + + procedure Check_Optional_Identifier (Arg : Node_Id; Id : Name_Id); + -- Checks if the given argument has an identifier, and if so, requires + -- it to match the given identifier name. If there is a non-matching + -- identifier, then an error message is given and Error_Resync raised. + + procedure Check_Required_Identifier (Arg : Node_Id; Id : Name_Id); + -- Same as Check_Optional_Identifier, except that the name is required + -- to be present and to match the given Id value. + + ---------- + -- Arg1 -- + ---------- + + function Arg1 return Node_Id is + begin + return First (Pragma_Argument_Associations (Pragma_Node)); + end Arg1; + + ---------- + -- Arg2 -- + ---------- + + function Arg2 return Node_Id is + begin + return Next (Arg1); + end Arg2; + + ---------- + -- Arg3 -- + ---------- + + function Arg3 return Node_Id is + begin + return Next (Arg2); + end Arg3; + + ---------- + -- Arg4 -- + ---------- + + function Arg4 return Node_Id is + begin + return Next (Arg3); + end Arg4; + + --------------------- + -- Check_Arg_Count -- + --------------------- + + procedure Check_Arg_Count (Required : Int) is + begin + if Arg_Count /= Required then + Error_Msg ("wrong number of arguments for pragma%", Pragma_Sloc); + raise Error_Resync; + end if; + end Check_Arg_Count; + + ---------------------------- + -- Check_Arg_Is_On_Or_Off -- + ---------------------------- + + procedure Check_Arg_Is_On_Or_Off (Arg : Node_Id) is + Argx : constant Node_Id := Expression (Arg); + + begin + if Nkind (Expression (Arg)) /= N_Identifier + or else (Chars (Argx) /= Name_On + and then + Chars (Argx) /= Name_Off) + then + Error_Msg_Name_2 := Name_On; + Error_Msg_Name_3 := Name_Off; + + Error_Msg + ("argument for pragma% must be% or%", Sloc (Argx)); + raise Error_Resync; + end if; + end Check_Arg_Is_On_Or_Off; + + --------------------------------- + -- Check_Arg_Is_String_Literal -- + --------------------------------- + + procedure Check_Arg_Is_String_Literal (Arg : Node_Id) is + begin + if Nkind (Expression (Arg)) /= N_String_Literal then + Error_Msg + ("argument for pragma% must be string literal", + Sloc (Expression (Arg))); + raise Error_Resync; + end if; + end Check_Arg_Is_String_Literal; + + ------------------------- + -- Check_No_Identifier -- + ------------------------- + + procedure Check_No_Identifier (Arg : Node_Id) is + begin + if Chars (Arg) /= No_Name then + Error_Msg_N ("pragma% does not permit named arguments", Arg); + raise Error_Resync; + end if; + end Check_No_Identifier; + + ------------------------------- + -- Check_Optional_Identifier -- + ------------------------------- + + procedure Check_Optional_Identifier (Arg : Node_Id; Id : Name_Id) is + begin + if Present (Arg) and then Chars (Arg) /= No_Name then + if Chars (Arg) /= Id then + Error_Msg_Name_2 := Id; + Error_Msg_N ("pragma% argument expects identifier%", Arg); + end if; + end if; + end Check_Optional_Identifier; + + ------------------------------- + -- Check_Required_Identifier -- + ------------------------------- + + procedure Check_Required_Identifier (Arg : Node_Id; Id : Name_Id) is + begin + if Chars (Arg) /= Id then + Error_Msg_Name_2 := Id; + Error_Msg_N ("pragma% argument must have identifier%", Arg); + end if; + end Check_Required_Identifier; + + ---------- + -- Prag -- + ---------- + +begin + Error_Msg_Name_1 := Pragma_Name; + + -- Count number of arguments. This loop also checks if any of the arguments + -- are Error, indicating a syntax error as they were parsed. If so, we + -- simply return, because we get into trouble with cascaded errors if we + -- try to perform our error checks on junk arguments. + + Arg_Count := 0; + + if Present (Pragma_Argument_Associations (Pragma_Node)) then + Arg_Node := Arg1; + + while Arg_Node /= Empty loop + Arg_Count := Arg_Count + 1; + + if Expression (Arg_Node) = Error then + return Error; + end if; + + Next (Arg_Node); + end loop; + end if; + + -- Remaining processing is pragma dependent + + case Get_Pragma_Id (Pragma_Name) is + + ------------ + -- Ada_83 -- + ------------ + + -- This pragma must be processed at parse time, since we want to set + -- the Ada 83 and Ada 95 switches properly at parse time to recognize + -- Ada 83 syntax or Ada 95 syntax as appropriate. + + when Pragma_Ada_83 => + Ada_83 := True; + Ada_95 := False; + + ------------ + -- Ada_95 -- + ------------ + + -- This pragma must be processed at parse time, since we want to set + -- the Ada 83 and Ada_95 switches properly at parse time to recognize + -- Ada 83 syntax or Ada 95 syntax as appropriate. + + when Pragma_Ada_95 => + Ada_83 := False; + Ada_95 := True; + + ----------- + -- Debug -- + ----------- + + -- pragma Debug (PROCEDURE_CALL_STATEMENT); + + -- This has to be processed by the parser because of the very peculiar + -- form of the second parameter, which is syntactically from a formal + -- point of view a function call (since it must be an expression), but + -- semantically we treat it as a procedure call (which has exactly the + -- same syntactic form, so that's why we can get away with this!) + + when Pragma_Debug => + Check_Arg_Count (1); + Check_No_Identifier (Arg1); + + declare + Expr : constant Node_Id := New_Copy (Expression (Arg1)); + + begin + if Nkind (Expr) /= N_Indexed_Component + and then Nkind (Expr) /= N_Function_Call + and then Nkind (Expr) /= N_Identifier + and then Nkind (Expr) /= N_Selected_Component + then + Error_Msg + ("argument of pragma% is not procedure call", Sloc (Expr)); + raise Error_Resync; + else + Set_Debug_Statement + (Pragma_Node, P_Statement_Name (Expr)); + end if; + end; + + ------------------------------- + -- Extensions_Allowed (GNAT) -- + ------------------------------- + + -- pragma Extensions_Allowed (Off | On) + + -- The processing for pragma Extensions_Allowed must be done at + -- parse time, since extensions mode may affect what is accepted. + + when Pragma_Extensions_Allowed => + Check_Arg_Count (1); + Check_No_Identifier (Arg1); + Check_Arg_Is_On_Or_Off (Arg1); + Opt.Extensions_Allowed := (Chars (Expression (Arg1)) = Name_On); + + ---------------- + -- List (2.8) -- + ---------------- + + -- pragma List (Off | On) + + -- The processing for pragma List must be done at parse time, + -- since a listing can be generated in parse only mode. + + when Pragma_List => + Check_Arg_Count (1); + Check_No_Identifier (Arg1); + Check_Arg_Is_On_Or_Off (Arg1); + + -- We unconditionally make a List_On entry for the pragma, so that + -- in the List (Off) case, the pragma will print even in a region + -- of code with listing turned off (this is required!) + + List_Pragmas.Increment_Last; + List_Pragmas.Table (List_Pragmas.Last) := + (Ptyp => List_On, Ploc => Sloc (Pragma_Node)); + + -- Now generate the list off entry for pragma List (Off) + + if Chars (Expression (Arg1)) = Name_Off then + List_Pragmas.Increment_Last; + List_Pragmas.Table (List_Pragmas.Last) := + (Ptyp => List_Off, Ploc => Semi); + end if; + + ---------------- + -- Page (2.8) -- + ---------------- + + -- pragma Page; + + -- Processing for this pragma must be done at parse time, since a + -- listing can be generated in parse only mode with semantics off. + + when Pragma_Page => + Check_Arg_Count (0); + List_Pragmas.Increment_Last; + List_Pragmas.Table (List_Pragmas.Last) := (Page, Semi); + + ----------------------------- + -- Source_File_Name (GNAT) -- + ----------------------------- + + -- There are five forms of this pragma: + + -- pragma Source_File_Name ( + -- [UNIT_NAME =>] unit_NAME, + -- BODY_FILE_NAME => STRING_LITERAL); + + -- pragma Source_File_Name ( + -- [UNIT_NAME =>] unit_NAME, + -- SPEC_FILE_NAME => STRING_LITERAL); + + -- pragma Source_File_Name ( + -- BODY_FILE_NAME => STRING_LITERAL + -- [, DOT_REPLACEMENT => STRING_LITERAL] + -- [, CASING => CASING_SPEC]); + + -- pragma Source_File_Name ( + -- SPEC_FILE_NAME => STRING_LITERAL + -- [, DOT_REPLACEMENT => STRING_LITERAL] + -- [, CASING => CASING_SPEC]); + + -- pragma Source_File_Name ( + -- SUBUNIT_FILE_NAME => STRING_LITERAL + -- [, DOT_REPLACEMENT => STRING_LITERAL] + -- [, CASING => CASING_SPEC]); + + -- CASING_SPEC ::= Uppercase | Lowercase | Mixedcase + + -- Note: we process this during parsing, since we need to have the + -- source file names set well before the semantic analysis starts, + -- since we load the spec and with'ed packages before analysis. + + when Pragma_Source_File_Name => Source_File_Name : declare + Unam : Unit_Name_Type; + Expr1 : Node_Id; + Pat : String_Ptr; + Typ : Character; + Dot : String_Ptr; + Cas : Casing_Type; + Nast : Nat; + + function Get_Fname (Arg : Node_Id) return Name_Id; + -- Process file name from unit name form of pragma + + function Get_String_Argument (Arg : Node_Id) return String_Ptr; + -- Process string literal value from argument + + procedure Process_Casing (Arg : Node_Id); + -- Process Casing argument of pattern form of pragma + + procedure Process_Dot_Replacement (Arg : Node_Id); + -- Process Dot_Replacement argument of patterm form of pragma + + --------------- + -- Get_Fname -- + --------------- + + function Get_Fname (Arg : Node_Id) return Name_Id is + begin + String_To_Name_Buffer (Strval (Expression (Arg))); + + for J in 1 .. Name_Len loop + if Is_Directory_Separator (Name_Buffer (J)) then + Error_Msg + ("directory separator character not allowed", + Sloc (Expression (Arg)) + Source_Ptr (J)); + end if; + end loop; + + return Name_Find; + end Get_Fname; + + ------------------------- + -- Get_String_Argument -- + ------------------------- + + function Get_String_Argument (Arg : Node_Id) return String_Ptr is + Str : String_Id; + + begin + if Nkind (Expression (Arg)) /= N_String_Literal + and then + Nkind (Expression (Arg)) /= N_Operator_Symbol + then + Error_Msg_N + ("argument for pragma% must be string literal", Arg); + raise Error_Resync; + end if; + + Str := Strval (Expression (Arg)); + + -- Check string has no wide chars + + for J in 1 .. String_Length (Str) loop + if Get_String_Char (Str, J) > 255 then + Error_Msg + ("wide character not allowed in pattern for pragma%", + Sloc (Expression (Arg2)) + Text_Ptr (J) - 1); + end if; + end loop; + + -- Acquire string + + String_To_Name_Buffer (Str); + return new String'(Name_Buffer (1 .. Name_Len)); + end Get_String_Argument; + + -------------------- + -- Process_Casing -- + -------------------- + + procedure Process_Casing (Arg : Node_Id) is + Expr : constant Node_Id := Expression (Arg); + + begin + Check_Required_Identifier (Arg, Name_Casing); + + if Nkind (Expr) = N_Identifier then + if Chars (Expr) = Name_Lowercase then + Cas := All_Lower_Case; + return; + elsif Chars (Expr) = Name_Uppercase then + Cas := All_Upper_Case; + return; + elsif Chars (Expr) = Name_Mixedcase then + Cas := Mixed_Case; + return; + end if; + end if; + + Error_Msg_N + ("Casing argument for pragma% must be " & + "one of Mixedcase, Lowercase, Uppercase", + Arg); + end Process_Casing; + + ----------------------------- + -- Process_Dot_Replacement -- + ----------------------------- + + procedure Process_Dot_Replacement (Arg : Node_Id) is + begin + Check_Required_Identifier (Arg, Name_Dot_Replacement); + Dot := Get_String_Argument (Arg); + end Process_Dot_Replacement; + + -- Start of processing for Source_File_Name pragma + + begin + -- We permit from 1 to 3 arguments + + if Arg_Count not in 1 .. 3 then + Check_Arg_Count (1); + end if; + + Expr1 := Expression (Arg1); + + -- If first argument is identifier or selected component, then + -- we have the specific file case of the Source_File_Name pragma, + -- and the first argument is a unit name. + + if Nkind (Expr1) = N_Identifier + or else + (Nkind (Expr1) = N_Selected_Component + and then + Nkind (Selector_Name (Expr1)) = N_Identifier) + then + Check_Arg_Count (2); + + Check_Optional_Identifier (Arg1, Name_Unit_Name); + Unam := Get_Unit_Name (Expr1); + + Check_Arg_Is_String_Literal (Arg2); + + if Chars (Arg2) = Name_Spec_File_Name then + Set_File_Name (Get_Spec_Name (Unam), Get_Fname (Arg2)); + + elsif Chars (Arg2) = Name_Body_File_Name then + Set_File_Name (Unam, Get_Fname (Arg2)); + + else + Error_Msg_N ("pragma% argument has incorrect identifier", Arg2); + return Pragma_Node; + end if; + + -- If the first argument is not an identifier, then we must have + -- the pattern form of the pragma, and the first argument must be + -- the pattern string with an appropriate name. + + else + if Chars (Arg1) = Name_Spec_File_Name then + Typ := 's'; + + elsif Chars (Arg1) = Name_Body_File_Name then + Typ := 'b'; + + elsif Chars (Arg1) = Name_Subunit_File_Name then + Typ := 'u'; + + elsif Chars (Arg1) = Name_Unit_Name then + Error_Msg_N + ("Unit_Name parameter for pragma% must be an identifier", + Arg1); + raise Error_Resync; + + else + Error_Msg_N ("pragma% argument has incorrect identifier", Arg1); + raise Error_Resync; + end if; + + Pat := Get_String_Argument (Arg1); + + -- Check pattern has exactly one asterisk + + Nast := 0; + for J in Pat'Range loop + if Pat (J) = '*' then + Nast := Nast + 1; + end if; + end loop; + + if Nast /= 1 then + Error_Msg_N + ("file name pattern must have exactly one * character", + Arg2); + return Pragma_Node; + end if; + + -- Set defaults for Casing and Dot_Separator parameters + + Cas := All_Lower_Case; + + Dot := new String'("."); + + -- Process second and third arguments if present + + if Arg_Count > 1 then + if Chars (Arg2) = Name_Casing then + Process_Casing (Arg2); + + if Arg_Count = 3 then + Process_Dot_Replacement (Arg3); + end if; + + else + Process_Dot_Replacement (Arg2); + + if Arg_Count = 3 then + Process_Casing (Arg3); + end if; + end if; + end if; + + Set_File_Name_Pattern (Pat, Typ, Dot, Cas); + end if; + end Source_File_Name; + + ----------------------------- + -- Source_Reference (GNAT) -- + ----------------------------- + + -- pragma Source_Reference + -- (INTEGER_LITERAL [, STRING_LITERAL] ); + + -- Processing for this pragma must be done at parse time, since error + -- messages needing the proper line numbers can be generated in parse + -- only mode with semantic checking turned off, and indeed we usually + -- turn off semantic checking anyway if any parse errors are found. + + when Pragma_Source_Reference => Source_Reference : declare + Fname : Name_Id; + + begin + if Arg_Count /= 1 then + Check_Arg_Count (2); + Check_No_Identifier (Arg2); + end if; + + -- Check that this is first line of file. We skip this test if + -- we are in syntax check only mode, since we may be dealing with + -- multiple compilation units. + + if Get_Physical_Line_Number (Pragma_Sloc) /= 1 + and then Num_SRef_Pragmas (Current_Source_File) = 0 + and then Operating_Mode /= Check_Syntax + then + Error_Msg + ("first % pragma must be first line of file", Pragma_Sloc); + raise Error_Resync; + end if; + + Check_No_Identifier (Arg1); + + if Arg_Count = 1 then + if Num_SRef_Pragmas (Current_Source_File) = 0 then + Error_Msg + ("file name required for first % pragma in file", + Pragma_Sloc); + raise Error_Resync; + + else + Fname := No_Name; + end if; + + -- File name present + + else + Check_Arg_Is_String_Literal (Arg2); + String_To_Name_Buffer (Strval (Expression (Arg2))); + Fname := Name_Find; + + if Num_SRef_Pragmas (Current_Source_File) > 0 then + if Fname /= Full_Ref_Name (Current_Source_File) then + Error_Msg + ("file name must be same in all % pragmas", Pragma_Sloc); + raise Error_Resync; + end if; + end if; + end if; + + if Nkind (Expression (Arg1)) /= N_Integer_Literal then + Error_Msg + ("argument for pragma% must be integer literal", + Sloc (Expression (Arg1))); + raise Error_Resync; + + -- OK, this source reference pragma is effective, however, we + -- ignore it if it is not in the first unit in the multiple unit + -- case. This is because the only purpose in this case is to + -- provide source pragmas for subsequent use by gnatchop. + + else + if Num_Library_Units = 1 then + Register_Source_Ref_Pragma + (Fname, + Strip_Directory (Fname), + UI_To_Int (Intval (Expression (Arg1))), + Get_Physical_Line_Number (Pragma_Sloc) + 1); + end if; + end if; + end Source_Reference; + + ------------------------- + -- Style_Checks (GNAT) -- + ------------------------- + + -- pragma Style_Checks (On | Off | ALL_CHECKS | STRING_LITERAL); + + -- This is processed by the parser since some of the style + -- checks take place during source scanning and parsing. + + when Pragma_Style_Checks => Style_Checks : declare + A : Node_Id; + S : String_Id; + C : Char_Code; + OK : Boolean := True; + + begin + -- Two argument case is only for semantics + + if Arg_Count = 2 then + null; + + else + Check_Arg_Count (1); + Check_No_Identifier (Arg1); + A := Expression (Arg1); + + if Nkind (A) = N_String_Literal then + S := Strval (A); + + declare + Slen : Natural := Natural (String_Length (S)); + Options : String (1 .. Slen); + J : Natural; + Ptr : Natural; + + begin + J := 1; + loop + C := Get_String_Char (S, Int (J)); + + if not In_Character_Range (C) then + OK := False; + Ptr := J; + exit; + + else + Options (J) := Get_Character (C); + end if; + + if J = Slen then + Set_Style_Check_Options (Options, OK, Ptr); + exit; + + else + J := J + 1; + end if; + end loop; + + if not OK then + Error_Msg + ("invalid style check option", + Sloc (Expression (Arg1)) + Source_Ptr (Ptr)); + raise Error_Resync; + end if; + end; + + elsif Nkind (A) /= N_Identifier then + OK := False; + + elsif Chars (A) = Name_All_Checks then + Stylesw.Set_Default_Style_Check_Options; + + elsif Chars (A) = Name_On then + Style_Check := True; + + elsif Chars (A) = Name_Off then + Style_Check := False; + + else + OK := False; + end if; + + if not OK then + Error_Msg ("incorrect argument for pragma%", Sloc (A)); + raise Error_Resync; + end if; + end if; + end Style_Checks; + + --------------------- + -- Warnings (GNAT) -- + --------------------- + + -- pragma Warnings (On | Off, [LOCAL_NAME]) + + -- The one argument case is processed by the parser, since it may + -- control parser warnings as well as semantic warnings, and in any + -- case we want to be absolutely sure that the range in the warnings + -- table is set well before any semantic analysis is performed. + + when Pragma_Warnings => + if Arg_Count = 1 then + Check_No_Identifier (Arg1); + Check_Arg_Is_On_Or_Off (Arg1); + + if Chars (Expression (Arg1)) = Name_On then + Set_Warnings_Mode_On (Pragma_Sloc); + else + Set_Warnings_Mode_Off (Pragma_Sloc); + end if; + end if; + + ----------------------- + -- All Other Pragmas -- + ----------------------- + + -- For all other pragmas, checking and processing is handled + -- entirely in Sem_Prag, and no further checking is done by Par. + + when Pragma_Abort_Defer | + Pragma_AST_Entry | + Pragma_All_Calls_Remote | + Pragma_Annotate | + Pragma_Assert | + Pragma_Asynchronous | + Pragma_Atomic | + Pragma_Atomic_Components | + Pragma_Attach_Handler | + Pragma_CPP_Class | + Pragma_CPP_Constructor | + Pragma_CPP_Virtual | + Pragma_CPP_Vtable | + Pragma_C_Pass_By_Copy | + Pragma_Comment | + Pragma_Common_Object | + Pragma_Complex_Representation | + Pragma_Component_Alignment | + Pragma_Controlled | + Pragma_Convention | + Pragma_Discard_Names | + Pragma_Eliminate | + Pragma_Elaborate | + Pragma_Elaborate_All | + Pragma_Elaborate_Body | + Pragma_Elaboration_Checks | + Pragma_Export | + Pragma_Export_Exception | + Pragma_Export_Function | + Pragma_Export_Object | + Pragma_Export_Procedure | + Pragma_Export_Valued_Procedure | + Pragma_Extend_System | + Pragma_External_Name_Casing | + Pragma_Finalize_Storage_Only | + Pragma_Float_Representation | + Pragma_Ident | + Pragma_Import | + Pragma_Import_Exception | + Pragma_Import_Function | + Pragma_Import_Object | + Pragma_Import_Procedure | + Pragma_Import_Valued_Procedure | + Pragma_Initialize_Scalars | + Pragma_Inline | + Pragma_Inline_Always | + Pragma_Inline_Generic | + Pragma_Inspection_Point | + Pragma_Interface | + Pragma_Interface_Name | + Pragma_Interrupt_Handler | + Pragma_Interrupt_Priority | + Pragma_Java_Constructor | + Pragma_Java_Interface | + Pragma_License | + Pragma_Link_With | + Pragma_Linker_Alias | + Pragma_Linker_Options | + Pragma_Linker_Section | + Pragma_Locking_Policy | + Pragma_Long_Float | + Pragma_Machine_Attribute | + Pragma_Main | + Pragma_Main_Storage | + Pragma_Memory_Size | + Pragma_No_Return | + Pragma_No_Run_Time | + Pragma_Normalize_Scalars | + Pragma_Optimize | + Pragma_Pack | + Pragma_Passive | + Pragma_Polling | + Pragma_Preelaborate | + Pragma_Priority | + Pragma_Propagate_Exceptions | + Pragma_Psect_Object | + Pragma_Pure | + Pragma_Pure_Function | + Pragma_Queuing_Policy | + Pragma_Remote_Call_Interface | + Pragma_Remote_Types | + Pragma_Restrictions | + Pragma_Restricted_Run_Time | + Pragma_Ravenscar | + Pragma_Reviewable | + Pragma_Share_Generic | + Pragma_Shared | + Pragma_Shared_Passive | + Pragma_Storage_Size | + Pragma_Storage_Unit | + Pragma_Stream_Convert | + Pragma_Subtitle | + Pragma_Suppress | + Pragma_Suppress_All | + Pragma_Suppress_Debug_Info | + Pragma_Suppress_Initialization | + Pragma_System_Name | + Pragma_Task_Dispatching_Policy | + Pragma_Task_Info | + Pragma_Task_Name | + Pragma_Task_Storage | + Pragma_Time_Slice | + Pragma_Title | + Pragma_Unchecked_Union | + Pragma_Unimplemented_Unit | + Pragma_Unreserve_All_Interrupts | + Pragma_Unsuppress | + Pragma_Use_VADS_Size | + Pragma_Volatile | + Pragma_Volatile_Components | + Pragma_Weak_External | + Pragma_Validity_Checks => + null; + + end case; + + return Pragma_Node; + + -------------------- + -- Error Handling -- + -------------------- + +exception + when Error_Resync => + return Error; + +end Prag; diff --git a/gcc/ada/par-sync.adb b/gcc/ada/par-sync.adb new file mode 100644 index 00000000000..d1ba793d9cd --- /dev/null +++ b/gcc/ada/par-sync.adb @@ -0,0 +1,312 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . S Y N C -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.19 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +separate (Par) +package body Sync is + + procedure Resync_Init; + -- This routine is called on initiating a resynchronization action + + procedure Resync_Resume; + -- This routine is called on completing a resynchronization action + + ------------------- + -- Resync_Choice -- + ------------------- + + procedure Resync_Choice is + begin + Resync_Init; + + -- Loop till we get a token that terminates a choice. Note that EOF is + -- one such token, so we are sure to get out of this loop eventually! + + while Token not in Token_Class_Cterm loop + Scan; + end loop; + + Resync_Resume; + end Resync_Choice; + + ------------------ + -- Resync_Cunit -- + ------------------ + + procedure Resync_Cunit is + begin + Resync_Init; + + while Token not in Token_Class_Cunit + and then Token /= Tok_EOF + loop + Scan; + end loop; + + Resync_Resume; + end Resync_Cunit; + + ----------------------- + -- Resync_Expression -- + ----------------------- + + procedure Resync_Expression is + Paren_Count : Int; + + begin + Resync_Init; + Paren_Count := 0; + + loop + -- Terminating tokens are those in class Eterm and also RANGE, + -- DIGITS or DELTA if not preceded by an apostrophe (if they are + -- preceded by an apostrophe, then they are attributes). In addiion, + -- at the outer parentheses level only, we also consider a comma, + -- right parenthesis or vertical bar to terminate an expression. + + if Token in Token_Class_Eterm + + or else (Token in Token_Class_Atkwd + and then Prev_Token /= Tok_Apostrophe) + + or else (Paren_Count = 0 + and then + (Token = Tok_Comma + or else Token = Tok_Right_Paren + or else Token = Tok_Vertical_Bar)) + then + -- A special check: if we stop on the ELSE of OR ELSE or the + -- THEN of AND THEN, keep going, because this is not really an + -- expression terminator after all. Also, keep going past WITH + -- since this can be part of an extension aggregate + + if (Token = Tok_Else and then Prev_Token = Tok_Or) + or else (Token = Tok_Then and then Prev_Token = Tok_And) + or else Token = Tok_With + then + null; + else + exit; + end if; + end if; + + if Token = Tok_Left_Paren then + Paren_Count := Paren_Count + 1; + + elsif Token = Tok_Right_Paren then + Paren_Count := Paren_Count - 1; + + end if; + + Scan; -- past token to be skipped + end loop; + + Resync_Resume; + end Resync_Expression; + + ----------------- + -- Resync_Init -- + ----------------- + + procedure Resync_Init is + begin + -- The following check makes sure we do not get stuck in an infinite + -- loop resynchonizing and getting nowhere. If we are called to do a + -- resynchronize and we are exactly at the same point that we left off + -- on the last resynchronize call, then we force at least one token to + -- be skipped so that we make progress! + + if Token_Ptr = Last_Resync_Point then + Scan; -- to skip at least one token + end if; + + -- Output extra error message if debug R flag is set + + if Debug_Flag_R then + Error_Msg_SC ("resynchronizing!"); + end if; + end Resync_Init; + + --------------------------- + -- Resync_Past_Semicolon -- + --------------------------- + + procedure Resync_Past_Semicolon is + begin + Resync_Init; + + loop + -- Done if we are at a semicolon + + if Token = Tok_Semicolon then + Scan; -- past semicolon + exit; + + -- Done if we are at a token which normally appears only after + -- a semicolon. One special glitch is that the keyword private is + -- in this category only if it does NOT appear after WITH. + + elsif Token in Token_Class_After_SM + and then (Token /= Tok_Private or else Prev_Token /= Tok_With) + then + exit; + + -- Otherwise keep going + + else + Scan; + end if; + end loop; + + -- Fall out of loop with resyncrhonization complete + + Resync_Resume; + end Resync_Past_Semicolon; + + ---------------------------------------------- + -- Resync_Past_Semicolon_Or_To_Loop_Or_Then -- + ---------------------------------------------- + + procedure Resync_Past_Semicolon_Or_To_Loop_Or_Then is + begin + Resync_Init; + + loop + -- Done if at semicolon + + if Token = Tok_Semicolon then + Scan; -- past the semicolon + exit; + + -- Done if we are at a token which normally appears only after + -- a semicolon. One special glitch is that the keyword private is + -- in this category only if it does NOT appear after WITH. + + elsif (Token in Token_Class_After_SM + and then (Token /= Tok_Private + or else Prev_Token /= Tok_With)) + then + exit; + + -- Done if we are at THEN or LOOP + + elsif Token = Tok_Then or else Token = Tok_Loop then + exit; + + -- Otherwise keep going + + else + Scan; + end if; + end loop; + + -- Fall out of loop with resyncrhonization complete + + Resync_Resume; + end Resync_Past_Semicolon_Or_To_Loop_Or_Then; + + ------------------- + -- Resync_Resume -- + ------------------- + + procedure Resync_Resume is + begin + -- Save resync point (see special test in Resync_Init) + + Last_Resync_Point := Token_Ptr; + + if Debug_Flag_R then + Error_Msg_SC ("resuming here!"); + end if; + end Resync_Resume; + + -------------------- + -- Resync_To_When -- + -------------------- + + procedure Resync_To_When is + begin + Resync_Init; + + loop + -- Done if at semicolon, WHEN or IS + + if Token = Tok_Semicolon + or else Token = Tok_When + or else Token = Tok_Is + then + exit; + + -- Otherwise keep going + + else + Scan; + end if; + end loop; + + -- Fall out of loop with resyncrhonization complete + + Resync_Resume; + end Resync_To_When; + + --------------------------- + -- Resync_Semicolon_List -- + --------------------------- + + procedure Resync_Semicolon_List is + Paren_Count : Int; + + begin + Resync_Init; + Paren_Count := 0; + + loop + if Token = Tok_EOF + or else Token = Tok_Semicolon + or else Token = Tok_Is + or else Token in Token_Class_After_SM + then + exit; + + elsif Token = Tok_Left_Paren then + Paren_Count := Paren_Count + 1; + + elsif Token = Tok_Right_Paren then + if Paren_Count = 0 then + exit; + else + Paren_Count := Paren_Count - 1; + end if; + end if; + + Scan; + end loop; + + Resync_Resume; + end Resync_Semicolon_List; + +end Sync; diff --git a/gcc/ada/par-tchk.adb b/gcc/ada/par-tchk.adb new file mode 100644 index 00000000000..4d49e7af738 --- /dev/null +++ b/gcc/ada/par-tchk.adb @@ -0,0 +1,812 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . T C H K -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.37 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +-- Token scan routines. + +-- Error recovery: none of the T_xxx or TF_xxx routines raise Error_Resync + +separate (Par) +package body Tchk is + + type Position is (SC, BC, AP); + -- Specify position of error message (see Error_Msg_SC/BC/AP) + + ----------------------- + -- Local Subprograms -- + ----------------------- + + procedure Check_Token (T : Token_Type; P : Position); + pragma Inline (Check_Token); + -- Called by T_xx routines to check for reserved keyword token. P is the + -- position of the error message if the token is missing (see Wrong_Token) + + procedure Wrong_Token (T : Token_Type; P : Position); + -- Called when scanning a reserved keyword when the keyword is not + -- present. T is the token type for the keyword, and P indicates the + -- position to be used to place a message relative to the current + -- token if the keyword is not located nearby. + + ----------------- + -- Check_Token -- + ----------------- + + procedure Check_Token (T : Token_Type; P : Position) is + begin + if Token = T then + Scan; + return; + else + Wrong_Token (T, P); + end if; + end Check_Token; + + ------------- + -- T_Abort -- + ------------- + + procedure T_Abort is + begin + Check_Token (Tok_Abort, SC); + end T_Abort; + + ------------- + -- T_Arrow -- + ------------- + + procedure T_Arrow is + begin + if Token = Tok_Arrow then + Scan; + + -- A little recovery helper, accept then in place of => + + elsif Token = Tok_Then then + Error_Msg_BC ("missing ""=>"""); + Scan; -- past THEN used in place of => + + elsif Token = Tok_Colon_Equal then + Error_Msg_SC (""":="" should be ""=>"""); + Scan; -- past := used in place of => + + else + Error_Msg_AP ("missing ""=>"""); + end if; + end T_Arrow; + + ---------- + -- T_At -- + ---------- + + procedure T_At is + begin + Check_Token (Tok_At, SC); + end T_At; + + ------------ + -- T_Body -- + ------------ + + procedure T_Body is + begin + Check_Token (Tok_Body, BC); + end T_Body; + + ----------- + -- T_Box -- + ----------- + + procedure T_Box is + begin + if Token = Tok_Box then + Scan; + else + Error_Msg_AP ("missing ""<>"""); + end if; + end T_Box; + + ------------- + -- T_Colon -- + ------------- + + procedure T_Colon is + begin + if Token = Tok_Colon then + Scan; + else + Error_Msg_AP ("missing "":"""); + end if; + end T_Colon; + + ------------------- + -- T_Colon_Equal -- + ------------------- + + procedure T_Colon_Equal is + begin + if Token = Tok_Colon_Equal then + Scan; + + elsif Token = Tok_Equal then + Error_Msg_SC ("""="" should be "":="""); + Scan; + + elsif Token = Tok_Colon then + Error_Msg_SC (""":"" should be "":="""); + Scan; + + elsif Token = Tok_Is then + Error_Msg_SC ("IS should be "":="""); + Scan; + + else + Error_Msg_AP ("missing "":="""); + end if; + end T_Colon_Equal; + + ------------- + -- T_Comma -- + ------------- + + procedure T_Comma is + begin + if Token = Tok_Comma then + Scan; + + else + if Token = Tok_Pragma then + P_Pragmas_Misplaced; + end if; + + if Token = Tok_Comma then + Scan; + else + Error_Msg_AP ("missing "","""); + end if; + end if; + + if Token = Tok_Pragma then + P_Pragmas_Misplaced; + end if; + end T_Comma; + + --------------- + -- T_Dot_Dot -- + --------------- + + procedure T_Dot_Dot is + begin + if Token = Tok_Dot_Dot then + Scan; + else + Error_Msg_AP ("missing "".."""); + end if; + end T_Dot_Dot; + + ----------- + -- T_For -- + ----------- + + procedure T_For is + begin + Check_Token (Tok_For, AP); + end T_For; + + ----------------------- + -- T_Greater_Greater -- + ----------------------- + + procedure T_Greater_Greater is + begin + if Token = Tok_Greater_Greater then + Scan; + else + Error_Msg_AP ("missing "">>"""); + end if; + end T_Greater_Greater; + + ------------------ + -- T_Identifier -- + ------------------ + + procedure T_Identifier is + begin + if Token = Tok_Identifier then + Scan; + elsif Token in Token_Class_Literal then + Error_Msg_SC ("identifier expected"); + Scan; + else + Error_Msg_AP ("identifier expected"); + end if; + end T_Identifier; + + ---------- + -- T_In -- + ---------- + + procedure T_In is + begin + Check_Token (Tok_In, AP); + end T_In; + + ---------- + -- T_Is -- + ---------- + + procedure T_Is is + begin + if Token = Tok_Is then + Scan; + + Ignore (Tok_Semicolon); + + -- Allow OF, => or = to substitute for IS with complaint + + elsif Token = Tok_Arrow + or else Token = Tok_Of + or else Token = Tok_Equal + then + Error_Msg_SC ("missing IS"); + Scan; -- token used in place of IS + else + Wrong_Token (Tok_Is, AP); + end if; + + while Token = Tok_Is loop + Error_Msg_SC ("extra IS ignored"); + Scan; + end loop; + end T_Is; + + ------------------ + -- T_Left_Paren -- + ------------------ + + procedure T_Left_Paren is + begin + if Token = Tok_Left_Paren then + Scan; + else + Error_Msg_AP ("missing ""("""); + end if; + end T_Left_Paren; + + ------------ + -- T_Loop -- + ------------ + + procedure T_Loop is + begin + if Token = Tok_Do then + Error_Msg_SC ("LOOP expected"); + Scan; + else + Check_Token (Tok_Loop, AP); + end if; + end T_Loop; + + ----------- + -- T_Mod -- + ----------- + + procedure T_Mod is + begin + Check_Token (Tok_Mod, AP); + end T_Mod; + + ----------- + -- T_New -- + ----------- + + procedure T_New is + begin + Check_Token (Tok_New, AP); + end T_New; + + ---------- + -- T_Of -- + ---------- + + procedure T_Of is + begin + Check_Token (Tok_Of, AP); + end T_Of; + + ---------- + -- T_Or -- + ---------- + + procedure T_Or is + begin + Check_Token (Tok_Or, AP); + end T_Or; + + --------------- + -- T_Private -- + --------------- + + procedure T_Private is + begin + Check_Token (Tok_Private, SC); + end T_Private; + + ------------- + -- T_Range -- + ------------- + + procedure T_Range is + begin + Check_Token (Tok_Range, AP); + end T_Range; + + -------------- + -- T_Record -- + -------------- + + procedure T_Record is + begin + Check_Token (Tok_Record, AP); + end T_Record; + + ------------------- + -- T_Right_Paren -- + ------------------- + + procedure T_Right_Paren is + begin + if Token = Tok_Right_Paren then + Scan; + else + Error_Msg_AP ("missing "")"""); + end if; + end T_Right_Paren; + + ----------------- + -- T_Semicolon -- + ----------------- + + procedure T_Semicolon is + begin + + if Token = Tok_Semicolon then + Scan; + + if Token = Tok_Semicolon then + Error_Msg_SC ("extra "";"" ignored"); + Scan; + end if; + + elsif Token = Tok_Colon then + Error_Msg_SC (""":"" should be "";"""); + Scan; + + elsif Token = Tok_Comma then + Error_Msg_SC (""","" should be "";"""); + Scan; + + elsif Token = Tok_Dot then + Error_Msg_SC ("""."" should be "";"""); + Scan; + + -- An interesting little kludge here. If the previous token is a + -- semicolon, then there is no way that we can legitimately need + -- another semicolon. This could only arise in an error situation + -- where an error has already been signalled. By simply ignoring + -- the request for a semicolon in this case, we avoid some spurious + -- missing semicolon messages. + + elsif Prev_Token = Tok_Semicolon then + return; + + -- If the current token is | then this is a reasonable + -- place to suggest the possibility of a "C" confusion :-) + + elsif Token = Tok_Vertical_Bar then + Error_Msg_SC ("unexpected occurrence of ""|"", did you mean OR'?"); + Resync_Past_Semicolon; + + -- Otherwise we really do have a missing semicolon + + else + Error_Msg_AP ("missing "";"""); + return; + end if; + + end T_Semicolon; + + ------------ + -- T_Then -- + ------------ + + procedure T_Then is + begin + Check_Token (Tok_Then, AP); + end T_Then; + + ------------ + -- T_Type -- + ------------ + + procedure T_Type is + begin + Check_Token (Tok_Type, BC); + end T_Type; + + ----------- + -- T_Use -- + ----------- + + procedure T_Use is + begin + Check_Token (Tok_Use, SC); + end T_Use; + + ------------ + -- T_When -- + ------------ + + procedure T_When is + begin + Check_Token (Tok_When, SC); + end T_When; + + ------------ + -- T_With -- + ------------ + + procedure T_With is + begin + Check_Token (Tok_With, BC); + end T_With; + + -------------- + -- TF_Arrow -- + -------------- + + procedure TF_Arrow is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Arrow then + Scan; -- skip arrow and we are done + + elsif Token = Tok_Colon_Equal then + T_Arrow; -- Let T_Arrow give the message + + else + T_Arrow; -- give missing arrow message + Save_Scan_State (Scan_State); -- at start of junk tokens + + loop + if Prev_Token_Ptr < Current_Line_Start + or else Token = Tok_Semicolon + or else Token = Tok_EOF + then + Restore_Scan_State (Scan_State); -- to where we were! + return; + end if; + + Scan; -- continue search! + + if Token = Tok_Arrow then + Scan; -- past arrow + return; + end if; + end loop; + end if; + end TF_Arrow; + + ----------- + -- TF_Is -- + ----------- + + procedure TF_Is is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Is then + T_Is; -- past IS and we are done + + -- Allow OF or => or = in place of IS (with error message) + + elsif Token = Tok_Of + or else Token = Tok_Arrow + or else Token = Tok_Equal + then + T_Is; -- give missing IS message and skip bad token + + else + T_Is; -- give missing IS message + Save_Scan_State (Scan_State); -- at start of junk tokens + + loop + if Prev_Token_Ptr < Current_Line_Start + or else Token = Tok_Semicolon + or else Token = Tok_EOF + then + Restore_Scan_State (Scan_State); -- to where we were! + return; + end if; + + Scan; -- continue search! + + if Token = Tok_Is + or else Token = Tok_Of + or else Token = Tok_Arrow + then + Scan; -- past IS or OF or => + return; + end if; + end loop; + end if; + end TF_Is; + + ------------- + -- TF_Loop -- + ------------- + + procedure TF_Loop is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Loop then + Scan; -- past LOOP and we are done + + -- Allow DO or THEN in place of LOOP + + elsif Token = Tok_Then or else Token = Tok_Do then + T_Loop; -- give missing LOOP message + + else + T_Loop; -- give missing LOOP message + Save_Scan_State (Scan_State); -- at start of junk tokens + + loop + if Prev_Token_Ptr < Current_Line_Start + or else Token = Tok_Semicolon + or else Token = Tok_EOF + then + Restore_Scan_State (Scan_State); -- to where we were! + return; + end if; + + Scan; -- continue search! + + if Token = Tok_Loop or else Token = Tok_Then then + Scan; -- past loop or then (message already generated) + return; + end if; + end loop; + end if; + end TF_Loop; + + -------------- + -- TF_Return-- + -------------- + + procedure TF_Return is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Return then + Scan; -- skip RETURN and we are done + + else + Error_Msg_SC ("missing RETURN"); + Save_Scan_State (Scan_State); -- at start of junk tokens + + loop + if Prev_Token_Ptr < Current_Line_Start + or else Token = Tok_Semicolon + or else Token = Tok_EOF + then + Restore_Scan_State (Scan_State); -- to where we were! + return; + end if; + + Scan; -- continue search! + + if Token = Tok_Return then + Scan; -- past RETURN + return; + end if; + end loop; + end if; + end TF_Return; + + ------------------ + -- TF_Semicolon -- + ------------------ + + procedure TF_Semicolon is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Semicolon then + T_Semicolon; + return; + + -- An interesting little kludge here. If the previous token is a + -- semicolon, then there is no way that we can legitimately need + -- another semicolon. This could only arise in an error situation + -- where an error has already been signalled. By simply ignoring + -- the request for a semicolon in this case, we avoid some spurious + -- missing semicolon messages. + + elsif Prev_Token = Tok_Semicolon then + return; + + else + if Token = Tok_Pragma then + P_Pragmas_Misplaced; + + if Token = Tok_Semicolon then + T_Semicolon; + return; + end if; + end if; + + T_Semicolon; -- give missing semicolon message + Save_Scan_State (Scan_State); -- at start of junk tokens + + loop + if Prev_Token_Ptr < Current_Line_Start + or else Token = Tok_EOF + then + Restore_Scan_State (Scan_State); -- to where we were + return; + end if; + + Scan; -- continue search + + if Token = Tok_Semicolon then + T_Semicolon; + return; + + elsif Token in Token_Class_After_SM then + return; + end if; + end loop; + end if; + end TF_Semicolon; + + ------------- + -- TF_Then -- + ------------- + + procedure TF_Then is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Then then + Scan; -- past THEN and we are done + + else + T_Then; -- give missing THEN message + Save_Scan_State (Scan_State); -- at start of junk tokens + + loop + if Prev_Token_Ptr < Current_Line_Start + or else Token = Tok_Semicolon + or else Token = Tok_EOF + then + Restore_Scan_State (Scan_State); -- to where we were + return; + end if; + + Scan; -- continue search! + + if Token = Tok_Then then + Scan; -- past THEN + return; + end if; + end loop; + end if; + end TF_Then; + + ------------ + -- TF_Use -- + ------------ + + procedure TF_Use is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Use then + Scan; -- past USE and we are done + + else + T_Use; -- give USE expected message + Save_Scan_State (Scan_State); -- at start of junk tokens + + loop + if Prev_Token_Ptr < Current_Line_Start + or else Token = Tok_Semicolon + or else Token = Tok_EOF + then + Restore_Scan_State (Scan_State); -- to where we were + return; + end if; + + Scan; -- continue search! + + if Token = Tok_Use then + Scan; -- past use + return; + end if; + end loop; + end if; + end TF_Use; + + ----------------- + -- Wrong_Token -- + ----------------- + + procedure Wrong_Token (T : Token_Type; P : Position) is + Missing : constant String := "missing "; + Image : constant String := Token_Type'Image (T); + Tok_Name : constant String := Image (5 .. Image'Length); + M : String (1 .. Missing'Length + Tok_Name'Length); + + begin + -- Set M to Missing & Tok_Name. + + M (1 .. Missing'Length) := Missing; + M (Missing'Length + 1 .. M'Last) := Tok_Name; + + if Token = Tok_Semicolon then + Scan; + + if Token = T then + Error_Msg_SP ("extra "";"" ignored"); + Scan; + else + Error_Msg_SP (M); + end if; + + elsif Token = Tok_Comma then + Scan; + + if Token = T then + Error_Msg_SP ("extra "","" ignored"); + Scan; + + else + Error_Msg_SP (M); + end if; + + else + case P is + when SC => Error_Msg_SC (M); + when BC => Error_Msg_BC (M); + when AP => Error_Msg_AP (M); + end case; + end if; + end Wrong_Token; + +end Tchk; diff --git a/gcc/ada/par-util.adb b/gcc/ada/par-util.adb new file mode 100644 index 00000000000..f8082b64ee6 --- /dev/null +++ b/gcc/ada/par-util.adb @@ -0,0 +1,638 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R . U T I L -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.64 $ +-- -- +-- 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 Uintp; use Uintp; + +with GNAT.Spelling_Checker; use GNAT.Spelling_Checker; + +separate (Par) +package body Util is + + --------------------- + -- Bad_Spelling_Of -- + --------------------- + + function Bad_Spelling_Of (T : Token_Type) return Boolean is + Tname : constant String := Token_Type'Image (T); + -- Characters of token name + + S : String (1 .. Tname'Last - 4); + -- Characters of token name folded to lower case, omitting TOK_ at start + + M1 : String (1 .. 42) := "incorrect spelling of keyword ************"; + M2 : String (1 .. 44) := "illegal abbreviation of keyword ************"; + -- Buffers used to construct error message + + P1 : constant := 30; + P2 : constant := 32; + -- Starting subscripts in M1, M2 for keyword name + + SL : constant Natural := S'Length; + -- Length of expected token name excluding TOK_ at start + + begin + if Token /= Tok_Identifier then + return False; + end if; + + for J in S'Range loop + S (J) := Fold_Lower (Tname (Integer (J) + 4)); + end loop; + + Get_Name_String (Token_Name); + + -- A special check for case of PROGRAM used for PROCEDURE + + if T = Tok_Procedure + and then Name_Len = 7 + and then Name_Buffer (1 .. 7) = "program" + then + Error_Msg_SC ("PROCEDURE expected"); + Token := T; + return True; + + -- A special check for an illegal abbrevation + + elsif Name_Len < S'Length + and then Name_Len >= 4 + and then Name_Buffer (1 .. Name_Len) = S (1 .. Name_Len) + then + for J in 1 .. S'Last loop + M2 (P2 + J - 1) := Fold_Upper (S (J)); + end loop; + + Error_Msg_SC (M2 (1 .. P2 - 1 + S'Last)); + Token := T; + return True; + end if; + + -- Now we go into the full circuit to check for a misspelling + + -- Never consider something a misspelling if either the actual or + -- expected string is less than 3 characters (before this check we + -- used to consider i to be a misspelled if in some cases!) + + if SL < 3 or else Name_Len < 3 then + return False; + + -- Special case: prefix matches, i.e. the leading characters of the + -- token that we have exactly match the required keyword. If there + -- are at least two characters left over, assume that we have a case + -- of two keywords joined together which should not be joined. + + elsif Name_Len > SL + 1 + and then S = Name_Buffer (1 .. SL) + then + Scan_Ptr := Token_Ptr + S'Length; + Error_Msg_S ("missing space"); + Token := T; + return True; + end if; + + if Is_Bad_Spelling_Of (Name_Buffer (1 .. Name_Len), S) then + + for J in 1 .. S'Last loop + M1 (P1 + J - 1) := Fold_Upper (S (J)); + end loop; + + Error_Msg_SC (M1 (1 .. P1 - 1 + S'Last)); + Token := T; + return True; + + else + return False; + end if; + + end Bad_Spelling_Of; + + ---------------------- + -- Check_95_Keyword -- + ---------------------- + + -- On entry, the caller has checked that current token is an identifier + -- whose name matches the name of the 95 keyword New_Tok. + + procedure Check_95_Keyword (Token_95, Next : Token_Type) is + Scan_State : Saved_Scan_State; + + begin + Save_Scan_State (Scan_State); -- at identifier/keyword + Scan; -- past identifier/keyword + + if Token = Next then + Restore_Scan_State (Scan_State); -- to identifier + Error_Msg_Name_1 := Token_Name; + Error_Msg_SC ("(Ada 83) keyword* cannot be used!"); + Token := Token_95; + else + Restore_Scan_State (Scan_State); -- to identifier + end if; + end Check_95_Keyword; + + ---------------------- + -- Check_Bad_Layout -- + ---------------------- + + procedure Check_Bad_Layout is + begin + if Style.RM_Column_Check and then Token_Is_At_Start_Of_Line + and then Start_Column <= Scope.Table (Scope.Last).Ecol + then + Error_Msg_BC ("(style) incorrect layout"); + end if; + end Check_Bad_Layout; + + -------------------------- + -- Check_Misspelling_Of -- + -------------------------- + + procedure Check_Misspelling_Of (T : Token_Type) is + begin + if Bad_Spelling_Of (T) then + null; + end if; + end Check_Misspelling_Of; + + ----------------------------- + -- Check_Simple_Expression -- + ----------------------------- + + procedure Check_Simple_Expression (E : Node_Id) is + begin + if Expr_Form = EF_Non_Simple then + Error_Msg_N ("this expression must be parenthesized", E); + end if; + end Check_Simple_Expression; + + --------------------------------------- + -- Check_Simple_Expression_In_Ada_83 -- + --------------------------------------- + + procedure Check_Simple_Expression_In_Ada_83 (E : Node_Id) is + begin + if Expr_Form = EF_Non_Simple then + if Ada_83 then + Error_Msg_N ("(Ada 83) this expression must be parenthesized!", E); + end if; + end if; + end Check_Simple_Expression_In_Ada_83; + + ------------------------ + -- Check_Subtype_Mark -- + ------------------------ + + function Check_Subtype_Mark (Mark : Node_Id) return Node_Id is + begin + if Nkind (Mark) = N_Identifier + or else Nkind (Mark) = N_Selected_Component + or else (Nkind (Mark) = N_Attribute_Reference + and then Is_Type_Attribute_Name (Attribute_Name (Mark))) + or else Mark = Error + then + return Mark; + else + Error_Msg ("subtype mark expected", Sloc (Mark)); + return Error; + end if; + end Check_Subtype_Mark; + + ------------------- + -- Comma_Present -- + ------------------- + + function Comma_Present return Boolean is + Scan_State : Saved_Scan_State; + Paren_Count : Nat; + + begin + -- First check, if a comma is present, then a comma is present! + + if Token = Tok_Comma then + T_Comma; + return True; + + -- If we have a right paren, then that is taken as ending the list + -- i.e. no comma is present. + + elsif Token = Tok_Right_Paren then + return False; + + -- If pragmas, then get rid of them and make a recursive call + -- to process what follows these pragmas. + + elsif Token = Tok_Pragma then + P_Pragmas_Misplaced; + return Comma_Present; + + -- At this stage we have an error, and the goal is to decide on whether + -- or not we should diagnose an error and report a (non-existent) + -- comma as being present, or simply to report no comma is present + + -- If we are a semicolon, then the question is whether we have a missing + -- right paren, or whether the semicolon should have been a comma. To + -- guess the right answer, we scan ahead keeping track of the paren + -- level, looking for a clue that helps us make the right decision. + + -- This approach is highly accurate in the single error case, and does + -- not make bad mistakes in the multiple error case (indeed we can't + -- really make a very bad decision at this point in any case). + + elsif Token = Tok_Semicolon then + Save_Scan_State (Scan_State); + Scan; -- past semicolon + + -- Check for being followed by identifier => which almost certainly + -- means we are still in a parameter list and the comma should have + -- been a semicolon (such a sequence could not follow a semicolon) + + if Token = Tok_Identifier then + Scan; + + if Token = Tok_Arrow then + goto Assume_Comma; + end if; + end if; + + -- If that test didn't work, loop ahead looking for a comma or + -- semicolon at the same parenthesis level. Always remember that + -- we can't go badly wrong in an error situation like this! + + Paren_Count := 0; + + -- Here is the look ahead loop, Paren_Count tells us whether the + -- token we are looking at is at the same paren level as the + -- suspicious semicolon that we are trying to figure out. + + loop + + -- If we hit another semicolon or an end of file, and we have + -- not seen a right paren or another comma on the way, then + -- probably the semicolon did end the list. Indeed that is + -- certainly the only single error correction possible here. + + if Token = Tok_Semicolon or else Token = Tok_EOF then + Restore_Scan_State (Scan_State); + return False; + + -- A comma at the same paren level as the semicolon is a strong + -- indicator that the semicolon should have been a comma, indeed + -- again this is the only possible single error correction. + + elsif Token = Tok_Comma then + exit when Paren_Count = 0; + + -- A left paren just bumps the paren count + + elsif Token = Tok_Left_Paren then + Paren_Count := Paren_Count + 1; + + -- A right paren that is at the same paren level as the semicolon + -- also means that the only possible single error correction is + -- to assume that the semicolon should have been a comma. If we + -- are not at the same paren level, then adjust the paren level. + + elsif Token = Tok_Right_Paren then + exit when Paren_Count = 0; + Paren_Count := Paren_Count - 1; + end if; + + -- Keep going, we haven't made a decision yet + + Scan; + end loop; + + -- If we fall through the loop, it means that we found a terminating + -- right paren or another comma. In either case it is reasonable to + -- assume that the semicolon was really intended to be a comma. Also + -- come here for the identifier arrow case. + + <<Assume_Comma>> + Restore_Scan_State (Scan_State); + Error_Msg_SC (""";"" illegal here, replaced by "","""); + Scan; -- past the semicolon + return True; + + -- If we are not at semicolon or a right paren, then we base the + -- decision on whether or not the next token can be part of an + -- expression. If not, then decide that no comma is present (the + -- caller will eventually generate a missing right parent message) + + elsif Token in Token_Class_Eterm then + return False; + + -- Otherwise we assume a comma is present, even if none is present, + -- since the next token must be part of an expression, so if we were + -- at the end of the list, then there is more than one error present. + + else + T_Comma; -- to give error + return True; + end if; + end Comma_Present; + + ----------------------- + -- Discard_Junk_List -- + ----------------------- + + procedure Discard_Junk_List (L : List_Id) is + begin + null; + end Discard_Junk_List; + + ----------------------- + -- Discard_Junk_Node -- + ----------------------- + + procedure Discard_Junk_Node (N : Node_Id) is + begin + null; + end Discard_Junk_Node; + + ------------ + -- Ignore -- + ------------ + + procedure Ignore (T : Token_Type) is + begin + if Token = T then + if T = Tok_Comma then + Error_Msg_SC ("unexpected "","" ignored"); + + elsif T = Tok_Left_Paren then + Error_Msg_SC ("unexpected ""("" ignored"); + + elsif T = Tok_Right_Paren then + Error_Msg_SC ("unexpected "")"" ignored"); + + elsif T = Tok_Semicolon then + Error_Msg_SC ("unexpected "";"" ignored"); + + else + declare + Tname : constant String := Token_Type'Image (Token); + Msg : String := "unexpected keyword ????????????????????????"; + + begin + -- Loop to copy characters of keyword name (ignoring Tok_) + + for J in 5 .. Tname'Last loop + Msg (J + 14) := Fold_Upper (Tname (J)); + end loop; + + Msg (Tname'Last + 15 .. Tname'Last + 22) := " ignored"; + Error_Msg_SC (Msg (1 .. Tname'Last + 22)); + end; + end if; + + Scan; -- Scan past ignored token + end if; + end Ignore; + + ---------------------------- + -- Is_Reserved_Identifier -- + ---------------------------- + + function Is_Reserved_Identifier return Boolean is + begin + if not Is_Reserved_Keyword (Token) then + return False; + + else + declare + Ident_Casing : constant Casing_Type := + Identifier_Casing (Current_Source_File); + + Key_Casing : constant Casing_Type := + Keyword_Casing (Current_Source_File); + + begin + -- If the casing of identifiers and keywords is different in + -- this source file, and the casing of this token matches the + -- keyword casing, then we return False, since it is pretty + -- clearly intended to be a keyword. + + if Ident_Casing /= Unknown + and then Key_Casing /= Unknown + and then Ident_Casing /= Key_Casing + and then Determine_Token_Casing = Key_Casing + then + return False; + + -- Otherwise assume that an identifier was intended + + else + return True; + end if; + end; + end if; + end Is_Reserved_Identifier; + + ---------------------- + -- Merge_Identifier -- + ---------------------- + + procedure Merge_Identifier (Prev : Node_Id; Nxt : Token_Type) is + begin + if Token /= Tok_Identifier then + return; + end if; + + declare + S : Saved_Scan_State; + T : Token_Type; + + begin + Save_Scan_State (S); + Scan; + T := Token; + Restore_Scan_State (S); + + if T /= Nxt then + return; + end if; + end; + + -- Check exactly one space between identifiers + + if Source (Token_Ptr - 1) /= ' ' + or else Int (Token_Ptr) /= + Int (Prev_Token_Ptr) + Length_Of_Name (Chars (Prev)) + 1 + then + return; + end if; + + -- Do the merge + + Get_Name_String (Chars (Token_Node)); + + declare + Buf : String (1 .. Name_Len) := Name_Buffer (1 .. Name_Len); + + begin + Get_Name_String (Chars (Prev)); + Add_Char_To_Name_Buffer ('_'); + Add_Str_To_Name_Buffer (Buf); + Set_Chars (Prev, Name_Find); + end; + + Error_Msg_Node_1 := Prev; + Error_Msg_SC + ("unexpected identifier, possibly & was meant here"); + Scan; + end Merge_Identifier; + + ------------------- + -- No_Constraint -- + ------------------- + + procedure No_Constraint is + begin + if Token in Token_Class_Consk then + Error_Msg_SC ("constraint not allowed here"); + Discard_Junk_Node (P_Constraint_Opt); + end if; + end No_Constraint; + + -------------------- + -- No_Right_Paren -- + -------------------- + + function No_Right_Paren (Expr : Node_Id) return Node_Id is + begin + if Token = Tok_Right_Paren then + Error_Msg_SC ("unexpected right parenthesis"); + Resync_Expression; + return Error; + else + return Expr; + end if; + end No_Right_Paren; + + --------------------- + -- Pop_Scope_Stack -- + --------------------- + + procedure Pop_Scope_Stack is + begin + pragma Assert (Scope.Last > 0); + Scope.Decrement_Last; + + if Debug_Flag_P then + Error_Msg_Uint_1 := UI_From_Int (Scope.Last); + Error_Msg_SC ("decrement scope stack ptr, new value = ^!"); + end if; + end Pop_Scope_Stack; + + ---------------------- + -- Push_Scope_Stack -- + ---------------------- + + procedure Push_Scope_Stack is + begin + Scope.Increment_Last; + Scope.Table (Scope.Last).Junk := False; + Scope.Table (Scope.Last).Node := Empty; + + if Debug_Flag_P then + Error_Msg_Uint_1 := UI_From_Int (Scope.Last); + Error_Msg_SC ("increment scope stack ptr, new value = ^!"); + end if; + end Push_Scope_Stack; + + ---------------------- + -- Separate_Present -- + ---------------------- + + function Separate_Present return Boolean is + Scan_State : Saved_Scan_State; + + begin + if Token = Tok_Separate then + return True; + + elsif Token /= Tok_Identifier then + return False; + + else + Save_Scan_State (Scan_State); + Scan; -- past identifier + + if Token = Tok_Semicolon then + Restore_Scan_State (Scan_State); + return Bad_Spelling_Of (Tok_Separate); + + else + Restore_Scan_State (Scan_State); + return False; + end if; + end if; + end Separate_Present; + + -------------------------- + -- Signal_Bad_Attribute -- + -------------------------- + + procedure Signal_Bad_Attribute is + begin + Error_Msg_N ("unrecognized attribute&", Token_Node); + + -- Check for possible misspelling + + Get_Name_String (Token_Name); + + declare + AN : constant String := Name_Buffer (1 .. Name_Len); + + begin + Error_Msg_Name_1 := First_Attribute_Name; + while Error_Msg_Name_1 <= Last_Attribute_Name loop + Get_Name_String (Error_Msg_Name_1); + + if Is_Bad_Spelling_Of + (AN, Name_Buffer (1 .. Name_Len)) + then + Error_Msg_N + ("\possible misspelling of %", Token_Node); + exit; + end if; + + Error_Msg_Name_1 := Error_Msg_Name_1 + 1; + end loop; + end; + end Signal_Bad_Attribute; + + ------------------------------- + -- Token_Is_At_Start_Of_Line -- + ------------------------------- + + function Token_Is_At_Start_Of_Line return Boolean is + begin + return (Token_Ptr = First_Non_Blank_Location or else Token = Tok_EOF); + end Token_Is_At_Start_Of_Line; + +end Util; diff --git a/gcc/ada/par.adb b/gcc/ada/par.adb new file mode 100644 index 00000000000..f45a83b4390 --- /dev/null +++ b/gcc/ada/par.adb @@ -0,0 +1,1181 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.126 $ +-- -- +-- 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 Atree; use Atree; +with Casing; use Casing; +with Csets; use Csets; +with Debug; use Debug; +with Elists; use Elists; +with Errout; use Errout; +with Fname; use Fname; +with Lib; use Lib; +with Namet; use Namet; +with Nlists; use Nlists; +with Nmake; use Nmake; +with Opt; use Opt; +with Output; use Output; +with Scans; use Scans; +with Scn; use Scn; +with Sinput; use Sinput; +with Sinput.L; use Sinput.L; +with Sinfo; use Sinfo; +with Snames; use Snames; +with Style; +with Table; + +function Par (Configuration_Pragmas : Boolean) return List_Id is + + Num_Library_Units : Natural := 0; + -- Count number of units parsed (relevant only in syntax check only mode, + -- since in semantics check mode only a single unit is permitted anyway) + + Unit_Node : Node_Id; + -- Stores compilation unit node for current unit + + Save_Config_Switches : Config_Switches_Type; + -- Variable used to save values of config switches while we parse the + -- new unit, to be restored on exit for proper recursive behavior. + + Loop_Block_Count : Nat := 0; + -- Counter used for constructing loop/block names (see the routine + -- Par.Ch5.Get_Loop_Block_Name) + + -------------------- + -- Error Recovery -- + -------------------- + + -- When an error is encountered, a call is made to one of the Error_Msg + -- routines to record the error. If the syntax scan is not derailed by the + -- error (e.g. a complaint that logical operators are inconsistent in an + -- EXPRESSION), then control returns from the Error_Msg call, and the + -- parse continues unimpeded. + + -- If on the other hand, the Error_Msg represents a situation from which + -- the parser cannot recover locally, the exception Error_Resync is raised + -- immediately after the call to Error_Msg. Handlers for Error_Resync + -- are located at strategic points to resynchronize the parse. For example, + -- when an error occurs in a statement, the handler skips to the next + -- semicolon and continues the scan from there. + + -- Each parsing procedure contains a note with the heading "Error recovery" + -- which shows if it can propagate the Error_Resync exception. In order + -- not to propagate the exception, a procedure must either contain its own + -- handler for this exception, or it must not call any other routines which + -- propagate the exception. + + -- Note: the arrangement of Error_Resync handlers is such that it should + -- never be possible to transfer control through a procedure which made + -- an entry in the scope stack, invalidating the contents of the stack. + + Error_Resync : exception; + -- Exception raised on error that is not handled locally, see above. + + Last_Resync_Point : Source_Ptr; + -- The resynchronization routines in Par.Sync run a risk of getting + -- stuck in an infinite loop if they do not skip a token, and the caller + -- keeps repeating the same resync call. On the other hand, if they skip + -- a token unconditionally, some recovery opportunities are missed. The + -- variable Last_Resync_Point records the token location previously set + -- by a Resync call, and if a subsequent Resync call occurs at the same + -- location, then the Resync routine does guarantee to skip a token. + + -------------------------------------------- + -- Handling Semicolon Used in Place of IS -- + -------------------------------------------- + + -- The following global variables are used in handling the error situation + -- of using a semicolon in place of IS in a subprogram declaration as in: + + -- procedure X (Y : Integer); + -- Q : Integer; + -- begin + -- ... + -- end; + + -- The two contexts in which this can appear are at the outer level, and + -- within a declarative region. At the outer level, we know something is + -- wrong as soon as we see the Q (or begin, if there are no declarations), + -- and we can immediately decide that the semicolon should have been IS. + + -- The situation in a declarative region is more complex. The declaration + -- of Q could belong to the outer region, and we do not know that we have + -- an error until we hit the begin. It is still not clear at this point + -- from a syntactic point of view that something is wrong, because the + -- begin could belong to the enclosing subprogram or package. However, we + -- can incorporate a bit of semantic knowledge and note that the body of + -- X is missing, so we definitely DO have an error. We diagnose this error + -- as semicolon in place of IS on the subprogram line. + + -- There are two styles for this diagnostic. If the begin immediately + -- follows the semicolon, then we can place a flag (IS expected) right + -- on the semicolon. Otherwise we do not detect the error until we hit + -- the begin which refers back to the line with the semicolon. + + -- To control the process in the second case, the following global + -- variables are set to indicate that we have a subprogram declaration + -- whose body is required and has not yet been found. The prefix SIS + -- stands for "Subprogram IS" handling. + + SIS_Entry_Active : Boolean; + -- Set True to indicate that an entry is active (i.e. that a subprogram + -- declaration has been encountered, and no body for this subprogram has + -- been encountered). The remaining fields are valid only if this is True. + + SIS_Labl : Node_Id; + -- Subprogram designator + + SIS_Sloc : Source_Ptr; + -- Source location of FUNCTION/PROCEDURE keyword + + SIS_Ecol : Column_Number; + -- Column number of FUNCTION/PROCEDURE keyword + + SIS_Semicolon_Sloc : Source_Ptr; + -- Source location of semicolon at end of subprogram declaration + + SIS_Declaration_Node : Node_Id; + -- Pointer to tree node for subprogram declaration + + SIS_Missing_Semicolon_Message : Error_Msg_Id; + -- Used to save message ID of missing semicolon message (which will be + -- modified to missing IS if necessary). Set to No_Error_Msg in the + -- normal (non-error) case. + + -- Five things can happen to an active SIS entry + + -- 1. If a BEGIN is encountered with an SIS entry active, then we have + -- exactly the situation in which we know the body of the subprogram is + -- missing. After posting an error message, we change the spec to a body, + -- rechaining the declarations that intervened between the spec and BEGIN. + + -- 2. Another subprogram declaration or body is encountered. In this + -- case the entry gets overwritten with the information for the new + -- subprogram declaration. We don't catch some nested cases this way, + -- but it doesn't seem worth the effort. + + -- 3. A nested declarative region (e.g. package declaration or package + -- body) is encountered. The SIS active indication is reset at the start + -- of such a nested region. Again, like case 2, this causes us to miss + -- some nested cases, but it doesn't seen worth the effort to stack and + -- unstack the SIS information. Maybe we will reconsider this if we ever + -- get a complaint about a missed case :-) + + -- 4. We encounter a valid pragma INTERFACE or IMPORT that effectively + -- supplies the missing body. In this case we reset the entry. + + -- 5. We encounter the end of the declarative region without encoutering + -- a BEGIN first. In this situation we simply reset the entry. We know + -- that there is a missing body, but it seems more reasonable to let the + -- later semantic checking discover this. + + -------------------------------------------- + -- Handling IS Used in Place of Semicolon -- + -------------------------------------------- + + -- This is a somewhat trickier situation, and we can't catch it in all + -- cases, but we do our best to detect common situations resulting from + -- a "cut and paste" operation which forgets to change the IS to semicolon. + -- Consider the following example: + + -- package body X is + -- procedure A; + -- procedure B is + -- procedure C; + -- ... + -- procedure D is + -- begin + -- ... + -- end; + -- begin + -- ... + -- end; + + -- The trouble is that the section of text from PROCEDURE B through END; + -- consitutes a valid procedure body, and the danger is that we find out + -- far too late that something is wrong (indeed most compilers will behave + -- uncomfortably on the above example). + + -- We have two approaches to helping to control this situation. First we + -- make every attempt to avoid swallowing the last END; if we can be + -- sure that some error will result from doing so. In particular, we won't + -- accept the END; unless it is exactly correct (in particular it must not + -- have incorrect name tokens), and we won't accept it if it is immediately + -- followed by end of file, WITH or SEPARATE (all tokens that unmistakeably + -- signal the start of a compilation unit, and which therefore allow us to + -- reserve the END; for the outer level.) For more details on this aspect + -- of the handling, see package Par.Endh. + + -- If we can avoid eating up the END; then the result in the absense of + -- any additional steps would be to post a missing END referring back to + -- the subprogram with the bogus IS. Similarly, if the enclosing package + -- has no BEGIN, then the result is a missing BEGIN message, which again + -- refers back to the subprogram header. + + -- Such an error message is not too bad (it's already a big improvement + -- over what many parsers do), but it's not ideal, because the declarations + -- following the IS have been absorbed into the wrong scope. In the above + -- case, this could result for example in a bogus complaint that the body + -- of D was missing from the package. + + -- To catch at least some of these cases, we take the following additional + -- steps. First, a subprogram body is marked as having a suspicious IS if + -- the declaration line is followed by a line which starts with a symbol + -- that can start a declaration in the same column, or to the left of the + -- column in which the FUNCTION or PROCEDURE starts (normal style is to + -- indent any declarations which really belong a subprogram). If such a + -- subprogram encounters a missing BEGIN or missing END, then we decide + -- that the IS should have been a semicolon, and the subprogram body node + -- is marked (by setting the Bad_Is_Detected flag true. Note that we do + -- not do this for library level procedures, only for nested procedures, + -- since for library level procedures, we must have a body. + + -- The processing for a declarative part checks to see if the last + -- declaration scanned is marked in this way, and if it is, the tree + -- is modified to reflect the IS being interpreted as a semicolon. + + --------------------------------------------------- + -- Parser Type Definitions and Control Variables -- + --------------------------------------------------- + + -- The following variable and associated type declaration are used by the + -- expression parsing routines to return more detailed information about + -- the categorization of a parsed expression. + + type Expr_Form_Type is ( + EF_Simple_Name, -- Simple name, i.e. possibly qualified identifier + EF_Name, -- Simple expression which could also be a name + EF_Simple, -- Simple expression which is not call or name + EF_Range_Attr, -- Range attribute reference + EF_Non_Simple); -- Expression that is not a simple expression + + Expr_Form : Expr_Form_Type; + + -- The following type is used for calls to P_Subprogram, P_Package, P_Task, + -- P_Protected to indicate which of several possibilities is acceptable. + + type Pf_Rec is record + Spcn : Boolean; -- True if specification OK + Decl : Boolean; -- True if declaration OK + Gins : Boolean; -- True if generic instantiation OK + Pbod : Boolean; -- True if proper body OK + Rnam : Boolean; -- True if renaming declaration OK + Stub : Boolean; -- True if body stub OK + Fil1 : Boolean; -- Filler to fill to 8 bits + Fil2 : Boolean; -- Filler to fill to 8 bits + end record; + pragma Pack (Pf_Rec); + + function T return Boolean renames True; + function F return Boolean renames False; + + Pf_Decl_Gins_Pbod_Rnam_Stub : constant Pf_Rec := + Pf_Rec'(F, T, T, T, T, T, F, F); + Pf_Decl : constant Pf_Rec := + Pf_Rec'(F, T, F, F, F, F, F, F); + Pf_Decl_Gins_Pbod_Rnam : constant Pf_Rec := + Pf_Rec'(F, T, T, T, T, F, F, F); + Pf_Decl_Pbod : constant Pf_Rec := + Pf_Rec'(F, T, F, T, F, F, F, F); + Pf_Pbod : constant Pf_Rec := + Pf_Rec'(F, F, F, T, F, F, F, F); + Pf_Spcn : constant Pf_Rec := + Pf_Rec'(T, F, F, F, F, F, F, F); + -- The above are the only allowed values of Pf_Rec arguments + + type SS_Rec is record + Eftm : Boolean; -- ELSIF can terminate sequence + Eltm : Boolean; -- ELSE can terminate sequence + Extm : Boolean; -- EXCEPTION can terminate sequence + Ortm : Boolean; -- OR can terminate sequence + Sreq : Boolean; -- at least one statement required + Tatm : Boolean; -- THEN ABORT can terminate sequence + Whtm : Boolean; -- WHEN can terminate sequence + Unco : Boolean; -- Unconditional terminate after one statement + end record; + pragma Pack (SS_Rec); + + SS_Eftm_Eltm_Sreq : constant SS_Rec := SS_Rec'(T, T, F, F, T, F, F, F); + SS_Eltm_Ortm_Tatm : constant SS_Rec := SS_Rec'(F, T, F, T, F, T, F, F); + SS_Extm_Sreq : constant SS_Rec := SS_Rec'(F, F, T, F, T, F, F, F); + SS_None : constant SS_Rec := SS_Rec'(F, F, F, F, F, F, F, F); + SS_Ortm_Sreq : constant SS_Rec := SS_Rec'(F, F, F, T, T, F, F, F); + SS_Sreq : constant SS_Rec := SS_Rec'(F, F, F, F, T, F, F, F); + SS_Sreq_Whtm : constant SS_Rec := SS_Rec'(F, F, F, F, T, F, T, F); + SS_Whtm : constant SS_Rec := SS_Rec'(F, F, F, F, F, F, T, F); + SS_Unco : constant SS_Rec := SS_Rec'(F, F, F, F, F, F, F, T); + + Label_List : Elist_Id; + -- List of label nodes for labels appearing in the current compilation. + -- Used by Par.Labl to construct the corresponding implicit declarations. + + ----------------- + -- Scope Table -- + ----------------- + + -- The scope table, also referred to as the scope stack, is used to + -- record the current scope context. It is organized as a stack, with + -- inner nested entries corresponding to higher entries on the stack. + -- An entry is made when the parser encounters the opening of a nested + -- construct (such as a record, task, package etc.), and then package + -- Par.Endh uses this stack to deal with END lines (including properly + -- dealing with END nesting errors). + + type SS_End_Type is + -- Type of end entry required for this scope. The last two entries are + -- used only in the subprogram body case to mark the case of a suspicious + -- IS, or a bad IS (i.e. suspicions confirmed by missing BEGIN or END). + -- See separate section on dealing with IS used in place of semicolon. + -- Note that for many purposes E_Name, E_Suspicious_Is and E_Bad_Is are + -- treated the same (E_Suspicious_Is and E_Bad_Is are simply special cases + -- of E_Name). They are placed at the end of the enumeration so that a + -- test for >= E_Name catches all three cases efficiently. + + (E_Dummy, -- dummy entry at outer level + E_Case, -- END CASE; + E_If, -- END IF; + E_Loop, -- END LOOP; + E_Record, -- END RECORD; + E_Select, -- END SELECT; + E_Name, -- END [name]; + E_Suspicious_Is, -- END [name]; (case of suspicious IS) + E_Bad_Is); -- END [name]; (case of bad IS) + + -- The following describes a single entry in the scope table + + type Scope_Table_Entry is record + Etyp : SS_End_Type; + -- Type of end entry, as per above description + + Lreq : Boolean; + -- A flag indicating whether the label, if present, is required to + -- appear on the end line. It is referenced only in the case of + -- Etyp = E_Name or E_Suspicious_Is where the name may or may not be + -- required (yes for labeled block, no in other cases). Note that for + -- all cases except begin, the question of whether a label is required + -- can be determined from the other fields (for loop, it is required if + -- it is present, and for the other constructs it is never required or + -- allowed). + + Ecol : Column_Number; + -- Contains the absolute column number (with tabs expanded) of the + -- the expected column of the end assuming normal Ada indentation + -- usage. If the RM_Column_Check mode is set, this value is used for + -- generating error messages about indentation. Otherwise it is used + -- only to control heuristic error recovery actions. + + Labl : Node_Id; + -- This field is used only for the LOOP and BEGIN cases, and is the + -- Node_Id value of the label name. For all cases except child units, + -- this value is an entity whose Chars field contains the name pointer + -- that identifies the label uniquely. For the child unit case the Labl + -- field references an N_Defining_Program_Unit_Name node for the name. + -- For cases other than LOOP or BEGIN, the Label field is set to Error, + -- indicating that it is an error to have a label on the end line. + + Decl : List_Id; + -- Points to the list of declarations (i.e. the declarative part) + -- associated with this construct. It is set only in the END [name] + -- cases, and is set to No_List for all other cases which do not have a + -- declarative unit associated with them. This is used for determining + -- the proper location for implicit label declarations. + + Node : Node_Id; + -- Empty except in the case of entries for IF and CASE statements, + -- in which case it contains the N_If_Statement or N_Case_Statement + -- node. This is used for setting the End_Span field. + + Sloc : Source_Ptr; + -- Source location of the opening token of the construct. This is + -- used to refer back to this line in error messages (such as missing + -- or incorrect end lines). The Sloc field is not used, and is not set, + -- if a label is present (the Labl field provides the text name of the + -- label in this case, which is fine for error messages). + + S_Is : Source_Ptr; + -- S_Is is relevant only if Etyp is set to E_Suspicious_Is or + -- E_Bad_Is. It records the location of the IS that is considered + -- to be suspicious. + + Junk : Boolean; + -- A boolean flag that is set true if the opening entry is the dubious + -- result of some prior error, e.g. a record entry where the record + -- keyword was missing. It is used to suppress the issuing of a + -- corresponding junk complaint about the end line (we do not want + -- to complain about a missing end record when there was no record). + end record; + + -- The following declares the scope table itself. The Last field is the + -- stack pointer, so that Scope.Table (Scope.Last) is the top entry. The + -- oldest entry, at Scope_Stack (0), is a dummy entry with Etyp set to + -- E_Dummy, and the other fields undefined. This dummy entry ensures that + -- Scope_Stack (Scope_Stack_Ptr).Etyp can always be tested, and that the + -- scope stack pointer is always in range. + + package Scope is new Table.Table ( + Table_Component_Type => Scope_Table_Entry, + Table_Index_Type => Int, + Table_Low_Bound => 0, + Table_Initial => 50, + Table_Increment => 100, + Table_Name => "Scope"); + + --------------------------------- + -- Parsing Routines by Chapter -- + --------------------------------- + + -- Uncommented declarations in this section simply parse the construct + -- corresponding to their name, and return an ID value for the Node or + -- List that is created. + + package Ch2 is + function P_Identifier return Node_Id; + function P_Pragma return Node_Id; + + function P_Pragmas_Opt return List_Id; + -- This function scans for a sequence of pragmas in other than a + -- declaration sequence or statement sequence context. All pragmas + -- can appear except pragmas Assert and Debug, which are only allowed + -- in a declaration or statement sequence context. + + procedure P_Pragmas_Misplaced; + -- Skips misplaced pragmas with a complaint + + procedure P_Pragmas_Opt (List : List_Id); + -- Parses optional pragmas and appends them to the List + end Ch2; + + package Ch3 is + Missing_Begin_Msg : Error_Msg_Id; + -- This variable is set by a call to P_Declarative_Part. Normally it + -- is set to No_Error_Msg, indicating that no special processing is + -- required by the caller. The special case arises when a statement + -- is found in the sequence of declarations. In this case the Id of + -- the message issued ("declaration expected") is preserved in this + -- variable, then the caller can change it to an appropriate missing + -- begin message if indeed the BEGIN is missing. + + function P_Access_Definition return Node_Id; + function P_Access_Type_Definition return Node_Id; + function P_Array_Type_Definition return Node_Id; + function P_Basic_Declarative_Items return List_Id; + function P_Constraint_Opt return Node_Id; + function P_Declarative_Part return List_Id; + function P_Defining_Identifier return Node_Id; + function P_Discrete_Choice_List return List_Id; + function P_Discrete_Range return Node_Id; + function P_Discrete_Subtype_Definition return Node_Id; + function P_Known_Discriminant_Part_Opt return List_Id; + function P_Signed_Integer_Type_Definition return Node_Id; + function P_Range return Node_Id; + function P_Range_Or_Subtype_Mark return Node_Id; + function P_Range_Constraint return Node_Id; + function P_Record_Definition return Node_Id; + function P_Subtype_Indication return Node_Id; + function P_Subtype_Mark return Node_Id; + function P_Subtype_Mark_Resync return Node_Id; + function P_Unknown_Discriminant_Part_Opt return Boolean; + + procedure P_Component_Items (Decls : List_Id); + -- Scan out one or more component items and append them to the + -- given list. Only scans out more than one declaration in the + -- case where the source has a single declaration with multiple + -- defining identifiers. + + function Init_Expr_Opt (P : Boolean := False) return Node_Id; + -- If an initialization expression is present (:= expression), then + -- it is scanned out and returned, otherwise Empty is returned if no + -- initialization expression is present. This procedure also handles + -- certain common error cases cleanly. The parameter P indicates if + -- a right paren can follow the expression (default = no right paren + -- allowed). + + procedure Skip_Declaration (S : List_Id); + -- Used when scanning statements to skip past a mispaced declaration + -- The declaration is scanned out and appended to the given list. + -- Token is known to be a declaration token (in Token_Class_Declk) + -- on entry, so there definition is a declaration to be scanned. + + function P_Subtype_Indication (Subtype_Mark : Node_Id) return Node_Id; + -- This version of P_Subtype_Indication is called when the caller has + -- already scanned out the subtype mark which is passed as a parameter. + + function P_Subtype_Mark_Attribute (Type_Node : Node_Id) return Node_Id; + -- Parse a subtype mark attribute. The caller has already parsed the + -- subtype mark, which is passed in as the argument, and has checked + -- that the current token is apostrophe. + + end Ch3; + + package Ch4 is + function P_Aggregate return Node_Id; + function P_Expression return Node_Id; + function P_Expression_No_Right_Paren return Node_Id; + function P_Expression_Or_Range_Attribute return Node_Id; + function P_Function_Name return Node_Id; + function P_Name return Node_Id; + function P_Qualified_Simple_Name return Node_Id; + function P_Qualified_Simple_Name_Resync return Node_Id; + function P_Simple_Expression return Node_Id; + function P_Simple_Expression_Or_Range_Attribute return Node_Id; + + function P_Qualified_Expression + (Subtype_Mark : Node_Id) + return Node_Id; + -- This routine scans out a qualified expression when the caller has + -- already scanned out the name and apostrophe of the construct. + + end Ch4; + + package Ch5 is + + function P_Statement_Name (Name_Node : Node_Id) return Node_Id; + -- Given a node representing a name (which is a call), converts it + -- to the syntactically corresponding procedure call statement. + + function P_Sequence_Of_Statements (SS_Flags : SS_Rec) return List_Id; + -- The argument indicates the acceptable termination tokens. + -- See body in Par.Ch5 for details of the use of this parameter. + + procedure Parse_Decls_Begin_End (Parent : Node_Id); + -- Parses declarations and handled statement sequence, setting + -- fields of Parent node appropriately. + + end Ch5; + + package Ch6 is + function P_Designator return Node_Id; + function P_Defining_Program_Unit_Name return Node_Id; + function P_Formal_Part return List_Id; + function P_Parameter_Profile return List_Id; + function P_Return_Statement return Node_Id; + function P_Subprogram_Specification return Node_Id; + + procedure P_Mode (Node : Node_Id); + -- Sets In_Present and/or Out_Present flags in Node scanning past + -- IN, OUT or IN OUT tokens in the source. + + function P_Subprogram (Pf_Flags : Pf_Rec) return Node_Id; + -- Scans out any construct starting with either of the keywords + -- PROCEDURE or FUNCTION. The parameter indicates which possible + -- possible kinds of construct (body, spec, instantiation etc.) + -- are permissible in the current context. + + end Ch6; + + package Ch7 is + function P_Package (Pf_Flags : Pf_Rec) return Node_Id; + -- Scans out any construct starting with the keyword PACKAGE. The + -- parameter indicates which possible kinds of construct (body, spec, + -- instantiation etc.) are permissible in the current context. + end Ch7; + + package Ch8 is + function P_Use_Clause return Node_Id; + end Ch8; + + package Ch9 is + function P_Abort_Statement return Node_Id; + function P_Abortable_Part return Node_Id; + function P_Accept_Statement return Node_Id; + function P_Delay_Statement return Node_Id; + function P_Entry_Body return Node_Id; + function P_Protected return Node_Id; + function P_Requeue_Statement return Node_Id; + function P_Select_Statement return Node_Id; + function P_Task return Node_Id; + function P_Terminate_Alternative return Node_Id; + end Ch9; + + package Ch10 is + function P_Compilation_Unit return Node_Id; + -- Note: this function scans a single compilation unit, and + -- checks that an end of file follows this unit, diagnosing + -- any unexpected input as an error, and then skipping it, so + -- that Token is set to Tok_EOF on return. An exception is in + -- syntax-only mode, where multiple compilation units are + -- permitted. In this case, P_Compilation_Unit does not check + -- for end of file and there may be more compilation units to + -- scan. The caller can uniquely detect this situation by the + -- fact that Token is not set to Tok_EOF on return. + end Ch10; + + package Ch11 is + function P_Handled_Sequence_Of_Statements return Node_Id; + function P_Raise_Statement return Node_Id; + + function Parse_Exception_Handlers return List_Id; + -- Parses the partial construct EXCEPTION followed by a list of + -- exception handlers which appears in a number of productions, + -- and returns the list of exception handlers. + + end Ch11; + + package Ch12 is + function P_Generic return Node_Id; + function P_Generic_Actual_Part_Opt return List_Id; + end Ch12; + + package Ch13 is + function P_Representation_Clause return Node_Id; + + function P_Code_Statement (Subtype_Mark : Node_Id) return Node_Id; + -- Function to parse a code statement. The caller has scanned out + -- the name to be used as the subtype mark (but has not checked that + -- it is suitable for use as a subtype mark, i.e. is either an + -- identifier or a selected component). The current token is an + -- apostrophe and the following token is either a left paren or + -- RANGE (the latter being an error to be caught by P_Code_Statement. + end Ch13; + + -- Note: the parsing for annexe J features (i.e. obsolescent features) + -- is found in the logical section where these features would be if + -- they were not obsolescent. In particular: + + -- Delta constraint is parsed by P_Delta_Constraint (3.5.9) + -- At clause is parsed by P_At_Clause (13.1) + -- Mod clause is parsed by P_Mod_Clause (13.5.1) + + ------------------ + -- End Handling -- + ------------------ + + -- Routines for handling end lines, including scope recovery + + package Endh is + + function Check_End return Boolean; + -- Called when an end sequence is required. In the absence of an error + -- situation, Token contains Tok_End on entry, but in a missing end + -- case, this may not be the case. Pop_End_Context is used to determine + -- the appropriate action to be taken. The returned result is True if + -- an End sequence was encountered and False if no End sequence was + -- present. This occurs if the END keyword encountered was determined + -- to be improper and deleted (i.e. Pop_End_Context set End_Action to + -- Skip_And_Reject). Note that the END sequence includes a semicolon, + -- except in the case of END RECORD, where a semicolon follows the END + -- RECORD, but is not part of the record type definition itself. + + procedure End_Skip; + -- Skip past an end sequence. On entry Token contains Tok_End, and we + -- we know that the end sequence is syntactically incorrect, and that + -- an appropriate error message has already been posted. The mission + -- is simply to position the scan pointer to be the best guess of the + -- position after the end sequence. We do not issue any additional + -- error messages while carrying this out. + + procedure End_Statements (Parent : Node_Id := Empty); + -- Called when an end is required or expected to terminate a sequence + -- of statements. The caller has already made an appropriate entry in + -- the Scope.Table to describe the expected form of the end. This can + -- only be used in cases where the only appropriate terminator is end. + -- If Parent is non-empty, then if a correct END line is encountered, + -- the End_Label field of Parent is set appropriately. + + end Endh; + + ------------------------------------ + -- Resynchronization After Errors -- + ------------------------------------ + + -- These procedures are used to resynchronize after errors. Following an + -- error which is not immediately locally recoverable, the exception + -- Error_Resync is raised. The handler for Error_Resync typically calls + -- one of these recovery procedures to resynchronize the source position + -- to a point from which parsing can be restarted. + + -- Note: these procedures output an information message that tokens are + -- being skipped, but this message is output only if the option for + -- Multiple_Errors_Per_Line is set in Options. + + package Sync is + + procedure Resync_Choice; + -- Used if an error occurs scanning a choice. The scan pointer is + -- advanced to the next vertical bar, arrow, or semicolon, whichever + -- comes first. We also quit if we encounter an end of file. + + procedure Resync_Expression; + -- Used if an error is detected during the parsing of an expression. + -- It skips past tokens until either a token which cannot be part of + -- an expression is encountered (an expression terminator), or if a + -- comma or right parenthesis or vertical bar is encountered at the + -- current parenthesis level (a parenthesis level counter is maintained + -- to carry out this test). + + procedure Resync_Past_Semicolon; + -- Used if an error occurs while scanning a sequence of declarations. + -- The scan pointer is positioned past the next semicolon and the scan + -- resumes. The scan is also resumed on encountering a token which + -- starts a declaration (but we make sure to skip at least one token + -- in this case, to avoid getting stuck in a loop). + + procedure Resync_Past_Semicolon_Or_To_Loop_Or_Then; + -- Used if an error occurs while scanning a sequence of statements. + -- The scan pointer is positioned past the next semicolon, or to the + -- next occurrence of either then or loop, and the scan resumes. + + procedure Resync_To_When; + -- Used when an error occurs scanning an entry index specification. + -- The scan pointer is positioned to the next WHEN (or to IS or + -- semicolon if either of these appear before WHEN, indicating + -- another error has occurred). + + procedure Resync_Semicolon_List; + -- Used if an error occurs while scanning a parenthesized list of items + -- separated by semicolons. The scan pointer is advanced to the next + -- semicolon or right parenthesis at the outer parenthesis level, or + -- to the next is or RETURN keyword occurence, whichever comes first. + + procedure Resync_Cunit; + -- Synchronize to next token which could be the start of a compilation + -- unit, or to the end of file token. + + end Sync; + + ------------------------- + -- Token Scan Routines -- + ------------------------- + + -- Routines to check for expected tokens + + package Tchk is + + -- Procedures with names of the form T_xxx, where Tok_xxx is a token + -- name, check that the current token matches the required token, and + -- if so, scan past it. If not, an error is issued indicating that + -- the required token is not present (xxx expected). In most cases, the + -- scan pointer is not moved in the not-found case, but there are some + -- exceptions to this, see for example T_Id, where the scan pointer is + -- moved across a literal appearing where an identifier is expected. + + procedure T_Abort; + procedure T_Arrow; + procedure T_At; + procedure T_Body; + procedure T_Box; + procedure T_Colon; + procedure T_Colon_Equal; + procedure T_Comma; + procedure T_Dot_Dot; + procedure T_For; + procedure T_Greater_Greater; + procedure T_Identifier; + procedure T_In; + procedure T_Is; + procedure T_Left_Paren; + procedure T_Loop; + procedure T_Mod; + procedure T_New; + procedure T_Of; + procedure T_Or; + procedure T_Private; + procedure T_Range; + procedure T_Record; + procedure T_Right_Paren; + procedure T_Semicolon; + procedure T_Then; + procedure T_Type; + procedure T_Use; + procedure T_When; + procedure T_With; + + -- Procedures have names of the form TF_xxx, where Tok_xxx is a token + -- name check that the current token matches the required token, and + -- if so, scan past it. If not, an error message is issued indicating + -- that the required token is not present (xxx expected). + + -- If the missing token is at the end of the line, then control returns + -- immediately after posting the message. If there are remaining tokens + -- on the current line, a search is conducted to see if the token + -- appears later on the current line, as follows: + + -- A call to Scan_Save is issued and a forward search for the token + -- is carried out. If the token is found on the current line before a + -- semicolon, then it is scanned out and the scan continues from that + -- point. If not the scan is restored to the point where it was missing. + + procedure TF_Arrow; + procedure TF_Is; + procedure TF_Loop; + procedure TF_Return; + procedure TF_Semicolon; + procedure TF_Then; + procedure TF_Use; + + end Tchk; + + ---------------------- + -- Utility Routines -- + ---------------------- + + package Util is + + function Bad_Spelling_Of (T : Token_Type) return Boolean; + -- This function is called in an error situation. It checks if the + -- current token is an identifier whose name is a plausible bad + -- spelling of the given keyword token, and if so, issues an error + -- message, sets Token from T, and returns True. Otherwise Token is + -- unchanged, and False is returned. + + procedure Check_Bad_Layout; + -- Check for bad indentation in RM checking mode. Used for statements + -- and declarations. Checks if current token is at start of line and + -- is exdented from the current expected end column, and if so an + -- error message is generated. + + procedure Check_Misspelling_Of (T : Token_Type); + pragma Inline (Check_Misspelling_Of); + -- This is similar to the function above, except that it does not + -- return a result. It is typically used in a situation where any + -- identifier is an error, and it makes sense to simply convert it + -- to the given token if it is a plausible misspelling of it. + + procedure Check_95_Keyword (Token_95, Next : Token_Type); + -- This routine checks if the token after the current one matches the + -- Next argument. If so, the scan is backed up to the current token + -- and Token_Type is changed to Token_95 after issuing an appropriate + -- error message ("(Ada 83) keyword xx cannot be used"). If not, + -- the scan is backed up with Token_Type unchanged. This routine + -- is used to deal with an attempt to use a 95 keyword in Ada 83 + -- mode. The caller has typically checked that the current token, + -- an identifier, matches one of the 95 keywords. + + procedure Check_Simple_Expression (E : Node_Id); + -- Given an expression E, that has just been scanned, so that Expr_Form + -- is still set, outputs an error if E is a non-simple expression. E is + -- not modified by this call. + + procedure Check_Simple_Expression_In_Ada_83 (E : Node_Id); + -- Like Check_Simple_Expression, except that the error message is only + -- given when operating in Ada 83 mode, and includes "in Ada 83". + + function Check_Subtype_Mark (Mark : Node_Id) return Node_Id; + -- Called to check that a node representing a name (or call) is + -- suitable for a subtype mark, i.e, that it is an identifier or + -- a selected component. If so, or if it is already Error, then + -- it is returned unchanged. Otherwise an error message is issued + -- and Error is returned. + + function Comma_Present return Boolean; + -- Used in comma delimited lists to determine if a comma is present, or + -- can reasonably be assumed to have been present (an error message is + -- generated in the latter case). If True is returned, the scan has been + -- positioned past the comma. If False is returned, the scan position + -- is unchanged. Note that all comma-delimited lists are terminated by + -- a right paren, so the only legitimate tokens when Comma_Present is + -- called are right paren and comma. If some other token is found, then + -- Comma_Present has the job of deciding whether it is better to pretend + -- a comma was present, post a message for a missing comma and return + -- True, or return False and let the caller diagnose the missing right + -- parenthesis. + + procedure Discard_Junk_Node (N : Node_Id); + procedure Discard_Junk_List (L : List_Id); + pragma Inline (Discard_Junk_Node); + pragma Inline (Discard_Junk_List); + -- These procedures do nothing at all, their effect is simply to discard + -- the argument. A typical use is to skip by some junk that is not + -- expected in the current context. + + procedure Ignore (T : Token_Type); + -- If current token matches T, then give an error message and skip + -- past it, otherwise the call has no effect at all. T may be any + -- reserved word token, or comma, left or right paren, or semicolon. + + function Is_Reserved_Identifier return Boolean; + -- Test if current token is a reserved identifier. This test is based + -- on the token being a keyword and being spelled in typical identifier + -- style (i.e. starting with an upper case letter). + + procedure Merge_Identifier (Prev : Node_Id; Nxt : Token_Type); + -- Called when the previous token is an identifier (whose Token_Node + -- value is given by Prev) to check if current token is an identifier + -- that can be merged with the previous one adding an underscore. The + -- merge is only attempted if the following token matches Nxt. If all + -- conditions are met, an error message is issued, and the merge is + -- carried out, modifying the Chars field of Prev. + + procedure No_Constraint; + -- Called in a place where no constraint is allowed, but one might + -- appear due to a common error (e.g. after the type mark in a procedure + -- parameter. If a constraint is present, an error message is posted, + -- and the constraint is scanned and discarded. + + function No_Right_Paren (Expr : Node_Id) return Node_Id; + -- Function to check for no right paren at end of expression, returns + -- its argument if no right paren, else flags paren and returns Error. + + procedure Push_Scope_Stack; + pragma Inline (Push_Scope_Stack); + -- Push a new entry onto the scope stack. Scope.Last (the stack pointer) + -- is incremented. The Junk field is preinitialized to False. The caller + -- is expected to fill in all remaining entries of the new new top stack + -- entry at Scope.Table (Scope.Last). + + procedure Pop_Scope_Stack; + -- Pop an entry off the top of the scope stack. Scope_Last (the scope + -- table stack pointer) is decremented by one. It is a fatal error to + -- try to pop off the dummy entry at the bottom of the stack (i.e. + -- Scope.Last must be non-zero at the time of call). + + function Separate_Present return Boolean; + -- Determines if the current token is either Tok_Separate, or an + -- identifier that is a possible misspelling of "separate" followed + -- by a semicolon. True is returned if so, otherwise False. + + procedure Signal_Bad_Attribute; + -- The current token is an identifier that is supposed to be an + -- attribute identifier but is not. This routine posts appropriate + -- error messages, including a check for a near misspelling. + + function Token_Is_At_Start_Of_Line return Boolean; + pragma Inline (Token_Is_At_Start_Of_Line); + -- Determines if the current token is the first token on the line + + end Util; + + --------------------------------------- + -- Specialized Syntax Check Routines -- + --------------------------------------- + + function Prag (Pragma_Node : Node_Id; Semi : Source_Ptr) return Node_Id; + -- This function is passed a tree for a pragma that has been scanned out. + -- The pragma is syntactically well formed according to the general syntax + -- for pragmas and the pragma identifier is for one of the recognized + -- pragmas. It performs specific syntactic checks for specific pragmas. + -- The result is the input node if it is OK, or Error otherwise. The + -- reason that this is separated out is to facilitate the addition + -- of implementation defined pragmas. The second parameter records the + -- location of the semicolon following the pragma (this is needed for + -- correct processing of the List and Page pragmas). The returned value + -- is a copy of Pragma_Node, or Error if an error is found. + + ------------------------- + -- Subsidiary Routines -- + ------------------------- + + procedure Labl; + -- This procedure creates implicit label declarations for all label that + -- are declared in the current unit. Note that this could conceptually + -- be done at the point where the labels are declared, but it is tricky + -- to do it then, since the tree is not hooked up at the point where the + -- label is declared (e.g. a sequence of statements is not yet attached + -- to its containing scope at the point a label in the sequence is found) + + procedure Load; + -- This procedure loads all subsidiary units that are required by this + -- unit, including with'ed units, specs for bodies, and parents for child + -- units. It does not load bodies for inlined procedures and generics, + -- since we don't know till semantic analysis is complete what is needed. + + ----------- + -- Stubs -- + ----------- + + -- The package bodies can see all routines defined in all other subpackages + + use Ch2; + use Ch3; + use Ch4; + use Ch5; + use Ch6; + use Ch7; + use Ch8; + use Ch9; + use Ch10; + use Ch11; + use Ch12; + use Ch13; + + use Endh; + use Tchk; + use Sync; + use Util; + + package body Ch2 is separate; + package body Ch3 is separate; + package body Ch4 is separate; + package body Ch5 is separate; + package body Ch6 is separate; + package body Ch7 is separate; + package body Ch8 is separate; + package body Ch9 is separate; + package body Ch10 is separate; + package body Ch11 is separate; + package body Ch12 is separate; + package body Ch13 is separate; + + package body Endh is separate; + package body Tchk is separate; + package body Sync is separate; + package body Util is separate; + + function Prag (Pragma_Node : Node_Id; Semi : Source_Ptr) return Node_Id + is separate; + + procedure Labl is separate; + procedure Load is separate; + + --------- + -- Par -- + --------- + +-- This function is the parse routine called at the outer level. It parses +-- the current compilation unit and adds implicit label declarations. + +begin + -- Deal with configuration pragmas case first + + if Configuration_Pragmas then + declare + Ecount : constant Int := Errors_Detected; + Pragmas : List_Id := Empty_List; + P_Node : Node_Id; + + begin + loop + if Token = Tok_EOF then + return Pragmas; + + elsif Token /= Tok_Pragma then + Error_Msg_SC ("only pragmas allowed in configuration file"); + return Error_List; + + else + P_Node := P_Pragma; + + if Errors_Detected > Ecount then + return Error_List; + end if; + + if Chars (P_Node) > Last_Configuration_Pragma_Name + and then Chars (P_Node) /= Name_Source_Reference + then + Error_Msg_SC + ("only configuration pragmas allowed " & + "in configuration file"); + return Error_List; + end if; + + Append (P_Node, Pragmas); + end if; + end loop; + end; + + -- Normal case of compilation unit + + else + Save_Opt_Config_Switches (Save_Config_Switches); + + -- Special processing for language defined units. For this purpose + -- we do NOT consider the renamings in annex J as predefined. That + -- allows users to compile their own versions of these files, and + -- in particular, in the VMS implementation, the DEC versions can + -- be substituted for the standard Ada 95 versions. + + if Is_Predefined_File_Name + (Fname => File_Name (Current_Source_File), + Renamings_Included => False) + then + Set_Opt_Config_Switches + (Is_Internal_File_Name (File_Name (Current_Source_File))); + + -- If this is the main unit, disallow compilation unless the -gnatg + -- (GNAT mode) switch is set (from a user point of view, the rule is + -- that language defined units cannot be recompiled). + + -- However, an exception is s-rpc, and its children. We test this + -- by looking at the character after the minus, the rule is that + -- System.RPC and its children are the only children in System + -- whose second level name can start with the letter r. + + Get_Name_String (File_Name (Current_Source_File)); + + if (Name_Len < 3 or else Name_Buffer (1 .. 3) /= "s-r") + and then Current_Source_Unit = Main_Unit + and then not GNAT_Mode + and then Operating_Mode = Generate_Code + then + Error_Msg_SC ("language defined units may not be recompiled"); + end if; + end if; + + -- The following loop runs more than once only in syntax check mode + -- where we allow multiple compilation units in the same file. + + loop + Set_Opt_Config_Switches + (Is_Internal_File_Name (File_Name (Current_Source_File))); + + -- Initialize scope table and other parser control variables + + Compiler_State := Parsing; + Scope.Init; + Scope.Increment_Last; + Scope.Table (0).Etyp := E_Dummy; + SIS_Entry_Active := False; + Last_Resync_Point := No_Location; + + Label_List := New_Elmt_List; + Unit_Node := P_Compilation_Unit; + + -- If we are not at an end of file, then this means that we are + -- in syntax scan mode, and we can have another compilation unit, + -- otherwise we will exit from the loop. + + exit when Token = Tok_EOF; + Restore_Opt_Config_Switches (Save_Config_Switches); + Set_Comes_From_Source_Default (False); + end loop; + + -- Now that we have completely parsed the source file, we can + -- complete the source file table entry. + + Complete_Source_File_Entry; + + -- An internal error check, the scope stack should now be empty + + pragma Assert (Scope.Last = 0); + + -- Remaining steps are to create implicit label declarations and to + -- load required subsidiary sources. These steps are required only + -- if we are doing semantic checking. + + if Operating_Mode /= Check_Syntax or else Debug_Flag_F then + Par.Labl; + Par.Load; + end if; + + -- Restore settings of switches saved on entry + + Restore_Opt_Config_Switches (Save_Config_Switches); + Set_Comes_From_Source_Default (False); + return Empty_List; + end if; + +end Par; diff --git a/gcc/ada/par.ads b/gcc/ada/par.ads new file mode 100644 index 00000000000..c1110a0b135 --- /dev/null +++ b/gcc/ada/par.ads @@ -0,0 +1,44 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P A R -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.14 $ -- +-- -- +-- Copyright (C) 1992,1993,1994,1995 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). -- +-- -- +------------------------------------------------------------------------------ + +-- The Par function and its subunits contains all the parsing routines +-- for the top down recursive descent parser that constructs the parse tree + +with Types; use Types; + +function Par (Configuration_Pragmas : Boolean) return List_Id; +-- Top level parsing routine. There are two cases: +-- +-- If Configuration_Pragmas is False, Par parses a compilation unit in the +-- current source file and sets the Cunit, Cunit_Entity and Unit_Name fields +-- of the units table entry for Current_Source_Unit. On return the parse tree +-- is complete, and decorated with any required implicit label declarations. +-- The value returned in this case is always No_List. +-- +-- If Configuration_Pragmas is True, Par parses a list of configuration +-- pragmas from the current source file, and returns the list of pragmas. diff --git a/gcc/ada/prj-attr.adb b/gcc/ada/prj-attr.adb new file mode 100644 index 00000000000..aa793025f8a --- /dev/null +++ b/gcc/ada/prj-attr.adb @@ -0,0 +1,211 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . A T T R -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.4 $ +-- -- +-- Copyright (C) 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 Ada.Characters.Handling; use Ada.Characters.Handling; +with Namet; use Namet; +with Output; use Output; + +package body Prj.Attr is + + -- Names end with '#' + -- Package names are preceded by 'P' + -- Attribute names are preceded by two capital letters: + -- 'S' for Single or 'L' for list, then + -- 'V' for single variable, 'A' for associative array, or 'B' for both. + -- End is indicated by two consecutive '#'. + + Initialisation_Data : constant String := + + -- project attributes + + "SVobject_dir#" & + "LVsource_dirs#" & + "LVsource_files#" & + "SVsource_list_file#" & + "SVlibrary_dir#" & + "SVlibrary_name#" & + "SVlibrary_kind#" & + "SVlibrary_elaboration#" & + "SVlibrary_version#" & + "LVmain#" & + + -- package Naming + + "Pnaming#" & + "SVspecification_append#" & + "SVbody_append#" & + "SVseparate_append#" & + "SVcasing#" & + "SVdot_replacement#" & + "SAspecification#" & + "SAbody_part#" & + + -- package Compiler + + "Pcompiler#" & + "LBswitches#" & + "SVlocal_configuration_pragmas#" & + + -- package gnatmake + + "Pgnatmake#" & + "LBswitches#" & + "SVglobal_configuration_pragmas#" & + + -- package gnatls + + "Pgnatls#" & + "LVswitches#" & + + -- package gnatbind + + "Pgnatbind#" & + "LBswitches#" & + + -- package gnatlink + + "Pgnatlink#" & + "LBswitches#" & + + "#"; + + ---------------- + -- Initialize -- + ---------------- + + procedure Initialize is + Start : Positive := Initialisation_Data'First; + Finish : Positive := Start; + Current_Package : Package_Node_Id := Empty_Package; + Current_Attribute : Attribute_Node_Id := Empty_Attribute; + Is_An_Attribute : Boolean := False; + Kind_1 : Variable_Kind := Undefined; + Kind_2 : Attribute_Kind := Single; + Package_Name : Name_Id := No_Name; + Attribute_Name : Name_Id := No_Name; + First_Attribute : Attribute_Node_Id := Attribute_First; + begin + + -- Make sure the two tables are empty + + Attributes.Set_Last (Attributes.First); + Package_Attributes.Set_Last (Package_Attributes.First); + + while Initialisation_Data (Start) /= '#' loop + Is_An_Attribute := True; + case Initialisation_Data (Start) is + when 'P' => + -- New allowed package + Start := Start + 1; + Finish := Start; + while Initialisation_Data (Finish) /= '#' loop + Finish := Finish + 1; + end loop; + Name_Len := Finish - Start; + Name_Buffer (1 .. Name_Len) := + To_Lower (Initialisation_Data (Start .. Finish - 1)); + Package_Name := Name_Find; + for Index in Package_First .. Package_Attributes.Last loop + if Package_Name = Package_Attributes.Table (Index).Name then + Write_Line ("Duplicate package name """ & + Initialisation_Data (Start .. Finish - 1) & + """ in Prj.Attr body."); + raise Program_Error; + end if; + end loop; + + Is_An_Attribute := False; + Current_Attribute := Empty_Attribute; + Package_Attributes.Increment_Last; + Current_Package := Package_Attributes.Last; + Package_Attributes.Table (Current_Package).Name := + Package_Name; + Start := Finish + 1; + when 'S' => + Kind_1 := Single; + when 'L' => + Kind_1 := List; + when others => + raise Program_Error; + end case; + + if Is_An_Attribute then + -- New attribute + Start := Start + 1; + case Initialisation_Data (Start) is + when 'V' => + Kind_2 := Single; + when 'A' => + Kind_2 := Associative_Array; + when 'B' => + Kind_2 := Both; + when others => + raise Program_Error; + end case; + Start := Start + 1; + Finish := Start; + while Initialisation_Data (Finish) /= '#' loop + Finish := Finish + 1; + end loop; + Name_Len := Finish - Start; + Name_Buffer (1 .. Name_Len) := + To_Lower (Initialisation_Data (Start .. Finish - 1)); + Attribute_Name := Name_Find; + Attributes.Increment_Last; + if Current_Attribute = Empty_Attribute then + First_Attribute := Attributes.Last; + if Current_Package /= Empty_Package then + Package_Attributes.Table (Current_Package).First_Attribute + := Attributes.Last; + end if; + else + -- Check that there are no duplicate attributes + for Index in First_Attribute .. Attributes.Last - 1 loop + if Attribute_Name = + Attributes.Table (Index).Name then + Write_Line ("Duplicate attribute name """ & + Initialisation_Data (Start .. Finish - 1) & + """ in Prj.Attr body."); + raise Program_Error; + end if; + end loop; + Attributes.Table (Current_Attribute).Next := + Attributes.Last; + end if; + Current_Attribute := Attributes.Last; + Attributes.Table (Current_Attribute) := + (Name => Attribute_Name, + Kind_1 => Kind_1, + Kind_2 => Kind_2, + Next => Empty_Attribute); + Start := Finish + 1; + end if; + end loop; + end Initialize; + +end Prj.Attr; diff --git a/gcc/ada/prj-attr.ads b/gcc/ada/prj-attr.ads new file mode 100644 index 00000000000..ba4bb2e543b --- /dev/null +++ b/gcc/ada/prj-attr.ads @@ -0,0 +1,108 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . A T T R -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.1 $ +-- -- +-- Copyright (C) 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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- This package defines allowed packages and attributes in GNAT project +-- files. + +with Types; use Types; +with Table; + +package Prj.Attr is + + -- Define the allowed attributes + + Attributes_Initial : constant := 50; + Attributes_Increment : constant := 50; + + Attribute_Node_Low_Bound : constant := 0; + Attribute_Node_High_Bound : constant := 099_999_999; + + type Attribute_Node_Id is + range Attribute_Node_Low_Bound .. Attribute_Node_High_Bound; + + First_Attribute_Node_Id : constant Attribute_Node_Id + := Attribute_Node_Low_Bound; + + Empty_Attribute : constant Attribute_Node_Id + := Attribute_Node_Low_Bound; + + type Attribute_Kind is (Single, Associative_Array, Both); + + type Attribute_Record is record + Name : Name_Id; + Kind_1 : Variable_Kind; + Kind_2 : Attribute_Kind; + Next : Attribute_Node_Id; + end record; + + package Attributes is + new Table.Table (Table_Component_Type => Attribute_Record, + Table_Index_Type => Attribute_Node_Id, + Table_Low_Bound => First_Attribute_Node_Id, + Table_Initial => Attributes_Initial, + Table_Increment => Attributes_Increment, + Table_Name => "Prj.Attr.Attributes"); + + Attribute_First : constant Attribute_Node_Id := First_Attribute_Node_Id + 1; + + -- Define the allowed packages + + Packages_Initial : constant := 10; + Packages_Increment : constant := 10; + + Package_Node_Low_Bound : constant := 0; + Package_Node_High_Bound : constant := 099_999_999; + + type Package_Node_Id is + range Package_Node_Low_Bound .. Package_Node_High_Bound; + + First_Package_Node_Id : constant Package_Node_Id + := Package_Node_Low_Bound; + + Empty_Package : constant Package_Node_Id := Package_Node_Low_Bound; + + type Package_Record is record + Name : Name_Id; + First_Attribute : Attribute_Node_Id; + end record; + + package Package_Attributes is + new Table.Table (Table_Component_Type => Package_Record, + Table_Index_Type => Package_Node_Id, + Table_Low_Bound => First_Package_Node_Id, + Table_Initial => Packages_Initial, + Table_Increment => Packages_Increment, + Table_Name => "Prj.Attr.Packages"); + + Package_First : constant Package_Node_Id := Package_Node_Low_Bound + 1; + + procedure Initialize; + -- Initialize the two tables above (Attributes and Package_Attributes). + -- This procedure should be called by Prj.Initialize. + +end Prj.Attr; diff --git a/gcc/ada/prj-com.adb b/gcc/ada/prj-com.adb new file mode 100644 index 00000000000..3447e18f57c --- /dev/null +++ b/gcc/ada/prj-com.adb @@ -0,0 +1,49 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . C O M -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.1 $ +-- -- +-- Copyright (C) 2000 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 Namet; use Namet; +with Stringt; use Stringt; + +package body Prj.Com is + + ---------- + -- Hash -- + ---------- + + function Hash (Name : Name_Id) return Header_Num is + begin + return Hash (Get_Name_String (Name)); + end Hash; + + function Hash (Name : String_Id) return Header_Num is + begin + String_To_Name_Buffer (Name); + return Hash (Name_Buffer (1 .. Name_Len)); + end Hash; + +end Prj.Com; diff --git a/gcc/ada/prj-com.ads b/gcc/ada/prj-com.ads new file mode 100644 index 00000000000..ddb7d0f8ef7 --- /dev/null +++ b/gcc/ada/prj-com.ads @@ -0,0 +1,92 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . C O M -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.3 $ -- +-- -- +-- Copyright (C) 1992-2000 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). -- +-- -- +------------------------------------------------------------------------------ + +-- The following package declares data types for GNAT project. +-- These data types are used in the bodies of the Prj hierarchy. + +with GNAT.HTable; +with Table; +with Types; use Types; + +package Prj.Com is + + -- At one point, this package was private. + -- It cannot be private, because it is used outside of + -- the Prj hierarchy. + + Tool_Name : Name_Id := No_Name; + + Current_Verbosity : Verbosity := Default; + + type Spec_Or_Body is + (Specification, Body_Part); + + type File_Name_Data is record + Name : Name_Id := No_Name; + Path : Name_Id := No_Name; + Project : Project_Id := No_Project; + Needs_Pragma : Boolean := False; + end record; + -- File and Path name of a spec or body. + + type File_Names_Data is array (Spec_Or_Body) of File_Name_Data; + + type Unit_Id is new Nat; + No_Unit : constant Unit_Id := 0; + type Unit_Data is record + Name : Name_Id := No_Name; + File_Names : File_Names_Data; + end record; + -- File and Path names of a unit, with a reference to its + -- GNAT Project File. + + package Units is new Table.Table + (Table_Component_Type => Unit_Data, + Table_Index_Type => Unit_Id, + Table_Low_Bound => 1, + Table_Initial => 100, + Table_Increment => 100, + Table_Name => "Prj.Com.Units"); + + type Header_Num is range 0 .. 2047; + + function Hash is new GNAT.HTable.Hash (Header_Num => Header_Num); + + function Hash (Name : Name_Id) return Header_Num; + + function Hash (Name : String_Id) return Header_Num; + + package Units_Htable is new GNAT.HTable.Simple_HTable + (Header_Num => Header_Num, + Element => Unit_Id, + No_Element => No_Unit, + Key => Name_Id, + Hash => Hash, + Equal => "="); + +end Prj.Com; diff --git a/gcc/ada/prj-dect.adb b/gcc/ada/prj-dect.adb new file mode 100644 index 00000000000..65f7e43a4b6 --- /dev/null +++ b/gcc/ada/prj-dect.adb @@ -0,0 +1,942 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . D E C T -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.5 $ +-- -- +-- Copyright (C) 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 Errout; use Errout; +with Prj.Strt; +with Prj.Tree; use Prj.Tree; +with Scans; use Scans; +with Sinfo; use Sinfo; +with Types; use Types; +with Prj.Attr; use Prj.Attr; + +package body Prj.Dect is + + type Zone is (In_Project, In_Package, In_Case_Construction); + + procedure Parse_Attribute_Declaration + (Attribute : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Parse an attribute declaration. + + procedure Parse_Case_Construction + (Case_Construction : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Parse a case construction + + procedure Parse_Declarative_Items + (Declarations : out Project_Node_Id; + In_Zone : Zone; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Parse declarative items. Depending on In_Zone, some declarative + -- items may be forbiden. + + procedure Parse_Package_Declaration + (Package_Declaration : out Project_Node_Id; + Current_Project : Project_Node_Id); + -- Parse a package declaration + + procedure Parse_String_Type_Declaration + (String_Type : out Project_Node_Id; + Current_Project : Project_Node_Id; + First_Attribute : Attribute_Node_Id); + -- type <name> is ( <literal_string> { , <literal_string> } ) ; + + procedure Parse_Variable_Declaration + (Variable : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Parse a variable assignment + -- <variable_Name> := <expression>; OR + -- <variable_Name> : <string_type_Name> := <string_expression>; + + ----------- + -- Parse -- + ----------- + + procedure Parse + (Declarations : out Project_Node_Id; + Current_Project : Project_Node_Id; + Modifying : Project_Node_Id) + is + First_Declarative_Item : Project_Node_Id := Empty_Node; + + begin + Declarations := Default_Project_Node (Of_Kind => N_Project_Declaration); + Set_Location_Of (Declarations, To => Token_Ptr); + Set_Modified_Project_Of (Declarations, To => Modifying); + Parse_Declarative_Items + (Declarations => First_Declarative_Item, + In_Zone => In_Project, + First_Attribute => Prj.Attr.Attribute_First, + Current_Project => Current_Project, + Current_Package => Empty_Node); + Set_First_Declarative_Item_Of + (Declarations, To => First_Declarative_Item); + end Parse; + + --------------------------------- + -- Parse_Attribute_Declaration -- + --------------------------------- + + procedure Parse_Attribute_Declaration + (Attribute : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + Current_Attribute : Attribute_Node_Id := First_Attribute; + + begin + Attribute := Default_Project_Node (Of_Kind => N_Attribute_Declaration); + Set_Location_Of (Attribute, To => Token_Ptr); + + -- Scan past "for" + + Scan; + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + Set_Name_Of (Attribute, To => Token_Name); + Set_Location_Of (Attribute, To => Token_Ptr); + while Current_Attribute /= Empty_Attribute + and then + Attributes.Table (Current_Attribute).Name /= Token_Name + loop + Current_Attribute := Attributes.Table (Current_Attribute).Next; + end loop; + + if Current_Attribute = Empty_Attribute then + Error_Msg ("undefined attribute", Token_Ptr); + end if; + + Scan; + end if; + + if Token = Tok_Left_Paren then + if Current_Attribute /= Empty_Attribute + and then Attributes.Table (Current_Attribute).Kind_2 = Single + then + Error_Msg ("this attribute cannot be an associative array", + Location_Of (Attribute)); + end if; + + Scan; + Expect (Tok_String_Literal, "literal string"); + + if Token = Tok_String_Literal then + Set_Associative_Array_Index_Of (Attribute, Strval (Token_Node)); + Scan; + end if; + + Expect (Tok_Right_Paren, ")"); + + if Token = Tok_Right_Paren then + Scan; + end if; + + else + if Current_Attribute /= Empty_Attribute + and then + Attributes.Table (Current_Attribute).Kind_2 = Associative_Array + then + Error_Msg ("this attribute need to be an associative array", + Location_Of (Attribute)); + end if; + end if; + + if Current_Attribute /= Empty_Attribute then + Set_Expression_Kind_Of + (Attribute, To => Attributes.Table (Current_Attribute).Kind_1); + end if; + + Expect (Tok_Use, "use"); + + if Token = Tok_Use then + Scan; + + declare + Expression_Location : constant Source_Ptr := Token_Ptr; + Expression : Project_Node_Id := Empty_Node; + + begin + Prj.Strt.Parse_Expression + (Expression => Expression, + Current_Project => Current_Project, + Current_Package => Current_Package); + Set_Expression_Of (Attribute, To => Expression); + + if Current_Attribute /= Empty_Attribute + and then Expression /= Empty_Node + and then Attributes.Table (Current_Attribute).Kind_1 /= + Expression_Kind_Of (Expression) + then + Error_Msg + ("wrong expression kind for the attribute", + Expression_Location); + end if; + end; + end if; + + end Parse_Attribute_Declaration; + + ----------------------------- + -- Parse_Case_Construction -- + ----------------------------- + + procedure Parse_Case_Construction + (Case_Construction : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + Current_Item : Project_Node_Id := Empty_Node; + Next_Item : Project_Node_Id := Empty_Node; + First_Case_Item : Boolean := True; + + Variable_Location : Source_Ptr := No_Location; + + String_Type : Project_Node_Id := Empty_Node; + + Case_Variable : Project_Node_Id := Empty_Node; + + First_Declarative_Item : Project_Node_Id := Empty_Node; + + First_Choice : Project_Node_Id := Empty_Node; + + begin + Case_Construction := + Default_Project_Node (Of_Kind => N_Case_Construction); + Set_Location_Of (Case_Construction, To => Token_Ptr); + + -- Scan past "case" + + Scan; + + -- Get the switch variable + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + Variable_Location := Token_Ptr; + Prj.Strt.Parse_Variable_Reference + (Variable => Case_Variable, + Current_Project => Current_Project, + Current_Package => Current_Package); + Set_Case_Variable_Reference_Of + (Case_Construction, To => Case_Variable); + + else + if Token /= Tok_Is then + Scan; + end if; + end if; + + if Case_Variable /= Empty_Node then + String_Type := String_Type_Of (Case_Variable); + + if String_Type = Empty_Node then + Error_Msg ("this variable is not typed", Variable_Location); + end if; + end if; + + Expect (Tok_Is, "is"); + + if Token = Tok_Is then + + -- Scan past "is" + + Scan; + end if; + + Prj.Strt.Start_New_Case_Construction (String_Type); + + When_Loop : + + while Token = Tok_When loop + + if First_Case_Item then + Current_Item := Default_Project_Node (Of_Kind => N_Case_Item); + Set_First_Case_Item_Of (Case_Construction, To => Current_Item); + First_Case_Item := False; + + else + Next_Item := Default_Project_Node (Of_Kind => N_Case_Item); + Set_Next_Case_Item (Current_Item, To => Next_Item); + Current_Item := Next_Item; + end if; + + Set_Location_Of (Current_Item, To => Token_Ptr); + + -- Scan past "when" + + Scan; + + if Token = Tok_Others then + + -- Scan past "others" + + Scan; + + Expect (Tok_Arrow, "=>"); + + -- Empty_Node in Field1 of a Case_Item indicates + -- the "when others =>" branch. + + Set_First_Choice_Of (Current_Item, To => Empty_Node); + + Parse_Declarative_Items + (Declarations => First_Declarative_Item, + In_Zone => In_Case_Construction, + First_Attribute => First_Attribute, + Current_Project => Current_Project, + Current_Package => Current_Package); + + -- "when others =>" must be the last branch, so save the + -- Case_Item and exit + + Set_First_Declarative_Item_Of + (Current_Item, To => First_Declarative_Item); + exit When_Loop; + + else + Prj.Strt.Parse_Choice_List (First_Choice => First_Choice); + Set_First_Choice_Of (Current_Item, To => First_Choice); + + Expect (Tok_Arrow, "=>"); + + Parse_Declarative_Items + (Declarations => First_Declarative_Item, + In_Zone => In_Case_Construction, + First_Attribute => First_Attribute, + Current_Project => Current_Project, + Current_Package => Current_Package); + + Set_First_Declarative_Item_Of + (Current_Item, To => First_Declarative_Item); + + end if; + end loop When_Loop; + + Prj.Strt.End_Case_Construction; + + Expect (Tok_End, "end case"); + + if Token = Tok_End then + + -- Scan past "end" + + Scan; + + Expect (Tok_Case, "case"); + + end if; + + -- Scan past "case" + + Scan; + + Expect (Tok_Semicolon, ";"); + + end Parse_Case_Construction; + + ----------------------------- + -- Parse_Declarative_Items -- + ----------------------------- + + procedure Parse_Declarative_Items + (Declarations : out Project_Node_Id; + In_Zone : Zone; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + Current_Declarative_Item : Project_Node_Id := Empty_Node; + Next_Declarative_Item : Project_Node_Id := Empty_Node; + Current_Declaration : Project_Node_Id := Empty_Node; + Item_Location : Source_Ptr := No_Location; + + begin + Declarations := Empty_Node; + + loop + -- We are always positioned at the token that precedes + -- the first token of the declarative element. + -- Scan past it + + Scan; + + Item_Location := Token_Ptr; + + case Token is + when Tok_Identifier => + + if In_Zone = In_Case_Construction then + Error_Msg ("a variable cannot be declared here", + Token_Ptr); + end if; + + Parse_Variable_Declaration + (Current_Declaration, + First_Attribute => First_Attribute, + Current_Project => Current_Project, + Current_Package => Current_Package); + + when Tok_For => + + Parse_Attribute_Declaration + (Attribute => Current_Declaration, + First_Attribute => First_Attribute, + Current_Project => Current_Project, + Current_Package => Current_Package); + + when Tok_Package => + + -- Package declaration + + if In_Zone /= In_Project then + Error_Msg ("a package cannot be declared here", Token_Ptr); + end if; + + Parse_Package_Declaration + (Package_Declaration => Current_Declaration, + Current_Project => Current_Project); + + when Tok_Type => + + -- Type String Declaration + + if In_Zone /= In_Project then + Error_Msg ("a string type cannot be declared here", + Token_Ptr); + end if; + + Parse_String_Type_Declaration + (String_Type => Current_Declaration, + Current_Project => Current_Project, + First_Attribute => First_Attribute); + + when Tok_Case => + + -- Case construction + + Parse_Case_Construction + (Case_Construction => Current_Declaration, + First_Attribute => First_Attribute, + Current_Project => Current_Project, + Current_Package => Current_Package); + + when others => + exit; + + -- We are leaving Parse_Declarative_Items positionned + -- at the first token after the list of declarative items. + -- It could be "end" (for a project, a package declaration or + -- a case construction) or "when" (for a case construction) + + end case; + + Expect (Tok_Semicolon, "; after declarative items"); + + if Current_Declarative_Item = Empty_Node then + Current_Declarative_Item := + Default_Project_Node (Of_Kind => N_Declarative_Item); + Declarations := Current_Declarative_Item; + + else + Next_Declarative_Item := + Default_Project_Node (Of_Kind => N_Declarative_Item); + Set_Next_Declarative_Item + (Current_Declarative_Item, To => Next_Declarative_Item); + Current_Declarative_Item := Next_Declarative_Item; + end if; + + Set_Current_Item_Node + (Current_Declarative_Item, To => Current_Declaration); + Set_Location_Of (Current_Declarative_Item, To => Item_Location); + + end loop; + + end Parse_Declarative_Items; + + ------------------------------- + -- Parse_Package_Declaration -- + ------------------------------- + + procedure Parse_Package_Declaration + (Package_Declaration : out Project_Node_Id; + Current_Project : Project_Node_Id) + is + First_Attribute : Attribute_Node_Id := Empty_Attribute; + Current_Package : Package_Node_Id := Empty_Package; + First_Declarative_Item : Project_Node_Id := Empty_Node; + + begin + Package_Declaration := + Default_Project_Node (Of_Kind => N_Package_Declaration); + Set_Location_Of (Package_Declaration, To => Token_Ptr); + + -- Scan past "package" + + Scan; + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + + Set_Name_Of (Package_Declaration, To => Token_Name); + + for Index in Package_Attributes.First .. Package_Attributes.Last loop + if Token_Name = Package_Attributes.Table (Index).Name then + First_Attribute := + Package_Attributes.Table (Index).First_Attribute; + Current_Package := Index; + exit; + end if; + end loop; + + if Current_Package = Empty_Package then + Error_Msg ("not an allowed package name", Token_Ptr); + + else + Set_Package_Id_Of (Package_Declaration, To => Current_Package); + + declare + Current : Project_Node_Id := First_Package_Of (Current_Project); + + begin + while Current /= Empty_Node + and then Name_Of (Current) /= Token_Name + loop + Current := Next_Package_In_Project (Current); + end loop; + + if Current /= Empty_Node then + Error_Msg + ("package declared twice in the same project", Token_Ptr); + + else + -- Add the package to the project list + + Set_Next_Package_In_Project + (Package_Declaration, + To => First_Package_Of (Current_Project)); + Set_First_Package_Of + (Current_Project, To => Package_Declaration); + end if; + end; + end if; + + -- Scan past the package name + + Scan; + + end if; + + if Token = Tok_Renames then + -- Scan past "renames" + Scan; + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + declare + Project_Name : Name_Id := Token_Name; + Clause : Project_Node_Id := + First_With_Clause_Of (Current_Project); + The_Project : Project_Node_Id := Empty_Node; + + begin + while Clause /= Empty_Node loop + The_Project := Project_Node_Of (Clause); + exit when Name_Of (The_Project) = Project_Name; + Clause := Next_With_Clause_Of (Clause); + end loop; + + if Clause = Empty_Node then + Error_Msg ("not an imported project", Token_Ptr); + else + Set_Project_Of_Renamed_Package_Of + (Package_Declaration, To => The_Project); + end if; + end; + + Scan; + Expect (Tok_Dot, "."); + + if Token = Tok_Dot then + Scan; + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + if Name_Of (Package_Declaration) /= Token_Name then + Error_Msg ("not the same package name", Token_Ptr); + elsif + Project_Of_Renamed_Package_Of (Package_Declaration) + /= Empty_Node + then + declare + Current : Project_Node_Id := + First_Package_Of + (Project_Of_Renamed_Package_Of + (Package_Declaration)); + + begin + while Current /= Empty_Node + and then Name_Of (Current) /= Token_Name + loop + Current := Next_Package_In_Project (Current); + end loop; + + if Current = Empty_Node then + Error_Msg + ("not a package declared by the project", + Token_Ptr); + end if; + end; + end if; + + Scan; + end if; + end if; + end if; + + Expect (Tok_Semicolon, ";"); + + elsif Token = Tok_Is then + + Parse_Declarative_Items + (Declarations => First_Declarative_Item, + In_Zone => In_Package, + First_Attribute => First_Attribute, + Current_Project => Current_Project, + Current_Package => Package_Declaration); + + Set_First_Declarative_Item_Of + (Package_Declaration, To => First_Declarative_Item); + + Expect (Tok_End, "end"); + + if Token = Tok_End then + + -- Scan past "end" + + Scan; + end if; + + -- We should have the name of the package after "end" + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier + and then Name_Of (Package_Declaration) /= No_Name + and then Token_Name /= Name_Of (Package_Declaration) + then + Error_Msg_Name_1 := Name_Of (Package_Declaration); + Error_Msg ("expected {", Token_Ptr); + end if; + + if Token /= Tok_Semicolon then + + -- Scan past the package name + + Scan; + end if; + + Expect (Tok_Semicolon, ";"); + + else + Error_Msg ("expected ""is"" or ""renames""", Token_Ptr); + end if; + + end Parse_Package_Declaration; + + ----------------------------------- + -- Parse_String_Type_Declaration -- + ----------------------------------- + + procedure Parse_String_Type_Declaration + (String_Type : out Project_Node_Id; + Current_Project : Project_Node_Id; + First_Attribute : Attribute_Node_Id) + is + Current : Project_Node_Id := Empty_Node; + First_String : Project_Node_Id := Empty_Node; + + begin + String_Type := + Default_Project_Node (Of_Kind => N_String_Type_Declaration); + + Set_Location_Of (String_Type, To => Token_Ptr); + + -- Scan past "type" + + Scan; + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + Set_Name_Of (String_Type, To => Token_Name); + + Current := First_String_Type_Of (Current_Project); + while Current /= Empty_Node + and then + Name_Of (Current) /= Token_Name + loop + Current := Next_String_Type (Current); + end loop; + + if Current /= Empty_Node then + Error_Msg ("duplicate string type name", Token_Ptr); + else + Current := First_Variable_Of (Current_Project); + while Current /= Empty_Node + and then Name_Of (Current) /= Token_Name + loop + Current := Next_Variable (Current); + end loop; + + if Current /= Empty_Node then + Error_Msg ("already a variable name", Token_Ptr); + else + Set_Next_String_Type + (String_Type, To => First_String_Type_Of (Current_Project)); + Set_First_String_Type_Of (Current_Project, To => String_Type); + end if; + end if; + + -- Scan past the name + + Scan; + end if; + + Expect (Tok_Is, "is"); + + if Token = Tok_Is then + Scan; + end if; + + Expect (Tok_Left_Paren, "("); + + if Token = Tok_Left_Paren then + Scan; + end if; + + Prj.Strt.Parse_String_Type_List (First_String => First_String); + Set_First_Literal_String (String_Type, To => First_String); + + Expect (Tok_Right_Paren, ")"); + + if Token = Tok_Right_Paren then + Scan; + end if; + + end Parse_String_Type_Declaration; + + -------------------------------- + -- Parse_Variable_Declaration -- + -------------------------------- + + procedure Parse_Variable_Declaration + (Variable : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + Expression_Location : Source_Ptr; + String_Type_Name : Name_Id := No_Name; + Project_String_Type_Name : Name_Id := No_Name; + Type_Location : Source_Ptr := No_Location; + Project_Location : Source_Ptr := No_Location; + Expression : Project_Node_Id := Empty_Node; + Variable_Name : constant Name_Id := Token_Name; + + begin + Variable := + Default_Project_Node (Of_Kind => N_Variable_Declaration); + Set_Name_Of (Variable, To => Variable_Name); + Set_Location_Of (Variable, To => Token_Ptr); + + -- Scan past the variable name + + Scan; + + if Token = Tok_Colon then + + -- Typed string variable declaration + + Scan; + Set_Kind_Of (Variable, N_Typed_Variable_Declaration); + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + String_Type_Name := Token_Name; + Type_Location := Token_Ptr; + Scan; + + if Token = Tok_Dot then + Project_String_Type_Name := String_Type_Name; + Project_Location := Type_Location; + + -- Scan past the dot + + Scan; + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + String_Type_Name := Token_Name; + Type_Location := Token_Ptr; + Scan; + else + String_Type_Name := No_Name; + end if; + end if; + + if String_Type_Name /= No_Name then + declare + Current : Project_Node_Id := + First_String_Type_Of (Current_Project); + + begin + if Project_String_Type_Name /= No_Name then + declare + The_Project_Name_And_Node : constant + Tree_Private_Part.Project_Name_And_Node := + Tree_Private_Part.Projects_Htable.Get + (Project_String_Type_Name); + + use Tree_Private_Part; + + begin + if The_Project_Name_And_Node = + Tree_Private_Part.No_Project_Name_And_Node + then + Error_Msg ("unknown project", Project_Location); + Current := Empty_Node; + else + Current := + First_String_Type_Of + (The_Project_Name_And_Node.Node); + end if; + end; + end if; + + while Current /= Empty_Node + and then Name_Of (Current) /= String_Type_Name + loop + Current := Next_String_Type (Current); + end loop; + + if Current = Empty_Node then + Error_Msg ("unknown string type", Type_Location); + else + Set_String_Type_Of + (Variable, To => Current); + end if; + end; + end if; + end if; + end if; + + Expect (Tok_Colon_Equal, ":="); + + if Token = Tok_Colon_Equal then + Scan; + end if; + + -- Get the single string or string list value + + Expression_Location := Token_Ptr; + + Prj.Strt.Parse_Expression + (Expression => Expression, + Current_Project => Current_Project, + Current_Package => Current_Package); + Set_Expression_Of (Variable, To => Expression); + + if Expression /= Empty_Node then + Set_Expression_Kind_Of + (Variable, To => Expression_Kind_Of (Expression)); + end if; + + declare + The_Variable : Project_Node_Id := Empty_Node; + + begin + if Current_Package /= Empty_Node then + The_Variable := First_Variable_Of (Current_Package); + elsif Current_Project /= Empty_Node then + The_Variable := First_Variable_Of (Current_Project); + end if; + + while The_Variable /= Empty_Node + and then Name_Of (The_Variable) /= Variable_Name + loop + The_Variable := Next_Variable (The_Variable); + end loop; + + if The_Variable = Empty_Node then + if Current_Package /= Empty_Node then + Set_Next_Variable + (Variable, To => First_Variable_Of (Current_Package)); + Set_First_Variable_Of (Current_Package, To => Variable); + + elsif Current_Project /= Empty_Node then + Set_Next_Variable + (Variable, To => First_Variable_Of (Current_Project)); + Set_First_Variable_Of (Current_Project, To => Variable); + end if; + + else + if Expression_Kind_Of (Variable) /= Undefined then + if Expression_Kind_Of (The_Variable) = Undefined then + Set_Expression_Kind_Of + (The_Variable, To => Expression_Kind_Of (Variable)); + + else + if Expression_Kind_Of (The_Variable) /= + Expression_Kind_Of (Variable) + then + Error_Msg ("wrong expression kind for the variable", + Expression_Location); + end if; + end if; + end if; + end if; + end; + + end Parse_Variable_Declaration; + +end Prj.Dect; diff --git a/gcc/ada/prj-dect.ads b/gcc/ada/prj-dect.ads new file mode 100644 index 00000000000..3072c573b62 --- /dev/null +++ b/gcc/ada/prj-dect.ads @@ -0,0 +1,41 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . D E C T -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.2 $ +-- -- +-- Copyright (C) 2000 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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- Parse a list of declarative items in a project file. + +with Prj.Tree; + +private package Prj.Dect is + + procedure Parse + (Declarations : out Prj.Tree.Project_Node_Id; + Current_Project : Prj.Tree.Project_Node_Id; + Modifying : Prj.Tree.Project_Node_Id); + -- Parse project declarative items. + +end Prj.Dect; diff --git a/gcc/ada/prj-env.adb b/gcc/ada/prj-env.adb new file mode 100644 index 00000000000..171a2d03c1a --- /dev/null +++ b/gcc/ada/prj-env.adb @@ -0,0 +1,1471 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . E N V -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.17 $ +-- -- +-- Copyright (C) 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 GNAT.OS_Lib; use GNAT.OS_Lib; +with Namet; use Namet; +with Opt; +with Osint; use Osint; +with Output; use Output; +with Prj.Com; use Prj.Com; +with Prj.Util; +with Snames; use Snames; +with Stringt; use Stringt; +with Table; + +package body Prj.Env is + + type Naming_Id is new Nat; + No_Naming : constant Naming_Id := 0; + + Ada_Path_Buffer : String_Access := new String (1 .. 1_000); + -- A buffer where values for ADA_INCLUDE_PATH + -- and ADA_OBJECTS_PATH are stored. + + Ada_Path_Length : Natural := 0; + -- Index of the last valid character in Ada_Path_Buffer. + + package Namings is new Table.Table ( + Table_Component_Type => Naming_Data, + Table_Index_Type => Naming_Id, + Table_Low_Bound => 1, + Table_Initial => 5, + Table_Increment => 100, + Table_Name => "Prj.Env.Namings"); + + Default_Naming : constant Naming_Id := Namings.First; + + Global_Configuration_Pragmas : Name_Id; + Local_Configuration_Pragmas : Name_Id; + + ----------------------- + -- Local Subprograms -- + ----------------------- + + function Body_Path_Name_Of (Unit : Unit_Id) return String; + -- Returns the path name of the body of a unit. + -- Compute it first, if necessary. + + function Spec_Path_Name_Of (Unit : Unit_Id) return String; + -- Returns the path name of the spec of a unit. + -- Compute it first, if necessary. + + procedure Add_To_Path (Path : String); + -- Add Path to global variable Ada_Path_Buffer + -- Increment Ada_Path_Length + + ---------------------- + -- Ada_Include_Path -- + ---------------------- + + function Ada_Include_Path (Project : Project_Id) return String_Access is + + procedure Add (Project : Project_Id); + -- Add all the source directories of a project to the path, + -- only if this project has not been visited. + -- Call itself recursively for projects being modified, + -- and imported projects. + -- Add the project to the list Seen if this is the first time + -- we call Add for this project. + + --------- + -- Add -- + --------- + + procedure Add (Project : Project_Id) is + begin + -- If Seen is empty, then the project cannot have been + -- visited. + + if not Projects.Table (Project).Seen then + Projects.Table (Project).Seen := True; + + declare + Data : Project_Data := Projects.Table (Project); + List : Project_List := Data.Imported_Projects; + + Current : String_List_Id := Data.Source_Dirs; + Source_Dir : String_Element; + + begin + -- Add to path all source directories of this project + + while Current /= Nil_String loop + if Ada_Path_Length > 0 then + Add_To_Path (Path => (1 => Path_Separator)); + end if; + + Source_Dir := String_Elements.Table (Current); + String_To_Name_Buffer (Source_Dir.Value); + + declare + New_Path : constant String := + Name_Buffer (1 .. Name_Len); + begin + Add_To_Path (New_Path); + end; + + Current := Source_Dir.Next; + end loop; + + -- Call Add to the project being modified, if any + + if Data.Modifies /= No_Project then + Add (Data.Modifies); + end if; + + -- Call Add for each imported project, if any + + while List /= Empty_Project_List loop + Add (Project_Lists.Table (List).Project); + List := Project_Lists.Table (List).Next; + end loop; + end; + end if; + + end Add; + + -- Start of processing for Ada_Include_Path + + begin + -- If it is the first time we call this function for + -- this project, compute the source path + + if Projects.Table (Project).Include_Path = null then + Ada_Path_Length := 0; + + for Index in 1 .. Projects.Last loop + Projects.Table (Index).Seen := False; + end loop; + + Add (Project); + Projects.Table (Project).Include_Path := + new String'(Ada_Path_Buffer (1 .. Ada_Path_Length)); + end if; + + return Projects.Table (Project).Include_Path; + end Ada_Include_Path; + + ---------------------- + -- Ada_Objects_Path -- + ---------------------- + + function Ada_Objects_Path + (Project : Project_Id; + Including_Libraries : Boolean := True) + return String_Access is + + procedure Add (Project : Project_Id); + -- Add all the object directory of a project to the path, + -- only if this project has not been visited. + -- Call itself recursively for projects being modified, + -- and imported projects. + -- Add the project to the list Seen if this is the first time + -- we call Add for this project. + + --------- + -- Add -- + --------- + + procedure Add (Project : Project_Id) is + begin + + -- If this project has not been seen yet + + if not Projects.Table (Project).Seen then + Projects.Table (Project).Seen := True; + + declare + Data : Project_Data := Projects.Table (Project); + List : Project_List := Data.Imported_Projects; + + begin + -- Add to path the object directory of this project + -- except if we don't include library project and + -- this is a library project. + + if (Data.Library and then Including_Libraries) + or else + (Data.Object_Directory /= No_Name + and then + (not Including_Libraries or else not Data.Library)) + then + if Ada_Path_Length > 0 then + Add_To_Path (Path => (1 => Path_Separator)); + end if; + + -- For a library project, att the library directory + + if Data.Library then + declare + New_Path : constant String := + Get_Name_String (Data.Library_Dir); + begin + Add_To_Path (New_Path); + end; + else + + -- For a non library project, add the object directory + declare + New_Path : constant String := + Get_Name_String (Data.Object_Directory); + begin + Add_To_Path (New_Path); + end; + end if; + end if; + + -- Call Add to the project being modified, if any + + if Data.Modifies /= No_Project then + Add (Data.Modifies); + end if; + + -- Call Add for each imported project, if any + + while List /= Empty_Project_List loop + Add (Project_Lists.Table (List).Project); + List := Project_Lists.Table (List).Next; + end loop; + end; + + end if; + end Add; + + -- Start of processing for Ada_Objects_Path + + begin + -- If it is the first time we call this function for + -- this project, compute the objects path + + if Projects.Table (Project).Objects_Path = null then + Ada_Path_Length := 0; + + for Index in 1 .. Projects.Last loop + Projects.Table (Index).Seen := False; + end loop; + + Add (Project); + Projects.Table (Project).Objects_Path := + new String'(Ada_Path_Buffer (1 .. Ada_Path_Length)); + end if; + + return Projects.Table (Project).Objects_Path; + end Ada_Objects_Path; + + ----------------- + -- Add_To_Path -- + ----------------- + + procedure Add_To_Path (Path : String) is + begin + -- If Ada_Path_Buffer is too small, double it + + if Ada_Path_Length + Path'Length > Ada_Path_Buffer'Last then + declare + New_Ada_Path_Buffer : constant String_Access := + new String + (1 .. Ada_Path_Buffer'Last + + Ada_Path_Buffer'Last); + + begin + New_Ada_Path_Buffer (1 .. Ada_Path_Length) := + Ada_Path_Buffer (1 .. Ada_Path_Length); + Ada_Path_Buffer := New_Ada_Path_Buffer; + end; + end if; + + Ada_Path_Buffer + (Ada_Path_Length + 1 .. Ada_Path_Length + Path'Length) := Path; + Ada_Path_Length := Ada_Path_Length + Path'Length; + end Add_To_Path; + + ----------------------- + -- Body_Path_Name_Of -- + ----------------------- + + function Body_Path_Name_Of (Unit : Unit_Id) return String is + Data : Unit_Data := Units.Table (Unit); + + begin + -- If we don't know the path name of the body of this unit, + -- we compute it, and we store it. + + if Data.File_Names (Body_Part).Path = No_Name then + declare + Current_Source : String_List_Id := + Projects.Table (Data.File_Names (Body_Part).Project).Sources; + Path : GNAT.OS_Lib.String_Access; + + begin + -- By default, put the file name + + Data.File_Names (Body_Part).Path := + Data.File_Names (Body_Part).Name; + + -- For each source directory + + while Current_Source /= Nil_String loop + String_To_Name_Buffer + (String_Elements.Table (Current_Source).Value); + Path := + Locate_Regular_File + (Namet.Get_Name_String + (Data.File_Names (Body_Part).Name), + Name_Buffer (1 .. Name_Len)); + + -- If the file is in this directory, + -- then we store the path, and we are done. + + if Path /= null then + Name_Len := Path'Length; + Name_Buffer (1 .. Name_Len) := Path.all; + Data.File_Names (Body_Part).Path := Name_Enter; + exit; + + else + Current_Source := + String_Elements.Table (Current_Source).Next; + end if; + end loop; + + Units.Table (Unit) := Data; + end; + end if; + + -- Returned the value stored + + return Namet.Get_Name_String (Data.File_Names (Body_Part).Path); + end Body_Path_Name_Of; + + -------------------------------- + -- Create_Config_Pragmas_File -- + -------------------------------- + + procedure Create_Config_Pragmas_File + (For_Project : Project_Id; + Main_Project : Project_Id) + is + File_Name : Temp_File_Name; + File : File_Descriptor := Invalid_FD; + + The_Packages : Package_Id; + Gnatmake : Prj.Package_Id; + Compiler : Prj.Package_Id; + + Current_Unit : Unit_Id := Units.First; + + First_Project : Project_List := Empty_Project_List; + + Current_Project : Project_List; + Current_Naming : Naming_Id; + + Global_Attribute : Variable_Value := Nil_Variable_Value; + Local_Attribute : Variable_Value := Nil_Variable_Value; + + Global_Attribute_Present : Boolean := False; + Local_Attribute_Present : Boolean := False; + + procedure Check (Project : Project_Id); + + procedure Check_Temp_File; + -- Check that a temporary file has been opened. + -- If not, create one, and put its name in the project data, + -- with the indication that it is a temporary file. + + procedure Copy_File (Name : String_Id); + -- Copy a configuration pragmas file into the temp file. + + procedure Put + (Unit_Name : Name_Id; + File_Name : Name_Id; + Unit_Kind : Spec_Or_Body); + -- Put an SFN pragma in the temporary file. + + procedure Put (File : File_Descriptor; S : String); + + procedure Put_Line (File : File_Descriptor; S : String); + + ----------- + -- Check -- + ----------- + + procedure Check (Project : Project_Id) is + Data : constant Project_Data := Projects.Table (Project); + + begin + if Current_Verbosity = High then + Write_Str ("Checking project file """); + Write_Str (Namet.Get_Name_String (Data.Name)); + Write_Str ("""."); + Write_Eol; + end if; + + -- Is this project in the list of the visited project? + + Current_Project := First_Project; + while Current_Project /= Empty_Project_List + and then Project_Lists.Table (Current_Project).Project /= Project + loop + Current_Project := Project_Lists.Table (Current_Project).Next; + end loop; + + -- If it is not, put it in the list, and visit it + + if Current_Project = Empty_Project_List then + Project_Lists.Increment_Last; + Project_Lists.Table (Project_Lists.Last) := + (Project => Project, Next => First_Project); + First_Project := Project_Lists.Last; + + -- Is the naming scheme of this project one that we know? + + Current_Naming := Default_Naming; + while Current_Naming <= Namings.Last and then + not Same_Naming_Scheme + (Left => Namings.Table (Current_Naming), + Right => Data.Naming) loop + Current_Naming := Current_Naming + 1; + end loop; + + -- If we don't know it, add it + + if Current_Naming > Namings.Last then + Namings.Increment_Last; + Namings.Table (Namings.Last) := Data.Naming; + + -- We need a temporary file to be created + + Check_Temp_File; + + -- Put the SFN pragmas for the naming scheme + + -- Spec + + Put_Line + (File, "pragma Source_File_Name"); + Put_Line + (File, " (Spec_File_Name => ""*" & + Namet.Get_Name_String (Data.Naming.Specification_Append) & + ""","); + Put_Line + (File, " Casing => " & + Image (Data.Naming.Casing) & ","); + Put_Line + (File, " Dot_Replacement => """ & + Namet.Get_Name_String (Data.Naming.Dot_Replacement) & + """);"); + + -- and body + + Put_Line + (File, "pragma Source_File_Name"); + Put_Line + (File, " (Body_File_Name => ""*" & + Namet.Get_Name_String (Data.Naming.Body_Append) & + ""","); + Put_Line + (File, " Casing => " & + Image (Data.Naming.Casing) & ","); + Put_Line + (File, " Dot_Replacement => """ & + Namet.Get_Name_String (Data.Naming.Dot_Replacement) & + """);"); + + -- and maybe separate + + if Data.Naming.Body_Append /= Data.Naming.Separate_Append then + Put_Line + (File, "pragma Source_File_Name"); + Put_Line + (File, " (Subunit_File_Name => ""*" & + Namet.Get_Name_String (Data.Naming.Separate_Append) & + ""","); + Put_Line + (File, " Casing => " & + Image (Data.Naming.Casing) & + ","); + Put_Line + (File, " Dot_Replacement => """ & + Namet.Get_Name_String (Data.Naming.Dot_Replacement) & + """);"); + end if; + end if; + + if Data.Modifies /= No_Project then + Check (Data.Modifies); + end if; + + declare + Current : Project_List := Data.Imported_Projects; + + begin + while Current /= Empty_Project_List loop + Check (Project_Lists.Table (Current).Project); + Current := Project_Lists.Table (Current).Next; + end loop; + end; + end if; + end Check; + + --------------------- + -- Check_Temp_File -- + --------------------- + + procedure Check_Temp_File is + begin + if File = Invalid_FD then + GNAT.OS_Lib.Create_Temp_File (File, Name => File_Name); + if File = Invalid_FD then + Osint.Fail + ("unable to create temporary configuration pragmas file"); + elsif Opt.Verbose_Mode then + Write_Str ("Creating temp file """); + Write_Str (File_Name); + Write_Line (""""); + end if; + end if; + end Check_Temp_File; + + --------------- + -- Copy_File -- + --------------- + + procedure Copy_File (Name : in String_Id) is + Input : File_Descriptor; + Buffer : String (1 .. 1_000); + Input_Length : Integer; + Output_Length : Integer; + + begin + Check_Temp_File; + String_To_Name_Buffer (Name); + + if Opt.Verbose_Mode then + Write_Str ("Copying config pragmas file """); + Write_Str (Name_Buffer (1 .. Name_Len)); + Write_Line (""" into temp file"); + end if; + + declare + Name : constant String := + Name_Buffer (1 .. Name_Len) & ASCII.NUL; + begin + Input := Open_Read (Name'Address, Binary); + end; + + if Input = Invalid_FD then + Osint.Fail + ("cannot open configuration pragmas file " & + Name_Buffer (1 .. Name_Len)); + end if; + + loop + Input_Length := Read (Input, Buffer'Address, Buffer'Length); + Output_Length := Write (File, Buffer'Address, Input_Length); + + if Output_Length /= Input_Length then + Osint.Fail ("disk full"); + end if; + + exit when Input_Length < Buffer'Length; + end loop; + + Close (Input); + + end Copy_File; + + --------- + -- Put -- + --------- + + procedure Put + (Unit_Name : Name_Id; + File_Name : Name_Id; + Unit_Kind : Spec_Or_Body) + is + begin + -- A temporary file needs to be open + + Check_Temp_File; + + -- Put the pragma SFN for the unit kind (spec or body) + + Put (File, "pragma Source_File_Name ("); + Put (File, Namet.Get_Name_String (Unit_Name)); + + if Unit_Kind = Specification then + Put (File, ", Spec_File_Name => """); + else + Put (File, ", Body_File_Name => """); + end if; + + Put (File, Namet.Get_Name_String (File_Name)); + Put_Line (File, """);"); + end Put; + + procedure Put (File : File_Descriptor; S : String) is + Last : Natural; + + begin + Last := Write (File, S (S'First)'Address, S'Length); + + if Last /= S'Length then + Osint.Fail ("Disk full"); + end if; + + if Current_Verbosity = High then + Write_Str (S); + end if; + end Put; + + -------------- + -- Put_Line -- + -------------- + + procedure Put_Line (File : File_Descriptor; S : String) is + S0 : String (1 .. S'Length + 1); + Last : Natural; + + begin + -- Add an ASCII.LF to the string. As this gnat.adc + -- is supposed to be used only by the compiler, we don't + -- care about the characters for the end of line. + -- The truth is we could have put a space, but it is + -- more convenient to be able to read gnat.adc during + -- development. And the development was done under UNIX. + -- Hence the ASCII.LF. + + S0 (1 .. S'Length) := S; + S0 (S0'Last) := ASCII.LF; + Last := Write (File, S0'Address, S0'Length); + + if Last /= S'Length + 1 then + Osint.Fail ("Disk full"); + end if; + + if Current_Verbosity = High then + Write_Line (S); + end if; + end Put_Line; + + -- Start of processing for Create_Config_Pragmas_File + + begin + + if not Projects.Table (For_Project).Config_Checked then + + -- Remove any memory of processed naming schemes, if any + + Namings.Set_Last (Default_Naming); + + -- Check the naming schemes + + Check (For_Project); + + -- Visit all the units and process those that need an SFN pragma + + while Current_Unit <= Units.Last loop + declare + Unit : constant Unit_Data := + Units.Table (Current_Unit); + + begin + if Unit.File_Names (Specification).Needs_Pragma then + Put (Unit.Name, + Unit.File_Names (Specification).Name, + Specification); + end if; + + if Unit.File_Names (Body_Part).Needs_Pragma then + Put (Unit.Name, + Unit.File_Names (Body_Part).Name, + Body_Part); + end if; + + Current_Unit := Current_Unit + 1; + end; + end loop; + + The_Packages := Projects.Table (Main_Project).Decl.Packages; + Gnatmake := + Prj.Util.Value_Of + (Name => Name_Gnatmake, + In_Packages => The_Packages); + + if Gnatmake /= No_Package then + Global_Attribute := Prj.Util.Value_Of + (Variable_Name => Global_Configuration_Pragmas, + In_Variables => Packages.Table (Gnatmake).Decl.Attributes); + Global_Attribute_Present := + Global_Attribute /= Nil_Variable_Value + and then String_Length (Global_Attribute.Value) > 0; + end if; + + The_Packages := Projects.Table (For_Project).Decl.Packages; + Compiler := + Prj.Util.Value_Of + (Name => Name_Compiler, + In_Packages => The_Packages); + + if Compiler /= No_Package then + Local_Attribute := Prj.Util.Value_Of + (Variable_Name => Local_Configuration_Pragmas, + In_Variables => Packages.Table (Compiler).Decl.Attributes); + Local_Attribute_Present := + Local_Attribute /= Nil_Variable_Value + and then String_Length (Local_Attribute.Value) > 0; + end if; + + if Global_Attribute_Present then + + if File /= Invalid_FD + or else Local_Attribute_Present + then + Copy_File (Global_Attribute.Value); + else + String_To_Name_Buffer (Global_Attribute.Value); + Projects.Table (For_Project).Config_File_Name := Name_Find; + end if; + end if; + + if Local_Attribute_Present then + + if File /= Invalid_FD then + Copy_File (Local_Attribute.Value); + + else + String_To_Name_Buffer (Local_Attribute.Value); + Projects.Table (For_Project).Config_File_Name := Name_Find; + end if; + + end if; + + if File /= Invalid_FD then + GNAT.OS_Lib.Close (File); + + if Opt.Verbose_Mode then + Write_Str ("Closing configuration file """); + Write_Str (File_Name); + Write_Line (""""); + end if; + + Name_Len := File_Name'Length; + Name_Buffer (1 .. Name_Len) := File_Name; + Projects.Table (For_Project).Config_File_Name := Name_Find; + Projects.Table (For_Project).Config_File_Temp := True; + end if; + + Projects.Table (For_Project).Config_Checked := True; + + end if; + + end Create_Config_Pragmas_File; + + ------------------------------------ + -- File_Name_Of_Library_Unit_Body -- + ------------------------------------ + + function File_Name_Of_Library_Unit_Body + (Name : String; + Project : Project_Id) + return String + is + Data : constant Project_Data := Projects.Table (Project); + Original_Name : String := Name; + + Extended_Spec_Name : String := + Name & Namet.Get_Name_String + (Data.Naming.Specification_Append); + Extended_Body_Name : String := + Name & Namet.Get_Name_String + (Data.Naming.Body_Append); + + Unit : Unit_Data; + + The_Original_Name : Name_Id; + The_Spec_Name : Name_Id; + The_Body_Name : Name_Id; + + begin + Canonical_Case_File_Name (Original_Name); + Name_Len := Original_Name'Length; + Name_Buffer (1 .. Name_Len) := Original_Name; + The_Original_Name := Name_Find; + + Canonical_Case_File_Name (Extended_Spec_Name); + Name_Len := Extended_Spec_Name'Length; + Name_Buffer (1 .. Name_Len) := Extended_Spec_Name; + The_Spec_Name := Name_Find; + + Canonical_Case_File_Name (Extended_Body_Name); + Name_Len := Extended_Body_Name'Length; + Name_Buffer (1 .. Name_Len) := Extended_Body_Name; + The_Body_Name := Name_Find; + + if Current_Verbosity = High then + Write_Str ("Looking for file name of """); + Write_Str (Name); + Write_Char ('"'); + Write_Eol; + Write_Str (" Extended Spec Name = """); + Write_Str (Extended_Spec_Name); + Write_Char ('"'); + Write_Eol; + Write_Str (" Extended Body Name = """); + Write_Str (Extended_Body_Name); + Write_Char ('"'); + Write_Eol; + end if; + + -- For every unit + + for Current in reverse Units.First .. Units.Last loop + Unit := Units.Table (Current); + + -- If it is a unit of the same project + + if Unit.File_Names (Body_Part).Project = Project then + declare + Current_Name : constant Name_Id := + Unit.File_Names (Body_Part).Name; + + begin + -- If there is a body + + if Current_Name /= No_Name then + if Current_Verbosity = High then + Write_Str (" Comparing with """); + Write_Str (Get_Name_String (Current_Name)); + Write_Char ('"'); + Write_Eol; + end if; + + -- If it has the name of the original name, + -- return the original name + + if Unit.Name = The_Original_Name + or else Current_Name = The_Original_Name + then + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Get_Name_String (Current_Name); + + -- If it has the name of the extended body name, + -- return the extended body name + + elsif Current_Name = The_Body_Name then + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Extended_Body_Name; + + else + if Current_Verbosity = High then + Write_Line (" not good"); + end if; + end if; + end if; + end; + end if; + + -- If it is a unit of the same project + + if Units.Table (Current).File_Names (Specification).Project = + Project + then + declare + Current_Name : constant Name_Id := + Unit.File_Names (Specification).Name; + + begin + -- If there is a spec + + if Current_Name /= No_Name then + if Current_Verbosity = High then + Write_Str (" Comparing with """); + Write_Str (Get_Name_String (Current_Name)); + Write_Char ('"'); + Write_Eol; + end if; + + -- If it has the same name as the original name, + -- return the original name + + if Unit.Name = The_Original_Name + or else Current_Name = The_Original_Name + then + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Get_Name_String (Current_Name); + + -- If it has the same name as the extended spec name, + -- return the extended spec name + + elsif Current_Name = The_Spec_Name then + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Extended_Spec_Name; + + else + if Current_Verbosity = High then + Write_Line (" not good"); + end if; + end if; + end if; + end; + end if; + + end loop; + + -- We don't know this file name, return an empty string + + return ""; + end File_Name_Of_Library_Unit_Body; + + ------------------------- + -- For_All_Object_Dirs -- + ------------------------- + + procedure For_All_Object_Dirs (Project : Project_Id) is + Seen : Project_List := Empty_Project_List; + + procedure Add (Project : Project_Id); + -- Process a project. Remember the processes visited to avoid + -- processing a project twice. Recursively process an eventual + -- modified project, and all imported projects. + + --------- + -- Add -- + --------- + + procedure Add (Project : Project_Id) is + Data : constant Project_Data := Projects.Table (Project); + List : Project_List := Data.Imported_Projects; + + begin + -- If the list of visited project is empty, then + -- for sure we never visited this project. + + if Seen = Empty_Project_List then + Project_Lists.Increment_Last; + Seen := Project_Lists.Last; + Project_Lists.Table (Seen) := + (Project => Project, Next => Empty_Project_List); + + else + -- Check if the project is in the list + + declare + Current : Project_List := Seen; + + begin + loop + -- If it is, then there is nothing else to do + + if Project_Lists.Table (Current).Project = Project then + return; + end if; + + exit when Project_Lists.Table (Current).Next = + Empty_Project_List; + Current := Project_Lists.Table (Current).Next; + end loop; + + -- This project has never been visited, add it + -- to the list. + + Project_Lists.Increment_Last; + Project_Lists.Table (Current).Next := Project_Lists.Last; + Project_Lists.Table (Project_Lists.Last) := + (Project => Project, Next => Empty_Project_List); + end; + end if; + + -- If there is an object directory, call Action + -- with its name + + if Data.Object_Directory /= No_Name then + Get_Name_String (Data.Object_Directory); + Action (Name_Buffer (1 .. Name_Len)); + end if; + + -- If we are modifying a project, visit it + + if Data.Modifies /= No_Project then + Add (Data.Modifies); + end if; + + -- And visit all imported projects + + while List /= Empty_Project_List loop + Add (Project_Lists.Table (List).Project); + List := Project_Lists.Table (List).Next; + end loop; + end Add; + + -- Start of processing for For_All_Object_Dirs + + begin + -- Visit this project, and its imported projects, + -- recursively + + Add (Project); + end For_All_Object_Dirs; + + ------------------------- + -- For_All_Source_Dirs -- + ------------------------- + + procedure For_All_Source_Dirs (Project : Project_Id) is + Seen : Project_List := Empty_Project_List; + + procedure Add (Project : Project_Id); + -- Process a project. Remember the processes visited to avoid + -- processing a project twice. Recursively process an eventual + -- modified project, and all imported projects. + + --------- + -- Add -- + --------- + + procedure Add (Project : Project_Id) is + Data : constant Project_Data := Projects.Table (Project); + List : Project_List := Data.Imported_Projects; + + begin + -- If the list of visited project is empty, then + -- for sure we never visited this project. + + if Seen = Empty_Project_List then + Project_Lists.Increment_Last; + Seen := Project_Lists.Last; + Project_Lists.Table (Seen) := + (Project => Project, Next => Empty_Project_List); + + else + -- Check if the project is in the list + + declare + Current : Project_List := Seen; + + begin + loop + -- If it is, then there is nothing else to do + + if Project_Lists.Table (Current).Project = Project then + return; + end if; + + exit when Project_Lists.Table (Current).Next = + Empty_Project_List; + Current := Project_Lists.Table (Current).Next; + end loop; + + -- This project has never been visited, add it + -- to the list. + + Project_Lists.Increment_Last; + Project_Lists.Table (Current).Next := Project_Lists.Last; + Project_Lists.Table (Project_Lists.Last) := + (Project => Project, Next => Empty_Project_List); + end; + end if; + + declare + Current : String_List_Id := Data.Source_Dirs; + The_String : String_Element; + + begin + -- Call action with the name of every source directorie + + while Current /= Nil_String loop + The_String := String_Elements.Table (Current); + String_To_Name_Buffer (The_String.Value); + Action (Name_Buffer (1 .. Name_Len)); + Current := The_String.Next; + end loop; + end; + + -- If we are modifying a project, visit it + + if Data.Modifies /= No_Project then + Add (Data.Modifies); + end if; + + -- And visit all imported projects + + while List /= Empty_Project_List loop + Add (Project_Lists.Table (List).Project); + List := Project_Lists.Table (List).Next; + end loop; + end Add; + + -- Start of processing for For_All_Source_Dirs + + begin + -- Visit this project, and its imported projects recursively + + Add (Project); + end For_All_Source_Dirs; + + ------------------- + -- Get_Reference -- + ------------------- + + procedure Get_Reference + (Source_File_Name : String; + Project : out Project_Id; + Path : out Name_Id) + is + begin + if Current_Verbosity > Default then + Write_Str ("Getting Reference_Of ("""); + Write_Str (Source_File_Name); + Write_Str (""") ... "); + end if; + + declare + Original_Name : String := Source_File_Name; + Unit : Unit_Data; + + begin + Canonical_Case_File_Name (Original_Name); + + for Id in Units.First .. Units.Last loop + Unit := Units.Table (Id); + + if (Unit.File_Names (Specification).Name /= No_Name + and then + Namet.Get_Name_String + (Unit.File_Names (Specification).Name) = Original_Name) + or else (Unit.File_Names (Specification).Path /= No_Name + and then + Namet.Get_Name_String + (Unit.File_Names (Specification).Path) = + Original_Name) + then + Project := Unit.File_Names (Specification).Project; + Path := Unit.File_Names (Specification).Path; + + if Current_Verbosity > Default then + Write_Str ("Done: Specification."); + Write_Eol; + end if; + + return; + + elsif (Unit.File_Names (Body_Part).Name /= No_Name + and then + Namet.Get_Name_String + (Unit.File_Names (Body_Part).Name) = Original_Name) + or else (Unit.File_Names (Body_Part).Path /= No_Name + and then Namet.Get_Name_String + (Unit.File_Names (Body_Part).Path) = + Original_Name) + then + Project := Unit.File_Names (Body_Part).Project; + Path := Unit.File_Names (Body_Part).Path; + + if Current_Verbosity > Default then + Write_Str ("Done: Body."); + Write_Eol; + end if; + + return; + end if; + + end loop; + end; + + Project := No_Project; + Path := No_Name; + + if Current_Verbosity > Default then + Write_Str ("Cannot be found."); + Write_Eol; + end if; + end Get_Reference; + + ---------------- + -- Initialize -- + ---------------- + + procedure Initialize is + Global : constant String := "global_configuration_pragmas"; + Local : constant String := "local_configuration_pragmas"; + begin + -- Put the standard GNAT naming scheme in the Namings table + + Namings.Increment_Last; + Namings.Table (Namings.Last) := Standard_Naming_Data; + Name_Len := Global'Length; + Name_Buffer (1 .. Name_Len) := Global; + Global_Configuration_Pragmas := Name_Find; + Name_Len := Local'Length; + Name_Buffer (1 .. Name_Len) := Local; + Local_Configuration_Pragmas := Name_Find; + end Initialize; + + ------------------------------------ + -- Path_Name_Of_Library_Unit_Body -- + ------------------------------------ + + function Path_Name_Of_Library_Unit_Body + (Name : String; + Project : Project_Id) + return String + is + Data : constant Project_Data := Projects.Table (Project); + Original_Name : String := Name; + + Extended_Spec_Name : String := + Name & Namet.Get_Name_String + (Data.Naming.Specification_Append); + Extended_Body_Name : String := + Name & Namet.Get_Name_String + (Data.Naming.Body_Append); + + First : Unit_Id := Units.First; + Current : Unit_Id; + Unit : Unit_Data; + + begin + Canonical_Case_File_Name (Original_Name); + Canonical_Case_File_Name (Extended_Spec_Name); + Canonical_Case_File_Name (Extended_Spec_Name); + + if Current_Verbosity = High then + Write_Str ("Looking for path name of """); + Write_Str (Name); + Write_Char ('"'); + Write_Eol; + Write_Str (" Extended Spec Name = """); + Write_Str (Extended_Spec_Name); + Write_Char ('"'); + Write_Eol; + Write_Str (" Extended Body Name = """); + Write_Str (Extended_Body_Name); + Write_Char ('"'); + Write_Eol; + end if; + + while First <= Units.Last + and then Units.Table (First).File_Names (Body_Part).Project /= Project + loop + First := First + 1; + end loop; + + Current := First; + while Current <= Units.Last loop + Unit := Units.Table (Current); + + if Unit.File_Names (Body_Part).Project = Project + and then Unit.File_Names (Body_Part).Name /= No_Name + then + declare + Current_Name : constant String := + Namet.Get_Name_String (Unit.File_Names (Body_Part).Name); + begin + if Current_Verbosity = High then + Write_Str (" Comparing with """); + Write_Str (Current_Name); + Write_Char ('"'); + Write_Eol; + end if; + + if Current_Name = Original_Name then + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Body_Path_Name_Of (Current); + + elsif Current_Name = Extended_Body_Name then + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Body_Path_Name_Of (Current); + + else + if Current_Verbosity = High then + Write_Line (" not good"); + end if; + end if; + end; + + elsif Unit.File_Names (Specification).Name /= No_Name then + declare + Current_Name : constant String := + Namet.Get_Name_String + (Unit.File_Names (Specification).Name); + + begin + if Current_Verbosity = High then + Write_Str (" Comparing with """); + Write_Str (Current_Name); + Write_Char ('"'); + Write_Eol; + end if; + + if Current_Name = Original_Name then + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Spec_Path_Name_Of (Current); + + elsif Current_Name = Extended_Spec_Name then + + if Current_Verbosity = High then + Write_Line (" OK"); + end if; + + return Spec_Path_Name_Of (Current); + + else + if Current_Verbosity = High then + Write_Line (" not good"); + end if; + end if; + end; + end if; + Current := Current + 1; + end loop; + + return ""; + end Path_Name_Of_Library_Unit_Body; + + ------------------- + -- Print_Sources -- + ------------------- + + procedure Print_Sources is + Unit : Unit_Data; + + begin + Write_Line ("List of Sources:"); + + for Id in Units.First .. Units.Last loop + Unit := Units.Table (Id); + Write_Str (" "); + Write_Line (Namet.Get_Name_String (Unit.Name)); + + if Unit.File_Names (Specification).Name /= No_Name then + if Unit.File_Names (Specification).Project = No_Project then + Write_Line (" No project"); + + else + Write_Str (" Project: "); + Get_Name_String + (Projects.Table + (Unit.File_Names (Specification).Project).Path_Name); + Write_Line (Name_Buffer (1 .. Name_Len)); + end if; + + Write_Str (" spec: "); + Write_Line + (Namet.Get_Name_String + (Unit.File_Names (Specification).Name)); + end if; + + if Unit.File_Names (Body_Part).Name /= No_Name then + if Unit.File_Names (Body_Part).Project = No_Project then + Write_Line (" No project"); + + else + Write_Str (" Project: "); + Get_Name_String + (Projects.Table + (Unit.File_Names (Body_Part).Project).Path_Name); + Write_Line (Name_Buffer (1 .. Name_Len)); + end if; + + Write_Str (" body: "); + Write_Line + (Namet.Get_Name_String + (Unit.File_Names (Body_Part).Name)); + end if; + + end loop; + + Write_Line ("end of List of Sources."); + end Print_Sources; + + ----------------------- + -- Spec_Path_Name_Of -- + ----------------------- + + function Spec_Path_Name_Of (Unit : Unit_Id) return String is + Data : Unit_Data := Units.Table (Unit); + + begin + if Data.File_Names (Specification).Path = No_Name then + declare + Current_Source : String_List_Id := + Projects.Table (Data.File_Names (Specification).Project).Sources; + Path : GNAT.OS_Lib.String_Access; + + begin + Data.File_Names (Specification).Path := + Data.File_Names (Specification).Name; + + while Current_Source /= Nil_String loop + String_To_Name_Buffer + (String_Elements.Table (Current_Source).Value); + Path := Locate_Regular_File + (Namet.Get_Name_String + (Data.File_Names (Specification).Name), + Name_Buffer (1 .. Name_Len)); + + if Path /= null then + Name_Len := Path'Length; + Name_Buffer (1 .. Name_Len) := Path.all; + Data.File_Names (Specification).Path := Name_Enter; + exit; + else + Current_Source := + String_Elements.Table (Current_Source).Next; + end if; + end loop; + + Units.Table (Unit) := Data; + end; + end if; + + return Namet.Get_Name_String (Data.File_Names (Specification).Path); + end Spec_Path_Name_Of; + +end Prj.Env; diff --git a/gcc/ada/prj-env.ads b/gcc/ada/prj-env.ads new file mode 100644 index 00000000000..272c559282a --- /dev/null +++ b/gcc/ada/prj-env.ads @@ -0,0 +1,99 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . E N V -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.10 $ +-- -- +-- Copyright (C) 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). -- +-- -- +------------------------------------------------------------------------------ + +-- This package implements services for Project-aware tools, related +-- to the environment (gnat.adc, ADA_INCLUDE_PATH, ADA_OBJECTS_PATH) + +with GNAT.OS_Lib; use GNAT.OS_Lib; + +package Prj.Env is + + procedure Initialize; + -- Put Standard_Naming_Data into Namings table (called by Prj.Initialize) + + procedure Print_Sources; + -- Output the list of sources, after Project files have been scanned + + procedure Create_Config_Pragmas_File + (For_Project : Project_Id; + Main_Project : Project_Id); + -- If there needs to have SFN pragmas, either for non standard naming + -- schemes or for individual units, or if Global_Configuration_Pragmas + -- has been specified in package gnatmake of the main project, or if + -- Local_Configuration_Pragmas has been specified in package Compiler + -- of the main project, build (if needed) a temporary file that contains + -- all configuration pragmas, and specify the configuration pragmas file + -- in the project data. + + function Ada_Include_Path (Project : Project_Id) return String_Access; + -- Get the ADA_INCLUDE_PATH of a Project file. For the first call, compute + -- it and cache it. + + function Ada_Objects_Path + (Project : Project_Id; + Including_Libraries : Boolean := True) + return String_Access; + -- Get the ADA_OBJECTS_PATH of a Project file. For the first call, compute + -- it and cache it. When Including_Libraries is False, do not include the + -- object directories of the library projects, and do not cache the result. + + function Path_Name_Of_Library_Unit_Body + (Name : String; + Project : Project_Id) + return String; + -- Returns the Path of a library unit. + + function File_Name_Of_Library_Unit_Body + (Name : String; + Project : Project_Id) + return String; + -- Returns the file name of a library unit, in canonical case. Name may or + -- may not have an extension (corresponding to the naming scheme of the + -- project). If there is no body with this name, but there is a spec, the + -- name of the spec is returned. If neither a body or a spec can be found, + -- return an empty string. + + procedure Get_Reference + (Source_File_Name : String; + Project : out Project_Id; + Path : out Name_Id); + -- Returns the project of a source. + + generic + with procedure Action (Path : String); + procedure For_All_Source_Dirs (Project : Project_Id); + -- Iterate through all the source directories of a project, + -- including those of imported or modified projects. + + generic + with procedure Action (Path : String); + procedure For_All_Object_Dirs (Project : Project_Id); + -- Iterate through all the object directories of a project, + -- including those of imported or modified projects. + +end Prj.Env; diff --git a/gcc/ada/prj-ext.adb b/gcc/ada/prj-ext.adb new file mode 100644 index 00000000000..b6f6ab8bb14 --- /dev/null +++ b/gcc/ada/prj-ext.adb @@ -0,0 +1,130 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . E X T -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.5 $ +-- -- +-- Copyright (C) 2000 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 GNAT.HTable; +with GNAT.OS_Lib; use GNAT.OS_Lib; +with Namet; use Namet; +with Prj.Com; use Prj.Com; +with Stringt; use Stringt; +with Types; use Types; + +package body Prj.Ext is + + package Htable is new GNAT.HTable.Simple_HTable + (Header_Num => Header_Num, + Element => String_Id, + No_Element => No_String, + Key => Name_Id, + Hash => Hash, + Equal => "="); + + --------- + -- Add -- + --------- + + procedure Add + (External_Name : String; + Value : String) + is + The_Key : Name_Id; + The_Value : String_Id; + + begin + Start_String; + Store_String_Chars (Value); + The_Value := End_String; + Name_Len := External_Name'Length; + Name_Buffer (1 .. Name_Len) := External_Name; + 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; + + -------------- + -- Value_Of -- + -------------- + + function Value_Of + (External_Name : Name_Id; + With_Default : String_Id := No_String) + return String_Id + is + The_Value : String_Id; + + begin + The_Value := Htable.Get (External_Name); + + if The_Value /= No_String then + return The_Value; + end if; + + -- Find if it is an environment. + -- If it is, put the value in the hash table. + + declare + Env_Value : constant String_Access := + Getenv (Get_Name_String (External_Name)); + + begin + if Env_Value /= null and then Env_Value'Length > 0 then + Start_String; + Store_String_Chars (Env_Value.all); + The_Value := End_String; + Htable.Set (External_Name, The_Value); + return The_Value; + + else + return With_Default; + end if; + end; + end Value_Of; + +end Prj.Ext; diff --git a/gcc/ada/prj-ext.ads b/gcc/ada/prj-ext.ads new file mode 100644 index 00000000000..4c12b786bcf --- /dev/null +++ b/gcc/ada/prj-ext.ads @@ -0,0 +1,51 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . E X T -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.3 $ +-- -- +-- Copyright (C) 2000 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). -- +-- -- +------------------------------------------------------------------------------ + +-- Set, Get and cache External reference, to be used as External functions +-- in project files. + +with Types; use Types; + +package Prj.Ext is + + procedure Add + (External_Name : String; + Value : String); + -- Add an external reference (or modify an existing one). + + function Value_Of + (External_Name : Name_Id; + With_Default : String_Id := No_String) + return String_Id; + -- Get the value of an external reference, and cache it for future uses. + + function Check (Declaration : String) return Boolean; + -- Check that an external declaration <external>=<value> is correct. + -- If it is correct, the external reference is Added. + +end Prj.Ext; diff --git a/gcc/ada/prj-nmsc.adb b/gcc/ada/prj-nmsc.adb new file mode 100644 index 00000000000..66031878d2b --- /dev/null +++ b/gcc/ada/prj-nmsc.adb @@ -0,0 +1,2236 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . N M S C -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.25 $ +-- -- +-- Copyright (C) 2000-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 Ada.Characters.Handling; use Ada.Characters.Handling; +with Ada.Strings; use Ada.Strings; +with Ada.Strings.Fixed; use Ada.Strings.Fixed; +with Ada.Strings.Maps.Constants; use Ada.Strings.Maps.Constants; +with Errout; use Errout; +with GNAT.Directory_Operations; use GNAT.Directory_Operations; +with GNAT.OS_Lib; use GNAT.OS_Lib; +with Namet; use Namet; +with Osint; use Osint; +with Output; use Output; +with Prj.Com; use Prj.Com; +with Prj.Util; use Prj.Util; +with Snames; use Snames; +with Stringt; use Stringt; +with Types; use Types; + +package body Prj.Nmsc is + + Dir_Sep : Character renames GNAT.OS_Lib.Directory_Separator; + + Error_Report : Put_Line_Access := null; + + procedure Check_Naming_Scheme (Naming : Naming_Data); + -- Check that the package Naming is correct. + + procedure Check_Naming_Scheme + (Name : Name_Id; + Unit : out Name_Id); + -- Check that a name is a valid unit name. + + procedure Error_Msg (Msg : String; Flag_Location : Source_Ptr); + -- Output an error message. + -- If Error_Report is null, simply call Errout.Error_Msg. + -- Otherwise, disregard Flag_Location and use Error_Report. + + function Get_Name_String (S : String_Id) return String; + -- Get the string from a String_Id + + procedure Get_Unit + (File_Name : Name_Id; + Naming : Naming_Data; + Unit_Name : out Name_Id; + Unit_Kind : out Spec_Or_Body; + Needs_Pragma : out Boolean); + -- Find out, from a file name, the unit name, the unit kind + -- and if a specific SFN pragma is needed. + -- If the file name corresponds to no unit, then Unit_Name + -- will be No_Name. + + function Is_Illegal_Append (This : String) return Boolean; + -- Returns True if the string This cannot be used as + -- a Specification_Append, a Body_Append or a Separate_Append. + + procedure Record_Source + (File_Name : Name_Id; + Path_Name : Name_Id; + Project : Project_Id; + Data : in out Project_Data; + Error_If_Invalid : Boolean; + Location : Source_Ptr; + Current_Source : in out String_List_Id); + -- Put a unit in the list of units of a project, if the file name + -- corresponds to a valid unit name. + -- If it does not correspond to a valid unit name, report an error + -- only if Error_If_Invalid is true. + + procedure Show_Source_Dirs (Project : Project_Id); + -- List all the source directories of a project. + + function Locate_Directory + (Name : Name_Id; + Parent : Name_Id) + return Name_Id; + -- Locate a directory. + -- Returns No_Name if directory does not exist. + + function Path_Name_Of + (File_Name : String_Id; + Directory : Name_Id) + return String; + -- Returns the path name of a (non project) file. + -- Returns an empty string if file cannot be found. + + function Path_Name_Of + (File_Name : String_Id; + Directory : String_Id) + return String; + -- Same as above except that Directory is a String_Id instead + -- of a Name_Id. + + ------------------------- + -- Check_Naming_Scheme -- + ------------------------- + + procedure Check_Naming_Scheme (Naming : Naming_Data) is + begin + -- Only check if we are not using the standard naming scheme + + if Naming /= Standard_Naming_Data then + declare + Dot_Replacement : constant String := + Get_Name_String + (Naming.Dot_Replacement); + Specification_Append : constant String := + Get_Name_String + (Naming.Specification_Append); + Body_Append : constant String := + Get_Name_String + (Naming.Body_Append); + Separate_Append : constant String := + Get_Name_String + (Naming.Separate_Append); + + begin + -- Dot_Replacement cannot + -- - be empty + -- - start or end with an alphanumeric + -- - be a single '_' + -- - start with an '_' followed by an alphanumeric + -- - contain a '.' except if it is "." + + if Dot_Replacement'Length = 0 + or else Is_Alphanumeric + (Dot_Replacement (Dot_Replacement'First)) + or else Is_Alphanumeric + (Dot_Replacement (Dot_Replacement'Last)) + or else (Dot_Replacement (Dot_Replacement'First) = '_' + and then + (Dot_Replacement'Length = 1 + or else + Is_Alphanumeric + (Dot_Replacement (Dot_Replacement'First + 1)))) + or else (Dot_Replacement'Length > 1 + and then + Index (Source => Dot_Replacement, + Pattern => ".") /= 0) + then + Error_Msg + ('"' & Dot_Replacement & + """ is illegal for Dot_Replacement.", + Naming.Dot_Repl_Loc); + end if; + + -- Appends cannot + -- - be empty + -- - start with an alphanumeric + -- - start with an '_' followed by an alphanumeric + + if Is_Illegal_Append (Specification_Append) then + Error_Msg + ('"' & Specification_Append & + """ is illegal for Specification_Append.", + Naming.Spec_Append_Loc); + end if; + + if Is_Illegal_Append (Body_Append) then + Error_Msg + ('"' & Body_Append & + """ is illegal for Body_Append.", + Naming.Body_Append_Loc); + end if; + + if Body_Append /= Separate_Append then + if Is_Illegal_Append (Separate_Append) then + Error_Msg + ('"' & Separate_Append & + """ is illegal for Separate_Append.", + Naming.Sep_Append_Loc); + end if; + end if; + + -- Specification_Append cannot have the same termination as + -- Body_Append or Separate_Append + + if Specification_Append'Length >= Body_Append'Length + and then + Body_Append (Body_Append'Last - + Specification_Append'Length + 1 .. + Body_Append'Last) = Specification_Append + then + Error_Msg + ("Body_Append (""" & + Body_Append & + """) cannot end with" & + " Specification_Append (""" & + Specification_Append & """).", + Naming.Body_Append_Loc); + end if; + + if Specification_Append'Length >= Separate_Append'Length + and then + Separate_Append + (Separate_Append'Last - Specification_Append'Length + 1 + .. + Separate_Append'Last) = Specification_Append + then + Error_Msg + ("Separate_Append (""" & + Separate_Append & + """) cannot end with" & + " Specification_Append (""" & + Specification_Append & """).", + Naming.Sep_Append_Loc); + end if; + end; + end if; + end Check_Naming_Scheme; + + procedure Check_Naming_Scheme + (Name : Name_Id; + Unit : out Name_Id) + is + The_Name : String := Get_Name_String (Name); + Need_Letter : Boolean := True; + Last_Underscore : Boolean := False; + OK : Boolean := The_Name'Length > 0; + + begin + for Index in The_Name'Range loop + if Need_Letter then + + -- We need a letter (at the beginning, and following a dot), + -- but we don't have one. + + if Is_Letter (The_Name (Index)) then + Need_Letter := False; + + else + OK := False; + + if Current_Verbosity = High then + Write_Int (Types.Int (Index)); + Write_Str (": '"); + Write_Char (The_Name (Index)); + Write_Line ("' is not a letter."); + end if; + + exit; + end if; + + elsif Last_Underscore + and then (The_Name (Index) = '_' or else The_Name (Index) = '.') + then + -- Two underscores are illegal, and a dot cannot follow + -- an underscore. + + OK := False; + + if Current_Verbosity = High then + Write_Int (Types.Int (Index)); + Write_Str (": '"); + Write_Char (The_Name (Index)); + Write_Line ("' is illegal here."); + end if; + + exit; + + elsif The_Name (Index) = '.' then + + -- We need a letter after a dot + + Need_Letter := True; + + elsif The_Name (Index) = '_' then + Last_Underscore := True; + + else + -- We need an letter or a digit + + Last_Underscore := False; + + if not Is_Alphanumeric (The_Name (Index)) then + OK := False; + + if Current_Verbosity = High then + Write_Int (Types.Int (Index)); + Write_Str (": '"); + Write_Char (The_Name (Index)); + Write_Line ("' is not alphanumeric."); + end if; + + exit; + end if; + end if; + end loop; + + -- We cannot end with an underscore or a dot + + OK := OK and then not Need_Letter and then not Last_Underscore; + + if OK then + Unit := Name; + else + -- We signal a problem with No_Name + + Unit := No_Name; + end if; + end Check_Naming_Scheme; + + procedure Check_Naming_Scheme + (Project : Project_Id; + Report_Error : Put_Line_Access) + is + Last_Source_Dir : String_List_Id := Nil_String; + Data : Project_Data := Projects.Table (Project); + + procedure Check_Unit_Names (List : Array_Element_Id); + -- Check that a list of unit names contains only valid names. + + procedure Find_Source_Dirs (From : String_Id; Location : Source_Ptr); + -- Find one or several source directories, and add them + -- to the list of source directories of the project. + + procedure Find_Sources; + -- Find all the sources in all of the source directories + -- of a project. + + procedure Get_Path_Name_And_Record_Source + (File_Name : String; + Location : Source_Ptr; + Current_Source : in out String_List_Id); + -- Find the path name of a source in the source directories and + -- record the source, if found. + + procedure Get_Sources_From_File + (Path : String; + Location : Source_Ptr); + -- Get the sources of a project from a text file + + ---------------------- + -- Check_Unit_Names -- + ---------------------- + + procedure Check_Unit_Names (List : Array_Element_Id) is + Current : Array_Element_Id := List; + Element : Array_Element; + Unit_Name : Name_Id; + + begin + -- Loop through elements of the string list + + while Current /= No_Array_Element loop + Element := Array_Elements.Table (Current); + + -- Check that it contains a valid unit name + + Check_Naming_Scheme (Element.Index, Unit_Name); + + if Unit_Name = No_Name then + Error_Msg_Name_1 := Element.Index; + Error_Msg + ("{ is not a valid unit name.", + Element.Value.Location); + + else + + if Current_Verbosity = High then + Write_Str (" Body_Part ("""); + Write_Str (Get_Name_String (Unit_Name)); + Write_Line (""")"); + end if; + + Element.Index := Unit_Name; + Array_Elements.Table (Current) := Element; + end if; + + Current := Element.Next; + end loop; + end Check_Unit_Names; + + ---------------------- + -- Find_Source_Dirs -- + ---------------------- + + procedure Find_Source_Dirs (From : String_Id; Location : Source_Ptr) is + + Directory : String (1 .. Integer (String_Length (From))); + Directory_Id : Name_Id; + Element : String_Element; + + procedure Recursive_Find_Dirs (Path : String_Id); + -- Find all the subdirectories (recursively) of Path + -- and add them to the list of source directories + -- of the project. + + ------------------------- + -- Recursive_Find_Dirs -- + ------------------------- + + procedure Recursive_Find_Dirs (Path : String_Id) is + Dir : Dir_Type; + Name : String (1 .. 250); + Last : Natural; + The_Path : String := Get_Name_String (Path) & Dir_Sep; + + The_Path_Last : Positive := The_Path'Last; + + begin + if The_Path'Length > 1 + and then + (The_Path (The_Path_Last - 1) = Dir_Sep + or else The_Path (The_Path_Last - 1) = '/') + then + The_Path_Last := The_Path_Last - 1; + end if; + + if Current_Verbosity = High then + Write_Str (" "); + Write_Line (The_Path (The_Path'First .. The_Path_Last)); + end if; + + String_Elements.Increment_Last; + Element := + (Value => Path, + Location => No_Location, + Next => Nil_String); + + -- Case of first source directory + + if Last_Source_Dir = Nil_String then + Data.Source_Dirs := String_Elements.Last; + + -- Here we already have source directories. + + else + -- Link the previous last to the new one + + String_Elements.Table (Last_Source_Dir).Next := + String_Elements.Last; + end if; + + -- And register this source directory as the new last + + Last_Source_Dir := String_Elements.Last; + String_Elements.Table (Last_Source_Dir) := Element; + + -- Now look for subdirectories + + Open (Dir, The_Path (The_Path'First .. The_Path_Last)); + + loop + Read (Dir, Name, Last); + exit when Last = 0; + + if Current_Verbosity = High then + Write_Str (" Checking "); + Write_Line (Name (1 .. Last)); + end if; + + if Name (1 .. Last) /= "." + and then Name (1 .. Last) /= ".." + then + -- Avoid . and .. + + declare + Path_Name : constant String := + The_Path (The_Path'First .. The_Path_Last) & + Name (1 .. Last); + + begin + if Is_Directory (Path_Name) then + + -- We have found a new subdirectory, + -- register it and find its own subdirectories. + + Start_String; + Store_String_Chars (Path_Name); + Recursive_Find_Dirs (End_String); + end if; + end; + end if; + end loop; + + Close (Dir); + + exception + when Directory_Error => + null; + end Recursive_Find_Dirs; + + -- Start of processing for Find_Source_Dirs + + begin + if Current_Verbosity = High then + Write_Str ("Find_Source_Dirs ("""); + end if; + + String_To_Name_Buffer (From); + Directory := Name_Buffer (1 .. Name_Len); + Directory_Id := Name_Find; + + if Current_Verbosity = High then + Write_Str (Directory); + Write_Line (""")"); + end if; + + -- First, check if we are looking for a directory tree, + -- indicated by "/**" at the end. + + if Directory'Length >= 3 + and then Directory (Directory'Last - 1 .. Directory'Last) = "**" + and then (Directory (Directory'Last - 2) = '/' + or else + Directory (Directory'Last - 2) = Dir_Sep) + then + Name_Len := Directory'Length - 3; + + if Name_Len = 0 then + -- This is the case of "/**": all directories + -- in the file system. + + Name_Len := 1; + Name_Buffer (1) := Directory (Directory'First); + + else + Name_Buffer (1 .. Name_Len) := + Directory (Directory'First .. Directory'Last - 3); + end if; + + if Current_Verbosity = High then + Write_Str ("Looking for all subdirectories of """); + Write_Str (Name_Buffer (1 .. Name_Len)); + Write_Line (""""); + end if; + + declare + Base_Dir : constant Name_Id := Name_Find; + Root : constant Name_Id := + Locate_Directory (Base_Dir, Data.Directory); + + begin + if Root = No_Name then + Error_Msg_Name_1 := Base_Dir; + if Location = No_Location then + Error_Msg ("{ is not a valid directory.", Data.Location); + else + Error_Msg ("{ is not a valid directory.", Location); + end if; + + else + -- We have an existing directory, + -- we register it and all of its subdirectories. + + if Current_Verbosity = High then + Write_Line ("Looking for source directories:"); + end if; + + Start_String; + Store_String_Chars (Get_Name_String (Root)); + Recursive_Find_Dirs (End_String); + + if Current_Verbosity = High then + Write_Line ("End of looking for source directories."); + end if; + end if; + end; + + -- We have a single directory + + else + declare + Path_Name : constant Name_Id := + Locate_Directory (Directory_Id, Data.Directory); + + begin + if Path_Name = No_Name then + Error_Msg_Name_1 := Directory_Id; + if Location = No_Location then + Error_Msg ("{ is not a valid directory", Data.Location); + else + Error_Msg ("{ is not a valid directory", Location); + end if; + else + + -- As it is an existing directory, we add it to + -- the list of directories. + + String_Elements.Increment_Last; + Start_String; + Store_String_Chars (Get_Name_String (Path_Name)); + Element.Value := End_String; + + if Last_Source_Dir = Nil_String then + + -- This is the first source directory + + Data.Source_Dirs := String_Elements.Last; + + else + -- We already have source directories, + -- link the previous last to the new one. + + String_Elements.Table (Last_Source_Dir).Next := + String_Elements.Last; + end if; + + -- And register this source directory as the new last + + Last_Source_Dir := String_Elements.Last; + String_Elements.Table (Last_Source_Dir) := Element; + end if; + end; + end if; + end Find_Source_Dirs; + + ------------------ + -- Find_Sources -- + ------------------ + + procedure Find_Sources is + Source_Dir : String_List_Id := Data.Source_Dirs; + Element : String_Element; + Dir : Dir_Type; + Current_Source : String_List_Id := Nil_String; + + begin + if Current_Verbosity = High then + Write_Line ("Looking for sources:"); + end if; + + -- For each subdirectory + + while Source_Dir /= Nil_String loop + begin + Element := String_Elements.Table (Source_Dir); + if Element.Value /= No_String then + declare + Source_Directory : String + (1 .. Integer (String_Length (Element.Value))); + begin + String_To_Name_Buffer (Element.Value); + Source_Directory := Name_Buffer (1 .. Name_Len); + if Current_Verbosity = High then + Write_Str ("Source_Dir = "); + Write_Line (Source_Directory); + end if; + + -- We look to every entry in the source directory + + Open (Dir, Source_Directory); + + loop + Read (Dir, Name_Buffer, Name_Len); + + if Current_Verbosity = High then + Write_Str (" Checking "); + Write_Line (Name_Buffer (1 .. Name_Len)); + end if; + + exit when Name_Len = 0; + + declare + Path_Access : constant GNAT.OS_Lib.String_Access := + Locate_Regular_File + (Name_Buffer (1 .. Name_Len), + Source_Directory); + + File_Name : Name_Id; + Path_Name : Name_Id; + + begin + -- If it is a regular file + + if Path_Access /= null then + File_Name := Name_Find; + Name_Len := Path_Access'Length; + Name_Buffer (1 .. Name_Len) := Path_Access.all; + Path_Name := Name_Find; + + -- We attempt to register it as a source. + -- However, there is no error if the file + -- does not contain a valid source (as + -- indicated by Error_If_Invalid => False). + -- But there is an error if we have a + -- duplicate unit name. + + Record_Source + (File_Name => File_Name, + Path_Name => Path_Name, + Project => Project, + Data => Data, + Error_If_Invalid => False, + Location => No_Location, + Current_Source => Current_Source); + + else + if Current_Verbosity = High then + Write_Line + (" Not a regular file."); + end if; + end if; + end; + end loop; + + Close (Dir); + end; + end if; + + exception + when Directory_Error => + null; + end; + + Source_Dir := Element.Next; + end loop; + + if Current_Verbosity = High then + Write_Line ("end Looking for sources."); + end if; + + -- If we have looked for sources and found none, then + -- it is an error. If a project is not supposed to contain + -- any source, then we never call Find_Sources. + + if Current_Source = Nil_String then + Error_Msg ("there are no sources in this project", + Data.Location); + end if; + end Find_Sources; + + ------------------------------------- + -- Get_Path_Name_And_Record_Source -- + ------------------------------------- + + procedure Get_Path_Name_And_Record_Source + (File_Name : String; + Location : Source_Ptr; + Current_Source : in out String_List_Id) + is + Source_Dir : String_List_Id := Data.Source_Dirs; + Element : String_Element; + Path_Name : GNAT.OS_Lib.String_Access; + Found : Boolean := False; + File : Name_Id; + + begin + if Current_Verbosity = High then + Write_Str (" Checking """); + Write_Str (File_Name); + Write_Line ("""."); + end if; + + -- We look in all source directories for this file name + + while Source_Dir /= Nil_String loop + Element := String_Elements.Table (Source_Dir); + + if Current_Verbosity = High then + Write_Str (" """); + Write_Str (Get_Name_String (Element.Value)); + Write_Str (""": "); + end if; + + Path_Name := + Locate_Regular_File + (File_Name, + Get_Name_String (Element.Value)); + + if Path_Name /= null then + if Current_Verbosity = High then + Write_Line ("OK"); + end if; + + Name_Len := File_Name'Length; + Name_Buffer (1 .. Name_Len) := File_Name; + File := Name_Find; + Name_Len := Path_Name'Length; + Name_Buffer (1 .. Name_Len) := Path_Name.all; + + -- We register the source. + -- We report an error if the file does not + -- correspond to a source. + + Record_Source + (File_Name => File, + Path_Name => Name_Find, + Project => Project, + Data => Data, + Error_If_Invalid => True, + Location => Location, + Current_Source => Current_Source); + Found := True; + exit; + + else + if Current_Verbosity = High then + Write_Line ("No"); + end if; + + Source_Dir := Element.Next; + end if; + end loop; + + if not Found then + Name_Len := File_Name'Length; + Name_Buffer (1 .. Name_Len) := File_Name; + Error_Msg_Name_1 := Name_Find; + Error_Msg + ("cannot find source {", Location); + end if; + end Get_Path_Name_And_Record_Source; + + --------------------------- + -- Get_Sources_From_File -- + --------------------------- + + procedure Get_Sources_From_File + (Path : String; + Location : Source_Ptr) + is + File : Prj.Util.Text_File; + Line : String (1 .. 250); + Last : Natural; + Current_Source : String_List_Id := Nil_String; + + Nmb_Errors : constant Nat := Errors_Detected; + + begin + if Current_Verbosity = High then + Write_Str ("Opening """); + Write_Str (Path); + Write_Line ("""."); + end if; + + -- We open the file + + Prj.Util.Open (File, Path); + + if not Prj.Util.Is_Valid (File) then + Error_Msg ("file does not exist", Location); + else + while not Prj.Util.End_Of_File (File) loop + Prj.Util.Get_Line (File, Line, Last); + + -- If the line is not empty and does not start with "--", + -- then it must contains a file name. + + if Last /= 0 + and then (Last = 1 or else Line (1 .. 2) /= "--") + then + Get_Path_Name_And_Record_Source + (File_Name => Line (1 .. Last), + Location => Location, + Current_Source => Current_Source); + exit when Nmb_Errors /= Errors_Detected; + end if; + end loop; + + Prj.Util.Close (File); + + end if; + + -- We should have found at least one source. + -- If not, report an error. + + if Current_Source = Nil_String then + Error_Msg ("this project has no source", Location); + end if; + end Get_Sources_From_File; + + -- Start of processing for Check_Naming_Scheme + + begin + + Error_Report := Report_Error; + + if Current_Verbosity = High then + Write_Line ("Starting to look for directories"); + end if; + + -- Let's check the object directory + + declare + Object_Dir : Variable_Value := + Util.Value_Of (Name_Object_Dir, Data.Decl.Attributes); + + begin + pragma Assert (Object_Dir.Kind = Single, + "Object_Dir is not a single string"); + + -- We set the object directory to its default + + Data.Object_Directory := Data.Directory; + + if not String_Equal (Object_Dir.Value, Empty_String) then + + String_To_Name_Buffer (Object_Dir.Value); + + if Name_Len = 0 then + Error_Msg ("Object_Dir cannot be empty", + Object_Dir.Location); + + else + -- We check that the specified object directory + -- does exist. + + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + + declare + Dir_Id : constant Name_Id := Name_Find; + + begin + Data.Object_Directory := + Locate_Directory (Dir_Id, Data.Directory); + + if Data.Object_Directory = No_Name then + Error_Msg_Name_1 := Dir_Id; + Error_Msg + ("the object directory { cannot be found", + Data.Location); + end if; + end; + end if; + end if; + end; + + if Current_Verbosity = High then + if Data.Object_Directory = No_Name then + Write_Line ("No object directory"); + else + Write_Str ("Object directory: """); + Write_Str (Get_Name_String (Data.Object_Directory)); + Write_Line (""""); + end if; + end if; + + -- Let's check the source directories + + declare + Source_Dirs : Variable_Value := + Util.Value_Of (Name_Source_Dirs, Data.Decl.Attributes); + + begin + + if Current_Verbosity = High then + Write_Line ("Starting to look for source directories"); + end if; + + pragma Assert (Source_Dirs.Kind = List, + "Source_Dirs is not a list"); + + if Source_Dirs.Default then + + -- No Source_Dirs specified: the single source directory + -- is the one containing the project file + + String_Elements.Increment_Last; + Data.Source_Dirs := String_Elements.Last; + Start_String; + Store_String_Chars (Get_Name_String (Data.Directory)); + String_Elements.Table (Data.Source_Dirs) := + (Value => End_String, + Location => No_Location, + Next => Nil_String); + + if Current_Verbosity = High then + Write_Line ("(Undefined) Single object directory:"); + Write_Str (" """); + Write_Str (Get_Name_String (Data.Directory)); + Write_Line (""""); + end if; + + elsif Source_Dirs.Values = Nil_String then + + -- If Source_Dirs is an empty string list, this means + -- that this project contains no source. + + if Data.Object_Directory = Data.Directory then + Data.Object_Directory := No_Name; + end if; + + Data.Source_Dirs := Nil_String; + + else + declare + Source_Dir : String_List_Id := Source_Dirs.Values; + Element : String_Element; + + begin + -- We will find the source directories for each + -- element of the list + + while Source_Dir /= Nil_String loop + Element := String_Elements.Table (Source_Dir); + Find_Source_Dirs (Element.Value, Element.Location); + Source_Dir := Element.Next; + end loop; + end; + end if; + + if Current_Verbosity = High then + Write_Line ("Puting source directories in canonical cases"); + end if; + + declare + Current : String_List_Id := Data.Source_Dirs; + Element : String_Element; + + begin + while Current /= Nil_String loop + Element := String_Elements.Table (Current); + if Element.Value /= No_String then + String_To_Name_Buffer (Element.Value); + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + Start_String; + Store_String_Chars (Name_Buffer (1 .. Name_Len)); + Element.Value := End_String; + String_Elements.Table (Current) := Element; + end if; + + Current := Element.Next; + end loop; + end; + end; + + -- Library Dir, Name, Version and Kind + + declare + Attributes : constant Prj.Variable_Id := Data.Decl.Attributes; + + Lib_Dir : Prj.Variable_Value := + Prj.Util.Value_Of (Snames.Name_Library_Dir, Attributes); + + Lib_Name : Prj.Variable_Value := + Prj.Util.Value_Of (Snames.Name_Library_Name, Attributes); + + Lib_Version : Prj.Variable_Value := + Prj.Util.Value_Of + (Snames.Name_Library_Version, Attributes); + + The_Lib_Kind : Prj.Variable_Value := + Prj.Util.Value_Of + (Snames.Name_Library_Kind, Attributes); + + begin + pragma Assert (Lib_Dir.Kind = Single); + + if Lib_Dir.Value = Empty_String then + + if Current_Verbosity = High then + Write_Line ("No library directory"); + end if; + + else + -- Find path name, check that it is a directory + + Stringt.String_To_Name_Buffer (Lib_Dir.Value); + + declare + Dir_Id : constant Name_Id := Name_Find; + + begin + Data.Library_Dir := + Locate_Directory (Dir_Id, Data.Directory); + + if Data.Library_Dir = No_Name then + Error_Msg ("not an existing directory", + Lib_Dir.Location); + + elsif Data.Library_Dir = Data.Object_Directory then + Error_Msg + ("library directory cannot be the same " & + "as object directory", + Lib_Dir.Location); + Data.Library_Dir := No_Name; + + else + if Current_Verbosity = High then + Write_Str ("Library directory ="""); + Write_Str (Get_Name_String (Data.Library_Dir)); + Write_Line (""""); + end if; + end if; + end; + end if; + + pragma Assert (Lib_Name.Kind = Single); + + if Lib_Name.Value = Empty_String then + if Current_Verbosity = High then + Write_Line ("No library name"); + end if; + + else + Stringt.String_To_Name_Buffer (Lib_Name.Value); + + if not Is_Letter (Name_Buffer (1)) then + Error_Msg ("must start with a letter", + Lib_Name.Location); + + else + Data.Library_Name := Name_Find; + + for Index in 2 .. Name_Len loop + if not Is_Alphanumeric (Name_Buffer (Index)) then + Data.Library_Name := No_Name; + Error_Msg ("only letters and digits are allowed", + Lib_Name.Location); + exit; + end if; + end loop; + + if Data.Library_Name /= No_Name + and then Current_Verbosity = High then + Write_Str ("Library name = """); + Write_Str (Get_Name_String (Data.Library_Name)); + Write_Line (""""); + end if; + end if; + end if; + + Data.Library := + Data.Library_Dir /= No_Name + and then + Data.Library_Name /= No_Name; + + if Data.Library then + if Current_Verbosity = High then + Write_Line ("This is a library project file"); + end if; + + pragma Assert (Lib_Version.Kind = Single); + + if Lib_Version.Value = Empty_String then + if Current_Verbosity = High then + Write_Line ("No library version specified"); + end if; + + else + Stringt.String_To_Name_Buffer (Lib_Version.Value); + Data.Lib_Internal_Name := Name_Find; + end if; + + pragma Assert (The_Lib_Kind.Kind = Single); + + if The_Lib_Kind.Value = Empty_String then + if Current_Verbosity = High then + Write_Line ("No library kind specified"); + end if; + + else + Stringt.String_To_Name_Buffer (The_Lib_Kind.Value); + + declare + Kind_Name : constant String := + Ada.Characters.Handling.To_Lower + (Name_Buffer (1 .. Name_Len)); + + OK : Boolean := True; + + begin + if Kind_Name = "static" then + Data.Library_Kind := Static; + + elsif Kind_Name = "dynamic" then + Data.Library_Kind := Dynamic; + + elsif Kind_Name = "relocatable" then + Data.Library_Kind := Relocatable; + + else + Error_Msg + ("illegal value for Library_Kind", + The_Lib_Kind.Location); + OK := False; + end if; + + if Current_Verbosity = High and then OK then + Write_Str ("Library kind = "); + Write_Line (Kind_Name); + end if; + end; + end if; + end if; + end; + + if Current_Verbosity = High then + Show_Source_Dirs (Project); + end if; + + declare + Naming_Id : constant Package_Id := + Util.Value_Of (Name_Naming, Data.Decl.Packages); + + Naming : Package_Element; + + begin + -- If there is a package Naming, we will put in Data.Naming + -- what is in this package Naming. + + if Naming_Id /= No_Package then + Naming := Packages.Table (Naming_Id); + + if Current_Verbosity = High then + Write_Line ("Checking ""Naming""."); + end if; + + declare + Bodies : constant Array_Element_Id := + Util.Value_Of (Name_Body_Part, Naming.Decl.Arrays); + + Specifications : constant Array_Element_Id := + Util.Value_Of + (Name_Specification, Naming.Decl.Arrays); + + begin + if Bodies /= No_Array_Element then + + -- We have elements in the array Body_Part + + if Current_Verbosity = High then + Write_Line ("Found Bodies."); + end if; + + Data.Naming.Bodies := Bodies; + Check_Unit_Names (Bodies); + + else + if Current_Verbosity = High then + Write_Line ("No Bodies."); + end if; + end if; + + if Specifications /= No_Array_Element then + + -- We have elements in the array Specification + + if Current_Verbosity = High then + Write_Line ("Found Specifications."); + end if; + + Data.Naming.Specifications := Specifications; + Check_Unit_Names (Specifications); + + else + if Current_Verbosity = High then + Write_Line ("No Specifications."); + end if; + end if; + end; + + -- We are now checking if variables Dot_Replacement, Casing, + -- Specification_Append, Body_Append and/or Separate_Append + -- exist. + -- For each variable, if it does not exist, we do nothing, + -- because we already have the default. + + -- Let's check Dot_Replacement + + declare + Dot_Replacement : constant Variable_Value := + Util.Value_Of + (Name_Dot_Replacement, + Naming.Decl.Attributes); + + begin + pragma Assert (Dot_Replacement.Kind = Single, + "Dot_Replacement is not a single string"); + + if not Dot_Replacement.Default then + + String_To_Name_Buffer (Dot_Replacement.Value); + + if Name_Len = 0 then + Error_Msg ("Dot_Replacement cannot be empty", + Dot_Replacement.Location); + + else + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + Data.Naming.Dot_Replacement := Name_Find; + Data.Naming.Dot_Repl_Loc := Dot_Replacement.Location; + end if; + + end if; + + end; + + if Current_Verbosity = High then + Write_Str (" Dot_Replacement = """); + Write_Str (Get_Name_String (Data.Naming.Dot_Replacement)); + Write_Char ('"'); + Write_Eol; + end if; + + -- Check Casing + + declare + Casing_String : constant Variable_Value := + Util.Value_Of (Name_Casing, Naming.Decl.Attributes); + + begin + pragma Assert (Casing_String.Kind = Single, + "Dot_Replacement is not a single string"); + + if not Casing_String.Default then + declare + Casing_Image : constant String := + Get_Name_String (Casing_String.Value); + + begin + declare + Casing : constant Casing_Type := + Value (Casing_Image); + + begin + Data.Naming.Casing := Casing; + end; + + exception + when Constraint_Error => + if Casing_Image'Length = 0 then + Error_Msg ("Casing cannot be an empty string", + Casing_String.Location); + + else + Name_Len := Casing_Image'Length; + Name_Buffer (1 .. Name_Len) := Casing_Image; + Error_Msg_Name_1 := Name_Find; + Error_Msg + ("{ is not a correct Casing", + Casing_String.Location); + end if; + end; + end if; + end; + + if Current_Verbosity = High then + Write_Str (" Casing = "); + Write_Str (Image (Data.Naming.Casing)); + Write_Char ('.'); + Write_Eol; + end if; + + -- Let's check Specification_Append + + declare + Specification_Append : constant Variable_Value := + Util.Value_Of + (Name_Specification_Append, + Naming.Decl.Attributes); + + begin + pragma Assert (Specification_Append.Kind = Single, + "Specification_Append is not a single string"); + + if not Specification_Append.Default then + String_To_Name_Buffer (Specification_Append.Value); + + if Name_Len = 0 then + Error_Msg ("Specification_Append cannot be empty", + Specification_Append.Location); + + else + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + Data.Naming.Specification_Append := Name_Find; + Data.Naming.Spec_Append_Loc := + Specification_Append.Location; + end if; + end if; + end; + + if Current_Verbosity = High then + Write_Str (" Specification_Append = """); + Write_Str (Get_Name_String (Data.Naming.Specification_Append)); + Write_Line ("""."); + end if; + + -- Check Body_Append + + declare + Body_Append : constant Variable_Value := + Util.Value_Of + (Name_Body_Append, Naming.Decl.Attributes); + + begin + pragma Assert (Body_Append.Kind = Single, + "Body_Append is not a single string"); + + if not Body_Append.Default then + + String_To_Name_Buffer (Body_Append.Value); + + if Name_Len = 0 then + Error_Msg ("Body_Append cannot be empty", + Body_Append.Location); + + else + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + Data.Naming.Body_Append := Name_Find; + Data.Naming.Body_Append_Loc := Body_Append.Location; + + -- As we have a new Body_Append, we set Separate_Append + -- to the same value. + + Data.Naming.Separate_Append := Data.Naming.Body_Append; + Data.Naming.Sep_Append_Loc := Data.Naming.Body_Append_Loc; + end if; + end if; + end; + + if Current_Verbosity = High then + Write_Str (" Body_Append = """); + Write_Str (Get_Name_String (Data.Naming.Body_Append)); + Write_Line ("""."); + end if; + + -- Check Separate_Append + + declare + Separate_Append : constant Variable_Value := + Util.Value_Of + (Name_Separate_Append, + Naming.Decl.Attributes); + + begin + pragma Assert (Separate_Append.Kind = Single, + "Separate_Append is not a single string"); + + if not Separate_Append.Default then + String_To_Name_Buffer (Separate_Append.Value); + + if Name_Len = 0 then + Error_Msg ("Separate_Append cannot be empty", + Separate_Append.Location); + + else + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + Data.Naming.Separate_Append := Name_Find; + Data.Naming.Sep_Append_Loc := Separate_Append.Location; + end if; + end if; + end; + + if Current_Verbosity = High then + Write_Str (" Separate_Append = """); + Write_Str (Get_Name_String (Data.Naming.Separate_Append)); + Write_Line ("""."); + Write_Line ("end Naming."); + end if; + + -- Now, we check if Data.Naming is valid + + Check_Naming_Scheme (Data.Naming); + end if; + end; + + -- If we have source directories, then let's find the sources. + + if Data.Source_Dirs /= Nil_String then + declare + Sources : constant Variable_Value := + Util.Value_Of + (Name_Source_Files, + Data.Decl.Attributes); + + Source_List_File : constant Variable_Value := + Util.Value_Of + (Name_Source_List_File, + Data.Decl.Attributes); + + begin + pragma Assert + (Sources.Kind = List, + "Source_Files is not a list"); + pragma Assert + (Source_List_File.Kind = Single, + "Source_List_File is not a single string"); + + if not Sources.Default then + if not Source_List_File.Default then + Error_Msg + ("?both variables source_files and " & + "source_list_file are present", + Source_List_File.Location); + end if; + + -- Sources is a list of file names + + declare + Current_Source : String_List_Id := Nil_String; + Current : String_List_Id := Sources.Values; + Element : String_Element; + + begin + while Current /= Nil_String loop + Element := String_Elements.Table (Current); + String_To_Name_Buffer (Element.Value); + + declare + File_Name : constant String := + Name_Buffer (1 .. Name_Len); + + begin + Get_Path_Name_And_Record_Source + (File_Name => File_Name, + Location => Element.Location, + Current_Source => Current_Source); + Current := Element.Next; + end; + end loop; + end; + + -- No source_files specified. + -- We check Source_List_File has been specified. + + elsif not Source_List_File.Default then + + -- Source_List_File is the name of the file + -- that contains the source file names + + declare + Source_File_Path_Name : constant String := + Path_Name_Of + (Source_List_File.Value, + Data.Directory); + + begin + if Source_File_Path_Name'Length = 0 then + String_To_Name_Buffer (Source_List_File.Value); + Error_Msg_Name_1 := Name_Find; + Error_Msg + ("file with sources { does not exist", + Source_List_File.Location); + + else + Get_Sources_From_File + (Source_File_Path_Name, + Source_List_File.Location); + end if; + end; + + else + -- Neither Source_Files nor Source_List_File has been + -- specified. + -- Find all the files that satisfy + -- the naming scheme in all the source directories. + + Find_Sources; + end if; + end; + end if; + + Projects.Table (Project) := Data; + end Check_Naming_Scheme; + + --------------- + -- Error_Msg -- + --------------- + + procedure Error_Msg (Msg : String; Flag_Location : Source_Ptr) is + begin + if Error_Report = null then + Errout.Error_Msg (Msg, Flag_Location); + + else + declare + Error_Buffer : String (1 .. 5_000); + Error_Last : Natural := 0; + Msg_Name : Natural := 0; + First : Positive := Msg'First; + + procedure Add (C : Character); + -- Add a character to the buffer + + procedure Add (S : String); + -- Add a string to the buffer + + procedure Add (Id : Name_Id); + -- Add a name to the buffer + + --------- + -- Add -- + --------- + + procedure Add (C : Character) is + begin + Error_Last := Error_Last + 1; + Error_Buffer (Error_Last) := C; + end Add; + + procedure Add (S : String) is + begin + Error_Buffer (Error_Last + 1 .. Error_Last + S'Length) := S; + Error_Last := Error_Last + S'Length; + end Add; + + procedure Add (Id : Name_Id) is + begin + Get_Name_String (Id); + Add (Name_Buffer (1 .. Name_Len)); + end Add; + + begin + if Msg (First) = '\' then + -- Continuation character, ignore. + First := First + 1; + + elsif Msg (First) = '?' then + -- Warning character. It is always the first one, + -- in this package. + First := First + 1; + Add ("Warning: "); + end if; + + for Index in First .. Msg'Last loop + if Msg (Index) = '{' or else Msg (Index) = '%' then + -- Include a name between double quotes. + Msg_Name := Msg_Name + 1; + Add ('"'); + + case Msg_Name is + when 1 => Add (Error_Msg_Name_1); + + when 2 => Add (Error_Msg_Name_2); + + when 3 => Add (Error_Msg_Name_3); + + when others => null; + end case; + + Add ('"'); + + else + Add (Msg (Index)); + end if; + + end loop; + + Error_Report (Error_Buffer (1 .. Error_Last)); + end; + end if; + end Error_Msg; + + --------------------- + -- Get_Name_String -- + --------------------- + + function Get_Name_String (S : String_Id) return String is + begin + if S = No_String then + return ""; + else + String_To_Name_Buffer (S); + return Name_Buffer (1 .. Name_Len); + end if; + end Get_Name_String; + + -------------- + -- Get_Unit -- + -------------- + + procedure Get_Unit + (File_Name : Name_Id; + Naming : Naming_Data; + Unit_Name : out Name_Id; + Unit_Kind : out Spec_Or_Body; + Needs_Pragma : out Boolean) + is + Canonical_Case_Name : Name_Id; + + begin + Needs_Pragma := False; + Get_Name_String (File_Name); + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + Canonical_Case_Name := Name_Find; + + if Naming.Bodies /= No_Array_Element then + + -- There are some specified file names for some bodies + -- of this project. Find out if File_Name is one of these bodies. + + declare + Current : Array_Element_Id := Naming.Bodies; + Element : Array_Element; + + begin + while Current /= No_Array_Element loop + Element := Array_Elements.Table (Current); + + if Element.Index /= No_Name then + String_To_Name_Buffer (Element.Value.Value); + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + + if Canonical_Case_Name = Name_Find then + + -- File_Name corresponds to one body. + -- So, we know it is a body, and we know the unit name. + + Unit_Kind := Body_Part; + Unit_Name := Element.Index; + Needs_Pragma := True; + return; + end if; + end if; + + Current := Element.Next; + end loop; + end; + end if; + + if Naming.Specifications /= No_Array_Element then + + -- There are some specified file names for some bodiesspecifications + -- of this project. Find out if File_Name is one of these + -- specifications. + + declare + Current : Array_Element_Id := Naming.Specifications; + Element : Array_Element; + + begin + while Current /= No_Array_Element loop + Element := Array_Elements.Table (Current); + + if Element.Index /= No_Name then + String_To_Name_Buffer (Element.Value.Value); + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + + if Canonical_Case_Name = Name_Find then + + -- File_Name corresponds to one specification. + -- So, we know it is a spec, and we know the unit name. + + Unit_Kind := Specification; + Unit_Name := Element.Index; + Needs_Pragma := True; + return; + end if; + + end if; + + Current := Element.Next; + end loop; + end; + end if; + + declare + File : String := Get_Name_String (Canonical_Case_Name); + First : Positive := File'First; + Last : Natural := File'Last; + + begin + -- Check if the end of the file name is Specification_Append + + Get_Name_String (Naming.Specification_Append); + + if File'Length > Name_Len + and then File (Last - Name_Len + 1 .. Last) = + Name_Buffer (1 .. Name_Len) + then + -- We have a spec + + Unit_Kind := Specification; + Last := Last - Name_Len; + + if Current_Verbosity = High then + Write_Str (" Specification: "); + Write_Line (File (First .. Last)); + end if; + + else + Get_Name_String (Naming.Body_Append); + + -- Check if the end of the file name is Body_Append + + if File'Length > Name_Len + and then File (Last - Name_Len + 1 .. Last) = + Name_Buffer (1 .. Name_Len) + then + -- We have a body + + Unit_Kind := Body_Part; + Last := Last - Name_Len; + + if Current_Verbosity = High then + Write_Str (" Body: "); + Write_Line (File (First .. Last)); + end if; + + elsif Naming.Separate_Append /= Naming.Body_Append then + Get_Name_String (Naming.Separate_Append); + + -- Check if the end of the file name is Separate_Append + + if File'Length > Name_Len + and then File (Last - Name_Len + 1 .. Last) = + Name_Buffer (1 .. Name_Len) + then + -- We have a separate (a body) + + Unit_Kind := Body_Part; + Last := Last - Name_Len; + + if Current_Verbosity = High then + Write_Str (" Separate: "); + Write_Line (File (First .. Last)); + end if; + + else + Last := 0; + end if; + + else + Last := 0; + end if; + end if; + + if Last = 0 then + + -- This is not a source file + + Unit_Name := No_Name; + Unit_Kind := Specification; + + if Current_Verbosity = High then + Write_Line (" Not a valid file name."); + end if; + + return; + end if; + + Get_Name_String (Naming.Dot_Replacement); + + if Name_Buffer (1 .. Name_Len) /= "." then + + -- If Dot_Replacement is not a single dot, + -- then there should not be any dot in the name. + + for Index in First .. Last loop + if File (Index) = '.' then + if Current_Verbosity = High then + Write_Line + (" Not a valid file name (some dot not replaced)."); + end if; + + Unit_Name := No_Name; + return; + + end if; + end loop; + + -- Replace the substring Dot_Replacement with dots + + declare + Index : Positive := First; + + begin + while Index <= Last - Name_Len + 1 loop + + if File (Index .. Index + Name_Len - 1) = + Name_Buffer (1 .. Name_Len) + then + File (Index) := '.'; + + if Name_Len > 1 and then Index < Last then + File (Index + 1 .. Last - Name_Len + 1) := + File (Index + Name_Len .. Last); + end if; + + Last := Last - Name_Len + 1; + end if; + + Index := Index + 1; + end loop; + end; + end if; + + -- Check if the casing is right + + declare + Src : String := File (First .. Last); + + begin + case Naming.Casing is + when All_Lower_Case => + Fixed.Translate + (Source => Src, + Mapping => Lower_Case_Map); + + when All_Upper_Case => + Fixed.Translate + (Source => Src, + Mapping => Upper_Case_Map); + + when Mixed_Case | Unknown => + null; + end case; + + if Src /= File (First .. Last) then + if Current_Verbosity = High then + Write_Line (" Not a valid file name (casing)."); + end if; + + Unit_Name := No_Name; + return; + end if; + + -- We put the name in lower case + + Fixed.Translate + (Source => Src, + Mapping => Lower_Case_Map); + + if Current_Verbosity = High then + Write_Str (" "); + Write_Line (Src); + end if; + + Name_Len := Src'Length; + Name_Buffer (1 .. Name_Len) := Src; + + -- Now, we check if this name is a valid unit name + + Check_Naming_Scheme (Name => Name_Find, Unit => Unit_Name); + end; + + end; + + end Get_Unit; + + ----------------------- + -- Is_Illegal_Append -- + ----------------------- + + function Is_Illegal_Append (This : String) return Boolean is + begin + return This'Length = 0 + or else Is_Alphanumeric (This (This'First)) + or else (This'Length >= 2 + and then This (This'First) = '_' + and then Is_Alphanumeric (This (This'First + 1))); + end Is_Illegal_Append; + + ---------------------- + -- Locate_Directory -- + ---------------------- + + function Locate_Directory + (Name : Name_Id; + Parent : Name_Id) + return Name_Id + is + The_Name : constant String := Get_Name_String (Name); + The_Parent : constant String := + Get_Name_String (Parent) & Dir_Sep; + + The_Parent_Last : Positive := The_Parent'Last; + + begin + if The_Parent'Length > 1 + and then (The_Parent (The_Parent_Last - 1) = Dir_Sep + or else The_Parent (The_Parent_Last - 1) = '/') + then + The_Parent_Last := The_Parent_Last - 1; + end if; + + if Current_Verbosity = High then + Write_Str ("Locate_Directory ("""); + Write_Str (The_Name); + Write_Str (""", """); + Write_Str (The_Parent); + Write_Line (""")"); + end if; + + if Is_Absolute_Path (The_Name) then + if Is_Directory (The_Name) then + return Name; + end if; + + else + declare + Full_Path : constant String := + The_Parent (The_Parent'First .. The_Parent_Last) & + The_Name; + + begin + if Is_Directory (Full_Path) then + Name_Len := Full_Path'Length; + Name_Buffer (1 .. Name_Len) := Full_Path; + return Name_Find; + end if; + end; + + end if; + + return No_Name; + end Locate_Directory; + + ------------------ + -- Path_Name_Of -- + ------------------ + + function Path_Name_Of + (File_Name : String_Id; + Directory : String_Id) + return String + is + Result : String_Access; + + begin + String_To_Name_Buffer (File_Name); + + declare + The_File_Name : constant String := Name_Buffer (1 .. Name_Len); + + begin + String_To_Name_Buffer (Directory); + Result := Locate_Regular_File + (File_Name => The_File_Name, + Path => Name_Buffer (1 .. Name_Len)); + end; + + if Result = null then + return ""; + else + Canonical_Case_File_Name (Result.all); + return Result.all; + end if; + end Path_Name_Of; + + function Path_Name_Of + (File_Name : String_Id; + Directory : Name_Id) + return String + is + Result : String_Access; + The_Directory : constant String := Get_Name_String (Directory); + + begin + String_To_Name_Buffer (File_Name); + Result := Locate_Regular_File + (File_Name => Name_Buffer (1 .. Name_Len), + Path => The_Directory); + + if Result = null then + return ""; + else + Canonical_Case_File_Name (Result.all); + return Result.all; + end if; + end Path_Name_Of; + + ------------------- + -- Record_Source -- + ------------------- + + procedure Record_Source + (File_Name : Name_Id; + Path_Name : Name_Id; + Project : Project_Id; + Data : in out Project_Data; + Error_If_Invalid : Boolean; + Location : Source_Ptr; + Current_Source : in out String_List_Id) + is + Unit_Name : Name_Id; + Unit_Kind : Spec_Or_Body; + Needs_Pragma : Boolean; + The_Location : Source_Ptr := Location; + + begin + -- Find out the unit name, the unit kind and if it needs + -- a specific SFN pragma. + + Get_Unit + (File_Name => File_Name, + Naming => Data.Naming, + Unit_Name => Unit_Name, + Unit_Kind => Unit_Kind, + Needs_Pragma => Needs_Pragma); + + -- If it is not a source file, report an error only if + -- Error_If_Invalid is true. + + if Unit_Name = No_Name then + if Error_If_Invalid then + Error_Msg_Name_1 := File_Name; + Error_Msg + ("{ is not a valid source file name", + Location); + + else + if Current_Verbosity = High then + Write_Str (" """); + Write_Str (Get_Name_String (File_Name)); + Write_Line (""" is not a valid source file name (ignored)."); + end if; + end if; + + else + -- Put the file name in the list of sources of the project + + String_Elements.Increment_Last; + Get_Name_String (File_Name); + Start_String; + Store_String_Chars (Name_Buffer (1 .. Name_Len)); + String_Elements.Table (String_Elements.Last) := + (Value => End_String, + Location => No_Location, + Next => Nil_String); + + if Current_Source = Nil_String then + Data.Sources := String_Elements.Last; + + else + String_Elements.Table (Current_Source).Next := + String_Elements.Last; + end if; + + Current_Source := String_Elements.Last; + + -- Put the unit in unit list + + declare + The_Unit : Unit_Id := Units_Htable.Get (Unit_Name); + The_Unit_Data : Unit_Data; + + begin + if Current_Verbosity = High then + Write_Str ("Putting "); + Write_Str (Get_Name_String (Unit_Name)); + Write_Line (" in the unit list."); + end if; + + -- The unit is already in the list, but may be it is + -- only the other unit kind (spec or body), or what is + -- in the unit list is a unit of a project we are modifying. + + if The_Unit /= Prj.Com.No_Unit then + The_Unit_Data := Units.Table (The_Unit); + + if The_Unit_Data.File_Names (Unit_Kind).Name = No_Name + or else (Data.Modifies /= No_Project + and then + The_Unit_Data.File_Names (Unit_Kind).Project = + Data.Modifies) + then + The_Unit_Data.File_Names (Unit_Kind) := + (Name => File_Name, + Path => Path_Name, + Project => Project, + Needs_Pragma => Needs_Pragma); + Units.Table (The_Unit) := The_Unit_Data; + + else + -- It is an error to have two units with the same name + -- and the same kind (spec or body). + + if The_Location = No_Location then + The_Location := Projects.Table (Project).Location; + end if; + + Error_Msg_Name_1 := Unit_Name; + Error_Msg ("duplicate source {", The_Location); + + Error_Msg_Name_1 := + Projects.Table + (The_Unit_Data.File_Names (Unit_Kind).Project).Name; + Error_Msg_Name_2 := + The_Unit_Data.File_Names (Unit_Kind).Path; + Error_Msg ("\ project file {, {", The_Location); + + Error_Msg_Name_1 := Projects.Table (Project).Name; + Error_Msg_Name_2 := Path_Name; + Error_Msg ("\ project file {, {", The_Location); + + end if; + + -- It is a new unit, create a new record + + else + Units.Increment_Last; + The_Unit := Units.Last; + Units_Htable.Set (Unit_Name, The_Unit); + The_Unit_Data.Name := Unit_Name; + The_Unit_Data.File_Names (Unit_Kind) := + (Name => File_Name, + Path => Path_Name, + Project => Project, + Needs_Pragma => Needs_Pragma); + Units.Table (The_Unit) := The_Unit_Data; + end if; + end; + end if; + end Record_Source; + + ---------------------- + -- Show_Source_Dirs -- + ---------------------- + + procedure Show_Source_Dirs (Project : Project_Id) is + Current : String_List_Id := Projects.Table (Project).Source_Dirs; + Element : String_Element; + + begin + Write_Line ("Source_Dirs:"); + + while Current /= Nil_String loop + Element := String_Elements.Table (Current); + Write_Str (" "); + Write_Line (Get_Name_String (Element.Value)); + Current := Element.Next; + end loop; + + Write_Line ("end Source_Dirs."); + end Show_Source_Dirs; + +end Prj.Nmsc; diff --git a/gcc/ada/prj-nmsc.ads b/gcc/ada/prj-nmsc.ads new file mode 100644 index 00000000000..5fcc00538da --- /dev/null +++ b/gcc/ada/prj-nmsc.ads @@ -0,0 +1,43 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . N M S C -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.3 $ +-- -- +-- Copyright (C) 2000-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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- Check the Naming Scheme of a project file, find the directories +-- and the source files. + +private package Prj.Nmsc is + + procedure Check_Naming_Scheme + (Project : Project_Id; + Report_Error : Put_Line_Access); + -- Check that the Naming Scheme of a project is legal. Find the + -- object directory, the source directories, and the source files. + -- Check the source files against the Naming Scheme. + -- If Report_Error is null , use the standard error reporting mechanism + -- (Errout). Otherwise, report errors using Report_Error. + +end Prj.Nmsc; diff --git a/gcc/ada/prj-pars.adb b/gcc/ada/prj-pars.adb new file mode 100644 index 00000000000..620d2e113a9 --- /dev/null +++ b/gcc/ada/prj-pars.adb @@ -0,0 +1,92 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . P A R S -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.19 $ +-- -- +-- Copyright (C) 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 Ada.Exceptions; use Ada.Exceptions; + +with Errout; use Errout; +with Output; use Output; +with Prj.Com; use Prj.Com; +with Prj.Part; +with Prj.Proc; +with Prj.Tree; use Prj.Tree; + +package body Prj.Pars is + + ----------- + -- Parse -- + ----------- + + procedure Parse + (Project : out Project_Id; + Project_File_Name : String) + is + Project_Tree : Project_Node_Id := Empty_Node; + The_Project : Project_Id := No_Project; + + begin + -- Parse the main project file into a tree + + Prj.Part.Parse + (Project => Project_Tree, + Project_File_Name => Project_File_Name, + Always_Errout_Finalize => False); + + -- If there were no error, process the tree + + if Project_Tree /= Empty_Node then + Prj.Proc.Process + (Project => The_Project, + From_Project_Node => Project_Tree, + Report_Error => null); + Errout.Finalize; + end if; + + Project := The_Project; + + exception + when X : others => + + -- Internal error + + Write_Line (Exception_Information (X)); + Write_Str ("Exception "); + Write_Str (Exception_Name (X)); + Write_Line (" raised, while processing project file"); + Project := No_Project; + end Parse; + + ------------------- + -- Set_Verbosity -- + ------------------- + + procedure Set_Verbosity (To : in Verbosity) is + begin + Current_Verbosity := To; + end Set_Verbosity; + +end Prj.Pars; diff --git a/gcc/ada/prj-pars.ads b/gcc/ada/prj-pars.ads new file mode 100644 index 00000000000..0adaf72f4c2 --- /dev/null +++ b/gcc/ada/prj-pars.ads @@ -0,0 +1,44 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . P A R S -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.8 $ +-- -- +-- Copyright (C) 2000-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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- Implements the parsing of project files. + +package Prj.Pars is + + procedure Set_Verbosity (To : Verbosity); + -- Set the verbosity when parsing the project files. + + procedure Parse + (Project : out Project_Id; + Project_File_Name : String); + -- Parse a project files and all its imported project files. + -- If parsing is successful, Project_Id is the project ID + -- of the main project file; otherwise, Project_Id is set + -- to No_Project. + +end Prj.Pars; diff --git a/gcc/ada/prj-part.adb b/gcc/ada/prj-part.adb new file mode 100644 index 00000000000..8100ad49e95 --- /dev/null +++ b/gcc/ada/prj-part.adb @@ -0,0 +1,871 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . P A R T -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.10 $ +-- -- +-- Copyright (C) 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 Ada.Characters.Handling; use Ada.Characters.Handling; +with Ada.Exceptions; use Ada.Exceptions; +with Errout; use Errout; +with GNAT.Directory_Operations; use GNAT.Directory_Operations; +with GNAT.OS_Lib; use GNAT.OS_Lib; +with Namet; use Namet; +with Osint; use Osint; +with Output; use Output; +with Prj.Com; use Prj.Com; +with Prj.Dect; +with Scans; use Scans; +with Scn; use Scn; +with Sinfo; use Sinfo; +with Sinput; use Sinput; +with Sinput.P; use Sinput.P; +with Stringt; use Stringt; +with Table; +with Types; use Types; + +pragma Elaborate_All (GNAT.OS_Lib); + +package body Prj.Part is + + Dir_Sep : Character renames GNAT.OS_Lib.Directory_Separator; + + Project_File_Extension : String := ".gpr"; + + Project_Path : String_Access; + -- The project path; initialized during package elaboration. + + Ada_Project_Path : constant String := "ADA_PROJECT_PATH"; + Prj_Path : constant String_Access := Getenv (Ada_Project_Path); + + ------------------------------------ + -- Local Packages and Subprograms -- + ------------------------------------ + + package Project_Stack is new Table.Table + (Table_Component_Type => Name_Id, + Table_Index_Type => Nat, + Table_Low_Bound => 1, + Table_Initial => 10, + Table_Increment => 10, + Table_Name => "Prj.Part.Project_Stack"); + -- This table is used to detect circular dependencies + -- for imported and modified projects. + + procedure Parse_Context_Clause + (Context_Clause : out Project_Node_Id; + Project_Directory : Name_Id); + -- Parse the context clause of a project + -- Does nothing if there is b\no context clause (if the current + -- token is not "with"). + + procedure Parse_Single_Project + (Project : out Project_Node_Id; + Path_Name : String; + Modified : Boolean); + -- Parse a project file. + -- Recursive procedure: it calls itself for imported and + -- modified projects. + + function Path_Name_Of + (File_Name : String; + Directory : String) + return String; + -- Returns the path name of a (non project) file. + -- Returns an empty string if file cannot be found. + + function Project_Path_Name_Of + (Project_File_Name : String; + Directory : String) + return String; + -- Returns the path name of a project file. + -- Returns an empty string if project file cannot be found. + + function Immediate_Directory_Of (Path_Name : Name_Id) return Name_Id; + -- Get the directory of the file with the specified path name. + -- This includes the directory separator as the last character. + -- Returns "./" if Path_Name contains no directory separator. + + function Simple_File_Name_Of (Path_Name : Name_Id) return Name_Id; + -- Returns the name of a file with the specified path name + -- with no directory information. + + function Project_Name_From (Path_Name : String) return Name_Id; + -- Returns the name of the project that corresponds to its path name. + -- Returns No_Name if the path name is invalid, because the corresponding + -- project name does not have the syntax of an ada identifier. + + ---------------------------- + -- Immediate_Directory_Of -- + ---------------------------- + + function Immediate_Directory_Of (Path_Name : Name_Id) return Name_Id is + begin + Get_Name_String (Path_Name); + + for Index in reverse 1 .. Name_Len loop + if Name_Buffer (Index) = '/' + or else Name_Buffer (Index) = Dir_Sep + then + -- Remove from name all characters after the last + -- directory separator. + + Name_Len := Index; + return Name_Find; + end if; + end loop; + + -- There is no directory separator in name. Return "./" or ".\". + Name_Len := 2; + Name_Buffer (1) := '.'; + Name_Buffer (2) := Dir_Sep; + return Name_Find; + end Immediate_Directory_Of; + + ----------- + -- Parse -- + ----------- + + procedure Parse + (Project : out Project_Node_Id; + Project_File_Name : String; + Always_Errout_Finalize : Boolean) + is + Current_Directory : constant String := Get_Current_Dir; + + begin + Project := Empty_Node; + + if Current_Verbosity >= Medium then + Write_Str ("ADA_PROJECT_PATH="""); + Write_Str (Project_Path.all); + Write_Line (""""); + end if; + + declare + Path_Name : constant String := + Project_Path_Name_Of (Project_File_Name, + Directory => Current_Directory); + + begin + -- Initialize the tables + + Tree_Private_Part.Project_Nodes.Set_Last (Empty_Node); + Tree_Private_Part.Projects_Htable.Reset; + + Errout.Initialize; + + -- And parse the main project file + + if Path_Name = "" then + Fail ("project file """ & Project_File_Name & """ not found"); + end if; + + Parse_Single_Project + (Project => Project, + Path_Name => Path_Name, + Modified => False); + + if Errout.Errors_Detected > 0 then + Project := Empty_Node; + end if; + + if Project = Empty_Node or else Always_Errout_Finalize then + Errout.Finalize; + end if; + end; + + exception + when X : others => + + -- Internal error + + Write_Line (Exception_Information (X)); + Write_Str ("Exception "); + Write_Str (Exception_Name (X)); + Write_Line (" raised, while processing project file"); + Project := Empty_Node; + end Parse; + + -------------------------- + -- Parse_Context_Clause -- + -------------------------- + + procedure Parse_Context_Clause + (Context_Clause : out Project_Node_Id; + Project_Directory : Name_Id) + is + Project_Directory_Path : constant String := + Get_Name_String (Project_Directory); + Current_With_Clause : Project_Node_Id := Empty_Node; + Next_With_Clause : Project_Node_Id := Empty_Node; + + begin + -- Assume no context clause + + Context_Clause := Empty_Node; + With_Loop : + + -- If Token is not "with", there is no context clause, + -- or we have exhausted the with clauses. + + while Token = Tok_With loop + Comma_Loop : + loop + -- Scan past "with" or "," + + Scan; + Expect (Tok_String_Literal, "literal string"); + + if Token /= Tok_String_Literal then + return; + end if; + + -- New with clause + + if Current_With_Clause = Empty_Node then + + -- First with clause of the context clause + + Current_With_Clause := Default_Project_Node + (Of_Kind => N_With_Clause); + Context_Clause := Current_With_Clause; + + else + Next_With_Clause := Default_Project_Node + (Of_Kind => N_With_Clause); + Set_Next_With_Clause_Of (Current_With_Clause, Next_With_Clause); + Current_With_Clause := Next_With_Clause; + end if; + + Set_String_Value_Of (Current_With_Clause, Strval (Token_Node)); + Set_Location_Of (Current_With_Clause, Token_Ptr); + String_To_Name_Buffer (String_Value_Of (Current_With_Clause)); + + declare + Original_Path : constant String := + Name_Buffer (1 .. Name_Len); + + Imported_Path_Name : constant String := + Project_Path_Name_Of + (Original_Path, + Project_Directory_Path); + + Withed_Project : Project_Node_Id := Empty_Node; + + begin + if Imported_Path_Name = "" then + + -- The project file cannot be found + + Name_Len := Original_Path'Length; + Name_Buffer (1 .. Name_Len) := Original_Path; + Error_Msg_Name_1 := Name_Find; + + Error_Msg ("unknown project file: {", Token_Ptr); + + else + -- Parse the imported project + + Parse_Single_Project + (Project => Withed_Project, + Path_Name => Imported_Path_Name, + Modified => False); + + if Withed_Project /= Empty_Node then + + -- If parsing was successful, record project name + -- and path name in with clause + + Set_Project_Node_Of (Current_With_Clause, Withed_Project); + Set_Name_Of (Current_With_Clause, + Name_Of (Withed_Project)); + Name_Len := Imported_Path_Name'Length; + Name_Buffer (1 .. Name_Len) := Imported_Path_Name; + Set_Path_Name_Of (Current_With_Clause, Name_Find); + end if; + end if; + end; + + Scan; + if Token = Tok_Semicolon then + + -- End of (possibly multiple) with clause; + -- Scan past the semicolon. + + Scan; + exit Comma_Loop; + + elsif Token /= Tok_Comma then + Error_Msg ("expected comma or semi colon", Token_Ptr); + exit Comma_Loop; + end if; + end loop Comma_Loop; + end loop With_Loop; + + end Parse_Context_Clause; + + -------------------------- + -- Parse_Single_Project -- + -------------------------- + + procedure Parse_Single_Project + (Project : out Project_Node_Id; + Path_Name : String; + Modified : Boolean) + is + Canonical_Path_Name : Name_Id; + Project_Directory : Name_Id; + Project_Scan_State : Saved_Project_Scan_State; + Source_Index : Source_File_Index; + + Modified_Project : Project_Node_Id := Empty_Node; + + A_Project_Name_And_Node : Tree_Private_Part.Project_Name_And_Node := + Tree_Private_Part.Projects_Htable.Get_First; + + Name_From_Path : constant Name_Id := Project_Name_From (Path_Name); + + use Tree_Private_Part; + + begin + Name_Len := Path_Name'Length; + Name_Buffer (1 .. Name_Len) := Path_Name; + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + Canonical_Path_Name := Name_Find; + + -- Check for a circular dependency + + for Index in 1 .. Project_Stack.Last loop + if Canonical_Path_Name = Project_Stack.Table (Index) then + Error_Msg ("circular dependency detected", Token_Ptr); + Error_Msg_Name_1 := Canonical_Path_Name; + Error_Msg ("\ { is imported by", Token_Ptr); + + for Current in reverse 1 .. Project_Stack.Last loop + Error_Msg_Name_1 := Project_Stack.Table (Current); + + if Error_Msg_Name_1 /= Canonical_Path_Name then + Error_Msg + ("\ { which itself is imported by", Token_Ptr); + + else + Error_Msg ("\ {", Token_Ptr); + exit; + end if; + end loop; + + Project := Empty_Node; + return; + end if; + end loop; + + Project_Stack.Increment_Last; + Project_Stack.Table (Project_Stack.Last) := Canonical_Path_Name; + + -- Check if the project file has already been parsed. + + while + A_Project_Name_And_Node /= Tree_Private_Part.No_Project_Name_And_Node + loop + if + Path_Name_Of (A_Project_Name_And_Node.Node) = Canonical_Path_Name + then + if Modified then + + if A_Project_Name_And_Node.Modified then + Error_Msg + ("cannot modify several times the same project file", + Token_Ptr); + + else + Error_Msg + ("cannot modify an imported project file", + Token_Ptr); + end if; + + elsif A_Project_Name_And_Node.Modified then + Error_Msg + ("cannot imported a modified project file", + Token_Ptr); + end if; + + Project := A_Project_Name_And_Node.Node; + Project_Stack.Decrement_Last; + return; + end if; + + A_Project_Name_And_Node := Tree_Private_Part.Projects_Htable.Get_Next; + end loop; + + -- We never encountered this project file + -- Save the scan state, load the project file and start to scan it. + + Save_Project_Scan_State (Project_Scan_State); + Source_Index := Load_Project_File (Path_Name); + + -- if we cannot find it, we stop + + if Source_Index = No_Source_File then + Project := Empty_Node; + Project_Stack.Decrement_Last; + return; + end if; + + Initialize_Scanner (Types.No_Unit, Source_Index); + + if Name_From_Path = No_Name then + + -- The project file name is not correct (no or bad extension, + -- or not following Ada identifier's syntax). + + Error_Msg_Name_1 := Canonical_Path_Name; + Error_Msg ("?{ is not a valid path name for a project file", + Token_Ptr); + end if; + + if Current_Verbosity >= Medium then + Write_Str ("Parsing """); + Write_Str (Path_Name); + Write_Char ('"'); + Write_Eol; + end if; + + Project_Directory := Immediate_Directory_Of (Canonical_Path_Name); + Project := Default_Project_Node (Of_Kind => N_Project); + Set_Directory_Of (Project, Project_Directory); + Set_Name_Of (Project, Simple_File_Name_Of (Canonical_Path_Name)); + Set_Path_Name_Of (Project, Canonical_Path_Name); + Set_Location_Of (Project, Token_Ptr); + + -- Is there any imported project? + + declare + First_With_Clause : Project_Node_Id := Empty_Node; + + begin + Parse_Context_Clause (Context_Clause => First_With_Clause, + Project_Directory => Project_Directory); + Set_First_With_Clause_Of (Project, First_With_Clause); + end; + + Expect (Tok_Project, "project"); + + -- Scan past "project" + + if Token = Tok_Project then + Set_Location_Of (Project, Token_Ptr); + Scan; + end if; + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + Set_Name_Of (Project, Token_Name); + + Get_Name_String (Token_Name); + Canonical_Case_File_Name (Name_Buffer (1 .. Name_Len)); + + declare + Expected_Name : constant Name_Id := Name_Find; + + begin + if Name_From_Path /= No_Name + and then Expected_Name /= Name_From_Path + then + -- The project name is not the one that was expected from + -- the file name. Report a warning. + + Error_Msg_Name_1 := Expected_Name; + Error_Msg ("?file name does not match unit name, " & + "should be `{" & Project_File_Extension & "`", + Token_Ptr); + end if; + end; + + declare + Project_Name : Name_Id := + Tree_Private_Part.Projects_Htable.Get_First.Name; + + begin + -- Check if we already have a project with this name + + while Project_Name /= No_Name + and then Project_Name /= Token_Name + loop + Project_Name := Tree_Private_Part.Projects_Htable.Get_Next.Name; + end loop; + + if Project_Name /= No_Name then + Error_Msg ("duplicate project name", Token_Ptr); + + else + Tree_Private_Part.Projects_Htable.Set + (K => Token_Name, + E => (Name => Token_Name, + Node => Project, + Modified => Modified)); + end if; + end; + + -- Scan past the project name + + Scan; + + end if; + + if Token = Tok_Modifying then + + -- We are modifying another project + + -- Scan past "modifying" + + Scan; + + Expect (Tok_String_Literal, "literal string"); + + if Token = Tok_String_Literal then + Set_Modified_Project_Path_Of (Project, Strval (Token_Node)); + String_To_Name_Buffer (Modified_Project_Path_Of (Project)); + + declare + Original_Path_Name : constant String := + Name_Buffer (1 .. Name_Len); + + Modified_Project_Path_Name : constant String := + Project_Path_Name_Of + (Original_Path_Name, + Get_Name_String + (Project_Directory)); + + begin + if Modified_Project_Path_Name = "" then + + -- We could not find the project file to modify + + Name_Len := Original_Path_Name'Length; + Name_Buffer (1 .. Name_Len) := Original_Path_Name; + Error_Msg_Name_1 := Name_Find; + + Error_Msg ("unknown project file: {", Token_Ptr); + + else + Parse_Single_Project + (Project => Modified_Project, + Path_Name => Modified_Project_Path_Name, + Modified => True); + end if; + end; + + -- Scan past the modified project path + + Scan; + end if; + end if; + + Expect (Tok_Is, "is"); + + declare + Project_Declaration : Project_Node_Id := Empty_Node; + + begin + -- No need to Scan past "is", Prj.Dect.Parse will do it. + + Prj.Dect.Parse + (Declarations => Project_Declaration, + Current_Project => Project, + Modifying => Modified_Project); + Set_Project_Declaration_Of (Project, Project_Declaration); + end; + + Expect (Tok_End, "end"); + + -- Scan past "end" + + if Token = Tok_End then + Scan; + end if; + + Expect (Tok_Identifier, "identifier"); + + if Token = Tok_Identifier then + + -- We check if this is the project name + + if To_Lower (Get_Name_String (Token_Name)) /= + Get_Name_String (Name_Of (Project)) + then + Error_Msg ("Expected """ & + Get_Name_String (Name_Of (Project)) & """", + Token_Ptr); + end if; + end if; + + if Token /= Tok_Semicolon then + Scan; + end if; + + Expect (Tok_Semicolon, ";"); + + -- Restore the scan state, in case we are not the main project + + Restore_Project_Scan_State (Project_Scan_State); + + Project_Stack.Decrement_Last; + end Parse_Single_Project; + + ------------------ + -- Path_Name_Of -- + ------------------ + + function Path_Name_Of + (File_Name : String; + Directory : String) + return String + is + Result : String_Access; + + begin + Result := Locate_Regular_File (File_Name => File_Name, + Path => Directory); + + if Result = null then + return ""; + + else + Canonical_Case_File_Name (Result.all); + return Result.all; + end if; + end Path_Name_Of; + + ----------------------- + -- Project_Name_From -- + ----------------------- + + function Project_Name_From (Path_Name : String) return Name_Id is + Canonical : String (1 .. Path_Name'Length) := Path_Name; + First : Natural := Canonical'Last; + Last : Positive := First; + + begin + if First = 0 then + return No_Name; + end if; + + Canonical_Case_File_Name (Canonical); + + while First > 0 + and then + Canonical (First) /= '.' + loop + First := First - 1; + end loop; + + if Canonical (First) = '.' then + if Canonical (First .. Last) = Project_File_Extension + and then First /= 1 + then + First := First - 1; + Last := First; + + while First > 0 + and then Canonical (First) /= '/' + and then Canonical (First) /= Dir_Sep + loop + First := First - 1; + end loop; + + else + return No_Name; + end if; + + else + return No_Name; + end if; + + if Canonical (First) = '/' + or else Canonical (First) = Dir_Sep + then + First := First + 1; + end if; + + Name_Len := Last - First + 1; + Name_Buffer (1 .. Name_Len) := To_Lower (Canonical (First .. Last)); + + if not Is_Letter (Name_Buffer (1)) then + return No_Name; + + else + for Index in 2 .. Name_Len - 1 loop + if Name_Buffer (Index) = '_' then + if Name_Buffer (Index + 1) = '_' then + return No_Name; + end if; + + elsif not Is_Alphanumeric (Name_Buffer (Index)) then + return No_Name; + end if; + + end loop; + + if not Is_Alphanumeric (Name_Buffer (Name_Len)) then + return No_Name; + + else + return Name_Find; + end if; + + end if; + end Project_Name_From; + + -------------------------- + -- Project_Path_Name_Of -- + -------------------------- + + function Project_Path_Name_Of + (Project_File_Name : String; + Directory : String) + return String + is + Result : String_Access; + + begin + -- First we try <file_name>.<extension> + + if Current_Verbosity = High then + Write_Str ("Project_Path_Name_Of ("""); + Write_Str (Project_File_Name); + Write_Str (""", """); + Write_Str (Directory); + Write_Line (""");"); + Write_Str (" Trying "); + Write_Str (Project_File_Name); + Write_Line (Project_File_Extension); + end if; + + Result := + Locate_Regular_File + (File_Name => Project_File_Name & Project_File_Extension, + Path => Project_Path.all); + + -- Then we try <file_name> + + if Result = null then + if Current_Verbosity = High then + Write_Str (" Trying "); + Write_Line (Project_File_Name); + end if; + + Result := + Locate_Regular_File + (File_Name => Project_File_Name, + Path => Project_Path.all); + + -- The we try <directory>/<file_name>.<extension> + + if Result = null then + if Current_Verbosity = High then + Write_Str (" Trying "); + Write_Str (Directory); + Write_Str (Project_File_Name); + Write_Line (Project_File_Extension); + end if; + + Result := + Locate_Regular_File + (File_Name => Directory & Project_File_Name & + Project_File_Extension, + Path => Project_Path.all); + + -- Then we try <directory>/<file_name> + + if Result = null then + if Current_Verbosity = High then + Write_Str (" Trying "); + Write_Str (Directory); + Write_Line (Project_File_Name); + end if; + + Result := + Locate_Regular_File + (File_Name => Directory & Project_File_Name, + Path => Project_Path.all); + end if; + end if; + end if; + + -- If we cannot find the project file, we return an empty string + + if Result = null then + return ""; + + else + declare + Final_Result : String + := GNAT.OS_Lib.Normalize_Pathname (Result.all); + begin + Free (Result); + Canonical_Case_File_Name (Final_Result); + return Final_Result; + end; + + end if; + + end Project_Path_Name_Of; + + ------------------------- + -- Simple_File_Name_Of -- + ------------------------- + + function Simple_File_Name_Of (Path_Name : Name_Id) return Name_Id is + begin + Get_Name_String (Path_Name); + + for Index in reverse 1 .. Name_Len loop + if Name_Buffer (Index) = '/' + or else Name_Buffer (Index) = Dir_Sep + then + exit when Index = Name_Len; + Name_Buffer (1 .. Name_Len - Index) := + Name_Buffer (Index + 1 .. Name_Len); + Name_Len := Name_Len - Index; + return Name_Find; + end if; + end loop; + + return No_Name; + + end Simple_File_Name_Of; + +begin + Canonical_Case_File_Name (Project_File_Extension); + + if Prj_Path.all = "" then + Project_Path := new String'("."); + + else + Project_Path := new String'("." & Path_Separator & Prj_Path.all); + end if; + +end Prj.Part; diff --git a/gcc/ada/prj-part.ads b/gcc/ada/prj-part.ads new file mode 100644 index 00000000000..d960b732b35 --- /dev/null +++ b/gcc/ada/prj-part.ads @@ -0,0 +1,46 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . P A R T -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.3 $ +-- -- +-- Copyright (C) 2000-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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- Implements the parsing of project files into a tree. + +with Prj.Tree; use Prj.Tree; + +package Prj.Part is + + procedure Parse + (Project : out Project_Node_Id; + Project_File_Name : String; + Always_Errout_Finalize : Boolean); + -- Parse a project file and all its imported project files + -- and create a tree. + -- Return the node for the project (or Empty_Node if parsing failed). + -- If Always_Errout_Finalize is True, Errout.Finalize is called + -- in all cases; otherwise, Errout.Finalize is only called if there are + -- errors (but not if there are only warnings). + +end Prj.Part; diff --git a/gcc/ada/prj-proc.adb b/gcc/ada/prj-proc.adb new file mode 100644 index 00000000000..4822596f964 --- /dev/null +++ b/gcc/ada/prj-proc.adb @@ -0,0 +1,1371 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . P R O C -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.16 $ +-- -- +-- Copyright (C) 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 Errout; use Errout; +with Namet; use Namet; +with Opt; +with Output; use Output; +with Prj.Attr; use Prj.Attr; +with Prj.Com; use Prj.Com; +with Prj.Ext; use Prj.Ext; +with Prj.Nmsc; use Prj.Nmsc; +with Stringt; use Stringt; + +with GNAT.HTable; + +package body Prj.Proc is + + Error_Report : Put_Line_Access := null; + + package Processed_Projects is new GNAT.HTable.Simple_HTable + (Header_Num => Header_Num, + Element => Project_Id, + No_Element => No_Project, + Key => Name_Id, + Hash => Hash, + Equal => "="); + -- This hash table contains all processed projects + + procedure Add (To_Exp : in out String_Id; Str : String_Id); + -- Concatenate two strings and returns another string if both + -- arguments are not null string. + + procedure Add_Attributes + (Decl : in out Declarations; + First : Attribute_Node_Id); + -- Add all attributes, starting with First, with their default + -- values to the package or project with declarations Decl. + + function Expression + (Project : Project_Id; + From_Project_Node : Project_Node_Id; + Pkg : Package_Id; + First_Term : Project_Node_Id; + Kind : Variable_Kind) + return Variable_Value; + -- From N_Expression project node From_Project_Node, compute the value + -- of an expression and return it as a Variable_Value. + + function Imported_Or_Modified_Project_From + (Project : Project_Id; + With_Name : Name_Id) + return Project_Id; + -- Find an imported or modified project of Project whose name is With_Name. + + function Package_From + (Project : Project_Id; + With_Name : Name_Id) + return Package_Id; + -- Find the package of Project whose name is With_Name. + + procedure Process_Declarative_Items + (Project : Project_Id; + From_Project_Node : Project_Node_Id; + Pkg : Package_Id; + Item : Project_Node_Id); + -- Process declarative items starting with From_Project_Node, and put them + -- in declarations Decl. This is a recursive procedure; it calls itself for + -- a package declaration or a case construction. + + procedure Recursive_Process + (Project : out Project_Id; + From_Project_Node : Project_Node_Id; + Modified_By : Project_Id); + -- Process project with node From_Project_Node in the tree. + -- Do nothing if From_Project_Node is Empty_Node. + -- If project has already been processed, simply return its project id. + -- Otherwise create a new project id, mark it as processed, call itself + -- recursively for all imported projects and a modified project, if any. + -- Then process the declarative items of the project. + + procedure Check (Project : in out Project_Id); + -- Set all projects to not checked, then call Recursive_Check for + -- the main project Project. + -- Project is set to No_Project if errors occurred. + + procedure Recursive_Check (Project : Project_Id); + -- If Project is marked as not checked, mark it as checked, + -- call Check_Naming_Scheme for the project, then call itself + -- for a possible modified project and all the imported projects + -- of Project. + + --------- + -- Add -- + --------- + + procedure Add (To_Exp : in out String_Id; Str : String_Id) is + begin + if To_Exp = Types.No_String or else String_Length (To_Exp) = 0 then + + -- To_Exp is nil or empty. The result is Str. + + To_Exp := Str; + + -- If Str is nil, then do not change To_Ext + + elsif Str /= No_String then + Start_String (To_Exp); + Store_String_Chars (Str); + To_Exp := End_String; + end if; + end Add; + + -------------------- + -- Add_Attributes -- + -------------------- + + procedure Add_Attributes + (Decl : in out Declarations; + First : Attribute_Node_Id) is + The_Attribute : Attribute_Node_Id := First; + Attribute_Data : Attribute_Record; + + begin + while The_Attribute /= Empty_Attribute loop + Attribute_Data := Attributes.Table (The_Attribute); + + if Attribute_Data.Kind_2 /= Associative_Array then + declare + New_Attribute : Variable_Value; + + begin + case Attribute_Data.Kind_1 is + + -- Undefined should not happen + + when Undefined => + pragma Assert + (False, "attribute with an undefined kind"); + raise Program_Error; + + -- Single attributes have a default value of empty string + + when Single => + New_Attribute := + (Kind => Single, + Location => No_Location, + Default => True, + Value => Empty_String); + + -- List attributes have a default value of nil list + + when List => + New_Attribute := + (Kind => List, + Location => No_Location, + Default => True, + Values => Nil_String); + + end case; + + Variable_Elements.Increment_Last; + Variable_Elements.Table (Variable_Elements.Last) := + (Next => Decl.Attributes, + Name => Attribute_Data.Name, + Value => New_Attribute); + Decl.Attributes := Variable_Elements.Last; + end; + end if; + + The_Attribute := Attributes.Table (The_Attribute).Next; + end loop; + + end Add_Attributes; + + ----------- + -- Check -- + ----------- + + procedure Check (Project : in out Project_Id) is + begin + -- Make sure that all projects are marked as not checked. + + for Index in 1 .. Projects.Last loop + Projects.Table (Index).Checked := False; + end loop; + + Recursive_Check (Project); + + if Errout.Errors_Detected > 0 then + Project := No_Project; + end if; + + end Check; + + ---------------- + -- Expression -- + ---------------- + + function Expression + (Project : Project_Id; + From_Project_Node : Project_Node_Id; + Pkg : Package_Id; + First_Term : Project_Node_Id; + Kind : Variable_Kind) + return Variable_Value + is + The_Term : Project_Node_Id := First_Term; + -- The term in the expression list + + The_Current_Term : Project_Node_Id := Empty_Node; + -- The current term node id + + Term_Kind : Variable_Kind; + -- The kind of the current term + + Result : Variable_Value (Kind => Kind); + -- The returned result + + Last : String_List_Id := Nil_String; + -- Reference to the last string elements in Result, when Kind is List. + + begin + Result.Location := Location_Of (From_Project_Node); + + -- Process each term of the expression, starting with First_Term + + while The_Term /= Empty_Node loop + + -- We get the term data and kind ... + + Term_Kind := Expression_Kind_Of (The_Term); + + The_Current_Term := Current_Term (The_Term); + + case Kind_Of (The_Current_Term) is + + when N_Literal_String => + + case Kind is + + when Undefined => + + -- Should never happen + + pragma Assert (False, "Undefined expression kind"); + raise Program_Error; + + when Single => + Add (Result.Value, String_Value_Of (The_Current_Term)); + + when List => + + String_Elements.Increment_Last; + + if Last = Nil_String then + + -- This can happen in an expression such as + -- () & "toto" + + Result.Values := String_Elements.Last; + + else + String_Elements.Table (Last).Next := + String_Elements.Last; + end if; + + Last := String_Elements.Last; + String_Elements.Table (Last) := + (Value => String_Value_Of (The_Current_Term), + Location => Location_Of (The_Current_Term), + Next => Nil_String); + + end case; + + when N_Literal_String_List => + + declare + String_Node : Project_Node_Id := + First_Expression_In_List (The_Current_Term); + + Value : Variable_Value; + + begin + if String_Node /= Empty_Node then + + -- If String_Node is nil, it is an empty list, + -- there is nothing to do + + Value := Expression + (Project => Project, + From_Project_Node => From_Project_Node, + Pkg => Pkg, + First_Term => Tree.First_Term (String_Node), + Kind => Single); + String_Elements.Increment_Last; + + if Result.Values = Nil_String then + + -- This literal string list is the first term + -- in a string list expression + + Result.Values := String_Elements.Last; + + else + String_Elements.Table (Last).Next := + String_Elements.Last; + end if; + + Last := String_Elements.Last; + String_Elements.Table (Last) := + (Value => Value.Value, + Location => Value.Location, + Next => Nil_String); + + loop + -- Add the other element of the literal string list + -- one after the other + + String_Node := + Next_Expression_In_List (String_Node); + + exit when String_Node = Empty_Node; + + Value := + Expression + (Project => Project, + From_Project_Node => From_Project_Node, + Pkg => Pkg, + First_Term => Tree.First_Term (String_Node), + Kind => Single); + + String_Elements.Increment_Last; + String_Elements.Table (Last).Next := + String_Elements.Last; + Last := String_Elements.Last; + String_Elements.Table (Last) := + (Value => Value.Value, + Location => Value.Location, + Next => Nil_String); + end loop; + + end if; + + end; + + when N_Variable_Reference | N_Attribute_Reference => + + declare + The_Project : Project_Id := Project; + The_Package : Package_Id := Pkg; + The_Name : Name_Id := No_Name; + The_Variable_Id : Variable_Id := No_Variable; + The_Variable : Variable; + Term_Project : constant Project_Node_Id := + Project_Node_Of (The_Current_Term); + Term_Package : constant Project_Node_Id := + Package_Node_Of (The_Current_Term); + + begin + if Term_Project /= Empty_Node and then + Term_Project /= From_Project_Node + then + -- This variable or attribute comes from another project + + The_Name := Name_Of (Term_Project); + The_Project := Imported_Or_Modified_Project_From + (Project => Project, With_Name => The_Name); + end if; + + if Term_Package /= Empty_Node then + + -- This is an attribute of a package + + The_Name := Name_Of (Term_Package); + The_Package := Projects.Table (The_Project).Decl.Packages; + + while The_Package /= No_Package + and then Packages.Table (The_Package).Name /= The_Name + loop + The_Package := Packages.Table (The_Package).Next; + end loop; + + pragma Assert + (The_Package /= No_Package, + "package not found."); + + elsif Kind_Of (The_Current_Term) = N_Attribute_Reference then + The_Package := No_Package; + end if; + + The_Name := Name_Of (The_Current_Term); + + if The_Package /= No_Package then + + -- First, if there is a package, look into the package + + if Kind_Of (The_Current_Term) = N_Variable_Reference then + The_Variable_Id := + Packages.Table (The_Package).Decl.Variables; + + else + The_Variable_Id := + Packages.Table (The_Package).Decl.Attributes; + end if; + + while The_Variable_Id /= No_Variable + and then + Variable_Elements.Table (The_Variable_Id).Name /= + The_Name + loop + The_Variable_Id := + Variable_Elements.Table (The_Variable_Id).Next; + end loop; + + end if; + + if The_Variable_Id = No_Variable then + + -- If we have not found it, look into the project + + if Kind_Of (The_Current_Term) = N_Variable_Reference then + The_Variable_Id := + Projects.Table (The_Project).Decl.Variables; + + else + The_Variable_Id := + Projects.Table (The_Project).Decl.Attributes; + end if; + + while The_Variable_Id /= No_Variable + and then + Variable_Elements.Table (The_Variable_Id).Name /= + The_Name + loop + The_Variable_Id := + Variable_Elements.Table (The_Variable_Id).Next; + end loop; + + end if; + + pragma Assert (The_Variable_Id /= No_Variable, + "variable or attribute not found"); + + The_Variable := Variable_Elements.Table (The_Variable_Id); + + case Kind is + + when Undefined => + + -- Should never happen + + pragma Assert (False, "undefined expression kind"); + null; + + when Single => + + case The_Variable.Value.Kind is + + when Undefined => + null; + + when Single => + Add (Result.Value, The_Variable.Value.Value); + + when List => + + -- Should never happen + + pragma Assert + (False, + "list cannot appear in single " & + "string expression"); + null; + + end case; + + when List => + case The_Variable.Value.Kind is + + when Undefined => + null; + + when Single => + String_Elements.Increment_Last; + + if Last = Nil_String then + + -- This can happen in an expression such as + -- () & Var + + Result.Values := String_Elements.Last; + + else + String_Elements.Table (Last).Next := + String_Elements.Last; + end if; + + Last := String_Elements.Last; + String_Elements.Table (Last) := + (Value => The_Variable.Value.Value, + Location => Location_Of (The_Current_Term), + Next => Nil_String); + + when List => + + declare + The_List : String_List_Id := + The_Variable.Value.Values; + + begin + while The_List /= Nil_String loop + String_Elements.Increment_Last; + + if Last = Nil_String then + Result.Values := String_Elements.Last; + + else + String_Elements.Table (Last).Next := + String_Elements.Last; + + end if; + + Last := String_Elements.Last; + String_Elements.Table (Last) := + (Value => + String_Elements.Table + (The_List).Value, + Location => Location_Of + (The_Current_Term), + Next => Nil_String); + The_List := + String_Elements.Table (The_List).Next; + + end loop; + end; + end case; + end case; + end; + + when N_External_Value => + String_To_Name_Buffer + (String_Value_Of (External_Reference_Of (The_Current_Term))); + + declare + Name : constant Name_Id := Name_Find; + Default : String_Id := No_String; + Value : String_Id := No_String; + + Default_Node : constant Project_Node_Id := + External_Default_Of (The_Current_Term); + + begin + if Default_Node /= Empty_Node then + Default := String_Value_Of (Default_Node); + end if; + + Value := Prj.Ext.Value_Of (Name, Default); + + if Value = No_String then + if Error_Report = null then + Error_Msg + ("undefined external reference", + Location_Of (The_Current_Term)); + + else + Error_Report + ("""" & Get_Name_String (Name) & + """ is an undefined external reference"); + end if; + + Value := Empty_String; + + end if; + + case Kind is + + when Undefined => + null; + + when Single => + Add (Result.Value, Value); + + when List => + String_Elements.Increment_Last; + + if Last = Nil_String then + Result.Values := String_Elements.Last; + + else + String_Elements.Table (Last).Next := + String_Elements.Last; + end if; + + Last := String_Elements.Last; + String_Elements.Table (Last) := + (Value => Value, + Location => Location_Of (The_Current_Term), + Next => Nil_String); + + end case; + + end; + + when others => + + -- Should never happen + + pragma Assert + (False, + "illegal node kind in an expression"); + raise Program_Error; + + end case; + + The_Term := Next_Term (The_Term); + + end loop; + return Result; + end Expression; + + --------------------------------------- + -- Imported_Or_Modified_Project_From -- + --------------------------------------- + + function Imported_Or_Modified_Project_From + (Project : Project_Id; + With_Name : Name_Id) + return Project_Id + is + Data : constant Project_Data := Projects.Table (Project); + List : Project_List := Data.Imported_Projects; + + begin + -- First check if it is the name of a modified project + + if Data.Modifies /= No_Project + and then Projects.Table (Data.Modifies).Name = With_Name + then + return Data.Modifies; + + else + -- Then check the name of each imported project + + while List /= Empty_Project_List + and then + Projects.Table + (Project_Lists.Table (List).Project).Name /= With_Name + + loop + List := Project_Lists.Table (List).Next; + end loop; + + pragma Assert + (List /= Empty_Project_List, + "project not found"); + + return Project_Lists.Table (List).Project; + end if; + + end Imported_Or_Modified_Project_From; + + ------------------ + -- Package_From -- + ------------------ + + function Package_From + (Project : Project_Id; + With_Name : Name_Id) + return Package_Id + is + Data : constant Project_Data := Projects.Table (Project); + Result : Package_Id := Data.Decl.Packages; + + begin + -- Check the name of each existing package of Project + + while Result /= No_Package + and then + Packages.Table (Result).Name /= With_Name + loop + Result := Packages.Table (Result).Next; + end loop; + + if Result = No_Package then + -- Should never happen + Write_Line ("package """ & Get_Name_String (With_Name) & + """ not found"); + raise Program_Error; + + else + return Result; + end if; + end Package_From; + + ------------- + -- Process -- + ------------- + + procedure Process + (Project : out Project_Id; + From_Project_Node : Project_Node_Id; + Report_Error : Put_Line_Access) + is + begin + Error_Report := Report_Error; + + -- Make sure there is no projects in the data structure + + Projects.Set_Last (No_Project); + Processed_Projects.Reset; + + -- And process the main project and all of the projects it depends on, + -- recursively + + Recursive_Process + (Project => Project, + From_Project_Node => From_Project_Node, + Modified_By => No_Project); + + if Errout.Errors_Detected > 0 then + Project := No_Project; + end if; + + if Project /= No_Project then + Check (Project); + end if; + + end Process; + + ------------------------------- + -- Process_Declarative_Items -- + ------------------------------- + + procedure Process_Declarative_Items + (Project : Project_Id; + From_Project_Node : Project_Node_Id; + Pkg : Package_Id; + Item : Project_Node_Id) is + + Current_Declarative_Item : Project_Node_Id := Item; + + Current_Item : Project_Node_Id := Empty_Node; + + begin + -- For each declarative item + + while Current_Declarative_Item /= Empty_Node loop + + -- Get its data + + Current_Item := Current_Item_Node (Current_Declarative_Item); + + -- And set Current_Declarative_Item to the next declarative item + -- ready for the next iteration + + Current_Declarative_Item := Next_Declarative_Item + (Current_Declarative_Item); + + case Kind_Of (Current_Item) is + + when N_Package_Declaration => + Packages.Increment_Last; + + declare + New_Pkg : constant Package_Id := Packages.Last; + The_New_Package : Package_Element; + + Project_Of_Renamed_Package : constant Project_Node_Id := + Project_Of_Renamed_Package_Of + (Current_Item); + + begin + The_New_Package.Name := Name_Of (Current_Item); + + if Pkg /= No_Package then + The_New_Package.Next := + Packages.Table (Pkg).Decl.Packages; + Packages.Table (Pkg).Decl.Packages := New_Pkg; + else + The_New_Package.Next := + Projects.Table (Project).Decl.Packages; + Projects.Table (Project).Decl.Packages := New_Pkg; + end if; + + Packages.Table (New_Pkg) := The_New_Package; + + if Project_Of_Renamed_Package /= Empty_Node then + + -- Renamed package + + declare + Project_Name : constant Name_Id := + Name_Of + (Project_Of_Renamed_Package); + + Renamed_Project : constant Project_Id := + Imported_Or_Modified_Project_From + (Project, Project_Name); + + Renamed_Package : constant Package_Id := + Package_From + (Renamed_Project, + Name_Of (Current_Item)); + + begin + Packages.Table (New_Pkg).Decl := + Packages.Table (Renamed_Package).Decl; + end; + + else + -- Set the default values of the attributes + + Add_Attributes + (Packages.Table (New_Pkg).Decl, + Package_Attributes.Table + (Package_Id_Of (Current_Item)).First_Attribute); + + Process_Declarative_Items + (Project => Project, + From_Project_Node => From_Project_Node, + Pkg => New_Pkg, + Item => First_Declarative_Item_Of + (Current_Item)); + end if; + + end; + + when N_String_Type_Declaration => + + -- There is nothing to process + + null; + + when N_Attribute_Declaration | + N_Typed_Variable_Declaration | + N_Variable_Declaration => + + pragma Assert (Expression_Of (Current_Item) /= Empty_Node, + "no expression for an object declaration"); + + declare + New_Value : constant Variable_Value := + Expression + (Project => Project, + From_Project_Node => From_Project_Node, + Pkg => Pkg, + First_Term => + Tree.First_Term (Expression_Of + (Current_Item)), + Kind => + Expression_Kind_Of (Current_Item)); + + The_Variable : Variable_Id := No_Variable; + + Current_Item_Name : constant Name_Id := + Name_Of (Current_Item); + + begin + if Kind_Of (Current_Item) = N_Typed_Variable_Declaration then + + if String_Equal (New_Value.Value, Empty_String) then + Error_Msg_Name_1 := Name_Of (Current_Item); + + if Error_Report = null then + Error_Msg + ("no value defined for %", + Location_Of (Current_Item)); + + else + Error_Report + ("no value defined for " & + Get_Name_String (Error_Msg_Name_1)); + end if; + + else + declare + Current_String : Project_Node_Id := + First_Literal_String + (String_Type_Of + (Current_Item)); + + begin + while Current_String /= Empty_Node + and then not String_Equal + (String_Value_Of (Current_String), + New_Value.Value) + loop + Current_String := + Next_Literal_String (Current_String); + end loop; + + if Current_String = Empty_Node then + String_To_Name_Buffer (New_Value.Value); + Error_Msg_Name_1 := Name_Find; + Error_Msg_Name_2 := Name_Of (Current_Item); + + if Error_Report = null then + Error_Msg + ("value { is illegal for typed string %", + Location_Of (Current_Item)); + + else + Error_Report + ("value """ & + Get_Name_String (Error_Msg_Name_1) & + """ is illegal for typed string """ & + Get_Name_String (Error_Msg_Name_2) & + """"); + end if; + end if; + end; + end if; + end if; + + if Kind_Of (Current_Item) /= N_Attribute_Declaration + or else + Associative_Array_Index_Of (Current_Item) = No_String + then + -- Usual case + + -- Code below really needs more comments ??? + + if Kind_Of (Current_Item) = N_Attribute_Declaration then + if Pkg /= No_Package then + The_Variable := + Packages.Table (Pkg).Decl.Attributes; + + else + The_Variable := + Projects.Table (Project).Decl.Attributes; + end if; + + else + if Pkg /= No_Package then + The_Variable := + Packages.Table (Pkg).Decl.Variables; + + else + The_Variable := + Projects.Table (Project).Decl.Variables; + end if; + + end if; + + while + The_Variable /= No_Variable + and then + Variable_Elements.Table (The_Variable).Name /= + Current_Item_Name + loop + The_Variable := + Variable_Elements.Table (The_Variable).Next; + end loop; + + if The_Variable = No_Variable then + pragma Assert + (Kind_Of (Current_Item) /= N_Attribute_Declaration, + "illegal attribute declaration"); + + Variable_Elements.Increment_Last; + The_Variable := Variable_Elements.Last; + + if Pkg /= No_Package then + Variable_Elements.Table (The_Variable) := + (Next => + Packages.Table (Pkg).Decl.Variables, + Name => Current_Item_Name, + Value => New_Value); + Packages.Table (Pkg).Decl.Variables := The_Variable; + + else + Variable_Elements.Table (The_Variable) := + (Next => + Projects.Table (Project).Decl.Variables, + Name => Current_Item_Name, + Value => New_Value); + Projects.Table (Project).Decl.Variables := + The_Variable; + end if; + + else + Variable_Elements.Table (The_Variable).Value := + New_Value; + + end if; + + else + -- Associative array attribute + + String_To_Name_Buffer + (Associative_Array_Index_Of (Current_Item)); + + declare + The_Array : Array_Id; + + The_Array_Element : Array_Element_Id := + No_Array_Element; + + Index_Name : constant Name_Id := Name_Find; + + begin + + if Pkg /= No_Package then + The_Array := Packages.Table (Pkg).Decl.Arrays; + + else + The_Array := Projects.Table (Project).Decl.Arrays; + end if; + + while + The_Array /= No_Array + and then Arrays.Table (The_Array).Name /= + Current_Item_Name + loop + The_Array := Arrays.Table (The_Array).Next; + end loop; + + if The_Array = No_Array then + Arrays.Increment_Last; + The_Array := Arrays.Last; + + if Pkg /= No_Package then + Arrays.Table (The_Array) := + (Name => Current_Item_Name, + Value => No_Array_Element, + Next => Packages.Table (Pkg).Decl.Arrays); + Packages.Table (Pkg).Decl.Arrays := The_Array; + + else + Arrays.Table (The_Array) := + (Name => Current_Item_Name, + Value => No_Array_Element, + Next => + Projects.Table (Project).Decl.Arrays); + Projects.Table (Project).Decl.Arrays := + The_Array; + end if; + + else + The_Array_Element := Arrays.Table (The_Array).Value; + end if; + + while The_Array_Element /= No_Array_Element + and then + Array_Elements.Table (The_Array_Element).Index /= + Index_Name + loop + The_Array_Element := + Array_Elements.Table (The_Array_Element).Next; + end loop; + + if The_Array_Element = No_Array_Element then + Array_Elements.Increment_Last; + The_Array_Element := Array_Elements.Last; + Array_Elements.Table (The_Array_Element) := + (Index => Index_Name, + Value => New_Value, + Next => Arrays.Table (The_Array).Value); + Arrays.Table (The_Array).Value := The_Array_Element; + + else + Array_Elements.Table (The_Array_Element).Value := + New_Value; + end if; + end; + end if; + end; + + when N_Case_Construction => + declare + The_Project : Project_Id := Project; + The_Package : Package_Id := Pkg; + The_Variable : Variable_Value := Nil_Variable_Value; + Case_Value : String_Id := No_String; + Case_Item : Project_Node_Id := Empty_Node; + Choice_String : Project_Node_Id := Empty_Node; + Decl_Item : Project_Node_Id := Empty_Node; + + begin + declare + Variable_Node : constant Project_Node_Id := + Case_Variable_Reference_Of + (Current_Item); + + Var_Id : Variable_Id := No_Variable; + Name : Name_Id := No_Name; + + begin + if Project_Node_Of (Variable_Node) /= Empty_Node then + Name := Name_Of (Project_Node_Of (Variable_Node)); + The_Project := + Imported_Or_Modified_Project_From (Project, Name); + end if; + + if Package_Node_Of (Variable_Node) /= Empty_Node then + Name := Name_Of (Package_Node_Of (Variable_Node)); + The_Package := Package_From (The_Project, Name); + end if; + + Name := Name_Of (Variable_Node); + + if The_Package /= No_Package then + Var_Id := Packages.Table (The_Package).Decl.Variables; + Name := Name_Of (Variable_Node); + while Var_Id /= No_Variable + and then + Variable_Elements.Table (Var_Id).Name /= Name + loop + Var_Id := Variable_Elements.Table (Var_Id).Next; + end loop; + end if; + + if Var_Id = No_Variable + and then Package_Node_Of (Variable_Node) = Empty_Node + then + Var_Id := Projects.Table (The_Project).Decl.Variables; + while Var_Id /= No_Variable + and then + Variable_Elements.Table (Var_Id).Name /= Name + loop + Var_Id := Variable_Elements.Table (Var_Id).Next; + end loop; + end if; + + if Var_Id = No_Variable then + + -- Should never happen + + Write_Line ("variable """ & + Get_Name_String (Name) & + """ not found"); + raise Program_Error; + end if; + + The_Variable := Variable_Elements.Table (Var_Id).Value; + + if The_Variable.Kind /= Single then + + -- Should never happen + + Write_Line ("variable""" & + Get_Name_String (Name) & + """ is not a single string variable"); + raise Program_Error; + end if; + + Case_Value := The_Variable.Value; + end; + + Case_Item := First_Case_Item_Of (Current_Item); + Case_Item_Loop : + while Case_Item /= Empty_Node loop + Choice_String := First_Choice_Of (Case_Item); + + if Choice_String = Empty_Node then + Decl_Item := First_Declarative_Item_Of (Case_Item); + exit Case_Item_Loop; + end if; + + Choice_Loop : + while Choice_String /= Empty_Node loop + if String_Equal (Case_Value, + String_Value_Of (Choice_String)) + then + Decl_Item := + First_Declarative_Item_Of (Case_Item); + exit Case_Item_Loop; + end if; + + Choice_String := + Next_Literal_String (Choice_String); + end loop Choice_Loop; + Case_Item := Next_Case_Item (Case_Item); + end loop Case_Item_Loop; + + if Decl_Item /= Empty_Node then + Process_Declarative_Items + (Project => Project, + From_Project_Node => From_Project_Node, + Pkg => Pkg, + Item => Decl_Item); + end if; + end; + + when others => + + -- Should never happen + + Write_Line ("Illegal declarative item: " & + Project_Node_Kind'Image (Kind_Of (Current_Item))); + raise Program_Error; + end case; + end loop; + end Process_Declarative_Items; + + --------------------- + -- Recursive_Check -- + --------------------- + + procedure Recursive_Check (Project : Project_Id) is + Data : Project_Data; + Imported_Project_List : Project_List := Empty_Project_List; + + begin + -- Do nothing if Project is No_Project, or Project has already + -- been marked as checked. + + if Project /= No_Project + and then not Projects.Table (Project).Checked + then + Data := Projects.Table (Project); + + -- Call itself for a possible modified project. + -- (if there is no modified project, then nothing happens). + + Recursive_Check (Data.Modifies); + + -- Call itself for all imported projects + + Imported_Project_List := Data.Imported_Projects; + while Imported_Project_List /= Empty_Project_List loop + Recursive_Check + (Project_Lists.Table (Imported_Project_List).Project); + Imported_Project_List := + Project_Lists.Table (Imported_Project_List).Next; + end loop; + + -- Mark project as checked + + Projects.Table (Project).Checked := True; + + if Opt.Verbose_Mode then + Write_Str ("Checking project file """); + Write_Str (Get_Name_String (Data.Name)); + Write_Line (""""); + end if; + + Prj.Nmsc.Check_Naming_Scheme (Project, Error_Report); + end if; + + end Recursive_Check; + + ----------------------- + -- Recursive_Process -- + ----------------------- + + procedure Recursive_Process + (Project : out Project_Id; + From_Project_Node : Project_Node_Id; + Modified_By : Project_Id) + is + With_Clause : Project_Node_Id; + + begin + if From_Project_Node = Empty_Node then + Project := No_Project; + + else + declare + Processed_Data : Project_Data := Empty_Project; + Imported : Project_List := Empty_Project_List; + Declaration_Node : Project_Node_Id := Empty_Node; + Name : constant Name_Id := + Name_Of (From_Project_Node); + + begin + Project := Processed_Projects.Get (Name); + + if Project /= No_Project then + return; + end if; + + Projects.Increment_Last; + Project := Projects.Last; + Processed_Projects.Set (Name, Project); + Processed_Data.Name := Name; + Processed_Data.Path_Name := Path_Name_Of (From_Project_Node); + Processed_Data.Location := Location_Of (From_Project_Node); + Processed_Data.Directory := Directory_Of (From_Project_Node); + Processed_Data.Modified_By := Modified_By; + Add_Attributes (Processed_Data.Decl, Attribute_First); + With_Clause := First_With_Clause_Of (From_Project_Node); + + while With_Clause /= Empty_Node loop + declare + New_Project : Project_Id; + New_Data : Project_Data; + + begin + Recursive_Process + (Project => New_Project, + From_Project_Node => Project_Node_Of (With_Clause), + Modified_By => No_Project); + New_Data := Projects.Table (New_Project); + + -- If we were the first project to import it, + -- set First_Referred_By to us. + + if New_Data.First_Referred_By = No_Project then + New_Data.First_Referred_By := Project; + Projects.Table (New_Project) := New_Data; + end if; + + -- Add this project to our list of imported projects + + Project_Lists.Increment_Last; + Project_Lists.Table (Project_Lists.Last) := + (Project => New_Project, Next => Empty_Project_List); + + -- Imported is the id of the last imported project. + -- If it is nil, then this imported project is our first. + + if Imported = Empty_Project_List then + Processed_Data.Imported_Projects := Project_Lists.Last; + + else + Project_Lists.Table (Imported).Next := Project_Lists.Last; + end if; + + Imported := Project_Lists.Last; + + With_Clause := Next_With_Clause_Of (With_Clause); + end; + end loop; + + Declaration_Node := Project_Declaration_Of (From_Project_Node); + + Recursive_Process + (Project => Processed_Data.Modifies, + From_Project_Node => Modified_Project_Of (Declaration_Node), + Modified_By => Project); + + Projects.Table (Project) := Processed_Data; + + Process_Declarative_Items + (Project => Project, + From_Project_Node => From_Project_Node, + Pkg => No_Package, + Item => First_Declarative_Item_Of + (Declaration_Node)); + + end; + end if; + end Recursive_Process; + +end Prj.Proc; diff --git a/gcc/ada/prj-proc.ads b/gcc/ada/prj-proc.ads new file mode 100644 index 00000000000..63259a42699 --- /dev/null +++ b/gcc/ada/prj-proc.ads @@ -0,0 +1,45 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . P R O C -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.2 $ +-- -- +-- Copyright (C) 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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- This package is used to convert a project file tree (see prj-tree.ads) to +-- project file data structures (see prj.ads), taking into account +-- the environment (external references). + +with Prj.Tree; use Prj.Tree; + +package Prj.Proc is + + procedure Process + (Project : out Project_Id; + From_Project_Node : Project_Node_Id; + Report_Error : Put_Line_Access); + -- Process a project file tree into project file data structures. + -- If Report_Error is null, use the standard error reporting mechanism + -- (Errout). Otherwise, report errors using Report_Error. + +end Prj.Proc; diff --git a/gcc/ada/prj-strt.adb b/gcc/ada/prj-strt.adb new file mode 100644 index 00000000000..790c632c2cf --- /dev/null +++ b/gcc/ada/prj-strt.adb @@ -0,0 +1,943 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . S T R T -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.12 $ +-- -- +-- Copyright (C) 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 Errout; use Errout; +with Prj.Attr; use Prj.Attr; +with Prj.Tree; use Prj.Tree; +with Scans; use Scans; +with Sinfo; use Sinfo; +with Stringt; use Stringt; +with Table; +with Types; use Types; + +package body Prj.Strt is + + Initial_Size : constant := 8; + + type Name_Location is record + Name : Name_Id := No_Name; + Location : Source_Ptr := No_Location; + end record; + -- Store the identifier and the location of a simple name + + type Name_Range is range 0 .. 3; + subtype Name_Index is Name_Range range 1 .. Name_Range'Last; + -- A Name may contain up to 3 simple names + + type Names is array (Name_Index) of Name_Location; + -- Used to store 1 to 3 simple_names. 2 simple names are for + -- <project>.<package>, <project>.<variable> or <package>.<variable>. + -- 3 simple names are for <project>.<package>.<variable>. + + type Choice_String is record + The_String : String_Id; + Already_Used : Boolean := False; + end record; + -- The string of a case label, and an indication that it has already + -- been used (to avoid duplicate case labels). + + Choices_Initial : constant := 10; + Choices_Increment : constant := 10; + + Choice_Node_Low_Bound : constant := 0; + Choice_Node_High_Bound : constant := 099_999_999; -- In practice, infinite + + type Choice_Node_Id is + range Choice_Node_Low_Bound .. Choice_Node_High_Bound; + + First_Choice_Node_Id : constant Choice_Node_Id := + Choice_Node_Low_Bound; + + Empty_Choice : constant Choice_Node_Id := + Choice_Node_Low_Bound; + + First_Choice_Id : constant Choice_Node_Id := First_Choice_Node_Id + 1; + + package Choices is + new Table.Table (Table_Component_Type => Choice_String, + Table_Index_Type => Choice_Node_Id, + Table_Low_Bound => First_Choice_Node_Id, + Table_Initial => Choices_Initial, + Table_Increment => Choices_Increment, + Table_Name => "Prj.Strt.Choices"); + -- Used to store the case labels and check that there is no duplicate. + + package Choice_Lasts is + new Table.Table (Table_Component_Type => Choice_Node_Id, + Table_Index_Type => Nat, + Table_Low_Bound => 1, + Table_Initial => 3, + Table_Increment => 3, + Table_Name => "Prj.Strt.Choice_Lasts"); + -- Used to store the indices of the choices in table Choices, + -- to distinguish nested case constructions. + + Choice_First : Choice_Node_Id := 0; + -- Index in table Choices of the first case label of the current + -- case construction. + -- 0 means no current case construction. + + procedure Add (This_String : String_Id); + -- Add a string to the case label list, indicating that it has not + -- yet been used. + + procedure External_Reference (External_Value : out Project_Node_Id); + -- Parse an external reference. Current token is "external". + + procedure Attribute_Reference + (Reference : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Parse an attribute reference. Current token is an apostrophe. + + procedure Terms + (Term : out Project_Node_Id; + Expr_Kind : in out Variable_Kind; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Recursive procedure to parse one term or several terms concatenated + -- using "&". + + --------- + -- Add -- + --------- + + procedure Add (This_String : String_Id) is + begin + Choices.Increment_Last; + Choices.Table (Choices.Last) := + (The_String => This_String, + Already_Used => False); + end Add; + + ------------------------- + -- Attribute_Reference -- + ------------------------- + + procedure Attribute_Reference + (Reference : out Project_Node_Id; + First_Attribute : Attribute_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + Current_Attribute : Attribute_Node_Id := First_Attribute; + + begin + Reference := Default_Project_Node (Of_Kind => N_Attribute_Reference); + Set_Location_Of (Reference, To => Token_Ptr); + Scan; -- past apostrophe + Expect (Tok_Identifier, "Identifier"); + + if Token = Tok_Identifier then + Set_Name_Of (Reference, To => Token_Name); + + while Current_Attribute /= Empty_Attribute + and then + Attributes.Table (Current_Attribute).Name /= Token_Name + loop + Current_Attribute := Attributes.Table (Current_Attribute).Next; + end loop; + + if Current_Attribute = Empty_Attribute then + Error_Msg ("unknown attribute", Token_Ptr); + Reference := Empty_Node; + + elsif + Attributes.Table (Current_Attribute).Kind_2 = Associative_Array + then + Error_Msg + ("associative array attribute cannot be referenced", + Token_Ptr); + Reference := Empty_Node; + + else + Set_Project_Node_Of (Reference, To => Current_Project); + Set_Package_Node_Of (Reference, To => Current_Package); + Set_Expression_Kind_Of + (Reference, To => Attributes.Table (Current_Attribute).Kind_1); + Scan; + end if; + end if; + end Attribute_Reference; + + --------------------------- + -- End_Case_Construction -- + --------------------------- + + procedure End_Case_Construction is + begin + if Choice_Lasts.Last = 1 then + Choice_Lasts.Set_Last (0); + Choices.Set_Last (First_Choice_Node_Id); + Choice_First := 0; + + elsif Choice_Lasts.Last = 2 then + Choice_Lasts.Set_Last (1); + Choices.Set_Last (Choice_Lasts.Table (1)); + Choice_First := 1; + + else + Choice_Lasts.Decrement_Last; + Choices.Set_Last (Choice_Lasts.Table (Choice_Lasts.Last)); + Choice_First := Choice_Lasts.Table (Choice_Lasts.Last - 1) + 1; + end if; + end End_Case_Construction; + + ------------------------ + -- External_Reference -- + ------------------------ + + procedure External_Reference (External_Value : out Project_Node_Id) is + Field_Id : Project_Node_Id := Empty_Node; + + begin + External_Value := + Default_Project_Node (Of_Kind => N_External_Value, + And_Expr_Kind => Single); + Set_Location_Of (External_Value, To => Token_Ptr); + + -- The current token is External + + -- Get the left parenthesis + + Scan; + Expect (Tok_Left_Paren, "("); + + -- Scan past the left parenthesis + + if Token = Tok_Left_Paren then + Scan; + end if; + + -- Get the name of the external reference + + Expect (Tok_String_Literal, "literal string"); + + if Token = Tok_String_Literal then + Field_Id := + Default_Project_Node (Of_Kind => N_Literal_String, + And_Expr_Kind => Single); + Set_String_Value_Of (Field_Id, To => Strval (Token_Node)); + Set_External_Reference_Of (External_Value, To => Field_Id); + + -- Scan past the first argument + + Scan; + + case Token is + + when Tok_Right_Paren => + + -- Scan past the right parenthesis + Scan; + + when Tok_Comma => + + -- Scan past the comma + + Scan; + + Expect (Tok_String_Literal, "literal string"); + + -- Get the default + + if Token = Tok_String_Literal then + Field_Id := + Default_Project_Node (Of_Kind => N_Literal_String, + And_Expr_Kind => Single); + Set_String_Value_Of (Field_Id, To => Strval (Token_Node)); + Set_External_Default_Of (External_Value, To => Field_Id); + Scan; + Expect (Tok_Right_Paren, ")"); + end if; + + -- Scan past the right parenthesis + if Token = Tok_Right_Paren then + Scan; + end if; + + when others => + Error_Msg ("',' or ')' expected", Token_Ptr); + end case; + end if; + end External_Reference; + + ----------------------- + -- Parse_Choice_List -- + ----------------------- + + procedure Parse_Choice_List (First_Choice : out Project_Node_Id) is + Current_Choice : Project_Node_Id := Empty_Node; + Next_Choice : Project_Node_Id := Empty_Node; + Choice_String : String_Id := No_String; + Found : Boolean := False; + + begin + First_Choice := + Default_Project_Node (Of_Kind => N_Literal_String, + And_Expr_Kind => Single); + Current_Choice := First_Choice; + + loop + Expect (Tok_String_Literal, "literal string"); + exit when Token /= Tok_String_Literal; + Set_Location_Of (Current_Choice, To => Token_Ptr); + Choice_String := Strval (Token_Node); + Set_String_Value_Of (Current_Choice, To => Choice_String); + + Found := False; + for Choice in Choice_First .. Choices.Last loop + if String_Equal (Choices.Table (Choice).The_String, + Choice_String) + then + Found := True; + + if Choices.Table (Choice).Already_Used then + Error_Msg ("duplicate case label", Token_Ptr); + else + Choices.Table (Choice).Already_Used := True; + end if; + + exit; + end if; + end loop; + + if not Found then + Error_Msg ("illegal case label", Token_Ptr); + end if; + + Scan; + + if Token = Tok_Vertical_Bar then + Next_Choice := + Default_Project_Node (Of_Kind => N_Literal_String, + And_Expr_Kind => Single); + Set_Next_Literal_String (Current_Choice, To => Next_Choice); + Current_Choice := Next_Choice; + Scan; + else + exit; + end if; + end loop; + end Parse_Choice_List; + + ---------------------- + -- Parse_Expression -- + ---------------------- + + procedure Parse_Expression + (Expression : out Project_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + First_Term : Project_Node_Id := Empty_Node; + Expression_Kind : Variable_Kind := Undefined; + + begin + Expression := Default_Project_Node (Of_Kind => N_Expression); + Set_Location_Of (Expression, To => Token_Ptr); + Terms (Term => First_Term, + Expr_Kind => Expression_Kind, + Current_Project => Current_Project, + Current_Package => Current_Package); + Set_First_Term (Expression, To => First_Term); + Set_Expression_Kind_Of (Expression, To => Expression_Kind); + end Parse_Expression; + + ---------------------------- + -- Parse_String_Type_List -- + ---------------------------- + + procedure Parse_String_Type_List (First_String : out Project_Node_Id) is + Last_String : Project_Node_Id := Empty_Node; + Next_String : Project_Node_Id := Empty_Node; + String_Value : String_Id := No_String; + + begin + First_String := + Default_Project_Node (Of_Kind => N_Literal_String, + And_Expr_Kind => Single); + Last_String := First_String; + + loop + Expect (Tok_String_Literal, "literal string"); + exit when Token /= Tok_String_Literal; + String_Value := Strval (Token_Node); + Set_String_Value_Of (Last_String, To => String_Value); + Set_Location_Of (Last_String, To => Token_Ptr); + + declare + Current : Project_Node_Id := First_String; + + begin + while Current /= Last_String loop + if String_Equal (String_Value_Of (Current), String_Value) then + Error_Msg ("duplicate value in type", Token_Ptr); + exit; + end if; + + Current := Next_Literal_String (Current); + end loop; + end; + + Scan; + + if Token /= Tok_Comma then + exit; + + else + Next_String := + Default_Project_Node (Of_Kind => N_Literal_String, + And_Expr_Kind => Single); + Set_Next_Literal_String (Last_String, To => Next_String); + Last_String := Next_String; + Scan; + end if; + end loop; + end Parse_String_Type_List; + + ------------------------------ + -- Parse_Variable_Reference -- + ------------------------------ + + procedure Parse_Variable_Reference + (Variable : out Project_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + The_Names : Names; + Last_Name : Name_Range := 0; + Current_Variable : Project_Node_Id := Empty_Node; + + The_Package : Project_Node_Id := Current_Package; + The_Project : Project_Node_Id := Current_Project; + + Specified_Project : Project_Node_Id := Empty_Node; + Specified_Package : Project_Node_Id := Empty_Node; + Look_For_Variable : Boolean := True; + First_Attribute : Attribute_Node_Id := Empty_Attribute; + Variable_Name : Name_Id; + + begin + for Index in The_Names'Range loop + Expect (Tok_Identifier, "identifier"); + + if Token /= Tok_Identifier then + Look_For_Variable := False; + exit; + end if; + + Last_Name := Last_Name + 1; + The_Names (Last_Name) := + (Name => Token_Name, + Location => Token_Ptr); + Scan; + exit when Token /= Tok_Dot; + Scan; + end loop; + + if Look_For_Variable then + if Token = Tok_Apostrophe then + + -- Attribute reference + + case Last_Name is + when 0 => + + -- Cannot happen + + null; + + when 1 => + for Index in Package_First .. Package_Attributes.Last loop + if Package_Attributes.Table (Index).Name = + The_Names (1).Name + then + First_Attribute := + Package_Attributes.Table (Index).First_Attribute; + exit; + end if; + end loop; + + if First_Attribute /= Empty_Attribute then + The_Package := First_Package_Of (Current_Project); + while The_Package /= Empty_Node + and then Name_Of (The_Package) /= The_Names (1).Name + loop + The_Package := Next_Package_In_Project (The_Package); + end loop; + + if The_Package = Empty_Node then + Error_Msg ("package not yet defined", + The_Names (1).Location); + end if; + + else + First_Attribute := Attribute_First; + The_Package := Empty_Node; + + declare + The_Project_Name_And_Node : + constant Tree_Private_Part.Project_Name_And_Node := + Tree_Private_Part.Projects_Htable.Get + (The_Names (1).Name); + + use Tree_Private_Part; + + begin + if The_Project_Name_And_Node = + Tree_Private_Part.No_Project_Name_And_Node + then + Error_Msg ("unknown project", + The_Names (1).Location); + else + The_Project := The_Project_Name_And_Node.Node; + end if; + end; + end if; + + when 2 => + declare + With_Clause : Project_Node_Id := + First_With_Clause_Of (Current_Project); + + begin + while With_Clause /= Empty_Node loop + The_Project := Project_Node_Of (With_Clause); + exit when Name_Of (The_Project) = The_Names (1).Name; + With_Clause := Next_With_Clause_Of (With_Clause); + end loop; + + if With_Clause = Empty_Node then + Error_Msg ("unknown project", + The_Names (1).Location); + The_Project := Empty_Node; + The_Package := Empty_Node; + First_Attribute := Attribute_First; + + else + The_Package := First_Package_Of (The_Project); + while The_Package /= Empty_Node + and then Name_Of (The_Package) /= The_Names (2).Name + loop + The_Package := + Next_Package_In_Project (The_Package); + end loop; + + if The_Package = Empty_Node then + Error_Msg ("package not declared in project", + The_Names (2).Location); + First_Attribute := Attribute_First; + + else + First_Attribute := + Package_Attributes.Table + (Package_Id_Of (The_Package)).First_Attribute; + end if; + end if; + end; + + when 3 => + Error_Msg + ("too many single names for an attribute reference", + The_Names (1).Location); + Scan; + Variable := Empty_Node; + return; + end case; + + Attribute_Reference + (Variable, + Current_Project => The_Project, + Current_Package => The_Package, + First_Attribute => First_Attribute); + return; + end if; + end if; + + Variable := + Default_Project_Node (Of_Kind => N_Variable_Reference); + + if Look_For_Variable then + case Last_Name is + when 0 => + + -- Cannot happen + + null; + + when 1 => + Set_Name_Of (Variable, To => The_Names (1).Name); + + -- Header comment needed ??? + + when 2 => + Set_Name_Of (Variable, To => The_Names (2).Name); + The_Package := First_Package_Of (Current_Project); + + while The_Package /= Empty_Node + and then Name_Of (The_Package) /= The_Names (1).Name + loop + The_Package := Next_Package_In_Project (The_Package); + end loop; + + if The_Package /= Empty_Node then + Specified_Package := The_Package; + The_Project := Empty_Node; + + else + declare + With_Clause : Project_Node_Id := + First_With_Clause_Of (Current_Project); + + begin + while With_Clause /= Empty_Node loop + The_Project := Project_Node_Of (With_Clause); + exit when Name_Of (The_Project) = The_Names (1).Name; + With_Clause := Next_With_Clause_Of (With_Clause); + end loop; + + if With_Clause = Empty_Node then + The_Project := + Modified_Project_Of + (Project_Declaration_Of (Current_Project)); + + if The_Project /= Empty_Node + and then + Name_Of (The_Project) /= The_Names (1).Name + then + The_Project := Empty_Node; + end if; + end if; + + if The_Project = Empty_Node then + Error_Msg ("unknown package or project", + The_Names (1).Location); + Look_For_Variable := False; + else + Specified_Project := The_Project; + end if; + end; + end if; + + -- Header comment needed ??? + + when 3 => + Set_Name_Of (Variable, To => The_Names (3).Name); + + declare + With_Clause : Project_Node_Id := + First_With_Clause_Of (Current_Project); + + begin + while With_Clause /= Empty_Node loop + The_Project := Project_Node_Of (With_Clause); + exit when Name_Of (The_Project) = The_Names (1).Name; + With_Clause := Next_With_Clause_Of (With_Clause); + end loop; + + if With_Clause = Empty_Node then + The_Project := + Modified_Project_Of + (Project_Declaration_Of (Current_Project)); + + if The_Project /= Empty_Node + and then Name_Of (The_Project) /= The_Names (1).Name + then + The_Project := Empty_Node; + end if; + end if; + + if The_Project = Empty_Node then + Error_Msg ("unknown package or project", + The_Names (1).Location); + Look_For_Variable := False; + + else + Specified_Project := The_Project; + The_Package := First_Package_Of (The_Project); + + while The_Package /= Empty_Node + and then Name_Of (The_Package) /= The_Names (2).Name + loop + The_Package := Next_Package_In_Project (The_Package); + end loop; + + if The_Package = Empty_Node then + Error_Msg ("unknown package", + The_Names (2).Location); + Look_For_Variable := False; + + else + Specified_Package := The_Package; + The_Project := Empty_Node; + end if; + end if; + end; + + end case; + end if; + + if Look_For_Variable then + Variable_Name := Name_Of (Variable); + Set_Project_Node_Of (Variable, To => Specified_Project); + Set_Package_Node_Of (Variable, To => Specified_Package); + + if The_Package /= Empty_Node then + Current_Variable := First_Variable_Of (The_Package); + + while Current_Variable /= Empty_Node + and then + Name_Of (Current_Variable) /= Variable_Name + loop + Current_Variable := Next_Variable (Current_Variable); + end loop; + end if; + + if Current_Variable = Empty_Node + and then The_Project /= Empty_Node + then + Current_Variable := First_Variable_Of (The_Project); + while Current_Variable /= Empty_Node + and then Name_Of (Current_Variable) /= Variable_Name + loop + Current_Variable := Next_Variable (Current_Variable); + end loop; + end if; + + if Current_Variable = Empty_Node then + Error_Msg ("unknown variable", The_Names (Last_Name).Location); + end if; + end if; + + if Current_Variable /= Empty_Node then + Set_Expression_Kind_Of + (Variable, To => Expression_Kind_Of (Current_Variable)); + + if Kind_Of (Current_Variable) = N_Typed_Variable_Declaration then + Set_String_Type_Of + (Variable, To => String_Type_Of (Current_Variable)); + end if; + end if; + end Parse_Variable_Reference; + + --------------------------------- + -- Start_New_Case_Construction -- + --------------------------------- + + procedure Start_New_Case_Construction (String_Type : Project_Node_Id) is + Current_String : Project_Node_Id; + + begin + if Choice_First = 0 then + Choice_First := 1; + Choices.Set_Last (First_Choice_Node_Id); + else + Choice_First := Choices.Last + 1; + end if; + + if String_Type /= Empty_Node then + Current_String := First_Literal_String (String_Type); + + while Current_String /= Empty_Node loop + Add (This_String => String_Value_Of (Current_String)); + Current_String := Next_Literal_String (Current_String); + end loop; + end if; + + Choice_Lasts.Increment_Last; + Choice_Lasts.Table (Choice_Lasts.Last) := Choices.Last; + + end Start_New_Case_Construction; + + ----------- + -- Terms -- + ----------- + + procedure Terms (Term : out Project_Node_Id; + Expr_Kind : in out Variable_Kind; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id) + is + Next_Term : Project_Node_Id := Empty_Node; + Term_Id : Project_Node_Id := Empty_Node; + Current_Expression : Project_Node_Id := Empty_Node; + Next_Expression : Project_Node_Id := Empty_Node; + Current_Location : Source_Ptr := No_Location; + Reference : Project_Node_Id := Empty_Node; + + begin + Term := Default_Project_Node (Of_Kind => N_Term); + Set_Location_Of (Term, To => Token_Ptr); + + case Token is + + when Tok_Left_Paren => + case Expr_Kind is + when Undefined => + Expr_Kind := List; + when List => + null; + when Single => + Expr_Kind := List; + Error_Msg + ("literal string list cannot appear in a string", + Token_Ptr); + end case; + + Term_Id := Default_Project_Node + (Of_Kind => N_Literal_String_List, + And_Expr_Kind => List); + Set_Current_Term (Term, To => Term_Id); + Set_Location_Of (Term, To => Token_Ptr); + + Scan; + if Token = Tok_Right_Paren then + Scan; + + else + loop + Current_Location := Token_Ptr; + Parse_Expression (Expression => Next_Expression, + Current_Project => Current_Project, + Current_Package => Current_Package); + + if Expression_Kind_Of (Next_Expression) = List then + Error_Msg ("single expression expected", + Current_Location); + end if; + + if Current_Expression = Empty_Node then + Set_First_Expression_In_List + (Term_Id, To => Next_Expression); + else + Set_Next_Expression_In_List + (Current_Expression, To => Next_Expression); + end if; + + Current_Expression := Next_Expression; + exit when Token /= Tok_Comma; + Scan; -- past the comma + end loop; + + Expect (Tok_Right_Paren, "("); + + if Token = Tok_Right_Paren then + Scan; + end if; + end if; + + when Tok_String_Literal => + if Expr_Kind = Undefined then + Expr_Kind := Single; + end if; + + Term_Id := Default_Project_Node (Of_Kind => N_Literal_String); + Set_Current_Term (Term, To => Term_Id); + Set_String_Value_Of (Term_Id, To => Strval (Token_Node)); + + Scan; + + when Tok_Identifier => + Current_Location := Token_Ptr; + Parse_Variable_Reference + (Variable => Reference, + Current_Project => Current_Project, + Current_Package => Current_Package); + Set_Current_Term (Term, To => Reference); + + if Reference /= Empty_Node then + if Expr_Kind = Undefined then + Expr_Kind := Expression_Kind_Of (Reference); + + elsif Expr_Kind = Single + and then Expression_Kind_Of (Reference) = List + then + Expr_Kind := List; + Error_Msg + ("list variable cannot appear in single string expression", + Current_Location); + end if; + end if; + + when Tok_Project => + Current_Location := Token_Ptr; + Scan; + Expect (Tok_Apostrophe, "'"); + + if Token = Tok_Apostrophe then + Attribute_Reference + (Reference => Reference, + First_Attribute => Prj.Attr.Attribute_First, + Current_Project => Current_Project, + Current_Package => Empty_Node); + Set_Current_Term (Term, To => Reference); + end if; + + if Reference /= Empty_Node then + if Expr_Kind = Undefined then + Expr_Kind := Expression_Kind_Of (Reference); + + elsif Expr_Kind = Single + and then Expression_Kind_Of (Reference) = List + then + Error_Msg + ("lists cannot appear in single string expression", + Current_Location); + end if; + end if; + + when Tok_External => + if Expr_Kind = Undefined then + Expr_Kind := Single; + end if; + + External_Reference (External_Value => Reference); + Set_Current_Term (Term, To => Reference); + + when others => + Error_Msg ("cannot be part of an expression", Token_Ptr); + Term := Empty_Node; + return; + end case; + + if Token = Tok_Ampersand then + Scan; + + Terms (Term => Next_Term, + Expr_Kind => Expr_Kind, + Current_Project => Current_Project, + Current_Package => Current_Package); + Set_Next_Term (Term, To => Next_Term); + + end if; + + end Terms; + +end Prj.Strt; diff --git a/gcc/ada/prj-strt.ads b/gcc/ada/prj-strt.ads new file mode 100644 index 00000000000..9bbdbeb8832 --- /dev/null +++ b/gcc/ada/prj-strt.ads @@ -0,0 +1,96 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . S T R T -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.4 $ +-- -- +-- Copyright (C) 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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- This package implements parsing of string expressions in project files. + +with Prj.Tree; use Prj.Tree; + +private package Prj.Strt is + + procedure Parse_String_Type_List (First_String : out Project_Node_Id); + -- Get the list of literal strings that are allowed for a typed string. + -- On entry, the current token is the first literal string following + -- a left parenthesis in a string type declaration such as: + -- type Toto is ("string_1", "string_2", "string_3"); + -- On exit, the current token is the right parenthesis. + -- The parameter First_String is a node that contained the first + -- literal string of the string type, linked with the following + -- literal strings. + -- + -- Report an error if + -- - a literal string is not found at the beginning of the list + -- or after a comma + -- - two literal strings in the list are equal + + procedure Start_New_Case_Construction (String_Type : Project_Node_Id); + -- This procedure is called at the beginning of a case construction + -- The parameter String_Type is the node for the string type + -- of the case label variable. + -- The different literal strings of the string type are stored + -- into a table to be checked against the case labels of the + -- case construction. + + procedure End_Case_Construction; + -- This procedure is called at the end of a case construction + -- to remove the case labels and to restore the previous state. + -- In particular, in the case of nested case constructions, + -- the case labels of the enclosing case construction are restored. + + procedure Parse_Choice_List + (First_Choice : out Project_Node_Id); + -- Get the label for a choice list. + -- Report an error if + -- - a case label is not a literal string + -- - a case label is not in the typed string list + -- - the same case label is repeated in the same case construction + + procedure Parse_Expression + (Expression : out Project_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Parse a simple string expression or a string list expression. + -- Current_Project is the node of the project file being parsed. + -- Current_Package is the node of the package being parsed, + -- or Empty_Node when we are at the project level (not in a package). + -- On exit, Expression is the node of the expression that has + -- been parsed. + + procedure Parse_Variable_Reference + (Variable : out Project_Node_Id; + Current_Project : Project_Node_Id; + Current_Package : Project_Node_Id); + -- Parse a variable or attribute reference. + -- Used internally (in expressions) and for case variables (in Prj.Dect). + -- Current_Package is the node of the package being parsed, + -- or Empty_Node when we are at the project level (not in a package). + -- On exit, Variable is the node of the variable or attribute reference. + -- A variable reference is made of one to three simple names. + -- An attribute reference is made of one or two simple names, + -- followed by an apostroph, followed by the attribute simple name. + +end Prj.Strt; diff --git a/gcc/ada/prj-tree.adb b/gcc/ada/prj-tree.adb new file mode 100644 index 00000000000..322e4aae39f --- /dev/null +++ b/gcc/ada/prj-tree.adb @@ -0,0 +1,1478 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . T R E E -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.7 $ +-- -- +-- Copyright (C) 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 Stringt; use Stringt; + +package body Prj.Tree is + + use Tree_Private_Part; + + -------------------------------- + -- Associative_Array_Index_Of -- + -------------------------------- + + function Associative_Array_Index_Of + (Node : Project_Node_Id) + return String_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Attribute_Declaration)); + return Project_Nodes.Table (Node).Value; + end Associative_Array_Index_Of; + + -------------------------------- + -- Case_Variable_Reference_Of -- + -------------------------------- + + function Case_Variable_Reference_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Construction); + return Project_Nodes.Table (Node).Field1; + end Case_Variable_Reference_Of; + + ----------------------- + -- Current_Item_Node -- + ----------------------- + + function Current_Item_Node + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Declarative_Item); + return Project_Nodes.Table (Node).Field1; + end Current_Item_Node; + + ------------------ + -- Current_Term -- + ------------------ + + function Current_Term + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Term); + return Project_Nodes.Table (Node).Field1; + end Current_Term; + + -------------------------- + -- Default_Project_Node -- + -------------------------- + + function Default_Project_Node + (Of_Kind : Project_Node_Kind; + And_Expr_Kind : Variable_Kind := Undefined) + return Project_Node_Id + is + begin + Project_Nodes.Increment_Last; + Project_Nodes.Table (Project_Nodes.Last) := + (Kind => Of_Kind, + Location => No_Location, + Directory => No_Name, + Expr_Kind => And_Expr_Kind, + Variables => Empty_Node, + Packages => Empty_Node, + Pkg_Id => Empty_Package, + Name => No_Name, + Path_Name => No_Name, + Value => No_String, + Field1 => Empty_Node, + Field2 => Empty_Node, + Field3 => Empty_Node); + return Project_Nodes.Last; + end Default_Project_Node; + + ------------------ + -- Directory_Of -- + ------------------ + + function Directory_Of (Node : Project_Node_Id) return Name_Id is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + return Project_Nodes.Table (Node).Directory; + end Directory_Of; + + ------------------------ + -- Expression_Kind_Of -- + ------------------------ + + function Expression_Kind_Of (Node : Project_Node_Id) return Variable_Kind is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Literal_String + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Expression + or else + Project_Nodes.Table (Node).Kind = N_Term + or else + Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Reference)); + + return Project_Nodes.Table (Node).Expr_Kind; + end Expression_Kind_Of; + + ------------------- + -- Expression_Of -- + ------------------- + + function Expression_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Attribute_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Variable_Declaration)); + + return Project_Nodes.Table (Node).Field1; + end Expression_Of; + + --------------------------- + -- External_Reference_Of -- + --------------------------- + + function External_Reference_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_External_Value); + return Project_Nodes.Table (Node).Field1; + end External_Reference_Of; + + ------------------------- + -- External_Default_Of -- + ------------------------- + + function External_Default_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_External_Value); + return Project_Nodes.Table (Node).Field2; + end External_Default_Of; + + ------------------------ + -- First_Case_Item_Of -- + ------------------------ + + function First_Case_Item_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Construction); + return Project_Nodes.Table (Node).Field2; + end First_Case_Item_Of; + + --------------------- + -- First_Choice_Of -- + --------------------- + + function First_Choice_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Item); + return Project_Nodes.Table (Node).Field1; + end First_Choice_Of; + + ------------------------------- + -- First_Declarative_Item_Of -- + ------------------------------- + + function First_Declarative_Item_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Project_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Case_Item + or else + Project_Nodes.Table (Node).Kind = N_Package_Declaration)); + + if Project_Nodes.Table (Node).Kind = N_Project_Declaration then + return Project_Nodes.Table (Node).Field1; + else + return Project_Nodes.Table (Node).Field2; + end if; + end First_Declarative_Item_Of; + + ------------------------------ + -- First_Expression_In_List -- + ------------------------------ + + function First_Expression_In_List + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Literal_String_List); + return Project_Nodes.Table (Node).Field1; + end First_Expression_In_List; + + -------------------------- + -- First_Literal_String -- + -------------------------- + + function First_Literal_String + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_String_Type_Declaration); + return Project_Nodes.Table (Node).Field1; + end First_Literal_String; + + ---------------------- + -- First_Package_Of -- + ---------------------- + + function First_Package_Of + (Node : Project_Node_Id) + return Package_Declaration_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + return Project_Nodes.Table (Node).Packages; + end First_Package_Of; + + -------------------------- + -- First_String_Type_Of -- + -------------------------- + + function First_String_Type_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + return Project_Nodes.Table (Node).Field3; + end First_String_Type_Of; + + ---------------- + -- First_Term -- + ---------------- + + function First_Term + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Expression); + return Project_Nodes.Table (Node).Field1; + end First_Term; + + ----------------------- + -- First_Variable_Of -- + ----------------------- + + function First_Variable_Of + (Node : Project_Node_Id) + return Variable_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Project + or else + Project_Nodes.Table (Node).Kind = N_Package_Declaration)); + + return Project_Nodes.Table (Node).Variables; + end First_Variable_Of; + + -------------------------- + -- First_With_Clause_Of -- + -------------------------- + + function First_With_Clause_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + return Project_Nodes.Table (Node).Field1; + end First_With_Clause_Of; + + ---------------- + -- Initialize -- + ---------------- + + procedure Initialize is + begin + Project_Nodes.Set_Last (Empty_Node); + Projects_Htable.Reset; + end Initialize; + + ------------- + -- Kind_Of -- + ------------- + + function Kind_Of (Node : Project_Node_Id) return Project_Node_Kind is + begin + pragma Assert (Node /= Empty_Node); + return Project_Nodes.Table (Node).Kind; + end Kind_Of; + + ----------------- + -- Location_Of -- + ----------------- + + function Location_Of (Node : Project_Node_Id) return Source_Ptr is + begin + pragma Assert (Node /= Empty_Node); + return Project_Nodes.Table (Node).Location; + end Location_Of; + + ------------------------- + -- Modified_Project_Of -- + ------------------------- + + function Modified_Project_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project_Declaration); + return Project_Nodes.Table (Node).Field2; + end Modified_Project_Of; + + ------------------------------ + -- Modified_Project_Path_Of -- + ------------------------------ + + function Modified_Project_Path_Of + (Node : Project_Node_Id) + return String_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + return Project_Nodes.Table (Node).Value; + end Modified_Project_Path_Of; + + ------------- + -- Name_Of -- + ------------- + + function Name_Of (Node : Project_Node_Id) return Name_Id is + begin + pragma Assert (Node /= Empty_Node); + return Project_Nodes.Table (Node).Name; + end Name_Of; + + -------------------- + -- Next_Case_Item -- + -------------------- + + function Next_Case_Item + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Item); + return Project_Nodes.Table (Node).Field3; + end Next_Case_Item; + + --------------------------- + -- Next_Declarative_Item -- + --------------------------- + + function Next_Declarative_Item + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Declarative_Item); + return Project_Nodes.Table (Node).Field2; + end Next_Declarative_Item; + + ----------------------------- + -- Next_Expression_In_List -- + ----------------------------- + + function Next_Expression_In_List + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Expression); + return Project_Nodes.Table (Node).Field2; + end Next_Expression_In_List; + + ------------------------- + -- Next_Literal_String -- + ------------------------- + + function Next_Literal_String + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Literal_String); + return Project_Nodes.Table (Node).Field1; + end Next_Literal_String; + + ----------------------------- + -- Next_Package_In_Project -- + ----------------------------- + + function Next_Package_In_Project + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Package_Declaration); + return Project_Nodes.Table (Node).Field3; + end Next_Package_In_Project; + + ---------------------- + -- Next_String_Type -- + ---------------------- + + function Next_String_Type + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_String_Type_Declaration); + return Project_Nodes.Table (Node).Field2; + end Next_String_Type; + + --------------- + -- Next_Term -- + --------------- + + function Next_Term + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Term); + return Project_Nodes.Table (Node).Field2; + end Next_Term; + + ------------------- + -- Next_Variable -- + ------------------- + + function Next_Variable + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Variable_Declaration)); + + return Project_Nodes.Table (Node).Field3; + end Next_Variable; + + ------------------------- + -- Next_With_Clause_Of -- + ------------------------- + + function Next_With_Clause_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_With_Clause); + return Project_Nodes.Table (Node).Field2; + end Next_With_Clause_Of; + + ------------------- + -- Package_Id_Of -- + ------------------- + + function Package_Id_Of (Node : Project_Node_Id) return Package_Node_Id is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Package_Declaration); + return Project_Nodes.Table (Node).Pkg_Id; + end Package_Id_Of; + + --------------------- + -- Package_Node_Of -- + --------------------- + + function Package_Node_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Reference)); + return Project_Nodes.Table (Node).Field2; + end Package_Node_Of; + + ------------------ + -- Path_Name_Of -- + ------------------ + + function Path_Name_Of (Node : Project_Node_Id) return Name_Id is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Project + or else + Project_Nodes.Table (Node).Kind = N_With_Clause)); + return Project_Nodes.Table (Node).Path_Name; + end Path_Name_Of; + + ---------------------------- + -- Project_Declaration_Of -- + ---------------------------- + + function Project_Declaration_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + return Project_Nodes.Table (Node).Field2; + end Project_Declaration_Of; + + --------------------- + -- Project_Node_Of -- + --------------------- + + function Project_Node_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_With_Clause + or else + Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Reference)); + return Project_Nodes.Table (Node).Field1; + end Project_Node_Of; + + ----------------------------------- + -- Project_Of_Renamed_Package_Of -- + ----------------------------------- + + function Project_Of_Renamed_Package_Of + (Node : Project_Node_Id) + return Project_Node_Id + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Package_Declaration); + return Project_Nodes.Table (Node).Field1; + end Project_Of_Renamed_Package_Of; + + ------------------------------------ + -- Set_Associative_Array_Index_Of -- + ------------------------------------ + + procedure Set_Associative_Array_Index_Of + (Node : Project_Node_Id; + To : String_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Attribute_Declaration); + Project_Nodes.Table (Node).Value := To; + end Set_Associative_Array_Index_Of; + + ------------------------------------ + -- Set_Case_Variable_Reference_Of -- + ------------------------------------ + + procedure Set_Case_Variable_Reference_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Construction); + Project_Nodes.Table (Node).Field1 := To; + end Set_Case_Variable_Reference_Of; + + --------------------------- + -- Set_Current_Item_Node -- + --------------------------- + + procedure Set_Current_Item_Node + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Declarative_Item); + Project_Nodes.Table (Node).Field1 := To; + end Set_Current_Item_Node; + + ---------------------- + -- Set_Current_Term -- + ---------------------- + + procedure Set_Current_Term + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Term); + Project_Nodes.Table (Node).Field1 := To; + end Set_Current_Term; + + ---------------------- + -- Set_Directory_Of -- + ---------------------- + + procedure Set_Directory_Of + (Node : Project_Node_Id; + To : Name_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + Project_Nodes.Table (Node).Directory := To; + end Set_Directory_Of; + + ---------------------------- + -- Set_Expression_Kind_Of -- + ---------------------------- + + procedure Set_Expression_Kind_Of + (Node : Project_Node_Id; + To : Variable_Kind) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Literal_String + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Expression + or else + Project_Nodes.Table (Node).Kind = N_Term + or else + Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Reference)); + Project_Nodes.Table (Node).Expr_Kind := To; + end Set_Expression_Kind_Of; + + ----------------------- + -- Set_Expression_Of -- + ----------------------- + + procedure Set_Expression_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Attribute_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Variable_Declaration)); + Project_Nodes.Table (Node).Field1 := To; + end Set_Expression_Of; + + ------------------------------- + -- Set_External_Reference_Of -- + ------------------------------- + + procedure Set_External_Reference_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_External_Value); + Project_Nodes.Table (Node).Field1 := To; + end Set_External_Reference_Of; + + ----------------------------- + -- Set_External_Default_Of -- + ----------------------------- + + procedure Set_External_Default_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_External_Value); + Project_Nodes.Table (Node).Field2 := To; + end Set_External_Default_Of; + + ---------------------------- + -- Set_First_Case_Item_Of -- + ---------------------------- + + procedure Set_First_Case_Item_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Construction); + Project_Nodes.Table (Node).Field2 := To; + end Set_First_Case_Item_Of; + + ------------------------- + -- Set_First_Choice_Of -- + ------------------------- + + procedure Set_First_Choice_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Item); + Project_Nodes.Table (Node).Field1 := To; + end Set_First_Choice_Of; + + ------------------------ + -- Set_Next_Case_Item -- + ------------------------ + + procedure Set_Next_Case_Item + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Case_Item); + Project_Nodes.Table (Node).Field3 := To; + end Set_Next_Case_Item; + + ----------------------------------- + -- Set_First_Declarative_Item_Of -- + ----------------------------------- + + procedure Set_First_Declarative_Item_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Project_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Case_Item + or else + Project_Nodes.Table (Node).Kind = N_Package_Declaration)); + + if Project_Nodes.Table (Node).Kind = N_Project_Declaration then + Project_Nodes.Table (Node).Field1 := To; + else + Project_Nodes.Table (Node).Field2 := To; + end if; + end Set_First_Declarative_Item_Of; + + ---------------------------------- + -- Set_First_Expression_In_List -- + ---------------------------------- + + procedure Set_First_Expression_In_List + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Literal_String_List); + Project_Nodes.Table (Node).Field1 := To; + end Set_First_Expression_In_List; + + ------------------------------ + -- Set_First_Literal_String -- + ------------------------------ + + procedure Set_First_Literal_String + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_String_Type_Declaration); + Project_Nodes.Table (Node).Field1 := To; + end Set_First_Literal_String; + + -------------------------- + -- Set_First_Package_Of -- + -------------------------- + + procedure Set_First_Package_Of + (Node : Project_Node_Id; + To : Package_Declaration_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + Project_Nodes.Table (Node).Packages := To; + end Set_First_Package_Of; + + ------------------------------ + -- Set_First_String_Type_Of -- + ------------------------------ + + procedure Set_First_String_Type_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + Project_Nodes.Table (Node).Field3 := To; + end Set_First_String_Type_Of; + + -------------------- + -- Set_First_Term -- + -------------------- + + procedure Set_First_Term + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Expression); + Project_Nodes.Table (Node).Field1 := To; + end Set_First_Term; + + --------------------------- + -- Set_First_Variable_Of -- + --------------------------- + + procedure Set_First_Variable_Of + (Node : Project_Node_Id; + To : Variable_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Project + or else + Project_Nodes.Table (Node).Kind = N_Package_Declaration)); + Project_Nodes.Table (Node).Variables := To; + end Set_First_Variable_Of; + + ------------------------------ + -- Set_First_With_Clause_Of -- + ------------------------------ + + procedure Set_First_With_Clause_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + Project_Nodes.Table (Node).Field1 := To; + end Set_First_With_Clause_Of; + + ----------------- + -- Set_Kind_Of -- + ----------------- + + procedure Set_Kind_Of + (Node : Project_Node_Id; + To : Project_Node_Kind) + is + begin + pragma Assert (Node /= Empty_Node); + Project_Nodes.Table (Node).Kind := To; + end Set_Kind_Of; + + --------------------- + -- Set_Location_Of -- + --------------------- + + procedure Set_Location_Of + (Node : Project_Node_Id; + To : Source_Ptr) + is + begin + pragma Assert (Node /= Empty_Node); + Project_Nodes.Table (Node).Location := To; + end Set_Location_Of; + + ----------------------------- + -- Set_Modified_Project_Of -- + ----------------------------- + + procedure Set_Modified_Project_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project_Declaration); + Project_Nodes.Table (Node).Field2 := To; + end Set_Modified_Project_Of; + + ---------------------------------- + -- Set_Modified_Project_Path_Of -- + ---------------------------------- + + procedure Set_Modified_Project_Path_Of + (Node : Project_Node_Id; + To : String_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + Project_Nodes.Table (Node).Value := To; + end Set_Modified_Project_Path_Of; + + ----------------- + -- Set_Name_Of -- + ----------------- + + procedure Set_Name_Of + (Node : Project_Node_Id; + To : Name_Id) + is + begin + pragma Assert (Node /= Empty_Node); + Project_Nodes.Table (Node).Name := To; + end Set_Name_Of; + + ------------------------------- + -- Set_Next_Declarative_Item -- + ------------------------------- + + procedure Set_Next_Declarative_Item + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Declarative_Item); + Project_Nodes.Table (Node).Field2 := To; + end Set_Next_Declarative_Item; + + --------------------------------- + -- Set_Next_Expression_In_List -- + --------------------------------- + + procedure Set_Next_Expression_In_List + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Expression); + Project_Nodes.Table (Node).Field2 := To; + end Set_Next_Expression_In_List; + + ----------------------------- + -- Set_Next_Literal_String -- + ----------------------------- + + procedure Set_Next_Literal_String + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Literal_String); + Project_Nodes.Table (Node).Field1 := To; + end Set_Next_Literal_String; + + --------------------------------- + -- Set_Next_Package_In_Project -- + --------------------------------- + + procedure Set_Next_Package_In_Project + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Package_Declaration); + Project_Nodes.Table (Node).Field3 := To; + end Set_Next_Package_In_Project; + + -------------------------- + -- Set_Next_String_Type -- + -------------------------- + + procedure Set_Next_String_Type + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_String_Type_Declaration); + Project_Nodes.Table (Node).Field2 := To; + end Set_Next_String_Type; + + ------------------- + -- Set_Next_Term -- + ------------------- + + procedure Set_Next_Term + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Term); + Project_Nodes.Table (Node).Field2 := To; + end Set_Next_Term; + + ----------------------- + -- Set_Next_Variable -- + ----------------------- + + procedure Set_Next_Variable + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration + or else + Project_Nodes.Table (Node).Kind = N_Variable_Declaration)); + Project_Nodes.Table (Node).Field3 := To; + end Set_Next_Variable; + + ----------------------------- + -- Set_Next_With_Clause_Of -- + ----------------------------- + + procedure Set_Next_With_Clause_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_With_Clause); + Project_Nodes.Table (Node).Field2 := To; + end Set_Next_With_Clause_Of; + + ----------------------- + -- Set_Package_Id_Of -- + ----------------------- + + procedure Set_Package_Id_Of + (Node : Project_Node_Id; + To : Package_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Package_Declaration); + Project_Nodes.Table (Node).Pkg_Id := To; + end Set_Package_Id_Of; + + ------------------------- + -- Set_Package_Node_Of -- + ------------------------- + + procedure Set_Package_Node_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Reference)); + Project_Nodes.Table (Node).Field2 := To; + end Set_Package_Node_Of; + + ---------------------- + -- Set_Path_Name_Of -- + ---------------------- + + procedure Set_Path_Name_Of + (Node : Project_Node_Id; + To : Name_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Project + or else + Project_Nodes.Table (Node).Kind = N_With_Clause)); + Project_Nodes.Table (Node).Path_Name := To; + end Set_Path_Name_Of; + + -------------------------------- + -- Set_Project_Declaration_Of -- + -------------------------------- + + procedure Set_Project_Declaration_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Project); + Project_Nodes.Table (Node).Field2 := To; + end Set_Project_Declaration_Of; + + ------------------------- + -- Set_Project_Node_Of -- + ------------------------- + + procedure Set_Project_Node_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_With_Clause + or else + Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Attribute_Reference)); + Project_Nodes.Table (Node).Field1 := To; + end Set_Project_Node_Of; + + --------------------------------------- + -- Set_Project_Of_Renamed_Package_Of -- + --------------------------------------- + + procedure Set_Project_Of_Renamed_Package_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + Project_Nodes.Table (Node).Kind = N_Package_Declaration); + Project_Nodes.Table (Node).Field1 := To; + end Set_Project_Of_Renamed_Package_Of; + + ------------------------ + -- Set_String_Type_Of -- + ------------------------ + + procedure Set_String_Type_Of + (Node : Project_Node_Id; + To : Project_Node_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration) + and then + Project_Nodes.Table (To).Kind = N_String_Type_Declaration); + + if Project_Nodes.Table (Node).Kind = N_Variable_Reference then + Project_Nodes.Table (Node).Field3 := To; + else + Project_Nodes.Table (Node).Field2 := To; + end if; + end Set_String_Type_Of; + + ------------------------- + -- Set_String_Value_Of -- + ------------------------- + + procedure Set_String_Value_Of + (Node : Project_Node_Id; + To : String_Id) + is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_With_Clause + or else + Project_Nodes.Table (Node).Kind = N_Literal_String)); + Project_Nodes.Table (Node).Value := To; + end Set_String_Value_Of; + + -------------------- + -- String_Type_Of -- + -------------------- + + function String_Type_Of (Node : Project_Node_Id) + return Project_Node_Id is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_Variable_Reference + or else + Project_Nodes.Table (Node).Kind = N_Typed_Variable_Declaration)); + + if Project_Nodes.Table (Node).Kind = N_Variable_Reference then + return Project_Nodes.Table (Node).Field3; + else + return Project_Nodes.Table (Node).Field2; + end if; + end String_Type_Of; + + --------------------- + -- String_Value_Of -- + --------------------- + + function String_Value_Of (Node : Project_Node_Id) return String_Id is + begin + pragma Assert + (Node /= Empty_Node + and then + (Project_Nodes.Table (Node).Kind = N_With_Clause + or else + Project_Nodes.Table (Node).Kind = N_Literal_String)); + return Project_Nodes.Table (Node).Value; + end String_Value_Of; + + -------------------- + -- Value_Is_Valid -- + -------------------- + + function Value_Is_Valid + (For_Typed_Variable : Project_Node_Id; + Value : String_Id) + return Boolean + is + begin + pragma Assert + (For_Typed_Variable /= Empty_Node + and then + (Project_Nodes.Table (For_Typed_Variable).Kind = + N_Typed_Variable_Declaration)); + + declare + Current_String : Project_Node_Id := + First_Literal_String + (String_Type_Of (For_Typed_Variable)); + + begin + while Current_String /= Empty_Node + and then + not String_Equal (String_Value_Of (Current_String), Value) + loop + Current_String := + Next_Literal_String (Current_String); + end loop; + + return Current_String /= Empty_Node; + end; + + end Value_Is_Valid; + +end Prj.Tree; diff --git a/gcc/ada/prj-tree.ads b/gcc/ada/prj-tree.ads new file mode 100644 index 00000000000..d32fcb19808 --- /dev/null +++ b/gcc/ada/prj-tree.ads @@ -0,0 +1,742 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . T R E E -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.9 $ +-- -- +-- Copyright (C) 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). -- +-- -- +------------------------------------------------------------------------------ + +-- This package defines the structure of the Project File tree. + +with GNAT.HTable; + +with Prj.Attr; use Prj.Attr; +with Prj.Com; use Prj.Com; +with Types; use Types; +with Table; + +package Prj.Tree is + + Project_Nodes_Initial : constant := 1_000; + -- Initial number of nodes in table Tree_Private_Part.Project_Nodes + Project_Nodes_Increment : constant := 100; + + Project_Node_Low_Bound : constant := 0; + Project_Node_High_Bound : constant := 099_999_999; -- In practice, infinite + + type Project_Node_Id is range + Project_Node_Low_Bound .. Project_Node_High_Bound; + -- The index of table Tree_Private_Part.Project_Nodes + + Empty_Node : constant Project_Node_Id := Project_Node_Low_Bound; + -- Designates no node in table Project_Nodes + First_Node_Id : constant Project_Node_Id := Project_Node_Low_Bound; + + subtype Variable_Node_Id is Project_Node_Id; + -- Used to designate a node whose expected kind is + -- N_Typed_Variable_Declaration, N_Variable_Declaration or + -- N_Variable_Reference. + subtype Package_Declaration_Id is Project_Node_Id; + -- Used to designate a node whose expected kind is + -- N_Project_Declaration. + + type Project_Node_Kind is + (N_Project, + N_With_Clause, + N_Project_Declaration, + N_Declarative_Item, + N_Package_Declaration, + N_String_Type_Declaration, + N_Literal_String, + N_Attribute_Declaration, + N_Typed_Variable_Declaration, + N_Variable_Declaration, + N_Expression, + N_Term, + N_Literal_String_List, + N_Variable_Reference, + N_External_Value, + N_Attribute_Reference, + N_Case_Construction, + N_Case_Item); + -- Each node in the tree is of a Project_Node_Kind + -- For the signification of the fields in each node of a + -- Project_Node_Kind, look at package Tree_Private_Part. + + procedure Initialize; + -- Initialize the Project File tree: empty the Project_Nodes table + -- and reset the Projects_Htable. + + function Default_Project_Node + (Of_Kind : Project_Node_Kind; + And_Expr_Kind : Variable_Kind := Undefined) + return Project_Node_Id; + -- Returns a Project_Node_Record with the specified Kind and + -- Expr_Kind; all the other components have default nil values. + + ---------------------- + -- Access Functions -- + ---------------------- + + -- The following query functions are part of the abstract interface + -- of the Project File tree + + function Name_Of (Node : Project_Node_Id) return Name_Id; + -- Valid for all non empty nodes. May return No_Name for nodes that have + -- no names. + + function Kind_Of (Node : Project_Node_Id) return Project_Node_Kind; + -- Valid for all non empty nodes + + function Location_Of (Node : Project_Node_Id) return Source_Ptr; + -- Valid for all non empty nodes + + function Directory_Of (Node : Project_Node_Id) return Name_Id; + -- Only valid for N_Project nodes. + + function Expression_Kind_Of (Node : Project_Node_Id) return Variable_Kind; + -- Only valid for N_Literal_String, N_Attribute_Declaration, + -- N_Variable_Declaration, N_Typed_Variable_Declaration, N_Expression, + -- N_Term, N_Variable_Reference or N_Attribute_Reference nodes. + + function First_Variable_Of + (Node : Project_Node_Id) + return Variable_Node_Id; + -- Only valid for N_Project or N_Package_Declaration nodes + + function First_Package_Of + (Node : Project_Node_Id) + return Package_Declaration_Id; + -- Only valid for N_Project nodes + + function Package_Id_Of (Node : Project_Node_Id) return Package_Node_Id; + -- Only valid for N_Package_Declaration nodes + + function Path_Name_Of (Node : Project_Node_Id) return Name_Id; + -- Only valid for N_Project and N_With_Clause nodes. + + function String_Value_Of (Node : Project_Node_Id) return String_Id; + -- Only valid for N_With_Clause or N_Literal_String nodes. + + function First_With_Clause_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Project nodes + + function Project_Declaration_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Project nodes + + function First_String_Type_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Project nodes + + function Modified_Project_Path_Of + (Node : Project_Node_Id) + return String_Id; + -- Only valid for N_With_Clause nodes + + function Project_Node_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Project nodes + + function Next_With_Clause_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_With_Clause nodes + + function First_Declarative_Item_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_With_Clause nodes + + function Modified_Project_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_With_Clause nodes + + function Current_Item_Node + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Declarative_Item nodes + + function Next_Declarative_Item + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Declarative_Item node + + function Project_Of_Renamed_Package_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Package_Declaration nodes. + -- May return Empty_Node. + + function Next_Package_In_Project + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Package_Declaration nodes + + function First_Literal_String + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_String_Type_Declaration nodes + + function Next_String_Type + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_String_Type_Declaration nodes + + function Next_Literal_String + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Literal_String nodes + + function Expression_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Attribute_Declaration, N_Typed_Variable_Declaration + -- or N_Variable_Declaration nodes + + function Value_Is_Valid + (For_Typed_Variable : Project_Node_Id; + Value : String_Id) + return Boolean; + -- Only valid for N_Typed_Variable_Declaration. Returns True if Value is + -- in the list of allowed strings for For_Typed_Variable. False otherwise. + + function Associative_Array_Index_Of + (Node : Project_Node_Id) + return String_Id; + -- Only valid for N_Attribute_Declaration. + -- Returns No_String for non associative array attributes. + + function Next_Variable + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Typed_Variable_Declaration or N_Variable_Declaration + -- nodes. + + function First_Term + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Expression nodes + + function Next_Expression_In_List + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Expression nodes + + function Current_Term + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Term nodes + + function Next_Term + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Term nodes + + function First_Expression_In_List + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Literal_String_List nodes + + function Package_Node_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Variable_Reference or N_Attribute_Reference nodes. + -- May return Empty_Node. + + function String_Type_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Variable_Reference or N_Typed_Variable_Declaration + -- nodes. + + function External_Reference_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_External_Value nodes + + function External_Default_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_External_Value nodes + + function Case_Variable_Reference_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Case_Construction nodes + + function First_Case_Item_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Case_Construction nodes + + function First_Choice_Of + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Case_Item nodes + + function Next_Case_Item + (Node : Project_Node_Id) + return Project_Node_Id; + -- Only valid for N_Case_Item nodes + + -------------------- + -- Set Procedures -- + -------------------- + + -- The following procedures are part of the abstract interface of + -- the Project File tree. + + -- Each Set_* procedure is valid only for the same Project_Node_Kind + -- nodes as the corresponding query function above. + + procedure Set_Name_Of + (Node : Project_Node_Id; + To : Name_Id); + + procedure Set_Kind_Of + (Node : Project_Node_Id; + To : Project_Node_Kind); + + procedure Set_Location_Of + (Node : Project_Node_Id; + To : Source_Ptr); + + procedure Set_Directory_Of + (Node : Project_Node_Id; + To : Name_Id); + + procedure Set_Expression_Kind_Of + (Node : Project_Node_Id; + To : Variable_Kind); + + procedure Set_First_Variable_Of + (Node : Project_Node_Id; + To : Variable_Node_Id); + + procedure Set_First_Package_Of + (Node : Project_Node_Id; + To : Package_Declaration_Id); + + procedure Set_Package_Id_Of + (Node : Project_Node_Id; + To : Package_Node_Id); + + procedure Set_Path_Name_Of + (Node : Project_Node_Id; + To : Name_Id); + + procedure Set_String_Value_Of + (Node : Project_Node_Id; + To : String_Id); + + procedure Set_First_With_Clause_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Project_Declaration_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_First_String_Type_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Modified_Project_Path_Of + (Node : Project_Node_Id; + To : String_Id); + + procedure Set_Project_Node_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_With_Clause_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_First_Declarative_Item_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Modified_Project_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Current_Item_Node + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_Declarative_Item + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Project_Of_Renamed_Package_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_Package_In_Project + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_First_Literal_String + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_String_Type + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_Literal_String + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Expression_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Associative_Array_Index_Of + (Node : Project_Node_Id; + To : String_Id); + + procedure Set_Next_Variable + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_First_Term + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_Expression_In_List + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Current_Term + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_Term + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_First_Expression_In_List + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Package_Node_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_String_Type_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_External_Reference_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_External_Default_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Case_Variable_Reference_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_First_Case_Item_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_First_Choice_Of + (Node : Project_Node_Id; + To : Project_Node_Id); + + procedure Set_Next_Case_Item + (Node : Project_Node_Id; + To : Project_Node_Id); + + ------------------------------- + -- Restricted Access Section -- + ------------------------------- + + package Tree_Private_Part is + + -- This is conceptually in the private part. + -- However, for efficiency, some packages are accessing it directly. + + type Project_Node_Record is record + + Kind : Project_Node_Kind; + + Location : Source_Ptr := No_Location; + + Directory : Name_Id := No_Name; + -- Only for N_Project + + Expr_Kind : Variable_Kind := Undefined; + -- See below for what Project_Node_Kind it is used + + Variables : Variable_Node_Id := Empty_Node; + -- First variable in a project or a package + + Packages : Package_Declaration_Id := Empty_Node; + -- First package declaration in a project + + Pkg_Id : Package_Node_Id := Empty_Package; + -- Only use in Package_Declaration + + Name : Name_Id := No_Name; + -- See below for what Project_Node_Kind it is used + + Path_Name : Name_Id := No_Name; + -- See below for what Project_Node_Kind it is used + + Value : String_Id := No_String; + -- See below for what Project_Node_Kind it is used + + Field1 : Project_Node_Id := Empty_Node; + -- See below the meaning for each Project_Node_Kind + + Field2 : Project_Node_Id := Empty_Node; + -- See below the meaning for each Project_Node_Kind + + Field3 : Project_Node_Id := Empty_Node; + -- See below the meaning for each Project_Node_Kind + + end record; + + -- type Project_Node_Kind is + + -- (N_Project, + -- -- Name: project name + -- -- Path_Name: project path name + -- -- Expr_Kind: Undefined + -- -- Field1: first with clause + -- -- Field2: project declaration + -- -- Field3: first string type + -- -- Value: modified project path name (if any) + + -- N_With_Clause, + -- -- Name: imported project name + -- -- Path_Name: imported project path name + -- -- Expr_Kind: Undefined + -- -- Field1: project node + -- -- Field2: next with clause + -- -- Field3: not used + -- -- Value: literal string withed + + -- N_Project_Declaration, + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: Undefined + -- -- Field1: first declarative item + -- -- Field2: modified project + -- -- Field3: not used + -- -- Value: not used + + -- N_Declarative_Item, + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: Undefined + -- -- Field1: current item node + -- -- Field2: next declarative item + -- -- Field3: not used + -- -- Value: not used + + -- N_Package_Declaration, + -- -- Name: package name + -- -- Path_Name: not used + -- -- Expr_Kind: Undefined + -- -- Field1: project of renamed package (if any) + -- -- Field2: first declarative item + -- -- Field3: next package in project + -- -- Value: not used + + -- N_String_Type_Declaration, + -- -- Name: type name + -- -- Path_Name: not used + -- -- Expr_Kind: Undefined + -- -- Field1: first literal string + -- -- Field2: next string type + -- -- Field3: not used + -- -- Value: not used + + -- N_Literal_String, + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: Single + -- -- Field1: next literal string + -- -- Field2: not used + -- -- Field3: not used + -- -- Value: string value + + -- N_Attribute_Declaration, + -- -- Name: attribute name + -- -- Path_Name: not used + -- -- Expr_Kind: attribute kind + -- -- Field1: expression + -- -- Field2: not used + -- -- Field3: not used + -- -- Value: associative array index + -- -- (if an associative array element) + + -- N_Typed_Variable_Declaration, + -- -- Name: variable name + -- -- Path_Name: not used + -- -- Expr_Kind: Single + -- -- Field1: expression + -- -- Field2: type of variable (N_String_Type_Declaration) + -- -- Field3: next variable + -- -- Value: not used + + -- N_Variable_Declaration, + -- -- Name: variable name + -- -- Path_Name: not used + -- -- Expr_Kind: variable kind + -- -- Field1: expression + -- -- Field2: not used + -- -- Field3 is used for next variable, instead of Field2, + -- -- so that it is the same field for + -- -- N_Variable_Declaration and + -- -- N_Typed_Variable_Declaration + -- -- Field3: next variable + -- -- Value: not used + + -- N_Expression, + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: expression kind + -- -- Field1: first term + -- -- Field2: next expression in list + -- -- Field3: not used + -- -- Value: not used + + -- N_Term, + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: term kind + -- -- Field1: current term + -- -- Field2: next term in the expression + -- -- Field3: not used + -- -- Value: not used + + -- N_Literal_String_List, + -- -- Designates a list of string expressions between brackets + -- -- separated by commas. The string expressions are not necessarily + -- -- literal strings. + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: List + -- -- Field1: first expression + -- -- Field2: not used + -- -- Field3: not used + -- -- Value: not used + + -- N_Variable_Reference, + -- -- Name: variable name + -- -- Path_Name: not used + -- -- Expr_Kind: variable kind + -- -- Field1: project (if specified) + -- -- Field2: package (if specified) + -- -- Field3: type of variable (N_String_Type_Declaration), if any + -- -- Value: not used + + -- N_External_Value, + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: Single + -- -- Field1: Name of the external reference (literal string) + -- -- Field2: Default (literal string) + -- -- Field3: not used + -- -- Value: not used + + -- N_Attribute_Reference, + -- -- Name: attribute name + -- -- Path_Name: not used + -- -- Expr_Kind: attribute kind + -- -- Field1: project + -- -- Field2: package (if attribute of a package) + -- -- Field3: not used + -- -- Value: not used + + -- N_Case_Construction, + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: Undefined + -- -- Field1: case variable reference + -- -- Field2: first case item + -- -- Field3: not used + -- -- Value: not used + + -- N_Case_Item); + -- -- Name: not used + -- -- Path_Name: not used + -- -- Expr_Kind: not used + -- -- Field1: first choice (literal string) + -- -- Field2: first declarative item + -- -- Field3: next case item + -- -- Value: not used + + package Project_Nodes is + new Table.Table (Table_Component_Type => Project_Node_Record, + Table_Index_Type => Project_Node_Id, + Table_Low_Bound => First_Node_Id, + Table_Initial => Project_Nodes_Initial, + Table_Increment => Project_Nodes_Increment, + Table_Name => "Project_Nodes"); + -- This table contains the syntactic tree of project data + -- from project files. + + type Project_Name_And_Node is record + Name : Name_Id; + -- Name of the project + Node : Project_Node_Id; + -- Node of the project in table Project_Nodes + Modified : Boolean; + -- True when the project is being modified by another project + end record; + + No_Project_Name_And_Node : constant Project_Name_And_Node := + (Name => No_Name, Node => Empty_Node, Modified => True); + + package Projects_Htable is new GNAT.HTable.Simple_HTable + (Header_Num => Header_Num, + Element => Project_Name_And_Node, + No_Element => No_Project_Name_And_Node, + Key => Name_Id, + Hash => Hash, + Equal => "="); + -- This hash table contains a mapping of project names to project nodes. + -- Note that this hash table contains only the nodes whose Kind is + -- N_Project. It is used to find the node of a project from its + -- name, and to verify if a project has already been parsed, knowing + -- its name. + + end Tree_Private_Part; + +end Prj.Tree; diff --git a/gcc/ada/prj-util.adb b/gcc/ada/prj-util.adb new file mode 100644 index 00000000000..6a94a0cfc4c --- /dev/null +++ b/gcc/ada/prj-util.adb @@ -0,0 +1,415 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . U T I L -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.8 $ -- +-- -- +-- Copyright (C) 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 Ada.Unchecked_Deallocation; + +with Namet; use Namet; +with Osint; +with Output; use Output; +with Stringt; use Stringt; + +package body Prj.Util is + + procedure Free is new Ada.Unchecked_Deallocation + (Text_File_Data, Text_File); + + ----------- + -- Close -- + ----------- + + procedure Close (File : in out Text_File) is + begin + if File = null then + Osint.Fail ("Close attempted on an invalid Text_File"); + end if; + + Close (File.FD); + Free (File); + end Close; + + ----------------- + -- End_Of_File -- + ----------------- + + function End_Of_File (File : Text_File) return Boolean is + begin + if File = null then + Osint.Fail ("End_Of_File attempted on an invalid Text_File"); + end if; + + return File.End_Of_File_Reached; + end End_Of_File; + + -------------- + -- Get_Line -- + -------------- + + procedure Get_Line + (File : Text_File; + Line : out String; + Last : out Natural) + is + C : Character; + + procedure Advance; + + ------------- + -- Advance -- + ------------- + + procedure Advance is + begin + if File.Cursor = File.Buffer_Len then + File.Buffer_Len := + Read + (FD => File.FD, + A => File.Buffer'Address, + N => File.Buffer'Length); + + if File.Buffer_Len = 0 then + File.End_Of_File_Reached := True; + return; + else + File.Cursor := 1; + end if; + + else + File.Cursor := File.Cursor + 1; + end if; + end Advance; + + -- Start of processing for Get_Line + + begin + if File = null then + Osint.Fail ("Get_Line attempted on an invalid Text_File"); + end if; + + Last := Line'First - 1; + + if not File.End_Of_File_Reached then + loop + C := File.Buffer (File.Cursor); + exit when C = ASCII.CR or else C = ASCII.LF; + Last := Last + 1; + Line (Last) := C; + Advance; + + if File.End_Of_File_Reached then + return; + end if; + + exit when Last = Line'Last; + end loop; + + if C = ASCII.CR or else C = ASCII.LF then + Advance; + + if File.End_Of_File_Reached then + return; + end if; + end if; + + if C = ASCII.CR + and then File.Buffer (File.Cursor) = ASCII.LF + then + Advance; + end if; + end if; + end Get_Line; + + -------------- + -- Is_Valid -- + -------------- + + function Is_Valid (File : Text_File) return Boolean is + begin + return File /= null; + end Is_Valid; + + ---------- + -- Open -- + ---------- + + procedure Open (File : out Text_File; Name : in String) is + FD : File_Descriptor; + File_Name : String (1 .. Name'Length + 1); + + begin + File_Name (1 .. Name'Length) := Name; + File_Name (File_Name'Last) := ASCII.NUL; + FD := Open_Read (Name => File_Name'Address, + Fmode => GNAT.OS_Lib.Text); + if FD = Invalid_FD then + File := null; + else + File := new Text_File_Data; + File.FD := FD; + File.Buffer_Len := + Read (FD => FD, + A => File.Buffer'Address, + N => File.Buffer'Length); + + if File.Buffer_Len = 0 then + File.End_Of_File_Reached := True; + else + File.Cursor := 1; + end if; + end if; + end Open; + + -------------- + -- Value_Of -- + -------------- + + function Value_Of + (Index : Name_Id; + In_Array : Array_Element_Id) + return Name_Id + is + Current : Array_Element_Id := In_Array; + Element : Array_Element; + + begin + while Current /= No_Array_Element loop + Element := Array_Elements.Table (Current); + + if Index = Element.Index then + exit when Element.Value.Kind /= Single; + exit when String_Length (Element.Value.Value) = 0; + String_To_Name_Buffer (Element.Value.Value); + return Name_Find; + else + Current := Element.Next; + end if; + end loop; + + return No_Name; + end Value_Of; + + function Value_Of + (Index : Name_Id; + In_Array : Array_Element_Id) + return Variable_Value + is + Current : Array_Element_Id := In_Array; + Element : Array_Element; + + begin + while Current /= No_Array_Element loop + Element := Array_Elements.Table (Current); + + if Index = Element.Index then + return Element.Value; + else + Current := Element.Next; + end if; + end loop; + + return Nil_Variable_Value; + end Value_Of; + + function Value_Of + (Name : Name_Id; + Attribute_Or_Array_Name : Name_Id; + In_Package : Package_Id) + return Variable_Value + is + The_Array : Array_Element_Id; + The_Attribute : Variable_Value := Nil_Variable_Value; + + begin + if In_Package /= No_Package then + + -- First, look if there is an array element that fits + + The_Array := + Value_Of + (Name => Attribute_Or_Array_Name, + In_Arrays => Packages.Table (In_Package).Decl.Arrays); + The_Attribute := + Value_Of + (Index => Name, + In_Array => The_Array); + + -- If there is no array element, look for a variable + + if The_Attribute = Nil_Variable_Value then + The_Attribute := + Value_Of + (Variable_Name => Attribute_Or_Array_Name, + In_Variables => Packages.Table (In_Package).Decl.Attributes); + end if; + end if; + + return The_Attribute; + end Value_Of; + + function Value_Of + (Index : Name_Id; + In_Array : Name_Id; + In_Arrays : Array_Id) + return Name_Id + is + Current : Array_Id := In_Arrays; + The_Array : Array_Data; + + begin + while Current /= No_Array loop + The_Array := Arrays.Table (Current); + if The_Array.Name = In_Array then + return Value_Of (Index, In_Array => The_Array.Value); + else + Current := The_Array.Next; + end if; + end loop; + + return No_Name; + end Value_Of; + + function Value_Of + (Name : Name_Id; + In_Arrays : Array_Id) + return Array_Element_Id + is + Current : Array_Id := In_Arrays; + The_Array : Array_Data; + + begin + while Current /= No_Array loop + The_Array := Arrays.Table (Current); + if The_Array.Name = Name then + return The_Array.Value; + else + Current := The_Array.Next; + end if; + end loop; + + return No_Array_Element; + end Value_Of; + + function Value_Of + (Name : Name_Id; + In_Packages : Package_Id) + return Package_Id + is + Current : Package_Id := In_Packages; + The_Package : Package_Element; + + begin + while Current /= No_Package loop + The_Package := Packages.Table (Current); + exit when The_Package.Name /= No_Name and then + The_Package.Name = Name; + Current := The_Package.Next; + end loop; + + return Current; + end Value_Of; + + function Value_Of + (Variable_Name : Name_Id; + In_Variables : Variable_Id) + return Variable_Value + is + Current : Variable_Id := In_Variables; + The_Variable : Variable; + + begin + while Current /= No_Variable loop + The_Variable := Variable_Elements.Table (Current); + + if Variable_Name = The_Variable.Name then + return The_Variable.Value; + else + Current := The_Variable.Next; + end if; + end loop; + + return Nil_Variable_Value; + end Value_Of; + + --------------- + -- Write_Str -- + --------------- + + procedure Write_Str + (S : String; + Max_Length : Positive; + Separator : Character) + is + First : Positive := S'First; + Last : Natural := S'Last; + + begin + -- Nothing to do for empty strings + + if S'Length > 0 then + -- Start on a new line if current line is already longer than + -- Max_Length. + + if Positive (Column) >= Max_Length then + Write_Eol; + end if; + + -- If length of remainder is longer than Max_Length, we need to + -- cut the remainder in several lines. + + while Positive (Column) + S'Last - First > Max_Length loop + -- Try the maximum length possible + + Last := First + Max_Length - Positive (Column); + + -- Look for last Separator in the line + + while Last >= First and then S (Last) /= Separator loop + Last := Last - 1; + end loop; + + -- If we do not find a separator, we output the maximum length + -- possible. + if Last < First then + Last := First + Max_Length - Positive (Column); + end if; + + Write_Line (S (First .. Last)); + + -- Set the beginning of the new remainder + + First := Last + 1; + + end loop; + + -- What is left goes to the buffer, without EOL + + Write_Str (S (First .. S'Last)); + + end if; + end Write_Str; + +end Prj.Util; diff --git a/gcc/ada/prj-util.ads b/gcc/ada/prj-util.ads new file mode 100644 index 00000000000..baef0404f0e --- /dev/null +++ b/gcc/ada/prj-util.ads @@ -0,0 +1,148 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J . U T I L -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.6 $ +-- -- +-- Copyright (C) 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). -- +-- -- +------------------------------------------------------------------------------ +-- +-- Utilities when using project files. + +with GNAT.OS_Lib; use GNAT.OS_Lib; +with Types; use Types; + +package Prj.Util is + + function Value_Of + (Index : Name_Id; + In_Array : Array_Element_Id) + return Name_Id; + -- Get a single string array component. + -- Returns No_Name if there is no component Index (case sensitive), + -- if In_Array is null, or if the component is a String list. + + function Value_Of + (Index : Name_Id; + In_Array : Array_Element_Id) + return Variable_Value; + -- Get a string array component (single String or String list). + -- Returns Nil_Variable_Value if there is no component Index + -- (case sensitive), or if In_Array is null. + + function Value_Of + (Name : Name_Id; + Attribute_Or_Array_Name : Name_Id; + In_Package : Package_Id) + return Variable_Value; + -- In a specific package, + -- - if there exists an array Variable_Or_Array_Name with an index + -- Name, returns the corresponding component, + -- - otherwise if there is a attribute Attribute_Or_Array_Name, + -- returns this attribute, + -- - otherwise, returns Nil_Variable_Value. + -- If In_Package is null, returns Nil_Variable_Value. + + function Value_Of + (Index : Name_Id; + In_Array : Name_Id; + In_Arrays : Array_Id) + return Name_Id; + -- Get a string array component in an array of an array list. + -- Returns No_Name if there is no component Index (case sensitive), + -- if In_Arrays is null, if In_Array is not found in In_Arrays, + -- or if the component is a String list. + + function Value_Of + (Name : Name_Id; + In_Arrays : Array_Id) + return Array_Element_Id; + -- Returns a specified array in an array list. + -- Returns No_Array_Element if In_Arrays is null or if Name is not the + -- name of an array in In_Arrays. + -- Assumption: Name is in lower case. + + function Value_Of + (Name : Name_Id; + In_Packages : Package_Id) + return Package_Id; + -- Returns a specified package in a package list. + -- Returns No_Package if In_Packages is null or if Name is not the + -- name of a package in Package_List. + -- Assumption: Name is in lower case. + + function Value_Of + (Variable_Name : Name_Id; + In_Variables : Variable_Id) + return Variable_Value; + -- Returns a specified variable in a variable list. + -- Returns null if In_Variables is null or if Variable_Name + -- is not the name of a variable in In_Variables. + -- Assumption: Variable_Name is in lower case. + + procedure Write_Str + (S : String; + Max_Length : Positive; + Separator : Character); + -- Output string S using Output.Write_Str. + -- If S is too long to fit in one line of Max_Length, cut it in + -- several lines, using Separator as the last character of each line, + -- if possible. + + type Text_File is limited private; + -- Represents a text file. + -- Default is invalid text file. + + function Is_Valid (File : Text_File) return Boolean; + -- Returns True if File designates an open text file that + -- has not yet been closed. + + procedure Open (File : out Text_File; Name : String); + -- Open a text file. If this procedure fails, File is invalid. + + function End_Of_File (File : Text_File) return Boolean; + -- Returns True if the end of the text file File has been + -- reached. Fails if File is invalid. + + procedure Get_Line + (File : Text_File; + Line : out String; + Last : out Natural); + -- Reads a line from an open text file. Fails if File is invalid. + + procedure Close (File : in out Text_File); + -- Close an open text file. File becomes invalid. + -- Fails if File is already invalid. + +private + + type Text_File_Data is record + FD : File_Descriptor := Invalid_FD; + Buffer : String (1 .. 1_000); + Buffer_Len : Natural; + Cursor : Natural := 0; + End_Of_File_Reached : Boolean := False; + end record; + + type Text_File is access Text_File_Data; + +end Prj.Util; diff --git a/gcc/ada/prj.adb b/gcc/ada/prj.adb new file mode 100644 index 00000000000..8e302117917 --- /dev/null +++ b/gcc/ada/prj.adb @@ -0,0 +1,286 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.16 $ +-- -- +-- Copyright (C) 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 Ada.Characters.Handling; use Ada.Characters.Handling; +with Errout; use Errout; +with GNAT.OS_Lib; use GNAT.OS_Lib; +with Namet; use Namet; +with Osint; use Osint; +with Prj.Attr; +with Prj.Com; +with Prj.Env; +with Scans; use Scans; +with Scn; +with Stringt; use Stringt; +with Sinfo.CN; +with Snames; use Snames; + +package body Prj is + + The_Empty_String : String_Id; + + subtype Known_Casing is Casing_Type range All_Upper_Case .. Mixed_Case; + + The_Casing_Images : array (Known_Casing) of String_Access := + (All_Lower_Case => new String'("lowercase"), + All_Upper_Case => new String'("UPPERCASE"), + Mixed_Case => new String'("MixedCase")); + + Initialized : Boolean := False; + + Standard_Dot_Replacement : constant Name_Id := + First_Name_Id + Character'Pos ('-'); + Standard_Specification_Append : Name_Id; + Standard_Body_Append : Name_Id; + + Std_Naming_Data : Naming_Data := + (Dot_Replacement => Standard_Dot_Replacement, + Dot_Repl_Loc => No_Location, + Casing => All_Lower_Case, + Specification_Append => No_Name, + Spec_Append_Loc => No_Location, + Body_Append => No_Name, + Body_Append_Loc => No_Location, + Separate_Append => No_Name, + Sep_Append_Loc => No_Location, + Specifications => No_Array_Element, + Bodies => No_Array_Element); + + Project_Empty : Project_Data := + (First_Referred_By => No_Project, + Name => No_Name, + Path_Name => No_Name, + Location => No_Location, + Directory => No_Name, + File_Name => No_Name, + Library => False, + Library_Dir => No_Name, + Library_Name => No_Name, + Library_Kind => Static, + Lib_Internal_Name => No_Name, + Lib_Elaboration => False, + Sources => Nil_String, + Source_Dirs => Nil_String, + Object_Directory => No_Name, + Modifies => No_Project, + Modified_By => No_Project, + Naming => Std_Naming_Data, + Decl => No_Declarations, + Imported_Projects => Empty_Project_List, + Include_Path => null, + Objects_Path => null, + Config_File_Name => No_Name, + Config_File_Temp => False, + Config_Checked => False, + Checked => False, + Seen => False, + Flag1 => False, + Flag2 => False); + + ------------------- + -- Empty_Project -- + ------------------- + + function Empty_Project return Project_Data is + begin + Initialize; + return Project_Empty; + end Empty_Project; + + ------------------ + -- Empty_String -- + ------------------ + + function Empty_String return String_Id is + begin + return The_Empty_String; + end Empty_String; + + ------------ + -- Expect -- + ------------ + + procedure Expect (The_Token : Token_Type; Token_Image : String) is + begin + if Token /= The_Token then + Error_Msg ("""" & Token_Image & """ expected", Token_Ptr); + end if; + end Expect; + + -------------------------------- + -- For_Every_Project_Imported -- + -------------------------------- + + procedure For_Every_Project_Imported + (By : Project_Id; + With_State : in out State) + is + + procedure Check (Project : Project_Id); + -- Check if a project has already been seen. + -- If not seen, mark it as seen, call Action, + -- and check all its imported projects. + + procedure Check (Project : Project_Id) is + List : Project_List; + + begin + if not Projects.Table (Project).Seen then + Projects.Table (Project).Seen := False; + Action (Project, With_State); + + List := Projects.Table (Project).Imported_Projects; + while List /= Empty_Project_List loop + Check (Project_Lists.Table (List).Project); + List := Project_Lists.Table (List).Next; + end loop; + end if; + end Check; + + begin + for Project in Projects.First .. Projects.Last loop + Projects.Table (Project).Seen := False; + end loop; + + Check (Project => By); + end For_Every_Project_Imported; + + ----------- + -- Image -- + ----------- + + function Image (Casing : Casing_Type) return String is + begin + return The_Casing_Images (Casing).all; + end Image; + + ---------------- + -- Initialize -- + ---------------- + + procedure Initialize is + begin + if not Initialized then + Initialized := True; + Stringt.Initialize; + Start_String; + The_Empty_String := End_String; + Name_Len := 4; + Name_Buffer (1 .. 4) := ".ads"; + Canonical_Case_File_Name (Name_Buffer (1 .. 4)); + Standard_Specification_Append := Name_Find; + Name_Buffer (4) := 'b'; + Canonical_Case_File_Name (Name_Buffer (1 .. 4)); + Standard_Body_Append := Name_Find; + Std_Naming_Data.Specification_Append := Standard_Specification_Append; + Std_Naming_Data.Body_Append := Standard_Body_Append; + Std_Naming_Data.Separate_Append := Standard_Body_Append; + Project_Empty.Naming := Std_Naming_Data; + Prj.Env.Initialize; + Prj.Attr.Initialize; + Set_Name_Table_Byte (Name_Project, Token_Type'Pos (Tok_Project)); + Set_Name_Table_Byte (Name_Modifying, Token_Type'Pos (Tok_Modifying)); + Set_Name_Table_Byte (Name_External, Token_Type'Pos (Tok_External)); + end if; + end Initialize; + + ------------ + -- Reset -- + ------------ + + procedure Reset is + begin + Projects.Init; + Project_Lists.Init; + Packages.Init; + Arrays.Init; + Variable_Elements.Init; + String_Elements.Init; + Prj.Com.Units.Init; + Prj.Com.Units_Htable.Reset; + end Reset; + + ------------------------ + -- Same_Naming_Scheme -- + ------------------------ + + function Same_Naming_Scheme + (Left, Right : Naming_Data) + return Boolean + is + begin + return Left.Dot_Replacement = Right.Dot_Replacement + and then Left.Casing = Right.Casing + and then Left.Specification_Append = Right.Specification_Append + and then Left.Body_Append = Right.Body_Append + and then Left.Separate_Append = Right.Separate_Append; + end Same_Naming_Scheme; + + ---------- + -- Scan -- + ---------- + + procedure Scan is + begin + Scn.Scan; + + -- Change operator symbol to literal strings, since that's the way + -- we treat all strings in a project file. + + if Token = Tok_Operator_Symbol then + Sinfo.CN.Change_Operator_Symbol_To_String_Literal (Token_Node); + Token := Tok_String_Literal; + end if; + end Scan; + + -------------------------- + -- Standard_Naming_Data -- + -------------------------- + + function Standard_Naming_Data return Naming_Data is + begin + Initialize; + return Std_Naming_Data; + end Standard_Naming_Data; + + ----------- + -- Value -- + ----------- + + function Value (Image : String) return Casing_Type is + begin + for Casing in The_Casing_Images'Range loop + if To_Lower (Image) = To_Lower (The_Casing_Images (Casing).all) then + return Casing; + end if; + end loop; + + raise Constraint_Error; + end Value; + +end Prj; diff --git a/gcc/ada/prj.ads b/gcc/ada/prj.ads new file mode 100644 index 00000000000..409a0717223 --- /dev/null +++ b/gcc/ada/prj.ads @@ -0,0 +1,416 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- P R J -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.18 $ +-- -- +-- Copyright (C) 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). -- +-- -- +------------------------------------------------------------------------------ + +-- The following package declares the data types for GNAT project. +-- These data types may be used by GNAT Project-aware tools. + +-- Children of these package implements various services on these data types. +-- See in particular Prj.Pars and Prj.Env. + +with Casing; use Casing; +with GNAT.OS_Lib; use GNAT.OS_Lib; +with Scans; use Scans; +with Table; +with Types; use Types; + +package Prj is + + type Put_Line_Access is access procedure (Line : String); + -- Use to customize error reporting in Prj.Proc and Prj.Nmsc. + + type Verbosity is (Default, Medium, High); + -- Verbosity when parsing GNAT Project Files. + -- Default is default (very quiet, if no errors). + -- Medium is more verbose. + -- High is extremely verbose. + + type Lib_Kind is (Static, Dynamic, Relocatable); + + function Empty_String return String_Id; + + type String_List_Id is new Nat; + Nil_String : constant String_List_Id := 0; + type String_Element is record + Value : String_Id := No_String; + Location : Source_Ptr := No_Location; + Next : String_List_Id := Nil_String; + end record; + -- To hold values for string list variables and array elements. + + package String_Elements is new Table.Table + (Table_Component_Type => String_Element, + Table_Index_Type => String_List_Id, + Table_Low_Bound => 1, + Table_Initial => 200, + Table_Increment => 100, + Table_Name => "Prj.String_Elements"); + -- The table for string elements in string lists. + + type Variable_Kind is (Undefined, List, Single); + -- Different kinds of variables + + type Variable_Value (Kind : Variable_Kind := Undefined) is record + Location : Source_Ptr := No_Location; + Default : Boolean := False; + case Kind is + when Undefined => + null; + when List => + Values : String_List_Id := Nil_String; + when Single => + Value : String_Id := No_String; + end case; + end record; + -- Values for variables and array elements + + Nil_Variable_Value : constant Variable_Value := + (Kind => Undefined, + Location => No_Location, + Default => False); + -- Value of a non existing variable or array element. + + type Variable_Id is new Nat; + No_Variable : constant Variable_Id := 0; + type Variable is record + Next : Variable_Id := No_Variable; + Name : Name_Id; + Value : Variable_Value; + end record; + -- To hold the list of variables in a project file and in packages. + + package Variable_Elements is new Table.Table + (Table_Component_Type => Variable, + Table_Index_Type => Variable_Id, + Table_Low_Bound => 1, + Table_Initial => 200, + Table_Increment => 100, + Table_Name => "Prj.Variable_Elements"); + -- The table of variable in list of variables. + + type Array_Element_Id is new Nat; + No_Array_Element : constant Array_Element_Id := 0; + type Array_Element is record + Index : Name_Id; + Value : Variable_Value; + Next : Array_Element_Id := No_Array_Element; + end record; + -- Each Array_Element represents an array element. + -- Each Array_Element is linked (Next) to the next array element, + -- if any, in the array. + + package Array_Elements is new Table.Table + (Table_Component_Type => Array_Element, + Table_Index_Type => Array_Element_Id, + Table_Low_Bound => 1, + Table_Initial => 200, + Table_Increment => 100, + Table_Name => "Prj.Array_Elements"); + -- The table that contains all array elements + + type Array_Id is new Nat; + No_Array : constant Array_Id := 0; + type Array_Data is record + Name : Name_Id := No_Name; + Value : Array_Element_Id := No_Array_Element; + Next : Array_Id := No_Array; + end record; + -- Each Array_Data represents an array. + -- Value is the id of the first element. + -- Next is the id of the next array in the project file or package. + + package Arrays is new Table.Table + (Table_Component_Type => Array_Data, + Table_Index_Type => Array_Id, + Table_Low_Bound => 1, + Table_Initial => 200, + Table_Increment => 100, + Table_Name => "Prj.Arrays"); + -- The table that contains all arrays + + type Package_Id is new Nat; + No_Package : constant Package_Id := 0; + type Declarations is record + Variables : Variable_Id := No_Variable; + Attributes : Variable_Id := No_Variable; + Arrays : Array_Id := No_Array; + Packages : Package_Id := No_Package; + end record; + + No_Declarations : constant Declarations := + (Variables => No_Variable, + Attributes => No_Variable, + Arrays => No_Array, + Packages => No_Package); + -- Declarations. Used in project structures and packages. + + type Package_Element is record + Name : Name_Id := No_Name; + Decl : Declarations := No_Declarations; + Parent : Package_Id := No_Package; + Next : Package_Id := No_Package; + end record; + -- A package. Includes declarations that may include + -- other packages. + + package Packages is new Table.Table + (Table_Component_Type => Package_Element, + Table_Index_Type => Package_Id, + Table_Low_Bound => 1, + Table_Initial => 100, + Table_Increment => 100, + Table_Name => "Prj.Packages"); + -- The table that contains all packages. + + function Image (Casing : Casing_Type) return String; + -- Similar to 'Image + + function Value (Image : String) return Casing_Type; + -- Similar to 'Value + -- This is to avoid s-valenu in the closure of the tools + -- Raises Constraint_Error if not a Casing_Type image. + + type Naming_Data is record + Dot_Replacement : Name_Id := No_Name; + -- The string to replace '.' in the source file name. + + Dot_Repl_Loc : Source_Ptr := No_Location; + -- The position in the project file source where + -- Dot_Replacement is defined. + + Casing : Casing_Type := All_Lower_Case; + -- The casing of the source file name. + + Specification_Append : Name_Id := No_Name; + -- The string to append to the unit name for the + -- source file name of a specification. + + Spec_Append_Loc : Source_Ptr := No_Location; + -- The position in the project file source where + -- Specification_Append is defined. + + Body_Append : Name_Id := No_Name; + -- The string to append to the unit name for the + -- source file name of a body. + + Body_Append_Loc : Source_Ptr := No_Location; + -- The position in the project file source where + -- Body_Append is defined. + + Separate_Append : Name_Id := No_Name; + -- The string to append to the unit name for the + -- source file name of a subunit. + + Sep_Append_Loc : Source_Ptr := No_Location; + -- The position in the project file source where + -- Separate_Append is defined. + + Specifications : Array_Element_Id := No_Array_Element; + -- An associative array mapping individual specifications + -- to source file names. + + Bodies : Array_Element_Id := No_Array_Element; + -- An associative array mapping individual bodies + -- to source file names. + + end record; + -- A naming scheme. + + function Standard_Naming_Data return Naming_Data; + pragma Inline (Standard_Naming_Data); + -- The standard GNAT naming scheme. + + function Same_Naming_Scheme + (Left, Right : Naming_Data) + return Boolean; + -- Returns True if Left and Right are the same naming scheme + -- not considering Specifications and Bodies. + + type Project_Id is new Nat; + No_Project : constant Project_Id := 0; + -- Id of a Project File + + type Project_List is new Nat; + Empty_Project_List : constant Project_List := 0; + -- A list of project files. + + type Project_Element is record + Project : Project_Id := No_Project; + Next : Project_List := Empty_Project_List; + end record; + -- Element in a list of project file. + -- Next is the id of the next project file in the list. + + package Project_Lists is new Table.Table + (Table_Component_Type => Project_Element, + Table_Index_Type => Project_List, + Table_Low_Bound => 1, + Table_Initial => 100, + Table_Increment => 100, + Table_Name => "Prj.Project_Lists"); + -- The table that contains the lists of project files. + + type Project_Data is record + First_Referred_By : Project_Id := No_Project; + -- The project, if any, that was the first to be known + -- as importing or modifying this project. + + Name : Name_Id := No_Name; + -- The name of the project. + + Path_Name : Name_Id := No_Name; + -- The path name of the project file. + + Location : Source_Ptr := No_Location; + -- The location in the project file source of the + -- reserved word project. + + Directory : Name_Id := No_Name; + -- The directory where the project file resides. + + File_Name : Name_Id := No_Name; + -- The file name of the project file. + + Library : Boolean := False; + -- True if this is a library project + + Library_Dir : Name_Id := No_Name; + -- If a library project, directory where resides the library + + Library_Name : Name_Id := No_Name; + -- If a library project, name of the library + + Library_Kind : Lib_Kind := Static; + -- If a library project, kind of library + + Lib_Internal_Name : Name_Id := No_Name; + -- If a library project, internal name store inside the library + + Lib_Elaboration : Boolean := False; + -- If a library project, indicate if <lib>init and <lib>final + -- procedures need to be defined. + + Sources : String_List_Id := Nil_String; + -- The list of all the source file names. + + Source_Dirs : String_List_Id := Nil_String; + -- The list of all the source directories. + + Object_Directory : Name_Id := No_Name; + -- The object directory of this project file. + + Modifies : Project_Id := No_Project; + -- The reference of the project file, if any, that this + -- project file modifies. + + Modified_By : Project_Id := No_Project; + -- The reference of the project file, if any, that + -- modifies this project file. + + Naming : Naming_Data := Standard_Naming_Data; + -- The naming scheme of this project file. + + Decl : Declarations := No_Declarations; + -- The declarations (variables, attributes and packages) + -- of this project file. + + Imported_Projects : Project_List := Empty_Project_List; + -- The list of all directly imported projects, if any. + + Include_Path : String_Access := null; + -- The cached value of ADA_INCLUDE_PATH for this project file. + + Objects_Path : String_Access := null; + -- The cached value of ADA_OBJECTS_PATH for this project file. + + Config_File_Name : Name_Id := No_Name; + -- The name of the configuration pragmas file, if any. + + Config_File_Temp : Boolean := False; + -- An indication that the configuration pragmas file is + -- a temporary file that must be deleted at the end. + + Config_Checked : Boolean := False; + -- A flag to avoid checking repetively the configuration pragmas file. + + Checked : Boolean := False; + -- A flag to avoid checking repetively the naming scheme of + -- this project file. + + -- Various flags that are used in an ad hoc manner + + Seen : Boolean := False; + Flag1 : Boolean := False; + Flag2 : Boolean := False; + + end record; + -- Project File representation. + + function Empty_Project return Project_Data; + -- Return the representation of an empty project. + + package Projects is new Table.Table ( + Table_Component_Type => Project_Data, + Table_Index_Type => Project_Id, + Table_Low_Bound => 1, + Table_Initial => 100, + Table_Increment => 100, + Table_Name => "Prj.Projects"); + -- The set of all project files. + + procedure Expect (The_Token : Token_Type; Token_Image : String); + -- Check that the current token is The_Token. If it is not, then + -- output an error message. + + procedure Initialize; + -- This procedure must be called before using any services from the Prj + -- hierarchy. Namet.Initialize must be called before Prj.Initialize. + + procedure Reset; + -- This procedure resets all the tables that are used when processing a + -- project file tree. Initialize must be called before the call to Reset. + + generic + type State is limited private; + with procedure Action + (Project : Project_Id; + With_State : in out State); + procedure For_Every_Project_Imported + (By : Project_Id; + With_State : in out State); + -- Call Action for each project imported directly or indirectly by project + -- By.-- Action is called according to the order of importation: if A + -- imports B, directly or indirectly, Action will be called for A before + -- it is called for B. With_State may be used by Action to choose a + -- behavior or to report some global result. + +private + + procedure Scan; + -- Calls Scn.Scan and change any Operator_Symbol to String_Literal + +end Prj; diff --git a/gcc/ada/raise.c b/gcc/ada/raise.c new file mode 100644 index 00000000000..43d630795a8 --- /dev/null +++ b/gcc/ada/raise.c @@ -0,0 +1,86 @@ +/**************************************************************************** + * * + * GNAT COMPILER COMPONENTS * + * * + * R A I S E * + * * + * C Implementation File * + * * + * $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. * + * * + * As a special exception, if you link this file with other files to * + * produce an executable, this file does not by itself cause the resulting * + * executable to be covered by the GNU General Public License. This except- * + * ion does not however invalidate any other reasons why the executable * + * file might be covered by the GNU Public License. * + * * + * 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). * + * * + ****************************************************************************/ + +/* Routines to support runtime exception handling */ + +#ifdef IN_RTS +#include "tconfig.h" +#include "tsystem.h" +#include <sys/stat.h> +#else +#include "config.h" +#include "system.h" +#endif + +#include "adaint.h" +#include "raise.h" + +/* We have not yet figured out how to import this directly */ + +void +_gnat_builtin_longjmp (ptr, flag) + void *ptr; + int flag ATTRIBUTE_UNUSED; +{ + __builtin_longjmp (ptr, 1); +} + +/* When an exception is raised for which no handler exists, the procedure + Ada.Exceptions.Unhandled_Exception is called, which performs the call to + adafinal to complete finalization, and then prints out the error messages + for the unhandled exception. The final step is to call this routine, which + performs any system dependent cleanup required. */ + +void +__gnat_unhandled_terminate () +{ + /* Special termination handling for VMS */ + +#ifdef VMS + { + long prvhnd; + + /* Remove the exception vector so it won't intercept any errors + in the call to exit, and go into and endless loop */ + + SYS$SETEXV (1, 0, 3, &prvhnd); + __gnat_os_exit (1); + } + +/* Termination handling for all other systems. */ + +#elif !defined (__RT__) + __gnat_os_exit (1); +#endif +} diff --git a/gcc/ada/raise.h b/gcc/ada/raise.h new file mode 100644 index 00000000000..8db83f4a2b8 --- /dev/null +++ b/gcc/ada/raise.h @@ -0,0 +1,71 @@ +/**************************************************************************** + * * + * GNAT COMPILER COMPONENTS * + * * + * R A I S E * + * * + * C Header File * + * * + * $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. * + * * + * As a special exception, if you link this file with other files to * + * produce an executable, this file does not by itself cause the resulting * + * executable to be covered by the GNU General Public License. This except- * + * ion does not however invalidate any other reasons why the executable * + * file might be covered by the GNU Public License. * + * * + * 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). * + * * + ****************************************************************************/ + +struct Exception_Data +{ + char Handled_By_Others; + char Lang; + int Name_Length; + char *Full_Name, Htable_Ptr; + int Import_Code; +}; + +typedef struct Exception_Data *Exception_Id; + +struct Exception_Occurrence +{ + int Max_Length; + Exception_Id Id; + int Msg_Length; + char Msg [0]; +}; + +typedef struct Exception_Occurrence *Exception_Occurrence_Access; + +extern void _gnat_builtin_longjmp PARAMS ((void *, int)); +extern void __gnat_unhandled_terminate PARAMS ((void)); +extern void *__gnat_malloc PARAMS ((__SIZE_TYPE__)); +extern void __gnat_free PARAMS ((void *)); +extern void *__gnat_realloc PARAMS ((void *, __SIZE_TYPE__)); +extern void __gnat_finalize PARAMS ((void)); +extern void set_gnat_exit_status PARAMS ((int)); +extern void __gnat_set_globals PARAMS ((int, int, int, int, int, int, + void (*) PARAMS ((void)), + int, int)); +extern void __gnat_initialize PARAMS ((void)); +extern void __gnat_init_float PARAMS ((void)); +extern void __gnat_install_handler PARAMS ((void)); + +extern int gnat_exit_status; + diff --git a/gcc/ada/repinfo.adb b/gcc/ada/repinfo.adb new file mode 100644 index 00000000000..9e711527e5b --- /dev/null +++ b/gcc/ada/repinfo.adb @@ -0,0 +1,1024 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- R E P I N F O -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.36 $ +-- -- +-- Copyright (C) 1999-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. -- +-- -- +-- As a special exception, if other files instantiate generics from this -- +-- unit, or you link this unit with other files to produce an executable, -- +-- this unit does not by itself cause the resulting executable to be -- +-- covered by the GNU General Public License. This exception does not -- +-- however invalidate any other reasons why the executable file might be -- +-- covered by the GNU Public License. -- +-- -- +-- 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 Alloc; use Alloc; +with Atree; use Atree; +with Casing; use Casing; +with Debug; use Debug; +with Einfo; use Einfo; +with Lib; use Lib; +with Namet; use Namet; +with Opt; use Opt; +with Output; use Output; +with Sinfo; use Sinfo; +with Sinput; use Sinput; +with Table; use Table; +with Uname; use Uname; +with Urealp; use Urealp; + +package body Repinfo is + + SSU : constant := 8; + -- Value for Storage_Unit, we do not want to get this from TTypes, since + -- this introduces problematic dependencies in ASIS, and in any case this + -- value is assumed to be 8 for the implementation of the DDA. + -- This is wrong for AAMP??? + + --------------------------------------- + -- Representation of gcc Expressions -- + --------------------------------------- + + -- This table is used only if Frontend_Layout_On_Target is False, + -- so that gigi lays out dynamic size/offset fields using encoded + -- gcc expressions. + + -- A table internal to this unit is used to hold the values of + -- back annotated expressions. This table is written out by -gnatt + -- and read back in for ASIS processing. + + -- Node values are stored as Uint values which are the negative of + -- the node index in this table. Constants appear as non-negative + -- Uint values. + + type Exp_Node is record + Expr : TCode; + Op1 : Node_Ref_Or_Val; + Op2 : Node_Ref_Or_Val; + Op3 : Node_Ref_Or_Val; + end record; + + package Rep_Table is new Table.Table ( + Table_Component_Type => Exp_Node, + Table_Index_Type => Nat, + Table_Low_Bound => 1, + Table_Initial => Alloc.Rep_Table_Initial, + Table_Increment => Alloc.Rep_Table_Increment, + Table_Name => "BE_Rep_Table"); + + -------------------------------------------------------------- + -- Representation of Front-End Dynamic Size/Offset Entities -- + -------------------------------------------------------------- + + package Dynamic_SO_Entity_Table is new Table.Table ( + Table_Component_Type => Entity_Id, + Table_Index_Type => Nat, + Table_Low_Bound => 1, + Table_Initial => Alloc.Rep_Table_Initial, + Table_Increment => Alloc.Rep_Table_Increment, + Table_Name => "FE_Rep_Table"); + + ----------------------- + -- Local Subprograms -- + ----------------------- + + Unit_Casing : Casing_Type; + -- Indentifier casing for current unit + + procedure Spaces (N : Natural); + -- Output given number of spaces + + function Back_End_Layout return Boolean; + -- Test for layout mode, True = back end, False = front end. This + -- function is used rather than checking the configuration parameter + -- because we do not want Repinfo to depend on Targparm (for ASIS) + + procedure List_Entities (Ent : Entity_Id); + -- This procedure lists the entities associated with the entity E, + -- starting with the First_Entity and using the Next_Entity link. + -- If a nested package is found, entities within the package are + -- recursively processed. + + procedure List_Name (Ent : Entity_Id); + -- List name of entity Ent in appropriate case. The name is listed with + -- full qualification up to but not including the compilation unit name. + + procedure List_Array_Info (Ent : Entity_Id); + -- List representation info for array type Ent + + procedure List_Object_Info (Ent : Entity_Id); + -- List representation info for object Ent + + procedure List_Record_Info (Ent : Entity_Id); + -- List representation info for record type Ent + + procedure List_Type_Info (Ent : Entity_Id); + -- List type info for type Ent + + function Rep_Not_Constant (Val : Node_Ref_Or_Val) return Boolean; + -- Returns True if Val represents a variable value, and False if it + -- represents a value that is fixed at compile time. + + procedure Write_Val (Val : Node_Ref_Or_Val; Paren : Boolean := False); + -- Given a representation value, write it out. No_Uint values or values + -- dependent on discriminants are written as two question marks. If the + -- flag Paren is set, then the output is surrounded in parentheses if + -- it is other than a simple value. + + --------------------- + -- Back_End_Layout -- + --------------------- + + function Back_End_Layout return Boolean is + begin + -- We have back end layout if the back end has made any entries in + -- the table of GCC expressions, otherwise we have front end layout. + + return Rep_Table.Last > 0; + end Back_End_Layout; + + ------------------------ + -- Create_Discrim_Ref -- + ------------------------ + + function Create_Discrim_Ref + (Discr : Entity_Id) + return Node_Ref + is + N : constant Uint := Discriminant_Number (Discr); + T : Nat; + + begin + Rep_Table.Increment_Last; + T := Rep_Table.Last; + Rep_Table.Table (T).Expr := Discrim_Val; + Rep_Table.Table (T).Op1 := N; + Rep_Table.Table (T).Op2 := No_Uint; + Rep_Table.Table (T).Op3 := No_Uint; + return UI_From_Int (-T); + end Create_Discrim_Ref; + + --------------------------- + -- Create_Dynamic_SO_Ref -- + --------------------------- + + function Create_Dynamic_SO_Ref + (E : Entity_Id) + return Dynamic_SO_Ref + is + T : Nat; + + begin + Dynamic_SO_Entity_Table.Increment_Last; + T := Dynamic_SO_Entity_Table.Last; + Dynamic_SO_Entity_Table.Table (T) := E; + return UI_From_Int (-T); + end Create_Dynamic_SO_Ref; + + ----------------- + -- Create_Node -- + ----------------- + + function Create_Node + (Expr : TCode; + Op1 : Node_Ref_Or_Val; + Op2 : Node_Ref_Or_Val := No_Uint; + Op3 : Node_Ref_Or_Val := No_Uint) + return Node_Ref + is + T : Nat; + + begin + Rep_Table.Increment_Last; + T := Rep_Table.Last; + Rep_Table.Table (T).Expr := Expr; + Rep_Table.Table (T).Op1 := Op1; + Rep_Table.Table (T).Op2 := Op2; + Rep_Table.Table (T).Op3 := Op3; + + return UI_From_Int (-T); + end Create_Node; + + --------------------------- + -- Get_Dynamic_SO_Entity -- + --------------------------- + + function Get_Dynamic_SO_Entity + (U : Dynamic_SO_Ref) + return Entity_Id + is + begin + return Dynamic_SO_Entity_Table.Table (-UI_To_Int (U)); + end Get_Dynamic_SO_Entity; + + ----------------------- + -- Is_Dynamic_SO_Ref -- + ----------------------- + + function Is_Dynamic_SO_Ref (U : SO_Ref) return Boolean is + begin + return U < Uint_0; + end Is_Dynamic_SO_Ref; + + ---------------------- + -- Is_Static_SO_Ref -- + ---------------------- + + function Is_Static_SO_Ref (U : SO_Ref) return Boolean is + begin + return U >= Uint_0; + end Is_Static_SO_Ref; + + --------- + -- lgx -- + --------- + + procedure lgx (U : Node_Ref_Or_Val) is + begin + List_GCC_Expression (U); + Write_Eol; + end lgx; + + ---------------------- + -- List_Array_Info -- + ---------------------- + + procedure List_Array_Info (Ent : Entity_Id) is + begin + List_Type_Info (Ent); + + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Component_Size use "); + Write_Val (Component_Size (Ent)); + Write_Line (";"); + end List_Array_Info; + + ------------------- + -- List_Entities -- + ------------------- + + procedure List_Entities (Ent : Entity_Id) is + E : Entity_Id; + + begin + if Present (Ent) then + E := First_Entity (Ent); + while Present (E) loop + if Comes_From_Source (E) or else Debug_Flag_AA then + + if Is_Record_Type (E) then + List_Record_Info (E); + + elsif Is_Array_Type (E) then + List_Array_Info (E); + + elsif List_Representation_Info >= 2 then + + if Is_Type (E) then + List_Type_Info (E); + + elsif Ekind (E) = E_Variable + or else + Ekind (E) = E_Constant + or else + Ekind (E) = E_Loop_Parameter + or else + Is_Formal (E) + then + List_Object_Info (E); + end if; + end if; + + -- Recurse over nested package, but not if they are + -- package renamings (in particular renamings of the + -- enclosing package, as for some Java bindings and + -- for generic instances). + + if (Ekind (E) = E_Package + and then No (Renamed_Object (E))) + or else + Ekind (E) = E_Protected_Type + or else + Ekind (E) = E_Task_Type + or else + Ekind (E) = E_Subprogram_Body + or else + Ekind (E) = E_Package_Body + or else + Ekind (E) = E_Task_Body + or else + Ekind (E) = E_Protected_Body + then + List_Entities (E); + end if; + end if; + + E := Next_Entity (E); + end loop; + end if; + end List_Entities; + + ------------------------- + -- List_GCC_Expression -- + ------------------------- + + procedure List_GCC_Expression (U : Node_Ref_Or_Val) is + + procedure P (Val : Node_Ref_Or_Val); + -- Internal recursive procedure to print expression + + procedure P (Val : Node_Ref_Or_Val) is + begin + if Val >= 0 then + UI_Write (Val, Decimal); + + else + declare + Node : Exp_Node renames Rep_Table.Table (-UI_To_Int (Val)); + + procedure Binop (S : String); + -- Output text for binary operator with S being operator name + + procedure Binop (S : String) is + begin + Write_Char ('('); + P (Node.Op1); + Write_Str (S); + P (Node.Op2); + Write_Char (')'); + end Binop; + + -- Start of processing for P + + begin + case Node.Expr is + when Cond_Expr => + Write_Str ("(if "); + P (Node.Op1); + Write_Str (" then "); + P (Node.Op2); + Write_Str (" else "); + P (Node.Op3); + Write_Str (" end)"); + + when Plus_Expr => + Binop (" + "); + + when Minus_Expr => + Binop (" - "); + + when Mult_Expr => + Binop (" * "); + + when Trunc_Div_Expr => + Binop (" /t "); + + when Ceil_Div_Expr => + Binop (" /c "); + + when Floor_Div_Expr => + Binop (" /f "); + + when Trunc_Mod_Expr => + Binop (" modt "); + + when Floor_Mod_Expr => + Binop (" modf "); + + when Ceil_Mod_Expr => + Binop (" modc "); + + when Exact_Div_Expr => + Binop (" /e "); + + when Negate_Expr => + Write_Char ('-'); + P (Node.Op1); + + when Min_Expr => + Binop (" min "); + + when Max_Expr => + Binop (" max "); + + when Abs_Expr => + Write_Str ("abs "); + P (Node.Op1); + + when Truth_Andif_Expr => + Binop (" and if "); + + when Truth_Orif_Expr => + Binop (" or if "); + + when Truth_And_Expr => + Binop (" and "); + + when Truth_Or_Expr => + Binop (" or "); + + when Truth_Xor_Expr => + Binop (" xor "); + + when Truth_Not_Expr => + Write_Str ("not "); + P (Node.Op1); + + when Lt_Expr => + Binop (" < "); + + when Le_Expr => + Binop (" <= "); + + when Gt_Expr => + Binop (" > "); + + when Ge_Expr => + Binop (" >= "); + + when Eq_Expr => + Binop (" == "); + + when Ne_Expr => + Binop (" != "); + + when Discrim_Val => + Write_Char ('#'); + UI_Write (Node.Op1); + + end case; + end; + end if; + end P; + + -- Start of processing for List_GCC_Expression + + begin + if U = No_Uint then + Write_Line ("??"); + else + P (U); + end if; + end List_GCC_Expression; + + --------------- + -- List_Name -- + --------------- + + procedure List_Name (Ent : Entity_Id) is + begin + if not Is_Compilation_Unit (Scope (Ent)) then + List_Name (Scope (Ent)); + Write_Char ('.'); + end if; + + Get_Unqualified_Decoded_Name_String (Chars (Ent)); + Set_Casing (Unit_Casing); + Write_Str (Name_Buffer (1 .. Name_Len)); + end List_Name; + + --------------------- + -- List_Object_Info -- + --------------------- + + procedure List_Object_Info (Ent : Entity_Id) is + begin + Write_Eol; + + if Known_Esize (Ent) then + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Size use "); + Write_Val (Esize (Ent)); + Write_Line (";"); + end if; + + if Known_Alignment (Ent) then + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Alignment use "); + Write_Val (Alignment (Ent)); + Write_Line (";"); + end if; + end List_Object_Info; + + ---------------------- + -- List_Record_Info -- + ---------------------- + + procedure List_Record_Info (Ent : Entity_Id) is + Comp : Entity_Id; + Esiz : Uint; + Cfbit : Uint; + Sunit : Uint; + + Max_Name_Length : Natural; + Max_Suni_Length : Natural; + + begin + List_Type_Info (Ent); + + Write_Str ("for "); + List_Name (Ent); + Write_Line (" use record"); + + -- First loop finds out max line length and max starting position + -- length, for the purpose of lining things up nicely. + + Max_Name_Length := 0; + Max_Suni_Length := 0; + + Comp := First_Entity (Ent); + while Present (Comp) loop + if Ekind (Comp) = E_Component + or else Ekind (Comp) = E_Discriminant + then + Get_Decoded_Name_String (Chars (Comp)); + Max_Name_Length := Natural'Max (Max_Name_Length, Name_Len); + + Cfbit := Component_Bit_Offset (Comp); + + if Rep_Not_Constant (Cfbit) then + UI_Image_Length := 2; + + else + -- Complete annotation in case not done + + Set_Normalized_Position (Comp, Cfbit / SSU); + Set_Normalized_First_Bit (Comp, Cfbit mod SSU); + + Esiz := Esize (Comp); + Sunit := Cfbit / SSU; + UI_Image (Sunit); + end if; + + if Unknown_Normalized_First_Bit (Comp) then + Set_Normalized_First_Bit (Comp, Uint_0); + end if; + + Max_Suni_Length := + Natural'Max (Max_Suni_Length, UI_Image_Length); + end if; + + Comp := Next_Entity (Comp); + end loop; + + -- Second loop does actual output based on those values + + Comp := First_Entity (Ent); + while Present (Comp) loop + if Ekind (Comp) = E_Component + or else Ekind (Comp) = E_Discriminant + then + declare + Esiz : constant Uint := Esize (Comp); + Bofs : constant Uint := Component_Bit_Offset (Comp); + Npos : constant Uint := Normalized_Position (Comp); + Fbit : constant Uint := Normalized_First_Bit (Comp); + Lbit : Uint; + + begin + Write_Str (" "); + Get_Decoded_Name_String (Chars (Comp)); + Set_Casing (Unit_Casing); + Write_Str (Name_Buffer (1 .. Name_Len)); + + for J in 1 .. Max_Name_Length - Name_Len loop + Write_Char (' '); + end loop; + + Write_Str (" at "); + + if Known_Static_Normalized_Position (Comp) then + UI_Image (Npos); + Spaces (Max_Suni_Length - UI_Image_Length); + Write_Str (UI_Image_Buffer (1 .. UI_Image_Length)); + + elsif Known_Component_Bit_Offset (Comp) + and then List_Representation_Info = 3 + then + Spaces (Max_Suni_Length - 2); + Write_Val (Bofs, Paren => True); + Write_Str (" / 8"); + + elsif Known_Normalized_Position (Comp) + and then List_Representation_Info = 3 + then + Spaces (Max_Suni_Length - 2); + Write_Val (Npos); + + else + Write_Str ("??"); + end if; + + Write_Str (" range "); + UI_Write (Fbit); + Write_Str (" .. "); + + if not Is_Dynamic_SO_Ref (Esize (Comp)) then + Lbit := Fbit + Esiz - 1; + + if Lbit < 10 then + Write_Char (' '); + end if; + + UI_Write (Lbit); + + elsif List_Representation_Info < 3 then + Write_Str ("??"); + + else -- List_Representation >= 3 + + Write_Val (Esiz, Paren => True); + + -- If in front end layout mode, then dynamic size is + -- stored in storage units, so renormalize for output + + if not Back_End_Layout then + Write_Str (" * "); + Write_Int (SSU); + end if; + + -- Add appropriate first bit offset + + if Fbit = 0 then + Write_Str (" - 1"); + + elsif Fbit = 1 then + null; + + else + Write_Str (" + "); + Write_Int (UI_To_Int (Fbit) - 1); + end if; + end if; + + Write_Line (";"); + end; + end if; + + Comp := Next_Entity (Comp); + end loop; + + Write_Line ("end record;"); + end List_Record_Info; + + ------------------- + -- List_Rep_Info -- + ------------------- + + procedure List_Rep_Info is + Col : Nat; + + begin + for U in Main_Unit .. Last_Unit loop + if In_Extended_Main_Source_Unit (Cunit_Entity (U)) then + Unit_Casing := Identifier_Casing (Source_Index (U)); + Write_Eol; + Write_Str ("Representation information for unit "); + Write_Unit_Name (Unit_Name (U)); + Col := Column; + Write_Eol; + + for J in 1 .. Col - 1 loop + Write_Char ('-'); + end loop; + + Write_Eol; + List_Entities (Cunit_Entity (U)); + end if; + end loop; + end List_Rep_Info; + + -------------------- + -- List_Type_Info -- + -------------------- + + procedure List_Type_Info (Ent : Entity_Id) is + begin + Write_Eol; + + -- If Esize and RM_Size are the same and known, list as Size. This + -- is a common case, which we may as well list in simple form. + + if Esize (Ent) = RM_Size (Ent) then + if Known_Esize (Ent) then + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Size use "); + Write_Val (Esize (Ent)); + Write_Line (";"); + end if; + + -- For now, temporary case, to be removed when gigi properly back + -- annotates RM_Size, if RM_Size is not set, then list Esize as + -- Size. This avoids odd Object_Size output till we fix things??? + + elsif Unknown_RM_Size (Ent) then + if Known_Esize (Ent) then + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Size use "); + Write_Val (Esize (Ent)); + Write_Line (";"); + end if; + + -- Otherwise list size values separately if they are set + + else + if Known_Esize (Ent) then + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Object_Size use "); + Write_Val (Esize (Ent)); + Write_Line (";"); + end if; + + -- Note on following check: The RM_Size of a discrete type can + -- legitimately be set to zero, so a special check is needed. + + if Known_RM_Size (Ent) or else Is_Discrete_Type (Ent) then + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Value_Size use "); + Write_Val (RM_Size (Ent)); + Write_Line (";"); + end if; + end if; + + if Known_Alignment (Ent) then + Write_Str ("for "); + List_Name (Ent); + Write_Str ("'Alignment use "); + Write_Val (Alignment (Ent)); + Write_Line (";"); + end if; + end List_Type_Info; + + ---------------------- + -- Rep_Not_Constant -- + ---------------------- + + function Rep_Not_Constant (Val : Node_Ref_Or_Val) return Boolean is + begin + if Val = No_Uint or else Val < 0 then + return True; + else + return False; + end if; + end Rep_Not_Constant; + + --------------- + -- Rep_Value -- + --------------- + + function Rep_Value + (Val : Node_Ref_Or_Val; + D : Discrim_List) + return Uint + is + function B (Val : Boolean) return Uint; + -- Returns Uint_0 for False, Uint_1 for True + + function T (Val : Node_Ref_Or_Val) return Boolean; + -- Returns True for 0, False for any non-zero (i.e. True) + + function V (Val : Node_Ref_Or_Val) return Uint; + -- Internal recursive routine to evaluate tree + + ------- + -- B -- + ------- + + function B (Val : Boolean) return Uint is + begin + if Val then + return Uint_1; + else + return Uint_0; + end if; + end B; + + ------- + -- T -- + ------- + + function T (Val : Node_Ref_Or_Val) return Boolean is + begin + if V (Val) = 0 then + return False; + else + return True; + end if; + end T; + + ------- + -- V -- + ------- + + function V (Val : Node_Ref_Or_Val) return Uint is + L, R, Q : Uint; + + begin + if Val >= 0 then + return Val; + + else + declare + Node : Exp_Node renames Rep_Table.Table (-UI_To_Int (Val)); + + begin + case Node.Expr is + when Cond_Expr => + if T (Node.Op1) then + return V (Node.Op2); + else + return V (Node.Op3); + end if; + + when Plus_Expr => + return V (Node.Op1) + V (Node.Op2); + + when Minus_Expr => + return V (Node.Op1) - V (Node.Op2); + + when Mult_Expr => + return V (Node.Op1) * V (Node.Op2); + + when Trunc_Div_Expr => + return V (Node.Op1) / V (Node.Op2); + + when Ceil_Div_Expr => + return + UR_Ceiling + (V (Node.Op1) / UR_From_Uint (V (Node.Op2))); + + when Floor_Div_Expr => + return + UR_Floor + (V (Node.Op1) / UR_From_Uint (V (Node.Op2))); + + when Trunc_Mod_Expr => + return V (Node.Op1) rem V (Node.Op2); + + when Floor_Mod_Expr => + return V (Node.Op1) mod V (Node.Op2); + + when Ceil_Mod_Expr => + L := V (Node.Op1); + R := V (Node.Op2); + Q := UR_Ceiling (L / UR_From_Uint (R)); + return L - R * Q; + + when Exact_Div_Expr => + return V (Node.Op1) / V (Node.Op2); + + when Negate_Expr => + return -V (Node.Op1); + + when Min_Expr => + return UI_Min (V (Node.Op1), V (Node.Op2)); + + when Max_Expr => + return UI_Max (V (Node.Op1), V (Node.Op2)); + + when Abs_Expr => + return UI_Abs (V (Node.Op1)); + + when Truth_Andif_Expr => + return B (T (Node.Op1) and then T (Node.Op2)); + + when Truth_Orif_Expr => + return B (T (Node.Op1) or else T (Node.Op2)); + + when Truth_And_Expr => + return B (T (Node.Op1) and T (Node.Op2)); + + when Truth_Or_Expr => + return B (T (Node.Op1) or T (Node.Op2)); + + when Truth_Xor_Expr => + return B (T (Node.Op1) xor T (Node.Op2)); + + when Truth_Not_Expr => + return B (not T (Node.Op1)); + + when Lt_Expr => + return B (V (Node.Op1) < V (Node.Op2)); + + when Le_Expr => + return B (V (Node.Op1) <= V (Node.Op2)); + + when Gt_Expr => + return B (V (Node.Op1) > V (Node.Op2)); + + when Ge_Expr => + return B (V (Node.Op1) >= V (Node.Op2)); + + when Eq_Expr => + return B (V (Node.Op1) = V (Node.Op2)); + + when Ne_Expr => + return B (V (Node.Op1) /= V (Node.Op2)); + + when Discrim_Val => + declare + Sub : constant Int := UI_To_Int (Node.Op1); + + begin + pragma Assert (Sub in D'Range); + return D (Sub); + end; + + end case; + end; + end if; + end V; + + -- Start of processing for Rep_Value + + begin + if Val = No_Uint then + return No_Uint; + + else + return V (Val); + end if; + end Rep_Value; + + ------------ + -- Spaces -- + ------------ + + procedure Spaces (N : Natural) is + begin + for J in 1 .. N loop + Write_Char (' '); + end loop; + end Spaces; + + --------------- + -- Tree_Read -- + --------------- + + procedure Tree_Read is + begin + Rep_Table.Tree_Read; + end Tree_Read; + + ---------------- + -- Tree_Write -- + ---------------- + + procedure Tree_Write is + begin + Rep_Table.Tree_Write; + end Tree_Write; + + --------------- + -- Write_Val -- + --------------- + + procedure Write_Val (Val : Node_Ref_Or_Val; Paren : Boolean := False) is + begin + if Rep_Not_Constant (Val) then + if List_Representation_Info < 3 then + Write_Str ("??"); + else + if Back_End_Layout then + Write_Char (' '); + List_GCC_Expression (Val); + Write_Char (' '); + else + Write_Name_Decoded (Chars (Get_Dynamic_SO_Entity (Val))); + end if; + end if; + + else + UI_Write (Val); + end if; + end Write_Val; + +end Repinfo; diff --git a/gcc/ada/repinfo.ads b/gcc/ada/repinfo.ads new file mode 100644 index 00000000000..0b41ba0864b --- /dev/null +++ b/gcc/ada/repinfo.ads @@ -0,0 +1,320 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- R E P I N F O -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.20 $ +-- -- +-- Copyright (C) 1999-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. -- +-- -- +-- As a special exception, if other files instantiate generics from this -- +-- unit, or you link this unit with other files to produce an executable, -- +-- this unit does not by itself cause the resulting executable to be -- +-- covered by the GNU General Public License. This exception does not -- +-- however invalidate any other reasons why the executable file might be -- +-- covered by the GNU Public License. -- +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +-- This package contains the routines to handle back annotation of the +-- tree to fill in representation information, and also the routine used +-- by -gnatR to print this information. This unit is used both in the +-- compiler and in ASIS (it is used in ASIS as part of the implementation +-- of the data decomposition annex. + +with Types; use Types; +with Uintp; use Uintp; + +package Repinfo is + + -------------------------------- + -- Representation Information -- + -------------------------------- + + -- The representation information of interest here is size and + -- component information for arrays and records. For primitive + -- types, the front end computes the Esize and RM_Size fields of + -- the corresponding entities as constant non-negative integers, + -- and the Uint values are stored directly in these fields. + + -- For composite types, there are three cases: + + -- 1. In some cases the front end knows the values statically, + -- for example in the ase where representation clauses or + -- pragmas specify the values. + + -- 2. If Backend_Layout is True, then the backend is responsible + -- for layout of all types and objects not laid out by the + -- front end. This includes all dynamic values, and also + -- static values (e.g. record sizes) when not set by the + -- front end. + + -- 3. If Backend_Layout is False, then the front end lays out + -- all data, according to target dependent size and alignment + -- information, creating dynamic inlinable functions where + -- needed in the case of sizes not known till runtime. + + ----------------------------- + -- Back-Annotation by Gigi -- + ----------------------------- + + -- The following interface is used by gigi if Backend_Layout is True. + + -- As part of the processing in gigi, the types are laid out and + -- appropriate values computed for the sizes and component positions + -- and sizes of records and arrays. + + -- The back-annotation circuit in gigi is responsible for updating the + -- relevant fields in the tree to reflect these computations, as follows: + + -- For E_Array_Type entities, the Component_Size field + + -- For all record and array types and subtypes, the Esize field, + -- which contains the Size (more accurately the Object_SIze) value + -- for the type or subtype. + + -- For E_Component and E_Distriminant entities, the Esize (size + -- of component) and Component_Bit_Offset fields. Note that gigi + -- does not (yet ???) back annotate Normalized_Position/First_Bit. + + -- There are three cases to consider: + + -- 1. The value is constant. In this case, the back annotation works + -- by simply storing the non-negative universal integer value in + -- the appropriate field corresponding to this constant size. + + -- 2. The value depends on variables other than discriminants of the + -- current record. In this case, the value is not known, even if + -- the complete data of the record is available, and gigi marks + -- this situation by storing the special value No_Uint. + + -- 3. The value depends on the discriminant values for the current + -- record. In this case, gigi back annotates the field with a + -- representation of the expression for computing the value in + -- terms of the discriminants. A negative Uint value is used to + -- represent the value of such an expression, as explained in + -- the following section. + + -- GCC expressions are represented with a Uint value that is negative. + -- See the body of this package for details on the representation used. + + -- One other case in which gigi back annotates GCC expressions is in + -- the Present_Expr field of an N_Variant node. This expression which + -- will always depend on discriminants, and hence always be represented + -- as a negative Uint value, provides an expression which, when evaluated + -- with a given set of discriminant values, indicates whether the variant + -- is present for that set of values (result is True, i.e. non-zero) or + -- not present (result is False, i.e. zero). + + subtype Node_Ref is Uint; + -- Subtype used for negative Uint values used to represent nodes + + subtype Node_Ref_Or_Val is Uint; + -- Subtype used for values that can either be a Node_Ref (negative) + -- or a value (non-negative) + + type TCode is range 0 .. 27; + -- Type used on Ada side to represent DEFTREECODE values defined in + -- tree.def. Only a subset of these tree codes can actually appear. + -- The names are the names from tree.def in Ada casing. + + -- name code description operands + + Cond_Expr : constant TCode := 1; -- conditional 3 + Plus_Expr : constant TCode := 2; -- addition 2 + Minus_Expr : constant TCode := 3; -- subtraction 2 + Mult_Expr : constant TCode := 4; -- multiplication 2 + Trunc_Div_Expr : constant TCode := 5; -- truncating division 2 + Ceil_Div_Expr : constant TCode := 6; -- division rounding up 2 + Floor_Div_Expr : constant TCode := 7; -- division rounding down 2 + Trunc_Mod_Expr : constant TCode := 8; -- mod for trunc_div 2 + Ceil_Mod_Expr : constant TCode := 9; -- mod for ceil_div 2 + Floor_Mod_Expr : constant TCode := 10; -- mod for floor_div 2 + Exact_Div_Expr : constant TCode := 11; -- exact div 2 + Negate_Expr : constant TCode := 12; -- negation 1 + Min_Expr : constant TCode := 13; -- minimum 2 + Max_Expr : constant TCode := 14; -- maximum 2 + Abs_Expr : constant TCode := 15; -- absolute value 1 + Truth_Andif_Expr : constant TCode := 16; -- Boolean and then 2 + Truth_Orif_Expr : constant TCode := 17; -- Boolean or else 2 + Truth_And_Expr : constant TCode := 18; -- Boolean and 2 + Truth_Or_Expr : constant TCode := 19; -- Boolean or 2 + Truth_Xor_Expr : constant TCode := 20; -- Boolean xor 2 + Truth_Not_Expr : constant TCode := 21; -- Boolean not 1 + Lt_Expr : constant TCode := 22; -- comparision < 2 + Le_Expr : constant TCode := 23; -- comparision <= 2 + Gt_Expr : constant TCode := 24; -- comparision > 2 + Ge_Expr : constant TCode := 25; -- comparision >= 2 + Eq_Expr : constant TCode := 26; -- comparision = 2 + Ne_Expr : constant TCode := 27; -- comparision /= 2 + + -- The following entry is used to represent a discriminant value in + -- the tree. It has a special tree code that does not correspond + -- directly to a gcc node. The single operand is the number of the + -- discriminant in the record (1 = first discriminant). + + Discrim_Val : constant TCode := 0; -- discriminant value 1 + + ------------------------ + -- The gigi Interface -- + ------------------------ + + -- The following declarations are for use by gigi for back annotation + + function Create_Node + (Expr : TCode; + Op1 : Node_Ref_Or_Val; + Op2 : Node_Ref_Or_Val := No_Uint; + Op3 : Node_Ref_Or_Val := No_Uint) + return Node_Ref; + -- Creates a node with using the tree code defined by Expr and from + -- 1-3 operands as required (unused operands set as shown to No_Uint) + -- Note that this call can be used to create a discriminant reference + -- by using (Expr => Discrim_Val, Op1 => discriminant_number). + + function Create_Discrim_Ref + (Discr : Entity_Id) + return Node_Ref; + -- Creates a refrerence to the discriminant whose entity is Discr. + + -------------------------------------------------------- + -- Front-End Interface for Dynamic Size/Offset Values -- + -------------------------------------------------------- + + -- If Backend_Layout is False, then the front-end deals with all + -- dynamic size and offset fields. There are two cases: + + -- 1. The value can be computed at the time of type freezing, and + -- is stored in a run-time constant. In this case, the field + -- contains a reference to this entity. In the case of sizes + -- the value stored is the size in storage units, since dynamic + -- sizes are always a multiple of storage units. + + -- 2. The size/offset depends on the value of discriminants at + -- run-time. In this case, the front end builds a function to + -- compute the value. This function has a single parameter + -- which is the discriminated record object in question. Any + -- references to discriminant values are simply references to + -- the appropriate discriminant in this single argument, and + -- to compute the required size/offset value at run time, the + -- code generator simply constructs a call to the function + -- with the appropriate argument. The size/offset field in + -- this case contains a reference to the function entity. + -- Note that as for case 1, if such a function is used to + -- return a size, then the size in storage units is returned, + -- not the size in bits. + + -- The interface here allows these created entities to be referenced + -- using negative Unit values, so that they can be stored in the + -- appropriate size and offset fields in the tree. + + -- In the case of components, if the location of the component is static, + -- then all four fields (Component_Bit_Offset, Normalized_Position, Esize, + -- and Normalized_First_Bit) are set to appropraite values. In the case of + -- a non-static component location, Component_Bit_Offset is not used and + -- is left set to Unknown. Normalized_Position and Normalized_First_Bit + -- are set appropriately. + + subtype SO_Ref is Uint; + -- Type used to represent a Uint value that represents a static or + -- dynamic size/offset value (non-negative if static, negative if + -- the size value is dynamic). + + subtype Dynamic_SO_Ref is Uint; + -- Type used to represent a negative Uint value used to store + -- a dynamic size/offset value. + + function Is_Dynamic_SO_Ref (U : SO_Ref) return Boolean; + pragma Inline (Is_Dynamic_SO_Ref); + -- Given a SO_Ref (Uint) value, returns True iff the SO_Ref value + -- represents a dynamic Size/Offset value (i.e. it is negative). + + function Is_Static_SO_Ref (U : SO_Ref) return Boolean; + pragma Inline (Is_Static_SO_Ref); + -- Given a SO_Ref (Uint) value, returns True iff the SO_Ref value + -- represents a static Size/Offset value (i.e. it is non-negative). + + function Create_Dynamic_SO_Ref + (E : Entity_Id) + return Dynamic_SO_Ref; + -- Given the Entity_Id for a constant (case 1), the Node_Id for an + -- expression (case 2), or the Entity_Id for a function (case 3), + -- this function returns a (negative) Uint value that can be used + -- to retrieve the entity or expression for later use. + + function Get_Dynamic_SO_Entity + (U : Dynamic_SO_Ref) + return Entity_Id; + -- Retrieve the Node_Id or Entity_Id stored by a previous call to + -- Create_Dynamic_SO_Ref. The approach is that the front end makes + -- the necessary Create_Dynamic_SO_Ref calls to associate the node + -- and entity id values and the back end makes Get_Dynamic_SO_Ref + -- calls to retrive them. + + -------------------- + -- ASIS_Interface -- + -------------------- + + type Discrim_List is array (Pos range <>) of Uint; + -- Type used to represent list of discriminant values + + function Rep_Value + (Val : Node_Ref_Or_Val; + D : Discrim_List) + return Uint; + -- Given the contents of a First_Bit_Position or Esize field containing + -- a node reference (i.e. a negative Uint value) and D, the list of + -- discriminant values, returns the interpreted value of this field. + -- For convenience, Rep_Value will take a non-negative Uint value + -- as an argument value, and return it unmodified. A No_Uint value is + -- also returned unmodified. + + procedure Tree_Read; + -- Read in the value of the Rep_Table + + ------------------------ + -- Compiler Interface -- + ------------------------ + + procedure List_Rep_Info; + -- Procedure to list representation information + + procedure Tree_Write; + -- Write out the value of the Rep_Table + + -------------------------- + -- Debugging Procedures -- + -------------------------- + + procedure List_GCC_Expression (U : Node_Ref_Or_Val); + -- Prints out given expression in symbolic form. Constants are listed + -- in decimal numeric form, Discriminants are listed with a # followed + -- by the discriminant number, and operators are output in appropriate + -- symbolic form No_Uint displays as two question marks. The output is + -- on a single line but has no line return after it. This procedure is + -- useful only if operating in backend layout mode. + + procedure lgx (U : Node_Ref_Or_Val); + -- In backend layout mode, this is like List_GCC_Expression, but + -- includes a line return at the end. If operating in front end + -- layout mode, then the name of the entity for the size (either + -- a function of a variable) is listed followed by a line return. + +end Repinfo; diff --git a/gcc/ada/repinfo.h b/gcc/ada/repinfo.h new file mode 100644 index 00000000000..305c818685c --- /dev/null +++ b/gcc/ada/repinfo.h @@ -0,0 +1,79 @@ +/**************************************************************************** + * * + * GNAT COMPILER COMPONENTS * + * * + * R E P I N F O * + * * + * C Header File * + * * + * $Revision: 1.1 $ + * * + * Copyright (C) 1999-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. * + * * + * As a special exception, if you link this file with other files to * + * produce an executable, this file does not by itself cause the resulting * + * executable to be covered by the GNU General Public License. This except- * + * ion does not however invalidate any other reasons why the executable * + * file might be covered by the GNU Public License. * + * * + * 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). * + * * + ****************************************************************************/ + +/* This file corresponds to the Ada file repinfo.ads. */ + +typedef Uint Node_Ref; +typedef Uint Node_Ref_Or_Val; +typedef char TCode; + +/* These are the values of TCcode that correspond to tree codes in tree.def, + except for the first, which is how we encode discriminants. */ + +#define Discrim_Val 0 +#define Cond_Expr 1 +#define Plus_Expr 2 +#define Minus_Expr 3 +#define Mult_Expr 4 +#define Trunc_Div_Expr 5 +#define Ceil_Div_Expr 6 +#define Floor_Div_Expr 7 +#define Trunc_Mod_Expr 8 +#define Ceil_Mod_Expr 9 +#define Floor_Mod_Expr 10 +#define Exact_Div_Expr 11 +#define Negate_Expr 12 +#define Min_Expr 13 +#define Max_Expr 14 +#define Abs_Expr 15 +#define Truth_Andif_Expr 16 +#define Truth_Orif_Expr 17 +#define Truth_And_Expr 18 +#define Truth_Or_Expr 19 +#define Truth_Xor_Expr 20 +#define Truth_Not_Expr 21 +#define Lt_Expr 22 +#define Le_Expr 23 +#define Gt_Expr 24 +#define Ge_Expr 25 +#define Eq_Expr 26 +#define Ne_Expr 27 + +/* Creates a node using the tree code defined by Expr and from 1-3 + operands as required (unused operands set as shown to No_Uint) Note + that this call can be used to create a discriminant reference by + using (Expr => Discrim_Val, Op1 => discriminant_number). */ +#define Create_Node repinfo__create_node +extern Node_Ref Create_Node PARAMS((TCode, Node_Ref_Or_Val, + Node_Ref_Or_Val, Node_Ref_Or_Val)); diff --git a/gcc/ada/restrict.adb b/gcc/ada/restrict.adb new file mode 100644 index 00000000000..a284cd42e82 --- /dev/null +++ b/gcc/ada/restrict.adb @@ -0,0 +1,458 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- R E S T R I C T -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.37 $ +-- -- +-- 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 Atree; use Atree; +with Casing; use Casing; +with Errout; use Errout; +with Exp_Util; use Exp_Util; +with Fname; use Fname; +with Fname.UF; use Fname.UF; +with Lib; use Lib; +with Namet; use Namet; +with Nmake; use Nmake; +with Opt; use Opt; +with Stand; use Stand; +with Uname; use Uname; + +package body Restrict is + + function Suppress_Restriction_Message (N : Node_Id) return Boolean; + -- N is the node for a possible restriction violation message, but + -- the message is to be suppressed if this is an internal file and + -- this file is not the main unit. + + ------------------- + -- Abort_Allowed -- + ------------------- + + function Abort_Allowed return Boolean is + begin + return + Restrictions (No_Abort_Statements) = False + or else + Restriction_Parameters (Max_Asynchronous_Select_Nesting) /= 0; + end Abort_Allowed; + + ------------------------------------ + -- Check_Elaboration_Code_Allowed -- + ------------------------------------ + + procedure Check_Elaboration_Code_Allowed (N : Node_Id) is + begin + -- Avoid calling Namet.Unlock/Lock except when there is an error. + -- Even in the error case it is a bit dubious, either gigi needs + -- the table locked or it does not! ??? + + if Restrictions (No_Elaboration_Code) + and then not Suppress_Restriction_Message (N) + then + Namet.Unlock; + Check_Restriction (No_Elaboration_Code, N); + Namet.Lock; + end if; + end Check_Elaboration_Code_Allowed; + + --------------------------- + -- Check_Restricted_Unit -- + --------------------------- + + procedure Check_Restricted_Unit (U : Unit_Name_Type; N : Node_Id) is + begin + if Suppress_Restriction_Message (N) then + return; + + elsif Is_Spec_Name (U) then + declare + Fnam : constant File_Name_Type := + Get_File_Name (U, Subunit => False); + R_Id : Restriction_Id; + + begin + if not Is_Predefined_File_Name (Fnam) then + return; + + -- Ada child unit spec, needs checking against list + + else + -- Pad name to 8 characters with blanks + + Get_Name_String (Fnam); + Name_Len := Name_Len - 4; + + while Name_Len < 8 loop + Name_Len := Name_Len + 1; + Name_Buffer (Name_Len) := ' '; + end loop; + + for J in Unit_Array'Range loop + if Name_Len = 8 + and then Name_Buffer (1 .. 8) = Unit_Array (J).Filenm + then + R_Id := Unit_Array (J).Res_Id; + Violations (R_Id) := True; + + if Restrictions (R_Id) then + declare + S : constant String := Restriction_Id'Image (R_Id); + + begin + Error_Msg_Unit_1 := U; + + Error_Msg_N + ("dependence on $ not allowed,", N); + + Name_Buffer (1 .. S'Last) := S; + Name_Len := S'Length; + Set_Casing (All_Lower_Case); + Error_Msg_Name_1 := Name_Enter; + Error_Msg_Sloc := Restrictions_Loc (R_Id); + + Error_Msg_N + ("\violates pragma Restriction (%) #", N); + return; + end; + end if; + end if; + end loop; + end if; + end; + end if; + end Check_Restricted_Unit; + + ----------------------- + -- Check_Restriction -- + ----------------------- + + -- Case of simple identifier (no parameter) + + procedure Check_Restriction (R : Restriction_Id; N : Node_Id) is + begin + Violations (R) := True; + + if Restrictions (R) + and then not Suppress_Restriction_Message (N) + then + declare + S : constant String := Restriction_Id'Image (R); + + begin + Name_Buffer (1 .. S'Last) := S; + Name_Len := S'Length; + Set_Casing (All_Lower_Case); + Error_Msg_Name_1 := Name_Enter; + Error_Msg_Sloc := Restrictions_Loc (R); + Error_Msg_N ("violation of restriction %#", N); + end; + end if; + end Check_Restriction; + + -- Case where a parameter is present (but no count) + + procedure Check_Restriction + (R : Restriction_Parameter_Id; + N : Node_Id) + is + begin + if Restriction_Parameters (R) = Uint_0 + and then not Suppress_Restriction_Message (N) + then + declare + Loc : constant Source_Ptr := Sloc (N); + S : constant String := + Restriction_Parameter_Id'Image (R); + + begin + Error_Msg_NE + ("& will be raised at run time?!", N, Standard_Storage_Error); + Name_Buffer (1 .. S'Last) := S; + Name_Len := S'Length; + Set_Casing (All_Lower_Case); + Error_Msg_Name_1 := Name_Enter; + Error_Msg_Sloc := Restriction_Parameters_Loc (R); + Error_Msg_N ("violation of restriction %?#!", N); + + Insert_Action (N, + Make_Raise_Storage_Error (Loc)); + end; + end if; + end Check_Restriction; + + -- Case where a parameter is present, with a count + + procedure Check_Restriction + (R : Restriction_Parameter_Id; + V : Uint; + N : Node_Id) + is + begin + if Restriction_Parameters (R) /= No_Uint + and then V > Restriction_Parameters (R) + and then not Suppress_Restriction_Message (N) + then + declare + S : constant String := Restriction_Parameter_Id'Image (R); + + begin + Name_Buffer (1 .. S'Last) := S; + Name_Len := S'Length; + Set_Casing (All_Lower_Case); + Error_Msg_Name_1 := Name_Enter; + Error_Msg_Sloc := Restriction_Parameters_Loc (R); + Error_Msg_N ("maximum value exceeded for restriction %#", N); + end; + end if; + end Check_Restriction; + + ------------------------------------------- + -- Compilation_Unit_Restrictions_Restore -- + ------------------------------------------- + + procedure Compilation_Unit_Restrictions_Restore + (R : Save_Compilation_Unit_Restrictions) + is + begin + for J in Compilation_Unit_Restrictions loop + Restrictions (J) := R (J); + end loop; + end Compilation_Unit_Restrictions_Restore; + + ---------------------------------------- + -- Compilation_Unit_Restrictions_Save -- + ---------------------------------------- + + function Compilation_Unit_Restrictions_Save + return Save_Compilation_Unit_Restrictions + is + R : Save_Compilation_Unit_Restrictions; + + begin + for J in Compilation_Unit_Restrictions loop + R (J) := Restrictions (J); + Restrictions (J) := False; + end loop; + + return R; + end Compilation_Unit_Restrictions_Save; + + ---------------------------------- + -- Disallow_In_No_Run_Time_Mode -- + ---------------------------------- + + procedure Disallow_In_No_Run_Time_Mode (Enode : Node_Id) is + begin + if No_Run_Time then + Error_Msg_N + ("this construct not allowed in No_Run_Time mode", Enode); + end if; + end Disallow_In_No_Run_Time_Mode; + + ------------------------ + -- Get_Restriction_Id -- + ------------------------ + + function Get_Restriction_Id + (N : Name_Id) + return Restriction_Id + is + J : Restriction_Id; + + begin + Get_Name_String (N); + Set_Casing (All_Upper_Case); + + J := Restriction_Id'First; + while J /= Not_A_Restriction_Id loop + declare + S : constant String := Restriction_Id'Image (J); + + begin + exit when S = Name_Buffer (1 .. Name_Len); + end; + + J := Restriction_Id'Succ (J); + end loop; + + return J; + end Get_Restriction_Id; + + ---------------------------------- + -- Get_Restriction_Parameter_Id -- + ---------------------------------- + + function Get_Restriction_Parameter_Id + (N : Name_Id) + return Restriction_Parameter_Id + is + J : Restriction_Parameter_Id; + + begin + Get_Name_String (N); + Set_Casing (All_Upper_Case); + + J := Restriction_Parameter_Id'First; + while J /= Not_A_Restriction_Parameter_Id loop + declare + S : constant String := Restriction_Parameter_Id'Image (J); + + begin + exit when S = Name_Buffer (1 .. Name_Len); + end; + + J := Restriction_Parameter_Id'Succ (J); + end loop; + + return J; + end Get_Restriction_Parameter_Id; + + ------------------------------- + -- No_Exception_Handlers_Set -- + ------------------------------- + + function No_Exception_Handlers_Set return Boolean is + begin + return Restrictions (No_Exception_Handlers); + end No_Exception_Handlers_Set; + + ------------------------ + -- Restricted_Profile -- + ------------------------ + + -- This implementation must be coordinated with Set_Restricted_Profile + + function Restricted_Profile return Boolean is + begin + return Restrictions (No_Abort_Statements) + and then Restrictions (No_Asynchronous_Control) + and then Restrictions (No_Entry_Queue) + and then Restrictions (No_Task_Hierarchy) + and then Restrictions (No_Task_Allocators) + and then Restrictions (No_Dynamic_Priorities) + and then Restrictions (No_Terminate_Alternatives) + and then Restrictions (No_Dynamic_Interrupts) + and then Restrictions (No_Protected_Type_Allocators) + and then Restrictions (No_Local_Protected_Objects) + and then Restrictions (No_Requeue) + and then Restrictions (No_Task_Attributes) + and then Restriction_Parameters (Max_Asynchronous_Select_Nesting) = 0 + and then Restriction_Parameters (Max_Task_Entries) = 0 + and then Restriction_Parameters (Max_Protected_Entries) <= 1 + and then Restriction_Parameters (Max_Select_Alternatives) = 0; + end Restricted_Profile; + + -------------------------- + -- Set_No_Run_Time_Mode -- + -------------------------- + + procedure Set_No_Run_Time_Mode is + begin + No_Run_Time := True; + Restrictions (No_Exception_Handlers) := True; + end Set_No_Run_Time_Mode; + + ------------------- + -- Set_Ravenscar -- + ------------------- + + procedure Set_Ravenscar is + begin + Set_Restricted_Profile; + Restrictions (Boolean_Entry_Barriers) := True; + Restrictions (No_Select_Statements) := True; + Restrictions (No_Calendar) := True; + Restrictions (Static_Storage_Size) := True; + Restrictions (No_Entry_Queue) := True; + Restrictions (No_Relative_Delay) := True; + Restrictions (No_Task_Termination) := True; + Restrictions (No_Implicit_Heap_Allocations) := True; + end Set_Ravenscar; + + ---------------------------- + -- Set_Restricted_Profile -- + ---------------------------- + + -- This must be coordinated with Restricted_Profile + + procedure Set_Restricted_Profile is + begin + Restrictions (No_Abort_Statements) := True; + Restrictions (No_Asynchronous_Control) := True; + Restrictions (No_Entry_Queue) := True; + Restrictions (No_Task_Hierarchy) := True; + Restrictions (No_Task_Allocators) := True; + Restrictions (No_Dynamic_Priorities) := True; + Restrictions (No_Terminate_Alternatives) := True; + Restrictions (No_Dynamic_Interrupts) := True; + Restrictions (No_Protected_Type_Allocators) := True; + Restrictions (No_Local_Protected_Objects) := True; + Restrictions (No_Requeue) := True; + Restrictions (No_Task_Attributes) := True; + + Restriction_Parameters (Max_Asynchronous_Select_Nesting) := Uint_0; + Restriction_Parameters (Max_Task_Entries) := Uint_0; + Restriction_Parameters (Max_Select_Alternatives) := Uint_0; + + if Restriction_Parameters (Max_Protected_Entries) /= Uint_0 then + Restriction_Parameters (Max_Protected_Entries) := Uint_1; + end if; + end Set_Restricted_Profile; + + ---------------------------------- + -- Suppress_Restriction_Message -- + ---------------------------------- + + function Suppress_Restriction_Message (N : Node_Id) return Boolean is + begin + -- If main unit is library unit, then we will output message + + if In_Extended_Main_Source_Unit (N) then + return False; + + -- If loaded by rtsfind, then suppress message + + elsif Sloc (N) <= No_Location then + return True; + + -- Otherwise suppress message if internal file + + else + return + Is_Internal_File_Name (Unit_File_Name (Get_Source_Unit (N))); + end if; + end Suppress_Restriction_Message; + + --------------------- + -- Tasking_Allowed -- + --------------------- + + function Tasking_Allowed return Boolean is + begin + return + Restriction_Parameters (Max_Tasks) /= 0; + end Tasking_Allowed; + +end Restrict; diff --git a/gcc/ada/restrict.ads b/gcc/ada/restrict.ads new file mode 100644 index 00000000000..426149efaaf --- /dev/null +++ b/gcc/ada/restrict.ads @@ -0,0 +1,253 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- R E S T R I C T -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.27 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +-- This package deals with the implementation of the Restrictions pragma + +with Rident; +with Types; use Types; +with Uintp; use Uintp; + +package Restrict is + + type Restriction_Id is new Rident.Restriction_Id; + -- The type Restriction_Id defines the set of restriction identifiers, + -- which take no parameter (i.e. they are either present or not present). + -- The actual definition is in the separate package Rident, so that it + -- can easily be accessed by the binder without dragging in lots of stuff. + + subtype Partition_Restrictions is + Restriction_Id range + Restriction_Id (Rident.Partition_Restrictions'First) .. + Restriction_Id (Rident.Partition_Restrictions'Last); + -- Range of restriction identifiers that are checked by the binder + + subtype Compilation_Unit_Restrictions is + Restriction_Id range + Restriction_Id (Rident.Compilation_Unit_Restrictions'First) .. + Restriction_Id (Rident.Compilation_Unit_Restrictions'Last); + -- Range of restriction identifiers not checked by binder + + type Restriction_Parameter_Id is new Rident.Restriction_Parameter_Id; + -- The type Restriction_Parameter_Id records cases where a parameter is + -- present in the corresponding pragma. These cases are not checked for + -- consistency by the binder. The actual definition is in the separate + -- package Rident for consistency. + + type Restrictions_Flags is array (Restriction_Id) of Boolean; + -- Type used for arrays indexed by Restriction_Id. + + Restrictions : Restrictions_Flags := (others => False); + -- Corresponding entry is False if restriction is not active, and + -- True if the restriction is active, i.e. if a pragma Restrictions + -- has been seen anywhere. Note that we are happy to pick up any + -- restrictions pragmas in with'ed units, since we are required to + -- be consistent at link time, and we might as well find the error + -- at compile time. Clients must NOT use this array for checking to + -- see if a restriction is violated, instead it is required that the + -- Check_Restrictions subprograms be used for this purpose. The only + -- legitimate direct use of this array is when the code is modified + -- as a result of the restriction in some way. + + Restrictions_Loc : array (Restriction_Id) of Source_Ptr; + -- Locations of Restrictions pragmas for error message purposes. + -- Valid only if corresponding entry in Restrictions is set. + + Main_Restrictions : Restrictions_Flags := (others => False); + -- This variable saves the cumulative restrictions in effect compiling + -- any unit that is part of the extended main unit (i.e. the compiled + -- unit, its spec if any, and its subunits if any). The reason we keep + -- track of this is for the information that goes to the binder about + -- restrictions that are set. The binder will identify a unit that has + -- a restrictions pragma for error message purposes, and we do not want + -- to pick up a restrictions pragma in a with'ed unit for this purpose. + + Violations : Restrictions_Flags := (others => False); + -- Corresponding entry is False if the restriction has not been + -- violated in the current main unit, and True if it has been violated. + + Restriction_Parameters : + array (Restriction_Parameter_Id) of Uint := (others => No_Uint); + -- This array indicates the setting of restriction parameter identifier + -- values. All values are initially set to No_Uint indicating that the + -- parameter is not set, and are set to the appropriate non-negative + -- value if a Restrictions pragma specifies the corresponding + -- restriction parameter identifier with an appropriate value. + + Restriction_Parameters_Loc : + array (Restriction_Parameter_Id) of Source_Ptr; + -- Locations of Restrictions pragmas for error message purposes. + -- Valid only if corresponding entry in Restriction_Parameters is + -- set to a value other than No_Uint. + + type Unit_Entry is record + Res_Id : Restriction_Id; + Filenm : String (1 .. 8); + end record; + + type Unit_Array_Type is array (Positive range <>) of Unit_Entry; + + Unit_Array : constant Unit_Array_Type := ( + (No_Asynchronous_Control, "a-astaco"), + (No_Calendar, "a-calend"), + (No_Calendar, "calendar"), + (No_Delay, "a-calend"), + (No_Delay, "calendar"), + (No_Dynamic_Priorities, "a-dynpri"), + (No_IO, "a-direio"), + (No_IO, "directio"), + (No_IO, "a-sequio"), + (No_IO, "sequenio"), + (No_IO, "a-ststio"), + (No_IO, "a-textio"), + (No_IO, "text_io "), + (No_IO, "a-witeio"), + (No_Task_Attributes, "a-tasatt"), + (No_Streams, "a-stream"), + (No_Unchecked_Conversion, "a-unccon"), + (No_Unchecked_Conversion, "unchconv"), + (No_Unchecked_Deallocation, "a-uncdea"), + (No_Unchecked_Deallocation, "unchdeal")); + -- This array defines the mapping between restriction identifiers and + -- predefined language files containing units for which the identifier + -- forbids semantic dependence. + + type Save_Compilation_Unit_Restrictions is private; + -- Type used for saving and restoring compilation unit restrictions. + -- See Compilation_Unit_Restrictions_[Save|Restore] subprograms. + + ----------------- + -- Subprograms -- + ----------------- + + procedure Check_Restricted_Unit (U : Unit_Name_Type; N : Node_Id); + -- Checks if loading of unit U is prohibited by the setting of some + -- restriction (e.g. No_IO restricts the loading of unit Ada.Text_IO). + -- If a restriction exists post error message at the given node. + + procedure Check_Restriction (R : Restriction_Id; N : Node_Id); + -- Checks that the given restriction is not set, and if it is set, an + -- appropriate message is posted on the given node. Also records the + -- violation in the violations array. Note that it is mandatory to + -- always use this routine to check if a restriction is violated. Such + -- checks must never be done directly by the caller, since otherwise + -- they are not properly recorded in the violations array. + + procedure Check_Restriction + (R : Restriction_Parameter_Id; + N : Node_Id); + -- Checks that the given restriction parameter identifier is not set to + -- zero. If it is set to zero, then the node N is replaced by a node + -- that raises Storage_Error, and a warning is issued. + + procedure Check_Restriction + (R : Restriction_Parameter_Id; + V : Uint; + N : Node_Id); + -- Checks that the count in V does not exceed the maximum value of the + -- restriction parameter value corresponding to the given restriction + -- parameter identifier (if it has been set). If the count in V exceeds + -- the maximum, then post an error message on node N. + + procedure Check_Elaboration_Code_Allowed (N : Node_Id); + -- Tests to see if elaboration code is allowed by the current restrictions + -- settings. This function is called by Gigi when it needs to define + -- an elaboration routine. If elaboration code is not allowed, an error + -- message is posted on the node given as argument. + + function No_Exception_Handlers_Set return Boolean; + -- Test to see if current restrictions settings specify that no exception + -- handlers are present. This function is called by Gigi when it needs to + -- expand an AT END clean up identifier with no exception handler. + + function Compilation_Unit_Restrictions_Save + return Save_Compilation_Unit_Restrictions; + -- This function saves the compilation unit restriction settings, and + -- resets them to False. This is used e.g. when compiling a with'ed + -- unit to avoid incorrectly propagating restrictions. Note that it + -- would not be wrong to also save and reset the partition restrictions, + -- since the binder would catch inconsistencies, but actually it is a + -- good thing to acquire restrictions from with'ed units if they are + -- required to be partition wide, because it allows the restriction + -- violation message to be given at compile time instead of link time. + + procedure Compilation_Unit_Restrictions_Restore + (R : Save_Compilation_Unit_Restrictions); + -- This is the corresponding restore procedure to restore restrictions + -- previously saved by Compilation_Unit_Restrictions_Save. + + procedure Disallow_In_No_Run_Time_Mode (Enode : Node_Id); + -- If in No_Run_Time mode, then the construct represented by Enode is + -- not permitted, and will be appropriately flagged. + + procedure Set_No_Run_Time_Mode; + -- Set the no run time mode, and associated restriction pragmas. + + function Get_Restriction_Id + (N : Name_Id) + return Restriction_Id; + -- Given an identifier name, determines if it is a valid restriction + -- identifier, and if so returns the corresponding Restriction_Id + -- value, otherwise returns Not_A_Restriction_Id. + + function Get_Restriction_Parameter_Id + (N : Name_Id) + return Restriction_Parameter_Id; + -- Given an identifier name, determines if it is a valid restriction + -- parameter identifier, and if so returns the corresponding + -- Restriction_Parameter_Id value, otherwise returns + -- Not_A_Restriction_Parameter_Id. + + function Abort_Allowed return Boolean; + pragma Inline (Abort_Allowed); + -- Tests to see if abort is allowed by the current restrictions settings. + -- For abort to be allowed, either No_Abort_Statements must be False, + -- or Max_Asynchronous_Select_Nesting must be non-zero. + + function Restricted_Profile return Boolean; + -- Tests to see if tasking operations follow the GNAT restricted run time + -- profile. + + procedure Set_Ravenscar; + -- Sets the set of rerstrictions fro Ravenscar + + procedure Set_Restricted_Profile; + -- Sets the set of restrictions for pragma Restricted_Run_Time + + function Tasking_Allowed return Boolean; + pragma Inline (Tasking_Allowed); + -- Tests to see if tasking operations are allowed by the current + -- restrictions settings. For tasking to be allowed Max_Tasks must + -- be non-zero. + +private + type Save_Compilation_Unit_Restrictions is + array (Compilation_Unit_Restrictions) of Boolean; + -- Type used for saving and restoring compilation unit restrictions. + -- See Compilation_Unit_Restrictions_[Save|Restore] subprograms. + +end Restrict; diff --git a/gcc/ada/rident.ads b/gcc/ada/rident.ads new file mode 100644 index 00000000000..3eb65408433 --- /dev/null +++ b/gcc/ada/rident.ads @@ -0,0 +1,139 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- R I D E N T -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.12 $ +-- -- +-- 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). -- +-- -- +------------------------------------------------------------------------------ + +-- This package defines the set of restriction identifiers. It is in a +-- separate package from Restrict so that it can be easily used by the +-- binder without dragging in a lot of stuff. + +package Rident is + + -- The following enumeration type defines the set of restriction + -- identifiers not taking a parameter that are implemented in GNAT. + -- To add a new restriction identifier, add an entry with the name + -- to be used in the pragma, and add appropriate calls to the + -- Check_Restriction routine. + + type Restriction_Id is ( + + -- The following cases are checked for consistency in the binder + + Boolean_Entry_Barriers, -- GNAT (Ravenscar) + No_Abort_Statements, -- (RM D.7(5), H.4(3)) + No_Access_Subprograms, -- (RM H.4(17)) + No_Allocators, -- (RM H.4(7)) + No_Asynchronous_Control, -- (RM D.9(10)) + No_Calendar, -- GNAT + No_Delay, -- (RM H.4(21)) + No_Dispatch, -- (RM H.4(19)) + No_Dynamic_Interrupts, -- GNAT + No_Dynamic_Priorities, -- (RM D.9(9)) + No_Enumeration_Maps, -- GNAT + No_Entry_Calls_In_Elaboration_Code, -- GNAT + No_Entry_Queue, -- GNAT + No_Exception_Handlers, -- GNAT + No_Exceptions, -- (RM H.4(12)) + No_Fixed_Point, -- (RM H.4(15)) + No_Floating_Point, -- (RM H.4(14)) + No_IO, -- (RM H.4(20)) + No_Implicit_Conditionals, -- GNAT + No_Implicit_Dynamic_Code, -- GNAT + No_Implicit_Heap_Allocations, -- (RM D.8(8), H.4(3)) + No_Implicit_Loops, -- GNAT + No_Local_Allocators, -- (RM H.4(8)) + No_Local_Protected_Objects, -- GNAT + No_Nested_Finalization, -- (RM D.7(4)) + No_Protected_Type_Allocators, -- GNAT + No_Protected_Types, -- (RM H.4(5)) + No_Recursion, -- (RM H.4(22)) + No_Reentrancy, -- (RM H.4(23)) + No_Relative_Delay, -- GNAT + No_Requeue, -- GNAT + No_Select_Statements, -- GNAT (Ravenscar) + No_Standard_Storage_Pools, -- GNAT + No_Streams, -- GNAT + No_Task_Allocators, -- (RM D.7(7)) + No_Task_Attributes, -- GNAT + No_Task_Hierarchy, -- (RM D.7(3), H.4(3)) + No_Task_Termination, -- GNAT + No_Terminate_Alternatives, -- (RM D.7(6)) + No_Unchecked_Access, -- (RM H.4(18)) + No_Unchecked_Conversion, -- (RM H.4(16)) + No_Unchecked_Deallocation, -- (RM H.4(9)) + No_Wide_Characters, -- GNAT + Static_Priorities, -- GNAT + Static_Storage_Size, -- GNAT + + -- The following cases do not require partition-wide checks + + Immediate_Reclamation, -- (RM H.4(10)) + No_Implementation_Attributes, -- GNAT + No_Implementation_Pragmas, -- GNAT + No_Implementation_Restrictions, -- GNAT + No_Elaboration_Code, -- GNAT + + Not_A_Restriction_Id); + + -- The following range of Restriction identifiers is checked for + -- consistency across a partition. The generated ali file is marked + -- for each entry to show one of three possibilities: + -- + -- Corresponding restriction is set (so unit does not violate it) + -- Corresponding restriction is not violated + -- Corresponding restriction is violated + + subtype Partition_Restrictions is + Restriction_Id range Boolean_Entry_Barriers .. Static_Storage_Size; + + -- The following set of Restriction identifiers is not checked for + -- consistency across a partition, and the generated ali files does + -- not carry any indications with respect to such restrictions. + + subtype Compilation_Unit_Restrictions is + Restriction_Id range Immediate_Reclamation .. No_Elaboration_Code; + + -- The following enumeration type defines the set of restriction + -- parameter identifiers taking a parameter that are implemented in + -- GNAT. To add a new restriction parameter identifier, add an entry + -- with the name to be used in the pragma, and add appropriate + -- calls to Check_Restriction. + + -- Note: the GNAT implementation currently only accomodates restriction + -- parameter identifiers whose expression value is a non-negative + -- integer. This is true for all language defined parameters. + + type Restriction_Parameter_Id is ( + Max_Asynchronous_Select_Nesting, -- (RM D.7(18), H.4(3)) + Max_Entry_Queue_Depth, -- GNAT + Max_Protected_Entries, -- (RM D.7(14)) + Max_Select_Alternatives, -- (RM D.7(12)) + Max_Storage_At_Blocking, -- (RM D.7(17)) + Max_Task_Entries, -- (RM D.7(13), H.4(3)) + Max_Tasks, -- (RM D.7(19), H.4(3)) + Not_A_Restriction_Parameter_Id); + +end Rident; diff --git a/gcc/ada/rtsfind.adb b/gcc/ada/rtsfind.adb new file mode 100644 index 00000000000..1299e1e2a13 --- /dev/null +++ b/gcc/ada/rtsfind.adb @@ -0,0 +1,913 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- R T S F I N D -- +-- -- +-- B o d y -- +-- -- +-- $Revision: 1.96 $ +-- -- +-- 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 Atree; use Atree; +with Casing; use Casing; +with Csets; use Csets; +with Debug; use Debug; +with Einfo; use Einfo; +with Elists; use Elists; +with Fname; use Fname; +with Fname.UF; use Fname.UF; +with Lib; use Lib; +with Lib.Load; use Lib.Load; +with Namet; use Namet; +with Nlists; use Nlists; +with Nmake; use Nmake; +with Output; use Output; +with Opt; use Opt; +with Restrict; use Restrict; +with Sem; use Sem; +with Sem_Ch7; use Sem_Ch7; +with Sem_Util; use Sem_Util; +with Sinfo; use Sinfo; +with Stand; use Stand; +with Snames; use Snames; +with Tbuild; use Tbuild; +with Uname; use Uname; + +package body Rtsfind is + + ---------------- + -- Unit table -- + ---------------- + + -- The unit table has one entry for each unit included in the definition + -- of the type RTU_Id in the spec. The table entries are initialized in + -- Initialize to set the Entity field to Empty, indicating that the + -- corresponding unit has not yet been loaded. The fields are set when + -- a unit is loaded to contain the defining entity for the unit, the + -- unit name, and the unit number. + + type RT_Unit_Table_Record is record + Entity : Entity_Id; + Uname : Unit_Name_Type; + Unum : Unit_Number_Type; + Withed : Boolean; + end record; + + RT_Unit_Table : array (RTU_Id) of RT_Unit_Table_Record; + + -------------------------- + -- Runtime Entity Table -- + -------------------------- + + -- There is one entry in the runtime entity table for each entity that is + -- included in the definition of the RE_Id type in the spec. The entries + -- are set by Initialize_Rtsfind to contain Empty, indicating that the + -- entity has not yet been located. Once the entity is located for the + -- first time, its ID is stored in this array, so that subsequent calls + -- for the same entity can be satisfied immediately. + + RE_Table : array (RE_Id) of Entity_Id; + + -------------------------- + -- Generation of WITH's -- + -------------------------- + + -- When a unit is implicitly loaded as a result of a call to RTE, it + -- is necessary to create an implicit with to ensure that the object + -- is correctly loaded by the binder. Such with statements are only + -- required when the request is from the extended main unit (if a + -- client needs a with, that will be taken care of when the client + -- is compiled. + + -- We always attach the with to the main unit. This is not perfectly + -- accurate in terms of elaboration requirements, but it is close + -- enough, since the units that are accessed using rtsfind do not + -- have delicate elaboration requirements. + + -- The flag Withed in the unit table record is initially set to False. + -- It is set True if a with has been generated for the main unit for + -- the corresponding unit. + + ----------------------- + -- Local Subprograms -- + ----------------------- + + procedure Load_Fail (S : String; U_Id : RTU_Id; Ent_Name : String := ""); + -- Internal procedure called if we can't find the entity or unit. + -- The parameter is a detailed error message that is to be given. + -- S is a reason for failing to compile the file. U_Id is the unit + -- id, and Ent_Name, if non-null, is the associated entity name. + + function Get_Unit_Name (U_Id : RTU_Id) return Unit_Name_Type; + -- Retrieves the Unit Name given a unit id represented by its + -- enumaration value in RTU_Id. + + procedure Load_RTU (U_Id : RTU_Id; Use_Setting : Boolean := False); + -- Load the unit whose Id is given if not already loaded. The unit is + -- loaded, analyzed, and added to the with list, and the entry in + -- RT_Unit_Table is updated to reflect the load. The second parameter + -- indicates the initial setting for the Is_Potentially_Use_Visible + -- flag of the entity for the loaded unit (if it is indeed loaded). + -- A value of False means nothing special need be done. A value of + -- True indicates that this flag must be set to True. It is needed + -- only in the Text_IO_Kludge procedure, which may materialize an + -- entity of Text_IO (or Wide_Text_IO) that was previously unknown. + + function RE_Chars (E : RE_Id) return Name_Id; + -- Given a RE_Id value returns the Chars of the corresponding entity. + + ------------------- + -- Get_Unit_Name -- + ------------------- + + function Get_Unit_Name (U_Id : RTU_Id) return Unit_Name_Type is + Uname_Chars : constant String := RTU_Id'Image (U_Id); + + begin + Name_Len := Uname_Chars'Length; + Name_Buffer (1 .. Name_Len) := Uname_Chars; + Set_Casing (All_Lower_Case); + + if U_Id in Ada_Child then + Name_Buffer (4) := '.'; + + if U_Id in Ada_Calendar_Child then + Name_Buffer (13) := '.'; + + elsif U_Id in Ada_Finalization_Child then + Name_Buffer (17) := '.'; + + elsif U_Id in Ada_Real_Time_Child then + Name_Buffer (14) := '.'; + + elsif U_Id in Ada_Streams_Child then + Name_Buffer (12) := '.'; + + elsif U_Id in Ada_Text_IO_Child then + Name_Buffer (12) := '.'; + + elsif U_Id in Ada_Wide_Text_IO_Child then + Name_Buffer (17) := '.'; + end if; + + elsif U_Id in Interfaces_Child then + Name_Buffer (11) := '.'; + + elsif U_Id in System_Child then + Name_Buffer (7) := '.'; + + if U_Id in System_Tasking_Child then + Name_Buffer (15) := '.'; + end if; + + if U_Id in System_Tasking_Restricted_Child then + Name_Buffer (26) := '.'; + end if; + + if U_Id in System_Tasking_Protected_Objects_Child then + Name_Buffer (33) := '.'; + end if; + + if U_Id in System_Tasking_Async_Delays_Child then + Name_Buffer (28) := '.'; + end if; + end if; + + -- Add %s at end for spec + + Name_Buffer (Name_Len + 1) := '%'; + Name_Buffer (Name_Len + 2) := 's'; + Name_Len := Name_Len + 2; + + return Name_Find; + end Get_Unit_Name; + + ---------------- + -- Initialize -- + ---------------- + + procedure Initialize is + begin + -- Initialize the unit table + + for J in RTU_Id loop + RT_Unit_Table (J).Entity := Empty; + end loop; + + for J in RE_Id loop + RE_Table (J) := Empty; + end loop; + end Initialize; + + ------------ + -- Is_RTE -- + ------------ + + function Is_RTE (Ent : Entity_Id; E : RE_Id) return Boolean is + E_Unit_Name : Unit_Name_Type; + Ent_Unit_Name : Unit_Name_Type; + + S : Entity_Id; + E1 : Entity_Id; + E2 : Entity_Id; + + begin + if No (Ent) then + return False; + + -- If E has already a corresponding entity, check it directly, + -- going to full views if they exist to deal with the incomplete + -- and private type cases properly. + + elsif Present (RE_Table (E)) then + E1 := Ent; + + if Is_Type (E1) and then Present (Full_View (E1)) then + E1 := Full_View (E1); + end if; + + E2 := RE_Table (E); + + if Is_Type (E2) and then Present (Full_View (E2)) then + E2 := Full_View (E2); + end if; + + return E1 = E2; + end if; + + -- If the unit containing E is not loaded, we already know that + -- the entity we have cannot have come from this unit. + + E_Unit_Name := Get_Unit_Name (RE_Unit_Table (E)); + + if not Is_Loaded (E_Unit_Name) then + return False; + end if; + + -- Here the unit containing the entity is loaded. We have not made + -- an explicit call to RTE to get the entity in question, but we may + -- have obtained a reference to it indirectly from some other entity + -- in the same unit, or some other unit that references it. + + -- Get the defining unit of the entity + + S := Scope (Ent); + + if Ekind (S) /= E_Package then + return False; + end if; + + Ent_Unit_Name := Get_Unit_Name (Unit_Declaration_Node (S)); + + -- If the defining unit of the entity we are testing is not the + -- unit containing E, then they cannot possibly match. + + if Ent_Unit_Name /= E_Unit_Name then + return False; + end if; + + -- If the units match, then compare the names (remember that no + -- overloading is permitted in entities fetched using Rtsfind). + + if RE_Chars (E) = Chars (Ent) then + RE_Table (E) := Ent; + + -- If front-end inlining is enabled, we may be within a body that + -- contains inlined functions, which has not been retrieved through + -- rtsfind, and therefore is not yet recorded in the RT_Unit_Table. + -- Add the unit information now, it must be fully available. + + declare + U : RT_Unit_Table_Record + renames RT_Unit_Table (RE_Unit_Table (E)); + begin + if No (U.Entity) then + U.Entity := S; + U.Uname := E_Unit_Name; + U.Unum := Get_Source_Unit (S); + end if; + end; + + return True; + else + return False; + end if; + end Is_RTE; + + ---------------------------- + -- Is_Text_IO_Kludge_Unit -- + ---------------------------- + + function Is_Text_IO_Kludge_Unit (Nam : Node_Id) return Boolean is + Prf : Node_Id; + Sel : Node_Id; + + begin + if Nkind (Nam) /= N_Expanded_Name then + return False; + end if; + + Prf := Prefix (Nam); + Sel := Selector_Name (Nam); + + if Nkind (Sel) /= N_Expanded_Name + or else Nkind (Prf) /= N_Identifier + or else Chars (Prf) /= Name_Ada + then + return False; + end if; + + Prf := Prefix (Sel); + Sel := Selector_Name (Sel); + + return + Nkind (Prf) = N_Identifier + and then + (Chars (Prf) = Name_Text_IO or else Chars (Prf) = Name_Wide_Text_IO) + and then + Nkind (Sel) = N_Identifier + and then + Chars (Sel) in Text_IO_Package_Name; + + end Is_Text_IO_Kludge_Unit; + + --------------- + -- Load_Fail -- + --------------- + + procedure Load_Fail (S : String; U_Id : RTU_Id; Ent_Name : String := "") is + begin + Set_Standard_Error; + + Write_Str ("fatal error: run-time library configuration error"); + Write_Eol; + + if Ent_Name /= "" then + Write_Str ("cannot locate """); + + -- Copy name skipping initial RE_ or RO_XX characters + + if Ent_Name (1 .. 2) = "RE" then + for J in 4 .. Ent_Name'Length loop + Name_Buffer (J - 3) := Ent_Name (J); + end loop; + else + for J in 7 .. Ent_Name'Length loop + Name_Buffer (J - 6) := Ent_Name (J); + end loop; + end if; + + Name_Len := Ent_Name'Length - 3; + Set_Casing (Mixed_Case); + Write_Str (Name_Buffer (1 .. Name_Len)); + Write_Str (""" in file """); + + else + Write_Str ("cannot load file """); + end if; + + Write_Name + (Get_File_Name (RT_Unit_Table (U_Id).Uname, Subunit => False)); + Write_Str (""" ("); + Write_Str (S); + Write_Char (')'); + Write_Eol; + Set_Standard_Output; + raise Unrecoverable_Error; + end Load_Fail; + + -------------- + -- Load_RTU -- + -------------- + + procedure Load_RTU (U_Id : RTU_Id; Use_Setting : Boolean := False) is + Loaded : Boolean; + U : RT_Unit_Table_Record renames RT_Unit_Table (U_Id); + Priv_Par : Elist_Id := New_Elmt_List; + Lib_Unit : Node_Id; + + procedure Save_Private_Visibility; + -- If the current unit is the body of child unit or the spec of a + -- private child unit, the private declarations of the parent (s) + -- are visible. If the unit to be loaded is another public sibling, + -- its compilation will affect the visibility of the common ancestors. + -- Indicate those that must be restored. + + procedure Restore_Private_Visibility; + -- Restore the visibility of ancestors after compiling RTU. + + -------------------------------- + -- Restore_Private_Visibility -- + -------------------------------- + + procedure Restore_Private_Visibility is + E_Par : Elmt_Id; + + begin + E_Par := First_Elmt (Priv_Par); + + while Present (E_Par) loop + if not In_Private_Part (Node (E_Par)) then + Install_Private_Declarations (Node (E_Par)); + end if; + + Next_Elmt (E_Par); + end loop; + end Restore_Private_Visibility; + + ----------------------------- + -- Save_Private_Visibility -- + ----------------------------- + + procedure Save_Private_Visibility is + Par : Entity_Id; + + begin + Par := Scope (Current_Scope); + + while Present (Par) + and then Par /= Standard_Standard + loop + if Ekind (Par) = E_Package + and then Is_Compilation_Unit (Par) + and then In_Private_Part (Par) + then + Append_Elmt (Par, Priv_Par); + end if; + + Par := Scope (Par); + end loop; + end Save_Private_Visibility; + + -- Start of processing for Load_RTU + + begin + -- Nothing to do if unit is already loaded + + if Present (U.Entity) then + return; + end if; + + -- Otherwise we need to load the unit, First build unit name + -- from the enumeration literal name in type RTU_Id. + + U.Uname := Get_Unit_Name (U_Id); + U.Withed := False; + Loaded := Is_Loaded (U.Uname); + + -- Now do the load call, note that setting Error_Node to Empty is + -- a signal to Load_Unit that we will regard a failure to find the + -- file as a fatal error, and that it should not output any kind + -- of diagnostics, since we will take care of it here. + + U.Unum := + Load_Unit + (Load_Name => U.Uname, + Required => False, + Subunit => False, + Error_Node => Empty); + + if U.Unum = No_Unit then + Load_Fail ("unit not found", U_Id); + + elsif Fatal_Error (U.Unum) then + Load_Fail ("parser errors", U_Id); + end if; + + -- Make sure that the unit is analyzed + + declare + Was_Analyzed : Boolean := Analyzed (Cunit (Current_Sem_Unit)); + + begin + -- Pretend that the current unit is analysed, in case it is + -- System or some such. This allows us to put some declarations, + -- such as exceptions and packed arrays of Boolean, into System + -- even though expanding them requires System... + + -- This is a bit odd but works fine. If the RTS unit does not depend + -- in any way on the current unit, then it never gets back into the + -- current unit's tree, and the change we make to the current unit + -- tree is never noticed by anyone (it is undone in a moment). That + -- is the normal situation. + + -- If the RTS Unit *does* depend on the current unit, for instance, + -- when you are compiling System, then you had better have finished + -- Analyzing the part of System that is depended on before you try + -- to load the RTS Unit. This means having the System ordered in an + -- appropriate manner. + + Set_Analyzed (Cunit (Current_Sem_Unit), True); + + if not Analyzed (Cunit (U.Unum)) then + + Save_Private_Visibility; + Semantics (Cunit (U.Unum)); + Restore_Private_Visibility; + + if Fatal_Error (U.Unum) then + Load_Fail ("semantic errors", U_Id); + end if; + end if; + + -- Undo the pretence + + Set_Analyzed (Cunit (Current_Sem_Unit), Was_Analyzed); + end; + + Lib_Unit := Unit (Cunit (U.Unum)); + U.Entity := Defining_Entity (Lib_Unit); + + if Use_Setting then + Set_Is_Potentially_Use_Visible (U.Entity, True); + end if; + end Load_RTU; + + -------------- + -- RE_Chars -- + -------------- + + function RE_Chars (E : RE_Id) return Name_Id is + RE_Name_Chars : constant String := RE_Id'Image (E); + + begin + -- Copy name skipping initial RE_ or RO_XX characters + + if RE_Name_Chars (1 .. 2) = "RE" then + for J in 4 .. RE_Name_Chars'Last loop + Name_Buffer (J - 3) := Fold_Lower (RE_Name_Chars (J)); + end loop; + + Name_Len := RE_Name_Chars'Length - 3; + + else + for J in 7 .. RE_Name_Chars'Last loop + Name_Buffer (J - 6) := Fold_Lower (RE_Name_Chars (J)); + end loop; + + Name_Len := RE_Name_Chars'Length - 6; + end if; + + return Name_Find; + end RE_Chars; + + --------- + -- RTE -- + --------- + + function RTE (E : RE_Id) return Entity_Id is + U_Id : constant RTU_Id := RE_Unit_Table (E); + U : RT_Unit_Table_Record renames RT_Unit_Table (U_Id); + + Ent : Entity_Id; + Lib_Unit : Node_Id; + Pkg_Ent : Entity_Id; + Ename : Name_Id; + Enode : Node_Id; + + procedure Check_RPC; + -- Reject programs that make use of distribution features not supported + -- on the current target. On such targets (VMS, Vxworks, others?) we + -- only provide a minimal body for System.Rpc that only supplies an + -- implementation of partition_id. + + function Find_Local_Entity (E : RE_Id) return Entity_Id; + -- This function is used when entity E is in this compilation's main + -- unit. It gets the value from the already compiled declaration. + + function Make_Unit_Name (N : Node_Id) return Node_Id; + -- If the unit is a child unit, build fully qualified name for use + -- in with_clause. + + --------------- + -- Check_RPC -- + --------------- + + procedure Check_RPC is + Body_Name : Unit_Name_Type; + Unum : Unit_Number_Type; + + begin + -- Bypass this check if debug flag -gnatdR set + + if Debug_Flag_RR then + return; + end if; + + -- Otherwise we need the check if we are going after one of + -- the critical entities in System.RPC in stubs mode. + + if (Distribution_Stub_Mode = Generate_Receiver_Stub_Body + or else + Distribution_Stub_Mode = Generate_Caller_Stub_Body) + and then (E = RE_Do_Rpc + or else E = RE_Do_Apc + or else E = RE_Params_Stream_Type + or else E = RE_RPC_Receiver) + then + -- Load body of System.Rpc, and abort if this is the body that is + -- provided by GNAT, for which these features are not supported + -- on current target. We identify the gnat body by the presence + -- of a local entity called Gnat in the first declaration. + + Lib_Unit := Unit (Cunit (U.Unum)); + Body_Name := Get_Body_Name (Get_Unit_Name (Lib_Unit)); + Unum := + Load_Unit + (Load_Name => Body_Name, + Required => False, + Subunit => False, + Error_Node => Empty, + Renamings => True); + + if Unum /= No_Unit then + declare + Decls : List_Id := Declarations (Unit (Cunit (Unum))); + + begin + if Present (Decls) + and then Nkind (First (Decls)) = N_Object_Declaration + and then + Chars (Defining_Identifier (First (Decls))) = Name_Gnat + then + Set_Standard_Error; + Write_Str ("distribution feature not supported"); + Write_Eol; + raise Unrecoverable_Error; + end if; + end; + end if; + end if; + end Check_RPC; + + ------------------------ + -- Find_System_Entity -- + ------------------------ + + function Find_Local_Entity (E : RE_Id) return Entity_Id is + RE_Str : String renames RE_Id'Image (E); + Ent : Entity_Id; + + Save_Nam : constant String := Name_Buffer (1 .. Name_Len); + -- Save name buffer and length over call + + begin + Name_Len := Natural'Max (0, RE_Str'Length - 3); + Name_Buffer (1 .. Name_Len) := + RE_Str (RE_Str'First + 3 .. RE_Str'Last); + + Ent := Entity_Id (Get_Name_Table_Info (Name_Find)); + + Name_Len := Save_Nam'Length; + Name_Buffer (1 .. Name_Len) := Save_Nam; + + return Ent; + end Find_Local_Entity; + + -------------------- + -- Make_Unit_Name -- + -------------------- + + function Make_Unit_Name (N : Node_Id) return Node_Id is + Nam : Node_Id; + Scop : Entity_Id; + + begin + Nam := New_Reference_To (U.Entity, Standard_Location); + Scop := Scope (U.Entity); + + if Nkind (N) = N_Defining_Program_Unit_Name then + while Scop /= Standard_Standard loop + Nam := + Make_Expanded_Name (Standard_Location, + Chars => Chars (U.Entity), + Prefix => New_Reference_To (Scop, Standard_Location), + Selector_Name => Nam); + Set_Entity (Nam, U.Entity); + + Scop := Scope (Scop); + end loop; + end if; + + return Nam; + end Make_Unit_Name; + + -- Start of processing for RTE + + begin + -- Doing a rtsfind in system.ads is special, as we cannot do this + -- when compiling System itself. So if we are compiling system then + -- we should already have acquired and processed the declaration + -- of the entity. The test is to see if this compilation's main unit + -- is System. If so, return the value from the already compiled + -- declaration and otherwise do a regular find. + + -- Not pleasant, but these kinds of annoying recursion when + -- writing an Ada compiler in Ada have to be broken somewhere! + + if Present (Main_Unit_Entity) + and then Chars (Main_Unit_Entity) = Name_System + and then Analyzed (Main_Unit_Entity) + and then not Is_Child_Unit (Main_Unit_Entity) + then + return Find_Local_Entity (E); + end if; + + Enode := Current_Error_Node; + + -- Load unit if unit not previously loaded + + if No (RE_Table (E)) then + Load_RTU (U_Id); + Lib_Unit := Unit (Cunit (U.Unum)); + + -- In the subprogram case, we are all done, the entity we want + -- is the entity for the subprogram itself. Note that we do not + -- bother to check that it is the entity that was requested. + -- the only way that could fail to be the case is if runtime is + -- hopelessly misconfigured, and it isn't worth testing for this. + + if Nkind (Lib_Unit) = N_Subprogram_Declaration then + RE_Table (E) := U.Entity; + + -- Otherwise we must have the package case, and here we have to + -- search the package entity chain for the entity we want. The + -- entity we want must be present in this chain, or we have a + -- misconfigured runtime. + + else + pragma Assert (Nkind (Lib_Unit) = N_Package_Declaration); + Ename := RE_Chars (E); + + Pkg_Ent := First_Entity (U.Entity); + + while Present (Pkg_Ent) loop + if Ename = Chars (Pkg_Ent) then + RE_Table (E) := Pkg_Ent; + Check_RPC; + goto Found; + end if; + + Next_Entity (Pkg_Ent); + end loop; + + -- If we didn't find the unit we want, something is wrong! + + Load_Fail ("entity not in package", U_Id, RE_Id'Image (E)); + raise Program_Error; + end if; + end if; + + -- See if we have to generate a with for this entity. We generate + -- a with if the current unit is part of the extended main code + -- unit, and if we have not already added the with. The with is + -- added to the appropriate unit (the current one). + + <<Found>> + if (not U.Withed) + and then + In_Extended_Main_Code_Unit (Cunit_Entity (Current_Sem_Unit)) + then + U.Withed := True; + + declare + Withn : Node_Id; + Lib_Unit : Node_Id; + + begin + Lib_Unit := Unit (Cunit (U.Unum)); + Withn := + Make_With_Clause (Standard_Location, + Name => + Make_Unit_Name + (Defining_Unit_Name (Specification (Lib_Unit)))); + Set_Library_Unit (Withn, Cunit (U.Unum)); + Set_Corresponding_Spec (Withn, U.Entity); + Set_First_Name (Withn, True); + Set_Implicit_With (Withn, True); + + Mark_Rewrite_Insertion (Withn); + Append (Withn, Context_Items (Cunit (Current_Sem_Unit))); + end; + end if; + + -- We can now obtain the entity. Check that the No_Run_Time condition + -- is not violated. Note that we do not signal the error if we detect + -- it in a runtime unit. This can only arise if the user explicitly + -- with'ed the runtime unit (or another runtime unit that uses it + -- transitively), or if some acceptable (e.g. inlined) entity is + -- fetched from a unit, some of whose other routines or entities + -- violate the conditions. In the latter case, it does not matter, + -- since we won't be using those entities. + + Ent := RE_Table (E); + + if Is_Subprogram (Ent) + and then not Is_Inlined (Ent) + and then Sloc (Enode) /= Standard_Location + and then not + Is_Predefined_File_Name (Unit_File_Name (Get_Source_Unit (Enode))) + then + Disallow_In_No_Run_Time_Mode (Enode); + end if; + + return Ent; + end RTE; + + -------------------- + -- Text_IO_Kludge -- + -------------------- + + procedure Text_IO_Kludge (Nam : Node_Id) is + Chrs : Name_Id; + + type Name_Map_Type is array (Text_IO_Package_Name) of RTU_Id; + + Name_Map : Name_Map_Type := Name_Map_Type'( + Name_Decimal_IO => Ada_Text_IO_Decimal_IO, + Name_Enumeration_IO => Ada_Text_IO_Enumeration_IO, + Name_Fixed_IO => Ada_Text_IO_Fixed_IO, + Name_Float_IO => Ada_Text_IO_Float_IO, + Name_Integer_IO => Ada_Text_IO_Integer_IO, + Name_Modular_IO => Ada_Text_IO_Modular_IO); + + Wide_Name_Map : Name_Map_Type := Name_Map_Type'( + Name_Decimal_IO => Ada_Wide_Text_IO_Decimal_IO, + Name_Enumeration_IO => Ada_Wide_Text_IO_Enumeration_IO, + Name_Fixed_IO => Ada_Wide_Text_IO_Fixed_IO, + Name_Float_IO => Ada_Wide_Text_IO_Float_IO, + Name_Integer_IO => Ada_Wide_Text_IO_Integer_IO, + Name_Modular_IO => Ada_Wide_Text_IO_Modular_IO); + + begin + -- Nothing to do if name is not identifier or a selected component + -- whose selector_name is not an identifier. + + if Nkind (Nam) = N_Identifier then + Chrs := Chars (Nam); + + elsif Nkind (Nam) = N_Selected_Component + and then Nkind (Selector_Name (Nam)) = N_Identifier + then + Chrs := Chars (Selector_Name (Nam)); + + else + return; + end if; + + -- Nothing to do if name is not one of the Text_IO subpackages + -- Otherwise look through loaded units, and if we find Text_IO + -- or Wide_Text_IO already loaded, then load the proper child. + + if Chrs in Text_IO_Package_Name then + for U in Main_Unit .. Last_Unit loop + Get_Name_String (Unit_File_Name (U)); + + if Name_Len = 12 then + + -- Here is where we do the loads if we find one of the + -- units Ada.Text_IO or Ada.Wide_Text_IO. An interesting + -- detail is that these units may already be used (i.e. + -- their In_Use flags may be set). Normally when the In_Use + -- flag is set, the Is_Potentially_Use_Visible flag of all + -- entities in the package is set, but the new entity we + -- are mysteriously adding was not there to have its flag + -- set at the time. So that's why we pass the extra parameter + -- to RTU_Find, to make sure the flag does get set now. + -- Given that those generic packages are in fact child units, + -- we must indicate that they are visible. + + if Name_Buffer (1 .. 12) = "a-textio.ads" then + Load_RTU (Name_Map (Chrs), In_Use (Cunit_Entity (U))); + Set_Is_Visible_Child_Unit + (RT_Unit_Table (Name_Map (Chrs)).Entity); + + elsif Name_Buffer (1 .. 12) = "a-witeio.ads" then + Load_RTU (Wide_Name_Map (Chrs), In_Use (Cunit_Entity (U))); + Set_Is_Visible_Child_Unit + (RT_Unit_Table (Wide_Name_Map (Chrs)).Entity); + end if; + end if; + end loop; + end if; + end Text_IO_Kludge; + +end Rtsfind; diff --git a/gcc/ada/rtsfind.ads b/gcc/ada/rtsfind.ads new file mode 100644 index 00000000000..11304f625a8 --- /dev/null +++ b/gcc/ada/rtsfind.ads @@ -0,0 +1,2324 @@ +------------------------------------------------------------------------------ +-- -- +-- GNAT COMPILER COMPONENTS -- +-- -- +-- R T S F I N D -- +-- -- +-- S p e c -- +-- -- +-- $Revision: 1.216 $ +-- -- +-- 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 Types; use Types; + +package Rtsfind is + +-- This package contains the routine that is used to obtain runtime library +-- entities, loading in the required runtime library packages on demand. It +-- is also used for such purposes as finding System.Address when System has +-- not been explicitly With'ed. + + ------------------------ + -- Runtime Unit Table -- + ------------------------ + + -- The following type includes an enumeration entry for each runtime + -- unit. The enumeration literal represents the fully qualified + -- name of the unit, as follows: + + -- Names of the form Ada_xxx are first level children of Ada, whose + -- name is Ada.xxx. For example, the name Ada_Tags refers to package + -- Ada.Tags. + + -- Names of the form Ada_Calendar_xxx are second level children + -- of Ada.Calendar. This is part of a temporary implementation of + -- delays; eventually, packages implementing delays will be found + -- relative to the package that declares the time type. + + -- Names of the form Interfaces_xxx are first level children of + -- Interfaces_CPP refers to package Interfaces.CPP + + -- Names of the form System_xxx are first level children of System, whose + -- name is System.xxx. For example, the name System_Str_Concat refers to + -- package System.Str_Concat. + + -- Names of the form System_Tasking_xxx are second level children of the + -- package System.Tasking. For example, System_Tasking_Stages refers to + -- refers to the package System.Tasking.Stages. + + -- Other names stand for themselves (e.g. System for package System) + + -- This list can contain both subprogram and package unit names. For + -- packages, the accessible entities in the package are separately + -- listed in the package entity table. The units must be either library + -- level package declarations, or library level subprogram declarations. + -- Generic units, library level instantiations and subprogram bodies + -- acting as specs may not be referenced (all these cases could be added + -- at the expense of additional complexity in the body of Rtsfind, but + -- it doesn't seem worth while, since the implementation controls the + -- set of units that are referenced, and this restrictions is easily met. + + -- IMPORTANT NOTE: the specs of packages and procedures with'ed using + -- this mechanism may not contain use clauses. This is because these + -- subprograms are compiled in the current visibility environment, and + -- it would be too much trouble to establish a clean environment for the + -- compilation. The presence of extraneous visible stuff has no effect + -- on the compilation except in the presence of use clauses (which might + -- result in unexpected ambiguities). + + type RTU_Id is ( + -- Runtime packages, for list of accessible entities in each + -- package see declarations in the runtime entity table below. + + RTU_Null, + -- Used as a null entry. Will cause an error if referenced. + + -- Children of Ada + + Ada_Calendar, + Ada_Exceptions, + Ada_Finalization, + Ada_Interrupts, + Ada_Real_Time, + Ada_Streams, + Ada_Tags, + Ada_Task_Identification, + + -- Children of Ada.Calendar + + Ada_Calendar_Delays, + + -- Children of Ada.Finalization + + Ada_Finalization_List_Controller, + + -- Children of Ada.Real_Time + + Ada_Real_Time_Delays, + + -- Children of Ada.Streams + + Ada_Streams_Stream_IO, + + -- Children of Ada.Text_IO (for Text_IO_Kludge) + + Ada_Text_IO_Decimal_IO, + Ada_Text_IO_Enumeration_IO, + Ada_Text_IO_Fixed_IO, + Ada_Text_IO_Float_IO, + Ada_Text_IO_Integer_IO, + Ada_Text_IO_Modular_IO, + + -- Children of Ada.Wide_Text_IO (for Text_IO_Kludge) + + Ada_Wide_Text_IO_Decimal_IO, + Ada_Wide_Text_IO_Enumeration_IO, + Ada_Wide_Text_IO_Fixed_IO, + Ada_Wide_Text_IO_Float_IO, + Ada_Wide_Text_IO_Integer_IO, + Ada_Wide_Text_IO_Modular_IO, + + -- Interfaces + + Interfaces, + + -- Children of Interfaces + + Interfaces_CPP, + Interfaces_Packed_Decimal, + + -- Package System + + System, + + -- Children of System + + System_Arith_64, + System_AST_Handling, + System_Assertions, + System_Aux_DEC, + System_Bit_Ops, + System_Checked_Pools, + System_Exception_Table, + System_Exceptions, + System_Delay_Operations, + System_Exn_Flt, + System_Exn_Int, + System_Exn_LFlt, + System_Exn_LInt, + System_Exn_LLF, + System_Exn_LLI, + System_Exn_SFlt, + System_Exn_SInt, + System_Exn_SSI, + System_Exp_Flt, + System_Exp_Int, + System_Exp_LFlt, + System_Exp_LInt, + System_Exp_LLF, + System_Exp_LLI, + System_Exp_LLU, + System_Exp_Mod, + System_Exp_SFlt, + System_Exp_SInt, + System_Exp_SSI, + System_Exp_Uns, + System_Fat_Flt, + System_Fat_LFlt, + System_Fat_LLF, + System_Fat_SFlt, + System_Finalization_Implementation, + System_Finalization_Root, + System_Fore, + System_Img_Bool, + System_Img_Char, + System_Img_Dec, + System_Img_Enum, + System_Img_Int, + System_Img_LLD, + System_Img_LLI, + System_Img_LLU, + System_Img_Name, + System_Img_Real, + System_Img_Uns, + System_Img_WChar, + System_Interrupts, + System_Machine_Code, + System_Mantissa, + System_Pack_03, + System_Pack_05, + System_Pack_06, + System_Pack_07, + System_Pack_09, + System_Pack_10, + System_Pack_11, + System_Pack_12, + System_Pack_13, + System_Pack_14, + System_Pack_15, + System_Pack_17, + System_Pack_18, + System_Pack_19, + System_Pack_20, + System_Pack_21, + System_Pack_22, + System_Pack_23, + System_Pack_24, + System_Pack_25, + System_Pack_26, + System_Pack_27, + System_Pack_28, + System_Pack_29, + System_Pack_30, + System_Pack_31, + System_Pack_33, + System_Pack_34, + System_Pack_35, + System_Pack_36, + System_Pack_37, + System_Pack_38, + System_Pack_39, + System_Pack_40, + System_Pack_41, + System_Pack_42, + System_Pack_43, + System_Pack_44, + System_Pack_45, + System_Pack_46, + System_Pack_47, + System_Pack_48, + System_Pack_49, + System_Pack_50, + System_Pack_51, + System_Pack_52, + System_Pack_53, + System_Pack_54, + System_Pack_55, + System_Pack_56, + System_Pack_57, + System_Pack_58, + System_Pack_59, + System_Pack_60, + System_Pack_61, + System_Pack_62, + System_Pack_63, + System_Parameters, + System_Partition_Interface, + System_Pool_Global, + System_Pool_Empty, + System_Pool_Local, + System_Pool_Size, + System_RPC, + System_Scalar_Values, + System_Secondary_Stack, + System_Shared_Storage, + System_Soft_Links, + System_Standard_Library, + System_Storage_Elements, + System_Storage_Pools, + System_Stream_Attributes, + System_String_Ops, + System_String_Ops_Concat_3, + System_String_Ops_Concat_4, + System_String_Ops_Concat_5, + System_Task_Info, + System_Tasking, + System_Unsigned_Types, + System_Val_Bool, + System_Val_Char, + System_Val_Dec, + System_Val_Enum, + System_Val_Int, + System_Val_LLD, + System_Val_LLI, + System_Val_LLU, + System_Val_Name, + System_Val_Real, + System_Val_Uns, + System_Val_WChar, + System_Vax_Float_Operations, + System_Version_Control, + System_VMS_Exception_Table, + System_WCh_StW, + System_WCh_WtS, + System_Wid_Bool, + System_Wid_Char, + System_Wid_Enum, + System_Wid_LLI, + System_Wid_LLU, + System_Wid_Name, + System_Wid_WChar, + System_WWd_Char, + System_WWd_Enum, + System_WWd_Wchar, + + -- Children of System.Tasking + + System_Tasking_Async_Delays, + System_Tasking_Async_Delays_Enqueue_Calendar, + System_Tasking_Async_Delays_Enqueue_RT, + System_Tasking_Protected_Objects, + System_Tasking_Protected_Objects_Entries, + System_Tasking_Protected_Objects_Operations, + System_Tasking_Protected_Objects_Single_Entry, + System_Tasking_Restricted_Stages, + System_Tasking_Rendezvous, + System_Tasking_Stages); + + subtype Ada_Child is RTU_Id + range Ada_Calendar .. Ada_Wide_Text_IO_Modular_IO; + -- Range of values for children or grand-children of Ada + + subtype Ada_Calendar_Child is Ada_Child + range Ada_Calendar_Delays .. Ada_Calendar_Delays; + -- Range of values for children of Ada.Calendar + + subtype Ada_Finalization_Child is Ada_Child range + Ada_Finalization_List_Controller .. Ada_Finalization_List_Controller; + -- Range of values for children of Ada.Finalization + + subtype Ada_Real_Time_Child is Ada_Child + range Ada_Real_Time_Delays .. Ada_Real_Time_Delays; + -- Range of values for children of Ada.Real_Time + + subtype Ada_Streams_Child is Ada_Child + range Ada_Streams_Stream_IO .. Ada_Streams_Stream_IO; + + subtype Ada_Text_IO_Child is Ada_Child + range Ada_Text_IO_Decimal_IO .. Ada_Text_IO_Modular_IO; + -- Range of values for children of Ada.Text_IO + + subtype Ada_Wide_Text_IO_Child is Ada_Child + range Ada_Wide_Text_IO_Decimal_IO .. Ada_Wide_Text_IO_Modular_IO; + -- Range of values for children of Ada.Text_IO + + subtype Interfaces_Child is RTU_Id + range Interfaces_CPP .. Interfaces_Packed_Decimal; + -- Range of values for children of Interfaces + + subtype System_Child is RTU_Id + range System_Arith_64 .. System_Tasking_Stages; + -- Range of values for children or grandchildren of System + + subtype System_Tasking_Child is System_Child + range System_Tasking_Async_Delays .. System_Tasking_Stages; + -- Range of values for children of System.Tasking + + subtype System_Tasking_Protected_Objects_Child is System_Tasking_Child + range System_Tasking_Protected_Objects_Entries .. + System_Tasking_Protected_Objects_Single_Entry; + -- Range of values for children of System.Tasking.Protected_Objects + + subtype System_Tasking_Restricted_Child is System_Tasking_Child + range System_Tasking_Restricted_Stages .. + System_Tasking_Restricted_Stages; + -- Range of values for children of System.Tasking.Restricted + + subtype System_Tasking_Async_Delays_Child is System_Tasking_Child + range System_Tasking_Async_Delays_Enqueue_Calendar .. + System_Tasking_Async_Delays_Enqueue_RT; + -- Range of values for children of System.Tasking.Async_Delays + + -------------------------- + -- Runtime Entity Table -- + -------------------------- + + -- This is the enumeration type used to define the argument passed to + -- the RTE function. The name must exactly match the name of the entity + -- involved, and in the case of a package entity, this name must uniquely + -- imply the package containing the entity. + + -- As far as possible, we avoid duplicate names in runtime packages, so + -- that the name RE_nnn uniquely identifies the entity nnn. In some cases, + -- it is impossible to avoid such duplication because the names come from + -- RM defined packages. In such cases, the name is of the form RO_XX_nnn + -- where XX is two letters used to differentiate the multiple occurrences + -- of the name xx, and nnn is the entity name. + + -- Note that not all entities in the units contained in the run-time unit + -- table are included in the following table, only those that actually + -- have to be referenced from generated code. + + -- Note on RE_Null. This value is used as a null entry where an RE_Id + -- value is required syntactically, but no real entry is required or + -- needed. Use of this value will cause a fatal error in an RTE call. + + type RE_Id is ( + + RE_Null, + + RE_Code_Loc, -- Ada.Exceptions + RE_Current_Target_Exception, -- Ada.Exceptions (JGNAT use only) + RE_Exception_Id, -- Ada.Exceptions + RE_Exception_Information, -- Ada.Exceptions + RE_Exception_Message, -- Ada.Exceptions + RE_Exception_Name_Simple, -- Ada.Exceptions + RE_Exception_Occurrence, -- Ada.Exceptions + RE_Null_Id, -- Ada.Exceptions + RE_Null_Occurrence, -- Ada.Exceptions + RE_Poll, -- Ada.Exceptions + RE_Raise_Exception, -- Ada.Exceptions + RE_Raise_Exception_Always, -- Ada.Exceptions + RE_Reraise_Occurrence, -- Ada.Exceptions + RE_Reraise_Occurrence_Always, -- Ada.Exceptions + RE_Reraise_Occurrence_No_Defer, -- Ada.Exceptions + RE_Save_Occurrence, -- Ada.Exceptions + + RE_Simple_List_Controller, -- Ada.Finalization.List_Controller + RE_List_Controller, -- Ada.Finalization.List_Controller + + RE_Interrupt_Id, -- Ada.Interrupts + + RE_Root_Stream_Type, -- Ada.Streams + RE_Stream_Element, -- Ada.Streams + RE_Stream_Element_Offset, -- Ada.Streams + RE_Stream_Element_Array, -- Ada.Streams + + RE_Stream_Access, -- Ada.Streams.Stream_IO + + RE_CW_Membership, -- Ada.Tags + RE_DT_Entry_Size, -- Ada.Tags + RE_DT_Prologue_Size, -- Ada.Tags + RE_External_Tag, -- Ada.Tags + RE_Get_Expanded_Name, -- Ada.Tags + RE_Get_External_Tag, -- Ada.Tags + RE_Get_Prim_Op_Address, -- Ada.Tags + RE_Get_RC_Offset, -- Ada.Tags + RE_Get_Remotely_Callable, -- Ada.Tags + RE_Get_TSD, -- Ada.Tags + RE_Inherit_DT, -- Ada.Tags + RE_Inherit_TSD, -- Ada.Tags + RE_Internal_Tag, -- Ada.Tags + RE_Register_Tag, -- Ada.Tags + RE_Set_Expanded_Name, -- Ada.Tags + RE_Set_External_Tag, -- Ada.Tags + RE_Set_Prim_Op_Address, -- Ada.Tags + RE_Set_RC_Offset, -- Ada.Tags + RE_Set_Remotely_Callable, -- Ada.Tags + RE_Set_TSD, -- Ada.Tags + RE_Tag_Error, -- Ada.Tags + RE_TSD_Entry_Size, -- Ada.Tags + RE_TSD_Prologue_Size, -- Ada.Tags + RE_Tag, -- Ada.Tags + RE_Address_Array, -- Ada.Tags + + RE_Current_Task, -- Ada.Task_Identification + RO_AT_Task_ID, -- Ada.Task_Identification + + RO_CA_Time, -- Ada.Calendar + + RO_CA_Delay_For, -- Ada.Calendar.Delays + RO_CA_Delay_Until, -- Ada.Calendar.Delays + RO_CA_To_Duration, -- Ada.Calendar.Delays + + RO_RT_Time, -- Ada.Real_Time + + RO_RT_Delay_Until, -- Ada.Real_Time.Delays + RO_RT_To_Duration, -- Ada.Real_Time.Delays + + RE_Integer_64, -- Interfaces + RE_Unsigned_8, -- Interfaces + RE_Unsigned_16, -- Interfaces + RE_Unsigned_32, -- Interfaces + RE_Unsigned_64, -- Interfaces + + RE_Vtable_Ptr, -- Interfaces.CPP + RE_Displaced_This, -- Interfaces.CPP + RE_CPP_CW_Membership, -- Interfaces.CPP + RE_CPP_DT_Entry_Size, -- Interfaces.CPP + RE_CPP_DT_Prologue_Size, -- Interfaces.CPP + RE_CPP_Get_Expanded_Name, -- Interfaces.CPP + RE_CPP_Get_External_Tag, -- Interfaces.CPP + RE_CPP_Get_Prim_Op_Address, -- Interfaces.CPP + RE_CPP_Get_RC_Offset, -- Interfaces.CPP + RE_CPP_Get_Remotely_Callable, -- Interfaces.CPP + RE_CPP_Get_TSD, -- Interfaces.CPP + RE_CPP_Inherit_DT, -- Interfaces.CPP + RE_CPP_Inherit_TSD, -- Interfaces.CPP + RE_CPP_Register_Tag, -- Interfaces.CPP + RE_CPP_Set_Expanded_Name, -- Interfaces.CPP + RE_CPP_Set_External_Tag, -- Interfaces.CPP + RE_CPP_Set_Prim_Op_Address, -- Interfaces.CPP + RE_CPP_Set_RC_Offset, -- Interfaces.CPP + RE_CPP_Set_Remotely_Callable, -- Interfaces.CPP + RE_CPP_Set_TSD, -- Interfaces.CPP + RE_CPP_TSD_Entry_Size, -- Interfaces.CPP + RE_CPP_TSD_Prologue_Size, -- Interfaces.CPP + + RE_Packed_Size, -- Interfaces.Packed_Decimal + RE_Packed_To_Int32, -- Interfaces.Packed_Decimal + RE_Packed_To_Int64, -- Interfaces.Packed_Decimal + RE_Int32_To_Packed, -- Interfaces.Packed_Decimal + RE_Int64_To_Packed, -- Interfaces.Packed_Decimal + + RE_Address, -- System + RE_Any_Priority, -- System + RE_Bit_Order, -- System + RE_Default_Priority, -- System + RE_High_Order_First, -- System + RE_Interrupt_Priority, -- System + RE_Lib_Stop, -- System + RE_Low_Order_First, -- System + RE_Max_Interrupt_Priority, -- System + RE_Max_Priority, -- System + RE_Null_Address, -- System + RE_Priority, -- System + + RE_Add_With_Ovflo_Check, -- System.Arith_64 + RE_Double_Divide, -- System.Arith_64 + RE_Multiply_With_Ovflo_Check, -- System.Arith_64 + RE_Scaled_Divide, -- System.Arith_64 + RE_Subtract_With_Ovflo_Check, -- System.Arith_64 + + RE_Create_AST_Handler, -- System.AST_Handling + + RE_Raise_Assert_Failure, -- System.Assertions + + RE_AST_Handler, -- System.Aux_DEC + RE_Import_Value, -- System.Aux_DEC + RE_No_AST_Handler, -- System.Aux_DEC + RE_Type_Class, -- System.Aux_DEC + RE_Type_Class_Enumeration, -- System.Aux_DEC + RE_Type_Class_Integer, -- System.Aux_DEC + RE_Type_Class_Fixed_Point, -- System.Aux_DEC + RE_Type_Class_Floating_Point, -- System.Aux_DEC + RE_Type_Class_Array, -- System.Aux_DEC + RE_Type_Class_Record, -- System.Aux_DEC + RE_Type_Class_Access, -- System.Aux_DEC + RE_Type_Class_Task, -- System.Aux_DEC + RE_Type_Class_Address, -- System.Aux_DEC + + RE_Bit_And, -- System.Bit_Ops + RE_Bit_Eq, -- System.Bit_Ops + RE_Bit_Not, -- System.Bit_Ops + RE_Bit_Or, -- System.Bit_Ops + RE_Bit_Xor, -- System.Bit_Ops + + RE_Checked_Pool, -- System.Checked_Pools + + RE_Register_Exception, -- System.Exception_Table + + RE_All_Others_Id, -- System.Exceptions + RE_Handler_Record, -- System.Exceptions + RE_Handler_Record_Ptr, -- System.Exceptions + RE_Others_Id, -- System.Exceptions + RE_Subprogram_Descriptor, -- System.Exceptions + RE_Subprogram_Descriptor_0, -- System.Exceptions + RE_Subprogram_Descriptor_1, -- System.Exceptions + RE_Subprogram_Descriptor_2, -- System.Exceptions + RE_Subprogram_Descriptor_3, -- System.Exceptions + RE_Subprogram_Descriptor_List, -- System.Exceptions + RE_Subprogram_Descriptor_Ptr, -- System.Exceptions + RE_Subprogram_Descriptors_Record, -- System.Exceptions + RE_Subprogram_Descriptors_Ptr, -- System.Exceptions + + RE_Exn_Float, -- System.Exn_Flt + + RE_Exn_Integer, -- System.Exn_Int + + RE_Exn_Long_Float, -- System.Exn_LFlt + + RE_Exn_Long_Integer, -- System.Exn_LInt + + RE_Exn_Long_Long_Float, -- System.Exn_LLF + + RE_Exn_Long_Long_Integer, -- System.Exn_LLI + + RE_Exn_Short_Float, -- System.Exn_SFlt + + RE_Exn_Short_Integer, -- System.Exn_SInt + + RE_Exn_Short_Short_Integer, -- System.Exn_SSI + + RE_Exp_Float, -- System.Exp_Flt + + RE_Exp_Integer, -- System.Exp_Int + + RE_Exp_Long_Float, -- System.Exp_LFlt + + RE_Exp_Long_Integer, -- System.Exp_LInt + + RE_Exp_Long_Long_Float, -- System.Exp_LLF + + RE_Exp_Long_Long_Integer, -- System.Exp_LLI + + RE_Exp_Long_Long_Unsigned, -- System.Exp_LLU + + RE_Exp_Modular, -- System.Exp_Mod + + RE_Exp_Short_Float, -- System.Exp_SFlt + + RE_Exp_Short_Integer, -- System.Exp_SInt + + RE_Exp_Short_Short_Integer, -- System.Exp_SSI + + RE_Exp_Unsigned, -- System.Exp_Uns + + RE_Fat_Float, -- System.Fat_Flt + + RE_Fat_Long_Float, -- System.Fat_LFlt + + RE_Fat_Long_Long_Float, -- System.Fat_LLF + + RE_Fat_Short_Float, -- System.Fat_SFlt + + RE_Attach_To_Final_List, -- System.Finalization_Implementation + RE_Finalize_List, -- System.Finalization_Implementation + RE_Finalize_One, -- System.Finalization_Implementation + RE_Global_Final_List, -- System.Finalization_Implementation + RE_Record_Controller, -- System.Finalization_Implementation + RE_Limited_Record_Controller, -- System.Finalization_Implementation + RE_Deep_Tag_Initialize, -- System.Finalization_Implementation + RE_Deep_Tag_Adjust, -- System.Finalization_Implementation + RE_Deep_Tag_Finalize, -- System.Finalization_Implementation + RE_Deep_Tag_Attach, -- System.Finalization_Implementation + RE_Deep_Rec_Initialize, -- System.Finalization_Implementation + RE_Deep_Rec_Adjust, -- System.Finalization_Implementation + RE_Deep_Rec_Finalize, -- System.Finalization_Implementation + + RE_Root_Controlled, -- System.Finalization_Root + RE_Finalizable, -- System.Finalization_Root + RE_Finalizable_Ptr, -- System.Finalization_Root + + RE_Fore, -- System.Fore + + RE_Image_Boolean, -- System.Img_Bool + + RE_Image_Character, -- System.Img_Char + + RE_Image_Decimal, -- System.Img_Dec + + RE_Image_Enumeration_8, -- System.Img_Enum + RE_Image_Enumeration_16, -- System.Img_Enum + RE_Image_Enumeration_32, -- System.Img_Enum + + RE_Image_Integer, -- System.Img_Int + + RE_Image_Long_Long_Decimal, -- System.Img_LLD + + RE_Image_Long_Long_Integer, -- System.Img_LLI + + RE_Image_Long_Long_Unsigned, -- System.Img_LLU + + RE_Image_Ordinary_Fixed_Point, -- System.Img_Real + RE_Image_Floating_Point, -- System.Img_Real + + RE_Image_Unsigned, -- System.Img_Uns + + RE_Image_Wide_Character, -- System.Img_WChar + + RE_Bind_Interrupt_To_Entry, -- System.Interrupts + RE_Default_Interrupt_Priority, -- System.Interrupts + RE_Dynamic_Interrupt_Protection, -- System.Interrupts + RE_Install_Handlers, -- System.Interrupts + RE_Register_Interrupt_Handler, -- System.Interrupts + RE_Static_Interrupt_Protection, -- System.Interrupts + + RE_Asm_Insn, -- System.Machine_Code + RE_Asm_Input_Operand, -- System.Machine_Code + RE_Asm_Output_Operand, -- System.Machine_Code + + RE_Mantissa_Value, -- System_Mantissa + + RE_Bits_03, -- System.Pack_03 + RE_Get_03, -- System.Pack_03 + RE_Set_03, -- System.Pack_03 + + RE_Bits_05, -- System.Pack_05 + RE_Get_05, -- System.Pack_05 + RE_Set_05, -- System.Pack_05 + + RE_Bits_06, -- System.Pack_06 + RE_Get_06, -- System.Pack_06 + RE_GetU_06, -- System.Pack_06 + RE_Set_06, -- System.Pack_06 + RE_SetU_06, -- System.Pack_06 + + RE_Bits_07, -- System.Pack_07 + RE_Get_07, -- System.Pack_07 + RE_Set_07, -- System.Pack_07 + + RE_Bits_09, -- System.Pack_09 + RE_Get_09, -- System.Pack_09 + RE_Set_09, -- System.Pack_09 + + RE_Bits_10, -- System.Pack_10 + RE_Get_10, -- System.Pack_10 + RE_GetU_10, -- System.Pack_10 + RE_Set_10, -- System.Pack_10 + RE_SetU_10, -- System.Pack_10 + + RE_Bits_11, -- System.Pack_11 + RE_Get_11, -- System.Pack_11 + RE_Set_11, -- System.Pack_11 + + RE_Bits_12, -- System.Pack_12 + RE_Get_12, -- System.Pack_12 + RE_GetU_12, -- System.Pack_12 + RE_Set_12, -- System.Pack_12 + RE_SetU_12, -- System.Pack_12 + + RE_Bits_13, -- System.Pack_13 + RE_Get_13, -- System.Pack_13 + RE_Set_13, -- System.Pack_13 + + RE_Bits_14, -- System.Pack_14 + RE_Get_14, -- System.Pack_14 + RE_GetU_14, -- System.Pack_14 + RE_Set_14, -- System.Pack_14 + RE_SetU_14, -- System.Pack_14 + + RE_Bits_15, -- System.Pack_15 + RE_Get_15, -- System.Pack_15 + RE_Set_15, -- System.Pack_15 + + RE_Bits_17, -- System.Pack_17 + RE_Get_17, -- System.Pack_17 + RE_Set_17, -- System.Pack_17 + + RE_Bits_18, -- System.Pack_18 + RE_Get_18, -- System.Pack_18 + RE_GetU_18, -- System.Pack_18 + RE_Set_18, -- System.Pack_18 + RE_SetU_18, -- System.Pack_18 + + RE_Bits_19, -- System.Pack_19 + RE_Get_19, -- System.Pack_19 + RE_Set_19, -- System.Pack_19 + + RE_Bits_20, -- System.Pack_20 + RE_Get_20, -- System.Pack_20 + RE_GetU_20, -- System.Pack_20 + RE_Set_20, -- System.Pack_20 + RE_SetU_20, -- System.Pack_20 + + RE_Bits_21, -- System.Pack_21 + RE_Get_21, -- System.Pack_21 + RE_Set_21, -- System.Pack_21 + + RE_Bits_22, -- System.Pack_22 + RE_Get_22, -- System.Pack_22 + RE_GetU_22, -- System.Pack_22 + RE_Set_22, -- System.Pack_22 + RE_SetU_22, -- System.Pack_22 + + RE_Bits_23, -- System.Pack_23 + RE_Get_23, -- System.Pack_23 + RE_Set_23, -- System.Pack_23 + + RE_Bits_24, -- System.Pack_24 + RE_Get_24, -- System.Pack_24 + RE_GetU_24, -- System.Pack_24 + RE_Set_24, -- System.Pack_24 + RE_SetU_24, -- System.Pack_24 + + RE_Bits_25, -- System.Pack_25 + RE_Get_25, -- System.Pack_25 + RE_Set_25, -- System.Pack_25 + + RE_Bits_26, -- System.Pack_26 + RE_Get_26, -- System.Pack_26 + RE_GetU_26, -- System.Pack_26 + RE_Set_26, -- System.Pack_26 + RE_SetU_26, -- System.Pack_26 + + RE_Bits_27, -- System.Pack_27 + RE_Get_27, -- System.Pack_27 + RE_Set_27, -- System.Pack_27 + + RE_Bits_28, -- System.Pack_28 + RE_Get_28, -- System.Pack_28 + RE_GetU_28, -- System.Pack_28 + RE_Set_28, -- System.Pack_28 + RE_SetU_28, -- System.Pack_28 + + RE_Bits_29, -- System.Pack_29 + RE_Get_29, -- System.Pack_29 + RE_Set_29, -- System.Pack_29 + + RE_Bits_30, -- System.Pack_30 + RE_Get_30, -- System.Pack_30 + RE_GetU_30, -- System.Pack_30 + RE_Set_30, -- System.Pack_30 + RE_SetU_30, -- System.Pack_30 + + RE_Bits_31, -- System.Pack_31 + RE_Get_31, -- System.Pack_31 + RE_Set_31, -- System.Pack_31 + + RE_Bits_33, -- System.Pack_33 + RE_Get_33, -- System.Pack_33 + RE_Set_33, -- System.Pack_33 + + RE_Bits_34, -- System.Pack_34 + RE_Get_34, -- System.Pack_34 + RE_GetU_34, -- System.Pack_34 + RE_Set_34, -- System.Pack_34 + RE_SetU_34, -- System.Pack_34 + + RE_Bits_35, -- System.Pack_35 + RE_Get_35, -- System.Pack_35 + RE_Set_35, -- System.Pack_35 + + RE_Bits_36, -- System.Pack_36 + RE_Get_36, -- System.Pack_36 + RE_GetU_36, -- System.Pack_36 + RE_Set_36, -- System.Pack_36 + RE_SetU_36, -- System.Pack_36 + + RE_Bits_37, -- System.Pack_37 + RE_Get_37, -- System.Pack_37 + RE_Set_37, -- System.Pack_37 + + RE_Bits_38, -- System.Pack_38 + RE_Get_38, -- System.Pack_38 + RE_GetU_38, -- System.Pack_38 + RE_Set_38, -- System.Pack_38 + RE_SetU_38, -- System.Pack_38 + + RE_Bits_39, -- System.Pack_39 + RE_Get_39, -- System.Pack_39 + RE_Set_39, -- System.Pack_39 + + RE_Bits_40, -- System.Pack_40 + RE_Get_40, -- System.Pack_40 + RE_GetU_40, -- System.Pack_40 + RE_Set_40, -- System.Pack_40 + RE_SetU_40, -- System.Pack_40 + + RE_Bits_41, -- System.Pack_41 + RE_Get_41, -- System.Pack_41 + RE_Set_41, -- System.Pack_41 + + RE_Bits_42, -- System.Pack_42 + RE_Get_42, -- System.Pack_42 + RE_GetU_42, -- System.Pack_42 + RE_Set_42, -- System.Pack_42 + RE_SetU_42, -- System.Pack_42 + + RE_Bits_43, -- System.Pack_43 + RE_Get_43, -- System.Pack_43 + RE_Set_43, -- System.Pack_43 + + RE_Bits_44, -- System.Pack_44 + RE_Get_44, -- System.Pack_44 + RE_GetU_44, -- System.Pack_44 + RE_Set_44, -- System.Pack_44 + RE_SetU_44, -- System.Pack_44 + + RE_Bits_45, -- System.Pack_45 + RE_Get_45, -- System.Pack_45 + RE_Set_45, -- System.Pack_45 + + RE_Bits_46, -- System.Pack_46 + RE_Get_46, -- System.Pack_46 + RE_GetU_46, -- System.Pack_46 + RE_Set_46, -- System.Pack_46 + RE_SetU_46, -- System.Pack_46 + + RE_Bits_47, -- System.Pack_47 + RE_Get_47, -- System.Pack_47 + RE_Set_47, -- System.Pack_47 + + RE_Bits_48, -- System.Pack_48 + RE_Get_48, -- System.Pack_48 + RE_GetU_48, -- System.Pack_48 + RE_Set_48, -- System.Pack_48 + RE_SetU_48, -- System.Pack_48 + + RE_Bits_49, -- System.Pack_49 + RE_Get_49, -- System.Pack_49 + RE_Set_49, -- System.Pack_49 + + RE_Bits_50, -- System.Pack_50 + RE_Get_50, -- System.Pack_50 + RE_GetU_50, -- System.Pack_50 + RE_Set_50, -- System.Pack_50 + RE_SetU_50, -- System.Pack_50 + + RE_Bits_51, -- System.Pack_51 + RE_Get_51, -- System.Pack_51 + RE_Set_51, -- System.Pack_51 + + RE_Bits_52, -- System.Pack_52 + RE_Get_52, -- System.Pack_52 + RE_GetU_52, -- System.Pack_52 + RE_Set_52, -- System.Pack_52 + RE_SetU_52, -- System.Pack_52 + + RE_Bits_53, -- System.Pack_53 + RE_Get_53, -- System.Pack_53 + RE_Set_53, -- System.Pack_53 + + RE_Bits_54, -- System.Pack_54 + RE_Get_54, -- System.Pack_54 + RE_GetU_54, -- System.Pack_54 + RE_Set_54, -- System.Pack_54 + RE_SetU_54, -- System.Pack_54 + + RE_Bits_55, -- System.Pack_55 + RE_Get_55, -- System.Pack_55 + RE_Set_55, -- System.Pack_55 + + RE_Bits_56, -- System.Pack_56 + RE_Get_56, -- System.Pack_56 + RE_GetU_56, -- System.Pack_56 + RE_Set_56, -- System.Pack_56 + RE_SetU_56, -- System.Pack_56 + + RE_Bits_57, -- System.Pack_57 + RE_Get_57, -- System.Pack_57 + RE_Set_57, -- System.Pack_57 + + RE_Bits_58, -- System.Pack_58 + RE_Get_58, -- System.Pack_58 + RE_GetU_58, -- System.Pack_58 + RE_Set_58, -- System.Pack_58 + RE_SetU_58, -- System.Pack_58 + + RE_Bits_59, -- System.Pack_59 + RE_Get_59, -- System.Pack_59 + RE_Set_59, -- System.Pack_59 + + RE_Bits_60, -- System.Pack_60 + RE_Get_60, -- System.Pack_60 + RE_GetU_60, -- System.Pack_60 + RE_Set_60, -- System.Pack_60 + RE_SetU_60, -- System.Pack_60 + + RE_Bits_61, -- System.Pack_61 + RE_Get_61, -- System.Pack_61 + RE_Set_61, -- System.Pack_61 + + RE_Bits_62, -- System.Pack_62 + RE_Get_62, -- System.Pack_62 + RE_GetU_62, -- System.Pack_62 + RE_Set_62, -- System.Pack_62 + RE_SetU_62, -- System.Pack_62 + + RE_Bits_63, -- System.Pack_63 + RE_Get_63, -- System.Pack_63 + RE_Set_63, -- System.Pack_63 + + RE_Adjust_Storage_Size, -- System_Parameters + RE_Default_Stack_Size, -- System.Parameters + RE_Garbage_Collected, -- System.Parameters + RE_Size_Type, -- System.Parameters + RE_Unspecified_Size, -- System.Parameters + + RE_Get_Active_Partition_Id, -- System.Partition_Interface + RE_Get_Passive_Partition_Id, -- System.Partition_Interface + RE_Get_Local_Partition_Id, -- System.Partition_Interface + RE_Get_RCI_Package_Receiver, -- System.Partition_Interface + RE_Get_Unique_Remote_Pointer, -- System.Partition_Interface + RE_RACW_Stub_Type, -- System.Partition_Interface + RE_RACW_Stub_Type_Access, -- System.Partition_Interface + RE_Raise_Program_Error_For_E_4_18, -- System.Partition_Interface + RE_Raise_Program_Error_Unknown_Tag, -- System.Partition_Interface + RE_Register_Passive_Package, -- System.Partition_Interface + RE_Register_Receiving_Stub, -- System.Partition_Interface + RE_RCI_Info, -- System.Partition_Interface + RE_Subprogram_Id, -- System.Partition_Interface + + RE_Global_Pool_Object, -- System.Pool_Global + + RE_Unbounded_Reclaim_Pool, -- System.Pool_Local + + RE_Stack_Bounded_Pool, -- System.Pool_Size + + RE_Do_Apc, -- System.RPC + RE_Do_Rpc, -- System.RPC + RE_Params_Stream_Type, -- System.RPC + RE_Partition_ID, -- System.RPC + RE_RPC_Receiver, -- System.RPC + + RE_IS_Is1, -- System.Scalar_Values + RE_IS_Is2, -- System.Scalar_Values + RE_IS_Is4, -- System.Scalar_Values + RE_IS_Is8, -- System.Scalar_Values + RE_IS_Iu1, -- System.Scalar_Values + RE_IS_Iu2, -- System.Scalar_Values + RE_IS_Iu4, -- System.Scalar_Values + RE_IS_Iu8, -- System.Scalar_Values + RE_IS_Isf, -- System.Scalar_Values + RE_IS_Ifl, -- System.Scalar_Values + RE_IS_Ilf, -- System.Scalar_Values + RE_IS_Ill, -- System.Scalar_Values + + RE_Mark_Id, -- System.Secondary_Stack + RE_SS_Allocate, -- System.Secondary_Stack + RE_SS_Pool, -- System.Secondary_Stack + RE_SS_Mark, -- System.Secondary_Stack + RE_SS_Release, -- System.Secondary_Stack + + RE_Shared_Var_Close, -- System.Shared_Storage + RE_Shared_Var_Lock, -- System.Shared_Storage + RE_Shared_Var_ROpen, -- System.Shared_Storage + RE_Shared_Var_Unlock, -- System.Shared_Storage + RE_Shared_Var_WOpen, -- System.Shared_Storage + + RE_Abort_Undefer_Direct, -- System.Standard_Library + RE_Exception_Data, -- System.Standard_Library + RE_Exception_Data_Ptr, -- System.Standard_Library + + RE_Integer_Address, -- System.Storage_Elements + RE_Storage_Offset, -- System.Storage_Elements + RE_Storage_Array, -- System.Storage_Elements + RE_To_Address, -- System.Storage_Elements + + RE_Root_Storage_Pool, -- System.Storage_Pools + + RE_Thin_Pointer, -- System.Stream_Attributes + RE_Fat_Pointer, -- System.Stream_Attributes + + RE_I_AD, -- System.Stream_Attributes + RE_I_AS, -- System.Stream_Attributes + RE_I_B, -- System.Stream_Attributes + RE_I_C, -- System.Stream_Attributes + RE_I_F, -- System.Stream_Attributes + RE_I_I, -- System.Stream_Attributes + RE_I_LF, -- System.Stream_Attributes + RE_I_LI, -- System.Stream_Attributes + RE_I_LLF, -- System.Stream_Attributes + RE_I_LLI, -- System.Stream_Attributes + RE_I_LLU, -- System.Stream_Attributes + RE_I_LU, -- System.Stream_Attributes + RE_I_SF, -- System.Stream_Attributes + RE_I_SI, -- System.Stream_Attributes + RE_I_SSI, -- System.Stream_Attributes + RE_I_SSU, -- System.Stream_Attributes + RE_I_SU, -- System.Stream_Attributes + RE_I_U, -- System.Stream_Attributes + RE_I_WC, -- System.Stream_Attributes + + RE_W_AD, -- System.Stream_Attributes + RE_W_AS, -- System.Stream_Attributes + RE_W_B, -- System.Stream_Attributes + RE_W_C, -- System.Stream_Attributes + RE_W_F, -- System.Stream_Attributes + RE_W_I, -- System.Stream_Attributes + RE_W_LF, -- System.Stream_Attributes + RE_W_LI, -- System.Stream_Attributes + RE_W_LLF, -- System.Stream_Attributes + RE_W_LLI, -- System.Stream_Attributes + RE_W_LLU, -- System.Stream_Attributes + RE_W_LU, -- System.Stream_Attributes + RE_W_SF, -- System.Stream_Attributes + RE_W_SI, -- System.Stream_Attributes + RE_W_SSI, -- System.Stream_Attributes + RE_W_SSU, -- System.Stream_Attributes + RE_W_SU, -- System.Stream_Attributes + RE_W_U, -- System.Stream_Attributes + RE_W_WC, -- System.Stream_Attributes + + RE_Str_Concat, -- System.String_Ops + RE_Str_Concat_CC, -- System.String_Ops + RE_Str_Concat_CS, -- System.String_Ops + RE_Str_Concat_SC, -- System.String_Ops + RE_Str_Equal, -- System.String_Ops + RE_Str_Normalize, -- System.String_Ops + RE_Wide_Str_Normalize, -- System.String_Ops + + RE_Str_Concat_3, -- System.String_Ops_Concat_3 + + RE_Str_Concat_4, -- System.String_Ops_Concat_4 + + RE_Str_Concat_5, -- System.String_Ops_Concat_5 + + RE_Free_Task_Image, -- System.Task_Info + RE_Task_Info_Type, -- System.Task_Info + RE_Task_Image_Type, -- System_Task_Info + RE_Unspecified_Task_Info, -- System.Task_Info + + RE_Library_Task_Level, -- System.Tasking + + RE_Task_Procedure_Access, -- System.Tasking + + RO_ST_Task_ID, -- System.Tasking + + RE_Call_Modes, -- System.Tasking + RE_Simple_Call, -- System.Tasking + RE_Conditional_Call, -- System.Tasking + RE_Asynchronous_Call, -- System.Tasking + RE_Timed_Call, -- System.Tasking + + RE_Task_List, -- System.Tasking + + RE_Accept_Alternative, -- System.Tasking + RE_Accept_List, -- System.Tasking + RE_Accept_List_Access, -- System.Tasking + RE_Max_Select, -- System.Tasking + RE_Max_Task_Entry, -- System.Tasking + RE_No_Rendezvous, -- System.Tasking + RE_Null_Task_Entry, -- System.Tasking + RE_Positive_Select_Index, -- System.Tasking + RE_Select_Index, -- System.Tasking + RE_Select_Modes, -- System.Tasking + RE_Else_Mode, -- System.Tasking + RE_Simple_Mode, -- System.Tasking + RE_Terminate_Mode, -- System.Tasking + RE_Delay_Mode, -- System.Tasking + RE_Task_Entry_Index, -- System.Tasking + RE_Self, -- System.Tasking + + RE_Master_Id, -- System.Tasking + RE_Unspecified_Priority, -- System.Tasking + + RE_Activation_Chain, -- System.Tasking + + RE_Abort_Defer, -- System.Soft_Links + RE_Abort_Undefer, -- System.Soft_Links + RE_Complete_Master, -- System.Soft_Links + RE_Current_Master, -- System.Soft_Links + RE_Enter_Master, -- System.Soft_Links + RE_Get_Current_Excep, -- System.Soft_Links + RE_Get_GNAT_Exception, -- System.Soft_Links + RE_Update_Exception, -- System.Soft_Links + + RE_Bits_1, -- System.Unsigned_Types + RE_Bits_2, -- System.Unsigned_Types + RE_Bits_4, -- System.Unsigned_Types + RE_Float_Unsigned, -- System.Unsigned_Types + RE_Long_Long_Unsigned, -- System.Unsigned_Types + RE_Packed_Byte, -- System.Unsigned_Types + RE_Packed_Bytes1, -- System.Unsigned_Types + RE_Packed_Bytes2, -- System.Unsigned_Types + RE_Packed_Bytes4, -- System.Unsigned_Types + RE_Unsigned, -- System.Unsigned_Types + + RE_Value_Boolean, -- System.Val_Bool + + RE_Value_Character, -- System.Val_Char + + RE_Value_Decimal, -- System.Val_Dec + + RE_Value_Enumeration_8, -- System.Val_Enum + RE_Value_Enumeration_16, -- System.Val_Enum + RE_Value_Enumeration_32, -- System.Val_Enum + + RE_Value_Integer, -- System.Val_Int + + RE_Value_Long_Long_Decimal, -- System.Val_LLD + + RE_Value_Long_Long_Integer, -- System.Val_LLI + + RE_Value_Long_Long_Unsigned, -- System.Val_LLU + + RE_Value_Real, -- System.Val_Real + + RE_Value_Unsigned, -- System.Val_Uns + + RE_Value_Wide_Character, -- System.Val_WChar + + RE_D, -- System.Vax_Float_Operations + RE_F, -- System.Vax_Float_Operations + RE_G, -- System.Vax_Float_Operations + RE_Q, -- System.Vax_Float_Operations + RE_S, -- System.Vax_Float_Operations + RE_T, -- System.Vax_Float_Operations + + RE_D_To_G, -- System.Vax_Float_Operations + RE_F_To_G, -- System.Vax_Float_Operations + RE_F_To_Q, -- System.Vax_Float_Operations + RE_F_To_S, -- System.Vax_Float_Operations + RE_G_To_D, -- System.Vax_Float_Operations + RE_G_To_F, -- System.Vax_Float_Operations + RE_G_To_Q, -- System.Vax_Float_Operations + RE_G_To_T, -- System.Vax_Float_Operations + RE_Q_To_F, -- System.Vax_Float_Operations + RE_Q_To_G, -- System.Vax_Float_Operations + RE_S_To_F, -- System.Vax_Float_Operations + RE_T_To_D, -- System.Vax_Float_Operations + RE_T_To_G, -- System.Vax_Float_Operations + + RE_Abs_F, -- System.Vax_Float_Operations + RE_Abs_G, -- System.Vax_Float_Operations + RE_Add_F, -- System.Vax_Float_Operations + RE_Add_G, -- System.Vax_Float_Operations + RE_Div_F, -- System.Vax_Float_Operations + RE_Div_G, -- System.Vax_Float_Operations + RE_Mul_F, -- System.Vax_Float_Operations + RE_Mul_G, -- System.Vax_Float_Operations + RE_Neg_F, -- System.Vax_Float_Operations + RE_Neg_G, -- System.Vax_Float_Operations + RE_Sub_F, -- System.Vax_Float_Operations + RE_Sub_G, -- System.Vax_Float_Operations + + RE_Eq_F, -- System.Vax_Float_Operations + RE_Eq_G, -- System.Vax_Float_Operations + RE_Le_F, -- System.Vax_Float_Operations + RE_Le_G, -- System.Vax_Float_Operations + RE_Lt_F, -- System.Vax_Float_Operations + RE_Lt_G, -- System.Vax_Float_Operations + + RE_Version_String, -- System.Version_Control + RE_Get_Version_String, -- System.Version_Control + + RE_Register_VMS_Exception, -- System.VMS_Exception_Table + + RE_String_To_Wide_String, -- System.WCh_StW + + RE_Wide_String_To_String, -- System.WCh_WtS + + RE_Wide_Width_Character, -- System.WWd_Char + + RE_Wide_Width_Enumeration_8, -- System.WWd_Enum + RE_Wide_Width_Enumeration_16, -- System.WWd_Enum + RE_Wide_Width_Enumeration_32, -- System.WWd_Enum + + RE_Wide_Width_Wide_Character, -- System.WWd_Wchar + + RE_Width_Boolean, -- System.Wid_Bool + + RE_Width_Character, -- System.Wid_Char + + RE_Width_Enumeration_8, -- System.Wid_Enum + RE_Width_Enumeration_16, -- System.Wid_Enum + RE_Width_Enumeration_32, -- System.Wid_Enum + + RE_Width_Long_Long_Integer, -- System.Wid_LLI + + RE_Width_Long_Long_Unsigned, -- System.Wid_LLU + + RE_Width_Wide_Character, -- System.Wid_WChar + + RE_Protected_Entry_Body_Array, -- Tasking.Protected_Objects.Entries + RE_Protection_Entries, -- Tasking.Protected_Objects.Entries + RE_Initialize_Protection_Entries, -- Tasking.Protected_Objects.Entries + RE_Lock_Entries, -- Tasking.Protected_Objects.Entries + RE_Lock_Read_Only_Entries, -- Tasking.Protected_Objects.Entries + RE_Unlock_Entries, -- Tasking.Protected_Objects.Entries + RE_Communication_Block, -- Protected_Objects.Operations + RE_Protected_Entry_Call, -- Protected_Objects.Operations + RE_Service_Entries, -- Protected_Objects.Operations + RE_Cancel_Protected_Entry_Call, -- Protected_Objects.Operations + RE_Enqueued, -- Protected_Objects.Operations + RE_Cancelled, -- Protected_Objects.Operations + RE_Complete_Entry_Body, -- Protected_Objects.Operations + RE_Exceptional_Complete_Entry_Body, -- Protected_Objects.Operations + RE_Requeue_Protected_Entry, -- Protected_Objects.Operations + RE_Requeue_Task_To_Protected_Entry, -- Protected_Objects.Operations + RE_Protected_Count, -- Protected_Objects.Operations + RE_Protected_Entry_Caller, -- Protected_Objects.Operations + RE_Timed_Protected_Entry_Call, -- Protected_Objects.Operations + + RE_Protection_Entry, -- Protected_Objects.Single_Entry + RE_Initialize_Protection_Entry, -- Protected_Objects.Single_Entry + RE_Lock_Entry, -- Protected_Objects.Single_Entry + RE_Lock_Read_Only_Entry, -- Protected_Objects.Single_Entry + RE_Unlock_Entry, -- Protected_Objects.Single_Entry + RE_Protected_Single_Entry_Call, -- Protected_Objects.Single_Entry + RE_Service_Entry, -- Protected_Objects.Single_Entry + RE_Complete_Single_Entry_Body, -- Protected_Objects.Single_Entry + RE_Exceptional_Complete_Single_Entry_Body, + RE_Protected_Count_Entry, -- Protected_Objects.Single_Entry + RE_Protected_Single_Entry_Caller, -- Protected_Objects.Single_Entry + RE_Timed_Protected_Single_Entry_Call, + + RE_Protected_Entry_Index, -- System.Tasking.Protected_Objects + RE_Entry_Body, -- System.Tasking.Protected_Objects + RE_Protection, -- System.Tasking.Protected_Objects + RE_Initialize_Protection, -- System.Tasking.Protected_Objects + RE_Finalize_Protection, -- System.Tasking.Protected_Objects + RE_Lock, -- System.Tasking.Protected_Objects + RE_Lock_Read_Only, -- System.Tasking.Protected_Objects + RE_Unlock, -- System.Tasking.Protected_Objects + + RE_Delay_Block, -- System.Tasking.Async_Delays + RE_Timed_Out, -- System.Tasking.Async_Delays + RE_Cancel_Async_Delay, -- System.Tasking.Async_Delays + RE_Enqueue_Duration, -- System.Tasking.Async_Delays + RE_Enqueue_Calendar, -- System.Tasking.Async_Delays + RE_Enqueue_RT, -- System.Tasking.Async_Delays + + RE_Accept_Call, -- System.Tasking.Rendezvous + RE_Accept_Trivial, -- System.Tasking.Rendezvous + RE_Callable, -- System.Tasking.Rendezvous + RE_Call_Simple, -- System.Tasking.Rendezvous + RE_Requeue_Task_Entry, -- System.Tasking.Rendezvous + RE_Requeue_Protected_To_Task_Entry, -- System.Tasking.Rendezvous + RE_Cancel_Task_Entry_Call, -- System.Tasking.Rendezvous + RE_Complete_Rendezvous, -- System.Tasking.Rendezvous + RE_Task_Count, -- System.Tasking.Rendezvous + RE_Exceptional_Complete_Rendezvous, -- System.Tasking.Rendezvous + RE_Selective_Wait, -- System.Tasking.Rendezvous + RE_Task_Entry_Call, -- System.Tasking.Rendezvous + RE_Task_Entry_Caller, -- System.Tasking.Rendezvous + RE_Timed_Task_Entry_Call, -- System.Tasking.Rendezvous + RE_Timed_Selective_Wait, -- System.Tasking.Rendezvous + + RE_Activate_Restricted_Tasks, -- System.Tasking.Restricted.Stages + RE_Complete_Restricted_Activation, -- System.Tasking.Restricted.Stages + RE_Create_Restricted_Task, -- System.Tasking.Restricted.Stages + RE_Complete_Restricted_Task, -- System.Tasking.Restricted.Stages + RE_Restricted_Terminated, -- System.Tasking.Restricted.Stages + + RE_Abort_Tasks, -- System.Tasking.Stages + RE_Activate_Tasks, -- System.Tasking.Stages + RE_Complete_Activation, -- System.Tasking.Stages + RE_Create_Task, -- System.Tasking.Stages + RE_Complete_Task, -- System.Tasking.Stages + RE_Free_Task, -- System.Tasking.Stages + RE_Expunge_Unactivated_Tasks, -- System.Tasking.Stages + RE_Terminated); -- System.Tasking.Stages + + -- The following declarations build a table that is indexed by the + -- RTE function to determine the unit containing the given entity. + -- This table is sorted in order of package names. + + RE_Unit_Table : array (RE_Id) of RTU_Id := ( + + RE_Null => RTU_Null, + + RE_Code_Loc => Ada_Exceptions, + RE_Current_Target_Exception => Ada_Exceptions, -- of JGNAT + RE_Exception_Id => Ada_Exceptions, + RE_Exception_Information => Ada_Exceptions, + RE_Exception_Message => Ada_Exceptions, + RE_Exception_Name_Simple => Ada_Exceptions, + RE_Exception_Occurrence => Ada_Exceptions, + RE_Null_Id => Ada_Exceptions, + RE_Null_Occurrence => Ada_Exceptions, + RE_Poll => Ada_Exceptions, + RE_Raise_Exception => Ada_Exceptions, + RE_Raise_Exception_Always => Ada_Exceptions, + RE_Reraise_Occurrence => Ada_Exceptions, + RE_Reraise_Occurrence_Always => Ada_Exceptions, + RE_Reraise_Occurrence_No_Defer => Ada_Exceptions, + RE_Save_Occurrence => Ada_Exceptions, + + RE_Simple_List_Controller => Ada_Finalization_List_Controller, + RE_List_Controller => Ada_Finalization_List_Controller, + + RE_Interrupt_Id => Ada_Interrupts, + + RE_Root_Stream_Type => Ada_Streams, + RE_Stream_Element => Ada_Streams, + RE_Stream_Element_Offset => Ada_Streams, + RE_Stream_Element_Array => Ada_Streams, + + RE_Stream_Access => Ada_Streams_Stream_IO, + + RE_CW_Membership => Ada_Tags, + RE_DT_Entry_Size => Ada_Tags, + RE_DT_Prologue_Size => Ada_Tags, + RE_External_Tag => Ada_Tags, + RE_Get_Expanded_Name => Ada_Tags, + RE_Get_External_Tag => Ada_Tags, + RE_Get_Prim_Op_Address => Ada_Tags, + RE_Get_RC_Offset => Ada_Tags, + RE_Get_Remotely_Callable => Ada_Tags, + RE_Get_TSD => Ada_Tags, + RE_Inherit_DT => Ada_Tags, + RE_Inherit_TSD => Ada_Tags, + RE_Internal_Tag => Ada_Tags, + RE_Register_Tag => Ada_Tags, + RE_Set_Expanded_Name => Ada_Tags, + RE_Set_External_Tag => Ada_Tags, + RE_Set_Prim_Op_Address => Ada_Tags, + RE_Set_RC_Offset => Ada_Tags, + RE_Set_Remotely_Callable => Ada_Tags, + RE_Set_TSD => Ada_Tags, + RE_Tag_Error => Ada_Tags, + RE_TSD_Entry_Size => Ada_Tags, + RE_TSD_Prologue_Size => Ada_Tags, + RE_Tag => Ada_Tags, + RE_Address_Array => Ada_Tags, + + RE_Current_Task => Ada_Task_Identification, + RO_AT_Task_ID => Ada_Task_Identification, + + RO_CA_Time => Ada_Calendar, + RO_CA_Delay_For => Ada_Calendar_Delays, + RO_CA_Delay_Until => Ada_Calendar_Delays, + RO_CA_To_Duration => Ada_Calendar_Delays, + + RO_RT_Time => Ada_Real_Time, + RO_RT_Delay_Until => Ada_Real_Time_Delays, + RO_RT_To_Duration => Ada_Real_Time_Delays, + + RE_Integer_64 => Interfaces, + RE_Unsigned_8 => Interfaces, + RE_Unsigned_16 => Interfaces, + RE_Unsigned_32 => Interfaces, + RE_Unsigned_64 => Interfaces, + + RE_Vtable_Ptr => Interfaces_CPP, + RE_Displaced_This => Interfaces_CPP, + RE_CPP_CW_Membership => Interfaces_CPP, + RE_CPP_DT_Entry_Size => Interfaces_CPP, + RE_CPP_DT_Prologue_Size => Interfaces_CPP, + RE_CPP_Get_Expanded_Name => Interfaces_CPP, + RE_CPP_Get_External_Tag => Interfaces_CPP, + RE_CPP_Get_Prim_Op_Address => Interfaces_CPP, + RE_CPP_Get_RC_Offset => Interfaces_CPP, + RE_CPP_Get_Remotely_Callable => Interfaces_CPP, + RE_CPP_Get_TSD => Interfaces_CPP, + RE_CPP_Inherit_DT => Interfaces_CPP, + RE_CPP_Inherit_TSD => Interfaces_CPP, + RE_CPP_Register_Tag => Interfaces_CPP, + RE_CPP_Set_Expanded_Name => Interfaces_CPP, + RE_CPP_Set_External_Tag => Interfaces_CPP, + RE_CPP_Set_Prim_Op_Address => Interfaces_CPP, + RE_CPP_Set_RC_Offset => Interfaces_CPP, + RE_CPP_Set_Remotely_Callable => Interfaces_CPP, + RE_CPP_Set_TSD => Interfaces_CPP, + RE_CPP_TSD_Entry_Size => Interfaces_CPP, + RE_CPP_TSD_Prologue_Size => Interfaces_CPP, + + RE_Packed_Size => Interfaces_Packed_Decimal, + RE_Packed_To_Int32 => Interfaces_Packed_Decimal, + RE_Packed_To_Int64 => Interfaces_Packed_Decimal, + RE_Int32_To_Packed => Interfaces_Packed_Decimal, + RE_Int64_To_Packed => Interfaces_Packed_Decimal, + + RE_Address => System, + RE_Any_Priority => System, + RE_Bit_Order => System, + RE_Default_Priority => System, + RE_High_Order_First => System, + RE_Interrupt_Priority => System, + RE_Lib_Stop => System, + RE_Low_Order_First => System, + RE_Max_Interrupt_Priority => System, + RE_Max_Priority => System, + RE_Null_Address => System, + RE_Priority => System, + + RE_Add_With_Ovflo_Check => System_Arith_64, + RE_Double_Divide => System_Arith_64, + RE_Multiply_With_Ovflo_Check => System_Arith_64, + RE_Scaled_Divide => System_Arith_64, + RE_Subtract_With_Ovflo_Check => System_Arith_64, + + RE_Create_AST_Handler => System_AST_Handling, + + RE_Raise_Assert_Failure => System_Assertions, + + RE_AST_Handler => System_Aux_DEC, + RE_Import_Value => System_Aux_DEC, + RE_No_AST_Handler => System_Aux_DEC, + RE_Type_Class => System_Aux_DEC, + RE_Type_Class_Enumeration => System_Aux_DEC, + RE_Type_Class_Integer => System_Aux_DEC, + RE_Type_Class_Fixed_Point => System_Aux_DEC, + RE_Type_Class_Floating_Point => System_Aux_DEC, + RE_Type_Class_Array => System_Aux_DEC, + RE_Type_Class_Record => System_Aux_DEC, + RE_Type_Class_Access => System_Aux_DEC, + RE_Type_Class_Task => System_Aux_DEC, + RE_Type_Class_Address => System_Aux_DEC, + + RE_Bit_And => System_Bit_Ops, + RE_Bit_Eq => System_Bit_Ops, + RE_Bit_Not => System_Bit_Ops, + RE_Bit_Or => System_Bit_Ops, + RE_Bit_Xor => System_Bit_Ops, + + RE_Checked_Pool => System_Checked_Pools, + + RE_Register_Exception => System_Exception_Table, + + RE_All_Others_Id => System_Exceptions, + RE_Handler_Record => System_Exceptions, + RE_Handler_Record_Ptr => System_Exceptions, + RE_Others_Id => System_Exceptions, + RE_Subprogram_Descriptor => System_Exceptions, + RE_Subprogram_Descriptor_0 => System_Exceptions, + RE_Subprogram_Descriptor_1 => System_Exceptions, + RE_Subprogram_Descriptor_2 => System_Exceptions, + RE_Subprogram_Descriptor_3 => System_Exceptions, + RE_Subprogram_Descriptor_List => System_Exceptions, + RE_Subprogram_Descriptor_Ptr => System_Exceptions, + RE_Subprogram_Descriptors_Record => System_Exceptions, + RE_Subprogram_Descriptors_Ptr => System_Exceptions, + + RE_Exn_Float => System_Exn_Flt, + + RE_Exn_Integer => System_Exn_Int, + + RE_Exn_Long_Float => System_Exn_LFlt, + + RE_Exn_Long_Integer => System_Exn_LInt, + + RE_Exn_Long_Long_Float => System_Exn_LLF, + + RE_Exn_Long_Long_Integer => System_Exn_LLI, + + RE_Exn_Short_Float => System_Exn_SFlt, + + RE_Exn_Short_Integer => System_Exn_SInt, + + RE_Exn_Short_Short_Integer => System_Exn_SSI, + + RE_Exp_Float => System_Exp_Flt, + + RE_Exp_Integer => System_Exp_Int, + + RE_Exp_Long_Float => System_Exp_LFlt, + + RE_Exp_Long_Integer => System_Exp_LInt, + + RE_Exp_Long_Long_Float => System_Exp_LLF, + + RE_Exp_Long_Long_Integer => System_Exp_LLI, + + RE_Exp_Long_Long_Unsigned => System_Exp_LLU, + + RE_Exp_Modular => System_Exp_Mod, + + RE_Exp_Short_Float => System_Exp_SFlt, + + RE_Exp_Short_Integer => System_Exp_SInt, + + RE_Exp_Short_Short_Integer => System_Exp_SSI, + + RE_Exp_Unsigned => System_Exp_Uns, + + RE_Fat_Float => System_Fat_Flt, + + RE_Fat_Long_Float => System_Fat_LFlt, + + RE_Fat_Long_Long_Float => System_Fat_LLF, + + RE_Fat_Short_Float => System_Fat_SFlt, + + RE_Attach_To_Final_List => System_Finalization_Implementation, + RE_Finalize_List => System_Finalization_Implementation, + RE_Finalize_One => System_Finalization_Implementation, + RE_Global_Final_List => System_Finalization_Implementation, + RE_Record_Controller => System_Finalization_Implementation, + RE_Limited_Record_Controller => System_Finalization_Implementation, + RE_Deep_Tag_Initialize => System_Finalization_Implementation, + RE_Deep_Tag_Adjust => System_Finalization_Implementation, + RE_Deep_Tag_Finalize => System_Finalization_Implementation, + RE_Deep_Tag_Attach => System_Finalization_Implementation, + RE_Deep_Rec_Initialize => System_Finalization_Implementation, + RE_Deep_Rec_Adjust => System_Finalization_Implementation, + RE_Deep_Rec_Finalize => System_Finalization_Implementation, + + RE_Root_Controlled => System_Finalization_Root, + RE_Finalizable => System_Finalization_Root, + RE_Finalizable_Ptr => System_Finalization_Root, + + RE_Fore => System_Fore, + + RE_Image_Boolean => System_Img_Bool, + + RE_Image_Character => System_Img_Char, + + RE_Image_Decimal => System_Img_Dec, + + RE_Image_Enumeration_8 => System_Img_Enum, + RE_Image_Enumeration_16 => System_Img_Enum, + RE_Image_Enumeration_32 => System_Img_Enum, + + RE_Image_Integer => System_Img_Int, + + RE_Image_Long_Long_Decimal => System_Img_LLD, + + RE_Image_Long_Long_Integer => System_Img_LLI, + + RE_Image_Long_Long_Unsigned => System_Img_LLU, + + RE_Image_Ordinary_Fixed_Point => System_Img_Real, + RE_Image_Floating_Point => System_Img_Real, + + RE_Image_Unsigned => System_Img_Uns, + + RE_Image_Wide_Character => System_Img_WChar, + + RE_Bind_Interrupt_To_Entry => System_Interrupts, + RE_Default_Interrupt_Priority => System_Interrupts, + RE_Dynamic_Interrupt_Protection => System_Interrupts, + RE_Install_Handlers => System_Interrupts, + RE_Register_Interrupt_Handler => System_Interrupts, + RE_Static_Interrupt_Protection => System_Interrupts, + + RE_Asm_Insn => System_Machine_Code, + RE_Asm_Input_Operand => System_Machine_Code, + RE_Asm_Output_Operand => System_Machine_Code, + + RE_Mantissa_Value => System_Mantissa, + + RE_Bits_03 => System_Pack_03, + RE_Get_03 => System_Pack_03, + RE_Set_03 => System_Pack_03, + + RE_Bits_05 => System_Pack_05, + RE_Get_05 => System_Pack_05, + RE_Set_05 => System_Pack_05, + + RE_Bits_06 => System_Pack_06, + RE_Get_06 => System_Pack_06, + RE_GetU_06 => System_Pack_06, + RE_Set_06 => System_Pack_06, + RE_SetU_06 => System_Pack_06, + + RE_Bits_07 => System_Pack_07, + RE_Get_07 => System_Pack_07, + RE_Set_07 => System_Pack_07, + + RE_Bits_09 => System_Pack_09, + RE_Get_09 => System_Pack_09, + RE_Set_09 => System_Pack_09, + + RE_Bits_10 => System_Pack_10, + RE_Get_10 => System_Pack_10, + RE_GetU_10 => System_Pack_10, + RE_Set_10 => System_Pack_10, + RE_SetU_10 => System_Pack_10, + + RE_Bits_11 => System_Pack_11, + RE_Get_11 => System_Pack_11, + RE_Set_11 => System_Pack_11, + + RE_Bits_12 => System_Pack_12, + RE_Get_12 => System_Pack_12, + RE_GetU_12 => System_Pack_12, + RE_Set_12 => System_Pack_12, + RE_SetU_12 => System_Pack_12, + + RE_Bits_13 => System_Pack_13, + RE_Get_13 => System_Pack_13, + RE_Set_13 => System_Pack_13, + + RE_Bits_14 => System_Pack_14, + RE_Get_14 => System_Pack_14, + RE_GetU_14 => System_Pack_14, + RE_Set_14 => System_Pack_14, + RE_SetU_14 => System_Pack_14, + + RE_Bits_15 => System_Pack_15, + RE_Get_15 => System_Pack_15, + RE_Set_15 => System_Pack_15, + + RE_Bits_17 => System_Pack_17, + RE_Get_17 => System_Pack_17, + RE_Set_17 => System_Pack_17, + + RE_Bits_18 => System_Pack_18, + RE_Get_18 => System_Pack_18, + RE_GetU_18 => System_Pack_18, + RE_Set_18 => System_Pack_18, + RE_SetU_18 => System_Pack_18, + + RE_Bits_19 => System_Pack_19, + RE_Get_19 => System_Pack_19, + RE_Set_19 => System_Pack_19, + + RE_Bits_20 => System_Pack_20, + RE_Get_20 => System_Pack_20, + RE_GetU_20 => System_Pack_20, + RE_Set_20 => System_Pack_20, + RE_SetU_20 => System_Pack_20, + + RE_Bits_21 => System_Pack_21, + RE_Get_21 => System_Pack_21, + RE_Set_21 => System_Pack_21, + + RE_Bits_22 => System_Pack_22, + RE_Get_22 => System_Pack_22, + RE_GetU_22 => System_Pack_22, + RE_Set_22 => System_Pack_22, + RE_SetU_22 => System_Pack_22, + + RE_Bits_23 => System_Pack_23, + RE_Get_23 => System_Pack_23, + RE_Set_23 => System_Pack_23, + + RE_Bits_24 => System_Pack_24, + RE_Get_24 => System_Pack_24, + RE_GetU_24 => System_Pack_24, + RE_Set_24 => System_Pack_24, + RE_SetU_24 => System_Pack_24, + + RE_Bits_25 => System_Pack_25, + RE_Get_25 => System_Pack_25, + RE_Set_25 => System_Pack_25, + + RE_Bits_26 => System_Pack_26, + RE_Get_26 => System_Pack_26, + RE_GetU_26 => System_Pack_26, + RE_Set_26 => System_Pack_26, + RE_SetU_26 => System_Pack_26, + + RE_Bits_27 => System_Pack_27, + RE_Get_27 => System_Pack_27, + RE_Set_27 => System_Pack_27, + + RE_Bits_28 => System_Pack_28, + RE_Get_28 => System_Pack_28, + RE_GetU_28 => System_Pack_28, + RE_Set_28 => System_Pack_28, + RE_SetU_28 => System_Pack_28, + + RE_Bits_29 => System_Pack_29, + RE_Get_29 => System_Pack_29, + RE_Set_29 => System_Pack_29, + + RE_Bits_30 => System_Pack_30, + RE_Get_30 => System_Pack_30, + RE_GetU_30 => System_Pack_30, + RE_Set_30 => System_Pack_30, + RE_SetU_30 => System_Pack_30, + + RE_Bits_31 => System_Pack_31, + RE_Get_31 => System_Pack_31, + RE_Set_31 => System_Pack_31, + + RE_Bits_33 => System_Pack_33, + RE_Get_33 => System_Pack_33, + RE_Set_33 => System_Pack_33, + + RE_Bits_34 => System_Pack_34, + RE_Get_34 => System_Pack_34, + RE_GetU_34 => System_Pack_34, + RE_Set_34 => System_Pack_34, + RE_SetU_34 => System_Pack_34, + + RE_Bits_35 => System_Pack_35, + RE_Get_35 => System_Pack_35, + RE_Set_35 => System_Pack_35, + + RE_Bits_36 => System_Pack_36, + RE_Get_36 => System_Pack_36, + RE_GetU_36 => System_Pack_36, + RE_Set_36 => System_Pack_36, + RE_SetU_36 => System_Pack_36, + + RE_Bits_37 => System_Pack_37, + RE_Get_37 => System_Pack_37, + RE_Set_37 => System_Pack_37, + + RE_Bits_38 => System_Pack_38, + RE_Get_38 => System_Pack_38, + RE_GetU_38 => System_Pack_38, + RE_Set_38 => System_Pack_38, + RE_SetU_38 => System_Pack_38, + + RE_Bits_39 => System_Pack_39, + RE_Get_39 => System_Pack_39, + RE_Set_39 => System_Pack_39, + + RE_Bits_40 => System_Pack_40, + RE_Get_40 => System_Pack_40, + RE_GetU_40 => System_Pack_40, + RE_Set_40 => System_Pack_40, + RE_SetU_40 => System_Pack_40, + + RE_Bits_41 => System_Pack_41, + RE_Get_41 => System_Pack_41, + RE_Set_41 => System_Pack_41, + + RE_Bits_42 => System_Pack_42, + RE_Get_42 => System_Pack_42, + RE_GetU_42 => System_Pack_42, + RE_Set_42 => System_Pack_42, + RE_SetU_42 => System_Pack_42, + + RE_Bits_43 => System_Pack_43, + RE_Get_43 => System_Pack_43, + RE_Set_43 => System_Pack_43, + + RE_Bits_44 => System_Pack_44, + RE_Get_44 => System_Pack_44, + RE_GetU_44 => System_Pack_44, + RE_Set_44 => System_Pack_44, + RE_SetU_44 => System_Pack_44, + + RE_Bits_45 => System_Pack_45, + RE_Get_45 => System_Pack_45, + RE_Set_45 => System_Pack_45, + + RE_Bits_46 => System_Pack_46, + RE_Get_46 => System_Pack_46, + RE_GetU_46 => System_Pack_46, + RE_Set_46 => System_Pack_46, + RE_SetU_46 => System_Pack_46, + + RE_Bits_47 => System_Pack_47, + RE_Get_47 => System_Pack_47, + RE_Set_47 => System_Pack_47, + + RE_Bits_48 => System_Pack_48, + RE_Get_48 => System_Pack_48, + RE_GetU_48 => System_Pack_48, + RE_Set_48 => System_Pack_48, + RE_SetU_48 => System_Pack_48, + + RE_Bits_49 => System_Pack_49, + RE_Get_49 => System_Pack_49, + RE_Set_49 => System_Pack_49, + + RE_Bits_50 => System_Pack_50, + RE_Get_50 => System_Pack_50, + RE_GetU_50 => System_Pack_50, + RE_Set_50 => System_Pack_50, + RE_SetU_50 => System_Pack_50, + + RE_Bits_51 => System_Pack_51, + RE_Get_51 => System_Pack_51, + RE_Set_51 => System_Pack_51, + + RE_Bits_52 => System_Pack_52, + RE_Get_52 => System_Pack_52, + RE_GetU_52 => System_Pack_52, + RE_Set_52 => System_Pack_52, + RE_SetU_52 => System_Pack_52, + + RE_Bits_53 => System_Pack_53, + RE_Get_53 => System_Pack_53, + RE_Set_53 => System_Pack_53, + + RE_Bits_54 => System_Pack_54, + RE_Get_54 => System_Pack_54, + RE_GetU_54 => System_Pack_54, + RE_Set_54 => System_Pack_54, + RE_SetU_54 => System_Pack_54, + + RE_Bits_55 => System_Pack_55, + RE_Get_55 => System_Pack_55, + RE_Set_55 => System_Pack_55, + + RE_Bits_56 => System_Pack_56, + RE_Get_56 => System_Pack_56, + RE_GetU_56 => System_Pack_56, + RE_Set_56 => System_Pack_56, + RE_SetU_56 => System_Pack_56, + + RE_Bits_57 => System_Pack_57, + RE_Get_57 => System_Pack_57, + RE_Set_57 => System_Pack_57, + + RE_Bits_58 => System_Pack_58, + RE_Get_58 => System_Pack_58, + RE_GetU_58 => System_Pack_58, + RE_Set_58 => System_Pack_58, + RE_SetU_58 => System_Pack_58, + + RE_Bits_59 => System_Pack_59, + RE_Get_59 => System_Pack_59, + RE_Set_59 => System_Pack_59, + + RE_Bits_60 => System_Pack_60, + RE_Get_60 => System_Pack_60, + RE_GetU_60 => System_Pack_60, + RE_Set_60 => System_Pack_60, + RE_SetU_60 => System_Pack_60, + + RE_Bits_61 => System_Pack_61, + RE_Get_61 => System_Pack_61, + RE_Set_61 => System_Pack_61, + + RE_Bits_62 => System_Pack_62, + RE_Get_62 => System_Pack_62, + RE_GetU_62 => System_Pack_62, + RE_Set_62 => System_Pack_62, + RE_SetU_62 => System_Pack_62, + + RE_Bits_63 => System_Pack_63, + RE_Get_63 => System_Pack_63, + RE_Set_63 => System_Pack_63, + + RE_Adjust_Storage_Size => System_Parameters, + RE_Default_Stack_Size => System_Parameters, + RE_Garbage_Collected => System_Parameters, + RE_Size_Type => System_Parameters, + RE_Unspecified_Size => System_Parameters, + + RE_Get_Active_Partition_Id => System_Partition_Interface, + RE_Get_Passive_Partition_Id => System_Partition_Interface, + RE_Get_Local_Partition_Id => System_Partition_Interface, + RE_Get_RCI_Package_Receiver => System_Partition_Interface, + RE_Get_Unique_Remote_Pointer => System_Partition_Interface, + RE_RACW_Stub_Type => System_Partition_Interface, + RE_RACW_Stub_Type_Access => System_Partition_Interface, + RE_Raise_Program_Error_For_E_4_18 => System_Partition_Interface, + RE_Raise_Program_Error_Unknown_Tag => System_Partition_Interface, + RE_Register_Passive_Package => System_Partition_Interface, + RE_Register_Receiving_Stub => System_Partition_Interface, + RE_RCI_Info => System_Partition_Interface, + RE_Subprogram_Id => System_Partition_Interface, + + RE_Global_Pool_Object => System_Pool_Global, + + RE_Unbounded_Reclaim_Pool => System_Pool_Local, + + RE_Stack_Bounded_Pool => System_Pool_Size, + + RE_Do_Apc => System_RPC, + RE_Do_Rpc => System_RPC, + RE_Params_Stream_Type => System_RPC, + RE_Partition_ID => System_RPC, + RE_RPC_Receiver => System_RPC, + + RE_IS_Is1 => System_Scalar_Values, + RE_IS_Is2 => System_Scalar_Values, + RE_IS_Is4 => System_Scalar_Values, + RE_IS_Is8 => System_Scalar_Values, + RE_IS_Iu1 => System_Scalar_Values, + RE_IS_Iu2 => System_Scalar_Values, + RE_IS_Iu4 => System_Scalar_Values, + RE_IS_Iu8 => System_Scalar_Values, + RE_IS_Isf => System_Scalar_Values, + RE_IS_Ifl => System_Scalar_Values, + RE_IS_Ilf => System_Scalar_Values, + RE_IS_Ill => System_Scalar_Values, + + RE_Mark_Id => System_Secondary_Stack, + RE_SS_Allocate => System_Secondary_Stack, + RE_SS_Mark => System_Secondary_Stack, + RE_SS_Pool => System_Secondary_Stack, + RE_SS_Release => System_Secondary_Stack, + + RE_Shared_Var_Close => System_Shared_Storage, + RE_Shared_Var_Lock => System_Shared_Storage, + RE_Shared_Var_ROpen => System_Shared_Storage, + RE_Shared_Var_Unlock => System_Shared_Storage, + RE_Shared_Var_WOpen => System_Shared_Storage, + + RE_Abort_Undefer_Direct => System_Standard_Library, + RE_Exception_Data => System_Standard_Library, + RE_Exception_Data_Ptr => System_Standard_Library, + + RE_Integer_Address => System_Storage_Elements, + RE_Storage_Offset => System_Storage_Elements, + RE_Storage_Array => System_Storage_Elements, + RE_To_Address => System_Storage_Elements, + + RE_Root_Storage_Pool => System_Storage_Pools, + + RE_Thin_Pointer => System_Stream_Attributes, + RE_Fat_Pointer => System_Stream_Attributes, + + RE_I_AD => System_Stream_Attributes, + RE_I_AS => System_Stream_Attributes, + RE_I_B => System_Stream_Attributes, + RE_I_C => System_Stream_Attributes, + RE_I_F => System_Stream_Attributes, + RE_I_I => System_Stream_Attributes, + RE_I_LF => System_Stream_Attributes, + RE_I_LI => System_Stream_Attributes, + RE_I_LLF => System_Stream_Attributes, + RE_I_LLI => System_Stream_Attributes, + RE_I_LLU => System_Stream_Attributes, + RE_I_LU => System_Stream_Attributes, + RE_I_SF => System_Stream_Attributes, + RE_I_SI => System_Stream_Attributes, + RE_I_SSI => System_Stream_Attributes, + RE_I_SSU => System_Stream_Attributes, + RE_I_SU => System_Stream_Attributes, + RE_I_U => System_Stream_Attributes, + RE_I_WC => System_Stream_Attributes, + + RE_W_AD => System_Stream_Attributes, + RE_W_AS => System_Stream_Attributes, + RE_W_B => System_Stream_Attributes, + RE_W_C => System_Stream_Attributes, + RE_W_F => System_Stream_Attributes, + RE_W_I => System_Stream_Attributes, + RE_W_LF => System_Stream_Attributes, + RE_W_LI => System_Stream_Attributes, + RE_W_LLF => System_Stream_Attributes, + RE_W_LLI => System_Stream_Attributes, + RE_W_LLU => System_Stream_Attributes, + RE_W_LU => System_Stream_Attributes, + RE_W_SF => System_Stream_Attributes, + RE_W_SI => System_Stream_Attributes, + RE_W_SSI => System_Stream_Attributes, + RE_W_SSU => System_Stream_Attributes, + RE_W_SU => System_Stream_Attributes, + RE_W_U => System_Stream_Attributes, + RE_W_WC => System_Stream_Attributes, + + RE_Str_Concat => System_String_Ops, + RE_Str_Equal => System_String_Ops, + RE_Str_Normalize => System_String_Ops, + RE_Wide_Str_Normalize => System_String_Ops, + RE_Str_Concat_CC => System_String_Ops, + RE_Str_Concat_CS => System_String_Ops, + RE_Str_Concat_SC => System_String_Ops, + + RE_Str_Concat_3 => System_String_Ops_Concat_3, + + RE_Str_Concat_4 => System_String_Ops_Concat_4, + + RE_Str_Concat_5 => System_String_Ops_Concat_5, + + RE_Free_Task_Image => System_Task_Info, + RE_Task_Info_Type => System_Task_Info, + RE_Task_Image_Type => System_Task_Info, + RE_Unspecified_Task_Info => System_Task_Info, + + RE_Library_Task_Level => System_Tasking, + + RE_Task_Procedure_Access => System_Tasking, + + RO_ST_Task_ID => System_Tasking, + + RE_Call_Modes => System_Tasking, + RE_Simple_Call => System_Tasking, + RE_Conditional_Call => System_Tasking, + RE_Asynchronous_Call => System_Tasking, + RE_Timed_Call => System_Tasking, + + RE_Task_List => System_Tasking, + + RE_Accept_Alternative => System_Tasking, + RE_Accept_List => System_Tasking, + RE_Accept_List_Access => System_Tasking, + RE_Max_Select => System_Tasking, + RE_Max_Task_Entry => System_Tasking, + RE_No_Rendezvous => System_Tasking, + RE_Null_Task_Entry => System_Tasking, + RE_Positive_Select_Index => System_Tasking, + RE_Select_Index => System_Tasking, + RE_Select_Modes => System_Tasking, + RE_Else_Mode => System_Tasking, + RE_Simple_Mode => System_Tasking, + RE_Terminate_Mode => System_Tasking, + RE_Delay_Mode => System_Tasking, + RE_Task_Entry_Index => System_Tasking, + RE_Self => System_Tasking, + + RE_Master_Id => System_Tasking, + RE_Unspecified_Priority => System_Tasking, + + RE_Activation_Chain => System_Tasking, + + RE_Abort_Defer => System_Soft_Links, + RE_Abort_Undefer => System_Soft_Links, + RE_Complete_Master => System_Soft_Links, + RE_Current_Master => System_Soft_Links, + RE_Enter_Master => System_Soft_Links, + RE_Get_Current_Excep => System_Soft_Links, + RE_Get_GNAT_Exception => System_Soft_Links, + RE_Update_Exception => System_Soft_Links, + + RE_Bits_1 => System_Unsigned_Types, + RE_Bits_2 => System_Unsigned_Types, + RE_Bits_4 => System_Unsigned_Types, + RE_Float_Unsigned => System_Unsigned_Types, + RE_Long_Long_Unsigned => System_Unsigned_Types, + RE_Packed_Byte => System_Unsigned_Types, + RE_Packed_Bytes1 => System_Unsigned_Types, + RE_Packed_Bytes2 => System_Unsigned_Types, + RE_Packed_Bytes4 => System_Unsigned_Types, + RE_Unsigned => System_Unsigned_Types, + + RE_Value_Boolean => System_Val_Bool, + + RE_Value_Character => System_Val_Char, + + RE_Value_Decimal => System_Val_Dec, + + RE_Value_Enumeration_8 => System_Val_Enum, + RE_Value_Enumeration_16 => System_Val_Enum, + RE_Value_Enumeration_32 => System_Val_Enum, + + RE_Value_Integer => System_Val_Int, + + RE_Value_Long_Long_Decimal => System_Val_LLD, + + RE_Value_Long_Long_Integer => System_Val_LLI, + + RE_Value_Long_Long_Unsigned => System_Val_LLU, + + RE_Value_Real => System_Val_Real, + + RE_Value_Unsigned => System_Val_Uns, + + RE_Value_Wide_Character => System_Val_WChar, + + RE_D => System_Vax_Float_Operations, + RE_F => System_Vax_Float_Operations, + RE_G => System_Vax_Float_Operations, + RE_Q => System_Vax_Float_Operations, + RE_S => System_Vax_Float_Operations, + RE_T => System_Vax_Float_Operations, + + RE_D_To_G => System_Vax_Float_Operations, + RE_F_To_G => System_Vax_Float_Operations, + RE_F_To_Q => System_Vax_Float_Operations, + RE_F_To_S => System_Vax_Float_Operations, + RE_G_To_D => System_Vax_Float_Operations, + RE_G_To_F => System_Vax_Float_Operations, + RE_G_To_Q => System_Vax_Float_Operations, + RE_G_To_T => System_Vax_Float_Operations, + RE_Q_To_F => System_Vax_Float_Operations, + RE_Q_To_G => System_Vax_Float_Operations, + RE_S_To_F => System_Vax_Float_Operations, + RE_T_To_D => System_Vax_Float_Operations, + RE_T_To_G => System_Vax_Float_Operations, + + RE_Abs_F => System_Vax_Float_Operations, + RE_Abs_G => System_Vax_Float_Operations, + RE_Add_F => System_Vax_Float_Operations, + RE_Add_G => System_Vax_Float_Operations, + RE_Div_F => System_Vax_Float_Operations, + RE_Div_G => System_Vax_Float_Operations, + RE_Mul_F => System_Vax_Float_Operations, + RE_Mul_G => System_Vax_Float_Operations, + RE_Neg_F => System_Vax_Float_Operations, + RE_Neg_G => System_Vax_Float_Operations, + RE_Sub_F => System_Vax_Float_Operations, + RE_Sub_G => System_Vax_Float_Operations, + + RE_Eq_F => System_Vax_Float_Operations, + RE_Eq_G => System_Vax_Float_Operations, + RE_Le_F => System_Vax_Float_Operations, + RE_Le_G => System_Vax_Float_Operations, + RE_Lt_F => System_Vax_Float_Operations, + RE_Lt_G => System_Vax_Float_Operations, + + RE_Version_String => System_Version_Control, + RE_Get_Version_String => System_Version_Control, + + RE_Register_VMS_Exception => System_VMS_Exception_Table, + + RE_String_To_Wide_String => System_WCh_StW, + + RE_Wide_String_To_String => System_WCh_WtS, + + RE_Wide_Width_Character => System_WWd_Char, + + RE_Wide_Width_Enumeration_8 => System_WWd_Enum, + RE_Wide_Width_Enumeration_16 => System_WWd_Enum, + RE_Wide_Width_Enumeration_32 => System_WWd_Enum, + + RE_Wide_Width_Wide_Character => System_WWd_Wchar, + + RE_Width_Boolean => System_Wid_Bool, + + RE_Width_Character => System_Wid_Char, + + RE_Width_Enumeration_8 => System_Wid_Enum, + RE_Width_Enumeration_16 => System_Wid_Enum, + RE_Width_Enumeration_32 => System_Wid_Enum, + + RE_Width_Long_Long_Integer => System_Wid_LLI, + + RE_Width_Long_Long_Unsigned => System_Wid_LLU, + + RE_Width_Wide_Character => System_Wid_WChar, + + RE_Protected_Entry_Body_Array => + System_Tasking_Protected_Objects_Entries, + RE_Protection_Entries => + System_Tasking_Protected_Objects_Entries, + RE_Initialize_Protection_Entries => + System_Tasking_Protected_Objects_Entries, + RE_Lock_Entries => + System_Tasking_Protected_Objects_Entries, + RE_Lock_Read_Only_Entries => + System_Tasking_Protected_Objects_Entries, + RE_Unlock_Entries => + System_Tasking_Protected_Objects_Entries, + RE_Communication_Block => + System_Tasking_Protected_Objects_Operations, + RE_Protected_Entry_Call => + System_Tasking_Protected_Objects_Operations, + RE_Service_Entries => + System_Tasking_Protected_Objects_Operations, + RE_Cancel_Protected_Entry_Call => + System_Tasking_Protected_Objects_Operations, + RE_Enqueued => + System_Tasking_Protected_Objects_Operations, + RE_Cancelled => + System_Tasking_Protected_Objects_Operations, + RE_Complete_Entry_Body => + System_Tasking_Protected_Objects_Operations, + RE_Exceptional_Complete_Entry_Body => + System_Tasking_Protected_Objects_Operations, + RE_Requeue_Protected_Entry => + System_Tasking_Protected_Objects_Operations, + RE_Requeue_Task_To_Protected_Entry => + System_Tasking_Protected_Objects_Operations, + RE_Protected_Count => + System_Tasking_Protected_Objects_Operations, + RE_Protected_Entry_Caller => + System_Tasking_Protected_Objects_Operations, + RE_Timed_Protected_Entry_Call => + System_Tasking_Protected_Objects_Operations, + + RE_Protection_Entry => + System_Tasking_Protected_Objects_Single_Entry, + RE_Initialize_Protection_Entry => + System_Tasking_Protected_Objects_Single_Entry, + RE_Lock_Entry => + System_Tasking_Protected_Objects_Single_Entry, + RE_Lock_Read_Only_Entry => + System_Tasking_Protected_Objects_Single_Entry, + RE_Unlock_Entry => + System_Tasking_Protected_Objects_Single_Entry, + RE_Protected_Single_Entry_Call => + System_Tasking_Protected_Objects_Single_Entry, + RE_Service_Entry => + System_Tasking_Protected_Objects_Single_Entry, + RE_Complete_Single_Entry_Body => + System_Tasking_Protected_Objects_Single_Entry, + RE_Exceptional_Complete_Single_Entry_Body => + System_Tasking_Protected_Objects_Single_Entry, + RE_Protected_Count_Entry => + System_Tasking_Protected_Objects_Single_Entry, + RE_Protected_Single_Entry_Caller => + System_Tasking_Protected_Objects_Single_Entry, + RE_Timed_Protected_Single_Entry_Call => + System_Tasking_Protected_Objects_Single_Entry, + + RE_Protected_Entry_Index => System_Tasking_Protected_Objects, + RE_Entry_Body => System_Tasking_Protected_Objects, + RE_Protection => System_Tasking_Protected_Objects, + RE_Initialize_Protection => System_Tasking_Protected_Objects, + RE_Finalize_Protection => System_Tasking_Protected_Objects, + RE_Lock => System_Tasking_Protected_Objects, + RE_Lock_Read_Only => System_Tasking_Protected_Objects, + RE_Unlock => System_Tasking_Protected_Objects, + + RE_Delay_Block => System_Tasking_Async_Delays, + RE_Timed_Out => System_Tasking_Async_Delays, + RE_Cancel_Async_Delay => System_Tasking_Async_Delays, + RE_Enqueue_Duration => System_Tasking_Async_Delays, + + RE_Enqueue_Calendar => + System_Tasking_Async_Delays_Enqueue_Calendar, + RE_Enqueue_RT => + System_Tasking_Async_Delays_Enqueue_RT, + + RE_Accept_Call => System_Tasking_Rendezvous, + RE_Accept_Trivial => System_Tasking_Rendezvous, + RE_Callable => System_Tasking_Rendezvous, + RE_Call_Simple => System_Tasking_Rendezvous, + RE_Cancel_Task_Entry_Call => System_Tasking_Rendezvous, + RE_Requeue_Task_Entry => System_Tasking_Rendezvous, + RE_Requeue_Protected_To_Task_Entry => System_Tasking_Rendezvous, + RE_Complete_Rendezvous => System_Tasking_Rendezvous, + RE_Task_Count => System_Tasking_Rendezvous, + RE_Exceptional_Complete_Rendezvous => System_Tasking_Rendezvous, + RE_Selective_Wait => System_Tasking_Rendezvous, + RE_Task_Entry_Call => System_Tasking_Rendezvous, + RE_Task_Entry_Caller => System_Tasking_Rendezvous, + RE_Timed_Task_Entry_Call => System_Tasking_Rendezvous, + RE_Timed_Selective_Wait => System_Tasking_Rendezvous, + + RE_Activate_Restricted_Tasks => System_Tasking_Restricted_Stages, + RE_Complete_Restricted_Activation => System_Tasking_Restricted_Stages, + RE_Create_Restricted_Task => System_Tasking_Restricted_Stages, + RE_Complete_Restricted_Task => System_Tasking_Restricted_Stages, + RE_Restricted_Terminated => System_Tasking_Restricted_Stages, + + RE_Abort_Tasks => System_Tasking_Stages, + RE_Activate_Tasks => System_Tasking_Stages, + RE_Complete_Activation => System_Tasking_Stages, + RE_Create_Task => System_Tasking_Stages, + RE_Complete_Task => System_Tasking_Stages, + RE_Free_Task => System_Tasking_Stages, + RE_Expunge_Unactivated_Tasks => System_Tasking_Stages, + RE_Terminated => System_Tasking_Stages); + + ----------------- + -- Subprograms -- + ----------------- + + procedure Initialize; + -- Procedure to initialize data structures used by RTE. Called at the + -- start of processing a new main source file. Must be called after + -- Initialize_Snames (since names it enters into name table must come + -- after names entered by Snames). + + function RTE (E : RE_Id) return Entity_Id; + -- Given the entity defined in the above tables, as identified by the + -- corresponding value in the RE_Id enumeration type, returns the Id + -- of the corresponding entity, first loading in (parsing, analyzing and + -- expanding) its spec if the unit has not already been loaded. If the + -- unit cannot be found, or if it does not contain the specified entity, + -- then an appropriate error message is output ("run-time configuration + -- error") and an Unrecoverable_Error exception is raised. + + function Is_RTE (Ent : Entity_Id; E : RE_Id) return Boolean; + -- This function determines if the given entity corresponds to the entity + -- referenced by RE_Id. It is similar in effect to (Ent = RTE (E)) except + -- that the latter would unconditionally load the unit containing E. For + -- this call, if the unit is not loaded, then a result of False is returned + -- immediately, since obviously Ent cannot be the entity in question if the + -- corresponding unit has not been loaded. + + procedure Text_IO_Kludge (Nam : Node_Id); + -- In Ada 83, and hence for compatibility in Ada 9X, package Text_IO has + -- generic subpackages (e.g. Integer_IO). They really should be child + -- packages, and in GNAT, they *are* child packages. To maintain the + -- required compatibility, this routine is called for package renamings + -- and generic instantiations, with the simple name of the referenced + -- package. If Text_IO has been with'ed and if the simple name of Nam + -- matches one of the subpackages of Text_IO, then this subpackage is + -- with'ed automatically. The important result of this approach is that + -- Text_IO does not drag in all the code for the subpackages unless they + -- are used. Our test is a little crude, and could drag in stuff when it + -- is not necessary, but that doesn't matter. Wide_Text_IO is handled in + -- a similar manner. + + function Is_Text_IO_Kludge_Unit (Nam : Node_Id) return Boolean; + -- Returns True if the given Nam is an Expanded Name, whose Prefix is + -- Ada, and whose selector is either Text_IO.xxx or Wide_Text_IO.xxx + -- where xxx is one of the subpackages of Text_IO that is specially + -- handled as described above for Text_IO_Kludge. + +end Rtsfind; |