| /*************************************************************************** |
| * __________ __ ___. |
| * Open \______ \ ____ ____ | | _\_ |__ _______ ___ |
| * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / |
| * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < |
| * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ |
| * \/ \/ \/ \/ \/ |
| * $Id$ |
| * |
| * Copyright (C) 2008 by Rafaël Carré |
| * |
| * 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 "../kernel-internal.h" |
| #include "lcd.h" |
| #ifdef HAVE_BOOTLOADER_USB_MODE |
| #include "usb.h" |
| #include "sysfont.h" |
| #endif /* HAVE_BOOTLOADER_USB_MODE */ |
| #include "backlight.h" |
| #include "button-target.h" |
| #include "common.h" |
| #include "rb-loader.h" |
| #include "loader_strerror.h" |
| #include "storage.h" |
| #include "file_internal.h" |
| #include "disk.h" |
| #include "panic.h" |
| #include "power.h" |
| |
| void show_logo(void); |
| |
| #ifdef HAVE_BOOTLOADER_USB_MODE |
| static void usb_mode(void) |
| { |
| if(usb_detect() != USB_INSERTED) |
| { |
| static const char msg[] = "Plug USB cable"; |
| reset_screen(); |
| lcd_putsxy( (LCD_WIDTH - (SYSFONT_WIDTH * sizeof(msg))) / 2, |
| (LCD_HEIGHT - SYSFONT_HEIGHT) / 2, msg); |
| lcd_update(); |
| |
| /* wait until USB is plugged */ |
| while(usb_detect() != USB_INSERTED) ; |
| } |
| |
| static const char msg[] = "Bootloader USB mode"; |
| reset_screen(); |
| lcd_putsxy( (LCD_WIDTH - (SYSFONT_WIDTH * sizeof(msg))) / 2, |
| (LCD_HEIGHT - SYSFONT_HEIGHT) / 2, msg); |
| lcd_update(); |
| |
| while(usb_detect() == USB_INSERTED) |
| sleep(HZ); |
| |
| reset_screen(); |
| lcd_update(); |
| } |
| #endif /* HAVE_BOOTLOADER_USB_MODE */ |
| |
| void main(void) NORETURN_ATTR; |
| void main(void) |
| { |
| unsigned char* loadbuffer; |
| int buffer_size; |
| void(*kernel_entry)(void); |
| int ret; |
| |
| system_init(); |
| kernel_init(); |
| |
| enable_irq(); |
| |
| lcd_init(); |
| show_logo(); |
| |
| backlight_init(); |
| |
| button_init_device(); |
| int btn = button_read_device(); |
| |
| #if !defined(SANSA_FUZE) && !defined(SANSA_CLIP) && !defined(SANSA_CLIPV2) \ |
| && !defined(SANSA_CLIPPLUS) && !defined(SANSA_CLIPZIP) |
| if (button_hold()) |
| { |
| verbose = true; |
| lcd_clear_display(); |
| printf("Hold switch on"); |
| printf("Shutting down..."); |
| sleep(HZ); |
| power_off(); |
| } |
| #endif |
| |
| /* Enable bootloader messages if any button is pressed */ |
| if (btn) |
| { |
| lcd_clear_display(); |
| verbose = true; |
| } |
| |
| ret = storage_init(); |
| if(ret < 0) |
| error(EATA, ret, true); |
| |
| filesystem_init(); |
| |
| #ifdef HAVE_BOOTLOADER_USB_MODE |
| usb_init(); |
| |
| /* Enter USB mode if USB is plugged and SELECT button is pressed */ |
| if(btn & BUTTON_SELECT) { |
| usb_start_monitoring(); |
| if(usb_detect() == USB_INSERTED) |
| usb_mode(); |
| } |
| #endif /* HAVE_BOOTLOADER_USB_MODE */ |
| |
| while((ret = disk_mount_all()) <= 0) |
| { |
| #ifdef HAVE_BOOTLOADER_USB_MODE |
| error(EDISK, ret, false); |
| usb_start_monitoring(); |
| usb_mode(); |
| #else |
| error(EDISK, ret, true); |
| #endif |
| } |
| |
| printf("Loading firmware"); |
| |
| loadbuffer = (unsigned char*)DRAM_ORIG; /* DRAM */ |
| buffer_size = (int)(loadbuffer + (DRAM_SIZE) - TTB_SIZE); |
| |
| while((ret = load_firmware(loadbuffer, BOOTFILE, buffer_size)) <= EFILE_EMPTY) |
| { |
| #ifdef HAVE_BOOTLOADER_USB_MODE |
| error(EBOOTFILE, ret, false); |
| usb_start_monitoring(); |
| usb_mode(); |
| #else |
| error(EBOOTFILE, ret, true); |
| #endif |
| } |
| |
| #if defined(SANSA_FUZEV2) || defined(SANSA_CLIPPLUS) || defined(SANSA_CLIPZIP) |
| /* It is necessary for proper detection AMSv2 variant 1. |
| * We should restore initial state of GPIOB_PIN(5) as it used for |
| * variant detection, but can be changed if we switch SD card. */ |
| if (amsv2_variant == 1) |
| GPIOB_PIN(5) = 1 << 5; |
| #endif |
| kernel_entry = (void*) loadbuffer; |
| commit_discard_idcache(); |
| printf("Executing"); |
| kernel_entry(); |
| printf("ERR: Failed to boot"); |
| |
| /* never returns */ |
| while(1) ; |
| } |