keepalive/flextimer/client/src/rflextimer.cpp
author hgs
Mon, 24 May 2010 20:51:35 +0300
changeset 32 5c4486441ae6
permissions -rw-r--r--
201021
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
32
hgs
parents:
diff changeset
     1
/*
hgs
parents:
diff changeset
     2
 * Copyright (c) 2010 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:  Implementation of RFlexTimer class
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
 * %version: 1 %
hgs
parents:
diff changeset
    20
 */
hgs
parents:
diff changeset
    21
#include "rflextimer.h"
hgs
parents:
diff changeset
    22
#include "flextimercommon.h"
hgs
parents:
diff changeset
    23
#include "OstTraceDefinitions.h"
hgs
parents:
diff changeset
    24
#ifdef OST_TRACE_COMPILER_IN_USE
hgs
parents:
diff changeset
    25
#include "rflextimerTraces.h"
hgs
parents:
diff changeset
    26
#endif
hgs
parents:
diff changeset
    27
hgs
parents:
diff changeset
    28
hgs
parents:
diff changeset
    29
_LIT( KFlexTimerSemaphoreName, "FlexTimerSemaphore" );
hgs
parents:
diff changeset
    30
hgs
parents:
diff changeset
    31
// Semaphore count value initialization to 1. The 1st to call Wait() will
hgs
parents:
diff changeset
    32
// pass the semaphore. Other processes attempting entry will wait until the
hgs
parents:
diff changeset
    33
// 1st one has called Signal() on the semaphore.
hgs
parents:
diff changeset
    34
const TInt KPassFirstWaitEntry = 1;
hgs
parents:
diff changeset
    35
hgs
parents:
diff changeset
    36
// ======== MEMBER FUNCTIONS ========
hgs
parents:
diff changeset
    37
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
    38
// Constructs the object.
hgs
parents:
diff changeset
    39
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
    40
//
hgs
parents:
diff changeset
    41
EXPORT_C RFlexTimer::RFlexTimer()
hgs
parents:
diff changeset
    42
    {
hgs
parents:
diff changeset
    43
    OstTrace1( TRACE_NORMAL, RFLEXTIMER_RFLEXTIMER,
hgs
parents:
diff changeset
    44
               "RFlexTimer::RFlexTimer;this=%x", this );
hgs
parents:
diff changeset
    45
    
hgs
parents:
diff changeset
    46
    }
hgs
parents:
diff changeset
    47
hgs
parents:
diff changeset
    48
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
    49
// Destructs the object.
hgs
parents:
diff changeset
    50
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
    51
//
hgs
parents:
diff changeset
    52
EXPORT_C RFlexTimer::~RFlexTimer()
hgs
parents:
diff changeset
    53
    {
hgs
parents:
diff changeset
    54
    OstTrace1( TRACE_NORMAL, DUP1_RFLEXTIMER_RFLEXTIMER,
hgs
parents:
diff changeset
    55
               "RFlexTimer::~RFlexTimer;this=%x", ( TUint )this );
hgs
parents:
diff changeset
    56
    
hgs
parents:
diff changeset
    57
    Close();
hgs
parents:
diff changeset
    58
    }
hgs
parents:
diff changeset
    59
hgs
parents:
diff changeset
    60
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
    61
// Connects to the server and create a session.
hgs
parents:
diff changeset
    62
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
    63
//
hgs
parents:
diff changeset
    64
