smsprotocols/smsstack/smsprot/Src/smsprov.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Fri, 16 Apr 2010 16:12:37 +0300
changeset 20 244d7c5f118e
parent 0 3553901f7fa8
child 24 6638e7f4bd8f
child 42 3adadc800673
permissions -rw-r--r--
Revision: 201015 Kit: 201015

// Copyright (c) 2001-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:
// Implements the CSmsProvider service access point (SAP) class.
// Includes
// 
//

/**
 @file
*/

#include "smsprot.h"

#include <es_ver.h>
#include <es_mbuf.h>

#include "Gsmuelem.h"
#include "gsmubuf.h"
#include "Gsmumsg.h"

#include "smsustrm.h"
#include "smspmain.h"
#include "smspfacadestor.h"

// CSmsProvider policies
//
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlDeleteSmsMessagePolicy,ECapabilityWriteUserData);
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlEnumerateSmsMessagesPolicy,ECapabilityReadUserData );
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlReadMessageSucceededPolicy,ECapabilityReadUserData );
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlReadMessageFailedPolicy,ECapabilityReadUserData );
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlSendSmsMessagePolicy,ECapabilityNetworkServices );
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlWriteSmsMessagePolicy,ECapabilityWriteUserData );
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlReadSmsParamsPolicy,ECapability_None );
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlCompleteReadSmsParamsPolicy,ECapability_None );
static _LIT_SECURITY_POLICY_C1(smsProviderIoctlWriteSmsParamsPolicy,ECapabilityWriteDeviceData );

// following not implemented as too paranoid
//static _LIT_SECURITY_POLICY_C1(smsProviderIoctlSelectModemPresentPolicy,ECapabilityWriteDeviceData );
//static _LIT_SECURITY_POLICY_C1(smsProviderIoctlSelectModemNotPresentPolicy,ECapabilityWriteDeviceData );
static _LIT_SECURITY_POLICY_C1(smsProviderSetLocalNamePolicy,ECapabilityNetworkServices);
static _LIT_SECURITY_POLICY_C1(smsProviderWritePolicy,ECapability_None);

/**
 *  2 Phase constructor.
 *  
 *  @param aProtocol a reference to the SMS protocol object.
 *  @leave Leaves if ContructL() leaves, or not enough memory is available.
 *  @return a new CSmsProvider object.
 *  
 */
CSmsProvider* CSmsProvider::NewL(CSmsProtocol& aProtocol)
    {
    LOGSMSPROT1("CSmsProvider::NewL");
    
    CSmsProvider* self =new(ELeave) CSmsProvider(aProtocol);
    CleanupStack::PushL(self);
    self->ConstructL();
    CleanupStack::Pop(self);

    LOGSMSPROT1("-> CSmsProvider::NewL - done");
    
    return self;
    }

/**
 *  C'tor
 *  
 *  @param aProtocol a reference to the SMS protocol object.
 */
CSmsProvider::CSmsProvider(CSmsProtocol& aProtocol)
: iProtocol(aProtocol),iEnumSocket(EFalse),iNumOfEnumeratedMessages(0)
    {
    }

/**
 *  2nd Phase of construction.
 *  Subscribes to the protocol as an observer, and creates our send and
 *  receive buffers.
 *  
 */
void CSmsProvider::ConstructL()
    {
    LOGSMSPROT1("CSmsProvider::ConstructL");
    
    iProtocol.AddSmsMessageObserverL(*this);
    SetObserverAddedToProtocol(ETrue);
    iRecvBufSegArray=new(ELeave) CArrayPtrFlat<CBufSeg>(8);
    iSendBufSeg = CBufSeg::NewL(KSmsMaxSegmentLength);
    
    LOGSMSPROT1("-> CSmsProvider::ConstructL - done");
    }

/**
 *  D'tor
 *  
 *  Removes this SAP as an observer of the SMS protocol, and
 *  frees the send and receive buffers.
 *  
 */
CSmsProvider::~CSmsProvider()
    {
    if( ObserverAddedToProtocol() )
        {
        iProtocol.RemoveSmsMessageObserver(*this);
        }
    if( iRecvBufSegArray )
        {
        iRecvBufSegArray->ResetAndDestroy();
        delete iRecvBufSegArray;
        }
    delete iSendBufSeg;
    }

/**
 *  Does nothing.  Implementation of pure virtual CServProviderBase::Start().
 *  
 */
void CSmsProvider::Start()
    {
    LOGSMSPROT1("CSmsProvider::Start");
    }

/**
 *  Returns the local address of this SAP.  Implementation of
 *  pure virtual CServProviderBase::LocalName().
 *  
 */
void CSmsProvider::LocalName(TSockAddr& aAddr) const
    {
    LOGSMSPROT1("CSmsProvider::LocalName");
    aAddr = iLocalAddress;
    }

/**
 *  Sets the local address of this SAP by binding it to the protocol.
 *  The protocol ensures that there are no duplicate observers, and
 *  then calls back on the CSmsProvider::SetLocalAddress() method
 *  to set the address.
 *  
 *  Implementation of the pure virtual CServProviderBase::SetLocalName().
 *  
 *  @capability NetworkServices
 */
TInt CSmsProvider::SetLocalName(TSockAddr& aAddr)
    {
    LOGSMSPROT1("CSmsProvider::SetLocalName");
    
    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderSetLocalNamePolicy,"CSmsProvider SetLocal Name policy check") != KErrNone) )
        {
        return KErrPermissionDenied;
        }
    TSmsAddr& smsAddr=static_cast<TSmsAddr&>(aAddr);
    if( ( smsAddr.SmsAddrFamily() == ESmsAddrApplication8BitPort  || smsAddr.SmsAddrFamily() == ESmsAddrApplication16BitPort ) && smsAddr.Port() == 0 )
        {
        if( !iProtocol.AllocateLocalAddress(smsAddr) )
            {
            return KErrInUse;
            }
        }
    return iProtocol.BindSmsMessageObserver(*this,smsAddr);
    }

