| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031 | {    $Id$    This file is part of the Free Component Library    Pascal source lexical scanner    Copyright (c) 2003 by      Areca Systems GmbH / Sebastian Guenther, [email protected]    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************}unit PScanner;interfaceuses SysUtils, Classes;resourcestring  SErrInvalidCharacter = 'Invalid character ''%s''';  SErrOpenString = 'String exceeds end of line';  SErrIncludeFileNotFound = 'Could not find include file ''%s''';  SErrIfXXXNestingLimitReached = 'Nesting of $IFxxx too deep';  SErrInvalidPPElse = '$ELSE without matching $IFxxx';  SErrInvalidPPEndif = '$ENDIF without matching $IFxxx';type  TToken = (    tkEOF,    tkWhitespace,    tkComment,    tkIdentifier,    tkString,    tkNumber,    tkChar,    // Simple (one-character) tokens    tkBraceOpen,        // '('    tkBraceClose,       // ')'    tkMul,              // '*'    tkPlus,             // '+'    tkComma,            // ','    tkMinus,            // '-'    tkDot,              // '.'    tkDivision,         // '/'    tkColon,            // ':'    tkSemicolon,        // ';'    tkLessThan,         // '<'    tkEqual,            // '='    tkGreaterThan,      // '>'    tkAt,               // '@'    tkSquaredBraceOpen, // '['    tkSquaredBraceClose,// ']'    tkCaret,            // '^'    // Two-character tokens    tkDotDot,           // '..'    tkAssign,           // ':='    tkNotEqual,         // '<>'    // Reserved words    tkabsolute,    tkand,    tkarray,    tkas,    tkasm,    tkbegin,    tkcase,    tkclass,    tkconst,    tkconstructor,    tkdestructor,    tkdiv,    tkdo,    tkdownto,    tkelse,    tkend,    tkexcept,    tkexports,    tkfalse,    tkfinalization,    tkfinally,    tkfor,    tkfunction,    tkgoto,    tkif,    tkimplementation,    tkin,    tkinherited,    tkinitialization,    tkinline,    tkinterface,    tkis,    tklabel,    tklibrary,    tkmod,    tknil,    tknot,    tkobject,    tkof,    tkon,    tkoperator,    tkor,    tkpacked,    tkprocedure,    tkprogram,    tkproperty,    tkraise,    tkrecord,    tkrepeat,    tkResourceString,    tkself,    tkset,    tkshl,    tkshr,//    tkstring,    tkthen,    tkto,    tktrue,    tktry,    tktype,    tkunit,    tkuntil,    tkuses,    tkvar,    tkwhile,    tkwith,    tkxor);  TLineReader = class  public    function IsEOF: Boolean; virtual; abstract;    function ReadLine: String; virtual; abstract;  end;  TFileLineReader = class(TLineReader)  private    FTextFile: Text;    FileOpened: Boolean;  public    constructor Create(const AFilename: String);    destructor Destroy; override;    function IsEOF: Boolean; override;    function ReadLine: String; override;  end;  TFileResolver = class  private    FIncludePaths: TStringList;  public    constructor Create;    destructor Destroy; override;    procedure AddIncludePath(const APath: String);    function FindSourceFile(const AName: String): TLineReader;    function FindIncludeFile(const AName: String): TLineReader;  end;  EScannerError       = class(Exception);  EFileNotFoundError  = class(Exception);  TPascalScannerPPSkipMode = (ppSkipNone, ppSkipIfBranch, ppSkipElseBranch,    ppSkipAll);  TPascalScanner = class  private    FFileResolver: TFileResolver;    FCurSourceFile: TLineReader;    FCurFilename: String;    FCurRow: Integer;    FCurToken: TToken;    FCurTokenString: String;    FCurLine: String;    FDefines: TStrings;    TokenStr: PChar;    FIncludeStack: TList;    // Preprocessor $IFxxx skipping data    PPSkipMode: TPascalScannerPPSkipMode;    PPIsSkipping: Boolean;    PPSkipStackIndex: Integer;    PPSkipModeStack: array[0..255] of TPascalScannerPPSkipMode;    PPIsSkippingStack: array[0..255] of Boolean;    function GetCurColumn: Integer;  protected    procedure Error(const Msg: String);    procedure Error(const Msg: String; Args: array of Const);    function DoFetchToken: TToken;  public    constructor Create(AFileResolver: TFileResolver);    destructor Destroy; override;    procedure OpenFile(const AFilename: String);    function FetchToken: TToken;    property FileResolver: TFileResolver read FFileResolver;    property CurSourceFile: TLineReader read FCurSourceFile;    property CurFilename: String read FCurFilename;    property CurLine: String read FCurLine;    property CurRow: Integer read FCurRow;    property CurColumn: Integer read GetCurColumn;    property CurToken: TToken read FCurToken;    property CurTokenString: String read FCurTokenString;    property Defines: TStrings read FDefines;  end;const  TokenInfos: array[TToken] of String = (    'EOF',    'Whitespace',    'Comment',    'Identifier',    'String',    'Number',    'Character',    '(',    ')',    '*',    '+',    ',',    '-',    '.',    '/',    ':',    ';',    '<',    '=',    '>',    '@',    '[',    ']',    '^',    '..',    ':=',    '<>',    // Reserved words    'absolute',    'and',    'array',    'as',    'asm',    'begin',    'case',    'class',    'const',    'constructor',    'destructor',    'div',    'do',    'downto',    'else',    'end',    'except',    'exports',    'false',    'finalization',    'finally',    'for',    'function',    'goto',    'if',    'implementation',    'in',    'inherited',    'initialization',    'inline',    'interface',    'is',    'label',    'library',    'mod',    'nil',    'not',    'object',    'of',    'on',    'operator',    'or',    'packed',    'procedure',    'program',    'property',    'raise',    'record',    'repeat',    'resourcestring',    'self',    'set',    'shl',    'shr',//    'string',    'then',    'to',    'true',    'try',    'type',    'unit',    'until',    'uses',    'var',    'while',    'with',    'xor'  );implementationtype  TIncludeStackItem = class    SourceFile: TLineReader;    Filename: String;    Token: TToken;    TokenString: String;    Line: String;    Row: Integer;    TokenStr: PChar;  end;constructor TFileLineReader.Create(const AFilename: String);begin  inherited Create;  Assign(FTextFile, AFilename);  Reset(FTextFile);  FileOpened := True;end;destructor TFileLineReader.Destroy;begin  if FileOpened then    Close(FTextFile);  inherited Destroy;end;function TFileLineReader.IsEOF: Boolean;begin  Result := EOF(FTextFile);end;function TFileLineReader.ReadLine: String;begin  ReadLn(FTextFile, Result);end;constructor TFileResolver.Create;begin  inherited Create;  FIncludePaths := TStringList.Create;end;destructor TFileResolver.Destroy;begin  FIncludePaths.Free;  inherited Destroy;end;procedure TFileResolver.AddIncludePath(const APath: String);begin  FIncludePaths.Add(IncludeTrailingPathDelimiter(ExpandFileName(APath)));end;function TFileResolver.FindSourceFile(const AName: String): TLineReader;begin  if not FileExists(AName) then    Raise EFileNotFoundError.create(Aname)  else    try      Result := TFileLineReader.Create(AName);    except      Result := nil;    end;end;function TFileResolver.FindIncludeFile(const AName: String): TLineReader;var  i: Integer;  FN : String;begin  Result := nil;  If FileExists(AName) then    Result := TFileLineReader.Create(AName)  else    begin    I:=0;    While (Result=Nil) and (I<FIncludePaths.Count) do      begin      Try        FN:=FIncludePaths[i]+AName;        If FileExists(FN) then          Result := TFileLineReader.Create(FN);      except        Result:=Nil;      end;      Inc(I);      end;    end;end;constructor TPascalScanner.Create(AFileResolver: TFileResolver);begin  inherited Create;  FFileResolver := AFileResolver;  FIncludeStack := TList.Create;  FDefines := TStringList.Create;end;destructor TPascalScanner.Destroy;begin  FDefines.Free;  // Dont' free the first element, because it is CurSourceFile  while FIncludeStack.Count > 1 do  begin    TFileResolver(FIncludeStack[1]).Free;    FIncludeStack.Delete(1);  end;  FIncludeStack.Free;  CurSourceFile.Free;  inherited Destroy;end;procedure TPascalScanner.OpenFile(const AFilename: String);begin  FCurSourceFile := FileResolver.FindSourceFile(AFilename);  FCurFilename := AFilename;end;function TPascalScanner.FetchToken: TToken;var  IncludeStackItem: TIncludeStackItem;begin  while True do  begin    Result := DoFetchToken;    if FCurToken = tkEOF then      if FIncludeStack.Count > 0 then      begin        CurSourceFile.Free;        IncludeStackItem :=          TIncludeStackItem(FIncludeStack[FIncludeStack.Count - 1]);        FIncludeStack.Delete(FIncludeStack.Count - 1);        FCurSourceFile := IncludeStackItem.SourceFile;        FCurFilename := IncludeStackItem.Filename;        FCurToken := IncludeStackItem.Token;        FCurTokenString := IncludeStackItem.TokenString;        FCurLine := IncludeStackItem.Line;        FCurRow := IncludeStackItem.Row;        TokenStr := IncludeStackItem.TokenStr;        IncludeStackItem.Free;        Result := FCurToken;      end else        break    else      if not PPIsSkipping then        break;  end;end;procedure TPascalScanner.Error(const Msg: String);begin  raise EScannerError.Create(Msg);end;procedure TPascalScanner.Error(const Msg: String; Args: array of Const);begin  raise EScannerError.CreateFmt(Msg, Args);end;function TPascalScanner.DoFetchToken: TToken;  function FetchLine: Boolean;  begin    if CurSourceFile.IsEOF then    begin      FCurLine := '';      TokenStr := nil;      Result := False;    end else    begin      FCurLine := CurSourceFile.ReadLine;      TokenStr := PChar(CurLine);      Result := True;      Inc(FCurRow);    end;  end;var  TokenStart, CurPos: PChar;  i: TToken;  OldLength, SectionLength, NestingLevel, Index: Integer;  Directive, Param: String;  IncludeStackItem: TIncludeStackItem;begin  if TokenStr = nil then    if not FetchLine then    begin      Result := tkEOF;      FCurToken := Result;      exit;    end;  FCurTokenString := '';  case TokenStr[0] of    #0:         // Empty line      begin        FetchLine;        Result := tkWhitespace;      end;    #9, ' ':      begin        Result := tkWhitespace;        repeat          Inc(TokenStr);          if TokenStr[0] = #0 then            if not FetchLine then            begin              FCurToken := Result;              exit;            end;        until not (TokenStr[0] in [#9, ' ']);      end;    '#':      begin        TokenStart := TokenStr;        Inc(TokenStr);        if TokenStr[0] = '$' then        begin          Inc(TokenStr);          repeat            Inc(TokenStr);          until not (TokenStr[0] in ['0'..'9', 'A'..'F', 'a'..'F']);        end else          repeat            Inc(TokenStr);          until not (TokenStr[0] in ['0'..'9']);        SectionLength := TokenStr - TokenStart;        SetLength(FCurTokenString, SectionLength);        if SectionLength > 0 then          Move(TokenStart^, FCurTokenString[1], SectionLength);        Result := tkChar;      end;    '$':      begin        TokenStart := TokenStr;        repeat          Inc(TokenStr);        until not (TokenStr[0] in ['0'..'9', 'A'..'F', 'a'..'F']);        SectionLength := TokenStr - TokenStart;        SetLength(FCurTokenString, SectionLength);        if SectionLength > 0 then          Move(TokenStart^, FCurTokenString[1], SectionLength);        Result := tkNumber;      end;    '%':      begin        TokenStart := TokenStr;        repeat          Inc(TokenStr);        until not (TokenStr[0] in ['0','1']);        SectionLength := TokenStr - TokenStart;        SetLength(FCurTokenString, SectionLength);        if SectionLength > 0 then          Move(TokenStart^, FCurTokenString[1], SectionLength);        Result := tkNumber;      end;    '''':      begin        Inc(TokenStr);        TokenStart := TokenStr;        OldLength := 0;        FCurTokenString := '';        while True do        begin          if TokenStr[0] = '''' then            if TokenStr[1] = '''' then            begin              SectionLength := TokenStr - TokenStart + 1;              SetLength(FCurTokenString, OldLength + SectionLength);              if SectionLength > 0 then                Move(TokenStart^, FCurTokenString[OldLength + 1], SectionLength);              Inc(OldLength, SectionLength);              Inc(TokenStr);              TokenStart := TokenStr+1;            end else              break;          if TokenStr[0] = #0 then            Error(SErrOpenString);          Inc(TokenStr);        end;        SectionLength := TokenStr - TokenStart;        SetLength(FCurTokenString, OldLength + SectionLength);        if SectionLength > 0 then          Move(TokenStart^, FCurTokenString[OldLength + 1], SectionLength);        Inc(TokenStr);        Result := tkString;      end;    '(':      begin        Inc(TokenStr);        if TokenStr[0] = '*' then        begin          // Old-style multi-line comment          Inc(TokenStr);          while (TokenStr[0] <> '*') or (TokenStr[1] <> ')') do          begin            if TokenStr[0] = #0 then            begin              if not FetchLine then              begin                Result := tkEOF;                FCurToken := Result;                exit;              end;            end else              Inc(TokenStr);          end;          Inc(TokenStr, 2);          Result := tkComment;        end else          Result := tkBraceOpen;      end;    ')':      begin        Inc(TokenStr);        Result := tkBraceClose;      end;    '*':      begin        Inc(TokenStr);        Result := tkMul;      end;    '+':      begin        Inc(TokenStr);        Result := tkPlus;      end;    ',':      begin        Inc(TokenStr);        Result := tkComma;      end;    '-':      begin        Inc(TokenStr);        Result := tkMinus;      end;    '.':      begin        Inc(TokenStr);        if TokenStr[0] = '.' then        begin          Inc(TokenStr);          Result := tkDotDot;        end else          Result := tkDot;      end;    '/':      begin        Inc(TokenStr);        if TokenStr[0] = '/' then       // Single-line comment        begin          Inc(TokenStr);          TokenStart := TokenStr;          FCurTokenString := '';          while TokenStr[0] <> #0 do            Inc(TokenStr);          SectionLength := TokenStr - TokenStart;          SetLength(FCurTokenString, SectionLength);          if SectionLength > 0 then            Move(TokenStart^, FCurTokenString[1], SectionLength);          Result := tkComment;          //WriteLn('Einzeiliger Kommentar: "', CurTokenString, '"');        end else          Result := tkDivision;      end;    '0'..'9':      begin        TokenStart := TokenStr;        while True do        begin          Inc(TokenStr);          case TokenStr[0] of            '.':              begin                if TokenStr[1] in ['0'..'9', 'e', 'E'] then                begin                  Inc(TokenStr);                  repeat                    Inc(TokenStr);                  until not (TokenStr[0] in ['0'..'9', 'e', 'E']);                end;                break;              end;            '0'..'9': ;            'e', 'E':              begin                Inc(TokenStr);                if TokenStr[0] = '-'  then                  Inc(TokenStr);                while TokenStr[0] in ['0'..'9'] do                  Inc(TokenStr);                break;              end;            else              break;          end;        end;        SectionLength := TokenStr - TokenStart;        SetLength(FCurTokenString, SectionLength);        if SectionLength > 0 then          Move(TokenStart^, FCurTokenString[1], SectionLength);        Result := tkNumber;      end;    ':':      begin        Inc(TokenStr);        if TokenStr[0] = '=' then        begin          Inc(TokenStr);          Result := tkAssign;        end else          Result := tkColon;      end;    ';':      begin        Inc(TokenStr);        Result := tkSemicolon;      end;    '<':      begin        Inc(TokenStr);        if TokenStr[0] = '>' then        begin          Inc(TokenStr);          Result := tkNotEqual;        end else          Result := tkLessThan;      end;    '=':      begin        Inc(TokenStr);        Result := tkEqual;      end;    '>':      begin        Inc(TokenStr);        Result := tkGreaterThan;      end;    '@':      begin        Inc(TokenStr);        Result := tkAt;      end;    '[':      begin        Inc(TokenStr);        Result := tkSquaredBraceOpen;      end;    ']':      begin        Inc(TokenStr);        Result := tkSquaredBraceClose;      end;    '^':      begin        Inc(TokenStr);        Result := tkCaret;      end;    '{':        // Multi-line comment      begin        Inc(TokenStr);        TokenStart := TokenStr;        FCurTokenString := '';        OldLength := 0;        NestingLevel := 0;        while (TokenStr[0] <> '}') or (NestingLevel > 0) do        begin          if TokenStr[0] = #0 then          begin            SectionLength := TokenStr - TokenStart + 1;            SetLength(FCurTokenString, OldLength + SectionLength);            if SectionLength > 1 then              Move(TokenStart^, FCurTokenString[OldLength + 1],                SectionLength - 1);            Inc(OldLength, SectionLength);            FCurTokenString[OldLength] := #10;            if not FetchLine then            begin              Result := tkEOF;              FCurToken := Result;              exit;            end;            TokenStart := TokenStr;          end else          begin            if TokenStr[0] = '{' then              Inc(NestingLevel)            else if TokenStr[0] = '}' then              Dec(NestingLevel);            Inc(TokenStr);          end;        end;        SectionLength := TokenStr - TokenStart;        SetLength(FCurTokenString, OldLength + SectionLength);        if SectionLength > 0 then          Move(TokenStart^, FCurTokenString[OldLength + 1], SectionLength);        Inc(TokenStr);        Result := tkComment;        //WriteLn('Kommentar: "', CurTokenString, '"');        if (Length(CurTokenString) > 0) and (CurTokenString[1] = '$') then        begin          TokenStart := @CurTokenString[2];          CurPos := TokenStart;          while (CurPos[0] <> ' ') and (CurPos[0] <> #0) do            Inc(CurPos);          SectionLength := CurPos - TokenStart;          SetLength(Directive, SectionLength);          if SectionLength > 0 then          begin            Move(TokenStart^, Directive[1], SectionLength);            Directive := UpperCase(Directive);            if CurPos[0] <> #0 then            begin              TokenStart := CurPos + 1;              CurPos := TokenStart;              while CurPos[0] <> #0 do                Inc(CurPos);              SectionLength := CurPos - TokenStart;              SetLength(Param, SectionLength);              if SectionLength > 0 then                Move(TokenStart^, Param[1], SectionLength);            end else              Param := '';            // WriteLn('Direktive: "', Directive, '", Param: "', Param, '"');            if (Directive = 'I') or (Directive = 'INCLUDE') then            begin              if not PPIsSkipping then              begin                IncludeStackItem := TIncludeStackItem.Create;                IncludeStackItem.SourceFile := CurSourceFile;                IncludeStackItem.Filename := CurFilename;                IncludeStackItem.Token := CurToken;                IncludeStackItem.TokenString := CurTokenString;                IncludeStackItem.Line := CurLine;                IncludeStackItem.Row := CurRow;                IncludeStackItem.TokenStr := TokenStr;                FIncludeStack.Add(IncludeStackItem);                FCurSourceFile := FileResolver.FindIncludeFile(Param);                if not Assigned(CurSourceFile) then                  Error(SErrIncludeFileNotFound, [Param]);                FCurFilename := Param;                FCurRow := 0;              end;            end else if Directive = 'DEFINE' then            begin              if not PPIsSkipping then              begin                Param := UpperCase(Param);                if Defines.IndexOf(Param) < 0 then                  Defines.Add(Param);              end;            end else if Directive = 'UNDEF' then            begin              if not PPIsSkipping then              begin                Param := UpperCase(Param);                Index := Defines.IndexOf(Param);                if Index >= 0 then                  Defines.Delete(Index);              end;            end else if Directive = 'IFDEF' then            begin              if PPSkipStackIndex = High(PPSkipModeStack) then                Error(SErrIfXXXNestingLimitReached);              PPSkipModeStack[PPSkipStackIndex] := PPSkipMode;              PPIsSkippingStack[PPSkipStackIndex] := PPIsSkipping;              Inc(PPSkipStackIndex);              if PPIsSkipping then              begin                PPSkipMode := ppSkipAll;                PPIsSkipping := True;              end else              begin                Param := UpperCase(Param);                Index := Defines.IndexOf(Param);                if Index < 0 then                begin                  PPSkipMode := ppSkipIfBranch;                  PPIsSkipping := True;                end else                  PPSkipMode := ppSkipElseBranch;              end;            end else if Directive = 'IFNDEF' then            begin              if PPSkipStackIndex = High(PPSkipModeStack) then                Error(SErrIfXXXNestingLimitReached);              PPSkipModeStack[PPSkipStackIndex] := PPSkipMode;              PPIsSkippingStack[PPSkipStackIndex] := PPIsSkipping;              Inc(PPSkipStackIndex);              if PPIsSkipping then              begin                PPSkipMode := ppSkipAll;                PPIsSkipping := True;              end else              begin                Param := UpperCase(Param);                Index := Defines.IndexOf(Param);                if Index >= 0 then                begin                  PPSkipMode := ppSkipIfBranch;                  PPIsSkipping := True;                end else                  PPSkipMode := ppSkipElseBranch;              end;            end else if Directive = 'IFOPT' then            begin              if PPSkipStackIndex = High(PPSkipModeStack) then                Error(SErrIfXXXNestingLimitReached);              PPSkipModeStack[PPSkipStackIndex] := PPSkipMode;              PPIsSkippingStack[PPSkipStackIndex] := PPIsSkipping;              Inc(PPSkipStackIndex);              if PPIsSkipping then              begin                PPSkipMode := ppSkipAll;                PPIsSkipping := True;              end else              begin                { !!!: Currently, options are not supported, so they are just                  assumed as not being set. }                PPSkipMode := ppSkipIfBranch;                PPIsSkipping := True;              end;            end else if Directive = 'IF' then            begin              if PPSkipStackIndex = High(PPSkipModeStack) then                Error(SErrIfXXXNestingLimitReached);              PPSkipModeStack[PPSkipStackIndex] := PPSkipMode;              PPIsSkippingStack[PPSkipStackIndex] := PPIsSkipping;              Inc(PPSkipStackIndex);              if PPIsSkipping then              begin                PPSkipMode := ppSkipAll;                PPIsSkipping := True;              end else              begin                { !!!: Currently, expressions are not supported, so they are                  just assumed as evaluating to false. }                PPSkipMode := ppSkipIfBranch;                PPIsSkipping := True;              end;            end else if Directive = 'ELSE' then            begin              if PPSkipStackIndex = 0 then                Error(SErrInvalidPPElse);              if PPSkipMode = ppSkipIfBranch then                PPIsSkipping := False              else if PPSkipMode = ppSkipElseBranch then                PPIsSkipping := True;            end else if Directive = 'ENDIF' then            begin              if PPSkipStackIndex = 0 then                Error(SErrInvalidPPEndif);              Dec(PPSkipStackIndex);              PPSkipMode := PPSkipModeStack[PPSkipStackIndex];              PPIsSkipping := PPIsSkippingStack[PPSkipStackIndex];            end;          end else            Directive := '';        end;      end;    'A'..'Z', 'a'..'z', '_':      begin        TokenStart := TokenStr;        repeat          Inc(TokenStr);        until not (TokenStr[0] in ['A'..'Z', 'a'..'z', '0'..'9', '_']);        SectionLength := TokenStr - TokenStart;        SetLength(FCurTokenString, SectionLength);        if SectionLength > 0 then          Move(TokenStart^, FCurTokenString[1], SectionLength);        // Check if this is a keyword or identifier        // !!!: Optimize this!        for i := tkAbsolute to tkXOR do          if CompareText(CurTokenString, TokenInfos[i]) = 0 then          begin            Result := i;            FCurToken := Result;            exit;          end;        Result := tkIdentifier;      end;  else    Error(SErrInvalidCharacter, [TokenStr[0]]);  end;  FCurToken := Result;end;function TPascalScanner.GetCurColumn: Integer;begin  Result := TokenStr - PChar(CurLine);end;end.{  $Log$  Revision 1.11  2005-02-14 17:13:17  peter    * truncate log}
 |