ARM指令集详解(超详细!带实例!)

上传人:小** 文档编号:39194674 上传时间:2021-11-10 格式:DOC 页数:27 大小:315KB
收藏 版权申诉 举报 下载
ARM指令集详解(超详细!带实例!)_第1页
第1页 / 共27页
ARM指令集详解(超详细!带实例!)_第2页
第2页 / 共27页
ARM指令集详解(超详细!带实例!)_第3页
第3页 / 共27页
资源描述:

《ARM指令集详解(超详细!带实例!)》由会员分享,可在线阅读,更多相关《ARM指令集详解(超详细!带实例!)(27页珍藏版)》请在装配图网上搜索。

1、算术和逻辑指令ADC : 带进位的加法(Ad dition with Carry)ADC条件S , , dest = op_1 + op_2 + carryADC将把两个操作数加起来,并把结果放置到目的寄存器中。它使用一 个进位标志位,这样就可以做比 32 位大的加法。下列例子将加两个 128 位的数。128 位结果 : 寄存器 0、 1、2、和 3第一个128 位数 : 寄存器 4 、5、6、和 7第二个128 位数 : 寄存器 8 、9、10 、和 11 。ADDSR0, R4, R8 ;加低端的字ADCSR1, R5, R9 ;加下一个字,带进位ADCSR2, R6, R10 ;加第三个

2、字,带进位ADCSR3, R7, R11 ;加高端的字,带进位如果如果要做这样的加法,不要忘记设置 S 后缀来更改进位标志。ADD : 加法(Add ition)ADD条件 S , , dest = op_1 + op_2ADD将把两个操作数加起来,把结果放置到目的寄存器中。 操作数1是一个寄存器,操作数 2 可以是一个寄存器,被移位的寄存器,或一个立即值 :ADD R0, R1, R2; R0 = R1 + R2ADD R0, R1, #256; R0 = R1 + 256ADD R0, R2, R3,LSL#1; R0 = R2 + (R3 1)加法可以在有符号和无符号数上进行。AND :

3、 逻辑与(logical AND)AND条件S , , dest = op_1 AND op_2AND将在两个操作数上进行逻辑与,把结果放置到目的寄存器中;对屏 蔽你要在上面工作的位很有用。 操作数 1 是一个寄存器,操作数 2 可 以是一个寄存器,被移位的寄存器,或一个立即值 :AND RO, RO, #3; RO =保持RO的位 0和1,丢弃其余的位。AND 的真值表 (二者都是 1 则结果为 1):Op_1 Op_2 结果O O OO 1 O1 O O1 1 1BIC : 位清除(Bit Clear)BIC 条件 S , , dest = op_1 AND (!op_2)BIC 是在一个

4、字中清除位的一种方法, 与 OR 位设置是相反的操作。 操 作数 2 是一个 32 位位掩码 (mask) 。如果如果在掩码中设置了某一位, 则清除这一位。未设置的掩码位指示此位保持不变。BIC R0, R0, #%1011; 清除 R0 中的位 0 、1、和 3 。保持其余的不变。BIC 真值表 :Op_1 Op_2 结果0 0 00 1 01 0 11 1 0译注:逻辑表达式为 Op_1 AND NOT Op_2EOR : 逻辑异或(logical Exclusive OR)EOR条件S , , dest = op_1 EOR op_2EOR将在两个操作数上进行逻辑异或,把结果放置到目的寄

5、存器中;对 反转特定的位有用。 操作数 1 是一个寄存器, 操作数 2 可以是一个寄 存器,被移位的寄存器,或一个立即值 :EOR R0, R0, #3;反转 R0 中的位 0 和 1EOR 真值表 (二者不同则结果为 1):Op_1 Op_2 结果0 0 00 1 11 0 11 1 0MOV : 传送(Mov e)MOV条 件S , dest = op_1MOV从另一个寄存器、被移位的寄存器、或一个立即值装载一个值到目的寄存器。你可以指定相同的寄存器来实现 NOP 指令的效果, 你还可 以专门移位一个寄存器 :MOV R0, R0; R0 = R0. NOP指令MOV R0, R0, LS

