嵌入式匯編語言學(xué)習(xí)經(jīng)驗(yàn)
最近參與了一個(gè)項(xiàng)目,需要用嵌入式匯編寫測(cè)試程序。匯編程序?qū)戇^,C程序也寫過,但是將它們混合起來寫還是第一次,完全沒有概念。只能上網(wǎng)找資料,自己慢慢摸索。
本文引用地址:http://cafeforensic.com/article/201904/399826.htm自己一直用Windows操作系統(tǒng),但是我們的項(xiàng)目是在Linux系統(tǒng)上,這兩者的匯編語言系統(tǒng)是不同的,windows一直是Intel匯編,而Linux是At&T匯編。為了調(diào)試的方便,我決定現(xiàn)在windows中嘗試著寫點(diǎn)嵌入式匯編,然后再在Linux中寫(主要原因是自己對(duì)Linux調(diào)試不熟悉)。
Windows嵌入式匯編
首先遇到的第一個(gè)問題是,格式到底是什么。上網(wǎng)查資料,千篇一律,都是_asm_ _volatile_(…);格式,但是自己弄的時(shí)候總是不行,說明網(wǎng)絡(luò)雖好,但是不可盲目相信。經(jīng)過多番測(cè)試,我發(fā)現(xiàn)windows下嵌入式匯編的格式只有_asm關(guān)鍵字,也沒有volatile;以前的括號(hào)現(xiàn)在也變成大括號(hào),最后也沒有引號(hào)。我自己寫的一個(gè)簡單測(cè)試程序?yàn)椋?/p>
#include
#include
//測(cè)試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嵌入式匯編的特點(diǎn)是:
1. 匯編是Intel匯編,而不是AT&T匯編;
2. 關(guān)鍵字是_asm,而不是其他,這個(gè)有可能和自己的環(huán)境有關(guān),如果你自己重新定義宏,那就是你自己定義的關(guān)鍵字;
3. 雖說有volatile關(guān)鍵字,但是用上卻報(bào)錯(cuò),具體我也不知道怎么用;
4. 沒有冒號(hào)指定輸入輸出,要輸入和輸出的數(shù)可以直接在匯編語言中使用;
5. 匯編代碼要用大括號(hào)包圍,而不是小括號(hào),而且最后也沒有分號(hào);
6. 如果不用大括號(hào)包圍,則必須每句匯編代碼之前都有_asm關(guān)鍵字。
Linux下嵌入式匯編
從上面的敘述中,可以看出貌似windows嵌入式匯編要簡單很多。
在windows中搞定之后,就要在Linux中進(jìn)行測(cè)試。我首先做的就是直接復(fù)制在windows中的程序到Linux中,很明顯自己的無知是要報(bào)錯(cuò)的。
有經(jīng)過多番測(cè)試,得到在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匯編,而且還要定義輸入輸出,寄存器用起來還要兩個(gè)%……總之,比較麻煩。但是我沒有貶低Linux的意思,只要用好了,功能還是很強(qiáng)大的。
Linux下嵌入式匯編的特點(diǎn)是:
1 匯編語言是AT&T,源操作數(shù)和目的操作數(shù)位置和Intel匯編相反,關(guān)于AT&T匯編細(xì)節(jié)請(qǐng)參考其他文獻(xiàn);
2. 嵌入式匯編的關(guān)鍵字也并不是_asm_ _volatile_,自己摸索出的正確的方式是:asm volatile,這個(gè)還是和自己機(jī)器的配置有個(gè),如果該關(guān)鍵字不正確,大家可以測(cè)試其他類似的關(guān)鍵字,如asm_ ,__asm等,volatile一樣;
3. Linux下嵌入式匯編有輸入輸出和修改標(biāo)識(shí),放在匯編代碼的末尾,用冒號(hào)隔開,第一個(gè)冒號(hào)后是輸入?yún)?shù),格式是”=r”(c語言變量,從匯編代碼獲得結(jié)果到c語言),其中引號(hào)中的字母有很多種格式,請(qǐng)參看其它文章,第二個(gè)冒號(hào)之后是輸入,格式是”r”(C語言定義的變量,在匯編語言中應(yīng)用),字母含義和輸出相同;第三個(gè)冒號(hào)之后表示內(nèi)存或者某個(gè)寄存器會(huì)在匯編執(zhí)行過程中改變,讓編譯器處理;
4. 由于有輸入和輸出選項(xiàng),所以要在匯編代碼中用匹配限制符來指代輸入輸出的值,格式就是%0---9,最多允許有10個(gè)輸入輸出,順序就是按輸出參數(shù),然后輸入?yún)?shù)排下來;
5. 因?yàn)槠ヅ湎拗品玫搅?,而AT&T匯編的寄存器本身也需要%,為了區(qū)別,就需要在寄存器前用兩個(gè)%,表示寄存器;
6. 匯編語句要用小括號(hào)包圍,而且最后還要有分號(hào);
7. 如果按上面代碼格式,在每一句匯編代碼之后都要有”/n”換行符以上是我寫嵌入式匯編的經(jīng)驗(yàn),希望對(duì)遇到同樣問題的朋友有所幫助。
評(píng)論