os/kernelhwsrv/kernel/eka/ewsrv/ws_main.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of the License "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// e32\ewsrv\ws_main.cpp
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include "ws_std.h"
sl@0
    19
#include <e32hal.h>
sl@0
    20
#include <hal.h>
sl@0
    21
#include <e32math.h>
sl@0
    22
#include <domainmanager.h>
sl@0
    23
#ifdef __WINS__
sl@0
    24
#include <e32wins.h>
sl@0
    25
#endif
sl@0
    26
sl@0
    27
GLREF_D CKeyTranslator *KeyTranslator;
sl@0
    28
GLREF_D CKeyRepeat* KeyRepeat;
sl@0
    29
sl@0
    30
// password notifier support functions
sl@0
    31
LOCAL_C void RenderPassword(RConsole *aCon, TInt aPWLeft, const TDesC& aPW);
sl@0
    32
sl@0
    33
_LIT(KShellProcessName, "ESHELL");
sl@0
    34
_LIT(KShellCommandLine, "/p");
sl@0
    35
sl@0
    36
//
sl@0
    37
// class CKeyRepeat
sl@0
    38
//
sl@0
    39
sl@0
    40
CKeyRepeat::CKeyRepeat(TInt aPriority) : CTimer(aPriority)
sl@0
    41
//
sl@0
    42
// Constructor. Set default repeat delay and rate
sl@0
    43
//
sl@0
    44
    {
sl@0
    45
    iDelay=EDefaultKeyRepeatDelay;
sl@0
    46
    iRate=EDefaultKeyRepeatRate;
sl@0
    47
    }
sl@0
    48
sl@0
    49
void CKeyRepeat::ConstructL()
sl@0
    50
    {
sl@0
    51
sl@0
    52
    CTimer::ConstructL();
sl@0
    53
    CActiveScheduler::Add(this);
sl@0
    54
    }
sl@0
    55
sl@0
    56
void CKeyRepeat::RunL()
sl@0
    57
//
sl@0
    58
// Send a repeat keypress to the window
sl@0
    59
//
sl@0
    60
    {
sl@0
    61
sl@0
    62
    After(iRate);
sl@0
    63
	CWsWindow::KeyPress(iKeyData);
sl@0
    64
    }
sl@0
    65
sl@0
    66
void CKeyRepeat::Request(TKeyData& aKeyData)
sl@0
    67
//
sl@0
    68
// Request a repeat event
sl@0
    69
//
sl@0
    70
    {
sl@0
    71
sl@0
    72
    iKeyData=aKeyData;
sl@0
    73
	Cancel();
sl@0
    74
    After(iDelay);
sl@0
    75
    }
sl@0
    76
sl@0
    77
void CKeyRepeat::SetRepeatTime(TInt aDelay,TInt aRate)
sl@0
    78
    {
sl@0
    79
sl@0
    80
    iDelay=aDelay;
sl@0
    81
    iRate=aRate;
sl@0
    82
    }
sl@0
    83
sl@0
    84
void CKeyRepeat::RepeatTime(TInt& aDelay,TInt& aRate)
sl@0
    85
    {
sl@0
    86
sl@0
    87
    aDelay=iDelay;
sl@0
    88
    aRate=iRate;
sl@0
    89
    }
sl@0
    90
sl@0
    91
//
sl@0
    92
// class CWsSession
sl@0
    93
//
sl@0
    94
sl@0
    95
CWsSession::CWsSession()
sl@0
    96
	{
sl@0
    97
	iTestFast = (UserSvr::DebugMask(2)&0x00000002) ? 1 : 0;
sl@0
    98
	}
sl@0
    99
sl@0
   100
CWsSession::~CWsSession()
sl@0
   101
//
sl@0
   102
// Destructor
sl@0
   103
//
sl@0
   104
	{
sl@0
   105
sl@0
   106
	delete iWindow;
sl@0
   107
	}
sl@0
   108
sl@0
   109
//
sl@0
   110
// class CWsServer
sl@0
   111
//
sl@0
   112
sl@0
   113
void CWsServer::New()
sl@0
   114
//
sl@0
   115
// Create a new CWsServer.
sl@0
   116
//
sl@0
   117
	{
sl@0
   118
sl@0
   119
	CWsServer *pS=new CWsServer(EPriority);
sl@0
   120
	__ASSERT_ALWAYS(pS!=NULL,Fault(ECreateServer));
sl@0
   121
	pS->SetPinClientDescriptors(EFalse); // don't pin because client interface can't cope with errors if pin fails under real or simulated OOM
sl@0
   122
	TInt r=pS->Start(KE32WindowServer);
sl@0
   123
	__ASSERT_ALWAYS(r==KErrNone,Fault(EStartServer));
sl@0
   124
	RProcess::Rendezvous(KErrNone);
sl@0
   125
sl@0
   126
	}
sl@0
   127
sl@0
   128
CWsServer::CWsServer(TInt aPriority)
sl@0
   129
//
sl@0
   130
// Constructor.
sl@0
   131
//
sl@0
   132
	: CServer2(aPriority)
sl@0
   133
	{
sl@0
   134
	}
sl@0
   135
sl@0
   136
CSession2* CWsServer::NewSessionL(const TVersion& aVersion,const RMessage2&) const
sl@0
   137
//
sl@0
   138
// Create a new client for this server.
sl@0
   139
//
sl@0
   140
	{
sl@0
   141
sl@0
   142
	TVersion v(KW32MajorVersionNumber,KW32MinorVersionNumber,KE32BuildVersionNumber);
sl@0
   143
	TBool r=User::QueryVersionSupported(v,aVersion);
sl@0
   144
	if (!r)
sl@0
   145
		User::Leave(KErrNotSupported);
sl@0
   146
	return new(ELeave) CWsSession;
sl@0
   147
	}
sl@0
   148
sl@0
   149
void CWsSession::ServiceL(const RMessage2& aMessage)
sl@0
   150
//									    
sl@0
   151
// Handle messages for this session.
sl@0
   152
//
sl@0
   153
	{
sl@0
   154
sl@0
   155
	iCurMsg = aMessage;
sl@0
   156
	CWsWindow::WaitOnService();
sl@0
   157
	CWsWindow* pW=iWindow;
sl@0
   158
	TInt r=EPrematureOperation;
sl@0
   159
	TBool delayCompletion=EFalse;
sl@0
   160
	switch (aMessage.Function())
sl@0
   161
		{
sl@0
   162
	case EConsoleCreate:
sl@0
   163
		{
sl@0
   164
		if (pW)
sl@0
   165
			{
sl@0
   166
			delete pW;
sl@0
   167
			iWindow=NULL;
sl@0
   168
			}
sl@0
   169
		pW=new CWsWindow;
sl@0
   170
		if (!pW)
sl@0
   171
			{ 
sl@0
   172
			r=EWindowOutOfMemory;
sl@0
   173
			break;
sl@0
   174
			}
sl@0
   175
		iWindow=pW;
sl@0
   176
		pW->iAllowResize=ETrue;
sl@0
   177
		pW->iIsVisible=ETrue;
sl@0
   178
		pW->iOnTop=EFalse;
sl@0
   179
		pW->SetCursorHeight(50);
sl@0
   180
		pW->iKQueue.SetOffset(_FOFF(SWsKey,iLink));
sl@0
   181
		break;
sl@0
   182
		}
sl@0
   183
	case EConsoleSet:
sl@0
   184
		{
sl@0
   185
		if (!pW)
sl@0
   186
			{
sl@0
   187
			pW=new(ELeave) CWsWindow;
sl@0
   188
			iWindow=pW;
sl@0
   189
			pW->iAllowResize=ETrue;
sl@0
   190
			pW->iIsVisible=ETrue;
sl@0
   191
			pW->iOnTop=EFalse;
sl@0
   192
			pW->SetCursorHeight(50);
sl@0
   193
			pW->iKQueue.SetOffset(_FOFF(SWsKey,iLink));
sl@0
   194
			}
sl@0
   195
		pW->iAllowSlide=ETrue;
sl@0
   196
		TFileName name;
sl@0
   197
		iCurMsg.ReadL(0, name);
sl@0
   198
		pW->iTitle=name;
sl@0
   199
		TPckgBuf<TSize> size;
sl@0
   200
		iCurMsg.ReadL(1, size);
sl@0
   201
		TRAP(r,pW->CreateL(size()));
sl@0
   202
		if (r != KErrNone)
sl@0
   203
			{
sl@0
   204
			delete pW;
sl@0
   205
			iWindow=NULL;			
sl@0
   206
			}
sl@0
   207
		break;
sl@0
   208
		}
sl@0
   209
	case EConsoleClearScreen:
sl@0
   210
		{
sl@0
   211
		if (pW)
sl@0
   212
			{
sl@0
   213
			pW->Clear();
sl@0
   214
			r=KErrNone;
sl@0
   215
			}
sl@0
   216
		break;
sl@0
   217
		}
sl@0
   218
	case EConsoleClearToEndOfLine:
sl@0
   219
		{
sl@0
   220
		if (pW)
sl@0
   221
			{
sl@0
   222
			pW->ClearToEndOfLine();
sl@0
   223
			r=KErrNone;
sl@0
   224
			}
sl@0
   225
		break;
sl@0
   226
		}
sl@0
   227
	case EConsoleSetTitle:
sl@0
   228
		{
sl@0
   229
		if (pW)
sl@0
   230
			{
sl@0
   231
			TFileName name;
sl@0
   232
			iCurMsg.ReadL(0, name);
sl@0
   233
			pW->SetTitle(name);
sl@0
   234
			r=KErrNone;
sl@0
   235
			}
sl@0
   236
		break;
sl@0
   237
		}
sl@0
   238
	case EConsoleSetSize:
sl@0
   239
		{
sl@0
   240
		if (pW)
sl@0
   241
			{
sl@0
   242
			TPckgBuf<TSize> size;
sl@0
   243
			iCurMsg.ReadL(0, size);
sl@0
   244
//			pW->SetSize(size());
sl@0
   245
//			r=KErrNone;
sl@0
   246
			r=KErrNotSupported;
sl@0
   247
			}
sl@0
   248
		break;
sl@0
   249
		}
sl@0
   250
	case EConsoleSetWindowPosAbs:
sl@0
   251
		{
sl@0
   252
		if (pW)
sl@0
   253
			{
sl@0
   254
			TPckgBuf<TPoint> point;
sl@0
   255
			iCurMsg.ReadL(0, point);
sl@0
   256
			pW->SetWindowPosAbs(point());
sl@0
   257
			r=KErrNone;
sl@0
   258
			}
sl@0
   259
		break;
sl@0
   260
		}
sl@0
   261
	case EConsoleSetCursorHeight:
sl@0
   262
		{
sl@0
   263
		if (pW)
sl@0
   264
			{
sl@0
   265
			pW->SetCursorHeight(aMessage.Int0());
sl@0
   266
			r=KErrNone;
sl@0
   267
			}
sl@0
   268
		break;
sl@0
   269
		}
sl@0
   270
	case EConsoleSetCursorPosAbs:
sl@0
   271
		{
sl@0
   272
		if (pW)
sl@0
   273
			{
sl@0
   274
			TPckgBuf<TPoint> point;
sl@0
   275
			iCurMsg.ReadL(0, point);
sl@0
   276
			pW->SetCursorPosAbs(point());
sl@0
   277
			r=KErrNone;
sl@0
   278
			}
sl@0
   279
		break;
sl@0
   280
		}
sl@0
   281
	case EConsoleSetCursorPosRel:
sl@0
   282
		{
sl@0
   283
		if (pW)
sl@0
   284
			{
sl@0
   285
			TPckgBuf<TPoint> point;
sl@0
   286
			iCurMsg.ReadL(0, point);
sl@0
   287
			pW->SetCursorPosRel(point());
sl@0
   288
			r=KErrNone;
sl@0
   289
			}
sl@0
   290
		break;
sl@0
   291
		}
sl@0
   292
	case EConsoleCursorPos:
sl@0
   293
		{
sl@0
   294
		if (pW)
sl@0
   295
			{
sl@0
   296
			TPckgBuf<TPoint> point;
sl@0
   297
			point()=pW->CursorPosition();
sl@0
   298
			aMessage.WriteL(0,point);
sl@0
   299
			r=KErrNone;
sl@0
   300
			}
sl@0
   301
		break;
sl@0
   302
		}
sl@0
   303
	case EConsoleSize:
sl@0
   304
		{
sl@0
   305
		if (pW)
sl@0
   306
			{
sl@0
   307
			TPckgBuf<TSize> size;
sl@0
   308
			size()=pW->Size();
sl@0
   309
			aMessage.WriteL(0,size);
sl@0
   310
			r=KErrNone;
sl@0
   311
			}
sl@0
   312
		break;
sl@0
   313
		}
sl@0
   314
	case EConsoleScreenSize:
sl@0
   315
		{
sl@0
   316
		if (pW)
sl@0
   317
			{
sl@0
   318
			TPckgBuf<TSize> size;
sl@0
   319
			size()=CWsWindow::ScreenSize;
sl@0
   320
			aMessage.WriteL(0,size);
sl@0
   321
			r=KErrNone;
sl@0
   322
			}
sl@0
   323
		break;
sl@0
   324
		}
sl@0
   325
	case EConsoleControl:
sl@0
   326
		{
sl@0
   327
		if (pW)
sl@0
   328
			{
sl@0
   329
			TBool indicator=ETrue;
sl@0
   330
			TInt offset=0;
sl@0
   331
			TBuf<0x100> b;
sl@0
   332
			do
sl@0
   333
				{
sl@0
   334
				iCurMsg.ReadL(0,b,offset);
sl@0
   335
				for (const TText* pB=b.Ptr();pB<b.Ptr()+b.Length();pB++)
sl@0
   336
					{
sl@0
   337
					switch(*pB)
sl@0
   338
						{
sl@0
   339
					case '+':
sl@0
   340
sl@0
   341
						indicator=ETrue;
sl@0
   342
						break;
sl@0
   343
					case '-':
sl@0
   344
						indicator=EFalse;
sl@0
   345
						break;
sl@0
   346
					case 'S':
sl@0
   347
						pW->ControlScrollBars(indicator);
sl@0
   348
						break;
sl@0
   349
					case 'W':
sl@0
   350
						pW->ControlWrapLock(indicator);
sl@0
   351
						break;
sl@0
   352
                    case 'P':
sl@0
   353
                        pW->ControlPointerEvents(indicator);
sl@0
   354
                        break;
sl@0
   355
					case 'L':
sl@0
   356
						pW->ControlScrollLock(indicator);
sl@0
   357
						break;
sl@0
   358
					case 'V':
sl@0
   359
						pW->ControlVisibility(indicator);
sl@0
   360
						break;
sl@0
   361
					case 'C':
sl@0
   362
						pW->ControlCursorRequired(indicator);
sl@0
   363
						break;
sl@0
   364
					case 'M':
sl@0
   365
						pW->ControlMaximised(indicator);
sl@0
   366
						break;
sl@0
   367
					case 'N':
sl@0
   368
						pW->ControlNewLineMode(indicator);
sl@0
   369
						break;
sl@0
   370
                    case 'O':
sl@0
   371
						pW->ControlOnTop(indicator);
sl@0
   372
						break;
sl@0
   373
					case 'I':
sl@0
   374
                        pW->ControlInformAllMouse(indicator);
sl@0
   375
                        break;
sl@0
   376
                    case 'R':
sl@0
   377
                        pW->ControlRawEventMode(indicator);
sl@0
   378
                        break;
sl@0
   379
					case 'A':
sl@0
   380
						pW->ControlAllowResize(indicator);
sl@0
   381
						}
sl@0
   382
					}
sl@0
   383
				offset+=b.Length();
sl@0
   384
				}
sl@0
   385
			while (b.Length()==b.MaxLength());
sl@0
   386
			r=KErrNone;
sl@0
   387
			}
sl@0
   388
		break;
sl@0
   389
		}
sl@0
   390
	case EConsoleWrite:
sl@0
   391
		{
sl@0
   392
		if (pW)
sl@0
   393
			{
sl@0
   394
			switch(iTestFast)
sl@0
   395
				{
sl@0
   396
			case 0:
sl@0
   397
				{
sl@0
   398
				TInt offset=0;
sl@0
   399
				TBuf<0x100> b;
sl@0
   400
				do
sl@0
   401
					{
sl@0
   402
					iCurMsg.ReadL(0,b,offset);
sl@0
   403
					pW->Write(b);
sl@0
   404
					offset+=b.Length();
sl@0
   405
					}
sl@0
   406
				while (b.Length()==b.MaxLength());
sl@0
   407
				pW->WriteDone();
sl@0
   408
				}
sl@0
   409
				r=KErrNone;
sl@0
   410
				break;
sl@0
   411
sl@0
   412
			case 1:
sl@0
   413
				{
sl@0
   414
				pW->Write(_L("Output suppressed because TESTFAST mode is set..."));
sl@0
   415
				pW->WriteDone();
sl@0
   416
				++iTestFast;
sl@0
   417
				}
sl@0
   418
				r=KErrNone;
sl@0
   419
				break;
sl@0
   420
sl@0
   421
			default:
sl@0
   422
				r=KErrNone;
sl@0
   423
				break;
sl@0
   424
				}
sl@0
   425
			}
sl@0
   426
		break;
sl@0
   427
		}
sl@0
   428
	case EConsoleRead:
sl@0
   429
		{
sl@0
   430
		if (pW)
sl@0
   431
			{
sl@0
   432
			if (pW->EnqueReadRequest(aMessage))
sl@0
   433
				{
sl@0
   434
				delayCompletion=ETrue;
sl@0
   435
				r=KErrNone;
sl@0
   436
				}
sl@0
   437
			else
sl@0
   438
				r=EDoubleReadRequest;
sl@0
   439
			}
sl@0
   440
		break;
sl@0
   441
		}
sl@0
   442
	case EConsoleReadCancel:
sl@0
   443
		{
sl@0
   444
		if (pW)
sl@0
   445
			{
sl@0
   446
			pW->DequeReadRequest();
sl@0
   447
			r=KErrNone;
sl@0
   448
			}
sl@0
   449
		break;
sl@0
   450
		}
sl@0
   451
	case EConsoleDestroy:
sl@0
   452
		{
sl@0
   453
		if (pW)
sl@0
   454
			{
sl@0
   455
			delete pW;
sl@0
   456
			iWindow=NULL;
sl@0
   457
			r=KErrNone;
sl@0
   458
			}
sl@0
   459
		break;
sl@0
   460
		}
sl@0
   461
	case EConsoleSetMode:
sl@0
   462
		{
sl@0
   463
		r=CWsWindow::SetMode((TVideoMode)aMessage.Int0());		
sl@0
   464
		break;
sl@0
   465
		}
sl@0
   466
	case EConsoleSetPaletteEntry:
sl@0
   467
		{
sl@0
   468
		CWsWindow::ScreenDriver->SetPaletteEntry((TColorIndex)aMessage.Int0(),(TUint8)aMessage.Int1(),(TUint8)aMessage.Int2(),(TUint8)aMessage.Int3());
sl@0
   469
		pW->Redraw();
sl@0
   470
		r=KErrNone;
sl@0
   471
		break;
sl@0
   472
		}
sl@0
   473
	case EConsoleGetPaletteEntry:
sl@0
   474
		{
sl@0
   475
		TUint8 r,g,b;
sl@0
   476
		TPckgBuf<TUint8> red,green,blue;
sl@0
   477
		CWsWindow::ScreenDriver->GetPaletteEntry((TColorIndex)aMessage.Int0(),r,g,b);
sl@0
   478
		red()=r; green()=g; blue()=b;
sl@0
   479
		aMessage.WriteL(1,red);
sl@0
   480
		aMessage.WriteL(2,green);
sl@0
   481
		aMessage.WriteL(3,blue);
sl@0
   482
		}
sl@0
   483
		r=KErrNone;
sl@0
   484
		break;
sl@0
   485
	case EConsoleSetTextColors:
sl@0
   486
		{
sl@0
   487
		if(pW)
sl@0
   488
			{
sl@0
   489
			pW->iFgColor=(TColorIndex)aMessage.Int0();
sl@0
   490
			pW->iBgColor=(TColorIndex)aMessage.Int1();
sl@0
   491
			}		
sl@0
   492
		r=KErrNone;
sl@0
   493
		break;
sl@0
   494
		}
sl@0
   495
	case EConsoleSetUIColors:
sl@0
   496
		{
sl@0
   497
		CWsWindow::WindowBgColor=(TColorIndex)aMessage.Int0();
sl@0
   498
		CWsWindow::BorderColor=(TColorIndex)aMessage.Int1();
sl@0
   499
		CWsWindow::ScreenColor=(TColorIndex)aMessage.Int2();
sl@0
   500
		CWsWindow::ChangeUIColors();
sl@0
   501
		r=KErrNone;
sl@0
   502
		break;
sl@0
   503
		}
sl@0
   504
	case EConsoleSetTextAttribute:
sl@0
   505
		{
sl@0
   506
		if(pW)
sl@0
   507
			pW->SetTextAttribute((TTextAttribute)aMessage.Int0());		
sl@0
   508
		r=KErrNone;
sl@0
   509
		break;
sl@0
   510
		}
sl@0
   511
	default:
sl@0
   512
		r=KErrNotSupported;
sl@0
   513
		}
sl@0
   514
	if (!delayCompletion)
sl@0
   515
		aMessage.Complete(r);
sl@0
   516
	CWsWindow::SignalService();
sl@0
   517
	}
