Blame src/utilities.h

Packit Service 9def5d
// Copyright (c) 2008, Google Inc.
Packit Service 9def5d
// All rights reserved.
Packit Service 9def5d
//
Packit Service 9def5d
// Redistribution and use in source and binary forms, with or without
Packit Service 9def5d
// modification, are permitted provided that the following conditions are
Packit Service 9def5d
// met:
Packit Service 9def5d
//
Packit Service 9def5d
//     * Redistributions of source code must retain the above copyright
Packit Service 9def5d
// notice, this list of conditions and the following disclaimer.
Packit Service 9def5d
//     * Redistributions in binary form must reproduce the above
Packit Service 9def5d
// copyright notice, this list of conditions and the following disclaimer
Packit Service 9def5d
// in the documentation and/or other materials provided with the
Packit Service 9def5d
// distribution.
Packit Service 9def5d
//     * Neither the name of Google Inc. nor the names of its
Packit Service 9def5d
// contributors may be used to endorse or promote products derived from
Packit Service 9def5d
// this software without specific prior written permission.
Packit Service 9def5d
//
Packit Service 9def5d
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit Service 9def5d
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit Service 9def5d
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit Service 9def5d
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
Packit Service 9def5d
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
Packit Service 9def5d
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
Packit Service 9def5d
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit Service 9def5d
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit Service 9def5d
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit Service 9def5d
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
Packit Service 9def5d
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit Service 9def5d
//
Packit Service 9def5d
// Author: Shinichiro Hamaji
Packit Service 9def5d
//
Packit Service 9def5d
// Define utilties for glog internal usage.
Packit Service 9def5d
Packit Service 9def5d
#ifndef UTILITIES_H__
Packit Service 9def5d
#define UTILITIES_H__
Packit Service 9def5d
Packit Service 9def5d
#if defined(WIN32) || defined(_WIN32) || defined(__WIN32__)
Packit Service 9def5d
# define OS_WINDOWS
Packit Service 9def5d
#elif defined(__CYGWIN__) || defined(__CYGWIN32__)
Packit Service 9def5d
# define OS_CYGWIN
Packit Service 9def5d
#elif defined(linux) || defined(__linux) || defined(__linux__)
Packit Service 9def5d
# define OS_LINUX
Packit Service 9def5d
#elif defined(macintosh) || defined(__APPLE__) || defined(__APPLE_CC__)
Packit Service 9def5d
# define OS_MACOSX
Packit Service 9def5d
#elif defined(__FreeBSD__)
Packit Service 9def5d
# define OS_FREEBSD
Packit Service 9def5d
#elif defined(__NetBSD__)
Packit Service 9def5d
# define OS_NETBSD
Packit Service 9def5d
#elif defined(__OpenBSD__)
Packit Service 9def5d
# define OS_OPENBSD
Packit Service 9def5d
#else
Packit Service 9def5d
// TODO(hamaji): Add other platforms.
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
// printf macros for size_t, in the style of inttypes.h
Packit Service 9def5d
#ifdef _LP64
Packit Service 9def5d
#define __PRIS_PREFIX "z"
Packit Service 9def5d
#else
Packit Service 9def5d
#define __PRIS_PREFIX
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
// Use these macros after a % in a printf format string
Packit Service 9def5d
// to get correct 32/64 bit behavior, like this:
Packit Service 9def5d
// size_t size = records.size();
Packit Service 9def5d
// printf("%"PRIuS"\n", size);
Packit Service 9def5d
Packit Service 9def5d
#define PRIdS __PRIS_PREFIX "d"
Packit Service 9def5d
#define PRIxS __PRIS_PREFIX "x"
Packit Service 9def5d
#define PRIuS __PRIS_PREFIX "u"
Packit Service 9def5d
#define PRIXS __PRIS_PREFIX "X"
Packit Service 9def5d
#define PRIoS __PRIS_PREFIX "o"
Packit Service 9def5d
Packit Service 9def5d
#include "base/mutex.h"  // This must go first so we get _XOPEN_SOURCE
Packit Service 9def5d
Packit Service 9def5d
#include <string>
Packit Service 9def5d
Packit Service 9def5d
#if defined(OS_WINDOWS)
Packit Service 9def5d
# include "port.h"
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
#include "config.h"
Packit Service 9def5d
#include "glog/logging.h"
Packit Service 9def5d
Packit Service 9def5d
// There are three different ways we can try to get the stack trace:
Packit Service 9def5d
//
Packit Service 9def5d
// 1) The libunwind library.  This is still in development, and as a
Packit Service 9def5d
//    separate library adds a new dependency, but doesn't need a frame
Packit Service 9def5d
//    pointer.  It also doesn't call malloc.
Packit Service 9def5d
//
Packit Service 9def5d
// 2) Our hand-coded stack-unwinder.  This depends on a certain stack
Packit Service 9def5d
//    layout, which is used by gcc (and those systems using a
Packit Service 9def5d
//    gcc-compatible ABI) on x86 systems, at least since gcc 2.95.
Packit Service 9def5d
//    It uses the frame pointer to do its work.
Packit Service 9def5d
//
Packit Service 9def5d
// 3) The gdb unwinder -- also the one used by the c++ exception code.
Packit Service 9def5d
//    It's obviously well-tested, but has a fatal flaw: it can call
Packit Service 9def5d
//    malloc() from the unwinder.  This is a problem because we're
Packit Service 9def5d
//    trying to use the unwinder to instrument malloc().
Packit Service 9def5d
//
Packit Service 9def5d
// Note: if you add a new implementation here, make sure it works
Packit Service 9def5d
// correctly when GetStackTrace() is called with max_depth == 0.
Packit Service 9def5d
// Some code may do that.
Packit Service 9def5d
Packit Service 9def5d
#if defined(HAVE_LIB_UNWIND)
Packit Service 9def5d
# define STACKTRACE_H "stacktrace_libunwind-inl.h"
Packit Service 9def5d
#elif !defined(NO_FRAME_POINTER)
Packit Service 9def5d
# if defined(__i386__) && __GNUC__ >= 2
Packit Service 9def5d
#  define STACKTRACE_H "stacktrace_x86-inl.h"
Packit Service 9def5d
# elif defined(__x86_64__) && __GNUC__ >= 2 && HAVE_UNWIND_H
Packit Service 9def5d
#  define STACKTRACE_H "stacktrace_x86_64-inl.h"
Packit Service 9def5d
# elif (defined(__ppc__) || defined(__PPC__)) && __GNUC__ >= 2
Packit Service 9def5d
#  define STACKTRACE_H "stacktrace_powerpc-inl.h"
Packit Service 9def5d
# endif
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
#if !defined(STACKTRACE_H) && defined(HAVE_EXECINFO_H)
Packit Service 9def5d
# define STACKTRACE_H "stacktrace_generic-inl.h"
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
#if defined(STACKTRACE_H)
Packit Service 9def5d
# define HAVE_STACKTRACE
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
// defined by gcc
Packit Service 9def5d
#if defined(__ELF__) && defined(OS_LINUX)
Packit Service 9def5d
# define HAVE_SYMBOLIZE
Packit Service 9def5d
#elif defined(OS_MACOSX) && defined(HAVE_DLADDR)
Packit Service 9def5d
// Use dladdr to symbolize.
Packit Service 9def5d
# define HAVE_SYMBOLIZE
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
#ifndef ARRAYSIZE
Packit Service 9def5d
// There is a better way, but this is good enough for our purpose.
Packit Service 9def5d
# define ARRAYSIZE(a) (sizeof(a) / sizeof(*(a)))
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
_START_GOOGLE_NAMESPACE_
Packit Service 9def5d
Packit Service 9def5d
namespace glog_internal_namespace_ {
Packit Service 9def5d
Packit Service 9def5d
#ifdef HAVE___ATTRIBUTE__
Packit Service 9def5d
# define ATTRIBUTE_NOINLINE __attribute__ ((noinline))
Packit Service 9def5d
# define HAVE_ATTRIBUTE_NOINLINE
Packit Service 9def5d
#else
Packit Service 9def5d
# define ATTRIBUTE_NOINLINE
Packit Service 9def5d
#endif
Packit Service 9def5d
Packit Service 9def5d
const char* ProgramInvocationShortName();
Packit Service 9def5d
Packit Service 9def5d
bool IsGoogleLoggingInitialized();
Packit Service 9def5d
Packit Service 9def5d
bool is_default_thread();
Packit Service 9def5d
Packit Service 9def5d
int64 CycleClock_Now();
Packit Service 9def5d
Packit Service 9def5d
int64 UsecToCycles(int64 usec);
Packit Service 9def5d
Packit Service 9def5d
typedef double WallTime;
Packit Service 9def5d
WallTime WallTime_Now();
Packit Service 9def5d
Packit Service 9def5d
int32 GetMainThreadPid();
Packit Service 9def5d
bool PidHasChanged();
Packit Service 9def5d
Packit Service 9def5d
pid_t GetTID();
Packit Service 9def5d
Packit Service 9def5d
const std::string& MyUserName();
Packit Service 9def5d
Packit Service 9def5d
// Get the part of filepath after the last path separator.
Packit Service 9def5d
// (Doesn't modify filepath, contrary to basename() in libgen.h.)
Packit Service 9def5d
const char* const_basename(const char* filepath);
Packit Service 9def5d
Packit Service 9def5d
// Wrapper of __sync_val_compare_and_swap. If the GCC extension isn't
Packit Service 9def5d
// defined, we try the CPU specific logics (we only support x86 and
Packit Service 9def5d
// x86_64 for now) first, then use a naive implementation, which has a
Packit Service 9def5d
// race condition.
Packit Service 9def5d
template<typename T>
Packit Service 9def5d
inline T sync_val_compare_and_swap(T* ptr, T oldval, T newval) {
Packit Service 9def5d
#if defined(HAVE___SYNC_VAL_COMPARE_AND_SWAP)
Packit Service 9def5d
  return __sync_val_compare_and_swap(ptr, oldval, newval);
Packit Service 9def5d
#elif defined(__GNUC__) && (defined(__i386__) || defined(__x86_64__))
Packit Service 9def5d
  T ret;
Packit Service 9def5d
  __asm__ __volatile__("lock; cmpxchg %1, (%2);"
Packit Service 9def5d
                       :"=a"(ret)
Packit Service 9def5d
                        // GCC may produces %sil or %dil for
Packit Service 9def5d
                        // constraint "r", but some of apple's gas
Packit Service 9def5d
                        // dosn't know the 8 bit registers.
Packit Service 9def5d
                        // We use "q" to avoid these registers.
Packit Service 9def5d
                       :"q"(newval), "q"(ptr), "a"(oldval)
Packit Service 9def5d
                       :"memory", "cc");
Packit Service 9def5d
  return ret;
Packit Service 9def5d
#else
Packit Service 9def5d
  T ret = *ptr;
Packit Service 9def5d
  if (ret == oldval) {
Packit Service 9def5d
    *ptr = newval;
Packit Service 9def5d
  }
Packit Service 9def5d
  return ret;
Packit Service 9def5d
#endif
Packit Service 9def5d
}
Packit Service 9def5d
Packit Service 9def5d
void DumpStackTraceToString(std::string* stacktrace);
Packit Service 9def5d
Packit Service 9def5d
struct CrashReason {
Packit Service 9def5d
  CrashReason() : filename(0), line_number(0), message(0), depth(0) {}
Packit Service 9def5d
Packit Service 9def5d
  const char* filename;
Packit Service 9def5d
  int line_number;
Packit Service 9def5d
  const char* message;
Packit Service 9def5d
Packit Service 9def5d
  // We'll also store a bit of stack trace context at the time of crash as
Packit Service 9def5d
  // it may not be available later on.
Packit Service 9def5d
  void* stack[32];
Packit Service 9def5d
  int depth;
Packit Service 9def5d
};
Packit Service 9def5d
Packit Service 9def5d
void SetCrashReason(const CrashReason* r);
Packit Service 9def5d
Packit Service 9def5d
void InitGoogleLoggingUtilities(const char* argv0);
Packit Service 9def5d
void ShutdownGoogleLoggingUtilities();
Packit Service 9def5d
Packit Service 9def5d
}  // namespace glog_internal_namespace_
Packit Service 9def5d
Packit Service 9def5d
_END_GOOGLE_NAMESPACE_
Packit Service 9def5d
Packit Service 9def5d
using namespace GOOGLE_NAMESPACE::glog_internal_namespace_;
Packit Service 9def5d
Packit Service 9def5d
#endif  // UTILITIES_H__