新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > ARM微處理器的編程模型之:異常中斷處理

ARM微處理器的編程模型之:異常中斷處理

作者: 時(shí)間:2013-09-13 來(lái)源:網(wǎng)絡(luò) 收藏

本文引用地址:http://butianyuan.cn/article/257085.htm

4.在特權(quán)模式下使用SWI異常處理

在特權(quán)模式下使用SWI異常處理,和IRQ/FIQ中斷嵌套基本類似。當(dāng)執(zhí)行SWI指令后,處理器執(zhí)行下面操作。

① 處理器進(jìn)入特權(quán)模式。

② 將程序狀態(tài)字內(nèi)容CPSR保存到SPSR_svc。

③ 返回地址放入LR_svc。

如果處理器已經(jīng)處于特權(quán)模式,再發(fā)生SWI異常,則LR_svc和SPSR_svc寄存器的值將丟失。

所以在特權(quán)模式下,調(diào)用SWI軟中斷異常,必須先將LR_svc和SPSR_svc寄存器的值壓棧保護(hù)。下面的例子顯示了一個(gè)可以在特權(quán)模式下調(diào)用的SWI處理函數(shù)。

AREA SWI_Area, CODE, READONLY

PRESERVE8

EXPORT SWI_Handler

IMPORT C_SWI_Handler

T_bit EQU 0x20

SWI_Handler

STMFD sp!,{r0-r3,r12,lr} ;寄存器壓棧保護(hù)

MOV r1, sp ;堆棧指針?lè)舝1作為參數(shù)傳遞.

MRS r0, spsr ;讀取spsr.

STMFD sp!, {r0, r3} ;將spsr壓棧保護(hù)

;

;

LDR r0,[lr,#-4] ;計(jì)算SWI指令地址.

BIC r0,r0,#0xFF000000 ;讀取SWI中斷向量號(hào).

; r0存放中斷向量號(hào)

; r1 堆棧指針

BL C_SWI_Handler ;調(diào)用C程序的SWI處理函數(shù).

LDMFD sp!, {r0, r3} ;從堆棧中讀取spsr.

MSR spsr_cf, r0 ;恢復(fù)spcr

LDMFD sp!, {r0-r3,r12,pc}^ ;恢復(fù)其他寄存器并返回.

END

5.從應(yīng)用程序中調(diào)用SWI

可從匯編語(yǔ)言或 C/C++ 中調(diào)用 SWI。

(1)從匯編應(yīng)用程序中調(diào)用SWI

從匯編語(yǔ)言程序中調(diào)用SWI,只要遵循AAPCS標(biāo)準(zhǔn)即可。調(diào)用前,設(shè)定所有必須的值并發(fā)出相關(guān)的 SWI。例如:

MOV r0, #65 ; 將軟中斷的子功能號(hào)放到r0中

SWI 0x0

注意

SWI指令和其他所以指令一樣,可以被條件執(zhí)行。

(2)從C應(yīng)用程序中調(diào)用SWI

在C或C++應(yīng)用程序中調(diào)用SWI,要將C語(yǔ)言的子程序用編譯器擴(kuò)展_swi聲明,例如:

__swi(0) void my_swi(int);

……

……

……

my_swi(65);

編譯器擴(kuò)展_swi確保了SWI以內(nèi)聯(lián)方式進(jìn)行編譯,而沒(méi)有額外的開(kāi)銷。但有如下的AAPCS限制。

· 函數(shù)調(diào)用參數(shù)只能使用r0~r3傳遞。

· 函數(shù)返回值只能通過(guò)r0~r3傳遞。

向內(nèi)聯(lián)的SWI函數(shù)傳遞參數(shù)和向?qū)嶋H的子函數(shù)傳遞參數(shù)基本類似。但返回值的情況比較復(fù)雜。如果有兩到四個(gè)返回值,則必須告訴編譯程序返回值是以結(jié)構(gòu)形式返回的,并使用__value_in_regs 偽操作聲明。這是因?yàn)榛诮Y(jié)構(gòu)值的函數(shù)通常被處理為一個(gè)void(空)型函數(shù),且第一個(gè)自變量必須為存放結(jié)果結(jié)構(gòu)的地址。

