1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-03 17:56:21 +03:00

0030762: Foundation Classes - include backtrace within OSD_SIGSEGV

Standard_Failure now holds an optional stack trace dump.
Added function Standard::StackTrace() dumping backtrace to the string.
SegvHandler within OSD_signal now appends backtrace to the message
if OSD::SignalStackTraceLength() is set to non-zero value
or environment variable "CSF_DEBUG_MODE" is set for debugging.

Added auxiliary macros Standard_NOINLINE disallowing function inlining.

Command "dsetsignal" has been extended by -strackTraceLength argument
for defining stack trace length within signals redirected to C++ exceptions.
Added "ddebugtraces" command for debugging purposes (adding stack traces to all exceptions).
This commit is contained in:
kgv 2019-06-03 08:06:24 +03:00 committed by bugmaster
parent 14eea8293d
commit 7fb9d6d573
16 changed files with 1167 additions and 329 deletions

View File

@ -377,6 +377,7 @@ void Draw_Appli(int argc, char** argv, const FDraw_InitAppli Draw_InitAppli)
// set signals
// *****************************************************************
OSD::SetSignal(Standard_False);
//OSD::SetSignalStackTraceLength (10);
#ifdef _WIN32
// in interactive mode, force Windows to report dll loading problems interactively

View File

@ -954,6 +954,7 @@ static int dsetsignal (Draw_Interpretor& theDI, Standard_Integer theArgNb, const
{
OSD_SignalMode aMode = OSD_SignalMode_Set;
Standard_Boolean aSetFPE = OSD::ToCatchFloatingSignals();
Standard_Integer aStackLen = OSD::SignalStackTraceLength();
// default for FPE signal is defined by CSF_FPE variable, if set
OSD_Environment aEnv("CSF_FPE");
@ -995,6 +996,14 @@ static int dsetsignal (Draw_Interpretor& theDI, Standard_Integer theArgNb, const
else if (anArg == "default")
{
}
else if (anArgIter + 1 < theArgNb
&& (anArg == "-stracktracelength"
|| anArg == "-stracktracelen"
|| anArg == "-stracklength"
|| anArg == "-stracklen"))
{
aStackLen = Draw::Atoi (theArgVec[++anArgIter]);
}
else
{
Message::SendFail() << "Syntax error: unknown argument '" << anArg << "'";
@ -1003,19 +1012,21 @@ static int dsetsignal (Draw_Interpretor& theDI, Standard_Integer theArgNb, const
}
OSD::SetSignal(aMode, aSetFPE);
OSD::SetSignalStackTraceLength (aStackLen);
// report actual status in the end
const char* aModeStr = 0;
switch (OSD::SignalMode())
{
default:
case OSD_SignalMode_AsIs: aModeStr = "asis"; break;
case OSD_SignalMode_Set: aModeStr = "set"; break;
case OSD_SignalMode_SetUnhandled: aModeStr = "unhandled"; break;
case OSD_SignalMode_Unset: aModeStr = "unset"; break;
default:
case OSD_SignalMode_AsIs: aModeStr = "asis"; break;
case OSD_SignalMode_Set: aModeStr = "set"; break;
case OSD_SignalMode_SetUnhandled: aModeStr = "unhandled"; break;
case OSD_SignalMode_Unset: aModeStr = "unset"; break;
}
theDI << "Signal mode: " << aModeStr << "\n"
<< "Catch FPE: " << (OSD::ToCatchFloatingSignals() ? "1" : "0") << "\n";
<< "Catch FPE: " << (OSD::ToCatchFloatingSignals() ? "1" : "0") << "\n"
<< "Stack Trace Length: " << aStackLen << "\n";
return 0;
}
@ -1112,6 +1123,27 @@ static int dtracelevel (Draw_Interpretor& theDI,
return 0;
}
//==============================================================================
//function : ddebugtraces
//purpose :
//==============================================================================
static int ddebugtraces (Draw_Interpretor& theDI, Standard_Integer theArgNb, const char** theArgVec)
{
if (theArgNb < 2)
{
theDI << Standard_Failure::DefaultStackTraceLength();
return 0;
}
else if (theArgNb != 2)
{
theDI << "Syntax error: wrong number of arguments";
return 1;
}
Standard_Failure::SetDefaultStackTraceLength (Draw::Atoi (theArgVec[1]));
return 0;
}
//==============================================================================
//function : dputs
//purpose :
@ -1243,7 +1275,11 @@ void Draw::BasicCommands(Draw_Interpretor& theCommands)
__FILE__, dmeminfo, g);
theCommands.Add("dperf","dperf [reset] -- show performance counters, reset if argument is provided",
__FILE__,dperf,g);
theCommands.Add("dsetsignal","dsetsignal [{asis|set|unhandled|unset}=set] [{0|1|default=$CSF_FPE}]\n -- set OSD signal handler, with FPE option if argument is given",
theCommands.Add("dsetsignal",
"dsetsignal [{asIs|set|unhandled|unset}=set] [{0|1|default=$CSF_FPE}]"
"\n\t\t: [-strackTraceLength Length]"
"\n\t\t: Sets OSD signal handler, with FPE option if argument is given."
"\n\t\t: -strackTraceLength specifies length of stack trace to put into exceptions redirected from signals.",
__FILE__,dsetsignal,g);
theCommands.Add("dparallel",
@ -1264,6 +1300,11 @@ void Draw::BasicCommands(Draw_Interpretor& theCommands)
__FILE__,decho,g);
theCommands.Add("dtracelevel", "dtracelevel [trace|info|warn|alarm|fail]",
__FILE__, dtracelevel, g);
theCommands.Add("ddebugtraces",
"ddebugtraces nbTraces"
"\n\t\t: Sets the number of lines for the stack trace within Standard_Failure constructor."
"\n\t\t: Intended for debug purposes.",
__FILE__, ddebugtraces, g);
theCommands.Add("dbreak", "raises Tcl exception if user has pressed Control-Break key",
__FILE__,dbreak,g);

View File

@ -134,6 +134,14 @@ public:
//! not. If yes then raises Exception_CTRL_BREAK.
Standard_EXPORT static void ControlBreak();
//! Returns a length of stack trace to be put into exception redirected from signal;
//! 0 by default meaning no stack trace.
//! @sa Standard_Failure::GetStackString()
Standard_EXPORT static Standard_Integer SignalStackTraceLength();
//! Sets a length of stack trace to be put into exception redirected from signal.
Standard_EXPORT static void SetSignalStackTraceLength (Standard_Integer theLength);
};
#endif // _OSD_HeaderFile

View File

@ -269,7 +269,7 @@ void OSD_ThreadPool::Launcher::wait()
}
aFailures = TCollection_AsciiString("Multiple exceptions:\n") + aFailures;
throw Standard_ProgramError (aFailures.ToCString());
throw Standard_ProgramError (aFailures.ToCString(), NULL);
}
// =======================================================================
@ -289,17 +289,17 @@ void OSD_ThreadPool::performJob (Handle(Standard_Failure)& theFailure,
{
TCollection_AsciiString aMsg = TCollection_AsciiString (aFailure.DynamicType()->Name())
+ ": " + aFailure.GetMessageString();
theFailure = new Standard_ProgramError (aMsg.ToCString());
theFailure = new Standard_ProgramError (aMsg.ToCString(), aFailure.GetStackString());
}
catch (std::exception& anStdException)
{
TCollection_AsciiString aMsg = TCollection_AsciiString (typeid(anStdException).name())
+ ": " + anStdException.what();
theFailure = new Standard_ProgramError (aMsg.ToCString());
theFailure = new Standard_ProgramError (aMsg.ToCString(), NULL);
}
catch (...)
{
theFailure = new Standard_ProgramError ("Error: Unknown exception");
theFailure = new Standard_ProgramError ("Error: Unknown exception", NULL);
}
}

View File

