kernel/eka/memmodel/epoc/flexible/mmu/mdefrag.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Thu, 15 Jul 2010 20:11:42 +0300
branchRCL_3
changeset 41 0ffb4e86fcc9
parent 28 5b5d147c7838
permissions -rw-r--r--
Revision: 201027 Kit: 2010127

// Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
//

#include <memmodel.h>
#include <ramalloc.h>
#include "mm.h"
#include "mmu.h"
#include "mmanager.h"
#include "mobject.h"
#include "mpager.h"
#include "mmapping.h"


TInt M::RamDefragFault(TAny* /*aExceptionInfo*/)
	{// Defag faults are handled by Mmu::HandlePageFault() on the flexible memory model.
	return KErrAbort;
	} 


EXPORT_C TInt Epoc::MovePhysicalPage(TPhysAddr aOld, TPhysAddr& aNew, TRamDefragPageToMove aPageToMove)
	{
	CHECK_PRECONDITIONS(MASK_THREAD_CRITICAL,"Epoc::MovePhysicalPage");
	__KTRACE_OPT(KMMU,Kern::Printf("Epoc::MovePhysicalPage() old=%08x pageToMove=%d",aOld,aPageToMove));
	// Mark aNew as invalid if a page is moved then this will be updated.
	// However, if the page couldn't be moved or is discarded then it must be invalid.
	aNew = KPhysAddrInvalid;

	TInt r = KErrNotFound;

	switch (aPageToMove)
		{
		case ERamDefragPage_Physical :
			break;

		case ERamDefragPage_PageTable :
			{
			// Assume aOld is a linear address in current thread and get the physical
			// address of the page table that maps it and move that.
			TLinAddr linAddr = (TLinAddr) aOld;
			DMemModelThread* thread = (DMemModelThread*)TheCurrentThread;
			// Get the os asid of current thread's process so no need to open reference on it.
			TUint osAsid = ((DMemModelProcess*)thread->iOwningProcess)->OsAsid();
			TUint offsetInMapping;
			TUint mapInstanceCount;
			DMemoryMapping* mapping = MM::FindMappingInAddressSpace(osAsid,linAddr,1,offsetInMapping,mapInstanceCount);
			if (!mapping)
				return r;
			MmuLock::Lock();
			TUint memoryIndex = (offsetInMapping >> KPageShift)+ mapping->iStartIndex;
			TPte* pte = mapping->FindPageTable(linAddr, memoryIndex);
			if (mapInstanceCount != mapping->MapInstanceCount() || !pte)
				{
				MmuLock::Unlock();
				mapping->Close();
				return r;
				}
			TPhysAddr physAddr = TheMmu.LinearToPhysical((TLinAddr)pte, KKernelOsAsid);
			__NK_ASSERT_DEBUG(physAddr != KPhysAddrInvalid);
			aOld = physAddr;	// Have physical address of page table page so move it.
			MmuLock::Unlock();
			mapping->Close();
			break;
			}

		case ERamDefragPage_PageTableInfo :
			{
			// Assume aOld is a linear address in current thread and get physical 
			// address of the page table info of the page table that maps it 
			// and move that.
			TLinAddr linAddr = (TLinAddr) aOld;
			DMemModelThread* thread = (DMemModelThread*)TheCurrentThread;
			// Get the os asid of current thread's process so no need to open reference on it.
			TUint osAsid = ((DMemModelProcess*)thread->iOwningProcess)->OsAsid();
			TUint offsetInMapping;
			TUint mapInstanceCount;
			DMemoryMapping* mapping = MM::FindMappingInAddressSpace(osAsid,linAddr,1,offsetInMapping,mapInstanceCount);
			if (!mapping)
				return r;
			MmuLock::Lock();
			TUint memoryIndex = (offsetInMapping >> KPageShift)+ mapping->iStartIndex;
			TPte* pte = mapping->FindPageTable(linAddr, memoryIndex);
			if (mapInstanceCount != mapping->MapInstanceCount() || !pte)
				{
				MmuLock::Unlock();
				mapping->Close();
				return r;
				}
			
			SPageTableInfo* pti = SPageTableInfo::FromPtPtr(pte);
			TPhysAddr physAddr = TheMmu.LinearToPhysical((TLinAddr)pti, KKernelOsAsid);
			__NK_ASSERT_DEBUG(physAddr != KPhysAddrInvalid);
			aOld = physAddr;	// Have physical address of page table info page so move it.
			MmuLock::Unlock();
			mapping->Close();
			break;
			}

		default :
			r = KErrNotSupported;
			return r;
		}

	RamAllocLock::Lock();

	// Move the page to any RAM zone.
	r = M::MovePage(aOld, aNew, KRamZoneInvalidId, 0);

	RamAllocLock::Unlock();
	return r;
	}


TInt M::MovePage(TPhysAddr aOld, TPhysAddr& aNew, TUint aBlockZoneId, TUint aMoveDisFlags)
	{
	// Returns this when page is not paged or managed or free but is a real RAM page.
	TInt r = KErrNotSupported;

	// get memory object corresponding to the page...
	DMemoryObject* memory = 0;
	MmuLock::Lock();
	SPageInfo* pi = SPageInfo::SafeFromPhysAddr(aOld & ~KPageMask);
	if (pi)
		{
		if (pi->PagedState() != SPageInfo::EUnpaged)
			{// The page is paged so let the pager handle it.
			return ThePager.DiscardPage(pi, aBlockZoneId, aMoveDisFlags);
			}
		switch (pi->Type())
			{
			case SPageInfo::EManaged:
				memory = pi->Owner();
				memory->Open();
				// move page, this will release the mmu lock.
				r = memory->iManager->MovePage(	memory, pi, aNew, aBlockZoneId, 
												(aMoveDisFlags & M::EMoveDisBlockRest)!=0);
				memory->AsyncClose();
				break;
			case SPageInfo::EUnused:
				r = KErrNotFound;	// This page is free so nothing to do.
				// Fall through..
			default:
				MmuLock::Unlock();
			}
		}
	else
		{// page info for aOld not found so aOld is not a RAM page...
		MmuLock::Unlock();
		r = KErrArgument;
		}
	return r;
	}


TInt M::MoveAndAllocPage(TPhysAddr aAddr, TZonePageType aPageType)
	{
	// Returns this when page is not paged or managed or free but is a real RAM page.
	TInt r = KErrNotSupported;

	// get memory object corresponding to the page...
	DMemoryObject* memory = 0;
	MmuLock::Lock();
	SPageInfo* pi = SPageInfo::SafeFromPhysAddr(aAddr & ~KPageMask);
	if(pi)
		{
		if (pi->PagedState() != SPageInfo::EUnpaged)
			{// The page is paged so let the pager handle it.
			return ThePager.DiscardAndAllocPage(pi, aPageType);
			}
		switch (pi->Type())
			{
			case SPageInfo::EManaged:
				memory = pi->Owner();
				memory->Open();
				// move page, this will release the mmu lock.
				r = memory->iManager->MoveAndAllocPage(memory, pi, aPageType);
				memory->AsyncClose();
				break;
			case SPageInfo::EUnused:
				r = KErrNone;	// This page is free so nothing to do.
				// Fall through..
			default:
				MmuLock::Unlock();
			}
		}
	else
		{// page info for aAddr not found so aAddr is not a RAM page...
		MmuLock::Unlock();
		r = KErrArgument;
		}
	return r;
	}