/**
 *  Called by the protocol to retrieve the remote name of the connection.
 *  This protocol is not connection oriented so this is not supported.
 *  
 *  Implementation of the pure virtual CServProviderBase::RemName().
 *  
 */
void CSmsProvider::RemName(TSockAddr& /*aAddr*/) const
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::RemName");
    BULLSEYE_RESTORE
    }

/**
 *  Called by the protocol to set the remote name of the connection.
 *  This protocol is not connection oriented so this is not supported.
 *  
 *  Implementation of the pure virtual CServProviderBase::SetRemName().
 *  
 */
TInt CSmsProvider::SetRemName(TSockAddr& /*aAddr*/)
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::SetRemName");
    return KErrNotSupported;
    BULLSEYE_RESTORE
    }

/**
 *  Returns the current value of an option setting for this SAP.
 *  No settings are currently defined.
 *  Implementation of pure virtual CServProviderBase::GetOption().
 *  
 */
TInt CSmsProvider::GetOption(TUint /*aLevel*/,TUint /*aName*/,TDes8& /*aOption*/) const
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::GetOption");
    return 0;
    BULLSEYE_RESTORE
    }

/**
 *  Called to perform specific IO control by the client.  All of the SMS protocol
 *  services are provided through this interface.
 *  
 *  The local address of this SAP must already be bound, and only one ioctl request
 *  may be outstanding at any one time.
 *  
 *  A resulting socket error of KErrEof can result from a KErrNoMemory during a
 *  preceding write to the socket.
 *  
 *  Implementation of pure virtual CServProviderBase::Ioctl().
 *  
 *  @param aLevel the IOCTL level.  Only KSolSmsProv is supported.
 *  @param aName the IOCTL name.
 *  @param aOption the IOCTL option.
 *  
 */
