os/graphics/windowing/windowserver/nonnga/SERVER/WSTOP.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) 1994-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 "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
// Top level window server code
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include "wstop.h"
sl@0
    19
sl@0
    20
#include <hal.h>
sl@0
    21
#include <graphics/fbsdefs.h>
sl@0
    22
#include "ANIM.H"
sl@0
    23
#include "gc.h"
sl@0
    24
#include "playbackgc.h"
sl@0
    25
#include "rootwin.h"
sl@0
    26
#include "windowgroup.h"
sl@0
    27
#include "EVENT.H"
sl@0
    28
#include "KEYCLICK.H"
sl@0
    29
#include <w32std.h>
sl@0
    30
#include "offscreenbitmap.h"
sl@0
    31
#include "panics.h"
sl@0
    32
#include "inifile.h"
sl@0
    33
#include "pointer.h"
sl@0
    34
#include "WSGRAPHICDRAWERFACTORY.H"
sl@0
    35
#include "redrawmsgwindow.h"
sl@0
    36
#include <Graphics/WSGRAPHICDRAWERINTERFACE.H>
sl@0
    37
#include "WsMemMgr.h"
sl@0
    38
#include "backedupwindow.h"
sl@0
    39
#include "wsfont.h"
sl@0
    40
#include "wstraces.h"
sl@0
    41
#ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP
sl@0
    42
#include <e32power.h>
sl@0
    43
#endif
sl@0
    44
#include "graphics/windowserverconstants.h"
sl@0
    45
sl@0
    46
typedef CDebugLogBase *(*CreateDebugLog)(TBool aIsFirst, TDesC &aParams);
sl@0
    47
sl@0
    48
CWsActiveScheduler* CWsActiveScheduler::Static()
sl@0
    49
	{
sl@0
    50
	return static_cast<CWsActiveScheduler*>(CActiveScheduler::Current());
sl@0
    51
	}
sl@0
    52
	
sl@0
    53
CWsActiveScheduler::CWsActiveScheduler(): iNumSamples(100)
sl@0
    54
	{
sl@0
    55
	iRun = User::FastCounter();
sl@0
    56
	if((KErrNone == HAL::Get(HALData::EFastCounterFrequency,iFastCounterFreq)) && iFastCounterFreq)
sl@0
    57
		{
sl@0
    58
		iData = reinterpret_cast<TSample*>(User::AllocZ(sizeof(TSample)*iNumSamples));
sl@0
    59
		}
sl@0
    60
	}
sl@0
    61
sl@0
    62
CWsActiveScheduler::~CWsActiveScheduler()
sl@0
    63
	{
sl@0
    64
	User::FreeZ(reinterpret_cast<void*&>(iData));
sl@0
    65
	}
sl@0
    66
	
sl@0
    67
void CWsActiveScheduler::PrepareDraw()
sl@0
    68
	{
sl@0
    69
	WS_ASSERT_DEBUG((ENormal == iState)||(EDrawn == iState),EWsPanicActiveScheduler);
sl@0
    70
	__DEBUG_ONLY(iState = EPreparingDraw;)
sl@0
    71
	iRunDraw = User::FastCounter();
sl@0
    72
	}
sl@0
    73
	
sl@0
    74
void CWsActiveScheduler::CancelPrepare()
sl@0
    75
	{
sl@0
    76
	WS_ASSERT_DEBUG(EPreparingDraw == iState,EWsPanicActiveScheduler);
sl@0
    77
	__DEBUG_ONLY(iState = ENormal;)
sl@0
    78
	}
sl@0
    79
	
sl@0
    80
void CWsActiveScheduler::StartDraw()
sl@0
    81
	{
sl@0
    82
	WS_TRACE_SERVER_STARTDRAW();
sl@0
    83
	WS_ASSERT_DEBUG(EPreparingDraw == iState,EWsPanicActiveScheduler);
sl@0
    84
	__DEBUG_ONLY(iState = EDrawing;)
sl@0
    85
	TTime now;
sl@0
    86
	now.UniversalTime();
sl@0
    87
	TUint64 duration = now.MicroSecondsFrom(iRunDraw).Int64();
sl@0
    88
	iPreparing += duration;
sl@0
    89
	iRunDraw = now;
sl@0
    90
	if(iData)
sl@0
    91
		{
sl@0
    92
		iData[iCurrent].iStart = iRunDraw;
sl@0
    93
		}
sl@0
    94
	iDraws++;
sl@0
    95
	}
sl@0
    96
sl@0
    97
void CWsActiveScheduler::DrawStats(TInt& aUpdatesPerSecond,TInt64& aPixelsPerSecond,TInt aWindowInSeconds) const
sl@0
    98
	{
sl@0
    99
	aUpdatesPerSecond=0;
sl@0
   100
	aPixelsPerSecond=0;
sl@0
   101
	if(iData)
sl@0
   102
		{
sl@0
   103
		// work out statistic
sl@0
   104
		TTime window;
sl@0
   105
		window.UniversalTime();
sl@0
   106
		window-=TTimeIntervalSeconds(aWindowInSeconds);
sl@0
   107
		TUint32 pixels=0;
sl@0
   108
		TInt64 duration=0;
sl@0
   109
		for(TInt ii=0;ii<iNumSamples;++ii)
sl@0
   110
			{
sl@0
   111
			if (iData[ii].iStart>=window)
sl@0
   112
				{
sl@0
   113
				pixels+=iData[ii].iPixels;
sl@0
   114
				duration+=iData[ii].iDuration;
sl@0
   115
				++aUpdatesPerSecond;
sl@0
   116
				}
sl@0
   117
			}
sl@0
   118
		if (aUpdatesPerSecond && duration)
sl@0
   119
			{
sl@0
   120
			aPixelsPerSecond=pixels;
sl@0
   121
			aPixelsPerSecond=(aPixelsPerSecond*1000000)/duration;
sl@0
   122
			aUpdatesPerSecond/=aWindowInSeconds;
sl@0
   123
			}
sl@0
   124
		}
sl@0
   125
	}
sl@0
   126
sl@0
   127
void CWsActiveScheduler::StopDraw(TInt aPixelsUpdated)
sl@0
   128
	{
sl@0
   129
	WS_TRACE_SERVER_STOPDRAW();
sl@0
   130
	WS_ASSERT_DEBUG(EDrawing == iState,EWsPanicActiveScheduler);
sl@0
   131
	__DEBUG_ONLY(iState = EDrawn;)
sl@0
   132
	TTime now;
sl@0
   133
	now.UniversalTime();
sl@0
   134
	TUint32 duration = now.MicroSecondsFrom(iRunDraw).Int64();
sl@0
   135
    // do recording
sl@0
   136
	if(iData)
sl@0
   137
		{
sl@0
   138
		iData[iCurrent].iDuration = now.MicroSecondsFrom(iData[iCurrent].iStart).Int64();
sl@0
   139
		iData[iCurrent].iPixels = aPixelsUpdated;
sl@0
   140
		if(++iCurrent >= iNumSamples)
sl@0
   141
			{
sl@0
   142
			iCurrent = 0;
sl@0
   143
			}
sl@0
   144
		}
sl@0
   145
	iDrawing += duration;
sl@0
   146
	}
