GCC内联汇编(INLINE ASSEMBLY)
  什么是内联汇编(Inline assembly)?
  1、这是GCC对C语言的扩张,是在C代码里面去写汇编代码
  2、可以直接在C的语句中插入汇编指令
  有何用处?
  1、C语言不足以完成所有CPU的指令, 特别是有一些特权指令,比如加载gdt表(Global Descriptor Table 全局描述符表),从而使用汇编代码来完成
  2、用汇编在C语言中手动优化,特别是在操作系统当中,使用汇编对操作系统的掌控更为,更加准确。
  如何工作?
  1、用给定的模板和约束来生成汇编指令
  2、才C函数内形成汇编代码
  下面一个小例子,简单看一下汇编代码和内联汇编的不同:
  GCC内联汇编-Example 1
  汇编代码 Assembly(*.S):
  movl $0xffff, %eax     //把0xffff给eax
  内联汇编 Inline assembly(*.c):
  asm("movl $0xffff, %%eax ")    //1、加了一个内联汇编关键字asm,把汇编代码括起来然后用引号表示一个字符串 2、寄存器前变为%%
  GCC内联汇编-语法
  asm("assmbler template"               //assmbler template是example 1 的那个字符串的内容(汇编语句)
  :output operands (optional)  //可选的输出操作数(输出寄存器)
  :input operands   (optional)  //可选的输入操作数(输入寄存器)
  :clobbers              (optional)  //会被修改的寄存器
  );
  GCC内联汇编-Example2
  Inline assembly(*.c)
  uint32_t cr0;                   //一个叫cr0的变量
  asm volatile("movl %%cr0, %0 " :"=r"(cr0));   //把cr0里面的内容读到%0这个寄存器当中, =r这个寄存器的内容会赋给cr0这个变量(输出操作)
  cr0 |= 0x80000000;                 //把某一个位设成1
  asm volatile("movl %0, %%cr0 " ::"r"(cr0));    //再把cr0变量里的内容写回到cr0寄存器当中。把cr0给一个寄存器,然后这个寄存器再把他里面的内容赋给cr0寄存器(输入操作)
  一些关键字的含义
  volatile:No reordering; No elimination. 不需要进一步的优化,不需要进一步的调整顺序
  %0  :The first constraint following. 第一个用到的寄存器
  r    :A constraint; GCC is free to use any register. 任意一个寄存器
  上面的内联汇编所对应的汇编代码如下
  Generated assembly code(*.s):
  movl %cr0, %ebx                             //cr0 --> ebx
  movl %ebx, 12(%esp)                     //ebx赋给一个局部变量12(%esp)
  orl    $-2147483648, 12(%esp)       //或操作
  movl 12(%esp), %eax                     //再给eax
  movl %eax, %cr0        //再把eax寄存器里面的内容赋给cr0
  GCC内联汇编-Example3
  Inline assembly(*.c)
  long _res, arg1 = 2, arg2 = 22, arg3 = 222, arg4 = 233;
  _asm_volatile("int $0x80"
  :"=a"(_res)
  :"0"(11), "b"(arg1), "c"(arg2), "d"(arg3), "S"(arg4));    //把arg1给ebx, arg2给ecx,arg3给edx, arg4给esi
  在内联汇编当中
  a = %eax, b = %ebx, c = %ecx, d = %edx, S = %esi, D = %edi, 0 = same as the first 第一个出现的寄存器
  Generated assembly code(*.s):
  mov $11, %eax
  movl-28(%ebp), %ebx
  movl-24(%ebp), %ecx
  movl-20(%ebp), %edx
  movl-16(%ebp), %esi
  int $0x80
  movl %edi, -12(%ebp)