void CSmsProvider::Ioctl(TUint aLevel,TUint aName,TDes8* aOption)
    {
    LOGSMSPROT3("CSmsProvider::Ioctl [aLevel=%d, aName=%d]", aLevel, aName);
    LOGSMSPROT2("CSmsProvider::Ioctl [provider=0x%08x]",this);
    
    // Panic in debug mode if this call is invalid in this SAPs current state
    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrUnbound,SmspPanic(KSmspPanicWrongSmsAddressFamily));
    __ASSERT_DEBUG(!IoctlOutstanding(),SmspPanic(KSmspPanicIoctlAlreadyOutstanding));
    // Gracefully handle invalid calls in release build
    if( iLocalAddress.SmsAddrFamily()==ESmsAddrUnbound )
        {
        iSocket->Error(KErrNotReady,MSocketNotify::EErrorIoctl);
        return;
        }
    if( IoctlOutstanding() )
        {
        iSocket->Error(KErrInUse,MSocketNotify::EErrorIoctl);
        return;
        }
    // General state is OK, now try to service the request
    iName=aName;
    switch( aLevel )
        {
        case KSolSmsProv:
            {
            switch( iName )
                {
                case KIoctlSupportOODClass0SmsMessages:
                    {
                    if( iProtocol.iReassemblyStore )
                        {
                        if( iProtocol.iReassemblyStore->IsSeparateClass0StoreSupported() )
                            {
                            iSocket->IoctlComplete(NULL);
                            }
                        else
                            {
                            iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                            }
                        }
                    else
                        {
                        iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                        }
                    } break;
                case KIoctlSendSmsMessage:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlSendSmsMessagePolicy,"CSmsProvider Ioctl SendSmsMessage policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrLocalOperation,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    __ASSERT_DEBUG(aOption!=NULL,SmspPanic(KSmspPanicOptionBufferNull));
                    // Handle bad requests gracefully
                    if( iLocalAddress.SmsAddrFamily()==ESmsAddrLocalOperation )
                        {
                        iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    if( aOption == NULL )
                        {
                        iSocket->Error(KErrArgument,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    // Read message from socket
                    CSmsMessage* smsmessage=NULL;
                    TRAPD(ret,(smsmessage=InternalizeMessageL()));
                    if( ret!=KErrNone )
                        {
                        iSendBufSeg->Reset();
                        iSocket->Error(ret,MSocketNotify::EErrorIoctl);
                        }
                    else
                        {
                            // Pass the message to the protocol for sending
                        TPckgBuf<TUint> buf;
                        buf.Copy(*aOption);
                        SetIoctlOutstanding(ETrue);
                        iProtocol.SendSmsMessage(smsmessage,*this, buf());
						}
                    } break;
                case KIoctlEnumerateSmsMessages:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlEnumerateSmsMessagesPolicy, "CSmsProvider Ioctl EnumerateSmsMessages policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    // Handle bad requests gracefully
                    if( iLocalAddress.SmsAddrFamily()==ESmsAddrSendOnly )
                        {
                        iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    
                    SetIoctlOutstanding(ETrue);
                    iProtocol.EnumeratePhone(*this);
                    } break;
                case KIoctlWriteSmsMessage:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlWriteSmsMessagePolicy,"CSmsProvider IoctlWriteSmsMessage policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    // Read message from the socket
                    CSmsMessage*smsmessage=NULL;
                    TRAPD(ret,(smsmessage=InternalizeMessageL()));
                    if( ret!=KErrNone )
                        {
                        iSendBufSeg->Reset();
                        iSocket->Error(ret,MSocketNotify::EErrorIoctl);
                        }
                    else
                        {
                        // Pass message to the protocol for writing
                        SetIoctlOutstanding(ETrue);
                        iProtocol.WriteSmsMessage(smsmessage,*this);
                        }
                    } break;
                case KIoctlDeleteSmsMessage:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlDeleteSmsMessagePolicy,"CSmsProvider Ioctl DeleteSmsMessage policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    // Handle bad requests gracefully
                    if( iLocalAddress.SmsAddrFamily()==ESmsAddrSendOnly )
                        {
                        iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    // Read message from the socket
                    CSmsMessage*smsmessage=NULL;
                    TRAPD(ret,(smsmessage=InternalizeMessageL()));
                    if( ret!=KErrNone )
                        {
                        LOGSMSPROT2("-> CSmsProvider::Ioctl - CSmsProvider::InternalizeMessageL [ret=%d]", ret);
                        iSendBufSeg->Reset();
                        iSocket->Error(ret, MSocketNotify::EErrorIoctl);
                        }
                    else
                        {
                        // Pass request to protocol
                        SetIoctlOutstanding(ETrue);
                        iProtocol.DeleteSmsMessage(smsmessage,*this);
                        }
                    } break;
                case KIoctlReadMessageSucceeded:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlReadMessageSucceededPolicy,"CSmsProvider Ioctl ReadMessageSucceeded policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    __ASSERT_DEBUG(NumSegments(iSegmentIndex==NumSegments(iRecvBufSegArray->At(0)->Size())),SmspPanic(KSmspPanicBadClientIoctlCall));
                    // Delete entry from reassemblystore
                    CSmsMessage*smsmessage=NULL;
                    TRAPD(ret,(smsmessage=InternalizeMessageL(iRecvBufSegArray->At(0))));
                    if( ret==KErrNone )
                        {
                        TRAP(ret,(iProtocol.DeleteSMSFromReaStoreL( *smsmessage )));
                        LOGSMSPROT2("-> CSmsProvider::Ioctl - CSmsProvider::DeleteSMSFromReaStoreL [ret=%d]", ret);
                        }
                    else
                        {
                        LOGSMSPROT2("-> CSmsProvider::Ioctl - CSmsProvider::InternalizeMessageL [ret=%d]", ret);
                        }
                    delete smsmessage;
                    // Looking for more sms left in the store
                    // This is now done after finishing the readprocess
                    iProtocol.MessageReadedSuccessfully();
                    if( iEnumSocket )
                        {
                        --iNumOfEnumeratedMessages;
                        LOGSMSPROT2("-> CSmsProvider::Ioctl - [iNumOfEnumeratedMessages=%d]", iNumOfEnumeratedMessages);
                        if( iNumOfEnumeratedMessages <= 0 )
                            {
                            iProtocol.iPhoneEnumerationObserver=NULL;
                            iEnumSocket=EFalse;
                            iProtocol.MessageReadedSuccessfully();
                            LOGSMSPROT1("-> CSmsProvider::Ioctl - [iNumOfEnumeratedMessages=NULL]");
                            }
                        }
                    // Remove the message from the receive buffer & complete
                    delete iRecvBufSegArray->At(0);
                    iRecvBufSegArray->At(0) = NULL;
                    iRecvBufSegArray->Delete(0);
    
                    iSegmentIndex=0;
                    iSocket->IoctlComplete(NULL);
                    } break;
                case KIoctlReadMessageFailed:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlReadMessageFailedPolicy,"CSmsProvider Ioctl ReadMessageFailed policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    __ASSERT_DEBUG(NumSegments(iSegmentIndex<=NumSegments(iRecvBufSegArray->At(0)->Size())),SmspPanic(KSmspPanicBadClientIoctlCall));
                    // Handle bad requests gracefully
                    if( iLocalAddress.SmsAddrFamily()==ESmsAddrSendOnly )
                        {
                        if( iEnumSocket )
                            {
                            --iNumOfEnumeratedMessages;
                            if( iNumOfEnumeratedMessages <= 0 )
                                {
                                iProtocol.iPhoneEnumerationObserver=NULL;
                                LOGSMSPROT1("-> CSmsProvider::Ioctl - fail [iNumOfEnumeratedMessages=NULL]");
                                iEnumSocket=EFalse;
                                iProtocol.MessageReadedSuccessfully();
                                }
                            }
                        iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    // Re-notify the socket that data is available, reset the segment
                    // index of the current message back to the start & complete
                    iSocket->NewData(iSegmentIndex);
                    iSegmentIndex=0;
                    iSocket->IoctlComplete(NULL);
                    } break;
                case KIoctlReadSmsParams:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlReadSmsParamsPolicy,"CSmsProvider Ioctl ReadSmsParams policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    // Handle bad requests gracefully
                    if( iLocalAddress.SmsAddrFamily()!=ESmsAddrLocalOperation )
                        {
                        iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                        return;
                        }
                   
                   // Pass request on to the protocol
                   SetIoctlOutstanding(ETrue);
                   iProtocol.ReadSmsParameters(*this);
                    } break;
                case KIoctlCompleteReadSmsParams:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlCompleteReadSmsParamsPolicy,"CSmsProvider Ioctl CompleteReadSmsParams policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    __ASSERT_DEBUG(NumSegments(iSegmentIndex==NumSegments(iRecvBufSegArray->At(0)->Size())),SmspPanic(KSmspPanicBadClientIoctlCall));
                    // Handle bad requests gracefully
                    if ( iLocalAddress.SmsAddrFamily()==ESmsAddrSendOnly )
                        {
                        iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    // Remove the parameter list from the receive buffer & complete
                    delete iRecvBufSegArray->At(0);
                    iRecvBufSegArray->At(0) = NULL;
                    iRecvBufSegArray->Delete(0);
                    iSegmentIndex=0;
                    iSocket->IoctlComplete(NULL);
                    } break; 
                case KIoctlWriteSmsParams:
                    {
                    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderIoctlWriteSmsParamsPolicy,"CSmsProvider Ioctl WriteSmsParams policy check") != KErrNone) )
                        {
                        iSocket->Error(KErrPermissionDenied,MSocketNotify::EErrorIoctl);
                        return;
                        }
                    // Read parameters from the socket
                    CMobilePhoneSmspList*mobilePhoneSmspList=NULL;
                    TRAPD(ret,(mobilePhoneSmspList=InternalizeParametersL()));
                    if( ret!=KErrNone )
                        {
                        iSendBufSeg->Reset();
                        iSocket->Error(ret,MSocketNotify::EErrorIoctl);
                        }
                    else
                        {
                        // Pass parameters to the protocol for writing.
                        // CSmsWriteParams takes ownership of mobilePhoneSmspList.
                        SetIoctlOutstanding(ETrue);
                        iProtocol.WriteSmsParameters(mobilePhoneSmspList, *this);
                        }
                    } break;
                default:
                    {
                    // Panic in debug build
                    __ASSERT_DEBUG(EFalse,SmspPanic(KSmspUndefinedName));
                    // Error gracefully in release build
                    iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
                    } break;
                }
            } break;
        default:
            {
            // Unsupported ioctl level, panic in debug build
            __ASSERT_DEBUG(EFalse,SmspPanic(KSmspUndefinedLevel));
            // Gracefully error in release build
            iSocket->Error(KErrNotSupported,MSocketNotify::EErrorIoctl);
            } break;
        }
    }

