rockbox/firmware/target/arm/ipod/power-ipod.c
Torne Wuff d4bfc15232 FS#10107: work around the issue on some ipods where after shutdown they do not power on.
Some ipods do not power on correctly after being shut down and require a hard
reset before they work again. This workaround shuts the device down via the OF
which prevents the problem from occurring. There is a cosmetic effect: the low
battery symbol appears briefly on power of.

Change applies to PP502x ipods: 4G/photo/video/mini1g/mini2g/nano1g

Patch by Boris Gjenero (dreamlayers), manual changes by me.


git-svn-id: svn://svn.rockbox.org/rockbox/trunk@24207 a1c6a512-1295-4272-9138-f99709370657
2010-01-09 19:04:51 +00:00

182 lines
5.1 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2002 by Linus Nielsen Feltzing
*
* 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 "cpu.h"
#include <stdbool.h>
#include "kernel.h"
#include "system.h"
#include "power.h"
#include "logf.h"
#include "pcf50605.h"
#include "usb.h"
#include "lcd.h"
#include "string.h"
#if CONFIG_CPU == PP5022 || CONFIG_CPU == PP5020
#include "rtc.h"
#endif
void power_init(void)
{
#if defined(IPOD_1G2G) || defined(IPOD_3G)
GPIOC_ENABLE |= 0x40; /* GPIO C6 is HDD power (low active) */
GPIOC_OUTPUT_VAL &= ~0x40; /* on by default */
GPIOC_OUTPUT_EN |= 0x40; /* enable output */
#endif
#ifndef IPOD_1G2G
pcf50605_init();
#endif
}
#if CONFIG_CHARGING
unsigned int power_input_status(void)
{
unsigned int status = POWER_INPUT_NONE;
#if defined(IPOD_NANO) || defined(IPOD_VIDEO)
if ((GPIOL_INPUT_VAL & 0x08) == 0)
status = POWER_INPUT_MAIN_CHARGER;
if ((GPIOL_INPUT_VAL & 0x10) != 0)
status |= POWER_INPUT_USB_CHARGER;
/* */
#elif defined(IPOD_4G) || defined(IPOD_COLOR) \
|| defined(IPOD_MINI) || defined(IPOD_MINI2G)
/* C2 is firewire power */
if ((GPIOC_INPUT_VAL & 0x04) == 0)
status = POWER_INPUT_MAIN_CHARGER;
if ((GPIOD_INPUT_VAL & 0x08) != 0)
status |= POWER_INPUT_USB_CHARGER;
/* */
#elif defined(IPOD_3G)
/* firewire power */
if ((GPIOC_INPUT_VAL & 0x10) == 0)
status = POWER_INPUT_MAIN_CHARGER;
/* */
#else
/* This needs filling in for other ipods. */
#endif
return status;
}
/* Returns true if the unit is charging the batteries. */
bool charging_state(void) {
#if defined(IPOD_COLOR)
/* 0x70000088 appears to be the input value for GPO32 bits.
Write a zero to 0x70000088 before reading.
To enable input set the corresponding 0x7000008C bit,
and clear the corresponding GPO32_ENABLE bit. */
outl(0, 0x70000088);
return (inl(0x70000088) & 1)?false:true;
#else
return (GPIOB_INPUT_VAL & 0x01)?false:true;
#endif
}
#endif /* CONFIG_CHARGING */
void ide_power_enable(bool on)
{
#if defined(IPOD_1G2G) || defined(IPOD_3G)
if (on)
GPIOC_OUTPUT_VAL &= ~0x40;
else
GPIOC_OUTPUT_VAL |= 0x40;
#elif defined(IPOD_4G) || defined(IPOD_COLOR) \
|| defined(IPOD_MINI) || defined(IPOD_MINI2G)
if (on)
{
GPIO_CLEAR_BITWISE(GPIOJ_OUTPUT_VAL, 0x04);
DEV_EN |= DEV_IDE0;
}
else
{
DEV_EN &= ~DEV_IDE0;
GPIO_SET_BITWISE(GPIOJ_OUTPUT_VAL, 0x04);
}
#elif defined(IPOD_VIDEO)
if (on)
{
GPO32_VAL &= ~0x40000000;
DEV_EN |= DEV_IDE0;
}
else
{
DEV_EN &= ~DEV_IDE0;
GPO32_VAL |= 0x40000000;
}
#else /* Nano */
(void)on; /* Do nothing. */
#endif
}
bool ide_powered(void)
{
#if defined(IPOD_1G2G) || defined(IPOD_3G)
return !(GPIOC_OUTPUT_VAL & 0x40);
#elif defined(IPOD_4G) || defined(IPOD_COLOR) \
|| defined(IPOD_MINI) || defined(IPOD_MINI2G)
return !(GPIOJ_OUTPUT_VAL & 0x04);
#elif defined(IPOD_VIDEO)
return !(GPO32_VAL & 0x40000000);
#else /* Nano */
return true; /* Pretend we are always powered */
#endif
}
void power_off(void)
{
#if defined(HAVE_LCD_COLOR) && !defined(HAVE_LCD_SHUTDOWN)
/* Clear the screen and backdrop to
remove ghosting effect on shutdown */
lcd_set_backdrop(NULL);
lcd_set_background(LCD_WHITE);
lcd_clear_display();
lcd_update();
sleep(HZ/16);
#endif
#ifndef BOOTLOADER
#if CONFIG_CPU == PP5022 || CONFIG_CPU == PP5020
/* When shut down by OF, wakeup via alarm is enabled. This resets the
alarm time so an unintended wakeup does not occur. */
if (!(pcf50605_wakeup_flags & 0x10))
rtc_enable_alarm(false);
#endif
#if defined(IPOD_1G2G)
/* we cannot turn off the 1st gen/ 2nd gen yet. Need to figure out sleep mode. */
system_reboot();
#elif CONFIG_CPU == PP5022
/* The OF in flash assumes boot failed because the battery is low.
If there is no charger connected, this leads to a shutdown.
*/
memcpy((void *)(0x4001ff00+8), "booting!", 8);
system_reboot();
#elif CONFIG_CPU == PP5020
memcpy((void *)(0x40017f00+8), "booting!", 8);
system_reboot();
#else
/* We don't turn off the ipod, we put it in a deep sleep */
pcf50605_standby_mode();
#endif
#endif
}