imservices/instantmessagingcache/imcacheclient/src/imcacheprocessstarter.cpp
author hgs
Wed, 03 Nov 2010 22:25:05 +0530
changeset 52 3d676fce9a4e
parent 51 61fad867f68e
permissions -rw-r--r--
201044_02
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
51
hgs
parents:
diff changeset
     1
/*
hgs
parents:
diff changeset
     2
* Copyright (c) 2008 Nokia Corporation and/or its subsidiary(-ies).
hgs
parents:
diff changeset
     3
* All rights reserved.
hgs
parents:
diff changeset
     4
* This component and the accompanying materials are made available
hgs
parents:
diff changeset
     5
* under the terms of "Eclipse Public License v1.0"
hgs
parents:
diff changeset
     6
* which accompanies this distribution, and is available
hgs
parents:
diff changeset
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
hgs
parents:
diff changeset
     8
*
hgs
parents:
diff changeset
     9
* Initial Contributors:
hgs
parents:
diff changeset
    10
* Nokia Corporation - initial contribution.
hgs
parents:
diff changeset
    11
*
hgs
parents:
diff changeset
    12
* Contributors:
hgs
parents:
diff changeset
    13
*
hgs
parents:
diff changeset
    14
* Description:  start the server
hgs
parents:
diff changeset
    15
*
hgs
parents:
diff changeset
    16
*/
hgs
parents:
diff changeset
    17
hgs
parents:
diff changeset
    18
hgs
parents:
diff changeset
    19
#include "imcacheprocessstarter.h"
hgs
parents:
diff changeset
    20
// logs
hgs
parents:
diff changeset
    21
#include "imcachedebugtrace.h"
hgs
parents:
diff changeset
    22
// system includes
hgs
parents:
diff changeset
    23
#include <e32std.h>
hgs
parents:
diff changeset
    24
#include <f32file.h>
hgs
parents:
diff changeset
    25
hgs
parents:
diff changeset
    26
hgs
parents:
diff changeset
    27
// CONSTANTS
hgs
parents:
diff changeset
    28
_LIT( KEka2ExeDir,"\\sys\\bin\\");
hgs
parents:
diff changeset
    29
_LIT( KEka2LaunchMutexExt, "[lMtx]" );
hgs
parents:
diff changeset
    30
const TInt KEka2SrvConnTries = 7;
hgs
parents:
diff changeset
    31
const TInt KEka2SrvConnInitialRetryWait = 500; //MicroSeconds => 0.0005s
hgs
parents:
diff changeset
    32
hgs
parents:
diff changeset
    33
hgs
parents:
diff changeset
    34
// ==============================================================
hgs
parents:
diff changeset
    35
// ====================== HELPER CLASS ==========================
hgs
parents:
diff changeset
    36
// ==============================================================
hgs
parents:
diff changeset
    37
hgs
parents:
diff changeset
    38
/**
hgs
parents:
diff changeset
    39
 * RSessionBase accessor to give to the ProcessStarter
hgs
parents:
diff changeset
    40
 * access to RSessionBase::CreateSession().
hgs
parents:
diff changeset
    41
 */
hgs
parents:
diff changeset
    42
class REka2SessionBaseAccessor : public RSessionBase
hgs
parents:
diff changeset
    43
    {
hgs
parents:
diff changeset
    44
    public: // Constructor
hgs
parents:
diff changeset
    45
        inline REka2SessionBaseAccessor()
hgs
parents:
diff changeset
    46
            {
hgs
parents:
diff changeset
    47
            }
hgs
parents:
diff changeset
    48
hgs
parents:
diff changeset
    49
    public: // New functions
hgs
parents:
diff changeset
    50
hgs
parents:
diff changeset
    51
        /**
hgs
parents:
diff changeset
    52
         * Public access to RSessionBase::CreateSession().
hgs
parents:
diff changeset
    53
         */
hgs
parents:
diff changeset
    54
        inline TInt CreateSession( const TDesC& aServer,
hgs
parents:
diff changeset
    55
                                   const TVersion& aVersion,
hgs
parents:
diff changeset
    56
                                   TInt aAsyncMessageSlots )
hgs
parents:
diff changeset
    57
            {
hgs
parents:
diff changeset
    58
            return RSessionBase::CreateSession( aServer,
hgs
parents:
diff changeset
    59
                                                aVersion,
hgs
parents:
diff changeset
    60
                                                aAsyncMessageSlots );
hgs
parents:
diff changeset
    61
            }
hgs
parents:
diff changeset
    62
    };
hgs
parents:
diff changeset
    63
hgs
parents:
diff changeset
    64
hgs
parents:
diff changeset
    65
// ==============================================================
hgs
parents:
diff changeset
    66
