基于s3c2410软中断服务的uC/OS-II任务切换

出处:21ic 发布于:2018-11-08 14:13:29

  1.关于软中断指令 软件中断指令(SWI)可以产生一个软件中断异常,这为应用程序调用系统例程提供了一种机制。  语法: SWI {} SWI_number SWI执行后的寄存器变化:
  lr_svc = SWI指令后面的指令地址 spsr_svc = cpsr pc = vectors + 0x08
  cpsr模式 = SVC cpsr I = 1(屏蔽IRQ中断)
  处理器执行SWI指令时,设置程序计数器pc为向量表的0x08偏移处,同事强制切换处理器模式到SVC模式,以便操作系统例程可以在特权模式下被调用。
  每个SWI指令有一个关联的SWI号(number),用于表示一个特定的功能调用或特性。
  【例子】 一个ARM工具箱中用于调试SWI的例子,是一个SWI号为0x123456的SWI调用。通常SWI指令是在用户模式下执行的。
  SWI执行前: cpsr = nzcVqift_USER pc = 0x00008000 lr = 0x003fffff ;lr = 4 r0 = 0x12
  执行指令: 0x00008000 SWI 0x123456
  SWI执行后: cpsr = nzcVqIft_SVC spsr = nzcVqift_USER pc = 0x00000008 lr = 0x00008004 r0 = 0x12
  SWI用于调用操作系统的例程,通常需要传递一些参数,这可以通过寄存器来完成。
  在上面的例子中,r0 用于传递参数0x12,返回值也通过寄存器来传递。
  处理软件中断调用的代码段称为中断处理程序(SWI Handler)。中断处理程序通过执行指令的地址获取软件中断号,指令地址是从lr计算出来的。
  SWI号由下式决定: SWI_number = AND NOT<0xff000000> 其中SWI instruction就是实际处理器执行的32位SWI指令
  SWI指令编码为: 31 - 28 27 - 24 23 - 0 cond 1 1 1 1 immed24
  指令的二进制代码的bit23-bit0是24bit的立即数,即SWI指令的中断号,通过屏蔽高8bit即可获得中断号。
  lr寄存器保存的是中断返回指令的地址,所以 [lr - 4] 就是执行SWI的执行代码。
  通过load指令拷贝整个SWI指令到寄存器,使用BIC屏蔽指令的高8位,获取SWI中断号。
  ;read the SWI instruction LDR r10, [lr, #-4] BIC r10, r10, #0xff000000 2. 周立功移植uC/OS-II到s3c2410的软中断服务级的任务切换 uC/OS-II的任务调度函数 uC/OS-II的任务级的调度是由函数OS_Sched( )完成的。
  void OS_Sched (void)
  {
  #if OS_CRITICAL_METHOD == 3 /* Allocate storage for CPU status register */
  OS_CPU_SR cpu_sr;
  #endif
  INT8U y;
  OS_ENTER_CRITICAL();
  if ((OSIntNesting == 0) && (OSLockNesting == 0)) { /* Sched. only if all ISRs done & not locked */
  y = OSUnMapTbl[OSRdyGrp]; /* Get pointer to HPT ready to run */
  OSPrioHighRdy = (INT8U)((y << 3) + OSUnMapTbl[OSRdyTbl[y]]);
  if (OSPrioHighRdy != OSPrioCur) { /* No Ctx Sw if current task is highest rdy */
  OSTCBHighRdy = OSTCBPrioTbl[OSPrioHighRdy];
  OSCtxSwCtr++; /* Increment context switch counter */
  OS_TASK_SW(); /* Perform a context switch */
  }
  }
  OS_EXIT_CRITICAL();
  }
  详细解释可以参考《嵌入式实时操作系统 uC/OS-II》,os_sched函数在确定所有就绪任务的优先级高于当前任务优先级时进行任务切换,通过OS_TASK_SW( )宏来调用。
  OS_TASK_SW( )宏实际上定义的是SWI软中断指令。见OS_CPU.H文件的代码:
  __swi(0x00) void OS_TASK_SW(void); /* 任务级任务切换函数 */
  __swi(0x01) void _OSStartHighRdy(void); /* 运行优先级的任务 */
  __swi(0x02) void OS_ENTER_CRITICAL(void); /* 关中断 */
  __swi(0x03) void OS_EXIT_CRITICAL(void); /* 开中断 */
  __swi(0x40) void *GetOSFunctionAddr(int Index); /* 获取系统服务函数入口 */
  __swi(0x41) void *GetUsrFunctionAddr(int Index);/* 获取自定义服务函数入口 */
  __swi(0x42) void OSISRBegin(void); /* 中断开始处理 */
  __swi(0x43) int OSISRNeedSwap(void); /* 判断中断是否需要切换 */
  __swi(0x80) void ChangeToSYSMode(void); /* 任务切换到系统模式 */
  __swi(0x81) void ChangeToUSRMode(void); /* 任务切换到用户模式 */
  __swi(0x82) void TaskIsARM(INT8U prio); /* 任务代码是ARM代码 */
  __swi(0x83) void TaskIsTHUMB(INT8U prio); /* 任务代码是THUMB */
  __swi(0x00) void OS_TASK_SW(void); 是与ADS相关的代码,通过反汇编可以看到,调用OS_TASK_SW实际上被替换成swi 0x00 软中断指令。执行此执行,pc会跳转到向量表的0x08偏移处。
  中断向量表:(见Startup.s文件)
  CODE32
  AREA vectors,CODE,READONLY
  ; 异常向量表
  Reset
  LDR PC, ResetAddr
  LDR PC, UndefinedAddr
  LDR PC, SWI_Addr
  LDR PC, PrefetchAddr
  LDR PC, DataAbortAddr
  DCD IRQ_Addr
  LDR PC, IRQ_Addr
  LDR PC, FIQ_Addr
  ResetAddr DCD ResetInit
  UndefinedAddr DCD Undefined
  SWI_Addr DCD SoftwareInterrupt
  PrefetchAddr DCD PrefetchAbort
  DataAbortAddr DCD DataAbort
  Nouse DCD 0
  IRQ_Addr DCD IRQ_Handler
  FIQ_Addr DCD FIQ_Handler
  执行SWI 0x00指令后,pc会跳转到SoftwareInterrupt代码处开始执行:
  见Os_cpu_a.s文件的SoftwareInterrupt函数:
  SoftwareInterrupt
  LDR SP, StackSvc ; 重新设置堆栈指针
  STMFD {R0-R3, R12, LR}
  MOV R1, SP ; R1指向参数存储位置
  MRS R3, SPSR
  TST R3, #T_bit ; 中断前是否是Thumb状态
  LDRNEH R0, [LR,#-2] ; 是: 取得Thumb状态SWI指令
  BICNE R0, R0, #0xff00
  LDREQ R0, [LR,#-4] ; 否: 取得arm状态SWI指令
  BICEQ R0, R0, #0xFF000000 ; 如上面所述,此处通过屏蔽SWI指令的高8位来获取SWI号,r0 = SWI号,R1指向参数存储位置
  CMP R0, #1
  LDRLO PC, =OSIntCtxSw ;为0时跳转到OSIntCtxSwdi地址处
  LDREQ PC, =__OSStartHighRdy ; 为1时,跳转到__OSStartHighRdy地址处。SWI 0x01为次任务切换
  BL SWI_Exception ;进入中断号散转函数
  LDMFD {R0-R3, R12, PC}^
  StackSvc DCD (SvcStackSpace + SVC_STACK_LEGTH * 4 - 4)
  以上就是任务切换软中断级服务的实现。

 

版权与免责声明

凡本网注明“出处:维库电子市场网”的所有作品,版权均属于维库电子市场网,转载请必须注明维库电子市场网,https://www.dzsc.com,违反者本网将追究相关法律责任。

本网转载并注明自其它出处的作品,目的在于传递更多信息,并不代表本网赞同其观点或证实其内容的真实性,不承担此类作品侵权行为的直接责任及连带责任。其他媒体、网站或个人从本网转载时,必须保留本网注明的作品出处,并自负版权等法律责任。

如涉及作品内容、版权等问题,请在作品发表之日起一周内与本网联系,否则视为放弃相关权利。

上传BOM文件: BOM文件
*公司名:
*联系人:
*手机号码:
QQ:
应用领域:

有效期:
OEM清单文件: OEM清单文件
*公司名:
*联系人:
*手机号码:
QQ:
有效期:

扫码下载APP,
一键连接广大的电子世界。

在线人工客服

买家服务:
卖家服务:

0571-85317607

客服在线时间周一至周五
9:00-17:30

关注官方微信号,
第一时间获取资讯。

建议反馈

联系人:

联系方式:

按住滑块,拖拽到最右边
>>
感谢您向阿库提出的宝贵意见,您的参与是维库提升服务的动力!意见一经采纳,将有感恩红包奉上哦!