Update contrib.
1 // Copyright (c) 1995-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 the License "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 // e32\ewsrv\ws_main.cpp
22 #include <domainmanager.h>
27 GLREF_D CKeyTranslator *KeyTranslator;
28 GLREF_D CKeyRepeat* KeyRepeat;
30 // password notifier support functions
31 LOCAL_C void RenderPassword(RConsole *aCon, TInt aPWLeft, const TDesC& aPW);
33 _LIT(KShellProcessName, "ESHELL");
34 _LIT(KShellCommandLine, "/p");
40 CKeyRepeat::CKeyRepeat(TInt aPriority) : CTimer(aPriority)
42 // Constructor. Set default repeat delay and rate
45 iDelay=EDefaultKeyRepeatDelay;
46 iRate=EDefaultKeyRepeatRate;
49 void CKeyRepeat::ConstructL()
53 CActiveScheduler::Add(this);
56 void CKeyRepeat::RunL()
58 // Send a repeat keypress to the window
63 CWsWindow::KeyPress(iKeyData);
66 void CKeyRepeat::Request(TKeyData& aKeyData)
68 // Request a repeat event
77 void CKeyRepeat::SetRepeatTime(TInt aDelay,TInt aRate)
84 void CKeyRepeat::RepeatTime(TInt& aDelay,TInt& aRate)
95 CWsSession::CWsSession()
97 iTestFast = (UserSvr::DebugMask(2)&0x00000002) ? 1 : 0;
100 CWsSession::~CWsSession()
113 void CWsServer::New()
115 // Create a new CWsServer.
119 CWsServer *pS=new CWsServer(EPriority);
120 __ASSERT_ALWAYS(pS!=NULL,Fault(ECreateServer));
121 pS->SetPinClientDescriptors(EFalse); // don't pin because client interface can't cope with errors if pin fails under real or simulated OOM
122 TInt r=pS->Start(KE32WindowServer);
123 __ASSERT_ALWAYS(r==KErrNone,Fault(EStartServer));
124 RProcess::Rendezvous(KErrNone);
128 CWsServer::CWsServer(TInt aPriority)
132 : CServer2(aPriority)
136 CSession2* CWsServer::NewSessionL(const TVersion& aVersion,const RMessage2&) const
138 // Create a new client for this server.
142 TVersion v(KW32MajorVersionNumber,KW32MinorVersionNumber,KE32BuildVersionNumber);
143 TBool r=User::QueryVersionSupported(v,aVersion);
145 User::Leave(KErrNotSupported);
146 return new(ELeave) CWsSession;
149 void CWsSession::ServiceL(const RMessage2& aMessage)
151 // Handle messages for this session.
156 CWsWindow::WaitOnService();
157 CWsWindow* pW=iWindow;
158 TInt r=EPrematureOperation;
159 TBool delayCompletion=EFalse;
160 switch (aMessage.Function())
172 r=EWindowOutOfMemory;
176 pW->iAllowResize=ETrue;
177 pW->iIsVisible=ETrue;
179 pW->SetCursorHeight(50);
180 pW->iKQueue.SetOffset(_FOFF(SWsKey,iLink));
187 pW=new(ELeave) CWsWindow;
189 pW->iAllowResize=ETrue;
190 pW->iIsVisible=ETrue;
192 pW->SetCursorHeight(50);
193 pW->iKQueue.SetOffset(_FOFF(SWsKey,iLink));
195 pW->iAllowSlide=ETrue;
197 iCurMsg.ReadL(0, name);
199 TPckgBuf<TSize> size;
200 iCurMsg.ReadL(1, size);
201 TRAP(r,pW->CreateL(size()));
209 case EConsoleClearScreen:
218 case EConsoleClearToEndOfLine:
222 pW->ClearToEndOfLine();
227 case EConsoleSetTitle:
232 iCurMsg.ReadL(0, name);
238 case EConsoleSetSize:
242 TPckgBuf<TSize> size;
243 iCurMsg.ReadL(0, size);
244 // pW->SetSize(size());
250 case EConsoleSetWindowPosAbs:
254 TPckgBuf<TPoint> point;
255 iCurMsg.ReadL(0, point);
256 pW->SetWindowPosAbs(point());
261 case EConsoleSetCursorHeight:
265 pW->SetCursorHeight(aMessage.Int0());
270 case EConsoleSetCursorPosAbs:
274 TPckgBuf<TPoint> point;
275 iCurMsg.ReadL(0, point);
276 pW->SetCursorPosAbs(point());
281 case EConsoleSetCursorPosRel:
285 TPckgBuf<TPoint> point;
286 iCurMsg.ReadL(0, point);
287 pW->SetCursorPosRel(point());
292 case EConsoleCursorPos:
296 TPckgBuf<TPoint> point;
297 point()=pW->CursorPosition();
298 aMessage.WriteL(0,point);
307 TPckgBuf<TSize> size;
309 aMessage.WriteL(0,size);
314 case EConsoleScreenSize:
318 TPckgBuf<TSize> size;
319 size()=CWsWindow::ScreenSize;
320 aMessage.WriteL(0,size);
325 case EConsoleControl:
329 TBool indicator=ETrue;
334 iCurMsg.ReadL(0,b,offset);
335 for (const TText* pB=b.Ptr();pB<b.Ptr()+b.Length();pB++)
347 pW->ControlScrollBars(indicator);
350 pW->ControlWrapLock(indicator);
353 pW->ControlPointerEvents(indicator);
356 pW->ControlScrollLock(indicator);
359 pW->ControlVisibility(indicator);
362 pW->ControlCursorRequired(indicator);
365 pW->ControlMaximised(indicator);
368 pW->ControlNewLineMode(indicator);
371 pW->ControlOnTop(indicator);
374 pW->ControlInformAllMouse(indicator);
377 pW->ControlRawEventMode(indicator);
380 pW->ControlAllowResize(indicator);
385 while (b.Length()==b.MaxLength());
402 iCurMsg.ReadL(0,b,offset);
406 while (b.Length()==b.MaxLength());
414 pW->Write(_L("Output suppressed because TESTFAST mode is set..."));
432 if (pW->EnqueReadRequest(aMessage))
434 delayCompletion=ETrue;
438 r=EDoubleReadRequest;
442 case EConsoleReadCancel:
446 pW->DequeReadRequest();
451 case EConsoleDestroy:
461 case EConsoleSetMode:
463 r=CWsWindow::SetMode((TVideoMode)aMessage.Int0());
466 case EConsoleSetPaletteEntry:
468 CWsWindow::ScreenDriver->SetPaletteEntry((TColorIndex)aMessage.Int0(),(TUint8)aMessage.Int1(),(TUint8)aMessage.Int2(),(TUint8)aMessage.Int3());
473 case EConsoleGetPaletteEntry:
476 TPckgBuf<TUint8> red,green,blue;
477 CWsWindow::ScreenDriver->GetPaletteEntry((TColorIndex)aMessage.Int0(),r,g,b);
478 red()=r; green()=g; blue()=b;
479 aMessage.WriteL(1,red);
480 aMessage.WriteL(2,green);
481 aMessage.WriteL(3,blue);
485 case EConsoleSetTextColors:
489 pW->iFgColor=(TColorIndex)aMessage.Int0();
490 pW->iBgColor=(TColorIndex)aMessage.Int1();
495 case EConsoleSetUIColors:
497 CWsWindow::WindowBgColor=(TColorIndex)aMessage.Int0();
498 CWsWindow::BorderColor=(TColorIndex)aMessage.Int1();
499 CWsWindow::ScreenColor=(TColorIndex)aMessage.Int2();
500 CWsWindow::ChangeUIColors();
504 case EConsoleSetTextAttribute:
507 pW->SetTextAttribute((TTextAttribute)aMessage.Int0());
514 if (!delayCompletion)
515 aMessage.Complete(r);
516 CWsWindow::SignalService();
519 void CWsSession::ServiceError(const RMessage2& aMessage,TInt aError)
521 if (!aMessage.IsNull())
525 aMessage.Panic(_L("WServ panic"),aError);
529 aMessage.Complete(aError);
534 CWsServer::~CWsServer()
547 // Create the CEvent active object.
551 CEvent *pE=new CEvent(EPriority);
552 __ASSERT_ALWAYS(pE!=NULL,Fault(ECreateEvent));
553 pE->CaptureKeys=new CCaptureKeys();
554 __ASSERT_ALWAYS(pE->CaptureKeys!=NULL,Fault(ECreateEvent));
556 pE->CaptureKeys->Construct();
558 CActiveScheduler::Add(pE);
559 UserSvr::CaptureEventHook();
565 // Destroy the CEvent active object
572 #pragma warning( disable : 4705 ) // statement has no effect
573 CEvent::CEvent(TInt aPriority)
580 #pragma warning( default : 4705 )
582 void CEvent::Request()
584 // Issue a request for the next event.
588 UserSvr::RequestEvent(iEvent,iStatus);
592 void CEvent::DoCancel()
594 // Cancel a pending event.
598 UserSvr::RequestEventCancel();
603 // Event has completed.
607 if (CWsWindow::RawEventMode())
610 CWsWindow::QueueRawEvent(iEvent.Event());
614 switch(iEvent.Event().Type())
616 case TRawEvent::ERedraw:
619 case TRawEvent::EButton1Down:
620 if(!CWsWindow::MouseIsCaptured)
622 CWsWindow::MouseMove(iEvent.Event().Pos());
623 CWsWindow::MouseLeftButton();
626 CWsWindow::InformTopMouse(iEvent.Event().Pos());
628 case TRawEvent::EButton1Up:
629 if(!CWsWindow::MouseIsCaptured)
631 CWsWindow::MouseMove(iEvent.Event().Pos());
632 CWsWindow::MouseLeftButtonUp();
635 case TRawEvent::EButton2Down:
637 case TRawEvent::EButton2Up:
639 case TRawEvent::EButton3Down:
641 case TRawEvent::EButton3Up:
643 case TRawEvent::EPointerMove:
644 CWsWindow::MouseMove(iEvent.Event().Pos());
646 case TRawEvent::EInactive:
649 case TRawEvent::EActive:
651 case TRawEvent::EUpdateModifiers:
652 KeyTranslator->UpdateModifiers(iEvent.Event().Modifiers());
654 case TRawEvent::EKeyDown:
657 if (KeyTranslator->TranslateKey(iEvent.Event().ScanCode(), EFalse,*CaptureKeys,keyData))
658 CWsWindow::KeyPress(keyData);
659 if (keyData.iModifiers&EModifierAutorepeatable)
660 KeyRepeat->Request(keyData);
663 case TRawEvent::EKeyUp:
667 if (KeyTranslator->TranslateKey(iEvent.Event().ScanCode(), ETrue,*CaptureKeys,keyData))
668 CWsWindow::KeyPress(keyData);
671 case TRawEvent::ESwitchOn:
672 case TRawEvent::ECaseOpen:
673 HAL::Set(HAL::EDisplayState, 1);
675 RDmDomainManager mgr;
676 TInt r = mgr.Connect();
678 User::Panic(_L("EWSRV SwitchOn0"), r);
679 TRequestStatus status;
680 mgr.RequestDomainTransition(KDmIdUiApps, EPwActive, status);
681 User::WaitForRequest(status);
682 if (status.Int() != KErrNone)
683 User::Panic(_L("EWSRV SwitchOn1"), status.Int());
687 case TRawEvent::EPointerSwitchOn:
689 User::Beep(440,250000);
692 case TRawEvent::ESwitchOff:
694 RDmDomainManager mgr;
695 TInt r = mgr.Connect();
697 User::Panic(_L("EWSRV SwitchOff0"), r);
698 TRequestStatus status;
699 mgr.RequestSystemTransition(EPwStandby, status);
700 User::WaitForRequest(status);
701 if (status.Int() != KErrNone)
702 User::Panic(_L("EWSRV SwitchOff1"), status.Int());
706 case TRawEvent::ECaseClose:
708 RDmDomainManager mgr;
709 TInt r = mgr.Connect();
711 User::Panic(_L("EWSRV CaseClosed"), r);
712 TRequestStatus status;
713 mgr.RequestDomainTransition(KDmIdUiApps, EPwStandby, status);
714 User::WaitForRequest(status);
715 if (status.Int() != KErrNone)
716 User::Panic(_L("EWSRV CaseClosed1"), status.Int());
719 HAL::Set(HAL::EDisplayState, 0);
721 case TRawEvent::ENone:
729 LOCAL_C void RenderPassword(RConsole *aCon, TInt aPWLeft, const TDesC& aPW)
730 // pre: aCon points to a console being used to read a password.
731 // aPWLeft is the column number from which the left brace should be drawn.
732 // aPasswd is a valid password.
733 // post: the password is rendered onto the console and followed by a '#' and
734 // padding spaces. Everything is enclosed in a pair of square brackets.
736 aCon->SetCursorPosAbs(TPoint(aPWLeft, 3));
737 aCon->Write(_L("["));
739 aCon->Write(_L("#"));
742 for (i = 0; i < KMaxMediaPassword - aPW.Length(); i++)
744 aCon->Write(_L(" "));
747 aCon->Write(_L("]"));
751 void CNotifierSession::RunPasswordWindowL(const RMessage2 &aMsg)
753 // Eight unicode chars are mapped to (up to) sixteen bytes. Remainder of array is
754 // zeroed. Message is completed in CNotifierSession::ServiceL().
757 // local copies of dialog data, 5 * (8 + 32 * 2) bytes
758 TBuf<0x20> line1, line2, unlockBtn, storeBtn, cancelBtn;
759 line1.Copy(_L("Password notifier"));
760 line2.Copy(_L("Enter password"));
761 unlockBtn.Copy(_L("Unlock"));
762 storeBtn.Copy(_L("Store"));
763 cancelBtn.Copy(_L("Cancel"));
765 TPckgBuf<TMediaPswdReplyNotifyInfoV1> reply;
767 // Format the console window.
769 const TInt KPasswordBarLen(1 + KMaxMediaPassword + 1 + 1);
770 const TInt KButtonBarLen(
771 1 + unlockBtn.Length() + 1
772 + 1 + 1 + cancelBtn.Length() + 1
773 + 1 + 1 + storeBtn.Length() + 1);
775 // Work out width of window.
776 // (Buttons are enclosed by angle brackets and separted by a space.)
777 // (Password is followed by '#' and delimited by square brackets.)
778 // If KButtonBarLen is at least as long as any other line then it will write
779 // to the bottom right corner and cause the console to scroll. To counter
780 // this, an extra padding character is added if necessary.
783 width = Max(line1.Length(), line2.Length());
784 width = Max(width, KPasswordBarLen);
785 width = KButtonBarLen >= width ? KButtonBarLen + 1: width;
787 // Create the window and render its contents.
791 con.Control(_L("-Visible"));
792 TInt r = con.Init(_L(""), TSize(width, 2 + 1 + 1 + 1 + 1));
795 PanicClient(aMsg, ENotifierPanicPasswordWindow);
796 User::Leave(KErrGeneral);
798 con.Control(_L("+Max -Cursor -AllowResize +OnTop"));
800 con.SetCursorPosAbs(TPoint(0, 0));
802 con.SetCursorPosAbs(TPoint(0, 1));
804 const TInt KPasswordLeft((width - KPasswordBarLen) / 2);
805 con.SetCursorPosAbs(TPoint(KPasswordLeft, 3));
806 con.Write(_L("[# ]"));
807 con.SetCursorPosAbs(TPoint((width - KButtonBarLen) / 2, 5));
809 con.Write(unlockBtn);
810 con.Write(_L("> <"));
812 con.Write(_L("> <"));
813 con.Write(cancelBtn);
816 // Allow the user to edit the password until they either press enter or escape.
819 TBuf<KMaxMediaPassword> pw;
821 TMediaPswdNotifyExitMode em(EMPEMUnlock); // avoid VC warning C4701 (used w/o init).
823 const TInt sendInfoLen = User::LeaveIfError(aMsg.GetDesLength(1));
825 if (sendInfoLen == sizeof(TMediaPswdSendNotifyInfoV1Debug))
827 // debug mode - wait for specified period and close notifier
829 TPckgBuf<TMediaPswdSendNotifyInfoV1Debug> sendDbg;
830 aMsg.ReadL(1, sendDbg);
831 if (sendDbg().iSleepPeriod >= 0)
832 User::After(sendDbg().iSleepPeriod);
837 TInt64 seed = now.Int64();
838 User::After(Math::Rand(seed) % -(sendDbg().iSleepPeriod));
841 reply().iEM = sendDbg().iEM;
842 Mem::Copy(reply().iPW, sendDbg().iPW, KMaxMediaPassword);
846 RenderPassword(&con, KPasswordLeft, pw);
852 TInt keyCode = key.Code();
867 em = EMPEMUnlockAndStore;
874 pw.SetLength(pw.Length() - 1);
875 RenderPassword(&con, KPasswordLeft, pw);
879 default: // interpret other keys as pw contents
881 // unicode encoding, so number of password characters is half byte length
882 if (ch.IsPrint() && pw.Length() < KMaxMediaPassword / 2)
885 RenderPassword(&con, KPasswordLeft, pw);
891 // Fill the TMediaPswdReplyNotifyInfoV1 structure.
893 if (em == EMPEMUnlock || em == EMPEMUnlockAndStore)
895 const TInt byteLen = pw.Length() * 2;
897 // zero entire array; and then copy in valid section of TMediaPassword,
898 // not converting Unicode to ASCII.
899 TPtr8 pt8(reply().iPW, KMaxMediaPassword); // length = 0
900 pt8.FillZ(KMaxMediaPassword); // length = KMaxMediaPassword
901 pt8.Zero(); // length = 0
903 pt8.Copy(reinterpret_cast<const TUint8 *>(pw.Ptr()), byteLen);
906 // Set exit mode to tell user how dialog handled.
909 } // else (sendInfoLen == sizeof(TMediaPswdSendNotifyInfoV1Debug))
913 aMsg.WriteL(2, reply);
917 // class MNotifierBase2
920 void MNotifierBase2::SetManager(MNotifierManager* aManager)
926 // class CNotifierServer
929 _LIT(__NOTIFIER_SERVER,"TextNotifierSrvr");
931 CNotifierServer* CNotifierServer::NewL()
933 CNotifierServer* server=new(ELeave) CNotifierServer(200);
934 CleanupStack::PushL(server);
935 server->ConstructL();
936 server->StartL(__NOTIFIER_NAME);
937 CleanupStack::Pop(); // server
941 CNotifierServer::~CNotifierServer()
947 void CNotifierServer::SetIsExiting()
952 TBool CNotifierServer::IsExiting() const
957 CNotifierServer::CNotifierServer(TInt aPriority)
958 : CServer2(aPriority)
961 void CNotifierServer::ConstructL()
963 iManager=CNotifierManager::NewL();
965 User::LeaveIfError(fs.Connect());
966 CleanupClosePushL(fs);
967 iManager->RegisterL(fs);
968 CleanupStack::PopAndDestroy(); // fs.Close()
971 CSession2* CNotifierServer::NewSessionL(const TVersion &aVersion,const RMessage2&) const
973 TVersion v(1,0,0); // !! liaise with E32
974 if (!User::QueryVersionSupported(v,aVersion))
975 User::Leave(KErrNotSupported);
976 return new(ELeave) CNotifierSession(*this);
980 // class CNotifierSession
983 CNotifierSession::CNotifierSession(const CNotifierServer& aServer)
986 iClientId=(TInt)this;
989 CNotifierSession::~CNotifierSession()
991 const CNotifierServer* server=static_cast<const CNotifierServer*>(Server());
992 if (!server->IsExiting())
994 server->Manager()->HandleClientExit(iClientId);
998 void CNotifierSession::ServiceL(const RMessage2 &aMessage)
1000 TBool completeMessage=ETrue;
1001 switch (aMessage.Function())
1003 case ENotifierNotify:
1004 DisplayAlertL(aMessage);
1006 case ENotifierNotifyCancel:
1007 // do nothing - this server doesn't support cancelling RNotifier::Notify - the client will just have to wait
1009 case ENotifierInfoPrint:
1010 DisplayInfoMsgL(aMessage);
1012 case EStartNotifier:
1013 DoStartNotifierL(aMessage);
1015 case ECancelNotifier:
1016 iServer->Manager()->NotifierCancel(TUid::Uid(aMessage.Int0()));
1018 case EUpdateNotifierAndGetResponse:
1019 case EUpdateNotifier:
1020 DoUpdateNotifierL(aMessage);
1022 case EStartNotifierAndGetResponse:
1024 if (aMessage.Int0()==KMediaPasswordNotifyUid)
1026 RunPasswordWindowL(aMessage);
1030 TBool cleanupComplete=ETrue;
1031 StartNotifierAndGetResponseL(aMessage,cleanupComplete);
1032 // if the plug-in starts successfully, it has
1033 // responsibility for completing the message (either
1034 // synchronously or asynchronously)
1035 completeMessage=EFalse;
1040 aMessage.Complete(KErrNotSupported);
1043 if (completeMessage && !aMessage.IsNull())
1045 aMessage.Complete(KErrNone);
1049 void CNotifierSession::DisplayAlertL(const RMessage2& aMessage)
1051 const TInt lengthOfCombinedBuffer=User::LeaveIfError(aMessage.GetDesLength(1));
1052 const TInt lengthOfLine1=(STATIC_CAST(TUint,aMessage.Int2())>>16);
1053 const TInt lengthOfLine2=(aMessage.Int2()&KMaxTUint16);
1054 const TInt lengthOfBut1=(STATIC_CAST(TUint,aMessage.Int3())>>16);
1055 const TInt lengthOfBut2=(aMessage.Int3()&KMaxTUint16);
1056 if (lengthOfCombinedBuffer!=lengthOfLine1+lengthOfLine2+lengthOfBut1+lengthOfBut2)
1058 PanicClient(aMessage,ENotifierPanicInconsistentDescriptorLengths);
1061 HBufC* const combinedBuffer=HBufC::NewLC(lengthOfCombinedBuffer);
1062 {TPtr combinedBuffer_asWritable(combinedBuffer->Des());
1063 aMessage.ReadL(1,combinedBuffer_asWritable);}
1064 const TPtrC line1(combinedBuffer->Left(lengthOfLine1));
1065 const TPtrC line2(combinedBuffer->Mid(lengthOfLine1,lengthOfLine2));
1066 const TPtrC but1(combinedBuffer->Mid(lengthOfLine1+lengthOfLine2,lengthOfBut1));
1067 const TPtrC but2(combinedBuffer->Mid(lengthOfLine1+lengthOfLine2+lengthOfBut1,lengthOfBut2));
1068 TInt buttons, len, offset;
1069 if (lengthOfBut2==0)
1077 len=lengthOfBut1+lengthOfBut2+5;
1079 if (lengthOfLine1>len)
1081 if (lengthOfLine2>len)
1086 con.ScreenSize(scsz);
1087 con.Control(_L("-Visible"));
1088 TInt ww=Min(len,scsz.iWidth-4);
1090 if ((lengthOfBut1+lengthOfBut2+5)>ww)
1092 if (lengthOfLine1>ww)
1094 if (lengthOfLine2>ww)
1096 con.Init(_L(""),TSize(ww,wh));
1097 con.Control(_L("+Max -Cursor -Allowresize +Ontop +Wrap"));
1099 con.SetCursorPosAbs(TPoint(0,1));
1102 offset=(len-lengthOfBut1-lengthOfBut2-5)/2;
1104 offset=(len-lengthOfBut1-2)/2;
1105 con.SetCursorPosAbs(TPoint(offset,2));
1111 con.Write(_L(" <"));
1123 while((keycode!=EKeyEscape&&keycode!=EKeyEnter&&buttons==2)||(keycode!=EKeyEnter&&buttons==1));
1124 if(keycode==EKeyEscape)
1129 aMessage.WriteL(0,TPckgC<TInt>(keycode));
1130 CleanupStack::PopAndDestroy(combinedBuffer);
1133 TInt CNotifierSession::InfoPrintThread(TAny* aMessage)
1135 TBuf<0x50> des; // 0x50 max size of message
1137 des=*(TBuf<0x50> *)aMessage;
1138 TInt l=des.Length();
1139 NotifierSemaphore.Signal();
1141 con.Control(_L("-Visible"));
1143 con.ScreenSize(size);
1144 TInt ww=Min(size.iWidth-6,l);
1145 TInt wh=(l+ww-1)/ww;
1148 con.Init(_L(""),TSize(ww,wh));
1149 con.Control(_L("+Maximise"));
1150 con.SetWindowPosAbs(TPoint(size.iWidth-ww-4,1));
1151 con.Control(_L("+Wrap +Ontop"));
1154 User::After(1300000);
1159 void CNotifierSession::DisplayInfoMsgL(const RMessage2& aMessage)
1162 TBuf<0x50> des; // 0x50 max size of message lines
1163 aMessage.ReadL(0,des);
1167 r=thread.Create(_L("Info Window"),InfoPrintThread,KDefaultStackSize,&User::Allocator(),(TAny*)&des);
1168 if(r==KErrAlreadyExists)
1169 User::After(200000);
1170 } while(r==KErrAlreadyExists);
1171 User::LeaveIfError(r);
1173 NotifierSemaphore.Wait();
1177 void CNotifierSession::DoStartNotifierL(const RMessage2& aMessage)
1179 const TUid targetUid={aMessage.Int0()};
1180 HBufC8* const inputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesLength(1)));
1181 {TPtr8 input(inputBuffer->Des());
1182 aMessage.ReadL(1,input);}
1184 iServer->Manager()->NotifierStartL(targetUid,*inputBuffer,&output,iClientId);
1186 aMessage.WriteL(2,output);
1187 CleanupStack::PopAndDestroy(inputBuffer);
1190 void CNotifierSession::DoUpdateNotifierL(const RMessage2& aMessage)
1192 const TUid targetUid={aMessage.Int0()};
1193 HBufC8* const inputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesLength(1)));
1194 {TPtr8 input(inputBuffer->Des());
1195 aMessage.ReadL(1, input);}
1196 HBufC8* const outputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesMaxLength(2)));
1197 {TPtr8 output(outputBuffer->Des());
1198 iServer->Manager()->NotifierUpdateL(targetUid,*inputBuffer,&output,iClientId);}
1199 aMessage.WriteL(2,*outputBuffer);
1200 CleanupStack::PopAndDestroy(2,inputBuffer);
1203 void CNotifierSession::StartNotifierAndGetResponseL(const RMessage2& aMessage,TBool& aCleanupComplete)
1205 const TUid targetUid={aMessage.Int0()};
1206 HBufC8* const inputBuffer=HBufC8::NewLC(User::LeaveIfError(aMessage.GetDesLength(1)));
1207 {TPtr8 input(inputBuffer->Des());
1208 aMessage.ReadL(1,input);}
1209 iServer->Manager()->NotifierStartAndGetResponseL(targetUid,*inputBuffer,2,aMessage,iClientId,aCleanupComplete);
1210 CleanupStack::PopAndDestroy(inputBuffer);
1213 void CNotifierSession::PanicClient(const RMessage2& aMessage,TNotifierPanic aCode)
1215 aMessage.Panic(__NOTIFIER_SERVER,aCode);
1219 // class CNotifierManager
1222 const TInt KNullClientId=0;
1224 CNotifierManager* CNotifierManager::NewL()
1226 CNotifierManager* self=new(ELeave) CNotifierManager;
1227 CleanupStack::PushL(self);
1229 CleanupStack::Pop(self);
1233 CNotifierManager::~CNotifierManager()
1237 const TInt count=iObservedList->Count();
1238 for (TInt ii=0;ii<count;ii++)
1240 (*iObservedList)[ii]->Release();
1242 delete iObservedList;
1246 const TInt count=iLibraries->Count();
1247 for (TInt ii=0;ii<count;ii++)
1249 (*iLibraries)[ii].Close();
1253 delete iChannelMonitor;
1254 delete iActivityMonitor;
1259 void CNotifierManager::RegisterL(RFs& aFs)
1261 #ifdef SUPPORT_OLD_PLUGIN_PATH
1263 for(old=0; old<2; old++)
1266 TFindFile* findFile=new(ELeave) TFindFile(aFs);
1267 CleanupStack::PushL(findFile);
1268 TParse* fileNameParser=new(ELeave) TParse;
1269 CleanupStack::PushL(fileNameParser);
1270 CDir* directory=NULL;
1272 #ifdef SUPPORT_OLD_PLUGIN_PATH
1273 _LIT(KNotifierPlugInOldSearchPath,"\\system\\tnotifiers\\");
1275 error=findFile->FindWildByDir(KNotifierPlugInExt, KNotifierPlugInOldSearchPath, directory);
1278 error=findFile->FindWildByDir(KNotifierPlugInExt, KNotifierPlugInSearchPath, directory);
1279 for (; error!=KErrNotFound; error=findFile->FindWild(directory))
1281 CleanupStack::PushL(directory);
1282 User::LeaveIfError(error);
1283 const TInt numberOfEntries=directory->Count();
1284 for (TInt i=0; i<numberOfEntries; ++i)
1286 const TEntry& entry=(*directory)[i];
1287 fileNameParser->SetNoWild(entry.iName, &findFile->File(), NULL); // findFile->File() returns a reference rather than an object, therefore taking the address of it is fine
1289 if (entry.iType[0].iUid==0x10000079)
1292 if( (entry.iType[1]==KUidTextNotifierPlugInV2))
1294 // Its a notifier...
1295 TPtrC path(fileNameParser->DriveAndPath());
1296 TPtrC name(fileNameParser->NameAndExt());
1297 DoAddPlugInL(path,name,entry.iType);
1301 CleanupStack::PopAndDestroy(); // directory
1305 CleanupStack::PopAndDestroy(2); // fileNameParser and findFile
1306 #ifdef SUPPORT_OLD_PLUGIN_PATH
1311 LOCAL_C void DeleteTemp(TAny* aPtr)
1313 CArrayPtr<MNotifierBase2>* array=REINTERPRET_CAST(CArrayPtr<MNotifierBase2>*,aPtr);
1314 const TInt count=array->Count();
1315 for (TInt ii=0;ii<count;ii++)
1317 (*array)[ii]->Release();
1324 void CNotifierManager::DoAddPlugInL(const TDesC& aPath,const TDesC& aFileName,const TUidType& aUidType)
1327 User::LeaveIfError(lib.Load(aFileName,aPath,aUidType));
1328 CleanupClosePushL(lib);
1329 iLibraries->AppendL(lib);
1330 CleanupStack::Pop(); // lib
1331 TLibraryFunction libEntry=lib.Lookup(1);
1332 CArrayPtr<MNotifierBase2>* array=REINTERPRET_CAST(CArrayPtr<MNotifierBase2>*,(libEntry)());
1333 User::LeaveIfNull(array);
1334 CleanupStack::PushL(TCleanupItem(DeleteTemp,array));
1335 while (array->Count()>0)
1337 MNotifierBase2* notif=(*array)[0];
1339 iObservedList->AppendL(notif);
1341 notif->SetManager(this);
1343 MNotifierBase2::TNotifierInfo info=notif->RegisterL();
1344 if (!iChannelMonitor->AlreadyHasChannel(info.iChannel))
1345 iChannelMonitor->AddNewChannelL(info.iChannel);
1347 CleanupStack::PopAndDestroy(); // array
1350 CNotifierManager::CNotifierManager()
1353 void CNotifierManager::ConstructL()
1355 iObservedList=new(ELeave) CArrayPtrSeg<MNotifierBase2>(6);
1356 iLibraries=new(ELeave) CArrayFixFlat<RLibrary>(2);
1357 iChannelMonitor=CChannelMonitor::NewL();
1358 iActivityMonitor=CActivityMonitor::NewL();
1359 iQueue=CNotifierQueue::NewL();
1362 struct SActivityCleanup
1364 CActivityMonitor* iMonitor;
1369 LOCAL_C void CleanupActivityMonitor(TAny* aPtr)
1371 SActivityCleanup& cleanup=*REINTERPRET_CAST(SActivityCleanup*,aPtr);
1372 cleanup.iMonitor->Remove(cleanup.iNotifier,cleanup.iClientId);
1375 void CNotifierManager::NotifierStartL(TUid aNotifierUid,const TDesC8& aBuffer,TPtrC8* aResponse,TInt aClientId)
1377 TInt result=KErrNotFound;
1378 const TInt count=iObservedList->Count();
1379 for (TInt ii=0;ii<count;ii++)
1381 MNotifierBase2* notif=(*iObservedList)[ii];
1382 MNotifierBase2::TNotifierInfo info=notif->Info();
1383 if (info.iUid==aNotifierUid)
1385 if (iActivityMonitor->IsNotifierActive(aNotifierUid,info.iChannel))
1387 result=KErrAlreadyExists;
1389 else if (info.iPriority>iChannelMonitor->ActivityLevel(info.iChannel))
1392 MNotifierBase2::TNotifierPriority priority;
1393 const TBool channelWasActive=iActivityMonitor->IsChannelActive(info.iChannel,notifier,priority);
1394 iActivityMonitor->AddL(info,aClientId);
1395 SActivityCleanup cleanup;
1396 cleanup.iMonitor=iActivityMonitor;
1397 cleanup.iNotifier=aNotifierUid;
1398 cleanup.iClientId=aClientId;
1399 CleanupStack::PushL(TCleanupItem(CleanupActivityMonitor,&cleanup));
1400 TPtrC8 response(notif->StartL(aBuffer));
1402 aResponse->Set(response);
1403 CleanupStack::Pop(); // cleanup;
1404 if (channelWasActive)
1406 for (TInt jj=0;jj<count;jj++)
1408 MNotifierBase2* notifForUpdate=(*iObservedList)[ii];
1409 MNotifierBase2::TNotifierInfo infoForUpdate=notifForUpdate->Info();
1410 if (infoForUpdate.iUid==notifier && infoForUpdate.iChannel==info.iChannel)
1412 TRAP_IGNORE(notifForUpdate->UpdateL(KNotifierPaused));
1416 iChannelMonitor->UpdateChannel(info.iChannel,info.iPriority);
1417 if (result!=ENotExtRequestQueued)
1419 result=ENotExtRequestCompleted;
1424 if (iQueue->IsAlreadyQueued(info.iUid,info.iChannel))
1426 result=KErrAlreadyExists;
1430 CQueueItem* queueCopy=CQueueItem::NewL(info,aBuffer,aClientId);
1431 CleanupStack::PushL(queueCopy);
1432 iQueue->QueueItemL(queueCopy);
1433 CleanupStack::Pop(); // queueCopy
1434 result=ENotExtRequestQueued;
1439 User::LeaveIfError(result);
1442 TInt CNotifierManager::NotifierUpdateL(TUid aNotifierUid,const TDesC8& aBuffer,TDes8* aResponse,TInt aClientId)
1444 TInt result=KErrNotFound;
1445 const TInt count=iObservedList->Count();
1446 for (TInt ii=0;ii<count;ii++)
1448 MNotifierBase2* notif=(*iObservedList)[ii];
1449 MNotifierBase2::TNotifierInfo info=notif->Info();
1450 if (info.iUid==aNotifierUid)
1452 if (iActivityMonitor->IsNotifierActive(aNotifierUid,info.iChannel))
1454 if (!iActivityMonitor->IsClientPresent(aNotifierUid,info.iChannel,aClientId))
1456 iActivityMonitor->AddL(info,aClientId);
1458 if (aResponse==NULL)
1460 notif->UpdateL(aBuffer);
1464 aResponse->Copy(notif->UpdateL(aBuffer));
1469 ; // not all channels have been started yet so update the queue
1477 void CNotifierManager::NotifierStartAndGetResponseL(TUid aNotifierUid,const TDesC8& aBuffer,TInt aReplySlot,
1478 const RMessage2& aMessage,TInt aClientId,TBool& aCleanupComplete)
1480 NotifierStartAndGetResponseL(aNotifierUid,TUid::Null(),aBuffer,aReplySlot,aMessage,aClientId,aCleanupComplete);
1483 void CNotifierManager::NotifierStartAndGetResponseL(TUid aNotifierUid,TUid aChannelUid,const TDesC8& aBuffer,TInt aReplySlot,
1484 const RMessage2& aMessage,TInt aClientId,TBool& aCleanupComplete)
1486 TInt result=KErrNotFound;
1487 const TInt count=iObservedList->Count();
1488 for (TInt ii=0;ii<count;ii++)
1490 MNotifierBase2* notif=(*iObservedList)[ii];
1491 MNotifierBase2::TNotifierInfo info=notif->Info();
1492 if (info.iUid==aNotifierUid && (aChannelUid==TUid::Null() || info.iChannel==aChannelUid))
1494 if (iActivityMonitor->IsNotifierActive(aNotifierUid,info.iChannel))
1496 notif->StartL(aBuffer,aReplySlot,aMessage); // asynch notifier can decide whether to support multiple clients
1499 else if (info.iPriority>iChannelMonitor->ActivityLevel(info.iChannel))
1502 MNotifierBase2::TNotifierPriority priority;
1503 const TBool channelWasActive=iActivityMonitor->IsChannelActive(info.iChannel,notifier,priority);
1504 iActivityMonitor->AddL(info,aClientId);
1505 SActivityCleanup activityCleanup;
1506 activityCleanup.iMonitor=iActivityMonitor;
1507 activityCleanup.iNotifier=aNotifierUid;
1508 activityCleanup.iClientId=aClientId;
1509 CleanupStack::PushL(TCleanupItem(CleanupActivityMonitor,&activityCleanup));
1510 aCleanupComplete=EFalse;
1511 // IMPORTANT, aMessage needs to be a full RMessage object until suport for V1 notifiers is removed
1512 // I.e. until CNotifierBaseAdaptor is removed
1513 notif->StartL(aBuffer,aReplySlot,aMessage);
1514 CleanupStack::Pop(&activityCleanup);
1515 if (channelWasActive)
1517 for (TInt jj=0;jj<count;jj++)
1519 MNotifierBase2* notifForUpdate=(*iObservedList)[ii];
1520 MNotifierBase2::TNotifierInfo infoForUpdate=notifForUpdate->Info();
1521 if (infoForUpdate.iUid==notifier && infoForUpdate.iChannel==info.iChannel)
1523 TRAP_IGNORE(notifForUpdate->UpdateL(KNotifierPaused));
1527 iChannelMonitor->UpdateChannel(info.iChannel,info.iPriority);
1532 if (iQueue->IsAlreadyQueued(info.iUid,info.iChannel))
1534 result=KErrAlreadyExists;
1538 CQueueItem* queueCopy=CQueueItem::NewL(info,aBuffer,aReplySlot,aMessage,aClientId);
1539 CleanupStack::PushL(queueCopy);
1540 iQueue->QueueItemL(queueCopy);
1541 CleanupStack::Pop(queueCopy);
1542 result=ENotExtRequestQueued;
1547 User::LeaveIfError(result);
1550 TInt CNotifierManager::NotifierCancel(TUid aNotifierUid)
1552 TInt result=KErrNotFound;
1553 const TInt count=iObservedList->Count();
1554 for (TInt ii=0;ii<count;ii++)
1556 MNotifierBase2* notif=(*iObservedList)[ii];
1557 MNotifierBase2::TNotifierInfo info=notif->Info();
1558 if (info.iUid==aNotifierUid)
1561 iActivityMonitor->RemoveNotifier(aNotifierUid,info.iChannel);
1562 MNotifierBase2::TNotifierPriority priority=MNotifierBase2::ENotifierPriorityLowest;
1564 //check channel activity and get highest priority on channnel
1565 if (iActivityMonitor->IsChannelActive(info.iChannel,notifier,priority))
1568 //check if priority of a queued item on the same channel is
1570 MNotifierBase2::TNotifierPriority queuePriority=
1571 (MNotifierBase2::TNotifierPriority)iQueue->GetHighestQueuePriority(info.iChannel);
1572 if (queuePriority>priority)
1574 iChannelMonitor->UpdateChannel(info.iChannel,MNotifierBase2::ENotifierPriorityLowest);
1575 CQueueItem* next=iQueue->FetchItem(info.iChannel);
1578 TUid notif=next->iInfo.iUid;
1579 TRAPD(err,StartFromQueueL(next));
1582 NotifierCancel(notif);
1588 for (TInt jj=0;jj<count;jj++)
1590 MNotifierBase2* notifForUpdate=(*iObservedList)[ii];
1591 MNotifierBase2::TNotifierInfo infoForUpdate=notifForUpdate->Info();
1592 if (infoForUpdate.iUid==notifier && infoForUpdate.iChannel==info.iChannel)
1594 TRAP_IGNORE(notifForUpdate->UpdateL(KNotifierPaused));
1597 iChannelMonitor->UpdateChannel(info.iChannel,priority);
1602 iChannelMonitor->UpdateChannel(info.iChannel,MNotifierBase2::ENotifierPriorityLowest);
1603 CQueueItem* next=iQueue->FetchItem(info.iChannel);
1606 TUid notif=next->iInfo.iUid;
1607 TRAPD(err,StartFromQueueL(next));
1610 NotifierCancel(notif);
1620 struct SCleanupMessage
1623 RMessage2* iMessage;
1626 LOCAL_C void CleanupStartAndGetResponse(TAny* aPtr)
1628 SCleanupMessage& cleanup=*REINTERPRET_CAST(SCleanupMessage*,aPtr);
1629 if (cleanup.iDoCleanup)
1631 cleanup.iMessage->Complete(KErrNoMemory);
1635 void CNotifierManager::StartFromQueueL(CQueueItem* aItem)
1637 CleanupStack::PushL(aItem);
1638 TPtr8 buffer=aItem->iBuffer->Des();
1639 if (aItem->iAsynchronous)
1641 SCleanupMessage cleanup;
1642 TBool doCleanup=ETrue;
1643 cleanup.iDoCleanup=&doCleanup;
1644 cleanup.iMessage=&aItem->iMessage;
1645 CleanupStack::PushL(TCleanupItem(CleanupStartAndGetResponse,&cleanup));
1646 // IMPORTANT, aItem->iMessage needs to be a full RMessage object until suport for V1 notifiers is removed
1647 // I.e. until CNotifierBaseAdaptor is removed
1648 NotifierStartAndGetResponseL(aItem->iInfo.iUid,aItem->iInfo.iChannel,buffer,aItem->iReplySlot,aItem->iMessage,aItem->iClientId,doCleanup);
1649 CleanupStack::Pop(&cleanup);
1653 NotifierStartL(aItem->iInfo.iUid,buffer,NULL,aItem->iClientId);
1655 CleanupStack::PopAndDestroy(); // aItem
1656 CQueueItem* update=iQueue->FetchItem(aItem->iInfo.iChannel);
1659 CleanupStack::PushL(update);
1660 NotifierUpdateL(update->iInfo.iUid,*update->iBuffer,NULL,update->iClientId);
1661 CleanupStack::PopAndDestroy(); // update
1662 update=iQueue->FetchItem(aItem->iInfo.iChannel);
1666 void CNotifierManager::HandleClientExit(TInt aClientId)
1668 TUid notifier=KNullUid;
1669 while (iActivityMonitor->NotifierForClient(notifier,aClientId))
1671 const TInt count=iObservedList->Count();
1672 for (TInt ii=0;ii<count;ii++)
1674 MNotifierBase2* notif=(*iObservedList)[ii];
1675 if (notif->Info().iUid==notifier)
1677 NotifierCancel(notifier);
1680 iActivityMonitor->Remove(notifier,aClientId);
1682 iActivityMonitor->RemoveClient(aClientId);
1683 iQueue->RemoveClient(aClientId);
1686 void CNotifierManager::StartNotifierL(TUid aNotifierUid,const TDesC8& aBuffer,TDes8& aResponse)
1688 TPtrC8 response(0,0);
1689 NotifierStartL(aNotifierUid,aBuffer, &response,KNullClientId);
1690 aResponse.Copy(response);
1693 void CNotifierManager::CancelNotifier(TUid aNotifierUid)
1695 NotifierCancel(aNotifierUid);
1698 void CNotifierManager::UpdateNotifierL(TUid aNotifierUid,const TDesC8& aBuffer,TDes8& aResponse)
1700 NotifierUpdateL(aNotifierUid,aBuffer,&aResponse,KNullClientId);
1704 // class CChannelMonitor
1707 CChannelMonitor* CChannelMonitor::NewL()
1709 CChannelMonitor* self=new(ELeave) CChannelMonitor;
1713 TBool CChannelMonitor::AlreadyHasChannel(TUid aChannel)const
1715 const TInt count=iMonitor.Count();
1716 for (TInt ii=0;ii<count;ii++)
1718 if (iMonitor[ii].iChannel==aChannel)
1724 TInt CChannelMonitor::ActivityLevel(TUid aChannel) const
1726 const TInt count=iMonitor.Count();
1727 for (TInt ii=0;ii<count;ii++)
1729 TChannelActivity activity=iMonitor[ii];
1730 if (activity.iChannel==aChannel)
1731 return activity.iHighestPriorityRunning;
1736 void CChannelMonitor::UpdateChannel(TUid aChannel,TInt aLevel)
1738 const TInt count=iMonitor.Count();
1739 for (TInt ii=0;ii<count;ii++)
1741 TChannelActivity& activity=iMonitor[ii];
1742 if (activity.iChannel==aChannel)
1744 activity.iHighestPriorityRunning=aLevel;
1750 CChannelMonitor::CChannelMonitor()
1755 // class CNotifierActivity
1758 CNotifierActivity* CNotifierActivity::NewLC(const MNotifierBase2::TNotifierInfo& aInfo,TInt aClientId)
1760 CNotifierActivity* self=new(ELeave) CNotifierActivity(aInfo);
1761 CleanupStack::PushL(self);
1762 self->ConstructL(aClientId);
1766 CNotifierActivity::~CNotifierActivity()
1768 iClientArray.Reset();
1771 TInt CNotifierActivity::Find(TInt aClientId) const
1773 TInt index=KErrNotFound;
1774 const TInt count=iClientArray.Count();
1775 for (TInt ii=0;ii<count;ii++)
1777 TInt clientId=iClientArray[ii];
1778 if (clientId==aClientId)
1787 CNotifierActivity::CNotifierActivity(const MNotifierBase2::TNotifierInfo& aInfo)
1788 : iInfo(aInfo), iClientArray(1)
1791 void CNotifierActivity::ConstructL(TInt aClientId)
1793 iClientArray.AppendL(aClientId);
1797 // class CActivityMonitor
1800 CActivityMonitor* CActivityMonitor::NewL()
1802 CActivityMonitor* self=new(ELeave) CActivityMonitor();
1806 CActivityMonitor::~CActivityMonitor()
1808 iMonitor.ResetAndDestroy();
1811 void CActivityMonitor::AddL(const MNotifierBase2::TNotifierInfo& aInfo,TInt aClientId)
1813 const TInt index=Find(aInfo.iUid,aInfo.iChannel);
1814 if (index==KErrNotFound)
1816 CNotifierActivity* activity=CNotifierActivity::NewLC(aInfo,aClientId);
1817 iMonitor.AppendL(activity);
1818 CleanupStack::Pop(); // activity
1822 iMonitor[index]->iClientArray.AppendL(aClientId);
1826 void CActivityMonitor::Remove(TUid aNotifierUid,TInt aClientId)
1828 const TInt index=Find(aNotifierUid);
1829 if (index!=KErrNotFound)
1831 CNotifierActivity* activity=iMonitor[index];
1832 const TInt clientIndex=activity->Find(aClientId);
1833 if (clientIndex!=KErrNotFound)
1835 if (activity->iClientArray.Count()==1)
1838 iMonitor.Delete(index);
1842 activity->iClientArray.Delete(index);
1848 void CActivityMonitor::RemoveNotifier(TUid aNotifierUid,TUid aChannel)
1850 const TInt index=Find(aNotifierUid,aChannel);
1851 if (index!=KErrNotFound)
1853 delete iMonitor[index];
1854 iMonitor.Delete(index);
1858 void CActivityMonitor::RemoveClient(TInt aClientId)
1861 while (ii<iMonitor.Count())
1863 CNotifierActivity* ptr=iMonitor[ii];
1864 TInt index=ptr->Find(aClientId);
1865 if (index!=KErrNotFound)
1867 ptr->iClientArray.Delete(index);
1869 if (ptr->iClientArray.Count()==0)
1871 iMonitor.Delete(ii);
1880 TBool CActivityMonitor::IsNotifierActive(TUid aNotifierUid,TUid aChannel) const
1882 const TInt index=Find(aNotifierUid,aChannel);
1883 return (index!=KErrNotFound);
1886 TBool CActivityMonitor::IsClientPresent(TUid aNotifierUid,TUid aChannel,TInt aClientId) const
1889 const TInt index=Find(aNotifierUid,aChannel);
1890 if (index!=KErrNotFound)
1892 found=(iMonitor[index]->Find(aClientId)!=KErrNotFound);
1897 TBool CActivityMonitor::IsChannelActive(TUid aChannel,TUid& aNotifier,MNotifierBase2::TNotifierPriority& aHighestPriority) const
1900 const TInt count=iMonitor.Count();
1901 for (TInt ii=0;ii<count;ii++)
1903 MNotifierBase2::TNotifierInfo info=iMonitor[ii]->iInfo;
1904 if (info.iChannel==aChannel)
1907 if ((MNotifierBase2::TNotifierPriority)info.iPriority>aHighestPriority)
1909 aNotifier=info.iUid;
1910 aHighestPriority=(MNotifierBase2::TNotifierPriority)info.iPriority;
1917 TBool CActivityMonitor::NotifierForClient(TUid& aNotifierUid,TInt aClientId) const
1919 TBool isOnlyClient=EFalse;
1920 aNotifierUid=KNullUid;
1921 const TInt count=iMonitor.Count();
1922 for (TInt ii=0;ii<count;ii++)
1924 CNotifierActivity* ptr=iMonitor[ii];
1925 if (ptr->Find(aClientId)!=KErrNotFound)
1927 aNotifierUid=ptr->iInfo.iUid;
1928 isOnlyClient=ptr->iClientArray.Count()==1;
1932 return isOnlyClient;
1935 CActivityMonitor::CActivityMonitor()
1939 TInt CActivityMonitor::Find(TUid aNotifierUid) const
1941 TInt index=KErrNotFound;
1942 const TInt count=iMonitor.Count();
1943 for (TInt ii=0;ii<count;ii++)
1945 if (iMonitor[ii]->iInfo.iUid==aNotifierUid)
1954 TInt CActivityMonitor::Find(TUid aNotifierUid,TUid aChannel) const
1956 TInt index=KErrNotFound;
1957 const TInt count=iMonitor.Count();
1958 for (TInt ii=0;ii<count;ii++)
1960 CNotifierActivity* ptr=iMonitor[ii];
1961 if (ptr->iInfo.iUid==aNotifierUid && ptr->iInfo.iChannel==aChannel)
1974 CQueueItem* CQueueItem::NewL(const MNotifierBase2::TNotifierInfo& aInfo,const TDesC8& aBuffer,
1975 TInt aReplySlot,const RMessage2& aMessage,TInt aClientId) //Asynchronous
1977 CQueueItem* self=new(ELeave) CQueueItem(aInfo);
1978 CleanupStack::PushL(self);
1979 self->ConstructL(aBuffer,aMessage,aClientId,aReplySlot);
1980 CleanupStack::Pop(); // self
1984 CQueueItem* CQueueItem::NewL(const MNotifierBase2::TNotifierInfo& aInfo,const TDesC8& aBuffer,TInt aClientId) //synchronous
1986 CQueueItem* self=new(ELeave) CQueueItem(aInfo);
1987 CleanupStack::PushL(self);
1988 self->ConstructL(aBuffer,aClientId);
1989 CleanupStack::Pop(); // self
1993 CQueueItem::~CQueueItem()
1998 CQueueItem::CQueueItem(const MNotifierBase2::TNotifierInfo& aInfo)
2002 void CQueueItem::ConstructL(const TDesC8& aBuffer,TInt aClientId)
2004 iBuffer=aBuffer.AllocL();
2005 iClientId=aClientId;
2006 iAsynchronous=EFalse;
2009 void CQueueItem::ConstructL(const TDesC8& aBuffer,const RMessage2& aMessage,TInt aClientId,TInt aReplySlot)
2011 iBuffer=aBuffer.AllocL();
2012 iAsynchronous=ETrue;
2014 iClientId=aClientId;
2015 iReplySlot=aReplySlot;
2019 // class CNotifierQueue
2022 CNotifierQueue* CNotifierQueue::NewL()
2024 CNotifierQueue* self=new(ELeave) CNotifierQueue;
2028 CQueueItem* CNotifierQueue::FetchItem(TUid aChannel)
2030 CQueueItem* result=NULL;
2031 const TInt count=iQueue.Count();
2032 TInt priority=MNotifierBase2::ENotifierPriorityLowest-1;
2033 TInt index=KErrNotFound;
2034 for (TInt ii=0;ii<count;ii++)
2036 CQueueItem* item=iQueue[ii];
2037 if (item->iInfo.iChannel==aChannel && item->iInfo.iPriority>priority)
2040 priority=item->iInfo.iPriority;
2044 if (index!=KErrNotFound)
2046 iQueue.Delete(index);
2051 TBool CNotifierQueue::IsAlreadyQueued(TUid aNotifier,TUid aChannel) const
2054 const TInt count=iQueue.Count();
2055 for (TInt ii=0;ii<count;ii++)
2057 CQueueItem* item=iQueue[ii];
2058 if (item->iInfo.iUid==aNotifier && item->iInfo.iChannel==aChannel)
2067 void CNotifierQueue::RemoveClient(TInt aClientId)
2069 const TInt count=iQueue.Count();
2070 for (TInt ii=count-1;ii>=0;ii--)
2072 CQueueItem* item=iQueue[ii];
2073 TInt clientId=item->iClientId;
2074 if (clientId==aClientId)
2082 TInt CNotifierQueue::GetHighestQueuePriority(TUid aChannel)
2084 const TInt count=iQueue.Count();
2085 TInt priority=MNotifierBase2::ENotifierPriorityLowest-1;
2087 for (TInt ii=0;ii<count;ii++)
2089 CQueueItem* item=iQueue[ii];
2090 if (item->iInfo.iChannel==aChannel && item->iInfo.iPriority>priority)
2092 priority=item->iInfo.iPriority;
2100 void CWsActiveScheduler::New()
2102 // Create and install the active scheduler.
2106 CWsActiveScheduler *pA=new CWsActiveScheduler;
2107 __ASSERT_ALWAYS(pA!=NULL,Fault(ECreateScheduler));
2108 CActiveScheduler::Install(pA);
2111 void CWsActiveScheduler::Error(TInt) const
2113 // Called if any Run() method leaves.
2119 TInt NotifierServerThread(TAny*)
2121 CTrapCleanup* CleanUpStack=CTrapCleanup::New();
2122 CWsActiveScheduler::New();
2123 TRAP_IGNORE(CNotifierServer::NewL());
2124 CNotifierSession::NotifierSemaphore.Signal();
2125 CWsActiveScheduler::Start();
2126 delete CleanUpStack;
2131 _LIT(KLitKeyDataDllNameBase, "EKDATA");
2132 _LIT(TwoDigExt,".%02d");
2134 GLDEF_C TInt E32Main()
2136 UserSvr::WsRegisterThread();
2137 UserSvr::WsRegisterSwitchOnScreenHandling(ETrue);
2138 User::SetProcessCritical(User::ESystemPermanent);
2139 User::SetCritical(User::ESystemPermanent);
2141 CWsActiveScheduler::New();
2146 KeyTranslator=CKeyTranslator::New();
2148 Fault(ENoKeyboardTranslator);
2150 // Change keyboard mapping according to information in the HAL
2151 // This code is the same as WSERV
2153 if (HAL::Get(HALData::EKeyboardIndex,keyboardIndex)==KErrNone)
2155 TBuf<16> keyDataDllName(KLitKeyDataDllNameBase);
2156 keyDataDllName.AppendFormat(TwoDigExt, keyboardIndex);
2157 KeyTranslator->ChangeKeyData(keyDataDllName);
2160 KeyRepeat=new(ELeave) CKeyRepeat(CKeyRepeat::EKeyRepeatPriority);
2161 TRAPD(r,KeyRepeat->ConstructL());
2163 User::Panic(_L("KEYREPEAT"),r);
2166 if (UserSvr::TestBootSequence())
2168 RDebug::Print(_L("WS_MAIN: TestBootSequence=TRUE, not loading ESHELL.EXE"));
2171 if (EmulatorAutoRun())
2172 { // don't start ESHELL if we used a self-bootstrapping EXE
2178 r=shell.Create(KShellProcessName, KShellCommandLine);
2179 __ASSERT_ALWAYS(r==KErrNone,Fault(ECreateShell));
2185 r=CNotifierSession::NotifierSemaphore.CreateLocal(0);
2187 Fault(ECreateNotifierSemaphore);
2188 r=t.Create(_L("NotifierServer"),NotifierServerThread,KDefaultStackSize,0x2000,0x100000,NULL);
2190 Fault(ECreateNotifierThread);
2192 CNotifierSession::NotifierSemaphore.Wait();
2194 CWsActiveScheduler::Start();
2195 UserSvr::ReleaseEventHook();