2010-03-06 21:28:37 +01:00
/*
* drivers / base / power / generic_ops . c - Generic PM callbacks for subsystems
*
* Copyright ( c ) 2010 Rafael J . Wysocki < rjw @ sisk . pl > , Novell Inc .
*
* This file is released under the GPLv2 .
*/
# include <linux/pm.h>
# include <linux/pm_runtime.h>
2011-05-27 07:12:15 -04:00
# include <linux/export.h>
2010-03-06 21:28:37 +01:00
# ifdef CONFIG_PM_RUNTIME
/**
* pm_generic_runtime_idle - Generic runtime idle callback for subsystems .
* @ dev : Device to handle .
*
* If PM operations are defined for the @ dev ' s driver and they include
* - > runtime_idle ( ) , execute it and return its error code , if nonzero .
* Otherwise , execute pm_runtime_suspend ( ) for the device and return 0.
*/
int pm_generic_runtime_idle ( struct device * dev )
{
const struct dev_pm_ops * pm = dev - > driver ? dev - > driver - > pm : NULL ;
if ( pm & & pm - > runtime_idle ) {
int ret = pm - > runtime_idle ( dev ) ;
if ( ret )
return ret ;
}
pm_runtime_suspend ( dev ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( pm_generic_runtime_idle ) ;
/**
* pm_generic_runtime_suspend - Generic runtime suspend callback for subsystems .
* @ dev : Device to suspend .
*
* If PM operations are defined for the @ dev ' s driver and they include
* - > runtime_suspend ( ) , execute it and return its error code . Otherwise ,
2010-11-25 23:41:19 +01:00
* return 0.
2010-03-06 21:28:37 +01:00
*/
int pm_generic_runtime_suspend ( struct device * dev )
{
const struct dev_pm_ops * pm = dev - > driver ? dev - > driver - > pm : NULL ;
int ret ;
2010-09-09 00:46:16 +02:00
ret = pm & & pm - > runtime_suspend ? pm - > runtime_suspend ( dev ) : 0 ;
2010-03-06 21:28:37 +01:00
return ret ;
}
EXPORT_SYMBOL_GPL ( pm_generic_runtime_suspend ) ;
/**
* pm_generic_runtime_resume - Generic runtime resume callback for subsystems .
* @ dev : Device to resume .
*
* If PM operations are defined for the @ dev ' s driver and they include
* - > runtime_resume ( ) , execute it and return its error code . Otherwise ,
2010-11-25 23:41:19 +01:00
* return 0.
2010-03-06 21:28:37 +01:00
*/
int pm_generic_runtime_resume ( struct device * dev )
{
const struct dev_pm_ops * pm = dev - > driver ? dev - > driver - > pm : NULL ;
int ret ;
2010-09-09 00:46:16 +02:00
ret = pm & & pm - > runtime_resume ? pm - > runtime_resume ( dev ) : 0 ;
2010-03-06 21:28:37 +01:00
return ret ;
}
EXPORT_SYMBOL_GPL ( pm_generic_runtime_resume ) ;
# endif /* CONFIG_PM_RUNTIME */
# ifdef CONFIG_PM_SLEEP
2011-05-17 23:26:21 +02:00
/**
* pm_generic_prepare - Generic routine preparing a device for power transition .
* @ dev : Device to prepare .
*
* Prepare a device for a system - wide power transition .
*/
int pm_generic_prepare ( struct device * dev )
{
struct device_driver * drv = dev - > driver ;
int ret = 0 ;
if ( drv & & drv - > pm & & drv - > pm - > prepare )
ret = drv - > pm - > prepare ( dev ) ;
return ret ;
}
2010-03-06 21:28:37 +01:00
/**
* __pm_generic_call - Generic suspend / freeze / poweroff / thaw subsystem callback .
* @ dev : Device to handle .
* @ event : PM transition of the system under way .
2011-07-01 22:12:59 +02:00
* @ bool : Whether or not this is the " noirq " stage .
2010-03-06 21:28:37 +01:00
*
2011-12-15 20:59:30 +01:00
* Execute the PM callback corresponding to @ event provided by the driver of
2011-12-15 20:59:23 +01:00
* @ dev , if defined , and return its error code . Return 0 if the callback is
* not present .
2010-03-06 21:28:37 +01:00
*/
2011-07-01 22:12:59 +02:00
static int __pm_generic_call ( struct device * dev , int event , bool noirq )
2010-03-06 21:28:37 +01:00
{
const struct dev_pm_ops * pm = dev - > driver ? dev - > driver - > pm : NULL ;
int ( * callback ) ( struct device * ) ;
2011-12-15 20:59:23 +01:00
if ( ! pm )
2010-03-06 21:28:37 +01:00
return 0 ;
switch ( event ) {
case PM_EVENT_SUSPEND :
2011-07-01 22:12:59 +02:00
callback = noirq ? pm - > suspend_noirq : pm - > suspend ;
2010-03-06 21:28:37 +01:00
break ;
case PM_EVENT_FREEZE :
2011-07-01 22:12:59 +02:00
callback = noirq ? pm - > freeze_noirq : pm - > freeze ;
2010-03-06 21:28:37 +01:00
break ;
case PM_EVENT_HIBERNATE :
2011-07-01 22:12:59 +02:00
callback = noirq ? pm - > poweroff_noirq : pm - > poweroff ;
2010-03-06 21:28:37 +01:00
break ;
2011-12-15 20:59:30 +01:00
case PM_EVENT_RESUME :
callback = noirq ? pm - > resume_noirq : pm - > resume ;
break ;
2010-03-06 21:28:37 +01:00
case PM_EVENT_THAW :
2011-07-01 22:12:59 +02:00
callback = noirq ? pm - > thaw_noirq : pm - > thaw ;
2010-03-06 21:28:37 +01:00
break ;
2011-12-15 20:59:30 +01:00
case PM_EVENT_RESTORE :
callback = noirq ? pm - > restore_noirq : pm - > restore ;
break ;
2010-03-06 21:28:37 +01:00
default :
callback = NULL ;
break ;
}
return callback ? callback ( dev ) : 0 ;
}
2011-07-01 22:12:59 +02:00
/**
* pm_generic_suspend_noirq - Generic suspend_noirq callback for subsystems .
* @ dev : Device to suspend .
*/
int pm_generic_suspend_noirq ( struct device * dev )
{
return __pm_generic_call ( dev , PM_EVENT_SUSPEND , true ) ;
}
EXPORT_SYMBOL_GPL ( pm_generic_suspend_noirq ) ;
2010-03-06 21:28:37 +01:00
/**
* pm_generic_suspend - Generic suspend callback for subsystems .
* @ dev : Device to suspend .
*/
int pm_generic_suspend ( struct device * dev )
{
2011-07-01 22:12:59 +02:00
return __pm_generic_call ( dev , PM_EVENT_SUSPEND , false ) ;
2010-03-06 21:28:37 +01:00
}
EXPORT_SYMBOL_GPL ( pm_generic_suspend ) ;
2011-07-01 22:12:59 +02:00
/**
* pm_generic_freeze_noirq - Generic freeze_noirq callback for subsystems .
* @ dev : Device to freeze .
*/
int pm_generic_freeze_noirq ( struct device * dev )
{
return __pm_generic_call ( dev , PM_EVENT_FREEZE , true ) ;
}
EXPORT_SYMBOL_GPL ( pm_generic_freeze_noirq ) ;
2010-03-06 21:28:37 +01:00
/**
* pm_generic_freeze - Generic freeze callback for subsystems .
* @ dev : Device to freeze .
*/
int pm_generic_freeze ( struct device * dev )
{
2011-07-01 22:12:59 +02:00
return __pm_generic_call ( dev , PM_EVENT_FREEZE , false ) ;
2010-03-06 21:28:37 +01:00
}
EXPORT_SYMBOL_GPL ( pm_generic_freeze ) ;
2011-07-01 22:12:59 +02:00
/**
* pm_generic_poweroff_noirq - Generic poweroff_noirq callback for subsystems .
* @ dev : Device to handle .
*/
int pm_generic_poweroff_noirq ( struct device * dev )
{
return __pm_generic_call ( dev , PM_EVENT_HIBERNATE , true ) ;
}
EXPORT_SYMBOL_GPL ( pm_generic_poweroff_noirq ) ;
2010-03-06 21:28:37 +01:00
/**
* pm_generic_poweroff - Generic poweroff callback for subsystems .
* @ dev : Device to handle .
*/
int pm_generic_poweroff ( struct device * dev )
{
2011-07-01 22:12:59 +02:00
return __pm_generic_call ( dev , PM_EVENT_HIBERNATE , false ) ;
2010-03-06 21:28:37 +01:00
}
EXPORT_SYMBOL_GPL ( pm_generic_poweroff ) ;
2011-07-01 22:12:59 +02:00
/**
* pm_generic_thaw_noirq - Generic thaw_noirq callback for subsystems .
* @ dev : Device to thaw .
*/
int pm_generic_thaw_noirq ( struct device * dev )
{
return __pm_generic_call ( dev , PM_EVENT_THAW , true ) ;
}
EXPORT_SYMBOL_GPL ( pm_generic_thaw_noirq ) ;
2010-03-06 21:28:37 +01:00
/**
* pm_generic_thaw - Generic thaw callback for subsystems .
* @ dev : Device to thaw .
*/
int pm_generic_thaw ( struct device * dev )
{
2011-07-01 22:12:59 +02:00
return __pm_generic_call ( dev , PM_EVENT_THAW , false ) ;
2010-03-06 21:28:37 +01:00
}
EXPORT_SYMBOL_GPL ( pm_generic_thaw ) ;
2011-07-01 22:12:59 +02:00
/**
* pm_generic_resume_noirq - Generic resume_noirq callback for subsystems .
* @ dev : Device to resume .
*/
int pm_generic_resume_noirq ( struct device * dev )
{
2011-12-15 20:59:30 +01:00
return __pm_generic_call ( dev , PM_EVENT_RESUME , true ) ;
2011-07-01 22:12:59 +02:00
}
EXPORT_SYMBOL_GPL ( pm_generic_resume_noirq ) ;
2010-03-06 21:28:37 +01:00
/**
* pm_generic_resume - Generic resume callback for subsystems .
* @ dev : Device to resume .
*/
int pm_generic_resume ( struct device * dev )
{
2011-12-15 20:59:30 +01:00
return __pm_generic_call ( dev , PM_EVENT_RESUME , false ) ;
2010-03-06 21:28:37 +01:00
}
EXPORT_SYMBOL_GPL ( pm_generic_resume ) ;
2011-07-01 22:12:59 +02:00
/**
* pm_generic_restore_noirq - Generic restore_noirq callback for subsystems .
* @ dev : Device to restore .
*/
int pm_generic_restore_noirq ( struct device * dev )
{
2011-12-15 20:59:30 +01:00
return __pm_generic_call ( dev , PM_EVENT_RESTORE , true ) ;
2011-07-01 22:12:59 +02:00
}
EXPORT_SYMBOL_GPL ( pm_generic_restore_noirq ) ;
2010-03-06 21:28:37 +01:00
/**
* pm_generic_restore - Generic restore callback for subsystems .
* @ dev : Device to restore .
*/
int pm_generic_restore ( struct device * dev )
{
2011-12-15 20:59:30 +01:00
return __pm_generic_call ( dev , PM_EVENT_RESTORE , false ) ;
2010-03-06 21:28:37 +01:00
}
EXPORT_SYMBOL_GPL ( pm_generic_restore ) ;
2011-05-17 23:26:21 +02:00
/**
* pm_generic_complete - Generic routine competing a device power transition .
* @ dev : Device to handle .
*
* Complete a device power transition during a system - wide power transition .
*/
void pm_generic_complete ( struct device * dev )
{
struct device_driver * drv = dev - > driver ;
if ( drv & & drv - > pm & & drv - > pm - > complete )
drv - > pm - > complete ( dev ) ;
/*
* Let runtime PM try to suspend devices that haven ' t been in use before
* going into the system - wide sleep state we ' re resuming from .
*/
pm_runtime_idle ( dev ) ;
}
2010-03-06 21:28:37 +01:00
# endif /* CONFIG_PM_SLEEP */