// ====================== PROCESSSTARTER ========================
hgs
parents:
diff changeset
    67
// ==============================================================
hgs
parents:
diff changeset
    68
hgs
parents:
diff changeset
    69
// --------------------------------------------------------------
hgs
parents:
diff changeset
    70
// IMCacheProcessStarter::FullExePathForClientLocation()
hgs
parents:
diff changeset
    71
// --------------------------------------------------------------
hgs
parents:
diff changeset
    72
//
hgs
parents:
diff changeset
    73
void IMCacheProcessStarter::FullExePathForClientLocation(
hgs
parents:
diff changeset
    74
    const TDesC& aExeName,
hgs
parents:
diff changeset
    75
    TFileName& aFullExePath )
hgs
parents:
diff changeset
    76
    {
hgs
parents:
diff changeset
    77
	TRACE( T_LIT("IMCacheProcessStarter::FullExePathForClientLocation begin") );
hgs
parents:
diff changeset
    78
    //Get drive (C:) where this client code is installed
hgs
parents:
diff changeset
    79
        {
hgs
parents:
diff changeset
    80
        TFileName tmp;
hgs
parents:
diff changeset
    81
        Dll::FileName( tmp );
hgs
parents:
diff changeset
    82
        aFullExePath.Copy( TParsePtrC( tmp ).Drive() );
hgs
parents:
diff changeset
    83
        }
hgs
parents:
diff changeset
    84
hgs
parents:
diff changeset
    85
    //Build the rest from the exe path
hgs
parents:
diff changeset
    86
    aFullExePath.Append( KEka2ExeDir );
hgs
parents:
diff changeset
    87
    aFullExePath.Append( aExeName );
hgs
parents:
diff changeset
    88
    TRACE( T_LIT("IMCacheProcessStarter::FullExePathForClientLocation end") );
hgs
parents:
diff changeset
    89
    }
hgs
parents:
diff changeset
    90
hgs
parents:
diff changeset
    91
// --------------------------------------------------------------
hgs
parents:
diff changeset
    92
// IMCacheProcessStarter::StartInstance()
hgs
parents:
diff changeset
    93
// --------------------------------------------------------------
hgs
parents:
diff changeset
    94
//
hgs
parents:
diff changeset
    95
TInt IMCacheProcessStarter::StartInstance(
hgs
parents:
diff changeset
    96
    const TDesC& aFullExePath,
hgs
parents:
diff changeset
    97
    const TDesC& aCommand )
hgs
parents:
diff changeset
    98
    {
hgs
parents:
diff changeset
    99
    TRACE( T_LIT("IMCacheProcessStarter::StartInstance begin") );
hgs
parents:
diff changeset
   100
    RMutex launchMutex;
hgs
parents:
diff changeset
   101
    TInt error = KErrNotFound;
hgs
parents:
diff changeset
   102
hgs
parents:
diff changeset
   103
        {
hgs
parents:
diff changeset
   104
        // Dynamic mutex name used to allow code share
hgs
parents:
diff changeset
   105
        TName launchMutexName( TParsePtrC( aFullExePath ).Name() );
hgs
parents:
diff changeset
   106
        launchMutexName.Append( KEka2LaunchMutexExt );
hgs
parents:
diff changeset
   107
hgs
parents:
diff changeset
   108
        // Open or Create mutex to serialize to access to server startup code.
hgs
parents:
diff changeset
   109
        // (race condition safe way)
hgs
parents:
diff changeset
   110
        while( error == KErrNotFound )
hgs
parents:
diff changeset
   111
            {
hgs
parents:
diff changeset
   112
            error = launchMutex.CreateGlobal( launchMutexName );
hgs
parents:
diff changeset
   113
            if( error != KErrAlreadyExists )
hgs
parents:
diff changeset
   114
                {
hgs
parents:
diff changeset
   115
                break;
hgs
parents:
diff changeset
   116
                }
hgs
parents:
diff changeset
   117
            error = launchMutex.OpenGlobal( launchMutexName );
hgs
parents:
diff changeset
   118
            }
hgs
parents:
diff changeset
   119
hgs
parents:
diff changeset
   120
        if( error != KErrNone )
hgs
parents:
diff changeset
   121
            {
hgs
parents:
diff changeset
   122
            TRACE( T_LIT("IMCacheProcessStarter::StartInstance end") );
hgs
parents:
diff changeset
   123
            return error;
hgs
parents:
diff changeset
   124
            }
hgs
parents:
diff changeset
   125
        }
hgs
parents:
diff changeset
   126
hgs
parents:
diff changeset
   127
hgs
parents:
diff changeset
   128
    launchMutex.Wait();
hgs
parents:
diff changeset
   129
hgs
parents:
diff changeset
   130
    //Serialized access
hgs
parents:
diff changeset
   131
    error = IMCacheProcessStarter::DoStartInstance( aFullExePath,
hgs
parents:
diff changeset
   132
                                             aCommand );
hgs
parents:
diff changeset
   133
hgs
parents:
diff changeset
   134
    launchMutex.Signal();
hgs
parents:
diff changeset
   135
    launchMutex.Close();
hgs
parents:
diff changeset
   136
	TRACE( T_LIT("IMCacheProcessStarter::StartInstance end") );
hgs
parents:
diff changeset
   137
    return error;
hgs
parents:
diff changeset
   138
    }
