5d236b2bfd
It's now easier to force rebuild of files depending on the svn revision version.c/version.h are generated once with new tools/genversion.sh Changes in the VCS are still not auto detected, so you'll have to remove builddir/version.* if you want to change the string in your binaries APPSVERSION is now called RBVERSION and is defined in the generated header instead of being defined by the Makefiles appsversion is now called rbversion (the plugin api number didn't change since old modules are still binary compatible) Change some bootloaders to use knwon-at-buildtime RBVERSION instead of "%s" + rbversion You'll need to run make clean to regenerate dependencies after the removal of apps/version.h To build binaries with a different version string, hand-edit tools/version.sh or tools/genversion.sh (which calls the former) git-svn-id: svn://svn.rockbox.org/rockbox/trunk@26320 a1c6a512-1295-4272-9138-f99709370657
317 lines
7.4 KiB
C
317 lines
7.4 KiB
C
/***************************************************************************
|
|
* __________ __ ___.
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
* \/ \/ \/ \/ \/
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2008 by Maurus Cuelenaere
|
|
*
|
|
* 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 "jz4740.h"
|
|
#include "backlight.h"
|
|
#include "font.h"
|
|
#include "lcd.h"
|
|
#include "file.h"
|
|
#include "usb.h"
|
|
#include "system.h"
|
|
#include "button.h"
|
|
#include "common.h"
|
|
#include "storage.h"
|
|
#include "disk.h"
|
|
#include "string.h"
|
|
#include "adc.h"
|
|
#include "version.h"
|
|
|
|
extern int show_logo(void);
|
|
extern void power_off(void);
|
|
|
|
static void show_splash(int timeout, const char *msg)
|
|
{
|
|
reset_screen();
|
|
lcd_putsxy( (LCD_WIDTH - (SYSFONT_WIDTH * strlen(msg))) / 2,
|
|
(LCD_HEIGHT - SYSFONT_HEIGHT) / 2, msg);
|
|
lcd_update();
|
|
|
|
sleep(timeout);
|
|
}
|
|
|
|
static void usb_mode(void)
|
|
{
|
|
int button;
|
|
|
|
/* Init USB */
|
|
usb_init();
|
|
usb_start_monitoring();
|
|
|
|
/* Wait for threads to connect */
|
|
show_splash(HZ/2, "Waiting for USB");
|
|
|
|
while (1)
|
|
{
|
|
button = button_get_w_tmo(HZ/2);
|
|
|
|
if (button == SYS_USB_CONNECTED)
|
|
break; /* Hit */
|
|
}
|
|
|
|
if (button == SYS_USB_CONNECTED)
|
|
{
|
|
/* Got the message - wait for disconnect */
|
|
show_splash(0, "Bootloader USB mode");
|
|
|
|
usb_acknowledge(SYS_USB_CONNECTED_ACK);
|
|
|
|
while (1)
|
|
{
|
|
button = button_get(true);
|
|
if (button == SYS_USB_DISCONNECTED)
|
|
{
|
|
usb_acknowledge(SYS_USB_DISCONNECTED_ACK);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static int boot_of(void)
|
|
{
|
|
int fd, rc, len, i, checksum = 0;
|
|
void (*kernel_entry)(int, void*, void*);
|
|
|
|
printf("Mounting disk...");
|
|
rc = disk_mount_all();
|
|
if (rc <= 0)
|
|
error(EDISK,rc);
|
|
|
|
/* TODO: get this from the NAND flash instead of SD */
|
|
fd = open("/ccpmp.bin", O_RDONLY);
|
|
if(fd < 0)
|
|
return EFILE_NOT_FOUND;
|
|
|
|
lseek(fd, 4, SEEK_SET);
|
|
rc = read(fd, (char*)&len, 4); /* CPU is LE */
|
|
if(rc < 4)
|
|
return EREAD_IMAGE_FAILED;
|
|
|
|
len += 8;
|
|
printf("Reading %d bytes...", len);
|
|
|
|
lseek(fd, 0, SEEK_SET);
|
|
rc = read(fd, (void*)0x80004000, len);
|
|
if(rc < len)
|
|
return EREAD_IMAGE_FAILED;
|
|
|
|
close(fd);
|
|
|
|
for(i=0; i<len; i++)
|
|
checksum += ((unsigned char*)0x80004000)[i];
|
|
|
|
*((unsigned int*)0x80004000) = checksum;
|
|
|
|
printf("Starting the OF...");
|
|
|
|
/* OF requires all clocks on */
|
|
__cpm_start_all();
|
|
|
|
disable_interrupt();
|
|
__dcache_writeback_all();
|
|
__icache_invalidate_all();
|
|
|
|
for(i=8000; i>0; i--)
|
|
asm volatile("nop\n");
|
|
|
|
kernel_entry = (void*) 0x80004008;
|
|
kernel_entry(0, "Jan 10 2008", "15:34:42"); /* Reversed from the SPL */
|
|
|
|
return 0; /* Shouldn't happen */
|
|
}
|
|
|
|
static int boot_rockbox(void)
|
|
{
|
|
int rc;
|
|
void (*kernel_entry)(void);
|
|
|
|
printf("Mounting disk...");
|
|
rc = disk_mount_all();
|
|
if (rc <= 0)
|
|
error(EDISK,rc);
|
|
|
|
printf("Loading firmware...");
|
|
rc = load_firmware((unsigned char *)CONFIG_SDRAM_START, BOOTFILE, 0x400000);
|
|
if(rc < 0)
|
|
return rc;
|
|
else
|
|
{
|
|
printf("Starting Rockbox...");
|
|
adc_close(); /* Disable SADC, seems to fix the re-init Rockbox does */
|
|
|
|
disable_interrupt();
|
|
kernel_entry = (void*) CONFIG_SDRAM_START;
|
|
kernel_entry();
|
|
|
|
return 0; /* Shouldn't happen */
|
|
}
|
|
}
|
|
|
|
static void reset_configuration(void)
|
|
{
|
|
int rc;
|
|
|
|
rc = disk_mount_all();
|
|
if (rc <= 0)
|
|
error(EDISK,rc);
|
|
|
|
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!");
|
|
}
|
|
|
|
#define RECT_X (LCD_WIDTH/8)
|
|
#define RECT_Y(i) (LCD_HEIGHT/20 + LCD_HEIGHT/10*i + RECT_HEIGHT*i)
|
|
#define RECT_WIDTH (LCD_WIDTH*3/4)
|
|
#define RECT_HEIGHT (LCD_HEIGHT/ARRAYLEN(strings) - LCD_HEIGHT/10)
|
|
#define TEXT_X(i) (RECT_X + RECT_WIDTH/2 - strlen(strings[i])*SYSFONT_WIDTH/2)
|
|
#define TEXT_Y(i) (RECT_Y(i) + RECT_HEIGHT/2 - SYSFONT_HEIGHT/2)
|
|
static int boot_menu(void)
|
|
{
|
|
const char* strings[] = {"Boot Rockbox", "Boot OF", "USB mode", "Reset Rockbox configuration"};
|
|
int button, touch, poweroff_repeat = 0;
|
|
unsigned int i;
|
|
|
|
verbose = true;
|
|
adc_init();
|
|
|
|
redraw:
|
|
lcd_clear_display();
|
|
for(i=0; i<ARRAYLEN(strings); i++)
|
|
{
|
|
lcd_drawrect(RECT_X, RECT_Y(i), RECT_WIDTH, RECT_HEIGHT);
|
|
lcd_putsxy(TEXT_X(i), TEXT_Y(i), strings[i]);
|
|
}
|
|
lcd_update();
|
|
|
|
while(1)
|
|
{
|
|
button = button_get_w_tmo(HZ/4);
|
|
if(button & BUTTON_TOUCHSCREEN)
|
|
{
|
|
touch = button_get_data();
|
|
unsigned int x = touch & 0xFFFF, y = touch >> 16;
|
|
int found = -1;
|
|
for(i=0; i<ARRAYLEN(strings); i++)
|
|
{
|
|
if(x > RECT_X && x < RECT_X+RECT_WIDTH &&
|
|
y > RECT_Y(i) && y < RECT_Y(i)+RECT_HEIGHT)
|
|
{
|
|
found = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch(found)
|
|
{
|
|
case 0:
|
|
reset_screen();
|
|
boot_rockbox();
|
|
break;
|
|
case 1:
|
|
reset_screen();
|
|
boot_of();
|
|
break;
|
|
case 2:
|
|
usb_mode();
|
|
break;
|
|
case 3:
|
|
reset_configuration();
|
|
break;
|
|
}
|
|
|
|
if(found != -1)
|
|
goto redraw;
|
|
}
|
|
else if(button & BUTTON_POWER)
|
|
{
|
|
if(poweroff_repeat++ > 8)
|
|
power_off();
|
|
}
|
|
else
|
|
poweroff_repeat = 0;
|
|
}
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
int rc;
|
|
#ifdef HAVE_TOUCHSCREEN
|
|
int dummy;
|
|
#endif
|
|
|
|
kernel_init();
|
|
lcd_init();
|
|
font_init();
|
|
lcd_setfont(FONT_SYSFIXED);
|
|
button_init();
|
|
backlight_init();
|
|
|
|
show_logo();
|
|
|
|
rc = storage_init();
|
|
if(rc)
|
|
error(EATA, rc);
|
|
|
|
/* Don't mount the disks yet, there could be file system/partition errors
|
|
which are fixable in USB mode */
|
|
|
|
#ifdef HAVE_TOUCHSCREEN
|
|
rc = button_read_device(&dummy);
|
|
#else
|
|
rc = button_read_device();
|
|
#endif
|
|
|
|
if(rc)
|
|
verbose = true;
|
|
|
|
#ifdef BUTTON_VOL_UP
|
|
if(rc & BUTTON_VOL_UP ||
|
|
#endif
|
|
#ifdef BUTTON_POWER
|
|
rc & BUTTON_POWER ||
|
|
#endif
|
|
0)
|
|
rc = boot_menu();
|
|
|
|
if(verbose)
|
|
reset_screen();
|
|
printf(MODEL_NAME" Rockbox Bootloader");
|
|
printf("Version " RBVERSION);
|
|
|
|
#ifdef HAS_BUTTON_HOLD
|
|
if(button_hold())
|
|
rc = boot_of();
|
|
else
|
|
#endif
|
|
rc = boot_rockbox();
|
|
|
|
if(rc < 0)
|
|
printf("Error: %s", strerror(rc));
|
|
|
|
/* Halt */
|
|
while (1)
|
|
core_idle();
|
|
|
|
return 0;
|
|
}
|