/* * $Id$ Copyright(C) 1999 by Antonio Linares. 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. You can contact me at: alinares@fivetech.com */ #include "extend.h" #include "ctoharb.h" #include "itemapi.h" #include "errorapi.h" #include "langapi.h" extern HARBOUR HB_ERRORNEW( void ); /* NOTE: This is called via its symbol name, so we should make sure */ /* that it gets linked. WARNING ! DON'T make this function static. */ void hb_errForceLink() { HB_ERRORNEW(); } PHB_ITEM hb_errNew( void ) { PHB_ITEM pReturn = hb_itemNew( NULL ); hb_vmPushSymbol( hb_dynsymGet( "ERRORNEW" )->pSymbol ); hb_vmPushNil(); hb_vmDo( 0 ); hb_itemCopy( pReturn, &stack.Return ); return pReturn; } WORD hb_errLaunch( PHB_ITEM pError ) { WORD wRetVal; if( pError ) { EVALINFO eval; PHB_ITEM pBlock; PHB_ITEM pObject; PHB_ITEM pResult; /* Check if we have a valid error handler */ if( ! IS_BLOCK( &errorBlock ) ) hb_errInternal( 9999, "No ERRORBLOCK() for error", NULL, NULL ); /* Launch the error handler: "lResult := EVAL( bErrorBlock, oError )" */ pBlock = hb_itemNew( NULL ); pObject = hb_itemNew( NULL ); hb_itemCopy( pBlock, &errorBlock ); hb_itemCopy( pObject, pError ); hb_evalNew( &eval, pBlock ); hb_evalPutParam( &eval, pObject ); pResult = hb_evalLaunch( &eval ); hb_evalRelease( &eval ); /* Check results */ /* TODO: Determine if there was a BREAK in the error handler. */ if( FALSE ) { hb_itemRelease( pResult ); /* TODO: Detect sequence level properly */ if( FALSE ) { /* TODO: Initiate a jump to the closest RECOVER statement */ wRetVal = E_BREAK; } else { /* TODO: QUIT correctly, without any message */ exit( 1 ); } } else { BOOL bFailure = FALSE; USHORT uiFlags = hb_errGetFlags( pError ); /* If the error block didn't return a logical value, */ /* or the canSubstitute flag has been set, consider it as a failure */ if( ! IS_LOGICAL( pResult ) || ( uiFlags & EF_CANSUBSTITUTE ) ) { bFailure = TRUE; } else { wRetVal = hb_itemGetL( pResult ) ? E_RETRY : E_DEFAULT; if( ( wRetVal == E_DEFAULT && !( uiFlags & EF_CANDEFAULT ) ) || ( wRetVal == E_RETRY && !( uiFlags & EF_CANRETRY ) ) ) { bFailure = TRUE; } } hb_itemRelease( pResult ); if( bFailure ) hb_errInternal( 9999, "Error recovery failure", NULL, NULL ); } } else wRetVal = E_RETRY; /* Clipper does this, undocumented */ return wRetVal; } /* This error launcher should be used in those situations, where the error handler is expected to return a value to be substituted as the result of a failed operation. */ /* NOTE: This should only be called when the EF_CANSUBSTITUE flag was set Since it this case the error handler will return the value to be substituted */ /* NOTE: The item pointer returned should be hb_itemRelease()-d by the caller. */ PHB_ITEM hb_errLaunchSubst( PHB_ITEM pError ) { PHB_ITEM pResult; if( pError ) { EVALINFO eval; PHB_ITEM pBlock; PHB_ITEM pObject; /* Check if we have a valid error handler */ if( ! IS_BLOCK( &errorBlock ) ) hb_errInternal( 9999, "No ERRORBLOCK() for error", NULL, NULL ); /* Launch the error handler: "xResult := EVAL( bErrorBlock, oError )" */ pBlock = hb_itemNew( NULL ); pObject = hb_itemNew( NULL ); hb_itemCopy( pBlock, &errorBlock ); hb_itemCopy( pObject, pError ); hb_evalNew( &eval, pBlock ); hb_evalPutParam( &eval, pObject ); pResult = hb_evalLaunch( &eval ); hb_evalRelease( &eval ); /* Check results */ /* TODO: Determine if there was a BREAK in the error handler. */ if( FALSE ) { /* TODO: Detect sequence level properly */ if( FALSE ) { /* TODO: Initiate a jump to the closest RECOVER statement */ /* QUESTION: Will Clipper return to the caller in this case ? */ } else { hb_itemRelease( pResult ); pResult = NULL; /* TODO: QUIT correctly, without any message */ exit( 1 ); } } else { /* If the canSubstitute flag has not been set, consider it as a failure. */ if( ! ( hb_errGetFlags( pError ) & EF_CANSUBSTITUTE ) ) { hb_errInternal( 9999, "Error recovery failure", NULL, NULL ); } } } else pResult = hb_itemNew( NULL ); return pResult; } void hb_errRelease( PHB_ITEM pError ) { /* NOTE: NULL pointer is checked by hb_itemRelease() */ hb_itemRelease( pError ); } char * hb_errGetDescription( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "DESCRIPTION" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asString.value; } PHB_ITEM hb_errPutDescription( PHB_ITEM pError, char * szDescription ) { hb_vmPushSymbol( hb_dynsymGet( "_DESCRIPTION" )->pSymbol ); hb_vmPush( pError ); hb_vmPushString( szDescription, strlen( szDescription ) ); hb_vmDo( 1 ); return pError; } char * hb_errGetFileName( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "FILENAME" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asString.value; } PHB_ITEM hb_errPutFileName( PHB_ITEM pError, char * szFileName ) { hb_vmPushSymbol( hb_dynsymGet( "_FILENAME" )->pSymbol ); hb_vmPush( pError ); hb_vmPushString( szFileName, strlen( szFileName ) ); hb_vmDo( 1 ); return pError; } USHORT hb_errGetGenCode( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "GENCODE" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asInteger.value; } PHB_ITEM hb_errPutGenCode( PHB_ITEM pError, USHORT uiGenCode ) { hb_vmPushSymbol( hb_dynsymGet( "_GENCODE" )->pSymbol ); hb_vmPush( pError ); hb_vmPushInteger( uiGenCode ); hb_vmDo( 1 ); return pError; } char * hb_errGetOperation( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "OPERATION" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asString.value; } PHB_ITEM hb_errPutOperation( PHB_ITEM pError, char * szOperation ) { hb_vmPushSymbol( hb_dynsymGet( "_OPERATION" )->pSymbol ); hb_vmPush( pError ); hb_vmPushString( szOperation, strlen( szOperation ) ); hb_vmDo( 1 ); return pError; } USHORT hb_errGetOsCode( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "OSCODE" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asInteger.value; } PHB_ITEM hb_errPutOsCode( PHB_ITEM pError, USHORT uiOsCode ) { hb_vmPushSymbol( hb_dynsymGet( "_OSCODE" )->pSymbol ); hb_vmPush( pError ); hb_vmPushInteger( uiOsCode ); hb_vmDo( 1 ); return pError; } USHORT hb_errGetSeverity( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "SEVERITY" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asInteger.value; } PHB_ITEM hb_errPutSeverity( PHB_ITEM pError, USHORT uiSeverity ) { hb_vmPushSymbol( hb_dynsymGet( "_SEVERITY" )->pSymbol ); hb_vmPush( pError ); hb_vmPushInteger( uiSeverity ); hb_vmDo( 1 ); return pError; } USHORT hb_errGetSubCode( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "SUBCODE" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asInteger.value; } PHB_ITEM hb_errPutSubCode( PHB_ITEM pError, USHORT uiSubCode ) { hb_vmPushSymbol( hb_dynsymGet( "_SUBCODE" )->pSymbol ); hb_vmPush( pError ); hb_vmPushInteger( uiSubCode ); hb_vmDo( 1 ); return pError; } char * hb_errGetSubSystem( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "SUBSYSTEM" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asString.value; } PHB_ITEM hb_errPutSubSystem( PHB_ITEM pError, char * szSubSystem ) { hb_vmPushSymbol( hb_dynsymGet( "_SUBSYSTEM" )->pSymbol ); hb_vmPush( pError ); hb_vmPushString( szSubSystem, strlen( szSubSystem ) ); hb_vmDo( 1 ); return pError; } USHORT hb_errGetTries( PHB_ITEM pError ) { hb_vmPushSymbol( hb_dynsymGet( "TRIES" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); return stack.Return.item.asInteger.value; } PHB_ITEM hb_errPutTries( PHB_ITEM pError, USHORT uiTries ) { hb_vmPushSymbol( hb_dynsymGet( "_TRIES" )->pSymbol ); hb_vmPush( pError ); hb_vmPushInteger( uiTries ); hb_vmDo( 1 ); return pError; } USHORT hb_errGetFlags( PHB_ITEM pError ) { USHORT uiFlags = EF_NONE; /* ; */ hb_vmPushSymbol( hb_dynsymGet( "CANRETRY" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); if( stack.Return.item.asLogical.value ) uiFlags |= EF_CANRETRY; /* ; */ hb_vmPushSymbol( hb_dynsymGet( "CANSUBSTITUTE" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); if( stack.Return.item.asLogical.value ) uiFlags |= EF_CANSUBSTITUTE; /* ; */ hb_vmPushSymbol( hb_dynsymGet( "CANDEFAULT" )->pSymbol ); hb_vmPush( pError ); hb_vmDo( 0 ); if( stack.Return.item.asLogical.value ) uiFlags |= EF_CANDEFAULT; /* ; */ return uiFlags; } PHB_ITEM hb_errPutFlags( PHB_ITEM pError, USHORT uiFlags ) { hb_vmPushSymbol( hb_dynsymGet( "_CANRETRY" )->pSymbol ); hb_vmPush( pError ); hb_vmPushLogical( ( uiFlags & EF_CANRETRY ) ? TRUE : FALSE ); hb_vmDo( 1 ); /* ; */ hb_vmPushSymbol( hb_dynsymGet( "_CANSUBSTITUTE" )->pSymbol ); hb_vmPush( pError ); hb_vmPushLogical( ( uiFlags & EF_CANSUBSTITUTE ) ? TRUE : FALSE ); hb_vmDo( 1 ); /* ; */ hb_vmPushSymbol( hb_dynsymGet( "_CANDEFAULT" )->pSymbol ); hb_vmPush( pError ); hb_vmPushLogical( ( uiFlags & EF_CANDEFAULT ) ? TRUE : FALSE ); hb_vmDo( 1 ); /* ; */ return pError; } /* Wrappers for hb_errLaunch() */ static WORD hb_errRT_New ( USHORT uiSeverity, char * szSubSystem, ULONG ulGenCode, ULONG ulSubCode, char * szDescription, char * szOperation, USHORT uiOsCode, USHORT uiFlags ) { PHB_ITEM pError = hb_errNew(); WORD wRetVal; hb_errPutSeverity( pError, uiSeverity ); hb_errPutSubSystem( pError, szSubSystem ); hb_errPutGenCode( pError, ulGenCode ); hb_errPutSubCode( pError, ulSubCode ); hb_errPutDescription( pError, szDescription ? szDescription : hb_langDGetErrorDesc( ulGenCode ) ); hb_errPutOperation( pError, szOperation ); hb_errPutOsCode( pError, uiOsCode ); hb_errPutFlags( pError, uiFlags ); wRetVal = hb_errLaunch( pError ); hb_errRelease( pError ); return wRetVal; } HARBOUR HB___ERRRT_BASE( void ) { hb_errRT_BASE( ( ULONG ) hb_parnl( 1 ), ( ULONG ) hb_parnl( 2 ), ISCHAR( 3 ) ? hb_parc( 3 ) : NULL, hb_parc( 4 ) ); } void hb_errRT_BASE( ULONG ulGenCode, ULONG ulSubCode, char * szDescription, char * szOperation ) { hb_errRT_New( ES_ERROR, HB_ERR_SS_BASE, ulGenCode, ulSubCode, szDescription, szOperation, 0, EF_NONE ); } WORD hb_errRT_BASE_Ext1( ULONG ulGenCode, ULONG ulSubCode, char * szDescription, char * szOperation, USHORT uiOsCode, USHORT uiFlags ) { return hb_errRT_New( ES_ERROR, HB_ERR_SS_BASE, ulGenCode, ulSubCode, szDescription, szOperation, uiOsCode, uiFlags ); } void hb_errRT_TERMINAL( ULONG ulGenCode, ULONG ulSubCode, char * szDescription, char * szOperation ) { hb_errRT_New( ES_ERROR, HB_ERR_SS_TERMINAL, ulGenCode, ulSubCode, szDescription, szOperation, 0, EF_NONE ); } void hb_errRT_DBCMD( ULONG ulGenCode, ULONG ulSubCode, char * szDescription, char * szOperation ) { hb_errRT_New( ES_ERROR, HB_ERR_SS_DBCMD, ulGenCode, ulSubCode, szDescription, szOperation, 0, EF_NONE ); } void hb_errRT_TOOLS( ULONG ulGenCode, ULONG ulSubCode, char * szDescription, char * szOperation ) { hb_errRT_New( ES_ERROR, HB_ERR_SS_BASE, ulGenCode, ulSubCode, szDescription, szOperation, 0, EF_NONE ); } /* NOTE: Use as minimal calls from here, as possible. */ /* Don't allocate memory from this function. */ void hb_errInternal( ULONG ulIntCode, char * szText, char * szPar1, char * szPar2 ) { printf( hb_consoleGetNewLine() ); printf( "Internal error %lu: ", ulIntCode ); printf( szText != NULL ? szText : hb_langDGetErrorIntr( ulIntCode ), szPar1, szPar2 ); printf( hb_consoleGetNewLine() ); hb_stackDispCall(); exit( EXIT_FAILURE ); }