/* ChibiOS/RT - Copyright (C) 2006,2007,2008,2009,2010 Giovanni Di Sirio. This file is part of ChibiOS/RT. ChibiOS/RT is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. ChibiOS/RT 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ /** * @page article_lifecycle Threads Lifecycle * In ChibiOS/RT threads are divided in two categories: * - Static threads. The memory used for static threads is allocated at * compile time so static threads are always there, there is no management * to be done. * - Dynamic threads. Dynamic threads are allocated at runtime from one of * the available allocators (see @ref heaps, @ref pools). * . * Dynamic threads create the problem of who is responsible of releasing * their memory because a thread cannot dispose its own memory.
* This is handled in ChibiOS/RT through the mechanism of "thread references", * When the @p CH_USE_DYNAMIC option is enabled the threads becomes objects * with a reference counter. The memory of a thread, if dynamic, is released * when the last reference to the thread is released while the thread is in * its @p THD_STATE_FINAL state.
* The following diagram explains the mechanism: * @dot digraph example { rankdir="LR"; node [shape=circle, fontname=Helvetica, fontsize=8, fixedsize="true", width="0.75", height="0.75"]; edge [fontname=Helvetica, fontsize=8]; init [label="No thread", style="bold"]; alive [label="Alive"]; final [label="Terminated"]; detached [label="Detached", style="bold"]; init -> alive [label="chThdCreateX()"]; alive -> alive [label="chThdAddRef()"]; alive -> alive [label="chThdRelease()\n[ref > 0]"]; alive -> detached [label="chThdRelease()\n[ref == 0]"]; alive -> init [label="chThdWait()\n[ref == 0]"]; alive -> final [label="chThdExit()\nreturn"]; final -> final [label="chThdAddRef()"]; final -> final [label="chThdRelease()\nchThdWait()\n[ref > 0]"]; final -> init [label="chThdRelease()\nchThdWait()\n[ref == 0]"]; } * @enddot *
* As you can see the simplest way to ensure that the memory is released is * that another threads performs a @p chThdWait() on the dynamic thread.
* If all the references to the threads are released while the thread is * still alive then the thread goes in a "detached" state and its memory * cannot be recovered unless there is a dedicated task in the system that * scans the threads through the @ref registry subsystem and frees the * terminated ones. */