当panic或者die被执行时,或者发生未定义指令时,如何被回调到

    Panic是Linux  kernel提供的一种复位机制,内核开发者在开发过程中,可以在异常流程中主动调用。

    另外,还有一部分模块或者子系统,希望系统在panic时,能主动通知一下本模块,以便做一些复位前的准备动作,相应的kernel也提供了该机制,就是让大家注册回调。每次panic在执行过程中,会遍历注册的回调函数并回调之。

实现机制:

1、在panic.c代码中,有一个链表的定义,

ATOMIC_NOTIFIER_HEAD(panic_notifier_list);

EXPORT_SYMBOL(panic_notifier_list);

该链表存储各个模块注册的panic回调函数,以便panic在发生时回调对应回调函数。

2、有了链表头,那么注册就简单了。

1)首先定义自己的回调函数

int my_panic_ notify(struct notifier_block *nb,

                            unsigned long event, void *buf)

{

       ………

       return 0;

}

static struct notifier_block my_panic_block = {

       .notifier_call = my_panic_ notify,

       .priority = INT_MAX,

};

然后,在合适的地方,注册到panic链表中(通常是在本模块初始化)

………….

atomic_notifier_chain_register(&panic_notifier_list, & my_panic_block);

…………

3、注册完成后,当系统中有调用panic函数时,panic在执行过程中就会回调注册的函数。

…………..

       /*

        * Run any panic handlers, including those that might need to

        * add information to the kmsg dump output.

        */

       atomic_notifier_call_chain(&panic_notifier_list, 0, buf);

…………….

附:同样,在die函数被调用时,也有类似的机制,不做详细介绍,大家可以看一下register_die_notifier/ unregister_die_notifier的相关实现。

未定义指令异常,参考register_undef_hook函数相关实现

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一段内核代码,定期执行打印间的任务,但是在系统休眠或panic停止打印: ``` #include <linux/kernel.h> #include <linux/module.h> #include <linux/init.h> #include <linux/time.h> #include <linux/timer.h> #include <linux/suspend.h> static struct timer_list my_timer; static void print_time(unsigned long data) { printk(KERN_INFO "Current time: %ld\n", (long)jiffies); mod_timer(&my_timer, jiffies + msecs_to_jiffies(1000)); // 1 second interval } static int __init my_init(void) { printk(KERN_INFO "Starting time printing task...\n"); setup_timer(&my_timer, print_time, 0); mod_timer(&my_timer, jiffies + msecs_to_jiffies(1000)); // start after 1 second return 0; } static void __exit my_exit(void) { del_timer_sync(&my_timer); printk(KERN_INFO "Stopping time printing task...\n"); } static int my_suspend(void) { del_timer_sync(&my_timer); return 0; } static int my_resume(void) { mod_timer(&my_timer, jiffies + msecs_to_jiffies(1000)); // resume after 1 second return 0; } static struct syscore_ops my_syscore_ops = { .suspend = my_suspend, .resume = my_resume, }; module_init(my_init); module_exit(my_exit); MODULE_LICENSE("GPL"); static int __init my_init(void) { printk(KERN_INFO "Starting time printing task...\n"); setup_timer(&my_timer, print_time, 0); mod_timer(&my_timer, jiffies + msecs_to_jiffies(1000)); // start after 1 second register_syscore_ops(&my_syscore_ops); return 0; } static void __exit my_exit(void) { unregister_syscore_ops(&my_syscore_ops); del_timer_sync(&my_timer); printk(KERN_INFO "Stopping time printing task...\n"); } static int my_suspend(void) { del_timer_sync(&my_timer); return 0; } static int my_resume(void) { mod_timer(&my_timer, jiffies + msecs_to_jiffies(1000)); // resume after 1 second return 0; } static struct syscore_ops my_syscore_ops = { .suspend = my_suspend, .resume = my_resume, }; module_init(my_init); module_exit(my_exit); MODULE_LICENSE("GPL"); ``` 这段代码在内核初始化会注册一个定器,每隔1秒打印一次当前间。如果系统进入睡眠状态,定器会被删除,直到系统恢复为止。如果系统panic,则定器也会停止。为了处理系统休眠和恢复,这段代码还注册了一个syscore操作,用于在系统进入和退出休眠状态启动和停止定器。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值