hgs
parents:
diff changeset
   139
hgs
parents:
diff changeset
   140
// --------------------------------------------------------------
hgs
parents:
diff changeset
   141
// IMCacheProcessStarter::ConnectToServer()
hgs
parents:
diff changeset
   142
// --------------------------------------------------------------
hgs
parents:
diff changeset
   143
//
hgs
parents:
diff changeset
   144
TInt IMCacheProcessStarter::ConnectToServer(
hgs
parents:
diff changeset
   145
    const TDesC& aFullExePath,
hgs
parents:
diff changeset
   146
    const TDesC& aCommand,
hgs
parents:
diff changeset
   147
    RSessionBase& aSessionToConnect,
hgs
parents:
diff changeset
   148
    const TDesC& aServerName,
hgs
parents:
diff changeset
   149
    const TVersion& aClientVersion,
hgs
parents:
diff changeset
   150
    TInt aAsyncMessageSlots )
hgs
parents:
diff changeset
   151
    {
hgs
parents:
diff changeset
   152
    TRACE( T_LIT("IMCacheProcessStarter::ConnectToServer begin") );
hgs
parents:
diff changeset
   153
    if( aSessionToConnect.Handle() != KNullHandle )
hgs
parents:
diff changeset
   154
        {
hgs
parents:
diff changeset
   155
        return KErrInUse;
hgs
parents:
diff changeset
   156
        }
hgs
parents:
diff changeset
   157
hgs
parents:
diff changeset
   158
    TInt err = KErrNone;
hgs
parents:
diff changeset
   159
    TInt startupWait = KEka2SrvConnInitialRetryWait;
hgs
parents:
diff changeset
   160
hgs
parents:
diff changeset
   161
    //Server connect and launch loop
hgs
parents:
diff changeset
   162
    for( TInt trie = 0 ; trie < KEka2SrvConnTries ; trie++ )
hgs
parents:
diff changeset
   163
        {
hgs
parents:
diff changeset
   164
        REka2SessionBaseAccessor acc;
hgs
parents:
diff changeset
   165
        err = acc.CreateSession( aServerName,
hgs
parents:
diff changeset
   166
                                 aClientVersion,
hgs
parents:
diff changeset
   167
                                 aAsyncMessageSlots );
hgs
parents:
diff changeset
   168
hgs
parents:
diff changeset
   169
        if( err == KErrNone )
hgs
parents:
diff changeset
   170
            {
hgs
parents:
diff changeset
   171
            //session ownership is now on client
hgs
parents:
diff changeset
   172
            aSessionToConnect = acc;
hgs
parents:
diff changeset
   173
            return KErrNone;
hgs
parents:
diff changeset
   174
            }
hgs
parents:
diff changeset
   175
hgs
parents:
diff changeset
   176
        else if( ( err == KErrNotFound ) ||
hgs
parents:
diff changeset
   177
                 ( err == KErrServerTerminated ) )
hgs
parents:
diff changeset
   178
            {
hgs
parents:
diff changeset
   179
            //Server missing or died when connecting
hgs
parents:
diff changeset
   180
            //Start a new server
hgs
parents:
diff changeset
   181
            err = IMCacheProcessStarter::StartInstance( aFullExePath,
hgs
parents:
diff changeset
   182
                                                 aCommand );
hgs
parents:
diff changeset
   183
hgs
parents:
diff changeset
   184
            //If process exist already, then all is fine
hgs
parents:
diff changeset
   185
            //(some other process started it between the origical connect and launch trie)
hgs
parents:
diff changeset
   186
            if( err == KErrAlreadyExists )
hgs
parents:
diff changeset
   187
                {
hgs
parents:
diff changeset
   188
                err = KErrNone;
hgs
parents:
diff changeset
   189
                }
hgs
parents:
diff changeset
   190
hgs
parents:
diff changeset
   191
            //If server process start failed, bail out.
hgs
parents:
diff changeset
   192
            if( err != KErrNone )
hgs
parents:
diff changeset
   193
                {
hgs
parents:
diff changeset
   194
                return err;
hgs
parents:
diff changeset
   195
                }
hgs
parents:
diff changeset
   196
hgs
parents:
diff changeset
   197
            //If this is 2nd or subsequent try,
hgs
parents:
diff changeset
   198
            //give some time for server to startup
hgs
parents:
diff changeset
   199
            if( trie > 0 )
hgs
parents:
diff changeset
   200
                {
hgs
parents:
diff changeset
   201
                // Code scanner warning : Use of User::After (id:92)
hgs
parents:
diff changeset
   202
                // it is required to be used here
hgs
parents:
diff changeset
   203
                User::After( startupWait ); // CSI: 92 # See above
hgs
parents:
diff changeset
   204
                startupWait = 2 * startupWait;
hgs
parents:
diff changeset
   205
                }
hgs
parents:
diff changeset
   206
            }
hgs
parents:
diff changeset
   207
hgs
parents:
diff changeset
   208
        else
hgs
parents:
diff changeset
   209
            {
hgs
parents:
diff changeset
   210
            //Server process start failed. Bail out.
hgs
parents:
diff changeset
   211
            return err;
hgs
parents:
diff changeset
   212
            }
hgs
parents:
diff changeset
   213
        }
hgs
parents:
diff changeset
   214
	TRACE( T_LIT("IMCacheProcessStarter::ConnectToServer end") );
hgs
parents:
diff changeset
   215
    return err;
hgs
parents:
diff changeset
   216
    }
