thread.c 22.5 KB
Newer Older
1 2 3
/*
 * File      : thread.c
 * This file is part of RT-Thread RTOS
D
dzzxzz 已提交
4
 * COPYRIGHT (C) 2006 - 2012, RT-Thread Development Team
5
 *
B
Bernard Xiong 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18
 *  This program 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 2 of the License, or
 *  (at your option) any later version.
 *
 *  This program 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, write to the Free Software Foundation, Inc.,
 *  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 20 21 22 23
 *
 * Change Logs:
 * Date           Author       Notes
 * 2006-03-28     Bernard      first version
 * 2006-04-29     Bernard      implement thread timer
24 25 26 27 28
 * 2006-04-30     Bernard      added THREAD_DEBUG
 * 2006-05-27     Bernard      fixed the rt_thread_yield bug
 * 2006-06-03     Bernard      fixed the thread timer init bug
 * 2006-08-10     Bernard      fixed the timer bug in thread_sleep
 * 2006-09-03     Bernard      changed rt_timer_delete to rt_timer_detach
29
 * 2006-09-03     Bernard      implement rt_thread_detach
30
 * 2008-02-16     Bernard      fixed the rt_thread_timeout bug
31 32
 * 2010-03-21     Bernard      change the errno of rt_thread_delay/sleep to
 *                             RT_EOK.
33
 * 2010-11-10     Bernard      add cleanup callback function in thread exit.
34 35
 * 2011-09-01     Bernard      fixed rt_thread_exit issue when the current
 *                             thread preempted, which reported by Jiaxing Lee.
36
 * 2011-09-08     Bernard      fixed the scheduling issue in rt_thread_startup.
37
 * 2012-12-29     Bernard      fixed compiling warning.
38
 * 2016-08-09     ArdaFu       add thread suspend and resume hook.
39 40
 * 2017-04-10     armink       fixed the rt_thread_delete and rt_thread_detach
                               bug when thread has not startup.
41 42 43 44 45 46
 */

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

extern rt_list_t rt_thread_priority_table[RT_THREAD_PRIORITY_MAX];
D
dzzxzz 已提交
47
extern struct rt_thread *rt_current_thread;
48 49
extern rt_list_t rt_thread_defunct;

50 51 52
#ifdef RT_USING_HOOK

static void (*rt_thread_suspend_hook)(rt_thread_t thread);
B
Bernard Xiong 已提交
53 54 55
static void (*rt_thread_resume_hook) (rt_thread_t thread);
static void (*rt_thread_inited_hook) (rt_thread_t thread);

56 57
/**
 * @ingroup Hook
58
 * This function sets a hook function when the system suspend a thread.
59 60 61 62 63 64 65 66 67
 *
 * @param hook the specified hook function
 *
 * @note the hook function must be simple and never be blocked or suspend.
 */
void rt_thread_suspend_sethook(void (*hook)(rt_thread_t thread))
{
    rt_thread_suspend_hook = hook;
}
B
Bernard Xiong 已提交
68

69 70
/**
 * @ingroup Hook
71
 * This function sets a hook function when the system resume a thread.
72 73 74 75 76 77 78 79 80
 *
 * @param hook the specified hook function
 *
 * @note the hook function must be simple and never be blocked or suspend.
 */
void rt_thread_resume_sethook(void (*hook)(rt_thread_t thread))
{
    rt_thread_resume_hook = hook;
}
B
Bernard Xiong 已提交
81 82 83

/**
 * @ingroup Hook
84
 * This function sets a hook function when a thread is initialized.
B
Bernard Xiong 已提交
85 86 87 88 89
 *
 * @param hook the specified hook function
 */
void rt_thread_inited_sethook(void (*hook)(rt_thread_t thread))
{
90
    rt_thread_inited_hook = hook;
B
Bernard Xiong 已提交
91 92
}

93 94
#endif

95
void rt_thread_exit(void)
96
{
97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
    struct rt_thread *thread;
    register rt_base_t level;

    /* get current thread */
    thread = rt_current_thread;

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

    /* remove from schedule */
    rt_schedule_remove_thread(thread);
    /* change stat */
    thread->stat = RT_THREAD_CLOSE;

    /* remove it from timer list */
112
    rt_timer_detach(&thread->thread_timer);
113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129

    if ((rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE) &&
        thread->cleanup == RT_NULL)
    {
        rt_object_detach((rt_object_t)thread);
    }
    else
    {
        /* insert to defunct thread list */
        rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
    }

    /* enable interrupt */
    rt_hw_interrupt_enable(level);

    /* switch to next task */
    rt_schedule();
130
}
131

