sipvoipprovider/src/svprtpobserverrequest.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Mon, 18 Jan 2010 20:12:36 +0200
changeset 0 a4daefaec16c
permissions -rw-r--r--
Revision: 201001 Kit: 201003

/*
* Copyright (c) 2008 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:  Provides RTP observation class
*
*/


#include <mcemediastream.h>
#include <mceaudiostream.h>
#include <mceaudiocodec.h>
#include "svprtpobserverrequest.h"
#include "svpsessionbase.h"
#include "svptimer.h"
#include "svpconsts.h"
#include "mccpcallobserver.h"
#include "svplogger.h"
#include "svpholdcontroller.h"

// LOCAL CONSTANTS
// RTCP timeout interval, 30 seconds.
const TInt KSVPRtcpTimeout = 30000;
// Drift 5 seconds, added to actual timeout allowing some randomness in
// actual timeout.
const TInt KSVPRtcpTimeoutDrift = 5000;

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::CSVPRtpObserverRequest
// ---------------------------------------------------------------------------
//
CSVPRtpObserverRequest::CSVPRtpObserverRequest(
    CSVPSessionBase& aSession  ) : iSession( aSession )
    {
    // No implementation required
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::~CSVPRtpObserverRequest
// ---------------------------------------------------------------------------
//
CSVPRtpObserverRequest::~CSVPRtpObserverRequest()
    {
    delete iTimer;
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::NewLC
// ---------------------------------------------------------------------------
//
CSVPRtpObserverRequest* CSVPRtpObserverRequest::NewLC(
    CSVPSessionBase& aSession )
    {
    CSVPRtpObserverRequest* self =
        new ( ELeave ) CSVPRtpObserverRequest( aSession );
    
    CleanupStack::PushL( self );
    self->ConstructL();
    return self;
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::NewL
// ---------------------------------------------------------------------------
//
CSVPRtpObserverRequest* CSVPRtpObserverRequest::NewL(
    CSVPSessionBase& aSession )
    {
    CSVPRtpObserverRequest* self = CSVPRtpObserverRequest::NewLC( aSession );
    CleanupStack::Pop( self );
    return self;
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::ConstructL
// ---------------------------------------------------------------------------
//
void CSVPRtpObserverRequest::ConstructL()
    {
    // NB, no need for a special timer ID.
    iTimer = CSVPTimer::NewL( *this, KErrNone );
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::SvpSession
// ---------------------------------------------------------------------------
//
const CSVPSessionBase* CSVPRtpObserverRequest::SvpSession() const
    {
    return &iSession;
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::MceSession
// ---------------------------------------------------------------------------
//
const CMceSession* CSVPRtpObserverRequest::MceSession() const
    {
    return &( iSession.Session() );
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::RtcpPacketReceived
// ---------------------------------------------------------------------------
//
void CSVPRtpObserverRequest::RtcpPacketReceived(
    TReceivedRtcpPacket /*aPacket*/,
    CMceSession& aSession, 
    CMceMediaStream& aStream )
    {
    SVPDEBUG2( "CSVPRtpObserverRequest::RtcpPacketReceived aStream.IsEnabled: %d",
        aStream.IsEnabled() )
    SVPDEBUG2( "CSVPRtpObserverRequest::RtcpPacketReceived aSession: 0x%x",
        &aSession )
    SVPDEBUG2( "CSVPRtpObserverRequest::RtcpPacketReceived iSession: 0x%x",
        &( iSession.Session() ) )
    
    // Check that the given session matches ours
    if ( &aSession == &( iSession.Session() ) )
        {
        SVPDEBUG1( "CSVPRtpObserverRequest::RtcpPacketReceived, (re)start timer" )
        
        // For the sake of "just to be sure", stop the timer first
        const TInt timeout( DetermineTimeoutValue( aStream ) );
        iTimer->Stop();
        iTimer->SetTime( timeout );
        iDisconnectSent = EFalse;
        }
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::Reset
// ---------------------------------------------------------------------------
//
void CSVPRtpObserverRequest::Reset()
    {
    SVPDEBUG1( "CSVPRtpObserverRequest::Reset" )
    
    iTimer->Stop();
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::TimedOut
// ---------------------------------------------------------------------------
//
void CSVPRtpObserverRequest::TimedOut( TInt /*aTimerId*/ )
    {
    // For the sake of "just to be sure", stop the timer first
    iTimer->Stop();
    
    if ( IsSessionInHoldState() )
        {
        SVPDEBUG1( "CSVPRtpObserverRequest::TimedOut, (re)start timer because Session is on HOLD" )
        iTimer->SetTime( KSVPRtcpTimeout );
        iDisconnectSent = EFalse;
        }
    else
        {
        SVPDEBUG2( "CSVPRtpObserverRequest::TimedOut iDisconnectSent: %d",
            iDisconnectSent )
        
        // First CCE must be notified about 'disconnecting', state. Then timer is
        // re-started so that CCE & UI can handle the events with time (i.e user
        // has actually time to see the UI notifications).
        // After timer has fired second time, then send 'idle' event to the
        // CCE.
        
        
        if ( !iDisconnectSent )
            {
            iSession.GetCCPSessionObserver().CallStateChanged(
                MCCPCallObserver::ECCPStateDisconnecting, &iSession ); 
            
            iTimer->SetTime( KSVPTerminatingTime );
            iDisconnectSent = ETrue;
            }
        else
            {
            iSession.GetCCPSessionObserver().CallStateChanged(
                MCCPCallObserver::ECCPStateIdle, &iSession ); 
            }
        }
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::DetermineTimeoutValue
// ---------------------------------------------------------------------------
//
TInt CSVPRtpObserverRequest::DetermineTimeoutValue(
    CMceMediaStream& aStream ) const
    {
    // NB: aStream is not const because call to Codecs() is not const => this
    // saves a const_cast and keeps some sanity in casting. Though, we do not
    // modify anything in given aStream...
    
    TInt timeout( KSVPRtcpTimeout );
    if ( KMceAudio == aStream.Type() )
        {
        CMceAudioStream& audio( static_cast<CMceAudioStream&>( aStream ) );
        
        const RPointerArray<CMceAudioCodec>& codecs( audio.Codecs() );
        const TInt count = codecs.Count();
        for( TInt k = 0; k < count; k++ )
            {
            // MCE API talks about ms' but returns somethings that look awfully
            // close to seconds... Oh, well...
            const TInt kpval(
                codecs[k]->KeepAliveTimer() * KSVPMilliSecondCoefficient );
            if ( kpval > timeout )
                {
                timeout = kpval;
                }
                
            SVPDEBUG2( "CSVPRtpObserverRequest::DetermineTimeoutValue k: %d", k )
            SVPDEBUG2( "CSVPRtpObserverRequest::DetermineTimeoutValue kpval: %d", kpval )
            }
            
        SVPDEBUG2( "CSVPRtpObserverRequest::DetermineTimeoutValue count: %d", count )
        }
    
    // Now add the drifting to the timeout.
    timeout += KSVPRtcpTimeoutDrift;
    
    SVPDEBUG2( "CSVPRtpObserverRequest::DetermineTimeoutValue timeout: %d", timeout )
    
    return timeout;
    }

// ---------------------------------------------------------------------------
// CSVPRtpObserverRequest::IsSessionInHoldState
// ---------------------------------------------------------------------------
//
TBool CSVPRtpObserverRequest::IsSessionInHoldState()
    {
    if ( SvpSession()->HasHoldController() &&
            ESVPOnHold == (SvpSession()->HoldController()).HoldState() )
        {
        SVPDEBUG1( "CSVPRtpObserverRequest::IsSessionInHoldState ETrue" )
        return ETrue;
        }
    else
        {
        SVPDEBUG1( "CSVPRtpObserverRequest::IsSessionInHoldState EFalse" )
        return EFalse;
        }
    }