xref: /nrf52832-nimble/rt-thread/src/thread.c (revision 104654410c56c573564690304ae786df310c91fc)
1*10465441SEvalZero /*
2*10465441SEvalZero  * Copyright (c) 2006-2018, RT-Thread Development Team
3*10465441SEvalZero  *
4*10465441SEvalZero  * SPDX-License-Identifier: Apache-2.0
5*10465441SEvalZero  *
6*10465441SEvalZero  * Change Logs:
7*10465441SEvalZero  * Date           Author       Notes
8*10465441SEvalZero  * 2006-03-28     Bernard      first version
9*10465441SEvalZero  * 2006-04-29     Bernard      implement thread timer
10*10465441SEvalZero  * 2006-04-30     Bernard      added THREAD_DEBUG
11*10465441SEvalZero  * 2006-05-27     Bernard      fixed the rt_thread_yield bug
12*10465441SEvalZero  * 2006-06-03     Bernard      fixed the thread timer init bug
13*10465441SEvalZero  * 2006-08-10     Bernard      fixed the timer bug in thread_sleep
14*10465441SEvalZero  * 2006-09-03     Bernard      changed rt_timer_delete to rt_timer_detach
15*10465441SEvalZero  * 2006-09-03     Bernard      implement rt_thread_detach
16*10465441SEvalZero  * 2008-02-16     Bernard      fixed the rt_thread_timeout bug
17*10465441SEvalZero  * 2010-03-21     Bernard      change the errno of rt_thread_delay/sleep to
18*10465441SEvalZero  *                             RT_EOK.
19*10465441SEvalZero  * 2010-11-10     Bernard      add cleanup callback function in thread exit.
20*10465441SEvalZero  * 2011-09-01     Bernard      fixed rt_thread_exit issue when the current
21*10465441SEvalZero  *                             thread preempted, which reported by Jiaxing Lee.
22*10465441SEvalZero  * 2011-09-08     Bernard      fixed the scheduling issue in rt_thread_startup.
23*10465441SEvalZero  * 2012-12-29     Bernard      fixed compiling warning.
24*10465441SEvalZero  * 2016-08-09     ArdaFu       add thread suspend and resume hook.
25*10465441SEvalZero  * 2017-04-10     armink       fixed the rt_thread_delete and rt_thread_detach
26*10465441SEvalZero  *                             bug when thread has not startup.
27*10465441SEvalZero  * 2018-11-22     Jesven       yield is same to rt_schedule
28*10465441SEvalZero  *                             add support for tasks bound to cpu
29*10465441SEvalZero  */
30*10465441SEvalZero 
31*10465441SEvalZero #include <rthw.h>
32*10465441SEvalZero #include <rtthread.h>
33*10465441SEvalZero 
34*10465441SEvalZero extern rt_list_t rt_thread_defunct;
35*10465441SEvalZero 
36*10465441SEvalZero #ifdef RT_USING_HOOK
37*10465441SEvalZero static void (*rt_thread_suspend_hook)(rt_thread_t thread);
38*10465441SEvalZero static void (*rt_thread_resume_hook) (rt_thread_t thread);
39*10465441SEvalZero static void (*rt_thread_inited_hook) (rt_thread_t thread);
40*10465441SEvalZero 
41*10465441SEvalZero /**
42*10465441SEvalZero  * @ingroup Hook
43*10465441SEvalZero  * This function sets a hook function when the system suspend a thread.
44*10465441SEvalZero  *
45*10465441SEvalZero  * @param hook the specified hook function
46*10465441SEvalZero  *
47*10465441SEvalZero  * @note the hook function must be simple and never be blocked or suspend.
48*10465441SEvalZero  */
rt_thread_suspend_sethook(void (* hook)(rt_thread_t thread))49*10465441SEvalZero void rt_thread_suspend_sethook(void (*hook)(rt_thread_t thread))
50*10465441SEvalZero {
51*10465441SEvalZero     rt_thread_suspend_hook = hook;
52*10465441SEvalZero }
53*10465441SEvalZero 
54*10465441SEvalZero /**
55*10465441SEvalZero  * @ingroup Hook
56*10465441SEvalZero  * This function sets a hook function when the system resume a thread.
57*10465441SEvalZero  *
58*10465441SEvalZero  * @param hook the specified hook function
59*10465441SEvalZero  *
60*10465441SEvalZero  * @note the hook function must be simple and never be blocked or suspend.
61*10465441SEvalZero  */
rt_thread_resume_sethook(void (* hook)(rt_thread_t thread))62*10465441SEvalZero void rt_thread_resume_sethook(void (*hook)(rt_thread_t thread))
63*10465441SEvalZero {
64*10465441SEvalZero     rt_thread_resume_hook = hook;
65*10465441SEvalZero }
66*10465441SEvalZero 
67*10465441SEvalZero /**
68*10465441SEvalZero  * @ingroup Hook
69*10465441SEvalZero  * This function sets a hook function when a thread is initialized.
70*10465441SEvalZero  *
71*10465441SEvalZero  * @param hook the specified hook function
72*10465441SEvalZero  */
rt_thread_inited_sethook(void (* hook)(rt_thread_t thread))73*10465441SEvalZero void rt_thread_inited_sethook(void (*hook)(rt_thread_t thread))
74*10465441SEvalZero {
75*10465441SEvalZero     rt_thread_inited_hook = hook;
76*10465441SEvalZero }
77*10465441SEvalZero 
78*10465441SEvalZero #endif
79*10465441SEvalZero 
rt_thread_exit(void)80*10465441SEvalZero void rt_thread_exit(void)
81*10465441SEvalZero {
82*10465441SEvalZero     struct rt_thread *thread;
83*10465441SEvalZero     register rt_base_t level;
84*10465441SEvalZero 
85*10465441SEvalZero     /* get current thread */
86*10465441SEvalZero     thread = rt_thread_self();
87*10465441SEvalZero 
88*10465441SEvalZero     /* disable interrupt */
89*10465441SEvalZero     level = rt_hw_interrupt_disable();
90*10465441SEvalZero 
91*10465441SEvalZero     /* remove from schedule */
92*10465441SEvalZero     rt_schedule_remove_thread(thread);
93*10465441SEvalZero     /* change stat */
94*10465441SEvalZero     thread->stat = RT_THREAD_CLOSE;
95*10465441SEvalZero 
96*10465441SEvalZero     /* remove it from timer list */
97*10465441SEvalZero     rt_timer_detach(&thread->thread_timer);
98*10465441SEvalZero 
99*10465441SEvalZero     if ((rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE) &&
100*10465441SEvalZero         thread->cleanup == RT_NULL)
101*10465441SEvalZero     {
102*10465441SEvalZero         rt_object_detach((rt_object_t)thread);
103*10465441SEvalZero     }
104*10465441SEvalZero     else
105*10465441SEvalZero     {
106*10465441SEvalZero         /* insert to defunct thread list */
107*10465441SEvalZero         rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
108*10465441SEvalZero     }
109*10465441SEvalZero 
110*10465441SEvalZero     /* enable interrupt */
111*10465441SEvalZero     rt_hw_interrupt_enable(level);
112*10465441SEvalZero 
113*10465441SEvalZero     /* switch to next task */
114*10465441SEvalZero     rt_schedule();
115*10465441SEvalZero }
116*10465441SEvalZero 
_rt_thread_init(struct rt_thread * thread,const char * name,void (* entry)(void * parameter),void * parameter,void * stack_start,rt_uint32_t stack_size,rt_uint8_t priority,rt_uint32_t tick)117*10465441SEvalZero static rt_err_t _rt_thread_init(struct rt_thread *thread,
118*10465441SEvalZero                                 const char       *name,
119*10465441SEvalZero                                 void (*entry)(void *parameter),
120*10465441SEvalZero                                 void             *parameter,
121*10465441SEvalZero                                 void             *stack_start,
122*10465441SEvalZero                                 rt_uint32_t       stack_size,
123*10465441SEvalZero                                 rt_uint8_t        priority,
124*10465441SEvalZero                                 rt_uint32_t       tick)
125*10465441SEvalZero {
126*10465441SEvalZero     /* init thread list */
127*10465441SEvalZero     rt_list_init(&(thread->tlist));
128*10465441SEvalZero 
129*10465441SEvalZero     thread->entry = (void *)entry;
130*10465441SEvalZero     thread->parameter = parameter;
131*10465441SEvalZero 
132*10465441SEvalZero     /* stack init */
133*10465441SEvalZero     thread->stack_addr = stack_start;
134*10465441SEvalZero     thread->stack_size = stack_size;
135*10465441SEvalZero 
136*10465441SEvalZero     /* init thread stack */
137*10465441SEvalZero     rt_memset(thread->stack_addr, '#', thread->stack_size);
138*10465441SEvalZero #ifdef ARCH_CPU_STACK_GROWS_UPWARD
139*10465441SEvalZero     thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter,
140*10465441SEvalZero                                           (void *)((char *)thread->stack_addr),
141*10465441SEvalZero                                           (void *)rt_thread_exit);
142*10465441SEvalZero #else
143*10465441SEvalZero     thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter,
144*10465441SEvalZero                                           (void *)((char *)thread->stack_addr + thread->stack_size - sizeof(rt_ubase_t)),
145*10465441SEvalZero                                           (void *)rt_thread_exit);
146*10465441SEvalZero #endif
147*10465441SEvalZero 
148*10465441SEvalZero     /* priority init */
149*10465441SEvalZero     RT_ASSERT(priority < RT_THREAD_PRIORITY_MAX);
150*10465441SEvalZero     thread->init_priority    = priority;
151*10465441SEvalZero     thread->current_priority = priority;
152*10465441SEvalZero 
153*10465441SEvalZero     thread->number_mask = 0;
154*10465441SEvalZero #if RT_THREAD_PRIORITY_MAX > 32
155*10465441SEvalZero     thread->number = 0;
156*10465441SEvalZero     thread->high_mask = 0;
157*10465441SEvalZero #endif
158*10465441SEvalZero 
159*10465441SEvalZero     /* tick init */
160*10465441SEvalZero     thread->init_tick      = tick;
161*10465441SEvalZero     thread->remaining_tick = tick;
162*10465441SEvalZero 
163*10465441SEvalZero     /* error and flags */
164*10465441SEvalZero     thread->error = RT_EOK;
165*10465441SEvalZero     thread->stat  = RT_THREAD_INIT;
166*10465441SEvalZero 
167*10465441SEvalZero #ifdef RT_USING_SMP
168*10465441SEvalZero     /* not bind on any cpu */
169*10465441SEvalZero     thread->bind_cpu = RT_CPUS_NR;
170*10465441SEvalZero     thread->oncpu = RT_CPU_DETACHED;
171*10465441SEvalZero 
172*10465441SEvalZero     /* lock init */
173*10465441SEvalZero     thread->scheduler_lock_nest = 0;
174*10465441SEvalZero     thread->cpus_lock_nest = 0;
175*10465441SEvalZero #endif /*RT_USING_SMP*/
176*10465441SEvalZero 
177*10465441SEvalZero     /* initialize cleanup function and user data */
178*10465441SEvalZero     thread->cleanup   = 0;
179*10465441SEvalZero     thread->user_data = 0;
180*10465441SEvalZero 
181*10465441SEvalZero     /* init thread timer */
182*10465441SEvalZero     rt_timer_init(&(thread->thread_timer),
183*10465441SEvalZero                   thread->name,
184*10465441SEvalZero                   rt_thread_timeout,
185*10465441SEvalZero                   thread,
186*10465441SEvalZero                   0,
187*10465441SEvalZero                   RT_TIMER_FLAG_ONE_SHOT);
188*10465441SEvalZero 
189*10465441SEvalZero     /* initialize signal */
190*10465441SEvalZero #ifdef RT_USING_SIGNALS
191*10465441SEvalZero     thread->sig_mask    = 0x00;
192*10465441SEvalZero     thread->sig_pending = 0x00;
193*10465441SEvalZero 
194*10465441SEvalZero     thread->sig_ret     = RT_NULL;
195*10465441SEvalZero     thread->sig_vectors = RT_NULL;
196*10465441SEvalZero     thread->si_list     = RT_NULL;
197*10465441SEvalZero #endif
198*10465441SEvalZero 
199*10465441SEvalZero #ifdef RT_USING_LWP
200*10465441SEvalZero     thread->lwp = RT_NULL;
201*10465441SEvalZero #endif
202*10465441SEvalZero 
203*10465441SEvalZero     RT_OBJECT_HOOK_CALL(rt_thread_inited_hook, (thread));
204*10465441SEvalZero 
205*10465441SEvalZero     return RT_EOK;
206*10465441SEvalZero }
207*10465441SEvalZero 
208*10465441SEvalZero /**
209*10465441SEvalZero  * @addtogroup Thread
210*10465441SEvalZero  */
211*10465441SEvalZero 
212*10465441SEvalZero /**@{*/
213*10465441SEvalZero 
214*10465441SEvalZero /**
215*10465441SEvalZero  * This function will initialize a thread, normally it's used to initialize a
216*10465441SEvalZero  * static thread object.
217*10465441SEvalZero  *
218*10465441SEvalZero  * @param thread the static thread object
219*10465441SEvalZero  * @param name the name of thread, which shall be unique
220*10465441SEvalZero  * @param entry the entry function of thread
221*10465441SEvalZero  * @param parameter the parameter of thread enter function
222*10465441SEvalZero  * @param stack_start the start address of thread stack
223*10465441SEvalZero  * @param stack_size the size of thread stack
224*10465441SEvalZero  * @param priority the priority of thread
225*10465441SEvalZero  * @param tick the time slice if there are same priority thread
226*10465441SEvalZero  *
227*10465441SEvalZero  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
228*10465441SEvalZero  */
rt_thread_init(struct rt_thread * thread,const char * name,void (* entry)(void * parameter),void * parameter,void * stack_start,rt_uint32_t stack_size,rt_uint8_t priority,rt_uint32_t tick)229*10465441SEvalZero rt_err_t rt_thread_init(struct rt_thread *thread,
230*10465441SEvalZero                         const char       *name,
231*10465441SEvalZero                         void (*entry)(void *parameter),
232*10465441SEvalZero                         void             *parameter,
233*10465441SEvalZero                         void             *stack_start,
234*10465441SEvalZero                         rt_uint32_t       stack_size,
235*10465441SEvalZero                         rt_uint8_t        priority,
236*10465441SEvalZero                         rt_uint32_t       tick)
237*10465441SEvalZero {
238*10465441SEvalZero     /* thread check */
239*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
240*10465441SEvalZero     RT_ASSERT(stack_start != RT_NULL);
241*10465441SEvalZero 
242*10465441SEvalZero     /* init thread object */
243*10465441SEvalZero     rt_object_init((rt_object_t)thread, RT_Object_Class_Thread, name);
244*10465441SEvalZero 
245*10465441SEvalZero     return _rt_thread_init(thread,
246*10465441SEvalZero                            name,
247*10465441SEvalZero                            entry,
248*10465441SEvalZero                            parameter,
249*10465441SEvalZero                            stack_start,
250*10465441SEvalZero                            stack_size,
251*10465441SEvalZero                            priority,
252*10465441SEvalZero                            tick);
253*10465441SEvalZero }
254*10465441SEvalZero RTM_EXPORT(rt_thread_init);
255*10465441SEvalZero 
256*10465441SEvalZero /**
257*10465441SEvalZero  * This function will return self thread object
258*10465441SEvalZero  *
259*10465441SEvalZero  * @return the self thread object
260*10465441SEvalZero  */
rt_thread_self(void)261*10465441SEvalZero rt_thread_t rt_thread_self(void)
262*10465441SEvalZero {
263*10465441SEvalZero #ifdef RT_USING_SMP
264*10465441SEvalZero     rt_base_t lock;
265*10465441SEvalZero     rt_thread_t self;
266*10465441SEvalZero 
267*10465441SEvalZero     lock = rt_hw_local_irq_disable();
268*10465441SEvalZero     self = rt_cpu_self()->current_thread;
269*10465441SEvalZero     rt_hw_local_irq_enable(lock);
270*10465441SEvalZero     return self;
271*10465441SEvalZero #else
272*10465441SEvalZero     extern rt_thread_t rt_current_thread;
273*10465441SEvalZero 
274*10465441SEvalZero     return rt_current_thread;
275*10465441SEvalZero #endif
276*10465441SEvalZero }
277*10465441SEvalZero RTM_EXPORT(rt_thread_self);
278*10465441SEvalZero 
279*10465441SEvalZero /**
280*10465441SEvalZero  * This function will start a thread and put it to system ready queue
281*10465441SEvalZero  *
282*10465441SEvalZero  * @param thread the thread to be started
283*10465441SEvalZero  *
284*10465441SEvalZero  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
285*10465441SEvalZero  */
rt_thread_startup(rt_thread_t thread)286*10465441SEvalZero rt_err_t rt_thread_startup(rt_thread_t thread)
287*10465441SEvalZero {
288*10465441SEvalZero     /* thread check */
289*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
290*10465441SEvalZero     RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_INIT);
291*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
292*10465441SEvalZero 
293*10465441SEvalZero     /* set current priority to init priority */
294*10465441SEvalZero     thread->current_priority = thread->init_priority;
295*10465441SEvalZero 
296*10465441SEvalZero     /* calculate priority attribute */
297*10465441SEvalZero #if RT_THREAD_PRIORITY_MAX > 32
298*10465441SEvalZero     thread->number      = thread->current_priority >> 3;            /* 5bit */
299*10465441SEvalZero     thread->number_mask = 1L << thread->number;
300*10465441SEvalZero     thread->high_mask   = 1L << (thread->current_priority & 0x07);  /* 3bit */
301*10465441SEvalZero #else
302*10465441SEvalZero     thread->number_mask = 1L << thread->current_priority;
303*10465441SEvalZero #endif
304*10465441SEvalZero 
305*10465441SEvalZero     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("startup a thread:%s with priority:%d\n",
306*10465441SEvalZero                                    thread->name, thread->init_priority));
307*10465441SEvalZero     /* change thread stat */
308*10465441SEvalZero     thread->stat = RT_THREAD_SUSPEND;
309*10465441SEvalZero     /* then resume it */
310*10465441SEvalZero     rt_thread_resume(thread);
311*10465441SEvalZero     if (rt_thread_self() != RT_NULL)
312*10465441SEvalZero     {
313*10465441SEvalZero         /* do a scheduling */
314*10465441SEvalZero         rt_schedule();
315*10465441SEvalZero     }
316*10465441SEvalZero 
317*10465441SEvalZero     return RT_EOK;
318*10465441SEvalZero }
319*10465441SEvalZero RTM_EXPORT(rt_thread_startup);
320*10465441SEvalZero 
321*10465441SEvalZero /**
322*10465441SEvalZero  * This function will detach a thread. The thread object will be removed from
323*10465441SEvalZero  * thread queue and detached/deleted from system object management.
324*10465441SEvalZero  *
325*10465441SEvalZero  * @param thread the thread to be deleted
326*10465441SEvalZero  *
327*10465441SEvalZero  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
328*10465441SEvalZero  */
rt_thread_detach(rt_thread_t thread)329*10465441SEvalZero rt_err_t rt_thread_detach(rt_thread_t thread)
330*10465441SEvalZero {
331*10465441SEvalZero     rt_base_t lock;
332*10465441SEvalZero 
333*10465441SEvalZero     /* thread check */
334*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
335*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
336*10465441SEvalZero     RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread));
337*10465441SEvalZero 
338*10465441SEvalZero     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
339*10465441SEvalZero     {
340*10465441SEvalZero         /* remove from schedule */
341*10465441SEvalZero         rt_schedule_remove_thread(thread);
342*10465441SEvalZero     }
343*10465441SEvalZero 
344*10465441SEvalZero     /* release thread timer */
345*10465441SEvalZero     rt_timer_detach(&(thread->thread_timer));
346*10465441SEvalZero 
347*10465441SEvalZero     /* change stat */
348*10465441SEvalZero     thread->stat = RT_THREAD_CLOSE;
349*10465441SEvalZero 
350*10465441SEvalZero     /* detach object */
351*10465441SEvalZero     rt_object_detach((rt_object_t)thread);
352*10465441SEvalZero 
353*10465441SEvalZero     if (thread->cleanup != RT_NULL)
354*10465441SEvalZero     {
355*10465441SEvalZero         /* disable interrupt */
356*10465441SEvalZero         lock = rt_hw_interrupt_disable();
357*10465441SEvalZero 
358*10465441SEvalZero         /* insert to defunct thread list */
359*10465441SEvalZero         rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
360*10465441SEvalZero 
361*10465441SEvalZero         /* enable interrupt */
362*10465441SEvalZero         rt_hw_interrupt_enable(lock);
363*10465441SEvalZero     }
364*10465441SEvalZero 
365*10465441SEvalZero     return RT_EOK;
366*10465441SEvalZero }
367*10465441SEvalZero RTM_EXPORT(rt_thread_detach);
368*10465441SEvalZero 
369*10465441SEvalZero 
370*10465441SEvalZero #ifdef RT_USING_HEAP
371*10465441SEvalZero /**
372*10465441SEvalZero  * This function will create a thread object and allocate thread object memory
373*10465441SEvalZero  * and stack.
374*10465441SEvalZero  *
375*10465441SEvalZero  * @param name the name of thread, which shall be unique
376*10465441SEvalZero  * @param entry the entry function of thread
377*10465441SEvalZero  * @param parameter the parameter of thread enter function
378*10465441SEvalZero  * @param stack_size the size of thread stack
379*10465441SEvalZero  * @param priority the priority of thread
380*10465441SEvalZero  * @param tick the time slice if there are same priority thread
381*10465441SEvalZero  *
382*10465441SEvalZero  * @return the created thread object
383*10465441SEvalZero  */
rt_thread_create(const char * name,void (* entry)(void * parameter),void * parameter,rt_uint32_t stack_size,rt_uint8_t priority,rt_uint32_t tick)384*10465441SEvalZero rt_thread_t rt_thread_create(const char *name,
385*10465441SEvalZero                              void (*entry)(void *parameter),
386*10465441SEvalZero                              void       *parameter,
387*10465441SEvalZero                              rt_uint32_t stack_size,
388*10465441SEvalZero                              rt_uint8_t  priority,
389*10465441SEvalZero                              rt_uint32_t tick)
390*10465441SEvalZero {
391*10465441SEvalZero     struct rt_thread *thread;
392*10465441SEvalZero     void *stack_start;
393*10465441SEvalZero 
394*10465441SEvalZero     thread = (struct rt_thread *)rt_object_allocate(RT_Object_Class_Thread,
395*10465441SEvalZero                                                     name);
396*10465441SEvalZero     if (thread == RT_NULL)
397*10465441SEvalZero         return RT_NULL;
398*10465441SEvalZero 
399*10465441SEvalZero     stack_start = (void *)RT_KERNEL_MALLOC(stack_size);
400*10465441SEvalZero     if (stack_start == RT_NULL)
401*10465441SEvalZero     {
402*10465441SEvalZero         /* allocate stack failure */
403*10465441SEvalZero         rt_object_delete((rt_object_t)thread);
404*10465441SEvalZero 
405*10465441SEvalZero         return RT_NULL;
406*10465441SEvalZero     }
407*10465441SEvalZero 
408*10465441SEvalZero     _rt_thread_init(thread,
409*10465441SEvalZero                     name,
410*10465441SEvalZero                     entry,
411*10465441SEvalZero                     parameter,
412*10465441SEvalZero                     stack_start,
413*10465441SEvalZero                     stack_size,
414*10465441SEvalZero                     priority,
415*10465441SEvalZero                     tick);
416*10465441SEvalZero 
417*10465441SEvalZero     return thread;
418*10465441SEvalZero }
419*10465441SEvalZero RTM_EXPORT(rt_thread_create);
420*10465441SEvalZero 
421*10465441SEvalZero /**
422*10465441SEvalZero  * This function will delete a thread. The thread object will be removed from
423*10465441SEvalZero  * thread queue and deleted from system object management in the idle thread.
424*10465441SEvalZero  *
425*10465441SEvalZero  * @param thread the thread to be deleted
426*10465441SEvalZero  *
427*10465441SEvalZero  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
428*10465441SEvalZero  */
rt_thread_delete(rt_thread_t thread)429*10465441SEvalZero rt_err_t rt_thread_delete(rt_thread_t thread)
430*10465441SEvalZero {
431*10465441SEvalZero     rt_base_t lock;
432*10465441SEvalZero 
433*10465441SEvalZero     /* thread check */
434*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
435*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
436*10465441SEvalZero     RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread) == RT_FALSE);
437*10465441SEvalZero 
438*10465441SEvalZero     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
439*10465441SEvalZero     {
440*10465441SEvalZero         /* remove from schedule */
441*10465441SEvalZero         rt_schedule_remove_thread(thread);
442*10465441SEvalZero     }
443*10465441SEvalZero 
444*10465441SEvalZero     /* release thread timer */
445*10465441SEvalZero     rt_timer_detach(&(thread->thread_timer));
446*10465441SEvalZero 
447*10465441SEvalZero     /* change stat */
448*10465441SEvalZero     thread->stat = RT_THREAD_CLOSE;
449*10465441SEvalZero 
450*10465441SEvalZero     /* disable interrupt */
451*10465441SEvalZero     lock = rt_hw_interrupt_disable();
452*10465441SEvalZero 
453*10465441SEvalZero     /* insert to defunct thread list */
454*10465441SEvalZero     rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
455*10465441SEvalZero 
456*10465441SEvalZero     /* enable interrupt */
457*10465441SEvalZero     rt_hw_interrupt_enable(lock);
458*10465441SEvalZero 
459*10465441SEvalZero     return RT_EOK;
460*10465441SEvalZero }
461*10465441SEvalZero RTM_EXPORT(rt_thread_delete);
462*10465441SEvalZero #endif
463*10465441SEvalZero 
464*10465441SEvalZero /**
465*10465441SEvalZero  * This function will let current thread yield processor, and scheduler will
466*10465441SEvalZero  * choose a highest thread to run. After yield processor, the current thread
467*10465441SEvalZero  * is still in READY state.
468*10465441SEvalZero  *
469*10465441SEvalZero  * @return RT_EOK
470*10465441SEvalZero  */
rt_thread_yield(void)471*10465441SEvalZero rt_err_t rt_thread_yield(void)
472*10465441SEvalZero {
473*10465441SEvalZero     rt_schedule();
474*10465441SEvalZero 
475*10465441SEvalZero     return RT_EOK;
476*10465441SEvalZero }
477*10465441SEvalZero RTM_EXPORT(rt_thread_yield);
478*10465441SEvalZero 
479*10465441SEvalZero /**
480*10465441SEvalZero  * This function will let current thread sleep for some ticks.
481*10465441SEvalZero  *
482*10465441SEvalZero  * @param tick the sleep ticks
483*10465441SEvalZero  *
484*10465441SEvalZero  * @return RT_EOK
485*10465441SEvalZero  */
rt_thread_sleep(rt_tick_t tick)486*10465441SEvalZero rt_err_t rt_thread_sleep(rt_tick_t tick)
487*10465441SEvalZero {
488*10465441SEvalZero     register rt_base_t temp;
489*10465441SEvalZero     struct rt_thread *thread;
490*10465441SEvalZero 
491*10465441SEvalZero     /* set to current thread */
492*10465441SEvalZero     thread = rt_thread_self();
493*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
494*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
495*10465441SEvalZero 
496*10465441SEvalZero     /* disable interrupt */
497*10465441SEvalZero     temp = rt_hw_interrupt_disable();
498*10465441SEvalZero 
499*10465441SEvalZero     /* suspend thread */
500*10465441SEvalZero     rt_thread_suspend(thread);
501*10465441SEvalZero 
502*10465441SEvalZero     /* reset the timeout of thread timer and start it */
503*10465441SEvalZero     rt_timer_control(&(thread->thread_timer), RT_TIMER_CTRL_SET_TIME, &tick);
504*10465441SEvalZero     rt_timer_start(&(thread->thread_timer));
505*10465441SEvalZero 
506*10465441SEvalZero     /* enable interrupt */
507*10465441SEvalZero     rt_hw_interrupt_enable(temp);
508*10465441SEvalZero 
509*10465441SEvalZero     rt_schedule();
510*10465441SEvalZero 
511*10465441SEvalZero     /* clear error number of this thread to RT_EOK */
512*10465441SEvalZero     if (thread->error == -RT_ETIMEOUT)
513*10465441SEvalZero         thread->error = RT_EOK;
514*10465441SEvalZero 
515*10465441SEvalZero     return RT_EOK;
516*10465441SEvalZero }
517*10465441SEvalZero 
518*10465441SEvalZero /**
519*10465441SEvalZero  * This function will let current thread delay for some ticks.
520*10465441SEvalZero  *
521*10465441SEvalZero  * @param tick the delay ticks
522*10465441SEvalZero  *
523*10465441SEvalZero  * @return RT_EOK
524*10465441SEvalZero  */
rt_thread_delay(rt_tick_t tick)525*10465441SEvalZero rt_err_t rt_thread_delay(rt_tick_t tick)
526*10465441SEvalZero {
527*10465441SEvalZero     return rt_thread_sleep(tick);
528*10465441SEvalZero }
529*10465441SEvalZero RTM_EXPORT(rt_thread_delay);
530*10465441SEvalZero 
531*10465441SEvalZero /**
532*10465441SEvalZero  * This function will let current thread delay for some milliseconds.
533*10465441SEvalZero  *
534*10465441SEvalZero  * @param tick the delay time
535*10465441SEvalZero  *
536*10465441SEvalZero  * @return RT_EOK
537*10465441SEvalZero  */
rt_thread_mdelay(rt_int32_t ms)538*10465441SEvalZero rt_err_t rt_thread_mdelay(rt_int32_t ms)
539*10465441SEvalZero {
540*10465441SEvalZero     rt_tick_t tick;
541*10465441SEvalZero 
542*10465441SEvalZero     tick = rt_tick_from_millisecond(ms);
543*10465441SEvalZero 
544*10465441SEvalZero     return rt_thread_sleep(tick);
545*10465441SEvalZero }
546*10465441SEvalZero RTM_EXPORT(rt_thread_mdelay);
547*10465441SEvalZero 
548*10465441SEvalZero /**
549*10465441SEvalZero  * This function will control thread behaviors according to control command.
550*10465441SEvalZero  *
551*10465441SEvalZero  * @param thread the specified thread to be controlled
552*10465441SEvalZero  * @param cmd the control command, which includes
553*10465441SEvalZero  *  RT_THREAD_CTRL_CHANGE_PRIORITY for changing priority level of thread;
554*10465441SEvalZero  *  RT_THREAD_CTRL_STARTUP for starting a thread;
555*10465441SEvalZero  *  RT_THREAD_CTRL_CLOSE for delete a thread;
556*10465441SEvalZero  *  RT_THREAD_CTRL_BIND_CPU for bind the thread to a CPU.
557*10465441SEvalZero  * @param arg the argument of control command
558*10465441SEvalZero  *
559*10465441SEvalZero  * @return RT_EOK
560*10465441SEvalZero  */
rt_thread_control(rt_thread_t thread,int cmd,void * arg)561*10465441SEvalZero rt_err_t rt_thread_control(rt_thread_t thread, int cmd, void *arg)
562*10465441SEvalZero {
563*10465441SEvalZero     register rt_base_t temp;
564*10465441SEvalZero 
565*10465441SEvalZero     /* thread check */
566*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
567*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
568*10465441SEvalZero 
569*10465441SEvalZero     switch (cmd)
570*10465441SEvalZero     {
571*10465441SEvalZero     case RT_THREAD_CTRL_CHANGE_PRIORITY:
572*10465441SEvalZero         /* disable interrupt */
573*10465441SEvalZero         temp = rt_hw_interrupt_disable();
574*10465441SEvalZero 
575*10465441SEvalZero         /* for ready thread, change queue */
576*10465441SEvalZero         if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_READY)
577*10465441SEvalZero         {
578*10465441SEvalZero             /* remove thread from schedule queue first */
579*10465441SEvalZero             rt_schedule_remove_thread(thread);
580*10465441SEvalZero 
581*10465441SEvalZero             /* change thread priority */
582*10465441SEvalZero             thread->current_priority = *(rt_uint8_t *)arg;
583*10465441SEvalZero 
584*10465441SEvalZero             /* recalculate priority attribute */
585*10465441SEvalZero #if RT_THREAD_PRIORITY_MAX > 32
586*10465441SEvalZero             thread->number      = thread->current_priority >> 3;            /* 5bit */
587*10465441SEvalZero             thread->number_mask = 1 << thread->number;
588*10465441SEvalZero             thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
589*10465441SEvalZero #else
590*10465441SEvalZero             thread->number_mask = 1 << thread->current_priority;
591*10465441SEvalZero #endif
592*10465441SEvalZero 
593*10465441SEvalZero             /* insert thread to schedule queue again */
594*10465441SEvalZero             rt_schedule_insert_thread(thread);
595*10465441SEvalZero         }
596*10465441SEvalZero         else
597*10465441SEvalZero         {
598*10465441SEvalZero             thread->current_priority = *(rt_uint8_t *)arg;
599*10465441SEvalZero 
600*10465441SEvalZero             /* recalculate priority attribute */
601*10465441SEvalZero #if RT_THREAD_PRIORITY_MAX > 32
602*10465441SEvalZero             thread->number      = thread->current_priority >> 3;            /* 5bit */
603*10465441SEvalZero             thread->number_mask = 1 << thread->number;
604*10465441SEvalZero             thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
605*10465441SEvalZero #else
606*10465441SEvalZero             thread->number_mask = 1 << thread->current_priority;
607*10465441SEvalZero #endif
608*10465441SEvalZero         }
609*10465441SEvalZero 
610*10465441SEvalZero         /* enable interrupt */
611*10465441SEvalZero         rt_hw_interrupt_enable(temp);
612*10465441SEvalZero         break;
613*10465441SEvalZero 
614*10465441SEvalZero     case RT_THREAD_CTRL_STARTUP:
615*10465441SEvalZero         return rt_thread_startup(thread);
616*10465441SEvalZero 
617*10465441SEvalZero #ifdef RT_USING_HEAP
618*10465441SEvalZero     case RT_THREAD_CTRL_CLOSE:
619*10465441SEvalZero         return rt_thread_delete(thread);
620*10465441SEvalZero #endif
621*10465441SEvalZero 
622*10465441SEvalZero #ifdef RT_USING_SMP
623*10465441SEvalZero     case RT_THREAD_CTRL_BIND_CPU:
624*10465441SEvalZero     {
625*10465441SEvalZero         rt_uint8_t cpu;
626*10465441SEvalZero 
627*10465441SEvalZero         if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
628*10465441SEvalZero         {
629*10465441SEvalZero             /* we only support bind cpu before started phase. */
630*10465441SEvalZero             return RT_ERROR;
631*10465441SEvalZero         }
632*10465441SEvalZero 
633*10465441SEvalZero         cpu = (rt_uint8_t)(size_t)arg;
634*10465441SEvalZero         thread->bind_cpu = cpu > RT_CPUS_NR? RT_CPUS_NR : cpu;
635*10465441SEvalZero         break;
636*10465441SEvalZero     }
637*10465441SEvalZero #endif /*RT_USING_SMP*/
638*10465441SEvalZero     default:
639*10465441SEvalZero         break;
640*10465441SEvalZero     }
641*10465441SEvalZero 
642*10465441SEvalZero     return RT_EOK;
643*10465441SEvalZero }
644*10465441SEvalZero RTM_EXPORT(rt_thread_control);
645*10465441SEvalZero 
646*10465441SEvalZero /**
647*10465441SEvalZero  * This function will suspend the specified thread.
648*10465441SEvalZero  *
649*10465441SEvalZero  * @param thread the thread to be suspended
650*10465441SEvalZero  *
651*10465441SEvalZero  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
652*10465441SEvalZero  *
653*10465441SEvalZero  * @note if suspend self thread, after this function call, the
654*10465441SEvalZero  * rt_schedule() must be invoked.
655*10465441SEvalZero  */
rt_thread_suspend(rt_thread_t thread)656*10465441SEvalZero rt_err_t rt_thread_suspend(rt_thread_t thread)
657*10465441SEvalZero {
658*10465441SEvalZero     register rt_base_t stat;
659*10465441SEvalZero     register rt_base_t temp;
660*10465441SEvalZero 
661*10465441SEvalZero     /* thread check */
662*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
663*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
664*10465441SEvalZero 
665*10465441SEvalZero     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend:  %s\n", thread->name));
666*10465441SEvalZero 
667*10465441SEvalZero     stat = thread->stat & RT_THREAD_STAT_MASK;
668*10465441SEvalZero     if ((stat != RT_THREAD_READY) && (stat != RT_THREAD_RUNNING))
669*10465441SEvalZero     {
670*10465441SEvalZero         RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend: thread disorder, 0x%2x\n",
671*10465441SEvalZero                                        thread->stat));
672*10465441SEvalZero         return -RT_ERROR;
673*10465441SEvalZero     }
674*10465441SEvalZero 
675*10465441SEvalZero     /* disable interrupt */
676*10465441SEvalZero     temp = rt_hw_interrupt_disable();
677*10465441SEvalZero     if (stat == RT_THREAD_RUNNING)
678*10465441SEvalZero     {
679*10465441SEvalZero         /* not suspend running status thread on other core */
680*10465441SEvalZero         RT_ASSERT(thread == rt_thread_self());
681*10465441SEvalZero     }
682*10465441SEvalZero 
683*10465441SEvalZero     /* change thread stat */
684*10465441SEvalZero     rt_schedule_remove_thread(thread);
685*10465441SEvalZero     thread->stat = RT_THREAD_SUSPEND | (thread->stat & ~RT_THREAD_STAT_MASK);
686*10465441SEvalZero 
687*10465441SEvalZero     /* stop thread timer anyway */
688*10465441SEvalZero     rt_timer_stop(&(thread->thread_timer));
689*10465441SEvalZero 
690*10465441SEvalZero     /* enable interrupt */
691*10465441SEvalZero     rt_hw_interrupt_enable(temp);
692*10465441SEvalZero 
693*10465441SEvalZero     RT_OBJECT_HOOK_CALL(rt_thread_suspend_hook, (thread));
694*10465441SEvalZero     return RT_EOK;
695*10465441SEvalZero }
696*10465441SEvalZero RTM_EXPORT(rt_thread_suspend);
697*10465441SEvalZero 
698*10465441SEvalZero /**
699*10465441SEvalZero  * This function will resume a thread and put it to system ready queue.
700*10465441SEvalZero  *
701*10465441SEvalZero  * @param thread the thread to be resumed
702*10465441SEvalZero  *
703*10465441SEvalZero  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
704*10465441SEvalZero  */
rt_thread_resume(rt_thread_t thread)705*10465441SEvalZero rt_err_t rt_thread_resume(rt_thread_t thread)
706*10465441SEvalZero {
707*10465441SEvalZero     register rt_base_t temp;
708*10465441SEvalZero 
709*10465441SEvalZero     /* thread check */
710*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
711*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
712*10465441SEvalZero 
713*10465441SEvalZero     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume:  %s\n", thread->name));
714*10465441SEvalZero 
715*10465441SEvalZero     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_SUSPEND)
716*10465441SEvalZero     {
717*10465441SEvalZero         RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume: thread disorder, %d\n",
718*10465441SEvalZero                                        thread->stat));
719*10465441SEvalZero 
720*10465441SEvalZero         return -RT_ERROR;
721*10465441SEvalZero     }
722*10465441SEvalZero 
723*10465441SEvalZero     /* disable interrupt */
724*10465441SEvalZero     temp = rt_hw_interrupt_disable();
725*10465441SEvalZero 
726*10465441SEvalZero     /* remove from suspend list */
727*10465441SEvalZero     rt_list_remove(&(thread->tlist));
728*10465441SEvalZero 
729*10465441SEvalZero     rt_timer_stop(&thread->thread_timer);
730*10465441SEvalZero 
731*10465441SEvalZero     /* enable interrupt */
732*10465441SEvalZero     rt_hw_interrupt_enable(temp);
733*10465441SEvalZero 
734*10465441SEvalZero     /* insert to schedule ready list */
735*10465441SEvalZero     rt_schedule_insert_thread(thread);
736*10465441SEvalZero 
737*10465441SEvalZero     RT_OBJECT_HOOK_CALL(rt_thread_resume_hook, (thread));
738*10465441SEvalZero     return RT_EOK;
739*10465441SEvalZero }
740*10465441SEvalZero RTM_EXPORT(rt_thread_resume);
741*10465441SEvalZero 
742*10465441SEvalZero /**
743*10465441SEvalZero  * This function is the timeout function for thread, normally which is invoked
744*10465441SEvalZero  * when thread is timeout to wait some resource.
745*10465441SEvalZero  *
746*10465441SEvalZero  * @param parameter the parameter of thread timeout function
747*10465441SEvalZero  */
rt_thread_timeout(void * parameter)748*10465441SEvalZero void rt_thread_timeout(void *parameter)
749*10465441SEvalZero {
750*10465441SEvalZero     struct rt_thread *thread;
751*10465441SEvalZero 
752*10465441SEvalZero     thread = (struct rt_thread *)parameter;
753*10465441SEvalZero 
754*10465441SEvalZero     /* thread check */
755*10465441SEvalZero     RT_ASSERT(thread != RT_NULL);
756*10465441SEvalZero     RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_SUSPEND);
757*10465441SEvalZero     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
758*10465441SEvalZero 
759*10465441SEvalZero     /* set error number */
760*10465441SEvalZero     thread->error = -RT_ETIMEOUT;
761*10465441SEvalZero 
762*10465441SEvalZero     /* remove from suspend list */
763*10465441SEvalZero     rt_list_remove(&(thread->tlist));
764*10465441SEvalZero 
765*10465441SEvalZero     /* insert to schedule ready list */
766*10465441SEvalZero     rt_schedule_insert_thread(thread);
767*10465441SEvalZero 
768*10465441SEvalZero     /* do schedule */
769*10465441SEvalZero     rt_schedule();
770*10465441SEvalZero }
771*10465441SEvalZero RTM_EXPORT(rt_thread_timeout);
772*10465441SEvalZero 
773*10465441SEvalZero /**
774*10465441SEvalZero  * This function will find the specified thread.
775*10465441SEvalZero  *
776*10465441SEvalZero  * @param name the name of thread finding
777*10465441SEvalZero  *
778*10465441SEvalZero  * @return the found thread
779*10465441SEvalZero  *
780*10465441SEvalZero  * @note please don't invoke this function in interrupt status.
781*10465441SEvalZero  */
rt_thread_find(char * name)782*10465441SEvalZero rt_thread_t rt_thread_find(char *name)
783*10465441SEvalZero {
784*10465441SEvalZero     struct rt_object_information *information;
785*10465441SEvalZero     struct rt_object *object;
786*10465441SEvalZero     struct rt_list_node *node;
787*10465441SEvalZero 
788*10465441SEvalZero     /* enter critical */
789*10465441SEvalZero     if (rt_thread_self() != RT_NULL)
790*10465441SEvalZero         rt_enter_critical();
791*10465441SEvalZero 
792*10465441SEvalZero     /* try to find device object */
793*10465441SEvalZero     information = rt_object_get_information(RT_Object_Class_Thread);
794*10465441SEvalZero     RT_ASSERT(information != RT_NULL);
795*10465441SEvalZero     for (node  = information->object_list.next;
796*10465441SEvalZero          node != &(information->object_list);
797*10465441SEvalZero          node  = node->next)
798*10465441SEvalZero     {
799*10465441SEvalZero         object = rt_list_entry(node, struct rt_object, list);
800*10465441SEvalZero         if (rt_strncmp(object->name, name, RT_NAME_MAX) == 0)
801*10465441SEvalZero         {
802*10465441SEvalZero             /* leave critical */
803*10465441SEvalZero             if (rt_thread_self() != RT_NULL)
804*10465441SEvalZero                 rt_exit_critical();
805*10465441SEvalZero 
806*10465441SEvalZero             return (rt_thread_t)object;
807*10465441SEvalZero         }
808*10465441SEvalZero     }
809*10465441SEvalZero 
810*10465441SEvalZero     /* leave critical */
811*10465441SEvalZero     if (rt_thread_self() != RT_NULL)
812*10465441SEvalZero         rt_exit_critical();
813*10465441SEvalZero 
814*10465441SEvalZero     /* not found */
815*10465441SEvalZero     return RT_NULL;
816*10465441SEvalZero }
817*10465441SEvalZero RTM_EXPORT(rt_thread_find);
818*10465441SEvalZero 
819*10465441SEvalZero /**@}*/
820