EXPORT_C TInt RFlexTimer::Connect()
hgs
parents:
diff changeset
    65
    {
hgs
parents:
diff changeset
    66
    OstTrace1( TRACE_NORMAL, RFLEXTIMER_CONNECT,
hgs
parents:
diff changeset
    67
               "RFlexTimer::Connect;this=%x", ( TUint )this );
hgs
parents:
diff changeset
    68
    
hgs
parents:
diff changeset
    69
    // Not opened handle-number
hgs
parents:
diff changeset
    70
    const TInt kHandleNotOpened( 0 );
hgs
parents:
diff changeset
    71
    
hgs
parents:
diff changeset
    72
    // Starts the server, if it does not already exist in the system.
hgs
parents:
diff changeset
    73
    TInt ret = StartServer();
hgs
parents:
diff changeset
    74
hgs
parents:
diff changeset
    75
    if ( ret == KErrNone )
hgs
parents:
diff changeset
    76
        { // No session, create it
hgs
parents:
diff changeset
    77
hgs
parents:
diff changeset
    78
        // Handle() is zero when initialized RHandleBase. Close() also zeroes 
hgs
parents:
diff changeset
    79
        // the handle. If CreateSession() fails, Handle() is still zero.
hgs
parents:
diff changeset
    80
        // If session is created ok, the Handle() contains the (non zero)
hgs
parents:
diff changeset
    81
        // handle-number. 
hgs
parents:
diff changeset
    82
        //
hgs
parents:
diff changeset
    83
        // Handle() can be used for checking is the handle opened or not.
hgs
parents:
diff changeset
    84
        //
hgs
parents:
diff changeset
    85
        if ( Handle() == kHandleNotOpened )
hgs
parents:
diff changeset
    86
            {
hgs
parents:
diff changeset
    87
            // Creates the session for this client.
hgs
parents:
diff changeset
    88
            ret = CreateSession( KFlexTimerServerName,
hgs
parents:
diff changeset
    89
                                 Version(),
hgs
parents:
diff changeset
    90
                                 KFlexTimerServerMessageSlots );
hgs
parents:
diff changeset
    91
            }
hgs
parents:
diff changeset
    92
        else
hgs
parents:
diff changeset
    93
            { // Session already exists - panic.
hgs
parents:
diff changeset
    94
hgs
parents:
diff changeset
    95
            OstTrace1( 
hgs
parents:
diff changeset
    96
                TRACE_NORMAL, 
hgs
parents:
diff changeset
    97
                DUP1_RFLEXTIMER_CONNECT,
hgs
parents:
diff changeset
    98
                "RFlexTimer::Connect already connected - PANIC;this=%x", 
hgs
parents:
diff changeset
    99
                ( TUint )this );
hgs
parents:
diff changeset
   100
        
hgs
parents:
diff changeset
   101
            User::Panic( KRFlexTimerPanicCat, EFlexTimerAlreadyConnected );
hgs
parents:
diff changeset
   102
            }
hgs
parents:
diff changeset
   103
        }
hgs
parents:
diff changeset
   104
    return ret;
hgs
parents:
diff changeset
   105
    }
hgs
parents:
diff changeset
   106
hgs
parents:
diff changeset
   107
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   108
// Cancels any outstanding request to the server.
hgs
parents:
diff changeset
   109
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   110
//
hgs
parents:
diff changeset
   111
EXPORT_C void RFlexTimer::Cancel()
hgs
parents:
diff changeset
   112
    {
hgs
parents:
diff changeset
   113
    OstTrace1( TRACE_NORMAL, RFLEXTIMER_CANCEL,
hgs
parents:
diff changeset
   114
               "RFlexTimer::Cancel;this=%x", ( TUint )this );
hgs
parents:
diff changeset
   115
    
hgs
parents:
diff changeset
   116
    SendReceive( EFlexTimerServCancelRequest );
hgs
parents:
diff changeset
   117
    }
hgs
parents:
diff changeset
   118
hgs
parents:
diff changeset
   119
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   120
// Fire timer at latest on the given interval. 32-bit interval.
hgs
parents:
diff changeset
   121
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   122
//
hgs
parents:
diff changeset
   123
EXPORT_C void RFlexTimer::After( TRequestStatus& aStatus,
hgs
parents:
diff changeset
   124
                                 TTimeIntervalMicroSeconds32 aInterval )
