筆者最近參與了一個項目,需要用嵌入式匯編寫測試程序。匯編程序?qū)戇^,C程序也寫過,但是將它們混合起來寫還是第一次,完全沒有概念。只能上網(wǎng)找資料,自己慢慢摸索。
自己一直用Windows操作系統(tǒng),但是我們的項目是在Linux系統(tǒng)上,這兩者的匯編語言系統(tǒng)是不同的,Windows一直是Intel匯編,而Linux是At&T匯編。為了調(diào)試的方便,我決定現(xiàn)在Windows中嘗試著寫點嵌入式匯編,然后再在Linux中寫(主要原因是自己對Linux調(diào)試不熟悉)。
首先遇到的第一個問題:格式到底是什么。上網(wǎng)查資料,千篇一律,都是_asm_ _volatile_(…); 格式,但是自己弄的時候總是不行。說明網(wǎng)絡(luò)雖好,但是不可盲目相信。
經(jīng)過多番測試,我發(fā)現(xiàn)Windows下嵌入式匯編的格式只有_asm關(guān)鍵字,也沒有volatile;,以前的括號現(xiàn)在也變成大括號,最后也沒有引號。我自己寫的一個簡單測試程序為:
#include
#include
//測試bsf指令
void test()
{
int value,index=32;
_asm {
mov eax,index
bsf ebx,eax
mov value,ebx
}
printf('value is %d/n',value);
}
void main()
{
printf('Hello World/n');
test();
}
Windows嵌入式匯編的特點是:
匯編是Intel匯編,而不是AT&T匯編;
關(guān)鍵字是_asm,而不是其他,這個有可能和自己的環(huán)境有關(guān),如果你自己重新定義宏,那就是你自己定義的關(guān)鍵字;
雖說有volatile關(guān)鍵字,但是用上卻報錯,具體我也不知道怎么用;
沒有冒號指定輸入輸出,要輸入和輸出的數(shù)可以直接在匯編語言中使用;
匯編代碼要用大括號包圍,而不是小括號,而且最后也沒有分號;
如果不用大括號包圍,則必須每句匯編代碼之前都有_asm關(guān)鍵字。
從上面的敘述中,可以看出貌似Windows嵌入式匯編要簡單很多。
在Windows中搞定之后,就要在Linux中進(jìn)行測試。我首先做的就是直接復(fù)制在Windows中的程序到Linux中,很明顯自己的無知是要報錯的。
有經(jīng)過多番測試,得到在Linux中執(zhí)行無誤的代碼為:
#include
void test()
{
int bsf=0,bsr=0,input=0x12345,flag;
asm volatile ('movl %3,%%eax/n'
'movl %3,%%ebx/n'
'bsf %3,%%cx/n'
'movzwl %%cx,%0/n'
'bsr %3,%%dx/n'
'movzwl %%dx,%1/n'
'pushf/n'
'pop %2/n'
:'=r'(bsf),'=r'(bsr),'=r'(flag)
:'m'(input)
);
printf('bsf value is %d/n',bsf);
printf('bsr value is %d/n',bsr);
printf('flag is %x/n',flag);
}
int main()
{
printf('Hello World/n');
test();
return 0;
}
從結(jié)構(gòu)來看,Linux下的要麻煩不少。首先用的匯編是Windows程序員不熟悉的AT&T匯編,而且還要定義輸入輸出,寄存器用起來還要兩個%……總之,比較麻煩。但是我沒有貶低Linux的意思,只要用好了,功能還是很強(qiáng)大的。
Linux下嵌入式匯編的特點是:
匯編語言是AT&T,源操作數(shù)和目的操作數(shù)位置和Intel匯編相反,關(guān)于AT&T匯編細(xì)節(jié)請參考其他文獻(xiàn)。
嵌入式匯編的關(guān)鍵字也并不是_asm_ _volatile_,自己摸索出的正確的方式是:asm volatile,這個還是和自己機(jī)器的配置有個,如果該關(guān)鍵字不正確,大家可以測試其他類似的關(guān)鍵字,如asm_ ,__asm等,volatile一樣。
Linux下嵌入式匯編有輸入輸出和修改標(biāo)識,放在匯編代碼的末尾,用冒號隔開,第一個冒號后是輸入?yún)?shù),格式是“=r”(C語言變量,從匯編代碼獲得結(jié)果到C語言),其中引號中的字母有很多種格式,請參看其它文章,第二個冒號之后是輸入,格式是“r”(C語言定義的變量,在匯編語言中應(yīng)用),字母含義和輸出相同。第三個冒號之后表示內(nèi)存或者某個寄存器會在匯編執(zhí)行過程中改變,讓編譯器處理;
由于有輸入和輸出選項,所以要在匯編代碼中用匹配限制符來指代輸入輸出的值,格式就是%0---9,最多允許有10個輸入輸出,順序就是按輸出參數(shù),然后輸入?yún)?shù)排下來。
因為匹配限制符用到了%,而AT&T匯編的寄存器本身也需要%,為了區(qū)別,就需要在寄存器前用兩個%,表示寄存器。
匯編語句要用小括號包圍,而且最后還要有分號。
如果按上面代碼格式,在每一句匯編代碼之后都要有'/n'換行符。
以上是筆者寫嵌入式匯編的經(jīng)驗,希望對遇到同樣問題的朋友有所幫助。
想學(xué)匯編,又不知從哪下手的,來“21ic公開課”吧,“零基礎(chǔ)學(xué)習(xí)匯編語言合集(上中下)“,全免費!