164b6ce7b2
Remove useless comments and header, and de duplicate a small declaration. Change-Id: I0d4475cbbd61ea964f8e586cf0e1f68eb91c4948 Reviewed-on: https://gerrit.libreoffice.org/6016 Reviewed-by: Arnaud Versini <arnaud.versini@libreoffice.org> Tested-by: Arnaud Versini <arnaud.versini@libreoffice.org>
278 lines
7 KiB
C++
278 lines
7 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
#include "osl/diagnose.h"
|
|
#include "system.h"
|
|
|
|
#ifndef HAVE_DLFCN_H
|
|
|
|
#if defined(LINUX) || defined(SOLARIS)
|
|
#define HAVE_DLFCN_H
|
|
#endif /* LINUX || SOLARIS */
|
|
|
|
#endif /* HAVE_DLFCN_H */
|
|
|
|
|
|
#ifdef HAVE_DLFCN_H
|
|
|
|
#ifndef INCLUDED_DLFCN_H
|
|
#include <dlfcn.h>
|
|
#define INCLUDED_DLFCN_H
|
|
#endif
|
|
|
|
#endif /* HAVE_DLFCN_H */
|
|
|
|
#include <pthread.h>
|
|
#include <stddef.h>
|
|
|
|
/************************************************************************/
|
|
/* Internal data structures and functions */
|
|
/************************************************************************/
|
|
|
|
static pthread_mutex_t g_mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
typedef pfunc_osl_printDebugMessage oslDebugMessageFunc;
|
|
static oslDebugMessageFunc volatile g_pDebugMessageFunc = 0;
|
|
|
|
typedef pfunc_osl_printDetailedDebugMessage oslDetailedDebugMessageFunc;
|
|
static oslDetailedDebugMessageFunc volatile g_pDetailedDebugMessageFunc = 0;
|
|
|
|
static void osl_diagnose_backtrace_Impl (
|
|
oslDebugMessageFunc f);
|
|
|
|
#define OSL_DIAGNOSE_OUTPUTMESSAGE(f, s) \
|
|
((f != 0) ? (*(f))((s)) : (void)fprintf(stderr, "%s", (s)))
|
|
|
|
#if defined (LINUX) || defined (SOLARIS)
|
|
|
|
static void osl_diagnose_frame_Impl (
|
|
oslDebugMessageFunc f,
|
|
int depth,
|
|
void * pc)
|
|
{
|
|
const char *fname = 0, *sname = 0;
|
|
void *fbase = 0, *saddr = 0;
|
|
ptrdiff_t offset;
|
|
char szMessage[1024];
|
|
|
|
#ifdef INCLUDED_DLFCN_H
|
|
Dl_info dli;
|
|
if (dladdr (pc, &dli) != 0)
|
|
{
|
|
fname = dli.dli_fname;
|
|
fbase = dli.dli_fbase;
|
|
sname = dli.dli_sname;
|
|
saddr = dli.dli_saddr;
|
|
}
|
|
#endif /* INCLUDED_DLFCN_H */
|
|
|
|
if (saddr)
|
|
offset = (ptrdiff_t)(pc) - (ptrdiff_t)(saddr);
|
|
else if (fbase)
|
|
offset = (ptrdiff_t)(pc) - (ptrdiff_t)(fbase);
|
|
else
|
|
offset = (ptrdiff_t)(pc);
|
|
|
|
snprintf (szMessage, sizeof(szMessage),
|
|
"Backtrace: [%d] %s: %s+0x%" SAL_PRI_PTRDIFFT "x\n",
|
|
depth,
|
|
fname ? fname : "<unknown>",
|
|
sname ? sname : "???",
|
|
offset);
|
|
|
|
OSL_DIAGNOSE_OUTPUTMESSAGE(f, szMessage);
|
|
}
|
|
#endif
|
|
|
|
#if defined(LINUX)
|
|
|
|
#include <execinfo.h>
|
|
|
|
#define FRAME_COUNT 64
|
|
#define FRAME_OFFSET 1
|
|
|
|
static void osl_diagnose_backtrace_Impl (oslDebugMessageFunc f)
|
|
{
|
|
void * ppFrames[FRAME_COUNT];
|
|
int i, n;
|
|
|
|
n = backtrace (ppFrames, FRAME_COUNT);
|
|
for (i = FRAME_OFFSET; i < n; i++)
|
|
{
|
|
osl_diagnose_frame_Impl (f, (i - FRAME_OFFSET), ppFrames[i]);
|
|
}
|
|
}
|
|
|
|
#elif defined(SOLARIS)
|
|
|
|
#include <setjmp.h>
|
|
#include <sys/frame.h>
|
|
|
|
#if defined(SPARC)
|
|
|
|
#if defined IS_LP64
|
|
|
|
#define FRAME_PTR_OFFSET 1
|
|
#define FRAME_OFFSET 0
|
|
#define STACK_BIAS 0x7ff
|
|
|
|
#else
|
|
|
|
#define FRAME_PTR_OFFSET 1
|
|
#define FRAME_OFFSET 0
|
|
#define STACK_BIAS 0
|
|
|
|
#endif
|
|
|
|
#elif defined(INTEL)
|
|
|
|
#define FRAME_PTR_OFFSET 3
|
|
#define FRAME_OFFSET 0
|
|
#define STACK_BIAS 0
|
|
|
|
#endif /* (SPARC || INTEL) */
|
|
|
|
static void osl_diagnose_backtrace_Impl (oslDebugMessageFunc f)
|
|
{
|
|
jmp_buf ctx;
|
|
long fpval;
|
|
struct frame * fp;
|
|
int i;
|
|
|
|
#if defined(SPARC)
|
|
asm("ta 3");
|
|
#endif /* SPARC */
|
|
setjmp (ctx);
|
|
|
|
fpval = ((long*)(ctx))[FRAME_PTR_OFFSET];
|
|
fp = (struct frame*)((char*)(fpval) + STACK_BIAS);
|
|
|
|
for (i = 0; (i < FRAME_OFFSET) && (fp != 0); i++)
|
|
fp = (struct frame*)((char*)(fp->fr_savfp) + STACK_BIAS);
|
|
|
|
for (i = 0; (fp != 0) && (fp->fr_savpc != 0); i++)
|
|
{
|
|
struct frame * prev = (struct frame*)((char*)(fp->fr_savfp) + STACK_BIAS);
|
|
osl_diagnose_frame_Impl (f, i, (void*)(fp->fr_savpc));
|
|
fp = (prev > fp) ? prev : 0;
|
|
}
|
|
}
|
|
|
|
#else /* (LINUX || SOLARIS) */
|
|
|
|
static void osl_diagnose_backtrace_Impl (oslDebugMessageFunc f)
|
|
{
|
|
(void) f;
|
|
/* not yet implemented */
|
|
}
|
|
|
|
#endif /* (LINUX || SOLARIS) */
|
|
|
|
namespace {
|
|
|
|
// getenv is not thread safe, so minimize use of result:
|
|
bool isEnv(char const * name) {
|
|
char * p = getenv(name);
|
|
return p != NULL && *p != '\0';
|
|
}
|
|
|
|
}
|
|
|
|
sal_Bool SAL_CALL osl_assertFailedLine (
|
|
const sal_Char* pszFileName,
|
|
sal_Int32 nLine,
|
|
const sal_Char* pszMessage)
|
|
{
|
|
oslDebugMessageFunc f = g_pDebugMessageFunc;
|
|
char szMessage[1024];
|
|
|
|
// after reporting the assertion, abort if told so by SAL_DIAGNOSE_ABORT, but *not* if
|
|
// assertions are routed to some external instance
|
|
static bool envAbort = isEnv( "SAL_DIAGNOSE_ABORT" );
|
|
static bool envBacktrace = isEnv( "SAL_DIAGNOSE_BACKTRACE" );
|
|
sal_Bool const doAbort = envAbort && f == NULL;
|
|
|
|
/* If there's a callback for detailed messages, use it */
|
|
if ( g_pDetailedDebugMessageFunc != NULL )
|
|
{
|
|
g_pDetailedDebugMessageFunc( pszFileName, nLine, pszMessage );
|
|
return sal_False;
|
|
}
|
|
|
|
/* format message into buffer */
|
|
if (pszMessage != 0)
|
|
{
|
|
snprintf(szMessage, sizeof(szMessage),
|
|
"Error: File %s, Line %" SAL_PRIdINT32 ": %s\n",
|
|
pszFileName, nLine, pszMessage);
|
|
}
|
|
else
|
|
{
|
|
snprintf(szMessage, sizeof(szMessage),
|
|
"Error: File %s, Line %" SAL_PRIdINT32 "\n",
|
|
pszFileName, nLine);
|
|
}
|
|
|
|
/* acquire lock to serialize output message(s) */
|
|
pthread_mutex_lock(&g_mutex);
|
|
|
|
/* output message buffer */
|
|
OSL_DIAGNOSE_OUTPUTMESSAGE(f, szMessage);
|
|
|
|
/* should we output backtrace? */
|
|
if( envBacktrace )
|
|
osl_diagnose_backtrace_Impl(f);
|
|
|
|
/* release lock and leave */
|
|
pthread_mutex_unlock(&g_mutex);
|
|
|
|
return doAbort;
|
|
}
|
|
|
|
void SAL_CALL osl_breakDebug()
|
|
{
|
|
abort();
|
|
}
|
|
|
|
sal_Int32 SAL_CALL osl_reportError (
|
|
sal_uInt32 nType,
|
|
const sal_Char* pszMessage)
|
|
{
|
|
(void) nType; /* unused */
|
|
fputs(pszMessage, stderr);
|
|
return 0;
|
|
}
|
|
|
|
oslDebugMessageFunc SAL_CALL osl_setDebugMessageFunc (
|
|
oslDebugMessageFunc pNewFunc)
|
|
{
|
|
oslDebugMessageFunc pOldFunc = g_pDebugMessageFunc;
|
|
g_pDebugMessageFunc = pNewFunc;
|
|
return pOldFunc;
|
|
}
|
|
|
|
pfunc_osl_printDetailedDebugMessage SAL_CALL osl_setDetailedDebugMessageFunc (
|
|
pfunc_osl_printDetailedDebugMessage pNewFunc)
|
|
{
|
|
oslDetailedDebugMessageFunc pOldFunc = g_pDetailedDebugMessageFunc;
|
|
g_pDetailedDebugMessageFunc = pNewFunc;
|
|
return pOldFunc;
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|