kernel/eka/euser/us_mes.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 0 a41df078684a
permissions -rw-r--r--
Revision: 201027 Kit: 2010127

// Copyright (c) 1995-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\euser\us_mes.cpp
// 
//

#include "us_std.h"




EXPORT_C void RServer2::Receive(RMessage2& aMessage)
//
// Receive a message for the server synchronously.
//
	{
	TRequestStatus s;
	Receive(aMessage,s);
	User::WaitForRequest(s);
	__ASSERT_ALWAYS(s==KErrNone,Panic(ETMesReceiveFailed));
	}



GLDEF_C void CompleteRequest(TRequestStatus &aStatus,TInt aCode)
	{
	TRequestStatus *pS=(&aStatus);
	User::RequestComplete(pS,aCode);
	}



EXPORT_C TInt RSessionBase::DoSendReceive(TInt aFunction,const TIpcArgs* aArgs) const
//
// Send a message and wait for the reply synchronously.
//
	{
	if (TUint(aFunction)<=TUint(KMaxTInt))
		return SendSync(aFunction,aArgs);

	Panic(ETMesBadFunctionNumber);
	return 0;
	}



EXPORT_C TInt RSessionBase::DoSend(TInt aFunction,const TIpcArgs* aArgs) const
//
// Send a blind message to the server.
//
	{
	if (TUint(aFunction)<=TUint(KMaxTInt))
		return SendAsync(aFunction,aArgs,NULL);

	Panic(ETMesBadFunctionNumber);
	return 0;
	}



EXPORT_C void RSessionBase::DoSendReceive(TInt aFunction,const TIpcArgs* aArgs,TRequestStatus &aStatus) const
//
// Send a message and wait for the reply asynchronously.
//
	{
	if (TUint(aFunction)<=TUint(KMaxTInt))
		{
		aStatus=KRequestPending;
		TInt r=SendAsync(aFunction,aArgs,&aStatus);
		if (r!=KErrNone)
			::CompleteRequest(aStatus,r);
		}
	else
		Panic(ETMesBadFunctionNumber);
	}



TInt RSubSessionBase::DoCreateSubSession(RSessionBase& aSession,TInt aFunction,const TIpcArgs* aArgs, TBool aAutoClose)
	{
	TIpcArgs a;
	if (aArgs!=NULL)
		{
		a.iArgs[0] = aArgs->iArgs[0];
		a.iArgs[1] = aArgs->iArgs[1];
		a.iArgs[2] = aArgs->iArgs[2];
		a.iFlags = aArgs->iFlags;
		}
	TPckgBuf<TInt> reply;
	a.Set(3,&reply);
	TInt r=aSession.SendReceive(aFunction,a);
	if (r==KErrNone)
		{
		iSubSessionHandle=reply();
		if (aAutoClose)
			{
			iSession=aSession;
			// set the caller's session handle to NULL to discourage the caller from closing it.
			aSession.SetHandle(KNullHandle);
			}
		else
			{
			// Set session handle with 'no close' set, to prevent CloseSubSession() 
			// from closing down the session
			iSession.SetHandleNC(aSession.Handle());
			}
		}
	else
		{
		iSubSessionHandle=0;
		iSession.SetHandle(KNullHandle);
		// Close the caller's session so it isn't left orphaned
		if (aAutoClose)
			aSession.Close();
		}
	return(r);
	}



EXPORT_C TInt RSubSessionBase::DoCreateSubSession(const RSessionBase& aSession,TInt aFunction,const TIpcArgs* aArgs)
	{
	// need to cast away const	
	return DoCreateSubSession( (RSessionBase&) aSession, aFunction, aArgs, EFalse);
	}



