First public contribution.
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Top level window server code
21 #include <graphics/fbsdefs.h>
25 #include "playbackgc.h"
27 #include "windowgroup.h"
33 #include "WSGRAPHICDRAWERFACTORY.H"
34 #include "redrawmsgwindow.h"
35 #include <graphics/WSGRAPHICDRAWERINTERFACE.H>
37 #include "backedupwindow.h"
39 #include "registeredsurfacemap.h"
42 #ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP
46 #include "registeredsurfacemap.h"
47 #include "windowelementset.h"
48 #include "wspluginmanager.h"
49 #include "renderorientationtracker.h"
51 // IDs of p&s properties that optionally contain callbacks that may be used
52 // to release singletons owned by libraries at shutdown in order to make
53 // the memory leak tests work.
54 // By convention, the ID is the same as the UID3 of the libary.
55 static TBool gReleaseSingletonsOnExit = EFalse;
56 static const TUid KOpenWfcImplCleanupKey = {0x10286FC4};
57 static const TUid KOpenWfcInteropCleanupKey = {0x10286FC5};
59 typedef CDebugLogBase *(*CreateDebugLog)(TBool aIsFirst, TDesC &aParams);
61 CWsActiveScheduler* CWsActiveScheduler::Static()
63 return static_cast<CWsActiveScheduler*>(CActiveScheduler::Current());
66 CWsActiveScheduler::CWsActiveScheduler(): iNumSamples(100)
68 iRun = User::FastCounter();
69 if((KErrNone == HAL::Get(HALData::EFastCounterFrequency,iFastCounterFreq)) && iFastCounterFreq)
71 iData = reinterpret_cast<TSample*>(User::AllocZ(sizeof(TSample)*iNumSamples));
75 CWsActiveScheduler::~CWsActiveScheduler()
77 User::FreeZ(reinterpret_cast<void*&>(iData));
79 void CWsActiveScheduler::PrepareDraw()
81 iRunDraw = User::FastCounter();
84 void CWsActiveScheduler::CancelPrepare()
88 void CWsActiveScheduler::StartDraw()
92 TUint64 duration = now.MicroSecondsFrom(iRunDraw).Int64();
93 iPreparing += duration;
97 iData[iCurrent].iStart = iRunDraw;
102 void CWsActiveScheduler::DrawStats(TInt& aUpdatesPerSecond,TInt64& aPixelsPerSecond,TInt aWindowInSeconds) const
104 aUpdatesPerSecond = 0;
105 aPixelsPerSecond = 0;
108 // work out statistic
110 window.UniversalTime();
111 window -= TTimeIntervalSeconds(aWindowInSeconds);
114 for(TUint i=0; i<iNumSamples; i++)
116 if(iData[i].iStart >= window)
118 pixels += iData[i].iPixels;
119 duration += iData[i].iDuration;
123 if(aUpdatesPerSecond && duration)
125 const TReal pixelsPerMicroSecond = (static_cast<TReal>(pixels) / static_cast<TReal>(duration));
126 aPixelsPerSecond = static_cast<TInt64>(pixelsPerMicroSecond * 1000000.0);
127 aUpdatesPerSecond /= aWindowInSeconds;
132 void CWsActiveScheduler::StopDraw(TInt aPixelsUpdated)
136 TUint32 duration = now.MicroSecondsFrom(iRunDraw).Int64();
140 iData[iCurrent].iDuration = now.MicroSecondsFrom(iData[iCurrent].iStart).Int64();
141 iData[iCurrent].iPixels = aPixelsUpdated;
142 if(++iCurrent >= iNumSamples)
147 iDrawing += duration;
150 void CWsActiveScheduler::WaitForAnyRequest()
153 stop.UniversalTime();
154 iTotal += stop.MicroSecondsFrom(iRun).Int64();
155 CActiveScheduler::WaitForAnyRequest();
157 iRun.UniversalTime();
158 iIdle += iRun.MicroSecondsFrom(stop).Int64();
162 void CWsActiveScheduler::Error(TInt /*aError*/) const
167 void CWsActiveScheduler::AccumReclaimedIdleTime(TInt64 aMicroSeconds)
169 iReclaimedIdleTime += aMicroSeconds;
173 GLREF_C void StateDump(CWsRootWindow* aRootWindow);
174 GLREF_C void HeapDump();
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;
181 GLDEF_D TPtr nullDescriptor(NULL,0);
183 LOCAL_D CWsActiveScheduler *TheActiveScheduler;
185 CScreen* CWsTop::iCurrentFocusScreen;
186 CArrayPtrFlat<CScreen>* CWsTop::iScreens ;
187 TInt CWsTop::iNumberOfScreens ;
188 CWsTop::CShellStarter* CWsTop::iShellStarter;
190 CWsPluginManager *CWsTop::iPluginManager;
191 CWindowServer *CWsTop::iServer;
192 RLibrary CWsTop::iDebugLib;
193 CWsShellLogon *CWsTop::iShell;
194 const CWsClient *CWsTop::iShellClient;
195 TBool CWsTop::iPreviousShellClient=EFalse;
196 TInt CWsTop::iShellBootMode;
197 TBool CWsTop::iShuttingDown;
198 TBool CWsTop::iIsFirstLog=ETrue;
199 CWsWindowBase *CWsTop::iWindowToSendOffEventsTo=NULL;
200 RTimer CWsTop::iTimer;
201 TBool CWsTop::iIgnoreSwitchOffEvent=EFalse;
202 TBool CWsTop::iFadeEnabled=ETrue;
203 TBool CWsTop::iFinishEveryFlush=EFalse;
204 TBool CWsTop::iMultiFocusPolicy=EFalse;
205 #if defined(__WINS__)
208 const CWsClient* CWsTop::iTriggerHeapCheckOnClientExit=NULL;
209 TWsCheckHeapOnDisconnectMode CWsTop::iHeapCheckMode=EWsCheckHeapOnDisconnectModeNone;
210 TInt CWsTop::iCheckHeapResult=KErrNotReady;
211 TBool CWsTop::iDoHeapCheckAndRestart=EFalse;
212 #define RFbsSession_SendCommand_ShutDownMessage 1 // A FBS message that is not published yet and probably never will be.
213 CWsRenderOrienationTracker* CWsTop::iRenderOrientationTracker=NULL;
215 static void DefineSingletonKey(const TUid& aSingletonKey)
217 * Defines a new property for a singleton key. WServ must only process
218 * singleton keys that it created to prevent a malicious process with the
219 * WriteDeviceData capability causing arbitrary functions to be executed.
221 * @param aSingeltonKey The UID of the singleton key to define.
225 TUid category = { t.SecureId().iId };
228 // Write access is restricted to THIS process
229 TInt err = prop.Define( category, aSingletonKey.iUid,
230 RProperty::EByteArray, TSecurityPolicy( t.SecureId() ),
231 TSecurityPolicy( t.SecureId() ), sizeof( TCallBack ) );
233 if ( err == KErrNone )
235 TCallBack cb( NULL, NULL );
236 TPckgC<TCallBack> cbPckg( cb );
238 // Any error should cause the properties to be ignored
239 err = prop.Set( category, aSingletonKey.iUid, cbPckg );
242 if ( err != KErrNone )
244 // A problem occured / the property already existed so for safety
245 // the release code should be skipped.
246 gReleaseSingletonsOnExit = EFalse;
253 static void DeleteSingleton( const TUid& aSingletonKey )
255 * Deletes a singleton object that was created on WServ's main heap.
257 * @pre The ws plugins have not been unloaded.
258 * @param aSingletonKey The UID of the singleton which correponds to an
259 * RProperty within WServ's category.
262 if ( gReleaseSingletonsOnExit )
265 TPckgBuf<TCallBack> cb;
267 TInt err = prop.Get(TUid::Uid(t.SecureId().iId), aSingletonKey.iUid, cb);
268 if (err == KErrNone && cb.Length() == sizeof(TCallBack) &&
269 cb().iFunction && cb().iPtr == &User::Heap())
271 // Callback is only invoked if the heap for the singleton was the
272 // WServ heap because the WServ memory leak tests only check this
276 // Errors are ignored because the purpose of this function is to free
277 // singletons in order top make memory leak checks pass.
283 void CWsTop::DeleteStaticsL()
286 delete iRenderOrientationTracker;
288 CClick::DeleteStatics();
290 CWsClient::DeleteStatics();
292 delete iShellStarter;
293 iServer->DestroySessionsForShutdown();
295 CWsBackedUpWindow::StaticDestroy();
299 CEventQueue::DeleteStaticsL();
301 CWsSpriteBase::DeleteStatics();
302 CWsGc::DeleteStatics();
303 CPlaybackGc::DeleteStatics();
304 CWsAnimDll::DeleteStatics();
305 CWsFontCache::DestroyInstance();
307 iScreens->ResetAndDestroy();
311 TWindowServerEvent::DeleteStatics();
313 // Free singletons on WServ heap created by libraries. Must be called
314 // BEFORE iPluginManager is deleted otherwise the library code could have
316 DeleteSingleton(KOpenWfcImplCleanupKey);
317 DeleteSingleton(KOpenWfcInteropCleanupKey);
320 if (!iDoHeapCheckAndRestart)
322 RFbsSession::GetSession()->SendCommand(RFbsSession_SendCommand_ShutDownMessage); //This line is just being tidy, never really does anything useful.
325 RFbsSession::Disconnect();
328 TWsPointer::DeleteStatics();
329 delete TheActiveScheduler;
331 // iPluginManager is not created in InitStaticsL because plugins need to be loaded
332 // before E32Main's loop for EcomSession allocations issues
333 delete iPluginManager;
336 RWsTextCursor *CWsTop::CurrentTextCursor()
338 if(iCurrentFocusScreen && FocusWindowGroup())
340 return FocusWindowGroup()->TextCursor();
348 CWsClient *CWsTop::FocusWindowGroupOwner()
350 return(FocusWindowGroup() ? FocusWindowGroup()->WsOwner():NULL);
353 void CWsTop::ClientDestroyed(const CWsClient *aClient)
355 if (aClient==iShellClient)
359 void CWsTop::NewSession(const CWsClient *aClient)
361 if (iShellClient==NULL && iShell)
363 #if defined(__WINS__)
365 proc=aClient->Client();
368 aClient->Client().Process(proc);
370 TFullName procName = proc.FullName();
371 // Before comparing the proc name with iShell name , truncate the proc name up to the actual name
372 // referring to the process, by removing the part which starts with ':', if exists.
373 TInt colonLocation = procName.Locate(':');
374 if( KErrNotFound != colonLocation)
376 procName = procName.Left(colonLocation);
378 if (procName ==iShell->FullName())
380 iShellClient=aClient;
381 if (!iPreviousShellClient)
383 iPreviousShellClient=ETrue;
384 aClient->Screen()->RootWindow()->SetColorIfClear();
387 #if !defined(__WINS__)
393 void CWsTop::SessionExited(CWsClient *aClient)
398 TInt err=aClient->Client().Process(proc);
399 if (err==KErrNone && proc.Id()!=RProcess().Id())
402 const_cast<RThread&>(aClient->Client()).Kill(0);
406 else if (iHeapCheckMode!=EWsCheckHeapOnDisconnectModeNone && aClient==iTriggerHeapCheckOnClientExit)
408 if (iHeapCheckMode==EWsCheckHeapOnDisconnectModeOnce)
410 iHeapCheckMode=EWsCheckHeapOnDisconnectModeNone;
412 iTriggerHeapCheckOnClientExit=NULL;
413 iDoHeapCheckAndRestart=ETrue;
416 if (iServer->SessionCount()==1 && iShellBootMode==EShellBootModeNoReboot && iShell==NULL)
421 void CWsTop::RunServerL()
424 CWsMemoryManager* memMgr = CWsMemoryManager::NewLC();
425 CActiveScheduler::Start();
426 CleanupStack::PopAndDestroy(memMgr);
430 void CWsTop::InitStaticsL()
432 iShuttingDown=EFalse;
433 // By default shell should be started.
434 TBool startShell = ETrue;
436 iCurrentFocusScreen = 0;
438 // The windows server has been invoked.
439 // This may have been from the system starter (via a
440 // start up rss file)
441 // This block looks for a "-NoShell" argument in the invocation.
442 // The existence of the argument means that shell does not need to be
443 // invoked from here because the new system starter
444 // is in charge, and it will do it if required.
446 _LIT(KNoShell,"-NOSHELL");
448 TInt argLen = User::CommandLineLength();
451 HBufC* arg = HBufC::NewLC(argLen);
452 TPtr argPtr = arg->Des();
453 User::CommandLine(argPtr);
456 if(KErrNotFound != argPtr.Find(KNoShell))
458 // Don't start the shell. It will be started if required by
459 // the system starter.
462 CleanupStack::PopAndDestroy(arg);
465 TheActiveScheduler=new(ELeave) CWsActiveScheduler;
466 CActiveScheduler::Install(TheActiveScheduler);
467 // WsIniFile is already created (before E32Main's Loop)
468 _LIT(KWSERVIniFileVarLogEnable,"LOGENABLE");
470 if (WsIniFile->FindVar(KWSERVIniFileVarLogEnable,loggingLevel))
472 EnableLogging(EDoNotReloadWsIni);
475 wsDebugLog->SetLoggingLevel(loggingLevel);
479 _LIT(KWSERVIniFileVarFadeEnable,"FADEDISABLE");
480 iFadeEnabled = !WsIniFile->FindVar(KWSERVIniFileVarFadeEnable);
482 _LIT(KWSERVIniFileVarFinishEveryFlush,"FINISHEVERYFLUSH");
483 iFinishEveryFlush = WsIniFile->FindVar(KWSERVIniFileVarFinishEveryFlush);
485 _LIT(KWSERVIniFileVarSwitchOffEvent,"IGNORESWITCHOFFEVENT");
486 iIgnoreSwitchOffEvent = WsIniFile->FindVar(KWSERVIniFileVarSwitchOffEvent);
488 iPluginManager = CWsPluginManager::NewL(); // need to be constructed before iServer!
489 iServer=CWindowServer::NewL();
490 CClick::InitStaticsL();
493 if (!wservProc.DefaultDataPaged())
495 iServer->SetPinClientDescriptors(ETrue);
497 User::LeaveIfError(FbsStartup());
498 User::LeaveIfError(RFbsSession::Connect());
499 User::LeaveIfError(iTimer.CreateLocal());
501 TWindowServerEvent::InitStaticsL();
502 CWsClient::InitStaticsL();
503 //-------------------------------------------
504 User::LeaveIfError( HAL::Get( HAL::EDisplayNumberOfScreens, iNumberOfScreens ) ) ;
505 // Check that the INI file matches the HAL
506 WS_ASSERT_ALWAYS(WsIniFile->NumberOfScreens()<=iNumberOfScreens, EWsPanicScreenInformationError);
508 iScreens = new (ELeave) CArrayPtrFlat<CScreen>( iNumberOfScreens ) ; //
509 // now construct screens for as long as there is information
512 for ( ii = 0 ; ii < iNumberOfScreens ; ++ii )
516 //---------------------------------------------
517 iCurrentFocusScreen = (*iScreens)[0] ;
520 CWsFontCache::CreateInstanceL();
521 CWsGc::InitStaticsL();
522 CPlaybackGc::InitStaticsL();
523 CWsSpriteBase::InitStaticsL();
524 CEventQueue::InitStaticsL();
527 CWsAnimDll::InitStaticsL();
530 _LIT(KWSERVIniFileVarReboot,"REBOOT");
531 WsIniFile->FindVar(KWSERVIniFileVarReboot,bootMode);
532 if (bootMode>=0 && bootMode<=2)
533 iShellBootMode=bootMode;
535 CWsBackedUpWindow::StaticInitL();
536 CWsRedrawMsgWindow::StaticInitL();
538 TWsPointer::InitStaticsL();
539 iShellStarter=new (ELeave) CShellStarter;
540 iShellStarter->ConstructL();
541 _LIT(KPreProcess,"REMOVEFADINGONFOCUSGAIN");
542 CWsWindowGroup::SetFocusGainPreprocessing(WsIniFile->FindVar(KPreProcess));
543 _LIT(KAbsFade,"ABSOLUTEFADING");
544 CWsClientWindow::SetAbsoluteFading(WsIniFile->FindVar(KAbsFade));
546 //Set the focus policy
547 _LIT(KFocusPolicy,"MULTIFOCUSPOLICY");
548 if(WsIniFile->FindVar(KFocusPolicy))
550 iMultiFocusPolicy = ETrue;
552 RProcess::Rendezvous(KErrNone);
553 // Start the shell from here unless the 'NoShell' option has been
554 // received indicating that the system starter will start the shell directly.
559 UserSvr::WsRegisterSwitchOnScreenHandling(ETrue);
561 iRenderOrientationTracker = CWsRenderOrienationTracker::NewL();
565 void CWsTop::InitScreenL( TInt aScreenNumber) // static
567 // create a new screen, read ini file for aScreenNumber (this happens in CScreen - just need to pass a screen number from here
568 CScreen* screen = new (ELeave) CScreen() ;
569 CleanupStack::PushL( screen ) ;
571 if (aScreenNumber==0)
573 TMachineInfoV1Buf machineBuf;
574 UserHal::MachineInfo(machineBuf);
575 TMachineInfoV1& machineInfo=*(TMachineInfoV1*)machineBuf.Ptr();
576 digitiserArea = TRect(-machineInfo.iOffsetToDisplayInPixels,machineInfo.iXYInputSizeInPixels);
578 screen->ConstructL(digitiserArea, aScreenNumber);
579 iScreens->AppendL( screen ) ;
580 CleanupStack::Pop( screen ) ;
583 void CWsTop::ClearAllRedrawStores()
586 if (iScreens && iScreens->Count()>=iNumberOfScreens)
587 { //All screens are initialized
588 for (ii=0; ii<iNumberOfScreens; ++ii)
590 CScreen *screen=(*iScreens)[ii];
591 CWsWindowGroup* group=screen->RootWindow()->Child();
594 CWsWindowBase* win=group->Child();
599 static_cast<CWsWindow*>(win)->Redraw()->ClearRedrawStore(ETrue);
600 //On orientation switch, window is no longer present on screen
601 static_cast<CWsWindow*>(win)->SetDrawnToScreen(EFalse);
602 if (win->BaseChild())
603 win=win->BaseChild();
608 if (win->NextSibling())
610 win=win->NextSibling();
613 win=win->BaseParent();
619 group=group->NextSibling();
621 TriggerRedraws(screen->RootWindow());
628 CActiveScheduler::Stop();
631 void CWsTop::TriggerRedraws(CWsRootWindow* aRootWindow)
633 for(CWsWindowGroup *group=aRootWindow->Child();group!=NULL;group=group->NextSibling())
634 group->WsOwner()->TriggerRedraw();
637 void CWsTop::StartShell()
639 TRAPD(err,iShell=new(ELeave) CWsShellLogon());
643 if ((err=fs.Connect())==KErrNone)
645 fs.SetNotifyUser(EFalse);
646 TRAP(err,iShell->ConstructL(fs));
653 RDebug::Print(_L("Failed to start shell: err=%d\n"),err);
657 iShellStarter->After(TTimeIntervalMicroSeconds32(1000000));
661 void CWsTop::ShellExited()
665 switch(iShellBootMode)
667 case EShellBootModeReboot:
670 case EShellBootModeNoReboot:
671 if (iServer->SessionCount()==0)
674 case EShellBootModeExit:
680 TInt CWsTop::SetSendOffEventsToShell(CWsClient *aClient,const TWsClCmdOffEventsToShell &aData)
682 CWsWindowBase *window=NULL;
686 aClient->PPanic(EWservPanicNoWindowSpecifed);
689 aClient->HandleToWindow(aData.window,&window);
692 if (iWindowToSendOffEventsTo!=NULL && aClient!=iShellClient) //Allow the shell to pinch it
693 return KErrAlreadyExists;
694 iWindowToSendOffEventsTo=window;
698 if (iWindowToSendOffEventsTo==window || (!window && aClient==iWindowToSendOffEventsTo->WsOwner()))
699 iWindowToSendOffEventsTo=NULL;
704 void CWsTop::StopWindowGettingOffEvents(CWsWindowBase* aWindow)
706 if (aWindow==iWindowToSendOffEventsTo)
707 iWindowToSendOffEventsTo=NULL;
710 /** Routes "Off" events to the shutdown manager if one is registered. Othwerwise calls Power API's directly to
711 set the power mode of the device accordingly.
714 @param aEvent Type of event that Wserv passes to client as OFF events.
715 It can be EEventSwitchOff, EEventCaseClosed, EEventKeySwitchOff or EEventRestartSystem.
716 @param aDoSwitchOff ETrue if the switch-off should be performed when no shutdown manager is registered
717 and IGNORESWITCHOFFEVENT is not defined in wsini.ini
719 void CWsTop::HandleSwitchOff(TInt aEvent,TBool aDoSwitchOff)
721 // If any shutdown manager is registered, forward the event to it
722 if (iWindowToSendOffEventsTo && iWindowToSendOffEventsTo->QueueEvent(aEvent))
724 // Otherwise, do the shutdown here
725 if(!iIgnoreSwitchOffEvent && aDoSwitchOff)
727 #ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP
728 // if LaF is not registered, restart or shutdown using Power API
729 if (aEvent == EEventRestartSystem) // restart
731 if (Power::EnableWakeupEvents(EPwRestart) == KErrNone)
733 // Should reboot/power-cycle the system, so no need to RequestWakeupEventNotification
737 else // everything else maps to standby for now
739 if (Power::EnableWakeupEvents(EPwStandby) == KErrNone)
741 // Prepare to wake up from standby
743 Power::RequestWakeupEventNotification(s);
745 User::WaitForRequest(s);
749 UserHal::SwitchOff();
754 void CWsTop::RedrawScreens()
756 // apply to all screens
758 for(screenNo=0; screenNo<iNumberOfScreens; ++screenNo)
760 CScreen* screen = Screen(screenNo);
761 TRegionFix<1> screenArea(TRect(screen->DrawableArea()));
765 void CWsTop::EnableLogging(TReloadWsIni aSetting)
770 if (wsDebugLogLevel==CDebugLogBase::ELogEverything)
772 wsDebugLogLevel=CDebugLogBase::ELogImportant;
773 wsDebugLog->SetLoggingLevel(wsDebugLogLevel);
779 if(aSetting == EDoReloadWsIni)
781 CIniFile* temp = NULL;
782 TRAPD(err, temp = CIniFile::NewL());
785 //loading successful, replace the previous one
792 _LIT(KWSERVDebugLogFileName,"DLOG");
793 TBuf<KMaxFileName> fname(KWSERVDebugLogFileName);
794 _LIT(KWSERVIniFileVarLog,"LOG");
795 if (WsIniFile->FindVar(KWSERVIniFileVarLog,dlog) && dlog.Length()==2)
798 TInt err=iDebugLib.Load(fname);
801 TUidType uid=iDebugLib.Type();
802 if (uid[1]!=KWservLoggingDllUid)
804 err=KErrNotSupported;
810 _LIT(KWSERVIniFileVarLogP,"LOGP");
811 if (!WsIniFile->FindVar(KWSERVIniFileVarLogP,params))
813 CreateDebugLog func=(CreateDebugLog)iDebugLib.Lookup(1);
816 TRAP(err,wsDebugLog=(*func)(iIsFirstLog, params));
820 wsDebugLogLevel=CDebugLogBase::ELogEverything;
831 if (iCurrentFocusScreen)
833 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::ELogging,(errorLoc<<8)-err);
838 void CWsTop::DisableLogging()
845 void CWsTop::LogCommand(RWsSession::TLoggingCommand aCommand)
849 case RWsSession::ELoggingEnable:
852 case RWsSession::ELoggingDisable:
855 case RWsSession::ELoggingStatusDump:
858 case RWsSession::ELoggingHeapDump:
864 void CWsTop::StateDump()
867 for (screenNo=0; screenNo<iNumberOfScreens; ++screenNo)
868 ::StateDump( (*iScreens)[screenNo]->RootWindow());
871 void CWsTop::SetCurrentFocusScreen(CScreen* aScreen)
873 iCurrentFocusScreen = aScreen;
876 TInt CWsTop::SetCurrentFocusScreen(TInt aScreenNo)
878 CScreen* newFocusScreen=CWsTop::Screen(aScreenNo);
879 if (newFocusScreen==iCurrentFocusScreen)
882 CWsWindowGroup* newFocusGroup=newFocusScreen->FocusWindowGroup();
886 CWsWindowGroup* oldFocusGroup=iCurrentFocusScreen->FocusWindowGroup();
888 oldFocusGroup->LostFocus();
889 iCurrentFocusScreen=newFocusScreen;
890 newFocusGroup->ReceivedFocus();
891 TWsPointer::UpdatePointerCursor();
892 TWindowServerEvent::SendFocusChangedEvents();
896 void CWsTop::SetCheckHeapOnDisconnectClient(const CWsClient* aClient)
898 iTriggerHeapCheckOnClientExit=aClient;
901 void CWsTop::SetCheckHeapOnDisconnectMode(TWsCheckHeapOnDisconnectMode aCheckHeapOnDisconnectMode)
903 iHeapCheckMode=aCheckHeapOnDisconnectMode;
906 TBool CWsTop::NeedsHeapCheckAndRestart(TInt aStartHeapCount)
908 if (!iDoHeapCheckAndRestart)
910 iDoHeapCheckAndRestart=EFalse;
911 iCheckHeapResult=User::Heap().Count()-aStartHeapCount;
912 if(iCheckHeapResult > 0)
914 const TUint32 orphanedCell = User::Heap().__DbgMarkEnd(aStartHeapCount);
915 RDebug::Printf("Memory leak detected in wserv. First orphaned cell: 0x%8x", orphanedCell);
920 TInt CWsTop::FetchCheckHeapResult()
922 TInt ret=iCheckHeapResult;
923 iCheckHeapResult=KErrNotReady;
928 This function looks for memory which isn't essential to the correct running of the window server
929 and frees some of it.
930 Returns true if some memory was freed, and false if it was unable to free anything.
931 Called from the memory manager in low memory conditions.
933 TBool CWsTop::ReleaseMemory()
935 return iServer->ReleaseMemory();
938 TBool CWsTop::MultiFocusPolicy()
940 return iMultiFocusPolicy;
943 void CWsTop::ClearSurfaceMap(CWsClient *aClient)
945 TInt tempScreens = CWsTop::NumberOfScreens();
946 for (TInt ii = 0; ii < tempScreens; ii++)
948 __DEBUG_ONLY(TInt err=)
949 CWsTop::Screen(ii)->SurfaceMap()->RemoveAll(*aClient);
950 WS_ASSERT_DEBUG((err==KErrNone||err==KErrNotFound||err==KErrInUse), EWsPanicSurfaceMapError);
954 TBool CWsTop::SearchDuplicateSurfaceId(const TSurfaceId& aSurfaceId)
956 TInt tempScreens = CWsTop::NumberOfScreens();
957 for (TInt ii = 0; ii < tempScreens; ii++)
959 if (CWsTop::Screen(ii)->WindowElements().SearchDuplicateSurfaceId(aSurfaceId))
968 Checks to see if the render orientation has changed, and publishes any new orientaion
969 via publish and subscribe
971 @see KRenderOrientationCategory
972 @see KRenderOrientationKey
974 void CWsTop::CheckRenderOrientation()
976 iRenderOrientationTracker->CheckRenderOrientation();
979 typedef TInt (*ShellEntryPoint)(TAny *);
981 #if defined(__WINS__)
982 LOCAL_D const TUint KHeapSize=0x10000;
983 LOCAL_D const TUint KMaxHeapSize=0x400000;
986 CWsShellLogon::CWsShellLogon() : CActive(EWsShellLogonPriority)
988 #if defined (__WINS__)
989 // Clear handles to NULL so we can later detect which one gets used
990 iShellThread.SetHandle(NULL);
991 iShellProcess.SetHandle(NULL);
995 CWsShellLogon::~CWsShellLogon()
998 #if defined (__WINS__)
999 iShellThread.Close();
1000 iShellProcess.Close();
1004 #if defined(__WINS__)
1009 void CWsShellLogon::ConstructL(RFs &)
1011 CActiveScheduler::Add(this);
1014 _LIT(KWSERVIniFileVarShellCmd,"SHELLCMD");
1015 WsIniFile->FindVar(KWSERVIniFileVarShellCmd,shellCmd);
1016 _LIT(KWSERVShellName,"SHELL");
1017 TPtrC startUp(KWSERVShellName);
1018 _LIT(KWSERVIniFileVarStartUp,"STARTUP");
1019 WsIniFile->FindVar(KWSERVIniFileVarStartUp,startUp);
1023 _LIT(KWSERVArmShellLocationPattern,"Z:\\sys\\bin\\.EXE");
1024 User::LeaveIfError(fileName.SetNoWild(startUp,&KWSERVArmShellLocationPattern,NULL));
1025 #if defined(__WINS__)
1026 TInt err=iShellProcess.Create(fileName.FullName(),shellCmd);
1027 if (err == KErrNone)
1030 iShellProcess.Resume();
1034 // Try loading the matching DLL name instead?
1035 _LIT(KWSERVShellExtension,"Z:.DLL");
1036 User::LeaveIfError(fileName.Set(KWSERVShellExtension,&startUp,NULL));
1037 User::LeaveIfError(iLib.Load(fileName.FullName()));
1038 ShellEntryPoint libFunc=(ShellEntryPoint)iLib.Lookup(1);
1040 User::Leave(KErrGeneral);
1046 _LIT(KWSERVWinsShellInstanceName,"Shell%d");
1047 name.Format(KWSERVWinsShellInstanceName,num++);
1048 ret=iShellThread.Create(name,libFunc,KDefaultStackSize,&shellCmd,&iLib,NULL,KHeapSize,KMaxHeapSize,EOwnerProcess);
1049 } while(ret==KErrAlreadyExists);
1050 User::LeaveIfError(ret);
1052 iShellThread.Resume();
1054 #else // not __WINS__
1055 User::LeaveIfError(iShell.Create(fileName.FullName(),shellCmd));
1063 void CWsShellLogon::DoCancel()
1065 #if defined (__WINS__)
1066 if(iShellThread.Handle())
1067 iShellThread.LogonCancel(iStatus);
1069 iShellProcess.LogonCancel(iStatus);
1071 iShell.LogonCancel(iStatus);
1075 void CWsShellLogon::RunL()
1077 if (iStatus.Int()!=KErrCancel)
1078 CWsTop::ShellExited();
1081 void CWsShellLogon::Request()
1083 #if defined (__WINS__)
1084 if(iShellThread.Handle())
1085 iShellThread.Logon(iStatus);
1087 iShellProcess.Logon(iStatus);
1089 iShell.Logon(iStatus);
1094 #if defined(__WINS__)
1095 TFullName CWsShellLogon::FullName()
1097 if(iShellThread.Handle())
1098 return(iShellThread.FullName());
1100 return(iShellProcess.FullName());
1103 TFullName CWsShellLogon::FullName()
1104 {return(iShell.FullName());}
1111 UserSvr::WsRegisterThread();
1114 // Set wserv's main thread to system permanent
1116 err = User::SetCritical(User::ESystemPermanent);
1119 WS_PANIC_ALWAYS(EWsPanicFailedToSetThread);
1121 // If in the future wserv becomes multi-threaded,
1122 // we can uncomment the following lines to set any new threads to be system permanent as well.
1123 // err = User::SetProcessCritical(User::ESystemPermanent);
1124 // if (err!=KErrNone)
1126 // WS_PANIC_ALWAYS(EWsPanicFailedToSetProcess);
1129 thread.SetPriority(EPriorityMore);
1130 err = User::RenameThread(KWSERVThreadName);
1140 CTrapCleanup* CleanUpStack=CTrapCleanup::New();
1141 TRAP(err, WsIniFile = CIniFile::NewL());
1144 WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
1147 _LIT(KMemLeakCheck, "MEMLEAKCHECK");
1148 if (WsIniFile->FindVar(KMemLeakCheck))
1150 RDebug::Printf("The Memory Leak Check is Enabled => ECOM plugins are preload");
1151 CWsPluginManager* pm = NULL;
1152 TRAP(err,pm = CWsPluginManager::NewL());
1156 WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
1160 // Define properties for singleton callbacks. This must only be done ONCE
1161 // to ensure the properties can't be hijacked.
1162 gReleaseSingletonsOnExit = ETrue;
1163 DefineSingletonKey(KOpenWfcInteropCleanupKey);
1164 DefineSingletonKey(KOpenWfcImplCleanupKey);
1169 User::Heap().__DbgMarkStart();
1170 startCount=User::Heap().Count();
1171 TRAP(err,CWsTop::RunServerL());
1174 WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
1176 UserSvr::ReleaseEventHook();
1177 } while (CWsTop::NeedsHeapCheckAndRestart(startCount));
1179 REComSession::FinalClose(); // Now we can unload the plugins' dlls
1181 delete CleanUpStack;