thread.c 25.4 KB
Newer Older
1
/*
mysterywolf's avatar
mysterywolf 已提交
2
 * Copyright (c) 2006-2021, RT-Thread Development Team
B
Bernard Xiong 已提交
3
 *
4
 * SPDX-License-Identifier: Apache-2.0
5 6 7 8 9
 *
 * Change Logs:
 * Date           Author       Notes
 * 2006-03-28     Bernard      first version
 * 2006-04-29     Bernard      implement thread timer
10 11 12 13 14
 * 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
15
 * 2006-09-03     Bernard      implement rt_thread_detach
16
 * 2008-02-16     Bernard      fixed the rt_thread_timeout bug
17 18
 * 2010-03-21     Bernard      change the errno of rt_thread_delay/sleep to
 *                             RT_EOK.
19
 * 2010-11-10     Bernard      add cleanup callback function in thread exit.
20 21
 * 2011-09-01     Bernard      fixed rt_thread_exit issue when the current
 *                             thread preempted, which reported by Jiaxing Lee.
22
 * 2011-09-08     Bernard      fixed the scheduling issue in rt_thread_startup.
23
 * 2012-12-29     Bernard      fixed compiling warning.
24
 * 2016-08-09     ArdaFu       add thread suspend and resume hook.
25
 * 2017-04-10     armink       fixed the rt_thread_delete and rt_thread_detach
S
shaojinchun 已提交
26 27 28
 *                             bug when thread has not startup.
 * 2018-11-22     Jesven       yield is same to rt_schedule
 *                             add support for tasks bound to cpu
29
 * 2021-02-24     Meco Man     rearrange rt_thread_control() - schedule the thread when close it
30 31 32
 */

#include <rthw.h>
S
shaojinchun 已提交
33
#include <rtthread.h>
34

35 36
#ifdef RT_USING_HOOK
static void (*rt_thread_suspend_hook)(rt_thread_t thread);
B
Bernard Xiong 已提交
37 38 39
static void (*rt_thread_resume_hook) (rt_thread_t thread);
static void (*rt_thread_inited_hook) (rt_thread_t thread);

40 41
/**
 * @ingroup Hook
42
 * This function sets a hook function when the system suspend a thread.
43 44 45 46 47 48 49 50 51
 *
 * @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 已提交
52

53 54
/**
 * @ingroup Hook
55
 * This function sets a hook function when the system resume a thread.
56 57 58 59 60 61 62 63 64
 *
 * @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 已提交
65 66 67

/**
 * @ingroup Hook
68
 * This function sets a hook function when a thread is initialized.
B
Bernard Xiong 已提交
69 70 71 72 73
 *
 * @param hook the specified hook function
 */
void rt_thread_inited_sethook(void (*hook)(rt_thread_t thread))
{
74
    rt_thread_inited_hook = hook;
B
Bernard Xiong 已提交
75 76
}

77
#endif /* RT_USING_HOOK */
78

79
/* must be invoke witch rt_hw_interrupt_disable */
80
static void _rt_thread_cleanup_execute(rt_thread_t thread)
81 82 83 84
{
    register rt_base_t level;
#ifdef RT_USING_MODULE
    struct rt_dlmodule *module = RT_NULL;
85
#endif /* RT_USING_MODULE */
86 87 88 89 90 91 92
    level = rt_hw_interrupt_disable();
#ifdef RT_USING_MODULE
    module = (struct rt_dlmodule*)thread->module_id;
    if (module)
    {
        dlmodule_destroy(module);
    }
93
#endif /* RT_USING_MODULE */
94 95 96 97 98 99
    /* invoke thread cleanup */
    if (thread->cleanup != RT_NULL)
        thread->cleanup(thread);

#ifdef RT_USING_SIGNALS
    rt_thread_free_sig(thread);
100
#endif /* RT_USING_SIGNALS */
101 102 103
    rt_hw_interrupt_enable(level);
}

