stack_trace.cc 2.6 KB
Newer Older
1 2 3 4 5
//  Copyright (c) 2013, Facebook, Inc.  All rights reserved.
//  This source code is licensed under the BSD-style license found in the
//  LICENSE file in the root directory of this source tree. An additional grant
//  of patent rights can be found in the PATENTS file in the same directory.
//
6 7 8 9 10 11 12 13 14 15 16
#include "util/stack_trace.h"

#ifdef OS_LINUX

#include <execinfo.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

17
namespace rocksdb {
18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33

static const char* GetExecutableName()
{
  static char name[1024];

  char link[1024];
  snprintf(link, sizeof(link), "/proc/%d/exe", getpid());
  auto read = readlink(link, name, sizeof(name));
  if (-1 == read) {
    return nullptr;
  } else {
    name[read] = 0;
    return name;
  }
}

34
void PrintStack(int first_frames_to_skip) {
35 36 37 38 39 40 41 42
  const int kMaxFrames = 100;
  void *frames[kMaxFrames];

  auto num_frames = backtrace(frames, kMaxFrames);
  auto symbols = backtrace_symbols(frames, num_frames);

  auto executable = GetExecutableName();

43 44
  for (int i = first_frames_to_skip; i < num_frames; ++i) {
    fprintf(stderr, "#%-2d  ", i - first_frames_to_skip);
45
    if (symbols) {
46
      fprintf(stderr, "%s ", symbols[i]);
47 48 49 50 51
    }
    if (executable) {
      // out source to addr2line, for the address translation
      const int kLineMax = 256;
      char cmd[kLineMax];
52
      sprintf(cmd, "addr2line %p -e %s -f -C 2>&1", frames[i], executable);
53 54 55 56
      auto f = popen(cmd, "r");
      if (f) {
        char line[kLineMax];
        while (fgets(line, sizeof(line), f)) {
57 58
          line[strlen(line) - 1] = 0; // remove newline
          fprintf(stderr, "%s\t", line);
59 60 61 62
        }
        pclose(f);
      }
    } else {
63
      fprintf(stderr, " %p", frames[i]);
64
    }
65
    fprintf(stderr, "\n");
66
  }
67
}
68

69 70 71 72 73 74
static void StackTraceHandler(int sig) {
  // reset to default handler
  signal(sig, SIG_DFL);
  fprintf(stderr, "Received signal %d (%s)\n", sig, strsignal(sig));
  // skip the top three signal handler related frames
  PrintStack(3);
75 76 77 78 79 80 81 82 83 84 85
  // re-signal to default handler (so we still get core dump if needed...)
  raise(sig);
}

void InstallStackTraceHandler() {
  // just use the plain old signal as it's simple and sufficient
  // for this use case
  signal(SIGILL, StackTraceHandler);
  signal(SIGSEGV, StackTraceHandler);
  signal(SIGBUS, StackTraceHandler);
  signal(SIGABRT, StackTraceHandler);
86 87 88

  printf("Installed stack trace handler for SIGILL SIGSEGV SIGBUS SIGABRT\n");

89 90
}

91
}   // namespace rocksdb
92 93 94

#else // no-op for non-linux system for now

95
namespace rocksdb {
96 97

void InstallStackTraceHandler() {}
98
void PrintStack(int first_frames_to_skip) {}
99 100 101 102

}

#endif // OS_LINUX