Files
harbour-core/harbour/source/rtl/math.c
Przemyslaw Czerpak 917c8057e6 2006-06-14 13:05 UTC+0200 Przemyslaw Czerpak (druzus/at/priv.onet.pl)
* harbour/config/linux/dir.cf
  * harbour/config/linux/gcc.cf
    * formatting

  * harbour/config/linux/global.cf
  * harbour/config/linux/install.cf
    * added -ldl to linked library list and RANLIB executing after library
      creation

  * harbour/contrib/ole/ole2.c
    * added #include "hbapicls.h"

  + harbour/include/hbapicls.h
  * harbour/include/hbapi.h
    + added HB_IT_ENUM item
    + added collect member to HB_IT_POINTER structure for POINTER items
      inspected by GC
    * changed the order of HB_CODEBLOCK members to safe few bytes on
      alignment
    * removed counters from HB_CODEBLOCK and HB_BASEARRAY structure
    + added hb_xRefInc(), hb_xRefDec(), hb_xRefFree(), hb_xRefCount(),
      hb_xRefResize() functions. They are internal Harbour VM functions
      and covered by internal API macro. Should not be used by 3-rd
      party code because we may change them in the future or even remove.
    + added hb_gcRefInc(), hb_gcRefDec(), hb_gcRefFree(), hb_gcRefCount()
      functions. These are also internal function which corresponds to the
      above hb_x*() ones.
    - removed hb_arrayRelease() function. This function is not longer
      necessary and it should never be public function - it was implemented
      in very dangerous way.
    - removed hb_codeblockDelete() function.
    - removed hb_memvarValueDecGarbageRef()
    + added USHORT usLen parameter to hb_codeblockMacroNew() - when sets
      to non 0 value it informs that PCODE should be copied to dynamically
      allocated buffer.
    * moved all classes/object functions to separate file: hbapicls.h
    + added hb_retptrGC()
    + added internal function hb_memvarGetItem()

  * harbour/include/hbapiitm.h
    - removed hb_itemForwardValue() and added macro which translates
      hb_itemForwardValue() to hb_itemMove()
    + added hb_itemUnShareString() and hb_itemReSizeString() functions
    + added hb_itemSetNil() macro

  * harbour/include/hbdefs.h
    * some cleanups in macro definitions

  * harbour/include/hbexprb.c
    * indenting

  * harbour/include/hbvm.h
  * harbour/include/hbinit.h
    + added hb_vmProcessSymbolsExt(), hb_vmRegisterSymbols(),
      hb_vmFreeSymbols(), hb_vmBeginSymbolGroup(),
      hb_vmInitSymbolGroup(), hb_vmExitSymbolGroup()
    * changed hb_vmProcessSymbols() to return address of register symbol
      table. For normal code it's the same address as given in parameter
      so it does not break backward binary compatibility.
    * changed symbol init macros to work correctly with modified address
      of symbol table

  * harbour/include/hbsetup.h
    * added support for HB_USE_PROFILER macro. Now profiler code is
      is disable in HVM by default
    * commented out HB_ASORT_OPT_ITEMCOPY - it's not longer used - see note.

  * harbour/include/hbstack.h
    * added missing parenthesis in hb_stackItem() macro

  * harbour/include/hbtypes.h
    * updated VM_PROCESS_DLL_SYMBOLS definition
    + added VM_PROCESS_SYMBOLS_EXT
    - removed HB_ARRAYRELEASE

  * harbour/include/hbver.h
    * changed revision number to 2

  * harbour/include/hbvmpub.h
    - remove pFunPtr from HB_DYNS and cover profiler member by
      #ifndef HB_NO_PROFILER
    + added HB_PCODEFUNC structure
    + added HB_FS_PCODEFUNC, HB_FS_DYNCODE and HB_FS_LOCAL - the last one
      is not used yet.

  * harbour/include/hbxvm.h
    * added some multipcode functions for some speed optimization in -gc3
      output

  * harbour/include/hbapicdp.h
  * harbour/source/codepage/cpbg866.c
  * harbour/source/codepage/cpbgiso.c
  * harbour/source/codepage/cpbgwin.c
  * harbour/source/codepage/cpeldos.c
  * harbour/source/codepage/cpelwin.c
  * harbour/source/codepage/cpesdos.c
  * harbour/source/codepage/cpesmwi.c
  * harbour/source/codepage/cpeswin.c
  * harbour/source/codepage/cpgedos.c
  * harbour/source/codepage/cpgewin.c
  * harbour/source/codepage/cphu852.c
  * harbour/source/codepage/cphuiso.c
  * harbour/source/codepage/cphuwin.c
  * harbour/source/codepage/cppl852.c
  * harbour/source/codepage/cppliso.c
  * harbour/source/codepage/cpplmaz.c
  * harbour/source/codepage/cpplwin.c
  * harbour/source/codepage/cppt850.c
  * harbour/source/codepage/cpptiso.c
  * harbour/source/codepage/cpru866.c
  * harbour/source/codepage/cprukoi.c
  * harbour/source/codepage/cpruwin.c
  * harbour/source/codepage/cpsl437.c
  * harbour/source/codepage/cpsl852.c
  * harbour/source/codepage/cpsliso.c
  * harbour/source/codepage/cpslwin.c
  * harbour/source/codepage/cpsrwin.c
  * harbour/source/common/hbfhnd.c
  * harbour/source/common/hbstr.c
  * harbour/source/common/hbver.c
  * harbour/source/compiler/gencli.c
  * harbour/source/rtl/fstemp.c
  * harbour/source/rtl/hbgtcore.c
  * harbour/source/rtl/langapi.c
  * harbour/source/rtl/gtalleg/gtalleg.c
  * harbour/source/rtl/gtalleg/ssf.h
  * harbour/source/rtl/gtcgi/gtcgi.c
  * harbour/source/rtl/gtcrs/gtcrs.c
  * harbour/source/rtl/gtdos/gtdos.c
  * harbour/source/rtl/gtos2/gtos2.c
  * harbour/source/rtl/gtpca/gtpca.c
  * harbour/source/rtl/gtsln/gtsln.c
  * harbour/source/rtl/gtstd/gtstd.c
  * harbour/source/rtl/gtwin/gtwin.c
  * harbour/source/rtl/gtwvt/gtwvt.c
  * harbour/source/rtl/gtxwc/gtxwc.c
    * cleaned the code to avoid warnings for some non ANSI C constructions

  * harbour/source/compiler/cmdcheck.c
    * formatting

  * harbour/source/compiler/gencc.c
    * added some multipcode optimizations

  * harbour/source/compiler/genc.c
    * keep function name set by compiler for static initstatics

  * harbour/source/compiler/harbour.c
    * extend init statics name with the number of used static variables
    * formatting

  * harbour/source/compiler/hbusage.c
    + added my name to developers list - I hope that there is enough
      of my code in Harbour (BTW for quite long time) ;-)

  * harbour/source/rdd/dbcmd.c
    - removed DBF2TEXT() function

  * harbour/source/rdd/dbf1.c
  * harbour/source/rdd/dbffpt/dbffpt1.c
    * generate RT error when someone tries to create DBF with memo fields
      without MEMO RDD linked.

  * harbour/source/rtl/errorapi.c
    * replace all hb_vmDo() with hb_vmSend()

  * harbour/source/rtl/math.c
    * use hb_errPutArgs() to pass parameters to error object

  * harbour/source/rtl/philes.c
  * harbour/source/rtl/space.c
  * harbour/source/rtl/strpeek.c
    * some minor optimizations

  * harbour/source/vm/arrays.c
    - removed hb_arrayRelease() function.
    * changed hb_arrayReleaseGarbage() to be safe for recursive call
    * updated to use GC counters

  * harbour/source/vm/arrayshb.c
    * use stack function/macros instead of direct accessing HB_STACK members

  * harbour/source/vm/asort.c
    * removed unused HB_ASORT_OPT_ITEMCOPY

  * harbour/source/vm/classes.c
    * added class code from HVM
    % added fast overloaded operator detection and execution
    + added hb_objHasOperator(), hb_objOperatorCall()
    + added separated overloading of "=" and "==" operators
    * operate on PHB_SYMB not PHB_FUNC - support for pure PCODE function
      without any machine code
    * use stack function/macros instead of direct accessing HB_STACK members
    * separated profiler code
    * some code cleaning

  * harbour/source/vm/codebloc.c
    + added USHORT usLen parameter to hb_codeblockMacroNew() - when sets
      to non 0 value it informs that PCODE should be copied to dynamically
      allocated buffer.
    * updated to use GC counters
    - removed hb_codeblockDelete() function.
    * changed hb_codeblockDeleteGarbage() to be safe for recursive call
    - removed restoring statics base in hb_codeblockEvaluate()

  * harbour/source/vm/debug.c
    * cleaned and optimized some code

  * harbour/source/vm/dynlibhb.c
    + added support for loading and unloading dynamic libraries with PCODE
    + added support for dynamic libraries in Linux (.so)
    + added HB_LIBERROR() - returns error string with last error in
      HB_LIBLOAD() - now works only in Linux

  * harbour/source/vm/dynsym.c
    * separated profiler code
    + use pDynSym->pSymbol->value.pFunPtr instead of pDynSym->pFunPtr

  * harbour/source/vm/estack.c
    + added #include "hbapicls.h"
    + added hb_stackPushReturn(), hb_stackPopReturn()

  * harbour/source/vm/eval.c
  * harbour/source/vm/extend.c
    * use stack function/macros instead of direct accessing HB_STACK members
    + added hb_retptrGC()

  * harbour/source/vm/fm.c
    + added hb_xRefInc(), hb_xRefDec(), hb_xRefFree(), hb_xRefCount(),
      hb_xRefResize() functions. They are internal Harbour VM functions
      and covered by internal API macro. Should not be used by 3-rd
      party code because we may change them in the future or even remove.

  * harbour/source/vm/garbage.c
    + added hb_gcRefInc(), hb_gcRefDec(), hb_gcRefFree(), hb_gcRefCount()
      functions. These are also internal function which corresponds to the
      above hb_x*() ones.
    * changed Step 3 of GC pass and execution of clean-up function to be
      safe for recursive calls and additional activity.
      Ryszard see my note and s_pDeletedBlock
    + added GC inspected HB_IT_POINTER items.

  * harbour/source/vm/hvm.c
    * separated profiler code
    * separated class code from HVM
    + added support for loading and unloading libraries with PCODE modules
    + added support for pure PCODE functions
    + added support for string item resizing and preallocating string buffer
    + added some new multipcode functions for -gc3 optimization
    + added hb_vmProcessSymbolsExt(), hb_vmRegisterSymbols(),
      hb_vmFreeSymbols(), hb_vmBeginSymbolGroup(),
      hb_vmInitSymbolGroup(), hb_vmExitSymbolGroup()
    * changed hb_vmProcessSymbols() to return address of register symbol
      table. For normal code it's the same address as given in parameter
      so it does not break backward binary compatibility.
    + use new code for operator overloading
    * changed FOR EACH to dynamically check iteration scope in
      hb_vmEnumNext()/hb_vmEnumPrev() and remove max number of iteration
      stored on HVM stack
    * temporary use hb_itemUnRefOnce() instead of hb_itemUnRefRefer() as
      workaround for possible GPF. I'd like Ryszard will chose the final
      version

  * harbour/source/vm/itemapi.c
    + added support for preallocating string buffer size
    + added hb_itemUnShareString() and hb_itemReSizeString()
    * updated string items to use counters allocated with each
      memory block by xh_xgrab()/hb_xalloc()
    * updated codeblock and array items to use GC reference counters
    + added GC inspected HB_IT_POINTER items.
    * generate RT error in hb_itemUnreOnce() for enumerator items
      out of scope
    + added hb_itemPutPtrGC()

  * harbour/source/vm/macro.c
    * use pDynSym->pSymbol->value.pFunPtr instead of pDynSym->pFunPtr

  * harbour/source/vm/maindllp.c
    * updated usage of hb_vmProcessSymbols()
    * added hb_vmProcessSymbolsExt()

  * harbour/source/vm/memvars.c
    + added hb_memvarGetItem()
    - removed hb_memvarValueDecGarbageRef()

  * harbour/source/vm/proc.c
    + added #include "hbapicls.h"

  * harbour/source/vm/pvalue.c
    * do not access stack structure directly but use hb_stack*()
      functions (macros)

  * harbour/source/vm/runner.c
    * synced with xHarbour

  * harbour/utils/hbtest/rt_misc.prg
   * added overloading "=" operator. Now "==" and "=" can be overloaded
     with different methods

   All names I used for new functions/macros can be changed. Please look at
   it and if you will some better propositions then we can use it.
   hb_vmProcessSymbolsExt() is added but not used yet.
