/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright (c) 1988 AT&T * All Rights Reserved * * Copyright 2009 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ /* * Utility functions */ #include <unistd.h> #include <signal.h> #include <locale.h> #include <string.h> #include "msg.h" #include "_libld.h" /* * Exit after cleaning up. */ int ld_exit(Ofl_desc *ofl) { /* * If we have created an output file remove it. */ if ((ofl->ofl_fd > 0) && ((ofl->ofl_flags1 & FLG_OF1_NONREG) == 0)) (void) unlink(ofl->ofl_name); /* * Inform any support library that the link-edit has failed. */ ld_sup_atexit(ofl, 1); /* * Wrap up debug output file if one is open */ dbg_cleanup(); return (1); } /* * Establish the signals we're interested in, and the handlers that need to be * reinstalled should any of these signals occur. */ typedef struct { int signo; void (* defhdl)(); } Signals; static Signals signals[] = { { SIGHUP, SIG_DFL }, { SIGINT, SIG_IGN }, { SIGQUIT, SIG_DFL }, { SIGBUS, SIG_DFL }, { SIGTERM, SIG_IGN }, { 0, 0 } }; static Ofl_desc *Ofl = 0; /* * Define our signal handler. */ static void /* ARGSUSED2 */ handler(int sig, siginfo_t *sip, void *utp) { struct sigaction nact; Signals * sigs; /* * Reset all ignore handlers regardless of how we got here. */ nact.sa_handler = SIG_IGN; nact.sa_flags = 0; (void) sigemptyset(&nact.sa_mask); for (sigs = signals; sigs->signo; sigs++) { if (sigs->defhdl == SIG_IGN) (void) sigaction(sigs->signo, &nact, NULL); } /* * The model for creating an output file is to ftruncate() it to the * required size and mmap() a mapping into which the new contents are * written. Neither of these operations guarantee that the required * disk blocks exist, and should we run out of disk space a bus error * is generated. * Other situations have been reported to result in ld catching a bus * error (one instance was a stale NFS handle from an unstable server). * Thus we catch all bus errors and hope we can decode a better error. */ if ((sig == SIGBUS) && sip && Ofl->ofl_name) { eprintf(Ofl->ofl_lml, ERR_FATAL, MSG_INTL(MSG_FIL_INTERRUPT), Ofl->ofl_name, strerror(sip->si_errno)); } /* * This assert(0) causes DEBUG enabled linkers to produce a core file. */ if ((sig != SIGHUP) && (sig != SIGINT)) assert(0); exit(ld_exit(Ofl)); } /* * Establish a signal handler for all signals we're interested in. */ void ld_init_sighandler(Ofl_desc *ofl) { struct sigaction nact, oact; Signals * sigs; Ofl = ofl; /* * Our heavy use of mmap() means that we are susceptible to * receiving a SIGBUS in low diskspace situations. The main * purpose of the signal handler is to handle that situation * gracefully, so that out of disk errors don't drop a core file. * * In rare cases, this will prevent us from getting a core from a * SIGBUS triggered by an internal alignment error in libld. * If -znosighandler is set, return without registering the * handler. This is primarily of use for debugging problems in * the field, and is not of general interest. */ if (ofl->ofl_flags1 & FLG_OF1_NOSGHND) return; /* * For each signal we're interested in set up a signal handler that * insures we clean up any output file we're in the middle of creating. */ nact.sa_sigaction = handler; (void) sigemptyset(&nact.sa_mask); for (sigs = signals; sigs->signo; sigs++) { if ((sigaction(sigs->signo, NULL, &oact) == 0) && (oact.sa_handler != SIG_IGN)) { nact.sa_flags = SA_SIGINFO; if (sigs->defhdl == SIG_DFL) nact.sa_flags |= (SA_RESETHAND | SA_NODEFER); (void) sigaction(sigs->signo, &nact, NULL); } } }