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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Top level window server code
21 #include <graphics/fbsdefs.h>
24 #include "playbackgc.h"
26 #include "windowgroup.h"
30 #include "offscreenbitmap.h"
34 #include "WSGRAPHICDRAWERFACTORY.H"
35 #include "redrawmsgwindow.h"
36 #include <Graphics/WSGRAPHICDRAWERINTERFACE.H>
38 #include "backedupwindow.h"
41 #ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP
44 #include "graphics/windowserverconstants.h"
46 typedef CDebugLogBase *(*CreateDebugLog)(TBool aIsFirst, TDesC &aParams);
48 CWsActiveScheduler* CWsActiveScheduler::Static()
50 return static_cast<CWsActiveScheduler*>(CActiveScheduler::Current());
53 CWsActiveScheduler::CWsActiveScheduler(): iNumSamples(100)
55 iRun = User::FastCounter();
56 if((KErrNone == HAL::Get(HALData::EFastCounterFrequency,iFastCounterFreq)) && iFastCounterFreq)
58 iData = reinterpret_cast<TSample*>(User::AllocZ(sizeof(TSample)*iNumSamples));
62 CWsActiveScheduler::~CWsActiveScheduler()
64 User::FreeZ(reinterpret_cast<void*&>(iData));
67 void CWsActiveScheduler::PrepareDraw()
69 WS_ASSERT_DEBUG((ENormal == iState)||(EDrawn == iState),EWsPanicActiveScheduler);
70 __DEBUG_ONLY(iState = EPreparingDraw;)
71 iRunDraw = User::FastCounter();
74 void CWsActiveScheduler::CancelPrepare()
76 WS_ASSERT_DEBUG(EPreparingDraw == iState,EWsPanicActiveScheduler);
77 __DEBUG_ONLY(iState = ENormal;)
80 void CWsActiveScheduler::StartDraw()
82 WS_TRACE_SERVER_STARTDRAW();
83 WS_ASSERT_DEBUG(EPreparingDraw == iState,EWsPanicActiveScheduler);
84 __DEBUG_ONLY(iState = EDrawing;)
87 TUint64 duration = now.MicroSecondsFrom(iRunDraw).Int64();
88 iPreparing += duration;
92 iData[iCurrent].iStart = iRunDraw;
97 void CWsActiveScheduler::DrawStats(TInt& aUpdatesPerSecond,TInt64& aPixelsPerSecond,TInt aWindowInSeconds) const
103 // work out statistic
105 window.UniversalTime();
106 window-=TTimeIntervalSeconds(aWindowInSeconds);
109 for(TInt ii=0;ii<iNumSamples;++ii)
111 if (iData[ii].iStart>=window)
113 pixels+=iData[ii].iPixels;
114 duration+=iData[ii].iDuration;
118 if (aUpdatesPerSecond && duration)
120 aPixelsPerSecond=pixels;
121 aPixelsPerSecond=(aPixelsPerSecond*1000000)/duration;
122 aUpdatesPerSecond/=aWindowInSeconds;
127 void CWsActiveScheduler::StopDraw(TInt aPixelsUpdated)
129 WS_TRACE_SERVER_STOPDRAW();
130 WS_ASSERT_DEBUG(EDrawing == iState,EWsPanicActiveScheduler);
131 __DEBUG_ONLY(iState = EDrawn;)
134 TUint32 duration = now.MicroSecondsFrom(iRunDraw).Int64();
138 iData[iCurrent].iDuration = now.MicroSecondsFrom(iData[iCurrent].iStart).Int64();
139 iData[iCurrent].iPixels = aPixelsUpdated;
140 if(++iCurrent >= iNumSamples)
145 iDrawing += duration;
148 void CWsActiveScheduler::WaitForAnyRequest()
150 WS_ASSERT_DEBUG((ENormal == iState)||(EDrawn == iState),EWsPanicActiveScheduler);
152 stop.UniversalTime();
153 iTotal += stop.MicroSecondsFrom(iRun).Int64();
154 CActiveScheduler::WaitForAnyRequest();
156 __DEBUG_ONLY(iState = ENormal;)
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;
184 #if defined(__WINS__)
185 LOCAL_D TPtrC shellCmd;
188 CScreen* CWsTop::iCurrentFocusScreen;
189 CArrayPtrFlat<CScreen>* CWsTop::iScreens ;
190 TInt CWsTop::iNumberOfScreens ;
191 CWsTop::CShellStarter* CWsTop::iShellStarter;
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__)
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()
218 CClick::DeleteStatics();
219 WsPointer::DeleteStatics();
220 CWsClient::DeleteStatics();
222 delete iShellStarter;
225 CWsBackedUpWindow::StaticDestroy();
226 TWindowServerEvent::DeleteStatics();
230 CEventQueue::DeleteStaticsL();
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();
239 iScreens->ResetAndDestroy() ;
242 if (!iDoHeapCheckAndRestart)
244 RFbsSession::GetSession()->SendCommand(RFbsSession_SendCommand_ShutDownMessage); //This line is just being tidy, never really does anything useful.
247 RFbsSession::Disconnect();
252 delete TheActiveScheduler;
253 WsGraphicDrawer::FinalClose();
256 RWsTextCursor *CWsTop::CurrentTextCursor()
258 return(FocusWindowGroup() ? FocusWindowGroup()->TextCursor():NULL);
261 CWsClient *CWsTop::FocusWindowGroupOwner()
263 return(FocusWindowGroup() ? FocusWindowGroup()->WsOwner():NULL);
266 void CWsTop::ClientDestroyed(const CWsClient *aClient)
268 if (aClient==iShellClient)
272 void CWsTop::NewSession(const CWsClient *aClient)
274 if (iShellClient==NULL && iShell)
276 #if defined(__WINS__)
278 proc=aClient->Client();
281 aClient->Client().Process(proc);
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)
289 procName = procName.Left(colonLocation);
291 if (procName ==iShell->FullName())
293 iShellClient=aClient;
294 if (!iPreviousShellClient)
296 iPreviousShellClient=ETrue;
297 aClient->Screen()->RootWindow()->SetColorIfClear();
300 #if !defined(__WINS__)
306 void CWsTop::SessionExited(CWsClient *aClient)
311 TInt err=aClient->Client().Process(proc);
312 if (err==KErrNone && proc.Id()!=RProcess().Id())
315 const_cast<RThread&>(aClient->Client()).Kill(0);
319 else if (iHeapCheckMode!=EWsCheckHeapOnDisconnectModeNone && aClient==iTriggerHeapCheckOnClientExit)
321 if (iHeapCheckMode==EWsCheckHeapOnDisconnectModeOnce)
323 iHeapCheckMode=EWsCheckHeapOnDisconnectModeNone;
325 iTriggerHeapCheckOnClientExit=NULL;
326 iDoHeapCheckAndRestart=ETrue;
329 if (iServer->SessionCount()==1 && iShellBootMode==EShellBootModeNoReboot && iShell==NULL)
334 void CWsTop::RunServerL()
337 CWsMemoryManager* memMgr = CWsMemoryManager::NewLC();
338 CActiveScheduler::Start();
339 CleanupStack::PopAndDestroy(memMgr);
343 void CWsTop::InitStaticsL()
345 iShuttingDown=EFalse;
346 // By default shell should be started.
347 TBool startShell = ETrue;
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.
357 _LIT(KNoShell,"-NOSHELL");
359 TInt argLen = User::CommandLineLength();
362 HBufC* arg = HBufC::NewLC(argLen);
363 TPtr argPtr = arg->Des();
364 User::CommandLine(argPtr);
367 if(KErrNotFound != argPtr.Find(KNoShell))
369 // Don't start the shell. It will be started if required by
370 // the system starter.
373 CleanupStack::PopAndDestroy(arg);
376 TheActiveScheduler=new(ELeave) CWsActiveScheduler;
377 CActiveScheduler::Install(TheActiveScheduler);
380 WsIniFile=CIniFile::NewL();
381 _LIT(KWSERVIniFileVarLogEnable,"LOGENABLE");
383 if (WsIniFile->FindVar(KWSERVIniFileVarLogEnable,loggingLevel))
385 EnableLogging(EDoNotReloadWsIni);
388 wsDebugLog->SetLoggingLevel(loggingLevel);
392 _LIT(KWSERVIniFileVarFadeEnable,"FADEDISABLE");
393 iFadeEnabled = !WsIniFile->FindVar(KWSERVIniFileVarFadeEnable);
395 _LIT(KWSERVIniFileVarFinishEveryFlush,"FINISHEVERYFLUSH");
396 iFinishEveryFlush = WsIniFile->FindVar(KWSERVIniFileVarFinishEveryFlush);
398 _LIT(KWSERVIniFileVarSwitchOffEvent,"IGNORESWITCHOFFEVENT");
399 iIgnoreSwitchOffEvent = WsIniFile->FindVar(KWSERVIniFileVarSwitchOffEvent);
401 iServer=CWindowServer::NewL();
402 CClick::InitStaticsL();
405 if (!wservProc.DefaultDataPaged())
407 iServer->SetPinClientDescriptors(ETrue);
410 iServer->StartL(KWSERVServerName);
412 User::LeaveIfError(FbsStartup());
413 User::LeaveIfError(RFbsSession::Connect());
414 User::LeaveIfError(iTimer.CreateLocal());
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);
422 iScreens = new (ELeave) CArrayPtrFlat<CScreen>( iNumberOfScreens ) ; //
423 // now construct screens for as long as there is information
426 for ( ii = 0 ; ii < iNumberOfScreens ; ++ii )
430 //---------------------------------------------
431 iCurrentFocusScreen = (*iScreens)[0] ;
433 CWsFontCache::CreateInstanceL();
434 CWsGc::InitStaticsL();
435 CPlaybackGc::InitStaticsL();
436 CWsSpriteBase::InitStaticsL();
437 CEventQueue::InitStaticsL();
440 CWsAnimDll::InitStaticsL();
441 CWsAnim::InitStaticsL();
444 _LIT(KWSERVIniFileVarReboot,"REBOOT");
445 WsIniFile->FindVar(KWSERVIniFileVarReboot,bootMode);
446 if (bootMode>=0 && bootMode<=2)
447 iShellBootMode=bootMode;
449 CWsBackedUpWindow::StaticInitL();
450 CWsRedrawMsgWindow::StaticInitL();
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));
460 //Set the focus policy
461 _LIT(KFocusPolicy,"MULTIFOCUSPOLICY");
462 if(WsIniFile->FindVar(KFocusPolicy))
464 iMultiFocusPolicy = ETrue;
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.
473 UserSvr::WsRegisterSwitchOnScreenHandling(ETrue);
477 void CWsTop::InitScreenL( TInt aScreenNumber) // static
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 ) ;
483 if (aScreenNumber==0)
485 TMachineInfoV1Buf machineBuf;
486 UserHal::MachineInfo(machineBuf);
487 TMachineInfoV1& machineInfo=*(TMachineInfoV1*)machineBuf.Ptr();
488 digitiserArea = TRect(-machineInfo.iOffsetToDisplayInPixels,machineInfo.iXYInputSizeInPixels);
490 screen->ConstructL(digitiserArea, aScreenNumber);
491 iScreens->AppendL( screen ) ;
492 CleanupStack::Pop( screen ) ;
495 void CWsTop::UpdateAllScreens()
498 for (ii=0; ii<iNumberOfScreens; ++ii)
500 (*iScreens)[ii]->Update();
504 void CWsTop::ClearAllRedrawStores()
507 for (ii=0; ii<iNumberOfScreens; ++ii)
509 CScreen *screen=(*iScreens)[ii];
510 CWsWindowGroup* group=screen->RootWindow()->Child();
513 CWsWindowBase* win=group->Child();
518 static_cast<CWsWindow*>(win)->Redraw()->ClearRedrawStore(ETrue);
519 if (win->BaseChild())
520 win=win->BaseChild();
525 if (win->NextSibling())
527 win=win->NextSibling();
530 win=win->BaseParent();
536 group=group->NextSibling();
538 TriggerRedraws(screen->RootWindow());
544 CActiveScheduler::Stop();
547 void CWsTop::TriggerRedraws(CWsRootWindow* aRootWindow)
549 for(CWsWindowGroup *group=aRootWindow->Child();group!=NULL;group=group->NextSibling())
550 group->WsOwner()->TriggerRedraw();
553 void CWsTop::StartShell()
555 TRAPD(err,iShell=new(ELeave) CWsShellLogon());
559 if ((err=fs.Connect())==KErrNone)
561 fs.SetNotifyUser(EFalse);
562 TRAP(err,iShell->ConstructL(fs));
569 RDebug::Print(_L("Failed to start shell: err=%d\n"),err);
573 iShellStarter->After(TTimeIntervalMicroSeconds32(1000000));
577 void CWsTop::ShellExited()
581 switch(iShellBootMode)
583 case EShellBootModeReboot:
586 case EShellBootModeNoReboot:
587 if (iServer->SessionCount()==0)
590 case EShellBootModeExit:
596 TInt CWsTop::SetSendOffEventsToShell(CWsClient *aClient,const TWsClCmdOffEventsToShell &aData)
598 CWsWindowBase *window=NULL;
602 aClient->PPanic(EWservPanicNoWindowSpecifed);
605 aClient->HandleToWindow(aData.window,&window);
608 if (iWindowToSendOffEventsTo!=NULL && aClient!=iShellClient) //Allow the shell to pinch it
609 return KErrAlreadyExists;
610 iWindowToSendOffEventsTo=window;
614 if (iWindowToSendOffEventsTo==window || (!window && aClient==iWindowToSendOffEventsTo->WsOwner()))
615 iWindowToSendOffEventsTo=NULL;
620 void CWsTop::StopWindowGettingOffEvents(CWsWindowBase* aWindow)
622 if (aWindow==iWindowToSendOffEventsTo)
623 iWindowToSendOffEventsTo=NULL;
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.
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
635 void CWsTop::HandleSwitchOff(TInt aEvent,TBool aDoSwitchOff)
637 // If any shutdown manager is registered, forward the event to it
638 if (iWindowToSendOffEventsTo && iWindowToSendOffEventsTo->QueueEvent(aEvent))
640 // Otherwise, do the shutdown here
641 if(!iIgnoreSwitchOffEvent && aDoSwitchOff)
643 #ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP
644 // if LaF is not registered, restart or shutdown using Power API
645 if (aEvent == EEventRestartSystem) // restart
647 if (Power::EnableWakeupEvents(EPwRestart) == KErrNone)
649 // Should reboot/power-cycle the system, so no need to RequestWakeupEventNotification
653 else // everything else maps to standby for now
655 if (Power::EnableWakeupEvents(EPwStandby) == KErrNone)
657 // Prepare to wake up from standby
659 Power::RequestWakeupEventNotification(s);
661 User::WaitForRequest(s);
665 UserHal::SwitchOff();
670 void CWsTop::RedrawScreens()
672 // apply to all screens
674 for(screenNo=0; screenNo<iNumberOfScreens; ++screenNo)
676 CScreen* screen = Screen(screenNo);
677 TRegionFix<1> screenArea(TRect(screen->DrawableArea()));
678 screen->Update(screenArea);
682 void CWsTop::EnableLogging(TReloadWsIni aSetting)
687 if (wsDebugLogLevel==CDebugLogBase::ELogEverything)
689 wsDebugLogLevel=CDebugLogBase::ELogImportant;
690 wsDebugLog->SetLoggingLevel(wsDebugLogLevel);
696 if (aSetting == EDoReloadWsIni)
698 CIniFile* temp = NULL;
699 TRAPD(err, temp = CIniFile::NewL());
702 //loading successful, replace the previous one
709 _LIT(KWSERVDebugLogFileName,"DLOG");
710 TBuf<KMaxFileName> fname(KWSERVDebugLogFileName);
711 _LIT(KWSERVIniFileVarLog,"LOG");
712 if (WsIniFile->FindVar(KWSERVIniFileVarLog,dlog) && dlog.Length()==2)
715 TInt err=iDebugLib.Load(fname);
718 TUidType uid=iDebugLib.Type();
719 if (uid[1]!=KWservLoggingDllUid)
721 err=KErrNotSupported;
727 _LIT(KWSERVIniFileVarLogP,"LOGP");
728 if (!WsIniFile->FindVar(KWSERVIniFileVarLogP,params))
730 CreateDebugLog func=(CreateDebugLog)iDebugLib.Lookup(1);
733 TRAP(err,wsDebugLog=(*func)(iIsFirstLog, params));
737 wsDebugLogLevel=CDebugLogBase::ELogEverything;
748 if (iCurrentFocusScreen)
750 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::ELogging,(errorLoc<<8)-err);
755 void CWsTop::DisableLogging()
762 void CWsTop::LogCommand(RWsSession::TLoggingCommand aCommand)
766 case RWsSession::ELoggingEnable:
769 case RWsSession::ELoggingDisable:
772 case RWsSession::ELoggingStatusDump:
775 case RWsSession::ELoggingHeapDump:
781 void CWsTop::StateDump()
784 for (screenNo=0; screenNo<iNumberOfScreens; ++screenNo)
785 ::StateDump( (*iScreens)[screenNo]->RootWindow());
788 void CWsTop::SetCurrentFocusScreen(CScreen* aScreen)
790 iCurrentFocusScreen = aScreen;
793 TInt CWsTop::SetCurrentFocusScreen(TInt aScreenNo)
795 CScreen* newFocusScreen=CWsTop::Screen(aScreenNo);
796 if (newFocusScreen==iCurrentFocusScreen)
799 CWsWindowGroup* newFocusGroup=newFocusScreen->FocusWindowGroup();
803 CWsWindowGroup* oldFocusGroup=iCurrentFocusScreen->FocusWindowGroup();
805 oldFocusGroup->LostFocus();
806 iCurrentFocusScreen=newFocusScreen;
807 newFocusGroup->ReceivedFocus();
808 WsPointer::UpdatePointerCursor();
809 TWindowServerEvent::SendFocusChangedEvents();
813 void CWsTop::SetCheckHeapOnDisconnectClient(const CWsClient* aClient)
815 iTriggerHeapCheckOnClientExit=aClient;
818 void CWsTop::SetCheckHeapOnDisconnectMode(TWsCheckHeapOnDisconnectMode aCheckHeapOnDisconnectMode)
820 iHeapCheckMode=aCheckHeapOnDisconnectMode;
823 TBool CWsTop::NeedsHeapCheckAndRestart(TInt aStartHeapCount)
825 if (!iDoHeapCheckAndRestart)
827 iDoHeapCheckAndRestart=EFalse;
828 iCheckHeapResult=User::Heap().Count()-aStartHeapCount;
829 if(iCheckHeapResult > 0)
831 const TUint32 orphanedCell = User::Heap().__DbgMarkEnd(aStartHeapCount);
832 RDebug::Printf("Memory leak detected in wserv. First orphaned cell: 0x%8x", orphanedCell);
837 TInt CWsTop::FetchCheckHeapResult()
839 TInt ret=iCheckHeapResult;
840 iCheckHeapResult=KErrNotReady;
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.
850 TBool CWsTop::ReleaseMemory()
852 return iServer->ReleaseMemory();
855 TBool CWsTop::MultiFocusPolicy()
857 return iMultiFocusPolicy;
860 typedef TInt (*ShellEntryPoint)(TAny *);
862 #if defined(__WINS__)
863 LOCAL_D const TUint KHeapSize=0x10000;
864 LOCAL_D const TUint KMaxHeapSize=0x400000;
867 CWsShellLogon::CWsShellLogon() : CActive(EWsShellLogonPriority)
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);
876 CWsShellLogon::~CWsShellLogon()
879 #if defined (__WINS__)
880 iShellThread.Close();
881 iShellProcess.Close();
885 #if defined(__WINS__)
890 void CWsShellLogon::ConstructL(RFs &)
892 CActiveScheduler::Add(this);
893 #if !defined(__WINS__)
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);
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);
912 iShellProcess.Resume();
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);
922 User::Leave(KErrGeneral);
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);
934 iShellThread.Resume();
936 #else // not __WINS__
937 User::LeaveIfError(iShell.Create(fileName.FullName(),shellCmd));
945 void CWsShellLogon::DoCancel()
947 #if defined (__WINS__)
948 if(iShellThread.Handle())
949 iShellThread.LogonCancel(iStatus);
951 iShellProcess.LogonCancel(iStatus);
953 iShell.LogonCancel(iStatus);
957 void CWsShellLogon::RunL()
959 if (iStatus.Int()!=KErrCancel)
960 CWsTop::ShellExited();
963 void CWsShellLogon::Request()
965 #if defined (__WINS__)
966 if(iShellThread.Handle())
967 iShellThread.Logon(iStatus);
969 iShellProcess.Logon(iStatus);
971 iShell.Logon(iStatus);
976 #if defined(__WINS__)
977 TFullName CWsShellLogon::FullName()
979 if(iShellThread.Handle())
980 return(iShellThread.FullName());
982 return(iShellProcess.FullName());
985 TFullName CWsShellLogon::FullName()
986 {return(iShell.FullName());}
995 UserSvr::WsRegisterThread();
998 // Set wserv's main thread to system permanent
1000 err = User::SetCritical(User::ESystemPermanent);
1003 WS_PANIC_ALWAYS(EWsPanicFailedToSetThread);
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)
1010 // WS_PANIC_ALWAYS(EWsPanicFailedToSetProcess);
1013 thread.SetPriority(EPriorityMore);
1014 err = User::RenameThread(KWSERVThreadName);
1026 User::Heap().__DbgMarkStart();
1027 startCount=User::Heap().Count();
1028 CTrapCleanup* CleanUpStack=CTrapCleanup::New();
1029 TRAP(err,CWsTop::RunServerL());
1032 WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
1034 UserSvr::ReleaseEventHook();
1035 delete CleanUpStack;
1036 } while (CWsTop::NeedsHeapCheckAndRestart(startCount));