hgs
parents:
diff changeset
   125
    {
hgs
parents:
diff changeset
   126
    OstTraceExt3( TRACE_NORMAL, RFLEXTIMER_AFTER32,
hgs
parents:
diff changeset
   127
                  "RFlexTimer::After32;this=%x;aStatus=%x;aInterval=%d",
hgs
parents:
diff changeset
   128
                  ( TUint )this, ( TUint )&( aStatus ), aInterval.Int() );
hgs
parents:
diff changeset
   129
    
hgs
parents:
diff changeset
   130
    const TTimeIntervalMicroSeconds32 ZERO_INTERVAL32(0);
hgs
parents:
diff changeset
   131
    __ASSERT_ALWAYS(aInterval >= ZERO_INTERVAL32,
hgs
parents:
diff changeset
   132
                    User::Panic(KRFlexTimerPanicCat,
hgs
parents:
diff changeset
   133
                                EFlexTimerAfterIntervalLessThanZero));
hgs
parents:
diff changeset
   134
    
hgs
parents:
diff changeset
   135
    TIpcArgs args( aInterval.Int(), 0 );
hgs
parents:
diff changeset
   136
    //asynchronous request for timer creation and timeout request.    
hgs
parents:
diff changeset
   137
    SendReceive( EFlexTimerServAfterRequest, args, aStatus );
hgs
parents:
diff changeset
   138
    }
hgs
parents:
diff changeset
   139
hgs
parents:
diff changeset
   140
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   141
// Fire timer at latest on the given interval. 64-bit interval.
hgs
parents:
diff changeset
   142
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   143
//
hgs
parents:
diff changeset
   144
EXPORT_C void RFlexTimer::After( TRequestStatus& aStatus,
hgs
parents:
diff changeset
   145
                                 TTimeIntervalMicroSeconds aInterval )
hgs
parents:
diff changeset
   146
    {
hgs
parents:
diff changeset
   147
    OstTraceExt3( TRACE_NORMAL, RFLEXTIMER_AFTER64,
hgs
parents:
diff changeset
   148
                  "RFlexTimer::After64;this=%x;aStatus=%x;aInterval=%lld",
hgs
parents:
diff changeset
   149
                  ( TUint )this, ( TUint )&( aStatus ), aInterval.Int64() );
hgs
parents:
diff changeset
   150
    
hgs
parents:
diff changeset
   151
    const TTimeIntervalMicroSeconds ZERO_INTERVAL(0);
hgs
parents:
diff changeset
   152
    __ASSERT_ALWAYS(aInterval >= ZERO_INTERVAL,
hgs
parents:
diff changeset
   153
                    User::Panic(KRFlexTimerPanicCat,
hgs
parents:
diff changeset
   154
                                EFlexTimerAfterIntervalLessThanZero));
hgs
parents:
diff changeset
   155
    
hgs
parents:
diff changeset
   156
    TIpcArgs args( I64LOW(aInterval.Int64()), I64HIGH(aInterval.Int64()) );
hgs
parents:
diff changeset
   157
    //asynchronous request for timer creation and timeout request.    
hgs
parents:
diff changeset
   158
    SendReceive( EFlexTimerServAfterRequest, args, aStatus );
hgs
parents:
diff changeset
   159
    }
hgs
parents:
diff changeset
   160
hgs
parents:
diff changeset
   161
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   162
// Fire timer at latest after the given number of ticks.
hgs
parents:
diff changeset
   163
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   164
//
hgs
parents:
diff changeset
   165
EXPORT_C void RFlexTimer::AfterTicks( TRequestStatus& aStatus, TInt aTicks )
hgs
parents:
diff changeset
   166
    {
hgs
parents:
diff changeset
   167
    OstTraceExt3( TRACE_NORMAL, RFLEXTIMER_AFTERTICKS,
hgs
parents:
diff changeset
   168
                  "RFlexTimer::AfterTicks;this=%x;aStatus=%d;aTicks=%d",
hgs
parents:
diff changeset
   169
                  ( TUint )this, ( TUint )&( aStatus ), aTicks );
hgs
parents:
diff changeset
   170
    
hgs
parents:
diff changeset
   171
    __ASSERT_ALWAYS(aTicks >= 0,
hgs
parents:
diff changeset
   172
                    User::Panic(KRFlexTimerPanicCat,
hgs
parents:
diff changeset
   173
                                EFlexTimerAfterTicksIntervalLessThanZero));
hgs
parents:
diff changeset
   174
hgs
parents:
diff changeset
   175
    TIpcArgs args( aTicks );
hgs
parents:
diff changeset
   176
    //asynchronous request for timer creation and timeout request.    
hgs
parents:
diff changeset
   177
    SendReceive( EFlexTimerServAfterTicksRequest, args, aStatus );
hgs
parents:
diff changeset
   178
    }
