Пример кода виртуальной памяти WinAPI C

#windows #winapi #visual-c

#Windows #winapi #visual-c

Вопрос:

У меня возникли некоторые проблемы с учебным кодом о распределении памяти. Ниже приведен код, который выполняет: 1) печать данных виртуальной памяти 2) выделение памяти 3) печать данных виртуальной памяти 4) освобождение памяти 5) печать данных виртуальной памяти

Я только начинаю изучать C для WinAPI. Может кто-нибудь помочь мне с кодом? Я не могу его скомпилировать.Я получаю некоторые ошибки (показаны ниже). Насколько я знаю, этот код корректно работал в более ранних версиях MS Visual Studio. В настоящее время я работаю с версией 2010. Заранее спасибо.

  #include "stdafx.h"

LPVOID myblock[5];
BOOL alloc = FALSE;

PCTSTR GetProtectText(MEMORY_BASIC_INFORMATION mbi)
{
    PCTSTR p = "Unknown";
    if(mbi.State == MEM_FREE) mbi.Protect = PAGE_NOACCESS;
    if(mbi.State == MEM_RESERVE) mbi.Protect = mbi.AllocationProtect;
    switch (mbi.Protect amp; ~(PAGE_GUARD | PAGE_NOCACHE | PAGE_WRITECOMBINE))
    {
        case PAGE_READONLY:          p = "-R--"; break;
        case PAGE_READWRITE:         p = "-RW-"; break;
        case PAGE_WRITECOPY:         p = "-RWC"; break;
        case PAGE_EXECUTE:           p = "E---"; break;
        case PAGE_EXECUTE_READ:      p = "ER--"; break;
        case PAGE_EXECUTE_READWRITE: p = "ERW-"; break;
        case PAGE_EXECUTE_WRITECOPY: p = "ERWC"; break;
        case PAGE_NOACCESS:          p = "----"; break;
    }
    return(p);
}

PCTSTR GetMemStorageText(MEMORY_BASIC_INFORMATION mbi)
{
   PCTSTR p = TEXT("Unknown");

   switch (mbi.State)
   {
        case MEM_FREE:      p = "Free";     break;
        case MEM_RESERVE:       p = "Reserve";  break;
        case MEM_COMMIT:
            switch (mbi.Type)
            {
                case MEM_FREE:  p = "Free";     break;
                case MEM_RESERVE:   p = "Reserve";  break;
                case MEM_IMAGE: p = "Image";    break;
                case MEM_MAPPED:    p = "Mapped";   break;
                case MEM_PRIVATE:   p = "Private";  break;
            }
            break;
   }
   return(p);
}

VOID MemoryMap()
{
    DWORD dwProcessId = GetCurrentProcessId(),
    dwNeeded = 0;
    HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, dwProcessId);
    HMODULE lpModule[40];  //handle to a module
    LPVOID lpModuleAddress[40];
    MODULEINFO ModuleInfo; //structure contains the module load address, size, and entry point. 

    EnumProcessModules(hProcess, lpModule, sizeof(HMODULE)*40, amp;dwNeeded);

    dwNeeded /= sizeof(HMODULE);

    for(UINT i = 0;i < dwNeeded;i  )
    {
        GetModuleInformation(hProcess, lpModule[i], amp;ModuleInfo, sizeof(ModuleInfo));
        lpModuleAddress[i] = ModuleInfo.lpBaseOfDll;
    }

    MEMORY_BASIC_INFORMATION mbi; //structure contains information about a range of                       //pages in the virtual address space of a process
    PVOID lpAddress = NULL;

    printf("  AddresstBlok SizetStoragettProtectn");
    CHAR str[255];

    while(VirtualQueryEx(hProcess, lpAddress, amp;mbi, sizeof(mbi)))
    {
        if(!(((UINT)lpAddress) % 0x10000))
        {
            printf("0x%pttt%s", lpAddress, GetMemStorageText(mbi));
            for(int i = 0;i < 5;i  )
                if(myblock[i] == lpAddress amp;amp; alloc) printf("n(My Al-location Block)");

            for(int i = 0;i < dwNeeded;i  )
                if(lpAddress == lpModuleAddress[i])
                {
                    GetModuleFileName(lpModule[i], str, 255);
                    printf("n(Image: %s)", str);
                }
            printf("n");
        }
        printf("  0x%pt0x%pt%stt%sn", lpAddress, (DWORD)mbi.RegionSize,
            GetMemStorageText(mbi), GetProtectText(mbi));
        lpAddress = ((PBYTE)lpAddress   mbi.RegionSize);
    };
}