6、L#3; R0 = R0 * 8如果 R15 是目的寄存器,将修改程序计数器或标志。这用于返回到调 用代码,方法是把连接寄存器的内容传送到 R15:MOV PC, R14;退出到调用者MOVS PC, R14;退出到调用者并恢复标志位( 不遵从 32-bit 体系 )MVN : 传送取反的值(Move N egative)MVN条 件S , dest = !op_1MVN从另一个寄存器、被移位的寄存器、或一个立即值装载一个值到目的寄存器。不同之处是在传送之前位被反转了,所以把一个被取反的值 传送到一个寄存器中。这是逻辑非操作而不是算术操作,这个取反的值 加 1 才是它的取负的值 :MVN R0

7、, #4; R0 = -5MVN R0, #0; R0 = -1ORR : 逻辑或(logical OR)ORR 条件S , , dest = op_1 OR op_2OR将在两个操作数上进行逻辑或, 把结果放置到目的寄存器中; 对设置 特定的位有用。操作数 1 是一个寄存器,操作数 2 可以是一个寄存器, 被移位的寄存器,或一个立即值 :ORR R0, R0, #3;设置 R0 中位 0 和 1OR 真值表 (二者中存在 1 则结果为 1):Op_1 Op_2 结果0 0 00 1 11 0 11 1 1RSB : 反向减法(Reverse Subtraction)RSB条 件S , , d

8、est = op_2 - op_1SUB用操作数two减去操作数one ,把结果放置到目的寄存器中。操作数 1 是一个寄存器,操作数 2 可以是一个寄存器,被移位的寄存器,或一个立即值 :RSB R0, R1, R2; R0 = R2 - R1RSB R0, R1, #256; R0 = 256 - R1RSB R0, R2, R3,LSL#1 ; R0 = (R3 1) - R2反向减法可以在有符号或无符号数上进行。RSC : 带借位的反向减法(Reverse Subtraction with Carry)RSC条件S , , dest = op_2 - op_1 - !carry同于SBC

9、但倒换了两个操作数的前后位置。SBC : 带借位的减法(Subtraction with Carry)SBC条件S , , dest = op_1 - op_2 - !carrySBC做两个操作数的减法,把结果放置到目的寄存器中。它使用进位标 志来表示借位,这样就可以做大于32位的减法。SUB和SBC生成进位标志的方式不同于常规,如果需要借位则 清除进位标志。所以,指令要 对进位标志进行一个 非操作 - 在指令执行期间自动的反转此位。SUB : 减法(Sub traction)SUB条件 S , , dest = op_1 - op_2SUB用操作数one减去操作数two,把结果放置到目的寄存

10、器中。操作数 1 是一个寄存器,操作数 2 可以是一个寄存器,被移位的寄存器, 或一个立即值 :SUB R0, R1, R2; R0 = R1 - R2SUB R0, R1, #256; R0 = R1 - 256SUB R0, R2, R3,LSL#1; R0 = R2 - (R3 1)减法可以在有符号和无符号数上进行。移位指令ARM 处理器组建了可以与数据处理指令 (ADC 、ADD 、AND 、BIC 、 CMN 、CMP 、EOR 、MOV 、MVN、ORR、RSB、SBC、SUB、TEQ、 TST) 一起使用的桶式移位器(barrel shifter)。你还可以使用桶式移位器 影响在

11、 LDR/STR 操作中的变址值。译注:移位操作在 ARM 指令集中不作为单独的指令使用, 它是指令格 式中是一个字段, 在汇编语言中表示为指令中的选项。 如果数据处理指 令的第二个操作数或者单一数据传送指令中的变址是寄存器, 则可以对 它进行各种移位操作。如果数据处理指令的第二个操作数是立即值,在 指令中用 8 位立即值和 4 位循环移位来表示它,所以对大于 255 的 立即值, 汇编器尝试通过在指令中设置循环移位数量来表示它,如果不 能表示则生成一个错误。 在逻辑类指令中, 逻辑运算指令由指令中 S 位 的设置或清除来确定是否影响进位标志,而比较指令的 S 位总是设置 的。在单一数据传送指

