[Back to DOS SWAG index] [Back to Main SWAG index] [Original]
{
Alexander Kugel
There was a discussion about how to trap floating point errors
in TP. Here is the solution that traps any kind of run-time
errors. The idea is not mine. I saw it in a russian book about TP
and OOP.
The idea is quite simple. Instead of trying to trap all kind of
errors, we can let TP to do the job For us. Whenever TP stops
execution of the Program ( because of a run time error or just
because the Program stops in a natural way ) it executes the
default Procedure of Exit : ExitProc. Then TP checks the status of
two Variables from the SYSTEM Unit : ErrorAddr and ExitCode. If
there was a run time error then ErrorAddr is not NIL and ExitCode
containes the run time error code. Otherwise ExitCode containes the
errorlevel that will be set For Dos and ErrorAddr is NIL.
Fortunatly we can easily redefine the ExitProc, and thus to
overtake the control from TP. The problem is that we got to be able
to get back or to jump to any point of the Program ( even to jump
inside a Procedure / Function). The author of the book claimed that
he took his routines from Turbo Professional.
Well, there are two Files you are gonna need. Save the first one
as JUMP.PAS Compile it as a Unit. The second one is a short Program
that shows how to use it. It asks For two numbers, divides the
first by the second and takes a natural logarithm of the result.
Try to divide by zero, logarithm of a negative number. Try entering
letters instead of numbers and see how the Program recovers.
The trapping works fine under Windows/Dos. To run it With
WindowS recompile the JUMP Unit For Windows target. Then add WinCrt
to the Uses statement and remove Mark/Release lines ( because there
is no Mark/Release For Windows ).
------------------------------jump.pas-----------------------------
}
Unit Jump;
Interface
Type
JumpRecord = Record
SpReg,
BpReg : Word;
JmpPt : Pointer;
end;
Procedure SetJump(Var JumpDest : JumpRecord);
{Storing SP,BP and the address}
Inline(
$5F/ {pop di }
$07/ {pop es }
$26/$89/$25/ {mov es:[di],sp }
$26/$89/$6D/$02/ {mov es:[di+2],bp }
$E8/$00/$00/ {call null }
{null: }
$58/ {pop ax }
$05/$0C/$00/ {add ax,12 }
$26/$89/$45/$04/ {mov es:[di+4],ax }
$26/$8C/$4D/$06); {mov es:[di+6],cs }
{next: }
Procedure LongJump(Var JumpDest : JumpRecord);
{Restore everything and jump}
Inline(
$5F/ {pop di }
$07/ {pop es }
$26/$8B/$25/ {mov sp,es:[di] }
$26/$8B/$6D/$02/ {mov bp,es:[di+2] }
$26/$FF/$6D/$04); {jmp far es:[di+4]}
Implementation
end.
{ ------------------------------try.pas------------------------------ }
Program Try;
Uses
Jump; {Uses Jump,WinCrt;}
Var
OldExit : Pointer;
MyAddr : JumpRecord;
MyHeap : Pointer;
a1,a2,
a3,a4 : Real;
{$F+}
Procedure MyExit;
{You can add your error handler here}
begin
if ErrorAddr <> Nil Then
begin
Case ExitCode of
106 : Writeln('Invalid numeric format');
200 : Writeln('Division by zero');
205 : Writeln('Floating point overflow');
206 : Writeln('Floating point underflow');
207 : Writeln('Invalid floating point operation');
else Writeln('Hmmm... How did you do that ?');
end;
ErrorAddr := Nil;
LongJump(MyAddr);
end;
ExitProc := OldExit;
end;
{$F-}
begin
OldExit := ExitProc;
Mark(MyHeap);
{Just an example of how to restore the heap }
{Actually we don't have to do that in }
{this Program, because we dont use heap }
{at all. But anyway here it goes }
{Don't forget to remove when compiling this }
{for Windows }
SetJump(MyAddr);
{We'll get back here whenever a run time }
{error occurs }
{This line should always be before }
{ ExitProc:=MyExit; }
{Don't ask me why... It's much easier For me}
{to follow the rule then to understand it :)}
ExitProc := @MyExit;
Release(MyHeap);
{restoring the heap after a run time error }
{Remove this if you are compiling it For }
{Windows }
{Try entering whatever you want at the }
{prompt. It should trap every runtime error}
{you could possibly get. }
Repeat
Writeln;
Write('Enter a number a1=');
Readln(a1);
Write('Enter a number a2=');
Readln(a2);
a3 := a1 / a2;
Writeln('a1/a2=', a3 : 10 : 5);
a4 := ln(a3);
Writeln('ln(a1/a2)=', a4 : 10 : 5);
Until a3 = 1;
end.
[Back to DOS SWAG index] [Back to Main SWAG index] [Original]