2

我正在四处浏览要做的事情,我想到了在 Windows 中的特定地址分配内存。

所以我在这里阅读了一些关于stackoverflow的问题,但没有一个真正提供一个工作示例,所以我不得不提出自己的问题,因为我真的很想尝试:

#include <Windows.h>
#include <iostream>

struct Variable {
    int var;
};

#define ACCESS() ((Variable*)0x50000000)

int main()
{
    DWORD ptr;
    VirtualAlloc((void*)0x50000000,sizeof(Variable),MEM_COMMIT | MEM_RESERVE | MEM_PHYSICAL,PAGE_READWRITE);                    
    VirtualProtect((void*)0x50000000,sizeof(Variable),PAGE_READWRITE,&ptr);
    ACCESS()->var = 5;
    while(!GetAsyncKeyState('Q')){}
}

但这总是会导致访问失效..

在特定地址分配数据的正确方法是什么?因为,这种方式不知何故行不通..但同时与“为什么不?”混淆。


编辑:

第一个答案后的第二个代码也不起作用:

#include <Windows.h>
#include <iostream>

struct Variable {
    int var;
};

#define ACCESS() ((Variable*)0x50000000)
int main()
{   
    std::cout << VirtualAlloc((void*)0x50000000,sizeof(Variable),MEM_COMMIT,PAGE_READWRITE) << std::endl;
    std::cout << GetLastError() << std::endl;
    ACCESS()->var = 6;
    std::cout << ACCESS() << std::endl;
    while(!GetAsyncKeyState('Q')){}
}

返回值为 0,表示失败,GetLastError() 等于 487,

根据此页面,487表示:

ERROR_INVALID_ADDRESS;尝试访问无效地址。

那么,我在互联网上阅读的所有内容和堆栈溢出都是假的?您>不能<在指定地址分配数据?如果这是真的,为什么我们还需要这个函数呢?


下一次编辑:

似乎完全有可能,工作代码:

#include <Windows.h>
#include <iostream>

struct Variable {
    int var;
};

#define ACCESS() ((Variable*)0x50000000)
int main()
{   
    std::cout << VirtualAlloc((void*)0x50000000,sizeof(Variable),MEM_COMMIT | MEM_RESERVE,PAGE_READWRITE) << std::endl;
    std::cout << GetLastError() << std::endl;
    std::cout << ACCESS() << std::endl;
    ACCESS()->var = 6;
    std::cout << ACCESS()->var << std::endl;
    while(!GetAsyncKeyState('Q')){}
}
4

1 回答 1

5

呼吁VirtualAlloc失败。如果您检查它的返回值,您会发现这一点。

它失败的原因直接来自文档MEM_PHYSICAL必须与其他值一起使用,MEM_RESERVE而不是其他值。无论如何,您可能都不想要MEM_PHYSICAL,这与AWE有关。删除标志将使分配成功。

呼吁VirtualProtect是不必要的。

以下应该有效:

#include <windows.h>
int main()
{
    VirtualAlloc((void*)0x50000000,0x1000,MEM_COMMIT|MEM_RESERVE,PAGE_READWRITE);
    *((int*)0x50000000) = 5;
}
于 2013-07-29T07:08:35.647 回答