hgs
parents:
diff changeset
   179
hgs
parents:
diff changeset
   180
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   181
// Fire timer between at latest by the given time value.
hgs
parents:
diff changeset
   182
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   183
//
hgs
parents:
diff changeset
   184
EXPORT_C void RFlexTimer::At( TRequestStatus& aStatus, const TTime& aTime )
hgs
parents:
diff changeset
   185
    {
hgs
parents:
diff changeset
   186
    OstTraceExt3( TRACE_NORMAL, RFLEXTIMER_AT,
hgs
parents:
diff changeset
   187
                  "RFlexTimer::At;this=%x;aStatus=%d;aTime=%lld",
hgs
parents:
diff changeset
   188
                  ( TUint )this, ( TUint )&( aStatus ), aTime.Int64() );
hgs
parents:
diff changeset
   189
    
hgs
parents:
diff changeset
   190
    TTime nowTime;
hgs
parents:
diff changeset
   191
    nowTime.HomeTime();
hgs
parents:
diff changeset
   192
    
hgs
parents:
diff changeset
   193
    __ASSERT_ALWAYS(aTime >= nowTime,
hgs
parents:
diff changeset
   194
                    User::Panic(KRFlexTimerPanicCat,
hgs
parents:
diff changeset
   195
                                EFlexTimerAtIntervalLessThanZero));
hgs
parents:
diff changeset
   196
    
hgs
parents:
diff changeset
   197
    TIpcArgs args( I64LOW(aTime.Int64()), I64HIGH(aTime.Int64()) );
hgs
parents:
diff changeset
   198
    //asynchronous request for timer creation and timeout request.    
hgs
parents:
diff changeset
   199
    SendReceive( EFlexTimerServAtRequest, args, aStatus );
hgs
parents:
diff changeset
   200
    }
hgs
parents:
diff changeset
   201
hgs
parents:
diff changeset
   202
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   203
// Fire timer between at latest by the given time value.
hgs
parents:
diff changeset
   204
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   205
//
hgs
parents:
diff changeset
   206
EXPORT_C void RFlexTimer::AtUTC( TRequestStatus& aStatus, const TTime& aTime )
hgs
parents:
diff changeset
   207
    {
hgs
parents:
diff changeset
   208
    OstTraceExt3( TRACE_NORMAL, RFLEXTIMER_ATUTC,
hgs
parents:
diff changeset
   209
                  "RFlexTimer::AtUTC;this=%x;aStatus=%d;aTime=%lld",
hgs
parents:
diff changeset
   210
                  ( TUint )this, ( TUint )&( aStatus ), aTime.Int64() );
hgs
parents:
diff changeset
   211
    
hgs
parents:
diff changeset
   212
    TTime nowTime;
hgs
parents:
diff changeset
   213
    nowTime.UniversalTime();
hgs
parents:
diff changeset
   214
    
hgs
parents:
diff changeset
   215
    __ASSERT_ALWAYS(aTime >= nowTime,
hgs
parents:
diff changeset
   216
                    User::Panic(KRFlexTimerPanicCat,
hgs
parents:
diff changeset
   217
                                EFlexTimerAtUTCIntervalLessThanZero));
hgs
parents:
diff changeset
   218
    
hgs
parents:
diff changeset
   219
    TIpcArgs args( I64LOW(aTime.Int64()), I64HIGH(aTime.Int64()) );
hgs
parents:
diff changeset
   220
    //asynchronous request for timer creation and timeout request.    
hgs
parents:
diff changeset
   221
    SendReceive( EFlexTimerServAtUTCRequest, args, aStatus );
hgs
parents:
diff changeset
   222
    }
