ImageVerifierCode 换一换
格式:DOCX , 页数:6 ,大小:20.46KB ,
资源ID:6524105      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/6524105.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(win进程虚拟内存数据搜索与修改.docx)为本站会员(b****6)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

win进程虚拟内存数据搜索与修改.docx

1、win进程虚拟内存数据搜索与修改win2000进程虚拟内存数据搜索与修改Windows2000下用户模式的内存扫描转帖 Sprite简述: 本文简要介绍了在Windows2000下实现内存扫描的基本理论和实现的办法。内存扫描是一项重要的技术,有相当广泛的应用范围:如病毒扫描、游戏修改等。Windows2000是一个完全保护的系统,且具有两种工作模式,即用户态和核心态(User Model and Kernel Model)。内存扫描也可分为用户态的内存扫描与核心态的内存扫描。本文主要讲述的是工作于用户态的内存扫描。 一相关理论 早期在DOS坏境下进行内存扫描是一件相对简单的事情。因为DOS工作

2、在CPU的实模式下,没有采用虚存技术也没有提供内存的保护机制,只要实实在在的扫描完所有的物理内存,一切工作也就完成了,早期有一些防毒软件就是用了这样的办法。当然为了提高效率,我们并不用扫描所有的内存区域,因为有些空间是没有被用到的,扫描这些地方也是只浪费时间。这可以通过遍历DOS系统的MCB(Memory Control Block)链,来得到实际内存的使用区域,从而使扫描的效率大大提高。相似的思路在Windows2000下的内存扫描也是适用的。Windows2000则是一个完全保护的系统,工作于CPU的保护模式下,引入了虚存技术。每个进程拥有独立的4GB的地址空间,其中低的2GB为进程的私有

3、空间,高的2GB为系统空间的映射(如果在Boot.ini文件中使用“/3GB”的开关可以使进程的私有空间增大到3GB,系统空间1GB)。对于每个进程来讲其虚拟的地址空间是连续的,实际上它们是以页面为单位离散的存在于物理内存中,一些可能被交换到硬盘上的页面文件中,而且还有大部分的空间是未提交(Uncommitted)的。因此在Windows2000中对进程的用户空间进行扫描必须依次对每个进程的空间进行扫描。一个进程的低2GB有空间的分布如下表:范围大小作用0x0 0xFFFF64 KB不可访问区域,只是用来防止非法的指针访问,访问该范围的地址会导致访问违例。0x100000x7FFEFFFF2

4、GB 减去至少192 KB进程的私有地址空间0x7FFDE0000x7FFDEFFF4 KB进程中第一个线程的线程环境块,即TEB(Thread environment block)0x7FFDF000 0x7FFDFFFF4 KB进程的进程环境块,即PEB(Process environment block)0x7FFE0000 0x7FFE0FFF4 KB一个共享的只读用户数据块,该块映射到到系统空间的一个数据块,其中存放的是一些系统信息如系统时间、时钟的滴答数、系统版本号等。这样访问这些信息的时候系统就不用切换到核心模式。0x7FFE10000x7FFEFFFF60 KB不可访问0x7F

5、FF0000 0x7FFFFFFF64 KB不可访问,用于防止线程的缓冲跨越两种模式空间的边界表1二实现从上表可以看出,我们要扫描范围的起点和终点不是从02GB,而只是其中的一部分。要得到这个起点和终点可以使用API函数GetSystemInfo,函数的原型如下:VOID GetSystemInfo( LPSYSTEM_INFO lpSystemInfo / system information); 而在结构SYSTEM_INFO中有两个域:lpMinimumApplicationAddress和 lpMaximumApplicationAddress(类型都是LPVOID) 中,我们就可以得

6、到一个应用程序可用的最小和最大的地址空间。这样我们就得到了要扫描的地址的起点和终点。那么是不是这起点和终点间所有的地址都要扫描呢?并不是这样的,因为一般情况下一个进程是用不着这么大(接近2GB)的地址空间的。因此一个进程的大部分地址空间都是未用(Free)或是保留(Reserved)的,真正用到的只是那些已提交(Committed)的内存而已。 内存页面可以有三种状态:未用(Free)、保留(Reserved)和提交(Committed)。一个未用的页面是指该页面未被保留或是提交,对一个进程来讲一个未用的页面是不可访问的,访问这样的页面将导致访问违例。进程可以要求系统保留一些页面以备后用,系统

7、返回一段保留的地址给进程,但是这些地址同样是不可访问的,进程若想使用这段地址空间,使用必须先提交。只有一个提交的页面才是一个真正可以访问的页面。不过你提交了一个页面,系统并不会马上分配物理页面,只有在该页面第一次被访问到时,系统才会分配页面并初始化。另外,这三个状态的两两之间都是可以相互转化的。相关的API函数有VirtualAlloc、VirtualAllocEx、VirtualFree、VirtualFreeEx等.这样我们的工作已大大减少了,只需要扫描那些提交的页面就好了。接下来要做的就是得到一个进程的已提交的页面范围。这就要用到另外两个API函数VirtualQuery和Virtual