hgs
parents:
diff changeset
   217
hgs
parents:
diff changeset
   218
// --------------------------------------------------------------
hgs
parents:
diff changeset
   219
// IMCacheProcessStarter::DoStartServerInstance()
hgs
parents:
diff changeset
   220
// --------------------------------------------------------------
hgs
parents:
diff changeset
   221
//
hgs
parents:
diff changeset
   222
TInt IMCacheProcessStarter::DoStartInstance(
hgs
parents:
diff changeset
   223
    const TDesC& aFullExePath,
hgs
parents:
diff changeset
   224
    const TDesC& aCommand )
hgs
parents:
diff changeset
   225
    {
hgs
parents:
diff changeset
   226
    TRACE( T_LIT("IMCacheProcessStarter::DoStartInstance begin") );
hgs
parents:
diff changeset
   227
    TInt error = KErrNone;
hgs
parents:
diff changeset
   228
hgs
parents:
diff changeset
   229
    //Create process
hgs
parents:
diff changeset
   230
    RProcess process;
hgs
parents:
diff changeset
   231
    error = process.Create( aFullExePath, aCommand );
hgs
parents:
diff changeset
   232
hgs
parents:
diff changeset
   233
    if( error == KErrNone )
hgs
parents:
diff changeset
   234
        {
hgs
parents:
diff changeset
   235
       
hgs
parents:
diff changeset
   236
            TRequestStatus rendezvousStatus;
hgs
parents:
diff changeset
   237
            process.Rendezvous( rendezvousStatus );
hgs
parents:
diff changeset
   238
hgs
parents:
diff changeset
   239
            process.Resume();
hgs
parents:
diff changeset
   240
            // Codescanner warning: user of User::WaitForRequest (Id:94)
hgs
parents:
diff changeset
   241
            // it is required to use at server startup
hgs
parents:
diff changeset
   242
            User::WaitForRequest( rendezvousStatus ); // CSI: 94 # See above
hgs
parents:
diff changeset
   243
            error = rendezvousStatus.Int();
hgs
parents:
diff changeset
   244
hgs
parents:
diff changeset
   245
            if( process.ExitType() != EExitPending )
hgs
parents:
diff changeset
   246
                {
hgs
parents:
diff changeset
   247
                //Something failed in server startup
hgs
parents:
diff changeset
   248
                //Force the error code to be always something
hgs
parents:
diff changeset
   249
                //else than KErrNone
hgs
parents:
diff changeset
   250
                if( error == KErrNone )
hgs
parents:
diff changeset
   251
                    {
hgs
parents:
diff changeset
   252
                    error = KErrServerTerminated;
hgs
parents:
diff changeset
   253
                    }
hgs
parents:
diff changeset
   254
                }
hgs
parents:
diff changeset
   255
           
hgs
parents:
diff changeset
   256
        }
hgs
parents:
diff changeset
   257
hgs
parents:
diff changeset
   258
    process.Close();
hgs
parents:
diff changeset
   259
	TRACE( T_LIT("IMCacheProcessStarter::DoStartInstance end") );
hgs
parents:
diff changeset
   260
    return error;
hgs
parents:
diff changeset
   261
    }
hgs
parents:
diff changeset
   262
hgs
parents:
diff changeset
   263
// END OF FILE
hgs
parents:
diff changeset
   264
hgs
parents:
diff changeset
   265
hgs
parents:
diff changeset
   266