Hi Anton, Following patch needs your attention.
Thanks, Ram > From: Pallala, Ramakrishna > Sent: Monday, June 25, 2012 6:07 PM > To: linux-kernel@vger.kernel.org > Cc: Anton Vorontsov; Anton Vorontsov; Pallala, Ramakrishna > Subject: [PATCH] power_supply: Added support for power supply attribute > sources > > On some platforms one driver(or HW chip) may not be able to provide all the > necessary attributes of the power supply connected to the platform or may > provide very limited info which can be used by core/primary drivers. > > For example a temperature sensor chip placed near the battery can be used to > report battery ambient temperature but it does not makes sense to register > sensor driver with power supply class. Or even a ADC driver or platform driver > may report power supply properties like voltage/current or charging status but > registering all those driver with power supply class is not a practical or > ideal > approach. > > This patch adds the generic support to register the drivers as power supply > attribute(properties) sources and adds an interface to read these attributes > from > power supply class drivers. > > Signed-off-by: Ramakrishna Pallala <ramakrishna.pall...@intel.com> > --- > Documentation/power/power_supply_class.txt | 30 +++++++++++ > drivers/power/power_supply_core.c | 78 > ++++++++++++++++++++++++++++ > include/linux/power_supply.h | 28 ++++++++++ > 3 files changed, 136 insertions(+), 0 deletions(-) > > diff --git a/Documentation/power/power_supply_class.txt > b/Documentation/power/power_supply_class.txt > index c0f62ae..f8ceb45 100644 > --- a/Documentation/power/power_supply_class.txt > +++ b/Documentation/power/power_supply_class.txt > @@ -130,6 +130,36 @@ while battery powers a load) TIME_TO_FULL - seconds > left for battery to be considered full (i.e. > while battery is charging) > > +Power supply attribute sources > +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ > +On some platforms one driver(or HW chip) may not be able to provide all > +the necessary attributes of the power supply connected to the platform. > + > +For example a temperature sensor chip placed near the battery can be > +used to report battery ambient temperature but it does not makes sense > +to register sensor driver with power supply class. Or even a ADC driver > +or platform driver may report power supply properties like > +voltage/current or charging status but registering all those driver > +with power supply class is not a practical or ideal approach. > + > +Power supply subsystem provides an interface to register and report > +about these power supply attributes to the primary driver which is > +registered with power supply class. > + > +Power supply attribute source driver can use the following functions to > +register/unregister as attributes source. > + > +int power_supply_attributes_register(struct device *parent, > + struct power_supply_attr_source *psy_attr); > + > +void power_supply_attributes_unregister( > + struct power_supply_attr_source *psy_attr); > + > +Power supply class driver(consumer driver) which needs to get power > +supply attributes can call the following function. > + > +int power_supply_get_external_attr( > + struct power_supply_attr_query *query); > > Battery <-> external power supply interaction > ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ > diff --git a/drivers/power/power_supply_core.c > b/drivers/power/power_supply_core.c > index ff990d2..25c47a3 100644 > --- a/drivers/power/power_supply_core.c > +++ b/drivers/power/power_supply_core.c > @@ -14,6 +14,7 @@ > #include <linux/types.h> > #include <linux/init.h> > #include <linux/slab.h> > +#include <linux/list.h> > #include <linux/device.h> > #include <linux/err.h> > #include <linux/power_supply.h> > @@ -26,6 +27,8 @@ EXPORT_SYMBOL_GPL(power_supply_class); > > static struct device_type power_supply_dev_type; > > +static LIST_HEAD(list_head_source_attr); > + > static int __power_supply_changed_work(struct device *dev, void *data) { > struct power_supply *psy = (struct power_supply *)data; @@ -164,6 > +167,34 @@ int power_supply_powers(struct power_supply *psy, struct device > *dev) } EXPORT_SYMBOL_GPL(power_supply_powers); > > +int power_supply_get_external_attr(struct power_supply_attr_query > +*query) { > + struct list_head *list; > + struct power_supply_attr_source *psy_attr; > + int ret = -ENODEV; > + > + if (!query || list_empty(&list_head_source_attr)) > + return -EINVAL; > + > + list_for_each(list, &list_head_source_attr) { > + psy_attr = list_entry(list, > + struct power_supply_attr_source, attr_pool); > + > + if (psy_attr->type != query->type) > + continue; > + > + ret = psy_attr->get_property(psy_attr, > + query->property, &query->res); > + if (ret < 0) > + continue; > + else > + break; > + } > + > + return ret; > +} > +EXPORT_SYMBOL_GPL(power_supply_get_external_attr); > + > static void power_supply_dev_release(struct device *dev) { > pr_debug("device: '%s': %s\n", dev_name(dev), __func__); @@ -289,6 > +320,53 @@ void power_supply_unregister(struct power_supply *psy) } > EXPORT_SYMBOL_GPL(power_supply_unregister); > > +int power_supply_attributes_register(struct device *parent, > + struct power_supply_attr_source *psy_attr) { > + struct device *dev; > + int rc; > + > + dev = kzalloc(sizeof(*dev), GFP_KERNEL); > + if (!dev) > + return -ENOMEM; > + > + device_initialize(dev); > + > + dev->parent = parent; > + dev->release = power_supply_dev_release; > + dev_set_drvdata(dev, psy_attr); > + psy_attr->dev = dev; > + > + rc = kobject_set_name(&dev->kobj, "%s", psy_attr->name); > + if (rc) > + goto kobject_set_name_failed; > + > + rc = device_add(dev); > + if (rc) > + goto device_add_failed; > + > + INIT_LIST_HEAD(&psy_attr->attr_pool); > + /* add to the list head */ > + list_add(&psy_attr->attr_pool, &list_head_source_attr); > + > + goto success; > + > +kobject_set_name_failed: > +device_add_failed: > + put_device(dev); > +success: > + return rc; > +} > +EXPORT_SYMBOL_GPL(power_supply_attributes_register); > + > +void power_supply_attributes_unregister(struct power_supply_attr_source > + *psy_attr) > +{ > + list_del(&psy_attr->attr_pool); > + device_unregister(psy_attr->dev); > +} > +EXPORT_SYMBOL_GPL(power_supply_attributes_unregister); > + > static int __init power_supply_class_init(void) { > power_supply_class = class_create(THIS_MODULE, "power_supply"); > diff --git a/include/linux/power_supply.h b/include/linux/power_supply.h index > 53f177d..af296c3 100644 > --- a/include/linux/power_supply.h > +++ b/include/linux/power_supply.h > @@ -193,6 +193,25 @@ struct power_supply { #endif }; > > +struct power_supply_attr_query { > + enum power_supply_property property; > + enum power_supply_type type; > + /* variable to store result */ > + union power_supply_propval res; > +}; > + > +struct power_supply_attr_source { > + const char *name; > + enum power_supply_type type; > + int (*get_property)(struct power_supply_attr_source *psy_attr, > + enum power_supply_property psp, > + union power_supply_propval *val); > + > + /* private */ > + struct device *dev; > + struct list_head attr_pool; > +}; > + > /* > * This is recommended structure to specify static power supply parameters. > * Generic one, parametrizable for different power supplies. Power supply @@ > - > 216,6 +235,8 @@ extern struct power_supply > *power_supply_get_by_name(char *name); extern void > power_supply_changed(struct power_supply *psy); extern int > power_supply_am_i_supplied(struct power_supply *psy); extern int > power_supply_set_battery_charged(struct power_supply *psy); > +extern int power_supply_get_external_attr( > + struct power_supply_attr_query *query); > > #ifdef CONFIG_POWER_SUPPLY > extern int power_supply_is_system_supplied(void); > @@ -226,6 +247,13 @@ static inline int power_supply_is_system_supplied(void) > { return -ENOSYS; } extern int power_supply_register(struct device *parent, > struct power_supply *psy); > extern void power_supply_unregister(struct power_supply *psy); > + > +extern int power_supply_attributes_register(struct device *parent, > + struct power_supply_attr_source *psy_attr); > + > +extern void power_supply_attributes_unregister( > + struct power_supply_attr_source *psy_attr); > + > extern int power_supply_powers(struct power_supply *psy, struct device *dev); > > /* For APM emulation, think legacy userspace. */ > -- > 1.7.0.4 -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/