104
static void _rt_thread_exit(void)
105
{
106 107 108 109
    struct rt_thread *thread;
    register rt_base_t level;

    /* get current thread */
S
shaojinchun 已提交
110
    thread = rt_thread_self();
111 112 113 114

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

115
    _rt_thread_cleanup_execute(thread);
116

117 118 119 120 121 122
    /* remove from schedule */
    rt_schedule_remove_thread(thread);
    /* change stat */
    thread->stat = RT_THREAD_CLOSE;

    /* remove it from timer list */
123
    rt_timer_detach(&thread->thread_timer);
124

125
    if (rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE)
126 127 128 129 130 131
    {
        rt_object_detach((rt_object_t)thread);
    }
    else
    {
        /* insert to defunct thread list */
F
fenghuijie 已提交
132
        rt_thread_defunct_enqueue(thread);
133 134 135 136
    }

    /* switch to next task */
    rt_schedule();
137 138 139

    /* enable interrupt */
    rt_hw_interrupt_enable(level);
140
}
141

D
dzzxzz 已提交
142
static rt_err_t _rt_thread_init(struct rt_thread *thread,
143 144 145 146 147 148 149
                                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)
150
{
151 152 153 154 155 156 157 158
    /* init thread list */
    rt_list_init(&(thread->tlist));

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

    /* stack init */
    thread->stack_addr = stack_start;
W
weety 已提交
159
    thread->stack_size = stack_size;
160 161 162

    /* init thread stack */
    rt_memset(thread->stack_addr, '#', thread->stack_size);
163 164 165
#ifdef ARCH_CPU_STACK_GROWS_UPWARD
    thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter,
                                          (void *)((char *)thread->stack_addr),
166
                                          (void *)_rt_thread_exit);
167
#else
168
    thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter,
169
                                          (rt_uint8_t *)((char *)thread->stack_addr + thread->stack_size - sizeof(rt_ubase_t)),
170
                                          (void *)_rt_thread_exit);
171
#endif /* ARCH_CPU_STACK_GROWS_UPWARD */
172 173 174 175 176 177

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

178 179 180
    thread->number_mask = 0;
#if RT_THREAD_PRIORITY_MAX > 32
    thread->number = 0;
B
Bernard Xiong 已提交
181
    thread->high_mask = 0;
182
#endif /* RT_THREAD_PRIORITY_MAX > 32 */
183

184 185 186 187 188 189 190 191
    /* tick init */
    thread->init_tick      = tick;
    thread->remaining_tick = tick;

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

S
shaojinchun 已提交
192 193 194 195 196 197 198 199
#ifdef RT_USING_SMP
    /* not bind on any cpu */
    thread->bind_cpu = RT_CPUS_NR;
    thread->oncpu = RT_CPU_DETACHED;

    /* lock init */
    thread->scheduler_lock_nest = 0;
    thread->cpus_lock_nest = 0;
S
shaojinchun 已提交
200
    thread->critical_lock_nest = 0;
201
#endif /* RT_USING_SMP */
S
shaojinchun 已提交
202

203 204 205 206
    /* initialize cleanup function and user data */
    thread->cleanup   = 0;
    thread->user_data = 0;

B
Bernard Xiong 已提交
207
    /* initialize thread timer */
208 209 210 211 212 213 214
    rt_timer_init(&(thread->thread_timer),
                  thread->name,
                  rt_thread_timeout,
                  thread,
                  0,
                  RT_TIMER_FLAG_ONE_SHOT);

B
bernard 已提交
215 216 217 218 219
    /* initialize signal */
#ifdef RT_USING_SIGNALS
    thread->sig_mask    = 0x00;
    thread->sig_pending = 0x00;

S
shaojinchun 已提交
220
#ifndef RT_USING_SMP
B
bernard 已提交
221
    thread->sig_ret     = RT_NULL;
222
#endif /* RT_USING_SMP */
B
bernard 已提交
223 224
    thread->sig_vectors = RT_NULL;
    thread->si_list     = RT_NULL;
225
#endif /* RT_USING_SIGNALS */
B
bernard 已提交
226

