Blame src/utilities.cc

Packit 18d29c
// Copyright (c) 2008, Google Inc.
Packit 18d29c
// All rights reserved.
Packit 18d29c
//
Packit 18d29c
// Redistribution and use in source and binary forms, with or without
Packit 18d29c
// modification, are permitted provided that the following conditions are
Packit 18d29c
// met:
Packit 18d29c
//
Packit 18d29c
//     * Redistributions of source code must retain the above copyright
Packit 18d29c
// notice, this list of conditions and the following disclaimer.
Packit 18d29c
//     * Redistributions in binary form must reproduce the above
Packit 18d29c
// copyright notice, this list of conditions and the following disclaimer
Packit 18d29c
// in the documentation and/or other materials provided with the
Packit 18d29c
// distribution.
Packit 18d29c
//     * Neither the name of Google Inc. nor the names of its
Packit 18d29c
// contributors may be used to endorse or promote products derived from
Packit 18d29c
// this software without specific prior written permission.
Packit 18d29c
//
Packit 18d29c
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit 18d29c
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit 18d29c
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit 18d29c
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
Packit 18d29c
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
Packit 18d29c
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
Packit 18d29c
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit 18d29c
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit 18d29c
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit 18d29c
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
Packit 18d29c
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit 18d29c
//
Packit 18d29c
// Author: Shinichiro Hamaji
Packit 18d29c
Packit 18d29c
#include "utilities.h"
Packit 18d29c
Packit 18d29c
#include <stdio.h>
Packit 18d29c
#include <stdlib.h>
Packit 18d29c
Packit 18d29c
#include <signal.h>
Packit 18d29c
#ifdef HAVE_SYS_TIME_H
Packit 18d29c
# include <sys/time.h>
Packit 18d29c
#endif
Packit 18d29c
#include <time.h>
Packit 18d29c
#if defined(HAVE_SYSCALL_H)
Packit 18d29c
#include <syscall.h>                 // for syscall()
Packit 18d29c
#elif defined(HAVE_SYS_SYSCALL_H)
Packit 18d29c
#include <sys/syscall.h>                 // for syscall()
Packit 18d29c
#endif
Packit 18d29c
#ifdef HAVE_SYSLOG_H
Packit 18d29c
# include <syslog.h>
Packit 18d29c
#endif
Packit 18d29c
Packit 18d29c
#include "base/googleinit.h"
Packit 18d29c
Packit 18d29c
using std::string;
Packit 18d29c
Packit 18d29c
_START_GOOGLE_NAMESPACE_
Packit 18d29c
Packit 18d29c
static const char* g_program_invocation_short_name = NULL;
Packit 18d29c
static pthread_t g_main_thread_id;
Packit 18d29c
Packit 18d29c
_END_GOOGLE_NAMESPACE_
Packit 18d29c
Packit 18d29c
// The following APIs are all internal.
Packit 18d29c
#ifdef HAVE_STACKTRACE
Packit 18d29c
Packit 18d29c
#include "stacktrace.h"
Packit 18d29c
#include "symbolize.h"
Packit 18d29c
#include "base/commandlineflags.h"
Packit 18d29c
Packit 18d29c
GLOG_DEFINE_bool(symbolize_stacktrace, true,
Packit 18d29c
                 "Symbolize the stack trace in the tombstone");