D
dzzxzz 已提交
132
static rt_err_t _rt_thread_init(struct rt_thread *thread,
133 134 135 136 137 138 139
                                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)
140
{
141 142 143 144 145 146 147 148
    /* init thread list */
    rt_list_init(&(thread->tlist));

    thread->entry = (void *)entry;
    thread->parameter = parameter;

    /* stack init */
    thread->stack_addr = stack_start;
W
weety 已提交
149
    thread->stack_size = stack_size;
150 151 152 153

    /* init thread stack */
    rt_memset(thread->stack_addr, '#', thread->stack_size);
    thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter,
154 155
                                          (void *)((char *)thread->stack_addr + thread->stack_size - 4),
                                          (void *)rt_thread_exit);
156 157 158 159 160 161

    /* priority init */
    RT_ASSERT(priority < RT_THREAD_PRIORITY_MAX);
    thread->init_priority    = priority;
    thread->current_priority = priority;

162 163 164
    thread->number_mask = 0;
#if RT_THREAD_PRIORITY_MAX > 32
    thread->number = 0;
B
Bernard Xiong 已提交
165
    thread->high_mask = 0;
166 167
#endif

168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187
    /* tick init */
    thread->init_tick      = tick;
    thread->remaining_tick = tick;

    /* error and flags */
    thread->error = RT_EOK;
    thread->stat  = RT_THREAD_INIT;

    /* initialize cleanup function and user data */
    thread->cleanup   = 0;
    thread->user_data = 0;

    /* init thread timer */
    rt_timer_init(&(thread->thread_timer),
                  thread->name,
                  rt_thread_timeout,
                  thread,
                  0,
                  RT_TIMER_FLAG_ONE_SHOT);

W
weety 已提交
188 189 190 191
#ifdef RT_USING_MODULE
    thread->module_id = RT_NULL;
#endif

B
bernard 已提交
192 193 194 195 196 197 198 199 200 201
    /* initialize signal */
#ifdef RT_USING_SIGNALS
    thread->sig_mask    = 0x00;
    thread->sig_pending = 0x00;

    thread->sig_ret     = RT_NULL;
    thread->sig_vectors = RT_NULL;
    thread->si_list     = RT_NULL;
#endif

202
    RT_OBJECT_HOOK_CALL(rt_thread_inited_hook, (thread));
B
Bernard Xiong 已提交
203

204
    return RT_EOK;
205 206 207 208 209 210
}

/**
 * @addtogroup Thread
 */

D
dogandog 已提交
211
/**@{*/
212 213

/**
B
bernard.xiong@gmail.com 已提交
214 215
 * This function will initialize a thread, normally it's used to initialize a
 * static thread object.
216 217 218 219 220 221 222 223 224 225
 *
 * @param thread the static thread object
 * @param name the name of thread, which shall be unique
 * @param entry the entry function of thread
 * @param parameter the parameter of thread enter function
 * @param stack_start the start address of thread stack
 * @param stack_size the size of thread stack
 * @param priority the priority of thread
 * @param tick the time slice if there are same priority thread
 *
B
bernard.xiong@gmail.com 已提交
226
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
227
 */
D
dzzxzz 已提交
228
rt_err_t rt_thread_init(struct rt_thread *thread,
229 230 231 232 233 234 235
                        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)
236
{
237 238 239 240 241 242 243 244 245 246 247 248 249 250 251
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
    RT_ASSERT(stack_start != RT_NULL);

    /* init thread object */
    rt_object_init((rt_object_t)thread, RT_Object_Class_Thread, name);

    return _rt_thread_init(thread,
                           name,
                           entry,
                           parameter,
                           stack_start,
                           stack_size,
                           priority,
                           tick);
252
}
253
RTM_EXPORT(rt_thread_init);
254 255 256 257 258 259

/**
 * This function will return self thread object
 *
 * @return the self thread object
 */