/**
 *  Cancels an outstanding ioctl.
 *  Since there can only be a single ioctl request outstanding, the parameters
 *  must match those of the original request.
 *  Implementation of the pure virtual CServProviderBase::CancelIoctl().
 *  
 *  @param aLevel the level of the ioctl request to cancel.
 *  @param aName the name of the ioctl request to cancel.
 *  
 */
void CSmsProvider::CancelIoctl(TUint aLevel, TUint aName)
    {
    LOGSMSPROT3("CSmsProvider::CancelIoctl [aLevel=%d, aName=%d]", aLevel, aName);
    
    // Panic in debug mode if this call is invalid in this SAPs current state
    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrUnbound,SmspPanic(KSmspPanicWrongSmsAddressFamily));
    __ASSERT_DEBUG(iName==aName,SmspPanic(ESmspBadIoctlName));
    if( iName != aName )
        {
        return;
        }
    switch( aLevel )
        {
        case KSolSmsProv:
            {
            // Request cancel via protocol
            switch( iName )
                {
                case KIoctlSendSmsMessage:
                    {
                    iProtocol.CancelSendSmsMessage(*this);
                    } break;
                case KIoctlEnumerateSmsMessages:
                    {
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    iProtocol.CancelEnumeratePhone(*this);
                    iEnumSocket=EFalse;
                    iProtocol.iPhoneEnumerationObserver=NULL;
                    LOGSMSPROT1("-> CSmsProvider::CancelIoctl - [iNumOfEnumeratedMessages=NULL]");
                    } break;
                case KIoctlWriteSmsMessage:
                    {
                    iProtocol.CancelWriteSmsMessage(*this);
                    } break;
                case KIoctlDeleteSmsMessage:
                    {
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    iProtocol.CancelDeleteSmsMessage(*this);
                    } break;
                case KIoctlReadMessageSucceeded:
                case KIoctlReadMessageFailed:
                    {
                    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly,SmspPanic(KSmspPanicWrongSmsAddressFamily));
                    } break;
                case KIoctlReadSmsParams:
                    {
                    iProtocol.CancelReadSmsParams();
                    } break;
                case KIoctlWriteSmsParams:
                    {
                    iProtocol.CancelWriteSmsParams();
                    } break;
                default:
                    {
                    __ASSERT_DEBUG(EFalse,SmspPanic(KSmspUndefinedName));
                    } break;
                }
            } break;
        default:
            {
            __ASSERT_DEBUG(EFalse,SmspPanic(KSmspUndefinedLevel));
            } break;
        }
    }

/**
 *  Sets an option on the SAP.  No options are currently implemented.
 *  
 *  Implements the pure virtual CServProviderBase::SetOption().
 *  
 */
TInt CSmsProvider::SetOption(TUint /*aLevel*/,TUint /*aName*/,const TDesC8& /*aOption*/)
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::SetOption()");
    return 0;
    BULLSEYE_RESTORE
    }

/**
 *  Called by the socket server to write data into the send queue.
 *  The data is a single segment of a serialized SMS message.
 *  
 *  A KErrNoMemory condition arising in this write will not be reported directly (due to
 *  the ESOCK design), instead it will be reflected in a KErrEof from the subsequent Ioctl()
 *  call.
 *  
 *  Implementation of the pure virtual CServProviderBase::Write().
 *  
 *  @param aBufChain the data to write into the send buffer.
 *  @param aOptions not used.
 *  @param aAddr not used.
 *  @return 1 (datagram written) on success, otherwise an error.
 *  
 *  @capability None
 */
TInt CSmsProvider::Write(RMBufChain& aBufChain, TUint /*aOptions*/, TSockAddr* /*aAddr*/)
    {
    LOGSMSPROT1("CSmsProvider::Write");
    
    if( !iSecurityChecker || (iSecurityChecker->CheckPolicy(smsProviderWritePolicy,"CSmsProvider Write policy check") != KErrNone) )
        {
        return KErrPermissionDenied;
        }
    __ASSERT_DEBUG(iLocalAddress.SmsAddrFamily()!=ESmsAddrUnbound,SmspPanic(KSmspPanicWrongSmsAddressFamily));
    
    /// @note: LOGIFH2A2 macro for logging esock write
    LOGSMSPROT2("-> CSmsProvider::Write [%d bytes]", aBufChain.Length());
#ifdef SMSLOGGERIF
	HBufC8* debugBuf = HBufC8::New(aBufChain.Length());
	if(debugBuf)
		{
		TPtr8 debugData = debugBuf->Des();
		aBufChain.CopyOut(debugData);
		LOGIF2(_L8("ESOCK WRITE: %S"),&debugData);
		LOGIFH2A2(_L8("ESOCK WRITE: "),debugData);
		delete debugBuf;
		}
#endif
    
    TUint bytesCopied = 0;
    // Append all of the mbufs to the send buffer
    TInt ret = KErrNone;
    TMBufIter iter(aBufChain);
    while( iter.More() )
        {
        RMBuf* p = iter++;
        TRAP(ret, iSendBufSeg->InsertL(iSendBufSeg->Size(), p->Ptr(), p->Length()));
        if(ret != KErrNone)
            break;
        bytesCopied += p->Length();
        }
    if( ret != KErrNone )
        {
        iSendBufSeg->Reset();	// it has always done this, but won't innocent data get zapped?
        }
    else
        {
        aBufChain.Free();	   // we accepted it all; flag these by consuming all buffers
        }
    return (ret == KErrNone)? 1: ret;
    }

