2006-06-22 23:06:14 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Win32 Base API
|
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
2011-07-26 14:33:22 +00:00
|
|
|
* FILE: dll/win32/kernel32/client/virtmem.c
|
2006-06-22 23:06:14 +00:00
|
|
|
* PURPOSE: Handles virtual memory APIs
|
|
|
|
* PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
|
1996-01-23 01:02:17 +00:00
|
|
|
*/
|
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
/* INCLUDES *******************************************************************/
|
1996-01-23 01:02:17 +00:00
|
|
|
|
2003-01-15 21:24:36 +00:00
|
|
|
#include <k32.h>
|
1996-01-23 01:02:17 +00:00
|
|
|
|
2004-10-30 22:18:17 +00:00
|
|
|
#define NDEBUG
|
2007-09-02 19:42:22 +00:00
|
|
|
#include <debug.h>
|
2004-10-30 22:18:17 +00:00
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
/* FUNCTIONS ******************************************************************/
|
1996-01-23 01:02:17 +00:00
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
LPVOID
|
|
|
|
NTAPI
|
|
|
|
VirtualAllocEx(IN HANDLE hProcess,
|
|
|
|
IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize,
|
|
|
|
IN DWORD flAllocationType,
|
|
|
|
IN DWORD flProtect)
|
1998-10-05 04:01:30 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
NTSTATUS Status;
|
2011-11-05 09:07:39 +00:00
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
/* Make sure the address is within the granularity of the system (64K) */
|
2018-02-15 21:34:48 +00:00
|
|
|
if ((lpAddress != NULL) &&
|
|
|
|
(lpAddress < UlongToPtr(BaseStaticServerData->SysInfo.AllocationGranularity)))
|
2011-07-26 14:33:22 +00:00
|
|
|
{
|
|
|
|
/* Fail the call */
|
|
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-11-05 09:07:39 +00:00
|
|
|
|
2023-03-09 16:08:08 +00:00
|
|
|
/* Allocate the memory */
|
|
|
|
Status = NtAllocateVirtualMemory(hProcess,
|
|
|
|
&lpAddress,
|
|
|
|
0,
|
|
|
|
&dwSize,
|
|
|
|
flAllocationType,
|
|
|
|
flProtect);
|
2011-11-05 09:07:39 +00:00
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
/* Check for status */
|
2006-06-22 23:06:14 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
2001-08-07 14:13:45 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* We failed */
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2006-06-22 23:06:14 +00:00
|
|
|
return NULL;
|
2001-08-07 14:13:45 +00:00
|
|
|
}
|
1998-10-05 04:01:30 +00:00
|
|
|
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Return the allocated address */
|
|
|
|
return lpAddress;
|
|
|
|
}
|
2001-08-07 14:13:45 +00:00
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
LPVOID
|
|
|
|
NTAPI
|
|
|
|
VirtualAlloc(IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize,
|
|
|
|
IN DWORD flAllocationType,
|
|
|
|
IN DWORD flProtect)
|
1998-10-05 04:01:30 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Call the extended API */
|
|
|
|
return VirtualAllocEx(GetCurrentProcess(),
|
|
|
|
lpAddress,
|
|
|
|
dwSize,
|
|
|
|
flAllocationType,
|
|
|
|
flProtect);
|
1998-10-05 04:01:30 +00:00
|
|
|
}
|
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
BOOL
|
|
|
|
NTAPI
|
|
|
|
VirtualFreeEx(IN HANDLE hProcess,
|
|
|
|
IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize,
|
|
|
|
IN DWORD dwFreeType)
|
1998-10-05 04:01:30 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
NTSTATUS Status;
|
2001-08-07 14:13:45 +00:00
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
/* Validate size and flags */
|
|
|
|
if (!(dwSize) || !(dwFreeType & MEM_RELEASE))
|
2001-08-07 14:13:45 +00:00
|
|
|
{
|
2008-12-28 13:03:05 +00:00
|
|
|
/* Free the memory */
|
|
|
|
Status = NtFreeVirtualMemory(hProcess,
|
2011-07-26 14:33:22 +00:00
|
|
|
&lpAddress,
|
|
|
|
&dwSize,
|
2008-12-28 13:03:05 +00:00
|
|
|
dwFreeType);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* We failed */
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2008-12-28 13:03:05 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return success */
|
|
|
|
return TRUE;
|
2001-08-07 14:13:45 +00:00
|
|
|
}
|
1998-10-05 04:01:30 +00:00
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
/* Invalid combo */
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(STATUS_INVALID_PARAMETER);
|
2008-12-28 13:03:05 +00:00
|
|
|
return FALSE;
|
2006-06-22 23:06:14 +00:00
|
|
|
}
|
2001-08-07 14:13:45 +00:00
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
BOOL
|
|
|
|
NTAPI
|
|
|
|
VirtualFree(IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize,
|
|
|
|
IN DWORD dwFreeType)
|
1996-01-23 01:02:17 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Call the extended API */
|
|
|
|
return VirtualFreeEx(GetCurrentProcess(),
|
|
|
|
lpAddress,
|
|
|
|
dwSize,
|
|
|
|
dwFreeType);
|
1996-01-23 01:02:17 +00:00
|
|
|
}
|
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
BOOL
|
|
|
|
NTAPI
|
|
|
|
VirtualProtect(IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize,
|
|
|
|
IN DWORD flNewProtect,
|
|
|
|
OUT PDWORD lpflOldProtect)
|
1998-10-05 04:01:30 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Call the extended API */
|
|
|
|
return VirtualProtectEx(GetCurrentProcess(),
|
|
|
|
lpAddress,
|
|
|
|
dwSize,
|
|
|
|
flNewProtect,
|
|
|
|
lpflOldProtect);
|
1998-10-05 04:01:30 +00:00
|
|
|
}
|
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
BOOL
|
|
|
|
NTAPI
|
|
|
|
VirtualProtectEx(IN HANDLE hProcess,
|
|
|
|
IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize,
|
|
|
|
IN DWORD flNewProtect,
|
|
|
|
OUT PDWORD lpflOldProtect)
|
1998-10-05 04:01:30 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
/* Change the protection */
|
|
|
|
Status = NtProtectVirtualMemory(hProcess,
|
|
|
|
&lpAddress,
|
|
|
|
&dwSize,
|
|
|
|
flNewProtect,
|
|
|
|
(PULONG)lpflOldProtect);
|
|
|
|
if (!NT_SUCCESS(Status))
|
2001-08-07 14:13:45 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* We failed */
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2006-06-22 23:06:14 +00:00
|
|
|
return FALSE;
|
2001-08-07 14:13:45 +00:00
|
|
|
}
|
1999-10-02 20:20:44 +00:00
|
|
|
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Return success */
|
|
|
|
return TRUE;
|
|
|
|
}
|
1999-10-02 20:20:44 +00:00
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
BOOL
|
|
|
|
NTAPI
|
|
|
|
VirtualLock(IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize)
|
1999-10-02 20:20:44 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
NTSTATUS Status;
|
2011-11-05 09:07:39 +00:00
|
|
|
SIZE_T RegionSize = dwSize;
|
2009-10-15 16:50:49 +00:00
|
|
|
PVOID BaseAddress = lpAddress;
|
2006-06-22 23:06:14 +00:00
|
|
|
|
|
|
|
/* Lock the memory */
|
|
|
|
Status = NtLockVirtualMemory(NtCurrentProcess(),
|
2009-10-15 16:50:49 +00:00
|
|
|
&BaseAddress,
|
|
|
|
&RegionSize,
|
|
|
|
MAP_PROCESS);
|
2006-06-22 23:06:14 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
2001-08-07 14:13:45 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* We failed */
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2006-06-22 23:06:14 +00:00
|
|
|
return FALSE;
|
2001-08-07 14:13:45 +00:00
|
|
|
}
|
1999-10-02 20:20:44 +00:00
|
|
|
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Return success */
|
|
|
|
return TRUE;
|
|
|
|
}
|
1999-10-02 20:20:44 +00:00
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2011-11-05 09:07:39 +00:00
|
|
|
SIZE_T
|
2006-06-22 23:06:14 +00:00
|
|
|
NTAPI
|
|
|
|
VirtualQuery(IN LPCVOID lpAddress,
|
|
|
|
OUT PMEMORY_BASIC_INFORMATION lpBuffer,
|
|
|
|
IN SIZE_T dwLength)
|
1999-10-02 20:20:44 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Call the extended API */
|
|
|
|
return VirtualQueryEx(NtCurrentProcess(),
|
|
|
|
lpAddress,
|
|
|
|
lpBuffer,
|
|
|
|
dwLength);
|
1999-10-02 20:20:44 +00:00
|
|
|
}
|
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2011-11-05 09:07:39 +00:00
|
|
|
SIZE_T
|
2006-06-22 23:06:14 +00:00
|
|
|
NTAPI
|
|
|
|
VirtualQueryEx(IN HANDLE hProcess,
|
|
|
|
IN LPCVOID lpAddress,
|
|
|
|
OUT PMEMORY_BASIC_INFORMATION lpBuffer,
|
|
|
|
IN SIZE_T dwLength)
|
1999-10-02 20:20:44 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
NTSTATUS Status;
|
2011-11-05 09:07:39 +00:00
|
|
|
SIZE_T ResultLength;
|
2006-06-22 23:06:14 +00:00
|
|
|
|
|
|
|
/* Query basic information */
|
|
|
|
Status = NtQueryVirtualMemory(hProcess,
|
|
|
|
(LPVOID)lpAddress,
|
|
|
|
MemoryBasicInformation,
|
|
|
|
lpBuffer,
|
2008-12-28 13:03:05 +00:00
|
|
|
dwLength,
|
2006-06-22 23:06:14 +00:00
|
|
|
&ResultLength);
|
|
|
|
if (!NT_SUCCESS(Status))
|
2001-08-07 14:13:45 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* We failed */
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2006-06-22 23:06:14 +00:00
|
|
|
return 0;
|
2001-08-07 14:13:45 +00:00
|
|
|
}
|
1999-10-02 20:20:44 +00:00
|
|
|
|
2006-06-22 23:06:14 +00:00
|
|
|
/* Return the length returned */
|
|
|
|
return ResultLength;
|
|
|
|
}
|
1999-10-02 20:20:44 +00:00
|
|
|
|
2003-07-10 18:50:51 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2006-06-22 23:06:14 +00:00
|
|
|
BOOL
|
|
|
|
NTAPI
|
|
|
|
VirtualUnlock(IN LPVOID lpAddress,
|
|
|
|
IN SIZE_T dwSize)
|
1999-10-02 20:20:44 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
NTSTATUS Status;
|
2011-11-05 09:07:39 +00:00
|
|
|
SIZE_T RegionSize = dwSize;
|
2009-10-15 16:50:49 +00:00
|
|
|
PVOID BaseAddress = lpAddress;
|
2011-11-05 09:07:39 +00:00
|
|
|
|
2009-10-15 16:50:49 +00:00
|
|
|
/* Lock the memory */
|
2006-06-22 23:06:14 +00:00
|
|
|
Status = NtUnlockVirtualMemory(NtCurrentProcess(),
|
2009-10-15 16:50:49 +00:00
|
|
|
&BaseAddress,
|
|
|
|
&RegionSize,
|
|
|
|
MAP_PROCESS);
|
2006-06-22 23:06:14 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
2001-08-07 14:13:45 +00:00
|
|
|
{
|
2006-06-22 23:06:14 +00:00
|
|
|
/* We failed */
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2006-06-22 23:06:14 +00:00
|
|
|
return FALSE;
|
2001-08-07 14:13:45 +00:00
|
|
|
}
|
2006-06-22 23:06:14 +00:00
|
|
|
|
|
|
|
/* Return success */
|
|
|
|
return TRUE;
|
1999-10-02 20:20:44 +00:00
|
|
|
}
|
2000-07-01 17:07:02 +00:00
|
|
|
|
2008-12-28 13:03:05 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
UINT
|
|
|
|
WINAPI
|
2011-07-26 14:33:22 +00:00
|
|
|
GetWriteWatch(IN DWORD dwFlags,
|
|
|
|
IN PVOID lpBaseAddress,
|
|
|
|
IN SIZE_T dwRegionSize,
|
|
|
|
IN PVOID *lpAddresses,
|
|
|
|
OUT PULONG_PTR lpdwCount,
|
|
|
|
OUT PULONG lpdwGranularity)
|
2008-12-28 13:03:05 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
Status = NtGetWriteWatch(GetCurrentProcess(),
|
|
|
|
dwFlags,
|
|
|
|
lpBaseAddress,
|
|
|
|
dwRegionSize,
|
|
|
|
lpAddresses,
|
|
|
|
lpdwCount,
|
|
|
|
lpdwGranularity);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2009-01-17 15:27:35 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
UINT
|
|
|
|
WINAPI
|
2011-07-26 14:33:22 +00:00
|
|
|
ResetWriteWatch(IN LPVOID lpBaseAddress,
|
|
|
|
IN SIZE_T dwRegionSize)
|
2009-01-17 15:27:35 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
Status = NtResetWriteWatch(NtCurrentProcess(),
|
|
|
|
lpBaseAddress,
|
|
|
|
dwRegionSize);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2008-12-28 13:03:05 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-01-17 15:27:35 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
|
|
|
WINAPI
|
2011-07-26 14:33:22 +00:00
|
|
|
AllocateUserPhysicalPages(IN HANDLE hProcess,
|
|
|
|
IN PULONG_PTR NumberOfPages,
|
|
|
|
OUT PULONG_PTR UserPfnArray)
|
2009-01-17 15:27:35 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
Status = NtAllocateUserPhysicalPages(hProcess, NumberOfPages, UserPfnArray);
|
2009-01-17 15:27:35 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2009-01-17 15:27:35 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
|
|
|
WINAPI
|
2011-07-26 14:33:22 +00:00
|
|
|
FreeUserPhysicalPages(IN HANDLE hProcess,
|
|
|
|
IN PULONG_PTR NumberOfPages,
|
|
|
|
IN PULONG_PTR PageArray)
|
2009-01-17 15:27:35 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
Status = NtFreeUserPhysicalPages(hProcess, NumberOfPages, PageArray);
|
2009-01-17 15:27:35 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2009-01-17 15:27:35 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
|
|
|
WINAPI
|
2011-07-26 14:33:22 +00:00
|
|
|
MapUserPhysicalPages(IN PVOID VirtualAddress,
|
|
|
|
IN ULONG_PTR NumberOfPages,
|
|
|
|
OUT PULONG_PTR PageArray OPTIONAL)
|
2009-01-17 15:27:35 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
2011-07-26 14:33:22 +00:00
|
|
|
Status = NtMapUserPhysicalPages(VirtualAddress, NumberOfPages, PageArray);
|
2009-01-17 15:27:35 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2009-01-17 15:27:35 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
|
|
|
WINAPI
|
2011-07-26 14:33:22 +00:00
|
|
|
MapUserPhysicalPagesScatter(IN PVOID *VirtualAddresses,
|
|
|
|
IN ULONG_PTR NumberOfPages,
|
|
|
|
OUT PULONG_PTR PageArray OPTIONAL)
|
2009-01-17 15:27:35 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
Status = NtMapUserPhysicalPagesScatter(VirtualAddresses,
|
|
|
|
NumberOfPages,
|
|
|
|
PageArray);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2011-07-23 18:54:29 +00:00
|
|
|
BaseSetLastNTError(Status);
|
2009-01-17 15:27:35 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2000-07-01 17:07:02 +00:00
|
|
|
/* EOF */
|