kernel/eka/memmodel/epoc/nvram.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 4 56f325a607ea
permissions -rw-r--r--
Revision: 201027 Kit: 2010127

// Copyright (c) 1997-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:
// e32\memmodel\epoc\nvram.cpp
// 
//

#include "plat_priv.h"

_LIT(KLitMachineConfigMutex,"MCConfMutex");
_LIT(KLitRamDriveMutex,"RamDriveMutex");
_LIT(KLitTheRamDriveChunk,"TheRamDriveChunk");

void K::InitNvRam()
	{
	__KTRACE_OPT(KBOOT,Kern::Printf("InitNvRam"));
	TInt r=K::MutexCreate(K::MachineConfigMutex, KLitMachineConfigMutex, NULL, EFalse, KMutexOrdMachineConfig);
	if (r!=KErrNone)
		K::Fault(K::EMachineConfigMutexCreateFailed);
	if (K::ColdStart)
		{
		TheSuperPage().iRamDriveSize=0;
		TheMachineConfig().iLogSize=0;
		TheMachineConfig().iLogMaxSize=0;
		}
#ifdef __MEMMODEL_FLEXIBLE__
	TheSuperPage().iRamDriveSize=0;
#endif
	SChunkCreateInfo c;
	TInt ramDriveSize=TheSuperPage().iRamDriveSize;
	c.iGlobal=EFalse;
	c.iAtt=TChunkCreate::ENormal;
	c.iForceFixed=EFalse;
#ifndef __MEMMODEL_FLEXIBLE__
	c.iOperations=SChunkCreateInfo::EAdjust|SChunkCreateInfo::EAdd;
#else
	c.iOperations=SChunkCreateInfo::EAdjust;
#endif
	c.iRunAddress=PP::RamDriveStartAddress;
	c.iPreallocated=ramDriveSize;
	c.iType=ERamDrive;
	c.iMaxSize=PP::RamDriveMaxSize;
	c.iInitialBottom=0;
	c.iInitialTop=0;
	c.iName.Set(KLitTheRamDriveChunk);
	c.iOwner=K::TheKernelProcess;
	TLinAddr runAddr;
	r=K::TheKernelProcess->NewChunk((DChunk*&)PP::TheRamDriveChunk,c,runAddr);
	if (r!=KErrNone)
		K::Fault(K::ERamDriveChunkCreateFailed);
	__KTRACE_OPT(KBOOT,Kern::Printf("Ram Drive size = %08x", ramDriveSize));
	r=TInternalRamDrive::Create();
	if (r!=KErrNone)
		K::Fault(K::ERamDriveInitFailed);

	__KTRACE_OPT(KBOOT,Kern::Printf("K::InitNvRam() completed"));
	}

TInt TInternalRamDrive::Create()
	{
	__KTRACE_OPT(KBOOT,Kern::Printf("TInternalRamDrive::Create()"));

	// create the RAM drive mutex
	TInt r=K::MutexCreate((DMutex*&)Mutex, KLitRamDriveMutex, NULL, EFalse, KMutexOrdRamDrive);
	if (r!=KErrNone)
		return r;
	__KTRACE_OPT(KBOOT,Kern::Printf("RAM drive mutex created at %08x",Mutex));
	return KErrNone;
	}

#ifndef __MEMMODEL_FLEXIBLE__
EXPORT_C TLinAddr TInternalRamDrive::Base()
//
// Return the Internal Ram Drive base address
//
	{
	return (TLinAddr)PP::TheRamDriveChunk->Base(&Kern::CurrentProcess());
	}
#endif

EXPORT_C TInt TInternalRamDrive::Size()
//
// Return the Internal Ram Drive size
//
	{
	return TheSuperPage().iRamDriveSize;
	}

EXPORT_C TInt TInternalRamDrive::Adjust(TInt aNewSize)
//
// Adjust the size of the internal ram drive
//
	{
	// If we are shrinking the drive, change the size now in case the
	// machine is reset half way through the chunk adjust
	if (aNewSize<0)
		return KErrArgument;
	if (aNewSize<TheSuperPage().iRamDriveSize)
		{
		TheSuperPage().iRamDriveSize=aNewSize;
		return PP::TheRamDriveChunk->Adjust(aNewSize);
		}

	// If we are growing the drive, change the size after the adjustment is complete
	// If a reset occurs in the middle of the adjust, the ram drive will be
	// restored to its original state before the adjustment.
	else if (aNewSize>TheSuperPage().iRamDriveSize)
		{
		if (aNewSize>PP::RamDriveMaxSize)
			return KErrDiskFull;
		TInt r=PP::TheRamDriveChunk->Adjust(aNewSize);
		if (r==KErrNoMemory)
			return(KErrDiskFull);
		else if(r==KErrNone)
			TheSuperPage().iRamDriveSize=aNewSize;
		return(r);
		}
	return KErrNone;
	}

EXPORT_C void TInternalRamDrive::Wait()
	{
	Kern::MutexWait(*Mutex);
	UNLOCK_USER_MEMORY();
	}

EXPORT_C void TInternalRamDrive::Signal()
	{
	LOCK_USER_MEMORY();
	Kern::MutexSignal(*Mutex);
	}