ef1abbbea0
Signed-off-by: Xiang Xiao <xiaoxiang@xiaomi.com>
581 lines
17 KiB
C
581 lines
17 KiB
C
/****************************************************************************
|
|
* sched/signal/sig_default.c
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright ownership. The
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance with the
|
|
* License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <sys/wait.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <pthread.h>
|
|
#include <signal.h>
|
|
#include <assert.h>
|
|
|
|
#include <nuttx/sched.h>
|
|
#include <nuttx/spinlock.h>
|
|
#include <nuttx/signal.h>
|
|
|
|
#include "group/group.h"
|
|
#include "sched/sched.h"
|
|
#include "task/task.h"
|
|
#include "signal/signal.h"
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
#if defined(CONFIG_SIG_SIGUSR1_ACTION) || defined(CONFIG_SIG_SIGUSR2_ACTION) || \
|
|
defined(CONFIG_SIG_SIGALRM_ACTION) || defined(CONFIG_SIG_SIGPOLL_ACTION) || \
|
|
defined(CONFIG_SIG_SIGKILL_ACTION) || defined(CONFIG_SIG_SIGPIPE_ACTION)
|
|
# define HAVE_NXSIG_ABNORMAL_TERMINANTION 1
|
|
#endif
|
|
|
|
/* Bit definitions for the struct nxsig_defaction_s 'flags' field */
|
|
|
|
#define SIG_FLAG_NOCATCH (1 << 0) /* Signal cannot be caught or ignored */
|
|
|
|
/****************************************************************************
|
|
* Private Types
|
|
****************************************************************************/
|
|
|
|
/* This type provides the default action associated with a signal */
|
|
|
|
struct nxsig_defaction_s
|
|
{
|
|
uint8_t signo; /* Signal number. Range 1..MAX_SIGNO */
|
|
uint8_t flags; /* See SIG_FLAG_ definitions */
|
|
_sa_handler_t action; /* Default signal action */
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Private Function Prototypes
|
|
****************************************************************************/
|
|
|
|
/* Default actions */
|
|
|
|
#ifdef HAVE_NXSIG_ABNORMAL_TERMINANTION
|
|
static void nxsig_abnormal_termination(int signo);
|
|
#endif
|
|
#ifdef CONFIG_SIG_SIGSTOP_ACTION
|
|
static void nxsig_null_action(int signo);
|
|
static void nxsig_stop_task(int signo);
|
|
#endif
|
|
|
|
/* Helpers */
|
|
|
|
static _sa_handler_t nxsig_default_action(int signo);
|
|
static void nxsig_setup_default_action(FAR struct task_group_s *group,
|
|
FAR const struct nxsig_defaction_s *
|
|
info);
|
|
|
|
/****************************************************************************
|
|
* Private Data
|
|
****************************************************************************/
|
|
|
|
/* NOTE: Default actions are not currently supported for very many signals.
|
|
* However, this array is set up to support an indefinite number of default
|
|
* signal actions.
|
|
*/
|
|
|
|
/* This table is used to make the default action for a signal to the correct
|
|
* signal handler. Signals whose default action is be ignored do not need
|
|
* to be in this table (e.g., SIGCHLD).
|
|
*/
|
|
|
|
static const struct nxsig_defaction_s g_defactions[] =
|
|
{
|
|
#ifdef CONFIG_SIG_SIGUSR1_ACTION
|
|
{ SIGUSR1, 0, nxsig_abnormal_termination },
|
|
#endif
|
|
#ifdef CONFIG_SIG_SIGUSR2_ACTION
|
|
{ SIGUSR2, 0, nxsig_abnormal_termination },
|
|
#endif
|
|
#ifdef CONFIG_SIG_SIGALRM_ACTION
|
|
{ SIGALRM, 0, nxsig_abnormal_termination },
|
|
#endif
|
|
#ifdef CONFIG_SIG_SIGPOLL_ACTION
|
|
{ SIGPOLL, 0, nxsig_abnormal_termination },
|
|
#endif
|
|
#ifdef CONFIG_SIG_SIGSTOP_ACTION
|
|
{ SIGSTOP, SIG_FLAG_NOCATCH, nxsig_stop_task },
|
|
{ SIGTSTP, 0, nxsig_stop_task },
|
|
{ SIGCONT, SIG_FLAG_NOCATCH, nxsig_null_action },
|
|
#endif
|
|
#ifdef CONFIG_SIG_SIGKILL_ACTION
|
|
{ SIGINT, 0, nxsig_abnormal_termination },
|
|
{ SIGKILL, SIG_FLAG_NOCATCH, nxsig_abnormal_termination },
|
|
{ SIGQUIT, 0, nxsig_abnormal_termination },
|
|
{ SIGTERM, 0, nxsig_abnormal_termination },
|
|
#endif
|
|
#ifdef CONFIG_SIG_SIGPIPE_ACTION
|
|
{ SIGPIPE, 0, nxsig_abnormal_termination }
|
|
#endif
|
|
};
|
|
|
|
#define NACTIONS (sizeof(g_defactions) / sizeof(struct nxsig_defaction_s))
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
/* TODO: Take into account pthread-specific signal behaviors */
|
|
|
|
/* Default Actions:
|
|
*
|
|
* - Abnormal termination of the process. The process is terminated with
|
|
* all the consequences of _exit() except that the status made available
|
|
* to wait() and waitpid() indicates abnormal termination by the
|
|
* specified signal.
|
|
* - Abnormal termination of the process. Additionally with the XSI
|
|
* extension, implementation-defined abnormal termination actions, such
|
|
* as creation of a core file, may occur.
|
|
* - Ignore the signal.
|
|
* - Stop the process.
|
|
* - Continue the process, if it is stopped; otherwise, ignore the signal.
|
|
*/
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_null_action
|
|
*
|
|
* Description:
|
|
* The do-nothing default signal action handler.
|
|
*
|
|
* Input Parameters:
|
|
* Standard signal handler parameters
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifdef CONFIG_SIG_SIGSTOP_ACTION
|
|
static void nxsig_null_action(int signo)
|
|
{
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_abnormal_termination
|
|
*
|
|
* Description:
|
|
* This is the handler for the abnormal termination default action.
|
|
*
|
|
* Input Parameters:
|
|
* Standard signal handler parameters
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifdef HAVE_NXSIG_ABNORMAL_TERMINANTION
|
|
static void nxsig_abnormal_termination(int signo)
|
|
{
|
|
FAR struct tcb_s *rtcb = (FAR struct tcb_s *)this_task();
|
|
|
|
/* Careful: In the multi-threaded task, the signal may be handled on a
|
|
* child pthread.
|
|
*/
|
|
|
|
#ifdef HAVE_GROUP_MEMBERS
|
|
/* Kill all of the children of the task. This will not kill the currently
|
|
* running task/pthread (this_task). It will kill the main thread of the
|
|
* task group if this_task is a pthread.
|
|
*/
|
|
|
|
group_kill_children(rtcb);
|
|
#endif
|
|
|
|
#ifndef CONFIG_DISABLE_PTHREAD
|
|
/* Check if the currently running task is actually a pthread */
|
|
|
|
if ((rtcb->flags & TCB_FLAG_TTYPE_MASK) == TCB_FLAG_TTYPE_PTHREAD)
|
|
{
|
|
/* Exit the final thread of the task group.
|
|
*
|
|
* REVISIT: This will not work if HAVE_GROUP_MEMBERS is not set.
|
|
*/
|
|
|
|
rtcb->flags &= ~TCB_FLAG_CANCEL_PENDING;
|
|
rtcb->flags |= TCB_FLAG_CANCEL_DOING;
|
|
#if !defined(CONFIG_BUILD_FLAT) && defined(__KERNEL__)
|
|
up_pthread_exit(((FAR struct pthread_tcb_s *)rtcb)->exit,
|
|
PTHREAD_CANCELED);
|
|
#else
|
|
pthread_exit(PTHREAD_CANCELED);
|
|
#endif
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
UNUSED(rtcb);
|
|
|
|
/* Exit to terminate the task (note that exit() vs. _exit() is used. */
|
|
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_stop_task
|
|
*
|
|
* Description:
|
|
* This is the handler for the stop default action.
|
|
*
|
|
* Input Parameters:
|
|
* Standard signal handler parameters
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifdef CONFIG_SIG_SIGSTOP_ACTION
|
|
static void nxsig_stop_task(int signo)
|
|
{
|
|
FAR struct tcb_s *rtcb = (FAR struct tcb_s *)this_task();
|
|
#if defined(CONFIG_SCHED_WAITPID) && !defined(CONFIG_SCHED_HAVE_PARENT)
|
|
FAR struct task_group_s *group;
|
|
|
|
DEBUGASSERT(rtcb != NULL && rtcb->group != NULL);
|
|
group = rtcb->group;
|
|
#endif
|
|
|
|
/* Careful: In the multi-threaded task, the signal may be handled on a
|
|
* child pthread.
|
|
*/
|
|
|
|
#ifdef HAVE_GROUP_MEMBERS
|
|
/* Suspend all of the children of the task. This will not suspend the
|
|
* currently running task/pthread (this_task). It will suspend the
|
|
* main thread of the task group if this_task is a pthread.
|
|
*/
|
|
|
|
group_suspend_children(rtcb);
|
|
#endif
|
|
|
|
/* Lock the scheduler so this thread is not pre-empted until after we
|
|
* call nxsched_suspend().
|
|
*/
|
|
|
|
sched_lock();
|
|
|
|
#if defined(CONFIG_SCHED_WAITPID) && !defined(CONFIG_SCHED_HAVE_PARENT)
|
|
/* Notify via waitpid if any parent is waiting for this task to EXIT
|
|
* or STOP. This action is only performed if WUNTRACED is set in the
|
|
* waitpid flags.
|
|
*/
|
|
|
|
if ((group->tg_waitflags & WUNTRACED) != 0)
|
|
{
|
|
/* Return zero for exit status (we are not exiting, however) */
|
|
|
|
if (group->tg_statloc != NULL)
|
|
{
|
|
*group->tg_statloc = 0;
|
|
group->tg_statloc = NULL;
|
|
}
|
|
|
|
/* tg_waitflags == 0 means that the flags are available to another
|
|
* caller of waitpid().
|
|
*/
|
|
|
|
group->tg_waitflags = 0;
|
|
|
|
/* Wakeup any tasks waiting for this task to exit or stop. */
|
|
|
|
while (group->tg_exitsem.semcount < 0)
|
|
{
|
|
/* Wake up the thread */
|
|
|
|
nxsem_post(&group->tg_exitsem);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* Then, finally, suspend this the final thread of the task group */
|
|
|
|
nxsched_suspend(rtcb);
|
|
sched_unlock();
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_default_action
|
|
*
|
|
* Description:
|
|
* Look up the default action associated with this signal
|
|
*
|
|
* Input Parameters:
|
|
* signo - The signal number to use in the query
|
|
*
|
|
* Returned Value:
|
|
* The default handler associated with this signal
|
|
*
|
|
****************************************************************************/
|
|
|
|
static _sa_handler_t nxsig_default_action(int signo)
|
|
{
|
|
int i;
|
|
|
|
/* Search the default action table for the entry associated with this
|
|
* signal.
|
|
*/
|
|
|
|
for (i = 0; i < NACTIONS; i++)
|
|
{
|
|
if (g_defactions[i].signo == signo)
|
|
{
|
|
return g_defactions[i].action;
|
|
}
|
|
}
|
|
|
|
/* No default action */
|
|
|
|
return SIG_IGN;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_setup_default_action
|
|
*
|
|
* Description:
|
|
* Setup the default action for a signal.
|
|
*
|
|
* This function is called early in task setup, prior to the creation of
|
|
* any pthreads so we should have exclusive access to the group structure.
|
|
*
|
|
* Input Parameters:
|
|
* group - The group that the task belongs in.
|
|
* action - The new default signal action
|
|
* signo - The signal that will produce this default action
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
****************************************************************************/
|
|
|
|
static void nxsig_setup_default_action(FAR struct task_group_s *group,
|
|
FAR const struct nxsig_defaction_s *
|
|
info)
|
|
{
|
|
/* Get the address of the handler for this signals default action. */
|
|
|
|
if (info->action != SIG_IGN)
|
|
{
|
|
struct sigaction sa;
|
|
|
|
/* Attach the signal handler.
|
|
*
|
|
* NOTE: nxsig_action will call nxsig_default(tcb, action, false).
|
|
* Don't be surprised.
|
|
*/
|
|
|
|
memset(&sa, 0, sizeof(sa));
|
|
sa.sa_handler = info->action;
|
|
sa.sa_flags = SA_SIGINFO;
|
|
nxsig_action(info->signo, &sa, NULL, true);
|
|
|
|
/* Indicate that the default signal handler has been attached */
|
|
|
|
nxsig_addset(&group->tg_sigdefault, (int)info->signo);
|
|
}
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_isdefault
|
|
*
|
|
* Description:
|
|
* Return true if the specified signal is set to the default action.
|
|
*
|
|
* Input Parameters:
|
|
* tcb - Identifies the thread associated with the default handler
|
|
* signo - The signal number to be queried
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) is returned on success; A negated errno value is returned
|
|
* on failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
bool nxsig_isdefault(FAR struct tcb_s *tcb, int signo)
|
|
{
|
|
FAR struct task_group_s *group;
|
|
int ret;
|
|
|
|
DEBUGASSERT(tcb != NULL && tcb->group != NULL && GOOD_SIGNO(signo));
|
|
group = tcb->group;
|
|
|
|
/* Return true if the signo is marked as using the default action. Return
|
|
* false in all other cases.
|
|
*/
|
|
|
|
ret = nxsig_ismember(&group->tg_sigdefault, signo);
|
|
return ret < 0 ? false : (bool)ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_iscatchable
|
|
*
|
|
* Description:
|
|
* Return true if the specified signal can be caught or ignored.
|
|
*
|
|
* Input Parameters:
|
|
* signo - The signal number to be queried
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) is returned on success; A negated errno value is returned
|
|
* on failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
bool nxsig_iscatchable(int signo)
|
|
{
|
|
int i;
|
|
|
|
/* Search the default action table for the entry associated with this
|
|
* signal.
|
|
*/
|
|
|
|
for (i = 0; i < NACTIONS; i++)
|
|
{
|
|
if (g_defactions[i].signo == signo)
|
|
{
|
|
return (g_defactions[i].flags & SIG_FLAG_NOCATCH) == 0;
|
|
}
|
|
}
|
|
|
|
/* If it is not in the table, then it is catchable */
|
|
|
|
return true;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_default
|
|
*
|
|
* Description:
|
|
* If 'defaction' is true, then return the default signal handler action
|
|
* for the specified signal and mark that the default signal handler is
|
|
* in place (it is not yet).
|
|
*
|
|
* If 'defaction' is false, then mark that the default signal handler is
|
|
* NOT in place and return SIG_IGN.
|
|
*
|
|
* This function is called form sigaction() to handle actions = SIG_DFL.
|
|
*
|
|
* Input Parameters:
|
|
* tcb - Identifies the thread associated with the default handler
|
|
* signo - The signal number whose behavior will be modified.
|
|
* defaction - True: the default action is in place
|
|
*
|
|
* Returned Value:
|
|
* The address of the default signal action handler is returne on success.
|
|
* SIG_IGN is returned if there is no default action.
|
|
*
|
|
****************************************************************************/
|
|
|
|
_sa_handler_t nxsig_default(FAR struct tcb_s *tcb, int signo, bool defaction)
|
|
{
|
|
FAR struct task_group_s *group;
|
|
_sa_handler_t handler = SIG_IGN;
|
|
irqstate_t flags;
|
|
|
|
DEBUGASSERT(tcb != NULL && tcb->group != NULL);
|
|
group = tcb->group;
|
|
|
|
/* Are we setting or unsetting the default action? */
|
|
|
|
if (defaction)
|
|
{
|
|
/* We are setting the default action. Look up the default action
|
|
* associated with signo.
|
|
*/
|
|
|
|
handler = nxsig_default_action(signo);
|
|
if (handler != SIG_IGN)
|
|
{
|
|
/* nxsig_addset() is not atomic (but neither is sigaction()) */
|
|
|
|
flags = spin_lock_irqsave(NULL);
|
|
nxsig_addset(&group->tg_sigdefault, signo);
|
|
spin_unlock_irqrestore(NULL, flags);
|
|
}
|
|
}
|
|
|
|
if (handler == SIG_IGN)
|
|
{
|
|
/* We are unsetting the default action. NOTE that nxsig_delset() is not
|
|
* atomic (but neither is sigaction()).
|
|
*/
|
|
|
|
flags = spin_lock_irqsave(NULL);
|
|
nxsig_delset(&group->tg_sigdefault, signo);
|
|
spin_unlock_irqrestore(NULL, flags);
|
|
}
|
|
|
|
return handler;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: nxsig_default_initialize
|
|
*
|
|
* Description:
|
|
* Set all signals to their default action. This is called from
|
|
* nxtask_start() to configure the newly started task.
|
|
*
|
|
* Input Parameters:
|
|
* tcb - Identifies the thread associated with the default handlers
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) is returned on success; A negated errno value is returned
|
|
* on failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
int nxsig_default_initialize(FAR struct tcb_s *tcb)
|
|
{
|
|
FAR struct task_group_s *group;
|
|
int i;
|
|
|
|
DEBUGASSERT(tcb != NULL && tcb->group != NULL);
|
|
group = tcb->group;
|
|
|
|
/* Initialize the set of default signal handlers */
|
|
|
|
sigemptyset(&group->tg_sigdefault);
|
|
|
|
/* Setup the default action for each signal in g_defactions[] */
|
|
|
|
for (i = 0; i < NACTIONS; i++)
|
|
{
|
|
nxsig_setup_default_action(group, &g_defactions[i]);
|
|
}
|
|
|
|
return OK;
|
|
}
|