pd->psy_nb.notifier_call = psy_changed; ret = power_supply_reg_notifier(&pd->psy_nb); if (ret) goto del_inst; /* force read initial power_supply values */ psy_changed(&pd->psy_nb, PSY_EVENT_PROP_CHANGED, pd->usb_psy);在另一个文件中有atomic_notifier_call_chain(&power_supply_notifier, PSY_EVENT_PROP_CHANGED, psy); kobject_uevent(&psy->dev.kobj, KOBJ_CHANGE);怎么通过中断相互调用
时间: 2024-02-17 19:26:03 浏览: 23
中断通常是由硬件设备引发的,它会打断CPU执行的当前任务,转而去执行中断服务程序。中断服务程序是在中断上下文中运行的,它的执行不能被打断,直到它完成为止。因此,在中断服务程序中,不能直接调用可能会睡眠的函数,否则会导致死锁或系统崩溃。
在这段代码中,如果在中断上下文中调用atomic_notifier_call_chain和kobject_uevent函数,可能会导致睡眠,因为这两个函数都会调用睡眠函数。因此,这两个函数不能在中断上下文中直接调用。
如果需要在中断服务程序中调用这些函数,可以使用工作队列或定时器来完成。将需要在中断上下文中执行的代码放入工作队列或定时器中,然后在合适的时机调度工作队列或定时器,以便在非中断上下文中执行这些代码。这样就可以避免在中断上下文中调用睡眠函数,从而保证系统的稳定性。
相关问题
static int psy_changed(struct notifier_block *nb, unsigned long evt, void ptr) pd->psy_nb.notifier_call = psy_changed; ret = power_supply_reg_notifier(&pd->psy_nb); if (ret) goto del_inst; / *force read initial power_supply values */ psy_changed(&pd->psy_nb, PSY_EVENT_PROP_CHANGED, pd->usb_psy); atomic_notifier_call_chain(&power_supply_notifier, PSY_EVENT_PROP_CHANGED, psy); kobject_uevent(&psy->dev.kobj, KOBJ_CHANGE);代码之间的关系
这段代码中的关系如下:
1. 首先定义了一个名为psy_changed的函数,该函数会在电源供应设备的属性变化时被调用。
2. 在代码中,pd->psy_nb.notifier_call被设置为psy_changed函数,以便在电源供应设备的属性变化时被调用。
3. 接着调用power_supply_reg_notifier函数,将pd->psy_nb注册为电源供应设备的notifier。如果注册失败,则跳转到del_inst标签处。
4. 为了强制读取电源供应设备的初始值,调用psy_changed函数,并传递PSY_EVENT_PROP_CHANGED和pd->usb_psy作为参数。
5. 接下来,调用atomic_notifier_call_chain函数,该函数会依次调用power_supply_notifier链表中的所有notifier,并传递PSY_EVENT_PROP_CHANGED和psy作为参数。
6. 最后,调用kobject_uevent函数,该函数会向用户空间发送一个内核对象的事件通知,通知用户空间有关电源供应设备的属性发生了变化。
atomic_notifier_call_chain和psy_changed(&pd->psy_nb, PSY_EVENT_PROP_CHANGED, pd->usb_psy);已经定义好,怎么在其他驱动文件调用psy_changed
如果你已经在一个驱动文件中定义了 `atomic_notifier_call_chain` 和 `psy_changed` 函数,并且想要在其他驱动文件中调用 `psy_changed` 函数,可以按照以下步骤进行操作:
1. 在需要调用 `psy_changed` 函数的驱动文件中,包含定义了 `psy_changed` 函数的头文件。
```c
#include "path/to/psy_changed.h"
```
2. 在需要调用 `psy_changed` 函数的地方,调用函数并传递需要的参数。
```c
struct power_supply *psy = ...; // 设置需要传递的参数
psy_changed(&pd->psy_nb, PSY_EVENT_PROP_CHANGED, psy);
```
这样,在其他驱动文件中就可以调用 `psy_changed` 函数了。需要注意的是,如果 `psy_changed` 函数中依赖于其他函数或全局变量,也需要在当前驱动文件中包含相应的头文件或定义相应的变量。
另外,`atomic_notifier_call_chain` 函数是一个通用的函数,可以在任何驱动文件中使用,用于回调注册的通知链。如果你需要在其他驱动文件中回调通知链,可以使用类似以下的代码:
```c
struct notifier_block *nb = ...; // 设置需要回调的通知链
atomic_notifier_call_chain(¬ification_chain, event, data);
```
其中 `notification_chain` 是一个 `notifier_block` 类型的链表,存储了多个需要回调的通知链。`event` 和 `data` 分别是通知链回调函数的参数。