不会开机的男孩

Win32汇编学习(1)

| Comments

终于把学校里面让人无语的论文搞定了。周末终于有时间干些自己想干的事了。想起了这2周做的关于编译原理的实验,代码优化这部分的确是个难题。哎,我实在是太笨了,其实答案就在自己电脑里。将c程序反汇编后,终于第一次感受到了debug 和release的区别。兴奋之余让我又产生了忧虑,自己汇编的能力太弱了,面对稍微复杂点的算法再加上编译优化后的汇编代码,真是欲哭无泪。痛下决心,准备好好学学汇编了。为了给自己一个动力,准备学习Win32汇编(和学校那个8086再见了)。

  第一天,不准备上难度了。从最简单的”HelloWorld”开始。

; HelloWorld.asm

comment * ----------------------------------------------
                 The First Assemble Application
                ---------------------------------------------- *

.386
.model flat, stdcall
option casemap:none

include \masm32\include\windows.inc

include \masm32\include\user32.inc
include \masm32\include\kernel32.inc

includelib \masm32\lib\user32.lib
includelib \masm32\lib\kernel32.lib

    .data
szCaption   db 'MessageBox', 0
szText      db 'Hello, World!', 0

    .code
start:
    invoke MessageBox,\     ; 调用函数名
        NULL,\                      ; 父窗口句柄
        offset szText,\           ; 文字
        offset szCaption,\      ; 标题
        MB_OK                    ; 按钮类型

    invoke ExitProcess, NULL
end start

一个简单的窗口就创建好了。一眼看上去发现和以前的8086还是有很多不同的。

首先8086和80386在寻址方式不同。

8086 通过 段地址*0x10 +偏移地址确定的。只能寻址1M,而80386 32根地址线寻址,空间达到了4G而且80386 通用寄存器大小为32位,所以不需要分段就能访问到地址。

那么.data,.code不是段的意思么? 不是。因为808386有分页机制,每个页可以自由制定属性,已经和8086代码和数据分段处理完全不同,实际上是把不同类型的数据或代码归类,再放到不同属性的内存页。

其次,8086的不安全,不方便的调用中断的方式改为了调用系统API的方式。我这里是采用的MASM,所以有invoke伪指令,其实这个指令也没别的,就是为我们编程省去了函数参数入栈的那些push。

最后,发现win32汇编和使用c/c++似乎没有复杂很多。

Comments