H
heyuanjie 已提交
227 228
#ifdef RT_USING_LWP
    thread->lwp = RT_NULL;
229
#endif /* RT_USING_LWP */
H
heyuanjie 已提交
230

F
fenghuijie 已提交
231 232 233 234
#ifdef RT_USING_CPU_USAGE
    thread->duration_tick = 0;
#endif

235
    RT_OBJECT_HOOK_CALL(rt_thread_inited_hook, (thread));
B
Bernard Xiong 已提交
236

237
    return RT_EOK;
238 239 240 241 242 243
}

/**
 * @addtogroup Thread
 */

D
dogandog 已提交
244
/**@{*/
245 246

/**
B
bernard.xiong@gmail.com 已提交
247 248
 * This function will initialize a thread, normally it's used to initialize a
 * static thread object.
249 250 251 252 253 254 255 256 257 258
 *
 * @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 已提交
259
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
260
 */
D
dzzxzz 已提交
261
rt_err_t rt_thread_init(struct rt_thread *thread,
262 263 264 265 266 267 268
                        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)
269
{
270 271 272 273
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
    RT_ASSERT(stack_start != RT_NULL);

B
Bernard Xiong 已提交
274
    /* initialize thread object */
275 276 277 278 279 280 281 282 283 284
    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);
285
}
286
RTM_EXPORT(rt_thread_init);
287 288 289 290 291 292

/**
 * This function will return self thread object
 *
 * @return the self thread object
 */
D
dzzxzz 已提交
293
rt_thread_t rt_thread_self(void)
294
{
S
shaojinchun 已提交
295 296 297 298 299 300 301 302 303 304 305
#ifdef RT_USING_SMP
    rt_base_t lock;
    rt_thread_t self;

    lock = rt_hw_local_irq_disable();
    self = rt_cpu_self()->current_thread;
    rt_hw_local_irq_enable(lock);
    return self;
#else
    extern rt_thread_t rt_current_thread;

306
    return rt_current_thread;
307
#endif /* RT_USING_SMP */
308
}
309
RTM_EXPORT(rt_thread_self);
310 311 312 313 314 315

/**
 * 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 已提交
316
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
317
 */
D
dzzxzz 已提交
318
rt_err_t rt_thread_startup(rt_thread_t thread)
319
{
320 321
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
B
bernard 已提交
322
    RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_INIT);
323
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
324

B
Bernard Xiong 已提交
325
    /* set current priority to initialize priority */
326
    thread->current_priority = thread->init_priority;
327

328
    /* calculate priority attribute */
329
#if RT_THREAD_PRIORITY_MAX > 32
330 331 332
    thread->number      = thread->current_priority >> 3;            /* 5bit */
    thread->number_mask = 1L << thread->number;
    thread->high_mask   = 1L << (thread->current_priority & 0x07);  /* 3bit */
333
#else
334
    thread->number_mask = 1L << thread->current_priority;
335
#endif /* RT_THREAD_PRIORITY_MAX > 32 */
336

337
    RT_DEBUG_LOG(RT_DEBUG_THREAD, ("startup a thread:%s with priority:%d\n",
338
                                   thread->name, thread->init_priority));
339 340 341 342 343 344 345 346 347 348 349
    /* 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;
350
}
351
RTM_EXPORT(rt_thread_startup);
352 353

/**
B
bernard.xiong@gmail.com 已提交
354 355
 * This function will detach a thread. The thread object will be removed from
 * thread queue and detached/deleted from system object management.
356 357 358
 *
 * @param thread the thread to be deleted
 *
B
bernard.xiong@gmail.com 已提交
359
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
360
 */
D
dzzxzz 已提交
361
rt_err_t rt_thread_detach(rt_thread_t thread)
362
{
T
tangyuxin 已提交
363 364
    rt_base_t lock;

365 366
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
367 368
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
    RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread));
369

C
cliff-cmc 已提交
370
    if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_CLOSE)
371 372
        return RT_EOK;

B
bernard 已提交
373
    if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