下面的例子顯示了對(duì)編號(hào)為0x0、0x1、0x2和0x3的SWI軟中斷的調(diào)用。其中,SWI0x0和SWI0x1傳遞兩個(gè)整型參數(shù)并返回一個(gè)單一結(jié)果;SWI0x2傳遞4個(gè)參數(shù)并返回一個(gè)單一結(jié)果;而SWI0x3傳遞4個(gè)參數(shù)并通過(guò)結(jié)構(gòu)體返回4個(gè)結(jié)果。

#include stdio.h>

#include swi.h

unsigned *swi_vec = (unsigned *)0x08;

extern void SWI_Handler(void);

int main( void )

{

int result1, result2;

struct four_results res_3;

Install_Handler( (unsigned) SWI_Handler, swi_vec );

printf(result1 = multiply_two(2,4) = %dn, result1 = multiply_two(2,4));

printf(result2 = multiply_two(3,6) = %dn, result2 = multiply_two(3,6));

printf(add_two( result1, result2 ) = %dn, add_two( result1, result2 ));

printf(add_multiply_two(2,4,3,6) = %dn, add_multiply_two(2,4,3,6));

res_3 = many_operations( 12, 4, 3, 1 );

printf(res_3.a = %dn, res_3.a );

printf(res_3.b = %dn, res_3.b );

printf(res_3.c = %dn, res_3.c );

printf(res_3.d = %dn, res_3.d );

return 0;

}

__swi(0) int multiply_two(int, int);

__swi(1) int add_two(int, int);

__swi(2) int add_multiply_two(int, int, int, int);

struct four_results

{

int a;

int b;

int c;

int d;

};

__swi(3) __value_in_regs struct four_results many_operations(int, int, int, int);

(3)應(yīng)用程序中動(dòng)態(tài)調(diào)用SWI

在某些情形下,需要調(diào)用直到運(yùn)行時(shí)才會(huì)知道其編號(hào)的 SWI。例如,當(dāng)有很多相關(guān)操作可在同一目標(biāo)上執(zhí)行,并且每一個(gè)操作都有其自己的 SWI 時(shí),就會(huì)發(fā)生這種情況。在此情況下,上一小節(jié)的方法不適用。

解決的方法有兩種。

· 在運(yùn)行時(shí)得到SWI功能號(hào),然后構(gòu)造出相應(yīng)的SWI指令的編碼,將該編碼保存在某個(gè)存儲(chǔ)單元中,將PC指針指向該單元,執(zhí)行指令。

· 使用一個(gè)通用的SWI程序,將運(yùn)行時(shí)需要調(diào)用的SWI功能號(hào)作為參數(shù)傳遞給該通用的SWI異常處理程序,通用的SWI程序根據(jù)參數(shù)值調(diào)用相應(yīng)的SWI處理程序完成需要的操作。

通過(guò)匯編語(yǔ)言可以實(shí)現(xiàn)第二種解決辦法:通過(guò)寄存器(通常為r0或r12)傳遞所需要的操作數(shù),這樣可以重新編寫(xiě)SWI處理程序,對(duì)相應(yīng)寄存器中的值進(jìn)行處理。

但有些情況下,為了節(jié)省程序開(kāi)銷,需要直接使用SWI中斷號(hào)對(duì)程序調(diào)用。例如,操作系統(tǒng)可能會(huì)使用單一的一條SWI指令并用寄存器來(lái)傳遞所需運(yùn)算的編號(hào)。這使得其他SWI空間可用于特定應(yīng)用程序的SWI。在一個(gè)特定的應(yīng)用程序中,如果從指令中提取SWI編號(hào)的開(kāi)銷太大,就可使用這個(gè)方法。(0x123456)和Thumb(0xAB)半主機(jī)方式的SWI就是這樣實(shí)現(xiàn)的。



評(píng)論


相關(guān)推薦

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

關(guān)閉