2009-03-16 22:33:49 +01:00
/*
* linux / kernel / irq / pm . c
*
* Copyright ( C ) 2009 Rafael J . Wysocki < rjw @ sisk . pl > , Novell Inc .
*
* This file contains power management functions related to interrupts .
*/
# include <linux/irq.h>
# include <linux/module.h>
# include <linux/interrupt.h>
# include "internals.h"
/**
* suspend_device_irqs - disable all currently enabled interrupt lines
*
2009-07-05 00:22:34 +02:00
* During system - wide suspend or hibernation device drivers need to be prevented
* from receiving interrupts and this function is provided for this purpose .
* It marks all interrupt lines in use , except for the timer ones , as disabled
2011-02-08 12:44:58 +01:00
* and sets the IRQS_SUSPENDED flag for each of them .
2009-03-16 22:33:49 +01:00
*/
void suspend_device_irqs ( void )
{
struct irq_desc * desc ;
int irq ;
for_each_irq_desc ( irq , desc ) {
unsigned long flags ;
2009-11-17 16:46:45 +01:00
raw_spin_lock_irqsave ( & desc - > lock , flags ) ;
2009-03-16 22:33:49 +01:00
__disable_irq ( desc , irq , true ) ;
2009-11-17 16:46:45 +01:00
raw_spin_unlock_irqrestore ( & desc - > lock , flags ) ;
2009-03-16 22:33:49 +01:00
}
for_each_irq_desc ( irq , desc )
2011-02-08 12:44:58 +01:00
if ( desc - > istate & IRQS_SUSPENDED )
2009-03-16 22:33:49 +01:00
synchronize_irq ( irq ) ;
}
EXPORT_SYMBOL_GPL ( suspend_device_irqs ) ;
/**
* resume_device_irqs - enable interrupt lines disabled by suspend_device_irqs ( )
*
* Enable all interrupt lines previously disabled by suspend_device_irqs ( ) that
2011-02-08 12:44:58 +01:00
* have the IRQS_SUSPENDED flag set .
2009-03-16 22:33:49 +01:00
*/
void resume_device_irqs ( void )
{
struct irq_desc * desc ;
int irq ;
for_each_irq_desc ( irq , desc ) {
unsigned long flags ;
2009-11-17 16:46:45 +01:00
raw_spin_lock_irqsave ( & desc - > lock , flags ) ;
2009-03-16 22:33:49 +01:00
__enable_irq ( desc , irq , true ) ;
2009-11-17 16:46:45 +01:00
raw_spin_unlock_irqrestore ( & desc - > lock , flags ) ;
2009-03-16 22:33:49 +01:00
}
}
EXPORT_SYMBOL_GPL ( resume_device_irqs ) ;
/**
* check_wakeup_irqs - check if any wake - up interrupts are pending
*/
int check_wakeup_irqs ( void )
{
struct irq_desc * desc ;
int irq ;
2011-03-11 21:22:14 +01:00
for_each_irq_desc ( irq , desc ) {
if ( irqd_is_wakeup_set ( & desc - > irq_data ) ) {
if ( desc - > istate & IRQS_PENDING )
return - EBUSY ;
continue ;
}
/*
* Check the non wakeup interrupts whether they need
* to be masked before finally going into suspend
* state . That ' s for hardware which has no wakeup
* source configuration facility . The chip
* implementation indicates that with
* IRQCHIP_MASK_ON_SUSPEND .
*/
if ( desc - > istate & IRQS_SUSPENDED & &
irq_desc_get_chip ( desc ) - > flags & IRQCHIP_MASK_ON_SUSPEND )
mask_irq ( desc ) ;
}
2009-03-16 22:33:49 +01:00
return 0 ;
}