12、令中指定移位的数量只能用立即值而不能用寄存面是给不同的移位类型的六个助记符LSL 逻辑左移ASL LSR ASR RORRRX算术左移逻辑右移算术右移循环右移 带扩展的循环右移ASL和LSL是等同的,可以自由互换。你可以用一个立即值 (从 0 到 31) 指定移位数量, 或用包含在 0 和 31 之间的一个值的寄存器指定移位数量。逻辑或算术左移(Logical or Arithmetic Shift Left)Rx, LSL #n orRx, ASL #n orRx, LSL Rn orRx, ASL Rn接受Rx的内容并按用或在寄存器Rn中指定的数量向高有效位方 向移位。最低有效位用零来填充

13、。除了概念上的第 33 位(就是被移出 的最小的那位 )之外丢弃移出最左端的高位, 如果逻辑类指令中 S 位被 设置了,则此位将成为从桶式移位器退出时进位标志的值。考虑下列 :MOV R1, #12MOV R0, R1, LSL#2在退出时,R0是48。这些指令形成的总和是 R0 = #12, LSL#2等同于BASIC 的 R0 = 12 shift。算术右移(Arithmetic Shift Right)Rx, ASR #n orRx, ASR Rn类似于LSR,但使用要被移位的寄存器(Rx)的第31位的值来填充高 位,用来保护补码表示中的符号。如果逻辑类指令中 S 位被设置了, 则把最后

14、被移出最右端的那位放置到进位标志中。它同于 BASIC 的 register = value shift 。循环右移(Rotate Right)Rx, ROR #n orRx, ROR Rn 循环右移类似于逻辑右移,但是把从右侧移出去的位放置到左侧, 如果 逻辑类指令中 S 位被设置了,则同时放置到进位标志中,这就是位的 循环。一个移位量为 32 的操作将导致输出与输入完全一致,因为所 有位都被移位了 32 个位置,又回到了开始时的位置 !带扩展的循环右移( R otate Right with e xtend)Rx, RRX这是一个 ROR#0 操作,它向右移动一个位置 - 不同之处是,它使

15、用 处理器的进位标志来提供一个要被移位的 33 位的数量。乘法指令指令格式这两个指令与普通算术指令在对操作数的限制上有所不同1. 给出的所有操作数、和目的寄存器必须为简单的寄存器。2. 你不能对操作数 2 使用立即值或被移位的寄存器。3. 目的寄存器和操作数 1 必须是不同的寄存器。4. 最后,你不能指定 R15 为目的寄存器。MLA : 带累加的乘法(Multip lication with Accumulate)MLA条件S , , , dest = (op_1 * op_2) + op_3MLA的行为同于MUL但它把操作数 3的值加到结果上。这在求总和时有用。MUL : 乘法(Mul t

16、iplication)MUL条 件S , , MUL提供32位整数乘法。如果操作数是有符号的,可以假定结果也是有符号的。比较指令指令格式译注: CMP 和 CMP 是算术指令, TEQ 和 TST 是逻辑指令。把它 们归入一类的原因是它们的 S 位总是设置的,就是说,它们总是影响 标志位。CMN : 比较取负的值(Compare Negative)CMN条 件P , status = op_1 - (- op_2)CMN同于CMP但它允许你与小负值(操作数2的取负的值)进行比较, 比如难于用其他方法实现的用于结束列表的-1。这样与 -1 比较将使用:CMN R0, #1;把 R0 与 -1 进