sl@0
   147
sl@0
   148
void CWsActiveScheduler::WaitForAnyRequest()
sl@0
   149
	{
sl@0
   150
	WS_ASSERT_DEBUG((ENormal == iState)||(EDrawn == iState),EWsPanicActiveScheduler);
sl@0
   151
	TTime stop;
sl@0
   152
	stop.UniversalTime();
sl@0
   153
	iTotal += stop.MicroSecondsFrom(iRun).Int64();
sl@0
   154
	CActiveScheduler::WaitForAnyRequest();
sl@0
   155
	iRequests++;
sl@0
   156
	__DEBUG_ONLY(iState = ENormal;)
sl@0
   157
	iRun.UniversalTime();
sl@0
   158
	iIdle += iRun.MicroSecondsFrom(stop).Int64();
sl@0
   159
	}
sl@0
   160
sl@0
   161
	
sl@0
   162
void CWsActiveScheduler::Error(TInt /*aError*/) const
sl@0
   163
	{
sl@0
   164
	iErrors++;
sl@0
   165
	}
sl@0
   166
sl@0
   167
void CWsActiveScheduler::AccumReclaimedIdleTime(TInt64 aMicroSeconds)
sl@0
   168
	{
sl@0
   169
	iReclaimedIdleTime += aMicroSeconds;
sl@0
   170
	}
sl@0
   171
	
sl@0
   172
sl@0
   173
GLREF_C void StateDump(CWsRootWindow* aRootWindow);
sl@0
   174
GLREF_C void HeapDump();
sl@0
   175
sl@0
   176
// Static objects that need to be destroyed on exit
sl@0
   177
GLDEF_D CDebugLogBase *wsDebugLog=NULL;
sl@0
   178
GLDEF_D TInt wsDebugLogLevel=0;
sl@0
   179
GLDEF_D CIniFile *WsIniFile=NULL;
sl@0
   180
//
sl@0
   181
GLDEF_D TPtr nullDescriptor(NULL,0);
sl@0
   182
sl@0
   183
LOCAL_D CWsActiveScheduler *TheActiveScheduler;
sl@0
   184
#if defined(__WINS__)
sl@0
   185
LOCAL_D TPtrC shellCmd;
sl@0
   186
#endif
sl@0
   187
sl@0
   188
CScreen* CWsTop::iCurrentFocusScreen;
sl@0
   189
CArrayPtrFlat<CScreen>* CWsTop::iScreens ;
sl@0
   190
TInt CWsTop::iNumberOfScreens ;
sl@0
   191
CWsTop::CShellStarter* CWsTop::iShellStarter;
sl@0
   192
sl@0
   193
CWindowServer *CWsTop::iServer;
sl@0
   194
RLibrary CWsTop::iDebugLib;
sl@0
   195
CWsShellLogon *CWsTop::iShell;
sl@0
   196
const CWsClient *CWsTop::iShellClient;
sl@0
   197
TBool CWsTop::iPreviousShellClient=EFalse;
sl@0
   198
TInt CWsTop::iShellBootMode;
sl@0
   199
TBool CWsTop::iShuttingDown;
sl@0
   200
TBool CWsTop::iIsFirstLog=ETrue;
sl@0
   201
CWsWindowBase *CWsTop::iWindowToSendOffEventsTo=NULL;
sl@0
   202
RTimer CWsTop::iTimer;
sl@0
   203
TBool CWsTop::iIgnoreSwitchOffEvent=EFalse;
sl@0
   204
TBool CWsTop::iFadeEnabled=ETrue;
sl@0
   205
TBool CWsTop::iFinishEveryFlush=EFalse;
sl@0
   206
TBool CWsTop::iMultiFocusPolicy=EFalse;
sl@0
   207
#if defined(__WINS__)
sl@0
   208
TFullName WsSemName;
sl@0
   209
#endif
sl@0
   210
const CWsClient* CWsTop::iTriggerHeapCheckOnClientExit=NULL;
sl@0
   211
TWsCheckHeapOnDisconnectMode CWsTop::iHeapCheckMode=EWsCheckHeapOnDisconnectModeNone;
sl@0
   212
TInt CWsTop::iCheckHeapResult=KErrNotReady;
sl@0
   213
TBool CWsTop::iDoHeapCheckAndRestart=EFalse;
sl@0
   214
#define RFbsSession_SendCommand_ShutDownMessage 1	// A FBS message that is not published yet and probably never will be.
sl@0
   215
void CWsTop::DeleteStaticsL()
sl@0
   216
	{
sl@0
   217
	iShuttingDown=ETrue;
sl@0
   218
	CClick::DeleteStatics();
sl@0
   219
	WsPointer::DeleteStatics();
sl@0
   220
	CWsClient::DeleteStatics();
sl@0
   221
//
sl@0
   222
	delete iShellStarter;
sl@0
   223
	delete iServer;
sl@0
   224
sl@0
   225
	CWsBackedUpWindow::StaticDestroy();
sl@0
   226
	TWindowServerEvent::DeleteStatics();
sl@0
   227
	delete iShell;
sl@0
   228
	DisableLogging();
sl@0
   229
sl@0
   230
	CEventQueue::DeleteStaticsL();
sl@0
   231
		
sl@0
   232
	CWsSpriteBase::DeleteStatics();
sl@0
   233
	CWsGc::DeleteStatics();
sl@0
   234
	CPlaybackGc::DeleteStatics();
sl@0
   235
	CWsAnim::DeleteStatics();		//This destroys a GC so must be done while FBSERV is still around
sl@0
   236
	CWsAnimDll::DeleteStatics();
sl@0
   237
	CWsFontCache::DestroyInstance();
sl@0
   238
sl@0
   239
	iScreens->ResetAndDestroy() ;
sl@0
   240
	delete iScreens ;
sl@0
   241
sl@0
   242
	if (!iDoHeapCheckAndRestart)
sl@0
   243
		{
sl@0
   244
		RFbsSession::GetSession()->SendCommand(RFbsSession_SendCommand_ShutDownMessage);		//This line is just being tidy, never really does anything useful.
sl@0
   245
		}
sl@0
   246
sl@0
   247
	RFbsSession::Disconnect();
sl@0
   248
	iDebugLib.Close();
sl@0
   249
	iTimer.Close();
sl@0
   250
//
sl@0
   251
	delete WsIniFile;
sl@0
   252
	delete TheActiveScheduler;
sl@0
   253
	WsGraphicDrawer::FinalClose();
sl@0
   254
	}
sl@0
   255
sl@0
   256
