power supply and reset changes for the v4.1 series
* new API for safe access of power supply function attrs * devres support for power supply (un)registration * new drivers / chips - generic syscon based poweroff driver - iio & charger driver for da9150 - fuel gauge driver for axp288 - bq27x00: add support for bq27510 - bq2415x: add support for bq24157s * twl4030-madc-battery: convert to iio consumer * misc fixes -----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQIcBAABCgAGBQJVK8KPAAoJENju1/PIO/qaKIIP/jYZr1NJ7zrSAfm/hiCS+1Kz f68H5zlC3IOD2YlH7Yx3EUnag4k8wXiwp5LlRFMmek78H6UtKgr274cVyBA730Qk qGYyI6rZqBi9nheIwbXugUTs65R+S1bSaLksZvNfTmnKRiHUpO5nsJJezusJM/O1 lSUJb166kR9agFL+7MSjOW4FTpqDCrndEmYOXiUmzpOS1+1ZJXPN8EMyktJxcZu9 fxE8smy2fr0rqkZoSX4irYI28DoAqb5jImv0/GlVLVKsGukzkOkwMdKunWbFywvT mg/mkyfpPb3xWNUPuyVRzhgsPnaNYm5uNwr2yWuFvyxNOTweyjRriGMflewkKscf aGJLOiFThJrdyTjscMCfdVuNr1a+Bzd9UdiNH1I6mS99zW7wLEukHJi2CH8+9UGE eDRrnb6wiKAqpvIDeIatEbQUTV5JXM0vWbKIHx7m1M3P5/hLGiOWguO7V2x4RsTp B21mXXj6/AUYA7IKIiPjNC7HJ71J80Gali40zOYRni9MFybi1SE9MzeHNdQhdbyn EW+3irBTruSl93qMpsCivxMhnSeclB/PuPvH6wPOfhftlAvN+9GhGaIQzLl0mo5E ckXbc23QfDCwcygCnkcQy+dgQOOttk6wDUkmhAiGriciO57NgTz9KnrhOexptw+h 2wCLGqBPSO2rADJ5RE9f =WbZu -----END PGP SIGNATURE----- Merge tag 'for-v4.1' of git://git.infradead.org/battery-2.6 Pull power supply and reset changes from Sebastian Reichel: - new API for safe access of power supply function attrs - devres support for power supply (un)registration - new drivers / chips: - generic syscon based poweroff driver - iio & charger driver for da9150 - fuel gauge driver for axp288 - bq27x00: add support for bq27510 - bq2415x: add support for bq24157s - twl4030-madc-battery: convert to iio consumer - misc fixes * tag 'for-v4.1' of git://git.infradead.org/battery-2.6: (66 commits) power: twl4030_madc_battery: Add missing MODULE_ALIAS power: twl4030-madc-battery: Convert to iio consumer. dt: power: Add docs for generic SYSCON poweroff driver. power: reset: Add generic SYSCON register mapped poweroff. power: max17042_battery: add missed blank power: max17042_battery: Use reg type instead of chip type power/reset: at91: big endian fixes for atsama5d3x power_supply: charger-manager: Fix dereferencing of ERR_PTR HID: input: Fix NULL pointer dereference when power_supply_register fails power: constify of_device_id array power/reset/rmobile-reset.c: Fix !HAS_IOMEM build power_supply: 88pm860x_charger: Fix possible NULL pointer dereference and use of initialized variable arm: mach-pxa: Decrement the power supply's device reference counter mfd: ab8500: Decrement the power supply's device reference counter power_supply: bq2415x_charger: Decrement the power supply's device reference counter power_supply: 88pm860x_charger: Decrement the power supply's device reference counter x86/olpc/xo15/sci: Use newly added power_supply_put API x86/olpc/xo1/sci: Use newly added power_supply_put API power_supply: charger-manager: Decrement the power supply's device reference counter power_supply: Increment power supply use counter when obtaining references ...
This commit is contained in:
commit
a21c1ea656
106 changed files with 5429 additions and 2098 deletions
16
Documentation/devicetree/bindings/iio/adc/da9150-gpadc.txt
Normal file
16
Documentation/devicetree/bindings/iio/adc/da9150-gpadc.txt
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
Dialog Semiconductor DA9150 IIO GPADC bindings
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible: "dlg,da9150-gpadc" for DA9150 IIO GPADC
|
||||||
|
- #io-channel-cells: Should be set to <1>
|
||||||
|
(See Documentation/devicetree/bindings/iio/iio-bindings.txt for further info)
|
||||||
|
|
||||||
|
For further information on GPADC channels, see device datasheet.
|
||||||
|
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
gpadc: da9150-gpadc {
|
||||||
|
compatible = "dlg,da9150-gpadc";
|
||||||
|
#io-channel-cells = <1>;
|
||||||
|
};
|
26
Documentation/devicetree/bindings/power/da9150-charger.txt
Normal file
26
Documentation/devicetree/bindings/power/da9150-charger.txt
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
Dialog Semiconductor DA9150 Charger Power Supply bindings
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible: "dlg,da9150-charger" for DA9150 Charger Power Supply
|
||||||
|
|
||||||
|
Optional properties:
|
||||||
|
- io-channels: List of phandle and IIO specifier pairs
|
||||||
|
- io-channel-names: List of channel names used by charger
|
||||||
|
["CHAN_IBUS", "CHAN_VBUS", "CHAN_TJUNC", "CHAN_VBAT"]
|
||||||
|
(See Documentation/devicetree/bindings/iio/iio-bindings.txt for further info)
|
||||||
|
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
da9150-charger {
|
||||||
|
compatible = "dlg,da9150-charger";
|
||||||
|
|
||||||
|
io-channels = <&gpadc 0>,
|
||||||
|
<&gpadc 2>,
|
||||||
|
<&gpadc 8>,
|
||||||
|
<&gpadc 5>;
|
||||||
|
io-channel-names = "CHAN_IBUS",
|
||||||
|
"CHAN_VBUS",
|
||||||
|
"CHAN_TJUNC",
|
||||||
|
"CHAN_VBAT";
|
||||||
|
};
|
|
@ -0,0 +1,23 @@
|
||||||
|
Generic SYSCON mapped register poweroff driver
|
||||||
|
|
||||||
|
This is a generic poweroff driver using syscon to map the poweroff register.
|
||||||
|
The poweroff is generally performed with a write to the poweroff register
|
||||||
|
defined by the register map pointed by syscon reference plus the offset
|
||||||
|
with the mask defined in the poweroff node.
|
||||||
|
|
||||||
|
Required properties:
|
||||||
|
- compatible: should contain "syscon-poweroff"
|
||||||
|
- regmap: this is phandle to the register map node
|
||||||
|
- offset: offset in the register map for the poweroff register (in bytes)
|
||||||
|
- mask: the poweroff value written to the poweroff register (32 bit access)
|
||||||
|
|
||||||
|
Default will be little endian mode, 32 bit access only.
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
|
||||||
|
poweroff {
|
||||||
|
compatible = "syscon-poweroff";
|
||||||
|
regmap = <®mapnode>;
|
||||||
|
offset = <0x0>;
|
||||||
|
mask = <0x7a>;
|
||||||
|
};
|
|
@ -3137,12 +3137,15 @@ S: Supported
|
||||||
F: Documentation/hwmon/da90??
|
F: Documentation/hwmon/da90??
|
||||||
F: drivers/gpio/gpio-da90??.c
|
F: drivers/gpio/gpio-da90??.c
|
||||||
F: drivers/hwmon/da90??-hwmon.c
|
F: drivers/hwmon/da90??-hwmon.c
|
||||||
|
F: drivers/iio/adc/da91??-*.c
|
||||||
F: drivers/input/misc/da90??_onkey.c
|
F: drivers/input/misc/da90??_onkey.c
|
||||||
F: drivers/input/touchscreen/da9052_tsi.c
|
F: drivers/input/touchscreen/da9052_tsi.c
|
||||||
F: drivers/leds/leds-da90??.c
|
F: drivers/leds/leds-da90??.c
|
||||||
F: drivers/mfd/da903x.c
|
F: drivers/mfd/da903x.c
|
||||||
F: drivers/mfd/da90??-*.c
|
F: drivers/mfd/da90??-*.c
|
||||||
|
F: drivers/mfd/da91??-*.c
|
||||||
F: drivers/power/da9052-battery.c
|
F: drivers/power/da9052-battery.c
|
||||||
|
F: drivers/power/da91??-*.c
|
||||||
F: drivers/regulator/da903x.c
|
F: drivers/regulator/da903x.c
|
||||||
F: drivers/regulator/da9???-regulator.[ch]
|
F: drivers/regulator/da9???-regulator.[ch]
|
||||||
F: drivers/rtc/rtc-da90??.c
|
F: drivers/rtc/rtc-da90??.c
|
||||||
|
@ -3152,6 +3155,7 @@ F: include/linux/mfd/da903x.h
|
||||||
F: include/linux/mfd/da9052/
|
F: include/linux/mfd/da9052/
|
||||||
F: include/linux/mfd/da9055/
|
F: include/linux/mfd/da9055/
|
||||||
F: include/linux/mfd/da9063/
|
F: include/linux/mfd/da9063/
|
||||||
|
F: include/linux/mfd/da9150/
|
||||||
F: include/sound/da[79]*.h
|
F: include/sound/da[79]*.h
|
||||||
F: sound/soc/codecs/da[79]*.[ch]
|
F: sound/soc/codecs/da[79]*.[ch]
|
||||||
|
|
||||||
|
|
|
@ -758,8 +758,10 @@ static void raumfeld_power_signal_charged(void)
|
||||||
struct power_supply *psy =
|
struct power_supply *psy =
|
||||||
power_supply_get_by_name(raumfeld_power_supplicants[0]);
|
power_supply_get_by_name(raumfeld_power_supplicants[0]);
|
||||||
|
|
||||||
if (psy)
|
if (psy) {
|
||||||
power_supply_set_battery_charged(psy);
|
power_supply_set_battery_charged(psy);
|
||||||
|
power_supply_put(psy);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int raumfeld_power_resume(void)
|
static int raumfeld_power_resume(void)
|
||||||
|
|
|
@ -61,7 +61,7 @@ static void battery_status_changed(void)
|
||||||
|
|
||||||
if (psy) {
|
if (psy) {
|
||||||
power_supply_changed(psy);
|
power_supply_changed(psy);
|
||||||
put_device(psy->dev);
|
power_supply_put(psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,7 +71,7 @@ static void ac_status_changed(void)
|
||||||
|
|
||||||
if (psy) {
|
if (psy) {
|
||||||
power_supply_changed(psy);
|
power_supply_changed(psy);
|
||||||
put_device(psy->dev);
|
power_supply_put(psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -83,7 +83,7 @@ static void battery_status_changed(void)
|
||||||
|
|
||||||
if (psy) {
|
if (psy) {
|
||||||
power_supply_changed(psy);
|
power_supply_changed(psy);
|
||||||
put_device(psy->dev);
|
power_supply_put(psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -93,7 +93,7 @@ static void ac_status_changed(void)
|
||||||
|
|
||||||
if (psy) {
|
if (psy) {
|
||||||
power_supply_changed(psy);
|
power_supply_changed(psy);
|
||||||
put_device(psy->dev);
|
power_supply_put(psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -95,13 +95,14 @@ static struct acpi_driver acpi_ac_driver = {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct acpi_ac {
|
struct acpi_ac {
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
|
struct power_supply_desc charger_desc;
|
||||||
struct acpi_device * device;
|
struct acpi_device * device;
|
||||||
unsigned long long state;
|
unsigned long long state;
|
||||||
struct notifier_block battery_nb;
|
struct notifier_block battery_nb;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define to_acpi_ac(x) container_of(x, struct acpi_ac, charger)
|
#define to_acpi_ac(x) power_supply_get_drvdata(x)
|
||||||
|
|
||||||
#ifdef CONFIG_ACPI_PROCFS_POWER
|
#ifdef CONFIG_ACPI_PROCFS_POWER
|
||||||
static const struct file_operations acpi_ac_fops = {
|
static const struct file_operations acpi_ac_fops = {
|
||||||
|
@ -275,7 +276,7 @@ static void acpi_ac_notify(struct acpi_device *device, u32 event)
|
||||||
dev_name(&device->dev), event,
|
dev_name(&device->dev), event,
|
||||||
(u32) ac->state);
|
(u32) ac->state);
|
||||||
acpi_notifier_call_chain(device, event, (u32) ac->state);
|
acpi_notifier_call_chain(device, event, (u32) ac->state);
|
||||||
kobject_uevent(&ac->charger.dev->kobj, KOBJ_CHANGE);
|
kobject_uevent(&ac->charger->dev.kobj, KOBJ_CHANGE);
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
|
@ -321,6 +322,7 @@ static struct dmi_system_id ac_dmi_table[] = {
|
||||||
|
|
||||||
static int acpi_ac_add(struct acpi_device *device)
|
static int acpi_ac_add(struct acpi_device *device)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
int result = 0;
|
int result = 0;
|
||||||
struct acpi_ac *ac = NULL;
|
struct acpi_ac *ac = NULL;
|
||||||
|
|
||||||
|
@ -341,19 +343,24 @@ static int acpi_ac_add(struct acpi_device *device)
|
||||||
if (result)
|
if (result)
|
||||||
goto end;
|
goto end;
|
||||||
|
|
||||||
ac->charger.name = acpi_device_bid(device);
|
psy_cfg.drv_data = ac;
|
||||||
|
|
||||||
|
ac->charger_desc.name = acpi_device_bid(device);
|
||||||
#ifdef CONFIG_ACPI_PROCFS_POWER
|
#ifdef CONFIG_ACPI_PROCFS_POWER
|
||||||
result = acpi_ac_add_fs(ac);
|
result = acpi_ac_add_fs(ac);
|
||||||
if (result)
|
if (result)
|
||||||
goto end;
|
goto end;
|
||||||
#endif
|
#endif
|
||||||
ac->charger.type = POWER_SUPPLY_TYPE_MAINS;
|
ac->charger_desc.type = POWER_SUPPLY_TYPE_MAINS;
|
||||||
ac->charger.properties = ac_props;
|
ac->charger_desc.properties = ac_props;
|
||||||
ac->charger.num_properties = ARRAY_SIZE(ac_props);
|
ac->charger_desc.num_properties = ARRAY_SIZE(ac_props);
|
||||||
ac->charger.get_property = get_ac_property;
|
ac->charger_desc.get_property = get_ac_property;
|
||||||
result = power_supply_register(&ac->device->dev, &ac->charger);
|
ac->charger = power_supply_register(&ac->device->dev,
|
||||||
if (result)
|
&ac->charger_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(ac->charger)) {
|
||||||
|
result = PTR_ERR(ac->charger);
|
||||||
goto end;
|
goto end;
|
||||||
|
}
|
||||||
|
|
||||||
printk(KERN_INFO PREFIX "%s [%s] (%s)\n",
|
printk(KERN_INFO PREFIX "%s [%s] (%s)\n",
|
||||||
acpi_device_name(device), acpi_device_bid(device),
|
acpi_device_name(device), acpi_device_bid(device),
|
||||||
|
@ -390,7 +397,7 @@ static int acpi_ac_resume(struct device *dev)
|
||||||
if (acpi_ac_get_state(ac))
|
if (acpi_ac_get_state(ac))
|
||||||
return 0;
|
return 0;
|
||||||
if (old_state != ac->state)
|
if (old_state != ac->state)
|
||||||
kobject_uevent(&ac->charger.dev->kobj, KOBJ_CHANGE);
|
kobject_uevent(&ac->charger->dev.kobj, KOBJ_CHANGE);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
|
@ -407,8 +414,7 @@ static int acpi_ac_remove(struct acpi_device *device)
|
||||||
|
|
||||||
ac = acpi_driver_data(device);
|
ac = acpi_driver_data(device);
|
||||||
|
|
||||||
if (ac->charger.dev)
|
power_supply_unregister(ac->charger);
|
||||||
power_supply_unregister(&ac->charger);
|
|
||||||
unregister_acpi_notifier(&ac->battery_nb);
|
unregister_acpi_notifier(&ac->battery_nb);
|
||||||
|
|
||||||
#ifdef CONFIG_ACPI_PROCFS_POWER
|
#ifdef CONFIG_ACPI_PROCFS_POWER
|
||||||
|
|
|
@ -117,7 +117,8 @@ enum {
|
||||||
struct acpi_battery {
|
struct acpi_battery {
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
struct mutex sysfs_lock;
|
struct mutex sysfs_lock;
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
struct power_supply_desc bat_desc;
|
||||||
struct acpi_device *device;
|
struct acpi_device *device;
|
||||||
struct notifier_block pm_nb;
|
struct notifier_block pm_nb;
|
||||||
unsigned long update_time;
|
unsigned long update_time;
|
||||||
|
@ -149,7 +150,7 @@ struct acpi_battery {
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define to_acpi_battery(x) container_of(x, struct acpi_battery, bat)
|
#define to_acpi_battery(x) power_supply_get_drvdata(x)
|
||||||
|
|
||||||
static inline int acpi_battery_present(struct acpi_battery *battery)
|
static inline int acpi_battery_present(struct acpi_battery *battery)
|
||||||
{
|
{
|
||||||
|
@ -608,40 +609,45 @@ static struct device_attribute alarm_attr = {
|
||||||
|
|
||||||
static int sysfs_add_battery(struct acpi_battery *battery)
|
static int sysfs_add_battery(struct acpi_battery *battery)
|
||||||
{
|
{
|
||||||
int result;
|
struct power_supply_config psy_cfg = { .drv_data = battery, };
|
||||||
|
|
||||||
if (battery->power_unit == ACPI_BATTERY_POWER_UNIT_MA) {
|
if (battery->power_unit == ACPI_BATTERY_POWER_UNIT_MA) {
|
||||||
battery->bat.properties = charge_battery_props;
|
battery->bat_desc.properties = charge_battery_props;
|
||||||
battery->bat.num_properties =
|
battery->bat_desc.num_properties =
|
||||||
ARRAY_SIZE(charge_battery_props);
|
ARRAY_SIZE(charge_battery_props);
|
||||||
} else {
|
} else {
|
||||||
battery->bat.properties = energy_battery_props;
|
battery->bat_desc.properties = energy_battery_props;
|
||||||
battery->bat.num_properties =
|
battery->bat_desc.num_properties =
|
||||||
ARRAY_SIZE(energy_battery_props);
|
ARRAY_SIZE(energy_battery_props);
|
||||||
}
|
}
|
||||||
|
|
||||||
battery->bat.name = acpi_device_bid(battery->device);
|
battery->bat_desc.name = acpi_device_bid(battery->device);
|
||||||
battery->bat.type = POWER_SUPPLY_TYPE_BATTERY;
|
battery->bat_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
battery->bat.get_property = acpi_battery_get_property;
|
battery->bat_desc.get_property = acpi_battery_get_property;
|
||||||
|
|
||||||
result = power_supply_register_no_ws(&battery->device->dev, &battery->bat);
|
battery->bat = power_supply_register_no_ws(&battery->device->dev,
|
||||||
|
&battery->bat_desc, &psy_cfg);
|
||||||
|
|
||||||
if (result)
|
if (IS_ERR(battery->bat)) {
|
||||||
|
int result = PTR_ERR(battery->bat);
|
||||||
|
|
||||||
|
battery->bat = NULL;
|
||||||
return result;
|
return result;
|
||||||
return device_create_file(battery->bat.dev, &alarm_attr);
|
}
|
||||||
|
return device_create_file(&battery->bat->dev, &alarm_attr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sysfs_remove_battery(struct acpi_battery *battery)
|
static void sysfs_remove_battery(struct acpi_battery *battery)
|
||||||
{
|
{
|
||||||
mutex_lock(&battery->sysfs_lock);
|
mutex_lock(&battery->sysfs_lock);
|
||||||
if (!battery->bat.dev) {
|
if (!battery->bat) {
|
||||||
mutex_unlock(&battery->sysfs_lock);
|
mutex_unlock(&battery->sysfs_lock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
device_remove_file(battery->bat.dev, &alarm_attr);
|
device_remove_file(&battery->bat->dev, &alarm_attr);
|
||||||
power_supply_unregister(&battery->bat);
|
power_supply_unregister(battery->bat);
|
||||||
battery->bat.dev = NULL;
|
battery->bat = NULL;
|
||||||
mutex_unlock(&battery->sysfs_lock);
|
mutex_unlock(&battery->sysfs_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -738,7 +744,7 @@ static int acpi_battery_update(struct acpi_battery *battery, bool resume)
|
||||||
return result;
|
return result;
|
||||||
acpi_battery_init_alarm(battery);
|
acpi_battery_init_alarm(battery);
|
||||||
}
|
}
|
||||||
if (!battery->bat.dev) {
|
if (!battery->bat) {
|
||||||
result = sysfs_add_battery(battery);
|
result = sysfs_add_battery(battery);
|
||||||
if (result)
|
if (result)
|
||||||
return result;
|
return result;
|
||||||
|
@ -764,7 +770,7 @@ static void acpi_battery_refresh(struct acpi_battery *battery)
|
||||||
{
|
{
|
||||||
int power_unit;
|
int power_unit;
|
||||||
|
|
||||||
if (!battery->bat.dev)
|
if (!battery->bat)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
power_unit = battery->power_unit;
|
power_unit = battery->power_unit;
|
||||||
|
@ -1062,11 +1068,11 @@ static void acpi_battery_remove_fs(struct acpi_device *device)
|
||||||
static void acpi_battery_notify(struct acpi_device *device, u32 event)
|
static void acpi_battery_notify(struct acpi_device *device, u32 event)
|
||||||
{
|
{
|
||||||
struct acpi_battery *battery = acpi_driver_data(device);
|
struct acpi_battery *battery = acpi_driver_data(device);
|
||||||
struct device *old;
|
struct power_supply *old;
|
||||||
|
|
||||||
if (!battery)
|
if (!battery)
|
||||||
return;
|
return;
|
||||||
old = battery->bat.dev;
|
old = battery->bat;
|
||||||
/*
|
/*
|
||||||
* On Acer Aspire V5-573G notifications are sometimes triggered too
|
* On Acer Aspire V5-573G notifications are sometimes triggered too
|
||||||
* early. For example, when AC is unplugged and notification is
|
* early. For example, when AC is unplugged and notification is
|
||||||
|
@ -1083,8 +1089,8 @@ static void acpi_battery_notify(struct acpi_device *device, u32 event)
|
||||||
acpi_battery_present(battery));
|
acpi_battery_present(battery));
|
||||||
acpi_notifier_call_chain(device, event, acpi_battery_present(battery));
|
acpi_notifier_call_chain(device, event, acpi_battery_present(battery));
|
||||||
/* acpi_battery_update could remove power_supply object */
|
/* acpi_battery_update could remove power_supply object */
|
||||||
if (old && battery->bat.dev)
|
if (old && battery->bat)
|
||||||
power_supply_changed(&battery->bat);
|
power_supply_changed(battery->bat);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int battery_notify(struct notifier_block *nb,
|
static int battery_notify(struct notifier_block *nb,
|
||||||
|
@ -1100,7 +1106,7 @@ static int battery_notify(struct notifier_block *nb,
|
||||||
if (!acpi_battery_present(battery))
|
if (!acpi_battery_present(battery))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (!battery->bat.dev) {
|
if (battery->bat) {
|
||||||
result = acpi_battery_get_info(battery);
|
result = acpi_battery_get_info(battery);
|
||||||
if (result)
|
if (result)
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -74,7 +74,8 @@ static const struct acpi_device_id sbs_device_ids[] = {
|
||||||
MODULE_DEVICE_TABLE(acpi, sbs_device_ids);
|
MODULE_DEVICE_TABLE(acpi, sbs_device_ids);
|
||||||
|
|
||||||
struct acpi_battery {
|
struct acpi_battery {
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
struct power_supply_desc bat_desc;
|
||||||
struct acpi_sbs *sbs;
|
struct acpi_sbs *sbs;
|
||||||
unsigned long update_time;
|
unsigned long update_time;
|
||||||
char name[8];
|
char name[8];
|
||||||
|
@ -101,10 +102,10 @@ struct acpi_battery {
|
||||||
u8 have_sysfs_alarm:1;
|
u8 have_sysfs_alarm:1;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define to_acpi_battery(x) container_of(x, struct acpi_battery, bat)
|
#define to_acpi_battery(x) power_supply_get_drvdata(x)
|
||||||
|
|
||||||
struct acpi_sbs {
|
struct acpi_sbs {
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
struct acpi_device *device;
|
struct acpi_device *device;
|
||||||
struct acpi_smb_hc *hc;
|
struct acpi_smb_hc *hc;
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
|
@ -115,7 +116,7 @@ struct acpi_sbs {
|
||||||
u8 charger_exists:1;
|
u8 charger_exists:1;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define to_acpi_sbs(x) container_of(x, struct acpi_sbs, charger)
|
#define to_acpi_sbs(x) power_supply_get_drvdata(x)
|
||||||
|
|
||||||
static int acpi_sbs_remove(struct acpi_device *device);
|
static int acpi_sbs_remove(struct acpi_device *device);
|
||||||
static int acpi_battery_get_state(struct acpi_battery *battery);
|
static int acpi_battery_get_state(struct acpi_battery *battery);
|
||||||
|
@ -303,6 +304,13 @@ static enum power_supply_property sbs_energy_battery_props[] = {
|
||||||
POWER_SUPPLY_PROP_MANUFACTURER,
|
POWER_SUPPLY_PROP_MANUFACTURER,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc acpi_sbs_charger_desc = {
|
||||||
|
.name = "sbs-charger",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = sbs_ac_props,
|
||||||
|
.num_properties = ARRAY_SIZE(sbs_ac_props),
|
||||||
|
.get_property = sbs_get_ac_property,
|
||||||
|
};
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
Smart Battery System Management
|
Smart Battery System Management
|
||||||
|
@ -519,6 +527,7 @@ static int acpi_battery_read(struct acpi_battery *battery)
|
||||||
static int acpi_battery_add(struct acpi_sbs *sbs, int id)
|
static int acpi_battery_add(struct acpi_sbs *sbs, int id)
|
||||||
{
|
{
|
||||||
struct acpi_battery *battery = &sbs->battery[id];
|
struct acpi_battery *battery = &sbs->battery[id];
|
||||||
|
struct power_supply_config psy_cfg = { .drv_data = battery, };
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
battery->id = id;
|
battery->id = id;
|
||||||
|
@ -528,23 +537,27 @@ static int acpi_battery_add(struct acpi_sbs *sbs, int id)
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
sprintf(battery->name, ACPI_BATTERY_DIR_NAME, id);
|
sprintf(battery->name, ACPI_BATTERY_DIR_NAME, id);
|
||||||
battery->bat.name = battery->name;
|
battery->bat_desc.name = battery->name;
|
||||||
battery->bat.type = POWER_SUPPLY_TYPE_BATTERY;
|
battery->bat_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
if (!acpi_battery_mode(battery)) {
|
if (!acpi_battery_mode(battery)) {
|
||||||
battery->bat.properties = sbs_charge_battery_props;
|
battery->bat_desc.properties = sbs_charge_battery_props;
|
||||||
battery->bat.num_properties =
|
battery->bat_desc.num_properties =
|
||||||
ARRAY_SIZE(sbs_charge_battery_props);
|
ARRAY_SIZE(sbs_charge_battery_props);
|
||||||
} else {
|
} else {
|
||||||
battery->bat.properties = sbs_energy_battery_props;
|
battery->bat_desc.properties = sbs_energy_battery_props;
|
||||||
battery->bat.num_properties =
|
battery->bat_desc.num_properties =
|
||||||
ARRAY_SIZE(sbs_energy_battery_props);
|
ARRAY_SIZE(sbs_energy_battery_props);
|
||||||
}
|
}
|
||||||
battery->bat.get_property = acpi_sbs_battery_get_property;
|
battery->bat_desc.get_property = acpi_sbs_battery_get_property;
|
||||||
result = power_supply_register(&sbs->device->dev, &battery->bat);
|
battery->bat = power_supply_register(&sbs->device->dev,
|
||||||
if (result)
|
&battery->bat_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(battery->bat)) {
|
||||||
|
result = PTR_ERR(battery->bat);
|
||||||
|
battery->bat = NULL;
|
||||||
goto end;
|
goto end;
|
||||||
|
}
|
||||||
|
|
||||||
result = device_create_file(battery->bat.dev, &alarm_attr);
|
result = device_create_file(&battery->bat->dev, &alarm_attr);
|
||||||
if (result)
|
if (result)
|
||||||
goto end;
|
goto end;
|
||||||
battery->have_sysfs_alarm = 1;
|
battery->have_sysfs_alarm = 1;
|
||||||
|
@ -559,28 +572,29 @@ static void acpi_battery_remove(struct acpi_sbs *sbs, int id)
|
||||||
{
|
{
|
||||||
struct acpi_battery *battery = &sbs->battery[id];
|
struct acpi_battery *battery = &sbs->battery[id];
|
||||||
|
|
||||||
if (battery->bat.dev) {
|
if (battery->bat) {
|
||||||
if (battery->have_sysfs_alarm)
|
if (battery->have_sysfs_alarm)
|
||||||
device_remove_file(battery->bat.dev, &alarm_attr);
|
device_remove_file(&battery->bat->dev, &alarm_attr);
|
||||||
power_supply_unregister(&battery->bat);
|
power_supply_unregister(battery->bat);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int acpi_charger_add(struct acpi_sbs *sbs)
|
static int acpi_charger_add(struct acpi_sbs *sbs)
|
||||||
{
|
{
|
||||||
int result;
|
int result;
|
||||||
|
struct power_supply_config psy_cfg = { .drv_data = sbs, };
|
||||||
|
|
||||||
result = acpi_ac_get_present(sbs);
|
result = acpi_ac_get_present(sbs);
|
||||||
if (result)
|
if (result)
|
||||||
goto end;
|
goto end;
|
||||||
|
|
||||||
sbs->charger_exists = 1;
|
sbs->charger_exists = 1;
|
||||||
sbs->charger.name = "sbs-charger";
|
sbs->charger = power_supply_register(&sbs->device->dev,
|
||||||
sbs->charger.type = POWER_SUPPLY_TYPE_MAINS;
|
&acpi_sbs_charger_desc, &psy_cfg);
|
||||||
sbs->charger.properties = sbs_ac_props;
|
if (IS_ERR(sbs->charger)) {
|
||||||
sbs->charger.num_properties = ARRAY_SIZE(sbs_ac_props);
|
result = PTR_ERR(sbs->charger);
|
||||||
sbs->charger.get_property = sbs_get_ac_property;
|
sbs->charger = NULL;
|
||||||
power_supply_register(&sbs->device->dev, &sbs->charger);
|
}
|
||||||
printk(KERN_INFO PREFIX "%s [%s]: AC Adapter [%s] (%s)\n",
|
printk(KERN_INFO PREFIX "%s [%s]: AC Adapter [%s] (%s)\n",
|
||||||
ACPI_SBS_DEVICE_NAME, acpi_device_bid(sbs->device),
|
ACPI_SBS_DEVICE_NAME, acpi_device_bid(sbs->device),
|
||||||
ACPI_AC_DIR_NAME, sbs->charger_present ? "on-line" : "off-line");
|
ACPI_AC_DIR_NAME, sbs->charger_present ? "on-line" : "off-line");
|
||||||
|
@ -590,8 +604,8 @@ static int acpi_charger_add(struct acpi_sbs *sbs)
|
||||||
|
|
||||||
static void acpi_charger_remove(struct acpi_sbs *sbs)
|
static void acpi_charger_remove(struct acpi_sbs *sbs)
|
||||||
{
|
{
|
||||||
if (sbs->charger.dev)
|
if (sbs->charger)
|
||||||
power_supply_unregister(&sbs->charger);
|
power_supply_unregister(sbs->charger);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void acpi_sbs_callback(void *context)
|
static void acpi_sbs_callback(void *context)
|
||||||
|
@ -605,7 +619,7 @@ static void acpi_sbs_callback(void *context)
|
||||||
if (sbs->charger_exists) {
|
if (sbs->charger_exists) {
|
||||||
acpi_ac_get_present(sbs);
|
acpi_ac_get_present(sbs);
|
||||||
if (sbs->charger_present != saved_charger_state)
|
if (sbs->charger_present != saved_charger_state)
|
||||||
kobject_uevent(&sbs->charger.dev->kobj, KOBJ_CHANGE);
|
kobject_uevent(&sbs->charger->dev.kobj, KOBJ_CHANGE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sbs->manager_present) {
|
if (sbs->manager_present) {
|
||||||
|
@ -617,7 +631,7 @@ static void acpi_sbs_callback(void *context)
|
||||||
acpi_battery_read(bat);
|
acpi_battery_read(bat);
|
||||||
if (saved_battery_state == bat->present)
|
if (saved_battery_state == bat->present)
|
||||||
continue;
|
continue;
|
||||||
kobject_uevent(&bat->bat.dev->kobj, KOBJ_CHANGE);
|
kobject_uevent(&bat->bat->dev.kobj, KOBJ_CHANGE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -339,7 +339,7 @@ static int hidinput_get_battery_property(struct power_supply *psy,
|
||||||
enum power_supply_property prop,
|
enum power_supply_property prop,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct hid_device *dev = container_of(psy, struct hid_device, battery);
|
struct hid_device *dev = power_supply_get_drvdata(psy);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
__u8 *buf;
|
__u8 *buf;
|
||||||
|
|
||||||
|
@ -397,26 +397,32 @@ static int hidinput_get_battery_property(struct power_supply *psy,
|
||||||
|
|
||||||
static bool hidinput_setup_battery(struct hid_device *dev, unsigned report_type, struct hid_field *field)
|
static bool hidinput_setup_battery(struct hid_device *dev, unsigned report_type, struct hid_field *field)
|
||||||
{
|
{
|
||||||
struct power_supply *battery = &dev->battery;
|
struct power_supply_desc *psy_desc = NULL;
|
||||||
int ret;
|
struct power_supply_config psy_cfg = { .drv_data = dev, };
|
||||||
unsigned quirks;
|
unsigned quirks;
|
||||||
s32 min, max;
|
s32 min, max;
|
||||||
|
|
||||||
if (field->usage->hid != HID_DC_BATTERYSTRENGTH)
|
if (field->usage->hid != HID_DC_BATTERYSTRENGTH)
|
||||||
return false; /* no match */
|
return false; /* no match */
|
||||||
|
|
||||||
if (battery->name != NULL)
|
if (dev->battery != NULL)
|
||||||
goto out; /* already initialized? */
|
goto out; /* already initialized? */
|
||||||
|
|
||||||
battery->name = kasprintf(GFP_KERNEL, "hid-%s-battery", dev->uniq);
|
psy_desc = kzalloc(sizeof(*psy_desc), GFP_KERNEL);
|
||||||
if (battery->name == NULL)
|
if (psy_desc == NULL)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
battery->type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_desc->name = kasprintf(GFP_KERNEL, "hid-%s-battery", dev->uniq);
|
||||||
battery->properties = hidinput_battery_props;
|
if (psy_desc->name == NULL) {
|
||||||
battery->num_properties = ARRAY_SIZE(hidinput_battery_props);
|
kfree(psy_desc);
|
||||||
battery->use_for_apm = 0;
|
goto out;
|
||||||
battery->get_property = hidinput_get_battery_property;
|
}
|
||||||
|
|
||||||
|
psy_desc->type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
|
psy_desc->properties = hidinput_battery_props;
|
||||||
|
psy_desc->num_properties = ARRAY_SIZE(hidinput_battery_props);
|
||||||
|
psy_desc->use_for_apm = 0;
|
||||||
|
psy_desc->get_property = hidinput_get_battery_property;
|
||||||
|
|
||||||
quirks = find_battery_quirk(dev);
|
quirks = find_battery_quirk(dev);
|
||||||
|
|
||||||
|
@ -439,27 +445,30 @@ static bool hidinput_setup_battery(struct hid_device *dev, unsigned report_type,
|
||||||
dev->battery_report_type = report_type;
|
dev->battery_report_type = report_type;
|
||||||
dev->battery_report_id = field->report->id;
|
dev->battery_report_id = field->report->id;
|
||||||
|
|
||||||
ret = power_supply_register(&dev->dev, battery);
|
dev->battery = power_supply_register(&dev->dev, psy_desc, &psy_cfg);
|
||||||
if (ret != 0) {
|
if (IS_ERR(dev->battery)) {
|
||||||
hid_warn(dev, "can't register power supply: %d\n", ret);
|
hid_warn(dev, "can't register power supply: %ld\n",
|
||||||
kfree(battery->name);
|
PTR_ERR(dev->battery));
|
||||||
battery->name = NULL;
|
kfree(psy_desc->name);
|
||||||
|
kfree(psy_desc);
|
||||||
|
dev->battery = NULL;
|
||||||
|
} else {
|
||||||
|
power_supply_powers(dev->battery, &dev->dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_powers(battery, &dev->dev);
|
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void hidinput_cleanup_battery(struct hid_device *dev)
|
static void hidinput_cleanup_battery(struct hid_device *dev)
|
||||||
{
|
{
|
||||||
if (!dev->battery.name)
|
if (!dev->battery)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
power_supply_unregister(&dev->battery);
|
power_supply_unregister(dev->battery);
|
||||||
kfree(dev->battery.name);
|
kfree(dev->battery->desc->name);
|
||||||
dev->battery.name = NULL;
|
kfree(dev->battery->desc);
|
||||||
|
dev->battery = NULL;
|
||||||
}
|
}
|
||||||
#else /* !CONFIG_HID_BATTERY_STRENGTH */
|
#else /* !CONFIG_HID_BATTERY_STRENGTH */
|
||||||
static bool hidinput_setup_battery(struct hid_device *dev, unsigned report_type,
|
static bool hidinput_setup_battery(struct hid_device *dev, unsigned report_type,
|
||||||
|
|
|
@ -815,7 +815,8 @@ struct sony_sc {
|
||||||
struct led_classdev *leds[MAX_LEDS];
|
struct led_classdev *leds[MAX_LEDS];
|
||||||
unsigned long quirks;
|
unsigned long quirks;
|
||||||
struct work_struct state_worker;
|
struct work_struct state_worker;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
|
struct power_supply_desc battery_desc;
|
||||||
int device_id;
|
int device_id;
|
||||||
__u8 *output_report_dmabuf;
|
__u8 *output_report_dmabuf;
|
||||||
|
|
||||||
|
@ -1660,7 +1661,7 @@ static int sony_battery_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 sony_sc *sc = container_of(psy, struct sony_sc, battery);
|
struct sony_sc *sc = power_supply_get_drvdata(psy);
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
u8 battery_charging, battery_capacity, cable_state;
|
u8 battery_charging, battery_capacity, cable_state;
|
||||||
|
@ -1699,6 +1700,7 @@ static int sony_battery_get_property(struct power_supply *psy,
|
||||||
|
|
||||||
static int sony_battery_probe(struct sony_sc *sc)
|
static int sony_battery_probe(struct sony_sc *sc)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = { .drv_data = sc, };
|
||||||
struct hid_device *hdev = sc->hdev;
|
struct hid_device *hdev = sc->hdev;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -1708,39 +1710,42 @@ static int sony_battery_probe(struct sony_sc *sc)
|
||||||
*/
|
*/
|
||||||
sc->battery_capacity = 100;
|
sc->battery_capacity = 100;
|
||||||
|
|
||||||
sc->battery.properties = sony_battery_props;
|
sc->battery_desc.properties = sony_battery_props;
|
||||||
sc->battery.num_properties = ARRAY_SIZE(sony_battery_props);
|
sc->battery_desc.num_properties = ARRAY_SIZE(sony_battery_props);
|
||||||
sc->battery.get_property = sony_battery_get_property;
|
sc->battery_desc.get_property = sony_battery_get_property;
|
||||||
sc->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
sc->battery_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
sc->battery.use_for_apm = 0;
|
sc->battery_desc.use_for_apm = 0;
|
||||||
sc->battery.name = kasprintf(GFP_KERNEL, "sony_controller_battery_%pMR",
|
sc->battery_desc.name = kasprintf(GFP_KERNEL,
|
||||||
sc->mac_address);
|
"sony_controller_battery_%pMR",
|
||||||
if (!sc->battery.name)
|
sc->mac_address);
|
||||||
|
if (!sc->battery_desc.name)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
ret = power_supply_register(&hdev->dev, &sc->battery);
|
sc->battery = power_supply_register(&hdev->dev, &sc->battery_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(sc->battery)) {
|
||||||
|
ret = PTR_ERR(sc->battery);
|
||||||
hid_err(hdev, "Unable to register battery device\n");
|
hid_err(hdev, "Unable to register battery device\n");
|
||||||
goto err_free;
|
goto err_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_powers(&sc->battery, &hdev->dev);
|
power_supply_powers(sc->battery, &hdev->dev);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_free:
|
err_free:
|
||||||
kfree(sc->battery.name);
|
kfree(sc->battery_desc.name);
|
||||||
sc->battery.name = NULL;
|
sc->battery_desc.name = NULL;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sony_battery_remove(struct sony_sc *sc)
|
static void sony_battery_remove(struct sony_sc *sc)
|
||||||
{
|
{
|
||||||
if (!sc->battery.name)
|
if (!sc->battery_desc.name)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
power_supply_unregister(&sc->battery);
|
power_supply_unregister(sc->battery);
|
||||||
kfree(sc->battery.name);
|
kfree(sc->battery_desc.name);
|
||||||
sc->battery.name = NULL;
|
sc->battery_desc.name = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -203,8 +203,7 @@ static int wiimod_battery_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 wiimote_data *wdata = container_of(psy, struct wiimote_data,
|
struct wiimote_data *wdata = power_supply_get_drvdata(psy);
|
||||||
battery);
|
|
||||||
int ret = 0, state;
|
int ret = 0, state;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
|
@ -238,42 +237,46 @@ static int wiimod_battery_get_property(struct power_supply *psy,
|
||||||
static int wiimod_battery_probe(const struct wiimod_ops *ops,
|
static int wiimod_battery_probe(const struct wiimod_ops *ops,
|
||||||
struct wiimote_data *wdata)
|
struct wiimote_data *wdata)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = { .drv_data = wdata, };
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
wdata->battery.properties = wiimod_battery_props;
|
wdata->battery_desc.properties = wiimod_battery_props;
|
||||||
wdata->battery.num_properties = ARRAY_SIZE(wiimod_battery_props);
|
wdata->battery_desc.num_properties = ARRAY_SIZE(wiimod_battery_props);
|
||||||
wdata->battery.get_property = wiimod_battery_get_property;
|
wdata->battery_desc.get_property = wiimod_battery_get_property;
|
||||||
wdata->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
wdata->battery_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
wdata->battery.use_for_apm = 0;
|
wdata->battery_desc.use_for_apm = 0;
|
||||||
wdata->battery.name = kasprintf(GFP_KERNEL, "wiimote_battery_%s",
|
wdata->battery_desc.name = kasprintf(GFP_KERNEL, "wiimote_battery_%s",
|
||||||
wdata->hdev->uniq);
|
wdata->hdev->uniq);
|
||||||
if (!wdata->battery.name)
|
if (!wdata->battery_desc.name)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
ret = power_supply_register(&wdata->hdev->dev, &wdata->battery);
|
wdata->battery = power_supply_register(&wdata->hdev->dev,
|
||||||
if (ret) {
|
&wdata->battery_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(wdata->battery)) {
|
||||||
hid_err(wdata->hdev, "cannot register battery device\n");
|
hid_err(wdata->hdev, "cannot register battery device\n");
|
||||||
|
ret = PTR_ERR(wdata->battery);
|
||||||
goto err_free;
|
goto err_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_powers(&wdata->battery, &wdata->hdev->dev);
|
power_supply_powers(wdata->battery, &wdata->hdev->dev);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_free:
|
err_free:
|
||||||
kfree(wdata->battery.name);
|
kfree(wdata->battery_desc.name);
|
||||||
wdata->battery.name = NULL;
|
wdata->battery_desc.name = NULL;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wiimod_battery_remove(const struct wiimod_ops *ops,
|
static void wiimod_battery_remove(const struct wiimod_ops *ops,
|
||||||
struct wiimote_data *wdata)
|
struct wiimote_data *wdata)
|
||||||
{
|
{
|
||||||
if (!wdata->battery.name)
|
if (!wdata->battery_desc.name)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
power_supply_unregister(&wdata->battery);
|
power_supply_unregister(wdata->battery);
|
||||||
kfree(wdata->battery.name);
|
kfree(wdata->battery_desc.name);
|
||||||
wdata->battery.name = NULL;
|
wdata->battery_desc.name = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct wiimod_ops wiimod_battery = {
|
static const struct wiimod_ops wiimod_battery = {
|
||||||
|
|
|
@ -147,7 +147,8 @@ struct wiimote_data {
|
||||||
struct led_classdev *leds[4];
|
struct led_classdev *leds[4];
|
||||||
struct input_dev *accel;
|
struct input_dev *accel;
|
||||||
struct input_dev *ir;
|
struct input_dev *ir;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
|
struct power_supply_desc battery_desc;
|
||||||
struct input_dev *mp;
|
struct input_dev *mp;
|
||||||
struct timer_list timer;
|
struct timer_list timer;
|
||||||
struct wiimote_debug *debug;
|
struct wiimote_debug *debug;
|
||||||
|
|
|
@ -119,8 +119,10 @@ struct wacom {
|
||||||
u8 img_lum; /* OLED matrix display brightness */
|
u8 img_lum; /* OLED matrix display brightness */
|
||||||
} led;
|
} led;
|
||||||
bool led_initialized;
|
bool led_initialized;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
struct power_supply ac;
|
struct power_supply *ac;
|
||||||
|
struct power_supply_desc battery_desc;
|
||||||
|
struct power_supply_desc ac_desc;
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline void wacom_schedule_work(struct wacom_wac *wacom_wac)
|
static inline void wacom_schedule_work(struct wacom_wac *wacom_wac)
|
||||||
|
@ -133,7 +135,7 @@ static inline void wacom_notify_battery(struct wacom_wac *wacom_wac)
|
||||||
{
|
{
|
||||||
struct wacom *wacom = container_of(wacom_wac, struct wacom, wacom_wac);
|
struct wacom *wacom = container_of(wacom_wac, struct wacom, wacom_wac);
|
||||||
|
|
||||||
power_supply_changed(&wacom->battery);
|
power_supply_changed(wacom->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern const struct hid_device_id wacom_ids[];
|
extern const struct hid_device_id wacom_ids[];
|
||||||
|
|
|
@ -944,7 +944,7 @@ static int wacom_battery_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 wacom *wacom = container_of(psy, struct wacom, battery);
|
struct wacom *wacom = power_supply_get_drvdata(psy);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -976,7 +976,7 @@ static int wacom_ac_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 wacom *wacom = container_of(psy, struct wacom, ac);
|
struct wacom *wacom = power_supply_get_drvdata(psy);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -998,42 +998,46 @@ static int wacom_ac_get_property(struct power_supply *psy,
|
||||||
static int wacom_initialize_battery(struct wacom *wacom)
|
static int wacom_initialize_battery(struct wacom *wacom)
|
||||||
{
|
{
|
||||||
static atomic_t battery_no = ATOMIC_INIT(0);
|
static atomic_t battery_no = ATOMIC_INIT(0);
|
||||||
int error;
|
struct power_supply_config psy_cfg = { .drv_data = wacom, };
|
||||||
unsigned long n;
|
unsigned long n;
|
||||||
|
|
||||||
if (wacom->wacom_wac.features.quirks & WACOM_QUIRK_BATTERY) {
|
if (wacom->wacom_wac.features.quirks & WACOM_QUIRK_BATTERY) {
|
||||||
|
struct power_supply_desc *bat_desc = &wacom->battery_desc;
|
||||||
|
struct power_supply_desc *ac_desc = &wacom->ac_desc;
|
||||||
n = atomic_inc_return(&battery_no) - 1;
|
n = atomic_inc_return(&battery_no) - 1;
|
||||||
|
|
||||||
wacom->battery.properties = wacom_battery_props;
|
bat_desc->properties = wacom_battery_props;
|
||||||
wacom->battery.num_properties = ARRAY_SIZE(wacom_battery_props);
|
bat_desc->num_properties = ARRAY_SIZE(wacom_battery_props);
|
||||||
wacom->battery.get_property = wacom_battery_get_property;
|
bat_desc->get_property = wacom_battery_get_property;
|
||||||
sprintf(wacom->wacom_wac.bat_name, "wacom_battery_%ld", n);
|
sprintf(wacom->wacom_wac.bat_name, "wacom_battery_%ld", n);
|
||||||
wacom->battery.name = wacom->wacom_wac.bat_name;
|
bat_desc->name = wacom->wacom_wac.bat_name;
|
||||||
wacom->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
bat_desc->type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
wacom->battery.use_for_apm = 0;
|
bat_desc->use_for_apm = 0;
|
||||||
|
|
||||||
wacom->ac.properties = wacom_ac_props;
|
ac_desc->properties = wacom_ac_props;
|
||||||
wacom->ac.num_properties = ARRAY_SIZE(wacom_ac_props);
|
ac_desc->num_properties = ARRAY_SIZE(wacom_ac_props);
|
||||||
wacom->ac.get_property = wacom_ac_get_property;
|
ac_desc->get_property = wacom_ac_get_property;
|
||||||
sprintf(wacom->wacom_wac.ac_name, "wacom_ac_%ld", n);
|
sprintf(wacom->wacom_wac.ac_name, "wacom_ac_%ld", n);
|
||||||
wacom->ac.name = wacom->wacom_wac.ac_name;
|
ac_desc->name = wacom->wacom_wac.ac_name;
|
||||||
wacom->ac.type = POWER_SUPPLY_TYPE_MAINS;
|
ac_desc->type = POWER_SUPPLY_TYPE_MAINS;
|
||||||
wacom->ac.use_for_apm = 0;
|
ac_desc->use_for_apm = 0;
|
||||||
|
|
||||||
error = power_supply_register(&wacom->hdev->dev,
|
wacom->battery = power_supply_register(&wacom->hdev->dev,
|
||||||
&wacom->battery);
|
&wacom->battery_desc, &psy_cfg);
|
||||||
if (error)
|
if (IS_ERR(wacom->battery))
|
||||||
return error;
|
return PTR_ERR(wacom->battery);
|
||||||
|
|
||||||
power_supply_powers(&wacom->battery, &wacom->hdev->dev);
|
power_supply_powers(wacom->battery, &wacom->hdev->dev);
|
||||||
|
|
||||||
error = power_supply_register(&wacom->hdev->dev, &wacom->ac);
|
wacom->ac = power_supply_register(&wacom->hdev->dev,
|
||||||
if (error) {
|
&wacom->ac_desc,
|
||||||
power_supply_unregister(&wacom->battery);
|
&psy_cfg);
|
||||||
return error;
|
if (IS_ERR(wacom->ac)) {
|
||||||
|
power_supply_unregister(wacom->battery);
|
||||||
|
return PTR_ERR(wacom->ac);
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_powers(&wacom->ac, &wacom->hdev->dev);
|
power_supply_powers(wacom->ac, &wacom->hdev->dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1042,11 +1046,11 @@ static int wacom_initialize_battery(struct wacom *wacom)
|
||||||
static void wacom_destroy_battery(struct wacom *wacom)
|
static void wacom_destroy_battery(struct wacom *wacom)
|
||||||
{
|
{
|
||||||
if ((wacom->wacom_wac.features.quirks & WACOM_QUIRK_BATTERY) &&
|
if ((wacom->wacom_wac.features.quirks & WACOM_QUIRK_BATTERY) &&
|
||||||
wacom->battery.dev) {
|
wacom->battery) {
|
||||||
power_supply_unregister(&wacom->battery);
|
power_supply_unregister(wacom->battery);
|
||||||
wacom->battery.dev = NULL;
|
wacom->battery = NULL;
|
||||||
power_supply_unregister(&wacom->ac);
|
power_supply_unregister(wacom->ac);
|
||||||
wacom->ac.dev = NULL;
|
wacom->ac = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -135,6 +135,15 @@ config AXP288_ADC
|
||||||
device. Depending on platform configuration, this general purpose ADC can
|
device. Depending on platform configuration, this general purpose ADC can
|
||||||
be used for sampling sensors such as thermal resistors.
|
be used for sampling sensors such as thermal resistors.
|
||||||
|
|
||||||
|
config DA9150_GPADC
|
||||||
|
tristate "Dialog DA9150 GPADC driver support"
|
||||||
|
depends on MFD_DA9150
|
||||||
|
help
|
||||||
|
Say yes here to build support for Dialog DA9150 GPADC.
|
||||||
|
|
||||||
|
This driver can also be built as a module. If chosen, the module name
|
||||||
|
will be da9150-gpadc.
|
||||||
|
|
||||||
config CC10001_ADC
|
config CC10001_ADC
|
||||||
tristate "Cosmic Circuits 10001 ADC driver"
|
tristate "Cosmic Circuits 10001 ADC driver"
|
||||||
depends on HAVE_CLK || REGULATOR
|
depends on HAVE_CLK || REGULATOR
|
||||||
|
|
|
@ -15,6 +15,7 @@ obj-$(CONFIG_AD7887) += ad7887.o
|
||||||
obj-$(CONFIG_AD799X) += ad799x.o
|
obj-$(CONFIG_AD799X) += ad799x.o
|
||||||
obj-$(CONFIG_AT91_ADC) += at91_adc.o
|
obj-$(CONFIG_AT91_ADC) += at91_adc.o
|
||||||
obj-$(CONFIG_AXP288_ADC) += axp288_adc.o
|
obj-$(CONFIG_AXP288_ADC) += axp288_adc.o
|
||||||
|
obj-$(CONFIG_DA9150_GPADC) += da9150-gpadc.o
|
||||||
obj-$(CONFIG_CC10001_ADC) += cc10001_adc.o
|
obj-$(CONFIG_CC10001_ADC) += cc10001_adc.o
|
||||||
obj-$(CONFIG_EXYNOS_ADC) += exynos_adc.o
|
obj-$(CONFIG_EXYNOS_ADC) += exynos_adc.o
|
||||||
obj-$(CONFIG_LP8788_ADC) += lp8788_adc.o
|
obj-$(CONFIG_LP8788_ADC) += lp8788_adc.o
|
||||||
|
|
407
drivers/iio/adc/da9150-gpadc.c
Normal file
407
drivers/iio/adc/da9150-gpadc.c
Normal file
|
@ -0,0 +1,407 @@
|
||||||
|
/*
|
||||||
|
* DA9150 GPADC Driver
|
||||||
|
*
|
||||||
|
* Copyright (c) 2014 Dialog Semiconductor
|
||||||
|
*
|
||||||
|
* Author: Adam Thomson <Adam.Thomson.Opensource@diasemi.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/kernel.h>
|
||||||
|
#include <linux/slab.h>
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/platform_device.h>
|
||||||
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/mutex.h>
|
||||||
|
#include <linux/completion.h>
|
||||||
|
#include <linux/iio/iio.h>
|
||||||
|
#include <linux/iio/machine.h>
|
||||||
|
#include <linux/iio/driver.h>
|
||||||
|
#include <linux/mfd/da9150/core.h>
|
||||||
|
#include <linux/mfd/da9150/registers.h>
|
||||||
|
|
||||||
|
/* Channels */
|
||||||
|
enum da9150_gpadc_hw_channel {
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOA_2V = 0,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOA_2V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOB_2V,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOB_2V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOC_2V,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOC_2V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOD_2V,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOD_2V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_IBUS_SENSE,
|
||||||
|
DA9150_GPADC_HW_CHAN_IBUS_SENSE_,
|
||||||
|
DA9150_GPADC_HW_CHAN_VBUS_DIV,
|
||||||
|
DA9150_GPADC_HW_CHAN_VBUS_DIV_,
|
||||||
|
DA9150_GPADC_HW_CHAN_ID,
|
||||||
|
DA9150_GPADC_HW_CHAN_ID_,
|
||||||
|
DA9150_GPADC_HW_CHAN_VSYS,
|
||||||
|
DA9150_GPADC_HW_CHAN_VSYS_,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOA_6V,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOA_6V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOB_6V,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOB_6V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOC_6V,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOC_6V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOD_6V,
|
||||||
|
DA9150_GPADC_HW_CHAN_GPIOD_6V_,
|
||||||
|
DA9150_GPADC_HW_CHAN_VBAT,
|
||||||
|
DA9150_GPADC_HW_CHAN_VBAT_,
|
||||||
|
DA9150_GPADC_HW_CHAN_TBAT,
|
||||||
|
DA9150_GPADC_HW_CHAN_TBAT_,
|
||||||
|
DA9150_GPADC_HW_CHAN_TJUNC_CORE,
|
||||||
|
DA9150_GPADC_HW_CHAN_TJUNC_CORE_,
|
||||||
|
DA9150_GPADC_HW_CHAN_TJUNC_OVP,
|
||||||
|
DA9150_GPADC_HW_CHAN_TJUNC_OVP_,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum da9150_gpadc_channel {
|
||||||
|
DA9150_GPADC_CHAN_GPIOA = 0,
|
||||||
|
DA9150_GPADC_CHAN_GPIOB,
|
||||||
|
DA9150_GPADC_CHAN_GPIOC,
|
||||||
|
DA9150_GPADC_CHAN_GPIOD,
|
||||||
|
DA9150_GPADC_CHAN_IBUS,
|
||||||
|
DA9150_GPADC_CHAN_VBUS,
|
||||||
|
DA9150_GPADC_CHAN_VSYS,
|
||||||
|
DA9150_GPADC_CHAN_VBAT,
|
||||||
|
DA9150_GPADC_CHAN_TBAT,
|
||||||
|
DA9150_GPADC_CHAN_TJUNC_CORE,
|
||||||
|
DA9150_GPADC_CHAN_TJUNC_OVP,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Private data */
|
||||||
|
struct da9150_gpadc {
|
||||||
|
struct da9150 *da9150;
|
||||||
|
struct device *dev;
|
||||||
|
|
||||||
|
struct mutex lock;
|
||||||
|
struct completion complete;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
static irqreturn_t da9150_gpadc_irq(int irq, void *data)
|
||||||
|
{
|
||||||
|
|
||||||
|
struct da9150_gpadc *gpadc = data;
|
||||||
|
|
||||||
|
complete(&gpadc->complete);
|
||||||
|
|
||||||
|
return IRQ_HANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_gpadc_read_adc(struct da9150_gpadc *gpadc, int hw_chan)
|
||||||
|
{
|
||||||
|
u8 result_regs[2];
|
||||||
|
int result;
|
||||||
|
|
||||||
|
mutex_lock(&gpadc->lock);
|
||||||
|
|
||||||
|
/* Set channel & enable measurement */
|
||||||
|
da9150_reg_write(gpadc->da9150, DA9150_GPADC_MAN,
|
||||||
|
(DA9150_GPADC_EN_MASK |
|
||||||
|
hw_chan << DA9150_GPADC_MUX_SHIFT));
|
||||||
|
|
||||||
|
/* Consume left-over completion from a previous timeout */
|
||||||
|
try_wait_for_completion(&gpadc->complete);
|
||||||
|
|
||||||
|
/* Check for actual completion */
|
||||||
|
wait_for_completion_timeout(&gpadc->complete, msecs_to_jiffies(5));
|
||||||
|
|
||||||
|
/* Read result and status from device */
|
||||||
|
da9150_bulk_read(gpadc->da9150, DA9150_GPADC_RES_A, 2, result_regs);
|
||||||
|
|
||||||
|
mutex_unlock(&gpadc->lock);
|
||||||
|
|
||||||
|
/* Check to make sure device really has completed reading */
|
||||||
|
if (result_regs[1] & DA9150_GPADC_RUN_MASK) {
|
||||||
|
dev_err(gpadc->dev, "Timeout on channel %d of GPADC\n",
|
||||||
|
hw_chan);
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* LSBs - 2 bits */
|
||||||
|
result = (result_regs[1] & DA9150_GPADC_RES_L_MASK) >>
|
||||||
|
DA9150_GPADC_RES_L_SHIFT;
|
||||||
|
/* MSBs - 8 bits */
|
||||||
|
result |= result_regs[0] << DA9150_GPADC_RES_L_BITS;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int da9150_gpadc_gpio_6v_voltage_now(int raw_val)
|
||||||
|
{
|
||||||
|
/* Convert to mV */
|
||||||
|
return (6 * ((raw_val * 1000) + 500)) / 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int da9150_gpadc_ibus_current_avg(int raw_val)
|
||||||
|
{
|
||||||
|
/* Convert to mA */
|
||||||
|
return (4 * ((raw_val * 1000) + 500)) / 2048;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int da9150_gpadc_vbus_21v_voltage_now(int raw_val)
|
||||||
|
{
|
||||||
|
/* Convert to mV */
|
||||||
|
return (21 * ((raw_val * 1000) + 500)) / 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int da9150_gpadc_vsys_6v_voltage_now(int raw_val)
|
||||||
|
{
|
||||||
|
/* Convert to mV */
|
||||||
|
return (3 * ((raw_val * 1000) + 500)) / 512;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_gpadc_read_processed(struct da9150_gpadc *gpadc, int channel,
|
||||||
|
int hw_chan, int *val)
|
||||||
|
{
|
||||||
|
int raw_val;
|
||||||
|
|
||||||
|
raw_val = da9150_gpadc_read_adc(gpadc, hw_chan);
|
||||||
|
if (raw_val < 0)
|
||||||
|
return raw_val;
|
||||||
|
|
||||||
|
switch (channel) {
|
||||||
|
case DA9150_GPADC_CHAN_GPIOA:
|
||||||
|
case DA9150_GPADC_CHAN_GPIOB:
|
||||||
|
case DA9150_GPADC_CHAN_GPIOC:
|
||||||
|
case DA9150_GPADC_CHAN_GPIOD:
|
||||||
|
*val = da9150_gpadc_gpio_6v_voltage_now(raw_val);
|
||||||
|
break;
|
||||||
|
case DA9150_GPADC_CHAN_IBUS:
|
||||||
|
*val = da9150_gpadc_ibus_current_avg(raw_val);
|
||||||
|
break;
|
||||||
|
case DA9150_GPADC_CHAN_VBUS:
|
||||||
|
*val = da9150_gpadc_vbus_21v_voltage_now(raw_val);
|
||||||
|
break;
|
||||||
|
case DA9150_GPADC_CHAN_VSYS:
|
||||||
|
*val = da9150_gpadc_vsys_6v_voltage_now(raw_val);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* No processing for other channels so return raw value */
|
||||||
|
*val = raw_val;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return IIO_VAL_INT;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_gpadc_read_scale(int channel, int *val, int *val2)
|
||||||
|
{
|
||||||
|
switch (channel) {
|
||||||
|
case DA9150_GPADC_CHAN_VBAT:
|
||||||
|
*val = 2932;
|
||||||
|
*val2 = 1000;
|
||||||
|
return IIO_VAL_FRACTIONAL;
|
||||||
|
case DA9150_GPADC_CHAN_TJUNC_CORE:
|
||||||
|
case DA9150_GPADC_CHAN_TJUNC_OVP:
|
||||||
|
*val = 1000000;
|
||||||
|
*val2 = 4420;
|
||||||
|
return IIO_VAL_FRACTIONAL;
|
||||||
|
default:
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_gpadc_read_offset(int channel, int *val)
|
||||||
|
{
|
||||||
|
switch (channel) {
|
||||||
|
case DA9150_GPADC_CHAN_VBAT:
|
||||||
|
*val = 1500000 / 2932;
|
||||||
|
return IIO_VAL_INT;
|
||||||
|
case DA9150_GPADC_CHAN_TJUNC_CORE:
|
||||||
|
case DA9150_GPADC_CHAN_TJUNC_OVP:
|
||||||
|
*val = -144;
|
||||||
|
return IIO_VAL_INT;
|
||||||
|
default:
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_gpadc_read_raw(struct iio_dev *indio_dev,
|
||||||
|
struct iio_chan_spec const *chan,
|
||||||
|
int *val, int *val2, long mask)
|
||||||
|
{
|
||||||
|
struct da9150_gpadc *gpadc = iio_priv(indio_dev);
|
||||||
|
|
||||||
|
if ((chan->channel < DA9150_GPADC_CHAN_GPIOA) ||
|
||||||
|
(chan->channel > DA9150_GPADC_CHAN_TJUNC_OVP))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
switch (mask) {
|
||||||
|
case IIO_CHAN_INFO_RAW:
|
||||||
|
case IIO_CHAN_INFO_PROCESSED:
|
||||||
|
return da9150_gpadc_read_processed(gpadc, chan->channel,
|
||||||
|
chan->address, val);
|
||||||
|
case IIO_CHAN_INFO_SCALE:
|
||||||
|
return da9150_gpadc_read_scale(chan->channel, val, val2);
|
||||||
|
case IIO_CHAN_INFO_OFFSET:
|
||||||
|
return da9150_gpadc_read_offset(chan->channel, val);
|
||||||
|
default:
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct iio_info da9150_gpadc_info = {
|
||||||
|
.read_raw = &da9150_gpadc_read_raw,
|
||||||
|
.driver_module = THIS_MODULE,
|
||||||
|
};
|
||||||
|
|
||||||
|
#define DA9150_GPADC_CHANNEL(_id, _hw_id, _type, chan_info, \
|
||||||
|
_ext_name) { \
|
||||||
|
.type = _type, \
|
||||||
|
.indexed = 1, \
|
||||||
|
.channel = DA9150_GPADC_CHAN_##_id, \
|
||||||
|
.address = DA9150_GPADC_HW_CHAN_##_hw_id, \
|
||||||
|
.info_mask_separate = chan_info, \
|
||||||
|
.extend_name = _ext_name, \
|
||||||
|
.datasheet_name = #_id, \
|
||||||
|
}
|
||||||
|
|
||||||
|
#define DA9150_GPADC_CHANNEL_RAW(_id, _hw_id, _type, _ext_name) \
|
||||||
|
DA9150_GPADC_CHANNEL(_id, _hw_id, _type, \
|
||||||
|
BIT(IIO_CHAN_INFO_RAW), _ext_name)
|
||||||
|
|
||||||
|
#define DA9150_GPADC_CHANNEL_SCALED(_id, _hw_id, _type, _ext_name) \
|
||||||
|
DA9150_GPADC_CHANNEL(_id, _hw_id, _type, \
|
||||||
|
BIT(IIO_CHAN_INFO_RAW) | \
|
||||||
|
BIT(IIO_CHAN_INFO_SCALE) | \
|
||||||
|
BIT(IIO_CHAN_INFO_OFFSET), \
|
||||||
|
_ext_name)
|
||||||
|
|
||||||
|
#define DA9150_GPADC_CHANNEL_PROCESSED(_id, _hw_id, _type, _ext_name) \
|
||||||
|
DA9150_GPADC_CHANNEL(_id, _hw_id, _type, \
|
||||||
|
BIT(IIO_CHAN_INFO_PROCESSED), _ext_name)
|
||||||
|
|
||||||
|
/* Supported channels */
|
||||||
|
static const struct iio_chan_spec da9150_gpadc_channels[] = {
|
||||||
|
DA9150_GPADC_CHANNEL_PROCESSED(GPIOA, GPIOA_6V, IIO_VOLTAGE, NULL),
|
||||||
|
DA9150_GPADC_CHANNEL_PROCESSED(GPIOB, GPIOB_6V, IIO_VOLTAGE, NULL),
|
||||||
|
DA9150_GPADC_CHANNEL_PROCESSED(GPIOC, GPIOC_6V, IIO_VOLTAGE, NULL),
|
||||||
|
DA9150_GPADC_CHANNEL_PROCESSED(GPIOD, GPIOD_6V, IIO_VOLTAGE, NULL),
|
||||||
|
DA9150_GPADC_CHANNEL_PROCESSED(IBUS, IBUS_SENSE, IIO_CURRENT, "ibus"),
|
||||||
|
DA9150_GPADC_CHANNEL_PROCESSED(VBUS, VBUS_DIV_, IIO_VOLTAGE, "vbus"),
|
||||||
|
DA9150_GPADC_CHANNEL_PROCESSED(VSYS, VSYS, IIO_VOLTAGE, "vsys"),
|
||||||
|
DA9150_GPADC_CHANNEL_SCALED(VBAT, VBAT, IIO_VOLTAGE, "vbat"),
|
||||||
|
DA9150_GPADC_CHANNEL_RAW(TBAT, TBAT, IIO_VOLTAGE, "tbat"),
|
||||||
|
DA9150_GPADC_CHANNEL_SCALED(TJUNC_CORE, TJUNC_CORE, IIO_TEMP,
|
||||||
|
"tjunc_core"),
|
||||||
|
DA9150_GPADC_CHANNEL_SCALED(TJUNC_OVP, TJUNC_OVP, IIO_TEMP,
|
||||||
|
"tjunc_ovp"),
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Default maps used by da9150-charger */
|
||||||
|
static struct iio_map da9150_gpadc_default_maps[] = {
|
||||||
|
{
|
||||||
|
.consumer_dev_name = "da9150-charger",
|
||||||
|
.consumer_channel = "CHAN_IBUS",
|
||||||
|
.adc_channel_label = "IBUS",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.consumer_dev_name = "da9150-charger",
|
||||||
|
.consumer_channel = "CHAN_VBUS",
|
||||||
|
.adc_channel_label = "VBUS",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.consumer_dev_name = "da9150-charger",
|
||||||
|
.consumer_channel = "CHAN_TJUNC",
|
||||||
|
.adc_channel_label = "TJUNC_CORE",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.consumer_dev_name = "da9150-charger",
|
||||||
|
.consumer_channel = "CHAN_VBAT",
|
||||||
|
.adc_channel_label = "VBAT",
|
||||||
|
},
|
||||||
|
{},
|
||||||
|
};
|
||||||
|
|
||||||
|
static int da9150_gpadc_probe(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct device *dev = &pdev->dev;
|
||||||
|
struct da9150 *da9150 = dev_get_drvdata(dev->parent);
|
||||||
|
struct da9150_gpadc *gpadc;
|
||||||
|
struct iio_dev *indio_dev;
|
||||||
|
int irq, ret;
|
||||||
|
|
||||||
|
indio_dev = devm_iio_device_alloc(dev, sizeof(*gpadc));
|
||||||
|
if (!indio_dev) {
|
||||||
|
dev_err(&pdev->dev, "Failed to allocate IIO device\n");
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
gpadc = iio_priv(indio_dev);
|
||||||
|
|
||||||
|
platform_set_drvdata(pdev, indio_dev);
|
||||||
|
gpadc->da9150 = da9150;
|
||||||
|
gpadc->dev = dev;
|
||||||
|
mutex_init(&gpadc->lock);
|
||||||
|
init_completion(&gpadc->complete);
|
||||||
|
|
||||||
|
irq = platform_get_irq_byname(pdev, "GPADC");
|
||||||
|
if (irq < 0) {
|
||||||
|
dev_err(dev, "Failed to get IRQ: %d\n", irq);
|
||||||
|
return irq;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = devm_request_threaded_irq(dev, irq, NULL, da9150_gpadc_irq,
|
||||||
|
IRQF_ONESHOT, "GPADC", gpadc);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(dev, "Failed to request IRQ %d: %d\n", irq, ret);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = iio_map_array_register(indio_dev, da9150_gpadc_default_maps);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(dev, "Failed to register IIO maps: %d\n", ret);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
indio_dev->name = dev_name(dev);
|
||||||
|
indio_dev->dev.parent = dev;
|
||||||
|
indio_dev->dev.of_node = pdev->dev.of_node;
|
||||||
|
indio_dev->info = &da9150_gpadc_info;
|
||||||
|
indio_dev->modes = INDIO_DIRECT_MODE;
|
||||||
|
indio_dev->channels = da9150_gpadc_channels;
|
||||||
|
indio_dev->num_channels = ARRAY_SIZE(da9150_gpadc_channels);
|
||||||
|
|
||||||
|
ret = iio_device_register(indio_dev);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(dev, "Failed to register IIO device: %d\n", ret);
|
||||||
|
goto iio_map_unreg;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
iio_map_unreg:
|
||||||
|
iio_map_array_unregister(indio_dev);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_gpadc_remove(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct iio_dev *indio_dev = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
|
iio_device_unregister(indio_dev);
|
||||||
|
iio_map_array_unregister(indio_dev);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct platform_driver da9150_gpadc_driver = {
|
||||||
|
.driver = {
|
||||||
|
.name = "da9150-gpadc",
|
||||||
|
},
|
||||||
|
.probe = da9150_gpadc_probe,
|
||||||
|
.remove = da9150_gpadc_remove,
|
||||||
|
};
|
||||||
|
|
||||||
|
module_platform_driver(da9150_gpadc_driver);
|
||||||
|
|
||||||
|
MODULE_DESCRIPTION("GPADC Driver for DA9150");
|
||||||
|
MODULE_AUTHOR("Adam Thomson <Adam.Thomson.Opensource@diasemi.com>");
|
||||||
|
MODULE_LICENSE("GPL");
|
|
@ -49,7 +49,9 @@ static void ab8500_power_off(void)
|
||||||
if (!psy)
|
if (!psy)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_ONLINE, &val);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_ONLINE,
|
||||||
|
&val);
|
||||||
|
power_supply_put(psy);
|
||||||
|
|
||||||
if (!ret && val.intval) {
|
if (!ret && val.intval) {
|
||||||
charger_present = true;
|
charger_present = true;
|
||||||
|
@ -63,8 +65,8 @@ static void ab8500_power_off(void)
|
||||||
/* Check if battery is known */
|
/* Check if battery is known */
|
||||||
psy = power_supply_get_by_name("ab8500_btemp");
|
psy = power_supply_get_by_name("ab8500_btemp");
|
||||||
if (psy) {
|
if (psy) {
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_TECHNOLOGY,
|
ret = power_supply_get_property(psy,
|
||||||
&val);
|
POWER_SUPPLY_PROP_TECHNOLOGY, &val);
|
||||||
if (!ret && val.intval != POWER_SUPPLY_TECHNOLOGY_UNKNOWN) {
|
if (!ret && val.intval != POWER_SUPPLY_TECHNOLOGY_UNKNOWN) {
|
||||||
printk(KERN_INFO
|
printk(KERN_INFO
|
||||||
"Charger \"%s\" is connected with known battery."
|
"Charger \"%s\" is connected with known battery."
|
||||||
|
@ -72,6 +74,7 @@ static void ab8500_power_off(void)
|
||||||
pss[i]);
|
pss[i]);
|
||||||
machine_restart("charging");
|
machine_restart("charging");
|
||||||
}
|
}
|
||||||
|
power_supply_put(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
shutdown:
|
shutdown:
|
||||||
|
|
|
@ -87,7 +87,7 @@ static struct resource axp20x_pek_resources[] = {
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct resource axp288_battery_resources[] = {
|
static struct resource axp288_fuel_gauge_resources[] = {
|
||||||
{
|
{
|
||||||
.start = AXP288_IRQ_QWBTU,
|
.start = AXP288_IRQ_QWBTU,
|
||||||
.end = AXP288_IRQ_QWBTU,
|
.end = AXP288_IRQ_QWBTU,
|
||||||
|
@ -350,9 +350,9 @@ static struct mfd_cell axp288_cells[] = {
|
||||||
.resources = axp288_charger_resources,
|
.resources = axp288_charger_resources,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
.name = "axp288_battery",
|
.name = "axp288_fuel_gauge",
|
||||||
.num_resources = ARRAY_SIZE(axp288_battery_resources),
|
.num_resources = ARRAY_SIZE(axp288_fuel_gauge_resources),
|
||||||
.resources = axp288_battery_resources,
|
.resources = axp288_fuel_gauge_resources,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
.name = "axp288_pmic_acpi",
|
.name = "axp288_pmic_acpi",
|
||||||
|
|
|
@ -177,7 +177,7 @@ struct compal_data{
|
||||||
unsigned char curr_pwm;
|
unsigned char curr_pwm;
|
||||||
|
|
||||||
/* Power supply */
|
/* Power supply */
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
struct power_supply_info psy_info;
|
struct power_supply_info psy_info;
|
||||||
char bat_model_name[BAT_MODEL_NAME_LEN + 1];
|
char bat_model_name[BAT_MODEL_NAME_LEN + 1];
|
||||||
char bat_manufacturer_name[BAT_MANUFACTURER_NAME_LEN + 1];
|
char bat_manufacturer_name[BAT_MANUFACTURER_NAME_LEN + 1];
|
||||||
|
@ -565,8 +565,7 @@ static int bat_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 compal_data *data;
|
struct compal_data *data = power_supply_get_drvdata(psy);
|
||||||
data = container_of(psy, struct compal_data, psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -875,13 +874,16 @@ static struct dmi_system_id __initdata compal_dmi_table[] = {
|
||||||
};
|
};
|
||||||
MODULE_DEVICE_TABLE(dmi, compal_dmi_table);
|
MODULE_DEVICE_TABLE(dmi, compal_dmi_table);
|
||||||
|
|
||||||
|
static const struct power_supply_desc psy_bat_desc = {
|
||||||
|
.name = DRIVER_NAME,
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = compal_bat_properties,
|
||||||
|
.num_properties = ARRAY_SIZE(compal_bat_properties),
|
||||||
|
.get_property = bat_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
static void initialize_power_supply_data(struct compal_data *data)
|
static void initialize_power_supply_data(struct compal_data *data)
|
||||||
{
|
{
|
||||||
data->psy.name = DRIVER_NAME;
|
|
||||||
data->psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
|
||||||
data->psy.properties = compal_bat_properties;
|
|
||||||
data->psy.num_properties = ARRAY_SIZE(compal_bat_properties);
|
|
||||||
data->psy.get_property = bat_get_property;
|
|
||||||
|
|
||||||
ec_read_sequence(BAT_MANUFACTURER_NAME_ADDR,
|
ec_read_sequence(BAT_MANUFACTURER_NAME_ADDR,
|
||||||
data->bat_manufacturer_name,
|
data->bat_manufacturer_name,
|
||||||
|
@ -1011,6 +1013,7 @@ static int compal_probe(struct platform_device *pdev)
|
||||||
int err;
|
int err;
|
||||||
struct compal_data *data;
|
struct compal_data *data;
|
||||||
struct device *hwmon_dev;
|
struct device *hwmon_dev;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
if (!extra_features)
|
if (!extra_features)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1026,9 +1029,9 @@ static int compal_probe(struct platform_device *pdev)
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
hwmon_dev = hwmon_device_register_with_groups(&pdev->dev,
|
hwmon_dev = devm_hwmon_device_register_with_groups(&pdev->dev,
|
||||||
"compal", data,
|
"compal", data,
|
||||||
compal_hwmon_groups);
|
compal_hwmon_groups);
|
||||||
if (IS_ERR(hwmon_dev)) {
|
if (IS_ERR(hwmon_dev)) {
|
||||||
err = PTR_ERR(hwmon_dev);
|
err = PTR_ERR(hwmon_dev);
|
||||||
goto remove;
|
goto remove;
|
||||||
|
@ -1036,7 +1039,13 @@ static int compal_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
/* Power supply */
|
/* Power supply */
|
||||||
initialize_power_supply_data(data);
|
initialize_power_supply_data(data);
|
||||||
power_supply_register(&compal_device->dev, &data->psy);
|
psy_cfg.drv_data = data;
|
||||||
|
data->psy = power_supply_register(&compal_device->dev, &psy_bat_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(data->psy)) {
|
||||||
|
err = PTR_ERR(data->psy);
|
||||||
|
goto remove;
|
||||||
|
}
|
||||||
|
|
||||||
platform_set_drvdata(pdev, data);
|
platform_set_drvdata(pdev, data);
|
||||||
|
|
||||||
|
@ -1071,7 +1080,7 @@ static int compal_remove(struct platform_device *pdev)
|
||||||
pwm_disable_control();
|
pwm_disable_control();
|
||||||
|
|
||||||
data = platform_get_drvdata(pdev);
|
data = platform_get_drvdata(pdev);
|
||||||
power_supply_unregister(&data->psy);
|
power_supply_unregister(data->psy);
|
||||||
|
|
||||||
sysfs_remove_group(&pdev->dev.kobj, &compal_platform_attr_group);
|
sysfs_remove_group(&pdev->dev.kobj, &compal_platform_attr_group);
|
||||||
|
|
||||||
|
|
|
@ -98,7 +98,7 @@ struct pm860x_battery_info {
|
||||||
struct i2c_client *i2c;
|
struct i2c_client *i2c;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
|
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
int status;
|
int status;
|
||||||
int irq_cc;
|
int irq_cc;
|
||||||
|
@ -798,9 +798,8 @@ out:
|
||||||
|
|
||||||
static void pm860x_external_power_changed(struct power_supply *psy)
|
static void pm860x_external_power_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct pm860x_battery_info *info;
|
struct pm860x_battery_info *info = dev_get_drvdata(psy->dev.parent);
|
||||||
|
|
||||||
info = container_of(psy, struct pm860x_battery_info, battery);
|
|
||||||
calc_resistor(info);
|
calc_resistor(info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -808,7 +807,7 @@ static int pm860x_batt_get_prop(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 pm860x_battery_info *info = dev_get_drvdata(psy->dev->parent);
|
struct pm860x_battery_info *info = dev_get_drvdata(psy->dev.parent);
|
||||||
int data;
|
int data;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -874,7 +873,7 @@ static int pm860x_batt_set_prop(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)
|
||||||
{
|
{
|
||||||
struct pm860x_battery_info *info = dev_get_drvdata(psy->dev->parent);
|
struct pm860x_battery_info *info = dev_get_drvdata(psy->dev.parent);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_CHARGE_FULL:
|
case POWER_SUPPLY_PROP_CHARGE_FULL:
|
||||||
|
@ -901,6 +900,16 @@ static enum power_supply_property pm860x_batt_props[] = {
|
||||||
POWER_SUPPLY_PROP_TEMP,
|
POWER_SUPPLY_PROP_TEMP,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc pm860x_battery_desc = {
|
||||||
|
.name = "battery-monitor",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = pm860x_batt_props,
|
||||||
|
.num_properties = ARRAY_SIZE(pm860x_batt_props),
|
||||||
|
.get_property = pm860x_batt_get_prop,
|
||||||
|
.set_property = pm860x_batt_set_prop,
|
||||||
|
.external_power_changed = pm860x_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static int pm860x_battery_probe(struct platform_device *pdev)
|
static int pm860x_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct pm860x_chip *chip = dev_get_drvdata(pdev->dev.parent);
|
struct pm860x_chip *chip = dev_get_drvdata(pdev->dev.parent);
|
||||||
|
@ -936,14 +945,6 @@ static int pm860x_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
pm860x_init_battery(info);
|
pm860x_init_battery(info);
|
||||||
|
|
||||||
info->battery.name = "battery-monitor";
|
|
||||||
info->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
|
||||||
info->battery.properties = pm860x_batt_props;
|
|
||||||
info->battery.num_properties = ARRAY_SIZE(pm860x_batt_props);
|
|
||||||
info->battery.get_property = pm860x_batt_get_prop;
|
|
||||||
info->battery.set_property = pm860x_batt_set_prop;
|
|
||||||
info->battery.external_power_changed = pm860x_external_power_changed;
|
|
||||||
|
|
||||||
if (pdata && pdata->max_capacity)
|
if (pdata && pdata->max_capacity)
|
||||||
info->max_capacity = pdata->max_capacity;
|
info->max_capacity = pdata->max_capacity;
|
||||||
else
|
else
|
||||||
|
@ -953,10 +954,11 @@ static int pm860x_battery_probe(struct platform_device *pdev)
|
||||||
else
|
else
|
||||||
info->resistor = 300; /* set default internal resistor */
|
info->resistor = 300; /* set default internal resistor */
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &info->battery);
|
info->battery = power_supply_register(&pdev->dev, &pm860x_battery_desc,
|
||||||
if (ret)
|
NULL);
|
||||||
return ret;
|
if (IS_ERR(info->battery))
|
||||||
info->battery.dev->parent = &pdev->dev;
|
return PTR_ERR(info->battery);
|
||||||
|
info->battery->dev.parent = &pdev->dev;
|
||||||
|
|
||||||
ret = request_threaded_irq(info->irq_cc, NULL,
|
ret = request_threaded_irq(info->irq_cc, NULL,
|
||||||
pm860x_coulomb_handler, IRQF_ONESHOT,
|
pm860x_coulomb_handler, IRQF_ONESHOT,
|
||||||
|
@ -981,7 +983,7 @@ static int pm860x_battery_probe(struct platform_device *pdev)
|
||||||
out_coulomb:
|
out_coulomb:
|
||||||
free_irq(info->irq_cc, info);
|
free_irq(info->irq_cc, info);
|
||||||
out_reg:
|
out_reg:
|
||||||
power_supply_unregister(&info->battery);
|
power_supply_unregister(info->battery);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -991,7 +993,7 @@ static int pm860x_battery_remove(struct platform_device *pdev)
|
||||||
|
|
||||||
free_irq(info->irq_batt, info);
|
free_irq(info->irq_batt, info);
|
||||||
free_irq(info->irq_cc, info);
|
free_irq(info->irq_cc, info);
|
||||||
power_supply_unregister(&info->battery);
|
power_supply_unregister(info->battery);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -102,7 +102,7 @@ struct pm860x_charger_info {
|
||||||
struct i2c_client *i2c_8606;
|
struct i2c_client *i2c_8606;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
|
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
int irq_nums;
|
int irq_nums;
|
||||||
int irq[7];
|
int irq[7];
|
||||||
|
@ -296,14 +296,20 @@ static int set_charging_fsm(struct pm860x_charger_info *info)
|
||||||
psy = power_supply_get_by_name(pm860x_supplied_to[0]);
|
psy = power_supply_get_by_name(pm860x_supplied_to[0]);
|
||||||
if (!psy)
|
if (!psy)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_VOLTAGE_NOW, &data);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
||||||
if (ret)
|
&data);
|
||||||
|
if (ret) {
|
||||||
|
power_supply_put(psy);
|
||||||
return ret;
|
return ret;
|
||||||
|
}
|
||||||
vbatt = data.intval / 1000;
|
vbatt = data.intval / 1000;
|
||||||
|
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_PRESENT, &data);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_PRESENT, &data);
|
||||||
if (ret)
|
if (ret) {
|
||||||
|
power_supply_put(psy);
|
||||||
return ret;
|
return ret;
|
||||||
|
}
|
||||||
|
power_supply_put(psy);
|
||||||
|
|
||||||
mutex_lock(&info->lock);
|
mutex_lock(&info->lock);
|
||||||
info->present = data.intval;
|
info->present = data.intval;
|
||||||
|
@ -414,7 +420,7 @@ static irqreturn_t pm860x_charger_handler(int irq, void *data)
|
||||||
|
|
||||||
set_charging_fsm(info);
|
set_charging_fsm(info);
|
||||||
|
|
||||||
power_supply_changed(&info->usb);
|
power_supply_changed(info->usb);
|
||||||
out:
|
out:
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -430,7 +436,7 @@ static irqreturn_t pm860x_temp_handler(int irq, void *data)
|
||||||
psy = power_supply_get_by_name(pm860x_supplied_to[0]);
|
psy = power_supply_get_by_name(pm860x_supplied_to[0]);
|
||||||
if (!psy)
|
if (!psy)
|
||||||
goto out;
|
goto out;
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_TEMP, &temp);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_TEMP, &temp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto out;
|
||||||
value = temp.intval / 10;
|
value = temp.intval / 10;
|
||||||
|
@ -446,6 +452,7 @@ static irqreturn_t pm860x_temp_handler(int irq, void *data)
|
||||||
|
|
||||||
set_charging_fsm(info);
|
set_charging_fsm(info);
|
||||||
out:
|
out:
|
||||||
|
power_supply_put(psy);
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -485,9 +492,10 @@ static irqreturn_t pm860x_done_handler(int irq, void *data)
|
||||||
psy = power_supply_get_by_name(pm860x_supplied_to[0]);
|
psy = power_supply_get_by_name(pm860x_supplied_to[0]);
|
||||||
if (!psy)
|
if (!psy)
|
||||||
goto out;
|
goto out;
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_VOLTAGE_NOW, &val);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
||||||
|
&val);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto out_psy_put;
|
||||||
vbatt = val.intval / 1000;
|
vbatt = val.intval / 1000;
|
||||||
/*
|
/*
|
||||||
* CHG_DONE interrupt is faster than CHG_DET interrupt when
|
* CHG_DONE interrupt is faster than CHG_DET interrupt when
|
||||||
|
@ -498,10 +506,13 @@ static irqreturn_t pm860x_done_handler(int irq, void *data)
|
||||||
*/
|
*/
|
||||||
ret = pm860x_reg_read(info->i2c, PM8607_STATUS_2);
|
ret = pm860x_reg_read(info->i2c, PM8607_STATUS_2);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out;
|
goto out_psy_put;
|
||||||
if (vbatt > CHARGE_THRESHOLD && ret & STATUS2_CHG)
|
if (vbatt > CHARGE_THRESHOLD && ret & STATUS2_CHG)
|
||||||
psy->set_property(psy, POWER_SUPPLY_PROP_CHARGE_FULL, &val);
|
power_supply_set_property(psy, POWER_SUPPLY_PROP_CHARGE_FULL,
|
||||||
|
&val);
|
||||||
|
|
||||||
|
out_psy_put:
|
||||||
|
power_supply_put(psy);
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&info->lock);
|
mutex_unlock(&info->lock);
|
||||||
dev_dbg(info->dev, "%s, Allowed: %d\n", __func__, info->allowed);
|
dev_dbg(info->dev, "%s, Allowed: %d\n", __func__, info->allowed);
|
||||||
|
@ -584,8 +595,7 @@ static int pm860x_usb_get_prop(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 pm860x_charger_info *info =
|
struct pm860x_charger_info *info = power_supply_get_drvdata(psy);
|
||||||
dev_get_drvdata(psy->dev->parent);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -645,9 +655,18 @@ static struct pm860x_irq_desc {
|
||||||
{ "vchg", pm860x_vchg_handler },
|
{ "vchg", pm860x_vchg_handler },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc pm860x_charger_desc = {
|
||||||
|
.name = "usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = pm860x_usb_props,
|
||||||
|
.num_properties = ARRAY_SIZE(pm860x_usb_props),
|
||||||
|
.get_property = pm860x_usb_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
static int pm860x_charger_probe(struct platform_device *pdev)
|
static int pm860x_charger_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct pm860x_chip *chip = dev_get_drvdata(pdev->dev.parent);
|
struct pm860x_chip *chip = dev_get_drvdata(pdev->dev.parent);
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct pm860x_charger_info *info;
|
struct pm860x_charger_info *info;
|
||||||
int ret;
|
int ret;
|
||||||
int count;
|
int count;
|
||||||
|
@ -685,16 +704,15 @@ static int pm860x_charger_probe(struct platform_device *pdev)
|
||||||
mutex_init(&info->lock);
|
mutex_init(&info->lock);
|
||||||
platform_set_drvdata(pdev, info);
|
platform_set_drvdata(pdev, info);
|
||||||
|
|
||||||
info->usb.name = "usb";
|
psy_cfg.drv_data = info;
|
||||||
info->usb.type = POWER_SUPPLY_TYPE_USB;
|
psy_cfg.supplied_to = pm860x_supplied_to;
|
||||||
info->usb.supplied_to = pm860x_supplied_to;
|
psy_cfg.num_supplicants = ARRAY_SIZE(pm860x_supplied_to);
|
||||||
info->usb.num_supplicants = ARRAY_SIZE(pm860x_supplied_to);
|
info->usb = power_supply_register(&pdev->dev, &pm860x_charger_desc,
|
||||||
info->usb.properties = pm860x_usb_props;
|
&psy_cfg);
|
||||||
info->usb.num_properties = ARRAY_SIZE(pm860x_usb_props);
|
if (IS_ERR(info->usb)) {
|
||||||
info->usb.get_property = pm860x_usb_get_prop;
|
ret = PTR_ERR(info->usb);
|
||||||
ret = power_supply_register(&pdev->dev, &info->usb);
|
|
||||||
if (ret)
|
|
||||||
goto out;
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
pm860x_init_charger(info);
|
pm860x_init_charger(info);
|
||||||
|
|
||||||
|
@ -711,7 +729,7 @@ static int pm860x_charger_probe(struct platform_device *pdev)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
out_irq:
|
out_irq:
|
||||||
power_supply_unregister(&info->usb);
|
power_supply_unregister(info->usb);
|
||||||
while (--i >= 0)
|
while (--i >= 0)
|
||||||
free_irq(info->irq[i], info);
|
free_irq(info->irq[i], info);
|
||||||
out:
|
out:
|
||||||
|
@ -723,7 +741,7 @@ static int pm860x_charger_remove(struct platform_device *pdev)
|
||||||
struct pm860x_charger_info *info = platform_get_drvdata(pdev);
|
struct pm860x_charger_info *info = platform_get_drvdata(pdev);
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
power_supply_unregister(&info->usb);
|
power_supply_unregister(info->usb);
|
||||||
free_irq(info->irq[0], info);
|
free_irq(info->irq[0], info);
|
||||||
for (i = 0; i < info->irq_nums; i++)
|
for (i = 0; i < info->irq_nums; i++)
|
||||||
free_irq(info->irq[i], info);
|
free_irq(info->irq[i], info);
|
||||||
|
|
|
@ -192,6 +192,27 @@ config BATTERY_DA9052
|
||||||
Say Y here to enable support for batteries charger integrated into
|
Say Y here to enable support for batteries charger integrated into
|
||||||
DA9052 PMIC.
|
DA9052 PMIC.
|
||||||
|
|
||||||
|
config CHARGER_DA9150
|
||||||
|
tristate "Dialog Semiconductor DA9150 Charger support"
|
||||||
|
depends on MFD_DA9150
|
||||||
|
depends on DA9150_GPADC
|
||||||
|
depends on IIO
|
||||||
|
help
|
||||||
|
Say Y here to enable support for charger unit of the DA9150
|
||||||
|
Integrated Charger & Fuel-Gauge IC.
|
||||||
|
|
||||||
|
This driver can also be built as a module. If so, the module will be
|
||||||
|
called da9150-charger.
|
||||||
|
|
||||||
|
config AXP288_FUEL_GAUGE
|
||||||
|
tristate "X-Powers AXP288 Fuel Gauge"
|
||||||
|
depends on MFD_AXP20X && IIO
|
||||||
|
help
|
||||||
|
Say yes here to have support for X-Power power management IC (PMIC)
|
||||||
|
Fuel Gauge. The device provides battery statistics and status
|
||||||
|
monitoring as well as alerts for battery over/under voltage and
|
||||||
|
over/under temperature.
|
||||||
|
|
||||||
config BATTERY_MAX17040
|
config BATTERY_MAX17040
|
||||||
tristate "Maxim MAX17040 Fuel Gauge"
|
tristate "Maxim MAX17040 Fuel Gauge"
|
||||||
depends on I2C
|
depends on I2C
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
ccflags-$(CONFIG_POWER_SUPPLY_DEBUG) := -DDEBUG
|
subdir-ccflags-$(CONFIG_POWER_SUPPLY_DEBUG) := -DDEBUG
|
||||||
|
|
||||||
power_supply-y := power_supply_core.o
|
power_supply-y := power_supply_core.o
|
||||||
power_supply-$(CONFIG_SYSFS) += power_supply_sysfs.o
|
power_supply-$(CONFIG_SYSFS) += power_supply_sysfs.o
|
||||||
|
@ -32,6 +32,7 @@ obj-$(CONFIG_BATTERY_SBS) += sbs-battery.o
|
||||||
obj-$(CONFIG_BATTERY_BQ27x00) += bq27x00_battery.o
|
obj-$(CONFIG_BATTERY_BQ27x00) += bq27x00_battery.o
|
||||||
obj-$(CONFIG_BATTERY_DA9030) += da9030_battery.o
|
obj-$(CONFIG_BATTERY_DA9030) += da9030_battery.o
|
||||||
obj-$(CONFIG_BATTERY_DA9052) += da9052-battery.o
|
obj-$(CONFIG_BATTERY_DA9052) += da9052-battery.o
|
||||||
|
obj-$(CONFIG_CHARGER_DA9150) += da9150-charger.o
|
||||||
obj-$(CONFIG_BATTERY_MAX17040) += max17040_battery.o
|
obj-$(CONFIG_BATTERY_MAX17040) += max17040_battery.o
|
||||||
obj-$(CONFIG_BATTERY_MAX17042) += max17042_battery.o
|
obj-$(CONFIG_BATTERY_MAX17042) += max17042_battery.o
|
||||||
obj-$(CONFIG_BATTERY_Z2) += z2_battery.o
|
obj-$(CONFIG_BATTERY_Z2) += z2_battery.o
|
||||||
|
@ -62,3 +63,4 @@ obj-$(CONFIG_POWER_AVS) += avs/
|
||||||
obj-$(CONFIG_CHARGER_SMB347) += smb347-charger.o
|
obj-$(CONFIG_CHARGER_SMB347) += smb347-charger.o
|
||||||
obj-$(CONFIG_CHARGER_TPS65090) += tps65090-charger.o
|
obj-$(CONFIG_CHARGER_TPS65090) += tps65090-charger.o
|
||||||
obj-$(CONFIG_POWER_RESET) += reset/
|
obj-$(CONFIG_POWER_RESET) += reset/
|
||||||
|
obj-$(CONFIG_AXP288_FUEL_GAUGE) += axp288_fuel_gauge.o
|
||||||
|
|
|
@ -45,9 +45,6 @@
|
||||||
#define BTEMP_BATCTRL_CURR_SRC_60UA 60
|
#define BTEMP_BATCTRL_CURR_SRC_60UA 60
|
||||||
#define BTEMP_BATCTRL_CURR_SRC_120UA 120
|
#define BTEMP_BATCTRL_CURR_SRC_120UA 120
|
||||||
|
|
||||||
#define to_ab8500_btemp_device_info(x) container_of((x), \
|
|
||||||
struct ab8500_btemp, btemp_psy);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct ab8500_btemp_interrupts - ab8500 interrupts
|
* struct ab8500_btemp_interrupts - ab8500 interrupts
|
||||||
* @name: name of the interrupt
|
* @name: name of the interrupt
|
||||||
|
@ -102,7 +99,7 @@ struct ab8500_btemp {
|
||||||
struct ab8500_gpadc *gpadc;
|
struct ab8500_gpadc *gpadc;
|
||||||
struct ab8500_fg *fg;
|
struct ab8500_fg *fg;
|
||||||
struct abx500_bm_data *bm;
|
struct abx500_bm_data *bm;
|
||||||
struct power_supply btemp_psy;
|
struct power_supply *btemp_psy;
|
||||||
struct ab8500_btemp_events events;
|
struct ab8500_btemp_events events;
|
||||||
struct ab8500_btemp_ranges btemp_ranges;
|
struct ab8500_btemp_ranges btemp_ranges;
|
||||||
struct workqueue_struct *btemp_wq;
|
struct workqueue_struct *btemp_wq;
|
||||||
|
@ -654,14 +651,14 @@ static void ab8500_btemp_periodic_work(struct work_struct *work)
|
||||||
if ((di->bat_temp != di->prev_bat_temp) || !di->initialized) {
|
if ((di->bat_temp != di->prev_bat_temp) || !di->initialized) {
|
||||||
di->initialized = true;
|
di->initialized = true;
|
||||||
di->bat_temp = bat_temp;
|
di->bat_temp = bat_temp;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
}
|
}
|
||||||
} else if (bat_temp < di->prev_bat_temp) {
|
} else if (bat_temp < di->prev_bat_temp) {
|
||||||
di->bat_temp--;
|
di->bat_temp--;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
} else if (bat_temp > di->prev_bat_temp) {
|
} else if (bat_temp > di->prev_bat_temp) {
|
||||||
di->bat_temp++;
|
di->bat_temp++;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
}
|
}
|
||||||
di->prev_bat_temp = bat_temp;
|
di->prev_bat_temp = bat_temp;
|
||||||
|
|
||||||
|
@ -689,7 +686,7 @@ static irqreturn_t ab8500_btemp_batctrlindb_handler(int irq, void *_di)
|
||||||
dev_err(di->dev, "Battery removal detected!\n");
|
dev_err(di->dev, "Battery removal detected!\n");
|
||||||
|
|
||||||
di->events.batt_rem = true;
|
di->events.batt_rem = true;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -715,7 +712,7 @@ static irqreturn_t ab8500_btemp_templow_handler(int irq, void *_di)
|
||||||
di->events.btemp_high = false;
|
di->events.btemp_high = false;
|
||||||
di->events.btemp_medhigh = false;
|
di->events.btemp_medhigh = false;
|
||||||
di->events.btemp_lowmed = false;
|
di->events.btemp_lowmed = false;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
|
@ -738,7 +735,7 @@ static irqreturn_t ab8500_btemp_temphigh_handler(int irq, void *_di)
|
||||||
di->events.btemp_medhigh = false;
|
di->events.btemp_medhigh = false;
|
||||||
di->events.btemp_lowmed = false;
|
di->events.btemp_lowmed = false;
|
||||||
di->events.btemp_low = false;
|
di->events.btemp_low = false;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -760,7 +757,7 @@ static irqreturn_t ab8500_btemp_lowmed_handler(int irq, void *_di)
|
||||||
di->events.btemp_medhigh = false;
|
di->events.btemp_medhigh = false;
|
||||||
di->events.btemp_high = false;
|
di->events.btemp_high = false;
|
||||||
di->events.btemp_low = false;
|
di->events.btemp_low = false;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -782,7 +779,7 @@ static irqreturn_t ab8500_btemp_medhigh_handler(int irq, void *_di)
|
||||||
di->events.btemp_lowmed = false;
|
di->events.btemp_lowmed = false;
|
||||||
di->events.btemp_high = false;
|
di->events.btemp_high = false;
|
||||||
di->events.btemp_low = false;
|
di->events.btemp_low = false;
|
||||||
power_supply_changed(&di->btemp_psy);
|
power_supply_changed(di->btemp_psy);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -884,9 +881,7 @@ static int ab8500_btemp_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 ab8500_btemp *di;
|
struct ab8500_btemp *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_btemp_device_info(psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_PRESENT:
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
|
@ -919,14 +914,14 @@ static int ab8500_btemp_get_ext_psy_data(struct device *dev, void *data)
|
||||||
|
|
||||||
psy = (struct power_supply *)data;
|
psy = (struct power_supply *)data;
|
||||||
ext = dev_get_drvdata(dev);
|
ext = dev_get_drvdata(dev);
|
||||||
di = to_ab8500_btemp_device_info(psy);
|
di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For all psy where the name of your driver
|
* For all psy where the name of your driver
|
||||||
* appears in any supplied_to
|
* appears in any supplied_to
|
||||||
*/
|
*/
|
||||||
for (i = 0; i < ext->num_supplicants; i++) {
|
for (i = 0; i < ext->num_supplicants; i++) {
|
||||||
if (!strcmp(ext->supplied_to[i], psy->name))
|
if (!strcmp(ext->supplied_to[i], psy->desc->name))
|
||||||
psy_found = true;
|
psy_found = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -934,16 +929,16 @@ static int ab8500_btemp_get_ext_psy_data(struct device *dev, void *data)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* Go through all properties for the psy */
|
/* Go through all properties for the psy */
|
||||||
for (j = 0; j < ext->num_properties; j++) {
|
for (j = 0; j < ext->desc->num_properties; j++) {
|
||||||
enum power_supply_property prop;
|
enum power_supply_property prop;
|
||||||
prop = ext->properties[j];
|
prop = ext->desc->properties[j];
|
||||||
|
|
||||||
if (ext->get_property(ext, prop, &ret))
|
if (power_supply_get_property(ext, prop, &ret))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
case POWER_SUPPLY_PROP_PRESENT:
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_MAINS:
|
case POWER_SUPPLY_TYPE_MAINS:
|
||||||
/* AC disconnected */
|
/* AC disconnected */
|
||||||
if (!ret.intval && di->events.ac_conn) {
|
if (!ret.intval && di->events.ac_conn) {
|
||||||
|
@ -990,10 +985,10 @@ static int ab8500_btemp_get_ext_psy_data(struct device *dev, void *data)
|
||||||
*/
|
*/
|
||||||
static void ab8500_btemp_external_power_changed(struct power_supply *psy)
|
static void ab8500_btemp_external_power_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct ab8500_btemp *di = to_ab8500_btemp_device_info(psy);
|
struct ab8500_btemp *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
class_for_each_device(power_supply_class, NULL,
|
class_for_each_device(power_supply_class, NULL,
|
||||||
&di->btemp_psy, ab8500_btemp_get_ext_psy_data);
|
di->btemp_psy, ab8500_btemp_get_ext_psy_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ab8500 btemp driver interrupts and their respective isr */
|
/* ab8500 btemp driver interrupts and their respective isr */
|
||||||
|
@ -1044,7 +1039,7 @@ static int ab8500_btemp_remove(struct platform_device *pdev)
|
||||||
destroy_workqueue(di->btemp_wq);
|
destroy_workqueue(di->btemp_wq);
|
||||||
|
|
||||||
flush_scheduled_work();
|
flush_scheduled_work();
|
||||||
power_supply_unregister(&di->btemp_psy);
|
power_supply_unregister(di->btemp_psy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1054,10 +1049,20 @@ static char *supply_interface[] = {
|
||||||
"ab8500_fg",
|
"ab8500_fg",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc ab8500_btemp_desc = {
|
||||||
|
.name = "ab8500_btemp",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = ab8500_btemp_props,
|
||||||
|
.num_properties = ARRAY_SIZE(ab8500_btemp_props),
|
||||||
|
.get_property = ab8500_btemp_get_property,
|
||||||
|
.external_power_changed = ab8500_btemp_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static int ab8500_btemp_probe(struct platform_device *pdev)
|
static int ab8500_btemp_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct device_node *np = pdev->dev.of_node;
|
struct device_node *np = pdev->dev.of_node;
|
||||||
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct ab8500_btemp *di;
|
struct ab8500_btemp *di;
|
||||||
int irq, i, ret = 0;
|
int irq, i, ret = 0;
|
||||||
u8 val;
|
u8 val;
|
||||||
|
@ -1089,17 +1094,9 @@ static int ab8500_btemp_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
di->initialized = false;
|
di->initialized = false;
|
||||||
|
|
||||||
/* BTEMP supply */
|
psy_cfg.supplied_to = supply_interface;
|
||||||
di->btemp_psy.name = "ab8500_btemp";
|
psy_cfg.num_supplicants = ARRAY_SIZE(supply_interface);
|
||||||
di->btemp_psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_cfg.drv_data = di;
|
||||||
di->btemp_psy.properties = ab8500_btemp_props;
|
|
||||||
di->btemp_psy.num_properties = ARRAY_SIZE(ab8500_btemp_props);
|
|
||||||
di->btemp_psy.get_property = ab8500_btemp_get_property;
|
|
||||||
di->btemp_psy.supplied_to = supply_interface;
|
|
||||||
di->btemp_psy.num_supplicants = ARRAY_SIZE(supply_interface);
|
|
||||||
di->btemp_psy.external_power_changed =
|
|
||||||
ab8500_btemp_external_power_changed;
|
|
||||||
|
|
||||||
|
|
||||||
/* Create a work queue for the btemp */
|
/* Create a work queue for the btemp */
|
||||||
di->btemp_wq =
|
di->btemp_wq =
|
||||||
|
@ -1140,9 +1137,11 @@ static int ab8500_btemp_probe(struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Register BTEMP power supply class */
|
/* Register BTEMP power supply class */
|
||||||
ret = power_supply_register(di->dev, &di->btemp_psy);
|
di->btemp_psy = power_supply_register(di->dev, &ab8500_btemp_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(di->btemp_psy)) {
|
||||||
dev_err(di->dev, "failed to register BTEMP psy\n");
|
dev_err(di->dev, "failed to register BTEMP psy\n");
|
||||||
|
ret = PTR_ERR(di->btemp_psy);
|
||||||
goto free_btemp_wq;
|
goto free_btemp_wq;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1171,7 +1170,7 @@ static int ab8500_btemp_probe(struct platform_device *pdev)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
free_irq:
|
free_irq:
|
||||||
power_supply_unregister(&di->btemp_psy);
|
power_supply_unregister(di->btemp_psy);
|
||||||
|
|
||||||
/* We also have to free all successfully registered irqs */
|
/* We also have to free all successfully registered irqs */
|
||||||
for (i = i - 1; i >= 0; i--) {
|
for (i = i - 1; i >= 0; i--) {
|
||||||
|
|
|
@ -435,7 +435,7 @@ static void ab8500_charger_set_usb_connected(struct ab8500_charger *di,
|
||||||
if (!connected)
|
if (!connected)
|
||||||
di->flags.vbus_drop_end = false;
|
di->flags.vbus_drop_end = false;
|
||||||
|
|
||||||
sysfs_notify(&di->usb_chg.psy.dev->kobj, NULL, "present");
|
sysfs_notify(&di->usb_chg.psy->dev.kobj, NULL, "present");
|
||||||
|
|
||||||
if (connected) {
|
if (connected) {
|
||||||
mutex_lock(&di->charger_attached_mutex);
|
mutex_lock(&di->charger_attached_mutex);
|
||||||
|
@ -1516,7 +1516,7 @@ static int ab8500_charger_ac_en(struct ux500_charger *charger,
|
||||||
|
|
||||||
dev_dbg(di->dev, "%s Disabled AC charging\n", __func__);
|
dev_dbg(di->dev, "%s Disabled AC charging\n", __func__);
|
||||||
}
|
}
|
||||||
ab8500_power_supply_changed(di, &di->ac_chg.psy);
|
ab8500_power_supply_changed(di, di->ac_chg.psy);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -1672,7 +1672,7 @@ static int ab8500_charger_usb_en(struct ux500_charger *charger,
|
||||||
cancel_delayed_work(&di->check_vbat_work);
|
cancel_delayed_work(&di->check_vbat_work);
|
||||||
|
|
||||||
}
|
}
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -1811,9 +1811,9 @@ static int ab8500_charger_watchdog_kick(struct ux500_charger *charger)
|
||||||
int ret;
|
int ret;
|
||||||
struct ab8500_charger *di;
|
struct ab8500_charger *di;
|
||||||
|
|
||||||
if (charger->psy.type == POWER_SUPPLY_TYPE_MAINS)
|
if (charger->psy->desc->type == POWER_SUPPLY_TYPE_MAINS)
|
||||||
di = to_ab8500_charger_ac_device_info(charger);
|
di = to_ab8500_charger_ac_device_info(charger);
|
||||||
else if (charger->psy.type == POWER_SUPPLY_TYPE_USB)
|
else if (charger->psy->desc->type == POWER_SUPPLY_TYPE_USB)
|
||||||
di = to_ab8500_charger_usb_device_info(charger);
|
di = to_ab8500_charger_usb_device_info(charger);
|
||||||
else
|
else
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
@ -1839,9 +1839,9 @@ static int ab8500_charger_update_charger_current(struct ux500_charger *charger,
|
||||||
int ret;
|
int ret;
|
||||||
struct ab8500_charger *di;
|
struct ab8500_charger *di;
|
||||||
|
|
||||||
if (charger->psy.type == POWER_SUPPLY_TYPE_MAINS)
|
if (charger->psy->desc->type == POWER_SUPPLY_TYPE_MAINS)
|
||||||
di = to_ab8500_charger_ac_device_info(charger);
|
di = to_ab8500_charger_ac_device_info(charger);
|
||||||
else if (charger->psy.type == POWER_SUPPLY_TYPE_USB)
|
else if (charger->psy->desc->type == POWER_SUPPLY_TYPE_USB)
|
||||||
di = to_ab8500_charger_usb_device_info(charger);
|
di = to_ab8500_charger_usb_device_info(charger);
|
||||||
else
|
else
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
@ -1879,7 +1879,7 @@ static int ab8540_charger_power_path_enable(struct ux500_charger *charger,
|
||||||
int ret;
|
int ret;
|
||||||
struct ab8500_charger *di;
|
struct ab8500_charger *di;
|
||||||
|
|
||||||
if (charger->psy.type == POWER_SUPPLY_TYPE_USB)
|
if (charger->psy->desc->type == POWER_SUPPLY_TYPE_USB)
|
||||||
di = to_ab8500_charger_usb_device_info(charger);
|
di = to_ab8500_charger_usb_device_info(charger);
|
||||||
else
|
else
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
@ -1910,7 +1910,7 @@ static int ab8540_charger_usb_pre_chg_enable(struct ux500_charger *charger,
|
||||||
int ret;
|
int ret;
|
||||||
struct ab8500_charger *di;
|
struct ab8500_charger *di;
|
||||||
|
|
||||||
if (charger->psy.type == POWER_SUPPLY_TYPE_USB)
|
if (charger->psy->desc->type == POWER_SUPPLY_TYPE_USB)
|
||||||
di = to_ab8500_charger_usb_device_info(charger);
|
di = to_ab8500_charger_usb_device_info(charger);
|
||||||
else
|
else
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
@ -1937,7 +1937,7 @@ static int ab8500_charger_get_ext_psy_data(struct device *dev, void *data)
|
||||||
struct ux500_charger *usb_chg;
|
struct ux500_charger *usb_chg;
|
||||||
|
|
||||||
usb_chg = (struct ux500_charger *)data;
|
usb_chg = (struct ux500_charger *)data;
|
||||||
psy = &usb_chg->psy;
|
psy = usb_chg->psy;
|
||||||
|
|
||||||
di = to_ab8500_charger_usb_device_info(usb_chg);
|
di = to_ab8500_charger_usb_device_info(usb_chg);
|
||||||
|
|
||||||
|
@ -1945,7 +1945,7 @@ static int ab8500_charger_get_ext_psy_data(struct device *dev, void *data)
|
||||||
|
|
||||||
/* For all psy where the driver name appears in any supplied_to */
|
/* For all psy where the driver name appears in any supplied_to */
|
||||||
for (i = 0; i < ext->num_supplicants; i++) {
|
for (i = 0; i < ext->num_supplicants; i++) {
|
||||||
if (!strcmp(ext->supplied_to[i], psy->name))
|
if (!strcmp(ext->supplied_to[i], psy->desc->name))
|
||||||
psy_found = true;
|
psy_found = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1953,16 +1953,16 @@ static int ab8500_charger_get_ext_psy_data(struct device *dev, void *data)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* Go through all properties for the psy */
|
/* Go through all properties for the psy */
|
||||||
for (j = 0; j < ext->num_properties; j++) {
|
for (j = 0; j < ext->desc->num_properties; j++) {
|
||||||
enum power_supply_property prop;
|
enum power_supply_property prop;
|
||||||
prop = ext->properties[j];
|
prop = ext->desc->properties[j];
|
||||||
|
|
||||||
if (ext->get_property(ext, prop, &ret))
|
if (power_supply_get_property(ext, prop, &ret))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
di->vbat = ret.intval / 1000;
|
di->vbat = ret.intval / 1000;
|
||||||
break;
|
break;
|
||||||
|
@ -1993,7 +1993,7 @@ static void ab8500_charger_check_vbat_work(struct work_struct *work)
|
||||||
struct ab8500_charger, check_vbat_work.work);
|
struct ab8500_charger, check_vbat_work.work);
|
||||||
|
|
||||||
class_for_each_device(power_supply_class, NULL,
|
class_for_each_device(power_supply_class, NULL,
|
||||||
&di->usb_chg.psy, ab8500_charger_get_ext_psy_data);
|
di->usb_chg.psy, ab8500_charger_get_ext_psy_data);
|
||||||
|
|
||||||
/* First run old_vbat is 0. */
|
/* First run old_vbat is 0. */
|
||||||
if (di->old_vbat == 0)
|
if (di->old_vbat == 0)
|
||||||
|
@ -2009,7 +2009,7 @@ static void ab8500_charger_check_vbat_work(struct work_struct *work)
|
||||||
di->vbat, di->old_vbat);
|
di->vbat, di->old_vbat);
|
||||||
ab8500_charger_set_vbus_in_curr(di,
|
ab8500_charger_set_vbus_in_curr(di,
|
||||||
di->max_usb_in_curr.usb_type_max);
|
di->max_usb_in_curr.usb_type_max);
|
||||||
power_supply_changed(&di->usb_chg.psy);
|
power_supply_changed(di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
di->old_vbat = di->vbat;
|
di->old_vbat = di->vbat;
|
||||||
|
@ -2049,7 +2049,7 @@ static void ab8500_charger_check_hw_failure_work(struct work_struct *work)
|
||||||
}
|
}
|
||||||
if (!(reg_value & MAIN_CH_NOK)) {
|
if (!(reg_value & MAIN_CH_NOK)) {
|
||||||
di->flags.mainextchnotok = false;
|
di->flags.mainextchnotok = false;
|
||||||
ab8500_power_supply_changed(di, &di->ac_chg.psy);
|
ab8500_power_supply_changed(di, di->ac_chg.psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (di->flags.vbus_ovv) {
|
if (di->flags.vbus_ovv) {
|
||||||
|
@ -2062,7 +2062,7 @@ static void ab8500_charger_check_hw_failure_work(struct work_struct *work)
|
||||||
}
|
}
|
||||||
if (!(reg_value & VBUS_OVV_TH)) {
|
if (!(reg_value & VBUS_OVV_TH)) {
|
||||||
di->flags.vbus_ovv = false;
|
di->flags.vbus_ovv = false;
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* If we still have a failure, schedule a new check */
|
/* If we still have a failure, schedule a new check */
|
||||||
|
@ -2132,8 +2132,8 @@ static void ab8500_charger_ac_work(struct work_struct *work)
|
||||||
di->ac.charger_connected = 0;
|
di->ac.charger_connected = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
ab8500_power_supply_changed(di, &di->ac_chg.psy);
|
ab8500_power_supply_changed(di, di->ac_chg.psy);
|
||||||
sysfs_notify(&di->ac_chg.psy.dev->kobj, NULL, "present");
|
sysfs_notify(&di->ac_chg.psy->dev.kobj, NULL, "present");
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ab8500_charger_usb_attached_work(struct work_struct *work)
|
static void ab8500_charger_usb_attached_work(struct work_struct *work)
|
||||||
|
@ -2240,7 +2240,7 @@ static void ab8500_charger_detect_usb_type_work(struct work_struct *work)
|
||||||
dev_dbg(di->dev, "%s di->vbus_detected = false\n", __func__);
|
dev_dbg(di->dev, "%s di->vbus_detected = false\n", __func__);
|
||||||
di->vbus_detected = false;
|
di->vbus_detected = false;
|
||||||
ab8500_charger_set_usb_connected(di, false);
|
ab8500_charger_set_usb_connected(di, false);
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
} else {
|
} else {
|
||||||
dev_dbg(di->dev, "%s di->vbus_detected = true\n", __func__);
|
dev_dbg(di->dev, "%s di->vbus_detected = true\n", __func__);
|
||||||
di->vbus_detected = true;
|
di->vbus_detected = true;
|
||||||
|
@ -2250,7 +2250,7 @@ static void ab8500_charger_detect_usb_type_work(struct work_struct *work)
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
ab8500_charger_set_usb_connected(di, true);
|
ab8500_charger_set_usb_connected(di, true);
|
||||||
ab8500_power_supply_changed(di,
|
ab8500_power_supply_changed(di,
|
||||||
&di->usb_chg.psy);
|
di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/*
|
/*
|
||||||
|
@ -2267,7 +2267,7 @@ static void ab8500_charger_detect_usb_type_work(struct work_struct *work)
|
||||||
ab8500_charger_set_usb_connected(di,
|
ab8500_charger_set_usb_connected(di,
|
||||||
true);
|
true);
|
||||||
ab8500_power_supply_changed(di,
|
ab8500_power_supply_changed(di,
|
||||||
&di->usb_chg.psy);
|
di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2295,7 +2295,7 @@ static void ab8500_charger_usb_link_attach_work(struct work_struct *work)
|
||||||
}
|
}
|
||||||
|
|
||||||
ab8500_charger_set_usb_connected(di, true);
|
ab8500_charger_set_usb_connected(di, true);
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2393,7 +2393,7 @@ static void ab8500_charger_usb_link_status_work(struct work_struct *work)
|
||||||
if (!(detected_chargers & USB_PW_CONN)) {
|
if (!(detected_chargers & USB_PW_CONN)) {
|
||||||
di->vbus_detected = false;
|
di->vbus_detected = false;
|
||||||
ab8500_charger_set_usb_connected(di, false);
|
ab8500_charger_set_usb_connected(di, false);
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2404,7 +2404,7 @@ static void ab8500_charger_usb_link_status_work(struct work_struct *work)
|
||||||
if (ret == -ENXIO) {
|
if (ret == -ENXIO) {
|
||||||
/* No valid charger type detected */
|
/* No valid charger type detected */
|
||||||
ab8500_charger_set_usb_connected(di, false);
|
ab8500_charger_set_usb_connected(di, false);
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -2463,7 +2463,7 @@ static void ab8500_charger_usb_state_changed_work(struct work_struct *work)
|
||||||
case AB8500_BM_USB_STATE_SUSPEND:
|
case AB8500_BM_USB_STATE_SUSPEND:
|
||||||
case AB8500_BM_USB_STATE_MAX:
|
case AB8500_BM_USB_STATE_MAX:
|
||||||
ab8500_charger_set_usb_connected(di, false);
|
ab8500_charger_set_usb_connected(di, false);
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case AB8500_BM_USB_STATE_RESUME:
|
case AB8500_BM_USB_STATE_RESUME:
|
||||||
|
@ -2486,7 +2486,7 @@ static void ab8500_charger_usb_state_changed_work(struct work_struct *work)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
ab8500_charger_set_usb_connected(di, true);
|
ab8500_charger_set_usb_connected(di, true);
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -2530,7 +2530,7 @@ static void ab8500_charger_check_usbchargernotok_work(struct work_struct *work)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (prev_status != di->flags.usbchargernotok)
|
if (prev_status != di->flags.usbchargernotok)
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2560,7 +2560,7 @@ static void ab8500_charger_check_main_thermal_prot_work(
|
||||||
else
|
else
|
||||||
di->flags.main_thermal_prot = false;
|
di->flags.main_thermal_prot = false;
|
||||||
|
|
||||||
ab8500_power_supply_changed(di, &di->ac_chg.psy);
|
ab8500_power_supply_changed(di, di->ac_chg.psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2590,7 +2590,7 @@ static void ab8500_charger_check_usb_thermal_prot_work(
|
||||||
else
|
else
|
||||||
di->flags.usb_thermal_prot = false;
|
di->flags.usb_thermal_prot = false;
|
||||||
|
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2651,7 +2651,7 @@ static irqreturn_t ab8500_charger_mainextchnotok_handler(int irq, void *_di)
|
||||||
|
|
||||||
dev_dbg(di->dev, "Main charger not ok\n");
|
dev_dbg(di->dev, "Main charger not ok\n");
|
||||||
di->flags.mainextchnotok = true;
|
di->flags.mainextchnotok = true;
|
||||||
ab8500_power_supply_changed(di, &di->ac_chg.psy);
|
ab8500_power_supply_changed(di, di->ac_chg.psy);
|
||||||
|
|
||||||
/* Schedule a new HW failure check */
|
/* Schedule a new HW failure check */
|
||||||
queue_delayed_work(di->charger_wq, &di->check_hw_failure_work, 0);
|
queue_delayed_work(di->charger_wq, &di->check_hw_failure_work, 0);
|
||||||
|
@ -2880,11 +2880,11 @@ static irqreturn_t ab8500_charger_chwdexp_handler(int irq, void *_di)
|
||||||
*/
|
*/
|
||||||
if (di->ac.charger_online) {
|
if (di->ac.charger_online) {
|
||||||
di->ac.wd_expired = true;
|
di->ac.wd_expired = true;
|
||||||
ab8500_power_supply_changed(di, &di->ac_chg.psy);
|
ab8500_power_supply_changed(di, di->ac_chg.psy);
|
||||||
}
|
}
|
||||||
if (di->usb.charger_online) {
|
if (di->usb.charger_online) {
|
||||||
di->usb.wd_expired = true;
|
di->usb.wd_expired = true;
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
|
@ -2927,7 +2927,7 @@ static irqreturn_t ab8500_charger_vbusovv_handler(int irq, void *_di)
|
||||||
|
|
||||||
dev_dbg(di->dev, "VBUS overvoltage detected\n");
|
dev_dbg(di->dev, "VBUS overvoltage detected\n");
|
||||||
di->flags.vbus_ovv = true;
|
di->flags.vbus_ovv = true;
|
||||||
ab8500_power_supply_changed(di, &di->usb_chg.psy);
|
ab8500_power_supply_changed(di, di->usb_chg.psy);
|
||||||
|
|
||||||
/* Schedule a new HW failure check */
|
/* Schedule a new HW failure check */
|
||||||
queue_delayed_work(di->charger_wq, &di->check_hw_failure_work, 0);
|
queue_delayed_work(di->charger_wq, &di->check_hw_failure_work, 0);
|
||||||
|
@ -3428,10 +3428,10 @@ static int ab8500_charger_remove(struct platform_device *pdev)
|
||||||
|
|
||||||
flush_scheduled_work();
|
flush_scheduled_work();
|
||||||
if (di->usb_chg.enabled)
|
if (di->usb_chg.enabled)
|
||||||
power_supply_unregister(&di->usb_chg.psy);
|
power_supply_unregister(di->usb_chg.psy);
|
||||||
|
|
||||||
if (di->ac_chg.enabled && !di->ac_chg.external)
|
if (di->ac_chg.enabled && !di->ac_chg.external)
|
||||||
power_supply_unregister(&di->ac_chg.psy);
|
power_supply_unregister(di->ac_chg.psy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -3442,10 +3442,27 @@ static char *supply_interface[] = {
|
||||||
"ab8500_btemp",
|
"ab8500_btemp",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc ab8500_ac_chg_desc = {
|
||||||
|
.name = "ab8500_ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = ab8500_charger_ac_props,
|
||||||
|
.num_properties = ARRAY_SIZE(ab8500_charger_ac_props),
|
||||||
|
.get_property = ab8500_charger_ac_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc ab8500_usb_chg_desc = {
|
||||||
|
.name = "ab8500_usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = ab8500_charger_usb_props,
|
||||||
|
.num_properties = ARRAY_SIZE(ab8500_charger_usb_props),
|
||||||
|
.get_property = ab8500_charger_usb_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
static int ab8500_charger_probe(struct platform_device *pdev)
|
static int ab8500_charger_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct device_node *np = pdev->dev.of_node;
|
struct device_node *np = pdev->dev.of_node;
|
||||||
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
||||||
|
struct power_supply_config ac_psy_cfg = {}, usb_psy_cfg = {};
|
||||||
struct ab8500_charger *di;
|
struct ab8500_charger *di;
|
||||||
int irq, i, charger_status, ret = 0, ch_stat;
|
int irq, i, charger_status, ret = 0, ch_stat;
|
||||||
|
|
||||||
|
@ -3483,15 +3500,15 @@ static int ab8500_charger_probe(struct platform_device *pdev)
|
||||||
di->autopower = false;
|
di->autopower = false;
|
||||||
di->invalid_charger_detect_state = 0;
|
di->invalid_charger_detect_state = 0;
|
||||||
|
|
||||||
|
/* AC and USB supply config */
|
||||||
|
ac_psy_cfg.supplied_to = supply_interface;
|
||||||
|
ac_psy_cfg.num_supplicants = ARRAY_SIZE(supply_interface);
|
||||||
|
ac_psy_cfg.drv_data = &di->ac_chg;
|
||||||
|
usb_psy_cfg.supplied_to = supply_interface;
|
||||||
|
usb_psy_cfg.num_supplicants = ARRAY_SIZE(supply_interface);
|
||||||
|
usb_psy_cfg.drv_data = &di->usb_chg;
|
||||||
|
|
||||||
/* AC supply */
|
/* AC supply */
|
||||||
/* power_supply base class */
|
|
||||||
di->ac_chg.psy.name = "ab8500_ac";
|
|
||||||
di->ac_chg.psy.type = POWER_SUPPLY_TYPE_MAINS;
|
|
||||||
di->ac_chg.psy.properties = ab8500_charger_ac_props;
|
|
||||||
di->ac_chg.psy.num_properties = ARRAY_SIZE(ab8500_charger_ac_props);
|
|
||||||
di->ac_chg.psy.get_property = ab8500_charger_ac_get_property;
|
|
||||||
di->ac_chg.psy.supplied_to = supply_interface;
|
|
||||||
di->ac_chg.psy.num_supplicants = ARRAY_SIZE(supply_interface),
|
|
||||||
/* ux500_charger sub-class */
|
/* ux500_charger sub-class */
|
||||||
di->ac_chg.ops.enable = &ab8500_charger_ac_en;
|
di->ac_chg.ops.enable = &ab8500_charger_ac_en;
|
||||||
di->ac_chg.ops.check_enable = &ab8500_charger_ac_check_enable;
|
di->ac_chg.ops.check_enable = &ab8500_charger_ac_check_enable;
|
||||||
|
@ -3511,14 +3528,6 @@ static int ab8500_charger_probe(struct platform_device *pdev)
|
||||||
&charger_notifier_list, &charger_nb);
|
&charger_notifier_list, &charger_nb);
|
||||||
|
|
||||||
/* USB supply */
|
/* USB supply */
|
||||||
/* power_supply base class */
|
|
||||||
di->usb_chg.psy.name = "ab8500_usb";
|
|
||||||
di->usb_chg.psy.type = POWER_SUPPLY_TYPE_USB;
|
|
||||||
di->usb_chg.psy.properties = ab8500_charger_usb_props;
|
|
||||||
di->usb_chg.psy.num_properties = ARRAY_SIZE(ab8500_charger_usb_props);
|
|
||||||
di->usb_chg.psy.get_property = ab8500_charger_usb_get_property;
|
|
||||||
di->usb_chg.psy.supplied_to = supply_interface;
|
|
||||||
di->usb_chg.psy.num_supplicants = ARRAY_SIZE(supply_interface),
|
|
||||||
/* ux500_charger sub-class */
|
/* ux500_charger sub-class */
|
||||||
di->usb_chg.ops.enable = &ab8500_charger_usb_en;
|
di->usb_chg.ops.enable = &ab8500_charger_usb_en;
|
||||||
di->usb_chg.ops.check_enable = &ab8500_charger_usb_check_enable;
|
di->usb_chg.ops.check_enable = &ab8500_charger_usb_check_enable;
|
||||||
|
@ -3616,18 +3625,24 @@ static int ab8500_charger_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
/* Register AC charger class */
|
/* Register AC charger class */
|
||||||
if (di->ac_chg.enabled) {
|
if (di->ac_chg.enabled) {
|
||||||
ret = power_supply_register(di->dev, &di->ac_chg.psy);
|
di->ac_chg.psy = power_supply_register(di->dev,
|
||||||
if (ret) {
|
&ab8500_ac_chg_desc,
|
||||||
|
&ac_psy_cfg);
|
||||||
|
if (IS_ERR(di->ac_chg.psy)) {
|
||||||
dev_err(di->dev, "failed to register AC charger\n");
|
dev_err(di->dev, "failed to register AC charger\n");
|
||||||
|
ret = PTR_ERR(di->ac_chg.psy);
|
||||||
goto free_charger_wq;
|
goto free_charger_wq;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Register USB charger class */
|
/* Register USB charger class */
|
||||||
if (di->usb_chg.enabled) {
|
if (di->usb_chg.enabled) {
|
||||||
ret = power_supply_register(di->dev, &di->usb_chg.psy);
|
di->usb_chg.psy = power_supply_register(di->dev,
|
||||||
if (ret) {
|
&ab8500_usb_chg_desc,
|
||||||
|
&usb_psy_cfg);
|
||||||
|
if (IS_ERR(di->usb_chg.psy)) {
|
||||||
dev_err(di->dev, "failed to register USB charger\n");
|
dev_err(di->dev, "failed to register USB charger\n");
|
||||||
|
ret = PTR_ERR(di->usb_chg.psy);
|
||||||
goto free_ac;
|
goto free_ac;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3650,8 +3665,8 @@ static int ab8500_charger_probe(struct platform_device *pdev)
|
||||||
if (charger_status & AC_PW_CONN) {
|
if (charger_status & AC_PW_CONN) {
|
||||||
di->ac.charger_connected = 1;
|
di->ac.charger_connected = 1;
|
||||||
di->ac_conn = true;
|
di->ac_conn = true;
|
||||||
ab8500_power_supply_changed(di, &di->ac_chg.psy);
|
ab8500_power_supply_changed(di, di->ac_chg.psy);
|
||||||
sysfs_notify(&di->ac_chg.psy.dev->kobj, NULL, "present");
|
sysfs_notify(&di->ac_chg.psy->dev.kobj, NULL, "present");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (charger_status & USB_PW_CONN) {
|
if (charger_status & USB_PW_CONN) {
|
||||||
|
@ -3712,10 +3727,10 @@ put_usb_phy:
|
||||||
usb_put_phy(di->usb_phy);
|
usb_put_phy(di->usb_phy);
|
||||||
free_usb:
|
free_usb:
|
||||||
if (di->usb_chg.enabled)
|
if (di->usb_chg.enabled)
|
||||||
power_supply_unregister(&di->usb_chg.psy);
|
power_supply_unregister(di->usb_chg.psy);
|
||||||
free_ac:
|
free_ac:
|
||||||
if (di->ac_chg.enabled)
|
if (di->ac_chg.enabled)
|
||||||
power_supply_unregister(&di->ac_chg.psy);
|
power_supply_unregister(di->ac_chg.psy);
|
||||||
free_charger_wq:
|
free_charger_wq:
|
||||||
destroy_workqueue(di->charger_wq);
|
destroy_workqueue(di->charger_wq);
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -57,9 +57,6 @@
|
||||||
#define interpolate(x, x1, y1, x2, y2) \
|
#define interpolate(x, x1, y1, x2, y2) \
|
||||||
((y1) + ((((y2) - (y1)) * ((x) - (x1))) / ((x2) - (x1))));
|
((y1) + ((((y2) - (y1)) * ((x) - (x1))) / ((x2) - (x1))));
|
||||||
|
|
||||||
#define to_ab8500_fg_device_info(x) container_of((x), \
|
|
||||||
struct ab8500_fg, fg_psy);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct ab8500_fg_interrupts - ab8500 fg interupts
|
* struct ab8500_fg_interrupts - ab8500 fg interupts
|
||||||
* @name: name of the interrupt
|
* @name: name of the interrupt
|
||||||
|
@ -229,7 +226,7 @@ struct ab8500_fg {
|
||||||
struct ab8500 *parent;
|
struct ab8500 *parent;
|
||||||
struct ab8500_gpadc *gpadc;
|
struct ab8500_gpadc *gpadc;
|
||||||
struct abx500_bm_data *bm;
|
struct abx500_bm_data *bm;
|
||||||
struct power_supply fg_psy;
|
struct power_supply *fg_psy;
|
||||||
struct workqueue_struct *fg_wq;
|
struct workqueue_struct *fg_wq;
|
||||||
struct delayed_work fg_periodic_work;
|
struct delayed_work fg_periodic_work;
|
||||||
struct delayed_work fg_low_bat_work;
|
struct delayed_work fg_low_bat_work;
|
||||||
|
@ -622,14 +619,14 @@ int ab8500_fg_inst_curr_finalize(struct ab8500_fg *di, int *res)
|
||||||
u8 low, high;
|
u8 low, high;
|
||||||
int val;
|
int val;
|
||||||
int ret;
|
int ret;
|
||||||
int timeout;
|
unsigned long timeout;
|
||||||
|
|
||||||
if (!completion_done(&di->ab8500_fg_complete)) {
|
if (!completion_done(&di->ab8500_fg_complete)) {
|
||||||
timeout = wait_for_completion_timeout(
|
timeout = wait_for_completion_timeout(
|
||||||
&di->ab8500_fg_complete,
|
&di->ab8500_fg_complete,
|
||||||
INS_CURR_TIMEOUT);
|
INS_CURR_TIMEOUT);
|
||||||
dev_dbg(di->dev, "Finalize time: %d ms\n",
|
dev_dbg(di->dev, "Finalize time: %d ms\n",
|
||||||
((INS_CURR_TIMEOUT - timeout) * 1000) / HZ);
|
jiffies_to_msecs(INS_CURR_TIMEOUT - timeout));
|
||||||
if (!timeout) {
|
if (!timeout) {
|
||||||
ret = -ETIME;
|
ret = -ETIME;
|
||||||
disable_irq(di->irq);
|
disable_irq(di->irq);
|
||||||
|
@ -716,7 +713,7 @@ fail:
|
||||||
int ab8500_fg_inst_curr_blocking(struct ab8500_fg *di)
|
int ab8500_fg_inst_curr_blocking(struct ab8500_fg *di)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
int timeout;
|
unsigned long timeout;
|
||||||
int res = 0;
|
int res = 0;
|
||||||
|
|
||||||
ret = ab8500_fg_inst_curr_start(di);
|
ret = ab8500_fg_inst_curr_start(di);
|
||||||
|
@ -731,7 +728,7 @@ int ab8500_fg_inst_curr_blocking(struct ab8500_fg *di)
|
||||||
&di->ab8500_fg_started,
|
&di->ab8500_fg_started,
|
||||||
INS_CURR_TIMEOUT);
|
INS_CURR_TIMEOUT);
|
||||||
dev_dbg(di->dev, "Start time: %d ms\n",
|
dev_dbg(di->dev, "Start time: %d ms\n",
|
||||||
((INS_CURR_TIMEOUT - timeout) * 1000) / HZ);
|
jiffies_to_msecs(INS_CURR_TIMEOUT - timeout));
|
||||||
if (!timeout) {
|
if (!timeout) {
|
||||||
ret = -ETIME;
|
ret = -ETIME;
|
||||||
dev_err(di->dev, "completion timed out [%d]\n",
|
dev_err(di->dev, "completion timed out [%d]\n",
|
||||||
|
@ -1391,7 +1388,7 @@ static void ab8500_fg_check_capacity_limits(struct ab8500_fg *di, bool init)
|
||||||
di->bat_cap.prev_percent,
|
di->bat_cap.prev_percent,
|
||||||
di->bat_cap.cap_scale.scaled_cap);
|
di->bat_cap.cap_scale.scaled_cap);
|
||||||
}
|
}
|
||||||
power_supply_changed(&di->fg_psy);
|
power_supply_changed(di->fg_psy);
|
||||||
if (di->flags.fully_charged && di->flags.force_full) {
|
if (di->flags.fully_charged && di->flags.force_full) {
|
||||||
dev_dbg(di->dev, "Battery full, notifying.\n");
|
dev_dbg(di->dev, "Battery full, notifying.\n");
|
||||||
di->flags.force_full = false;
|
di->flags.force_full = false;
|
||||||
|
@ -1850,7 +1847,7 @@ static void ab8500_fg_check_hw_failure_work(struct work_struct *work)
|
||||||
if (!di->flags.bat_ovv) {
|
if (!di->flags.bat_ovv) {
|
||||||
dev_dbg(di->dev, "Battery OVV\n");
|
dev_dbg(di->dev, "Battery OVV\n");
|
||||||
di->flags.bat_ovv = true;
|
di->flags.bat_ovv = true;
|
||||||
power_supply_changed(&di->fg_psy);
|
power_supply_changed(di->fg_psy);
|
||||||
}
|
}
|
||||||
/* Not yet recovered from ovv, reschedule this test */
|
/* Not yet recovered from ovv, reschedule this test */
|
||||||
queue_delayed_work(di->fg_wq, &di->fg_check_hw_failure_work,
|
queue_delayed_work(di->fg_wq, &di->fg_check_hw_failure_work,
|
||||||
|
@ -1858,7 +1855,7 @@ static void ab8500_fg_check_hw_failure_work(struct work_struct *work)
|
||||||
} else {
|
} else {
|
||||||
dev_dbg(di->dev, "Battery recovered from OVV\n");
|
dev_dbg(di->dev, "Battery recovered from OVV\n");
|
||||||
di->flags.bat_ovv = false;
|
di->flags.bat_ovv = false;
|
||||||
power_supply_changed(&di->fg_psy);
|
power_supply_changed(di->fg_psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2096,9 +2093,7 @@ static int ab8500_fg_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 ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If battery is identified as unknown and charging of unknown
|
* If battery is identified as unknown and charging of unknown
|
||||||
|
@ -2181,14 +2176,14 @@ static int ab8500_fg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
|
|
||||||
psy = (struct power_supply *)data;
|
psy = (struct power_supply *)data;
|
||||||
ext = dev_get_drvdata(dev);
|
ext = dev_get_drvdata(dev);
|
||||||
di = to_ab8500_fg_device_info(psy);
|
di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For all psy where the name of your driver
|
* For all psy where the name of your driver
|
||||||
* appears in any supplied_to
|
* appears in any supplied_to
|
||||||
*/
|
*/
|
||||||
for (i = 0; i < ext->num_supplicants; i++) {
|
for (i = 0; i < ext->num_supplicants; i++) {
|
||||||
if (!strcmp(ext->supplied_to[i], psy->name))
|
if (!strcmp(ext->supplied_to[i], psy->desc->name))
|
||||||
psy_found = true;
|
psy_found = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2196,16 +2191,16 @@ static int ab8500_fg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* Go through all properties for the psy */
|
/* Go through all properties for the psy */
|
||||||
for (j = 0; j < ext->num_properties; j++) {
|
for (j = 0; j < ext->desc->num_properties; j++) {
|
||||||
enum power_supply_property prop;
|
enum power_supply_property prop;
|
||||||
prop = ext->properties[j];
|
prop = ext->desc->properties[j];
|
||||||
|
|
||||||
if (ext->get_property(ext, prop, &ret))
|
if (power_supply_get_property(ext, prop, &ret))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
switch (ret.intval) {
|
switch (ret.intval) {
|
||||||
case POWER_SUPPLY_STATUS_UNKNOWN:
|
case POWER_SUPPLY_STATUS_UNKNOWN:
|
||||||
|
@ -2244,7 +2239,7 @@ static int ab8500_fg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
};
|
};
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
if (!di->flags.batt_id_received &&
|
if (!di->flags.batt_id_received &&
|
||||||
di->bm->batt_id != BATTERY_UNKNOWN) {
|
di->bm->batt_id != BATTERY_UNKNOWN) {
|
||||||
|
@ -2274,7 +2269,7 @@ static int ab8500_fg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_TEMP:
|
case POWER_SUPPLY_PROP_TEMP:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
if (di->flags.batt_id_received)
|
if (di->flags.batt_id_received)
|
||||||
di->bat_temp = ret.intval;
|
di->bat_temp = ret.intval;
|
||||||
|
@ -2399,10 +2394,10 @@ out:
|
||||||
*/
|
*/
|
||||||
static void ab8500_fg_external_power_changed(struct power_supply *psy)
|
static void ab8500_fg_external_power_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct ab8500_fg *di = to_ab8500_fg_device_info(psy);
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
class_for_each_device(power_supply_class, NULL,
|
class_for_each_device(power_supply_class, NULL,
|
||||||
&di->fg_psy, ab8500_fg_get_ext_psy_data);
|
di->fg_psy, ab8500_fg_get_ext_psy_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2580,9 +2575,7 @@ static ssize_t ab8505_powercut_flagtime_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_FLAG_TIME_REG, ®_value);
|
AB8505_RTC_PCUT_FLAG_TIME_REG, ®_value);
|
||||||
|
@ -2605,9 +2598,7 @@ static ssize_t ab8505_powercut_flagtime_write(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
long unsigned reg_value;
|
long unsigned reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
reg_value = simple_strtoul(buf, NULL, 10);
|
reg_value = simple_strtoul(buf, NULL, 10);
|
||||||
|
|
||||||
|
@ -2633,9 +2624,7 @@ static ssize_t ab8505_powercut_maxtime_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_MAX_TIME_REG, ®_value);
|
AB8505_RTC_PCUT_MAX_TIME_REG, ®_value);
|
||||||
|
@ -2659,9 +2648,7 @@ static ssize_t ab8505_powercut_maxtime_write(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
int reg_value;
|
int reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
reg_value = simple_strtoul(buf, NULL, 10);
|
reg_value = simple_strtoul(buf, NULL, 10);
|
||||||
if (reg_value > 0x7F) {
|
if (reg_value > 0x7F) {
|
||||||
|
@ -2686,9 +2673,7 @@ static ssize_t ab8505_powercut_restart_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_RESTART_REG, ®_value);
|
AB8505_RTC_PCUT_RESTART_REG, ®_value);
|
||||||
|
@ -2711,9 +2696,7 @@ static ssize_t ab8505_powercut_restart_write(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
int reg_value;
|
int reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
reg_value = simple_strtoul(buf, NULL, 10);
|
reg_value = simple_strtoul(buf, NULL, 10);
|
||||||
if (reg_value > 0xF) {
|
if (reg_value > 0xF) {
|
||||||
|
@ -2739,9 +2722,7 @@ static ssize_t ab8505_powercut_timer_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_TIME_REG, ®_value);
|
AB8505_RTC_PCUT_TIME_REG, ®_value);
|
||||||
|
@ -2764,9 +2745,7 @@ static ssize_t ab8505_powercut_restart_counter_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_RESTART_REG, ®_value);
|
AB8505_RTC_PCUT_RESTART_REG, ®_value);
|
||||||
|
@ -2789,9 +2768,7 @@ static ssize_t ab8505_powercut_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
|
AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
|
||||||
|
@ -2812,9 +2789,7 @@ static ssize_t ab8505_powercut_write(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
int reg_value;
|
int reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
reg_value = simple_strtoul(buf, NULL, 10);
|
reg_value = simple_strtoul(buf, NULL, 10);
|
||||||
if (reg_value > 0x1) {
|
if (reg_value > 0x1) {
|
||||||
|
@ -2840,9 +2815,7 @@ static ssize_t ab8505_powercut_flag_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
|
AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
|
||||||
|
@ -2865,9 +2838,7 @@ static ssize_t ab8505_powercut_debounce_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_DEBOUNCE_REG, ®_value);
|
AB8505_RTC_PCUT_DEBOUNCE_REG, ®_value);
|
||||||
|
@ -2890,9 +2861,7 @@ static ssize_t ab8505_powercut_debounce_write(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
int reg_value;
|
int reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
reg_value = simple_strtoul(buf, NULL, 10);
|
reg_value = simple_strtoul(buf, NULL, 10);
|
||||||
if (reg_value > 0x7) {
|
if (reg_value > 0x7) {
|
||||||
|
@ -2917,9 +2886,7 @@ static ssize_t ab8505_powercut_enable_status_read(struct device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg_value;
|
u8 reg_value;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
|
||||||
AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
|
AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
|
||||||
|
@ -2954,44 +2921,38 @@ static struct device_attribute ab8505_fg_sysfs_psy_attrs[] = {
|
||||||
ab8505_powercut_enable_status_read, NULL),
|
ab8505_powercut_enable_status_read, NULL),
|
||||||
};
|
};
|
||||||
|
|
||||||
static int ab8500_fg_sysfs_psy_create_attrs(struct device *dev)
|
static int ab8500_fg_sysfs_psy_create_attrs(struct ab8500_fg *di)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
|
||||||
struct ab8500_fg *di;
|
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
if (((is_ab8505(di->parent) || is_ab9540(di->parent)) &&
|
if (((is_ab8505(di->parent) || is_ab9540(di->parent)) &&
|
||||||
abx500_get_chip_id(dev->parent) >= AB8500_CUT2P0)
|
abx500_get_chip_id(di->dev) >= AB8500_CUT2P0)
|
||||||
|| is_ab8540(di->parent)) {
|
|| is_ab8540(di->parent)) {
|
||||||
for (i = 0; i < ARRAY_SIZE(ab8505_fg_sysfs_psy_attrs); i++)
|
for (i = 0; i < ARRAY_SIZE(ab8505_fg_sysfs_psy_attrs); i++)
|
||||||
if (device_create_file(dev,
|
if (device_create_file(&di->fg_psy->dev,
|
||||||
&ab8505_fg_sysfs_psy_attrs[i]))
|
&ab8505_fg_sysfs_psy_attrs[i]))
|
||||||
goto sysfs_psy_create_attrs_failed_ab8505;
|
goto sysfs_psy_create_attrs_failed_ab8505;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
sysfs_psy_create_attrs_failed_ab8505:
|
sysfs_psy_create_attrs_failed_ab8505:
|
||||||
dev_err(dev, "Failed creating sysfs psy attrs for ab8505.\n");
|
dev_err(&di->fg_psy->dev, "Failed creating sysfs psy attrs for ab8505.\n");
|
||||||
while (i--)
|
while (i--)
|
||||||
device_remove_file(dev, &ab8505_fg_sysfs_psy_attrs[i]);
|
device_remove_file(&di->fg_psy->dev,
|
||||||
|
&ab8505_fg_sysfs_psy_attrs[i]);
|
||||||
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ab8500_fg_sysfs_psy_remove_attrs(struct device *dev)
|
static void ab8500_fg_sysfs_psy_remove_attrs(struct ab8500_fg *di)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
|
||||||
struct ab8500_fg *di;
|
|
||||||
|
|
||||||
di = to_ab8500_fg_device_info(psy);
|
|
||||||
|
|
||||||
if (((is_ab8505(di->parent) || is_ab9540(di->parent)) &&
|
if (((is_ab8505(di->parent) || is_ab9540(di->parent)) &&
|
||||||
abx500_get_chip_id(dev->parent) >= AB8500_CUT2P0)
|
abx500_get_chip_id(di->dev) >= AB8500_CUT2P0)
|
||||||
|| is_ab8540(di->parent)) {
|
|| is_ab8540(di->parent)) {
|
||||||
for (i = 0; i < ARRAY_SIZE(ab8505_fg_sysfs_psy_attrs); i++)
|
for (i = 0; i < ARRAY_SIZE(ab8505_fg_sysfs_psy_attrs); i++)
|
||||||
(void)device_remove_file(dev, &ab8505_fg_sysfs_psy_attrs[i]);
|
(void)device_remove_file(&di->fg_psy->dev,
|
||||||
|
&ab8505_fg_sysfs_psy_attrs[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3056,17 +3017,20 @@ static int ab8500_fg_remove(struct platform_device *pdev)
|
||||||
ab8500_fg_sysfs_exit(di);
|
ab8500_fg_sysfs_exit(di);
|
||||||
|
|
||||||
flush_scheduled_work();
|
flush_scheduled_work();
|
||||||
ab8500_fg_sysfs_psy_remove_attrs(di->fg_psy.dev);
|
ab8500_fg_sysfs_psy_remove_attrs(di);
|
||||||
power_supply_unregister(&di->fg_psy);
|
power_supply_unregister(di->fg_psy);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ab8500 fg driver interrupts and their respective isr */
|
/* ab8500 fg driver interrupts and their respective isr */
|
||||||
static struct ab8500_fg_interrupts ab8500_fg_irq[] = {
|
static struct ab8500_fg_interrupts ab8500_fg_irq_th[] = {
|
||||||
{"NCONV_ACCU", ab8500_fg_cc_convend_handler},
|
{"NCONV_ACCU", ab8500_fg_cc_convend_handler},
|
||||||
{"BATT_OVV", ab8500_fg_batt_ovv_handler},
|
{"BATT_OVV", ab8500_fg_batt_ovv_handler},
|
||||||
{"LOW_BAT_F", ab8500_fg_lowbatf_handler},
|
{"LOW_BAT_F", ab8500_fg_lowbatf_handler},
|
||||||
{"CC_INT_CALIB", ab8500_fg_cc_int_calib_handler},
|
{"CC_INT_CALIB", ab8500_fg_cc_int_calib_handler},
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct ab8500_fg_interrupts ab8500_fg_irq_bh[] = {
|
||||||
{"CCEOC", ab8500_fg_cc_data_end_handler},
|
{"CCEOC", ab8500_fg_cc_data_end_handler},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -3075,10 +3039,20 @@ static char *supply_interface[] = {
|
||||||
"ab8500_usb",
|
"ab8500_usb",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc ab8500_fg_desc = {
|
||||||
|
.name = "ab8500_fg",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = ab8500_fg_props,
|
||||||
|
.num_properties = ARRAY_SIZE(ab8500_fg_props),
|
||||||
|
.get_property = ab8500_fg_get_property,
|
||||||
|
.external_power_changed = ab8500_fg_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static int ab8500_fg_probe(struct platform_device *pdev)
|
static int ab8500_fg_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct device_node *np = pdev->dev.of_node;
|
struct device_node *np = pdev->dev.of_node;
|
||||||
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct ab8500_fg *di;
|
struct ab8500_fg *di;
|
||||||
int i, irq;
|
int i, irq;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
@ -3110,14 +3084,9 @@ static int ab8500_fg_probe(struct platform_device *pdev)
|
||||||
di->parent = dev_get_drvdata(pdev->dev.parent);
|
di->parent = dev_get_drvdata(pdev->dev.parent);
|
||||||
di->gpadc = ab8500_gpadc_get("ab8500-gpadc.0");
|
di->gpadc = ab8500_gpadc_get("ab8500-gpadc.0");
|
||||||
|
|
||||||
di->fg_psy.name = "ab8500_fg";
|
psy_cfg.supplied_to = supply_interface;
|
||||||
di->fg_psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_cfg.num_supplicants = ARRAY_SIZE(supply_interface);
|
||||||
di->fg_psy.properties = ab8500_fg_props;
|
psy_cfg.drv_data = di;
|
||||||
di->fg_psy.num_properties = ARRAY_SIZE(ab8500_fg_props);
|
|
||||||
di->fg_psy.get_property = ab8500_fg_get_property;
|
|
||||||
di->fg_psy.supplied_to = supply_interface;
|
|
||||||
di->fg_psy.num_supplicants = ARRAY_SIZE(supply_interface),
|
|
||||||
di->fg_psy.external_power_changed = ab8500_fg_external_power_changed;
|
|
||||||
|
|
||||||
di->bat_cap.max_mah_design = MILLI_TO_MICRO *
|
di->bat_cap.max_mah_design = MILLI_TO_MICRO *
|
||||||
di->bm->bat_type[di->bm->batt_id].charge_full_design;
|
di->bm->bat_type[di->bm->batt_id].charge_full_design;
|
||||||
|
@ -3178,9 +3147,10 @@ static int ab8500_fg_probe(struct platform_device *pdev)
|
||||||
di->flags.batt_id_received = false;
|
di->flags.batt_id_received = false;
|
||||||
|
|
||||||
/* Register FG power supply class */
|
/* Register FG power supply class */
|
||||||
ret = power_supply_register(di->dev, &di->fg_psy);
|
di->fg_psy = power_supply_register(di->dev, &ab8500_fg_desc, &psy_cfg);
|
||||||
if (ret) {
|
if (IS_ERR(di->fg_psy)) {
|
||||||
dev_err(di->dev, "failed to register FG psy\n");
|
dev_err(di->dev, "failed to register FG psy\n");
|
||||||
|
ret = PTR_ERR(di->fg_psy);
|
||||||
goto free_inst_curr_wq;
|
goto free_inst_curr_wq;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3194,21 +3164,36 @@ static int ab8500_fg_probe(struct platform_device *pdev)
|
||||||
init_completion(&di->ab8500_fg_started);
|
init_completion(&di->ab8500_fg_started);
|
||||||
init_completion(&di->ab8500_fg_complete);
|
init_completion(&di->ab8500_fg_complete);
|
||||||
|
|
||||||
/* Register interrupts */
|
/* Register primary interrupt handlers */
|
||||||
for (i = 0; i < ARRAY_SIZE(ab8500_fg_irq); i++) {
|
for (i = 0; i < ARRAY_SIZE(ab8500_fg_irq_th); i++) {
|
||||||
irq = platform_get_irq_byname(pdev, ab8500_fg_irq[i].name);
|
irq = platform_get_irq_byname(pdev, ab8500_fg_irq_th[i].name);
|
||||||
ret = request_threaded_irq(irq, NULL, ab8500_fg_irq[i].isr,
|
ret = request_irq(irq, ab8500_fg_irq_th[i].isr,
|
||||||
IRQF_SHARED | IRQF_NO_SUSPEND,
|
IRQF_SHARED | IRQF_NO_SUSPEND,
|
||||||
ab8500_fg_irq[i].name, di);
|
ab8500_fg_irq_th[i].name, di);
|
||||||
|
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
dev_err(di->dev, "failed to request %s IRQ %d: %d\n"
|
dev_err(di->dev, "failed to request %s IRQ %d: %d\n",
|
||||||
, ab8500_fg_irq[i].name, irq, ret);
|
ab8500_fg_irq_th[i].name, irq, ret);
|
||||||
goto free_irq;
|
goto free_irq;
|
||||||
}
|
}
|
||||||
dev_dbg(di->dev, "Requested %s IRQ %d: %d\n",
|
dev_dbg(di->dev, "Requested %s IRQ %d: %d\n",
|
||||||
ab8500_fg_irq[i].name, irq, ret);
|
ab8500_fg_irq_th[i].name, irq, ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Register threaded interrupt handler */
|
||||||
|
irq = platform_get_irq_byname(pdev, ab8500_fg_irq_bh[0].name);
|
||||||
|
ret = request_threaded_irq(irq, NULL, ab8500_fg_irq_bh[0].isr,
|
||||||
|
IRQF_SHARED | IRQF_NO_SUSPEND | IRQF_ONESHOT,
|
||||||
|
ab8500_fg_irq_bh[0].name, di);
|
||||||
|
|
||||||
|
if (ret != 0) {
|
||||||
|
dev_err(di->dev, "failed to request %s IRQ %d: %d\n",
|
||||||
|
ab8500_fg_irq_bh[0].name, irq, ret);
|
||||||
|
goto free_irq;
|
||||||
|
}
|
||||||
|
dev_dbg(di->dev, "Requested %s IRQ %d: %d\n",
|
||||||
|
ab8500_fg_irq_bh[0].name, irq, ret);
|
||||||
|
|
||||||
di->irq = platform_get_irq_byname(pdev, "CCEOC");
|
di->irq = platform_get_irq_byname(pdev, "CCEOC");
|
||||||
disable_irq(di->irq);
|
disable_irq(di->irq);
|
||||||
di->nbr_cceoc_irq_cnt = 0;
|
di->nbr_cceoc_irq_cnt = 0;
|
||||||
|
@ -3221,7 +3206,7 @@ static int ab8500_fg_probe(struct platform_device *pdev)
|
||||||
goto free_irq;
|
goto free_irq;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = ab8500_fg_sysfs_psy_create_attrs(di->fg_psy.dev);
|
ret = ab8500_fg_sysfs_psy_create_attrs(di);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(di->dev, "failed to create FG psy\n");
|
dev_err(di->dev, "failed to create FG psy\n");
|
||||||
ab8500_fg_sysfs_exit(di);
|
ab8500_fg_sysfs_exit(di);
|
||||||
|
@ -3243,13 +3228,15 @@ static int ab8500_fg_probe(struct platform_device *pdev)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
free_irq:
|
free_irq:
|
||||||
power_supply_unregister(&di->fg_psy);
|
power_supply_unregister(di->fg_psy);
|
||||||
|
|
||||||
/* We also have to free all successfully registered irqs */
|
/* We also have to free all registered irqs */
|
||||||
for (i = i - 1; i >= 0; i--) {
|
for (i = 0; i < ARRAY_SIZE(ab8500_fg_irq_th); i++) {
|
||||||
irq = platform_get_irq_byname(pdev, ab8500_fg_irq[i].name);
|
irq = platform_get_irq_byname(pdev, ab8500_fg_irq_th[i].name);
|
||||||
free_irq(irq, di);
|
free_irq(irq, di);
|
||||||
}
|
}
|
||||||
|
irq = platform_get_irq_byname(pdev, ab8500_fg_irq_bh[0].name);
|
||||||
|
free_irq(irq, di);
|
||||||
free_inst_curr_wq:
|
free_inst_curr_wq:
|
||||||
destroy_workqueue(di->fg_wq);
|
destroy_workqueue(di->fg_wq);
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -50,9 +50,6 @@
|
||||||
#define CHARGALG_CURR_STEP_LOW 0
|
#define CHARGALG_CURR_STEP_LOW 0
|
||||||
#define CHARGALG_CURR_STEP_HIGH 100
|
#define CHARGALG_CURR_STEP_HIGH 100
|
||||||
|
|
||||||
#define to_abx500_chargalg_device_info(x) container_of((x), \
|
|
||||||
struct abx500_chargalg, chargalg_psy);
|
|
||||||
|
|
||||||
enum abx500_chargers {
|
enum abx500_chargers {
|
||||||
NO_CHG,
|
NO_CHG,
|
||||||
AC_CHG,
|
AC_CHG,
|
||||||
|
@ -256,7 +253,7 @@ struct abx500_chargalg {
|
||||||
struct ab8500 *parent;
|
struct ab8500 *parent;
|
||||||
struct abx500_chargalg_current_step_status curr_status;
|
struct abx500_chargalg_current_step_status curr_status;
|
||||||
struct abx500_bm_data *bm;
|
struct abx500_bm_data *bm;
|
||||||
struct power_supply chargalg_psy;
|
struct power_supply *chargalg_psy;
|
||||||
struct ux500_charger *ac_chg;
|
struct ux500_charger *ac_chg;
|
||||||
struct ux500_charger *usb_chg;
|
struct ux500_charger *usb_chg;
|
||||||
struct abx500_chargalg_events events;
|
struct abx500_chargalg_events events;
|
||||||
|
@ -695,7 +692,7 @@ static void abx500_chargalg_stop_charging(struct abx500_chargalg *di)
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
di->charge_status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
||||||
di->maintenance_chg = false;
|
di->maintenance_chg = false;
|
||||||
cancel_delayed_work(&di->chargalg_wd_work);
|
cancel_delayed_work(&di->chargalg_wd_work);
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -715,7 +712,7 @@ static void abx500_chargalg_hold_charging(struct abx500_chargalg *di)
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
di->maintenance_chg = false;
|
di->maintenance_chg = false;
|
||||||
cancel_delayed_work(&di->chargalg_wd_work);
|
cancel_delayed_work(&di->chargalg_wd_work);
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -842,7 +839,7 @@ static void abx500_chargalg_end_of_charge(struct abx500_chargalg *di)
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_FULL;
|
di->charge_status = POWER_SUPPLY_STATUS_FULL;
|
||||||
di->maintenance_chg = true;
|
di->maintenance_chg = true;
|
||||||
dev_dbg(di->dev, "EOC reached!\n");
|
dev_dbg(di->dev, "EOC reached!\n");
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
} else {
|
} else {
|
||||||
dev_dbg(di->dev,
|
dev_dbg(di->dev,
|
||||||
" EOC limit reached for the %d"
|
" EOC limit reached for the %d"
|
||||||
|
@ -987,10 +984,10 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
|
|
||||||
psy = (struct power_supply *)data;
|
psy = (struct power_supply *)data;
|
||||||
ext = dev_get_drvdata(dev);
|
ext = dev_get_drvdata(dev);
|
||||||
di = to_abx500_chargalg_device_info(psy);
|
di = power_supply_get_drvdata(psy);
|
||||||
/* For all psy where the driver name appears in any supplied_to */
|
/* For all psy where the driver name appears in any supplied_to */
|
||||||
for (i = 0; i < ext->num_supplicants; i++) {
|
for (i = 0; i < ext->num_supplicants; i++) {
|
||||||
if (!strcmp(ext->supplied_to[i], psy->name))
|
if (!strcmp(ext->supplied_to[i], psy->desc->name))
|
||||||
psy_found = true;
|
psy_found = true;
|
||||||
}
|
}
|
||||||
if (!psy_found)
|
if (!psy_found)
|
||||||
|
@ -1001,29 +998,31 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
* property because of handling that sysfs entry on its own, this is
|
* property because of handling that sysfs entry on its own, this is
|
||||||
* the place to get the battery capacity.
|
* the place to get the battery capacity.
|
||||||
*/
|
*/
|
||||||
if (!ext->get_property(ext, POWER_SUPPLY_PROP_CAPACITY, &ret)) {
|
if (!power_supply_get_property(ext, POWER_SUPPLY_PROP_CAPACITY, &ret)) {
|
||||||
di->batt_data.percent = ret.intval;
|
di->batt_data.percent = ret.intval;
|
||||||
capacity_updated = true;
|
capacity_updated = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Go through all properties for the psy */
|
/* Go through all properties for the psy */
|
||||||
for (j = 0; j < ext->num_properties; j++) {
|
for (j = 0; j < ext->desc->num_properties; j++) {
|
||||||
enum power_supply_property prop;
|
enum power_supply_property prop;
|
||||||
prop = ext->properties[j];
|
prop = ext->desc->properties[j];
|
||||||
|
|
||||||
/* Initialize chargers if not already done */
|
/*
|
||||||
|
* Initialize chargers if not already done.
|
||||||
|
* The ab8500_charger*/
|
||||||
if (!di->ac_chg &&
|
if (!di->ac_chg &&
|
||||||
ext->type == POWER_SUPPLY_TYPE_MAINS)
|
ext->desc->type == POWER_SUPPLY_TYPE_MAINS)
|
||||||
di->ac_chg = psy_to_ux500_charger(ext);
|
di->ac_chg = psy_to_ux500_charger(ext);
|
||||||
else if (!di->usb_chg &&
|
else if (!di->usb_chg &&
|
||||||
ext->type == POWER_SUPPLY_TYPE_USB)
|
ext->desc->type == POWER_SUPPLY_TYPE_USB)
|
||||||
di->usb_chg = psy_to_ux500_charger(ext);
|
di->usb_chg = psy_to_ux500_charger(ext);
|
||||||
|
|
||||||
if (ext->get_property(ext, prop, &ret))
|
if (power_supply_get_property(ext, prop, &ret))
|
||||||
continue;
|
continue;
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
case POWER_SUPPLY_PROP_PRESENT:
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
/* Battery present */
|
/* Battery present */
|
||||||
if (ret.intval)
|
if (ret.intval)
|
||||||
|
@ -1070,7 +1069,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_ONLINE:
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_TYPE_MAINS:
|
case POWER_SUPPLY_TYPE_MAINS:
|
||||||
|
@ -1115,7 +1114,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_HEALTH:
|
case POWER_SUPPLY_PROP_HEALTH:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_TYPE_MAINS:
|
case POWER_SUPPLY_TYPE_MAINS:
|
||||||
|
@ -1198,7 +1197,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
di->batt_data.volt = ret.intval / 1000;
|
di->batt_data.volt = ret.intval / 1000;
|
||||||
break;
|
break;
|
||||||
|
@ -1214,7 +1213,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_VOLTAGE_AVG:
|
case POWER_SUPPLY_PROP_VOLTAGE_AVG:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_MAINS:
|
case POWER_SUPPLY_TYPE_MAINS:
|
||||||
/* AVG is used to indicate when we are
|
/* AVG is used to indicate when we are
|
||||||
* in CV mode */
|
* in CV mode */
|
||||||
|
@ -1239,7 +1238,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
if (ret.intval)
|
if (ret.intval)
|
||||||
di->events.batt_unknown = false;
|
di->events.batt_unknown = false;
|
||||||
|
@ -1257,7 +1256,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_CURRENT_NOW:
|
case POWER_SUPPLY_PROP_CURRENT_NOW:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_MAINS:
|
case POWER_SUPPLY_TYPE_MAINS:
|
||||||
di->chg_info.ac_curr =
|
di->chg_info.ac_curr =
|
||||||
ret.intval / 1000;
|
ret.intval / 1000;
|
||||||
|
@ -1275,7 +1274,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case POWER_SUPPLY_PROP_CURRENT_AVG:
|
case POWER_SUPPLY_PROP_CURRENT_AVG:
|
||||||
switch (ext->type) {
|
switch (ext->desc->type) {
|
||||||
case POWER_SUPPLY_TYPE_BATTERY:
|
case POWER_SUPPLY_TYPE_BATTERY:
|
||||||
di->batt_data.avg_curr = ret.intval / 1000;
|
di->batt_data.avg_curr = ret.intval / 1000;
|
||||||
break;
|
break;
|
||||||
|
@ -1311,7 +1310,7 @@ static int abx500_chargalg_get_ext_psy_data(struct device *dev, void *data)
|
||||||
*/
|
*/
|
||||||
static void abx500_chargalg_external_power_changed(struct power_supply *psy)
|
static void abx500_chargalg_external_power_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct abx500_chargalg *di = to_abx500_chargalg_device_info(psy);
|
struct abx500_chargalg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Trigger execution of the algorithm instantly and read
|
* Trigger execution of the algorithm instantly and read
|
||||||
|
@ -1336,7 +1335,7 @@ static void abx500_chargalg_algorithm(struct abx500_chargalg *di)
|
||||||
|
|
||||||
/* Collect data from all power_supply class devices */
|
/* Collect data from all power_supply class devices */
|
||||||
class_for_each_device(power_supply_class, NULL,
|
class_for_each_device(power_supply_class, NULL,
|
||||||
&di->chargalg_psy, abx500_chargalg_get_ext_psy_data);
|
di->chargalg_psy, abx500_chargalg_get_ext_psy_data);
|
||||||
|
|
||||||
abx500_chargalg_end_of_charge(di);
|
abx500_chargalg_end_of_charge(di);
|
||||||
abx500_chargalg_check_temp(di);
|
abx500_chargalg_check_temp(di);
|
||||||
|
@ -1478,7 +1477,7 @@ static void abx500_chargalg_algorithm(struct abx500_chargalg *di)
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
di->charge_status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
||||||
di->maintenance_chg = false;
|
di->maintenance_chg = false;
|
||||||
abx500_chargalg_state_to(di, STATE_SUSPENDED);
|
abx500_chargalg_state_to(di, STATE_SUSPENDED);
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
/* Intentional fallthrough */
|
/* Intentional fallthrough */
|
||||||
|
|
||||||
case STATE_SUSPENDED:
|
case STATE_SUSPENDED:
|
||||||
|
@ -1576,7 +1575,7 @@ static void abx500_chargalg_algorithm(struct abx500_chargalg *di)
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
di->eoc_cnt = 0;
|
di->eoc_cnt = 0;
|
||||||
di->maintenance_chg = false;
|
di->maintenance_chg = false;
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -1624,7 +1623,7 @@ static void abx500_chargalg_algorithm(struct abx500_chargalg *di)
|
||||||
di->bm->bat_type[
|
di->bm->bat_type[
|
||||||
di->bm->batt_id].maint_a_cur_lvl);
|
di->bm->batt_id].maint_a_cur_lvl);
|
||||||
abx500_chargalg_state_to(di, STATE_MAINTENANCE_A);
|
abx500_chargalg_state_to(di, STATE_MAINTENANCE_A);
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
/* Intentional fallthrough*/
|
/* Intentional fallthrough*/
|
||||||
|
|
||||||
case STATE_MAINTENANCE_A:
|
case STATE_MAINTENANCE_A:
|
||||||
|
@ -1644,7 +1643,7 @@ static void abx500_chargalg_algorithm(struct abx500_chargalg *di)
|
||||||
di->bm->bat_type[
|
di->bm->bat_type[
|
||||||
di->bm->batt_id].maint_b_cur_lvl);
|
di->bm->batt_id].maint_b_cur_lvl);
|
||||||
abx500_chargalg_state_to(di, STATE_MAINTENANCE_B);
|
abx500_chargalg_state_to(di, STATE_MAINTENANCE_B);
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
/* Intentional fallthrough*/
|
/* Intentional fallthrough*/
|
||||||
|
|
||||||
case STATE_MAINTENANCE_B:
|
case STATE_MAINTENANCE_B:
|
||||||
|
@ -1663,7 +1662,7 @@ static void abx500_chargalg_algorithm(struct abx500_chargalg *di)
|
||||||
abx500_chargalg_stop_maintenance_timer(di);
|
abx500_chargalg_stop_maintenance_timer(di);
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
abx500_chargalg_state_to(di, STATE_TEMP_LOWHIGH);
|
abx500_chargalg_state_to(di, STATE_TEMP_LOWHIGH);
|
||||||
power_supply_changed(&di->chargalg_psy);
|
power_supply_changed(di->chargalg_psy);
|
||||||
/* Intentional fallthrough */
|
/* Intentional fallthrough */
|
||||||
|
|
||||||
case STATE_TEMP_LOWHIGH:
|
case STATE_TEMP_LOWHIGH:
|
||||||
|
@ -1779,9 +1778,7 @@ static int abx500_chargalg_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 abx500_chargalg *di;
|
struct abx500_chargalg *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
di = to_abx500_chargalg_device_info(psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -2034,7 +2031,7 @@ static int abx500_chargalg_remove(struct platform_device *pdev)
|
||||||
/* Delete the work queue */
|
/* Delete the work queue */
|
||||||
destroy_workqueue(di->chargalg_wq);
|
destroy_workqueue(di->chargalg_wq);
|
||||||
|
|
||||||
power_supply_unregister(&di->chargalg_psy);
|
power_supply_unregister(di->chargalg_psy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -2043,10 +2040,20 @@ static char *supply_interface[] = {
|
||||||
"ab8500_fg",
|
"ab8500_fg",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc abx500_chargalg_desc = {
|
||||||
|
.name = "abx500_chargalg",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = abx500_chargalg_props,
|
||||||
|
.num_properties = ARRAY_SIZE(abx500_chargalg_props),
|
||||||
|
.get_property = abx500_chargalg_get_property,
|
||||||
|
.external_power_changed = abx500_chargalg_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static int abx500_chargalg_probe(struct platform_device *pdev)
|
static int abx500_chargalg_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct device_node *np = pdev->dev.of_node;
|
struct device_node *np = pdev->dev.of_node;
|
||||||
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
struct abx500_bm_data *plat = pdev->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct abx500_chargalg *di;
|
struct abx500_chargalg *di;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
|
@ -2074,16 +2081,9 @@ static int abx500_chargalg_probe(struct platform_device *pdev)
|
||||||
di->dev = &pdev->dev;
|
di->dev = &pdev->dev;
|
||||||
di->parent = dev_get_drvdata(pdev->dev.parent);
|
di->parent = dev_get_drvdata(pdev->dev.parent);
|
||||||
|
|
||||||
/* chargalg supply */
|
psy_cfg.supplied_to = supply_interface;
|
||||||
di->chargalg_psy.name = "abx500_chargalg";
|
psy_cfg.num_supplicants = ARRAY_SIZE(supply_interface);
|
||||||
di->chargalg_psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_cfg.drv_data = di;
|
||||||
di->chargalg_psy.properties = abx500_chargalg_props;
|
|
||||||
di->chargalg_psy.num_properties = ARRAY_SIZE(abx500_chargalg_props);
|
|
||||||
di->chargalg_psy.get_property = abx500_chargalg_get_property;
|
|
||||||
di->chargalg_psy.supplied_to = supply_interface;
|
|
||||||
di->chargalg_psy.num_supplicants = ARRAY_SIZE(supply_interface),
|
|
||||||
di->chargalg_psy.external_power_changed =
|
|
||||||
abx500_chargalg_external_power_changed;
|
|
||||||
|
|
||||||
/* Initilialize safety timer */
|
/* Initilialize safety timer */
|
||||||
hrtimer_init(&di->safety_timer, CLOCK_REALTIME, HRTIMER_MODE_ABS);
|
hrtimer_init(&di->safety_timer, CLOCK_REALTIME, HRTIMER_MODE_ABS);
|
||||||
|
@ -2115,9 +2115,11 @@ static int abx500_chargalg_probe(struct platform_device *pdev)
|
||||||
di->chg_info.prev_conn_chg = -1;
|
di->chg_info.prev_conn_chg = -1;
|
||||||
|
|
||||||
/* Register chargalg power supply class */
|
/* Register chargalg power supply class */
|
||||||
ret = power_supply_register(di->dev, &di->chargalg_psy);
|
di->chargalg_psy = power_supply_register(di->dev, &abx500_chargalg_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(di->chargalg_psy)) {
|
||||||
dev_err(di->dev, "failed to register chargalg psy\n");
|
dev_err(di->dev, "failed to register chargalg psy\n");
|
||||||
|
ret = PTR_ERR(di->chargalg_psy);
|
||||||
goto free_chargalg_wq;
|
goto free_chargalg_wq;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2138,7 +2140,7 @@ static int abx500_chargalg_probe(struct platform_device *pdev)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
free_psy:
|
free_psy:
|
||||||
power_supply_unregister(&di->chargalg_psy);
|
power_supply_unregister(di->chargalg_psy);
|
||||||
free_chargalg_wq:
|
free_chargalg_wq:
|
||||||
destroy_workqueue(di->chargalg_wq);
|
destroy_workqueue(di->chargalg_wq);
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -15,10 +15,10 @@
|
||||||
#include <linux/apm-emulation.h>
|
#include <linux/apm-emulation.h>
|
||||||
|
|
||||||
|
|
||||||
#define PSY_PROP(psy, prop, val) (psy->get_property(psy, \
|
#define PSY_PROP(psy, prop, val) (power_supply_get_property(psy, \
|
||||||
POWER_SUPPLY_PROP_##prop, val))
|
POWER_SUPPLY_PROP_##prop, val))
|
||||||
|
|
||||||
#define _MPSY_PROP(prop, val) (main_battery->get_property(main_battery, \
|
#define _MPSY_PROP(prop, val) (power_supply_get_property(main_battery, \
|
||||||
prop, val))
|
prop, val))
|
||||||
|
|
||||||
#define MPSY_PROP(prop, val) _MPSY_PROP(POWER_SUPPLY_PROP_##prop, val)
|
#define MPSY_PROP(prop, val) _MPSY_PROP(POWER_SUPPLY_PROP_##prop, val)
|
||||||
|
@ -48,7 +48,7 @@ static int __find_main_battery(struct device *dev, void *data)
|
||||||
|
|
||||||
bp->bat = dev_get_drvdata(dev);
|
bp->bat = dev_get_drvdata(dev);
|
||||||
|
|
||||||
if (bp->bat->use_for_apm) {
|
if (bp->bat->desc->use_for_apm) {
|
||||||
/* nice, we explicitly asked to report this battery. */
|
/* nice, we explicitly asked to report this battery. */
|
||||||
bp->main = bp->bat;
|
bp->main = bp->bat;
|
||||||
return 1;
|
return 1;
|
||||||
|
|
1154
drivers/power/axp288_fuel_gauge.c
Normal file
1154
drivers/power/axp288_fuel_gauge.c
Normal file
File diff suppressed because it is too large
Load diff
|
@ -13,12 +13,6 @@
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU General Public License for more details.
|
* GNU General Public License for more details.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU General Public License along
|
|
||||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
||||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Datasheets:
|
* Datasheets:
|
||||||
* http://www.ti.com/product/bq24150
|
* http://www.ti.com/product/bq24150
|
||||||
* http://www.ti.com/product/bq24150a
|
* http://www.ti.com/product/bq24150a
|
||||||
|
@ -26,6 +20,8 @@
|
||||||
* http://www.ti.com/product/bq24153
|
* http://www.ti.com/product/bq24153
|
||||||
* http://www.ti.com/product/bq24153a
|
* http://www.ti.com/product/bq24153a
|
||||||
* http://www.ti.com/product/bq24155
|
* http://www.ti.com/product/bq24155
|
||||||
|
* http://www.ti.com/product/bq24157s
|
||||||
|
* http://www.ti.com/product/bq24158
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
|
@ -147,6 +143,7 @@ enum bq2415x_chip {
|
||||||
BQ24155,
|
BQ24155,
|
||||||
BQ24156,
|
BQ24156,
|
||||||
BQ24156A,
|
BQ24156A,
|
||||||
|
BQ24157S,
|
||||||
BQ24158,
|
BQ24158,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -162,18 +159,20 @@ static char *bq2415x_chip_name[] = {
|
||||||
"bq24155",
|
"bq24155",
|
||||||
"bq24156",
|
"bq24156",
|
||||||
"bq24156a",
|
"bq24156a",
|
||||||
|
"bq24157s",
|
||||||
"bq24158",
|
"bq24158",
|
||||||
};
|
};
|
||||||
|
|
||||||
struct bq2415x_device {
|
struct bq2415x_device {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct bq2415x_platform_data init_data;
|
struct bq2415x_platform_data init_data;
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
|
struct power_supply_desc charger_desc;
|
||||||
struct delayed_work work;
|
struct delayed_work work;
|
||||||
struct power_supply *notify_psy;
|
struct power_supply *notify_psy;
|
||||||
struct notifier_block nb;
|
struct notifier_block nb;
|
||||||
enum bq2415x_mode reported_mode;/* mode reported by hook function */
|
enum bq2415x_mode reported_mode;/* mode reported by hook function */
|
||||||
enum bq2415x_mode mode; /* current configured mode */
|
enum bq2415x_mode mode; /* currently configured mode */
|
||||||
enum bq2415x_chip chip;
|
enum bq2415x_chip chip;
|
||||||
const char *timer_error;
|
const char *timer_error;
|
||||||
char *model;
|
char *model;
|
||||||
|
@ -352,8 +351,7 @@ static int bq2415x_exec_command(struct bq2415x_device *bq,
|
||||||
BQ2415X_BIT_CE);
|
BQ2415X_BIT_CE);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
else
|
return ret > 0 ? 0 : 1;
|
||||||
return ret > 0 ? 0 : 1;
|
|
||||||
case BQ2415X_CHARGER_ENABLE:
|
case BQ2415X_CHARGER_ENABLE:
|
||||||
return bq2415x_i2c_write_bit(bq, BQ2415X_REG_CONTROL,
|
return bq2415x_i2c_write_bit(bq, BQ2415X_REG_CONTROL,
|
||||||
0, BQ2415X_BIT_CE);
|
0, BQ2415X_BIT_CE);
|
||||||
|
@ -426,20 +424,17 @@ static enum bq2415x_chip bq2415x_detect_chip(struct bq2415x_device *bq)
|
||||||
case 0:
|
case 0:
|
||||||
if (bq->chip == BQ24151A)
|
if (bq->chip == BQ24151A)
|
||||||
return bq->chip;
|
return bq->chip;
|
||||||
else
|
return BQ24151;
|
||||||
return BQ24151;
|
|
||||||
case 1:
|
case 1:
|
||||||
if (bq->chip == BQ24150A ||
|
if (bq->chip == BQ24150A ||
|
||||||
bq->chip == BQ24152 ||
|
bq->chip == BQ24152 ||
|
||||||
bq->chip == BQ24155)
|
bq->chip == BQ24155)
|
||||||
return bq->chip;
|
return bq->chip;
|
||||||
else
|
return BQ24150;
|
||||||
return BQ24150;
|
|
||||||
case 2:
|
case 2:
|
||||||
if (bq->chip == BQ24153A)
|
if (bq->chip == BQ24153A)
|
||||||
return bq->chip;
|
return bq->chip;
|
||||||
else
|
return BQ24153;
|
||||||
return BQ24153;
|
|
||||||
default:
|
default:
|
||||||
return BQUNKNOWN;
|
return BQUNKNOWN;
|
||||||
}
|
}
|
||||||
|
@ -450,9 +445,10 @@ static enum bq2415x_chip bq2415x_detect_chip(struct bq2415x_device *bq)
|
||||||
case 0:
|
case 0:
|
||||||
if (bq->chip == BQ24156A)
|
if (bq->chip == BQ24156A)
|
||||||
return bq->chip;
|
return bq->chip;
|
||||||
else
|
return BQ24156;
|
||||||
return BQ24156;
|
|
||||||
case 2:
|
case 2:
|
||||||
|
if (bq->chip == BQ24157S)
|
||||||
|
return bq->chip;
|
||||||
return BQ24158;
|
return BQ24158;
|
||||||
default:
|
default:
|
||||||
return BQUNKNOWN;
|
return BQUNKNOWN;
|
||||||
|
@ -480,24 +476,22 @@ static int bq2415x_detect_revision(struct bq2415x_device *bq)
|
||||||
case BQ24152:
|
case BQ24152:
|
||||||
if (ret >= 0 && ret <= 3)
|
if (ret >= 0 && ret <= 3)
|
||||||
return ret;
|
return ret;
|
||||||
else
|
return -1;
|
||||||
return -1;
|
|
||||||
case BQ24153:
|
case BQ24153:
|
||||||
case BQ24153A:
|
case BQ24153A:
|
||||||
case BQ24156:
|
case BQ24156:
|
||||||
case BQ24156A:
|
case BQ24156A:
|
||||||
|
case BQ24157S:
|
||||||
case BQ24158:
|
case BQ24158:
|
||||||
if (ret == 3)
|
if (ret == 3)
|
||||||
return 0;
|
return 0;
|
||||||
else if (ret == 1)
|
else if (ret == 1)
|
||||||
return 1;
|
return 1;
|
||||||
else
|
return -1;
|
||||||
return -1;
|
|
||||||
case BQ24155:
|
case BQ24155:
|
||||||
if (ret == 3)
|
if (ret == 3)
|
||||||
return 3;
|
return 3;
|
||||||
else
|
return -1;
|
||||||
return -1;
|
|
||||||
case BQUNKNOWN:
|
case BQUNKNOWN:
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
@ -791,7 +785,7 @@ static int bq2415x_set_mode(struct bq2415x_device *bq, enum bq2415x_mode mode)
|
||||||
bq2415x_set_default_value(bq, battery_regulation_voltage);
|
bq2415x_set_default_value(bq, battery_regulation_voltage);
|
||||||
|
|
||||||
bq->mode = mode;
|
bq->mode = mode;
|
||||||
sysfs_notify(&bq->charger.dev->kobj, NULL, "mode");
|
sysfs_notify(&bq->charger->dev.kobj, NULL, "mode");
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -816,7 +810,8 @@ static int bq2415x_notifier_call(struct notifier_block *nb,
|
||||||
|
|
||||||
dev_dbg(bq->dev, "notifier call was called\n");
|
dev_dbg(bq->dev, "notifier call was called\n");
|
||||||
|
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_CURRENT_MAX, &prop);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_CURRENT_MAX,
|
||||||
|
&prop);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
return NOTIFY_OK;
|
return NOTIFY_OK;
|
||||||
|
|
||||||
|
@ -874,7 +869,7 @@ static void bq2415x_set_autotimer(struct bq2415x_device *bq, int state)
|
||||||
static void bq2415x_timer_error(struct bq2415x_device *bq, const char *msg)
|
static void bq2415x_timer_error(struct bq2415x_device *bq, const char *msg)
|
||||||
{
|
{
|
||||||
bq->timer_error = msg;
|
bq->timer_error = msg;
|
||||||
sysfs_notify(&bq->charger.dev->kobj, NULL, "timer");
|
sysfs_notify(&bq->charger->dev.kobj, NULL, "timer");
|
||||||
dev_err(bq->dev, "%s\n", msg);
|
dev_err(bq->dev, "%s\n", msg);
|
||||||
if (bq->automode > 0)
|
if (bq->automode > 0)
|
||||||
bq->automode = 0;
|
bq->automode = 0;
|
||||||
|
@ -892,7 +887,7 @@ static void bq2415x_timer_work(struct work_struct *work)
|
||||||
int boost;
|
int boost;
|
||||||
|
|
||||||
if (bq->automode > 0 && (bq->reported_mode != bq->mode)) {
|
if (bq->automode > 0 && (bq->reported_mode != bq->mode)) {
|
||||||
sysfs_notify(&bq->charger.dev->kobj, NULL, "reported_mode");
|
sysfs_notify(&bq->charger->dev.kobj, NULL, "reported_mode");
|
||||||
bq2415x_set_mode(bq, bq->reported_mode);
|
bq2415x_set_mode(bq, bq->reported_mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -998,8 +993,7 @@ static int bq2415x_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 bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -1030,12 +1024,14 @@ static int bq2415x_power_supply_init(struct bq2415x_device *bq)
|
||||||
int ret;
|
int ret;
|
||||||
int chip;
|
int chip;
|
||||||
char revstr[8];
|
char revstr[8];
|
||||||
|
struct power_supply_config psy_cfg = { .drv_data = bq, };
|
||||||
|
|
||||||
bq->charger.name = bq->name;
|
bq->charger_desc.name = bq->name;
|
||||||
bq->charger.type = POWER_SUPPLY_TYPE_USB;
|
bq->charger_desc.type = POWER_SUPPLY_TYPE_USB;
|
||||||
bq->charger.properties = bq2415x_power_supply_props;
|
bq->charger_desc.properties = bq2415x_power_supply_props;
|
||||||
bq->charger.num_properties = ARRAY_SIZE(bq2415x_power_supply_props);
|
bq->charger_desc.num_properties =
|
||||||
bq->charger.get_property = bq2415x_power_supply_get_property;
|
ARRAY_SIZE(bq2415x_power_supply_props);
|
||||||
|
bq->charger_desc.get_property = bq2415x_power_supply_get_property;
|
||||||
|
|
||||||
ret = bq2415x_detect_chip(bq);
|
ret = bq2415x_detect_chip(bq);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
|
@ -1058,10 +1054,11 @@ static int bq2415x_power_supply_init(struct bq2415x_device *bq)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(bq->dev, &bq->charger);
|
bq->charger = power_supply_register(bq->dev, &bq->charger_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(bq->charger)) {
|
||||||
kfree(bq->model);
|
kfree(bq->model);
|
||||||
return ret;
|
return PTR_ERR(bq->charger);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1073,7 +1070,7 @@ static void bq2415x_power_supply_exit(struct bq2415x_device *bq)
|
||||||
if (bq->automode > 0)
|
if (bq->automode > 0)
|
||||||
bq->automode = 0;
|
bq->automode = 0;
|
||||||
cancel_delayed_work_sync(&bq->work);
|
cancel_delayed_work_sync(&bq->work);
|
||||||
power_supply_unregister(&bq->charger);
|
power_supply_unregister(bq->charger);
|
||||||
kfree(bq->model);
|
kfree(bq->model);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1085,8 +1082,7 @@ static ssize_t bq2415x_sysfs_show_status(struct device *dev,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
enum bq2415x_command command;
|
enum bq2415x_command command;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -1119,8 +1115,7 @@ static ssize_t bq2415x_sysfs_set_timer(struct device *dev,
|
||||||
size_t count)
|
size_t count)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
if (strncmp(buf, "auto", 4) == 0)
|
if (strncmp(buf, "auto", 4) == 0)
|
||||||
|
@ -1141,8 +1136,7 @@ static ssize_t bq2415x_sysfs_show_timer(struct device *dev,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
|
|
||||||
if (bq->timer_error)
|
if (bq->timer_error)
|
||||||
return sprintf(buf, "%s\n", bq->timer_error);
|
return sprintf(buf, "%s\n", bq->timer_error);
|
||||||
|
@ -1166,8 +1160,7 @@ static ssize_t bq2415x_sysfs_set_mode(struct device *dev,
|
||||||
size_t count)
|
size_t count)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
enum bq2415x_mode mode;
|
enum bq2415x_mode mode;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
|
@ -1219,8 +1212,7 @@ static ssize_t bq2415x_sysfs_show_mode(struct device *dev,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
ssize_t ret = 0;
|
ssize_t ret = 0;
|
||||||
|
|
||||||
if (bq->automode > 0)
|
if (bq->automode > 0)
|
||||||
|
@ -1257,8 +1249,7 @@ static ssize_t bq2415x_sysfs_show_reported_mode(struct device *dev,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
|
|
||||||
if (bq->automode < 0)
|
if (bq->automode < 0)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -1286,8 +1277,7 @@ static ssize_t bq2415x_sysfs_set_registers(struct device *dev,
|
||||||
size_t count)
|
size_t count)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
ssize_t ret = 0;
|
ssize_t ret = 0;
|
||||||
unsigned int reg;
|
unsigned int reg;
|
||||||
unsigned int val;
|
unsigned int val;
|
||||||
|
@ -1322,8 +1312,7 @@ static ssize_t bq2415x_sysfs_show_registers(struct device *dev,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
ssize_t ret = 0;
|
ssize_t ret = 0;
|
||||||
|
|
||||||
ret += bq2415x_sysfs_print_reg(bq, BQ2415X_REG_STATUS, buf+ret);
|
ret += bq2415x_sysfs_print_reg(bq, BQ2415X_REG_STATUS, buf+ret);
|
||||||
|
@ -1341,8 +1330,7 @@ static ssize_t bq2415x_sysfs_set_limit(struct device *dev,
|
||||||
size_t count)
|
size_t count)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
long val;
|
long val;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -1373,8 +1361,7 @@ static ssize_t bq2415x_sysfs_show_limit(struct device *dev,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (strcmp(attr->attr.name, "current_limit") == 0)
|
if (strcmp(attr->attr.name, "current_limit") == 0)
|
||||||
|
@ -1402,8 +1389,7 @@ static ssize_t bq2415x_sysfs_set_enable(struct device *dev,
|
||||||
size_t count)
|
size_t count)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
enum bq2415x_command command;
|
enum bq2415x_command command;
|
||||||
long val;
|
long val;
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -1438,8 +1424,7 @@ static ssize_t bq2415x_sysfs_show_enable(struct device *dev,
|
||||||
char *buf)
|
char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq2415x_device *bq = container_of(psy, struct bq2415x_device,
|
struct bq2415x_device *bq = power_supply_get_drvdata(psy);
|
||||||
charger);
|
|
||||||
enum bq2415x_command command;
|
enum bq2415x_command command;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -1530,13 +1515,13 @@ static const struct attribute_group bq2415x_sysfs_attr_group = {
|
||||||
|
|
||||||
static int bq2415x_sysfs_init(struct bq2415x_device *bq)
|
static int bq2415x_sysfs_init(struct bq2415x_device *bq)
|
||||||
{
|
{
|
||||||
return sysfs_create_group(&bq->charger.dev->kobj,
|
return sysfs_create_group(&bq->charger->dev.kobj,
|
||||||
&bq2415x_sysfs_attr_group);
|
&bq2415x_sysfs_attr_group);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bq2415x_sysfs_exit(struct bq2415x_device *bq)
|
static void bq2415x_sysfs_exit(struct bq2415x_device *bq)
|
||||||
{
|
{
|
||||||
sysfs_remove_group(&bq->charger.dev->kobj, &bq2415x_sysfs_attr_group);
|
sysfs_remove_group(&bq->charger->dev.kobj, &bq2415x_sysfs_attr_group);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* main bq2415x probe function */
|
/* main bq2415x probe function */
|
||||||
|
@ -1609,27 +1594,27 @@ static int bq2415x_probe(struct i2c_client *client,
|
||||||
ret = of_property_read_u32(np, "ti,current-limit",
|
ret = of_property_read_u32(np, "ti,current-limit",
|
||||||
&bq->init_data.current_limit);
|
&bq->init_data.current_limit);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_2;
|
goto error_3;
|
||||||
ret = of_property_read_u32(np, "ti,weak-battery-voltage",
|
ret = of_property_read_u32(np, "ti,weak-battery-voltage",
|
||||||
&bq->init_data.weak_battery_voltage);
|
&bq->init_data.weak_battery_voltage);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_2;
|
goto error_3;
|
||||||
ret = of_property_read_u32(np, "ti,battery-regulation-voltage",
|
ret = of_property_read_u32(np, "ti,battery-regulation-voltage",
|
||||||
&bq->init_data.battery_regulation_voltage);
|
&bq->init_data.battery_regulation_voltage);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_2;
|
goto error_3;
|
||||||
ret = of_property_read_u32(np, "ti,charge-current",
|
ret = of_property_read_u32(np, "ti,charge-current",
|
||||||
&bq->init_data.charge_current);
|
&bq->init_data.charge_current);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_2;
|
goto error_3;
|
||||||
ret = of_property_read_u32(np, "ti,termination-current",
|
ret = of_property_read_u32(np, "ti,termination-current",
|
||||||
&bq->init_data.termination_current);
|
&bq->init_data.termination_current);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_2;
|
goto error_3;
|
||||||
ret = of_property_read_u32(np, "ti,resistor-sense",
|
ret = of_property_read_u32(np, "ti,resistor-sense",
|
||||||
&bq->init_data.resistor_sense);
|
&bq->init_data.resistor_sense);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_2;
|
goto error_3;
|
||||||
} else {
|
} else {
|
||||||
memcpy(&bq->init_data, pdata, sizeof(bq->init_data));
|
memcpy(&bq->init_data, pdata, sizeof(bq->init_data));
|
||||||
}
|
}
|
||||||
|
@ -1639,19 +1624,19 @@ static int bq2415x_probe(struct i2c_client *client,
|
||||||
ret = bq2415x_power_supply_init(bq);
|
ret = bq2415x_power_supply_init(bq);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(bq->dev, "failed to register power supply: %d\n", ret);
|
dev_err(bq->dev, "failed to register power supply: %d\n", ret);
|
||||||
goto error_2;
|
goto error_3;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = bq2415x_sysfs_init(bq);
|
ret = bq2415x_sysfs_init(bq);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(bq->dev, "failed to create sysfs entries: %d\n", ret);
|
dev_err(bq->dev, "failed to create sysfs entries: %d\n", ret);
|
||||||
goto error_3;
|
goto error_4;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = bq2415x_set_defaults(bq);
|
ret = bq2415x_set_defaults(bq);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(bq->dev, "failed to set default values: %d\n", ret);
|
dev_err(bq->dev, "failed to set default values: %d\n", ret);
|
||||||
goto error_4;
|
goto error_5;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bq->notify_psy) {
|
if (bq->notify_psy) {
|
||||||
|
@ -1659,7 +1644,7 @@ static int bq2415x_probe(struct i2c_client *client,
|
||||||
ret = power_supply_reg_notifier(&bq->nb);
|
ret = power_supply_reg_notifier(&bq->nb);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(bq->dev, "failed to reg notifier: %d\n", ret);
|
dev_err(bq->dev, "failed to reg notifier: %d\n", ret);
|
||||||
goto error_5;
|
goto error_6;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Query for initial reported_mode and set it */
|
/* Query for initial reported_mode and set it */
|
||||||
|
@ -1679,11 +1664,14 @@ static int bq2415x_probe(struct i2c_client *client,
|
||||||
dev_info(bq->dev, "driver registered\n");
|
dev_info(bq->dev, "driver registered\n");
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
error_6:
|
||||||
error_5:
|
error_5:
|
||||||
error_4:
|
|
||||||
bq2415x_sysfs_exit(bq);
|
bq2415x_sysfs_exit(bq);
|
||||||
error_3:
|
error_4:
|
||||||
bq2415x_power_supply_exit(bq);
|
bq2415x_power_supply_exit(bq);
|
||||||
|
error_3:
|
||||||
|
if (bq->notify_psy)
|
||||||
|
power_supply_put(bq->notify_psy);
|
||||||
error_2:
|
error_2:
|
||||||
kfree(name);
|
kfree(name);
|
||||||
error_1:
|
error_1:
|
||||||
|
@ -1700,8 +1688,10 @@ static int bq2415x_remove(struct i2c_client *client)
|
||||||
{
|
{
|
||||||
struct bq2415x_device *bq = i2c_get_clientdata(client);
|
struct bq2415x_device *bq = i2c_get_clientdata(client);
|
||||||
|
|
||||||
if (bq->notify_psy)
|
if (bq->notify_psy) {
|
||||||
power_supply_unreg_notifier(&bq->nb);
|
power_supply_unreg_notifier(&bq->nb);
|
||||||
|
power_supply_put(bq->notify_psy);
|
||||||
|
}
|
||||||
|
|
||||||
bq2415x_sysfs_exit(bq);
|
bq2415x_sysfs_exit(bq);
|
||||||
bq2415x_power_supply_exit(bq);
|
bq2415x_power_supply_exit(bq);
|
||||||
|
@ -1731,6 +1721,7 @@ static const struct i2c_device_id bq2415x_i2c_id_table[] = {
|
||||||
{ "bq24155", BQ24155 },
|
{ "bq24155", BQ24155 },
|
||||||
{ "bq24156", BQ24156 },
|
{ "bq24156", BQ24156 },
|
||||||
{ "bq24156a", BQ24156A },
|
{ "bq24156a", BQ24156A },
|
||||||
|
{ "bq24157s", BQ24157S },
|
||||||
{ "bq24158", BQ24158 },
|
{ "bq24158", BQ24158 },
|
||||||
{},
|
{},
|
||||||
};
|
};
|
||||||
|
|
|
@ -152,8 +152,8 @@
|
||||||
struct bq24190_dev_info {
|
struct bq24190_dev_info {
|
||||||
struct i2c_client *client;
|
struct i2c_client *client;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
char model_name[I2C_NAME_SIZE];
|
char model_name[I2C_NAME_SIZE];
|
||||||
kernel_ulong_t model;
|
kernel_ulong_t model;
|
||||||
unsigned int gpio_int;
|
unsigned int gpio_int;
|
||||||
|
@ -423,8 +423,7 @@ static ssize_t bq24190_sysfs_show(struct device *dev,
|
||||||
struct device_attribute *attr, char *buf)
|
struct device_attribute *attr, char *buf)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq24190_dev_info *bdi =
|
struct bq24190_dev_info *bdi = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct bq24190_dev_info, charger);
|
|
||||||
struct bq24190_sysfs_field_info *info;
|
struct bq24190_sysfs_field_info *info;
|
||||||
int ret;
|
int ret;
|
||||||
u8 v;
|
u8 v;
|
||||||
|
@ -444,8 +443,7 @@ static ssize_t bq24190_sysfs_store(struct device *dev,
|
||||||
struct device_attribute *attr, const char *buf, size_t count)
|
struct device_attribute *attr, const char *buf, size_t count)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
struct bq24190_dev_info *bdi =
|
struct bq24190_dev_info *bdi = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct bq24190_dev_info, charger);
|
|
||||||
struct bq24190_sysfs_field_info *info;
|
struct bq24190_sysfs_field_info *info;
|
||||||
int ret;
|
int ret;
|
||||||
u8 v;
|
u8 v;
|
||||||
|
@ -469,13 +467,13 @@ static int bq24190_sysfs_create_group(struct bq24190_dev_info *bdi)
|
||||||
{
|
{
|
||||||
bq24190_sysfs_init_attrs();
|
bq24190_sysfs_init_attrs();
|
||||||
|
|
||||||
return sysfs_create_group(&bdi->charger.dev->kobj,
|
return sysfs_create_group(&bdi->charger->dev.kobj,
|
||||||
&bq24190_sysfs_attr_group);
|
&bq24190_sysfs_attr_group);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bq24190_sysfs_remove_group(struct bq24190_dev_info *bdi)
|
static void bq24190_sysfs_remove_group(struct bq24190_dev_info *bdi)
|
||||||
{
|
{
|
||||||
sysfs_remove_group(&bdi->charger.dev->kobj, &bq24190_sysfs_attr_group);
|
sysfs_remove_group(&bdi->charger->dev.kobj, &bq24190_sysfs_attr_group);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static int bq24190_sysfs_create_group(struct bq24190_dev_info *bdi)
|
static int bq24190_sysfs_create_group(struct bq24190_dev_info *bdi)
|
||||||
|
@ -807,8 +805,7 @@ static int bq24190_charger_set_voltage(struct bq24190_dev_info *bdi,
|
||||||
static int bq24190_charger_get_property(struct power_supply *psy,
|
static int bq24190_charger_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property psp, union power_supply_propval *val)
|
enum power_supply_property psp, union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct bq24190_dev_info *bdi =
|
struct bq24190_dev_info *bdi = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct bq24190_dev_info, charger);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
||||||
|
@ -861,8 +858,7 @@ static int bq24190_charger_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)
|
||||||
{
|
{
|
||||||
struct bq24190_dev_info *bdi =
|
struct bq24190_dev_info *bdi = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct bq24190_dev_info, charger);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
||||||
|
@ -922,18 +918,15 @@ static char *bq24190_charger_supplied_to[] = {
|
||||||
"main-battery",
|
"main-battery",
|
||||||
};
|
};
|
||||||
|
|
||||||
static void bq24190_charger_init(struct power_supply *charger)
|
static const struct power_supply_desc bq24190_charger_desc = {
|
||||||
{
|
.name = "bq24190-charger",
|
||||||
charger->name = "bq24190-charger";
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
charger->type = POWER_SUPPLY_TYPE_USB;
|
.properties = bq24190_charger_properties,
|
||||||
charger->properties = bq24190_charger_properties;
|
.num_properties = ARRAY_SIZE(bq24190_charger_properties),
|
||||||
charger->num_properties = ARRAY_SIZE(bq24190_charger_properties);
|
.get_property = bq24190_charger_get_property,
|
||||||
charger->supplied_to = bq24190_charger_supplied_to;
|
.set_property = bq24190_charger_set_property,
|
||||||
charger->num_supplicants = ARRAY_SIZE(bq24190_charger_supplied_to);
|
.property_is_writeable = bq24190_charger_property_is_writeable,
|
||||||
charger->get_property = bq24190_charger_get_property;
|
};
|
||||||
charger->set_property = bq24190_charger_set_property;
|
|
||||||
charger->property_is_writeable = bq24190_charger_property_is_writeable;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Battery power supply property routines */
|
/* Battery power supply property routines */
|
||||||
|
|
||||||
|
@ -1102,8 +1095,7 @@ static int bq24190_battery_set_temp_alert_max(struct bq24190_dev_info *bdi,
|
||||||
static int bq24190_battery_get_property(struct power_supply *psy,
|
static int bq24190_battery_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property psp, union power_supply_propval *val)
|
enum power_supply_property psp, union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct bq24190_dev_info *bdi =
|
struct bq24190_dev_info *bdi = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct bq24190_dev_info, battery);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
||||||
|
@ -1144,8 +1136,7 @@ static int bq24190_battery_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)
|
||||||
{
|
{
|
||||||
struct bq24190_dev_info *bdi =
|
struct bq24190_dev_info *bdi = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct bq24190_dev_info, battery);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
dev_dbg(bdi->dev, "prop: %d\n", psp);
|
||||||
|
@ -1193,16 +1184,15 @@ static enum power_supply_property bq24190_battery_properties[] = {
|
||||||
POWER_SUPPLY_PROP_SCOPE,
|
POWER_SUPPLY_PROP_SCOPE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static void bq24190_battery_init(struct power_supply *battery)
|
static const struct power_supply_desc bq24190_battery_desc = {
|
||||||
{
|
.name = "bq24190-battery",
|
||||||
battery->name = "bq24190-battery";
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
battery->type = POWER_SUPPLY_TYPE_BATTERY;
|
.properties = bq24190_battery_properties,
|
||||||
battery->properties = bq24190_battery_properties;
|
.num_properties = ARRAY_SIZE(bq24190_battery_properties),
|
||||||
battery->num_properties = ARRAY_SIZE(bq24190_battery_properties);
|
.get_property = bq24190_battery_get_property,
|
||||||
battery->get_property = bq24190_battery_get_property;
|
.set_property = bq24190_battery_set_property,
|
||||||
battery->set_property = bq24190_battery_set_property;
|
.property_is_writeable = bq24190_battery_property_is_writeable,
|
||||||
battery->property_is_writeable = bq24190_battery_property_is_writeable;
|
};
|
||||||
}
|
|
||||||
|
|
||||||
static irqreturn_t bq24190_irq_handler_thread(int irq, void *data)
|
static irqreturn_t bq24190_irq_handler_thread(int irq, void *data)
|
||||||
{
|
{
|
||||||
|
@ -1269,8 +1259,8 @@ static irqreturn_t bq24190_irq_handler_thread(int irq, void *data)
|
||||||
* interrupt received).
|
* interrupt received).
|
||||||
*/
|
*/
|
||||||
if (alert_userspace && !bdi->first_time) {
|
if (alert_userspace && !bdi->first_time) {
|
||||||
power_supply_changed(&bdi->charger);
|
power_supply_changed(bdi->charger);
|
||||||
power_supply_changed(&bdi->battery);
|
power_supply_changed(bdi->battery);
|
||||||
bdi->first_time = false;
|
bdi->first_time = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1362,6 +1352,7 @@ static int bq24190_probe(struct i2c_client *client,
|
||||||
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
||||||
struct device *dev = &client->dev;
|
struct device *dev = &client->dev;
|
||||||
struct bq24190_platform_data *pdata = client->dev.platform_data;
|
struct bq24190_platform_data *pdata = client->dev.platform_data;
|
||||||
|
struct power_supply_config charger_cfg = {}, battery_cfg = {};
|
||||||
struct bq24190_dev_info *bdi;
|
struct bq24190_dev_info *bdi;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -1416,19 +1407,23 @@ static int bq24190_probe(struct i2c_client *client,
|
||||||
goto out2;
|
goto out2;
|
||||||
}
|
}
|
||||||
|
|
||||||
bq24190_charger_init(&bdi->charger);
|
charger_cfg.drv_data = bdi;
|
||||||
|
charger_cfg.supplied_to = bq24190_charger_supplied_to;
|
||||||
ret = power_supply_register(dev, &bdi->charger);
|
charger_cfg.num_supplicants = ARRAY_SIZE(bq24190_charger_supplied_to),
|
||||||
if (ret) {
|
bdi->charger = power_supply_register(dev, &bq24190_charger_desc,
|
||||||
|
&charger_cfg);
|
||||||
|
if (IS_ERR(bdi->charger)) {
|
||||||
dev_err(dev, "Can't register charger\n");
|
dev_err(dev, "Can't register charger\n");
|
||||||
|
ret = PTR_ERR(bdi->charger);
|
||||||
goto out2;
|
goto out2;
|
||||||
}
|
}
|
||||||
|
|
||||||
bq24190_battery_init(&bdi->battery);
|
battery_cfg.drv_data = bdi;
|
||||||
|
bdi->battery = power_supply_register(dev, &bq24190_battery_desc,
|
||||||
ret = power_supply_register(dev, &bdi->battery);
|
&battery_cfg);
|
||||||
if (ret) {
|
if (IS_ERR(bdi->battery)) {
|
||||||
dev_err(dev, "Can't register battery\n");
|
dev_err(dev, "Can't register battery\n");
|
||||||
|
ret = PTR_ERR(bdi->battery);
|
||||||
goto out3;
|
goto out3;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1441,9 +1436,9 @@ static int bq24190_probe(struct i2c_client *client,
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
out4:
|
out4:
|
||||||
power_supply_unregister(&bdi->battery);
|
power_supply_unregister(bdi->battery);
|
||||||
out3:
|
out3:
|
||||||
power_supply_unregister(&bdi->charger);
|
power_supply_unregister(bdi->charger);
|
||||||
out2:
|
out2:
|
||||||
pm_runtime_disable(dev);
|
pm_runtime_disable(dev);
|
||||||
out1:
|
out1:
|
||||||
|
@ -1462,8 +1457,8 @@ static int bq24190_remove(struct i2c_client *client)
|
||||||
pm_runtime_put_sync(bdi->dev);
|
pm_runtime_put_sync(bdi->dev);
|
||||||
|
|
||||||
bq24190_sysfs_remove_group(bdi);
|
bq24190_sysfs_remove_group(bdi);
|
||||||
power_supply_unregister(&bdi->battery);
|
power_supply_unregister(bdi->battery);
|
||||||
power_supply_unregister(&bdi->charger);
|
power_supply_unregister(bdi->charger);
|
||||||
pm_runtime_disable(bdi->dev);
|
pm_runtime_disable(bdi->dev);
|
||||||
|
|
||||||
if (bdi->gpio_int)
|
if (bdi->gpio_int)
|
||||||
|
@ -1499,8 +1494,8 @@ static int bq24190_pm_resume(struct device *dev)
|
||||||
pm_runtime_put_sync(bdi->dev);
|
pm_runtime_put_sync(bdi->dev);
|
||||||
|
|
||||||
/* Things may have changed while suspended so alert upper layer */
|
/* Things may have changed while suspended so alert upper layer */
|
||||||
power_supply_changed(&bdi->charger);
|
power_supply_changed(bdi->charger);
|
||||||
power_supply_changed(&bdi->battery);
|
power_supply_changed(bdi->battery);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,14 +44,15 @@
|
||||||
#define BQ24735_DEVICE_ID 0xff
|
#define BQ24735_DEVICE_ID 0xff
|
||||||
|
|
||||||
struct bq24735 {
|
struct bq24735 {
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
struct i2c_client *client;
|
struct power_supply_desc charger_desc;
|
||||||
struct bq24735_platform *pdata;
|
struct i2c_client *client;
|
||||||
|
struct bq24735_platform *pdata;
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline struct bq24735 *to_bq24735(struct power_supply *psy)
|
static inline struct bq24735 *to_bq24735(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
return container_of(psy, struct bq24735, charger);
|
return power_supply_get_drvdata(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum power_supply_property bq24735_charger_properties[] = {
|
static enum power_supply_property bq24735_charger_properties[] = {
|
||||||
|
@ -192,9 +193,7 @@ static int bq24735_charger_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 bq24735 *charger;
|
struct bq24735 *charger = to_bq24735(psy);
|
||||||
|
|
||||||
charger = container_of(psy, struct bq24735, charger);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_ONLINE:
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
|
@ -248,7 +247,8 @@ static int bq24735_charger_probe(struct i2c_client *client,
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
struct bq24735 *charger;
|
struct bq24735 *charger;
|
||||||
struct power_supply *supply;
|
struct power_supply_desc *supply_desc;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
char *name;
|
char *name;
|
||||||
|
|
||||||
charger = devm_kzalloc(&client->dev, sizeof(*charger), GFP_KERNEL);
|
charger = devm_kzalloc(&client->dev, sizeof(*charger), GFP_KERNEL);
|
||||||
|
@ -277,16 +277,18 @@ static int bq24735_charger_probe(struct i2c_client *client,
|
||||||
|
|
||||||
charger->client = client;
|
charger->client = client;
|
||||||
|
|
||||||
supply = &charger->charger;
|
supply_desc = &charger->charger_desc;
|
||||||
|
|
||||||
supply->name = name;
|
supply_desc->name = name;
|
||||||
supply->type = POWER_SUPPLY_TYPE_MAINS;
|
supply_desc->type = POWER_SUPPLY_TYPE_MAINS;
|
||||||
supply->properties = bq24735_charger_properties;
|
supply_desc->properties = bq24735_charger_properties;
|
||||||
supply->num_properties = ARRAY_SIZE(bq24735_charger_properties);
|
supply_desc->num_properties = ARRAY_SIZE(bq24735_charger_properties);
|
||||||
supply->get_property = bq24735_charger_get_property;
|
supply_desc->get_property = bq24735_charger_get_property;
|
||||||
supply->supplied_to = charger->pdata->supplied_to;
|
|
||||||
supply->num_supplicants = charger->pdata->num_supplicants;
|
psy_cfg.supplied_to = charger->pdata->supplied_to;
|
||||||
supply->of_node = client->dev.of_node;
|
psy_cfg.num_supplicants = charger->pdata->num_supplicants;
|
||||||
|
psy_cfg.of_node = client->dev.of_node;
|
||||||
|
psy_cfg.drv_data = charger;
|
||||||
|
|
||||||
i2c_set_clientdata(client, charger);
|
i2c_set_clientdata(client, charger);
|
||||||
|
|
||||||
|
@ -341,8 +343,10 @@ static int bq24735_charger_probe(struct i2c_client *client,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&client->dev, supply);
|
charger->charger = power_supply_register(&client->dev, supply_desc,
|
||||||
if (ret < 0) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(charger->charger)) {
|
||||||
|
ret = PTR_ERR(charger->charger);
|
||||||
dev_err(&client->dev, "Failed to register power supply: %d\n",
|
dev_err(&client->dev, "Failed to register power supply: %d\n",
|
||||||
ret);
|
ret);
|
||||||
goto err_free_name;
|
goto err_free_name;
|
||||||
|
@ -354,7 +358,8 @@ static int bq24735_charger_probe(struct i2c_client *client,
|
||||||
IRQF_TRIGGER_RISING |
|
IRQF_TRIGGER_RISING |
|
||||||
IRQF_TRIGGER_FALLING |
|
IRQF_TRIGGER_FALLING |
|
||||||
IRQF_ONESHOT,
|
IRQF_ONESHOT,
|
||||||
supply->name, supply);
|
supply_desc->name,
|
||||||
|
charger->charger);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(&client->dev,
|
dev_err(&client->dev,
|
||||||
"Unable to register IRQ %d err %d\n",
|
"Unable to register IRQ %d err %d\n",
|
||||||
|
@ -365,7 +370,7 @@ static int bq24735_charger_probe(struct i2c_client *client,
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
err_unregister_supply:
|
err_unregister_supply:
|
||||||
power_supply_unregister(supply);
|
power_supply_unregister(charger->charger);
|
||||||
err_free_name:
|
err_free_name:
|
||||||
if (name != charger->pdata->name)
|
if (name != charger->pdata->name)
|
||||||
kfree(name);
|
kfree(name);
|
||||||
|
@ -381,10 +386,10 @@ static int bq24735_charger_remove(struct i2c_client *client)
|
||||||
devm_free_irq(&charger->client->dev, charger->client->irq,
|
devm_free_irq(&charger->client->dev, charger->client->irq,
|
||||||
&charger->charger);
|
&charger->charger);
|
||||||
|
|
||||||
power_supply_unregister(&charger->charger);
|
power_supply_unregister(charger->charger);
|
||||||
|
|
||||||
if (charger->charger.name != charger->pdata->name)
|
if (charger->charger_desc.name != charger->pdata->name)
|
||||||
kfree(charger->charger.name);
|
kfree(charger->charger_desc.name);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,14 +16,12 @@
|
||||||
* IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
|
* IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
|
||||||
* WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
* WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
*
|
*
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Datasheets:
|
* Datasheets:
|
||||||
* http://focus.ti.com/docs/prod/folders/print/bq27000.html
|
* http://focus.ti.com/docs/prod/folders/print/bq27000.html
|
||||||
* http://focus.ti.com/docs/prod/folders/print/bq27500.html
|
* http://focus.ti.com/docs/prod/folders/print/bq27500.html
|
||||||
* http://www.ti.com/product/bq27425-g1
|
* http://www.ti.com/product/bq27425-g1
|
||||||
* http://www.ti.com/product/BQ27742-G1
|
* http://www.ti.com/product/BQ27742-G1
|
||||||
|
* http://www.ti.com/product/BQ27510-G3
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/device.h>
|
#include <linux/device.h>
|
||||||
|
@ -74,6 +72,10 @@
|
||||||
|
|
||||||
#define BQ27742_POWER_AVG 0x76
|
#define BQ27742_POWER_AVG 0x76
|
||||||
|
|
||||||
|
#define BQ27510_REG_SOC 0x20
|
||||||
|
#define BQ27510_REG_DCAP 0x2E /* Design capacity */
|
||||||
|
#define BQ27510_REG_CYCT 0x1E /* Cycle count total */
|
||||||
|
|
||||||
/* bq27425 register addresses are same as bq27x00 addresses minus 4 */
|
/* bq27425 register addresses are same as bq27x00 addresses minus 4 */
|
||||||
#define BQ27425_REG_OFFSET 0x04
|
#define BQ27425_REG_OFFSET 0x04
|
||||||
#define BQ27425_REG_SOC (0x1C + BQ27425_REG_OFFSET)
|
#define BQ27425_REG_SOC (0x1C + BQ27425_REG_OFFSET)
|
||||||
|
@ -87,7 +89,7 @@ struct bq27x00_access_methods {
|
||||||
int (*read)(struct bq27x00_device_info *di, u8 reg, bool single);
|
int (*read)(struct bq27x00_device_info *di, u8 reg, bool single);
|
||||||
};
|
};
|
||||||
|
|
||||||
enum bq27x00_chip { BQ27000, BQ27500, BQ27425, BQ27742};
|
enum bq27x00_chip { BQ27000, BQ27500, BQ27425, BQ27742, BQ27510};
|
||||||
|
|
||||||
struct bq27x00_reg_cache {
|
struct bq27x00_reg_cache {
|
||||||
int temperature;
|
int temperature;
|
||||||
|
@ -114,7 +116,7 @@ struct bq27x00_device_info {
|
||||||
unsigned long last_update;
|
unsigned long last_update;
|
||||||
struct delayed_work work;
|
struct delayed_work work;
|
||||||
|
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
|
||||||
struct bq27x00_access_methods bus;
|
struct bq27x00_access_methods bus;
|
||||||
|
|
||||||
|
@ -174,6 +176,24 @@ static enum power_supply_property bq27742_battery_props[] = {
|
||||||
POWER_SUPPLY_PROP_HEALTH,
|
POWER_SUPPLY_PROP_HEALTH,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static enum power_supply_property bq27510_battery_props[] = {
|
||||||
|
POWER_SUPPLY_PROP_STATUS,
|
||||||
|
POWER_SUPPLY_PROP_PRESENT,
|
||||||
|
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
||||||
|
POWER_SUPPLY_PROP_CURRENT_NOW,
|
||||||
|
POWER_SUPPLY_PROP_CAPACITY,
|
||||||
|
POWER_SUPPLY_PROP_CAPACITY_LEVEL,
|
||||||
|
POWER_SUPPLY_PROP_TEMP,
|
||||||
|
POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW,
|
||||||
|
POWER_SUPPLY_PROP_TECHNOLOGY,
|
||||||
|
POWER_SUPPLY_PROP_CHARGE_FULL,
|
||||||
|
POWER_SUPPLY_PROP_CHARGE_NOW,
|
||||||
|
POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN,
|
||||||
|
POWER_SUPPLY_PROP_CYCLE_COUNT,
|
||||||
|
POWER_SUPPLY_PROP_POWER_AVG,
|
||||||
|
POWER_SUPPLY_PROP_HEALTH,
|
||||||
|
};
|
||||||
|
|
||||||
static unsigned int poll_interval = 360;
|
static unsigned int poll_interval = 360;
|
||||||
module_param(poll_interval, uint, 0644);
|
module_param(poll_interval, uint, 0644);
|
||||||
MODULE_PARM_DESC(poll_interval, "battery poll interval in seconds - " \
|
MODULE_PARM_DESC(poll_interval, "battery poll interval in seconds - " \
|
||||||
|
@ -198,7 +218,8 @@ static inline int bq27x00_read(struct bq27x00_device_info *di, u8 reg,
|
||||||
*/
|
*/
|
||||||
static bool bq27xxx_is_chip_version_higher(struct bq27x00_device_info *di)
|
static bool bq27xxx_is_chip_version_higher(struct bq27x00_device_info *di)
|
||||||
{
|
{
|
||||||
if (di->chip == BQ27425 || di->chip == BQ27500 || di->chip == BQ27742)
|
if (di->chip == BQ27425 || di->chip == BQ27500 || di->chip == BQ27742
|
||||||
|
|| di->chip == BQ27510)
|
||||||
return true;
|
return true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -213,6 +234,8 @@ static int bq27x00_battery_read_rsoc(struct bq27x00_device_info *di)
|
||||||
|
|
||||||
if (di->chip == BQ27500 || di->chip == BQ27742)
|
if (di->chip == BQ27500 || di->chip == BQ27742)
|
||||||
rsoc = bq27x00_read(di, BQ27500_REG_SOC, false);
|
rsoc = bq27x00_read(di, BQ27500_REG_SOC, false);
|
||||||
|
else if (di->chip == BQ27510)
|
||||||
|
rsoc = bq27x00_read(di, BQ27510_REG_SOC, false);
|
||||||
else if (di->chip == BQ27425)
|
else if (di->chip == BQ27425)
|
||||||
rsoc = bq27x00_read(di, BQ27425_REG_SOC, false);
|
rsoc = bq27x00_read(di, BQ27425_REG_SOC, false);
|
||||||
else
|
else
|
||||||
|
@ -286,6 +309,8 @@ static int bq27x00_battery_read_ilmd(struct bq27x00_device_info *di)
|
||||||
if (bq27xxx_is_chip_version_higher(di)) {
|
if (bq27xxx_is_chip_version_higher(di)) {
|
||||||
if (di->chip == BQ27425)
|
if (di->chip == BQ27425)
|
||||||
ilmd = bq27x00_read(di, BQ27425_REG_DCAP, false);
|
ilmd = bq27x00_read(di, BQ27425_REG_DCAP, false);
|
||||||
|
else if (di->chip == BQ27510)
|
||||||
|
ilmd = bq27x00_read(di, BQ27510_REG_DCAP, false);
|
||||||
else
|
else
|
||||||
ilmd = bq27x00_read(di, BQ27500_REG_DCAP, false);
|
ilmd = bq27x00_read(di, BQ27500_REG_DCAP, false);
|
||||||
} else
|
} else
|
||||||
|
@ -354,7 +379,10 @@ static int bq27x00_battery_read_cyct(struct bq27x00_device_info *di)
|
||||||
{
|
{
|
||||||
int cyct;
|
int cyct;
|
||||||
|
|
||||||
cyct = bq27x00_read(di, BQ27x00_REG_CYCT, false);
|
if (di->chip == BQ27510)
|
||||||
|
cyct = bq27x00_read(di, BQ27510_REG_CYCT, false);
|
||||||
|
else
|
||||||
|
cyct = bq27x00_read(di, BQ27x00_REG_CYCT, false);
|
||||||
if (cyct < 0)
|
if (cyct < 0)
|
||||||
dev_err(di->dev, "error reading cycle count total\n");
|
dev_err(di->dev, "error reading cycle count total\n");
|
||||||
|
|
||||||
|
@ -425,6 +453,10 @@ static int bq27x00_battery_read_health(struct bq27x00_device_info *di)
|
||||||
else
|
else
|
||||||
tval = POWER_SUPPLY_HEALTH_GOOD;
|
tval = POWER_SUPPLY_HEALTH_GOOD;
|
||||||
return tval;
|
return tval;
|
||||||
|
} else if (di->chip == BQ27510) {
|
||||||
|
if (tval & BQ27500_FLAG_OTC)
|
||||||
|
return POWER_SUPPLY_HEALTH_OVERHEAT;
|
||||||
|
return POWER_SUPPLY_HEALTH_GOOD;
|
||||||
} else {
|
} else {
|
||||||
if (tval & BQ27000_FLAG_EDV1)
|
if (tval & BQ27000_FLAG_EDV1)
|
||||||
tval = POWER_SUPPLY_HEALTH_DEAD;
|
tval = POWER_SUPPLY_HEALTH_DEAD;
|
||||||
|
@ -440,6 +472,7 @@ static void bq27x00_update(struct bq27x00_device_info *di)
|
||||||
{
|
{
|
||||||
struct bq27x00_reg_cache cache = {0, };
|
struct bq27x00_reg_cache cache = {0, };
|
||||||
bool is_bq27500 = di->chip == BQ27500;
|
bool is_bq27500 = di->chip == BQ27500;
|
||||||
|
bool is_bq27510 = di->chip == BQ27510;
|
||||||
bool is_bq27425 = di->chip == BQ27425;
|
bool is_bq27425 = di->chip == BQ27425;
|
||||||
bool is_bq27742 = di->chip == BQ27742;
|
bool is_bq27742 = di->chip == BQ27742;
|
||||||
bool flags_1b = !(is_bq27500 || is_bq27742);
|
bool flags_1b = !(is_bq27500 || is_bq27742);
|
||||||
|
@ -449,7 +482,7 @@ static void bq27x00_update(struct bq27x00_device_info *di)
|
||||||
/* read error */
|
/* read error */
|
||||||
cache.flags = -1;
|
cache.flags = -1;
|
||||||
if (cache.flags >= 0) {
|
if (cache.flags >= 0) {
|
||||||
if (!is_bq27500 && !is_bq27425 && !is_bq27742
|
if (!is_bq27500 && !is_bq27425 && !is_bq27742 && !is_bq27510
|
||||||
&& (cache.flags & BQ27000_FLAG_CI)) {
|
&& (cache.flags & BQ27000_FLAG_CI)) {
|
||||||
dev_info(di->dev, "battery is not calibrated! ignoring capacity values\n");
|
dev_info(di->dev, "battery is not calibrated! ignoring capacity values\n");
|
||||||
cache.capacity = -ENODATA;
|
cache.capacity = -ENODATA;
|
||||||
|
@ -461,7 +494,7 @@ static void bq27x00_update(struct bq27x00_device_info *di)
|
||||||
cache.health = -ENODATA;
|
cache.health = -ENODATA;
|
||||||
} else {
|
} else {
|
||||||
cache.capacity = bq27x00_battery_read_rsoc(di);
|
cache.capacity = bq27x00_battery_read_rsoc(di);
|
||||||
if (is_bq27742)
|
if (is_bq27742 || is_bq27510)
|
||||||
cache.time_to_empty =
|
cache.time_to_empty =
|
||||||
bq27x00_battery_read_time(di,
|
bq27x00_battery_read_time(di,
|
||||||
BQ27x00_REG_TTE);
|
BQ27x00_REG_TTE);
|
||||||
|
@ -498,7 +531,7 @@ static void bq27x00_update(struct bq27x00_device_info *di)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (di->cache.capacity != cache.capacity)
|
if (di->cache.capacity != cache.capacity)
|
||||||
power_supply_changed(&di->bat);
|
power_supply_changed(di->bat);
|
||||||
|
|
||||||
if (memcmp(&di->cache, &cache, sizeof(cache)) != 0)
|
if (memcmp(&di->cache, &cache, sizeof(cache)) != 0)
|
||||||
di->cache = cache;
|
di->cache = cache;
|
||||||
|
@ -570,7 +603,7 @@ static int bq27x00_battery_status(struct bq27x00_device_info *di,
|
||||||
status = POWER_SUPPLY_STATUS_FULL;
|
status = POWER_SUPPLY_STATUS_FULL;
|
||||||
else if (di->cache.flags & BQ27000_FLAG_CHGS)
|
else if (di->cache.flags & BQ27000_FLAG_CHGS)
|
||||||
status = POWER_SUPPLY_STATUS_CHARGING;
|
status = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
else if (power_supply_am_i_supplied(&di->bat))
|
else if (power_supply_am_i_supplied(di->bat))
|
||||||
status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
status = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
||||||
else
|
else
|
||||||
status = POWER_SUPPLY_STATUS_DISCHARGING;
|
status = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
|
@ -642,15 +675,12 @@ static int bq27x00_simple_value(int value,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define to_bq27x00_device_info(x) container_of((x), \
|
|
||||||
struct bq27x00_device_info, bat);
|
|
||||||
|
|
||||||
static int bq27x00_battery_get_property(struct power_supply *psy,
|
static int bq27x00_battery_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)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct bq27x00_device_info *di = to_bq27x00_device_info(psy);
|
struct bq27x00_device_info *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
mutex_lock(&di->lock);
|
mutex_lock(&di->lock);
|
||||||
if (time_is_before_jiffies(di->last_update + 5 * HZ)) {
|
if (time_is_before_jiffies(di->last_update + 5 * HZ)) {
|
||||||
|
@ -728,35 +758,47 @@ static int bq27x00_battery_get_property(struct power_supply *psy,
|
||||||
|
|
||||||
static void bq27x00_external_power_changed(struct power_supply *psy)
|
static void bq27x00_external_power_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct bq27x00_device_info *di = to_bq27x00_device_info(psy);
|
struct bq27x00_device_info *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
cancel_delayed_work_sync(&di->work);
|
cancel_delayed_work_sync(&di->work);
|
||||||
schedule_delayed_work(&di->work, 0);
|
schedule_delayed_work(&di->work, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int bq27x00_powersupply_init(struct bq27x00_device_info *di)
|
static int bq27x00_powersupply_init(struct bq27x00_device_info *di,
|
||||||
|
const char *name)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
struct power_supply_desc *psy_desc;
|
||||||
|
struct power_supply_config psy_cfg = { .drv_data = di, };
|
||||||
|
|
||||||
di->bat.type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_desc = devm_kzalloc(di->dev, sizeof(*psy_desc), GFP_KERNEL);
|
||||||
|
if (!psy_desc)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
psy_desc->name = name;
|
||||||
|
psy_desc->type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
if (di->chip == BQ27425) {
|
if (di->chip == BQ27425) {
|
||||||
di->bat.properties = bq27425_battery_props;
|
psy_desc->properties = bq27425_battery_props;
|
||||||
di->bat.num_properties = ARRAY_SIZE(bq27425_battery_props);
|
psy_desc->num_properties = ARRAY_SIZE(bq27425_battery_props);
|
||||||
} else if (di->chip == BQ27742) {
|
} else if (di->chip == BQ27742) {
|
||||||
di->bat.properties = bq27742_battery_props;
|
psy_desc->properties = bq27742_battery_props;
|
||||||
di->bat.num_properties = ARRAY_SIZE(bq27742_battery_props);
|
psy_desc->num_properties = ARRAY_SIZE(bq27742_battery_props);
|
||||||
|
} else if (di->chip == BQ27510) {
|
||||||
|
psy_desc->properties = bq27510_battery_props;
|
||||||
|
psy_desc->num_properties = ARRAY_SIZE(bq27510_battery_props);
|
||||||
} else {
|
} else {
|
||||||
di->bat.properties = bq27x00_battery_props;
|
psy_desc->properties = bq27x00_battery_props;
|
||||||
di->bat.num_properties = ARRAY_SIZE(bq27x00_battery_props);
|
psy_desc->num_properties = ARRAY_SIZE(bq27x00_battery_props);
|
||||||
}
|
}
|
||||||
di->bat.get_property = bq27x00_battery_get_property;
|
psy_desc->get_property = bq27x00_battery_get_property;
|
||||||
di->bat.external_power_changed = bq27x00_external_power_changed;
|
psy_desc->external_power_changed = bq27x00_external_power_changed;
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&di->work, bq27x00_battery_poll);
|
INIT_DELAYED_WORK(&di->work, bq27x00_battery_poll);
|
||||||
mutex_init(&di->lock);
|
mutex_init(&di->lock);
|
||||||
|
|
||||||
ret = power_supply_register(di->dev, &di->bat);
|
di->bat = power_supply_register_no_ws(di->dev, psy_desc, &psy_cfg);
|
||||||
if (ret) {
|
if (IS_ERR(di->bat)) {
|
||||||
|
ret = PTR_ERR(di->bat);
|
||||||
dev_err(di->dev, "failed to register battery: %d\n", ret);
|
dev_err(di->dev, "failed to register battery: %d\n", ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -780,7 +822,7 @@ static void bq27x00_powersupply_unregister(struct bq27x00_device_info *di)
|
||||||
|
|
||||||
cancel_delayed_work_sync(&di->work);
|
cancel_delayed_work_sync(&di->work);
|
||||||
|
|
||||||
power_supply_unregister(&di->bat);
|
power_supply_unregister(di->bat);
|
||||||
|
|
||||||
mutex_destroy(&di->lock);
|
mutex_destroy(&di->lock);
|
||||||
}
|
}
|
||||||
|
@ -844,37 +886,34 @@ static int bq27x00_battery_probe(struct i2c_client *client,
|
||||||
if (num < 0)
|
if (num < 0)
|
||||||
return num;
|
return num;
|
||||||
|
|
||||||
name = kasprintf(GFP_KERNEL, "%s-%d", id->name, num);
|
name = devm_kasprintf(&client->dev, GFP_KERNEL, "%s-%d", id->name, num);
|
||||||
if (!name) {
|
if (!name) {
|
||||||
dev_err(&client->dev, "failed to allocate device name\n");
|
dev_err(&client->dev, "failed to allocate device name\n");
|
||||||
retval = -ENOMEM;
|
retval = -ENOMEM;
|
||||||
goto batt_failed_1;
|
goto batt_failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
di = devm_kzalloc(&client->dev, sizeof(*di), GFP_KERNEL);
|
di = devm_kzalloc(&client->dev, sizeof(*di), GFP_KERNEL);
|
||||||
if (!di) {
|
if (!di) {
|
||||||
dev_err(&client->dev, "failed to allocate device info data\n");
|
dev_err(&client->dev, "failed to allocate device info data\n");
|
||||||
retval = -ENOMEM;
|
retval = -ENOMEM;
|
||||||
goto batt_failed_2;
|
goto batt_failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
di->id = num;
|
di->id = num;
|
||||||
di->dev = &client->dev;
|
di->dev = &client->dev;
|
||||||
di->chip = id->driver_data;
|
di->chip = id->driver_data;
|
||||||
di->bat.name = name;
|
|
||||||
di->bus.read = &bq27x00_read_i2c;
|
di->bus.read = &bq27x00_read_i2c;
|
||||||
|
|
||||||
retval = bq27x00_powersupply_init(di);
|
retval = bq27x00_powersupply_init(di, name);
|
||||||
if (retval)
|
if (retval)
|
||||||
goto batt_failed_2;
|
goto batt_failed;
|
||||||
|
|
||||||
i2c_set_clientdata(client, di);
|
i2c_set_clientdata(client, di);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
batt_failed_2:
|
batt_failed:
|
||||||
kfree(name);
|
|
||||||
batt_failed_1:
|
|
||||||
mutex_lock(&battery_mutex);
|
mutex_lock(&battery_mutex);
|
||||||
idr_remove(&battery_id, num);
|
idr_remove(&battery_id, num);
|
||||||
mutex_unlock(&battery_mutex);
|
mutex_unlock(&battery_mutex);
|
||||||
|
@ -888,8 +927,6 @@ static int bq27x00_battery_remove(struct i2c_client *client)
|
||||||
|
|
||||||
bq27x00_powersupply_unregister(di);
|
bq27x00_powersupply_unregister(di);
|
||||||
|
|
||||||
kfree(di->bat.name);
|
|
||||||
|
|
||||||
mutex_lock(&battery_mutex);
|
mutex_lock(&battery_mutex);
|
||||||
idr_remove(&battery_id, di->id);
|
idr_remove(&battery_id, di->id);
|
||||||
mutex_unlock(&battery_mutex);
|
mutex_unlock(&battery_mutex);
|
||||||
|
@ -902,6 +939,7 @@ static const struct i2c_device_id bq27x00_id[] = {
|
||||||
{ "bq27500", BQ27500 },
|
{ "bq27500", BQ27500 },
|
||||||
{ "bq27425", BQ27425 },
|
{ "bq27425", BQ27425 },
|
||||||
{ "bq27742", BQ27742 },
|
{ "bq27742", BQ27742 },
|
||||||
|
{ "bq27510", BQ27510 },
|
||||||
{},
|
{},
|
||||||
};
|
};
|
||||||
MODULE_DEVICE_TABLE(i2c, bq27x00_id);
|
MODULE_DEVICE_TABLE(i2c, bq27x00_id);
|
||||||
|
@ -977,6 +1015,7 @@ static int bq27000_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct bq27x00_device_info *di;
|
struct bq27x00_device_info *di;
|
||||||
struct bq27000_platform_data *pdata = pdev->dev.platform_data;
|
struct bq27000_platform_data *pdata = pdev->dev.platform_data;
|
||||||
|
const char *name;
|
||||||
|
|
||||||
if (!pdata) {
|
if (!pdata) {
|
||||||
dev_err(&pdev->dev, "no platform_data supplied\n");
|
dev_err(&pdev->dev, "no platform_data supplied\n");
|
||||||
|
@ -999,10 +1038,10 @@ static int bq27000_battery_probe(struct platform_device *pdev)
|
||||||
di->dev = &pdev->dev;
|
di->dev = &pdev->dev;
|
||||||
di->chip = BQ27000;
|
di->chip = BQ27000;
|
||||||
|
|
||||||
di->bat.name = pdata->name ?: dev_name(&pdev->dev);
|
name = pdata->name ?: dev_name(&pdev->dev);
|
||||||
di->bus.read = &bq27000_read_platform;
|
di->bus.read = &bq27000_read_platform;
|
||||||
|
|
||||||
return bq27x00_powersupply_init(di);
|
return bq27x00_powersupply_init(di, name);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int bq27000_battery_remove(struct platform_device *pdev)
|
static int bq27000_battery_remove(struct platform_device *pdev)
|
||||||
|
|
|
@ -103,10 +103,11 @@ static bool is_batt_present(struct charger_manager *cm)
|
||||||
if (!psy)
|
if (!psy)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
ret = psy->get_property(psy,
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_PRESENT,
|
||||||
POWER_SUPPLY_PROP_PRESENT, &val);
|
&val);
|
||||||
if (ret == 0 && val.intval)
|
if (ret == 0 && val.intval)
|
||||||
present = true;
|
present = true;
|
||||||
|
power_supply_put(psy);
|
||||||
break;
|
break;
|
||||||
case CM_CHARGER_STAT:
|
case CM_CHARGER_STAT:
|
||||||
for (i = 0; cm->desc->psy_charger_stat[i]; i++) {
|
for (i = 0; cm->desc->psy_charger_stat[i]; i++) {
|
||||||
|
@ -118,8 +119,9 @@ static bool is_batt_present(struct charger_manager *cm)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_PRESENT,
|
ret = power_supply_get_property(psy,
|
||||||
&val);
|
POWER_SUPPLY_PROP_PRESENT, &val);
|
||||||
|
power_supply_put(psy);
|
||||||
if (ret == 0 && val.intval) {
|
if (ret == 0 && val.intval) {
|
||||||
present = true;
|
present = true;
|
||||||
break;
|
break;
|
||||||
|
@ -155,7 +157,9 @@ static bool is_ext_pwr_online(struct charger_manager *cm)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_ONLINE, &val);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_ONLINE,
|
||||||
|
&val);
|
||||||
|
power_supply_put(psy);
|
||||||
if (ret == 0 && val.intval) {
|
if (ret == 0 && val.intval) {
|
||||||
online = true;
|
online = true;
|
||||||
break;
|
break;
|
||||||
|
@ -183,8 +187,9 @@ static int get_batt_uV(struct charger_manager *cm, int *uV)
|
||||||
if (!fuel_gauge)
|
if (!fuel_gauge)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
ret = fuel_gauge->get_property(fuel_gauge,
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_VOLTAGE_NOW, &val);
|
POWER_SUPPLY_PROP_VOLTAGE_NOW, &val);
|
||||||
|
power_supply_put(fuel_gauge);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
@ -223,20 +228,26 @@ static bool is_charging(struct charger_manager *cm)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 2. The charger should be online (ext-power) */
|
/* 2. The charger should be online (ext-power) */
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_ONLINE, &val);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_ONLINE,
|
||||||
|
&val);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_warn(cm->dev, "Cannot read ONLINE value from %s\n",
|
dev_warn(cm->dev, "Cannot read ONLINE value from %s\n",
|
||||||
cm->desc->psy_charger_stat[i]);
|
cm->desc->psy_charger_stat[i]);
|
||||||
|
power_supply_put(psy);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (val.intval == 0)
|
if (val.intval == 0) {
|
||||||
|
power_supply_put(psy);
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* 3. The charger should not be FULL, DISCHARGING,
|
* 3. The charger should not be FULL, DISCHARGING,
|
||||||
* or NOT_CHARGING.
|
* or NOT_CHARGING.
|
||||||
*/
|
*/
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_STATUS, &val);
|
ret = power_supply_get_property(psy, POWER_SUPPLY_PROP_STATUS,
|
||||||
|
&val);
|
||||||
|
power_supply_put(psy);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_warn(cm->dev, "Cannot read STATUS value from %s\n",
|
dev_warn(cm->dev, "Cannot read STATUS value from %s\n",
|
||||||
cm->desc->psy_charger_stat[i]);
|
cm->desc->psy_charger_stat[i]);
|
||||||
|
@ -264,6 +275,7 @@ static bool is_full_charged(struct charger_manager *cm)
|
||||||
struct charger_desc *desc = cm->desc;
|
struct charger_desc *desc = cm->desc;
|
||||||
union power_supply_propval val;
|
union power_supply_propval val;
|
||||||
struct power_supply *fuel_gauge;
|
struct power_supply *fuel_gauge;
|
||||||
|
bool is_full = false;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
int uV;
|
int uV;
|
||||||
|
|
||||||
|
@ -279,30 +291,38 @@ static bool is_full_charged(struct charger_manager *cm)
|
||||||
val.intval = 0;
|
val.intval = 0;
|
||||||
|
|
||||||
/* Not full if capacity of fuel gauge isn't full */
|
/* Not full if capacity of fuel gauge isn't full */
|
||||||
ret = fuel_gauge->get_property(fuel_gauge,
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_CHARGE_FULL, &val);
|
POWER_SUPPLY_PROP_CHARGE_FULL, &val);
|
||||||
if (!ret && val.intval > desc->fullbatt_full_capacity)
|
if (!ret && val.intval > desc->fullbatt_full_capacity) {
|
||||||
return true;
|
is_full = true;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Full, if it's over the fullbatt voltage */
|
/* Full, if it's over the fullbatt voltage */
|
||||||
if (desc->fullbatt_uV > 0) {
|
if (desc->fullbatt_uV > 0) {
|
||||||
ret = get_batt_uV(cm, &uV);
|
ret = get_batt_uV(cm, &uV);
|
||||||
if (!ret && uV >= desc->fullbatt_uV)
|
if (!ret && uV >= desc->fullbatt_uV) {
|
||||||
return true;
|
is_full = true;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Full, if the capacity is more than fullbatt_soc */
|
/* Full, if the capacity is more than fullbatt_soc */
|
||||||
if (desc->fullbatt_soc > 0) {
|
if (desc->fullbatt_soc > 0) {
|
||||||
val.intval = 0;
|
val.intval = 0;
|
||||||
|
|
||||||
ret = fuel_gauge->get_property(fuel_gauge,
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_CAPACITY, &val);
|
POWER_SUPPLY_PROP_CAPACITY, &val);
|
||||||
if (!ret && val.intval >= desc->fullbatt_soc)
|
if (!ret && val.intval >= desc->fullbatt_soc) {
|
||||||
return true;
|
is_full = true;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
out:
|
||||||
|
power_supply_put(fuel_gauge);
|
||||||
|
return is_full;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -575,14 +595,18 @@ static int cm_get_battery_temperature_by_psy(struct charger_manager *cm,
|
||||||
int *temp)
|
int *temp)
|
||||||
{
|
{
|
||||||
struct power_supply *fuel_gauge;
|
struct power_supply *fuel_gauge;
|
||||||
|
int ret;
|
||||||
|
|
||||||
fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
|
fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
|
||||||
if (!fuel_gauge)
|
if (!fuel_gauge)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
return fuel_gauge->get_property(fuel_gauge,
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_TEMP,
|
POWER_SUPPLY_PROP_TEMP,
|
||||||
(union power_supply_propval *)temp);
|
(union power_supply_propval *)temp);
|
||||||
|
power_supply_put(fuel_gauge);
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int cm_get_battery_temperature(struct charger_manager *cm,
|
static int cm_get_battery_temperature(struct charger_manager *cm,
|
||||||
|
@ -861,10 +885,9 @@ static int charger_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 charger_manager *cm = container_of(psy,
|
struct charger_manager *cm = power_supply_get_drvdata(psy);
|
||||||
struct charger_manager, charger_psy);
|
|
||||||
struct charger_desc *desc = cm->desc;
|
struct charger_desc *desc = cm->desc;
|
||||||
struct power_supply *fuel_gauge;
|
struct power_supply *fuel_gauge = NULL;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
int uV;
|
int uV;
|
||||||
|
|
||||||
|
@ -900,26 +923,26 @@ static int charger_get_property(struct power_supply *psy,
|
||||||
ret = -ENODEV;
|
ret = -ENODEV;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ret = fuel_gauge->get_property(fuel_gauge,
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_CURRENT_NOW, val);
|
POWER_SUPPLY_PROP_CURRENT_NOW, val);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_TEMP:
|
case POWER_SUPPLY_PROP_TEMP:
|
||||||
case POWER_SUPPLY_PROP_TEMP_AMBIENT:
|
case POWER_SUPPLY_PROP_TEMP_AMBIENT:
|
||||||
return cm_get_battery_temperature(cm, &val->intval);
|
return cm_get_battery_temperature(cm, &val->intval);
|
||||||
case POWER_SUPPLY_PROP_CAPACITY:
|
case POWER_SUPPLY_PROP_CAPACITY:
|
||||||
fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
|
|
||||||
if (!fuel_gauge) {
|
|
||||||
ret = -ENODEV;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!is_batt_present(cm)) {
|
if (!is_batt_present(cm)) {
|
||||||
/* There is no battery. Assume 100% */
|
/* There is no battery. Assume 100% */
|
||||||
val->intval = 100;
|
val->intval = 100;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = fuel_gauge->get_property(fuel_gauge,
|
fuel_gauge = power_supply_get_by_name(cm->desc->psy_fuel_gauge);
|
||||||
|
if (!fuel_gauge) {
|
||||||
|
ret = -ENODEV;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_CAPACITY, val);
|
POWER_SUPPLY_PROP_CAPACITY, val);
|
||||||
if (ret)
|
if (ret)
|
||||||
break;
|
break;
|
||||||
|
@ -975,7 +998,7 @@ static int charger_get_property(struct power_supply *psy,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = fuel_gauge->get_property(fuel_gauge,
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_CHARGE_NOW,
|
POWER_SUPPLY_PROP_CHARGE_NOW,
|
||||||
val);
|
val);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -993,6 +1016,8 @@ static int charger_get_property(struct power_supply *psy,
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
if (fuel_gauge)
|
||||||
|
power_supply_put(fuel_gauge);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1015,7 +1040,7 @@ static enum power_supply_property default_charger_props[] = {
|
||||||
*/
|
*/
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply psy_default = {
|
static const struct power_supply_desc psy_default = {
|
||||||
.name = "battery",
|
.name = "battery",
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
.properties = default_charger_props,
|
.properties = default_charger_props,
|
||||||
|
@ -1396,7 +1421,7 @@ static int charger_manager_register_sysfs(struct charger_manager *cm)
|
||||||
dev_info(cm->dev, "'%s' regulator's externally_control is %d\n",
|
dev_info(cm->dev, "'%s' regulator's externally_control is %d\n",
|
||||||
charger->regulator_name, charger->externally_control);
|
charger->regulator_name, charger->externally_control);
|
||||||
|
|
||||||
ret = sysfs_create_group(&cm->charger_psy.dev->kobj,
|
ret = sysfs_create_group(&cm->charger_psy->dev.kobj,
|
||||||
&charger->attr_g);
|
&charger->attr_g);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
dev_err(cm->dev, "Cannot create sysfs entry of %s regulator\n",
|
dev_err(cm->dev, "Cannot create sysfs entry of %s regulator\n",
|
||||||
|
@ -1424,13 +1449,13 @@ static int cm_init_thermal_data(struct charger_manager *cm,
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
/* Verify whether fuel gauge provides battery temperature */
|
/* Verify whether fuel gauge provides battery temperature */
|
||||||
ret = fuel_gauge->get_property(fuel_gauge,
|
ret = power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_TEMP, &val);
|
POWER_SUPPLY_PROP_TEMP, &val);
|
||||||
|
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
cm->charger_psy.properties[cm->charger_psy.num_properties] =
|
cm->charger_psy_desc.properties[cm->charger_psy_desc.num_properties] =
|
||||||
POWER_SUPPLY_PROP_TEMP;
|
POWER_SUPPLY_PROP_TEMP;
|
||||||
cm->charger_psy.num_properties++;
|
cm->charger_psy_desc.num_properties++;
|
||||||
cm->desc->measure_battery_temp = true;
|
cm->desc->measure_battery_temp = true;
|
||||||
}
|
}
|
||||||
#ifdef CONFIG_THERMAL
|
#ifdef CONFIG_THERMAL
|
||||||
|
@ -1441,9 +1466,9 @@ static int cm_init_thermal_data(struct charger_manager *cm,
|
||||||
return PTR_ERR(cm->tzd_batt);
|
return PTR_ERR(cm->tzd_batt);
|
||||||
|
|
||||||
/* Use external thermometer */
|
/* Use external thermometer */
|
||||||
cm->charger_psy.properties[cm->charger_psy.num_properties] =
|
cm->charger_psy_desc.properties[cm->charger_psy_desc.num_properties] =
|
||||||
POWER_SUPPLY_PROP_TEMP_AMBIENT;
|
POWER_SUPPLY_PROP_TEMP_AMBIENT;
|
||||||
cm->charger_psy.num_properties++;
|
cm->charger_psy_desc.num_properties++;
|
||||||
cm->desc->measure_battery_temp = true;
|
cm->desc->measure_battery_temp = true;
|
||||||
ret = 0;
|
ret = 0;
|
||||||
}
|
}
|
||||||
|
@ -1459,7 +1484,7 @@ static int cm_init_thermal_data(struct charger_manager *cm,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct of_device_id charger_manager_match[] = {
|
static const struct of_device_id charger_manager_match[] = {
|
||||||
{
|
{
|
||||||
.compatible = "charger-manager",
|
.compatible = "charger-manager",
|
||||||
},
|
},
|
||||||
|
@ -1603,6 +1628,7 @@ static int charger_manager_probe(struct platform_device *pdev)
|
||||||
int j = 0;
|
int j = 0;
|
||||||
union power_supply_propval val;
|
union power_supply_propval val;
|
||||||
struct power_supply *fuel_gauge;
|
struct power_supply *fuel_gauge;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
if (IS_ERR(desc)) {
|
if (IS_ERR(desc)) {
|
||||||
dev_err(&pdev->dev, "No platform data (desc) found\n");
|
dev_err(&pdev->dev, "No platform data (desc) found\n");
|
||||||
|
@ -1617,6 +1643,7 @@ static int charger_manager_probe(struct platform_device *pdev)
|
||||||
/* Basic Values. Unspecified are Null or 0 */
|
/* Basic Values. Unspecified are Null or 0 */
|
||||||
cm->dev = &pdev->dev;
|
cm->dev = &pdev->dev;
|
||||||
cm->desc = desc;
|
cm->desc = desc;
|
||||||
|
psy_cfg.drv_data = cm;
|
||||||
|
|
||||||
/* Initialize alarm timer */
|
/* Initialize alarm timer */
|
||||||
if (alarmtimer_get_rtcdev()) {
|
if (alarmtimer_get_rtcdev()) {
|
||||||
|
@ -1672,13 +1699,7 @@ static int charger_manager_probe(struct platform_device *pdev)
|
||||||
desc->psy_charger_stat[i]);
|
desc->psy_charger_stat[i]);
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
}
|
power_supply_put(psy);
|
||||||
|
|
||||||
fuel_gauge = power_supply_get_by_name(desc->psy_fuel_gauge);
|
|
||||||
if (!fuel_gauge) {
|
|
||||||
dev_err(&pdev->dev, "Cannot find power supply \"%s\"\n",
|
|
||||||
desc->psy_fuel_gauge);
|
|
||||||
return -ENODEV;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (desc->polling_interval_ms == 0 ||
|
if (desc->polling_interval_ms == 0 ||
|
||||||
|
@ -1696,40 +1717,46 @@ static int charger_manager_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
platform_set_drvdata(pdev, cm);
|
platform_set_drvdata(pdev, cm);
|
||||||
|
|
||||||
memcpy(&cm->charger_psy, &psy_default, sizeof(psy_default));
|
memcpy(&cm->charger_psy_desc, &psy_default, sizeof(psy_default));
|
||||||
|
|
||||||
if (!desc->psy_name)
|
if (!desc->psy_name)
|
||||||
strncpy(cm->psy_name_buf, psy_default.name, PSY_NAME_MAX);
|
strncpy(cm->psy_name_buf, psy_default.name, PSY_NAME_MAX);
|
||||||
else
|
else
|
||||||
strncpy(cm->psy_name_buf, desc->psy_name, PSY_NAME_MAX);
|
strncpy(cm->psy_name_buf, desc->psy_name, PSY_NAME_MAX);
|
||||||
cm->charger_psy.name = cm->psy_name_buf;
|
cm->charger_psy_desc.name = cm->psy_name_buf;
|
||||||
|
|
||||||
/* Allocate for psy properties because they may vary */
|
/* Allocate for psy properties because they may vary */
|
||||||
cm->charger_psy.properties = devm_kzalloc(&pdev->dev,
|
cm->charger_psy_desc.properties = devm_kzalloc(&pdev->dev,
|
||||||
sizeof(enum power_supply_property)
|
sizeof(enum power_supply_property)
|
||||||
* (ARRAY_SIZE(default_charger_props) +
|
* (ARRAY_SIZE(default_charger_props) +
|
||||||
NUM_CHARGER_PSY_OPTIONAL), GFP_KERNEL);
|
NUM_CHARGER_PSY_OPTIONAL), GFP_KERNEL);
|
||||||
if (!cm->charger_psy.properties)
|
if (!cm->charger_psy_desc.properties)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
memcpy(cm->charger_psy.properties, default_charger_props,
|
memcpy(cm->charger_psy_desc.properties, default_charger_props,
|
||||||
sizeof(enum power_supply_property) *
|
sizeof(enum power_supply_property) *
|
||||||
ARRAY_SIZE(default_charger_props));
|
ARRAY_SIZE(default_charger_props));
|
||||||
cm->charger_psy.num_properties = psy_default.num_properties;
|
cm->charger_psy_desc.num_properties = psy_default.num_properties;
|
||||||
|
|
||||||
/* Find which optional psy-properties are available */
|
/* Find which optional psy-properties are available */
|
||||||
if (!fuel_gauge->get_property(fuel_gauge,
|
fuel_gauge = power_supply_get_by_name(desc->psy_fuel_gauge);
|
||||||
POWER_SUPPLY_PROP_CHARGE_NOW, &val)) {
|
if (!fuel_gauge) {
|
||||||
cm->charger_psy.properties[cm->charger_psy.num_properties] =
|
dev_err(&pdev->dev, "Cannot find power supply \"%s\"\n",
|
||||||
POWER_SUPPLY_PROP_CHARGE_NOW;
|
desc->psy_fuel_gauge);
|
||||||
cm->charger_psy.num_properties++;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
if (!fuel_gauge->get_property(fuel_gauge,
|
if (!power_supply_get_property(fuel_gauge,
|
||||||
|
POWER_SUPPLY_PROP_CHARGE_NOW, &val)) {
|
||||||
|
cm->charger_psy_desc.properties[cm->charger_psy_desc.num_properties] =
|
||||||
|
POWER_SUPPLY_PROP_CHARGE_NOW;
|
||||||
|
cm->charger_psy_desc.num_properties++;
|
||||||
|
}
|
||||||
|
if (!power_supply_get_property(fuel_gauge,
|
||||||
POWER_SUPPLY_PROP_CURRENT_NOW,
|
POWER_SUPPLY_PROP_CURRENT_NOW,
|
||||||
&val)) {
|
&val)) {
|
||||||
cm->charger_psy.properties[cm->charger_psy.num_properties] =
|
cm->charger_psy_desc.properties[cm->charger_psy_desc.num_properties] =
|
||||||
POWER_SUPPLY_PROP_CURRENT_NOW;
|
POWER_SUPPLY_PROP_CURRENT_NOW;
|
||||||
cm->charger_psy.num_properties++;
|
cm->charger_psy_desc.num_properties++;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = cm_init_thermal_data(cm, fuel_gauge);
|
ret = cm_init_thermal_data(cm, fuel_gauge);
|
||||||
|
@ -1737,14 +1764,16 @@ static int charger_manager_probe(struct platform_device *pdev)
|
||||||
dev_err(&pdev->dev, "Failed to initialize thermal data\n");
|
dev_err(&pdev->dev, "Failed to initialize thermal data\n");
|
||||||
cm->desc->measure_battery_temp = false;
|
cm->desc->measure_battery_temp = false;
|
||||||
}
|
}
|
||||||
|
power_supply_put(fuel_gauge);
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&cm->fullbatt_vchk_work, fullbatt_vchk);
|
INIT_DELAYED_WORK(&cm->fullbatt_vchk_work, fullbatt_vchk);
|
||||||
|
|
||||||
ret = power_supply_register(NULL, &cm->charger_psy);
|
cm->charger_psy = power_supply_register(NULL, &cm->charger_psy_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(cm->charger_psy)) {
|
||||||
dev_err(&pdev->dev, "Cannot register charger-manager with name \"%s\"\n",
|
dev_err(&pdev->dev, "Cannot register charger-manager with name \"%s\"\n",
|
||||||
cm->charger_psy.name);
|
cm->charger_psy_desc.name);
|
||||||
return ret;
|
return PTR_ERR(cm->charger_psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Register extcon device for charger cable */
|
/* Register extcon device for charger cable */
|
||||||
|
@ -1790,7 +1819,7 @@ err_reg_sysfs:
|
||||||
struct charger_regulator *charger;
|
struct charger_regulator *charger;
|
||||||
|
|
||||||
charger = &desc->charger_regulators[i];
|
charger = &desc->charger_regulators[i];
|
||||||
sysfs_remove_group(&cm->charger_psy.dev->kobj,
|
sysfs_remove_group(&cm->charger_psy->dev.kobj,
|
||||||
&charger->attr_g);
|
&charger->attr_g);
|
||||||
}
|
}
|
||||||
err_reg_extcon:
|
err_reg_extcon:
|
||||||
|
@ -1808,7 +1837,7 @@ err_reg_extcon:
|
||||||
regulator_put(desc->charger_regulators[i].consumer);
|
regulator_put(desc->charger_regulators[i].consumer);
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_unregister(&cm->charger_psy);
|
power_supply_unregister(cm->charger_psy);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -1840,7 +1869,7 @@ static int charger_manager_remove(struct platform_device *pdev)
|
||||||
for (i = 0 ; i < desc->num_charger_regulators ; i++)
|
for (i = 0 ; i < desc->num_charger_regulators ; i++)
|
||||||
regulator_put(desc->charger_regulators[i].consumer);
|
regulator_put(desc->charger_regulators[i].consumer);
|
||||||
|
|
||||||
power_supply_unregister(&cm->charger_psy);
|
power_supply_unregister(cm->charger_psy);
|
||||||
|
|
||||||
try_charger_enable(cm, false);
|
try_charger_enable(cm, false);
|
||||||
|
|
||||||
|
@ -1999,7 +2028,7 @@ static bool find_power_supply(struct charger_manager *cm,
|
||||||
bool found = false;
|
bool found = false;
|
||||||
|
|
||||||
for (i = 0; cm->desc->psy_charger_stat[i]; i++) {
|
for (i = 0; cm->desc->psy_charger_stat[i]; i++) {
|
||||||
if (!strcmp(psy->name, cm->desc->psy_charger_stat[i])) {
|
if (!strcmp(psy->desc->name, cm->desc->psy_charger_stat[i])) {
|
||||||
found = true;
|
found = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@ static int wakeup_enabled;
|
||||||
|
|
||||||
struct collie_bat {
|
struct collie_bat {
|
||||||
int status;
|
int status;
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
int full_chrg;
|
int full_chrg;
|
||||||
|
|
||||||
struct mutex work_lock; /* protects data */
|
struct mutex work_lock; /* protects data */
|
||||||
|
@ -98,7 +98,7 @@ static int collie_bat_get_property(struct power_supply *psy,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct collie_bat *bat = container_of(psy, struct collie_bat, psy);
|
struct collie_bat *bat = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
if (bat->is_present && !bat->is_present(bat)
|
if (bat->is_present && !bat->is_present(bat)
|
||||||
&& psp != POWER_SUPPLY_PROP_PRESENT) {
|
&& psp != POWER_SUPPLY_PROP_PRESENT) {
|
||||||
|
@ -155,14 +155,14 @@ static irqreturn_t collie_bat_gpio_isr(int irq, void *data)
|
||||||
static void collie_bat_update(struct collie_bat *bat)
|
static void collie_bat_update(struct collie_bat *bat)
|
||||||
{
|
{
|
||||||
int old;
|
int old;
|
||||||
struct power_supply *psy = &bat->psy;
|
struct power_supply *psy = bat->psy;
|
||||||
|
|
||||||
mutex_lock(&bat->work_lock);
|
mutex_lock(&bat->work_lock);
|
||||||
|
|
||||||
old = bat->status;
|
old = bat->status;
|
||||||
|
|
||||||
if (bat->is_present && !bat->is_present(bat)) {
|
if (bat->is_present && !bat->is_present(bat)) {
|
||||||
printk(KERN_NOTICE "%s not present\n", psy->name);
|
printk(KERN_NOTICE "%s not present\n", psy->desc->name);
|
||||||
bat->status = POWER_SUPPLY_STATUS_UNKNOWN;
|
bat->status = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
bat->full_chrg = -1;
|
bat->full_chrg = -1;
|
||||||
} else if (power_supply_am_i_supplied(psy)) {
|
} else if (power_supply_am_i_supplied(psy)) {
|
||||||
|
@ -220,18 +220,20 @@ static enum power_supply_property collie_bat_bu_props[] = {
|
||||||
POWER_SUPPLY_PROP_PRESENT,
|
POWER_SUPPLY_PROP_PRESENT,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc collie_bat_main_desc = {
|
||||||
|
.name = "main-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = collie_bat_main_props,
|
||||||
|
.num_properties = ARRAY_SIZE(collie_bat_main_props),
|
||||||
|
.get_property = collie_bat_get_property,
|
||||||
|
.external_power_changed = collie_bat_external_power_changed,
|
||||||
|
.use_for_apm = 1,
|
||||||
|
};
|
||||||
|
|
||||||
static struct collie_bat collie_bat_main = {
|
static struct collie_bat collie_bat_main = {
|
||||||
.status = POWER_SUPPLY_STATUS_DISCHARGING,
|
.status = POWER_SUPPLY_STATUS_DISCHARGING,
|
||||||
.full_chrg = -1,
|
.full_chrg = -1,
|
||||||
.psy = {
|
.psy = NULL,
|
||||||
.name = "main-battery",
|
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
|
||||||
.properties = collie_bat_main_props,
|
|
||||||
.num_properties = ARRAY_SIZE(collie_bat_main_props),
|
|
||||||
.get_property = collie_bat_get_property,
|
|
||||||
.external_power_changed = collie_bat_external_power_changed,
|
|
||||||
.use_for_apm = 1,
|
|
||||||
},
|
|
||||||
|
|
||||||
.gpio_full = COLLIE_GPIO_CO,
|
.gpio_full = COLLIE_GPIO_CO,
|
||||||
.gpio_charge_on = COLLIE_GPIO_CHARGE_ON,
|
.gpio_charge_on = COLLIE_GPIO_CHARGE_ON,
|
||||||
|
@ -249,18 +251,19 @@ static struct collie_bat collie_bat_main = {
|
||||||
.adc_temp_divider = 10000,
|
.adc_temp_divider = 10000,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc collie_bat_bu_desc = {
|
||||||
|
.name = "backup-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = collie_bat_bu_props,
|
||||||
|
.num_properties = ARRAY_SIZE(collie_bat_bu_props),
|
||||||
|
.get_property = collie_bat_get_property,
|
||||||
|
.external_power_changed = collie_bat_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static struct collie_bat collie_bat_bu = {
|
static struct collie_bat collie_bat_bu = {
|
||||||
.status = POWER_SUPPLY_STATUS_UNKNOWN,
|
.status = POWER_SUPPLY_STATUS_UNKNOWN,
|
||||||
.full_chrg = -1,
|
.full_chrg = -1,
|
||||||
|
.psy = NULL,
|
||||||
.psy = {
|
|
||||||
.name = "backup-battery",
|
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
|
||||||
.properties = collie_bat_bu_props,
|
|
||||||
.num_properties = ARRAY_SIZE(collie_bat_bu_props),
|
|
||||||
.get_property = collie_bat_get_property,
|
|
||||||
.external_power_changed = collie_bat_external_power_changed,
|
|
||||||
},
|
|
||||||
|
|
||||||
.gpio_full = -1,
|
.gpio_full = -1,
|
||||||
.gpio_charge_on = -1,
|
.gpio_charge_on = -1,
|
||||||
|
@ -319,6 +322,7 @@ static int collie_bat_resume(struct ucb1x00_dev *dev)
|
||||||
static int collie_bat_probe(struct ucb1x00_dev *dev)
|
static int collie_bat_probe(struct ucb1x00_dev *dev)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
struct power_supply_config psy_main_cfg = {}, psy_bu_cfg = {};
|
||||||
|
|
||||||
if (!machine_is_collie())
|
if (!machine_is_collie())
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
@ -334,12 +338,23 @@ static int collie_bat_probe(struct ucb1x00_dev *dev)
|
||||||
|
|
||||||
INIT_WORK(&bat_work, collie_bat_work);
|
INIT_WORK(&bat_work, collie_bat_work);
|
||||||
|
|
||||||
ret = power_supply_register(&dev->ucb->dev, &collie_bat_main.psy);
|
psy_main_cfg.drv_data = &collie_bat_main;
|
||||||
if (ret)
|
collie_bat_main.psy = power_supply_register(&dev->ucb->dev,
|
||||||
|
&collie_bat_main_desc,
|
||||||
|
&psy_main_cfg);
|
||||||
|
if (IS_ERR(collie_bat_main.psy)) {
|
||||||
|
ret = PTR_ERR(collie_bat_main.psy);
|
||||||
goto err_psy_reg_main;
|
goto err_psy_reg_main;
|
||||||
ret = power_supply_register(&dev->ucb->dev, &collie_bat_bu.psy);
|
}
|
||||||
if (ret)
|
|
||||||
|
psy_main_cfg.drv_data = &collie_bat_bu;
|
||||||
|
collie_bat_bu.psy = power_supply_register(&dev->ucb->dev,
|
||||||
|
&collie_bat_bu_desc,
|
||||||
|
&psy_bu_cfg);
|
||||||
|
if (IS_ERR(collie_bat_bu.psy)) {
|
||||||
|
ret = PTR_ERR(collie_bat_bu.psy);
|
||||||
goto err_psy_reg_bu;
|
goto err_psy_reg_bu;
|
||||||
|
}
|
||||||
|
|
||||||
ret = request_irq(gpio_to_irq(COLLIE_GPIO_CO),
|
ret = request_irq(gpio_to_irq(COLLIE_GPIO_CO),
|
||||||
collie_bat_gpio_isr,
|
collie_bat_gpio_isr,
|
||||||
|
@ -354,9 +369,9 @@ static int collie_bat_probe(struct ucb1x00_dev *dev)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_irq:
|
err_irq:
|
||||||
power_supply_unregister(&collie_bat_bu.psy);
|
power_supply_unregister(collie_bat_bu.psy);
|
||||||
err_psy_reg_bu:
|
err_psy_reg_bu:
|
||||||
power_supply_unregister(&collie_bat_main.psy);
|
power_supply_unregister(collie_bat_main.psy);
|
||||||
err_psy_reg_main:
|
err_psy_reg_main:
|
||||||
|
|
||||||
/* see comment in collie_bat_remove */
|
/* see comment in collie_bat_remove */
|
||||||
|
@ -369,8 +384,8 @@ static void collie_bat_remove(struct ucb1x00_dev *dev)
|
||||||
{
|
{
|
||||||
free_irq(gpio_to_irq(COLLIE_GPIO_CO), &collie_bat_main);
|
free_irq(gpio_to_irq(COLLIE_GPIO_CO), &collie_bat_main);
|
||||||
|
|
||||||
power_supply_unregister(&collie_bat_bu.psy);
|
power_supply_unregister(collie_bat_bu.psy);
|
||||||
power_supply_unregister(&collie_bat_main.psy);
|
power_supply_unregister(collie_bat_main.psy);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Now cancel the bat_work. We won't get any more schedules,
|
* Now cancel the bat_work. We won't get any more schedules,
|
||||||
|
|
|
@ -89,7 +89,8 @@ struct da9030_battery_thresholds {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct da9030_charger {
|
struct da9030_charger {
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
|
struct power_supply_desc psy_desc;
|
||||||
|
|
||||||
struct device *master;
|
struct device *master;
|
||||||
|
|
||||||
|
@ -245,7 +246,7 @@ static void da9030_set_charge(struct da9030_charger *charger, int on)
|
||||||
|
|
||||||
da903x_write(charger->master, DA9030_CHARGE_CONTROL, val);
|
da903x_write(charger->master, DA9030_CHARGE_CONTROL, val);
|
||||||
|
|
||||||
power_supply_changed(&charger->psy);
|
power_supply_changed(charger->psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void da9030_charger_check_state(struct da9030_charger *charger)
|
static void da9030_charger_check_state(struct da9030_charger *charger)
|
||||||
|
@ -341,8 +342,7 @@ static int da9030_battery_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 da9030_charger *charger;
|
struct da9030_charger *charger = power_supply_get_drvdata(psy);
|
||||||
charger = container_of(psy, struct da9030_charger, psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -447,16 +447,16 @@ static void da9030_battery_convert_thresholds(struct da9030_charger *charger,
|
||||||
|
|
||||||
static void da9030_battery_setup_psy(struct da9030_charger *charger)
|
static void da9030_battery_setup_psy(struct da9030_charger *charger)
|
||||||
{
|
{
|
||||||
struct power_supply *psy = &charger->psy;
|
struct power_supply_desc *psy_desc = &charger->psy_desc;
|
||||||
struct power_supply_info *info = charger->battery_info;
|
struct power_supply_info *info = charger->battery_info;
|
||||||
|
|
||||||
psy->name = info->name;
|
psy_desc->name = info->name;
|
||||||
psy->use_for_apm = info->use_for_apm;
|
psy_desc->use_for_apm = info->use_for_apm;
|
||||||
psy->type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_desc->type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
psy->get_property = da9030_battery_get_property;
|
psy_desc->get_property = da9030_battery_get_property;
|
||||||
|
|
||||||
psy->properties = da9030_battery_props;
|
psy_desc->properties = da9030_battery_props;
|
||||||
psy->num_properties = ARRAY_SIZE(da9030_battery_props);
|
psy_desc->num_properties = ARRAY_SIZE(da9030_battery_props);
|
||||||
};
|
};
|
||||||
|
|
||||||
static int da9030_battery_charger_init(struct da9030_charger *charger)
|
static int da9030_battery_charger_init(struct da9030_charger *charger)
|
||||||
|
@ -494,6 +494,7 @@ static int da9030_battery_charger_init(struct da9030_charger *charger)
|
||||||
static int da9030_battery_probe(struct platform_device *pdev)
|
static int da9030_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct da9030_charger *charger;
|
struct da9030_charger *charger;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct da9030_battery_info *pdata = pdev->dev.platform_data;
|
struct da9030_battery_info *pdata = pdev->dev.platform_data;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -541,9 +542,13 @@ static int da9030_battery_probe(struct platform_device *pdev)
|
||||||
goto err_notifier;
|
goto err_notifier;
|
||||||
|
|
||||||
da9030_battery_setup_psy(charger);
|
da9030_battery_setup_psy(charger);
|
||||||
ret = power_supply_register(&pdev->dev, &charger->psy);
|
psy_cfg.drv_data = charger;
|
||||||
if (ret)
|
charger->psy = power_supply_register(&pdev->dev, &charger->psy_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(charger->psy)) {
|
||||||
|
ret = PTR_ERR(charger->psy);
|
||||||
goto err_ps_register;
|
goto err_ps_register;
|
||||||
|
}
|
||||||
|
|
||||||
charger->debug_file = da9030_bat_create_debugfs(charger);
|
charger->debug_file = da9030_bat_create_debugfs(charger);
|
||||||
platform_set_drvdata(pdev, charger);
|
platform_set_drvdata(pdev, charger);
|
||||||
|
@ -571,7 +576,7 @@ static int da9030_battery_remove(struct platform_device *dev)
|
||||||
DA9030_EVENT_CHIOVER | DA9030_EVENT_TBAT);
|
DA9030_EVENT_CHIOVER | DA9030_EVENT_TBAT);
|
||||||
cancel_delayed_work_sync(&charger->work);
|
cancel_delayed_work_sync(&charger->work);
|
||||||
da9030_set_charge(charger, 0);
|
da9030_set_charge(charger, 0);
|
||||||
power_supply_unregister(&charger->psy);
|
power_supply_unregister(charger->psy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -169,7 +169,7 @@ static u32 const vc_tbl[3][68][2] = {
|
||||||
|
|
||||||
struct da9052_battery {
|
struct da9052_battery {
|
||||||
struct da9052 *da9052;
|
struct da9052 *da9052;
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
struct notifier_block nb;
|
struct notifier_block nb;
|
||||||
int charger_type;
|
int charger_type;
|
||||||
int status;
|
int status;
|
||||||
|
@ -452,7 +452,7 @@ static irqreturn_t da9052_bat_irq(int irq, void *data)
|
||||||
|
|
||||||
if (irq == DA9052_IRQ_CHGEND || irq == DA9052_IRQ_DCIN ||
|
if (irq == DA9052_IRQ_CHGEND || irq == DA9052_IRQ_DCIN ||
|
||||||
irq == DA9052_IRQ_VBUS || irq == DA9052_IRQ_TBAT) {
|
irq == DA9052_IRQ_VBUS || irq == DA9052_IRQ_TBAT) {
|
||||||
power_supply_changed(&bat->psy);
|
power_supply_changed(bat->psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
|
@ -499,8 +499,7 @@ static int da9052_bat_get_property(struct power_supply *psy,
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
int illegal;
|
int illegal;
|
||||||
struct da9052_battery *bat = container_of(psy, struct da9052_battery,
|
struct da9052_battery *bat = power_supply_get_drvdata(psy);
|
||||||
psy);
|
|
||||||
|
|
||||||
ret = da9052_bat_check_presence(bat, &illegal);
|
ret = da9052_bat_check_presence(bat, &illegal);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
|
@ -561,7 +560,7 @@ static enum power_supply_property da9052_bat_props[] = {
|
||||||
POWER_SUPPLY_PROP_TECHNOLOGY,
|
POWER_SUPPLY_PROP_TECHNOLOGY,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply template_battery = {
|
static struct power_supply_desc psy_desc = {
|
||||||
.name = "da9052-bat",
|
.name = "da9052-bat",
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
.properties = da9052_bat_props,
|
.properties = da9052_bat_props,
|
||||||
|
@ -591,6 +590,7 @@ static s32 da9052_bat_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct da9052_pdata *pdata;
|
struct da9052_pdata *pdata;
|
||||||
struct da9052_battery *bat;
|
struct da9052_battery *bat;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
int ret;
|
int ret;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
@ -599,8 +599,9 @@ static s32 da9052_bat_probe(struct platform_device *pdev)
|
||||||
if (!bat)
|
if (!bat)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
psy_cfg.drv_data = bat;
|
||||||
|
|
||||||
bat->da9052 = dev_get_drvdata(pdev->dev.parent);
|
bat->da9052 = dev_get_drvdata(pdev->dev.parent);
|
||||||
bat->psy = template_battery;
|
|
||||||
bat->charger_type = DA9052_NOCHARGER;
|
bat->charger_type = DA9052_NOCHARGER;
|
||||||
bat->status = POWER_SUPPLY_STATUS_UNKNOWN;
|
bat->status = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
bat->health = POWER_SUPPLY_HEALTH_UNKNOWN;
|
bat->health = POWER_SUPPLY_HEALTH_UNKNOWN;
|
||||||
|
@ -608,9 +609,9 @@ static s32 da9052_bat_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
pdata = bat->da9052->dev->platform_data;
|
pdata = bat->da9052->dev->platform_data;
|
||||||
if (pdata != NULL && pdata->use_for_apm)
|
if (pdata != NULL && pdata->use_for_apm)
|
||||||
bat->psy.use_for_apm = pdata->use_for_apm;
|
psy_desc.use_for_apm = pdata->use_for_apm;
|
||||||
else
|
else
|
||||||
bat->psy.use_for_apm = 1;
|
psy_desc.use_for_apm = 1;
|
||||||
|
|
||||||
for (i = 0; i < ARRAY_SIZE(da9052_bat_irqs); i++) {
|
for (i = 0; i < ARRAY_SIZE(da9052_bat_irqs); i++) {
|
||||||
ret = da9052_request_irq(bat->da9052,
|
ret = da9052_request_irq(bat->da9052,
|
||||||
|
@ -625,9 +626,11 @@ static s32 da9052_bat_probe(struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &bat->psy);
|
bat->psy = power_supply_register(&pdev->dev, &psy_desc, &psy_cfg);
|
||||||
if (ret)
|
if (IS_ERR(bat->psy)) {
|
||||||
|
ret = PTR_ERR(bat->psy);
|
||||||
goto err;
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
platform_set_drvdata(pdev, bat);
|
platform_set_drvdata(pdev, bat);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -646,7 +649,7 @@ static int da9052_bat_remove(struct platform_device *pdev)
|
||||||
for (i = 0; i < ARRAY_SIZE(da9052_bat_irqs); i++)
|
for (i = 0; i < ARRAY_SIZE(da9052_bat_irqs); i++)
|
||||||
da9052_free_irq(bat->da9052, da9052_bat_irq_bits[i], bat);
|
da9052_free_irq(bat->da9052, da9052_bat_irq_bits[i], bat);
|
||||||
|
|
||||||
power_supply_unregister(&bat->psy);
|
power_supply_unregister(bat->psy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
694
drivers/power/da9150-charger.c
Normal file
694
drivers/power/da9150-charger.c
Normal file
|
@ -0,0 +1,694 @@
|
||||||
|
/*
|
||||||
|
* DA9150 Charger Driver
|
||||||
|
*
|
||||||
|
* Copyright (c) 2014 Dialog Semiconductor
|
||||||
|
*
|
||||||
|
* Author: Adam Thomson <Adam.Thomson.Opensource@diasemi.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/kernel.h>
|
||||||
|
#include <linux/slab.h>
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/platform_device.h>
|
||||||
|
#include <linux/of.h>
|
||||||
|
#include <linux/of_platform.h>
|
||||||
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/power_supply.h>
|
||||||
|
#include <linux/notifier.h>
|
||||||
|
#include <linux/usb/phy.h>
|
||||||
|
#include <linux/iio/consumer.h>
|
||||||
|
#include <linux/mfd/da9150/core.h>
|
||||||
|
#include <linux/mfd/da9150/registers.h>
|
||||||
|
|
||||||
|
/* Private data */
|
||||||
|
struct da9150_charger {
|
||||||
|
struct da9150 *da9150;
|
||||||
|
struct device *dev;
|
||||||
|
|
||||||
|
struct power_supply *usb;
|
||||||
|
struct power_supply *battery;
|
||||||
|
struct power_supply *supply_online;
|
||||||
|
|
||||||
|
struct usb_phy *usb_phy;
|
||||||
|
struct notifier_block otg_nb;
|
||||||
|
struct work_struct otg_work;
|
||||||
|
unsigned long usb_event;
|
||||||
|
|
||||||
|
struct iio_channel *ibus_chan;
|
||||||
|
struct iio_channel *vbus_chan;
|
||||||
|
struct iio_channel *tjunc_chan;
|
||||||
|
struct iio_channel *vbat_chan;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline int da9150_charger_supply_online(struct da9150_charger *charger,
|
||||||
|
struct power_supply *psy,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
val->intval = (psy == charger->supply_online) ? 1 : 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Charger Properties */
|
||||||
|
static int da9150_charger_vbus_voltage_now(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
int v_val, ret;
|
||||||
|
|
||||||
|
/* Read processed value - mV units */
|
||||||
|
ret = iio_read_channel_processed(charger->vbus_chan, &v_val);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* Convert voltage to expected uV units */
|
||||||
|
val->intval = v_val * 1000;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_ibus_current_avg(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
int i_val, ret;
|
||||||
|
|
||||||
|
/* Read processed value - mA units */
|
||||||
|
ret = iio_read_channel_processed(charger->ibus_chan, &i_val);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* Convert current to expected uA units */
|
||||||
|
val->intval = i_val * 1000;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_tjunc_temp(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
int t_val, ret;
|
||||||
|
|
||||||
|
/* Read processed value - 0.001 degrees C units */
|
||||||
|
ret = iio_read_channel_processed(charger->tjunc_chan, &t_val);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* Convert temp to expect 0.1 degrees C units */
|
||||||
|
val->intval = t_val / 100;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum power_supply_property da9150_charger_props[] = {
|
||||||
|
POWER_SUPPLY_PROP_ONLINE,
|
||||||
|
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
||||||
|
POWER_SUPPLY_PROP_CURRENT_AVG,
|
||||||
|
POWER_SUPPLY_PROP_TEMP,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int da9150_charger_get_prop(struct power_supply *psy,
|
||||||
|
enum power_supply_property psp,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger = dev_get_drvdata(psy->dev.parent);
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
switch (psp) {
|
||||||
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
|
ret = da9150_charger_supply_online(charger, psy, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
||||||
|
ret = da9150_charger_vbus_voltage_now(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_CURRENT_AVG:
|
||||||
|
ret = da9150_charger_ibus_current_avg(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_TEMP:
|
||||||
|
ret = da9150_charger_tjunc_temp(charger, val);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ret = -EINVAL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Battery Properties */
|
||||||
|
static int da9150_charger_battery_status(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
u8 reg;
|
||||||
|
|
||||||
|
/* Check to see if battery is discharging */
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_STATUS_H);
|
||||||
|
|
||||||
|
if (((reg & DA9150_VBUS_STAT_MASK) == DA9150_VBUS_STAT_OFF) ||
|
||||||
|
((reg & DA9150_VBUS_STAT_MASK) == DA9150_VBUS_STAT_WAIT)) {
|
||||||
|
val->intval = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_STATUS_J);
|
||||||
|
|
||||||
|
/* Now check for other states */
|
||||||
|
switch (reg & DA9150_CHG_STAT_MASK) {
|
||||||
|
case DA9150_CHG_STAT_ACT:
|
||||||
|
case DA9150_CHG_STAT_PRE:
|
||||||
|
case DA9150_CHG_STAT_CC:
|
||||||
|
case DA9150_CHG_STAT_CV:
|
||||||
|
val->intval = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
|
break;
|
||||||
|
case DA9150_CHG_STAT_OFF:
|
||||||
|
case DA9150_CHG_STAT_SUSP:
|
||||||
|
case DA9150_CHG_STAT_TEMP:
|
||||||
|
case DA9150_CHG_STAT_TIME:
|
||||||
|
case DA9150_CHG_STAT_BAT:
|
||||||
|
val->intval = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
||||||
|
break;
|
||||||
|
case DA9150_CHG_STAT_FULL:
|
||||||
|
val->intval = POWER_SUPPLY_STATUS_FULL;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
val->intval = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_battery_health(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
u8 reg;
|
||||||
|
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_STATUS_J);
|
||||||
|
|
||||||
|
/* Check if temperature limit reached */
|
||||||
|
switch (reg & DA9150_CHG_TEMP_MASK) {
|
||||||
|
case DA9150_CHG_TEMP_UNDER:
|
||||||
|
val->intval = POWER_SUPPLY_HEALTH_COLD;
|
||||||
|
return 0;
|
||||||
|
case DA9150_CHG_TEMP_OVER:
|
||||||
|
val->intval = POWER_SUPPLY_HEALTH_OVERHEAT;
|
||||||
|
return 0;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check for other health states */
|
||||||
|
switch (reg & DA9150_CHG_STAT_MASK) {
|
||||||
|
case DA9150_CHG_STAT_ACT:
|
||||||
|
case DA9150_CHG_STAT_PRE:
|
||||||
|
val->intval = POWER_SUPPLY_HEALTH_DEAD;
|
||||||
|
break;
|
||||||
|
case DA9150_CHG_STAT_TIME:
|
||||||
|
val->intval = POWER_SUPPLY_HEALTH_UNSPEC_FAILURE;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
val->intval = POWER_SUPPLY_HEALTH_GOOD;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_battery_present(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
u8 reg;
|
||||||
|
|
||||||
|
/* Check if battery present or removed */
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_STATUS_J);
|
||||||
|
if ((reg & DA9150_CHG_STAT_MASK) == DA9150_CHG_STAT_BAT)
|
||||||
|
val->intval = 0;
|
||||||
|
else
|
||||||
|
val->intval = 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_battery_charge_type(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
u8 reg;
|
||||||
|
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_STATUS_J);
|
||||||
|
|
||||||
|
switch (reg & DA9150_CHG_STAT_MASK) {
|
||||||
|
case DA9150_CHG_STAT_CC:
|
||||||
|
val->intval = POWER_SUPPLY_CHARGE_TYPE_FAST;
|
||||||
|
break;
|
||||||
|
case DA9150_CHG_STAT_ACT:
|
||||||
|
case DA9150_CHG_STAT_PRE:
|
||||||
|
case DA9150_CHG_STAT_CV:
|
||||||
|
val->intval = POWER_SUPPLY_CHARGE_TYPE_TRICKLE;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
val->intval = POWER_SUPPLY_CHARGE_TYPE_NONE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_battery_voltage_min(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
u8 reg;
|
||||||
|
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_PPR_CHGCTRL_C);
|
||||||
|
|
||||||
|
/* Value starts at 2500 mV, 50 mV increments, presented in uV */
|
||||||
|
val->intval = ((reg & DA9150_CHG_VFAULT_MASK) * 50000) + 2500000;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_battery_voltage_now(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
int v_val, ret;
|
||||||
|
|
||||||
|
/* Read processed value - mV units */
|
||||||
|
ret = iio_read_channel_processed(charger->vbat_chan, &v_val);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
val->intval = v_val * 1000;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_battery_current_max(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
int reg;
|
||||||
|
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_PPR_CHGCTRL_D);
|
||||||
|
|
||||||
|
/* 25mA increments */
|
||||||
|
val->intval = reg * 25000;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_battery_voltage_max(struct da9150_charger *charger,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
u8 reg;
|
||||||
|
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_PPR_CHGCTRL_B);
|
||||||
|
|
||||||
|
/* Value starts at 3650 mV, 25 mV increments, presented in uV */
|
||||||
|
val->intval = ((reg & DA9150_CHG_VBAT_MASK) * 25000) + 3650000;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum power_supply_property da9150_charger_bat_props[] = {
|
||||||
|
POWER_SUPPLY_PROP_STATUS,
|
||||||
|
POWER_SUPPLY_PROP_ONLINE,
|
||||||
|
POWER_SUPPLY_PROP_HEALTH,
|
||||||
|
POWER_SUPPLY_PROP_PRESENT,
|
||||||
|
POWER_SUPPLY_PROP_CHARGE_TYPE,
|
||||||
|
POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN,
|
||||||
|
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
||||||
|
POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX,
|
||||||
|
POWER_SUPPLY_PROP_CONSTANT_CHARGE_VOLTAGE_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int da9150_charger_battery_get_prop(struct power_supply *psy,
|
||||||
|
enum power_supply_property psp,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger = dev_get_drvdata(psy->dev.parent);
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
switch (psp) {
|
||||||
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
ret = da9150_charger_battery_status(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
|
ret = da9150_charger_supply_online(charger, psy, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_HEALTH:
|
||||||
|
ret = da9150_charger_battery_health(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
|
ret = da9150_charger_battery_present(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
||||||
|
ret = da9150_charger_battery_charge_type(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN:
|
||||||
|
ret = da9150_charger_battery_voltage_min(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
||||||
|
ret = da9150_charger_battery_voltage_now(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_CONSTANT_CHARGE_CURRENT_MAX:
|
||||||
|
ret = da9150_charger_battery_current_max(charger, val);
|
||||||
|
break;
|
||||||
|
case POWER_SUPPLY_PROP_CONSTANT_CHARGE_VOLTAGE_MAX:
|
||||||
|
ret = da9150_charger_battery_voltage_max(charger, val);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ret = -EINVAL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static irqreturn_t da9150_charger_chg_irq(int irq, void *data)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger = data;
|
||||||
|
|
||||||
|
power_supply_changed(charger->battery);
|
||||||
|
|
||||||
|
return IRQ_HANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static irqreturn_t da9150_charger_tjunc_irq(int irq, void *data)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger = data;
|
||||||
|
|
||||||
|
/* Nothing we can really do except report this. */
|
||||||
|
dev_crit(charger->dev, "TJunc over temperature!!!\n");
|
||||||
|
power_supply_changed(charger->usb);
|
||||||
|
|
||||||
|
return IRQ_HANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static irqreturn_t da9150_charger_vfault_irq(int irq, void *data)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger = data;
|
||||||
|
|
||||||
|
/* Nothing we can really do except report this. */
|
||||||
|
dev_crit(charger->dev, "VSYS under voltage!!!\n");
|
||||||
|
power_supply_changed(charger->usb);
|
||||||
|
power_supply_changed(charger->battery);
|
||||||
|
|
||||||
|
return IRQ_HANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static irqreturn_t da9150_charger_vbus_irq(int irq, void *data)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger = data;
|
||||||
|
u8 reg;
|
||||||
|
|
||||||
|
reg = da9150_reg_read(charger->da9150, DA9150_STATUS_H);
|
||||||
|
|
||||||
|
/* Charger plugged in or battery only */
|
||||||
|
switch (reg & DA9150_VBUS_STAT_MASK) {
|
||||||
|
case DA9150_VBUS_STAT_OFF:
|
||||||
|
case DA9150_VBUS_STAT_WAIT:
|
||||||
|
charger->supply_online = charger->battery;
|
||||||
|
break;
|
||||||
|
case DA9150_VBUS_STAT_CHG:
|
||||||
|
charger->supply_online = charger->usb;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
dev_warn(charger->dev, "Unknown VBUS state - reg = 0x%x\n",
|
||||||
|
reg);
|
||||||
|
charger->supply_online = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
power_supply_changed(charger->usb);
|
||||||
|
power_supply_changed(charger->battery);
|
||||||
|
|
||||||
|
return IRQ_HANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void da9150_charger_otg_work(struct work_struct *data)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger =
|
||||||
|
container_of(data, struct da9150_charger, otg_work);
|
||||||
|
|
||||||
|
switch (charger->usb_event) {
|
||||||
|
case USB_EVENT_ID:
|
||||||
|
/* Enable OTG Boost */
|
||||||
|
da9150_set_bits(charger->da9150, DA9150_PPR_BKCTRL_A,
|
||||||
|
DA9150_VBUS_MODE_MASK, DA9150_VBUS_MODE_OTG);
|
||||||
|
break;
|
||||||
|
case USB_EVENT_NONE:
|
||||||
|
/* Revert to charge mode */
|
||||||
|
power_supply_changed(charger->usb);
|
||||||
|
power_supply_changed(charger->battery);
|
||||||
|
da9150_set_bits(charger->da9150, DA9150_PPR_BKCTRL_A,
|
||||||
|
DA9150_VBUS_MODE_MASK, DA9150_VBUS_MODE_CHG);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_otg_ncb(struct notifier_block *nb, unsigned long val,
|
||||||
|
void *priv)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger =
|
||||||
|
container_of(nb, struct da9150_charger, otg_nb);
|
||||||
|
|
||||||
|
dev_dbg(charger->dev, "DA9150 OTG notify %lu\n", val);
|
||||||
|
|
||||||
|
charger->usb_event = val;
|
||||||
|
schedule_work(&charger->otg_work);
|
||||||
|
|
||||||
|
return NOTIFY_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_register_irq(struct platform_device *pdev,
|
||||||
|
irq_handler_t handler,
|
||||||
|
const char *irq_name)
|
||||||
|
{
|
||||||
|
struct device *dev = &pdev->dev;
|
||||||
|
struct da9150_charger *charger = platform_get_drvdata(pdev);
|
||||||
|
int irq, ret;
|
||||||
|
|
||||||
|
irq = platform_get_irq_byname(pdev, irq_name);
|
||||||
|
if (irq < 0) {
|
||||||
|
dev_err(dev, "Failed to get IRQ CHG_STATUS: %d\n", irq);
|
||||||
|
return irq;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = request_threaded_irq(irq, NULL, handler, IRQF_ONESHOT, irq_name,
|
||||||
|
charger);
|
||||||
|
if (ret)
|
||||||
|
dev_err(dev, "Failed to request IRQ %d: %d\n", irq, ret);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void da9150_charger_unregister_irq(struct platform_device *pdev,
|
||||||
|
const char *irq_name)
|
||||||
|
{
|
||||||
|
struct device *dev = &pdev->dev;
|
||||||
|
struct da9150_charger *charger = platform_get_drvdata(pdev);
|
||||||
|
int irq;
|
||||||
|
|
||||||
|
irq = platform_get_irq_byname(pdev, irq_name);
|
||||||
|
if (irq < 0) {
|
||||||
|
dev_err(dev, "Failed to get IRQ CHG_STATUS: %d\n", irq);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
free_irq(irq, charger);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc usb_desc = {
|
||||||
|
.name = "da9150-usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = da9150_charger_props,
|
||||||
|
.num_properties = ARRAY_SIZE(da9150_charger_props),
|
||||||
|
.get_property = da9150_charger_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc battery_desc = {
|
||||||
|
.name = "da9150-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = da9150_charger_bat_props,
|
||||||
|
.num_properties = ARRAY_SIZE(da9150_charger_bat_props),
|
||||||
|
.get_property = da9150_charger_battery_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int da9150_charger_probe(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct device *dev = &pdev->dev;
|
||||||
|
struct da9150 *da9150 = dev_get_drvdata(dev->parent);
|
||||||
|
struct da9150_charger *charger;
|
||||||
|
u8 reg;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
charger = devm_kzalloc(dev, sizeof(struct da9150_charger), GFP_KERNEL);
|
||||||
|
if (!charger)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
platform_set_drvdata(pdev, charger);
|
||||||
|
charger->da9150 = da9150;
|
||||||
|
charger->dev = dev;
|
||||||
|
|
||||||
|
/* Acquire ADC channels */
|
||||||
|
charger->ibus_chan = iio_channel_get(dev, "CHAN_IBUS");
|
||||||
|
if (IS_ERR(charger->ibus_chan)) {
|
||||||
|
ret = PTR_ERR(charger->ibus_chan);
|
||||||
|
goto ibus_chan_fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
charger->vbus_chan = iio_channel_get(dev, "CHAN_VBUS");
|
||||||
|
if (IS_ERR(charger->vbus_chan)) {
|
||||||
|
ret = PTR_ERR(charger->vbus_chan);
|
||||||
|
goto vbus_chan_fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
charger->tjunc_chan = iio_channel_get(dev, "CHAN_TJUNC");
|
||||||
|
if (IS_ERR(charger->tjunc_chan)) {
|
||||||
|
ret = PTR_ERR(charger->tjunc_chan);
|
||||||
|
goto tjunc_chan_fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
charger->vbat_chan = iio_channel_get(dev, "CHAN_VBAT");
|
||||||
|
if (IS_ERR(charger->vbat_chan)) {
|
||||||
|
ret = PTR_ERR(charger->vbat_chan);
|
||||||
|
goto vbat_chan_fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Register power supplies */
|
||||||
|
charger->usb = power_supply_register(dev, &usb_desc, NULL);
|
||||||
|
if (IS_ERR(charger->usb)) {
|
||||||
|
ret = PTR_ERR(charger->usb);
|
||||||
|
goto usb_fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
charger->battery = power_supply_register(dev, &battery_desc, NULL);
|
||||||
|
if (IS_ERR(charger->battery)) {
|
||||||
|
ret = PTR_ERR(charger->battery);
|
||||||
|
goto battery_fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get initial online supply */
|
||||||
|
reg = da9150_reg_read(da9150, DA9150_STATUS_H);
|
||||||
|
|
||||||
|
switch (reg & DA9150_VBUS_STAT_MASK) {
|
||||||
|
case DA9150_VBUS_STAT_OFF:
|
||||||
|
case DA9150_VBUS_STAT_WAIT:
|
||||||
|
charger->supply_online = charger->battery;
|
||||||
|
break;
|
||||||
|
case DA9150_VBUS_STAT_CHG:
|
||||||
|
charger->supply_online = charger->usb;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
dev_warn(dev, "Unknown VBUS state - reg = 0x%x\n", reg);
|
||||||
|
charger->supply_online = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Setup OTG reporting & configuration */
|
||||||
|
charger->usb_phy = devm_usb_get_phy(dev, USB_PHY_TYPE_USB2);
|
||||||
|
if (!IS_ERR_OR_NULL(charger->usb_phy)) {
|
||||||
|
INIT_WORK(&charger->otg_work, da9150_charger_otg_work);
|
||||||
|
charger->otg_nb.notifier_call = da9150_charger_otg_ncb;
|
||||||
|
usb_register_notifier(charger->usb_phy, &charger->otg_nb);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Register IRQs */
|
||||||
|
ret = da9150_charger_register_irq(pdev, da9150_charger_chg_irq,
|
||||||
|
"CHG_STATUS");
|
||||||
|
if (ret < 0)
|
||||||
|
goto chg_irq_fail;
|
||||||
|
|
||||||
|
ret = da9150_charger_register_irq(pdev, da9150_charger_tjunc_irq,
|
||||||
|
"CHG_TJUNC");
|
||||||
|
if (ret < 0)
|
||||||
|
goto tjunc_irq_fail;
|
||||||
|
|
||||||
|
ret = da9150_charger_register_irq(pdev, da9150_charger_vfault_irq,
|
||||||
|
"CHG_VFAULT");
|
||||||
|
if (ret < 0)
|
||||||
|
goto vfault_irq_fail;
|
||||||
|
|
||||||
|
ret = da9150_charger_register_irq(pdev, da9150_charger_vbus_irq,
|
||||||
|
"CHG_VBUS");
|
||||||
|
if (ret < 0)
|
||||||
|
goto vbus_irq_fail;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
|
||||||
|
vbus_irq_fail:
|
||||||
|
da9150_charger_unregister_irq(pdev, "CHG_VFAULT");
|
||||||
|
vfault_irq_fail:
|
||||||
|
da9150_charger_unregister_irq(pdev, "CHG_TJUNC");
|
||||||
|
tjunc_irq_fail:
|
||||||
|
da9150_charger_unregister_irq(pdev, "CHG_STATUS");
|
||||||
|
chg_irq_fail:
|
||||||
|
if (!IS_ERR_OR_NULL(charger->usb_phy))
|
||||||
|
usb_unregister_notifier(charger->usb_phy, &charger->otg_nb);
|
||||||
|
battery_fail:
|
||||||
|
power_supply_unregister(charger->usb);
|
||||||
|
|
||||||
|
usb_fail:
|
||||||
|
iio_channel_release(charger->vbat_chan);
|
||||||
|
|
||||||
|
vbat_chan_fail:
|
||||||
|
iio_channel_release(charger->tjunc_chan);
|
||||||
|
|
||||||
|
tjunc_chan_fail:
|
||||||
|
iio_channel_release(charger->vbus_chan);
|
||||||
|
|
||||||
|
vbus_chan_fail:
|
||||||
|
iio_channel_release(charger->ibus_chan);
|
||||||
|
|
||||||
|
ibus_chan_fail:
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int da9150_charger_remove(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
struct da9150_charger *charger = platform_get_drvdata(pdev);
|
||||||
|
int irq;
|
||||||
|
|
||||||
|
/* Make sure IRQs are released before unregistering power supplies */
|
||||||
|
irq = platform_get_irq_byname(pdev, "CHG_VBUS");
|
||||||
|
free_irq(irq, charger);
|
||||||
|
|
||||||
|
irq = platform_get_irq_byname(pdev, "CHG_VFAULT");
|
||||||
|
free_irq(irq, charger);
|
||||||
|
|
||||||
|
irq = platform_get_irq_byname(pdev, "CHG_TJUNC");
|
||||||
|
free_irq(irq, charger);
|
||||||
|
|
||||||
|
irq = platform_get_irq_byname(pdev, "CHG_STATUS");
|
||||||
|
free_irq(irq, charger);
|
||||||
|
|
||||||
|
if (!IS_ERR_OR_NULL(charger->usb_phy))
|
||||||
|
usb_unregister_notifier(charger->usb_phy, &charger->otg_nb);
|
||||||
|
|
||||||
|
power_supply_unregister(charger->battery);
|
||||||
|
power_supply_unregister(charger->usb);
|
||||||
|
|
||||||
|
/* Release ADC channels */
|
||||||
|
iio_channel_release(charger->ibus_chan);
|
||||||
|
iio_channel_release(charger->vbus_chan);
|
||||||
|
iio_channel_release(charger->tjunc_chan);
|
||||||
|
iio_channel_release(charger->vbat_chan);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct platform_driver da9150_charger_driver = {
|
||||||
|
.driver = {
|
||||||
|
.name = "da9150-charger",
|
||||||
|
},
|
||||||
|
.probe = da9150_charger_probe,
|
||||||
|
.remove = da9150_charger_remove,
|
||||||
|
};
|
||||||
|
|
||||||
|
module_platform_driver(da9150_charger_driver);
|
||||||
|
|
||||||
|
MODULE_DESCRIPTION("Charger Driver for DA9150");
|
||||||
|
MODULE_AUTHOR("Adam Thomson <Adam.Thomson.Opensource@diasemi.com>");
|
||||||
|
MODULE_LICENSE("GPL");
|
|
@ -53,7 +53,8 @@ struct ds2760_device_info {
|
||||||
int charge_status; /* POWER_SUPPLY_STATUS_* */
|
int charge_status; /* POWER_SUPPLY_STATUS_* */
|
||||||
|
|
||||||
int full_counter;
|
int full_counter;
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
struct power_supply_desc bat_desc;
|
||||||
struct device *w1_dev;
|
struct device *w1_dev;
|
||||||
struct workqueue_struct *monitor_wqueue;
|
struct workqueue_struct *monitor_wqueue;
|
||||||
struct delayed_work monitor_work;
|
struct delayed_work monitor_work;
|
||||||
|
@ -254,7 +255,7 @@ static void ds2760_battery_update_status(struct ds2760_device_info *di)
|
||||||
if (di->charge_status == POWER_SUPPLY_STATUS_UNKNOWN)
|
if (di->charge_status == POWER_SUPPLY_STATUS_UNKNOWN)
|
||||||
di->full_counter = 0;
|
di->full_counter = 0;
|
||||||
|
|
||||||
if (power_supply_am_i_supplied(&di->bat)) {
|
if (power_supply_am_i_supplied(di->bat)) {
|
||||||
if (di->current_uA > 10000) {
|
if (di->current_uA > 10000) {
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
di->charge_status = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
di->full_counter = 0;
|
di->full_counter = 0;
|
||||||
|
@ -287,7 +288,7 @@ static void ds2760_battery_update_status(struct ds2760_device_info *di)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (di->charge_status != old_charge_status)
|
if (di->charge_status != old_charge_status)
|
||||||
power_supply_changed(&di->bat);
|
power_supply_changed(di->bat);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ds2760_battery_write_status(struct ds2760_device_info *di,
|
static void ds2760_battery_write_status(struct ds2760_device_info *di,
|
||||||
|
@ -346,12 +347,9 @@ static void ds2760_battery_work(struct work_struct *work)
|
||||||
queue_delayed_work(di->monitor_wqueue, &di->monitor_work, interval);
|
queue_delayed_work(di->monitor_wqueue, &di->monitor_work, interval);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define to_ds2760_device_info(x) container_of((x), struct ds2760_device_info, \
|
|
||||||
bat);
|
|
||||||
|
|
||||||
static void ds2760_battery_external_power_changed(struct power_supply *psy)
|
static void ds2760_battery_external_power_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct ds2760_device_info *di = to_ds2760_device_info(psy);
|
struct ds2760_device_info *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
dev_dbg(di->dev, "%s\n", __func__);
|
dev_dbg(di->dev, "%s\n", __func__);
|
||||||
|
|
||||||
|
@ -377,7 +375,7 @@ static void ds2760_battery_set_charged_work(struct work_struct *work)
|
||||||
* that error.
|
* that error.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (!power_supply_am_i_supplied(&di->bat))
|
if (!power_supply_am_i_supplied(di->bat))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
bias = (signed char) di->current_raw +
|
bias = (signed char) di->current_raw +
|
||||||
|
@ -396,7 +394,7 @@ static void ds2760_battery_set_charged_work(struct work_struct *work)
|
||||||
|
|
||||||
static void ds2760_battery_set_charged(struct power_supply *psy)
|
static void ds2760_battery_set_charged(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct ds2760_device_info *di = to_ds2760_device_info(psy);
|
struct ds2760_device_info *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
/* postpone the actual work by 20 secs. This is for debouncing GPIO
|
/* postpone the actual work by 20 secs. This is for debouncing GPIO
|
||||||
* signals and to let the current value settle. See AN4188. */
|
* signals and to let the current value settle. See AN4188. */
|
||||||
|
@ -407,7 +405,7 @@ static int ds2760_battery_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 ds2760_device_info *di = to_ds2760_device_info(psy);
|
struct ds2760_device_info *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -458,7 +456,7 @@ static int ds2760_battery_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)
|
||||||
{
|
{
|
||||||
struct ds2760_device_info *di = to_ds2760_device_info(psy);
|
struct ds2760_device_info *di = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_CHARGE_FULL:
|
case POWER_SUPPLY_PROP_CHARGE_FULL:
|
||||||
|
@ -508,6 +506,7 @@ static enum power_supply_property ds2760_battery_props[] = {
|
||||||
|
|
||||||
static int ds2760_battery_probe(struct platform_device *pdev)
|
static int ds2760_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
char status;
|
char status;
|
||||||
int retval = 0;
|
int retval = 0;
|
||||||
struct ds2760_device_info *di;
|
struct ds2760_device_info *di;
|
||||||
|
@ -520,20 +519,22 @@ static int ds2760_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
platform_set_drvdata(pdev, di);
|
platform_set_drvdata(pdev, di);
|
||||||
|
|
||||||
di->dev = &pdev->dev;
|
di->dev = &pdev->dev;
|
||||||
di->w1_dev = pdev->dev.parent;
|
di->w1_dev = pdev->dev.parent;
|
||||||
di->bat.name = dev_name(&pdev->dev);
|
di->bat_desc.name = dev_name(&pdev->dev);
|
||||||
di->bat.type = POWER_SUPPLY_TYPE_BATTERY;
|
di->bat_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
di->bat.properties = ds2760_battery_props;
|
di->bat_desc.properties = ds2760_battery_props;
|
||||||
di->bat.num_properties = ARRAY_SIZE(ds2760_battery_props);
|
di->bat_desc.num_properties = ARRAY_SIZE(ds2760_battery_props);
|
||||||
di->bat.get_property = ds2760_battery_get_property;
|
di->bat_desc.get_property = ds2760_battery_get_property;
|
||||||
di->bat.set_property = ds2760_battery_set_property;
|
di->bat_desc.set_property = ds2760_battery_set_property;
|
||||||
di->bat.property_is_writeable =
|
di->bat_desc.property_is_writeable =
|
||||||
ds2760_battery_property_is_writeable;
|
ds2760_battery_property_is_writeable;
|
||||||
di->bat.set_charged = ds2760_battery_set_charged;
|
di->bat_desc.set_charged = ds2760_battery_set_charged;
|
||||||
di->bat.external_power_changed =
|
di->bat_desc.external_power_changed =
|
||||||
ds2760_battery_external_power_changed;
|
ds2760_battery_external_power_changed;
|
||||||
|
|
||||||
|
psy_cfg.drv_data = di;
|
||||||
|
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_UNKNOWN;
|
di->charge_status = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
|
|
||||||
/* enable sleep mode feature */
|
/* enable sleep mode feature */
|
||||||
|
@ -555,9 +556,10 @@ static int ds2760_battery_probe(struct platform_device *pdev)
|
||||||
if (current_accum)
|
if (current_accum)
|
||||||
ds2760_battery_set_current_accum(di, current_accum);
|
ds2760_battery_set_current_accum(di, current_accum);
|
||||||
|
|
||||||
retval = power_supply_register(&pdev->dev, &di->bat);
|
di->bat = power_supply_register(&pdev->dev, &di->bat_desc, &psy_cfg);
|
||||||
if (retval) {
|
if (IS_ERR(di->bat)) {
|
||||||
dev_err(di->dev, "failed to register battery\n");
|
dev_err(di->dev, "failed to register battery\n");
|
||||||
|
retval = PTR_ERR(di->bat);
|
||||||
goto batt_failed;
|
goto batt_failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -574,7 +576,7 @@ static int ds2760_battery_probe(struct platform_device *pdev)
|
||||||
goto success;
|
goto success;
|
||||||
|
|
||||||
workqueue_failed:
|
workqueue_failed:
|
||||||
power_supply_unregister(&di->bat);
|
power_supply_unregister(di->bat);
|
||||||
batt_failed:
|
batt_failed:
|
||||||
di_alloc_failed:
|
di_alloc_failed:
|
||||||
success:
|
success:
|
||||||
|
@ -588,7 +590,7 @@ static int ds2760_battery_remove(struct platform_device *pdev)
|
||||||
cancel_delayed_work_sync(&di->monitor_work);
|
cancel_delayed_work_sync(&di->monitor_work);
|
||||||
cancel_delayed_work_sync(&di->set_charged_work);
|
cancel_delayed_work_sync(&di->set_charged_work);
|
||||||
destroy_workqueue(di->monitor_wqueue);
|
destroy_workqueue(di->monitor_wqueue);
|
||||||
power_supply_unregister(&di->bat);
|
power_supply_unregister(di->bat);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -610,7 +612,7 @@ static int ds2760_battery_resume(struct platform_device *pdev)
|
||||||
struct ds2760_device_info *di = platform_get_drvdata(pdev);
|
struct ds2760_device_info *di = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
di->charge_status = POWER_SUPPLY_STATUS_UNKNOWN;
|
di->charge_status = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
power_supply_changed(&di->bat);
|
power_supply_changed(di->bat);
|
||||||
|
|
||||||
mod_delayed_work(di->monitor_wqueue, &di->monitor_work, HZ);
|
mod_delayed_work(di->monitor_wqueue, &di->monitor_work, HZ);
|
||||||
|
|
||||||
|
|
|
@ -37,7 +37,8 @@
|
||||||
|
|
||||||
struct ds2780_device_info {
|
struct ds2780_device_info {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
struct power_supply_desc bat_desc;
|
||||||
struct device *w1_dev;
|
struct device *w1_dev;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -52,7 +53,7 @@ static const char manufacturer[] = "Maxim/Dallas";
|
||||||
static inline struct ds2780_device_info *
|
static inline struct ds2780_device_info *
|
||||||
to_ds2780_device_info(struct power_supply *psy)
|
to_ds2780_device_info(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
return container_of(psy, struct ds2780_device_info, bat);
|
return power_supply_get_drvdata(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct power_supply *to_power_supply(struct device *dev)
|
static inline struct power_supply *to_power_supply(struct device *dev)
|
||||||
|
@ -757,6 +758,7 @@ static const struct attribute_group ds2780_attr_group = {
|
||||||
|
|
||||||
static int ds2780_battery_probe(struct platform_device *pdev)
|
static int ds2780_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct ds2780_device_info *dev_info;
|
struct ds2780_device_info *dev_info;
|
||||||
|
|
||||||
|
@ -770,25 +772,29 @@ static int ds2780_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
dev_info->dev = &pdev->dev;
|
dev_info->dev = &pdev->dev;
|
||||||
dev_info->w1_dev = pdev->dev.parent;
|
dev_info->w1_dev = pdev->dev.parent;
|
||||||
dev_info->bat.name = dev_name(&pdev->dev);
|
dev_info->bat_desc.name = dev_name(&pdev->dev);
|
||||||
dev_info->bat.type = POWER_SUPPLY_TYPE_BATTERY;
|
dev_info->bat_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
dev_info->bat.properties = ds2780_battery_props;
|
dev_info->bat_desc.properties = ds2780_battery_props;
|
||||||
dev_info->bat.num_properties = ARRAY_SIZE(ds2780_battery_props);
|
dev_info->bat_desc.num_properties = ARRAY_SIZE(ds2780_battery_props);
|
||||||
dev_info->bat.get_property = ds2780_battery_get_property;
|
dev_info->bat_desc.get_property = ds2780_battery_get_property;
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &dev_info->bat);
|
psy_cfg.drv_data = dev_info;
|
||||||
if (ret) {
|
|
||||||
|
dev_info->bat = power_supply_register(&pdev->dev, &dev_info->bat_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(dev_info->bat)) {
|
||||||
dev_err(dev_info->dev, "failed to register battery\n");
|
dev_err(dev_info->dev, "failed to register battery\n");
|
||||||
|
ret = PTR_ERR(dev_info->bat);
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_group(&dev_info->bat.dev->kobj, &ds2780_attr_group);
|
ret = sysfs_create_group(&dev_info->bat->dev.kobj, &ds2780_attr_group);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev_info->dev, "failed to create sysfs group\n");
|
dev_err(dev_info->dev, "failed to create sysfs group\n");
|
||||||
goto fail_unregister;
|
goto fail_unregister;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_bin_file(&dev_info->bat.dev->kobj,
|
ret = sysfs_create_bin_file(&dev_info->bat->dev.kobj,
|
||||||
&ds2780_param_eeprom_bin_attr);
|
&ds2780_param_eeprom_bin_attr);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev_info->dev,
|
dev_err(dev_info->dev,
|
||||||
|
@ -796,7 +802,7 @@ static int ds2780_battery_probe(struct platform_device *pdev)
|
||||||
goto fail_remove_group;
|
goto fail_remove_group;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_bin_file(&dev_info->bat.dev->kobj,
|
ret = sysfs_create_bin_file(&dev_info->bat->dev.kobj,
|
||||||
&ds2780_user_eeprom_bin_attr);
|
&ds2780_user_eeprom_bin_attr);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev_info->dev,
|
dev_err(dev_info->dev,
|
||||||
|
@ -807,12 +813,12 @@ static int ds2780_battery_probe(struct platform_device *pdev)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fail_remove_bin_file:
|
fail_remove_bin_file:
|
||||||
sysfs_remove_bin_file(&dev_info->bat.dev->kobj,
|
sysfs_remove_bin_file(&dev_info->bat->dev.kobj,
|
||||||
&ds2780_param_eeprom_bin_attr);
|
&ds2780_param_eeprom_bin_attr);
|
||||||
fail_remove_group:
|
fail_remove_group:
|
||||||
sysfs_remove_group(&dev_info->bat.dev->kobj, &ds2780_attr_group);
|
sysfs_remove_group(&dev_info->bat->dev.kobj, &ds2780_attr_group);
|
||||||
fail_unregister:
|
fail_unregister:
|
||||||
power_supply_unregister(&dev_info->bat);
|
power_supply_unregister(dev_info->bat);
|
||||||
fail:
|
fail:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -821,10 +827,13 @@ static int ds2780_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct ds2780_device_info *dev_info = platform_get_drvdata(pdev);
|
struct ds2780_device_info *dev_info = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
/* remove attributes */
|
/*
|
||||||
sysfs_remove_group(&dev_info->bat.dev->kobj, &ds2780_attr_group);
|
* Remove attributes before unregistering power supply
|
||||||
|
* because 'bat' will be freed on power_supply_unregister() call.
|
||||||
|
*/
|
||||||
|
sysfs_remove_group(&dev_info->bat->dev.kobj, &ds2780_attr_group);
|
||||||
|
|
||||||
power_supply_unregister(&dev_info->bat);
|
power_supply_unregister(dev_info->bat);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,7 +35,8 @@
|
||||||
|
|
||||||
struct ds2781_device_info {
|
struct ds2781_device_info {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
struct power_supply_desc bat_desc;
|
||||||
struct device *w1_dev;
|
struct device *w1_dev;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -50,7 +51,7 @@ static const char manufacturer[] = "Maxim/Dallas";
|
||||||
static inline struct ds2781_device_info *
|
static inline struct ds2781_device_info *
|
||||||
to_ds2781_device_info(struct power_supply *psy)
|
to_ds2781_device_info(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
return container_of(psy, struct ds2781_device_info, bat);
|
return power_supply_get_drvdata(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct power_supply *to_power_supply(struct device *dev)
|
static inline struct power_supply *to_power_supply(struct device *dev)
|
||||||
|
@ -328,7 +329,7 @@ static int ds2781_get_status(struct ds2781_device_info *dev_info, int *status)
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
if (power_supply_am_i_supplied(&dev_info->bat)) {
|
if (power_supply_am_i_supplied(dev_info->bat)) {
|
||||||
if (capacity == 100)
|
if (capacity == 100)
|
||||||
*status = POWER_SUPPLY_STATUS_FULL;
|
*status = POWER_SUPPLY_STATUS_FULL;
|
||||||
else if (current_uA > 50000)
|
else if (current_uA > 50000)
|
||||||
|
@ -752,6 +753,7 @@ static const struct attribute_group ds2781_attr_group = {
|
||||||
|
|
||||||
static int ds2781_battery_probe(struct platform_device *pdev)
|
static int ds2781_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct ds2781_device_info *dev_info;
|
struct ds2781_device_info *dev_info;
|
||||||
|
|
||||||
|
@ -763,25 +765,29 @@ static int ds2781_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
dev_info->dev = &pdev->dev;
|
dev_info->dev = &pdev->dev;
|
||||||
dev_info->w1_dev = pdev->dev.parent;
|
dev_info->w1_dev = pdev->dev.parent;
|
||||||
dev_info->bat.name = dev_name(&pdev->dev);
|
dev_info->bat_desc.name = dev_name(&pdev->dev);
|
||||||
dev_info->bat.type = POWER_SUPPLY_TYPE_BATTERY;
|
dev_info->bat_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
dev_info->bat.properties = ds2781_battery_props;
|
dev_info->bat_desc.properties = ds2781_battery_props;
|
||||||
dev_info->bat.num_properties = ARRAY_SIZE(ds2781_battery_props);
|
dev_info->bat_desc.num_properties = ARRAY_SIZE(ds2781_battery_props);
|
||||||
dev_info->bat.get_property = ds2781_battery_get_property;
|
dev_info->bat_desc.get_property = ds2781_battery_get_property;
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &dev_info->bat);
|
psy_cfg.drv_data = dev_info;
|
||||||
if (ret) {
|
|
||||||
|
dev_info->bat = power_supply_register(&pdev->dev, &dev_info->bat_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(dev_info->bat)) {
|
||||||
dev_err(dev_info->dev, "failed to register battery\n");
|
dev_err(dev_info->dev, "failed to register battery\n");
|
||||||
|
ret = PTR_ERR(dev_info->bat);
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_group(&dev_info->bat.dev->kobj, &ds2781_attr_group);
|
ret = sysfs_create_group(&dev_info->bat->dev.kobj, &ds2781_attr_group);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev_info->dev, "failed to create sysfs group\n");
|
dev_err(dev_info->dev, "failed to create sysfs group\n");
|
||||||
goto fail_unregister;
|
goto fail_unregister;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_bin_file(&dev_info->bat.dev->kobj,
|
ret = sysfs_create_bin_file(&dev_info->bat->dev.kobj,
|
||||||
&ds2781_param_eeprom_bin_attr);
|
&ds2781_param_eeprom_bin_attr);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev_info->dev,
|
dev_err(dev_info->dev,
|
||||||
|
@ -789,7 +795,7 @@ static int ds2781_battery_probe(struct platform_device *pdev)
|
||||||
goto fail_remove_group;
|
goto fail_remove_group;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = sysfs_create_bin_file(&dev_info->bat.dev->kobj,
|
ret = sysfs_create_bin_file(&dev_info->bat->dev.kobj,
|
||||||
&ds2781_user_eeprom_bin_attr);
|
&ds2781_user_eeprom_bin_attr);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev_info->dev,
|
dev_err(dev_info->dev,
|
||||||
|
@ -800,12 +806,12 @@ static int ds2781_battery_probe(struct platform_device *pdev)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fail_remove_bin_file:
|
fail_remove_bin_file:
|
||||||
sysfs_remove_bin_file(&dev_info->bat.dev->kobj,
|
sysfs_remove_bin_file(&dev_info->bat->dev.kobj,
|
||||||
&ds2781_param_eeprom_bin_attr);
|
&ds2781_param_eeprom_bin_attr);
|
||||||
fail_remove_group:
|
fail_remove_group:
|
||||||
sysfs_remove_group(&dev_info->bat.dev->kobj, &ds2781_attr_group);
|
sysfs_remove_group(&dev_info->bat->dev.kobj, &ds2781_attr_group);
|
||||||
fail_unregister:
|
fail_unregister:
|
||||||
power_supply_unregister(&dev_info->bat);
|
power_supply_unregister(dev_info->bat);
|
||||||
fail:
|
fail:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -814,10 +820,13 @@ static int ds2781_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct ds2781_device_info *dev_info = platform_get_drvdata(pdev);
|
struct ds2781_device_info *dev_info = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
/* remove attributes */
|
/*
|
||||||
sysfs_remove_group(&dev_info->bat.dev->kobj, &ds2781_attr_group);
|
* Remove attributes before unregistering power supply
|
||||||
|
* because 'bat' will be freed on power_supply_unregister() call.
|
||||||
|
*/
|
||||||
|
sysfs_remove_group(&dev_info->bat->dev.kobj, &ds2781_attr_group);
|
||||||
|
|
||||||
power_supply_unregister(&dev_info->bat);
|
power_supply_unregister(dev_info->bat);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,11 +53,12 @@ struct ds278x_battery_ops {
|
||||||
int (*get_battery_capacity)(struct ds278x_info *info, int *capacity);
|
int (*get_battery_capacity)(struct ds278x_info *info, int *capacity);
|
||||||
};
|
};
|
||||||
|
|
||||||
#define to_ds278x_info(x) container_of(x, struct ds278x_info, battery)
|
#define to_ds278x_info(x) power_supply_get_drvdata(x)
|
||||||
|
|
||||||
struct ds278x_info {
|
struct ds278x_info {
|
||||||
struct i2c_client *client;
|
struct i2c_client *client;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
|
struct power_supply_desc battery_desc;
|
||||||
struct ds278x_battery_ops *ops;
|
struct ds278x_battery_ops *ops;
|
||||||
struct delayed_work bat_work;
|
struct delayed_work bat_work;
|
||||||
int id;
|
int id;
|
||||||
|
@ -285,7 +286,7 @@ static void ds278x_bat_update(struct ds278x_info *info)
|
||||||
ds278x_get_status(info, &info->status);
|
ds278x_get_status(info, &info->status);
|
||||||
|
|
||||||
if ((old_status != info->status) || (old_capacity != info->capacity))
|
if ((old_status != info->status) || (old_capacity != info->capacity))
|
||||||
power_supply_changed(&info->battery);
|
power_supply_changed(info->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ds278x_bat_work(struct work_struct *work)
|
static void ds278x_bat_work(struct work_struct *work)
|
||||||
|
@ -306,7 +307,7 @@ static enum power_supply_property ds278x_battery_props[] = {
|
||||||
POWER_SUPPLY_PROP_TEMP,
|
POWER_SUPPLY_PROP_TEMP,
|
||||||
};
|
};
|
||||||
|
|
||||||
static void ds278x_power_supply_init(struct power_supply *battery)
|
static void ds278x_power_supply_init(struct power_supply_desc *battery)
|
||||||
{
|
{
|
||||||
battery->type = POWER_SUPPLY_TYPE_BATTERY;
|
battery->type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
battery->properties = ds278x_battery_props;
|
battery->properties = ds278x_battery_props;
|
||||||
|
@ -319,8 +320,8 @@ static int ds278x_battery_remove(struct i2c_client *client)
|
||||||
{
|
{
|
||||||
struct ds278x_info *info = i2c_get_clientdata(client);
|
struct ds278x_info *info = i2c_get_clientdata(client);
|
||||||
|
|
||||||
power_supply_unregister(&info->battery);
|
power_supply_unregister(info->battery);
|
||||||
kfree(info->battery.name);
|
kfree(info->battery_desc.name);
|
||||||
|
|
||||||
mutex_lock(&battery_lock);
|
mutex_lock(&battery_lock);
|
||||||
idr_remove(&battery_id, info->id);
|
idr_remove(&battery_id, info->id);
|
||||||
|
@ -377,6 +378,7 @@ static int ds278x_battery_probe(struct i2c_client *client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
struct ds278x_platform_data *pdata = client->dev.platform_data;
|
struct ds278x_platform_data *pdata = client->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct ds278x_info *info;
|
struct ds278x_info *info;
|
||||||
int ret;
|
int ret;
|
||||||
int num;
|
int num;
|
||||||
|
@ -404,8 +406,9 @@ static int ds278x_battery_probe(struct i2c_client *client,
|
||||||
goto fail_info;
|
goto fail_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
info->battery.name = kasprintf(GFP_KERNEL, "%s-%d", client->name, num);
|
info->battery_desc.name = kasprintf(GFP_KERNEL, "%s-%d",
|
||||||
if (!info->battery.name) {
|
client->name, num);
|
||||||
|
if (!info->battery_desc.name) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto fail_name;
|
goto fail_name;
|
||||||
}
|
}
|
||||||
|
@ -417,16 +420,19 @@ static int ds278x_battery_probe(struct i2c_client *client,
|
||||||
info->client = client;
|
info->client = client;
|
||||||
info->id = num;
|
info->id = num;
|
||||||
info->ops = &ds278x_ops[id->driver_data];
|
info->ops = &ds278x_ops[id->driver_data];
|
||||||
ds278x_power_supply_init(&info->battery);
|
ds278x_power_supply_init(&info->battery_desc);
|
||||||
|
psy_cfg.drv_data = info;
|
||||||
|
|
||||||
info->capacity = 100;
|
info->capacity = 100;
|
||||||
info->status = POWER_SUPPLY_STATUS_FULL;
|
info->status = POWER_SUPPLY_STATUS_FULL;
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&info->bat_work, ds278x_bat_work);
|
INIT_DELAYED_WORK(&info->bat_work, ds278x_bat_work);
|
||||||
|
|
||||||
ret = power_supply_register(&client->dev, &info->battery);
|
info->battery = power_supply_register(&client->dev,
|
||||||
if (ret) {
|
&info->battery_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(info->battery)) {
|
||||||
dev_err(&client->dev, "failed to register battery\n");
|
dev_err(&client->dev, "failed to register battery\n");
|
||||||
|
ret = PTR_ERR(info->battery);
|
||||||
goto fail_register;
|
goto fail_register;
|
||||||
} else {
|
} else {
|
||||||
schedule_delayed_work(&info->bat_work, DS278x_DELAY);
|
schedule_delayed_work(&info->bat_work, DS278x_DELAY);
|
||||||
|
@ -435,7 +441,7 @@ static int ds278x_battery_probe(struct i2c_client *client,
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fail_register:
|
fail_register:
|
||||||
kfree(info->battery.name);
|
kfree(info->battery_desc.name);
|
||||||
fail_name:
|
fail_name:
|
||||||
kfree(info);
|
kfree(info);
|
||||||
fail_info:
|
fail_info:
|
||||||
|
|
|
@ -44,7 +44,8 @@ static const char *const gab_chan_name[] = {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct gab {
|
struct gab {
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
|
struct power_supply_desc psy_desc;
|
||||||
struct iio_channel *channel[GAB_MAX_CHAN_TYPE];
|
struct iio_channel *channel[GAB_MAX_CHAN_TYPE];
|
||||||
struct gab_platform_data *pdata;
|
struct gab_platform_data *pdata;
|
||||||
struct delayed_work bat_work;
|
struct delayed_work bat_work;
|
||||||
|
@ -55,7 +56,7 @@ struct gab {
|
||||||
|
|
||||||
static struct gab *to_generic_bat(struct power_supply *psy)
|
static struct gab *to_generic_bat(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
return container_of(psy, struct gab, psy);
|
return power_supply_get_drvdata(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void gab_ext_power_changed(struct power_supply *psy)
|
static void gab_ext_power_changed(struct power_supply *psy)
|
||||||
|
@ -151,7 +152,7 @@ static int gab_get_property(struct power_supply *psy,
|
||||||
|
|
||||||
adc_bat = to_generic_bat(psy);
|
adc_bat = to_generic_bat(psy);
|
||||||
if (!adc_bat) {
|
if (!adc_bat) {
|
||||||
dev_err(psy->dev, "no battery infos ?!\n");
|
dev_err(&psy->dev, "no battery infos ?!\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
pdata = adc_bat->pdata;
|
pdata = adc_bat->pdata;
|
||||||
|
@ -159,7 +160,7 @@ static int gab_get_property(struct power_supply *psy,
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
gab_get_status(adc_bat);
|
val->intval = gab_get_status(adc_bat);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_CHARGE_EMPTY_DESIGN:
|
case POWER_SUPPLY_PROP_CHARGE_EMPTY_DESIGN:
|
||||||
val->intval = 0;
|
val->intval = 0;
|
||||||
|
@ -210,7 +211,7 @@ static void gab_work(struct work_struct *work)
|
||||||
pdata = adc_bat->pdata;
|
pdata = adc_bat->pdata;
|
||||||
status = adc_bat->status;
|
status = adc_bat->status;
|
||||||
|
|
||||||
is_plugged = power_supply_am_i_supplied(&adc_bat->psy);
|
is_plugged = power_supply_am_i_supplied(adc_bat->psy);
|
||||||
adc_bat->cable_plugged = is_plugged;
|
adc_bat->cable_plugged = is_plugged;
|
||||||
|
|
||||||
if (!is_plugged)
|
if (!is_plugged)
|
||||||
|
@ -221,7 +222,7 @@ static void gab_work(struct work_struct *work)
|
||||||
adc_bat->status = POWER_SUPPLY_STATUS_CHARGING;
|
adc_bat->status = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
|
|
||||||
if (status != adc_bat->status)
|
if (status != adc_bat->status)
|
||||||
power_supply_changed(&adc_bat->psy);
|
power_supply_changed(adc_bat->psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static irqreturn_t gab_charged(int irq, void *dev_id)
|
static irqreturn_t gab_charged(int irq, void *dev_id)
|
||||||
|
@ -239,7 +240,8 @@ static irqreturn_t gab_charged(int irq, void *dev_id)
|
||||||
static int gab_probe(struct platform_device *pdev)
|
static int gab_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct gab *adc_bat;
|
struct gab *adc_bat;
|
||||||
struct power_supply *psy;
|
struct power_supply_desc *psy_desc;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct gab_platform_data *pdata = pdev->dev.platform_data;
|
struct gab_platform_data *pdata = pdev->dev.platform_data;
|
||||||
enum power_supply_property *properties;
|
enum power_supply_property *properties;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
@ -252,32 +254,34 @@ static int gab_probe(struct platform_device *pdev)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
psy = &adc_bat->psy;
|
psy_cfg.drv_data = adc_bat;
|
||||||
psy->name = pdata->battery_info.name;
|
psy_desc = &adc_bat->psy_desc;
|
||||||
|
psy_desc->name = pdata->battery_info.name;
|
||||||
|
|
||||||
/* bootup default values for the battery */
|
/* bootup default values for the battery */
|
||||||
adc_bat->cable_plugged = false;
|
adc_bat->cable_plugged = false;
|
||||||
adc_bat->status = POWER_SUPPLY_STATUS_DISCHARGING;
|
adc_bat->status = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
psy->type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_desc->type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
psy->get_property = gab_get_property;
|
psy_desc->get_property = gab_get_property;
|
||||||
psy->external_power_changed = gab_ext_power_changed;
|
psy_desc->external_power_changed = gab_ext_power_changed;
|
||||||
adc_bat->pdata = pdata;
|
adc_bat->pdata = pdata;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* copying the static properties and allocating extra memory for holding
|
* copying the static properties and allocating extra memory for holding
|
||||||
* the extra configurable properties received from platform data.
|
* the extra configurable properties received from platform data.
|
||||||
*/
|
*/
|
||||||
psy->properties = kcalloc(ARRAY_SIZE(gab_props) +
|
psy_desc->properties = kcalloc(ARRAY_SIZE(gab_props) +
|
||||||
ARRAY_SIZE(gab_chan_name),
|
ARRAY_SIZE(gab_chan_name),
|
||||||
sizeof(*psy->properties), GFP_KERNEL);
|
sizeof(*psy_desc->properties),
|
||||||
if (!psy->properties) {
|
GFP_KERNEL);
|
||||||
|
if (!psy_desc->properties) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto first_mem_fail;
|
goto first_mem_fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(psy->properties, gab_props, sizeof(gab_props));
|
memcpy(psy_desc->properties, gab_props, sizeof(gab_props));
|
||||||
properties = (enum power_supply_property *)
|
properties = (enum power_supply_property *)
|
||||||
((char *)psy->properties + sizeof(gab_props));
|
((char *)psy_desc->properties + sizeof(gab_props));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* getting channel from iio and copying the battery properties
|
* getting channel from iio and copying the battery properties
|
||||||
|
@ -291,7 +295,7 @@ static int gab_probe(struct platform_device *pdev)
|
||||||
adc_bat->channel[chan] = NULL;
|
adc_bat->channel[chan] = NULL;
|
||||||
} else {
|
} else {
|
||||||
/* copying properties for supported channels only */
|
/* copying properties for supported channels only */
|
||||||
memcpy(properties + sizeof(*(psy->properties)) * index,
|
memcpy(properties + sizeof(*(psy_desc->properties)) * index,
|
||||||
&gab_dyn_props[chan],
|
&gab_dyn_props[chan],
|
||||||
sizeof(gab_dyn_props[chan]));
|
sizeof(gab_dyn_props[chan]));
|
||||||
index++;
|
index++;
|
||||||
|
@ -310,11 +314,13 @@ static int gab_probe(struct platform_device *pdev)
|
||||||
* as come channels may be not be supported by the device.So
|
* as come channels may be not be supported by the device.So
|
||||||
* we need to take care of that.
|
* we need to take care of that.
|
||||||
*/
|
*/
|
||||||
psy->num_properties = ARRAY_SIZE(gab_props) + index;
|
psy_desc->num_properties = ARRAY_SIZE(gab_props) + index;
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, psy);
|
adc_bat->psy = power_supply_register(&pdev->dev, psy_desc, &psy_cfg);
|
||||||
if (ret)
|
if (IS_ERR(adc_bat->psy)) {
|
||||||
|
ret = PTR_ERR(adc_bat->psy);
|
||||||
goto err_reg_fail;
|
goto err_reg_fail;
|
||||||
|
}
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&adc_bat->bat_work, gab_work);
|
INIT_DELAYED_WORK(&adc_bat->bat_work, gab_work);
|
||||||
|
|
||||||
|
@ -342,14 +348,14 @@ static int gab_probe(struct platform_device *pdev)
|
||||||
err_gpio:
|
err_gpio:
|
||||||
gpio_free(pdata->gpio_charge_finished);
|
gpio_free(pdata->gpio_charge_finished);
|
||||||
gpio_req_fail:
|
gpio_req_fail:
|
||||||
power_supply_unregister(psy);
|
power_supply_unregister(adc_bat->psy);
|
||||||
err_reg_fail:
|
err_reg_fail:
|
||||||
for (chan = 0; chan < ARRAY_SIZE(gab_chan_name); chan++) {
|
for (chan = 0; chan < ARRAY_SIZE(gab_chan_name); chan++) {
|
||||||
if (adc_bat->channel[chan])
|
if (adc_bat->channel[chan])
|
||||||
iio_channel_release(adc_bat->channel[chan]);
|
iio_channel_release(adc_bat->channel[chan]);
|
||||||
}
|
}
|
||||||
second_mem_fail:
|
second_mem_fail:
|
||||||
kfree(psy->properties);
|
kfree(psy_desc->properties);
|
||||||
first_mem_fail:
|
first_mem_fail:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -360,7 +366,7 @@ static int gab_remove(struct platform_device *pdev)
|
||||||
struct gab *adc_bat = platform_get_drvdata(pdev);
|
struct gab *adc_bat = platform_get_drvdata(pdev);
|
||||||
struct gab_platform_data *pdata = adc_bat->pdata;
|
struct gab_platform_data *pdata = adc_bat->pdata;
|
||||||
|
|
||||||
power_supply_unregister(&adc_bat->psy);
|
power_supply_unregister(adc_bat->psy);
|
||||||
|
|
||||||
if (gpio_is_valid(pdata->gpio_charge_finished)) {
|
if (gpio_is_valid(pdata->gpio_charge_finished)) {
|
||||||
free_irq(gpio_to_irq(pdata->gpio_charge_finished), adc_bat);
|
free_irq(gpio_to_irq(pdata->gpio_charge_finished), adc_bat);
|
||||||
|
@ -372,7 +378,7 @@ static int gab_remove(struct platform_device *pdev)
|
||||||
iio_channel_release(adc_bat->channel[chan]);
|
iio_channel_release(adc_bat->channel[chan]);
|
||||||
}
|
}
|
||||||
|
|
||||||
kfree(adc_bat->psy.properties);
|
kfree(adc_bat->psy_desc.properties);
|
||||||
cancel_delayed_work(&adc_bat->bat_work);
|
cancel_delayed_work(&adc_bat->bat_work);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,8 +30,8 @@ struct goldfish_battery_data {
|
||||||
int irq;
|
int irq;
|
||||||
spinlock_t lock;
|
spinlock_t lock;
|
||||||
|
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
struct power_supply ac;
|
struct power_supply *ac;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define GOLDFISH_BATTERY_READ(data, addr) \
|
#define GOLDFISH_BATTERY_READ(data, addr) \
|
||||||
|
@ -67,8 +67,7 @@ static int goldfish_ac_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 goldfish_battery_data *data = container_of(psy,
|
struct goldfish_battery_data *data = power_supply_get_drvdata(psy);
|
||||||
struct goldfish_battery_data, ac);
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -86,8 +85,7 @@ static int goldfish_battery_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 goldfish_battery_data *data = container_of(psy,
|
struct goldfish_battery_data *data = power_supply_get_drvdata(psy);
|
||||||
struct goldfish_battery_data, battery);
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -139,20 +137,36 @@ static irqreturn_t goldfish_battery_interrupt(int irq, void *dev_id)
|
||||||
status &= BATTERY_INT_MASK;
|
status &= BATTERY_INT_MASK;
|
||||||
|
|
||||||
if (status & BATTERY_STATUS_CHANGED)
|
if (status & BATTERY_STATUS_CHANGED)
|
||||||
power_supply_changed(&data->battery);
|
power_supply_changed(data->battery);
|
||||||
if (status & AC_STATUS_CHANGED)
|
if (status & AC_STATUS_CHANGED)
|
||||||
power_supply_changed(&data->ac);
|
power_supply_changed(data->ac);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&data->lock, irq_flags);
|
spin_unlock_irqrestore(&data->lock, irq_flags);
|
||||||
return status ? IRQ_HANDLED : IRQ_NONE;
|
return status ? IRQ_HANDLED : IRQ_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc battery_desc = {
|
||||||
|
.properties = goldfish_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(goldfish_battery_props),
|
||||||
|
.get_property = goldfish_battery_get_property,
|
||||||
|
.name = "battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc ac_desc = {
|
||||||
|
.properties = goldfish_ac_props,
|
||||||
|
.num_properties = ARRAY_SIZE(goldfish_ac_props),
|
||||||
|
.get_property = goldfish_ac_get_property,
|
||||||
|
.name = "ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
};
|
||||||
|
|
||||||
static int goldfish_battery_probe(struct platform_device *pdev)
|
static int goldfish_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
struct resource *r;
|
struct resource *r;
|
||||||
struct goldfish_battery_data *data;
|
struct goldfish_battery_data *data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL);
|
data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL);
|
||||||
if (data == NULL)
|
if (data == NULL)
|
||||||
|
@ -160,18 +174,6 @@ static int goldfish_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
spin_lock_init(&data->lock);
|
spin_lock_init(&data->lock);
|
||||||
|
|
||||||
data->battery.properties = goldfish_battery_props;
|
|
||||||
data->battery.num_properties = ARRAY_SIZE(goldfish_battery_props);
|
|
||||||
data->battery.get_property = goldfish_battery_get_property;
|
|
||||||
data->battery.name = "battery";
|
|
||||||
data->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
|
||||||
|
|
||||||
data->ac.properties = goldfish_ac_props;
|
|
||||||
data->ac.num_properties = ARRAY_SIZE(goldfish_ac_props);
|
|
||||||
data->ac.get_property = goldfish_ac_get_property;
|
|
||||||
data->ac.name = "ac";
|
|
||||||
data->ac.type = POWER_SUPPLY_TYPE_MAINS;
|
|
||||||
|
|
||||||
r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||||
if (r == NULL) {
|
if (r == NULL) {
|
||||||
dev_err(&pdev->dev, "platform_get_resource failed\n");
|
dev_err(&pdev->dev, "platform_get_resource failed\n");
|
||||||
|
@ -195,14 +197,17 @@ static int goldfish_battery_probe(struct platform_device *pdev)
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &data->ac);
|
psy_cfg.drv_data = data;
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &data->battery);
|
data->ac = power_supply_register(&pdev->dev, &ac_desc, &psy_cfg);
|
||||||
if (ret) {
|
if (IS_ERR(data->ac))
|
||||||
power_supply_unregister(&data->ac);
|
return PTR_ERR(data->ac);
|
||||||
return ret;
|
|
||||||
|
data->battery = power_supply_register(&pdev->dev, &battery_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(data->battery)) {
|
||||||
|
power_supply_unregister(data->ac);
|
||||||
|
return PTR_ERR(data->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
platform_set_drvdata(pdev, data);
|
platform_set_drvdata(pdev, data);
|
||||||
|
@ -216,8 +221,8 @@ static int goldfish_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct goldfish_battery_data *data = platform_get_drvdata(pdev);
|
struct goldfish_battery_data *data = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
power_supply_unregister(&data->battery);
|
power_supply_unregister(data->battery);
|
||||||
power_supply_unregister(&data->ac);
|
power_supply_unregister(data->ac);
|
||||||
battery_data = NULL;
|
battery_data = NULL;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,7 +32,8 @@ struct gpio_charger {
|
||||||
unsigned int irq;
|
unsigned int irq;
|
||||||
bool wakeup_enabled;
|
bool wakeup_enabled;
|
||||||
|
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
|
struct power_supply_desc charger_desc;
|
||||||
};
|
};
|
||||||
|
|
||||||
static irqreturn_t gpio_charger_irq(int irq, void *devid)
|
static irqreturn_t gpio_charger_irq(int irq, void *devid)
|
||||||
|
@ -46,7 +47,7 @@ static irqreturn_t gpio_charger_irq(int irq, void *devid)
|
||||||
|
|
||||||
static inline struct gpio_charger *psy_to_gpio_charger(struct power_supply *psy)
|
static inline struct gpio_charger *psy_to_gpio_charger(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
return container_of(psy, struct gpio_charger, charger);
|
return power_supply_get_drvdata(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int gpio_charger_get_property(struct power_supply *psy,
|
static int gpio_charger_get_property(struct power_supply *psy,
|
||||||
|
@ -127,8 +128,9 @@ struct gpio_charger_platform_data *gpio_charger_parse_dt(struct device *dev)
|
||||||
static int gpio_charger_probe(struct platform_device *pdev)
|
static int gpio_charger_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
const struct gpio_charger_platform_data *pdata = pdev->dev.platform_data;
|
const struct gpio_charger_platform_data *pdata = pdev->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct gpio_charger *gpio_charger;
|
struct gpio_charger *gpio_charger;
|
||||||
struct power_supply *charger;
|
struct power_supply_desc *charger_desc;
|
||||||
int ret;
|
int ret;
|
||||||
int irq;
|
int irq;
|
||||||
|
|
||||||
|
@ -154,16 +156,18 @@ static int gpio_charger_probe(struct platform_device *pdev)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
charger = &gpio_charger->charger;
|
charger_desc = &gpio_charger->charger_desc;
|
||||||
|
|
||||||
charger->name = pdata->name ? pdata->name : "gpio-charger";
|
charger_desc->name = pdata->name ? pdata->name : "gpio-charger";
|
||||||
charger->type = pdata->type;
|
charger_desc->type = pdata->type;
|
||||||
charger->properties = gpio_charger_properties;
|
charger_desc->properties = gpio_charger_properties;
|
||||||
charger->num_properties = ARRAY_SIZE(gpio_charger_properties);
|
charger_desc->num_properties = ARRAY_SIZE(gpio_charger_properties);
|
||||||
charger->get_property = gpio_charger_get_property;
|
charger_desc->get_property = gpio_charger_get_property;
|
||||||
charger->supplied_to = pdata->supplied_to;
|
|
||||||
charger->num_supplicants = pdata->num_supplicants;
|
psy_cfg.supplied_to = pdata->supplied_to;
|
||||||
charger->of_node = pdev->dev.of_node;
|
psy_cfg.num_supplicants = pdata->num_supplicants;
|
||||||
|
psy_cfg.of_node = pdev->dev.of_node;
|
||||||
|
psy_cfg.drv_data = gpio_charger;
|
||||||
|
|
||||||
ret = gpio_request(pdata->gpio, dev_name(&pdev->dev));
|
ret = gpio_request(pdata->gpio, dev_name(&pdev->dev));
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -178,8 +182,10 @@ static int gpio_charger_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
gpio_charger->pdata = pdata;
|
gpio_charger->pdata = pdata;
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, charger);
|
gpio_charger->charger = power_supply_register(&pdev->dev,
|
||||||
if (ret < 0) {
|
charger_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(gpio_charger->charger)) {
|
||||||
|
ret = PTR_ERR(gpio_charger->charger);
|
||||||
dev_err(&pdev->dev, "Failed to register power supply: %d\n",
|
dev_err(&pdev->dev, "Failed to register power supply: %d\n",
|
||||||
ret);
|
ret);
|
||||||
goto err_gpio_free;
|
goto err_gpio_free;
|
||||||
|
@ -189,7 +195,7 @@ static int gpio_charger_probe(struct platform_device *pdev)
|
||||||
if (irq > 0) {
|
if (irq > 0) {
|
||||||
ret = request_any_context_irq(irq, gpio_charger_irq,
|
ret = request_any_context_irq(irq, gpio_charger_irq,
|
||||||
IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING,
|
IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING,
|
||||||
dev_name(&pdev->dev), charger);
|
dev_name(&pdev->dev), gpio_charger->charger);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
dev_warn(&pdev->dev, "Failed to request irq: %d\n", ret);
|
dev_warn(&pdev->dev, "Failed to request irq: %d\n", ret);
|
||||||
else
|
else
|
||||||
|
@ -213,9 +219,9 @@ static int gpio_charger_remove(struct platform_device *pdev)
|
||||||
struct gpio_charger *gpio_charger = platform_get_drvdata(pdev);
|
struct gpio_charger *gpio_charger = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
if (gpio_charger->irq)
|
if (gpio_charger->irq)
|
||||||
free_irq(gpio_charger->irq, &gpio_charger->charger);
|
free_irq(gpio_charger->irq, gpio_charger->charger);
|
||||||
|
|
||||||
power_supply_unregister(&gpio_charger->charger);
|
power_supply_unregister(gpio_charger->charger);
|
||||||
|
|
||||||
gpio_free(gpio_charger->pdata->gpio);
|
gpio_free(gpio_charger->pdata->gpio);
|
||||||
|
|
||||||
|
@ -241,7 +247,7 @@ static int gpio_charger_resume(struct device *dev)
|
||||||
|
|
||||||
if (device_may_wakeup(dev) && gpio_charger->wakeup_enabled)
|
if (device_may_wakeup(dev) && gpio_charger->wakeup_enabled)
|
||||||
disable_irq_wake(gpio_charger->irq);
|
disable_irq_wake(gpio_charger->irq);
|
||||||
power_supply_changed(&gpio_charger->charger);
|
power_supply_changed(gpio_charger->charger);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -107,8 +107,8 @@ struct pmic_power_module_info {
|
||||||
unsigned int batt_prev_charge_full; /* in mAS */
|
unsigned int batt_prev_charge_full; /* in mAS */
|
||||||
unsigned int batt_charge_rate; /* in units per second */
|
unsigned int batt_charge_rate; /* in units per second */
|
||||||
|
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct power_supply batt;
|
struct power_supply *batt;
|
||||||
int irq; /* GPE_ID or IRQ# */
|
int irq; /* GPE_ID or IRQ# */
|
||||||
struct workqueue_struct *monitor_wqueue;
|
struct workqueue_struct *monitor_wqueue;
|
||||||
struct delayed_work monitor_battery;
|
struct delayed_work monitor_battery;
|
||||||
|
@ -404,8 +404,7 @@ static int pmic_usb_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 pmic_power_module_info *pbi = container_of(psy,
|
struct pmic_power_module_info *pbi = power_supply_get_drvdata(psy);
|
||||||
struct pmic_power_module_info, usb);
|
|
||||||
|
|
||||||
/* update pmic_power_module_info members */
|
/* update pmic_power_module_info members */
|
||||||
pmic_battery_read_status(pbi);
|
pmic_battery_read_status(pbi);
|
||||||
|
@ -444,8 +443,7 @@ static int pmic_battery_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 pmic_power_module_info *pbi = container_of(psy,
|
struct pmic_power_module_info *pbi = power_supply_get_drvdata(psy);
|
||||||
struct pmic_power_module_info, batt);
|
|
||||||
|
|
||||||
/* update pmic_power_module_info members */
|
/* update pmic_power_module_info members */
|
||||||
pmic_battery_read_status(pbi);
|
pmic_battery_read_status(pbi);
|
||||||
|
@ -640,6 +638,25 @@ static void pmic_battery_handle_intrpt(struct work_struct *work)
|
||||||
__func__);
|
__func__);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Description of power supplies
|
||||||
|
*/
|
||||||
|
static const struct power_supply_desc pmic_usb_desc = {
|
||||||
|
.name = "pmic-usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = pmic_usb_props,
|
||||||
|
.num_properties = ARRAY_SIZE(pmic_usb_props),
|
||||||
|
.get_property = pmic_usb_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc pmic_batt_desc = {
|
||||||
|
.name = "pmic-batt",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = pmic_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(pmic_battery_props),
|
||||||
|
.get_property = pmic_battery_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* pmic_battery_probe - pmic battery initialize
|
* pmic_battery_probe - pmic battery initialize
|
||||||
* @irq: pmic battery device irq
|
* @irq: pmic battery device irq
|
||||||
|
@ -653,6 +670,7 @@ static int probe(int irq, struct device *dev)
|
||||||
{
|
{
|
||||||
int retval = 0;
|
int retval = 0;
|
||||||
struct pmic_power_module_info *pbi;
|
struct pmic_power_module_info *pbi;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
dev_dbg(dev, "pmic-battery: found pmic battery device\n");
|
dev_dbg(dev, "pmic-battery: found pmic battery device\n");
|
||||||
|
|
||||||
|
@ -666,6 +684,7 @@ static int probe(int irq, struct device *dev)
|
||||||
pbi->dev = dev;
|
pbi->dev = dev;
|
||||||
pbi->irq = irq;
|
pbi->irq = irq;
|
||||||
dev_set_drvdata(dev, pbi);
|
dev_set_drvdata(dev, pbi);
|
||||||
|
psy_cfg.drv_data = pbi;
|
||||||
|
|
||||||
/* initialize all required framework before enabling interrupts */
|
/* initialize all required framework before enabling interrupts */
|
||||||
INIT_WORK(&pbi->handler, pmic_battery_handle_intrpt);
|
INIT_WORK(&pbi->handler, pmic_battery_handle_intrpt);
|
||||||
|
@ -687,16 +706,12 @@ static int probe(int irq, struct device *dev)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* register pmic-batt with power supply subsystem */
|
/* register pmic-batt with power supply subsystem */
|
||||||
pbi->batt.name = "pmic-batt";
|
pbi->batt = power_supply_register(dev, &pmic_usb_desc, &psy_cfg);
|
||||||
pbi->batt.type = POWER_SUPPLY_TYPE_BATTERY;
|
if (IS_ERR(pbi->batt)) {
|
||||||
pbi->batt.properties = pmic_battery_props;
|
|
||||||
pbi->batt.num_properties = ARRAY_SIZE(pmic_battery_props);
|
|
||||||
pbi->batt.get_property = pmic_battery_get_property;
|
|
||||||
retval = power_supply_register(dev, &pbi->batt);
|
|
||||||
if (retval) {
|
|
||||||
dev_err(dev,
|
dev_err(dev,
|
||||||
"%s(): failed to register pmic battery device with power supply subsystem\n",
|
"%s(): failed to register pmic battery device with power supply subsystem\n",
|
||||||
__func__);
|
__func__);
|
||||||
|
retval = PTR_ERR(pbi->batt);
|
||||||
goto power_reg_failed;
|
goto power_reg_failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -707,16 +722,12 @@ static int probe(int irq, struct device *dev)
|
||||||
queue_delayed_work(pbi->monitor_wqueue, &pbi->monitor_battery, HZ * 1);
|
queue_delayed_work(pbi->monitor_wqueue, &pbi->monitor_battery, HZ * 1);
|
||||||
|
|
||||||
/* register pmic-usb with power supply subsystem */
|
/* register pmic-usb with power supply subsystem */
|
||||||
pbi->usb.name = "pmic-usb";
|
pbi->usb = power_supply_register(dev, &pmic_batt_desc, &psy_cfg);
|
||||||
pbi->usb.type = POWER_SUPPLY_TYPE_USB;
|
if (IS_ERR(pbi->usb)) {
|
||||||
pbi->usb.properties = pmic_usb_props;
|
|
||||||
pbi->usb.num_properties = ARRAY_SIZE(pmic_usb_props);
|
|
||||||
pbi->usb.get_property = pmic_usb_get_property;
|
|
||||||
retval = power_supply_register(dev, &pbi->usb);
|
|
||||||
if (retval) {
|
|
||||||
dev_err(dev,
|
dev_err(dev,
|
||||||
"%s(): failed to register pmic usb device with power supply subsystem\n",
|
"%s(): failed to register pmic usb device with power supply subsystem\n",
|
||||||
__func__);
|
__func__);
|
||||||
|
retval = PTR_ERR(pbi->usb);
|
||||||
goto power_reg_failed_1;
|
goto power_reg_failed_1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -728,7 +739,7 @@ static int probe(int irq, struct device *dev)
|
||||||
return retval;
|
return retval;
|
||||||
|
|
||||||
power_reg_failed_1:
|
power_reg_failed_1:
|
||||||
power_supply_unregister(&pbi->batt);
|
power_supply_unregister(pbi->batt);
|
||||||
power_reg_failed:
|
power_reg_failed:
|
||||||
cancel_delayed_work_sync(&pbi->monitor_battery);
|
cancel_delayed_work_sync(&pbi->monitor_battery);
|
||||||
requestirq_failed:
|
requestirq_failed:
|
||||||
|
@ -762,8 +773,8 @@ static int platform_pmic_battery_remove(struct platform_device *pdev)
|
||||||
cancel_delayed_work_sync(&pbi->monitor_battery);
|
cancel_delayed_work_sync(&pbi->monitor_battery);
|
||||||
destroy_workqueue(pbi->monitor_wqueue);
|
destroy_workqueue(pbi->monitor_wqueue);
|
||||||
|
|
||||||
power_supply_unregister(&pbi->usb);
|
power_supply_unregister(pbi->usb);
|
||||||
power_supply_unregister(&pbi->batt);
|
power_supply_unregister(pbi->batt);
|
||||||
|
|
||||||
cancel_work_sync(&pbi->handler);
|
cancel_work_sync(&pbi->handler);
|
||||||
kfree(pbi);
|
kfree(pbi);
|
||||||
|
|
|
@ -93,7 +93,7 @@ static void micro_battery_work(struct work_struct *work)
|
||||||
|
|
||||||
static int get_capacity(struct power_supply *b)
|
static int get_capacity(struct power_supply *b)
|
||||||
{
|
{
|
||||||
struct micro_battery *mb = dev_get_drvdata(b->dev->parent);
|
struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
|
||||||
|
|
||||||
switch (mb->flag & 0x07) {
|
switch (mb->flag & 0x07) {
|
||||||
case MICRO_BATT_STATUS_HIGH:
|
case MICRO_BATT_STATUS_HIGH:
|
||||||
|
@ -113,7 +113,7 @@ static int get_capacity(struct power_supply *b)
|
||||||
|
|
||||||
static int get_status(struct power_supply *b)
|
static int get_status(struct power_supply *b)
|
||||||
{
|
{
|
||||||
struct micro_battery *mb = dev_get_drvdata(b->dev->parent);
|
struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
|
||||||
|
|
||||||
if (mb->flag == MICRO_BATT_STATUS_UNKNOWN)
|
if (mb->flag == MICRO_BATT_STATUS_UNKNOWN)
|
||||||
return POWER_SUPPLY_STATUS_UNKNOWN;
|
return POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
|
@ -132,7 +132,7 @@ static int micro_batt_get_property(struct power_supply *b,
|
||||||
enum power_supply_property psp,
|
enum power_supply_property psp,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct micro_battery *mb = dev_get_drvdata(b->dev->parent);
|
struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
||||||
|
@ -180,7 +180,7 @@ static int micro_ac_get_property(struct power_supply *b,
|
||||||
enum power_supply_property psp,
|
enum power_supply_property psp,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct micro_battery *mb = dev_get_drvdata(b->dev->parent);
|
struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_ONLINE:
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
|
@ -202,7 +202,7 @@ static enum power_supply_property micro_batt_power_props[] = {
|
||||||
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
POWER_SUPPLY_PROP_VOLTAGE_NOW,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply micro_batt_power = {
|
static const struct power_supply_desc micro_batt_power_desc = {
|
||||||
.name = "main-battery",
|
.name = "main-battery",
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
.properties = micro_batt_power_props,
|
.properties = micro_batt_power_props,
|
||||||
|
@ -215,7 +215,7 @@ static enum power_supply_property micro_ac_power_props[] = {
|
||||||
POWER_SUPPLY_PROP_ONLINE,
|
POWER_SUPPLY_PROP_ONLINE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply micro_ac_power = {
|
static const struct power_supply_desc micro_ac_power_desc = {
|
||||||
.name = "ac",
|
.name = "ac",
|
||||||
.type = POWER_SUPPLY_TYPE_MAINS,
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
.properties = micro_ac_power_props,
|
.properties = micro_ac_power_props,
|
||||||
|
@ -223,9 +223,12 @@ static struct power_supply micro_ac_power = {
|
||||||
.get_property = micro_ac_get_property,
|
.get_property = micro_ac_get_property,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct power_supply *micro_batt_power, *micro_ac_power;
|
||||||
|
|
||||||
static int micro_batt_probe(struct platform_device *pdev)
|
static int micro_batt_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct micro_battery *mb;
|
struct micro_battery *mb;
|
||||||
|
int ret;
|
||||||
|
|
||||||
mb = devm_kzalloc(&pdev->dev, sizeof(*mb), GFP_KERNEL);
|
mb = devm_kzalloc(&pdev->dev, sizeof(*mb), GFP_KERNEL);
|
||||||
if (!mb)
|
if (!mb)
|
||||||
|
@ -233,14 +236,36 @@ static int micro_batt_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
mb->micro = dev_get_drvdata(pdev->dev.parent);
|
mb->micro = dev_get_drvdata(pdev->dev.parent);
|
||||||
mb->wq = create_singlethread_workqueue("ipaq-battery-wq");
|
mb->wq = create_singlethread_workqueue("ipaq-battery-wq");
|
||||||
|
if (!mb->wq)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&mb->update, micro_battery_work);
|
INIT_DELAYED_WORK(&mb->update, micro_battery_work);
|
||||||
platform_set_drvdata(pdev, mb);
|
platform_set_drvdata(pdev, mb);
|
||||||
queue_delayed_work(mb->wq, &mb->update, 1);
|
queue_delayed_work(mb->wq, &mb->update, 1);
|
||||||
power_supply_register(&pdev->dev, µ_batt_power);
|
|
||||||
power_supply_register(&pdev->dev, µ_ac_power);
|
micro_batt_power = power_supply_register(&pdev->dev,
|
||||||
|
µ_batt_power_desc, NULL);
|
||||||
|
if (IS_ERR(micro_batt_power)) {
|
||||||
|
ret = PTR_ERR(micro_batt_power);
|
||||||
|
goto batt_err;
|
||||||
|
}
|
||||||
|
|
||||||
|
micro_ac_power = power_supply_register(&pdev->dev,
|
||||||
|
µ_ac_power_desc, NULL);
|
||||||
|
if (IS_ERR(micro_ac_power)) {
|
||||||
|
ret = PTR_ERR(micro_ac_power);
|
||||||
|
goto ac_err;
|
||||||
|
}
|
||||||
|
|
||||||
dev_info(&pdev->dev, "iPAQ micro battery driver\n");
|
dev_info(&pdev->dev, "iPAQ micro battery driver\n");
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
ac_err:
|
||||||
|
power_supply_unregister(micro_ac_power);
|
||||||
|
batt_err:
|
||||||
|
cancel_delayed_work_sync(&mb->update);
|
||||||
|
destroy_workqueue(mb->wq);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int micro_batt_remove(struct platform_device *pdev)
|
static int micro_batt_remove(struct platform_device *pdev)
|
||||||
|
@ -248,9 +273,10 @@ static int micro_batt_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct micro_battery *mb = platform_get_drvdata(pdev);
|
struct micro_battery *mb = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
power_supply_unregister(µ_ac_power);
|
power_supply_unregister(micro_ac_power);
|
||||||
power_supply_unregister(µ_batt_power);
|
power_supply_unregister(micro_batt_power);
|
||||||
cancel_delayed_work_sync(&mb->update);
|
cancel_delayed_work_sync(&mb->update);
|
||||||
|
destroy_workqueue(mb->wq);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,11 +57,12 @@ static u16 isp170x_id[] = {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct isp1704_charger {
|
struct isp1704_charger {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
struct usb_phy *phy;
|
struct power_supply_desc psy_desc;
|
||||||
struct notifier_block nb;
|
struct usb_phy *phy;
|
||||||
struct work_struct work;
|
struct notifier_block nb;
|
||||||
|
struct work_struct work;
|
||||||
|
|
||||||
/* properties */
|
/* properties */
|
||||||
char model[8];
|
char model[8];
|
||||||
|
@ -259,10 +260,10 @@ static void isp1704_charger_work(struct work_struct *data)
|
||||||
|
|
||||||
/* detect wall charger */
|
/* detect wall charger */
|
||||||
if (isp1704_charger_detect_dcp(isp)) {
|
if (isp1704_charger_detect_dcp(isp)) {
|
||||||
isp->psy.type = POWER_SUPPLY_TYPE_USB_DCP;
|
isp->psy_desc.type = POWER_SUPPLY_TYPE_USB_DCP;
|
||||||
isp->current_max = 1800;
|
isp->current_max = 1800;
|
||||||
} else {
|
} else {
|
||||||
isp->psy.type = POWER_SUPPLY_TYPE_USB;
|
isp->psy_desc.type = POWER_SUPPLY_TYPE_USB;
|
||||||
isp->current_max = 500;
|
isp->current_max = 500;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -271,7 +272,7 @@ static void isp1704_charger_work(struct work_struct *data)
|
||||||
usb_gadget_connect(isp->phy->otg->gadget);
|
usb_gadget_connect(isp->phy->otg->gadget);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isp->psy.type != POWER_SUPPLY_TYPE_USB_DCP) {
|
if (isp->psy_desc.type != POWER_SUPPLY_TYPE_USB_DCP) {
|
||||||
/*
|
/*
|
||||||
* Only 500mA here or high speed chirp
|
* Only 500mA here or high speed chirp
|
||||||
* handshaking may break
|
* handshaking may break
|
||||||
|
@ -280,14 +281,14 @@ static void isp1704_charger_work(struct work_struct *data)
|
||||||
isp->current_max = 500;
|
isp->current_max = 500;
|
||||||
|
|
||||||
if (isp->current_max > 100)
|
if (isp->current_max > 100)
|
||||||
isp->psy.type = POWER_SUPPLY_TYPE_USB_CDP;
|
isp->psy_desc.type = POWER_SUPPLY_TYPE_USB_CDP;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case USB_EVENT_NONE:
|
case USB_EVENT_NONE:
|
||||||
isp->online = false;
|
isp->online = false;
|
||||||
isp->present = 0;
|
isp->present = 0;
|
||||||
isp->current_max = 0;
|
isp->current_max = 0;
|
||||||
isp->psy.type = POWER_SUPPLY_TYPE_USB;
|
isp->psy_desc.type = POWER_SUPPLY_TYPE_USB;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Disable data pullups. We need to prevent the controller from
|
* Disable data pullups. We need to prevent the controller from
|
||||||
|
@ -306,7 +307,7 @@ static void isp1704_charger_work(struct work_struct *data)
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_changed(&isp->psy);
|
power_supply_changed(isp->psy);
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&lock);
|
mutex_unlock(&lock);
|
||||||
}
|
}
|
||||||
|
@ -326,8 +327,7 @@ static int isp1704_charger_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 isp1704_charger *isp =
|
struct isp1704_charger *isp = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct isp1704_charger, psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_PRESENT:
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
|
@ -403,6 +403,7 @@ static int isp1704_charger_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct isp1704_charger *isp;
|
struct isp1704_charger *isp;
|
||||||
int ret = -ENODEV;
|
int ret = -ENODEV;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
struct isp1704_charger_data *pdata = dev_get_platdata(&pdev->dev);
|
struct isp1704_charger_data *pdata = dev_get_platdata(&pdev->dev);
|
||||||
struct device_node *np = pdev->dev.of_node;
|
struct device_node *np = pdev->dev.of_node;
|
||||||
|
@ -454,15 +455,19 @@ static int isp1704_charger_probe(struct platform_device *pdev)
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto fail1;
|
goto fail1;
|
||||||
|
|
||||||
isp->psy.name = "isp1704";
|
isp->psy_desc.name = "isp1704";
|
||||||
isp->psy.type = POWER_SUPPLY_TYPE_USB;
|
isp->psy_desc.type = POWER_SUPPLY_TYPE_USB;
|
||||||
isp->psy.properties = power_props;
|
isp->psy_desc.properties = power_props;
|
||||||
isp->psy.num_properties = ARRAY_SIZE(power_props);
|
isp->psy_desc.num_properties = ARRAY_SIZE(power_props);
|
||||||
isp->psy.get_property = isp1704_charger_get_property;
|
isp->psy_desc.get_property = isp1704_charger_get_property;
|
||||||
|
|
||||||
ret = power_supply_register(isp->dev, &isp->psy);
|
psy_cfg.drv_data = isp;
|
||||||
if (ret)
|
|
||||||
|
isp->psy = power_supply_register(isp->dev, &isp->psy_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(isp->psy)) {
|
||||||
|
ret = PTR_ERR(isp->psy);
|
||||||
goto fail1;
|
goto fail1;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* REVISIT: using work in order to allow the usb notifications to be
|
* REVISIT: using work in order to allow the usb notifications to be
|
||||||
|
@ -498,7 +503,7 @@ static int isp1704_charger_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
fail2:
|
fail2:
|
||||||
power_supply_unregister(&isp->psy);
|
power_supply_unregister(isp->psy);
|
||||||
fail1:
|
fail1:
|
||||||
isp1704_charger_set_power(isp, 0);
|
isp1704_charger_set_power(isp, 0);
|
||||||
fail0:
|
fail0:
|
||||||
|
@ -512,7 +517,7 @@ static int isp1704_charger_remove(struct platform_device *pdev)
|
||||||
struct isp1704_charger *isp = platform_get_drvdata(pdev);
|
struct isp1704_charger *isp = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
usb_unregister_notifier(isp->phy, &isp->nb);
|
usb_unregister_notifier(isp->phy, &isp->nb);
|
||||||
power_supply_unregister(&isp->psy);
|
power_supply_unregister(isp->psy);
|
||||||
isp1704_charger_set_power(isp, 0);
|
isp1704_charger_set_power(isp, 0);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -46,7 +46,8 @@ struct jz_battery {
|
||||||
|
|
||||||
struct completion read_completion;
|
struct completion read_completion;
|
||||||
|
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
|
struct power_supply_desc battery_desc;
|
||||||
struct delayed_work work;
|
struct delayed_work work;
|
||||||
|
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
|
@ -54,7 +55,7 @@ struct jz_battery {
|
||||||
|
|
||||||
static inline struct jz_battery *psy_to_jz_battery(struct power_supply *psy)
|
static inline struct jz_battery *psy_to_jz_battery(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
return container_of(psy, struct jz_battery, battery);
|
return power_supply_get_drvdata(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static irqreturn_t jz_battery_irq_handler(int irq, void *devid)
|
static irqreturn_t jz_battery_irq_handler(int irq, void *devid)
|
||||||
|
@ -213,7 +214,7 @@ static void jz_battery_update(struct jz_battery *jz_battery)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (has_changed)
|
if (has_changed)
|
||||||
power_supply_changed(&jz_battery->battery);
|
power_supply_changed(jz_battery->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum power_supply_property jz_battery_properties[] = {
|
static enum power_supply_property jz_battery_properties[] = {
|
||||||
|
@ -242,8 +243,9 @@ static int jz_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct jz_battery_platform_data *pdata = pdev->dev.parent->platform_data;
|
struct jz_battery_platform_data *pdata = pdev->dev.parent->platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct jz_battery *jz_battery;
|
struct jz_battery *jz_battery;
|
||||||
struct power_supply *battery;
|
struct power_supply_desc *battery_desc;
|
||||||
struct resource *mem;
|
struct resource *mem;
|
||||||
|
|
||||||
if (!pdata) {
|
if (!pdata) {
|
||||||
|
@ -271,14 +273,17 @@ static int jz_battery_probe(struct platform_device *pdev)
|
||||||
if (IS_ERR(jz_battery->base))
|
if (IS_ERR(jz_battery->base))
|
||||||
return PTR_ERR(jz_battery->base);
|
return PTR_ERR(jz_battery->base);
|
||||||
|
|
||||||
battery = &jz_battery->battery;
|
battery_desc = &jz_battery->battery_desc;
|
||||||
battery->name = pdata->info.name;
|
battery_desc->name = pdata->info.name;
|
||||||
battery->type = POWER_SUPPLY_TYPE_BATTERY;
|
battery_desc->type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
battery->properties = jz_battery_properties;
|
battery_desc->properties = jz_battery_properties;
|
||||||
battery->num_properties = ARRAY_SIZE(jz_battery_properties);
|
battery_desc->num_properties = ARRAY_SIZE(jz_battery_properties);
|
||||||
battery->get_property = jz_battery_get_property;
|
battery_desc->get_property = jz_battery_get_property;
|
||||||
battery->external_power_changed = jz_battery_external_power_changed;
|
battery_desc->external_power_changed =
|
||||||
battery->use_for_apm = 1;
|
jz_battery_external_power_changed;
|
||||||
|
battery_desc->use_for_apm = 1;
|
||||||
|
|
||||||
|
psy_cfg.drv_data = jz_battery;
|
||||||
|
|
||||||
jz_battery->pdata = pdata;
|
jz_battery->pdata = pdata;
|
||||||
jz_battery->pdev = pdev;
|
jz_battery->pdev = pdev;
|
||||||
|
@ -330,9 +335,11 @@ static int jz_battery_probe(struct platform_device *pdev)
|
||||||
else
|
else
|
||||||
jz4740_adc_set_config(pdev->dev.parent, JZ_ADC_CONFIG_BAT_MB, 0);
|
jz4740_adc_set_config(pdev->dev.parent, JZ_ADC_CONFIG_BAT_MB, 0);
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &jz_battery->battery);
|
jz_battery->battery = power_supply_register(&pdev->dev, battery_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(jz_battery->battery)) {
|
||||||
dev_err(&pdev->dev, "power supply battery register failed.\n");
|
dev_err(&pdev->dev, "power supply battery register failed.\n");
|
||||||
|
ret = PTR_ERR(jz_battery->battery);
|
||||||
goto err_free_charge_irq;
|
goto err_free_charge_irq;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -364,7 +371,7 @@ static int jz_battery_remove(struct platform_device *pdev)
|
||||||
gpio_free(jz_battery->pdata->gpio_charge);
|
gpio_free(jz_battery->pdata->gpio_charge);
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_unregister(&jz_battery->battery);
|
power_supply_unregister(jz_battery->battery);
|
||||||
|
|
||||||
free_irq(jz_battery->irq, jz_battery);
|
free_irq(jz_battery->irq, jz_battery);
|
||||||
|
|
||||||
|
|
|
@ -80,9 +80,9 @@ enum lp8727_die_temp {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct lp8727_psy {
|
struct lp8727_psy {
|
||||||
struct power_supply ac;
|
struct power_supply *ac;
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct power_supply batt;
|
struct power_supply *batt;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct lp8727_chg {
|
struct lp8727_chg {
|
||||||
|
@ -242,9 +242,9 @@ static void lp8727_delayed_func(struct work_struct *_work)
|
||||||
lp8727_id_detection(pchg, idno, vbus);
|
lp8727_id_detection(pchg, idno, vbus);
|
||||||
lp8727_enable_chgdet(pchg);
|
lp8727_enable_chgdet(pchg);
|
||||||
|
|
||||||
power_supply_changed(&pchg->psy->ac);
|
power_supply_changed(pchg->psy->ac);
|
||||||
power_supply_changed(&pchg->psy->usb);
|
power_supply_changed(pchg->psy->usb);
|
||||||
power_supply_changed(&pchg->psy->batt);
|
power_supply_changed(pchg->psy->batt);
|
||||||
}
|
}
|
||||||
|
|
||||||
static irqreturn_t lp8727_isr_func(int irq, void *ptr)
|
static irqreturn_t lp8727_isr_func(int irq, void *ptr)
|
||||||
|
@ -311,12 +311,12 @@ static int lp8727_charger_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 lp8727_chg *pchg = dev_get_drvdata(psy->dev->parent);
|
struct lp8727_chg *pchg = dev_get_drvdata(psy->dev.parent);
|
||||||
|
|
||||||
if (psp != POWER_SUPPLY_PROP_ONLINE)
|
if (psp != POWER_SUPPLY_PROP_ONLINE)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
val->intval = lp8727_is_charger_attached(psy->name, pchg->devid);
|
val->intval = lp8727_is_charger_attached(psy->desc->name, pchg->devid);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -337,14 +337,14 @@ static int lp8727_battery_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 lp8727_chg *pchg = dev_get_drvdata(psy->dev->parent);
|
struct lp8727_chg *pchg = dev_get_drvdata(psy->dev.parent);
|
||||||
struct lp8727_platform_data *pdata = pchg->pdata;
|
struct lp8727_platform_data *pdata = pchg->pdata;
|
||||||
enum lp8727_die_temp temp;
|
enum lp8727_die_temp temp;
|
||||||
u8 read;
|
u8 read;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
if (!lp8727_is_charger_attached(psy->name, pchg->devid)) {
|
if (!lp8727_is_charger_attached(psy->desc->name, pchg->devid)) {
|
||||||
val->intval = POWER_SUPPLY_STATUS_DISCHARGING;
|
val->intval = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -400,13 +400,13 @@ static int lp8727_battery_get_property(struct power_supply *psy,
|
||||||
|
|
||||||
static void lp8727_charger_changed(struct power_supply *psy)
|
static void lp8727_charger_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct lp8727_chg *pchg = dev_get_drvdata(psy->dev->parent);
|
struct lp8727_chg *pchg = dev_get_drvdata(psy->dev.parent);
|
||||||
u8 eoc_level;
|
u8 eoc_level;
|
||||||
u8 ichg;
|
u8 ichg;
|
||||||
u8 val;
|
u8 val;
|
||||||
|
|
||||||
/* skip if no charger exists */
|
/* skip if no charger exists */
|
||||||
if (!lp8727_is_charger_attached(psy->name, pchg->devid))
|
if (!lp8727_is_charger_attached(psy->desc->name, pchg->devid))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* update charging parameters */
|
/* update charging parameters */
|
||||||
|
@ -418,8 +418,34 @@ static void lp8727_charger_changed(struct power_supply *psy)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc lp8727_ac_desc = {
|
||||||
|
.name = "ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = lp8727_charger_prop,
|
||||||
|
.num_properties = ARRAY_SIZE(lp8727_charger_prop),
|
||||||
|
.get_property = lp8727_charger_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc lp8727_usb_desc = {
|
||||||
|
.name = "usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = lp8727_charger_prop,
|
||||||
|
.num_properties = ARRAY_SIZE(lp8727_charger_prop),
|
||||||
|
.get_property = lp8727_charger_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc lp8727_batt_desc = {
|
||||||
|
.name = "main_batt",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = lp8727_battery_prop,
|
||||||
|
.num_properties = ARRAY_SIZE(lp8727_battery_prop),
|
||||||
|
.get_property = lp8727_battery_get_property,
|
||||||
|
.external_power_changed = lp8727_charger_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static int lp8727_register_psy(struct lp8727_chg *pchg)
|
static int lp8727_register_psy(struct lp8727_chg *pchg)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {}; /* Only for ac and usb */
|
||||||
struct lp8727_psy *psy;
|
struct lp8727_psy *psy;
|
||||||
|
|
||||||
psy = devm_kzalloc(pchg->dev, sizeof(*psy), GFP_KERNEL);
|
psy = devm_kzalloc(pchg->dev, sizeof(*psy), GFP_KERNEL);
|
||||||
|
@ -428,44 +454,28 @@ static int lp8727_register_psy(struct lp8727_chg *pchg)
|
||||||
|
|
||||||
pchg->psy = psy;
|
pchg->psy = psy;
|
||||||
|
|
||||||
psy->ac.name = "ac";
|
psy_cfg.supplied_to = battery_supplied_to;
|
||||||
psy->ac.type = POWER_SUPPLY_TYPE_MAINS;
|
psy_cfg.num_supplicants = ARRAY_SIZE(battery_supplied_to);
|
||||||
psy->ac.properties = lp8727_charger_prop;
|
|
||||||
psy->ac.num_properties = ARRAY_SIZE(lp8727_charger_prop);
|
|
||||||
psy->ac.get_property = lp8727_charger_get_property;
|
|
||||||
psy->ac.supplied_to = battery_supplied_to;
|
|
||||||
psy->ac.num_supplicants = ARRAY_SIZE(battery_supplied_to);
|
|
||||||
|
|
||||||
if (power_supply_register(pchg->dev, &psy->ac))
|
psy->ac = power_supply_register(pchg->dev, &lp8727_ac_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(psy->ac))
|
||||||
goto err_psy_ac;
|
goto err_psy_ac;
|
||||||
|
|
||||||
psy->usb.name = "usb";
|
psy->usb = power_supply_register(pchg->dev, &lp8727_usb_desc,
|
||||||
psy->usb.type = POWER_SUPPLY_TYPE_USB;
|
&psy_cfg);
|
||||||
psy->usb.properties = lp8727_charger_prop;
|
if (IS_ERR(psy->usb))
|
||||||
psy->usb.num_properties = ARRAY_SIZE(lp8727_charger_prop);
|
|
||||||
psy->usb.get_property = lp8727_charger_get_property;
|
|
||||||
psy->usb.supplied_to = battery_supplied_to;
|
|
||||||
psy->usb.num_supplicants = ARRAY_SIZE(battery_supplied_to);
|
|
||||||
|
|
||||||
if (power_supply_register(pchg->dev, &psy->usb))
|
|
||||||
goto err_psy_usb;
|
goto err_psy_usb;
|
||||||
|
|
||||||
psy->batt.name = "main_batt";
|
psy->batt = power_supply_register(pchg->dev, &lp8727_batt_desc, NULL);
|
||||||
psy->batt.type = POWER_SUPPLY_TYPE_BATTERY;
|
if (IS_ERR(psy->batt))
|
||||||
psy->batt.properties = lp8727_battery_prop;
|
|
||||||
psy->batt.num_properties = ARRAY_SIZE(lp8727_battery_prop);
|
|
||||||
psy->batt.get_property = lp8727_battery_get_property;
|
|
||||||
psy->batt.external_power_changed = lp8727_charger_changed;
|
|
||||||
|
|
||||||
if (power_supply_register(pchg->dev, &psy->batt))
|
|
||||||
goto err_psy_batt;
|
goto err_psy_batt;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_psy_batt:
|
err_psy_batt:
|
||||||
power_supply_unregister(&psy->usb);
|
power_supply_unregister(psy->usb);
|
||||||
err_psy_usb:
|
err_psy_usb:
|
||||||
power_supply_unregister(&psy->ac);
|
power_supply_unregister(psy->ac);
|
||||||
err_psy_ac:
|
err_psy_ac:
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
}
|
}
|
||||||
|
@ -477,9 +487,9 @@ static void lp8727_unregister_psy(struct lp8727_chg *pchg)
|
||||||
if (!psy)
|
if (!psy)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
power_supply_unregister(&psy->ac);
|
power_supply_unregister(psy->ac);
|
||||||
power_supply_unregister(&psy->usb);
|
power_supply_unregister(psy->usb);
|
||||||
power_supply_unregister(&psy->batt);
|
power_supply_unregister(psy->batt);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_OF
|
#ifdef CONFIG_OF
|
||||||
|
|
|
@ -105,8 +105,8 @@ struct lp8788_chg_irq {
|
||||||
*/
|
*/
|
||||||
struct lp8788_charger {
|
struct lp8788_charger {
|
||||||
struct lp8788 *lp;
|
struct lp8788 *lp;
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
struct work_struct charger_work;
|
struct work_struct charger_work;
|
||||||
struct iio_channel *chan[LP8788_NUM_CHG_ADC];
|
struct iio_channel *chan[LP8788_NUM_CHG_ADC];
|
||||||
struct lp8788_chg_irq irqs[LP8788_MAX_CHG_IRQS];
|
struct lp8788_chg_irq irqs[LP8788_MAX_CHG_IRQS];
|
||||||
|
@ -148,7 +148,7 @@ static int lp8788_charger_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 lp8788_charger *pchg = dev_get_drvdata(psy->dev->parent);
|
struct lp8788_charger *pchg = dev_get_drvdata(psy->dev.parent);
|
||||||
u8 read;
|
u8 read;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -337,7 +337,7 @@ static int lp8788_battery_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 lp8788_charger *pchg = dev_get_drvdata(psy->dev->parent);
|
struct lp8788_charger *pchg = dev_get_drvdata(psy->dev.parent);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -397,36 +397,50 @@ static int lp8788_update_charger_params(struct platform_device *pdev,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc lp8788_psy_charger_desc = {
|
||||||
|
.name = LP8788_CHARGER_NAME,
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = lp8788_charger_prop,
|
||||||
|
.num_properties = ARRAY_SIZE(lp8788_charger_prop),
|
||||||
|
.get_property = lp8788_charger_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc lp8788_psy_battery_desc = {
|
||||||
|
.name = LP8788_BATTERY_NAME,
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = lp8788_battery_prop,
|
||||||
|
.num_properties = ARRAY_SIZE(lp8788_battery_prop),
|
||||||
|
.get_property = lp8788_battery_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
static int lp8788_psy_register(struct platform_device *pdev,
|
static int lp8788_psy_register(struct platform_device *pdev,
|
||||||
struct lp8788_charger *pchg)
|
struct lp8788_charger *pchg)
|
||||||
{
|
{
|
||||||
pchg->charger.name = LP8788_CHARGER_NAME;
|
struct power_supply_config charger_cfg = {};
|
||||||
pchg->charger.type = POWER_SUPPLY_TYPE_MAINS;
|
|
||||||
pchg->charger.properties = lp8788_charger_prop;
|
|
||||||
pchg->charger.num_properties = ARRAY_SIZE(lp8788_charger_prop);
|
|
||||||
pchg->charger.get_property = lp8788_charger_get_property;
|
|
||||||
pchg->charger.supplied_to = battery_supplied_to;
|
|
||||||
pchg->charger.num_supplicants = ARRAY_SIZE(battery_supplied_to);
|
|
||||||
|
|
||||||
if (power_supply_register(&pdev->dev, &pchg->charger))
|
charger_cfg.supplied_to = battery_supplied_to;
|
||||||
|
charger_cfg.num_supplicants = ARRAY_SIZE(battery_supplied_to);
|
||||||
|
|
||||||
|
pchg->charger = power_supply_register(&pdev->dev,
|
||||||
|
&lp8788_psy_charger_desc,
|
||||||
|
&charger_cfg);
|
||||||
|
if (IS_ERR(pchg->charger))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
|
|
||||||
pchg->battery.name = LP8788_BATTERY_NAME;
|
pchg->battery = power_supply_register(&pdev->dev,
|
||||||
pchg->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
&lp8788_psy_battery_desc, NULL);
|
||||||
pchg->battery.properties = lp8788_battery_prop;
|
if (IS_ERR(pchg->battery)) {
|
||||||
pchg->battery.num_properties = ARRAY_SIZE(lp8788_battery_prop);
|
power_supply_unregister(pchg->charger);
|
||||||
pchg->battery.get_property = lp8788_battery_get_property;
|
|
||||||
|
|
||||||
if (power_supply_register(&pdev->dev, &pchg->battery))
|
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void lp8788_psy_unregister(struct lp8788_charger *pchg)
|
static void lp8788_psy_unregister(struct lp8788_charger *pchg)
|
||||||
{
|
{
|
||||||
power_supply_unregister(&pchg->battery);
|
power_supply_unregister(pchg->battery);
|
||||||
power_supply_unregister(&pchg->charger);
|
power_supply_unregister(pchg->charger);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void lp8788_charger_event(struct work_struct *work)
|
static void lp8788_charger_event(struct work_struct *work)
|
||||||
|
@ -470,8 +484,8 @@ static irqreturn_t lp8788_charger_irq_thread(int virq, void *ptr)
|
||||||
case LP8788_INT_EOC:
|
case LP8788_INT_EOC:
|
||||||
case LP8788_INT_BATT_LOW:
|
case LP8788_INT_BATT_LOW:
|
||||||
case LP8788_INT_NO_BATT:
|
case LP8788_INT_NO_BATT:
|
||||||
power_supply_changed(&pchg->charger);
|
power_supply_changed(pchg->charger);
|
||||||
power_supply_changed(&pchg->battery);
|
power_supply_changed(pchg->battery);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -59,7 +59,8 @@ enum ltc294x_reg {
|
||||||
|
|
||||||
struct ltc294x_info {
|
struct ltc294x_info {
|
||||||
struct i2c_client *client; /* I2C Client pointer */
|
struct i2c_client *client; /* I2C Client pointer */
|
||||||
struct power_supply supply; /* Supply pointer */
|
struct power_supply *supply; /* Supply pointer */
|
||||||
|
struct power_supply_desc supply_desc; /* Supply description */
|
||||||
struct delayed_work work; /* Work scheduler */
|
struct delayed_work work; /* Work scheduler */
|
||||||
int num_regs; /* Number of registers (chip type) */
|
int num_regs; /* Number of registers (chip type) */
|
||||||
int id; /* Identifier of ltc294x chip */
|
int id; /* Identifier of ltc294x chip */
|
||||||
|
@ -294,8 +295,7 @@ static int ltc294x_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property prop,
|
enum power_supply_property prop,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct ltc294x_info *info =
|
struct ltc294x_info *info = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct ltc294x_info, supply);
|
|
||||||
|
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
case POWER_SUPPLY_PROP_CHARGE_NOW:
|
case POWER_SUPPLY_PROP_CHARGE_NOW:
|
||||||
|
@ -317,8 +317,7 @@ static int ltc294x_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)
|
||||||
{
|
{
|
||||||
struct ltc294x_info *info =
|
struct ltc294x_info *info = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct ltc294x_info, supply);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_CHARGE_NOW:
|
case POWER_SUPPLY_PROP_CHARGE_NOW:
|
||||||
|
@ -345,7 +344,7 @@ static void ltc294x_update(struct ltc294x_info *info)
|
||||||
|
|
||||||
if (charge != info->charge) {
|
if (charge != info->charge) {
|
||||||
info->charge = charge;
|
info->charge = charge;
|
||||||
power_supply_changed(&info->supply);
|
power_supply_changed(info->supply);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -371,8 +370,8 @@ static int ltc294x_i2c_remove(struct i2c_client *client)
|
||||||
struct ltc294x_info *info = i2c_get_clientdata(client);
|
struct ltc294x_info *info = i2c_get_clientdata(client);
|
||||||
|
|
||||||
cancel_delayed_work(&info->work);
|
cancel_delayed_work(&info->work);
|
||||||
power_supply_unregister(&info->supply);
|
power_supply_unregister(info->supply);
|
||||||
kfree(info->supply.name);
|
kfree(info->supply_desc.name);
|
||||||
mutex_lock(<c294x_lock);
|
mutex_lock(<c294x_lock);
|
||||||
idr_remove(<c294x_id, info->id);
|
idr_remove(<c294x_id, info->id);
|
||||||
mutex_unlock(<c294x_lock);
|
mutex_unlock(<c294x_lock);
|
||||||
|
@ -382,6 +381,7 @@ static int ltc294x_i2c_remove(struct i2c_client *client)
|
||||||
static int ltc294x_i2c_probe(struct i2c_client *client,
|
static int ltc294x_i2c_probe(struct i2c_client *client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct ltc294x_info *info;
|
struct ltc294x_info *info;
|
||||||
int ret;
|
int ret;
|
||||||
int num;
|
int num;
|
||||||
|
@ -406,8 +406,9 @@ static int ltc294x_i2c_probe(struct i2c_client *client,
|
||||||
i2c_set_clientdata(client, info);
|
i2c_set_clientdata(client, info);
|
||||||
|
|
||||||
info->num_regs = id->driver_data;
|
info->num_regs = id->driver_data;
|
||||||
info->supply.name = kasprintf(GFP_KERNEL, "%s-%d", client->name, num);
|
info->supply_desc.name = kasprintf(GFP_KERNEL, "%s-%d", client->name,
|
||||||
if (!info->supply.name) {
|
num);
|
||||||
|
if (!info->supply_desc.name) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto fail_name;
|
goto fail_name;
|
||||||
}
|
}
|
||||||
|
@ -440,30 +441,32 @@ static int ltc294x_i2c_probe(struct i2c_client *client,
|
||||||
} else {
|
} else {
|
||||||
if (prescaler_exp > LTC2941_MAX_PRESCALER_EXP)
|
if (prescaler_exp > LTC2941_MAX_PRESCALER_EXP)
|
||||||
prescaler_exp = LTC2941_MAX_PRESCALER_EXP;
|
prescaler_exp = LTC2941_MAX_PRESCALER_EXP;
|
||||||
info->Qlsb = ((58 * 50000) / r_sense) /
|
info->Qlsb = ((85 * 50000) / r_sense) /
|
||||||
(128 / (1 << prescaler_exp));
|
(128 / (1 << prescaler_exp));
|
||||||
}
|
}
|
||||||
|
|
||||||
info->client = client;
|
info->client = client;
|
||||||
info->id = num;
|
info->id = num;
|
||||||
info->supply.type = POWER_SUPPLY_TYPE_BATTERY;
|
info->supply_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
info->supply.properties = ltc294x_properties;
|
info->supply_desc.properties = ltc294x_properties;
|
||||||
if (info->num_regs >= LTC294X_REG_TEMPERATURE_LSB)
|
if (info->num_regs >= LTC294X_REG_TEMPERATURE_LSB)
|
||||||
info->supply.num_properties =
|
info->supply_desc.num_properties =
|
||||||
ARRAY_SIZE(ltc294x_properties);
|
ARRAY_SIZE(ltc294x_properties);
|
||||||
else if (info->num_regs >= LTC294X_REG_CURRENT_LSB)
|
else if (info->num_regs >= LTC294X_REG_CURRENT_LSB)
|
||||||
info->supply.num_properties =
|
info->supply_desc.num_properties =
|
||||||
ARRAY_SIZE(ltc294x_properties) - 1;
|
ARRAY_SIZE(ltc294x_properties) - 1;
|
||||||
else if (info->num_regs >= LTC294X_REG_VOLTAGE_LSB)
|
else if (info->num_regs >= LTC294X_REG_VOLTAGE_LSB)
|
||||||
info->supply.num_properties =
|
info->supply_desc.num_properties =
|
||||||
ARRAY_SIZE(ltc294x_properties) - 2;
|
ARRAY_SIZE(ltc294x_properties) - 2;
|
||||||
else
|
else
|
||||||
info->supply.num_properties =
|
info->supply_desc.num_properties =
|
||||||
ARRAY_SIZE(ltc294x_properties) - 3;
|
ARRAY_SIZE(ltc294x_properties) - 3;
|
||||||
info->supply.get_property = ltc294x_get_property;
|
info->supply_desc.get_property = ltc294x_get_property;
|
||||||
info->supply.set_property = ltc294x_set_property;
|
info->supply_desc.set_property = ltc294x_set_property;
|
||||||
info->supply.property_is_writeable = ltc294x_property_is_writeable;
|
info->supply_desc.property_is_writeable = ltc294x_property_is_writeable;
|
||||||
info->supply.external_power_changed = NULL;
|
info->supply_desc.external_power_changed = NULL;
|
||||||
|
|
||||||
|
psy_cfg.drv_data = info;
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&info->work, ltc294x_work);
|
INIT_DELAYED_WORK(&info->work, ltc294x_work);
|
||||||
|
|
||||||
|
@ -473,9 +476,11 @@ static int ltc294x_i2c_probe(struct i2c_client *client,
|
||||||
goto fail_comm;
|
goto fail_comm;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&client->dev, &info->supply);
|
info->supply = power_supply_register(&client->dev, &info->supply_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(info->supply)) {
|
||||||
dev_err(&client->dev, "failed to register ltc2941\n");
|
dev_err(&client->dev, "failed to register ltc2941\n");
|
||||||
|
ret = PTR_ERR(info->supply);
|
||||||
goto fail_register;
|
goto fail_register;
|
||||||
} else {
|
} else {
|
||||||
schedule_delayed_work(&info->work, LTC294X_WORK_DELAY * HZ);
|
schedule_delayed_work(&info->work, LTC294X_WORK_DELAY * HZ);
|
||||||
|
@ -484,7 +489,7 @@ static int ltc294x_i2c_probe(struct i2c_client *client,
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fail_register:
|
fail_register:
|
||||||
kfree(info->supply.name);
|
kfree(info->supply_desc.name);
|
||||||
fail_comm:
|
fail_comm:
|
||||||
fail_name:
|
fail_name:
|
||||||
fail_info:
|
fail_info:
|
||||||
|
|
|
@ -22,12 +22,9 @@
|
||||||
#include <linux/mfd/max14577.h>
|
#include <linux/mfd/max14577.h>
|
||||||
|
|
||||||
struct max14577_charger {
|
struct max14577_charger {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct max14577 *max14577;
|
struct max14577 *max14577;
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
|
|
||||||
unsigned int charging_state;
|
|
||||||
unsigned int battery_state;
|
|
||||||
|
|
||||||
struct max14577_charger_platform_data *pdata;
|
struct max14577_charger_platform_data *pdata;
|
||||||
};
|
};
|
||||||
|
@ -57,10 +54,10 @@ static enum max14577_muic_charger_type maxim_get_charger_type(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int max14577_get_charger_state(struct max14577_charger *chg)
|
static int max14577_get_charger_state(struct max14577_charger *chg, int *val)
|
||||||
{
|
{
|
||||||
struct regmap *rmap = chg->max14577->regmap;
|
struct regmap *rmap = chg->max14577->regmap;
|
||||||
int state = POWER_SUPPLY_STATUS_DISCHARGING;
|
int ret;
|
||||||
u8 reg_data;
|
u8 reg_data;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -74,23 +71,32 @@ static int max14577_get_charger_state(struct max14577_charger *chg)
|
||||||
* - handle properly dead-battery charging (respect timer)
|
* - handle properly dead-battery charging (respect timer)
|
||||||
* - handle timers (fast-charge and prequal) /MBCCHGERR/
|
* - handle timers (fast-charge and prequal) /MBCCHGERR/
|
||||||
*/
|
*/
|
||||||
max14577_read_reg(rmap, MAX14577_CHG_REG_CHG_CTRL2, ®_data);
|
ret = max14577_read_reg(rmap, MAX14577_CHG_REG_CHG_CTRL2, ®_data);
|
||||||
if ((reg_data & CHGCTRL2_MBCHOSTEN_MASK) == 0)
|
if (ret < 0)
|
||||||
goto state_set;
|
goto out;
|
||||||
|
|
||||||
|
if ((reg_data & CHGCTRL2_MBCHOSTEN_MASK) == 0) {
|
||||||
|
*val = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = max14577_read_reg(rmap, MAX14577_CHG_REG_STATUS3, ®_data);
|
||||||
|
if (ret < 0)
|
||||||
|
goto out;
|
||||||
|
|
||||||
max14577_read_reg(rmap, MAX14577_CHG_REG_STATUS3, ®_data);
|
|
||||||
if (reg_data & STATUS3_CGMBC_MASK) {
|
if (reg_data & STATUS3_CGMBC_MASK) {
|
||||||
/* Charger or USB-cable is connected */
|
/* Charger or USB-cable is connected */
|
||||||
if (reg_data & STATUS3_EOC_MASK)
|
if (reg_data & STATUS3_EOC_MASK)
|
||||||
state = POWER_SUPPLY_STATUS_FULL;
|
*val = POWER_SUPPLY_STATUS_FULL;
|
||||||
else
|
else
|
||||||
state = POWER_SUPPLY_STATUS_CHARGING;
|
*val = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
goto state_set;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
state_set:
|
*val = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
chg->charging_state = state;
|
|
||||||
return state;
|
out:
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -98,8 +104,10 @@ state_set:
|
||||||
* - POWER_SUPPLY_CHARGE_TYPE_NONE
|
* - POWER_SUPPLY_CHARGE_TYPE_NONE
|
||||||
* - POWER_SUPPLY_CHARGE_TYPE_FAST
|
* - POWER_SUPPLY_CHARGE_TYPE_FAST
|
||||||
*/
|
*/
|
||||||
static int max14577_get_charge_type(struct max14577_charger *chg)
|
static int max14577_get_charge_type(struct max14577_charger *chg, int *val)
|
||||||
{
|
{
|
||||||
|
int ret, charging;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* TODO: CHARGE_TYPE_TRICKLE (VCHGR_RC or EOC)?
|
* TODO: CHARGE_TYPE_TRICKLE (VCHGR_RC or EOC)?
|
||||||
* As spec says:
|
* As spec says:
|
||||||
|
@ -108,18 +116,29 @@ static int max14577_get_charge_type(struct max14577_charger *chg)
|
||||||
* top-off timer starts. The device continues to trickle
|
* top-off timer starts. The device continues to trickle
|
||||||
* charge the battery until the top-off timer runs out."
|
* charge the battery until the top-off timer runs out."
|
||||||
*/
|
*/
|
||||||
if (max14577_get_charger_state(chg) == POWER_SUPPLY_STATUS_CHARGING)
|
ret = max14577_get_charger_state(chg, &charging);
|
||||||
return POWER_SUPPLY_CHARGE_TYPE_FAST;
|
if (ret < 0)
|
||||||
return POWER_SUPPLY_CHARGE_TYPE_NONE;
|
return ret;
|
||||||
|
|
||||||
|
if (charging == POWER_SUPPLY_STATUS_CHARGING)
|
||||||
|
*val = POWER_SUPPLY_CHARGE_TYPE_FAST;
|
||||||
|
else
|
||||||
|
*val = POWER_SUPPLY_CHARGE_TYPE_NONE;
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int max14577_get_online(struct max14577_charger *chg)
|
static int max14577_get_online(struct max14577_charger *chg, int *val)
|
||||||
{
|
{
|
||||||
struct regmap *rmap = chg->max14577->regmap;
|
struct regmap *rmap = chg->max14577->regmap;
|
||||||
u8 reg_data;
|
u8 reg_data;
|
||||||
|
int ret;
|
||||||
enum max14577_muic_charger_type chg_type;
|
enum max14577_muic_charger_type chg_type;
|
||||||
|
|
||||||
max14577_read_reg(rmap, MAX14577_MUIC_REG_STATUS2, ®_data);
|
ret = max14577_read_reg(rmap, MAX14577_MUIC_REG_STATUS2, ®_data);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
reg_data = ((reg_data & STATUS2_CHGTYP_MASK) >> STATUS2_CHGTYP_SHIFT);
|
reg_data = ((reg_data & STATUS2_CHGTYP_MASK) >> STATUS2_CHGTYP_SHIFT);
|
||||||
chg_type = maxim_get_charger_type(chg->max14577->dev_type, reg_data);
|
chg_type = maxim_get_charger_type(chg->max14577->dev_type, reg_data);
|
||||||
switch (chg_type) {
|
switch (chg_type) {
|
||||||
|
@ -129,14 +148,17 @@ static int max14577_get_online(struct max14577_charger *chg)
|
||||||
case MAX14577_CHARGER_TYPE_SPECIAL_1A:
|
case MAX14577_CHARGER_TYPE_SPECIAL_1A:
|
||||||
case MAX14577_CHARGER_TYPE_DEAD_BATTERY:
|
case MAX14577_CHARGER_TYPE_DEAD_BATTERY:
|
||||||
case MAX77836_CHARGER_TYPE_SPECIAL_BIAS:
|
case MAX77836_CHARGER_TYPE_SPECIAL_BIAS:
|
||||||
return 1;
|
*val = 1;
|
||||||
|
break;
|
||||||
case MAX14577_CHARGER_TYPE_NONE:
|
case MAX14577_CHARGER_TYPE_NONE:
|
||||||
case MAX14577_CHARGER_TYPE_DOWNSTREAM_PORT:
|
case MAX14577_CHARGER_TYPE_DOWNSTREAM_PORT:
|
||||||
case MAX14577_CHARGER_TYPE_RESERVED:
|
case MAX14577_CHARGER_TYPE_RESERVED:
|
||||||
case MAX77836_CHARGER_TYPE_RESERVED:
|
case MAX77836_CHARGER_TYPE_RESERVED:
|
||||||
default:
|
default:
|
||||||
return 0;
|
*val = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -145,30 +167,38 @@ static int max14577_get_online(struct max14577_charger *chg)
|
||||||
* - POWER_SUPPLY_HEALTH_OVERVOLTAGE
|
* - POWER_SUPPLY_HEALTH_OVERVOLTAGE
|
||||||
* - POWER_SUPPLY_HEALTH_GOOD
|
* - POWER_SUPPLY_HEALTH_GOOD
|
||||||
*/
|
*/
|
||||||
static int max14577_get_battery_health(struct max14577_charger *chg)
|
static int max14577_get_battery_health(struct max14577_charger *chg, int *val)
|
||||||
{
|
{
|
||||||
struct regmap *rmap = chg->max14577->regmap;
|
struct regmap *rmap = chg->max14577->regmap;
|
||||||
int state = POWER_SUPPLY_HEALTH_GOOD;
|
int ret;
|
||||||
u8 reg_data;
|
u8 reg_data;
|
||||||
enum max14577_muic_charger_type chg_type;
|
enum max14577_muic_charger_type chg_type;
|
||||||
|
|
||||||
max14577_read_reg(rmap, MAX14577_MUIC_REG_STATUS2, ®_data);
|
ret = max14577_read_reg(rmap, MAX14577_MUIC_REG_STATUS2, ®_data);
|
||||||
|
if (ret < 0)
|
||||||
|
goto out;
|
||||||
|
|
||||||
reg_data = ((reg_data & STATUS2_CHGTYP_MASK) >> STATUS2_CHGTYP_SHIFT);
|
reg_data = ((reg_data & STATUS2_CHGTYP_MASK) >> STATUS2_CHGTYP_SHIFT);
|
||||||
chg_type = maxim_get_charger_type(chg->max14577->dev_type, reg_data);
|
chg_type = maxim_get_charger_type(chg->max14577->dev_type, reg_data);
|
||||||
if (chg_type == MAX14577_CHARGER_TYPE_DEAD_BATTERY) {
|
if (chg_type == MAX14577_CHARGER_TYPE_DEAD_BATTERY) {
|
||||||
state = POWER_SUPPLY_HEALTH_DEAD;
|
*val = POWER_SUPPLY_HEALTH_DEAD;
|
||||||
goto state_set;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
max14577_read_reg(rmap, MAX14577_CHG_REG_STATUS3, ®_data);
|
ret = max14577_read_reg(rmap, MAX14577_CHG_REG_STATUS3, ®_data);
|
||||||
|
if (ret < 0)
|
||||||
|
goto out;
|
||||||
|
|
||||||
if (reg_data & STATUS3_OVP_MASK) {
|
if (reg_data & STATUS3_OVP_MASK) {
|
||||||
state = POWER_SUPPLY_HEALTH_OVERVOLTAGE;
|
*val = POWER_SUPPLY_HEALTH_OVERVOLTAGE;
|
||||||
goto state_set;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
state_set:
|
/* Not dead, not overvoltage */
|
||||||
chg->battery_state = state;
|
*val = POWER_SUPPLY_HEALTH_GOOD;
|
||||||
return state;
|
|
||||||
|
out:
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -176,9 +206,11 @@ state_set:
|
||||||
* The max14577 chip doesn't report any status of battery presence.
|
* The max14577 chip doesn't report any status of battery presence.
|
||||||
* Lets assume that it will always be used with some battery.
|
* Lets assume that it will always be used with some battery.
|
||||||
*/
|
*/
|
||||||
static int max14577_get_present(struct max14577_charger *chg)
|
static int max14577_get_present(struct max14577_charger *chg, int *val)
|
||||||
{
|
{
|
||||||
return 1;
|
*val = 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int max14577_set_fast_charge_timer(struct max14577_charger *chg,
|
static int max14577_set_fast_charge_timer(struct max14577_charger *chg,
|
||||||
|
@ -389,26 +421,24 @@ static int max14577_charger_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 max14577_charger *chg = container_of(psy,
|
struct max14577_charger *chg = power_supply_get_drvdata(psy);
|
||||||
struct max14577_charger,
|
|
||||||
charger);
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
val->intval = max14577_get_charger_state(chg);
|
ret = max14577_get_charger_state(chg, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
||||||
val->intval = max14577_get_charge_type(chg);
|
ret = max14577_get_charge_type(chg, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_HEALTH:
|
case POWER_SUPPLY_PROP_HEALTH:
|
||||||
val->intval = max14577_get_battery_health(chg);
|
ret = max14577_get_battery_health(chg, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_PRESENT:
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
val->intval = max14577_get_present(chg);
|
ret = max14577_get_present(chg, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_ONLINE:
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
val->intval = max14577_get_online(chg);
|
ret = max14577_get_online(chg, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_MODEL_NAME:
|
case POWER_SUPPLY_PROP_MODEL_NAME:
|
||||||
BUILD_BUG_ON(ARRAY_SIZE(model_names) != MAXIM_DEVICE_TYPE_NUM);
|
BUILD_BUG_ON(ARRAY_SIZE(model_names) != MAXIM_DEVICE_TYPE_NUM);
|
||||||
|
@ -424,6 +454,14 @@ static int max14577_charger_get_property(struct power_supply *psy,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc max14577_charger_desc = {
|
||||||
|
.name = "max14577-charger",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = max14577_charger_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max14577_charger_props),
|
||||||
|
.get_property = max14577_charger_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_OF
|
#ifdef CONFIG_OF
|
||||||
static struct max14577_charger_platform_data *max14577_charger_dt_init(
|
static struct max14577_charger_platform_data *max14577_charger_dt_init(
|
||||||
struct platform_device *pdev)
|
struct platform_device *pdev)
|
||||||
|
@ -531,6 +569,7 @@ static DEVICE_ATTR(fast_charge_timer, S_IRUGO | S_IWUSR,
|
||||||
static int max14577_charger_probe(struct platform_device *pdev)
|
static int max14577_charger_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct max14577_charger *chg;
|
struct max14577_charger *chg;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct max14577 *max14577 = dev_get_drvdata(pdev->dev.parent);
|
struct max14577 *max14577 = dev_get_drvdata(pdev->dev.parent);
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -550,21 +589,18 @@ static int max14577_charger_probe(struct platform_device *pdev)
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
chg->charger.name = "max14577-charger",
|
|
||||||
chg->charger.type = POWER_SUPPLY_TYPE_BATTERY,
|
|
||||||
chg->charger.properties = max14577_charger_props,
|
|
||||||
chg->charger.num_properties = ARRAY_SIZE(max14577_charger_props),
|
|
||||||
chg->charger.get_property = max14577_charger_get_property,
|
|
||||||
|
|
||||||
ret = device_create_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
ret = device_create_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(&pdev->dev, "failed: create sysfs entry\n");
|
dev_err(&pdev->dev, "failed: create sysfs entry\n");
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &chg->charger);
|
psy_cfg.drv_data = chg;
|
||||||
if (ret) {
|
chg->charger = power_supply_register(&pdev->dev, &max14577_charger_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(chg->charger)) {
|
||||||
dev_err(&pdev->dev, "failed: power supply register\n");
|
dev_err(&pdev->dev, "failed: power supply register\n");
|
||||||
|
ret = PTR_ERR(chg->charger);
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -585,7 +621,7 @@ static int max14577_charger_remove(struct platform_device *pdev)
|
||||||
struct max14577_charger *chg = platform_get_drvdata(pdev);
|
struct max14577_charger *chg = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
device_remove_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
device_remove_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
||||||
power_supply_unregister(&chg->charger);
|
power_supply_unregister(chg->charger);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,7 +40,7 @@
|
||||||
struct max17040_chip {
|
struct max17040_chip {
|
||||||
struct i2c_client *client;
|
struct i2c_client *client;
|
||||||
struct delayed_work work;
|
struct delayed_work work;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
struct max17040_platform_data *pdata;
|
struct max17040_platform_data *pdata;
|
||||||
|
|
||||||
/* State Of Connect */
|
/* State Of Connect */
|
||||||
|
@ -57,8 +57,7 @@ static int max17040_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 max17040_chip *chip = container_of(psy,
|
struct max17040_chip *chip = power_supply_get_drvdata(psy);
|
||||||
struct max17040_chip, battery);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -188,7 +187,8 @@ static void max17040_work(struct work_struct *work)
|
||||||
max17040_get_online(chip->client);
|
max17040_get_online(chip->client);
|
||||||
max17040_get_status(chip->client);
|
max17040_get_status(chip->client);
|
||||||
|
|
||||||
schedule_delayed_work(&chip->work, MAX17040_DELAY);
|
queue_delayed_work(system_power_efficient_wq, &chip->work,
|
||||||
|
MAX17040_DELAY);
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum power_supply_property max17040_battery_props[] = {
|
static enum power_supply_property max17040_battery_props[] = {
|
||||||
|
@ -198,12 +198,20 @@ static enum power_supply_property max17040_battery_props[] = {
|
||||||
POWER_SUPPLY_PROP_CAPACITY,
|
POWER_SUPPLY_PROP_CAPACITY,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc max17040_battery_desc = {
|
||||||
|
.name = "battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.get_property = max17040_get_property,
|
||||||
|
.properties = max17040_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max17040_battery_props),
|
||||||
|
};
|
||||||
|
|
||||||
static int max17040_probe(struct i2c_client *client,
|
static int max17040_probe(struct i2c_client *client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct max17040_chip *chip;
|
struct max17040_chip *chip;
|
||||||
int ret;
|
|
||||||
|
|
||||||
if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE))
|
if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE))
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
@ -216,24 +224,21 @@ static int max17040_probe(struct i2c_client *client,
|
||||||
chip->pdata = client->dev.platform_data;
|
chip->pdata = client->dev.platform_data;
|
||||||
|
|
||||||
i2c_set_clientdata(client, chip);
|
i2c_set_clientdata(client, chip);
|
||||||
|
psy_cfg.drv_data = chip;
|
||||||
|
|
||||||
chip->battery.name = "battery";
|
chip->battery = power_supply_register(&client->dev,
|
||||||
chip->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
&max17040_battery_desc, &psy_cfg);
|
||||||
chip->battery.get_property = max17040_get_property;
|
if (IS_ERR(chip->battery)) {
|
||||||
chip->battery.properties = max17040_battery_props;
|
|
||||||
chip->battery.num_properties = ARRAY_SIZE(max17040_battery_props);
|
|
||||||
|
|
||||||
ret = power_supply_register(&client->dev, &chip->battery);
|
|
||||||
if (ret) {
|
|
||||||
dev_err(&client->dev, "failed: power supply register\n");
|
dev_err(&client->dev, "failed: power supply register\n");
|
||||||
return ret;
|
return PTR_ERR(chip->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
max17040_reset(client);
|
max17040_reset(client);
|
||||||
max17040_get_version(client);
|
max17040_get_version(client);
|
||||||
|
|
||||||
INIT_DEFERRABLE_WORK(&chip->work, max17040_work);
|
INIT_DEFERRABLE_WORK(&chip->work, max17040_work);
|
||||||
schedule_delayed_work(&chip->work, MAX17040_DELAY);
|
queue_delayed_work(system_power_efficient_wq, &chip->work,
|
||||||
|
MAX17040_DELAY);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -242,7 +247,7 @@ static int max17040_remove(struct i2c_client *client)
|
||||||
{
|
{
|
||||||
struct max17040_chip *chip = i2c_get_clientdata(client);
|
struct max17040_chip *chip = i2c_get_clientdata(client);
|
||||||
|
|
||||||
power_supply_unregister(&chip->battery);
|
power_supply_unregister(chip->battery);
|
||||||
cancel_delayed_work(&chip->work);
|
cancel_delayed_work(&chip->work);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -263,7 +268,8 @@ static int max17040_resume(struct device *dev)
|
||||||
struct i2c_client *client = to_i2c_client(dev);
|
struct i2c_client *client = to_i2c_client(dev);
|
||||||
struct max17040_chip *chip = i2c_get_clientdata(client);
|
struct max17040_chip *chip = i2c_get_clientdata(client);
|
||||||
|
|
||||||
schedule_delayed_work(&chip->work, MAX17040_DELAY);
|
queue_delayed_work(system_power_efficient_wq, &chip->work,
|
||||||
|
MAX17040_DELAY);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -63,13 +63,10 @@
|
||||||
#define dP_ACC_100 0x1900
|
#define dP_ACC_100 0x1900
|
||||||
#define dP_ACC_200 0x3200
|
#define dP_ACC_200 0x3200
|
||||||
|
|
||||||
#define MAX17042_IC_VERSION 0x0092
|
|
||||||
#define MAX17047_IC_VERSION 0x00AC /* same for max17050 */
|
|
||||||
|
|
||||||
struct max17042_chip {
|
struct max17042_chip {
|
||||||
struct i2c_client *client;
|
struct i2c_client *client;
|
||||||
struct regmap *regmap;
|
struct regmap *regmap;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
enum max170xx_chip_type chip_type;
|
enum max170xx_chip_type chip_type;
|
||||||
struct max17042_platform_data *pdata;
|
struct max17042_platform_data *pdata;
|
||||||
struct work_struct work;
|
struct work_struct work;
|
||||||
|
@ -96,8 +93,7 @@ static int max17042_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 max17042_chip *chip = container_of(psy,
|
struct max17042_chip *chip = power_supply_get_drvdata(psy);
|
||||||
struct max17042_chip, battery);
|
|
||||||
struct regmap *map = chip->regmap;
|
struct regmap *map = chip->regmap;
|
||||||
int ret;
|
int ret;
|
||||||
u32 data;
|
u32 data;
|
||||||
|
@ -132,7 +128,7 @@ static int max17042_get_property(struct power_supply *psy,
|
||||||
val->intval *= 20000; /* Units of LSB = 20mV */
|
val->intval *= 20000; /* Units of LSB = 20mV */
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN:
|
case POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN:
|
||||||
if (chip->chip_type == MAX17042)
|
if (chip->chip_type == MAXIM_DEVICE_TYPE_MAX17042)
|
||||||
ret = regmap_read(map, MAX17042_V_empty, &data);
|
ret = regmap_read(map, MAX17042_V_empty, &data);
|
||||||
else
|
else
|
||||||
ret = regmap_read(map, MAX17047_V_empty, &data);
|
ret = regmap_read(map, MAX17047_V_empty, &data);
|
||||||
|
@ -272,6 +268,7 @@ static inline void max17042_override_por(struct regmap *map,
|
||||||
static inline void max10742_unlock_model(struct max17042_chip *chip)
|
static inline void max10742_unlock_model(struct max17042_chip *chip)
|
||||||
{
|
{
|
||||||
struct regmap *map = chip->regmap;
|
struct regmap *map = chip->regmap;
|
||||||
|
|
||||||
regmap_write(map, MAX17042_MLOCKReg1, MODEL_UNLOCK1);
|
regmap_write(map, MAX17042_MLOCKReg1, MODEL_UNLOCK1);
|
||||||
regmap_write(map, MAX17042_MLOCKReg2, MODEL_UNLOCK2);
|
regmap_write(map, MAX17042_MLOCKReg2, MODEL_UNLOCK2);
|
||||||
}
|
}
|
||||||
|
@ -289,6 +286,7 @@ static inline void max17042_write_model_data(struct max17042_chip *chip,
|
||||||
{
|
{
|
||||||
struct regmap *map = chip->regmap;
|
struct regmap *map = chip->regmap;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < size; i++)
|
for (i = 0; i < size; i++)
|
||||||
regmap_write(map, addr + i,
|
regmap_write(map, addr + i,
|
||||||
chip->pdata->config_data->cell_char_tbl[i]);
|
chip->pdata->config_data->cell_char_tbl[i]);
|
||||||
|
@ -379,7 +377,8 @@ static void max17042_write_config_regs(struct max17042_chip *chip)
|
||||||
regmap_write(map, MAX17042_FilterCFG,
|
regmap_write(map, MAX17042_FilterCFG,
|
||||||
config->filter_cfg);
|
config->filter_cfg);
|
||||||
regmap_write(map, MAX17042_RelaxCFG, config->relax_cfg);
|
regmap_write(map, MAX17042_RelaxCFG, config->relax_cfg);
|
||||||
if (chip->chip_type == MAX17047)
|
if (chip->chip_type == MAXIM_DEVICE_TYPE_MAX17047 ||
|
||||||
|
chip->chip_type == MAXIM_DEVICE_TYPE_MAX17050)
|
||||||
regmap_write(map, MAX17047_FullSOCThr,
|
regmap_write(map, MAX17047_FullSOCThr,
|
||||||
config->full_soc_thresh);
|
config->full_soc_thresh);
|
||||||
}
|
}
|
||||||
|
@ -392,7 +391,7 @@ static void max17042_write_custom_regs(struct max17042_chip *chip)
|
||||||
max17042_write_verify_reg(map, MAX17042_RCOMP0, config->rcomp0);
|
max17042_write_verify_reg(map, MAX17042_RCOMP0, config->rcomp0);
|
||||||
max17042_write_verify_reg(map, MAX17042_TempCo, config->tcompc0);
|
max17042_write_verify_reg(map, MAX17042_TempCo, config->tcompc0);
|
||||||
max17042_write_verify_reg(map, MAX17042_ICHGTerm, config->ichgt_term);
|
max17042_write_verify_reg(map, MAX17042_ICHGTerm, config->ichgt_term);
|
||||||
if (chip->chip_type == MAX17042) {
|
if (chip->chip_type == MAXIM_DEVICE_TYPE_MAX17042) {
|
||||||
regmap_write(map, MAX17042_EmptyTempCo, config->empty_tempco);
|
regmap_write(map, MAX17042_EmptyTempCo, config->empty_tempco);
|
||||||
max17042_write_verify_reg(map, MAX17042_K_empty0,
|
max17042_write_verify_reg(map, MAX17042_K_empty0,
|
||||||
config->kempty0);
|
config->kempty0);
|
||||||
|
@ -501,14 +500,14 @@ static inline void max17042_override_por_values(struct max17042_chip *chip)
|
||||||
|
|
||||||
max17042_override_por(map, MAX17042_FullCAP, config->fullcap);
|
max17042_override_por(map, MAX17042_FullCAP, config->fullcap);
|
||||||
max17042_override_por(map, MAX17042_FullCAPNom, config->fullcapnom);
|
max17042_override_por(map, MAX17042_FullCAPNom, config->fullcapnom);
|
||||||
if (chip->chip_type == MAX17042)
|
if (chip->chip_type == MAXIM_DEVICE_TYPE_MAX17042)
|
||||||
max17042_override_por(map, MAX17042_SOC_empty,
|
max17042_override_por(map, MAX17042_SOC_empty,
|
||||||
config->socempty);
|
config->socempty);
|
||||||
max17042_override_por(map, MAX17042_LAvg_empty, config->lavg_empty);
|
max17042_override_por(map, MAX17042_LAvg_empty, config->lavg_empty);
|
||||||
max17042_override_por(map, MAX17042_dQacc, config->dqacc);
|
max17042_override_por(map, MAX17042_dQacc, config->dqacc);
|
||||||
max17042_override_por(map, MAX17042_dPacc, config->dpacc);
|
max17042_override_por(map, MAX17042_dPacc, config->dpacc);
|
||||||
|
|
||||||
if (chip->chip_type == MAX17042)
|
if (chip->chip_type == MAXIM_DEVICE_TYPE_MAX17042)
|
||||||
max17042_override_por(map, MAX17042_V_empty, config->vempty);
|
max17042_override_por(map, MAX17042_V_empty, config->vempty);
|
||||||
else
|
else
|
||||||
max17042_override_por(map, MAX17047_V_empty, config->vempty);
|
max17042_override_por(map, MAX17047_V_empty, config->vempty);
|
||||||
|
@ -529,7 +528,6 @@ static int max17042_init_chip(struct max17042_chip *chip)
|
||||||
{
|
{
|
||||||
struct regmap *map = chip->regmap;
|
struct regmap *map = chip->regmap;
|
||||||
int ret;
|
int ret;
|
||||||
int val;
|
|
||||||
|
|
||||||
max17042_override_por_values(chip);
|
max17042_override_por_values(chip);
|
||||||
/* After Power up, the MAX17042 requires 500mS in order
|
/* After Power up, the MAX17042 requires 500mS in order
|
||||||
|
@ -572,8 +570,7 @@ static int max17042_init_chip(struct max17042_chip *chip)
|
||||||
max17042_load_new_capacity_params(chip);
|
max17042_load_new_capacity_params(chip);
|
||||||
|
|
||||||
/* Init complete, Clear the POR bit */
|
/* Init complete, Clear the POR bit */
|
||||||
regmap_read(map, MAX17042_STATUS, &val);
|
regmap_update_bits(map, MAX17042_STATUS, STATUS_POR_BIT, 0x0);
|
||||||
regmap_write(map, MAX17042_STATUS, val & (~STATUS_POR_BIT));
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -604,7 +601,7 @@ static irqreturn_t max17042_thread_handler(int id, void *dev)
|
||||||
max17042_set_soc_threshold(chip, 1);
|
max17042_set_soc_threshold(chip, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_changed(&chip->battery);
|
power_supply_changed(chip->battery);
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -664,10 +661,28 @@ static const struct regmap_config max17042_regmap_config = {
|
||||||
.val_format_endian = REGMAP_ENDIAN_NATIVE,
|
.val_format_endian = REGMAP_ENDIAN_NATIVE,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc max17042_psy_desc = {
|
||||||
|
.name = "max170xx_battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.get_property = max17042_get_property,
|
||||||
|
.properties = max17042_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max17042_battery_props),
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc max17042_no_current_sense_psy_desc = {
|
||||||
|
.name = "max170xx_battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.get_property = max17042_get_property,
|
||||||
|
.properties = max17042_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max17042_battery_props) - 2,
|
||||||
|
};
|
||||||
|
|
||||||
static int max17042_probe(struct i2c_client *client,
|
static int max17042_probe(struct i2c_client *client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
||||||
|
const struct power_supply_desc *max17042_desc = &max17042_psy_desc;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct max17042_chip *chip;
|
struct max17042_chip *chip;
|
||||||
int ret;
|
int ret;
|
||||||
int i;
|
int i;
|
||||||
|
@ -694,29 +709,13 @@ static int max17042_probe(struct i2c_client *client,
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_set_clientdata(client, chip);
|
i2c_set_clientdata(client, chip);
|
||||||
|
chip->chip_type = id->driver_data;
|
||||||
regmap_read(chip->regmap, MAX17042_DevName, &val);
|
psy_cfg.drv_data = chip;
|
||||||
if (val == MAX17042_IC_VERSION) {
|
|
||||||
dev_dbg(&client->dev, "chip type max17042 detected\n");
|
|
||||||
chip->chip_type = MAX17042;
|
|
||||||
} else if (val == MAX17047_IC_VERSION) {
|
|
||||||
dev_dbg(&client->dev, "chip type max17047/50 detected\n");
|
|
||||||
chip->chip_type = MAX17047;
|
|
||||||
} else {
|
|
||||||
dev_err(&client->dev, "device version mismatch: %x\n", val);
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
|
|
||||||
chip->battery.name = "max170xx_battery";
|
|
||||||
chip->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
|
||||||
chip->battery.get_property = max17042_get_property;
|
|
||||||
chip->battery.properties = max17042_battery_props;
|
|
||||||
chip->battery.num_properties = ARRAY_SIZE(max17042_battery_props);
|
|
||||||
|
|
||||||
/* When current is not measured,
|
/* When current is not measured,
|
||||||
* CURRENT_NOW and CURRENT_AVG properties should be invisible. */
|
* CURRENT_NOW and CURRENT_AVG properties should be invisible. */
|
||||||
if (!chip->pdata->enable_current_sense)
|
if (!chip->pdata->enable_current_sense)
|
||||||
chip->battery.num_properties -= 2;
|
max17042_desc = &max17042_no_current_sense_psy_desc;
|
||||||
|
|
||||||
if (chip->pdata->r_sns == 0)
|
if (chip->pdata->r_sns == 0)
|
||||||
chip->pdata->r_sns = MAX17042_DEFAULT_SNS_RESISTOR;
|
chip->pdata->r_sns = MAX17042_DEFAULT_SNS_RESISTOR;
|
||||||
|
@ -733,21 +732,22 @@ static int max17042_probe(struct i2c_client *client,
|
||||||
regmap_write(chip->regmap, MAX17042_LearnCFG, 0x0007);
|
regmap_write(chip->regmap, MAX17042_LearnCFG, 0x0007);
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&client->dev, &chip->battery);
|
chip->battery = power_supply_register(&client->dev, max17042_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(chip->battery)) {
|
||||||
dev_err(&client->dev, "failed: power supply register\n");
|
dev_err(&client->dev, "failed: power supply register\n");
|
||||||
return ret;
|
return PTR_ERR(chip->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (client->irq) {
|
if (client->irq) {
|
||||||
ret = request_threaded_irq(client->irq, NULL,
|
ret = request_threaded_irq(client->irq, NULL,
|
||||||
max17042_thread_handler,
|
max17042_thread_handler,
|
||||||
IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
|
IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
|
||||||
chip->battery.name, chip);
|
chip->battery->desc->name, chip);
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
regmap_read(chip->regmap, MAX17042_CONFIG, &val);
|
regmap_update_bits(chip->regmap, MAX17042_CONFIG,
|
||||||
val |= CONFIG_ALRT_BIT_ENBL;
|
CONFIG_ALRT_BIT_ENBL,
|
||||||
regmap_write(chip->regmap, MAX17042_CONFIG, val);
|
CONFIG_ALRT_BIT_ENBL);
|
||||||
max17042_set_soc_threshold(chip, 1);
|
max17042_set_soc_threshold(chip, 1);
|
||||||
} else {
|
} else {
|
||||||
client->irq = 0;
|
client->irq = 0;
|
||||||
|
@ -773,7 +773,7 @@ static int max17042_remove(struct i2c_client *client)
|
||||||
|
|
||||||
if (client->irq)
|
if (client->irq)
|
||||||
free_irq(client->irq, chip);
|
free_irq(client->irq, chip);
|
||||||
power_supply_unregister(&chip->battery);
|
power_supply_unregister(chip->battery);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -823,9 +823,9 @@ MODULE_DEVICE_TABLE(of, max17042_dt_match);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static const struct i2c_device_id max17042_id[] = {
|
static const struct i2c_device_id max17042_id[] = {
|
||||||
{ "max17042", 0 },
|
{ "max17042", MAXIM_DEVICE_TYPE_MAX17042 },
|
||||||
{ "max17047", 1 },
|
{ "max17047", MAXIM_DEVICE_TYPE_MAX17047 },
|
||||||
{ "max17050", 2 },
|
{ "max17050", MAXIM_DEVICE_TYPE_MAX17050 },
|
||||||
{ }
|
{ }
|
||||||
};
|
};
|
||||||
MODULE_DEVICE_TABLE(i2c, max17042_id);
|
MODULE_DEVICE_TABLE(i2c, max17042_id);
|
||||||
|
|
|
@ -22,14 +22,14 @@
|
||||||
#include <linux/mfd/max77693.h>
|
#include <linux/mfd/max77693.h>
|
||||||
#include <linux/mfd/max77693-private.h>
|
#include <linux/mfd/max77693-private.h>
|
||||||
|
|
||||||
static const char *max77693_charger_name = "max77693-charger";
|
#define MAX77693_CHARGER_NAME "max77693-charger"
|
||||||
static const char *max77693_charger_model = "MAX77693";
|
static const char *max77693_charger_model = "MAX77693";
|
||||||
static const char *max77693_charger_manufacturer = "Maxim Integrated";
|
static const char *max77693_charger_manufacturer = "Maxim Integrated";
|
||||||
|
|
||||||
struct max77693_charger {
|
struct max77693_charger {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct max77693_dev *max77693;
|
struct max77693_dev *max77693;
|
||||||
struct power_supply charger;
|
struct power_supply *charger;
|
||||||
|
|
||||||
u32 constant_volt;
|
u32 constant_volt;
|
||||||
u32 min_system_volt;
|
u32 min_system_volt;
|
||||||
|
@ -38,13 +38,14 @@ struct max77693_charger {
|
||||||
u32 charge_input_threshold_volt;
|
u32 charge_input_threshold_volt;
|
||||||
};
|
};
|
||||||
|
|
||||||
static int max77693_get_charger_state(struct regmap *regmap)
|
static int max77693_get_charger_state(struct regmap *regmap, int *val)
|
||||||
{
|
{
|
||||||
int state;
|
int ret;
|
||||||
unsigned int data;
|
unsigned int data;
|
||||||
|
|
||||||
if (regmap_read(regmap, MAX77693_CHG_REG_CHG_DETAILS_01, &data) < 0)
|
ret = regmap_read(regmap, MAX77693_CHG_REG_CHG_DETAILS_01, &data);
|
||||||
return POWER_SUPPLY_STATUS_UNKNOWN;
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
data &= CHG_DETAILS_01_CHG_MASK;
|
data &= CHG_DETAILS_01_CHG_MASK;
|
||||||
data >>= CHG_DETAILS_01_CHG_SHIFT;
|
data >>= CHG_DETAILS_01_CHG_SHIFT;
|
||||||
|
@ -56,35 +57,36 @@ static int max77693_get_charger_state(struct regmap *regmap)
|
||||||
case MAX77693_CHARGING_TOP_OFF:
|
case MAX77693_CHARGING_TOP_OFF:
|
||||||
/* In high temp the charging current is reduced, but still charging */
|
/* In high temp the charging current is reduced, but still charging */
|
||||||
case MAX77693_CHARGING_HIGH_TEMP:
|
case MAX77693_CHARGING_HIGH_TEMP:
|
||||||
state = POWER_SUPPLY_STATUS_CHARGING;
|
*val = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
break;
|
break;
|
||||||
case MAX77693_CHARGING_DONE:
|
case MAX77693_CHARGING_DONE:
|
||||||
state = POWER_SUPPLY_STATUS_FULL;
|
*val = POWER_SUPPLY_STATUS_FULL;
|
||||||
break;
|
break;
|
||||||
case MAX77693_CHARGING_TIMER_EXPIRED:
|
case MAX77693_CHARGING_TIMER_EXPIRED:
|
||||||
case MAX77693_CHARGING_THERMISTOR_SUSPEND:
|
case MAX77693_CHARGING_THERMISTOR_SUSPEND:
|
||||||
state = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
*val = POWER_SUPPLY_STATUS_NOT_CHARGING;
|
||||||
break;
|
break;
|
||||||
case MAX77693_CHARGING_OFF:
|
case MAX77693_CHARGING_OFF:
|
||||||
case MAX77693_CHARGING_OVER_TEMP:
|
case MAX77693_CHARGING_OVER_TEMP:
|
||||||
case MAX77693_CHARGING_WATCHDOG_EXPIRED:
|
case MAX77693_CHARGING_WATCHDOG_EXPIRED:
|
||||||
state = POWER_SUPPLY_STATUS_DISCHARGING;
|
*val = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
break;
|
break;
|
||||||
case MAX77693_CHARGING_RESERVED:
|
case MAX77693_CHARGING_RESERVED:
|
||||||
default:
|
default:
|
||||||
state = POWER_SUPPLY_STATUS_UNKNOWN;
|
*val = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
return state;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int max77693_get_charge_type(struct regmap *regmap)
|
static int max77693_get_charge_type(struct regmap *regmap, int *val)
|
||||||
{
|
{
|
||||||
int state;
|
int ret;
|
||||||
unsigned int data;
|
unsigned int data;
|
||||||
|
|
||||||
if (regmap_read(regmap, MAX77693_CHG_REG_CHG_DETAILS_01, &data) < 0)
|
ret = regmap_read(regmap, MAX77693_CHG_REG_CHG_DETAILS_01, &data);
|
||||||
return POWER_SUPPLY_CHARGE_TYPE_UNKNOWN;
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
data &= CHG_DETAILS_01_CHG_MASK;
|
data &= CHG_DETAILS_01_CHG_MASK;
|
||||||
data >>= CHG_DETAILS_01_CHG_SHIFT;
|
data >>= CHG_DETAILS_01_CHG_SHIFT;
|
||||||
|
@ -96,13 +98,13 @@ static int max77693_get_charge_type(struct regmap *regmap)
|
||||||
* 100 and 250 mA. It is higher than prequalification current.
|
* 100 and 250 mA. It is higher than prequalification current.
|
||||||
*/
|
*/
|
||||||
case MAX77693_CHARGING_TOP_OFF:
|
case MAX77693_CHARGING_TOP_OFF:
|
||||||
state = POWER_SUPPLY_CHARGE_TYPE_TRICKLE;
|
*val = POWER_SUPPLY_CHARGE_TYPE_TRICKLE;
|
||||||
break;
|
break;
|
||||||
case MAX77693_CHARGING_FAST_CONST_CURRENT:
|
case MAX77693_CHARGING_FAST_CONST_CURRENT:
|
||||||
case MAX77693_CHARGING_FAST_CONST_VOLTAGE:
|
case MAX77693_CHARGING_FAST_CONST_VOLTAGE:
|
||||||
/* In high temp the charging current is reduced, but still charging */
|
/* In high temp the charging current is reduced, but still charging */
|
||||||
case MAX77693_CHARGING_HIGH_TEMP:
|
case MAX77693_CHARGING_HIGH_TEMP:
|
||||||
state = POWER_SUPPLY_CHARGE_TYPE_FAST;
|
*val = POWER_SUPPLY_CHARGE_TYPE_FAST;
|
||||||
break;
|
break;
|
||||||
case MAX77693_CHARGING_DONE:
|
case MAX77693_CHARGING_DONE:
|
||||||
case MAX77693_CHARGING_TIMER_EXPIRED:
|
case MAX77693_CHARGING_TIMER_EXPIRED:
|
||||||
|
@ -110,14 +112,14 @@ static int max77693_get_charge_type(struct regmap *regmap)
|
||||||
case MAX77693_CHARGING_OFF:
|
case MAX77693_CHARGING_OFF:
|
||||||
case MAX77693_CHARGING_OVER_TEMP:
|
case MAX77693_CHARGING_OVER_TEMP:
|
||||||
case MAX77693_CHARGING_WATCHDOG_EXPIRED:
|
case MAX77693_CHARGING_WATCHDOG_EXPIRED:
|
||||||
state = POWER_SUPPLY_CHARGE_TYPE_NONE;
|
*val = POWER_SUPPLY_CHARGE_TYPE_NONE;
|
||||||
break;
|
break;
|
||||||
case MAX77693_CHARGING_RESERVED:
|
case MAX77693_CHARGING_RESERVED:
|
||||||
default:
|
default:
|
||||||
state = POWER_SUPPLY_CHARGE_TYPE_UNKNOWN;
|
*val = POWER_SUPPLY_CHARGE_TYPE_UNKNOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
return state;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -129,69 +131,78 @@ static int max77693_get_charge_type(struct regmap *regmap)
|
||||||
* - POWER_SUPPLY_HEALTH_UNKNOWN
|
* - POWER_SUPPLY_HEALTH_UNKNOWN
|
||||||
* - POWER_SUPPLY_HEALTH_UNSPEC_FAILURE
|
* - POWER_SUPPLY_HEALTH_UNSPEC_FAILURE
|
||||||
*/
|
*/
|
||||||
static int max77693_get_battery_health(struct regmap *regmap)
|
static int max77693_get_battery_health(struct regmap *regmap, int *val)
|
||||||
{
|
{
|
||||||
int state;
|
int ret;
|
||||||
unsigned int data;
|
unsigned int data;
|
||||||
|
|
||||||
if (regmap_read(regmap, MAX77693_CHG_REG_CHG_DETAILS_01, &data) < 0)
|
ret = regmap_read(regmap, MAX77693_CHG_REG_CHG_DETAILS_01, &data);
|
||||||
return POWER_SUPPLY_HEALTH_UNKNOWN;
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
data &= CHG_DETAILS_01_BAT_MASK;
|
data &= CHG_DETAILS_01_BAT_MASK;
|
||||||
data >>= CHG_DETAILS_01_BAT_SHIFT;
|
data >>= CHG_DETAILS_01_BAT_SHIFT;
|
||||||
|
|
||||||
switch (data) {
|
switch (data) {
|
||||||
case MAX77693_BATTERY_NOBAT:
|
case MAX77693_BATTERY_NOBAT:
|
||||||
state = POWER_SUPPLY_HEALTH_DEAD;
|
*val = POWER_SUPPLY_HEALTH_DEAD;
|
||||||
break;
|
break;
|
||||||
case MAX77693_BATTERY_PREQUALIFICATION:
|
case MAX77693_BATTERY_PREQUALIFICATION:
|
||||||
case MAX77693_BATTERY_GOOD:
|
case MAX77693_BATTERY_GOOD:
|
||||||
case MAX77693_BATTERY_LOWVOLTAGE:
|
case MAX77693_BATTERY_LOWVOLTAGE:
|
||||||
state = POWER_SUPPLY_HEALTH_GOOD;
|
*val = POWER_SUPPLY_HEALTH_GOOD;
|
||||||
break;
|
break;
|
||||||
case MAX77693_BATTERY_TIMER_EXPIRED:
|
case MAX77693_BATTERY_TIMER_EXPIRED:
|
||||||
/*
|
/*
|
||||||
* Took longer to charge than expected, charging suspended.
|
* Took longer to charge than expected, charging suspended.
|
||||||
* Damaged battery?
|
* Damaged battery?
|
||||||
*/
|
*/
|
||||||
state = POWER_SUPPLY_HEALTH_SAFETY_TIMER_EXPIRE;
|
*val = POWER_SUPPLY_HEALTH_SAFETY_TIMER_EXPIRE;
|
||||||
break;
|
break;
|
||||||
case MAX77693_BATTERY_OVERVOLTAGE:
|
case MAX77693_BATTERY_OVERVOLTAGE:
|
||||||
state = POWER_SUPPLY_HEALTH_OVERVOLTAGE;
|
*val = POWER_SUPPLY_HEALTH_OVERVOLTAGE;
|
||||||
break;
|
break;
|
||||||
case MAX77693_BATTERY_OVERCURRENT:
|
case MAX77693_BATTERY_OVERCURRENT:
|
||||||
state = POWER_SUPPLY_HEALTH_UNSPEC_FAILURE;
|
*val = POWER_SUPPLY_HEALTH_UNSPEC_FAILURE;
|
||||||
break;
|
break;
|
||||||
case MAX77693_BATTERY_RESERVED:
|
case MAX77693_BATTERY_RESERVED:
|
||||||
default:
|
default:
|
||||||
state = POWER_SUPPLY_HEALTH_UNKNOWN;
|
*val = POWER_SUPPLY_HEALTH_UNKNOWN;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return state;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int max77693_get_present(struct regmap *regmap)
|
static int max77693_get_present(struct regmap *regmap, int *val)
|
||||||
{
|
{
|
||||||
unsigned int data;
|
unsigned int data;
|
||||||
|
int ret;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read CHG_INT_OK register. High DETBAT bit here should be
|
* Read CHG_INT_OK register. High DETBAT bit here should be
|
||||||
* equal to value 0x0 in CHG_DETAILS_01/BAT field.
|
* equal to value 0x0 in CHG_DETAILS_01/BAT field.
|
||||||
*/
|
*/
|
||||||
regmap_read(regmap, MAX77693_CHG_REG_CHG_INT_OK, &data);
|
ret = regmap_read(regmap, MAX77693_CHG_REG_CHG_INT_OK, &data);
|
||||||
if (data & CHG_INT_OK_DETBAT_MASK)
|
if (ret < 0)
|
||||||
return 0;
|
return ret;
|
||||||
return 1;
|
|
||||||
|
*val = (data & CHG_INT_OK_DETBAT_MASK) ? 0 : 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int max77693_get_online(struct regmap *regmap)
|
static int max77693_get_online(struct regmap *regmap, int *val)
|
||||||
{
|
{
|
||||||
unsigned int data;
|
unsigned int data;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = regmap_read(regmap, MAX77693_CHG_REG_CHG_INT_OK, &data);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
*val = (data & CHG_INT_OK_CHGIN_MASK) ? 1 : 0;
|
||||||
|
|
||||||
regmap_read(regmap, MAX77693_CHG_REG_CHG_INT_OK, &data);
|
|
||||||
if (data & CHG_INT_OK_CHGIN_MASK)
|
|
||||||
return 1;
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -209,27 +220,25 @@ static int max77693_charger_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 max77693_charger *chg = container_of(psy,
|
struct max77693_charger *chg = power_supply_get_drvdata(psy);
|
||||||
struct max77693_charger,
|
|
||||||
charger);
|
|
||||||
struct regmap *regmap = chg->max77693->regmap;
|
struct regmap *regmap = chg->max77693->regmap;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
val->intval = max77693_get_charger_state(regmap);
|
ret = max77693_get_charger_state(regmap, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
case POWER_SUPPLY_PROP_CHARGE_TYPE:
|
||||||
val->intval = max77693_get_charge_type(regmap);
|
ret = max77693_get_charge_type(regmap, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_HEALTH:
|
case POWER_SUPPLY_PROP_HEALTH:
|
||||||
val->intval = max77693_get_battery_health(regmap);
|
ret = max77693_get_battery_health(regmap, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_PRESENT:
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
val->intval = max77693_get_present(regmap);
|
ret = max77693_get_present(regmap, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_ONLINE:
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
val->intval = max77693_get_online(regmap);
|
ret = max77693_get_online(regmap, &val->intval);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_MODEL_NAME:
|
case POWER_SUPPLY_PROP_MODEL_NAME:
|
||||||
val->strval = max77693_charger_model;
|
val->strval = max77693_charger_model;
|
||||||
|
@ -244,6 +253,14 @@ static int max77693_charger_get_property(struct power_supply *psy,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc max77693_charger_desc = {
|
||||||
|
.name = MAX77693_CHARGER_NAME,
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = max77693_charger_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max77693_charger_props),
|
||||||
|
.get_property = max77693_charger_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
static ssize_t device_attr_store(struct device *dev,
|
static ssize_t device_attr_store(struct device *dev,
|
||||||
struct device_attribute *attr, const char *buf, size_t count,
|
struct device_attribute *attr, const char *buf, size_t count,
|
||||||
int (*fn)(struct max77693_charger *, unsigned long))
|
int (*fn)(struct max77693_charger *, unsigned long))
|
||||||
|
@ -659,6 +676,7 @@ static int max77693_dt_init(struct device *dev, struct max77693_charger *chg)
|
||||||
static int max77693_charger_probe(struct platform_device *pdev)
|
static int max77693_charger_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct max77693_charger *chg;
|
struct max77693_charger *chg;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct max77693_dev *max77693 = dev_get_drvdata(pdev->dev.parent);
|
struct max77693_dev *max77693 = dev_get_drvdata(pdev->dev.parent);
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -678,11 +696,7 @@ static int max77693_charger_probe(struct platform_device *pdev)
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
chg->charger.name = max77693_charger_name;
|
psy_cfg.drv_data = chg;
|
||||||
chg->charger.type = POWER_SUPPLY_TYPE_BATTERY;
|
|
||||||
chg->charger.properties = max77693_charger_props;
|
|
||||||
chg->charger.num_properties = ARRAY_SIZE(max77693_charger_props);
|
|
||||||
chg->charger.get_property = max77693_charger_get_property;
|
|
||||||
|
|
||||||
ret = device_create_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
ret = device_create_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -703,9 +717,12 @@ static int max77693_charger_probe(struct platform_device *pdev)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &chg->charger);
|
chg->charger = power_supply_register(&pdev->dev,
|
||||||
if (ret) {
|
&max77693_charger_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(chg->charger)) {
|
||||||
dev_err(&pdev->dev, "failed: power supply register\n");
|
dev_err(&pdev->dev, "failed: power supply register\n");
|
||||||
|
ret = PTR_ERR(chg->charger);
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -727,7 +744,7 @@ static int max77693_charger_remove(struct platform_device *pdev)
|
||||||
device_remove_file(&pdev->dev, &dev_attr_top_off_threshold_current);
|
device_remove_file(&pdev->dev, &dev_attr_top_off_threshold_current);
|
||||||
device_remove_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
device_remove_file(&pdev->dev, &dev_attr_fast_charge_timer);
|
||||||
|
|
||||||
power_supply_unregister(&chg->charger);
|
power_supply_unregister(chg->charger);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,7 +31,8 @@
|
||||||
struct max8903_data {
|
struct max8903_data {
|
||||||
struct max8903_pdata pdata;
|
struct max8903_pdata pdata;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
|
struct power_supply_desc psy_desc;
|
||||||
bool fault;
|
bool fault;
|
||||||
bool usb_in;
|
bool usb_in;
|
||||||
bool ta_in;
|
bool ta_in;
|
||||||
|
@ -47,8 +48,7 @@ static int max8903_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 max8903_data *data = container_of(psy,
|
struct max8903_data *data = power_supply_get_drvdata(psy);
|
||||||
struct max8903_data, psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -104,17 +104,17 @@ static irqreturn_t max8903_dcin(int irq, void *_data)
|
||||||
dev_dbg(data->dev, "TA(DC-IN) Charger %s.\n", ta_in ?
|
dev_dbg(data->dev, "TA(DC-IN) Charger %s.\n", ta_in ?
|
||||||
"Connected" : "Disconnected");
|
"Connected" : "Disconnected");
|
||||||
|
|
||||||
old_type = data->psy.type;
|
old_type = data->psy_desc.type;
|
||||||
|
|
||||||
if (data->ta_in)
|
if (data->ta_in)
|
||||||
data->psy.type = POWER_SUPPLY_TYPE_MAINS;
|
data->psy_desc.type = POWER_SUPPLY_TYPE_MAINS;
|
||||||
else if (data->usb_in)
|
else if (data->usb_in)
|
||||||
data->psy.type = POWER_SUPPLY_TYPE_USB;
|
data->psy_desc.type = POWER_SUPPLY_TYPE_USB;
|
||||||
else
|
else
|
||||||
data->psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
data->psy_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
|
|
||||||
if (old_type != data->psy.type)
|
if (old_type != data->psy_desc.type)
|
||||||
power_supply_changed(&data->psy);
|
power_supply_changed(data->psy);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -143,17 +143,17 @@ static irqreturn_t max8903_usbin(int irq, void *_data)
|
||||||
dev_dbg(data->dev, "USB Charger %s.\n", usb_in ?
|
dev_dbg(data->dev, "USB Charger %s.\n", usb_in ?
|
||||||
"Connected" : "Disconnected");
|
"Connected" : "Disconnected");
|
||||||
|
|
||||||
old_type = data->psy.type;
|
old_type = data->psy_desc.type;
|
||||||
|
|
||||||
if (data->ta_in)
|
if (data->ta_in)
|
||||||
data->psy.type = POWER_SUPPLY_TYPE_MAINS;
|
data->psy_desc.type = POWER_SUPPLY_TYPE_MAINS;
|
||||||
else if (data->usb_in)
|
else if (data->usb_in)
|
||||||
data->psy.type = POWER_SUPPLY_TYPE_USB;
|
data->psy_desc.type = POWER_SUPPLY_TYPE_USB;
|
||||||
else
|
else
|
||||||
data->psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
data->psy_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
|
|
||||||
if (old_type != data->psy.type)
|
if (old_type != data->psy_desc.type)
|
||||||
power_supply_changed(&data->psy);
|
power_supply_changed(data->psy);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -184,6 +184,7 @@ static int max8903_probe(struct platform_device *pdev)
|
||||||
struct max8903_data *data;
|
struct max8903_data *data;
|
||||||
struct device *dev = &pdev->dev;
|
struct device *dev = &pdev->dev;
|
||||||
struct max8903_pdata *pdata = pdev->dev.platform_data;
|
struct max8903_pdata *pdata = pdev->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
int gpio;
|
int gpio;
|
||||||
int ta_in = 0;
|
int ta_in = 0;
|
||||||
|
@ -280,17 +281,20 @@ static int max8903_probe(struct platform_device *pdev)
|
||||||
data->ta_in = ta_in;
|
data->ta_in = ta_in;
|
||||||
data->usb_in = usb_in;
|
data->usb_in = usb_in;
|
||||||
|
|
||||||
data->psy.name = "max8903_charger";
|
data->psy_desc.name = "max8903_charger";
|
||||||
data->psy.type = (ta_in) ? POWER_SUPPLY_TYPE_MAINS :
|
data->psy_desc.type = (ta_in) ? POWER_SUPPLY_TYPE_MAINS :
|
||||||
((usb_in) ? POWER_SUPPLY_TYPE_USB :
|
((usb_in) ? POWER_SUPPLY_TYPE_USB :
|
||||||
POWER_SUPPLY_TYPE_BATTERY);
|
POWER_SUPPLY_TYPE_BATTERY);
|
||||||
data->psy.get_property = max8903_get_property;
|
data->psy_desc.get_property = max8903_get_property;
|
||||||
data->psy.properties = max8903_charger_props;
|
data->psy_desc.properties = max8903_charger_props;
|
||||||
data->psy.num_properties = ARRAY_SIZE(max8903_charger_props);
|
data->psy_desc.num_properties = ARRAY_SIZE(max8903_charger_props);
|
||||||
|
|
||||||
ret = power_supply_register(dev, &data->psy);
|
psy_cfg.drv_data = data;
|
||||||
if (ret) {
|
|
||||||
|
data->psy = power_supply_register(dev, &data->psy_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(data->psy)) {
|
||||||
dev_err(dev, "failed: power supply register.\n");
|
dev_err(dev, "failed: power supply register.\n");
|
||||||
|
ret = PTR_ERR(data->psy);
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -339,7 +343,7 @@ err_dc_irq:
|
||||||
if (pdata->dc_valid)
|
if (pdata->dc_valid)
|
||||||
free_irq(gpio_to_irq(pdata->dok), data);
|
free_irq(gpio_to_irq(pdata->dok), data);
|
||||||
err_psy:
|
err_psy:
|
||||||
power_supply_unregister(&data->psy);
|
power_supply_unregister(data->psy);
|
||||||
err:
|
err:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -357,7 +361,7 @@ static int max8903_remove(struct platform_device *pdev)
|
||||||
free_irq(gpio_to_irq(pdata->uok), data);
|
free_irq(gpio_to_irq(pdata->uok), data);
|
||||||
if (pdata->dc_valid)
|
if (pdata->dc_valid)
|
||||||
free_irq(gpio_to_irq(pdata->dok), data);
|
free_irq(gpio_to_irq(pdata->dok), data);
|
||||||
power_supply_unregister(&data->psy);
|
power_supply_unregister(data->psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -68,9 +68,9 @@ struct max8925_power_info {
|
||||||
struct i2c_client *gpm;
|
struct i2c_client *gpm;
|
||||||
struct i2c_client *adc;
|
struct i2c_client *adc;
|
||||||
|
|
||||||
struct power_supply ac;
|
struct power_supply *ac;
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
int irq_base;
|
int irq_base;
|
||||||
unsigned ac_online:1;
|
unsigned ac_online:1;
|
||||||
unsigned usb_online:1;
|
unsigned usb_online:1;
|
||||||
|
@ -196,7 +196,7 @@ static int max8925_ac_get_prop(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 max8925_power_info *info = dev_get_drvdata(psy->dev->parent);
|
struct max8925_power_info *info = dev_get_drvdata(psy->dev.parent);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -230,7 +230,7 @@ static int max8925_usb_get_prop(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 max8925_power_info *info = dev_get_drvdata(psy->dev->parent);
|
struct max8925_power_info *info = dev_get_drvdata(psy->dev.parent);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -264,7 +264,7 @@ static int max8925_bat_get_prop(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 max8925_power_info *info = dev_get_drvdata(psy->dev->parent);
|
struct max8925_power_info *info = dev_get_drvdata(psy->dev.parent);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -347,6 +347,30 @@ static enum power_supply_property max8925_battery_props[] = {
|
||||||
POWER_SUPPLY_PROP_STATUS,
|
POWER_SUPPLY_PROP_STATUS,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc ac_desc = {
|
||||||
|
.name = "max8925-ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = max8925_ac_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max8925_ac_props),
|
||||||
|
.get_property = max8925_ac_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc usb_desc = {
|
||||||
|
.name = "max8925-usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = max8925_usb_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max8925_usb_props),
|
||||||
|
.get_property = max8925_usb_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc battery_desc = {
|
||||||
|
.name = "max8925-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = max8925_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max8925_battery_props),
|
||||||
|
.get_property = max8925_bat_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
#define REQUEST_IRQ(_irq, _name) \
|
#define REQUEST_IRQ(_irq, _name) \
|
||||||
do { \
|
do { \
|
||||||
ret = request_threaded_irq(chip->irq_base + _irq, NULL, \
|
ret = request_threaded_irq(chip->irq_base + _irq, NULL, \
|
||||||
|
@ -482,6 +506,7 @@ max8925_power_dt_init(struct platform_device *pdev)
|
||||||
static int max8925_power_probe(struct platform_device *pdev)
|
static int max8925_power_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct max8925_chip *chip = dev_get_drvdata(pdev->dev.parent);
|
struct max8925_chip *chip = dev_get_drvdata(pdev->dev.parent);
|
||||||
|
struct power_supply_config psy_cfg = {}; /* Only for ac and usb */
|
||||||
struct max8925_power_pdata *pdata = NULL;
|
struct max8925_power_pdata *pdata = NULL;
|
||||||
struct max8925_power_info *info;
|
struct max8925_power_info *info;
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -502,40 +527,29 @@ static int max8925_power_probe(struct platform_device *pdev)
|
||||||
info->adc = chip->adc;
|
info->adc = chip->adc;
|
||||||
platform_set_drvdata(pdev, info);
|
platform_set_drvdata(pdev, info);
|
||||||
|
|
||||||
info->ac.name = "max8925-ac";
|
psy_cfg.supplied_to = pdata->supplied_to;
|
||||||
info->ac.type = POWER_SUPPLY_TYPE_MAINS;
|
psy_cfg.num_supplicants = pdata->num_supplicants;
|
||||||
info->ac.properties = max8925_ac_props;
|
|
||||||
info->ac.num_properties = ARRAY_SIZE(max8925_ac_props);
|
info->ac = power_supply_register(&pdev->dev, &ac_desc, &psy_cfg);
|
||||||
info->ac.get_property = max8925_ac_get_prop;
|
if (IS_ERR(info->ac)) {
|
||||||
info->ac.supplied_to = pdata->supplied_to;
|
ret = PTR_ERR(info->ac);
|
||||||
info->ac.num_supplicants = pdata->num_supplicants;
|
|
||||||
ret = power_supply_register(&pdev->dev, &info->ac);
|
|
||||||
if (ret)
|
|
||||||
goto out;
|
goto out;
|
||||||
info->ac.dev->parent = &pdev->dev;
|
}
|
||||||
|
info->ac->dev.parent = &pdev->dev;
|
||||||
|
|
||||||
info->usb.name = "max8925-usb";
|
info->usb = power_supply_register(&pdev->dev, &usb_desc, &psy_cfg);
|
||||||
info->usb.type = POWER_SUPPLY_TYPE_USB;
|
if (IS_ERR(info->usb)) {
|
||||||
info->usb.properties = max8925_usb_props;
|
ret = PTR_ERR(info->usb);
|
||||||
info->usb.num_properties = ARRAY_SIZE(max8925_usb_props);
|
|
||||||
info->usb.get_property = max8925_usb_get_prop;
|
|
||||||
info->usb.supplied_to = pdata->supplied_to;
|
|
||||||
info->usb.num_supplicants = pdata->num_supplicants;
|
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &info->usb);
|
|
||||||
if (ret)
|
|
||||||
goto out_usb;
|
goto out_usb;
|
||||||
info->usb.dev->parent = &pdev->dev;
|
}
|
||||||
|
info->usb->dev.parent = &pdev->dev;
|
||||||
|
|
||||||
info->battery.name = "max8925-battery";
|
info->battery = power_supply_register(&pdev->dev, &battery_desc, NULL);
|
||||||
info->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
if (IS_ERR(info->battery)) {
|
||||||
info->battery.properties = max8925_battery_props;
|
ret = PTR_ERR(info->battery);
|
||||||
info->battery.num_properties = ARRAY_SIZE(max8925_battery_props);
|
|
||||||
info->battery.get_property = max8925_bat_get_prop;
|
|
||||||
ret = power_supply_register(&pdev->dev, &info->battery);
|
|
||||||
if (ret)
|
|
||||||
goto out_battery;
|
goto out_battery;
|
||||||
info->battery.dev->parent = &pdev->dev;
|
}
|
||||||
|
info->battery->dev.parent = &pdev->dev;
|
||||||
|
|
||||||
info->batt_detect = pdata->batt_detect;
|
info->batt_detect = pdata->batt_detect;
|
||||||
info->topoff_threshold = pdata->topoff_threshold;
|
info->topoff_threshold = pdata->topoff_threshold;
|
||||||
|
@ -547,9 +561,9 @@ static int max8925_power_probe(struct platform_device *pdev)
|
||||||
max8925_init_charger(chip, info);
|
max8925_init_charger(chip, info);
|
||||||
return 0;
|
return 0;
|
||||||
out_battery:
|
out_battery:
|
||||||
power_supply_unregister(&info->battery);
|
power_supply_unregister(info->battery);
|
||||||
out_usb:
|
out_usb:
|
||||||
power_supply_unregister(&info->ac);
|
power_supply_unregister(info->ac);
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -559,9 +573,9 @@ static int max8925_power_remove(struct platform_device *pdev)
|
||||||
struct max8925_power_info *info = platform_get_drvdata(pdev);
|
struct max8925_power_info *info = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
if (info) {
|
if (info) {
|
||||||
power_supply_unregister(&info->ac);
|
power_supply_unregister(info->ac);
|
||||||
power_supply_unregister(&info->usb);
|
power_supply_unregister(info->usb);
|
||||||
power_supply_unregister(&info->battery);
|
power_supply_unregister(info->battery);
|
||||||
max8925_deinit_charger(info);
|
max8925_deinit_charger(info);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
struct charger_data {
|
struct charger_data {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct max8997_dev *iodev;
|
struct max8997_dev *iodev;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
};
|
};
|
||||||
|
|
||||||
static enum power_supply_property max8997_battery_props[] = {
|
static enum power_supply_property max8997_battery_props[] = {
|
||||||
|
@ -44,8 +44,7 @@ static int max8997_battery_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 charger_data *charger = container_of(psy,
|
struct charger_data *charger = power_supply_get_drvdata(psy);
|
||||||
struct charger_data, battery);
|
|
||||||
struct i2c_client *i2c = charger->iodev->i2c;
|
struct i2c_client *i2c = charger->iodev->i2c;
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg;
|
u8 reg;
|
||||||
|
@ -86,12 +85,21 @@ static int max8997_battery_get_property(struct power_supply *psy,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc max8997_battery_desc = {
|
||||||
|
.name = "max8997_pmic",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.get_property = max8997_battery_get_property,
|
||||||
|
.properties = max8997_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max8997_battery_props),
|
||||||
|
};
|
||||||
|
|
||||||
static int max8997_battery_probe(struct platform_device *pdev)
|
static int max8997_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct charger_data *charger;
|
struct charger_data *charger;
|
||||||
struct max8997_dev *iodev = dev_get_drvdata(pdev->dev.parent);
|
struct max8997_dev *iodev = dev_get_drvdata(pdev->dev.parent);
|
||||||
struct max8997_platform_data *pdata = dev_get_platdata(iodev->dev);
|
struct max8997_platform_data *pdata = dev_get_platdata(iodev->dev);
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
if (!pdata)
|
if (!pdata)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -147,19 +155,18 @@ static int max8997_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
platform_set_drvdata(pdev, charger);
|
platform_set_drvdata(pdev, charger);
|
||||||
|
|
||||||
charger->battery.name = "max8997_pmic";
|
|
||||||
charger->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
|
||||||
charger->battery.get_property = max8997_battery_get_property;
|
|
||||||
charger->battery.properties = max8997_battery_props;
|
|
||||||
charger->battery.num_properties = ARRAY_SIZE(max8997_battery_props);
|
|
||||||
|
|
||||||
charger->dev = &pdev->dev;
|
charger->dev = &pdev->dev;
|
||||||
charger->iodev = iodev;
|
charger->iodev = iodev;
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &charger->battery);
|
psy_cfg.drv_data = charger;
|
||||||
if (ret) {
|
|
||||||
|
charger->battery = power_supply_register(&pdev->dev,
|
||||||
|
&max8997_battery_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(charger->battery)) {
|
||||||
dev_err(&pdev->dev, "failed: power supply register\n");
|
dev_err(&pdev->dev, "failed: power supply register\n");
|
||||||
return ret;
|
return PTR_ERR(charger->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -169,7 +176,7 @@ static int max8997_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct charger_data *charger = platform_get_drvdata(pdev);
|
struct charger_data *charger = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
power_supply_unregister(&charger->battery);
|
power_supply_unregister(charger->battery);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
struct max8998_battery_data {
|
struct max8998_battery_data {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct max8998_dev *iodev;
|
struct max8998_dev *iodev;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
};
|
};
|
||||||
|
|
||||||
static enum power_supply_property max8998_battery_props[] = {
|
static enum power_supply_property max8998_battery_props[] = {
|
||||||
|
@ -43,8 +43,7 @@ static int max8998_battery_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 max8998_battery_data *max8998 = container_of(psy,
|
struct max8998_battery_data *max8998 = power_supply_get_drvdata(psy);
|
||||||
struct max8998_battery_data, battery);
|
|
||||||
struct i2c_client *i2c = max8998->iodev->i2c;
|
struct i2c_client *i2c = max8998->iodev->i2c;
|
||||||
int ret;
|
int ret;
|
||||||
u8 reg;
|
u8 reg;
|
||||||
|
@ -75,10 +74,19 @@ static int max8998_battery_get_property(struct power_supply *psy,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc max8998_battery_desc = {
|
||||||
|
.name = "max8998_pmic",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.get_property = max8998_battery_get_property,
|
||||||
|
.properties = max8998_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(max8998_battery_props),
|
||||||
|
};
|
||||||
|
|
||||||
static int max8998_battery_probe(struct platform_device *pdev)
|
static int max8998_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct max8998_dev *iodev = dev_get_drvdata(pdev->dev.parent);
|
struct max8998_dev *iodev = dev_get_drvdata(pdev->dev.parent);
|
||||||
struct max8998_platform_data *pdata = dev_get_platdata(iodev->dev);
|
struct max8998_platform_data *pdata = dev_get_platdata(iodev->dev);
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct max8998_battery_data *max8998;
|
struct max8998_battery_data *max8998;
|
||||||
struct i2c_client *i2c;
|
struct i2c_client *i2c;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
@ -161,15 +169,15 @@ static int max8998_battery_probe(struct platform_device *pdev)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
max8998->battery.name = "max8998_pmic";
|
psy_cfg.drv_data = max8998;
|
||||||
max8998->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
|
||||||
max8998->battery.get_property = max8998_battery_get_property;
|
|
||||||
max8998->battery.properties = max8998_battery_props;
|
|
||||||
max8998->battery.num_properties = ARRAY_SIZE(max8998_battery_props);
|
|
||||||
|
|
||||||
ret = power_supply_register(max8998->dev, &max8998->battery);
|
max8998->battery = power_supply_register(max8998->dev,
|
||||||
if (ret) {
|
&max8998_battery_desc,
|
||||||
dev_err(max8998->dev, "failed: power supply register\n");
|
&psy_cfg);
|
||||||
|
if (IS_ERR(max8998->battery)) {
|
||||||
|
ret = PTR_ERR(max8998->battery);
|
||||||
|
dev_err(max8998->dev, "failed: power supply register: %d\n",
|
||||||
|
ret);
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -182,7 +190,7 @@ static int max8998_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct max8998_battery_data *max8998 = platform_get_drvdata(pdev);
|
struct max8998_battery_data *max8998 = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
power_supply_unregister(&max8998->battery);
|
power_supply_unregister(max8998->battery);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -81,7 +81,7 @@ static enum power_supply_property olpc_ac_props[] = {
|
||||||
POWER_SUPPLY_PROP_ONLINE,
|
POWER_SUPPLY_PROP_ONLINE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply olpc_ac = {
|
static const struct power_supply_desc olpc_ac_desc = {
|
||||||
.name = "olpc-ac",
|
.name = "olpc-ac",
|
||||||
.type = POWER_SUPPLY_TYPE_MAINS,
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
.properties = olpc_ac_props,
|
.properties = olpc_ac_props,
|
||||||
|
@ -89,6 +89,8 @@ static struct power_supply olpc_ac = {
|
||||||
.get_property = olpc_ac_get_prop,
|
.get_property = olpc_ac_get_prop,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct power_supply *olpc_ac;
|
||||||
|
|
||||||
static char bat_serial[17]; /* Ick */
|
static char bat_serial[17]; /* Ick */
|
||||||
|
|
||||||
static int olpc_bat_get_status(union power_supply_propval *val, uint8_t ec_byte)
|
static int olpc_bat_get_status(union power_supply_propval *val, uint8_t ec_byte)
|
||||||
|
@ -574,21 +576,23 @@ static struct device_attribute olpc_bat_error = {
|
||||||
* Initialisation
|
* Initialisation
|
||||||
*********************************************************************/
|
*********************************************************************/
|
||||||
|
|
||||||
static struct power_supply olpc_bat = {
|
static struct power_supply_desc olpc_bat_desc = {
|
||||||
.name = "olpc-battery",
|
.name = "olpc-battery",
|
||||||
.get_property = olpc_bat_get_property,
|
.get_property = olpc_bat_get_property,
|
||||||
.use_for_apm = 1,
|
.use_for_apm = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct power_supply *olpc_bat;
|
||||||
|
|
||||||
static int olpc_battery_suspend(struct platform_device *pdev,
|
static int olpc_battery_suspend(struct platform_device *pdev,
|
||||||
pm_message_t state)
|
pm_message_t state)
|
||||||
{
|
{
|
||||||
if (device_may_wakeup(olpc_ac.dev))
|
if (device_may_wakeup(&olpc_ac->dev))
|
||||||
olpc_ec_wakeup_set(EC_SCI_SRC_ACPWR);
|
olpc_ec_wakeup_set(EC_SCI_SRC_ACPWR);
|
||||||
else
|
else
|
||||||
olpc_ec_wakeup_clear(EC_SCI_SRC_ACPWR);
|
olpc_ec_wakeup_clear(EC_SCI_SRC_ACPWR);
|
||||||
|
|
||||||
if (device_may_wakeup(olpc_bat.dev))
|
if (device_may_wakeup(&olpc_bat->dev))
|
||||||
olpc_ec_wakeup_set(EC_SCI_SRC_BATTERY | EC_SCI_SRC_BATSOC
|
olpc_ec_wakeup_set(EC_SCI_SRC_BATTERY | EC_SCI_SRC_BATSOC
|
||||||
| EC_SCI_SRC_BATERR);
|
| EC_SCI_SRC_BATERR);
|
||||||
else
|
else
|
||||||
|
@ -619,52 +623,54 @@ static int olpc_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
/* Ignore the status. It doesn't actually matter */
|
/* Ignore the status. It doesn't actually matter */
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &olpc_ac);
|
olpc_ac = power_supply_register(&pdev->dev, &olpc_ac_desc, NULL);
|
||||||
if (ret)
|
if (IS_ERR(olpc_ac))
|
||||||
return ret;
|
return PTR_ERR(olpc_ac);
|
||||||
|
|
||||||
if (olpc_board_at_least(olpc_board_pre(0xd0))) { /* XO-1.5 */
|
if (olpc_board_at_least(olpc_board_pre(0xd0))) { /* XO-1.5 */
|
||||||
olpc_bat.properties = olpc_xo15_bat_props;
|
olpc_bat_desc.properties = olpc_xo15_bat_props;
|
||||||
olpc_bat.num_properties = ARRAY_SIZE(olpc_xo15_bat_props);
|
olpc_bat_desc.num_properties = ARRAY_SIZE(olpc_xo15_bat_props);
|
||||||
} else { /* XO-1 */
|
} else { /* XO-1 */
|
||||||
olpc_bat.properties = olpc_xo1_bat_props;
|
olpc_bat_desc.properties = olpc_xo1_bat_props;
|
||||||
olpc_bat.num_properties = ARRAY_SIZE(olpc_xo1_bat_props);
|
olpc_bat_desc.num_properties = ARRAY_SIZE(olpc_xo1_bat_props);
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &olpc_bat);
|
olpc_bat = power_supply_register(&pdev->dev, &olpc_bat_desc, NULL);
|
||||||
if (ret)
|
if (IS_ERR(olpc_bat)) {
|
||||||
|
ret = PTR_ERR(olpc_bat);
|
||||||
goto battery_failed;
|
goto battery_failed;
|
||||||
|
}
|
||||||
|
|
||||||
ret = device_create_bin_file(olpc_bat.dev, &olpc_bat_eeprom);
|
ret = device_create_bin_file(&olpc_bat->dev, &olpc_bat_eeprom);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto eeprom_failed;
|
goto eeprom_failed;
|
||||||
|
|
||||||
ret = device_create_file(olpc_bat.dev, &olpc_bat_error);
|
ret = device_create_file(&olpc_bat->dev, &olpc_bat_error);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_failed;
|
goto error_failed;
|
||||||
|
|
||||||
if (olpc_ec_wakeup_available()) {
|
if (olpc_ec_wakeup_available()) {
|
||||||
device_set_wakeup_capable(olpc_ac.dev, true);
|
device_set_wakeup_capable(&olpc_ac->dev, true);
|
||||||
device_set_wakeup_capable(olpc_bat.dev, true);
|
device_set_wakeup_capable(&olpc_bat->dev, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
error_failed:
|
error_failed:
|
||||||
device_remove_bin_file(olpc_bat.dev, &olpc_bat_eeprom);
|
device_remove_bin_file(&olpc_bat->dev, &olpc_bat_eeprom);
|
||||||
eeprom_failed:
|
eeprom_failed:
|
||||||
power_supply_unregister(&olpc_bat);
|
power_supply_unregister(olpc_bat);
|
||||||
battery_failed:
|
battery_failed:
|
||||||
power_supply_unregister(&olpc_ac);
|
power_supply_unregister(olpc_ac);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int olpc_battery_remove(struct platform_device *pdev)
|
static int olpc_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
device_remove_file(olpc_bat.dev, &olpc_bat_error);
|
device_remove_file(&olpc_bat->dev, &olpc_bat_error);
|
||||||
device_remove_bin_file(olpc_bat.dev, &olpc_bat_eeprom);
|
device_remove_bin_file(&olpc_bat->dev, &olpc_bat_eeprom);
|
||||||
power_supply_unregister(&olpc_bat);
|
power_supply_unregister(olpc_bat);
|
||||||
power_supply_unregister(&olpc_ac);
|
power_supply_unregister(olpc_ac);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,9 +33,9 @@ struct pcf50633_mbc {
|
||||||
int adapter_online;
|
int adapter_online;
|
||||||
int usb_online;
|
int usb_online;
|
||||||
|
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct power_supply adapter;
|
struct power_supply *adapter;
|
||||||
struct power_supply ac;
|
struct power_supply *ac;
|
||||||
};
|
};
|
||||||
|
|
||||||
int pcf50633_mbc_usb_curlim_set(struct pcf50633 *pcf, int ma)
|
int pcf50633_mbc_usb_curlim_set(struct pcf50633 *pcf, int ma)
|
||||||
|
@ -104,7 +104,7 @@ int pcf50633_mbc_usb_curlim_set(struct pcf50633 *pcf, int ma)
|
||||||
PCF50633_MBCC1_CHGENA, PCF50633_MBCC1_CHGENA);
|
PCF50633_MBCC1_CHGENA, PCF50633_MBCC1_CHGENA);
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_changed(&mbc->usb);
|
power_supply_changed(mbc->usb);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -278,9 +278,9 @@ pcf50633_mbc_irq_handler(int irq, void *data)
|
||||||
else if (irq == PCF50633_IRQ_ADPREM)
|
else if (irq == PCF50633_IRQ_ADPREM)
|
||||||
mbc->adapter_online = 0;
|
mbc->adapter_online = 0;
|
||||||
|
|
||||||
power_supply_changed(&mbc->ac);
|
power_supply_changed(mbc->ac);
|
||||||
power_supply_changed(&mbc->usb);
|
power_supply_changed(mbc->usb);
|
||||||
power_supply_changed(&mbc->adapter);
|
power_supply_changed(mbc->adapter);
|
||||||
|
|
||||||
if (mbc->pcf->pdata->mbc_event_callback)
|
if (mbc->pcf->pdata->mbc_event_callback)
|
||||||
mbc->pcf->pdata->mbc_event_callback(mbc->pcf, irq);
|
mbc->pcf->pdata->mbc_event_callback(mbc->pcf, irq);
|
||||||
|
@ -290,8 +290,7 @@ static int adapter_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 pcf50633_mbc *mbc = container_of(psy,
|
struct pcf50633_mbc *mbc = power_supply_get_drvdata(psy);
|
||||||
struct pcf50633_mbc, adapter);
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -309,7 +308,7 @@ static int usb_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 pcf50633_mbc *mbc = container_of(psy, struct pcf50633_mbc, usb);
|
struct pcf50633_mbc *mbc = power_supply_get_drvdata(psy);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
u8 usblim = pcf50633_reg_read(mbc->pcf, PCF50633_REG_MBCC7) &
|
u8 usblim = pcf50633_reg_read(mbc->pcf, PCF50633_REG_MBCC7) &
|
||||||
PCF50633_MBCC7_USB_MASK;
|
PCF50633_MBCC7_USB_MASK;
|
||||||
|
@ -330,7 +329,7 @@ static int ac_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 pcf50633_mbc *mbc = container_of(psy, struct pcf50633_mbc, ac);
|
struct pcf50633_mbc *mbc = power_supply_get_drvdata(psy);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
u8 usblim = pcf50633_reg_read(mbc->pcf, PCF50633_REG_MBCC7) &
|
u8 usblim = pcf50633_reg_read(mbc->pcf, PCF50633_REG_MBCC7) &
|
||||||
PCF50633_MBCC7_USB_MASK;
|
PCF50633_MBCC7_USB_MASK;
|
||||||
|
@ -366,8 +365,33 @@ static const u8 mbc_irq_handlers[] = {
|
||||||
PCF50633_IRQ_LOWBAT,
|
PCF50633_IRQ_LOWBAT,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc pcf50633_mbc_adapter_desc = {
|
||||||
|
.name = "adapter",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = power_props,
|
||||||
|
.num_properties = ARRAY_SIZE(power_props),
|
||||||
|
.get_property = &adapter_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc pcf50633_mbc_usb_desc = {
|
||||||
|
.name = "usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = power_props,
|
||||||
|
.num_properties = ARRAY_SIZE(power_props),
|
||||||
|
.get_property = usb_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc pcf50633_mbc_ac_desc = {
|
||||||
|
.name = "ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = power_props,
|
||||||
|
.num_properties = ARRAY_SIZE(power_props),
|
||||||
|
.get_property = ac_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
static int pcf50633_mbc_probe(struct platform_device *pdev)
|
static int pcf50633_mbc_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct pcf50633_mbc *mbc;
|
struct pcf50633_mbc *mbc;
|
||||||
int ret;
|
int ret;
|
||||||
int i;
|
int i;
|
||||||
|
@ -385,49 +409,36 @@ static int pcf50633_mbc_probe(struct platform_device *pdev)
|
||||||
pcf50633_register_irq(mbc->pcf, mbc_irq_handlers[i],
|
pcf50633_register_irq(mbc->pcf, mbc_irq_handlers[i],
|
||||||
pcf50633_mbc_irq_handler, mbc);
|
pcf50633_mbc_irq_handler, mbc);
|
||||||
|
|
||||||
|
psy_cfg.supplied_to = mbc->pcf->pdata->batteries;
|
||||||
|
psy_cfg.num_supplicants = mbc->pcf->pdata->num_batteries;
|
||||||
|
psy_cfg.drv_data = mbc;
|
||||||
|
|
||||||
/* Create power supplies */
|
/* Create power supplies */
|
||||||
mbc->adapter.name = "adapter";
|
mbc->adapter = power_supply_register(&pdev->dev,
|
||||||
mbc->adapter.type = POWER_SUPPLY_TYPE_MAINS;
|
&pcf50633_mbc_adapter_desc,
|
||||||
mbc->adapter.properties = power_props;
|
&psy_cfg);
|
||||||
mbc->adapter.num_properties = ARRAY_SIZE(power_props);
|
if (IS_ERR(mbc->adapter)) {
|
||||||
mbc->adapter.get_property = &adapter_get_property;
|
|
||||||
mbc->adapter.supplied_to = mbc->pcf->pdata->batteries;
|
|
||||||
mbc->adapter.num_supplicants = mbc->pcf->pdata->num_batteries;
|
|
||||||
|
|
||||||
mbc->usb.name = "usb";
|
|
||||||
mbc->usb.type = POWER_SUPPLY_TYPE_USB;
|
|
||||||
mbc->usb.properties = power_props;
|
|
||||||
mbc->usb.num_properties = ARRAY_SIZE(power_props);
|
|
||||||
mbc->usb.get_property = usb_get_property;
|
|
||||||
mbc->usb.supplied_to = mbc->pcf->pdata->batteries;
|
|
||||||
mbc->usb.num_supplicants = mbc->pcf->pdata->num_batteries;
|
|
||||||
|
|
||||||
mbc->ac.name = "ac";
|
|
||||||
mbc->ac.type = POWER_SUPPLY_TYPE_MAINS;
|
|
||||||
mbc->ac.properties = power_props;
|
|
||||||
mbc->ac.num_properties = ARRAY_SIZE(power_props);
|
|
||||||
mbc->ac.get_property = ac_get_property;
|
|
||||||
mbc->ac.supplied_to = mbc->pcf->pdata->batteries;
|
|
||||||
mbc->ac.num_supplicants = mbc->pcf->pdata->num_batteries;
|
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &mbc->adapter);
|
|
||||||
if (ret) {
|
|
||||||
dev_err(mbc->pcf->dev, "failed to register adapter\n");
|
dev_err(mbc->pcf->dev, "failed to register adapter\n");
|
||||||
|
ret = PTR_ERR(mbc->adapter);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &mbc->usb);
|
mbc->usb = power_supply_register(&pdev->dev, &pcf50633_mbc_usb_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(mbc->usb)) {
|
||||||
dev_err(mbc->pcf->dev, "failed to register usb\n");
|
dev_err(mbc->pcf->dev, "failed to register usb\n");
|
||||||
power_supply_unregister(&mbc->adapter);
|
power_supply_unregister(mbc->adapter);
|
||||||
|
ret = PTR_ERR(mbc->usb);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &mbc->ac);
|
mbc->ac = power_supply_register(&pdev->dev, &pcf50633_mbc_ac_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(mbc->ac)) {
|
||||||
dev_err(mbc->pcf->dev, "failed to register ac\n");
|
dev_err(mbc->pcf->dev, "failed to register ac\n");
|
||||||
power_supply_unregister(&mbc->adapter);
|
power_supply_unregister(mbc->adapter);
|
||||||
power_supply_unregister(&mbc->usb);
|
power_supply_unregister(mbc->usb);
|
||||||
|
ret = PTR_ERR(mbc->ac);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -454,9 +465,9 @@ static int pcf50633_mbc_remove(struct platform_device *pdev)
|
||||||
pcf50633_free_irq(mbc->pcf, mbc_irq_handlers[i]);
|
pcf50633_free_irq(mbc->pcf, mbc_irq_handlers[i]);
|
||||||
|
|
||||||
sysfs_remove_group(&pdev->dev.kobj, &mbc_attr_group);
|
sysfs_remove_group(&pdev->dev.kobj, &mbc_attr_group);
|
||||||
power_supply_unregister(&mbc->usb);
|
power_supply_unregister(mbc->usb);
|
||||||
power_supply_unregister(&mbc->adapter);
|
power_supply_unregister(mbc->adapter);
|
||||||
power_supply_unregister(&mbc->ac);
|
power_supply_unregister(mbc->ac);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,6 +34,7 @@ static struct timer_list charger_timer;
|
||||||
static struct timer_list supply_timer;
|
static struct timer_list supply_timer;
|
||||||
static struct timer_list polling_timer;
|
static struct timer_list polling_timer;
|
||||||
static int polling;
|
static int polling;
|
||||||
|
static struct power_supply *pda_psy_ac, *pda_psy_usb;
|
||||||
|
|
||||||
#if IS_ENABLED(CONFIG_USB_PHY)
|
#if IS_ENABLED(CONFIG_USB_PHY)
|
||||||
static struct usb_phy *transceiver;
|
static struct usb_phy *transceiver;
|
||||||
|
@ -58,7 +59,7 @@ static int pda_power_get_property(struct power_supply *psy,
|
||||||
{
|
{
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_ONLINE:
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
if (psy->type == POWER_SUPPLY_TYPE_MAINS)
|
if (psy->desc->type == POWER_SUPPLY_TYPE_MAINS)
|
||||||
val->intval = pdata->is_ac_online ?
|
val->intval = pdata->is_ac_online ?
|
||||||
pdata->is_ac_online() : 0;
|
pdata->is_ac_online() : 0;
|
||||||
else
|
else
|
||||||
|
@ -80,21 +81,17 @@ static char *pda_power_supplied_to[] = {
|
||||||
"backup-battery",
|
"backup-battery",
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply pda_psy_ac = {
|
static const struct power_supply_desc pda_psy_ac_desc = {
|
||||||
.name = "ac",
|
.name = "ac",
|
||||||
.type = POWER_SUPPLY_TYPE_MAINS,
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
.supplied_to = pda_power_supplied_to,
|
|
||||||
.num_supplicants = ARRAY_SIZE(pda_power_supplied_to),
|
|
||||||
.properties = pda_power_props,
|
.properties = pda_power_props,
|
||||||
.num_properties = ARRAY_SIZE(pda_power_props),
|
.num_properties = ARRAY_SIZE(pda_power_props),
|
||||||
.get_property = pda_power_get_property,
|
.get_property = pda_power_get_property,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply pda_psy_usb = {
|
static const struct power_supply_desc pda_psy_usb_desc = {
|
||||||
.name = "usb",
|
.name = "usb",
|
||||||
.type = POWER_SUPPLY_TYPE_USB,
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
.supplied_to = pda_power_supplied_to,
|
|
||||||
.num_supplicants = ARRAY_SIZE(pda_power_supplied_to),
|
|
||||||
.properties = pda_power_props,
|
.properties = pda_power_props,
|
||||||
.num_properties = ARRAY_SIZE(pda_power_props),
|
.num_properties = ARRAY_SIZE(pda_power_props),
|
||||||
.get_property = pda_power_get_property,
|
.get_property = pda_power_get_property,
|
||||||
|
@ -147,12 +144,12 @@ static void supply_timer_func(unsigned long unused)
|
||||||
{
|
{
|
||||||
if (ac_status == PDA_PSY_TO_CHANGE) {
|
if (ac_status == PDA_PSY_TO_CHANGE) {
|
||||||
ac_status = new_ac_status;
|
ac_status = new_ac_status;
|
||||||
power_supply_changed(&pda_psy_ac);
|
power_supply_changed(pda_psy_ac);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (usb_status == PDA_PSY_TO_CHANGE) {
|
if (usb_status == PDA_PSY_TO_CHANGE) {
|
||||||
usb_status = new_usb_status;
|
usb_status = new_usb_status;
|
||||||
power_supply_changed(&pda_psy_usb);
|
power_supply_changed(pda_psy_usb);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,9 +173,9 @@ static void charger_timer_func(unsigned long unused)
|
||||||
|
|
||||||
static irqreturn_t power_changed_isr(int irq, void *power_supply)
|
static irqreturn_t power_changed_isr(int irq, void *power_supply)
|
||||||
{
|
{
|
||||||
if (power_supply == &pda_psy_ac)
|
if (power_supply == pda_psy_ac)
|
||||||
ac_status = PDA_PSY_TO_CHANGE;
|
ac_status = PDA_PSY_TO_CHANGE;
|
||||||
else if (power_supply == &pda_psy_usb)
|
else if (power_supply == pda_psy_usb)
|
||||||
usb_status = PDA_PSY_TO_CHANGE;
|
usb_status = PDA_PSY_TO_CHANGE;
|
||||||
else
|
else
|
||||||
return IRQ_NONE;
|
return IRQ_NONE;
|
||||||
|
@ -262,6 +259,7 @@ static int otg_handle_notification(struct notifier_block *nb,
|
||||||
|
|
||||||
static int pda_power_probe(struct platform_device *pdev)
|
static int pda_power_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
dev = &pdev->dev;
|
dev = &pdev->dev;
|
||||||
|
@ -309,10 +307,11 @@ static int pda_power_probe(struct platform_device *pdev)
|
||||||
usb_irq = platform_get_resource_byname(pdev, IORESOURCE_IRQ, "usb");
|
usb_irq = platform_get_resource_byname(pdev, IORESOURCE_IRQ, "usb");
|
||||||
|
|
||||||
if (pdata->supplied_to) {
|
if (pdata->supplied_to) {
|
||||||
pda_psy_ac.supplied_to = pdata->supplied_to;
|
psy_cfg.supplied_to = pdata->supplied_to;
|
||||||
pda_psy_ac.num_supplicants = pdata->num_supplicants;
|
psy_cfg.num_supplicants = pdata->num_supplicants;
|
||||||
pda_psy_usb.supplied_to = pdata->supplied_to;
|
} else {
|
||||||
pda_psy_usb.num_supplicants = pdata->num_supplicants;
|
psy_cfg.supplied_to = pda_power_supplied_to;
|
||||||
|
psy_cfg.num_supplicants = ARRAY_SIZE(pda_power_supplied_to);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if IS_ENABLED(CONFIG_USB_PHY)
|
#if IS_ENABLED(CONFIG_USB_PHY)
|
||||||
|
@ -326,17 +325,19 @@ static int pda_power_probe(struct platform_device *pdev)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (pdata->is_ac_online) {
|
if (pdata->is_ac_online) {
|
||||||
ret = power_supply_register(&pdev->dev, &pda_psy_ac);
|
pda_psy_ac = power_supply_register(&pdev->dev,
|
||||||
if (ret) {
|
&pda_psy_ac_desc, &psy_cfg);
|
||||||
|
if (IS_ERR(pda_psy_ac)) {
|
||||||
dev_err(dev, "failed to register %s power supply\n",
|
dev_err(dev, "failed to register %s power supply\n",
|
||||||
pda_psy_ac.name);
|
pda_psy_ac_desc.name);
|
||||||
|
ret = PTR_ERR(pda_psy_ac);
|
||||||
goto ac_supply_failed;
|
goto ac_supply_failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ac_irq) {
|
if (ac_irq) {
|
||||||
ret = request_irq(ac_irq->start, power_changed_isr,
|
ret = request_irq(ac_irq->start, power_changed_isr,
|
||||||
get_irq_flags(ac_irq), ac_irq->name,
|
get_irq_flags(ac_irq), ac_irq->name,
|
||||||
&pda_psy_ac);
|
pda_psy_ac);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev, "request ac irq failed\n");
|
dev_err(dev, "request ac irq failed\n");
|
||||||
goto ac_irq_failed;
|
goto ac_irq_failed;
|
||||||
|
@ -347,17 +348,20 @@ static int pda_power_probe(struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pdata->is_usb_online) {
|
if (pdata->is_usb_online) {
|
||||||
ret = power_supply_register(&pdev->dev, &pda_psy_usb);
|
pda_psy_usb = power_supply_register(&pdev->dev,
|
||||||
if (ret) {
|
&pda_psy_usb_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(pda_psy_usb)) {
|
||||||
dev_err(dev, "failed to register %s power supply\n",
|
dev_err(dev, "failed to register %s power supply\n",
|
||||||
pda_psy_usb.name);
|
pda_psy_usb_desc.name);
|
||||||
|
ret = PTR_ERR(pda_psy_usb);
|
||||||
goto usb_supply_failed;
|
goto usb_supply_failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (usb_irq) {
|
if (usb_irq) {
|
||||||
ret = request_irq(usb_irq->start, power_changed_isr,
|
ret = request_irq(usb_irq->start, power_changed_isr,
|
||||||
get_irq_flags(usb_irq),
|
get_irq_flags(usb_irq),
|
||||||
usb_irq->name, &pda_psy_usb);
|
usb_irq->name, pda_psy_usb);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(dev, "request usb irq failed\n");
|
dev_err(dev, "request usb irq failed\n");
|
||||||
goto usb_irq_failed;
|
goto usb_irq_failed;
|
||||||
|
@ -394,21 +398,21 @@ static int pda_power_probe(struct platform_device *pdev)
|
||||||
#if IS_ENABLED(CONFIG_USB_PHY)
|
#if IS_ENABLED(CONFIG_USB_PHY)
|
||||||
otg_reg_notifier_failed:
|
otg_reg_notifier_failed:
|
||||||
if (pdata->is_usb_online && usb_irq)
|
if (pdata->is_usb_online && usb_irq)
|
||||||
free_irq(usb_irq->start, &pda_psy_usb);
|
free_irq(usb_irq->start, pda_psy_usb);
|
||||||
#endif
|
#endif
|
||||||
usb_irq_failed:
|
usb_irq_failed:
|
||||||
if (pdata->is_usb_online)
|
if (pdata->is_usb_online)
|
||||||
power_supply_unregister(&pda_psy_usb);
|
power_supply_unregister(pda_psy_usb);
|
||||||
usb_supply_failed:
|
usb_supply_failed:
|
||||||
if (pdata->is_ac_online && ac_irq)
|
if (pdata->is_ac_online && ac_irq)
|
||||||
free_irq(ac_irq->start, &pda_psy_ac);
|
free_irq(ac_irq->start, pda_psy_ac);
|
||||||
#if IS_ENABLED(CONFIG_USB_PHY)
|
#if IS_ENABLED(CONFIG_USB_PHY)
|
||||||
if (!IS_ERR_OR_NULL(transceiver))
|
if (!IS_ERR_OR_NULL(transceiver))
|
||||||
usb_put_phy(transceiver);
|
usb_put_phy(transceiver);
|
||||||
#endif
|
#endif
|
||||||
ac_irq_failed:
|
ac_irq_failed:
|
||||||
if (pdata->is_ac_online)
|
if (pdata->is_ac_online)
|
||||||
power_supply_unregister(&pda_psy_ac);
|
power_supply_unregister(pda_psy_ac);
|
||||||
ac_supply_failed:
|
ac_supply_failed:
|
||||||
if (ac_draw) {
|
if (ac_draw) {
|
||||||
regulator_put(ac_draw);
|
regulator_put(ac_draw);
|
||||||
|
@ -424,9 +428,9 @@ wrongid:
|
||||||
static int pda_power_remove(struct platform_device *pdev)
|
static int pda_power_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
if (pdata->is_usb_online && usb_irq)
|
if (pdata->is_usb_online && usb_irq)
|
||||||
free_irq(usb_irq->start, &pda_psy_usb);
|
free_irq(usb_irq->start, pda_psy_usb);
|
||||||
if (pdata->is_ac_online && ac_irq)
|
if (pdata->is_ac_online && ac_irq)
|
||||||
free_irq(ac_irq->start, &pda_psy_ac);
|
free_irq(ac_irq->start, pda_psy_ac);
|
||||||
|
|
||||||
if (polling)
|
if (polling)
|
||||||
del_timer_sync(&polling_timer);
|
del_timer_sync(&polling_timer);
|
||||||
|
@ -434,9 +438,9 @@ static int pda_power_remove(struct platform_device *pdev)
|
||||||
del_timer_sync(&supply_timer);
|
del_timer_sync(&supply_timer);
|
||||||
|
|
||||||
if (pdata->is_usb_online)
|
if (pdata->is_usb_online)
|
||||||
power_supply_unregister(&pda_psy_usb);
|
power_supply_unregister(pda_psy_usb);
|
||||||
if (pdata->is_ac_online)
|
if (pdata->is_ac_online)
|
||||||
power_supply_unregister(&pda_psy_ac);
|
power_supply_unregister(pda_psy_ac);
|
||||||
#if IS_ENABLED(CONFIG_USB_PHY)
|
#if IS_ENABLED(CONFIG_USB_PHY)
|
||||||
if (!IS_ERR_OR_NULL(transceiver))
|
if (!IS_ERR_OR_NULL(transceiver))
|
||||||
usb_put_phy(transceiver);
|
usb_put_phy(transceiver);
|
||||||
|
|
|
@ -216,7 +216,7 @@ static int pm2xxx_charger_ovv_mngt(struct pm2xxx_charger *pm2, int val)
|
||||||
{
|
{
|
||||||
dev_err(pm2->dev, "Overvoltage detected\n");
|
dev_err(pm2->dev, "Overvoltage detected\n");
|
||||||
pm2->flags.ovv = true;
|
pm2->flags.ovv = true;
|
||||||
power_supply_changed(&pm2->ac_chg.psy);
|
power_supply_changed(pm2->ac_chg.psy);
|
||||||
|
|
||||||
/* Schedule a new HW failure check */
|
/* Schedule a new HW failure check */
|
||||||
queue_delayed_work(pm2->charger_wq, &pm2->check_hw_failure_work, 0);
|
queue_delayed_work(pm2->charger_wq, &pm2->check_hw_failure_work, 0);
|
||||||
|
@ -229,7 +229,7 @@ static int pm2xxx_charger_wd_exp_mngt(struct pm2xxx_charger *pm2, int val)
|
||||||
dev_dbg(pm2->dev , "20 minutes watchdog expired\n");
|
dev_dbg(pm2->dev , "20 minutes watchdog expired\n");
|
||||||
|
|
||||||
pm2->ac.wd_expired = true;
|
pm2->ac.wd_expired = true;
|
||||||
power_supply_changed(&pm2->ac_chg.psy);
|
power_supply_changed(pm2->ac_chg.psy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -573,7 +573,7 @@ static int pm2xxx_charger_update_charger_current(struct ux500_charger *charger,
|
||||||
struct pm2xxx_charger *pm2;
|
struct pm2xxx_charger *pm2;
|
||||||
u8 val;
|
u8 val;
|
||||||
|
|
||||||
if (charger->psy.type == POWER_SUPPLY_TYPE_MAINS)
|
if (charger->psy->desc->type == POWER_SUPPLY_TYPE_MAINS)
|
||||||
pm2 = to_pm2xxx_charger_ac_device_info(charger);
|
pm2 = to_pm2xxx_charger_ac_device_info(charger);
|
||||||
else
|
else
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
@ -816,7 +816,7 @@ static int pm2xxx_charger_ac_en(struct ux500_charger *charger,
|
||||||
|
|
||||||
dev_dbg(pm2->dev, "PM2301: " "Disabled AC charging\n");
|
dev_dbg(pm2->dev, "PM2301: " "Disabled AC charging\n");
|
||||||
}
|
}
|
||||||
power_supply_changed(&pm2->ac_chg.psy);
|
power_supply_changed(pm2->ac_chg.psy);
|
||||||
|
|
||||||
error_occured:
|
error_occured:
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -827,7 +827,7 @@ static int pm2xxx_charger_watchdog_kick(struct ux500_charger *charger)
|
||||||
int ret;
|
int ret;
|
||||||
struct pm2xxx_charger *pm2;
|
struct pm2xxx_charger *pm2;
|
||||||
|
|
||||||
if (charger->psy.type == POWER_SUPPLY_TYPE_MAINS)
|
if (charger->psy->desc->type == POWER_SUPPLY_TYPE_MAINS)
|
||||||
pm2 = to_pm2xxx_charger_ac_device_info(charger);
|
pm2 = to_pm2xxx_charger_ac_device_info(charger);
|
||||||
else
|
else
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
@ -845,8 +845,8 @@ static void pm2xxx_charger_ac_work(struct work_struct *work)
|
||||||
struct pm2xxx_charger, ac_work);
|
struct pm2xxx_charger, ac_work);
|
||||||
|
|
||||||
|
|
||||||
power_supply_changed(&pm2->ac_chg.psy);
|
power_supply_changed(pm2->ac_chg.psy);
|
||||||
sysfs_notify(&pm2->ac_chg.psy.dev->kobj, NULL, "present");
|
sysfs_notify(&pm2->ac_chg.psy->dev.kobj, NULL, "present");
|
||||||
};
|
};
|
||||||
|
|
||||||
static void pm2xxx_charger_check_hw_failure_work(struct work_struct *work)
|
static void pm2xxx_charger_check_hw_failure_work(struct work_struct *work)
|
||||||
|
@ -862,7 +862,7 @@ static void pm2xxx_charger_check_hw_failure_work(struct work_struct *work)
|
||||||
if (!(reg_value & (PM2XXX_INT4_S_ITVPWR1OVV |
|
if (!(reg_value & (PM2XXX_INT4_S_ITVPWR1OVV |
|
||||||
PM2XXX_INT4_S_ITVPWR2OVV))) {
|
PM2XXX_INT4_S_ITVPWR2OVV))) {
|
||||||
pm2->flags.ovv = false;
|
pm2->flags.ovv = false;
|
||||||
power_supply_changed(&pm2->ac_chg.psy);
|
power_supply_changed(pm2->ac_chg.psy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -895,7 +895,7 @@ static void pm2xxx_charger_check_main_thermal_prot_work(
|
||||||
| PM2XXX_INT5_S_ITTHERMALSHUTDOWNFALL))
|
| PM2XXX_INT5_S_ITTHERMALSHUTDOWNFALL))
|
||||||
pm2->flags.main_thermal_prot = false;
|
pm2->flags.main_thermal_prot = false;
|
||||||
|
|
||||||
power_supply_changed(&pm2->ac_chg.psy);
|
power_supply_changed(pm2->ac_chg.psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct pm2xxx_interrupts pm2xxx_int = {
|
static struct pm2xxx_interrupts pm2xxx_int = {
|
||||||
|
@ -989,6 +989,7 @@ static int pm2xxx_wall_charger_probe(struct i2c_client *i2c_client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
struct pm2xxx_platform_data *pl_data = i2c_client->dev.platform_data;
|
struct pm2xxx_platform_data *pl_data = i2c_client->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct pm2xxx_charger *pm2;
|
struct pm2xxx_charger *pm2;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
u8 val;
|
u8 val;
|
||||||
|
@ -1042,13 +1043,14 @@ static int pm2xxx_wall_charger_probe(struct i2c_client *i2c_client,
|
||||||
|
|
||||||
/* AC supply */
|
/* AC supply */
|
||||||
/* power_supply base class */
|
/* power_supply base class */
|
||||||
pm2->ac_chg.psy.name = pm2->pdata->label;
|
pm2->ac_chg_desc.name = pm2->pdata->label;
|
||||||
pm2->ac_chg.psy.type = POWER_SUPPLY_TYPE_MAINS;
|
pm2->ac_chg_desc.type = POWER_SUPPLY_TYPE_MAINS;
|
||||||
pm2->ac_chg.psy.properties = pm2xxx_charger_ac_props;
|
pm2->ac_chg_desc.properties = pm2xxx_charger_ac_props;
|
||||||
pm2->ac_chg.psy.num_properties = ARRAY_SIZE(pm2xxx_charger_ac_props);
|
pm2->ac_chg_desc.num_properties = ARRAY_SIZE(pm2xxx_charger_ac_props);
|
||||||
pm2->ac_chg.psy.get_property = pm2xxx_charger_ac_get_property;
|
pm2->ac_chg_desc.get_property = pm2xxx_charger_ac_get_property;
|
||||||
pm2->ac_chg.psy.supplied_to = pm2->pdata->supplied_to;
|
|
||||||
pm2->ac_chg.psy.num_supplicants = pm2->pdata->num_supplicants;
|
psy_cfg.supplied_to = pm2->pdata->supplied_to;
|
||||||
|
psy_cfg.num_supplicants = pm2->pdata->num_supplicants;
|
||||||
/* pm2xxx_charger sub-class */
|
/* pm2xxx_charger sub-class */
|
||||||
pm2->ac_chg.ops.enable = &pm2xxx_charger_ac_en;
|
pm2->ac_chg.ops.enable = &pm2xxx_charger_ac_en;
|
||||||
pm2->ac_chg.ops.kick_wd = &pm2xxx_charger_watchdog_kick;
|
pm2->ac_chg.ops.kick_wd = &pm2xxx_charger_watchdog_kick;
|
||||||
|
@ -1093,9 +1095,11 @@ static int pm2xxx_wall_charger_probe(struct i2c_client *i2c_client,
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Register AC charger class */
|
/* Register AC charger class */
|
||||||
ret = power_supply_register(pm2->dev, &pm2->ac_chg.psy);
|
pm2->ac_chg.psy = power_supply_register(pm2->dev, &pm2->ac_chg_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(pm2->ac_chg.psy)) {
|
||||||
dev_err(pm2->dev, "failed to register AC charger\n");
|
dev_err(pm2->dev, "failed to register AC charger\n");
|
||||||
|
ret = PTR_ERR(pm2->ac_chg.psy);
|
||||||
goto free_regulator;
|
goto free_regulator;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1167,8 +1171,8 @@ static int pm2xxx_wall_charger_probe(struct i2c_client *i2c_client,
|
||||||
ab8500_override_turn_on_stat(~AB8500_POW_KEY_1_ON,
|
ab8500_override_turn_on_stat(~AB8500_POW_KEY_1_ON,
|
||||||
AB8500_MAIN_CH_DET);
|
AB8500_MAIN_CH_DET);
|
||||||
pm2->ac_conn = true;
|
pm2->ac_conn = true;
|
||||||
power_supply_changed(&pm2->ac_chg.psy);
|
power_supply_changed(pm2->ac_chg.psy);
|
||||||
sysfs_notify(&pm2->ac_chg.psy.dev->kobj, NULL, "present");
|
sysfs_notify(&pm2->ac_chg.psy->dev.kobj, NULL, "present");
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1183,7 +1187,7 @@ unregister_pm2xxx_interrupt:
|
||||||
free_irq(gpio_to_irq(pm2->pdata->gpio_irq_number), pm2);
|
free_irq(gpio_to_irq(pm2->pdata->gpio_irq_number), pm2);
|
||||||
unregister_pm2xxx_charger:
|
unregister_pm2xxx_charger:
|
||||||
/* unregister power supply */
|
/* unregister power supply */
|
||||||
power_supply_unregister(&pm2->ac_chg.psy);
|
power_supply_unregister(pm2->ac_chg.psy);
|
||||||
free_regulator:
|
free_regulator:
|
||||||
/* disable the regulator */
|
/* disable the regulator */
|
||||||
regulator_put(pm2->regu);
|
regulator_put(pm2->regu);
|
||||||
|
@ -1218,7 +1222,7 @@ static int pm2xxx_wall_charger_remove(struct i2c_client *i2c_client)
|
||||||
/* disable the regulator */
|
/* disable the regulator */
|
||||||
regulator_put(pm2->regu);
|
regulator_put(pm2->regu);
|
||||||
|
|
||||||
power_supply_unregister(&pm2->ac_chg.psy);
|
power_supply_unregister(pm2->ac_chg.psy);
|
||||||
|
|
||||||
if (gpio_is_valid(pm2->lpn_pin))
|
if (gpio_is_valid(pm2->lpn_pin))
|
||||||
gpio_free(pm2->lpn_pin);
|
gpio_free(pm2->lpn_pin);
|
||||||
|
|
|
@ -486,6 +486,7 @@ struct pm2xxx_charger {
|
||||||
struct work_struct check_main_thermal_prot_work;
|
struct work_struct check_main_thermal_prot_work;
|
||||||
struct delayed_work check_hw_failure_work;
|
struct delayed_work check_hw_failure_work;
|
||||||
struct ux500_charger ac_chg;
|
struct ux500_charger ac_chg;
|
||||||
|
struct power_supply_desc ac_chg_desc;
|
||||||
struct pm2xxx_charger_event_flags flags;
|
struct pm2xxx_charger_event_flags flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -17,13 +17,14 @@
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
|
||||||
static struct pmu_battery_dev {
|
static struct pmu_battery_dev {
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
struct power_supply_desc bat_desc;
|
||||||
struct pmu_battery_info *pbi;
|
struct pmu_battery_info *pbi;
|
||||||
char name[16];
|
char name[16];
|
||||||
int propval;
|
int propval;
|
||||||
} *pbats[PMU_MAX_BATTERIES];
|
} *pbats[PMU_MAX_BATTERIES];
|
||||||
|
|
||||||
#define to_pmu_battery_dev(x) container_of(x, struct pmu_battery_dev, bat)
|
#define to_pmu_battery_dev(x) power_supply_get_drvdata(x)
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
* Power
|
* Power
|
||||||
|
@ -49,7 +50,7 @@ static enum power_supply_property pmu_ac_props[] = {
|
||||||
POWER_SUPPLY_PROP_ONLINE,
|
POWER_SUPPLY_PROP_ONLINE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply pmu_ac = {
|
static const struct power_supply_desc pmu_ac_desc = {
|
||||||
.name = "pmu-ac",
|
.name = "pmu-ac",
|
||||||
.type = POWER_SUPPLY_TYPE_MAINS,
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
.properties = pmu_ac_props,
|
.properties = pmu_ac_props,
|
||||||
|
@ -57,6 +58,8 @@ static struct power_supply pmu_ac = {
|
||||||
.get_property = pmu_get_ac_prop,
|
.get_property = pmu_get_ac_prop,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct power_supply *pmu_ac;
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
* Battery properties
|
* Battery properties
|
||||||
*********************************************************************/
|
*********************************************************************/
|
||||||
|
@ -142,7 +145,7 @@ static struct platform_device *bat_pdev;
|
||||||
|
|
||||||
static int __init pmu_bat_init(void)
|
static int __init pmu_bat_init(void)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret = 0;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
bat_pdev = platform_device_register_simple("pmu-battery",
|
bat_pdev = platform_device_register_simple("pmu-battery",
|
||||||
|
@ -152,25 +155,32 @@ static int __init pmu_bat_init(void)
|
||||||
goto pdev_register_failed;
|
goto pdev_register_failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(&bat_pdev->dev, &pmu_ac);
|
pmu_ac = power_supply_register(&bat_pdev->dev, &pmu_ac_desc, NULL);
|
||||||
if (ret)
|
if (IS_ERR(pmu_ac)) {
|
||||||
|
ret = PTR_ERR(pmu_ac);
|
||||||
goto ac_register_failed;
|
goto ac_register_failed;
|
||||||
|
}
|
||||||
|
|
||||||
for (i = 0; i < pmu_battery_count; i++) {
|
for (i = 0; i < pmu_battery_count; i++) {
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct pmu_battery_dev *pbat = kzalloc(sizeof(*pbat),
|
struct pmu_battery_dev *pbat = kzalloc(sizeof(*pbat),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!pbat)
|
if (!pbat)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
sprintf(pbat->name, "PMU_battery_%d", i);
|
sprintf(pbat->name, "PMU_battery_%d", i);
|
||||||
pbat->bat.name = pbat->name;
|
pbat->bat_desc.name = pbat->name;
|
||||||
pbat->bat.properties = pmu_bat_props;
|
pbat->bat_desc.properties = pmu_bat_props;
|
||||||
pbat->bat.num_properties = ARRAY_SIZE(pmu_bat_props);
|
pbat->bat_desc.num_properties = ARRAY_SIZE(pmu_bat_props);
|
||||||
pbat->bat.get_property = pmu_bat_get_property;
|
pbat->bat_desc.get_property = pmu_bat_get_property;
|
||||||
pbat->pbi = &pmu_batteries[i];
|
pbat->pbi = &pmu_batteries[i];
|
||||||
|
psy_cfg.drv_data = pbat;
|
||||||
|
|
||||||
ret = power_supply_register(&bat_pdev->dev, &pbat->bat);
|
pbat->bat = power_supply_register(&bat_pdev->dev,
|
||||||
if (ret) {
|
&pbat->bat_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(pbat->bat)) {
|
||||||
|
ret = PTR_ERR(pbat->bat);
|
||||||
kfree(pbat);
|
kfree(pbat);
|
||||||
goto battery_register_failed;
|
goto battery_register_failed;
|
||||||
}
|
}
|
||||||
|
@ -183,10 +193,10 @@ battery_register_failed:
|
||||||
while (i--) {
|
while (i--) {
|
||||||
if (!pbats[i])
|
if (!pbats[i])
|
||||||
continue;
|
continue;
|
||||||
power_supply_unregister(&pbats[i]->bat);
|
power_supply_unregister(pbats[i]->bat);
|
||||||
kfree(pbats[i]);
|
kfree(pbats[i]);
|
||||||
}
|
}
|
||||||
power_supply_unregister(&pmu_ac);
|
power_supply_unregister(pmu_ac);
|
||||||
ac_register_failed:
|
ac_register_failed:
|
||||||
platform_device_unregister(bat_pdev);
|
platform_device_unregister(bat_pdev);
|
||||||
pdev_register_failed:
|
pdev_register_failed:
|
||||||
|
@ -201,10 +211,10 @@ static void __exit pmu_bat_exit(void)
|
||||||
for (i = 0; i < PMU_MAX_BATTERIES; i++) {
|
for (i = 0; i < PMU_MAX_BATTERIES; i++) {
|
||||||
if (!pbats[i])
|
if (!pbats[i])
|
||||||
continue;
|
continue;
|
||||||
power_supply_unregister(&pbats[i]->bat);
|
power_supply_unregister(pbats[i]->bat);
|
||||||
kfree(pbats[i]);
|
kfree(pbats[i]);
|
||||||
}
|
}
|
||||||
power_supply_unregister(&pmu_ac);
|
power_supply_unregister(pmu_ac);
|
||||||
platform_device_unregister(bat_pdev);
|
platform_device_unregister(bat_pdev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -40,16 +40,16 @@ static bool __power_supply_is_supplied_by(struct power_supply *supplier,
|
||||||
|
|
||||||
/* Support both supplied_to and supplied_from modes */
|
/* Support both supplied_to and supplied_from modes */
|
||||||
if (supply->supplied_from) {
|
if (supply->supplied_from) {
|
||||||
if (!supplier->name)
|
if (!supplier->desc->name)
|
||||||
return false;
|
return false;
|
||||||
for (i = 0; i < supply->num_supplies; i++)
|
for (i = 0; i < supply->num_supplies; i++)
|
||||||
if (!strcmp(supplier->name, supply->supplied_from[i]))
|
if (!strcmp(supplier->desc->name, supply->supplied_from[i]))
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
if (!supply->name)
|
if (!supply->desc->name)
|
||||||
return false;
|
return false;
|
||||||
for (i = 0; i < supplier->num_supplicants; i++)
|
for (i = 0; i < supplier->num_supplicants; i++)
|
||||||
if (!strcmp(supplier->supplied_to[i], supply->name))
|
if (!strcmp(supplier->supplied_to[i], supply->desc->name))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,8 +62,8 @@ static int __power_supply_changed_work(struct device *dev, void *data)
|
||||||
struct power_supply *pst = dev_get_drvdata(dev);
|
struct power_supply *pst = dev_get_drvdata(dev);
|
||||||
|
|
||||||
if (__power_supply_is_supplied_by(psy, pst)) {
|
if (__power_supply_is_supplied_by(psy, pst)) {
|
||||||
if (pst->external_power_changed)
|
if (pst->desc->external_power_changed)
|
||||||
pst->external_power_changed(pst);
|
pst->desc->external_power_changed(pst);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -75,7 +75,7 @@ static void power_supply_changed_work(struct work_struct *work)
|
||||||
struct power_supply *psy = container_of(work, struct power_supply,
|
struct power_supply *psy = container_of(work, struct power_supply,
|
||||||
changed_work);
|
changed_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);
|
||||||
/*
|
/*
|
||||||
|
@ -93,7 +93,7 @@ static void power_supply_changed_work(struct work_struct *work)
|
||||||
power_supply_update_leds(psy);
|
power_supply_update_leds(psy);
|
||||||
atomic_notifier_call_chain(&power_supply_notifier,
|
atomic_notifier_call_chain(&power_supply_notifier,
|
||||||
PSY_EVENT_PROP_CHANGED, psy);
|
PSY_EVENT_PROP_CHANGED, psy);
|
||||||
kobject_uevent(&psy->dev->kobj, KOBJ_CHANGE);
|
kobject_uevent(&psy->dev.kobj, KOBJ_CHANGE);
|
||||||
spin_lock_irqsave(&psy->changed_lock, flags);
|
spin_lock_irqsave(&psy->changed_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,7 +103,7 @@ static void power_supply_changed_work(struct work_struct *work)
|
||||||
* to true.
|
* to true.
|
||||||
*/
|
*/
|
||||||
if (likely(!psy->changed))
|
if (likely(!psy->changed))
|
||||||
pm_relax(psy->dev);
|
pm_relax(&psy->dev);
|
||||||
spin_unlock_irqrestore(&psy->changed_lock, flags);
|
spin_unlock_irqrestore(&psy->changed_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -111,11 +111,11 @@ void power_supply_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
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);
|
||||||
psy->changed = true;
|
psy->changed = true;
|
||||||
pm_stay_awake(psy->dev);
|
pm_stay_awake(&psy->dev);
|
||||||
spin_unlock_irqrestore(&psy->changed_lock, flags);
|
spin_unlock_irqrestore(&psy->changed_lock, flags);
|
||||||
schedule_work(&psy->changed_work);
|
schedule_work(&psy->changed_work);
|
||||||
}
|
}
|
||||||
|
@ -138,9 +138,9 @@ static int __power_supply_populate_supplied_from(struct device *dev,
|
||||||
break;
|
break;
|
||||||
|
|
||||||
if (np == epsy->of_node) {
|
if (np == epsy->of_node) {
|
||||||
dev_info(psy->dev, "%s: Found supply : %s\n",
|
dev_info(&psy->dev, "%s: Found supply : %s\n",
|
||||||
psy->name, epsy->name);
|
psy->desc->name, epsy->desc->name);
|
||||||
psy->supplied_from[i-1] = (char *)epsy->name;
|
psy->supplied_from[i-1] = (char *)epsy->desc->name;
|
||||||
psy->num_supplies++;
|
psy->num_supplies++;
|
||||||
of_node_put(np);
|
of_node_put(np);
|
||||||
break;
|
break;
|
||||||
|
@ -158,7 +158,7 @@ static int power_supply_populate_supplied_from(struct power_supply *psy)
|
||||||
error = class_for_each_device(power_supply_class, NULL, psy,
|
error = class_for_each_device(power_supply_class, NULL, psy,
|
||||||
__power_supply_populate_supplied_from);
|
__power_supply_populate_supplied_from);
|
||||||
|
|
||||||
dev_dbg(psy->dev, "%s %d\n", __func__, error);
|
dev_dbg(&psy->dev, "%s %d\n", __func__, error);
|
||||||
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
@ -220,7 +220,7 @@ static int power_supply_check_supplies(struct power_supply *psy)
|
||||||
of_node_put(np);
|
of_node_put(np);
|
||||||
|
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_dbg(psy->dev, "Failed to find supply!\n");
|
dev_dbg(&psy->dev, "Failed to find supply!\n");
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
} while (np);
|
} while (np);
|
||||||
|
@ -230,17 +230,18 @@ static int power_supply_check_supplies(struct power_supply *psy)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* All supplies found, allocate char ** array for filling */
|
/* All supplies found, allocate char ** array for filling */
|
||||||
psy->supplied_from = devm_kzalloc(psy->dev, sizeof(psy->supplied_from),
|
psy->supplied_from = devm_kzalloc(&psy->dev, sizeof(psy->supplied_from),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!psy->supplied_from) {
|
if (!psy->supplied_from) {
|
||||||
dev_err(psy->dev, "Couldn't allocate memory for supply list\n");
|
dev_err(&psy->dev, "Couldn't allocate memory for supply list\n");
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
*psy->supplied_from = devm_kzalloc(psy->dev, sizeof(char *) * (cnt - 1),
|
*psy->supplied_from = devm_kzalloc(&psy->dev,
|
||||||
|
sizeof(char *) * (cnt - 1),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!*psy->supplied_from) {
|
if (!*psy->supplied_from) {
|
||||||
dev_err(psy->dev, "Couldn't allocate memory for supply list\n");
|
dev_err(&psy->dev, "Couldn't allocate memory for supply list\n");
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -260,7 +261,8 @@ static int __power_supply_am_i_supplied(struct device *dev, void *data)
|
||||||
struct power_supply *epsy = dev_get_drvdata(dev);
|
struct power_supply *epsy = dev_get_drvdata(dev);
|
||||||
|
|
||||||
if (__power_supply_is_supplied_by(epsy, psy))
|
if (__power_supply_is_supplied_by(epsy, psy))
|
||||||
if (!epsy->get_property(epsy, POWER_SUPPLY_PROP_ONLINE, &ret))
|
if (!epsy->desc->get_property(epsy, POWER_SUPPLY_PROP_ONLINE,
|
||||||
|
&ret))
|
||||||
return ret.intval;
|
return ret.intval;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -273,7 +275,7 @@ int power_supply_am_i_supplied(struct power_supply *psy)
|
||||||
error = class_for_each_device(power_supply_class, NULL, psy,
|
error = class_for_each_device(power_supply_class, NULL, psy,
|
||||||
__power_supply_am_i_supplied);
|
__power_supply_am_i_supplied);
|
||||||
|
|
||||||
dev_dbg(psy->dev, "%s %d\n", __func__, error);
|
dev_dbg(&psy->dev, "%s %d\n", __func__, error);
|
||||||
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
@ -286,8 +288,9 @@ static int __power_supply_is_system_supplied(struct device *dev, void *data)
|
||||||
unsigned int *count = data;
|
unsigned int *count = data;
|
||||||
|
|
||||||
(*count)++;
|
(*count)++;
|
||||||
if (psy->type != POWER_SUPPLY_TYPE_BATTERY)
|
if (psy->desc->type != POWER_SUPPLY_TYPE_BATTERY)
|
||||||
if (!psy->get_property(psy, POWER_SUPPLY_PROP_ONLINE, &ret))
|
if (!psy->desc->get_property(psy, POWER_SUPPLY_PROP_ONLINE,
|
||||||
|
&ret))
|
||||||
return ret.intval;
|
return ret.intval;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -314,8 +317,10 @@ EXPORT_SYMBOL_GPL(power_supply_is_system_supplied);
|
||||||
|
|
||||||
int power_supply_set_battery_charged(struct power_supply *psy)
|
int power_supply_set_battery_charged(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
if (psy->type == POWER_SUPPLY_TYPE_BATTERY && psy->set_charged) {
|
if (atomic_read(&psy->use_cnt) >= 0 &&
|
||||||
psy->set_charged(psy);
|
psy->desc->type == POWER_SUPPLY_TYPE_BATTERY &&
|
||||||
|
psy->desc->set_charged) {
|
||||||
|
psy->desc->set_charged(psy);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -328,28 +333,74 @@ static int power_supply_match_device_by_name(struct device *dev, const void *dat
|
||||||
const char *name = data;
|
const char *name = data;
|
||||||
struct power_supply *psy = dev_get_drvdata(dev);
|
struct power_supply *psy = dev_get_drvdata(dev);
|
||||||
|
|
||||||
return strcmp(psy->name, name) == 0;
|
return strcmp(psy->desc->name, name) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* power_supply_get_by_name() - Search for a power supply and returns its ref
|
||||||
|
* @name: Power supply name to fetch
|
||||||
|
*
|
||||||
|
* If power supply was found, it increases reference count for the
|
||||||
|
* internal power supply's device. The user should power_supply_put()
|
||||||
|
* after usage.
|
||||||
|
*
|
||||||
|
* Return: On success returns a reference to a power supply with
|
||||||
|
* matching name equals to @name, a NULL otherwise.
|
||||||
|
*/
|
||||||
struct power_supply *power_supply_get_by_name(const char *name)
|
struct power_supply *power_supply_get_by_name(const char *name)
|
||||||
{
|
{
|
||||||
|
struct power_supply *psy = NULL;
|
||||||
struct device *dev = class_find_device(power_supply_class, NULL, name,
|
struct device *dev = class_find_device(power_supply_class, NULL, name,
|
||||||
power_supply_match_device_by_name);
|
power_supply_match_device_by_name);
|
||||||
|
|
||||||
return dev ? dev_get_drvdata(dev) : NULL;
|
if (dev) {
|
||||||
|
psy = dev_get_drvdata(dev);
|
||||||
|
atomic_inc(&psy->use_cnt);
|
||||||
|
}
|
||||||
|
|
||||||
|
return psy;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(power_supply_get_by_name);
|
EXPORT_SYMBOL_GPL(power_supply_get_by_name);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* power_supply_put() - Drop reference obtained with power_supply_get_by_name
|
||||||
|
* @psy: Reference to put
|
||||||
|
*
|
||||||
|
* The reference to power supply should be put before unregistering
|
||||||
|
* the power supply.
|
||||||
|
*/
|
||||||
|
void power_supply_put(struct power_supply *psy)
|
||||||
|
{
|
||||||
|
might_sleep();
|
||||||
|
|
||||||
|
atomic_dec(&psy->use_cnt);
|
||||||
|
put_device(&psy->dev);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_put);
|
||||||
|
|
||||||
#ifdef CONFIG_OF
|
#ifdef CONFIG_OF
|
||||||
static int power_supply_match_device_node(struct device *dev, const void *data)
|
static int power_supply_match_device_node(struct device *dev, const void *data)
|
||||||
{
|
{
|
||||||
return dev->parent && dev->parent->of_node == data;
|
return dev->parent && dev->parent->of_node == data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* power_supply_get_by_phandle() - Search for a power supply and returns its ref
|
||||||
|
* @np: Pointer to device node holding phandle property
|
||||||
|
* @phandle_name: Name of property holding a power supply name
|
||||||
|
*
|
||||||
|
* If power supply was found, it increases reference count for the
|
||||||
|
* internal power supply's device. The user should power_supply_put()
|
||||||
|
* after usage.
|
||||||
|
*
|
||||||
|
* Return: On success returns a reference to a power supply with
|
||||||
|
* matching name equals to value under @property, NULL or ERR_PTR otherwise.
|
||||||
|
*/
|
||||||
struct power_supply *power_supply_get_by_phandle(struct device_node *np,
|
struct power_supply *power_supply_get_by_phandle(struct device_node *np,
|
||||||
const char *property)
|
const char *property)
|
||||||
{
|
{
|
||||||
struct device_node *power_supply_np;
|
struct device_node *power_supply_np;
|
||||||
|
struct power_supply *psy = NULL;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
|
|
||||||
power_supply_np = of_parse_phandle(np, property, 0);
|
power_supply_np = of_parse_phandle(np, property, 0);
|
||||||
|
@ -361,21 +412,70 @@ struct power_supply *power_supply_get_by_phandle(struct device_node *np,
|
||||||
|
|
||||||
of_node_put(power_supply_np);
|
of_node_put(power_supply_np);
|
||||||
|
|
||||||
return dev ? dev_get_drvdata(dev) : NULL;
|
if (dev) {
|
||||||
|
psy = dev_get_drvdata(dev);
|
||||||
|
atomic_inc(&psy->use_cnt);
|
||||||
|
}
|
||||||
|
|
||||||
|
return psy;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(power_supply_get_by_phandle);
|
EXPORT_SYMBOL_GPL(power_supply_get_by_phandle);
|
||||||
#endif /* CONFIG_OF */
|
#endif /* CONFIG_OF */
|
||||||
|
|
||||||
|
int power_supply_get_property(struct power_supply *psy,
|
||||||
|
enum power_supply_property psp,
|
||||||
|
union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
if (atomic_read(&psy->use_cnt) <= 0)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
return psy->desc->get_property(psy, psp, val);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_get_property);
|
||||||
|
|
||||||
|
int power_supply_set_property(struct power_supply *psy,
|
||||||
|
enum power_supply_property psp,
|
||||||
|
const union power_supply_propval *val)
|
||||||
|
{
|
||||||
|
if (atomic_read(&psy->use_cnt) <= 0 || !psy->desc->set_property)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
return psy->desc->set_property(psy, psp, val);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_set_property);
|
||||||
|
|
||||||
|
int power_supply_property_is_writeable(struct power_supply *psy,
|
||||||
|
enum power_supply_property psp)
|
||||||
|
{
|
||||||
|
if (atomic_read(&psy->use_cnt) <= 0 ||
|
||||||
|
!psy->desc->property_is_writeable)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
return psy->desc->property_is_writeable(psy, psp);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_property_is_writeable);
|
||||||
|
|
||||||
|
void power_supply_external_power_changed(struct power_supply *psy)
|
||||||
|
{
|
||||||
|
if (atomic_read(&psy->use_cnt) <= 0 ||
|
||||||
|
!psy->desc->external_power_changed)
|
||||||
|
return;
|
||||||
|
|
||||||
|
psy->desc->external_power_changed(psy);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_external_power_changed);
|
||||||
|
|
||||||
int power_supply_powers(struct power_supply *psy, struct device *dev)
|
int power_supply_powers(struct power_supply *psy, struct device *dev)
|
||||||
{
|
{
|
||||||
return sysfs_create_link(&psy->dev->kobj, &dev->kobj, "powers");
|
return sysfs_create_link(&psy->dev.kobj, &dev->kobj, "powers");
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(power_supply_powers);
|
EXPORT_SYMBOL_GPL(power_supply_powers);
|
||||||
|
|
||||||
static void power_supply_dev_release(struct device *dev)
|
static void power_supply_dev_release(struct device *dev)
|
||||||
{
|
{
|
||||||
|
struct power_supply *psy = container_of(dev, struct power_supply, dev);
|
||||||
pr_debug("device: '%s': %s\n", dev_name(dev), __func__);
|
pr_debug("device: '%s': %s\n", dev_name(dev), __func__);
|
||||||
kfree(dev);
|
kfree(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
int power_supply_reg_notifier(struct notifier_block *nb)
|
int power_supply_reg_notifier(struct notifier_block *nb)
|
||||||
|
@ -400,7 +500,7 @@ static int power_supply_read_temp(struct thermal_zone_device *tzd,
|
||||||
|
|
||||||
WARN_ON(tzd == NULL);
|
WARN_ON(tzd == NULL);
|
||||||
psy = tzd->devdata;
|
psy = tzd->devdata;
|
||||||
ret = psy->get_property(psy, POWER_SUPPLY_PROP_TEMP, &val);
|
ret = psy->desc->get_property(psy, POWER_SUPPLY_PROP_TEMP, &val);
|
||||||
|
|
||||||
/* Convert tenths of degree Celsius to milli degree Celsius. */
|
/* Convert tenths of degree Celsius to milli degree Celsius. */
|
||||||
if (!ret)
|
if (!ret)
|
||||||
|
@ -417,14 +517,14 @@ static int psy_register_thermal(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (psy->no_thermal)
|
if (psy->desc->no_thermal)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* Register battery zone device psy reports temperature */
|
/* Register battery zone device psy reports temperature */
|
||||||
for (i = 0; i < psy->num_properties; i++) {
|
for (i = 0; i < psy->desc->num_properties; i++) {
|
||||||
if (psy->properties[i] == POWER_SUPPLY_PROP_TEMP) {
|
if (psy->desc->properties[i] == POWER_SUPPLY_PROP_TEMP) {
|
||||||
psy->tzd = thermal_zone_device_register(psy->name, 0, 0,
|
psy->tzd = thermal_zone_device_register(psy->desc->name,
|
||||||
psy, &psy_tzd_ops, NULL, 0, 0);
|
0, 0, psy, &psy_tzd_ops, NULL, 0, 0);
|
||||||
return PTR_ERR_OR_ZERO(psy->tzd);
|
return PTR_ERR_OR_ZERO(psy->tzd);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -447,7 +547,7 @@ static int ps_get_max_charge_cntl_limit(struct thermal_cooling_device *tcd,
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
psy = tcd->devdata;
|
psy = tcd->devdata;
|
||||||
ret = psy->get_property(psy,
|
ret = psy->desc->get_property(psy,
|
||||||
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT_MAX, &val);
|
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT_MAX, &val);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
*state = val.intval;
|
*state = val.intval;
|
||||||
|
@ -463,7 +563,7 @@ static int ps_get_cur_chrage_cntl_limit(struct thermal_cooling_device *tcd,
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
psy = tcd->devdata;
|
psy = tcd->devdata;
|
||||||
ret = psy->get_property(psy,
|
ret = psy->desc->get_property(psy,
|
||||||
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT, &val);
|
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT, &val);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
*state = val.intval;
|
*state = val.intval;
|
||||||
|
@ -480,7 +580,7 @@ static int ps_set_cur_charge_cntl_limit(struct thermal_cooling_device *tcd,
|
||||||
|
|
||||||
psy = tcd->devdata;
|
psy = tcd->devdata;
|
||||||
val.intval = state;
|
val.intval = state;
|
||||||
ret = psy->set_property(psy,
|
ret = psy->desc->set_property(psy,
|
||||||
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT, &val);
|
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT, &val);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -497,11 +597,11 @@ static int psy_register_cooler(struct power_supply *psy)
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
/* Register for cooling device if psy can control charging */
|
/* Register for cooling device if psy can control charging */
|
||||||
for (i = 0; i < psy->num_properties; i++) {
|
for (i = 0; i < psy->desc->num_properties; i++) {
|
||||||
if (psy->properties[i] ==
|
if (psy->desc->properties[i] ==
|
||||||
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT) {
|
POWER_SUPPLY_PROP_CHARGE_CONTROL_LIMIT) {
|
||||||
psy->tcd = thermal_cooling_device_register(
|
psy->tcd = thermal_cooling_device_register(
|
||||||
(char *)psy->name,
|
(char *)psy->desc->name,
|
||||||
psy, &psy_tcd_ops);
|
psy, &psy_tcd_ops);
|
||||||
return PTR_ERR_OR_ZERO(psy->tcd);
|
return PTR_ERR_OR_ZERO(psy->tcd);
|
||||||
}
|
}
|
||||||
|
@ -535,15 +635,21 @@ static void psy_unregister_cooler(struct power_supply *psy)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static int __power_supply_register(struct device *parent,
|
static struct power_supply *__must_check
|
||||||
struct power_supply *psy, bool ws)
|
__power_supply_register(struct device *parent,
|
||||||
|
const struct power_supply_desc *desc,
|
||||||
|
const struct power_supply_config *cfg,
|
||||||
|
bool ws)
|
||||||
{
|
{
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
|
struct power_supply *psy;
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
psy = kzalloc(sizeof(*psy), GFP_KERNEL);
|
||||||
if (!dev)
|
if (!psy)
|
||||||
return -ENOMEM;
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
|
dev = &psy->dev;
|
||||||
|
|
||||||
device_initialize(dev);
|
device_initialize(dev);
|
||||||
|
|
||||||
|
@ -552,9 +658,16 @@ static int __power_supply_register(struct device *parent,
|
||||||
dev->parent = parent;
|
dev->parent = parent;
|
||||||
dev->release = power_supply_dev_release;
|
dev->release = power_supply_dev_release;
|
||||||
dev_set_drvdata(dev, psy);
|
dev_set_drvdata(dev, psy);
|
||||||
psy->dev = dev;
|
psy->desc = desc;
|
||||||
|
atomic_inc(&psy->use_cnt);
|
||||||
|
if (cfg) {
|
||||||
|
psy->drv_data = cfg->drv_data;
|
||||||
|
psy->of_node = cfg->of_node;
|
||||||
|
psy->supplied_to = cfg->supplied_to;
|
||||||
|
psy->num_supplicants = cfg->num_supplicants;
|
||||||
|
}
|
||||||
|
|
||||||
rc = dev_set_name(dev, "%s", psy->name);
|
rc = dev_set_name(dev, "%s", desc->name);
|
||||||
if (rc)
|
if (rc)
|
||||||
goto dev_set_name_failed;
|
goto dev_set_name_failed;
|
||||||
|
|
||||||
|
@ -589,7 +702,7 @@ static int __power_supply_register(struct device *parent,
|
||||||
|
|
||||||
power_supply_changed(psy);
|
power_supply_changed(psy);
|
||||||
|
|
||||||
return 0;
|
return psy;
|
||||||
|
|
||||||
create_triggers_failed:
|
create_triggers_failed:
|
||||||
psy_unregister_cooler(psy);
|
psy_unregister_cooler(psy);
|
||||||
|
@ -602,33 +715,155 @@ wakeup_init_failed:
|
||||||
check_supplies_failed:
|
check_supplies_failed:
|
||||||
dev_set_name_failed:
|
dev_set_name_failed:
|
||||||
put_device(dev);
|
put_device(dev);
|
||||||
return rc;
|
return ERR_PTR(rc);
|
||||||
}
|
}
|
||||||
|
|
||||||
int power_supply_register(struct device *parent, struct power_supply *psy)
|
/**
|
||||||
|
* power_supply_register() - Register new power supply
|
||||||
|
* @parent: Device to be a parent of power supply's device
|
||||||
|
* @desc: Description of power supply, must be valid through whole
|
||||||
|
* lifetime of this power supply
|
||||||
|
* @cfg: Run-time specific configuration accessed during registering,
|
||||||
|
* may be NULL
|
||||||
|
*
|
||||||
|
* Return: A pointer to newly allocated power_supply on success
|
||||||
|
* or ERR_PTR otherwise.
|
||||||
|
* Use power_supply_unregister() on returned power_supply pointer to release
|
||||||
|
* resources.
|
||||||
|
*/
|
||||||
|
struct power_supply *__must_check power_supply_register(struct device *parent,
|
||||||
|
const struct power_supply_desc *desc,
|
||||||
|
const struct power_supply_config *cfg)
|
||||||
{
|
{
|
||||||
return __power_supply_register(parent, psy, true);
|
return __power_supply_register(parent, desc, cfg, true);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(power_supply_register);
|
EXPORT_SYMBOL_GPL(power_supply_register);
|
||||||
|
|
||||||
int power_supply_register_no_ws(struct device *parent, struct power_supply *psy)
|
/**
|
||||||
|
* power_supply_register() - Register new non-waking-source power supply
|
||||||
|
* @parent: Device to be a parent of power supply's device
|
||||||
|
* @desc: Description of power supply, must be valid through whole
|
||||||
|
* lifetime of this power supply
|
||||||
|
* @cfg: Run-time specific configuration accessed during registering,
|
||||||
|
* may be NULL
|
||||||
|
*
|
||||||
|
* Return: A pointer to newly allocated power_supply on success
|
||||||
|
* or ERR_PTR otherwise.
|
||||||
|
* Use power_supply_unregister() on returned power_supply pointer to release
|
||||||
|
* resources.
|
||||||
|
*/
|
||||||
|
struct power_supply *__must_check
|
||||||
|
power_supply_register_no_ws(struct device *parent,
|
||||||
|
const struct power_supply_desc *desc,
|
||||||
|
const struct power_supply_config *cfg)
|
||||||
{
|
{
|
||||||
return __power_supply_register(parent, psy, false);
|
return __power_supply_register(parent, desc, cfg, false);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(power_supply_register_no_ws);
|
EXPORT_SYMBOL_GPL(power_supply_register_no_ws);
|
||||||
|
|
||||||
|
static void devm_power_supply_release(struct device *dev, void *res)
|
||||||
|
{
|
||||||
|
struct power_supply **psy = res;
|
||||||
|
|
||||||
|
power_supply_unregister(*psy);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* power_supply_register() - Register managed power supply
|
||||||
|
* @parent: Device to be a parent of power supply's device
|
||||||
|
* @desc: Description of power supply, must be valid through whole
|
||||||
|
* lifetime of this power supply
|
||||||
|
* @cfg: Run-time specific configuration accessed during registering,
|
||||||
|
* may be NULL
|
||||||
|
*
|
||||||
|
* Return: A pointer to newly allocated power_supply on success
|
||||||
|
* or ERR_PTR otherwise.
|
||||||
|
* The returned power_supply pointer will be automatically unregistered
|
||||||
|
* on driver detach.
|
||||||
|
*/
|
||||||
|
struct power_supply *__must_check
|
||||||
|
devm_power_supply_register(struct device *parent,
|
||||||
|
const struct power_supply_desc *desc,
|
||||||
|
const struct power_supply_config *cfg)
|
||||||
|
{
|
||||||
|
struct power_supply **ptr, *psy;
|
||||||
|
|
||||||
|
ptr = devres_alloc(devm_power_supply_release, sizeof(*ptr), GFP_KERNEL);
|
||||||
|
|
||||||
|
if (!ptr)
|
||||||
|
return ERR_PTR(-ENOMEM);
|
||||||
|
psy = __power_supply_register(parent, desc, cfg, true);
|
||||||
|
if (IS_ERR(psy)) {
|
||||||
|
devres_free(ptr);
|
||||||
|
} else {
|
||||||
|
*ptr = psy;
|
||||||
|
devres_add(parent, ptr);
|
||||||
|
}
|
||||||
|
return psy;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(devm_power_supply_register);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* power_supply_register() - Register managed non-waking-source power supply
|
||||||
|
* @parent: Device to be a parent of power supply's device
|
||||||
|
* @desc: Description of power supply, must be valid through whole
|
||||||
|
* lifetime of this power supply
|
||||||
|
* @cfg: Run-time specific configuration accessed during registering,
|
||||||
|
* may be NULL
|
||||||
|
*
|
||||||
|
* Return: A pointer to newly allocated power_supply on success
|
||||||
|
* or ERR_PTR otherwise.
|
||||||
|
* The returned power_supply pointer will be automatically unregistered
|
||||||
|
* on driver detach.
|
||||||
|
*/
|
||||||
|
struct power_supply *__must_check
|
||||||
|
devm_power_supply_register_no_ws(struct device *parent,
|
||||||
|
const struct power_supply_desc *desc,
|
||||||
|
const struct power_supply_config *cfg)
|
||||||
|
{
|
||||||
|
struct power_supply **ptr, *psy;
|
||||||
|
|
||||||
|
ptr = devres_alloc(devm_power_supply_release, sizeof(*ptr), GFP_KERNEL);
|
||||||
|
|
||||||
|
if (!ptr)
|
||||||
|
return ERR_PTR(-ENOMEM);
|
||||||
|
psy = __power_supply_register(parent, desc, cfg, false);
|
||||||
|
if (IS_ERR(psy)) {
|
||||||
|
devres_free(ptr);
|
||||||
|
} else {
|
||||||
|
*ptr = psy;
|
||||||
|
devres_add(parent, ptr);
|
||||||
|
}
|
||||||
|
return psy;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(devm_power_supply_register_no_ws);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* power_supply_unregister() - Remove this power supply from system
|
||||||
|
* @psy: Pointer to power supply to unregister
|
||||||
|
*
|
||||||
|
* Remove this power supply from the system. The resources of power supply
|
||||||
|
* will be freed here or on last power_supply_put() call.
|
||||||
|
*/
|
||||||
void power_supply_unregister(struct power_supply *psy)
|
void power_supply_unregister(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
|
WARN_ON(atomic_dec_return(&psy->use_cnt));
|
||||||
cancel_work_sync(&psy->changed_work);
|
cancel_work_sync(&psy->changed_work);
|
||||||
sysfs_remove_link(&psy->dev->kobj, "powers");
|
sysfs_remove_link(&psy->dev.kobj, "powers");
|
||||||
power_supply_remove_triggers(psy);
|
power_supply_remove_triggers(psy);
|
||||||
psy_unregister_cooler(psy);
|
psy_unregister_cooler(psy);
|
||||||
psy_unregister_thermal(psy);
|
psy_unregister_thermal(psy);
|
||||||
device_init_wakeup(psy->dev, false);
|
device_init_wakeup(&psy->dev, false);
|
||||||
device_unregister(psy->dev);
|
device_unregister(&psy->dev);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(power_supply_unregister);
|
EXPORT_SYMBOL_GPL(power_supply_unregister);
|
||||||
|
|
||||||
|
void *power_supply_get_drvdata(struct power_supply *psy)
|
||||||
|
{
|
||||||
|
return psy->drv_data;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(power_supply_get_drvdata);
|
||||||
|
|
||||||
static int __init power_supply_class_init(void)
|
static int __init power_supply_class_init(void)
|
||||||
{
|
{
|
||||||
power_supply_class = class_create(THIS_MODULE, "power_supply");
|
power_supply_class = class_create(THIS_MODULE, "power_supply");
|
||||||
|
|
|
@ -25,10 +25,10 @@ static void power_supply_update_bat_leds(struct power_supply *psy)
|
||||||
unsigned long delay_on = 0;
|
unsigned long delay_on = 0;
|
||||||
unsigned long delay_off = 0;
|
unsigned long delay_off = 0;
|
||||||
|
|
||||||
if (psy->get_property(psy, POWER_SUPPLY_PROP_STATUS, &status))
|
if (psy->desc->get_property(psy, POWER_SUPPLY_PROP_STATUS, &status))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
dev_dbg(psy->dev, "%s %d\n", __func__, status.intval);
|
dev_dbg(&psy->dev, "%s %d\n", __func__, status.intval);
|
||||||
|
|
||||||
switch (status.intval) {
|
switch (status.intval) {
|
||||||
case POWER_SUPPLY_STATUS_FULL:
|
case POWER_SUPPLY_STATUS_FULL:
|
||||||
|
@ -58,21 +58,21 @@ static void power_supply_update_bat_leds(struct power_supply *psy)
|
||||||
static int power_supply_create_bat_triggers(struct power_supply *psy)
|
static int power_supply_create_bat_triggers(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
psy->charging_full_trig_name = kasprintf(GFP_KERNEL,
|
psy->charging_full_trig_name = kasprintf(GFP_KERNEL,
|
||||||
"%s-charging-or-full", psy->name);
|
"%s-charging-or-full", psy->desc->name);
|
||||||
if (!psy->charging_full_trig_name)
|
if (!psy->charging_full_trig_name)
|
||||||
goto charging_full_failed;
|
goto charging_full_failed;
|
||||||
|
|
||||||
psy->charging_trig_name = kasprintf(GFP_KERNEL,
|
psy->charging_trig_name = kasprintf(GFP_KERNEL,
|
||||||
"%s-charging", psy->name);
|
"%s-charging", psy->desc->name);
|
||||||
if (!psy->charging_trig_name)
|
if (!psy->charging_trig_name)
|
||||||
goto charging_failed;
|
goto charging_failed;
|
||||||
|
|
||||||
psy->full_trig_name = kasprintf(GFP_KERNEL, "%s-full", psy->name);
|
psy->full_trig_name = kasprintf(GFP_KERNEL, "%s-full", psy->desc->name);
|
||||||
if (!psy->full_trig_name)
|
if (!psy->full_trig_name)
|
||||||
goto full_failed;
|
goto full_failed;
|
||||||
|
|
||||||
psy->charging_blink_full_solid_trig_name = kasprintf(GFP_KERNEL,
|
psy->charging_blink_full_solid_trig_name = kasprintf(GFP_KERNEL,
|
||||||
"%s-charging-blink-full-solid", psy->name);
|
"%s-charging-blink-full-solid", psy->desc->name);
|
||||||
if (!psy->charging_blink_full_solid_trig_name)
|
if (!psy->charging_blink_full_solid_trig_name)
|
||||||
goto charging_blink_full_solid_failed;
|
goto charging_blink_full_solid_failed;
|
||||||
|
|
||||||
|
@ -115,10 +115,10 @@ static void power_supply_update_gen_leds(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
union power_supply_propval online;
|
union power_supply_propval online;
|
||||||
|
|
||||||
if (psy->get_property(psy, POWER_SUPPLY_PROP_ONLINE, &online))
|
if (psy->desc->get_property(psy, POWER_SUPPLY_PROP_ONLINE, &online))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
dev_dbg(psy->dev, "%s %d\n", __func__, online.intval);
|
dev_dbg(&psy->dev, "%s %d\n", __func__, online.intval);
|
||||||
|
|
||||||
if (online.intval)
|
if (online.intval)
|
||||||
led_trigger_event(psy->online_trig, LED_FULL);
|
led_trigger_event(psy->online_trig, LED_FULL);
|
||||||
|
@ -128,7 +128,8 @@ static void power_supply_update_gen_leds(struct power_supply *psy)
|
||||||
|
|
||||||
static int power_supply_create_gen_triggers(struct power_supply *psy)
|
static int power_supply_create_gen_triggers(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
psy->online_trig_name = kasprintf(GFP_KERNEL, "%s-online", psy->name);
|
psy->online_trig_name = kasprintf(GFP_KERNEL, "%s-online",
|
||||||
|
psy->desc->name);
|
||||||
if (!psy->online_trig_name)
|
if (!psy->online_trig_name)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -147,7 +148,7 @@ static void power_supply_remove_gen_triggers(struct power_supply *psy)
|
||||||
|
|
||||||
void power_supply_update_leds(struct power_supply *psy)
|
void power_supply_update_leds(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
if (psy->type == POWER_SUPPLY_TYPE_BATTERY)
|
if (psy->desc->type == POWER_SUPPLY_TYPE_BATTERY)
|
||||||
power_supply_update_bat_leds(psy);
|
power_supply_update_bat_leds(psy);
|
||||||
else
|
else
|
||||||
power_supply_update_gen_leds(psy);
|
power_supply_update_gen_leds(psy);
|
||||||
|
@ -155,14 +156,14 @@ void power_supply_update_leds(struct power_supply *psy)
|
||||||
|
|
||||||
int power_supply_create_triggers(struct power_supply *psy)
|
int power_supply_create_triggers(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
if (psy->type == POWER_SUPPLY_TYPE_BATTERY)
|
if (psy->desc->type == POWER_SUPPLY_TYPE_BATTERY)
|
||||||
return power_supply_create_bat_triggers(psy);
|
return power_supply_create_bat_triggers(psy);
|
||||||
return power_supply_create_gen_triggers(psy);
|
return power_supply_create_gen_triggers(psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
void power_supply_remove_triggers(struct power_supply *psy)
|
void power_supply_remove_triggers(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
if (psy->type == POWER_SUPPLY_TYPE_BATTERY)
|
if (psy->desc->type == POWER_SUPPLY_TYPE_BATTERY)
|
||||||
power_supply_remove_bat_triggers(psy);
|
power_supply_remove_bat_triggers(psy);
|
||||||
else
|
else
|
||||||
power_supply_remove_gen_triggers(psy);
|
power_supply_remove_gen_triggers(psy);
|
||||||
|
|
|
@ -74,9 +74,9 @@ static ssize_t power_supply_show_property(struct device *dev,
|
||||||
union power_supply_propval value;
|
union power_supply_propval value;
|
||||||
|
|
||||||
if (off == POWER_SUPPLY_PROP_TYPE) {
|
if (off == POWER_SUPPLY_PROP_TYPE) {
|
||||||
value.intval = psy->type;
|
value.intval = psy->desc->type;
|
||||||
} else {
|
} else {
|
||||||
ret = psy->get_property(psy, off, &value);
|
ret = power_supply_get_property(psy, off, &value);
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
if (ret == -ENODATA)
|
if (ret == -ENODATA)
|
||||||
|
@ -125,7 +125,7 @@ static ssize_t power_supply_store_property(struct device *dev,
|
||||||
|
|
||||||
value.intval = long_val;
|
value.intval = long_val;
|
||||||
|
|
||||||
ret = psy->set_property(psy, off, &value);
|
ret = psy->desc->set_property(psy, off, &value);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
@ -218,12 +218,12 @@ 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;
|
||||||
|
|
||||||
for (i = 0; i < psy->num_properties; i++) {
|
for (i = 0; i < psy->desc->num_properties; i++) {
|
||||||
int property = psy->properties[i];
|
int property = psy->desc->properties[i];
|
||||||
|
|
||||||
if (property == attrno) {
|
if (property == attrno) {
|
||||||
if (psy->property_is_writeable &&
|
if (psy->desc->property_is_writeable &&
|
||||||
psy->property_is_writeable(psy, property) > 0)
|
power_supply_property_is_writeable(psy, property) > 0)
|
||||||
mode |= S_IWUSR;
|
mode |= S_IWUSR;
|
||||||
|
|
||||||
return mode;
|
return mode;
|
||||||
|
@ -279,14 +279,14 @@ int power_supply_uevent(struct device *dev, struct kobj_uevent_env *env)
|
||||||
|
|
||||||
dev_dbg(dev, "uevent\n");
|
dev_dbg(dev, "uevent\n");
|
||||||
|
|
||||||
if (!psy || !psy->dev) {
|
if (!psy || !psy->desc) {
|
||||||
dev_dbg(dev, "No power supply yet\n");
|
dev_dbg(dev, "No power supply yet\n");
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
dev_dbg(dev, "POWER_SUPPLY_NAME=%s\n", psy->name);
|
dev_dbg(dev, "POWER_SUPPLY_NAME=%s\n", psy->desc->name);
|
||||||
|
|
||||||
ret = add_uevent_var(env, "POWER_SUPPLY_NAME=%s", psy->name);
|
ret = add_uevent_var(env, "POWER_SUPPLY_NAME=%s", psy->desc->name);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
@ -294,11 +294,11 @@ int power_supply_uevent(struct device *dev, struct kobj_uevent_env *env)
|
||||||
if (!prop_buf)
|
if (!prop_buf)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
for (j = 0; j < psy->num_properties; j++) {
|
for (j = 0; j < psy->desc->num_properties; j++) {
|
||||||
struct device_attribute *attr;
|
struct device_attribute *attr;
|
||||||
char *line;
|
char *line;
|
||||||
|
|
||||||
attr = &power_supply_attrs[psy->properties[j]];
|
attr = &power_supply_attrs[psy->desc->properties[j]];
|
||||||
|
|
||||||
ret = power_supply_show_property(dev, attr, prop_buf);
|
ret = power_supply_show_property(dev, attr, prop_buf);
|
||||||
if (ret == -ENODEV || ret == -ENODATA) {
|
if (ret == -ENODEV || ret == -ENODATA) {
|
||||||
|
|
|
@ -151,9 +151,17 @@ config POWER_RESET_SYSCON
|
||||||
help
|
help
|
||||||
Reboot support for generic SYSCON mapped register reset.
|
Reboot support for generic SYSCON mapped register reset.
|
||||||
|
|
||||||
|
config POWER_RESET_SYSCON_POWEROFF
|
||||||
|
bool "Generic SYSCON regmap poweroff driver"
|
||||||
|
depends on OF
|
||||||
|
select MFD_SYSCON
|
||||||
|
help
|
||||||
|
Poweroff support for generic SYSCON mapped register poweroff.
|
||||||
|
|
||||||
config POWER_RESET_RMOBILE
|
config POWER_RESET_RMOBILE
|
||||||
tristate "Renesas R-Mobile reset driver"
|
tristate "Renesas R-Mobile reset driver"
|
||||||
depends on ARCH_RMOBILE || COMPILE_TEST
|
depends on ARCH_RMOBILE || COMPILE_TEST
|
||||||
|
depends on HAS_IOMEM
|
||||||
help
|
help
|
||||||
Reboot support for Renesas R-Mobile and SH-Mobile SoCs.
|
Reboot support for Renesas R-Mobile and SH-Mobile SoCs.
|
||||||
|
|
||||||
|
|
|
@ -17,4 +17,5 @@ obj-$(CONFIG_POWER_RESET_VEXPRESS) += vexpress-poweroff.o
|
||||||
obj-$(CONFIG_POWER_RESET_XGENE) += xgene-reboot.o
|
obj-$(CONFIG_POWER_RESET_XGENE) += xgene-reboot.o
|
||||||
obj-$(CONFIG_POWER_RESET_KEYSTONE) += keystone-reset.o
|
obj-$(CONFIG_POWER_RESET_KEYSTONE) += keystone-reset.o
|
||||||
obj-$(CONFIG_POWER_RESET_SYSCON) += syscon-reboot.o
|
obj-$(CONFIG_POWER_RESET_SYSCON) += syscon-reboot.o
|
||||||
|
obj-$(CONFIG_POWER_RESET_SYSCON_POWEROFF) += syscon-poweroff.o
|
||||||
obj-$(CONFIG_POWER_RESET_RMOBILE) += rmobile-reset.o
|
obj-$(CONFIG_POWER_RESET_RMOBILE) += rmobile-reset.o
|
||||||
|
|
|
@ -140,7 +140,7 @@ static int at91_poweroff_probe(struct platform_device *pdev)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct of_device_id at91_poweroff_of_match[] = {
|
static const struct of_device_id at91_poweroff_of_match[] = {
|
||||||
{ .compatible = "atmel,at91sam9260-shdwc", },
|
{ .compatible = "atmel,at91sam9260-shdwc", },
|
||||||
{ .compatible = "atmel,at91sam9rl-shdwc", },
|
{ .compatible = "atmel,at91sam9rl-shdwc", },
|
||||||
{ .compatible = "atmel,at91sam9x5-shdwc", },
|
{ .compatible = "atmel,at91sam9x5-shdwc", },
|
||||||
|
|
|
@ -73,8 +73,8 @@ static int at91sam9260_restart(struct notifier_block *this, unsigned long mode,
|
||||||
: "r" (at91_ramc_base[0]),
|
: "r" (at91_ramc_base[0]),
|
||||||
"r" (at91_rstc_base),
|
"r" (at91_rstc_base),
|
||||||
"r" (1),
|
"r" (1),
|
||||||
"r" (AT91_SDRAMC_LPCB_POWER_DOWN),
|
"r" cpu_to_le32(AT91_SDRAMC_LPCB_POWER_DOWN),
|
||||||
"r" (AT91_RSTC_KEY | AT91_RSTC_PERRST | AT91_RSTC_PROCRST));
|
"r" cpu_to_le32(AT91_RSTC_KEY | AT91_RSTC_PERRST | AT91_RSTC_PROCRST));
|
||||||
|
|
||||||
return NOTIFY_DONE;
|
return NOTIFY_DONE;
|
||||||
}
|
}
|
||||||
|
@ -116,8 +116,8 @@ static int at91sam9g45_restart(struct notifier_block *this, unsigned long mode,
|
||||||
"r" (at91_ramc_base[1]),
|
"r" (at91_ramc_base[1]),
|
||||||
"r" (at91_rstc_base),
|
"r" (at91_rstc_base),
|
||||||
"r" (1),
|
"r" (1),
|
||||||
"r" (AT91_DDRSDRC_LPCB_POWER_DOWN),
|
"r" cpu_to_le32(AT91_DDRSDRC_LPCB_POWER_DOWN),
|
||||||
"r" (AT91_RSTC_KEY | AT91_RSTC_PERRST | AT91_RSTC_PROCRST)
|
"r" cpu_to_le32(AT91_RSTC_KEY | AT91_RSTC_PERRST | AT91_RSTC_PROCRST)
|
||||||
: "r0");
|
: "r0");
|
||||||
|
|
||||||
return NOTIFY_DONE;
|
return NOTIFY_DONE;
|
||||||
|
@ -152,14 +152,14 @@ static void __init at91_reset_status(struct platform_device *pdev)
|
||||||
pr_info("AT91: Starting after %s\n", reason);
|
pr_info("AT91: Starting after %s\n", reason);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct of_device_id at91_ramc_of_match[] = {
|
static const struct of_device_id at91_ramc_of_match[] = {
|
||||||
{ .compatible = "atmel,at91sam9260-sdramc", },
|
{ .compatible = "atmel,at91sam9260-sdramc", },
|
||||||
{ .compatible = "atmel,at91sam9g45-ddramc", },
|
{ .compatible = "atmel,at91sam9g45-ddramc", },
|
||||||
{ .compatible = "atmel,sama5d3-ddramc", },
|
{ .compatible = "atmel,sama5d3-ddramc", },
|
||||||
{ /* sentinel */ }
|
{ /* sentinel */ }
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct of_device_id at91_reset_of_match[] = {
|
static const struct of_device_id at91_reset_of_match[] = {
|
||||||
{ .compatible = "atmel,at91sam9260-rstc", .data = at91sam9260_restart },
|
{ .compatible = "atmel,at91sam9260-rstc", .data = at91sam9260_restart },
|
||||||
{ .compatible = "atmel,at91sam9g45-rstc", .data = at91sam9g45_restart },
|
{ .compatible = "atmel,at91sam9g45-rstc", .data = at91sam9g45_restart },
|
||||||
{ /* sentinel */ }
|
{ /* sentinel */ }
|
||||||
|
|
|
@ -64,7 +64,7 @@ static int hisi_reboot_probe(struct platform_device *pdev)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct of_device_id hisi_reboot_of_match[] = {
|
static const struct of_device_id hisi_reboot_of_match[] = {
|
||||||
{ .compatible = "hisilicon,sysctrl" },
|
{ .compatible = "hisilicon,sysctrl" },
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
|
@ -70,7 +70,7 @@ static struct notifier_block rsctrl_restart_nb = {
|
||||||
.priority = 128,
|
.priority = 128,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct of_device_id rsctrl_of_match[] = {
|
static const struct of_device_id rsctrl_of_match[] = {
|
||||||
{.compatible = "ti,keystone-reset", },
|
{.compatible = "ti,keystone-reset", },
|
||||||
{},
|
{},
|
||||||
};
|
};
|
||||||
|
|
|
@ -97,7 +97,7 @@ static struct notifier_block st_restart_nb = {
|
||||||
.priority = 192,
|
.priority = 192,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct of_device_id st_reset_of_match[] = {
|
static const struct of_device_id st_reset_of_match[] = {
|
||||||
{
|
{
|
||||||
.compatible = "st,stih415-restart",
|
.compatible = "st,stih415-restart",
|
||||||
.data = (void *)&stih415_reset,
|
.data = (void *)&stih415_reset,
|
||||||
|
|
102
drivers/power/reset/syscon-poweroff.c
Normal file
102
drivers/power/reset/syscon-poweroff.c
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
/*
|
||||||
|
* Generic Syscon Poweroff Driver
|
||||||
|
*
|
||||||
|
* Copyright (c) 2015, National Instruments Corp.
|
||||||
|
* Author: Moritz Fischer <moritz.fischer@ettus.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.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <linux/kallsyms.h>
|
||||||
|
#include <linux/delay.h>
|
||||||
|
#include <linux/io.h>
|
||||||
|
#include <linux/notifier.h>
|
||||||
|
#include <linux/mfd/syscon.h>
|
||||||
|
#include <linux/of_address.h>
|
||||||
|
#include <linux/of_device.h>
|
||||||
|
#include <linux/platform_device.h>
|
||||||
|
#include <linux/pm.h>
|
||||||
|
#include <linux/regmap.h>
|
||||||
|
|
||||||
|
static struct regmap *map;
|
||||||
|
static u32 offset;
|
||||||
|
static u32 mask;
|
||||||
|
|
||||||
|
void syscon_poweroff(void)
|
||||||
|
{
|
||||||
|
/* Issue the poweroff */
|
||||||
|
regmap_write(map, offset, mask);
|
||||||
|
|
||||||
|
mdelay(1000);
|
||||||
|
|
||||||
|
pr_emerg("Unable to poweroff system\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int syscon_poweroff_probe(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
char symname[KSYM_NAME_LEN];
|
||||||
|
|
||||||
|
map = syscon_regmap_lookup_by_phandle(pdev->dev.of_node, "regmap");
|
||||||
|
if (IS_ERR(map)) {
|
||||||
|
dev_err(&pdev->dev, "unable to get syscon");
|
||||||
|
return PTR_ERR(map);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (of_property_read_u32(pdev->dev.of_node, "offset", &offset)) {
|
||||||
|
dev_err(&pdev->dev, "unable to read 'offset'");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (of_property_read_u32(pdev->dev.of_node, "mask", &mask)) {
|
||||||
|
dev_err(&pdev->dev, "unable to read 'mask'");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pm_power_off) {
|
||||||
|
lookup_symbol_name((ulong)pm_power_off, symname);
|
||||||
|
dev_err(&pdev->dev,
|
||||||
|
"pm_power_off already claimed %p %s",
|
||||||
|
pm_power_off, symname);
|
||||||
|
return -EBUSY;
|
||||||
|
}
|
||||||
|
|
||||||
|
pm_power_off = syscon_poweroff;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int syscon_poweroff_remove(struct platform_device *pdev)
|
||||||
|
{
|
||||||
|
if (pm_power_off == syscon_poweroff)
|
||||||
|
pm_power_off = NULL;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct of_device_id syscon_poweroff_of_match[] = {
|
||||||
|
{ .compatible = "syscon-poweroff" },
|
||||||
|
{}
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct platform_driver syscon_poweroff_driver = {
|
||||||
|
.probe = syscon_poweroff_probe,
|
||||||
|
.remove = syscon_poweroff_remove,
|
||||||
|
.driver = {
|
||||||
|
.name = "syscon-poweroff",
|
||||||
|
.of_match_table = syscon_poweroff_of_match,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
static int __init syscon_poweroff_register(void)
|
||||||
|
{
|
||||||
|
return platform_driver_register(&syscon_poweroff_driver);
|
||||||
|
}
|
||||||
|
device_initcall(syscon_poweroff_register);
|
|
@ -76,7 +76,7 @@ static int syscon_reboot_probe(struct platform_device *pdev)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct of_device_id syscon_reboot_of_match[] = {
|
static const struct of_device_id syscon_reboot_of_match[] = {
|
||||||
{ .compatible = "syscon-reboot" },
|
{ .compatible = "syscon-reboot" },
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
|
@ -80,7 +80,7 @@ DEVICE_ATTR(active, S_IRUGO | S_IWUSR, vexpress_reset_active_show,
|
||||||
|
|
||||||
enum vexpress_reset_func { FUNC_RESET, FUNC_SHUTDOWN, FUNC_REBOOT };
|
enum vexpress_reset_func { FUNC_RESET, FUNC_SHUTDOWN, FUNC_REBOOT };
|
||||||
|
|
||||||
static struct of_device_id vexpress_reset_of_match[] = {
|
static const struct of_device_id vexpress_reset_of_match[] = {
|
||||||
{
|
{
|
||||||
.compatible = "arm,vexpress-reset",
|
.compatible = "arm,vexpress-reset",
|
||||||
.data = (void *)FUNC_RESET,
|
.data = (void *)FUNC_RESET,
|
||||||
|
|
|
@ -87,7 +87,7 @@ static int xgene_reboot_probe(struct platform_device *pdev)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct of_device_id xgene_reboot_of_match[] = {
|
static const struct of_device_id xgene_reboot_of_match[] = {
|
||||||
{ .compatible = "apm,xgene-reboot" },
|
{ .compatible = "apm,xgene-reboot" },
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
|
@ -72,8 +72,7 @@ static int rt5033_battery_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 rt5033_battery *battery = container_of(psy,
|
struct rt5033_battery *battery = power_supply_get_drvdata(psy);
|
||||||
struct rt5033_battery, psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
||||||
|
@ -102,16 +101,25 @@ static enum power_supply_property rt5033_battery_props[] = {
|
||||||
POWER_SUPPLY_PROP_CAPACITY,
|
POWER_SUPPLY_PROP_CAPACITY,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct regmap_config rt5033_battery_regmap_config = {
|
static const struct regmap_config rt5033_battery_regmap_config = {
|
||||||
.reg_bits = 8,
|
.reg_bits = 8,
|
||||||
.val_bits = 8,
|
.val_bits = 8,
|
||||||
.max_register = RT5033_FUEL_REG_END,
|
.max_register = RT5033_FUEL_REG_END,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc rt5033_battery_desc = {
|
||||||
|
.name = "rt5033-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.get_property = rt5033_battery_get_property,
|
||||||
|
.properties = rt5033_battery_props,
|
||||||
|
.num_properties = ARRAY_SIZE(rt5033_battery_props),
|
||||||
|
};
|
||||||
|
|
||||||
static int rt5033_battery_probe(struct i2c_client *client,
|
static int rt5033_battery_probe(struct i2c_client *client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct rt5033_battery *battery;
|
struct rt5033_battery *battery;
|
||||||
u32 ret;
|
u32 ret;
|
||||||
|
|
||||||
|
@ -131,16 +139,13 @@ static int rt5033_battery_probe(struct i2c_client *client,
|
||||||
}
|
}
|
||||||
|
|
||||||
i2c_set_clientdata(client, battery);
|
i2c_set_clientdata(client, battery);
|
||||||
|
psy_cfg.drv_data = battery;
|
||||||
|
|
||||||
battery->psy.name = "rt5033-battery";
|
battery->psy = power_supply_register(&client->dev,
|
||||||
battery->psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
&rt5033_battery_desc, &psy_cfg);
|
||||||
battery->psy.get_property = rt5033_battery_get_property;
|
if (IS_ERR(battery->psy)) {
|
||||||
battery->psy.properties = rt5033_battery_props;
|
|
||||||
battery->psy.num_properties = ARRAY_SIZE(rt5033_battery_props);
|
|
||||||
|
|
||||||
ret = power_supply_register(&client->dev, &battery->psy);
|
|
||||||
if (ret) {
|
|
||||||
dev_err(&client->dev, "Failed to register power supply\n");
|
dev_err(&client->dev, "Failed to register power supply\n");
|
||||||
|
ret = PTR_ERR(battery->psy);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -151,7 +156,7 @@ static int rt5033_battery_remove(struct i2c_client *client)
|
||||||
{
|
{
|
||||||
struct rt5033_battery *battery = i2c_get_clientdata(client);
|
struct rt5033_battery *battery = i2c_get_clientdata(client);
|
||||||
|
|
||||||
power_supply_unregister(&battery->psy);
|
power_supply_unregister(battery->psy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,8 @@
|
||||||
|
|
||||||
struct rx51_device_info {
|
struct rx51_device_info {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct power_supply bat;
|
struct power_supply *bat;
|
||||||
|
struct power_supply_desc bat_desc;
|
||||||
struct iio_channel *channel_temp;
|
struct iio_channel *channel_temp;
|
||||||
struct iio_channel *channel_bsi;
|
struct iio_channel *channel_bsi;
|
||||||
struct iio_channel *channel_vbat;
|
struct iio_channel *channel_vbat;
|
||||||
|
@ -161,8 +162,7 @@ static int rx51_battery_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 rx51_device_info *di = container_of((psy),
|
struct rx51_device_info *di = power_supply_get_drvdata(psy);
|
||||||
struct rx51_device_info, bat);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
||||||
|
@ -204,6 +204,7 @@ static enum power_supply_property rx51_battery_props[] = {
|
||||||
|
|
||||||
static int rx51_battery_probe(struct platform_device *pdev)
|
static int rx51_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
struct rx51_device_info *di;
|
struct rx51_device_info *di;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -214,11 +215,13 @@ static int rx51_battery_probe(struct platform_device *pdev)
|
||||||
platform_set_drvdata(pdev, di);
|
platform_set_drvdata(pdev, di);
|
||||||
|
|
||||||
di->dev = &pdev->dev;
|
di->dev = &pdev->dev;
|
||||||
di->bat.name = dev_name(&pdev->dev);
|
di->bat_desc.name = dev_name(&pdev->dev);
|
||||||
di->bat.type = POWER_SUPPLY_TYPE_BATTERY;
|
di->bat_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
di->bat.properties = rx51_battery_props;
|
di->bat_desc.properties = rx51_battery_props;
|
||||||
di->bat.num_properties = ARRAY_SIZE(rx51_battery_props);
|
di->bat_desc.num_properties = ARRAY_SIZE(rx51_battery_props);
|
||||||
di->bat.get_property = rx51_battery_get_property;
|
di->bat_desc.get_property = rx51_battery_get_property;
|
||||||
|
|
||||||
|
psy_cfg.drv_data = di;
|
||||||
|
|
||||||
di->channel_temp = iio_channel_get(di->dev, "temp");
|
di->channel_temp = iio_channel_get(di->dev, "temp");
|
||||||
if (IS_ERR(di->channel_temp)) {
|
if (IS_ERR(di->channel_temp)) {
|
||||||
|
@ -238,9 +241,11 @@ static int rx51_battery_probe(struct platform_device *pdev)
|
||||||
goto error_channel_bsi;
|
goto error_channel_bsi;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = power_supply_register(di->dev, &di->bat);
|
di->bat = power_supply_register(di->dev, &di->bat_desc, &psy_cfg);
|
||||||
if (ret)
|
if (IS_ERR(di->bat)) {
|
||||||
|
ret = PTR_ERR(di->bat);
|
||||||
goto error_channel_vbat;
|
goto error_channel_vbat;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -259,7 +264,7 @@ static int rx51_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct rx51_device_info *di = platform_get_drvdata(pdev);
|
struct rx51_device_info *di = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
power_supply_unregister(&di->bat);
|
power_supply_unregister(di->bat);
|
||||||
|
|
||||||
iio_channel_release(di->channel_vbat);
|
iio_channel_release(di->channel_vbat);
|
||||||
iio_channel_release(di->channel_bsi);
|
iio_channel_release(di->channel_bsi);
|
||||||
|
|
|
@ -28,7 +28,7 @@
|
||||||
#define JITTER_DELAY 500 /* ms */
|
#define JITTER_DELAY 500 /* ms */
|
||||||
|
|
||||||
struct s3c_adc_bat {
|
struct s3c_adc_bat {
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
struct s3c_adc_client *client;
|
struct s3c_adc_client *client;
|
||||||
struct s3c_adc_bat_pdata *pdata;
|
struct s3c_adc_bat_pdata *pdata;
|
||||||
int volt_value;
|
int volt_value;
|
||||||
|
@ -73,10 +73,10 @@ static int s3c_adc_backup_bat_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 s3c_adc_bat *bat = container_of(psy, struct s3c_adc_bat, psy);
|
struct s3c_adc_bat *bat = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
if (!bat) {
|
if (!bat) {
|
||||||
dev_err(psy->dev, "%s: no battery infos ?!\n", __func__);
|
dev_err(&psy->dev, "%s: no battery infos ?!\n", __func__);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -105,17 +105,17 @@ static int s3c_adc_backup_bat_get_property(struct power_supply *psy,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct s3c_adc_bat backup_bat = {
|
static const struct power_supply_desc backup_bat_desc = {
|
||||||
.psy = {
|
.name = "backup-battery",
|
||||||
.name = "backup-battery",
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
.properties = s3c_adc_backup_bat_props,
|
||||||
.properties = s3c_adc_backup_bat_props,
|
.num_properties = ARRAY_SIZE(s3c_adc_backup_bat_props),
|
||||||
.num_properties = ARRAY_SIZE(s3c_adc_backup_bat_props),
|
.get_property = s3c_adc_backup_bat_get_property,
|
||||||
.get_property = s3c_adc_backup_bat_get_property,
|
.use_for_apm = 1,
|
||||||
.use_for_apm = 1,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct s3c_adc_bat backup_bat;
|
||||||
|
|
||||||
static enum power_supply_property s3c_adc_main_bat_props[] = {
|
static enum power_supply_property s3c_adc_main_bat_props[] = {
|
||||||
POWER_SUPPLY_PROP_STATUS,
|
POWER_SUPPLY_PROP_STATUS,
|
||||||
POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN,
|
POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN,
|
||||||
|
@ -141,7 +141,7 @@ static int s3c_adc_bat_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 s3c_adc_bat *bat = container_of(psy, struct s3c_adc_bat, psy);
|
struct s3c_adc_bat *bat = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
int new_level;
|
int new_level;
|
||||||
int full_volt;
|
int full_volt;
|
||||||
|
@ -149,7 +149,7 @@ static int s3c_adc_bat_get_property(struct power_supply *psy,
|
||||||
unsigned int lut_size;
|
unsigned int lut_size;
|
||||||
|
|
||||||
if (!bat) {
|
if (!bat) {
|
||||||
dev_err(psy->dev, "no battery infos ?!\n");
|
dev_err(&psy->dev, "no battery infos ?!\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -232,18 +232,18 @@ static int s3c_adc_bat_get_property(struct power_supply *psy,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct s3c_adc_bat main_bat = {
|
static const struct power_supply_desc main_bat_desc = {
|
||||||
.psy = {
|
.name = "main-battery",
|
||||||
.name = "main-battery",
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
.properties = s3c_adc_main_bat_props,
|
||||||
.properties = s3c_adc_main_bat_props,
|
.num_properties = ARRAY_SIZE(s3c_adc_main_bat_props),
|
||||||
.num_properties = ARRAY_SIZE(s3c_adc_main_bat_props),
|
.get_property = s3c_adc_bat_get_property,
|
||||||
.get_property = s3c_adc_bat_get_property,
|
.external_power_changed = s3c_adc_bat_ext_power_changed,
|
||||||
.external_power_changed = s3c_adc_bat_ext_power_changed,
|
.use_for_apm = 1,
|
||||||
.use_for_apm = 1,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct s3c_adc_bat main_bat;
|
||||||
|
|
||||||
static void s3c_adc_bat_work(struct work_struct *work)
|
static void s3c_adc_bat_work(struct work_struct *work)
|
||||||
{
|
{
|
||||||
struct s3c_adc_bat *bat = &main_bat;
|
struct s3c_adc_bat *bat = &main_bat;
|
||||||
|
@ -251,7 +251,7 @@ static void s3c_adc_bat_work(struct work_struct *work)
|
||||||
int is_plugged;
|
int is_plugged;
|
||||||
static int was_plugged;
|
static int was_plugged;
|
||||||
|
|
||||||
is_plugged = power_supply_am_i_supplied(&bat->psy);
|
is_plugged = power_supply_am_i_supplied(bat->psy);
|
||||||
bat->cable_plugged = is_plugged;
|
bat->cable_plugged = is_plugged;
|
||||||
if (is_plugged != was_plugged) {
|
if (is_plugged != was_plugged) {
|
||||||
was_plugged = is_plugged;
|
was_plugged = is_plugged;
|
||||||
|
@ -279,7 +279,7 @@ static void s3c_adc_bat_work(struct work_struct *work)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_changed(&bat->psy);
|
power_supply_changed(bat->psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static irqreturn_t s3c_adc_bat_charged(int irq, void *dev_id)
|
static irqreturn_t s3c_adc_bat_charged(int irq, void *dev_id)
|
||||||
|
@ -310,16 +310,25 @@ static int s3c_adc_bat_probe(struct platform_device *pdev)
|
||||||
main_bat.cable_plugged = 0;
|
main_bat.cable_plugged = 0;
|
||||||
main_bat.status = POWER_SUPPLY_STATUS_DISCHARGING;
|
main_bat.status = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &main_bat.psy);
|
main_bat.psy = power_supply_register(&pdev->dev, &main_bat_desc, NULL);
|
||||||
if (ret)
|
if (IS_ERR(main_bat.psy)) {
|
||||||
|
ret = PTR_ERR(main_bat.psy);
|
||||||
goto err_reg_main;
|
goto err_reg_main;
|
||||||
|
}
|
||||||
if (pdata->backup_volt_mult) {
|
if (pdata->backup_volt_mult) {
|
||||||
|
const struct power_supply_config psy_cfg
|
||||||
|
= { .drv_data = &backup_bat, };
|
||||||
|
|
||||||
backup_bat.client = client;
|
backup_bat.client = client;
|
||||||
backup_bat.pdata = pdev->dev.platform_data;
|
backup_bat.pdata = pdev->dev.platform_data;
|
||||||
backup_bat.volt_value = -1;
|
backup_bat.volt_value = -1;
|
||||||
ret = power_supply_register(&pdev->dev, &backup_bat.psy);
|
backup_bat.psy = power_supply_register(&pdev->dev,
|
||||||
if (ret)
|
&backup_bat_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(backup_bat.psy)) {
|
||||||
|
ret = PTR_ERR(backup_bat.psy);
|
||||||
goto err_reg_backup;
|
goto err_reg_backup;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&bat_work, s3c_adc_bat_work);
|
INIT_DELAYED_WORK(&bat_work, s3c_adc_bat_work);
|
||||||
|
@ -360,9 +369,9 @@ err_irq:
|
||||||
gpio_free(pdata->gpio_charge_finished);
|
gpio_free(pdata->gpio_charge_finished);
|
||||||
err_gpio:
|
err_gpio:
|
||||||
if (pdata->backup_volt_mult)
|
if (pdata->backup_volt_mult)
|
||||||
power_supply_unregister(&backup_bat.psy);
|
power_supply_unregister(backup_bat.psy);
|
||||||
err_reg_backup:
|
err_reg_backup:
|
||||||
power_supply_unregister(&main_bat.psy);
|
power_supply_unregister(main_bat.psy);
|
||||||
err_reg_main:
|
err_reg_main:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -372,9 +381,9 @@ static int s3c_adc_bat_remove(struct platform_device *pdev)
|
||||||
struct s3c_adc_client *client = platform_get_drvdata(pdev);
|
struct s3c_adc_client *client = platform_get_drvdata(pdev);
|
||||||
struct s3c_adc_bat_pdata *pdata = pdev->dev.platform_data;
|
struct s3c_adc_bat_pdata *pdata = pdev->dev.platform_data;
|
||||||
|
|
||||||
power_supply_unregister(&main_bat.psy);
|
power_supply_unregister(main_bat.psy);
|
||||||
if (pdata->backup_volt_mult)
|
if (pdata->backup_volt_mult)
|
||||||
power_supply_unregister(&backup_bat.psy);
|
power_supply_unregister(backup_bat.psy);
|
||||||
|
|
||||||
s3c_adc_release(client);
|
s3c_adc_release(client);
|
||||||
|
|
||||||
|
|
|
@ -156,7 +156,7 @@ static enum power_supply_property sbs_properties[] = {
|
||||||
|
|
||||||
struct sbs_info {
|
struct sbs_info {
|
||||||
struct i2c_client *client;
|
struct i2c_client *client;
|
||||||
struct power_supply power_supply;
|
struct power_supply *power_supply;
|
||||||
struct sbs_platform_data *pdata;
|
struct sbs_platform_data *pdata;
|
||||||
bool is_present;
|
bool is_present;
|
||||||
bool gpio_detect;
|
bool gpio_detect;
|
||||||
|
@ -391,7 +391,7 @@ static int sbs_get_battery_property(struct i2c_client *client,
|
||||||
chip->last_state = val->intval;
|
chip->last_state = val->intval;
|
||||||
else if (chip->last_state != val->intval) {
|
else if (chip->last_state != val->intval) {
|
||||||
cancel_delayed_work_sync(&chip->work);
|
cancel_delayed_work_sync(&chip->work);
|
||||||
power_supply_changed(&chip->power_supply);
|
power_supply_changed(chip->power_supply);
|
||||||
chip->poll_time = 0;
|
chip->poll_time = 0;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -556,8 +556,7 @@ static int sbs_get_property(struct power_supply *psy,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct sbs_info *chip = container_of(psy,
|
struct sbs_info *chip = power_supply_get_drvdata(psy);
|
||||||
struct sbs_info, power_supply);
|
|
||||||
struct i2c_client *client = chip->client;
|
struct i2c_client *client = chip->client;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -638,7 +637,7 @@ static int sbs_get_property(struct power_supply *psy,
|
||||||
if (!chip->gpio_detect &&
|
if (!chip->gpio_detect &&
|
||||||
chip->is_present != (ret >= 0)) {
|
chip->is_present != (ret >= 0)) {
|
||||||
chip->is_present = (ret >= 0);
|
chip->is_present = (ret >= 0);
|
||||||
power_supply_changed(&chip->power_supply);
|
power_supply_changed(chip->power_supply);
|
||||||
}
|
}
|
||||||
|
|
||||||
done:
|
done:
|
||||||
|
@ -671,9 +670,7 @@ static irqreturn_t sbs_irq(int irq, void *devid)
|
||||||
|
|
||||||
static void sbs_external_power_changed(struct power_supply *psy)
|
static void sbs_external_power_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct sbs_info *chip;
|
struct sbs_info *chip = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
chip = container_of(psy, struct sbs_info, power_supply);
|
|
||||||
|
|
||||||
if (chip->ignore_changes > 0) {
|
if (chip->ignore_changes > 0) {
|
||||||
chip->ignore_changes--;
|
chip->ignore_changes--;
|
||||||
|
@ -712,7 +709,7 @@ static void sbs_delayed_work(struct work_struct *work)
|
||||||
|
|
||||||
if (chip->last_state != ret) {
|
if (chip->last_state != ret) {
|
||||||
chip->poll_time = 0;
|
chip->poll_time = 0;
|
||||||
power_supply_changed(&chip->power_supply);
|
power_supply_changed(chip->power_supply);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (chip->poll_time > 0) {
|
if (chip->poll_time > 0) {
|
||||||
|
@ -796,42 +793,48 @@ static struct sbs_platform_data *sbs_of_populate_pdata(
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
static const struct power_supply_desc sbs_default_desc = {
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = sbs_properties,
|
||||||
|
.num_properties = ARRAY_SIZE(sbs_properties),
|
||||||
|
.get_property = sbs_get_property,
|
||||||
|
.external_power_changed = sbs_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static int sbs_probe(struct i2c_client *client,
|
static int sbs_probe(struct i2c_client *client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
struct sbs_info *chip;
|
struct sbs_info *chip;
|
||||||
|
struct power_supply_desc *sbs_desc;
|
||||||
struct sbs_platform_data *pdata = client->dev.platform_data;
|
struct sbs_platform_data *pdata = client->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
int rc;
|
int rc;
|
||||||
int irq;
|
int irq;
|
||||||
char *name;
|
|
||||||
|
|
||||||
name = kasprintf(GFP_KERNEL, "sbs-%s", dev_name(&client->dev));
|
sbs_desc = devm_kmemdup(&client->dev, &sbs_default_desc,
|
||||||
if (!name) {
|
sizeof(*sbs_desc), GFP_KERNEL);
|
||||||
dev_err(&client->dev, "Failed to allocate device name\n");
|
if (!sbs_desc)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
sbs_desc->name = devm_kasprintf(&client->dev, GFP_KERNEL, "sbs-%s",
|
||||||
|
dev_name(&client->dev));
|
||||||
|
if (!sbs_desc->name)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
|
||||||
|
|
||||||
chip = kzalloc(sizeof(struct sbs_info), GFP_KERNEL);
|
chip = kzalloc(sizeof(struct sbs_info), GFP_KERNEL);
|
||||||
if (!chip) {
|
if (!chip)
|
||||||
rc = -ENOMEM;
|
return -ENOMEM;
|
||||||
goto exit_free_name;
|
|
||||||
}
|
|
||||||
|
|
||||||
chip->client = client;
|
chip->client = client;
|
||||||
chip->enable_detection = false;
|
chip->enable_detection = false;
|
||||||
chip->gpio_detect = false;
|
chip->gpio_detect = false;
|
||||||
chip->power_supply.name = name;
|
psy_cfg.of_node = client->dev.of_node;
|
||||||
chip->power_supply.type = POWER_SUPPLY_TYPE_BATTERY;
|
psy_cfg.drv_data = chip;
|
||||||
chip->power_supply.properties = sbs_properties;
|
|
||||||
chip->power_supply.num_properties = ARRAY_SIZE(sbs_properties);
|
|
||||||
chip->power_supply.get_property = sbs_get_property;
|
|
||||||
chip->power_supply.of_node = client->dev.of_node;
|
|
||||||
/* ignore first notification of external change, it is generated
|
/* ignore first notification of external change, it is generated
|
||||||
* from the power_supply_register call back
|
* from the power_supply_register call back
|
||||||
*/
|
*/
|
||||||
chip->ignore_changes = 1;
|
chip->ignore_changes = 1;
|
||||||
chip->last_state = POWER_SUPPLY_STATUS_UNKNOWN;
|
chip->last_state = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
chip->power_supply.external_power_changed = sbs_external_power_changed;
|
|
||||||
|
|
||||||
pdata = sbs_of_populate_pdata(client);
|
pdata = sbs_of_populate_pdata(client);
|
||||||
|
|
||||||
|
@ -870,7 +873,7 @@ static int sbs_probe(struct i2c_client *client,
|
||||||
|
|
||||||
rc = request_irq(irq, sbs_irq,
|
rc = request_irq(irq, sbs_irq,
|
||||||
IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING,
|
IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING,
|
||||||
dev_name(&client->dev), &chip->power_supply);
|
dev_name(&client->dev), chip->power_supply);
|
||||||
if (rc) {
|
if (rc) {
|
||||||
dev_warn(&client->dev, "Failed to request irq: %d\n", rc);
|
dev_warn(&client->dev, "Failed to request irq: %d\n", rc);
|
||||||
gpio_free(pdata->battery_detect);
|
gpio_free(pdata->battery_detect);
|
||||||
|
@ -892,10 +895,12 @@ skip_gpio:
|
||||||
goto exit_psupply;
|
goto exit_psupply;
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = power_supply_register(&client->dev, &chip->power_supply);
|
chip->power_supply = power_supply_register(&client->dev, sbs_desc,
|
||||||
if (rc) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(chip->power_supply)) {
|
||||||
dev_err(&client->dev,
|
dev_err(&client->dev,
|
||||||
"%s: Failed to register power supply\n", __func__);
|
"%s: Failed to register power supply\n", __func__);
|
||||||
|
rc = PTR_ERR(chip->power_supply);
|
||||||
goto exit_psupply;
|
goto exit_psupply;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -910,15 +915,12 @@ skip_gpio:
|
||||||
|
|
||||||
exit_psupply:
|
exit_psupply:
|
||||||
if (chip->irq)
|
if (chip->irq)
|
||||||
free_irq(chip->irq, &chip->power_supply);
|
free_irq(chip->irq, chip->power_supply);
|
||||||
if (chip->gpio_detect)
|
if (chip->gpio_detect)
|
||||||
gpio_free(pdata->battery_detect);
|
gpio_free(pdata->battery_detect);
|
||||||
|
|
||||||
kfree(chip);
|
kfree(chip);
|
||||||
|
|
||||||
exit_free_name:
|
|
||||||
kfree(name);
|
|
||||||
|
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -927,15 +929,14 @@ static int sbs_remove(struct i2c_client *client)
|
||||||
struct sbs_info *chip = i2c_get_clientdata(client);
|
struct sbs_info *chip = i2c_get_clientdata(client);
|
||||||
|
|
||||||
if (chip->irq)
|
if (chip->irq)
|
||||||
free_irq(chip->irq, &chip->power_supply);
|
free_irq(chip->irq, chip->power_supply);
|
||||||
if (chip->gpio_detect)
|
if (chip->gpio_detect)
|
||||||
gpio_free(chip->pdata->battery_detect);
|
gpio_free(chip->pdata->battery_detect);
|
||||||
|
|
||||||
power_supply_unregister(&chip->power_supply);
|
power_supply_unregister(chip->power_supply);
|
||||||
|
|
||||||
cancel_delayed_work_sync(&chip->work);
|
cancel_delayed_work_sync(&chip->work);
|
||||||
|
|
||||||
kfree(chip->power_supply.name);
|
|
||||||
kfree(chip);
|
kfree(chip);
|
||||||
chip = NULL;
|
chip = NULL;
|
||||||
|
|
||||||
|
|
|
@ -139,9 +139,9 @@ struct smb347_charger {
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct regmap *regmap;
|
struct regmap *regmap;
|
||||||
struct power_supply mains;
|
struct power_supply *mains;
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
bool mains_online;
|
bool mains_online;
|
||||||
bool usb_online;
|
bool usb_online;
|
||||||
bool charging_enabled;
|
bool charging_enabled;
|
||||||
|
@ -741,7 +741,7 @@ static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
*/
|
*/
|
||||||
if (stat_c & STAT_C_CHARGER_ERROR) {
|
if (stat_c & STAT_C_CHARGER_ERROR) {
|
||||||
dev_err(smb->dev, "charging stopped due to charger error\n");
|
dev_err(smb->dev, "charging stopped due to charger error\n");
|
||||||
power_supply_changed(&smb->battery);
|
power_supply_changed(smb->battery);
|
||||||
handled = true;
|
handled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -752,7 +752,7 @@ static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
*/
|
*/
|
||||||
if (irqstat_c & (IRQSTAT_C_TERMINATION_IRQ | IRQSTAT_C_TAPER_IRQ)) {
|
if (irqstat_c & (IRQSTAT_C_TERMINATION_IRQ | IRQSTAT_C_TAPER_IRQ)) {
|
||||||
if (irqstat_c & IRQSTAT_C_TERMINATION_STAT)
|
if (irqstat_c & IRQSTAT_C_TERMINATION_STAT)
|
||||||
power_supply_changed(&smb->battery);
|
power_supply_changed(smb->battery);
|
||||||
dev_dbg(smb->dev, "going to HW maintenance mode\n");
|
dev_dbg(smb->dev, "going to HW maintenance mode\n");
|
||||||
handled = true;
|
handled = true;
|
||||||
}
|
}
|
||||||
|
@ -766,7 +766,7 @@ static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
|
|
||||||
if (irqstat_d & IRQSTAT_D_CHARGE_TIMEOUT_STAT)
|
if (irqstat_d & IRQSTAT_D_CHARGE_TIMEOUT_STAT)
|
||||||
dev_warn(smb->dev, "charging stopped due to timeout\n");
|
dev_warn(smb->dev, "charging stopped due to timeout\n");
|
||||||
power_supply_changed(&smb->battery);
|
power_supply_changed(smb->battery);
|
||||||
handled = true;
|
handled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -778,9 +778,9 @@ static irqreturn_t smb347_interrupt(int irq, void *data)
|
||||||
if (smb347_update_ps_status(smb) > 0) {
|
if (smb347_update_ps_status(smb) > 0) {
|
||||||
smb347_start_stop_charging(smb);
|
smb347_start_stop_charging(smb);
|
||||||
if (smb->pdata->use_mains)
|
if (smb->pdata->use_mains)
|
||||||
power_supply_changed(&smb->mains);
|
power_supply_changed(smb->mains);
|
||||||
if (smb->pdata->use_usb)
|
if (smb->pdata->use_usb)
|
||||||
power_supply_changed(&smb->usb);
|
power_supply_changed(smb->usb);
|
||||||
}
|
}
|
||||||
handled = true;
|
handled = true;
|
||||||
}
|
}
|
||||||
|
@ -842,7 +842,8 @@ static int smb347_irq_init(struct smb347_charger *smb,
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
ret = request_threaded_irq(irq, NULL, smb347_interrupt,
|
ret = request_threaded_irq(irq, NULL, smb347_interrupt,
|
||||||
IRQF_TRIGGER_FALLING, client->name, smb);
|
IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
|
||||||
|
client->name, smb);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto fail_gpio;
|
goto fail_gpio;
|
||||||
|
|
||||||
|
@ -934,8 +935,7 @@ static int smb347_mains_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property prop,
|
enum power_supply_property prop,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct smb347_charger *smb =
|
struct smb347_charger *smb = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct smb347_charger, mains);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
|
@ -976,8 +976,7 @@ static int smb347_usb_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property prop,
|
enum power_supply_property prop,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct smb347_charger *smb =
|
struct smb347_charger *smb = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct smb347_charger, usb);
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
switch (prop) {
|
switch (prop) {
|
||||||
|
@ -1063,8 +1062,7 @@ static int smb347_battery_get_property(struct power_supply *psy,
|
||||||
enum power_supply_property prop,
|
enum power_supply_property prop,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct smb347_charger *smb =
|
struct smb347_charger *smb = power_supply_get_drvdata(psy);
|
||||||
container_of(psy, struct smb347_charger, battery);
|
|
||||||
const struct smb347_charger_platform_data *pdata = smb->pdata;
|
const struct smb347_charger_platform_data *pdata = smb->pdata;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -1188,11 +1186,36 @@ static const struct regmap_config smb347_regmap = {
|
||||||
.readable_reg = smb347_readable_reg,
|
.readable_reg = smb347_readable_reg,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc smb347_mains_desc = {
|
||||||
|
.name = "smb347-mains",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.get_property = smb347_mains_get_property,
|
||||||
|
.properties = smb347_mains_properties,
|
||||||
|
.num_properties = ARRAY_SIZE(smb347_mains_properties),
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc smb347_usb_desc = {
|
||||||
|
.name = "smb347-usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.get_property = smb347_usb_get_property,
|
||||||
|
.properties = smb347_usb_properties,
|
||||||
|
.num_properties = ARRAY_SIZE(smb347_usb_properties),
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc smb347_battery_desc = {
|
||||||
|
.name = "smb347-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.get_property = smb347_battery_get_property,
|
||||||
|
.properties = smb347_battery_properties,
|
||||||
|
.num_properties = ARRAY_SIZE(smb347_battery_properties),
|
||||||
|
};
|
||||||
|
|
||||||
static int smb347_probe(struct i2c_client *client,
|
static int smb347_probe(struct i2c_client *client,
|
||||||
const struct i2c_device_id *id)
|
const struct i2c_device_id *id)
|
||||||
{
|
{
|
||||||
static char *battery[] = { "smb347-battery" };
|
static char *battery[] = { "smb347-battery" };
|
||||||
const struct smb347_charger_platform_data *pdata;
|
const struct smb347_charger_platform_data *pdata;
|
||||||
|
struct power_supply_config mains_usb_cfg = {}, battery_cfg = {};
|
||||||
struct device *dev = &client->dev;
|
struct device *dev = &client->dev;
|
||||||
struct smb347_charger *smb;
|
struct smb347_charger *smb;
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -1222,49 +1245,35 @@ static int smb347_probe(struct i2c_client *client,
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
mains_usb_cfg.supplied_to = battery;
|
||||||
|
mains_usb_cfg.num_supplicants = ARRAY_SIZE(battery);
|
||||||
|
mains_usb_cfg.drv_data = smb;
|
||||||
if (smb->pdata->use_mains) {
|
if (smb->pdata->use_mains) {
|
||||||
smb->mains.name = "smb347-mains";
|
smb->mains = power_supply_register(dev, &smb347_mains_desc,
|
||||||
smb->mains.type = POWER_SUPPLY_TYPE_MAINS;
|
&mains_usb_cfg);
|
||||||
smb->mains.get_property = smb347_mains_get_property;
|
if (IS_ERR(smb->mains))
|
||||||
smb->mains.properties = smb347_mains_properties;
|
return PTR_ERR(smb->mains);
|
||||||
smb->mains.num_properties = ARRAY_SIZE(smb347_mains_properties);
|
|
||||||
smb->mains.supplied_to = battery;
|
|
||||||
smb->mains.num_supplicants = ARRAY_SIZE(battery);
|
|
||||||
ret = power_supply_register(dev, &smb->mains);
|
|
||||||
if (ret < 0)
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (smb->pdata->use_usb) {
|
if (smb->pdata->use_usb) {
|
||||||
smb->usb.name = "smb347-usb";
|
smb->usb = power_supply_register(dev, &smb347_usb_desc,
|
||||||
smb->usb.type = POWER_SUPPLY_TYPE_USB;
|
&mains_usb_cfg);
|
||||||
smb->usb.get_property = smb347_usb_get_property;
|
if (IS_ERR(smb->usb)) {
|
||||||
smb->usb.properties = smb347_usb_properties;
|
|
||||||
smb->usb.num_properties = ARRAY_SIZE(smb347_usb_properties);
|
|
||||||
smb->usb.supplied_to = battery;
|
|
||||||
smb->usb.num_supplicants = ARRAY_SIZE(battery);
|
|
||||||
ret = power_supply_register(dev, &smb->usb);
|
|
||||||
if (ret < 0) {
|
|
||||||
if (smb->pdata->use_mains)
|
if (smb->pdata->use_mains)
|
||||||
power_supply_unregister(&smb->mains);
|
power_supply_unregister(smb->mains);
|
||||||
return ret;
|
return PTR_ERR(smb->usb);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
smb->battery.name = "smb347-battery";
|
battery_cfg.drv_data = smb;
|
||||||
smb->battery.type = POWER_SUPPLY_TYPE_BATTERY;
|
smb->battery = power_supply_register(dev, &smb347_battery_desc,
|
||||||
smb->battery.get_property = smb347_battery_get_property;
|
&battery_cfg);
|
||||||
smb->battery.properties = smb347_battery_properties;
|
if (IS_ERR(smb->battery)) {
|
||||||
smb->battery.num_properties = ARRAY_SIZE(smb347_battery_properties);
|
|
||||||
|
|
||||||
|
|
||||||
ret = power_supply_register(dev, &smb->battery);
|
|
||||||
if (ret < 0) {
|
|
||||||
if (smb->pdata->use_usb)
|
if (smb->pdata->use_usb)
|
||||||
power_supply_unregister(&smb->usb);
|
power_supply_unregister(smb->usb);
|
||||||
if (smb->pdata->use_mains)
|
if (smb->pdata->use_mains)
|
||||||
power_supply_unregister(&smb->mains);
|
power_supply_unregister(smb->mains);
|
||||||
return ret;
|
return PTR_ERR(smb->battery);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1294,11 +1303,11 @@ static int smb347_remove(struct i2c_client *client)
|
||||||
gpio_free(smb->pdata->irq_gpio);
|
gpio_free(smb->pdata->irq_gpio);
|
||||||
}
|
}
|
||||||
|
|
||||||
power_supply_unregister(&smb->battery);
|
power_supply_unregister(smb->battery);
|
||||||
if (smb->pdata->use_usb)
|
if (smb->pdata->use_usb)
|
||||||
power_supply_unregister(&smb->usb);
|
power_supply_unregister(smb->usb);
|
||||||
if (smb->pdata->use_mains)
|
if (smb->pdata->use_mains)
|
||||||
power_supply_unregister(&smb->mains);
|
power_supply_unregister(smb->mains);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -153,12 +153,12 @@ static char *test_power_ac_supplied_to[] = {
|
||||||
"test_battery",
|
"test_battery",
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply test_power_supplies[] = {
|
static struct power_supply *test_power_supplies[TEST_POWER_NUM];
|
||||||
|
|
||||||
|
static const struct power_supply_desc test_power_desc[] = {
|
||||||
[TEST_AC] = {
|
[TEST_AC] = {
|
||||||
.name = "test_ac",
|
.name = "test_ac",
|
||||||
.type = POWER_SUPPLY_TYPE_MAINS,
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
.supplied_to = test_power_ac_supplied_to,
|
|
||||||
.num_supplicants = ARRAY_SIZE(test_power_ac_supplied_to),
|
|
||||||
.properties = test_power_ac_props,
|
.properties = test_power_ac_props,
|
||||||
.num_properties = ARRAY_SIZE(test_power_ac_props),
|
.num_properties = ARRAY_SIZE(test_power_ac_props),
|
||||||
.get_property = test_power_get_ac_property,
|
.get_property = test_power_get_ac_property,
|
||||||
|
@ -173,14 +173,25 @@ static struct power_supply test_power_supplies[] = {
|
||||||
[TEST_USB] = {
|
[TEST_USB] = {
|
||||||
.name = "test_usb",
|
.name = "test_usb",
|
||||||
.type = POWER_SUPPLY_TYPE_USB,
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
.supplied_to = test_power_ac_supplied_to,
|
|
||||||
.num_supplicants = ARRAY_SIZE(test_power_ac_supplied_to),
|
|
||||||
.properties = test_power_ac_props,
|
.properties = test_power_ac_props,
|
||||||
.num_properties = ARRAY_SIZE(test_power_ac_props),
|
.num_properties = ARRAY_SIZE(test_power_ac_props),
|
||||||
.get_property = test_power_get_usb_property,
|
.get_property = test_power_get_usb_property,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_config test_power_configs[] = {
|
||||||
|
{
|
||||||
|
/* test_ac */
|
||||||
|
.supplied_to = test_power_ac_supplied_to,
|
||||||
|
.num_supplicants = ARRAY_SIZE(test_power_ac_supplied_to),
|
||||||
|
}, {
|
||||||
|
/* test_battery */
|
||||||
|
}, {
|
||||||
|
/* test_usb */
|
||||||
|
.supplied_to = test_power_ac_supplied_to,
|
||||||
|
.num_supplicants = ARRAY_SIZE(test_power_ac_supplied_to),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
static int __init test_power_init(void)
|
static int __init test_power_init(void)
|
||||||
{
|
{
|
||||||
|
@ -188,12 +199,16 @@ static int __init test_power_init(void)
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
BUILD_BUG_ON(TEST_POWER_NUM != ARRAY_SIZE(test_power_supplies));
|
BUILD_BUG_ON(TEST_POWER_NUM != ARRAY_SIZE(test_power_supplies));
|
||||||
|
BUILD_BUG_ON(TEST_POWER_NUM != ARRAY_SIZE(test_power_configs));
|
||||||
|
|
||||||
for (i = 0; i < ARRAY_SIZE(test_power_supplies); i++) {
|
for (i = 0; i < ARRAY_SIZE(test_power_supplies); i++) {
|
||||||
ret = power_supply_register(NULL, &test_power_supplies[i]);
|
test_power_supplies[i] = power_supply_register(NULL,
|
||||||
if (ret) {
|
&test_power_desc[i],
|
||||||
|
&test_power_configs[i]);
|
||||||
|
if (IS_ERR(test_power_supplies[i])) {
|
||||||
pr_err("%s: failed to register %s\n", __func__,
|
pr_err("%s: failed to register %s\n", __func__,
|
||||||
test_power_supplies[i].name);
|
test_power_desc[i].name);
|
||||||
|
ret = PTR_ERR(test_power_supplies[i]);
|
||||||
goto failed;
|
goto failed;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -202,7 +217,7 @@ static int __init test_power_init(void)
|
||||||
return 0;
|
return 0;
|
||||||
failed:
|
failed:
|
||||||
while (--i >= 0)
|
while (--i >= 0)
|
||||||
power_supply_unregister(&test_power_supplies[i]);
|
power_supply_unregister(test_power_supplies[i]);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
module_init(test_power_init);
|
module_init(test_power_init);
|
||||||
|
@ -216,13 +231,13 @@ static void __exit test_power_exit(void)
|
||||||
usb_online = 0;
|
usb_online = 0;
|
||||||
battery_status = POWER_SUPPLY_STATUS_DISCHARGING;
|
battery_status = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
for (i = 0; i < ARRAY_SIZE(test_power_supplies); i++)
|
for (i = 0; i < ARRAY_SIZE(test_power_supplies); i++)
|
||||||
power_supply_changed(&test_power_supplies[i]);
|
power_supply_changed(test_power_supplies[i]);
|
||||||
pr_info("%s: 'changed' event sent, sleeping for 10 seconds...\n",
|
pr_info("%s: 'changed' event sent, sleeping for 10 seconds...\n",
|
||||||
__func__);
|
__func__);
|
||||||
ssleep(10);
|
ssleep(10);
|
||||||
|
|
||||||
for (i = 0; i < ARRAY_SIZE(test_power_supplies); i++)
|
for (i = 0; i < ARRAY_SIZE(test_power_supplies); i++)
|
||||||
power_supply_unregister(&test_power_supplies[i]);
|
power_supply_unregister(test_power_supplies[i]);
|
||||||
|
|
||||||
module_initialized = false;
|
module_initialized = false;
|
||||||
}
|
}
|
||||||
|
@ -320,7 +335,7 @@ static inline void signal_power_supply_changed(struct power_supply *psy)
|
||||||
static int param_set_ac_online(const char *key, const struct kernel_param *kp)
|
static int param_set_ac_online(const char *key, const struct kernel_param *kp)
|
||||||
{
|
{
|
||||||
ac_online = map_get_value(map_ac_online, key, ac_online);
|
ac_online = map_get_value(map_ac_online, key, ac_online);
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_AC]);
|
signal_power_supply_changed(test_power_supplies[TEST_AC]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -333,7 +348,7 @@ static int param_get_ac_online(char *buffer, const struct kernel_param *kp)
|
||||||
static int param_set_usb_online(const char *key, const struct kernel_param *kp)
|
static int param_set_usb_online(const char *key, const struct kernel_param *kp)
|
||||||
{
|
{
|
||||||
usb_online = map_get_value(map_ac_online, key, usb_online);
|
usb_online = map_get_value(map_ac_online, key, usb_online);
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_USB]);
|
signal_power_supply_changed(test_power_supplies[TEST_USB]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -347,7 +362,7 @@ static int param_set_battery_status(const char *key,
|
||||||
const struct kernel_param *kp)
|
const struct kernel_param *kp)
|
||||||
{
|
{
|
||||||
battery_status = map_get_value(map_status, key, battery_status);
|
battery_status = map_get_value(map_status, key, battery_status);
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_BATTERY]);
|
signal_power_supply_changed(test_power_supplies[TEST_BATTERY]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -361,7 +376,7 @@ static int param_set_battery_health(const char *key,
|
||||||
const struct kernel_param *kp)
|
const struct kernel_param *kp)
|
||||||
{
|
{
|
||||||
battery_health = map_get_value(map_health, key, battery_health);
|
battery_health = map_get_value(map_health, key, battery_health);
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_BATTERY]);
|
signal_power_supply_changed(test_power_supplies[TEST_BATTERY]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -375,7 +390,7 @@ static int param_set_battery_present(const char *key,
|
||||||
const struct kernel_param *kp)
|
const struct kernel_param *kp)
|
||||||
{
|
{
|
||||||
battery_present = map_get_value(map_present, key, battery_present);
|
battery_present = map_get_value(map_present, key, battery_present);
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_AC]);
|
signal_power_supply_changed(test_power_supplies[TEST_AC]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -391,7 +406,7 @@ static int param_set_battery_technology(const char *key,
|
||||||
{
|
{
|
||||||
battery_technology = map_get_value(map_technology, key,
|
battery_technology = map_get_value(map_technology, key,
|
||||||
battery_technology);
|
battery_technology);
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_BATTERY]);
|
signal_power_supply_changed(test_power_supplies[TEST_BATTERY]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -412,7 +427,7 @@ static int param_set_battery_capacity(const char *key,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
battery_capacity = tmp;
|
battery_capacity = tmp;
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_BATTERY]);
|
signal_power_supply_changed(test_power_supplies[TEST_BATTERY]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -427,7 +442,7 @@ static int param_set_battery_voltage(const char *key,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
battery_voltage = tmp;
|
battery_voltage = tmp;
|
||||||
signal_power_supply_changed(&test_power_supplies[TEST_BATTERY]);
|
signal_power_supply_changed(test_power_supplies[TEST_BATTERY]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,7 +26,7 @@ static struct work_struct bat_work;
|
||||||
|
|
||||||
struct tosa_bat {
|
struct tosa_bat {
|
||||||
int status;
|
int status;
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
int full_chrg;
|
int full_chrg;
|
||||||
|
|
||||||
struct mutex work_lock; /* protects data */
|
struct mutex work_lock; /* protects data */
|
||||||
|
@ -61,7 +61,7 @@ static unsigned long tosa_read_bat(struct tosa_bat *bat)
|
||||||
mutex_lock(&bat_lock);
|
mutex_lock(&bat_lock);
|
||||||
gpio_set_value(bat->gpio_bat, 1);
|
gpio_set_value(bat->gpio_bat, 1);
|
||||||
msleep(5);
|
msleep(5);
|
||||||
value = wm97xx_read_aux_adc(dev_get_drvdata(bat->psy.dev->parent),
|
value = wm97xx_read_aux_adc(dev_get_drvdata(bat->psy->dev.parent),
|
||||||
bat->adc_bat);
|
bat->adc_bat);
|
||||||
gpio_set_value(bat->gpio_bat, 0);
|
gpio_set_value(bat->gpio_bat, 0);
|
||||||
mutex_unlock(&bat_lock);
|
mutex_unlock(&bat_lock);
|
||||||
|
@ -81,7 +81,7 @@ static unsigned long tosa_read_temp(struct tosa_bat *bat)
|
||||||
mutex_lock(&bat_lock);
|
mutex_lock(&bat_lock);
|
||||||
gpio_set_value(bat->gpio_temp, 1);
|
gpio_set_value(bat->gpio_temp, 1);
|
||||||
msleep(5);
|
msleep(5);
|
||||||
value = wm97xx_read_aux_adc(dev_get_drvdata(bat->psy.dev->parent),
|
value = wm97xx_read_aux_adc(dev_get_drvdata(bat->psy->dev.parent),
|
||||||
bat->adc_temp);
|
bat->adc_temp);
|
||||||
gpio_set_value(bat->gpio_temp, 0);
|
gpio_set_value(bat->gpio_temp, 0);
|
||||||
mutex_unlock(&bat_lock);
|
mutex_unlock(&bat_lock);
|
||||||
|
@ -96,7 +96,7 @@ static int tosa_bat_get_property(struct power_supply *psy,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct tosa_bat *bat = container_of(psy, struct tosa_bat, psy);
|
struct tosa_bat *bat = power_supply_get_drvdata(psy);
|
||||||
|
|
||||||
if (bat->is_present && !bat->is_present(bat)
|
if (bat->is_present && !bat->is_present(bat)
|
||||||
&& psp != POWER_SUPPLY_PROP_PRESENT) {
|
&& psp != POWER_SUPPLY_PROP_PRESENT) {
|
||||||
|
@ -158,14 +158,14 @@ static irqreturn_t tosa_bat_gpio_isr(int irq, void *data)
|
||||||
static void tosa_bat_update(struct tosa_bat *bat)
|
static void tosa_bat_update(struct tosa_bat *bat)
|
||||||
{
|
{
|
||||||
int old;
|
int old;
|
||||||
struct power_supply *psy = &bat->psy;
|
struct power_supply *psy = bat->psy;
|
||||||
|
|
||||||
mutex_lock(&bat->work_lock);
|
mutex_lock(&bat->work_lock);
|
||||||
|
|
||||||
old = bat->status;
|
old = bat->status;
|
||||||
|
|
||||||
if (bat->is_present && !bat->is_present(bat)) {
|
if (bat->is_present && !bat->is_present(bat)) {
|
||||||
printk(KERN_NOTICE "%s not present\n", psy->name);
|
printk(KERN_NOTICE "%s not present\n", psy->desc->name);
|
||||||
bat->status = POWER_SUPPLY_STATUS_UNKNOWN;
|
bat->status = POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
bat->full_chrg = -1;
|
bat->full_chrg = -1;
|
||||||
} else if (power_supply_am_i_supplied(psy)) {
|
} else if (power_supply_am_i_supplied(psy)) {
|
||||||
|
@ -222,18 +222,38 @@ static enum power_supply_property tosa_bat_bu_props[] = {
|
||||||
POWER_SUPPLY_PROP_PRESENT,
|
POWER_SUPPLY_PROP_PRESENT,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc tosa_bat_main_desc = {
|
||||||
|
.name = "main-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = tosa_bat_main_props,
|
||||||
|
.num_properties = ARRAY_SIZE(tosa_bat_main_props),
|
||||||
|
.get_property = tosa_bat_get_property,
|
||||||
|
.external_power_changed = tosa_bat_external_power_changed,
|
||||||
|
.use_for_apm = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc tosa_bat_jacket_desc = {
|
||||||
|
.name = "jacket-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = tosa_bat_main_props,
|
||||||
|
.num_properties = ARRAY_SIZE(tosa_bat_main_props),
|
||||||
|
.get_property = tosa_bat_get_property,
|
||||||
|
.external_power_changed = tosa_bat_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc tosa_bat_bu_desc = {
|
||||||
|
.name = "backup-battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = tosa_bat_bu_props,
|
||||||
|
.num_properties = ARRAY_SIZE(tosa_bat_bu_props),
|
||||||
|
.get_property = tosa_bat_get_property,
|
||||||
|
.external_power_changed = tosa_bat_external_power_changed,
|
||||||
|
};
|
||||||
|
|
||||||
static struct tosa_bat tosa_bat_main = {
|
static struct tosa_bat tosa_bat_main = {
|
||||||
.status = POWER_SUPPLY_STATUS_DISCHARGING,
|
.status = POWER_SUPPLY_STATUS_DISCHARGING,
|
||||||
.full_chrg = -1,
|
.full_chrg = -1,
|
||||||
.psy = {
|
.psy = NULL,
|
||||||
.name = "main-battery",
|
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
|
||||||
.properties = tosa_bat_main_props,
|
|
||||||
.num_properties = ARRAY_SIZE(tosa_bat_main_props),
|
|
||||||
.get_property = tosa_bat_get_property,
|
|
||||||
.external_power_changed = tosa_bat_external_power_changed,
|
|
||||||
.use_for_apm = 1,
|
|
||||||
},
|
|
||||||
|
|
||||||
.gpio_full = TOSA_GPIO_BAT0_CRG,
|
.gpio_full = TOSA_GPIO_BAT0_CRG,
|
||||||
.gpio_charge_off = TOSA_GPIO_CHARGE_OFF,
|
.gpio_charge_off = TOSA_GPIO_CHARGE_OFF,
|
||||||
|
@ -254,14 +274,7 @@ static struct tosa_bat tosa_bat_main = {
|
||||||
static struct tosa_bat tosa_bat_jacket = {
|
static struct tosa_bat tosa_bat_jacket = {
|
||||||
.status = POWER_SUPPLY_STATUS_DISCHARGING,
|
.status = POWER_SUPPLY_STATUS_DISCHARGING,
|
||||||
.full_chrg = -1,
|
.full_chrg = -1,
|
||||||
.psy = {
|
.psy = NULL,
|
||||||
.name = "jacket-battery",
|
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
|
||||||
.properties = tosa_bat_main_props,
|
|
||||||
.num_properties = ARRAY_SIZE(tosa_bat_main_props),
|
|
||||||
.get_property = tosa_bat_get_property,
|
|
||||||
.external_power_changed = tosa_bat_external_power_changed,
|
|
||||||
},
|
|
||||||
|
|
||||||
.is_present = tosa_jacket_bat_is_present,
|
.is_present = tosa_jacket_bat_is_present,
|
||||||
.gpio_full = TOSA_GPIO_BAT1_CRG,
|
.gpio_full = TOSA_GPIO_BAT1_CRG,
|
||||||
|
@ -283,15 +296,7 @@ static struct tosa_bat tosa_bat_jacket = {
|
||||||
static struct tosa_bat tosa_bat_bu = {
|
static struct tosa_bat tosa_bat_bu = {
|
||||||
.status = POWER_SUPPLY_STATUS_UNKNOWN,
|
.status = POWER_SUPPLY_STATUS_UNKNOWN,
|
||||||
.full_chrg = -1,
|
.full_chrg = -1,
|
||||||
|
.psy = NULL,
|
||||||
.psy = {
|
|
||||||
.name = "backup-battery",
|
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
|
||||||
.properties = tosa_bat_bu_props,
|
|
||||||
.num_properties = ARRAY_SIZE(tosa_bat_bu_props),
|
|
||||||
.get_property = tosa_bat_get_property,
|
|
||||||
.external_power_changed = tosa_bat_external_power_changed,
|
|
||||||
},
|
|
||||||
|
|
||||||
.gpio_full = -1,
|
.gpio_full = -1,
|
||||||
.gpio_charge_off = -1,
|
.gpio_charge_off = -1,
|
||||||
|
@ -345,6 +350,9 @@ static int tosa_bat_resume(struct platform_device *dev)
|
||||||
static int tosa_bat_probe(struct platform_device *dev)
|
static int tosa_bat_probe(struct platform_device *dev)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
struct power_supply_config main_psy_cfg = {},
|
||||||
|
jacket_psy_cfg = {},
|
||||||
|
bu_psy_cfg = {};
|
||||||
|
|
||||||
if (!machine_is_tosa())
|
if (!machine_is_tosa())
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
@ -358,15 +366,31 @@ static int tosa_bat_probe(struct platform_device *dev)
|
||||||
|
|
||||||
INIT_WORK(&bat_work, tosa_bat_work);
|
INIT_WORK(&bat_work, tosa_bat_work);
|
||||||
|
|
||||||
ret = power_supply_register(&dev->dev, &tosa_bat_main.psy);
|
main_psy_cfg.drv_data = &tosa_bat_main;
|
||||||
if (ret)
|
tosa_bat_main.psy = power_supply_register(&dev->dev,
|
||||||
|
&tosa_bat_main_desc,
|
||||||
|
&main_psy_cfg);
|
||||||
|
if (IS_ERR(tosa_bat_main.psy)) {
|
||||||
|
ret = PTR_ERR(tosa_bat_main.psy);
|
||||||
goto err_psy_reg_main;
|
goto err_psy_reg_main;
|
||||||
ret = power_supply_register(&dev->dev, &tosa_bat_jacket.psy);
|
}
|
||||||
if (ret)
|
|
||||||
|
jacket_psy_cfg.drv_data = &tosa_bat_jacket;
|
||||||
|
tosa_bat_jacket.psy = power_supply_register(&dev->dev,
|
||||||
|
&tosa_bat_jacket_desc,
|
||||||
|
&jacket_psy_cfg);
|
||||||
|
if (IS_ERR(tosa_bat_jacket.psy)) {
|
||||||
|
ret = PTR_ERR(tosa_bat_jacket.psy);
|
||||||
goto err_psy_reg_jacket;
|
goto err_psy_reg_jacket;
|
||||||
ret = power_supply_register(&dev->dev, &tosa_bat_bu.psy);
|
}
|
||||||
if (ret)
|
|
||||||
|
bu_psy_cfg.drv_data = &tosa_bat_bu;
|
||||||
|
tosa_bat_bu.psy = power_supply_register(&dev->dev, &tosa_bat_bu_desc,
|
||||||
|
&bu_psy_cfg);
|
||||||
|
if (IS_ERR(tosa_bat_bu.psy)) {
|
||||||
|
ret = PTR_ERR(tosa_bat_bu.psy);
|
||||||
goto err_psy_reg_bu;
|
goto err_psy_reg_bu;
|
||||||
|
}
|
||||||
|
|
||||||
ret = request_irq(gpio_to_irq(TOSA_GPIO_BAT0_CRG),
|
ret = request_irq(gpio_to_irq(TOSA_GPIO_BAT0_CRG),
|
||||||
tosa_bat_gpio_isr,
|
tosa_bat_gpio_isr,
|
||||||
|
@ -395,11 +419,11 @@ static int tosa_bat_probe(struct platform_device *dev)
|
||||||
err_req_jacket:
|
err_req_jacket:
|
||||||
free_irq(gpio_to_irq(TOSA_GPIO_BAT0_CRG), &tosa_bat_main);
|
free_irq(gpio_to_irq(TOSA_GPIO_BAT0_CRG), &tosa_bat_main);
|
||||||
err_req_main:
|
err_req_main:
|
||||||
power_supply_unregister(&tosa_bat_bu.psy);
|
power_supply_unregister(tosa_bat_bu.psy);
|
||||||
err_psy_reg_bu:
|
err_psy_reg_bu:
|
||||||
power_supply_unregister(&tosa_bat_jacket.psy);
|
power_supply_unregister(tosa_bat_jacket.psy);
|
||||||
err_psy_reg_jacket:
|
err_psy_reg_jacket:
|
||||||
power_supply_unregister(&tosa_bat_main.psy);
|
power_supply_unregister(tosa_bat_main.psy);
|
||||||
err_psy_reg_main:
|
err_psy_reg_main:
|
||||||
|
|
||||||
/* see comment in tosa_bat_remove */
|
/* see comment in tosa_bat_remove */
|
||||||
|
@ -415,9 +439,9 @@ static int tosa_bat_remove(struct platform_device *dev)
|
||||||
free_irq(gpio_to_irq(TOSA_GPIO_BAT1_CRG), &tosa_bat_jacket);
|
free_irq(gpio_to_irq(TOSA_GPIO_BAT1_CRG), &tosa_bat_jacket);
|
||||||
free_irq(gpio_to_irq(TOSA_GPIO_BAT0_CRG), &tosa_bat_main);
|
free_irq(gpio_to_irq(TOSA_GPIO_BAT0_CRG), &tosa_bat_main);
|
||||||
|
|
||||||
power_supply_unregister(&tosa_bat_bu.psy);
|
power_supply_unregister(tosa_bat_bu.psy);
|
||||||
power_supply_unregister(&tosa_bat_jacket.psy);
|
power_supply_unregister(tosa_bat_jacket.psy);
|
||||||
power_supply_unregister(&tosa_bat_main.psy);
|
power_supply_unregister(tosa_bat_main.psy);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Now cancel the bat_work. We won't get any more schedules,
|
* Now cancel the bat_work. We won't get any more schedules,
|
||||||
|
|
|
@ -43,7 +43,7 @@ struct tps65090_charger {
|
||||||
int irq;
|
int irq;
|
||||||
struct task_struct *poll_task;
|
struct task_struct *poll_task;
|
||||||
bool passive_mode;
|
bool passive_mode;
|
||||||
struct power_supply ac;
|
struct power_supply *ac;
|
||||||
struct tps65090_platform_data *pdata;
|
struct tps65090_platform_data *pdata;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -135,8 +135,7 @@ static int tps65090_ac_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 tps65090_charger *charger = container_of(psy,
|
struct tps65090_charger *charger = power_supply_get_drvdata(psy);
|
||||||
struct tps65090_charger, ac);
|
|
||||||
|
|
||||||
if (psp == POWER_SUPPLY_PROP_ONLINE) {
|
if (psp == POWER_SUPPLY_PROP_ONLINE) {
|
||||||
val->intval = charger->ac_online;
|
val->intval = charger->ac_online;
|
||||||
|
@ -190,7 +189,7 @@ static irqreturn_t tps65090_charger_isr(int irq, void *dev_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (charger->prev_ac_online != charger->ac_online)
|
if (charger->prev_ac_online != charger->ac_online)
|
||||||
power_supply_changed(&charger->ac);
|
power_supply_changed(charger->ac);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -229,10 +228,19 @@ static int tps65090_charger_poll_task(void *data)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc tps65090_charger_desc = {
|
||||||
|
.name = "tps65090-ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.get_property = tps65090_ac_get_property,
|
||||||
|
.properties = tps65090_ac_props,
|
||||||
|
.num_properties = ARRAY_SIZE(tps65090_ac_props),
|
||||||
|
};
|
||||||
|
|
||||||
static int tps65090_charger_probe(struct platform_device *pdev)
|
static int tps65090_charger_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct tps65090_charger *cdata;
|
struct tps65090_charger *cdata;
|
||||||
struct tps65090_platform_data *pdata;
|
struct tps65090_platform_data *pdata;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
uint8_t status1 = 0;
|
uint8_t status1 = 0;
|
||||||
int ret;
|
int ret;
|
||||||
int irq;
|
int irq;
|
||||||
|
@ -259,19 +267,16 @@ static int tps65090_charger_probe(struct platform_device *pdev)
|
||||||
cdata->dev = &pdev->dev;
|
cdata->dev = &pdev->dev;
|
||||||
cdata->pdata = pdata;
|
cdata->pdata = pdata;
|
||||||
|
|
||||||
cdata->ac.name = "tps65090-ac";
|
psy_cfg.supplied_to = pdata->supplied_to;
|
||||||
cdata->ac.type = POWER_SUPPLY_TYPE_MAINS;
|
psy_cfg.num_supplicants = pdata->num_supplicants;
|
||||||
cdata->ac.get_property = tps65090_ac_get_property;
|
psy_cfg.of_node = pdev->dev.of_node;
|
||||||
cdata->ac.properties = tps65090_ac_props;
|
psy_cfg.drv_data = cdata;
|
||||||
cdata->ac.num_properties = ARRAY_SIZE(tps65090_ac_props);
|
|
||||||
cdata->ac.supplied_to = pdata->supplied_to;
|
|
||||||
cdata->ac.num_supplicants = pdata->num_supplicants;
|
|
||||||
cdata->ac.of_node = pdev->dev.of_node;
|
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &cdata->ac);
|
cdata->ac = power_supply_register(&pdev->dev, &tps65090_charger_desc,
|
||||||
if (ret) {
|
&psy_cfg);
|
||||||
|
if (IS_ERR(cdata->ac)) {
|
||||||
dev_err(&pdev->dev, "failed: power supply register\n");
|
dev_err(&pdev->dev, "failed: power supply register\n");
|
||||||
return ret;
|
return PTR_ERR(cdata->ac);
|
||||||
}
|
}
|
||||||
|
|
||||||
irq = platform_get_irq(pdev, 0);
|
irq = platform_get_irq(pdev, 0);
|
||||||
|
@ -301,7 +306,7 @@ static int tps65090_charger_probe(struct platform_device *pdev)
|
||||||
goto fail_unregister_supply;
|
goto fail_unregister_supply;
|
||||||
}
|
}
|
||||||
cdata->ac_online = 1;
|
cdata->ac_online = 1;
|
||||||
power_supply_changed(&cdata->ac);
|
power_supply_changed(cdata->ac);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (irq != -ENXIO) {
|
if (irq != -ENXIO) {
|
||||||
|
@ -328,7 +333,7 @@ static int tps65090_charger_probe(struct platform_device *pdev)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fail_unregister_supply:
|
fail_unregister_supply:
|
||||||
power_supply_unregister(&cdata->ac);
|
power_supply_unregister(cdata->ac);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -339,12 +344,12 @@ static int tps65090_charger_remove(struct platform_device *pdev)
|
||||||
|
|
||||||
if (cdata->irq == -ENXIO)
|
if (cdata->irq == -ENXIO)
|
||||||
kthread_stop(cdata->poll_task);
|
kthread_stop(cdata->poll_task);
|
||||||
power_supply_unregister(&cdata->ac);
|
power_supply_unregister(cdata->ac);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct of_device_id of_tps65090_charger_match[] = {
|
static const struct of_device_id of_tps65090_charger_match[] = {
|
||||||
{ .compatible = "ti,tps65090-charger", },
|
{ .compatible = "ti,tps65090-charger", },
|
||||||
{ /* end */ }
|
{ /* end */ }
|
||||||
};
|
};
|
||||||
|
|
|
@ -87,8 +87,8 @@ MODULE_PARM_DESC(allow_usb, "Allow USB charge drawing default current");
|
||||||
|
|
||||||
struct twl4030_bci {
|
struct twl4030_bci {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
struct power_supply ac;
|
struct power_supply *ac;
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct usb_phy *transceiver;
|
struct usb_phy *transceiver;
|
||||||
struct notifier_block usb_nb;
|
struct notifier_block usb_nb;
|
||||||
struct work_struct work;
|
struct work_struct work;
|
||||||
|
@ -318,8 +318,8 @@ static irqreturn_t twl4030_charger_interrupt(int irq, void *arg)
|
||||||
struct twl4030_bci *bci = arg;
|
struct twl4030_bci *bci = arg;
|
||||||
|
|
||||||
dev_dbg(bci->dev, "CHG_PRES irq\n");
|
dev_dbg(bci->dev, "CHG_PRES irq\n");
|
||||||
power_supply_changed(&bci->ac);
|
power_supply_changed(bci->ac);
|
||||||
power_supply_changed(&bci->usb);
|
power_supply_changed(bci->usb);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -347,8 +347,8 @@ static irqreturn_t twl4030_bci_interrupt(int irq, void *arg)
|
||||||
|
|
||||||
if (irqs1 & (TWL4030_ICHGLOW | TWL4030_ICHGEOC)) {
|
if (irqs1 & (TWL4030_ICHGLOW | TWL4030_ICHGEOC)) {
|
||||||
/* charger state change, inform the core */
|
/* charger state change, inform the core */
|
||||||
power_supply_changed(&bci->ac);
|
power_supply_changed(bci->ac);
|
||||||
power_supply_changed(&bci->usb);
|
power_supply_changed(bci->usb);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* various monitoring events, for now we just log them here */
|
/* various monitoring events, for now we just log them here */
|
||||||
|
@ -463,7 +463,7 @@ static int twl4030_bci_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 twl4030_bci *bci = dev_get_drvdata(psy->dev->parent);
|
struct twl4030_bci *bci = dev_get_drvdata(psy->dev.parent);
|
||||||
int is_charging;
|
int is_charging;
|
||||||
int state;
|
int state;
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -472,7 +472,7 @@ static int twl4030_bci_get_property(struct power_supply *psy,
|
||||||
if (state < 0)
|
if (state < 0)
|
||||||
return state;
|
return state;
|
||||||
|
|
||||||
if (psy->type == POWER_SUPPLY_TYPE_USB)
|
if (psy->desc->type == POWER_SUPPLY_TYPE_USB)
|
||||||
is_charging = state & TWL4030_MSTATEC_USB;
|
is_charging = state & TWL4030_MSTATEC_USB;
|
||||||
else
|
else
|
||||||
is_charging = state & TWL4030_MSTATEC_AC;
|
is_charging = state & TWL4030_MSTATEC_AC;
|
||||||
|
@ -488,7 +488,7 @@ static int twl4030_bci_get_property(struct power_supply *psy,
|
||||||
/* charging must be active for meaningful result */
|
/* charging must be active for meaningful result */
|
||||||
if (!is_charging)
|
if (!is_charging)
|
||||||
return -ENODATA;
|
return -ENODATA;
|
||||||
if (psy->type == POWER_SUPPLY_TYPE_USB) {
|
if (psy->desc->type == POWER_SUPPLY_TYPE_USB) {
|
||||||
ret = twl4030bci_read_adc_val(TWL4030_BCIVBUS);
|
ret = twl4030bci_read_adc_val(TWL4030_BCIVBUS);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -558,6 +558,22 @@ twl4030_bci_parse_dt(struct device *dev)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
static const struct power_supply_desc twl4030_bci_ac_desc = {
|
||||||
|
.name = "twl4030_ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = twl4030_charger_props,
|
||||||
|
.num_properties = ARRAY_SIZE(twl4030_charger_props),
|
||||||
|
.get_property = twl4030_bci_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc twl4030_bci_usb_desc = {
|
||||||
|
.name = "twl4030_usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = twl4030_charger_props,
|
||||||
|
.num_properties = ARRAY_SIZE(twl4030_charger_props),
|
||||||
|
.get_property = twl4030_bci_get_property,
|
||||||
|
};
|
||||||
|
|
||||||
static int __init twl4030_bci_probe(struct platform_device *pdev)
|
static int __init twl4030_bci_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct twl4030_bci *bci;
|
struct twl4030_bci *bci;
|
||||||
|
@ -584,28 +600,21 @@ static int __init twl4030_bci_probe(struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
|
|
||||||
platform_set_drvdata(pdev, bci);
|
platform_set_drvdata(pdev, bci);
|
||||||
bci->ac.name = "twl4030_ac";
|
|
||||||
bci->ac.type = POWER_SUPPLY_TYPE_MAINS;
|
|
||||||
bci->ac.properties = twl4030_charger_props;
|
|
||||||
bci->ac.num_properties = ARRAY_SIZE(twl4030_charger_props);
|
|
||||||
bci->ac.get_property = twl4030_bci_get_property;
|
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &bci->ac);
|
bci->ac = power_supply_register(&pdev->dev, &twl4030_bci_ac_desc,
|
||||||
if (ret) {
|
NULL);
|
||||||
|
if (IS_ERR(bci->ac)) {
|
||||||
|
ret = PTR_ERR(bci->ac);
|
||||||
dev_err(&pdev->dev, "failed to register ac: %d\n", ret);
|
dev_err(&pdev->dev, "failed to register ac: %d\n", ret);
|
||||||
goto fail_register_ac;
|
goto fail_register_ac;
|
||||||
}
|
}
|
||||||
|
|
||||||
bci->usb.name = "twl4030_usb";
|
|
||||||
bci->usb.type = POWER_SUPPLY_TYPE_USB;
|
|
||||||
bci->usb.properties = twl4030_charger_props;
|
|
||||||
bci->usb.num_properties = ARRAY_SIZE(twl4030_charger_props);
|
|
||||||
bci->usb.get_property = twl4030_bci_get_property;
|
|
||||||
|
|
||||||
bci->usb_reg = regulator_get(bci->dev, "bci3v1");
|
bci->usb_reg = regulator_get(bci->dev, "bci3v1");
|
||||||
|
|
||||||
ret = power_supply_register(&pdev->dev, &bci->usb);
|
bci->usb = power_supply_register(&pdev->dev, &twl4030_bci_usb_desc,
|
||||||
if (ret) {
|
NULL);
|
||||||
|
if (IS_ERR(bci->usb)) {
|
||||||
|
ret = PTR_ERR(bci->usb);
|
||||||
dev_err(&pdev->dev, "failed to register usb: %d\n", ret);
|
dev_err(&pdev->dev, "failed to register usb: %d\n", ret);
|
||||||
goto fail_register_usb;
|
goto fail_register_usb;
|
||||||
}
|
}
|
||||||
|
@ -670,9 +679,9 @@ fail_unmask_interrupts:
|
||||||
fail_bci_irq:
|
fail_bci_irq:
|
||||||
free_irq(bci->irq_chg, bci);
|
free_irq(bci->irq_chg, bci);
|
||||||
fail_chg_irq:
|
fail_chg_irq:
|
||||||
power_supply_unregister(&bci->usb);
|
power_supply_unregister(bci->usb);
|
||||||
fail_register_usb:
|
fail_register_usb:
|
||||||
power_supply_unregister(&bci->ac);
|
power_supply_unregister(bci->ac);
|
||||||
fail_register_ac:
|
fail_register_ac:
|
||||||
fail_no_battery:
|
fail_no_battery:
|
||||||
kfree(bci);
|
kfree(bci);
|
||||||
|
@ -700,8 +709,8 @@ static int __exit twl4030_bci_remove(struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
free_irq(bci->irq_bci, bci);
|
free_irq(bci->irq_bci, bci);
|
||||||
free_irq(bci->irq_chg, bci);
|
free_irq(bci->irq_chg, bci);
|
||||||
power_supply_unregister(&bci->usb);
|
power_supply_unregister(bci->usb);
|
||||||
power_supply_unregister(&bci->ac);
|
power_supply_unregister(bci->ac);
|
||||||
kfree(bci);
|
kfree(bci);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -19,10 +19,14 @@
|
||||||
#include <linux/sort.h>
|
#include <linux/sort.h>
|
||||||
#include <linux/i2c/twl4030-madc.h>
|
#include <linux/i2c/twl4030-madc.h>
|
||||||
#include <linux/power/twl4030_madc_battery.h>
|
#include <linux/power/twl4030_madc_battery.h>
|
||||||
|
#include <linux/iio/consumer.h>
|
||||||
|
|
||||||
struct twl4030_madc_battery {
|
struct twl4030_madc_battery {
|
||||||
struct power_supply psy;
|
struct power_supply *psy;
|
||||||
struct twl4030_madc_bat_platform_data *pdata;
|
struct twl4030_madc_bat_platform_data *pdata;
|
||||||
|
struct iio_channel *channel_temp;
|
||||||
|
struct iio_channel *channel_ichg;
|
||||||
|
struct iio_channel *channel_vbat;
|
||||||
};
|
};
|
||||||
|
|
||||||
static enum power_supply_property twl4030_madc_bat_props[] = {
|
static enum power_supply_property twl4030_madc_bat_props[] = {
|
||||||
|
@ -38,43 +42,34 @@ static enum power_supply_property twl4030_madc_bat_props[] = {
|
||||||
POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW,
|
POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW,
|
||||||
};
|
};
|
||||||
|
|
||||||
static int madc_read(int index)
|
static int madc_read(struct iio_channel *channel)
|
||||||
{
|
{
|
||||||
struct twl4030_madc_request req;
|
int val, err;
|
||||||
int val;
|
err = iio_read_channel_processed(channel, &val);
|
||||||
|
if (err < 0)
|
||||||
|
return err;
|
||||||
|
|
||||||
req.channels = index;
|
return val;
|
||||||
req.method = TWL4030_MADC_SW2;
|
|
||||||
req.type = TWL4030_MADC_WAIT;
|
|
||||||
req.do_avg = 0;
|
|
||||||
req.raw = false;
|
|
||||||
req.func_cb = NULL;
|
|
||||||
|
|
||||||
val = twl4030_madc_conversion(&req);
|
|
||||||
if (val < 0)
|
|
||||||
return val;
|
|
||||||
|
|
||||||
return req.rbuf[ffs(index) - 1];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int twl4030_madc_bat_get_charging_status(void)
|
static int twl4030_madc_bat_get_charging_status(struct twl4030_madc_battery *bt)
|
||||||
{
|
{
|
||||||
return (madc_read(TWL4030_MADC_ICHG) > 0) ? 1 : 0;
|
return (madc_read(bt->channel_ichg) > 0) ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int twl4030_madc_bat_get_voltage(void)
|
static int twl4030_madc_bat_get_voltage(struct twl4030_madc_battery *bt)
|
||||||
{
|
{
|
||||||
return madc_read(TWL4030_MADC_VBAT);
|
return madc_read(bt->channel_vbat);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int twl4030_madc_bat_get_current(void)
|
static int twl4030_madc_bat_get_current(struct twl4030_madc_battery *bt)
|
||||||
{
|
{
|
||||||
return madc_read(TWL4030_MADC_ICHG) * 1000;
|
return madc_read(bt->channel_ichg) * 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int twl4030_madc_bat_get_temp(void)
|
static int twl4030_madc_bat_get_temp(struct twl4030_madc_battery *bt)
|
||||||
{
|
{
|
||||||
return madc_read(TWL4030_MADC_BTEMP) * 10;
|
return madc_read(bt->channel_temp) * 10;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int twl4030_madc_bat_voltscale(struct twl4030_madc_battery *bat,
|
static int twl4030_madc_bat_voltscale(struct twl4030_madc_battery *bat,
|
||||||
|
@ -84,7 +79,7 @@ static int twl4030_madc_bat_voltscale(struct twl4030_madc_battery *bat,
|
||||||
int i, res = 0;
|
int i, res = 0;
|
||||||
|
|
||||||
/* choose charging curve */
|
/* choose charging curve */
|
||||||
if (twl4030_madc_bat_get_charging_status())
|
if (twl4030_madc_bat_get_charging_status(bat))
|
||||||
calibration = bat->pdata->charging;
|
calibration = bat->pdata->charging;
|
||||||
else
|
else
|
||||||
calibration = bat->pdata->discharging;
|
calibration = bat->pdata->discharging;
|
||||||
|
@ -113,29 +108,28 @@ static int twl4030_madc_bat_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 twl4030_madc_battery *bat = container_of(psy,
|
struct twl4030_madc_battery *bat = power_supply_get_drvdata(psy);
|
||||||
struct twl4030_madc_battery, psy);
|
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
if (twl4030_madc_bat_voltscale(bat,
|
if (twl4030_madc_bat_voltscale(bat,
|
||||||
twl4030_madc_bat_get_voltage()) > 95)
|
twl4030_madc_bat_get_voltage(bat)) > 95)
|
||||||
val->intval = POWER_SUPPLY_STATUS_FULL;
|
val->intval = POWER_SUPPLY_STATUS_FULL;
|
||||||
else {
|
else {
|
||||||
if (twl4030_madc_bat_get_charging_status())
|
if (twl4030_madc_bat_get_charging_status(bat))
|
||||||
val->intval = POWER_SUPPLY_STATUS_CHARGING;
|
val->intval = POWER_SUPPLY_STATUS_CHARGING;
|
||||||
else
|
else
|
||||||
val->intval = POWER_SUPPLY_STATUS_DISCHARGING;
|
val->intval = POWER_SUPPLY_STATUS_DISCHARGING;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
case POWER_SUPPLY_PROP_VOLTAGE_NOW:
|
||||||
val->intval = twl4030_madc_bat_get_voltage() * 1000;
|
val->intval = twl4030_madc_bat_get_voltage(bat) * 1000;
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
case POWER_SUPPLY_PROP_TECHNOLOGY:
|
||||||
val->intval = POWER_SUPPLY_TECHNOLOGY_LION;
|
val->intval = POWER_SUPPLY_TECHNOLOGY_LION;
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_CURRENT_NOW:
|
case POWER_SUPPLY_PROP_CURRENT_NOW:
|
||||||
val->intval = twl4030_madc_bat_get_current();
|
val->intval = twl4030_madc_bat_get_current(bat);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_PRESENT:
|
case POWER_SUPPLY_PROP_PRESENT:
|
||||||
/* assume battery is always present */
|
/* assume battery is always present */
|
||||||
|
@ -143,23 +137,23 @@ static int twl4030_madc_bat_get_property(struct power_supply *psy,
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_CHARGE_NOW: {
|
case POWER_SUPPLY_PROP_CHARGE_NOW: {
|
||||||
int percent = twl4030_madc_bat_voltscale(bat,
|
int percent = twl4030_madc_bat_voltscale(bat,
|
||||||
twl4030_madc_bat_get_voltage());
|
twl4030_madc_bat_get_voltage(bat));
|
||||||
val->intval = (percent * bat->pdata->capacity) / 100;
|
val->intval = (percent * bat->pdata->capacity) / 100;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case POWER_SUPPLY_PROP_CAPACITY:
|
case POWER_SUPPLY_PROP_CAPACITY:
|
||||||
val->intval = twl4030_madc_bat_voltscale(bat,
|
val->intval = twl4030_madc_bat_voltscale(bat,
|
||||||
twl4030_madc_bat_get_voltage());
|
twl4030_madc_bat_get_voltage(bat));
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_CHARGE_FULL:
|
case POWER_SUPPLY_PROP_CHARGE_FULL:
|
||||||
val->intval = bat->pdata->capacity;
|
val->intval = bat->pdata->capacity;
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_TEMP:
|
case POWER_SUPPLY_PROP_TEMP:
|
||||||
val->intval = twl4030_madc_bat_get_temp();
|
val->intval = twl4030_madc_bat_get_temp(bat);
|
||||||
break;
|
break;
|
||||||
case POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW: {
|
case POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW: {
|
||||||
int percent = twl4030_madc_bat_voltscale(bat,
|
int percent = twl4030_madc_bat_voltscale(bat,
|
||||||
twl4030_madc_bat_get_voltage());
|
twl4030_madc_bat_get_voltage(bat));
|
||||||
/* in mAh */
|
/* in mAh */
|
||||||
int chg = (percent * (bat->pdata->capacity/1000))/100;
|
int chg = (percent * (bat->pdata->capacity/1000))/100;
|
||||||
|
|
||||||
|
@ -176,12 +170,19 @@ static int twl4030_madc_bat_get_property(struct power_supply *psy,
|
||||||
|
|
||||||
static void twl4030_madc_bat_ext_changed(struct power_supply *psy)
|
static void twl4030_madc_bat_ext_changed(struct power_supply *psy)
|
||||||
{
|
{
|
||||||
struct twl4030_madc_battery *bat = container_of(psy,
|
power_supply_changed(psy);
|
||||||
struct twl4030_madc_battery, psy);
|
|
||||||
|
|
||||||
power_supply_changed(&bat->psy);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct power_supply_desc twl4030_madc_bat_desc = {
|
||||||
|
.name = "twl4030_battery",
|
||||||
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
|
.properties = twl4030_madc_bat_props,
|
||||||
|
.num_properties = ARRAY_SIZE(twl4030_madc_bat_props),
|
||||||
|
.get_property = twl4030_madc_bat_get_property,
|
||||||
|
.external_power_changed = twl4030_madc_bat_ext_changed,
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
static int twl4030_cmp(const void *a, const void *b)
|
static int twl4030_cmp(const void *a, const void *b)
|
||||||
{
|
{
|
||||||
return ((struct twl4030_madc_bat_calibration *)b)->voltage -
|
return ((struct twl4030_madc_bat_calibration *)b)->voltage -
|
||||||
|
@ -192,19 +193,31 @@ static int twl4030_madc_battery_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct twl4030_madc_battery *twl4030_madc_bat;
|
struct twl4030_madc_battery *twl4030_madc_bat;
|
||||||
struct twl4030_madc_bat_platform_data *pdata = pdev->dev.platform_data;
|
struct twl4030_madc_bat_platform_data *pdata = pdev->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
twl4030_madc_bat = kzalloc(sizeof(*twl4030_madc_bat), GFP_KERNEL);
|
twl4030_madc_bat = devm_kzalloc(&pdev->dev, sizeof(*twl4030_madc_bat),
|
||||||
|
GFP_KERNEL);
|
||||||
if (!twl4030_madc_bat)
|
if (!twl4030_madc_bat)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
twl4030_madc_bat->psy.name = "twl4030_battery";
|
twl4030_madc_bat->channel_temp = iio_channel_get(&pdev->dev, "temp");
|
||||||
twl4030_madc_bat->psy.type = POWER_SUPPLY_TYPE_BATTERY;
|
if (IS_ERR(twl4030_madc_bat->channel_temp)) {
|
||||||
twl4030_madc_bat->psy.properties = twl4030_madc_bat_props;
|
ret = PTR_ERR(twl4030_madc_bat->channel_temp);
|
||||||
twl4030_madc_bat->psy.num_properties =
|
goto err;
|
||||||
ARRAY_SIZE(twl4030_madc_bat_props);
|
}
|
||||||
twl4030_madc_bat->psy.get_property = twl4030_madc_bat_get_property;
|
|
||||||
twl4030_madc_bat->psy.external_power_changed =
|
twl4030_madc_bat->channel_ichg = iio_channel_get(&pdev->dev, "ichg");
|
||||||
twl4030_madc_bat_ext_changed;
|
if (IS_ERR(twl4030_madc_bat->channel_ichg)) {
|
||||||
|
ret = PTR_ERR(twl4030_madc_bat->channel_ichg);
|
||||||
|
goto err_temp;
|
||||||
|
}
|
||||||
|
|
||||||
|
twl4030_madc_bat->channel_vbat = iio_channel_get(&pdev->dev, "vbat");
|
||||||
|
if (IS_ERR(twl4030_madc_bat->channel_vbat)) {
|
||||||
|
ret = PTR_ERR(twl4030_madc_bat->channel_vbat);
|
||||||
|
goto err_ichg;
|
||||||
|
}
|
||||||
|
|
||||||
/* sort charging and discharging calibration data */
|
/* sort charging and discharging calibration data */
|
||||||
sort(pdata->charging, pdata->charging_size,
|
sort(pdata->charging, pdata->charging_size,
|
||||||
|
@ -216,17 +229,36 @@ static int twl4030_madc_battery_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
twl4030_madc_bat->pdata = pdata;
|
twl4030_madc_bat->pdata = pdata;
|
||||||
platform_set_drvdata(pdev, twl4030_madc_bat);
|
platform_set_drvdata(pdev, twl4030_madc_bat);
|
||||||
power_supply_register(&pdev->dev, &twl4030_madc_bat->psy);
|
psy_cfg.drv_data = twl4030_madc_bat;
|
||||||
|
twl4030_madc_bat->psy = power_supply_register(&pdev->dev,
|
||||||
|
&twl4030_madc_bat_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(twl4030_madc_bat->psy)) {
|
||||||
|
ret = PTR_ERR(twl4030_madc_bat->psy);
|
||||||
|
goto err_vbat;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
err_vbat:
|
||||||
|
iio_channel_release(twl4030_madc_bat->channel_vbat);
|
||||||
|
err_ichg:
|
||||||
|
iio_channel_release(twl4030_madc_bat->channel_ichg);
|
||||||
|
err_temp:
|
||||||
|
iio_channel_release(twl4030_madc_bat->channel_temp);
|
||||||
|
err:
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int twl4030_madc_battery_remove(struct platform_device *pdev)
|
static int twl4030_madc_battery_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct twl4030_madc_battery *bat = platform_get_drvdata(pdev);
|
struct twl4030_madc_battery *bat = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
power_supply_unregister(&bat->psy);
|
power_supply_unregister(bat->psy);
|
||||||
kfree(bat);
|
|
||||||
|
iio_channel_release(bat->channel_vbat);
|
||||||
|
iio_channel_release(bat->channel_ichg);
|
||||||
|
iio_channel_release(bat->channel_temp);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -243,3 +275,4 @@ module_platform_driver(twl4030_madc_battery_driver);
|
||||||
MODULE_LICENSE("GPL");
|
MODULE_LICENSE("GPL");
|
||||||
MODULE_AUTHOR("Lukas Märdian <lukas@goldelico.com>");
|
MODULE_AUTHOR("Lukas Märdian <lukas@goldelico.com>");
|
||||||
MODULE_DESCRIPTION("twl4030_madc battery driver");
|
MODULE_DESCRIPTION("twl4030_madc battery driver");
|
||||||
|
MODULE_ALIAS("platform:twl4030_madc_battery");
|
||||||
|
|
|
@ -21,7 +21,8 @@
|
||||||
|
|
||||||
struct wm831x_backup {
|
struct wm831x_backup {
|
||||||
struct wm831x *wm831x;
|
struct wm831x *wm831x;
|
||||||
struct power_supply backup;
|
struct power_supply *backup;
|
||||||
|
struct power_supply_desc backup_desc;
|
||||||
char name[20];
|
char name[20];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -115,7 +116,7 @@ static int wm831x_backup_get_prop(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 wm831x_backup *devdata = dev_get_drvdata(psy->dev->parent);
|
struct wm831x_backup *devdata = dev_get_drvdata(psy->dev.parent);
|
||||||
struct wm831x *wm831x = devdata->wm831x;
|
struct wm831x *wm831x = devdata->wm831x;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
|
@ -166,8 +167,6 @@ static int wm831x_backup_probe(struct platform_device *pdev)
|
||||||
struct wm831x *wm831x = dev_get_drvdata(pdev->dev.parent);
|
struct wm831x *wm831x = dev_get_drvdata(pdev->dev.parent);
|
||||||
struct wm831x_pdata *wm831x_pdata = wm831x->dev->platform_data;
|
struct wm831x_pdata *wm831x_pdata = wm831x->dev->platform_data;
|
||||||
struct wm831x_backup *devdata;
|
struct wm831x_backup *devdata;
|
||||||
struct power_supply *backup;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
devdata = devm_kzalloc(&pdev->dev, sizeof(struct wm831x_backup),
|
devdata = devm_kzalloc(&pdev->dev, sizeof(struct wm831x_backup),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
|
@ -177,8 +176,6 @@ static int wm831x_backup_probe(struct platform_device *pdev)
|
||||||
devdata->wm831x = wm831x;
|
devdata->wm831x = wm831x;
|
||||||
platform_set_drvdata(pdev, devdata);
|
platform_set_drvdata(pdev, devdata);
|
||||||
|
|
||||||
backup = &devdata->backup;
|
|
||||||
|
|
||||||
/* We ignore configuration failures since we can still read
|
/* We ignore configuration failures since we can still read
|
||||||
* back the status without enabling the charger (which may
|
* back the status without enabling the charger (which may
|
||||||
* already be enabled anyway).
|
* already be enabled anyway).
|
||||||
|
@ -192,21 +189,22 @@ static int wm831x_backup_probe(struct platform_device *pdev)
|
||||||
snprintf(devdata->name, sizeof(devdata->name),
|
snprintf(devdata->name, sizeof(devdata->name),
|
||||||
"wm831x-backup");
|
"wm831x-backup");
|
||||||
|
|
||||||
backup->name = devdata->name;
|
devdata->backup_desc.name = devdata->name;
|
||||||
backup->type = POWER_SUPPLY_TYPE_BATTERY;
|
devdata->backup_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
backup->properties = wm831x_backup_props;
|
devdata->backup_desc.properties = wm831x_backup_props;
|
||||||
backup->num_properties = ARRAY_SIZE(wm831x_backup_props);
|
devdata->backup_desc.num_properties = ARRAY_SIZE(wm831x_backup_props);
|
||||||
backup->get_property = wm831x_backup_get_prop;
|
devdata->backup_desc.get_property = wm831x_backup_get_prop;
|
||||||
ret = power_supply_register(&pdev->dev, backup);
|
devdata->backup = power_supply_register(&pdev->dev,
|
||||||
|
&devdata->backup_desc, NULL);
|
||||||
|
|
||||||
return ret;
|
return PTR_ERR_OR_ZERO(devdata->backup);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int wm831x_backup_remove(struct platform_device *pdev)
|
static int wm831x_backup_remove(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct wm831x_backup *devdata = platform_get_drvdata(pdev);
|
struct wm831x_backup *devdata = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
power_supply_unregister(&devdata->backup);
|
power_supply_unregister(devdata->backup);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,9 +21,12 @@
|
||||||
|
|
||||||
struct wm831x_power {
|
struct wm831x_power {
|
||||||
struct wm831x *wm831x;
|
struct wm831x *wm831x;
|
||||||
struct power_supply wall;
|
struct power_supply *wall;
|
||||||
struct power_supply usb;
|
struct power_supply *usb;
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
|
struct power_supply_desc wall_desc;
|
||||||
|
struct power_supply_desc usb_desc;
|
||||||
|
struct power_supply_desc battery_desc;
|
||||||
char wall_name[20];
|
char wall_name[20];
|
||||||
char usb_name[20];
|
char usb_name[20];
|
||||||
char battery_name[20];
|
char battery_name[20];
|
||||||
|
@ -67,7 +70,7 @@ static int wm831x_wall_get_prop(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 wm831x_power *wm831x_power = dev_get_drvdata(psy->dev->parent);
|
struct wm831x_power *wm831x_power = dev_get_drvdata(psy->dev.parent);
|
||||||
struct wm831x *wm831x = wm831x_power->wm831x;
|
struct wm831x *wm831x = wm831x_power->wm831x;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
|
@ -98,7 +101,7 @@ static int wm831x_usb_get_prop(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 wm831x_power *wm831x_power = dev_get_drvdata(psy->dev->parent);
|
struct wm831x_power *wm831x_power = dev_get_drvdata(psy->dev.parent);
|
||||||
struct wm831x *wm831x = wm831x_power->wm831x;
|
struct wm831x *wm831x = wm831x_power->wm831x;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
|
@ -393,7 +396,7 @@ static int wm831x_bat_get_prop(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 wm831x_power *wm831x_power = dev_get_drvdata(psy->dev->parent);
|
struct wm831x_power *wm831x_power = dev_get_drvdata(psy->dev.parent);
|
||||||
struct wm831x *wm831x = wm831x_power->wm831x;
|
struct wm831x *wm831x = wm831x_power->wm831x;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
|
@ -451,7 +454,7 @@ static irqreturn_t wm831x_bat_irq(int irq, void *data)
|
||||||
/* The battery charger is autonomous so we don't need to do
|
/* The battery charger is autonomous so we don't need to do
|
||||||
* anything except kick user space */
|
* anything except kick user space */
|
||||||
if (wm831x_power->have_battery)
|
if (wm831x_power->have_battery)
|
||||||
power_supply_changed(&wm831x_power->battery);
|
power_supply_changed(wm831x_power->battery);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -482,9 +485,9 @@ static irqreturn_t wm831x_pwr_src_irq(int irq, void *data)
|
||||||
|
|
||||||
/* Just notify for everything - little harm in overnotifying. */
|
/* Just notify for everything - little harm in overnotifying. */
|
||||||
if (wm831x_power->have_battery)
|
if (wm831x_power->have_battery)
|
||||||
power_supply_changed(&wm831x_power->battery);
|
power_supply_changed(wm831x_power->battery);
|
||||||
power_supply_changed(&wm831x_power->usb);
|
power_supply_changed(wm831x_power->usb);
|
||||||
power_supply_changed(&wm831x_power->wall);
|
power_supply_changed(wm831x_power->wall);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
@ -494,9 +497,6 @@ static int wm831x_power_probe(struct platform_device *pdev)
|
||||||
struct wm831x *wm831x = dev_get_drvdata(pdev->dev.parent);
|
struct wm831x *wm831x = dev_get_drvdata(pdev->dev.parent);
|
||||||
struct wm831x_pdata *wm831x_pdata = wm831x->dev->platform_data;
|
struct wm831x_pdata *wm831x_pdata = wm831x->dev->platform_data;
|
||||||
struct wm831x_power *power;
|
struct wm831x_power *power;
|
||||||
struct power_supply *usb;
|
|
||||||
struct power_supply *battery;
|
|
||||||
struct power_supply *wall;
|
|
||||||
int ret, irq, i;
|
int ret, irq, i;
|
||||||
|
|
||||||
power = kzalloc(sizeof(struct wm831x_power), GFP_KERNEL);
|
power = kzalloc(sizeof(struct wm831x_power), GFP_KERNEL);
|
||||||
|
@ -506,10 +506,6 @@ static int wm831x_power_probe(struct platform_device *pdev)
|
||||||
power->wm831x = wm831x;
|
power->wm831x = wm831x;
|
||||||
platform_set_drvdata(pdev, power);
|
platform_set_drvdata(pdev, power);
|
||||||
|
|
||||||
usb = &power->usb;
|
|
||||||
battery = &power->battery;
|
|
||||||
wall = &power->wall;
|
|
||||||
|
|
||||||
if (wm831x_pdata && wm831x_pdata->wm831x_num) {
|
if (wm831x_pdata && wm831x_pdata->wm831x_num) {
|
||||||
snprintf(power->wall_name, sizeof(power->wall_name),
|
snprintf(power->wall_name, sizeof(power->wall_name),
|
||||||
"wm831x-wall.%d", wm831x_pdata->wm831x_num);
|
"wm831x-wall.%d", wm831x_pdata->wm831x_num);
|
||||||
|
@ -531,23 +527,28 @@ static int wm831x_power_probe(struct platform_device *pdev)
|
||||||
*/
|
*/
|
||||||
wm831x_config_battery(wm831x);
|
wm831x_config_battery(wm831x);
|
||||||
|
|
||||||
wall->name = power->wall_name;
|
power->wall_desc.name = power->wall_name;
|
||||||
wall->type = POWER_SUPPLY_TYPE_MAINS;
|
power->wall_desc.type = POWER_SUPPLY_TYPE_MAINS;
|
||||||
wall->properties = wm831x_wall_props;
|
power->wall_desc.properties = wm831x_wall_props;
|
||||||
wall->num_properties = ARRAY_SIZE(wm831x_wall_props);
|
power->wall_desc.num_properties = ARRAY_SIZE(wm831x_wall_props);
|
||||||
wall->get_property = wm831x_wall_get_prop;
|
power->wall_desc.get_property = wm831x_wall_get_prop;
|
||||||
ret = power_supply_register(&pdev->dev, wall);
|
power->wall = power_supply_register(&pdev->dev, &power->wall_desc,
|
||||||
if (ret)
|
NULL);
|
||||||
|
if (IS_ERR(power->wall)) {
|
||||||
|
ret = PTR_ERR(power->wall);
|
||||||
goto err_kmalloc;
|
goto err_kmalloc;
|
||||||
|
}
|
||||||
|
|
||||||
usb->name = power->usb_name,
|
power->usb_desc.name = power->usb_name,
|
||||||
usb->type = POWER_SUPPLY_TYPE_USB;
|
power->usb_desc.type = POWER_SUPPLY_TYPE_USB;
|
||||||
usb->properties = wm831x_usb_props;
|
power->usb_desc.properties = wm831x_usb_props;
|
||||||
usb->num_properties = ARRAY_SIZE(wm831x_usb_props);
|
power->usb_desc.num_properties = ARRAY_SIZE(wm831x_usb_props);
|
||||||
usb->get_property = wm831x_usb_get_prop;
|
power->usb_desc.get_property = wm831x_usb_get_prop;
|
||||||
ret = power_supply_register(&pdev->dev, usb);
|
power->usb = power_supply_register(&pdev->dev, &power->usb_desc, NULL);
|
||||||
if (ret)
|
if (IS_ERR(power->usb)) {
|
||||||
|
ret = PTR_ERR(power->usb);
|
||||||
goto err_wall;
|
goto err_wall;
|
||||||
|
}
|
||||||
|
|
||||||
ret = wm831x_reg_read(wm831x, WM831X_CHARGER_CONTROL_1);
|
ret = wm831x_reg_read(wm831x, WM831X_CHARGER_CONTROL_1);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
|
@ -555,14 +556,18 @@ static int wm831x_power_probe(struct platform_device *pdev)
|
||||||
power->have_battery = ret & WM831X_CHG_ENA;
|
power->have_battery = ret & WM831X_CHG_ENA;
|
||||||
|
|
||||||
if (power->have_battery) {
|
if (power->have_battery) {
|
||||||
battery->name = power->battery_name;
|
power->battery_desc.name = power->battery_name;
|
||||||
battery->properties = wm831x_bat_props;
|
power->battery_desc.properties = wm831x_bat_props;
|
||||||
battery->num_properties = ARRAY_SIZE(wm831x_bat_props);
|
power->battery_desc.num_properties = ARRAY_SIZE(wm831x_bat_props);
|
||||||
battery->get_property = wm831x_bat_get_prop;
|
power->battery_desc.get_property = wm831x_bat_get_prop;
|
||||||
battery->use_for_apm = 1;
|
power->battery_desc.use_for_apm = 1;
|
||||||
ret = power_supply_register(&pdev->dev, battery);
|
power->battery = power_supply_register(&pdev->dev,
|
||||||
if (ret)
|
&power->battery_desc,
|
||||||
goto err_usb;
|
NULL);
|
||||||
|
if (IS_ERR(power->battery)) {
|
||||||
|
ret = PTR_ERR(power->battery);
|
||||||
|
goto err_usb;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
irq = wm831x_irq(wm831x, platform_get_irq_byname(pdev, "SYSLO"));
|
irq = wm831x_irq(wm831x, platform_get_irq_byname(pdev, "SYSLO"));
|
||||||
|
@ -615,11 +620,11 @@ err_syslo:
|
||||||
free_irq(irq, power);
|
free_irq(irq, power);
|
||||||
err_battery:
|
err_battery:
|
||||||
if (power->have_battery)
|
if (power->have_battery)
|
||||||
power_supply_unregister(battery);
|
power_supply_unregister(power->battery);
|
||||||
err_usb:
|
err_usb:
|
||||||
power_supply_unregister(usb);
|
power_supply_unregister(power->usb);
|
||||||
err_wall:
|
err_wall:
|
||||||
power_supply_unregister(wall);
|
power_supply_unregister(power->wall);
|
||||||
err_kmalloc:
|
err_kmalloc:
|
||||||
kfree(power);
|
kfree(power);
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -645,9 +650,9 @@ static int wm831x_power_remove(struct platform_device *pdev)
|
||||||
free_irq(irq, wm831x_power);
|
free_irq(irq, wm831x_power);
|
||||||
|
|
||||||
if (wm831x_power->have_battery)
|
if (wm831x_power->have_battery)
|
||||||
power_supply_unregister(&wm831x_power->battery);
|
power_supply_unregister(wm831x_power->battery);
|
||||||
power_supply_unregister(&wm831x_power->wall);
|
power_supply_unregister(wm831x_power->wall);
|
||||||
power_supply_unregister(&wm831x_power->usb);
|
power_supply_unregister(wm831x_power->usb);
|
||||||
kfree(wm831x_power);
|
kfree(wm831x_power);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -196,14 +196,14 @@ static irqreturn_t wm8350_charger_handler(int irq, void *data)
|
||||||
break;
|
break;
|
||||||
case WM8350_IRQ_CHG_TO:
|
case WM8350_IRQ_CHG_TO:
|
||||||
dev_err(wm8350->dev, "charger timeout\n");
|
dev_err(wm8350->dev, "charger timeout\n");
|
||||||
power_supply_changed(&power->battery);
|
power_supply_changed(power->battery);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WM8350_IRQ_CHG_BAT_HOT:
|
case WM8350_IRQ_CHG_BAT_HOT:
|
||||||
case WM8350_IRQ_CHG_BAT_COLD:
|
case WM8350_IRQ_CHG_BAT_COLD:
|
||||||
case WM8350_IRQ_CHG_START:
|
case WM8350_IRQ_CHG_START:
|
||||||
case WM8350_IRQ_CHG_END:
|
case WM8350_IRQ_CHG_END:
|
||||||
power_supply_changed(&power->battery);
|
power_supply_changed(power->battery);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WM8350_IRQ_CHG_FAST_RDY:
|
case WM8350_IRQ_CHG_FAST_RDY:
|
||||||
|
@ -231,9 +231,9 @@ static irqreturn_t wm8350_charger_handler(int irq, void *data)
|
||||||
case WM8350_IRQ_EXT_WALL_FB:
|
case WM8350_IRQ_EXT_WALL_FB:
|
||||||
wm8350_charger_config(wm8350, policy);
|
wm8350_charger_config(wm8350, policy);
|
||||||
case WM8350_IRQ_EXT_BAT_FB: /* Fall through */
|
case WM8350_IRQ_EXT_BAT_FB: /* Fall through */
|
||||||
power_supply_changed(&power->battery);
|
power_supply_changed(power->battery);
|
||||||
power_supply_changed(&power->usb);
|
power_supply_changed(power->usb);
|
||||||
power_supply_changed(&power->ac);
|
power_supply_changed(power->ac);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -250,7 +250,7 @@ static int wm8350_ac_get_prop(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 wm8350 *wm8350 = dev_get_drvdata(psy->dev->parent);
|
struct wm8350 *wm8350 = dev_get_drvdata(psy->dev.parent);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -280,7 +280,7 @@ static int wm8350_usb_get_prop(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 wm8350 *wm8350 = dev_get_drvdata(psy->dev->parent);
|
struct wm8350 *wm8350 = dev_get_drvdata(psy->dev.parent);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -346,7 +346,7 @@ static int wm8350_bat_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 wm8350 *wm8350 = dev_get_drvdata(psy->dev->parent);
|
struct wm8350 *wm8350 = dev_get_drvdata(psy->dev.parent);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -382,6 +382,30 @@ static enum power_supply_property wm8350_bat_props[] = {
|
||||||
POWER_SUPPLY_PROP_CHARGE_TYPE,
|
POWER_SUPPLY_PROP_CHARGE_TYPE,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc wm8350_ac_desc = {
|
||||||
|
.name = "wm8350-ac",
|
||||||
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
|
.properties = wm8350_ac_props,
|
||||||
|
.num_properties = ARRAY_SIZE(wm8350_ac_props),
|
||||||
|
.get_property = wm8350_ac_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc wm8350_battery_desc = {
|
||||||
|
.name = "wm8350-battery",
|
||||||
|
.properties = wm8350_bat_props,
|
||||||
|
.num_properties = ARRAY_SIZE(wm8350_bat_props),
|
||||||
|
.get_property = wm8350_bat_get_property,
|
||||||
|
.use_for_apm = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct power_supply_desc wm8350_usb_desc = {
|
||||||
|
.name = "wm8350-usb",
|
||||||
|
.type = POWER_SUPPLY_TYPE_USB,
|
||||||
|
.properties = wm8350_usb_props,
|
||||||
|
.num_properties = ARRAY_SIZE(wm8350_usb_props),
|
||||||
|
.get_property = wm8350_usb_get_prop,
|
||||||
|
};
|
||||||
|
|
||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
* Initialisation
|
* Initialisation
|
||||||
*********************************************************************/
|
*********************************************************************/
|
||||||
|
@ -447,37 +471,24 @@ static int wm8350_power_probe(struct platform_device *pdev)
|
||||||
struct wm8350 *wm8350 = platform_get_drvdata(pdev);
|
struct wm8350 *wm8350 = platform_get_drvdata(pdev);
|
||||||
struct wm8350_power *power = &wm8350->power;
|
struct wm8350_power *power = &wm8350->power;
|
||||||
struct wm8350_charger_policy *policy = power->policy;
|
struct wm8350_charger_policy *policy = power->policy;
|
||||||
struct power_supply *usb = &power->usb;
|
|
||||||
struct power_supply *battery = &power->battery;
|
|
||||||
struct power_supply *ac = &power->ac;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ac->name = "wm8350-ac";
|
power->ac = power_supply_register(&pdev->dev, &wm8350_ac_desc, NULL);
|
||||||
ac->type = POWER_SUPPLY_TYPE_MAINS;
|
if (IS_ERR(power->ac))
|
||||||
ac->properties = wm8350_ac_props;
|
return PTR_ERR(power->ac);
|
||||||
ac->num_properties = ARRAY_SIZE(wm8350_ac_props);
|
|
||||||
ac->get_property = wm8350_ac_get_prop;
|
|
||||||
ret = power_supply_register(&pdev->dev, ac);
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
battery->name = "wm8350-battery";
|
power->battery = power_supply_register(&pdev->dev, &wm8350_battery_desc,
|
||||||
battery->properties = wm8350_bat_props;
|
NULL);
|
||||||
battery->num_properties = ARRAY_SIZE(wm8350_bat_props);
|
if (IS_ERR(power->battery)) {
|
||||||
battery->get_property = wm8350_bat_get_property;
|
ret = PTR_ERR(power->battery);
|
||||||
battery->use_for_apm = 1;
|
|
||||||
ret = power_supply_register(&pdev->dev, battery);
|
|
||||||
if (ret)
|
|
||||||
goto battery_failed;
|
goto battery_failed;
|
||||||
|
}
|
||||||
|
|
||||||
usb->name = "wm8350-usb",
|
power->usb = power_supply_register(&pdev->dev, &wm8350_usb_desc, NULL);
|
||||||
usb->type = POWER_SUPPLY_TYPE_USB;
|
if (IS_ERR(power->usb)) {
|
||||||
usb->properties = wm8350_usb_props;
|
ret = PTR_ERR(power->usb);
|
||||||
usb->num_properties = ARRAY_SIZE(wm8350_usb_props);
|
|
||||||
usb->get_property = wm8350_usb_get_prop;
|
|
||||||
ret = power_supply_register(&pdev->dev, usb);
|
|
||||||
if (ret)
|
|
||||||
goto usb_failed;
|
goto usb_failed;
|
||||||
|
}
|
||||||
|
|
||||||
ret = device_create_file(&pdev->dev, &dev_attr_charger_state);
|
ret = device_create_file(&pdev->dev, &dev_attr_charger_state);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
|
@ -494,9 +505,9 @@ static int wm8350_power_probe(struct platform_device *pdev)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
usb_failed:
|
usb_failed:
|
||||||
power_supply_unregister(battery);
|
power_supply_unregister(power->battery);
|
||||||
battery_failed:
|
battery_failed:
|
||||||
power_supply_unregister(ac);
|
power_supply_unregister(power->ac);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -508,9 +519,9 @@ static int wm8350_power_remove(struct platform_device *pdev)
|
||||||
|
|
||||||
free_charger_irq(wm8350);
|
free_charger_irq(wm8350);
|
||||||
device_remove_file(&pdev->dev, &dev_attr_charger_state);
|
device_remove_file(&pdev->dev, &dev_attr_charger_state);
|
||||||
power_supply_unregister(&power->battery);
|
power_supply_unregister(power->battery);
|
||||||
power_supply_unregister(&power->ac);
|
power_supply_unregister(power->ac);
|
||||||
power_supply_unregister(&power->usb);
|
power_supply_unregister(power->usb);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -32,20 +32,20 @@ static enum power_supply_property *prop;
|
||||||
|
|
||||||
static unsigned long wm97xx_read_bat(struct power_supply *bat_ps)
|
static unsigned long wm97xx_read_bat(struct power_supply *bat_ps)
|
||||||
{
|
{
|
||||||
struct wm97xx_pdata *wmdata = bat_ps->dev->parent->platform_data;
|
struct wm97xx_pdata *wmdata = bat_ps->dev.parent->platform_data;
|
||||||
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
||||||
|
|
||||||
return wm97xx_read_aux_adc(dev_get_drvdata(bat_ps->dev->parent),
|
return wm97xx_read_aux_adc(dev_get_drvdata(bat_ps->dev.parent),
|
||||||
pdata->batt_aux) * pdata->batt_mult /
|
pdata->batt_aux) * pdata->batt_mult /
|
||||||
pdata->batt_div;
|
pdata->batt_div;
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned long wm97xx_read_temp(struct power_supply *bat_ps)
|
static unsigned long wm97xx_read_temp(struct power_supply *bat_ps)
|
||||||
{
|
{
|
||||||
struct wm97xx_pdata *wmdata = bat_ps->dev->parent->platform_data;
|
struct wm97xx_pdata *wmdata = bat_ps->dev.parent->platform_data;
|
||||||
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
||||||
|
|
||||||
return wm97xx_read_aux_adc(dev_get_drvdata(bat_ps->dev->parent),
|
return wm97xx_read_aux_adc(dev_get_drvdata(bat_ps->dev.parent),
|
||||||
pdata->temp_aux) * pdata->temp_mult /
|
pdata->temp_aux) * pdata->temp_mult /
|
||||||
pdata->temp_div;
|
pdata->temp_div;
|
||||||
}
|
}
|
||||||
|
@ -54,7 +54,7 @@ static int wm97xx_bat_get_property(struct power_supply *bat_ps,
|
||||||
enum power_supply_property psp,
|
enum power_supply_property psp,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct wm97xx_pdata *wmdata = bat_ps->dev->parent->platform_data;
|
struct wm97xx_pdata *wmdata = bat_ps->dev.parent->platform_data;
|
||||||
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -105,7 +105,7 @@ static void wm97xx_bat_external_power_changed(struct power_supply *bat_ps)
|
||||||
static void wm97xx_bat_update(struct power_supply *bat_ps)
|
static void wm97xx_bat_update(struct power_supply *bat_ps)
|
||||||
{
|
{
|
||||||
int old_status = bat_status;
|
int old_status = bat_status;
|
||||||
struct wm97xx_pdata *wmdata = bat_ps->dev->parent->platform_data;
|
struct wm97xx_pdata *wmdata = bat_ps->dev.parent->platform_data;
|
||||||
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
struct wm97xx_batt_pdata *pdata = wmdata->batt_pdata;
|
||||||
|
|
||||||
mutex_lock(&work_lock);
|
mutex_lock(&work_lock);
|
||||||
|
@ -117,7 +117,7 @@ static void wm97xx_bat_update(struct power_supply *bat_ps)
|
||||||
POWER_SUPPLY_STATUS_UNKNOWN;
|
POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
|
|
||||||
if (old_status != bat_status) {
|
if (old_status != bat_status) {
|
||||||
pr_debug("%s: %i -> %i\n", bat_ps->name, old_status,
|
pr_debug("%s: %i -> %i\n", bat_ps->desc->name, old_status,
|
||||||
bat_status);
|
bat_status);
|
||||||
power_supply_changed(bat_ps);
|
power_supply_changed(bat_ps);
|
||||||
}
|
}
|
||||||
|
@ -125,7 +125,8 @@ static void wm97xx_bat_update(struct power_supply *bat_ps)
|
||||||
mutex_unlock(&work_lock);
|
mutex_unlock(&work_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct power_supply bat_ps = {
|
static struct power_supply *bat_psy;
|
||||||
|
static struct power_supply_desc bat_psy_desc = {
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
.get_property = wm97xx_bat_get_property,
|
.get_property = wm97xx_bat_get_property,
|
||||||
.external_power_changed = wm97xx_bat_external_power_changed,
|
.external_power_changed = wm97xx_bat_external_power_changed,
|
||||||
|
@ -134,7 +135,7 @@ static struct power_supply bat_ps = {
|
||||||
|
|
||||||
static void wm97xx_bat_work(struct work_struct *work)
|
static void wm97xx_bat_work(struct work_struct *work)
|
||||||
{
|
{
|
||||||
wm97xx_bat_update(&bat_ps);
|
wm97xx_bat_update(bat_psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static irqreturn_t wm97xx_chrg_irq(int irq, void *data)
|
static irqreturn_t wm97xx_chrg_irq(int irq, void *data)
|
||||||
|
@ -237,18 +238,20 @@ static int wm97xx_bat_probe(struct platform_device *dev)
|
||||||
dev_info(&dev->dev, "Please consider setting proper battery "
|
dev_info(&dev->dev, "Please consider setting proper battery "
|
||||||
"name in platform definition file, falling "
|
"name in platform definition file, falling "
|
||||||
"back to name \"wm97xx-batt\"\n");
|
"back to name \"wm97xx-batt\"\n");
|
||||||
bat_ps.name = "wm97xx-batt";
|
bat_psy_desc.name = "wm97xx-batt";
|
||||||
} else
|
} else
|
||||||
bat_ps.name = pdata->batt_name;
|
bat_psy_desc.name = pdata->batt_name;
|
||||||
|
|
||||||
bat_ps.properties = prop;
|
bat_psy_desc.properties = prop;
|
||||||
bat_ps.num_properties = props;
|
bat_psy_desc.num_properties = props;
|
||||||
|
|
||||||
ret = power_supply_register(&dev->dev, &bat_ps);
|
bat_psy = power_supply_register(&dev->dev, &bat_psy_desc, NULL);
|
||||||
if (!ret)
|
if (!IS_ERR(bat_psy)) {
|
||||||
schedule_work(&bat_work);
|
schedule_work(&bat_work);
|
||||||
else
|
} else {
|
||||||
|
ret = PTR_ERR(bat_psy);
|
||||||
goto err4;
|
goto err4;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
err4:
|
err4:
|
||||||
|
@ -273,7 +276,7 @@ static int wm97xx_bat_remove(struct platform_device *dev)
|
||||||
gpio_free(pdata->charge_gpio);
|
gpio_free(pdata->charge_gpio);
|
||||||
}
|
}
|
||||||
cancel_work_sync(&bat_work);
|
cancel_work_sync(&bat_work);
|
||||||
power_supply_unregister(&bat_ps);
|
power_supply_unregister(bat_psy);
|
||||||
kfree(prop);
|
kfree(prop);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,12 +21,13 @@
|
||||||
#define Z2_DEFAULT_NAME "Z2"
|
#define Z2_DEFAULT_NAME "Z2"
|
||||||
|
|
||||||
struct z2_charger {
|
struct z2_charger {
|
||||||
struct z2_battery_info *info;
|
struct z2_battery_info *info;
|
||||||
int bat_status;
|
int bat_status;
|
||||||
struct i2c_client *client;
|
struct i2c_client *client;
|
||||||
struct power_supply batt_ps;
|
struct power_supply *batt_ps;
|
||||||
struct mutex work_lock;
|
struct power_supply_desc batt_ps_desc;
|
||||||
struct work_struct bat_work;
|
struct mutex work_lock;
|
||||||
|
struct work_struct bat_work;
|
||||||
};
|
};
|
||||||
|
|
||||||
static unsigned long z2_read_bat(struct z2_charger *charger)
|
static unsigned long z2_read_bat(struct z2_charger *charger)
|
||||||
|
@ -44,8 +45,7 @@ static int z2_batt_get_property(struct power_supply *batt_ps,
|
||||||
enum power_supply_property psp,
|
enum power_supply_property psp,
|
||||||
union power_supply_propval *val)
|
union power_supply_propval *val)
|
||||||
{
|
{
|
||||||
struct z2_charger *charger = container_of(batt_ps, struct z2_charger,
|
struct z2_charger *charger = power_supply_get_drvdata(batt_ps);
|
||||||
batt_ps);
|
|
||||||
struct z2_battery_info *info = charger->info;
|
struct z2_battery_info *info = charger->info;
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
|
@ -85,8 +85,8 @@ static int z2_batt_get_property(struct power_supply *batt_ps,
|
||||||
|
|
||||||
static void z2_batt_ext_power_changed(struct power_supply *batt_ps)
|
static void z2_batt_ext_power_changed(struct power_supply *batt_ps)
|
||||||
{
|
{
|
||||||
struct z2_charger *charger = container_of(batt_ps, struct z2_charger,
|
struct z2_charger *charger = power_supply_get_drvdata(batt_ps);
|
||||||
batt_ps);
|
|
||||||
schedule_work(&charger->bat_work);
|
schedule_work(&charger->bat_work);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -106,9 +106,10 @@ static void z2_batt_update(struct z2_charger *charger)
|
||||||
POWER_SUPPLY_STATUS_UNKNOWN;
|
POWER_SUPPLY_STATUS_UNKNOWN;
|
||||||
|
|
||||||
if (old_status != charger->bat_status) {
|
if (old_status != charger->bat_status) {
|
||||||
pr_debug("%s: %i -> %i\n", charger->batt_ps.name, old_status,
|
pr_debug("%s: %i -> %i\n", charger->batt_ps->desc->name,
|
||||||
charger->bat_status);
|
old_status,
|
||||||
power_supply_changed(&charger->batt_ps);
|
charger->bat_status);
|
||||||
|
power_supply_changed(charger->batt_ps);
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_unlock(&charger->work_lock);
|
mutex_unlock(&charger->work_lock);
|
||||||
|
@ -166,16 +167,17 @@ static int z2_batt_ps_init(struct z2_charger *charger, int props)
|
||||||
"Please consider setting proper battery "
|
"Please consider setting proper battery "
|
||||||
"name in platform definition file, falling "
|
"name in platform definition file, falling "
|
||||||
"back to name \" Z2_DEFAULT_NAME \"\n");
|
"back to name \" Z2_DEFAULT_NAME \"\n");
|
||||||
charger->batt_ps.name = Z2_DEFAULT_NAME;
|
charger->batt_ps_desc.name = Z2_DEFAULT_NAME;
|
||||||
} else
|
} else
|
||||||
charger->batt_ps.name = info->batt_name;
|
charger->batt_ps_desc.name = info->batt_name;
|
||||||
|
|
||||||
charger->batt_ps.properties = prop;
|
charger->batt_ps_desc.properties = prop;
|
||||||
charger->batt_ps.num_properties = props;
|
charger->batt_ps_desc.num_properties = props;
|
||||||
charger->batt_ps.type = POWER_SUPPLY_TYPE_BATTERY;
|
charger->batt_ps_desc.type = POWER_SUPPLY_TYPE_BATTERY;
|
||||||
charger->batt_ps.get_property = z2_batt_get_property;
|
charger->batt_ps_desc.get_property = z2_batt_get_property;
|
||||||
charger->batt_ps.external_power_changed = z2_batt_ext_power_changed;
|
charger->batt_ps_desc.external_power_changed =
|
||||||
charger->batt_ps.use_for_apm = 1;
|
z2_batt_ext_power_changed;
|
||||||
|
charger->batt_ps_desc.use_for_apm = 1;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -187,6 +189,7 @@ static int z2_batt_probe(struct i2c_client *client,
|
||||||
int props = 1; /* POWER_SUPPLY_PROP_PRESENT */
|
int props = 1; /* POWER_SUPPLY_PROP_PRESENT */
|
||||||
struct z2_charger *charger;
|
struct z2_charger *charger;
|
||||||
struct z2_battery_info *info = client->dev.platform_data;
|
struct z2_battery_info *info = client->dev.platform_data;
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
if (info == NULL) {
|
if (info == NULL) {
|
||||||
dev_err(&client->dev,
|
dev_err(&client->dev,
|
||||||
|
@ -203,6 +206,7 @@ static int z2_batt_probe(struct i2c_client *client,
|
||||||
charger->info = info;
|
charger->info = info;
|
||||||
charger->client = client;
|
charger->client = client;
|
||||||
i2c_set_clientdata(client, charger);
|
i2c_set_clientdata(client, charger);
|
||||||
|
psy_cfg.drv_data = charger;
|
||||||
|
|
||||||
mutex_init(&charger->work_lock);
|
mutex_init(&charger->work_lock);
|
||||||
|
|
||||||
|
@ -230,16 +234,20 @@ static int z2_batt_probe(struct i2c_client *client,
|
||||||
|
|
||||||
INIT_WORK(&charger->bat_work, z2_batt_work);
|
INIT_WORK(&charger->bat_work, z2_batt_work);
|
||||||
|
|
||||||
ret = power_supply_register(&client->dev, &charger->batt_ps);
|
charger->batt_ps = power_supply_register(&client->dev,
|
||||||
if (ret)
|
&charger->batt_ps_desc,
|
||||||
|
&psy_cfg);
|
||||||
|
if (IS_ERR(charger->batt_ps)) {
|
||||||
|
ret = PTR_ERR(charger->batt_ps);
|
||||||
goto err4;
|
goto err4;
|
||||||
|
}
|
||||||
|
|
||||||
schedule_work(&charger->bat_work);
|
schedule_work(&charger->bat_work);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err4:
|
err4:
|
||||||
kfree(charger->batt_ps.properties);
|
kfree(charger->batt_ps_desc.properties);
|
||||||
err3:
|
err3:
|
||||||
if (info->charge_gpio >= 0 && gpio_is_valid(info->charge_gpio))
|
if (info->charge_gpio >= 0 && gpio_is_valid(info->charge_gpio))
|
||||||
free_irq(gpio_to_irq(info->charge_gpio), charger);
|
free_irq(gpio_to_irq(info->charge_gpio), charger);
|
||||||
|
@ -257,9 +265,9 @@ static int z2_batt_remove(struct i2c_client *client)
|
||||||
struct z2_battery_info *info = charger->info;
|
struct z2_battery_info *info = charger->info;
|
||||||
|
|
||||||
cancel_work_sync(&charger->bat_work);
|
cancel_work_sync(&charger->bat_work);
|
||||||
power_supply_unregister(&charger->batt_ps);
|
power_supply_unregister(charger->batt_ps);
|
||||||
|
|
||||||
kfree(charger->batt_ps.properties);
|
kfree(charger->batt_ps_desc.properties);
|
||||||
if (info->charge_gpio >= 0 && gpio_is_valid(info->charge_gpio)) {
|
if (info->charge_gpio >= 0 && gpio_is_valid(info->charge_gpio)) {
|
||||||
free_irq(gpio_to_irq(info->charge_gpio), charger);
|
free_irq(gpio_to_irq(info->charge_gpio), charger);
|
||||||
gpio_free(info->charge_gpio);
|
gpio_free(info->charge_gpio);
|
||||||
|
|
|
@ -82,8 +82,8 @@ struct bat_response {
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply nvec_bat_psy;
|
static struct power_supply *nvec_bat_psy;
|
||||||
static struct power_supply nvec_psy;
|
static struct power_supply *nvec_psy;
|
||||||
|
|
||||||
static int nvec_power_notifier(struct notifier_block *nb,
|
static int nvec_power_notifier(struct notifier_block *nb,
|
||||||
unsigned long event_type, void *data)
|
unsigned long event_type, void *data)
|
||||||
|
@ -98,7 +98,7 @@ static int nvec_power_notifier(struct notifier_block *nb,
|
||||||
if (res->sub_type == 0) {
|
if (res->sub_type == 0) {
|
||||||
if (power->on != res->plu) {
|
if (power->on != res->plu) {
|
||||||
power->on = res->plu;
|
power->on = res->plu;
|
||||||
power_supply_changed(&nvec_psy);
|
power_supply_changed(nvec_psy);
|
||||||
}
|
}
|
||||||
return NOTIFY_STOP;
|
return NOTIFY_STOP;
|
||||||
}
|
}
|
||||||
|
@ -167,7 +167,7 @@ static int nvec_power_bat_notifier(struct notifier_block *nb,
|
||||||
}
|
}
|
||||||
power->bat_cap = res->plc[1];
|
power->bat_cap = res->plc[1];
|
||||||
if (status_changed)
|
if (status_changed)
|
||||||
power_supply_changed(&nvec_bat_psy);
|
power_supply_changed(nvec_bat_psy);
|
||||||
break;
|
break;
|
||||||
case VOLTAGE:
|
case VOLTAGE:
|
||||||
power->bat_voltage_now = res->plu * 1000;
|
power->bat_voltage_now = res->plu * 1000;
|
||||||
|
@ -225,7 +225,7 @@ static int nvec_power_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 nvec_power *power = dev_get_drvdata(psy->dev->parent);
|
struct nvec_power *power = dev_get_drvdata(psy->dev.parent);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_ONLINE:
|
case POWER_SUPPLY_PROP_ONLINE:
|
||||||
|
@ -241,7 +241,7 @@ static int nvec_battery_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 nvec_power *power = dev_get_drvdata(psy->dev->parent);
|
struct nvec_power *power = dev_get_drvdata(psy->dev.parent);
|
||||||
|
|
||||||
switch (psp) {
|
switch (psp) {
|
||||||
case POWER_SUPPLY_PROP_STATUS:
|
case POWER_SUPPLY_PROP_STATUS:
|
||||||
|
@ -323,7 +323,7 @@ static char *nvec_power_supplied_to[] = {
|
||||||
"battery",
|
"battery",
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply nvec_bat_psy = {
|
static const struct power_supply_desc nvec_bat_psy_desc = {
|
||||||
.name = "battery",
|
.name = "battery",
|
||||||
.type = POWER_SUPPLY_TYPE_BATTERY,
|
.type = POWER_SUPPLY_TYPE_BATTERY,
|
||||||
.properties = nvec_battery_props,
|
.properties = nvec_battery_props,
|
||||||
|
@ -331,11 +331,9 @@ static struct power_supply nvec_bat_psy = {
|
||||||
.get_property = nvec_battery_get_property,
|
.get_property = nvec_battery_get_property,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct power_supply nvec_psy = {
|
static const struct power_supply_desc nvec_psy_desc = {
|
||||||
.name = "ac",
|
.name = "ac",
|
||||||
.type = POWER_SUPPLY_TYPE_MAINS,
|
.type = POWER_SUPPLY_TYPE_MAINS,
|
||||||
.supplied_to = nvec_power_supplied_to,
|
|
||||||
.num_supplicants = ARRAY_SIZE(nvec_power_supplied_to),
|
|
||||||
.properties = nvec_power_props,
|
.properties = nvec_power_props,
|
||||||
.num_properties = ARRAY_SIZE(nvec_power_props),
|
.num_properties = ARRAY_SIZE(nvec_power_props),
|
||||||
.get_property = nvec_power_get_property,
|
.get_property = nvec_power_get_property,
|
||||||
|
@ -373,9 +371,11 @@ static void nvec_power_poll(struct work_struct *work)
|
||||||
|
|
||||||
static int nvec_power_probe(struct platform_device *pdev)
|
static int nvec_power_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct power_supply *psy;
|
struct power_supply **psy;
|
||||||
|
const struct power_supply_desc *psy_desc;
|
||||||
struct nvec_power *power;
|
struct nvec_power *power;
|
||||||
struct nvec_chip *nvec = dev_get_drvdata(pdev->dev.parent);
|
struct nvec_chip *nvec = dev_get_drvdata(pdev->dev.parent);
|
||||||
|
struct power_supply_config psy_cfg = {};
|
||||||
|
|
||||||
power = devm_kzalloc(&pdev->dev, sizeof(struct nvec_power), GFP_NOWAIT);
|
power = devm_kzalloc(&pdev->dev, sizeof(struct nvec_power), GFP_NOWAIT);
|
||||||
if (power == NULL)
|
if (power == NULL)
|
||||||
|
@ -387,6 +387,9 @@ static int nvec_power_probe(struct platform_device *pdev)
|
||||||
switch (pdev->id) {
|
switch (pdev->id) {
|
||||||
case AC:
|
case AC:
|
||||||
psy = &nvec_psy;
|
psy = &nvec_psy;
|
||||||
|
psy_desc = &nvec_psy_desc;
|
||||||
|
psy_cfg.supplied_to = nvec_power_supplied_to;
|
||||||
|
psy_cfg.num_supplicants = ARRAY_SIZE(nvec_power_supplied_to);
|
||||||
|
|
||||||
power->notifier.notifier_call = nvec_power_notifier;
|
power->notifier.notifier_call = nvec_power_notifier;
|
||||||
|
|
||||||
|
@ -395,6 +398,7 @@ static int nvec_power_probe(struct platform_device *pdev)
|
||||||
break;
|
break;
|
||||||
case BAT:
|
case BAT:
|
||||||
psy = &nvec_bat_psy;
|
psy = &nvec_bat_psy;
|
||||||
|
psy_desc = &nvec_bat_psy_desc;
|
||||||
|
|
||||||
power->notifier.notifier_call = nvec_power_bat_notifier;
|
power->notifier.notifier_call = nvec_power_bat_notifier;
|
||||||
break;
|
break;
|
||||||
|
@ -407,7 +411,9 @@ static int nvec_power_probe(struct platform_device *pdev)
|
||||||
if (pdev->id == BAT)
|
if (pdev->id == BAT)
|
||||||
get_bat_mfg_data(power);
|
get_bat_mfg_data(power);
|
||||||
|
|
||||||
return power_supply_register(&pdev->dev, psy);
|
*psy = power_supply_register(&pdev->dev, psy_desc, &psy_cfg);
|
||||||
|
|
||||||
|
return PTR_ERR_OR_ZERO(*psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nvec_power_remove(struct platform_device *pdev)
|
static int nvec_power_remove(struct platform_device *pdev)
|
||||||
|
@ -418,10 +424,10 @@ static int nvec_power_remove(struct platform_device *pdev)
|
||||||
nvec_unregister_notifier(power->nvec, &power->notifier);
|
nvec_unregister_notifier(power->nvec, &power->notifier);
|
||||||
switch (pdev->id) {
|
switch (pdev->id) {
|
||||||
case AC:
|
case AC:
|
||||||
power_supply_unregister(&nvec_psy);
|
power_supply_unregister(nvec_psy);
|
||||||
break;
|
break;
|
||||||
case BAT:
|
case BAT:
|
||||||
power_supply_unregister(&nvec_bat_psy);
|
power_supply_unregister(nvec_bat_psy);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -514,10 +514,10 @@ struct hid_device { /* device report descriptor */
|
||||||
#ifdef CONFIG_HID_BATTERY_STRENGTH
|
#ifdef CONFIG_HID_BATTERY_STRENGTH
|
||||||
/*
|
/*
|
||||||
* Power supply information for HID devices which report
|
* Power supply information for HID devices which report
|
||||||
* battery strength. power_supply is registered iff
|
* battery strength. power_supply was successfully registered if
|
||||||
* battery.name is non-NULL.
|
* battery is non-NULL.
|
||||||
*/
|
*/
|
||||||
struct power_supply battery;
|
struct power_supply *battery;
|
||||||
__s32 battery_min;
|
__s32 battery_min;
|
||||||
__s32 battery_max;
|
__s32 battery_max;
|
||||||
__s32 battery_report_type;
|
__s32 battery_report_type;
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue