游戏开发论坛

 找回密码
 立即注册
搜索
查看: 2237|回复: 3

转载—使用 Visual Basic 通过32位地址访问内存

[复制链接]

42

主题

140

帖子

203

积分

中级会员

Rank: 3Rank: 3

积分
203
发表于 2003-11-5 20:11:00 | 显示全部楼层 |阅读模式
使用 Visual Basic 通过32位地址访问内存

2001年7月6日

马尼拉,菲律宾

作者:Chris Vega [gwapo@models.com]

  当我们谈论“真的”指针和内存地址,我们大都会想到 Visual Basic 的局限性,比如,由于 VB 没有作为变量声明的指针数据类型,它不能直接访问内存。当某些场合需要一个变量的“地址”而不是它的值的时候,这一点混淆就显得特别明显。例如,那个变量位于内存(当前进程、其它进程或者动态链接库的虚拟空间)中的何处。

  是的,VB 确实“没有”指针变量,但是你是否曾试过将一个正规的 VB 数据类型转变为一个指针?你是否认为这是不可能的?好吧,还是再想一下,因为在 Visual Basic 中(从发行版本5开始),Microsoft 提供了一系列便利的函数以将你的正规变量转换为指针,它们是:

1] VarPtr - 返回一个变量或者数组元素的地址
StrPtr - 返回字符串的地址

  Visual Basic 中除字符串以外的变量,都位于它的内存位置上,你可以通过调用 VarPtr 函数获取这个变量的地址。字符串实际上是作为 BSTR 储存的,这是一个指向“字符数组的指针”的指针,这样你就需要 StrPtr 以得到“字符数组的指针”的地址,而不是用 VarPtr 获得 BSTR 的地址。

范例:
Dim ptrMyPointer As Long
Dim intMyInteger As Integer
Dim strMyString As String * 25

' 这就是一个调用
ptrMyPointer = VarPtr(intMyInteger)
' 将内存中 intMyInteger 这个变量的32位地址赋予 ptrMyPointer

strMyString = "变量的地址:" & Hex(ptrMyPointer)
MsgBox strMyString


' 这是另一个调用
ptrMyPointer = StrPtr(strMyString)
' 给出字符数组首元素的地址,例如,字符串的第一个字母。


2] VarPtrArray - 返回变量数组的地址
VarPtrStringArray - 返回字符传数组的地址

  Visual Basic 中数组被包存在 SAFEARRAY 结构中,你需要使用 VarPtrArray 函数以获取数组的地址,但是在使用该函数之前,你必须手工把它从 msvbvm50.dll 中声明到 VB 程序中。

范例:

' 对于 VB 5
' ========
Declare Function VarPtrArray Lib "msvbvm50.dll" Alias "VarPtr" (Var() as Any) As Long

' 对于 VB 6
' ========
Declare Function VarPtrArray Lib "msvbvm60.dll" Alias "VarPtr" (Var() as Any) As Long

' 调用

Dim lngSafeArrayAddress As Long
Dim lngArrayOfLongs(6) As Long
Dim i As Long

Randomize
For i = 0 to 6
 lngArrayOfLongs = Int(Rnd * &HFFFF)
Next

lngSafeArrayAddress = VarPtrArray(lngArrayOfLongs())

' 返回数组 lngArrayOfLongs 的安全地址,s of an Array , you
' 你可以将这些地址用于快速排序或其它用途。


  事实上,VarPtrStringArray 更难以用于你的程序中,因为你需要创建一个类型库并手工将该类型库引用到 VB 程序中。要做一个类型库,你需要一个 MIDL 编译器,它是一个用于将 *.odl 文件编译成类型库的命令行工具。
  对于 VB5,创建一个文本文件并且保存为 VB5StrPtr.odl,加入以下内容:

----------开始剪切--------------------------------------------------
#define RTCALL _stdcall
[uuid(6E814F00-7439-11D2-98D2-00C04FAD90E7),lcid (0), version(5.0), helpstring("VarPtrStringArray Support for VB5")]
library PtrLib
{
 importlib ("stdole2.tlb");
 [dllname("msvbvm50.dll")]
 module ArrayPtr
{
  [entry("VarPtr")]
  long RTCALL VarPtrStringArray([in] SAFEARRAY (BSTR) *Ptr);
}
}
----------结束剪切-------------------------------------------------

  用以下命令编译: MIDL /t VB5StrPtr.odl

  对于 VB6,创建一个文本文件并且保存为 VB6StrPtr.odl,加入以下内容:

