📄 sched_arm9.s
字号:
#define PID_OFT 0
#define STATE_OFT 4
#define COUNT_OFT 8
#define TIMER_OFT 12
#define PRIORITY_OFT 16
#define CONTENT_OFT 20
#define KERNELSTACK_OFT 100
#define SIZEOF_TASK_STRUCT 100+1024*4
@ 应该转换成虚拟地址
#define SDRAM_BASE 0x30000000
#define SDRAM_SIZE 0x04000000
@ 寄存器切换
@ __switch_to(struct task_struct pcur,struct task_struct pnext):
@ r0 = &pcur, r1 = &pnext
@ 对于linux,__switch_to在arch/arm/kernel/entry-armv.S中定义,以下代码参考它
@ 原进程处于usr/sys模式时(现在处于irq模式:调用__switch_to时处于irq模式的话,原进程必为usr/sys模式):
@ 在task[i].content中保存:sp_svc(在此未变,不必再保存),spsr_irq(即cpsr_usr/sys),sp_usr/sys,spsr_usr/sys(不用保存,无此寄存器),"进入irq时,irq堆栈中保存的{r0-r12,lr}"
@ 原进程处于svc模式时:
@ 在task[i].content中保存:sp_usr/sys,cpsr,sp_svc,spsr_svc,{r0-r12,lr}
@ 假设处于原来进程状态,给task[i].content中的内容名(int表示被中断时):sp_backup,cpsr_int,sp_int,spsr_int,r0-r12,lr
@ 恢复时,根据task[i].content中的内容:
@ 如果spsr_int为svc模式,则先进入sys模式:sp=sp_backup,cpsr=cpsr_int(现在为svc模式),sp=sp_int,spsr=spsr_int,然后恢复{r0-r12,lr}到{r0-r12,pc}
@ 如果spsr_int为usr/sys模式,则先进入svc模式:sp=sp_backup,cpsr=cpsr_int(现在为usr/sys模式),sp=sp_int,然后恢复{r0-r12,lr}到{r0-r12,pc}
@ ENTRY(__switch_to)
.global __switch_to
__switch_to: @ __switch_to有可能从irq和svc模式调用
mrs r2, cpsr
bic r2, r2,#0xffffffe0
cmp r2, #0x12 @ 判断是否为irq模式
beq irq_mode_save
cmp r2, #0x13
beq svc_mode_save
mov pc,lr
irq_mode_save:
@ 保存原进程(usr/sys模式):
add r0, r0,#CONTENT_OFT
ldr sp, =(SDRAM_BASE+0x100000-14*4)
add r0, r0,#4 @ 对应sp_svc,不必再存
mrs r2, spsr @ 读出spsr,保存
stmia r0!,{r2}
stmia r0!,{sp}^ @ usr/sys模式下的sp
mov r2, #0x13 @ 对于usr/sys模式,无spsr,自己写值进去:0x13(表示svc模式)
stmia r0!,{r2}
ldmia sp!,{r2-r8} @ 把进入中断时保存的寄存器转移到当前task struct中
stmia r0!,{r2-r8}
ldmia sp!,{r2-r8}
stmia r0!,{r2-r8}
@ 要切换了...
do_switch:
ldr r0, [r1,#PID_OFT] @ 要切换到的进程的PID
mov r0, r0, lsl #25 @ PID存在寄存器的最高7位
mcr p15, 0, r0, c13, c0, 0 @ 写next_pid,从此,VA<32M的取址计算公式就变了,不过现在的VA总是大于32M的
add r1, r1, #CONTENT_OFT @ r1 = &task[next].content
ldmia r1!,{r2-r5} @ sp_backup,cpsr_int,sp_int,spsr_int
mov r0, r5
bic r0, r0,#0xffffffe0
cmp r0, #0x13 @ 判断是sp_backup对应的模式是否为svc模式
msreq cpsr_c,#0xd3 @ 若是,进入svc模式
msrne cpsr_c,#0xdf @ 否则,进入sys模式
mov sp, r2 @ 设置sp_backup
msr cpsr_cxsf,r3 @ 被中断时的cpsr_int
mov sp,r4 @ sp_int
msrne spsr_cxsf, r5 @ usr/sys模式是没有spsr的
@ 清中断
@ mov r0, #0x4a000000
@ mov r2, #0x400
@ str r2, [r0]
@ add r0, r0,#0x10
@ ldr r2, [r0]
@ str r2, [r0]
@ mov pc, #0
mov r0,r1
ldmia r0,{r0-r12,pc} @ 终于切换了
svc_mode_save:
@ 保存原进程(svc模式):
add r0, r0,#CONTENT_OFT
stmia r0!,{sp}^ @ 对应sp_usr/sys
mrs r2, cpsr @ 读出cpsr,保存
stmia r0!,{r2}
stmia r0!,{sp} @ svc模式下的sp
mrs r2, spsr @ 对应spsr_svc
stmia r0!,{r2}
stmia r0!,{r0-r12,lr}
b do_switch
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -