rockbox/bootloader/rk27xx.c

221 lines
4.8 KiB
C

#include <stdio.h>
#include <inttypes.h>
#include "config.h"
#include "system.h"
#include "../kernel-internal.h"
#include "gcc_extensions.h"
#include "lcd.h"
#include "font.h"
#include "backlight.h"
#include "adc.h"
#include "button-target.h"
#include "button.h"
#include "common.h"
#include "storage.h"
#include "file_internal.h"
#include "disk.h"
#include "panic.h"
#include "power.h"
#include "string.h"
#include "file.h"
#include "crc32-rkw.h"
#include "rkw-loader.h"
#include "version.h"
#include "i2c-rk27xx.h"
#include "loader_strerror.h"
/* beginning of DRAM */
#define DRAM_ORIG 0x60000000
/* bootloader code runs from 0x60700000
* so we cannot load more code to not overwrite ourself
*/
#define LOAD_SIZE 0x700000
extern void show_logo( void );
/* This function setup bare minimum
* and jumps to rom in order to activate
* hardcoded rkusb mode
*/
static void enter_rkusb(void)
{
asm volatile (
/* Turn off cache */
"ldr r0, =0xefff0000 \n"
"ldrh r1, [r0] \n"
"strh r1, [r0] \n"
/* Turn off interrupts */
"mrs r0, cpsr \n"
"bic r0, r0, #0x1f \n"
"orr r0, r0, #0xd3 \n"
"msr cpsr, r0 \n"
/* Disable iram remap */
"mov r0, #0x18000000 \n"
"add r0, r0, #0x1c000 \n"
"mov r1, #0 \n"
"str r1, [r0, #4] \n"
/* Ungate all clocks */
"str r1, [r0, #0x18] \n"
/* Read SCU_ID to determine
* which version of bootrom we have
* 2706A has ID 0xa1000604
* 2706B and 2705 have ID 0xa100027b
*/
"ldr r1, [r0] \n"
"ldr r2, =0xa1000604 \n"
"cmp r1, r2 \n"
"bne rk27xx_new \n"
/* Setup stacks in unmapped
* iram just as rom will do.
*
* We know about two versions
* of bootrom which are very similar
* but memory addresses are slightly
* different.
*/
"rk27xx_old: \n"
"ldr r1, =0x18200258 \n"
"ldr r0, =0xaf0 \n"
"b jump_to_rom \n"
"rk27xx_new: \n"
"ldr r1, =0x18200274 \n"
"ldr r0, =0xec0 \n"
"jump_to_rom: \n"
"msr cpsr, #0xd2 \n"
"add r1, r1, #0x200 \n"
"mov sp, r1 \n"
"msr cpsr, #0xd3 \n"
"add r1, r1, #0x400 \n"
"mov sp, r1 \n"
/* Finaly jump to rkusb handler
* in bootrom.
*/
"bx r0 \n"
);
}
void main(void) NORETURN_ATTR;
void main(void)
{
char filename[MAX_PATH];
unsigned char* loadbuffer;
void(*kernel_entry)(void);
int ret;
enum {rb, of} boot = rb;
power_init();
system_init();
kernel_init();
i2c_init();
enable_irq();
adc_init();
lcd_init();
backlight_init();
button_init_device();
font_init();
lcd_setfont(FONT_SYSFIXED);
show_logo();
int btn = button_read_device();
/* if there is some other button pressed
* besides POWER/PLAY we boot into OF
*/
if ((btn & ~POWEROFF_BUTTON))
boot = of;
/* if we are woken up by USB insert boot into OF */
if (DEV_INFO & (1<<20))
boot = of;
lcd_clear_display();
ret = storage_init();
if(ret < 0)
error(EATA, ret, true);
filesystem_init();
while((ret = disk_mount_all()) <= 0)
error(EDISK, ret, true);
loadbuffer = (unsigned char*)DRAM_ORIG; /* DRAM */
if (boot == rb)
snprintf(filename,sizeof(filename), BOOTDIR "/%s", BOOTFILE);
else if (boot == of)
snprintf(filename,sizeof(filename), BOOTDIR "/%s", "BASE.RKW");
printf("Bootloader version: %s", rbversion);
printf("Loading: %s", filename);
ret = load_rkw(loadbuffer, filename, LOAD_SIZE);
if (ret <= EFILE_EMPTY)
{
error(EBOOTFILE, ret, false);
/* if we boot rockbox we shutdown on error
* if we boot OF we fall back to rkusb mode on error
*/
if (boot == rb)
{
power_off();
}
else
{
/* give visual feedback what we are doing */
printf("Entering rockchip USB mode...");
lcd_update();
enter_rkusb();
}
}
else
{
/* print 'Loading OK' */
printf("Loading OK");
sleep(HZ);
}
/* jump to entrypoint */
kernel_entry = (void*) loadbuffer;
commit_discard_idcache();
printf("Executing");
kernel_entry();
/* this should never be reached actually */
printf("ERR: Failed to boot");
sleep(5*HZ);
if (boot == rb)
{
power_off();
}
else
{
/* give visual feedback what we are doing */
printf("Entering rockchip USB mode...");
lcd_update();
enter_rkusb();
}
/* hang */
while(1);
}