sl@0
   518
sl@0
   519
void CWsSession::ServiceError(const RMessage2& aMessage,TInt aError)
sl@0
   520
	{
sl@0
   521
	if (!aMessage.IsNull())
sl@0
   522
		{
sl@0
   523
		if (aError>0)
sl@0
   524
			{
sl@0
   525
			aMessage.Panic(_L("WServ panic"),aError);
sl@0
   526
			}
sl@0
   527
		else
sl@0
   528
			{
sl@0
   529
			aMessage.Complete(aError);
sl@0
   530
			}
sl@0
   531
		}
sl@0
   532
	}
sl@0
   533
sl@0
   534
CWsServer::~CWsServer()
sl@0
   535
//
sl@0
   536
// Destructor
sl@0
   537
//
sl@0
   538
	{
sl@0
   539
	}
sl@0
   540
sl@0
   541
//
sl@0
   542
// class CEvent
sl@0
   543
//
sl@0
   544
sl@0
   545
void CEvent::New()
sl@0
   546
//
sl@0
   547
// Create the CEvent active object.
sl@0
   548
//
sl@0
   549
	{
sl@0
   550
sl@0
   551
	CEvent *pE=new CEvent(EPriority);
sl@0
   552
	__ASSERT_ALWAYS(pE!=NULL,Fault(ECreateEvent));
sl@0
   553
	pE->CaptureKeys=new CCaptureKeys();
sl@0
   554
	__ASSERT_ALWAYS(pE->CaptureKeys!=NULL,Fault(ECreateEvent));
sl@0
   555
sl@0
   556
	pE->CaptureKeys->Construct();
sl@0
   557
sl@0
   558
	CActiveScheduler::Add(pE);
sl@0
   559
	UserSvr::CaptureEventHook();
sl@0
   560
	pE->Request();
sl@0
   561
	}
sl@0
   562
sl@0
   563
CEvent::~CEvent()
sl@0
   564
//
sl@0
   565
// Destroy the CEvent active object
sl@0
   566
//
sl@0
   567
	{
sl@0
   568
sl@0
   569
	Cancel();
sl@0
   570
	}
sl@0
   571
sl@0
   572
#pragma warning( disable : 4705 )	// statement has no effect
sl@0
   573
CEvent::CEvent(TInt aPriority)
sl@0
   574
//
sl@0
   575
// Constructor
sl@0
   576
//
sl@0
   577
	: CActive(aPriority)
sl@0
   578
	{
sl@0
   579
	}
sl@0
   580
#pragma warning( default : 4705 )
sl@0
   581
sl@0
   582
void CEvent::Request()
sl@0
   583
//
sl@0
   584
// Issue a request for the next event.
sl@0
   585
//
sl@0
   586
	{
sl@0
   587
sl@0
   588
	UserSvr::RequestEvent(iEvent,iStatus);
sl@0
   589
	SetActive();
sl@0
   590
	}
sl@0
   591
sl@0
   592
void CEvent::DoCancel()
sl@0
   593
//
sl@0
   594
// Cancel a pending event.
sl@0
   595
//
sl@0
   596
	{
sl@0
   597
sl@0
   598
	UserSvr::RequestEventCancel();
sl@0
   599
	}
sl@0
   600
sl@0
   601
void CEvent::RunL()
sl@0
   602
//
sl@0
   603
// Event has completed.
sl@0
   604
//
sl@0
   605
	{
sl@0
   606
sl@0
   607
    if (CWsWindow::RawEventMode())
sl@0
   608
        {
sl@0
   609
        KeyRepeat->Cancel();
sl@0
   610
        CWsWindow::QueueRawEvent(iEvent.Event());
sl@0
   611
        Request();
sl@0
   612
        return;
sl@0
   613
        }
sl@0
   614
	switch(iEvent.Event().Type())
sl@0
   615
		{
sl@0
   616
	case TRawEvent::ERedraw:
sl@0
   617
		CWsWindow::Redraw();
sl@0
   618
		break;
sl@0
   619
	case TRawEvent::EButton1Down:
sl@0
   620
        if(!CWsWindow::MouseIsCaptured)
sl@0
   621
            {
sl@0
   622
      		CWsWindow::MouseMove(iEvent.Event().Pos());
sl@0
   623
  	    	CWsWindow::MouseLeftButton();
sl@0
   624
            }
sl@0
   625
        else
sl@0
   626
            CWsWindow::InformTopMouse(iEvent.Event().Pos());
sl@0
   627
		break;
sl@0
   628
	case TRawEvent::EButton1Up:
sl@0
   629
		if(!CWsWindow::MouseIsCaptured)
sl@0
   630
			{
sl@0
   631
			CWsWindow::MouseMove(iEvent.Event().Pos());
sl@0
   632
			CWsWindow::MouseLeftButtonUp();
sl@0
   633
			}
sl@0
   634
		break;
sl@0
   635
	case TRawEvent::EButton2Down:
sl@0
   636
		break;
sl@0
   637
	case TRawEvent::EButton2Up:
sl@0
   638
		break;
sl@0
   639
	case TRawEvent::EButton3Down:
sl@0
   640
		break;
sl@0
   641
	case TRawEvent::EButton3Up:
sl@0
   642
		break;
sl@0
   643
	case TRawEvent::EPointerMove:
sl@0
   644
		CWsWindow::MouseMove(iEvent.Event().Pos());
sl@0
   645
		break;
sl@0
   646
	case TRawEvent::EInactive:
sl@0
   647
        KeyRepeat->Cancel();
sl@0
   648
        break;
sl@0
   649
    case TRawEvent::EActive:
sl@0
   650
        break;
sl@0
   651
    case TRawEvent::EUpdateModifiers:
sl@0
   652
        KeyTranslator->UpdateModifiers(iEvent.Event().Modifiers());
sl@0
   653
        break;
sl@0
   654
	case TRawEvent::EKeyDown:
sl@0
   655
		{
sl@0
   656
		TKeyData keyData;
sl@0
   657
		if (KeyTranslator->TranslateKey(iEvent.Event().ScanCode(), EFalse,*CaptureKeys,keyData))
sl@0
   658
			CWsWindow::KeyPress(keyData);
sl@0
   659
		if (keyData.iModifiers&EModifierAutorepeatable)
sl@0
   660
			KeyRepeat->Request(keyData);
sl@0
   661
		break;
sl@0
   662
		}
sl@0
   663
	case TRawEvent::EKeyUp:
sl@0
   664
		{
sl@0
   665
		TKeyData keyData;
sl@0
   666
		KeyRepeat->Cancel();
sl@0
   667
		if (KeyTranslator->TranslateKey(iEvent.Event().ScanCode(), ETrue,*CaptureKeys,keyData))
sl@0
   668
			CWsWindow::KeyPress(keyData);
sl@0
   669
		break;
sl@0
   670
		}
sl@0
   671
	case TRawEvent::ESwitchOn:
sl@0
   672
	case TRawEvent::ECaseOpen:
sl@0
   673
		HAL::Set(HAL::EDisplayState, 1);
sl@0
   674
			{
sl@0
   675
			RDmDomainManager mgr; 
sl@0
   676
			TInt r = mgr.Connect();
sl@0
   677
			if (r != KErrNone)
sl@0
   678
				User::Panic(_L("EWSRV SwitchOn0"), r);
sl@0
   679
			TRequestStatus status;
sl@0
   680
			mgr.RequestDomainTransition(KDmIdUiApps, EPwActive, status);
sl@0
   681
			User::WaitForRequest(status);
sl@0
   682
			if (status.Int() != KErrNone)
sl@0
   683
				User::Panic(_L("EWSRV SwitchOn1"), status.Int());
sl@0
   684
			mgr.Close();
sl@0
   685
			}
sl@0
   686
		break;
sl@0
   687
	case TRawEvent::EPointerSwitchOn:
sl@0
   688
#if defined(_DEBUG)
sl@0
   689
    	User::Beep(440,250000);
sl@0
   690
#endif
sl@0
   691
		break;
sl@0
   692
	case TRawEvent::ESwitchOff:
sl@0
   693
			{
sl@0
   694
			RDmDomainManager mgr; 
sl@0
   695
			TInt r = mgr.Connect();
sl@0
   696
			if (r != KErrNone)
sl@0
   697
				User::Panic(_L("EWSRV SwitchOff0"), r);
sl@0
   698
			TRequestStatus status;
sl@0
   699
			mgr.RequestSystemTransition(EPwStandby, status);
sl@0
   700
			User::WaitForRequest(status);
sl@0
   701
			if (status.Int() != KErrNone)
sl@0
   702
				User::Panic(_L("EWSRV SwitchOff1"), status.Int());
sl@0
   703
			mgr.Close();
sl@0
   704
			}
sl@0
   705
		break;
sl@0
   706
	case TRawEvent::ECaseClose:
sl@0
   707
			{
sl@0
   708
			RDmDomainManager mgr; 
sl@0
   709
			TInt r = mgr.Connect();
sl@0
   710
			if (r != KErrNone)
sl@0
   711
				User::Panic(_L("EWSRV CaseClosed"), r);
sl@0
   712
			TRequestStatus status;
sl@0
   713
			mgr.RequestDomainTransition(KDmIdUiApps, EPwStandby, status);
sl@0
   714
			User::WaitForRequest(status);
sl@0
   715
			if (status.Int() != KErrNone)
sl@0
   716
				User::Panic(_L("EWSRV CaseClosed1"), status.Int());
sl@0
   717
			mgr.Close();
sl@0
   718
			}
sl@0
   719
		HAL::Set(HAL::EDisplayState, 0);
sl@0
   720
		break;
sl@0
   721
	case TRawEvent::ENone:
sl@0
   722
		break;
sl@0
   723
	default:
sl@0
   724
		break;
sl@0
   725
		}                                       
sl@0
   726
	Request();
sl@0
   727
	}
