context_gcc.S 5.4 KB
Newer Older
B
Bernard Xiong 已提交
1 2 3 4 5 6 7 8 9 10 11 12
/*
 * File      : context_gcc.S
 * This file is part of RT-Thread RTOS
 * COPYRIGHT (C) 2009 - 2011, RT-Thread Development Team
 *
 * The license and distribution terms for this file may be
 * found in the file LICENSE in this distribution or at
 * http://www.rt-thread.org/license/LICENSE
 *
 * Change Logs:
 * Date         Author    Notes
 * 2009-10-11   Bernard   First version
13 14 15 16
 * 2010-12-29   onelife   Modify for EFM32
 * 2011-06-17   onelife   Merge all of the assembly source code into context_gcc.S
 * 2011-07-12   onelife   Add interrupt context check function
 * 2013-06-18   aozima    add restore MSP feature.
B
Bernard Xiong 已提交
17 18
 */
 
19 20 21 22 23 24
    .cpu    cortex-m3
    .fpu    softvfp
    .syntax unified
    .thumb
    .text

25
    .equ    SCB_VTOR, 0xE000ED08            /* Vector Table Offset Register */
26 27 28 29 30
    .equ    ICSR, 0xE000ED04                /* interrupt control state register */
    .equ    PENDSVSET_BIT, 0x10000000       /* value to trigger PendSV exception */
    
    .equ    SHPR3, 0xE000ED20               /* system priority register (3) */
    .equ    PENDSV_PRI_LOWEST, 0x00FF0000   /* PendSV priority value (lowest) */
B
Bernard Xiong 已提交
31 32 33 34

/*
 * rt_base_t rt_hw_interrupt_disable();
 */
35 36
    .global rt_hw_interrupt_disable
    .type rt_hw_interrupt_disable, %function
B
Bernard Xiong 已提交
37
rt_hw_interrupt_disable:
38 39 40
    MRS     R0, PRIMASK
    CPSID   I
    BX      LR
B
Bernard Xiong 已提交
41 42 43 44

/*
 * void rt_hw_interrupt_enable(rt_base_t level);
 */
45 46
    .global rt_hw_interrupt_enable
    .type rt_hw_interrupt_enable, %function
B
Bernard Xiong 已提交
47
rt_hw_interrupt_enable:
48 49
    MSR     PRIMASK, R0
    BX      LR
B
Bernard Xiong 已提交
50 51 52 53 54 55

/*
 * void rt_hw_context_switch(rt_uint32 from, rt_uint32 to);
 * R0 --> from
 * R1 --> to
 */
56 57 58 59
    .global rt_hw_context_switch_interrupt
    .type rt_hw_context_switch_interrupt, %function
    .global rt_hw_context_switch
    .type rt_hw_context_switch, %function
B
Bernard Xiong 已提交
60 61
rt_hw_context_switch_interrupt:
rt_hw_context_switch:
62 63 64 65 66 67 68
    /* set rt_thread_switch_interrupt_flag to 1 */
    LDR     R2, =rt_thread_switch_interrupt_flag
    LDR     R3, [R2]
    CMP     R3, #1
    BEQ     _reswitch
    MOV     R3, #1
    STR     R3, [R2]
B
Bernard Xiong 已提交
69

70 71
    LDR     R2, =rt_interrupt_from_thread   /* set rt_interrupt_from_thread */
    STR     R0, [R2]
B
Bernard Xiong 已提交
72 73

_reswitch:
74 75
    LDR     R2, =rt_interrupt_to_thread     /* set rt_interrupt_to_thread */
    STR     R1, [R2]
B
Bernard Xiong 已提交
76

77 78 79 80
    LDR     R0, =ICSR           /* trigger the PendSV exception (causes context switch) */
    LDR     R1, =PENDSVSET_BIT
    STR     R1, [R0]
    BX      LR
B
Bernard Xiong 已提交
81 82 83 84 85

/* R0 --> swith from thread stack
 * R1 --> swith to thread stack
 * psr, pc, LR, R12, R3, R2, R1, R0 are pushed into [from] stack
 */
86 87
    .global PendSV_Handler
    .type PendSV_Handler, %function
B
Bernard Xiong 已提交
88
PendSV_Handler:
89 90 91
    /* disable interrupt to protect context switch */
    MRS     R2, PRIMASK
    CPSID   I
B
Bernard Xiong 已提交
92

93 94 95 96
    /* get rt_thread_switch_interrupt_flag */
    LDR     R0, =rt_thread_switch_interrupt_flag
    LDR     R1, [R0]
    CBZ     R1, pendsv_exit         /* pendsv aLReady handled */
