Blame view

arch/arm/mach-pxa/sharpsl_pm.c 27 KB
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
  /*
   * Battery and Power Management code for the Sharp SL-C7xx and SL-Cxx00
   * series of PDAs
   *
   * Copyright (c) 2004-2005 Richard Purdie
   *
   * Based on code written by Sharp for 2.4 kernels
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
   *
   */
  
  #undef DEBUG
  
  #include <linux/module.h>
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
18
  #include <linux/kernel.h>
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
19
  #include <linux/interrupt.h>
c5e1ae972   Richard Purdie   [ARM] 3154/1: Sha...
20
  #include <linux/platform_device.h>
61fde514c   Russell King   [ARM] Fix more ap...
21
  #include <linux/apm-emulation.h>
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
22
  #include <linux/timer.h>
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
23
24
25
  #include <linux/delay.h>
  #include <linux/leds.h>
  #include <linux/suspend.h>
d5af27783   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
26
  #include <linux/gpio.h>
23019a733   Rob Herring   ARM: pxa: use com...
27
  #include <linux/io.h>
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
28

078abcf95   Richard Purdie   [ARM] 3096/1: Add...
29
  #include <asm/mach-types.h>
a09e64fbc   Russell King   [ARM] Move includ...
30
  #include <mach/pm.h>
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
31
  #include <mach/pxa2xx-regs.h>
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
32
  #include <mach/regs-rtc.h>
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
  #include <mach/sharpsl_pm.h>
  
  /*
   * Constants
   */
  #define SHARPSL_CHARGE_ON_TIME_INTERVAL        (msecs_to_jiffies(1*60*1000))  /* 1 min */
  #define SHARPSL_CHARGE_FINISH_TIME             (msecs_to_jiffies(10*60*1000)) /* 10 min */
  #define SHARPSL_BATCHK_TIME                    (msecs_to_jiffies(15*1000))    /* 15 sec */
  #define SHARPSL_BATCHK_TIME_SUSPEND            (60*10)                        /* 10 min */
  
  #define SHARPSL_WAIT_CO_TIME                   15  /* 15 sec */
  #define SHARPSL_WAIT_DISCHARGE_ON              100 /* 100 msec */
  #define SHARPSL_CHECK_BATTERY_WAIT_TIME_TEMP   10  /* 10 msec */
  #define SHARPSL_CHECK_BATTERY_WAIT_TIME_VOLT   10  /* 10 msec */
  #define SHARPSL_CHECK_BATTERY_WAIT_TIME_ACIN   10  /* 10 msec */
  #define SHARPSL_CHARGE_WAIT_TIME               15  /* 15 msec */
  #define SHARPSL_CHARGE_CO_CHECK_TIME           5   /* 5 msec */
  #define SHARPSL_CHARGE_RETRY_CNT               1   /* eqv. 10 min */
  
  /*
   * Prototypes
   */
  #ifdef CONFIG_PM
  static int sharpsl_off_charge_battery(void);
  static int sharpsl_check_battery_voltage(void);
  static int sharpsl_fatal_check(void);
  #endif
  static int sharpsl_check_battery_temp(void);
  static int sharpsl_ac_check(void);
  static int sharpsl_average_value(int ad);
  static void sharpsl_average_clear(void);
  static void sharpsl_charge_toggle(struct work_struct *private_);
  static void sharpsl_battery_thread(struct work_struct *private_);
  
  
  /*
   * Variables
   */
  struct sharpsl_pm_status sharpsl_pm;
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
72
73
  static DECLARE_DELAYED_WORK(toggle_charger, sharpsl_charge_toggle);
  static DECLARE_DELAYED_WORK(sharpsl_bat, sharpsl_battery_thread);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
74
  DEFINE_LED_TRIGGER(sharpsl_charge_led_trigger);
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
75

0ba01ebcb   Pavel Machek   [ARM] pxa/zaurus:...
76
  struct battery_thresh sharpsl_battery_levels_acin[] = {
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
  	{ 213, 100},
  	{ 212,  98},
  	{ 211,  95},
  	{ 210,  93},
  	{ 209,  90},
  	{ 208,  88},
  	{ 207,  85},
  	{ 206,  83},
  	{ 205,  80},
  	{ 204,  78},
  	{ 203,  75},
  	{ 202,  73},
  	{ 201,  70},
  	{ 200,  68},
  	{ 199,  65},
  	{ 198,  63},
  	{ 197,  60},
  	{ 196,  58},
  	{ 195,  55},
  	{ 194,  53},
  	{ 193,  50},
  	{ 192,  48},
  	{ 192,  45},
  	{ 191,  43},
  	{ 191,  40},
  	{ 190,  38},
  	{ 190,  35},
  	{ 189,  33},
  	{ 188,  30},
  	{ 187,  28},
  	{ 186,  25},
  	{ 185,  23},
  	{ 184,  20},
  	{ 183,  18},
  	{ 182,  15},
  	{ 181,  13},
  	{ 180,  10},
  	{ 179,   8},
  	{ 178,   5},
  	{   0,   0},
  };
0ba01ebcb   Pavel Machek   [ARM] pxa/zaurus:...
118
  struct battery_thresh sharpsl_battery_levels_noac[] = {
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
  	{ 213, 100},
  	{ 212,  98},
  	{ 211,  95},
  	{ 210,  93},
  	{ 209,  90},
  	{ 208,  88},
  	{ 207,  85},
  	{ 206,  83},
  	{ 205,  80},
  	{ 204,  78},
  	{ 203,  75},
  	{ 202,  73},
  	{ 201,  70},
  	{ 200,  68},
  	{ 199,  65},
  	{ 198,  63},
  	{ 197,  60},
  	{ 196,  58},
  	{ 195,  55},
  	{ 194,  53},
  	{ 193,  50},
  	{ 192,  48},
  	{ 191,  45},
  	{ 190,  43},
  	{ 189,  40},
  	{ 188,  38},
  	{ 187,  35},
  	{ 186,  33},
  	{ 185,  30},
  	{ 184,  28},
  	{ 183,  25},
  	{ 182,  23},
  	{ 181,  20},
  	{ 180,  18},
  	{ 179,  15},
  	{ 178,  13},
  	{ 177,  10},
  	{ 176,   8},
  	{ 175,   5},
  	{   0,   0},
  };
