Skip to content

Commit

Permalink
详解ARM汇编条件标志
Browse files Browse the repository at this point in the history
  • Loading branch information
CYRUS-STUDIO committed Nov 4, 2024
1 parent 55081d1 commit 6334886
Show file tree
Hide file tree
Showing 13 changed files with 593 additions and 106 deletions.
114 changes: 114 additions & 0 deletions content/posts/详解ARM汇编条件标志.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
+++
title = '详解ARM汇编条件标志'
date = 2024-11-04T20:19:51.993649+08:00
draft = false
+++

> 版权归作者所有,如有转发,请注明文章出处:<https://cyrus-studio.github.io/blog/>
# __条件标志__


在 ARM 指令集中,条件标志是控制指令执行的一种机制,它们用于实现条件分支、比较和其他逻辑操作。

我们平时使用 IDA 调试程序时,在 general registers 窗口中看到的条件标志
![image.png]()

ARM 处理器通常使用四个主要的条件标志,它们的状态影响指令的执行。这些条件标志包括:

1\. N (Negative): 负标志,用于指示最后一次操作的结果是否为负值。若结果为负,则 N 被置为 1;否则为 0。

2\. Z (Zero): 零标志,用于指示最后一次操作的结果是否为零。如果结果为零,则 Z 被置为 1;否则为 0。

3\. C (Carry): 进位标志,用于指示加法操作是否产生了进位或减法操作是否没有借位。对于加法,若产生进位,则 C 被置为 1;对于减法,若没有借位,则 C 被置为 1。

4\. V (Overflow): 溢出标志,用于指示最后一次算术操作是否发生了溢出。当两个有符号数相加或相减时,如果结果的符号与操作数的符号不一致,表示发生了溢出,V 被置为 1。

在 ARM 指令中,条件执行可以通过附加条件码实现,例如:

- EQ (Equal): 当 Z = 1 时执行(相等)。

- NE (Not Equal): 当 Z = 0 时执行(不相等)。

- GT (Greater Than): 当 Z = 0 且 N = V 时执行(大于)。

- LT (Less Than): 当 N ≠ V 时执行(小于)。

- GE (Greater Than or Equal): 当 N = V 时执行(大于或等于)。

- LE (Less Than or Equal): 当 Z = 1 或 N ≠ V 时执行(小于或等于)。

这些条件标志和条件码使得 ARM 架构能够高效地进行复杂的控制流和决策逻辑,从而优化程序的执行效率。

# __条件标志如何影响指令的执行__


下面以一段汇编代码解释条件标志的变化过程,以及如何通过条件码用于指令影响汇编指令的走向。
```
.data
a: .word 5 // 定义变量 a,值为 5
b: .word 10 // 定义变量 b,值为 10
max_value: .word 0 // 存储最大值的变量
.text
.global _start
_start:
// 读取 a 和 b 的值
LDR R0, =a // 将 a 的地址加载到 R0
LDR R1, [R0] // 将 a 的值加载到 R1 (R1 = 5)
LDR R0, =b // 将 b 的地址加载到 R0
LDR R2, [R0] // 将 b 的值加载到 R2 (R2 = 10)
// 比较 a 和 b
CMP R1, R2 // 比较 R1 (a) 和 R2 (b)
// 根据 R1 和 R2 的值,设置条件标志
// 如果 R1 < R2:
// N = 1, Z = 0, C = 1, V = 0
// 如果 R1 == R2:
// N = 0, Z = 1, C = 1, V = 0
// 如果 R1 > R2:
// N = 0, Z = 0, C = 0, V = 0
// 根据比较结果设置最大值
BEQ a_equals_b // 如果 Z = 1 (相等),跳转到 a_equals_b
BGT a_greater // 如果 N = 0 且 Z = 0,(a > b),跳转到 a_greater
// 如果到这里,说明 b > a
STR R2, =max_value // b 是最大值,存储 b 的值
B end // 跳转到 end
a_equals_b:
// a 和 b 相等
STR R1, =max_value // 存储任一值,a 或 b 都可以
B end // 跳转到 end
a_greater:
// a 大于 b
STR R1, =max_value // 存储 a 的值
end:
// 结束程序
MOV R7, #1 // 系统调用号,退出
SWI 0 // 触发系统调用
```

# __CMP 指令是如何影响条件标志__


在 ARM 汇编中,CMP 指令用于比较两个寄存器的值。具体来说,CMP R1,R2 指令会将寄存器 R1 的值减去寄存器 R2 的值,但不会将结果存储在任何寄存器中。这一操作的主要目的是更新条件标志,以便后续的条件执行指令可以根据比较结果做出决策。