8、QueryEx。两个函数的功能相似,不同就是VirtualQuery只是查询本进程而VirtualQueryEx可以查询指定进程的内存空间信息,后者正是我们所需要的,函数原型如下:DWORD VirtualQueryEx( HANDLE hProcess, / handle to process LPCVOID lpAddress, / address of region PMEMORY_BASIC_INFORMATION lpBuffer, / information buffer SIZE_T dwLength / size of buffer);第一个参数是进程的句柄;第二个参数是内存地

9、址指针;第三个参数是指向MEMORY_BASIC_INFORMATION结构的指针,用于返回内存空间的信息;第四个参数是lpBuffer的长度。再来看一下结构MEMORY_BASIC_INFORMATION的声明:typedef struct _MEMORY_BASIC_INFORMATION PVOID BaseAddress; PVOID AllocationBase; DWORD AllocationProtect; SIZE_T RegionSize; DWORD State; DWORD Protect; DWORD Type; MEMORY_BASIC_INFORMATION, *

10、PMEMORY_BASIC_INFORMATION; 第一个参数是查询内存块的基地址;第二个参数指的是用VirtualAlloc分配该内存时实际分配的基地址,可以小于BaseAddress,也就是说BaseAddress一定包含在AllocationBase分配的范围内;第三个参数指的是分配该页面时,页面的一些属性,如PAGE_READWRITE、PAGE_EXECUTE等(其它属性可参考Platform SDK);第四个参数指的是从BaseAddress开始,具有相同属性的页面的大小。第五参数指的是页面的状态,有三种可能值:MEM_COMMIT、MEM_FREE和MEM_RESERVE,这个

11、参数对我们来说是最重要的了,从中我们便可知指定内存页面的状态了;第六个参数指的是页面的属性,其可能的取值与AllocationProtect相同;最后一个参数指明了该内存块的类型,有三种可能值:MEM_IMAGE 、MEM_MAPPED和MEM_PRIVATE。这样我们就可得到进程中需要扫描的地址范围了。到这里剩下的问题就是要读取指定的进程的指定的地地址空间的内容了。这里要用到的是用于调试程序和错误处理(Debugging and Error Handling)的API函数。在“Platform SDK: Debugging and Error Handling”章节中,介绍了一部分与程序调试

12、和错误处理相关的API函数,有许多是很有用,例如我们下面用到的ReadProcessMemory和WriteProcessMemory,它们原型如下:BOOL ReadProcessMemory( HANDLE hProcess, / handle to the process LPCVOID lpBaseAddress, / base of memory area LPVOID lpBuffer, / data buffer SIZE_T nSize, / number of bytes to read SIZE_T * lpNumberOfBytesRead / number of byt

13、es read);BOOL WriteProcessMemory( HANDLE hProcess, / handle to process LPVOID lpBaseAddress, / base of memory area LPCVOID lpBuffer, / data buffer SIZE_T nSize, / count of bytes to write SIZE_T * lpNumberOfBytesWritten / count of bytes written);参数很简单从它们的名字都可以猜出其意义了,这里就不多做说明了。要说明的是要对一个进程进行ReadProcess

14、Memory操作,当前进程对要读的进程必须有PROCESS_VM_READ访问权。要对一个进程进行WriteProcessMemory操作,当前进程对要写的进程必须有PROCESS_VM_WRITE 和PROCESS_VM_OPERATION访问权。要获得一个进程的句柄和对这个进程的一些控制权可以使用API函数OpenProcess得到,其使用不做详细说明了,只给出其原型:HANDLE OpenProcess( DWORD dwDesiredAccess, / access flag BOOL bInheritHandle, / handle inheritance option DWORD

15、dwProcessId / process identifier); 这样对一个进程的用户地址空间内存扫描的流程基本就阐述清楚了。三 相关的问题: 在实际操作中会遇到一些问题。如果我们指定了写相关的访问权(如PROCESS_VM_WRITE、PROCESS_SET_INFORMATION、PROCESS_ALL_ACCESS等),用OpenProcess打开一些普通进程是没什么问题,但要是打开的是系统安全进程(如System、Winlogon、smss、csrss、services、lsass等)或是一些注册为服务的进程时,就会遇到“访问拒绝”的错误,这是为了系统的安全而采取的保护手段。说明了

16、当前的进程没有足够的权限来进行此操作。在进程控制结构中有一个“访问令牌”(Access tokens),里面包含有本进程的权限信息。一些常用的权限如表1所示(摘自Inside Windows2000,Third Edition)。 权限名 权限含义SeBackup在备份的时候绕过安全检查SeDebug可对一个进程进行调试SeShutdown可关闭本地系统SeTakeOwnerShip在没有得到自由访问权的情况下得到一个对象的所有权表2 要对一个任意进程(包括系统安全进程和服务进程)进行指定了写相关的访问权的OpenProcess操作,只要当前进程具有SeDeDebug权限就可以了。要是一个用户

