--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/kernel/eka/include/nkern/arm/entry.h Mon Oct 19 15:55:17 2009 +0100
@@ -0,0 +1,129 @@
+// Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
+// All rights reserved.
+// This component and the accompanying materials are made available
+// under the terms of the License "Eclipse Public License v1.0"
+// which accompanies this distribution, and is available
+// at the URL "http://www.eclipse.org/legal/epl-v10.html".
+//
+// Initial Contributors:
+// Nokia Corporation - initial contribution.
+//
+// Contributors:
+//
+// Description:
+// e32/include/nkern/arm/entry.h
+//
+//
+
+extern "C" {
+
+extern void __ArmVectorReset();
+extern void __ArmVectorUndef();
+extern void __ArmVectorSwi();
+extern void __ArmVectorAbortPrefetch();
+extern void __ArmVectorAbortData();
+extern void __ArmVectorReserved();
+extern void __ArmVectorIrq();
+extern void __ArmVectorFiq();
+
+#define __DECLARE_UNDEFINED_INSTRUCTION_HANDLER asm(".word __ArmVectorUndef ")
+#define __DECLARE_PREFETCH_ABORT_HANDLER asm(".word __ArmVectorAbortPrefetch ")
+#define __DECLARE_DATA_ABORT_HANDLER asm(".word __ArmVectorAbortData ")
+
+/* NOTE: We must ensure that this code goes at the beginning of the kernel image.
+*/
+__NAKED__ void __this_must_go_at_the_beginning_of_the_kernel_image()
+ {
+ asm("ldr pc, __reset_vector "); // 00 = Reset vector
+ asm("ldr pc, __undef_vector "); // 04 = Undefined instruction vector
+ asm("ldr pc, __swi_vector "); // 08 = SWI vector
+ asm("ldr pc, __pabt_vector "); // 0C = Prefetch abort vector
+ asm("ldr pc, __dabt_vector "); // 10 = Data abort vector
+ asm("ldr pc, __unused_vector "); // 14 = unused
+ asm("b HandleIrq "); // 18 = IRQ vector
+ // 1C = FIQ vector, code in situ
+ asm("ldr r12, __ArmInterrupt "); // THIS MUST BE AN IDEMPOTENT INSTRUCTION TO AVOID A PROBLEM WITH XSCALE PXA255
+ asm("sub lr, lr, #4 ");
+ asm("str lr, [sp, #-4]! ");
+ // we assume FIQ handler preserves r0-r7 but not r8-r12
+ // hence must be assembler, so stack misalignment OK
+#if defined(__CPU_ARM_HAS_WORKING_CLREX)
+ CLREX
+#elif defined(__CPU_ARM_HAS_LDREX_STREX)
+ STREX(8,14, 13); // dummy STREX to reset exclusivity monitor
+#endif
+#ifdef __USER_MEMORY_GUARDS_ENABLED__
+ USER_MEMORY_GUARD_ON(,lr,r8);
+ asm("str lr, [sp, #-4]! ");
+#endif
+#ifdef BTRACE_CPU_USAGE
+ asm("ldrb r8, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iCpuUsageFilter));
+ asm("ldr lr, _ArmVectorFiq ");
+ asm("mov r10, #%a0" : : "i" ((TInt)(BTrace::ECpuUsage<<BTrace::ECategoryIndex*8)+(BTrace::EFiqStart<<BTrace::ESubCategoryIndex*8)) );
+ asm("cmp r8,#0");
+ asm("bne btrace_fiq");
+ asm("ldr pc, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iFiqHandler)); // call FIQ handler, return to ArmVectorFiq
+
+ asm("btrace_fiq:"); // call trace handler before fiq handler...
+ asm("stmdb sp!, {r0-r3} ");
+ asm("add r0, r10, #%a0" : : "i" ((TInt)4) ); // add size of trace into header
+ asm("mov lr, pc");
+ asm("ldr pc, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iBTraceHandler));
+ asm("ldr r12, __ArmInterrupt ");
+ asm("ldmia sp!, {r0-r3} ");
+#endif
+ asm("ldr lr, _ArmVectorFiq ");
+ asm("ldr pc, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iFiqHandler)); // call FIQ handler, return to ArmVectorFiq
+
+ asm("HandleIrq: ");
+ asm("sub lr, lr, #4 ");
+ asm("stmfd sp!, {r0-r3,r12,lr} ");
+#if defined(__CPU_ARM_HAS_WORKING_CLREX)
+ CLREX
+#elif defined(__CPU_ARM_HAS_LDREX_STREX)
+ STREX(12, 0, 13); // dummy STREX to reset exclusivity monitor
+#endif
+#ifdef __USER_MEMORY_GUARDS_ENABLED__
+ USER_MEMORY_GUARD_ON(,lr,r12);
+ asm("str lr, [sp, #-8]! ");
+#endif
+ asm("ldr r12, __ArmInterrupt ");
+#ifdef BTRACE_CPU_USAGE
+ asm("mov r0, #%a0" : : "i" ((TInt)(BTrace::ECpuUsage<<BTrace::ECategoryIndex*8)+(BTrace::EIrqStart<<BTrace::ESubCategoryIndex*8)) );
+ asm("add r0, r0, #%a0" : : "i" ((TInt)4) ); // add size of trace into header
+ asm("ldrb r1, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iCpuUsageFilter));
+ asm("ldr lr, _ArmVectorIrq ");
+ asm("cmp r1,#0");
+ asm("bne btrace_irq");
+ asm("ldr pc, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iIrqHandler)); // call IRQ handler, return to ArmVectorIrq
+
+ asm("btrace_irq:"); // call trace handler before irq handler...
+ asm("mov lr, pc");
+ asm("ldr pc, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iBTraceHandler));
+ asm("ldr r12, __ArmInterrupt ");
+#endif
+ asm("ldr lr, _ArmVectorIrq ");
+ asm("ldr pc, [r12,#%a0]" : : "i" _FOFF(SArmInterruptInfo,iIrqHandler)); // call IRQ handler, return to ArmVectorIrq
+
+ asm("__reset_vector:");
+ asm(".word __ArmVectorReset ");
+ asm("__undef_vector:");
+ __DECLARE_UNDEFINED_INSTRUCTION_HANDLER;
+ asm("__swi_vector:");
+ asm(".word __ArmVectorSwi ");
+ asm("__pabt_vector:");
+ __DECLARE_PREFETCH_ABORT_HANDLER;
+ asm("__dabt_vector:");
+ __DECLARE_DATA_ABORT_HANDLER;
+ asm("__unused_vector:");
+ asm(".word __ArmVectorReserved ");
+
+ asm("__ArmInterrupt: ");
+ asm(".word ArmInterruptInfo ");
+ asm("_ArmVectorIrq:");
+ asm(".word __ArmVectorIrq");
+ asm("_ArmVectorFiq:");
+ asm(".word __ArmVectorFiq ");
+ }
+}
+