sl@0
   728
sl@0
   729
LOCAL_C void RenderPassword(RConsole *aCon, TInt aPWLeft, const TDesC& aPW)
sl@0
   730
// pre:		aCon points to a console being used to read a password.
sl@0
   731
//			aPWLeft is the column number from which the left brace should be drawn.
sl@0
   732
//			aPasswd is a valid password.
sl@0
   733
// post:	the password is rendered onto the console and followed by a '#' and
sl@0
   734
//			padding spaces.  Everything is enclosed in a pair of square brackets.
sl@0
   735
	{
sl@0
   736
	aCon->SetCursorPosAbs(TPoint(aPWLeft, 3));
sl@0
   737
	aCon->Write(_L("["));
sl@0
   738
	aCon->Write(aPW);
sl@0
   739
	aCon->Write(_L("#"));
sl@0
   740
sl@0
   741
	TInt i;
sl@0
   742
	for (i = 0; i < KMaxMediaPassword - aPW.Length(); i++)
sl@0
   743
		{
sl@0
   744
		aCon->Write(_L(" "));
sl@0
   745
		}
sl@0
   746
sl@0
   747
	aCon->Write(_L("]"));
sl@0
   748
	}
sl@0
   749
sl@0
   750
sl@0
   751
void CNotifierSession::RunPasswordWindowL(const RMessage2 &aMsg)
sl@0
   752
//
sl@0
   753
// Eight unicode chars are mapped to (up to) sixteen bytes.  Remainder of array is
sl@0
   754
// zeroed.  Message is completed in CNotifierSession::ServiceL().
sl@0
   755
// 
sl@0
   756
	{
sl@0
   757
	// local copies of dialog data, 5 * (8 + 32 * 2) bytes
sl@0
   758
	TBuf<0x20> line1, line2, unlockBtn, storeBtn, cancelBtn;
sl@0
   759
	line1.Copy(_L("Password notifier"));
sl@0
   760
	line2.Copy(_L("Enter password"));
sl@0
   761
	unlockBtn.Copy(_L("Unlock"));
sl@0
   762
	storeBtn.Copy(_L("Store"));
sl@0
   763
	cancelBtn.Copy(_L("Cancel"));
sl@0
   764
sl@0
   765
	TPckgBuf<TMediaPswdReplyNotifyInfoV1> reply;
sl@0
   766
sl@0
   767
	// Format the console window.
sl@0
   768
sl@0
   769
	const TInt KPasswordBarLen(1 + KMaxMediaPassword + 1 + 1);
sl@0
   770
	const TInt KButtonBarLen(
sl@0
   771
				1 + unlockBtn.Length() + 1
sl@0
   772
		+ 1 +	1 + cancelBtn.Length() + 1
sl@0
   773
		+ 1 +	1 + storeBtn.Length() + 1);
sl@0
   774
sl@0
   775
	// Work out width of window.
sl@0
   776
	// (Buttons are enclosed by angle brackets and separted by a space.)
sl@0
   777
	// (Password is followed by '#' and delimited by square brackets.)
sl@0
   778
	// If KButtonBarLen is at least as long as any other line then it will write
sl@0
   779
	// to the bottom right corner and cause the console to scroll.  To counter
sl@0
   780
	// this, an extra padding character is added if necessary.
sl@0
   781
sl@0
   782
	TInt width;
sl@0
   783
	width = Max(line1.Length(), line2.Length());
sl@0
   784
	width = Max(width, KPasswordBarLen);
sl@0
   785
	width = KButtonBarLen >= width ? KButtonBarLen + 1: width;
sl@0
   786
sl@0
   787
	// Create the window and render its contents.
sl@0
   788
sl@0
   789
	RConsole con;
sl@0
   790
	con.Create();
sl@0
   791
	con.Control(_L("-Visible"));
sl@0
   792
	TInt r = con.Init(_L(""), TSize(width, 2 + 1 + 1 + 1 + 1));
sl@0
   793
	if (KErrNone != r)
sl@0
   794
		{
sl@0
   795
		PanicClient(aMsg, ENotifierPanicPasswordWindow);
sl@0
   796
		User::Leave(KErrGeneral);
sl@0
   797
		}
sl@0
   798
	con.Control(_L("+Max -Cursor -AllowResize +OnTop"));
sl@0
   799
sl@0
   800
	con.SetCursorPosAbs(TPoint(0, 0));
sl@0
   801
	con.Write(line1);
sl@0
   802
	con.SetCursorPosAbs(TPoint(0, 1));
sl@0
   803
	con.Write(line2);
sl@0
   804
	const TInt KPasswordLeft((width - KPasswordBarLen) / 2);
sl@0
   805
	con.SetCursorPosAbs(TPoint(KPasswordLeft, 3));
sl@0
   806
	con.Write(_L("[#                ]"));
sl@0
   807
	con.SetCursorPosAbs(TPoint((width - KButtonBarLen) / 2, 5));
sl@0
   808
	con.Write(_L("<"));
sl@0
   809
	con.Write(unlockBtn);
sl@0
   810
	con.Write(_L("> <"));
sl@0
   811
	con.Write(storeBtn);
sl@0
   812
	con.Write(_L("> <"));
sl@0
   813
	con.Write(cancelBtn);
sl@0
   814
	con.Write(_L(">"));
sl@0
   815
sl@0
   816
	// Allow the user to edit the password until they either press enter or escape.
sl@0
   817
sl@0
   818
	TBool done(EFalse);
sl@0
   819
	TBuf<KMaxMediaPassword> pw;
sl@0
   820
	pw.Zero();
sl@0
   821
	TMediaPswdNotifyExitMode em(EMPEMUnlock);	// avoid VC warning C4701 (used w/o init).
sl@0
   822
sl@0
   823
	const TInt sendInfoLen = User::LeaveIfError(aMsg.GetDesLength(1));
sl@0
   824
sl@0
   825
	if (sendInfoLen == sizeof(TMediaPswdSendNotifyInfoV1Debug))
sl@0
   826
		{
sl@0
   827
		// debug mode - wait for specified period and close notifier
sl@0
   828
sl@0
   829
		TPckgBuf<TMediaPswdSendNotifyInfoV1Debug> sendDbg;
sl@0
   830
		aMsg.ReadL(1, sendDbg);
sl@0
   831
		if (sendDbg().iSleepPeriod >= 0)
sl@0
   832
			User::After(sendDbg().iSleepPeriod);
sl@0
   833
		else
sl@0
   834
			{
sl@0
   835
			TTime now;
sl@0
   836
			now.HomeTime();
sl@0
   837
			TInt64 seed = now.Int64();
sl@0
   838
			User::After(Math::Rand(seed) % -(sendDbg().iSleepPeriod));
sl@0
   839
			}
sl@0
   840
sl@0
   841
		reply().iEM = sendDbg().iEM;
sl@0
   842
		Mem::Copy(reply().iPW, sendDbg().iPW, KMaxMediaPassword);
sl@0
   843
		}
sl@0
   844
	else
sl@0
   845
		{
sl@0
   846
		RenderPassword(&con, KPasswordLeft, pw);
sl@0
   847
		do
sl@0
   848
			{
sl@0
   849
			TConsoleKey key;
sl@0
   850
sl@0
   851
			con.Read(key);
sl@0
   852
			TInt keyCode = key.Code();
sl@0
   853
sl@0
   854
			switch (keyCode)
sl@0
   855
				{
sl@0
   856
				case EKeyEscape:
sl@0
   857
					em = EMPEMCancel;
sl@0
   858
					done = ETrue;
sl@0
   859
					break;
sl@0
   860
sl@0
   861
				case EKeyEnter:
sl@0
   862
					em = EMPEMUnlock;
sl@0
   863
					done = ETrue;
sl@0
   864
					break;
sl@0
   865
				
sl@0
   866
				case EKeySpace:
sl@0
   867
					em = EMPEMUnlockAndStore;
sl@0
   868
					done = ETrue;
sl@0
   869
					break;
sl@0
   870
				
sl@0
   871
				case EKeyBackspace:
sl@0
   872
					if (pw.Length() > 0)
sl@0
   873
						{
sl@0
   874
						pw.SetLength(pw.Length() - 1);
sl@0
   875
						RenderPassword(&con, KPasswordLeft, pw);
sl@0
   876
						}
sl@0
   877
					break;
sl@0
   878
sl@0
   879
				default:							// interpret other keys as pw contents
sl@0
   880
					TChar ch(keyCode);
sl@0
   881
					// unicode encoding, so number of password characters is half byte length
sl@0
   882
					if (ch.IsPrint() && pw.Length() < KMaxMediaPassword / 2)
sl@0
   883
						{
sl@0
   884
						pw.Append(ch);
sl@0
   885
						RenderPassword(&con, KPasswordLeft, pw);
sl@0
   886
						}
sl@0
   887
					break;
sl@0
   888
				}
sl@0
   889
			} while (! done);
sl@0
   890
sl@0
   891
		// Fill the TMediaPswdReplyNotifyInfoV1 structure.
sl@0
   892
sl@0
   893
		if (em == EMPEMUnlock || em == EMPEMUnlockAndStore)
sl@0
   894
			{
sl@0
   895
			const TInt byteLen = pw.Length() * 2;
sl@0
   896
sl@0
   897
			// zero entire array; and then copy in valid section of TMediaPassword,
sl@0
   898
			// not converting Unicode to ASCII.
sl@0
   899
			TPtr8 pt8(reply().iPW, KMaxMediaPassword);	// length = 0
sl@0
   900
			pt8.FillZ(KMaxMediaPassword);				// length = KMaxMediaPassword
sl@0
   901
			pt8.Zero();									// length = 0
sl@0
   902
														// length = byteLen
sl@0
   903
			pt8.Copy(reinterpret_cast<const TUint8 *>(pw.Ptr()), byteLen);
sl@0
   904
			}
sl@0
   905
		
sl@0
   906
		// Set exit mode to tell user how dialog handled.
sl@0
   907
sl@0
   908
		reply().iEM = em;
sl@0
   909
		}	// else (sendInfoLen == sizeof(TMediaPswdSendNotifyInfoV1Debug))
sl@0
   910
	
sl@0
   911
	con.Destroy();
sl@0
   912
sl@0
   913
	aMsg.WriteL(2, reply);
sl@0
   914
	}
sl@0
   915
sl@0
   916
//
sl@0
   917
// class MNotifierBase2
sl@0
   918
//
sl@0
   919
sl@0
   920
void MNotifierBase2::SetManager(MNotifierManager* aManager)
sl@0
   921
	{
sl@0
   922
	iManager=aManager;
sl@0
   923
	}
sl@0
   924
sl@0
   925
//
sl@0
   926
// class CNotifierServer
sl@0
   927
//
sl@0
   928
sl@0
   929
_LIT(__NOTIFIER_SERVER,"TextNotifierSrvr");
sl@0
   930
sl@0
   931
