|  | /* Copyright (C) 1991-2016 Free Software Foundation, Inc. | 
|  | This file is part of the GNU C Library. | 
|  |  | 
|  | The GNU C Library is free software; you can redistribute it and/or | 
|  | modify it under the terms of the GNU Lesser General Public | 
|  | License as published by the Free Software Foundation; either | 
|  | version 2.1 of the License, or (at your option) any later version. | 
|  |  | 
|  | The GNU C Library is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | Lesser General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU Lesser General Public | 
|  | License along with the GNU C Library; if not, see | 
|  | <http://www.gnu.org/licenses/>.  */ | 
|  |  | 
|  | #include <libc-lock.h> | 
|  | #include <signal.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <unistd.h> | 
|  |  | 
|  | /* Try to get a machine dependent instruction which will make the | 
|  | program crash.  This is used in case everything else fails.  */ | 
|  | #include <abort-instr.h> | 
|  | #ifndef ABORT_INSTRUCTION | 
|  | /* No such instruction is available.  */ | 
|  | # define ABORT_INSTRUCTION | 
|  | #endif | 
|  |  | 
|  | #include <libio/libioP.h> | 
|  | #define fflush(s) _IO_flush_all_lockp (0) | 
|  |  | 
|  | /* Exported variable to locate abort message in core files etc.  */ | 
|  | struct abort_msg_s *__abort_msg __attribute__ ((nocommon)); | 
|  | libc_hidden_def (__abort_msg) | 
|  |  | 
|  | /* We must avoid to run in circles.  Therefore we remember how far we | 
|  | already got.  */ | 
|  | static int stage; | 
|  |  | 
|  | /* We should be prepared for multiple threads trying to run abort.  */ | 
|  | __libc_lock_define_initialized_recursive (static, lock); | 
|  |  | 
|  |  | 
|  | /* Cause an abnormal program termination with core-dump.  */ | 
|  | void | 
|  | abort (void) | 
|  | { | 
|  | struct sigaction act; | 
|  | sigset_t sigs; | 
|  |  | 
|  | /* First acquire the lock.  */ | 
|  | __libc_lock_lock_recursive (lock); | 
|  |  | 
|  | /* Now it's for sure we are alone.  But recursive calls are possible.  */ | 
|  |  | 
|  | /* Unlock SIGABRT.  */ | 
|  | if (stage == 0) | 
|  | { | 
|  | ++stage; | 
|  | if (__sigemptyset (&sigs) == 0 && | 
|  | __sigaddset (&sigs, SIGABRT) == 0) | 
|  | __sigprocmask (SIG_UNBLOCK, &sigs, (sigset_t *) NULL); | 
|  | } | 
|  |  | 
|  | /* Flush all streams.  We cannot close them now because the user | 
|  | might have registered a handler for SIGABRT.  */ | 
|  | if (stage == 1) | 
|  | { | 
|  | ++stage; | 
|  | fflush (NULL); | 
|  | } | 
|  |  | 
|  | /* Send signal which possibly calls a user handler.  */ | 
|  | if (stage == 2) | 
|  | { | 
|  | /* This stage is special: we must allow repeated calls of | 
|  | `abort' when a user defined handler for SIGABRT is installed. | 
|  | This is risky since the `raise' implementation might also | 
|  | fail but I don't see another possibility.  */ | 
|  | int save_stage = stage; | 
|  |  | 
|  | stage = 0; | 
|  | __libc_lock_unlock_recursive (lock); | 
|  |  | 
|  | raise (SIGABRT); | 
|  |  | 
|  | __libc_lock_lock_recursive (lock); | 
|  | stage = save_stage + 1; | 
|  | } | 
|  |  | 
|  | /* There was a handler installed.  Now remove it.  */ | 
|  | if (stage == 3) | 
|  | { | 
|  | ++stage; | 
|  | memset (&act, '\0', sizeof (struct sigaction)); | 
|  | act.sa_handler = SIG_DFL; | 
|  | __sigfillset (&act.sa_mask); | 
|  | act.sa_flags = 0; | 
|  | __sigaction (SIGABRT, &act, NULL); | 
|  | } | 
|  |  | 
|  | /* Now close the streams which also flushes the output the user | 
|  | defined handler might has produced.  */ | 
|  | if (stage == 4) | 
|  | { | 
|  | ++stage; | 
|  | __fcloseall (); | 
|  | } | 
|  |  | 
|  | /* Try again.  */ | 
|  | if (stage == 5) | 
|  | { | 
|  | ++stage; | 
|  | raise (SIGABRT); | 
|  | } | 
|  |  | 
|  | /* Now try to abort using the system specific command.  */ | 
|  | if (stage == 6) | 
|  | { | 
|  | ++stage; | 
|  | ABORT_INSTRUCTION; | 
|  | } | 
|  |  | 
|  | /* If we can't signal ourselves and the abort instruction failed, exit.  */ | 
|  | if (stage == 7) | 
|  | { | 
|  | ++stage; | 
|  | _exit (127); | 
|  | } | 
|  |  | 
|  | /* If even this fails try to use the provided instruction to crash | 
|  | or otherwise make sure we never return.  */ | 
|  | while (1) | 
|  | /* Try for ever and ever.  */ | 
|  | ABORT_INSTRUCTION; | 
|  | } | 
|  | libc_hidden_def (abort) |