@ -21,6 +21,7 @@
#include <Standard_WarningDisableFunctionCast.hxx>
static OSD_SignalMode OSD_WasSetSignal = OSD_SignalMode_AsIs;
static Standard_Integer OSD_SignalStackTraceLength = 0;
//=======================================================================
//function : SignalMode
@ -31,6 +32,24 @@ OSD_SignalMode OSD::SignalMode()
return OSD_WasSetSignal;
}
// =======================================================================
// function : SignalStackTraceLength
// purpose :
// =======================================================================
Standard_Integer OSD::SignalStackTraceLength()
{
return OSD_SignalStackTraceLength;
}
// =======================================================================
// function : SetSignalStackTraceLength
// purpose :
// =======================================================================
void OSD::SetSignalStackTraceLength (Standard_Integer theLength)
{
OSD_SignalStackTraceLength = theLength;
}
#ifdef _WIN32
//---------------------------- Windows NT System --------------------------------
@ -85,7 +104,7 @@ static Standard_Boolean fMsgBox;
// used to forbid simultaneous execution of setting / executing handlers
static Standard_Mutex THE_SIGNAL_MUTEX;
static LONG __fastcall _osd_raise ( DWORD, LPSTR );
static LONG __fastcall _osd_raise (DWORD theCode, const char* theMsg, const char* theStack);
static BOOL WINAPI _osd_ctrl_break_handler ( DWORD );
#if ! defined(OCCT_UWP) && !defined(__MINGW32__) && !defined(__CYGWIN32__)
@ -96,172 +115,208 @@ static LONG _osd_debug ( void );
# define _OSD_FPX ( _EM_INVALID | _EM_DENORMAL | _EM_ZERODIVIDE | _EM_OVERFLOW )
#ifdef OCC_CONVERT_SIGNALS
#define THROW_OR_JUMP(Type,Message) Type::NewInstance(Message)->Jump()
#define THROW_OR_JUMP(Type,Message,Stack) Type::NewInstance(Message,Stack)->Jump()
#else
#define THROW_OR_JUMP(Type,Message) throw Type(Message)
#define THROW_OR_JUMP(Type,Message,Stack) throw Type(Message,Stack)
#endif
//=======================================================================
//function : CallHandler
//purpose :
//=======================================================================
static LONG CallHandler (DWORD dwExceptionCode,
ptrdiff_t ExceptionInformation1,
ptrdiff_t ExceptionInformation0)
static LONG CallHandler (DWORD theExceptionCode,
EXCEPTION_POINTERS* theExcPtr)
{
ptrdiff_t ExceptionInformation1 = 0, ExceptionInformation0 = 0;
if (theExcPtr != NULL)
{
ExceptionInformation1 = theExcPtr->ExceptionRecord->ExceptionInformation[1];
ExceptionInformation0 = theExcPtr->ExceptionRecord->ExceptionInformation[0];
}
Standard_Mutex::Sentry aSentry (THE_SIGNAL_MUTEX); // lock the mutex to prevent simultaneous handling
static char aBuffer[2048];
static wchar_t buffer[2048];
int flterr = 0;
buffer[0] = '\0' ;
// std::cout << "CallHandler " << dwExceptionCode << std::endl ;
switch ( dwExceptionCode ) {
bool isFloatErr = false;
aBuffer[0] = '\0';
switch (theExceptionCode)
{
case EXCEPTION_FLT_DENORMAL_OPERAND:
// std::cout << "CallHandler : EXCEPTION_FLT_DENORMAL_OPERAND:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT DENORMAL OPERAND");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT DENORMAL OPERAND");
isFloatErr = true;
break;
}
case EXCEPTION_FLT_DIVIDE_BY_ZERO:
// std::cout << "CallHandler : EXCEPTION_FLT_DIVIDE_BY_ZERO:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT DIVIDE BY ZERO");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT DIVIDE BY ZERO");
isFloatErr = true;
break;
}
case EXCEPTION_FLT_INEXACT_RESULT:
// std::cout << "CallHandler : EXCEPTION_FLT_INEXACT_RESULT:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT INEXACT RESULT");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT INEXACT RESULT");
isFloatErr = true;
break;
}
case EXCEPTION_FLT_INVALID_OPERATION:
// std::cout << "CallHandler : EXCEPTION_FLT_INVALID_OPERATION:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT INVALID OPERATION");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT INVALID OPERATION");
isFloatErr = true;
break;
}
case EXCEPTION_FLT_OVERFLOW:
// std::cout << "CallHandler : EXCEPTION_FLT_OVERFLOW:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT OVERFLOW");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT OVERFLOW");
isFloatErr = true;
break;
}
case EXCEPTION_FLT_STACK_CHECK:
// std::cout << "CallHandler : EXCEPTION_FLT_STACK_CHECK:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT STACK CHECK");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT STACK CHECK");
isFloatErr = true;
break;
}
case EXCEPTION_FLT_UNDERFLOW:
// std::cout << "CallHandler : EXCEPTION_FLT_UNDERFLOW:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT UNDERFLOW");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT UNDERFLOW");
isFloatErr = true;
break;
}
case STATUS_FLOAT_MULTIPLE_TRAPS:
// std::cout << "CallHandler : EXCEPTION_FLT_UNDERFLOW:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT MULTIPLE TRAPS (possible overflow in conversion of double to integer)");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT MULTIPLE TRAPS (possible overflow in conversion of double to integer)");
isFloatErr = true;
break;
}
case STATUS_FLOAT_MULTIPLE_FAULTS:
// std::cout << "CallHandler : EXCEPTION_FLT_UNDERFLOW:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"FLT MULTIPLE FAULTS");
flterr = 1 ;
break ;
{
strcat_s (aBuffer, sizeof(aBuffer), "FLT MULTIPLE FAULTS");
isFloatErr = true;
break;
}
case STATUS_NO_MEMORY:
// std::cout << "CallHandler : STATUS_NO_MEMORY:" << std::endl ;
THROW_OR_JUMP (OSD_Exception_STATUS_NO_MEMORY, "MEMORY ALLOCATION ERROR ( no room in the process heap )");
{
THROW_OR_JUMP (OSD_Exception_STATUS_NO_MEMORY, "MEMORY ALLOCATION ERROR ( no room in the process heap )", NULL);
break;
}
case EXCEPTION_ACCESS_VIOLATION:
// std::cout << "CallHandler : EXCEPTION_ACCESS_VIOLATION:" << std::endl ;
StringCchPrintfW (buffer, _countof(buffer), L"%s%s%s0x%.8p%s%s%s", L"ACCESS VIOLATION",
fMsgBox ? L"\n" : L" ", L"at address ",
ExceptionInformation1 ,
L" during '",
ExceptionInformation0 ? L"WRITE" : L"READ",
L"' operation");
{
_snprintf_s (aBuffer, sizeof(aBuffer), _TRUNCATE, "%s%s%s0x%.8p%s%s%s", "ACCESS VIOLATION",
fMsgBox ? "\n" : " ",
"at address ", (void* )ExceptionInformation1,
" during '", ExceptionInformation0 ? "WRITE" : "READ", "' operation");
break;
}
case EXCEPTION_ARRAY_BOUNDS_EXCEEDED:
// std::cout << "CallHandler : EXCEPTION_ARRAY_BOUNDS_EXCEEDED:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"ARRAY BOUNDS EXCEEDED");
{
strcat_s (aBuffer, sizeof(aBuffer), "ARRAY BOUNDS EXCEEDED");
break;
}
case EXCEPTION_DATATYPE_MISALIGNMENT:
// std::cout << "CallHandler : EXCEPTION_DATATYPE_MISALIGNMENT:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"DATATYPE MISALIGNMENT");
{
strcat_s (aBuffer, sizeof(aBuffer), "DATATYPE MISALIGNMENT");
break;
}
case EXCEPTION_ILLEGAL_INSTRUCTION:
// std::cout << "CallHandler : EXCEPTION_ILLEGAL_INSTRUCTION:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"ILLEGAL INSTRUCTION");
{
strcat_s (aBuffer, sizeof(aBuffer), "ILLEGAL INSTRUCTION");
break;
}
case EXCEPTION_IN_PAGE_ERROR:
// std::cout << "CallHandler : EXCEPTION_IN_PAGE_ERROR:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"IN_PAGE ERROR");
{
strcat_s (aBuffer, sizeof(aBuffer), "IN_PAGE ERROR");
break;
}
case EXCEPTION_INT_DIVIDE_BY_ZERO:
// std::cout << "CallHandler : EXCEPTION_INT_DIVIDE_BY_ZERO:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"INTEGER DIVISION BY ZERO");
{
strcat_s (aBuffer, sizeof(aBuffer), "INTEGER DIVISION BY ZERO");
break;
}
case EXCEPTION_INT_OVERFLOW:
// std::cout << "CallHandler : EXCEPTION_INT_OVERFLOW:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"INTEGER OVERFLOW");
{
strcat_s (aBuffer, sizeof(aBuffer), "INTEGER OVERFLOW");
break;
}
case EXCEPTION_INVALID_DISPOSITION:
// std::cout << "CallHandler : EXCEPTION_INVALID_DISPOSITION:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"INVALID DISPOSITION");
{
strcat_s (aBuffer, sizeof(aBuffer), "INVALID DISPOSITION");
break;
}
case EXCEPTION_NONCONTINUABLE_EXCEPTION:
// std::cout << "CallHandler : EXCEPTION_NONCONTINUABLE_EXCEPTION:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"NONCONTINUABLE EXCEPTION");
{
strcat_s (aBuffer, sizeof(aBuffer), "NONCONTINUABLE EXCEPTION");
break;
}
case EXCEPTION_PRIV_INSTRUCTION:
// std::cout << "CallHandler : EXCEPTION_PRIV_INSTRUCTION:" << std::endl ;
StringCchCopyW (buffer, _countof(buffer), L"PRIVELEGED INSTRUCTION ENCOUNTERED");
{
strcat_s (aBuffer, sizeof(aBuffer), "PRIVELEGED INSTRUCTION ENCOUNTERED");
break;
}
case EXCEPTION_STACK_OVERFLOW:
// std::cout << "CallHandler : EXCEPTION_STACK_OVERFLOW:" << std::endl ;
{
#if defined( _MSC_VER ) && ( _MSC_VER >= 1300 ) && !defined(OCCT_UWP)
// try recovering from stack overflow: available in MS VC++ 7.0
// try recovering from stack overflow: available in MS VC++ 7.0
if (!_resetstkoflw())
StringCchCopyW (buffer, _countof(buffer), L"Unrecoverable STACK OVERFLOW");
{
strcat_s (aBuffer, sizeof(aBuffer), "Unrecoverable STACK OVERFLOW");
}
else
#endif
StringCchCopyW (buffer, _countof(buffer), L"STACK OVERFLOW");
{
strcat_s (aBuffer, sizeof(aBuffer), "STACK OVERFLOW");
}
break;
}
default:
StringCchPrintfW (buffer, _countof(buffer), L"unknown exception code 0x%x, params 0x%p 0x%p",
dwExceptionCode, ExceptionInformation1, ExceptionInformation0 );
} // end switch
{
_snprintf_s (aBuffer, sizeof(aBuffer), _TRUNCATE, "unknown exception code 0x%x, params 0x%p 0x%p",
theExceptionCode, (void* )ExceptionInformation1, (void* )ExceptionInformation0);
}
}
// reset FPE state (before message box, otherwise it may fail to show up)
if ( flterr ) {
if (isFloatErr)
{
OSD::SetFloatingSignal (Standard_True);
}
#if ! defined(OCCT_UWP) && !defined(__MINGW32__) && !defined(__CYGWIN32__)
// provide message to the user with possibility to stop
size_t idx;
StringCchLengthW (buffer, _countof(buffer),&idx);
if ( idx && fMsgBox && dwExceptionCode != EXCEPTION_NONCONTINUABLE_EXCEPTION ) {
MessageBeep ( MB_ICONHAND );
int aChoice = ::MessageBoxW (0, buffer, L"OCCT Exception Handler", MB_ABORTRETRYIGNORE | MB_ICONSTOP);
const int aStackLength = OSD_SignalStackTraceLength;
const int aStackBufLen = Max (aStackLength * 200, 2048);
char* aStackBuffer = aStackLength != 0 ? (char* )alloca (aStackBufLen) : NULL;
if (aStackBuffer != NULL)
{
memset (aStackBuffer, 0, aStackBufLen);
Standard::StackTrace (aStackBuffer, aStackBufLen, aStackLength, theExcPtr->ContextRecord);
}
#if !defined(OCCT_UWP) && !defined(__MINGW32__) && !defined(__CYGWIN32__)
// provide message to the user with possibility to stop
if (aBuffer[0] != '\0'
&& fMsgBox
&& theExceptionCode != EXCEPTION_NONCONTINUABLE_EXCEPTION)
{
MessageBeep (MB_ICONHAND);
char aMsgBoxBuffer[2048];
strcat_s (aMsgBoxBuffer, sizeof(aMsgBoxBuffer), aBuffer);
if (aStackBuffer != NULL)
{
strcat_s (aMsgBoxBuffer, sizeof(aMsgBoxBuffer), aStackBuffer);
}
int aChoice = ::MessageBoxA (0, aMsgBoxBuffer, "OCCT Exception Handler", MB_ABORTRETRYIGNORE | MB_ICONSTOP);
if (aChoice == IDRETRY)
{
_osd_debug();
DebugBreak();
} else if (aChoice == IDABORT)
exit(0xFFFF);
}
else if (aChoice == IDABORT)
{
exit (0xFFFF);
}
}
#endif
char aBufferA[2048];
WideCharToMultiByte(CP_UTF8, 0, buffer, -1, aBufferA, sizeof(aBufferA), NULL, NULL);
return _osd_raise(dwExceptionCode, aBufferA);
return _osd_raise (theExceptionCode, aBuffer, aStackBuffer);
}
//=======================================================================
@ -279,38 +334,38 @@ static void SIGWntHandler (int signum, int sub_code)
std::cout << "signal error" << std::endl ;
switch( sub_code ) {
case _FPE_INVALID :
CallHandler( EXCEPTION_FLT_INVALID_OPERATION ,0,0) ;
CallHandler (EXCEPTION_FLT_INVALID_OPERATION, NULL);
break ;
case _FPE_DENORMAL :
CallHandler( EXCEPTION_FLT_DENORMAL_OPERAND ,0,0) ;
CallHandler (EXCEPTION_FLT_DENORMAL_OPERAND, NULL);
break ;
case _FPE_ZERODIVIDE :
CallHandler( EXCEPTION_FLT_DIVIDE_BY_ZERO ,0,0) ;
CallHandler (EXCEPTION_FLT_DIVIDE_BY_ZERO, NULL);
break ;
case _FPE_OVERFLOW :
CallHandler( EXCEPTION_FLT_OVERFLOW ,0,0) ;
CallHandler (EXCEPTION_FLT_OVERFLOW, NULL);
break ;
case _FPE_UNDERFLOW :
CallHandler( EXCEPTION_FLT_UNDERFLOW ,0,0) ;
CallHandler (EXCEPTION_FLT_UNDERFLOW, NULL);
break ;
case _FPE_INEXACT :
CallHandler( EXCEPTION_FLT_INEXACT_RESULT ,0,0) ;
CallHandler (EXCEPTION_FLT_INEXACT_RESULT, NULL);
break ;
default:
std::cout << "SIGWntHandler(default) -> throw Standard_NumericError(\"Floating Point Error\");" << std::endl;
THROW_OR_JUMP (Standard_NumericError, "Floating Point Error");
THROW_OR_JUMP (Standard_NumericError, "Floating Point Error", NULL);
break ;
}
break ;
case SIGSEGV :
if ( signal( signum, (void(*)(int))SIGWntHandler ) == SIG_ERR )
std::cout << "signal error" << std::endl ;
CallHandler( EXCEPTION_ACCESS_VIOLATION ,0,0) ;
CallHandler (EXCEPTION_ACCESS_VIOLATION, NULL);
break ;
case SIGILL :
if ( signal( signum, (void(*)(int))SIGWntHandler ) == SIG_ERR )
std::cout << "signal error" << std::endl ;
CallHandler( EXCEPTION_ILLEGAL_INSTRUCTION ,0,0) ;
CallHandler (EXCEPTION_ILLEGAL_INSTRUCTION, NULL);
break ;
default:
std::cout << "SIGWntHandler unexpected signal : " << signum << std::endl ;
@ -337,12 +392,7 @@ static void SIGWntHandler (int signum, int sub_code)
static void TranslateSE( unsigned int theCode, EXCEPTION_POINTERS* theExcPtr )
{
Standard_Mutex::Sentry aSentry (THE_SIGNAL_MUTEX); // lock the mutex to prevent simultaneous handling
ptrdiff_t info1 = 0, info0 = 0;
if ( theExcPtr ) {
info1 = theExcPtr->ExceptionRecord->ExceptionInformation[1];
info0 = theExcPtr->ExceptionRecord->ExceptionInformation[0];
}
CallHandler(theCode, info1, info0);
CallHandler (theCode, theExcPtr);
}
#endif
@ -354,11 +404,8 @@ static void TranslateSE( unsigned int theCode, EXCEPTION_POINTERS* theExcPtr )
//=======================================================================
static LONG WINAPI WntHandler (EXCEPTION_POINTERS *lpXP)
{
DWORD dwExceptionCode = lpXP->ExceptionRecord->ExceptionCode;
return CallHandler (dwExceptionCode,
lpXP->ExceptionRecord->ExceptionInformation[1],
lpXP->ExceptionRecord->ExceptionInformation[0]);
DWORD dwExceptionCode = lpXP->ExceptionRecord->ExceptionCode;
return CallHandler (dwExceptionCode, lpXP);
}
//=======================================================================
@ -422,6 +469,11 @@ void OSD::SetSignal (OSD_SignalMode theSignalMode,
{
std::cout << "Environment variable CSF_DEBUG_MODE setted.\n";
fMsgBox = Standard_True;
if (OSD_SignalStackTraceLength == 0)
{
// enable stack trace if CSF_DEBUG_MODE is set
OSD_SignalStackTraceLength = 10;
}
}
else
{
@ -508,67 +560,71 @@ static BOOL WINAPI _osd_ctrl_break_handler ( DWORD dwCode ) {
//============================================================================
//==== _osd_raise
//============================================================================
static LONG __fastcall _osd_raise ( DWORD dwCode, LPSTR msg )
static LONG __fastcall _osd_raise (DWORD theCode, const char* theMsg, const char* theStack)
{
if (msg[0] == '\x03') ++msg;
const char* aMsg = theMsg;
if (aMsg[0] == '\x03')
{
++aMsg;
}
switch (dwCode)
switch (theCode)
{
case EXCEPTION_ACCESS_VIOLATION:
THROW_OR_JUMP (OSD_Exception_ACCESS_VIOLATION, msg);
THROW_OR_JUMP (OSD_Exception_ACCESS_VIOLATION, aMsg, theStack);
break;
case EXCEPTION_ARRAY_BOUNDS_EXCEEDED:
THROW_OR_JUMP (OSD_Exception_ARRAY_BOUNDS_EXCEEDED, msg);
THROW_OR_JUMP (OSD_Exception_ARRAY_BOUNDS_EXCEEDED, aMsg, theStack);
break;
case EXCEPTION_DATATYPE_MISALIGNMENT:
THROW_OR_JUMP (Standard_ProgramError, msg);
THROW_OR_JUMP (Standard_ProgramError, aMsg, theStack);
break;
case EXCEPTION_ILLEGAL_INSTRUCTION:
THROW_OR_JUMP (OSD_Exception_ILLEGAL_INSTRUCTION, msg);
THROW_OR_JUMP (OSD_Exception_ILLEGAL_INSTRUCTION, aMsg, theStack);
break;
case EXCEPTION_IN_PAGE_ERROR:
THROW_OR_JUMP (OSD_Exception_IN_PAGE_ERROR, msg);
THROW_OR_JUMP (OSD_Exception_IN_PAGE_ERROR, aMsg, theStack);
break;
case EXCEPTION_INT_DIVIDE_BY_ZERO:
THROW_OR_JUMP (Standard_DivideByZero, msg);
THROW_OR_JUMP (Standard_DivideByZero, aMsg, theStack);
break;
case EXCEPTION_INT_OVERFLOW:
THROW_OR_JUMP (OSD_Exception_INT_OVERFLOW, msg);
THROW_OR_JUMP (OSD_Exception_INT_OVERFLOW, aMsg, theStack);
break;
case EXCEPTION_INVALID_DISPOSITION:
THROW_OR_JUMP (OSD_Exception_INVALID_DISPOSITION, msg);
THROW_OR_JUMP (OSD_Exception_INVALID_DISPOSITION, aMsg, theStack);
break;
case EXCEPTION_NONCONTINUABLE_EXCEPTION:
THROW_OR_JUMP (OSD_Exception_NONCONTINUABLE_EXCEPTION, msg);
THROW_OR_JUMP (OSD_Exception_NONCONTINUABLE_EXCEPTION, aMsg, theStack);
break;
case EXCEPTION_PRIV_INSTRUCTION:
THROW_OR_JUMP (OSD_Exception_PRIV_INSTRUCTION, msg);
THROW_OR_JUMP (OSD_Exception_PRIV_INSTRUCTION, aMsg, theStack);
break;
case EXCEPTION_STACK_OVERFLOW:
THROW_OR_JUMP (OSD_Exception_STACK_OVERFLOW, msg);
THROW_OR_JUMP (OSD_Exception_STACK_OVERFLOW, aMsg, theStack);
break;
case EXCEPTION_FLT_DIVIDE_BY_ZERO:
THROW_OR_JUMP (Standard_DivideByZero, msg);
THROW_OR_JUMP (Standard_DivideByZero, aMsg, theStack);
break;
case EXCEPTION_FLT_STACK_CHECK:
case EXCEPTION_FLT_OVERFLOW:
THROW_OR_JUMP (Standard_Overflow, msg);
THROW_OR_JUMP (Standard_Overflow, aMsg, theStack);
break;
case EXCEPTION_FLT_UNDERFLOW:
THROW_OR_JUMP (Standard_Underflow, msg);
THROW_OR_JUMP (Standard_Underflow, aMsg, theStack);
break;
case EXCEPTION_FLT_INVALID_OPERATION:
case EXCEPTION_FLT_DENORMAL_OPERAND:
case EXCEPTION_FLT_INEXACT_RESULT:
case STATUS_FLOAT_MULTIPLE_TRAPS:
case STATUS_FLOAT_MULTIPLE_FAULTS:
THROW_OR_JUMP (Standard_NumericError, msg);
THROW_OR_JUMP (Standard_NumericError, aMsg, theStack);
break;
default:
break;
} // end switch
}
return EXCEPTION_EXECUTE_HANDLER;
} // end _osd_raise
}
#if ! defined(OCCT_UWP) && !defined(__MINGW32__) && !defined(__CYGWIN32__)
//============================================================================
@ -849,32 +905,41 @@ static void Handler (const int theSignal)
#ifdef SA_SIGINFO
static void SegvHandler(const int theSignal,
siginfo_t *ip,
siginfo_t* theSigInfo,
const Standard_Address theContext)
{
(void)theSignal; // silence GCC warnings
(void)theSignal;
(void)theContext;
if (theSigInfo != NULL)
{
sigset_t set;
sigemptyset (&set);
sigaddset (&set, SIGSEGV);
sigprocmask (SIG_UNBLOCK, &set, NULL);
void* anAddress = theSigInfo->si_addr;
{
char aMsg[100];
sprintf (aMsg, "SIGSEGV 'segmentation violation' detected. Address %lx.", (long )anAddress);
// std::cout << "OSD::SegvHandler activated(SA_SIGINFO)" << std::endl ;
if ( ip != NULL ) {
sigset_t set;
sigemptyset(&set);
sigaddset(&set, SIGSEGV);
sigprocmask (SIG_UNBLOCK, &set, NULL) ;
void *address = ip->si_addr ;
{
char Msg[100];
sprintf(Msg,"SIGSEGV 'segmentation violation' detected. Address %lx",
(long ) address ) ;
OSD_SIGSEGV::NewInstance(Msg)->Jump();
}
const int aStackLength = OSD_SignalStackTraceLength;
const int aStackBufLen = Max (aStackLength * 200, 2048);
char* aStackBuffer = aStackLength != 0 ? (char* )alloca (aStackBufLen) : NULL;
if (aStackBuffer != NULL)
{
memset (aStackBuffer, 0, aStackBufLen);
Standard::StackTrace (aStackBuffer, aStackBufLen, aStackLength);
}
OSD_SIGSEGV::NewInstance (aMsg, aStackBuffer)->Jump();
}
}
#ifdef OCCT_DEBUG
else {
std::cout << "Wrong undefined address." << std::endl ;
else
{
std::cout << "Wrong undefined address." << std::endl;
}
#endif
exit(SIGSEGV);
exit (SIGSEGV);
}
#elif defined (_hpux) || defined(HPUX)
@ -882,30 +947,26 @@ static void SegvHandler(const int theSignal,
// pour version 09.07
static void SegvHandler(const int theSignal,
siginfo_t *ip,
siginfo_t* theSigInfo,
const Standard_Address theContext)
{
unsigned long Space ;
unsigned long Offset ;
char Msg[100] ;
if ( theContext != NULL ) {
Space = ((struct sigcontext *)theContext)->sc_sl.sl_ss.ss_cr20 ;
Offset = ((struct sigcontext *)theContext)->sc_sl.sl_ss.ss_cr21 ;
// std::cout << "Wrong address = " << hex(Offset) << std::endl ;
if (theContext != NULL)
{
unsigned long aSpace = ((struct sigcontext *)theContext)->sc_sl.sl_ss.ss_cr20;
unsigned long anOffset = ((struct sigcontext *)theContext)->sc_sl.sl_ss.ss_cr21;
{
sprintf(Msg,"SIGSEGV 'segmentation violation' detected. Address %lx",Offset) ;
OSD_SIGSEGV::Jump(Msg);
// scp->sc_pcoq_head = scp->sc_pcoq_tail ; Permettrait de continuer a
// scp->sc_pcoq_tail = scp->sc_pcoq_tail + 0x4 ; l'intruction suivant le segv.
char aMsg[100];
sprintf (aMsg, "SIGSEGV 'segmentation violation' detected. Address %lx", anOffset);
OSD_SIGSEGV::NewInstance (aMsg)->Jump();
}
}
#ifdef OCCT_DEBUG
else {
std::cout << "Wrong undefined address." << std::endl ;
else
{
std::cout << "Wrong undefined address." << std::endl;
}
#endif
exit(SIGSEGV);
exit (SIGSEGV);
}
#endif

View File

@ -2396,6 +2396,88 @@ static Standard_Integer OCC6143 (Draw_Interpretor& di, Standard_Integer argc, co
return 0;
}
//! Auxiliary functions for printing synthetic backtrace
class MyTestInterface : public Standard_Transient
{
public:
virtual int Standard_NOINLINE testMethod3 (int* theIntPtr, bool theToPrintStack) = 0;
};
class MyTestClass : public MyTestInterface
{
public:
MyTestClass() {}
virtual int Standard_NOINLINE testMethod3 (int* theIntPtr, bool theToPrintStack)
{
if (theToPrintStack)
{
char aMsg[4096] = {};
Standard::StackTrace (aMsg, 4096, 10);
std::cout << aMsg << "\n";
return 0;
}
*theIntPtr = 4;
return *theIntPtr;
}
};
static int Standard_NOINLINE myTestFunction2 (int* theIntPtr, bool theToPrintStack)
{
Handle(MyTestInterface) aTest = new MyTestClass();
return aTest->testMethod3 (theIntPtr, theToPrintStack);
}
static void Standard_NOINLINE myTestFunction1 (bool theToPrintStack)
{
int* anIntPtr = NULL;
myTestFunction2 (anIntPtr, theToPrintStack);
}
static Standard_NOINLINE Standard_Integer OCC30762 (Draw_Interpretor& theDI,
Standard_Integer theNbArgs,
const char** )
{
if (theNbArgs != 1)
{
theDI << "Syntax error: wrong number of arguments";
return 1;
}
// just print stack
std::cout << "Test normal backtrace...\n";
myTestFunction1 (true);
// test access violation
{
try
{
OCC_CATCH_SIGNALS
std::cout << "Test segmentation Fault...\n";
myTestFunction1 (false);
std::cout << "Error: writing by NULL address - no exception is raised!\n";
}
#ifdef _WIN32
catch (OSD_Exception_ACCESS_VIOLATION const& aSegException)
#else
catch (OSD_SIGSEGV const& aSegException)
#endif
{
theDI << " Caught (";
theDI << aSegException.GetMessageString();
theDI << aSegException.GetStackString();
theDI << ")... OK\n";
}
catch (Standard_Failure const& anException)
{
theDI << " Caught (";
theDI << anException.GetMessageString();
theDI << anException.GetStackString();
theDI << ")... KO\n";
}
}
return 0;
}
//! Auxiliary functor.
struct TestParallelFunctor
{
@ -4949,6 +5031,7 @@ void QABugs::Commands_11(Draw_Interpretor& theCommands) {
theCommands.Add("OCC5698", "OCC5698 wire", __FILE__, OCC5698, group);
theCommands.Add("OCC6143", "OCC6143 catching signals", __FILE__, OCC6143, group);
theCommands.Add("OCC30775", "OCC30775 catching signals in threads", __FILE__, OCC30775, group);
theCommands.Add("OCC30762", "OCC30762 printing backtrace", __FILE__, OCC30762, group);
theCommands.Add("OCC7141", "OCC7141 [nCount] aPath", __FILE__, OCC7141, group);
theCommands.Add("OCC7372", "OCC7372", __FILE__, OCC7372, group);
theCommands.Add("OCC8169", "OCC8169 edge1 edge2 plane", __FILE__, OCC8169, group);

View File

@ -87,6 +87,7 @@ Standard_ShortReal.cxx
Standard_ShortReal.hxx
Standard_Size.hxx
Standard_SStream.hxx
Standard_StackTrace.cxx
Standard_Std.hxx
Standard_Stream.hxx
Standard_Strtod.cxx

View File

@ -84,6 +84,28 @@ public:
//! Returns non-zero if some memory has been actually freed.
Standard_EXPORT static Standard_Integer Purge();
//! Appends backtrace to a message buffer.
//! Stack information might be incomplete in case of stripped binaries.
//! Implementation details:
//! - Not implemented for Android, iOS, QNX and UWP platforms.
//! - On non-Windows platform, this function is a wrapper to backtrace() system call.
//! - On Windows (Win32) platform, the function loads DbgHelp.dll dynamically,
//! and no stack will be provided if this or companion libraries (SymSrv.dll, SrcSrv.dll, etc.) will not be found;
//! .pdb symbols should be provided on Windows platform to retrieve a meaningful stack;
//! only x86_64 CPU architecture is currently implemented.
//! @param theBuffer [in] [out] message buffer to extend
//! @param theBufferSize [in] message buffer size
//! @param theNbTraces [in] maximum number of stack traces
//! @param theContext [in] optional platform-dependent frame context;
//! in case of DbgHelp (Windows) should be a pointer to CONTEXT
//! @param theNbTopSkip [in] number of traces on top of the stack to skip
//! @return TRUE on success
Standard_EXPORT static Standard_Boolean StackTrace (char* theBuffer,
const int theBufferSize,
const int theNbTraces,
void* theContext = NULL,
const int theNbTopSkip = 0);
};
// include definition of handle to make it always visible

View File

@ -32,7 +32,9 @@ class C1 : public C2 { \
void Throw () const Standard_OVERRIDE { throw *this; } \
public: \
C1() : C2() {} \
C1(const Standard_CString theMessage) : C2(theMessage) {} \
C1(Standard_CString theMessage) : C2(theMessage) {} \
C1(Standard_CString theMessage, Standard_CString theStackTrace) \
: C2 (theMessage, theStackTrace) {} \
static void Raise(const Standard_CString theMessage = "") { \
Handle(C1) _E = new C1; \
_E->Reraise(theMessage); \
@ -41,7 +43,8 @@ public: \
Handle(C1) _E = new C1; \
_E->Reraise (theMessage); \
} \
static Handle(C1) NewInstance(const Standard_CString theMessage = "") { return new C1(theMessage); } \
static Handle(C1) NewInstance(Standard_CString theMessage = "") { return new C1(theMessage); } \
static Handle(C1) NewInstance(Standard_CString theMessage, Standard_CString theStackTrace) { return new C1(theMessage, theStackTrace); } \
DEFINE_STANDARD_RTTI_INLINE(C1,C2) \
};

View File

@ -12,9 +12,9 @@
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#include <Standard_Failure.hxx>
#include <Standard_ErrorHandler.hxx>
#include <Standard_Failure.hxx>
#include <Standard_Macro.hxx>
#include <Standard_NoSuchObject.hxx>
#include <Standard_PCharacter.hxx>
@ -22,173 +22,341 @@
#include <Standard_TypeMismatch.hxx>
#include <string.h>
IMPLEMENT_STANDARD_RTTIEXT(Standard_Failure,Standard_Transient)
static Standard_CString allocate_message(const Standard_CString AString)
namespace
{
Standard_CString aStr = 0;
if(AString) {
const Standard_Size aLen = strlen(AString);
aStr = (Standard_CString) malloc(aLen+sizeof(Standard_Integer)+1);
if (aStr) {
Standard_PCharacter pStr=(Standard_PCharacter)aStr;
strcpy(pStr+sizeof(Standard_Integer),AString);
*((Standard_Integer*)aStr) = 1;
//! Global last failure object returned by Standard_Failure::Caught().
static Standard_THREADLOCAL Handle(Standard_Failure) Standard_Failure_RaisedError;
//! Global parameter defining default length of stack trace.
static Standard_Integer Standard_Failure_DefaultStackTraceLength = 0;
}
// =======================================================================
// function : StringRef::allocate_message
// purpose :
// =======================================================================
Standard_Failure::StringRef* Standard_Failure::StringRef::allocate_message (const Standard_CString theString)
{
if (theString == NULL
|| *theString == '\0')
{
return NULL;
}
const Standard_Size aLen = strlen (theString);
StringRef* aStrPtr = (StringRef* )malloc (aLen + sizeof(Standard_Integer) + 1);
if (aStrPtr != NULL)
{
strcpy ((char* )&aStrPtr->Message[0], theString);
aStrPtr->Counter = 1;
}
return aStrPtr;
}
// =======================================================================
// function : StringRef::copy_message
// purpose :
// =======================================================================
Standard_Failure::StringRef* Standard_Failure::StringRef::copy_message (Standard_Failure::StringRef* theString)
{
if (theString == NULL)
{
return NULL;
}
++theString->Counter;
return theString;
}
// =======================================================================
// function : StringRef::deallocate_message
// purpose :
// =======================================================================
void Standard_Failure::StringRef::deallocate_message (Standard_Failure::StringRef* theString)
{
if (theString != NULL)
{
if (--theString->Counter == 0)
{
free ((void* )theString);
}
}
return aStr;
}
static Standard_CString copy_message(Standard_CString aMessage)
// =======================================================================
// function : Standard_Failure
// purpose :
// =======================================================================
Standard_Failure::Standard_Failure()
: myMessage (NULL),
myStackTrace (NULL)
{
Standard_CString aStr = 0;
if(aMessage) {
aStr = aMessage;
(*((Standard_Integer*)aStr))++;
}
return aStr;
}
static void deallocate_message(Standard_CString aMessage)
{
if(aMessage) {
(*((Standard_Integer*)aMessage))--;
if(*((Standard_Integer*)aMessage)==0)
free((void*)aMessage);
const Standard_Integer aStackLength = Standard_Failure_DefaultStackTraceLength;
if (aStackLength > 0)
{
int aStackBufLen = Max (aStackLength * 200, 2048);
char* aStackBuffer = (char* )alloca (aStackBufLen);
if (aStackBuffer != NULL)
{
memset (aStackBuffer, 0, aStackBufLen);
if (Standard::StackTrace (aStackBuffer, aStackBufLen, aStackLength, NULL, 1))
{
myStackTrace = StringRef::allocate_message (aStackBuffer);
}
}
}
}
// ******************************************************************
// Standard_Failure *
// ******************************************************************
static Standard_THREADLOCAL Handle(Standard_Failure) RaisedError;
// ------------------------------------------------------------------
//
// ------------------------------------------------------------------
Standard_Failure::Standard_Failure ()
: myMessage(NULL)
// =======================================================================
// function : Standard_Failure
// purpose :
// =======================================================================
Standard_Failure::Standard_Failure (const Standard_CString theDesc)
: myMessage (NULL),
myStackTrace (NULL)
{
myMessage = StringRef::allocate_message (theDesc);
const Standard_Integer aStackLength = Standard_Failure_DefaultStackTraceLength;
if (aStackLength > 0)
{
int aStackBufLen = Max (aStackLength * 200, 2048);
char* aStackBuffer = (char* )alloca (aStackBufLen);
if (aStackBuffer != NULL)
{
memset (aStackBuffer, 0, aStackBufLen);
Standard::StackTrace (aStackBuffer, aStackBufLen, aStackLength, NULL, 1);
myStackTrace = StringRef::allocate_message (aStackBuffer);
}
}
}
// ------------------------------------------------------------------
// Create returns mutable Failure;
// ------------------------------------------------------------------
Standard_Failure::Standard_Failure (const Standard_CString AString)
: myMessage(NULL)
// =======================================================================
// function : Standard_Failure
// purpose :
// =======================================================================
Standard_Failure::Standard_Failure (const Standard_CString theDesc,
const Standard_CString theStackTrace)
: myMessage (NULL),
myStackTrace (NULL)
{
myMessage = allocate_message(AString);
myMessage = StringRef::allocate_message (theDesc);
myStackTrace = StringRef::allocate_message (theStackTrace);
}
Standard_Failure::Standard_Failure (const Standard_Failure& theFailure)
: Standard_Transient(theFailure)
// =======================================================================
// function : Standard_Failure
// purpose :
// =======================================================================
Standard_Failure::Standard_Failure (const Standard_Failure& theFailure)
: Standard_Transient (theFailure),
myMessage (NULL),
myStackTrace (NULL)
{
myMessage = copy_message(theFailure.myMessage);
myMessage = StringRef::copy_message (theFailure.myMessage);
myStackTrace = StringRef::copy_message (theFailure.myStackTrace);
}
// =======================================================================
// function : ~Standard_Failure
// purpose :
// =======================================================================
Standard_Failure::~Standard_Failure()
{
deallocate_message(myMessage);
StringRef::deallocate_message (myMessage);
StringRef::deallocate_message (myStackTrace);
}
void Standard_Failure::SetMessageString(const Standard_CString AString)
// =======================================================================
// function : GetMessageString
// purpose :
// =======================================================================
Standard_CString Standard_Failure::GetMessageString() const
{
if ( AString == GetMessageString() ) return;
deallocate_message(myMessage);
myMessage = allocate_message(AString);
return myMessage != NULL
? myMessage->GetMessage()
: "";
}
// ------------------------------------------------------------------
// Caught (myclass) returns mutable Failure raises NoSuchObject ;
// ------------------------------------------------------------------
Handle(Standard_Failure) Standard_Failure::Caught()
// =======================================================================
// function : SetMessageString
// purpose :
// =======================================================================
void Standard_Failure::SetMessageString (const Standard_CString theDesc)
{
return RaisedError ;
if (theDesc == GetMessageString())
{
return;
}
StringRef::deallocate_message (myMessage);
myMessage = StringRef::allocate_message (theDesc);
}
// ------------------------------------------------------------------
// Raise (myclass; aMessage: CString = "") ;
// ------------------------------------------------------------------
void Standard_Failure::Raise (const Standard_CString AString)
// =======================================================================
// function : GetStackString
// purpose :
// =======================================================================
Standard_CString Standard_Failure::GetStackString() const
{
return myStackTrace != NULL
? myStackTrace->GetMessage()
: "";
}
// =======================================================================
// function : SetStackString
// purpose :
// =======================================================================
void Standard_Failure::SetStackString (const Standard_CString theStack)
{
if (theStack == GetStackString())
{
return;
}
StringRef::deallocate_message (myStackTrace);
myStackTrace = StringRef::allocate_message (theStack);
}
// =======================================================================
// function : Caught
// purpose :
// =======================================================================
Handle(Standard_Failure) Standard_Failure::Caught()
{
return Standard_Failure_RaisedError;
}
// =======================================================================
// function : Raise
// purpose :
// =======================================================================
void Standard_Failure::Raise (const Standard_CString theDesc)
{
Handle(Standard_Failure) E = new Standard_Failure() ;
E->Reraise (AString) ;
Handle(Standard_Failure) aFailure = new Standard_Failure();
aFailure->Reraise (theDesc);
}
// ------------------------------------------------------------------
// Raise(myclass; aReason: in out SStream) ;
// ------------------------------------------------------------------
void Standard_Failure::Raise (const Standard_SStream& AReason)
// =======================================================================
// function : Raise
// purpose :
// =======================================================================
void Standard_Failure::Raise (const Standard_SStream& theReason)
{
Handle(Standard_Failure) E = new Standard_Failure();
E->Reraise (AReason);
Handle(Standard_Failure) aFailure = new Standard_Failure();
aFailure->Reraise (theReason);
}
// ------------------------------------------------------------------
// Reraise (me: mutable; aMessage: CString) ;
// ------------------------------------------------------------------
void Standard_Failure::Reraise (const Standard_CString AString)
// =======================================================================
// function : Reraise
// purpose :
// =======================================================================
void Standard_Failure::Reraise (const Standard_CString theDesc)
{
SetMessageString(AString);
SetMessageString (theDesc);
Reraise();
}
void Standard_Failure::Reraise (const Standard_SStream& AReason)
// =======================================================================
// function : Reraise
// purpose :
// =======================================================================
void Standard_Failure::Reraise (const Standard_SStream& theReason)
{
SetMessageString(AReason.str().c_str());
SetMessageString (theReason.str().c_str());
Reraise();
}
void Standard_Failure::Reraise ()
// =======================================================================
// function : Reraise
// purpose :
// =======================================================================
void Standard_Failure::Reraise()
{
RaisedError = this;
Standard_Failure_RaisedError = this;
Throw();
}
// =======================================================================
// function : Jump
// purpose :
// =======================================================================
void Standard_Failure::Jump()
{
#if defined (OCC_CONVERT_SIGNALS)
Standard_ErrorHandler::Error (this);
Standard_ErrorHandler::Abort (this);
#else
RaisedError = this;
Standard_Failure_RaisedError = this;
Throw();
#endif
}
// ------------------------------------------------------------------
// Throw (me) is virtual ;
// ------------------------------------------------------------------
// =======================================================================
// function : Throw
// purpose :
// =======================================================================
void Standard_Failure::Throw() const
{
throw *this;
}
// ------------------------------------------------------------------
// Print (me; s: in out OStream) returns OStream;
// ------------------------------------------------------------------
void Standard_Failure::Print (Standard_OStream& AStream) const
// =======================================================================
// function : Print
// purpose :
// =======================================================================
void Standard_Failure::Print (Standard_OStream& theStream) const
{
if(myMessage){
AStream << DynamicType() << ": " << GetMessageString();
}
else {
AStream << DynamicType();
}
if (myMessage != NULL)
{
theStream << DynamicType() << ": " << GetMessageString();
}
else
{
theStream << DynamicType();
}
if (myStackTrace != NULL)
{
theStream << GetStackString();
}
}
Handle(Standard_Failure) Standard_Failure::NewInstance(const Standard_CString AString)
// =======================================================================
// function : NewInstance
// purpose :
// =======================================================================
Handle(Standard_Failure) Standard_Failure::NewInstance (Standard_CString theString)
{
return new Standard_Failure(AString) ;
return new Standard_Failure (theString);
}
//=======================================================================
//function : GetMessageString
//purpose : Returns error message
//=======================================================================
Standard_CString Standard_Failure::GetMessageString () const
// =======================================================================
// function : NewInstance
// purpose :
// =======================================================================
Handle(Standard_Failure) Standard_Failure::NewInstance (Standard_CString theMessage,
Standard_CString theStackTrace)
{
return (myMessage ? myMessage+sizeof(Standard_Integer) : "");
return new Standard_Failure (theMessage, theStackTrace);
}
// =======================================================================
// function : GetNbStackTraces
// purpose :
// =======================================================================
Standard_Integer Standard_Failure::DefaultStackTraceLength()
{
return Standard_Failure_DefaultStackTraceLength;
}
// =======================================================================
// function : SetNbStackTraces
// purpose :
// =======================================================================
void Standard_Failure::SetDefaultStackTraceLength (Standard_Integer theNbStackTraces)
{
Standard_Failure_DefaultStackTraceLength = theNbStackTraces;
}

View File

@ -17,28 +17,20 @@
#ifndef _Standard_Failure_HeaderFile
#define _Standard_Failure_HeaderFile
#include <Standard.hxx>
#include <Standard_Type.hxx>
#include <Standard_CString.hxx>
#include <Standard_Transient.hxx>
#include <Standard_OStream.hxx>
#include <Standard_SStream.hxx>
class Standard_NoSuchObject;
class Standard_Failure;
DEFINE_STANDARD_HANDLE(Standard_Failure, Standard_Transient)
//! Forms the root of the entire exception hierarchy.
class Standard_Failure : public Standard_Transient
{
public:
//! Creates a status object of type "Failure".
Standard_EXPORT Standard_Failure();
@ -46,11 +38,18 @@ public:
Standard_EXPORT Standard_Failure (const Standard_Failure& f);
//! Creates a status object of type "Failure".
Standard_EXPORT Standard_Failure(const Standard_CString aString);
//! @param theDesc [in] exception description
Standard_EXPORT Standard_Failure (const Standard_CString theDesc);
//! Creates a status object of type "Failure" with stack trace.
//! @param theDesc [in] exception description
//! @param theStackTrace [in] associated stack trace
Standard_EXPORT Standard_Failure (const Standard_CString theDesc,
const Standard_CString theStackTrace);
//! Assignment operator
Standard_EXPORT Standard_Failure& operator= (const Standard_Failure& f);
//! Destructor
Standard_EXPORT ~Standard_Failure();
@ -63,15 +62,23 @@ public:
Standard_EXPORT virtual Standard_CString GetMessageString() const;
//! Sets error message
Standard_EXPORT virtual void SetMessageString (const Standard_CString aMessage);
Standard_EXPORT virtual void SetMessageString (const Standard_CString theMessage);
//! Returns the stack trace string
Standard_EXPORT virtual Standard_CString GetStackString() const;
//! Sets the stack trace string
Standard_EXPORT virtual void SetStackString (const Standard_CString theStack);
Standard_EXPORT void Reraise();
Standard_EXPORT void Reraise (const Standard_CString aMessage);
//! Reraises a caught exception and changes its error message.
Standard_EXPORT void Reraise (const Standard_SStream& aReason);
public:
//! Raises an exception of type "Failure" and associates
//! an error message to it. The message can be printed
//! in an exception handler.
@ -82,13 +89,24 @@ public:
//! at run-time.
Standard_EXPORT static void Raise (const Standard_SStream& aReason);
//! Used to construct an instance of the exception object
//! as a handle. Shall be used to protect against possible
//! construction of exception object in C stack -- that is
//! dangerous since some of methods require that object
//! was allocated dynamically.
Standard_EXPORT static Handle(Standard_Failure) NewInstance (const Standard_CString aMessage);
//! Used to construct an instance of the exception object as a handle.
//! Shall be used to protect against possible construction of exception object in C stack,
//! which is dangerous since some of methods require that object was allocated dynamically.
Standard_EXPORT static Handle(Standard_Failure) NewInstance (Standard_CString theMessage);
//! Used to construct an instance of the exception object as a handle.
Standard_EXPORT static Handle(Standard_Failure) NewInstance (Standard_CString theMessage,
Standard_CString theStackTrace);
//! Returns the default length of stack trace to be captured by Standard_Failure constructor;
//! 0 by default meaning no stack trace.
Standard_EXPORT static Standard_Integer DefaultStackTraceLength();
//! Sets default length of stack trace to be captured by Standard_Failure constructor.
Standard_EXPORT static void SetDefaultStackTraceLength (Standard_Integer theNbStackTraces);
public:
//! Used to throw CASCADE exception from C signal handler.
//! On platforms that do not allow throwing C++ exceptions
//! from this handler (e.g. Linux), uses longjump to get to
@ -102,41 +120,66 @@ public:
Standard_DEPRECATED("This method is deprecated (not thread-safe), use standard C++ mechanism instead")
Standard_EXPORT static Handle(Standard_Failure) Caught();
DEFINE_STANDARD_RTTIEXT(Standard_Failure,Standard_Transient)
protected:
//! Used only if standard C++ exceptions are used.
//! Throws exception of the same type as this by C++ throw,
//! and stores current object as last thrown exception,
//! to be accessible by method Caught()
Standard_EXPORT virtual void Throw() const;
private:
//! Reference-counted string,
//! Memory block is allocated with an extra 4-byte header (int representing number of references)
//! using low-level malloc() to avoid exceptions.
struct StringRef
{
Standard_Integer Counter;
Standard_Character Message[1];
//! Return message string.
Standard_CString GetMessage() const { return (Standard_CString )&Message[0]; }
//! Allocate reference-counted message string.
static StringRef* allocate_message (Standard_CString theString);
//! Copy reference-counted message string.
static StringRef* copy_message (StringRef* theString);
//! Release reference-counted message string.
static void deallocate_message (StringRef* theString);
};
private:
Standard_CString myMessage;
StringRef* myMessage;
StringRef* myStackTrace;
};
inline Standard_OStream& operator << (Standard_OStream& AStream,
const Handle(Standard_Failure)& AFailure)
// =======================================================================
// function : operator<<
// purpose :
// =======================================================================
inline Standard_OStream& operator<< (Standard_OStream& theStream,
const Handle(Standard_Failure)& theFailure)
{
AFailure->Print(AStream);
return AStream;
theFailure->Print (theStream);
return theStream;
}
inline Standard_OStream& operator << (Standard_OStream& AStream,
const Standard_Failure& AFailure)
// =======================================================================
// function : operator<<
// purpose :
// =======================================================================
inline Standard_OStream& operator<< (Standard_OStream& theStream,
const Standard_Failure& theFailure)
{
AFailure.Print(AStream);
return AStream;
theFailure.Print (theStream);
return theStream;
}
#endif // _Standard_Failure_HeaderFile

View File

@ -90,6 +90,17 @@
#define Standard_UNUSED
#endif
//! @def Standard_NOINLINE
//! Macro for disallowing function inlining.
//! Expands to "__attribute__((noinline))" on GCC and CLang.
#if defined(__clang__) || (defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 1)))
#define Standard_NOINLINE __attribute__((noinline))
#elif defined(_MSC_VER)
#define Standard_NOINLINE __declspec(noinline)
#else
#define Standard_NOINLINE
#endif
//! @def Standard_THREADLOCAL
//! Define Standard_THREADLOCAL modifier as C++11 thread_local keyword where it is available.
#if defined(__clang__)

View File

@ -89,12 +89,20 @@ void Standard_OutOfMemory::Raise(Standard_SStream& theMessage)
// global instance must be allocated at load-time
static Handle(Standard_OutOfMemory) anOutOfMemInstance = new Standard_OutOfMemory;
Handle(Standard_OutOfMemory) Standard_OutOfMemory::NewInstance(const Standard_CString theMessage)
Handle(Standard_OutOfMemory) Standard_OutOfMemory::NewInstance (Standard_CString theMessage)
{
anOutOfMemInstance->SetMessageString (theMessage);
return anOutOfMemInstance;
}
Handle(Standard_OutOfMemory) Standard_OutOfMemory::NewInstance (Standard_CString theMessage,
Standard_CString theStackTrace)
{
anOutOfMemInstance->SetMessageString (theMessage);
anOutOfMemInstance->SetStackString (theStackTrace);
return anOutOfMemInstance;
}
//=======================================================================
//function : Throw
//purpose :

View File

@ -68,7 +68,11 @@ public:
Standard_EXPORT static void Raise(Standard_SStream& theMessage);
//! Returns global instance of exception
Standard_EXPORT static Handle(Standard_OutOfMemory) NewInstance(const Standard_CString theMessage = "");
Standard_EXPORT static Handle(Standard_OutOfMemory) NewInstance (Standard_CString theMessage = "");
//! Returns global instance of exception
Standard_EXPORT static Handle(Standard_OutOfMemory) NewInstance (Standard_CString theMessage,
Standard_CString theStackTrace);
DEFINE_STANDARD_RTTIEXT(Standard_OutOfMemory,Standard_ProgramError)

View File

@ -0,0 +1,362 @@
// Created on: 2020-11-30
// Copyright (c) 2020 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and/or modify it under
// the terms of the GNU Lesser General Public License version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#include <Standard.hxx>
#include <Message.hxx>
#include <Standard_Mutex.hxx>
#if defined(__APPLE__)
#import <TargetConditionals.h>
#endif
#if defined(__EMSCRIPTEN__)
#include <emscripten/emscripten.h>
#elif defined(__ANDROID__)
//#include <unwind.h>
#elif defined(__QNX__)
//#include <backtrace.h> // requires linking to libbacktrace
#elif !defined(_WIN32) && !(defined(TARGET_OS_IPHONE) && TARGET_OS_IPHONE)
#include <execinfo.h>
#elif defined(_WIN32) && !defined(OCCT_UWP)
#include <Standard_WarningsDisable.hxx>
#include <dbghelp.h>
#include <Standard_WarningsRestore.hxx>
//! This is a wrapper of DbgHelp library loaded dynamically.
//! DbgHelp is coming with Windows SDK, so that technically it is always available.
//! However, it's usage requires extra steps:
//! - .pdb files are necessary to resolve function names;
//! Normal release DLLs without PDBs will show no much useful info.
//! - DbgHelp.dll and friends (SymSrv.dll, SrcSrv.dll) should be packaged with application;
//! DbgHelp.dll coming with system might be of other incompatible version
//! (some applications load it dynamically to avoid packaging extra DLL,
//! with a extra hacks determining library version)
class Standard_DbgHelper
{
public: // dbgHelp.dll function types
typedef BOOL (WINAPI *SYMINITIALIZEPROC) (HANDLE, PCSTR, BOOL);
typedef BOOL (WINAPI *STACKWALK64PROC) (DWORD, HANDLE, HANDLE, LPSTACKFRAME64,
PVOID, PREAD_PROCESS_MEMORY_ROUTINE64,
PFUNCTION_TABLE_ACCESS_ROUTINE64,
PGET_MODULE_BASE_ROUTINE64, PTRANSLATE_ADDRESS_ROUTINE64);
typedef BOOL (WINAPI *SYMCLEANUPPROC) (HANDLE);
typedef BOOL (WINAPI *SYMFROMADDRPROC) (HANDLE, DWORD64, PDWORD64, PSYMBOL_INFO);
public:
//! Return global instance.
static Standard_DbgHelper& GetDbgHelper()
{
static Standard_DbgHelper aDbgHelper;
return aDbgHelper;
}
//! Return global mutex.
static Standard_Mutex& Mutex()
{
static Standard_Mutex THE_MUTEX_LOCK;
return THE_MUTEX_LOCK;
}
public:
SYMINITIALIZEPROC SymInitialize;
SYMCLEANUPPROC SymCleanup;
STACKWALK64PROC StackWalk64;
PFUNCTION_TABLE_ACCESS_ROUTINE64 SymFunctionTableAccess64;
PGET_MODULE_BASE_ROUTINE64 SymGetModuleBase64;
SYMFROMADDRPROC SymFromAddr;
//! Return TRUE if library has been loaded.
bool IsLoaded() const { return myDbgHelpLib != NULL; }
//! Return loading error message.
const char* ErrorMessage() const { return myError; }
private:
//! Main constructor.
Standard_DbgHelper()
: SymInitialize (NULL),
SymCleanup (NULL),
StackWalk64 (NULL),
SymFunctionTableAccess64 (NULL),
SymGetModuleBase64 (NULL),
SymFromAddr (NULL),
myDbgHelpLib (LoadLibraryW (L"DbgHelp.dll")),
myError (NULL)
{
if (myDbgHelpLib == NULL)
{
myError = "Standard_DbgHelper, Failed to load DbgHelp.dll";
return;
}
if ((SymInitialize = (SYMINITIALIZEPROC) GetProcAddress (myDbgHelpLib, "SymInitialize")) == NULL)
{
myError = "Standard_DbgHelper, Function not found in DbgHelp.dll: SymInitialize";
unload();
return;
}
if ((SymCleanup = (SYMCLEANUPPROC) GetProcAddress (myDbgHelpLib, "SymCleanup")) == NULL)
{
myError = "Standard_DbgHelper, Function not found in DbgHelp.dll: SymCleanup";
unload();
return;
}
if ((StackWalk64 = (STACKWALK64PROC) GetProcAddress (myDbgHelpLib, "StackWalk64")) == NULL)
{
myError = "Standard_DbgHelper, Function not found in DbgHelp.dll: StackWalk64";
unload();
return;
}
if ((SymFunctionTableAccess64 = (PFUNCTION_TABLE_ACCESS_ROUTINE64) GetProcAddress (myDbgHelpLib, "SymFunctionTableAccess64")) == NULL)
{
myError = "Standard_DbgHelper, Function not found in DbgHelp.dll: SymFunctionTableAccess64";
unload();
return;
}
if ((SymGetModuleBase64 = (PGET_MODULE_BASE_ROUTINE64) GetProcAddress (myDbgHelpLib, "SymGetModuleBase64")) == NULL)
{
myError = "Standard_DbgHelper, Function not found in DbgHelp.dll: SymGetModuleBase64";
unload();
return;
}
if ((SymFromAddr = (SYMFROMADDRPROC) GetProcAddress (myDbgHelpLib, "SymFromAddr")) == NULL)
{
myError = "Standard_DbgHelper, Function not found in DbgHelp.dll: SymFromAddr";
unload();
return;
}
}
//! Destructor.
~Standard_DbgHelper()
{
// we could unload library here, but don't do it as it is kept loaded
//unload();
}
//! Unload library.
void unload()
{
if (myDbgHelpLib != NULL)
{
FreeLibrary (myDbgHelpLib);
myDbgHelpLib = NULL;
}
}
private:
Standard_DbgHelper (const Standard_DbgHelper& );
Standard_DbgHelper& operator= (const Standard_DbgHelper& );
private:
HMODULE myDbgHelpLib; //!< handle to DbgHelp
const char* myError; //!< loading error message
};
#endif
//=======================================================================
//function : StackTrace
//purpose :
//=======================================================================
Standard_Boolean Standard::StackTrace (char* theBuffer,
const int theBufferSize,
const int theNbTraces = 10,
void* theContext,
const int theNbTopSkip)
{
(void )theContext;
if (theBufferSize < 1
|| theNbTraces < 1
|| theBuffer == NULL
|| theNbTopSkip < 0)
{
return false;
}
#if defined(__EMSCRIPTEN__)
// theNbTraces is ignored
// EM_LOG_JS_STACK?
return emscripten_get_callstack (EM_LOG_C_STACK | EM_LOG_DEMANGLE | EM_LOG_NO_PATHS | EM_LOG_FUNC_PARAMS, theBuffer, theBufferSize) > 0;
#elif defined(__ANDROID__)
Message::SendTrace ("Standard::StackTrace() is not implemented for this platform");
return false;
#elif defined(__QNX__)
// bt_get_backtrace()
Message::SendTrace ("Standard::StackTrace() is not implemented for this platform");
return false;
#elif defined(OCCT_UWP) || (defined(TARGET_OS_IPHONE) && TARGET_OS_IPHONE)
Message::SendTrace ("Standard::StackTrace() is not implemented for this platform");
return false;
#elif defined(_WIN32)
// Each CPU architecture requires manual stack frame setup,
// and 32-bit version requires also additional hacks to retrieve current context;
// this implementation currently covers only x86_64 architecture.
#if defined(_M_X64)
int aNbTraces = theNbTraces;
const HANDLE anHProcess = GetCurrentProcess();
const HANDLE anHThread = GetCurrentThread();
CONTEXT aCtx;
if (theContext != NULL)
{
memcpy (&aCtx, theContext, sizeof(aCtx));
}
else
{
++aNbTraces;
memset (&aCtx, 0, sizeof(aCtx));
aCtx.ContextFlags = CONTEXT_FULL;
RtlCaptureContext (&aCtx);
}
// DbgHelp is not thread-safe library, hence global lock is used for serial access
Standard_Mutex::Sentry aSentry (Standard_DbgHelper::Mutex());
Standard_DbgHelper& aDbgHelp = Standard_DbgHelper::GetDbgHelper();
if (!aDbgHelp.IsLoaded())
{
strcat_s (theBuffer, theBufferSize, "\n==Backtrace==\n");
strcat_s (theBuffer, theBufferSize, aDbgHelp.ErrorMessage());
strcat_s (theBuffer, theBufferSize, "\n=============");
return false;
}
aDbgHelp.SymInitialize (anHProcess, NULL, TRUE);
DWORD anImage = 0;
STACKFRAME64 aStackFrame;
memset (&aStackFrame, 0, sizeof(aStackFrame));
anImage = IMAGE_FILE_MACHINE_AMD64;
aStackFrame.AddrPC.Offset = aCtx.Rip;
aStackFrame.AddrPC.Mode = AddrModeFlat;
aStackFrame.AddrFrame.Offset = aCtx.Rsp;
aStackFrame.AddrFrame.Mode = AddrModeFlat;
aStackFrame.AddrStack.Offset = aCtx.Rsp;
aStackFrame.AddrStack.Mode = AddrModeFlat;
char aModBuffer[MAX_PATH] = {};
char aSymBuffer[sizeof(SYMBOL_INFO) + MAX_SYM_NAME * sizeof(CHAR)];
SYMBOL_INFO* aSymbol = (SYMBOL_INFO*) aSymBuffer;
aSymbol->SizeOfStruct = sizeof(SYMBOL_INFO);
aSymbol->MaxNameLen = MAX_SYM_NAME;
int aTopSkip = theNbTopSkip + 1; // skip this function call and specified extra number
strcat_s (theBuffer, theBufferSize, "\n==Backtrace==");
for (int aLineIter = 0; aLineIter < aNbTraces; ++aLineIter)
{
BOOL aRes = aDbgHelp.StackWalk64 (anImage, anHProcess, anHThread,
&aStackFrame, &aCtx, NULL,
aDbgHelp.SymFunctionTableAccess64, aDbgHelp.SymGetModuleBase64, NULL);
if (!aRes)
{
break;
}
if (theContext == NULL && aTopSkip > 0)
{
--aTopSkip;
continue;
}
if (aStackFrame.AddrPC.Offset == 0)
{
break;
}
strcat_s (theBuffer, theBufferSize, "\n");
const DWORD64 aModuleBase = aDbgHelp.SymGetModuleBase64 (anHProcess, aStackFrame.AddrPC.Offset);
if (aModuleBase != 0
&& GetModuleFileNameA ((HINSTANCE) aModuleBase, aModBuffer, MAX_PATH))
{
strcat_s (theBuffer, theBufferSize, aModBuffer);
}
DWORD64 aDisp = 0;
strcat_s (theBuffer, theBufferSize, "(");
if (aDbgHelp.SymFromAddr (anHProcess, aStackFrame.AddrPC.Offset, &aDisp, aSymbol))
{
strcat_s (theBuffer, theBufferSize, aSymbol->Name);
}
else
{
strcat_s (theBuffer, theBufferSize, "???");
}
strcat_s (theBuffer, theBufferSize, ")");
}
strcat_s (theBuffer, theBufferSize, "\n=============");
aDbgHelp.SymCleanup (anHProcess);
return true;
#else
Message::SendTrace ("Standard::StackTrace() is not implemented for this CPU architecture");
return false;
#endif
#else
const int aTopSkip = theNbTopSkip + 1; // skip this function call and specified extra number
int aNbTraces = theNbTraces + aTopSkip;
void** aStackArr = (void** )alloca (sizeof(void*) * aNbTraces);
if (aStackArr == NULL)
{
return false;
}
aNbTraces = ::backtrace (aStackArr, aNbTraces);
if (aNbTraces <= 1)
{
return false;
}
aNbTraces -= aTopSkip;
char** aStrings = ::backtrace_symbols (aStackArr + aTopSkip, aNbTraces);
if (aStrings == NULL)
{
return false;
}
const size_t aLenInit = strlen (theBuffer);
size_t aLimit = (size_t) theBufferSize - aLenInit - 1;
if (aLimit > 14)
{
strcat (theBuffer, "\n==Backtrace==");
aLimit -= 14;
}
for (int aLineIter = 0; aLineIter < aNbTraces; ++aLineIter)
{
const size_t aLen = strlen (aStrings[aLineIter]);
if (aLen + 1 >= aLimit)
{
break;
}
strcat (theBuffer, "\n");
strcat (theBuffer, aStrings[aLineIter]);
aLimit -= aLen + 1;
}
free (aStrings);
if (aLimit > 14)
{
strcat (theBuffer, "\n=============");
}
return true;
#endif
}

View File

@ -0,0 +1,22 @@
puts "================"
puts "0030762: Foundation Classes - include backtrace within OSD_SIGSEGV on Linux"
puts "================"
puts ""
pload QAcommands
dsetsignal set -strackTraceLength 10
set IsDone [catch {set aResult [OCC30762]} result]
if { ${IsDone} != 0 } {
puts "result = ${result}"
puts "Error: command raised exception"
} else {
if { [string first "testMethod3" $aResult] != -1 } {
puts "OK test case"
} else {
# stack trace might be missing due to stripped symbols or optimized code
#puts "Error: backtrace is not printed"
puts "Warning: backtrace is not printed"
}
}