duke@435: /* stefank@2314: * Copyright (c) 2003, 2010, Oracle and/or its affiliates. All rights reserved. duke@435: * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. duke@435: * duke@435: * This code is free software; you can redistribute it and/or modify it duke@435: * under the terms of the GNU General Public License version 2 only, as duke@435: * published by the Free Software Foundation. duke@435: * duke@435: * This code is distributed in the hope that it will be useful, but WITHOUT duke@435: * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or duke@435: * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License duke@435: * version 2 for more details (a copy is included in the LICENSE file that duke@435: * accompanied this code). duke@435: * duke@435: * You should have received a copy of the GNU General Public License version duke@435: * 2 along with this work; if not, write to the Free Software Foundation, duke@435: * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. duke@435: * trims@1907: * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA trims@1907: * or visit www.oracle.com if you need additional information or have any trims@1907: * questions. duke@435: * duke@435: */ duke@435: stefank@2314: #include "precompiled.hpp" stefank@2314: #include "runtime/arguments.hpp" stefank@2314: #include "runtime/os.hpp" stefank@2314: #include "runtime/thread.hpp" stefank@2314: #include "utilities/vmError.hpp" duke@435: duke@435: #include duke@435: #include duke@435: #include duke@435: #include duke@435: #include duke@435: duke@435: void VMError::show_message_box(char *buf, int buflen) { duke@435: bool yes; duke@435: do { duke@435: error_string(buf, buflen); duke@435: int len = (int)strlen(buf); duke@435: char *p = &buf[len]; duke@435: duke@435: jio_snprintf(p, buflen - len, duke@435: "\n\n" duke@435: "Do you want to debug the problem?\n\n" vlivanov@4154: "To debug, run 'gdb /proc/%d/exe %d'; then switch to thread " UINTX_FORMAT " (" INTPTR_FORMAT ")\n" duke@435: "Enter 'yes' to launch gdb automatically (PATH must include gdb)\n" duke@435: "Otherwise, press RETURN to abort...", duke@435: os::current_process_id(), os::current_process_id(), twisti@2350: os::current_thread_id(), os::current_thread_id()); duke@435: duke@435: yes = os::message_box("Unexpected Error", buf); duke@435: duke@435: if (yes) { duke@435: // yes, user asked VM to launch debugger duke@435: jio_snprintf(buf, buflen, "gdb /proc/%d/exe %d", duke@435: os::current_process_id(), os::current_process_id()); duke@435: duke@435: os::fork_and_exec(buf); duke@435: yes = false; duke@435: } duke@435: } while (yes); duke@435: } duke@435: duke@435: // Space for our "saved" signal flags and handlers duke@435: static int resettedSigflags[2]; duke@435: static address resettedSighandler[2]; duke@435: duke@435: static void save_signal(int idx, int sig) duke@435: { duke@435: struct sigaction sa; duke@435: sigaction(sig, NULL, &sa); duke@435: resettedSigflags[idx] = sa.sa_flags; duke@435: resettedSighandler[idx] = (sa.sa_flags & SA_SIGINFO) duke@435: ? CAST_FROM_FN_PTR(address, sa.sa_sigaction) duke@435: : CAST_FROM_FN_PTR(address, sa.sa_handler); duke@435: } duke@435: duke@435: int VMError::get_resetted_sigflags(int sig) { duke@435: if(SIGSEGV == sig) { duke@435: return resettedSigflags[0]; duke@435: } else if(SIGBUS == sig) { duke@435: return resettedSigflags[1]; duke@435: } duke@435: return -1; duke@435: } duke@435: duke@435: address VMError::get_resetted_sighandler(int sig) { duke@435: if(SIGSEGV == sig) { duke@435: return resettedSighandler[0]; duke@435: } else if(SIGBUS == sig) { duke@435: return resettedSighandler[1]; duke@435: } duke@435: return NULL; duke@435: } duke@435: duke@435: static void crash_handler(int sig, siginfo_t* info, void* ucVoid) { duke@435: // unmask current signal duke@435: sigset_t newset; duke@435: sigemptyset(&newset); duke@435: sigaddset(&newset, sig); duke@435: sigprocmask(SIG_UNBLOCK, &newset, NULL); duke@435: duke@435: VMError err(NULL, sig, NULL, info, ucVoid); duke@435: err.report_and_die(); duke@435: } duke@435: duke@435: void VMError::reset_signal_handlers() { duke@435: // Save sigflags for resetted signals duke@435: save_signal(0, SIGSEGV); duke@435: save_signal(1, SIGBUS); duke@435: os::signal(SIGSEGV, CAST_FROM_FN_PTR(void *, crash_handler)); duke@435: os::signal(SIGBUS, CAST_FROM_FN_PTR(void *, crash_handler)); duke@435: }