mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-25 12:41:40 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			293 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			293 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| {
 | |
|     $Id$
 | |
|     This file is part of the Free Pascal run time library.
 | |
|     Copyright (c) 1999-2000 by Michael Van Canneyt
 | |
|     member of the Free Pascal development team
 | |
| 
 | |
|     See the file COPYING.FPC, included in this distribution,
 | |
|     for details about the copyright.
 | |
| 
 | |
|     This program is distributed in the hope that it will be useful,
 | |
|     but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 | |
| 
 | |
|  **********************************************************************}
 | |
| 
 | |
| {****************************************************************************
 | |
|                                 Exception support
 | |
| ****************************************************************************}
 | |
| 
 | |
| 
 | |
| Const
 | |
|   { Type of exception. Currently only one. }
 | |
|   FPC_EXCEPTION   = 1;
 | |
| 
 | |
|   { types of frames for the exception address stack }
 | |
|   cExceptionFrame = 1;
 | |
|   cFinalizeFrame  = 2;
 | |
| 
 | |
| Type
 | |
|   PExceptAddr = ^TExceptAddr;
 | |
|   TExceptAddr = record
 | |
|     buf       : pjmp_buf;
 | |
|     frametype : Longint;
 | |
|     next      : PExceptAddr;
 | |
|   end;
 | |
| 
 | |
| 
 | |
|   TExceptObjectClass = Class of TObject;
 | |
| 
 | |
| Const
 | |
|   CatchAllExceptions = -1;
 | |
| {$ifdef MT}
 | |
| ThreadVar
 | |
| {$else MT}
 | |
| Var
 | |
| {$endif MT}
 | |
|   ExceptAddrStack   : PExceptAddr;
 | |
|   ExceptObjectStack : PExceptObject;
 | |
| 
 | |
| Function RaiseList : PExceptObject;
 | |
| 
 | |
| begin
 | |
|   RaiseList:=ExceptObjectStack;
 | |
| end;
 | |
| 
 | |
| {$ifndef HAS_ADDR_STACK_ON_STACK}
 | |
| Function PushExceptAddr (Ft: Longint): PJmp_buf ;
 | |
|   [Public, Alias : 'FPC_PUSHEXCEPTADDR'];saveregisters;
 | |
| {$else HAS_ADDR_STACK_ON_HEAP}
 | |
| Function PushExceptAddr (Ft: Longint;_buf,_newaddr : pointer): PJmp_buf ;
 | |
|   [Public, Alias : 'FPC_PUSHEXCEPTADDR'];saveregisters;
 | |
| {$endif HAS_ADDR_STACK_ON_STACK}
 | |
| 
 | |
| var
 | |
|   Buf : PJmp_buf;
 | |
|   NewAddr : PExceptAddr;
 | |
| begin
 | |
| {$ifdef excdebug}
 | |
|   writeln ('In PushExceptAddr');
 | |
| {$endif}
 | |
|   If ExceptAddrstack=Nil then
 | |
|     begin
 | |
| {$ifndef HAS_ADDR_STACK_ON_STACK}
 | |
|       New(ExceptAddrStack);
 | |
| {$else HAS_ADDR_STACK_ON_STACK}
 | |
|       ExceptAddrStack:=PExceptAddr(_newaddr);
 | |
| {$endif HAS_ADDR_STACK_ON_STACK}
 | |
|       ExceptAddrStack^.Next:=Nil;
 | |
|     end
 | |
|   else
 | |
|     begin
 | |
| {$ifndef HAS_ADDR_STACK_ON_STACK}
 | |
|       New(NewAddr);
 | |
| {$else HAS_ADDR_STACK_ON_STACK}
 | |
|       NewAddr:=PExceptAddr(_newaddr);
 | |
| {$endif HAS_ADDR_STACK_ON_STACK}
 | |
|       NewAddr^.Next:=ExceptAddrStack;
 | |
|       ExceptAddrStack:=NewAddr;
 | |
|     end;
 | |
| {$ifndef HAS_ADDR_STACK_ON_STACK}
 | |
|   new(buf);
 | |
| {$else HAS_ADDR_STACK_ON_STACK}
 | |
|   buf:=PJmp_Buf(_buf);
 | |
| {$endif HAS_ADDR_STACK_ON_STACK}
 | |
