mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-05 20:48:06 +02:00
1391 lines
43 KiB
ObjectPascal
1391 lines
43 KiB
ObjectPascal
{
|
|
Copyright (c) 1998-2002 by Peter Vreman
|
|
|
|
This unit handles the verbose management
|
|
|
|
This program 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 program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
****************************************************************************
|
|
}
|
|
unit verbose;
|
|
|
|
{$i fpcdefs.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
{$IFNDEF USE_FAKE_SYSUTILS}
|
|
sysutils,
|
|
{$ELSE}
|
|
fksysutl,
|
|
{$ENDIF}
|
|
cutils,
|
|
globtype,finput,
|
|
cmsgs;
|
|
|
|
{$ifndef EXTERN_MSG}
|
|
{$i msgtxt.inc}
|
|
{$endif}
|
|
|
|
{$i msgidx.inc}
|
|
|
|
Const
|
|
{ Levels }
|
|
V_None = $0;
|
|
V_Fatal = $1;
|
|
V_Error = $2;
|
|
V_Normal = $4; { doesn't show a text like Error: }
|
|
V_Warning = $8;
|
|
V_Note = $10;
|
|
V_Hint = $20;
|
|
V_LineInfoMask = $fff;
|
|
{ From here by default no line info }
|
|
V_Info = $1000;
|
|
V_Status = $2000;
|
|
V_Used = $4000;
|
|
V_Tried = $8000;
|
|
V_Conditional = $10000;
|
|
V_Debug = $20000;
|
|
V_Executable = $40000;
|
|
V_LevelMask = $fffffff;
|
|
V_All = V_LevelMask;
|
|
V_Default = V_Fatal + V_Error + V_Normal;
|
|
{ Flags }
|
|
V_LineInfo = $10000000;
|
|
|
|
var
|
|
msg : pmessage;
|
|
paraprintnodetree : byte;
|
|
|
|
type
|
|
tmsgqueueevent = procedure(const s:TMsgStr;v,w:longint) of object;
|
|
|
|
const
|
|
msgfilename : string = '';
|
|
|
|
procedure SetRedirectFile(const fn:string);
|
|
function SetVerbosity(const s:string):boolean;
|
|
procedure PrepareReport;
|
|
|
|
function CheckVerbosity(v:longint):boolean;
|
|
function SetMessageVerbosity(v:longint;state:tmsgstate):boolean;
|
|
procedure RestoreLocalVerbosity(pstate : pmessagestaterecord);
|
|
procedure FreeLocalVerbosity(var fstate : pmessagestaterecord);
|
|
|
|
function ChangeMessageVerbosity(s: string; var i: integer;state:tmsgstate): boolean;
|
|
procedure ShowStatus;
|
|
function ErrorCount:longint;
|
|
procedure SetErrorFlags(const s:string);
|
|
procedure GenerateError;
|
|
procedure Internalerror(i:longint);noreturn;
|
|
procedure Comment(l:longint;s:ansistring);
|
|
function MessageStr(w:longint):TMsgStr;
|
|
procedure Message(w:longint;onqueue:tmsgqueueevent=nil);
|
|
procedure Message1(w:longint;const s1:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
procedure Message2(w:longint;const s1,s2:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
procedure Message3(w:longint;const s1,s2,s3:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
procedure Message4(w:longint;const s1,s2,s3,s4:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
procedure MessagePos(const pos:tfileposinfo;w:longint;onqueue:tmsgqueueevent=nil);
|
|
procedure MessagePos1(const pos:tfileposinfo;w:longint;const s1:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
procedure MessagePos2(const pos:tfileposinfo;w:longint;const s1,s2:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
procedure MessagePos3(const pos:tfileposinfo;w:longint;const s1,s2,s3:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
procedure MessagePos4(const pos:tfileposinfo;w:longint;const s1,s2,s3,s4:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
|
|
{ message calls with codegenerror support }
|
|
procedure cgmessage(t : longint);
|
|
procedure cgmessage1(t : longint;const s : TMsgStr);
|
|
procedure cgmessage2(t : longint;const s1,s2 : TMsgStr);
|
|
procedure cgmessage3(t : longint;const s1,s2,s3 : TMsgStr);
|
|
procedure CGMessagePos(const pos:tfileposinfo;t:longint);
|
|
procedure CGMessagePos1(const pos:tfileposinfo;t:longint;const s1:TMsgStr);
|
|
procedure CGMessagePos2(const pos:tfileposinfo;t:longint;const s1,s2:TMsgStr);
|
|
procedure CGMessagePos3(const pos:tfileposinfo;t:longint;const s1,s2,s3:TMsgStr);
|
|
|
|
procedure FlushOutput;
|
|
|
|
procedure InitVerbose;
|
|
procedure DoneVerbose;
|
|
|
|
|
|
const
|
|
printnodespacing = ' ';
|
|
var
|
|
{ indention used when writing a node tree to the screen }
|
|
printnodeindention : string;
|
|
|
|
|
|
{ Node dumping support functions }
|
|
procedure printnodeindent; inline;
|
|
procedure printnodeunindent; inline;
|
|
{$ifdef DEBUG_NODE_XML}
|
|
function SanitiseXMLString(const S: ansistring): ansistring;
|
|
function WritePointer(const P: Pointer): ansistring;
|
|
function WriteConstPUInt(const P: TConstPtrUInt): ansistring;
|
|
function WriteGUID(const GUID: TGUID): ansistring;
|
|
{$endif DEBUG_NODE_XML}
|
|
|
|
implementation
|
|
|
|
uses
|
|
comphook,fmodule,constexp,globals,cfileutl,switches;
|
|
|
|
{****************************************************************************
|
|
Extra Handlers for default compiler
|
|
****************************************************************************}
|
|
|
|
procedure DoneRedirectFile;
|
|
begin
|
|
if status.use_redir then
|
|
begin
|
|
close(status.redirfile);
|
|
status.use_redir:=false;
|
|
end;
|
|
if status.use_bugreport then
|
|
begin
|
|
close(status.reportbugfile);
|
|
status.use_bugreport:=false;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure SetRedirectFile(const fn:string);
|
|
begin
|
|
{ close old redirection file because FileRedirection is handled in both passes }
|
|
if status.use_redir then
|
|
close(status.redirfile);
|
|
|
|
assign(status.redirfile,fn);
|
|
{$push}{$I-}
|
|
append(status.redirfile);
|
|
if ioresult <> 0 then
|
|
begin
|
|
assign(status.redirfile,fn);
|
|
rewrite(status.redirfile);
|
|
end;
|
|
{$pop}
|
|
status.use_redir:=(ioresult=0);
|
|
end;
|
|
|
|
|
|
procedure PrepareReport;
|
|
var
|
|
fn : string;
|
|
begin
|
|
if status.use_bugreport then
|
|
exit;
|
|
fn:='fpcdebug.txt';
|
|
assign(status.reportbugfile,fn);
|
|
{$push}{$I-}
|
|
append(status.reportbugfile);
|
|
if ioresult <> 0 then
|
|
rewrite(status.reportbugfile);
|
|
{$pop}
|
|
status.use_bugreport:=(ioresult=0);
|
|
if status.use_bugreport then
|
|
writeln(status.reportbugfile,'FPC bug report file');
|
|
end;
|
|
|
|
procedure RestoreLocalVerbosity(pstate : pmessagestaterecord);
|
|
begin
|
|
msg^.ResetStates;
|
|
while assigned(pstate) do
|
|
begin
|
|
SetMessageVerbosity(pstate^.value,pstate^.state);
|
|
pstate:=pstate^.next;
|
|
end;
|
|
end;
|
|
|
|
procedure FreeLocalVerbosity(var fstate : pmessagestaterecord);
|
|
var pstate : pmessagestaterecord;
|
|
begin
|
|
pstate:=unaligned(fstate);
|
|
while assigned(pstate) do
|
|
begin
|
|
unaligned(fstate):=pstate^.next;
|
|
freemem(pstate);
|
|
pstate:=unaligned(fstate);
|
|
end;
|
|
end;
|
|
|
|
function ChangeMessageVerbosity(s: string; var i : integer;state:tmsgstate): boolean;
|
|
var
|
|
tok : string;
|
|
msgnr, code : longint;
|
|
begin
|
|
{ delete everything up to and including 'm' }
|
|
delete(s,1,i);
|
|
{ the rest of the string must be message numbers }
|
|
inc(i,length(s)+1);
|
|
result:=false;
|
|
repeat
|
|
tok:=GetToken(s,',');
|
|
if (tok='') then
|
|
break;
|
|
val(tok, msgnr, code);
|
|
if (code<>0) then
|
|
exit;
|
|
if not msg^.setverbosity(msgnr,state) then
|
|
exit
|
|
else
|
|
recordpendingmessagestate(msgnr, state);
|
|
until false;
|
|
result:=true;
|
|
end;
|
|
|
|
function SetMessageVerbosity(v:longint;state:tmsgstate):boolean;
|
|
begin
|
|
result:=msg^.setverbosity(v,state);
|
|
end;
|
|
|
|
function CheckVerbosity(v:longint):boolean;
|
|
begin
|
|
result:=do_checkverbosity(v);
|
|
end;
|
|
|
|
|
|
function SetVerbosity(const s:string):boolean;
|
|
const
|
|
message_verbosity:array[boolean] of tmsgstate=(ms_off_global,ms_on_global);
|
|
var
|
|
m : Longint;
|
|
i : Integer;
|
|
inverse : boolean;
|
|
c : char;
|
|
begin
|
|
Setverbosity:=false;
|
|
val(s,m,i);
|
|
if (i=0) and (s<>'') then
|
|
status.verbosity:=m
|
|
else
|
|
begin
|
|
i:=1;
|
|
while i<=length(s) do
|
|
begin
|
|
c:=upcase(s[i]);
|
|
inverse:=false;
|
|
{ on/off ? }
|
|
if (i<length(s)) then
|
|
case s[i+1] of
|
|
'-' : begin
|
|
inc(i);
|
|
inverse:=true;
|
|
end;
|
|
'+' : inc(i);
|
|
end;
|
|
{ handle switch }
|
|
case c of
|
|
{ Special cases }
|
|
'0' : status.verbosity:=V_Default;
|
|
'A' : status.verbosity:=V_All;
|
|
'B' : begin
|
|
if inverse then
|
|
status.print_source_path:=false
|
|
else
|
|
status.print_source_path:=true;
|
|
end;
|
|
'M' : if not ChangeMessageVerbosity(s,i,message_verbosity[inverse]) then
|
|
begin
|
|
result:=false;
|
|
exit
|
|
end;
|
|
'P' : begin
|
|
if inverse then
|
|
paraprintnodetree:=0
|
|
else
|
|
paraprintnodetree:=1;
|
|
end;
|
|
'Q' : begin
|
|
if inverse then
|
|
status.showmsgnrs:=false
|
|
else
|
|
status.showmsgnrs:=true;
|
|
end;
|
|
'R' : begin
|
|
if inverse then
|
|
begin
|
|
status.use_gccoutput:=false;
|
|
status.use_stderr:=false;
|
|
end
|
|
else
|
|
begin
|
|
status.use_gccoutput:=true;
|
|
status.use_stderr:=true;
|
|
end;
|
|
end;
|
|
'V' : PrepareReport;
|
|
'Z' : begin
|
|
if inverse then
|
|
status.use_stderr:=false
|
|
else
|
|
status.use_stderr:=true;
|
|
end;
|
|
{ Normal cases - do an or }
|
|
'C' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Conditional)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Conditional;
|
|
'D' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Debug)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Debug;
|
|
'E' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Error)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Error;
|
|
'H' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Hint)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Hint;
|
|
'I' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Info)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Info;
|
|
'L' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Status)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Status;
|
|
'N' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Note)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Note;
|
|
'S' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_TimeStamps)
|
|
else
|
|
status.verbosity:=status.verbosity or V_TimeStamps;
|
|
'T' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Tried)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Tried;
|
|
'U' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Used)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Used;
|
|
'W' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Warning)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Warning;
|
|
'X' : if inverse then
|
|
status.verbosity:=status.verbosity and (not V_Executable)
|
|
else
|
|
status.verbosity:=status.verbosity or V_Executable;
|
|
end;
|
|
inc(i);
|
|
end;
|
|
end;
|
|
if status.verbosity=0 then
|
|
status.verbosity:=V_Default;
|
|
setverbosity:=true;
|
|
end;
|
|
|
|
|
|
procedure Loadprefixes;
|
|
|
|
function loadprefix(w:longint):string;
|
|
var
|
|
s : string;
|
|
idx : longint;
|
|
begin
|
|
s:=msg^.get(w,[]);
|
|
idx:=pos('_',s);
|
|
if idx>0 then
|
|
Loadprefix:=Copy(s,idx+1,255)
|
|
else
|
|
Loadprefix:=s;
|
|
end;
|
|
|
|
begin
|
|
{ Load the prefixes }
|
|
fatalstr:=Loadprefix(general_i_fatal);
|
|
errorstr:=Loadprefix(general_i_error);
|
|
warningstr:=Loadprefix(general_i_warning);
|
|
notestr:=Loadprefix(general_i_note);
|
|
hintstr:=Loadprefix(general_i_hint);
|
|
end;
|
|
|
|
|
|
procedure LoadMsgFile(const fn:string);
|
|
begin
|
|
{ reload the internal messages if not already loaded }
|
|
{$ifndef EXTERN_MSG}
|
|
if not msg^.msgintern then
|
|
msg^.LoadIntern(@msgtxt,msgtxtsize,msgtxt_codepage);
|
|
{$endif}
|
|
if not msg^.LoadExtern(fn) then
|
|
begin
|
|
{$ifdef EXTERN_MSG}
|
|
writeln('Fatal: Cannot find error message file.');
|
|
halt(3);
|
|
{$else}
|
|
msg^.LoadIntern(@msgtxt,msgtxtsize,msgtxt_codepage);
|
|
{$endif}
|
|
end;
|
|
{ reload the prefixes using the new messages }
|
|
Loadprefixes;
|
|
end;
|
|
|
|
|
|
procedure MaybeLoadMessageFile;
|
|
begin
|
|
{ Load new message file }
|
|
if (msgfilename<>'') then
|
|
begin
|
|
LoadMsgFile(msgfilename);
|
|
msgfilename:='';
|
|
end;
|
|
end;
|
|
|
|
|
|
var
|
|
lastfileidx,
|
|
lastmoduleidx : longint;
|
|
|
|
|
|
Procedure UpdateStatus;
|
|
var
|
|
module : tmodule;
|
|
begin
|
|
{ fix status }
|
|
status.currentline:=current_filepos.line;
|
|
status.currentcolumn:=current_filepos.column;
|
|
if (current_filepos.moduleindex <> lastmoduleidx) or
|
|
(current_filepos.fileindex <> lastfileidx) then
|
|
begin
|
|
module:=get_module(current_filepos.moduleindex);
|
|
if assigned(module) and assigned(module.sourcefiles) then
|
|
begin
|
|
{ update status record }
|
|
status.currentmodule:=module.modulename^;
|
|
status.currentsourceppufilename:=module.ppufilename;
|
|
status.currentmodulestate:=ModuleStateStr[module.state];
|
|
status.currentsource:=module.sourcefiles.get_file_name(current_filepos.fileindex);
|
|
status.currentsourcepath:=module.sourcefiles.get_file_path(current_filepos.fileindex);
|
|
status.sources_avail:=module.sources_avail;
|
|
{ if currentsourcepath is relative, make it absolute }
|
|
if not path_absolute(status.currentsourcepath) then
|
|
status.currentsourcepath:=GetCurrentDir+status.currentsourcepath;
|
|
|
|
{ update lastfileidx only if name known PM }
|
|
if status.currentsource<>'' then
|
|
lastfileidx:=current_filepos.fileindex
|
|
else
|
|
lastfileidx:=0;
|
|
|
|
lastmoduleidx:=module.unit_index;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure ShowStatus;
|
|
begin
|
|
UpdateStatus;
|
|
if do_status() then
|
|
raise ECompilerAbort.Create;
|
|
end;
|
|
|
|
|
|
function ErrorCount:longint;
|
|
begin
|
|
ErrorCount:=status.errorcount;
|
|
end;
|
|
|
|
|
|
procedure SetErrorFlags(const s:string);
|
|
var
|
|
code : integer;
|
|
i,j,l : longint;
|
|
begin
|
|
{ empty string means error count = 1 for backward compatibility (PFV) }
|
|
if s='' then
|
|
begin
|
|
status.maxerrorcount:=1;
|
|
exit;
|
|
end;
|
|
i:=0;
|
|
while (i<length(s)) do
|
|
begin
|
|
inc(i);
|
|
case s[i] of
|
|
'0'..'9' :
|
|
begin
|
|
j:=i;
|
|
while (j<=length(s)) and (s[j] in ['0'..'9']) do
|
|
inc(j);
|
|
val(copy(s,i,j-i),l,code);
|
|
if code<>0 then
|
|
l:=1;
|
|
status.maxerrorcount:=l;
|
|
i:=j-1;
|
|
end;
|
|
'w','W' :
|
|
begin
|
|
if (i<length(s)) and (s[i+1]='-') then
|
|
begin
|
|
inc(i);
|
|
status.errorwarning:=false;
|
|
end
|
|
else
|
|
begin
|
|
status.errorwarning:=true;
|
|
{ Enable writing of warnings, to avoid getting errors without any message }
|
|
status.verbosity:=status.verbosity or V_Warning;
|
|
end;
|
|
end;
|
|
'n','N' :
|
|
begin
|
|
if (i<length(s)) and (s[i+1]='-') then
|
|
begin
|
|
inc(i);
|
|
status.errornote:=false;
|
|
end
|
|
else
|
|
begin
|
|
status.errornote:=true;
|
|
{ Enable writing of notes, to avoid getting errors without any message }
|
|
status.verbosity:=status.verbosity or V_Note;
|
|
end;
|
|
|
|
end;
|
|
'h','H' :
|
|
begin
|
|
if (i<length(s)) and (s[i+1]='-') then
|
|
begin
|
|
inc(i);
|
|
status.errorhint:=false;
|
|
end
|
|
else
|
|
begin
|
|
status.errorhint:=true;
|
|
{ Enable writing of hints, to avoid getting errors without any message }
|
|
status.verbosity:=status.verbosity or V_Hint;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure GenerateError;
|
|
begin
|
|
inc(status.errorcount);
|
|
end;
|
|
|
|
|
|
procedure internalerror(i : longint);noreturn;
|
|
begin
|
|
UpdateStatus;
|
|
do_internalerror(i);
|
|
GenerateError;
|
|
raise ECompilerAbort.Create;
|
|
end;
|
|
|
|
|
|
procedure Comment(l:longint;s:ansistring);
|
|
var
|
|
dostop : boolean;
|
|
begin
|
|
dostop:=((l and V_Fatal)<>0);
|
|
if ((l and V_Error)<>0) or
|
|
((l and V_Fatal)<>0) or
|
|
(status.errorwarning and ((l and V_Warning)<>0)) or
|
|
(status.errornote and ((l and V_Note)<>0)) or
|
|
(status.errorhint and ((l and V_Hint)<>0)) then
|
|
GenerateError
|
|
else
|
|
if l and V_Warning <> 0 then
|
|
inc(status.countWarnings)
|
|
else
|
|
if l and V_Note <> 0 then
|
|
inc(status.countNotes)
|
|
else
|
|
if l and V_Hint <> 0 then
|
|
inc(status.countHints);
|
|
{ check verbosity level }
|
|
if not CheckVerbosity(l) then
|
|
exit;
|
|
if (l and V_LineInfoMask)<>0 then
|
|
l:=l or V_LineInfo;
|
|
{ Create status info }
|
|
UpdateStatus;
|
|
{ Fix replacements }
|
|
DefaultReplacements(s);
|
|
{ show comment }
|
|
if do_comment(l,s) or dostop then
|
|
raise ECompilerAbort.Create;
|
|
if (status.errorcount>=status.maxerrorcount) and not status.skip_error then
|
|
begin
|
|
Message1(unit_f_errors_in_unit,tostr(status.errorcount));
|
|
status.skip_error:=true;
|
|
raise ECompilerAbort.Create;
|
|
end;
|
|
end;
|
|
|
|
function GetMessageState(m:longint):tmsgstate;
|
|
var
|
|
i: integer;
|
|
begin
|
|
i:=m div 1000;
|
|
{ get the default state }
|
|
Result:=msg^.msgstates[i]^[m mod 1000];
|
|
|
|
{ and search at the current unit settings }
|
|
{ todo }
|
|
end;
|
|
|
|
Procedure Msg2Comment(s:ansistring;w:longint;onqueue:tmsgqueueevent);
|
|
var
|
|
idx,i,v : longint;
|
|
dostop : boolean;
|
|
doqueue : boolean;
|
|
st : tmsgstate;
|
|
ch : char;
|
|
begin
|
|
{Reset}
|
|
dostop:=false;
|
|
doqueue:=false;
|
|
v:=0;
|
|
{Parse options}
|
|
idx:=pos('_',s);
|
|
if idx=0 then
|
|
v:=V_None
|
|
else
|
|
if (idx >= 1) And (idx <= 5) then
|
|
begin
|
|
for i:=1 to idx do
|
|
begin
|
|
ch:=upcase(s[i]);
|
|
case ch of
|
|
'F' :
|
|
begin
|
|
v:=v or V_Fatal;
|
|
GenerateError;
|
|
dostop:=true;
|
|
end;
|
|
'E','W','N','H':
|
|
begin
|
|
if ch='E' then
|
|
st:=ms_error
|
|
else
|
|
st:=GetMessageState(w);
|
|
{ We only want to know about local value }
|
|
st:= tmsgstate(ord(st) and ms_local_mask);
|
|
if st=ms_error then
|
|
begin
|
|
v:=v or V_Error;
|
|
GenerateError;
|
|
end
|
|
else if st<>ms_off then
|
|
case ch of
|
|
'W':
|
|
begin
|
|
v:=v or V_Warning;
|
|
if CheckVerbosity(V_Warning) then
|
|
if status.errorwarning then
|
|
GenerateError
|
|
else
|
|
inc(status.countWarnings);
|
|
end;
|
|
'N' :
|
|
begin
|
|
v:=v or V_Note;
|
|
if CheckVerbosity(V_Note) then
|
|
if status.errornote then
|
|
GenerateError
|
|
else
|
|
inc(status.countNotes);
|
|
end;
|
|
'H' :
|
|
begin
|
|
v:=v or V_Hint;
|
|
if CheckVerbosity(V_Hint) then
|
|
if status.errorhint then
|
|
GenerateError
|
|
else
|
|
inc(status.countHints);
|
|
end;
|
|
end;
|
|
end;
|
|
'O' :
|
|
v:=v or V_Normal;
|
|
'I' :
|
|
v:=v or V_Info;
|
|
'L' :
|
|
v:=v or V_LineInfo;
|
|
'U' :
|
|
v:=v or V_Used;
|
|
'T' :
|
|
v:=v or V_Tried;
|
|
'C' :
|
|
v:=v or V_Conditional;
|
|
'D' :
|
|
v:=v or V_Debug;
|
|
'X' :
|
|
v:=v or V_Executable;
|
|
'S' :
|
|
dostop:=true;
|
|
'_' : ;
|
|
end;
|
|
end;
|
|
end;
|
|
Delete(s,1,idx);
|
|
{ check verbosity level }
|
|
if not CheckVerbosity(v) then
|
|
begin
|
|
doqueue := onqueue <> nil;
|
|
if not doqueue then
|
|
exit;
|
|
end;
|
|
if (v and V_LineInfoMask)<>0 then
|
|
v:=v or V_LineInfo;
|
|
{ fix status }
|
|
UpdateStatus;
|
|
{ Fix replacements }
|
|
DefaultReplacements(s);
|
|
if status.showmsgnrs and ((v and V_Normal)=0) then
|
|
s:='('+tostr(w)+') '+s;
|
|
if doqueue then
|
|
begin
|
|
onqueue(s,v,w);
|
|
exit;
|
|
end;
|
|
{ show comment }
|
|
if do_comment(v,s) or dostop then
|
|
raise ECompilerAbort.Create;
|
|
if (status.errorcount>=status.maxerrorcount) and not status.skip_error then
|
|
begin
|
|
Message1(unit_f_errors_in_unit,tostr(status.errorcount));
|
|
status.skip_error:=true;
|
|
raise ECompilerAbort.Create;
|
|
end;
|
|
end;
|
|
|
|
|
|
function MessageStr(w:longint):TMsgStr;
|
|
begin
|
|
MaybeLoadMessageFile;
|
|
MessageStr:=msg^.Get(w,[]);
|
|
end;
|
|
|
|
|
|
procedure Message(w:longint;onqueue:tmsgqueueevent=nil);
|
|
begin
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[]),w,onqueue);
|
|
end;
|
|
|
|
|
|
procedure Message1(w:longint;const s1:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
|
|
begin
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1]),w,onqueue);
|
|
end;
|
|
|
|
|
|
procedure Message2(w:longint;const s1,s2:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
begin
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1,s2]),w,onqueue);
|
|
end;
|
|
|
|
|
|
procedure Message3(w:longint;const s1,s2,s3:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
begin
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1,s2,s3]),w,onqueue);
|
|
end;
|
|
|
|
|
|
procedure Message4(w:longint;const s1,s2,s3,s4:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
begin
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1,s2,s3,s4]),w,onqueue);
|
|
end;
|
|
|
|
|
|
procedure MessagePos(const pos:tfileposinfo;w:longint;onqueue:tmsgqueueevent=nil);
|
|
var
|
|
oldpos : tfileposinfo;
|
|
begin
|
|
oldpos:=current_filepos;
|
|
current_filepos:=pos;
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[]),w,onqueue);
|
|
current_filepos:=oldpos;
|
|
end;
|
|
|
|
|
|
procedure MessagePos1(const pos:tfileposinfo;w:longint;const s1:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
var
|
|
oldpos : tfileposinfo;
|
|
begin
|
|
oldpos:=current_filepos;
|
|
current_filepos:=pos;
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1]),w,onqueue);
|
|
current_filepos:=oldpos;
|
|
end;
|
|
|
|
|
|
procedure MessagePos2(const pos:tfileposinfo;w:longint;const s1,s2:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
var
|
|
oldpos : tfileposinfo;
|
|
begin
|
|
oldpos:=current_filepos;
|
|
current_filepos:=pos;
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1,s2]),w,onqueue);
|
|
current_filepos:=oldpos;
|
|
end;
|
|
|
|
|
|
procedure MessagePos3(const pos:tfileposinfo;w:longint;const s1,s2,s3:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
var
|
|
oldpos : tfileposinfo;
|
|
begin
|
|
oldpos:=current_filepos;
|
|
current_filepos:=pos;
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1,s2,s3]),w,onqueue);
|
|
current_filepos:=oldpos;
|
|
end;
|
|
|
|
|
|
procedure MessagePos4(const pos:tfileposinfo;w:longint;const s1,s2,s3,s4:TMsgStr;onqueue:tmsgqueueevent=nil);
|
|
var
|
|
oldpos : tfileposinfo;
|
|
begin
|
|
oldpos:=current_filepos;
|
|
current_filepos:=pos;
|
|
MaybeLoadMessageFile;
|
|
Msg2Comment(msg^.Get(w,[s1,s2,s3,s4]),w,onqueue);
|
|
current_filepos:=oldpos;
|
|
end;
|
|
|
|
|
|
{*****************************************************************************
|
|
override the message calls to set codegenerror
|
|
*****************************************************************************}
|
|
|
|
procedure cgmessage(t : longint);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.Message(t);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
procedure cgmessage1(t : longint;const s : TMsgStr);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.Message1(t,s);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
procedure cgmessage2(t : longint;const s1,s2 : TMsgStr);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.Message2(t,s1,s2);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
procedure cgmessage3(t : longint;const s1,s2,s3 : TMsgStr);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.Message3(t,s1,s2,s3);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure cgmessagepos(const pos:tfileposinfo;t : longint);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.MessagePos(pos,t);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
procedure cgmessagepos1(const pos:tfileposinfo;t : longint;const s1 : TMsgStr);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.MessagePos1(pos,t,s1);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
procedure cgmessagepos2(const pos:tfileposinfo;t : longint;const s1,s2 : TMsgStr);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.MessagePos2(pos,t,s1,s2);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
procedure cgmessagepos3(const pos:tfileposinfo;t : longint;const s1,s2,s3 : TMsgStr);
|
|
var
|
|
olderrorcount : longint;
|
|
begin
|
|
if not(codegenerror) then
|
|
begin
|
|
olderrorcount:=Errorcount;
|
|
verbose.MessagePos3(pos,t,s1,s2,s3);
|
|
codegenerror:=olderrorcount<>Errorcount;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure FlushOutput;
|
|
begin
|
|
if not (Status.Use_StdErr) then (* StdErr is flushed after every line *)
|
|
begin
|
|
if Status.Use_Redir then
|
|
Flush(Status.RedirFile)
|
|
else
|
|
Flush(Output);
|
|
end;
|
|
end;
|
|
|
|
|
|
{*****************************************************************************
|
|
Initialization
|
|
*****************************************************************************}
|
|
|
|
procedure InitVerbose;
|
|
begin
|
|
{ Init }
|
|
msg:=new(pmessage,Init(20,msgidxmax));
|
|
if msg=nil then
|
|
begin
|
|
writeln('Fatal: MsgIdx Wrong');
|
|
halt(3);
|
|
end;
|
|
{$ifndef EXTERN_MSG}
|
|
msg^.LoadIntern(@msgtxt,msgtxtsize,msgtxt_codepage);
|
|
{$else EXTERN_MSG}
|
|
LoadMsgFile(exepath+'errore.msg');
|
|
{$endif EXTERN_MSG}
|
|
FillChar(Status,sizeof(TCompilerStatus),0);
|
|
status.verbosity:=V_Default;
|
|
Status.MaxErrorCount:=50;
|
|
Status.codesize:=aword(-1);
|
|
Status.datasize:=aword(-1);
|
|
Loadprefixes;
|
|
lastfileidx:=-1;
|
|
lastmoduleidx:=-1;
|
|
status.currentmodule:='';
|
|
status.currentsourceppufilename:='';
|
|
status.currentsource:='';
|
|
status.currentsourcepath:='';
|
|
{ Register internalerrorproc for cutils/cclasses }
|
|
internalerrorproc:=@internalerror;
|
|
end;
|
|
|
|
|
|
procedure DoneVerbose;
|
|
begin
|
|
if assigned(msg) then
|
|
begin
|
|
dispose(msg,Done);
|
|
msg:=nil;
|
|
end;
|
|
DoneRedirectFile;
|
|
end;
|
|
|
|
|
|
procedure printnodeindent; inline;
|
|
begin
|
|
printnodeindention:=printnodeindention+printnodespacing;
|
|
end;
|
|
|
|
|
|
procedure printnodeunindent; inline;
|
|
begin
|
|
delete(printnodeindention,1,length(printnodespacing));
|
|
end;
|
|
|
|
{$ifdef DEBUG_NODE_XML}
|
|
function WritePointer(const P: Pointer): ansistring;
|
|
begin
|
|
case PtrUInt(P) of
|
|
0:
|
|
WritePointer := 'nil';
|
|
1..$FFFF:
|
|
WritePointer := '$' + hexstr(PtrUInt(P), 4);
|
|
{$if sizeof(Pointer) > 4}
|
|
$10000..$FFFFFFFF:
|
|
WritePointer := '$' + hexstr(PtrUInt(P), 8);
|
|
{$ifend sizeof(Pointer) > 4}
|
|
else
|
|
WritePointer := '$' + hexstr(PtrUInt(P), 2*sizeof(Pointer));
|
|
end;
|
|
end;
|
|
|
|
|
|
function WriteConstPUInt(const P: TConstPtrUInt): ansistring;
|
|
begin
|
|
case P of
|
|
0:
|
|
WriteConstPUInt := 'nil';
|
|
1..$FFFF:
|
|
WriteConstPUInt := '$' + hexstr(P, 4);
|
|
{$if sizeof(TConstPtrUInt) > 4}
|
|
$10000..$FFFFFFFF:
|
|
WriteConstPUInt := '$' + hexstr(P, 8);
|
|
{$ifend sizeof(TConstPtrUInt) >= 4}
|
|
else
|
|
WriteConstPUInt := '$' + hexstr(P, 2*sizeof(TConstPtrUInt));
|
|
end;
|
|
end;
|
|
|
|
|
|
function WriteGUID(const GUID: TGUID): ansistring;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := '{' + hexstr(GUID.D1, 8) + '-' + hexstr(GUID.D2, 4) + '-' + hexstr(GUID.D3, 4) + '-';
|
|
for i := 0 to 7 do
|
|
Result := Result + hexstr(GUID.D4[i], 2);
|
|
|
|
Result := Result + '}';
|
|
end;
|
|
|
|
|
|
function SanitiseXMLString(const S: ansistring): ansistring;
|
|
var
|
|
X, UTF8Len, UTF8Char, CurrentChar: Integer;
|
|
needs_quoting, in_quotes, add_end_quote: Boolean;
|
|
DoASCII: Boolean;
|
|
|
|
{ Write the given byte as #xxx }
|
|
procedure EncodeControlChar(Value: Byte);
|
|
begin
|
|
if X = Length(Result) then
|
|
add_end_quote := False;
|
|
|
|
Delete(Result, X, 1);
|
|
if in_quotes then
|
|
begin
|
|
Insert('#' + tostr(Value) + '''', Result, X);
|
|
|
|
{ If the entire string consists of control characters, it
|
|
doesn't need quoting, so only set the flag here }
|
|
needs_quoting := True;
|
|
|
|
in_quotes := False;
|
|
end
|
|
else
|
|
Insert('#' + tostr(Value), Result, X);
|
|
end;
|
|
|
|
{ Write the given byte as either a plain character or an XML keyword }
|
|
procedure EncodeStandardChar(Value: Byte);
|
|
begin
|
|
if not in_quotes then
|
|
begin
|
|
in_quotes := True;
|
|
if (X < Length(Result)) then
|
|
begin
|
|
needs_quoting := True;
|
|
Insert('''', Result, X + 1)
|
|
end;
|
|
end;
|
|
|
|
{ Check the character for anything that could be mistaken for an XML element }
|
|
case CurrentChar of
|
|
Ord('#'):
|
|
{ Required to differentiate '#27' from the escape code #27, for example }
|
|
needs_quoting:=true;
|
|
|
|
Ord('<'):
|
|
begin
|
|
Delete(Result, X, 1);
|
|
Insert('<', Result, X);
|
|
end;
|
|
Ord('>'):
|
|
begin
|
|
Delete(Result, X, 1);
|
|
Insert('>', Result, X);
|
|
end;
|
|
Ord('&'):
|
|
begin
|
|
Delete(Result, X, 1);
|
|
Insert('&', Result, X);
|
|
end;
|
|
Ord('"'):
|
|
begin
|
|
needs_quoting := True;
|
|
Delete(Result, X, 1);
|
|
Insert('"', Result, X);
|
|
end;
|
|
Ord(''''):
|
|
begin
|
|
needs_quoting:=true;
|
|
{ Simply double it like in pascal strings }
|
|
Insert('''', Result, X);
|
|
end;
|
|
else
|
|
{ Do nothing };
|
|
end;
|
|
end;
|
|
|
|
{ Convert character between $80 and $FF to UTF-8 }
|
|
procedure EncodeExtendedChar(Value: Byte);
|
|
begin
|
|
if not in_quotes then
|
|
begin
|
|
in_quotes := True;
|
|
if (X < Length(Result)) then
|
|
begin
|
|
needs_quoting := True;
|
|
Insert('''', Result, X + 1)
|
|
end;
|
|
end;
|
|
|
|
case Value of
|
|
$80..$BF: { Add $C2 before the value }
|
|
Insert(#$C2, Result, X);
|
|
$C0..$FF: { Zero the $40 bit and add $C3 before the value }
|
|
begin
|
|
Result[X] := Char(Byte(Result[X]) and $BF);
|
|
Insert(#$C3, Result, X);
|
|
end;
|
|
else
|
|
{ Previous conditions should prevent this procedure from being
|
|
called if Value < $80 }
|
|
InternalError(2019061901);
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
needs_quoting := False;
|
|
Result := S;
|
|
|
|
{ Gets set to True if an invalid UTF-8 sequence is found }
|
|
DoASCII := False;
|
|
|
|
{ By setting in_quotes to false here, we can exclude the single
|
|
quotation marks surrounding the string if it doesn't contain any
|
|
control characters, or consists entirely of control characters. }
|
|
in_quotes := False;
|
|
|
|
add_end_quote := True;
|
|
|
|
X := Length(Result);
|
|
while X > 0 do
|
|
begin
|
|
CurrentChar := Ord(Result[X]);
|
|
|
|
{ Control characters and extended characters need special handling }
|
|
case CurrentChar of
|
|
$00..$1F, $7F:
|
|
EncodeControlChar(CurrentChar);
|
|
|
|
$20..$7E:
|
|
EncodeStandardChar(CurrentChar);
|
|
|
|
{ UTF-8 continuation byte }
|
|
$80..$BF:
|
|
begin
|
|
if not in_quotes then
|
|
begin
|
|
in_quotes := True;
|
|
if (X < Length(Result)) then
|
|
begin
|
|
needs_quoting := True;
|
|
Insert('''', Result, X + 1)
|
|
end;
|
|
end;
|
|
|
|
UTF8Char := CurrentChar and $3F; { The data bits of the continuation byte }
|
|
UTF8Len := 1; { This variable actually holds 1 less than the length }
|
|
|
|
{ By setting DoASCII to true, it marks the string as 'invalid UTF-8'
|
|
automatically if it reaches the beginning of the string unexpectedly }
|
|
DoASCII := True;
|
|
|
|
Dec(X);
|
|
while X > 0 do
|
|
begin
|
|
CurrentChar := Ord(Result[X]);
|
|
|
|
case CurrentChar of
|
|
{ A standard character here is invalid UTF-8 }
|
|
$00..$7F:
|
|
Break;
|
|
|
|
{ Another continuation byte }
|
|
$80..$BF:
|
|
begin
|
|
UTF8Char := UTF8Char or ((CurrentChar and $3F) shl (6 * UTF8Len));
|
|
|
|
Inc(UTF8Len);
|
|
if UTF8Len >= 4 then
|
|
{ Sequence too long }
|
|
Break;
|
|
end;
|
|
|
|
{ Lead byte for 2-byte sequences }
|
|
$C2..$DF:
|
|
begin
|
|
if UTF8Len <> 1 then Break;
|
|
|
|
UTF8Char := UTF8Char or ((CurrentChar and $1F) shl 6);
|
|
|
|
{ Check to see if the code is in range and not part of an 'overlong' sequence }
|
|
case UTF8Char of
|
|
$0080..$07FF:
|
|
DoASCII := False;
|
|
else
|
|
{ Do nothing - DoASCII is already true }
|
|
end;
|
|
Break;
|
|
end;
|
|
|
|
{ Lead byte for 3-byte sequences }
|
|
$E0..$EF:
|
|
begin
|
|
if UTF8Len <> 2 then Break;
|
|
|
|
UTF8Char := UTF8Char or ((CurrentChar and $0F) shl 12);
|
|
|
|
{ Check to see if the code is in range and not part of an 'overlong' sequence }
|
|
case UTF8Char of
|
|
$0800..$D7FF, $E000..$FFFF: { $D800..$DFFF is reserved and hence invalid }
|
|
DoASCII := False;
|
|
else
|
|
{ Do nothing - DoASCII is already true }
|
|
end;
|
|
Break;
|
|
end;
|
|
|
|
{ Lead byte for 4-byte sequences }
|
|
$F0..$F4:
|
|
begin
|
|
if UTF8Len <> 3 then Break;
|
|
|
|
UTF8Char := UTF8Char or ((CurrentChar and $07) shl 18);
|
|
|
|
{ Check to see if the code is in range and not part of an 'overlong' sequence }
|
|
case UTF8Char of
|
|
$010000..$10FFFF:
|
|
DoASCII := False;
|
|
else
|
|
{ Do nothing - DoASCII is already true }
|
|
end;
|
|
Break;
|
|
end;
|
|
|
|
{ Invalid character }
|
|
else
|
|
Break;
|
|
end;
|
|
end;
|
|
|
|
if DoASCII then
|
|
Break;
|
|
|
|
{ If all is fine, we don't need to encode any more characters }
|
|
end;
|
|
|
|
{ Invalid UTF-8 bytes and lead bytes without continuation bytes }
|
|
$C0..$FF:
|
|
begin
|
|
DoASCII := True;
|
|
Break;
|
|
end;
|
|
end;
|
|
|
|
Dec(X);
|
|
end;
|
|
|
|
{ UTF-8 failed, so encode the string as plain ASCII }
|
|
if DoASCII then
|
|
begin
|
|
{ Reset the flags and Result }
|
|
needs_quoting := False;
|
|
Result := S;
|
|
in_quotes := False;
|
|
add_end_quote := True;
|
|
|
|
for X := Length(Result) downto 1 do
|
|
begin
|
|
CurrentChar := Ord(Result[X]);
|
|
|
|
{ Control characters and extended characters need special handling }
|
|
case CurrentChar of
|
|
$00..$1F, $7F:
|
|
EncodeControlChar(CurrentChar);
|
|
|
|
$20..$7E:
|
|
EncodeStandardChar(CurrentChar);
|
|
|
|
{ Extended characters }
|
|
else
|
|
EncodeExtendedChar(CurrentChar);
|
|
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
if needs_quoting then
|
|
begin
|
|
if in_quotes then
|
|
Result := '''' + Result;
|
|
|
|
if add_end_quote then
|
|
Result := Result + '''';
|
|
end;
|
|
end;
|
|
{$endif DEBUG_NODE_XML}
|
|
|
|
|
|
initialization
|
|
constexp.internalerrorproc:=@internalerror;
|
|
finalization
|
|
{ Be sure to close the redirect files to flush all data }
|
|
DoneRedirectFile;
|
|
end.
|