首先遇到的第一个问题是,格式到底是什么。上网查资料,前篇一律,都是_asm_ _volatile_(…);格式,但是自己弄的时候总是不行,说明网络虽好,但是不可盲目相信。经过多番测试,我发现windows下嵌入式汇编的格式只有_asm关键字,也没有volatile;以前的括号现在也变成大括号,最后也没有引号。我自己写的一个简单测试程序为:
#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嵌入式汇编的特点是:
1. 汇编是Intel汇编,而不是AT&T汇编;
2. 关键字是_asm,而不是其他,这个有可能和自己的环境有关,如果你自己重新定义宏,那就是你自己定义的关键字;
3. 虽说有volatile关键字,但是用上却报错,具体我也不知道怎么用;
4. 没有冒号指定输入输出,要输入和输出的数可以直接在汇编语言中使用;
5. 汇编代码要用大括号包围,而不是小括号,而且最后也没有分号;
6. 如果不用大括号包围,则必须每句汇编代码之前都有_asm关键字。
从上面的叙述中,可以看出貌似windows嵌入式汇编要简单很多。
在windows中搞定之后,就要在Linux中进行测试。我首先做的就是直接复制在windows中的程序到Linux中,很明显自己的无知是要报错的。
有经过多番测试,得到在Linux中执行无误的代码为:
#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;
}
从结构来看,Linux下的要麻烦不少。首先用的汇编是windows程序员不熟悉的AT&T汇编,而且还要定义输入输出,寄存器用起来还要两个%……总之,比较麻烦。但是我没有贬低Linux的意思,只要用好了,功能还是很强大的。
Linux下嵌入式汇编的特点是:
1. 汇编语言是AT&T,源操作数和目的操作数位置和Intel汇编相反,关于AT&T汇编细节请参考其他文献;
2. 嵌入式汇编的关键字也并不是_asm_ _volatile_,自己摸索出的正确的方式是:asm volatile,这个还是和自己机器的配置有个,如果该关键字不正确,大家可以测试其他类似的关键字,如asm_ ,__asm等,volatile一样;
3. Linux下嵌入式汇编有输入输出和修改标识,放在汇编代码的末尾,用冒号隔开,第一个冒号后是输入参数,格式是”=r”(c语言变量,从汇编代码获得结果到c语言),其中引号中的字母有很多种格式,请参看其它文章,第二个冒号之后是输入,格式是”r”(C语言定义的变量,在汇编语言中应用),字母含义和输出相同;第三个冒号之后表示内存或者某个寄存器会在汇编执行过程中改变,让编译器处理;
4. 由于有输入和输出选项,所以要在汇编代码中用匹配限制符来指代输入输出的值,格式就是%0---9,最多允许有10个输入输出,顺序就是按输出参数,然后输入参数排下来;
5. 因为匹配限制符用到了%,而AT&T汇编的寄存器本身也需要%,为了区别,就需要在寄存器前用两个%,表示寄存器;
6. 汇编语句要用小括号包围,而且最后还要有分号;
7. 如果按上面代码格式,在每一句汇编代码之后都要有”/n”换行符
以上是我写嵌入式汇编的经验,希望对遇到同样问题的朋友有所帮助。