|   ExceptAddrStack^.Buf:=Buf;
 | |
|   ExceptAddrStack^.FrameType:=ft;
 | |
|   PushExceptAddr:=Buf;
 | |
| end;
 | |
| 
 | |
| 
 | |
| Procedure PushExceptObj (Obj : TObject; AnAddr,AFrame : Pointer);
 | |
|   [Public, Alias : 'FPC_PUSHEXCEPTOBJECT'];saveregisters;
 | |
| var
 | |
|   Newobj : PExceptObject;
 | |
| begin
 | |
| {$ifdef excdebug}
 | |
|   writeln ('In PushExceptObject');
 | |
| {$endif}
 | |
|   If ExceptObjectStack=Nil then
 | |
|     begin
 | |
|       New(ExceptObjectStack);
 | |
|       ExceptObjectStack^.Next:=Nil;
 | |
|     end
 | |
|   else
 | |
|     begin
 | |
|       New(NewObj);
 | |
|       NewObj^.Next:=ExceptObjectStack;
 | |
|       ExceptObjectStack:=NewObj;
 | |
|     end;
 | |
|   ExceptObjectStack^.FObject:=Obj;
 | |
|   ExceptObjectStack^.Addr:=AnAddr;
 | |
|   ExceptObjectStack^.Frame:=AFrame;
 | |
| end;
 | |
| 
 | |
| 
 | |
| Procedure DoUnHandledException;
 | |
| begin
 | |
|   If ExceptProc<>Nil then
 | |
|     If ExceptObjectStack<>Nil then
 | |
|       TExceptPRoc(ExceptProc)(ExceptObjectStack^.FObject,ExceptObjectStack^.Addr,ExceptObjectStack^.Frame);
 | |
|   RunError(217);
 | |
| end;
 | |
| 
 | |
| 
 | |
| Function Raiseexcept (Obj : TObject; AnAddr,AFrame : Pointer) : TObject;[Public, Alias : 'FPC_RAISEEXCEPTION'];
 | |
| begin
 | |
| {$ifdef excdebug}
 | |
|   writeln ('In RaiseException');
 | |
| {$endif}
 | |
|   Raiseexcept:=nil;
 | |
|   PushExceptObj(Obj,AnAddr,AFrame);
 | |
|   If ExceptAddrStack=Nil then
 | |
|     DoUnhandledException;
 | |
|   if (RaiseProc <> nil) and (ExceptObjectStack <> nil) then
 | |
|     RaiseProc(Obj, AnAddr, AFrame);
 | |
|   longjmp(ExceptAddrStack^.Buf^,FPC_Exception);
 | |
| end;
 | |
| 
 | |
| 
 | |
| Procedure PopAddrStack;[Public, Alias : 'FPC_POPADDRSTACK'];
 | |
| {$ifndef HAS_ADDR_STACK_ON_STACK}
 | |
| var
 | |
|   hp : PExceptAddr;
 | |
| {$endif HAS_ADDR_STACK_ON_STACK}
 | |
| begin
 | |
| {$ifdef excdebug}
 | |
|   writeln ('In Popaddrstack');
 | |
| {$endif}
 | |
|   If ExceptAddrStack=nil then
 | |
|     begin
 | |
|       writeln ('At end of ExceptionAddresStack');
 | |
|       halt (255);
 | |
|     end
 | |
|   else
 | |
|     begin
 | |
| {$ifndef HAS_ADDR_STACK_ON_STACK}
 | |
|       hp:=ExceptAddrStack;
 | |
|       ExceptAddrStack:=ExceptAddrStack^.Next;
 | |
|       dispose(hp^.buf);
 | |
|       dispose(hp);
 | |
| {$else HAS_ADDR_STACK_ON_STACK}
 | |
|       ExceptAddrStack:=ExceptAddrStack^.Next;
 | |
| {$endif HAS_ADDR_STACK_ON_STACK}
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function PopObjectStack : TObject;[Public, Alias : 'FPC_POPOBJECTSTACK'];
 | |
| var
 | |
|   hp : PExceptObject;
 | |
| begin
 | |
| {$ifdef excdebug}
 | |
|   writeln ('In PopObjectstack');
 | |
| {$endif}
 | |
|   If ExceptObjectStack=nil then
 | |
|     begin
 | |
