aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authornaruse <naruse@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2017-09-01 01:13:53 +0000
committernaruse <naruse@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2017-09-01 01:13:53 +0000
commit114ff9b3167b5db3227241cb8393510713d8b58c (patch)
tree761894b66b190a26b00c6d1ffab040908aee34ce
parentada1f6b2e72370e1e3dfcd2c30a2ea9dc99c78cd (diff)
downloadruby-114ff9b3167b5db3227241cb8393510713d8b58c.tar.gz
Fix C level backtrace on Darwin
SEGV caused by invalid instruction call. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@59710 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--test/ruby/test_vm_dump.rb20
-rw-r--r--vm_dump.c9
2 files changed, 27 insertions, 2 deletions
diff --git a/test/ruby/test_vm_dump.rb b/test/ruby/test_vm_dump.rb
new file mode 100644
index 0000000000..9ea725f233
--- /dev/null
+++ b/test/ruby/test_vm_dump.rb
@@ -0,0 +1,20 @@
+# frozen_string_literal: true
+require 'test/unit'
+
+class TestVMDump < Test::Unit::TestCase
+ def assert_vm_dump_works(args)
+ assert_in_out_err(args, "", [], [:*, /^.* main \+ \d+$/, :*, /^\[IMPORTANT\]/, :*])
+ end
+
+ def test_darwin_invalid_call
+ assert_vm_dump_works(['-rfiddle', '-eFiddle::Function.new(Fiddle::Pointer.new(1), [], Fiddle::TYPE_VOID).call'])
+ end
+
+ def test_darwin_segv_in_syscall
+ assert_vm_dump_works('-e1.times{Process.kill :SEGV,$$}')
+ end
+
+ def test_darwin_invalid_access
+ assert_vm_dump_works(['-rfiddle', '-eFiddle.dlunwrap(100).class'])
+ end
+end
diff --git a/vm_dump.c b/vm_dump.c
index 37028a292d..9c91c13bd5 100644
--- a/vm_dump.c
+++ b/vm_dump.c
@@ -413,6 +413,7 @@ rb_vmdebug_thread_dump_state(VALUE self)
# elif defined(__APPLE__) && defined(__x86_64__) && defined(HAVE_LIBUNWIND_H)
# define UNW_LOCAL_ONLY
# include <libunwind.h>
+# include <sys/mman.h>
# undef backtrace
int
backtrace(void **trace, int size)
@@ -439,6 +440,8 @@ darwin_sigtramp:
/* darwin's bundled libunwind doesn't support signal trampoline */
{
ucontext_t *uctx;
+ char vec[1];
+ int r;
/* get _sigtramp's ucontext_t and set values to cursor
* http://www.opensource.apple.com/source/Libc/Libc-825.25/i386/sys/_sigtramp.s
* http://www.opensource.apple.com/source/libunwind/libunwind-35.1/src/unw_getcontext.s
@@ -462,8 +465,10 @@ darwin_sigtramp:
unw_set_reg(&cursor, UNW_X86_64_R14, uctx->uc_mcontext->__ss.__r14);
unw_set_reg(&cursor, UNW_X86_64_R15, uctx->uc_mcontext->__ss.__r15);
ip = uctx->uc_mcontext->__ss.__rip;
- if (!ip || (((char*)ip)[-2] == 0x0f && ((char*)ip)[-1] == 5)) {
- /* NULL reference or signal received in syscall */
+ r = mincore((const void *)ip, 1, vec);
+ if (r || !vec[0] || memcmp((const char *)ip-2, "\x0f\x05", 2) == 0) {
+ /* if segv is caused by invalid call or signal received in syscall */
+ /* the frame is invalid; skip */
trace[n++] = (void *)ip;
ip = *(unw_word_t*)uctx->uc_mcontext->__ss.__rsp;
}