2008-04-30 20:05:33 +04:00
/*
* reg - virtual - consumer . c
*
* Copyright 2008 Wolfson Microelectronics PLC .
*
* Author : Mark Brown < broonie @ opensource . wolfsonmicro . com >
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation ; either version 2 of the
* License , or ( at your option ) any later version .
*/
# include <linux/err.h>
# include <linux/mutex.h>
# include <linux/platform_device.h>
# include <linux/regulator/consumer.h>
struct virtual_consumer_data {
struct mutex lock ;
struct regulator * regulator ;
int enabled ;
int min_uV ;
int max_uV ;
int min_uA ;
int max_uA ;
unsigned int mode ;
} ;
2009-06-17 18:45:06 +04:00
static void update_voltage_constraints ( struct device * dev ,
struct virtual_consumer_data * data )
2008-04-30 20:05:33 +04:00
{
int ret ;
if ( data - > min_uV & & data - > max_uV
& & data - > min_uV < = data - > max_uV ) {
2009-06-17 18:45:07 +04:00
dev_dbg ( dev , " Requesting %d-%duV \n " ,
data - > min_uV , data - > max_uV ) ;
2008-04-30 20:05:33 +04:00
ret = regulator_set_voltage ( data - > regulator ,
2009-06-17 18:45:06 +04:00
data - > min_uV , data - > max_uV ) ;
2008-04-30 20:05:33 +04:00
if ( ret ! = 0 ) {
2009-06-17 18:45:06 +04:00
dev_err ( dev ,
" regulator_set_voltage() failed: %d \n " , ret ) ;
2008-04-30 20:05:33 +04:00
return ;
}
}
if ( data - > min_uV & & data - > max_uV & & ! data - > enabled ) {
2009-06-17 18:45:07 +04:00
dev_dbg ( dev , " Enabling regulator \n " ) ;
2008-04-30 20:05:33 +04:00
ret = regulator_enable ( data - > regulator ) ;
if ( ret = = 0 )
data - > enabled = 1 ;
else
2009-06-17 18:45:06 +04:00
dev_err ( dev , " regulator_enable() failed: %d \n " ,
2008-04-30 20:05:33 +04:00
ret ) ;
}
if ( ! ( data - > min_uV & & data - > max_uV ) & & data - > enabled ) {
2009-06-17 18:45:07 +04:00
dev_dbg ( dev , " Disabling regulator \n " ) ;
2008-04-30 20:05:33 +04:00
ret = regulator_disable ( data - > regulator ) ;
if ( ret = = 0 )
data - > enabled = 0 ;
else
2009-06-17 18:45:06 +04:00
dev_err ( dev , " regulator_disable() failed: %d \n " ,
2008-04-30 20:05:33 +04:00
ret ) ;
}
}
2009-06-17 18:45:06 +04:00
static void update_current_limit_constraints ( struct device * dev ,
struct virtual_consumer_data * data )
2008-04-30 20:05:33 +04:00
{
int ret ;
if ( data - > max_uA
& & data - > min_uA < = data - > max_uA ) {
2009-06-17 18:45:07 +04:00
dev_dbg ( dev , " Requesting %d-%duA \n " ,
data - > min_uA , data - > max_uA ) ;
2008-04-30 20:05:33 +04:00
ret = regulator_set_current_limit ( data - > regulator ,
data - > min_uA , data - > max_uA ) ;
if ( ret ! = 0 ) {
2009-06-17 18:45:06 +04:00
dev_err ( dev ,
" regulator_set_current_limit() failed: %d \n " ,
ret ) ;
2008-04-30 20:05:33 +04:00
return ;
}
}
if ( data - > max_uA & & ! data - > enabled ) {
2009-06-17 18:45:07 +04:00
dev_dbg ( dev , " Enabling regulator \n " ) ;
2008-04-30 20:05:33 +04:00
ret = regulator_enable ( data - > regulator ) ;
if ( ret = = 0 )
data - > enabled = 1 ;
else
2009-06-17 18:45:06 +04:00
dev_err ( dev , " regulator_enable() failed: %d \n " ,
2008-04-30 20:05:33 +04:00
ret ) ;
}
if ( ! ( data - > min_uA & & data - > max_uA ) & & data - > enabled ) {
2009-06-17 18:45:07 +04:00
dev_dbg ( dev , " Disabling regulator \n " ) ;
2008-04-30 20:05:33 +04:00
ret = regulator_disable ( data - > regulator ) ;
if ( ret = = 0 )
data - > enabled = 0 ;
else
2009-06-17 18:45:06 +04:00
dev_err ( dev , " regulator_disable() failed: %d \n " ,
2008-04-30 20:05:33 +04:00
ret ) ;
}
}
static ssize_t show_min_uV ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %d \n " , data - > min_uV ) ;
}
static ssize_t set_min_uV ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
long val ;
if ( strict_strtol ( buf , 10 , & val ) ! = 0 )
return count ;
mutex_lock ( & data - > lock ) ;
data - > min_uV = val ;
2009-06-17 18:45:06 +04:00
update_voltage_constraints ( dev , data ) ;
2008-04-30 20:05:33 +04:00
mutex_unlock ( & data - > lock ) ;
return count ;
}
static ssize_t show_max_uV ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %d \n " , data - > max_uV ) ;
}
static ssize_t set_max_uV ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
long val ;
if ( strict_strtol ( buf , 10 , & val ) ! = 0 )
return count ;
mutex_lock ( & data - > lock ) ;
data - > max_uV = val ;
2009-06-17 18:45:06 +04:00
update_voltage_constraints ( dev , data ) ;
2008-04-30 20:05:33 +04:00
mutex_unlock ( & data - > lock ) ;
return count ;
}
static ssize_t show_min_uA ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %d \n " , data - > min_uA ) ;
}
static ssize_t set_min_uA ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
long val ;
if ( strict_strtol ( buf , 10 , & val ) ! = 0 )
return count ;
mutex_lock ( & data - > lock ) ;
data - > min_uA = val ;
2009-06-17 18:45:06 +04:00
update_current_limit_constraints ( dev , data ) ;
2008-04-30 20:05:33 +04:00
mutex_unlock ( & data - > lock ) ;
return count ;
}
static ssize_t show_max_uA ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %d \n " , data - > max_uA ) ;
}
static ssize_t set_max_uA ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
long val ;
if ( strict_strtol ( buf , 10 , & val ) ! = 0 )
return count ;
mutex_lock ( & data - > lock ) ;
data - > max_uA = val ;
2009-06-17 18:45:06 +04:00
update_current_limit_constraints ( dev , data ) ;
2008-04-30 20:05:33 +04:00
mutex_unlock ( & data - > lock ) ;
return count ;
}
static ssize_t show_mode ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
switch ( data - > mode ) {
case REGULATOR_MODE_FAST :
return sprintf ( buf , " fast \n " ) ;
case REGULATOR_MODE_NORMAL :
return sprintf ( buf , " normal \n " ) ;
case REGULATOR_MODE_IDLE :
return sprintf ( buf , " idle \n " ) ;
case REGULATOR_MODE_STANDBY :
return sprintf ( buf , " standby \n " ) ;
default :
return sprintf ( buf , " unknown \n " ) ;
}
}
static ssize_t set_mode ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t count )
{
struct virtual_consumer_data * data = dev_get_drvdata ( dev ) ;
unsigned int mode ;
int ret ;
regulator: minor cleanup of virtual consumer
On Thu, 15 Jan 2009 16:10:22 -0800
Andrew Morton <akpm@linux-foundation.org> wrote:
> On Wed, 14 Jan 2009 13:16:27 -0800
> David Brownell <david-b@pacbell.net> wrote:
>
> > From: David Brownell <dbrownell@users.sourceforge.net>
> >
> > Minor cleanup to the regulator set_mode sysfs support:
> > switch to sysfs_streq() in set_mode(), which is also
> > a code shrink. Use the same strings that get_mode()
> > uses, shrinking data too.
> >
> > Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
> > ---
> > drivers/regulator/virtual.c | 8 ++++----
> > 1 file changed, 4 insertions(+), 4 deletions(-)
> >
> > --- a/drivers/regulator/virtual.c
> > +++ b/drivers/regulator/virtual.c
> > @@ -226,13 +226,13 @@ static ssize_t set_mode(struct device *d
> > unsigned int mode;
> > int ret;
> >
> > - if (strncmp(buf, "fast", strlen("fast")) == 0)
> > + if (sysfs_streq(buf, "fast\n") == 0)
> > mode = REGULATOR_MODE_FAST;
> > - else if (strncmp(buf, "normal", strlen("normal")) == 0)
> > + else if (sysfs_streq(buf, "normal\n") == 0)
> > mode = REGULATOR_MODE_NORMAL;
> > - else if (strncmp(buf, "idle", strlen("idle")) == 0)
> > + else if (sysfs_streq(buf, "idle\n") == 0)
> > mode = REGULATOR_MODE_IDLE;
> > - else if (strncmp(buf, "standby", strlen("standby")) == 0)
> > + else if (sysfs_streq(buf, "standby\n") == 0)
> > mode = REGULATOR_MODE_STANDBY;
>
> we don't need the \n's, do we?
oh, it's for the string sharing. Sneaky.
I wonder how many people will try to fix that up for us?
Acked-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
Signed-off-by: Liam Girdwood <lrg@slimlogic.co.uk>
2009-01-16 03:13:01 +03:00
/*
* sysfs_streq ( ) doesn ' t need the \ n ' s , but we add them so the strings
* will be shared with show_mode ( ) , above .
*/
2009-04-26 12:49:30 +04:00
if ( sysfs_streq ( buf , " fast \n " ) )
2008-04-30 20:05:33 +04:00
mode = REGULATOR_MODE_FAST ;
2009-04-26 12:49:30 +04:00
else if ( sysfs_streq ( buf , " normal \n " ) )
2008-04-30 20:05:33 +04:00
mode = REGULATOR_MODE_NORMAL ;
2009-04-26 12:49:30 +04:00
else if ( sysfs_streq ( buf , " idle \n " ) )
2008-04-30 20:05:33 +04:00
mode = REGULATOR_MODE_IDLE ;
2009-04-26 12:49:30 +04:00
else if ( sysfs_streq ( buf , " standby \n " ) )
2008-04-30 20:05:33 +04:00
mode = REGULATOR_MODE_STANDBY ;
else {
dev_err ( dev , " Configuring invalid mode \n " ) ;
return count ;
}
mutex_lock ( & data - > lock ) ;
ret = regulator_set_mode ( data - > regulator , mode ) ;
if ( ret = = 0 )
data - > mode = mode ;
else
dev_err ( dev , " Failed to configure mode: %d \n " , ret ) ;
mutex_unlock ( & data - > lock ) ;
return count ;
}
static DEVICE_ATTR ( min_microvolts , 0666 , show_min_uV , set_min_uV ) ;
static DEVICE_ATTR ( max_microvolts , 0666 , show_max_uV , set_max_uV ) ;
static DEVICE_ATTR ( min_microamps , 0666 , show_min_uA , set_min_uA ) ;
static DEVICE_ATTR ( max_microamps , 0666 , show_max_uA , set_max_uA ) ;
static DEVICE_ATTR ( mode , 0666 , show_mode , set_mode ) ;
2009-01-19 16:37:06 +03:00
static struct device_attribute * attributes [ ] = {
2008-04-30 20:05:33 +04:00
& dev_attr_min_microvolts ,
& dev_attr_max_microvolts ,
& dev_attr_min_microamps ,
& dev_attr_max_microamps ,
& dev_attr_mode ,
} ;
static int regulator_virtual_consumer_probe ( struct platform_device * pdev )
{
char * reg_id = pdev - > dev . platform_data ;
struct virtual_consumer_data * drvdata ;
int ret , i ;
drvdata = kzalloc ( sizeof ( struct virtual_consumer_data ) , GFP_KERNEL ) ;
if ( drvdata = = NULL ) {
2009-07-21 19:00:27 +04:00
return - ENOMEM ;
2008-04-30 20:05:33 +04:00
}
mutex_init ( & drvdata - > lock ) ;
drvdata - > regulator = regulator_get ( & pdev - > dev , reg_id ) ;
if ( IS_ERR ( drvdata - > regulator ) ) {
ret = PTR_ERR ( drvdata - > regulator ) ;
2009-06-15 23:01:01 +04:00
dev_err ( & pdev - > dev , " Failed to obtain supply '%s': %d \n " ,
reg_id , ret ) ;
2008-04-30 20:05:33 +04:00
goto err ;
}
for ( i = 0 ; i < ARRAY_SIZE ( attributes ) ; i + + ) {
ret = device_create_file ( & pdev - > dev , attributes [ i ] ) ;
2009-07-21 19:00:27 +04:00
if ( ret ! = 0 ) {
dev_err ( & pdev - > dev , " Failed to create attr %d: %d \n " ,
i , ret ) ;
goto err_regulator ;
}
2008-04-30 20:05:33 +04:00
}
drvdata - > mode = regulator_get_mode ( drvdata - > regulator ) ;
platform_set_drvdata ( pdev , drvdata ) ;
return 0 ;
2009-07-21 19:00:27 +04:00
err_regulator :
regulator_put ( drvdata - > regulator ) ;
2008-04-30 20:05:33 +04:00
err :
for ( i = 0 ; i < ARRAY_SIZE ( attributes ) ; i + + )
device_remove_file ( & pdev - > dev , attributes [ i ] ) ;
kfree ( drvdata ) ;
return ret ;
}
static int regulator_virtual_consumer_remove ( struct platform_device * pdev )
{
struct virtual_consumer_data * drvdata = platform_get_drvdata ( pdev ) ;
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( attributes ) ; i + + )
device_remove_file ( & pdev - > dev , attributes [ i ] ) ;
if ( drvdata - > enabled )
regulator_disable ( drvdata - > regulator ) ;
regulator_put ( drvdata - > regulator ) ;
kfree ( drvdata ) ;
return 0 ;
}
static struct platform_driver regulator_virtual_consumer_driver = {
. probe = regulator_virtual_consumer_probe ,
. remove = regulator_virtual_consumer_remove ,
. driver = {
. name = " reg-virt-consumer " ,
} ,
} ;
static int __init regulator_virtual_consumer_init ( void )
{
return platform_driver_register ( & regulator_virtual_consumer_driver ) ;
}
module_init ( regulator_virtual_consumer_init ) ;
static void __exit regulator_virtual_consumer_exit ( void )
{
platform_driver_unregister ( & regulator_virtual_consumer_driver ) ;
}
module_exit ( regulator_virtual_consumer_exit ) ;
MODULE_AUTHOR ( " Mark Brown <broonie@opensource.wolfsonmicro.com> " ) ;
MODULE_DESCRIPTION ( " Virtual regulator consumer " ) ;
MODULE_LICENSE ( " GPL " ) ;
2009-04-28 14:13:55 +04:00
MODULE_ALIAS ( " platform:reg-virt-consumer " ) ;