forked from len0rd/rockbox
		
	This allows targets to report the actual discharging or charging current if they are able to. Change-Id: I0b538e6ac94346f1434e45f83c8da8c1260a53a3
		
			
				
	
	
		
			196 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			196 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2002 by Heikki Hannikainen, Uwe Freese
 | |
|  * Revisions copyright (C) 2005 by Gerald Van Baren
 | |
|  *
 | |
|  * 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| #include "config.h"
 | |
| #include "system.h"
 | |
| #include <time.h>
 | |
| #include "kernel.h"
 | |
| #include "powermgmt.h"
 | |
| #include "power.h"
 | |
| 
 | |
| #define BATT_MINMVOLT   3300      /* minimum millivolts of battery */
 | |
| #define BATT_MAXMVOLT   4300      /* maximum millivolts of battery */
 | |
| #define BATT_MAXRUNTIME (10 * 60) /* maximum runtime with full battery in
 | |
|                                      minutes */
 | |
| /* Number of millivolts to discharge the battery every second */
 | |
| #define BATT_DISCHARGE_STEP ((BATT_MAXMVOLT - BATT_MINMVOLT) / 100)
 | |
| /* Number of millivolts to charge the battery every second */
 | |
| #define BATT_CHARGE_STEP (BATT_DISCHARGE_STEP * 2)
 | |
| #if CONFIG_CHARGING >= CHARGING_MONITOR
 | |
| /* Number of seconds to externally power before discharging again */
 | |
| #define POWER_AFTER_CHARGE_TICKS (8 * HZ)
 | |
| #endif
 | |
| 
 | |
| static bool charging = false;
 | |
| static unsigned int batt_millivolts = BATT_MAXMVOLT;
 | |
| static unsigned int batt_percent = 100;
 | |
| static unsigned int batt_runtime = BATT_MAXRUNTIME;
 | |
| static unsigned int batt_current = 0;
 | |
| 
 | |
| void powermgmt_init_target(void) {}
 | |
| 
 | |
| static void battery_status_update(void)
 | |
| {
 | |
|     /* Delay next battery update until tick */
 | |
|     static long update_after_tick = 0;
 | |
| #if CONFIG_CHARGING >= CHARGING_MONITOR
 | |
|     /* When greater than 0, the tick to unplug the external power at */
 | |
|     static unsigned int ext_power_until_tick = 0;
 | |
| #endif
 | |
| 
 | |
|     if(TIME_BEFORE(current_tick, update_after_tick))
 | |
|         return;
 | |
| 
 | |
|     update_after_tick = current_tick + HZ;
 | |
| 
 | |
| #if CONFIG_CHARGING >= CHARGING_MONITOR
 | |
|     /* Handle period of being externally powered */
 | |
|     if (ext_power_until_tick > 0) {
 | |
|         if (TIME_AFTER(current_tick, ext_power_until_tick)) {
 | |
|             /* Pretend the charger was disconnected */
 | |
|             charger_input_state = CHARGER_UNPLUGGED;
 | |
|             ext_power_until_tick = 0;
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     if (charging) {
 | |
|         batt_millivolts += BATT_CHARGE_STEP;
 | |
|         if (batt_millivolts >= BATT_MAXMVOLT) {
 | |
|             charging = false;
 | |
| #if CONFIG_CHARGING >= CHARGING_MONITOR
 | |
|             /* Keep external power until tick */
 | |
|             ext_power_until_tick = current_tick + POWER_AFTER_CHARGE_TICKS;
 | |
| #elif CONFIG_CHARGING
 | |
|             /* Pretend the charger was disconnected */
 | |
|             charger_input_state = CHARGER_UNPLUGGED;
 | |
| #endif
 | |
|         }
 | |
|     } else {
 | |
|         batt_millivolts -= BATT_DISCHARGE_STEP;
 | |
|         if (batt_millivolts <= BATT_MINMVOLT) {
 | |
|             charging = true;
 | |
| #if CONFIG_CHARGING
 | |
|             /* Pretend the charger was connected */
 | |
|             charger_input_state = CHARGER_PLUGGED;
 | |
| #endif
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     batt_percent = (batt_millivolts - BATT_MINMVOLT) / (BATT_MAXMVOLT - BATT_MINMVOLT);
 | |
|     batt_runtime = batt_percent * BATT_MAXRUNTIME;
 | |
|     /* current is completely bogus... */
 | |
|     batt_current = charging ? BATT_CHARGE_STEP : BATT_DISCHARGE_STEP;
 | |
| }
 | |
| 
 | |
| const unsigned short battery_level_dangerous[BATTERY_TYPES_COUNT] = { 3200 };
 | |
| const unsigned short battery_level_shutoff[BATTERY_TYPES_COUNT] = { 3200 };
 | |
| 
 | |
| /* make the simulated curve nicely linear */
 | |
| const unsigned short percent_to_volt_discharge[BATTERY_TYPES_COUNT][11] =
 | |
| { { 3300, 3400, 3500, 3600, 3700, 3800, 3900, 4000, 4100, 4200, 4300 } };
 | |
| const unsigned short percent_to_volt_charge[11] =
 | |
| { 3300, 3400, 3500, 3600, 3700, 3800, 3900, 4000, 4100, 4200, 4300  };
 | |
| 
 | |
| #if CONFIG_BATTERY_MEASURE & VOLTAGE_MEASURE
 | |
| int _battery_voltage(void)
 | |
| {
 | |
|     battery_status_update();
 | |
|     return batt_millivolts;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #if CONFIG_BATTERY_MEASURE & PERCENTAGE_MEASURE
 | |
| int _battery_level(void)
 | |
| {
 | |
|     battery_status_update();
 | |
|     return batt_percent;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #if (CONFIG_BATTERY_MEASURE & TIME_MEASURE)
 | |
| int _battery_time(void)
 | |
| {
 | |
|     battery_status_update();
 | |
|     return batt_runtime;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #if (CONFIG_BATTERY_MEASURE & CURRENT_MEASURE)
 | |
| int _battery_current(void)
 | |
| {
 | |
|     battery_status_update();
 | |
|     return batt_current;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #if CONFIG_CHARGING
 | |
| unsigned int power_input_status(void)
 | |
| {
 | |
|     unsigned int status = charger_input_state >= CHARGER_PLUGGED
 | |
|             ? POWER_INPUT_CHARGER : POWER_INPUT_NONE;
 | |
| 
 | |
| #ifdef HAVE_BATTERY_SWITCH
 | |
|     status |= POWER_INPUT_BATTERY;
 | |
| #endif
 | |
| 
 | |
|     return status;
 | |
| }
 | |
| 
 | |
| bool charging_state(void)
 | |
| {
 | |
|     return charging;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_ACCESSORY_SUPPLY
 | |
| void accessory_supply_set(bool enable)
 | |
| {
 | |
|     (void)enable;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_LINEOUT_POWEROFF
 | |
| void lineout_set(bool enable)
 | |
| {
 | |
|     (void)enable;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
| bool remote_detect(void)
 | |
| {
 | |
|     return true;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_BATTERY_SWITCH
 | |
| unsigned int input_millivolts(void)
 | |
| {
 | |
|     if ((power_input_status() & POWER_INPUT_BATTERY) == 0) {
 | |
|         /* Just return a safe value if battery isn't connected */
 | |
|         return 4050;
 | |
|     }
 | |
| 
 | |
|     return battery_voltage();
 | |
| }
 | |
| #endif
 |