RWsTextCursor *CWsTop::CurrentTextCursor()
sl@0
   257
	{
sl@0
   258
	return(FocusWindowGroup() ? FocusWindowGroup()->TextCursor():NULL);
sl@0
   259
	}
sl@0
   260
sl@0
   261
CWsClient *CWsTop::FocusWindowGroupOwner()
sl@0
   262
	{
sl@0
   263
	return(FocusWindowGroup() ? FocusWindowGroup()->WsOwner():NULL);
sl@0
   264
	}
sl@0
   265
sl@0
   266
void CWsTop::ClientDestroyed(const CWsClient *aClient)
sl@0
   267
	{
sl@0
   268
	if (aClient==iShellClient)
sl@0
   269
		iShellClient=NULL;
sl@0
   270
	}
sl@0
   271
sl@0
   272
void CWsTop::NewSession(const CWsClient *aClient)
sl@0
   273
	{
sl@0
   274
	if (iShellClient==NULL && iShell)
sl@0
   275
		{
sl@0
   276
#if defined(__WINS__)
sl@0
   277
		RThread proc;
sl@0
   278
		proc=aClient->Client();
sl@0
   279
#else
sl@0
   280
		RProcess proc;
sl@0
   281
		aClient->Client().Process(proc);
sl@0
   282
#endif
sl@0
   283
		TFullName procName = proc.FullName();
sl@0
   284
		// Before comparing the proc name with iShell name , truncate the proc name up to the actual name
sl@0
   285
		// referring to the process, by removing the part which starts with ':', if exists.
sl@0
   286
		TInt colonLocation = procName.Locate(':');
sl@0
   287
		if( KErrNotFound != colonLocation)
sl@0
   288
			{
sl@0
   289
			procName = procName.Left(colonLocation);
sl@0
   290
			}
sl@0
   291
		if (procName ==iShell->FullName())
sl@0
   292
			{
sl@0
   293
			iShellClient=aClient;
sl@0
   294
			if (!iPreviousShellClient)
sl@0
   295
				{
sl@0
   296
				iPreviousShellClient=ETrue;
sl@0
   297
				aClient->Screen()->RootWindow()->SetColorIfClear();
sl@0
   298
				}
sl@0
   299
			}
sl@0
   300
#if !defined(__WINS__)
sl@0
   301
		proc.Close();
sl@0
   302
#endif
sl@0
   303
		}	
sl@0
   304
	}
sl@0
   305
sl@0
   306
void CWsTop::SessionExited(CWsClient *aClient)
sl@0
   307
	{
sl@0
   308
	if (iShuttingDown)
sl@0
   309
		{
sl@0
   310
		RProcess proc;
sl@0
   311
		TInt err=aClient->Client().Process(proc);
sl@0
   312
		if (err==KErrNone && proc.Id()!=RProcess().Id())
sl@0
   313
			proc.Kill(0);
sl@0
   314
		else
sl@0
   315
			const_cast<RThread&>(aClient->Client()).Kill(0);
sl@0
   316
		if (err==KErrNone)
sl@0
   317
			proc.Close();
sl@0
   318
		}
sl@0
   319
	else if (iHeapCheckMode!=EWsCheckHeapOnDisconnectModeNone && aClient==iTriggerHeapCheckOnClientExit)
sl@0
   320
		{
sl@0
   321
		if (iHeapCheckMode==EWsCheckHeapOnDisconnectModeOnce)
sl@0
   322
			{
sl@0
   323
			iHeapCheckMode=EWsCheckHeapOnDisconnectModeNone;
sl@0
   324
			}
sl@0
   325
		iTriggerHeapCheckOnClientExit=NULL;
sl@0
   326
		iDoHeapCheckAndRestart=ETrue;
sl@0
   327
		Exit();
sl@0
   328
		}
sl@0
   329
	if (iServer->SessionCount()==1 && iShellBootMode==EShellBootModeNoReboot && iShell==NULL)
sl@0
   330
		StartShell();
sl@0
   331
	}
sl@0
   332
sl@0
   333
sl@0
   334
void CWsTop::RunServerL()
sl@0
   335
	{
sl@0
   336
	InitStaticsL();
sl@0
   337
	CWsMemoryManager* memMgr = CWsMemoryManager::NewLC();
sl@0
   338
	CActiveScheduler::Start();
sl@0
   339
	CleanupStack::PopAndDestroy(memMgr);
sl@0
   340
	DeleteStaticsL();
sl@0
   341
	}
sl@0
   342
sl@0
   343