我们可以在执行比较 CMP R1, R2 时观察到条件标志的变化。
| 步骤 | R1 (a) | R2 (b) | N (Negative) | Z (Zero) | C (Carry) | V (Overflow) | 说明 |
| --- | --- | --- | --- | --- | --- | --- | --- |
| 读取 a 的值 | 5 | - | - | - | - | - | 加载 a 的值到 R1 |
| 读取 b 的值 | 5 | 10 | - | - | - | - | 加载 b 的值到 R2 |
| 比较 R1 和 R2 | 5 | 10 | 1 | 0 | 1 | 0 | 执行 CMP R1, R2 ,结果 5 - 10(借位) |
| BEQ 检查 | 5 | 10 | 1 | 0 | 1 | 0 | Z = 0(不相等),不跳转到 a_equals_b |
| BGT 检查 | 5 | 10 | 1 | 0 | 1 | 0 | N = 1,Z = 0(a 不大于 b),不跳转到 a_greater |
| 存储最大值 | - | 10 | 1 | 0 | 1 | 0 | 存储 b 的值到 max_v |


34 changes: 17 additions & 17 deletions public/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,23 @@
<h1>CYRUS STUDIO</h1>
<ul class="posts-list">

<li class="posts-list-item">
<a class="posts-list-item-title" href="https://cyrus-studio.github.io/blog/posts/%E8%AF%A6%E8%A7%A3arm%E6%B1%87%E7%BC%96%E6%9D%A1%E4%BB%B6%E6%A0%87%E5%BF%97/">详解ARM汇编条件标志</a>
<span class="posts-list-item-description">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-calendar">
<title>calendar</title>
<rect x="3" y="4" width="18" height="18" rx="2" ry="2"></rect><line x1="16" y1="2" x2="16" y2="6"></line><line x1="8" y1="2" x2="8" y2="6"></line><line x1="3" y1="10" x2="21" y2="10"></line>
</svg>
Nov 4, 2024
<span class="posts-list-item-separator">-</span>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-clock">
<title>clock</title>
<circle cx="12" cy="12" r="10"></circle><polyline points="12 6 12 12 16 14"></polyline>
</svg>
3 min read
</span>
</li>

<li class="posts-list-item">
<a class="posts-list-item-title" href="https://cyrus-studio.github.io/blog/posts/thumb-%E6%B1%87%E7%BC%96%E6%8C%87%E4%BB%A4%E9%9B%86thumb-%E6%8C%87%E4%BB%A4%E7%BC%96%E7%A0%81%E6%96%B9%E5%BC%8F%E7%BC%96%E8%AF%91-thumb-%E6%B1%87%E7%BC%96%E4%BB%A3%E7%A0%81/">Thumb 汇编指令集,Thumb 指令编码方式,编译 Thumb 汇编代码</a>
<span class="posts-list-item-description">
Expand Down Expand Up @@ -207,23 +224,6 @@ <h1>CYRUS STUDIO</h1>
</span>
</li>

<li class="posts-list-item">
<a class="posts-list-item-title" href="https://cyrus-studio.github.io/blog/posts/%E4%BD%BF%E7%94%A8miniconda%E7%AE%A1%E7%90%86python%E7%8E%AF%E5%A2%83/">使用Miniconda管理Python环境</a>
<span class="posts-list-item-description">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-calendar">
<title>calendar</title>
<rect x="3" y="4" width="18" height="18" rx="2" ry="2"></rect><line x1="16" y1="2" x2="16" y2="6"></line><line x1="8" y1="2" x2="8" y2="6"></line><line x1="3" y1="10" x2="21" y2="10"></line>
</svg>
Oct 27, 2024
<span class="posts-list-item-separator">-</span>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-clock">
<title>clock</title>
<circle cx="12" cy="12" r="10"></circle><polyline points="12 6 12 12 16 14"></polyline>
</svg>
2 min read
</span>
</li>

</ul>