17、行比较详情参照CMP指令。CMP : 比较(Compare)CMP条 件P , status = op_1 - op_2CMP允许把一个寄存器的内容如另一个寄存器的内容或立即值进行比较,更改状态标志来允许进行条件执行。它进行一次减法,但不存储结 果,而是正确的更改标志。 标志表示的是操作数 1 比操作数 2 如何(大 小等)。如果操作数 1 大于操作操作数 2,则此后的有 GT 后缀的指 令将可以执行。明显的,你不需要显式的指定 S后缀来更改状态标志 如果你指定了 它则被忽略。TEQ : 测试等价(Test Eq uivalence)TEQ条件P vop 1, Status = op_1 EO

18、R op_2TEQ类似于TST。区别是这里的概念上的计算是EOR而不是AND。这提供了一种查看两个操作数是否相同而又不影响进位标志(不象CMP那样)的方法。加上P后缀的TEQ还可用于改变 R15中的标志(在26-bit 模式中 )。详情请参照 psr.html ,在 32-bit 模式下如何做请参见这里。TST : 测试位(Test bits)TST条 件P , Status = op_1 AND op_2TST类似于CMP不产生放置到目的寄存器中的结果。 而是在给出的两个 操作数上进行操作并把结果反映到状态标志上。使用TST来检查是否设置了特定的位。操作数 1 是要测试的数据字而操作数 2

19、是一个位掩 码。经过测试后,如果匹配则设置 Zero标志,否则清除它。象CMP那 样,你不需要指定 S 后缀。TST R0, #%1;测试在 R0 中是否设置了位 0 。B : 分支( Branch)B条件 地址B是最简单的分支。一旦遇到一个 B指令,ARM处理器将立即跳转到给定的地 址,从那里继续执行。注意存储在分支指令中的实际的值是相对当前的 R15 的值的一个偏移量;而不 是一个绝对地址。它的值由汇编器来计算, 它是 24 位有符号数,左移两位后有符号扩展为 32 位, 表示的有效偏移为 26 位(+/- 32 M) 。在其他处理器上,你可能经常见到这样的指令 :OPT 1LDA &70

20、CMP #0BEQ ZeroSTA &72.Zero RTS( 取自 Acorn Electron User Guide issue 1 page 213)在 ARM 处理器上,它们将变成下面这些东西 :OPT 1ADR R1, #&70LDR R0, R1CMP #0BEQ ZeroSTR R0, R1, #2.ZeroMOV PC, R14这不是一个很好的例子, 但你可以构想如何更好的去条件执行而不是分支。 另一 方面,如果你有大段的代码或者你的代码使用状态标志, 那么你可以使用条件执 行来实现各类分支 : 这样一个单一的简单条件执行指令可以替代在其他处理器 中存在的所有这些分支和跳转指令

21、。OPT 1ADR R1, #&70LDR R0, R1CMP R0, #0STRNE R0, R1, #2MOV PC, R14BL : 带连接的分支( Branch with Link)BL条件 地址BL 是另一个分支指令。 就在分支之前, 在寄存器 14 中装载上 R15 的内容。 你 可以重新装载 R14 到 R15 中来返回到在这个分支之后的那个指令, 它是子例程的一个基本但强力的实现。它的作用在屏幕装载器 2 ( 例子 4) 中得 以很好的展现 .load_new_formatBLswitch_screen_modeBLget_screen_infoBLload_palettene

22、w_loopMOVR1, R5BLread_byteCMPR0, #255BLEQ read_loopSTRB R0, R2, #1!. 在这里我们见到在装载器循环之前调用了三个子例程。接着,一旦满足了条 件执行就在循环中调用了 read_byte 子例程。条件执行ARM处理器的一个非常特殊的特征是它的条件执行。 我们指的不是基本的如果进位则分支,ARM使这个逻 辑阶段进一步深化为如果进位则 XXX - 这里的 XXX 是任何东西。为了举例,下面是 Intel 8086 处理器分支指令的一个列表JA Jump if AboveJAE Jump if Above or EqualJB Jump