void CWsTop::InitStaticsL()
sl@0
   344
	{
sl@0
   345
	iShuttingDown=EFalse;
sl@0
   346
	// By default shell should be started.
sl@0
   347
	TBool startShell = ETrue;
sl@0
   348
sl@0
   349
	// The windows server has been invoked.
sl@0
   350
	// This may have been from the system starter (via a
sl@0
   351
	// start up rss file)  
sl@0
   352
	// This block looks for a "-NoShell" argument in the invocation.
sl@0
   353
	// The existence of the argument means that shell does not need to be 
sl@0
   354
	// invoked from here because the new system starter
sl@0
   355
	// is in charge, and it will do it if required.
sl@0
   356
	
sl@0
   357
	_LIT(KNoShell,"-NOSHELL");
sl@0
   358
	
sl@0
   359
	TInt argLen = User::CommandLineLength();
sl@0
   360
	if(argLen)
sl@0
   361
		{
sl@0
   362
		HBufC* arg = HBufC::NewLC(argLen);
sl@0
   363
		TPtr argPtr = arg->Des();
sl@0
   364
		User::CommandLine(argPtr);
sl@0
   365
		argPtr.UpperCase();
sl@0
   366
sl@0
   367
		if(KErrNotFound != argPtr.Find(KNoShell))
sl@0
   368
			{
sl@0
   369
			// Don't start the shell. It will be started if required by 
sl@0
   370
			// the system starter.
sl@0
   371
			startShell = EFalse;
sl@0
   372
			}
sl@0
   373
		CleanupStack::PopAndDestroy(arg);
sl@0
   374
		}
sl@0
   375
		
sl@0
   376
	TheActiveScheduler=new(ELeave) CWsActiveScheduler;
sl@0
   377
	CActiveScheduler::Install(TheActiveScheduler);
sl@0
   378
	
sl@0
   379
//
sl@0
   380
	WsIniFile=CIniFile::NewL();
sl@0
   381
	_LIT(KWSERVIniFileVarLogEnable,"LOGENABLE");
sl@0
   382
	TInt loggingLevel;
sl@0
   383
	if (WsIniFile->FindVar(KWSERVIniFileVarLogEnable,loggingLevel))
sl@0
   384
		{
sl@0
   385
		EnableLogging(EDoNotReloadWsIni);
sl@0
   386
		if (wsDebugLog)
sl@0
   387
			{
sl@0
   388
			wsDebugLog->SetLoggingLevel(loggingLevel);
sl@0
   389
			}
sl@0
   390
		}
sl@0
   391
		
sl@0
   392
	_LIT(KWSERVIniFileVarFadeEnable,"FADEDISABLE");
sl@0
   393
	iFadeEnabled = !WsIniFile->FindVar(KWSERVIniFileVarFadeEnable);
sl@0
   394
sl@0
   395
	_LIT(KWSERVIniFileVarFinishEveryFlush,"FINISHEVERYFLUSH");
sl@0
   396
	iFinishEveryFlush = WsIniFile->FindVar(KWSERVIniFileVarFinishEveryFlush);
sl@0
   397
	
sl@0
   398
	_LIT(KWSERVIniFileVarSwitchOffEvent,"IGNORESWITCHOFFEVENT");
sl@0
   399
	iIgnoreSwitchOffEvent = WsIniFile->FindVar(KWSERVIniFileVarSwitchOffEvent);	
sl@0
   400
sl@0
   401
	iServer=CWindowServer::NewL();
sl@0
   402
	CClick::InitStaticsL();
sl@0
   403
	
sl@0
   404
	RProcess wservProc;
sl@0
   405
	if (!wservProc.DefaultDataPaged())
sl@0
   406
	{
sl@0
   407
		iServer->SetPinClientDescriptors(ETrue);
sl@0
   408
	}
sl@0
   409
//
sl@0
   410
	iServer->StartL(KWSERVServerName);
sl@0
   411
//
sl@0
   412
	User::LeaveIfError(FbsStartup());
sl@0
   413
	User::LeaveIfError(RFbsSession::Connect());
sl@0
   414
	User::LeaveIfError(iTimer.CreateLocal());
sl@0
   415
sl@0
   416
	TWindowServerEvent::InitStaticsL();
sl@0
   417
	//-------------------------------------------
sl@0
   418
	User::LeaveIfError(  HAL::Get( HAL::EDisplayNumberOfScreens, iNumberOfScreens ) ) ; 
sl@0
   419
	// Check that the INI file matches the HAL
sl@0
   420
	WS_ASSERT_ALWAYS(WsIniFile->NumberOfScreens()<=iNumberOfScreens, EWsPanicScreenInformationError);
sl@0
   421
sl@0
   422
	iScreens = new (ELeave) CArrayPtrFlat<CScreen>( iNumberOfScreens ) ;  //
sl@0
   423
	// now construct screens for as long as there is information
sl@0
   424
	
sl@0
   425
	TInt ii ;
sl@0
   426
	for ( ii = 0 ; ii < iNumberOfScreens ; ++ii )
sl@0
   427
		{
sl@0
   428
		InitScreenL( ii ) ;
sl@0
   429
		}
sl@0
   430
	//---------------------------------------------
sl@0
   431
	iCurrentFocusScreen = (*iScreens)[0] ;
sl@0
   432
sl@0
   433
	CWsFontCache::CreateInstanceL();
sl@0
   434
	CWsGc::InitStaticsL();
sl@0
   435
	CPlaybackGc::InitStaticsL();
sl@0
   436
	CWsSpriteBase::InitStaticsL();
sl@0
   437
	CEventQueue::InitStaticsL();
sl@0
   438
sl@0
   439
//
sl@0
   440
	CWsAnimDll::InitStaticsL();
sl@0
   441
	CWsAnim::InitStaticsL();
sl@0
   442
//
sl@0
   443
	TInt bootMode=0;
sl@0
   444
	_LIT(KWSERVIniFileVarReboot,"REBOOT");
sl@0
   445
	WsIniFile->FindVar(KWSERVIniFileVarReboot,bootMode);
sl@0
   446
	if (bootMode>=0 && bootMode<=2)
sl@0
   447
		iShellBootMode=bootMode;
sl@0
   448
//
sl@0
   449
	CWsBackedUpWindow::StaticInitL();
sl@0
   450
	CWsRedrawMsgWindow::StaticInitL();
sl@0
   451
//
sl@0
   452
	WsPointer::InitStaticsL();
sl@0
   453
	iShellStarter=new (ELeave) CShellStarter;
sl@0
   454
	iShellStarter->ConstructL();
sl@0
   455
	_LIT(KPreProcess,"REMOVEFADINGONFOCUSGAIN");
sl@0
   456
	CWsWindowGroup::SetFocusGainPreprocessing(WsIniFile->FindVar(KPreProcess));
sl@0
   457
	_LIT(KAbsFade,"ABSOLUTEFADING");
sl@0
   458
	CWsClientWindow::SetAbsoluteFading(WsIniFile->FindVar(KAbsFade));
sl@0
   459
sl@0
   460
//Set the focus policy
sl@0
   461
	_LIT(KFocusPolicy,"MULTIFOCUSPOLICY");
sl@0
   462
	if(WsIniFile->FindVar(KFocusPolicy))
sl@0
   463
		{
sl@0
   464
		iMultiFocusPolicy = ETrue;	
sl@0
   465
		}
sl@0
   466
	RProcess::Rendezvous(KErrNone);
sl@0
   467
	// Start the shell from here unless the 'NoShell' option has been 
sl@0
   468
	// received indicating that the system starter will start the shell directly.
sl@0
   469
	if(startShell)
sl@0
   470
		{
sl@0
   471
		StartShell();
sl@0
   472
		}
sl@0
   473
	UserSvr::WsRegisterSwitchOnScreenHandling(ETrue);
sl@0
   474
	}
sl@0
   475
sl@0
   476
sl@0
   477
void CWsTop::InitScreenL( TInt aScreenNumber)  // static
sl@0
   478
	{
sl@0
   479
	// create a new screen,  read ini file for aScreenNumber (this happens in CScreen - just need to pass a screen number from here
sl@0
   480
	CScreen* screen = new (ELeave) CScreen() ;
sl@0
   481
	CleanupStack::PushL( screen ) ;
sl@0
   482
	TRect digitiserArea;
sl@0
   483
	if (aScreenNumber==0)
sl@0
   484
		{
sl@0
   485
		TMachineInfoV1Buf machineBuf;
sl@0
   486
		UserHal::MachineInfo(machineBuf);
sl@0
   487
		TMachineInfoV1& machineInfo=*(TMachineInfoV1*)machineBuf.Ptr();
sl@0
   488
		digitiserArea = TRect(-machineInfo.iOffsetToDisplayInPixels,machineInfo.iXYInputSizeInPixels);
sl@0
   489
		}
sl@0
   490
	screen->ConstructL(digitiserArea, aScreenNumber);
sl@0
   491
	iScreens->AppendL( screen ) ; 
sl@0
   492
	CleanupStack::Pop( screen ) ;
sl@0
   493
	}
sl@0
   494
sl@0
   495
