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