#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")