25af3b0f7   Eric Miao   Revert "[ARM] pxa...
160
  /* MAX1111 Commands */
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
161
162
163
164
  #define MAXCTRL_PD0      (1u << 0)
  #define MAXCTRL_PD1      (1u << 1)
  #define MAXCTRL_SGL      (1u << 2)
  #define MAXCTRL_UNI      (1u << 3)
25af3b0f7   Eric Miao   Revert "[ARM] pxa...
165
  #define MAXCTRL_SEL_SH   4
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
166
  #define MAXCTRL_STR      (1u << 7)
25af3b0f7   Eric Miao   Revert "[ARM] pxa...
167

438d7dc2b   Haojian Zhuang   ARM: pxa: fix mix...
168
  extern int max1111_read_channel(int);
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
169
170
171
  /*
   * Read MAX1111 ADC
   */
b7557de41   Richard Purdie   [ARM] 3228/1: Sha...
172
  int sharpsl_pm_pxa_read_max1111(int channel)
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
173
  {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
174
175
  	/* Ugly, better move this function into another module */
  	if (machine_is_tosa())
f8703dc8c   Richard Purdie   [ARM] 3564/1: sha...
176
  	    return 0;
f16177c20   Eric Miao   hwmon: add max111...
177
178
179
180
  	/* max1111 accepts channels from 0-3, however,
  	 * it is encoded from 0-7 here in the code.
  	 */
  	return max1111_read_channel(channel >> 1);
078abcf95   Richard Purdie   [ARM] 3096/1: Add...
181
  }
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
  static int get_percentage(int voltage)
  {
  	int i = sharpsl_pm.machinfo->bat_levels - 1;
  	int bl_status = sharpsl_pm.machinfo->backlight_get_status ? sharpsl_pm.machinfo->backlight_get_status() : 0;
  	struct battery_thresh *thresh;
  
  	if (sharpsl_pm.charge_mode == CHRG_ON)
  		thresh = bl_status ? sharpsl_pm.machinfo->bat_levels_acin_bl : sharpsl_pm.machinfo->bat_levels_acin;
  	else
  		thresh = bl_status ? sharpsl_pm.machinfo->bat_levels_noac_bl : sharpsl_pm.machinfo->bat_levels_noac;
  
  	while (i > 0 && (voltage > thresh[i].voltage))
  		i--;
  
  	return thresh[i].percentage;
  }
  
  static int get_apm_status(int voltage)
  {
  	int low_thresh, high_thresh;
  
  	if (sharpsl_pm.charge_mode == CHRG_ON) {
  		high_thresh = sharpsl_pm.machinfo->status_high_acin;
  		low_thresh = sharpsl_pm.machinfo->status_low_acin;
  	} else {
  		high_thresh = sharpsl_pm.machinfo->status_high_noac;
  		low_thresh = sharpsl_pm.machinfo->status_low_noac;
  	}
  
  	if (voltage >= high_thresh)
  		return APM_BATTERY_STATUS_HIGH;
  	if (voltage >= low_thresh)
  		return APM_BATTERY_STATUS_LOW;
  	return APM_BATTERY_STATUS_CRITICAL;
  }
  
  void sharpsl_battery_kick(void)
  {
  	schedule_delayed_work(&sharpsl_bat, msecs_to_jiffies(125));
  }
  EXPORT_SYMBOL(sharpsl_battery_kick);
  
  
  static void sharpsl_battery_thread(struct work_struct *private_)
  {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
227
  	int voltage, percent, apm_status, i;
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
228
229
230
231
232
233
234
235
236
237
  
  	if (!sharpsl_pm.machinfo)
  		return;
  
  	sharpsl_pm.battstat.ac_status = (sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_ACIN) ? APM_AC_ONLINE : APM_AC_OFFLINE);
  
  	/* Corgi cannot confirm when battery fully charged so periodically kick! */
  	if (!sharpsl_pm.machinfo->batfull_irq && (sharpsl_pm.charge_mode == CHRG_ON)
  			&& time_after(jiffies, sharpsl_pm.charge_start_time +  SHARPSL_CHARGE_ON_TIME_INTERVAL))
  		schedule_delayed_work(&toggle_charger, 0);
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
238
  	for (i = 0; i < 5; i++) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
239
  		voltage = sharpsl_pm.machinfo->read_devdata(SHARPSL_BATT_VOLT);
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
240
  		if (voltage > 0)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
241
  			break;
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
242
243
244
245
246
  	}
  	if (voltage <= 0) {
  		voltage = sharpsl_pm.machinfo->bat_levels_noac[0].voltage;
  		dev_warn(sharpsl_pm.dev, "Warning: Cannot read main battery!
  ");
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
247
248
249
250
251
252
253
  	}
  
  	voltage = sharpsl_average_value(voltage);
  	apm_status = get_apm_status(voltage);
  	percent = get_percentage(voltage);
  
  	/* At low battery voltages, the voltage has a tendency to start
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
254
255
256
257
  	   creeping back up so we try to avoid this here */
  	if ((sharpsl_pm.battstat.ac_status == APM_AC_ONLINE)
  	    || (apm_status == APM_BATTERY_STATUS_HIGH)
  	    || percent <= sharpsl_pm.battstat.mainbat_percent) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
258
259
260
261
262
263
264
265
  		sharpsl_pm.battstat.mainbat_voltage = voltage;
  		sharpsl_pm.battstat.mainbat_status = apm_status;
  		sharpsl_pm.battstat.mainbat_percent = percent;
  	}
  
  	dev_dbg(sharpsl_pm.dev, "Battery: voltage: %d, status: %d, percentage: %d, time: %ld
  ", voltage,
  			sharpsl_pm.battstat.mainbat_status, sharpsl_pm.battstat.mainbat_percent, jiffies);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
