power: supply: core: implement extension API
Various drivers, mostly in platform/x86 extend the ACPI battery driver with additional sysfs attributes to implement more UAPIs than are exposed through ACPI by using various side-channels, like WMI, nonstandard ACPI or EC communication. While the created sysfs attributes look similar to the attributes provided by the powersupply core, there are various deficiencies: * They don't show up in uevent payload. * They can't be queried with the standard in-kernel APIs. * They don't work with triggers. * The extending driver has to reimplement all of the parsing, formatting and sysfs display logic. * Writing a extension driver is completely different from writing a normal power supply driver. This extension API avoids all of these issues. An extension is just a "struct power_supply_ext" with the same kind of callbacks as in a normal "struct power_supply_desc". The API is meant to be used via battery_hook_register(), the same way as the current extensions. Signed-off-by: Thomas Weißschuh <linux@weissschuh.net> Reviewed-by: Armin Wolf <W_Armin@gmx.de> Link: https://lore.kernel.org/r/20241211-power-supply-extensions-v6-1-9d9dc3f3d387@weissschuh.net Signed-off-by: Sebastian Reichel <sebastian.reichel@collabora.com>
This commit is contained in:
parent
57e5a9a85b
commit
6037802bba
4 changed files with 228 additions and 10 deletions
|
@ -9,6 +9,8 @@
|
||||||
* Modified: 2004, Oct Szabolcs Gyurko
|
* Modified: 2004, Oct Szabolcs Gyurko
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <linux/lockdep.h>
|
||||||
|
|
||||||
struct device;
|
struct device;
|
||||||
struct device_type;
|
struct device_type;
|
||||||
struct power_supply;
|
struct power_supply;
|
||||||
|
@ -17,6 +19,21 @@ extern int power_supply_property_is_writeable(struct power_supply *psy,
|
||||||
enum power_supply_property psp);
|
enum power_supply_property psp);
|
||||||
extern bool power_supply_has_property(struct power_supply *psy,
|
extern bool power_supply_has_property(struct power_supply *psy,
|
||||||
enum power_supply_property psp);
|
enum power_supply_property psp);
|
||||||
|
extern bool power_supply_ext_has_property(const struct power_supply_ext *ext,
|
||||||
|
enum power_supply_property psp);
|
||||||
|
|
||||||
|
struct power_supply_ext_registration {
|
||||||
|
struct list_head list_head;
|
||||||
|
const struct power_supply_ext *ext;
|
||||||
|
void *data;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Make sure that the macro is a single expression */
|
||||||
|
#define power_supply_for_each_extension(pos, psy) \
|
||||||
|
if ( ({ lockdep_assert_held(&(psy)->extensions_sem); 0; }) ) \
|
||||||
|
; \
|
||||||
|
else \
|
||||||
|
list_for_each_entry(pos, &(psy)->extensions, list_head) \
|
||||||
|
|
||||||
#ifdef CONFIG_SYSFS
|
#ifdef CONFIG_SYSFS
|
||||||
|
|
||||||
|
|
|
@ -80,6 +80,7 @@ static int __power_supply_changed_work(struct power_supply *pst, void *data)
|
||||||
|
|
||||||
static void power_supply_changed_work(struct work_struct *work)
|
static void power_supply_changed_work(struct work_struct *work)
|
||||||
{
|
{
|
||||||
|
int ret;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
struct power_supply *psy = container_of(work, struct power_supply,
|
struct power_supply *psy = container_of(work, struct power_supply,
|
||||||
changed_work);
|
changed_work);
|
||||||
|
@ -87,6 +88,16 @@ static void power_supply_changed_work(struct work_struct *work)
|
||||||
dev_dbg(&psy->dev, "%s\n", __func__);
|
dev_dbg(&psy->dev, "%s\n", __func__);
|
||||||
|
|
||||||
spin_lock_irqsave(&psy->changed_lock, flags);
|
spin_lock_irqsave(&psy->changed_lock, flags);
|
||||||
|
|
||||||
|
if (unlikely(psy->update_groups)) {
|
||||||
|
psy->update_groups = false;
|
||||||
|
spin_unlock_irqrestore(&psy->changed_lock, flags);
|
||||||
|
ret = sysfs_update_groups(&psy->dev.kobj, power_supply_dev_type.groups);
|
||||||
|
if (ret)
|
||||||
|
dev_warn(&psy->dev, "failed to update sysfs groups: %pe\n", ERR_PTR(ret));
|
||||||
|
spin_lock_irqsave(&psy->changed_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Check 'changed' here to avoid issues due to race between
|
* Check 'changed' here to avoid issues due to race between
|
||||||
* power_supply_changed() and this routine. In worst case
|
* power_supply_changed() and this routine. In worst case
|
||||||
|
@ -1208,15 +1219,34 @@ static bool psy_desc_has_property(const struct power_supply_desc *psy_desc,
|
||||||
return found;
|
return found;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool power_supply_ext_has_property(const struct power_supply_ext *psy_ext,
|
||||||
|
enum power_supply_property psp)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < psy_ext->num_properties; i++)
|
||||||
|
if (psy_ext->properties[i] == psp)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool power_supply_has_property(struct power_supply *psy,
|
bool power_supply_has_property(struct power_supply *psy,
|
||||||
enum power_supply_property psp)
|
enum power_supply_property psp)
|
||||||
{
|
{
|
||||||
|
struct power_supply_ext_registration *reg;
|
||||||
|
|
||||||
if (psy_desc_has_property(psy->desc, psp))
|
if (psy_desc_has_property(psy->desc, psp))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (power_supply_battery_info_has_prop(psy->battery_info, psp))
|
if (power_supply_battery_info_has_prop(psy->battery_info, psp))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
power_supply_for_each_extension(reg, psy) {
|
||||||
|
if (power_supply_ext_has_property(reg->ext, psp))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1224,12 +1254,21 @@ int power_supply_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property psp,
|
enum power_supply_property psp,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
|
struct power_supply_ext_registration *reg;
|
||||||
|
|
||||||
if (atomic_read(&psy->use_cnt) <= 0) {
|
if (atomic_read(&psy->use_cnt) <= 0) {
|
||||||
if (!psy->initialized)
|
if (!psy->initialized)
|
||||||
return -EAGAIN;
|
return -EAGAIN;
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
scoped_guard(rwsem_read, &psy->extensions_sem) {
|
||||||
|
power_supply_for_each_extension(reg, psy) {
|
||||||
|
if (power_supply_ext_has_property(reg->ext, psp))
|
||||||
|
return reg->ext->get_property(psy, reg->ext, reg->data, psp, val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (psy_desc_has_property(psy->desc, psp))
|
if (psy_desc_has_property(psy->desc, psp))
|
||||||
return psy->desc->get_property(psy, psp, val);
|
return psy->desc->get_property(psy, psp, val);
|
||||||
else if (power_supply_battery_info_has_prop(psy->battery_info, psp))
|
else if (power_supply_battery_info_has_prop(psy->battery_info, psp))
|
||||||
|
@ -1243,7 +1282,24 @@ int power_supply_set_property(struct power_supply *psy,
|
||||||
enum power_supply_property psp,
|
enum power_supply_property psp,
|
||||||
const union power_supply_propval *val)
|
const union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
if (atomic_read(&psy->use_cnt) <= 0 || !psy->desc->set_property)
|
struct power_supply_ext_registration *reg;
|
||||||
|
|
||||||
|
if (atomic_read(&psy->use_cnt) <= 0)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
scoped_guard(rwsem_read, &psy->extensions_sem) {
|
||||||
|
power_supply_for_each_extension(reg, psy) {
|
||||||
|
if (power_supply_ext_has_property(reg->ext, psp)) {
|
||||||
|
if (reg->ext->set_property)
|
||||||
|
return reg->ext->set_property(psy, reg->ext, reg->data,
|
||||||
|
psp, val);
|
||||||
|
else
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!psy->desc->set_property)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
return psy->desc->set_property(psy, psp, val);
|
return psy->desc->set_property(psy, psp, val);
|
||||||
|
@ -1253,7 +1309,22 @@ EXPORT_SYMBOL_GPL(power_supply_set_property);
|
||||||
int power_supply_property_is_writeable(struct power_supply *psy,
|
int power_supply_property_is_writeable(struct power_supply *psy,
|
||||||
enum power_supply_property psp)
|
enum power_supply_property psp)
|
||||||
{
|
{
|
||||||
return psy->desc->property_is_writeable && psy->desc->property_is_writeable(psy, psp);
|
struct power_supply_ext_registration *reg;
|
||||||
|
|
||||||
|
power_supply_for_each_extension(reg, psy) {
|
||||||
|
if (power_supply_ext_has_property(reg->ext, psp)) {
|
||||||
|
if (reg->ext->property_is_writeable)
|
||||||
|
return reg->ext->property_is_writeable(psy, reg->ext,
|
||||||
|
reg->data, psp);
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!psy->desc->property_is_writeable)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return psy->desc->property_is_writeable(psy, psp);
|
||||||
}
|
}
|
||||||
|
|
||||||
void power_supply_external_power_changed(struct power_supply *psy)
|
void power_supply_external_power_changed(struct power_supply *psy)
|
||||||
|
@ -1272,6 +1343,76 @@ int power_supply_powers(struct power_supply *psy, struct device *dev)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(power_supply_powers);
|
EXPORT_SYMBOL_GPL(power_supply_powers);
|
||||||
|
|
||||||
|
static int power_supply_update_sysfs_and_hwmon(struct power_supply *psy)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&psy->changed_lock, flags);
|
||||||
|
psy->update_groups = true;
|
||||||
|
spin_unlock_irqrestore(&psy->changed_lock, flags);
|
||||||
|
|
||||||
|
power_supply_changed(psy);
|
||||||
|
|
||||||
|
power_supply_remove_hwmon_sysfs(psy);
|
||||||
|
return power_supply_add_hwmon_sysfs(psy);
|
||||||
|
}
|
||||||
|
|
||||||
|
int power_supply_register_extension(struct power_supply *psy, const struct power_supply_ext *ext,
|
||||||
|
void *data)
|
||||||
|
{
|
||||||
|
struct power_supply_ext_registration *reg;
|
||||||
|
size_t i;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (!psy || !ext || !ext->properties || !ext->num_properties)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
guard(rwsem_write)(&psy->extensions_sem);
|
||||||
|
|
||||||
|
for (i = 0; i < ext->num_properties; i++)
|
||||||
|
if (power_supply_has_property(psy, ext->properties[i]))
|
||||||
|
return -EEXIST;
|
||||||
|
|
||||||
|
reg = kmalloc(sizeof(*reg), GFP_KERNEL);
|
||||||
|
if (!reg)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
reg->ext = ext;
|
||||||
|
reg->data = data;
|
||||||
|
list_add(®->list_head, &psy->extensions);
|
||||||
|
|
||||||
|
ret = power_supply_update_sysfs_and_hwmon(psy);
|
||||||
|
if (ret)
|
||||||
|
goto sysfs_hwmon_failed;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
sysfs_hwmon_failed:
|
||||||
|
list_del(®->list_head);
|
||||||
|
kfree(reg);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_register_extension);
|
||||||
|
|
||||||
|
void power_supply_unregister_extension(struct power_supply *psy, const struct power_supply_ext *ext)
|
||||||
|
{
|
||||||
|
struct power_supply_ext_registration *reg;
|
||||||
|
|
||||||
|
guard(rwsem_write)(&psy->extensions_sem);
|
||||||
|
|
||||||
|
power_supply_for_each_extension(reg, psy) {
|
||||||
|
if (reg->ext == ext) {
|
||||||
|
list_del(®->list_head);
|
||||||
|
kfree(reg);
|
||||||
|
power_supply_update_sysfs_and_hwmon(psy);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dev_warn(&psy->dev, "Trying to unregister invalid extension");
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_unregister_extension);
|
||||||
|
|
||||||
static void power_supply_dev_release(struct device *dev)
|
static void power_supply_dev_release(struct device *dev)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = to_power_supply(dev);
|
struct power_supply *psy = to_power_supply(dev);
|
||||||
|
@ -1426,6 +1567,9 @@ __power_supply_register(struct device *parent,
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock_init(&psy->changed_lock);
|
spin_lock_init(&psy->changed_lock);
|
||||||
|
init_rwsem(&psy->extensions_sem);
|
||||||
|
INIT_LIST_HEAD(&psy->extensions);
|
||||||
|
|
||||||
rc = device_add(dev);
|
rc = device_add(dev);
|
||||||
if (rc)
|
if (rc)
|
||||||
goto device_add_failed;
|
goto device_add_failed;
|
||||||
|
@ -1438,13 +1582,15 @@ __power_supply_register(struct device *parent,
|
||||||
if (rc)
|
if (rc)
|
||||||
goto register_thermal_failed;
|
goto register_thermal_failed;
|
||||||
|
|
||||||
rc = power_supply_create_triggers(psy);
|
scoped_guard(rwsem_read, &psy->extensions_sem) {
|
||||||
if (rc)
|
rc = power_supply_create_triggers(psy);
|
||||||
goto create_triggers_failed;
|
if (rc)
|
||||||
|
goto create_triggers_failed;
|
||||||
|
|
||||||
rc = power_supply_add_hwmon_sysfs(psy);
|
rc = power_supply_add_hwmon_sysfs(psy);
|
||||||
if (rc)
|
if (rc)
|
||||||
goto add_hwmon_sysfs_failed;
|
goto add_hwmon_sysfs_failed;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Update use_cnt after any uevents (most notably from device_add()).
|
* Update use_cnt after any uevents (most notably from device_add()).
|
||||||
|
|
|
@ -299,6 +299,27 @@ static ssize_t power_supply_show_enum_with_available(
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ssize_t power_supply_show_charge_behaviour(struct device *dev,
|
||||||
|
struct power_supply *psy,
|
||||||
|
union power_supply_propval *value,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct power_supply_ext_registration *reg;
|
||||||
|
|
||||||
|
scoped_guard(rwsem_read, &psy->extensions_sem) {
|
||||||
|
power_supply_for_each_extension(reg, psy) {
|
||||||
|
if (power_supply_ext_has_property(reg->ext,
|
||||||
|
POWER_SUPPLY_PROP_CHARGE_BEHAVIOUR))
|
||||||
|
return power_supply_charge_behaviour_show(dev,
|
||||||
|
reg->ext->charge_behaviours,
|
||||||
|
value->intval, buf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return power_supply_charge_behaviour_show(dev, psy->desc->charge_behaviours,
|
||||||
|
value->intval, buf);
|
||||||
|
}
|
||||||
|
|
||||||
static ssize_t power_supply_format_property(struct device *dev,
|
static ssize_t power_supply_format_property(struct device *dev,
|
||||||
bool uevent,
|
bool uevent,
|
||||||
struct device_attribute *attr,
|
struct device_attribute *attr,
|
||||||
|
@ -338,8 +359,7 @@ static ssize_t power_supply_format_property(struct device *dev,
|
||||||
case POWER_SUPPLY_PROP_CHARGE_BEHAVIOUR:
|
case POWER_SUPPLY_PROP_CHARGE_BEHAVIOUR:
|
||||||
if (uevent) /* no possible values in uevents */
|
if (uevent) /* no possible values in uevents */
|
||||||
goto default_format;
|
goto default_format;
|
||||||
ret = power_supply_charge_behaviour_show(dev, psy->desc->charge_behaviours,
|
ret = power_supply_show_charge_behaviour(dev, psy, &value, buf);
|
||||||
value.intval, buf);
|
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_CHARGE_TYPES:
|
case POWER_SUPPLY_PROP_CHARGE_TYPES:
|
||||||
if (uevent) /* no possible values in uevents */
|
if (uevent) /* no possible values in uevents */
|
||||||
|
@ -422,6 +442,8 @@ static umode_t power_supply_attr_is_visible(struct kobject *kobj,
|
||||||
if (attrno == POWER_SUPPLY_PROP_TYPE)
|
if (attrno == POWER_SUPPLY_PROP_TYPE)
|
||||||
return mode;
|
return mode;
|
||||||
|
|
||||||
|
guard(rwsem_read)(&psy->extensions_sem);
|
||||||
|
|
||||||
if (power_supply_has_property(psy, attrno)) {
|
if (power_supply_has_property(psy, attrno)) {
|
||||||
if (power_supply_property_is_writeable(psy, attrno) > 0)
|
if (power_supply_property_is_writeable(psy, attrno) > 0)
|
||||||
mode |= S_IWUSR;
|
mode |= S_IWUSR;
|
||||||
|
|
|
@ -15,6 +15,8 @@
|
||||||
#include <linux/device.h>
|
#include <linux/device.h>
|
||||||
#include <linux/workqueue.h>
|
#include <linux/workqueue.h>
|
||||||
#include <linux/leds.h>
|
#include <linux/leds.h>
|
||||||
|
#include <linux/rwsem.h>
|
||||||
|
#include <linux/list.h>
|
||||||
#include <linux/spinlock.h>
|
#include <linux/spinlock.h>
|
||||||
#include <linux/notifier.h>
|
#include <linux/notifier.h>
|
||||||
|
|
||||||
|
@ -283,6 +285,27 @@ struct power_supply_desc {
|
||||||
int use_for_apm;
|
int use_for_apm;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct power_supply_ext {
|
||||||
|
u8 charge_behaviours;
|
||||||
|
const enum power_supply_property *properties;
|
||||||
|
size_t num_properties;
|
||||||
|
|
||||||
|
int (*get_property)(struct power_supply *psy,
|
||||||
|
const struct power_supply_ext *ext,
|
||||||
|
void *data,
|
||||||
|
enum power_supply_property psp,
|
||||||
|
union power_supply_propval *val);
|
||||||
|
int (*set_property)(struct power_supply *psy,
|
||||||
|
const struct power_supply_ext *ext,
|
||||||
|
void *data,
|
||||||
|
enum power_supply_property psp,
|
||||||
|
const union power_supply_propval *val);
|
||||||
|
int (*property_is_writeable)(struct power_supply *psy,
|
||||||
|
const struct power_supply_ext *ext,
|
||||||
|
void *data,
|
||||||
|
enum power_supply_property psp);
|
||||||
|
};
|
||||||
|
|
||||||
struct power_supply {
|
struct power_supply {
|
||||||
const struct power_supply_desc *desc;
|
const struct power_supply_desc *desc;
|
||||||
|
|
||||||
|
@ -302,10 +325,13 @@ struct power_supply {
|
||||||
struct delayed_work deferred_register_work;
|
struct delayed_work deferred_register_work;
|
||||||
spinlock_t changed_lock;
|
spinlock_t changed_lock;
|
||||||
bool changed;
|
bool changed;
|
||||||
|
bool update_groups;
|
||||||
bool initialized;
|
bool initialized;
|
||||||
bool removing;
|
bool removing;
|
||||||
atomic_t use_cnt;
|
atomic_t use_cnt;
|
||||||
struct power_supply_battery_info *battery_info;
|
struct power_supply_battery_info *battery_info;
|
||||||
|
struct rw_semaphore extensions_sem; /* protects "extensions" */
|
||||||
|
struct list_head extensions;
|
||||||
#ifdef CONFIG_THERMAL
|
#ifdef CONFIG_THERMAL
|
||||||
struct thermal_zone_device *tzd;
|
struct thermal_zone_device *tzd;
|
||||||
struct thermal_cooling_device *tcd;
|
struct thermal_cooling_device *tcd;
|
||||||
|
@ -882,6 +908,13 @@ devm_power_supply_register(struct device *parent,
|
||||||
extern void power_supply_unregister(struct power_supply *psy);
|
extern void power_supply_unregister(struct power_supply *psy);
|
||||||
extern int power_supply_powers(struct power_supply *psy, struct device *dev);
|
extern int power_supply_powers(struct power_supply *psy, struct device *dev);
|
||||||
|
|
||||||
|
extern int __must_check
|
||||||
|
power_supply_register_extension(struct power_supply *psy,
|
||||||
|
const struct power_supply_ext *ext,
|
||||||
|
void *data);
|
||||||
|
extern void power_supply_unregister_extension(struct power_supply *psy,
|
||||||
|
const struct power_supply_ext *ext);
|
||||||
|
|
||||||
#define to_power_supply(device) container_of(device, struct power_supply, dev)
|
#define to_power_supply(device) container_of(device, struct power_supply, dev)
|
||||||
|
|
||||||
extern void *power_supply_get_drvdata(struct power_supply *psy);
|
extern void *power_supply_get_drvdata(struct power_supply *psy);
|
||||||
|
|
Loading…
Add table
Reference in a new issue