1 /* drivers/power/wm831x_charger_display.c
3 * battery detect driver for the rk2818
5 * This software is licensed under the terms of the GNU General Public
6 * License version 2, as published by the Free Software Foundation, and
7 * may be copied, distributed, and modified under those terms.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
15 #include <linux/module.h>
16 #include <linux/init.h>
17 #include <linux/slab.h>
18 #include <linux/kernel.h>
19 #include <linux/errno.h>
20 #include <linux/input.h>
21 #include <linux/interrupt.h>
23 #include <linux/delay.h>
24 #include <linux/suspend.h>
25 #include <linux/linux_logo.h>
26 #include <linux/regulator/consumer.h>
27 #include <linux/platform_device.h>
28 #include <linux/workqueue.h>
29 #include <linux/mfd/wm831x/core.h>
30 #include <mach/gpio.h>
31 #include <mach/iomux.h>
32 #include <mach/rk29_iomap.h>
34 #include <linux/mfd/wm831x/core.h>
35 #include <linux/mfd/wm831x/auxadc.h>
36 #include <linux/mfd/wm831x/pmu.h>
37 #include <linux/mfd/wm831x/pdata.h>
38 #include <linux/power_supply.h>
41 #define READ_ON_PIN_CNT 20/*11*/
42 #define BACKLIGHT_CNT 2
44 #define MC_OPEN 7/*10*/
45 #define BAT_CHARGING 1
46 #define BAT_DISCHARGING 0
47 #define BL_DELAY_TIME (8*1000)
50 #define SET_BACKLIGHT_ON 1
52 #define SUSPEND_SYSTEM 3
55 #define DBG(x...) printk(x)
60 extern void request_suspend_state(suspend_state_t new_state);
61 extern void kernel_restart(char *cmd);
62 extern void kernel_power_off(void);
64 extern unsigned g_vbus_status_register;
65 extern unsigned long *g_pcd_for_charger;
67 extern int rk29_backlight_ctrl(int open);
69 extern void fb_show_charge_logo(struct linux_logo *logo);
71 extern void kernel_restart_prepare(char *cmd);
72 extern int dwc_vbus_status( void );
73 extern int dwc_otg_pcd_check_vbus_detech( unsigned long pdata );
75 #if defined(CONFIG_LOGO_A22_CLUT224)
76 extern struct linux_logo logo_a22_charger01_clut224;
77 extern struct linux_logo logo_a22_charger02_clut224;
78 extern struct linux_logo logo_a22_charger03_clut224;
79 extern struct linux_logo logo_a22_charger04_clut224;
80 extern struct linux_logo logo_a22_charger05_clut224;
81 extern struct linux_logo logo_a22_charger06_clut224;
82 extern struct linux_logo logo_a22_charger07_clut224;
83 extern struct linux_logo logo_a22_charger08_clut224;
86 extern struct fb_info *g_fb0_inf;
88 static struct linux_logo* g_chargerlogo[8]= {
89 #if defined(CONFIG_LOGO_A22_CLUT224)
90 &logo_a22_charger01_clut224,&logo_a22_charger02_clut224,&logo_a22_charger03_clut224,&logo_a22_charger04_clut224,
91 &logo_a22_charger05_clut224,&logo_a22_charger06_clut224,&logo_a22_charger07_clut224,&logo_a22_charger08_clut224
97 struct wm831x *wm831x;
108 static int charger_logo_display(struct linux_logo *logo)
110 fb_show_charge_logo(logo);
111 fb_show_logo(g_fb0_inf, 0);
115 static int charger_backlight_ctrl(int open)
117 DBG("%s:open=%d\n",__FUNCTION__,open);
118 return rk29_backlight_ctrl(open);
121 static int wm831x_read_on_pin_status(struct wm831x_chg *wm831x_chg)
127 printk("err:%s:wm831x_chg address is 0\n",__FUNCTION__);
131 ret = wm831x_reg_read(wm831x_chg->wm831x, WM831X_ON_PIN_CONTROL);
135 return !(ret & WM831X_ON_PIN_STS) ? 1 : 0;
139 static int wm831x_read_chg_status(struct wm831x_chg *wm831x_chg)
141 int ret, usb_chg = 0, wall_chg = 0;
145 printk("err:%s:wm831x_chg address is 0\n",__FUNCTION__);
149 ret = wm831x_reg_read(wm831x_chg->wm831x, WM831X_SYSTEM_STATUS);
153 if (ret & WM831X_PWR_USB)
155 if (ret & WM831X_PWR_WALL)
158 return ((usb_chg | wall_chg) ? 1 : 0);
161 static int wm831x_bat_check_status(struct wm831x *wm831x, int *status)
165 ret = wm831x_reg_read(wm831x, WM831X_SYSTEM_STATUS);
169 if (ret & WM831X_PWR_SRC_BATT) {
170 *status = POWER_SUPPLY_STATUS_DISCHARGING;
174 ret = wm831x_reg_read(wm831x, WM831X_CHARGER_STATUS);
179 switch (ret & WM831X_CHG_STATE_MASK) {
180 case WM831X_CHG_STATE_OFF:
181 *status = POWER_SUPPLY_STATUS_NOT_CHARGING;
183 case WM831X_CHG_STATE_TRICKLE:
184 case WM831X_CHG_STATE_FAST:
185 *status = POWER_SUPPLY_STATUS_CHARGING;
189 *status = POWER_SUPPLY_STATUS_UNKNOWN;
196 static int wm831x_read_bat_charging_status(struct wm831x_chg *wm831x_chg)
202 printk("err:%s:g_wm831x_power address is 0\n",__FUNCTION__);
206 ret = wm831x_bat_check_status(wm831x_chg->wm831x, &status);
209 if (status == POWER_SUPPLY_STATUS_CHARGING)
214 static int wm831x_read_batt_voltage(struct wm831x_chg *wm831x_chg)
220 printk("err:%s:wm831x_chg address is 0\n",__FUNCTION__);
224 ret = wm831x_auxadc_read_uv(wm831x_chg->wm831x, WM831X_AUX_BATT);
229 static int get_charger_logo_start_num(struct wm831x_chg *wm831x_chg)
231 int rlogonum, bat_vol;
233 /*check charger voltage*/
234 bat_vol = wm831x_read_batt_voltage(wm831x_chg);
235 if(bat_vol <= 3610) {
238 else if(bat_vol <= 3700) {
241 else if(bat_vol <= 3760) {
244 else if(bat_vol <= 3840) {
247 else if(bat_vol <= 3900) {
250 else if(bat_vol <= 3990) {
253 else if(bat_vol <= 4130) {
256 else if(bat_vol <= 4200) {
266 static int wm831x_check_on_pin(struct wm831x_chg *wm831x_chg)
269 ret = wm831x_read_on_pin_status(wm831x_chg);
272 if(wm831x_chg->cnt_on++ > 1000)
273 wm831x_chg->cnt_on = 1000;
277 //control backlight if press on pin
278 if(wm831x_chg->cnt_on >= 1)
280 wm831x_chg->flag_bl = !wm831x_chg->flag_bl;
281 charger_backlight_ctrl(wm831x_chg->flag_bl);
282 wm831x_chg->cnt_on = 0;
283 if(wm831x_chg->flag_bl)
285 wm831x_chg->flag_suspend = 0;
286 wm831x_chg->cnt_disp = 0;
294 static int rk29_charger_display(struct wm831x_chg *wm831x_chg)
297 struct linux_logo* chargerlogo[8];
300 wm831x_chg->flag_chg = wm831x_read_chg_status(wm831x_chg);
301 if(!wm831x_chg->flag_chg)
306 wm831x_chg->flag_chg = wm831x_read_chg_status(wm831x_chg);
307 if(!wm831x_chg->flag_chg)
310 status = wm831x_read_bat_charging_status(wm831x_chg);
313 chargerlogo[i] = g_chargerlogo[i];
315 if(status == BAT_CHARGING)
317 for(i=get_charger_logo_start_num(wm831x_chg); i<8; i++ )
319 wm831x_chg->flag_chg = wm831x_read_chg_status(wm831x_chg);
320 if(!wm831x_chg->flag_chg)
323 ret = charger_logo_display(chargerlogo[i]);
325 DBG("%s:i=%d\n",__FUNCTION__,i);
328 wm831x_check_on_pin(wm831x_chg);
332 else if(status == BAT_DISCHARGING)
334 charger_logo_display(chargerlogo[7]);
336 wm831x_check_on_pin(wm831x_chg);
339 //suspend when timeout(100*200ms)
340 if(wm831x_chg->cnt_disp++ > 100)
342 if(wm831x_chg->flag_suspend == 0)
344 wm831x_chg->flag_suspend = 1;
345 wm831x_chg->cnt_disp = 0;
346 wm831x_chg->flag_bl = 0;
347 charger_backlight_ctrl(wm831x_chg->flag_bl);
349 wm831x_chg->cnt_disp = 0;
352 printk("%s,status=%d,cnt_on=%d,cnt_disp=%d\n",__FUNCTION__,status,wm831x_chg->cnt_on,wm831x_chg->cnt_disp);
354 //open system if long time press
355 if(wm831x_chg->cnt_on > 4)
357 wm831x_chg->cnt_on = 0;
358 wm831x_chg->flag_bl = 1;
359 charger_backlight_ctrl(wm831x_chg->flag_bl);
360 wm831x_chg->flag_suspend = 0;
361 wm831x_chg->cnt_disp = 0;
371 static int __devinit wm831x_chg_probe(struct platform_device *pdev)
373 struct wm831x *wm831x = dev_get_drvdata(pdev->dev.parent);;
374 struct wm831x_chg *wm831x_chg;
377 wm831x_chg = kzalloc(sizeof(struct wm831x_chg), GFP_KERNEL);
379 dev_err(&pdev->dev, "Can't allocate data\n");
382 printk("%s:start\n",__FUNCTION__);
383 wm831x_chg->wm831x = wm831x;
384 wm831x_chg->flag_chg = 0;
385 wm831x_chg->logo_id = 0;
386 wm831x_chg->flag_bl = 1;
387 wm831x_chg->cnt_on = 0;
388 wm831x_chg->flag_suspend = 0;
390 platform_set_drvdata(pdev, wm831x_chg);
392 ret = rk29_charger_display(wm831x_chg);
394 wm831x_chg->flag_chg = 0;
395 wm831x_chg->flag_bl = 1;
396 wm831x_chg->cnt_on = 0;
397 wm831x_chg->flag_suspend = 0;
399 printk("%s:exit\n",__FUNCTION__);
405 static int __devexit wm831x_chg_remove(struct platform_device *pdev)
407 struct wm831x_chg *wm831x_chg = platform_get_drvdata(pdev);
414 static struct platform_driver wm831x_chg_driver = {
415 .probe = wm831x_chg_probe,
416 .remove = __devexit_p(wm831x_chg_remove),
418 .name = "wm831x_charger_display",
419 .owner = THIS_MODULE,
423 static int __init wm831x_chg_init(void)
425 return platform_driver_register(&wm831x_chg_driver);
427 late_initcall(wm831x_chg_init);
429 static void __exit wm831x_chg_exit(void)
431 platform_driver_unregister(&wm831x_chg_driver);
433 module_exit(wm831x_chg_exit);
437 MODULE_LICENSE("GPL");
438 MODULE_AUTHOR("linjh<linjh@rock-chips.com>");
439 MODULE_DESCRIPTION("charger display");