D
dzzxzz 已提交
260
rt_thread_t rt_thread_self(void)
261
{
262
    return rt_current_thread;
263
}
264
RTM_EXPORT(rt_thread_self);
265 266 267 268 269 270

/**
 * This function will start a thread and put it to system ready queue
 *
 * @param thread the thread to be started
 *
B
bernard.xiong@gmail.com 已提交
271
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
272
 */
D
dzzxzz 已提交
273
rt_err_t rt_thread_startup(rt_thread_t thread)
274
{
275 276
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
B
bernard 已提交
277
    RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_INIT);
278
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
279

280 281
    /* set current priority to init priority */
    thread->current_priority = thread->init_priority;
282

283
    /* calculate priority attribute */
284
#if RT_THREAD_PRIORITY_MAX > 32
285 286 287
    thread->number      = thread->current_priority >> 3;            /* 5bit */
    thread->number_mask = 1L << thread->number;
    thread->high_mask   = 1L << (thread->current_priority & 0x07);  /* 3bit */
288
#else
289
    thread->number_mask = 1L << thread->current_priority;
290 291
#endif

292
    RT_DEBUG_LOG(RT_DEBUG_THREAD, ("startup a thread:%s with priority:%d\n",
293
                                   thread->name, thread->init_priority));
294 295 296 297 298 299 300 301 302 303 304
    /* change thread stat */
    thread->stat = RT_THREAD_SUSPEND;
    /* then resume it */
    rt_thread_resume(thread);
    if (rt_thread_self() != RT_NULL)
    {
        /* do a scheduling */
        rt_schedule();
    }

    return RT_EOK;
305
}
306
RTM_EXPORT(rt_thread_startup);
307 308

/**
B
bernard.xiong@gmail.com 已提交
309 310
 * This function will detach a thread. The thread object will be removed from
 * thread queue and detached/deleted from system object management.
311 312 313
 *
 * @param thread the thread to be deleted
 *
B
bernard.xiong@gmail.com 已提交
314
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
315
 */
D
dzzxzz 已提交
316
rt_err_t rt_thread_detach(rt_thread_t thread)
317
{
318
    rt_base_t lock;
319

320 321
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
322 323
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
    RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread));
324

B
bernard 已提交
325
    if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
326 327 328 329
    {
        /* remove from schedule */
        rt_schedule_remove_thread(thread);
    }
330

331 332
    /* release thread timer */
    rt_timer_detach(&(thread->thread_timer));
333

334 335
    /* change stat */
    thread->stat = RT_THREAD_CLOSE;
336

337 338
    /* detach object */
    rt_object_detach((rt_object_t)thread);
339

340 341 342 343
    if (thread->cleanup != RT_NULL)
    {
        /* disable interrupt */
        lock = rt_hw_interrupt_disable();
344

345 346
        /* insert to defunct thread list */
        rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
347

348 349 350
        /* enable interrupt */
        rt_hw_interrupt_enable(lock);
    }
351

352
    return RT_EOK;
353
}
354 355
RTM_EXPORT(rt_thread_detach);

356 357

#ifdef RT_USING_HEAP
358 359 360 361 362 363 364 365 366 367 368 369 370 371
/**
 * This function will create a thread object and allocate thread object memory
 * and stack.
 *
 * @param name the name of thread, which shall be unique
 * @param entry the entry function of thread
 * @param parameter the parameter of thread enter function
 * @param stack_size the size of thread stack
 * @param priority the priority of thread
 * @param tick the time slice if there are same priority thread
 *
 * @return the created thread object
 */
rt_thread_t rt_thread_create(const char *name,
372 373 374 375 376
                             void (*entry)(void *parameter),
                             void       *parameter,
                             rt_uint32_t stack_size,
                             rt_uint8_t  priority,
                             rt_uint32_t tick)
377
{
378 379 380 381 382 383 384 385
    struct rt_thread *thread;
    void *stack_start;

    thread = (struct rt_thread *)rt_object_allocate(RT_Object_Class_Thread,
                                                    name);
    if (thread == RT_NULL)
        return RT_NULL;

B
Bernard Xiong 已提交
386
    stack_start = (void *)RT_KERNEL_MALLOC(stack_size);
387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404
    if (stack_start == RT_NULL)
    {
        /* allocate stack failure */
        rt_object_delete((rt_object_t)thread);

        return RT_NULL;
    }

    _rt_thread_init(thread,
                    name,
                    entry,
                    parameter,
                    stack_start,
                    stack_size,
                    priority,
                    tick);

    return thread;
405
}
406
RTM_EXPORT(rt_thread_create);
407

