内联汇编基础知识 作者:释雪
#define randasm __asm _emit 0x4A __asm _emit 0x43 __asm _emit 0x4B …__asm { randasm}
内联汇编不可以是一个宏汇编程序,你不可以使用MASM中的宏定义指令以及宏操作符。但内联汇编是可以使用C\C++中的预理指令来定义宏。
在处理段时,你只能使用寄存器,而不是通过名字,因为在内联汇编中这是非法的。而且段必须显式地使用寄存器,如: ES:[BX]
在内联汇编使用操作符LENGTH, SIZE, 和 TYPE可以来对变量以及类型进行长度的测量,你可以使用它们来求得C\C++中的变量及类型的长度:
*LENGTH操作符可以返回在一个变量数组中的元素个数,如果返回为1则代表这不是一个变量数组。
*SIZE操作符可以求得一个变量及类型的总长度。这个值也可以由LENGTH与TYPE积来求得。
*TYPE操作符可以求得一个变量及类型的长度,与SIZE不同的是,如果变量名是一个数组的话,则返加这个数组中单个元素的长度。
具体情况请看下表:
包含着内联汇编的程序可以使用/Zi选项编译,从而来进行代码级的调试工作。这样,你就可以同时在C\C++程序段与内联汇编段中设置断点,如果你使用/Fas选项允许混合汇编与C\C++源程序调试方式,那么你就可以看到混合着汇编与源程序的代码集合。
Visual C++编译器允许你在内联汇编程序中使用Intel处理器的MMX指令集。不过如果使用MMX指令集编译器会发生警告。更多的信息请查看MSDN的Web站点。
二、 关于内联汇编的具体使用说明: 因为内联汇编不需要编译与链接过程,因此它比一个汇编程序更为方便。由于它能够访问C\C++中的变量及函数,所以它能够更好和你C\C++代码融为一体。内联汇编可以在以下方面进行编程:
如果你计划在不同机器上运行程序的话,那么你应该分别放置不同机种的专一汇编代码。因为内联汇编有一定的机器专一性,它不完全支持所有MASM中的宏与数据类型。
VC不支持C++中的asm关键字,所以你需要使用__asm(两个下划线)关键字来编写你的内联汇编代码。
你可以使用这个关键字来编写一个内联代码段,如:
__asm{ mov al, 2 mov dx, 0xD007 out al, dx}
__asm mov al, 2__asm mov dx, 0xD007__asm out al, dx
int array[10];__asm mov array[6], 0 ; 期望达到C中的array[6] = 0功能,但这是错误的。
__asm mov array[6 * TYPE int], 0 ; array[6] = 0;
__asm mov eax, var ;将变量var中的值赋给eax寄存器中。
struct first_type{ char *weasel; int same_name;} hal;
__asm{ mov ebx, OFFSET hal ;取得hal对象的首地址 mov ecx, [ebx]hal.same_name ;加上same_name偏移值,则可以访问到成员same_name mov esi, [ebx]hal.weasel ;加上weasel偏移值。}
#include <stdio.h>int power2( int num, int power );void main( void ){ printf( "3 times 2 to the power of 5 is %d\n", \ power2( 3, 5) );}int power2( int num, int power ){ __asm { mov eax, num ; 取得第一个参数 mov ecx, power ; 取得第二个参数 shl eax, cl ; EAX = EAX * CL } //在函数中,返回值是由eax负责往回传递的。(顺便问一句ax与eax有什么不同啊?是不是一样的?)}
void func( void ){ goto C_Dest; /* Legal: correct case */ goto c_dest; /* Error: incorrect case在C中大小写区分。*/ goto A_Dest; /* Legal: correct case */ goto a_dest; /* Legal: incorrect case */ __asm { jmp C_Dest ; Legal: correct case jmp c_dest ; Legal: incorrect case jmp A_Dest ; Legal: correct case jmp a_dest ; Legal: incorrect case a_dest: ; __asm label } C_Dest: /* C label */ return;}
__asm{ mov eax,2; push; 参数b等于2 mov eax,3; push; 参数a等于3 call CAdd;调用CAdd函数 mov Result,eax;所有函数的返回值都被存放在eax。于是,Result等于5}
#define PORTIO __asm \/* Port output */ \{ \ __asm mov al, 2 \ __asm mov dx, 0xD007 \ __asm out al, dx \}
#include <iostream.h>struct MyData{ int nMember1; int * lpMember2;}; void main(){ MyData sample; __asm//这是对成员变量赋值 { mov eax,12; mov sample.nMember1,eax; } cout <<sample.nMember1<<endl; __asm//这是对成员指针赋值 { lea eax,sample.nMember1; mov sample.lpMember2,eax; } cout <<*sample.lpMember2<<endl; __asm//这是对指针所指向的变量赋值 { mov ebx,sample.lpMember2; mov eax,5; mov [ebx],eax; } cout <<sample.nMember1<<endl;}