Linux下节能中断唤醒
Linux下节能中断唤醒在linux系统进入节能,如进入mem之后,可以通过中断唤醒系统。选择在节能模式下,是否支持该中断唤醒系统是本文要讲的内容。例如,在触摸屏的suspend方法中,调用enable_irq_wake,则设置支持触摸屏中断唤醒系统休眠,在resume方法中,调用disable_irq_wake关闭支持。enable_irq_wake就该函数而言,
·
Linux下节能中断唤醒
在linux系统进入节能,如进入mem之后,可以通过中断唤醒系统。选择在节能模式下,是否支持该中断唤醒系统是本文要讲的内容。
例如,在触摸屏的suspend方法中,调用enable_irq_wake,则设置支持触摸屏中断唤醒系统休眠,在resume方法中,调用disable_irq_wake关闭支持。
enable_irq_wake
就该函数而言,可以从两方面思考。一,必须在进入suspend之前设置支持该中断,二,在中断唤醒表中标记对应的中断号,支持其中断唤醒。
enable_irq_wake调用函数set_irq_wake(irq,1),跟踪set_irq_wake.该函数首先获得中断描述符,禁止所有中断,调用set_irq_wake_real设置中断(步骤一),接着设置desc->status|=IRQ_WAKEUP(步骤二),该status会在进入suspend之前,通过检查status的标志位是否有IRQ_WAKEUP来决定是否支持中断唤醒休眠。
步骤一,set_irq_wake_real,该函数如下,
318 static int set_irq_wake_real(unsigned int irq, unsigned int on)
319 {
320 struct irq_desc *desc = irq_to_desc(irq);
321 int ret = -ENXIO;
322
323 if (desc->chip->set_wake)
324 ret = desc->chip->set_wake(irq, on);
325
326 return ret;
327 }
该函数实际调用的是中断相关芯片的set_wake操作。
对于freescale的mx25而言(与体系相关),在MACHINE_START中,通过machine_desc,不同的体系有不同的machine_desc,配置中断,在set_irq_handler被调用时,有set_irq_chip设置chip操作,set_irq_chip实际就是不同的中断号对应的chip指针指向mxc_avic_chip.
static struct irq_chip mxc_avic_chip = {
.ack = mxc_mask_irq,
.mask = mxc_mask_irq,
.unmask = mxc_unmask_irq,
.set_wake = mxc_set_wake_irq,
};
Chip的所有操作,包括startup,enable,disable都是在上述方法的基础上封装的,具体与寄存器配置有关。
Set_wake方法实际就是mxc系列在suspend期间允许相关中断的寄存器配置操作。
步骤二,desc->status|=IRQ_WAKEUP,
关于IRQ_WAKEUP,有两处涉及,
(1)check_wakeup_irqs,该函数被sysdev_suspend所调用。在准备进入suspend时,会调用sysdev_suspend,该函数中通过调用check_wakeup_irqs,检查设置IRQ_WAKEUP的中断是否还有在pending状态,即进入中断未退出的,如果有,则禁止进入suspend。
(2)对于不同的体系,进入节能配置允许中断唤醒的操作各不相同。所以Linux下有platform_suspend_ops提供出来,从而区别不同的方法统一操作。
在late_initcall(mx25_pm_init)中,调用suspend_set_ops的方法设置platform_suspend_ops设置ops.
struct platform_suspend_ops mx25_suspend_ops = {
.begin = mx25_pm_begin,
.valid = mx25_pm_valid,
.prepare = mx25_suspend_prepare,
.enter = mx25_suspend_enter,
.finish = mx25_suspend_finish,
};
在sysdev_suspend成功之后,会调用mx25的enter方法,即mx25_suspend_enter,该函数最终调用mxc_cpu_lp_set(“mem”模式suspend)。该函数中,
if ((desc->status & IRQ_WAKEUP) != 0)
lpimr &= ~(1 << (i - 32));
配置lpimr寄存器,该寄存器就是允许休眠中断唤醒的中断设置。
整体上睡眠中断唤醒的流程就这样,而disable_irq_wakeup与enable_irq_wakeup只是简单的类似操作。
更多推荐
已为社区贡献3条内容
所有评论(0)