enable_irq_wake(irq);有嵌套层次,多次执行后如果去执行一次disable_irq_wake,结果不能够disable irq wake,
要enable和disable的次数相同
static inline int enable_irq_wake(unsigned int irq)
{
return set_irq_wake(irq, 1);
}
static inline int disable_irq_wake(unsigned int irq)
{
return set_irq_wake(irq, 0);
}
/**
*set_irq_wake - control irq power management wakeup
*@irq:interrupt to control
*@on:enable/disable power management wakeup
*
*Enable/disable power management wakeup mode, which is
*disabled by default. Enables and disables must match,
*just as they match for non-wakeup mode support.
*
*Wakeup mode lets this IRQ wake the system from sleep
*states like "suspend to RAM".
*/
int set_irq_wake(unsigned int irq, unsigned int on)
{
struct irq_desc *desc = irq_to_desc(irq);
unsigned long flags;
int ret = 0;
/* wakeup-capable irqs can be shared between drivers that
* don't need to have the same sleep mode behaviors.
*/
spin_lock_irqsave(&desc->lock, flags);
if (on) {
if (desc->wake_depth++ == 0) {
ret = set_irq_wake_real(irq, on);
if (ret)
desc->wake_depth = 0;
else
desc->status |= IRQ_WAKEUP;
}
} else {
if (desc->wake_depth == 0) {
WARN(1, "Unbalanced IRQ %d wake disable/n", irq);
} else if (--desc->wake_depth == 0) {
ret = set_irq_wake_real(irq, on);
if (ret)
desc->wake_depth = 1;
else
desc->status &= ~IRQ_WAKEUP;
}
}
spin_unlock_irqrestore(&desc->lock, flags);
return ret;
}
EXPORT_SYMBOL(set_irq_wake);
enable
struct irq_desc *desc = irq_to_desc(irq);
desc->wake_depth = 0;
enable_irq_wake(irq);
disable
struct irq_desc *desc = irq_to_desc(irq);
desc->wake_depth = 1;
disable_irq_wake(irq);