/************************************************************************ * sched/sched_releasetcb.c * * Copyright (C) 2007, 2009, 2012-2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name NuttX nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ************************************************************************/ /************************************************************************ * Included Files ************************************************************************/ #include #include #include #include #include #include "os_internal.h" #include "group/group.h" #include "timer/timer.h" /************************************************************************ * Private Functions ************************************************************************/ /************************************************************************ * Name: sched_releasepid * * Description: When a task is destroyed, this function must * be called to make its process ID available for re-use. ************************************************************************/ static void sched_releasepid(pid_t pid) { int hash_ndx = PIDHASH(pid); /* Make any pid associated with this hash available. Note: * no special precautions need be taken here because the * following action is atomic */ g_pidhash[hash_ndx].tcb = NULL; g_pidhash[hash_ndx].pid = INVALID_PROCESS_ID; #ifdef CONFIG_SCHED_CPULOAD /* Decrement the total CPU load count held by this thread from the * total for all threads. Then we can reset the count on this * defunct thread to zero. */ g_cpuload_total -= g_pidhash[hash_ndx].ticks; g_pidhash[hash_ndx].ticks = 0; #endif } /************************************************************************ * Public Functions ************************************************************************/ /************************************************************************ * Name: sched_releasetcb * * Description: * Free all resources contained in a TCB * * Parameters: * tcb - The TCB to be released * ttype - The type of the TCB to be released * * This thread type is normally available in the flags field of the TCB, * however, there are certain error recovery contexts where the TCB my * not be fully initialized when sched_releasetcb is called. * * Return Value: * OK on success; ERROR on failure * * Assumptions: * Interrupts are disabled. * ************************************************************************/ int sched_releasetcb(FAR struct tcb_s *tcb, uint8_t ttype) { int ret = OK; #if defined(CONFIG_CUSTOM_STACK) || !defined(CONFIG_NUTTX_KERNEL) int i; #endif if (tcb) { /* Relase any timers that the task might hold. We do this * before release the PID because it may still be trying to * deliver signals (although interrupts are should be * disabled here). */ #ifndef CONFIG_DISABLE_POSIX_TIMERS #ifdef CONFIG_HAVE_WEAKFUNCTIONS if (timer_deleteall != NULL) #endif { timer_deleteall(tcb->pid); } #endif /* Release the task's process ID if one was assigned. PID * zero is reserved for the IDLE task. The TCB of the IDLE * task is never release so a value of zero simply means that * the process ID was never allocated to this TCB. */ if (tcb->pid) { sched_releasepid(tcb->pid); } /* Delete the thread's stack if one has been allocated */ #ifndef CONFIG_CUSTOM_STACK if (tcb->stack_alloc_ptr) { up_release_stack(tcb, ttype); } #endif /* Delete the tasks's allocated DSpace region (external modules only) */ #ifdef CONFIG_PIC if (tcb->dspace) { if (tcb->dspace->crefs <= 1) { sched_kfree(tcb->dspace); } else { tcb->dspace->crefs--; } } #endif #if defined(CONFIG_CUSTOM_STACK) || !defined(CONFIG_NUTTX_KERNEL) /* Release command line arguments that were allocated for task * start/re-start. * * NOTE: In the kernel mode build, the arguments were saved on * the task's stack and will be cleaned up when the stack memory * is released. Nothing need be done here in that case. */ #ifndef CONFIG_DISABLE_PTHREAD if (ttype != TCB_FLAG_TTYPE_PTHREAD) #endif { FAR struct task_tcb_s *ttcb = (FAR struct task_tcb_s *)tcb; for (i = 1; i < CONFIG_MAX_TASK_ARGS+1 && ttcb->argv[i]; i++) { sched_kfree((FAR void*)ttcb->argv[i]); } } #endif /* CONFIG_CUSTOM_STACK || !CONFIG_NUTTX_KERNEL */ /* Release this thread's reference to the address environment */ #ifdef CONFIG_ADDRENV ret = up_addrenv_release(tcb); #endif /* Leave the group (if we did not already leady in task_exithook.c) */ #ifdef HAVE_TASK_GROUP group_leave(tcb); #endif /* And, finally, release the TCB itself */ sched_kfree(tcb); } return ret; }