23、if BelowJBE Jump if Below or EqualJC Jump if CarryJCXZ Jump if CXZero (CX is a register that can be used for loop counts) JE Jump if EqualJG Jump if Greater thanJGE Jump if Greater than or EqualJL Jump if Less thanJLE Jump if Less Than or EqualJMP JuMPJNA Jump if Not AboveJNAE Jump if Not Above or E

24、qualJNB Jump if Not BelowJNBE Jump if Not Below or EqualJNC Jump if No CarryJNE Jump if Not EqualJNG Jump if Not Greater thanJNGE Jump if Not Greater than or EqualJNL Jump if Not Less thanJNLE Jump if Not Less than or EqualJNO Jump if Not OverflowJNP Jump if Not ParityJNS Jump if Not SignJNZ Jump if

25、 Not ZeroJO Jump if OverflowJP Jump if ParityJPE Jump if Parity EvenJPO Jump if Parity OddJS Jump if SignJZ Jump if Zero80386 添加了 :JECXZ Jump if ECX Zero作为对比,ARM处理器只提供了 :B 分支BL 带连接的分支 但 ARM 提供了条件执行,你可以不受这个表面上不灵活的方式的限制 BEQ Branch if EQualBNE Branch if Not EqualBVS Branch if overflow SetBVC Branch if

26、overflow ClearBHI Branch if HigherBLS Branch if Lower or the SameBPL Branch if PLusBMI Branch if MinusBCS Branch if Carry SetBCC Branch if Carry ClearBGE Branch if Greater tha n or EqualBGT Branch if Greater Tha nBLE Branch if Less tha n or EqualBLT Branch if Less Tha nBLEQ Branch with Link if EQual

27、BLLT Branch with Link if Less Than还有两个代码,AL - ALways,缺省条件所以不须指定NV- NeVer,不是非常有用。你无论如何不要使用这个代码当你发现所有Bxx指令实际上是同一个指令的时候,紧要关头就到了。接着你会想,如果你可以在一个分支指令上加上所有这些条件,那么对一个寄存器装载指令能否加上它们?答案是可以。下面是可获得的条件代码的列表:EQ :等于如果一次比较之后设置了 Z标志。NE :不等于如果一次比较之后清除了 Z标志。VS :溢出设置如果在一次算术操作之后设置了 V标志,计算的结果不适合放入一个 32bit目标寄存器中。VC :溢出清除如果

28、清除了 V标志,与VS相反。HI :高于( 无符号 )如果一次比较之后设置了 C 标志并 清除了 Z 标志。LS :低于或同于 ( 无符号 )如果一次比较操作之后清除了C 标志 或设置了 Z 标志。PL :正号 如果一次算术操作之后清除了 原因是它不是负数 .N 。出于定义正号的目的, 零是正数的MI :负号如果一次算术操作之后设置了N 标志。CS : 进位设置 如果一次算术操作或移位操作之后设置了 C 标志,操作的结果不能表示 为 32bit 。你可以把 C 标志当作结果的第 33 位。CC : 进位清除与 CS 相反。GE : 大于或等于 (有符号) 如果一次比较之后 . 设置了 N 标志

29、 并设置了 V 标志 或者 .清除了 N 标志 并清除了 V 标志GT : 大于(有符号) 如果一次比较之后 . 设置了 N 标志 并设置了 V 标志 或者 .清除了 N 标志 并清除了 V 标志 并且.清除了 Z 标志。LE : 小于或等于 (有符号) 如果一次比较之后 . 设置了 N 标志 并清除了 V 标志 或者 .清除了 N 标志 并设置了 V 标志 并且.设置了 Z 标志。LT : 小于(有符号)如果一次比较之后 .设置了 N 标志 并清除了 V 标志 或者 .清除了 N 标志 并设置了 V 标志AL : 总是 缺省条件,所以不用明显声明NV : 从不 不是特别有用,它表示应当永远不

