xref: /openbmc/linux/drivers/power/supply/lego_ev3_battery.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
153db8858SDavid Lechner /*
253db8858SDavid Lechner  * Battery driver for LEGO MINDSTORMS EV3
353db8858SDavid Lechner  *
453db8858SDavid Lechner  * Copyright (C) 2017 David Lechner <david@lechnology.com>
553db8858SDavid Lechner  *
653db8858SDavid Lechner  * This program is free software; you can redistribute it and/or modify
753db8858SDavid Lechner  * it under the terms of the GNU General Public License version 2 as
853db8858SDavid Lechner  * published by the Free Software Foundation.
953db8858SDavid Lechner 
1053db8858SDavid Lechner  * This program is distributed "as is" WITHOUT ANY WARRANTY of any
1153db8858SDavid Lechner  * kind, whether express or implied; without even the implied warranty
1253db8858SDavid Lechner  * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
1353db8858SDavid Lechner  * GNU General Public License for more details.
1453db8858SDavid Lechner  */
1553db8858SDavid Lechner 
1653db8858SDavid Lechner #include <linux/delay.h>
1753db8858SDavid Lechner #include <linux/err.h>
1853db8858SDavid Lechner #include <linux/gpio/consumer.h>
1953db8858SDavid Lechner #include <linux/iio/consumer.h>
2053db8858SDavid Lechner #include <linux/iio/types.h>
2153db8858SDavid Lechner #include <linux/kernel.h>
2253db8858SDavid Lechner #include <linux/module.h>
23*2ce8284cSRob Herring #include <linux/mod_devicetable.h>
2453db8858SDavid Lechner #include <linux/platform_device.h>
2553db8858SDavid Lechner #include <linux/power_supply.h>
2653db8858SDavid Lechner 
2753db8858SDavid Lechner struct lego_ev3_battery {
2853db8858SDavid Lechner 	struct iio_channel *iio_v;
2953db8858SDavid Lechner 	struct iio_channel *iio_i;
3053db8858SDavid Lechner 	struct gpio_desc *rechargeable_gpio;
3153db8858SDavid Lechner 	struct power_supply *psy;
3253db8858SDavid Lechner 	int technology;
3353db8858SDavid Lechner 	int v_max;
3453db8858SDavid Lechner 	int v_min;
3553db8858SDavid Lechner };
3653db8858SDavid Lechner 
lego_ev3_battery_get_property(struct power_supply * psy,enum power_supply_property psp,union power_supply_propval * val)3753db8858SDavid Lechner static int lego_ev3_battery_get_property(struct power_supply *psy,
3853db8858SDavid Lechner 					 enum power_supply_property psp,
3953db8858SDavid Lechner 					 union power_supply_propval *val)
4053db8858SDavid Lechner {
4153db8858SDavid Lechner 	struct lego_ev3_battery *batt = power_supply_get_drvdata(psy);
426e92cecbSDavid Lechner 	int ret, val2;
4353db8858SDavid Lechner 
4453db8858SDavid Lechner 	switch (psp) {
4553db8858SDavid Lechner 	case POWER_SUPPLY_PROP_TECHNOLOGY:
4653db8858SDavid Lechner 		val->intval = batt->technology;
4753db8858SDavid Lechner 		break;
4853db8858SDavid Lechner 	case POWER_SUPPLY_PROP_VOLTAGE_NOW:
4953db8858SDavid Lechner 		/* battery voltage is iio channel * 2 + Vce of transistor */
506e92cecbSDavid Lechner 		ret = iio_read_channel_processed(batt->iio_v, &val->intval);
516e92cecbSDavid Lechner 		if (ret)
526e92cecbSDavid Lechner 			return ret;
536e92cecbSDavid Lechner 
5453db8858SDavid Lechner 		val->intval *= 2000;
559c727241SDavid Lechner 		val->intval += 50000;
566e92cecbSDavid Lechner 
5753db8858SDavid Lechner 		/* plus adjust for shunt resistor drop */
586e92cecbSDavid Lechner 		ret = iio_read_channel_processed(batt->iio_i, &val2);
596e92cecbSDavid Lechner 		if (ret)
606e92cecbSDavid Lechner 			return ret;
616e92cecbSDavid Lechner 
6253db8858SDavid Lechner 		val2 *= 1000;
6353db8858SDavid Lechner 		val2 /= 15;
6453db8858SDavid Lechner 		val->intval += val2;
6553db8858SDavid Lechner 		break;
6653db8858SDavid Lechner 	case POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN:
6753db8858SDavid Lechner 		val->intval = batt->v_max;
6853db8858SDavid Lechner 		break;
6953db8858SDavid Lechner 	case POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN:
7053db8858SDavid Lechner 		val->intval = batt->v_min;
7153db8858SDavid Lechner 		break;
7253db8858SDavid Lechner 	case POWER_SUPPLY_PROP_CURRENT_NOW:
7353db8858SDavid Lechner 		/* battery current is iio channel / 15 / 0.05 ohms */
746e92cecbSDavid Lechner 		ret = iio_read_channel_processed(batt->iio_i, &val->intval);
756e92cecbSDavid Lechner 		if (ret)
766e92cecbSDavid Lechner 			return ret;
776e92cecbSDavid Lechner 
7853db8858SDavid Lechner 		val->intval *= 20000;
7953db8858SDavid Lechner 		val->intval /= 15;
8053db8858SDavid Lechner 		break;
8153db8858SDavid Lechner 	case POWER_SUPPLY_PROP_SCOPE:
8253db8858SDavid Lechner 		val->intval = POWER_SUPPLY_SCOPE_SYSTEM;
8353db8858SDavid Lechner 		break;
8453db8858SDavid Lechner 	default:
8553db8858SDavid Lechner 		return -EINVAL;
8653db8858SDavid Lechner 	}
8753db8858SDavid Lechner 
8853db8858SDavid Lechner 	return 0;
8953db8858SDavid Lechner }
9053db8858SDavid Lechner 
lego_ev3_battery_set_property(struct power_supply * psy,enum power_supply_property psp,const union power_supply_propval * val)9153db8858SDavid Lechner static int lego_ev3_battery_set_property(struct power_supply *psy,
9253db8858SDavid Lechner 					 enum power_supply_property psp,
9353db8858SDavid Lechner 					 const union power_supply_propval *val)
9453db8858SDavid Lechner {
9553db8858SDavid Lechner 	struct lego_ev3_battery *batt = power_supply_get_drvdata(psy);
9653db8858SDavid Lechner 
9753db8858SDavid Lechner 	switch (psp) {
9853db8858SDavid Lechner 	case POWER_SUPPLY_PROP_TECHNOLOGY:
9953db8858SDavid Lechner 		/*
10053db8858SDavid Lechner 		 * Only allow changing technology from Unknown to NiMH. Li-ion
10153db8858SDavid Lechner 		 * batteries are automatically detected and should not be
10253db8858SDavid Lechner 		 * overridden. Rechargeable AA batteries, on the other hand,
10353db8858SDavid Lechner 		 * cannot be automatically detected, and so must be manually
10453db8858SDavid Lechner 		 * specified. This should only be set once during system init,
10553db8858SDavid Lechner 		 * so there is no mechanism to go back to Unknown.
10653db8858SDavid Lechner 		 */
10753db8858SDavid Lechner 		if (batt->technology != POWER_SUPPLY_TECHNOLOGY_UNKNOWN)
10853db8858SDavid Lechner 			return -EINVAL;
10953db8858SDavid Lechner 		switch (val->intval) {
11053db8858SDavid Lechner 		case POWER_SUPPLY_TECHNOLOGY_NiMH:
11153db8858SDavid Lechner 			batt->technology = POWER_SUPPLY_TECHNOLOGY_NiMH;
11253db8858SDavid Lechner 			batt->v_max = 7800000;
11353db8858SDavid Lechner 			batt->v_min = 5400000;
11453db8858SDavid Lechner 			break;
11553db8858SDavid Lechner 		default:
11653db8858SDavid Lechner 			return -EINVAL;
11753db8858SDavid Lechner 		}
11853db8858SDavid Lechner 		break;
11953db8858SDavid Lechner 	default:
12053db8858SDavid Lechner 		return -EINVAL;
12153db8858SDavid Lechner 	}
12253db8858SDavid Lechner 
12353db8858SDavid Lechner 	return 0;
12453db8858SDavid Lechner }
12553db8858SDavid Lechner 
lego_ev3_battery_property_is_writeable(struct power_supply * psy,enum power_supply_property psp)12653db8858SDavid Lechner static int lego_ev3_battery_property_is_writeable(struct power_supply *psy,
12753db8858SDavid Lechner 						  enum power_supply_property psp)
12853db8858SDavid Lechner {
12953db8858SDavid Lechner 	struct lego_ev3_battery *batt = power_supply_get_drvdata(psy);
13053db8858SDavid Lechner 
13153db8858SDavid Lechner 	return psp == POWER_SUPPLY_PROP_TECHNOLOGY &&
13253db8858SDavid Lechner 		batt->technology == POWER_SUPPLY_TECHNOLOGY_UNKNOWN;
13353db8858SDavid Lechner }
13453db8858SDavid Lechner 
13553db8858SDavid Lechner static enum power_supply_property lego_ev3_battery_props[] = {
13653db8858SDavid Lechner 	POWER_SUPPLY_PROP_TECHNOLOGY,
13753db8858SDavid Lechner 	POWER_SUPPLY_PROP_VOLTAGE_NOW,
13853db8858SDavid Lechner 	POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN,
13953db8858SDavid Lechner 	POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN,
14053db8858SDavid Lechner 	POWER_SUPPLY_PROP_CURRENT_NOW,
14153db8858SDavid Lechner 	POWER_SUPPLY_PROP_SCOPE,
14253db8858SDavid Lechner };
14353db8858SDavid Lechner 
14453db8858SDavid Lechner static const struct power_supply_desc lego_ev3_battery_desc = {
14553db8858SDavid Lechner 	.name			= "lego-ev3-battery",
14653db8858SDavid Lechner 	.type			= POWER_SUPPLY_TYPE_BATTERY,
14753db8858SDavid Lechner 	.properties		= lego_ev3_battery_props,
14853db8858SDavid Lechner 	.num_properties		= ARRAY_SIZE(lego_ev3_battery_props),
14953db8858SDavid Lechner 	.get_property		= lego_ev3_battery_get_property,
15053db8858SDavid Lechner 	.set_property		= lego_ev3_battery_set_property,
15153db8858SDavid Lechner 	.property_is_writeable	= lego_ev3_battery_property_is_writeable,
15253db8858SDavid Lechner };
15353db8858SDavid Lechner 
lego_ev3_battery_probe(struct platform_device * pdev)15453db8858SDavid Lechner static int lego_ev3_battery_probe(struct platform_device *pdev)
15553db8858SDavid Lechner {
15653db8858SDavid Lechner 	struct device *dev = &pdev->dev;
15753db8858SDavid Lechner 	struct lego_ev3_battery *batt;
15853db8858SDavid Lechner 	struct power_supply_config psy_cfg = {};
15953db8858SDavid Lechner 	int err;
16053db8858SDavid Lechner 
16153db8858SDavid Lechner 	batt = devm_kzalloc(dev, sizeof(*batt), GFP_KERNEL);
16253db8858SDavid Lechner 	if (!batt)
16353db8858SDavid Lechner 		return -ENOMEM;
16453db8858SDavid Lechner 
16553db8858SDavid Lechner 	platform_set_drvdata(pdev, batt);
16653db8858SDavid Lechner 
16753db8858SDavid Lechner 	batt->iio_v = devm_iio_channel_get(dev, "voltage");
16853db8858SDavid Lechner 	err = PTR_ERR_OR_ZERO(batt->iio_v);
169e03e3601SKrzysztof Kozlowski 	if (err)
170e03e3601SKrzysztof Kozlowski 		return dev_err_probe(dev, err,
171e03e3601SKrzysztof Kozlowski 				     "Failed to get voltage iio channel\n");
17253db8858SDavid Lechner 
17353db8858SDavid Lechner 	batt->iio_i = devm_iio_channel_get(dev, "current");
17453db8858SDavid Lechner 	err = PTR_ERR_OR_ZERO(batt->iio_i);
175e03e3601SKrzysztof Kozlowski 	if (err)
176e03e3601SKrzysztof Kozlowski 		return dev_err_probe(dev, err,
177e03e3601SKrzysztof Kozlowski 				     "Failed to get current iio channel\n");
17853db8858SDavid Lechner 
17953db8858SDavid Lechner 	batt->rechargeable_gpio = devm_gpiod_get(dev, "rechargeable", GPIOD_IN);
18053db8858SDavid Lechner 	err = PTR_ERR_OR_ZERO(batt->rechargeable_gpio);
181e03e3601SKrzysztof Kozlowski 	if (err)
182e03e3601SKrzysztof Kozlowski 		return dev_err_probe(dev, err,
183e03e3601SKrzysztof Kozlowski 				     "Failed to get rechargeable gpio\n");
18453db8858SDavid Lechner 
18553db8858SDavid Lechner 	/*
18653db8858SDavid Lechner 	 * The rechargeable battery indication switch cannot be changed without
18753db8858SDavid Lechner 	 * removing the battery, so we only need to read it once.
18853db8858SDavid Lechner 	 */
18953db8858SDavid Lechner 	if (gpiod_get_value(batt->rechargeable_gpio)) {
19053db8858SDavid Lechner 		/* 2-cell Li-ion, 7.4V nominal */
19153db8858SDavid Lechner 		batt->technology = POWER_SUPPLY_TECHNOLOGY_LION;
19253db8858SDavid Lechner 		batt->v_max = 84000000;
19353db8858SDavid Lechner 		batt->v_min = 60000000;
19453db8858SDavid Lechner 	} else {
19553db8858SDavid Lechner 		/* 6x AA Alkaline, 9V nominal */
19653db8858SDavid Lechner 		batt->technology = POWER_SUPPLY_TECHNOLOGY_UNKNOWN;
19753db8858SDavid Lechner 		batt->v_max = 90000000;
19853db8858SDavid Lechner 		batt->v_min = 48000000;
19953db8858SDavid Lechner 	}
20053db8858SDavid Lechner 
20153db8858SDavid Lechner 	psy_cfg.of_node = pdev->dev.of_node;
20253db8858SDavid Lechner 	psy_cfg.drv_data = batt;
20353db8858SDavid Lechner 
20453db8858SDavid Lechner 	batt->psy = devm_power_supply_register(dev, &lego_ev3_battery_desc,
20553db8858SDavid Lechner 					       &psy_cfg);
20653db8858SDavid Lechner 	err = PTR_ERR_OR_ZERO(batt->psy);
20753db8858SDavid Lechner 	if (err) {
20853db8858SDavid Lechner 		dev_err(dev, "failed to register power supply\n");
20953db8858SDavid Lechner 		return err;
21053db8858SDavid Lechner 	}
21153db8858SDavid Lechner 
21253db8858SDavid Lechner 	return 0;
21353db8858SDavid Lechner }
21453db8858SDavid Lechner 
21553db8858SDavid Lechner static const struct of_device_id of_lego_ev3_battery_match[] = {
21653db8858SDavid Lechner 	{ .compatible = "lego,ev3-battery", },
21753db8858SDavid Lechner 	{ }
21853db8858SDavid Lechner };
21953db8858SDavid Lechner MODULE_DEVICE_TABLE(of, of_lego_ev3_battery_match);
22053db8858SDavid Lechner 
22153db8858SDavid Lechner static struct platform_driver lego_ev3_battery_driver = {
22253db8858SDavid Lechner 	.driver	= {
22353db8858SDavid Lechner 		.name		= "lego-ev3-battery",
22453db8858SDavid Lechner 		.of_match_table = of_lego_ev3_battery_match,
22553db8858SDavid Lechner 	},
22653db8858SDavid Lechner 	.probe	= lego_ev3_battery_probe,
22753db8858SDavid Lechner };
22853db8858SDavid Lechner module_platform_driver(lego_ev3_battery_driver);
22953db8858SDavid Lechner 
23053db8858SDavid Lechner MODULE_LICENSE("GPL");
23153db8858SDavid Lechner MODULE_AUTHOR("David Lechner <david@lechnology.com>");
23253db8858SDavid Lechner MODULE_DESCRIPTION("LEGO MINDSTORMS EV3 Battery Driver");
233