/****************************************************************************** Copyright 2008-2009, Freie Universitaet Berlin (FUB). All rights reserved. These sources were developed at the Freie Universitaet Berlin, Computer Systems and Telematics group (http://cst.mi.fu-berlin.de). ------------------------------------------------------------------------------- This file is part of FeuerWare. 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 3 of the License, or (at your option) any later version. FeuerWare is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see http://www.gnu.org/licenses/ . -------------------------------------------------------------------------------- For further information and questions please use the web site http://scatterweb.mi.fu-berlin.de and the mailinglist (subscription via web site) scatterweb@lists.spline.inf.fu-berlin.de *******************************************************************************/ /** * @ingroup arm_common * @{ */ /** * @file * @internal * @brief ARM bootloader * * @author Freie Universität Berlin, Computer Systems & Telematics, FeuerWhere project * @author Heiko Will * @author Michael Baar * @version $Revision$ * @since 19.08.2008 * * @note $Id$ */ #include #include #include #include void FIQ_Routine (void) __attribute__ ((interrupt("FIQ"))); //void SWI_Routine (void) __attribute__ ((interrupt("SWI"))); void UNDEF_Routine (void) __attribute__ ((interrupt("UNDEF"))); void IRQ_Routine (void) { printf("Kernel Panic,\nEarly IRQ call\n"); while(1){}; } /*-----------------------------------------------------------------------------------*/ void FIQ_Routine (void) { printf("Kernel Panic,\nEarly FIQ call\n"); while(1){}; } /*-----------------------------------------------------------------------------------*/ void SWI_Routine (void) { printf("Kernel Panic,\nEarly SWI call\n"); while(1){}; } /*-----------------------------------------------------------------------------------*/ void DEBUG_Routine (void) { printf("DEBUG hit."); while(1){}; } /*-----------------------------------------------------------------------------------*/ volatile int arm_abortflag = 0; void abtorigin(const char* vector, u_long* lnk_ptr1) { register u_long* lnk_ptr2; register unsigned long* sp; register unsigned int cpsr, spsr; __asm__ __volatile__("mrs %0, cpsr" : "=r" (cpsr) ); // copy current mode __asm__ __volatile__("mrs %0, spsr" : "=r" (spsr) ); // copy dabt generating mode __asm__ __volatile__("msr cpsr_c, %0" :: "r" (spsr) ); // switch to dabt generating mode __asm__ __volatile__("mov %0, lr" : "=r" (lnk_ptr2)); // copy lr __asm__ __volatile__("mov %0, sp" : "=r" (sp)); // copy sp __asm__ __volatile__("msr cpsr_c, %0" :: "r" (cpsr) ); // switch back to abt mode printf("#! %s abort at %p (0x%08lX) originating from %p (0x%08lX) in mode 0x%X\n", vector, (void*)lnk_ptr1, *(lnk_ptr1), (void*)lnk_ptr2, *(lnk_ptr2), spsr ); stdio_flush(); exit(1); } /*-----------------------------------------------------------------------------------*/ void UNDEF_Routine (void) { register u_long* lnk_ptr; __asm__ __volatile__("sub %0, lr, #8" : "=r" (lnk_ptr) ); // get aborting instruction if( arm_abortflag == 0 ) { arm_abortflag = 1; // remember state (if printing should fail again) abtorigin("undef", lnk_ptr); } exit(1); } /*-----------------------------------------------------------------------------------*/ void PABT_Routine (void) { register u_long* lnk_ptr; __asm__ __volatile__("sub %0, lr, #8" : "=r" (lnk_ptr) ); // get aborting instruction if( arm_abortflag == 0 ) { arm_abortflag = 1; // remember state (if printing should fail again) abtorigin("pabt", lnk_ptr); } exit(1); } /*-----------------------------------------------------------------------------------*/ void DABT_Routine (void) { register u_long* lnk_ptr; __asm__ __volatile__("sub %0, lr, #8" : "=r" (lnk_ptr) ); // get aborting instruction if( arm_abortflag == 0 ) { arm_abortflag = 1; // remember state (if printing should fail again) abtorigin("data", lnk_ptr); } exit(1); } /*-----------------------------------------------------------------------------------*/ static inline void bl_init_data(void) { extern unsigned int _etext; extern unsigned int _data; extern unsigned int _edata; extern unsigned int __bss_start; extern unsigned int __bss_end; register unsigned int* p1; register unsigned int* p2; register unsigned int* p3; // initialize data from flash // (linker script ensures that data is 32-bit aligned) p1 = &_etext; p2 = &_data; p3 = &_edata; while( p2 < p3 ) *p2++ = *p1++; // clear bss // (linker script ensures that bss is 32-bit aligned) p1 = &__bss_start; p2 = &__bss_end; while( p1 < p2 ) *p1++ = 0; } /*-----------------------------------------------------------------------------------*/ void bootloader(void) { extern void bl_uart_init(void); extern void bl_init_ports(void); extern void bl_init_clks(void); extern void bl_blink(void); /* board specific setup of clocks */ bl_init_clks(); /* initialize bss and data */ bl_init_data(); /* board specific setup of i/o pins */ bl_init_ports(); /* short blinking of both of the LEDs on startup */ bl_blink(); /* board specific setup of UART */ bl_uart_init(); printf("Board initialized.\n"); } /** @} */