macro restore_user_regs
ldr r1,[sp, #S_PSR]
ldr lr,[sp, #S_PC]! @ !用來控制基址變址尋址的最終新地址是否進行回寫操作,
@ 執(zhí)行l(wèi)dr之后sp被回寫成sp+#S_PC基址變址尋址的新地址
msr spsr,r1 @ 把cpsr的值保存到spsr中
ldmdb sp,{r0 - lr}^ @ lr=[sp-1*4],r13=[sp-2*4],r12=[sp-3*4],......,r0=[sp-15*4]
@ 因為沒對pc賦值,所以^的表示將數(shù)據(jù)恢復到User模式的[r0-lr]寄存器組中[gliethttp]
mov r0,r0
add sp,sp,#S_FRAME_SIZE - S_PC
movs pc,lr
.endm
其他指令正在學習中[隨時補充gliethttp]
-----------------------------
1.ldr ip,[sp],#4 將sp中內(nèi)容存入ip,之后sp=sp+4;
ldr ip,[sp,#4] 將sp+4這個新地址下內(nèi)容存入ip,之后sp值保持不變
ldr ip,[sp,#4]!將sp+4這個新地址下內(nèi)容存入ip,之后sp=sp+4將新地址值賦給sp
str ip,[sp],#4 將ip存入sp地址處,之后sp=sp+4;
str ip,[sp,#4] 將ip存入sp+4這個新地址,之后sp值保持不變
str ip,[sp,#4]!將ip存入sp+4這個新地址,之后sp=sp+4將新地址值賦給sp
-----------------------------
2.movs r1,#3 ;movs將導致ALU被更改,因為r1賦值非0,即操作結果r0非0,所以ALU的Z標志清0
bne 1f ;因為Z=0,說明不等,所以向前跳到標號1:所在處繼續(xù)執(zhí)行其他語句
-----------------------------
3.LDM表示裝載,STM表示存儲.
LDMED LDMIB 預先增加裝載
LDMFD LDMIA 過后增加裝載
LDMEA LDMDB 預先減少裝載
LDMFA LDMDA 過后減少裝載
STMFA STMIB 預先增加存儲
STMEA STMIA 過后增加存儲
STMFD STMDB 預先減少存儲
STMED STMDA 過后減少存儲
注意ED不同于IB;只對于預先減少裝是相同的.在存儲的時候,ED是過后減少的.
FD、ED、FA、和 EA 指定是滿棧還是空棧,是升序棧還是降序棧.
對于存儲STM而言
先加后存 FA 姑且這么來記,先加(first add),存數(shù)據(jù)
后加先存 EA 姑且這么來記,存數(shù)據(jù),后加end add
先減后存 FD 姑且這么來記,先減first dec,存數(shù)據(jù)
后減先存 ED 姑且這么來記,存數(shù)據(jù),后減end dec
然后記憶LDM,LDM是STM的反相彈出動作,所以
因為是先加后存,所以后減先取 FA 就成了與STM對應的取數(shù)據(jù),后減
因為是后加先存,所以先減后取 EA 就成了與STM對應的先減,取數(shù)據(jù)
因為是先減后存,所以后加先取 FD 就成了與STM對應的取數(shù)據(jù),后加
因為是后減先存,所以先加后取 ED 就成了與STM對應的先加,取數(shù)據(jù)
我想通過上面的變態(tài)方式可以比較容易的記住這套指令[gliethttp]
一個滿棧的棧指針指向上次寫的最后一個數(shù)據(jù)單元,而空棧的棧指針指向第一個空閑單元.
一個降序棧是在內(nèi)存中反向增長(就是說,從應用程序空間結束處開始反向增長)而升序棧在內(nèi)存中正向增長.
其他形式簡單的描述指令的行為,意思分別是
IA過后增加(Increment After)、
IB預先增加(Increment Before)、
DA過后減少(Decrement After)、
DB預先減少(Decrement Before).
RISC OS使用傳統(tǒng)的滿降序棧.在使用符合APCS規(guī)定的編譯器的時候,它通常把你的棧指針設置在應用程序空間的
結束處并接著使用一個FD(滿降序-Full Descending)棧.如果你與一個高級語言(BASIC或C)一起工作,你將別無選擇.
棧指針(傳統(tǒng)上是R13)指向一個滿降序棧.你必須繼續(xù)這個格式,或則建立并管理你自己的棧.
4.teq r1,#0 //r1-0,將結果送入狀態(tài)標志,如果r1和0相減的結果為0,那么ALU的Z置位,否則Z清0
bne reschedule//ne表示Z非0,即:不等,那么執(zhí)行reschedule函數(shù)
-----------------------------
5.使用tst來檢查是否設置了特定的位
tst r1,#0x80 //按位and操作,檢測r1的0x1<<7,即第7位是否置1,按位與之后結果為0,那么ALU的Z置位
beq reset //如果Z置位,即:以上按位與操作結果是0,那么跳轉到reset標號執(zhí)行
-----------------------------
6.'^'的理解
'^'是一個后綴標志,不能在User模式和Sys系統(tǒng)模式下使用該標志.該標志有兩個存在目的:
6.1.對于LDM操作,同時恢復的寄存器中含有pc(r15)寄存器,那么指令執(zhí)行的同時cpu自動將spsr拷貝到cpsr中
如:在IRQ中斷返回代碼中[如下為ads環(huán)境下的代碼gliethttp]
ldmfd {r4} //讀取sp中保存的的spsr值到r4中
msr spsr_cxsf,r4 //對spsr的所有控制為進行寫操作,將r4的值全部注入spsr
ldmfd {r0-r12,lr,pc}^//當指令執(zhí)行完畢,pc跳轉之前,將spsr的值自動拷貝到cpsr中[gliethttp]
6.2.數(shù)據(jù)的送入、送出發(fā)生在User用戶模式下的寄存器,而非當前模式寄存器
如:ldmdb sp,{r0 - lr}^;表示sp棧中的數(shù)據(jù)回復到User分組寄存器r0-lr中,而不是恢復到當前模式寄存器r0-lr 當然對于User,System,IRQ,SVC,Abort,Undefined這6種模式來說[gliethttp]r0-r12是共用的,只是r13和r14
為分別獨有,對于FIQ模式,僅僅r0-r7是和前6中模式的r0-r7共用,r8-r14都是FIQ模式下專有.
7.spsr_cxsf,cpsr_cxsf的理解
c - control field mask byte(PSR[7:0])
x - extension field mask byte(PSR[15:8])
s - status field mask byte(PSR[23:16)
f - flags field mask byte(PSR[31:24]).
老式聲明方式:cpsr_flg,cpsr_all在ADS中已經(jīng)不在支持
cpsr_flg對應cpsr_f
cpsr_all對應cpsr_cxsf
需要使用專用指令對cpsr和spsr操作:mrs,msr
mrs tmp,cpsr //讀取CPSR的值
bic tmp,tmp,#0x80 //如果第7位為1,將其清0
msr cpsr_c,tmp //對控制位區(qū)psr[7:0]進行寫操作
-----------------------------
8.cpsr的理解
CPSR = Current Program Status Register
SPSR = Saved Program Status Registers
CPSR寄存器(和保存它的SPSR寄存器)
N,Z,C,V稱為ALU狀態(tài)標志
N:如果結果是負數(shù)則置位
Z:如果結果是零則置位
C:如果發(fā)生進位則置位
V:如果發(fā)生溢出則置位
I:置位表示禁用IRQ中斷,清0表示使能IRQ
F:置位表示禁用FIQ中斷,清0表示使能FIQ
T:置位表示系統(tǒng)運行在Thumb態(tài),清0表示運行在ARM態(tài)
M[4:0]:
10000 User模式,和System系統(tǒng)模式一樣
10001 FIQ模式
10010 IRQ模式
10011 SVC超級管理模式
10111 Abort數(shù)據(jù)異常模式
11011 Undefined未定義指令模式
11111 System系統(tǒng)模式,和User模式一樣
舉例:
ands r2,r2,#7 使用運算結果改變標志位,如果運算結果r2=0,那么Z置位,EQ相等判斷成立
subs r2,r2,#1 使用運算結果改變標志位,如果運算結果r2=0,那么Z置位,EQ相等判斷成立
beq wordcopy
EQ : 等于
NE : 不等
CS : 無符號>=
CC : 無符號<
MI : 負數(shù)
PL : 非負[>=0]
VS : 溢出
VC : 無溢出
HI : 無符號>
LS : 無符號<=
GE : 有符號>=
LT : 有符號<
GT : 有符號>
LE : 有符號<=
AL : 總是[默認]