Создание страниц Guard

Страница охраны предоставляет однократный сигнал тревоги для доступа к страницам памяти. Это может быть полезно для приложения, которому необходимо отслеживать рост больших динамических структур данных. Например, существуют операционные системы, использующие страницы защиты для реализации автоматической проверки стека.

Чтобы создать страницу защиты, задайте модификатор защиты страницы PAGE_GUARD для страницы. Это значение можно указать вместе с другими модификаторами защиты страниц в функциях VirtualAlloc, VirtualAllocEx, VirtualProtect и VirtualProtectEx . Модификатор PAGE_GUARD можно использовать с любыми другими модификаторами защиты страниц, кроме PAGE_NOACCESS.

Если программа пытается получить доступ к адресу на странице защиты, система создает исключение STATUS_GUARD_PAGE_VIOLATION (0x80000001). Система также очищает модификатор PAGE_GUARD , удаляя состояние страницы защиты страницы памяти. Система не остановит следующую попытку доступа к странице памяти с STATUS_GUARD_PAGE_VIOLATION исключением.

Если во время системной службы возникает исключение страницы защиты, служба завершается сбоем и обычно возвращает некоторый индикатор состояния сбоя. Так как система также удаляет состояние страницы защиты соответствующей страницы памяти, следующий вызов той же системной службы не завершится ошибкой из-за исключения STATUS_GUARD_PAGE_VIOLATION (если, конечно, кто-то не перенаправит страницу защиты).

Следующая короткая программа иллюстрирует поведение защиты страницы guard.

/* A program to demonstrate the use of guard pages of memory. Allocate
   a page of memory as a guard page, then try to access the page. That
   will fail, but doing so releases the lock on the guard page, so the
   next access works correctly.

   The output will look like this. The actual address may vary.

   This computer has a page size of 4096.
   Committed 4096 bytes at address 0x00520000
   Cannot lock at 00520000, error = 0x80000001
   2nd Lock Achieved at 00520000

   This sample does not show how to use the guard page fault to
   "grow" a dynamic array, such as a stack. */

#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <tchar.h>

int main()
{
  LPVOID lpvAddr;               // address of the test memory
  DWORD dwPageSize;             // amount of memory to allocate.
  BOOL bLocked;                 // address of the guarded memory
  SYSTEM_INFO sSysInfo;         // useful information about the system

  GetSystemInfo(&sSysInfo);     // initialize the structure

  _tprintf(TEXT("This computer has page size %d.\n"), sSysInfo.dwPageSize);

  dwPageSize = sSysInfo.dwPageSize;

  // Try to allocate the memory.

  lpvAddr = VirtualAlloc(NULL, dwPageSize,
                         MEM_RESERVE | MEM_COMMIT,
                         PAGE_READONLY | PAGE_GUARD);

  if(lpvAddr == NULL) {
    _tprintf(TEXT("VirtualAlloc failed. Error: %ld\n"),
             GetLastError());
    return 1;

  } else {
    _ftprintf(stderr, TEXT("Committed %lu bytes at address 0x%lp\n"),
              dwPageSize, lpvAddr);
  }

  // Try to lock the committed memory. This fails the first time 
  // because of the guard page.

  bLocked = VirtualLock(lpvAddr, dwPageSize);
  if (!bLocked) {
    _ftprintf(stderr, TEXT("Cannot lock at %lp, error = 0x%lx\n"),
              lpvAddr, GetLastError());
  } else {
    _ftprintf(stderr, TEXT("Lock Achieved at %lp\n"), lpvAddr);
  }

  // Try to lock the committed memory again. This succeeds the second
  // time because the guard page status was removed by the first 
  // access attempt.

  bLocked = VirtualLock(lpvAddr, dwPageSize);

  if (!bLocked) {
    _ftprintf(stderr, TEXT("Cannot get 2nd lock at %lp, error = %lx\n"),
              lpvAddr, GetLastError());
  } else {
    _ftprintf(stderr, TEXT("2nd Lock Achieved at %lp\n"), lpvAddr);
  }

  return 0;
}

Первая попытка заблокировать блок памяти завершается сбоем, что вызывает исключение STATUS_GUARD_PAGE_VIOLATION . Вторая попытка завершается успешно, так как защита страницы блока памяти была отключена первой попыткой.