Skip to content

Commit 6037802

Browse files
t-8chsre
authored andcommitted
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 <[email protected]> Reviewed-by: Armin Wolf <[email protected]> Link: https://lore.kernel.org/r/20241211-power-supply-extensions-v6-1-9d9dc3f3d387@weissschuh.net Signed-off-by: Sebastian Reichel <[email protected]>
1 parent 57e5a9a commit 6037802

File tree

4 files changed

+228
-10
lines changed

4 files changed

+228
-10
lines changed

drivers/power/supply/power_supply.h

+17
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@
99
* Modified: 2004, Oct Szabolcs Gyurko
1010
*/
1111

12+
#include <linux/lockdep.h>
13+
1214
struct device;
1315
struct device_type;
1416
struct power_supply;
@@ -17,6 +19,21 @@ extern int power_supply_property_is_writeable(struct power_supply *psy,
1719
enum power_supply_property psp);
1820
extern bool power_supply_has_property(struct power_supply *psy,
1921
enum power_supply_property psp);
22+
extern bool power_supply_ext_has_property(const struct power_supply_ext *ext,
23+
enum power_supply_property psp);
24+
25+
struct power_supply_ext_registration {
26+
struct list_head list_head;
27+
const struct power_supply_ext *ext;
28+
void *data;
29+
};
30+
31+
/* Make sure that the macro is a single expression */
32+
#define power_supply_for_each_extension(pos, psy) \
33+
if ( ({ lockdep_assert_held(&(psy)->extensions_sem); 0; }) ) \
34+
; \
35+
else \
36+
list_for_each_entry(pos, &(psy)->extensions, list_head) \
2037

2138
#ifdef CONFIG_SYSFS
2239

drivers/power/supply/power_supply_core.c

+154-8
Original file line numberDiff line numberDiff line change
@@ -80,13 +80,24 @@ static int __power_supply_changed_work(struct power_supply *pst, void *data)
8080

