2014-05-20 14:22:24 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
|
|
|
*
|
2014-07-31 19:45:27 +02:00
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
2014-05-20 14:22:24 +02:00
|
|
|
*/
|
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/* GCC ARM assembler */
|
|
|
|
|
|
|
|
.text
|
|
|
|
.code 32
|
|
|
|
.align 4 /* 0 */
|
|
|
|
|
|
|
|
/* Constants */
|
|
|
|
|
|
|
|
.equ USERMODE, 0x10
|
|
|
|
.equ FIQMODE, 0x11
|
|
|
|
.equ IRQMODE, 0x12
|
|
|
|
.equ SVCMODE, 0x13
|
|
|
|
.equ ABORTMODE, 0x17
|
|
|
|
.equ UNDEFMODE, 0x1b
|
|
|
|
.equ SYSTEMMODE, 0x1f
|
|
|
|
.equ MODEMASK, 0x1f
|
|
|
|
.equ NOINT, 0xc0 /* Disable both IRQ & FIR */
|
|
|
|
.equ TBIT, 0x20
|
|
|
|
.equ IRQMASK, 0x80
|
|
|
|
.equ FIQMASK, 0x40
|
|
|
|
|
|
|
|
|
|
|
|
/* External references */
|
|
|
|
|
2014-04-10 22:28:35 +02:00
|
|
|
.extern sched_active_thread
|
2010-10-28 11:22:57 +02:00
|
|
|
.extern sched_context_switch_request
|
|
|
|
.extern sched_run
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/* Public functions declared in this file */
|
2010-10-28 11:22:57 +02:00
|
|
|
.global arm_irq_handler
|
|
|
|
.global cpu_switch_context_exit
|
2010-09-22 15:10:42 +02:00
|
|
|
.global task_return
|
|
|
|
.global ctx_switch
|
|
|
|
|
|
|
|
ctx_switch:
|
|
|
|
/* Save return address on stack */
|
|
|
|
stmfd sp!, {lr}
|
|
|
|
|
|
|
|
/* disable interrupts */
|
|
|
|
mov lr, #NOINT|SVCMODE
|
|
|
|
msr CPSR_c, lr
|
|
|
|
|
|
|
|
ctx_switch2:
|
|
|
|
/* get stack pointer from user mode into lr */
|
|
|
|
stmfd sp, {sp}^
|
|
|
|
sub sp, sp, #4
|
|
|
|
ldmfd sp!, {lr}
|
|
|
|
|
|
|
|
/* save registers on user mode stack pointed to by lr */
|
|
|
|
stmfd lr, {r0-r14}^
|
|
|
|
sub lr, lr, #60
|
|
|
|
|
|
|
|
/* restore saved return address from stack */
|
|
|
|
ldmfd sp!, {r1}
|
|
|
|
|
|
|
|
/* get spsr into register */
|
|
|
|
mrs r0, spsr
|
|
|
|
|
|
|
|
/* store return address and spsr on user mode stack */
|
|
|
|
stmfd lr!, {r0, r1}
|
|
|
|
|
2014-04-10 22:28:35 +02:00
|
|
|
/* save user mode stack pointer in *sched_active_thread */
|
|
|
|
ldr r1, =sched_active_thread /* r1 = &sched_active_thread */
|
|
|
|
ldr r1, [r1] /* r1 = *r1 = sched_active_thread */
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2014-02-11 18:15:43 +01:00
|
|
|
str lr, [r1] /* store stack pointer in tasks tcb*/
|
2010-09-22 15:10:42 +02:00
|
|
|
/* now the calling task has all its registers saved on its stack and it's SP is saved in its tcb */
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2014-04-10 22:28:35 +02:00
|
|
|
/* call scheduler so sched_active_thread points to the next task */
|
2010-10-28 11:22:57 +02:00
|
|
|
bl sched_run
|
|
|
|
b task_return
|
2010-09-22 15:10:42 +02:00
|
|
|
/* never coming back */
|
|
|
|
|
2010-10-28 11:22:57 +02:00
|
|
|
cpu_switch_context_exit:
|
2010-09-22 15:10:42 +02:00
|
|
|
mov r0, #NOINT|SVCMODE
|
|
|
|
msr cpsr, r0
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2014-04-10 22:28:35 +02:00
|
|
|
/* call scheduler so sched_active_thread points to the next task */
|
2010-10-28 11:22:57 +02:00
|
|
|
bl sched_run
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2010-10-28 11:22:57 +02:00
|
|
|
/* continue in task_return: */
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2010-10-28 11:22:57 +02:00
|
|
|
task_return:
|
2010-09-22 15:10:42 +02:00
|
|
|
/* load tcb->stackpointer in r0 */
|
2014-04-10 22:28:35 +02:00
|
|
|
ldr r0, =sched_active_thread /* r0 = &sched_active_thread */
|
|
|
|
ldr r0, [r0] /* r0 = *r0 = sched_active_thread */
|
2010-09-22 15:10:42 +02:00
|
|
|
ldr r0, [r0]
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/* restore saved spsr and return address from tasks stack */
|
|
|
|
ldmfd r0!, {r1,lr}
|
|
|
|
msr SPSR, r1
|
|
|
|
|
|
|
|
/* Restore all registers, PC & cpsr */
|
|
|
|
ldmfd r0, {r0-lr}^
|
2023-05-19 11:08:08 +02:00
|
|
|
/* According to ARM Architecture Reference Manual DDI 0100I,
|
|
|
|
* ldms which affect banked registers must be followed by a nop.
|
|
|
|
*/
|
|
|
|
nop
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/* return to task */
|
|
|
|
movs pc, lr
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------
|
|
|
|
*
|
|
|
|
*----------------------------------------------------------------------------*/
|
2010-10-28 11:22:57 +02:00
|
|
|
arm_irq_handler:
|
2014-08-02 12:21:55 +02:00
|
|
|
sub lr, lr, #4
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/* save interrupted tasks PC onto stack */
|
|
|
|
stmfd sp!, {lr}
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/* save all registers on isr stack */
|
|
|
|
stmfd sp!, {r0-r12}
|
|
|
|
|
|
|
|
/* save spsr on stack */
|
|
|
|
MRS R0, SPSR
|
|
|
|
STMFD SP!, {R0}
|
|
|
|
|
2014-02-11 18:15:43 +01:00
|
|
|
MRS R1, CPSR
|
|
|
|
MSR SPSR, R1
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/* jump into vic interrupt */
|
|
|
|
mov r0, #0xffffff00 /* lpc23xx */
|
2013-11-11 12:30:06 +01:00
|
|
|
ldr r0, [r0]
|
|
|
|
add lr,pc,#4
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
mov pc, r0
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/* restore spsr from stack */
|
|
|
|
LDMFD SP!, {R0}
|
2014-02-11 18:15:43 +01:00
|
|
|
MSR SPSR, R0
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/* check if context switch was requested by irq */
|
2010-10-28 11:22:57 +02:00
|
|
|
ldr r0, =sched_context_switch_request
|
2010-09-22 15:10:42 +02:00
|
|
|
ldr r0, [r0]
|
|
|
|
|
|
|
|
cmp r0, #0x00
|
|
|
|
bne switch_context_int
|
|
|
|
|
|
|
|
exit_irq_int:
|
2014-02-11 18:15:43 +01:00
|
|
|
/* recover general purpose registers */
|
2010-09-22 15:10:42 +02:00
|
|
|
ldmfd sp!, {r0-r12}
|
|
|
|
|
|
|
|
/* recover tasks PC into lr */
|
|
|
|
ldmfd sp!, {lr}
|
|
|
|
|
|
|
|
/* now jump back into task */
|
|
|
|
movs pc, lr
|
|
|
|
|
|
|
|
switch_context_int:
|
2014-02-11 18:15:43 +01:00
|
|
|
/* recover general purpose registers */
|
2010-09-22 15:10:42 +02:00
|
|
|
ldmfd sp!, {r0-r12}
|
|
|
|
|
|
|
|
b ctx_switch2
|