-----------开始剪切--------------------------------------------------
#define RTCALL _stdcall
[uuid(C6799410-4431-11d2-A7F1-00A0C91110C3),lcid (0), version(6.0), helpstring("VarPtrStringArray Support for VB6")]
library PtrLib
{
 importlib ("stdole2.tlb");
 [dllname("msvbvm60.dll")]
 module ArrayPtr
{
  [entry("VarPtr")]
  long RTCALL VarPtrStringArray([in] SAFEARRAY (BSTR) *Ptr);
}
}
----------结束剪切-------------------------------------------------

  用以下命令编译: MIDL /t VB6StrPtr.odl


  现在,你有了类型库,将该类型库引用到 VB 程序中,然后你可以用以下方式获取字符串数组:

Dim MyArrayOfStrings(3) As String
Dim AddressOfArray As Long
MyArrayOfStrings(0)="Chris"
MyArrayOfStrings(1)="Vega"
MyArrayOfStrings(2)="gwapo@models.com"

' 调用
AddressOfArray = VarPtrStringArray ( MyArrayOfStrings() )

' 给出数组首元素的地址,而且是该首元素的第一个字符,
' 例如,这里是字符“C”在内存中的地址


' *** 怎样?你没有 MIDL 编译器?或者你不愿麻烦自己创建类型库并手工引用?
' 这里有一种足够容易的简单方法。
' 因为 StrPtr 函数具有获得字符串地址的能力,而字符串数组的元素全部都是字符串,
' 所以你应该清楚了,你可以对数组的首元素进行这个调用

AddressOfArray = StrPtr ( MyArrayOfStrings(0) )

' 返回更上述方法完全相同的结果



3] ObjPtr - 返回一个对象的地址

  面向对象编程由众多对象组成,而这些对象和它的众多属性一起保存在内存中,作为一种结构化的布局。你需要调用 ObjPtr 函数来获取它的位置。

范例:
' 你要知道 Form1 处于内存何处,本方法告诉你它在线程中的地址

Dim objMyObject As New Form1
MsgBox "Form1 位于:" & Hex( ObjPtr( objMyObject ) )



  好,到此为止你一定在想:无论如何,怎么才能把这些地址变成实际有用的东西呢?其实如果你这样想答案就很清楚了:地址是一个内存中的位置,而你的变量中保存的就是一个内存中的位置,并且有它本身在内存中的位置。搞糊涂了?我们让它简单一点,你可以简单的认为这个地址是保存数据的位置,数据是可读写的,而你需要通过这个地址来读写它。唔,Visual Basic 能够做这种事情吗?

  不能,如果你只是简单的考虑 Visual Basic 的能力的话,但是你的程序可以使用 API 函数。我现在谈到的 API 是从 KERNEL32.DLL 输出的运行库,名为 RtlMoveMemory 和 RtlCopyMemory。

  它太吸引人了。首先我们已经找到了通过把变量转变为指针来得到内存地址的方法,现在我们又有了读写这些地址所指内容的方法。但是只要将这两个声明中的任一个加入你的程序中,而不是全部,因为它们的功能是一样的。我建议使用第二个,因为它被所有的 Windows 系统支持,而 RtlCopyMemory 则不然。


Private Declare Sub CopyMemory _
Lib "kernel32" Alias _
"RtlCopyMemory" _
(Destination As Any, _
Source As Any, _
ByVal length As Long)
' RtlCopyMemory 将一块内存的内容复制到另一块中。

' 或者

Private Declare Sub CopyMemory _
Lib "kernel32" Alias _
"RtlMoveMemory" _
(Destination As Any, _
Source As Any, _
ByVal length As Long)

' RtlMoveMemory 可以向前或向后移动内存,匹配的或不匹配的,
' 以 4字节的块为单位,后面为所有保留的字节。

参数:

Destination
指向要移动的目标。

Source
指向要复制的内存。

