2011-05-01 13:02:46 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Copyright (C) 2011 by amaury Pouly
|
|
|
|
*
|
|
|
|
* Based on Rockbox iriver bootloader by Linus Nielsen Feltzing
|
|
|
|
* and the ipodlinux bootloader by Daniel Palffy and Bernard Leach
|
|
|
|
*
|
|
|
|
* 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 <stdio.h>
|
|
|
|
#include <system.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include "config.h"
|
|
|
|
#include "gcc_extensions.h"
|
|
|
|
#include "lcd.h"
|
|
|
|
#include "backlight.h"
|
|
|
|
#include "button-target.h"
|
|
|
|
#include "common.h"
|
|
|
|
#include "storage.h"
|
|
|
|
#include "disk.h"
|
|
|
|
#include "panic.h"
|
|
|
|
#include "power.h"
|
2011-05-11 22:38:09 +00:00
|
|
|
#include "system-target.h"
|
2011-07-09 17:14:05 +00:00
|
|
|
#include "fmradio_i2c.h"
|
2011-06-17 22:30:58 +00:00
|
|
|
|
|
|
|
#include "usb.h"
|
2011-09-05 11:29:32 +00:00
|
|
|
#include "usb-target.h"
|
|
|
|
|
2011-09-13 23:38:08 +00:00
|
|
|
extern char loadaddress[];
|
|
|
|
extern char loadaddressend[];
|
2011-09-05 11:29:32 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_BOOTLOADER_USB_MODE
|
|
|
|
static void usb_mode(int connect_timeout)
|
|
|
|
{
|
|
|
|
int button;
|
|
|
|
|
|
|
|
usb_init();
|
|
|
|
usb_start_monitoring();
|
|
|
|
|
|
|
|
/* Wait for threads to connect or cable is pulled */
|
|
|
|
printf("USB: Connecting");
|
|
|
|
|
|
|
|
long end_tick = current_tick + connect_timeout;
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
button = button_get_w_tmo(HZ/10);
|
|
|
|
|
|
|
|
if(button == SYS_USB_CONNECTED)
|
|
|
|
break; /* Hit */
|
|
|
|
|
|
|
|
if(TIME_AFTER(current_tick, end_tick))
|
|
|
|
{
|
|
|
|
/* Timed out waiting for the connect - will happen when connected
|
|
|
|
* to a charger through the USB port */
|
|
|
|
printf("USB: Timed out");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!usb_plugged())
|
|
|
|
break; /* Cable pulled */
|
|
|
|
}
|
|
|
|
|
|
|
|
if(button == SYS_USB_CONNECTED)
|
|
|
|
{
|
|
|
|
/* Got the message - wait for disconnect */
|
|
|
|
printf("Bootloader USB mode");
|
|
|
|
|
|
|
|
usb_acknowledge(SYS_USB_CONNECTED_ACK);
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
button = button_get_w_tmo(HZ/2);
|
|
|
|
if(button == SYS_USB_DISCONNECTED)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Put drivers initialized for USB connection into a known state */
|
|
|
|
usb_close();
|
|
|
|
}
|
|
|
|
#else /* !HAVE_BOOTLOADER_USB_MODE */
|
|
|
|
static void usb_mode(int connect_timeout)
|
|
|
|
{
|
|
|
|
(void) connect_timeout;
|
|
|
|
}
|
|
|
|
#endif /* HAVE_BOOTLOADER_USB_MODE */
|
2011-05-01 13:02:46 +00:00
|
|
|
|
2011-09-23 20:40:52 +00:00
|
|
|
void main(uint32_t arg, uint32_t addr) NORETURN_ATTR;
|
|
|
|
void main(uint32_t arg, uint32_t addr)
|
2011-05-01 13:02:46 +00:00
|
|
|
{
|
|
|
|
unsigned char* loadbuffer;
|
|
|
|
int buffer_size;
|
|
|
|
void(*kernel_entry)(void);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
system_init();
|
|
|
|
kernel_init();
|
|
|
|
|
2011-09-05 11:29:32 +00:00
|
|
|
power_init();
|
2011-05-01 13:02:46 +00:00
|
|
|
enable_irq();
|
|
|
|
|
|
|
|
lcd_init();
|
2011-07-02 02:12:10 +00:00
|
|
|
lcd_clear_display();
|
|
|
|
lcd_update();
|
2011-05-01 13:02:46 +00:00
|
|
|
|
|
|
|
backlight_init();
|
|
|
|
|
2011-09-05 11:29:32 +00:00
|
|
|
button_init();
|
2011-05-01 13:02:46 +00:00
|
|
|
|
2011-06-17 22:30:58 +00:00
|
|
|
//button_debug_screen();
|
2011-09-23 20:40:52 +00:00
|
|
|
printf("arg=%x addr=%x", arg, addr);
|
2011-09-05 11:29:32 +00:00
|
|
|
|
|
|
|
#ifdef SANSA_FUZEPLUS
|
|
|
|
extern void imx233_mmc_disable_window(void);
|
|
|
|
if(arg == 0xfee1dead)
|
|
|
|
{
|
|
|
|
printf("Disable MMC window.");
|
|
|
|
imx233_mmc_disable_window();
|
|
|
|
}
|
|
|
|
#endif
|
2011-06-17 22:30:58 +00:00
|
|
|
|
2011-05-01 13:02:46 +00:00
|
|
|
ret = storage_init();
|
|
|
|
if(ret < 0)
|
|
|
|
error(EATA, ret, true);
|
|
|
|
|
2011-09-05 18:32:40 +00:00
|
|
|
/* NOTE: allow disk_init and disk_mount_all to fail since we can do USB after.
|
|
|
|
* We need this order to determine the correct logical sector size */
|
2011-05-01 13:02:46 +00:00
|
|
|
while(!disk_init(IF_MV(0)))
|
2011-09-05 11:29:32 +00:00
|
|
|
printf("disk_init failed!");
|
2011-05-01 13:02:46 +00:00
|
|
|
|
2011-09-05 11:29:32 +00:00
|
|
|
if((ret = disk_mount_all()) <= 0)
|
|
|
|
error(EDISK, ret, false);
|
2011-05-01 13:02:46 +00:00
|
|
|
|
2011-09-05 13:58:29 +00:00
|
|
|
if(usb_plugged())
|
|
|
|
usb_mode(HZ);
|
2011-07-22 15:45:58 +00:00
|
|
|
|
2011-05-01 13:02:46 +00:00
|
|
|
printf("Loading firmware");
|
|
|
|
|
2011-09-13 23:38:08 +00:00
|
|
|
loadbuffer = (unsigned char*)loadaddress;
|
|
|
|
buffer_size = (int)(loadaddressend - loadaddress);
|
2011-05-01 13:02:46 +00:00
|
|
|
|
|
|
|
while((ret = load_firmware(loadbuffer, BOOTFILE, buffer_size)) < 0)
|
|
|
|
{
|
|
|
|
error(EBOOTFILE, ret, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
kernel_entry = (void*) loadbuffer;
|
|
|
|
printf("Executing");
|
2011-09-13 23:38:08 +00:00
|
|
|
disable_interrupt(IRQ_FIQ_STATUS);
|
|
|
|
commit_discard_idcache();
|
2011-05-01 13:02:46 +00:00
|
|
|
kernel_entry();
|
|
|
|
printf("ERR: Failed to boot");
|
|
|
|
|
|
|
|
/* never returns */
|
|
|
|
while(1) ;
|
|
|
|
}
|