#include #include #include #include "config.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 "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" /* 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" /* setup stacks in unmapped * iram just as rom will do */ "msr cpsr, #0xd2 \n" "ldr r1, =0x18200274 \n" "add r1, r1, #0x200 \n" "mov sp, r1 \n" "msr cpsr, #0xd3 \n" "add r1, r1, #0x400 \n" "mov sp, r1 \n" /* jump to main() in rom * just before dfu handler */ "ldr r0, =0xec0 \n" "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(); 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); while(!disk_init(IF_MV(0))) panicf("disk_init failed!"); 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 < 0) { printf(rkw_strerror(ret)); lcd_update(); sleep(5*HZ); /* 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(rkw_strerror(0)); 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); }