mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-22 10:22:08 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			329 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			329 lines
		
	
	
		
			8.7 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;
 | |
| interface
 | |
| 
 | |
| 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_Info         = $10000;
 | |
|   V_Status       = $20000;
 | |
|   V_Used         = $40000;
 | |
|   V_Tried        = $80000;
 | |
|   V_Debug        = $100000;
 | |
|   V_Declarations = $200000;
 | |
|   V_Executable   = $400000;
 | |
|   V_ShowFile     = $ffff;
 | |
|   V_All          = $ffffffff;
 | |
|   V_Default      = V_Fatal + V_Error + V_Normal;
 | |
| 
 | |
| 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 }
 | |
|   { Settings for the output }
 | |
|     verbosity     : longint;
 | |
|     maxerrorcount : longint;
 | |
|     skip_error,
 | |
|     use_stderr,
 | |
|     use_redir,
 | |
|     use_gccoutput : 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;
 | |
| {$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;
 | |
| const
 | |
|   do_stop          : tstopprocedure   = def_stop;
 | |
|   do_halt          : thaltprocedure   = def_halt;
 | |
|   do_status        : tstatusfunction  = def_status;
 | |
|   do_comment       : tcommentfunction = def_comment;
 | |
|   do_internalerror : tinternalerrorfunction = def_internalerror;
 | |
| 
 | |
| 
 | |
| 
 | |
| implementation
 | |
| 
 | |
| {$ifdef USEEXCEPT}
 | |
|   uses tpexcept;
 | |
| {$endif USEEXCEPT}
 | |
| 
 | |
| {****************************************************************************
 | |
|                           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;
 | |
|   {$ifndef TP}
 | |
|     {$ifopt H+}
 | |
|       setlength(gccfilename,length(s));
 | |
|     {$else}
 | |
|       gccfilename[0]:=s[0];
 | |
|     {$endif}
 | |
|   {$else}
 | |
|     gccfilename[0]:=s[0];
 | |
|   {$endif}
 | |
| 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
 | |
| {$ifndef USEEXCEPT}
 | |
|   Halt(1);
 | |
| {$else USEEXCEPT}
 | |
|   Halt(1);
 | |
| {$endif USEEXCEPT}
 | |
| 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
 | |
|   { RHIDE expect gcc like error output }
 | |
|   rh_errorstr='error: ';
 | |
|   rh_warningstr='warning: ';
 | |
|   fatalstr='Fatal: ';
 | |
|   errorstr='Error: ';
 | |
|   warningstr='Warning: ';
 | |
|   notestr='Note: ';
 | |
|   hintstr='Hint: ';
 | |
| var
 | |
|   hs : string;
 | |
| begin
 | |
|   def_comment:=false; { never stop }
 | |
|   if (status.verbosity and Level)=Level then
 | |
|    begin
 | |
|      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)+': '
 | |
|            else
 | |
|              hs:=status.currentsource+'('+tostr(status.currentline)
 | |
|                  +','+tostr(status.currentcolumn)+') '+hs;
 | |
|          end
 | |
|         else
 | |
|          begin
 | |
|            if status.use_gccoutput then
 | |
|              hs:=gccfilename(status.currentsource)+': '+hs+tostr(status.currentline)+': '
 | |
|            else
 | |
|              hs:=status.currentsource+'('+tostr(status.currentline)+') '+hs;
 | |
|          end;
 | |
|       end;
 | |
|    { add the message to the text }
 | |
|      hs:=hs+s;
 | |
| {$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;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function def_internalerror(i : longint) : boolean;
 | |
| begin
 | |
|   do_comment(V_Fatal,'Internal error '+tostr(i));
 | |
|   def_internalerror:=true;
 | |
| end;
 | |
| 
 | |
| 
 | |
| end.
 | |
| {
 | |
|   $Log$
 | |
|   Revision 1.20  2000-01-07 01:14:23  peter
 | |
|     * updated copyright to 2000
 | |
| 
 | |
|   Revision 1.19  1999/11/18 15:34:45  pierre
 | |
|     * Notes/Hints for local syms changed to
 | |
|       Set_varstate function
 | |
| 
 | |
|   Revision 1.18  1999/09/07 14:03:48  pierre
 | |
|    + added do_halt procedure
 | |
| 
 | |
|   Revision 1.17  1999/08/05 16:52:53  peter
 | |
|     * V_Fatal=1, all other V_ are also increased
 | |
|     * Check for local procedure when assigning procvar
 | |
|     * fixed comment parsing because directives
 | |
|     * oldtp mode directives better supported
 | |
|     * added some messages to errore.msg
 | |
| 
 | |
|   Revision 1.16  1999/05/04 21:44:38  florian
 | |
|     * changes to compile it with Delphi 4.0
 | |
| 
 | |
|   Revision 1.15  1999/01/15 12:27:23  peter
 | |
|     * removed path from output, was there only for debugging
 | |
| 
 | |
|   Revision 1.14  1999/01/14 21:47:09  peter
 | |
|     * status.currentmodule is now also updated
 | |
|     + status.currentsourcepath
 | |
| 
 | |
|   Revision 1.13  1998/12/11 00:03:12  peter
 | |
|     + globtype,tokens,version unit splitted from globals
 | |
| 
 | |
| }
 | 
