Linux内核中的通知链机制解析

Linux内核中的通知链机制解析

概述

在Linux内核这个庞大的系统中,各个子系统之间经常需要进行信息交互。当一个子系统发生某些事件时,其他子系统可能需要及时获知这些变化。为此,Linux内核提供了一种称为"通知链"(Notification Chains)的机制,它允许内核中的不同组件订阅并接收来自其他组件的异步事件通知。

通知链的基本概念

通知链本质上是一个回调函数的链表,当特定事件发生时,链表中注册的所有回调函数都会被依次调用。这种机制类似于设计模式中的"观察者模式",其中:

  • 发布者(Producer):产生事件并通知所有订阅者的组件
  • 订阅者(Subscriber):注册回调函数以接收特定事件通知的组件

通知链机制仅用于内核内部的通信,内核与用户空间之间的通信则使用其他机制。

通知链的数据结构

回调函数类型

通知链中的每个回调函数都是notifier_fn_t类型:

typedef int (*notifier_fn_t)(struct notifier_block *nb, 
                           unsigned long action, 
                           void *data);

参数说明:

  • nb:指向通知块(notifier_block)的指针
  • action:事件类型标识符
  • data:事件相关的附加数据

返回值:

  • NOTIFY_DONE:订阅者不关心此通知
  • NOTIFY_OK:通知处理成功
  • NOTIFY_BAD:处理过程中出错
  • NOTIFY_STOP:通知处理完成,但不应再调用后续回调

通知块结构

通知链的基本构建块是notifier_block结构:

struct notifier_block {
    notifier_fn_t notifier_call;  // 回调函数指针
    struct notifier_block __rcu *next;  // 链表中下一个通知块
    int priority;  // 回调优先级
};

通知链的类型

Linux内核提供了四种不同类型的通知链,主要区别在于它们的同步保护机制:

  1. 阻塞通知链(Blocking Notifier Chains)

    • 回调在进程上下文中执行
    • 使用读写信号量(rw_semaphore)保护
    • 调用可能被阻塞
  2. SRCU通知链(SRCU Notifier Chains)

    • 也是进程上下文中执行
    • 使用SRCU(可睡眠RCU)机制保护
    • 允许在读端临界区中阻塞
  3. 原子通知链(Atomic Notifier Chains)

    • 在中断或原子上下文中执行
    • 使用自旋锁(spinlock)保护
    • 调用不能被阻塞
  4. 原始通知链(Raw Notifier Chains)

    • 无任何锁保护
    • 调用者自行负责同步
    • 适用于需要特殊同步机制的场景

通知链的使用

初始化通知链

每种通知链都有对应的初始化宏/函数:

// 阻塞通知链
BLOCKING_NOTIFIER_HEAD(name)

// 原子通知链
ATOMIC_NOTIFIER_HEAD(name) 

// 原始通知链
RAW_NOTIFIER_HEAD(name)

// SRCU通知链
srcu_init_notifier(&name)

注册/注销通知

订阅者可以通过以下函数注册回调:

int blocking_notifier_chain_register(struct blocking_notifier_head *nh,
                                   struct notifier_block *nb);

int atomic_notifier_chain_register(struct atomic_notifier_head *nh,
                                 struct notifier_block *nb);
// 其他类型类似

注销则使用对应的unregister函数。

发送通知

生产者使用以下函数发送通知:

int blocking_notifier_call_chain(struct blocking_notifier_head *nh,
                               unsigned long val, void *v);

int atomic_notifier_call_chain(struct atomic_notifier_head *nh,
                             unsigned long val, void *v);
// 其他类型类似

实际应用示例

以内核模块为例,内核维护了一个模块状态变化的通知链:

static BLOCKING_NOTIFIER_HEAD(module_notify_list);

模块状态有三种可能的变化:

  • MODULE_STATE_LIVE:模块已加载完成
  • MODULE_STATE_COMING:模块正在加载
  • MODULE_STATE_GOING:模块正在卸载

其他子系统可以注册回调来跟踪模块状态变化。例如,内核的tracepoint子系统就注册了模块通知:

static struct notifier_block tracepoint_module_nb = {
    .notifier_call = tracepoint_module_notify,
    .priority = 0,
};

register_module_notifier(&tracepoint_module_nb);

当模块状态变化时(如通过init_moduledelete_module系统调用),内核会调用:

blocking_notifier_call_chain(&module_notify_list, event, mod);

从而通知所有注册的订阅者。

总结

通知链机制是Linux内核中实现发布-订阅模式的核心基础设施,它提供了一种灵活的方式让内核各组件之间进行松耦合的通信。通过四种不同类型的通知链,内核可以根据不同的上下文和同步需求选择最合适的实现方式。理解这一机制对于深入分析内核各子系统间的交互至关重要。

创作声明:本文部分内容由AI辅助生成(AIGC),仅供参考

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

尚舰舸Elsie

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值