Linux Audio

Check our new training course

Loading...
Note: File does not exist in v3.15.
  1/*
  2 * This program is free software; you can redistribute it and/or modify
  3 * it under the terms of the GNU General Public License version 2 as
  4 * published by the Free Software Foundation.
  5 *
  6 * h3xxx atmel micro companion support, battery subdevice
  7 * based on previous kernel 2.4 version
  8 * Author : Alessandro Gardich <gremlin@gremlin.it>
  9 * Author : Linus Walleij <linus.walleij@linaro.org>
 10 *
 11 */
 12
 13#include <linux/module.h>
 14#include <linux/init.h>
 15#include <linux/platform_device.h>
 16#include <linux/mfd/ipaq-micro.h>
 17#include <linux/power_supply.h>
 18#include <linux/workqueue.h>
 19
 20#define BATT_PERIOD 100000 /* 100 seconds in milliseconds */
 21
 22#define MICRO_BATT_CHEM_ALKALINE	0x01
 23#define MICRO_BATT_CHEM_NICD		0x02
 24#define MICRO_BATT_CHEM_NIMH		0x03
 25#define MICRO_BATT_CHEM_LION		0x04
 26#define MICRO_BATT_CHEM_LIPOLY		0x05
 27#define MICRO_BATT_CHEM_NOT_INSTALLED	0x06
 28#define MICRO_BATT_CHEM_UNKNOWN		0xff
 29
 30#define MICRO_BATT_STATUS_HIGH		0x01
 31#define MICRO_BATT_STATUS_LOW		0x02
 32#define MICRO_BATT_STATUS_CRITICAL	0x04
 33#define MICRO_BATT_STATUS_CHARGING	0x08
 34#define MICRO_BATT_STATUS_CHARGEMAIN	0x10
 35#define MICRO_BATT_STATUS_DEAD		0x20 /* Battery will not charge */
 36#define MICRO_BATT_STATUS_NOTINSTALLED	0x20 /* For expansion pack batteries */
 37#define MICRO_BATT_STATUS_FULL		0x40 /* Battery fully charged */
 38#define MICRO_BATT_STATUS_NOBATTERY	0x80
 39#define MICRO_BATT_STATUS_UNKNOWN	0xff
 40
 41struct micro_battery {
 42	struct ipaq_micro *micro;
 43	struct workqueue_struct *wq;
 44	struct delayed_work update;
 45	u8 ac;
 46	u8 chemistry;
 47	unsigned int voltage;
 48	u16 temperature;
 49	u8 flag;
 50};
 51
 52static void micro_battery_work(struct work_struct *work)
 53{
 54	struct micro_battery *mb = container_of(work,
 55				struct micro_battery, update.work);
 56	struct ipaq_micro_msg msg_battery = {
 57		.id = MSG_BATTERY,
 58	};
 59	struct ipaq_micro_msg msg_sensor = {
 60		.id = MSG_THERMAL_SENSOR,
 61	};
 62
 63	/* First send battery message */
 64	ipaq_micro_tx_msg_sync(mb->micro, &msg_battery);
 65	if (msg_battery.rx_len < 4)
 66		pr_info("ERROR");
 67
 68	/*
 69	 * Returned message format:
 70	 * byte 0:   0x00 = Not plugged in
 71	 *           0x01 = AC adapter plugged in
 72	 * byte 1:   chemistry
 73	 * byte 2:   voltage LSB
 74	 * byte 3:   voltage MSB
 75	 * byte 4:   flags
 76	 * byte 5-9: same for battery 2
 77	 */
 78	mb->ac = msg_battery.rx_data[0];
 79	mb->chemistry = msg_battery.rx_data[1];
 80	mb->voltage = ((((unsigned short)msg_battery.rx_data[3] << 8) +
 81			msg_battery.rx_data[2]) * 5000L) * 1000 / 1024;
 82	mb->flag = msg_battery.rx_data[4];
 83
 84	if (msg_battery.rx_len == 9)
 85		pr_debug("second battery ignored\n");
 86
 87	/* Then read the sensor */
 88	ipaq_micro_tx_msg_sync(mb->micro, &msg_sensor);
 89	mb->temperature = msg_sensor.rx_data[1] << 8 | msg_sensor.rx_data[0];
 90
 91	queue_delayed_work(mb->wq, &mb->update, msecs_to_jiffies(BATT_PERIOD));
 92}
 93
 94static int get_capacity(struct power_supply *b)
 95{
 96	struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
 97
 98	switch (mb->flag & 0x07) {
 99	case MICRO_BATT_STATUS_HIGH:
100		return 100;
101		break;
102	case MICRO_BATT_STATUS_LOW:
103		return 50;
104		break;
105	case MICRO_BATT_STATUS_CRITICAL:
106		return 5;
107		break;
108	default:
109		break;
110	}
111	return 0;
112}
113
114static int get_status(struct power_supply *b)
115{
116	struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
117
118	if (mb->flag == MICRO_BATT_STATUS_UNKNOWN)
119		return POWER_SUPPLY_STATUS_UNKNOWN;
120
121	if (mb->flag & MICRO_BATT_STATUS_FULL)
122		return POWER_SUPPLY_STATUS_FULL;
123
124	if ((mb->flag & MICRO_BATT_STATUS_CHARGING) ||
125		(mb->flag & MICRO_BATT_STATUS_CHARGEMAIN))
126		return POWER_SUPPLY_STATUS_CHARGING;
127
128	return POWER_SUPPLY_STATUS_DISCHARGING;
129}
130
131static int micro_batt_get_property(struct power_supply *b,
132					enum power_supply_property psp,
133					union power_supply_propval *val)
134{
135	struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
136
137	switch (psp) {
138	case POWER_SUPPLY_PROP_TECHNOLOGY:
139		switch (mb->chemistry) {
140		case MICRO_BATT_CHEM_NICD:
141			val->intval = POWER_SUPPLY_TECHNOLOGY_NiCd;
142			break;
143		case MICRO_BATT_CHEM_NIMH:
144			val->intval = POWER_SUPPLY_TECHNOLOGY_NiMH;
145			break;
146		case MICRO_BATT_CHEM_LION:
147			val->intval = POWER_SUPPLY_TECHNOLOGY_LION;
148			break;
149		case MICRO_BATT_CHEM_LIPOLY:
150			val->intval = POWER_SUPPLY_TECHNOLOGY_LIPO;
151			break;
152		default:
153			val->intval = POWER_SUPPLY_TECHNOLOGY_UNKNOWN;
154			break;
155		};
156		break;
157	case POWER_SUPPLY_PROP_STATUS:
158		val->intval = get_status(b);
159		break;
160	case POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN:
161		val->intval = 4700000;
162		break;
163	case POWER_SUPPLY_PROP_CAPACITY:
164		val->intval = get_capacity(b);
165		break;
166	case POWER_SUPPLY_PROP_TEMP:
167		val->intval = mb->temperature;
168		break;
169	case POWER_SUPPLY_PROP_VOLTAGE_NOW:
170		val->intval = mb->voltage;
171		break;
172	default:
173		return -EINVAL;
174	};
175
176	return 0;
177}
178
179static int micro_ac_get_property(struct power_supply *b,
180				 enum power_supply_property psp,
181				 union power_supply_propval *val)
182{
183	struct micro_battery *mb = dev_get_drvdata(b->dev.parent);
184
185	switch (psp) {
186	case POWER_SUPPLY_PROP_ONLINE:
187		val->intval = mb->ac;
188		break;
189	default:
190		return -EINVAL;
191	};
192
193	return 0;
194}
195
196static enum power_supply_property micro_batt_power_props[] = {
197	POWER_SUPPLY_PROP_TECHNOLOGY,
198	POWER_SUPPLY_PROP_STATUS,
199	POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN,
200	POWER_SUPPLY_PROP_CAPACITY,
201	POWER_SUPPLY_PROP_TEMP,
202	POWER_SUPPLY_PROP_VOLTAGE_NOW,
203};
204
205static const struct power_supply_desc micro_batt_power_desc = {
206	.name			= "main-battery",
207	.type			= POWER_SUPPLY_TYPE_BATTERY,
208	.properties		= micro_batt_power_props,
209	.num_properties		= ARRAY_SIZE(micro_batt_power_props),
210	.get_property		= micro_batt_get_property,
211	.use_for_apm		= 1,
212};
213
214static enum power_supply_property micro_ac_power_props[] = {
215	POWER_SUPPLY_PROP_ONLINE,
216};
217
218static const struct power_supply_desc micro_ac_power_desc = {
219	.name			= "ac",
220	.type			= POWER_SUPPLY_TYPE_MAINS,
221	.properties		= micro_ac_power_props,
222	.num_properties		= ARRAY_SIZE(micro_ac_power_props),
223	.get_property		= micro_ac_get_property,
224};
225
226static struct power_supply *micro_batt_power, *micro_ac_power;
227
228static int micro_batt_probe(struct platform_device *pdev)
229{
230	struct micro_battery *mb;
231	int ret;
232
233	mb = devm_kzalloc(&pdev->dev, sizeof(*mb), GFP_KERNEL);
234	if (!mb)
235		return -ENOMEM;
236
237	mb->micro = dev_get_drvdata(pdev->dev.parent);
238	mb->wq = create_singlethread_workqueue("ipaq-battery-wq");
239	if (!mb->wq)
240		return -ENOMEM;
241
242	INIT_DELAYED_WORK(&mb->update, micro_battery_work);
243	platform_set_drvdata(pdev, mb);
244	queue_delayed_work(mb->wq, &mb->update, 1);
245
246	micro_batt_power = power_supply_register(&pdev->dev,
247						 &micro_batt_power_desc, NULL);
248	if (IS_ERR(micro_batt_power)) {
249		ret = PTR_ERR(micro_batt_power);
250		goto batt_err;
251	}
252
253	micro_ac_power = power_supply_register(&pdev->dev,
254					       &micro_ac_power_desc, NULL);
255	if (IS_ERR(micro_ac_power)) {
256		ret = PTR_ERR(micro_ac_power);
257		goto ac_err;
258	}
259
260	dev_info(&pdev->dev, "iPAQ micro battery driver\n");
261	return 0;
262
263ac_err:
264	power_supply_unregister(micro_ac_power);
265batt_err:
266	cancel_delayed_work_sync(&mb->update);
267	destroy_workqueue(mb->wq);
268	return ret;
269}
270
271static int micro_batt_remove(struct platform_device *pdev)
272
273{
274	struct micro_battery *mb = platform_get_drvdata(pdev);
275
276	power_supply_unregister(micro_ac_power);
277	power_supply_unregister(micro_batt_power);
278	cancel_delayed_work_sync(&mb->update);
279	destroy_workqueue(mb->wq);
280
281	return 0;
282}
283
284static int __maybe_unused micro_batt_suspend(struct device *dev)
285{
286	struct micro_battery *mb = dev_get_drvdata(dev);
287
288	cancel_delayed_work_sync(&mb->update);
289	return 0;
290}
291
292static int __maybe_unused micro_batt_resume(struct device *dev)
293{
294	struct micro_battery *mb = dev_get_drvdata(dev);
295
296	queue_delayed_work(mb->wq, &mb->update, msecs_to_jiffies(BATT_PERIOD));
297	return 0;
298}
299
300static const struct dev_pm_ops micro_batt_dev_pm_ops = {
301	SET_SYSTEM_SLEEP_PM_OPS(micro_batt_suspend, micro_batt_resume)
302};
303
304static struct platform_driver micro_batt_device_driver = {
305	.driver		= {
306		.name	= "ipaq-micro-battery",
307		.pm	= &micro_batt_dev_pm_ops,
308	},
309	.probe		= micro_batt_probe,
310	.remove		= micro_batt_remove,
311};
312module_platform_driver(micro_batt_device_driver);
313
314MODULE_LICENSE("GPL");
315MODULE_DESCRIPTION("driver for iPAQ Atmel micro battery");
316MODULE_ALIAS("platform:battery-ipaq-micro");