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