⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sched_arm9.s

📁 这是一个不到2000行的操作系统
💻 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中定义,以下代码参考它@@ 进程上下文content中的内容为(int表示被中断时的状态):sp_another_state,cpsr_int,sp_int,spsr_int,lr_usr/sys,{r0-r12,lr}_int@ 用户进程只可能处于两个状态:用户态(usr/sys),内核态(svc),它们互为another_state@	@ 原进程处于usr/sys模式时(现在处于irq模式:调用__switch_to时处于irq模式的话,原进程必为usr/sys模式):	@ 在pcur->content中需要保存:sp_svc(在此未变,不必再保存),spsr_irq(即cpsr_usr/sys),sp_usr/sys,spsr_usr/sys(不用保存,无此寄存器),lr_usr/sys,"进入irq时,irq堆栈中保存的{r0-r12,lr}"@ 原进程处于svc模式时:	@ 在pcur->content中需要保存:sp_usr/sys,cpsr,sp_svc,spsr_svc,lr_usr/sys,{r0-r12,lr}@@ 恢复时,根据pnext->content中的内容:	@ 如果spsr_int为svc模式,则先进入svc模式,令sp=sp_another_state,然后切换模式cpsr=cpsr_int(现在为usr/sys模式),最后恢复所有寄存器:sp=sp_int,spsr=spsr_int,lr_usr/sys,{r0-r12,pc}@ 如果spsr_int为usr/sys模式,则先进入usr/sys模式,令sp=sp_another_state,然后切换模式cpsr=cpsr_int(现在为svc模式),最后恢复所有寄存器:sp=sp_int,lr_usr/sys,{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,lrirq_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}	stmia	r0!,{lr}^		@ save lr_usr/sys	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 = pnext->content	ldmia	r1!,{r2-r5}		@ sp_another_state,cpsr_int,sp_int,spsr_int	mov	r0, r5	bic	r0, r0,#0xffffffe0	cmp	r0, #0x13		@ 判断是sp_another_state对应的模式是否为svc模式	msreq	cpsr_c,#0xd3	@ 若是,进入svc模式	msrne	cpsr_c,#0xdf	@ 否则,进入sys模式	mov	sp, r2		@ 设置sp_another_state	@ 清中断	mov	r0, #0x4a000000	mov	r2, #0x400	str	r2, [r0]	add	r0, r0,#0x10	ldr	r2, [r0]	str	r2, [r0]	msr	cpsr_cxsf,r3	@ 被中断时的cpsr_int	mov	sp,r4		@ sp_int	msrne	spsr_cxsf, r5	@ 现在处于svc模式,而usr/sys模式是没有spsr的		mov	r0,r1	ldmeqia	r0!,{lr}^		@ lr_usr/sys	ldmneia	r0!,{lr}^	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!,{lr}^		@ lr_usr/sys	stmia	r0!,{r0-r12,lr}	b	do_switch

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -