multimediacommsengine/mmcesrv/mmceserver/src/mcestateserverestablishing.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 02 Feb 2010 01:04:58 +0200
changeset 0 1bce908db942
permissions -rw-r--r--
Revision: 201003 Kit: 201005

/*
* Copyright (c) 2005 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 "mcestateserverestablishing.h"
#include "mceactionset.h"
#include "mcefcactionset.h"
#include "mcenatmacros.h"

// -----------------------------------------------------------------------------
// CMceStateServerEstablishing::CMceStateServerEstablishing
// -----------------------------------------------------------------------------
//
CMceStateServerEstablishing::CMceStateServerEstablishing ()
    : CMceState( KMceStateServerEstablishing )
	{
	}

// -----------------------------------------------------------------------------
// CMceStateServerEstablishing::~CMceStateServerEstablishing
// -----------------------------------------------------------------------------
//
CMceStateServerEstablishing::~CMceStateServerEstablishing()
	{
	}

// -----------------------------------------------------------------------------
// CMceStateServerEstablishing::DoAcceptL
// -----------------------------------------------------------------------------
//
TBool CMceStateServerEstablishing::DoAcceptL( TMceStateTransitionEvent& aEvent )
	{
    //not accepted	
	User::LeaveIfError( aEvent.EventSource() == EMceSip ? 
                        KErrTotalLossOfPrecision :
                        KErrNone );

    User::LeaveIfError(
        aEvent.Code() == EMceItcUpdate ||
        aEvent.Code() == EMceItcRejectSession ||      
        aEvent.Code() == EMceItcEnable ||
        aEvent.Code() == EMceItcDisable ||
        IsExtensionRequestEvent( aEvent )

        ? KErrNone : KErrTotalLossOfPrecision );
    
	return ETrue;
	}

// -----------------------------------------------------------------------------
// CMceStateServerEstablishing::EntryL
// -----------------------------------------------------------------------------
//
void CMceStateServerEstablishing::EntryL( TMceStateTransitionEvent& aEvent )
	{

	if ( IsExtensionRequestEvent( aEvent ))
		{
		HandleExtensionRequestEventL( aEvent );	
		}

	else
		{
		CMceSipSession& session = aEvent.Session();
		CSIPServerTransaction& request = session.Request();
	    CMceSipSession::TSubState subState = session.SubState();
		
		switch( aEvent.Code() )
	        {
	        case EMceItcUpdate:
	            {
	            EntryUpdateL( aEvent );
	            break;
	            }
	        case EMceItcRejectSession:
	            {
	            CDesC8Array* tempHeaders = session.Body()->iSIPHeaders;
	            CleanupStack::PushL( tempHeaders );
	            session.Body()->iSIPHeaders = 0;
	            session.Actions().SendDeclineL( request, aEvent.ParamClientMessage() );
	            
	            if ( !session.Body()->iSIPHeaders )
	                {
	                session.Body()->iSIPHeaders = tempHeaders;
	                CleanupStack::Pop( tempHeaders );
	                }
	            else
	                {
	                CleanupStack::PopAndDestroy( tempHeaders );
	                }
	            
	            if ( subState == CMceSipSession::EOffering )
	                {
	                session.Actions().StopMedia();
	                session.FCActions().ReleaseFC();
	                }
	            else if ( subState == CMceSipSession::EUpdating )
	                {
	                session.Actions().UpdateFailed();
	                }
	            break;
	            }
	        case EMceItcEnable:
	        case EMceItcDisable:
	            {
	            session.Actions().ControlMediaL( aEvent.ParamIDs(), 
	                                             (TMceItcFunctions)aEvent.Code() );
	            break;
	            } 
	        default:
	            {
	            //NOP
	            break;
	            }
	        } 
		}
	}

// -----------------------------------------------------------------------------
// CMceStateServerEstablishing::EntryUpdateL
// -----------------------------------------------------------------------------
//
void
CMceStateServerEstablishing::EntryUpdateL( TMceStateTransitionEvent& aEvent )
    {
	CMceSipSession& session = aEvent.Session();
	CSIPServerTransaction& request = session.Request();
    CMceSipSession::TSubState subState = session.SubState();
	TMceReturnStatus status = KErrNone;

    //update body
    session.Actions().UpdateBodyL( aEvent.ParamIDs(), 
                                   aEvent.ParamClientMessage() );
    //update MM                                  
    status = session.Actions().Update();
	
    if ( status == KMceReady )
        {
        // Send 183 with SDP if needed
        if ( MCE_NEED_TO_RECEIVE( session ) &&
             session.Extensions().Local( CMceSipExtensions::E100rel ) 
             != CMceSipExtensions::ENotSupported &&
             session.Extensions().Remote( CMceSipExtensions::E100rel ) 
             != CMceSipExtensions::ENotSupported )
            {
            // encode
            NAT_WAIT_NO_EXEC( session, session.Actions().EncodeL() )
            
            session.FCActions().UpdateFCAnswerL( *session.Offer() );
            session.Actions().SendResponseL( 
                            request,
                            KMceSipSessionProgress,
                            SipStrConsts::EPhraseSessionProgress );
           //For Long Sesion after this no need to updated key until session
           //established
           if ( session.ActiveBody().SecureSession() )
            	{
            	session.ActiveBody().SecureSession()->iKeyNeedUpdated = EFalse;
            	}                
            // start reserving
           session.Actions().ReserveL();
            // change state
           session.Actions().StateChanged( 
                KMceStateAcknowledgementRequired );
            }
        else
            {
            if ( subState != CMceSipSession::EAnswering )
                {
                // directly send 200 OK
                // encode body
                NAT_WAIT_NO_EXEC( session, session.Actions().EncodeL() )
                
                session.Actions().UpdateSucceed(); //no affect if answering
                session.FCActions().UpdateFCAnswerL( *session.Offer() );
                session.Actions().SendAnswerL( request );
                }
            }
        }
    else if ( MCE_IS_ERROR( status ) )
        {
        MceSip::DiscardRequest( request, KMceSipServerInternalError );
        
        if ( subState == CMceSipSession::EAnswering )
            {
			session.Actions().StopMedia();
            session.FCActions().ReleaseFC();
            //exit same as if client rejects
            aEvent.Code() = EMceItcRejectSession;
            }
        else if ( subState == CMceSipSession::EUpdating )
            {
            session.Actions().UpdateFailed();
            }
        else
            {
            //NOP
            }
        }

    aEvent.ParamStatus() = status;
    }

// -----------------------------------------------------------------------------
// CMceStateServerEstablishing::ExitL
// -----------------------------------------------------------------------------
//
void CMceStateServerEstablishing::ExitL( TMceStateTransitionEvent& aEvent )
	{
	
	if ( IsExtensionRequestEvent( aEvent ))
		{
		//Do Nothing;
		}
	
	else
		{
		CMceSipSession& session = aEvent.Session();
		TInt status = aEvent.ParamStatus();
	    CMceSipSession::TSubState subState = session.SubState();

	    switch ( aEvent.Code() )
	        {
	        case EMceItcUpdate:
	            {
	            if ( status < KErrNone ) //only in updating
	                {
	                //Client must use get session function get the original
	                session.Actions().StateChanged( KMceStateEstablished );
	                session.Actions().ClientStateChanged( aEvent.ParamIDs(), 
	                                                      CMceSession::EEstablished,
	                                                      status );
	                }
	            else if ( status == KMceAsync ) //both first and update
	                {
	                session.Actions().StateChanged( KMceStateUpdating );
	                session.Actions().ClientStateChanged( aEvent.ParamIDs(), 
	                                                      CMceSession::EReserving );
	                }
	            else //KReady
	                {
	                if ( subState == CMceSipSession::EAnswering )
	                    {
	                    // first INVITE, need to wait client to accept
	                                        									
						session.Actions().StateChanged( KMceStateUpdated );					
										
						session.Actions().ClientStateChanged( aEvent.ParamIDs(), 
																CMceSession::EReserving );
											
	                    session.Actions().ClientStateChangedL( CMceSession::EProceeding );
	                   	                 
	                    }
	                else
	                    {
                        if ( session.Request().Type() == SIPStrings::StringF( SipStrConsts::EUpdate ) )
                            {
                            // UPDATE, not refreshing
                            session.Actions().StateChanged( KMceStateEstablished );
                            session.Actions().ClientStateChanged( aEvent.ParamIDs(), 
                                                                  CMceSession::EEstablished,
                                                                  status );
                            session.Actions().ClientStateChangedL( CMceSession::EEstablished, session.Request() );
                            }
                        else
                            {
	                        // re-INVITE, answer sent
	                        session.Actions().StateChanged( KMceStateAnswering );
	                        session.Actions().ClientStateChanged( aEvent.ParamIDs(), 
	                                                          CMceSession::EAnswering );
                            }
	                    }
	                }
	            break;
	            }
	        case EMceItcRejectSession:
	            {
	            if ( subState == CMceSipSession::EAnswering )
	                { // original session rejected
	                session.Actions().StateChanged( KMceStateTerminated );
	                session.Actions().ClientStateChanged( aEvent.ParamIDs(), 
	                                                      CMceSession::ETerminated,
	                                                      status );
	                }
	            else 
	                { // update rejected
	                session.Actions().StateChanged( KMceStateEstablished );
	                session.Actions().ClientStateChanged( aEvent.ParamIDs(), 
	                                                      CMceSession::EEstablished );
	                }
	            break;
	            }
	        default:
	            {
	            //NOP
	            break;
	            }
	        }
		}
	}


// End of file