CNotifierServer* CNotifierServer::NewL()
sl@0
   932
	{
sl@0
   933
	CNotifierServer* server=new(ELeave) CNotifierServer(200);
sl@0
   934
	CleanupStack::PushL(server);
sl@0
   935
	server->ConstructL();
sl@0
   936
	server->StartL(__NOTIFIER_NAME);
sl@0
   937
	CleanupStack::Pop(); // server
sl@0
   938
	return server;
sl@0
   939
	}
sl@0
   940
sl@0
   941
CNotifierServer::~CNotifierServer()
sl@0
   942
	{
sl@0
   943
	SetIsExiting();
sl@0
   944
	delete iManager;
sl@0
   945
	}
sl@0
   946
sl@0
   947
void CNotifierServer::SetIsExiting()
sl@0
   948
	{
sl@0
   949
	iExiting=ETrue;
sl@0
   950
	}
sl@0
   951
sl@0
   952
TBool CNotifierServer::IsExiting() const
sl@0
   953
	{
sl@0
   954
	return iExiting;
sl@0
   955
	}
sl@0
   956
sl@0
   957
CNotifierServer::CNotifierServer(TInt aPriority)
sl@0
   958
	: CServer2(aPriority)
sl@0
   959
	{}
sl@0
   960
sl@0
   961
void CNotifierServer::ConstructL()
sl@0
   962
	{
sl@0
   963
	iManager=CNotifierManager::NewL();
sl@0
   964
	RFs fs;
sl@0
   965
	User::LeaveIfError(fs.Connect());
sl@0
   966
	CleanupClosePushL(fs);
sl@0
   967
	iManager->RegisterL(fs);
sl@0
   968
	CleanupStack::PopAndDestroy(); // fs.Close()
sl@0
   969
	}
sl@0
   970
sl@0
   971
CSession2* CNotifierServer::NewSessionL(const TVersion &aVersion,const RMessage2&) const
sl@0
   972
	{
sl@0
   973
	TVersion v(1,0,0); // !! liaise with E32
sl@0
   974
	if (!User::QueryVersionSupported(v,aVersion))
sl@0
   975
		User::Leave(KErrNotSupported);
sl@0
   976
	return new(ELeave) CNotifierSession(*this);
sl@0
   977
	}
sl@0
   978
sl@0
   979
//
sl@0
   980
// class CNotifierSession
sl@0
   981
//
sl@0
   982
sl@0
   983
CNotifierSession::CNotifierSession(const CNotifierServer& aServer)
sl@0
   984
	{
sl@0
   985
	iServer=&aServer;
sl@0
   986
	iClientId=(TInt)this;
sl@0
   987
	}
sl@0
   988
sl@0
   989
CNotifierSession::~CNotifierSession()
sl@0
   990
	{
sl@0
   991
	const CNotifierServer* server=static_cast<const CNotifierServer*>(Server());
sl@0
   992
	if (!server->IsExiting())
sl@0
   993
		{
sl@0
   994
		server->Manager()->HandleClientExit(iClientId);
sl@0
   995
		}
sl@0
   996
	}
sl@0
   997
sl@0
   998
void CNotifierSession::ServiceL(const RMessage2 &aMessage)
sl@0
   999
	{
sl@0
  1000
	TBool completeMessage=ETrue;
sl@0
  1001
	switch (aMessage.Function())
sl@0
  1002
		{
sl@0
  1003
	case ENotifierNotify:
sl@0
  1004
		DisplayAlertL(aMessage);
sl@0
  1005
		break;
sl@0
  1006
	case ENotifierNotifyCancel:
sl@0
  1007
		// do nothing - this server doesn't support cancelling RNotifier::Notify - the client will just have to wait
sl@0
  1008
		break;
sl@0
  1009
	case ENotifierInfoPrint:
sl@0
  1010
		DisplayInfoMsgL(aMessage);
sl@0
  1011
		break;
sl@0
  1012
	case EStartNotifier:
sl@0
  1013
		DoStartNotifierL(aMessage);
sl@0
  1014
		break;
sl@0
  1015
	case ECancelNotifier:
sl@0
  1016
		iServer->Manager()->NotifierCancel(TUid::Uid(aMessage.Int0()));
sl@0
  1017
		break;
sl@0
  1018
	case EUpdateNotifierAndGetResponse:
sl@0
  1019
	case EUpdateNotifier:
sl@0
  1020
		DoUpdateNotifierL(aMessage);
sl@0
  1021
		break;
sl@0
  1022
	case EStartNotifierAndGetResponse:
sl@0
  1023
		{
sl@0
  1024
		if (aMessage.Int0()==KMediaPasswordNotifyUid)
sl@0
  1025
			{
sl@0
  1026
			RunPasswordWindowL(aMessage);
sl@0
  1027
			}
sl@0
  1028
		else
sl@0
  1029
			{
sl@0
  1030
			TBool cleanupComplete=ETrue;
sl@0
  1031
			StartNotifierAndGetResponseL(aMessage,cleanupComplete);
sl@0
  1032
			 // if the plug-in starts successfully, it has
sl@0
  1033
			 // responsibility for completing the message (either
sl@0
  1034
			 // synchronously or asynchronously)
sl@0
  1035
			completeMessage=EFalse;
sl@0
  1036
			}
sl@0
  1037
		}
sl@0
  1038
		break;
sl@0
  1039
	default:
sl@0
  1040
		aMessage.Complete(KErrNotSupported);
sl@0
  1041
		break;
sl@0
  1042
		}
sl@0
  1043
	if (completeMessage && !aMessage.IsNull())
sl@0
  1044
		{
sl@0
  1045
		aMessage.Complete(KErrNone);
sl@0
  1046
		}
sl@0
  1047
	}
sl@0
  1048
sl@0
  1049
void CNotifierSession::DisplayAlertL(const RMessage2& aMessage)
sl@0
  1050
	{
sl@0
  1051
	const TInt lengthOfCombinedBuffer=User::LeaveIfError(aMessage.GetDesLength(1));
sl@0
  1052
	const TInt lengthOfLine1=(STATIC_CAST(TUint,aMessage.Int2())>>16);
sl@0
  1053
	const TInt lengthOfLine2=(aMessage.Int2()&KMaxTUint16);
sl@0
  1054
	const TInt lengthOfBut1=(STATIC_CAST(TUint,aMessage.Int3())>>16);
sl@0
  1055
	const TInt lengthOfBut2=(aMessage.Int3()&KMaxTUint16);
sl@0
  1056
	if (lengthOfCombinedBuffer!=lengthOfLine1+lengthOfLine2+lengthOfBut1+lengthOfBut2)
sl@0
  1057
		{
sl@0
  1058
		PanicClient(aMessage,ENotifierPanicInconsistentDescriptorLengths);
sl@0
  1059
		return;
sl@0
  1060
		}
sl@0
  1061
	HBufC* const combinedBuffer=HBufC::NewLC(lengthOfCombinedBuffer);
sl@0
  1062
	{TPtr combinedBuffer_asWritable(combinedBuffer->Des());
sl@0
  1063
	aMessage.ReadL(1,combinedBuffer_asWritable);}
sl@0
  1064
	const TPtrC line1(combinedBuffer->Left(lengthOfLine1));
sl@0
  1065
	const TPtrC line2(combinedBuffer->Mid(lengthOfLine1,lengthOfLine2));
sl@0
  1066
	const TPtrC but1(combinedBuffer->Mid(lengthOfLine1+lengthOfLine2,lengthOfBut1));
sl@0
  1067
	const TPtrC but2(combinedBuffer->Mid(lengthOfLine1+lengthOfLine2+lengthOfBut1,lengthOfBut2));
sl@0
  1068
	TInt buttons, len, offset;
sl@0
  1069
	if (lengthOfBut2==0)
sl@0
  1070
		{
sl@0
  1071
		buttons=1;
sl@0
  1072
		len=lengthOfBut1+2;
sl@0
  1073
		}
sl@0
  1074
	else
sl@0
  1075
		{
sl@0
  1076
		buttons=2;
sl@0
  1077
		len=lengthOfBut1+lengthOfBut2+5;
sl@0
  1078
		}
sl@0
  1079
	if (lengthOfLine1>len)
sl@0
  1080
		len=lengthOfLine1;
sl@0
  1081
	if (lengthOfLine2>len)
sl@0
  1082
		len=lengthOfLine2;
sl@0
  1083
	RConsole con;
sl@0
  1084
	con.Create();
sl@0
  1085
	TSize scsz;
sl@0
  1086
	con.ScreenSize(scsz);
sl@0
  1087
	con.Control(_L("-Visible"));
sl@0
  1088
	TInt ww=Min(len,scsz.iWidth-4);
sl@0
  1089
	TInt wh=3;
sl@0
  1090
	if ((lengthOfBut1+lengthOfBut2+5)>ww)
sl@0
  1091
		wh++;
sl@0
  1092
	if (lengthOfLine1>ww)
sl@0
  1093
		wh++;
sl@0
  1094
	if (lengthOfLine2>ww)
sl@0
  1095
		wh++;
sl@0
  1096
	con.Init(_L(""),TSize(ww,wh));
sl@0
  1097
	con.Control(_L("+Max -Cursor -Allowresize +Ontop +Wrap"));
sl@0
  1098
	con.Write(line1);
sl@0
  1099
	con.SetCursorPosAbs(TPoint(0,1));
sl@0
  1100
	con.Write(line2);
sl@0
  1101
	if (buttons==2)
sl@0
  1102
		offset=(len-lengthOfBut1-lengthOfBut2-5)/2;
sl@0
  1103
	else
sl@0
  1104
		offset=(len-lengthOfBut1-2)/2;
sl@0
  1105
	con.SetCursorPosAbs(TPoint(offset,2));
sl@0
  1106
	con.Write(_L("<"));
sl@0
  1107
	con.Write(but1);
sl@0
  1108
	con.Write(_L(">"));
sl@0
  1109
	if(buttons==2)
sl@0
  1110
		{
sl@0
  1111
		con.Write(_L(" <"));
sl@0
  1112
		con.Write(but2);
sl@0
  1113
		con.Write(_L(">"));
sl@0
  1114
		}
sl@0
  1115
	
sl@0
  1116
	TConsoleKey key;
sl@0
  1117
	TInt keycode;
sl@0
  1118
	do
sl@0
  1119
		{
sl@0
  1120
		con.Read(key);
sl@0
  1121
		keycode=key.Code();
sl@0
  1122
		}
sl@0
  1123
	while((keycode!=EKeyEscape&&keycode!=EKeyEnter&&buttons==2)||(keycode!=EKeyEnter&&buttons==1));
sl@0
  1124
	if(keycode==EKeyEscape)
sl@0
  1125
		keycode=0;
sl@0
  1126
	else
sl@0
  1127
		keycode=1;
sl@0
  1128
	con.Destroy();
sl@0
  1129
	aMessage.WriteL(0,TPckgC<TInt>(keycode));
sl@0
  1130
	CleanupStack::PopAndDestroy(combinedBuffer);
sl@0
  1131
	}
sl@0
  1132
sl@0
  1133
TInt CNotifierSession::InfoPrintThread(TAny* aMessage)
sl@0
  1134
	{
sl@0
  1135
	TBuf<0x50> des; // 0x50 max size of message
sl@0
  1136
	RConsole con;
sl@0
  1137
	des=*(TBuf<0x50> *)aMessage;
sl@0
  1138
	TInt l=des.Length();
sl@0
  1139
	NotifierSemaphore.Signal();
sl@0
  1140
	con.Create();
sl@0
  1141
	con.Control(_L("-Visible"));
sl@0
  1142
	TSize size;
sl@0
  1143
	con.ScreenSize(size);
sl@0
  1144
	TInt ww=Min(size.iWidth-6,l);
sl@0
  1145
	TInt wh=(l+ww-1)/ww;
sl@0
  1146
	if (wh==0)
sl@0
  1147
		wh=1;
sl@0
  1148
	con.Init(_L(""),TSize(ww,wh));
sl@0
  1149
	con.Control(_L("+Maximise"));
sl@0
  1150
	con.SetWindowPosAbs(TPoint(size.iWidth-ww-4,1));
sl@0
  1151
	con.Control(_L("+Wrap +Ontop"));
sl@0
  1152
	con.Write(des);
sl@0
  1153
sl@0
  1154
	User::After(1300000);
sl@0
  1155
	con.Destroy();
sl@0
  1156
	return KErrNone;
sl@0
  1157
	}
sl@0
  1158
sl@0
  1159
void CNotifierSession::DisplayInfoMsgL(const RMessage2& aMessage)
sl@0
  1160
	{
sl@0
  1161
	TInt r;
sl@0
  1162
	TBuf<0x50> des; // 0x50 max size of message lines
sl@0
  1163
	aMessage.ReadL(0,des);
sl@0
  1164
	RThread thread;
sl@0
  1165
	do
sl@0
  1166
		{
sl@0
  1167
		r=thread.Create(_L("Info Window"),InfoPrintThread,KDefaultStackSize,&User::Allocator(),(TAny*)&des);
sl@0
  1168
		if(r==KErrAlreadyExists)
sl@0
  1169
			User::After(200000);
sl@0
  1170
		} while(r==KErrAlreadyExists);
sl@0
  1171
		User::LeaveIfError(r);
sl@0
  1172
	thread.Resume();
sl@0
  1173
	NotifierSemaphore.Wait();
sl@0
  1174
	thread.Close();
sl@0
  1175
	}
sl@0
  1176
sl@0
  1177
void CNotifierSession::DoStartNotifierL(const RMessage2& aMessage)
sl@0
  1178
	{
sl@0
  1179
	const TUid targetUid={aMessage.Int0()};
sl@0
  1180
	HBufC8* const inputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesLength(1)));
sl@0
  1181
	{TPtr8 input(inputBuffer->Des());
sl@0
  1182
	aMessage.ReadL(1,input);}
sl@0
  1183
	TPtrC8 output(0,0);
sl@0
  1184
	iServer->Manager()->NotifierStartL(targetUid,*inputBuffer,&output,iClientId);
sl@0
  1185
	if(aMessage.Int2())
sl@0
  1186
		aMessage.WriteL(2,output);
