rockbox/bootloader/xduoox3.c

238 lines
5.1 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2016 by Roman Stolyarov
*
* 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 "jz4760b.h"
#include "../kernel-internal.h"
#include "backlight.h"
#include "font.h"
#include "lcd.h"
#include "file.h"
#ifdef HAVE_BOOTLOADER_USB_MODE
#include "usb.h"
#endif
#include "system.h"
#include "button.h"
#include "common.h"
#include "rb-loader.h"
#include "loader_strerror.h"
#include "storage.h"
#include "file_internal.h"
#include "disk.h"
#include "string.h"
#include "adc.h"
#include "version.h"
#include "xdebug.h"
#define SHOW_LOGO
extern void show_logo(void);
extern void power_off(void);
static int lcd_inited = 0;
void init_lcd(void)
{
if(lcd_inited)
return;
lcd_init();
font_init();
lcd_setfont(FONT_SYSFIXED);
lcd_clear_display();
lcd_update();
backlight_init();
lcd_inited = true;
}
#ifdef HAVE_BOOTLOADER_USB_MODE
static void show_splash(int timeout, const char *msg)
{
init_lcd();
reset_screen();
lcd_putsxy( (LCD_WIDTH - (SYSFONT_WIDTH * strlen(msg))) / 2,
(LCD_HEIGHT - SYSFONT_HEIGHT) / 2, msg);
lcd_update();
sleep(timeout);
}
static int usb_inited = 0;
static void usb_mode(void)
{
int button = 0;
/* Init USB, but only once */
if (!usb_inited) {
usb_init();
usb_start_monitoring();
usb_inited = 1;
}
init_lcd();
reset_screen();
/* Wait for USB */
show_splash(0, "Waiting for USB");
while(1) {
button = button_get_w_tmo(HZ/2);
if (button == BUTTON_POWER)
return;
if (button == SYS_USB_CONNECTED)
break; /* Hit */
}
/* Got the message - wait for disconnect */
show_splash(0, "Bootloader USB mode");
usb_acknowledge(SYS_USB_CONNECTED_ACK);
while(1) {
button = button_get_w_tmo(HZ/2);
if (button == SYS_USB_DISCONNECTED)
break;
}
show_splash(HZ*2, "USB disconnected");
}
#endif
/* Jump to loaded binary */
void exec(void* addr) __attribute__((noinline, noreturn, section(".icode")));
void exec(void* addr)
{
commit_discard_idcache();
typedef void(*entry_fn)(void) __attribute__((noreturn));
entry_fn fn = (entry_fn)addr;
fn();
}
static int boot_rockbox(void)
{
int rc;
printf("Mounting disk...");
while((rc = disk_mount_all()) <= 0) {
verbose = true;
#ifdef HAVE_BOOTLOADER_USB_MODE
error(EDISK, rc, false);
usb_mode();
#else
error(EDISK, rc, true);
#endif
}
printf("Loading firmware...");
rc = load_firmware((unsigned char *)CONFIG_SDRAM_START, BOOTFILE, 0x400000);
if(rc <= EFILE_EMPTY) {
printf("!! " BOOTFILE);
return rc;
} else {
printf("Starting Rockbox...");
adc_close(); /* Disable SADC, seems to fix the re-init Rockbox does */
disable_interrupt();
exec((void*) CONFIG_SDRAM_START);
return 0; /* Shouldn't happen */
}
}
#if 0
static void reset_configuration(void)
{
int rc;
rc = disk_mount_all();
if (rc <= 0)
{
verbose = true;
error(EDISK,rc, true);
}
if(rename(ROCKBOX_DIR "/config.cfg", ROCKBOX_DIR "/config.old") == 0)
show_splash(HZ/2, "Configuration reset successfully!");
else
show_splash(HZ/2, "Couldn't reset configuration!");
}
#endif
int main(void)
{
int rc;
serial_puts("\n\nSPL Stage 2\n\n");
system_init();
kernel_init();
init_lcd();
#ifdef SHOW_LOGO
show_logo();
#endif
button_init();
enable_irq();
int btn = button_read_device();
if(btn & BUTTON_POWER) {
verbose = true;
}
rc = storage_init();
if(rc) {
verbose = true;
error(EATA, rc, true);
}
filesystem_init();
#ifdef HAVE_BOOTLOADER_USB_MODE
/* Enter USB mode if USB is plugged and POWER button is pressed */
if (btn & BUTTON_POWER) {
usb_mode();
reset_screen();
}
#endif /* HAVE_BOOTLOADER_USB_MODE */
#ifndef SHOW_LOGO
printf(MODEL_NAME" Rockbox Bootloader");
printf("Version %s", rbversion);
#endif
rc = boot_rockbox();
if(rc <= EFILE_EMPTY) {
verbose = true;
printf("Error: %s", loader_strerror(rc));
}
/* Power off */
sleep(5*HZ);
power_off();
return 0;
}