2006-06-14 11:21:26 +00:00

764 lines
21 KiB
C

/*
* $Id$
*/
/*
* Harbour Project source code:
* Math functions
*
* Copyright 1999 Matthew Hamilton <mhamilton@bunge.com.au>
*
* Functions for user defined math error handlers, changes and fixes
* Copyright 2001/2002 IntTec GmbH, Freiburg, Germany,
* Author: Martin Vogel <vogel@inttec.de>
*
* www - http://www.harbour-project.org
*
* 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, 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 software; see the file COPYING. If not, write to
* the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
* Boston, MA 02111-1307 USA (or visit the web site http://www.gnu.org/).
*
* As a special exception, the Harbour Project gives permission for
* additional uses of the text contained in its release of Harbour.
*
* The exception is that, if you link the Harbour libraries with other
* files to produce an executable, this does not by itself cause the
* resulting executable to be covered by the GNU General Public License.
* Your use of that executable is in no way restricted on account of
* linking the Harbour library code into it.
*
* This exception does not however invalidate any other reasons why
* the executable file might be covered by the GNU General Public License.
*
* This exception applies only to the code released by the Harbour
* Project under the name Harbour. If you copy code from other
* Harbour Project or Free Software Foundation releases into a copy of
* Harbour, as the General Public License permits, the exception does
* not apply to the code that you add in this way. To avoid misleading
* anyone as to the status of such modified files, you must delete
* this exception notice from them.
*
* If you write modifications of your own for Harbour, it is your choice
* whether to permit this exception to apply to your modifications.
* If you do not wish that, delete this exception notice.
*
*/
#if defined(__DJGPP__)
#include <libm/math.h>
_LIB_VERSION_TYPE _LIB_VERSION = _XOPEN_;
#else
#include <math.h>
#endif
#include "hbapi.h"
#include "hbapiitm.h"
#include "hbapierr.h"
#include "hbvm.h"
#include "hbmath.h"
#if defined(HB_MATH_ERRNO)
# include <errno.h>
#endif
#if defined(HB_OS_SUNOS)
# include <ieeefp.h>
#endif
/*
* ************************************************************
* Harbour Math functions Part I:
* handling math errors, C math lib redirection
* ************************************************************
*/
static HB_MATH_EXCEPTION s_hb_exc = {HB_MATH_ERR_NONE, "", "", 0.0, 0.0, 0.0, 1, 0, 0};
/* reset math error information */
void hb_mathResetError (void)
{
HB_TRACE (HB_TR_DEBUG, ("hb_mathResetError()"));
s_hb_exc.type = HB_MATH_ERR_NONE;
s_hb_exc.funcname = "";
s_hb_exc.error = "";
s_hb_exc.arg1 = 0.0;
s_hb_exc.arg2 = 0.0;
s_hb_exc.retval = 0.0;
s_hb_exc.retvalwidth = -1; /* we don't know */
s_hb_exc.retvaldec = -1; /* use standard SET DECIMALS */
s_hb_exc.handled = 1;
return;
}
/* get last math error */
int hb_mathGetLastError (HB_MATH_EXCEPTION * phb_exc)
{
HB_TRACE (HB_TR_DEBUG, ("hb_mathGetLastError(%p)", phb_exc));
if (phb_exc != NULL)
{
phb_exc->type = s_hb_exc.type;
phb_exc->funcname = s_hb_exc.funcname;
phb_exc->error = s_hb_exc.error;
phb_exc->arg1 = s_hb_exc.arg1;
phb_exc->arg2 = s_hb_exc.arg2;
phb_exc->retval = s_hb_exc.retval;
phb_exc->retvalwidth = s_hb_exc.retvalwidth;
phb_exc->retvaldec = s_hb_exc.retvaldec;
phb_exc->handled = s_hb_exc.handled;
}
return (s_hb_exc.type);
}
/* is it reasonable to install math error handlers ? This depends on the C math lib we are using ! */
int hb_mathIsMathErr (void)
{
HB_TRACE (HB_TR_DEBUG, ("hb_mathIsMathErr()"));
#if defined(HB_MATH_HANDLER)
return (1);
#else
return (0);
#endif
}
/* route C math lib errors to Harbour error handling */
#if defined(HB_MATH_HANDLER)
int matherr (struct exception * err)
{
int retval;
HB_MATH_HANDLERPROC mathHandler;
HB_TRACE (HB_TR_DEBUG, ("matherr(%p)", err));
/* map math error types */
switch (err->type)
{
case DOMAIN:
{
s_hb_exc.type = HB_MATH_ERR_DOMAIN;
s_hb_exc.error = "Argument not in domain of function";
}; break;
case SING:
{
s_hb_exc.type = HB_MATH_ERR_SING;
s_hb_exc.error = "Calculation results in singularity";
}; break;
case OVERFLOW:
{
s_hb_exc.type = HB_MATH_ERR_OVERFLOW;
s_hb_exc.error = "Calculation result too large to represent";
}; break;
case UNDERFLOW:
{
s_hb_exc.type = HB_MATH_ERR_UNDERFLOW;
s_hb_exc.error = "Calculation result too small to represent";
}; break;
case TLOSS:
{
s_hb_exc.type = HB_MATH_ERR_TLOSS;
s_hb_exc.error = "Total loss of significant digits";
}; break;
case PLOSS:
{
s_hb_exc.type = HB_MATH_ERR_PLOSS;
s_hb_exc.error = "Partial loss of significant digits";
}; break;
default:
{
s_hb_exc.type = HB_MATH_ERR_UNKNOWN;
s_hb_exc.error = "Unknown math error";
}; break;
}
s_hb_exc.funcname = (char *)err->name; /* (char *) Avoid warning in DJGPP */
s_hb_exc.arg1 = err->arg1;
s_hb_exc.arg2 = err->arg2;
s_hb_exc.retval = err->retval;
s_hb_exc.handled = 0;
mathHandler = hb_mathGetHandler();
if (mathHandler != NULL)
{
retval = (*(mathHandler))(&s_hb_exc);
err->retval = s_hb_exc.retval;
}
else
{
/* there is no custom math handler */
retval = 1; /* don't print any message, don't set errno and use return value provided by C RTL */
}
return (retval);
}
#elif defined(HB_MATH_ERRNO)
int hb_mathErrSet( double dResult, double arg1, double arg2, char * szFunc, int errCode )
{
HB_MATH_HANDLERPROC mathHandler;
HB_TRACE (HB_TR_DEBUG, ("hb_mathErrSet(%f, %d)", dResult, errCode));
switch (errCode)
{
case EDOM:
case ERANGE:
case EOVERFLOW:
break;
default:
if ( isnan( dResult ) )
{
errCode = EDOM;
}
#if defined(HB_OS_SUNOS)
else if ( !finite( dResult ) )
#else
else if ( isinf( dResult ) )
#endif
{
errCode = ERANGE;
}
}
if ( errCode == 0 )
{
return 0;
}
hb_mathResetError();
/* map math error types */
switch (errCode)
{
case EDOM:
s_hb_exc.type = HB_MATH_ERR_DOMAIN;
s_hb_exc.error = "Argument not in domain of function";
break;
case ERANGE:
s_hb_exc.type = HB_MATH_ERR_SING;
s_hb_exc.error = "Calculation results in singularity";
break;
case EOVERFLOW:
s_hb_exc.type = HB_MATH_ERR_OVERFLOW;
s_hb_exc.error = "Calculation result too large to represent";
break;
default:
s_hb_exc.type = HB_MATH_ERR_UNKNOWN;
s_hb_exc.error = "Unknown math error";
break;
}
s_hb_exc.funcname = szFunc;
s_hb_exc.arg1 = arg1;
s_hb_exc.arg2 = arg2;
s_hb_exc.retval = dResult;
s_hb_exc.handled = 0;
mathHandler = hb_mathGetHandler();
if (mathHandler != NULL)
{
(*(mathHandler))(&s_hb_exc);
}
return 1;
}
#endif
/*
* ************************************************************
* Harbour Math functions Part II:
* handling math errors, Harbour default handling routine
* ************************************************************
*/
static int s_hb_matherr_mode = HB_MATH_ERRMODE_DEFAULT; /* TODO: make this thread safe */
/* set error handling mode of hb_matherr() */
int hb_mathSetErrMode (int imode)
{
int oldmode;
HB_TRACE (HB_TR_DEBUG, ("hb_mathSetErrMode (%i)", imode));
oldmode = s_hb_matherr_mode;
if ((imode == HB_MATH_ERRMODE_DEFAULT) ||
(imode == HB_MATH_ERRMODE_CDEFAULT) ||
(imode == HB_MATH_ERRMODE_USER) ||
(imode == HB_MATH_ERRMODE_USERDEFAULT) ||
(imode == HB_MATH_ERRMODE_USERCDEFAULT))
{
s_hb_matherr_mode = imode;
}
return (oldmode);
}
/* get error handling mode of hb_matherr() */
int hb_mathGetErrMode (void)
{
HB_TRACE (HB_TR_DEBUG, ("hb_mathGetErrMode()"));
return (s_hb_matherr_mode);
}
/* Harbour equivalent to mathSet/GetErrMode */
HB_FUNC( HB_MATHERMODE ) /* ([<nNewMode>]) -> <nOldMode> */
{
hb_retni (hb_mathGetErrMode());
/* set new mode */
if (ISNUM (1))
{
hb_mathSetErrMode( hb_parni( 1 ) );
}
return;
}
/* Harbour default math error handling routine */
int hb_matherr (HB_MATH_EXCEPTION * pexc)
{
int mode = hb_mathGetErrMode();
HB_TRACE(HB_TR_DEBUG, ("hb_matherr(%p)",pexc));
if ((pexc == NULL) ||
((pexc != NULL) && (pexc->handled != 0)))
{
/* error already handled by other handlers ! */
return (1);
}
if ((mode == HB_MATH_ERRMODE_USER) || (mode == HB_MATH_ERRMODE_USERDEFAULT) ||
(mode == HB_MATH_ERRMODE_USERCDEFAULT))
{
PHB_ITEM pMatherrResult;
PHB_ITEM pArg1 = hb_itemPutND(NULL, pexc->arg1);
PHB_ITEM pArg2 = hb_itemPutND(NULL, pexc->arg2);
PHB_ITEM pError;
/* create an error object */
/* NOTE: In case of HB_MATH_ERRMODE_USER[C]DEFAULT, I am setting both EF_CANSUBSTITUTE and EF_CANDEFAULT to .T. here.
This is forbidden according to the original Cl*pper docs, but I think this reflects the situation best here:
The error handler can either substitute the errorneous value (by returning a numeric value) or choose the
default error handling (by returning .F., as usual) [martin vogel]*/
pError = hb_errRT_New_Subst (ES_ERROR, "MATH", EG_NUMERR, pexc->type,
pexc->error, pexc->funcname, 0, EF_CANSUBSTITUTE |
(mode == HB_MATH_ERRMODE_USER ? 0: EF_CANDEFAULT));
/* Assign the new array to the object data item. */
/* NOTE: Unfortunately, we cannot decide whether one or two parameters have been used when the
math function has been called, so we always take two */
hb_errPutArgs( pError, 2, pArg1, pArg2 );
/* launch error codeblock */
pMatherrResult = hb_errLaunchSubst (pError);
hb_errRelease (pError);
if (pMatherrResult != NULL)
{
if (HB_IS_NUMERIC (pMatherrResult))
{
pexc->retval = hb_itemGetND (pMatherrResult);
hb_itemGetNLen (pMatherrResult, &(pexc->retvalwidth), &(pexc->retvaldec));
pexc->handled = 1;
}
hb_itemRelease (pMatherrResult);
}
}
/* math exception not handled by Harbour error routine above ? */
if (pexc->handled == 0)
{
int iret;
switch (mode)
{
case HB_MATH_ERRMODE_USER:
{
/* user failed to handle the math exception, so quit the app [yes, that's the meaning of this mode !!] */
iret = 0;
hb_vmRequestQuit();
}; break;
case HB_MATH_ERRMODE_DEFAULT:
case HB_MATH_ERRMODE_USERDEFAULT:
{
/* return 1 to suppress C RTL error msgs, but leave error handling to the calling Harbour routine */
iret = 1;
}; break;
case HB_MATH_ERRMODE_CDEFAULT:
case HB_MATH_ERRMODE_USERCDEFAULT:
{
/* use the correction value supplied in pexc->retval */
pexc->handled = 1;
iret = 1;
}; break;
default:
iret = 1;
}
return (iret);
}
return (1); /* error handling successful */
}
/*
* ************************************************************
* Harbour Math functions Part III:
* (de)installing and (de)activating custom math error handlers
* ************************************************************
*/
/* static slot for current math error handler, this is hb_matherr by default */
static HB_MATH_HANDLERPROC s_mathHandlerProc = hb_matherr; /* TODO: make this thread safe */
/* install a harbour-like math error handler (that will be called by the matherr() function), return old handler */
HB_MATH_HANDLERPROC hb_mathSetHandler (HB_MATH_HANDLERPROC handlerproc)
{
HB_MATH_HANDLERPROC oldHandlerProc;
HB_TRACE (HB_TR_DEBUG, ("hb_mathSetHandler (%p)", handlerproc));
oldHandlerProc = s_mathHandlerProc;
s_mathHandlerProc = handlerproc;
return ((HB_MATH_HANDLERPROC)oldHandlerProc);
}
/* get current harbour-like math error handler */
HB_MATH_HANDLERPROC hb_mathGetHandler (void)
{
HB_TRACE (HB_TR_DEBUG, ("hb_mathGetHandler ()"));
return ((HB_MATH_HANDLERPROC)s_mathHandlerProc);
}
/*
* ************************************************************
* Harbour Math functions Part IV:
* example of hb_mathSet/GetHandler: add a new math handler that
* calls a given codeblock for every math error
* ************************************************************
*/
static PHB_ITEM spMathErrorBlock = NULL;
static HB_MATH_HANDLERPROC sPrevMathHandler = NULL;
static int hb_matherrblock( HB_MATH_EXCEPTION * pexc )
{
int retval;
/* call codeblock for both case: handled and unhandled exceptions */
if( spMathErrorBlock != NULL )
{
PHB_ITEM pArray, pRet;
PHB_ITEM pType, pFuncname, pError, pArg1, pArg2, pRetval, pHandled;
pType = hb_itemPutNI (NULL, pexc->type);
pFuncname = hb_itemPutC (NULL, pexc->funcname);
pError = hb_itemPutC (NULL, pexc->error);
pArg1 = hb_itemPutND (NULL, pexc->arg1);
pArg2 = hb_itemPutND (NULL, pexc->arg2);
pRetval = hb_itemPutNDLen (NULL, pexc->retval, pexc->retvalwidth, pexc->retvaldec);
pHandled = hb_itemPutL (NULL, pexc->handled);
pArray = hb_itemArrayNew (2);
hb_itemArrayPut (pArray, 1, pRetval);
hb_itemArrayPut (pArray, 2, pHandled);
/* launch error codeblock that can
a) change the members of the array = {dRetval, lHandled} to set the return value of the math C RTL routine and
the <exception handled flag> and it
b) can return an integer value to set the return value of matherr().
NOTE that these values are only used if lHandled was .F. and is set to .T. within the codeblock */
pRet = hb_itemDo (spMathErrorBlock, 6, pType, pFuncname, pError, pArg1, pArg2, pArray);
hb_itemRelease (pType);
hb_itemRelease (pFuncname);
hb_itemRelease (pError);
hb_itemRelease (pArg1);
hb_itemRelease (pArg2);
hb_itemRelease (pRetval);
hb_itemRelease (pHandled);
if( pexc->handled )
{
/* math exception has already been handled, so codeblock call above was only informative */
retval = 1;
}
else
{
/* exception handled by codeblock ? */
pHandled = hb_itemArrayGet (pArray, 2);
if (pHandled != NULL)
{
pexc->handled = hb_itemGetL( pHandled );
hb_itemRelease( pHandled );
}
if (pexc->handled)
{
/* YES ! */
/* extract retval for math routine and matherr() */
pRetval = hb_itemArrayGet (pArray, 1);
if (pRetval != NULL)
{
pexc->retval = hb_itemGetND (pRetval);
hb_itemGetNLen (pRetval, &(pexc->retvalwidth), &(pexc->retvaldec));
hb_itemRelease (pRetval);
}
if ((pRet != NULL) && HB_IS_NUMERIC (pRet))
{
retval = hb_itemGetNI (pRet); /* block may also return 0 to force C math lib warnings */
hb_itemRelease (pRet);
}
else
{
retval = 1; /* default return value to suppress C math lib warnings */
}
}
else
{
/* NO ! */
retval = 1;
}
}
hb_itemRelease (pArray);
}
else
{
retval = 1; /* default return value to suppress C math lib warnings */
}
if (sPrevMathHandler != NULL)
{
if (pexc->handled)
{
/* the error is handled, so simply inform the previous handler */
(*sPrevMathHandler)(pexc);
}
else
{
/* else go on error handling within previous handler */
retval = (*sPrevMathHandler)(pexc);
}
}
return retval;
}
/* set/get math error block */
HB_FUNC( HB_MATHERBLOCK ) /* ([<nNewErrorBlock>]) -> <nOldErrorBlock> */
{
/* immediately install hb_matherrblock and keep it permanently installed !
This is not dangerous because hb_matherrorblock will always call the previous error handler */
if (sPrevMathHandler == NULL)
{
sPrevMathHandler = hb_mathSetHandler (hb_matherrblock);
}
/* return old math handler */
if (spMathErrorBlock == NULL)
{
hb_ret();
}
else
{
hb_itemReturn (spMathErrorBlock);
}
if (hb_pcount() > 0)
{
/* set new error block */
PHB_ITEM pNewErrorBlock = hb_param (1, HB_IT_BLOCK);
if (pNewErrorBlock != NULL)
{
if (spMathErrorBlock == NULL)
{
spMathErrorBlock = hb_itemNew (NULL);
}
hb_itemCopy (spMathErrorBlock, pNewErrorBlock);
}
else
{
/* a parameter other than a block has been passed -> delete error handler ! */
if (spMathErrorBlock != NULL)
{
hb_itemRelease (spMathErrorBlock);
spMathErrorBlock = NULL;
}
}
}
return;
}
/*
* ************************************************************
* Harbour Math functions Part V:
* EXP(), LOG(), SQRT()
* ************************************************************
*/
HB_FUNC( EXP )
{
if( ISNUM( 1 ) )
{
HB_MATH_EXCEPTION hb_exc;
double dResult;
#if defined(HB_MATH_ERRNO)
errno = 0;
dResult = exp( hb_parnd( 1 ) );
if ( hb_mathErrSet( dResult, hb_parnd (1), 0.0, "EXP", errno ) )
#else
hb_mathResetError();
dResult = exp( hb_parnd( 1 ) );
if ( hb_mathIsMathErr() )
#endif
{
/* the C-RTL provides a kind of matherr() mechanism */
int iLastError = hb_mathGetLastError( &hb_exc );
if( iLastError != HB_MATH_ERR_NONE )
{
if( hb_exc.handled )
{
hb_retndlen( hb_exc.retval, hb_exc.retvalwidth, hb_exc.retvaldec );
}
else
{
/* math exception is up to the Harbour function, so do this as Clipper compatible as possible */
if ( iLastError == HB_MATH_ERR_OVERFLOW )
{
hb_retndlen( HUGE_VAL, -1, -1 );
}
else
{
hb_retnd( 0.0 );
}
}
return;
}
}
hb_retnd( dResult );
}
else
{
hb_errRT_BASE_SubstR( EG_ARG, 1096, NULL, "EXP", 1, hb_paramError( 1 ) );
}
}
HB_FUNC( LOG )
{
if( ISNUM ( 1 ) )
{
HB_MATH_EXCEPTION hb_exc;
double dResult;
#if defined(HB_MATH_ERRNO)
errno = 0;
dResult = log( hb_parnd( 1 ) );
if ( hb_mathErrSet( dResult, hb_parnd (1), 0.0, "LOG", errno ) )
#else
hb_mathResetError();
dResult = log( hb_parnd( 1 ) );
if ( hb_mathIsMathErr() )
#endif
{
/* the C-RTL provides a kind of matherr() mechanism */
int iLastError = hb_mathGetLastError( &hb_exc );
if( iLastError != HB_MATH_ERR_NONE )
{
if( hb_exc.handled )
{
hb_retndlen( hb_exc.retval, hb_exc.retvalwidth, hb_exc.retvaldec );
}
else
{
/* math exception is up to the Harbour function, so do this as Clipper compatible as possible */
switch( iLastError )
{
case HB_MATH_ERR_SING: /* argument to log was 0.0 */
case HB_MATH_ERR_DOMAIN: /* argument to log was < 0.0 */
{
hb_retndlen( -HUGE_VAL, -1, -1 ); /* return -infinity */
}; break;
default:
{
hb_retnd( 0.0 );
}
}
}
return;
}
}
hb_retnd( dResult );
}
else
{
hb_errRT_BASE_SubstR( EG_ARG, 1095, NULL, "LOG", 1, hb_paramError( 1 ) );
}
}
HB_FUNC( SQRT )
{
if( ISNUM( 1 ) )
{
HB_MATH_EXCEPTION hb_exc;
double dResult;
#if defined(HB_MATH_ERRNO)
errno = 0;
dResult = sqrt( hb_parnd( 1 ) );
if ( hb_mathErrSet( dResult, hb_parnd( 1 ), 0.0, "SQRT", errno ) )
#else
hb_mathResetError();
dResult = sqrt( hb_parnd( 1 ) );
if ( hb_mathIsMathErr() )
#endif
{
/* the C-RTL provides a kind of matherr() mechanism */
int iLastError = hb_mathGetLastError( &hb_exc );
if( iLastError != HB_MATH_ERR_NONE )
{
if( hb_exc.handled )
{
hb_retndlen( hb_exc.retval, hb_exc.retvalwidth, hb_exc.retvaldec );
}
else
{
/* math exception is up to the Harbour function, so do this as Clipper compatible as possible */
hb_retnd( 0.0 ); /* return 0.0 on all errors (all (?) of type DOMAIN) */
}
return;
}
}
hb_retnd( dResult );
}
else
{
hb_errRT_BASE_SubstR( EG_ARG, 1097, NULL, "SQRT", 1, hb_paramError( 1 ) );
}
}