sl@0
  1187
	CleanupStack::PopAndDestroy(inputBuffer);
sl@0
  1188
	}
sl@0
  1189
sl@0
  1190
void CNotifierSession::DoUpdateNotifierL(const RMessage2& aMessage)
sl@0
  1191
	{
sl@0
  1192
	const TUid targetUid={aMessage.Int0()};
sl@0
  1193
	HBufC8* const inputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesLength(1)));
sl@0
  1194
	{TPtr8 input(inputBuffer->Des());
sl@0
  1195
	aMessage.ReadL(1, input);}
sl@0
  1196
	HBufC8* const outputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesMaxLength(2)));
sl@0
  1197
	{TPtr8 output(outputBuffer->Des());
sl@0
  1198
	iServer->Manager()->NotifierUpdateL(targetUid,*inputBuffer,&output,iClientId);}
sl@0
  1199
	aMessage.WriteL(2,*outputBuffer);
sl@0
  1200
	CleanupStack::PopAndDestroy(2,inputBuffer);
sl@0
  1201
	}
sl@0
  1202
sl@0
  1203
void CNotifierSession::StartNotifierAndGetResponseL(const RMessage2& aMessage,TBool& aCleanupComplete)
sl@0
  1204
	{
sl@0
  1205
	const TUid targetUid={aMessage.Int0()};
sl@0
  1206
	HBufC8* const inputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesLength(1)));
sl@0
  1207
	{TPtr8 input(inputBuffer->Des());
sl@0
  1208
	aMessage.ReadL(1,input);}
sl@0
  1209
	iServer->Manager()->NotifierStartAndGetResponseL(targetUid,*inputBuffer,2,aMessage,iClientId,aCleanupComplete);
sl@0
  1210
	CleanupStack::PopAndDestroy(inputBuffer);
sl@0
  1211
	}
sl@0
  1212
sl@0
  1213
void CNotifierSession::PanicClient(const RMessage2& aMessage,TNotifierPanic aCode)
sl@0
  1214
	{
sl@0
  1215
	aMessage.Panic(__NOTIFIER_SERVER,aCode);
sl@0
  1216
	}
sl@0
  1217
sl@0
  1218
//
sl@0
  1219
// class CNotifierManager
sl@0
  1220
//
sl@0
  1221
sl@0
  1222
const TInt KNullClientId=0;
sl@0
  1223
sl@0
  1224
CNotifierManager* CNotifierManager::NewL()
sl@0
  1225
	{
sl@0
  1226
	CNotifierManager* self=new(ELeave) CNotifierManager;
sl@0
  1227
	CleanupStack::PushL(self);
sl@0
  1228
	self->ConstructL();
sl@0
  1229
	CleanupStack::Pop(self);
sl@0
  1230
	return self;
sl@0
  1231
	}
sl@0
  1232
sl@0
  1233
CNotifierManager::~CNotifierManager()
sl@0
  1234
	{
sl@0
  1235
	if (iObservedList)
sl@0
  1236
		{
sl@0
  1237
		const TInt count=iObservedList->Count();
sl@0
  1238
		for (TInt ii=0;ii<count;ii++)
sl@0
  1239
			{
sl@0
  1240
			(*iObservedList)[ii]->Release();
sl@0
  1241
			}
sl@0
  1242
		delete iObservedList;
sl@0
  1243
		}
sl@0
  1244
	if (iLibraries)
sl@0
  1245
		{
sl@0
  1246
		const TInt count=iLibraries->Count();
sl@0
  1247
		for (TInt ii=0;ii<count;ii++)
sl@0
  1248
			{
sl@0
  1249
			(*iLibraries)[ii].Close();
sl@0
  1250
			}
sl@0
  1251
		delete iLibraries;
sl@0
  1252
		}
sl@0
  1253
	delete iChannelMonitor;
sl@0
  1254
	delete iActivityMonitor;
sl@0
  1255
	delete iQueue;
sl@0
  1256
	}
sl@0
  1257
sl@0
  1258
sl@0
  1259
void CNotifierManager::RegisterL(RFs& aFs)
sl@0
  1260
	{
sl@0
  1261
#ifdef SUPPORT_OLD_PLUGIN_PATH
sl@0
  1262
	TBool old;
sl@0
  1263
	for(old=0; old<2; old++)
sl@0
  1264
	{
sl@0
  1265
#endif
sl@0
  1266
	TFindFile* findFile=new(ELeave) TFindFile(aFs);
sl@0
  1267
	CleanupStack::PushL(findFile);
sl@0
  1268
	TParse* fileNameParser=new(ELeave) TParse;
sl@0
  1269
	CleanupStack::PushL(fileNameParser);
sl@0
  1270
	CDir* directory=NULL;
sl@0
  1271
	TInt error;
sl@0
  1272
#ifdef SUPPORT_OLD_PLUGIN_PATH
sl@0
  1273
	_LIT(KNotifierPlugInOldSearchPath,"\\system\\tnotifiers\\");
sl@0
  1274
	if(old)
sl@0
  1275
		error=findFile->FindWildByDir(KNotifierPlugInExt, KNotifierPlugInOldSearchPath, directory);
sl@0
  1276
	else
sl@0
  1277
#endif
sl@0
  1278
	error=findFile->FindWildByDir(KNotifierPlugInExt, KNotifierPlugInSearchPath, directory);
sl@0
  1279
	for (; error!=KErrNotFound; error=findFile->FindWild(directory))
sl@0
  1280
		{
sl@0
  1281
		CleanupStack::PushL(directory);
sl@0
  1282
		User::LeaveIfError(error);
sl@0
  1283
		const TInt numberOfEntries=directory->Count();
sl@0
  1284
		for (TInt i=0; i<numberOfEntries; ++i)
sl@0
  1285
			{
sl@0
  1286
			const TEntry& entry=(*directory)[i];
sl@0
  1287
			fileNameParser->SetNoWild(entry.iName, &findFile->File(), NULL); // findFile->File() returns a reference rather than an object, therefore taking the address of it is fine
sl@0
  1288
sl@0
  1289
			if (entry.iType[0].iUid==0x10000079)
sl@0
  1290
				{
sl@0
  1291
				 // It's a DLL
sl@0
  1292
				if( (entry.iType[1]==KUidTextNotifierPlugInV2))
sl@0
  1293
					{
sl@0
  1294
					// Its a notifier...
sl@0
  1295
					TPtrC path(fileNameParser->DriveAndPath());
sl@0
  1296
					TPtrC name(fileNameParser->NameAndExt());
sl@0
  1297
					DoAddPlugInL(path,name,entry.iType);
sl@0
  1298
					}
sl@0
  1299
				}
sl@0
  1300
			}
sl@0
  1301
		CleanupStack::PopAndDestroy(); // directory
sl@0
  1302
		directory=NULL;
sl@0
  1303
		}
sl@0
  1304
	delete directory;
sl@0
  1305
	CleanupStack::PopAndDestroy(2); // fileNameParser and findFile
sl@0
  1306
#ifdef SUPPORT_OLD_PLUGIN_PATH
sl@0
  1307
	}
sl@0
  1308
#endif
sl@0
  1309
	}
sl@0
  1310
sl@0
  1311
LOCAL_C void DeleteTemp(TAny* aPtr)
sl@0
  1312
	{
sl@0
  1313
	CArrayPtr<MNotifierBase2>* array=REINTERPRET_CAST(CArrayPtr<MNotifierBase2>*,aPtr);
sl@0
  1314
	const TInt count=array->Count();
sl@0
  1315
	for (TInt ii=0;ii<count;ii++)
sl@0
  1316
		{
sl@0
  1317
		(*array)[ii]->Release();
sl@0
  1318
		}
sl@0
  1319
	delete array;
sl@0
  1320
	}
sl@0
  1321
sl@0
  1322
sl@0
  1323
sl@0
  1324
void CNotifierManager::DoAddPlugInL(const TDesC& aPath,const TDesC& aFileName,const TUidType& aUidType)
sl@0
  1325
	{
sl@0
  1326
	RLibrary lib;
sl@0
  1327
	User::LeaveIfError(lib.Load(aFileName,aPath,aUidType));
sl@0
  1328
	CleanupClosePushL(lib);
sl@0
  1329
	iLibraries->AppendL(lib);
sl@0
  1330
	CleanupStack::Pop(); // lib
sl@0
  1331
	TLibraryFunction libEntry=lib.Lookup(1);
sl@0
  1332
	CArrayPtr<MNotifierBase2>* array=REINTERPRET_CAST(CArrayPtr<MNotifierBase2>*,(libEntry)());
sl@0
  1333
	User::LeaveIfNull(array);
sl@0
  1334
	CleanupStack::PushL(TCleanupItem(DeleteTemp,array));
sl@0
  1335
	while (array->Count()>0)
sl@0
  1336
		{
sl@0
  1337
		MNotifierBase2* notif=(*array)[0];
sl@0
  1338
			{
sl@0
  1339
			iObservedList->AppendL(notif);
sl@0
  1340
			array->Delete(0);
sl@0
  1341
			notif->SetManager(this);
sl@0
  1342
			}
sl@0
  1343
		MNotifierBase2::TNotifierInfo info=notif->RegisterL();
sl@0
  1344
		if (!iChannelMonitor->AlreadyHasChannel(info.iChannel))
sl@0
  1345
			iChannelMonitor->AddNewChannelL(info.iChannel);
sl@0
  1346
		}
sl@0
  1347
	CleanupStack::PopAndDestroy(); // array
sl@0
  1348
	}
sl@0
  1349
sl@0
  1350
CNotifierManager::CNotifierManager()
sl@0
  1351
	{}
sl@0
  1352
sl@0
  1353
void CNotifierManager::ConstructL()
sl@0
  1354
	{
sl@0
  1355
	iObservedList=new(ELeave) CArrayPtrSeg<MNotifierBase2>(6);
sl@0
  1356
	iLibraries=new(ELeave) CArrayFixFlat<RLibrary>(2);
sl@0
  1357
	iChannelMonitor=CChannelMonitor::NewL();
sl@0
  1358
	iActivityMonitor=CActivityMonitor::NewL();
sl@0
  1359
	iQueue=CNotifierQueue::NewL();
sl@0
  1360
	}
sl@0
  1361
sl@0
  1362
struct SActivityCleanup
sl@0
  1363
	{
sl@0
  1364
	CActivityMonitor* iMonitor;
sl@0
  1365
	TUid iNotifier;
sl@0
  1366
	TInt iClientId;
sl@0
  1367
	};
sl@0
  1368
sl@0
  1369
LOCAL_C void CleanupActivityMonitor(TAny* aPtr)
sl@0
  1370
	{
sl@0
  1371
	SActivityCleanup& cleanup=*REINTERPRET_CAST(SActivityCleanup*,aPtr);
sl@0
  1372
	cleanup.iMonitor->Remove(cleanup.iNotifier,cleanup.iClientId);
sl@0
  1373
	}
sl@0
  1374
sl@0
  1375
void CNotifierManager::NotifierStartL(TUid aNotifierUid,const TDesC8& aBuffer,TPtrC8* aResponse,TInt aClientId)
sl@0
  1376
	{
sl@0
  1377
	TInt result=KErrNotFound;
sl@0
  1378
	const TInt count=iObservedList->Count();
sl@0
  1379
	for (TInt ii=0;ii<count;ii++)
sl@0
  1380
		{
sl@0
  1381
		MNotifierBase2* notif=(*iObservedList)[ii];
sl@0
  1382
		MNotifierBase2::TNotifierInfo info=notif->Info();
sl@0
  1383
		if (info.iUid==aNotifierUid)
sl@0
  1384
			{
sl@0
  1385
			if (iActivityMonitor->IsNotifierActive(aNotifierUid,info.iChannel))
sl@0
  1386
				{
sl@0
  1387
				result=KErrAlreadyExists;
sl@0
  1388
				}
sl@0
  1389
			else if (info.iPriority>iChannelMonitor->ActivityLevel(info.iChannel))
sl@0
  1390
				{
sl@0
  1391
				TUid notifier;
sl@0
  1392
				MNotifierBase2::TNotifierPriority priority;
sl@0
  1393
				const TBool channelWasActive=iActivityMonitor->IsChannelActive(info.iChannel,notifier,priority);
sl@0
  1394
				iActivityMonitor->AddL(info,aClientId);
sl@0
  1395
				SActivityCleanup cleanup;
sl@0
  1396
				cleanup.iMonitor=iActivityMonitor;
sl@0
  1397
				cleanup.iNotifier=aNotifierUid;
sl@0
  1398
				cleanup.iClientId=aClientId;
sl@0
  1399
				CleanupStack::PushL(TCleanupItem(CleanupActivityMonitor,&cleanup));
sl@0
  1400
				TPtrC8 response(notif->StartL(aBuffer));
sl@0
  1401
				if(aResponse)
sl@0
  1402
					aResponse->Set(response);
sl@0
  1403
				CleanupStack::Pop(); // cleanup;
sl@0
  1404
				if (channelWasActive)
sl@0
  1405
					{
sl@0
  1406
					for (TInt jj=0;jj<count;jj++)
sl@0
  1407
						{
sl@0
  1408
						MNotifierBase2* notifForUpdate=(*iObservedList)[ii];
sl@0
  1409
						MNotifierBase2::TNotifierInfo infoForUpdate=notifForUpdate->Info();
sl@0
  1410
						if (infoForUpdate.iUid==notifier && infoForUpdate.iChannel==info.iChannel)
sl@0
  1411
							{
sl@0
  1412
							TRAP_IGNORE(notifForUpdate->UpdateL(KNotifierPaused));
sl@0
  1413
							}
sl@0
  1414
						}
sl@0
  1415
					}
sl@0
  1416
				iChannelMonitor->UpdateChannel(info.iChannel,info.iPriority);
sl@0
  1417
				if (result!=ENotExtRequestQueued)
sl@0
  1418
					{
sl@0
  1419
					result=ENotExtRequestCompleted;
sl@0
  1420
					}
sl@0
  1421
				}
sl@0
  1422
			else
sl@0
  1423
				{
sl@0
  1424
				if (iQueue->IsAlreadyQueued(info.iUid,info.iChannel))
sl@0
  1425
					{
sl@0
  1426
					result=KErrAlreadyExists;
sl@0
  1427
					}
sl@0
  1428
				else
sl@0
  1429
					{
sl@0
  1430
					CQueueItem* queueCopy=CQueueItem::NewL(info,aBuffer,aClientId);
sl@0
  1431
					CleanupStack::PushL(queueCopy);
sl@0
  1432
					iQueue->QueueItemL(queueCopy);
sl@0
  1433
					CleanupStack::Pop(); // queueCopy
sl@0
  1434
					result=ENotExtRequestQueued;
sl@0
  1435
					}
sl@0
  1436
				}
sl@0
  1437
			}
sl@0
  1438
		}
sl@0
  1439
	User::LeaveIfError(result);
sl@0
  1440
	}