30、执行这个指令。是穷人的NOP。包含 NV 是为了完整性 (与 AL 相对) ,你不应该在你的代码中使用它。有一个在最后的条件代码 S,它以相反的方式工作。当用于一个指令的时候,导 致更改状态标志。 这不是自动发生的 - 除非这些指令的目的是设置状态。 例如: ADD R0, R0, R1ADDS R0, R0, R1ADDEQS R0, R0, R1 第一个例子是一个基本的加法 (把 R1 的值增加到 R0) ,它不影响状态寄存器。 第二个例子是同一个加法,只不过它导致更改状态寄存器。最后一个例子是同一个加法, 更改状态寄存器。 不同在于它是一个有条件的指令。 只有前一个操作的结果是 EQ (

31、 如果设置了 Z 标志)的时候它才执行。下面是条件执行的一个工作中的例子。你把寄存器 0 与存储在寄存器 10 中内 容相比较。如果不等于R10,则调用一个软件中断,增加它并分支回来再次做这些。 否则清 除 R10 并返回到调用它的那部分代码 (它的地址存储在 R14)。标记循环开始位置把 R0 与 R10 相比较 不等于: 调用 SWI &40017向 R0 加 1分支到 loop等于 : 设置 R10 为零 条件执行的一个例子.loopCMP R0, R10SWINE &40017ADDNE R0, R0, #1BNE loopMOV R10, #0LDMFD R13!, R0-R12,P

32、C返回到调用者注解:SWI编号就象我写的这样。在RISC OS下,它是给Econet_Dolmmediate 的编号。不要字面的接受它,这只是一个例子!你可能以前没见过LDM FD它从栈中装载多个寄存器。在这个例子中, 我们从一个完全正式的栈中装载 R0至R12和R14。关于寄存器装载和存储的 更多信息请参阅str.html 。我说要装载R14。那么为什么要把它放入 PC中?原因是此时R14存 储的值包含返回地址。我们也可以采用:LDMFD R13!, R0-R12,R14MOV PC, R14但是直接恢复到PC中可以省略这个MOV语句。最后,这些寄存器很有可能被一个 SWI调用所占用(依赖于

33、在调用期间 执行的代码),所以你最好把你的重要的寄存器压入栈中,以后在恢复它们。SWI指令SWI :软件中断(Soft ware I nterrupt)SWI条件 指令格式 这是一个简单的设施,但可能是最常用的。多数操作系统设施是用SWI提供的 没有SWI的RISC OS是不可想象的。Nava Whiteford 解释了 SWI 是如何工作的(最初在 Frobnicate issue 12?).SWI是什么?SWI表示Software Interrupt 。在RISC OS中使用SWI来访问操作系统例程 或第三方生产的模块。许多应用使用模块来给其他应用提供低层外部访问。SWI的例子有:文件器S

34、WI,它辅助读写磁盘、设置属性等。打印机驱动器SWI,用来辅助使用打印并行端口。FreeNet/Acorn TCP/IP 协议栈 SWI,用 TCP/IP 协议在 In ternet上发送和接收数据。在以这种方式使用的时候,SWI允许操作系统拥有一个模块结构,这意味着用来 建立完整的操作系统的所需的代码可以被分割成许多小的部分 (模块)和一个模 块处理程序(handler)。当SWI处理程序得到对特定的例程编号的一个请求的时候,它找到这个例程的 位置并执行它,并传递(有关的)任何数据。它是如何工作的?首先查看一下如何使用它。一个 SWI指令(汇编语言)看起来如下:SWI &02或SWI OS_

35、Write0这些指令实际上是相同的, 将被汇编成相同的指令。 唯一的不同是第二个指令使 用一个字符串来表示 SWI 编号 &02 。在使用采用了字符串编号的程序的时候,在执行之前首先查找这个字符串。 在这里我们不想处理字符串, 因为它不能给出它要进行什么的一个真实表示。 它 们通常用于增进一个程序的清晰程度,但不是实际执行的指令。让我们再次看一下第一个指令 :SWI &02这是什么意思 ? 字面的意思是进入 SWI 处理程序并传递值 &02。在 RISC OS 中 这意味着执行编号是 &02 的例程。它是如何这么作的 ? 它如何传递 SWI 编号和进入 SWI 处理程序 ?如果你查看内存的开始

