mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-11-04 14:39:36 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			408 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			408 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
{
 | 
						|
    $Id$
 | 
						|
    Copyright (c) 1998-2000 by Peter Vreman
 | 
						|
 | 
						|
    This unit handles the compilerhooks for output to external programs
 | 
						|
 | 
						|
    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 comphook;
 | 
						|
 | 
						|
{$i defines.inc}
 | 
						|
 | 
						|
interface
 | 
						|
 | 
						|
uses
 | 
						|
  finput;
 | 
						|
 | 
						|
Const
 | 
						|
{ <$10000 will show file and line }
 | 
						|
  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_Macro        = $100;
 | 
						|
  V_Procedure    = $200;
 | 
						|
  V_Conditional  = $400;
 | 
						|
  V_Assem        = $800;
 | 
						|
  V_Declarations = $1000;
 | 
						|
  V_Info         = $10000;
 | 
						|
  V_Status       = $20000;
 | 
						|
  V_Used         = $40000;
 | 
						|
  V_Tried        = $80000;
 | 
						|
  V_Debug        = $100000;
 | 
						|
  V_Executable   = $200000;
 | 
						|
  V_ShowFile     = $ffff;
 | 
						|
  V_All          = $ffffffff;
 | 
						|
  V_Default      = V_Fatal + V_Error + V_Normal;
 | 
						|
 | 
						|
const
 | 
						|
  { RHIDE expect gcc like error output }
 | 
						|
  fatalstr      : string[20] = 'Fatal:';
 | 
						|
  errorstr      : string[20] = 'Error:';
 | 
						|
  warningstr    : string[20] = 'Warning:';
 | 
						|
  notestr       : string[20] = 'Note:';
 | 
						|
  hintstr       : string[20] = 'Hint:';
 | 
						|
 | 
						|
type
 | 
						|
  PCompilerStatus = ^TCompilerStatus;
 | 
						|
  TCompilerStatus = record
 | 
						|
  { Current status }
 | 
						|
    currentmodule,
 | 
						|
    currentsourcepath,
 | 
						|
    currentsource : string;   { filename }
 | 
						|
    currentline,
 | 
						|
    currentcolumn : longint;  { current line and column }
 | 
						|
  { Total Status }
 | 
						|
    compiledlines : longint;  { the number of lines which are compiled }
 | 
						|
    errorcount    : longint;  { number of generated errors }
 | 
						|
  { program info }
 | 
						|
    isexe,
 | 
						|
    islibrary     : boolean;
 | 
						|
  { Settings for the output }
 | 
						|
    verbosity     : longint;
 | 
						|
    maxerrorcount : longint;
 | 
						|
    errorwarning,
 | 
						|
    errornote,
 | 
						|
    errorhint,
 | 
						|
    skip_error,
 | 
						|
    use_stderr,
 | 
						|
    use_redir,
 | 
						|
    use_gccoutput,
 | 
						|
    compiling_current : boolean;
 | 
						|
  { Redirection support }
 | 
						|
    redirfile : text;
 | 
						|
  end;
 | 
						|
var
 | 
						|
  status : tcompilerstatus;
 | 
						|
 | 
						|
{ Default Functions }
 | 
						|
procedure def_stop;
 | 
						|
procedure def_halt(i : longint);
 | 
						|
Function  def_status:boolean;
 | 
						|
Function  def_comment(Level:Longint;const s:string):boolean;
 | 
						|
function  def_internalerror(i:longint):boolean;
 | 
						|
procedure def_initsymbolinfo;
 | 
						|
procedure def_donesymbolinfo;
 | 
						|
procedure def_extractsymbolinfo;
 | 
						|
function  def_openinputfile(const filename: string): tinputfile;
 | 
						|
Function  def_getnamedfiletime(Const F : String) : Longint;
 | 
						|
{$ifdef DEBUG}
 | 
						|
{ allow easy stopping in GDB
 | 
						|
  using
 | 
						|
  b DEF_GDB_STOP
 | 
						|
  cond 1 LEVEL <= 8 }
 | 
						|
procedure def_gdb_stop(level : longint);
 | 
						|
{$endif DEBUG}
 | 
						|
{ Function redirecting for IDE support }
 | 
						|
type
 | 
						|
  tstopprocedure         = procedure;
 | 
						|
  thaltprocedure         = procedure(i : longint);
 | 
						|
  tstatusfunction        = function:boolean;
 | 
						|
  tcommentfunction       = function(Level:Longint;const s:string):boolean;
 | 
						|
  tinternalerrorfunction = function(i:longint):boolean;
 | 
						|
 | 
						|
  tinitsymbolinfoproc = procedure;
 | 
						|
  tdonesymbolinfoproc = procedure;
 | 
						|
  textractsymbolinfoproc = procedure;
 | 
						|
  topeninputfilefunc = function(const filename: string): tinputfile;
 | 
						|
  tgetnamedfiletimefunc = function(const filename: string): longint;
 | 
						|
 | 
						|
const
 | 
						|
  do_stop          : tstopprocedure   = {$ifdef FPCPROCVAR}@{$endif}def_stop;
 | 
						|
  do_halt          : thaltprocedure   = {$ifdef FPCPROCVAR}@{$endif}def_halt;
 | 
						|
  do_status        : tstatusfunction  = {$ifdef FPCPROCVAR}@{$endif}def_status;
 | 
						|
  do_comment       : tcommentfunction = {$ifdef FPCPROCVAR}@{$endif}def_comment;
 | 
						|
  do_internalerror : tinternalerrorfunction = {$ifdef FPCPROCVAR}@{$endif}def_internalerror;
 | 
						|
 | 
						|
  do_initsymbolinfo : tinitsymbolinfoproc = {$ifdef FPCPROCVAR}@{$endif}def_initsymbolinfo;
 | 
						|
  do_donesymbolinfo : tdonesymbolinfoproc = {$ifdef FPCPROCVAR}@{$endif}def_donesymbolinfo;
 | 
						|
  do_extractsymbolinfo : textractsymbolinfoproc = {$ifdef FPCPROCVAR}@{$endif}def_extractsymbolinfo;
 | 
						|
 | 
						|
  do_openinputfile : topeninputfilefunc = {$ifdef FPCPROCVAR}@{$endif}def_openinputfile;
 | 
						|
  do_getnamedfiletime : tgetnamedfiletimefunc = {$ifdef FPCPROCVAR}@{$endif}def_getnamedfiletime;
 | 
						|
 | 
						|
implementation
 | 
						|
 | 
						|
  uses
 | 
						|
{$ifdef delphi}
 | 
						|
   dmisc
 | 
						|
{$else}
 | 
						|
   dos
 | 
						|
{$endif}
 | 
						|
   ;
 | 
						|
 | 
						|
{****************************************************************************
 | 
						|
                          Helper Routines
 | 
						|
****************************************************************************}
 | 
						|
 | 
						|
function gccfilename(const s : string) : string;
 | 
						|
var
 | 
						|
  i : longint;
 | 
						|
begin
 | 
						|
  for i:=1to length(s) do
 | 
						|
   begin
 | 
						|
     case s[i] of
 | 
						|
      '\' : gccfilename[i]:='/';
 | 
						|
 'A'..'Z' : gccfilename[i]:=chr(ord(s[i])+32);
 | 
						|
     else
 | 
						|
      gccfilename[i]:=s[i];
 | 
						|
     end;
 | 
						|
   end;
 | 
						|
  gccfilename[0]:=s[0];
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
function tostr(i : longint) : string;
 | 
						|
var
 | 
						|
  hs : string;
 | 
						|
begin
 | 
						|
  str(i,hs);
 | 
						|
  tostr:=hs;
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
{****************************************************************************
 | 
						|
                         Predefined default Handlers
 | 
						|
****************************************************************************}
 | 
						|
 | 
						|
{ predefined handler when then compiler stops }
 | 
						|
procedure def_stop;
 | 
						|
begin
 | 
						|
  Halt(1);
 | 
						|
end;
 | 
						|
 | 
						|
{$ifdef DEBUG}
 | 
						|
{ allow easy stopping in GDB
 | 
						|
  using
 | 
						|
  b DEF_GDB_STOP
 | 
						|
  cond 1 LEVEL <= 8 }
 | 
						|
procedure def_gdb_stop(level : longint);
 | 
						|
begin
 | 
						|
  { Its only a dummy for GDB }
 | 
						|
end;
 | 
						|
{$endif DEBUG}
 | 
						|
 | 
						|
procedure def_halt(i : longint);
 | 
						|
begin
 | 
						|
  halt(i);
 | 
						|
end;
 | 
						|
 | 
						|
function def_status:boolean;
 | 
						|
begin
 | 
						|
  def_status:=false; { never stop }
 | 
						|
{ Status info?, Called every line }
 | 
						|
  if ((status.verbosity and V_Status)<>0) then
 | 
						|
   begin
 | 
						|
{$ifndef Delphi}
 | 
						|
     if (status.compiledlines=1) then
 | 
						|
       WriteLn(memavail shr 10,' Kb Free');
 | 
						|
{$endif Delphi}
 | 
						|
     if (status.currentline>0) and (status.currentline mod 100=0) then
 | 
						|
{$ifdef FPC}
 | 
						|
       WriteLn(status.currentline,' ',memavail shr 10,'/',system.heapsize shr 10,' Kb Free');
 | 
						|
{$else}
 | 
						|
  {$ifndef Delphi}
 | 
						|
       WriteLn(status.currentline,' ',memavail shr 10,' Kb Free');
 | 
						|
  {$endif Delphi}
 | 
						|
{$endif}
 | 
						|
   end
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
Function def_comment(Level:Longint;const s:string):boolean;
 | 
						|
const
 | 
						|
  rh_errorstr   = 'error:';
 | 
						|
  rh_warningstr = 'warning:';
 | 
						|
var
 | 
						|
  hs : string;
 | 
						|
begin
 | 
						|
  def_comment:=false; { never stop }
 | 
						|
  hs:='';
 | 
						|
  if not(status.use_gccoutput) then
 | 
						|
    begin
 | 
						|
      if (status.verbosity and Level)=V_Hint then
 | 
						|
        hs:=hintstr;
 | 
						|
      if (status.verbosity and Level)=V_Note then
 | 
						|
        hs:=notestr;
 | 
						|
      if (status.verbosity and Level)=V_Warning then
 | 
						|
        hs:=warningstr;
 | 
						|
      if (status.verbosity and Level)=V_Error then
 | 
						|
        hs:=errorstr;
 | 
						|
      if (status.verbosity and Level)=V_Fatal then
 | 
						|
        hs:=fatalstr;
 | 
						|
    end
 | 
						|
  else
 | 
						|
    begin
 | 
						|
      if (status.verbosity and Level)=V_Hint then
 | 
						|
        hs:=rh_warningstr;
 | 
						|
      if (status.verbosity and Level)=V_Note then
 | 
						|
        hs:=rh_warningstr;
 | 
						|
      if (status.verbosity and Level)=V_Warning then
 | 
						|
        hs:=rh_warningstr;
 | 
						|
      if (status.verbosity and Level)=V_Error then
 | 
						|
        hs:=rh_errorstr;
 | 
						|
      if (status.verbosity and Level)=V_Fatal then
 | 
						|
        hs:=rh_errorstr;
 | 
						|
    end;
 | 
						|
  if (Level<=V_ShowFile) and (status.currentsource<>'') and (status.currentline>0) then
 | 
						|
   begin
 | 
						|
     { Adding the column should not confuse RHIDE,
 | 
						|
     even if it does not yet use it PM
 | 
						|
     but only if it is after error or warning !! PM }
 | 
						|
     if status.currentcolumn>0 then
 | 
						|
      begin
 | 
						|
        if status.use_gccoutput then
 | 
						|
          hs:=gccfilename(status.currentsource)+':'+tostr(status.currentline)+': '+hs+' '+
 | 
						|
              tostr(status.currentcolumn)+': '+s
 | 
						|
        else
 | 
						|
          hs:=status.currentsource+'('+tostr(status.currentline)+
 | 
						|
              ','+tostr(status.currentcolumn)+') '+hs+' '+s;
 | 
						|
      end
 | 
						|
     else
 | 
						|
      begin
 | 
						|
        if status.use_gccoutput then
 | 
						|
          hs:=gccfilename(status.currentsource)+': '+hs+' '+tostr(status.currentline)+': '+s
 | 
						|
        else
 | 
						|
          hs:=status.currentsource+'('+tostr(status.currentline)+') '+hs+' '+s;
 | 
						|
      end;
 | 
						|
   end
 | 
						|
  else
 | 
						|
   begin
 | 
						|
     if hs<>'' then
 | 
						|
      hs:=hs+' '+s
 | 
						|
     else
 | 
						|
      hs:=s;
 | 
						|
   end;
 | 
						|
{$ifdef FPC}
 | 
						|
  if status.use_stderr then
 | 
						|
   begin
 | 
						|
     writeln(stderr,hs);
 | 
						|
     flush(stderr);
 | 
						|
   end
 | 
						|
  else
 | 
						|
{$endif}
 | 
						|
   begin
 | 
						|
     if status.use_redir then
 | 
						|
      writeln(status.redirfile,hs)
 | 
						|
     else
 | 
						|
      writeln(hs);
 | 
						|
   end;
 | 
						|
{$ifdef DEBUG}
 | 
						|
  def_gdb_stop(level);
 | 
						|
{$endif DEBUG}
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
function def_internalerror(i : longint) : boolean;
 | 
						|
begin
 | 
						|
  do_comment(V_Fatal,'Internal error '+tostr(i));
 | 
						|
  def_internalerror:=true;
 | 
						|
end;
 | 
						|
 | 
						|
procedure def_initsymbolinfo;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure def_donesymbolinfo;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure def_extractsymbolinfo;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
function  def_openinputfile(const filename: string): tinputfile;
 | 
						|
begin
 | 
						|
  def_openinputfile:=tdosinputfile.create(filename);
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
Function def_GetNamedFileTime (Const F : String) : Longint;
 | 
						|
var
 | 
						|
  L : Longint;
 | 
						|
  info : SearchRec;
 | 
						|
begin
 | 
						|
  l:=-1;
 | 
						|
  {$ifdef delphi}
 | 
						|
    dmisc.FindFirst (F,archive+readonly+hidden,info);
 | 
						|
  {$else delphi}
 | 
						|
    FindFirst (F,archive+readonly+hidden,info);
 | 
						|
  {$endif delphi}
 | 
						|
  if DosError=0 then
 | 
						|
   l:=info.time;
 | 
						|
  FindClose(info);
 | 
						|
  def_GetNamedFileTime:=l;
 | 
						|
end;
 | 
						|
 | 
						|
end.
 | 
						|
{
 | 
						|
  $Log$
 | 
						|
  Revision 1.16  2001-08-04 10:23:54  peter
 | 
						|
    * updates so it works with the ide
 | 
						|
 | 
						|
  Revision 1.15  2001/06/07 21:25:57  peter
 | 
						|
    * Regenerated
 | 
						|
 | 
						|
  Revision 1.14  2001/06/06 17:20:21  jonas
 | 
						|
    * fixed wrong typed constant procvars in preparation of my fix which will
 | 
						|
      disallow them in FPC mode (plus some other unmerged changes since
 | 
						|
      LAST_MERGE)
 | 
						|
 | 
						|
  Revision 1.13  2001/02/05 20:47:00  peter
 | 
						|
    * support linux unit for ver1_0 compilers
 | 
						|
 | 
						|
  Revision 1.12  2001/01/21 20:32:45  marco
 | 
						|
   * Renamefest. Compiler part. Not that hard.
 | 
						|
 | 
						|
  Revision 1.11  2000/12/26 15:58:29  peter
 | 
						|
    * check for verbosity in verbose instead of comphook
 | 
						|
 | 
						|
  Revision 1.10  2000/12/25 00:07:25  peter
 | 
						|
    + new tlinkedlist class (merge of old tstringqueue,tcontainer and
 | 
						|
      tlinkedlist objects)
 | 
						|
 | 
						|
  Revision 1.9  2000/11/13 15:26:12  marco
 | 
						|
   * Renamefest
 | 
						|
 | 
						|
  Revision 1.8  2000/09/30 16:07:20  peter
 | 
						|
    * prefix fix (merged)
 | 
						|
 | 
						|
  Revision 1.7  2000/09/24 21:33:46  peter
 | 
						|
    * message updates merges
 | 
						|
 | 
						|
  Revision 1.6  2000/09/24 15:06:13  peter
 | 
						|
    * use defines.inc
 | 
						|
 | 
						|
  Revision 1.5  2000/08/27 16:11:50  peter
 | 
						|
    * moved some util functions from globals,cobjects to cutils
 | 
						|
    * splitted files into finput,fmodule
 | 
						|
 | 
						|
  Revision 1.4  2000/08/13 13:04:15  peter
 | 
						|
    * -vb update
 | 
						|
 | 
						|
  Revision 1.3  2000/08/12 15:30:45  peter
 | 
						|
    * IDE patch for stream reading (merged)
 | 
						|
 | 
						|
  Revision 1.2  2000/07/13 11:32:38  michael
 | 
						|
  + removed logs
 | 
						|
 | 
						|
}
 |