Expand Down
9 changes: 8 additions & 1 deletion public/index.xml
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,15 @@
<description>Recent content on CYRUS STUDIO</description>
<generator>Hugo</generator>
<language>zh-cn</language>
<lastBuildDate>Mon, 04 Nov 2024 19:17:49 +0800</lastBuildDate>
<lastBuildDate>Mon, 04 Nov 2024 20:19:51 +0800</lastBuildDate>
<atom:link href="https://cyrus-studio.github.io/blog/index.xml" rel="self" type="application/rss+xml" />
<item>
<title>详解ARM汇编条件标志</title>
<link>https://cyrus-studio.github.io/blog/posts/%E8%AF%A6%E8%A7%A3arm%E6%B1%87%E7%BC%96%E6%9D%A1%E4%BB%B6%E6%A0%87%E5%BF%97/</link>
<pubDate>Mon, 04 Nov 2024 20:19:51 +0800</pubDate>
<guid>https://cyrus-studio.github.io/blog/posts/%E8%AF%A6%E8%A7%A3arm%E6%B1%87%E7%BC%96%E6%9D%A1%E4%BB%B6%E6%A0%87%E5%BF%97/</guid>
<description>版权归作者所有,如有转发,请注明文章出处:https://cyrus-studio.github.io/blog/&#xA;条件标志 在 ARM 指令集中,条件标志是控制指令执行的一种机制,它们用于实现条件分支、比较和其他逻辑操作。&#xA;我们平时使用 IDA 调试程序时,在 general registers 窗口中看到的条件标志 ARM 处理器通常使用四个主要的条件标志,它们的状态影响指令的执行。这些条件标志包括:&#xA;1. N (Negative): 负标志,用于指示最后一次操作的结果是否为负值。若结果为负,则 N 被置为 1;否则为 0。&#xA;2. Z (Zero): 零标志,用于指示最后一次操作的结果是否为零。如果结果为零,则 Z 被置为 1;否则为 0。&#xA;3. C (Carry): 进位标志,用于指示加法操作是否产生了进位或减法操作是否没有借位。对于加法,若产生进位,则 C 被置为 1;对于减法,若没有借位,则 C 被置为 1。&#xA;4. V (Overflow): 溢出标志,用于指示最后一次算术操作是否发生了溢出。当两个有符号数相加或相减时,如果结果的符号与操作数的符号不一致,表示发生了溢出,V 被置为 1。&#xA;在 ARM 指令中,条件执行可以通过附加条件码实现,例如:&#xA;EQ (Equal): 当 Z = 1 时执行(相等)。&#xA;NE (Not Equal): 当 Z = 0 时执行(不相等)。&#xA;GT (Greater Than): 当 Z = 0 且 N = V 时执行(大于)。</description>
</item>
<item>
<title>Thumb 汇编指令集,Thumb 指令编码方式,编译 Thumb 汇编代码</title>
<link>https://cyrus-studio.github.io/blog/posts/thumb-%E6%B1%87%E7%BC%96%E6%8C%87%E4%BB%A4%E9%9B%86thumb-%E6%8C%87%E4%BB%A4%E7%BC%96%E7%A0%81%E6%96%B9%E5%BC%8F%E7%BC%96%E8%AF%91-thumb-%E6%B1%87%E7%BC%96%E4%BB%A3%E7%A0%81/</link>
Expand Down
34 changes: 17 additions & 17 deletions public/page/2/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,23 @@
<h1>CYRUS STUDIO</h1>
<ul class="posts-list">

<li class="posts-list-item">
<a class="posts-list-item-title" href="https://cyrus-studio.github.io/blog/posts/%E4%BD%BF%E7%94%A8miniconda%E7%AE%A1%E7%90%86python%E7%8E%AF%E5%A2%83/">使用Miniconda管理Python环境</a>
<span class="posts-list-item-description">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-calendar">
<title>calendar</title>
<rect x="3" y="4" width="18" height="18" rx="2" ry="2"></rect><line x1="16" y1="2" x2="16" y2="6"></line><line x1="8" y1="2" x2="8" y2="6"></line><line x1="3" y1="10" x2="21" y2="10"></line>
</svg>
Oct 27, 2024
<span class="posts-list-item-separator">-</span>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-clock">
<title>clock</title>
<circle cx="12" cy="12" r="10"></circle><polyline points="12 6 12 12 16 14"></polyline>
</svg>
2 min read
</span>
</li>

<li class="posts-list-item">
<a class="posts-list-item-title" href="https://cyrus-studio.github.io/blog/posts/%E4%BD%BF%E7%94%A8-ida-tracing-%E5%8A%A8%E6%80%81%E8%B7%9F%E8%B8%AA-jni-%E6%96%B9%E6%B3%95%E6%89%A7%E8%A1%8C%E8%BF%87%E7%A8%8B/">使用 IDA Tracing 动态跟踪 JNI 方法执行过程</a>
<span class="posts-list-item-description">
Expand Down Expand Up @@ -207,23 +224,6 @@ <h1>CYRUS STUDIO</h1>
</span>
</li>

<li class="posts-list-item">
<a class="posts-list-item-title" href="https://cyrus-studio.github.io/blog/posts/android%E4%B8%8B%E5%8F%8D%E8%B0%83%E8%AF%95%E4%B8%8E%E5%8F%8D%E5%8F%8D%E8%B0%83%E8%AF%95/">Android下反调试与反反调试</a>
<span class="posts-list-item-description">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-calendar">
<title>calendar</title>
<rect x="3" y="4" width="18" height="18" rx="2" ry="2"></rect><line x1="16" y1="2" x2="16" y2="6"></line><line x1="8" y1="2" x2="8" y2="6"></line><line x1="3" y1="10" x2="21" y2="10"></line>
</svg>
Sep 19, 2024
<span class="posts-list-item-separator">-</span>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="icon icon-clock">
<title>clock</title>
<circle cx="12" cy="12" r="10"></circle><polyline points="12 6 12 12 16 14"></polyline>
</svg>
5 min read
</span>
</li>

</ul>


Expand Down
Loading

0 comments on commit 6334886

Please sign in to comment.