{ *************************************************************************** * * * This source is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * This code 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. See the GNU * * General Public License for more details. * * * * A copy of the GNU General Public License is available on the World * * Wide Web at . You can also * * obtain it by writing to the Free Software Foundation, * * Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1335, USA. * * * *************************************************************************** Author: Mattias Gaertner Abstract: TLinkScanner scans a source file, reacts to compiler directives, replaces macros and reads include files. It builds one source and a link list. The resulting source is called the cleaned source. A link points from a position of the cleaned source to its position in the real source. The link list makes it possible to find scanned code in the source files. } unit LinkScanner; {$mode objfpc}{$H+} {$I codetools.inc} { $DEFINE ShowIgnoreErrorAfter} // debugging { $DEFINE ShowUpdateCleanedSrc} { $DEFINE VerboseIncludeSearch} { $DEFINE VerboseUpdateNeeded} interface uses {$IFDEF MEM_CHECK} MemCheck, {$ENDIF} // RTL + FCL Classes, SysUtils, math, Laz_AVL_Tree, // CodeTools CodeToolsStrConsts, CodeToolMemManager, FileProcs, ExprEval, SourceLog, KeywordFuncLists, BasicCodeTools, // LazUtils LazFileUtils, LazUtilities, LazDbgLog; var FPCSystemUnitName: String = 'system'; DelphiSystemUnitName: String = 'System'; const PascalCompilerDefine = ExternalMacroStart+'Compiler'; MacroUseHeapTrc = ExternalMacroStart+'UseHeapTrcUnit'; MacroUseLineInfo = ExternalMacroStart+'UseLineInfo'; MacroUselnfodwrf = ExternalMacroStart+'Uselnfodwrf'; MacroUseValgrind = ExternalMacroStart+'UseValgrind'; MacroUseProfiler = ExternalMacroStart+'UseProfiler'; MacroUseFPCylix = ExternalMacroStart+'UseFPCylix'; MacroUseSysThrds = ExternalMacroStart+'UseSysThrds'; MacroControllerUnit = ExternalMacroStart+'ControllerUnit'; type TLinkScanner = class; //---------------------------------------------------------------------------- TOnGetSource = function(Sender: TObject; Code: Pointer): TSourceLog of object; TOnLoadSource = function(Sender: TObject; const AFilename: string; OnlyIfExists: boolean): pointer of object; TOnGetSourceStatus = procedure(Sender: TObject; Code: Pointer; var ReadOnly: boolean) of object; TOnDeleteSource = procedure(Sender: TObject; Code: Pointer; Pos, Len: integer) of object; TOnGetFileName = function(Sender: TObject; Code: Pointer): string of object; TOnCheckFileOnDisk = function(Code: Pointer): boolean of object; TOnGetInitValues = function(Scanner: TLinkScanner; Code: Pointer; out ChangeStep: integer): TExpressionEvaluator of object; TOnIncludeCode = procedure(ParentCode, IncludeCode: Pointer) of object; TOnSetWriteLock = procedure(Lock: boolean) of object; TLSOnGetGlobalChangeSteps = procedure(out SourcesChangeStep, FilesChangeStep: int64; out InitValuesChangeStep: integer) of object; { TSourceLink is used to map between the codefiles and the cleaned source } TSourceLinkKind = ( slkCode, slkMissingIncludeFile, slkSkipStart, // start of skipped code due to IFDEFs {#3 slkSkipEnd, // end of skipped code due to IFDEFs #3} slkCompilerString // e.g. {$I %FPCVERSION%} ); TSourceLinkKinds = set of TSourceLinkKind; PSourceLink = ^TSourceLink; TSourceLink = record CleanedPos: integer; SrcPos: integer; Code: Pointer; Kind: TSourceLinkKind; Next: PSourceLink; end; TSourceLinkMacro = record Name: PChar; Code: Pointer; Src: string; SrcFilename: string; StartPos, EndPos: integer; end; PSourceLinkMacro = ^TSourceLinkMacro; { TSourceChangeStep is used to save the ChangeStep of every used file A ChangeStep is switching to or from an include file } PSourceChangeStep = ^TSourceChangeStep; TSourceChangeStep = record Code: Pointer; ChangeStep: integer; Next: PSourceChangeStep; end; TLinkScannerRange = ( lsrNone, // undefined lsrInit, // init, but do not scan any code lsrSourceType, // read till source type (e.g. keyword program or unit) lsrSourceName, // read till source name lsrInterfaceStart, // read till keyword interface lsrMainUsesSectionStart, // uses section of interface/program lsrMainUsesSectionEnd, // uses section of interface/program lsrImplementationStart, // scan at least to interface end (e.g. till implementation keyword) lsrImplementationUsesSectionStart, // uses section of implementation lsrImplementationUsesSectionEnd, // uses section of implementation lsrInitializationStart, lsrFinalizationStart, lsrEnd // scan till 'end.' ); TCommentStyle = ( CommentNone, CommentCurly, // {} CommentRound, // (* *) CommentLine // // ); TCompilerMode = ( cmFPC, cmDELPHI, cmDELPHIUNICODE, cmGPC, cmTP, cmOBJFPC, cmMacPas, cmISO, cmExtPas ); { TCompilerModeSwitch - see fpc/compiler/globtype.pas tmodeswitch } TCompilerModeSwitch = ( //cms_FPC,cms_ObjFPC,cms_Delphi,cms_TP7,cms_Mac,cms_ISO,cms_ExtPas, cmsAdd_pointer, { ? } cmsClass, { delphi class model } cmsObjpas, { load objpas unit } cmsResult, { result in functions } cmsString_pchar, { pchar 2 string conversion } cmsCvar_support, { cvar variable directive } cmsNested_comment, { nested comments } cmsTp_procvar, { tp style procvars (no @ needed) } cmsMac_procvar, { macpas style procvars } cmsRepeat_forward, { repeating forward declarations is needed } cmsPointer_2_procedure,{ allows the assignement of pointers to procedure variables } cmsAutoderef, { does auto dereferencing of struct. vars, e.g. a.b -> a^.b } cmsInitfinal, { initialization/finalization for units } cmsDefault_ansistring, { ansistring turned on by default } cmsOut, { support the calling convention OUT } cmsDefault_para, { support default parameters } cmsHintdirective, { support hint directives } cmsDuplicate_names, { allow locals/paras to have duplicate names of globals } cmsProperty, { allow properties } cmsDefault_inline, { allow inline proc directive } cmsExcept, { allow exception-related keywords } cmsObjectiveC1, { support interfacing with Objective-C (1.0) } cmsObjectiveC2, { support interfacing with Objective-C (2.0), includes cmsObjectiveC1 } cmsNestedProcVars, { support nested procedural variables } cmsNonLocalGoto, { support non local gotos (like iso pascal) } cmsAdvancedRecords, { advanced record syntax with visibility sections, methods and properties } cmsISOLike_unary_minus,{ unary minus like in iso pascal: same precedence level as binary minus/plus } cmsSystemcodepage, { use system codepage as compiler codepage by default, emit ansistrings with system codepage } cmsFinalFields, { allows declaring fields as "final", which means they must be initialised in the (class) constructor and are constant from then on (same as final fields in Java) } cmsDefault_unicodestring, { makes the default string type in $h+ mode unicodestring rather than ansistring; similarly, char becomes unicodechar rather than ansichar } cmsTypeHelpers, { allows the declaration of "type helper" (non-Delphi) or "record helper" (Delphi) for primitive types } cmsCBlocks, { support for http://en.wikipedia.org/wiki/Blocks_(C_language_extension) } cmsISOlike_IO, { I/O as it required by an ISO compatible compiler } cmsISOLike_Program_Para, { program parameters as it required by an ISO compatible compiler } cmsISOLike_Mod, { mod operation as it is required by an iso compatible compiler } cmsPrefixedAttributes, { allow Delphi attributes, disable FPC [] proc modifier } cmsExternalClass, { pas2js: allow class external [pkgname] name [symbol] } cmsIgnoreAttributes, { pas2js: ignore attributes } cmsIgnoreInterfaces { pas2js: ignore class interfaces } ); TCompilerModeSwitches = set of TCompilerModeSwitch; const // see fpc/compiler/globals.pp DefaultCompilerModeSwitches: array[TCompilerMode] of TCompilerModeSwitches = ( // cmFPC [cmsString_pchar,cmsNested_comment,cmsRepeat_forward,cmsCvar_support, cmsInitfinal,cmsHintdirective,cmsProperty,cmsDefault_inline, cmsResult], // cmDELPHI [cmsClass,cmsObjpas,cmsResult,cmsString_pchar, cmsPointer_2_procedure,cmsAutoderef,cmsTp_procvar,cmsInitfinal,cmsDefault_ansistring, cmsOut,cmsDefault_para,cmsDuplicate_names,cmsHintdirective, cmsProperty,cmsDefault_inline,cmsExcept,cmsAdvancedRecords, cmsPrefixedAttributes], // cmDELPHIUNICODE [cmsClass,cmsObjpas,cmsResult,cmsString_pchar, cmsPointer_2_procedure,cmsAutoderef,cmsTp_procvar,cmsInitfinal, cmsOut,cmsDefault_para,cmsDuplicate_names,cmsHintdirective, cmsProperty,cmsDefault_inline,cmsExcept,cmsAdvancedRecords, cmsSystemcodepage,cmsDefault_unicodestring, cmsPrefixedAttributes], // cmGPC [cmsTp_procvar], // cmTP [cmsResult,cmsTp_procvar,cmsDuplicate_names], // cmOBJFPC [cmsClass,cmsObjpas,cmsResult,cmsString_pchar,cmsNested_comment, cmsRepeat_forward,cmsCvar_support,cmsInitfinal,cmsOut,cmsDefault_para, cmsHintdirective,cmsProperty,cmsDefault_inline,cmsExcept], // cmMacPas [cmsCvar_support,cmsMac_procvar,cmsNestedProcVars, cmsNonLocalGoto,cmsISOLike_unary_minus,cmsDefault_inline], // cmISO [cmsTp_procvar,cmsDuplicate_names,cmsNestedProcVars,cmsNonLocalGoto, cmsISOLike_unary_minus], // cmExtPas [cmsTp_procvar,cmsDuplicate_names,cmsNestedProcVars,cmsNonLocalGoto, cmsISOLike_unary_minus,cmsISOlike_IO, cmsISOLike_Program_Para, cmsISOLike_Mod] ); cmAllModesWithGeneric = [cmDELPHI,cmDELPHIUNICODE,cmOBJFPC]; type // see fpcsrc/compiler/globtype.pas toptimizerswitch TOptimizerSwitch = ( cs_opt_none, cs_opt_level1,cs_opt_level2,cs_opt_level3,cs_opt_level4, cs_opt_regvar,cs_opt_uncertain,cs_opt_size,cs_opt_stackframe, cs_opt_peephole,cs_opt_loopunroll,cs_opt_tailrecursion,cs_opt_nodecse, cs_opt_nodedfa,cs_opt_loopstrength,cs_opt_scheduler,cs_opt_autoinline,cs_useebp,cs_userbp, cs_opt_reorder_fields,cs_opt_fastmath, cs_opt_dead_values, cs_opt_remove_emtpy_proc, cs_opt_constant_propagate, cs_opt_dead_store_eliminate, cs_opt_forcenostackframe ); toptimizerswitches = set of toptimizerswitch; const OptimizerSwitchStr : array[toptimizerswitch] of string[17] = ('', 'LEVEL1','LEVEL2','LEVEL3','LEVEL4', 'REGVAR','UNCERTAIN','SIZE','STACKFRAME', 'PEEPHOLE','LOOPUNROLL','TAILREC','CSE', 'DFA','STRENGTH','SCHEDULE','AUTOINLINE','USEEBP','USERBP', 'ORDERFIELDS','FASTMATH','DEADVALUES','REMOVEEMPTYPROCS', 'CONSTPROP', 'DEADSTORE','FORCENOSTACKFRAME' ); type TPascalCompiler = ( pcFPC, pcDelphi, pcPas2js); type TLSSkippingDirective = ( lssdNone, lssdTillElse, lssdTillEndIf ); TLSDirectiveKind = ( lsdkNone, // if lsdkIf, lsdkIfC, lsdkIfDef, lsdkIfNDef, lsdkIfOpt, // else lsdkElIfC, lsdkElse, lsdkElseC, lsdkElseIf, // end lsdkEndC, lsdkEndif, lsdkIfEnd, // misc lsdkDefine, lsdkInclude, lsdkIncludePath, lsdkLongSwitch, lsdkMacro, lsdkMode, lsdkModeSwitch, lsdkSetC, lsdkShortSwitch, lsdkThreading, lsdkUndef ); TLSDirectiveKinds = set of TLSDirectiveKind; const lsdkAllIf = [lsdkIf,lsdkIfC,lsdkIfDef,lsdkIfNDef,lsdkIfOpt]; lsdkAllElse = [lsdkElIfC,lsdkElse,lsdkElseC,lsdkElseIf]; lsdkAllEnd = [lsdkEndC,lsdkEndif,lsdkIfEnd]; type TLSDirectiveState = ( lsdsActive, // was executed lsdsInactive,// was executed, but expression result was false and following code was skipped lsdsSkipped // was not executed ); TLSDirectiveStates = set of TLSDirectiveState; TLSDirective = record CleanPos: integer; Level: integer; State: TLSDirectiveState; Code: Pointer; // TCodeBuffer SrcPos: integer; // 1-based position in Code Kind: TLSDirectiveKind; end; PLSDirective = ^TLSDirective; PPLSDirective = ^PLSDirective; type { TMissingIncludeFile is a missing include file together with all params involved in the search } TMissingIncludeFile = class public IncludePath: string; Filename: string; constructor Create(const AFilename, AIncludePath: string); function CalcMemSize: PtrUInt; end; { TMissingIncludeFiles is a list of TMissingIncludeFile } TMissingIncludeFiles = class(TList) private function GetIncFile(Index: Integer): TMissingIncludeFile; procedure SetIncFile(Index: Integer; const AValue: TMissingIncludeFile); public procedure Clear; override; procedure Delete(Index: Integer); function CalcMemSize: PtrUInt; property Items[Index: Integer]: TMissingIncludeFile read GetIncFile write SetIncFile; default; end; TDirectiveSequenceItemValue = record CleanPos: integer; Value: string; end; TSequenceDirective = (sdScopedEnums); TDirectiveSequenceItem = class private FItems: array of TDirectiveSequenceItemValue; FLastItem: integer; public constructor Create; procedure Add(const AValue: string; ACleanPos: integer); function FindValue(const ACleanPos: integer; out Value: string): Boolean; procedure Clear(FreeMemory: boolean); function CalcMemSize: PtrUInt; end; TDirectiveSequence = class private FDirectives: array[TSequenceDirective] of TDirectiveSequenceItem; public constructor Create; destructor Destroy; override; procedure Add(ADirective: TSequenceDirective; const ADirectiveValue: string; ACleanPos: Integer); function FindValue(ADirective: TSequenceDirective; ACleanPos: Integer; out Value: string): Boolean; procedure Clear(FreeMemory: boolean); function CalcMemSize: PtrUInt; end; { LinkScanner Token Types } TLSTokenType = ( lsttNone, lsttSrcEnd, // no more tokens lsttWord, lsttEqual, lsttPoint, lsttSemicolon, lsttComma, lsttStringConstant, lsttEnd ); { Error handling } { ELinkScannerError } ELinkScannerError = class(Exception) Sender: TLinkScanner; Id: int64; constructor Create(ASender: TLinkScanner; TheId: int64; const AMessage: string); end; ELinkScannerErrors = class of ELinkScannerError; TLinkScannerProgress = function(Sender: TLinkScanner): boolean of object; ELinkScannerAbort = class(ELinkScannerError); ELinkScannerConsistency = class(ELinkScannerError); { ELinkScannerEditError } ELinkScannerEditError = class(ELinkScannerError) Buffer: Pointer; BufferPos: integer; constructor Create(ASender: TLinkScanner; TheId: int64; const AMessage: string; ABuffer: Pointer; ABufferPos: integer); end; TLinkScannerState = ( lssSourcesChanged, // used source buffers changed lssInitValuesChanged, // used init values changed lssFilesChanged, // used files on disk changed lssIgnoreMissingIncludeFiles ); TLinkScannerStates = set of TLinkScannerState; { TLinkScanner } TLinkScanner = class(TObject) private FLinks: PSourceLink; // list of TSourceLink, sorted for CleanedPos FLinkCount: integer; FLinkCapacity: integer; FCleanedSrc: string; FLastCleanedSrcLen: integer; FOnGetSource: TOnGetSource; FOnGetFileName: TOnGetFileName; FOnGetSourceStatus: TOnGetSourceStatus; FOnLoadSource: TOnLoadSource; FOnDeleteSource: TOnDeleteSource; FOnCheckFileOnDisk: TOnCheckFileOnDisk; FOnGetInitValues: TOnGetInitValues; FOnIncludeCode: TOnIncludeCode; FOnProgress: TLinkScannerProgress; FIgnoreErrorAfterCode: Pointer; FIgnoreErrorAfterCursorPos: integer; FInitValues: TExpressionEvaluator; FInitValuesChangeStep: integer; FSourceChangeSteps: TFPList; // list of PSourceChangeStep sorted with Code FChangeStep: integer; FMainSourceFilename: string; FMainCode: pointer; FScanTill: TLinkScannerRange; FNestedComments: boolean; // for speed reasons keep this flag redundant with the CompilerModeSwitches FStates: TLinkScannerStates; FHiddenUsedUnits: string; // comma separated // global write lock FOnSetGlobalWriteLock: TOnSetWriteLock; FGlobalSourcesChangeStep: int64; FGlobalFilesChangeStep: int64; FGlobalInitValuesChangeStep: integer; function GetLinks(Index: integer): TSourceLink; inline; function GetLinkP(Index: integer): PSourceLink; inline; procedure SetLinks(Index: integer; const Value: TSourceLink); procedure SetSource(ACode: Pointer); // set current source procedure AddSourceChangeStep(ACode: pointer; AChangeStep: integer); procedure AddLink(ASrcPos: integer; ACode: Pointer; AKind: TSourceLinkKind = slkCode); procedure IncreaseChangeStep; inline; procedure SetMainCode(const Value: pointer); procedure SetScanTill(const Value: TLinkScannerRange); function GetIgnoreMissingIncludeFiles: boolean; procedure SetIgnoreMissingIncludeFiles(const Value: boolean); function TokenIs(const AToken: string): boolean; function UpTokenIs(const AToken: string): boolean; private // parsing CommentStyle: TCommentStyle; CommentLevel: integer; CommentStartPos: integer; // position of '{', '(*', '//' CommentInnerStartPos: integer; // position after '{', '(*', '//' CommentInnerEndPos: integer; // position of '}', '*)', #10 CommentEndPos: integer; // postion after '}', '*)', #10 CopiedSrcPos: integer; IfLevel: integer; procedure ReadNextToken; function ReturnFromIncludeFileAndIsEnd: boolean; function ReadIdentifier: string; function ReadUpperIdentifier: string; procedure ReadSpace; inline; procedure ReadCurlyComment; procedure ReadLineComment; procedure ReadRoundComment; procedure CommentEndNotFound(id: int64); procedure EndComment; inline; procedure IncCommentLevel; inline; procedure DecCommentLevel; inline; procedure HandleDirective; procedure UpdateCleanedSource(NewCopiedSrcPos: integer); function ReturnFromIncludeFile: boolean; function ParseKeyWord(StartPos: integer; LastTokenType: TLSTokenType): boolean; function DoEndToken: boolean; inline; function DoSourceTypeToken: boolean; inline; function DoInterfaceToken: boolean; inline; function DoImplementationToken: boolean; inline; function DoFinalizationToken: boolean; inline; function DoInitializationToken: boolean; inline; function DoUsesToken: boolean; inline; function TokenIsWord(p: PChar): boolean; inline; private // directives FDirectives: PLSDirective; FDirectivesCount: integer; FDirectivesCapacity: integer; FDirectivesSorted: PPLSDirective; // array of PLSDirective to items of FDirectives FDirectiveName: string; FDirectiveCleanPos: integer; FDirectivesStored: boolean; FMacrosOn: boolean; FMissingIncludeFiles: TMissingIncludeFiles; FIncludeStack: TFPList; // list of TSourceLink FOnGetGlobalChangeSteps: TLSOnGetGlobalChangeSteps; FSkippingDirectives: TLSSkippingDirective; FSkipIfLevel: integer; FStoreDirectives: integer; FCompilerMode: TCompilerMode; FCompilerModeSwitches: TCompilerModeSwitches; FPascalCompiler: TPascalCompiler; FMacros: PSourceLinkMacro; FMacroCount, fMacroCapacity: integer; FDirectiveSequence: TDirectiveSequence; function GetDirectives(Index: integer): PLSDirective; inline; function GetDirectivesSorted(Index: integer): PLSDirective; inline; procedure SetCompilerMode(const AValue: TCompilerMode); procedure SetPascalCompiler(const AValue: TPascalCompiler); procedure SkipTillEndifElse(SkippingUntil: TLSSkippingDirective); procedure SortDirectives; function InternalIfDirective: boolean; procedure EndSkipping; procedure AddSkipComment(IsStart: boolean); procedure SetDirectiveValueWithSequence(ADirective: TSequenceDirective; const ADirectiveValue: string); function IfdefDirective: boolean; function IfCDirective: boolean; function IfndefDirective: boolean; function IfDirective: boolean; function IfOptDirective: boolean; function EndifDirective: boolean; function EndCDirective: boolean; function IfEndDirective: boolean; function ElseDirective: boolean; function ElseCDirective: boolean; function ElseIfDirective: boolean; function ElIfCDirective: boolean; function DefineDirective: boolean; function UndefDirective: boolean; function SetCDirective: boolean; function IncludeDirective: boolean; function IncludePathDirective: boolean; function ShortSwitchDirective: boolean; function ReadNextSwitchDirective: boolean; function LongSwitchDirective: boolean; function LongSwitchDirectiveWithSequence(const ADirective: TSequenceDirective): boolean; function MacroDirective: boolean; function ModeDirective: boolean; function ModeSwitchDirective: boolean; function ThreadingDirective: boolean; function DoDirective(StartPos, DirLen: integer): boolean; function IncludeFile(const AFilename: string): boolean; procedure PushIncludeLink(ACleanedPos, ASrcPos: integer; ACode: Pointer); function PopIncludeLink: TSourceLink; function GetIncludeFileIsMissing: boolean; function MissingIncludeFilesNeedsUpdate: boolean; procedure ClearMissingIncludeFiles; // code macros procedure AddMacroValue(MacroName: PChar; ValueStart, ValueEnd: integer); procedure ClearMacros; function IndexOfMacro(MacroName: PChar; InsertPos: boolean): integer; procedure AddMacroSource(MacroID: integer); protected // error: the error is in range Succ(ScannedRange) LastErrorMessage: string; LastErrorSrcPos: integer; LastErrorCode: pointer; LastErrorIsValid: boolean; LastErrorBehindIgnorePosition: boolean; LastErrorCheckedForIgnored: boolean; LastErrorId: int64; CleanedIgnoreErrorAfterPosition: integer;// ignore if valid and >= procedure RaiseExceptionFmt(id: int64; const AMessage: string; const Args: array of const); procedure RaiseException(id: int64; const AMessage: string); procedure RaiseExceptionClass(id: int64; const AMessage: string; ExceptionClass: ELinkScannerErrors); procedure RaiseEditException(id: int64; const AMessage: string; ABuffer: Pointer; ABufferPos: integer); procedure RaiseConsistencyException(id: int64; const AMessage: string); procedure ClearLastError; procedure RaiseLastError; procedure DoCheckAbort; public // current values, positions, source, flags CleanedLen: integer; Src: string; // current parsed source (= TCodeBuffer(Code).Source) SrcPos: integer; // current position (1-based in Src) TokenStart: integer; // start position of current token TokenType: TLSTokenType; SrcLen: integer; // length of current source Code: pointer; // current code object (TCodeBuffer) Values: TExpressionEvaluator; SrcFilename: string;// current parsed filename (= TCodeBuffer(Code).Filename) IsUnit: boolean; SourceName: string; ScannedRange: TLinkScannerRange; // excluding the section with a syntax error function MainFilename: string; property ChangeStep: integer read FChangeStep; // see CTInvalidChangeStamp // links property Links[Index: integer]: TSourceLink read GetLinks write SetLinks; property LinkP[Index: integer]: PSourceLink read GetLinkP; property LinkCount: integer read FLinkCount; function LinkIndexAtCleanPos(ACleanPos: integer): integer; function LinkIndexAtCursorPos(ACursorPos: integer; ACode: Pointer): integer; function LinkSize(Index: integer): integer; function LinkSize_Inline(Index: integer): integer; inline; function LinkCleanedEndPos(Index: integer): integer; function LinkCleanedEndPos_Inline(Index: integer): integer; inline; function LinkSourceLog(Index: integer): TSourceLog; function FindFirstSiblingLink(LinkIndex: integer): integer; function FindParentLink(LinkIndex: integer): integer; function LinkIndexNearCursorPos(ACursorPos: integer; ACode: Pointer; var CursorInLink: boolean): integer; function CreateTreeOfSourceCodes: TAVLTree; // directives property Directives[Index: integer]: PLSDirective read GetDirectives; property DirectivesSorted[Index: integer]: PLSDirective read GetDirectivesSorted; // sorted for Code and SrcPos property DirectiveCount: integer read FDirectivesCount; procedure ClearDirectives(FreeMemory: boolean); function StoreDirectives: boolean; inline; // store directives on next Scan procedure DemandStoreDirectives; // increase internal counter to StoreDirectives procedure ReleaseStoreDirectives; // decrease internal counter to StoreDirectives property DirectivesStored: boolean read FDirectivesStored; // directives were stored on last scan function FindDirective(aCode: Pointer; aSrcPos: integer; out FirstSortedIndex, LastSortedIndex: integer): boolean; function FindFirstDirective(aCode: Pointer; aSrcPos: integer; const AllowedStates: TLSDirectiveStates): PLSDirective; function GetDirectiveValueAt(ADirective: TSequenceDirective; ACleanPos: integer): string; // source mapping (Cleaned <-> Original) function CleanedSrc: string; function CursorToCleanPos(ACursorPos: integer; ACode: pointer; out ACleanPos: integer): integer; // 0=valid CleanPos //-1=CursorPos was skipped, CleanPos between two links // 1=CursorPos beyond scanned code function CleanedPosToCursor(ACleanedPos: integer; out ACursorPos: integer; out ACode: Pointer): boolean; function CleanedPosToStr(ACleanedPos: integer): string; function LastErrorIsInFrontOfCleanedPos(ACleanedPos: integer): boolean; procedure RaiseLastErrorIfInFrontOfCleanedPos(ACleanedPos: integer); // ranges function WholeRangeIsWritable(CleanStartPos, CleanEndPos: integer; ErrorOnFail: boolean): boolean; procedure FindCodeInRange(CleanStartPos, CleanEndPos: integer; UniqueSortedCodeList: TFPList); procedure DeleteRange(CleanStartPos,CleanEndPos: integer); // scanning procedure Scan(Range: TLinkScannerRange; CheckFilesOnDisk: boolean); function UpdateNeeded(Range: TLinkScannerRange; CheckFilesOnDisk: boolean): boolean; procedure SetIgnoreErrorAfter(ACursorPos: integer; ACode: Pointer); procedure ClearIgnoreErrorAfter; function IgnoreErrAfterPositionIsInFrontOfLastErrMessage: boolean; function IgnoreErrorAfterCleanedPos: integer;// before using this, check if valid! function IgnoreErrorAfterValid: boolean; function CleanPosIsAfterIgnorePos(CleanPos: integer): boolean; function LoadSourceCaseLoUp(const AFilename: string; AllowVirtual: boolean = false): pointer; function SearchIncludeFile(AFilename: string; out NewCode: Pointer; var MissingIncludeFile: TMissingIncludeFile): boolean; function GuessMisplacedIfdefEndif(StartCursorPos: integer; StartCode: pointer; out EndCursorPos: integer; out EndCode: Pointer): boolean; function GetHiddenUsedUnits: string; // comma separated // global write lock procedure ActivateGlobalWriteLock; procedure DeactivateGlobalWriteLock; property OnGetGlobalChangeSteps: TLSOnGetGlobalChangeSteps read FOnGetGlobalChangeSteps write FOnGetGlobalChangeSteps; property OnSetGlobalWriteLock: TOnSetWriteLock read FOnSetGlobalWriteLock write FOnSetGlobalWriteLock; // properties property OnGetSource: TOnGetSource read FOnGetSource write FOnGetSource; property OnLoadSource: TOnLoadSource read FOnLoadSource write FOnLoadSource; property OnDeleteSource: TOnDeleteSource read FOnDeleteSource write FOnDeleteSource; property OnGetSourceStatus: TOnGetSourceStatus read FOnGetSourceStatus write FOnGetSourceStatus; property OnGetFileName: TOnGetFileName read FOnGetFileName write FOnGetFileName; property OnCheckFileOnDisk: TOnCheckFileOnDisk read FOnCheckFileOnDisk write FOnCheckFileOnDisk; property OnGetInitValues: TOnGetInitValues read FOnGetInitValues write FOnGetInitValues; property OnIncludeCode: TOnIncludeCode read FOnIncludeCode write FOnIncludeCode; property OnProgress: TLinkScannerProgress read FOnProgress write FOnProgress; property IgnoreMissingIncludeFiles: boolean read GetIgnoreMissingIncludeFiles write SetIgnoreMissingIncludeFiles; property InitialValues: TExpressionEvaluator read FInitValues write FInitValues; property MainCode: pointer read FMainCode write SetMainCode; property IncludeFileIsMissing: boolean read GetIncludeFileIsMissing; property NestedComments: boolean read FNestedComments; property CompilerMode: TCompilerMode read FCompilerMode write SetCompilerMode; property CompilerModeSwitches: TCompilerModeSwitches read FCompilerModeSwitches write FCompilerModeSwitches; property PascalCompiler: TPascalCompiler read FPascalCompiler write SetPascalCompiler; property ScanTill: TLinkScannerRange read FScanTill write SetScanTill; procedure Clear; procedure ConsistencyCheck; procedure WriteDebugReport; procedure CalcMemSize(Stats: TCTMemStats); constructor Create; destructor Destroy; override; end; //---------------------------------------------------------------------------- // memory system for PSourceLink(s) TPSourceLinkMemManager = class(TCodeToolMemManager) protected procedure FreeFirstItem; override; public procedure DisposePSourceLink(Link: PSourceLink); function NewPSourceLink: PSourceLink; end; // memory system for PSourceLink(s) TPSourceChangeStepMemManager = class(TCodeToolMemManager) protected procedure FreeFirstItem; override; public procedure DisposePSourceChangeStep(Step: PSourceChangeStep); function NewPSourceChangeStep: PSourceChangeStep; end; const // upper case CompilerModeNames: array[TCompilerMode] of string=( 'FPC', 'DELPHI', 'DELPHIUNICODE', 'GPC', 'TP', 'OBJFPC', 'MACPAS', 'ISO', 'EXTPAS' ); // upper case (see fpc/compiler/globtype.pas modeswitchstr ) CompilerModeSwitchNames: array[TCompilerModeSwitch] of string=( 'POINTERARITHMETICS', 'CLASS', 'OBJPAS', 'RESULT', 'PCHARTOSTRING', 'CVAR', 'NESTEDCOMMENTS', 'CLASSICPROCVARS', 'MACPROCVARS', 'REPEATFORWARD', 'POINTERTOPROCVAR', 'AUTODEREF', 'INITFINAL', 'ANSISTRINGS', 'OUT', 'DEFAULTPARAMETERS', 'HINTDIRECTIVE', 'DUPLICATELOCALS', 'PROPERTIES', 'ALLOWINLINE', 'EXCEPTIONS', 'OBJECTIVEC1', 'OBJECTIVEC2', 'NESTEDPROCVARS', 'NONLOCALGOTO', 'ADVANCEDRECORDS', 'ISOUNARYMINUS', 'SYSTEMCODEPAGE', 'FINALFIELDS', 'UNICODESTRINGS', 'TYPEHELPERS', 'CBLOCKS', 'ISOIO', 'ISOPROGRAMPARAS', 'ISOMOD', 'PREFIXEDATTRIBUTES', 'EXTERNALCLASS', 'IGNOREATTRIBUTES', 'IGNOREINTERFACES' ); // upper case PascalCompilerNames: array[TPascalCompiler] of string=( 'FPC', 'DELPHI', 'PAS2JS' ); const DirectiveSequenceName: array [TSequenceDirective] of string = ('SCOPEDENUMS'); var CompilerModeVars: array[TCompilerMode] of string; PSourceLinkMemManager: TPSourceLinkMemManager; PSourceChangeStepMemManager: TPSourceChangeStepMemManager; function StrToCompilerMode(const aName: string): TCompilerMode; function StrToPascalCompiler(const aName: string): TPascalCompiler; procedure AddCodeToUniqueList(ACode: Pointer; UniqueSortedCodeList: TFPList); function IndexOfCodeInUniqueList(ACode: Pointer; UniqueSortedCodeList: TList): integer; function IndexOfCodeInUniqueList(ACode: Pointer; UniqueSortedCodeList: TFPList): integer; function dbgs(r: TLinkScannerRange): string; overload; function dbgs(const ModeSwitches: TCompilerModeSwitches): string; overload; function dbgs(k: TSourceLinkKind): string; overload; function dbgs(s: TLSDirectiveState): string; overload; function dbgs(s: TLSDirectiveKind): string; overload; implementation // useful procs ---------------------------------------------------------------- function IndexOfCodeInUniqueList(ACode: Pointer; UniqueSortedCodeList: TList): integer; var l,m,r: integer; begin l:=0; r:=UniqueSortedCodeList.Count-1; m:=0; while r>=l do begin m:=(l+r) shr 1; if ACodeUniqueSortedCodeList[m] then l:=m+1 else begin Result:=m; exit; end; end; Result:=-1; end; function IndexOfCodeInUniqueList(ACode: Pointer; UniqueSortedCodeList: TFPList): integer; var l,m,r: integer; begin l:=0; r:=UniqueSortedCodeList.Count-1; m:=0; while r>=l do begin m:=(l+r) shr 1; if ACodeUniqueSortedCodeList[m] then l:=m+1 else begin Result:=m; exit; end; end; Result:=-1; end; function dbgs(r: TLinkScannerRange): string; overload; begin WriteStr(Result, r); end; function dbgs(const ModeSwitches: TCompilerModeSwitches): string; var ms: TCompilerModeSwitch; begin Result:=''; for ms:=Low(TCompilerModeSwitches) to high(TCompilerModeSwitches) do if ms in ModeSwitches then begin Result:=Result+CompilerModeSwitchNames[ms]+','; end; System.Delete(Result,length(Result),1); // cut comma Result:='['+Result+']'; end; function dbgs(k: TSourceLinkKind): string; begin case k of slkCode: Result:='Code'; slkMissingIncludeFile: Result:='MissingInc'; slkSkipStart: Result:='SkipStart'; slkSkipEnd: Result:='SkipEnd'; else Result:='?'; end; end; function dbgs(s: TLSDirectiveState): string; begin case s of lsdsActive: Result:='Active'; lsdsInactive: Result:='Inactive'; lsdsSkipped: Result:='Skipped'; end; end; function dbgs(s: TLSDirectiveKind): string; begin WriteStr(Result,s); end; function StrToCompilerMode(const aName: string): TCompilerMode; begin for Result:=low(Result) to high(Result) do if SysUtils.CompareText(aName,CompilerModeNames[Result])=0 then exit; Result:=cmFPC; end; function StrToPascalCompiler(const aName: string): TPascalCompiler; begin for Result:=low(Result) to high(Result) do if SysUtils.CompareText(aName,PascalCompilerNames[Result])=0 then exit; Result:=pcFPC; end; procedure AddCodeToUniqueList(ACode: Pointer; UniqueSortedCodeList: TFPList); var l,m,r: integer; begin l:=0; r:=UniqueSortedCodeList.Count-1; m:=0; while r>=l do begin m:=(l+r) shr 1; if ACodeUniqueSortedCodeList[m] then l:=m+1 else exit; end; if (mUniqueSortedCodeList[m]) then inc(m); UniqueSortedCodeList.Insert(m,ACode); end; function CompareUpToken(const UpToken: string; const Txt: string; TxtStartPos, TxtEndPos: integer): boolean; var len, i: integer; begin Result:=false; len:=TxtEndPos-TxtStartPos; if len<>length(UpToken) then exit; i:=1; while iUpChars[Txt[TxtStartPos]]) then exit; inc(i); inc(TxtStartPos); end; Result:=true; end; function CompareLSDirectiveCodeSrcPos(Item1, Item2: Pointer): Integer; var Dir1: PLSDirective absolute Item1; Dir2: PLSDirective absolute Item2; begin Result:=ComparePointers(Dir1^.Code,Dir2^.Code); if Result<>0 then exit; Result:=Dir1^.SrcPos-Dir2^.SrcPos; end; function CompareLSDirectiveCodeSrcPosCleanPos(Item1, Item2: Pointer): Integer; var Dir1: PLSDirective absolute Item1; Dir2: PLSDirective absolute Item2; begin Result:=ComparePointers(Dir1^.Code,Dir2^.Code); if Result<>0 then exit; Result:=Dir1^.SrcPos-Dir2^.SrcPos; if Result<>0 then exit; Result:=Dir1^.CleanPos-Dir2^.CleanPos; end; { TDirectiveSequenceItem } constructor TDirectiveSequenceItem.Create; begin FLastItem := -1; end; procedure TDirectiveSequenceItem.Add(const AValue: string; ACleanPos: integer); begin if (FLastItem >= 0) and (ACleanPos <= FItems[FLastItem].CleanPos) then raise Exception.Create('Internal error: TDirectiveSequenceItem.Add: ACleanPos not sorted.'); if Length(FItems) = 0 then SetLength(FItems, 1) else if FLastItem = High(FItems) then SetLength(FItems, Length(FItems)+Min(128, Length(FItems))); Inc(FLastItem); FItems[FLastItem].CleanPos := ACleanPos; FItems[FLastItem].Value := AValue; end; function TDirectiveSequenceItem.CalcMemSize: PtrUInt; var Item: TDirectiveSequenceItemValue; begin Result:=PtrUInt(InstanceSize) +PtrUInt(Length(FItems))*PtrUInt(SizeOf(TDirectiveSequenceItemValue)); for Item in FItems do Inc(Result, MemSizeString(Item.Value)); end; procedure TDirectiveSequenceItem.Clear(FreeMemory: boolean); begin if FreeMemory then SetLength(FItems, 0); FLastItem := -1; end; function TDirectiveSequenceItem.FindValue(const ACleanPos: integer; out Value: string): Boolean; function BinarySearch: integer; var I, Max, Min: Integer; ResIndex, ResCleanPos: integer; begin Max := FLastItem; Min := 0; ResIndex := -1; ResCleanPos := -1; while (Min <= Max) do begin I := (Max+Min) div 2; if (FItems[I].CleanPos = ACleanPos) then Exit(I) else if (FItems[I].CleanPos < ACleanPos) then begin if ResCleanPos < FItems[I].CleanPos then begin //store the closest ResIndex := I; ResCleanPos := FItems[I].CleanPos; end; Min := I + 1; end else begin Max := I - 1; end; end; Result := ResIndex; end; var ItemIndex: integer; begin ItemIndex := BinarySearch; Result := ItemIndex >= 0; if Result then Value := FItems[ItemIndex].Value else Value := ''; end; { TDirectiveSequence } constructor TDirectiveSequence.Create; var I: TSequenceDirective; begin for I := Low(FDirectives) to High(FDirectives) do FDirectives[I] := TDirectiveSequenceItem.Create; end; procedure TDirectiveSequence.Add(ADirective: TSequenceDirective; const ADirectiveValue: string; ACleanPos: Integer); begin FDirectives[ADirective].Add(ADirectiveValue, ACleanPos); end; function TDirectiveSequence.CalcMemSize: PtrUInt; var Item: TDirectiveSequenceItem; begin Result:=PtrUInt(InstanceSize); for Item in FDirectives do Inc(Result, Item.CalcMemSize); end; procedure TDirectiveSequence.Clear(FreeMemory: boolean); var Item: TDirectiveSequenceItem; begin for Item in FDirectives do Item.Clear(FreeMemory); end; destructor TDirectiveSequence.Destroy; var Item: TDirectiveSequenceItem; begin for Item in FDirectives do Item.Free; inherited Destroy; end; function TDirectiveSequence.FindValue(ADirective: TSequenceDirective; ACleanPos: Integer; out Value: string): Boolean; begin Result := FDirectives[ADirective].FindValue(ACleanPos, Value); end; { TLinkScanner } // inline function TLinkScanner.GetLinks(Index: integer): TSourceLink; begin Result:=FLinks[Index]; end; // inline function TLinkScanner.GetLinkP(Index: integer): PSourceLink; begin Result:=@FLinks[Index]; end; // inline procedure TLinkScanner.IncreaseChangeStep; begin if FChangeStep=$7fffffff then FChangeStep:=-$7fffffff else inc(FChangeStep); end; // inline procedure TLinkScanner.ReadSpace; begin while (SrcPos<=SrcLen) and (IsSpaceChar[Src[SrcPos]]) do inc(SrcPos); end; // inline procedure TLinkScanner.EndComment; begin CommentStyle:=CommentNone; end; // inline procedure TLinkScanner.IncCommentLevel; begin if FNestedComments then inc(CommentLevel) else CommentLevel:=1; end; // inline procedure TLinkScanner.DecCommentLevel; begin if FNestedComments then dec(CommentLevel) else CommentLevel:=0; end; // inline function TLinkScanner.DoEndToken: boolean; begin TokenType:=lsttEnd; Result:=true; end; // inline function TLinkScanner.DoSourceTypeToken: boolean; // program, unit, library, package // unit unit1; // unit a.b.unit1 platform; // unit unit1 unimplemented; begin if ScannedRange<>lsrInit then exit(false); Result:=true; ScannedRange:=lsrSourceType; IsUnit:=TokenIsWord('UNIT'); if ScannedRange=ScanTill then exit; repeat ReadNextToken; // read identifier if TokenType=lsttWord then begin if SourceName<>'' then SourceName:=SourceName+'.'; SourceName:=SourceName+GetIdentifier(@Src[TokenStart]); ReadNextToken; // read ';' or '.' or hint modifier end; until TokenType<>lsttPoint; ScannedRange:=lsrSourceName; if ScannedRange=ScanTill then exit; end; // inline function TLinkScanner.DoInterfaceToken: boolean; begin if ord(ScannedRange)>=ord(lsrInterfaceStart) then exit(false); ScannedRange:=lsrInterfaceStart; Result:=true; end; // inline function TLinkScanner.DoFinalizationToken: boolean; begin if ord(ScannedRange)>=ord(lsrFinalizationStart) then exit(false); ScannedRange:=lsrFinalizationStart; Result:=true; end; // inline function TLinkScanner.DoInitializationToken: boolean; begin if ord(ScannedRange)>=ord(lsrInitializationStart) then exit(false); ScannedRange:=lsrInitializationStart; Result:=true; end; function TLinkScanner.DoUsesToken: boolean; // uses name, name in 'string'; begin if ord(ScannedRange)<=ord(lsrInterfaceStart) then ScannedRange:=lsrMainUsesSectionStart else if ScannedRange=lsrImplementationStart then ScannedRange:=lsrImplementationUsesSectionStart else exit(false); repeat // read unit name repeat ReadNextToken; if (TokenType<>lsttWord) or WordIsKeyWord.DoItCaseInsensitive(@Src[SrcPos]) then exit(false); ReadNextToken; until TokenType<>lsttPoint; if TokenIs('in') then begin // read "in" filename ReadNextToken; if TokenType=lsttStringConstant then ReadNextToken; end; if TokenType=lsttSemicolon then break; if TokenType<>lsttComma then begin // syntax error -> this token does not belong to the uses section SrcPos:=TokenStart; break; end; until false; ScannedRange:=succ(ScannedRange); // lsrMainUsesSectionEnd, lsrImplementationUsesSectionEnd; Result:=true; end; // inline function TLinkScanner.DoImplementationToken: boolean; begin if ord(ScannedRange)>=ord(lsrImplementationStart) then exit(false); ScannedRange:=lsrImplementationStart; Result:=true; end; // inline function TLinkScanner.TokenIsWord(p: PChar): boolean; begin Result:=(TokenType=lsttWord) and (CompareIdentifiers(p,@Src[TokenStart])=0); end; // inline function TLinkScanner.GetDirectives(Index: integer): PLSDirective; begin Result:=@FDirectives[Index]; end; // inline function TLinkScanner.GetDirectivesSorted(Index: integer): PLSDirective; begin Result:=FDirectivesSorted[Index]; end; function TLinkScanner.GetDirectiveValueAt(ADirective: TSequenceDirective; ACleanPos: integer): string; begin if not FDirectiveSequence.FindValue(ADirective, ACleanPos, Result) then Result := FInitValues.Variables[DirectiveSequenceName[ADirective]]; end; // inline function TLinkScanner.LinkSize_Inline(Index: integer): integer; var Link: PSourceLink; begin Link:=@FLinks[Index]; if Index+10; end; procedure TLinkScanner.SortDirectives; var i: Integer; begin // try to keep memory allocated ReAllocMem(FDirectivesSorted,SizeOf(Pointer)*FDirectivesCapacity); for i:=0 to FDirectivesCount-1 do FDirectivesSorted[i]:=@FDirectives[i]; for i:=FDirectivesCount to FDirectivesCapacity-1 do FDirectivesSorted[i]:=nil; MergeSort(PPointer(FDirectivesSorted),FDirectivesCount,@CompareLSDirectiveCodeSrcPosCleanPos); end; procedure TLinkScanner.AddLink(ASrcPos: integer; ACode: Pointer; AKind: TSourceLinkKind); var NewCapacity: Integer; Link: PSourceLink; begin if (LinkCount>0) and (FLinks[FLinkCount-1].CleanedPos=CleanedLen+1) then begin // last link is empty => remove {$IFDEF ShowUpdateCleanedSrc} Link:=@FLinks[FLinkCount-1]; debugln(['TLinkScanner.AddLink removing empty link: ',dbgs(Link^.Kind)]); {$ENDIF} dec(FLinkCount); end else if FLinkCount=FLinkCapacity then begin NewCapacity:=FLinkCapacity*2; if NewCapacity<16 then NewCapacity:=16; ReAllocMem(FLinks,NewCapacity*SizeOf(TSourceLink)); FLinkCapacity:=NewCapacity; end; Link:=@FLinks[FLinkCount]; with Link^ do begin CleanedPos:=CleanedLen+1; SrcPos:=ASrcPos; Code:=ACode; Kind:=AKind; end; inc(FLinkCount); end; function TLinkScanner.CleanedSrc: string; begin if length(FCleanedSrc)<>CleanedLen then begin SetLength(FCleanedSrc,CleanedLen); end; Result:=FCleanedSrc; if FLastCleanedSrcLen=LinkCount) then IndexOutOfBounds; Result:=LinkSize_Inline(Index); end; function TLinkScanner.LinkCleanedEndPos(Index: integer): integer; begin Result:=LinkSize(Index)+FLinks[Index].CleanedPos; end; function TLinkScanner.LinkSourceLog(Index: integer): TSourceLog; begin if Assigned(OnGetSource) then Result:=OnGetSource(Self,FLinks[Index].Code) else Result:=nil; end; function TLinkScanner.FindFirstSiblingLink(LinkIndex: integer): integer; { find link at the start of the code e.g. The resulting link SrcPos is always 1 if LinkIndex is in the main code, the result will be 0 if LinkIndex is in an include file, the result will be the first link of the include file. If the include file is included multiple times, it is treated as if they are different files. } var LastIndex: integer; begin Result:=LinkIndex; if LinkIndex>=0 then begin LastIndex:=LinkIndex; while (Result>=0) do begin if FLinks[Result].Code=FLinks[LinkIndex].Code then begin if Links[Result].SrcPos>FLinks[LastIndex].SrcPos then begin // the include file was (in-)directly included by itself // -> skip Result:=FindParentLink(Result); end else if FLinks[Result].SrcPos=1 then begin // start found exit; end; LastIndex:=Result; end; dec(Result); end; end; end; function TLinkScanner.FindParentLink(LinkIndex: integer): integer; // a parent link is the link of the include directive // or in other words: the link in front of the first sibling link begin Result:=FindFirstSiblingLink(LinkIndex); if Result>=0 then dec(Result); end; function TLinkScanner.LinkIndexNearCursorPos(ACursorPos: integer; ACode: Pointer; var CursorInLink: boolean): integer; // returns the nearest link at cursorpos // (either covering the cursorpos or in front) var CurLinkSize: integer; BestLinkIndex: integer; begin BestLinkIndex:=-1; Result:=0; CursorInLink:=false; while Result=FLinks[Result].SrcPos) then begin CurLinkSize:=LinkSize(Result); if ACursorPosnil then FDirectivesSorted[0]:=nil; end; FDirectiveSequence.Clear(FreeMemory); end; procedure TLinkScanner.DemandStoreDirectives; begin inc(FStoreDirectives); end; procedure TLinkScanner.ReleaseStoreDirectives; begin if FStoreDirectives=0 then raise Exception.Create(''); dec(FStoreDirectives); if FStoreDirectives=0 then ClearDirectives(true); end; function TLinkScanner.FindDirective(aCode: Pointer; aSrcPos: integer; out FirstSortedIndex, LastSortedIndex: integer): boolean; var l: Integer; r: Integer; m: Integer; Dir: TLSDirective; cmp: Integer; begin Dir.Code:=aCode; Dir.SrcPos:=aSrcPos; l:=0; r:=FDirectivesCount-1; while l<=r do begin m:=(l+r) div 2; cmp:=CompareLSDirectiveCodeSrcPos(@Dir,DirectivesSorted[m]); if cmp<0 then r:=m-1 else if cmp>0 then l:=m+1 else begin // found FirstSortedIndex:=m; LastSortedIndex:=m; while (FirstSortedIndex>0) and (CompareLSDirectiveCodeSrcPos(@Dir,DirectivesSorted[FirstSortedIndex-1])=0) do dec(FirstSortedIndex); while (LastSortedIndex+1CleanedLen) then exit; // binary search through the links l:=0; r:=LinkCount-1; while l<=r do begin m:=(l+r) div 2; if m=FLinks[m+1].CleanedPos then l:=m+1 else begin Result:=m; exit; end; end else begin if ACleanPos>=FLinks[m].CleanedPos then begin Result:=m; exit; end else ConsistencyError2; end; end; ConsistencyError1; end; function TLinkScanner.LinkIndexAtCursorPos(ACursorPos: integer; ACode: Pointer ): integer; var CurLinkSize: integer; begin Result:=0; while Result=FLinks[Result].SrcPos) then begin CurLinkSize:=LinkSize(Result); if ACursorPosSrcLen) and ReturnFromIncludeFileAndIsEnd then exit; //DebugLn([' TLinkScanner.ReadNextToken SrcPos=',SrcPos,' SrcLen=',SrcLen,' "',copy(Src,SrcPos,5),'"']); // Skip all spaces and comments p:=@Src[SrcPos]; while true do begin case p^ of #0: begin SrcPos:=p-PChar(Src)+1; if (SrcPos>SrcLen) then begin if ReturnFromIncludeFileAndIsEnd then exit; if (SrcPos>SrcLen) then break; end else inc(SrcPos); p:=@Src[SrcPos]; end; '{' : begin SrcPos:=p-PChar(Src)+1; ReadCurlyComment; p:=@Src[SrcPos]; end; '/': if p[1]='/' then begin SrcPos:=p-PChar(Src)+1; ReadLineComment; p:=@Src[SrcPos]; end else break; '(': if p[1]='*' then begin SrcPos:=p-PChar(Src)+1; ReadRoundComment; p:=@Src[SrcPos]; end else break; ' ',#9,#10,#13: repeat inc(p); until not (p^ in [' ',#9,#10,#13]); else break; end; end; TokenStart:=p-PChar(Src)+1; // read token c1:=p^; case c1 of '_','A'..'Z','a'..'z': begin // keyword or identifier inc(p); while IsIdentChar[p^] do inc(p); TokenType:=lsttWord; SrcPos:=p-PChar(Src)+1; if FMacrosOn then begin MacroID:=IndexOfMacro(@Src[TokenStart],false); if MacroID>=0 then begin AddMacroSource(MacroID); end; end; end; '&': begin // identifier with "&" character or an octal number inc(p); case p^ of '_','A'..'Z','a'..'z'://identifier: &uses begin inc(p); while IsIdentChar[p^] do inc(p); TokenType:=lsttWord; end; '0'..'7'://octal number: &10 begin inc(p); while IsOctNumberChar[p^] do inc(p); TokenType:=lsttNone; end; else TokenType:=lsttNone; end; SrcPos:=p-PChar(Src)+1; end; '''','#': begin TokenType:=lsttStringConstant; while true do begin case p^ of #0: begin SrcPos:=p-PChar(Src)+1; if SrcPos>SrcLen then break; inc(p); end; '#': begin inc(p); while IsNumberChar[p^] do inc(p); end; '''': begin inc(p); while true do begin case p^ of #0: begin SrcPos:=p-PChar(Src)+1; if SrcPos>SrcLen then break; inc(p); end; '''': begin inc(p); break; end; #10,#13: break; else inc(p); end; end; end; else break; end; end; SrcPos:=p-PChar(Src)+1; end; '0'..'9': begin TokenType:=lsttNone; inc(p); while IsNumberChar[p^] do inc(p); if (p^='.') and (p[1]<>'.') then begin // real type number inc(p); while IsNumberChar[p^] do inc(p); if (p^ in ['E','e']) then begin // read exponent inc(p); if (p^ in ['-','+']) then inc(p); while IsNumberChar[p^] do inc(p); end; end; SrcPos:=p-PChar(Src)+1; end; '%': // boolean begin TokenType:=lsttNone; inc(p); while p^ in ['0'..'1'] do inc(p); SrcPos:=p-PChar(Src)+1; end; '$': // hex begin TokenType:=lsttNone; inc(p); while IsHexNumberChar[p^] do inc(p); SrcPos:=p-PChar(Src)+1; end; '=': begin SrcPos:=p-PChar(Src)+2; TokenType:=lsttEqual; end; '.': begin SrcPos:=p-PChar(Src)+2; TokenType:=lsttPoint; end; ';': begin SrcPos:=p-PChar(Src)+2; TokenType:=lsttSemicolon; end; ',': begin SrcPos:=p-PChar(Src)+2; TokenType:=lsttComma; end; else TokenType:=lsttNone; inc(p); c2:=p^; // test for double char operators // :=, +=, -=, /=, *=, <>, <=, >=, **, ><, .. if ((c2='=') and (IsEqualOperatorStartChar[c1])) or ((c1='<') and (c2='>')) or ((c1='>') and (c2='<')) or ((c1='.') and (c2='.')) or ((c1='*') and (c2='*')) then inc(p); SrcPos:=p-PChar(Src)+1; end; end; {$IFDEF RangeChecking}{$R+}{$UNDEF RangeChecking}{$ENDIF} procedure TLinkScanner.Scan(Range: TLinkScannerRange; CheckFilesOnDisk: boolean); var LastTokenType: TLSTokenType; cm: TCompilerMode; pc: TPascalCompiler; s: string; LastProgressPos: integer; CheckForAbort: boolean; NewSrcLen: Integer; begin if (not UpdateNeeded(Range,CheckFilesOnDisk)) then begin // input is the same as last time -> output is the same // -> if there was an error and it was in a needed range, raise it again if LastErrorIsValid then begin // the error has happened in ScannedRange if ord(ScannedRange)>=ord(Range) then begin // error is behind needed range => ok end else if (not IgnoreErrorAfterValid) or (not IgnoreErrAfterPositionIsInFrontOfLastErrMessage) then RaiseLastError; end; exit; end; {$IFDEF CTDEBUG} DebugLn('TLinkScanner.Scan A -------- Range=',dbgs(Range)); {$ENDIF} ScanTill:=Range; Clear; if Assigned(OnGetGlobalChangeSteps) then OnGetGlobalChangeSteps(FGlobalSourcesChangeStep,FGlobalFilesChangeStep, FGlobalInitValuesChangeStep); FStates:=FStates-[lssSourcesChanged,lssFilesChanged,lssInitValuesChanged]; {$IFDEF CTDEBUG} DebugLn('TLinkScanner.Scan B '); {$ENDIF} SetSource(FMainCode); NewSrcLen:=length(Src); if NewSrcLen'' then begin for pc:=Low(TPascalCompiler) to High(TPascalCompiler) do if (s=PascalCompilerNames[pc]) then PascalCompiler:=pc; end else if InitialValues.IsDefined('pas2js') then PascalCompiler:=pcPas2js else if InitialValues.IsDefined('delphi') and not InitialValues.IsDefined('fpc') then PascalCompiler:=pcDelphi; // compiler mode for cm:=Low(TCompilerMode) to High(TCompilerMode) do if FInitValues.IsDefined(CompilerModeVars[cm]) then begin CompilerMode:=cm; break; end; // nested comments override if Values.IsDefined(ExternalMacroStart+'NestedComments') then FNestedComments:=true; //DebugLn(['TLinkScanner.Scan ',MainFilename,' ',PascalCompilerNames[PascalCompiler],' ',CompilerModeNames[CompilerMode],' FNestedComments=',FNestedComments,' ModeSwitches=',dbgs(CompilerModeSwitches)]); //DebugLn(Values.AsString); FMacrosOn:=(Values.Variables['MACROS']<>'0'); if Src='' then exit; // begin scanning AddLink(SrcPos,Code); LastTokenType:=lsttNone; LastProgressPos:=0; CheckForAbort:=Assigned(OnProgress); {$IFDEF CTDEBUG} DebugLn('TLinkScanner.Scan F ',dbgs(SrcLen)); {$ENDIF} ScannedRange:=lsrInit; if ScanTill=lsrInit then exit; try try ReadNextToken; if TokenIsWord('USES') then DoUsesToken else SrcPos:=TokenStart; while ord(ScanTill)>ord(ScannedRange) do begin // check every 100.000 bytes for abort if CheckForAbort and ((LastProgressPos-CopiedSrcPos)>100000) then begin LastProgressPos:=CopiedSrcPos; DoCheckAbort; end; //debugln(['TLinkScanner.Scan Token ',dbgstr(Src,TokenStart,SrcPos-TokenStart)]); ReadNextToken; if TokenType=lsttWord then ParseKeyWord(TokenStart,LastTokenType); //writeln('TLinkScanner.Scan G "',copy(Src,TokenStart,SrcPos-TokenStart),'" LastTokenType=',LastTokenType,' TokenType=',TokenType); if (LastTokenType=lsttEnd) and (TokenType=lsttPoint) then begin //DebugLn(['TLinkScanner.Scan END. ',MainFilename]); ScannedRange:=lsrEnd; break; end; if (SrcPos>SrcLen) and ReturnFromIncludeFileAndIsEnd then break; LastTokenType:=TokenType; end; finally {$IFDEF ShowUpdateCleanedSrc} DebugLn('TLinkScanner.Scan UpdatePos=',DbgS(SrcPos-1)); {$ENDIF} if (SrcPos>CopiedSrcPos) then UpdateCleanedSource(SrcPos-1); if FDirectivesCount>0 then SortDirectives; if FSkippingDirectives<>lssdNone then begin {$IFDEF ShowUpdateCleanedSrc} DebugLn(['TLinkScanner.Scan missing $ENDIF']); {$ENDIF} end; end; IncreaseChangeStep; FLastCleanedSrcLen:=CleanedLen; except on E: ELinkScannerError do begin if (not IgnoreErrorAfterValid) or (not IgnoreErrAfterPositionIsInFrontOfLastErrMessage) then raise; {$IFDEF ShowIgnoreErrorAfter} DebugLn('TLinkScanner.Scan IGNORING ERROR: ',LastErrorMessage); {$ENDIF} end; end; {$IFDEF CTDEBUG} DebugLn('TLinkScanner.Scan END ',dbgs(CleanedLen),' ',dbgs(ScannedRange)); {$ENDIF} end; procedure TLinkScanner.SetLinks(Index: integer; const Value: TSourceLink); begin FLinks[Index]:=Value; end; procedure TLinkScanner.ReadCurlyComment; // a normal pascal {} comment var p: PChar; begin CommentStyle:=CommentCurly; CommentStartPos:=SrcPos; IncCommentLevel; CommentInnerStartPos:=SrcPos+1; p:=@Src[SrcPos]; inc(p); // HandleSwitches can dec CommentLevel while true do begin case p^ of #0: begin SrcPos:=p-PChar(Src)+1; if SrcPos>SrcLen then break; end; '{' : IncCommentLevel; '}' : begin DecCommentLevel; if CommentLevel=0 then begin inc(p); break; end; end; end; inc(p); end; SrcPos:=p-PChar(Src)+1; CommentEndPos:=SrcPos; CommentInnerEndPos:=SrcPos-1; if (CommentLevel>0) then CommentEndNotFound(20170422130048); // handle compiler switches if Src[CommentInnerStartPos]='$' then HandleDirective; EndComment; end; procedure TLinkScanner.ReadLineComment; // a // newline comment var p: PChar; begin CommentStyle:=CommentLine; CommentStartPos:=SrcPos; IncCommentLevel; p:=@Src[SrcPos]; inc(p,2); CommentInnerStartPos:=SrcPos; while not (p^ in [#0,#10,#13]) do inc(p); DecCommentLevel; if p^<>#0 then inc(p); SrcPos:=p-PChar(Src)+1; CommentEndPos:=SrcPos; CommentInnerEndPos:=SrcPos-1; // handle compiler switches (ignore) EndComment; end; procedure TLinkScanner.ReadRoundComment; // a delphi comment (* *) var p: PChar; begin CommentStyle:=CommentLine; CommentStartPos:=SrcPos; IncCommentLevel; CommentInnerStartPos:=SrcPos+2; p:=@Src[SrcPos]; inc(p,2); while true do begin case p^ of #0: begin SrcPos:=p-PChar(Src)+1; if SrcPos>SrcLen then break; end; '*': begin inc(p); if p^=')' then begin inc(p); DecCommentLevel; if CommentLevel=0 then break; end; end; '(': begin inc(p); if FNestedComments and (p^='*') then begin inc(p); IncCommentLevel; end; end; else inc(p); end; end; SrcPos:=p-PChar(Src)+1; CommentEndPos:=SrcPos; CommentInnerEndPos:=SrcPos-2; if (CommentLevel>0) then CommentEndNotFound(20170422130050); // handle compiler switches if Src[CommentInnerStartPos]='$' then HandleDirective; EndComment; end; procedure TLinkScanner.CommentEndNotFound(id: int64); begin SrcPos:=CommentStartPos; RaiseException(id,ctsCommentEndNotFound); end; procedure TLinkScanner.UpdateCleanedSource(NewCopiedSrcPos: integer); // add new parsed code to cleaned source string procedure RaiseInvalid; begin debugln(['TLinkScanner.UpdateCleanedSource inconsistency found: Srclen=',SrcLen,'=',length(Src),' FCleanedSrc=',CleanedLen,'/',length(FCleanedSrc),' CopiedSrcPos=',CopiedSrcPos,' NewCopiedSrcPos=',NewCopiedSrcPos,' AddLen=',NewCopiedSrcPos-CopiedSrcPos]); RaiseConsistencyException(20170422130054,'TLinkScanner.UpdateCleanedSource inconsistency found AddLen='+dbgs(NewCopiedSrcPos-CopiedSrcPos)); end; var AddLen: integer; begin if NewCopiedSrcPos>SrcLen then NewCopiedSrcPos:=SrcLen+1; if NewCopiedSrcPos=CopiedSrcPos then exit; AddLen:=NewCopiedSrcPos-CopiedSrcPos; if AddLen<=0 then RaiseInvalid; if AddLen>length(FCleanedSrc)-CleanedLen then begin // expand cleaned source string by at least 1024 SetLength(FCleanedSrc,length(FCleanedSrc)+SrcLen+1024); end; System.Move(Src[CopiedSrcPos+1],FCleanedSrc[CleanedLen+1],AddLen); inc(CleanedLen,AddLen); {$IFDEF ShowUpdateCleanedSrc} DebugLn('TLinkScanner.UpdateCleanedSource A ', DbgS(CopiedSrcPos),'-',DbgS(NewCopiedSrcPos),'="', StringToPascalConst(copy(Src,CopiedSrcPos+1,20)), '".."',StringToPascalConst(copy(Src,NewCopiedSrcPos-19,20)),'"'); {$ENDIF} CopiedSrcPos:=NewCopiedSrcPos; end; procedure TLinkScanner.AddSourceChangeStep(ACode: pointer; AChangeStep: integer); procedure RaiseCodeNil; begin RaiseConsistencyException(20170422130109,'TLinkScanner.AddSourceChangeStep ACode=nil'); end; var l,r,m: integer; NewSrcChangeStep: PSourceChangeStep; c: pointer; begin //DebugLn('[TLinkScanner.AddSourceChangeStep] ',DbgS(ACode)); if ACode=nil then RaiseCodeNil; l:=0; r:=FSourceChangeSteps.Count-1; m:=0; c:=nil; while (l<=r) do begin m:=(l+r) shr 1; c:=PSourceChangeStep(FSourceChangeSteps[m])^.Code; if cACode then r:=m-1 else exit; end; NewSrcChangeStep:=PSourceChangeStepMemManager.NewPSourceChangeStep; NewSrcChangeStep^.Code:=ACode; NewSrcChangeStep^.ChangeStep:=AChangeStep; if (FSourceChangeSteps.Count>0) and (c=1) then begin ATokenLen:=length(AToken); if ATokenLen=SrcPos-TokenStart then begin for i:=1 to ATokenLen do if AToken[i]<>Src[TokenStart-1+i] then exit; Result:=true; end; end; end; function TLinkScanner.UpTokenIs(const AToken: string): boolean; var ATokenLen: integer; i: integer; begin Result:=false; if (SrcPos<=SrcLen+1) and (TokenStart>=1) then begin ATokenLen:=length(AToken); if ATokenLen=SrcPos-TokenStart then begin for i:=1 to ATokenLen do if AToken[i]<>UpChars[Src[TokenStart-1+i]] then exit; Result:=true; end; end; end; procedure TLinkScanner.ConsistencyCheck; var i: integer; begin if (FLinks=nil) xor (FLinkCapacity=0) then RaiseCatchableException(''); if FLinks<>nil then begin for i:=0 to FLinkCount-1 do begin if (FLinks[i].Code=nil) and (FLinks[i].Kind=slkCode) then RaiseCatchableException(''); if (FLinks[i].CleanedPos<1) or (FLinks[i].CleanedPos>SrcLen) then RaiseCatchableException(''); end; end; if SrcLen<>length(Src) then RaiseCatchableException(''); if Values<>nil then Values.ConsistencyCheck; end; procedure TLinkScanner.WriteDebugReport; var i: integer; begin // header DebugLn(''); DebugLn('[TLinkScanner.WriteDebugReport]', ' ChangeStepCount=',dbgs(FSourceChangeSteps.Count), ' LinkCount=',dbgs(LinkCount), ' CleanedLen=',dbgs(CleanedLen)); // time stamps for i:=0 to FSourceChangeSteps.Count-1 do begin DebugLn(' ChangeStep ',dbgs(i),': ' ,' Code=',dbgs(PSourceChangeStep(FSourceChangeSteps[i])^.Code) ,' ChangeStep=',dbgs(PSourceChangeStep(FSourceChangeSteps[i])^.ChangeStep)); end; // links for i:=0 to LinkCount-1 do begin DebugLn([' Link ',i,':' ,' CleanedPos=',FLinks[i].CleanedPos ,' SrcPos=',FLinks[i].SrcPos ,' Code=',dbgs(FLinks[i].Code) ,' Kind=',dbgs(FLinks[i].Kind) ,' Src="',dbgstr(CleanedSrc,FLinks[i].CleanedPos,Min(50,LinkSize(i))),'"' ]); end; end; procedure TLinkScanner.CalcMemSize(Stats: TCTMemStats); begin Stats.Add('TLinkScanner', PtrUInt(InstanceSize) +MemSizeString(FMainSourceFilename) +length(FDirectiveName) +MemSizeString(LastErrorMessage) +MemSizeString(SrcFilename)); Stats.Add('TLinkScanner.CleanedSrc',MemSizeString(FCleanedSrc)); // Note: Src belongs to the codebuffer if FLinks<>nil then Stats.Add('TLinkScanner.FLinks', FLinkCapacity*SizeOf(TSourceLink)); if FInitValues<>nil then Stats.Add('TLinkScanner.FInitValues', FInitValues.CalcMemSize(false)); // FInitValues are copies of strings of TDefineTree if FSourceChangeSteps<>nil then Stats.Add('TLinkScanner.FSourceChangeSteps', FSourceChangeSteps.InstanceSize +FSourceChangeSteps.Capacity*SizeOf(TSourceChangeStep)); if FIncludeStack<>nil then Stats.Add('TLinkScanner.FIncludeStack', FIncludeStack.InstanceSize+FIncludeStack.Capacity*SizeOf(TSourceLink)); if Values<>nil then Stats.Add('TLinkScanner.Values', Values.CalcMemSize(true,FInitValues)); if FDirectiveSequence<>nil then Stats.Add('TLinkScanner.FDirectiveSequence', FDirectiveSequence.CalcMemSize); if FMissingIncludeFiles<>nil then Stats.Add('TLinkScanner.FMissingIncludeFiles', FMissingIncludeFiles.InstanceSize); end; function TLinkScanner.UpdateNeeded( Range: TLinkScannerRange; CheckFilesOnDisk: boolean): boolean; { the clean source must be rebuilt if 1. a former check says so 2. scanrange increased 3. unit source changed 4. one of its include files changed 5. init values changed (e.g. initial compiler defines) 6. a missing include file can now be found } var i: integer; SrcLog: TSourceLog; NewInitValues: TExpressionEvaluator; NewInitValuesChangeStep: integer; SrcChange: PSourceChangeStep; CurSourcesChangeStep, CurFilesChangeStep: int64; CurInitValuesChangeStep: integer; begin Result:=true; if Range=lsrNone then exit(false); if not Assigned(FOnCheckFileOnDisk) then CheckFilesOnDisk:=false; // use the last check result if [lssSourcesChanged,lssInitValuesChanged]*FStates<>[] then exit; if CheckFilesOnDisk and (lssFilesChanged in FStates) then exit; // check options if StoreDirectives and (not DirectivesStored) then exit; // check if range increased // Note: if there was an error, then a range increase will raise the same error // and no update is needed if (ord(Range)>ord(ScannedRange)) and (not LastErrorIsValid) then begin {$IFDEF VerboseUpdateNeeded} DebugLn(['TLinkScanner.UpdateNeeded because range increased from ',dbgs(ScannedRange),' to ',dbgs(Range),' ',ExtractFilename(MainFilename)]); {$ENDIF} exit; end; // do a quick test: check the global change steps for sources and values if Assigned(OnGetGlobalChangeSteps) then begin OnGetGlobalChangeSteps(CurSourcesChangeStep,CurFilesChangeStep,CurInitValuesChangeStep); if (CurSourcesChangeStep=FGlobalSourcesChangeStep) and (CurInitValuesChangeStep=FGlobalInitValuesChangeStep) and ((not CheckFilesOnDisk) or (CurFilesChangeStep=FGlobalSourcesChangeStep)) then begin // sources and values did not change since last check //debugln(['TLinkScanner.UpdateNeeded global change steps still the same: ',MainFilename]); Result:=false; exit; end; end else begin CurSourcesChangeStep:=1; CurFilesChangeStep:=1; CurInitValuesChangeStep:=1; end; // check initvalues //if ExtractFileNameOnly(MainFilename)='androidr14' then //debugln(['TLinkScanner.UpdateNeeded FGlobalInitValuesChangeStep=',FGlobalInitValuesChangeStep,' CurInitValuesChangeStep=',CurInitValuesChangeStep]); if FGlobalInitValuesChangeStep<>CurInitValuesChangeStep then begin FGlobalInitValuesChangeStep:=CurInitValuesChangeStep; if Assigned(FOnGetInitValues) then begin NewInitValues:=FOnGetInitValues(Self,Code,NewInitValuesChangeStep); if (NewInitValues<>nil) and (NewInitValuesChangeStep<>FInitValuesChangeStep) and (not FInitValues.Equals(NewInitValues)) then begin {$IFDEF VerboseUpdateNeeded} //if ExtractFileNameOnly(MainFilename)='androidr14' then DebugLn(['TLinkScanner.UpdateNeeded because InitValues changed ',MainFilename]); {$ENDIF} Include(FStates,lssInitValuesChanged); exit; end; end; end; // check all used codebuffers if FGlobalSourcesChangeStep<>CurSourcesChangeStep then begin FGlobalSourcesChangeStep:=CurSourcesChangeStep; if Assigned(FOnGetSource) then begin for i:=0 to FSourceChangeSteps.Count-1 do begin SrcChange:=PSourceChangeStep(FSourceChangeSteps[i]); SrcLog:=FOnGetSource(Self,SrcChange^.Code); //debugln(['TLinkScanner.UpdateNeeded ',ExtractFilename(MainFilename),' i=',i,' File=',FOnGetFileName(Self,SrcLog),' Last=',SrcChange^.ChangeStep,' Now=',SrcLog.ChangeStep]); if SrcChange^.ChangeStep<>SrcLog.ChangeStep then begin {$IFDEF VerboseUpdateNeeded} DebugLn(['TLinkScanner.UpdateNeeded because source buffer changed: ',OnGetFileName(Self,SrcLog),' MainFilename=',MainFilename]); {$ENDIF} Include(FStates,lssSourcesChanged); exit; end; end; end; end; // check all file dates if CheckFilesOnDisk then begin if FGlobalFilesChangeStep<>CurFilesChangeStep then begin FGlobalFilesChangeStep:=CurFilesChangeStep; if Assigned(FOnGetSource) then begin // if files changed on disk, reload them for i:=0 to FSourceChangeSteps.Count-1 do begin SrcChange:=PSourceChangeStep(FSourceChangeSteps[i]); SrcLog:=FOnGetSource(Self,SrcChange^.Code); if FOnCheckFileOnDisk(SrcLog) then begin {$IFDEF VerboseUpdateNeeded} DebugLn(['TLinkScanner.UpdateNeeded because file on disk changed: ',OnGetFileName(Self,SrcLog),' MainFilename=',MainFilename]); {$ENDIF} Include(FStates,lssFilesChanged); exit; end; end; end; end; end; // check missing include files if CheckFilesOnDisk and MissingIncludeFilesNeedsUpdate then begin {$IFDEF VerboseUpdateNeeded} DebugLn(['TLinkScanner.UpdateNeeded because MissingIncludeFilesNeedsUpdate']); {$ENDIF} Include(FStates,lssFilesChanged); exit; end; // no update needed :) //DebugLn('TLinkScanner.UpdateNeeded END'); Result:=false; end; procedure TLinkScanner.SetIgnoreErrorAfter(ACursorPos: integer; ACode: Pointer); begin if (FIgnoreErrorAfterCode=ACode) and (FIgnoreErrorAfterCursorPos=ACursorPos) then exit; FIgnoreErrorAfterCode:=ACode; FIgnoreErrorAfterCursorPos:=ACursorPos; LastErrorCheckedForIgnored:=false; {$IFDEF ShowIgnoreErrorAfter} DbgOut('TLinkScanner.SetIgnoreErrorAfter '); if FIgnoreErrorAfterCode<>nil then DbgOut(OnGetFileName(Self,FIgnoreErrorAfterCode)) else DbgOut('nil'); DbgOut(' ',dbgs(FIgnoreErrorAfterCursorPos)); DebugLn(''); {$ENDIF} end; procedure TLinkScanner.ClearIgnoreErrorAfter; begin SetIgnoreErrorAfter(0,nil); end; function TLinkScanner.IgnoreErrAfterPositionIsInFrontOfLastErrMessage: boolean; var CleanResult: integer; begin //DebugLn('TLinkScanner.IgnoreErrAfterPositionIsInFrontOfLastErrMessage'); //DebugLn([' LastErrorCheckedForIgnored=',LastErrorCheckedForIgnored, // ' LastErrorBehindIgnorePosition=',LastErrorBehindIgnorePosition]); if LastErrorCheckedForIgnored then Result:=LastErrorBehindIgnorePosition else begin CleanedIgnoreErrorAfterPosition:=-1; if (FIgnoreErrorAfterCode<>nil) and (FIgnoreErrorAfterCursorPos>0) then begin CleanResult:=CursorToCleanPos(FIgnoreErrorAfterCursorPos, FIgnoreErrorAfterCode,CleanedIgnoreErrorAfterPosition); {$IFDEF ShowIgnoreErrorAfter} DebugLn([' CleanResult=',CleanResult, ' CleanedIgnoreErrorAfterPosition=',CleanedIgnoreErrorAfterPosition, ' FIgnoreErrorAfterCursorPos=',FIgnoreErrorAfterCursorPos, ' CleanedLen=',CleanedLen, ' LastErrorIsValid=',LastErrorIsValid]); {$ENDIF} if (CleanResult=0) or (CleanResult=-1) or (not LastErrorIsValid) then begin Result:=true; end else begin Result:=false; end; end else begin Result:=false; end; LastErrorBehindIgnorePosition:=Result; LastErrorCheckedForIgnored:=true; end; {$IFDEF ShowIgnoreErrorAfter} DebugLn('TLinkScanner.IgnoreErrAfterPositionIsInFrontOfLastErrMessage Result=',dbgs(Result)); {$ENDIF} end; function TLinkScanner.IgnoreErrorAfterCleanedPos: integer; begin if IgnoreErrAfterPositionIsInFrontOfLastErrMessage then Result:=CleanedIgnoreErrorAfterPosition else Result:=-1; {$IFDEF ShowIgnoreErrorAfter} DebugLn('TLinkScanner.IgnoreErrorAfterCleanedPos Result=',dbgs(Result)); {$ENDIF} end; function TLinkScanner.IgnoreErrorAfterValid: boolean; begin Result:=(FIgnoreErrorAfterCode<>nil); {$IFDEF ShowIgnoreErrorAfter} DebugLn('TLinkScanner.IgnoreErrorAfterValid Result=',dbgs(Result)); {$ENDIF} end; function TLinkScanner.CleanPosIsAfterIgnorePos(CleanPos: integer): boolean; var p: LongInt; begin if IgnoreErrorAfterValid then begin p:=IgnoreErrorAfterCleanedPos; if p<1 then Result:=false else Result:=CleanPos>=p; end else begin Result:=false end; end; function TLinkScanner.LastErrorIsInFrontOfCleanedPos(ACleanedPos: integer ): boolean; begin Result:=LastErrorIsValid and (CleanedLen skip rest of code AToken.StartPos:=ASrcLen; end; else // in different kind of comment -> ignore end; '*': // turbo pascal comment end if (AToken.StartPos skip rest of code AToken.StartPos:=ASrcLen; end; else // in different kind of comment -> ignore end; end; #10,#13: // line end if AToken.Range in [trDelphiComment] then begin AToken.EndPos:=AToken.StartPos+1; if (AToken.StartPosASrc[AToken.StartPos]) then inc(AToken.EndPos); AToken.TheType:=ttLineEnd; AToken.Range:=trCode; exit; end else begin // in different kind of comment -> ignore end; '''': // skip string constant begin inc(AToken.StartPos); while (AToken.StartPos<=ASrcLen) do begin if (not (ASrc[AToken.StartPos] in ['''',#10,#13])) then begin inc(AToken.StartPos); end else begin break; end; end; end; end; inc(AToken.StartPos); end; // at the end of the code AToken.EndPos:=AToken.StartPos; AToken.TheType:=ttNone; Result:=false; end; procedure FreeIfStack(var IfStack: TFPList); var i: integer; AnIf: PIf; begin if IfStack=nil then exit; for i:=0 to IfStack.Count-1 do begin AnIf:=PIf(IfStack[i]); AnIf^.Expression:=''; Dispose(AnIf); end; IfStack.Free; IfStack:=nil; end; function InitGuessMisplaced(out CurToken: TToken; ACode: Pointer; out ASrc: string; out ASrcLen: integer): boolean; var ASrcLog: TSourceLog; begin Result:=false; // get source if (FOnGetSource=nil) then exit; ASrcLog:=FOnGetSource(Self,ACode); if ASrcLog=nil then exit; ASrc:=ASrcLog.Source; ASrcLen:=length(ASrc); CurToken.StartPos:=1; CurToken.EndPos:=1; CurToken.Range:=trCode; CurToken.TheType:=ttNone; CurToken.NestedComments:=NestedComments; Result:=true; end; function ReadDirectiveType(const ASrc: string; AToken: TToken): TDirectiveType; const DIR_RST: array[0..5] of TDirectiveType = ( dtIfDef, dtIfNDef, dtIfOpt, dtIf, dtElse, dtEndif ); DIR_TXT: array[0..5] of PChar = ( 'IFDEF', 'IFNDEF', 'IFOPT', 'IF', 'ELSE', 'ENDIF' ); var ASrcLen, p: integer; n: Integer; begin Result:=dtUnknown; ASrcLen:=length(ASrc); p:=AToken.EndPos; if (p misplaced directive found EndCursorPos:=CurToken.EndPos; EndCode:=ACode; DebugLn('GuessMisplacedIfdefEndif $ELSE has no $IF'); Result:=true; exit; end; PIf(IfStack[IfStack.Count-1])^.HasElse:=true; end; dtEndif: begin if (IfStack.Count=0) then begin // this $ENDIF has no $IF // -> misplaced directive found EndCursorPos:=CurToken.EndPos; EndCode:=ACode; DebugLn('GuessMisplacedIfdefEndif $ENDIF has no $IF'); Result:=true; exit; end; PopIfFromStack(IfStack); end; end; end; until CurToken.TheType=ttNone; if IfStack.Count>0 then begin // there is an $IF without $ENDIF // -> misplaced directive found EndCursorPos:=PIf(IfStack[IfStack.Count-1])^.StartPos+1; EndCode:=ACode; DebugLn('GuessMisplacedIfdefEndif $IF without $ENDIF'); Result:=true; exit; end; finally FreeIfStack(IfStack); end; end; var LinkID, i, BestSrcPos: integer; LastCode: Pointer; SearchedCodes: TFPList; begin Result:=false; EndCursorPos:=0; EndCode:=nil; if StartCode=nil then exit; // search link before start position LinkID:=-1; BestSrcPos:=0; i:=0; while i=FLinks[i].SrcPos) then begin if (LinkID<0) or (BestSrcPos=LinkCount then exit; if FLinks[LinkID].Code=nil then continue; until (FLinks[LinkID].Code<>LastCode) and (SearchedCodes.IndexOf(FLinks[LinkID].Code)<0); end; finally SearchedCodes.Free; end; end; function TLinkScanner.GetHiddenUsedUnits: string; var Controller: String; AnUnitName: String; p: Integer; begin if FHiddenUsedUnits='' then begin // see fpc/compiler/pmodules.pp loaddefaultunits if PascalCompiler=pcDelphi then FHiddenUsedUnits:=DelphiSystemUnitName else FHiddenUsedUnits:=FPCSystemUnitName; if InitialValues.IsDefined('VER1_0') then begin if InitialValues.IsDefined('LINUX') then FHiddenUsedUnits:='SYSLINUX' else if InitialValues.IsDefined('BSD') then FHiddenUsedUnits:='SYSBSD' else if InitialValues.IsDefined('WIN32') then FHiddenUsedUnits:='SYSWIN32'; end; if InitialValues.IsDefined(MacroUseLineInfo) then FHiddenUsedUnits:=FHiddenUsedUnits+',lineinfo' else if InitialValues.IsDefined(MacroUselnfodwrf) then FHiddenUsedUnits:=FHiddenUsedUnits+',lnfodwrf'; if InitialValues.IsDefined(MacroUseValgrind) then FHiddenUsedUnits:=FHiddenUsedUnits+',cmem'; if InitialValues.IsDefined(MacroUseHeapTrc) then FHiddenUsedUnits:=FHiddenUsedUnits+',HeapTrc'; if InitialValues.IsDefined('FPC_HAS_WinLikeResources') then begin // ToDo: fpintres FHiddenUsedUnits:=FHiddenUsedUnits+',fpextres'; end; if (cmsObjpas in CompilerModeSwitches) and (PascalCompiler=pcFPC) then FHiddenUsedUnits:=FHiddenUsedUnits+',ObjPas'; if (CompilerMode=cmMacPas) then FHiddenUsedUnits:=FHiddenUsedUnits+',MacPas'; if (CompilerMode=cmISO) then FHiddenUsedUnits:=FHiddenUsedUnits+',ISO7185'; if (CompilerMode=cmExtPas) then FHiddenUsedUnits:=FHiddenUsedUnits+',ISO7185,ExtPas'; if (cmsCBlocks in CompilerModeSwitches) and (PascalCompiler=pcFPC) then FHiddenUsedUnits:=FHiddenUsedUnits+',BlockRTL'; if (cmsDefault_unicodestring in CompilerModeSwitches) then FHiddenUsedUnits:=FHiddenUsedUnits+',UUChar'; if CompilerMode=cmISO then FHiddenUsedUnits:=FHiddenUsedUnits+',iso7185'; if cmsObjectiveC1 in CompilerModeSwitches then FHiddenUsedUnits:=FHiddenUsedUnits+',ObjC,ObjCBase'; if InitialValues.IsDefined(MacroUseProfiler) then // Note: only valid on i386_go32v2,i386_watcom FHiddenUsedUnits:=FHiddenUsedUnits+',profile'; if InitialValues.IsDefined(MacroUseSysThrds) then FHiddenUsedUnits:=FHiddenUsedUnits+',SysThrds'; if InitialValues.IsDefined(MacroUseFPCylix) then FHiddenUsedUnits:=FHiddenUsedUnits+',fpcylix,dynlibs'; Controller:=InitialValues[MacroControllerUnit]; if (Controller<>'') and IsDottedIdentifier(Controller) then FHiddenUsedUnits:=FHiddenUsedUnits+','+Controller; // check if this is a hidden used unit AnUnitName:=ExtractFileNameOnly(MainFilename); if AnUnitName<>'' then begin if AnUnitName='system' then FHiddenUsedUnits:='' else begin p:=length(FHiddenUsedUnits); while p>=1 do begin while (p>1) and (FHiddenUsedUnits[p-1]<>',') do dec(p); if (CompareDottedIdentifiers(@FHiddenUsedUnits[p],PChar(AnUnitName))=0) or (IsUnit and (SourceName<>'') and (CompareDottedIdentifiers(@FHiddenUsedUnits[p],PChar(SourceName))=0)) then begin // this unit is a hidden unit => remove this and all behind from list if p>1 then System.Delete(FHiddenUsedUnits,p-1,length(FHiddenUsedUnits)) else FHiddenUsedUnits:=''; break; end; dec(p); // skip comma end; end; end; //debugln(['TLinkScanner.GetHiddenUsedUnits ',dbgs(CompilerModeSwitches),' ',PascalCompilerNames[PascalCompiler],' Mode=',CompilerModeNames[CompilerMode],' units=',FHiddenUsedUnits]); end; Result:=FHiddenUsedUnits; end; procedure TLinkScanner.SetMainCode(const Value: pointer); begin if FMainCode=Value then exit; FMainCode:=Value; FMainSourceFilename:=FOnGetFileName(Self,FMainCode); Clear; end; procedure TLinkScanner.SetScanTill(const Value: TLinkScannerRange); var OldScanRange: TLinkScannerRange; begin if FScanTill=Value then exit; OldScanRange:=FScanTill; FScanTill := Value; if ord(OldScanRange)'' then begin if (SrcPos<=SrcLen) and (Src[SrcPos] in ['-','+']) then begin if Src[SrcPos]='-' then Values.Variables[FDirectiveName] := '0' else Values.Variables[FDirectiveName] := '1'; inc(SrcPos); Result:=ReadNextSwitchDirective; end else begin if c='I' then Result:=IncludeDirective else Result:=LongSwitchDirective; end; end else Result:=true; end; function TLinkScanner.DoDirective(StartPos, DirLen: integer): boolean; var p: PChar; begin Result:=false; if StartPos>SrcLen then exit; p:=@Src[StartPos]; //DebugLn(['TLinkScanner.DoDirective ',copy(Src,StartPos,DirLen),' FSkippingDirectives=',ord(FSkippingDirectives)]); if FSkippingDirectives=lssdNone then begin if DirLen=1 then begin Result:=(CompilerSwitchesNames[UpChars[p^]]<>'') and ShortSwitchDirective; end else begin case UpChars[p^] of 'A': case UpChars[p[1]] of 'L': if CompareIdentifiers(p,'ALIGN')=0 then Result:=LongSwitchDirective; 'S': if CompareIdentifiers(p,'ASSERTIONS')=0 then Result:=LongSwitchDirective; end; 'B': if CompareIdentifiers(p,'BOOLEVAL')=0 then Result:=LongSwitchDirective; 'D': case UpChars[p[1]] of 'E': case UpChars[p[2]] of 'F': if CompareIdentifiers(p,'DEFINE')=0 then Result:=DefineDirective; 'B': if CompareIdentifiers(p,'DEBUGINFO')=0 then Result:=LongSwitchDirective; end; end; 'E': case UpChars[p[1]] of 'L': case UpChars[p[2]] of 'I': if CompareIdentifiers(p,'ELIFC')=0 then Result:=ElIfCDirective; 'S': case UpChars[p[3]] of 'E': if CompareIdentifiers(p,'ELSE')=0 then Result:=ElseDirective else if CompareIdentifiers(p,'ELSEC')=0 then Result:=ElseCDirective else if CompareIdentifiers(p,'ELSEIF')=0 then Result:=ElseIfDirective; end; end; 'N': if CompareIdentifiers(p,'ENDC')=0 then Result:=EndCDirective else if CompareIdentifiers(p,'ENDIF')=0 then Result:=EndIfDirective; 'X': if CompareIdentifiers(p,'EXTENDEDSYNTAX')=0 then Result:=LongSwitchDirective; end; 'I': case UpChars[p[1]] of 'F': case UpChars[p[2]] of 'C': if CompareIdentifiers(p,'IFC')=0 then Result:=IfCDirective; 'D': if CompareIdentifiers(p,'IFDEF')=0 then Result:=IfDefDirective; 'E': if CompareIdentifiers(p,'IFEND')=0 then Result:=IfEndDirective; 'N': if CompareIdentifiers(p,'IFNDEF')=0 then Result:=IfndefDirective; 'O': if CompareIdentifiers(p,'IFOPT')=0 then Result:=IfOptDirective; else if DirLen=2 then Result:=IfDirective; end; 'N': if CompareIdentifiers(p,'INCLUDE')=0 then Result:=IncludeDirective else if CompareIdentifiers(p,'INCLUDEPATH')=0 then Result:=IncludePathDirective; 'O': if CompareIdentifiers(p,'IOCHECKS')=0 then Result:=LongSwitchDirective; end; 'L': if CompareIdentifiers(p,'LOCALSYMBOLS')=0 then Result:=LongSwitchDirective else if CompareIdentifiers(p,'LONGSTRINGS')=0 then Result:=LongSwitchDirective; 'M': if CompareIdentifiers(p,'MODE')=0 then Result:=ModeDirective else if CompareIdentifiers(p,'MODESWITCH')=0 then Result:=ModeSwitchDirective else if CompareIdentifiers(p,'MACRO')=0 then Result:=MacroDirective; 'O': if CompareIdentifiers(p,'OPENSTRINGS')=0 then Result:=LongSwitchDirective else if CompareIdentifiers(p,'OVERFLOWCHECKS')=0 then Result:=LongSwitchDirective; 'R': if CompareIdentifiers(p,'RANGECHECKS')=0 then Result:=LongSwitchDirective else if CompareIdentifiers(p,'REFERENCEINFO')=0 then Result:=LongSwitchDirective; 'S': if CompareIdentifiers(p,'SETC')=0 then Result:=SetCDirective else if CompareIdentifiers(p,'STACKFRAMES')=0 then Result:=LongSwitchDirective else if CompareIdentifiers(p,'SCOPEDENUMS')=0 then Result:=LongSwitchDirectiveWithSequence(sdScopedEnums); 'T': if CompareIdentifiers(p,'THREADING')=0 then Result:=ThreadingDirective else if CompareIdentifiers(p,'TYPEADDRESS')=0 then Result:=LongSwitchDirective else if CompareIdentifiers(p,'TYPEINFO')=0 then Result:=LongSwitchDirective; 'U': if CompareIdentifiers(p,'UNDEF')=0 then Result:=UndefDirective; 'V': if CompareIdentifiers(p,'VARSTRINGCHECKS')=0 then Result:=LongSwitchDirective; end; end; end else begin // skipping code => read only IF directives case UpChars[p^] of 'E': case UpChars[p[1]] of 'L': case UpChars[p[2]] of 'I': if CompareIdentifiers(p,'ELIFC')=0 then Result:=ElIfCDirective; 'S': case UpChars[p[3]] of 'E': if CompareIdentifiers(p,'ELSE')=0 then Result:=ElseDirective else if CompareIdentifiers(p,'ELSEC')=0 then Result:=ElseCDirective else if CompareIdentifiers(p,'ELSEIF')=0 then Result:=ElseIfDirective; end; end; 'N': if CompareIdentifiers(p,'ENDC')=0 then Result:=EndCDirective else if CompareIdentifiers(p,'ENDIF')=0 then Result:=EndIfDirective; end; 'I': case UpChars[p[1]] of 'F': case UpChars[p[2]] of 'C': if CompareIdentifiers(p,'IFC')=0 then Result:=IfCDirective; 'D': if CompareIdentifiers(p,'IFDEF')=0 then Result:=IfDefDirective; 'E': if CompareIdentifiers(p,'IFEND')=0 then Result:=IfEndDirective; 'N': if CompareIdentifiers(p,'IFNDEF')=0 then Result:=IfndefDirective; 'O': if CompareIdentifiers(p,'IFOPT')=0 then Result:=IfOptDirective; else if DirLen=2 then Result:=IfDirective; end; end; end; end; end; function TLinkScanner.LongSwitchDirective: boolean; // example: {$ASSERTIONS ON comment} var ValStart: integer; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkLongSwitch; ReadSpace; ValStart:=SrcPos; while (SrcPos<=SrcLen) and IsWordChar[Src[SrcPos]] do inc(SrcPos); if CompareUpToken('ON',Src,ValStart,SrcPos) then Values.Variables[FDirectiveName] := '1' else if CompareUpToken('OFF',Src,ValStart,SrcPos) then Values.Variables[FDirectiveName] := '0' else if CompareUpToken('PRELOAD',Src,ValStart,SrcPos) and (FDirectiveName='ASSERTIONS') then Values.Variables[FDirectiveName] := 'PRELOAD' else if (FDirectiveName='LOCALSYMBOLS') then // ignore "localsymbols " else if (FDirectiveName='RANGECHECKS') then // ignore "rangechecks " else if (FDirectiveName='ALIGN') then // set record align size else begin RaiseExceptionFmt(20170422130112,ctsInvalidFlagValueForDirective, [copy(Src,ValStart,SrcPos-ValStart),FDirectiveName]); end; Result:=ReadNextSwitchDirective; end; function TLinkScanner.LongSwitchDirectiveWithSequence( const ADirective: TSequenceDirective): boolean; var ValStart: integer; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkLongSwitch; ReadSpace; ValStart:=SrcPos; while (SrcPos<=SrcLen) and IsWordChar[Src[SrcPos]] do inc(SrcPos); if CompareUpToken('ON',Src,ValStart,SrcPos) then SetDirectiveValueWithSequence(ADirective, '1') else if CompareUpToken('OFF',Src,ValStart,SrcPos) then SetDirectiveValueWithSequence(ADirective, '0') else begin RaiseExceptionFmt(20170422130115,ctsInvalidFlagValueForDirective, [copy(Src,ValStart,SrcPos-ValStart),FDirectiveName]); end; Result:=ReadNextSwitchDirective; end; function TLinkScanner.MacroDirective: boolean; var ValStart: LongInt; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkMacro; ReadSpace; ValStart:=SrcPos; while (SrcPos<=SrcLen) and (IsWordChar[Src[SrcPos]]) do inc(SrcPos); if CompareUpToken('ON',Src,ValStart,SrcPos) then FMacrosOn:=true else if CompareUpToken('OFF',Src,ValStart,SrcPos) then FMacrosOn:=false else RaiseExceptionFmt(20170422130118,ctsInvalidFlagValueForDirective, [copy(Src,ValStart,SrcPos-ValStart),FDirectiveName]); Result:=true; end; function TLinkScanner.ModeDirective: boolean; // $MODE DEFAULT, OBJFPC, TP, FPC, GPC, DELPHI var ValStart: integer; AMode: TCompilerMode; ModeValid: boolean; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkMode; ReadSpace; ValStart:=SrcPos; while (SrcPos<=SrcLen) and (IsIdentChar[Src[SrcPos]]) do inc(SrcPos); // undefine all mode macros for AMode:=Low(TCompilerMode) to High(TCompilerMode) do Values.Undefine(CompilerModeVars[AMode]); // define new mode macro if CompareUpToken('DEFAULT',Src,ValStart,SrcPos) then begin // set mode to initial mode for AMode:=Low(TCompilerMode) to High(TCompilerMode) do if FInitValues.IsDefined(CompilerModeVars[AMode]) then begin CompilerMode:=AMode; break; end; end else begin ModeValid:=false; for AMode:=Low(TCompilerMode) to High(TCompilerMode) do if CompareUpToken(CompilerModeNames[AMode],Src,ValStart,SrcPos) then begin CompilerMode:=AMode; ModeValid:=true; break; end; if not ModeValid then RaiseExceptionFmt(20170422130122,ctsInvalidMode,[copy(Src,ValStart,SrcPos-ValStart)]); end; Result:=true; end; function TLinkScanner.ModeSwitchDirective: boolean; // $MODESWITCH objectivec1 // $MODESWITCH objectivec1 on|off // $MODESWITCH systemcodepage- var p, ValStart: PChar; ModeSwitch: TCompilerModeSwitch; Switches: TCompilerModeSwitches; Enable: Boolean; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkModeSwitch; ReadSpace; p:=@Src[SrcPos]; ValStart:=p; while IsIdentChar[p^] do inc(p); Result:=false; for ModeSwitch := Succ(Low(ModeSwitch)) to High(ModeSwitch) do begin if CompareIdentifiers(@CompilerModeSwitchNames[ModeSwitch][1],ValStart)=0 then begin Result:=true; Switches:=[ModeSwitch]; Enable:=true; if p^='-' then Enable:=false else if IsSpaceChar[p^] then begin repeat inc(p); until not IsSpaceChar[p^]; if CompareIdentifiers(p,'off')=0 then Enable:=false; end; case ModeSwitch of cmsObjectiveC2: Include(Switches,cmsObjectiveC1); end; if Enable then begin FCompilerModeSwitches:=FCompilerModeSwitches+Switches; case ModeSwitch of cmsDefault_unicodestring: begin Values.Variables['FPC_UNICODESTRINGS'] := '1'; Values.Variables['UNICODE'] := '1'; end; end; end else begin FCompilerModeSwitches:=FCompilerModeSwitches-Switches; case ModeSwitch of cmsDefault_unicodestring: begin Values.Undefine('FPC_UNICODESTRINGS'); Values.Undefine('UNICODE'); end; end; end; exit; end; end; RaiseExceptionFmt(20170422130125,ctsInvalidModeSwitch,[GetIdentifier(ValStart)]); end; function TLinkScanner.ThreadingDirective: boolean; // example: {$threading on} var ValStart: integer; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkThreading; ReadSpace; ValStart:=SrcPos; while (SrcPos<=SrcLen) and (IsWordChar[Src[SrcPos]]) do inc(SrcPos); if CompareUpToken('ON',Src,ValStart,SrcPos) then begin // define Values.Variables[MacroUseSysThrds]:='1'; end else begin // undefine Values.Undefine(MacroUseSysThrds); end; Result:=true; end; function TLinkScanner.ReadNextSwitchDirective: boolean; var DirStart, DirLen: integer; begin ReadSpace; if (SrcPos<=SrcLen) and (Src[SrcPos]=',') then begin inc(SrcPos); DirStart:=SrcPos; while (SrcPos<=SrcLen) and (IsIdentChar[Src[SrcPos]]) do inc(SrcPos); DirLen:=SrcPos-DirStart; if DirLen>255 then DirLen:=255; FDirectiveName:=copy(Src,DirStart,DirLen); Result:=DoDirective(DirStart,DirLen); end else Result:=true; end; function TLinkScanner.IfdefDirective: boolean; // {$ifdef name comment} var VariableName: string; begin inc(IfLevel); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkIfdef; if FSkippingDirectives<>lssdNone then exit(true); ReadSpace; VariableName:=ReadUpperIdentifier; if (VariableName<>'') and (not Values.IsDefined(VariableName)) then SkipTillEndifElse(lssdTillElse); Result:=true; end; function TLinkScanner.IfCDirective: boolean; // {$ifc expression} or indirectly called by {$elifc expression} begin //DebugLn(['TLinkScanner.IfCDirective FSkippingDirectives=',ord(FSkippingDirectives),' IfLevel=',IfLevel]); inc(IfLevel); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkIfC; if FSkippingDirectives<>lssdNone then exit(true); Result:=InternalIfDirective; end; function TLinkScanner.ReadIdentifier: string; var StartPos: integer; begin StartPos:=SrcPos; if (SrcPos<=SrcLen) and (IsIdentStartChar[Src[SrcPos]]) then begin inc(SrcPos); while (SrcPos<=SrcLen) and (IsIdentChar[Src[SrcPos]]) do inc(SrcPos); Result:=copy(Src,StartPos,SrcPos-StartPos); end else Result:=''; end; function TLinkScanner.ReadUpperIdentifier: string; var StartPos: integer; begin StartPos:=SrcPos; if (SrcPos<=SrcLen) and (IsIdentStartChar[Src[SrcPos]]) then begin inc(SrcPos); while (SrcPos<=SrcLen) and (IsIdentChar[Src[SrcPos]]) do inc(SrcPos); Result:=UpperCaseStr(copy(Src,StartPos,SrcPos-StartPos)); end else Result:=''; end; function TLinkScanner.IfndefDirective: boolean; // {$ifndef name comment} var VariableName: string; begin inc(IfLevel); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkIfndef; if FSkippingDirectives<>lssdNone then exit(true); ReadSpace; VariableName:=ReadUpperIdentifier; if (VariableName<>'') and (Values.IsDefined(VariableName)) then SkipTillEndifElse(lssdTillElse); Result:=true; end; function TLinkScanner.EndifDirective: boolean; // {$endif comment} procedure RaiseAWithoutB; begin RaiseExceptionFmt(20170422130128,ctsAwithoutB,['$ENDIF','$IF']) end; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkEndif; if IfLevel<=0 then RaiseAWithoutB; dec(IfLevel); if (IfLevellssdNone) then begin {$IFDEF ShowUpdateCleanedSrc} debugln(['TLinkScanner.EndifDirective end skip']); {$ENDIF} EndSkipping; end; if StoreDirectives then begin FDirectives[FDirectivesCount-1].Level:=IfLevel; end; Result:=true; end; function TLinkScanner.EndCDirective: boolean; // {$endc comment} procedure RaiseAWithoutB; begin RaiseExceptionFmt(20170422130131,ctsAwithoutB,['$ENDC','$IFC']) end; begin //DebugLn(['TLinkScanner.EndCDirective FSkippingDirectives=',ord(FSkippingDirectives),' IfLevel=',IfLevel]); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkEndC; if IfLevel<=0 then RaiseAWithoutB; dec(IfLevel); if (IfLevellssdNone) then begin {$IFDEF ShowUpdateCleanedSrc} debugln(['TLinkScanner.EndCDirective end skip']); {$ENDIF} EndSkipping; end; Result:=true; end; function TLinkScanner.IfEndDirective: boolean; // {$IfEnd comment} procedure RaiseAWithoutB; begin RaiseExceptionFmt(20170422130134,ctsAwithoutB,['$IfEnd','$ElseIf']) end; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkIfEnd; if IfLevel<=0 then RaiseAWithoutB; dec(IfLevel); if (IfLevellssdNone) then begin {$IFDEF ShowUpdateCleanedSrc} debugln(['TLinkScanner.IfEndDirective end skip']); {$ENDIF} EndSkipping; end; Result:=true; end; function TLinkScanner.ElseDirective: boolean; // {$else comment} procedure RaiseAWithoutB; begin RaiseExceptionFmt(20170422130138,ctsAwithoutB,['$ELSE','$IF']); end; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkElse; if IfLevel=0 then RaiseAWithoutB; case FSkippingDirectives of lssdNone: // last block was executed, skip all other SkipTillEndifElse(lssdTillEndIf); lssdTillElse: if IfLevel=FSkipIfLevel then begin {$IFDEF ShowUpdateCleanedSrc} debugln(['TLinkScanner.ElseDirective skipped front, using ELSE part']); {$ENDIF} EndSkipping; end; end; Result:=true; end; function TLinkScanner.ElseCDirective: boolean; // {$elsec comment} procedure RaiseAWithoutB; begin RaiseExceptionFmt(20170422130140,ctsAwithoutB,['$ELSEC','$IFC']); end; begin //DebugLn(['TLinkScanner.ElseCDirective FSkippingDirectives=',ord(FSkippingDirectives),' IfLevel=',IfLevel]); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkElseC; if IfLevel=0 then RaiseAWithoutB; case FSkippingDirectives of lssdNone: // last block was executed, skip all other SkipTillEndifElse(lssdTillEndIf); lssdTillElse: if IfLevel=FSkipIfLevel then begin {$IFDEF ShowUpdateCleanedSrc} debugln(['TLinkScanner.ElseCDirective skipped front, using ELSEC part']); {$ENDIF} EndSkipping; end; end; Result:=true; end; function TLinkScanner.ElseIfDirective: boolean; // {$elseif expression} procedure RaiseAWithoutB; begin RaiseExceptionFmt(20170422130143,ctsAwithoutB,['$ELSEIF','$IF']); end; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkElseIf; if IfLevel=0 then RaiseAWithoutB; case FSkippingDirectives of lssdNone: // last block was executed, skip all other SkipTillEndifElse(lssdTillEndIf); lssdTillElse: if IfLevel=FSkipIfLevel then exit(InternalIfDirective); end; Result:=true; end; function TLinkScanner.ElIfCDirective: boolean; // {$elifc expression} procedure RaiseAWithoutB; begin RaiseExceptionFmt(20170422130146,ctsAwithoutB,['$ELIFC','$IFC']); end; begin //DebugLn(['TLinkScanner.ElIfCDirective FSkippingDirectives=',ord(FSkippingDirectives),' IfLevel=',IfLevel]); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkElseC; if IfLevel=0 then RaiseAWithoutB; case FSkippingDirectives of lssdNone: // last block was executed, skip all other SkipTillEndifElse(lssdTillEndIf); lssdTillElse: if IfLevel=FSkipIfLevel then exit(InternalIfDirective); end; Result:=true; end; function TLinkScanner.DefineDirective: boolean; // {$define name} or {$define name:=value} var VariableName, NewValue: string; NamePos: LongInt; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkDefine; ReadSpace; NamePos:=SrcPos; VariableName:=ReadUpperIdentifier; if (VariableName<>'') then begin ReadSpace; if FMacrosOn and (SrcPos store the value inc(SrcPos,2); ReadSpace; NewValue:=copy(Src,SrcPos,CommentInnerEndPos-SrcPos); if CompareIdentifiers(PChar(NewValue),'false')=0 then NewValue:='0' else if CompareIdentifiers(PChar(NewValue),'true')=0 then NewValue:='1'; Values.Variables[VariableName]:=NewValue; AddMacroValue(@Src[NamePos],SrcPos,CommentInnerEndPos); end else begin // flag Values.Variables[VariableName]:='1'; end; end; Result:=true; end; function TLinkScanner.UndefDirective: boolean; // {$undefine name} var VariableName: string; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkUndef; ReadSpace; VariableName:=ReadUpperIdentifier; if (VariableName<>'') then Values.Undefine(VariableName); Result:=true; end; function TLinkScanner.SetCDirective: boolean; // {$setc name} or {$setc name:=value} var VariableName, NewValue: string; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkSetC; ReadSpace; VariableName:=ReadUpperIdentifier; if (VariableName<>'') then begin ReadSpace; if FMacrosOn and (SrcPoslength(FCleanedSrc)-CleanedLen then begin // expand cleaned source string by at least 1024 SetLength(FCleanedSrc,length(FCleanedSrc)+1024); end; AddLink(1,nil,slkCompilerString); inc(CleanedLen); FCleanedSrc[CleanedLen]:=''''; inc(CleanedLen); FCleanedSrc[CleanedLen]:=''''; // continue after directive CopiedSrcPos:=CommentEndPos-1; AddLink(CommentEndPos,Code); end else begin IncFilename:=Trim(copy(Src,SrcPos,CommentInnerEndPos-SrcPos)); if (IncFilename<>'') and (IncFilename[1]='''') then begin if (IncFilename[length(IncFilename)]='''') then IncFilename:=copy(IncFilename,2,length(IncFilename)-2) else begin SrcPos:=CommentInnerEndPos; RaiseException(20170422130149,'missing '''); end; end; ForcePathDelims(IncFilename); {$IFDEF ShowUpdateCleanedSrc} DebugLn('TLinkScanner.IncludeDirective A IncFilename="',IncFilename,'" UpdatePos=',DbgS(CommentEndPos-1)); {$ENDIF} UpdateCleanedSource(CommentEndPos-1); // put old position on stack PushIncludeLink(CleanedLen,CommentEndPos,Code); // load include file Result:=IncludeFile(IncFilename); if Result then begin if (SrcPos<=SrcLen) then CommentEndPos:=SrcPos else ReturnFromIncludeFile; end else begin PopIncludeLink; end; end; //DebugLn('[TLinkScanner.IncludeDirective] END ',CommentEndPos,',',SrcPos,',',SrcLen); end; function TLinkScanner.IncludePathDirective: boolean; // {$includepath path_addition} var AddPath, PathDivider: string; begin if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkIncludePath; inc(SrcPos); AddPath:=Trim(copy(Src,SrcPos,CommentInnerEndPos-SrcPos)); PathDivider:=':'; Values.Variables[ExternalMacroStart+'INCPATH']:= Values.Variables[ExternalMacroStart+'INCPATH']+PathDivider+AddPath; Result:=true; end; function TLinkScanner.LoadSourceCaseLoUp(const AFilename: string; AllowVirtual: boolean): pointer; var Path, FileNameOnly: string; SecondaryFileNameOnly: String; begin {$IFDEF VerboseIncludeSearch} debugln(['TLinkScanner.LoadSourceCaseLoUp AFilename="',AFilename,'" AllowVirtual=',AllowVirtual]); {$ENDIF} Result:=nil; Path:=ResolveDots(ExtractFilePath(AFilename)); if (not AllowVirtual) and (Path<>'') and (not FilenameIsAbsolute(Path)) then exit; FileNameOnly:=ExtractFilename(AFilename); Result:=FOnLoadSource(Self,Path+FileNameOnly,true); if (Result<>nil) then exit; SecondaryFileNameOnly:=LowerCase(FileNameOnly); if (SecondaryFileNameOnly<>FileNameOnly) then begin Result:=FOnLoadSource(Self,Path+SecondaryFileNameOnly,true); if (Result<>nil) then exit; end; SecondaryFileNameOnly:=UpperCaseStr(FileNameOnly); if (SecondaryFileNameOnly<>FileNameOnly) then begin Result:=FOnLoadSource(Self,Path+SecondaryFileNameOnly,true); if (Result<>nil) then exit; end; end; function TLinkScanner.SearchIncludeFile(AFilename: string; out NewCode: Pointer; var MissingIncludeFile: TMissingIncludeFile): boolean; var PathStart, PathEnd: integer; IncludePath, CurPath: string; ExpFilename: string; HasPathDelims: Boolean; function SearchPath(const APath, RelFilename: string): boolean; begin Result:=false; if APath='' then exit; {$IFDEF VerboseIncludeSearch} DebugLn('TLinkScanner.SearchPath CurIncPath="',APath,'" / "',RelFilename,'"'); {$ENDIF} ExpFilename:=AppendPathDelim(APath)+RelFilename; if not FilenameIsAbsolute(ExpFilename) then ExpFilename:=ExtractFilePath(FMainSourceFilename)+ExpFilename; NewCode:=LoadSourceCaseLoUp(ExpFilename); Result:=NewCode<>nil; end; procedure SetMissingIncludeFile; begin if MissingIncludeFile=nil then MissingIncludeFile:=TMissingIncludeFile.Create(AFilename,''); MissingIncludeFile.IncludePath:=IncludePath; end; function SearchCasedInIncPath(const RelFilename: string): boolean; begin if FilenameIsAbsolute(FMainSourceFilename) then begin // main source has absolute filename // search in directory of unit ExpFilename:=ExtractFilePath(FMainSourceFilename)+RelFilename; NewCode:=LoadSourceCaseLoUp(ExpFilename); Result:=(NewCode<>nil); if Result then exit; // search in directory of source of include directive if FilenameIsAbsolute(SrcFilename) and (CompareFilenames(SrcFilename,FMainSourceFilename)<>0) then begin ExpFilename:=ExtractFilePath(SrcFilename)+RelFilename; NewCode:=LoadSourceCaseLoUp(ExpFilename); Result:=(NewCode<>nil); if Result then exit; end; end else begin // main source is virtual -> allow virtual include file NewCode:=LoadSourceCaseLoUp(RelFilename,true); Result:=(NewCode<>nil); if Result then exit; end; // then search the include file in directories defines in fpc.cfg (by -Fi option) if FindIncFileInCfgCache(AFilename,ExpFilename) then begin NewCode:=LoadSourceCaseLoUp(ExpFilename); Result:=(NewCode<>nil); if Result then exit; end; // then search the include file in the include path if not HasPathDelims then begin if MissingIncludeFile=nil then IncludePath:=Values.Variables[ExternalMacroStart+'INCPATH'] else IncludePath:=MissingIncludeFile.IncludePath; {$IFDEF VerboseIncludeSearch} DebugLn('TLinkScanner.SearchIncludeFile IncPath="',IncludePath,'"'); {$ENDIF} PathStart:=1; PathEnd:=PathStart; while PathEnd<=length(IncludePath) do begin if IncludePath[PathEnd]=';' then begin if PathEnd>PathStart then begin CurPath:=TrimFilename(copy(IncludePath,PathStart,PathEnd-PathStart)); Result:=SearchPath(CurPath,RelFilename); if Result then exit; end; PathStart:=PathEnd+1; PathEnd:=PathStart; end else inc(PathEnd); end; if PathEnd>PathStart then begin CurPath:=TrimFilename(copy(IncludePath,PathStart,PathEnd-PathStart)); Result:=SearchPath(CurPath,RelFilename); if Result then exit; end; end; Result:=false; end; begin {$IFDEF VerboseIncludeSearch} DebugLn('TLinkScanner.SearchIncludeFile Filename="',AFilename,'"'); {$ENDIF} NewCode:=nil; IncludePath:=''; // beware of 'dir/file.inc' HasPathDelims:=(System.Pos('/',AFilename)>0) or (System.Pos('\',AFilename)>0); if HasPathDelims then ForcePathDelims(AFilename); AFilename:=ResolveDots(AFilename); if not Assigned(FOnLoadSource) then begin NewCode:=nil; SetMissingIncludeFile; Result:=false; exit; end; // if include filename is absolute then load it directly if FilenameIsAbsolute(AFilename) then begin NewCode:=LoadSourceCaseLoUp(AFilename); Result:=(NewCode<>nil); if not Result then SetMissingIncludeFile; exit; end; // first search without touching the extension {$IFDEF VerboseIncludeSearch} debugln(['TLinkScanner.SearchIncludeFile FMainSourceFilename="',FMainSourceFilename,'" SrcFile="',SrcFilename,'" AFilename="',AFilename,'"']); {$ENDIF} if SearchCasedInIncPath(AFilename) then exit(true); if ExtractFileExt(AFilename)='' then begin // search with the default file extensions if SearchCasedInIncPath(AFilename+'.inc') then exit(true); if SearchCasedInIncPath(AFilename+'.pp') then exit(true); if SearchCasedInIncPath(AFilename+'.pas') then exit(true); end; SetMissingIncludeFile; Result:=false; end; function TLinkScanner.IncludeFile(const AFilename: string): boolean; var NewCode: Pointer; MissingIncludeFile: TMissingIncludeFile; begin MissingIncludeFile:=nil; Result:=SearchIncludeFile(AFilename,NewCode,MissingIncludeFile); if Result then begin // change source if Assigned(FOnIncludeCode) then FOnIncludeCode(FMainCode,NewCode); SetSource(NewCode); AddLink(SrcPos,Code); end else begin if MissingIncludeFile<>nil then begin if FMissingIncludeFiles=nil then FMissingIncludeFiles:=TMissingIncludeFiles.Create; FMissingIncludeFiles.Add(MissingIncludeFile); end; if (not IgnoreMissingIncludeFiles) then begin // ToDo: add an event to let application improve the error message RaiseExceptionFmt(20170422130152,ctsIncludeFileNotFound,[AFilename]) end else begin // add a dummy link AddLink(SrcPos,nil,slkMissingIncludeFile); AddLink(SrcPos,Code); end; end; end; function TLinkScanner.IfDirective: boolean; // {$if expression} or indirectly called by {$elseif expression} begin inc(IfLevel); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkIf; if FSkippingDirectives<>lssdNone then exit(true); Result:=InternalIfDirective; end; function TLinkScanner.IfOptDirective: boolean; // {$ifopt o+} or {$ifopt o-} var Option, c: char; begin inc(IfLevel); if StoreDirectives then FDirectives[FDirectivesCount-1].Kind:=lsdkIfOpt; if FSkippingDirectives<>lssdNone then exit(true); Result:=true; inc(SrcPos); Option:=UpChars[Src[SrcPos]]; if (IsWordChar[Option]) and (CompilerSwitchesNames[Option]<>'') then begin inc(SrcPos); if (SrcPos<=SrcLen) then begin c:=Src[SrcPos]; if c in ['+','-'] then begin if (c='-')<>(Values.Variables[CompilerSwitchesNames[Option]]='0') then begin SkipTillEndifElse(lssdTillElse); exit; end; end; end; end; end; procedure TLinkScanner.SetIgnoreMissingIncludeFiles(const Value: boolean); begin if Value then Include(FStates,lssIgnoreMissingIncludeFiles) else Exclude(FStates,lssIgnoreMissingIncludeFiles); end; procedure TLinkScanner.PushIncludeLink(ACleanedPos, ASrcPos: integer; ACode: Pointer); procedure RaiseIncludeCircleDetected; begin RaiseException(20170422130154,ctsIncludeCircleDetected); end; var NewLink: PSourceLink; i: integer; begin for i:=0 to FIncludeStack.Count-1 do if PSourceLink(FIncludeStack[i])^.Code=ACode then RaiseIncludeCircleDetected; NewLink:=PSourceLinkMemManager.NewPSourceLink; with NewLink^ do begin CleanedPos:=ACleanedPos; SrcPos:=ASrcPos; Code:=ACode; end; FIncludeStack.Add(NewLink); end; function TLinkScanner.PopIncludeLink: TSourceLink; var PLink: PSourceLink; begin PLink:=PSourceLink(FIncludeStack[FIncludeStack.Count-1]); Result:=PLink^; PSourceLinkMemManager.DisposePSourceLink(PLink); FIncludeStack.Delete(FIncludeStack.Count-1); end; function TLinkScanner.GetIncludeFileIsMissing: boolean; begin Result:=(FMissingIncludeFiles<>nil); end; function TLinkScanner.MissingIncludeFilesNeedsUpdate: boolean; var i: integer; MissingIncludeFile: TMissingIncludeFile; NewCode: Pointer; begin Result:=false; if (not IncludeFileIsMissing) or IgnoreMissingIncludeFiles then exit; { last scan missed an include file (i.e. was not in searchpath) -> Check all missing include files again } for i:=0 to FMissingIncludeFiles.Count-1 do begin MissingIncludeFile:=FMissingIncludeFiles[i]; if SearchIncludeFile(MissingIncludeFile.Filename,NewCode,MissingIncludeFile) then begin Result:=true; exit; end; end; end; procedure TLinkScanner.ClearMissingIncludeFiles; begin FreeAndNil(FMissingIncludeFiles); end; procedure TLinkScanner.AddMacroValue(MacroName: PChar; ValueStart, ValueEnd: integer); var i: LongInt; Macro: PSourceLinkMacro; begin i:=IndexOfMacro(MacroName,false); if i<0 then begin // insert new macro i:=IndexOfMacro(MacroName,true); if FMacroCount=fMacroCapacity then begin fMacroCapacity:=fMacroCapacity*2; if fMacroCapacity<4 then fMacroCapacity:=4; ReAllocMem(FMacros,SizeOf(TSourceLinkMacro)*fMacroCapacity); end; if i0 then l:=m+1 else begin Result:=m; exit; end; end; if InsertPos then begin if cmp>0 then inc(m); Result:=m; end else begin Result:=-1; end; end; procedure TLinkScanner.AddMacroSource(MacroID: integer); var Macro: PSourceLinkMacro; OldCode: Pointer; OldSrc: String; OldSrcFilename: String; begin Macro:=@FMacros[MacroID]; //DebugLn(['TLinkScanner.AddMacroSource ID=',MacroID,' ',GetIdentifier(Macro^.Name)]); // update cleaned source UpdateCleanedSource(TokenStart-1); // store old code pos OldCode:=Code; OldSrc:=Src; OldSrcFilename:=SrcFilename; //DebugLn(['TLinkScanner.AddMacroSource BEFORE CleanedSrc=',dbgstr(copy(FCleanedSrc,CleanedLen-19,20))]); // add macro source AddLink(Macro^.StartPos,Macro^.Code); Code:=Macro^.Code; Src:=Macro^.Src; SrcLen:=length(Src); SrcFilename:=Macro^.SrcFilename; CopiedSrcPos:=Macro^.StartPos-1; UpdateCleanedSource(Macro^.EndPos-1); //DebugLn(['TLinkScanner.AddMacroSource MACRO CleanedSrc=',dbgstr(copy(FCleanedSrc,CleanedLen-19,20))]); // restore code pos Code:=OldCode; Src:=OldSrc; SrcLen:=length(Src); SrcFilename:=OldSrcFilename; CopiedSrcPos:=SrcPos-1; AddLink(SrcPos,Code); // clear token type TokenType:=lsttNone; // SrcPos was not touched and still stands behind the macro name //DebugLn(['TLinkScanner.AddMacroSource END Token=',copy(Src,TokenStart,SrcPos-TokenStart)]); end; procedure TLinkScanner.AddSkipComment(IsStart: boolean); begin //DebugLn(['TLinkScanner.AddSkipComment InFront="',dbgstr(CleanedSrc,CleanedLen-12,13),'" isstart=',IsStart]); // insert {#3 or #3} if 2>length(FCleanedSrc)-CleanedLen then begin // expand cleaned source string by at least 1024 SetLength(FCleanedSrc,length(FCleanedSrc)+1024); end; if IsStart then begin AddLink(1,nil,slkSkipStart); inc(CleanedLen); FCleanedSrc[CleanedLen]:='{'; inc(CleanedLen); FCleanedSrc[CleanedLen]:=#3; end else begin if (LinkCount>0) and (FLinks[FLinkCount-1].CleanedPos=CleanedLen+1) then begin // last link is empty dec(FLinkCount); {$IFDEF ShowUpdateCleanedSrc} debugln(['TLinkScanner.AddSkipComment SkipEnd: removing empty link: ',dbgs(FLinks[FLinkCount].Kind)]); {$ENDIF} if (LinkCount>0) and (FLinks[FLinkCount-1].Kind=slkSkipStart) then begin // remove unneeded SkipStart dec(FLinkCount); CleanedLen:=FLinks[FLinkCount].CleanedPos; exit; end; end; AddLink(1,nil,slkSkipEnd); inc(CleanedLen); FCleanedSrc[CleanedLen]:=#3; inc(CleanedLen); FCleanedSrc[CleanedLen]:='}'; end; // SrcPos was not touched and still stands at the same position //DebugLn(['TLinkScanner.AddSkipComment END']); end; function TLinkScanner.ReturnFromIncludeFile: boolean; var OldPos: TSourceLink; begin if (SrcPos-1>CopiedSrcPos) then begin {$IFDEF ShowUpdateCleanedSrc} DebugLn('TLinkScanner.ReturnFromIncludeFile A UpdatePos=',DbgS(SrcPos-1)); {$ENDIF} UpdateCleanedSource(SrcPos-1); end; while SrcPos>SrcLen do begin Result:=FIncludeStack.Count>0; if not Result then exit; OldPos:=PopIncludeLink; SetSource(OldPos.Code); SrcPos:=OldPos.SrcPos; CopiedSrcPos:=SrcPos-1; AddLink(SrcPos,Code); end; Result:=SrcPos<=SrcLen; end; function TLinkScanner.ParseKeyWord(StartPos: integer; LastTokenType: TLSTokenType): boolean; var p: PChar; begin if StartPos>SrcLen then exit(false); p:=@Src[StartPos]; //writeln('TLinkScanner.ParseKeyWord ',copy(Src,StartPos,WordLen)); case UpChars[p^] of 'E': if CompareIdentifiers(p,'END')=0 then exit(DoEndToken); 'F': if CompareIdentifiers(p,'FINALIZATION')=0 then exit(DoFinalizationToken); 'I': case UpChars[p[1]] of 'M': if CompareIdentifiers(p,'IMPLEMENTATION')=0 then exit(DoImplementationToken); 'N': case UpChars[p[2]] of 'I': if CompareIdentifiers(p,'INITIALIZATION')=0 then exit(DoInitializationToken); 'T': if (LastTokenType<>lsttEqual) and (CompareIdentifiers(p,'INTERFACE')=0) then exit(DoInterfaceToken); end; end; 'L': if CompareIdentifiers(p,'LIBRARY')=0 then exit(DoSourceTypeToken); 'P': case UpChars[p[1]] of 'R': if CompareIdentifiers(p,'PROGRAM')=0 then exit(DoSourceTypeToken); 'A': if CompareIdentifiers(p,'PACKAGE')=0 then exit(DoSourceTypeToken); end; 'U': case UpChars[p[1]] of 'N': if CompareIdentifiers(p,'UNIT')=0 then exit(DoSourceTypeToken); 'S': if CompareIdentifiers(p,'USES')=0 then exit(DoUsesToken); end; end; Result:=false; end; procedure TLinkScanner.SkipTillEndifElse(SkippingUntil: TLSSkippingDirective); procedure RaiseAlreadySkipping; begin raise Exception.Create('TLinkScanner.SkipTillEndifElse inconsistency: already skipping ' +' Old='+dbgs(ord(FSkippingDirectives)) +' New='+dbgs(ord(SkippingUntil))); end; var p: PChar; begin if FDirectivesCount>0 then FDirectives[FDirectivesCount-1].State:=lsdsInactive; if FSkippingDirectives<>lssdNone then begin FSkippingDirectives:=SkippingUntil; exit; end; FSkippingDirectives:=SkippingUntil; SrcPos:=CommentEndPos; {$IFDEF ShowUpdateCleanedSrc} DebugLn('TLinkScanner.SkipTillEndifElse A UpdatePos=',DbgS(SrcPos-1),' Src=',DbgStr(Src,SrcPos-15,15)+'|'+DbgStr(Src,SrcPos,15)); {$ENDIF} UpdateCleanedSource(SrcPos-1); AddSkipComment(true); AddLink(SrcPos,Code); // parse till $else, $elseif or $endif FSkipIfLevel:=IfLevel; if (SrcPos<=SrcLen) then begin p:=@Src[SrcPos]; while true do begin case p^ of '{': begin SrcPos:=p-PChar(Src)+1; ReadCurlyComment; if (FSkippingDirectives=lssdNone) or (SrcPos>SrcLen) then break; p:=@Src[SrcPos]; end; '/': if p[1]='/' then begin SrcPos:=p-PChar(Src)+1; ReadLineComment; if (FSkippingDirectives=lssdNone) or (SrcPos>SrcLen) then break; p:=@Src[SrcPos]; end else inc(p); '(': if p[1]='*' then begin SrcPos:=p-PChar(Src)+1; ReadRoundComment; if (FSkippingDirectives=lssdNone) or (SrcPos>SrcLen) then break; p:=@Src[SrcPos]; end else inc(p); '''': begin // skip string constant inc(p); while not (p^ in ['''',#0,#10,#13]) do inc(p); if p^='''' then inc(p); end; #0: begin // FPC allows that corresponding IFDEF and ENDIF are in different files SrcPos:=p-PChar(Src)+1; if (SrcPos>SrcLen) then begin if not ReturnFromIncludeFile then begin CopiedSrcPos:=SrcLen+1; break; end; p:=@Src[SrcPos]; end else inc(p); end; else inc(p); end; end; SrcPos:=p-PChar(Src)+1; end else begin CopiedSrcPos:=SrcLen+1; end; {$IFDEF ShowUpdateCleanedSrc} DebugLn('TLinkScanner.SkipTillEndifElse B Continuing after: ', ' Src=',DbgStr(copy(Src,CommentStartPos-15,15))+'|'+DbgStr(copy(Src,CommentStartPos,15))); {$ENDIF} FSkippingDirectives:=lssdNone; end; procedure TLinkScanner.SetCompilerMode(const AValue: TCompilerMode); var OldModeSwitches, EnabledModeSwitches, DisabledModeSwitches: TCompilerModeSwitches; begin if FCompilerMode=AValue then exit; Values.Undefine(CompilerModeVars[FCompilerMode]); FCompilerMode:=AValue; OldModeSwitches:=FCompilerModeSwitches; FCompilerModeSwitches:=DefaultCompilerModeSwitches[CompilerMode]; FNestedComments:=cmsNested_comment in CompilerModeSwitches; Values.Variables[CompilerModeVars[FCompilerMode]]:='1'; EnabledModeSwitches:=FCompilerModeSwitches-OldModeSwitches; DisabledModeSwitches:=OldModeSwitches-FCompilerModeSwitches; if cmsDefault_unicodestring in EnabledModeSwitches then begin Values.Variables['FPC_UNICODESTRINGS'] := '1'; Values.Variables['UNICODE'] := '1'; end else if cmsDefault_unicodestring in DisabledModeSwitches then begin Values.Undefine('FPC_UNICODESTRINGS'); Values.Undefine('UNICODE'); end; end; procedure TLinkScanner.SetPascalCompiler(const AValue: TPascalCompiler); begin if FPascalCompiler=AValue then Exit; FPascalCompiler:=AValue; case PascalCompiler of pcPas2js: ; end; end; procedure TLinkScanner.SetDirectiveValueWithSequence( ADirective: TSequenceDirective; const ADirectiveValue: string); begin Values.Variables[DirectiveSequenceName[ADirective]] := ADirectiveValue; FDirectiveSequence.Add(ADirective, ADirectiveValue, FDirectiveCleanPos); end; function TLinkScanner.GetIgnoreMissingIncludeFiles: boolean; begin Result:=lssIgnoreMissingIncludeFiles in FStates; end; function TLinkScanner.InternalIfDirective: boolean; // {$if expression} or {$ifc expression} // or indirectly called by {$elifc expression} or {$elseif expression} procedure RaiseMissingExpr; begin RaiseException(20170422130156,'missing expression'); end; var ExprResult: Boolean; begin //DebugLn(['TLinkScanner.InternalIfDirective FSkippingDirectives=',ord(FSkippingDirectives),' IfLevel=',IfLevel]); inc(SrcPos); if SrcPos>SrcLen then RaiseMissingExpr; ExprResult:=Values.EvalBoolean(@Src[SrcPos],CommentInnerEndPos-SrcPos); Result:=true; //DebugLn(['TLinkScanner.InternalIfDirective ExprResult=',ExprResult]); if Values.ErrorPosition>=0 then begin inc(SrcPos,Values.ErrorPosition); RaiseException(20170422130200,Values.ErrorMsg) end else if ExprResult then begin // expression evaluates to true => stop skipping and parse block if FDirectivesCount>0 then FDirectives[FDirectivesCount-1].State:=lsdsActive; if FSkippingDirectives<>lssdNone then begin {$IFDEF ShowUpdateCleanedSrc} debugln(['TLinkScanner.InternalIfDirective skipped front, using ELIFC part']); {$ENDIF} EndSkipping; end; end else begin // expression evaluates to false => skip this block SkipTillEndifElse(lssdTillElse); end; end; procedure TLinkScanner.EndSkipping; procedure ErrorNotSkipping; begin debugln(['ErrorNotSkipping internal error, please report this bug']); CTDumpStack; end; var Dir: PLSDirective; begin if FSkippingDirectives=lssdNone then begin ErrorNotSkipping; exit; end; FSkippingDirectives:=lssdNone; UpdateCleanedSource(CommentStartPos-1); AddSkipComment(false); AddLink(CommentStartPos,Code); //debugln(['TLinkScanner.EndSkipping ',StoreDirectives]); if StoreDirectives then begin // update cleaned position of directive Dir:=@FDirectives[FDirectivesCount-1]; Dir^.CleanPos:=CleanedLen+1; Dir^.State:=lsdsActive; end; FSkipIfLevel:=-1; end; function TLinkScanner.CursorToCleanPos(ACursorPos: integer; ACode: pointer; out ACleanPos: integer): integer; // 0=valid CleanPos //-1=CursorPos was skipped, CleanPos is between two links // 1=CursorPos beyond scanned code // ACleanPos starts at 1 type TLinkQuality = (qNone,qSkipped,qSkippedInFront,qDisabled,qClean); var i, BestCleanPos: integer; Link: PSourceLink; BestQuality: TLinkQuality; LinkEnd: Integer; Enabled: Boolean; begin Result:=1; ACleanPos:=0; if ACode=nil then exit; i:=0; BestQuality:=qNone; BestCleanPos:=0; Enabled:=true; while i save position and continue search if BestQualitynil; end; ACursorPos:=ACleanedPos-FLinks[LinkID].CleanedPos+FLinks[LinkID].SrcPos; end; var l,r,m: integer; begin Result:=(ACleanedPos>=1) and (ACleanedPos<=CleanedLen); if Result then begin // ACleanedPos in Cleaned Code -> binary search through the links l:=0; r:=LinkCount-1; while l<=r do begin m:=(l+r) div 2; if m=FLinks[m+1].CleanedPos then l:=m+1 else begin Found(m); exit; end; end else begin if ACleanedPos>=FLinks[m].CleanedPos then begin Found(m); exit; end else ConsistencyCheckI(2); end; end; ConsistencyCheckI(1); end; ACode:=nil; ACursorPos:=0; end; function TLinkScanner.CleanedPosToStr(ACleanedPos: integer): string; var p: integer; ACode: Pointer; begin if CleanedPosToCursor(ACleanedPos,p,ACode) then begin Result:=TSourceLog(ACode).AbsoluteToLineColStr(p); end else begin Result:='p='+IntToStr(ACleanedPos)+',y=?,x=?'; end; end; function TLinkScanner.WholeRangeIsWritable(CleanStartPos, CleanEndPos: integer; ErrorOnFail: boolean): boolean; procedure EditError(const AMessage: string; ACode: Pointer); begin if ErrorOnFail then RaiseEditException(20170422130202,AMessage,ACode,0); end; var ACode: Pointer; LinkIndex: integer; CodeIsReadOnly: boolean; begin Result:=false; if (CleanStartPos<1) or (CleanStartPos>=CleanEndPos) or (CleanEndPos>CleanedLen+1) or (not Assigned(FOnGetSourceStatus)) then begin EditError('TLinkScanner.WholeRangeIsWritable: Invalid range',nil); exit; end; LinkIndex:=LinkIndexAtCleanPos(CleanStartPos); if LinkIndex<0 then begin EditError('TLinkScanner.WholeRangeIsWritable: position out of scan range',nil); exit; end; ACode:=FLinks[LinkIndex].Code; CodeIsReadOnly:=false; FOnGetSourceStatus(Self,ACode,CodeIsReadOnly); if CodeIsReadOnly then begin EditError(ctsfileIsReadOnly, ACode); exit; end; repeat inc(LinkIndex); if (LinkIndex>=LinkCount) or (FLinks[LinkIndex].CleanedPos>CleanEndPos) then begin Result:=true; exit; end; if ACode<>FLinks[LinkIndex].Code then begin ACode:=FLinks[LinkIndex].Code; if ACode<>nil then begin FOnGetSourceStatus(Self,ACode,CodeIsReadOnly); if CodeIsReadOnly then begin EditError(ctsfileIsReadOnly, ACode); exit; end; end; end; until false; end; procedure TLinkScanner.FindCodeInRange(CleanStartPos, CleanEndPos: integer; UniqueSortedCodeList: TFPList); var ACode: Pointer; LinkIndex: integer; Link: PSourceLink; begin if (CleanStartPos<1) or (CleanStartPos>CleanEndPos) or (CleanEndPos>CleanedLen+1) or (UniqueSortedCodeList=nil) then exit; LinkIndex:=LinkIndexAtCleanPos(CleanStartPos); if LinkIndex<0 then exit; ACode:=FLinks[LinkIndex].Code; if ACode<>nil then AddCodeToUniqueList(ACode,UniqueSortedCodeList); repeat inc(LinkIndex); if (LinkIndex>=LinkCount) then exit; Link:=@FLinks[LinkIndex]; if (Link^.CleanedPos>CleanEndPos) then exit; if (ACode<>Link^.Code) then begin ACode:=Link^.Code; if ACode<>nil then AddCodeToUniqueList(ACode,UniqueSortedCodeList); end; until false; end; procedure TLinkScanner.DeleteRange(CleanStartPos,CleanEndPos: integer); { delete all code in links (=parsed code) starting with the last link before you call this, test with WholeRangeIsWritable this can do unexpected things if - include files are included twice - compiler directives like IFDEF - ENDIF are partially destroyed ToDo: keep include directives } var LinkIndex, StartPos, Len, aLinkSize: integer; ACode: Pointer; begin if CleanStartPos<1 then CleanStartPos:=1; if CleanEndPos>CleanedLen then CleanEndPos:=CleanedLen+1; if (CleanStartPos>=CleanEndPos) or (not Assigned(FOnDeleteSource)) then exit; LinkIndex:=LinkIndexAtCleanPos(CleanEndPos-1); //debugln(['TLinkScanner.DeleteRange CleanStartPos=',CleanStartPos,' CleanEndPos=',CleanEndPos,' LinkIndex=',LinkIndex]); while LinkIndex>=0 do begin StartPos:=CleanStartPos-FLinks[LinkIndex].CleanedPos; if StartPos<0 then StartPos:=0; aLinkSize:=LinkSize(LinkIndex); //debugln(['TLinkScanner.DeleteRange LinkIndex=',LinkIndex,' aLinkSize=',aLinkSize,' StartPosInLink=',StartPos]); Len:=CleanEndPos-FLinks[LinkIndex].CleanedPos; if Len>aLinkSize then Len:=aLinkSize; dec(Len,StartPos); inc(StartPos,FLinks[LinkIndex].SrcPos); //DebugLn(['[TLinkScanner.DeleteRange] Pos=',StartPos,'-',StartPos+Len,' ',dbgstr(copy(Src,StartPos,Len))]); ACode:=FLinks[LinkIndex].Code; if ACode<>nil then FOnDeleteSource(Self,ACode,StartPos,Len); if FLinks[LinkIndex].CleanedPos<=CleanStartPos then break; dec(LinkIndex); end; end; procedure TLinkScanner.ActivateGlobalWriteLock; begin if Assigned(OnSetGlobalWriteLock) then OnSetGlobalWriteLock(true); end; procedure TLinkScanner.DeactivateGlobalWriteLock; begin if Assigned(OnSetGlobalWriteLock) then OnSetGlobalWriteLock(false); end; procedure TLinkScanner.RaiseExceptionFmt(id: int64; const AMessage: string; const Args: array of const); begin RaiseException(id,Format(AMessage,args)); end; procedure TLinkScanner.RaiseException(id: int64; const AMessage: string); begin RaiseExceptionClass(id,AMessage,ELinkScannerError); end; procedure TLinkScanner.RaiseExceptionClass(id: int64; const AMessage: string; ExceptionClass: ELinkScannerErrors); begin LastErrorMessage:=AMessage; LastErrorSrcPos:=SrcPos; LastErrorCode:=Code; LastErrorCheckedForIgnored:=false; LastErrorId:=id; LastErrorIsValid:=true; raise ExceptionClass.Create(Self,id,AMessage); end; procedure TLinkScanner.RaiseEditException(id: int64; const AMessage: string; ABuffer: Pointer; ABufferPos: integer); begin raise ELinkScannerEditError.Create(Self,id,AMessage,ABuffer,ABufferPos); end; procedure TLinkScanner.RaiseConsistencyException(id: int64; const AMessage: string); begin RaiseExceptionClass(id,AMessage,ELinkScannerConsistency); end; procedure TLinkScanner.ClearLastError; begin LastErrorIsValid:=false; LastErrorCheckedForIgnored:=false; end; procedure TLinkScanner.RaiseLastError; begin SrcPos:=LastErrorSrcPos; Code:=LastErrorCode; RaiseException(20170422130205,LastErrorMessage); end; procedure TLinkScanner.DoCheckAbort; begin if not Assigned(OnProgress) then exit; if OnProgress(Self) then exit; // raise abort exception RaiseExceptionClass(20170422130207,'Abort',ELinkScannerAbort); end; function TLinkScanner.MainFilename: string; begin if Assigned(OnGetFileName) and (FMainCode<>nil) then Result:=OnGetFileName(Self,FMainCode) else Result:=''; end; { ELinkScannerError } constructor ELinkScannerError.Create(ASender: TLinkScanner; TheId: int64; const AMessage: string); begin inherited Create(AMessage); Sender:=ASender; id:=TheId; end; { TPSourceLinkMemManager } procedure TPSourceLinkMemManager.FreeFirstItem; var Link: PSourceLink; begin Link:=PSourceLink(FFirstFree); PSourceLink(FFirstFree):=Link^.Next; Dispose(Link); end; procedure TPSourceLinkMemManager.DisposePSourceLink(Link: PSourceLink); begin if (FFreeCount free Link Dispose(Link); {$IFDEF DebugCTMemManager} inc(FFreedCount); {$ENDIF} end; dec(FCount); end; function TPSourceLinkMemManager.NewPSourceLink: PSourceLink; begin if FFirstFree<>nil then begin // take from free list Result:=PSourceLink(FFirstFree); PSourceLink(FFirstFree):=Result^.Next; Result^.Next:=nil; dec(FFreeCount); end else begin // free list empty -> create new PSourceLink New(Result); FillChar(Result^,SizeOf(TSourceLink),0); {$IFDEF DebugCTMemManager} inc(FAllocatedCount); {$ENDIF} end; inc(FCount); end; { TPSourceChangeStep } procedure TPSourceChangeStepMemManager.FreeFirstItem; var Step: PSourceChangeStep; begin Step:=PSourceChangeStep(FFirstFree); PSourceChangeStep(FFirstFree):=Step^.Next; Dispose(Step); end; procedure TPSourceChangeStepMemManager.DisposePSourceChangeStep( Step: PSourceChangeStep); begin if (FFreeCount free Step Dispose(Step); {$IFDEF DebugCTMemManager} inc(FFreedCount); {$ENDIF} end; dec(FCount); end; function TPSourceChangeStepMemManager.NewPSourceChangeStep: PSourceChangeStep; begin if FFirstFree<>nil then begin // take from free list Result:=PSourceChangeStep(FFirstFree); PSourceChangeStep(FFirstFree):=Result^.Next; Result^.Next:=nil; dec(FFreeCount); end else begin // free list empty -> create new PSourceChangeStep New(Result); FillChar(Result^,SizeOf(TSourceChangeStep),0); {$IFDEF DebugCTMemManager} inc(FAllocatedCount); {$ENDIF} end; inc(FCount); end; { TMissingIncludeFile } constructor TMissingIncludeFile.Create(const AFilename, AIncludePath: string); begin inherited Create; Filename:=AFilename; IncludePath:=AIncludePath; end; function TMissingIncludeFile.CalcMemSize: PtrUInt; begin Result:=PtrUInt(InstanceSize) +MemSizeString(IncludePath) +MemSizeString(Filename); end; { TMissingIncludeFiles } function TMissingIncludeFiles.GetIncFile(Index: Integer): TMissingIncludeFile; begin Result:=TMissingIncludeFile(Get(Index)); end; procedure TMissingIncludeFiles.SetIncFile(Index: Integer; const AValue: TMissingIncludeFile); begin Put(Index,AValue); end; procedure TMissingIncludeFiles.Clear; var i: integer; begin for i:=0 to Count-1 do Items[i].Free; inherited Clear; end; procedure TMissingIncludeFiles.Delete(Index: Integer); begin Items[Index].Free; inherited Delete(Index); end; function TMissingIncludeFiles.CalcMemSize: PtrUInt; var i: Integer; begin Result:=PtrUInt(InstanceSize) +SizeOf(Pointer)*PtrUInt(Capacity); for i:=0 to Count-1 do inc(Result,Items[i].CalcMemSize); end; //------------------------------------------------------------------------------ procedure InternalInit; var CompMode: TCompilerMode; begin for CompMode:=Low(TCompilerMode) to High(TCompilerMode) do CompilerModeVars[CompMode]:='FPC_'+CompilerModeNames[CompMode]; PSourceLinkMemManager:=TPSourceLinkMemManager.Create; PSourceChangeStepMemManager:=TPSourceChangeStepMemManager.Create; end; procedure InternalFinal; begin PSourceChangeStepMemManager.Free; PSourceLinkMemManager.Free; end; { ELinkScannerEditError } constructor ELinkScannerEditError.Create(ASender: TLinkScanner; TheId: int64; const AMessage: string; ABuffer: Pointer; ABufferPos: integer); begin inherited Create(ASender,TheId,AMessage); Buffer:=ABuffer; BufferPos:=ABufferPos; end; initialization InternalInit; finalization InternalFinal; end.