idle.c 4.8 KB
Newer Older
1 2 3
/*
 * File      : idle.c
 * This file is part of RT-Thread RTOS
D
dzzxzz 已提交
4
 * COPYRIGHT (C) 2006 - 2012, RT-Thread Development Team
5 6 7
 *
 * The license and distribution terms for this file may be
 * found in the file LICENSE in this distribution or at
B
bernard.xiong 已提交
8
 * http://www.rt-thread.org/license/LICENSE
9 10 11 12
 *
 * Change Logs:
 * Date           Author       Notes
 * 2006-03-23     Bernard      the first version
13
 * 2010-11-10     Bernard      add cleanup callback function in thread exit.
14 15 16 17 18
 */

#include <rthw.h>
#include <rtthread.h>

19
#ifndef IDLE_THREAD_STACK_SIZE
20
#if defined (RT_USING_HOOK) || defined(RT_USING_HEAP)
21
#define IDLE_THREAD_STACK_SIZE  256
22
#else
23
#define IDLE_THREAD_STACK_SIZE  128
24
#endif
25
#endif
26 27

static struct rt_thread idle;
28
ALIGN(RT_ALIGN_SIZE)
29 30 31 32 33 34 35 36
static rt_uint8_t rt_thread_stack[IDLE_THREAD_STACK_SIZE];

extern rt_list_t rt_thread_defunct;

#ifdef RT_USING_HOOK
/**
 * @addtogroup Hook
 */
D
dzzxzz 已提交
37

38 39 40 41 42 43 44 45 46 47 48 49 50
/*@{*/

static void (*rt_thread_idle_hook)();

/**
 * This function will set a hook function to idle thread loop.
 *
 * @param hook the specified hook function
 *
 * @note the hook function must be simple and never be blocked or suspend.
 */
void rt_thread_idle_sethook(void (*hook)())
{
51
    rt_thread_idle_hook = hook;
52
}
D
dzzxzz 已提交
53

54 55 56
/*@}*/
#endif

B
bernard.xiong 已提交
57
/**
B
bernard.xiong@gmail.com 已提交
58 59 60
 * @ingroup Thread
 *
 * This function will perform system background job when system idle.
B
bernard.xiong 已提交
61 62
 */
void rt_thread_idle_excute(void)
63
{
64 65 66 67 68
    /* check the defunct thread list */
    if (!rt_list_isempty(&rt_thread_defunct))
    {
        rt_base_t lock;
        rt_thread_t thread;
B
bernard.xiong 已提交
69
#ifdef RT_USING_MODULE
70
        rt_module_t module = RT_NULL;
71
#endif
72 73 74 75 76 77 78 79 80 81 82 83
        RT_DEBUG_NOT_IN_INTERRUPT;

        /* disable interrupt */
        lock = rt_hw_interrupt_disable();

        /* re-check whether list is empty */
        if (!rt_list_isempty(&rt_thread_defunct))
        {
            /* get defunct thread */
            thread = rt_list_entry(rt_thread_defunct.next,
                                   struct rt_thread,
                                   tlist);
qiuyiuestc's avatar
qiuyiuestc 已提交
84
#ifdef RT_USING_MODULE
85 86 87 88 89 90 91 92 93
            /* get thread's parent module */
            module = (rt_module_t)thread->module_id;

            /* if the thread is module's main thread */
            if (module != RT_NULL && module->module_thread == thread)
            {
                /* detach module's main thread */
                module->module_thread = RT_NULL;
            }
qiuyiuestc's avatar
qiuyiuestc 已提交
94
#endif
95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
            /* remove defunct thread */
            rt_list_remove(&(thread->tlist));
            /* invoke thread cleanup */
            if (thread->cleanup != RT_NULL)
                thread->cleanup(thread);

            /* if it's a system object, not delete it */
            if (rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE)
            {
                /* enable interrupt */
                rt_hw_interrupt_enable(lock);

                return;
            }
        }
        else
        {
            /* enable interrupt */
            rt_hw_interrupt_enable(lock);

            /* may the defunct thread list is removed by others, just return */
            return;
        }

        /* enable interrupt */
        rt_hw_interrupt_enable(lock);
qiuyiuestc's avatar
qiuyiuestc 已提交
121

122
#ifdef RT_USING_HEAP
qiuyiuestc's avatar
qiuyiuestc 已提交
123
#if defined(RT_USING_MODULE) && defined(RT_USING_SLAB)
124 125 126 127
        /* the thread belongs to an application module */
        if (thread->flags & RT_OBJECT_FLAG_MODULE)
            rt_module_free((rt_module_t)thread->module_id, thread->stack_addr);
        else
qiuyiuestc's avatar
qiuyiuestc 已提交
128
#endif
129 130 131 132
        /* release thread's stack */
        rt_free(thread->stack_addr);
        /* delete thread object */
        rt_object_delete((rt_object_t)thread);
qiuyiuestc's avatar
qiuyiuestc 已提交
133
#endif
qiuyiuestc's avatar
qiuyiuestc 已提交
134 135

#ifdef RT_USING_MODULE
136 137 138 139 140 141 142 143 144 145 146 147 148 149 150
        if (module != RT_NULL)
        {
            extern rt_err_t rt_module_destroy(rt_module_t module);

            /* if sub thread list and main thread are all empty */
            if ((module->module_thread == RT_NULL) &&
                rt_list_isempty(&module->module_object[RT_Object_Class_Thread].object_list))
            {
                module->nref --;
            }

            /* destroy module */
            if (module->nref == 0)
                rt_module_destroy(module);
        }
qiuyiuestc's avatar
qiuyiuestc 已提交
151
#endif
152
    }
B
bernard.xiong 已提交
153 154
}

D
dzzxzz 已提交
155
static void rt_thread_idle_entry(void *parameter)
B
bernard.xiong 已提交
156
{
157 158 159 160 161 162 163 164 165
    while (1)
    {
        #ifdef RT_USING_HOOK
        if (rt_thread_idle_hook != RT_NULL)
            rt_thread_idle_hook();
        #endif

        rt_thread_idle_excute();
    }
166 167 168
}

/**
B
bernard.xiong@gmail.com 已提交
169 170
 * @ingroup SymstemInit
 *
171 172 173 174
 * This function will initialize idle thread, then start it.
 *
 * @note this function must be invoked when system init.
 */
D
dzzxzz 已提交
175
void rt_thread_idle_init(void)
176
{
177 178 179 180 181 182 183 184 185 186 187 188
    /* initialize thread */
    rt_thread_init(&idle,
                   "tidle",
                   rt_thread_idle_entry,
                   RT_NULL,
                   &rt_thread_stack[0],
                   sizeof(rt_thread_stack),
                   RT_THREAD_PRIORITY_MAX - 1,
                   32);

    /* startup */
    rt_thread_startup(&idle);
189
}