大页面申请及使用

大页面的的应用场景及注意事项如下(出自北风网 VC++ 系列视频教程,请到官网下载尊重版权):

  • 大页面的内存是不能换页的,只能常驻于RAM中,不会存在页面文件中 (即大页面不会换页,物理内存紧张时慎用)
  • 大页面的内存算作进程的私有页面,不会统计在进程工作集中
  • 因为大页面尺寸远大于小页面尺寸,分配一个大页面必须要有连续的足够大块的物理内存,这可能导致系统进行耗时的内存碎片整理工作
  • 大页面始终是可读写的,并且是充分利用了高速缓存(TLB)
  • 大页面的分配不受作业对象对进程工作集大小的限制
  • 在Intel安腾架构上的WOW64系统不支持32位应用大页面,除非重编译应用为64位的

参考代码

#include <tchar.h>
#include <windows.h>
#include <time.h>

void Privilege(TCHAR* pszPrivilege, BOOL bEnable)
{
    HANDLE              hToken = NULL;
    TOKEN_PRIVILEGES    tp = {};
    OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &hToken);
    LookupPrivilegeValue(NULL, pszPrivilege, &tp.Privileges[0].Luid);
    tp.PrivilegeCount = 1;
    tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
    AdjustTokenPrivileges(hToken, FALSE, &tp, 0, (PTOKEN_PRIVILEGES)NULL, 0);
    CloseHandle(hToken);
}

int _tmain(int argc, _TCHAR* argv[])
{
    // 设置随机数种子
    srand((unsigned int)time(NULL));
    // 提权
    Privilege(TEXT("SeLockMemoryPrivilege"), TRUE);
    // 获取大页面大小
    SIZE_T szLargePage = GetLargePageMinimum();
    // 保留空间 64 * szLargePage
    PVOID pBuffer = VirtualAlloc(0, 64 * szLargePage, MEM_RESERVE, PAGE_READWRITE);
    // 使用 MEM_LARGE_PAGES 标志提交内存
    VirtualAlloc(pBuffer, 2 * szLargePage, MEM_COMMIT | MEM_LARGE_PAGES, PAGE_READWRITE);
    // 测试内存数据
    double* pdbArray = (double*)pBuffer;
    for (int i = 0; i < (int)(2 * szLargePage / sizeof(double)); i++)
    {
        pdbArray[i] = 1.0f * rand();
    }
    // 释放内存
    VirtualFree(pBuffer, 0, MEM_RELEASE);
    return 0;
}

发表评论