/**
 *  Called by the socket server to retrieve data that this SAP has indicated
 *  is waiting in its buffers.
 *  
 *  Implentation of the pure virtual CServProviderBase::GetData().
 *  
 *  Once the provider has indicated new data is available using
 *  MSocketNotify::NewData(), the socket server will call on this method
 *  to retrieve the serialized SMS message in segments.
 *  Once the sockets client has streamed out the entire message, it
 *  will call on ioctl with KIoctlReadMessageSucceeded which resets the internal
 *  counters.
 *  
 *  @param aBufChain the buffer to insert the data.
 *  @param aLength not used.
 *  @param aOptions not used.
 *  @param aAddr not used.
 *  @return 1 (datagram read) on success, otherwise an error.
 *  
 */
TInt CSmsProvider::GetData(RMBufChain& aBufChain, TUint /*aLength*/, TUint /*aOptions*/, TSockAddr* /*aAddr*/)
    {
    __ASSERT_DEBUG((iLocalAddress.SmsAddrFamily()!=ESmsAddrUnbound) && (iLocalAddress.SmsAddrFamily()!=ESmsAddrSendOnly),SmspPanic(KSmspPanicWrongSmsAddressFamily));

    LOGSMSPROT2("CSmsProvider::GetData [provider=0x%08x]", this);
    
    // Get the segmented buffer of first message
    CBufSeg* recvbufseg=iRecvBufSegArray->At(0);
    TInt size=recvbufseg->Size();
    __ASSERT_DEBUG(iSegmentIndex<NumSegments(size),SmspPanic(KSmspPanicBadClientMessageRead));

    // Caculate the position of the next segment of the serialized message,
    // insert into the buffer parameter and update our segment counter
    TInt pos=iSegmentIndex*KSmsMaxSegmentLength;
    TInt length = pos+KSmsMaxSegmentLength>size? size-pos: KSmsMaxSegmentLength;
    
    TRAPD(err, aBufChain.AllocL(length));
    if( err == KErrNone )
        {
        // For want of a segmented buffer copy for Mbufs we have a little loop. Because we're
        // reading consecutive data out of the CBufSeg there shouldn't be a bad performance hit
        // (see CBufSeg::Ptr() doco; CBufBase::Read() uses this)
        TInt segPos = 0;
        TMBufIter iter(aBufChain);
        while( segPos < length )
            {
            RMBuf* p = iter++;
            TInt readSize = Min(p->Size(), length - segPos);
            recvbufseg->Read(pos + segPos, p->Buffer(), readSize);
            segPos += readSize;
            }
        }
    else if( err == KErrNoMBufs )
        {
        return KErrNoMBufs;	// ask ESock to call us back when some buffers are free
        }
    else
        {
        iSocket->Error(err, MSocketNotify::EErrorRecv);
        }
    ++iSegmentIndex;
    return 1;	// datagrams are counted as atoms not bytes
    }

/**
 *  Called by the socket server to indicate the provider should connect to
 *  a peer.  Not a connection oriented protocol so this is not implemented.
 *  
 *  Implementation of the pure virtual CServProviderBase::ActiveOpen().
 *  
 */
void CSmsProvider::ActiveOpen()
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::ActiveOpen [does nothing]");
    BULLSEYE_RESTORE
    }

/**
 *  Called by the socket server to indicate the provider should connect to
 *  a peer.  Not a connection oriented protocol so this is not implemented.
 *  
 *  Implementation of the pure virtual CServProviderBase::ActiveOpen().
 *  
 */
void CSmsProvider::ActiveOpen(const TDesC8& /*aConnectionData*/)
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::ActiveOpen [does nothing]");
    BULLSEYE_RESTORE
    }

/**
 *  Called by the socket server to indicate the provider should wait for an
 *  incoming client connection request.  Not a connection oriented protocol
 *  so this is not implemented.
 *  
 *  Implementation of the pure virtual CServiceProviderBase::PassiveOpen().
 *  
 */
TInt CSmsProvider::PassiveOpen(TUint /*aQueSize*/)
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::PassiveOpen [not supported]");
    return KErrNotSupported;
    BULLSEYE_RESTORE
    }

/**
 *  Called by the socket server to indicate the provider should wait for an
 *  incoming client connection request.  Not a connection oriented protocol
 *  so this is not implemented.
 *  
 *  Implementation of the pure virtual CServiceProviderBase::PassiveOpen().
 *  
 */
TInt CSmsProvider::PassiveOpen(TUint /*aQueSize*/,const TDesC8& /*aConnectionData*/)
    {
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
    LOGSMSPROT1("CSmsProvider::PassiveOpen [not supported]");
    return KErrNotSupported;
    BULLSEYE_RESTORE
    }

/**
 *  Called by the socket server to shutdown a connection.
 *  
 *  Implementation of the pure virtual CServProviderBase::Shutdown().
 *  
 */
void CSmsProvider::Shutdown(TCloseType aOption)
    {
    LOGSMSPROT2("CSmsProvider::Shutdown [aOption=%d]", aOption);
    
    TInt messagesInBuffer = iRecvBufSegArray->Count();
    for( TInt index = 0; index < messagesInBuffer; ++index )
        {
        // Read message from the socket
        CSmsMessage*smsmessage=NULL;
        TRAPD(ret,(smsmessage=InternalizeMessageL(iRecvBufSegArray->At(index))));
        if( ret == KErrNone )
            {
            TRAP(ret, (iProtocol.iReassemblyStore->SetMessagePassedToClientL(*smsmessage, EFalse)));
            LOGSMSPROT2("-> CSmsProvider::Shutdown - SetMessagePassedToClientL [ret=%d]", ret);
            }
        else
            {
            LOGSMSPROT2("-> CSmsProvider::Shutdown - CSmsProvider::InternalizeMessageL leave [ret=%d]", ret);
            }
        delete smsmessage;
        }
    
    if( iEnumSocket && iProtocol.iPhoneEnumerationObserver == this )
        {
        iEnumSocket=EFalse;
        iProtocol.iPhoneEnumerationObserver=NULL;
        }
    if( ObserverAddedToProtocol() )
        {
        iProtocol.CancelSendSmsMessage(*this);
        }    
    if( aOption!=CServProviderBase::EImmediate )
        {
        iSocket->CanClose();
        }
    } 

