http://blog.fishc.com/671.html
让编程改变世界
Change the world by program
使用MASM
经过上一讲的准备工作,相信大家已经搭建好了 Win32 汇编的工作环境,并已经知道编译、链接一个程序的过程和原理了。
现在,我们让例子回归到经典:
#include <stdio.h>
int main(void)
{
Printf(“Hello, worldn”);
} // 事实上想想,这不正是初生的婴儿?!
#include <stdio.h>
int main(void)
{
Printf(“Hello, worldn”);
} // 事实上想想,这不正是初生的婴儿?!
Win32汇编源程序的结构
麻雀虽小,五脏俱全。刚刚那个C语言的”Hello, world”程序包含了C语言中的最基本的格式。
在C语言的源程序中,我们不需要为堆栈段、数据段和代码段的定义而烦恼,编译器会自己解决。
回顾一下,在DOS 下的汇编这段代码会变成什么样?
在例子中我们看到,stack、data、code都找到了自己的小窝。
回归主题,在Win32 汇编语言下,小麻雀”Hello World” 又会变成什么样子呢?
是不是又不同了?但是,我们怎么就发觉Win32 汇编其实是前边两种形态的集大成者?!
接下来,小甲鱼带大家逐段来理解和接受这个新先的语言!
模式定义
程序的第一部分是模式和源程序格式的定义语句
.386 .model flat,stdcall option casemap:none
这些指令定义了程序使用的指令集、工作模式和格式。
1)指定使用的指令集
.386语句是汇编语句的伪指令,类似的指令还有:.8086、.186、.286、.386/.386p、.486/.486p和.586/.586p等,用于告诉编译器在本程序中使用的指令集。
在DOS的汇编中默认使用的是8086指令集,那时候如果在源程序中写入80386所特有的指令或使用32位的寄存器就会报错。
Win32环境工作在80386及以上的处理器中,所以这一句.386是必不可少的。
另外,后面带p的伪指令则表示程序中可以使用特权指令,如:mov cr0,eax
这一类指令必须在特权级0上运行,如果只指定.386,那么使用普通的指令是可以的,编译时到这一句就会报错。
如果我们要写的程序是VxD等驱动程序,中间要用到特权指令,那么必须定义.386p,在应用程序级别的Win32编程中,程序都是运行在优先级3上,不会用到特权指令,只需定义.386就够了。
80486和Pentium处理器指令是80386处理器指令的超集,同样道理,如果程序中要用80486处理器或Pentium处理器的指令,则必须定义.486或.586。
另外,Intel公司的80×86系列处理器从Pentium MMX开始增加了MMX指令集,为了使用MMX指令,除了定义.586之外,还要加上一句.mmx伪指令:
.386 .mmx
2)model语句
.model语句在低版本的宏汇编中已经存在,用来定义程序工作的模式,它的使用方法是:
.model 内存模式 [,语言模式] [,其他模式]
内存模式的定义影响最后生成的可执行文件,可执行文件的规模从小到大,可以有很多种类型。
详见下表:
Windows?程序运行在保护模式下,系统把每一个Win32应用程序都放到分开的虚拟地址空间中去运行。
也就是说,每一个应用程序都拥有其相互独立的4GB地址空间。
对Win32程序来说,只有一种内存模式,即flat(平坦)模式,意思是内存是很平坦地从0延伸到4GB,再没有64KB段大小限制。
对比一下DOS的Hello World和Win32的Hello World开始部分的不同,DOS程序中有这样语句
mov ax,data mov ds,ax
意思是把数据段寄存器DS指向data数据段,data数据段在前面已经用data segment语句定义,只要DS不重新设置,那么从此以后指令中涉及的数据默认将从data数据段中取得。
所以下面的语句是从data数据段取出szHello字符串的地址后再显示:
mov ah,9 mov dx,offset szHello int 21h
纵观Win32汇编的源程序,没有一处可以找到ds或es等段寄存器的使用。
因为所有的4GB空间用32位的寄存器全部都能访问到了,不必在头脑中随时记着当前使用的是哪个数据段,这就是平坦内存模式带来的好处。
如果定义了.model flat,MASM自动为各种段寄存器做了如下定义:
ASSUME cs:FLAT, ds:FLAT, ss:FLAT, es:FLAT, fs:ERROR, gs:ERROR
也就是说,CS,DS,SS和ES段全部使用平坦模式,FS和GS寄存默认不使用,这时若在源程序中使用FS或GS,在编译时会报错。
如果有必要使用它们,只需在使用前用下面的语句声明一下就可以了:
assume fs:nothing, gs:nothing
或者
assume fs:flat, gs:flat
在Win32汇编中,.model语句中还应该指定语言模式,即子程序和调用方式。
例子中用的是stdcall,它指出了调用子程序或Win32 API时参数传递的次序和堆栈平衡的方法。
相对于stdcall,不同的语言类型还有C, SysCall, BASIC, FORTRAN 和PASCALL,虽然各种高级语言在调用子程序时都是使用堆栈来传递参数。
Windows的API调用使用是的stdcall格式,所以在Win32汇编中没有选择,必须在.model中加上stdcall参数。
话题:理解stdcall和cdecl
(1)_stdcall调用
_stdcall是Pascal程序的缺省调用方式,参数采用从右到左的压栈方式,被调函数自身在返回前清空堆栈。WIN32 Api都采用_stdcall调用方式。
(2)_cdecl调用
_cdecl是C/C++的缺省调用方式,参数采用从右到左的压栈方式,传送参数的内存栈由调用者维护。
_cedcl约定的函数只能被C/C++调用,每一个调用它的函数都包含清空堆栈的代码,所以产生的可执行文件大小会比调用_stdcall函数的大。