374 375 376 377
    {
        /* remove from schedule */
        rt_schedule_remove_thread(thread);
    }
378

379
    _rt_thread_cleanup_execute(thread);
380

381 382
    /* release thread timer */
    rt_timer_detach(&(thread->thread_timer));
383

384 385
    /* change stat */
    thread->stat = RT_THREAD_CLOSE;
386

F
fenghuijie 已提交
387 388 389 390 391 392 393 394 395 396 397 398 399
    if (rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE)
    {
        rt_object_detach((rt_object_t)thread);
    }
    else
    {
        /* disable interrupt */
        lock = rt_hw_interrupt_disable();
        /* insert to defunct thread list */
        rt_thread_defunct_enqueue(thread);
        /* enable interrupt */
        rt_hw_interrupt_enable(lock);
    }
T
tangyuxin 已提交
400

401
    return RT_EOK;
402
}
403 404
RTM_EXPORT(rt_thread_detach);

405
#ifdef RT_USING_HEAP
406 407 408 409 410 411 412 413 414 415 416 417 418 419
/**
 * 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,
420 421 422 423 424
                             void (*entry)(void *parameter),
                             void       *parameter,
                             rt_uint32_t stack_size,
                             rt_uint8_t  priority,
                             rt_uint32_t tick)
425
{
426 427 428 429 430 431 432 433
    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 已提交
434
    stack_start = (void *)RT_KERNEL_MALLOC(stack_size);
435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452
    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;
453
}
454
RTM_EXPORT(rt_thread_create);
455

456
/**
B
bernard.xiong@gmail.com 已提交
457
 * This function will delete a thread. The thread object will be removed from
458
 * thread queue and deleted from system object management in the idle thread.
459 460 461
 *
 * @param thread the thread to be deleted
 *
B
bernard.xiong@gmail.com 已提交
462
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
463
 */
D
dzzxzz 已提交
464
rt_err_t rt_thread_delete(rt_thread_t thread)
465
{
466
    rt_base_t lock;
467

468 469
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
470 471
    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);
472

C
cliff-cmc 已提交
473
    if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_CLOSE)
474 475
        return RT_EOK;

B
bernard 已提交
476
    if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
477 478 479 480
    {
        /* remove from schedule */
        rt_schedule_remove_thread(thread);
    }
481

482
    _rt_thread_cleanup_execute(thread);
483

484 485
    /* release thread timer */
    rt_timer_detach(&(thread->thread_timer));
486

487 488
    /* disable interrupt */
    lock = rt_hw_interrupt_disable();
489

490 491 492
    /* change stat */
    thread->stat = RT_THREAD_CLOSE;

493
    /* insert to defunct thread list */
F
fenghuijie 已提交
494
    rt_thread_defunct_enqueue(thread);
495

496 497
    /* enable interrupt */
    rt_hw_interrupt_enable(lock);
498

499
    return RT_EOK;
500
}
501
RTM_EXPORT(rt_thread_delete);
502
#endif /* RT_USING_HEAP */
503 504

/**
B
bernard.xiong@gmail.com 已提交
505 506 507
 * 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.
508
 *
B
bernard.xiong@gmail.com 已提交
509
 * @return RT_EOK
510
 */
D
dzzxzz 已提交
511
rt_err_t rt_thread_yield(void)
512
{
513 514 515 516 517 518 519
    struct rt_thread *thread;
    rt_base_t lock;

    thread = rt_thread_self();
    lock = rt_hw_interrupt_disable();
    thread->remaining_tick = thread->init_tick;
    thread->stat |= RT_THREAD_STAT_YIELD;
S
shaojinchun 已提交
520
    rt_schedule();
521
    rt_hw_interrupt_enable(lock);
522

523
    return RT_EOK;
524
}
525
RTM_EXPORT(rt_thread_yield);
526 527 528 529 530 531

/**
 * This function will let current thread sleep for some ticks.
 *
 * @param tick the sleep ticks
 *
B
bernard.xiong@gmail.com 已提交
532
 * @return RT_EOK
533
 */