/**
 *  Called by the socket server to shutdown a connection.  Simply calls
 *  the single parameter version shutdown method.
 *  
 *  Implementation of the pure virtual CServProviderBase::Shutdown().
 *  
 */
void CSmsProvider::Shutdown(TCloseType aOption, const TDesC8& /*aDisconnectionData*/)
	{
	LOGSMSPROT1("CSmsProvider::Shutdown");
	Shutdown(aOption);
	}

/**
 *  Called by the socket server to indicate the provider should choose
 *  a local address and bind to it.  Not supported by this protocol.
 *  
 *  Implementation of the pure virtual CServProviderBase::AutoBind().
 *  
 */
void CSmsProvider::AutoBind()
	{
    // Ignore in code coverage - not intended to be used
    BULLSEYE_OFF    
	LOGSMSPROT1("CSmsProvider::AutoBind [does nothing]");
    BULLSEYE_RESTORE
	}

/**
 *  Called by the SMS protocol to obtain the local address of this SAP.
 *  
 *  @return the local address of this SAP.
 *  
 */
const TSmsAddr& CSmsProvider::GetLocalAddress() const
	{
	LOGSMSPROT1("CSmsProvider::GetLocalAddress");
	return iLocalAddress;
	}

/**
 *  Called by the SMS protocol to bind a local address to this SAP.
 *  
 */
void CSmsProvider::SetLocalAddress(const TSmsAddr& aSmsAddr)
	{
	LOGSMSPROT1("CSmsProvider::SetLocalAddress");
	iLocalAddress = aSmsAddr;
	}

/**
 *  Called by the protocol when the modem connection state has changed.
 *  Notifies any standard KIoctlSelect requests of this event.
 *  
 *  @param aStatus either KIoctlSelectModemPresent or KIoctlSelectModemNotPresent.
 *  
 */
void CSmsProvider::ModemNotificationCompleted(TInt aStatus)
	{
	LOGSMSPROT2("CSmsProvider::ModemNotificationCompleted [aStatus=%d]", aStatus);

	if( !IoctlOutstanding() )
	    {
		iSocket->Error(aStatus,MSocketNotify::EErrorIoctl);
	    }
	}

/**
 *  Called by the protocol when a message send request made by this
 *  SAP has completed.
 *  
 *  @param aStatus the result of the send.
 *  
 */
void CSmsProvider::MessageSendCompleted(TInt aStatus)
    {
    LOGSMSPROT2("CSmsProvider::MessageSendCompleted [aStatus=%d]", aStatus);
    
    iSocket->Error(aStatus,MSocketNotify::EErrorIoctl);
    SetIoctlOutstanding(EFalse);
    }

/**
 *  Called by the protocol when a message has been received for this SAP.
 *  
 *  @param aSmsMessage the received message.
 *  @return KErrNone if the message was serialized to the receive buffer successfully.
 *  
 */
TInt CSmsProvider::MessageReceived(const CSmsMessage& aSmsMessage,TDes& /*aDes*/)
    {
    LOGSMSPROT1("CSmsProvider::MessageReceived");
    
    // Attempt to serial the message to the receive buffer & notify
    // the socket of the new data
    TInt numnewsegments=0;
    TRAPD(ret,(numnewsegments=ExternalizeMessageL(aSmsMessage,ETrue)));
    if( ret==KErrNone )
        {
        iSocket->NewData(numnewsegments);
        }
    return ret;
    }

/**
 *  Informs protocol whether client confirms received message
 */
TBool CSmsProvider::ClientConfirmsMessage() const
    {
    LOGSMSPROT1("CSmsProvider::ClientConfirmsMessage");
    
    return ETrue;
    }

/**
 *  Informs protocol whether address is ued by the observer
 */
TInt CSmsProvider::SmsAddrIsDuplicate(const MSmsMessageObserver* aObserver, const TSmsAddr& aAddr) const
    {
    LOGSMSPROT1("CSmsProvider::SmsAddrIsDuplicate");
    
    if( this == aObserver )
        {
        return EFalse;
        }
    return iLocalAddress==aAddr;
    }

/**
 *  Called by the protocol when an enumeration of the phone's message stores
 *  requested by this SAP has completed.
 *  
 *  @param aStatus the result of the enumeration.
 *  
 */
void CSmsProvider::EnumeratePhoneCompleted(TInt aStatus)
    {
    LOGSMSPROT2("CSmsProvider::EnumeratePhoneCompleted [aStatus=%d]", aStatus);
    
    // Attempt to serialize all enumerated messages to the receive buffer
    TInt numnewsegments=0;
    TInt count=0;
    if( aStatus==KErrNone )
        {
        // Save current message count in case we need to rollback on error
        TInt nummessages=iRecvBufSegArray->Count();
        TRAP(aStatus, (numnewsegments=ExternalizeEnumeratedMessagesL(count)));
        if( aStatus==KErrNone )
            {
            // Success, obtain the message count and notify socket of the new data
            if( numnewsegments>0 )
                {
                iSocket->NewData(numnewsegments);
                }
            }
        else
            {
            // Error, rollback the messages we added
            for( TInt i=(iRecvBufSegArray->Count()-nummessages)-1; i>=0; i-- )
                {
                delete iRecvBufSegArray->At(i);
                iRecvBufSegArray->Delete(i);
                }
            }
        }
    // On success, complete with the count of messages enumerated
    if( aStatus==KErrNone )
        {
        TPckgBuf<TInt> buf;
        buf()=count;
        if( count>0 )
            {
            iEnumSocket=ETrue;
            }
        else
            {
            iEnumSocket=EFalse;
            iProtocol.iPhoneEnumerationObserver=NULL;
            }
        iNumOfEnumeratedMessages=count;
        iSocket->IoctlComplete(&buf);
        }
    else
        {
        iEnumSocket=EFalse;
        if( iProtocol.iPhoneEnumerationObserver == this)
            {
            iProtocol.iPhoneEnumerationObserver=NULL;
            }
        iSocket->Error(aStatus,MSocketNotify::EErrorIoctl);
        }
    SetIoctlOutstanding(EFalse);
    }

