c - 如果我们在 SMP 上的中断处理程序中休眠会发生什么

标签 c linux linux-kernel linux-device-driver interrupt-handling

如果我们在 SMP 机器上的中断处理程序中休眠会发生什么,

我写了一个示例 keyboard driver并在中断处理程序上添加 sleep

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/interrupt.h>
#include <linux/delay.h>
#include <linux/sched/signal.h>

MODULE_LICENSE("GPL");
static int irq = 1,  dev = 0xaa, counter = 0;

static irqreturn_t keyboard_handler(int irq, void *dev)
{
        pr_info("Keyboard Counter:%d\n", counter++);
        msleep(1000);
        return IRQ_NONE;
}
/* registering irq */
static int test_interrupt_init(void)
{
        pr_info("%s: In init\n", __func__);
        return request_irq(irq, keyboard_handler, IRQF_SHARED,"my_keyboard_handler", &dev);
}

static void test_interrupt_exit(void)
{
        pr_info("%s: In exit\n", __func__);
        synchronize_irq(irq); /* synchronize interrupt */
        free_irq(irq, &dev);
}

module_init(test_interrupt_init);
module_exit(test_interrupt_exit);

系统运行了几分钟,然后崩溃了。为什么系统不能在禁用一个处理器的情况下工作,因为定时器中断将在另一个 CPU 上触发并可以调度进程。

使用 kgdb 设置捕获的反向跟踪:

Program received signal SIGSEGV, Segmentation fault.
[Switching to Thread 1254]
0x0000000000000000 in irq_stack_union ()

(gdb) bt
#0  0x0000000000000000 in irq_stack_union ()
#1  0xffffffff810ad8e4 in ttwu_activate (en_flags=<optimized out>, p=<optimized out>, rq=<optimized out>)
    at kernel/sched/core.c:1638
#2  ttwu_do_activate (rq=0xffff888237422c40, p=0xffffffff82213780 <init_task>, wake_flags=9, 
    rf=0xffffc900022b3f10) at kernel/sched/core.c:1697
#3  0xffffffff810aec00 in sched_ttwu_pending () at kernel/sched/core.c:1740
#4  0xffffffff810aedcd in scheduler_ipi () at kernel/sched/core.c:1771
#5  0xffffffff81a01aef in reschedule_interrupt () at arch/x86/entry/entry_64.S:888
#6  0xffffc900022b3f58 in ?? ()
#7  0xffffffff81a01aea in reschedule_interrupt () at arch/x86/entry/entry_64.S:888
#8  0x0000000000000002 in irq_stack_union ()
#9  0x00007fcec3421b40 in ?? ()
#10 0x0000000000000006 in irq_stack_union ()
#11 0x00007fceb00008c0 in ?? ()
#12 0x0000000000000002 in irq_stack_union ()
#13 0x00000000020bd380 in ?? ()
#14 0x0012c8d2cc413914 in ?? ()
Cannot access memory at address 0x5000

最佳答案

键盘中断随时可能发生——包括在内核调用期间。通常,这没关系:中断发生,驱动程序执行它的操作,中断处理程序返回,然后内核继续。

但是如果您在中断处理程序中sleep(),则内核处于中间状态。其他处理器无法执行内核调用,因为它已经很忙了。每个人都将被迫暂停等待内核——它不会回来。难怪它会 panic !

关于c - 如果我们在 SMP 上的中断处理程序中休眠会发生什么,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54058292/

相关文章:

php - apache/php 创建文件,但同一页面不允许编辑文件

c - linux上PPC64下一个进程可以有多少内存

linux - linux下只有中断向量能识别中断源吗?

将字符串十进制转换为十六进制十进制

linux - 系统调用中的系统调用

c - C 可变参数是否使用名为 'end' 的关键字?

linux - 使用 awk 获取恶意软件病毒事件所需的帮助

linux - 构建 Linux Kernel 4.18 时出现编译时错误

c - libcurl gnutls_handshake() 失败 : A TLS warning alert has been received

c - 获取地址信息(): how to distinguish between resolution failure and non-resolvable hostname?