- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在尝试编写 C 代码,其效果等同于以下汇编代码:
addq %rsi, %rdi
imulq %rdx, %rdi
movq %rdi, %rax
sarq $15, %rax
salq $31, %rax
andq %rdi, %rax
ret
有一个函数原型(prototype)
长解码(长x,长y,长z)
参数 x、y 和 z 在寄存器 %rdi、%rsi 和 %rdx 中传递。代码将返回值存储在寄存器 %rax 中。
到目前为止,我有以下 C 代码:
long decode(long x, long y, long z)
{
long w;
x = x+y;
x = x*z;
w = x;
w = ((x>>15)<<31);
return x&w;
}
其中,当使用 gcc -O2 -S -c filename.c 编译时产生以下内容:
addq %rdi, %rsi
imulq %rdx, %rsi
movq %rsi, %rax
sarq $15, %rax
salq $31, %rax
andq %rsi, %rax
ret
显然,寄存器 %rdi 和 %rsi 被交换了。
所以,如果我通过交换 x 和 y 来更改函数,它看起来像这样:
long decode4(long x, long y, long z)
{
long w;
y = x + y;
y = y * z;
w = y;
w = ((y>>15)<<31);
return y&w;
}
而且,程序集又是这样的:
addq %rdi, %rsi
imulq %rdx, %rsi
movq %rsi, %rax
sarq $15, %rax
salq $31, %rax
andq %rsi, %rax
ret
y 和 x 的交换,并没有改变生成的汇编代码中的任何东西。关于如何解决该问题的任何想法?谢谢!
最佳答案
您对代码的解释确实没有任何问题:
long decode(long x, long y, long z)
{
long w;
x = x+y;
x = x*z;
w = x;
w = ((x>>15)<<31);
return x&w;
}
它可以稍微简化,但它没有任何问题,因为输出仅在所用寄存器的反转方面有所不同。可观察到的结果将是相同的。您声明此要求:
I am trying to write C code that will have an effect equivalent to the following assembly
两者等价,所以我相信您的解决方案已经满足了作业要求。
有时这种类型的事情可以归结为所使用的编译器。我注意到我可以通过 GCC 4.6.4 和 -O2
优化级别获得您正在寻找的准确输出。您可以在 godbolt 上使用此代码输出是:
decode:
addq %rsi, %rdi
imulq %rdx, %rdi
movq %rdi, %rax
sarq $15, %rax
salq $31, %rax
andq %rdi, %rax
ret
这似乎与您类(class)的输出完全匹配。使用相同版本的 GCC (4.6.4),您可以获得相同的输出:
long decode(long x, long y, long z)
{
x = (x + y) * z;
return x & ((x >> 15) << 31);
}
关于c - 汇编语言到 C 问题与寄存器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49343708/
我不确定 cbw命令实际上可以。我有一段代码: mov ax,0FF0h cbw idiv ah cbw后ax的值如何变化? 最佳答案 cbw指令将一个字节符号扩展为一个字。在这种情况下,它将采用
MIPS 架构中的.s 和.asm 有什么区别? 最佳答案 .s 和.asm 之间没有天生的区别 如果你愿意,你可以用 .s 命名文本文件,用 .asm 命名十六进制文件,但这只是为了让它们保持正直。
我正在业余时间学习汇编语言,以成为更好的开发人员。 我在概念层面上理解基于堆栈的机器和基于寄存器的机器之间的区别,但我想知道基于堆栈的机器实际上是如何实现的。如果是虚拟机,例如JVM 或 .NET,在
几周后我开始上大学(软件工程学位),并且知道我将在第一年学习的一个模块是“建筑基础”或类似的东西;基本上它是在学习“汇编”中的编程。 我想做一些背景学习,所以我为此做好了准备(我在其他一些单元上做了同
我是汇编语言的新手。我目前正在经历这个 Assembly Guide .我对 LEA 指令有疑问。我对LEA指令的理解是LEA用source operand的有效地址加载destination ope
x86 汇编中是否有模运算符或指令之类的东西? 最佳答案 如果您的模数/除数是一个已知常数,并且您关心性能,请参阅 this和 this .对于直到运行时才知道的循环不变值,乘法逆甚至是可能的,例如见
我是汇编语言的新手。我目前正在经历这个 Assembly Guide .我对 LEA 指令有疑问。我对LEA指令的理解是LEA用source operand的有效地址加载destination ope
我相信这对大多数人来说非常简单,但我对 x86 汇编语言不是很熟悉。我只是想自学。 我在 window 里。在我读到的任何地方,我都被告知要使用 INT 21返回操作系统。这退出了程序,但我收到一条错
我正在尝试输入一个字符串,然后查看该字符串中的最后一个值是否为 EOL 字符。我想我会使用读入的字符串的长度,然后将它添加到缓冲区的地址以找到最后一个元素。这似乎不起作用。 编辑:很抱歉我没有包含更多
; This program checks for even or odd parities inside of an array. Displays 1 if even or 0 if odd. I
我一直在努力掌握 x86 汇编语言,并且想知道是否有一个与 movl $1, %eax 等效的快速简短的语言。就在那时,我认为列出该语言中经常使用的习语可能是个好主意。 这可能包括首选使用 xorl
有人知道如何在 ARMv6 汇编语言中开始注释(例如“#”、“;”、“/”、“/*”)吗? 最佳答案 应该是“;”根据我在 wikipedia article: 上找到的一些示例代码 loop
; This program checks for even or odd parities inside of an array. Displays 1 if even or 0 if odd. I
我是一名利用业余时间学习汇编语言的大学生。我注意到虽然有像 add 和 mul 这样的指令,但算术运算符经常在指令中使用。例如: mov eax,[ebx+ecx] 它等价于以下吗? add ebx,
我真的是学习汇编语言的新手,并且刚刚开始研究汇编语言,所以我想知道也许有些人可以帮助我解决一个问题。我有一个作业,告诉我将汇编语言指令与C代码进行比较,并告诉我哪些C代码与汇编指令等效。所以这是组装说
假设我有以下代码: C++ 代码和内联汇编代码如下: 方程式: X=40+5 Y=3*12 ---> X=45 Y=36 测试 if x < y - print x+y else print x-y
我正在编写一个程序,其中的要求如下: 有一个名为 Name 的字符串变量,该变量设置为您的全名,包括空格。 只清除 AL 寄存器,不清除 EAX 寄存器的其余部分。 遍历 Name 变量中的每个字符。
我有一个任务,描述如下: - Reads printable characters (20h-7Fh) from the keyboard without echoing - Uppercase le
我正在查看 switch 语句的汇编语言代码。 我了解代码的工作原理以及情况。我的问题是如何决定案例名称? 下面是汇编语言代码,下面是我的解释。我基本上只需要使用跳转表并填写案例名称。 1 8
我是汇编语言新手,我想到了这个问题:构建代码、宏或过程哪个是更好的解决方案?虽然宏中包含的代码只是复制到所需的部分,并且允许执行代码而无需昂贵的跳转,但它确实添加了必须在程序执行时加载的额外指令。此外
我是一名优秀的程序员,十分优秀!