C指標原理(7)-C內嵌彙編
阿新 • • 發佈:2018-11-30
如何在內聯彙編中訪問C程式的區域性變數呢,請看下面這段程式碼。
#include <stdio.h> int main(void){ //不使用全域性變數,必須使用擴充套件GNU的asm //格式為:asm("彙編程式碼":輸出位置:輸入位置:改動的暫存器列表) //a為eax,ax,al;b為ebx等;c為ecx等;d為edx等;S為esi或si;D為edi或di //+讀和寫;=寫;%如果必要,運算元可以和下一個運算元切換;&在行內函數完成之前,可以刪除或重新使用運算元 int xa=6; int xb=2; int result; //ansi c標準的asm有其它用,所以用__asm__,__volatile__表示內聯彙編部分不用優化(可以用volatile,但是ansi c不行),以防優化破壞內聯程式碼組織結構 asm volatile( "add %%ebx,%%eax\n\t" "movl $2,%%ecx\n\t" "mul %%ecx\n\t" "movl %%eax,%%edx" :"=d"(result):"a"(xa),"b"(xb):"%ecx");//注意擴充套件方式使用2個%表示 printf("%d\n",result); return 0; }
這個例子完成這個計算:(xa+xb)2=(6+2)2=16
不使用全域性變數與彙編程式碼互動,我們必須使用擴充套件GNU的asm ,格式為:
asm("彙編程式碼":輸出位置:輸入位置:改動的暫存器列表)
彙編程式碼中涉及暫存器部分的使用2個“%”,如:使用%%eax表示eax暫存器
輸出位置、輸入位置的特殊命名規則為:
a為eax,ax,al;b為ebx等;c為ecx等;d為edx等;S為esi或si;D為edi或di
+讀和寫
=寫
%如果必要,運算元可以和下一個運算元切換
&在行內函數完成之前,可以刪除或重新使用運算元
上述程式碼中,彙編程式碼部分為
輸出位置、輸入位置、改動的暫存器列表部分為:
:"=d"(result):"a"(xa),"b"(xb):"%ecx"
先來看彙編程式碼部分,使用雙%號表示暫存器,比如:
"add %%ebx,%%eax\n\t"
關於輸出位置、輸入位置部分,可以這麼理解:將變數與暫存器繫結,繫結後,對暫存器的操作就是對變數的操作。
:"=d"(result):"a"(xa),"b"(xb)
將result與暫存器edx繫結,xa與暫存器eax繫結,xb與暫存器ebx繫結。
%ecx屬於需要改動的暫存器