/**
 *  Called by the protocol when a request to write a message to
 *  a phone store by this SAP has completed.
 *  Implements the pure virtual MSmsMessageObserver::MessageWriteCompleted().
 *  
 *  @param aStatus the result of the write operation.
 *  
 */
void CSmsProvider::MessageWriteCompleted(TInt aStatus, const CSmsMessage* aSmsMessage)
	{
	LOGSMSPROT2("CSmsProvider::MessageWriteCompleted [aStatus=%d]", aStatus);

	// If no errors at present populate the buffer
	if( aStatus == KErrNone )
	    {
		TRAP(aStatus, (PopulateBufferWithPDUSlotsL(*aSmsMessage)));
	    }
	if( aStatus != KErrNone )
		{
		iSocket->Error(aStatus,MSocketNotify::EErrorIoctl);
		}
	SetIoctlOutstanding(EFalse);
	} // CSmsProvider::MessageWriteCompleted


/**
 *  Create and populate a buffer containing store type and PDU slot indexes
 *  
 *  @param aSmsMessage the message containing the slot information.
 *  
 */
void CSmsProvider::PopulateBufferWithPDUSlotsL(const CSmsMessage& aSmsMessage)
    {
    LOGSMSPROT1("CSmsProvider::PopulateBufferWithPDUSlotsL");
    
    // Create buffer for store id and PDU slot indexes based on size of slot array
    
    HBufC8* buf = HBufC8::NewL(aSmsMessage.iSlotArray.Count()+1 * sizeof(TUint));
    buf->Des().Append(aSmsMessage.Storage());
    
    TInt count = aSmsMessage.iSlotArray.Count();
    for( TInt index=0; index<count; ++index )
        {
        buf->Des().Append(aSmsMessage.iSlotArray[index].iIndex);
        }
    
    TPtr8 textBufPtr(buf->Des());
    
    iSocket->IoctlComplete(&textBufPtr);
    delete buf;

    LOGSMSPROT1("-> CSmsProvider::PopulateBufferWithPDUSlotsL - done");
    }

/**
 *  Called by the protocol when a request to delete a message from
 *  a phone store by this SAP has completed.
 *  Implements the pure virtual MSmsMessageObserver::MessageDeleteCompleted().
 *  
 *  @param aStatus the result of the deletion.
 *  
 */
void CSmsProvider::MessageDeleteCompleted(TInt aStatus)
    {
    LOGSMSPROT2("CSmsProvider::MessageDeleteCompleted [aStatus=%d]", aStatus);
    iSocket->Error(aStatus,MSocketNotify::EErrorIoctl);
    SetIoctlOutstanding(EFalse);
    }

/**
 *  Called by the CSmsReadParams object when a read all SMS parameter sets
 *  requested by this SAP has completed.
 *  
 *  @param aStatus the result of the read operation.
 *  @param aSmspList SMS parameter list
 *  
 */
void CSmsProvider::ReadSmsParamsCompleted(TInt aStatus, CMobilePhoneSmspList* aSmspList)
    {
    LOGSMSPROT2("CSmsProvider::ReadSmsParamsCompleted [aStatus=%d]", aStatus);
    
    TInt numNewSegments=0;
    
    if( aStatus == KErrNone )
        {
        // Attempt to serial the parameters to the receive buffer & notify
        // the socket of the new data
        __ASSERT_DEBUG(aSmspList != NULL,SmspPanic(KSmspPanicParameterBufferNull));
        TRAP(aStatus,(numNewSegments=ExternalizeParametersL(*aSmspList)));
        }
    if( aStatus == KErrNone )
        {
        iSocket->NewData(numNewSegments);
        TPckgBuf<TInt> buf;
        buf()=1;
        iSocket->IoctlComplete(&buf);
        }
    else
        {
        iSocket->Error(aStatus,MSocketNotify::EErrorIoctl);
        }
    SetIoctlOutstanding(EFalse);
    }

/**
 *  Called by the CSmsWriteParams object when a request to write SMS
 *  parameters to a phone store by this SAP has completed.
 *  
 *  @param aStatus the result of the write operation.
 *  
 */
void CSmsProvider::WriteSmsParamsCompleted(TInt aStatus)
    {
    LOGSMSPROT2("CSmsProvider::WriteSmsParamsCompleted [aStatus=%d]", aStatus);
    iSocket->Error(aStatus,MSocketNotify::EErrorIoctl);
    SetIoctlOutstanding(EFalse);
    }

/**
 *  Internal function called after a phone store enumeration request to
 *  serialize the messages to the receive buffer.
 *  
 *  @leave Leaves if any individual ExternalizeMessageL() call leaves.
 *  @return the total number of segments the messages were split into.
 *  
 */
TInt CSmsProvider::ExternalizeEnumeratedMessagesL(TInt& aCount)
    {
    LOGSMSPROT1("CSmsProvider::ExternalizeEnumeratedMessagesL");
    
    TInt numnewsegments(0);
    numnewsegments=iProtocol.ExternalizeEnumeratedMessagesL(*this,aCount);
    
    LOGSMSPROT1("-> CSmsProvider::ExternalizeEnumeratedMessagesL - done");
    
    return numnewsegments;
    }

/**
 *  Internal function used to serialize a message into the receive buffer.
 *  Each serialized message is split into smaller segments which form the
 *  basic unit of data passed back to the socket server in the GetData() method.
 *  
 *  @param aSmsMessage the message to serialize.
 *  @param aAppend specifies whether the message is inserted at the start or end of the buffer.
 *  @leave Leaves if the message could not be serialized or inserted into the buffer.
 *  @return the number of segments the message was split into.
 *  
 */
