Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
signal-信号
信号内部机制实现的疑问
发布于 2024-07-09 09:45:20 浏览:374
订阅该版
我的问题是:下面的第3种情况,为什么不采用和第一种情况一样的处理信号的方式去处理,而是要采用构建栈,在_signal_entry函数中去处理 在信号(signal)的_signal_deliver函数中,信号的处理分位3种情况,如下: 1. 如果目标线程是挂起状态时,恢复成就绪状态,设置信号标志,最终信号的处理函数rt_thread_handle_sig(RT_TRUE);会在rt_schedule();中执行 ```c #ifdef RT_USING_SIGNALS /* check stat of thread for signal */ level = rt_hw_interrupt_disable(); if (RT_SCHED_CTX(rt_current_thread).stat & RT_THREAD_STAT_SIGNAL_PENDING) { extern void rt_thread_handle_sig(rt_bool_t clean_state); RT_SCHED_CTX(rt_current_thread).stat &= ~RT_THREAD_STAT_SIGNAL_PENDING; rt_hw_interrupt_enable(level); /* check signal status */ rt_thread_handle_sig(RT_TRUE); } else { rt_hw_interrupt_enable(level); } ``` 2. 如果目标线程就是当前线程,会在_signal_deliver(rt_thread_ttid)函数直接执行信号的处理函数rt_thread_handle_sig(RT_TRUE); ```c if (tid == rt_thread_self()) { /* add signal state */ RT_SCHED_CTX(tid).stat |= RT_THREAD_STAT_SIGNAL; rt_spin_unlock_irqrestore(&_thread_signal_lock, level); /* do signal action in self thread context */ if (rt_interrupt_get_nest() == 0) { rt_thread_handle_sig(RT_TRUE); } } ``` 3.如果目标线程处于就绪态,则在目标线程的栈上构建新的栈,且通过rt_hw_stack_init把线程entry改为_signal_entry,然后在_signal_entry函数中去执行信号的处理函数rt_thread_handle_sig(RT_TRUE);,执行完后再去切换到之前的栈 ```c /* point to the signal handle entry */ RT_SCHED_CTX(tid).stat &= ~RT_THREAD_STAT_SIGNAL_PENDING; tid->sig_ret = tid->sp; tid->sp = rt_hw_stack_init((void *)_signal_entry, RT_NULL, (void *)((char *)tid->sig_ret - 32), RT_NULL); ``` ```c static void _signal_entry(void *parameter) { RT_UNUSED(parameter); rt_thread_t tid = rt_thread_self(); /* handle signal */ rt_thread_handle_sig(RT_FALSE); #ifdef RT_USING_SMP #else /* return to thread */ tid->sp = tid->sig_ret; tid->sig_ret = RT_NULL; #endif /* RT_USING_SMP */ LOG_D("switch back to: 0x%08x\n", tid->sp); RT_SCHED_CTX(tid).stat &= ~RT_THREAD_STAT_SIGNAL; #ifdef RT_USING_SMP rt_hw_context_switch_to((rt_base_t)¶meter, tid); #else rt_hw_context_switch_to((rt_ubase_t)&(tid->sp)); #endif /* RT_USING_SMP */ } ``` static void _signal_deliver(rt_thread_ttid)函数 ```c static void _signal_deliver(rt_thread_t tid) { rt_base_t level; level = rt_spin_lock_irqsave(&_thread_signal_lock); /* thread is not interested in pended signals */ if (!(tid->sig_pending & tid->sig_mask)) { rt_spin_unlock_irqrestore(&_thread_signal_lock, level); return; } if ((RT_SCHED_CTX(tid).stat & RT_THREAD_SUSPEND_MASK) == RT_THREAD_SUSPEND_MASK) { /* resume thread to handle signal */ #ifdef RT_USING_SMART rt_thread_wakeup(tid); #else rt_thread_resume(tid); #endif /* add signal state */ RT_SCHED_CTX(tid).stat |= (RT_THREAD_STAT_SIGNAL | RT_THREAD_STAT_SIGNAL_PENDING); rt_spin_unlock_irqrestore(&_thread_signal_lock, level); /* re-schedule */ rt_schedule(); } else { if (tid == rt_thread_self()) { /* add signal state */ RT_SCHED_CTX(tid).stat |= RT_THREAD_STAT_SIGNAL; rt_spin_unlock_irqrestore(&_thread_signal_lock, level); /* do signal action in self thread context */ if (rt_interrupt_get_nest() == 0) { rt_thread_handle_sig(RT_TRUE); } } else if (!((RT_SCHED_CTX(tid).stat & RT_THREAD_STAT_SIGNAL_MASK) & RT_THREAD_STAT_SIGNAL)) { /* add signal state */ RT_SCHED_CTX(tid).stat |= (RT_THREAD_STAT_SIGNAL | RT_THREAD_STAT_SIGNAL_PENDING); #ifdef RT_USING_SMP { int cpu_id; cpu_id = RT_SCHED_CTX(tid).oncpu; if ((cpu_id != RT_CPU_DETACHED) && (cpu_id != rt_cpu_get_id())) { rt_uint32_t cpu_mask; cpu_mask = RT_CPU_MASK ^ (1 << cpu_id); rt_hw_ipi_send(RT_SCHEDULE_IPI, cpu_mask); } } #else /* point to the signal handle entry */ RT_SCHED_CTX(tid).stat &= ~RT_THREAD_STAT_SIGNAL_PENDING; tid->sig_ret = tid->sp; tid->sp = rt_hw_stack_init((void *)_signal_entry, RT_NULL, (void *)((char *)tid->sig_ret - 32), RT_NULL); #endif /* RT_USING_SMP */ rt_spin_unlock_irqrestore(&_thread_signal_lock, level); LOG_D("signal stack pointer @ 0x%08x", tid->sp); /* re-schedule */ rt_schedule(); } else { rt_spin_unlock_irqrestore(&_thread_signal_lock, level); } } } ```
查看更多
1
个回答
默认排序
按发布时间排序
AAA水果批发小雷
2024-07-09
这家伙很懒,什么也没写!
大佬们,信号的内部机制函数_signal_deliver(rt_thread_ttid),对信号的处理分三种情况,第三种情况该怎么理解啊,
撰写答案
登录
注册新账号
关注者
1
被浏览
374
关于作者
AAA水果批发小雷
这家伙很懒,什么也没写!
提问
1
回答
1
被采纳
0
关注TA
发私信
相关问题
1
signal内核是不是有问题,还是我方法不对
2
关于信号(signal)处理机制的疑问
3
请问rt-thread信号(signal)如何使用?
4
rtthread signal的使用
5
RT-Thread有没有命令可以终止正执行的命令
6
_signal_entry() 函数中dbg_enter在哪里定义呢?
7
怎么找到函数对应的头文件?
8
rt_signal_wait 代码逻辑问题
9
rt-thread内核的libc_signal.h头文件没有gcc编译器的宏定义
10
信号SIGUSR1(10)跟 SIGUSR1(12)的意思
推荐文章
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
【RT-Thread】【ci】【scons】将ci.attachconfig.yml和scons结合使用
2
Rt-thread中OTA下载后,bootloader不搬程序
3
ulog 日志 LOG_HEX 输出时间改为本地日期时间
4
在RT-Thread Studio中构建前执行python命令
5
研究一了一段时间RTT,直接标准版上手太难,想用nano,但又舍不得组件
热门标签
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
I2C_IIC
ESP8266
UART
WIZnet_W5500
ota在线升级
cubemx
PWM
flash
freemodbus
BSP
packages_软件包
潘多拉开发板_Pandora
定时器
ADC
flashDB
GD32
socket
编译报错
中断
Debug
rt_mq_消息队列_msg_queue
SFUD
msh
keil_MDK
ulog
C++_cpp
MicroPython
本月问答贡献
xusiwei1236
8
个答案
2
次被采纳
踩姑娘的小蘑菇
1
个答案
2
次被采纳
用户名由3_15位
9
个答案
1
次被采纳
bernard
4
个答案
1
次被采纳
RTT_逍遥
3
个答案
1
次被采纳
本月文章贡献
聚散无由
2
篇文章
15
次点赞
catcatbing
2
篇文章
5
次点赞
Wade
2
篇文章
4
次点赞
Ghost_Girls
1
篇文章
7
次点赞
xiaorui
1
篇文章
2
次点赞
回到
顶部
发布
问题
分享
好友
手机
浏览
扫码手机浏览
投诉
建议
回到
底部