Packit 18d29c
Packit 18d29c
_START_GOOGLE_NAMESPACE_
Packit 18d29c
Packit 18d29c
typedef void DebugWriter(const char*, void*);
Packit 18d29c
Packit 18d29c
// The %p field width for printf() functions is two characters per byte.
Packit 18d29c
// For some environments, add two extra bytes for the leading "0x".
Packit 18d29c
static const int kPrintfPointerFieldWidth = 2 + 2 * sizeof(void*);
Packit 18d29c
Packit 18d29c
static void DebugWriteToStderr(const char* data, void *) {
Packit 18d29c
  // This one is signal-safe.
Packit 18d29c
  if (write(STDERR_FILENO, data, strlen(data)) < 0) {
Packit 18d29c
    // Ignore errors.
Packit 18d29c
  }
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
void DebugWriteToString(const char* data, void *arg) {
Packit 18d29c
  reinterpret_cast<string*>(arg)->append(data);
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
#ifdef HAVE_SYMBOLIZE
Packit 18d29c
// Print a program counter and its symbol name.
Packit 18d29c
static void DumpPCAndSymbol(DebugWriter *writerfn, void *arg, void *pc,
Packit 18d29c
                            const char * const prefix) {
Packit 18d29c
  char tmp[1024];
Packit 18d29c
  const char *symbol = "(unknown)";
Packit 18d29c
  // Symbolizes the previous address of pc because pc may be in the
Packit 18d29c
  // next function.  The overrun happens when the function ends with
Packit 18d29c
  // a call to a function annotated noreturn (e.g. CHECK).
Packit 18d29c
  if (Symbolize(reinterpret_cast<char *>(pc) - 1, tmp, sizeof(tmp))) {
Packit 18d29c
      symbol = tmp;
Packit 18d29c
  }
Packit 18d29c
  char buf[1024];
Packit 18d29c
  snprintf(buf, sizeof(buf), "%s@ %*p  %s\n",
Packit 18d29c
           prefix, kPrintfPointerFieldWidth, pc, symbol);
Packit 18d29c
  writerfn(buf, arg);
Packit 18d29c
}
Packit 18d29c
#endif
Packit 18d29c
Packit 18d29c
static void DumpPC(DebugWriter *writerfn, void *arg, void *pc,
Packit 18d29c
                   const char * const prefix) {
Packit 18d29c
  char buf[100];
Packit 18d29c
  snprintf(buf, sizeof(buf), "%s@ %*p\n",
Packit 18d29c
           prefix, kPrintfPointerFieldWidth, pc);
Packit 18d29c
  writerfn(buf, arg);
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
// Dump current stack trace as directed by writerfn
Packit 18d29c
static void DumpStackTrace(int skip_count, DebugWriter *writerfn, void *arg) {
Packit 18d29c
  // Print stack trace
Packit 18d29c
  void* stack[32];
Packit 18d29c
  int depth = GetStackTrace(stack, ARRAYSIZE(stack), skip_count+1);
Packit 18d29c
  for (int i = 0; i < depth; i++) {
Packit 18d29c
#if defined(HAVE_SYMBOLIZE)
Packit 18d29c
    if (FLAGS_symbolize_stacktrace) {
Packit 18d29c
      DumpPCAndSymbol(writerfn, arg, stack[i], "    ");
Packit 18d29c
    } else {
Packit 18d29c
      DumpPC(writerfn, arg, stack[i], "    ");
Packit 18d29c
    }
Packit 18d29c
#else
Packit 18d29c
    DumpPC(writerfn, arg, stack[i], "    ");
Packit 18d29c
#endif
Packit 18d29c
  }
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
static void DumpStackTraceAndExit() {
Packit 18d29c
  DumpStackTrace(1, DebugWriteToStderr, NULL);
Packit 18d29c
Packit 18d29c
  // TOOD(hamaji): Use signal instead of sigaction?
Packit 18d29c
#ifdef HAVE_SIGACTION
Packit 18d29c
  if (IsFailureSignalHandlerInstalled()) {
Packit 18d29c
    // Set the default signal handler for SIGABRT, to avoid invoking our
Packit 18d29c
    // own signal handler installed by InstallFailureSignalHandler().
Packit 18d29c
    struct sigaction sig_action;
Packit 18d29c
    memset(&sig_action, 0, sizeof(sig_action));
Packit 18d29c
    sigemptyset(&sig_action.sa_mask);
Packit 18d29c
    sig_action.sa_handler = SIG_DFL;
Packit 18d29c
    sigaction(SIGABRT, &sig_action, NULL);
Packit 18d29c
  }
Packit 18d29c
#endif  // HAVE_SIGACTION
Packit 18d29c
Packit 18d29c
  abort();
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
_END_GOOGLE_NAMESPACE_
Packit 18d29c
Packit 18d29c
#endif  // HAVE_STACKTRACE
Packit 18d29c
Packit 18d29c
_START_GOOGLE_NAMESPACE_
Packit 18d29c
Packit 18d29c
namespace glog_internal_namespace_ {
Packit 18d29c
Packit 18d29c
const char* ProgramInvocationShortName() {
Packit 18d29c
  if (g_program_invocation_short_name != NULL) {
Packit 18d29c
    return g_program_invocation_short_name;
Packit 18d29c
  } else {
Packit 18d29c
    // TODO(hamaji): Use /proc/self/cmdline and so?
Packit 18d29c
    return "UNKNOWN";
Packit 18d29c
  }
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
bool IsGoogleLoggingInitialized() {
Packit 18d29c
  return g_program_invocation_short_name != NULL;
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
bool is_default_thread() {
Packit 18d29c
  if (g_program_invocation_short_name == NULL) {
Packit 18d29c
    // InitGoogleLogging() not yet called, so unlikely to be in a different
Packit 18d29c
    // thread
Packit 18d29c
    return true;
Packit 18d29c
  } else {
Packit 18d29c
    return pthread_equal(pthread_self(), g_main_thread_id);
Packit 18d29c
  }
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
#ifdef OS_WINDOWS
Packit 18d29c
struct timeval {
Packit 18d29c
  long tv_sec, tv_usec;
Packit 18d29c
};
Packit 18d29c
Packit 18d29c
// Based on: http://www.google.com/codesearch/p?hl=en#dR3YEbitojA/os_win32.c&q=GetSystemTimeAsFileTime%20license:bsd
Packit 18d29c
// See COPYING for copyright information.
Packit 18d29c
static int gettimeofday(struct timeval *tv, void* tz) {
Packit 18d29c
#define EPOCHFILETIME (116444736000000000ULL)
Packit 18d29c
  FILETIME ft;
Packit 18d29c
  LARGE_INTEGER li;
Packit 18d29c
  uint64 tt;
Packit 18d29c
Packit 18d29c
  GetSystemTimeAsFileTime(&ft;;
Packit 18d29c
  li.LowPart = ft.dwLowDateTime;
Packit 18d29c
  li.HighPart = ft.dwHighDateTime;
Packit 18d29c
  tt = (li.QuadPart - EPOCHFILETIME) / 10;
Packit 18d29c
  tv->tv_sec = tt / 1000000;
Packit 18d29c
  tv->tv_usec = tt % 1000000;
Packit 18d29c
Packit 18d29c
  return 0;
Packit 18d29c
}
Packit 18d29c
#endif
Packit 18d29c
Packit 18d29c
int64 CycleClock_Now() {
Packit 18d29c
  // TODO(hamaji): temporary impementation - it might be too slow.
Packit 18d29c
  struct timeval tv;
Packit 18d29c
  gettimeofday(&tv, NULL);
Packit 18d29c
  return static_cast<int64>(tv.tv_sec) * 1000000 + tv.tv_usec;
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
int64 UsecToCycles(int64 usec) {
Packit 18d29c
  return usec;
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
WallTime WallTime_Now() {
Packit 18d29c
  // Now, cycle clock is retuning microseconds since the epoch.
Packit 18d29c
  return CycleClock_Now() * 0.000001;
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
static int32 g_main_thread_pid = getpid();
Packit 18d29c
int32 GetMainThreadPid() {
Packit 18d29c
  return g_main_thread_pid;
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
bool PidHasChanged() {
Packit 18d29c
  int32 pid = getpid();
Packit 18d29c
  if (g_main_thread_pid == pid) {
Packit 18d29c
    return false;
Packit 18d29c
  }
Packit 18d29c
  g_main_thread_pid = pid;
Packit 18d29c
  return true;
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
pid_t GetTID() {
Packit 18d29c
  // On Linux and MacOSX, we try to use gettid().
Packit 18d29c
#if defined OS_LINUX || defined OS_MACOSX
Packit 18d29c
#ifndef __NR_gettid
Packit 18d29c
#ifdef OS_MACOSX
Packit 18d29c
#define __NR_gettid SYS_gettid
Packit 18d29c
#elif ! defined __i386__
Packit 18d29c
#error "Must define __NR_gettid for non-x86 platforms"
Packit 18d29c
#else
Packit 18d29c
#define __NR_gettid 224
Packit 18d29c
#endif
Packit 18d29c
#endif
Packit 18d29c
  static bool lacks_gettid = false;
Packit 18d29c
  if (!lacks_gettid) {
Packit 18d29c
    pid_t tid = syscall(__NR_gettid);
Packit 18d29c
    if (tid != -1) {
Packit 18d29c
      return tid;
Packit 18d29c
    }
Packit 18d29c
    // Technically, this variable has to be volatile, but there is a small
Packit 18d29c
    // performance penalty in accessing volatile variables and there should
Packit 18d29c
    // not be any serious adverse effect if a thread does not immediately see
Packit 18d29c
    // the value change to "true".
Packit 18d29c
    lacks_gettid = true;
Packit 18d29c
  }
Packit 18d29c
#endif  // OS_LINUX || OS_MACOSX
Packit 18d29c
Packit 18d29c
  // If gettid() could not be used, we use one of the following.
Packit 18d29c
#if defined OS_LINUX
Packit 18d29c
  return getpid();  // Linux:  getpid returns thread ID when gettid is absent
Packit 18d29c
#elif defined OS_WINDOWS || defined OS_CYGWIN
Packit 18d29c
  return GetCurrentThreadId();
Packit 18d29c
#else
Packit 18d29c
  // If none of the techniques above worked, we use pthread_self().
Packit 18d29c
  return (pid_t)(uintptr_t)pthread_self();
Packit 18d29c
#endif
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
const char* const_basename(const char* filepath) {
Packit 18d29c
  const char* base = strrchr(filepath, '/');
Packit 18d29c
#ifdef OS_WINDOWS  // Look for either path separator in Windows
Packit 18d29c
  if (!base)
Packit 18d29c
    base = strrchr(filepath, '\\');
Packit 18d29c
#endif
Packit 18d29c
  return base ? (base+1) : filepath;
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
static string g_my_user_name;
Packit 18d29c
const string& MyUserName() {
Packit 18d29c
  return g_my_user_name;
Packit 18d29c
}
Packit 18d29c
static void MyUserNameInitializer() {
Packit 18d29c
  // TODO(hamaji): Probably this is not portable.
Packit 18d29c
#if defined(OS_WINDOWS)
Packit 18d29c
  const char* user = getenv("USERNAME");
Packit 18d29c
#else
Packit 18d29c
  const char* user = getenv("USER");
Packit 18d29c
#endif
Packit 18d29c
  if (user != NULL) {
Packit 18d29c
    g_my_user_name = user;
Packit 18d29c
  } else {
Packit 18d29c
    g_my_user_name = "invalid-user";
Packit 18d29c
  }
Packit 18d29c
}
Packit 18d29c
REGISTER_MODULE_INITIALIZER(utilities, MyUserNameInitializer());
Packit 18d29c
Packit 18d29c
#ifdef HAVE_STACKTRACE
Packit 18d29c
void DumpStackTraceToString(string* stacktrace) {
Packit 18d29c
  DumpStackTrace(1, DebugWriteToString, stacktrace);
Packit 18d29c
}
Packit 18d29c
#endif
Packit 18d29c
Packit 18d29c
// We use an atomic operation to prevent problems with calling CrashReason
Packit 18d29c
// from inside the Mutex implementation (potentially through RAW_CHECK).
Packit 18d29c
static const CrashReason* g_reason = 0;
Packit 18d29c
Packit 18d29c
void SetCrashReason(const CrashReason* r) {
Packit 18d29c
  sync_val_compare_and_swap(&g_reason,
Packit 18d29c
                            reinterpret_cast<const CrashReason*>(0),
Packit 18d29c
                            r);
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
void InitGoogleLoggingUtilities(const char* argv0) {
Packit 18d29c
  CHECK(!IsGoogleLoggingInitialized())
Packit 18d29c
      << "You called InitGoogleLogging() twice!";
Packit 18d29c
  const char* slash = strrchr(argv0, '/');
Packit 18d29c
#ifdef OS_WINDOWS
Packit 18d29c
  if (!slash)  slash = strrchr(argv0, '\\');
Packit 18d29c
#endif
Packit 18d29c
  g_program_invocation_short_name = slash ? slash + 1 : argv0;
Packit 18d29c
  g_main_thread_id = pthread_self();
Packit 18d29c
Packit 18d29c
#ifdef HAVE_STACKTRACE
Packit 18d29c
  InstallFailureFunction(&DumpStackTraceAndExit);
Packit 18d29c
#endif
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
void ShutdownGoogleLoggingUtilities() {
Packit 18d29c
  CHECK(IsGoogleLoggingInitialized())
Packit 18d29c
      << "You called ShutdownGoogleLogging() without calling InitGoogleLogging() first!";
Packit 18d29c
  g_program_invocation_short_name = NULL;
Packit 18d29c
#ifdef HAVE_SYSLOG_H
Packit 18d29c
  closelog();
Packit 18d29c
#endif
Packit 18d29c
}
Packit 18d29c
Packit 18d29c
}  // namespace glog_internal_namespace_
Packit 18d29c
Packit 18d29c
_END_GOOGLE_NAMESPACE_
Packit 18d29c
Packit 18d29c
// Make an implementation of stacktrace compiled.
Packit 18d29c
#ifdef STACKTRACE_H
Packit 18d29c
# include STACKTRACE_H
Packit 18d29c
#endif