2008-03-09 14:11:42 +00:00
|
|
|
/*
|
2005-01-26 13:58:37 +00:00
|
|
|
* COPYRIGHT: See COPYING in the top directory
|
2005-05-09 01:38:29 +00:00
|
|
|
* PROJECT: ReactOS kernel
|
2005-01-26 13:58:37 +00:00
|
|
|
* FILE: ntoskrnl/mm/rmap.c
|
|
|
|
* PURPOSE: Kernel memory managment functions
|
2005-05-09 01:38:29 +00:00
|
|
|
*
|
2005-01-26 13:58:37 +00:00
|
|
|
* PROGRAMMERS: David Welch (welch@cwcom.net)
|
2001-12-31 01:53:46 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
|
2004-08-15 16:39:12 +00:00
|
|
|
#include <ntoskrnl.h>
|
2014-11-10 16:26:55 +00:00
|
|
|
#include <cache/section/newmm.h>
|
2001-12-31 01:53:46 +00:00
|
|
|
#define NDEBUG
|
2008-08-30 16:31:06 +00:00
|
|
|
#include <debug.h>
|
2001-12-31 01:53:46 +00:00
|
|
|
|
2005-11-28 23:25:31 +00:00
|
|
|
#if defined (ALLOC_PRAGMA)
|
|
|
|
#pragma alloc_text(INIT, MmInitializeRmapList)
|
|
|
|
#endif
|
|
|
|
|
2001-12-31 01:53:46 +00:00
|
|
|
/* TYPES ********************************************************************/
|
|
|
|
|
|
|
|
/* GLOBALS ******************************************************************/
|
|
|
|
|
2003-01-11 15:28:18 +00:00
|
|
|
static NPAGED_LOOKASIDE_LIST RmapLookasideList;
|
[CACHE]
The cache manager rewrite I started years ago has finally appeared in
ReactOS' trunk and although at this point it's not quite perfectly
integrated, it's enough to boot up the bootcd or livecd. To check out
the more mature original, check out arty-newcc-reactos, branch
arty-newcc on bitbucket.org . Amine Khaldi encouraged me quite a bit
to not give up on it, and was able to reach out and be an advocate
when i really wasn't able to. Others agree that the time has come to
begin removing the old cache manager. I expect the remaining problems
in the version going to trunk will be taken care of relatively
quickly.
The motivation for this effort lies in the particularly hairy
relationship between ReactOS' cache manager and data sections. This
code completely removes page sharing between cache manager and section
and reimagines cache manager as being a facility layered on the memory
manager, not really caring about individual pages, but simply managing
data section objects where caching might occur.
It took me about 2 years to do the first pass of this rewrite and most
of this year to fix some lingering issues, properly implement demand
paging in ReactOS (code which didn't come with this patch in a
recognizable form), and finish getting the PrivateCacheMap and
SharedCacheMap relationship correct.
Currently, the new ntoskrnl/cache directory contains an own
implementation of data file sections. After things have settled down,
we can begin to deprecate and remove the parts of ReactOS' section
implementation that depend on a close relationship with cache
manager. Eventually, I think that the extra code added to
ntoskrnl/cache/section will be removed and ReactOS' own sections will
replace the use of the special MM_CACHE_SECTION_SEGMENT in the cache
path.
Note also, that this makes all cache manager (and new section parts)
use wide file offsets. If my section code were to take over other
parts of the ReactOS memory manager, they would also benefit from
these improvements.
I invite anyone who wants to to peek at this code and fix whatever
bugs can be found.
svn path=/trunk/; revision=49423
2010-11-02 02:32:39 +00:00
|
|
|
FAST_MUTEX RmapListLock;
|
2001-12-31 01:53:46 +00:00
|
|
|
|
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
|
2013-02-01 09:39:54 +00:00
|
|
|
_IRQL_requires_max_(DISPATCH_LEVEL)
|
|
|
|
static
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
RmapListFree(
|
|
|
|
_In_ __drv_freesMem(Mem) PVOID P)
|
|
|
|
{
|
|
|
|
ExFreePoolWithTag(P, TAG_RMAP);
|
|
|
|
}
|
|
|
|
|
2005-09-14 01:05:50 +00:00
|
|
|
VOID
|
|
|
|
INIT_FUNCTION
|
|
|
|
NTAPI
|
2001-12-31 01:53:46 +00:00
|
|
|
MmInitializeRmapList(VOID)
|
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
ExInitializeFastMutex(&RmapListLock);
|
|
|
|
ExInitializeNPagedLookasideList (&RmapLookasideList,
|
|
|
|
NULL,
|
|
|
|
RmapListFree,
|
|
|
|
0,
|
|
|
|
sizeof(MM_RMAP_ENTRY),
|
|
|
|
TAG_RMAP,
|
|
|
|
50);
|
2001-12-31 01:53:46 +00:00
|
|
|
}
|
|
|
|
|
2002-08-14 20:58:39 +00:00
|
|
|
NTSTATUS
|
2005-09-14 01:05:50 +00:00
|
|
|
NTAPI
|
2010-07-15 22:50:12 +00:00
|
|
|
MmPageOutPhysicalAddress(PFN_NUMBER Page)
|
2002-08-14 20:58:39 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY entry;
|
|
|
|
PMEMORY_AREA MemoryArea;
|
|
|
|
PMMSUPPORT AddressSpace;
|
|
|
|
ULONG Type;
|
|
|
|
PVOID Address;
|
|
|
|
PEPROCESS Process;
|
|
|
|
ULONGLONG Offset;
|
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
entry = MmGetRmapListHeadPage(Page);
|
[NEWCC]
A reintegration checkpoint for the NewCC branch, brought to you by Team NewCC.
Differences with current ReactOS trunk:
* A new memory area type, MEMORY_AREA_CACHE, is added, which represents a mapped region of a file. In NEWCC mode, user sections are MEMORY_AREA_CACHE type as well, and obey new semantics. In non-NEWCC mode, they aren't used.
* A way of claiming a page entry for a specific thread's work is added. Placing the special SWAPENTRY value MM_WAIT_ENTRY in a page table, or in a section page table should indicate that memory management code is intended to wait for another thread to make some status change before checking the state of the page entry again. In code that uses this convention, a return value of STATUS_SUCCESS + 1 is used to indicate that the caller should use the MiWaitForPageEvent macro to wait until somebody has change the state of a wait entry before checking again. This is a lighter weight mechanism than PAGEOPs.
* A way of asking the caller to perform some blocking operation without locks held is provided. This replaces some spaghettified code in which locks are repeatedly taken and broken by code that performs various blocking operations. Using this mechanism, it is possible to do a small amount of non-blocking work, fill in a request, then return STATUS_MORE_PROCESSING_REQUIRED to request that locks be dropped and the blocking operation be carried out. A MM_REQUIRED_RESOURCES structure is provided to consumers of this contract to use to accumulate state across many blocking operations. Several functions wrapping blocking operations are provided in ntoskrnl/cache/reqtools.c.
* Image section pages are no longer direct mapped. This is done to simplify consolidation of ownership of pages under the data section system. At a later time, it may be possible to make data pages directly available to image sections for the same file. This is likely the only direct performance impact this code makes on non-NEWCC mode.
RMAPs:
* A new type of RMAP entry is introduced, distinguished by RMAP_IS_SEGMENT(Address) of the rmap entry. This kind of entry contains a pointer to a section page table node in the Process pointer, which in turn links back to the MM_SECTION_SEGMENT it belongs to. Therefore, a page belonging only to a segment (that is, a segment page that isn't mapped) can exist and be evicted using the normal page eviction mechanism in balance.c. Each of the rmap function has been modified to deal with segment rmaps.
* The low 8 bits of the Address field in a segment rmap denote the entry number in the generic table node pointed to by Process that points to the page the rmap belongs to. By combining them, you can determine the file offset the page belongs to.
* In NEWCC mode, MmSharePageEntry/UnsharePageEntry are not used, and instead the page reference count is used to keep track of the number of mappings of a page, allowing the last reference expiring to allow the page to be recycled without much intervention. These are still used in non-NEWCC mode. One change has been made, the count fields have been narrowed by 1 bit to make room for a dirty bit in SSE entries, needed when a page is present but unmapped.
Section page tables:
* The section page tables are now implemented using RtlGenericTables. This enables a fairly compact representation of section page tables without having the existence of a section object imply 4k of fake PDEs. In addition, each node in the generic table has a wide file offset that is a multiple of 256 pages, or 1 megabyte total. Besides needing wide file offsets, the only other visible change caused by the switch to generic tables for section page tables is the need to lock the section segment before interacting with the section page table.
Eviction:
* Page eviction in cache sections is accomplished by MmpPageOutPhysicalAddress. In the case of a shared page, it tries to remove all mappings of the indicated page. If this process fails at any point, the page will simply be drawn back into the target address spaces. After succeeding at this, if TRUE has been accumulated into the page's dirty bit in the section page table, it is written back, and then permanently removed.
NewCC mode:
* NEWCC mode is introduced, which rewrites the file cache to a set of cache stripes actively mapped, along with unmapped section data.
* NewCC is more authentic in its interpretation of the external interface to the windows cache than the current cache manager, implementing each of the cache manager functions according to the documented interface with no preconceived ideas about how anything should be implemented internally. Cache stripes are implemented on top of section objects, using the same memory manager paths, and therefore economizing code and complexity. This replaces a rather complicated system in which pages can be owned by the cache manager and the memory manager simultaneously and they must cooperate in a fairly sophisticated way to manage them. Since they're quite interdependent in the current code, modifying either is very difficult. In NEWCC, they have a clear division of labor and thus can be worked on independently.
* Several third party filesystems that use the kernel Cc interface work properly using NEWCC, including matt wu's ext3 driver.
* In contrast with code that tries to make CcInitializeCacheMap and CcUninitializeCacheMap into a pair that supports reference counting, NEWCC lazily initializes the shared and private cache maps as needed and uses the presence of a PrivateCacheMap on at least one file pointing to the SharedCacheMap as an indication that the FILE_OBJECT reference in the SharedCacheMap should still be held. When the last PrivateCacheMap is discarded, that's the appropriate time to tear down caching for a specific file, as the SharedCacheMap data is allowed to be saved and reused. We honor this by making the SharedCacheMap into a depot for keeping track of the PrivateCacheMap objects associated with views of a file.
svn path=/trunk/; revision=55833
2012-02-23 12:03:06 +00:00
|
|
|
|
|
|
|
#ifdef NEWCC
|
2014-10-05 06:33:34 +00:00
|
|
|
// Special case for NEWCC: we can have a page that's only in a segment
|
|
|
|
// page table
|
|
|
|
if (entry && RMAP_IS_SEGMENT(entry->Address) && entry->Next == NULL)
|
|
|
|
{
|
|
|
|
/* NEWCC does locking itself */
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return MmpPageOutPhysicalAddress(Page);
|
|
|
|
}
|
[NEWCC]
A reintegration checkpoint for the NewCC branch, brought to you by Team NewCC.
Differences with current ReactOS trunk:
* A new memory area type, MEMORY_AREA_CACHE, is added, which represents a mapped region of a file. In NEWCC mode, user sections are MEMORY_AREA_CACHE type as well, and obey new semantics. In non-NEWCC mode, they aren't used.
* A way of claiming a page entry for a specific thread's work is added. Placing the special SWAPENTRY value MM_WAIT_ENTRY in a page table, or in a section page table should indicate that memory management code is intended to wait for another thread to make some status change before checking the state of the page entry again. In code that uses this convention, a return value of STATUS_SUCCESS + 1 is used to indicate that the caller should use the MiWaitForPageEvent macro to wait until somebody has change the state of a wait entry before checking again. This is a lighter weight mechanism than PAGEOPs.
* A way of asking the caller to perform some blocking operation without locks held is provided. This replaces some spaghettified code in which locks are repeatedly taken and broken by code that performs various blocking operations. Using this mechanism, it is possible to do a small amount of non-blocking work, fill in a request, then return STATUS_MORE_PROCESSING_REQUIRED to request that locks be dropped and the blocking operation be carried out. A MM_REQUIRED_RESOURCES structure is provided to consumers of this contract to use to accumulate state across many blocking operations. Several functions wrapping blocking operations are provided in ntoskrnl/cache/reqtools.c.
* Image section pages are no longer direct mapped. This is done to simplify consolidation of ownership of pages under the data section system. At a later time, it may be possible to make data pages directly available to image sections for the same file. This is likely the only direct performance impact this code makes on non-NEWCC mode.
RMAPs:
* A new type of RMAP entry is introduced, distinguished by RMAP_IS_SEGMENT(Address) of the rmap entry. This kind of entry contains a pointer to a section page table node in the Process pointer, which in turn links back to the MM_SECTION_SEGMENT it belongs to. Therefore, a page belonging only to a segment (that is, a segment page that isn't mapped) can exist and be evicted using the normal page eviction mechanism in balance.c. Each of the rmap function has been modified to deal with segment rmaps.
* The low 8 bits of the Address field in a segment rmap denote the entry number in the generic table node pointed to by Process that points to the page the rmap belongs to. By combining them, you can determine the file offset the page belongs to.
* In NEWCC mode, MmSharePageEntry/UnsharePageEntry are not used, and instead the page reference count is used to keep track of the number of mappings of a page, allowing the last reference expiring to allow the page to be recycled without much intervention. These are still used in non-NEWCC mode. One change has been made, the count fields have been narrowed by 1 bit to make room for a dirty bit in SSE entries, needed when a page is present but unmapped.
Section page tables:
* The section page tables are now implemented using RtlGenericTables. This enables a fairly compact representation of section page tables without having the existence of a section object imply 4k of fake PDEs. In addition, each node in the generic table has a wide file offset that is a multiple of 256 pages, or 1 megabyte total. Besides needing wide file offsets, the only other visible change caused by the switch to generic tables for section page tables is the need to lock the section segment before interacting with the section page table.
Eviction:
* Page eviction in cache sections is accomplished by MmpPageOutPhysicalAddress. In the case of a shared page, it tries to remove all mappings of the indicated page. If this process fails at any point, the page will simply be drawn back into the target address spaces. After succeeding at this, if TRUE has been accumulated into the page's dirty bit in the section page table, it is written back, and then permanently removed.
NewCC mode:
* NEWCC mode is introduced, which rewrites the file cache to a set of cache stripes actively mapped, along with unmapped section data.
* NewCC is more authentic in its interpretation of the external interface to the windows cache than the current cache manager, implementing each of the cache manager functions according to the documented interface with no preconceived ideas about how anything should be implemented internally. Cache stripes are implemented on top of section objects, using the same memory manager paths, and therefore economizing code and complexity. This replaces a rather complicated system in which pages can be owned by the cache manager and the memory manager simultaneously and they must cooperate in a fairly sophisticated way to manage them. Since they're quite interdependent in the current code, modifying either is very difficult. In NEWCC, they have a clear division of labor and thus can be worked on independently.
* Several third party filesystems that use the kernel Cc interface work properly using NEWCC, including matt wu's ext3 driver.
* In contrast with code that tries to make CcInitializeCacheMap and CcUninitializeCacheMap into a pair that supports reference counting, NEWCC lazily initializes the shared and private cache maps as needed and uses the presence of a PrivateCacheMap on at least one file pointing to the SharedCacheMap as an indication that the FILE_OBJECT reference in the SharedCacheMap should still be held. When the last PrivateCacheMap is discarded, that's the appropriate time to tear down caching for a specific file, as the SharedCacheMap data is allowed to be saved and reused. We honor this by making the SharedCacheMap into a depot for keeping track of the PrivateCacheMap objects associated with views of a file.
svn path=/trunk/; revision=55833
2012-02-23 12:03:06 +00:00
|
|
|
#endif
|
|
|
|
|
2014-10-05 06:33:34 +00:00
|
|
|
while (entry && RMAP_IS_SEGMENT(entry->Address))
|
|
|
|
entry = entry->Next;
|
|
|
|
|
|
|
|
if (entry == NULL)
|
|
|
|
{
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return(STATUS_UNSUCCESSFUL);
|
|
|
|
}
|
|
|
|
|
|
|
|
Process = entry->Process;
|
|
|
|
|
|
|
|
Address = entry->Address;
|
|
|
|
|
|
|
|
if ((((ULONG_PTR)Address) & 0xFFF) != 0)
|
|
|
|
{
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Address < MmSystemRangeStart)
|
|
|
|
{
|
|
|
|
if (!ExAcquireRundownProtection(&Process->RundownProtect))
|
|
|
|
{
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return STATUS_PROCESS_IS_TERMINATING;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = ObReferenceObjectByPointer(Process, PROCESS_ALL_ACCESS, NULL, KernelMode);
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
ExReleaseRundownProtection(&Process->RundownProtect);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
AddressSpace = &Process->Vm;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
AddressSpace = MmGetKernelAddressSpace();
|
|
|
|
}
|
|
|
|
|
|
|
|
MmLockAddressSpace(AddressSpace);
|
|
|
|
MemoryArea = MmLocateMemoryAreaByAddress(AddressSpace, Address);
|
|
|
|
if (MemoryArea == NULL || MemoryArea->DeleteInProgress)
|
|
|
|
{
|
|
|
|
MmUnlockAddressSpace(AddressSpace);
|
|
|
|
if (Address < MmSystemRangeStart)
|
|
|
|
{
|
2010-09-28 16:44:18 +00:00
|
|
|
ExReleaseRundownProtection(&Process->RundownProtect);
|
2004-04-10 22:36:07 +00:00
|
|
|
ObDereferenceObject(Process);
|
2014-10-05 06:33:34 +00:00
|
|
|
}
|
|
|
|
return(STATUS_UNSUCCESSFUL);
|
|
|
|
}
|
|
|
|
Type = MemoryArea->Type;
|
|
|
|
if (Type == MEMORY_AREA_SECTION_VIEW)
|
|
|
|
{
|
|
|
|
ULONG_PTR Entry;
|
|
|
|
Offset = MemoryArea->Data.SectionData.ViewOffset.QuadPart +
|
2015-05-16 20:10:03 +00:00
|
|
|
((ULONG_PTR)Address - MA_GetStartingAddress(MemoryArea));
|
2014-10-05 06:33:34 +00:00
|
|
|
|
|
|
|
MmLockSectionSegment(MemoryArea->Data.SectionData.Segment);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get or create a pageop
|
|
|
|
*/
|
|
|
|
Entry = MmGetPageEntrySectionSegment(MemoryArea->Data.SectionData.Segment,
|
|
|
|
(PLARGE_INTEGER)&Offset);
|
2016-11-20 10:51:26 +00:00
|
|
|
if (Entry && MM_IS_WAIT_PTE(Entry))
|
2014-10-05 06:33:34 +00:00
|
|
|
{
|
|
|
|
MmUnlockSectionSegment(MemoryArea->Data.SectionData.Segment);
|
|
|
|
MmUnlockAddressSpace(AddressSpace);
|
|
|
|
if (Address < MmSystemRangeStart)
|
|
|
|
{
|
|
|
|
ExReleaseRundownProtection(&Process->RundownProtect);
|
|
|
|
ObDereferenceObject(Process);
|
|
|
|
}
|
|
|
|
return(STATUS_UNSUCCESSFUL);
|
|
|
|
}
|
|
|
|
|
|
|
|
MmSetPageEntrySectionSegment(MemoryArea->Data.SectionData.Segment, (PLARGE_INTEGER)&Offset, MAKE_SWAP_SSE(MM_WAIT_ENTRY));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Release locks now we have a page op.
|
|
|
|
*/
|
|
|
|
MmUnlockSectionSegment(MemoryArea->Data.SectionData.Segment);
|
|
|
|
MmUnlockAddressSpace(AddressSpace);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Do the actual page out work.
|
|
|
|
*/
|
|
|
|
Status = MmPageOutSectionView(AddressSpace, MemoryArea, Address, Entry);
|
|
|
|
}
|
|
|
|
else if (Type == MEMORY_AREA_CACHE)
|
|
|
|
{
|
|
|
|
/* NEWCC does locking itself */
|
|
|
|
MmUnlockAddressSpace(AddressSpace);
|
|
|
|
Status = MmpPageOutPhysicalAddress(Page);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Address < MmSystemRangeStart)
|
|
|
|
{
|
|
|
|
ExReleaseRundownProtection(&Process->RundownProtect);
|
|
|
|
ObDereferenceObject(Process);
|
|
|
|
}
|
|
|
|
return(Status);
|
2001-12-31 01:53:46 +00:00
|
|
|
}
|
|
|
|
|
2002-08-14 20:58:39 +00:00
|
|
|
VOID
|
2005-09-14 01:05:50 +00:00
|
|
|
NTAPI
|
2010-07-15 22:50:12 +00:00
|
|
|
MmSetCleanAllRmaps(PFN_NUMBER Page)
|
2002-08-14 20:58:39 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY current_entry;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
if (current_entry == NULL)
|
|
|
|
{
|
|
|
|
DPRINT1("MmIsDirtyRmap: No rmaps.\n");
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
while (current_entry != NULL)
|
|
|
|
{
|
|
|
|
if (!RMAP_IS_SEGMENT(current_entry->Address))
|
|
|
|
MmSetCleanPage(current_entry->Process, current_entry->Address);
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
}
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
2002-08-14 20:58:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
2005-09-14 01:05:50 +00:00
|
|
|
NTAPI
|
2010-07-15 22:50:12 +00:00
|
|
|
MmSetDirtyAllRmaps(PFN_NUMBER Page)
|
2002-08-14 20:58:39 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY current_entry;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
if (current_entry == NULL)
|
|
|
|
{
|
|
|
|
DPRINT1("MmIsDirtyRmap: No rmaps.\n");
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
while (current_entry != NULL)
|
|
|
|
{
|
|
|
|
if (!RMAP_IS_SEGMENT(current_entry->Address))
|
|
|
|
MmSetDirtyPage(current_entry->Process, current_entry->Address);
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
}
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
2002-08-14 20:58:39 +00:00
|
|
|
}
|
|
|
|
|
2006-01-08 06:23:17 +00:00
|
|
|
BOOLEAN
|
2005-09-14 01:05:50 +00:00
|
|
|
NTAPI
|
2010-07-15 22:50:12 +00:00
|
|
|
MmIsDirtyPageRmap(PFN_NUMBER Page)
|
2002-08-14 20:58:39 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY current_entry;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
if (current_entry == NULL)
|
|
|
|
{
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
while (current_entry != NULL)
|
|
|
|
{
|
|
|
|
if (
|
|
|
|
!RMAP_IS_SEGMENT(current_entry->Address) &&
|
|
|
|
MmIsDirtyPage(current_entry->Process, current_entry->Address))
|
|
|
|
{
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return(TRUE);
|
|
|
|
}
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
}
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return(FALSE);
|
2002-08-14 20:58:39 +00:00
|
|
|
}
|
|
|
|
|
2002-05-13 18:10:41 +00:00
|
|
|
VOID
|
2005-09-14 01:05:50 +00:00
|
|
|
NTAPI
|
2010-07-15 22:50:12 +00:00
|
|
|
MmInsertRmap(PFN_NUMBER Page, PEPROCESS Process,
|
2004-04-10 22:36:07 +00:00
|
|
|
PVOID Address)
|
2002-05-13 18:10:41 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY current_entry;
|
|
|
|
PMM_RMAP_ENTRY new_entry;
|
|
|
|
ULONG PrevSize;
|
|
|
|
if (!RMAP_IS_SEGMENT(Address))
|
|
|
|
Address = (PVOID)PAGE_ROUND_DOWN(Address);
|
|
|
|
|
|
|
|
new_entry = ExAllocateFromNPagedLookasideList(&RmapLookasideList);
|
|
|
|
if (new_entry == NULL)
|
|
|
|
{
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
new_entry->Address = Address;
|
|
|
|
new_entry->Process = (PEPROCESS)Process;
|
2009-06-17 12:44:05 +00:00
|
|
|
#if DBG
|
2007-03-26 20:28:13 +00:00
|
|
|
#ifdef __GNUC__
|
2014-10-05 06:33:34 +00:00
|
|
|
new_entry->Caller = __builtin_return_address(0);
|
2007-03-02 18:25:23 +00:00
|
|
|
#else
|
2014-10-05 06:33:34 +00:00
|
|
|
new_entry->Caller = _ReturnAddress();
|
2007-03-02 18:25:23 +00:00
|
|
|
#endif
|
2007-10-19 23:21:45 +00:00
|
|
|
#endif
|
2002-05-13 18:10:41 +00:00
|
|
|
|
2014-10-05 06:33:34 +00:00
|
|
|
if (
|
|
|
|
!RMAP_IS_SEGMENT(Address) &&
|
|
|
|
MmGetPfnForProcess(Process, Address) != Page)
|
|
|
|
{
|
|
|
|
DPRINT1("Insert rmap (%d, 0x%.8X) 0x%.8X which doesn't match physical "
|
|
|
|
"address 0x%.8X\n", Process ? Process->UniqueProcessId : 0,
|
|
|
|
Address,
|
|
|
|
MmGetPfnForProcess(Process, Address) << PAGE_SHIFT,
|
|
|
|
Page << PAGE_SHIFT);
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
new_entry->Next = current_entry;
|
2009-06-17 12:44:05 +00:00
|
|
|
#if DBG
|
2014-10-05 06:33:34 +00:00
|
|
|
while (current_entry)
|
|
|
|
{
|
|
|
|
if (current_entry->Address == new_entry->Address && current_entry->Process == new_entry->Process)
|
|
|
|
{
|
|
|
|
DbgPrint("MmInsertRmap tries to add a second rmap entry for address %p\n current caller ",
|
|
|
|
current_entry->Address);
|
|
|
|
DbgPrint("%p", new_entry->Caller);
|
|
|
|
DbgPrint("\n previous caller ");
|
|
|
|
DbgPrint("%p", current_entry->Caller);
|
|
|
|
DbgPrint("\n");
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
}
|
2005-08-16 20:36:03 +00:00
|
|
|
#endif
|
2014-10-05 06:33:34 +00:00
|
|
|
MmSetRmapListHeadPage(Page, new_entry);
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
if (!RMAP_IS_SEGMENT(Address))
|
|
|
|
{
|
|
|
|
if (Process == NULL)
|
|
|
|
{
|
|
|
|
Process = PsInitialSystemProcess;
|
|
|
|
}
|
|
|
|
if (Process)
|
|
|
|
{
|
|
|
|
PrevSize = InterlockedExchangeAddUL(&Process->Vm.WorkingSetSize, PAGE_SIZE);
|
|
|
|
if (PrevSize >= Process->Vm.PeakWorkingSetSize)
|
|
|
|
{
|
|
|
|
Process->Vm.PeakWorkingSetSize = PrevSize + PAGE_SIZE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2001-12-31 01:53:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
2005-09-14 01:05:50 +00:00
|
|
|
NTAPI
|
2010-07-15 22:50:12 +00:00
|
|
|
MmDeleteAllRmaps(PFN_NUMBER Page, PVOID Context,
|
2006-05-18 20:32:17 +00:00
|
|
|
VOID (*DeleteMapping)(PVOID Context, PEPROCESS Process,
|
2004-04-10 22:36:07 +00:00
|
|
|
PVOID Address))
|
2001-12-31 01:53:46 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY current_entry;
|
|
|
|
PMM_RMAP_ENTRY previous_entry;
|
|
|
|
PEPROCESS Process;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
if (current_entry == NULL)
|
|
|
|
{
|
|
|
|
DPRINT1("MmDeleteAllRmaps: No rmaps.\n");
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
|
|
|
}
|
|
|
|
MmSetRmapListHeadPage(Page, NULL);
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
|
|
|
|
while (current_entry != NULL)
|
|
|
|
{
|
|
|
|
previous_entry = current_entry;
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
if (!RMAP_IS_SEGMENT(previous_entry->Address))
|
|
|
|
{
|
|
|
|
if (DeleteMapping)
|
|
|
|
{
|
|
|
|
DeleteMapping(Context, previous_entry->Process,
|
|
|
|
previous_entry->Address);
|
|
|
|
}
|
|
|
|
Process = previous_entry->Process;
|
|
|
|
ExFreeToNPagedLookasideList(&RmapLookasideList, previous_entry);
|
|
|
|
if (Process == NULL)
|
|
|
|
{
|
|
|
|
Process = PsInitialSystemProcess;
|
|
|
|
}
|
|
|
|
if (Process)
|
|
|
|
{
|
|
|
|
(void)InterlockedExchangeAddUL(&Process->Vm.WorkingSetSize, -PAGE_SIZE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ExFreeToNPagedLookasideList(&RmapLookasideList, previous_entry);
|
|
|
|
}
|
|
|
|
}
|
2001-12-31 01:53:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
2005-09-14 01:05:50 +00:00
|
|
|
NTAPI
|
2010-07-15 22:50:12 +00:00
|
|
|
MmDeleteRmap(PFN_NUMBER Page, PEPROCESS Process,
|
2004-04-10 22:36:07 +00:00
|
|
|
PVOID Address)
|
2001-12-31 01:53:46 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY current_entry, previous_entry;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
previous_entry = NULL;
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
|
|
|
|
while (current_entry != NULL)
|
|
|
|
{
|
|
|
|
if (current_entry->Process == (PEPROCESS)Process &&
|
|
|
|
current_entry->Address == Address)
|
|
|
|
{
|
|
|
|
if (previous_entry == NULL)
|
2011-12-25 18:21:05 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
MmSetRmapListHeadPage(Page, current_entry->Next);
|
2011-12-25 18:21:05 +00:00
|
|
|
}
|
2014-10-05 06:33:34 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
previous_entry->Next = current_entry->Next;
|
|
|
|
}
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
ExFreeToNPagedLookasideList(&RmapLookasideList, current_entry);
|
|
|
|
if (!RMAP_IS_SEGMENT(Address))
|
2011-12-25 18:21:05 +00:00
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
if (Process == NULL)
|
|
|
|
{
|
|
|
|
Process = PsInitialSystemProcess;
|
|
|
|
}
|
|
|
|
if (Process)
|
|
|
|
{
|
|
|
|
(void)InterlockedExchangeAddUL(&Process->Vm.WorkingSetSize, -PAGE_SIZE);
|
|
|
|
}
|
2011-12-25 18:21:05 +00:00
|
|
|
}
|
2014-10-05 06:33:34 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
previous_entry = current_entry;
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
}
|
|
|
|
KeBugCheck(MEMORY_MANAGEMENT);
|
[CACHE]
The cache manager rewrite I started years ago has finally appeared in
ReactOS' trunk and although at this point it's not quite perfectly
integrated, it's enough to boot up the bootcd or livecd. To check out
the more mature original, check out arty-newcc-reactos, branch
arty-newcc on bitbucket.org . Amine Khaldi encouraged me quite a bit
to not give up on it, and was able to reach out and be an advocate
when i really wasn't able to. Others agree that the time has come to
begin removing the old cache manager. I expect the remaining problems
in the version going to trunk will be taken care of relatively
quickly.
The motivation for this effort lies in the particularly hairy
relationship between ReactOS' cache manager and data sections. This
code completely removes page sharing between cache manager and section
and reimagines cache manager as being a facility layered on the memory
manager, not really caring about individual pages, but simply managing
data section objects where caching might occur.
It took me about 2 years to do the first pass of this rewrite and most
of this year to fix some lingering issues, properly implement demand
paging in ReactOS (code which didn't come with this patch in a
recognizable form), and finish getting the PrivateCacheMap and
SharedCacheMap relationship correct.
Currently, the new ntoskrnl/cache directory contains an own
implementation of data file sections. After things have settled down,
we can begin to deprecate and remove the parts of ReactOS' section
implementation that depend on a close relationship with cache
manager. Eventually, I think that the extra code added to
ntoskrnl/cache/section will be removed and ReactOS' own sections will
replace the use of the special MM_CACHE_SECTION_SEGMENT in the cache
path.
Note also, that this makes all cache manager (and new section parts)
use wide file offsets. If my section code were to take over other
parts of the ReactOS memory manager, they would also benefit from
these improvements.
I invite anyone who wants to to peek at this code and fix whatever
bugs can be found.
svn path=/trunk/; revision=49423
2010-11-02 02:32:39 +00:00
|
|
|
}
|
|
|
|
|
2012-03-29 04:43:44 +00:00
|
|
|
/*
|
|
|
|
|
|
|
|
Return the process pointer given when a previous call to MmInsertRmap was
|
|
|
|
called with a process and address pointer that conform to the segment rmap
|
|
|
|
schema. In short, this requires the address part to be 0xffffff00 + n
|
|
|
|
where n is between 0 and 255. When such an rmap exists, it specifies a
|
|
|
|
segment rmap in which the process part is a pointer to a slice of a section
|
|
|
|
page table, and the low 8 bits of the address represent a page index in the
|
2012-05-04 11:32:07 +00:00
|
|
|
page table slice. Together, this information is used by
|
2012-03-29 04:43:44 +00:00
|
|
|
MmGetSectionAssociation to determine which page entry points to this page in
|
|
|
|
the segment page table.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
[CACHE]
The cache manager rewrite I started years ago has finally appeared in
ReactOS' trunk and although at this point it's not quite perfectly
integrated, it's enough to boot up the bootcd or livecd. To check out
the more mature original, check out arty-newcc-reactos, branch
arty-newcc on bitbucket.org . Amine Khaldi encouraged me quite a bit
to not give up on it, and was able to reach out and be an advocate
when i really wasn't able to. Others agree that the time has come to
begin removing the old cache manager. I expect the remaining problems
in the version going to trunk will be taken care of relatively
quickly.
The motivation for this effort lies in the particularly hairy
relationship between ReactOS' cache manager and data sections. This
code completely removes page sharing between cache manager and section
and reimagines cache manager as being a facility layered on the memory
manager, not really caring about individual pages, but simply managing
data section objects where caching might occur.
It took me about 2 years to do the first pass of this rewrite and most
of this year to fix some lingering issues, properly implement demand
paging in ReactOS (code which didn't come with this patch in a
recognizable form), and finish getting the PrivateCacheMap and
SharedCacheMap relationship correct.
Currently, the new ntoskrnl/cache directory contains an own
implementation of data file sections. After things have settled down,
we can begin to deprecate and remove the parts of ReactOS' section
implementation that depend on a close relationship with cache
manager. Eventually, I think that the extra code added to
ntoskrnl/cache/section will be removed and ReactOS' own sections will
replace the use of the special MM_CACHE_SECTION_SEGMENT in the cache
path.
Note also, that this makes all cache manager (and new section parts)
use wide file offsets. If my section code were to take over other
parts of the ReactOS memory manager, they would also benefit from
these improvements.
I invite anyone who wants to to peek at this code and fix whatever
bugs can be found.
svn path=/trunk/; revision=49423
2010-11-02 02:32:39 +00:00
|
|
|
PVOID
|
|
|
|
NTAPI
|
|
|
|
MmGetSegmentRmap(PFN_NUMBER Page, PULONG RawOffset)
|
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PCACHE_SECTION_PAGE_TABLE Result = NULL;
|
|
|
|
PMM_RMAP_ENTRY current_entry;//, previous_entry;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
//previous_entry = NULL;
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
while (current_entry != NULL)
|
|
|
|
{
|
|
|
|
if (RMAP_IS_SEGMENT(current_entry->Address))
|
|
|
|
{
|
|
|
|
Result = (PCACHE_SECTION_PAGE_TABLE)current_entry->Process;
|
|
|
|
*RawOffset = (ULONG_PTR)current_entry->Address & ~RMAP_SEGMENT_MASK;
|
|
|
|
InterlockedIncrementUL(&Result->Segment->ReferenceCount);
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
//previous_entry = current_entry;
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
}
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
return NULL;
|
[CACHE]
The cache manager rewrite I started years ago has finally appeared in
ReactOS' trunk and although at this point it's not quite perfectly
integrated, it's enough to boot up the bootcd or livecd. To check out
the more mature original, check out arty-newcc-reactos, branch
arty-newcc on bitbucket.org . Amine Khaldi encouraged me quite a bit
to not give up on it, and was able to reach out and be an advocate
when i really wasn't able to. Others agree that the time has come to
begin removing the old cache manager. I expect the remaining problems
in the version going to trunk will be taken care of relatively
quickly.
The motivation for this effort lies in the particularly hairy
relationship between ReactOS' cache manager and data sections. This
code completely removes page sharing between cache manager and section
and reimagines cache manager as being a facility layered on the memory
manager, not really caring about individual pages, but simply managing
data section objects where caching might occur.
It took me about 2 years to do the first pass of this rewrite and most
of this year to fix some lingering issues, properly implement demand
paging in ReactOS (code which didn't come with this patch in a
recognizable form), and finish getting the PrivateCacheMap and
SharedCacheMap relationship correct.
Currently, the new ntoskrnl/cache directory contains an own
implementation of data file sections. After things have settled down,
we can begin to deprecate and remove the parts of ReactOS' section
implementation that depend on a close relationship with cache
manager. Eventually, I think that the extra code added to
ntoskrnl/cache/section will be removed and ReactOS' own sections will
replace the use of the special MM_CACHE_SECTION_SEGMENT in the cache
path.
Note also, that this makes all cache manager (and new section parts)
use wide file offsets. If my section code were to take over other
parts of the ReactOS memory manager, they would also benefit from
these improvements.
I invite anyone who wants to to peek at this code and fix whatever
bugs can be found.
svn path=/trunk/; revision=49423
2010-11-02 02:32:39 +00:00
|
|
|
}
|
|
|
|
|
2012-03-29 04:43:44 +00:00
|
|
|
/*
|
|
|
|
|
|
|
|
Remove the section rmap associated with the indicated page, if it exists.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
[CACHE]
The cache manager rewrite I started years ago has finally appeared in
ReactOS' trunk and although at this point it's not quite perfectly
integrated, it's enough to boot up the bootcd or livecd. To check out
the more mature original, check out arty-newcc-reactos, branch
arty-newcc on bitbucket.org . Amine Khaldi encouraged me quite a bit
to not give up on it, and was able to reach out and be an advocate
when i really wasn't able to. Others agree that the time has come to
begin removing the old cache manager. I expect the remaining problems
in the version going to trunk will be taken care of relatively
quickly.
The motivation for this effort lies in the particularly hairy
relationship between ReactOS' cache manager and data sections. This
code completely removes page sharing between cache manager and section
and reimagines cache manager as being a facility layered on the memory
manager, not really caring about individual pages, but simply managing
data section objects where caching might occur.
It took me about 2 years to do the first pass of this rewrite and most
of this year to fix some lingering issues, properly implement demand
paging in ReactOS (code which didn't come with this patch in a
recognizable form), and finish getting the PrivateCacheMap and
SharedCacheMap relationship correct.
Currently, the new ntoskrnl/cache directory contains an own
implementation of data file sections. After things have settled down,
we can begin to deprecate and remove the parts of ReactOS' section
implementation that depend on a close relationship with cache
manager. Eventually, I think that the extra code added to
ntoskrnl/cache/section will be removed and ReactOS' own sections will
replace the use of the special MM_CACHE_SECTION_SEGMENT in the cache
path.
Note also, that this makes all cache manager (and new section parts)
use wide file offsets. If my section code were to take over other
parts of the ReactOS memory manager, they would also benefit from
these improvements.
I invite anyone who wants to to peek at this code and fix whatever
bugs can be found.
svn path=/trunk/; revision=49423
2010-11-02 02:32:39 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
MmDeleteSectionAssociation(PFN_NUMBER Page)
|
|
|
|
{
|
2014-10-05 06:33:34 +00:00
|
|
|
PMM_RMAP_ENTRY current_entry, previous_entry;
|
|
|
|
|
|
|
|
ExAcquireFastMutex(&RmapListLock);
|
|
|
|
previous_entry = NULL;
|
|
|
|
current_entry = MmGetRmapListHeadPage(Page);
|
|
|
|
while (current_entry != NULL)
|
|
|
|
{
|
|
|
|
if (RMAP_IS_SEGMENT(current_entry->Address))
|
|
|
|
{
|
|
|
|
if (previous_entry == NULL)
|
|
|
|
{
|
|
|
|
MmSetRmapListHeadPage(Page, current_entry->Next);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
previous_entry->Next = current_entry->Next;
|
|
|
|
}
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
|
|
|
ExFreeToNPagedLookasideList(&RmapLookasideList, current_entry);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
previous_entry = current_entry;
|
|
|
|
current_entry = current_entry->Next;
|
|
|
|
}
|
|
|
|
ExReleaseFastMutex(&RmapListLock);
|
2001-12-31 01:53:46 +00:00
|
|
|
}
|