hgs
parents:
diff changeset
   223
hgs
parents:
diff changeset
   224
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   225
// Sets the window size in which alignment is possible for the timer.
hgs
parents:
diff changeset
   226
// This is a synchronous command. 32-bit function version.
hgs
parents:
diff changeset
   227
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   228
//
hgs
parents:
diff changeset
   229
EXPORT_C TInt RFlexTimer::Configure( TTimeIntervalMicroSeconds32 aWindowSize )
hgs
parents:
diff changeset
   230
    {
hgs
parents:
diff changeset
   231
    OstTraceExt2( TRACE_NORMAL, RFLEXTIMER_CONFIGURE,
hgs
parents:
diff changeset
   232
                  "RFlexTimer::Configure32;this=%x;aWindowSize=%d",
hgs
parents:
diff changeset
   233
                  ( TUint )this, aWindowSize.Int() );
hgs
parents:
diff changeset
   234
    
hgs
parents:
diff changeset
   235
    const TTimeIntervalMicroSeconds32 ZERO_INTERVAL32(0);
hgs
parents:
diff changeset
   236
    __ASSERT_ALWAYS(aWindowSize >= ZERO_INTERVAL32,
hgs
parents:
diff changeset
   237
                    User::Panic(KRFlexTimerPanicCat,
hgs
parents:
diff changeset
   238
                                EFlexTimerWindowLessThanZero));
hgs
parents:
diff changeset
   239
    
hgs
parents:
diff changeset
   240
    TInt64 transfer = MAKE_TINT64( 0, aWindowSize.Int() );
hgs
parents:
diff changeset
   241
        
hgs
parents:
diff changeset
   242
    // Regardless of the user function, the window size is always transmitted
hgs
parents:
diff changeset
   243
    // as a 64-bit integer.
hgs
parents:
diff changeset
   244
    TIpcArgs args( EConfigureRequestWindowSize, 
hgs
parents:
diff changeset
   245
                   I64LOW(transfer), I64HIGH(transfer) );
hgs
parents:
diff changeset
   246
hgs
parents:
diff changeset
   247
    return SendReceive( EFlexTimerServConfigureRequest, args );
hgs
parents:
diff changeset
   248
    }
hgs
parents:
diff changeset
   249
hgs
parents:
diff changeset
   250
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   251
// Sets the window size in which alignment is possible for the timer.
hgs
parents:
diff changeset
   252
// This is a synchronous command. 64-bit function version.
hgs
parents:
diff changeset
   253
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   254
//
hgs
parents:
diff changeset
   255
EXPORT_C TInt RFlexTimer::Configure( TTimeIntervalMicroSeconds aWindowSize )
hgs
parents:
diff changeset
   256
    {
hgs
parents:
diff changeset
   257
    OstTraceExt2( TRACE_NORMAL, DUP1_RFLEXTIMER_CONFIGURE,
hgs
parents:
diff changeset
   258
                  "RFlexTimer::Configure64;this=%x;aWindowSize=%lld",
hgs
parents:
diff changeset
   259
                  ( TUint )this, aWindowSize.Int64() );
hgs
parents:
diff changeset
   260
    
hgs
parents:
diff changeset
   261
    const TTimeIntervalMicroSeconds ZERO_INTERVAL64(0);
hgs
parents:
diff changeset
   262
    __ASSERT_ALWAYS(aWindowSize >= ZERO_INTERVAL64,
hgs
parents:
diff changeset
   263
                    User::Panic(KRFlexTimerPanicCat,
hgs
parents:
diff changeset
   264
                                EFlexTimerWindowLessThanZero));
hgs
parents:
diff changeset
   265
    
hgs
parents:
diff changeset
   266
    // Regardless of the user function, the window size is always transmitted
hgs
parents:
diff changeset
   267
    // as a 64-bit integer.
hgs
parents:
diff changeset
   268
    TIpcArgs args( EConfigureRequestWindowSize,
hgs
parents:
diff changeset
   269
                   I64LOW(aWindowSize.Int64()),
hgs
parents:
diff changeset
   270
                   I64HIGH(aWindowSize.Int64()) );
hgs
parents:
diff changeset
   271
hgs
parents:
diff changeset
   272
    return SendReceive( EFlexTimerServConfigureRequest, args  );
hgs
parents:
diff changeset
   273
    }
