笔者最近参与了一个项目,需要用嵌入式汇编写测试程序。汇编程序写过,C程序也写过,但是将它们混合起来写还是第一次,完全没有概念。只能上网找资料,自己慢慢摸索。 自己一直用Windows操作系统,但是我们的项目是在Linux系统上,这两者的汇编语言系统是不同的,Windows一直是Intel汇编,而Linux是At&T汇编。为了调试的方便,我决定现在Windows中尝试着写点嵌入式汇编,然后再在Linux中写(主要原因是自己对Linux调试不熟悉)。 首先遇到的第一个问题:格式到底是什么。上网查资料,千篇一律,都是_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嵌入式汇编的特点是:
从上面的叙述中,可以看出貌似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下嵌入式汇编的特点是: 汇编语言是AT&T,源操作数和目的操作数位置和Intel汇编相反,关于AT&T汇编细节请参考其他文献。 嵌入式汇编的关键字也并不是_asm_ _volatile_,自己摸索出的正确的方式是:asm volatile,这个还是和自己机器的配置有个,如果该关键字不正确,大家可以测试其他类似的关键字,如asm_ ,__asm等,volatile一样。 Linux下嵌入式汇编有输入输出和修改标识,放在汇编代码的末尾,用冒号隔开,第一个冒号后是输入参数,格式是“=r”(C语言变量,从汇编代码获得结果到C语言),其中引号中的字母有很多种格式,请参看其它文章,第二个冒号之后是输入,格式是“r”(C语言定义的变量,在汇编语言中应用),字母含义和输出相同。第三个冒号之后表示内存或者某个寄存器会在汇编执行过程中改变,让编译器处理; 由于有输入和输出选项,所以要在汇编代码中用匹配限制符来指代输入输出的值,格式就是%0---9,最多允许有10个输入输出,顺序就是按输出参数,然后输入参数排下来。 因为匹配限制符用到了%,而AT&T汇编的寄存器本身也需要%,为了区别,就需要在寄存器前用两个%,表示寄存器。 汇编语句要用小括号包围,而且最后还要有分号。 如果按上面代码格式,在每一句汇编代码之后都要有'/n'换行符。 以上是笔者写嵌入式汇编的经验,希望对遇到同样问题的朋友有所帮助。 想学汇编,又不知从哪下手的,来“21ic公开课”吧,“零基础学习汇编语言合集(上中下)“,全免费! |
|
来自: 西北望msm66g9f > 《培训》