/**
Creates a new sub-session within an existing session. The new sub-session takes 
ownership of the session so that when the sub-session is closed, the session is 
closed too. If the creation of the sub-session fails, the session is closed immediately.
In other words, this method will always take ownership of the session, whether it succeeds
or not and the caller should never need to close it.

@param aSession The session to which this sub-session will belong.
@param aFunction The opcode specifying the requested service;
                the server should interpret this as a request to create
                a sub-session.
@param aArgs	The arguments to be sent to the server as part of the 
				sub-session create request. The fourth argument is not 
				sent to the server, instead it is replaced with a descriptor 
				reference to the 32bit value where the server should store 
				the handle of the created sub-session.

@return KErrNone if successful, otherwise one of the other system-wide error
        codes.

*/
EXPORT_C TInt RSubSessionBase::CreateAutoCloseSubSession(RSessionBase& aSession,TInt aFunction,const TIpcArgs& aArgs)
	{
	return DoCreateSubSession(aSession, aFunction, &aArgs, ETrue);
	}



/**
Returns a copy of the session associated with this sub-session.

@return a copy of the session
*/
EXPORT_C const RSessionBase RSubSessionBase::Session() const
	{
	RSessionBase session = iSession;

	// If this is a "normal" subsession, then the ENoClose flag will be set
	// to prevent the closing of the subsession from closing down the main session
	// so in this case we need to turn the ENoClose flag OFF to allow someone
	// to use the returned session to call RSessionBase::Close().
	// If this is a "autoclose" subsession, then the ENoClose flag will be clear
	// to allow the closing of the subsession to close down the main session
	// so in this case we need to turn the ENoClose flag ON to stop someone from
	// using the returned session to call RSessionBase::Close().
	session.iHandle ^= CObjectIx::ENoClose;


	return session;
	}



/**
Closes the sub-session.

@param aFunction The opcode specifying the requested service;
                 the server should interpret this as a request to close
                 the sub-session.
*/
EXPORT_C void RSubSessionBase::CloseSubSession(TInt aFunction)
	{
	if (iSubSessionHandle)
		{
		iSession.SendReceive(aFunction,TIpcArgs(TIpcArgs::ENothing,TIpcArgs::ENothing,TIpcArgs::ENothing,iSubSessionHandle));
		iSubSessionHandle=KNullHandle;

		// Close the session - will only work if CObjectIx::ENoClose is clear 
		// i.e. the sub-session was created using CreateAutoCloseSubSession()
		iSession.Close();
		}
	}



EXPORT_C TInt RSubSessionBase::DoSend(TInt aFunction,const TIpcArgs* aArgs) const
//
// Blind send. 
//
	{
	TIpcArgs a;
	if(aArgs)
		{
		a.iArgs[0] = aArgs->iArgs[0];
		a.iArgs[1] = aArgs->iArgs[1];
		a.iArgs[2] = aArgs->iArgs[2];
		a.iFlags = aArgs->iFlags&((1<<(3*TIpcArgs::KBitsPerType))-1);
		}
	a.iArgs[3] = iSubSessionHandle;
	return(iSession.Send(aFunction,a));
	}



EXPORT_C void RSubSessionBase::DoSendReceive(TInt aFunction,const TIpcArgs* aArgs,TRequestStatus &aStatus) const
//
// Send and wait for reply asynchronously.
//
	{
	TIpcArgs a;
	if(aArgs)
		{
		a.iArgs[0] = aArgs->iArgs[0];
		a.iArgs[1] = aArgs->iArgs[1];
		a.iArgs[2] = aArgs->iArgs[2];
		a.iFlags = aArgs->iFlags&((1<<(3*TIpcArgs::KBitsPerType))-1);
		}
	a.iArgs[3] = iSubSessionHandle;
	iSession.SendReceive(aFunction,a,aStatus);
	}



EXPORT_C TInt RSubSessionBase::DoSendReceive(TInt aFunction,const TIpcArgs* aArgs) const
//
// Send and wait for reply synchronously.
//
	{
	TIpcArgs a;
	if(aArgs)
		{
		a.iArgs[0] = aArgs->iArgs[0];
		a.iArgs[1] = aArgs->iArgs[1];
		a.iArgs[2] = aArgs->iArgs[2];
		a.iFlags = aArgs->iFlags&((1<<(3*TIpcArgs::KBitsPerType))-1);
		}
	a.iArgs[3] = iSubSessionHandle;
	return(iSession.SendReceive(aFunction,a));
	}