Length
指定要复制的字节数。


  为了使它更容易使用,你可以把下面内容复制粘贴到 modMemory.bas 中:

------------开始剪切------------------------------------------------------------------

Attribute VB_Name = "modMemory"
' =============================================================================
' 复制内存 API
' =============================================================================
Private Declare Sub CopyMemory _
Lib "kernel32" Alias _
"RtlMoveMemory" _
(Destination As Any, _
Source As Any, _
ByVal length As Long)

' =============================================================================
' 数据长度
' =============================================================================
Public Enum e_BinaryData
DefineByte = 1 ' 8 位数据
DefineWord = 2 ' 16 位数据
DefineDoubleWord = 4 ' 32 位数据
DefineQuadWord = 8 ' 64 位数据
End Enum

' =============================================================================
' 允许直接读 MemPointer 指向的内存
' 用和 Asm 一样的字节数定义 (DB, DW, DD, DX)
' =============================================================================
Function ReadMem(ByVal MemPointer As Long, _
SizeInBytes As e_BinaryData)
Select Case SizeInBytes
Case DefineByte
Dim DB As Byte
CopyMemory DB, ByVal MemPointer, 1
ReadMem = DB
Case DefineWord
Dim DW As Integer
CopyMemory DW, ByVal MemPointer, 2
ReadMem = DW
Case DefineDoubleWord
Dim DD As Long
CopyMemory DD, ByVal MemPointer, 4
ReadMem = DD
Case DefineQuadWord
Dim DX As Double
CopyMemory DX, ByVal MemPointer, 8
ReadMem = DX
End Select
End Function

' =============================================================================
' 允许直接写 MemPointer 指向的内存
' 用和 Asm 一样的字节数定义 (DB, DW, DD, DX)
' =============================================================================
Sub WriteMem(ByVal MemPointer As Long, _
SizeInBytes As e_BinaryData, _
ByVal DataToWrite)
CopyMemory ByVal MemPointer, VarPtr(DataToWrite), SizeInBytes
End Sub

------------结束剪切---------------------------------------------------------------


用例:

通过内存为变量赋值:

Dim ptrVariable As Long
Dim xCounter As Long

ptrVariable = VarPtr(ptrVariable)
WriteMem ptrVariable, DefineWord, &HFFFF
' 与 ptrVariable = &HFFFF 等价


读内存的内容,使用:

ptrVariable = ReadMem(ptrVariable, DefineWord)


  现在我们能够获得指针并访问它们了。但是如果你一步步跟着以上步骤看下来,你可能奇怪一条原本的 Visual Basic 赋值操作比这里介绍的直接内存赋值操作快得多。然而本文旨在指出可以使用 Visual Basic 访问内存,而这一点的主要意义不仅在于读取和分析变量,接下来,你可以通过获得内存地址简单地处理运行的 DLL。同时利用 modMemory.bas 和 PE (Portable Executable) 文件格式的知识,你可以分析 DLL 主体,看看它们是如何处理的。最好的是,可以获取它所有输出函数的列表;差点忘记,可以把它们 spy 出来或者干脆获取函数体的副本进行反汇编,比低级语言访问更多的内容,这也是 C 语言被称为工业标准的原因;现在你可以书写跟 C 表现相同的 Visual Basic 程序

140

主题

1228

帖子

1233

积分

金牌会员

Rank: 6Rank: 6

积分
1233
QQ
发表于 2003-11-6 10:31:00 | 显示全部楼层

Re:转载—使用 Visual Basic 通过32位地址访问内存

不错!

6

主题

103

帖子

103

积分

注册会员

Rank: 2

积分
103
发表于 2006-9-10 23:45:00 | 显示全部楼层

Re:转载—使用 Visual Basic 通过32位地址访问内存

好东西

6

主题

103

帖子

103

积分

注册会员

Rank: 2

积分
103
发表于 2006-9-11 01:03:00 | 显示全部楼层

Re:转载—使用 Visual Basic 通过32位地址访问内存

再顶
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

作品发布|文章投稿|广告合作|关于本站|游戏开发论坛 ( 闽ICP备17032699号-3 )

GMT+8, 2026-1-25 08:49

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

快速回复 返回顶部 返回列表