- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我需要使用 VirtualAlloc 为我的项目分配可执行内存,以便 JIT 将自定义脚本格式重新编译为 x86/etc。但我感到困惑的是,其他人似乎都没有注意到,而且它的行为似乎明显缺乏细节。
我知道它分配“虚拟”内存,这意味着它可以是物理上的任何东西(RAM/磁盘),但在使用时它可以简单地被视为“内存”。但是,例如,如果我做类似的事情:
#define MB 1024*1024
auto pAddr = VirtualAlloc(NULL, 8*MB, MEM_RESERVE, PAGE_NOACCESS);
VirtualAlloc(pAddr + 4*MB, 1*MB, MEM_COMMIT, PAGE_EXECUTE_READWRITE); // commit 1MB, 4MB's into the reserved memory
是只使用了 1MB 还是 5MB?显然,我并不期望它是 5MB - 我只是不知道在这种看似显而易见的情况下会发生什么。有效吗?保留内存中的任何范围都可以自由提交和取消提交吗?更重要的是,它可以乱序使用还是应该增量分配(根据 MSDN 文档,看起来你可以用它来做)。还是 VirtualAlloc 只对一次分配“页面”感到满意?
我发现的每个示例似乎只对向我展示如何分配页面感兴趣——这可能只是最基本的用途,但远非最实用——但我想用它来为脚本分配编译代码,这可能是偶尔在执行期间重新编译。我需要尝试为这些分配创建某种接口(interface),这样我就可以简单地说“为这个脚本编译提供一些内存”,它会自动返回以前未使用的提交空间或提交一些新空间 - 所以任何提示如何最好地从虚拟内存分配(例如,最好不要取消提交可能会再次提交的内存?)也将受到赞赏。
最佳答案
好的;我认为我明白你的意思,我希望这会澄清事情。
从概念上讲,VirtualAlloc 在单独的页面上工作。
为简单起见,让我们考虑一个 32 位 x86 进程。虚拟地址空间是从第 0 页到第 1048575 页的一系列页面。这些页面中的每一个都可能保留也可能不保留;如果保留,则可能会或可能不会提交。 (如果提交,它也将有零个或多个内存保护选项,并且页面可能处于各种其他状态,但我们现在几乎可以忽略所有这些。)
无法只保留或提交页面的一部分,或者同一页面的两个部分具有不同的内存保护选项。相反,保留和/或提交的页面是否连续并不重要。
如果您使用特定的起始地址和区域大小调用 VirtualAlloc,那么它会作用于指定虚拟地址区域中包含一个或多个字节的每一页。地址和大小仅用于计算要作用于哪些页面。参数是地址而不是页码的唯一原因是为了简化程序员的工作。
从概念上讲,一次覆盖多个页面的 VirtualAlloc 调用等同于为每个页面调用一次 VirtualAlloc。唯一的区别(效率除外)是一次对多个页面进行操作是原子的,因此在整个范围内要么失败要么成功。
请特别注意,如果您成功多次调用 VirtualAlloc 覆盖连续的页面,则无法告诉后记这些页面是单独分配的。操作系统只记得页面处于什么状态,而不是它是如何到达那里的。 [附录:糟糕;这是错误的。 VirtualQuery
的文档说它可以判断连续的页面是否属于同一分配。也许它们被标记了唯一的分配 ID 或其他东西。我不相信内存管理器实际上使用了这些信息,但显然它 被保留了。]
请记住,HeapCreate 函数已经允许您创建一个堆,其内存块允许代码执行。除非您的应用程序有非常特殊的需求,否则您不太可能通过编写自己的堆管理器获得任何好处。
关于c++ - VirtualAlloc 的困惑 - 它只适用于页面吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22183228/
当使用 VirtualAlloc API 分配和提交具有 2 次幂页面边界大小的虚拟内存区域时,例如: void* address = VirtualAlloc(0, 0x10000, MEM_COM
我知道 C/C++ 风格的程序有内存部分、堆栈、堆、.text 等。但是当我使用 VirtualAlloc 时,它从哪里分配内存?我不认为它是堆,因为我可以只使用 HeapAlloc。 建议将不胜感激
virtualpointer=(char*) VirtualAlloc (NULL, (unsigned __int64) (1<<31), MEM_RESERVE, PAGE_READWRITE);
出于某种原因,当我的应用程序在某些配置的硬盘上运行时(RAID、随机分区、通过单独的 IDE Controller ,而不是在操作系统分区上),VirtualAlloc 返回 null 与 ERROR
我正在尝试使用 VirtualAlloc 来保留和提交一 block 内存,然后再次扩展该 block 。不幸的是,尽管 VirtualQuery 说请求的地址范围是免费的,但它返回 NULL 和错误
我正在为正在优化的 (Win7) C++ 例程编写回归测试,该例程以前释放并重新分配了许多巨大的缓冲区:内存流失。我想证明在测试期间,程序没有分配任何大内存区域(比如 16M 或更大),而是有效地重新
我试图从一个二进制文件中读取一些数据到一个用 VirtualAlloc 分配的缓冲区中。问题是我遇到了一个“糟糕的 Pr”问题并且无法执行 fread。这是我的代码: fseek(myfile,0,
VirtualAlloc 返回的内存块是否总是与页面大小对齐?换句话说,VirtualAlloc 的返回值和页面大小的模数总是为零吗? 最佳答案 嗯,是。 毕竟,你调用 VirtualAlloc 来分
在我的应用程序中,我试图在启动时通过 VirtualAlloc 分配一大块内存(大约 1GB-2GB),然后我可以稍后将其分割以供整个应用程序的其余部分使用。在 Debug模式下,我想在 Virtua
假设我使用 VirtualAlloc() 从 0x06000000 到 0x06010000 分配了几页连续内存(即 16 个 4KB 页),其中 PAGE_READWRITE保护。但过了一段时间,我
我正在使用 C# 开发应用程序 SFX/Protector,我希望 protected 程序集从字节数组执行,而不是将其写入硬盘,以便逆向工程更加困难。 我在一个字节数组中有一个程序(它有一个有效的入
我在使用 virtualalloc 时有一些奇怪的行为。我在使用 C++,Visual Studio 2010。 我有两件事要分配,我正在使用 VirtualAlloc(我有我的理由,与问题无关) 1
我正在将代码从 C++ 转换为我的 VB.NET 应用程序。这是 C++ 代码: typedef int (__stdcall *init_t)(uint32_t value,uint32_t pa
我想知道是否有比malloc/free低一级的跨平台分配器。 例如,我想要一些在 Linux 中简单地调用 sbrk 并在 Windows 中调用 VirtualAlloc 的东西(可能还有两个类似的
重要:在您在这里花费太多时间之前向下滚动到“最终更新”。事实证明,主要的教训是提防单元测试套件中其他测试的副作用,并始终在孤立地重现事物,然后再仓促下结论! 从表面上看,以下 64 位代码使用 Vir
我需要使用 VirtualAlloc/VirtualAllocEx 做什么? 举个例子,我发现的一个案例——如果我分配了 4 GB 的虚拟内存,那么如果我不使用所有虚拟内存,那么我就不会花费物理内存,
我想知道是否有可能确定一个虚拟地址(指针)是否属于以前的 VirtualAlloc 调用(如果可能的话不写入页面)。如果 VirtualFree 与 MEM_RELEASE 一起使用,它可以自动设置
在 VirtualFree 的 msdn 文档中 BOOL WINAPI VirtualFree( _In_ LPVOID lpAddress, _In_ SIZE_T dwSize, _
我需要使用 VirtualAlloc 为我的项目分配可执行内存,以便 JIT 将自定义脚本格式重新编译为 x86/etc。但我感到困惑的是,其他人似乎都没有注意到,而且它的行为似乎明显缺乏细节。 我知
我想调整由 MS 窗口的 VirtualAlloc 分配的内存区域的大小。查看 VirtualFree 文档,可以只部分取消提交一个区域,但不可能部分释放它。即可以释放部分物理内存,但不能释放部分虚拟
我是一名优秀的程序员,十分优秀!