D
dzzxzz 已提交
534
rt_err_t rt_thread_sleep(rt_tick_t tick)
535
{
536 537
    register rt_base_t temp;
    struct rt_thread *thread;
538

539
    /* set to current thread */
S
shaojinchun 已提交
540
    thread = rt_thread_self();
541
    RT_ASSERT(thread != RT_NULL);
542
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
543

S
shaojinchun 已提交
544 545 546
    /* disable interrupt */
    temp = rt_hw_interrupt_disable();

547 548
    /* suspend thread */
    rt_thread_suspend(thread);
549

550 551 552
    /* 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));
553

554 555
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);
556

557
    rt_schedule();
558

559 560 561
    /* clear error number of this thread to RT_EOK */
    if (thread->error == -RT_ETIMEOUT)
        thread->error = RT_EOK;
562

563
    return RT_EOK;
564 565 566 567 568 569 570
}

/**
 * This function will let current thread delay for some ticks.
 *
 * @param tick the delay ticks
 *
B
bernard.xiong@gmail.com 已提交
571
 * @return RT_EOK
572 573 574
 */
rt_err_t rt_thread_delay(rt_tick_t tick)
{
575
    return rt_thread_sleep(tick);
576
}
577
RTM_EXPORT(rt_thread_delay);
578

579
/**
B
Bernard Xiong 已提交
580
 * This function will let current thread delay until (*tick + inc_tick).
581 582 583 584 585 586
 *
 * @param tick the tick of last wakeup.
 * @param inc_tick the increment tick
 *
 * @return RT_EOK
 */
B
Bernard Xiong 已提交
587
rt_err_t rt_thread_delay_until(rt_tick_t *tick, rt_tick_t inc_tick)
588 589 590
{
    register rt_base_t level;
    struct rt_thread *thread;
591
    rt_tick_t cur_tick;
592

B
Bernard Xiong 已提交
593 594
    RT_ASSERT(tick != RT_NULL);

595 596 597 598 599 600 601 602
    /* set to current thread */
    thread = rt_thread_self();
    RT_ASSERT(thread != RT_NULL);
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);

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

603
    cur_tick = rt_tick_get();
604
    if (cur_tick - *tick < inc_tick)
605
    {
606 607 608 609
        rt_tick_t left_tick;

        *tick += inc_tick;
        left_tick = *tick - cur_tick;
610 611 612 613 614

        /* suspend thread */
        rt_thread_suspend(thread);

        /* reset the timeout of thread timer and start it */
615
        rt_timer_control(&(thread->thread_timer), RT_TIMER_CTRL_SET_TIME, &left_tick);
616 617 618 619 620 621 622 623 624 625 626 627 628 629 630
        rt_timer_start(&(thread->thread_timer));

        /* enable interrupt */
        rt_hw_interrupt_enable(level);

        rt_schedule();

        /* clear error number of this thread to RT_EOK */
        if (thread->error == -RT_ETIMEOUT)
        {
            thread->error = RT_EOK;
        }
    }
    else
    {
631
        *tick = cur_tick;
632 633 634 635 636
        rt_hw_interrupt_enable(level);
    }

    return RT_EOK;
}
637
RTM_EXPORT(rt_thread_delay_until);
638

H
heyuanjie 已提交
639 640 641
/**
 * This function will let current thread delay for some milliseconds.
 *
S
suhui 已提交
642
 * @param ms the delay ms time
H
heyuanjie 已提交
643 644 645 646 647 648 649 650 651 652 653 654 655
 *
 * @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 已提交
656 657 658 659 660
/**
 * 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
661
 *  RT_THREAD_CTRL_CHANGE_PRIORITY for changing priority level of thread;
B
bernard.xiong@gmail.com 已提交
662
 *  RT_THREAD_CTRL_STARTUP for starting a thread;
S
shaojinchun 已提交
663 664
 *  RT_THREAD_CTRL_CLOSE for delete a thread;
 *  RT_THREAD_CTRL_BIND_CPU for bind the thread to a CPU.
B
bernard.xiong@gmail.com 已提交
665 666 667 668
 * @param arg the argument of control command
 *
 * @return RT_EOK
 */
