X86&&X64 汇编学习——内联基础

偏执的太偏执、 2022-09-23 15:50 407阅读 0赞
  1. 前言
  2. 现在已经进入了64位时代,但是几乎所有的汇编教程道还停留在16位,32位上。总是让人用起来有种脱轨的感觉,诚如上面所言,我们很难很难找到讲授X64汇编的文档,手册,甚至教程。
  3. 但是我还是需要使用的,没办法需要自己慢慢查资料摸索。最近在回顾X86的内嵌汇编,所以想着在做32位内嵌汇编的同时学习64位汇编。
  4. X64汇编的学习你至少要知道X86 32位汇编的基础知识。

在C语言中内嵌汇编访问全局变量

X86

  1. #include<stdio.h>
  2. int a = 10;
  3. int b = 20;
  4. int result;
  5. int main(){
  6. __asm__ __volatile__ ("movl a, %eax\n\t"
  7. "movl b, %ebx\n\t"
  8. "imull %ebx,%eax\n\t"
  9. "movl %eax,result\n\t"
  10. );
  11. printf("the answer is %d\n",result);
  12. return 0;
  13. }

这段程序的目的用result 来保存 a 与 b 的乘积,典型的X86用法。我们再来看看X64的用法:

  1. #include<stdio.h>
  2. int a = 10;
  3. int b = 20;
  4. int result;
  5. int main(){
  6. __asm__ __volatile__ ("movq a, %rax\n\t"
  7. "movq b, %rbx\n\t"
  8. "imulq %rbx,%rax\n\t"
  9. "movq %rax,result\n\t"
  10. );
  11. printf("the answer is %d\n",result);
  12. return 0;
  13. }

首先指令上:movl —-> movq 转变成了64位指令 imull —-> imulq 其次看寄存器: e*x —-> r*x 32位寄存器转换成空了64位寄存器

扩展ASM

__asm__ __volatile__(“assembly code”:output locations : input locations : changed registers); 汇编代码,输出代码,输入代码,修改寄存器/破坏性描述

使用寄存器

在使用寄存器的时候,我们需要注意的是:内存和寄存器交互使用,需要用“%”转义寄存器。64位下随机选择寄存器使用“g”参数。 例如我们将变量b的值赋值给变量a.

  1. #include<stdio.h>
  2. int main(){
  3. int a = 10;
  4. int b = 20;
  5. __asm__ __volatile__("movq %1,%%rbx\n\t"
  6. "movq %%rbx,%0":"=m"(a):"g"(b));
  7. printf("a = %d\n",a);
  8. }

替换占位符

简单来说就是做了标记,使用[] 给变量起一个别名

  1. #include<stdio.h>
  2. int main(){
  3. int a = 10;
  4. int b = 20;
  5. __asm__ __volatile__("movq %[name_a],%%rbx\n\t"
  6. "movq %%rbx,%[name_b]":[name_b]"=m"(a):[name_a]"g"(b));
  7. printf("a = %d\n",a);
  8. }

效果和之前时一样的。

寄存器破坏

如果使用再输入输出中没有声明但是已经使用的寄存器,需要在破坏描述符部分说明:

  1. #include<stdio.h>
  2. int main(){
  3. int data1 = 10;
  4. int result = 20;
  5. __asm__ __volatile__("movq %1,%%rax\n\t"
  6. "movq %%rax,%0":"=m"(result):"g"(data1):"%rax");
  7. printf("result = %d\n",result);
  8. }

内联跳转,宏一样处理。

发表评论

表情:
评论列表 (有 0 条评论,407人围观)

还没有评论,来说两句吧...

相关阅读