sl@0
  1441
sl@0
  1442
TInt CNotifierManager::NotifierUpdateL(TUid aNotifierUid,const TDesC8& aBuffer,TDes8* aResponse,TInt aClientId)
sl@0
  1443
	{
sl@0
  1444
	TInt result=KErrNotFound;
sl@0
  1445
	const TInt count=iObservedList->Count();
sl@0
  1446
	for (TInt ii=0;ii<count;ii++)
sl@0
  1447
		{
sl@0
  1448
		MNotifierBase2* notif=(*iObservedList)[ii];
sl@0
  1449
		MNotifierBase2::TNotifierInfo info=notif->Info();
sl@0
  1450
		if (info.iUid==aNotifierUid)
sl@0
  1451
			{
sl@0
  1452
			if (iActivityMonitor->IsNotifierActive(aNotifierUid,info.iChannel))
sl@0
  1453
				{
sl@0
  1454
				if (!iActivityMonitor->IsClientPresent(aNotifierUid,info.iChannel,aClientId))
sl@0
  1455
					{
sl@0
  1456
					iActivityMonitor->AddL(info,aClientId);
sl@0
  1457
					}
sl@0
  1458
				if (aResponse==NULL)
sl@0
  1459
					{
sl@0
  1460
					notif->UpdateL(aBuffer);
sl@0
  1461
					}
sl@0
  1462
				else
sl@0
  1463
					{
sl@0
  1464
					aResponse->Copy(notif->UpdateL(aBuffer));
sl@0
  1465
					}
sl@0
  1466
				}
sl@0
  1467
			else
sl@0
  1468
				{
sl@0
  1469
				; // not all channels have been started yet so update the queue
sl@0
  1470
				}
sl@0
  1471
			result=KErrNone;
sl@0
  1472
			}
sl@0
  1473
		}
sl@0
  1474
	return result;
sl@0
  1475
	}
sl@0
  1476
sl@0
  1477
void CNotifierManager::NotifierStartAndGetResponseL(TUid aNotifierUid,const TDesC8& aBuffer,TInt aReplySlot,
sl@0
  1478
														  const RMessage2& aMessage,TInt aClientId,TBool& aCleanupComplete)
sl@0
  1479
	{
sl@0
  1480
	NotifierStartAndGetResponseL(aNotifierUid,TUid::Null(),aBuffer,aReplySlot,aMessage,aClientId,aCleanupComplete);
sl@0
  1481
	}
sl@0
  1482
sl@0
  1483
void CNotifierManager::NotifierStartAndGetResponseL(TUid aNotifierUid,TUid aChannelUid,const TDesC8& aBuffer,TInt aReplySlot,
sl@0
  1484
														  const RMessage2& aMessage,TInt aClientId,TBool& aCleanupComplete)
sl@0
  1485
	{
sl@0
  1486
	TInt result=KErrNotFound;
sl@0
  1487
	const TInt count=iObservedList->Count();
sl@0
  1488
	for (TInt ii=0;ii<count;ii++)
sl@0
  1489
		{
sl@0
  1490
		MNotifierBase2* notif=(*iObservedList)[ii];
sl@0
  1491
		MNotifierBase2::TNotifierInfo info=notif->Info();
sl@0
  1492
		if (info.iUid==aNotifierUid && (aChannelUid==TUid::Null() || info.iChannel==aChannelUid))
sl@0
  1493
			{
sl@0
  1494
			if (iActivityMonitor->IsNotifierActive(aNotifierUid,info.iChannel))
sl@0
  1495
				{
sl@0
  1496
				notif->StartL(aBuffer,aReplySlot,aMessage); // asynch notifier can decide whether to support multiple clients
sl@0
  1497
				result=KErrNone;
sl@0
  1498
				}
sl@0
  1499
			else if (info.iPriority>iChannelMonitor->ActivityLevel(info.iChannel))
sl@0
  1500
				{
sl@0
  1501
				TUid notifier;
sl@0
  1502
				MNotifierBase2::TNotifierPriority priority;
sl@0
  1503
				const TBool channelWasActive=iActivityMonitor->IsChannelActive(info.iChannel,notifier,priority);
sl@0
  1504
				iActivityMonitor->AddL(info,aClientId);
sl@0
  1505
				SActivityCleanup activityCleanup;
sl@0
  1506
				activityCleanup.iMonitor=iActivityMonitor;
sl@0
  1507
				activityCleanup.iNotifier=aNotifierUid;
sl@0
  1508
				activityCleanup.iClientId=aClientId;
sl@0
  1509
				CleanupStack::PushL(TCleanupItem(CleanupActivityMonitor,&activityCleanup));
sl@0
  1510
				aCleanupComplete=EFalse;
sl@0
  1511
				// IMPORTANT, aMessage needs to be a full RMessage object until suport for V1 notifiers is removed
sl@0
  1512
				// I.e. until CNotifierBaseAdaptor is removed
sl@0
  1513
				notif->StartL(aBuffer,aReplySlot,aMessage);
sl@0
  1514
				CleanupStack::Pop(&activityCleanup);
sl@0
  1515
				if (channelWasActive)
sl@0
  1516
					{
sl@0
  1517
					for (TInt jj=0;jj<count;jj++)
sl@0
  1518
						{
sl@0
  1519
						MNotifierBase2* notifForUpdate=(*iObservedList)[ii];
sl@0
  1520
						MNotifierBase2::TNotifierInfo infoForUpdate=notifForUpdate->Info();
sl@0
  1521
						if (infoForUpdate.iUid==notifier && infoForUpdate.iChannel==info.iChannel)
sl@0
  1522
							{
sl@0
  1523
							TRAP_IGNORE(notifForUpdate->UpdateL(KNotifierPaused));
sl@0
  1524
							}
sl@0
  1525
						}
sl@0
  1526
					}
sl@0
  1527
				iChannelMonitor->UpdateChannel(info.iChannel,info.iPriority);
sl@0
  1528
				result=KErrNone;
sl@0
  1529
				}
sl@0
  1530
			else
sl@0
  1531
				{
sl@0
  1532
				if (iQueue->IsAlreadyQueued(info.iUid,info.iChannel))
sl@0
  1533
					{
sl@0
  1534
					result=KErrAlreadyExists;
sl@0
  1535
					}
sl@0
  1536
				else
sl@0
  1537
					{
sl@0
  1538
					CQueueItem* queueCopy=CQueueItem::NewL(info,aBuffer,aReplySlot,aMessage,aClientId);
sl@0
  1539
					CleanupStack::PushL(queueCopy);
sl@0
  1540
					iQueue->QueueItemL(queueCopy);
sl@0
  1541
					CleanupStack::Pop(queueCopy);
sl@0
  1542
					result=ENotExtRequestQueued;
sl@0
  1543
					}
sl@0
  1544
				}
sl@0
  1545
			}
sl@0
  1546
		}
sl@0
  1547
	User::LeaveIfError(result);
sl@0
  1548
	}
sl@0
  1549
sl@0
  1550
TInt CNotifierManager::NotifierCancel(TUid aNotifierUid)
sl@0
  1551
	{
sl@0
  1552
	TInt result=KErrNotFound;
sl@0
  1553
	const TInt count=iObservedList->Count();
sl@0
  1554
	for (TInt ii=0;ii<count;ii++)
sl@0
  1555
		{
sl@0
  1556
		MNotifierBase2* notif=(*iObservedList)[ii];
sl@0
  1557
		MNotifierBase2::TNotifierInfo info=notif->Info();
sl@0
  1558
		if (info.iUid==aNotifierUid)
sl@0
  1559
			{
sl@0
  1560
			notif->Cancel();
sl@0
  1561
			iActivityMonitor->RemoveNotifier(aNotifierUid,info.iChannel);
sl@0
  1562
			MNotifierBase2::TNotifierPriority priority=MNotifierBase2::ENotifierPriorityLowest;
sl@0
  1563
			TUid notifier;
sl@0
  1564
			//check channel activity and get highest priority on channnel
sl@0
  1565
			if (iActivityMonitor->IsChannelActive(info.iChannel,notifier,priority))
sl@0
  1566
				{
sl@0
  1567
sl@0
  1568
				//check if priority of a queued item on the same channel is
sl@0
  1569
				//greater 
sl@0
  1570
				MNotifierBase2::TNotifierPriority queuePriority=
sl@0
  1571
				(MNotifierBase2::TNotifierPriority)iQueue->GetHighestQueuePriority(info.iChannel);
sl@0
  1572
				if (queuePriority>priority)
sl@0
  1573
					{
sl@0
  1574
					iChannelMonitor->UpdateChannel(info.iChannel,MNotifierBase2::ENotifierPriorityLowest);
sl@0
  1575
					CQueueItem* next=iQueue->FetchItem(info.iChannel);
sl@0
  1576
					if (next)
sl@0
  1577
						{
sl@0
  1578
						TUid notif=next->iInfo.iUid;
sl@0
  1579
						TRAPD(err,StartFromQueueL(next));
sl@0
  1580
						if (err!=KErrNone)
sl@0
  1581
							{
sl@0
  1582
							NotifierCancel(notif);
sl@0
  1583
							}
sl@0
  1584
						}
sl@0
  1585
					 }
sl@0
  1586
					else
sl@0
  1587
					{
sl@0
  1588
					for (TInt jj=0;jj<count;jj++)
sl@0
  1589
						{
sl@0
  1590
						MNotifierBase2* notifForUpdate=(*iObservedList)[ii];
sl@0
  1591
						MNotifierBase2::TNotifierInfo infoForUpdate=notifForUpdate->Info();
sl@0
  1592
						if (infoForUpdate.iUid==notifier && infoForUpdate.iChannel==info.iChannel)
sl@0
  1593
							{
sl@0
  1594
							TRAP_IGNORE(notifForUpdate->UpdateL(KNotifierPaused));
sl@0
  1595
							}
sl@0
  1596
						}
sl@0
  1597
					iChannelMonitor->UpdateChannel(info.iChannel,priority);
sl@0
  1598
					}
sl@0
  1599
				}
sl@0
  1600
			else
sl@0
  1601
				{
sl@0
  1602
				iChannelMonitor->UpdateChannel(info.iChannel,MNotifierBase2::ENotifierPriorityLowest);
sl@0
  1603
				CQueueItem* next=iQueue->FetchItem(info.iChannel);
sl@0
  1604
				if (next)
sl@0
  1605
					{
sl@0
  1606
					TUid notif=next->iInfo.iUid;
sl@0
  1607
					TRAPD(err,StartFromQueueL(next));
sl@0
  1608
					if (err!=KErrNone)
sl@0
  1609
						{
sl@0
  1610
						NotifierCancel(notif);
sl@0
  1611
						}
sl@0
  1612
					}
sl@0
  1613
				}
sl@0
  1614
			result=KErrNone;
sl@0
  1615
			}
sl@0
  1616
		}
sl@0
  1617
	return result;
sl@0
  1618
	}
sl@0
  1619
sl@0
  1620
struct SCleanupMessage
sl@0
  1621
	{
sl@0
  1622
	TBool* iDoCleanup;
sl@0
  1623
	RMessage2* iMessage;
sl@0
  1624
	};
sl@0
  1625
sl@0
  1626
LOCAL_C void CleanupStartAndGetResponse(TAny* aPtr)
sl@0
  1627
	{
sl@0
  1628
	SCleanupMessage& cleanup=*REINTERPRET_CAST(SCleanupMessage*,aPtr);
sl@0
  1629
	if (cleanup.iDoCleanup)
sl@0
  1630
		{
sl@0
  1631
		cleanup.iMessage->Complete(KErrNoMemory);
sl@0
  1632
		}
sl@0
  1633
	}
sl@0
  1634
sl@0
  1635
void CNotifierManager::StartFromQueueL(CQueueItem* aItem)
sl@0
  1636
	{
sl@0
  1637
	CleanupStack::PushL(aItem);
sl@0
  1638
	TPtr8 buffer=aItem->iBuffer->Des();
sl@0
  1639
	if (aItem->iAsynchronous)
sl@0
  1640
		{
sl@0
  1641
		SCleanupMessage cleanup;
sl@0
  1642
		TBool doCleanup=ETrue;
sl@0
  1643
		cleanup.iDoCleanup=&doCleanup;
sl@0
  1644
		cleanup.iMessage=&aItem->iMessage;
sl@0
  1645
		CleanupStack::PushL(TCleanupItem(CleanupStartAndGetResponse,&cleanup));
sl@0
  1646
		// IMPORTANT, aItem->iMessage needs to be a full RMessage object until suport for V1 notifiers is removed
sl@0
  1647
		// I.e. until CNotifierBaseAdaptor is removed
sl@0
  1648
		NotifierStartAndGetResponseL(aItem->iInfo.iUid,aItem->iInfo.iChannel,buffer,aItem->iReplySlot,aItem->iMessage,aItem->iClientId,doCleanup);
sl@0
  1649
		CleanupStack::Pop(&cleanup);
sl@0
  1650
		}
sl@0
  1651
	else
sl@0
  1652
		{
sl@0
  1653
		NotifierStartL(aItem->iInfo.iUid,buffer,NULL,aItem->iClientId);
sl@0
  1654
		}
sl@0
  1655
	CleanupStack::PopAndDestroy(); // aItem
sl@0
  1656
	CQueueItem* update=iQueue->FetchItem(aItem->iInfo.iChannel);
sl@0
  1657
	while (update)
sl@0
  1658
		{
sl@0
  1659
		CleanupStack::PushL(update);
sl@0
  1660
		NotifierUpdateL(update->iInfo.iUid,*update->iBuffer,NULL,update->iClientId);
sl@0
  1661
		CleanupStack::PopAndDestroy(); // update
sl@0
  1662
		update=iQueue->FetchItem(aItem->iInfo.iChannel);
sl@0
  1663
		}
sl@0
  1664
	}
