diff options
author | nobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2014-05-18 07:38:54 +0000 |
---|---|---|
committer | nobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2014-05-18 07:38:54 +0000 |
commit | f4f71d6e6bfc7a678626734eef13364b1d0d78e5 (patch) | |
tree | 1eecfef175066949b4bc0b2b61705890d08e5fe2 /signal.c | |
parent | 3622efd2bd3fba8b927b3b76ba37f1e33710da22 (diff) |
signal.c: check stack overflow by SP
* signal.c (check_stack_overflow): raise SystemStackError if SP
register and fault address is in the same page or the next, on
x86 Mac OS X. [EXPERIMENTAL]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@45995 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'signal.c')
-rw-r--r-- | signal.c | 22 |
1 files changed, 17 insertions, 5 deletions
@@ -700,17 +700,29 @@ rb_get_next_signal(void) #if defined(USE_SIGALTSTACK) || defined(_WIN32) NORETURN(void ruby_thread_stack_overflow(rb_thread_t *th)); -#if defined(HAVE_UCONTEXT_H) && defined __linux__ && (defined __i386__ || defined __x86_64__) +#if !(defined(HAVE_UCONTEXT_H) && (defined __i386__ || defined __x86_64__)) +#elif defined __linux__ +# define USE_UCONTEXT_REG 1 +#elif defined __APPLE__ # define USE_UCONTEXT_REG 1 #endif #ifdef USE_UCONTEXT_REG static void check_stack_overflow(const uintptr_t addr, const ucontext_t *ctx) { -# if defined REG_RSP - const greg_t sp = ctx->uc_mcontext.gregs[REG_RSP]; -# else - const greg_t sp = ctx->uc_mcontext.gregs[REG_ESP]; + const struct mcontext *mctx = ctx->uc_mcontext; +# if defined __linux__ +# if defined REG_RSP + const greg_t sp = mctx->gregs[REG_RSP]; +# else + const greg_t sp = mctx->gregs[REG_ESP]; +# endif +# elif defined __APPLE__ +# if defined(__LP64__) + const uintptr_t sp = mctx->ss.rsp; +# else + const uintptr_t sp = mctx->ss.esp; +# endif # endif enum {pagesize = 4096}; const uintptr_t sp_page = (uintptr_t)sp / pagesize; |