VOID Allocation()
{
    alloc = TRUE;
    myblock[0] = VirtualAlloc(NULL, 1024, MEM_RESERVE, PAGE_READONLY);
    myblock[1] = VirtualAlloc(NULL, 1024*2, MEM_RESERVE, PAGE_READWRITE);
    myblock[2] = VirtualAlloc(NULL, 1024*8, MEM_RESERVE, PAGE_EXECUTE);
    myblock[3] = VirtualAlloc(NULL, 1024*16, MEM_RESERVE|MEM_COMMIT, PAGE_EXECUTE_READ);
    myblock[4] = VirtualAlloc(NULL, 1024*16, MEM_RESERVE|MEM_COMMIT, PAGE_EXECUTE_READWRITE);
}

VOID Release()
{
    for(int i = 0;i < 5;i  )
        VirtualFree(myblock[i], 0, MEM_RELEASE);
    alloc = FALSE;
}



INT _tmain(INT argc, _TCHAR* argv[])
{
    printf("nProcess Virtual Memory Before Allocation Memory:n");
    MemoryMap();
    printf("nProcess Virtual Memory After Allocation Memory:n");
    Allocation();
    MemoryMap();
    printf("nProcess Virtual Memory After Release Memory:n");
    Release();
    MemoryMap();
    getch();
    return 0;
}

1>------ Build started: Project: VirtMemory, Configuration: Debug Win32 ------
1>Build started 06.10.2011 23:51:01.
1>InitializeBuildStatus:
1>  Touching ".DebugVirtMemory.unsuccessfulbuild".
1>ClCompile:
1>  VirtMemory.cpp
1>e:виртуальная памятьkod1virtmemory.cpp(80): warning C4018: '<' : signed/unsigned mismatch
1>e:виртуальная памятьkod1virtmemory.cpp(123): warning C4996: 'getch': The POSIX name for this item is deprecated. Instead, use the ISO C   conformant name: _getch. See online help for details.
1>          d:program filesmicrosoft visual studio 10.0vcincludeconio.h(128) : see declaration of 'getch'
1>ManifestResourceCompile:
1>  All outputs are up-to-date.
1>VirtMemory.obj : error LNK2019: unresolved external symbol _GetModuleInformation@16 referenced in function "void __cdecl MemoryMap(void)" (?MemoryMap@@YAXXZ)
1>VirtMemory.obj : error LNK2019: unresolved external symbol _EnumProcessModules@16 referenced in function "void __cdecl MemoryMap(void)" (?MemoryMap@@YAXXZ)
1>.DebugVirtMemory.exe : fatal error LNK1120: 2 unresolved externals
1>
1>Build FAILED.
1>
1>Time Elapsed 00:00:01.43
enter code here
  

Комментарии:

1. Вам нужно будет добавить psapi.lib в параметр дополнительных зависимостей компоновщика.

2. Google говорит мне это тоже. Извините за скучный вопрос, но как я могу это сделать?. Заранее спасибо.

3. Спасибо за ваш ответ. Я нашел эти настройки и скомпилировал код.

Ответ №1:

GetModuleInformation и EnumProcessModules потребуют, чтобы вы правильно включили и связали с psapi.lib

Вы можете сделать это легко в верхней части вашего проекта добавить:

 #include <Psapi.h>
#pragma comment(lib, "psapi.lib")