------------------------------------------------------------------------------ -- -- -- GNAT COMPILER COMPONENTS -- -- -- -- P A R . S Y N C -- -- -- -- B o d y -- -- -- -- Copyright (C) 1992-2022, Free Software Foundation, Inc. -- -- -- -- GNAT is free software; you can redistribute it and/or modify it under -- -- terms of the GNU General Public License as published by the Free Soft- -- -- ware Foundation; either version 3, or (at your option) any later ver- -- -- sion. GNAT is distributed in the hope that it will be useful, but WITH- -- -- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -- -- or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License -- -- for more details. You should have received a copy of the GNU General -- -- Public License distributed with GNAT; see file COPYING3. If not, go to -- -- http://www.gnu.org/licenses for a complete copy of the license. -- -- -- -- GNAT was originally developed by the GNAT team at New York University. -- -- Extensive contributions were provided by Ada Core Technologies Inc. -- -- -- ------------------------------------------------------------------------------ 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 | 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 addition, -- 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 in Tok_Comma | Tok_Right_Paren | 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 resynchronizing 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_Malformed_Aspect -- ---------------------------------- procedure Resync_Past_Malformed_Aspect is begin Resync_Init; loop -- A comma may separate two aspect specifications, but it may also -- delimit multiple arguments of a single aspect. if Token = Tok_Comma then declare Scan_State : Saved_Scan_State; begin Save_Scan_State (Scan_State); Scan; -- past comma -- The identifier following the comma is a valid aspect, the -- current malformed aspect has been successfully skipped. if Token = Tok_Identifier and then Is_Aspect_Id (Token_Name) then Restore_Scan_State (Scan_State); exit; -- The comma is delimiting multiple arguments of an aspect else Restore_Scan_State (Scan_State); end if; end; -- An IS signals the last aspect specification when the related -- context is a body. elsif Token = Tok_Is then exit; -- A semicolon signals the last aspect specification elsif Token = Tok_Semicolon then exit; -- In the case of a mistyped semicolon, any token which follows a -- semicolon signals the last aspect specification. elsif Token in Token_Class_After_SM then exit; end if; -- Keep on resyncing Scan; end loop; -- Fall out of loop with resynchronization complete Resync_Resume; end Resync_Past_Malformed_Aspect; --------------------------- -- 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 resynchronization 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 in Tok_Then | Tok_Loop then exit; -- Otherwise keep going else Scan; end if; end loop; -- Fall out of loop with resynchronization 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_Semicolon_List -- --------------------------- procedure Resync_Semicolon_List is Paren_Count : Int; begin Resync_Init; Paren_Count := 0; loop if Token in Tok_EOF | Tok_Semicolon | Tok_Is | 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; ------------------------- -- Resync_To_Semicolon -- ------------------------- procedure Resync_To_Semicolon is begin Resync_Init; loop -- Done if we are at a semicolon if Token = Tok_Semicolon then 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 resynchronization complete Resync_Resume; end Resync_To_Semicolon; -------------------- -- Resync_To_When -- -------------------- procedure Resync_To_When is begin Resync_Init; loop -- Done if at semicolon, WHEN or IS if Token in Tok_Semicolon | Tok_When | Tok_Is then exit; -- Otherwise keep going else Scan; end if; end loop; -- Fall out of loop with resynchronization complete Resync_Resume; end Resync_To_When; end Sync;