8181
static void power_supply_changed_work(struct work_struct *work)
8282
{
83+
int ret;
8384
unsigned long flags;
8485
struct power_supply *psy = container_of(work, struct power_supply,
8586
changed_work);
8687

8788
dev_dbg(&psy->dev, "%s\n", __func__);
8889

8990
spin_lock_irqsave(&psy->changed_lock, flags);
91+
92+
if (unlikely(psy->update_groups)) {
93+
psy->update_groups = false;
94+
spin_unlock_irqrestore(&psy->changed_lock, flags);
95+
ret = sysfs_update_groups(&psy->dev.kobj, power_supply_dev_type.groups);
96+
if (ret)
97+
dev_warn(&psy->dev, "failed to update sysfs groups: %pe\n", ERR_PTR(ret));
98+
spin_lock_irqsave(&psy->changed_lock, flags);
99+
}
100+
90101
/*
91102
* Check 'changed' here to avoid issues due to race between
92103
* power_supply_changed() and this routine. In worst case
@@ -1208,28 +1219,56 @@ static bool psy_desc_has_property(const struct power_supply_desc *psy_desc,
12081219
return found;
12091220
}
12101221

1222+
bool power_supply_ext_has_property(const struct power_supply_ext *psy_ext,
1223+
enum power_supply_property psp)
1224+
{
1225+
int i;
1226+
1227+
for (i = 0; i < psy_ext->num_properties; i++)
1228+
if (psy_ext->properties[i] == psp)
1229+
return true;
1230+
1231+
return false;
1232+
}
1233+
12111234
bool power_supply_has_property(struct power_supply *psy,
12121235
enum power_supply_property psp)
12131236
{
1237+
struct power_supply_ext_registration *reg;
1238+
12141239
if (psy_desc_has_property(psy->desc, psp))
12151240
return true;
12161241

12171242
if (power_supply_battery_info_has_prop(psy->battery_info, psp))
12181243
return true;
12191244

1245+
power_supply_for_each_extension(reg, psy) {
1246+
if (power_supply_ext_has_property(reg->ext, psp))
1247+
return true;
1248+
}
1249+
12201250
return false;
12211251
}
12221252

12231253
int power_supply_get_property(struct power_supply *psy,
12241254
enum power_supply_property psp,
12251255
union power_supply_propval *val)
12261256
{
1257+
struct power_supply_ext_registration *reg;
1258+
12271259
if (atomic_read(&psy->use_cnt) <= 0) {
12281260
if (!psy->initialized)
12291261
return -EAGAIN;
12301262
return -ENODEV;
12311263
}
12321264

1265+
scoped_guard(rwsem_read, &psy->extensions_sem) {
1266+
power_supply_for_each_extension(reg, psy) {
1267+
if (power_supply_ext_has_property(reg->ext, psp))
1268+
return reg->ext->get_property(psy, reg->ext, reg->data, psp, val);
1269+
}
1270+
}
1271+
12331272
if (psy_desc_has_property(psy->desc, psp))
12341273
return psy->desc->get_property(psy, psp, val);
12351274
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,
12431282
enum power_supply_property psp,
12441283
const union power_supply_propval *val)
12451284
{
1246-
if (atomic_read(&psy->use_cnt) <= 0 || !psy->desc->set_property)
1285+
struct power_supply_ext_registration *reg;
1286+
1287+
if (atomic_read(&psy->use_cnt) <= 0)
1288+
return -ENODEV;
1289+
1290+
scoped_guard(rwsem_read, &psy->extensions_sem) {
1291+
power_supply_for_each_extension(reg, psy) {
1292+
if (power_supply_ext_has_property(reg->ext, psp)) {
1293+
if (reg->ext->set_property)
1294+
return reg->ext->set_property(psy, reg->ext, reg->data,
1295+
psp, val);
1296+
else
1297+
return -ENODEV;
1298+
}
1299+
}
1300+
}
1301+
1302+
if (!psy->desc->set_property)
12471303
return -ENODEV;
12481304

12491305
return psy->desc->set_property(psy, psp, val);
@@ -1253,7 +1309,22 @@ EXPORT_SYMBOL_GPL(power_supply_set_property);
12531309
int power_supply_property_is_writeable(struct power_supply *psy,
12541310
enum power_supply_property psp)
12551311
{
1256-
return psy->desc->property_is_writeable && psy->desc->property_is_writeable(psy, psp);
1312+
struct power_supply_ext_registration *reg;
1313+
1314+
power_supply_for_each_extension(reg, psy) {
1315+
if (power_supply_ext_has_property(reg->ext, psp)) {
1316+
if (reg->ext->property_is_writeable)
1317+
return reg->ext->property_is_writeable(psy, reg->ext,
1318+
reg->data, psp);
1319+
else
1320+
return 0;
1321+
}
1322+
}
1323+
1324+
if (!psy->desc->property_is_writeable)
1325+
return 0;
1326+
1327+
return psy->desc->property_is_writeable(psy, psp);
12571328
}
12581329

12591330
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)
12721343
}
12731344
EXPORT_SYMBOL_GPL(power_supply_powers);
12741345

1346+
static int power_supply_update_sysfs_and_hwmon(struct power_supply *psy)
1347+
{
1348+
unsigned long flags;
1349+
1350+
spin_lock_irqsave(&psy->changed_lock, flags);
1351+
psy->update_groups = true;
1352+
spin_unlock_irqrestore(&psy->changed_lock, flags);
1353+
1354+
power_supply_changed(psy);
1355+
1356+
power_supply_remove_hwmon_sysfs(psy);
1357+
return power_supply_add_hwmon_sysfs(psy);
1358+
}
1359+
1360+
int power_supply_register_extension(struct power_supply *psy, const struct power_supply_ext *ext,
1361+
void *data)
1362+
{
1363+
struct power_supply_ext_registration *reg;
1364+
size_t i;
1365+
int ret;
1366+
1367+
if (!psy || !ext || !ext->properties || !ext->num_properties)
1368+
return -EINVAL;
1369+
1370+
guard(rwsem_write)(&psy->extensions_sem);
1371+
1372+
for (i = 0; i < ext->num_properties; i++)
1373+
if (power_supply_has_property(psy, ext->properties[i]))
1374+
return -EEXIST;
1375+
1376+
reg = kmalloc(sizeof(*reg), GFP_KERNEL);
1377+
if (!reg)
1378+
return -ENOMEM;
1379+
1380+
reg->ext = ext;
1381+
reg->data = data;
1382+
list_add(&reg->list_head, &psy->extensions);
1383+
1384+
ret = power_supply_update_sysfs_and_hwmon(psy);
1385+
if (ret)
1386+
goto sysfs_hwmon_failed;
1387+
1388+
return 0;
1389+
1390+
sysfs_hwmon_failed:
1391+
list_del(&reg->list_head);
1392+
kfree(reg);
1393+
return ret;
1394+
}
1395+
EXPORT_SYMBOL_GPL(power_supply_register_extension);
1396+
1397+
void power_supply_unregister_extension(struct power_supply *psy, const struct power_supply_ext *ext)
1398+
{
1399+
struct power_supply_ext_registration *reg;
1400+
1401+
guard(rwsem_write)(&psy->extensions_sem);
1402+
1403+
power_supply_for_each_extension(reg, psy) {
1404+
if (reg->ext == ext) {
1405+
list_del(&reg->list_head);
1406+
kfree(reg);
1407+
power_supply_update_sysfs_and_hwmon(psy);
1408+
return;
1409+
}
1410+
}
1411+
1412+
dev_warn(&psy->dev, "Trying to unregister invalid extension");
1413+
}
1414+
EXPORT_SYMBOL_GPL(power_supply_unregister_extension);
1415+
12751416
static void power_supply_dev_release(struct device *dev)
12761417
{
12771418
struct power_supply *psy = to_power_supply(dev);
@@ -1426,6 +1567,9 @@ __power_supply_register(struct device *parent,
14261567
}
14271568

14281569
spin_lock_init(&psy->changed_lock);
1570+
init_rwsem(&psy->extensions_sem);
1571+
INIT_LIST_HEAD(&psy->extensions);
1572+
14291573
rc = device_add(dev);
14301574
if (rc)
14311575
goto device_add_failed;
@@ -1438,13 +1582,15 @@ __power_supply_register(struct device *parent,
14381582
if (rc)
14391583
goto register_thermal_failed;
14401584

1441-
rc = power_supply_create_triggers(psy);
1442-
if (rc)
1443-
goto create_triggers_failed;
1585+
scoped_guard(rwsem_read, &psy->extensions_sem) {
1586+
rc = power_supply_create_triggers(psy);
1587+
if (rc)
1588+
goto create_triggers_failed;
14441589

1445-
rc = power_supply_add_hwmon_sysfs(psy);
1446-
if (rc)
1447-
goto add_hwmon_sysfs_failed;
1590+
rc = power_supply_add_hwmon_sysfs(psy);
1591+
if (rc)
1592+
goto add_hwmon_sysfs_failed;
1593+
}
14481594

14491595
/*
14501596
* Update use_cnt after any uevents (most notably from device_add()).

drivers/power/supply/power_supply_sysfs.c

+24-2
Original file line numberDiff line numberDiff line change
@@ -299,6 +299,27 @@ static ssize_t power_supply_show_enum_with_available(
299299
return count;
300300
}
301301

302+
static ssize_t power_supply_show_charge_behaviour(struct device *dev,
303+
struct power_supply *psy,
304+
union power_supply_propval *value,
305+
char *buf)
306+
{
307+
struct power_supply_ext_registration *reg;
308+
309+
scoped_guard(rwsem_read, &psy->extensions_sem) {
310+
power_supply_for_each_extension(reg, psy) {
311+
if (power_supply_ext_has_property(reg->ext,
312+
POWER_SUPPLY_PROP_CHARGE_BEHAVIOUR))
313+
return power_supply_charge_behaviour_show(dev,
314+
reg->ext->charge_behaviours,
315+
value->intval, buf);
316+
}
317+
}
318+
319+
return power_supply_charge_behaviour_show(dev, psy->desc->charge_behaviours,
320+
value->intval, buf);
321+
}
322+
302323
static ssize_t power_supply_format_property(struct device *dev,
303324
bool uevent,
304325
struct device_attribute *attr,
@@ -338,8 +359,7 @@ static ssize_t power_supply_format_property(struct device *dev,
338359
case POWER_SUPPLY_PROP_CHARGE_BEHAVIOUR:
339360
if (uevent) /* no possible values in uevents */
340361
goto default_format;
341-
ret = power_supply_charge_behaviour_show(dev, psy->desc->charge_behaviours,
342-
value.intval, buf);
362+
ret = power_supply_show_charge_behaviour(dev, psy, &value, buf);
343363
break;
344364
case POWER_SUPPLY_PROP_CHARGE_TYPES:
345365
if (uevent) /* no possible values in uevents */
@@ -422,6 +442,8 @@ static umode_t power_supply_attr_is_visible(struct kobject *kobj,
422442
if (attrno == POWER_SUPPLY_PROP_TYPE)
423443
return mode;
424444

445+
guard(rwsem_read)(&psy->extensions_sem);
446+
425447
if (power_supply_has_property(psy, attrno)) {
426448
if (power_supply_property_is_writeable(psy, attrno) > 0)
427449
mode |= S_IWUSR;

include/linux/power_supply.h

+33
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@
1515
#include <linux/device.h>
1616
#include <linux/workqueue.h>
1717
#include <linux/leds.h>
18+
#include <linux/rwsem.h>
19+
#include <linux/list.h>
1820
#include <linux/spinlock.h>
1921
#include <linux/notifier.h>
2022

@@ -283,6 +285,27 @@ struct power_supply_desc {
283285
int use_for_apm;
284286
};
285287

288+
struct power_supply_ext {
289+
u8 charge_behaviours;
290+
const enum power_supply_property *properties;
291+
size_t num_properties;
292+
293+
int (*get_property)(struct power_supply *psy,
294+
const struct power_supply_ext *ext,
295+
void *data,
296+
enum power_supply_property psp,
297+
union power_supply_propval *val);
298+
int (*set_property)(struct power_supply *psy,
299+
const struct power_supply_ext *ext,
300+
void *data,
301+
enum power_supply_property psp,
302+
const union power_supply_propval *val);
303+
int (*property_is_writeable)(struct power_supply *psy,
304+
const struct power_supply_ext *ext,
305+
void *data,
306+
enum power_supply_property psp);
307+
};
308+
286309
struct power_supply {
287310
const struct power_supply_desc *desc;
288311

@@ -302,10 +325,13 @@ struct power_supply {
302325
struct delayed_work deferred_register_work;
303326
spinlock_t changed_lock;
304327
bool changed;
328+
bool update_groups;
305329
bool initialized;
306330
bool removing;
307331
atomic_t use_cnt;
308332
struct power_supply_battery_info *battery_info;
333+
struct rw_semaphore extensions_sem; /* protects "extensions" */
334+
struct list_head extensions;
309335
#ifdef CONFIG_THERMAL
310336
struct thermal_zone_device *tzd;
311337
struct thermal_cooling_device *tcd;
@@ -882,6 +908,13 @@ devm_power_supply_register(struct device *parent,
882908
extern void power_supply_unregister(struct power_supply *psy);
883909
extern int power_supply_powers(struct power_supply *psy, struct device *dev);
884910

911+
extern int __must_check
912+
power_supply_register_extension(struct power_supply *psy,
913+
const struct power_supply_ext *ext,
914+
void *data);
915+
extern void power_supply_unregister_extension(struct power_supply *psy,
916+
const struct power_supply_ext *ext);
917+
885918
#define to_power_supply(device) container_of(device, struct power_supply, dev)
886919

887920
extern void *power_supply_get_drvdata(struct power_supply *psy);

0 commit comments

Comments
 (0)