hgs
parents:
diff changeset
   274
hgs
parents:
diff changeset
   275
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   276
// Gets the version number.
hgs
parents:
diff changeset
   277
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   278
//
hgs
parents:
diff changeset
   279
TVersion RFlexTimer::Version() const
hgs
parents:
diff changeset
   280
    {
hgs
parents:
diff changeset
   281
    return ( TVersion( KFlexTimerServMajorVersionNumber,
hgs
parents:
diff changeset
   282
                       KFlexTimerServMinorVersionNumber,
hgs
parents:
diff changeset
   283
                       KFlexTimerServBuildVersionNumber ) );
hgs
parents:
diff changeset
   284
    }
hgs
parents:
diff changeset
   285
hgs
parents:
diff changeset
   286
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   287
// Connects to the server. If server does not exist, it is created.
hgs
parents:
diff changeset
   288
// ---------------------------------------------------------------------------
hgs
parents:
diff changeset
   289
//
hgs
parents:
diff changeset
   290
TInt RFlexTimer::StartServer()
hgs
parents:
diff changeset
   291
    {
hgs
parents:
diff changeset
   292
    OstTrace1( TRACE_NORMAL, RFLEXTIMER_STARTSERVER,
hgs
parents:
diff changeset
   293
               "RFlexTimer::StartServer;this=%x", ( TUint )this );
hgs
parents:
diff changeset
   294
    
hgs
parents:
diff changeset
   295
    TFindServer findServer( KFlexTimerServerName );
hgs
parents:
diff changeset
   296
    TFullName serverName;
hgs
parents:
diff changeset
   297
hgs
parents:
diff changeset
   298
    // See if the server is already started. 
hgs
parents:
diff changeset
   299
    TInt ret = findServer.Next( serverName );
hgs
parents:
diff changeset
   300
    
hgs
parents:
diff changeset
   301
    if ( ret != KErrNone )
hgs
parents:
diff changeset
   302
        {
hgs
parents:
diff changeset
   303
        //Server was not found so create one.
hgs
parents:
diff changeset
   304
        RProcess serverProcess;
hgs
parents:
diff changeset
   305
        TBuf<1> serverParameters;
hgs
parents:
diff changeset
   306
hgs
parents:
diff changeset
   307
        // Semaphore to guard the opening of the server process.
hgs
parents:
diff changeset
   308
        RSemaphore semaphore;
hgs
parents:
diff changeset
   309
hgs
parents:
diff changeset
   310
        // Prevent two clients from  creating the serverProcess at the same
hgs
parents:
diff changeset
   311
        // time.
hgs
parents:
diff changeset
   312
        if ( ( ret = semaphore.CreateGlobal( 
hgs
parents:
diff changeset
   313
                KFlexTimerSemaphoreName, KPassFirstWaitEntry ) ) != KErrNone )
hgs
parents:
diff changeset
   314
            {
hgs
parents:
diff changeset
   315
            ret = semaphore.OpenGlobal( KFlexTimerSemaphoreName );
hgs
parents:
diff changeset
   316
            }
hgs
parents:
diff changeset
   317
hgs
parents:
diff changeset
   318
        if ( ret == KErrNone )
hgs
parents:
diff changeset
   319
            {
hgs
parents:
diff changeset
   320
            semaphore.Wait();
hgs
parents:
diff changeset
   321
hgs
parents:
diff changeset
   322
            // See again if the server is already started. This doublechecking
hgs
parents:
diff changeset
   323
            // is necessary, since two or processes may have resolved the
hgs
parents:
diff changeset
   324
            // first findserver before any server was created.
hgs
parents:
diff changeset
   325
hgs
parents:
diff changeset
   326
            // TFindServer sets its content only when instantiated.
hgs
parents:
diff changeset
   327
            // Just using findServer.Next() does not work in here.
hgs
parents:
diff changeset
   328
            TFindServer findServerAgain( KFlexTimerServerName );
hgs
parents:
diff changeset
   329
            if ( findServerAgain.Next( serverName ) != KErrNone )
hgs
parents:
diff changeset
   330
                {
hgs
parents:
diff changeset
   331
                // Load the executable for the server.
hgs
parents:
diff changeset
   332
                ret = serverProcess.Create( KFlexTimerServerExe,
hgs
parents:
diff changeset
   333
                                            serverParameters,
hgs
parents:
diff changeset
   334
                                            EOwnerThread );
hgs
parents:
diff changeset
   335
hgs
parents:
diff changeset
   336
                if ( ret == KErrNone )
hgs
parents:
diff changeset
   337
                    {
hgs
parents:
diff changeset
   338
                    // Server has been created successfully. It is initially 
hgs
parents:
diff changeset
   339
                    // in suspended state. Now resume the server process.
hgs
parents:
diff changeset
   340
                    serverProcess.Resume();
hgs
parents:
diff changeset
   341
hgs
parents:
diff changeset
   342
                    // Wait until the server process has been started.
hgs
parents:
diff changeset
   343
                    TRequestStatus status;
hgs
parents:
diff changeset
   344
                    serverProcess.Rendezvous( status );
hgs
parents:
diff changeset
   345
                    User::WaitForRequest( status );
hgs
parents:
diff changeset
   346
hgs
parents:
diff changeset
   347
                    // Check if server has panicked during initialization.
hgs
parents:
diff changeset
   348
                    ret = serverProcess.ExitType();
hgs
parents:
diff changeset
   349
                    if ( ret == EExitPanic )
hgs
parents:
diff changeset
   350
                        {
hgs
parents:
diff changeset
   351
                        OstTrace1( TRACE_NORMAL, DUP1_RFLEXTIMER_STARTSERVER,
hgs
parents:
diff changeset
   352
                                "RFlexTimer::StartServer;this=%x; "
hgs
parents:
diff changeset
   353
                                "ERROR: Server paniced",
hgs
parents:
diff changeset
   354
                                ( TUint )this );
hgs
parents:
diff changeset
   355
hgs
parents:
diff changeset
   356
                        ret = KErrServerTerminated;
hgs
parents:
diff changeset
   357
                        }
hgs
parents:
diff changeset
   358
                    else
hgs
parents:
diff changeset
   359
                        {
hgs
parents:
diff changeset
   360
                        ret = status.Int();
hgs
parents:
diff changeset
   361
                        }
hgs
parents:
diff changeset
   362
hgs
parents:
diff changeset
   363
                    // The server process stands on its own. This handle can
hgs
parents:
diff changeset
   364
                    // be closed. 
hgs
parents:
diff changeset
   365
                    serverProcess.Close();
hgs
parents:
diff changeset
   366
                    }
hgs
parents:
diff changeset
   367
                else
hgs
parents:
diff changeset
   368
                    {
hgs
parents:
diff changeset
   369
                    OstTrace1( TRACE_NORMAL, DUP2_RFLEXTIMER_STARTSERVER,
hgs
parents:
diff changeset
   370
                            "RFlexTimer::StartServer;this=%x; "
hgs
parents:
diff changeset
   371
                            "ERROR: Server creation failed",
hgs
parents:
diff changeset
   372
                            ( TUint )this );
hgs
parents:
diff changeset
   373
                    }
hgs
parents:
diff changeset
   374
                }
hgs
parents:
diff changeset
   375
            
hgs
parents:
diff changeset
   376
            semaphore.Signal();
hgs
parents:
diff changeset
   377
            semaphore.Close();
hgs
parents:
diff changeset
   378
            }
hgs
parents:
diff changeset
   379
        }
hgs
parents:
diff changeset
   380
hgs
parents:
diff changeset
   381
    return ret;
hgs
parents:
diff changeset
   382
    }
hgs
parents:
diff changeset
   383