void CWsTop::UpdateAllScreens()
sl@0
   496
	{
sl@0
   497
	TInt ii;
sl@0
   498
	for (ii=0; ii<iNumberOfScreens; ++ii)
sl@0
   499
		{
sl@0
   500
		(*iScreens)[ii]->Update();
sl@0
   501
		}
sl@0
   502
	}
sl@0
   503
sl@0
   504
void CWsTop::ClearAllRedrawStores()
sl@0
   505
	{
sl@0
   506
	TInt ii;
sl@0
   507
	for (ii=0; ii<iNumberOfScreens; ++ii)
sl@0
   508
		{
sl@0
   509
		CScreen *screen=(*iScreens)[ii];
sl@0
   510
		CWsWindowGroup* group=screen->RootWindow()->Child();
sl@0
   511
		while (group!=NULL)
sl@0
   512
			{
sl@0
   513
			CWsWindowBase* win=group->Child();
sl@0
   514
			if (win)
sl@0
   515
				{
sl@0
   516
				while (win!=group)
sl@0
   517
					{
sl@0
   518
					static_cast<CWsWindow*>(win)->Redraw()->ClearRedrawStore(ETrue);
sl@0
   519
					if (win->BaseChild())
sl@0
   520
						win=win->BaseChild();
sl@0
   521
					else
sl@0
   522
						{
sl@0
   523
						do	
sl@0
   524
							{
sl@0
   525
							if (win->NextSibling())
sl@0
   526
								{
sl@0
   527
								win=win->NextSibling();
sl@0
   528
								break;
sl@0
   529
								}
sl@0
   530
							win=win->BaseParent();
sl@0
   531
							}
sl@0
   532
						while (win!=group);
sl@0
   533
						}
sl@0
   534
					}
sl@0
   535
				}
sl@0
   536
			group=group->NextSibling();
sl@0
   537
			}
sl@0
   538
		TriggerRedraws(screen->RootWindow());
sl@0
   539
		}
sl@0
   540
	}
sl@0
   541
sl@0
   542
void CWsTop::Exit()
sl@0
   543
	{
sl@0
   544
	CActiveScheduler::Stop();
sl@0
   545
	}
sl@0
   546
sl@0
   547
void CWsTop::TriggerRedraws(CWsRootWindow* aRootWindow)
sl@0
   548
	{
sl@0
   549
	for(CWsWindowGroup *group=aRootWindow->Child();group!=NULL;group=group->NextSibling())
sl@0
   550
		group->WsOwner()->TriggerRedraw();
sl@0
   551
	}
sl@0
   552
sl@0
   553
void CWsTop::StartShell()
sl@0
   554
	{
sl@0
   555
	TRAPD(err,iShell=new(ELeave) CWsShellLogon());
sl@0
   556
	if (err==KErrNone)
sl@0
   557
		{
sl@0
   558
		RFs fs;
sl@0
   559
		if ((err=fs.Connect())==KErrNone)
sl@0
   560
			{
sl@0
   561
			fs.SetNotifyUser(EFalse);
sl@0
   562
			TRAP(err,iShell->ConstructL(fs));
sl@0
   563
			fs.Close();
sl@0
   564
			}
sl@0
   565
		}
sl@0
   566
	if (err!=KErrNone)
sl@0
   567
		{
sl@0
   568
#ifdef _DEBUG
sl@0
   569
		RDebug::Print(_L("Failed to start shell: err=%d\n"),err);
sl@0
   570
#endif
sl@0
   571
		delete iShell;
sl@0
   572
		iShell=NULL;
sl@0
   573
		iShellStarter->After(TTimeIntervalMicroSeconds32(1000000));
sl@0
   574
		}
sl@0
   575
	}
sl@0
   576
sl@0
   577
void CWsTop::ShellExited()
sl@0
   578
	{
sl@0
   579
	delete iShell;
sl@0
   580
	iShell=NULL;
sl@0
   581
	switch(iShellBootMode)
sl@0
   582
		{
sl@0
   583
		case EShellBootModeReboot:
sl@0
   584
			StartShell();
sl@0
   585
			break;
sl@0
   586
		case EShellBootModeNoReboot:
sl@0
   587
			if (iServer->SessionCount()==0)
sl@0
   588
				StartShell();
sl@0
   589
			break;
sl@0
   590
		case EShellBootModeExit:
sl@0
   591
			Exit();
sl@0
   592
			break;
sl@0
   593
		}
sl@0
   594
	}
sl@0
   595
sl@0
   596
TInt CWsTop::SetSendOffEventsToShell(CWsClient *aClient,const TWsClCmdOffEventsToShell &aData)
sl@0
   597
	{
sl@0
   598
	CWsWindowBase *window=NULL;
sl@0
   599
	if (aData.window==0)
sl@0
   600
		{
sl@0
   601
		if (aData.on)
sl@0
   602
			aClient->PPanic(EWservPanicNoWindowSpecifed);
sl@0
   603
		}
sl@0
   604
	else
sl@0
   605
		aClient->HandleToWindow(aData.window,&window);
sl@0
   606
	if (aData.on)
sl@0
   607
		{
sl@0
   608
		if (iWindowToSendOffEventsTo!=NULL && aClient!=iShellClient)		//Allow the shell to pinch it
sl@0
   609
			return KErrAlreadyExists;
sl@0
   610
		iWindowToSendOffEventsTo=window;
sl@0
   611
		}
sl@0
   612
	else
sl@0
   613
		{
sl@0
   614
		if (iWindowToSendOffEventsTo==window || (!window && aClient==iWindowToSendOffEventsTo->WsOwner()))
sl@0
   615
			iWindowToSendOffEventsTo=NULL;
sl@0
   616
		}
sl@0
   617
	return KErrNone;
sl@0
   618
	}
sl@0
   619
sl@0
   620
void CWsTop::StopWindowGettingOffEvents(CWsWindowBase* aWindow)
sl@0
   621
	{
sl@0
   622
	if (aWindow==iWindowToSendOffEventsTo)
sl@0
   623
		iWindowToSendOffEventsTo=NULL;
sl@0
   624
	}
sl@0
   625
sl@0
   626
/** Routes "Off" events to the shutdown manager if one is registered. Othwerwise calls Power API's directly to
sl@0
   627
set the power mode of the device accordingly.
sl@0
   628
@internalTechnology
sl@0
   629
@released
sl@0
   630
@param aEvent Type of event that Wserv passes to client as OFF events. 
sl@0
   631
It can be EEventSwitchOff, EEventCaseClosed, EEventKeySwitchOff or EEventRestartSystem.
sl@0
   632
@param aDoSwitchOff ETrue if the switch-off should be performed when no shutdown manager is registered
sl@0
   633
and IGNORESWITCHOFFEVENT is not defined in wsini.ini
sl@0
   634
*/
sl@0
   635
