Update contrib.
1 // Copyright (c) 1994-2010 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.
27 #include "windowgroup.h"
30 #include "../CLIENT/w32comm.h"
33 #include <u32hal.h> // EHalGroupEmulator
36 GLREF_C void HeapDump();
38 GLREF_D CDebugLogBase *wsDebugLog;
40 GLREF_D TPtr nullDescriptor;
42 TWsCmdHeaderBase CWsClient::iCurrentCommand;
43 TBuf8<EClientBufferMaxSize> CWsClient::iCmdBuf;
44 TUint CWsClient::iConnectionId=CDebugLogBase::EDummyConnectionId+1;
45 CArrayFixFlat<TWsCursorArrayItem> *CWsClient::iSystemPointerCursors=NULL;
46 TInt CWsClient::iDefaultSystemPointerCursorIndex=0; //Negative when there isn't one
47 CWsPointerCursor *CWsClient::iDefaultSystemPointerCursor;
48 CWsClient *CWsClient::iSystemPointerCursorListOwner=NULL;
49 CArrayFixFlat<TWsCursorArrayItem> *CWsClient::iTextCursorArray=NULL;
50 TInt CWsClient::iReply;
51 TInt CWsClient::iReplyOffset;
52 CWsClient *CWsClient::iCurrentClient;
55 Used for enforcing the redraw calling convention (in preparation for BR2412) in emulator builds.
56 When enabled this will panic any client calling a CWindowGc draw operation outside a
57 RWindow::BeginRedraw() / RWindow::EndRedraw() pair (known as non-redraw drawing).
59 Enable by adding "debug_wserv_exe_EnforceRedrawCallingConvention X" to epoc.ini
60 where X is either 0 (zero) for "off" or 1 (one) for "on".
62 Then enable globaly in WServ AutoFlush by defining __AUTO_FLUSH in ../client/client.h
63 or locally by calling RWsSession::SetAutoFlush(ETrue) for a specific client programatically,
64 or locally pressing Ctrl-Alt-Shift-F in the emulator.
66 TBool CWsClient::iDebug_EnforceRedrawCallingConvention = EFalse;
68 _LIT(KWSERVSessionPanicCategory,"WSERV");
70 TKeyArrayFix CursorKey(_FOFF(TWsCursorArrayItem,iIndex),ECmpTInt);
72 static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_WriteDeviceData,ECapabilityWriteDeviceData);
73 static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_SwEvent,ECapabilitySwEvent);
74 static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_PowerMgmt,ECapabilityPowerMgmt);
76 CWsClient::CWsClient(RThread aClient) : iClient(aClient), iGraphicMessageQueue(this), iIsInitialised(EFalse)
79 ,iRemoveKeyCode(ETrue)
82 iScreen=CWsTop::Screen(); //## Need to find better way to set this
85 CWsClient::~CWsClient()
87 WindowServer().RemoveAllGraphicDrawers(*this); // deindexes all graphic drawers owned by this client
89 delete iTempCustomTextCursor.iCursor;
90 FreeSystemPointerCursorList();
91 CWsTop::ClientDestroyed(this);
94 _LIT(ClientDestuct,"Client %d destructing");
95 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,ClientDestuct, iConnectionHandle);
97 iInternalFlags|=EClientIsClosing;
101 delete iPriorityKeyEvent;
103 CWsTop::SessionExited(this);
104 iScreen->Update(); /* Mathias: Don't care about multiple screens yet
105 CWsTop::UpdateAllScreens(EFalse);
110 void CWsClient::CompleteInitializationL()
112 iObjectIndex=new(ELeave) CWsObjectIx();
113 iObjectIndex->ConstructL();
114 iEventQueue=new(ELeave) CEventQueue(this);
115 iEventQueue->ConstructL();
116 iRedrawQueue=new(ELeave) CRedrawQueue(this);
117 iRedrawQueue->ConstructL();
118 iPriorityKeyEvent=new(ELeave) CPriorityKey(this);
119 CWsCliObj::NewL(this);
120 iComputeMode=RWsSession::EPriorityControlComputeOff;
121 CWsTop::NewSession(this);
124 TBool halValue = EFalse;
125 if (UserSvr::HalFunction(EHalGroupEmulator, EEmulatorHalIntProperty,
126 (TAny*)"debug_wserv_exe_EnforceRedrawCallingConvention", &halValue) == KErrNone)
128 iDebug_EnforceRedrawCallingConvention = halValue;
132 iIsInitialised = ETrue;
135 TBool CWsClient::DebugEnforceRedrawCallingConvention()
137 return iDebug_EnforceRedrawCallingConvention;
140 void CWsClient::StartInitializationL(TUint aConnectionHandle)
143 wsDebugLog->NewClient(aConnectionHandle);
146 PPanic(EWservPanicReInitialise);
150 iConnectionHandle=aConnectionHandle;
151 CompleteInitializationL();
155 void CWsClient::HandleToWindow(TInt handle,CWsWindowBase **pWin)
157 // Convert a handle to object checking it is of the correct type.
160 if ((*pWin=(CWsWindowBase *)HandleToObjUntyped(handle))==NULL ||
161 ((*pWin)->Type()!=WS_HANDLE_WINDOW && (*pWin)->Type()!=WS_HANDLE_GROUP_WINDOW))
162 PPanic(EWservPanicWindow);
165 void CWsClient::HandleToClientWindow(TInt handle,CWsClientWindow **pWin)
167 // Convert a handle to object checking it is of the correct type.
170 if ((*pWin=(CWsClientWindow *)HandleToObj(handle, WS_HANDLE_WINDOW))==NULL)
171 PPanic(EWservPanicWindow);
174 void CWsClient::CreateNewPointerCursorL(const TWsClCmdCreatePointerCursor &aCmd)
176 CWsPointerCursor *pc=new(ELeave) CWsPointerCursor(this);
177 CleanupStack::PushL(pc);
178 pc->ConstructL(aCmd);
182 // Create a new custom text cursor
183 void CWsClient::StartSetCustomTextCursorL(const TWsClCmdCustomTextCursorData& aCmd)
185 if (!iTextCursorArray)
187 const TInt textCursorArrayGranularity = 4;
188 iTextCursorArray = new(ELeave) CArrayFixFlat<TWsCursorArrayItem>(textCursorArrayGranularity);
191 if (FindCursorArrayItem(iTextCursorArray, aCmd.identifier, arrayIndex))
192 User::Leave(KErrAlreadyExists);
193 delete iTempCustomTextCursor.iCursor;
194 iTempCustomTextCursor.iCursor = NULL;
195 iTempCustomTextCursor.iCursor = new(ELeave) CWsCustomTextCursor(this, aCmd.alignment);
196 static_cast<CWsCustomTextCursor*>(iTempCustomTextCursor.iCursor)->ConstructL(aCmd.flags);
197 iTempCustomTextCursor.iIndex = aCmd.identifier;
200 // Add new custom text cursor to global list
201 void CWsClient::CompleteSetCustomTextCursorL(TInt aError)
203 if (aError != KErrNone)
205 delete iTempCustomTextCursor.iCursor;
206 iTempCustomTextCursor.iCursor = NULL;
210 TWsCursorArrayItem entry = iTempCustomTextCursor;
211 iTempCustomTextCursor.iCursor = NULL;
212 CleanupStack::PushL(entry.iCursor);
215 if (FindCursorArrayItem(iTextCursorArray, entry.iIndex, arrayIndex))
217 User::Leave(KErrAlreadyExists);
221 iTextCursorArray->InsertIsqL(entry, CursorKey);
224 CleanupStack::Pop(entry.iCursor);
227 CWsCustomTextCursor* CWsClient::FindCustomTextCursor(TInt aIdentifier)
230 if (!FindCursorArrayItem(iTextCursorArray, aIdentifier, arrayIndex))
234 return TextCursor(arrayIndex);
237 void CWsClient::CreateNewSpriteL(const TWsClCmdCreateSprite &aCmd)
239 CWsSprite *sprite=new(ELeave) CWsSprite(this);
240 CleanupStack::PushL(sprite);
241 sprite->ConstructL(aCmd);
245 void CWsClient::CreateNewBitmapL(const TWsClCmdCreateBitmap &aCmd)
247 DWsBitmap *bitmap=new(ELeave) DWsBitmap(this);
248 CleanupStack::PushL(bitmap);
249 bitmap->ConstructL(aCmd);
253 /** Creates a new window.
255 @param aCmd The command received from the client
259 void CWsClient::CreateNewWindowL(const TWsClCmdCreateWindow &aCmd)
261 CWsWindowBase *parent;
262 HandleToWindow(aCmd.parent,&parent);
263 CWsClientWindow *win=NULL;
264 TBool deviceIsInvalid=EFalse;
265 CScreen* screen = parent->Screen();
267 if (parent->WinType()==EWinTypeGroup)
269 __ASSERT_DEBUG(!((CWsWindowGroup*)parent)->ScreenDeviceDeleted(),PPanic(EWservPanicGroupWinScreenDeviceDeleted));
270 win=new(ELeave) CWsClientWindow(this, screen);
271 deviceIsInvalid=!((CWsWindowGroup *)parent)->ScreenDeviceValid();
275 win=new(ELeave) CWsClientWindow(this, screen);
277 CleanupStack::PushL(win);
278 win->ConstructL(aCmd,parent,deviceIsInvalid);
279 CleanupStack::Pop(win);
282 void CWsClient::CreateNewWindowGroupL(const TWsClCmdCreateWindowGroup &aCmd)
284 CWsWindowGroup::NewL(this, NULL, aCmd); //Screen is set inside the ConstructL since support for multiple screens was introduced
287 void CWsClient::CreateNewAnimDllL(const TWsClCmdUnion &aParams)
289 CWsAnimDll *animDll=new(ELeave) CWsAnimDll(this);
290 CleanupStack::PushL(animDll);
291 animDll->LoadL(BufferTPtr((TText *)(aParams.LoadAnimDll+1),aParams.LoadAnimDll->length));
295 void CWsClient::CreateNewScreenDeviceL( TInt aDefaultScreenNumber, TUint aClientScreenDevicePointer)
297 DWsScreenDevice *screenDevice=new(ELeave) DWsScreenDevice( this, aDefaultScreenNumber,aClientScreenDevicePointer);
298 CleanupStack::PushL(screenDevice);
299 screenDevice->ConstructL();
300 CleanupStack::Pop(screenDevice);
301 if (iPrimaryScreenDevice==NULL)
303 iPrimaryScreenDevice=screenDevice;
304 // When client create screen device, change default screen to the one specified.
305 // Client should do this immediately after establishing session
306 iScreen = iPrimaryScreenDevice->Screen();
307 InitialiseScreenDevices();
311 void CWsClient::InitialiseScreenDevices()
313 const TWsObject* ptr=iObjectIndex->FirstObject();
314 const TWsObject* end=ptr+iObjectIndex->Length();
315 WS_ASSERT_DEBUG(ptr->iObject==NULL, EWsPanicObjectIndexError);
318 if (ptr->iObject && ptr->iObject->Type()==WS_HANDLE_GROUP_WINDOW)
320 CWsWindowGroup *gw =STATIC_CAST(CWsWindowGroup*,ptr->iObject);
321 if(gw->Device()==NULL)
323 gw->SetScreenDevice(iPrimaryScreenDevice);
329 void CWsClient::CreateNewClickL(const TUid& aUid)
331 CClick *click=new(ELeave) CClick(this);
332 CleanupStack::PushL(click);
333 click->ConstructL(aUid);
334 CleanupStack::Pop(click);
337 void CWsClient::RequestComplete(TRequestStatus * &aStatus, TInt aErr)
339 Client().RequestComplete(aStatus,aErr);
342 void CWsClient::PanicCurrentClient(TClientPanic aPanic)
344 iCurrentClient->PPanic(aPanic);
347 void CWsClient::PPanic(TClientPanic aPanic) const
348 //This function is allowed to leave with out the 'L' convention for special reasons
350 SessionPanic(aPanic);
351 User::Leave(EPanicLeave);
354 void CWsClient::SessionPanic(TClientPanic aReason) const
357 wsDebugLog->Panic(iConnectionHandle, aReason);
358 if (!iInternalFlags&EPanicClientAsSoonAsPossible) // keep the first error code
360 iInternalFlags|=EPanicClientAsSoonAsPossible;
361 iPanicReason=aReason;
365 void CWsClient::SessionTerminate()
368 wsDebugLog->Panic(iConnectionHandle, 0);
370 const RThread thread=Client();
372 if (thread.Process(process)==KErrNone)
374 process.Terminate(0);
379 void CWsClient::ReplyBuf(const TDesC16 &aDes)
381 WS_ASSERT_DEBUG(!iCurrentClient->ClientMessage().IsNull(), EWsPanicInvalidMessageHandle);
382 if(iCurrentClient->ClientMessage().Write(KReplyBufferMessageSlot,aDes,iReplyOffset) != KErrNone)
383 PanicCurrentClient(EWservPanicDescriptor);
384 iReplyOffset+=aDes.Length();
386 wsDebugLog->ReplyBuf(aDes);
389 void CWsClient::ReplyBuf(const TDesC8 &aDes)
391 WS_ASSERT_DEBUG(!iCurrentClient->ClientMessage().IsNull(), EWsPanicInvalidMessageHandle);
392 if(iCurrentClient->ClientMessage().Write(KReplyBufferMessageSlot,aDes,iReplyOffset) != KErrNone)
393 PanicCurrentClient(EWservPanicDescriptor);
394 iReplyOffset+=aDes.Length();
396 wsDebugLog->ReplyBuf(aDes);
399 void CWsClient::ReplyBuf(const TAny *aSource, TInt aLength)
401 // Send a buffer to the client process.
404 TPtrC8 src(reinterpret_cast<const TUint8*>(aSource),aLength);
408 void CWsClient::ReplySize(const TSize &aSize)
410 ReplyBuf(&aSize,sizeof(aSize));
413 void CWsClient::ReplyPoint(const TPoint &aPoint)
415 ReplyBuf(&aPoint,sizeof(aPoint));
418 void CWsClient::ReplyRect(const TRect &aRect)
420 ReplyBuf(&aRect,sizeof(aRect));
423 void CWsClient::SetReply(TInt reply)
427 wsDebugLog->Reply(reply);
430 const TUint8 *CWsClient::EndOfCommandBuffer()
432 return(iCmdBuf.Ptr()+iCmdBuf.Size());
435 const TPtrC CWsClient::BufferTPtr(TText *aStart,TInt aLen)
438 if (!BufferTPtrGc(aStart,aLen,ptr))
439 PanicCurrentClient(EWservPanicBufferPtr);
443 TBool CWsClient::BufferTPtrGc(TText* aStart,TInt aLen, TPtrC& aPtr)
445 if (iCurrentCommand.iOpcode>0)
447 if ((REINTERPRET_CAST(TUint8*,aStart)<iCmdBuf.Ptr()
448 || REINTERPRET_CAST(TUint8*,aStart+aLen)>(iCmdBuf.Ptr()+iCmdBuf.Size())))
453 if (aLen>=iCurrentCommand.iCmdLength)
456 aPtr.Set(aStart,aLen);
460 const TPtrC8 CWsClient::BufferTPtr8(TUint8* aStart,TInt aLen)
462 if (iCurrentCommand.iOpcode>0 && (REINTERPRET_CAST(TUint8*,aStart)<iCmdBuf.Ptr()
463 || REINTERPRET_CAST(TUint8*,aStart+aLen)>(iCmdBuf.Ptr()+iCmdBuf.Size())))
464 PanicCurrentClient(EWservPanicBufferPtr);
465 return(TPtrC8(aStart,aLen));
469 Process a command buffer
474 void CWsClient::CommandBufL()
478 wsDebugLog->CommandBuf(iConnectionHandle);
479 RThread client = Client();
480 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, client.FullName());
484 CWsObject *destObj=NULL;
485 const TUint8 *nextCmd=iCmdBuf.Ptr();
486 const TUint8 *endCmd=nextCmd+iCmdBuf.Length();
490 const TWsCmdHeader *pCmd=(TWsCmdHeader *)nextCmd;
491 TUint opcode=pCmd->iBase.iOpcode;
492 TInt headerLen=sizeof(pCmd->iBase);
493 iCurrentCommand=pCmd->iBase;
495 // For performance reasons the handle is only included
496 // if it is different from the previous command. The EWsOpcodeHandle
497 // flag indicates whether a new handle has been included in the
498 // current command. If not we use the same handle as the previous
500 if (opcode&EWsOpcodeHandle)
502 opcode&=~EWsOpcodeHandle;
503 iCurrentCommand.iOpcode=reinterpret_cast<TUint16&>(opcode);
504 destObj=HandleToObjUntyped(pCmd->iDestHandle);
505 headerLen=sizeof(*pCmd);
508 const TAny *cmdParams=nextCmd;
509 nextCmd+=pCmd->iBase.iCmdLength;
510 if (destObj==NULL || nextCmd>endCmd) // Invalid handle or Corrupt buffer
512 SessionPanic(destObj==NULL ? EWservPanicHandle : EWservPanicBuffer);
516 iLastCommand=(nextCmd==endCmd);
518 // Storing destObj->Type() to a temporary variable objType allows the value of destObj->Type()
519 // to be used if destObj is deleted during destObj->CommandL().
520 WH_HANDLES objType=destObj->Type();
522 wsDebugLog->Command(objType, opcode, cmdParams, destObj->LogHandle());
523 destObj->CommandL(opcode, cmdParams);
525 } while(nextCmd<endCmd);
528 User::Heap().Check();
532 void CWsClient::CommandL(TInt aOpcode, const RMessage2& aMessage)
536 case EWsClOpEventReady:
537 EventReady(aMessage);
539 case EWsClOpPriorityKeyReady:
540 PriorityKeyEventReady(aMessage);
542 case EWsClOpRedrawReady:
543 RedrawEventReady(aMessage);
545 case EWsClOpGraphicMessageReady:
546 iGraphicMessageQueue.EventReady(aMessage);
550 PPanic(EWservPanicOpcode);
556 void CWsClient::CommandL(TInt aOpcode, const TAny *aCmdData)
562 case EWsClOpCreateWindowGroup:
563 CreateNewWindowGroupL(*pData.CreateWindowGroup);
565 case EWsClOpCreateWindow:
566 CreateNewWindowL(*pData.CreateWindow);
568 case EWsClOpCreateGc:
571 case EWsClOpCreateAnimDll:
572 if (!CheckBuffer(pData.LoadAnimDll->length, KMaxFileName))
573 PanicCurrentClient(EWservPanicBufferPtr);
574 CreateNewAnimDllL(pData);
576 case EWsClOpCreateGraphic:
577 CWsGraphicDrawerObject::NewL(this,pData);
579 case EWsClOpCreateScreenDevice:
581 TInt screenNumber = pData.CreateScreenDevice->screenNumber;
582 TUint clientScreenDevicePointer = pData.CreateScreenDevice->clientScreenDevicePointer;
583 if (screenNumber<0 || screenNumber>=CWsTop::NumberOfScreens())
585 PPanic(EWservPanicScreenNumber);
589 CreateNewScreenDeviceL(screenNumber,clientScreenDevicePointer);
593 case EWsClOpCreateSprite:
594 CreateNewSpriteL(*pData.CreateSprite);
596 case EWsClOpCreatePointerCursor:
597 CreateNewPointerCursorL(*pData.CreatePointerCursor);
599 case EWsClOpStartSetCustomTextCursor:
600 StartSetCustomTextCursorL(*pData.CustomTextCursorData);
602 case EWsClOpCompleteSetCustomTextCursor:
603 CompleteSetCustomTextCursorL(*pData.Int);
605 case EWsClOpCreateBitmap:
606 CreateNewBitmapL(*pData.CreateBitmap);
608 case EWsClOpCreateDirectScreenAccess:
609 CWsDirectScreenAccess::NewL(this);
611 case EWsClOpCreateClick:
612 CreateNewClickL(*pData.Uid);
614 case EWsClOpSetHotKey:
616 if(!KSecurityPolicy_SwEvent().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetHotKey API")))
618 User::Leave(KErrPermissionDenied);
620 TWindowServerEvent::SetHotKeyL(*pData.SetHotKey);
623 case EWsClOpClearHotKeys:
625 if(!KSecurityPolicy_SwEvent().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::ClearHotKeys API")))
627 User::Leave(KErrPermissionDenied);
629 TWindowServerEvent::ClearHotKeysL(*pData.UInt);
632 case EWsClOpRestoreDefaultHotKey:
633 TWindowServerEvent::ResetDefaultHotKeyL(*pData.UInt);
635 case EWsClOpSetShadowVector:
637 for(TInt i=0;i<CWsTop::NumberOfScreens();i++)
639 CScreen *screen = CWsTop::Screen(i);
640 screen->SetShadowVector(*pData.Point);
644 case EWsClOpShadowVector:
646 TPoint vector(iScreen->ShadowVector());
647 ReplyBuf(&vector,sizeof(TPoint));
650 case EWsClOpSetKeyboardRepeatRate:
652 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetKeyboardRepeatRate API")))
654 User::Leave(KErrPermissionDenied);
656 if ((pData.SetKeyboardRepeatRate->initial.Int()<0) || (pData.SetKeyboardRepeatRate->time.Int()<0))
658 User::Leave(KErrArgument);
660 CKeyboardRepeat::SetRepeatTime(pData.SetKeyboardRepeatRate->initial,pData.SetKeyboardRepeatRate->time);
663 case EWsClOpGetKeyboardRepeatRate:
665 SKeyRepeatSettings settings;
666 CKeyboardRepeat::GetRepeatTime(settings.iInitialTime,settings.iTime);
667 ReplyBuf(&settings,sizeof(settings));
670 case EWsClOpSetDoubleClick:
672 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetDoubleClick API")))
674 User::Leave(KErrPermissionDenied);
676 WsPointer::SetDoubleClick(pData.SetDoubleClick->interval,pData.SetDoubleClick->distance);
679 case EWsClOpGetDoubleClickSettings:
681 SDoubleClickSettings settings;
682 WsPointer::GetDoubleClickSettings(settings.iInterval,settings.iDistance);
683 ReplyBuf(&settings,sizeof(settings));
686 case EWsClOpEventReady:
688 case EWsClOpGetEvent:
691 case EWsClOpPurgePointerEvents:
692 PurgePointerEvents();
694 case EWsClOpEventReadyCancel:
697 case EWsClOpRedrawReady:
699 case EWsClOpRedrawReadyCancel:
702 case EWsClOpGetRedraw:
705 case EWsClOpPriorityKeyReady:
707 case EWsClOpPriorityKeyReadyCancel:
708 CancelPriorityKeyEvent();
710 case EWsClOpGetPriorityKey:
711 GetPriorityKeyData();
713 case EWsClOpNumWindowGroups:
714 SetReply(CWsWindowGroup::NumWindowGroups(EFalse, *pData.Int));
716 case EWsClOpNumWindowGroupsAllPriorities:
717 SetReply(CWsWindowGroup::NumWindowGroups(ETrue, 0));
719 case EWsClOpNumWindowGroupsOnScreen:
721 TInt screenNumber=pData.NumWinGroups->screenNumber;
722 if (screenNumber<0 || screenNumber>=CWsTop::NumberOfScreens())
724 PPanic(EWservPanicScreenNumber);
728 SetReply(CWsWindowGroup::NumWindowGroupsOnScreen(CWsTop::Screen(screenNumber)->RootWindow()->Child(),(pData.NumWinGroups->priority==EAllPriorities),pData.NumWinGroups->priority));
732 case EWsClOpWindowGroupList:
734 TInt screenNumber=pData.WindowGroupList->screenNumber;
735 if (screenNumber<KDummyScreenNumber || screenNumber>=CWsTop::NumberOfScreens())
737 PPanic(EWservPanicScreenNumber);
741 SetReply(CWsWindowGroup::SendWindowGroupListL(screenNumber,(pData.WindowGroupList->priority==EAllPriorities), pData.WindowGroupList->priority, pData.WindowGroupList->count));
745 case EWsClOpWindowGroupListAllPriorities:
747 TInt screenNumber=pData.WindowGroupList->screenNumber;
748 if (screenNumber<KDummyScreenNumber || screenNumber>=CWsTop::NumberOfScreens())
750 PPanic(EWservPanicScreenNumber);
754 SetReply(CWsWindowGroup::SendWindowGroupListL(screenNumber,ETrue, 0, pData.WindowGroupList->count));
758 case EWsClOpWindowGroupListAndChain:
759 SetReply(CWsWindowGroup::SendWindowGroupListAndChainL(EFalse, pData.WindowGroupList->priority, pData.WindowGroupList->count));
761 case EWsClOpWindowGroupListAndChainAllPriorities:
762 SetReply(CWsWindowGroup::SendWindowGroupListAndChainL(ETrue, 0, pData.WindowGroupList->count));
764 case EWsClOpGetDefaultOwningWindow:
766 TInt screenNumber = *pData.Int;
767 if (screenNumber<KDummyScreenNumber || screenNumber>=CWsTop::NumberOfScreens())
769 PPanic(EWservPanicScreenNumber);
773 CScreen* screen = (screenNumber ==KDummyScreenNumber) ? iScreen : CWsTop::Screen(screenNumber);
774 SetReply(screen->DefaultOwningWindowGroup() ? screen->DefaultOwningWindowGroup()->Identifier():0);
778 case EWsClOpGetFocusWindowGroup:
780 TInt screenNumber = *pData.Int;
781 if (screenNumber<KDummyScreenNumber || screenNumber>=CWsTop::NumberOfScreens())
783 PPanic(EWservPanicScreenNumber);
787 CWsWindowGroup::GetFocusWindowGroupL(screenNumber);
791 case EWsClOpSetWindowGroupOrdinalPosition:
792 CWsWindowGroup::WindowGroupFromIdentifierL(pData.SetWindowGroupOrdinalPosition->identifier)->SetOrdinalPosition(pData.SetWindowGroupOrdinalPosition->position);
794 case EWsClOpGetWindowGroupHandle:
795 SetReply(CWsWindowGroup::WindowGroupFromIdentifierL(*pData.Int)->ClientHandle());
797 case EWsClOpGetWindowGroupOrdinalPriority:
798 SetReply(CWsWindowGroup::WindowGroupFromIdentifierL(*pData.Int)->OrdinalPriority());
800 case EWsClOpGetWindowGroupClientThreadId:
802 TThreadId id=CWsWindowGroup::WindowGroupFromIdentifierL(*pData.Int)->WsOwner()->Client().Id();
803 ReplyBuf(&id,sizeof(id));
806 case EWsClOpSendEventToWindowGroup:
808 CWsWindowGroup *group=CWsWindowGroup::WindowGroupFromIdentifierL(pData.SendEventToWindowGroup->parameter);
809 TWsEvent event=pData.SendEventToWindowGroup->event;
810 event.SetHandle(group->ClientHandle());
811 // Events in enum TEventCode is protected by capabilities
812 if (group->WsOwner()!=this && event.Type()>=EEventNull && event.Type()<EEventUser)
814 if (event.Type()<EEventPowerMgmt || event.Type()>=EEventReserved)
816 if(!KSecurityPolicy_SwEvent().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SendEventToWindowGroup API")))
817 User::Leave(KErrPermissionDenied);
821 if (!KSecurityPolicy_PowerMgmt().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SendEventToWindowGroup API")))
822 User::Leave(KErrPermissionDenied);
825 if (event.Type()==EEventKey && event.Key()->iRepeats!=0)
826 CKeyboardRepeat::CancelRepeat(NULL); //Otherwise we will trip an invarient
827 if (!group->EventQueue()->QueueEvent(event))
828 User::Leave(KErrNoMemory);
831 case EWsClOpSendEventToAllWindowGroup:
832 case EWsClOpSendEventToAllWindowGroupPriority:
833 case EWsClOpSendEventToOneWindowGroupPerClient:
835 TWsEvent event=pData.SendEventToWindowGroup->event;
838 User::Leave(KErrArgument);
840 if(event.Type()<EEventUser)
842 if (event.Type()<EEventPowerMgmt || event.Type()>=EEventReserved)
844 if(!KSecurityPolicy_SwEvent().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SendEventToAllWindowGroup API")))
845 User::Leave(KErrPermissionDenied);
849 if (!KSecurityPolicy_PowerMgmt().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SendEventToAllWindowGroup API")))
850 User::Leave(KErrPermissionDenied);
853 if (!CWsWindowGroup::SendEventToAllGroups(aOpcode!=EWsClOpSendEventToAllWindowGroupPriority
854 ,aOpcode==EWsClOpSendEventToOneWindowGroupPerClient,*pData.SendEventToWindowGroup))
855 User::Leave(KErrNoMemory);
858 case EWsClOpSendMessageToWindowGroup:
860 CWsWindowGroup *group=CWsWindowGroup::WindowGroupFromIdentifierL(pData.SendMessageToWindowGroup->identifierOrPriority);
861 if (group->WsOwner()!=this)
863 if (!KSecurityPolicy_SwEvent().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SendMessageToWindowGroup API")))
865 User::Leave(KErrPermissionDenied);
868 group->QueueMessageL(pData.SendMessageToWindowGroup->uid, pData.SendMessageToWindowGroup->dataLength, *this);
871 case EWsClOpSendMessageToAllWindowGroups:
872 case EWsClOpSendMessageToAllWindowGroupsPriority:
874 if ((pData.SendMessageToWindowGroup->dataLength<0) || (pData.SendMessageToWindowGroup->dataLength>=(KMaxTInt/2)))
876 User::Leave(KErrArgument);
878 CWsWindowGroup::SendMessageToAllGroupsL(*this,aOpcode==EWsClOpSendMessageToAllWindowGroups,*pData.SendMessageToWindowGroup);
881 case EWsClOpFetchMessage:
882 CWsWindowGroup::WindowGroupFromIdentifierL(pData.FetchMessage->windowGroupIdentifier)->FetchMessageL();
884 case EWsClOpGetWindowGroupNameFromIdentifier:
885 ReplyGroupName(CWsWindowGroup::WindowGroupFromIdentifierL(pData.GetWindowGroupNameFromIdentifier->identifier)->GroupName(),pData.GetWindowGroupNameFromIdentifier->maxLength);
887 case EWsClOpFindWindowGroupIdentifier:
889 if (pData.FindWindowGroupIdentifier->length<0)
890 User::Leave(KErrArgument);
891 TPtrC ptr(BufferTPtr((TText *)(pData.FindWindowGroupIdentifier+1),pData.FindWindowGroupIdentifier->length));
892 SetReply(CWsWindowGroup::FindWindowGroupL(this, pData.FindWindowGroupIdentifier->identifier,
893 pData.FindWindowGroupIdentifier->offset,&ptr,NULL)->Identifier());
896 case EWsClOpFindWindowGroupIdentifierThread:
897 SetReply(CWsWindowGroup::FindWindowGroupL(this, pData.FindWindowGroupIdentifierThread->identifier,0,NULL,
898 &pData.FindWindowGroupIdentifierThread->threadId)->Identifier());
900 case EWsClOpSetBackgroundColor:
901 for(TInt i=0;i<CWsTop::NumberOfScreens();i++)
903 CWsTop::Screen(i)->RootWindow()->SetColor(*pData.rgb);
906 case EWsClOpGetBackgroundColor:
907 SetReply(iScreen->RootWindow()->BackColor().Internal());
909 case EWsClOpClaimSystemPointerCursorList:
911 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::ClaimSystemPointerCursorList API")))
913 User::Leave(KErrPermissionDenied);
915 ClaimSystemPointerCursorListL();
918 case EWsClOpFreeSystemPointerCursorList:
919 FreeSystemPointerCursorList();
921 case EWsClOpSetSystemPointerCursor:
922 CWsObject *pointercursor;
923 if ((pointercursor=HandleToObj(pData.SetSystemPointerCursor->handle, WS_HANDLE_POINTER_CURSOR))==NULL)
924 PPanic(EWservPanicSprite);
925 SetSystemPointerCursorL(pData.SetSystemPointerCursor->number, (CWsPointerCursor *)pointercursor);
927 case EWsClOpClearSystemPointerCursor:
928 ClearSystemPointerCursor(*pData.Int);
930 case EWsClOpSetPointerCursorArea:
932 if(KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetPointerCursorArea API")))
934 if (!iScreen->IsValidScreenSizeMode(*pData.Int))
935 PPanic(EWservPanicScreenModeNumber);
936 iScreen->SetPointerCursorArea(pData.SetPointerCursorArea->mode,pData.SetPointerCursorArea->area);
940 case EWsClOpPointerCursorArea:
941 if (!iScreen->IsValidScreenSizeMode(*pData.Int))
942 PPanic(EWservPanicScreenModeNumber);
943 ReplyRect(iScreen->GetPointerCursorArea(*pData.Int));
945 case EWsClOpSetPointerCursorMode:
947 CWsWindowGroup* focusWinGp=CWsTop::FocusWindowGroup();
948 if (focusWinGp && focusWinGp->WsOwner()==this)
950 WsPointer::SetPointerCursorMode(*pData.Mode);
951 WsPointer::UpdatePointerCursor();
955 case EWsClOpSetClientCursorMode :
957 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetPointerCursorModeIfFocused API")))
959 User::Leave(KErrPermissionDenied);
961 WsPointer::SetPointerCursorMode(*pData.Mode);
962 WsPointer::UpdatePointerCursor();
965 case EWsClOpPointerCursorMode:
966 SetReply(WsPointer::PointerCursorMode());
968 case EWsClOpSetDefaultSystemPointerCursor:
969 SetDefaultSystemPointerCursor(*pData.Int);
971 case EWsClOpClearDefaultSystemPointerCursor:
972 SetDefaultSystemPointerCursor(ENoDefaultSystemPointerCursor);
974 case EWsClOpSetPointerCursorPosition:
976 CWsWindowGroup* focusWinGp=CWsTop::FocusWindowGroup();
977 if ((!focusWinGp || focusWinGp->WsOwner()!=this)&&
978 (!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetPointerCursorPosition API"))))
980 User::Leave(KErrPermissionDenied);
982 WsPointer::SetPointerCursorPos(*pData.Point);
985 case EWsClOpPointerCursorPosition:
986 ReplyPoint(WsPointer::PointerCursorPos());
988 case EWsClOpSetModifierState:
990 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetModifierState API")))
992 User::Leave(KErrPermissionDenied);
994 TWindowServerEvent::SetModifierState(pData.SetModifierState->modifier,pData.SetModifierState->state);
997 case EWsClOpGetModifierState:
998 SetReply(TWindowServerEvent::GetModifierState());
1000 case EWsClOpHeapCount:
1001 SetReply(CWsMemoryManager::Static()->Count());
1003 case EWsClOpDebugInfo:
1004 DebugInfoL(pData.DebugInfo->iFunction,pData.DebugInfo->iParam,EFalse);
1006 case EWsClOpDebugInfoReplyBuf:
1007 DebugInfoL(pData.DebugInfo->iFunction,pData.DebugInfo->iParam,ETrue);
1009 case EWsClOpResourceCount:
1010 SetReply(iObjectIndex->Count());
1012 case EWsClOpHeapSetFail:
1013 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::HeapSetFail API")))
1015 PPanic(EWservPanicPermissionDenied);
1017 #if !defined(_DEBUG)
1018 if (pData.HeapSetFail->type!=RHeap::ENone)
1019 TWindowServerEvent::NotifyOom();
1021 User::__DbgSetAllocFail(RHeap::EUser,pData.HeapSetFail->type,pData.HeapSetFail->value);
1022 //__UHEAP_SETFAIL(pData.HeapSetFail->type,pData.HeapSetFail->value);
1024 case EWsClOpRawEvent:
1026 if(!KSecurityPolicy_SwEvent().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SimulateRawEvent API")))
1028 PPanic(EWservPanicPermissionDenied);
1030 TRawEvent event(*pData.RawEvent);
1031 if (WsPointer::PreProcessEvent(event))
1032 TWindowServerEvent::ProcessRawEvent(event);
1035 case EWsClOpKeyEvent:
1037 if(!KSecurityPolicy_SwEvent().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SimulateKeyEvent API")))
1039 PPanic(EWservPanicPermissionDenied);
1041 TWindowServerEvent::ProcessKeyEvent(*pData.KeyEvent,0);
1044 case EWsClOpLogMessage:
1047 if (CheckBuffer(*pData.Int, KLogMessageLength))
1048 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,BufferTPtr((TText *)(pData.Int+1),*pData.Int),0);
1051 case EWsClOpPasswordEntered:
1052 CWsPassword::PasswordEntered(this);
1054 case EWsClOpComputeMode:
1055 SetComputeMode(*pData.ComputeMode);
1057 case EWsClOpSendOffEventsToShell:
1059 if(!KSecurityPolicy_PowerMgmt().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::RequestOffEvents API")))
1061 User::Leave(KErrPermissionDenied);
1063 SetReply(CWsTop::SetSendOffEventsToShell(this,*pData.OffEventsToShell));
1066 case EWsClOpGetDefModeMaxNumColors:
1068 SDefModeMaxNumColors colors;
1069 TInt screenNumber = *pData.Int;
1070 if (screenNumber<KDummyScreenNumber || screenNumber>=CWsTop::NumberOfScreens())
1072 PPanic(EWservPanicScreenNumber);
1076 CScreen* screen = (screenNumber==KDummyScreenNumber)?iScreen:CWsTop::Screen(screenNumber);
1077 screen->MaxNumColors(colors.iColors,colors.iGrays);
1078 colors.iDisplayMode=screen->FirstDefaultDisplayMode();
1080 ReplyBuf(&colors,sizeof(colors));
1083 case EWsClOpGetColorModeList:
1085 TInt screenNumber = *pData.Int;
1086 if (screenNumber<KDummyScreenNumber || screenNumber>=CWsTop::NumberOfScreens())
1088 PPanic(EWservPanicScreenNumber);
1092 SetReply((screenNumber==KDummyScreenNumber) ? iScreen->ColorModesFlag() : CWsTop::Screen(screenNumber)->ColorModesFlag());
1096 case EWsClOpSetDefaultFadingParams:
1098 if(KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetDefaultFadingParameters API")))
1100 iScreen->SetFadingParams(WservEncoding::ExtractFirst8BitValue(*pData.UInt),WservEncoding::ExtractSecond8BitValue(*pData.UInt));
1104 case EWsClOpPrepareForSwitchOff:
1106 if(KSecurityPolicy_PowerMgmt().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::PrepareForSwitchOff API")))
1108 // Andy - this used to stop the heartbeat - should it now stop animations?
1112 case EWsClOpSetFaded:
1114 // Deprecated - retained for BC with applications that retrieve the fade count
1115 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetSystemFaded API")))
1117 User::Leave(KErrPermissionDenied);
1121 if (pData.SetSystemFaded->UseDefaultMap())
1123 iScreen->GetFadingParams(blackMap,whiteMap);
1127 pData.SetSystemFaded->GetFadingParams(blackMap,whiteMap);
1129 iScreen->RootWindow()->SetSystemFaded(pData.SetSystemFaded->Faded(),blackMap,whiteMap);
1130 if (CWsTop::IsFadeEnabled())
1132 Screen()->AcceptFadeRequest( iScreen->RootWindow(),
1133 pData.SetSystemFaded->Faded(),
1139 case EWsClOpLogCommand:
1140 CWsTop::LogCommand(*pData.LogCommand);
1142 #if defined(__WINS__)
1143 case EWsClOpRemoveKeyCode:
1144 iRemoveKeyCode=*pData.Bool;
1146 case EWsClOpSimulateXyInput:
1147 WsPointer::SetXyInputType(static_cast<TXYInputType>(*pData.XyInput));
1150 case EWsClOpNoFlickerFree:
1152 // should only be used by WServ test code
1153 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for EWsClOpNoFlickerFree message")))
1155 PPanic(EWservPanicPermissionDenied);
1157 iScreen->FreeOffScreenBitmap();
1160 case EWsClOpSetFocusScreen:
1162 if(!KSecurityPolicy_WriteDeviceData().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::SetFocusScreen API")))
1164 User::Leave(KErrPermissionDenied);
1166 TInt focusScreen=*pData.Int;
1167 if (focusScreen>=0 && focusScreen<CWsTop::NumberOfScreens())
1168 SetReply(CWsTop::SetCurrentFocusScreen(focusScreen));
1170 SessionPanic(EWservPanicScreenNumber);
1173 case EWsClOpGetFocusScreen:
1174 SetReply(CWsTop::CurrentFocusScreen()->ScreenNumber());
1176 case EWsClOpGetNumberOfScreens:
1177 SetReply(CWsTop::NumberOfScreens());
1179 case EWsClOpClearAllRedrawStores:
1180 CWsTop::ClearAllRedrawStores();
1182 case EWsClOpGetGraphicMessage:
1183 iGraphicMessageQueue.GetGraphicMessage();
1185 case EWsClOpGraphicMessageCancel:
1186 iGraphicMessageQueue.CancelRead();
1188 case EWsClOpGraphicAbortMessage:
1189 iGraphicMessageQueue.AbortMessage(*pData.Int);
1191 case EWsClOpGraphicFetchHeaderMessage:
1192 SetReply(iGraphicMessageQueue.TopClientHandle());
1195 PPanic(EWservPanicOpcode);
1200 void CWsClient::DebugInfoL(TInt aFunction, TInt aParam, TBool aHasReplyBuf) const
1204 case EWsDebugInfoHeap:
1207 TWsDebugHeapInfo heapInfo;
1208 RHeap& heap=User::Heap();
1209 heapInfo.iCount=heap.AllocSize(heapInfo.iTotal);
1210 heapInfo.iAvailable=heap.Available(heapInfo.iLargestAvailable);
1211 ReplyBuf(&heapInfo,sizeof(heapInfo));
1213 SetReply(KErrArgument);
1215 case EWsDebugSetCheckHeapOnDisconnectClient:
1216 if (!KSecurityPolicy_PowerMgmt().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::DebugInfoL API")))
1218 User::Leave(KErrPermissionDenied);
1220 CWsTop::SetCheckHeapOnDisconnectClient(this);
1222 case EWsDebugSetCheckHeapOnDisconnectMode:
1223 if (!KSecurityPolicy_PowerMgmt().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWsSession::DebugInfoL API")))
1225 User::Leave(KErrPermissionDenied);
1227 CWsTop::SetCheckHeapOnDisconnectMode(STATIC_CAST(TWsCheckHeapOnDisconnectMode,aParam));
1229 case EWsDebugFetchCheckHeapResult:
1230 SetReply(CWsTop::FetchCheckHeapResult());
1233 SetReply(KErrNotSupported);
1238 void CWsClient::ReplyGroupName(HBufC *aName, TInt aMaxLength) const
1242 if (aName->Length()>aMaxLength)
1244 ReplyBuf(aName->Left(aMaxLength));
1245 SetReply(KErrOverflow);
1251 ReplyBuf(nullDescriptor);
1254 void CWsClient::TriggerRedraw()
1256 RedrawQueue()->TriggerRedraw();
1259 void CWsClient::UpdateWindowOrdinalPrioritys()
1261 for(CWsWindowGroup *win=iScreen->RootWindow()->Child();win;win=win->NextSibling())
1262 if (win->WsOwner()==this)
1263 win->UpdateOrdinalPriority(ETrue);
1266 void CWsClient::DeleteSystemPointerListEntry(TInt aIndex)
1268 PointerCursor (aIndex)->Close();
1269 iSystemPointerCursors->Delete(aIndex);
1272 CWsPointerCursor *CWsClient::SystemPointerCursor(TInt aIndex)
1275 if (iSystemPointerCursors)
1277 if (FindCursorArrayItem(iSystemPointerCursors, aIndex, arrayIndex))
1279 return PointerCursor (arrayIndex);
1281 // Cursor not defined so try for default cursor
1282 if (FindCursorArrayItem(iSystemPointerCursors, 0, arrayIndex))
1284 return PointerCursor (arrayIndex);
1287 // If that fails simply return NULL for no cursor
1291 void CWsClient::SetSystemPointerCursorL(TInt aIndex, CWsPointerCursor *aCursor)
1293 if (iSystemPointerCursorListOwner!=this)
1294 PPanic(EWservPanicNotSystemPointerCursorListOwner);
1296 if (FindCursorArrayItem(iSystemPointerCursors, aIndex, arrayIndex))
1298 PointerCursor (arrayIndex)->Close();
1299 PointerCursor (arrayIndex) = aCursor;
1303 TWsCursorArrayItem entry;
1304 entry.iIndex=aIndex;
1305 entry.iCursor=aCursor;
1306 iSystemPointerCursors->InsertIsqL(entry, CursorKey);
1309 if (aIndex==iDefaultSystemPointerCursorIndex)
1310 iDefaultSystemPointerCursor=aCursor;
1311 WsPointer::UpdatePointerCursor();
1314 void CWsClient::ClearSystemPointerCursor(TInt aIndex)
1316 if (iSystemPointerCursorListOwner!=this)
1317 PPanic(EWservPanicNotSystemPointerCursorListOwner);
1319 if (FindCursorArrayItem(iSystemPointerCursors, aIndex, arrayIndex))
1321 DeleteSystemPointerListEntry(arrayIndex);
1322 if (aIndex==iDefaultSystemPointerCursorIndex)
1323 iDefaultSystemPointerCursor=NULL;
1327 void CWsClient::ClaimSystemPointerCursorListL()
1329 if (iSystemPointerCursorListOwner)
1330 User::Leave(KErrInUse);
1331 const TInt systemPointerCursorGranularity = 4;
1332 iSystemPointerCursors = new(ELeave) CArrayFixFlat<TWsCursorArrayItem> (systemPointerCursorGranularity);
1333 iSystemPointerCursorListOwner=this;
1336 void CWsClient::FreeSystemPointerCursorList()
1338 if (iSystemPointerCursorListOwner==this)
1340 iSystemPointerCursorListOwner=NULL;
1341 while(iSystemPointerCursors->Count()>0)
1342 DeleteSystemPointerListEntry(0);
1343 iDefaultSystemPointerCursor=NULL;
1344 iDefaultSystemPointerCursorIndex=0;
1345 delete iSystemPointerCursors;
1346 iSystemPointerCursors=NULL;
1350 void CWsClient::SetDefaultSystemPointerCursor(TInt aIndex)
1353 if (iSystemPointerCursorListOwner!=this)
1354 PPanic(EWservPanicNotSystemPointerCursorListOwner);
1355 iDefaultSystemPointerCursorIndex=aIndex;
1356 iDefaultSystemPointerCursor=NULL;
1357 if (aIndex != ENoDefaultSystemPointerCursor &&
1358 FindCursorArrayItem(iSystemPointerCursors, aIndex, arrayIndex))
1359 iDefaultSystemPointerCursor = PointerCursor (arrayIndex);
1361 iDefaultSystemPointerCursor=NULL;
1362 WsPointer::UpdatePointerCursor();
1365 TBool CWsClient::FindCursorArrayItem(CArrayFixFlat<TWsCursorArrayItem>* aCursorArray,
1366 TInt aIndex,TInt& aPosition)
1370 return EFalse; // No hit if the array isn't even allocated
1372 TWsCursorArrayItem entry;
1373 entry.iIndex=aIndex;
1374 return aCursorArray->FindIsq(entry, CursorKey, aPosition)==KErrNone;
1377 void CWsClient::SetClientPriority()
1379 if (iComputeMode!=RWsSession::EPriorityControlDisabled)
1381 Client().SetProcessPriority(
1382 iComputeMode==RWsSession::EPriorityControlComputeOn || CWsTop::FocusWindowGroupOwner()!=this ?
1383 EPriorityBackground :
1384 EPriorityForeground);
1388 void CWsClient::SetComputeMode(RWsSession::TComputeMode aComputeMode)
1390 if (aComputeMode!=RWsSession::EPriorityControlDisabled &&
1391 aComputeMode!=RWsSession::EPriorityControlComputeOn &&
1392 aComputeMode!=RWsSession::EPriorityControlComputeOff)
1393 PPanic(EWservPanicSetComputeMode);
1394 iComputeMode=aComputeMode;
1395 SetClientPriority();
1398 void CWsClient::CompleteMessage(const RMessage2& aMessage,TInt aReason)
1400 if (!aMessage.IsNull())
1402 if (iInternalFlags&EPanicClientAsSoonAsPossible)
1404 aMessage.Panic(KWSERVSessionPanicCategory,iPanicReason);
1405 iInternalFlags&=~EPanicClientAsSoonAsPossible;
1409 aMessage.Complete(aReason);
1414 void CWsClient::ServiceError(const RMessage2& /*aMessage*/,TInt aError)
1416 CompleteMessage(iClientMessage,aError);
1419 void CWsClient::ServiceL(const RMessage2 &aMessage)
1421 // Handle messages for the window server server.
1424 iClientMessage=aMessage; // from now on use always the message stored in the session
1425 if (iInternalFlags&EPanicClientAsSoonAsPossible)
1427 iClientMessage.Panic(KWSERVSessionPanicCategory,iPanicReason);
1431 iPanicReason=KErrNone;
1433 TBool completeRequest=ETrue;
1434 DoServiceL(iClientMessage, completeRequest);
1435 if (completeRequest)
1437 if(!iResponseHandle)
1439 CompleteMessage(iClientMessage,iReply);
1443 // Prior to 9.0 RMessagePtr2.Complete doesn't have the option of sending a handle back
1444 // However, since the security is lower simply sending the handle id is ok
1445 iClientMessage.Complete(*iResponseHandle);
1446 iResponseHandle = NULL;
1452 void CWsClient::SetResponseHandle(RHandleBase* aHandle)
1454 iResponseHandle = aHandle;
1457 void CWsClient::DoServiceL(const RMessage2& aMessage, TBool& aCompleteRequest)
1459 const TInt function=aMessage.Function();
1462 case EWservMessInit:
1463 StartInitializationL(iConnectionId++);
1465 case EWservMessSyncMsgBuf:
1466 case EWservMessCommandBuffer:
1468 if (!IsInitialised())
1470 PPanic(EWservPanicUninitialisedClient);
1473 const TInt err=aMessage.Read(KBufferMessageSlot,iCmdBuf);
1476 TRAPD(error,CommandBufL());
1477 iCurrentClient=NULL;
1479 if (!iLastCommand && err!=KErrNone)
1481 SessionPanic(EWservPanicFunctionLeave);
1487 else if (err!=KErrDied)
1489 PPanic(EWservPanicDescriptor);
1491 if(function == EWservMessCommandBuffer && CWsTop::FinishEveryFlush())
1493 if(Screen()->IsUpdatePending())
1494 Screen()->DoRedrawNow();
1497 aCompleteRequest=(function!=EWservMessAsynchronousService);
1500 case EWservMessShutdown:
1502 if(!KSecurityPolicy_PowerMgmt().CheckPolicy(ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for EWservMessShutdown message ")))
1504 PPanic(EWservPanicPermissionDenied);
1506 if (aMessage.Int0()==EWservShutdownCheck)
1510 PPanic(EWservPanicHandle);
1514 case EWservMessFinish:
1516 if(Screen()->IsUpdatePending())
1517 Screen()->DoRedrawNow();
1521 if (function&EWservMessAsynchronousService)
1523 TRAPD(err,CommandL((function&~EWservMessAsynchronousService), aMessage));
1524 aCompleteRequest=(err!=KErrNone);
1525 WS_ASSERT_DEBUG((!(iInternalFlags&EPanicClientAsSoonAsPossible))==(!aCompleteRequest), EWsPanicPanicFlagError);
1527 else if (function&EWservMessAnimDllAsyncCommand)
1529 CWsAnimDll* const animDll=STATIC_CAST(CWsAnimDll*,HandleToObj(aMessage.Int0(), WS_HANDLE_ANIM_DLL));
1532 SessionPanic(EWservPanicHandle);
1535 // it looks wrong to call CommandReply here (not AsyncCommandReply, or something, which doesn't exist), but basically, we can't add a virtual AsyncCommandReplyL to CAnim to correspond to RAnim::AsyncCommandReply as that would break binary and source compatibility for Anim plug-ins; instead asynchronous commands are done by the plug-in having to complete the RMessagePtr2 returned by iFunctions->Message() (or a copy of of that object) for asynchronous commands only
1536 TRAPD(err,animDll->AnimObjectL(aMessage.Int1())->CommandReply(function&~EWservMessAnimDllAsyncCommand,NULL));
1537 aCompleteRequest=(err!=KErrNone);
1538 WS_ASSERT_DEBUG((!(iInternalFlags&EPanicClientAsSoonAsPossible))==(!aCompleteRequest), EWsPanicPanicFlagError);
1542 SetReply(KErrNotSupported);
1548 void CWsClient::RemoteRead(TDes16& aDes, TInt aOffset)
1550 if (iClientMessage.Read(KRemoteBufferMessageSlot,aDes,aOffset)!=KErrNone)
1552 SessionPanic(EWservPanicDescriptor);
1556 void CWsClient::RemoteRead(TDes8& aDes, TInt aOffset)
1558 if (iClientMessage.Read(KRemoteBufferMessageSlot,aDes,aOffset)!=KErrNone)
1560 SessionPanic(EWservPanicDescriptor);
1564 void CWsClient::RemoteReadL(TDes16& aDes, TInt aOffset)
1566 iClientMessage.ReadL(KRemoteBufferMessageSlot,aDes,aOffset);
1569 void CWsClient::RemoteReadL(TDes8& aDes, TInt aOffset)
1571 iClientMessage.ReadL(KRemoteBufferMessageSlot,aDes,aOffset);
1574 void CWsClient::DeleteStatics()
1576 if (iTextCursorArray)
1578 const TInt count = iTextCursorArray->Count();
1579 for (TInt index=0;index<count;index++)
1581 delete iTextCursorArray->At(index).iCursor;
1583 delete iTextCursorArray;
1584 iTextCursorArray = NULL;
1588 // CWsClient implementing MWsClient \\\\\\\\\\\\\\\\\\\\\\\\
1590 TBool CWsClient::HasCapability(TCapability aCapability) const
1592 return iClient.HasCapability(aCapability);
1595 TSecureId CWsClient::SecureId() const
1597 return iClient.SecureId();
1600 TVendorId CWsClient::VendorId() const
1602 return iClient.VendorId();
1605 Makes a new copy of the aData. so it could be deleted after this call.
1607 TInt CWsClient::SendMessage(const CWsGraphicDrawer* aOnBehalfOf,const TDesC8& aData)
1609 CWsGraphicMessageQueue::CMessage* msg = CWsGraphicMessageQueue::CMessage::New(aData);
1612 return SendMessage(aOnBehalfOf,*msg);
1617 TInt CWsClient::SendMessage(const CWsGraphicDrawer* aOnBehalfOf,CWsMessageData& aData)
1619 WS_ASSERT_DEBUG(aData.Data().Size(), EWsPanicWsGraphic);
1620 const CWsGraphicDrawerObject* obj = DrawerObject(aOnBehalfOf);
1621 WS_ASSERT_DEBUG(obj, EWsPanicWsGraphic);
1624 // assign message id
1625 if(iMessageIdSeq == KMaxTInt) // Wrap if iMessageIdSeq has reached KMaxTInt
1628 // correct other handles
1629 aData.iClientHandle = (obj->ClientHandle() | (EWsGraphMessageTypeUser & 0x03));
1630 aData.iDrawer = obj->Drawer();
1631 aData.iId = iMessageIdSeq;
1632 iGraphicMessageQueue.Queue(&aData);
1633 return iMessageIdSeq;
1638 /** adds a message to the message queue
1639 @return a postive number to uniquely identify the message
1643 CWsGraphicDrawerObject* CWsClient::DrawerObject(const CWsGraphicDrawer* aDrawer)
1645 const TInt count = ObjectIndex()->Length();
1646 for(TInt i=0; i<count; i++)
1648 CWsObject* obj = const_cast<CWsObject*>(ObjectIndex()->At(i));
1649 if(obj && (WS_HANDLE_GRAPHIC_DRAWER == obj->Type()))
1651 CWsGraphicDrawerObject* candidate = static_cast<CWsGraphicDrawerObject*>(obj);
1652 if(candidate->Drawer() == aDrawer)
1661 const CWsGraphicDrawerObject* CWsClient::DrawerObject(const CWsGraphicDrawer* aDrawer) const
1663 CWsObjectIx* objectIndex = const_cast<CWsClient*>(this)->ObjectIndex();
1664 const TInt count = objectIndex->Length();
1665 for(TInt i=0; i<count; i++)
1667 CWsObject* obj = const_cast<CWsObject*>(objectIndex->At(i));
1668 if(obj && (WS_HANDLE_GRAPHIC_DRAWER == obj->Type()))
1670 const CWsGraphicDrawerObject* candidate = static_cast<CWsGraphicDrawerObject*>(obj);
1671 if(candidate->Drawer() == aDrawer)
1682 CWsCliObj* CWsCliObj::NewL(CWsClient *aOwner)
1684 CWsCliObj* self = new(ELeave) CWsCliObj(aOwner);
1685 CleanupStack::PushL(self);
1687 CleanupStack::Pop(self);
1691 CWsCliObj::CWsCliObj(CWsClient *aOwner) : CWsObject(aOwner,WS_HANDLE_CLIENT)
1693 __DECLARE_NAME(_S("CWsCliObj"));
1696 void CWsCliObj::ConstructL()
1701 void CWsCliObj::CommandL(TInt aOpcode, const TAny *aCmdData)
1703 iWsOwner->CommandL(aOpcode,aCmdData);
1706 CWsObject* CWsClient::HandleToObjUntyped(TInt aHandle)
1708 return(iObjectIndex->HandleToObject(aHandle));
1711 CWsObject* CWsClient::HandleToObj(TInt aHandle, WH_HANDLES aType)
1713 CWsObject* object = HandleToObjUntyped(aHandle);
1714 if (object && object->Type() == aType)