Raw-OS源代码分析之系统tick中断
分析的内核版本号截止到2014-04-15,基于1.05正式版,blogs会及时跟进最新版本号的内核开发进度,若源代码凝视出现”???”字样,则是未深究理解部分。
Raw-OS官方站点:http://www.raw-os.org/
Raw-OS托管地址:https://github.com/jorya/raw-os/
在讨论同优先级任务切换时粗略粘贴系统时钟中断ISR的代码,上节已经分析过同优先级任务轮转时tick的部分代码:
void tick_isr(){ /* 中断ISR进入。进入中断时必须先调用 */ raw_enter_interrupt(); /* 宏配置中先不要实现task 0。还没研究过源代码,不懂怎么去解释??? */ #ifdef (CONFIG_RAW_TASK_0 == 1) /* task 0 转发??? */ task_0_tick_post(); #else /* 调用系统时间tick处理函数 */ raw_time_tick(); #endif /* 中断ISR退出,退出中断时必须配套raw_enter_interrupt()使用,这里可能发生中断任务切换 */ raw_finish_int(); }
那么运行流程一般流程是这种:
在这里具体讨论这个raw_enter_interrupt()和raw_finish_int()函数,前者是入中断ISR时首先调用。后者是退出中断ISR时最后调用,那么这两个函数做了什么工作,看看源代码凝视:
/* 进入中断ISR时首先调用 */ RAW_U16 raw_enter_interrupt(void) { RAW_SR_ALLOC(); /* 检查中断嵌套层数,默认100,超过中断嵌套层数阀值马上返回 */ if (raw_int_nesting >= INT_NESTED_LEVEL) { return RAW_EXCEED_INT_NESTED_LEVEL; } /* 这里首先禁止CPU中断,目的是保证raw_int_nesting操作是原子操作 */ RAW_CPU_DISABLE(); /* 中断嵌套层数++,在中断中也能够会发生中断。在这里统计中断嵌套层数 */ raw_int_nesting++; RAW_CPU_ENABLE(); return RAW_SUCCESS; }
/* 退出中断ISR时和raw_enter_interrupt()配套调用 */ RAW_VOID raw_finish_int(void) { /* 移植相关。这个宏是定义存放一个CPU状态寄存器的暂时变量,通常是unsigned int类型 */ RAW_SR_ALLOC(); /* 移植相关,保存当前任务的CPU状态字到上面定义的暂时变量中 */ USER_CPU_INT_DISABLE(); /* * raw_int_nesting是在raw_enter_interrupt()调用时++的 * 为0时表明中断没有发生,没有中断发生调用此函数马上返回 */ if (raw_int_nesting == 0) { USER_CPU_INT_ENABLE(); return; } /* 退出中断时,中断嵌套层数-- */ raw_int_nesting--; /* 假设此时中断嵌套层数>0,表明还有没完毕的中断。返回,优先完毕中断处理 */ if (raw_int_nesting) { USER_CPU_INT_ENABLE(); return; } /* 系统上锁时,即系统禁止调度时。返回 */ if (raw_sched_lock) { USER_CPU_INT_ENABLE(); /* if interrupt happen here, it may cause raw_int_nesting equal 0 ???*/ return; } /* 获取就绪队列中最高优先级任务 */ get_ready_task(&raw_ready_queue); /* 假设获取的最高优先级任务是当前执行的任务,即不须要进行任务切换。返回当前任务继续执行 */ if (high_ready_obj == raw_task_active) { USER_CPU_INT_ENABLE(); return; } /* trace系统调试,作用??? */ TRACE_INT_TASK_SWITCH(raw_task_active, high_ready_obj); /* 移植相关,中断中的任务切换函数,在这里实现tick ISR中进行系统调度任务 */ raw_int_switch(); /* 移植相关。任务切换后恢复切换后任务使用的CPU状态字 */ USER_CPU_INT_ENABLE(); }
再看看那有关移植相关的两个宏
/* * 当系统宏配置文件不打开RAW_CPU_INT_DIS_MEASURE_CHECK这个选项是,下面两段代码是一致的,即 * RAW_CPU_DISABLE() == USER_CPU_INT_DISABLE() * RAW_CPU_ENABLE() == RAW_CPU_DISABLE() * * 功能:RAW_SR_ALLOC()定义一个SR类型的暂时变量tmp。USER_CPU_INT_DISABLE()。保存当前CPU状态机到tmp * USER_CPU_INT_ENABLE()读取CPU状态机,恢复到CPU状态机寄存器 */ void xx_func(void){ RAW_SR_ALLOC(); USER_CPU_INT_DISABLE(); ...... USER_CPU_INT_ENABLE(); } void xx_func(void){ RAW_SR_ALLOC(); RAW_CPU_DISABLE(); ...... RAW_CPU_ENABLE(); }
很好理解,这一节就这样结束吧~