void CWsTop::HandleSwitchOff(TInt aEvent,TBool aDoSwitchOff)
sl@0
   636
	{
sl@0
   637
	// If any shutdown manager is registered, forward the event to it
sl@0
   638
	if (iWindowToSendOffEventsTo && iWindowToSendOffEventsTo->QueueEvent(aEvent))
sl@0
   639
		return;
sl@0
   640
	// Otherwise, do the shutdown here
sl@0
   641
	if(!iIgnoreSwitchOffEvent && aDoSwitchOff)
sl@0
   642
		{
sl@0
   643
#ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP		
sl@0
   644
		// if LaF is not registered, restart or shutdown using Power API 
sl@0
   645
		if (aEvent == EEventRestartSystem) // restart
sl@0
   646
			{
sl@0
   647
			if (Power::EnableWakeupEvents(EPwRestart) == KErrNone)
sl@0
   648
				{
sl@0
   649
				// Should reboot/power-cycle the system, so no need to RequestWakeupEventNotification
sl@0
   650
				Power::PowerDown();
sl@0
   651
				}
sl@0
   652
			}
sl@0
   653
		else // everything else maps to standby for now
sl@0
   654
			{
sl@0
   655
			if (Power::EnableWakeupEvents(EPwStandby) == KErrNone)
sl@0
   656
				{
sl@0
   657
				// Prepare to wake up from standby
sl@0
   658
				TRequestStatus s;
sl@0
   659
				Power::RequestWakeupEventNotification(s);
sl@0
   660
				Power::PowerDown();
sl@0
   661
				User::WaitForRequest(s);
sl@0
   662
				}
sl@0
   663
			}
sl@0
   664
#else
sl@0
   665
		UserHal::SwitchOff();
sl@0
   666
#endif			
sl@0
   667
		}
sl@0
   668
	}
sl@0
   669
sl@0
   670
void CWsTop::RedrawScreens()
sl@0
   671
	{
sl@0
   672
	// apply to all screens
sl@0
   673
	TInt screenNo;
sl@0
   674
	for(screenNo=0; screenNo<iNumberOfScreens; ++screenNo)
sl@0
   675
		{
sl@0
   676
		CScreen* screen = Screen(screenNo);
sl@0
   677
		TRegionFix<1> screenArea(TRect(screen->DrawableArea()));
sl@0
   678
		screen->Update(screenArea);
sl@0
   679
		}
sl@0
   680
	}
sl@0
   681
sl@0
   682
void CWsTop::EnableLogging(TReloadWsIni aSetting)
sl@0
   683
	{
sl@0
   684
	TInt errorLoc=0;
sl@0
   685
	if (wsDebugLog)
sl@0
   686
		{
sl@0
   687
		if (wsDebugLogLevel==CDebugLogBase::ELogEverything)
sl@0
   688
			{
sl@0
   689
			wsDebugLogLevel=CDebugLogBase::ELogImportant;
sl@0
   690
			wsDebugLog->SetLoggingLevel(wsDebugLogLevel);
sl@0
   691
			return;
sl@0
   692
			}
sl@0
   693
		DisableLogging();
sl@0
   694
		}
sl@0
   695
sl@0
   696
     if (aSetting == EDoReloadWsIni)
sl@0
   697
        {
sl@0
   698
        CIniFile* temp = NULL;
sl@0
   699
        TRAPD(err, temp = CIniFile::NewL());
sl@0
   700
        if (err == KErrNone)
sl@0
   701
            {
sl@0
   702
            //loading successful, replace the previous one
sl@0
   703
            delete WsIniFile;
sl@0
   704
            WsIniFile = temp;
sl@0
   705
            }
sl@0
   706
        }
sl@0
   707
sl@0
   708
	TPtrC dlog;
sl@0
   709
	_LIT(KWSERVDebugLogFileName,"DLOG");
sl@0
   710
	TBuf<KMaxFileName> fname(KWSERVDebugLogFileName);
sl@0
   711
	_LIT(KWSERVIniFileVarLog,"LOG");
sl@0
   712
	if (WsIniFile->FindVar(KWSERVIniFileVarLog,dlog) && dlog.Length()==2)
sl@0
   713
		{
sl@0
   714
		fname.Append(dlog);
sl@0
   715
		TInt err=iDebugLib.Load(fname);
sl@0
   716
		if (err==KErrNone)
sl@0
   717
			{
sl@0
   718
			TUidType uid=iDebugLib.Type();
sl@0
   719
			if (uid[1]!=KWservLoggingDllUid)
sl@0
   720
				{
sl@0
   721
				err=KErrNotSupported;
sl@0
   722
				errorLoc=2;
sl@0
   723
				}
sl@0
   724
			else
sl@0
   725
				{
sl@0
   726
				TPtrC params;
sl@0
   727
				_LIT(KWSERVIniFileVarLogP,"LOGP");
sl@0
   728
				if (!WsIniFile->FindVar(KWSERVIniFileVarLogP,params))
sl@0
   729
					params.Set(NULL,0);
sl@0
   730
				CreateDebugLog func=(CreateDebugLog)iDebugLib.Lookup(1);
sl@0
   731
				if (func!=NULL)
sl@0
   732
					{
sl@0
   733
					TRAP(err,wsDebugLog=(*func)(iIsFirstLog, params));
sl@0
   734
					if (err==KErrNone)
sl@0
   735
						{
sl@0
   736
						iIsFirstLog=EFalse;
sl@0
   737
						wsDebugLogLevel=CDebugLogBase::ELogEverything;
sl@0
   738
						}
sl@0
   739
					else
sl@0
   740
						errorLoc=4;
sl@0
   741
					}
sl@0
   742
				else
sl@0
   743
					errorLoc=3;
sl@0
   744
				}
sl@0
   745
			}
sl@0
   746
		else
sl@0
   747
			errorLoc=1;
sl@0
   748
		if (iCurrentFocusScreen)
sl@0
   749
			{
sl@0
   750
			TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::ELogging,(errorLoc<<8)-err);
sl@0
   751
			}
sl@0
   752
		}
sl@0
   753
	}
sl@0
   754
sl@0
   755
void CWsTop::DisableLogging()
sl@0
   756
	{
sl@0
   757
	delete wsDebugLog;
sl@0
   758
	wsDebugLog=NULL;
sl@0
   759
	iDebugLib.Close();
sl@0
   760
	}
sl@0
   761
sl@0
   762
void CWsTop::LogCommand(RWsSession::TLoggingCommand aCommand)
sl@0
   763
	{
sl@0
   764
	switch(aCommand)
sl@0
   765
		{
sl@0
   766
	case RWsSession::ELoggingEnable:
sl@0
   767
		EnableLogging();
sl@0
   768
		break;
sl@0
   769
	case RWsSession::ELoggingDisable:
sl@0
   770
		DisableLogging();
sl@0
   771
		break;
sl@0
   772
	case RWsSession::ELoggingStatusDump:
sl@0
   773
		StateDump();
sl@0
   774
		break;
sl@0
   775
	case RWsSession::ELoggingHeapDump:
sl@0
   776
		HeapDump();
sl@0
   777
		break;
sl@0
   778
		}
sl@0
   779
	}
