persistentstorage/centralrepository/cenrepcli/clirep.cpp
author Shabe Razvi <shaber@symbian.org>
Tue, 19 Oct 2010 15:57:30 +0100
changeset 54 a0e1d366428c
parent 0 08ec8eefde2f
permissions -rw-r--r--
Workaround for Bug 3854 - featuremgr bld.inf no longer exports features.dat for emulator

// Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "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 "clirep.h"
#include <e32math.h>
#include "srvparams.h"
#include "srvreqs.h"

using namespace NCentralRepositoryConstants;

RRepositorySession* CClientRepository::Session()
	{
	return static_cast<RRepositorySession*>(Dll::Tls());
	}
	
CClientRepository* CClientRepository::NewLC(TUid aRepositoryUid)
	{
	CClientRepository* rep = new(ELeave) CClientRepository();
	CleanupStack::PushL(rep);
	rep->ConstructL(aRepositoryUid);
	return rep;
	}

void CClientRepository::ConstructL(TUid aRepositoryUid)
	{
	RRepositorySession* session = Session();
	
	if(session == NULL)
		{
		session = new (ELeave) RRepositorySession();
		CleanupStack::PushL(session);
		User::LeaveIfError(Dll::SetTls(session));
		CleanupStack::Pop(session);
		User::LeaveIfError(session->Connect());
		}
	else
		{
		session->IncrementSubSessionCounter();
		}

	iSubSession = new (ELeave) RRepositorySubSession();
	User::LeaveIfError(iSubSession->Open(session, EInitialise, TIpcArgs(aRepositoryUid.iUid)));
	}

CClientRepository::CClientRepository()
	{
	}

CClientRepository::~CClientRepository()
	{
	if(iSubSession)
		{
		iSubSession->Close();
		delete iSubSession;
		}
	RRepositorySession* session = Session();
	if(session && session->DecrementSubSessionCounter() == 0)
		{
		//The last subSesssion is closed. Time to close the session.
		session->Close();
		delete session;
		Dll::FreeTls();
		//SetSession(NULL);
		}
	}

TInt CClientRepository::Create(TUint32 aId, TInt aVal)
	{
	return iSubSession->SendReceive(ECreateInt, TIpcArgs(aId, aVal));
	}

TInt CClientRepository::Create(TUint32 aId, const TReal& aVal)
	{
	TPckg<TReal> p(aVal);
	return iSubSession->SendReceive(ECreateReal, TIpcArgs(aId, &p));
	}
	
TInt CClientRepository::Create(TUint32 aId, const TDesC8& aVal)
	{
	return iSubSession->SendReceive(ECreateString, TIpcArgs(aId, &aVal));
	}

TInt CClientRepository::Create(TUint32 aId, const TDesC16& aVal)
	{
	TPtrC8 ptr8((const TUint8*)aVal.Ptr(), aVal.Size());
	return iSubSession->SendReceive(ECreateString, TIpcArgs(aId, &ptr8));
	}

TInt CClientRepository::Delete(TUint32 aId)
	{
	return iSubSession->SendReceive(EDelete, TIpcArgs(aId));
	}
	
TInt CClientRepository::Delete(TUint32 aPartialKey, TUint32 aMask, TUint32 &aErrorKey)
	{
	aErrorKey = KUnspecifiedKey; // set in case not filled by server
	TPckg<TUint32> p(aErrorKey);
	return iSubSession->SendReceive(EDeleteRange, TIpcArgs(aPartialKey, aMask, &p));
	}

TInt CClientRepository::Get(TUint32 aId, TInt& aVal)
	{
	TPckg<TInt> p(aVal);
	return iSubSession->SendReceive(EGetInt, TIpcArgs(aId, &p));
	}

TInt CClientRepository::Set(TUint32 aId, TInt aVal)
	{
	return iSubSession->SendReceive(ESetInt, TIpcArgs(aId, aVal));
	}

TInt CClientRepository::Get(TUint32 aId, TReal& aVal)
	{
	TPckg<TReal> p(aVal);
	return iSubSession->SendReceive(EGetReal, TIpcArgs(aId, &p));
	}