36、 32 字节(位于 0-&1C) 并反汇编它们 (查开实际的 ARM 指令)你将见到如下 :地址 内容 反汇编00000000 : 0.? : E5000030 : STR R0,R0,# -4800000004 : . o ? : E59FF31C : LDR PC,&0000032800000008 : . o ? 0000000C : . o ? 00000010 : . o ?00000014 : . o ? 00000018 : . o ?0000001C :E59FF31C : LDRE59FF31C : LDRE59FF31C : LDRE59FF31C : LDRE59FF31

37、C : LDROV2? : E3A0A632 : MPC,&0000032CPC,&00000330PC,&00000334PC,&00000338PC,&0000033CR10,#&3200000让我们仔细看一下。除了第一个和最后一个指令之外 ( 它们是特殊情况 ) 你见到的都是把一个新值装 载到 PC ( 程序计数器 ) 的指令,它们告诉计算机到哪里去执行下一个指令。还展示了这个值是从内存中的一个地址接受来的。 ( 你可以在 !Zap 主菜单上使用“ Read Memory选项去自己查看一下。)这看起来好象与 SWI 没多少关系,下面做进一步的说明。一个 SWI 所做的一切就是把模式改变成

38、超级用户并设置 PC 来执行在地址 &08 处的下一个指令 !把处理器转换到超级用户模式会切换掉两个寄存器 r13 和 r14 并用 r13_svc 和 r14_svc 替换它们。在进入超级用户模式的时候,还把 r14_svc 设置为在这个 SWI 指令之后的地 址。这个实际上就象一个连接到地址 &08 的分支指令 (BL &08) ,但带有用于一些数 据(SWI编号)的空间。象我说过的那样,地址 &08 包含跳转到另一个地址的一个指令, 就是实际的 SWI 程序的地址 !此时你可能会想“稍等一会 ! 还有 SWI 编号呢 ?”。实际上处理器忽略这个值本 身。 SWI 处理程序使用传递来的 r

39、14_svc 的值来获取它。下面是完成它的步骤 (在存储寄存器 r0-r12 之后):1.它从 r14 中减去 4 来获得 SWI 指令的地址。2.把这个指令装载到一个寄存器。3. 清除这个指令的高端 8 位,去掉了 OpCode 而只剩下的 SWI 编号4. 使用这个值来找到要被执行的代码的例程的地址 (使用查找表等 ) 。5. 恢复寄存器 r0-r12 。6. 使处理器离开超级用户模式。7. 跳转到这个例程的地址。容易吧! ;)下面是一个例子,来自 ARM610 datasheet:0x08 B SupervisorEntryTableDCD ZeroRtnDCD ReadCRtnDCD

40、WriteIRtnZero EQU 0ReadC EQU 256WriteI EQU 512; SWI 包含需要的例程在位 8-23 中和数据 ( 如果有的话 ) 在位 0-7 中 ; 假定 R13_svc 指向了一个合适的栈STMFD R13, r0-r2 , R14; 保存工作寄存器和返回地址LDR R0,R14,#-4; 得到 SWI 指令。BIC R0,R0, #0xFF000000; 清除高端的 8 位。MOV R1, R0, LSR #8; 得到例程偏移量。ADR R2, EntryTable; 得到入口表 (EntryTable) 的开始地址LDR R15,R2,R1,LSL #2; 分支到正确的例程WriteIRtn; 写 R0 中的位 0 - 7 中的字符LDMFD R13, r0-r2 , R15A; 恢复工作空间,并返回、恢复处理器模式和标志 这就是 SWI 指令的基本处理步骤。

展开阅读全文
温馨提示:
1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
2: 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
3.本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
关于我们 - 网站声明 - 网站地图 - 资源地图 - 友情链接 - 网站客服 - 联系我们

copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!