AT&T與Intel彙編語法的比較

GCC採用的是AT&T的彙編格式, 也叫GAS格式(Gnu ASembler GNU彙編器), 而微軟採用Intel的彙編格式. 
語法上主要有以下幾個不同. 
1、寄存器命名原則

AT&T 彙編格式中,寄存器名要加上 '%' 作爲前綴;而在 Intel 彙編格式中,寄存器名不需要加前綴。

AT&T

Intel

說明

%eax

eax

Intel的不帶百分號


2、源/目的操作數順序

AT&T Intel 格式中的源操作數和目標操作數的位置正好相反。 Intel 彙編格式中,目標操作數在源操作數的左邊;而在 AT&T 彙編格式中,目標操作數在源操作數的右邊。

AT&T

Intel

說明

movl %eax, %ebx

mov ebx, eax

Intel的目的操作數在前,源操作數在後


3、常數/立即數的格式

AT&T 彙編格式中,用 '$' 前綴表示一個立即操作數;而在 Intel 彙編格式中,立即數的表示不用帶任何前綴。

AT&T

Intel

說明

movl $_value,%ebx

mov eax,_value

Intel的立即數前面不帶$符號

movl $0xd00d,%ebx

mov ebx,0xd00d

規則同樣適用於16進制的立即數


4、操作數長度標識

AT&T 彙編格式中,操作數的字長由操作符的最後一個字母決定,後綴'b''w''l'分別表示操作數爲字節(byte8 比特)、字(word16 比特)和長字(long32比特);而在 Intel 彙編格式中,操作數的字長是用 "byte ptr" "word ptr" 等前綴來表示的。

AT&T

Intel

說明

movw %ax,%bx

mov bx,ax

Intel的彙編中, 操作數的長度並不通過指令符號來標識

在AT&T的格式中, 每個操作都有一個字符後綴, 表明操作數的大小. 例如:mov指令有三種形式:

movb  傳送字節

movw  傳送字

movl   傳送雙字

因爲在許多機器上, 32位數都稱爲長字(long word), 這是沿用以16位字爲標準的時代的歷史習慣造成的.

---------摘自《深入理解計算機系統》


5、尋址方式

AT&T

Intel

imm32(basepointer,indexpointer,indexscale)

[basepointer + indexpointer*indexscale + imm32)

兩種尋址的實際結果都應該是

imm32 +basepointer + indexpointer*indexscale

舉例:

AT&T 格式

Intel 格式

movl -4(%ebp), %eax

mov eax, [ebp - 4]

movl array(, %eax, 4), %eax

mov eax, [eax*4 + array]

movw array(%ebx, %eax, 4), %cx

mov cx, [ebx + 4*eax + array]

movb $4, %fs:(%eax)

mov fs:eax, 4

6、跳轉指令

AT&T 彙編格式中,絕對轉移和調用指令(jump/call)的操作數前要加上'*'作爲前綴,而在 Intel 格式中則不需要。

遠程轉移指令和遠程子調用指令的操作碼,在 AT&T 彙編格式中爲 "ljump" "lcall",而在 Intel 彙編格式中則爲 "jmp far" "call far",即:

AT&T 格式ljump $section, $offset

Intel 格式 jmp far section:offset

AT&T
格式lcall $section, $offset
Intel
格式call far section:offset
 

與之相應的遠程返回指令則爲:

AT&T 格式:lret $stack_adjust
 Intel
格式: ret far stack_adjust
 

AT&T的彙編格式中, 跳轉指令有點特殊.

直接跳轉, 即跳轉目標是作爲指令的一部分編碼的.

       例如: jmp Label_1

間接跳轉, 即跳轉目標是從寄存器或存儲器位置中讀出的. 寫法是在" * "後面跟一個操作數指示符.

       例如: jmp *%eax 用寄存器%eax中的值作爲跳轉目標

                jmp *(%eax) 以%eax中的值作爲讀入的地址, 從存儲器中讀出跳轉目標

--------摘自《深入理解計算機系統》

 

下面是一些尋址的例子:

AT&T: `-4(%ebp)'        相當於 Intel: `[ebp - 4]'

AT&T: `foo(,%eax,4)' 相當於 Intel: `[foo + eax*4]'

AT&T: `foo(,1)'          相當於 Intel `[foo]'

AT&T: `%gs:foo'          相當於 Intel`gs:foo' 
例子摘自http://sourceware.org/binutils/docs/as/i386_002dMemory.html#i386_002dMemory

 

發佈了19 篇原創文章 · 獲贊 16 · 訪問量 10萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章