B
bernard 已提交
669
rt_err_t rt_thread_control(rt_thread_t thread, int cmd, void *arg)
670
{
671
    register rt_base_t temp;
672

673 674
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
675
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
676

677 678
    switch (cmd)
    {
679
        case RT_THREAD_CTRL_CHANGE_PRIORITY:
680
        {
681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699
            /* disable interrupt */
            temp = rt_hw_interrupt_disable();

            /* for ready thread, change queue */
            if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_READY)
            {
                /* remove thread from schedule queue first */
                rt_schedule_remove_thread(thread);

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

                /* recalculate priority attribute */
    #if RT_THREAD_PRIORITY_MAX > 32
                thread->number      = thread->current_priority >> 3;            /* 5bit */
                thread->number_mask = 1 << thread->number;
                thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
    #else
                thread->number_mask = 1 << thread->current_priority;
700
    #endif /* RT_THREAD_PRIORITY_MAX > 32 */
701 702 703 704 705 706 707 708 709 710 711 712 713 714 715

                /* insert thread to schedule queue again */
                rt_schedule_insert_thread(thread);
            }
            else
            {
                thread->current_priority = *(rt_uint8_t *)arg;

                /* recalculate priority attribute */
    #if RT_THREAD_PRIORITY_MAX > 32
                thread->number      = thread->current_priority >> 3;            /* 5bit */
                thread->number_mask = 1 << thread->number;
                thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
    #else
                thread->number_mask = 1 << thread->current_priority;
716
    #endif /* RT_THREAD_PRIORITY_MAX > 32 */
717 718 719 720 721
            }

            /* enable interrupt */
            rt_hw_interrupt_enable(temp);
            break;
722
        }
723

724 725 726
        case RT_THREAD_CTRL_STARTUP:
        {
            return rt_thread_startup(thread);
727
        }
728

729
        case RT_THREAD_CTRL_CLOSE:
730
        {
731 732 733 734 735 736 737 738 739 740 741
            rt_err_t rt_err;

            if (rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE)
            {
                rt_err = rt_thread_detach(thread);
            }
    #ifdef RT_USING_HEAP
            else
            {
                rt_err = rt_thread_delete(thread);
            }
742
    #endif /* RT_USING_HEAP */
743 744
            rt_schedule();
            return rt_err;
745
        }
746 747 748

    #ifdef RT_USING_SMP
        case RT_THREAD_CTRL_BIND_CPU:
749
        {
750
            rt_uint8_t cpu;
751

752 753 754 755 756
            if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
            {
                /* we only support bind cpu before started phase. */
                return RT_ERROR;
            }
S
shaojinchun 已提交
757

758 759 760
            cpu = (rt_uint8_t)(size_t)arg;
            thread->bind_cpu = cpu > RT_CPUS_NR? RT_CPUS_NR : cpu;
            break;
S
shaojinchun 已提交
761
        }
762
    #endif /* RT_USING_SMP */
S
shaojinchun 已提交
763

764 765
        default:
            break;
766
    }
767

768
    return RT_EOK;
769
}
770
RTM_EXPORT(rt_thread_control);
771 772 773 774 775 776

/**
 * This function will suspend the specified thread.
 *
 * @param thread the thread to be suspended
 *
B
bernard.xiong@gmail.com 已提交
777
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
778
 *
B
bernard.xiong@gmail.com 已提交
779 780
 * @note if suspend self thread, after this function call, the
 * rt_schedule() must be invoked.
781
 */
D
dzzxzz 已提交
782
rt_err_t rt_thread_suspend(rt_thread_t thread)
783
{
784
    register rt_base_t stat;
785
    register rt_base_t temp;
786

787 788
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
789
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
790

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

793 794
    stat = thread->stat & RT_THREAD_STAT_MASK;
    if ((stat != RT_THREAD_READY) && (stat != RT_THREAD_RUNNING))
795
    {
796
        RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend: thread disorder, 0x%2x\n",
797
                                       thread->stat));