TInt CClientRepository::Set(TUint32 aId, const TReal& aVal)
	{
	TPckg<TReal> p(aVal);
	return iSubSession->SendReceive(ESetReal, TIpcArgs(aId, &p));
	}

TInt CClientRepository::Get(TUint32 aId, TDes8& aVal)
	{
	TPckg<TInt> p(aVal.MaxLength());
	return iSubSession->SendReceive(EGetString, TIpcArgs(aId, &aVal, &p));
	}

TInt CClientRepository::Get(TUint32 aId, TDes8& aVal, TInt& aActualLen)
	{
	aActualLen = aVal.MaxLength();
	TPckg<TInt> p(aActualLen);
	return iSubSession->SendReceive(EGetString, TIpcArgs(aId, &aVal, &p));
	}

TInt CClientRepository::Set(TUint32 aId, const TDesC8& aVal)
	{
	return iSubSession->SendReceive(ESetString, TIpcArgs(aId, &aVal));
	}

TInt CClientRepository::Get(TUint32 aId, TDes& aVal)
	{
	TPtr8 ptr8((TUint8*)aVal.Ptr(), 0, aVal.MaxSize());
	
	TPckg<TInt> p(ptr8.MaxLength());
	
	TInt r = iSubSession->SendReceive(EGetString, TIpcArgs(aId, &ptr8, &p));

	if(r==KErrNone || r==KErrOverflow)
		{
		TInt len = ptr8.Length();
		// note the following handles the case where client is getting an odd-length 8-bit
		// descriptor into 16-bit aVal. Round up length and ensure the extra byte is zero.
		if(len&1)
			{
			ptr8.SetLength(len+1);   // set the length before trying to write the value			
			ptr8[len] = 0;
			}
		aVal.SetLength((len + 1)/2);
		}

	return r;
	}
	
TInt CClientRepository::Get(TUint32 aId, TDes& aVal, TInt& aActualLen)
	{
	TPtr8 ptr8((TUint8*)aVal.Ptr(), 0, aVal.MaxSize());

	aActualLen = ptr8.MaxLength();
	TPckg<TInt> p(aActualLen);

	TInt r = iSubSession->SendReceive(EGetString, TIpcArgs(aId, &ptr8, &p));

	if(r==KErrNone || r==KErrOverflow)
		{
		TInt len = ptr8.Length();
		// note the following handles the case where client is getting an odd-length 8-bit
		// descriptor into 16-bit aVal. Round up length and ensure the extra byte is zero.
		if(len&1)
			{
			ptr8.SetLength(len+1);   // set the length before trying to write the value			
			ptr8[len] = 0;
			}
		aVal.SetLength((len + 1)/2);
		aActualLen = ((aActualLen + 1)/2);
		}

	return r;
	}

TInt CClientRepository::Set(TUint32 aId, const TDesC& aVal)
	{
	TPtrC8 ptr8((const TUint8*)aVal.Ptr(), aVal.Size());
	return iSubSession->SendReceive(ESetString, TIpcArgs(aId, &ptr8));
	}

TInt CClientRepository::GetMeta(TUint32 aId, TUint32& aMeta)
	{
	TPckg<TUint32> p(aMeta);
	return iSubSession->SendReceive(EGetMeta, TIpcArgs(aId, &p));
	}

TInt CClientRepository::Move(TUint32 aSourcePartialId, TUint32 aTargetPartialId,
                             TUint32 aIdMask, TUint32 &aErrorId)
	{
	aErrorId = KUnspecifiedKey; // set in case not filled by server
	TPckg<TUint32> p(aErrorId);
	TKeyFilter srcKeyIdentifier = {aSourcePartialId, aIdMask};
	TKeyFilter tgtKeyIdentifier = {aTargetPartialId, aIdMask};
	TPckg<TKeyFilter> pSrc(srcKeyIdentifier);
	TPckg<TKeyFilter> pTrg(tgtKeyIdentifier);
	
	TInt r = iSubSession->SendReceive(EMove, TIpcArgs(&pSrc, &pTrg, &p));
	
	return r;
	}
	
