<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應用 > 初探WindowsCE異常和中斷服務程序

          初探WindowsCE異常和中斷服務程序

          作者: 時間:2012-10-31 來源:網(wǎng)絡(luò) 收藏

          PROLOG_END

          和上面一樣,的入口處都是例行公事的計算返回位置以抵消流水線誤差。再將要用到的寄存器壓入STACK_IRQ。

          ; Test interlocked API status.

          ;INTERLOCKED_START EQU USER_KPAGE 0x380

          ;INTERLOCKED_END EQU USER_KPAGE 0x400

          sub r0, lr, #INTERLOCKED_START

          cmp r0, #INTERLOCKED_END-INTERLOCKED_START

          bllo CheckInterlockedRestart

          上面這部分的內(nèi)容是關(guān)于互鎖的檢測,由于如信號量這些同步手段都必須作為原子操作進行,不允許打斷。所以如果發(fā)生在互鎖API的執(zhí)行過程中,就需要專門的處理了。這些API都是放在INTERLOCKED_START和INTERLOCKED_END之間的,通過LR很容易就檢查出是否是INTERLOCKEDXXX的過程中。這里并不關(guān)心互鎖的實現(xiàn)就繞開這部分代碼繼續(xù)往下看,當作沒有發(fā)生在interlock過程處理。

          ;

          ; CAREFUL! The stack frame is being altered here. It's ok since

          ; the only routine relying on this was the Interlock Check. Note that

          ; we re-push LR onto the stack so that the incoming argument area to

          ; OEMInterruptHandler will be correct.

          ;

          mrs r1, spsr ; (r1) = saved status reg

          stmfd sp!, {r1} ; save SPSR onto the IRQ stack

          mov r0,lr ; parAMEter to OEMInterruptHandler

          msr cpsr_c, #SVC_MODE:OR:0x80 ; switch to supervisor mode w/IRQs disabled

          stmfd sp!, {lr} ; save LR onto the SVC stack

          stmfd sp!, {r0} ; save IRQ LR (in R0) onto the SVC stack (param)

          ;

          ; Now we call the OEM's interrupt handler code. It is up to them to

          ; enable interrupts if they so desire. We can't do it for them since

          ; there's only on interrupt and they haven't yet defined their nesting.

          ;

          CALL OEMInterruptHandler

          ldmfd sp!, {r1} ; dummy pop (parameter)

          ldmfd sp!, {lr} ; restore SVC LR from the SVC stack

          msr cpsr_c, #IRQ_MODE:OR:0x80 ; switch back to IRQ mode w/IRQs disabled

          ; Restore the saved program status register from the stack.

          ;

          ldmfd sp!, {r1} ; restore IRQ SPSR from the IRQ stack

          msr spsr, r1 ; (r1) = saved status reg

          ldr lr, =KData ; (lr) = ptr to KDataStruct

          cmp r0, #SYSINTR_RESCHED ;->時間片已到,進行調(diào)度

          beq

          %B20

          ; interrupted, reschedule again

          msr  spsr, r2

          ldr  lr, [r0, #TcxPc-TcxR3]

          ldmdb  r0,

          movs  pc,

          lr

          ; return to user or system mode

          HandleException是實際進行處理的函數(shù),針對上面沒有處理完的進一步分析并進行處理。這個函數(shù)是沒有公開代碼的,所以沒有辦法進一步深入下去。由于處理的類型比較多所以這個異常處理函數(shù)的代碼量是相當大的,因此會耗費相對比較多的時鐘周期,在之前的代碼中我們都是在關(guān)閉的情況下進行異常處理,如果在這里還不打開中斷的話整個異常處理過程會相當?shù)拈L,這樣會很大程度上影響系統(tǒng)的實時性,所以在這里調(diào)用HandleException之前是將中斷重新打開的,待到處理完成再將中斷關(guān)閉。對于這些異常,如果不能處理就只有兩種情況:1.結(jié)束該進程/線程。2.掛起系統(tǒng)。第二種情況下掛起系統(tǒng)HandleException是不會返回的。因此,只有異常處理正常流程和結(jié)束線程的可能。對于返回的情況,這個時候如果返回觸發(fā)異常的地址繼續(xù)運行的話,仍然會導致異常,所以結(jié)束進程/線程都需要重新調(diào)度才能完成了。對于異常處理成功的情形,就不必調(diào)度了,直接就可以返回產(chǎn)生異常的地方繼續(xù)執(zhí)行。在這里還要考慮套嵌(這里僅僅是指系統(tǒng)模式和兼管模式的異常套嵌)的情形,也就是中斷/異常已經(jīng)進入調(diào)度狀態(tài)又再次產(chǎn)生中斷/異常,這個時候就強行取消上一次調(diào)度,進而重新調(diào)度。這用于調(diào)度過程中遇到異常恢復和剝奪的情況,如果不屬于這種情況的話就直接恢復寄存器狀態(tài)并且返回中斷點繼續(xù)執(zhí)行。

          ; Return to a non-preemptible privileged mode.

          ;

          ;   (r0) = ptr to THREAD structure

          ;   (r2) = target mode

          30  msr  cpsr,

          r2

          ; switch to target mode

          add  r0, r0, #TcxR0

          ldmia  r0,

          ; reload all registers return

          通過HandleException處理以后,已經(jīng)完成了所有異常的處理,所以這里只是考慮反回的情況,由于這里不包含用戶模式下的處理,所以這里處理的都是特權(quán)模式,完全可以訪問kdata區(qū)域,這里就直接利用Kdata區(qū)域中的線程備份來完成恢復寄存器和返回。


          上一頁 1 2 下一頁

          關(guān)鍵詞: 服務 程序 中斷 異常 WindowsCE 初探

          評論


          相關(guān)推薦

          技術(shù)專區(qū)

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();