μC/OS-II 移植笔记 2(FreeScale 68HCS12 核单片机)
首先是函数和全局变量的声明。
- ;***************************************************************************
- ;PUBLICDECLARATIONS
- ;***************************************************************************
- xdefOSCPUSaveSR
- xdefOSCPURestoreSR
- xdefOSStartHighRdy
- xdefOSCtxSw
- xdefOSIntCtxSw
- xdefOSTickISR
- ;***************************************************************************
- ;EXTERNALDECLARATIONS
- ;***************************************************************************
- xrefOSIntExit
- xrefOSIntNesting
- xrefOSPrioCur
- xrefOSPrioHighRdy
- xrefOSRunning
- xrefOSTaskSwHook
- xrefOSTCBCur
- xrefOSTCBHighRdy
- xrefOSTimeTick
然后是临界区的相关代码。
- OSCPUSaveSR:
- tfrccr,b;Itsassumedthat8-bitreturnvalueisinregisterB
- sei;Disableinterrupts
- rts;ReturntocallerwithBcontainingthepreviousCCR
- OSCPURestoreSR:
- tfrb,ccr;BcontainstheCCRvaluetorestore,movetoCCR
- rts
下面的代码是重点部分,首先是 OSStartHighRdy 函数,虽然这个函数只在 OSStart 函数中被调用一次,在之后的程序生命周期中就再也用不到了,但这次调用至关重要,决定了用户任务是否能被调度起来。因此代码中给出的注释尽可能的详细,我想看过注释后就不需要我解释什么了。
- ;***********************************************************************
- ;STARTHIGHESTPRIORITYTASKREADY-TO-RUN
- ;
- ;Description:ThisfunctioniscalledbyOSStart()tostart
- ;thehighestprioritytaskthatwascreatedbyyour
- ;applicationbeforecallingOSStart().
- ;
- ;Arguments:none
- ;
- ;Note(s):1)Thestackframeisassumedtolookasfollows:
- ;
- ;OSTCBHighRdy->OSTCBStkPtr+0CCR
- ;+1B
- ;+2A
- ;+3X(H)
- ;+4X(L)
- ;+5Y(H)
- ;+6Y(L)
- ;+7PC(H)
- ;+8PC(L)
- ;
- ;2)OSStartHighRdy()MUST:
- ;a)CallOSTaskSwHook()then,
- ;b)SetOSRunningtoTRUE,
- ;c)Switchtothehighestprioritytaskbyloading
- ;thestackpointerofthehighestprioritytask
- ;intotheSPregisterandexecutean
- ;RTIinstruction.
- ;************************************************************************
- OSStartHighRdy:
- jsrOSTaskSwHook;4~,调用OSTaskSwHook
- ldab#$01;2~,OSRunning=1
- stabOSRunning;4~
- ldxOSTCBHighRdy;3~,将OSTCBHighRdy的地址放到寄存器X
- lds0,x;3~,将OSTCBHighRdy->OSTCBStkPtr放到SP
- nop
- rti;4~,Runtask
其实上面的代码也可以放到OS_CPU_C.C 中,下面是个示例:
- #pragmaCODE_SEGNON_BANKED
- #pragmaTRAP_PROCSAVE_NO_REGS
- voidOSStartHighRdy(void)
- {
- __asmjsrOSTaskSwHook;//OSTaskSwHook();
- __asmldab#$01;
- __asmstabOSRunning;//OSRunning=TRUE;
- __asm
- {
- ldxOSTCBHighRdy
- lds0,x
- nop
- }
- }
上面代码中 #pragma TRAP_PROC SAVE_NO_REGS 表示这是个中断处理函数,编译器不为之保存任何寄存器内容。 虽然 OSStartHighRdy 并不是个真正的中断处理函数,但它模拟却模拟了中断处理函数的行为,函数结束时调用 rti 而不是 rts 命令。
下面是任务切换的代码,注释已经足够的详细了,如果有不明白的地方建议将 Jean J.Labrosse 的书再仔细读读。
- OSCtxSw:
- ldyOSTCBCur;3~,OSTCBCur->OSTCBStkPtr=StackPointer
- sts0,y
- OSIntCtxSw:
- jsrOSTaskSwHook;4~,Callusertaskswitchhook
- ldxOSTCBHighRdy;3~,OSTCBCur=OSTCBHighRdy
- stxOSTCBCur
- ldabOSPrioHighRdy;3~,OSPrioCur=OSPrioHighRdy
- stabOSPrioCur
- lds0,x;3~,LoadSPinto68HC12
- nop
- rti;8~,Runtask
可以看到,上面两个函数公用了大部分的函数体。上面的代码也可以直接写到 OS_CPU_C.C 中,不过写成 C 函数后就不能公用函数体了。
最后一部分是时钟中断程序,我使用RTI中断作为周期性时钟源。
- OSTickISR:
- incOSIntNesting;4~,NotifyuC/OS-IIaboutISR
- ldabOSIntNesting;4~,if(OSIntNesting==1){
- cmpb#$01
- bneOSTickISR1
- ldyOSTCBCur;OSTCBCur->OSTCBStkPtr=StackPointer
- sts0,y;}
- OSTickISR1:
- BSET$37,#128;CRGFLG_RTIF=1,这句是反汇编出来的,应该没错
- jsrOSTimeTick
- jsrOSIntExit;6~+,NotifyuC/OS-IIaboutendofISR
- rti;12~,Returnfrominterrupt,nohigherprioritytasksready.
中断程序的大部分代码都比较简答,只有下面这句我下了番功夫才写出来:
BSET $37, #128
与这行代码功能相同的 C 代码是:CRGFLG_RTIF = 1
我将 C代码直接生成汇编代码的结果是:BSET _CRGFLG,#128
可是直接拿到汇编文件中却无法编译,提示说 _CRGFLG 没有定义。一番查找才确定了_CRGFLG = 0x37。
2.3 OS_CPU_C.C
由于大部分的移植代码都放到了汇编文件中,OS_CPU_C.C 中的工作就很少了。OS_CPU_C.C 最重要的工作是 OSTaskStkInit 函数,并且网上流传的大多数 68HC12 内核的移植代码的这部分都或多或少的有问题。下面先给出我的代码:
- OS_STK*OSTaskStkInit(void(*task)(void*pd),void*p_arg,OS_STK*ptos,INT16Uopt)
- {
- INT16U*wstk;
- INT8U*bstk;
- (void)opt;/*optisnotused,preventwarning*/
- ptos--;/*需要这么调整一下栈顶地址,否则存的第一个int16的低Byte会溢出堆栈*/
- wstk=(INT16U*)ptos;/*Loadstackpointer*/
- *wstk--=(INT16U)task;/*Returnaddress.Format:PCH:PCL*/
- *wstk--=(INT16U)task;/*Returnaddress.Format:PCH:PCL*/
- *wstk--=(INT16U)0x2222;/*YRegister*/
- *wstk--=(INT16U)0x1111;/*XRegister*/
- *wstk=(INT16U)p_arg;/*Simulatecalltofunctionwithargument(InDRegister)*/
- bstk=(INT8U*)wstk;/*ConvertWORDptrtoBYTEptrtosetCCR*/
- bstk--;
- *bstk=(0xC0);/*CCRRegister(DisableSTOPinstructionandXIRQ)*/
- return((OS_STK*)bstk);/*Returnpointertonewtop-of-stack*/
- }
其中有几点需要特别注意:
(1)68HC12 内核与 68HC11 内核一个大的区别就是 SP 指向的是实栈顶。老的68HC11的移植代码都是 *--wstk = XXXX。移植到68HC12 内核就要改为*wstk-- = XXXX。否则会浪费掉堆栈的前两个字节。
(2)先要执行 ptos--;否则第一个双字节会有一半溢出堆栈空间。
(3)任务的参数传递是通过寄存器 D 的,而不是堆栈。网上代码多数是:
- *wstk--=(INT16U)p_arg;
- *wstk--=(INT16U)task;
这样参数是传递不进来的,只有像我的代码中这样写才是正确的。
(4)代码中 *wstk-- = (INT16U)task; 重复了两遍,千万别以为这是我的笔误。堆栈中先存的(INT16U)task实际上是 task 函数的返回地址。虽然 μC/OS-II 要求任务不能返回,但是作为 C 语言的调用约定,在调用一个 C 函数之前要将 C 函数的返回地址先入栈。因此我将 task 的地址重复了两次,实际上第一的地址是什么都不重要,因为程序运行中觉得不会用到。甚至不要这行也行,还能节省堆栈中两个字节的空间。不过我还是选择了保留这行,使其看起来更加符合 C 语言的调用规范。
除此之外,OS_CPU_C.C 还包括一系列的 Hook 函数:
- #ifOS_CPU_HOOKS_EN>0&&OS_VERSION>203
- voidOSInitHookBegin(void)
- {
- #ifOS_TMR_EN>0
- OSTmrCtr=0;
- #endif
- }
- voidOSInitHookEnd(void)
- {
- }
- #endif
- #ifOS_CPU_HOOKS_EN>0
- voidOSTaskCreateHook(OS_TCB*ptcb)
- {
- #ifOS_APP_HOOKS_EN>0
- App_TaskCreateHook(ptcb);
- #else
- (void)ptcb;
- #endif
- }
- voidOSTaskDelHook(OS_TCB*ptcb)
- {
- #ifOS_APP_HOOKS_EN>0
- App_TaskDelHook(ptcb);
- #else
- (void)ptcb;
- #endif
- }
- voidOSTaskStatHook(void)
- {
- #ifOS_APP_HOOKS_EN>0
- App_TaskStatHook();
- #endif
- }
- voidOSTaskSwHook(void)
- {
- #ifOS_APP_HOOKS_EN>0
- App_TaskSwHook();
- #endif
- }
- #endif
- #ifOS_CPU_HOOKS_EN>0&&OS_VERSION>=251
- voidOSTaskIdleHook(void)
- {
- #ifOS_APP_HOOKS_EN>0
- App_TaskIdleHook();
- #endif
- }
- #endif
- #ifOS_CPU_HOOKS_EN>0&&OS_VERSION>203
- voidOSTCBInitHook(OS_TCB*ptcb)
- {
- #ifOS_APP_HOOKS_EN>0
- App_TCBInitHook(ptcb);
- #else
- (void)ptcb;/*Preventcompilerwarning*/
- #endif
- }
- #endif
- #ifOS_CPU_HOOKS_EN>0
- voidOSTimeTickHook(void)
- {
- #ifOS_APP_HOOKS_EN>0
- App_TimeTickHook();
- #endif
- #ifOS_TMR_EN>0
- OSTmrCtr++;
- if(OSTmrCtr>=(OS_TICKS_PER_SEC/OS_TMR_CFG_TICKS_PER_SEC))
- {
- OSTmrCtr=0;
- OSTmrSignal();
- }
- #endif
- }
- #endif
代码中 OS_APP_HOOKS_EN 和 OS_TMR_EN 在v2.52 版本中还没出现,我在这里这样写是为了移植到后面版本时更轻松。
至此,移植代码就基本完成了。不过这样还不能运行,因为两个中断处理函数(OSCtxSw和OSTickISR)还没有和对应的中断产生关联。将这二者关联起来的方法有几种,比如直接在 PRM 文件中制定,我用了种比较笨的办法,从网上找了个 vector.c 文件,虽然看起来不是很优雅,但确实是正确的代码。
- /*******************************************************************
- *
- *FreescaleMC9S12DP256ISRVectorDefinitions
- *
- *FileName:vectors.c
- *Version:1.0
- *Date:Jun/22/2004
- *Programmer:EricShufro
- ********************************************************************/
- /********************************************************************
- *EXTERNALISRFUNCTIONPROTOTYPES
- *********************************************************************/
- externvoidnear_Startup(void);/*StartupRoutine.*/
- externvoidnearOSTickISR(void);/*OSTimeTickRoutine.*/
- externvoidnearOSCtxSw(void);/*OSContectSwitchRoutine.*/
- externvoidnearSCI1_ISR(void);/*SCI1Routine.*/
- externvoidnearSCI0_ISR(void);/*SCI0Routine.*/
- /*
- ************************************************************************
- *DUMMYINTERRUPTSERVICEROUTINES
- *
- *Description:Whenaspuriousinterruptoccurs,theprocessorwill
- *jumptothededicateddefaulthandlerandstaythere
- *sothatthesourceinterruptmaybeidentifiedand
- *debugged.
- *
- *Notes:DoNotModify
- ************************************************************************
- */
- #pragmaCODE_SEG__NEAR_SEGNON_BANKED
- __interruptvoidsoftware_trap64(void){for(;;);}
- __interruptvoidsoftware_trap63(void){for(;;);}
- __interruptvoidsoftware_trap62(void){for(;;);}
- __interruptvoidsoftware_trap61(void){for(;;);}
- __interruptvoidsoftware_trap60(void){for(;;);}
- __interruptvoidsoftware_trap59(void){for(;;);}
- __interruptvoidsoftware_trap58(void){for(;;);}
- __interruptvoidsoftware_trap57(void){for(;;);}
- __interruptvoidsoftware_trap56(void){for(;;);}
- __interruptvoidsoftware_trap55(void){for(;;);}
- __interruptvoidsoftware_trap54(void){for(;;);}
- __interruptvoidsoftware_trap53(void){for(;;);}
- __interruptvoidsoftware_trap52(void){for(;;);}
- __interruptvoidsoftware_trap51(void){for(;;);}
- __interruptvoidsoftware_trap50(void){for(;;);}
- __interruptvoidsoftware_trap49(void){for(;;);}
- __interruptvoidsoftware_trap48(void){for(;;);}
- __interruptvoidsoftware_trap47(void){for(;;);}
- __interruptvoidsoftware_trap46(void){for(;;);}
- __interruptvoidsoftware_trap45(void){for(;;);}
- __interruptvoidsoftware_trap44(void){for(;;);}
- __interruptvoidsoftware_trap43(void){for(;;);}
- __interruptvoidsoftware_trap42(void){for(;;);}
- __interruptvoidsoftware_trap41(void){for(;;);}
- __interruptvoidsoftware_trap40(void){for(;;);}
- __interruptvoidsoftware_trap39(void){for(;;);}
- __interruptvoidsoftware_trap38(void){for(;;);}
- __interruptvoidsoftware_trap37(void){for(;;);}
- __interruptvoidsoftware_trap36(void){for(;;);}
- __interruptvoidsoftware_trap35(void){for(;;);}
- __interruptvoidsoftware_trap34(void){for(;;);}
- __interruptvoidsoftware_trap33(void){for(;;);}
- __interruptvoidsoftware_trap32(void){for(;;);}
- __interruptvoidsoftware_trap31(void){for(;;);}
- __interruptvoidsoftware_trap30(void){for(;;);}
- __interruptvoidsoftware_trap29(void){for(;;);}
- __interruptvoidsoftware_trap28(void){for(;;);}
- __interruptvoidsoftware_trap27(void){for(;;);}
- __interruptvoidsoftware_trap26(void){for(;;);}
- __interruptvoidsoftware_trap25(void){for(;;);}
- __interruptvoidsoftware_trap24(void){for(;;);}
- __interruptvoidsoftware_trap23(void){for(;;);}
- __interruptvoidsoftware_trap22(void){for(;;);}
- __interruptvoidsoftware_trap21(void){for(;;);}
- __interruptvoidsoftware_trap20(void){for(;;);}
- __interruptvoidsoftware_trap19(void){for(;;);}
- __interruptvoidsoftware_trap18(void){for(;;);}
- __interruptvoidsoftware_trap17(void){for(;;);}
- __interruptvoidsoftware_trap16(void){for(;;);}
- __interruptvoidsoftware_trap15(void){for(;;);}
- __interruptvoidsoftware_trap14(void){for(;;);}
- __interruptvoidsoftware_trap13(void){for(;;);}
- __interruptvoidsoftware_trap12(void){for(;;);}
- __interruptvoidsoftware_trap11(void){for(;;);}
- __interruptvoidsoftware_trap10(void){for(;;);}
- __interruptvoidsoftware_trap09(void){for(;;);}
- __interruptvoidsoftware_trap08(void){for(;;);}
- __interruptvoidsoftware_trap07(void){for(;;);}
- __interruptvoidsoftware_trap06(void){for(;;);}
- __interruptvoidsoftware_trap05(void){for(;;);}
- __interruptvoidsoftware_trap04(void){for(;;);}
- __interruptvoidsoftware_trap03(void){for(;;);}
- __interruptvoidsoftware_trap02(void){for(;;);}
- __interruptvoidsoftware_trap01(void){for(;;);}
- #pragmaCODE_SEGDEFAULT
- /***********************************************************************
- *INTERRUPTVECTORS
- ***********************************************************************/
- typedefvoid(*neartIsrFunc)(void);
- consttIsrFunc_vect[]@0xFF80={/*Interrupttable*/
- software_trap63,/*63RESERVED*/
- software_trap62,/*62RESERVED*/
- software_trap61,/*61RESERVED*/
- software_trap60,/*60RESERVED*/
- software_trap59,/*59RESERVED*/
- software_trap58,/*58RESERVED*/
- software_trap57,/*57PWMEmergencyShutdown*/
- software_trap56,/*56PortPInterrupt*/
- software_trap55,/*55CAN4transmit*/
- software_trap54,/*54CAN4receive*/
- software_trap53,/*53CAN4errors*/
- software_trap52,/*52CAN4wake-up*/
- software_trap51,/*51CAN3transmit*/
- software_trap50,/*50CAN3receive*/
- software_trap49,/*49CAN3errors*/
- software_trap48,/*48CAN3wake-up*/
- software_trap47,/*47CAN2transmit*/
- software_trap46,/*46CAN2receive*/
- software_trap45,/*45CAN2errors*/
- software_trap44,/*44CAN2wake-up*/
- software_trap43,/*43CAN1transmit*/
- software_trap42,/*42CAN1receive*/
- software_trap41,/*41CAN1errors*/
- software_trap40,/*40CAN1wake-up*/
- software_trap39,/*39CAN0transmit*/
- software_trap38,/*38CAN0receive*/
- software_trap37,/*37CAN0errors*/
- software_trap36,/*36CAN0wake-up*/
- software_trap35,/*35FLASH*/
- software_trap34,/*34EEPROM*/
- software_trap33,/*33SPI2*/
- software_trap32,/*32SPI1*/
- software_trap31,/*31IICBus*/
- software_trap30,/*30BDLC*/
- software_trap29,/*29CRGSelfClockMode*/
- software_trap28,/*28CRGPLLlock*/
- software_trap27,/*27PulseAccumulatorBOverflow*/
- software_trap26,/*26ModulusDownCounterunderflow*/
- software_trap25,/*25PortH*/
- software_trap24,/*24PortJ*/
- software_trap23,/*23ATD1*/
- software_trap22,/*22ATD0*/
- SCI1_ISR,/*21SC11*/
- SCI0_ISR,/*20SCI0*/
- software_trap19,/*19SPI0*/
- software_trap18,/*18Pulseaccumulatorinputedge*/
- software_trap17,/*17PulseaccumulatorAoverflow*/
- software_trap16,/*16EnhancedCaptureTimerOverflow*/
- software_trap15,/*15EnhancedCaptureTimerchannel7*/
- software_trap14,/*14EnhancedCaptureTimerchannel6*/
- software_trap13,/*13EnhancedCaptureTimerchannel5*/
- software_trap12,/*12EnhancedCaptureTimerchannel4*/
- software_trap11,/*11EnhancedCaptureTimerchannel3*/
- software_trap10,/*10EnhancedCaptureTimerchannel2*/
- software_trap09,/*09EnhancedCaptureTimerchannel1*/
- software_trap08,/*08EnhancedCaptureTimerchannel0*/
- OSTickISR,/*07RealTimeInterrupt*/
- software_trap06,/*06IRQ*/
- software_trap05,/*05XIRQ*/
- OSCtxSw,/*04SWI-BreakpointonHCS12SerialMon.*/
- software_trap03,/*03Unimplementedinstructiontrap*/
- software_trap02,/*02COPfailurereset*/
- software_trap01//,/*01Clockmonitorfailreset*/
- //_Startup/*00Resetvector*/
- };
后记:
当我完成全部移植工作并测试通过后,我又重新审视了一遍整个移植过程,发现走了许多弯路。这些弯路基本都是因为我对C编译器的特性,尤其是内联汇编的处理不熟悉造成的。比如中断处理函数,其实可以直接写到 OS_CPU_C.C 中。就可以省略了 vector.c 文件了。其实我一开始也是这样做的,但是最初的中断处理函数混合了C 语句和汇编语句,产生了各种莫名奇妙的错误。比如下面的RTI中断处理函数代码:
- interruptVectorNumber_VrtivoidOSTickISR(void)
- {
- OSIntNesting++;//4~,NotifyuC/OS-IIaboutISR
- if(OSIntNesting==1)
- {
- __asm
- {
- ldxOSTCBCur//3~,OSTCBCur->OSTCBStkPtr=StackPointer
- sts0,x//3~,}
- }
- }
- CRGFLG_RTIF=1;//clearinterruptflag.
- OSTimeTick();//6~+,CalluC/OS-IIstickupdatingfunction
- OSIntExit();//6~+,NotifyuC/OS-IIaboutendofISR
- }
对比后来的汇编代码,其实已经离成功很近了,只要将其中的C 语句全部用汇编写成来大功告成了:
- interruptVectorNumber_VrtivoidOSTickISR(void)
- {
- __asm
- {
- incOSIntNesting
- ldabOSIntNesting
- cmpb#$01
- bneOSTickISR1
- ldxOSTCBCur
- sts0,x
- OSTickISR1:
- BSET_CRGFLG,#128
- jsrOSTimeTick
- jsrOSIntExit
- }
- }
其他的代码也一样,都这样改写后就完全不需要 vector.c 文件了。但这里还是将这些本可以省略掉的代码保留下来了,是想记录下一条真实的探索路程。
加入微信
获取电子行业最新资讯
搜索微信公众号:EEPW
或用微信扫描左侧二维码