sl@0
   780
sl@0
   781
void CWsTop::StateDump()
sl@0
   782
	{
sl@0
   783
	TInt screenNo;
sl@0
   784
	for (screenNo=0; screenNo<iNumberOfScreens; ++screenNo)
sl@0
   785
		::StateDump( (*iScreens)[screenNo]->RootWindow());
sl@0
   786
	}
sl@0
   787
	
sl@0
   788
void CWsTop::SetCurrentFocusScreen(CScreen* aScreen)	
sl@0
   789
	{
sl@0
   790
	iCurrentFocusScreen = aScreen;
sl@0
   791
	}
sl@0
   792
sl@0
   793
TInt CWsTop::SetCurrentFocusScreen(TInt aScreenNo)
sl@0
   794
	{
sl@0
   795
	CScreen* newFocusScreen=CWsTop::Screen(aScreenNo);
sl@0
   796
	if (newFocusScreen==iCurrentFocusScreen)
sl@0
   797
		return KErrNone;
sl@0
   798
sl@0
   799
	CWsWindowGroup* newFocusGroup=newFocusScreen->FocusWindowGroup();
sl@0
   800
	if (!newFocusGroup)
sl@0
   801
		return KErrNotReady;
sl@0
   802
sl@0
   803
	CWsWindowGroup* oldFocusGroup=iCurrentFocusScreen->FocusWindowGroup();
sl@0
   804
	if (oldFocusGroup)
sl@0
   805
		oldFocusGroup->LostFocus();
sl@0
   806
	iCurrentFocusScreen=newFocusScreen;
sl@0
   807
	newFocusGroup->ReceivedFocus();
sl@0
   808
	WsPointer::UpdatePointerCursor();
sl@0
   809
	TWindowServerEvent::SendFocusChangedEvents();
sl@0
   810
	return KErrNone;
sl@0
   811
	}
sl@0
   812
sl@0
   813
void CWsTop::SetCheckHeapOnDisconnectClient(const CWsClient* aClient)
sl@0
   814
	{
sl@0
   815
	iTriggerHeapCheckOnClientExit=aClient;
sl@0
   816
	}
sl@0
   817
sl@0
   818
void CWsTop::SetCheckHeapOnDisconnectMode(TWsCheckHeapOnDisconnectMode aCheckHeapOnDisconnectMode)
sl@0
   819
	{
sl@0
   820
	iHeapCheckMode=aCheckHeapOnDisconnectMode;
sl@0
   821
	}
sl@0
   822
sl@0
   823
TBool CWsTop::NeedsHeapCheckAndRestart(TInt aStartHeapCount)
sl@0
   824
	{
sl@0
   825
	if (!iDoHeapCheckAndRestart)
sl@0
   826
		return(EFalse);
sl@0
   827
	iDoHeapCheckAndRestart=EFalse;
sl@0
   828
	iCheckHeapResult=User::Heap().Count()-aStartHeapCount;
sl@0
   829
	if(iCheckHeapResult > 0)
sl@0
   830
	    {
sl@0
   831
        const TUint32 orphanedCell = User::Heap().__DbgMarkEnd(aStartHeapCount);
sl@0
   832
        RDebug::Printf("Memory leak detected in wserv. First orphaned cell: 0x%8x", orphanedCell);
sl@0
   833
        }
sl@0
   834
	return(ETrue);;
sl@0
   835
	}
sl@0
   836
sl@0
   837
TInt CWsTop::FetchCheckHeapResult()
sl@0
   838
	{
sl@0
   839
	TInt ret=iCheckHeapResult;
sl@0
   840
	iCheckHeapResult=KErrNotReady;
sl@0
   841
	return(ret);
sl@0
   842
	}
sl@0
   843
sl@0
   844
/**
sl@0
   845
This function looks for memory which isn't essential to the correct running of the window server
sl@0
   846
and frees some of it.
sl@0
   847
Returns true if some memory was freed, and false if it was unable to free anything.
sl@0
   848
Called from the memory manager in low memory conditions.
sl@0
   849
*/
sl@0
   850
TBool CWsTop::ReleaseMemory()
sl@0
   851
	{
sl@0
   852
	return iServer->ReleaseMemory();
sl@0
   853
	}
sl@0
   854
	
sl@0
   855
TBool CWsTop::MultiFocusPolicy()
sl@0
   856
	{
sl@0
   857
	return	iMultiFocusPolicy;
sl@0
   858
	}
sl@0
   859
	
sl@0
   860
typedef TInt (*ShellEntryPoint)(TAny *);
sl@0
   861
sl@0
   862
#if defined(__WINS__)
sl@0
   863
LOCAL_D const TUint KHeapSize=0x10000;
sl@0
   864
LOCAL_D const TUint KMaxHeapSize=0x400000;
sl@0
   865
#endif
sl@0
   866
sl@0
   867
CWsShellLogon::CWsShellLogon() : CActive(EWsShellLogonPriority)
sl@0
   868
	{
sl@0
   869
#if defined (__WINS__)
sl@0
   870
	// Clear handles to NULL so we can later detect which one gets used
sl@0
   871
	iShellThread.SetHandle(NULL);
sl@0
   872
	iShellProcess.SetHandle(NULL);
sl@0
   873
#endif
sl@0
   874
	}
sl@0
   875
sl@0
   876
CWsShellLogon::~CWsShellLogon()
sl@0
   877
	{
sl@0
   878
	Cancel();
sl@0
   879
#if defined (__WINS__)
sl@0
   880
	iShellThread.Close();
sl@0
   881
	iShellProcess.Close();
sl@0
   882
#else
sl@0
   883
	iShell.Close();
sl@0
   884
#endif
sl@0
   885
#if defined(__WINS__)
sl@0
   886
	iLib.Close();
sl@0
   887
#endif
sl@0
   888
	}
sl@0
   889
sl@0
   890
