2013-09-22 02:03:52 +00:00
|
|
|
// -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*-
|
2007-03-22 03:00:33 +00:00
|
|
|
// Copyright (c) 2005, Google Inc.
|
|
|
|
// All rights reserved.
|
2014-02-08 23:30:36 +00:00
|
|
|
//
|
2007-03-22 03:00:33 +00:00
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are
|
|
|
|
// met:
|
2014-02-08 23:30:36 +00:00
|
|
|
//
|
2007-03-22 03:00:33 +00:00
|
|
|
// * Redistributions of source code must retain the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer.
|
|
|
|
// * Redistributions in binary form must reproduce the above
|
|
|
|
// copyright notice, this list of conditions and the following disclaimer
|
|
|
|
// in the documentation and/or other materials provided with the
|
|
|
|
// distribution.
|
|
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
|
|
// contributors may be used to endorse or promote products derived from
|
|
|
|
// this software without specific prior written permission.
|
2014-02-08 23:30:36 +00:00
|
|
|
//
|
2007-03-22 03:00:33 +00:00
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
// ---
|
|
|
|
// Author: Sanjay Ghemawat
|
|
|
|
//
|
2007-04-16 20:49:32 +00:00
|
|
|
// Produce stack trace.
|
|
|
|
//
|
|
|
|
// There are three different ways we can try to get the stack trace:
|
|
|
|
//
|
|
|
|
// 1) Our hand-coded stack-unwinder. This depends on a certain stack
|
|
|
|
// layout, which is used by gcc (and those systems using a
|
|
|
|
// gcc-compatible ABI) on x86 systems, at least since gcc 2.95.
|
|
|
|
// It uses the frame pointer to do its work.
|
|
|
|
//
|
|
|
|
// 2) The libunwind library. This is still in development, and as a
|
|
|
|
// separate library adds a new dependency, abut doesn't need a frame
|
|
|
|
// pointer. It also doesn't call malloc.
|
|
|
|
//
|
|
|
|
// 3) The gdb unwinder -- also the one used by the c++ exception code.
|
|
|
|
// It's obviously well-tested, but has a fatal flaw: it can call
|
|
|
|
// malloc() from the unwinder. This is a problem because we're
|
|
|
|
// trying to use the unwinder to instrument malloc().
|
|
|
|
//
|
|
|
|
// Note: if you add a new implementation here, make sure it works
|
|
|
|
// correctly when GetStackTrace() is called with max_depth == 0.
|
|
|
|
// Some code may do that.
|
2007-03-22 03:00:33 +00:00
|
|
|
|
2009-04-21 17:10:29 +00:00
|
|
|
#include <config.h>
|
2014-02-08 23:30:36 +00:00
|
|
|
#include <stdlib.h> // for getenv
|
|
|
|
#include <string.h> // for strcmp
|
|
|
|
#include <stdio.h> // for fprintf
|
|
|
|
#include "gperftools/stacktrace.h"
|
|
|
|
#include "base/commandlineflags.h"
|
|
|
|
#include "base/googleinit.h"
|
|
|
|
|
|
|
|
|
|
|
|
// we're using plain struct and not class to avoid any possible issues
|
|
|
|
// during initialization. Struct of pointers is easy to init at
|
|
|
|
// link-time.
|
|
|
|
struct GetStackImplementation {
|
|
|
|
int (*GetStackFramesPtr)(void** result, int* sizes, int max_depth,
|
|
|
|
int skip_count);
|
|
|
|
|
|
|
|
int (*GetStackFramesWithContextPtr)(void** result, int* sizes, int max_depth,
|
|
|
|
int skip_count, const void *uc);
|
|
|
|
|
|
|
|
int (*GetStackTracePtr)(void** result, int max_depth,
|
|
|
|
int skip_count);
|
|
|
|
|
|
|
|
int (*GetStackTraceWithContextPtr)(void** result, int max_depth,
|
|
|
|
int skip_count, const void *uc);
|
|
|
|
|
|
|
|
const char *name;
|
|
|
|
};
|
|
|
|
|
|
|
|
#if HAVE_DECL_BACKTRACE
|
|
|
|
#define STACKTRACE_INL_HEADER "stacktrace_generic-inl.h"
|
|
|
|
#define GST_SUFFIX generic
|
|
|
|
#include "stacktrace_impl_setup-inl.h"
|
|
|
|
#undef GST_SUFFIX
|
|
|
|
#undef STACKTRACE_INL_HEADER
|
|
|
|
#define HAVE_GST_generic
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAVE_LIBUNWIND_H
|
|
|
|
#define STACKTRACE_INL_HEADER "stacktrace_libunwind-inl.h"
|
|
|
|
#define GST_SUFFIX libunwind
|
|
|
|
#include "stacktrace_impl_setup-inl.h"
|
|
|
|
#undef GST_SUFFIX
|
|
|
|
#undef STACKTRACE_INL_HEADER
|
|
|
|
#define HAVE_GST_libunwind
|
|
|
|
#endif // HAVE_LIBUNWIND_H
|
|
|
|
|
|
|
|
#if defined(__i386__) || defined(__x86_64__)
|
|
|
|
#define STACKTRACE_INL_HEADER "stacktrace_x86-inl.h"
|
|
|
|
#define GST_SUFFIX x86
|
|
|
|
#include "stacktrace_impl_setup-inl.h"
|
|
|
|
#undef GST_SUFFIX
|
|
|
|
#undef STACKTRACE_INL_HEADER
|
|
|
|
#define HAVE_GST_x86
|
|
|
|
#endif // i386 || x86_64
|
|
|
|
|
|
|
|
#if defined(__ppc__) || defined(__PPC__)
|
|
|
|
#define STACKTRACE_INL_HEADER "stacktrace_powerpc-inl.h"
|
|
|
|
#define GST_SUFFIX ppc
|
|
|
|
#include "stacktrace_impl_setup-inl.h"
|
|
|
|
#undef GST_SUFFIX
|
|
|
|
#undef STACKTRACE_INL_HEADER
|
|
|
|
#define HAVE_GST_ppc
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__arm__)
|
|
|
|
#define STACKTRACE_INL_HEADER "stacktrace_arm-inl.h"
|
|
|
|
#define GST_SUFFIX arm
|
|
|
|
#include "stacktrace_impl_setup-inl.h"
|
|
|
|
#undef GST_SUFFIX
|
|
|
|
#undef STACKTRACE_INL_HEADER
|
|
|
|
#define HAVE_GST_arm
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef TCMALLOC_ENABLE_INSTRUMENT_STACKTRACE
|
|
|
|
#define STACKTRACE_INL_HEADER "stacktrace_instrument-inl.h"
|
|
|
|
#define GST_SUFFIX instrument
|
|
|
|
#include "stacktrace_impl_setup-inl.h"
|
|
|
|
#undef GST_SUFFIX
|
|
|
|
#undef STACKTRACE_INL_HEADER
|
|
|
|
#define HAVE_GST_instrument
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// The Windows case -- probably cygwin and mingw will use one of the
|
|
|
|
// x86-includes above, but if not, we can fall back to windows intrinsics.
|
|
|
|
#if defined(_WIN32) || defined(__CYGWIN__) || defined(__CYGWIN32__) || defined(__MINGW32__)
|
|
|
|
#define STACKTRACE_INL_HEADER "stacktrace_win32-inl.h"
|
|
|
|
#define GST_SUFFIX win32
|
|
|
|
#include "stacktrace_impl_setup-inl.h"
|
|
|
|
#undef GST_SUFFIX
|
|
|
|
#undef STACKTRACE_INL_HEADER
|
|
|
|
#define HAVE_GST_win32
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static GetStackImplementation *all_impls[] = {
|
|
|
|
#ifdef HAVE_GST_generic
|
|
|
|
&impl__generic,
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_GST_libunwind
|
|
|
|
&impl__libunwind,
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_GST_x86
|
|
|
|
&impl__x86,
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_GST_arm
|
|
|
|
&impl__arm,
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_GST_ppc
|
|
|
|
&impl__ppc,
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_GST_instrument
|
|
|
|
&impl__instrument,
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_GST_win32
|
|
|
|
&impl__win32,
|
|
|
|
#endif
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
// ppc and i386 implementations prefer arch-specific asm implementations.
|
|
|
|
// arm's asm implementation is broken
|
|
|
|
#if defined(__i386__) || defined(__x86_64__) || defined(__ppc__) || defined(__PPC__)
|
|
|
|
#if !defined(NO_FRAME_POINTER)
|
|
|
|
#define TCMALLOC_DONT_PREFER_LIBUNWIND
|
|
|
|
#endif
|
|
|
|
#endif
|
2010-03-23 20:39:55 +00:00
|
|
|
|
2014-02-08 23:30:36 +00:00
|
|
|
#if defined(HAVE_GST_instrument)
|
|
|
|
static GetStackImplementation *get_stack_impl = &impl__instrument;
|
|
|
|
#elif defined(HAVE_GST_win32)
|
|
|
|
static GetStackImplementation *get_stack_impl = &impl__win32;
|
|
|
|
#elif defined(HAVE_GST_x86) && defined(TCMALLOC_DONT_PREFER_LIBUNWIND)
|
|
|
|
static GetStackImplementation *get_stack_impl = &impl__x86;
|
|
|
|
#elif defined(HAVE_GST_ppc) && defined(TCMALLOC_DONT_PREFER_LIBUNWIND)
|
|
|
|
static GetStackImplementation *get_stack_impl = &impl__ppc;
|
|
|
|
#elif defined(HAVE_GST_libunwind)
|
|
|
|
static GetStackImplementation *get_stack_impl = &impl__libunwind;
|
|
|
|
#elif defined(HAVE_GST_arm)
|
|
|
|
static GetStackImplementation *get_stack_impl = &impl__arm;
|
|
|
|
#elif defined(HAVE_GST_generic)
|
|
|
|
static GetStackImplementation *get_stack_impl = &impl__generic;
|
2009-03-11 20:50:03 +00:00
|
|
|
#elif 0
|
|
|
|
// This is for the benefit of code analysis tools that may have
|
|
|
|
// trouble with the computed #include above.
|
2011-07-16 01:07:10 +00:00
|
|
|
# include "stacktrace_x86-inl.h"
|
|
|
|
# include "stacktrace_libunwind-inl.h"
|
|
|
|
# include "stacktrace_generic-inl.h"
|
|
|
|
# include "stacktrace_powerpc-inl.h"
|
|
|
|
# include "stacktrace_win32-inl.h"
|
2011-11-17 00:11:42 +00:00
|
|
|
# include "stacktrace_arm-inl.h"
|
2014-02-08 23:30:36 +00:00
|
|
|
# include "stacktrace_instrument-inl.h"
|
2007-04-16 20:49:32 +00:00
|
|
|
#else
|
2014-02-08 23:30:36 +00:00
|
|
|
#error Cannot calculate stack trace: will need to write for your environment
|
2007-03-22 03:00:33 +00:00
|
|
|
#endif
|
2014-02-08 23:30:36 +00:00
|
|
|
|
|
|
|
static int ATTRIBUTE_NOINLINE frame_forcer(int rv) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
PERFTOOLS_DLL_DECL int GetStackFrames(void** result, int* sizes, int max_depth,
|
|
|
|
int skip_count) {
|
|
|
|
return frame_forcer(get_stack_impl->GetStackFramesPtr(result, sizes, max_depth, skip_count));
|
|
|
|
}
|
|
|
|
|
|
|
|
PERFTOOLS_DLL_DECL int GetStackFramesWithContext(void** result, int* sizes, int max_depth,
|
|
|
|
int skip_count, const void *uc) {
|
|
|
|
return frame_forcer(get_stack_impl->GetStackFramesWithContextPtr(
|
|
|
|
result, sizes, max_depth,
|
|
|
|
skip_count, uc));
|
|
|
|
}
|
|
|
|
|
|
|
|
PERFTOOLS_DLL_DECL int GetStackTrace(void** result, int max_depth,
|
|
|
|
int skip_count) {
|
|
|
|
return frame_forcer(get_stack_impl->GetStackTracePtr(result, max_depth, skip_count));
|
|
|
|
}
|
|
|
|
|
|
|
|
PERFTOOLS_DLL_DECL int GetStackTraceWithContext(void** result, int max_depth,
|
|
|
|
int skip_count, const void *uc) {
|
|
|
|
return frame_forcer(get_stack_impl->GetStackTraceWithContextPtr(
|
|
|
|
result, max_depth, skip_count, uc));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init_default_stack_impl_inner(void) {
|
|
|
|
char *val = getenv("TCMALLOC_STACKTRACE_METHOD");
|
|
|
|
if (!val || !*val) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (GetStackImplementation **p = all_impls; *p; p++) {
|
|
|
|
GetStackImplementation *c = *p;
|
|
|
|
if (strcmp(c->name, val) == 0) {
|
|
|
|
get_stack_impl = c;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr, "Unknown or unsupported stacktrace method requested: %s. Ignoring it\n", val);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init_default_stack_impl(void) {
|
|
|
|
init_default_stack_impl_inner();
|
|
|
|
if (EnvToBool("TCMALLOC_STACKTRACE_METHOD_VERBOSE", false)) {
|
|
|
|
fprintf(stderr, "Chosen stacktrace method is %s\nSupported methods:\n", get_stack_impl->name);
|
|
|
|
for (GetStackImplementation **p = all_impls; *p; p++) {
|
|
|
|
GetStackImplementation *c = *p;
|
|
|
|
fprintf(stderr, "* %s\n", c->name);
|
|
|
|
}
|
|
|
|
fputs("\n", stderr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
REGISTER_MODULE_INITIALIZER(stacktrace_init_default_stack_impl, init_default_stack_impl());
|