408
/**
B
bernard.xiong@gmail.com 已提交
409
 * This function will delete a thread. The thread object will be removed from
410
 * thread queue and deleted from system object management in the idle thread.
411 412 413
 *
 * @param thread the thread to be deleted
 *
B
bernard.xiong@gmail.com 已提交
414
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
415
 */
D
dzzxzz 已提交
416
rt_err_t rt_thread_delete(rt_thread_t thread)
417
{
418
    rt_base_t lock;
419

420 421
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
422 423
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
    RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread) == RT_FALSE);
424

B
bernard 已提交
425
    if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
426 427 428 429
    {
        /* remove from schedule */
        rt_schedule_remove_thread(thread);
    }
430

431 432
    /* release thread timer */
    rt_timer_detach(&(thread->thread_timer));
433

434 435
    /* change stat */
    thread->stat = RT_THREAD_CLOSE;
436

437 438
    /* disable interrupt */
    lock = rt_hw_interrupt_disable();
439

440 441
    /* insert to defunct thread list */
    rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
442

443 444
    /* enable interrupt */
    rt_hw_interrupt_enable(lock);
445

446
    return RT_EOK;
447
}
448
RTM_EXPORT(rt_thread_delete);
449 450 451
#endif

/**
B
bernard.xiong@gmail.com 已提交
452 453 454
 * This function will let current thread yield processor, and scheduler will
 * choose a highest thread to run. After yield processor, the current thread
 * is still in READY state.
455
 *
B
bernard.xiong@gmail.com 已提交
456
 * @return RT_EOK
457
 */
D
dzzxzz 已提交
458
rt_err_t rt_thread_yield(void)
459
{
460 461
    register rt_base_t level;
    struct rt_thread *thread;
462

463 464
    /* disable interrupt */
    level = rt_hw_interrupt_disable();
465

466 467
    /* set to current thread */
    thread = rt_current_thread;
468

469
    /* if the thread stat is READY and on ready queue list */
B
bernard 已提交
470
    if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_READY &&
471 472 473 474
        thread->tlist.next != thread->tlist.prev)
    {
        /* remove thread from thread list */
        rt_list_remove(&(thread->tlist));
475

476 477 478
        /* put thread to end of ready queue */
        rt_list_insert_before(&(rt_thread_priority_table[thread->current_priority]),
                              &(thread->tlist));
479

480 481
        /* enable interrupt */
        rt_hw_interrupt_enable(level);
482

483
        rt_schedule();
484

485 486
        return RT_EOK;
    }
487

488 489
    /* enable interrupt */
    rt_hw_interrupt_enable(level);
490

491
    return RT_EOK;
492
}
493
RTM_EXPORT(rt_thread_yield);
494 495 496 497 498 499

/**
 * This function will let current thread sleep for some ticks.
 *
 * @param tick the sleep ticks
 *
B
bernard.xiong@gmail.com 已提交
500
 * @return RT_EOK
501
 */
D
dzzxzz 已提交
502
rt_err_t rt_thread_sleep(rt_tick_t tick)
503
{
504 505
    register rt_base_t temp;
    struct rt_thread *thread;
506

507 508 509 510 511
    /* disable interrupt */
    temp = rt_hw_interrupt_disable();
    /* set to current thread */
    thread = rt_current_thread;
    RT_ASSERT(thread != RT_NULL);
512
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
513

514 515
    /* suspend thread */
    rt_thread_suspend(thread);
516

517 518 519
    /* reset the timeout of thread timer and start it */
    rt_timer_control(&(thread->thread_timer), RT_TIMER_CTRL_SET_TIME, &tick);
    rt_timer_start(&(thread->thread_timer));
520

521 522
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);
523

524
    rt_schedule();
525

526 527 528
    /* clear error number of this thread to RT_EOK */
    if (thread->error == -RT_ETIMEOUT)
        thread->error = RT_EOK;
529

530
    return RT_EOK;
531 532 533 534 535 536 537
}

