From c5b8a2c9ecf16d95c69f05851751d56b8a36012d Mon Sep 17 00:00:00 2001 From: "erik.corry@gmail.com" Date: Wed, 19 May 2010 07:36:25 +0000 Subject: [PATCH] Fixes to Solaris build. signbit and StackWalker. This is a commit for ry (coldredlemur@gmail.com). See http://codereview.chromium.org/2092007 git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@4677 ce2b1a6d-e550-0410-aec6-3dcde31c8c00 --- src/platform-solaris.cc | 83 +++++++++++++++++++++++++++++++++++++------------ src/platform.h | 8 +++++ 2 files changed, 71 insertions(+), 20 deletions(-) diff --git a/src/platform-solaris.cc b/src/platform-solaris.cc index 0d9547b..0ae1ecf 100644 --- a/src/platform-solaris.cc +++ b/src/platform-solaris.cc @@ -35,7 +35,8 @@ #include // for stack alignment #include // getpagesize(), usleep() #include // mmap() -#include // backtrace(), backtrace_symbols() +#include // walkstack(), getcontext() +#include // dladdr #include #include // for sched_yield #include @@ -53,6 +54,24 @@ #include "platform.h" +// It seems there is a bug in some Solaris distributions (experienced in +// SunOS 5.10 Generic_141445-09) which make it difficult or impossible to +// access signbit() despite the availability of other C99 math functions. +#ifndef signbit +// Test sign - usually defined in math.h +int signbit(double x) { + // We need to take care of the special case of both positive and negative + // versions of zero. + if (x == 0) { + return fpclass(x) & FP_NZERO; + } else { + // This won't detect negative NaN but that should be okay since we don't + // assume that behavior. + return x < 0; + } +} +#endif // signbit + namespace v8 { namespace internal { @@ -231,31 +250,55 @@ void OS::LogSharedLibraryAddresses() { } -int OS::StackWalk(Vector frames) { - int frames_size = frames.length(); - ScopedVector addresses(frames_size); +struct StackWalker { + Vector& frames; + int index; +}; - int frames_count = backtrace(addresses.start(), frames_size); - char** symbols = backtrace_symbols(addresses.start(), frames_count); - if (symbols == NULL) { - return kStackWalkError; - } +static int StackWalkCallback(uintptr_t pc, int signo, void* data) { + struct StackWalker* walker = static_cast(data); + Dl_info info; + + int i = walker->index; + + walker->frames[i].address = reinterpret_cast(pc); - for (int i = 0; i < frames_count; i++) { - frames[i].address = addresses[i]; - // Format a text representation of the frame based on the information - // available. - SNPrintF(MutableCStrVector(frames[i].text, kStackWalkMaxTextLen), - "%s", - symbols[i]); - // Make sure line termination is in place. - frames[i].text[kStackWalkMaxTextLen - 1] = '\0'; + // Make sure line termination is in place. + walker->frames[i].text[OS::kStackWalkMaxTextLen - 1] = '\0'; + + Vector text = MutableCStrVector(walker->frames[i].text, + OS::kStackWalkMaxTextLen); + + if (dladdr(reinterpret_cast(pc), &info) == 0) { + OS::SNPrintF(text, "[0x%p]", pc); + } else if ((info.dli_fname != NULL && info.dli_sname != NULL)) { + // We have symbol info. + OS::SNPrintF(text, "%s'%s+0x%x", info.dli_fname, info.dli_sname, pc); + } else { + // No local symbol info. + OS::SNPrintF(text, + "%s'0x%p [0x%p]", + info.dli_fname, + pc - reinterpret_cast(info.dli_fbase), + pc); } + walker->index++; + return 0; +} - free(symbols); - return frames_count; +int OS::StackWalk(Vector frames) { + ucontext_t ctx; + struct StackWalker walker = { frames, 0 }; + + if (getcontext(&ctx) < 0) return kStackWalkError; + + if (!walkcontext(&ctx, StackWalkCallback, &walker)) { + return kStackWalkError; + } + + return walker.index; } diff --git a/src/platform.h b/src/platform.h index 7156441..606e5b4 100644 --- a/src/platform.h +++ b/src/platform.h @@ -83,6 +83,14 @@ int random(); #endif // WIN32 + +#ifdef __sun +# ifndef signbit +int signbit(double x); +# endif +#endif + + // GCC specific stuff #ifdef __GNUC__ -- 2.7.4