win32汇编 之2 009

http://blog.fishc.com/671.html

让编程改变世界

Change the world by program


 

使用MASM

 

经过上一讲的准备工作,相信大家已经搭建好了 Win32 汇编的工作环境,并已经知道编译、链接一个程序的过程和原理了。

现在,我们让例子回归到经典:




#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函数的大。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值