sl@0
  1665
sl@0
  1666
void CNotifierManager::HandleClientExit(TInt aClientId)
sl@0
  1667
	{
sl@0
  1668
	TUid notifier=KNullUid;
sl@0
  1669
	while (iActivityMonitor->NotifierForClient(notifier,aClientId))
sl@0
  1670
		{
sl@0
  1671
		const TInt count=iObservedList->Count();
sl@0
  1672
		for (TInt ii=0;ii<count;ii++)
sl@0
  1673
			{
sl@0
  1674
			MNotifierBase2* notif=(*iObservedList)[ii];
sl@0
  1675
			if (notif->Info().iUid==notifier)
sl@0
  1676
				{
sl@0
  1677
				NotifierCancel(notifier);
sl@0
  1678
				}
sl@0
  1679
			}
sl@0
  1680
		iActivityMonitor->Remove(notifier,aClientId);
sl@0
  1681
		}
sl@0
  1682
	iActivityMonitor->RemoveClient(aClientId);
sl@0
  1683
	iQueue->RemoveClient(aClientId);
sl@0
  1684
	}
sl@0
  1685
sl@0
  1686
void CNotifierManager::StartNotifierL(TUid aNotifierUid,const TDesC8& aBuffer,TDes8& aResponse)
sl@0
  1687
	{
sl@0
  1688
	TPtrC8 response(0,0);
sl@0
  1689
	NotifierStartL(aNotifierUid,aBuffer, &response,KNullClientId);
sl@0
  1690
	aResponse.Copy(response);
sl@0
  1691
	}
sl@0
  1692
sl@0
  1693
void CNotifierManager::CancelNotifier(TUid aNotifierUid)
sl@0
  1694
	{
sl@0
  1695
	NotifierCancel(aNotifierUid);
sl@0
  1696
	}
sl@0
  1697
sl@0
  1698
void CNotifierManager::UpdateNotifierL(TUid aNotifierUid,const TDesC8& aBuffer,TDes8& aResponse)
sl@0
  1699
	{
sl@0
  1700
	NotifierUpdateL(aNotifierUid,aBuffer,&aResponse,KNullClientId);
sl@0
  1701
	}
sl@0
  1702
sl@0
  1703
//
sl@0
  1704
// class CChannelMonitor
sl@0
  1705
//
sl@0
  1706
sl@0
  1707
CChannelMonitor* CChannelMonitor::NewL()
sl@0
  1708
	{
sl@0
  1709
	CChannelMonitor* self=new(ELeave) CChannelMonitor;
sl@0
  1710
	return self;
sl@0
  1711
	}
sl@0
  1712
sl@0
  1713
TBool CChannelMonitor::AlreadyHasChannel(TUid aChannel)const
sl@0
  1714
	{
sl@0
  1715
	const TInt count=iMonitor.Count();
sl@0
  1716
	for (TInt ii=0;ii<count;ii++)
sl@0
  1717
		{
sl@0
  1718
		if (iMonitor[ii].iChannel==aChannel)
sl@0
  1719
			return ETrue;
sl@0
  1720
		}
sl@0
  1721
	return EFalse;
sl@0
  1722
	}
sl@0
  1723
sl@0
  1724
TInt CChannelMonitor::ActivityLevel(TUid aChannel) const
sl@0
  1725
	{
sl@0
  1726
	const TInt count=iMonitor.Count();
sl@0
  1727
	for (TInt ii=0;ii<count;ii++)
sl@0
  1728
		{
sl@0
  1729
		TChannelActivity activity=iMonitor[ii];
sl@0
  1730
		if (activity.iChannel==aChannel)
sl@0
  1731
			return activity.iHighestPriorityRunning;
sl@0
  1732
		}
sl@0
  1733
	return 0;
sl@0
  1734
	}
sl@0
  1735
sl@0
  1736
void CChannelMonitor::UpdateChannel(TUid aChannel,TInt aLevel)
sl@0
  1737
	{
sl@0
  1738
	const TInt count=iMonitor.Count();
sl@0
  1739
	for (TInt ii=0;ii<count;ii++)
sl@0
  1740
		{
sl@0
  1741
		TChannelActivity& activity=iMonitor[ii];
sl@0
  1742
		if (activity.iChannel==aChannel)
sl@0
  1743
			{
sl@0
  1744
			activity.iHighestPriorityRunning=aLevel;
sl@0
  1745
			break;
sl@0
  1746
			}
sl@0
  1747
		}
sl@0
  1748
	}
sl@0
  1749
sl@0
  1750
CChannelMonitor::CChannelMonitor()
sl@0
  1751
	:iMonitor(3)
sl@0
  1752
	{}
sl@0
  1753
sl@0
  1754
//
sl@0
  1755
// class CNotifierActivity
sl@0
  1756
//
sl@0
  1757
sl@0
  1758
CNotifierActivity* CNotifierActivity::NewLC(const MNotifierBase2::TNotifierInfo& aInfo,TInt aClientId)
sl@0
  1759
	{ // static
sl@0
  1760
	CNotifierActivity* self=new(ELeave) CNotifierActivity(aInfo);
sl@0
  1761
	CleanupStack::PushL(self);
sl@0
  1762
	self->ConstructL(aClientId);
sl@0
  1763
	return self;
sl@0
  1764
	}
sl@0
  1765
sl@0
  1766
CNotifierActivity::~CNotifierActivity()
sl@0
  1767
	{
sl@0
  1768
	iClientArray.Reset();
sl@0
  1769
	}
sl@0
  1770
sl@0
  1771
TInt CNotifierActivity::Find(TInt aClientId) const
sl@0
  1772
	{
sl@0
  1773
	TInt index=KErrNotFound;
sl@0
  1774
	const TInt count=iClientArray.Count();
sl@0
  1775
	for (TInt ii=0;ii<count;ii++)
sl@0
  1776
		{
sl@0
  1777
		TInt clientId=iClientArray[ii];
sl@0
  1778
		if (clientId==aClientId)
sl@0
  1779
			{
sl@0
  1780
			index=ii;
sl@0
  1781
			break;
sl@0
  1782
			}
sl@0
  1783
		}
sl@0
  1784
	return index;
sl@0
  1785
	}
sl@0
  1786
sl@0
  1787
CNotifierActivity::CNotifierActivity(const MNotifierBase2::TNotifierInfo& aInfo)
sl@0
  1788
	: iInfo(aInfo), iClientArray(1)
sl@0
  1789
	{}
sl@0
  1790
sl@0
  1791
void CNotifierActivity::ConstructL(TInt aClientId)
sl@0
  1792
	{
sl@0
  1793
	iClientArray.AppendL(aClientId);
sl@0
  1794
	}
sl@0
  1795
sl@0
  1796
//
sl@0
  1797
// class CActivityMonitor
sl@0
  1798
//
sl@0
  1799
sl@0
  1800
CActivityMonitor* CActivityMonitor::NewL()
sl@0
  1801
	{ // static
sl@0
  1802
	CActivityMonitor* self=new(ELeave) CActivityMonitor();
sl@0
  1803
	return self;
sl@0
  1804
	}
sl@0
  1805
sl@0
  1806
CActivityMonitor::~CActivityMonitor()
sl@0
  1807
	{
sl@0
  1808
	iMonitor.ResetAndDestroy();
sl@0
  1809
	}
sl@0
  1810
sl@0
  1811
void CActivityMonitor::AddL(const MNotifierBase2::TNotifierInfo& aInfo,TInt aClientId)
sl@0
  1812
	{
sl@0
  1813
	const TInt index=Find(aInfo.iUid,aInfo.iChannel);
sl@0
  1814
	if (index==KErrNotFound)
sl@0
  1815
		{
sl@0
  1816
		CNotifierActivity* activity=CNotifierActivity::NewLC(aInfo,aClientId);
sl@0
  1817
		iMonitor.AppendL(activity);
sl@0
  1818
		CleanupStack::Pop(); // activity
sl@0
  1819
		}
sl@0
  1820
	else
sl@0
  1821
		{
sl@0
  1822
		iMonitor[index]->iClientArray.AppendL(aClientId);
sl@0
  1823
		}
sl@0
  1824
	}
sl@0
  1825
sl@0
  1826
void CActivityMonitor::Remove(TUid aNotifierUid,TInt aClientId)
sl@0
  1827
	{
sl@0
  1828
	const TInt index=Find(aNotifierUid);
sl@0
  1829
	if (index!=KErrNotFound)
sl@0
  1830
		{
sl@0
  1831
		CNotifierActivity* activity=iMonitor[index];
sl@0
  1832
		const TInt clientIndex=activity->Find(aClientId);
sl@0
  1833
		if (clientIndex!=KErrNotFound)
sl@0
  1834
			{
sl@0
  1835
			if (activity->iClientArray.Count()==1)
sl@0
  1836
				{
sl@0
  1837
				delete activity;
sl@0
  1838
				iMonitor.Delete(index);
sl@0
  1839
				}
sl@0
  1840
			else
sl@0
  1841
				{
sl@0
  1842
				activity->iClientArray.Delete(index);
sl@0
  1843
				}
sl@0
  1844
			}
sl@0
  1845
		}
sl@0
  1846
	}
sl@0
  1847
sl@0
  1848
void CActivityMonitor::RemoveNotifier(TUid aNotifierUid,TUid aChannel)
sl@0
  1849
	{
sl@0
  1850
	const TInt index=Find(aNotifierUid,aChannel);
sl@0
  1851
	if (index!=KErrNotFound)
sl@0
  1852
		{
sl@0
  1853
		delete iMonitor[index];
sl@0
  1854
		iMonitor.Delete(index);
sl@0
  1855
		}
sl@0
  1856
	}
sl@0
  1857
sl@0
  1858
void CActivityMonitor::RemoveClient(TInt aClientId)
sl@0
  1859
	{
sl@0
  1860
	TInt ii=0;
sl@0
  1861
	while (ii<iMonitor.Count())
sl@0
  1862
		{
sl@0
  1863
		CNotifierActivity* ptr=iMonitor[ii];
sl@0
  1864
		TInt index=ptr->Find(aClientId);
sl@0
  1865
		if (index!=KErrNotFound)
sl@0
  1866
			{
sl@0
  1867
			ptr->iClientArray.Delete(index);
sl@0
  1868
			}
sl@0
  1869
		if (ptr->iClientArray.Count()==0)
sl@0
  1870
			{
sl@0
  1871
			iMonitor.Delete(ii);
sl@0
  1872
			}
sl@0
  1873
		else
sl@0
  1874
			{
sl@0
  1875
			++ii;
sl@0
  1876
			}
sl@0
  1877
		}
sl@0
  1878
	}
sl@0
  1879
sl@0
  1880
TBool CActivityMonitor::IsNotifierActive(TUid aNotifierUid,TUid aChannel) const
sl@0
  1881
	{
sl@0
  1882
	const TInt index=Find(aNotifierUid,aChannel);
sl@0
  1883
	return (index!=KErrNotFound);
sl@0
  1884
	}
sl@0
  1885
sl@0
  1886
TBool CActivityMonitor::IsClientPresent(TUid aNotifierUid,TUid aChannel,TInt aClientId) const
sl@0
  1887
	{
sl@0
  1888
	TBool found=EFalse;
sl@0
  1889
	const TInt index=Find(aNotifierUid,aChannel);
sl@0
  1890
	if (index!=KErrNotFound)
sl@0
  1891
		{
sl@0
  1892
		found=(iMonitor[index]->Find(aClientId)!=KErrNotFound);
sl@0
  1893
		}
sl@0
  1894
	return found;
sl@0
  1895
	}
sl@0
  1896
sl@0
  1897
TBool CActivityMonitor::IsChannelActive(TUid aChannel,TUid& aNotifier,MNotifierBase2::TNotifierPriority& aHighestPriority) const
sl@0
  1898
	{
sl@0
  1899
	TBool ret=EFalse;
sl@0
  1900
	const TInt count=iMonitor.Count();
sl@0
  1901
	for (TInt ii=0;ii<count;ii++)
sl@0
  1902
		{
sl@0
  1903
		MNotifierBase2::TNotifierInfo info=iMonitor[ii]->iInfo;
sl@0
  1904
		if (info.iChannel==aChannel)
sl@0
  1905
			{
sl@0
  1906
			ret=ETrue;
sl@0
  1907
			if ((MNotifierBase2::TNotifierPriority)info.iPriority>aHighestPriority)
sl@0
  1908
				{
sl@0
  1909
				aNotifier=info.iUid;
sl@0
  1910
				aHighestPriority=(MNotifierBase2::TNotifierPriority)info.iPriority;
sl@0
  1911
				}
sl@0
  1912
			}
sl@0
  1913
		}
sl@0
  1914
	return ret;
sl@0
  1915
	}
sl@0
  1916
sl@0
  1917
TBool CActivityMonitor::NotifierForClient(TUid& aNotifierUid,TInt aClientId) const
sl@0
  1918
	{
sl@0
  1919
	TBool isOnlyClient=EFalse;
sl@0
  1920
	aNotifierUid=KNullUid;
sl@0
  1921
	const TInt count=iMonitor.Count();
sl@0
  1922
	for (TInt ii=0;ii<count;ii++)
sl@0
  1923
		{
sl@0
  1924
		CNotifierActivity* ptr=iMonitor[ii];
sl@0
  1925
		if (ptr->Find(aClientId)!=KErrNotFound)
sl@0
  1926
			{
sl@0
  1927
			aNotifierUid=ptr->iInfo.iUid;
sl@0
  1928
			isOnlyClient=ptr->iClientArray.Count()==1;
sl@0
  1929
			break;
sl@0
  1930
			}
sl@0
  1931
		}
sl@0
  1932
	return isOnlyClient;
sl@0
  1933
	}
