Fasm -- Win32汇编学习--15
来源:互联网 发布:php开源手机商城系统 编辑:程序博客网 时间:2024/04/27 17:02
Fasm -- Win32汇编学习--15
课中我们将要讲解内存映射文件并且演示如何运用它。您将会发现使用内存映射文件是非常简单的。
如果您仔细研究了前一节课的例子,就会发现一个严重的缺陷:如果您想读取的内容大于系统分配的内存块该怎么办? 如果您想搜索的字符串刚好超过内存块的边界又该如何处理? 对于第一个问题,您也许会说,只要不断地的读不久解决了吗? 至于第二个问题,您又会说在内存块的边界处做一些特别的处理,例如放上一些标志就可以了。 原理上确实可以行的通,但是随着问题的复杂程度加深而显得非常难以处理。 其中第二个问题是有名的边界判断问题,程序中许许多多的错误都是由此引起。想一想,如果我们能够分配一个能够容纳整个文件的大内存块该多好啊,这样两个问题不都迎刃而解了吗? 是的,Win32的内存映射文件确实允许我们分配一个装的下现实中可能存在的足够大的文件的内存。
(译者注: 也就是说我们可以通过内存映射的机制将整个文件的大小,直接翻到用户地址空间中,我们不需要去考虑这个文件多大,或者分配的内存块多大,操作系统会帮我们完成这些事情,并且并不是说我们只能映射整个文件到用户地址空间,我们还可以映射部分等,)
利用内存映射文件您可以认为操作系统已经把文件全部装入了内存(译者注:其实本身就是操作系统将文件全部映射到我们的用户地址空间上),此时已经映射到我们的地址空间商了,我们只要操作相应返回的内存块的指针我们就可以在这些空间商修改任意数据,嘿嘿。 当然前提是你在创建文件映射对象时候要设置可读可写的属性。(译者注:米办法,windows 利用系统空间的特性,以保护这些对象的安全性)。这样您甚至不需要调用那些分配、释放内存块、和文件输入/输出的api函数,另外你可以把这用作不同进程共享数据的一种方法。运用内存映射文件实际上并没有进行实际的文件操作,它更像为每个进程保留一个看的见的内存空间。至于把文件映射文件当成进程间共享数据的办法来用,则要“加倍小心”。因为你不得不处理数据的同步问题,否则你得到的将是过时的或者是错误的数据甚至崩溃。本课中我们将主要讲述内存映射文件,将不设计进程间额度同步。Win32的内存映射文件使用非常广泛,譬如即使是系统的核心模块----pe格式文件装载器也用到了内存映射文件,因为pe格式的文件并不是一次性加载到内存来的,譬如它在首次加载时只加载必须加载的部分,而其他部分在用到时在加载,这正好用到了内存映射文件的长处。实际商大多数文件存取都和PE加载器类似,所以您在处理该类文件也应该充分利用内存映射文件。
内存映射文件本身也有局限性,譬如你一旦生成了一个内存映射文件,所以您在那个会话期间是不能修改其大小的。所以内存映射文件对于只读文件和不会影响其大小的文件操作是非常有用的。当然这并不意味着会引起文件大小的文件操作就不能用内存映射文件,您可以事先估计一下操作后生成的文件大小,然后生成这么大一块的内存映射文件,然后文件的长度就可以增加到这么一个大小。我们解释的够多了,接下来看看实现的细节。
1.调用CreateFile打开您想要映射的文件。
2.调用CreateFileMapping,其中要求传入先前CreateFile返回的句柄,该函数生成一个建立在CreateFile函数创建的文件对象基础上的内存映射对象。
3.调用MapViewOfFile函数映射整个文件的一个区域或者整个文件到内存。该函数返回指向映射到内存的第一个字节的指针,用该指针来读写文件。
4.调用UnmapViewOfFile来解除文件映射。
5.调用CloseHandle来关闭内存映射文件。注意必须传入内存映射文件的句柄。
6.调用CloseHandle来关闭文件。注意必须传入由CreateFile创建的文件的句柄。
例子:
下面的例子允许用户通过“打开文件”对话框来打开一个文件,然后用内存映射文件来打开该文件,如果成功,窗口的标题条会显示打开的文件的名称,您可以通过选择“File/Save”菜单项来把换名保存。该程序将会把打开的文件的内容存到新文件中去。注意,这整个过程您根本就没有用到GlobalAlloc这样的分配内存的函数。
format PE GUI 4.0
include 'win32ax.inc'
IDM_OPEN equ 1000
IDM_SAVE equ 1001
IDM_EXIT equ 1002
EDITID equ 1003
.data
hInstance rd 1
hEdit rd 1
hFile rd 1
hFileWrite rd 1
hMapFile rd 1
hMenu rd 1
lpMemory rd 1
lpSizeWrite rd 1
szClassName db 'xfish', 0
szWndName db '偶的程序', 0
szEditName db 'edit', 0
lpFilterName db 'all file (*.*)', 0, '*.*', 0, 0
szBuffer db 100 dup (?)
lpOfn OPENFILENAME <>
.code
entry $
invoke GetModuleHandle, NULL
mov [hInstance], eax
stdcall _WinMain, [hInstance], NULL, NULL, NULL
invoke ExitProcess, NULL
proc _WinMain hInstance:DWORD, hPrevInstance:DWORD, lpCmdLine:DWORD, nCmdShow:DWORD
local @wc:WNDCLASS
local @msg:MSG
local @hWin:DWORD
; +-------------------------------------------+
; | Register Class |
; +-------------------------------------------+
invoke RtlZeroMemory, addr @wc, sizeof.WNDCLASS
mov [@wc.style], CS_VREDRAW or CS_HREDRAW
mov [@wc.lpfnWndProc], _WndProc
memmov @wc.hInstance, hInstance
mov [@wc.hbrBackground], COLOR_WINDOW + 1
mov [@wc.lpszMenuName], 30
mov [@wc.lpszClassName], szClassName
invoke LoadIcon, NULL, IDI_ASTERISK
mov [@wc.hIcon], eax
invoke LoadCursor, NULL, IDC_ARROW
mov [@wc.hCursor], eax
invoke RegisterClass, addr @wc
; +------------------------------------------+
; | Create Window |
; +------------------------------------------+
invoke CreateWindowEx,/
NULL,/
szClassName,/
szWndName,/
WS_OVERLAPPEDWINDOW,/
CW_USEDEFAULT,/
CW_USEDEFAULT,/
300,/
200,/
NULL,/
NULL,/
[hInstance],/
NULL
mov [@hWin], eax
invoke ShowWindow, [@hWin], SW_SHOWNORMAL
invoke UpdateWindow, [@hWin]
; +-----------------------------------------+
; | Msg Loop |
; +-----------------------------------------+
msgloop:
invoke GetMessage, addr @msg, NULL, NULL, NULL
or eax, eax
je msgend
invoke TranslateMessage, addr @msg
invoke DispatchMessage, addr @msg
jmp msgloop
msgend:
mov eax, [@msg.wParam]
ret
endp
proc _WndProc hWnd:DWORD, uMsg:DWORD, wParam:DWORD, lParam:DWORD
cmp [uMsg], WM_CREATE
je wndcreate
cmp [uMsg], WM_SIZE
je wndsize
cmp [uMsg], WM_COMMAND
je menucmd
cmp [uMsg], WM_DESTROY
je wndend
invoke DefWindowProc, [hWnd], [uMsg], [wParam], [lParam]
ret
wndcreate:
; +-------------------------------------+
; | 创建编辑框 |
; +-------------------------------------+
invoke CreateWindowEx,/
NULL,/
szEditName,/
NULL,/
WS_CHILD or WS_VISIBLE or WS_HSCROLL or/
WS_VSCROLL or ES_MULTILINE or ES_LEFT,/
0, 0, 0, 0,/
[hWnd],/
EDITID,/
[hInstance],/
NULL
mov [hEdit], eax
invoke SetFocus, [hEdit]
invoke GetMenu, [hWnd]
mov [hMenu], eax
; +-------------------------------------+
; | 初始化lpofn |
; +-------------------------------------+
mov [lpOfn.lStructSize], sizeof.OPENFILENAME
memmov lpOfn.hInstance, hInstance
mov [lpOfn.lpstrFilter], lpFilterName
mov [lpOfn.lpstrFile], szBuffer
mov [lpOfn.nMaxFile], 100
jmp quit
wndsize:
push edx
mov eax, [lParam]
movzx edx, ax ;edx = 宽
shr eax, 16 ;eax = 高
invoke MoveWindow, [hEdit], 0, 0, edx, eax, TRUE
invoke GetLastError
pop edx
jmp quit
menucmd:
mov eax, [wParam]
cmp ax, IDM_OPEN
je menuopen
cmp ax, IDM_SAVE
je menusave
cmp ax, IDM_EXIT
je menuexit
jmp quit
menuopen:
mov [lpOfn.Flags], OFN_EXPLORER or OFN_FILEMUSTEXIST or /
OFN_PATHMUSTEXIST or OFN_READONLY
invoke GetOpenFileName, lpOfn
or eax, eax
je quit
invoke CreateFile,/
szBuffer,/
GENERIC_READ,/
NULL,/
NULL,/
OPEN_EXISTING,/
FILE_ATTRIBUTE_ARCHIVE,/
NULL
test eax, -1
je quit
mov [hFile], eax
invoke CreateFileMapping,/
[hFile],/
NULL,/
PAGE_READONLY,/
0,0,NULL
mov [hMapFile], eax
push edx
mov edx, szBuffer
movzx eax, [lpOfn.nFileOffset]
add eax, edx
invoke SetWindowText, [hWnd], eax
pop edx
invoke EnableMenuItem, [hMenu], IDM_OPEN, MF_GRAYED
invoke EnableMenuItem, [hMenu], IDM_SAVE, MF_ENABLED
jmp quit
menusave:
mov [lpOfn.Flags], OFN_EXPLORER or OFN_LONGNAMES or OFN_HIDEREADONLY
invoke GetSaveFileName, lpOfn
or eax, eax
je quit
invoke CreateFile,/
szBuffer,/
GENERIC_WRITE or GENERIC_READ,/
FILE_SHARE_READ or FILE_SHARE_WRITE,/
NULL,/
CREATE_NEW,/
FILE_ATTRIBUTE_ARCHIVE,/
NULL
test eax, -1
je menuerror
mov [hFileWrite], eax
invoke MapViewOfFile,/
[hMapFile],/
FILE_MAP_READ,/
NULL,/
NULL,/
0
or eax, eax
je quit
mov [lpMemory], eax
invoke GetFileSize, [hFile], NULL
invoke WriteFile,/
[hFileWrite],/
[lpMemory],/
eax,/
lpSizeWrite,/
NULL
invoke UnmapViewOfFile, [lpMemory]
invoke CloseHandle, [hFileWrite]
call _CloseHandle
invoke SetWindowText, [hWnd], szWndName
invoke EnableMenuItem, [hMenu], IDM_OPEN, MF_ENABLED
invoke EnableMenuItem, [hMenu], IDM_SAVE, MF_GRAYED
jmp quit
menuerror:
invoke MessageBox, NULL, '失败', 'test', MB_OK
jmp quit
testhandle:
call _CloseHandle
jmp quitmessage
menuexit:
wndend:
cmp [hMapFile], 0
jnz testhandle
quitmessage:
invoke PostQuitMessage, 0
quit:
xor eax, eax
ret
endp
proc _CloseHandle
invoke CloseHandle, [hMapFile]
mov [hMapFile], 0
invoke CloseHandle, [hFile]
ret
endp
.import
library kernel32, 'kernel32.dll',/
user32, 'user32.dll',/
comdlg32, 'comdlg32.dll'
include 'api/kernel32.inc'
include 'api/user32.inc'
include 'api/comdlg32.inc'
section '.rsrc' data readable resource
include 'string.fnt'
directory RT_MENU, Menu
resource Menu, 30,/
LANG_NEUTRAL, MenuMain
menu MenuMain
menuitem str_file_chs, 0, MFR_POPUP + MFR_END
menuitem str_open_chs, 1000, 0
menuitem str_save_chs, 1001, 0
menuseparator
menuitem str_exit_chs, 1002, MFR_END
分析:
invoke CreateFile,/
szBuffer,/
GENERIC_READ,/
NULL,/
NULL,/
OPEN_EXISTING,/
FILE_ATTRIBUTE_ARCHIVE,/
NULL
当用户选择一个文件的时候,我们调用CreateFile函数来打开,注意我们指定GENERIC_READ(一般的读)来表示我们打开的文件只能够读出,把dwShareMode设成0,表示我们不想其他进程在我们操作文件时来存取该文件。
invoke CreateFileMapping,/
[hFile],/
NULL,/
PAGE_READONLY,/
0,0,NULL
mov [hMapFile], eax
然后我们调用CreateFileMapping来创建一个文件映射对象,因为我们windows的各种资源都是以对象的形式来呈现给我们的编程人员的,操作每个对象之前我们必须取得对象的句柄。(我的想法:句柄就是这些对象数据块在系统地址空间的索引值,我们创建一个对象,windows默认在系统空间创建其对象,这些对象拥有一些安全属性等等,这样的好处是为了保护这些对象的安全,以及可以共享其对象)。我们创建一个文件映射对象后,返回的是这个对象的句柄,而这个对象在系统空间创立,并且写入了相关的属性,例如什么权限,属性等。
您应当知道该函数并没有必要把整个文件映射到内存中去,您可以用该函数来只映射文件的一部分。您可以在参数dwMaximumSizeHigh和dwMaximumSizeLow中指定内存映射文件的大小,如果您指定的值大于实际的文件,则实际的文件将增长到指定的大小,如果想要映射的内存大小正好和文件的实际大小相等,则把两个参数中都设成为0。您可以设定lpFileMappingAttributes为NULL,让WINDOWS赋予该内存映射文件于缺省的安全属性。
flProtect定义了内存映射文件的保护属性,我们指定它为PAGE_READONLY来规定该内存映射文件只能够读。注意该属性不能和CreateFile中指定的属性相矛盾,否则就不能生成内存映射文件。
lpName指定内存映射文件的名称,如果您想要该内存映射文件同时可以供其它的进程使用,就必须给它取个名称。不过在我们的例子中,只有我们的进程使用该内存映射文件故我们忽略该参数。
push edx
mov edx, szBuffer
movzx eax, [lpOfn.nFileOffset]
add eax, edx
invoke SetWindowText, [hWnd], eax
pop edx
这句指令是用来将我们窗口的菜单栏的标题设置为我们打开的文件的名称。如果函数CreateFileMapping调用成功,我们把窗口的标题条换成被打开文件的名称。保存在缓冲区中的文件名是带有路径的全文件名,所以为了只显示文件名我们需要利用OPENFILENAME结构体中的成员nFileOffset的值来找到文件名的起始地址。
invoke EnableMenuItem,hMenu,IDM_OPEN,MF_GRAYED
invoke EnableMenuItem,hMenu,IDM_SAVE,MF_ENABLED
为了避免用户一次性打开多个文件,我们让“打开文件”菜单项呈灰色显示,使得打开文件的菜单项失效。函数EnableMenuItem可以用来改变菜单项的属性。 之后用户可能保存文件或者直接关闭应用程序。如果用户选择关闭应用程序,则事先必须关闭内存映射文件和打开的文件, 代码如下:
cmp [hMapFile], 0
jnz testhandle
testhandle:
call _CloseHandle
jmp quitmessage
在上面的代码段中,当WINDOWS的消息处理过程接收到WM_DESTROY消息后,它首先检测hMapFile值是否为0。如果不为0则表示相关的文件未关闭,这样就需要调用CloseMapFile来关闭它们。
proc _CloseHandle
invoke CloseHandle, [hMapFile]
mov [hMapFile], 0
invoke CloseHandle, [hFile]
ret
endp
上述过程调用是用来关闭内存映射文件和原来打开的文件的,这样可以使得程序退出时没有资源泄漏。如果用户选择保存文件的话,就弹出一个“保存文件”对话框,当用户输入了新文件的名称后,我们调用CreateFile函数来创建新文件---输出文件。
invoke MapViewOfFile,/
[hMapFile],/
FILE_MAP_READ,/
NULL,/
NULL,/
0
在输出文件创建后,我们调用MapViewOfFile函数来映射希望映射到我们用户地址空间的部分。
dwDesiredAccess用来指定我们想对文件进行的操作。在我们例子中,我们只想读,故指定标志FILE_MAP_READ。
dwFileOffsetHigh 和 dwFileOffsetLow 用来指定打开文件中欲映射的起始偏移位置。我们的例子中想映射整个的文件,故指定它们的值为0。
dwNumberOfBytesToMap 用来指定欲映射的字节数,如果想映射整个的文件,设定该值为0。
调用MapViewOfFile后,我们希望的部分就已经映射到内存中去了。您将得到一个指向起始内存块的指针。
invoke GetFileSize, [hFile], NULL
调用该函数可以得到文件的大小,其值通过eax传送,如果文件的长度超过4G,那么文件长度DWORD的高值部分(也即超过4G的部分)保存在FileSizeHighWord中。因为我们估计一般的文件将没有这么大,故忽略该值。
invoke WriteFile,/
[hFileWrite],/
[lpMemory],/
eax,/
lpSizeWrite,/
NULL
把内存映射文件中的数据写到输出文件中去。
invoke UnmapViewOfFile, [lpMemory]
写完后,我们解除映射。此时windows会撤销之前的内存块。
invoke SetWindowText, [hWnd], szWndName
invoke EnableMenuItem, [hMenu], IDM_OPEN, MF_ENABLED
invoke EnableMenuItem, [hMenu], IDM_SAVE, MF_GRAYED
最后我们修改标题,然后激活菜单栏的“打开”菜单按钮。
- Fasm -- Win32汇编学习--15
- Fasm---Win32汇编学习1
- Fasm---Win32汇编学习2
- Fasm---Win32汇编学习3
- FASM---Win32汇编学习4
- Fasm---Win32汇编学习5
- FASM--Win32汇编学习6
- Fasm---Win32汇编学习7
- Fasm---Win32汇编学习8
- Fasm---Win32汇编学习9
- FASM--Win32汇编学习10
- FASM -- Win32汇编学习11
- Fasm--Win32汇编学习12
- FASM -- Win32汇编学习13
- FASM---Win32汇编学习14
- Fasm ----Win32汇编专题---1
- Fasm---Win32汇编专题-----2
- Fasm汇编器源码学习中
- 字符引用与实体引用
- JDBC对数据库的事务操作
- 惊爆--唐山大地震的惊天谎言!!!
- 汶川大地震后问题
- 唐山地震二十九年祭
- Fasm -- Win32汇编学习--15
- 用C#2.0实现网络蜘蛛(WebSpider)
- HDOJ1800-------字符串Hash大杂烩
- 给黑白颜色的哀悼页面上添加祈福的红色蜡烛
- 20080519 梦
- 2年了
- LINUX下建立临时文件: mkstemp
- 北川邓家“刘汉小学”无一死亡奇迹背后的真相
- Oracle 连表查询 inner join;left join; right join!