仓库迁移
This commit is contained in:
		
							
								
								
									
										120
									
								
								source/rt_thread/src/clock.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								source/rt_thread/src/clock.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,120 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-03-12     Bernard      first version | ||||
|  * 2006-05-27     Bernard      add support for same priority thread schedule | ||||
|  * 2006-08-10     Bernard      remove the last rt_schedule in rt_tick_increase | ||||
|  * 2010-03-08     Bernard      remove rt_passed_second | ||||
|  * 2010-05-20     Bernard      fix the tick exceeds the maximum limits | ||||
|  * 2010-07-13     Bernard      fix rt_tick_from_millisecond issue found by kuronca | ||||
|  * 2011-06-26     Bernard      add rt_tick_set function. | ||||
|  * 2018-11-22     Jesven       add per cpu tick | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| static rt_tick_t rt_tick = 0; | ||||
|  | ||||
| /** | ||||
|  * This function will initialize system tick and set it to zero. | ||||
|  * @ingroup SystemInit | ||||
|  * | ||||
|  * @deprecated since 1.1.0, this function does not need to be invoked | ||||
|  * in the system initialization. | ||||
|  */ | ||||
| void rt_system_tick_init(void) | ||||
| { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Clock | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will return current tick from operating system startup | ||||
|  * | ||||
|  * @return current tick | ||||
|  */ | ||||
| rt_tick_t rt_tick_get(void) | ||||
| { | ||||
|     /* return the global tick */ | ||||
|     return rt_tick; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set current tick | ||||
|  */ | ||||
| void rt_tick_set(rt_tick_t tick) | ||||
| { | ||||
|     rt_base_t level; | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     rt_tick = tick; | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will notify kernel there is one tick passed. Normally, | ||||
|  * this function is invoked by clock ISR. | ||||
|  */ | ||||
| void rt_tick_increase(void) | ||||
| { | ||||
|     struct rt_thread *thread; | ||||
|  | ||||
|     /* increase the global tick */ | ||||
|     ++ rt_tick; | ||||
|  | ||||
|     /* check time slice */ | ||||
|     thread = rt_thread_self(); | ||||
|  | ||||
|     -- thread->remaining_tick; | ||||
|     if (thread->remaining_tick == 0) | ||||
|     { | ||||
|         /* change to initialized tick */ | ||||
|         thread->remaining_tick = thread->init_tick; | ||||
|  | ||||
|         /* yield */ | ||||
|         rt_thread_yield(); | ||||
|     } | ||||
|  | ||||
|     /* check timer */ | ||||
|     rt_timer_check(); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will calculate the tick from millisecond. | ||||
|  * | ||||
|  * @param ms the specified millisecond | ||||
|  *           - Negative Number wait forever | ||||
|  *           - Zero not wait | ||||
|  *           - Max 0x7fffffff | ||||
|  * | ||||
|  * @return the calculated tick | ||||
|  */ | ||||
| rt_tick_t rt_tick_from_millisecond(rt_int32_t ms) | ||||
| { | ||||
|     rt_tick_t tick; | ||||
|  | ||||
|     if (ms < 0) | ||||
|     { | ||||
|         tick = (rt_tick_t)RT_WAITING_FOREVER; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         tick = RT_TICK_PER_SECOND * (ms / 1000); | ||||
|         tick += (RT_TICK_PER_SECOND * (ms % 1000) + 999) / 1000; | ||||
|     } | ||||
|  | ||||
|     /* return the calculated tick */ | ||||
|     return tick; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
|  | ||||
							
								
								
									
										246
									
								
								source/rt_thread/src/components.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										246
									
								
								source/rt_thread/src/components.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,246 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2012-09-20     Bernard      Change the name to components.c | ||||
|  *                             And all components related header files. | ||||
|  * 2012-12-23     Bernard      fix the pthread initialization issue. | ||||
|  * 2013-06-23     Bernard      Add the init_call for components initialization. | ||||
|  * 2013-07-05     Bernard      Remove initialization feature for MS VC++ compiler | ||||
|  * 2015-02-06     Bernard      Remove the MS VC++ support and move to the kernel | ||||
|  * 2015-05-04     Bernard      Rename it to components.c because compiling issue | ||||
|  *                             in some IDEs. | ||||
|  * 2015-07-29     Arda.Fu      Add support to use RT_USING_USER_MAIN with IAR | ||||
|  * 2018-11-22     Jesven       Add secondary cpu boot up | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #ifdef RT_USING_USER_MAIN | ||||
| #ifndef RT_MAIN_THREAD_STACK_SIZE | ||||
| #define RT_MAIN_THREAD_STACK_SIZE     2048 | ||||
| #endif | ||||
| #ifndef RT_MAIN_THREAD_PRIORITY | ||||
| #define RT_MAIN_THREAD_PRIORITY       (RT_THREAD_PRIORITY_MAX / 3) | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| #ifdef RT_USING_COMPONENTS_INIT | ||||
| /* | ||||
|  * Components Initialization will initialize some driver and components as following | ||||
|  * order: | ||||
|  * rti_start         --> 0 | ||||
|  * BOARD_EXPORT      --> 1 | ||||
|  * rti_board_end     --> 1.end | ||||
|  * | ||||
|  * DEVICE_EXPORT     --> 2 | ||||
|  * COMPONENT_EXPORT  --> 3 | ||||
|  * FS_EXPORT         --> 4 | ||||
|  * ENV_EXPORT        --> 5 | ||||
|  * APP_EXPORT        --> 6 | ||||
|  * | ||||
|  * rti_end           --> 6.end | ||||
|  * | ||||
|  * These automatically initialization, the driver or component initial function must | ||||
|  * be defined with: | ||||
|  * INIT_BOARD_EXPORT(fn); | ||||
|  * INIT_DEVICE_EXPORT(fn); | ||||
|  * ... | ||||
|  * INIT_APP_EXPORT(fn); | ||||
|  * etc. | ||||
|  */ | ||||
| static int rti_start(void) | ||||
| { | ||||
|     return 0; | ||||
| } | ||||
| INIT_EXPORT(rti_start, "0"); | ||||
|  | ||||
| static int rti_board_start(void) | ||||
| { | ||||
|     return 0; | ||||
| } | ||||
| INIT_EXPORT(rti_board_start, "0.end"); | ||||
|  | ||||
| static int rti_board_end(void) | ||||
| { | ||||
|     return 0; | ||||
| } | ||||
| INIT_EXPORT(rti_board_end, "1.end"); | ||||
|  | ||||
| static int rti_end(void) | ||||
| { | ||||
|     return 0; | ||||
| } | ||||
| INIT_EXPORT(rti_end, "6.end"); | ||||
|  | ||||
| /** | ||||
|  * RT-Thread Components Initialization for board | ||||
|  */ | ||||
| void rt_components_board_init(void) | ||||
| { | ||||
| #if RT_DEBUG_INIT | ||||
|     int result; | ||||
|     const struct rt_init_desc *desc; | ||||
|     for (desc = &__rt_init_desc_rti_board_start; desc < &__rt_init_desc_rti_board_end; desc ++) | ||||
|     { | ||||
|         rt_kprintf("initialize %s", desc->fn_name); | ||||
|         result = desc->fn(); | ||||
|         rt_kprintf(":%d done\n", result); | ||||
|     } | ||||
| #else | ||||
|     volatile const init_fn_t *fn_ptr; | ||||
|  | ||||
|     for (fn_ptr = &__rt_init_rti_board_start; fn_ptr < &__rt_init_rti_board_end; fn_ptr++) | ||||
|     { | ||||
|         (*fn_ptr)(); | ||||
|     } | ||||
| #endif | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * RT-Thread Components Initialization | ||||
|  */ | ||||
| void rt_components_init(void) | ||||
| { | ||||
| #if RT_DEBUG_INIT | ||||
|     int result; | ||||
|     const struct rt_init_desc *desc; | ||||
|  | ||||
|     rt_kprintf("do components initialization.\n"); | ||||
|     for (desc = &__rt_init_desc_rti_board_end; desc < &__rt_init_desc_rti_end; desc ++) | ||||
|     { | ||||
|         rt_kprintf("initialize %s", desc->fn_name); | ||||
|         result = desc->fn(); | ||||
|         rt_kprintf(":%d done\n", result); | ||||
|     } | ||||
| #else | ||||
|     volatile const init_fn_t *fn_ptr; | ||||
|  | ||||
|     for (fn_ptr = &__rt_init_rti_board_end; fn_ptr < &__rt_init_rti_end; fn_ptr ++) | ||||
|     { | ||||
|         (*fn_ptr)(); | ||||
|     } | ||||
| #endif | ||||
| } | ||||
| #endif   /* RT_USING_COMPONENTS_INIT */ | ||||
|  | ||||
| #ifdef RT_USING_USER_MAIN | ||||
|  | ||||
| void rt_application_init(void); | ||||
| void rt_hw_board_init(void); | ||||
| int rtthread_startup(void); | ||||
|  | ||||
| #if defined(__CC_ARM) || defined(__CLANG_ARM) | ||||
| extern int $Super$$main(void); | ||||
| /* re-define main function */ | ||||
| int $Sub$$main(void) | ||||
| { | ||||
|     rtthread_startup(); | ||||
|     return 0; | ||||
| } | ||||
| #elif defined(__ICCARM__) | ||||
| extern int main(void); | ||||
| /* __low_level_init will auto called by IAR cstartup */ | ||||
| extern void __iar_data_init3(void); | ||||
| int __low_level_init(void) | ||||
| { | ||||
|     // call IAR table copy function. | ||||
|     __iar_data_init3(); | ||||
|     rtthread_startup(); | ||||
|     return 0; | ||||
| } | ||||
| #elif defined(__GNUC__) | ||||
| /* Add -eentry to arm-none-eabi-gcc argument */ | ||||
| int entry(void) | ||||
| { | ||||
|     rtthread_startup(); | ||||
|     return 0; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifndef RT_USING_HEAP | ||||
| /* if there is not enable heap, we should use static thread and stack. */ | ||||
| ALIGN(8) | ||||
| static rt_uint8_t main_stack[RT_MAIN_THREAD_STACK_SIZE]; | ||||
| struct rt_thread main_thread; | ||||
| #endif | ||||
|  | ||||
| /* the system main thread */ | ||||
| void main_thread_entry(void *parameter) | ||||
| { | ||||
|     extern int main(void); | ||||
|     extern int $Super$$main(void); | ||||
|  | ||||
| #ifdef RT_USING_COMPONENTS_INIT | ||||
|     /* RT-Thread components initialization */ | ||||
|     rt_components_init(); | ||||
| #endif | ||||
|     /* invoke system main function */ | ||||
| #if defined(__CC_ARM) || defined(__CLANG_ARM) | ||||
|     $Super$$main(); /* for ARMCC. */ | ||||
| #elif defined(__ICCARM__) || defined(__GNUC__) | ||||
|     main(); | ||||
| #endif | ||||
| } | ||||
|  | ||||
| void rt_application_init(void) | ||||
| { | ||||
|     rt_thread_t tid; | ||||
|  | ||||
| #ifdef RT_USING_HEAP | ||||
|     tid = rt_thread_create("main", main_thread_entry, RT_NULL, | ||||
|                            RT_MAIN_THREAD_STACK_SIZE, RT_MAIN_THREAD_PRIORITY, 20); | ||||
|     RT_ASSERT(tid != RT_NULL); | ||||
| #else | ||||
|     rt_err_t result; | ||||
|  | ||||
|     tid = &main_thread; | ||||
|     result = rt_thread_init(tid, "main", main_thread_entry, RT_NULL, | ||||
|                             main_stack, sizeof(main_stack), RT_MAIN_THREAD_PRIORITY, 20); | ||||
|     RT_ASSERT(result == RT_EOK); | ||||
|  | ||||
|     /* if not define RT_USING_HEAP, using to eliminate the warning */ | ||||
|     (void)result; | ||||
| #endif | ||||
|  | ||||
|     rt_thread_startup(tid); | ||||
| } | ||||
|  | ||||
| int rtthread_startup(void) | ||||
| { | ||||
|     rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* board level initialization | ||||
|      * NOTE: please initialize heap inside board initialization. | ||||
|      */ | ||||
|     rt_hw_board_init(); | ||||
|  | ||||
|     /* show RT-Thread version */ | ||||
|     rt_show_version(); | ||||
|  | ||||
|     /* timer system initialization */ | ||||
|     rt_system_timer_init(); | ||||
|  | ||||
|     /* scheduler system initialization */ | ||||
|     rt_system_scheduler_init(); | ||||
|  | ||||
|     /* create init_thread */ | ||||
|     rt_application_init(); | ||||
|  | ||||
|     /* timer thread initialization */ | ||||
|     rt_system_timer_thread_init(); | ||||
|  | ||||
|     /* idle thread initialization */ | ||||
|     rt_thread_idle_init(); | ||||
|  | ||||
|     /* start scheduler */ | ||||
|     rt_system_scheduler_start(); | ||||
|  | ||||
|     /* never reach here */ | ||||
|     return 0; | ||||
| } | ||||
| #endif | ||||
							
								
								
									
										14
									
								
								source/rt_thread/src/cpu.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								source/rt_thread/src/cpu.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-10-30     Bernard      The first version | ||||
|  */ | ||||
|  | ||||
| #include <rtthread.h> | ||||
| #include <rthw.h> | ||||
|  | ||||
| /* nothing on non-smp version */ | ||||
							
								
								
									
										235
									
								
								source/rt_thread/src/idle.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										235
									
								
								source/rt_thread/src/idle.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,235 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-03-23     Bernard      the first version | ||||
|  * 2010-11-10     Bernard      add cleanup callback function in thread exit. | ||||
|  * 2012-12-29     Bernard      fix compiling warning. | ||||
|  * 2013-12-21     Grissiom     let rt_thread_idle_excute loop until there is no | ||||
|  *                             dead thread. | ||||
|  * 2016-08-09     ArdaFu       add method to get the handler of the idle thread. | ||||
|  * 2018-02-07     Bernard      lock scheduler to protect tid->cleanup. | ||||
|  * 2018-07-14     armink       add idle hook list | ||||
|  * 2018-11-22     Jesven       add per cpu idle task | ||||
|  *                             combine the code of primary and secondary cpu | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #if defined (RT_USING_HOOK) | ||||
| #ifndef RT_USING_IDLE_HOOK | ||||
| #define RT_USING_IDLE_HOOK | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| #ifndef IDLE_THREAD_STACK_SIZE | ||||
| #if defined (RT_USING_IDLE_HOOK) || defined(RT_USING_HEAP) | ||||
| #define IDLE_THREAD_STACK_SIZE  256 | ||||
| #else | ||||
| #define IDLE_THREAD_STACK_SIZE  128 | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| extern rt_list_t rt_thread_defunct; | ||||
|  | ||||
| static struct rt_thread idle; | ||||
| ALIGN(RT_ALIGN_SIZE) | ||||
| static rt_uint8_t rt_thread_stack[IDLE_THREAD_STACK_SIZE]; | ||||
|  | ||||
| #ifdef RT_USING_IDLE_HOOK | ||||
| #ifndef RT_IDLE_HOOK_LIST_SIZE | ||||
| #define RT_IDLE_HOOK_LIST_SIZE  4 | ||||
| #endif | ||||
|  | ||||
| static void (*idle_hook_list[RT_IDLE_HOOK_LIST_SIZE])(void); | ||||
|  | ||||
| /** | ||||
|  * @ingroup Hook | ||||
|  * This function sets a hook function to idle thread loop. When the system performs | ||||
|  * idle loop, this hook function should be invoked. | ||||
|  * | ||||
|  * @param hook the specified hook function | ||||
|  * | ||||
|  * @return RT_EOK: set OK | ||||
|  *         -RT_EFULL: hook list is full | ||||
|  * | ||||
|  * @note the hook function must be simple and never be blocked or suspend. | ||||
|  */ | ||||
| rt_err_t rt_thread_idle_sethook(void (*hook)(void)) | ||||
| { | ||||
|     rt_size_t i; | ||||
|     rt_base_t level; | ||||
|     rt_err_t ret = -RT_EFULL; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     for (i = 0; i < RT_IDLE_HOOK_LIST_SIZE; i++) | ||||
|     { | ||||
|         if (idle_hook_list[i] == RT_NULL) | ||||
|         { | ||||
|             idle_hook_list[i] = hook; | ||||
|             ret = RT_EOK; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * delete the idle hook on hook list | ||||
|  * | ||||
|  * @param hook the specified hook function | ||||
|  * | ||||
|  * @return RT_EOK: delete OK | ||||
|  *         -RT_ENOSYS: hook was not found | ||||
|  */ | ||||
| rt_err_t rt_thread_idle_delhook(void (*hook)(void)) | ||||
| { | ||||
|     rt_size_t i; | ||||
|     rt_base_t level; | ||||
|     rt_err_t ret = -RT_ENOSYS; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     for (i = 0; i < RT_IDLE_HOOK_LIST_SIZE; i++) | ||||
|     { | ||||
|         if (idle_hook_list[i] == hook) | ||||
|         { | ||||
|             idle_hook_list[i] = RT_NULL; | ||||
|             ret = RT_EOK; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| #endif | ||||
|  | ||||
| #ifdef RT_USING_HEAP | ||||
| /* Return whether there is defunctional thread to be deleted. */ | ||||
| rt_inline int _has_defunct_thread(void) | ||||
| { | ||||
|     /* The rt_list_isempty has prototype of "int rt_list_isempty(const rt_list_t *l)". | ||||
|      * So the compiler has a good reason that the rt_thread_defunct list does | ||||
|      * not change within rt_thread_idle_excute thus optimize the "while" loop | ||||
|      * into a "if". | ||||
|      * | ||||
|      * So add the volatile qualifier here. */ | ||||
|     const volatile rt_list_t *l = (const volatile rt_list_t *)&rt_thread_defunct; | ||||
|  | ||||
|     return l->next != l; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * @ingroup Thread | ||||
|  * | ||||
|  * This function will perform system background job when system idle. | ||||
|  */ | ||||
| void rt_thread_idle_excute(void) | ||||
| { | ||||
|     /* Loop until there is no dead thread. So one call to rt_thread_idle_excute | ||||
|      * will do all the cleanups. */ | ||||
|     /* disable interrupt */ | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
| #ifdef RT_USING_HEAP | ||||
|     while (1) | ||||
|     { | ||||
|         rt_base_t lock; | ||||
|         rt_thread_t thread; | ||||
|  | ||||
|         lock = rt_hw_interrupt_disable(); | ||||
|  | ||||
|         /* check whether list is empty */ | ||||
|         if (!_has_defunct_thread()) | ||||
|         { | ||||
|             rt_hw_interrupt_enable(lock); | ||||
|             break; | ||||
|         } | ||||
|         /* get defunct thread */ | ||||
|         thread = rt_list_entry(rt_thread_defunct.next, | ||||
|                 struct rt_thread, | ||||
|                 tlist); | ||||
|         /* remove defunct thread */ | ||||
|         rt_list_remove(&(thread->tlist)); | ||||
|         /* release thread's stack */ | ||||
|         RT_KERNEL_FREE(thread->stack_addr); | ||||
|         /* delete thread object */ | ||||
|         rt_object_delete((rt_object_t)thread); | ||||
|         rt_hw_interrupt_enable(lock); | ||||
|     } | ||||
| #endif | ||||
| } | ||||
|  | ||||
| extern void rt_system_power_manager(void); | ||||
| static void rt_thread_idle_entry(void *parameter) | ||||
| { | ||||
|     while (1) | ||||
|     { | ||||
|  | ||||
| #ifdef RT_USING_IDLE_HOOK | ||||
|         rt_size_t i; | ||||
|  | ||||
|         for (i = 0; i < RT_IDLE_HOOK_LIST_SIZE; i++) | ||||
|         { | ||||
|             if (idle_hook_list[i] != RT_NULL) | ||||
|             { | ||||
|                 idle_hook_list[i](); | ||||
|             } | ||||
|         } | ||||
| #endif | ||||
|  | ||||
|         rt_thread_idle_excute(); | ||||
| #ifdef RT_USING_PM | ||||
|         rt_system_power_manager(); | ||||
| #endif | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * | ||||
|  * This function will initialize idle thread, then start it. | ||||
|  * | ||||
|  * @note this function must be invoked when system init. | ||||
|  */ | ||||
| void rt_thread_idle_init(void) | ||||
| { | ||||
|     /* initialize thread */ | ||||
|     rt_thread_init(&idle, | ||||
|                    "tidle", | ||||
|                    rt_thread_idle_entry, | ||||
|                    RT_NULL, | ||||
|                    &rt_thread_stack[0], | ||||
|                    sizeof(rt_thread_stack), | ||||
|                    RT_THREAD_PRIORITY_MAX - 1, | ||||
|                    32); | ||||
|  | ||||
|     /* startup */ | ||||
|     rt_thread_startup(&idle); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup Thread | ||||
|  * | ||||
|  * This function will get the handler of the idle thread. | ||||
|  * | ||||
|  */ | ||||
| rt_thread_t rt_thread_idle_gethandler(void) | ||||
| { | ||||
|     return (rt_thread_t)(&idle); | ||||
| } | ||||
							
								
								
									
										2502
									
								
								source/rt_thread/src/ipc.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2502
									
								
								source/rt_thread/src/ipc.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										113
									
								
								source/rt_thread/src/irq.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								source/rt_thread/src/irq.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-02-24     Bernard      first version | ||||
|  * 2006-05-03     Bernard      add IRQ_DEBUG | ||||
|  * 2016-08-09     ArdaFu       add interrupt enter and leave hook. | ||||
|  * 2018-11-22     Jesven       rt_interrupt_get_nest function add disable irq | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #ifdef RT_USING_HOOK | ||||
|  | ||||
| static void (*rt_interrupt_enter_hook)(void); | ||||
| static void (*rt_interrupt_leave_hook)(void); | ||||
|  | ||||
| /** | ||||
|  * @ingroup Hook | ||||
|  * This function set a hook function when the system enter a interrupt | ||||
|  * | ||||
|  * @note the hook function must be simple and never be blocked or suspend. | ||||
|  */ | ||||
| void rt_interrupt_enter_sethook(void (*hook)(void)) | ||||
| { | ||||
|     rt_interrupt_enter_hook = hook; | ||||
| } | ||||
| /** | ||||
|  * @ingroup Hook | ||||
|  * This function set a hook function when the system exit a interrupt. | ||||
|  * | ||||
|  * @note the hook function must be simple and never be blocked or suspend. | ||||
|  */ | ||||
| void rt_interrupt_leave_sethook(void (*hook)(void)) | ||||
| { | ||||
|     rt_interrupt_leave_hook = hook; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /* #define IRQ_DEBUG */ | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Kernel | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| volatile rt_uint8_t rt_interrupt_nest; | ||||
|  | ||||
| /** | ||||
|  * This function will be invoked by BSP, when enter interrupt service routine | ||||
|  * | ||||
|  * @note please don't invoke this routine in application | ||||
|  * | ||||
|  * @see rt_interrupt_leave | ||||
|  */ | ||||
| void rt_interrupt_enter(void) | ||||
| { | ||||
|     rt_base_t level; | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_IRQ, ("irq coming..., irq nest:%d\n", | ||||
|                                 rt_interrupt_nest)); | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     rt_interrupt_nest ++; | ||||
|     RT_OBJECT_HOOK_CALL(rt_interrupt_enter_hook,()); | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will be invoked by BSP, when leave interrupt service routine | ||||
|  * | ||||
|  * @note please don't invoke this routine in application | ||||
|  * | ||||
|  * @see rt_interrupt_enter | ||||
|  */ | ||||
| void rt_interrupt_leave(void) | ||||
| { | ||||
|     rt_base_t level; | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_IRQ, ("irq leave, irq nest:%d\n", | ||||
|                                 rt_interrupt_nest)); | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     rt_interrupt_nest --; | ||||
|     RT_OBJECT_HOOK_CALL(rt_interrupt_leave_hook,()); | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will return the nest of interrupt. | ||||
|  * | ||||
|  * User application can invoke this function to get whether current | ||||
|  * context is interrupt context. | ||||
|  * | ||||
|  * @return the number of nested interrupts. | ||||
|  */ | ||||
| RT_WEAK rt_uint8_t rt_interrupt_get_nest(void) | ||||
| { | ||||
|     rt_uint8_t ret; | ||||
|     rt_base_t level; | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     ret = rt_interrupt_nest; | ||||
|     rt_hw_interrupt_enable(level); | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
							
								
								
									
										1349
									
								
								source/rt_thread/src/kservice.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1349
									
								
								source/rt_thread/src/kservice.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										713
									
								
								source/rt_thread/src/mem.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										713
									
								
								source/rt_thread/src/mem.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,713 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2008-7-12      Bernard      the first version | ||||
|  * 2010-06-09     Bernard      fix the end stub of heap | ||||
|  *                             fix memory check in rt_realloc function | ||||
|  * 2010-07-13     Bernard      fix RT_ALIGN issue found by kuronca | ||||
|  * 2010-10-14     Bernard      fix rt_realloc issue when realloc a NULL pointer. | ||||
|  * 2017-07-14     armink       fix rt_realloc issue when new size is 0 | ||||
|  * 2018-10-02     Bernard      Add 64bit support | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Copyright (c) 2001-2004 Swedish Institute of Computer Science. | ||||
|  * All rights reserved. | ||||
|  * | ||||
|  * Redistribution and use in source and binary forms, with or without modification, | ||||
|  * are permitted provided that the following conditions are met: | ||||
|  * | ||||
|  * 1. Redistributions of source code must retain the above copyright notice, | ||||
|  *    this list of conditions and the following disclaimer. | ||||
|  * 2. Redistributions in binary form must reproduce the above copyright notice, | ||||
|  *    this list of conditions and the following disclaimer in the documentation | ||||
|  *    and/or other materials provided with the distribution. | ||||
|  * 3. The name of the author may not be used to endorse or promote products | ||||
|  *    derived from this software without specific prior written permission. | ||||
|  * | ||||
|  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED | ||||
|  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF | ||||
|  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT | ||||
|  * SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | ||||
|  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT | ||||
|  * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||
|  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | ||||
|  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING | ||||
|  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY | ||||
|  * OF SUCH DAMAGE. | ||||
|  * | ||||
|  * This file is part of the lwIP TCP/IP stack. | ||||
|  * | ||||
|  * Author: Adam Dunkels <adam@sics.se> | ||||
|  *         Simon Goldschmidt | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #ifndef RT_USING_MEMHEAP_AS_HEAP | ||||
|  | ||||
| /* #define RT_MEM_DEBUG */ | ||||
| #define RT_MEM_STATS | ||||
|  | ||||
| #if defined (RT_USING_HEAP) && defined (RT_USING_SMALL_MEM) | ||||
| #ifdef RT_USING_HOOK | ||||
| static void (*rt_malloc_hook)(void *ptr, rt_size_t size); | ||||
| static void (*rt_free_hook)(void *ptr); | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Hook | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when a memory | ||||
|  * block is allocated from heap memory. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_malloc_sethook(void (*hook)(void *ptr, rt_size_t size)) | ||||
| { | ||||
|     rt_malloc_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when a memory | ||||
|  * block is released to heap memory. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_free_sethook(void (*hook)(void *ptr)) | ||||
| { | ||||
|     rt_free_hook = hook; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
|  | ||||
| #endif | ||||
|  | ||||
| #define HEAP_MAGIC 0x1ea0 | ||||
| struct heap_mem | ||||
| { | ||||
|     /* magic and used flag */ | ||||
|     rt_uint16_t magic; | ||||
|     rt_uint16_t used; | ||||
| #ifdef ARCH_CPU_64BIT | ||||
|     rt_uint32_t resv; | ||||
| #endif | ||||
|  | ||||
|     rt_size_t next, prev; | ||||
|  | ||||
| #ifdef RT_USING_MEMTRACE | ||||
| #ifdef ARCH_CPU_64BIT | ||||
|     rt_uint8_t thread[8]; | ||||
| #else | ||||
|     rt_uint8_t thread[4];   /* thread name */ | ||||
| #endif | ||||
| #endif | ||||
| }; | ||||
|  | ||||
| /** pointer to the heap: for alignment, heap_ptr is now a pointer instead of an array */ | ||||
| static rt_uint8_t *heap_ptr; | ||||
|  | ||||
| /** the last entry, always unused! */ | ||||
| static struct heap_mem *heap_end; | ||||
|  | ||||
| #ifdef ARCH_CPU_64BIT | ||||
| #define MIN_SIZE 24 | ||||
| #else | ||||
| #define MIN_SIZE 12 | ||||
| #endif | ||||
|  | ||||
| #define MIN_SIZE_ALIGNED     RT_ALIGN(MIN_SIZE, RT_ALIGN_SIZE) | ||||
| #define SIZEOF_STRUCT_MEM    RT_ALIGN(sizeof(struct heap_mem), RT_ALIGN_SIZE) | ||||
|  | ||||
| static struct heap_mem *lfree;   /* pointer to the lowest free block */ | ||||
|  | ||||
| static struct rt_semaphore heap_sem; | ||||
| static rt_size_t mem_size_aligned; | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
| static rt_size_t used_mem, max_mem; | ||||
| #endif | ||||
| #ifdef RT_USING_MEMTRACE | ||||
| rt_inline void rt_mem_setname(struct heap_mem *mem, const char *name) | ||||
| { | ||||
|     int index; | ||||
|     for (index = 0; index < sizeof(mem->thread); index ++) | ||||
|     { | ||||
|         if (name[index] == '\0') break; | ||||
|         mem->thread[index] = name[index]; | ||||
|     } | ||||
|  | ||||
|     for (; index < sizeof(mem->thread); index ++) | ||||
|     { | ||||
|         mem->thread[index] = ' '; | ||||
|     } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| static void plug_holes(struct heap_mem *mem) | ||||
| { | ||||
|     struct heap_mem *nmem; | ||||
|     struct heap_mem *pmem; | ||||
|  | ||||
|     RT_ASSERT((rt_uint8_t *)mem >= heap_ptr); | ||||
|     RT_ASSERT((rt_uint8_t *)mem < (rt_uint8_t *)heap_end); | ||||
|     RT_ASSERT(mem->used == 0); | ||||
|  | ||||
|     /* plug hole forward */ | ||||
|     nmem = (struct heap_mem *)&heap_ptr[mem->next]; | ||||
|     if (mem != nmem && | ||||
|         nmem->used == 0 && | ||||
|         (rt_uint8_t *)nmem != (rt_uint8_t *)heap_end) | ||||
|     { | ||||
|         /* if mem->next is unused and not end of heap_ptr, | ||||
|          * combine mem and mem->next | ||||
|          */ | ||||
|         if (lfree == nmem) | ||||
|         { | ||||
|             lfree = mem; | ||||
|         } | ||||
|         mem->next = nmem->next; | ||||
|         ((struct heap_mem *)&heap_ptr[nmem->next])->prev = (rt_uint8_t *)mem - heap_ptr; | ||||
|     } | ||||
|  | ||||
|     /* plug hole backward */ | ||||
|     pmem = (struct heap_mem *)&heap_ptr[mem->prev]; | ||||
|     if (pmem != mem && pmem->used == 0) | ||||
|     { | ||||
|         /* if mem->prev is unused, combine mem and mem->prev */ | ||||
|         if (lfree == mem) | ||||
|         { | ||||
|             lfree = pmem; | ||||
|         } | ||||
|         pmem->next = mem->next; | ||||
|         ((struct heap_mem *)&heap_ptr[mem->next])->prev = (rt_uint8_t *)pmem - heap_ptr; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * | ||||
|  * This function will initialize system heap memory. | ||||
|  * | ||||
|  * @param begin_addr the beginning address of system heap memory. | ||||
|  * @param end_addr the end address of system heap memory. | ||||
|  */ | ||||
| void rt_system_heap_init(void *begin_addr, void *end_addr) | ||||
| { | ||||
|     struct heap_mem *mem; | ||||
|     rt_ubase_t begin_align = RT_ALIGN((rt_ubase_t)begin_addr, RT_ALIGN_SIZE); | ||||
|     rt_ubase_t end_align   = RT_ALIGN_DOWN((rt_ubase_t)end_addr, RT_ALIGN_SIZE); | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     /* alignment addr */ | ||||
|     if ((end_align > (2 * SIZEOF_STRUCT_MEM)) && | ||||
|         ((end_align - 2 * SIZEOF_STRUCT_MEM) >= begin_align)) | ||||
|     { | ||||
|         /* calculate the aligned memory size */ | ||||
|         mem_size_aligned = end_align - begin_align - 2 * SIZEOF_STRUCT_MEM; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         rt_kprintf("mem init, error begin address 0x%x, and end address 0x%x\n", | ||||
|                    (rt_ubase_t)begin_addr, (rt_ubase_t)end_addr); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     /* point to begin address of heap */ | ||||
|     heap_ptr = (rt_uint8_t *)begin_align; | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEM, ("mem init, heap begin address 0x%x, size %d\n", | ||||
|                                 (rt_ubase_t)heap_ptr, mem_size_aligned)); | ||||
|  | ||||
|     /* initialize the start of the heap */ | ||||
|     mem        = (struct heap_mem *)heap_ptr; | ||||
|     mem->magic = HEAP_MAGIC; | ||||
|     mem->next  = mem_size_aligned + SIZEOF_STRUCT_MEM; | ||||
|     mem->prev  = 0; | ||||
|     mem->used  = 0; | ||||
| #ifdef RT_USING_MEMTRACE | ||||
|     rt_mem_setname(mem, "INIT"); | ||||
| #endif | ||||
|  | ||||
|     /* initialize the end of the heap */ | ||||
|     heap_end        = (struct heap_mem *)&heap_ptr[mem->next]; | ||||
|     heap_end->magic = HEAP_MAGIC; | ||||
|     heap_end->used  = 1; | ||||
|     heap_end->next  = mem_size_aligned + SIZEOF_STRUCT_MEM; | ||||
|     heap_end->prev  = mem_size_aligned + SIZEOF_STRUCT_MEM; | ||||
| #ifdef RT_USING_MEMTRACE | ||||
|     rt_mem_setname(heap_end, "INIT"); | ||||
| #endif | ||||
|  | ||||
|     rt_sem_init(&heap_sem, "heap", 1, RT_IPC_FLAG_FIFO); | ||||
|  | ||||
|     /* initialize the lowest-free pointer to the start of the heap */ | ||||
|     lfree = (struct heap_mem *)heap_ptr; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @addtogroup MM | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * Allocate a block of memory with a minimum of 'size' bytes. | ||||
|  * | ||||
|  * @param size is the minimum size of the requested block in bytes. | ||||
|  * | ||||
|  * @return pointer to allocated memory or NULL if no free memory was found. | ||||
|  */ | ||||
| void *rt_malloc(rt_size_t size) | ||||
| { | ||||
|     rt_size_t ptr, ptr2; | ||||
|     struct heap_mem *mem, *mem2; | ||||
|  | ||||
|     if (size == 0) | ||||
|         return RT_NULL; | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     if (size != RT_ALIGN(size, RT_ALIGN_SIZE)) | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEM, ("malloc size %d, but align to %d\n", | ||||
|                                     size, RT_ALIGN(size, RT_ALIGN_SIZE))); | ||||
|     else | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEM, ("malloc size %d\n", size)); | ||||
|  | ||||
|     /* alignment size */ | ||||
|     size = RT_ALIGN(size, RT_ALIGN_SIZE); | ||||
|  | ||||
|     if (size > mem_size_aligned) | ||||
|     { | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEM, ("no memory\n")); | ||||
|  | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     /* every data block must be at least MIN_SIZE_ALIGNED long */ | ||||
|     if (size < MIN_SIZE_ALIGNED) | ||||
|         size = MIN_SIZE_ALIGNED; | ||||
|  | ||||
|     /* take memory semaphore */ | ||||
|     rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
|     for (ptr = (rt_uint8_t *)lfree - heap_ptr; | ||||
|          ptr < mem_size_aligned - size; | ||||
|          ptr = ((struct heap_mem *)&heap_ptr[ptr])->next) | ||||
|     { | ||||
|         mem = (struct heap_mem *)&heap_ptr[ptr]; | ||||
|  | ||||
|         if ((!mem->used) && (mem->next - (ptr + SIZEOF_STRUCT_MEM)) >= size) | ||||
|         { | ||||
|             /* mem is not used and at least perfect fit is possible: | ||||
|              * mem->next - (ptr + SIZEOF_STRUCT_MEM) gives us the 'user data size' of mem */ | ||||
|  | ||||
|             if (mem->next - (ptr + SIZEOF_STRUCT_MEM) >= | ||||
|                 (size + SIZEOF_STRUCT_MEM + MIN_SIZE_ALIGNED)) | ||||
|             { | ||||
|                 /* (in addition to the above, we test if another struct heap_mem (SIZEOF_STRUCT_MEM) containing | ||||
|                  * at least MIN_SIZE_ALIGNED of data also fits in the 'user data space' of 'mem') | ||||
|                  * -> split large block, create empty remainder, | ||||
|                  * remainder must be large enough to contain MIN_SIZE_ALIGNED data: if | ||||
|                  * mem->next - (ptr + (2*SIZEOF_STRUCT_MEM)) == size, | ||||
|                  * struct heap_mem would fit in but no data between mem2 and mem2->next | ||||
|                  * @todo we could leave out MIN_SIZE_ALIGNED. We would create an empty | ||||
|                  *       region that couldn't hold data, but when mem->next gets freed, | ||||
|                  *       the 2 regions would be combined, resulting in more free memory | ||||
|                  */ | ||||
|                 ptr2 = ptr + SIZEOF_STRUCT_MEM + size; | ||||
|  | ||||
|                 /* create mem2 struct */ | ||||
|                 mem2       = (struct heap_mem *)&heap_ptr[ptr2]; | ||||
|                 mem2->magic = HEAP_MAGIC; | ||||
|                 mem2->used = 0; | ||||
|                 mem2->next = mem->next; | ||||
|                 mem2->prev = ptr; | ||||
| #ifdef RT_USING_MEMTRACE | ||||
|                 rt_mem_setname(mem2, "    "); | ||||
| #endif | ||||
|  | ||||
|                 /* and insert it between mem and mem->next */ | ||||
|                 mem->next = ptr2; | ||||
|                 mem->used = 1; | ||||
|  | ||||
|                 if (mem2->next != mem_size_aligned + SIZEOF_STRUCT_MEM) | ||||
|                 { | ||||
|                     ((struct heap_mem *)&heap_ptr[mem2->next])->prev = ptr2; | ||||
|                 } | ||||
| #ifdef RT_MEM_STATS | ||||
|                 used_mem += (size + SIZEOF_STRUCT_MEM); | ||||
|                 if (max_mem < used_mem) | ||||
|                     max_mem = used_mem; | ||||
| #endif | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 /* (a mem2 struct does no fit into the user data space of mem and mem->next will always | ||||
|                  * be used at this point: if not we have 2 unused structs in a row, plug_holes should have | ||||
|                  * take care of this). | ||||
|                  * -> near fit or excact fit: do not split, no mem2 creation | ||||
|                  * also can't move mem->next directly behind mem, since mem->next | ||||
|                  * will always be used at this point! | ||||
|                  */ | ||||
|                 mem->used = 1; | ||||
| #ifdef RT_MEM_STATS | ||||
|                 used_mem += mem->next - ((rt_uint8_t *)mem - heap_ptr); | ||||
|                 if (max_mem < used_mem) | ||||
|                     max_mem = used_mem; | ||||
| #endif | ||||
|             } | ||||
|             /* set memory block magic */ | ||||
|             mem->magic = HEAP_MAGIC; | ||||
| #ifdef RT_USING_MEMTRACE | ||||
|             if (rt_thread_self()) | ||||
|                 rt_mem_setname(mem, rt_thread_self()->name); | ||||
|             else | ||||
|                 rt_mem_setname(mem, "NONE"); | ||||
| #endif | ||||
|  | ||||
|             if (mem == lfree) | ||||
|             { | ||||
|                 /* Find next free block after mem and update lowest free pointer */ | ||||
|                 while (lfree->used && lfree != heap_end) | ||||
|                     lfree = (struct heap_mem *)&heap_ptr[lfree->next]; | ||||
|  | ||||
|                 RT_ASSERT(((lfree == heap_end) || (!lfree->used))); | ||||
|             } | ||||
|  | ||||
|             rt_sem_release(&heap_sem); | ||||
|             RT_ASSERT((rt_ubase_t)mem + SIZEOF_STRUCT_MEM + size <= (rt_ubase_t)heap_end); | ||||
|             RT_ASSERT((rt_ubase_t)((rt_uint8_t *)mem + SIZEOF_STRUCT_MEM) % RT_ALIGN_SIZE == 0); | ||||
|             RT_ASSERT((((rt_ubase_t)mem) & (RT_ALIGN_SIZE - 1)) == 0); | ||||
|  | ||||
|             RT_DEBUG_LOG(RT_DEBUG_MEM, | ||||
|                          ("allocate memory at 0x%x, size: %d\n", | ||||
|                           (rt_ubase_t)((rt_uint8_t *)mem + SIZEOF_STRUCT_MEM), | ||||
|                           (rt_ubase_t)(mem->next - ((rt_uint8_t *)mem - heap_ptr)))); | ||||
|  | ||||
|             RT_OBJECT_HOOK_CALL(rt_malloc_hook, | ||||
|                                 (((void *)((rt_uint8_t *)mem + SIZEOF_STRUCT_MEM)), size)); | ||||
|  | ||||
|             /* return the memory data except mem struct */ | ||||
|             return (rt_uint8_t *)mem + SIZEOF_STRUCT_MEM; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     rt_sem_release(&heap_sem); | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will change the previously allocated memory block. | ||||
|  * | ||||
|  * @param rmem pointer to memory allocated by rt_malloc | ||||
|  * @param newsize the required new size | ||||
|  * | ||||
|  * @return the changed memory block address | ||||
|  */ | ||||
| void *rt_realloc(void *rmem, rt_size_t newsize) | ||||
| { | ||||
|     rt_size_t size; | ||||
|     rt_size_t ptr, ptr2; | ||||
|     struct heap_mem *mem, *mem2; | ||||
|     void *nmem; | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     /* alignment size */ | ||||
|     newsize = RT_ALIGN(newsize, RT_ALIGN_SIZE); | ||||
|     if (newsize > mem_size_aligned) | ||||
|     { | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEM, ("realloc: out of memory\n")); | ||||
|  | ||||
|         return RT_NULL; | ||||
|     } | ||||
|     else if (newsize == 0) | ||||
|     { | ||||
|         rt_free(rmem); | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     /* allocate a new memory block */ | ||||
|     if (rmem == RT_NULL) | ||||
|         return rt_malloc(newsize); | ||||
|  | ||||
|     rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
|     if ((rt_uint8_t *)rmem < (rt_uint8_t *)heap_ptr || | ||||
|         (rt_uint8_t *)rmem >= (rt_uint8_t *)heap_end) | ||||
|     { | ||||
|         /* illegal memory */ | ||||
|         rt_sem_release(&heap_sem); | ||||
|  | ||||
|         return rmem; | ||||
|     } | ||||
|  | ||||
|     mem = (struct heap_mem *)((rt_uint8_t *)rmem - SIZEOF_STRUCT_MEM); | ||||
|  | ||||
|     ptr = (rt_uint8_t *)mem - heap_ptr; | ||||
|     size = mem->next - ptr - SIZEOF_STRUCT_MEM; | ||||
|     if (size == newsize) | ||||
|     { | ||||
|         /* the size is the same as */ | ||||
|         rt_sem_release(&heap_sem); | ||||
|  | ||||
|         return rmem; | ||||
|     } | ||||
|  | ||||
|     if (newsize + SIZEOF_STRUCT_MEM + MIN_SIZE < size) | ||||
|     { | ||||
|         /* split memory block */ | ||||
| #ifdef RT_MEM_STATS | ||||
|         used_mem -= (size - newsize); | ||||
| #endif | ||||
|  | ||||
|         ptr2 = ptr + SIZEOF_STRUCT_MEM + newsize; | ||||
|         mem2 = (struct heap_mem *)&heap_ptr[ptr2]; | ||||
|         mem2->magic = HEAP_MAGIC; | ||||
|         mem2->used = 0; | ||||
|         mem2->next = mem->next; | ||||
|         mem2->prev = ptr; | ||||
| #ifdef RT_USING_MEMTRACE | ||||
|         rt_mem_setname(mem2, "    "); | ||||
| #endif | ||||
|         mem->next = ptr2; | ||||
|         if (mem2->next != mem_size_aligned + SIZEOF_STRUCT_MEM) | ||||
|         { | ||||
|             ((struct heap_mem *)&heap_ptr[mem2->next])->prev = ptr2; | ||||
|         } | ||||
|  | ||||
|         if (mem2 < lfree) | ||||
|         { | ||||
|             /* the splited struct is now the lowest */ | ||||
|             lfree = mem2; | ||||
|         } | ||||
|  | ||||
|         plug_holes(mem2); | ||||
|  | ||||
|         rt_sem_release(&heap_sem); | ||||
|  | ||||
|         return rmem; | ||||
|     } | ||||
|     rt_sem_release(&heap_sem); | ||||
|  | ||||
|     /* expand memory */ | ||||
|     nmem = rt_malloc(newsize); | ||||
|     if (nmem != RT_NULL) /* check memory */ | ||||
|     { | ||||
|         rt_memcpy(nmem, rmem, size < newsize ? size : newsize); | ||||
|         rt_free(rmem); | ||||
|     } | ||||
|  | ||||
|     return nmem; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will contiguously allocate enough space for count objects | ||||
|  * that are size bytes of memory each and returns a pointer to the allocated | ||||
|  * memory. | ||||
|  * | ||||
|  * The allocated memory is filled with bytes of value zero. | ||||
|  * | ||||
|  * @param count number of objects to allocate | ||||
|  * @param size size of the objects to allocate | ||||
|  * | ||||
|  * @return pointer to allocated memory / NULL pointer if there is an error | ||||
|  */ | ||||
| void *rt_calloc(rt_size_t count, rt_size_t size) | ||||
| { | ||||
|     void *p; | ||||
|  | ||||
|     /* allocate 'count' objects of size 'size' */ | ||||
|     p = rt_malloc(count * size); | ||||
|  | ||||
|     /* zero the memory */ | ||||
|     if (p) | ||||
|         rt_memset(p, 0, count * size); | ||||
|  | ||||
|     return p; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will release the previously allocated memory block by | ||||
|  * rt_malloc. The released memory block is taken back to system heap. | ||||
|  * | ||||
|  * @param rmem the address of memory which will be released | ||||
|  */ | ||||
| void rt_free(void *rmem) | ||||
| { | ||||
|     struct heap_mem *mem; | ||||
|  | ||||
|     if (rmem == RT_NULL) | ||||
|         return; | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     RT_ASSERT((((rt_ubase_t)rmem) & (RT_ALIGN_SIZE - 1)) == 0); | ||||
|     RT_ASSERT((rt_uint8_t *)rmem >= (rt_uint8_t *)heap_ptr && | ||||
|               (rt_uint8_t *)rmem < (rt_uint8_t *)heap_end); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_free_hook, (rmem)); | ||||
|  | ||||
|     if ((rt_uint8_t *)rmem < (rt_uint8_t *)heap_ptr || | ||||
|         (rt_uint8_t *)rmem >= (rt_uint8_t *)heap_end) | ||||
|     { | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEM, ("illegal memory\n")); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     /* Get the corresponding struct heap_mem ... */ | ||||
|     mem = (struct heap_mem *)((rt_uint8_t *)rmem - SIZEOF_STRUCT_MEM); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEM, | ||||
|                  ("release memory 0x%x, size: %d\n", | ||||
|                   (rt_ubase_t)rmem, | ||||
|                   (rt_ubase_t)(mem->next - ((rt_uint8_t *)mem - heap_ptr)))); | ||||
|  | ||||
|  | ||||
|     /* protect the heap from concurrent access */ | ||||
|     rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
|     /* ... which has to be in a used state ... */ | ||||
|     if (!mem->used || mem->magic != HEAP_MAGIC) | ||||
|     { | ||||
|         rt_kprintf("to free a bad data block:\n"); | ||||
|         rt_kprintf("mem: 0x%08x, used flag: %d, magic code: 0x%04x\n", mem, mem->used, mem->magic); | ||||
|     } | ||||
|     RT_ASSERT(mem->used); | ||||
|     RT_ASSERT(mem->magic == HEAP_MAGIC); | ||||
|     /* ... and is now unused. */ | ||||
|     mem->used  = 0; | ||||
|     mem->magic = HEAP_MAGIC; | ||||
| #ifdef RT_USING_MEMTRACE | ||||
|     rt_mem_setname(mem, "    "); | ||||
| #endif | ||||
|  | ||||
|     if (mem < lfree) | ||||
|     { | ||||
|         /* the newly freed struct is now the lowest */ | ||||
|         lfree = mem; | ||||
|     } | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
|     used_mem -= (mem->next - ((rt_uint8_t *)mem - heap_ptr)); | ||||
| #endif | ||||
|  | ||||
|     /* finally, see if prev or next are free also */ | ||||
|     plug_holes(mem); | ||||
|     rt_sem_release(&heap_sem); | ||||
| } | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
| void rt_memory_info(rt_uint32_t *total, | ||||
|                     rt_uint32_t *used, | ||||
|                     rt_uint32_t *max_used) | ||||
| { | ||||
|     if (total != RT_NULL) | ||||
|         *total = mem_size_aligned; | ||||
|     if (used  != RT_NULL) | ||||
|         *used = used_mem; | ||||
|     if (max_used != RT_NULL) | ||||
|         *max_used = max_mem; | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_FINSH | ||||
| #include <finsh.h> | ||||
|  | ||||
| void list_mem(void) | ||||
| { | ||||
|     rt_kprintf("total memory: %d\n", mem_size_aligned); | ||||
|     rt_kprintf("used memory : %d\n", used_mem); | ||||
|     rt_kprintf("maximum allocated memory: %d\n", max_mem); | ||||
| } | ||||
| FINSH_FUNCTION_EXPORT(list_mem, list memory usage information) | ||||
|  | ||||
| #ifdef RT_USING_MEMTRACE | ||||
| int memcheck(void) | ||||
| { | ||||
|     int position; | ||||
|     rt_ubase_t level; | ||||
|     struct heap_mem *mem; | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     for (mem = (struct heap_mem *)heap_ptr; mem != heap_end; mem = (struct heap_mem *)&heap_ptr[mem->next]) | ||||
|     { | ||||
|         position = (rt_ubase_t)mem - (rt_ubase_t)heap_ptr; | ||||
|         if (position < 0) goto __exit; | ||||
|         if (position > (int)mem_size_aligned) goto __exit; | ||||
|         if (mem->magic != HEAP_MAGIC) goto __exit; | ||||
|         if (mem->used != 0 && mem->used != 1) goto __exit; | ||||
|     } | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return 0; | ||||
| __exit: | ||||
|     rt_kprintf("Memory block wrong:\n"); | ||||
|     rt_kprintf("address: 0x%08x\n", mem); | ||||
|     rt_kprintf("  magic: 0x%04x\n", mem->magic); | ||||
|     rt_kprintf("   used: %d\n", mem->used); | ||||
|     rt_kprintf("  size: %d\n", mem->next - position - SIZEOF_STRUCT_MEM); | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
| MSH_CMD_EXPORT(memcheck, check memory data); | ||||
|  | ||||
| int memtrace(int argc, char **argv) | ||||
| { | ||||
|     struct heap_mem *mem; | ||||
|  | ||||
|     list_mem(); | ||||
|  | ||||
|     rt_kprintf("\nmemory heap address:\n"); | ||||
|     rt_kprintf("heap_ptr: 0x%08x\n", heap_ptr); | ||||
|     rt_kprintf("lfree   : 0x%08x\n", lfree); | ||||
|     rt_kprintf("heap_end: 0x%08x\n", heap_end); | ||||
|  | ||||
|     rt_kprintf("\n--memory item information --\n"); | ||||
|     for (mem = (struct heap_mem *)heap_ptr; mem != heap_end; mem = (struct heap_mem *)&heap_ptr[mem->next]) | ||||
|     { | ||||
|         int position = (rt_ubase_t)mem - (rt_ubase_t)heap_ptr; | ||||
|         int size; | ||||
|  | ||||
|         rt_kprintf("[0x%08x - ", mem); | ||||
|  | ||||
|         size = mem->next - position - SIZEOF_STRUCT_MEM; | ||||
|         if (size < 1024) | ||||
|             rt_kprintf("%5d", size); | ||||
|         else if (size < 1024 * 1024) | ||||
|             rt_kprintf("%4dK", size / 1024); | ||||
|         else | ||||
|             rt_kprintf("%4dM", size / (1024 * 1024)); | ||||
|  | ||||
|         rt_kprintf("] %c%c%c%c", mem->thread[0], mem->thread[1], mem->thread[2], mem->thread[3]); | ||||
|         if (mem->magic != HEAP_MAGIC) | ||||
|             rt_kprintf(": ***\n"); | ||||
|         else | ||||
|             rt_kprintf("\n"); | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
| MSH_CMD_EXPORT(memtrace, dump memory trace information); | ||||
| #endif /* end of RT_USING_MEMTRACE */ | ||||
| #endif /* end of RT_USING_FINSH    */ | ||||
|  | ||||
| #endif | ||||
|  | ||||
| /**@}*/ | ||||
|  | ||||
| #endif /* end of RT_USING_HEAP */ | ||||
| #endif /* end of RT_USING_MEMHEAP_AS_HEAP */ | ||||
							
								
								
									
										722
									
								
								source/rt_thread/src/memheap.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										722
									
								
								source/rt_thread/src/memheap.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,722 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * File      : memheap.c | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2012-04-10     Bernard      first implementation | ||||
|  * 2012-10-16     Bernard      add the mutex lock for heap object. | ||||
|  * 2012-12-29     Bernard      memheap can be used as system heap. | ||||
|  *                             change mutex lock to semaphore lock. | ||||
|  * 2013-04-10     Bernard      add rt_memheap_realloc function. | ||||
|  * 2013-05-24     Bernard      fix the rt_memheap_realloc issue. | ||||
|  * 2013-07-11     Grissiom     fix the memory block splitting issue. | ||||
|  * 2013-07-15     Grissiom     optimize rt_memheap_realloc | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #ifdef RT_USING_MEMHEAP | ||||
|  | ||||
| /* dynamic pool magic and mask */ | ||||
| #define RT_MEMHEAP_MAGIC        0x1ea01ea0 | ||||
| #define RT_MEMHEAP_MASK         0xfffffffe | ||||
| #define RT_MEMHEAP_USED         0x01 | ||||
| #define RT_MEMHEAP_FREED        0x00 | ||||
|  | ||||
| #define RT_MEMHEAP_IS_USED(i)   ((i)->magic & RT_MEMHEAP_USED) | ||||
| #define RT_MEMHEAP_MINIALLOC    12 | ||||
|  | ||||
| #define RT_MEMHEAP_SIZE         RT_ALIGN(sizeof(struct rt_memheap_item), RT_ALIGN_SIZE) | ||||
| #define MEMITEM_SIZE(item)      ((rt_ubase_t)item->next - (rt_ubase_t)item - RT_MEMHEAP_SIZE) | ||||
|  | ||||
| /* | ||||
|  * The initialized memory pool will be: | ||||
|  * +-----------------------------------+--------------------------+ | ||||
|  * | whole freed memory block          | Used Memory Block Tailer | | ||||
|  * +-----------------------------------+--------------------------+ | ||||
|  * | ||||
|  * block_list --> whole freed memory block | ||||
|  * | ||||
|  * The length of Used Memory Block Tailer is 0, | ||||
|  * which is prevents block merging across list | ||||
|  */ | ||||
| rt_err_t rt_memheap_init(struct rt_memheap *memheap, | ||||
|                          const char        *name, | ||||
|                          void              *start_addr, | ||||
|                          rt_size_t         size) | ||||
| { | ||||
|     struct rt_memheap_item *item; | ||||
|  | ||||
|     RT_ASSERT(memheap != RT_NULL); | ||||
|  | ||||
|     /* initialize pool object */ | ||||
|     rt_object_init(&(memheap->parent), RT_Object_Class_MemHeap, name); | ||||
|  | ||||
|     memheap->start_addr     = start_addr; | ||||
|     memheap->pool_size      = RT_ALIGN_DOWN(size, RT_ALIGN_SIZE); | ||||
|     memheap->available_size = memheap->pool_size - (2 * RT_MEMHEAP_SIZE); | ||||
|     memheap->max_used_size  = memheap->pool_size - memheap->available_size; | ||||
|  | ||||
|     /* initialize the free list header */ | ||||
|     item            = &(memheap->free_header); | ||||
|     item->magic     = RT_MEMHEAP_MAGIC; | ||||
|     item->pool_ptr  = memheap; | ||||
|     item->next      = RT_NULL; | ||||
|     item->prev      = RT_NULL; | ||||
|     item->next_free = item; | ||||
|     item->prev_free = item; | ||||
|  | ||||
|     /* set the free list to free list header */ | ||||
|     memheap->free_list = item; | ||||
|  | ||||
|     /* initialize the first big memory block */ | ||||
|     item            = (struct rt_memheap_item *)start_addr; | ||||
|     item->magic     = RT_MEMHEAP_MAGIC; | ||||
|     item->pool_ptr  = memheap; | ||||
|     item->next      = RT_NULL; | ||||
|     item->prev      = RT_NULL; | ||||
|     item->next_free = item; | ||||
|     item->prev_free = item; | ||||
|  | ||||
|     item->next = (struct rt_memheap_item *) | ||||
|                  ((rt_uint8_t *)item + memheap->available_size + RT_MEMHEAP_SIZE); | ||||
|     item->prev = item->next; | ||||
|  | ||||
|     /* block list header */ | ||||
|     memheap->block_list = item; | ||||
|  | ||||
|     /* place the big memory block to free list */ | ||||
|     item->next_free = memheap->free_list->next_free; | ||||
|     item->prev_free = memheap->free_list; | ||||
|     memheap->free_list->next_free->prev_free = item; | ||||
|     memheap->free_list->next_free            = item; | ||||
|  | ||||
|     /* move to the end of memory pool to build a small tailer block, | ||||
|      * which prevents block merging | ||||
|      */ | ||||
|     item = item->next; | ||||
|     /* it's a used memory block */ | ||||
|     item->magic     = RT_MEMHEAP_MAGIC | RT_MEMHEAP_USED; | ||||
|     item->pool_ptr  = memheap; | ||||
|     item->next      = (struct rt_memheap_item *)start_addr; | ||||
|     item->prev      = (struct rt_memheap_item *)start_addr; | ||||
|     /* not in free list */ | ||||
|     item->next_free = item->prev_free = RT_NULL; | ||||
|  | ||||
|     /* initialize semaphore lock */ | ||||
|     rt_sem_init(&(memheap->lock), name, 1, RT_IPC_FLAG_FIFO); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                  ("memory heap: start addr 0x%08x, size %d, free list header 0x%08x\n", | ||||
|                   start_addr, size, &(memheap->free_header))); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| rt_err_t rt_memheap_detach(struct rt_memheap *heap) | ||||
| { | ||||
|     RT_ASSERT(heap); | ||||
|     RT_ASSERT(rt_object_get_type(&heap->parent) == RT_Object_Class_MemHeap); | ||||
|     RT_ASSERT(rt_object_is_systemobject(&heap->parent)); | ||||
|  | ||||
|     rt_sem_detach(&heap->lock); | ||||
|     rt_object_detach(&(heap->parent)); | ||||
|  | ||||
|     /* Return a successful completion. */ | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| void *rt_memheap_alloc(struct rt_memheap *heap, rt_size_t size) | ||||
| { | ||||
|     rt_err_t result; | ||||
|     rt_uint32_t free_size; | ||||
|     struct rt_memheap_item *header_ptr; | ||||
|  | ||||
|     RT_ASSERT(heap != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&heap->parent) == RT_Object_Class_MemHeap); | ||||
|  | ||||
|     /* align allocated size */ | ||||
|     size = RT_ALIGN(size, RT_ALIGN_SIZE); | ||||
|     if (size < RT_MEMHEAP_MINIALLOC) | ||||
|         size = RT_MEMHEAP_MINIALLOC; | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, ("allocate %d on heap:%8.*s", | ||||
|                                     size, RT_NAME_MAX, heap->parent.name)); | ||||
|  | ||||
|     if (size < heap->available_size) | ||||
|     { | ||||
|         /* search on free list */ | ||||
|         free_size = 0; | ||||
|  | ||||
|         /* lock memheap */ | ||||
|         result = rt_sem_take(&(heap->lock), RT_WAITING_FOREVER); | ||||
|         if (result != RT_EOK) | ||||
|         { | ||||
|             rt_set_errno(result); | ||||
|  | ||||
|             return RT_NULL; | ||||
|         } | ||||
|  | ||||
|         /* get the first free memory block */ | ||||
|         header_ptr = heap->free_list->next_free; | ||||
|         while (header_ptr != heap->free_list && free_size < size) | ||||
|         { | ||||
|             /* get current freed memory block size */ | ||||
|             free_size = MEMITEM_SIZE(header_ptr); | ||||
|             if (free_size < size) | ||||
|             { | ||||
|                 /* move to next free memory block */ | ||||
|                 header_ptr = header_ptr->next_free; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* determine if the memory is available. */ | ||||
|         if (free_size >= size) | ||||
|         { | ||||
|             /* a block that satisfies the request has been found. */ | ||||
|  | ||||
|             /* determine if the block needs to be split. */ | ||||
|             if (free_size >= (size + RT_MEMHEAP_SIZE + RT_MEMHEAP_MINIALLOC)) | ||||
|             { | ||||
|                 struct rt_memheap_item *new_ptr; | ||||
|  | ||||
|                 /* split the block. */ | ||||
|                 new_ptr = (struct rt_memheap_item *) | ||||
|                           (((rt_uint8_t *)header_ptr) + size + RT_MEMHEAP_SIZE); | ||||
|  | ||||
|                 RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                              ("split: block[0x%08x] nextm[0x%08x] prevm[0x%08x] to new[0x%08x]\n", | ||||
|                               header_ptr, | ||||
|                               header_ptr->next, | ||||
|                               header_ptr->prev, | ||||
|                               new_ptr)); | ||||
|  | ||||
|                 /* mark the new block as a memory block and freed. */ | ||||
|                 new_ptr->magic = RT_MEMHEAP_MAGIC; | ||||
|  | ||||
|                 /* put the pool pointer into the new block. */ | ||||
|                 new_ptr->pool_ptr = heap; | ||||
|  | ||||
|                 /* break down the block list */ | ||||
|                 new_ptr->prev          = header_ptr; | ||||
|                 new_ptr->next          = header_ptr->next; | ||||
|                 header_ptr->next->prev = new_ptr; | ||||
|                 header_ptr->next       = new_ptr; | ||||
|  | ||||
|                 /* remove header ptr from free list */ | ||||
|                 header_ptr->next_free->prev_free = header_ptr->prev_free; | ||||
|                 header_ptr->prev_free->next_free = header_ptr->next_free; | ||||
|                 header_ptr->next_free = RT_NULL; | ||||
|                 header_ptr->prev_free = RT_NULL; | ||||
|  | ||||
|                 /* insert new_ptr to free list */ | ||||
|                 new_ptr->next_free = heap->free_list->next_free; | ||||
|                 new_ptr->prev_free = heap->free_list; | ||||
|                 heap->free_list->next_free->prev_free = new_ptr; | ||||
|                 heap->free_list->next_free            = new_ptr; | ||||
|                 RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, ("new ptr: next_free 0x%08x, prev_free 0x%08x\n", | ||||
|                                                 new_ptr->next_free, | ||||
|                                                 new_ptr->prev_free)); | ||||
|  | ||||
|                 /* decrement the available byte count.  */ | ||||
|                 heap->available_size = heap->available_size - | ||||
|                                        size - | ||||
|                                        RT_MEMHEAP_SIZE; | ||||
|                 if (heap->pool_size - heap->available_size > heap->max_used_size) | ||||
|                     heap->max_used_size = heap->pool_size - heap->available_size; | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 /* decrement the entire free size from the available bytes count. */ | ||||
|                 heap->available_size = heap->available_size - free_size; | ||||
|                 if (heap->pool_size - heap->available_size > heap->max_used_size) | ||||
|                     heap->max_used_size = heap->pool_size - heap->available_size; | ||||
|  | ||||
|                 /* remove header_ptr from free list */ | ||||
|                 RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                              ("one block: block[0x%08x], next_free 0x%08x, prev_free 0x%08x\n", | ||||
|                               header_ptr, | ||||
|                               header_ptr->next_free, | ||||
|                               header_ptr->prev_free)); | ||||
|  | ||||
|                 header_ptr->next_free->prev_free = header_ptr->prev_free; | ||||
|                 header_ptr->prev_free->next_free = header_ptr->next_free; | ||||
|                 header_ptr->next_free = RT_NULL; | ||||
|                 header_ptr->prev_free = RT_NULL; | ||||
|             } | ||||
|  | ||||
|             /* Mark the allocated block as not available. */ | ||||
|             header_ptr->magic |= RT_MEMHEAP_USED; | ||||
|  | ||||
|             /* release lock */ | ||||
|             rt_sem_release(&(heap->lock)); | ||||
|  | ||||
|             /* Return a memory address to the caller.  */ | ||||
|             RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                          ("alloc mem: memory[0x%08x], heap[0x%08x], size: %d\n", | ||||
|                           (void *)((rt_uint8_t *)header_ptr + RT_MEMHEAP_SIZE), | ||||
|                           header_ptr, | ||||
|                           size)); | ||||
|  | ||||
|             return (void *)((rt_uint8_t *)header_ptr + RT_MEMHEAP_SIZE); | ||||
|         } | ||||
|  | ||||
|         /* release lock */ | ||||
|         rt_sem_release(&(heap->lock)); | ||||
|     } | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, ("allocate memory: failed\n")); | ||||
|  | ||||
|     /* Return the completion status.  */ | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| void *rt_memheap_realloc(struct rt_memheap *heap, void *ptr, rt_size_t newsize) | ||||
| { | ||||
|     rt_err_t result; | ||||
|     rt_size_t oldsize; | ||||
|     struct rt_memheap_item *header_ptr; | ||||
|     struct rt_memheap_item *new_ptr; | ||||
|  | ||||
|     RT_ASSERT(heap); | ||||
|     RT_ASSERT(rt_object_get_type(&heap->parent) == RT_Object_Class_MemHeap); | ||||
|  | ||||
|     if (newsize == 0) | ||||
|     { | ||||
|         rt_memheap_free(ptr); | ||||
|  | ||||
|         return RT_NULL; | ||||
|     } | ||||
|     /* align allocated size */ | ||||
|     newsize = RT_ALIGN(newsize, RT_ALIGN_SIZE); | ||||
|     if (newsize < RT_MEMHEAP_MINIALLOC) | ||||
|         newsize = RT_MEMHEAP_MINIALLOC; | ||||
|  | ||||
|     if (ptr == RT_NULL) | ||||
|     { | ||||
|         return rt_memheap_alloc(heap, newsize); | ||||
|     } | ||||
|  | ||||
|     /* get memory block header and get the size of memory block */ | ||||
|     header_ptr = (struct rt_memheap_item *) | ||||
|                  ((rt_uint8_t *)ptr - RT_MEMHEAP_SIZE); | ||||
|     oldsize = MEMITEM_SIZE(header_ptr); | ||||
|     /* re-allocate memory */ | ||||
|     if (newsize > oldsize) | ||||
|     { | ||||
|         void *new_ptr; | ||||
|         struct rt_memheap_item *next_ptr; | ||||
|  | ||||
|         /* lock memheap */ | ||||
|         result = rt_sem_take(&(heap->lock), RT_WAITING_FOREVER); | ||||
|         if (result != RT_EOK) | ||||
|         { | ||||
|             rt_set_errno(result); | ||||
|             return RT_NULL; | ||||
|         } | ||||
|  | ||||
|         next_ptr = header_ptr->next; | ||||
|  | ||||
|         /* header_ptr should not be the tail */ | ||||
|         RT_ASSERT(next_ptr > header_ptr); | ||||
|  | ||||
|         /* check whether the following free space is enough to expand */ | ||||
|         if (!RT_MEMHEAP_IS_USED(next_ptr)) | ||||
|         { | ||||
|             rt_int32_t nextsize; | ||||
|  | ||||
|             nextsize = MEMITEM_SIZE(next_ptr); | ||||
|             RT_ASSERT(next_ptr > 0); | ||||
|  | ||||
|             /* Here is the ASCII art of the situation that we can make use of | ||||
|              * the next free node without alloc/memcpy, |*| is the control | ||||
|              * block: | ||||
|              * | ||||
|              *      oldsize           free node | ||||
|              * |*|-----------|*|----------------------|*| | ||||
|              *         newsize          >= minialloc | ||||
|              * |*|----------------|*|-----------------|*| | ||||
|              */ | ||||
|             if (nextsize + oldsize > newsize + RT_MEMHEAP_MINIALLOC) | ||||
|             { | ||||
|                 /* decrement the entire free size from the available bytes count. */ | ||||
|                 heap->available_size = heap->available_size - (newsize - oldsize); | ||||
|                 if (heap->pool_size - heap->available_size > heap->max_used_size) | ||||
|                     heap->max_used_size = heap->pool_size - heap->available_size; | ||||
|  | ||||
|                 /* remove next_ptr from free list */ | ||||
|                 RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                              ("remove block: block[0x%08x], next_free 0x%08x, prev_free 0x%08x", | ||||
|                               next_ptr, | ||||
|                               next_ptr->next_free, | ||||
|                               next_ptr->prev_free)); | ||||
|  | ||||
|                 next_ptr->next_free->prev_free = next_ptr->prev_free; | ||||
|                 next_ptr->prev_free->next_free = next_ptr->next_free; | ||||
|                 next_ptr->next->prev = next_ptr->prev; | ||||
|                 next_ptr->prev->next = next_ptr->next; | ||||
|  | ||||
|                 /* build a new one on the right place */ | ||||
|                 next_ptr = (struct rt_memheap_item *)((char *)ptr + newsize); | ||||
|  | ||||
|                 RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                              ("new free block: block[0x%08x] nextm[0x%08x] prevm[0x%08x]", | ||||
|                               next_ptr, | ||||
|                               next_ptr->next, | ||||
|                               next_ptr->prev)); | ||||
|  | ||||
|                 /* mark the new block as a memory block and freed. */ | ||||
|                 next_ptr->magic = RT_MEMHEAP_MAGIC; | ||||
|  | ||||
|                 /* put the pool pointer into the new block. */ | ||||
|                 next_ptr->pool_ptr = heap; | ||||
|  | ||||
|                 next_ptr->prev          = header_ptr; | ||||
|                 next_ptr->next          = header_ptr->next; | ||||
|                 header_ptr->next->prev = next_ptr; | ||||
|                 header_ptr->next       = next_ptr; | ||||
|  | ||||
|                 /* insert next_ptr to free list */ | ||||
|                 next_ptr->next_free = heap->free_list->next_free; | ||||
|                 next_ptr->prev_free = heap->free_list; | ||||
|                 heap->free_list->next_free->prev_free = next_ptr; | ||||
|                 heap->free_list->next_free            = next_ptr; | ||||
|                 RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, ("new ptr: next_free 0x%08x, prev_free 0x%08x", | ||||
|                                                 next_ptr->next_free, | ||||
|                                                 next_ptr->prev_free)); | ||||
|  | ||||
|                 /* release lock */ | ||||
|                 rt_sem_release(&(heap->lock)); | ||||
|  | ||||
|                 return ptr; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* release lock */ | ||||
|         rt_sem_release(&(heap->lock)); | ||||
|  | ||||
|         /* re-allocate a memory block */ | ||||
|         new_ptr = (void *)rt_memheap_alloc(heap, newsize); | ||||
|         if (new_ptr != RT_NULL) | ||||
|         { | ||||
|             rt_memcpy(new_ptr, ptr, oldsize < newsize ? oldsize : newsize); | ||||
|             rt_memheap_free(ptr); | ||||
|         } | ||||
|  | ||||
|         return new_ptr; | ||||
|     } | ||||
|  | ||||
|     /* don't split when there is less than one node space left */ | ||||
|     if (newsize + RT_MEMHEAP_SIZE + RT_MEMHEAP_MINIALLOC >= oldsize) | ||||
|         return ptr; | ||||
|  | ||||
|     /* lock memheap */ | ||||
|     result = rt_sem_take(&(heap->lock), RT_WAITING_FOREVER); | ||||
|     if (result != RT_EOK) | ||||
|     { | ||||
|         rt_set_errno(result); | ||||
|  | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     /* split the block. */ | ||||
|     new_ptr = (struct rt_memheap_item *) | ||||
|               (((rt_uint8_t *)header_ptr) + newsize + RT_MEMHEAP_SIZE); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                  ("split: block[0x%08x] nextm[0x%08x] prevm[0x%08x] to new[0x%08x]\n", | ||||
|                   header_ptr, | ||||
|                   header_ptr->next, | ||||
|                   header_ptr->prev, | ||||
|                   new_ptr)); | ||||
|  | ||||
|     /* mark the new block as a memory block and freed. */ | ||||
|     new_ptr->magic = RT_MEMHEAP_MAGIC; | ||||
|     /* put the pool pointer into the new block. */ | ||||
|     new_ptr->pool_ptr = heap; | ||||
|  | ||||
|     /* break down the block list */ | ||||
|     new_ptr->prev          = header_ptr; | ||||
|     new_ptr->next          = header_ptr->next; | ||||
|     header_ptr->next->prev = new_ptr; | ||||
|     header_ptr->next       = new_ptr; | ||||
|  | ||||
|     /* determine if the block can be merged with the next neighbor. */ | ||||
|     if (!RT_MEMHEAP_IS_USED(new_ptr->next)) | ||||
|     { | ||||
|         struct rt_memheap_item *free_ptr; | ||||
|  | ||||
|         /* merge block with next neighbor. */ | ||||
|         free_ptr = new_ptr->next; | ||||
|         heap->available_size = heap->available_size - MEMITEM_SIZE(free_ptr); | ||||
|  | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                      ("merge: right node 0x%08x, next_free 0x%08x, prev_free 0x%08x\n", | ||||
|                       header_ptr, header_ptr->next_free, header_ptr->prev_free)); | ||||
|  | ||||
|         free_ptr->next->prev = new_ptr; | ||||
|         new_ptr->next   = free_ptr->next; | ||||
|  | ||||
|         /* remove free ptr from free list */ | ||||
|         free_ptr->next_free->prev_free = free_ptr->prev_free; | ||||
|         free_ptr->prev_free->next_free = free_ptr->next_free; | ||||
|     } | ||||
|  | ||||
|     /* insert the split block to free list */ | ||||
|     new_ptr->next_free = heap->free_list->next_free; | ||||
|     new_ptr->prev_free = heap->free_list; | ||||
|     heap->free_list->next_free->prev_free = new_ptr; | ||||
|     heap->free_list->next_free            = new_ptr; | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, ("new free ptr: next_free 0x%08x, prev_free 0x%08x\n", | ||||
|                                     new_ptr->next_free, | ||||
|                                     new_ptr->prev_free)); | ||||
|  | ||||
|     /* increment the available byte count.  */ | ||||
|     heap->available_size = heap->available_size + MEMITEM_SIZE(new_ptr); | ||||
|  | ||||
|     /* release lock */ | ||||
|     rt_sem_release(&(heap->lock)); | ||||
|  | ||||
|     /* return the old memory block */ | ||||
|     return ptr; | ||||
| } | ||||
|  | ||||
| void rt_memheap_free(void *ptr) | ||||
| { | ||||
|     rt_err_t result; | ||||
|     struct rt_memheap *heap; | ||||
|     struct rt_memheap_item *header_ptr, *new_ptr; | ||||
|     rt_uint32_t insert_header; | ||||
|  | ||||
|     /* NULL check */ | ||||
|     if (ptr == RT_NULL) return; | ||||
|  | ||||
|     /* set initial status as OK */ | ||||
|     insert_header = 1; | ||||
|     new_ptr       = RT_NULL; | ||||
|     header_ptr    = (struct rt_memheap_item *) | ||||
|                     ((rt_uint8_t *)ptr - RT_MEMHEAP_SIZE); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, ("free memory: memory[0x%08x], block[0x%08x]\n", | ||||
|                                     ptr, header_ptr)); | ||||
|  | ||||
|     /* check magic */ | ||||
|     RT_ASSERT((header_ptr->magic & RT_MEMHEAP_MASK) == RT_MEMHEAP_MAGIC); | ||||
|     RT_ASSERT(header_ptr->magic & RT_MEMHEAP_USED); | ||||
|     /* check whether this block of memory has been over-written. */ | ||||
|     RT_ASSERT((header_ptr->next->magic & RT_MEMHEAP_MASK) == RT_MEMHEAP_MAGIC); | ||||
|  | ||||
|     /* get pool ptr */ | ||||
|     heap = header_ptr->pool_ptr; | ||||
|  | ||||
|     RT_ASSERT(heap); | ||||
|     RT_ASSERT(rt_object_get_type(&heap->parent) == RT_Object_Class_MemHeap); | ||||
|  | ||||
|     /* lock memheap */ | ||||
|     result = rt_sem_take(&(heap->lock), RT_WAITING_FOREVER); | ||||
|     if (result != RT_EOK) | ||||
|     { | ||||
|         rt_set_errno(result); | ||||
|  | ||||
|         return ; | ||||
|     } | ||||
|  | ||||
|     /* Mark the memory as available. */ | ||||
|     header_ptr->magic &= ~RT_MEMHEAP_USED; | ||||
|     /* Adjust the available number of bytes. */ | ||||
|     heap->available_size = heap->available_size + MEMITEM_SIZE(header_ptr); | ||||
|  | ||||
|     /* Determine if the block can be merged with the previous neighbor. */ | ||||
|     if (!RT_MEMHEAP_IS_USED(header_ptr->prev)) | ||||
|     { | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, ("merge: left node 0x%08x\n", | ||||
|                                         header_ptr->prev)); | ||||
|  | ||||
|         /* adjust the available number of bytes. */ | ||||
|         heap->available_size = heap->available_size + RT_MEMHEAP_SIZE; | ||||
|  | ||||
|         /* yes, merge block with previous neighbor. */ | ||||
|         (header_ptr->prev)->next = header_ptr->next; | ||||
|         (header_ptr->next)->prev = header_ptr->prev; | ||||
|  | ||||
|         /* move header pointer to previous. */ | ||||
|         header_ptr = header_ptr->prev; | ||||
|         /* don't insert header to free list */ | ||||
|         insert_header = 0; | ||||
|     } | ||||
|  | ||||
|     /* determine if the block can be merged with the next neighbor. */ | ||||
|     if (!RT_MEMHEAP_IS_USED(header_ptr->next)) | ||||
|     { | ||||
|         /* adjust the available number of bytes. */ | ||||
|         heap->available_size = heap->available_size + RT_MEMHEAP_SIZE; | ||||
|  | ||||
|         /* merge block with next neighbor. */ | ||||
|         new_ptr = header_ptr->next; | ||||
|  | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                      ("merge: right node 0x%08x, next_free 0x%08x, prev_free 0x%08x\n", | ||||
|                       new_ptr, new_ptr->next_free, new_ptr->prev_free)); | ||||
|  | ||||
|         new_ptr->next->prev = header_ptr; | ||||
|         header_ptr->next    = new_ptr->next; | ||||
|  | ||||
|         /* remove new ptr from free list */ | ||||
|         new_ptr->next_free->prev_free = new_ptr->prev_free; | ||||
|         new_ptr->prev_free->next_free = new_ptr->next_free; | ||||
|     } | ||||
|  | ||||
|     if (insert_header) | ||||
|     { | ||||
|         /* no left merge, insert to free list */ | ||||
|         header_ptr->next_free = heap->free_list->next_free; | ||||
|         header_ptr->prev_free = heap->free_list; | ||||
|         heap->free_list->next_free->prev_free = header_ptr; | ||||
|         heap->free_list->next_free            = header_ptr; | ||||
|  | ||||
|         RT_DEBUG_LOG(RT_DEBUG_MEMHEAP, | ||||
|                      ("insert to free list: next_free 0x%08x, prev_free 0x%08x\n", | ||||
|                       header_ptr->next_free, header_ptr->prev_free)); | ||||
|     } | ||||
|  | ||||
|     /* release lock */ | ||||
|     rt_sem_release(&(heap->lock)); | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_MEMHEAP_AS_HEAP | ||||
| static struct rt_memheap _heap; | ||||
|  | ||||
| void rt_system_heap_init(void *begin_addr, void *end_addr) | ||||
| { | ||||
|     /* initialize a default heap in the system */ | ||||
|     rt_memheap_init(&_heap, | ||||
|                     "heap", | ||||
|                     begin_addr, | ||||
|                     (rt_uint32_t)end_addr - (rt_uint32_t)begin_addr); | ||||
| } | ||||
|  | ||||
| void *rt_malloc(rt_size_t size) | ||||
| { | ||||
|     void *ptr; | ||||
|  | ||||
|     /* try to allocate in system heap */ | ||||
|     ptr = rt_memheap_alloc(&_heap, size); | ||||
|     if (ptr == RT_NULL) | ||||
|     { | ||||
|         struct rt_object *object; | ||||
|         struct rt_list_node *node; | ||||
|         struct rt_memheap *heap; | ||||
|         struct rt_object_information *information; | ||||
|  | ||||
|         /* try to allocate on other memory heap */ | ||||
|         information = rt_object_get_information(RT_Object_Class_MemHeap); | ||||
|         RT_ASSERT(information != RT_NULL); | ||||
|         for (node  = information->object_list.next; | ||||
|              node != &(information->object_list); | ||||
|              node  = node->next) | ||||
|         { | ||||
|             object = rt_list_entry(node, struct rt_object, list); | ||||
|             heap   = (struct rt_memheap *)object; | ||||
|  | ||||
|             RT_ASSERT(heap); | ||||
|             RT_ASSERT(rt_object_get_type(&heap->parent) == RT_Object_Class_MemHeap); | ||||
|  | ||||
|             /* not allocate in the default system heap */ | ||||
|             if (heap == &_heap) | ||||
|                 continue; | ||||
|  | ||||
|             ptr = rt_memheap_alloc(heap, size); | ||||
|             if (ptr != RT_NULL) | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return ptr; | ||||
| } | ||||
|  | ||||
| void rt_free(void *rmem) | ||||
| { | ||||
|     rt_memheap_free(rmem); | ||||
| } | ||||
|  | ||||
| void *rt_realloc(void *rmem, rt_size_t newsize) | ||||
| { | ||||
|     void *new_ptr; | ||||
|     struct rt_memheap_item *header_ptr; | ||||
|  | ||||
|     if (rmem == RT_NULL) | ||||
|         return rt_malloc(newsize); | ||||
|  | ||||
|     if (newsize == 0) | ||||
|     { | ||||
|         rt_free(rmem); | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     /* get old memory item */ | ||||
|     header_ptr = (struct rt_memheap_item *) | ||||
|                  ((rt_uint8_t *)rmem - RT_MEMHEAP_SIZE); | ||||
|  | ||||
|     new_ptr = rt_memheap_realloc(header_ptr->pool_ptr, rmem, newsize); | ||||
|     if (new_ptr == RT_NULL && newsize != 0) | ||||
|     { | ||||
|         /* allocate memory block from other memheap */ | ||||
|         new_ptr = rt_malloc(newsize); | ||||
|         if (new_ptr != RT_NULL && rmem != RT_NULL) | ||||
|         { | ||||
|             rt_size_t oldsize; | ||||
|  | ||||
|             /* get the size of old memory block */ | ||||
|             oldsize = MEMITEM_SIZE(header_ptr); | ||||
|             if (newsize > oldsize) | ||||
|                 rt_memcpy(new_ptr, rmem, oldsize); | ||||
|             else | ||||
|                 rt_memcpy(new_ptr, rmem, newsize); | ||||
|  | ||||
|             rt_free(rmem); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return new_ptr; | ||||
| } | ||||
|  | ||||
| void *rt_calloc(rt_size_t count, rt_size_t size) | ||||
| { | ||||
|     void *ptr; | ||||
|     rt_size_t total_size; | ||||
|  | ||||
|     total_size = count * size; | ||||
|     ptr = rt_malloc(total_size); | ||||
|     if (ptr != RT_NULL) | ||||
|     { | ||||
|         /* clean memory */ | ||||
|         rt_memset(ptr, 0, total_size); | ||||
|     } | ||||
|  | ||||
|     return ptr; | ||||
| } | ||||
|  | ||||
| void rt_memory_info(rt_uint32_t *total, | ||||
|                     rt_uint32_t *used, | ||||
|                     rt_uint32_t *max_used) | ||||
| { | ||||
|     if (total != RT_NULL) | ||||
|         *total = _heap.pool_size; | ||||
|  | ||||
|     if (used  != RT_NULL) | ||||
|         *used = _heap.pool_size - _heap.available_size; | ||||
|  | ||||
|     if (max_used != RT_NULL) | ||||
|         *max_used = _heap.max_used_size; | ||||
| } | ||||
|  | ||||
| #endif | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										449
									
								
								source/rt_thread/src/mempool.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										449
									
								
								source/rt_thread/src/mempool.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,449 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-05-27     Bernard      implement memory pool | ||||
|  * 2006-06-03     Bernard      fix the thread timer init bug | ||||
|  * 2006-06-30     Bernard      fix the allocate/free block bug | ||||
|  * 2006-08-04     Bernard      add hook support | ||||
|  * 2006-08-10     Bernard      fix interrupt bug in rt_mp_alloc | ||||
|  * 2010-07-13     Bernard      fix RT_ALIGN issue found by kuronca | ||||
|  * 2010-10-26     yi.qiu       add module support in rt_mp_delete | ||||
|  * 2011-01-24     Bernard      add object allocation check. | ||||
|  * 2012-03-22     Bernard      fix align issue in rt_mp_init and rt_mp_create. | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #ifdef RT_USING_MEMPOOL | ||||
|  | ||||
| #ifdef RT_USING_HOOK | ||||
| static void (*rt_mp_alloc_hook)(struct rt_mempool *mp, void *block); | ||||
| static void (*rt_mp_free_hook)(struct rt_mempool *mp, void *block); | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Hook | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when a memory | ||||
|  * block is allocated from memory pool. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_mp_alloc_sethook(void (*hook)(struct rt_mempool *mp, void *block)) | ||||
| { | ||||
|     rt_mp_alloc_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when a memory | ||||
|  * block is released to memory pool. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_mp_free_sethook(void (*hook)(struct rt_mempool *mp, void *block)) | ||||
| { | ||||
|     rt_mp_free_hook = hook; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * @addtogroup MM | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will initialize a memory pool object, normally which is used | ||||
|  * for static object. | ||||
|  * | ||||
|  * @param mp the memory pool object | ||||
|  * @param name the name of memory pool | ||||
|  * @param start the star address of memory pool | ||||
|  * @param size the total size of memory pool | ||||
|  * @param block_size the size for each block | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_mp_init(struct rt_mempool *mp, | ||||
|                     const char        *name, | ||||
|                     void              *start, | ||||
|                     rt_size_t          size, | ||||
|                     rt_size_t          block_size) | ||||
| { | ||||
|     rt_uint8_t *block_ptr; | ||||
|     register rt_size_t offset; | ||||
|  | ||||
|     /* parameter check */ | ||||
|     RT_ASSERT(mp != RT_NULL); | ||||
|     RT_ASSERT(name != RT_NULL); | ||||
|     RT_ASSERT(start != RT_NULL); | ||||
|     RT_ASSERT(size > 0 && block_size > 0); | ||||
|  | ||||
|     /* initialize object */ | ||||
|     rt_object_init(&(mp->parent), RT_Object_Class_MemPool, name); | ||||
|  | ||||
|     /* initialize memory pool */ | ||||
|     mp->start_address = start; | ||||
|     mp->size = RT_ALIGN_DOWN(size, RT_ALIGN_SIZE); | ||||
|  | ||||
|     /* align the block size */ | ||||
|     block_size = RT_ALIGN(block_size, RT_ALIGN_SIZE); | ||||
|     mp->block_size = block_size; | ||||
|  | ||||
|     /* align to align size byte */ | ||||
|     mp->block_total_count = mp->size / (mp->block_size + sizeof(rt_uint8_t *)); | ||||
|     mp->block_free_count  = mp->block_total_count; | ||||
|  | ||||
|     /* initialize suspended thread list */ | ||||
|     rt_list_init(&(mp->suspend_thread)); | ||||
|  | ||||
|     /* initialize free block list */ | ||||
|     block_ptr = (rt_uint8_t *)mp->start_address; | ||||
|     for (offset = 0; offset < mp->block_total_count; offset ++) | ||||
|     { | ||||
|         *(rt_uint8_t **)(block_ptr + offset * (block_size + sizeof(rt_uint8_t *))) = | ||||
|             (rt_uint8_t *)(block_ptr + (offset + 1) * (block_size + sizeof(rt_uint8_t *))); | ||||
|     } | ||||
|  | ||||
|     *(rt_uint8_t **)(block_ptr + (offset - 1) * (block_size + sizeof(rt_uint8_t *))) = | ||||
|         RT_NULL; | ||||
|  | ||||
|     mp->block_list = block_ptr; | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will detach a memory pool from system object management. | ||||
|  * | ||||
|  * @param mp the memory pool object | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_mp_detach(struct rt_mempool *mp) | ||||
| { | ||||
|     struct rt_thread *thread; | ||||
|     register rt_ubase_t level; | ||||
|  | ||||
|     /* parameter check */ | ||||
|     RT_ASSERT(mp != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&mp->parent) == RT_Object_Class_MemPool); | ||||
|     RT_ASSERT(rt_object_is_systemobject(&mp->parent)); | ||||
|  | ||||
|     /* wake up all suspended threads */ | ||||
|     while (!rt_list_isempty(&(mp->suspend_thread))) | ||||
|     { | ||||
|         /* disable interrupt */ | ||||
|         level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|         /* get next suspend thread */ | ||||
|         thread = rt_list_entry(mp->suspend_thread.next, struct rt_thread, tlist); | ||||
|         /* set error code to RT_ERROR */ | ||||
|         thread->error = -RT_ERROR; | ||||
|  | ||||
|         /* | ||||
|          * resume thread | ||||
|          * In rt_thread_resume function, it will remove current thread from | ||||
|          * suspend list | ||||
|          */ | ||||
|         rt_thread_resume(thread); | ||||
|  | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(level); | ||||
|     } | ||||
|  | ||||
|     /* detach object */ | ||||
|     rt_object_detach(&(mp->parent)); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_HEAP | ||||
| /** | ||||
|  * This function will create a mempool object and allocate the memory pool from | ||||
|  * heap. | ||||
|  * | ||||
|  * @param name the name of memory pool | ||||
|  * @param block_count the count of blocks in memory pool | ||||
|  * @param block_size the size for each block | ||||
|  * | ||||
|  * @return the created mempool object | ||||
|  */ | ||||
| rt_mp_t rt_mp_create(const char *name, | ||||
|                      rt_size_t   block_count, | ||||
|                      rt_size_t   block_size) | ||||
| { | ||||
|     rt_uint8_t *block_ptr; | ||||
|     struct rt_mempool *mp; | ||||
|     register rt_size_t offset; | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     /* parameter check */ | ||||
|     RT_ASSERT(name != RT_NULL); | ||||
|     RT_ASSERT(block_count > 0 && block_size > 0); | ||||
|  | ||||
|     /* allocate object */ | ||||
|     mp = (struct rt_mempool *)rt_object_allocate(RT_Object_Class_MemPool, name); | ||||
|     /* allocate object failed */ | ||||
|     if (mp == RT_NULL) | ||||
|         return RT_NULL; | ||||
|  | ||||
|     /* initialize memory pool */ | ||||
|     block_size     = RT_ALIGN(block_size, RT_ALIGN_SIZE); | ||||
|     mp->block_size = block_size; | ||||
|     mp->size       = (block_size + sizeof(rt_uint8_t *)) * block_count; | ||||
|  | ||||
|     /* allocate memory */ | ||||
|     mp->start_address = rt_malloc((block_size + sizeof(rt_uint8_t *)) * | ||||
|                                   block_count); | ||||
|     if (mp->start_address == RT_NULL) | ||||
|     { | ||||
|         /* no memory, delete memory pool object */ | ||||
|         rt_object_delete(&(mp->parent)); | ||||
|  | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     mp->block_total_count = block_count; | ||||
|     mp->block_free_count  = mp->block_total_count; | ||||
|  | ||||
|     /* initialize suspended thread list */ | ||||
|     rt_list_init(&(mp->suspend_thread)); | ||||
|  | ||||
|     /* initialize free block list */ | ||||
|     block_ptr = (rt_uint8_t *)mp->start_address; | ||||
|     for (offset = 0; offset < mp->block_total_count; offset ++) | ||||
|     { | ||||
|         *(rt_uint8_t **)(block_ptr + offset * (block_size + sizeof(rt_uint8_t *))) | ||||
|             = block_ptr + (offset + 1) * (block_size + sizeof(rt_uint8_t *)); | ||||
|     } | ||||
|  | ||||
|     *(rt_uint8_t **)(block_ptr + (offset - 1) * (block_size + sizeof(rt_uint8_t *))) | ||||
|         = RT_NULL; | ||||
|  | ||||
|     mp->block_list = block_ptr; | ||||
|  | ||||
|     return mp; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will delete a memory pool and release the object memory. | ||||
|  * | ||||
|  * @param mp the memory pool object | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_mp_delete(rt_mp_t mp) | ||||
| { | ||||
|     struct rt_thread *thread; | ||||
|     register rt_ubase_t level; | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     /* parameter check */ | ||||
|     RT_ASSERT(mp != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&mp->parent) == RT_Object_Class_MemPool); | ||||
|     RT_ASSERT(rt_object_is_systemobject(&mp->parent) == RT_FALSE); | ||||
|  | ||||
|     /* wake up all suspended threads */ | ||||
|     while (!rt_list_isempty(&(mp->suspend_thread))) | ||||
|     { | ||||
|         /* disable interrupt */ | ||||
|         level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|         /* get next suspend thread */ | ||||
|         thread = rt_list_entry(mp->suspend_thread.next, struct rt_thread, tlist); | ||||
|         /* set error code to RT_ERROR */ | ||||
|         thread->error = -RT_ERROR; | ||||
|  | ||||
|         /* | ||||
|          * resume thread | ||||
|          * In rt_thread_resume function, it will remove current thread from | ||||
|          * suspend list | ||||
|          */ | ||||
|         rt_thread_resume(thread); | ||||
|  | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(level); | ||||
|     } | ||||
|  | ||||
|     /* release allocated room */ | ||||
|     rt_free(mp->start_address); | ||||
|  | ||||
|     /* detach object */ | ||||
|     rt_object_delete(&(mp->parent)); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * This function will allocate a block from memory pool | ||||
|  * | ||||
|  * @param mp the memory pool object | ||||
|  * @param time the waiting time | ||||
|  * | ||||
|  * @return the allocated memory block or RT_NULL on allocated failed | ||||
|  */ | ||||
| void *rt_mp_alloc(rt_mp_t mp, rt_int32_t time) | ||||
| { | ||||
|     rt_uint8_t *block_ptr; | ||||
|     register rt_base_t level; | ||||
|     struct rt_thread *thread; | ||||
|     rt_uint32_t before_sleep = 0; | ||||
|  | ||||
|     /* parameter check */ | ||||
|     RT_ASSERT(mp != RT_NULL); | ||||
|  | ||||
|     /* get current thread */ | ||||
|     thread = rt_thread_self(); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     while (mp->block_free_count == 0) | ||||
|     { | ||||
|         /* memory block is unavailable. */ | ||||
|         if (time == 0) | ||||
|         { | ||||
|             /* enable interrupt */ | ||||
|             rt_hw_interrupt_enable(level); | ||||
|  | ||||
|             rt_set_errno(-RT_ETIMEOUT); | ||||
|  | ||||
|             return RT_NULL; | ||||
|         } | ||||
|  | ||||
|         RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|         thread->error = RT_EOK; | ||||
|  | ||||
|         /* need suspend thread */ | ||||
|         rt_thread_suspend(thread); | ||||
|         rt_list_insert_after(&(mp->suspend_thread), &(thread->tlist)); | ||||
|  | ||||
|         if (time > 0) | ||||
|         { | ||||
|             /* get the start tick of timer */ | ||||
|             before_sleep = rt_tick_get(); | ||||
|  | ||||
|             /* init thread timer and start it */ | ||||
|             rt_timer_control(&(thread->thread_timer), | ||||
|                              RT_TIMER_CTRL_SET_TIME, | ||||
|                              &time); | ||||
|             rt_timer_start(&(thread->thread_timer)); | ||||
|         } | ||||
|  | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(level); | ||||
|  | ||||
|         /* do a schedule */ | ||||
|         rt_schedule(); | ||||
|  | ||||
|         if (thread->error != RT_EOK) | ||||
|             return RT_NULL; | ||||
|  | ||||
|         if (time > 0) | ||||
|         { | ||||
|             time -= rt_tick_get() - before_sleep; | ||||
|             if (time < 0) | ||||
|                 time = 0; | ||||
|         } | ||||
|         /* disable interrupt */ | ||||
|         level = rt_hw_interrupt_disable(); | ||||
|     } | ||||
|  | ||||
|     /* memory block is available. decrease the free block counter */ | ||||
|     mp->block_free_count--; | ||||
|  | ||||
|     /* get block from block list */ | ||||
|     block_ptr = mp->block_list; | ||||
|     RT_ASSERT(block_ptr != RT_NULL); | ||||
|  | ||||
|     /* Setup the next free node. */ | ||||
|     mp->block_list = *(rt_uint8_t **)block_ptr; | ||||
|  | ||||
|     /* point to memory pool */ | ||||
|     *(rt_uint8_t **)block_ptr = (rt_uint8_t *)mp; | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_mp_alloc_hook, | ||||
|                         (mp, (rt_uint8_t *)(block_ptr + sizeof(rt_uint8_t *)))); | ||||
|  | ||||
|     return (rt_uint8_t *)(block_ptr + sizeof(rt_uint8_t *)); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will release a memory block | ||||
|  * | ||||
|  * @param block the address of memory block to be released | ||||
|  */ | ||||
| void rt_mp_free(void *block) | ||||
| { | ||||
|     rt_uint8_t **block_ptr; | ||||
|     struct rt_mempool *mp; | ||||
|     struct rt_thread *thread; | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* parameter check */ | ||||
|     if (block == RT_NULL) return; | ||||
|  | ||||
|     /* get the control block of pool which the block belongs to */ | ||||
|     block_ptr = (rt_uint8_t **)((rt_uint8_t *)block - sizeof(rt_uint8_t *)); | ||||
|     mp        = (struct rt_mempool *)*block_ptr; | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_mp_free_hook, (mp, block)); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* increase the free block count */ | ||||
|     mp->block_free_count ++; | ||||
|  | ||||
|     /* link the block into the block list */ | ||||
|     *block_ptr = mp->block_list; | ||||
|     mp->block_list = (rt_uint8_t *)block_ptr; | ||||
|  | ||||
|     if (!rt_list_isempty(&(mp->suspend_thread))) | ||||
|     { | ||||
|         /* get the suspended thread */ | ||||
|         thread = rt_list_entry(mp->suspend_thread.next, | ||||
|                                struct rt_thread, | ||||
|                                tlist); | ||||
|  | ||||
|         /* set error */ | ||||
|         thread->error = RT_EOK; | ||||
|  | ||||
|         /* resume thread */ | ||||
|         rt_thread_resume(thread); | ||||
|  | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(level); | ||||
|  | ||||
|         /* do a schedule */ | ||||
|         rt_schedule(); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
|  | ||||
| #endif | ||||
|  | ||||
							
								
								
									
										548
									
								
								source/rt_thread/src/object.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										548
									
								
								source/rt_thread/src/object.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,548 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-03-14     Bernard      the first version | ||||
|  * 2006-04-21     Bernard      change the scheduler lock to interrupt lock | ||||
|  * 2006-05-18     Bernard      fix the object init bug | ||||
|  * 2006-08-03     Bernard      add hook support | ||||
|  * 2007-01-28     Bernard      rename RT_OBJECT_Class_Static to RT_Object_Class_Static | ||||
|  * 2010-10-26     yi.qiu       add module support in rt_object_allocate and rt_object_free | ||||
|  * 2017-12-10     Bernard      Add object_info enum. | ||||
|  * 2018-01-25     Bernard      Fix the object find issue when enable MODULE. | ||||
|  */ | ||||
|  | ||||
| #include <rtthread.h> | ||||
| #include <rthw.h> | ||||
|  | ||||
| /* | ||||
|  * define object_info for the number of rt_object_container items. | ||||
|  */ | ||||
| enum rt_object_info_type | ||||
| { | ||||
|     RT_Object_Info_Thread = 0,                         /**< The object is a thread. */ | ||||
| #ifdef RT_USING_SEMAPHORE | ||||
|     RT_Object_Info_Semaphore,                          /**< The object is a semaphore. */ | ||||
| #endif | ||||
| #ifdef RT_USING_MUTEX | ||||
|     RT_Object_Info_Mutex,                              /**< The object is a mutex. */ | ||||
| #endif | ||||
| #ifdef RT_USING_EVENT | ||||
|     RT_Object_Info_Event,                              /**< The object is a event. */ | ||||
| #endif | ||||
| #ifdef RT_USING_MAILBOX | ||||
|     RT_Object_Info_MailBox,                            /**< The object is a mail box. */ | ||||
| #endif | ||||
| #ifdef RT_USING_MESSAGEQUEUE | ||||
|     RT_Object_Info_MessageQueue,                       /**< The object is a message queue. */ | ||||
| #endif | ||||
| #ifdef RT_USING_MEMHEAP | ||||
|     RT_Object_Info_MemHeap,                            /**< The object is a memory heap */ | ||||
| #endif | ||||
| #ifdef RT_USING_MEMPOOL | ||||
|     RT_Object_Info_MemPool,                            /**< The object is a memory pool. */ | ||||
| #endif | ||||
| #ifdef RT_USING_DEVICE | ||||
|     RT_Object_Info_Device,                             /**< The object is a device */ | ||||
| #endif | ||||
|     RT_Object_Info_Timer,                              /**< The object is a timer. */ | ||||
|     RT_Object_Info_Unknown,                            /**< The object is unknown. */ | ||||
| }; | ||||
|  | ||||
| #define _OBJ_CONTAINER_LIST_INIT(c)     \ | ||||
|     {&(rt_object_container[c].object_list), &(rt_object_container[c].object_list)} | ||||
| static struct rt_object_information rt_object_container[RT_Object_Info_Unknown] = | ||||
| { | ||||
|     /* initialize object container - thread */ | ||||
|     {RT_Object_Class_Thread, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_Thread), sizeof(struct rt_thread)}, | ||||
| #ifdef RT_USING_SEMAPHORE | ||||
|     /* initialize object container - semaphore */ | ||||
|     {RT_Object_Class_Semaphore, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_Semaphore), sizeof(struct rt_semaphore)}, | ||||
| #endif | ||||
| #ifdef RT_USING_MUTEX | ||||
|     /* initialize object container - mutex */ | ||||
|     {RT_Object_Class_Mutex, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_Mutex), sizeof(struct rt_mutex)}, | ||||
| #endif | ||||
| #ifdef RT_USING_EVENT | ||||
|     /* initialize object container - event */ | ||||
|     {RT_Object_Class_Event, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_Event), sizeof(struct rt_event)}, | ||||
| #endif | ||||
| #ifdef RT_USING_MAILBOX | ||||
|     /* initialize object container - mailbox */ | ||||
|     {RT_Object_Class_MailBox, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_MailBox), sizeof(struct rt_mailbox)}, | ||||
| #endif | ||||
| #ifdef RT_USING_MESSAGEQUEUE | ||||
|     /* initialize object container - message queue */ | ||||
|     {RT_Object_Class_MessageQueue, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_MessageQueue), sizeof(struct rt_messagequeue)}, | ||||
| #endif | ||||
| #ifdef RT_USING_MEMHEAP | ||||
|     /* initialize object container - memory heap */ | ||||
|     {RT_Object_Class_MemHeap, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_MemHeap), sizeof(struct rt_memheap)}, | ||||
| #endif | ||||
| #ifdef RT_USING_MEMPOOL | ||||
|     /* initialize object container - memory pool */ | ||||
|     {RT_Object_Class_MemPool, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_MemPool), sizeof(struct rt_mempool)}, | ||||
| #endif | ||||
| #ifdef RT_USING_DEVICE | ||||
|     /* initialize object container - device */ | ||||
|     {RT_Object_Class_Device, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_Device), sizeof(struct rt_device)}, | ||||
| #endif | ||||
|     /* initialize object container - timer */ | ||||
|     {RT_Object_Class_Timer, _OBJ_CONTAINER_LIST_INIT(RT_Object_Info_Timer), sizeof(struct rt_timer)}, | ||||
| }; | ||||
|  | ||||
| #ifdef RT_USING_HOOK | ||||
| static void (*rt_object_attach_hook)(struct rt_object *object); | ||||
| static void (*rt_object_detach_hook)(struct rt_object *object); | ||||
| void (*rt_object_trytake_hook)(struct rt_object *object); | ||||
| void (*rt_object_take_hook)(struct rt_object *object); | ||||
| void (*rt_object_put_hook)(struct rt_object *object); | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Hook | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when object | ||||
|  * attaches to kernel object system. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_object_attach_sethook(void (*hook)(struct rt_object *object)) | ||||
| { | ||||
|     rt_object_attach_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when object | ||||
|  * detaches from kernel object system. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_object_detach_sethook(void (*hook)(struct rt_object *object)) | ||||
| { | ||||
|     rt_object_detach_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when object | ||||
|  * is taken from kernel object system. | ||||
|  * | ||||
|  * The object is taken means: | ||||
|  * semaphore - semaphore is taken by thread | ||||
|  * mutex - mutex is taken by thread | ||||
|  * event - event is received by thread | ||||
|  * mailbox - mail is received by thread | ||||
|  * message queue - message is received by thread | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_object_trytake_sethook(void (*hook)(struct rt_object *object)) | ||||
| { | ||||
|     rt_object_trytake_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when object | ||||
|  * have been taken from kernel object system. | ||||
|  * | ||||
|  * The object have been taken means: | ||||
|  * semaphore - semaphore have been taken by thread | ||||
|  * mutex - mutex have been taken by thread | ||||
|  * event - event have been received by thread | ||||
|  * mailbox - mail have been received by thread | ||||
|  * message queue - message have been received by thread | ||||
|  * timer - timer is started | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_object_take_sethook(void (*hook)(struct rt_object *object)) | ||||
| { | ||||
|     rt_object_take_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when object | ||||
|  * is put to kernel object system. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_object_put_sethook(void (*hook)(struct rt_object *object)) | ||||
| { | ||||
|     rt_object_put_hook = hook; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * | ||||
|  * This function will initialize system object management. | ||||
|  * | ||||
|  * @deprecated since 0.3.0, this function does not need to be invoked | ||||
|  * in the system initialization. | ||||
|  */ | ||||
| void rt_system_object_init(void) | ||||
| { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @addtogroup KernelObject | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will return the specified type of object information. | ||||
|  * | ||||
|  * @param type the type of object, which can be | ||||
|  *             RT_Object_Class_Thread/Semaphore/Mutex... etc | ||||
|  * | ||||
|  * @return the object type information or RT_NULL | ||||
|  */ | ||||
| struct rt_object_information * | ||||
| rt_object_get_information(enum rt_object_class_type type) | ||||
| { | ||||
|     int index; | ||||
|  | ||||
|     for (index = 0; index < RT_Object_Info_Unknown; index ++) | ||||
|         if (rt_object_container[index].type == type) return &rt_object_container[index]; | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will return the length of object list in object container. | ||||
|  * | ||||
|  * @param type the type of object, which can be | ||||
|  *             RT_Object_Class_Thread/Semaphore/Mutex... etc | ||||
|  * @return the length of object list | ||||
|  */ | ||||
| int rt_object_get_length(enum rt_object_class_type type) | ||||
| { | ||||
|     int count = 0; | ||||
|     rt_ubase_t level; | ||||
|     struct rt_list_node *node = RT_NULL; | ||||
|     struct rt_object_information *information = RT_NULL; | ||||
|  | ||||
|     information = rt_object_get_information((enum rt_object_class_type)type); | ||||
|     if (information == RT_NULL) return 0; | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     /* get the count of objects */ | ||||
|     rt_list_for_each(node, &(information->object_list)) | ||||
|     { | ||||
|         count ++; | ||||
|     } | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return count; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will copy the object pointer of the specified type, | ||||
|  * with the maximum size specified by maxlen. | ||||
|  * | ||||
|  * @param type the type of object, which can be | ||||
|  *             RT_Object_Class_Thread/Semaphore/Mutex... etc | ||||
|  * @param pointers the pointers will be saved to | ||||
|  * @param maxlen the maximum number of pointers can be saved | ||||
|  * | ||||
|  * @return the copied number of object pointers | ||||
|  */ | ||||
| int rt_object_get_pointers(enum rt_object_class_type type, rt_object_t *pointers, int maxlen) | ||||
| { | ||||
|     int index = 0; | ||||
|     rt_ubase_t level; | ||||
|  | ||||
|     struct rt_object *object; | ||||
|     struct rt_list_node *node = RT_NULL; | ||||
|     struct rt_object_information *information = RT_NULL; | ||||
|  | ||||
|     if (maxlen <= 0) return 0; | ||||
|  | ||||
|     information = rt_object_get_information((enum rt_object_class_type)type); | ||||
|     if (information == RT_NULL) return 0; | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     /* retrieve pointer of object */ | ||||
|     rt_list_for_each(node, &(information->object_list)) | ||||
|     { | ||||
|         object = rt_list_entry(node, struct rt_object, list); | ||||
|  | ||||
|         pointers[index] = object; | ||||
|         index ++; | ||||
|  | ||||
|         if (index >= maxlen) break; | ||||
|     } | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return index; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will initialize an object and add it to object system | ||||
|  * management. | ||||
|  * | ||||
|  * @param object the specified object to be initialized. | ||||
|  * @param type the object type. | ||||
|  * @param name the object name. In system, the object's name must be unique. | ||||
|  */ | ||||
| void rt_object_init(struct rt_object         *object, | ||||
|                     enum rt_object_class_type type, | ||||
|                     const char               *name) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|     struct rt_list_node *node = RT_NULL; | ||||
|     struct rt_object_information *information; | ||||
|  | ||||
|     /* get object information */ | ||||
|     information = rt_object_get_information(type); | ||||
|     RT_ASSERT(information != RT_NULL); | ||||
|  | ||||
|     /* check object type to avoid re-initialization */ | ||||
|  | ||||
|     /* enter critical */ | ||||
|     rt_enter_critical(); | ||||
|     /* try to find object */ | ||||
|     for (node  = information->object_list.next; | ||||
|             node != &(information->object_list); | ||||
|             node  = node->next) | ||||
|     { | ||||
|         struct rt_object *obj; | ||||
|  | ||||
|         obj = rt_list_entry(node, struct rt_object, list); | ||||
|         if (obj) /* skip warning when disable debug */ | ||||
|         { | ||||
|             RT_ASSERT(obj != object); | ||||
|         } | ||||
|     } | ||||
|     /* leave critical */ | ||||
|     rt_exit_critical(); | ||||
|  | ||||
|     /* initialize object's parameters */ | ||||
|     /* set object type to static */ | ||||
|     object->type = type | RT_Object_Class_Static; | ||||
|     /* copy name */ | ||||
|     rt_strncpy(object->name, name, RT_NAME_MAX); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_object_attach_hook, (object)); | ||||
|  | ||||
|     /* lock interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* insert object into information object list */ | ||||
|     rt_list_insert_after(&(information->object_list), &(object->list)); | ||||
|  | ||||
|     /* unlock interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will detach a static object from object system, | ||||
|  * and the memory of static object is not freed. | ||||
|  * | ||||
|  * @param object the specified object to be detached. | ||||
|  */ | ||||
| void rt_object_detach(rt_object_t object) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|  | ||||
|     /* object check */ | ||||
|     RT_ASSERT(object != RT_NULL); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_object_detach_hook, (object)); | ||||
|  | ||||
|     /* reset object type */ | ||||
|     object->type = 0; | ||||
|  | ||||
|     /* lock interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* remove from old list */ | ||||
|     rt_list_remove(&(object->list)); | ||||
|  | ||||
|     /* unlock interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_HEAP | ||||
| /** | ||||
|  * This function will allocate an object from object system | ||||
|  * | ||||
|  * @param type the type of object | ||||
|  * @param name the object name. In system, the object's name must be unique. | ||||
|  * | ||||
|  * @return object | ||||
|  */ | ||||
| rt_object_t rt_object_allocate(enum rt_object_class_type type, const char *name) | ||||
| { | ||||
|     struct rt_object *object; | ||||
|     register rt_base_t temp; | ||||
|     struct rt_object_information *information; | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     /* get object information */ | ||||
|     information = rt_object_get_information(type); | ||||
|     RT_ASSERT(information != RT_NULL); | ||||
|  | ||||
|     object = (struct rt_object *)RT_KERNEL_MALLOC(information->object_size); | ||||
|     if (object == RT_NULL) | ||||
|     { | ||||
|         /* no memory can be allocated */ | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     /* clean memory data of object */ | ||||
|     rt_memset(object, 0x0, information->object_size); | ||||
|  | ||||
|     /* initialize object's parameters */ | ||||
|  | ||||
|     /* set object type */ | ||||
|     object->type = type; | ||||
|  | ||||
|     /* set object flag */ | ||||
|     object->flag = 0; | ||||
|  | ||||
|     /* copy name */ | ||||
|     rt_strncpy(object->name, name, RT_NAME_MAX); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_object_attach_hook, (object)); | ||||
|  | ||||
|     /* lock interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* insert object into information object list */ | ||||
|     rt_list_insert_after(&(information->object_list), &(object->list)); | ||||
|  | ||||
|     /* unlock interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
|  | ||||
|     /* return object */ | ||||
|     return object; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will delete an object and release object memory. | ||||
|  * | ||||
|  * @param object the specified object to be deleted. | ||||
|  */ | ||||
| void rt_object_delete(rt_object_t object) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|  | ||||
|     /* object check */ | ||||
|     RT_ASSERT(object != RT_NULL); | ||||
|     RT_ASSERT(!(object->type & RT_Object_Class_Static)); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_object_detach_hook, (object)); | ||||
|  | ||||
|     /* reset object type */ | ||||
|     object->type = RT_Object_Class_Null; | ||||
|  | ||||
|     /* lock interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* remove from old list */ | ||||
|     rt_list_remove(&(object->list)); | ||||
|  | ||||
|     /* unlock interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
|  | ||||
|     /* free the memory of object */ | ||||
|     RT_KERNEL_FREE(object); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * This function will judge the object is system object or not. | ||||
|  * Normally, the system object is a static object and the type | ||||
|  * of object set to RT_Object_Class_Static. | ||||
|  * | ||||
|  * @param object the specified object to be judged. | ||||
|  * | ||||
|  * @return RT_TRUE if a system object, RT_FALSE for others. | ||||
|  */ | ||||
| rt_bool_t rt_object_is_systemobject(rt_object_t object) | ||||
| { | ||||
|     /* object check */ | ||||
|     RT_ASSERT(object != RT_NULL); | ||||
|  | ||||
|     if (object->type & RT_Object_Class_Static) | ||||
|         return RT_TRUE; | ||||
|  | ||||
|     return RT_FALSE; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will return the type of object without | ||||
|  * RT_Object_Class_Static flag. | ||||
|  * | ||||
|  * @param object the specified object to be get type. | ||||
|  * | ||||
|  * @return the type of object. | ||||
|  */ | ||||
| rt_uint8_t rt_object_get_type(rt_object_t object) | ||||
| { | ||||
|     /* object check */ | ||||
|     RT_ASSERT(object != RT_NULL); | ||||
|  | ||||
|     return object->type & ~RT_Object_Class_Static; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will find specified name object from object | ||||
|  * container. | ||||
|  * | ||||
|  * @param name the specified name of object. | ||||
|  * @param type the type of object | ||||
|  * | ||||
|  * @return the found object or RT_NULL if there is no this object | ||||
|  * in object container. | ||||
|  * | ||||
|  * @note this function shall not be invoked in interrupt status. | ||||
|  */ | ||||
| rt_object_t rt_object_find(const char *name, rt_uint8_t type) | ||||
| { | ||||
|     struct rt_object *object = RT_NULL; | ||||
|     struct rt_list_node *node = RT_NULL; | ||||
|     struct rt_object_information *information = RT_NULL; | ||||
|  | ||||
|     information = rt_object_get_information((enum rt_object_class_type)type); | ||||
|  | ||||
|     /* parameter check */ | ||||
|     if ((name == RT_NULL) || (information == RT_NULL)) return RT_NULL; | ||||
|  | ||||
|     /* which is invoke in interrupt status */ | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     /* enter critical */ | ||||
|     rt_enter_critical(); | ||||
|  | ||||
|     /* try to find object */ | ||||
|     rt_list_for_each(node, &(information->object_list)) | ||||
|     { | ||||
|         object = rt_list_entry(node, struct rt_object, list); | ||||
|         if (rt_strncmp(object->name, name, RT_NAME_MAX) == 0) | ||||
|         { | ||||
|             /* leave critical */ | ||||
|             rt_exit_critical(); | ||||
|  | ||||
|             return object; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /* leave critical */ | ||||
|     rt_exit_critical(); | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
							
								
								
									
										423
									
								
								source/rt_thread/src/scheduler.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										423
									
								
								source/rt_thread/src/scheduler.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,423 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-03-17     Bernard      the first version | ||||
|  * 2006-04-28     Bernard      fix the scheduler algorthm | ||||
|  * 2006-04-30     Bernard      add SCHEDULER_DEBUG | ||||
|  * 2006-05-27     Bernard      fix the scheduler algorthm for same priority | ||||
|  *                             thread schedule | ||||
|  * 2006-06-04     Bernard      rewrite the scheduler algorithm | ||||
|  * 2006-08-03     Bernard      add hook support | ||||
|  * 2006-09-05     Bernard      add 32 priority level support | ||||
|  * 2006-09-24     Bernard      add rt_system_scheduler_start function | ||||
|  * 2009-09-16     Bernard      fix _rt_scheduler_stack_check | ||||
|  * 2010-04-11     yi.qiu       add module feature | ||||
|  * 2010-07-13     Bernard      fix the maximal number of rt_scheduler_lock_nest | ||||
|  *                             issue found by kuronca | ||||
|  * 2010-12-13     Bernard      add defunct list initialization even if not use heap. | ||||
|  * 2011-05-10     Bernard      clean scheduler debug log. | ||||
|  * 2013-12-21     Grissiom     add rt_critical_level | ||||
|  * 2018-11-22     Jesven       remove the current task from ready queue | ||||
|  *                             add per cpu ready queue | ||||
|  *                             add _get_highest_priority_thread to find highest priority task | ||||
|  *                             rt_schedule_insert_thread won't insert current task to ready queue | ||||
|  *                             in smp version, rt_hw_context_switch_interrupt maybe switch to | ||||
|  *                               new task directly | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <rtthread.h> | ||||
| #include <rthw.h> | ||||
|  | ||||
| rt_list_t rt_thread_priority_table[RT_THREAD_PRIORITY_MAX]; | ||||
| rt_uint32_t rt_thread_ready_priority_group; | ||||
|  | ||||
| #if RT_THREAD_PRIORITY_MAX > 32 | ||||
| /* Maximum priority level, 256 */ | ||||
| rt_uint8_t rt_thread_ready_table[32]; | ||||
| #endif | ||||
|  | ||||
|  | ||||
| extern volatile rt_uint8_t rt_interrupt_nest; | ||||
| static rt_int16_t rt_scheduler_lock_nest; | ||||
| struct rt_thread *rt_current_thread = RT_NULL; | ||||
| rt_uint8_t rt_current_priority; | ||||
|  | ||||
|  | ||||
| rt_list_t rt_thread_defunct; | ||||
|  | ||||
| #ifdef RT_USING_HOOK | ||||
| static void (*rt_scheduler_hook)(struct rt_thread *from, struct rt_thread *to); | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Hook | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when thread | ||||
|  * switch happens. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void | ||||
| rt_scheduler_sethook(void (*hook)(struct rt_thread *from, struct rt_thread *to)) | ||||
| { | ||||
|     rt_scheduler_hook = hook; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
| #endif | ||||
|  | ||||
| #ifdef RT_USING_OVERFLOW_CHECK | ||||
| static void _rt_scheduler_stack_check(struct rt_thread *thread) | ||||
| { | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|  | ||||
| #if defined(ARCH_CPU_STACK_GROWS_UPWARD) | ||||
|     if (*((rt_uint8_t *)((rt_ubase_t)thread->stack_addr + thread->stack_size - 1)) != '#' || | ||||
| #else | ||||
|     if (*((rt_uint8_t *)thread->stack_addr) != '#' || | ||||
| #endif | ||||
|         (rt_ubase_t)thread->sp <= (rt_ubase_t)thread->stack_addr || | ||||
|         (rt_ubase_t)thread->sp > | ||||
|         (rt_ubase_t)thread->stack_addr + (rt_ubase_t)thread->stack_size) | ||||
|     { | ||||
|         rt_ubase_t level; | ||||
|  | ||||
|         rt_kprintf("thread:%s stack overflow\n", thread->name); | ||||
| #ifdef RT_USING_FINSH | ||||
|         { | ||||
|             extern long list_thread(void); | ||||
|             list_thread(); | ||||
|         } | ||||
| #endif | ||||
|         level = rt_hw_interrupt_disable(); | ||||
|         while (level); | ||||
|     } | ||||
| #if defined(ARCH_CPU_STACK_GROWS_UPWARD) | ||||
|     else if ((rt_ubase_t)thread->sp > ((rt_ubase_t)thread->stack_addr + thread->stack_size)) | ||||
|     { | ||||
|         rt_kprintf("warning: %s stack is close to the top of stack address.\n", | ||||
|                    thread->name); | ||||
|     } | ||||
| #else | ||||
|     else if ((rt_ubase_t)thread->sp <= ((rt_ubase_t)thread->stack_addr + 32)) | ||||
|     { | ||||
|         rt_kprintf("warning: %s stack is close to end of stack address.\n", | ||||
|                    thread->name); | ||||
|     } | ||||
| #endif | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * This function will initialize the system scheduler | ||||
|  */ | ||||
| void rt_system_scheduler_init(void) | ||||
| { | ||||
|     register rt_base_t offset; | ||||
|  | ||||
|     rt_scheduler_lock_nest = 0; | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, ("start scheduler: max priority 0x%02x\n", | ||||
|                                       RT_THREAD_PRIORITY_MAX)); | ||||
|  | ||||
|     for (offset = 0; offset < RT_THREAD_PRIORITY_MAX; offset ++) | ||||
|     { | ||||
|         rt_list_init(&rt_thread_priority_table[offset]); | ||||
|     } | ||||
|  | ||||
|     rt_current_priority = RT_THREAD_PRIORITY_MAX - 1; | ||||
|     rt_current_thread = RT_NULL; | ||||
|  | ||||
|     /* initialize ready priority group */ | ||||
|     rt_thread_ready_priority_group = 0; | ||||
|  | ||||
| #if RT_THREAD_PRIORITY_MAX > 32 | ||||
|     /* initialize ready table */ | ||||
|     rt_memset(rt_thread_ready_table, 0, sizeof(rt_thread_ready_table)); | ||||
| #endif | ||||
|  | ||||
|     /* initialize thread defunct */ | ||||
|     rt_list_init(&rt_thread_defunct); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * This function will startup scheduler. It will select one thread | ||||
|  * with the highest priority level, then switch to it. | ||||
|  */ | ||||
| void rt_system_scheduler_start(void) | ||||
| { | ||||
|     register struct rt_thread *to_thread; | ||||
|     register rt_ubase_t highest_ready_priority; | ||||
|  | ||||
| #if RT_THREAD_PRIORITY_MAX > 32 | ||||
|     register rt_ubase_t number; | ||||
|  | ||||
|     number = __rt_ffs(rt_thread_ready_priority_group) - 1; | ||||
|     highest_ready_priority = (number << 3) + __rt_ffs(rt_thread_ready_table[number]) - 1; | ||||
| #else | ||||
|     highest_ready_priority = __rt_ffs(rt_thread_ready_priority_group) - 1; | ||||
| #endif | ||||
|  | ||||
|     /* get switch to thread */ | ||||
|     to_thread = rt_list_entry(rt_thread_priority_table[highest_ready_priority].next, | ||||
|                               struct rt_thread, | ||||
|                               tlist); | ||||
|  | ||||
|     rt_current_thread = to_thread; | ||||
|  | ||||
|     /* switch to new thread */ | ||||
|     rt_hw_context_switch_to((rt_uint32_t)&to_thread->sp); | ||||
|  | ||||
|     /* never come back */ | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Thread | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will perform one schedule. It will select one thread | ||||
|  * with the highest priority level, then switch to it. | ||||
|  */ | ||||
| void rt_schedule(void) | ||||
| { | ||||
|     rt_base_t level; | ||||
|     struct rt_thread *to_thread; | ||||
|     struct rt_thread *from_thread; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* check the scheduler is enabled or not */ | ||||
|     if (rt_scheduler_lock_nest == 0) | ||||
|     { | ||||
|         register rt_ubase_t highest_ready_priority; | ||||
|  | ||||
| #if RT_THREAD_PRIORITY_MAX <= 32 | ||||
|         highest_ready_priority = __rt_ffs(rt_thread_ready_priority_group) - 1; | ||||
| #else | ||||
|         register rt_ubase_t number; | ||||
|  | ||||
|         number = __rt_ffs(rt_thread_ready_priority_group) - 1; | ||||
|         highest_ready_priority = (number << 3) + __rt_ffs(rt_thread_ready_table[number]) - 1; | ||||
| #endif | ||||
|  | ||||
|         /* get switch to thread */ | ||||
|         to_thread = rt_list_entry(rt_thread_priority_table[highest_ready_priority].next, | ||||
|                                   struct rt_thread, | ||||
|                                   tlist); | ||||
|  | ||||
|         /* if the destination thread is not the same as current thread */ | ||||
|         if (to_thread != rt_current_thread) | ||||
|         { | ||||
|             rt_current_priority = (rt_uint8_t)highest_ready_priority; | ||||
|             from_thread         = rt_current_thread; | ||||
|             rt_current_thread   = to_thread; | ||||
|  | ||||
|             RT_OBJECT_HOOK_CALL(rt_scheduler_hook, (from_thread, to_thread)); | ||||
|  | ||||
|             /* switch to new thread */ | ||||
|             RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, | ||||
|                          ("[%d]switch to priority#%d " | ||||
|                           "thread:%.*s(sp:0x%p), " | ||||
|                           "from thread:%.*s(sp: 0x%p)\n", | ||||
|                           rt_interrupt_nest, highest_ready_priority, | ||||
|                           RT_NAME_MAX, to_thread->name, to_thread->sp, | ||||
|                           RT_NAME_MAX, from_thread->name, from_thread->sp)); | ||||
|  | ||||
| #ifdef RT_USING_OVERFLOW_CHECK | ||||
|             _rt_scheduler_stack_check(to_thread); | ||||
| #endif | ||||
|  | ||||
|             if (rt_interrupt_nest == 0) | ||||
|             { | ||||
|                 rt_hw_context_switch((rt_ubase_t)&from_thread->sp, | ||||
|                                      (rt_ubase_t)&to_thread->sp); | ||||
|  | ||||
|                 /* enable interrupt */ | ||||
|                 rt_hw_interrupt_enable(level); | ||||
|  | ||||
|                 return ; | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, ("switch in interrupt\n")); | ||||
|  | ||||
|                 rt_hw_context_switch_interrupt((rt_ubase_t)&from_thread->sp, | ||||
|                                                (rt_ubase_t)&to_thread->sp); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * This function will insert a thread to system ready queue. The state of | ||||
|  * thread will be set as READY and remove from suspend queue. | ||||
|  * | ||||
|  * @param thread the thread to be inserted | ||||
|  * @note Please do not invoke this function in user application. | ||||
|  */ | ||||
| void rt_schedule_insert_thread(struct rt_thread *thread) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|  | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* change stat */ | ||||
|     thread->stat = RT_THREAD_READY | (thread->stat & ~RT_THREAD_STAT_MASK); | ||||
|  | ||||
|     /* insert thread to ready list */ | ||||
|     rt_list_insert_before(&(rt_thread_priority_table[thread->current_priority]), | ||||
|                           &(thread->tlist)); | ||||
|  | ||||
|     /* set priority mask */ | ||||
| #if RT_THREAD_PRIORITY_MAX <= 32 | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, ("insert thread[%.*s], the priority: %d\n", | ||||
|                                       RT_NAME_MAX, thread->name, thread->current_priority)); | ||||
| #else | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, | ||||
|                  ("insert thread[%.*s], the priority: %d 0x%x %d\n", | ||||
|                   RT_NAME_MAX, | ||||
|                   thread->name, | ||||
|                   thread->number, | ||||
|                   thread->number_mask, | ||||
|                   thread->high_mask)); | ||||
| #endif | ||||
|  | ||||
| #if RT_THREAD_PRIORITY_MAX > 32 | ||||
|     rt_thread_ready_table[thread->number] |= thread->high_mask; | ||||
| #endif | ||||
|     rt_thread_ready_priority_group |= thread->number_mask; | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * This function will remove a thread from system ready queue. | ||||
|  * | ||||
|  * @param thread the thread to be removed | ||||
|  * | ||||
|  * @note Please do not invoke this function in user application. | ||||
|  */ | ||||
| void rt_schedule_remove_thread(struct rt_thread *thread) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|  | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
| #if RT_THREAD_PRIORITY_MAX <= 32 | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, ("remove thread[%.*s], the priority: %d\n", | ||||
|                                       RT_NAME_MAX, thread->name, | ||||
|                                       thread->current_priority)); | ||||
| #else | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, | ||||
|                  ("remove thread[%.*s], the priority: %d 0x%x %d\n", | ||||
|                   RT_NAME_MAX, | ||||
|                   thread->name, | ||||
|                   thread->number, | ||||
|                   thread->number_mask, | ||||
|                   thread->high_mask)); | ||||
| #endif | ||||
|  | ||||
|     /* remove thread from ready list */ | ||||
|     rt_list_remove(&(thread->tlist)); | ||||
|     if (rt_list_isempty(&(rt_thread_priority_table[thread->current_priority]))) | ||||
|     { | ||||
| #if RT_THREAD_PRIORITY_MAX > 32 | ||||
|         rt_thread_ready_table[thread->number] &= ~thread->high_mask; | ||||
|         if (rt_thread_ready_table[thread->number] == 0) | ||||
|         { | ||||
|             rt_thread_ready_priority_group &= ~thread->number_mask; | ||||
|         } | ||||
| #else | ||||
|         rt_thread_ready_priority_group &= ~thread->number_mask; | ||||
| #endif | ||||
|     } | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will lock the thread scheduler. | ||||
|  */ | ||||
| void rt_enter_critical(void) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* | ||||
|      * the maximal number of nest is RT_UINT16_MAX, which is big | ||||
|      * enough and does not check here | ||||
|      */ | ||||
|     rt_scheduler_lock_nest ++; | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will unlock the thread scheduler. | ||||
|  */ | ||||
| void rt_exit_critical(void) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     rt_scheduler_lock_nest --; | ||||
|     if (rt_scheduler_lock_nest <= 0) | ||||
|     { | ||||
|         rt_scheduler_lock_nest = 0; | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(level); | ||||
|  | ||||
|         if (rt_current_thread) | ||||
|         { | ||||
|             /* if scheduler is started, do a schedule */ | ||||
|             rt_schedule(); | ||||
|         } | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(level); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get the scheduler lock level | ||||
|  * | ||||
|  * @return the level of the scheduler lock. 0 means unlocked. | ||||
|  */ | ||||
| rt_uint16_t rt_critical_level(void) | ||||
| { | ||||
|     return rt_scheduler_lock_nest; | ||||
| } | ||||
| /**@}*/ | ||||
|  | ||||
							
								
								
									
										931
									
								
								source/rt_thread/src/slab.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										931
									
								
								source/rt_thread/src/slab.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,931 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * File      : slab.c | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2008-07-12     Bernard      the first version | ||||
|  * 2010-07-13     Bernard      fix RT_ALIGN issue found by kuronca | ||||
|  * 2010-10-23     yi.qiu       add module memory allocator | ||||
|  * 2010-12-18     yi.qiu       fix zone release bug | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * KERN_SLABALLOC.C - Kernel SLAB memory allocator | ||||
|  * | ||||
|  * Copyright (c) 2003,2004 The DragonFly Project.  All rights reserved. | ||||
|  * | ||||
|  * This code is derived from software contributed to The DragonFly Project | ||||
|  * by Matthew Dillon <dillon@backplane.com> | ||||
|  * | ||||
|  * Redistribution and use in source and binary forms, with or without | ||||
|  * modification, are permitted provided that the following conditions | ||||
|  * are met: | ||||
|  * | ||||
|  * 1. Redistributions of source code must retain the above copyright | ||||
|  *    notice, this list of conditions and the following disclaimer. | ||||
|  * 2. Redistributions in binary form must reproduce the above copyright | ||||
|  *    notice, this list of conditions and the following disclaimer in | ||||
|  *    the documentation and/or other materials provided with the | ||||
|  *    distribution. | ||||
|  * 3. Neither the name of The DragonFly Project nor the names of its | ||||
|  *    contributors may be used to endorse or promote products derived | ||||
|  *    from this software without specific, prior written permission. | ||||
|  * | ||||
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | ||||
|  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | ||||
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | ||||
|  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE | ||||
|  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | ||||
|  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING, | ||||
|  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||||
|  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED | ||||
|  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
|  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | ||||
|  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | ||||
|  * SUCH DAMAGE. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #define RT_MEM_STATS | ||||
|  | ||||
| #if defined (RT_USING_HEAP) && defined (RT_USING_SLAB) | ||||
| /* some statistical variable */ | ||||
| #ifdef RT_MEM_STATS | ||||
| static rt_size_t used_mem, max_mem; | ||||
| #endif | ||||
|  | ||||
| #ifdef RT_USING_HOOK | ||||
| static void (*rt_malloc_hook)(void *ptr, rt_size_t size); | ||||
| static void (*rt_free_hook)(void *ptr); | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Hook | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when a memory | ||||
|  * block is allocated from heap memory. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_malloc_sethook(void (*hook)(void *ptr, rt_size_t size)) | ||||
| { | ||||
|     rt_malloc_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when a memory | ||||
|  * block is released to heap memory. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_free_sethook(void (*hook)(void *ptr)) | ||||
| { | ||||
|     rt_free_hook = hook; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
|  | ||||
| #endif | ||||
|  | ||||
| /* | ||||
|  * slab allocator implementation | ||||
|  * | ||||
|  * A slab allocator reserves a ZONE for each chunk size, then lays the | ||||
|  * chunks out in an array within the zone.  Allocation and deallocation | ||||
|  * is nearly instantanious, and fragmentation/overhead losses are limited | ||||
|  * to a fixed worst-case amount. | ||||
|  * | ||||
|  * The downside of this slab implementation is in the chunk size | ||||
|  * multiplied by the number of zones.  ~80 zones * 128K = 10MB of VM per cpu. | ||||
|  * In a kernel implementation all this memory will be physical so | ||||
|  * the zone size is adjusted downward on machines with less physical | ||||
|  * memory.  The upside is that overhead is bounded... this is the *worst* | ||||
|  * case overhead. | ||||
|  * | ||||
|  * Slab management is done on a per-cpu basis and no locking or mutexes | ||||
|  * are required, only a critical section.  When one cpu frees memory | ||||
|  * belonging to another cpu's slab manager an asynchronous IPI message | ||||
|  * will be queued to execute the operation.   In addition, both the | ||||
|  * high level slab allocator and the low level zone allocator optimize | ||||
|  * M_ZERO requests, and the slab allocator does not have to pre initialize | ||||
|  * the linked list of chunks. | ||||
|  * | ||||
|  * XXX Balancing is needed between cpus.  Balance will be handled through | ||||
|  * asynchronous IPIs primarily by reassigning the z_Cpu ownership of chunks. | ||||
|  * | ||||
|  * XXX If we have to allocate a new zone and M_USE_RESERVE is set, use of | ||||
|  * the new zone should be restricted to M_USE_RESERVE requests only. | ||||
|  * | ||||
|  *  Alloc Size  Chunking        Number of zones | ||||
|  *  0-127       8               16 | ||||
|  *  128-255     16              8 | ||||
|  *  256-511     32              8 | ||||
|  *  512-1023    64              8 | ||||
|  *  1024-2047   128             8 | ||||
|  *  2048-4095   256             8 | ||||
|  *  4096-8191   512             8 | ||||
|  *  8192-16383  1024            8 | ||||
|  *  16384-32767 2048            8 | ||||
|  *  (if RT_MM_PAGE_SIZE is 4K the maximum zone allocation is 16383) | ||||
|  * | ||||
|  *  Allocations >= zone_limit go directly to kmem. | ||||
|  * | ||||
|  *          API REQUIREMENTS AND SIDE EFFECTS | ||||
|  * | ||||
|  *    To operate as a drop-in replacement to the FreeBSD-4.x malloc() we | ||||
|  *    have remained compatible with the following API requirements: | ||||
|  * | ||||
|  *    + small power-of-2 sized allocations are power-of-2 aligned (kern_tty) | ||||
|  *    + all power-of-2 sized allocations are power-of-2 aligned (twe) | ||||
|  *    + malloc(0) is allowed and returns non-RT_NULL (ahc driver) | ||||
|  *    + ability to allocate arbitrarily large chunks of memory | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Chunk structure for free elements | ||||
|  */ | ||||
| typedef struct slab_chunk | ||||
| { | ||||
|     struct slab_chunk *c_next; | ||||
| } slab_chunk; | ||||
|  | ||||
| /* | ||||
|  * The IN-BAND zone header is placed at the beginning of each zone. | ||||
|  */ | ||||
| typedef struct slab_zone | ||||
| { | ||||
|     rt_int32_t  z_magic;        /* magic number for sanity check */ | ||||
|     rt_int32_t  z_nfree;        /* total free chunks / ualloc space in zone */ | ||||
|     rt_int32_t  z_nmax;         /* maximum free chunks */ | ||||
|  | ||||
|     struct slab_zone *z_next;   /* zoneary[] link if z_nfree non-zero */ | ||||
|     rt_uint8_t  *z_baseptr;     /* pointer to start of chunk array */ | ||||
|  | ||||
|     rt_int32_t  z_uindex;       /* current initial allocation index */ | ||||
|     rt_int32_t  z_chunksize;    /* chunk size for validation */ | ||||
|  | ||||
|     rt_int32_t  z_zoneindex;    /* zone index */ | ||||
|     slab_chunk  *z_freechunk;   /* free chunk list */ | ||||
| } slab_zone; | ||||
|  | ||||
| #define ZALLOC_SLAB_MAGIC       0x51ab51ab | ||||
| #define ZALLOC_ZONE_LIMIT       (16 * 1024)     /* max slab-managed alloc */ | ||||
| #define ZALLOC_MIN_ZONE_SIZE    (32 * 1024)     /* minimum zone size */ | ||||
| #define ZALLOC_MAX_ZONE_SIZE    (128 * 1024)    /* maximum zone size */ | ||||
| #define NZONES                  72              /* number of zones */ | ||||
| #define ZONE_RELEASE_THRESH     2               /* threshold number of zones */ | ||||
|  | ||||
| static slab_zone *zone_array[NZONES];   /* linked list of zones NFree > 0 */ | ||||
| static slab_zone *zone_free;            /* whole zones that have become free */ | ||||
|  | ||||
| static int zone_free_cnt; | ||||
| static int zone_size; | ||||
| static int zone_limit; | ||||
| static int zone_page_cnt; | ||||
|  | ||||
| /* | ||||
|  * Misc constants.  Note that allocations that are exact multiples of | ||||
|  * RT_MM_PAGE_SIZE, or exceed the zone limit, fall through to the kmem module. | ||||
|  */ | ||||
| #define MIN_CHUNK_SIZE      8       /* in bytes */ | ||||
| #define MIN_CHUNK_MASK      (MIN_CHUNK_SIZE - 1) | ||||
|  | ||||
| /* | ||||
|  * Array of descriptors that describe the contents of each page | ||||
|  */ | ||||
| #define PAGE_TYPE_FREE      0x00 | ||||
| #define PAGE_TYPE_SMALL     0x01 | ||||
| #define PAGE_TYPE_LARGE     0x02 | ||||
| struct memusage | ||||
| { | ||||
|     rt_uint32_t type: 2 ;       /* page type */ | ||||
|     rt_uint32_t size: 30;       /* pages allocated or offset from zone */ | ||||
| }; | ||||
| static struct memusage *memusage = RT_NULL; | ||||
| #define btokup(addr)    \ | ||||
|     (&memusage[((rt_ubase_t)(addr) - heap_start) >> RT_MM_PAGE_BITS]) | ||||
|  | ||||
| static rt_ubase_t heap_start, heap_end; | ||||
|  | ||||
| /* page allocator */ | ||||
| struct rt_page_head | ||||
| { | ||||
|     struct rt_page_head *next;      /* next valid page */ | ||||
|     rt_size_t page;                 /* number of page  */ | ||||
|  | ||||
|     /* dummy */ | ||||
|     char dummy[RT_MM_PAGE_SIZE - (sizeof(struct rt_page_head *) + sizeof(rt_size_t))]; | ||||
| }; | ||||
| static struct rt_page_head *rt_page_list; | ||||
| static struct rt_semaphore heap_sem; | ||||
|  | ||||
| void *rt_page_alloc(rt_size_t npages) | ||||
| { | ||||
|     struct rt_page_head *b, *n; | ||||
|     struct rt_page_head **prev; | ||||
|  | ||||
|     if (npages == 0) | ||||
|         return RT_NULL; | ||||
|  | ||||
|     /* lock heap */ | ||||
|     rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|     for (prev = &rt_page_list; (b = *prev) != RT_NULL; prev = &(b->next)) | ||||
|     { | ||||
|         if (b->page > npages) | ||||
|         { | ||||
|             /* splite pages */ | ||||
|             n       = b + npages; | ||||
|             n->next = b->next; | ||||
|             n->page = b->page - npages; | ||||
|             *prev   = n; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         if (b->page == npages) | ||||
|         { | ||||
|             /* this node fit, remove this node */ | ||||
|             *prev = b->next; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /* unlock heap */ | ||||
|     rt_sem_release(&heap_sem); | ||||
|  | ||||
|     return b; | ||||
| } | ||||
|  | ||||
| void rt_page_free(void *addr, rt_size_t npages) | ||||
| { | ||||
|     struct rt_page_head *b, *n; | ||||
|     struct rt_page_head **prev; | ||||
|  | ||||
|     RT_ASSERT(addr != RT_NULL); | ||||
|     RT_ASSERT((rt_ubase_t)addr % RT_MM_PAGE_SIZE == 0); | ||||
|     RT_ASSERT(npages != 0); | ||||
|  | ||||
|     n = (struct rt_page_head *)addr; | ||||
|  | ||||
|     /* lock heap */ | ||||
|     rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
|     for (prev = &rt_page_list; (b = *prev) != RT_NULL; prev = &(b->next)) | ||||
|     { | ||||
|         RT_ASSERT(b->page > 0); | ||||
|         RT_ASSERT(b > n || b + b->page <= n); | ||||
|  | ||||
|         if (b + b->page == n) | ||||
|         { | ||||
|             if (b + (b->page += npages) == b->next) | ||||
|             { | ||||
|                 b->page += b->next->page; | ||||
|                 b->next  = b->next->next; | ||||
|             } | ||||
|  | ||||
|             goto _return; | ||||
|         } | ||||
|  | ||||
|         if (b == n + npages) | ||||
|         { | ||||
|             n->page = b->page + npages; | ||||
|             n->next = b->next; | ||||
|             *prev   = n; | ||||
|  | ||||
|             goto _return; | ||||
|         } | ||||
|  | ||||
|         if (b > n + npages) | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     n->page = npages; | ||||
|     n->next = b; | ||||
|     *prev   = n; | ||||
|  | ||||
| _return: | ||||
|     /* unlock heap */ | ||||
|     rt_sem_release(&heap_sem); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Initialize the page allocator | ||||
|  */ | ||||
| static void rt_page_init(void *addr, rt_size_t npages) | ||||
| { | ||||
|     RT_ASSERT(addr != RT_NULL); | ||||
|     RT_ASSERT(npages != 0); | ||||
|  | ||||
|     rt_page_list = RT_NULL; | ||||
|     rt_page_free(addr, npages); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * | ||||
|  * This function will init system heap | ||||
|  * | ||||
|  * @param begin_addr the beginning address of system page | ||||
|  * @param end_addr the end address of system page | ||||
|  */ | ||||
| void rt_system_heap_init(void *begin_addr, void *end_addr) | ||||
| { | ||||
|     rt_uint32_t limsize, npages; | ||||
|  | ||||
|     RT_DEBUG_NOT_IN_INTERRUPT; | ||||
|  | ||||
|     /* align begin and end addr to page */ | ||||
|     heap_start = RT_ALIGN((rt_ubase_t)begin_addr, RT_MM_PAGE_SIZE); | ||||
|     heap_end   = RT_ALIGN_DOWN((rt_ubase_t)end_addr, RT_MM_PAGE_SIZE); | ||||
|  | ||||
|     if (heap_start >= heap_end) | ||||
|     { | ||||
|         rt_kprintf("rt_system_heap_init, wrong address[0x%x - 0x%x]\n", | ||||
|                    (rt_ubase_t)begin_addr, (rt_ubase_t)end_addr); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     limsize = heap_end - heap_start; | ||||
|     npages  = limsize / RT_MM_PAGE_SIZE; | ||||
|  | ||||
|     /* initialize heap semaphore */ | ||||
|     rt_sem_init(&heap_sem, "heap", 1, RT_IPC_FLAG_FIFO); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SLAB, ("heap[0x%x - 0x%x], size 0x%x, 0x%x pages\n", | ||||
|                                  heap_start, heap_end, limsize, npages)); | ||||
|  | ||||
|     /* init pages */ | ||||
|     rt_page_init((void *)heap_start, npages); | ||||
|  | ||||
|     /* calculate zone size */ | ||||
|     zone_size = ZALLOC_MIN_ZONE_SIZE; | ||||
|     while (zone_size < ZALLOC_MAX_ZONE_SIZE && (zone_size << 1) < (limsize / 1024)) | ||||
|         zone_size <<= 1; | ||||
|  | ||||
|     zone_limit = zone_size / 4; | ||||
|     if (zone_limit > ZALLOC_ZONE_LIMIT) | ||||
|         zone_limit = ZALLOC_ZONE_LIMIT; | ||||
|  | ||||
|     zone_page_cnt = zone_size / RT_MM_PAGE_SIZE; | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SLAB, ("zone size 0x%x, zone page count 0x%x\n", | ||||
|                                  zone_size, zone_page_cnt)); | ||||
|  | ||||
|     /* allocate memusage array */ | ||||
|     limsize  = npages * sizeof(struct memusage); | ||||
|     limsize  = RT_ALIGN(limsize, RT_MM_PAGE_SIZE); | ||||
|     memusage = rt_page_alloc(limsize / RT_MM_PAGE_SIZE); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SLAB, ("memusage 0x%x, size 0x%x\n", | ||||
|                                  (rt_ubase_t)memusage, limsize)); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Calculate the zone index for the allocation request size and set the | ||||
|  * allocation request size to that particular zone's chunk size. | ||||
|  */ | ||||
| rt_inline int zoneindex(rt_size_t *bytes) | ||||
| { | ||||
|     /* unsigned for shift opt */ | ||||
|     rt_ubase_t n = (rt_ubase_t)(*bytes); | ||||
|  | ||||
|     if (n < 128) | ||||
|     { | ||||
|         *bytes = n = (n + 7) & ~7; | ||||
|  | ||||
|         /* 8 byte chunks, 16 zones */ | ||||
|         return (n / 8 - 1); | ||||
|     } | ||||
|     if (n < 256) | ||||
|     { | ||||
|         *bytes = n = (n + 15) & ~15; | ||||
|  | ||||
|         return (n / 16 + 7); | ||||
|     } | ||||
|     if (n < 8192) | ||||
|     { | ||||
|         if (n < 512) | ||||
|         { | ||||
|             *bytes = n = (n + 31) & ~31; | ||||
|  | ||||
|             return (n / 32 + 15); | ||||
|         } | ||||
|         if (n < 1024) | ||||
|         { | ||||
|             *bytes = n = (n + 63) & ~63; | ||||
|  | ||||
|             return (n / 64 + 23); | ||||
|         } | ||||
|         if (n < 2048) | ||||
|         { | ||||
|             *bytes = n = (n + 127) & ~127; | ||||
|  | ||||
|             return (n / 128 + 31); | ||||
|         } | ||||
|         if (n < 4096) | ||||
|         { | ||||
|             *bytes = n = (n + 255) & ~255; | ||||
|  | ||||
|             return (n / 256 + 39); | ||||
|         } | ||||
|         *bytes = n = (n + 511) & ~511; | ||||
|  | ||||
|         return (n / 512 + 47); | ||||
|     } | ||||
|     if (n < 16384) | ||||
|     { | ||||
|         *bytes = n = (n + 1023) & ~1023; | ||||
|  | ||||
|         return (n / 1024 + 55); | ||||
|     } | ||||
|  | ||||
|     rt_kprintf("Unexpected byte count %d", n); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @addtogroup MM | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will allocate a block from system heap memory. | ||||
|  * - If the nbytes is less than zero, | ||||
|  * or | ||||
|  * - If there is no nbytes sized memory valid in system, | ||||
|  * the RT_NULL is returned. | ||||
|  * | ||||
|  * @param size the size of memory to be allocated | ||||
|  * | ||||
|  * @return the allocated memory | ||||
|  */ | ||||
| void *rt_malloc(rt_size_t size) | ||||
| { | ||||
|     slab_zone *z; | ||||
|     rt_int32_t zi; | ||||
|     slab_chunk *chunk; | ||||
|     struct memusage *kup; | ||||
|  | ||||
|     /* zero size, return RT_NULL */ | ||||
|     if (size == 0) | ||||
|         return RT_NULL; | ||||
|  | ||||
|     /* | ||||
|      * Handle large allocations directly.  There should not be very many of | ||||
|      * these so performance is not a big issue. | ||||
|      */ | ||||
|     if (size >= zone_limit) | ||||
|     { | ||||
|         size = RT_ALIGN(size, RT_MM_PAGE_SIZE); | ||||
|  | ||||
|         chunk = rt_page_alloc(size >> RT_MM_PAGE_BITS); | ||||
|         if (chunk == RT_NULL) | ||||
|             return RT_NULL; | ||||
|  | ||||
|         /* set kup */ | ||||
|         kup = btokup(chunk); | ||||
|         kup->type = PAGE_TYPE_LARGE; | ||||
|         kup->size = size >> RT_MM_PAGE_BITS; | ||||
|  | ||||
|         RT_DEBUG_LOG(RT_DEBUG_SLAB, | ||||
|                      ("malloc a large memory 0x%x, page cnt %d, kup %d\n", | ||||
|                       size, | ||||
|                       size >> RT_MM_PAGE_BITS, | ||||
|                       ((rt_ubase_t)chunk - heap_start) >> RT_MM_PAGE_BITS)); | ||||
|  | ||||
|         /* lock heap */ | ||||
|         rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
|         used_mem += size; | ||||
|         if (used_mem > max_mem) | ||||
|             max_mem = used_mem; | ||||
| #endif | ||||
|         goto done; | ||||
|     } | ||||
|  | ||||
|     /* lock heap */ | ||||
|     rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
|     /* | ||||
|      * Attempt to allocate out of an existing zone.  First try the free list, | ||||
|      * then allocate out of unallocated space.  If we find a good zone move | ||||
|      * it to the head of the list so later allocations find it quickly | ||||
|      * (we might have thousands of zones in the list). | ||||
|      * | ||||
|      * Note: zoneindex() will panic of size is too large. | ||||
|      */ | ||||
|     zi = zoneindex(&size); | ||||
|     RT_ASSERT(zi < NZONES); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_SLAB, ("try to malloc 0x%x on zone: %d\n", size, zi)); | ||||
|  | ||||
|     if ((z = zone_array[zi]) != RT_NULL) | ||||
|     { | ||||
|         RT_ASSERT(z->z_nfree > 0); | ||||
|  | ||||
|         /* Remove us from the zone_array[] when we become empty */ | ||||
|         if (--z->z_nfree == 0) | ||||
|         { | ||||
|             zone_array[zi] = z->z_next; | ||||
|             z->z_next = RT_NULL; | ||||
|         } | ||||
|  | ||||
|         /* | ||||
|          * No chunks are available but nfree said we had some memory, so | ||||
|          * it must be available in the never-before-used-memory area | ||||
|          * governed by uindex.  The consequences are very serious if our zone | ||||
|          * got corrupted so we use an explicit rt_kprintf rather then a KASSERT. | ||||
|          */ | ||||
|         if (z->z_uindex + 1 != z->z_nmax) | ||||
|         { | ||||
|             z->z_uindex = z->z_uindex + 1; | ||||
|             chunk = (slab_chunk *)(z->z_baseptr + z->z_uindex * size); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             /* find on free chunk list */ | ||||
|             chunk = z->z_freechunk; | ||||
|  | ||||
|             /* remove this chunk from list */ | ||||
|             z->z_freechunk = z->z_freechunk->c_next; | ||||
|         } | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
|         used_mem += z->z_chunksize; | ||||
|         if (used_mem > max_mem) | ||||
|             max_mem = used_mem; | ||||
| #endif | ||||
|  | ||||
|         goto done; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * If all zones are exhausted we need to allocate a new zone for this | ||||
|      * index. | ||||
|      * | ||||
|      * At least one subsystem, the tty code (see CROUND) expects power-of-2 | ||||
|      * allocations to be power-of-2 aligned.  We maintain compatibility by | ||||
|      * adjusting the base offset below. | ||||
|      */ | ||||
|     { | ||||
|         rt_int32_t off; | ||||
|  | ||||
|         if ((z = zone_free) != RT_NULL) | ||||
|         { | ||||
|             /* remove zone from free zone list */ | ||||
|             zone_free = z->z_next; | ||||
|             -- zone_free_cnt; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             /* unlock heap, since page allocator will think about lock */ | ||||
|             rt_sem_release(&heap_sem); | ||||
|  | ||||
|             /* allocate a zone from page */ | ||||
|             z = rt_page_alloc(zone_size / RT_MM_PAGE_SIZE); | ||||
|             if (z == RT_NULL) | ||||
|             { | ||||
|                 chunk = RT_NULL; | ||||
|                 goto __exit; | ||||
|             } | ||||
|  | ||||
|             /* lock heap */ | ||||
|             rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
|             RT_DEBUG_LOG(RT_DEBUG_SLAB, ("alloc a new zone: 0x%x\n", | ||||
|                                          (rt_ubase_t)z)); | ||||
|  | ||||
|             /* set message usage */ | ||||
|             for (off = 0, kup = btokup(z); off < zone_page_cnt; off ++) | ||||
|             { | ||||
|                 kup->type = PAGE_TYPE_SMALL; | ||||
|                 kup->size = off; | ||||
|  | ||||
|                 kup ++; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* clear to zero */ | ||||
|         rt_memset(z, 0, sizeof(slab_zone)); | ||||
|  | ||||
|         /* offset of slab zone struct in zone */ | ||||
|         off = sizeof(slab_zone); | ||||
|  | ||||
|         /* | ||||
|          * Guarentee power-of-2 alignment for power-of-2-sized chunks. | ||||
|          * Otherwise just 8-byte align the data. | ||||
|          */ | ||||
|         if ((size | (size - 1)) + 1 == (size << 1)) | ||||
|             off = (off + size - 1) & ~(size - 1); | ||||
|         else | ||||
|             off = (off + MIN_CHUNK_MASK) & ~MIN_CHUNK_MASK; | ||||
|  | ||||
|         z->z_magic     = ZALLOC_SLAB_MAGIC; | ||||
|         z->z_zoneindex = zi; | ||||
|         z->z_nmax      = (zone_size - off) / size; | ||||
|         z->z_nfree     = z->z_nmax - 1; | ||||
|         z->z_baseptr   = (rt_uint8_t *)z + off; | ||||
|         z->z_uindex    = 0; | ||||
|         z->z_chunksize = size; | ||||
|  | ||||
|         chunk = (slab_chunk *)(z->z_baseptr + z->z_uindex * size); | ||||
|  | ||||
|         /* link to zone array */ | ||||
|         z->z_next = zone_array[zi]; | ||||
|         zone_array[zi] = z; | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
|         used_mem += z->z_chunksize; | ||||
|         if (used_mem > max_mem) | ||||
|             max_mem = used_mem; | ||||
| #endif | ||||
|     } | ||||
|  | ||||
| done: | ||||
|     rt_sem_release(&heap_sem); | ||||
|     RT_OBJECT_HOOK_CALL(rt_malloc_hook, ((char *)chunk, size)); | ||||
|  | ||||
| __exit: | ||||
|     return chunk; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will change the size of previously allocated memory block. | ||||
|  * | ||||
|  * @param ptr the previously allocated memory block | ||||
|  * @param size the new size of memory block | ||||
|  * | ||||
|  * @return the allocated memory | ||||
|  */ | ||||
| void *rt_realloc(void *ptr, rt_size_t size) | ||||
| { | ||||
|     void *nptr; | ||||
|     slab_zone *z; | ||||
|     struct memusage *kup; | ||||
|  | ||||
|     if (ptr == RT_NULL) | ||||
|         return rt_malloc(size); | ||||
|     if (size == 0) | ||||
|     { | ||||
|         rt_free(ptr); | ||||
|  | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * Get the original allocation's zone.  If the new request winds up | ||||
|      * using the same chunk size we do not have to do anything. | ||||
|      */ | ||||
|     kup = btokup((rt_ubase_t)ptr & ~RT_MM_PAGE_MASK); | ||||
|     if (kup->type == PAGE_TYPE_LARGE) | ||||
|     { | ||||
|         rt_size_t osize; | ||||
|  | ||||
|         osize = kup->size << RT_MM_PAGE_BITS; | ||||
|         if ((nptr = rt_malloc(size)) == RT_NULL) | ||||
|             return RT_NULL; | ||||
|         rt_memcpy(nptr, ptr, size > osize ? osize : size); | ||||
|         rt_free(ptr); | ||||
|  | ||||
|         return nptr; | ||||
|     } | ||||
|     else if (kup->type == PAGE_TYPE_SMALL) | ||||
|     { | ||||
|         z = (slab_zone *)(((rt_ubase_t)ptr & ~RT_MM_PAGE_MASK) - | ||||
|                           kup->size * RT_MM_PAGE_SIZE); | ||||
|         RT_ASSERT(z->z_magic == ZALLOC_SLAB_MAGIC); | ||||
|  | ||||
|         zoneindex(&size); | ||||
|         if (z->z_chunksize == size) | ||||
|             return (ptr); /* same chunk */ | ||||
|  | ||||
|         /* | ||||
|          * Allocate memory for the new request size.  Note that zoneindex has | ||||
|          * already adjusted the request size to the appropriate chunk size, which | ||||
|          * should optimize our bcopy().  Then copy and return the new pointer. | ||||
|          */ | ||||
|         if ((nptr = rt_malloc(size)) == RT_NULL) | ||||
|             return RT_NULL; | ||||
|  | ||||
|         rt_memcpy(nptr, ptr, size > z->z_chunksize ? z->z_chunksize : size); | ||||
|         rt_free(ptr); | ||||
|  | ||||
|         return nptr; | ||||
|     } | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will contiguously allocate enough space for count objects | ||||
|  * that are size bytes of memory each and returns a pointer to the allocated | ||||
|  * memory. | ||||
|  * | ||||
|  * The allocated memory is filled with bytes of value zero. | ||||
|  * | ||||
|  * @param count number of objects to allocate | ||||
|  * @param size size of the objects to allocate | ||||
|  * | ||||
|  * @return pointer to allocated memory / NULL pointer if there is an error | ||||
|  */ | ||||
| void *rt_calloc(rt_size_t count, rt_size_t size) | ||||
| { | ||||
|     void *p; | ||||
|  | ||||
|     /* allocate 'count' objects of size 'size' */ | ||||
|     p = rt_malloc(count * size); | ||||
|  | ||||
|     /* zero the memory */ | ||||
|     if (p) | ||||
|         rt_memset(p, 0, count * size); | ||||
|  | ||||
|     return p; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will release the previous allocated memory block by rt_malloc. | ||||
|  * The released memory block is taken back to system heap. | ||||
|  * | ||||
|  * @param ptr the address of memory which will be released | ||||
|  */ | ||||
| void rt_free(void *ptr) | ||||
| { | ||||
|     slab_zone *z; | ||||
|     slab_chunk *chunk; | ||||
|     struct memusage *kup; | ||||
|  | ||||
|     /* free a RT_NULL pointer */ | ||||
|     if (ptr == RT_NULL) | ||||
|         return ; | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_free_hook, (ptr)); | ||||
|  | ||||
|     /* get memory usage */ | ||||
| #if RT_DEBUG_SLAB | ||||
|     { | ||||
|         rt_ubase_t addr = ((rt_ubase_t)ptr & ~RT_MM_PAGE_MASK); | ||||
|         RT_DEBUG_LOG(RT_DEBUG_SLAB, | ||||
|                      ("free a memory 0x%x and align to 0x%x, kup index %d\n", | ||||
|                       (rt_ubase_t)ptr, | ||||
|                       (rt_ubase_t)addr, | ||||
|                       ((rt_ubase_t)(addr) - heap_start) >> RT_MM_PAGE_BITS)); | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|     kup = btokup((rt_ubase_t)ptr & ~RT_MM_PAGE_MASK); | ||||
|     /* release large allocation */ | ||||
|     if (kup->type == PAGE_TYPE_LARGE) | ||||
|     { | ||||
|         rt_ubase_t size; | ||||
|  | ||||
|         /* lock heap */ | ||||
|         rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|         /* clear page counter */ | ||||
|         size = kup->size; | ||||
|         kup->size = 0; | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
|         used_mem -= size * RT_MM_PAGE_SIZE; | ||||
| #endif | ||||
|         rt_sem_release(&heap_sem); | ||||
|  | ||||
|         RT_DEBUG_LOG(RT_DEBUG_SLAB, | ||||
|                      ("free large memory block 0x%x, page count %d\n", | ||||
|                       (rt_ubase_t)ptr, size)); | ||||
|  | ||||
|         /* free this page */ | ||||
|         rt_page_free(ptr, size); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     /* lock heap */ | ||||
|     rt_sem_take(&heap_sem, RT_WAITING_FOREVER); | ||||
|  | ||||
|     /* zone case. get out zone. */ | ||||
|     z = (slab_zone *)(((rt_ubase_t)ptr & ~RT_MM_PAGE_MASK) - | ||||
|                       kup->size * RT_MM_PAGE_SIZE); | ||||
|     RT_ASSERT(z->z_magic == ZALLOC_SLAB_MAGIC); | ||||
|  | ||||
|     chunk          = (slab_chunk *)ptr; | ||||
|     chunk->c_next  = z->z_freechunk; | ||||
|     z->z_freechunk = chunk; | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
|     used_mem -= z->z_chunksize; | ||||
| #endif | ||||
|  | ||||
|     /* | ||||
|      * Bump the number of free chunks.  If it becomes non-zero the zone | ||||
|      * must be added back onto the appropriate list. | ||||
|      */ | ||||
|     if (z->z_nfree++ == 0) | ||||
|     { | ||||
|         z->z_next = zone_array[z->z_zoneindex]; | ||||
|         zone_array[z->z_zoneindex] = z; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * If the zone becomes totally free, and there are other zones we | ||||
|      * can allocate from, move this zone to the FreeZones list.  Since | ||||
|      * this code can be called from an IPI callback, do *NOT* try to mess | ||||
|      * with kernel_map here.  Hysteresis will be performed at malloc() time. | ||||
|      */ | ||||
|     if (z->z_nfree == z->z_nmax && | ||||
|         (z->z_next || zone_array[z->z_zoneindex] != z)) | ||||
|     { | ||||
|         slab_zone **pz; | ||||
|  | ||||
|         RT_DEBUG_LOG(RT_DEBUG_SLAB, ("free zone 0x%x\n", | ||||
|                                      (rt_ubase_t)z, z->z_zoneindex)); | ||||
|  | ||||
|         /* remove zone from zone array list */ | ||||
|         for (pz = &zone_array[z->z_zoneindex]; z != *pz; pz = &(*pz)->z_next) | ||||
|             ; | ||||
|         *pz = z->z_next; | ||||
|  | ||||
|         /* reset zone */ | ||||
|         z->z_magic = -1; | ||||
|  | ||||
|         /* insert to free zone list */ | ||||
|         z->z_next = zone_free; | ||||
|         zone_free = z; | ||||
|  | ||||
|         ++ zone_free_cnt; | ||||
|  | ||||
|         /* release zone to page allocator */ | ||||
|         if (zone_free_cnt > ZONE_RELEASE_THRESH) | ||||
|         { | ||||
|             register rt_base_t i; | ||||
|  | ||||
|             z         = zone_free; | ||||
|             zone_free = z->z_next; | ||||
|             -- zone_free_cnt; | ||||
|  | ||||
|             /* set message usage */ | ||||
|             for (i = 0, kup = btokup(z); i < zone_page_cnt; i ++) | ||||
|             { | ||||
|                 kup->type = PAGE_TYPE_FREE; | ||||
|                 kup->size = 0; | ||||
|                 kup ++; | ||||
|             } | ||||
|  | ||||
|             /* unlock heap */ | ||||
|             rt_sem_release(&heap_sem); | ||||
|  | ||||
|             /* release pages */ | ||||
|             rt_page_free(z, zone_size / RT_MM_PAGE_SIZE); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
|     /* unlock heap */ | ||||
|     rt_sem_release(&heap_sem); | ||||
| } | ||||
|  | ||||
| #ifdef RT_MEM_STATS | ||||
| void rt_memory_info(rt_uint32_t *total, | ||||
|                     rt_uint32_t *used, | ||||
|                     rt_uint32_t *max_used) | ||||
| { | ||||
|     if (total != RT_NULL) | ||||
|         *total = heap_end - heap_start; | ||||
|  | ||||
|     if (used  != RT_NULL) | ||||
|         *used = used_mem; | ||||
|  | ||||
|     if (max_used != RT_NULL) | ||||
|         *max_used = max_mem; | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_FINSH | ||||
| #include <finsh.h> | ||||
|  | ||||
| void list_mem(void) | ||||
| { | ||||
|     rt_kprintf("total memory: %d\n", heap_end - heap_start); | ||||
|     rt_kprintf("used memory : %d\n", used_mem); | ||||
|     rt_kprintf("maximum allocated memory: %d\n", max_mem); | ||||
| } | ||||
| FINSH_FUNCTION_EXPORT(list_mem, list memory usage information) | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| /**@}*/ | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										834
									
								
								source/rt_thread/src/thread.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										834
									
								
								source/rt_thread/src/thread.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,834 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-03-28     Bernard      first version | ||||
|  * 2006-04-29     Bernard      implement thread timer | ||||
|  * 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 | ||||
|  * 2006-09-03     Bernard      implement rt_thread_detach | ||||
|  * 2008-02-16     Bernard      fixed the rt_thread_timeout bug | ||||
|  * 2010-03-21     Bernard      change the errno of rt_thread_delay/sleep to | ||||
|  *                             RT_EOK. | ||||
|  * 2010-11-10     Bernard      add cleanup callback function in thread exit. | ||||
|  * 2011-09-01     Bernard      fixed rt_thread_exit issue when the current | ||||
|  *                             thread preempted, which reported by Jiaxing Lee. | ||||
|  * 2011-09-08     Bernard      fixed the scheduling issue in rt_thread_startup. | ||||
|  * 2012-12-29     Bernard      fixed compiling warning. | ||||
|  * 2016-08-09     ArdaFu       add thread suspend and resume hook. | ||||
|  * 2017-04-10     armink       fixed the rt_thread_delete and rt_thread_detach | ||||
|                                bug when thread has not startup. | ||||
|  * 2018-11-22     Jesven       yield is same to rt_schedule | ||||
|  *                             add support for tasks bound to cpu | ||||
|  */ | ||||
|  | ||||
| #include <rthw.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| extern rt_list_t rt_thread_priority_table[RT_THREAD_PRIORITY_MAX]; | ||||
| extern struct rt_thread *rt_current_thread; | ||||
| extern rt_list_t rt_thread_defunct; | ||||
|  | ||||
| #ifdef RT_USING_HOOK | ||||
|  | ||||
| static void (*rt_thread_suspend_hook)(rt_thread_t thread); | ||||
| static void (*rt_thread_resume_hook) (rt_thread_t thread); | ||||
| static void (*rt_thread_inited_hook) (rt_thread_t thread); | ||||
|  | ||||
| /** | ||||
|  * @ingroup Hook | ||||
|  * This function sets a hook function when the system suspend a thread. | ||||
|  * | ||||
|  * @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; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup Hook | ||||
|  * This function sets a hook function when the system resume a thread. | ||||
|  * | ||||
|  * @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; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup Hook | ||||
|  * This function sets a hook function when a thread is initialized. | ||||
|  * | ||||
|  * @param hook the specified hook function | ||||
|  */ | ||||
| void rt_thread_inited_sethook(void (*hook)(rt_thread_t thread)) | ||||
| { | ||||
|     rt_thread_inited_hook = hook; | ||||
| } | ||||
|  | ||||
| #endif | ||||
|  | ||||
| /* must be invoke witch rt_hw_interrupt_disable */ | ||||
| static void _thread_cleanup_execute(rt_thread_t thread) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* invoke thread cleanup */ | ||||
|     if (thread->cleanup != RT_NULL) | ||||
|         thread->cleanup(thread); | ||||
|  | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| void rt_thread_exit(void) | ||||
| { | ||||
|     struct rt_thread *thread; | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* get current thread */ | ||||
|     thread = rt_current_thread; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     _thread_cleanup_execute(thread); | ||||
|  | ||||
|     /* remove from schedule */ | ||||
|     rt_schedule_remove_thread(thread); | ||||
|     /* change stat */ | ||||
|     thread->stat = RT_THREAD_CLOSE; | ||||
|  | ||||
|     /* remove it from timer list */ | ||||
|     rt_timer_detach(&thread->thread_timer); | ||||
|  | ||||
|     if (rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE) | ||||
|     { | ||||
|         rt_object_detach((rt_object_t)thread); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         /* insert to defunct thread list */ | ||||
|         rt_list_insert_after(&rt_thread_defunct, &(thread->tlist)); | ||||
|     } | ||||
|  | ||||
|     /* switch to next task */ | ||||
|     rt_schedule(); | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| static rt_err_t _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) | ||||
| { | ||||
|     /* init thread list */ | ||||
|     rt_list_init(&(thread->tlist)); | ||||
|  | ||||
|     thread->entry = (void *)entry; | ||||
|     thread->parameter = parameter; | ||||
|  | ||||
|     /* stack init */ | ||||
|     thread->stack_addr = stack_start; | ||||
|     thread->stack_size = stack_size; | ||||
|  | ||||
|     /* init thread stack */ | ||||
|     rt_memset(thread->stack_addr, '#', thread->stack_size); | ||||
| #ifdef ARCH_CPU_STACK_GROWS_UPWARD | ||||
|     thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter, | ||||
|                                           (void *)((char *)thread->stack_addr), | ||||
|                                           (void *)rt_thread_exit); | ||||
| #else | ||||
|     thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter, | ||||
|                                           (rt_uint8_t *)((char *)thread->stack_addr + thread->stack_size - sizeof(rt_ubase_t)), | ||||
|                                           (void *)rt_thread_exit); | ||||
| #endif | ||||
|  | ||||
|     /* priority init */ | ||||
|     RT_ASSERT(priority < RT_THREAD_PRIORITY_MAX); | ||||
|     thread->init_priority    = priority; | ||||
|     thread->current_priority = priority; | ||||
|  | ||||
|     thread->number_mask = 0; | ||||
| #if RT_THREAD_PRIORITY_MAX > 32 | ||||
|     thread->number = 0; | ||||
|     thread->high_mask = 0; | ||||
| #endif | ||||
|  | ||||
|     /* 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; | ||||
|  | ||||
|     /* initialize thread timer */ | ||||
|     rt_timer_init(&(thread->thread_timer), | ||||
|                   thread->name, | ||||
|                   rt_thread_timeout, | ||||
|                   thread, | ||||
|                   0, | ||||
|                   RT_TIMER_FLAG_ONE_SHOT); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_thread_inited_hook, (thread)); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Thread | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will initialize a thread, normally it's used to initialize a | ||||
|  * static thread object. | ||||
|  * | ||||
|  * @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 | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t 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) | ||||
| { | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT(stack_start != RT_NULL); | ||||
|  | ||||
|     /* initialize 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); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will return self thread object | ||||
|  * | ||||
|  * @return the self thread object | ||||
|  */ | ||||
| rt_thread_t rt_thread_self(void) | ||||
| { | ||||
|     return rt_current_thread; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will start a thread and put it to system ready queue | ||||
|  * | ||||
|  * @param thread the thread to be started | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_thread_startup(rt_thread_t thread) | ||||
| { | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_INIT); | ||||
|     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread); | ||||
|  | ||||
|     /* set current priority to initialize priority */ | ||||
|     thread->current_priority = thread->init_priority; | ||||
|  | ||||
|     /* calculate priority attribute */ | ||||
| #if RT_THREAD_PRIORITY_MAX > 32 | ||||
|     thread->number      = thread->current_priority >> 3;            /* 5bit */ | ||||
|     thread->number_mask = 1L << thread->number; | ||||
|     thread->high_mask   = 1L << (thread->current_priority & 0x07);  /* 3bit */ | ||||
| #else | ||||
|     thread->number_mask = 1L << thread->current_priority; | ||||
| #endif | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("startup a thread:%s with priority:%d\n", | ||||
|                                    thread->name, thread->init_priority)); | ||||
|     /* 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; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will detach a thread. The thread object will be removed from | ||||
|  * thread queue and detached/deleted from system object management. | ||||
|  * | ||||
|  * @param thread the thread to be deleted | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_thread_detach(rt_thread_t thread) | ||||
| { | ||||
|     rt_base_t lock; | ||||
|  | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread); | ||||
|     RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread)); | ||||
|  | ||||
|     if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_CLOSE) | ||||
|         return RT_EOK; | ||||
|  | ||||
|     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT) | ||||
|     { | ||||
|         /* remove from schedule */ | ||||
|         rt_schedule_remove_thread(thread); | ||||
|     } | ||||
|  | ||||
|     _thread_cleanup_execute(thread); | ||||
|  | ||||
|     /* release thread timer */ | ||||
|     rt_timer_detach(&(thread->thread_timer)); | ||||
|  | ||||
|     /* change stat */ | ||||
|     thread->stat = RT_THREAD_CLOSE; | ||||
|  | ||||
|     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_list_insert_after(&rt_thread_defunct, &(thread->tlist)); | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(lock); | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_HEAP | ||||
| /** | ||||
|  * 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, | ||||
|                              void (*entry)(void *parameter), | ||||
|                              void       *parameter, | ||||
|                              rt_uint32_t stack_size, | ||||
|                              rt_uint8_t  priority, | ||||
|                              rt_uint32_t tick) | ||||
| { | ||||
|     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; | ||||
|  | ||||
|     stack_start = (void *)RT_KERNEL_MALLOC(stack_size); | ||||
|     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; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will delete a thread. The thread object will be removed from | ||||
|  * thread queue and deleted from system object management in the idle thread. | ||||
|  * | ||||
|  * @param thread the thread to be deleted | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_thread_delete(rt_thread_t thread) | ||||
| { | ||||
|     rt_base_t lock; | ||||
|  | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     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); | ||||
|  | ||||
|     if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_CLOSE) | ||||
|         return RT_EOK; | ||||
|  | ||||
|     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT) | ||||
|     { | ||||
|         /* remove from schedule */ | ||||
|         rt_schedule_remove_thread(thread); | ||||
|     } | ||||
|  | ||||
|     _thread_cleanup_execute(thread); | ||||
|  | ||||
|     /* release thread timer */ | ||||
|     rt_timer_detach(&(thread->thread_timer)); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     lock = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* change stat */ | ||||
|     thread->stat = RT_THREAD_CLOSE; | ||||
|  | ||||
|     /* insert to defunct thread list */ | ||||
|     rt_list_insert_after(&rt_thread_defunct, &(thread->tlist)); | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(lock); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * 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. | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_thread_yield(void) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|     struct rt_thread *thread; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* set to current thread */ | ||||
|     thread = rt_current_thread; | ||||
|  | ||||
|     /* if the thread stat is READY and on ready queue list */ | ||||
|     if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_READY && | ||||
|         thread->tlist.next != thread->tlist.prev) | ||||
|     { | ||||
|         /* remove thread from thread list */ | ||||
|         rt_list_remove(&(thread->tlist)); | ||||
|  | ||||
|         /* put thread to end of ready queue */ | ||||
|         rt_list_insert_before(&(rt_thread_priority_table[thread->current_priority]), | ||||
|                               &(thread->tlist)); | ||||
|  | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(level); | ||||
|  | ||||
|         rt_schedule(); | ||||
|  | ||||
|         return RT_EOK; | ||||
|     } | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will let current thread sleep for some ticks. | ||||
|  * | ||||
|  * @param tick the sleep ticks | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_thread_sleep(rt_tick_t tick) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|     struct rt_thread *thread; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|     /* set to current thread */ | ||||
|     thread = rt_current_thread; | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread); | ||||
|  | ||||
|     /* suspend thread */ | ||||
|     rt_thread_suspend(thread); | ||||
|  | ||||
|     /* 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)); | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
|  | ||||
|     rt_schedule(); | ||||
|  | ||||
|     /* clear error number of this thread to RT_EOK */ | ||||
|     if (thread->error == -RT_ETIMEOUT) | ||||
|         thread->error = RT_EOK; | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will let current thread delay for some ticks. | ||||
|  * | ||||
|  * @param tick the delay ticks | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_thread_delay(rt_tick_t tick) | ||||
| { | ||||
|     return rt_thread_sleep(tick); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will let current thread delay until (*tick + inc_tick). | ||||
|  * | ||||
|  * @param tick the tick of last wakeup. | ||||
|  * @param inc_tick the increment tick | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_thread_delay_until(rt_tick_t *tick, rt_tick_t inc_tick) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|     struct rt_thread *thread; | ||||
|  | ||||
|     RT_ASSERT(tick != RT_NULL); | ||||
|  | ||||
|     /* 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(); | ||||
|  | ||||
|     if (rt_tick_get() - *tick < inc_tick) | ||||
|     { | ||||
|         *tick = *tick + inc_tick - rt_tick_get(); | ||||
|  | ||||
|         /* suspend thread */ | ||||
|         rt_thread_suspend(thread); | ||||
|  | ||||
|         /* 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)); | ||||
|  | ||||
|         /* 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 | ||||
|     { | ||||
|         rt_hw_interrupt_enable(level); | ||||
|     } | ||||
|  | ||||
|     /* get the wakeup tick */ | ||||
|     *tick = rt_tick_get(); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will let current thread delay for some milliseconds. | ||||
|  * | ||||
|  * @param ms the delay ms 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); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * 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 | ||||
|  *  RT_THREAD_CTRL_CHANGE_PRIORITY for changing priority level of thread; | ||||
|  *  RT_THREAD_CTRL_STARTUP for starting a thread; | ||||
|  *  RT_THREAD_CTRL_CLOSE for delete a thread; | ||||
|  *  RT_THREAD_CTRL_BIND_CPU for bind the thread to a CPU. | ||||
|  * @param arg the argument of control command | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_thread_control(rt_thread_t thread, int cmd, void *arg) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|  | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread); | ||||
|  | ||||
|     switch (cmd) | ||||
|     { | ||||
|     case RT_THREAD_CTRL_CHANGE_PRIORITY: | ||||
|         /* 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; | ||||
| #endif | ||||
|  | ||||
|             /* 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; | ||||
| #endif | ||||
|         } | ||||
|  | ||||
|         /* enable interrupt */ | ||||
|         rt_hw_interrupt_enable(temp); | ||||
|         break; | ||||
|  | ||||
|     case RT_THREAD_CTRL_STARTUP: | ||||
|         return rt_thread_startup(thread); | ||||
|  | ||||
|     case RT_THREAD_CTRL_CLOSE: | ||||
|  | ||||
|         if (rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE) | ||||
|         { | ||||
|             return rt_thread_detach(thread); | ||||
|         } | ||||
| #ifdef RT_USING_HEAP | ||||
|         else | ||||
|         { | ||||
|             return rt_thread_delete(thread); | ||||
|         } | ||||
| #endif | ||||
|  | ||||
|     default: | ||||
|         break; | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will suspend the specified thread. | ||||
|  * | ||||
|  * @param thread the thread to be suspended | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  * | ||||
|  * @note if suspend self thread, after this function call, the | ||||
|  * rt_schedule() must be invoked. | ||||
|  */ | ||||
| rt_err_t rt_thread_suspend(rt_thread_t thread) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|  | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend:  %s\n", thread->name)); | ||||
|  | ||||
|     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_READY) | ||||
|     { | ||||
|         RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend: thread disorder, 0x%2x\n", | ||||
|                                        thread->stat)); | ||||
|  | ||||
|         return -RT_ERROR; | ||||
|     } | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* change thread stat */ | ||||
|     rt_schedule_remove_thread(thread); | ||||
|     thread->stat = RT_THREAD_SUSPEND | (thread->stat & ~RT_THREAD_STAT_MASK); | ||||
|  | ||||
|     /* stop thread timer anyway */ | ||||
|     rt_timer_stop(&(thread->thread_timer)); | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_thread_suspend_hook, (thread)); | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will resume a thread and put it to system ready queue. | ||||
|  * | ||||
|  * @param thread the thread to be resumed | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_thread_resume(rt_thread_t thread) | ||||
| { | ||||
|     register rt_base_t temp; | ||||
|  | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume:  %s\n", thread->name)); | ||||
|  | ||||
|     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_SUSPEND) | ||||
|     { | ||||
|         RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume: thread disorder, %d\n", | ||||
|                                        thread->stat)); | ||||
|  | ||||
|         return -RT_ERROR; | ||||
|     } | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     temp = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     /* remove from suspend list */ | ||||
|     rt_list_remove(&(thread->tlist)); | ||||
|  | ||||
|     rt_timer_stop(&thread->thread_timer); | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(temp); | ||||
|  | ||||
|     /* insert to schedule ready list */ | ||||
|     rt_schedule_insert_thread(thread); | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_thread_resume_hook, (thread)); | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function is the timeout function for thread, normally which is invoked | ||||
|  * when thread is timeout to wait some resource. | ||||
|  * | ||||
|  * @param parameter the parameter of thread timeout function | ||||
|  */ | ||||
| void rt_thread_timeout(void *parameter) | ||||
| { | ||||
|     struct rt_thread *thread; | ||||
|  | ||||
|     thread = (struct rt_thread *)parameter; | ||||
|  | ||||
|     /* thread check */ | ||||
|     RT_ASSERT(thread != RT_NULL); | ||||
|     RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_SUSPEND); | ||||
|     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread); | ||||
|  | ||||
|     /* set error number */ | ||||
|     thread->error = -RT_ETIMEOUT; | ||||
|  | ||||
|     /* remove from suspend list */ | ||||
|     rt_list_remove(&(thread->tlist)); | ||||
|  | ||||
|     /* insert to schedule ready list */ | ||||
|     rt_schedule_insert_thread(thread); | ||||
|  | ||||
|     /* do schedule */ | ||||
|     rt_schedule(); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will find the specified thread. | ||||
|  * | ||||
|  * @param name the name of thread finding | ||||
|  * | ||||
|  * @return the found thread | ||||
|  * | ||||
|  * @note please don't invoke this function in interrupt status. | ||||
|  */ | ||||
| rt_thread_t rt_thread_find(char *name) | ||||
| { | ||||
|     return (rt_thread_t)rt_object_find(name, RT_Object_Class_Thread); | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
							
								
								
									
										763
									
								
								source/rt_thread/src/timer.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										763
									
								
								source/rt_thread/src/timer.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,763 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2006-03-12     Bernard      first version | ||||
|  * 2006-04-29     Bernard      implement thread timer | ||||
|  * 2006-06-04     Bernard      implement rt_timer_control | ||||
|  * 2006-08-10     Bernard      fix the periodic timer bug | ||||
|  * 2006-09-03     Bernard      implement rt_timer_detach | ||||
|  * 2009-11-11     LiJin        add soft timer | ||||
|  * 2010-05-12     Bernard      fix the timer check bug. | ||||
|  * 2010-11-02     Charlie      re-implement tick overflow issue | ||||
|  * 2012-12-15     Bernard      fix the next timeout issue in soft timer | ||||
|  * 2014-07-12     Bernard      does not lock scheduler when invoking soft-timer | ||||
|  *                             timeout function. | ||||
|  */ | ||||
|  | ||||
| #include <rtthread.h> | ||||
| #include <rthw.h> | ||||
|  | ||||
| /* hard timer list */ | ||||
| static rt_list_t rt_timer_list[RT_TIMER_SKIP_LIST_LEVEL]; | ||||
|  | ||||
| #ifdef RT_USING_TIMER_SOFT | ||||
|  | ||||
| #define RT_SOFT_TIMER_IDLE              1 | ||||
| #define RT_SOFT_TIMER_BUSY              0 | ||||
|  | ||||
| #ifndef RT_TIMER_THREAD_STACK_SIZE | ||||
| #define RT_TIMER_THREAD_STACK_SIZE     512 | ||||
| #endif | ||||
|  | ||||
| #ifndef RT_TIMER_THREAD_PRIO | ||||
| #define RT_TIMER_THREAD_PRIO           0 | ||||
| #endif | ||||
|  | ||||
| /* soft timer status */ | ||||
| static rt_uint8_t soft_timer_status = RT_SOFT_TIMER_IDLE; | ||||
| /* soft timer list */ | ||||
| static rt_list_t rt_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL]; | ||||
| static struct rt_thread timer_thread; | ||||
| ALIGN(RT_ALIGN_SIZE) | ||||
| static rt_uint8_t timer_thread_stack[RT_TIMER_THREAD_STACK_SIZE]; | ||||
| #endif | ||||
|  | ||||
| #ifdef RT_USING_HOOK | ||||
| extern void (*rt_object_take_hook)(struct rt_object *object); | ||||
| extern void (*rt_object_put_hook)(struct rt_object *object); | ||||
| static void (*rt_timer_enter_hook)(struct rt_timer *timer); | ||||
| static void (*rt_timer_exit_hook)(struct rt_timer *timer); | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Hook | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when enter | ||||
|  * timer timeout callback function. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_timer_enter_sethook(void (*hook)(struct rt_timer *timer)) | ||||
| { | ||||
|     rt_timer_enter_hook = hook; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will set a hook function, which will be invoked when exit | ||||
|  * timer timeout callback function. | ||||
|  * | ||||
|  * @param hook the hook function | ||||
|  */ | ||||
| void rt_timer_exit_sethook(void (*hook)(struct rt_timer *timer)) | ||||
| { | ||||
|     rt_timer_exit_hook = hook; | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
| #endif | ||||
|  | ||||
| static void _rt_timer_init(rt_timer_t timer, | ||||
|                            void (*timeout)(void *parameter), | ||||
|                            void      *parameter, | ||||
|                            rt_tick_t  time, | ||||
|                            rt_uint8_t flag) | ||||
| { | ||||
|     int i; | ||||
|  | ||||
|     /* set flag */ | ||||
|     timer->parent.flag  = flag; | ||||
|  | ||||
|     /* set deactivated */ | ||||
|     timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|  | ||||
|     timer->timeout_func = timeout; | ||||
|     timer->parameter    = parameter; | ||||
|  | ||||
|     timer->timeout_tick = 0; | ||||
|     timer->init_tick    = time; | ||||
|  | ||||
|     /* initialize timer list */ | ||||
|     for (i = 0; i < RT_TIMER_SKIP_LIST_LEVEL; i++) | ||||
|     { | ||||
|         rt_list_init(&(timer->row[i])); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* the fist timer always in the last row */ | ||||
| static rt_tick_t rt_timer_list_next_timeout(rt_list_t timer_list[]) | ||||
| { | ||||
|     struct rt_timer *timer; | ||||
|     register rt_base_t level; | ||||
|     rt_tick_t timeout_tick = RT_TICK_MAX; | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     if (!rt_list_isempty(&timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1])) | ||||
|     { | ||||
|         timer = rt_list_entry(timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1].next, | ||||
|                               struct rt_timer, row[RT_TIMER_SKIP_LIST_LEVEL - 1]); | ||||
|         timeout_tick = timer->timeout_tick; | ||||
|     } | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return timeout_tick; | ||||
| } | ||||
|  | ||||
| rt_inline void _rt_timer_remove(rt_timer_t timer) | ||||
| { | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < RT_TIMER_SKIP_LIST_LEVEL; i++) | ||||
|     { | ||||
|         rt_list_remove(&timer->row[i]); | ||||
|     } | ||||
| } | ||||
|  | ||||
| #if RT_DEBUG_TIMER | ||||
| static int rt_timer_count_height(struct rt_timer *timer) | ||||
| { | ||||
|     int i, cnt = 0; | ||||
|  | ||||
|     for (i = 0; i < RT_TIMER_SKIP_LIST_LEVEL; i++) | ||||
|     { | ||||
|         if (!rt_list_isempty(&timer->row[i])) | ||||
|             cnt++; | ||||
|     } | ||||
|     return cnt; | ||||
| } | ||||
|  | ||||
| void rt_timer_dump(rt_list_t timer_heads[]) | ||||
| { | ||||
|     rt_list_t *list; | ||||
|  | ||||
|     for (list = timer_heads[RT_TIMER_SKIP_LIST_LEVEL - 1].next; | ||||
|          list != &timer_heads[RT_TIMER_SKIP_LIST_LEVEL - 1]; | ||||
|          list = list->next) | ||||
|     { | ||||
|         struct rt_timer *timer = rt_list_entry(list, | ||||
|                                                struct rt_timer, | ||||
|                                                row[RT_TIMER_SKIP_LIST_LEVEL - 1]); | ||||
|         rt_kprintf("%d", rt_timer_count_height(timer)); | ||||
|     } | ||||
|     rt_kprintf("\n"); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * @addtogroup Clock | ||||
|  */ | ||||
|  | ||||
| /**@{*/ | ||||
|  | ||||
| /** | ||||
|  * This function will initialize a timer, normally this function is used to | ||||
|  * initialize a static timer object. | ||||
|  * | ||||
|  * @param timer the static timer object | ||||
|  * @param name the name of timer | ||||
|  * @param timeout the timeout function | ||||
|  * @param parameter the parameter of timeout function | ||||
|  * @param time the tick of timer | ||||
|  * @param flag the flag of timer | ||||
|  */ | ||||
| void rt_timer_init(rt_timer_t  timer, | ||||
|                    const char *name, | ||||
|                    void (*timeout)(void *parameter), | ||||
|                    void       *parameter, | ||||
|                    rt_tick_t   time, | ||||
|                    rt_uint8_t  flag) | ||||
| { | ||||
|     /* timer check */ | ||||
|     RT_ASSERT(timer != RT_NULL); | ||||
|  | ||||
|     /* timer object initialization */ | ||||
|     rt_object_init((rt_object_t)timer, RT_Object_Class_Timer, name); | ||||
|  | ||||
|     _rt_timer_init(timer, timeout, parameter, time, flag); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will detach a timer from timer management. | ||||
|  * | ||||
|  * @param timer the static timer object | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK; RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_timer_detach(rt_timer_t timer) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* timer check */ | ||||
|     RT_ASSERT(timer != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&timer->parent) == RT_Object_Class_Timer); | ||||
|     RT_ASSERT(rt_object_is_systemobject(&timer->parent)); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     _rt_timer_remove(timer); | ||||
|     /* stop timer */ | ||||
|     timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     rt_object_detach((rt_object_t)timer); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_HEAP | ||||
| /** | ||||
|  * This function will create a timer | ||||
|  * | ||||
|  * @param name the name of timer | ||||
|  * @param timeout the timeout function | ||||
|  * @param parameter the parameter of timeout function | ||||
|  * @param time the tick of timer | ||||
|  * @param flag the flag of timer | ||||
|  * | ||||
|  * @return the created timer object | ||||
|  */ | ||||
| rt_timer_t rt_timer_create(const char *name, | ||||
|                            void (*timeout)(void *parameter), | ||||
|                            void       *parameter, | ||||
|                            rt_tick_t   time, | ||||
|                            rt_uint8_t  flag) | ||||
| { | ||||
|     struct rt_timer *timer; | ||||
|  | ||||
|     /* allocate a object */ | ||||
|     timer = (struct rt_timer *)rt_object_allocate(RT_Object_Class_Timer, name); | ||||
|     if (timer == RT_NULL) | ||||
|     { | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     _rt_timer_init(timer, timeout, parameter, time, flag); | ||||
|  | ||||
|     return timer; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will delete a timer and release timer memory | ||||
|  * | ||||
|  * @param timer the timer to be deleted | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK; RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_timer_delete(rt_timer_t timer) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* timer check */ | ||||
|     RT_ASSERT(timer != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&timer->parent) == RT_Object_Class_Timer); | ||||
|     RT_ASSERT(rt_object_is_systemobject(&timer->parent) == RT_FALSE); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     _rt_timer_remove(timer); | ||||
|     /* stop timer */ | ||||
|     timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     rt_object_delete((rt_object_t)timer); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * This function will start the timer | ||||
|  * | ||||
|  * @param timer the timer to be started | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_timer_start(rt_timer_t timer) | ||||
| { | ||||
|     unsigned int row_lvl; | ||||
|     rt_list_t *timer_list; | ||||
|     register rt_base_t level; | ||||
|     rt_list_t *row_head[RT_TIMER_SKIP_LIST_LEVEL]; | ||||
|     unsigned int tst_nr; | ||||
|     static unsigned int random_nr; | ||||
|  | ||||
|     /* timer check */ | ||||
|     RT_ASSERT(timer != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&timer->parent) == RT_Object_Class_Timer); | ||||
|  | ||||
|     /* stop timer firstly */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     /* remove timer from list */ | ||||
|     _rt_timer_remove(timer); | ||||
|     /* change status of timer */ | ||||
|     timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_object_take_hook, (&(timer->parent))); | ||||
|  | ||||
|     /* | ||||
|      * get timeout tick, | ||||
|      * the max timeout tick shall not great than RT_TICK_MAX/2 | ||||
|      */ | ||||
|     RT_ASSERT(timer->init_tick < RT_TICK_MAX / 2); | ||||
|     timer->timeout_tick = rt_tick_get() + timer->init_tick; | ||||
|  | ||||
| #ifdef RT_USING_TIMER_SOFT | ||||
|     if (timer->parent.flag & RT_TIMER_FLAG_SOFT_TIMER) | ||||
|     { | ||||
|         /* insert timer to soft timer list */ | ||||
|         timer_list = rt_soft_timer_list; | ||||
|     } | ||||
|     else | ||||
| #endif | ||||
|     { | ||||
|         /* insert timer to system timer list */ | ||||
|         timer_list = rt_timer_list; | ||||
|     } | ||||
|  | ||||
|     row_head[0]  = &timer_list[0]; | ||||
|     for (row_lvl = 0; row_lvl < RT_TIMER_SKIP_LIST_LEVEL; row_lvl++) | ||||
|     { | ||||
|         for (; row_head[row_lvl] != timer_list[row_lvl].prev; | ||||
|              row_head[row_lvl]  = row_head[row_lvl]->next) | ||||
|         { | ||||
|             struct rt_timer *t; | ||||
|             rt_list_t *p = row_head[row_lvl]->next; | ||||
|  | ||||
|             /* fix up the entry pointer */ | ||||
|             t = rt_list_entry(p, struct rt_timer, row[row_lvl]); | ||||
|  | ||||
|             /* If we have two timers that timeout at the same time, it's | ||||
|              * preferred that the timer inserted early get called early. | ||||
|              * So insert the new timer to the end the the some-timeout timer | ||||
|              * list. | ||||
|              */ | ||||
|             if ((t->timeout_tick - timer->timeout_tick) == 0) | ||||
|             { | ||||
|                 continue; | ||||
|             } | ||||
|             else if ((t->timeout_tick - timer->timeout_tick) < RT_TICK_MAX / 2) | ||||
|             { | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         if (row_lvl != RT_TIMER_SKIP_LIST_LEVEL - 1) | ||||
|             row_head[row_lvl + 1] = row_head[row_lvl] + 1; | ||||
|     } | ||||
|  | ||||
|     /* Interestingly, this super simple timer insert counter works very very | ||||
|      * well on distributing the list height uniformly. By means of "very very | ||||
|      * well", I mean it beats the randomness of timer->timeout_tick very easily | ||||
|      * (actually, the timeout_tick is not random and easy to be attacked). */ | ||||
|     random_nr++; | ||||
|     tst_nr = random_nr; | ||||
|  | ||||
|     rt_list_insert_after(row_head[RT_TIMER_SKIP_LIST_LEVEL - 1], | ||||
|                          &(timer->row[RT_TIMER_SKIP_LIST_LEVEL - 1])); | ||||
|     for (row_lvl = 2; row_lvl <= RT_TIMER_SKIP_LIST_LEVEL; row_lvl++) | ||||
|     { | ||||
|         if (!(tst_nr & RT_TIMER_SKIP_LIST_MASK)) | ||||
|             rt_list_insert_after(row_head[RT_TIMER_SKIP_LIST_LEVEL - row_lvl], | ||||
|                                  &(timer->row[RT_TIMER_SKIP_LIST_LEVEL - row_lvl])); | ||||
|         else | ||||
|             break; | ||||
|         /* Shift over the bits we have tested. Works well with 1 bit and 2 | ||||
|          * bits. */ | ||||
|         tst_nr >>= (RT_TIMER_SKIP_LIST_MASK + 1) >> 1; | ||||
|     } | ||||
|  | ||||
|     timer->parent.flag |= RT_TIMER_FLAG_ACTIVATED; | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
| #ifdef RT_USING_TIMER_SOFT | ||||
|     if (timer->parent.flag & RT_TIMER_FLAG_SOFT_TIMER) | ||||
|     { | ||||
|         /* check whether timer thread is ready */ | ||||
|         if ((soft_timer_status == RT_SOFT_TIMER_IDLE) && | ||||
|            ((timer_thread.stat & RT_THREAD_STAT_MASK) == RT_THREAD_SUSPEND)) | ||||
|         { | ||||
|             /* resume timer thread to check soft timer */ | ||||
|             rt_thread_resume(&timer_thread); | ||||
|             rt_schedule(); | ||||
|         } | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will stop the timer | ||||
|  * | ||||
|  * @param timer the timer to be stopped | ||||
|  * | ||||
|  * @return the operation status, RT_EOK on OK, -RT_ERROR on error | ||||
|  */ | ||||
| rt_err_t rt_timer_stop(rt_timer_t timer) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* timer check */ | ||||
|     RT_ASSERT(timer != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&timer->parent) == RT_Object_Class_Timer); | ||||
|  | ||||
|     if (!(timer->parent.flag & RT_TIMER_FLAG_ACTIVATED)) | ||||
|         return -RT_ERROR; | ||||
|  | ||||
|     RT_OBJECT_HOOK_CALL(rt_object_put_hook, (&(timer->parent))); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     _rt_timer_remove(timer); | ||||
|     /* change status */ | ||||
|     timer->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will get or set some options of the timer | ||||
|  * | ||||
|  * @param timer the timer to be get or set | ||||
|  * @param cmd the control command | ||||
|  * @param arg the argument | ||||
|  * | ||||
|  * @return RT_EOK | ||||
|  */ | ||||
| rt_err_t rt_timer_control(rt_timer_t timer, int cmd, void *arg) | ||||
| { | ||||
|     register rt_base_t level; | ||||
|  | ||||
|     /* timer check */ | ||||
|     RT_ASSERT(timer != RT_NULL); | ||||
|     RT_ASSERT(rt_object_get_type(&timer->parent) == RT_Object_Class_Timer); | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     switch (cmd) | ||||
|     { | ||||
|     case RT_TIMER_CTRL_GET_TIME: | ||||
|         *(rt_tick_t *)arg = timer->init_tick; | ||||
|         break; | ||||
|  | ||||
|     case RT_TIMER_CTRL_SET_TIME: | ||||
|         timer->init_tick = *(rt_tick_t *)arg; | ||||
|         break; | ||||
|  | ||||
|     case RT_TIMER_CTRL_SET_ONESHOT: | ||||
|         timer->parent.flag &= ~RT_TIMER_FLAG_PERIODIC; | ||||
|         break; | ||||
|  | ||||
|     case RT_TIMER_CTRL_SET_PERIODIC: | ||||
|         timer->parent.flag |= RT_TIMER_FLAG_PERIODIC; | ||||
|         break; | ||||
|  | ||||
|     case RT_TIMER_CTRL_GET_STATE: | ||||
|         if(timer->parent.flag & RT_TIMER_FLAG_ACTIVATED) | ||||
|         { | ||||
|             /*timer is start and run*/ | ||||
|             *(rt_tick_t *)arg = RT_TIMER_FLAG_ACTIVATED; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             /*timer is stop*/ | ||||
|             *(rt_tick_t *)arg = RT_TIMER_FLAG_DEACTIVATED; | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|     default: | ||||
|         break; | ||||
|     } | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will check timer list, if a timeout event happens, the | ||||
|  * corresponding timeout function will be invoked. | ||||
|  * | ||||
|  * @note this function shall be invoked in operating system timer interrupt. | ||||
|  */ | ||||
| void rt_timer_check(void) | ||||
| { | ||||
|     struct rt_timer *t; | ||||
|     rt_tick_t current_tick; | ||||
|     register rt_base_t level; | ||||
|     rt_list_t list; | ||||
|  | ||||
|     rt_list_init(&list); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_TIMER, ("timer check enter\n")); | ||||
|  | ||||
|     current_tick = rt_tick_get(); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     while (!rt_list_isempty(&rt_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1])) | ||||
|     { | ||||
|         t = rt_list_entry(rt_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1].next, | ||||
|                           struct rt_timer, row[RT_TIMER_SKIP_LIST_LEVEL - 1]); | ||||
|  | ||||
|         /* | ||||
|          * It supposes that the new tick shall less than the half duration of | ||||
|          * tick max. | ||||
|          */ | ||||
|         if ((current_tick - t->timeout_tick) < RT_TICK_MAX / 2) | ||||
|         { | ||||
|             RT_OBJECT_HOOK_CALL(rt_timer_enter_hook, (t)); | ||||
|  | ||||
|             /* remove timer from timer list firstly */ | ||||
|             _rt_timer_remove(t); | ||||
|             if (!(t->parent.flag & RT_TIMER_FLAG_PERIODIC)) | ||||
|             { | ||||
|                 t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|             } | ||||
|             /* add timer to temporary list  */ | ||||
|             rt_list_insert_after(&list, &(t->row[RT_TIMER_SKIP_LIST_LEVEL - 1])); | ||||
|             /* call timeout function */ | ||||
|             t->timeout_func(t->parameter); | ||||
|  | ||||
|             /* re-get tick */ | ||||
|             current_tick = rt_tick_get(); | ||||
|  | ||||
|             RT_OBJECT_HOOK_CALL(rt_timer_exit_hook, (t)); | ||||
|             RT_DEBUG_LOG(RT_DEBUG_TIMER, ("current tick: %d\n", current_tick)); | ||||
|  | ||||
|             /* Check whether the timer object is detached or started again */ | ||||
|             if (rt_list_isempty(&list)) | ||||
|             { | ||||
|                 continue; | ||||
|             } | ||||
|             rt_list_remove(&(t->row[RT_TIMER_SKIP_LIST_LEVEL - 1])); | ||||
|             if ((t->parent.flag & RT_TIMER_FLAG_PERIODIC) && | ||||
|                 (t->parent.flag & RT_TIMER_FLAG_ACTIVATED)) | ||||
|             { | ||||
|                 /* start it */ | ||||
|                 t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|                 rt_timer_start(t); | ||||
|             } | ||||
|         } | ||||
|         else break; | ||||
|     } | ||||
|  | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_TIMER, ("timer check leave\n")); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This function will return the next timeout tick in the system. | ||||
|  * | ||||
|  * @return the next timeout tick in the system | ||||
|  */ | ||||
| rt_tick_t rt_timer_next_timeout_tick(void) | ||||
| { | ||||
|     return rt_timer_list_next_timeout(rt_timer_list); | ||||
| } | ||||
|  | ||||
| #ifdef RT_USING_TIMER_SOFT | ||||
| /** | ||||
|  * This function will check software-timer list, if a timeout event happens, the | ||||
|  * corresponding timeout function will be invoked. | ||||
|  */ | ||||
| void rt_soft_timer_check(void) | ||||
| { | ||||
|     rt_tick_t current_tick; | ||||
|     struct rt_timer *t; | ||||
|     register rt_base_t level; | ||||
|     rt_list_t list; | ||||
|  | ||||
|     rt_list_init(&list); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_TIMER, ("software timer check enter\n")); | ||||
|  | ||||
|     /* disable interrupt */ | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|     while (!rt_list_isempty(&rt_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1])) | ||||
|     { | ||||
|         t = rt_list_entry(rt_soft_timer_list[RT_TIMER_SKIP_LIST_LEVEL - 1].next, | ||||
|                             struct rt_timer, row[RT_TIMER_SKIP_LIST_LEVEL - 1]); | ||||
|  | ||||
|         current_tick = rt_tick_get(); | ||||
|  | ||||
|         /* | ||||
|          * It supposes that the new tick shall less than the half duration of | ||||
|          * tick max. | ||||
|          */ | ||||
|         if ((current_tick - t->timeout_tick) < RT_TICK_MAX / 2) | ||||
|         { | ||||
|             RT_OBJECT_HOOK_CALL(rt_timer_enter_hook, (t)); | ||||
|  | ||||
|             /* remove timer from timer list firstly */ | ||||
|             _rt_timer_remove(t); | ||||
|             if (!(t->parent.flag & RT_TIMER_FLAG_PERIODIC)) | ||||
|             { | ||||
|                 t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|             } | ||||
|             /* add timer to temporary list  */ | ||||
|             rt_list_insert_after(&list, &(t->row[RT_TIMER_SKIP_LIST_LEVEL - 1])); | ||||
|  | ||||
|             soft_timer_status = RT_SOFT_TIMER_BUSY; | ||||
|             /* enable interrupt */ | ||||
|             rt_hw_interrupt_enable(level); | ||||
|  | ||||
|             /* call timeout function */ | ||||
|             t->timeout_func(t->parameter); | ||||
|  | ||||
|             RT_OBJECT_HOOK_CALL(rt_timer_exit_hook, (t)); | ||||
|             RT_DEBUG_LOG(RT_DEBUG_TIMER, ("current tick: %d\n", current_tick)); | ||||
|  | ||||
|             /* disable interrupt */ | ||||
|             level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|             soft_timer_status = RT_SOFT_TIMER_IDLE; | ||||
|             /* Check whether the timer object is detached or started again */ | ||||
|             if (rt_list_isempty(&list)) | ||||
|             { | ||||
|                 continue; | ||||
|             } | ||||
|             rt_list_remove(&(t->row[RT_TIMER_SKIP_LIST_LEVEL - 1])); | ||||
|             if ((t->parent.flag & RT_TIMER_FLAG_PERIODIC) && | ||||
|                 (t->parent.flag & RT_TIMER_FLAG_ACTIVATED)) | ||||
|             { | ||||
|                 /* start it */ | ||||
|                 t->parent.flag &= ~RT_TIMER_FLAG_ACTIVATED; | ||||
|                 rt_timer_start(t); | ||||
|             } | ||||
|         } | ||||
|         else break; /* not check anymore */ | ||||
|     } | ||||
|     /* enable interrupt */ | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     RT_DEBUG_LOG(RT_DEBUG_TIMER, ("software timer check leave\n")); | ||||
| } | ||||
|  | ||||
| /* system timer thread entry */ | ||||
| static void rt_thread_timer_entry(void *parameter) | ||||
| { | ||||
|     rt_tick_t next_timeout; | ||||
|  | ||||
|     while (1) | ||||
|     { | ||||
|         /* get the next timeout tick */ | ||||
|         next_timeout = rt_timer_list_next_timeout(rt_soft_timer_list); | ||||
|         if (next_timeout == RT_TICK_MAX) | ||||
|         { | ||||
|             /* no software timer exist, suspend self. */ | ||||
|             rt_thread_suspend(rt_thread_self()); | ||||
|             rt_schedule(); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             rt_tick_t current_tick; | ||||
|  | ||||
|             /* get current tick */ | ||||
|             current_tick = rt_tick_get(); | ||||
|  | ||||
|             if ((next_timeout - current_tick) < RT_TICK_MAX / 2) | ||||
|             { | ||||
|                 /* get the delta timeout tick */ | ||||
|                 next_timeout = next_timeout - current_tick; | ||||
|                 rt_thread_delay(next_timeout); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* check software timer */ | ||||
|         rt_soft_timer_check(); | ||||
|     } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * | ||||
|  * This function will initialize system timer | ||||
|  */ | ||||
| void rt_system_timer_init(void) | ||||
| { | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < sizeof(rt_timer_list) / sizeof(rt_timer_list[0]); i++) | ||||
|     { | ||||
|         rt_list_init(rt_timer_list + i); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @ingroup SystemInit | ||||
|  * | ||||
|  * This function will initialize system timer thread | ||||
|  */ | ||||
| void rt_system_timer_thread_init(void) | ||||
| { | ||||
| #ifdef RT_USING_TIMER_SOFT | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; | ||||
|          i < sizeof(rt_soft_timer_list) / sizeof(rt_soft_timer_list[0]); | ||||
|          i++) | ||||
|     { | ||||
|         rt_list_init(rt_soft_timer_list + i); | ||||
|     } | ||||
|  | ||||
|     /* start software timer thread */ | ||||
|     rt_thread_init(&timer_thread, | ||||
|                    "timer", | ||||
|                    rt_thread_timer_entry, | ||||
|                    RT_NULL, | ||||
|                    &timer_thread_stack[0], | ||||
|                    sizeof(timer_thread_stack), | ||||
|                    RT_TIMER_THREAD_PRIO, | ||||
|                    10); | ||||
|  | ||||
|     /* startup */ | ||||
|     rt_thread_startup(&timer_thread); | ||||
| #endif | ||||
| } | ||||
|  | ||||
| /**@}*/ | ||||
		Reference in New Issue
	
	Block a user
	 andy
					andy