17、是Administrator或是被给予了相应的权限,就可以具有该权限。可是,就算我们用Administrator帐号对一个系统安全进程执行OpenProcess(PROCESS_ALL_ACCESS,FALSE, dwProcessID)还是会遇到“访问拒绝”的错误。什么原因呢?原来在默认的情况下进程的一些访问权限是没有被使能(Enabled)的,所以我们要做的首先是使能这些权限。与此相关的一些API函数有OpenProcessToken、LookupPrivilegevalue、AdjustTokenPrivileges。我们要修改一个进程的访问令牌,首先要获得进程访问令牌的句柄,这可以通过

18、OpenProcessToken得到,函数的原型如下:BOOL OpenProcessToken( HANDLE ProcessHandle, DWORD DesiredAccess,PHANDLE TokenHandle);第一参数是要修改访问权限的进程句柄;第三个参数就是返回的访问令牌指针;第二个参数指定你要进行的操作类型,如要修改令牌我们要指定第二个参数为TOKEN_ADJUST_PRIVILEGES(其它一些参数可参考Platform SDK)。通过这个函数我们就可以得到当前进程的访问令牌的句柄(指定函数的第一个参数为GetCurrentProcess()就可以了)。接着我们可以调用A

19、djustTokenPrivileges对这个访问令牌进行修改。AdjustTokenPrivileges的原型如下:BOOL AdjustTokenPrivileges( HANDLE TokenHandle, / handle to token BOOL DisableAllPrivileges, / disabling option PTOKEN_PRIVILEGES NewState, / privilege information DWORD BufferLength, / size of buffer PTOKEN_PRIVILEGES PreviousState, / origi

20、nal state buffer PDWORD ReturnLength / required buffer size);第一个参数是访问令牌的句柄;第二个参数决定是进行权限修改还是除能(Disable)所有权限;第三个参数指明要修改的权限,是一个指向TOKEN_PRIVILEGES结构的指针,该结构包含一个数组,数据组的每个项指明了权限的类型和要进行的操作; 第四个参数是结构PreviousState的长度,如果PreviousState为空,该参数应为NULL;第五个参数也是一个指向TOKEN_PRIVILEGES结构的指针,存放修改前的访问权限的信息,可空;最后一个参数为实际Previo

21、usState结构返回的大小。在使用这个函数前再看一下TOKEN_PRIVILEGES这个结构,其声明如下:typedef struct _TOKEN_PRIVILEGES DWORD PrivilegeCount; LUID_AND_ATTRIBUTES Privileges; TOKEN_PRIVILEGES, *PTOKEN_PRIVILEGES; PrivilegeCount指的数组原素的个数,接着是一个LUID_AND_ATTRIBUTES类型的数组,再来看一下LUID_AND_ATTRIBUTES这个结构的内容,声明如下:typedef struct _LUID_AND_ATTRI

22、BUTES LUID Luid; DWORD Attributes; LUID_AND_ATTRIBUTES, *PLUID_AND_ATTRIBUTES第二个参数就指明了我们要进行的操作类型,有三个可选项: SE_PRIVILEGE_ENABLED、SE_PRIVILEGE_ENABLED_BY_DEFAULT、SE_PRIVILEGE_USED_FOR_ACCESS。要使能一个权限就指定Attributes为SE_PRIVILEGE_ENABLED。第一个参数就是指权限的类型,是一个LUID的值,LUID就是指locally unique identifier,我想GUID大家是比较熟悉的

23、,和GUID的要求保证全局唯一不同,LUID只要保证局部唯一,就是指在系统的每一次运行期间保证是唯一的就可以了。另外和GUID相同的一点,LUID也是一个64位的值,相信大家都看过GUID那一大串的值,我们要怎么样才能知道一个权限对应的LUID值是多少呢?这就要用到另外一个API函数LookupPrivilegevalue,其原形如下:BOOL LookupPrivilegevalue( LPCTSTR lpSystemName, / system name LPCTSTR lpName, / privilege name PLUID lpLuid / locally unique ident

24、ifier);第一个参数是系统的名称,如果是本地系统只要指明为NULL就可以了,第三个参数就是返回LUID的指针,第二个参数就是指明了权限的名称,如“SeDebugPrivilege”。在Winnt.h中还定义了一些权限名称的宏,如:#define SE_BACKUP_NAME TEXT(SeBackupPrivilege)#define SE_RESTORE_NAME TEXT(SeRestorePrivilege)#define SE_SHUTDOWN_NAME TEXT(SeShutdownPrivilege)#define SE_DEBUG_NAME TEXT(SeDebugPrivilege)这样通过这三个函数的调用,我们就可以用OpenProcess(PROCESS_ALL_ACCESS,FALSE, dwProcessID)来打获得任意进程的句柄,并且指定了所有的访问权。四 总结 用户模式的内存扫描还是具有想当的局限性,它不能完全扫描Windows2000的全部内存空间。要对系统空间进行扫描,在Windows2000下,用户模式的应用程序是不能实现的。要实现对系统空间的扫描,必须通过工作于核心模式的程序驱动程序来实现。

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1