- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在编写自己的汇编程序并尝试对 ADC 指令进行编码,我对立即值有疑问,尤其是在将 8 位值添加到 AX 寄存器时。
添加 16 位值时:adc ax, 0xff33
被编码为 15 33 ff
哪个是正确的。
但是如果 adc ax, 0x33
是否重要?被编码为 15 33 00
?
Nasm 将其编码为 83 d0 33
这显然是正确的,但我的方法也正确吗?
最佳答案
x86 通常有超过 1 种有效的指令编码方式。例如大多数 op reg, reg
指令都可以选择通过 op r/m, reg
或 op reg, r/m
操作码进行编码。
是的,通常您希望汇编程序始终为指令选择最短的编码。 NASM 甚至将 mov rax, 1
(mov r64, sign_extended_imm32
的 7 个字节)优化为 x86-64 的 mov eax, 1
(5 个字节),更改操作数大小以使用写入 32 位寄存器的零扩展,而不是 32 位的显式符号扩展即时。
在可用时使用符号扩展 imm8 编码总是好的
它对于 16 位长度相等,但对于 32 位操作数大小较短,因此它简化了您的代码以始终选择 imm8
。
操作数大小为 32 位时,op eax, imm32
为 5 个字节,而 op r/m32, imm8
仍为 3 个字节。 (不包括设置操作数大小或其他内容所需的任何前缀;两者都是相同的。)
imm8 编码的性能优势
如果需要操作数大小前缀(例如在 adc ax, 0x33
的 32 位模式下),使用带有操作数大小前缀的 adc ax/eax/rax, imm16/32/32
编码将在 Intel CPU 上创建 LCP 停顿(长度更改前缀意味着前缀更改长度指令的其余部分。这不会发生在 imm8 编码中,因为无论操作数大小如何,它仍然是(前缀)+ 操作码 + modrm + imm8。
请参阅 Agner Fog's microarch.pdf 中的 x86 tag wiki 和其他性能链接。另请参阅 x86 instruction encoding how to choose opcode,它与此相同,除了 adc
是一个特殊情况。
在 adc
/sbb
的特定情况下,避免 ax, imm16
编码还有另一个优势:通过 Haswell 参见 Sandybridge 上的 Which Intel microarchitecture introduced the ADC reg,0 single-uop special case?,adc ax, 0
被特例视为单 uop 指令,而不是 3 输入 uop 的普通 2(斧头,旗帜,立即)。
但是这种特殊的 shell 不适用于 no-ModRM 短格式编码,因此 3 字节的 adc ax, imm16
仍然解码为 2 uops 。只有 imm8
形式的解码器在解码为单个 uop 之前检查立即数是否为零。 (它仍然不适用于 adc al, imm8
。)
所以总是尽可能选择符号扩展的 imm8 也是最佳选择, 即使在 adc ax,0
不需要操作数大小前缀的 16 位模式下也是如此,因此不会发生 LCP 停顿问题。
大多数汇编程序不提供覆盖来避免 no-ModRM 短格式。在设计它们时,除了有意延长指令以获得对齐而不在循环顶部或其他分支目标之前添加 NOP 之外,没有其他性能用例:What methods can be used to efficiently extend instruction length on modern x86?
如果您正在设计一种新的 asm 语法风格,您可能会考虑允许使用 override 关键字对编码进行更多控制。对于现有设计,查看 NASM 的 strict
和 nosplit
关键字,以及 GAS 的 {vex2}
、 {vex3}
、 {disp32}
等“前缀”
nosplit
以强制对 LEA 进行更长更有效的编码。 {disp32}
等,以及 {load}
或 {store}
以选择您喜欢的 op r/m, r
与 op r, r/m
编码中的哪一个。)a32 mov eax, [0x123456]
编码的 moffs
会导致 Intel CPU 上的 LCP 停顿。对于绝对寻址,它比 modrm+SIB+disp32 短,但可能更慢。 mov rax,1
(5 字节)与 mov rax, strict dword 1
(7 字节)与 mov rax, strict qword 1
(10 字节 imm64
编码)关于assembly - x86 汇编 16 位与 8 位立即操作数编码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56524046/
int enter_path(char** path) { char* standard = "./questions.txt"; printf("\n\t%s\n\t%s",
我有以下几行代码: #define PORT 9987 和 char *ptr = (char *)&PORT; 这似乎适用于我的服务器代码。但是当我在我的客户端代码中写它时,它给出了这个错误信息:
大家好,我在成员函数中有以下内容 int tt = 6; vector>& temp = m_egressCandidatesByDestAndOtMode[tt]; set& egressCandi
我知道您可以通过以下方式在正则表达式中使用 NOT 操作数: [^AB] :匹配除 "A" 之外的任何内容或"B" A(?!B) :匹配"A" ,后面不跟 "B" (?
我的代码如下,下面还解释了为什么会发生左值; typedef struct test_item { char id[MENU_NAME_LEN + NULL_SPACE]; MenuF
我正在审查一些 javascript 代码,程序员在几个地方使用了 >>。我试图在谷歌上搜索但找不到这个操作数/运算符的作用。所以我来了。下面的代码示例: var triplet=(((binarra
我使用以下行(希望这是最佳实践,如果不正确请纠正我)来处理命令行选项: #!/usr/bin/bash read -r -d '' HELP &2 for i in "${invalid_opti
我正在尝试编辑一个计时器应用程序,出现了这行代码。我该如何解决? let styleMask: Int = NSClosableWindowMask | NSTitledWindowMask 错误是:
我可以得到两个特定日期之间的差异,这将等于日期总数。现在我想将工作日除以总天数并得到整数输出。 @IBAction func go(_ sender: UIButton) { let con
我的项目有一个问题,它应该使用一个线程将每一行相加,然后将它们全部相加,但是我收到一个错误,指出左值需要作为一元 '&"操作数 pthread_create(&tid, NULL, &sum_line
我的代码有问题。有以下功能: static Poly PolyFromCoeff(int coeff); static Mono MonoFromPoly(const Poly *p, int exp
在 C# 中是否没有字符串的 OR 操作数? 我正在查看 Microsoft C# 操作数页面 - 没有关于字符串的任何类型的 OR。 我有一个要写的 if 语句: if (Convert.ToStr
下面的函数左移一个double操作数: double shl(double x,unsigned long long n) { unsigned long long* p = (unsigne
我在 Linux 中使用了以下简单的 ksh 脚本 #!/bin/ksh set -x ### Process list of *.dat files if [ -f *.dat ] then pri
我有一个使用 Entity Framework 的查询。它有许多不同的操作数,我对其优先级感到困惑。我得到了错误的结果。我需要所有 IsPaid == true 或 IsPaid == null 的记
我有以下代码来尝试创建一个约束数组以添加到 View 中: let views = ["button": button] let metrics = ["margin": 16] var constr
这个问题在这里已经有了答案: How to compare one value against multiple values - Swift (8 个答案) 关闭 6 年前。 我有一种情况,我必须
我使用 jquery $.ajax 将请求发送到服务器,它返回 JSON。 $.ajax({ url: 'moreMonth.ajax', data: { startIndex: id },
我的问题是程序没有按照“他”的预期读取代码。 我有 if (hero.getPos() == (6 | 11 | 16)) { move = new Object[] {"Up", "Righ
我在对象中创建线程时遇到问题。错误是需要作为一元“&”操作数的左值 CPP文件 #include "AirQ.h" static int i=0; AirQ::AirQ(int pinNo, bool
我是一名优秀的程序员,十分优秀!