2011-05-01 13:02:46 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Copyright (C) 2011 by Amaury Pouly
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2011-07-23 11:45:22 +00:00
|
|
|
#include "config.h"
|
|
|
|
#include "system.h"
|
|
|
|
#include "power.h"
|
2011-11-30 18:47:31 +00:00
|
|
|
#include "string.h"
|
2011-12-31 18:50:03 +00:00
|
|
|
#include "usb.h"
|
2011-07-23 11:45:22 +00:00
|
|
|
#include "system-target.h"
|
2012-01-27 23:43:04 +00:00
|
|
|
#include "power-imx233.h"
|
2012-02-01 16:51:01 +00:00
|
|
|
#include "pinctrl-imx233.h"
|
2012-03-15 13:59:20 +00:00
|
|
|
#include "fmradio_i2c.h"
|
2011-09-05 11:29:32 +00:00
|
|
|
|
2011-12-24 19:20:12 +00:00
|
|
|
struct current_step_bit_t
|
|
|
|
{
|
|
|
|
unsigned current;
|
|
|
|
uint32_t bit;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* in decreasing order */
|
|
|
|
static struct current_step_bit_t g_charger_current_bits[] =
|
|
|
|
{
|
|
|
|
{ 400, HW_POWER_CHARGE__BATTCHRG_I__400mA },
|
|
|
|
{ 200, HW_POWER_CHARGE__BATTCHRG_I__200mA },
|
|
|
|
{ 100, HW_POWER_CHARGE__BATTCHRG_I__100mA },
|
|
|
|
{ 50, HW_POWER_CHARGE__BATTCHRG_I__50mA },
|
|
|
|
{ 20, HW_POWER_CHARGE__BATTCHRG_I__20mA },
|
|
|
|
{ 10, HW_POWER_CHARGE__BATTCHRG_I__10mA }
|
|
|
|
};
|
|
|
|
|
|
|
|
/* in decreasing order */
|
|
|
|
static struct current_step_bit_t g_charger_stop_current_bits[] =
|
|
|
|
{
|
|
|
|
{ 100, HW_POWER_CHARGE__STOP_ILIMIT__100mA },
|
|
|
|
{ 50, HW_POWER_CHARGE__STOP_ILIMIT__50mA },
|
|
|
|
{ 20, HW_POWER_CHARGE__STOP_ILIMIT__20mA },
|
|
|
|
{ 10, HW_POWER_CHARGE__STOP_ILIMIT__10mA }
|
|
|
|
};
|
|
|
|
|
|
|
|
/* in decreasing order */
|
|
|
|
static struct current_step_bit_t g_4p2_charge_limit_bits[] =
|
|
|
|
{
|
|
|
|
{ 400, HW_POWER_5VCTRL__CHARGE_4P2_ILIMIT__400mA },
|
|
|
|
{ 200, HW_POWER_5VCTRL__CHARGE_4P2_ILIMIT__200mA },
|
|
|
|
{ 100, HW_POWER_5VCTRL__CHARGE_4P2_ILIMIT__100mA },
|
|
|
|
{ 50, HW_POWER_5VCTRL__CHARGE_4P2_ILIMIT__50mA },
|
|
|
|
{ 20, HW_POWER_5VCTRL__CHARGE_4P2_ILIMIT__20mA },
|
|
|
|
{ 10, HW_POWER_5VCTRL__CHARGE_4P2_ILIMIT__10mA }
|
|
|
|
};
|
|
|
|
|
2011-09-05 11:29:32 +00:00
|
|
|
void INT_VDD5V(void)
|
|
|
|
{
|
|
|
|
if(HW_POWER_CTRL & HW_POWER_CTRL__VBUSVALID_IRQ)
|
|
|
|
{
|
|
|
|
if(HW_POWER_STS & HW_POWER_STS__VBUSVALID)
|
|
|
|
usb_insert_int();
|
|
|
|
else
|
|
|
|
usb_remove_int();
|
|
|
|
/* reverse polarity */
|
|
|
|
__REG_TOG(HW_POWER_CTRL) = HW_POWER_CTRL__POLARITY_VBUSVALID;
|
|
|
|
/* enable int */
|
|
|
|
__REG_CLR(HW_POWER_CTRL) = HW_POWER_CTRL__VBUSVALID_IRQ;
|
|
|
|
}
|
|
|
|
}
|
2011-07-23 11:45:22 +00:00
|
|
|
|
|
|
|
void power_init(void)
|
|
|
|
{
|
2011-12-24 19:20:12 +00:00
|
|
|
/* setup vbusvalid parameters: set threshold to 4v and power up comparators */
|
|
|
|
__REG_CLR(HW_POWER_5VCTRL) = HW_POWER_5VCTRL__VBUSVALID_TRSH_BM;
|
|
|
|
__REG_SET(HW_POWER_5VCTRL) = HW_POWER_5VCTRL__VBUSVALID_TRSH_4V |
|
|
|
|
HW_POWER_5VCTRL__PWRUP_VBUS_CMPS;
|
|
|
|
/* enable vbusvalid detection method for the dcdc (improves efficiency) */
|
|
|
|
__REG_SET(HW_POWER_5VCTRL) = HW_POWER_5VCTRL__VBUSVALID_5VDETECT;
|
2011-09-05 11:29:32 +00:00
|
|
|
/* clear vbusvalid irq and set correct polarity */
|
|
|
|
__REG_CLR(HW_POWER_CTRL) = HW_POWER_CTRL__VBUSVALID_IRQ;
|
|
|
|
if(HW_POWER_STS & HW_POWER_STS__VBUSVALID)
|
|
|
|
__REG_CLR(HW_POWER_CTRL) = HW_POWER_CTRL__POLARITY_VBUSVALID;
|
|
|
|
else
|
|
|
|
__REG_SET(HW_POWER_CTRL) = HW_POWER_CTRL__POLARITY_VBUSVALID;
|
|
|
|
__REG_SET(HW_POWER_CTRL) = HW_POWER_CTRL__ENIRQ_VBUS_VALID;
|
2012-05-19 11:23:17 +00:00
|
|
|
imx233_icoll_enable_interrupt(INT_SRC_VDD5V, true);
|
2011-12-24 19:20:12 +00:00
|
|
|
/* setup linear regulator offsets to 25 mV below to prevent contention between
|
|
|
|
* linear regulators and DCDC */
|
|
|
|
__FIELD_SET(HW_POWER_VDDDCTRL, LINREG_OFFSET, 2);
|
|
|
|
__FIELD_SET(HW_POWER_VDDACTRL, LINREG_OFFSET, 2);
|
|
|
|
__FIELD_SET(HW_POWER_VDDIOCTRL, LINREG_OFFSET, 2);
|
2012-01-27 23:43:04 +00:00
|
|
|
/* enable a few bits controlling the DC-DC as recommended by Freescale */
|
|
|
|
__REG_SET(HW_POWER_LOOPCTRL) = HW_POWER_LOOPCTRL__TOGGLE_DIF |
|
|
|
|
HW_POWER_LOOPCTRL__EN_CM_HYST;
|
|
|
|
__FIELD_SET(HW_POWER_LOOPCTRL, EN_RCSCALE, HW_POWER_LOOPCTRL__EN_RCSCALE__2X);
|
2011-07-23 11:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void power_off(void)
|
|
|
|
{
|
2011-11-14 21:07:52 +00:00
|
|
|
/* wait a bit, useful for the user to stop touching anything */
|
2011-11-14 19:46:13 +00:00
|
|
|
sleep(HZ / 2);
|
2012-02-01 16:51:01 +00:00
|
|
|
#ifdef SANSA_FUZEPLUS
|
|
|
|
/* This pin seems to be important to shutdown the hardware properly */
|
2012-05-09 22:08:03 +00:00
|
|
|
imx233_pinctrl_acquire_pin(0, 9, "power off");
|
2012-02-01 16:51:01 +00:00
|
|
|
imx233_set_pin_function(0, 9, PINCTRL_FUNCTION_GPIO);
|
|
|
|
imx233_enable_gpio_output(0, 9, true);
|
|
|
|
imx233_set_gpio_output(0, 9, true);
|
|
|
|
#endif
|
2011-07-23 11:45:22 +00:00
|
|
|
/* power down */
|
|
|
|
HW_POWER_RESET = HW_POWER_RESET__UNLOCK | HW_POWER_RESET__PWD;
|
|
|
|
while(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int power_input_status(void)
|
|
|
|
{
|
2011-12-31 18:50:03 +00:00
|
|
|
return (usb_detect() == USB_INSERTED)
|
2011-12-31 18:43:59 +00:00
|
|
|
? POWER_INPUT_MAIN_CHARGER : POWER_INPUT_NONE;
|
2011-07-23 11:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool charging_state(void)
|
|
|
|
{
|
2011-12-24 19:20:12 +00:00
|
|
|
return HW_POWER_STS & HW_POWER_STS__CHRGSTS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void imx233_power_set_charge_current(unsigned current)
|
|
|
|
{
|
|
|
|
__REG_CLR(HW_POWER_CHARGE) = HW_POWER_CHARGE__BATTCHRG_I_BM;
|
|
|
|
/* find closest current LOWER THAN OR EQUAL TO the expected current */
|
|
|
|
for(unsigned i = 0; i < ARRAYLEN(g_charger_current_bits); i++)
|
|
|
|
if(current >= g_charger_current_bits[i].current)
|
|
|
|
{
|
|
|
|
current -= g_charger_current_bits[i].current;
|
|
|
|
__REG_SET(HW_POWER_CHARGE) = g_charger_current_bits[i].bit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void imx233_power_set_stop_current(unsigned current)
|
|
|
|
{
|
|
|
|
__REG_CLR(HW_POWER_CHARGE) = HW_POWER_CHARGE__STOP_ILIMIT_BM;
|
|
|
|
/* find closest current GREATHER THAN OR EQUAL TO the expected current */
|
|
|
|
unsigned sum = 0;
|
|
|
|
for(unsigned i = 0; i < ARRAYLEN(g_charger_stop_current_bits); i++)
|
|
|
|
sum += g_charger_stop_current_bits[i].current;
|
|
|
|
for(unsigned i = 0; i < ARRAYLEN(g_charger_stop_current_bits); i++)
|
|
|
|
{
|
|
|
|
sum -= g_charger_stop_current_bits[i].current;
|
|
|
|
if(current > sum)
|
|
|
|
{
|
|
|
|
current -= g_charger_stop_current_bits[i].current;
|
|
|
|
__REG_SET(HW_POWER_CHARGE) = g_charger_stop_current_bits[i].bit;
|
|
|
|
}
|
|
|
|
}
|
2011-07-23 11:45:22 +00:00
|
|
|
}
|
2011-11-30 18:47:31 +00:00
|
|
|
|
|
|
|
struct imx233_power_info_t imx233_power_get_info(unsigned flags)
|
|
|
|
{
|
|
|
|
static int dcdc_freqsel[8] = {
|
|
|
|
[HW_POWER_MISC__FREQSEL__RES] = 0,
|
|
|
|
[HW_POWER_MISC__FREQSEL__20MHz] = 20000,
|
|
|
|
[HW_POWER_MISC__FREQSEL__24MHz] = 24000,
|
|
|
|
[HW_POWER_MISC__FREQSEL__19p2MHz] = 19200,
|
|
|
|
[HW_POWER_MISC__FREQSEL__14p4MHz] = 14200,
|
|
|
|
[HW_POWER_MISC__FREQSEL__18MHz] = 18000,
|
|
|
|
[HW_POWER_MISC__FREQSEL__21p6MHz] = 21600,
|
|
|
|
[HW_POWER_MISC__FREQSEL__17p28MHz] = 17280,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct imx233_power_info_t s;
|
|
|
|
memset(&s, 0, sizeof(s));
|
|
|
|
if(flags & POWER_INFO_VDDD)
|
|
|
|
{
|
|
|
|
s.vddd = HW_POWER_VDDDCTRL__TRG_MIN + HW_POWER_VDDDCTRL__TRG_STEP * __XTRACT(HW_POWER_VDDDCTRL, TRG);
|
|
|
|
s.vddd_linreg = HW_POWER_VDDDCTRL & HW_POWER_VDDDCTRL__ENABLE_LINREG;
|
2011-12-24 19:20:12 +00:00
|
|
|
s.vddd_linreg_offset = __XTRACT(HW_POWER_VDDDCTRL, LINREG_OFFSET) == 0 ? 0 :
|
|
|
|
__XTRACT(HW_POWER_VDDDCTRL, LINREG_OFFSET) == 1 ? 25 : -25;
|
2011-11-30 18:47:31 +00:00
|
|
|
}
|
|
|
|
if(flags & POWER_INFO_VDDA)
|
|
|
|
{
|
|
|
|
s.vdda = HW_POWER_VDDACTRL__TRG_MIN + HW_POWER_VDDACTRL__TRG_STEP * __XTRACT(HW_POWER_VDDACTRL, TRG);
|
|
|
|
s.vdda_linreg = HW_POWER_VDDACTRL & HW_POWER_VDDACTRL__ENABLE_LINREG;
|
2011-12-24 19:20:12 +00:00
|
|
|
s.vdda_linreg_offset = __XTRACT(HW_POWER_VDDACTRL, LINREG_OFFSET) == 0 ? 0 :
|
|
|
|
__XTRACT(HW_POWER_VDDACTRL, LINREG_OFFSET) == 1 ? 25 : -25;
|
2011-11-30 18:47:31 +00:00
|
|
|
}
|
|
|
|
if(flags & POWER_INFO_VDDIO)
|
2011-12-24 19:20:12 +00:00
|
|
|
{
|
2011-11-30 18:47:31 +00:00
|
|
|
s.vddio = HW_POWER_VDDIOCTRL__TRG_MIN + HW_POWER_VDDIOCTRL__TRG_STEP * __XTRACT(HW_POWER_VDDIOCTRL, TRG);
|
2011-12-24 19:20:12 +00:00
|
|
|
s.vddio_linreg_offset = __XTRACT(HW_POWER_VDDIOCTRL, LINREG_OFFSET) == 0 ? 0 :
|
|
|
|
__XTRACT(HW_POWER_VDDIOCTRL, LINREG_OFFSET) == 1 ? 25 : -25;
|
|
|
|
}
|
2011-11-30 18:47:31 +00:00
|
|
|
if(flags & POWER_INFO_VDDMEM)
|
|
|
|
{
|
|
|
|
s.vddmem = HW_POWER_VDDMEMCTRL__TRG_MIN + HW_POWER_VDDMEMCTRL__TRG_STEP * __XTRACT(HW_POWER_VDDMEMCTRL, TRG);
|
|
|
|
s.vddmem_linreg = HW_POWER_VDDMEMCTRL & HW_POWER_VDDMEMCTRL__ENABLE_LINREG;
|
|
|
|
}
|
|
|
|
if(flags & POWER_INFO_DCDC)
|
|
|
|
{
|
|
|
|
s.dcdc_sel_pllclk = HW_POWER_MISC & HW_POWER_MISC__SEL_PLLCLK;
|
|
|
|
s.dcdc_freqsel = dcdc_freqsel[__XTRACT(HW_POWER_MISC, FREQSEL)];
|
|
|
|
}
|
2011-12-24 19:20:12 +00:00
|
|
|
if(flags & POWER_INFO_CHARGE)
|
|
|
|
{
|
|
|
|
for(unsigned i = 0; i < ARRAYLEN(g_charger_current_bits); i++)
|
|
|
|
if(HW_POWER_CHARGE & g_charger_current_bits[i].bit)
|
|
|
|
s.charge_current += g_charger_current_bits[i].current;
|
|
|
|
for(unsigned i = 0; i < ARRAYLEN(g_charger_stop_current_bits); i++)
|
|
|
|
if(HW_POWER_CHARGE & g_charger_stop_current_bits[i].bit)
|
|
|
|
s.stop_current += g_charger_stop_current_bits[i].current;
|
|
|
|
s.charging = HW_POWER_STS & HW_POWER_STS__CHRGSTS;
|
|
|
|
s.batt_adj = HW_POWER_BATTMONITOR & HW_POWER_BATTMONITOR__ENBATADJ;
|
|
|
|
}
|
|
|
|
if(flags & POWER_INFO_4P2)
|
|
|
|
{
|
|
|
|
s._4p2_enable = HW_POWER_DCDC4P2 & HW_POWER_DCDC4P2__ENABLE_4P2;
|
|
|
|
s._4p2_dcdc = HW_POWER_DCDC4P2 & HW_POWER_DCDC4P2__ENABLE_DCDC;
|
|
|
|
s._4p2_cmptrip = __XTRACT(HW_POWER_DCDC4P2, CMPTRIP);
|
|
|
|
s._4p2_dropout = __XTRACT(HW_POWER_DCDC4P2, DROPOUT_CTRL);
|
|
|
|
}
|
|
|
|
if(flags & POWER_INFO_5V)
|
|
|
|
{
|
|
|
|
s._5v_pwd_charge_4p2 = HW_POWER_5VCTRL & HW_POWER_5VCTRL__PWD_CHARGE_4P2;
|
|
|
|
s._5v_dcdc_xfer = HW_POWER_5VCTRL & HW_POWER_5VCTRL__DCDC_XFER;
|
|
|
|
s._5v_enable_dcdc = HW_POWER_5VCTRL & HW_POWER_5VCTRL__ENABLE_DCDC;
|
|
|
|
for(unsigned i = 0; i < ARRAYLEN(g_4p2_charge_limit_bits); i++)
|
|
|
|
if(HW_POWER_5VCTRL & g_4p2_charge_limit_bits[i].bit)
|
|
|
|
s._5v_charge_4p2_limit += g_4p2_charge_limit_bits[i].current;
|
|
|
|
s._5v_vbusvalid_detect = HW_POWER_5VCTRL & HW_POWER_5VCTRL__VBUSVALID_5VDETECT;
|
|
|
|
s._5v_vbus_cmps = HW_POWER_5VCTRL & HW_POWER_5VCTRL__PWRUP_VBUS_CMPS;
|
|
|
|
s._5v_vbusvalid_thr =
|
|
|
|
__XTRACT(HW_POWER_5VCTRL, VBUSVALID_TRSH) == 0 ?
|
|
|
|
2900
|
|
|
|
: 3900 + __XTRACT(HW_POWER_5VCTRL, VBUSVALID_TRSH) * 100;
|
|
|
|
}
|
2011-11-30 18:47:31 +00:00
|
|
|
return s;
|
|
|
|
}
|