diff options
author | Ryan Dahl <ry@tinyclouds.org> | 2010-05-14 23:36:17 +0000 |
---|---|---|
committer | Ryan Dahl <ry@tinyclouds.org> | 2010-05-14 16:38:17 -0700 |
commit | ac3a2d899a12b7eabf01eb72b4c1de68b6a34e2b (patch) | |
tree | 22b3c7b53b069541088c2340e5d60f0c5811c7f7 /deps/v8 | |
parent | a9b962a653162c0ff025698367aead308cb18b8e (diff) | |
download | node-ac3a2d899a12b7eabf01eb72b4c1de68b6a34e2b.tar.gz |
Changes to work on Solaris 10
Diffstat (limited to 'deps/v8')
-rw-r--r-- | deps/v8/src/platform-solaris.cc | 78 | ||||
-rw-r--r-- | deps/v8/src/platform.h | 7 |
2 files changed, 65 insertions, 20 deletions
diff --git a/deps/v8/src/platform-solaris.cc b/deps/v8/src/platform-solaris.cc index 0d9547b8b..16617cec1 100644 --- a/deps/v8/src/platform-solaris.cc +++ b/deps/v8/src/platform-solaris.cc @@ -35,7 +35,8 @@ #include <sys/stack.h> // for stack alignment #include <unistd.h> // getpagesize(), usleep() #include <sys/mman.h> // mmap() -#include <execinfo.h> // backtrace(), backtrace_symbols() +#include <ucontext.h> // walkstack(), getcontext() +#include <dlfcn.h> // dladdr1 #include <pthread.h> #include <sched.h> // for sched_yield #include <semaphore.h> @@ -53,6 +54,19 @@ #include "platform.h" +#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 { + return x < 0; + } +} +#endif // signbit + namespace v8 { namespace internal { @@ -231,31 +245,55 @@ void OS::LogSharedLibraryAddresses() { } -int OS::StackWalk(Vector<OS::StackFrame> frames) { - int frames_size = frames.length(); - ScopedVector<void*> addresses(frames_size); +struct stack_walker { + Vector<OS::StackFrame> &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 stack_walker * walker = static_cast<struct stack_walker *>(data); + Dl_info info; + + int i = walker->index; + + walker->frames[i].address = (void*)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<char> text = MutableCStrVector(walker->frames[i].text, + OS::kStackWalkMaxTextLen); + + if (dladdr((void*)pc, &info) == 0) { + OS::SNPrintF(text, "[0x%p]", pc); + } else if ((info.dli_fname != NULL && info.dli_sname != NULL)) { + // we have containing 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, + (unsigned long)pc - (unsigned long)info.dli_fbase, + pc); } + walker->index++; + return 0; +} - free(symbols); - return frames_count; +int OS::StackWalk(Vector<OS::StackFrame> frames) { + ucontext_t ctx; + struct stack_walker walker = { frames, 0 }; + + if (getcontext(&ctx) < 0) return kStackWalkError; + + if (!walkcontext(&ctx, StackWalkCallback, (void*)(&walker))) { + return kStackWalkError; + } + + return walker.index; } diff --git a/deps/v8/src/platform.h b/deps/v8/src/platform.h index 7156441b3..f89b34241 100644 --- a/deps/v8/src/platform.h +++ b/deps/v8/src/platform.h @@ -83,6 +83,13 @@ int random(); #endif // WIN32 +#ifdef __sun +# if !defined(signbit) +int signbit(double x); +# endif +#endif + + // GCC specific stuff #ifdef __GNUC__ |