B
Bernard Xiong 已提交
97

98 99 100
    /* clear rt_thread_switch_interrupt_flag to 0 */
    MOV     R1, #0
    STR     R1, [R0]
B
Bernard Xiong 已提交
101

102 103 104
    LDR     R0, =rt_interrupt_from_thread
    LDR     R1, [R0]
    CBZ     R1, swtich_to_thread    /* skip register save at the first time */
B
Bernard Xiong 已提交
105

106 107 108 109
    MRS     R1, PSP                 /* get from thread stack pointer */
    STMFD   R1!, {R4 - R11}         /* push R4 - R11 register */
    LDR     R0, [R0]
    STR     R1, [R0]                /* update from thread stack pointer */
B
Bernard Xiong 已提交
110 111

swtich_to_thread:
112 113 114
    LDR     R1, =rt_interrupt_to_thread
    LDR     R1, [R1]
    LDR     R1, [R1]                /* load thread stack pointer */
B
Bernard Xiong 已提交
115

116 117
    LDMFD   R1!, {R4 - R11}         /* pop R4 - R11 register */
    MSR     PSP, R1                 /* update stack pointer */
B
Bernard Xiong 已提交
118 119

pendsv_exit:
120 121
    /* restore interrupt */
    MSR     PRIMASK, R2
B
Bernard Xiong 已提交
122

123 124
    ORR     LR, LR, #0x04
    BX      LR
B
Bernard Xiong 已提交
125 126 127 128 129

/*
 * void rt_hw_context_switch_to(rt_uint32 to);
 * R0 --> to
 */
130 131
    .global rt_hw_context_switch_to
    .type rt_hw_context_switch_to, %function
B
Bernard Xiong 已提交
132
rt_hw_context_switch_to:
133 134 135 136 137 138 139
    LDR     R1, =rt_interrupt_to_thread
    STR     R0, [R1]

    /* set from thread to 0 */
    LDR     R1, =rt_interrupt_from_thread
    MOV     R0, #0
    STR     R0, [R1]
B
Bernard Xiong 已提交
140

141 142 143 144
    /* set interrupt flag to 1 */
    LDR     R1, =rt_thread_switch_interrupt_flag
    MOV     R0, #1
    STR     R0, [R1]
B
Bernard Xiong 已提交
145

146 147 148 149 150 151
    /* set the PendSV exception priority */
    LDR     R0, =SHPR3
    LDR     R1, =PENDSV_PRI_LOWEST
    LDR.W   R2, [R0,#0]             /* read */
    ORR     R1, R1, R2              /* modify */
    STR     R1, [R0]                /* write-back */
B
Bernard Xiong 已提交
152

153 154 155
    LDR     R0, =ICSR               /* trigger the PendSV exception (causes context switch) */
    LDR     R1, =PENDSVSET_BIT
    STR     R1, [R0]
B
Bernard Xiong 已提交
156

157 158 159 160 161 162
    /* restore MSP */
    LDR     r0, =SCB_VTOR
    LDR     r0, [r0]
    LDR     r0, [r0]
    NOP
    MSR     msp, r0
B
Bernard Xiong 已提交
163

164
    CPSIE   I                       /* enable interrupts at processor level */
B
Bernard Xiong 已提交
165

166
    /* never reach here! */
B
Bernard Xiong 已提交
167 168

/* compatible with old version */
169 170
    .global rt_hw_interrupt_thread_switch
    .type rt_hw_interrupt_thread_switch, %function
B
Bernard Xiong 已提交
171
rt_hw_interrupt_thread_switch:
172 173
    BX      LR
    NOP
B
Bernard Xiong 已提交
174

175 176
    .global HardFault_Handler
    .type HardFault_Handler, %function
B
Bernard Xiong 已提交
177 178 179 180 181 182 183 184 185 186 187 188 189 190
HardFault_Handler:
    /* get current context */
    MRS     R0, PSP                 /* get fault thread stack pointer */
    PUSH    {LR}
    BL      rt_hw_hard_fault_exception
    POP     {LR}

    ORR     LR, LR, #0x04
    BX      LR

/*
 * rt_uint32_t rt_hw_interrupt_check(void);
 * R0 --> state
 */
191 192
    .global rt_hw_interrupt_check
    .type rt_hw_interrupt_check, %function
B
Bernard Xiong 已提交
193
rt_hw_interrupt_check:
194 195
    MRS     R0, IPSR
    BX      LR