TInt CSmsProvider::ExternalizeMessageL(const CSmsMessage& aSmsMessage,TBool aAppend)
    {
    LOGSMSPROT1("CSmsProvider::ExternalizeMessageL()");
    
    // Create a new segmented buffer for the serialization of this message
    CBufSeg* recvbufseg = CBufSeg::NewL(KSmsMaxSegmentLength);
    CleanupStack::PushL(recvbufseg);
    
    // Attempt to serialize this message into the buffer
    RBufWriteStream writestream(*recvbufseg);
    writestream.Open(*recvbufseg);
    CleanupClosePushL(writestream);
    writestream << aSmsMessage;
    
    // Append / insert this buffer at the end / start of the other serialized message buffers
    if( aAppend )
        {
        iRecvBufSegArray->AppendL(recvbufseg);
        }
    else
        {
        iRecvBufSegArray->InsertL(0,recvbufseg);
        }
    CleanupStack::PopAndDestroy();  //  writestream
    CleanupStack::Pop();            //  recvbufseg
    
    LOGSMSPROT1("-> CSmsProvider::ExternalizeMessageL - done");
    
    return NumSegments(recvbufseg->Size());
    }

/**
 *  Internal function to deserialize a message from the send buffer.
 *  Each serialized message is split into smaller segments which form the basic
 *  unit of data passed to this SAP from the socket server in the Write() method.
 *  
 *  @leave Leaves if the message could not be de-serialized.
 *  @return the de-serialized CSmsMessage object.
 *  
 */
CSmsMessage* CSmsProvider::InternalizeMessageL()
    {
    LOGSMSPROT1("CSmsProvider::InternalizeMessageL()");
    
    // Initialize the read stream with the buffer
    RBufReadStream readstream(*iSendBufSeg);
    readstream.Open(*iSendBufSeg,0);
    CleanupClosePushL(readstream);

    // Create a buffer and message to store the result
    CSmsBufferBase* buffer = CSmsBuffer::NewL();
    CSmsMessage* smsmessage= CSmsMessage::NewL(iProtocol.FileSession(), CSmsPDU::ESmsDeliver,buffer);
    CleanupStack::PushL(smsmessage);

    // De-serialize the message from using the read stream
    readstream >> *smsmessage;

    CleanupStack::Pop();  //  smsmessage
    CleanupStack::PopAndDestroy();  //  readstream
    iSendBufSeg->Reset();

    LOGSMSPROT1("-> CSmsProvider::InternalizeMessageL - done");
    
    return smsmessage;
    }

/**
 *  Internal function used to serialize SMS parameters into the receive buffer.
 *  Each serialized parameter object is split into smaller segments which form the
 *  basic unit of data passed back to the socket server in the GetData() method.
 *  
 *  @param aMobilePhoneSmspList the parameters to serialize.
 *  @leave Leaves if the parameters could not be serialized or inserted into the buffer.
 *  @return the number of segments the parameters was split into.
 *  
 */
TInt CSmsProvider::ExternalizeParametersL(const CMobilePhoneSmspList& aMobilePhoneSmspList)
    {
    LOGSMSPROT1("CSmsProvider::ExternalizeParametersL");
    
    // Create a new segmented buffer for the serialization of this message
    CBufSeg* recvBufSeg = CBufSeg::NewL(KSmsMaxSegmentLength);
    CleanupStack::PushL(recvBufSeg);

    // Attempt to serialize this message into the buffer
    RBufWriteStream writeStream(*recvBufSeg);
    writeStream.Open(*recvBufSeg);
    CleanupClosePushL(writeStream);
    writeStream << aMobilePhoneSmspList;
    writeStream.CommitL();

    // Append start of the other serialized message buffers
    CleanupStack::PopAndDestroy();	//writeStream
    iRecvBufSegArray->InsertL(0,recvBufSeg);
    CleanupStack::Pop(recvBufSeg);

    LOGSMSPROT1("-> CSmsProvider::ExternalizeParametersL - done");
    
    return NumSegments(recvBufSeg->Size());
    }

/**
 *  Internal function to retrieve a SMS parameters from the send buffer.
 *  Each serialized object is split into smaller segments which form the basic
 *  unit of data passed to this SAP from the socket server in the Write() method.
 *  
 *  @leave Leaves if the parameters could not retrieved.
 *  @return the retrieved CMobilePhoneSmspList object.
 *  
 */
CMobilePhoneSmspList* CSmsProvider::InternalizeParametersL()
    {
    LOGSMSPROT1("CSmsProvider::InternalizeParametersL");
    
    // Initialize the read stream with the buffer
    RBufReadStream readStream(*iSendBufSeg);
    readStream.Open(*iSendBufSeg,0);
    CleanupClosePushL(readStream);

    // Create a parameter object to store the result
    CMobilePhoneSmspList* mobilePhoneSmspList = CMobilePhoneSmspList::NewL();
    CleanupStack::PushL(mobilePhoneSmspList);

    // De-serialize the message from using the read stream
    readStream >> *mobilePhoneSmspList;

    CleanupStack::Pop(mobilePhoneSmspList);
    CleanupStack::PopAndDestroy();	//readStream
    iSendBufSeg->Reset();

    LOGSMSPROT1("-> CSmsProvider::InternalizeParametersL - done");
    
    return mobilePhoneSmspList;
    }

CSmsMessage* CSmsProvider::InternalizeMessageL(	CBufSeg* aBufSeg)
    {
    LOGSMSPROT1("CSmsProvider::InternalizeMessageL");
    
    RBufReadStream readstream(*aBufSeg);
    readstream.Open(*aBufSeg,0);
    CleanupClosePushL(readstream);
    CSmsBufferBase* buffer = CSmsBuffer::NewL();
    CSmsMessage* smsmessage= CSmsMessage::NewL(iProtocol.FileSession(),CSmsPDU::ESmsDeliver,buffer);
    
    CleanupStack::PushL(smsmessage);
    readstream >> *smsmessage;
    
    CleanupStack::Pop();  //  smsmessage
    CleanupStack::PopAndDestroy();  //  readsream
    
    LOGSMSPROT1("-> CSmsProvider::InternalizeMessageL - done");
    
    return smsmessage;
    }

TInt CSmsProvider::SecurityCheck(MProvdSecurityChecker* aSecurityChecker)
    {
    LOGSMSPROT1("CSmsProvider::SecurityCheck");
    iSecurityChecker = aSecurityChecker;
    return KErrNone;
    }