|     writeln ('At end of ExceptionObjectStack');
 | |
|     halt (1);
 | |
|     end
 | |
|   else
 | |
|     begin
 | |
|        { we need to return the exception object to dispose it }
 | |
|        PopObjectStack:=ExceptObjectStack^.FObject;
 | |
|        hp:=ExceptObjectStack;
 | |
|        ExceptObjectStack:=ExceptObjectStack^.next;
 | |
|        dispose(hp);
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| { this is for popping exception objects when a second exception is risen }
 | |
| { in an except/on                                                        }
 | |
| function PopSecondObjectStack : TObject;[Public, Alias : 'FPC_POPSECONDOBJECTSTACK'];
 | |
| var
 | |
|   hp : PExceptObject;
 | |
| begin
 | |
| {$ifdef excdebug}
 | |
|   writeln ('In PopObjectstack');
 | |
| {$endif}
 | |
|   If not(assigned(ExceptObjectStack)) or
 | |
|      not(assigned(ExceptObjectStack^.next)) then
 | |
|     begin
 | |
|     writeln ('At end of ExceptionObjectStack');
 | |
|     halt (1);
 | |
|     end
 | |
|   else
 | |
|     begin
 | |
|        { we need to return the exception object to dispose it }
 | |
|        PopSecondObjectStack:=ExceptObjectStack^.next^.FObject;
 | |
|        hp:=ExceptObjectStack^.next;
 | |
|        ExceptObjectStack^.next:=hp^.next;
 | |
|        dispose(hp);
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| Procedure ReRaise;[Public, Alias : 'FPC_RERAISE'];
 | |
| begin
 | |
| {$ifdef excdebug}
 | |
|   writeln ('In reraise');
 | |
| {$endif}
 | |
|   If ExceptAddrStack=Nil then
 | |
|     DoUnHandledException;
 | |
|   longjmp(ExceptAddrStack^.Buf^,FPC_Exception);
 | |
| end;
 | |
| 
 | |
| 
 | |
| Function Catches(Objtype : TExceptObjectClass) : TObject;[Public, Alias : 'FPC_CATCHES'];
 | |
| begin
 | |
|   If ExceptObjectStack=Nil then
 | |
|    begin
 | |
|      Writeln ('Internal error.');
 | |
|      halt (255);
 | |
|    end;
 | |
|   if Not ((Objtype = TExceptObjectClass(CatchAllExceptions)) or
 | |
|          (ExceptObjectStack^.FObject is ObjType)) then
 | |
|     Catches:=Nil
 | |
|   else
 | |
|     begin
 | |
|       // catch !
 | |
|       Catches:=ExceptObjectStack^.FObject;
 | |
|       { this can't be done, because there could be a reraise (PFV)
 | |
|        PopObjectStack;
 | |
| 
 | |
|        Also the PopAddrStack shouldn't be done, we do it now
 | |
|        immediatly in the exception handler (FK)
 | |
|       PopAddrStack; }
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| Procedure DestroyException(o : TObject);[Public, Alias : 'FPC_DESTROYEXCEPTION'];
 | |
| begin
 | |
|   { with free we're on the really save side }
 | |
|   o.Free;
 | |
| end;
 | |
| 
 | |
| 
 | |
| Procedure InitExceptions;
 | |
| {
 | |
|   Initialize exceptionsupport
 | |
| }
 | |
| begin
 | |
|   ExceptObjectstack:=Nil;
 | |
|   ExceptAddrStack:=Nil;
 | |
| end;
 | |
| {
 | |
|   $Log$
 | |
|   Revision 1.6  2001-04-13 22:30:04  peter
 | |
|     * remove warnings
 | |
| 
 | |
|   Revision 1.5  2001/01/24 21:47:18  florian
 | |
|     + more MT stuff added
 | |
| 
 | |
|   Revision 1.4  2001/01/05 17:35:50  florian
 | |
|   * the info about exception frames is stored now on the stack
 | |
|   instead on the heap
 | |
| 
 | |
|   Revision 1.3  2000/09/30 07:38:07  sg
 | |
|   * Added 'RaiseProc': A user-definable callback procedure which gets
 | |
|     called whenever an exception is being raised
 | |
| 
 | |
|   Revision 1.2  2000/07/13 11:33:42  michael
 | |
|   + removed logs
 | |
| }
 | 