void CWsShellLogon::ConstructL(RFs &)
sl@0
   891
	{
sl@0
   892
	CActiveScheduler::Add(this);
sl@0
   893
#if !defined(__WINS__)
sl@0
   894
	TPtrC shellCmd;
sl@0
   895
#endif
sl@0
   896
	_LIT(KWSERVIniFileVarShellCmd,"SHELLCMD");
sl@0
   897
	WsIniFile->FindVar(KWSERVIniFileVarShellCmd,shellCmd);
sl@0
   898
	_LIT(KWSERVShellName,"SHELL");
sl@0
   899
	TPtrC startUp(KWSERVShellName);
sl@0
   900
	_LIT(KWSERVIniFileVarStartUp,"STARTUP");
sl@0
   901
	WsIniFile->FindVar(KWSERVIniFileVarStartUp,startUp);
sl@0
   902
sl@0
   903
sl@0
   904
	TParse fileName;
sl@0
   905
	_LIT(KWSERVArmShellLocationPattern,"Z:\\sys\\bin\\.EXE");
sl@0
   906
	User::LeaveIfError(fileName.SetNoWild(startUp,&KWSERVArmShellLocationPattern,NULL));
sl@0
   907
#if defined(__WINS__)
sl@0
   908
	TInt err=iShellProcess.Create(fileName.FullName(),shellCmd);
sl@0
   909
	if (err == KErrNone)
sl@0
   910
		{
sl@0
   911
		Request();
sl@0
   912
		iShellProcess.Resume();
sl@0
   913
		}
sl@0
   914
	else
sl@0
   915
		{
sl@0
   916
		// Try loading the matching DLL name instead?
sl@0
   917
		_LIT(KWSERVShellExtension,"Z:.DLL");
sl@0
   918
		User::LeaveIfError(fileName.Set(KWSERVShellExtension,&startUp,NULL));
sl@0
   919
		User::LeaveIfError(iLib.Load(fileName.FullName()));
sl@0
   920
		ShellEntryPoint libFunc=(ShellEntryPoint)iLib.Lookup(1);
sl@0
   921
		if (!libFunc)
sl@0
   922
			User::Leave(KErrGeneral);
sl@0
   923
		TBuf<256> name;
sl@0
   924
		TInt num=0;
sl@0
   925
		TInt ret=KErrNone;
sl@0
   926
		do
sl@0
   927
			{
sl@0
   928
			_LIT(KWSERVWinsShellInstanceName,"Shell%d");
sl@0
   929
			name.Format(KWSERVWinsShellInstanceName,num++);
sl@0
   930
			ret=iShellThread.Create(name,libFunc,KDefaultStackSize,&shellCmd,&iLib,NULL,KHeapSize,KMaxHeapSize,EOwnerProcess);
sl@0
   931
			} while(ret==KErrAlreadyExists);
sl@0
   932
		User::LeaveIfError(ret);
sl@0
   933
		Request();
sl@0
   934
		iShellThread.Resume();
sl@0
   935
		}
sl@0
   936
#else // not __WINS__
sl@0
   937
	User::LeaveIfError(iShell.Create(fileName.FullName(),shellCmd));
sl@0
   938
	Request();
sl@0
   939
	iShell.Resume();
sl@0
   940
#endif	
sl@0
   941
	return;
sl@0
   942
sl@0
   943
	}
sl@0
   944
sl@0
   945
void CWsShellLogon::DoCancel()
sl@0
   946
	{
sl@0
   947
#if defined (__WINS__)
sl@0
   948
	if(iShellThread.Handle())
sl@0
   949
		iShellThread.LogonCancel(iStatus);
sl@0
   950
	else
sl@0
   951
		iShellProcess.LogonCancel(iStatus);
sl@0
   952
#else
sl@0
   953
	iShell.LogonCancel(iStatus);
sl@0
   954
#endif
sl@0
   955
	}
sl@0
   956
sl@0
   957
void CWsShellLogon::RunL()
sl@0
   958
	{
sl@0
   959
	if (iStatus.Int()!=KErrCancel)
sl@0
   960
		CWsTop::ShellExited();
sl@0
   961
	}
sl@0
   962
sl@0
   963
void CWsShellLogon::Request()
sl@0
   964
	{
sl@0
   965
#if defined (__WINS__)
sl@0
   966
	if(iShellThread.Handle())
sl@0
   967
		iShellThread.Logon(iStatus);
sl@0
   968
	else
sl@0
   969
		iShellProcess.Logon(iStatus);
sl@0
   970
#else
sl@0
   971
	iShell.Logon(iStatus);
sl@0
   972
#endif
sl@0
   973
	SetActive();
sl@0
   974
	}
sl@0
   975
sl@0
   976
#if defined(__WINS__) 
sl@0
   977
TFullName CWsShellLogon::FullName()
sl@0
   978
	{
sl@0
   979
	if(iShellThread.Handle())
sl@0
   980
		return(iShellThread.FullName());
sl@0
   981
	else
sl@0
   982
		return(iShellProcess.FullName());
sl@0
   983
	}
sl@0
   984
#else
sl@0
   985
TFullName CWsShellLogon::FullName()
sl@0
   986
	{return(iShell.FullName());}
sl@0
   987
#endif
sl@0
   988
sl@0
   989
sl@0
   990
sl@0
   991
TInt E32Main()
sl@0
   992
	{
sl@0
   993
	__UHEAP_MARK;
sl@0
   994
sl@0
   995
	UserSvr::WsRegisterThread();
sl@0
   996
sl@0
   997
	RThread thread;
sl@0
   998
	// Set wserv's main thread to system permanent
sl@0
   999
	TInt err;
sl@0
  1000
	err = User::SetCritical(User::ESystemPermanent);
sl@0
  1001
	if (err!=KErrNone)
sl@0
  1002
		{
sl@0
  1003
		WS_PANIC_ALWAYS(EWsPanicFailedToSetThread);
sl@0
  1004
		}
sl@0
  1005
// If in the future wserv becomes multi-threaded,
sl@0
  1006
// we can uncomment the following lines to set any new threads to be system permanent as well.
sl@0
  1007
//	err = User::SetProcessCritical(User::ESystemPermanent); 
sl@0
  1008
//	if (err!=KErrNone)
sl@0
  1009
//		{
sl@0
  1010
//		WS_PANIC_ALWAYS(EWsPanicFailedToSetProcess);
sl@0
  1011
//		}
sl@0
  1012
sl@0
  1013
	thread.SetPriority(EPriorityMore);
sl@0
  1014
	err = User::RenameThread(KWSERVThreadName);
sl@0
  1015
	if (err==KErrNone)
sl@0
  1016
		{
sl@0
  1017
		thread.Close();
sl@0
  1018
		}
sl@0
  1019
	else
sl@0
  1020
		{
sl@0
  1021
		return err;
sl@0
  1022
		}
sl@0
  1023
	TInt startCount;
sl@0
  1024
	do
sl@0
  1025
		{
sl@0
  1026
		User::Heap().__DbgMarkStart();
sl@0
  1027
		startCount=User::Heap().Count();
sl@0
  1028
		CTrapCleanup* CleanUpStack=CTrapCleanup::New();
sl@0
  1029
		TRAP(err,CWsTop::RunServerL());
sl@0
  1030
		if (err!=KErrNone)
sl@0
  1031
			{
sl@0
  1032
			WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
sl@0
  1033
			}
sl@0
  1034
		UserSvr::ReleaseEventHook();
sl@0
  1035
		delete CleanUpStack;
sl@0
  1036
		} while	(CWsTop::NeedsHeapCheckAndRestart(startCount));
sl@0
  1037
sl@0
  1038
	__UHEAP_MARKEND;
sl@0
  1039
sl@0
  1040
	return(err);
sl@0
  1041
	}
sl@0
  1042
sl@0
  1043