//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindL(TUint32 aPartialId, TUint32 aIdMask,
	RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFind, TIpcArgs(&pIdentifier, 0, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindEqL(TUint32 aPartialId, TUint32 aIdMask, TInt aVal,
	RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindEqInt, TIpcArgs(&pIdentifier, aVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindEqL(TUint32 aPartialId, TUint32 aIdMask,
	const TReal& aVal, RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TPckg<TReal> pVal(aVal);
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindEqReal, TIpcArgs(&pIdentifier, &pVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindEqL(TUint32 aPartialId, TUint32 aIdMask,
	const TDesC8& aVal, RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindEqString, TIpcArgs(&pIdentifier, &aVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindEqL(TUint32 aPartialId, TUint32 aIdMask,
	const TDesC& aVal, RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TPtrC8 pVal((const TUint8*)aVal.Ptr(), aVal.Length()*2);
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindEqString, TIpcArgs(&pIdentifier, &pVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindNeqL(TUint32 aPartialId, TUint32 aIdMask,
	TInt aVal, RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindNeqInt, TIpcArgs(&pIdentifier, aVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindNeqL(TUint32 aPartialId, TUint32 aIdMask,
	const TReal& aVal, RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TPckg<TReal> pVal(aVal);
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindNeqReal, TIpcArgs(&pIdentifier, &pVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindNeqL(TUint32 aPartialId, TUint32 aIdMask,
	const TDesC8& aVal, RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindNeqString, TIpcArgs(&pIdentifier, &aVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

//Calls FailTransaction if it Leaves. This is the pattern for all client-side failure of
//operations valid in transactions.
TInt CClientRepository::FindNeqL(TUint32 aPartialId, TUint32 aIdMask,
	const TDesC& aVal, RArray<TUint32>& aFoundIds)
	{
    CleanupFailTransactionPushL();
	aFoundIds.Reset();	
	
	TPtrC8 pVal((const TUint8*)aVal.Ptr(), aVal.Length()*2);
	TFixedArray<TUint32, KCentRepFindWithLenghtBufSize> uids;
	TUint32* start = uids.Begin();
	TPtr8 ptr(reinterpret_cast<TUint8*>(start), uids.Count() * uids.Length());
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);

	TInt r = iSubSession->SendReceive(EFindNeqString, TIpcArgs(&pIdentifier, &pVal, &ptr));

	if(r == KErrNone)
		{
		r = GetFindResult(uids, aFoundIds);
		if (r==KErrNoMemory)
			User::LeaveNoMemory();
		}
		
    CleanupStack::Pop();
	
	return r;
	}

/** Private helper function for all the Find~L functions.
No need to call FailTransaction since all the methods that call this method calls
FailTransaction prior to this method.
@internalComponent
*/
TInt CClientRepository::GetFindResult(const TFixedArray<TUint32, KCentRepFindWithLenghtBufSize>& aUids, RArray<TUint32>& aFoundIds)
	{
	iClientErr = KErrNone;	
	const TUint32 numFound = aUids[0];
	const TUint32 numInitial = numFound > KCentRepFindBufSize ? KCentRepFindBufSize : numFound;
	const TUint32 numFinal = numFound > KCentRepFindBufSize ? numFound - KCentRepFindBufSize : 0;
	
	for(TUint32 i = 1; i <= numInitial; i++)
		{
		//initialise client error first
		iClientErr=aFoundIds.Append(aUids[i]);
		if (iClientErr!=KErrNone)
			return iClientErr;		
		}
	
	if(numFinal)
		{
		TAny* tempBuf = User::Alloc(numFinal * sizeof(TUint32));
		if (tempBuf==NULL)
			{
			return KErrNoMemory;
			}
		TPtr8 p(static_cast<TUint8*>(tempBuf), numFinal * sizeof(TUint32));
		TInt r = iSubSession->SendReceive(EGetFindResult, TIpcArgs(&p));
		if (r == KErrNone)
			{ 
			for(TUint32 i = 0; i < numFinal; i++)
				{
				iClientErr=aFoundIds.Append(static_cast<const TUint32*>(tempBuf)[i]);
				if (iClientErr!=KErrNone)
					{
					User::Free(tempBuf);
					return iClientErr;
					}
				}
			}
		User::Free(tempBuf);
		}		
	return iClientErr;
	}

TInt CClientRepository::NotifyRequest(TUint32 aId, TRequestStatus& aStatus)
	{
	TInt r = iSubSession->SendReceive(ENotifyRequestCheck, TIpcArgs(aId));
	if(r==KErrNone)
		iSubSession->SendReceive(ENotifyRequest, TIpcArgs(aId), aStatus);
	return r;
	}

TInt CClientRepository::NotifyCancel(TUint32 aId)
	{
	return iSubSession->SendReceive(ENotifyCancel, TIpcArgs(aId));
	}

TInt CClientRepository::NotifyCancelAll()
	{
	return iSubSession->SendReceive(ENotifyCancelAll);
	}

TInt CClientRepository::NotifyRequest(TUint32 aPartialId, TUint32 aIdMask,
	TRequestStatus& aStatus)
	{
	iSubSession->SendReceive(EGroupNotifyRequest,
		TIpcArgs(aPartialId, aIdMask), aStatus);
	return KErrNone;
	}

TInt CClientRepository::NotifyCancel(TUint32 aPartialId, TUint32 aIdMask)
	{
	TKeyFilter keyIdentifier = {aPartialId, aIdMask};
	TPckg<TKeyFilter> pIdentifier(keyIdentifier);
	
	return iSubSession->SendReceive(EGroupNotifyCancel, TIpcArgs(&pIdentifier));
	}

TInt CClientRepository::Reset()
	{
	return iSubSession->SendReceive(EResetAll);
	}

TInt CClientRepository::Reset(TUint32 aId)
	{
	return iSubSession->SendReceive(EReset, TIpcArgs(aId));
	}

TInt CClientRepository::StartTransaction(TTransactionMode aMode)
	{
	return iSubSession->SendReceive(ETransactionStart, TIpcArgs(aMode));
	}

void CClientRepository::StartTransaction(TTransactionMode aMode, TRequestStatus& aStatus)
	{
	iSubSession->SendReceive(ETransactionStart, TIpcArgs(aMode), aStatus);
	}

TInt CClientRepository::CommitTransaction(TUint32& aKeyInfo)
	{
	// set to KUnspecifiedKey in case failure happens before setting in server
	aKeyInfo = KUnspecifiedKey;
	TPckg<TUint32> p(aKeyInfo);
	return iSubSession->SendReceive(ETransactionCommit, TIpcArgs(&p));
	}

void CClientRepository::CommitTransaction(TDes8& aKeyInfo, TRequestStatus& aStatus)
	{
	// set to KUnspecifiedKey in case failure happens before setting in server
	aKeyInfo.Copy(TPckg<TUint32>(KUnspecifiedKey));
	iSubSession->SendReceive(ETransactionCommit, TIpcArgs(&aKeyInfo), aStatus);
	}
	
void CClientRepository::CancelTransaction()
	{
    iSubSession->SendReceive(ETransactionCancel);
	}

static void CancelTransactionCleanupOperation(TAny* aRepository)
    {
    static_cast<CClientRepository*>(aRepository)->CancelTransaction();
    }

// So CancelTransaction is called in case of Leave. Must pop with CleanupStack::Pop() or similar
void CClientRepository::CleanupCancelTransactionPushL()
    {
    CleanupStack::PushL(TCleanupItem(CancelTransactionCleanupOperation, this));
    }
    
void CClientRepository::FailTransaction()
	{
	if (iClientErr==KErrNone)
		iSubSession->SendReceive(ETransactionFail,TIpcArgs(KErrAbort));
	else
		iSubSession->SendReceive(ETransactionFail,TIpcArgs(iClientErr));
	//reset the internal client code
	iClientErr=KErrNone;
	}
	
// So FailTransaction is called in case of Leave. Must pop with CleanupStack::Pop() or similar
static void FailTransactionCleanupOperation(TAny* aRepository)
    {
    static_cast<CClientRepository*>(aRepository)->FailTransaction();
    }
	
void CClientRepository::CleanupFailTransactionPushL()
	{
	CleanupStack::PushL(TCleanupItem(FailTransactionCleanupOperation, this));
	}

TInt CClientRepository::TransactionState()
	{
	TInt iValue;
	
	TPckg<TInt> p(iValue);
	
	iSubSession->SendReceive(ETransactionState, TIpcArgs(&p));
	
	return iValue;
	}

TInt RRepositorySubSession::Open(RRepositorySession* aSession,TInt aFunction,const TIpcArgs& aArgs)
	{
	iSession = aSession;
	return(CreateSubSession(*aSession, aFunction, aArgs));
	}
	
void RRepositorySubSession::Close()
	{
	RSubSessionBase::CloseSubSession(EClose);
	}

TInt RRepositorySubSession::SendReceive(TInt aFunction) const
	{
	return RSubSessionBase::SendReceive(aFunction);
	}
	
TInt RRepositorySubSession::SendReceive(TInt aFunction, const TIpcArgs& aArgs) const
	{
	return RSubSessionBase::SendReceive(aFunction, aArgs);
	}
	
void RRepositorySubSession::SendReceive(TInt aFunction, const TIpcArgs& aArgs, TRequestStatus& aStatus) const
	{
	RSubSessionBase::SendReceive(aFunction, aArgs, aStatus);
	}

inline TInt RRepositorySession::IncrementSubSessionCounter()
	{
	return ++iSubSessionCounter;
	}
	
RRepositorySession ::RRepositorySession()
	:iSubSessionCounter(1)
	{
	}
	
inline TInt RRepositorySession::DecrementSubSessionCounter()
	{
	ASSERT(iSubSessionCounter > 0);
	return --iSubSessionCounter;
	}

#if defined(__CENTREP_SERVER_PERFTEST__) || defined(__CENTREP_SERVER_MEMTEST__) || defined(__CENTREP_SERVER_CACHETEST__)
TInt RRepositorySession::SendReceive(TInt aFunction) const
	{
	return RSessionBase::SendReceive(aFunction);
	}

TInt RRepositorySession::SendReceive(TInt aFunction,
	const TIpcArgs& aArgs) const
	{
	return RSessionBase::SendReceive(aFunction, aArgs);
	}

void RRepositorySession::SendReceive(TInt aFunction, const TIpcArgs& aArgs,
	TRequestStatus& aStatus) const
	{
	RSessionBase::SendReceive(aFunction, aArgs, aStatus);
	}
#endif

LOCAL_C TInt StartServer();

TInt RRepositorySession::Connect()
	{
   	const TVersion KVersion(KServerMajorVersion, KServerMinorVersion,
   		KServerBuildVersion);
   	TInt retry = 2;
   	TInt err = KErrGeneral;
   	// Use unlimited message slots as we can call subscribe multiple times per
   	// session.
   	TInt numMessageSlots = -1; 
   	for(;;)
   		{
   		// Try to create a new session with the server.
   		err = CreateSession(KServerName, KVersion, numMessageSlots);
   		if((err != KErrNotFound) && (err != KErrServerTerminated))
   			break; //completed
   		// Server not running, try to start it.
   		if(--retry==0)
			break; // Failed.
   		err = StartServer();
   		if((err != KErrNone) && (err != KErrAlreadyExists))
			break;	// Launched server
   		}
   	return err;
	}

//
// Start the server process or thread
//
LOCAL_C TInt StartServer()
	{
	const TUidType serverUid(KNullUid,KNullUid,KServerUid3);
	//
	// EPOC and EKA2 is easy, we just create a new server process. Simultaneous
	// launching of two such processes should be detected when the second one
	// attempts to create the server object, failing with KErrAlreadyExists.
	//
	RProcess server;
	TInt r=server.Create(KServerImg,KNullDesC,serverUid);

	if (r!=KErrNone)
		return r;
	TRequestStatus stat;
	server.Rendezvous(stat);
	if (stat!=KRequestPending)
		server.Kill(0);		// abort startup
	else
		server.Resume();	// logon OK - start the server
	User::WaitForRequest(stat);		// wait for start or death
	// we can't use the 'exit reason' if the server panicked as this
	// is the panic 'reason' and may be '0' which cannot be distinguished
	// from KErrNone
	r=(server.ExitType()==EExitPanic) ? KErrGeneral : stat.Int();
	server.Close();
	return r;
	}