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