/****************************************************************************** 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 RIOT. 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. RIOT 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 * @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); extern void bl_config_init(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(); /* UART setup */ bl_uart_init(); puts("Board initialized."); } /** @} */