/**
 * This function will let current thread delay for some ticks.
 *
 * @param tick the delay ticks
 *
B
bernard.xiong@gmail.com 已提交
538
 * @return RT_EOK
539 540 541
 */
rt_err_t rt_thread_delay(rt_tick_t tick)
{
542
    return rt_thread_sleep(tick);
543
}
544
RTM_EXPORT(rt_thread_delay);
545

H
heyuanjie 已提交
546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562
/**
 * This function will let current thread delay for some milliseconds.
 *
 * @param tick the delay time
 *
 * @return RT_EOK
 */
rt_err_t rt_thread_mdelay(rt_int32_t ms)
{
    rt_tick_t tick;

    tick = rt_tick_from_millisecond(ms);

    return rt_thread_sleep(tick);
}
RTM_EXPORT(rt_thread_mdelay);

B
bernard.xiong@gmail.com 已提交
563 564 565 566 567
/**
 * This function will control thread behaviors according to control command.
 *
 * @param thread the specified thread to be controlled
 * @param cmd the control command, which includes
568
 *  RT_THREAD_CTRL_CHANGE_PRIORITY for changing priority level of thread;
B
bernard.xiong@gmail.com 已提交
569 570 571 572 573 574
 *  RT_THREAD_CTRL_STARTUP for starting a thread;
 *  RT_THREAD_CTRL_CLOSE for delete a thread.
 * @param arg the argument of control command
 *
 * @return RT_EOK
 */
B
bernard 已提交
575
rt_err_t rt_thread_control(rt_thread_t thread, int cmd, void *arg)
576
{
577
    register rt_base_t temp;
578

579 580
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
581
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
582

583 584 585 586 587
    switch (cmd)
    {
    case RT_THREAD_CTRL_CHANGE_PRIORITY:
        /* disable interrupt */
        temp = rt_hw_interrupt_disable();
588

589
        /* for ready thread, change queue */
B
bernard 已提交
590
        if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_READY)
591 592 593
        {
            /* remove thread from schedule queue first */
            rt_schedule_remove_thread(thread);
594

595 596
            /* change thread priority */
            thread->current_priority = *(rt_uint8_t *)arg;
597

598
            /* recalculate priority attribute */
599
#if RT_THREAD_PRIORITY_MAX > 32
600 601 602
            thread->number      = thread->current_priority >> 3;            /* 5bit */
            thread->number_mask = 1 << thread->number;
            thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
603
#else
604
            thread->number_mask = 1 << thread->current_priority;
605 606
#endif

607 608 609 610 611 612
            /* insert thread to schedule queue again */
            rt_schedule_insert_thread(thread);
        }
        else
        {
            thread->current_priority = *(rt_uint8_t *)arg;
613

614
            /* recalculate priority attribute */
615
#if RT_THREAD_PRIORITY_MAX > 32
616 617 618
            thread->number      = thread->current_priority >> 3;            /* 5bit */
            thread->number_mask = 1 << thread->number;
            thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
619
#else
620
            thread->number_mask = 1 << thread->current_priority;
621
#endif
622
        }
623

624 625 626
        /* enable interrupt */
        rt_hw_interrupt_enable(temp);
        break;
627

628 629
    case RT_THREAD_CTRL_STARTUP:
        return rt_thread_startup(thread);
630 631

#ifdef RT_USING_HEAP
632 633
    case RT_THREAD_CTRL_CLOSE:
        return rt_thread_delete(thread);
634 635
#endif

636 637 638
    default:
        break;
    }
639

640
    return RT_EOK;
641
}
642
RTM_EXPORT(rt_thread_control);
643 644 645 646 647 648

/**
 * This function will suspend the specified thread.
 *
 * @param thread the thread to be suspended
 *
B
bernard.xiong@gmail.com 已提交
649
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
650
 *
B
bernard.xiong@gmail.com 已提交
651 652
 * @note if suspend self thread, after this function call, the
 * rt_schedule() must be invoked.
653
 */
D
dzzxzz 已提交
654
rt_err_t rt_thread_suspend(rt_thread_t thread)
655
{
656
    register rt_base_t temp;
657

658 659
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
660
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
661

662
    RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend:  %s\n", thread->name));
663

B
bernard 已提交
664
    if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_READY)
665
    {
666
        RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend: thread disorder, 0x%2x\n",
667
                                       thread->stat));
B
Bernard Xiong 已提交
668

