Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
RT-Thread一般讨论
rt_schedule中need_insert_from_thread的问题
发布于 2022-05-14 19:07:55 浏览:778
订阅该版
[tocm] RT-Thread支持时间片调度算法,相同优先级的thread按其设定的ticks 轮流执行。 先上`rt_schedule`代码 ![image-20220514174332285.png](https://oss-club.rt-thread.org/uploads/20220514/801be0f38967021a13f4b77412c4e0fd.png.webp) need_insert_from_thread 标志主要是用来标记是否从当前任务切换到其他任务,切出意味着需要把当前任务插入到对应优先级的readylist。 ## 情况1 这个很好理解:没有和当前任务task1优先级相同的任务,其优先级readylist为空。 1. 假如有一个低优先级task2任务, systic中断里 rt_timer_check()发现超时;或者资源就绪把它唤醒,都会发起一次调度,把它插入对应readylist,这时_scheduler_get_highest_priority_thread得到的是task2,无需切换。 2. 或者当前任务时间片用完,主动发起调度,发现自己还是老大,那就继续运行吧 3. task主动调用rt_thread_yield,礼让CPU资源,结果和上面一样 ## 情况2 假设还有一个task2和当前任务task1优先级相同, 同样可能因为超时;或者资源就绪把它唤醒,发起一次调度,把它插入对应readylist。 这时task1的时间片还未用完,依旧无需切换,让它等着吧。 ## 情况3 可以直接列出剩余的三种情况, 1) ``` if (rt_current_thread->current_priority == highest_ready_priority && (rt_current_thread->stat & RT_THREAD_STAT_YIELD_MASK) != 0) ``` 和情况2相同,当前任务的时间片用完了或者主动礼让,把自己插入对应readylist前 2) ``` if (rt_current_thread->current_priority > highest_ready_priority && (rt_current_thread->stat & RT_THREAD_STAT_YIELD_MASK) != 0) ``` 这个情况实际是不存在的,因为在SysTick_Handler 调用 的rt_tick_increase,一旦发现时间片用完或者主动礼让就会发起一次调度,清除了RT_THREAD_STAT_YIELD 再次因超时或者资源唤醒进来,也不满足这个条件 ``` /** * @brief 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; rt_base_t level; RT_OBJECT_HOOK_CALL(rt_tick_hook, ()); level = rt_hw_interrupt_disable(); /* increase the global tick */ #ifdef RT_USING_SMP rt_cpu_self()->tick ++; #else ++ rt_tick; #endif /* RT_USING_SMP */ /* 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; thread->stat |= RT_THREAD_STAT_YIELD; rt_hw_interrupt_enable(level); rt_schedule(); } else { rt_hw_interrupt_enable(level); } /* check timer */ rt_timer_check(); ``` 3) ``` if (rt_current_thread->current_priority > highest_ready_priority && (rt_current_thread->stat & RT_THREAD_STAT_YIELD_MASK) == 0) ``` 新就绪的任务优先级高于当前任务,虽然当前任务时间片有剩余,也必须切出,同时保留剩余时间片 ## 插入到readylist ![image-20220514185246202.png](https://oss-club.rt-thread.org/uploads/20220514/6c836ed28fd54e3ff036d66317e161ec.png.webp) ``` 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(); /* it's current thread, it should be RUNNING thread */ if (thread == rt_current_thread) { thread->stat = RT_THREAD_RUNNING | (thread->stat & ~RT_THREAD_STAT_MASK); goto __exit; } /* READY thread, insert to ready queue */ 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)); RT_DEBUG_LOG(RT_DEBUG_SCHEDULER, ("insert thread[%.*s], the priority: %d\n", RT_NAME_MAX, thread->name, thread->current_priority)); /* set priority mask */ #if RT_THREAD_PRIORITY_MAX > 32 rt_thread_ready_table[thread->number] |= thread->high_mask; #endif /* RT_THREAD_PRIORITY_MAX > 32 */ rt_thread_ready_priority_group |= thread->number_mask; __exit: /* enable interrupt */ rt_hw_interrupt_enable(temp); } ``` ## 问题 问题出在情况3的第三种情景下: **新就绪的任务优先级高于当前任务,当前任务时间片有剩余,** 插入动作没有问题,但是使用rt_list_insert_before把未执行完时间片的任务插入到其优先级readylist的最后面,是不是对其不太公平,毕竟是高优先级的任务打断了它。 是不是应该按下面分类处理比较好, 使用rt_list_insert_after把它插在list->next位置,下次轮到该优先级时,首先执行,直到其时间片用完! ``` if (rt_current_thread->current_priority < highest_ready_priority) { to_thread = rt_current_thread;// no task of same priority } else if (rt_current_thread->current_priority == highest_ready_priority && (rt_current_thread->stat & RT_THREAD_STAT_YIELD_MASK) == 0) { to_thread = rt_current_thread; // another tasks of same priority ,not preemption } else if (rt_current_thread->current_priority == highest_ready_priority && (rt_current_thread->stat & RT_THREAD_STAT_YIELD_MASK) != 0) { need_insert_from_thread = 1; // insert before } else if (rt_current_thread->current_priority > highest_ready_priority && (rt_current_thread->stat & RT_THREAD_STAT_YIELD_MASK) == 0) { need_insert_from_thread = 2;// insert after to make sure the remained ticks can be used continually } // else if (rt_current_thread->current_priority > highest_ready_priority && (rt_current_thread->stat & RT_THREAD_STAT_YIELD_MASK) != 0) // { // need_insert_from_thread = 1; // } ```
查看更多
3
个回答
默认排序
按发布时间排序
lqwuliang
2022-06-27
天下没有解决不了的bug,只有发现不了的bug.
分析得有道理,按这分析,我觉得正常逻辑应该是,高优先级打断了它,它住后等,等高优先级执行完后,就应该再执行它,
flyboy
2022-05-16
Do my self();
很棒,分析的很深入👍
rtt_newer
2023-09-05
这家伙很懒,什么也没写!
已清楚!
撰写答案
登录
注册新账号
关注者
0
被浏览
778
关于作者
blta
这家伙很懒,什么也没写!
提问
3
回答
9
被采纳
2
关注TA
发私信
相关问题
1
有关动态模块加载的一篇论文
2
最近的调程序总结
3
晕掉了,这么久都不见layer2的踪影啊
4
继续K9ii的历程
5
[GUI相关] FreeType 2
6
[GUI相关]嵌入式系统中文输入法的设计
7
20081101 RT-Thread开发者聚会总结
8
嵌入式系统基础
9
linux2.4.19在at91rm9200 上的寄存器设置
10
[转]基于嵌入式Linux的通用触摸屏校准程序
推荐文章
1
RT-Thread应用项目汇总
2
玩转RT-Thread系列教程
3
国产MCU移植系列教程汇总,欢迎查看!
4
机器人操作系统 (ROS2) 和 RT-Thread 通信
5
五分钟玩转RT-Thread新社区
6
【技术三千问】之《玩转ART-Pi》,看这篇就够了!干货汇总
7
关于STM32H7开发板上使用SDIO接口驱动SD卡挂载文件系统的问题总结
8
STM32的“GPU”——DMA2D实例详解
9
RT-Thread隐藏的宝藏之completion
10
【ART-PI】RT-Thread 开启RTC 与 Alarm组件
最新文章
1
freemodbus主机在freertos的适配,参考rtthread例程
2
开源共生 商业共赢 | RT-Thread 2024开发者大会议程正式发布!
3
【24嵌入式设计大赛】基于RT-Thread星火一号的智慧家居系统
4
RT-Thread EtherKit开源以太网硬件正式发布
5
还在担心bsp不好维护吗?快使用yml管理主线bsp
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
USB
DMA
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
FAL
rt-smart
ESP8266
I2C_IIC
WIZnet_W5500
UART
ota在线升级
PWM
cubemx
freemodbus
flash
packages_软件包
BSP
潘多拉开发板_Pandora
定时器
ADC
GD32
flashDB
socket
中断
Debug
编译报错
msh
SFUD
rt_mq_消息队列_msg_queue
keil_MDK
ulog
MicroPython
C++_cpp
本月问答贡献
a1012112796
20
个答案
3
次被采纳
张世争
12
个答案
3
次被采纳
踩姑娘的小蘑菇
7
个答案
3
次被采纳
用户名由3_15位
13
个答案
2
次被采纳
rv666
9
个答案
2
次被采纳
本月文章贡献
程序员阿伟
9
篇文章
2
次点赞
hhart
3
篇文章
4
次点赞
RTT_逍遥
1
篇文章
8
次点赞
大龄码农
1
篇文章
5
次点赞
ThinkCode
1
篇文章
1
次点赞
回到
顶部
发布
问题
分享
好友
手机
浏览
扫码手机浏览
投诉
建议
回到
底部