sl@0
  1934
sl@0
  1935
CActivityMonitor::CActivityMonitor()
sl@0
  1936
	: iMonitor(1)
sl@0
  1937
	{}
sl@0
  1938
sl@0
  1939
TInt CActivityMonitor::Find(TUid aNotifierUid) const
sl@0
  1940
	{
sl@0
  1941
	TInt index=KErrNotFound;
sl@0
  1942
	const TInt count=iMonitor.Count();
sl@0
  1943
	for (TInt ii=0;ii<count;ii++)
sl@0
  1944
		{
sl@0
  1945
		if (iMonitor[ii]->iInfo.iUid==aNotifierUid)
sl@0
  1946
			{
sl@0
  1947
			index=ii;
sl@0
  1948
			break;
sl@0
  1949
			}
sl@0
  1950
		}
sl@0
  1951
	return index;
sl@0
  1952
	}
sl@0
  1953
sl@0
  1954
TInt CActivityMonitor::Find(TUid aNotifierUid,TUid aChannel) const
sl@0
  1955
	{
sl@0
  1956
	TInt index=KErrNotFound;
sl@0
  1957
	const TInt count=iMonitor.Count();
sl@0
  1958
	for (TInt ii=0;ii<count;ii++)
sl@0
  1959
		{
sl@0
  1960
		CNotifierActivity* ptr=iMonitor[ii];
sl@0
  1961
		if (ptr->iInfo.iUid==aNotifierUid && ptr->iInfo.iChannel==aChannel)
sl@0
  1962
			{
sl@0
  1963
			index=ii;
sl@0
  1964
			break;
sl@0
  1965
			}
sl@0
  1966
		}
sl@0
  1967
	return index;
sl@0
  1968
	}
sl@0
  1969
sl@0
  1970
//
sl@0
  1971
// class CQueueItem
sl@0
  1972
//
sl@0
  1973
sl@0
  1974
CQueueItem* CQueueItem::NewL(const MNotifierBase2::TNotifierInfo& aInfo,const TDesC8& aBuffer,
sl@0
  1975
									TInt aReplySlot,const RMessage2& aMessage,TInt aClientId) //Asynchronous
sl@0
  1976
	{
sl@0
  1977
	CQueueItem* self=new(ELeave) CQueueItem(aInfo);
sl@0
  1978
	CleanupStack::PushL(self);
sl@0
  1979
	self->ConstructL(aBuffer,aMessage,aClientId,aReplySlot);
sl@0
  1980
	CleanupStack::Pop(); // self
sl@0
  1981
	return self;
sl@0
  1982
	}
sl@0
  1983
sl@0
  1984
CQueueItem* CQueueItem::NewL(const MNotifierBase2::TNotifierInfo& aInfo,const TDesC8& aBuffer,TInt aClientId) //synchronous
sl@0
  1985
	{
sl@0
  1986
	CQueueItem* self=new(ELeave) CQueueItem(aInfo);
sl@0
  1987
	CleanupStack::PushL(self);
sl@0
  1988
	self->ConstructL(aBuffer,aClientId);
sl@0
  1989
	CleanupStack::Pop(); // self
sl@0
  1990
	return self;
sl@0
  1991
	}
sl@0
  1992
sl@0
  1993
CQueueItem::~CQueueItem()
sl@0
  1994
	{
sl@0
  1995
	delete iBuffer;
sl@0
  1996
	}
sl@0
  1997
sl@0
  1998
CQueueItem::CQueueItem(const MNotifierBase2::TNotifierInfo& aInfo)
sl@0
  1999
	: iInfo(aInfo)
sl@0
  2000
	{}
sl@0
  2001
sl@0
  2002
void CQueueItem::ConstructL(const TDesC8& aBuffer,TInt aClientId)
sl@0
  2003
	{
sl@0
  2004
	iBuffer=aBuffer.AllocL();
sl@0
  2005
	iClientId=aClientId;
sl@0
  2006
	iAsynchronous=EFalse;
sl@0
  2007
	}
sl@0
  2008
sl@0
  2009
void CQueueItem::ConstructL(const TDesC8& aBuffer,const RMessage2& aMessage,TInt aClientId,TInt aReplySlot)
sl@0
  2010
	{
sl@0
  2011
	iBuffer=aBuffer.AllocL();
sl@0
  2012
	iAsynchronous=ETrue;
sl@0
  2013
	iMessage=aMessage;
sl@0
  2014
	iClientId=aClientId;
sl@0
  2015
	iReplySlot=aReplySlot;
sl@0
  2016
	}
sl@0
  2017
sl@0
  2018
//
sl@0
  2019
// class CNotifierQueue
sl@0
  2020
//
sl@0
  2021
sl@0
  2022
CNotifierQueue* CNotifierQueue::NewL()
sl@0
  2023
	{
sl@0
  2024
	CNotifierQueue* self=new(ELeave) CNotifierQueue;
sl@0
  2025
	return self;
sl@0
  2026
	}
sl@0
  2027
sl@0
  2028
CQueueItem* CNotifierQueue::FetchItem(TUid aChannel)
sl@0
  2029
	{
sl@0
  2030
	CQueueItem* result=NULL;
sl@0
  2031
	const TInt count=iQueue.Count();
sl@0
  2032
	TInt priority=MNotifierBase2::ENotifierPriorityLowest-1;
sl@0
  2033
	TInt index=KErrNotFound;
sl@0
  2034
	for (TInt ii=0;ii<count;ii++)
sl@0
  2035
		{
sl@0
  2036
		CQueueItem* item=iQueue[ii];
sl@0
  2037
		if (item->iInfo.iChannel==aChannel && item->iInfo.iPriority>priority)
sl@0
  2038
			{
sl@0
  2039
			index=ii;
sl@0
  2040
			priority=item->iInfo.iPriority;
sl@0
  2041
			result=item;
sl@0
  2042
			}
sl@0
  2043
		}
sl@0
  2044
	if (index!=KErrNotFound)
sl@0
  2045
		{
sl@0
  2046
		iQueue.Delete(index);
sl@0
  2047
		}
sl@0
  2048
	return result;
sl@0
  2049
	}
sl@0
  2050
sl@0
  2051
TBool CNotifierQueue::IsAlreadyQueued(TUid aNotifier,TUid aChannel) const
sl@0
  2052
	{
sl@0
  2053
	TBool ret=EFalse;
sl@0
  2054
	const TInt count=iQueue.Count();
sl@0
  2055
	for (TInt ii=0;ii<count;ii++)
sl@0
  2056
		{
sl@0
  2057
		CQueueItem* item=iQueue[ii];
sl@0
  2058
		if (item->iInfo.iUid==aNotifier && item->iInfo.iChannel==aChannel)
sl@0
  2059
			{
sl@0
  2060
			ret=ETrue;
sl@0
  2061
			break;
sl@0
  2062
			}
sl@0
  2063
		}
sl@0
  2064
	return ret;
sl@0
  2065
	}
sl@0
  2066
sl@0
  2067
void CNotifierQueue::RemoveClient(TInt aClientId)
sl@0
  2068
	{
sl@0
  2069
	const TInt count=iQueue.Count();
sl@0
  2070
	for (TInt ii=count-1;ii>=0;ii--)
sl@0
  2071
		{
sl@0
  2072
		CQueueItem* item=iQueue[ii];
sl@0
  2073
		TInt clientId=item->iClientId;
sl@0
  2074
		if (clientId==aClientId)
sl@0
  2075
			{
sl@0
  2076
			iQueue.Delete(ii);
sl@0
  2077
			}
sl@0
  2078
		}
sl@0
  2079
	}
sl@0
  2080
sl@0
  2081
sl@0
  2082
TInt CNotifierQueue::GetHighestQueuePriority(TUid aChannel)
sl@0
  2083
	{
sl@0
  2084
	const TInt count=iQueue.Count();
sl@0
  2085
	TInt priority=MNotifierBase2::ENotifierPriorityLowest-1;
sl@0
  2086
sl@0
  2087
	for (TInt ii=0;ii<count;ii++)
sl@0
  2088
		{
sl@0
  2089
		CQueueItem* item=iQueue[ii];
sl@0
  2090
		if (item->iInfo.iChannel==aChannel && item->iInfo.iPriority>priority)
sl@0
  2091
			{
sl@0
  2092
			priority=item->iInfo.iPriority;
sl@0
  2093
			}
sl@0
  2094
		}
sl@0
  2095
sl@0
  2096
	return priority;
sl@0
  2097
	}
sl@0
  2098
sl@0
  2099
sl@0
  2100
void CWsActiveScheduler::New()
sl@0
  2101
//
sl@0
  2102
// Create and install the active scheduler.
sl@0
  2103
//
sl@0
  2104
	{
sl@0
  2105
sl@0
  2106
	CWsActiveScheduler *pA=new CWsActiveScheduler;
sl@0
  2107
	__ASSERT_ALWAYS(pA!=NULL,Fault(ECreateScheduler));
sl@0
  2108
	CActiveScheduler::Install(pA);
sl@0
  2109
	}
sl@0
  2110
sl@0
  2111
void CWsActiveScheduler::Error(TInt) const
sl@0
  2112
//
sl@0
  2113
// Called if any Run() method leaves.
sl@0
  2114
//
sl@0
  2115
	{
sl@0
  2116
	}
sl@0
  2117
sl@0
  2118
sl@0
  2119
TInt NotifierServerThread(TAny*)
sl@0
  2120
	{
sl@0
  2121
	CTrapCleanup* CleanUpStack=CTrapCleanup::New();
sl@0
  2122
	CWsActiveScheduler::New();
sl@0
  2123
	TRAP_IGNORE(CNotifierServer::NewL());
sl@0
  2124
	CNotifierSession::NotifierSemaphore.Signal();
sl@0
  2125
	CWsActiveScheduler::Start();
sl@0
  2126
	delete CleanUpStack;
sl@0
  2127
	return(0);
sl@0
  2128
	}
sl@0
  2129
sl@0
  2130
sl@0
  2131
_LIT(KLitKeyDataDllNameBase, "EKDATA");
sl@0
  2132
_LIT(TwoDigExt,".%02d");
sl@0
  2133
sl@0
  2134
GLDEF_C TInt E32Main()
sl@0
  2135
	{
sl@0
  2136
	UserSvr::WsRegisterThread();
sl@0
  2137
	UserSvr::WsRegisterSwitchOnScreenHandling(ETrue);
sl@0
  2138
	User::SetProcessCritical(User::ESystemPermanent);
sl@0
  2139
	User::SetCritical(User::ESystemPermanent);
sl@0
  2140
sl@0
  2141
	CWsActiveScheduler::New();
sl@0
  2142
	CWsServer::New();
sl@0
  2143
	CWsWindow::New();
sl@0
  2144
	CEvent::New();
sl@0
  2145
    
sl@0
  2146
	KeyTranslator=CKeyTranslator::New();
sl@0
  2147
	if (!KeyTranslator)
sl@0
  2148
		Fault(ENoKeyboardTranslator);
sl@0
  2149
sl@0
  2150
//  Change keyboard mapping according to information in the HAL
sl@0
  2151
//	This code is the same as WSERV
sl@0
  2152
	TInt keyboardIndex;
sl@0
  2153
	if (HAL::Get(HALData::EKeyboardIndex,keyboardIndex)==KErrNone)
sl@0
  2154
		{
sl@0
  2155
		TBuf<16> keyDataDllName(KLitKeyDataDllNameBase);
sl@0
  2156
		keyDataDllName.AppendFormat(TwoDigExt, keyboardIndex);
sl@0
  2157
		KeyTranslator->ChangeKeyData(keyDataDllName);
sl@0
  2158
		}
sl@0
  2159
sl@0
  2160
    KeyRepeat=new(ELeave) CKeyRepeat(CKeyRepeat::EKeyRepeatPriority);
sl@0
  2161
	TRAPD(r,KeyRepeat->ConstructL());
sl@0
  2162
	if (r!=KErrNone)
sl@0
  2163
		User::Panic(_L("KEYREPEAT"),r);
sl@0
  2164
sl@0
  2165
#ifndef __WINS__
sl@0
  2166
    if (UserSvr::TestBootSequence())
sl@0
  2167
		{
sl@0
  2168
		RDebug::Print(_L("WS_MAIN: TestBootSequence=TRUE, not loading ESHELL.EXE"));
sl@0
  2169
		}
sl@0
  2170
#else
sl@0
  2171
    if (EmulatorAutoRun())
sl@0
  2172
    	{	// don't start ESHELL if we used a self-bootstrapping EXE
sl@0
  2173
    	}
sl@0
  2174
#endif
sl@0
  2175
	else
sl@0
  2176
		{
sl@0
  2177
		RProcess shell;
sl@0
  2178
		r=shell.Create(KShellProcessName, KShellCommandLine);
sl@0
  2179
		__ASSERT_ALWAYS(r==KErrNone,Fault(ECreateShell));
sl@0
  2180
		shell.Resume();
sl@0
  2181
		shell.Close();
sl@0
  2182
		}
sl@0
  2183
sl@0
  2184
	RThread t;
sl@0
  2185
	r=CNotifierSession::NotifierSemaphore.CreateLocal(0);
sl@0
  2186
	if (r!=KErrNone)
sl@0
  2187
		Fault(ECreateNotifierSemaphore);
sl@0
  2188
	r=t.Create(_L("NotifierServer"),NotifierServerThread,KDefaultStackSize,0x2000,0x100000,NULL);
sl@0
  2189
	if (r!=KErrNone)
sl@0
  2190
		Fault(ECreateNotifierThread);
sl@0
  2191
	t.Resume();
sl@0
  2192
	CNotifierSession::NotifierSemaphore.Wait();
sl@0
  2193
sl@0
  2194
	CWsActiveScheduler::Start();
sl@0
  2195
	UserSvr::ReleaseEventHook();
sl@0
  2196
	return(KErrNone);
sl@0
  2197
	}
sl@0
  2198