669 670
        return -RT_ERROR;
    }
671

672 673
    /* disable interrupt */
    temp = rt_hw_interrupt_disable();
674

675
    /* change thread stat */
676
    thread->stat = RT_THREAD_SUSPEND | (thread->stat & ~RT_THREAD_STAT_MASK);
677
    rt_schedule_remove_thread(thread);
678

679 680 681
    /* stop thread timer anyway */
    rt_timer_stop(&(thread->thread_timer));

682 683
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);
684

685
    RT_OBJECT_HOOK_CALL(rt_thread_suspend_hook, (thread));
686
    return RT_EOK;
687
}
688
RTM_EXPORT(rt_thread_suspend);
689 690 691 692 693 694

/**
 * This function will resume a thread and put it to system ready queue.
 *
 * @param thread the thread to be resumed
 *
B
bernard.xiong@gmail.com 已提交
695
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
696
 */
D
dzzxzz 已提交
697
rt_err_t rt_thread_resume(rt_thread_t thread)
698
{
699
    register rt_base_t temp;
700

701 702
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
703
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
704

705
    RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume:  %s\n", thread->name));
706

B
bernard 已提交
707
    if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_SUSPEND)
708 709
    {
        RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume: thread disorder, %d\n",
710
                                       thread->stat));
711

712 713
        return -RT_ERROR;
    }
714

715 716
    /* disable interrupt */
    temp = rt_hw_interrupt_disable();
717

718 719
    /* remove from suspend list */
    rt_list_remove(&(thread->tlist));
720

721
    rt_timer_stop(&thread->thread_timer);
722

723 724
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);
725

726 727
    /* insert to schedule ready list */
    rt_schedule_insert_thread(thread);
728

729
    RT_OBJECT_HOOK_CALL(rt_thread_resume_hook, (thread));
730
    return RT_EOK;
731
}
732
RTM_EXPORT(rt_thread_resume);
733 734

/**
B
bernard.xiong@gmail.com 已提交
735
 * This function is the timeout function for thread, normally which is invoked
736
 * when thread is timeout to wait some resource.
737 738 739
 *
 * @param parameter the parameter of thread timeout function
 */
D
dzzxzz 已提交
740
void rt_thread_timeout(void *parameter)
741
{
742
    struct rt_thread *thread;
743

744
    thread = (struct rt_thread *)parameter;
745

746 747
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
B
bernard 已提交
748
    RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_SUSPEND);
749
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
750

751 752
    /* set error number */
    thread->error = -RT_ETIMEOUT;
753

754 755
    /* remove from suspend list */
    rt_list_remove(&(thread->tlist));
756

757 758
    /* insert to schedule ready list */
    rt_schedule_insert_thread(thread);
759

760 761
    /* do schedule */
    rt_schedule();
762
}
763
RTM_EXPORT(rt_thread_timeout);
764 765 766 767 768 769

/**
 * This function will find the specified thread.
 *
 * @param name the name of thread finding
 *
B
bernard.xiong@gmail.com 已提交
770 771 772
 * @return the found thread
 *
 * @note please don't invoke this function in interrupt status.
773
 */
D
dzzxzz 已提交
774
rt_thread_t rt_thread_find(char *name)
775
{
776 777 778 779 780 781 782 783 784
    struct rt_object_information *information;
    struct rt_object *object;
    struct rt_list_node *node;

    /* enter critical */
    if (rt_thread_self() != RT_NULL)
        rt_enter_critical();

    /* try to find device object */
785 786
    information = rt_object_get_information(RT_Object_Class_Thread);
    RT_ASSERT(information != RT_NULL);
787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807
    for (node  = information->object_list.next;
         node != &(information->object_list);
         node  = node->next)
    {
        object = rt_list_entry(node, struct rt_object, list);
        if (rt_strncmp(object->name, name, RT_NAME_MAX) == 0)
        {
            /* leave critical */
            if (rt_thread_self() != RT_NULL)
                rt_exit_critical();

            return (rt_thread_t)object;
        }
    }

    /* leave critical */
    if (rt_thread_self() != RT_NULL)
        rt_exit_critical();

    /* not found */
    return RT_NULL;
808
}
809
RTM_EXPORT(rt_thread_find);
810

D
dogandog 已提交
811
/**@}*/