266
267
  	/* Suspend if critical battery level */
  	if ((sharpsl_pm.battstat.ac_status != APM_AC_ONLINE)
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
268
269
  	     && (sharpsl_pm.battstat.mainbat_status == APM_BATTERY_STATUS_CRITICAL)
  	     && !(sharpsl_pm.flags & SHARPSL_APM_QUEUED)) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
  		sharpsl_pm.flags |= SHARPSL_APM_QUEUED;
  		dev_err(sharpsl_pm.dev, "Fatal Off
  ");
  		apm_queue_event(APM_CRITICAL_SUSPEND);
  	}
  
  	schedule_delayed_work(&sharpsl_bat, SHARPSL_BATCHK_TIME);
  }
  
  void sharpsl_pm_led(int val)
  {
  	if (val == SHARPSL_LED_ERROR) {
  		dev_err(sharpsl_pm.dev, "Charging Error!
  ");
  	} else if (val == SHARPSL_LED_ON) {
  		dev_dbg(sharpsl_pm.dev, "Charge LED On
  ");
  		led_trigger_event(sharpsl_charge_led_trigger, LED_FULL);
  	} else {
  		dev_dbg(sharpsl_pm.dev, "Charge LED Off
  ");
  		led_trigger_event(sharpsl_charge_led_trigger, LED_OFF);
  	}
  }
  
  static void sharpsl_charge_on(void)
  {
  	dev_dbg(sharpsl_pm.dev, "Turning Charger On
  ");
  
  	sharpsl_pm.full_count = 0;
  	sharpsl_pm.charge_mode = CHRG_ON;
  	schedule_delayed_work(&toggle_charger, msecs_to_jiffies(250));
  	schedule_delayed_work(&sharpsl_bat, msecs_to_jiffies(500));
  }
  
  static void sharpsl_charge_off(void)
  {
  	dev_dbg(sharpsl_pm.dev, "Turning Charger Off
  ");
  
  	sharpsl_pm.machinfo->charge(0);
  	sharpsl_pm_led(SHARPSL_LED_OFF);
  	sharpsl_pm.charge_mode = CHRG_OFF;
  
  	schedule_delayed_work(&sharpsl_bat, 0);
  }
  
  static void sharpsl_charge_error(void)
  {
  	sharpsl_pm_led(SHARPSL_LED_ERROR);
  	sharpsl_pm.machinfo->charge(0);
  	sharpsl_pm.charge_mode = CHRG_ERROR;
  }
  
  static void sharpsl_charge_toggle(struct work_struct *private_)
  {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
327
328
  	dev_dbg(sharpsl_pm.dev, "Toggling Charger at time: %lx
  ", jiffies);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
  
  	if (!sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_ACIN)) {
  		sharpsl_charge_off();
  		return;
  	} else if ((sharpsl_check_battery_temp() < 0) || (sharpsl_ac_check() < 0)) {
  		sharpsl_charge_error();
  		return;
  	}
  
  	sharpsl_pm_led(SHARPSL_LED_ON);
  	sharpsl_pm.machinfo->charge(0);
  	mdelay(SHARPSL_CHARGE_WAIT_TIME);
  	sharpsl_pm.machinfo->charge(1);
  
  	sharpsl_pm.charge_start_time = jiffies;
  }
  
  static void sharpsl_ac_timer(unsigned long data)
  {
  	int acin = sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_ACIN);
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
349
350
  	dev_dbg(sharpsl_pm.dev, "AC Status: %d
  ", acin);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
351
352
353
354
355
356
357
358
359
  
  	sharpsl_average_clear();
  	if (acin && (sharpsl_pm.charge_mode != CHRG_ON))
  		sharpsl_charge_on();
  	else if (sharpsl_pm.charge_mode == CHRG_ON)
  		sharpsl_charge_off();
  
  	schedule_delayed_work(&sharpsl_bat, 0);
  }
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
360
  static irqreturn_t sharpsl_ac_isr(int irq, void *dev_id)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
  {
  	/* Delay the event slightly to debounce */
  	/* Must be a smaller delay than the chrg_full_isr below */
  	mod_timer(&sharpsl_pm.ac_timer, jiffies + msecs_to_jiffies(250));
  
  	return IRQ_HANDLED;
  }
  
  static void sharpsl_chrg_full_timer(unsigned long data)
  {
  	dev_dbg(sharpsl_pm.dev, "Charge Full at time: %lx
  ", jiffies);
  
  	sharpsl_pm.full_count++;
  
  	if (!sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_ACIN)) {
  		dev_dbg(sharpsl_pm.dev, "Charge Full: AC removed - stop charging!
  ");
  		if (sharpsl_pm.charge_mode == CHRG_ON)
  			sharpsl_charge_off();
  	} else if (sharpsl_pm.full_count < 2) {
  		dev_dbg(sharpsl_pm.dev, "Charge Full: Count too low
  ");
  		schedule_delayed_work(&toggle_charger, 0);
  	} else if (time_after(jiffies, sharpsl_pm.charge_start_time + SHARPSL_CHARGE_FINISH_TIME)) {
  		dev_dbg(sharpsl_pm.dev, "Charge Full: Interrupt generated too slowly - retry.
  ");
  		schedule_delayed_work(&toggle_charger, 0);
  	} else {
  		sharpsl_charge_off();
  		sharpsl_pm.charge_mode = CHRG_DONE;
  		dev_dbg(sharpsl_pm.dev, "Charge Full: Charging Finished
  ");
  	}
  }
  
  /* Charging Finished Interrupt (Not present on Corgi) */
  /* Can trigger at the same time as an AC status change so
     delay until after that has been processed */
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
400
  static irqreturn_t sharpsl_chrg_full_isr(int irq, void *dev_id)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
401
402
403
404
405
406
407
408
409
  {
  	if (sharpsl_pm.flags & SHARPSL_SUSPENDED)
  		return IRQ_HANDLED;
  
  	/* delay until after any ac interrupt */
  	mod_timer(&sharpsl_pm.chrg_full_timer, jiffies + msecs_to_jiffies(500));
  
  	return IRQ_HANDLED;
  }
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
410
  static irqreturn_t sharpsl_fatal_isr(int irq, void *dev_id)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
  {
  	int is_fatal = 0;
  
  	if (!sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_LOCK)) {
  		dev_err(sharpsl_pm.dev, "Battery now Unlocked! Suspending.
  ");
  		is_fatal = 1;
  	}
  
  	if (!sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_FATAL)) {
  		dev_err(sharpsl_pm.dev, "Fatal Batt Error! Suspending.
  ");
  		is_fatal = 1;
  	}
  
  	if (!(sharpsl_pm.flags & SHARPSL_APM_QUEUED) && is_fatal) {
  		sharpsl_pm.flags |= SHARPSL_APM_QUEUED;
  		apm_queue_event(APM_CRITICAL_SUSPEND);
  	}
  
  	return IRQ_HANDLED;
  }
  
  /*
   * Maintain an average of the last 10 readings
   */
  #define SHARPSL_CNV_VALUE_NUM    10
  static int sharpsl_ad_index;
  
  static void sharpsl_average_clear(void)
  {
  	sharpsl_ad_index = 0;
  }
  
  static int sharpsl_average_value(int ad)
  {
  	int i, ad_val = 0;
  	static int sharpsl_ad[SHARPSL_CNV_VALUE_NUM+1];
  
  	if (sharpsl_pm.battstat.mainbat_status != APM_BATTERY_STATUS_HIGH) {
  		sharpsl_ad_index = 0;
  		return ad;
  	}
  
  	sharpsl_ad[sharpsl_ad_index] = ad;
  	sharpsl_ad_index++;
  	if (sharpsl_ad_index >= SHARPSL_CNV_VALUE_NUM) {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
458
  		for (i = 0; i < (SHARPSL_CNV_VALUE_NUM-1); i++)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
459
460
461
  			sharpsl_ad[i] = sharpsl_ad[i+1];
  		sharpsl_ad_index = SHARPSL_CNV_VALUE_NUM - 1;
  	}
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
462
  	for (i = 0; i < sharpsl_ad_index; i++)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
463
  		ad_val += sharpsl_ad[i];
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
464
  	return ad_val / sharpsl_ad_index;
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
465
466
467
468
469
470
471
472
473
474
475
476
  }
  
  /*
   * Take an array of 5 integers, remove the maximum and minimum values
   * and return the average.
   */
  static int get_select_val(int *val)
  {
  	int i, j, k, temp, sum = 0;
  
  	/* Find MAX val */
  	temp = val[0];
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
477
478
  	j = 0;
  	for (i = 1; i < 5; i++) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
479
480
481
482
483
484
485
486
  		if (temp < val[i]) {
  			temp = val[i];
  			j = i;
  		}
  	}
  
  	/* Find MIN val */
  	temp = val[4];
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
487
488
  	k = 4;
  	for (i = 3; i >= 0; i--) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
489
490
491
492
493
  		if (temp > val[i]) {
  			temp = val[i];
  			k = i;
  		}
  	}
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
494
495
  	for (i = 0; i < 5; i++)
  		if (i != j && i != k)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
496
497
498
499
  			sum += val[i];
  
  	dev_dbg(sharpsl_pm.dev, "Average: %d from values: %d, %d, %d, %d, %d
  ", sum/3, val[0], val[1], val[2], val[3], val[4]);
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
500
  	return sum/3;
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
501
502
503
504
505
506
507
  }
  
  static int sharpsl_check_battery_temp(void)
  {
  	int val, i, buff[5];
  
  	/* Check battery temperature */
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
508
  	for (i = 0; i < 5; i++) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
  		mdelay(SHARPSL_CHECK_BATTERY_WAIT_TIME_TEMP);
  		sharpsl_pm.machinfo->measure_temp(1);
  		mdelay(SHARPSL_CHECK_BATTERY_WAIT_TIME_TEMP);
  		buff[i] = sharpsl_pm.machinfo->read_devdata(SHARPSL_BATT_TEMP);
  		sharpsl_pm.machinfo->measure_temp(0);
  	}
  
  	val = get_select_val(buff);
  
  	dev_dbg(sharpsl_pm.dev, "Temperature: %d
  ", val);
  	if (val > sharpsl_pm.machinfo->charge_on_temp) {
  		printk(KERN_WARNING "Not charging: temperature out of limits.
  ");
  		return -1;
  	}
  
  	return 0;
  }
  
  #ifdef CONFIG_PM
  static int sharpsl_check_battery_voltage(void)
  {
  	int val, i, buff[5];
  
  	/* disable charge, enable discharge */
  	sharpsl_pm.machinfo->charge(0);
  	sharpsl_pm.machinfo->discharge(1);
  	mdelay(SHARPSL_WAIT_DISCHARGE_ON);
  
  	if (sharpsl_pm.machinfo->discharge1)
  		sharpsl_pm.machinfo->discharge1(1);
  
  	/* Check battery voltage */
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
543
  	for (i = 0; i < 5; i++) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
  		buff[i] = sharpsl_pm.machinfo->read_devdata(SHARPSL_BATT_VOLT);
  		mdelay(SHARPSL_CHECK_BATTERY_WAIT_TIME_VOLT);
  	}
  
  	if (sharpsl_pm.machinfo->discharge1)
  		sharpsl_pm.machinfo->discharge1(0);
  
  	sharpsl_pm.machinfo->discharge(0);
  
  	val = get_select_val(buff);
  	dev_dbg(sharpsl_pm.dev, "Battery Voltage: %d
  ", val);
  
  	if (val < sharpsl_pm.machinfo->charge_on_volt)
  		return -1;
  
  	return 0;
  }
  #endif
  
  static int sharpsl_ac_check(void)
  {
  	int temp, i, buff[5];
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
567
  	for (i = 0; i < 5; i++) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
568
569
570
571
572
  		buff[i] = sharpsl_pm.machinfo->read_devdata(SHARPSL_ACIN_VOLT);
  		mdelay(SHARPSL_CHECK_BATTERY_WAIT_TIME_ACIN);
  	}
  
  	temp = get_select_val(buff);
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
573
574
  	dev_dbg(sharpsl_pm.dev, "AC Voltage: %d
  ", temp);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
575
576
  
  	if ((temp > sharpsl_pm.machinfo->charge_acin_high) || (temp < sharpsl_pm.machinfo->charge_acin_low)) {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
577
578
  		dev_err(sharpsl_pm.dev, "Error: AC check failed: voltage %d.
  ", temp);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
579
580
581
582
583
584
585
586
587
588
  		return -1;
  	}
  
  	return 0;
  }
  
  #ifdef CONFIG_PM
  static int sharpsl_pm_suspend(struct platform_device *pdev, pm_message_t state)
  {
  	sharpsl_pm.flags |= SHARPSL_SUSPENDED;
fe413ec32   Tejun Heo   arm/sharpsl: don'...
589
590
  	flush_delayed_work_sync(&toggle_charger);
  	flush_delayed_work_sync(&sharpsl_bat);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
  
  	if (sharpsl_pm.charge_mode == CHRG_ON)
  		sharpsl_pm.flags |= SHARPSL_DO_OFFLINE_CHRG;
  	else
  		sharpsl_pm.flags &= ~SHARPSL_DO_OFFLINE_CHRG;
  
  	return 0;
  }
  
  static int sharpsl_pm_resume(struct platform_device *pdev)
  {
  	/* Clear the reset source indicators as they break the bootloader upon reboot */
  	RCSR = 0x0f;
  	sharpsl_average_clear();
  	sharpsl_pm.flags &= ~SHARPSL_APM_QUEUED;
  	sharpsl_pm.flags &= ~SHARPSL_SUSPENDED;
  
  	return 0;
  }
  
  static void corgi_goto_sleep(unsigned long alarm_time, unsigned int alarm_enable, suspend_state_t state)
  {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
613
614
  	dev_dbg(sharpsl_pm.dev, "Time is: %08x
  ", RCNR);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
615

b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
616
617
  	dev_dbg(sharpsl_pm.dev, "Offline Charge Activate = %d
  ", sharpsl_pm.flags & SHARPSL_DO_OFFLINE_CHRG);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
  	/* not charging and AC-IN! */
  
  	if ((sharpsl_pm.flags & SHARPSL_DO_OFFLINE_CHRG) && (sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_ACIN))) {
  		dev_dbg(sharpsl_pm.dev, "Activating Offline Charger...
  ");
  		sharpsl_pm.charge_mode = CHRG_OFF;
  		sharpsl_pm.flags &= ~SHARPSL_DO_OFFLINE_CHRG;
  		sharpsl_off_charge_battery();
  	}
  
  	sharpsl_pm.machinfo->presuspend();
  
  	PEDR = 0xffffffff; /* clear it */
  
  	sharpsl_pm.flags &= ~SHARPSL_ALARM_ACTIVE;
  	if ((sharpsl_pm.charge_mode == CHRG_ON) && ((alarm_enable && ((alarm_time - RCNR) > (SHARPSL_BATCHK_TIME_SUSPEND + 30))) || !alarm_enable)) {
  		RTSR &= RTSR_ALE;
  		RTAR = RCNR + SHARPSL_BATCHK_TIME_SUSPEND;
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
636
637
  		dev_dbg(sharpsl_pm.dev, "Charging alarm at: %08x
  ", RTAR);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
638
639
640
641
  		sharpsl_pm.flags |= SHARPSL_ALARM_ACTIVE;
  	} else if (alarm_enable) {
  		RTSR &= RTSR_ALE;
  		RTAR = alarm_time;
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
642
643
  		dev_dbg(sharpsl_pm.dev, "User alarm at: %08x
  ", RTAR);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
644
645
646
647
648
649
650
651
  	} else {
  		dev_dbg(sharpsl_pm.dev, "No alarms set.
  ");
  	}
  
  	pxa_pm_enter(state);
  
  	sharpsl_pm.machinfo->postsuspend();
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
652
653
  	dev_dbg(sharpsl_pm.dev, "Corgi woken up from suspend: %08x
  ", PEDR);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
654
655
656
657
  }
  
  static int corgi_enter_suspend(unsigned long alarm_time, unsigned int alarm_enable, suspend_state_t state)
  {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
658
  	if (!sharpsl_pm.machinfo->should_wakeup(!(sharpsl_pm.flags & SHARPSL_ALARM_ACTIVE) && alarm_enable)) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
659
660
661
662
663
664
  		if (!(sharpsl_pm.flags & SHARPSL_ALARM_ACTIVE)) {
  			dev_dbg(sharpsl_pm.dev, "No user triggered wakeup events and not charging. Strange. Suspend.
  ");
  			corgi_goto_sleep(alarm_time, alarm_enable, state);
  			return 1;
  		}
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
665
  		if (sharpsl_off_charge_battery()) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
666
667
668
669
670
671
672
673
  			dev_dbg(sharpsl_pm.dev, "Charging. Suspend...
  ");
  			corgi_goto_sleep(alarm_time, alarm_enable, state);
  			return 1;
  		}
  		dev_dbg(sharpsl_pm.dev, "User triggered wakeup in offline charger.
  ");
  	}
99f329a2b   Pavel Machek   [ARM] pxa/sharpsl...
674
675
  	if ((!sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_LOCK)) ||
  	    (!sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_FATAL)))	{
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
  		dev_err(sharpsl_pm.dev, "Fatal condition. Suspend.
  ");
  		corgi_goto_sleep(alarm_time, alarm_enable, state);
  		return 1;
  	}
  
  	return 0;
  }
  
  static int corgi_pxa_pm_enter(suspend_state_t state)
  {
  	unsigned long alarm_time = RTAR;
  	unsigned int alarm_status = ((RTSR & RTSR_ALE) != 0);
  
  	dev_dbg(sharpsl_pm.dev, "SharpSL suspending for first time.
  ");
  
  	corgi_goto_sleep(alarm_time, alarm_status, state);
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
694
  	while (corgi_enter_suspend(alarm_time, alarm_status, state))
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
  		{}
  
  	if (sharpsl_pm.machinfo->earlyresume)
  		sharpsl_pm.machinfo->earlyresume();
  
  	dev_dbg(sharpsl_pm.dev, "SharpSL resuming...
  ");
  
  	return 0;
  }
  
  /*
   * Check for fatal battery errors
   * Fatal returns -1
   */
  static int sharpsl_fatal_check(void)
  {
  	int buff[5], temp, i, acin;
  
  	dev_dbg(sharpsl_pm.dev, "sharpsl_fatal_check entered
  ");
  
  	/* Check AC-Adapter */
  	acin = sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_ACIN);
  
  	if (acin && (sharpsl_pm.charge_mode == CHRG_ON)) {
  		sharpsl_pm.machinfo->charge(0);
  		udelay(100);
  		sharpsl_pm.machinfo->discharge(1);	/* enable discharge */
  		mdelay(SHARPSL_WAIT_DISCHARGE_ON);
  	}
  
  	if (sharpsl_pm.machinfo->discharge1)
  		sharpsl_pm.machinfo->discharge1(1);
  
  	/* Check battery : check inserting battery ? */
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
731
  	for (i = 0; i < 5; i++) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
  		buff[i] = sharpsl_pm.machinfo->read_devdata(SHARPSL_BATT_VOLT);
  		mdelay(SHARPSL_CHECK_BATTERY_WAIT_TIME_VOLT);
  	}
  
  	if (sharpsl_pm.machinfo->discharge1)
  		sharpsl_pm.machinfo->discharge1(0);
  
  	if (acin && (sharpsl_pm.charge_mode == CHRG_ON)) {
  		udelay(100);
  		sharpsl_pm.machinfo->charge(1);
  		sharpsl_pm.machinfo->discharge(0);
  	}
  
  	temp = get_select_val(buff);
  	dev_dbg(sharpsl_pm.dev, "sharpsl_fatal_check: acin: %d, discharge voltage: %d, no discharge: %ld
  ", acin, temp, sharpsl_pm.machinfo->read_devdata(SHARPSL_BATT_VOLT));
  
  	if ((acin && (temp < sharpsl_pm.machinfo->fatal_acin_volt)) ||
  			(!acin && (temp < sharpsl_pm.machinfo->fatal_noacin_volt)))
  		return -1;
  	return 0;
  }
  
  static int sharpsl_off_charge_error(void)
  {
  	dev_err(sharpsl_pm.dev, "Offline Charger: Error occurred.
  ");
  	sharpsl_pm.machinfo->charge(0);
  	sharpsl_pm_led(SHARPSL_LED_ERROR);
  	sharpsl_pm.charge_mode = CHRG_ERROR;
  	return 1;
  }
  
  /*
   * Charging Control while suspended
   * Return 1 - go straight to sleep
   * Return 0 - sleep or wakeup depending on other factors
   */
  static int sharpsl_off_charge_battery(void)
  {
  	int time;
  
  	dev_dbg(sharpsl_pm.dev, "Charge Mode: %d
  ", sharpsl_pm.charge_mode);
  
  	if (sharpsl_pm.charge_mode == CHRG_OFF) {
  		dev_dbg(sharpsl_pm.dev, "Offline Charger: Step 1
  ");
  
  		/* AC Check */
  		if ((sharpsl_ac_check() < 0) || (sharpsl_check_battery_temp() < 0))
  			return sharpsl_off_charge_error();
  
  		/* Start Charging */
  		sharpsl_pm_led(SHARPSL_LED_ON);
  		sharpsl_pm.machinfo->charge(0);
  		mdelay(SHARPSL_CHARGE_WAIT_TIME);
  		sharpsl_pm.machinfo->charge(1);
  
  		sharpsl_pm.charge_mode = CHRG_ON;
  		sharpsl_pm.full_count = 0;
  
  		return 1;
  	} else if (sharpsl_pm.charge_mode != CHRG_ON) {
  		return 1;
  	}
  
  	if (sharpsl_pm.full_count == 0) {
  		int time;
  
  		dev_dbg(sharpsl_pm.dev, "Offline Charger: Step 2
  ");
  
  		if ((sharpsl_check_battery_temp() < 0) || (sharpsl_check_battery_voltage() < 0))
  			return sharpsl_off_charge_error();
  
  		sharpsl_pm.machinfo->charge(0);
  		mdelay(SHARPSL_CHARGE_WAIT_TIME);
  		sharpsl_pm.machinfo->charge(1);
  		sharpsl_pm.charge_mode = CHRG_ON;
  
  		mdelay(SHARPSL_CHARGE_CO_CHECK_TIME);
  
  		time = RCNR;
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
816
  		while (1) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
  			/* Check if any wakeup event had occurred */
  			if (sharpsl_pm.machinfo->charger_wakeup() != 0)
  				return 0;
  			/* Check for timeout */
  			if ((RCNR - time) > SHARPSL_WAIT_CO_TIME)
  				return 1;
  			if (sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_CHRGFULL)) {
  				dev_dbg(sharpsl_pm.dev, "Offline Charger: Charge full occurred. Retrying to check
  ");
  				sharpsl_pm.full_count++;
  				sharpsl_pm.machinfo->charge(0);
  				mdelay(SHARPSL_CHARGE_WAIT_TIME);
  				sharpsl_pm.machinfo->charge(1);
  				return 1;
  			}
  		}
  	}
  
  	dev_dbg(sharpsl_pm.dev, "Offline Charger: Step 3
  ");
  
  	mdelay(SHARPSL_CHARGE_CO_CHECK_TIME);
  
  	time = RCNR;
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
841
  	while (1) {
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
842
  		/* Check if any wakeup event had occurred */
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
843
  		if (sharpsl_pm.machinfo->charger_wakeup())
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
  			return 0;
  		/* Check for timeout */
  		if ((RCNR-time) > SHARPSL_WAIT_CO_TIME) {
  			if (sharpsl_pm.full_count > SHARPSL_CHARGE_RETRY_CNT) {
  				dev_dbg(sharpsl_pm.dev, "Offline Charger: Not charged sufficiently. Retrying.
  ");
  				sharpsl_pm.full_count = 0;
  			}
  			sharpsl_pm.full_count++;
  			return 1;
  		}
  		if (sharpsl_pm.machinfo->read_devdata(SHARPSL_STATUS_CHRGFULL)) {
  			dev_dbg(sharpsl_pm.dev, "Offline Charger: Charging complete.
  ");
  			sharpsl_pm_led(SHARPSL_LED_OFF);
  			sharpsl_pm.machinfo->charge(0);
  			sharpsl_pm.charge_mode = CHRG_DONE;
  			return 1;
  		}
  	}
  }
  #else
  #define sharpsl_pm_suspend	NULL
  #define sharpsl_pm_resume	NULL
  #endif
  
  static ssize_t battery_percentage_show(struct device *dev, struct device_attribute *attr, char *buf)
  {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
872
873
  	return sprintf(buf, "%d
  ", sharpsl_pm.battstat.mainbat_percent);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
874
875
876
877
  }
  
  static ssize_t battery_voltage_show(struct device *dev, struct device_attribute *attr, char *buf)
  {
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
878
879
  	return sprintf(buf, "%d
  ", sharpsl_pm.battstat.mainbat_voltage);
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
  }
  
  static DEVICE_ATTR(battery_percentage, 0444, battery_percentage_show, NULL);
  static DEVICE_ATTR(battery_voltage, 0444, battery_voltage_show, NULL);
  
  extern void (*apm_get_power_status)(struct apm_power_info *);
  
  static void sharpsl_apm_get_power_status(struct apm_power_info *info)
  {
  	info->ac_line_status = sharpsl_pm.battstat.ac_status;
  
  	if (sharpsl_pm.charge_mode == CHRG_ON)
  		info->battery_status = APM_BATTERY_STATUS_CHARGING;
  	else
  		info->battery_status = sharpsl_pm.battstat.mainbat_status;
  
  	info->battery_flag = (1 << info->battery_status);
  	info->battery_life = sharpsl_pm.battstat.mainbat_percent;
  }
  
  #ifdef CONFIG_PM
2f55ac072   Lionel Debroux   suspend: constify...
901
  static const struct platform_suspend_ops sharpsl_pm_ops = {
51cdd9289   Pavel Machek   [ARM] pxa/sharpsl...
902
903
  	.prepare	= pxa_pm_prepare,
  	.finish		= pxa_pm_finish,
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
904
905
906
907
  	.enter		= corgi_pxa_pm_enter,
  	.valid		= suspend_valid_only_mem,
  };
  #endif
91a99dfc6   Uwe Kleine-König   platform-drivers:...
908
  static int __devinit sharpsl_pm_probe(struct platform_device *pdev)
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
  {
  	int ret;
  
  	if (!pdev->dev.platform_data)
  		return -EINVAL;
  
  	sharpsl_pm.dev = &pdev->dev;
  	sharpsl_pm.machinfo = pdev->dev.platform_data;
  	sharpsl_pm.charge_mode = CHRG_OFF;
  	sharpsl_pm.flags = 0;
  
  	init_timer(&sharpsl_pm.ac_timer);
  	sharpsl_pm.ac_timer.function = sharpsl_ac_timer;
  
  	init_timer(&sharpsl_pm.chrg_full_timer);
  	sharpsl_pm.chrg_full_timer.function = sharpsl_chrg_full_timer;
  
  	led_trigger_register_simple("sharpsl-charge", &sharpsl_charge_led_trigger);
  
  	sharpsl_pm.machinfo->init();
d5af27783   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
929
930
931
932
933
934
  	gpio_request(sharpsl_pm.machinfo->gpio_acin, "AC IN");
  	gpio_direction_input(sharpsl_pm.machinfo->gpio_acin);
  	gpio_request(sharpsl_pm.machinfo->gpio_batfull, "Battery Full");
  	gpio_direction_input(sharpsl_pm.machinfo->gpio_batfull);
  	gpio_request(sharpsl_pm.machinfo->gpio_batlock, "Battery Lock");
  	gpio_direction_input(sharpsl_pm.machinfo->gpio_batlock);
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
935
936
  
  	/* Register interrupt handlers */
6384fdadb   Haojian Zhuang   ARM: pxa: rename ...
937
938
939
  	if (request_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_acin), sharpsl_ac_isr, IRQF_DISABLED | IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, "AC Input Detect", sharpsl_ac_isr)) {
  		dev_err(sharpsl_pm.dev, "Could not get irq %d.
  ", PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_acin));
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
940
  	}
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
941

6384fdadb   Haojian Zhuang   ARM: pxa: rename ...
942
943
944
  	if (request_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_batlock), sharpsl_fatal_isr, IRQF_DISABLED | IRQF_TRIGGER_FALLING, "Battery Cover", sharpsl_fatal_isr)) {
  		dev_err(sharpsl_pm.dev, "Could not get irq %d.
  ", PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_batlock));
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
945
  	}
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
946
947
  
  	if (sharpsl_pm.machinfo->gpio_fatal) {
6384fdadb   Haojian Zhuang   ARM: pxa: rename ...
948
949
950
  		if (request_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_fatal), sharpsl_fatal_isr, IRQF_DISABLED | IRQF_TRIGGER_FALLING, "Fatal Battery", sharpsl_fatal_isr)) {
  			dev_err(sharpsl_pm.dev, "Could not get irq %d.
  ", PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_fatal));
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
951
  		}
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
952
  	}
b64b0b76c   Pavel Machek   [ARM] pxa/zaurus:...
953
  	if (sharpsl_pm.machinfo->batfull_irq) {
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
954
  		/* Register interrupt handler. */
6384fdadb   Haojian Zhuang   ARM: pxa: rename ...
955
956
957
  		if (request_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_batfull), sharpsl_chrg_full_isr, IRQF_DISABLED | IRQF_TRIGGER_RISING, "CO", sharpsl_chrg_full_isr)) {
  			dev_err(sharpsl_pm.dev, "Could not get irq %d.
  ", PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_batfull));
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
958
  		}
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
959
  	}
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
  	ret = device_create_file(&pdev->dev, &dev_attr_battery_percentage);
  	ret |= device_create_file(&pdev->dev, &dev_attr_battery_voltage);
  	if (ret != 0)
  		dev_warn(&pdev->dev, "Failed to register attributes (%d)
  ", ret);
  
  	apm_get_power_status = sharpsl_apm_get_power_status;
  
  #ifdef CONFIG_PM
  	suspend_set_ops(&sharpsl_pm_ops);
  #endif
  
  	mod_timer(&sharpsl_pm.ac_timer, jiffies + msecs_to_jiffies(250));
  
  	return 0;
  }
  
  static int sharpsl_pm_remove(struct platform_device *pdev)
  {
  	suspend_set_ops(NULL);
  
  	device_remove_file(&pdev->dev, &dev_attr_battery_percentage);
  	device_remove_file(&pdev->dev, &dev_attr_battery_voltage);
  
  	led_trigger_unregister_simple(sharpsl_charge_led_trigger);
6384fdadb   Haojian Zhuang   ARM: pxa: rename ...
985
986
  	free_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_acin), sharpsl_ac_isr);
  	free_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_batlock), sharpsl_fatal_isr);
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
987
988
  
  	if (sharpsl_pm.machinfo->gpio_fatal)
6384fdadb   Haojian Zhuang   ARM: pxa: rename ...
989
  		free_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_fatal), sharpsl_fatal_isr);
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
990
991
  
  	if (sharpsl_pm.machinfo->batfull_irq)
6384fdadb   Haojian Zhuang   ARM: pxa: rename ...
992
  		free_irq(PXA_GPIO_TO_IRQ(sharpsl_pm.machinfo->gpio_batfull), sharpsl_chrg_full_isr);
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
993

d5af27783   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
994
995
996
  	gpio_free(sharpsl_pm.machinfo->gpio_batlock);
  	gpio_free(sharpsl_pm.machinfo->gpio_batfull);
  	gpio_free(sharpsl_pm.machinfo->gpio_acin);
d48898a3c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
997
998
  	if (sharpsl_pm.machinfo->exit)
  		sharpsl_pm.machinfo->exit();
78731d33c   Dmitry Eremin-Solenikov   [ARM] pxa/sharpsl...
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
  
  	del_timer_sync(&sharpsl_pm.chrg_full_timer);
  	del_timer_sync(&sharpsl_pm.ac_timer);
  
  	return 0;
  }
  
  static struct platform_driver sharpsl_pm_driver = {
  	.probe		= sharpsl_pm_probe,
  	.remove		= sharpsl_pm_remove,
  	.suspend	= sharpsl_pm_suspend,
  	.resume		= sharpsl_pm_resume,
  	.driver		= {
  		.name		= "sharpsl-pm",
  	},
  };
  
  static int __devinit sharpsl_pm_init(void)
  {
  	return platform_driver_register(&sharpsl_pm_driver);
  }
  
  static void sharpsl_pm_exit(void)
  {
  	platform_driver_unregister(&sharpsl_pm_driver);
  }
  
  late_initcall(sharpsl_pm_init);
  module_exit(sharpsl_pm_exit);