798 799
        return -RT_ERROR;
    }
800

801 802
    /* disable interrupt */
    temp = rt_hw_interrupt_disable();
803 804 805 806 807
    if (stat == RT_THREAD_RUNNING)
    {
        /* not suspend running status thread on other core */
        RT_ASSERT(thread == rt_thread_self());
    }
808

809 810
    /* change thread stat */
    rt_schedule_remove_thread(thread);
811
    thread->stat = RT_THREAD_SUSPEND | (thread->stat & ~RT_THREAD_STAT_MASK);
812

813 814 815
    /* stop thread timer anyway */
    rt_timer_stop(&(thread->thread_timer));

816 817
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);
818

819
    RT_OBJECT_HOOK_CALL(rt_thread_suspend_hook, (thread));
820
    return RT_EOK;
821
}
822
RTM_EXPORT(rt_thread_suspend);
823 824 825 826 827 828

/**
 * 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 已提交
829
 * @return the operation status, RT_EOK on OK, -RT_ERROR on error
830
 */
D
dzzxzz 已提交
831
rt_err_t rt_thread_resume(rt_thread_t thread)
832
{
833
    register rt_base_t temp;
834

835 836
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
837
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
838

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

B
bernard 已提交
841
    if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_SUSPEND)
842 843
    {
        RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume: thread disorder, %d\n",
844
                                       thread->stat));
845

846 847
        return -RT_ERROR;
    }
848

849 850
    /* disable interrupt */
    temp = rt_hw_interrupt_disable();
851

852 853
    /* remove from suspend list */
    rt_list_remove(&(thread->tlist));
854

855
    rt_timer_stop(&thread->thread_timer);
856

857 858
    /* insert to schedule ready list */
    rt_schedule_insert_thread(thread);
859

F
fenghuijie 已提交
860 861 862
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);

863
    RT_OBJECT_HOOK_CALL(rt_thread_resume_hook, (thread));
864
    return RT_EOK;
865
}
866
RTM_EXPORT(rt_thread_resume);
867 868

/**
B
bernard.xiong@gmail.com 已提交
869
 * This function is the timeout function for thread, normally which is invoked
870
 * when thread is timeout to wait some resource.
871 872 873
 *
 * @param parameter the parameter of thread timeout function
 */
D
dzzxzz 已提交
874
void rt_thread_timeout(void *parameter)
875
{
876
    struct rt_thread *thread;
F
fenghuijie 已提交
877
    register rt_base_t temp;
878

879
    thread = (struct rt_thread *)parameter;
880

881 882
    /* thread check */
    RT_ASSERT(thread != RT_NULL);
B
bernard 已提交
883
    RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_SUSPEND);
884
    RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
885

F
fenghuijie 已提交
886 887 888
    /* disable interrupt */
    temp = rt_hw_interrupt_disable();

889 890
    /* set error number */
    thread->error = -RT_ETIMEOUT;
891

892 893
    /* remove from suspend list */
    rt_list_remove(&(thread->tlist));
894

895 896
    /* insert to schedule ready list */
    rt_schedule_insert_thread(thread);
897

F
fenghuijie 已提交
898 899 900
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);

901 902
    /* do schedule */
    rt_schedule();
903
}
904
RTM_EXPORT(rt_thread_timeout);
905 906 907 908 909 910

/**
 * This function will find the specified thread.
 *
 * @param name the name of thread finding
 *
B
bernard.xiong@gmail.com 已提交
911 912 913
 * @return the found thread
 *
 * @note please don't invoke this function in interrupt status.
914
 */
D
dzzxzz 已提交
915
rt_thread_t rt_thread_find(char *name)
916
{
917
    return (rt_thread_t)rt_object_find(name, RT_Object_Class_Thread);
918
}
919
RTM_EXPORT(rt_thread_find);
920

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