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 // Window server event queue handling
24 #include "advancedpointereventhelper.h"
25 #include "debughelper.h"
27 GLREF_D CDebugLogBase* wsDebugLog;
30 #define __CHECK_QUEUE() CheckQueue()
31 #define __ZAP_EVENTS(pointer,len) ZapEvents(pointer,len)
32 #define __ZAP_EVENT(pointer) ZapEvent(pointer)
34 #define __CHECK_QUEUE()
35 #define __ZAP_EVENTS(pointer,len)
36 #define __ZAP_EVENT(pointer)
39 const TInt ECopyBufSize=4;
41 TDblQue<CEventQueue> CEventQueue::iQueueList(_FOFF(CEventQueue,iLink));
42 TWsEvent* CEventQueue::iGlobalEventQueue=NULL;
43 TInt CEventQueue::iGlobalEventQueueSize=0;
44 TInt CEventQueue::iNumConnections=0;
45 RMutex CEventQueue::iMutex;
46 TWsEvent CEventQueue::iNullEvent;
50 CEventBase::~CEventBase()
52 if (!iEventMsg.IsNull() && CWsTop::ShuttingDown())
54 iEventMsg.Complete(KErrServerTerminated);
58 CEventBase::CEventBase(CWsClient* aOwner) : iWsOwner(aOwner)
62 void CEventBase::SignalEvent(TInt aCode)
65 wsDebugLog->SignalEvent(iWsOwner->ConnectionHandle());
66 iEventMsg.Complete(aCode);
67 iEventSignalledState|=EEventFlagSignalled;
70 inline TBool CEventBase::IsEventCancelled()
72 return (iEventSignalledState & EEventFlagCancelled);
76 If there is an outstanding read cancel it.
78 void CEventBase::CancelRead()
80 if (!iEventMsg.IsNull())
82 iEventMsg.Complete(KErrCancel);
84 iEventSignalledState|=EEventFlagCancelled;
87 void CEventBase::GetData(TAny* aData, TInt aLen)
89 if (!(iEventSignalledState&EEventFlagSignalled))
91 iWsOwner->PPanic(EWservPanicUnsignalledEventData);
93 iEventSignalledState&=~EEventFlagSignalled;
94 CWsClient::ReplyBuf(aData,aLen);
98 Queue a read of an event notification
100 void CEventBase::EventReadyCheck()
102 if ((iEventSignalledState&EEventFlagSignalled && !(iEventSignalledState&EEventFlagCancelled)) || !iEventMsg.IsNull())
104 #ifdef LOG_WSERV_EVENTS
105 RDebug::Printf("_WSEVENT: CEventBase::EventReadyCheck, Event is not ready");
107 iWsOwner->PPanic(EWservPanicReadOutstanding);
111 void CEventBase::EventReady(const RMessagePtr2& aEventMsg)
117 // CEventQueue - Queue organisation functions
120 void CEventQueue::CheckQueue()
122 TDblQueIter<CEventQueue> iter(iQueueList);
125 while((qptr=iter++)!=NULL)
127 WS_ASSERT_DEBUG(qptr->iQueueSize>=0 && qptr->iQueueSize<=iGlobalEventQueueSize, EWsPanicCheckEventQueue);
128 WS_ASSERT_DEBUG(qptr->iQueueSize==0 || qptr->iHead<qptr->iQueueSize, EWsPanicCheckEventQueue);
129 WS_ASSERT_DEBUG(qptr->iCount<=qptr->iQueueSize, EWsPanicCheckEventQueue);
130 WS_ASSERT_DEBUG(qptr->iEventPtr>=iGlobalEventQueue && qptr->iEventPtr<=(iGlobalEventQueue+iGlobalEventQueueSize), EWsPanicCheckEventQueue);
132 for(TInt index=0;index<iCount;index++)
134 TWsEvent* ev=EventPtr(index);
135 WS_ASSERT_DEBUG(ev->Type()!=EEventMarkInvalid, EWsPanicCheckEventQueue);
139 void CEventQueue::ZapEvent(TWsEvent* aTarget)
141 aTarget->SetType(EEventMarkInvalid);
142 aTarget->SetHandle(555);
145 void CEventQueue::ZapEvents(TWsEvent* aTarget, TInt aLen)
147 for(TInt index=0;index<aLen;index++)
148 ZapEvent(aTarget+index);
153 void CEventQueue::EventCopy(TWsEvent* aTarget, TWsEvent* aSource, TInt aNumEvents)
155 WS_ASSERT_DEBUG(aTarget>=iGlobalEventQueue, EWsPanicEventQueueCopy);
156 WS_ASSERT_DEBUG((aTarget+aNumEvents)<=(iGlobalEventQueue+iGlobalEventQueueSize), EWsPanicEventQueueCopy);
157 Mem::Copy(aTarget,aSource,aNumEvents*sizeof(TWsEvent));
160 TInt CEventQueue::RequiredQueueSize(TInt aNumConnections)
162 return(aNumConnections*EMinQueueSize+EMaxQueueSize+EExtraQueueSize);
166 Adjust the Global Queue Size.
167 @param aNewSize the size for event queue.
169 void CEventQueue::AdjustQueueSizeL(TInt aNewSize)
171 TWsEvent* oldQ=iGlobalEventQueue;
172 if (aNewSize < iGlobalEventQueueSize)
174 // Re-alloc wont move the cell down to a lower address, this means once this cell is
175 // high up in memory it may never move down again, thus wasting loads of global heap space.
176 const CEventQueue* last=iQueueList.Last();
177 if ((last->iEventPtr + last->iQueueSize) >= (iGlobalEventQueue + aNewSize))
181 TInt allocSize = aNewSize * sizeof(TWsEvent);
182 iGlobalEventQueue=static_cast<TWsEvent*>(User::AllocL(allocSize));
183 Mem::Copy(iGlobalEventQueue,oldQ,allocSize);
188 iGlobalEventQueue = static_cast<TWsEvent*>(User::ReAllocL(iGlobalEventQueue, aNewSize * sizeof(TWsEvent)));
190 __ZAP_EVENTS(iGlobalEventQueue + iGlobalEventQueueSize, aNewSize - iGlobalEventQueueSize);
191 iGlobalEventQueueSize = aNewSize;
192 // coverity[use_after_free]
193 TInt diff=(reinterpret_cast<TInt8*>(iGlobalEventQueue)-reinterpret_cast<TInt8*>(oldQ));
196 TDblQueIter<CEventQueue> iter(iQueueList);
198 while((qptr=iter++)!=NULL)
200 qptr->iEventPtr=reinterpret_cast<TWsEvent*>(reinterpret_cast<TInt8*>(qptr->iEventPtr)+diff);
205 void CEventQueue::AddQueueL()
208 if ((iNumConnections%EQueueGranularity)==0)
210 const TInt newSize = RequiredQueueSize(iNumConnections + EQueueGranularity);
211 TRAPD(err,AdjustQueueSizeL(newSize));
220 if (iQueueList.IsEmpty())
221 iEventPtr=iGlobalEventQueue;
224 CEventQueue* qptr=iQueueList.Last();
225 iEventPtr=qptr->iEventPtr+qptr->iQueueSize;
227 iQueueList.AddLast(*this);
229 //Initialize the queue size to at least EMinQueueSize
230 //Check the queue by doing standard queue compression.
231 TBool isExpanded = ETrue;
232 do {isExpanded = Expand(EEventPriorityLow);} while ((iQueueSize < EMinQueueSize) && isExpanded);
233 while (iQueueSize < EMinQueueSize)
235 //Cannot get enough spaces by doing standard queue compression,
236 //try to grow the global queue.
237 TInt sizeRequired = EMinQueueSize - iQueueSize;
238 const TInt newSize = iGlobalEventQueueSize + sizeRequired;
239 TRAPD(err, AdjustQueueSizeL(newSize));
242 //Cannot get enough spaces by growing global queue.
243 //try to purge the oldest events from inactive clients.
244 TInt numEventCleared = PurgeInactiveEvents(sizeRequired);
245 if (numEventCleared == 0)
252 while (doExpand(ECompressNoPurge)) {};
258 void CEventQueue::RemoveQueue()
261 if (iEventPtr) // If this is still NULL this class hasn't been linked into the Q list
263 __ZAP_EVENTS(iEventPtr, iQueueSize);
265 if (--iNumConnections==0)
267 User::Free(iGlobalEventQueue);
268 iGlobalEventQueue=NULL;
269 iGlobalEventQueueSize=0;
271 else if ((iNumConnections%EQueueGranularity)==0)
273 TDblQueIter<CEventQueue> iter(iQueueList);
274 CEventQueue* qptr=iter++;
275 qptr->Compress(ECompressNoPurge);
276 while((qptr=iter++)!=NULL)
280 const TInt newSize = RequiredQueueSize(iNumConnections);
281 TRAP_IGNORE(AdjustQueueSizeL(newSize));
282 // Can easily leave as we need to allocate the new smaller queue before freeing
283 // the old queue. But if it does it doesn't matter as we'll simply be left with
284 // a larger than necessary buffer until the next realloc
292 void CEventQueue::IncreaseQueueSize(TInt aNumSpaces)
294 if ((iQueueSize+aNumSpaces)>EMaxQueueSize)
295 aNumSpaces=EMaxQueueSize-iQueueSize;
296 EventCopy(iEventPtr+iHead+aNumSpaces, iEventPtr+iHead, iQueueSize-iHead);
297 __ZAP_EVENTS(iEventPtr+iHead, aNumSpaces);
298 iQueueSize+=aNumSpaces;
299 iHead=(iHead+aNumSpaces)%iQueueSize;
302 TBool CEventQueue::Expand(TWservEventPriorities aPriority)
304 if (iQueueSize==EMaxQueueSize)
306 if (aPriority==EEventPriorityHigh)
309 if (iCount<iQueueSize)
310 return(ETrue); // Success!
313 else if (doExpand(ECompressNoPurge) ||
314 doExpand(ECompressPurge1) ||
315 (aPriority==EEventPriorityHigh && doExpand(ECompressPurge2)))
317 return(EFalse); // Failure
320 TBool CEventQueue::doExpand(TCompressMode aCompressMode)
322 TDblQueIter<CEventQueue> iter(iQueueList);
324 CEventQueue* qptr=NULL;
327 // while loop from last queue till current queue, moving all queues up
328 // to get all the space from between them
329 while((qptr=iter)!=this)
331 spare=qptr->SqueezeUp();
333 qptr->Compress(aCompressMode);
337 // current queue, if we have space then expand the same and return
338 spare=FollowingGap();
341 IncreaseQueueSize(spare);
346 // while loop from current queue till first queue, looking for a gap
347 // between queues and using the first one it finds
350 while((qptr=iter)!=NULL)
352 expanded=qptr->SqueezeUp();
355 return MoveDownAndExpand(iter, expanded);
358 qptr->Compress(aCompressMode);
362 // Even by doing all the above if we did not find space then check if first
363 // queue has some space before it. If so then make use of it and movedown all the
364 // queue till the current queue and then use the space for expansion
367 if (qptr->iEventPtr>iGlobalEventQueue)
369 return MoveDownAndExpand(iter, qptr->iEventPtr-iGlobalEventQueue);
373 return(EFalse); // Failed to expand enough room
376 // This function moves the queue down by given amount and repeats this until
377 // the current queue and then expand the same
378 TBool CEventQueue::MoveDownAndExpand(TDblQueIter<CEventQueue> &aIter, TInt aExpand)
380 CEventQueue* qptr=NULL;
384 qptr->MoveDown(aExpand);
387 IncreaseQueueSize(aExpand);
395 void CEventQueue::MoveDown(TInt aMove)
401 EventCopy(iEventPtr-aMove,iEventPtr,iQueueSize);
405 /*void CEventQueue::LogUpDownEvents(TChar aChar)
416 for (index=0;index<iCount;index++)
418 event=EventPtr(index);
419 if (event->Type()==EEventPointer)
421 if (event->Pointer()->iType==TPointerEvent::EButton1Down
422 || event->Pointer()->iType==TPointerEvent::EButton1Up)
425 if (event->Pointer()->iType==TPointerEvent::EButton1Down)
429 buf.AppendNum(index);
434 wsDebugLog->MiscMessage(ELogImportant,buf);
437 inline void CEventQueue::IncEventPointer(TWsEvent*& aEventPtr)
439 // iEventPtr[iQueueSize] is the element beyond the array, used for efficient bounds checking of the circular buffer only, do not access!!
440 if(++aEventPtr==&iEventPtr[iQueueSize])
446 inline void CEventQueue::DecEventPointer(TWsEvent*& aEventPtr)
448 if(aEventPtr--==iEventPtr)
450 aEventPtr=&iEventPtr[iQueueSize - 1];
455 Starting from aEventToPurge searches the queue for matching event by iterating towards end of queue.
456 Matching event will be a pointer event with the same window handle and pointer number
457 as aEventToPurge, but will have type aMatchingType. If matching event is found, it will
458 be removed from the queue and search will finish.
460 Search will be stopped if an event of type aSearchTerminator is found with the same pointer number
463 If search is not stopped by aSearchTerminator and matching event is not found, TWsPointer
464 class is notified that matching event has not been removed, so if it arrives in the future,
465 TWsPointer class will be able to ignore it.
467 void CEventQueue::PurgeEventPairs(TWsEvent* aEventToPurge, TPointerEvent::TType aMatchingType, TPointerEvent::TType aSearchTerminator)
469 TWsEvent* eventToMatch = aEventToPurge;
470 TWsEvent* lastEvent = EventPtr(iCount);
471 for(IncEventPointer(eventToMatch);eventToMatch!=lastEvent;IncEventPointer(eventToMatch))
473 if ( (eventToMatch->Type()==EEventPointer) // Must be checked first to ensure it is pointer data checked later
474 && (TAdvancedPointerEventHelper::PointerNumber(*eventToMatch) == TAdvancedPointerEventHelper::PointerNumber(*aEventToPurge))) // same pointer
476 if ((eventToMatch->Pointer()->iType==aMatchingType) // correct event type
477 && (eventToMatch->Handle()==aEventToPurge->Handle())) // same window
479 *eventToMatch=iNullEvent;
482 else if (eventToMatch->Pointer()->iType==aSearchTerminator) // stop searching for mathing type
488 TWsPointer::UnmatchedEventPurged(aMatchingType, aEventToPurge);
492 Starting from pointer event aBasePointerEvent searches the queue for next pointer event for
493 the same pointer by iterating towards end of queue.
495 @param aBasePointerEvent must be of type EEventPointer
496 @return Next pointer event in the queue for the same pointer as aBasePointerEvent
497 if it has the same handle as aBasePointerEvent or NULL if it has different handle.
498 NULL if there is no next pointer event for the same pointer in the queue.
500 TWsEvent* CEventQueue::NextPointerEvent(TWsEvent* aBasePointerEvent)
502 WS_ASSERT_DEBUG(aBasePointerEvent->Type() == EEventPointer, EWsPanicCheckEventQueue);
503 TWsEvent* currentEvent = aBasePointerEvent;
504 TWsEvent* lastEvent = EventPtr(iCount);
505 TUint8 pointerNumber = TAdvancedPointerEventHelper::PointerNumber(*aBasePointerEvent);
506 TUint handle = aBasePointerEvent->Handle();
507 for(IncEventPointer(currentEvent);currentEvent!=lastEvent;IncEventPointer(currentEvent))
509 if ((currentEvent->Type() == EEventPointer) &&
510 (TAdvancedPointerEventHelper::PointerNumber(*currentEvent) == pointerNumber))
512 if (currentEvent->Handle() == handle)
526 Checks if aEventToPurge should be purged. If it can be purged, then events that should
527 be purged together with aEventToPurge (matching events) are overwritten with iNullEvent.
529 @return ETrue if aEventToPurge should be purged, EFalse otherwise.
531 TBool CEventQueue::CheckPurgePointerEvent(TWsEvent* aEventToPurge)
533 switch(aEventToPurge->Pointer()->iType)
535 case TPointerEvent::EDrag:
536 case TPointerEvent::EMove:
537 case TPointerEvent::EButtonRepeat:
538 case TPointerEvent::ESwitchOn:
540 case TPointerEvent::EButton1Down:
541 PurgeEventPairs(aEventToPurge,TPointerEvent::EButton1Up, TPointerEvent::ENullType);
543 case TPointerEvent::EButton2Down:
544 PurgeEventPairs(aEventToPurge,TPointerEvent::EButton2Up, TPointerEvent::ENullType);
546 case TPointerEvent::EButton3Down:
547 PurgeEventPairs(aEventToPurge,TPointerEvent::EButton3Up, TPointerEvent::ENullType);
549 case TPointerEvent::EEnterHighPressure:
550 PurgeEventPairs(aEventToPurge,TPointerEvent::EExitHighPressure, TPointerEvent::EButton1Up);
552 case TPointerEvent::EEnterCloseProximity:
554 TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
555 if (nextEvent != NULL)
557 switch(nextEvent->Pointer()->iType)
559 case TPointerEvent::EExitCloseProximity:
560 *nextEvent = iNullEvent;
561 case TPointerEvent::EOutOfRange:
567 case TPointerEvent::EExitCloseProximity:
569 TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
570 if (nextEvent != NULL)
572 switch(nextEvent->Pointer()->iType)
574 case TPointerEvent::EEnterCloseProximity:
575 *nextEvent = iNullEvent;
576 case TPointerEvent::EOutOfRange:
582 case TPointerEvent::EOutOfRange:
584 TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
585 if ((nextEvent != NULL) &&
586 (nextEvent->Pointer()->iType == TPointerEvent::EOutOfRange))
592 case TPointerEvent::EExitHighPressure:
594 TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
595 if ((nextEvent != NULL) &&
596 (nextEvent->Pointer()->iType == TPointerEvent::EButton1Up))
602 case TPointerEvent::EButton1Up:
603 case TPointerEvent::EButton2Up:
604 case TPointerEvent::EButton3Up:
610 /** Purgable events are:
611 Pointer Up/Down pairs belonging to the same pointer, button and window
612 Pointer moves & drags
615 Key Ups if not foreground connection
616 Focus lost/gained pairs
617 EEnterHighPressure/EExitHighPressure pairs belonging to the same pointer
618 EEnterCloseProximity/EExitCloseProximity pairs belonging to the same pointer
619 Lone EEnterHighPressure, EExitHighPressure if followed by Up for the same pointer
620 Lone EEnterCloseProximity, EExitCloseProximity if followed by EOutOfRange for the same pointer
621 EOutOfRange if followed by another EOutOfRange for the same pointer
623 Events that must not be purged:
624 Key ups for foreground connections queue
625 Lone pointer ups, must be delivered to match preceeding pointer down
626 Lone EExitHighPressure if not followed by Up, must be delivered to match preceeding EEnterHighPressure
627 Lone EEnterCloseProximity, EExitCloseProximity not followed by EOutOfRange for the same pointer
628 Lone focus lost/gained messages
630 void CEventQueue::Purge()
632 TWsEvent* eventToPurge;
633 TWsEvent* eventToMatch;
635 TInt indexToPurge=iCount;
636 while(indexToPurge>0)
638 eventToPurge=EventPtr(--indexToPurge);
639 switch(eventToPurge->Type())
643 case EEventMarkInvalid:
645 WS_PANIC_DEBUG(EWsPanicCheckEventQueue);
649 case EEventPointerEnter:
650 case EEventPointerExit:
652 breakLoopAndRemoveEvent:
653 #ifdef LOG_WSERV_EVENTS
654 RDebug::Print(_L("_WSEVENT: CEventQueue::Purge(), The event to be purged is %S"), &WsEventName(*eventToPurge));
656 RemoveEvent(indexToPurge);
659 if (iQueueList.First()!=this)
660 goto breakLoopAndRemoveEvent;
663 if (iQueueList.First()!=this)
664 goto breakLoopAndRemoveEvent;
665 for (indexToMatch=indexToPurge+1;indexToMatch<iCount;indexToMatch++)
667 eventToMatch=EventPtr(indexToMatch);
668 if (eventToMatch->Type()==EEventKeyUp && eventToMatch->Key()->iScanCode==eventToPurge->Key()->iScanCode)
670 *eventToMatch=iNullEvent;
671 goto breakLoopAndRemoveEvent;
675 case EEventModifiersChanged:
676 for (indexToMatch=indexToPurge;indexToMatch>0;)
678 eventToMatch=EventPtr(--indexToMatch);
679 if (eventToMatch->Type()==EEventModifiersChanged)
681 eventToPurge->ModifiersChanged()->iChangedModifiers|=eventToMatch->ModifiersChanged()->iChangedModifiers;
682 indexToPurge=indexToMatch;
683 goto breakLoopAndRemoveEvent;
687 case EEventPointerBufferReady:
688 CWsPointerBuffer::DiscardPointerMoveBuffer(eventToPurge->Handle());
689 goto breakLoopAndRemoveEvent;
690 case EEventFocusLost:
691 case EEventFocusGained:
692 if ((indexToPurge+1)<iCount)
694 eventToMatch=EventPtr(indexToPurge+1);
695 if (eventToMatch->Type()==EEventFocusLost || eventToMatch->Type()==EEventFocusGained)
697 *eventToMatch=iNullEvent;
698 goto breakLoopAndRemoveEvent;
703 if ((indexToPurge+1)<iCount && EventPtr(indexToPurge+1)->Type()==EEventSwitchOn)
704 goto breakLoopAndRemoveEvent;
707 if (CheckPurgePointerEvent(eventToPurge))
709 goto breakLoopAndRemoveEvent;
716 void CEventQueue::PurgePointerEvents()
718 TWsEvent* eventToPurge;
719 TInt indexToPurge=iCount;
720 while(indexToPurge>0)
722 eventToPurge=EventPtr(--indexToPurge);
723 switch(eventToPurge->Type())
725 case EEventPointerBufferReady:
726 CWsPointerBuffer::DiscardPointerMoveBuffer(eventToPurge->Handle());
727 RemoveEvent(indexToPurge);
730 if (CheckPurgePointerEvent(eventToPurge))
732 RemoveEvent(indexToPurge);
740 Purge requested number of oldest events from inactive event queue.
741 @param aSizeRequired the total events required to be cleared.
742 @return The number of events cleared.
744 TInt CEventQueue::PurgeInactiveEvents(const TInt& aSizeRequired)
746 TInt numEventsCleared = 0;
750 TDblQueIter<CEventQueue> iter(iQueueList);
752 while ((qptr = iter++) != NULL && (aSizeRequired > numEventsCleared))
754 if ((qptr->IsEventCancelled() || (qptr->iEventMsg.IsNull() && !qptr->iEventSignalledState)) &&
755 (qptr->iQueueSize > EMinQueueSize))
757 // we have a client that is not listening with a size larger than min queue size.
758 // so lets remove it's oldest event until the number of removed events meet the requirement.
759 qptr->RemoveEvent(0);
764 } while ((aSizeRequired > numEventsCleared) && isRemoved);
765 return numEventsCleared;
768 void CEventQueue::Compress(TCompressMode aCompressMode)
771 // The different purge modes are
773 // ECompressNoPurge, // Don't purge anything
774 // ECompressPurge1, // Don't purge foreground queue
775 // ECompressPurge2, // Purge all queues
777 if (aCompressMode==ECompressPurge2 ||
778 (this!=iQueueList.First() && aCompressMode==ECompressPurge1))
780 TInt compress=iQueueSize-(iCount>EMinQueueSize?iCount:EMinQueueSize);
783 compress=(compress+1)/2; // Compress half the free space in the queue
784 TWsEvent* head=EventPtr(0);
785 TWsEvent* tail=EventPtr(iCount);
788 EventCopy(iEventPtr+compress,iEventPtr,tail-iEventPtr);
793 EventCopy(iEventPtr+compress,head,iCount);
797 iQueueSize-=compress;
801 void CEventQueue::MoveUp(TInt aNumEvents)
807 EventCopy(iEventPtr+aNumEvents,iEventPtr,iQueueSize);
808 iEventPtr+=aNumEvents;
811 TInt CEventQueue::FollowingGap() const
813 TDblQueIter<CEventQueue> iter(iQueueList);
815 iter.Set(*(CEventQueue *)this);
818 if ((qptr=iter)!=NULL)
821 end=iGlobalEventQueue+iGlobalEventQueueSize;
822 return(end-(iEventPtr+iQueueSize));
825 TInt CEventQueue::SqueezeUp()
827 TInt gap=FollowingGap();
832 void CEventQueue::SqueezeDown()
834 TDblQueIter<CEventQueue> iter(iQueueList);
837 CEventQueue* qptr=iter;
840 Compress(ECompressNoPurge);
841 TInt gap=qptr->FollowingGap();
846 void CEventQueue::MoveToFront()
848 if (this==iQueueList.First())
853 TDblQueIter<CEventQueue> iter(iQueueList);
855 while((qptr=iter--)!=NULL)
858 qptr->Compress(ECompressNoPurge);
859 gap=qptr->SqueezeUp();
862 EventCopy(iGlobalEventQueue,iEventPtr,iQueueSize);
865 EventCopy(iGlobalEventQueue,iEventPtr,gap);
867 TWsEvent copyBuf[ECopyBufSize]; // temp buffer, can copy upto ECopyBufSize events at a time
868 TInt eventsToGo=iQueueSize-gap;
872 TInt copy=Min(eventsToGo,ECopyBufSize);
873 Mem::Copy(©Buf[0],iEventPtr,copy*sizeof(TWsEvent));
876 while((qptr=iter--)!=NULL)
878 EventCopy(iGlobalEventQueue+iQueueSize,©Buf[0],copy);
882 } while(eventsToGo>0);
884 iEventPtr=iGlobalEventQueue;
886 iQueueList.AddFirst(*this);
893 CEventQueue::CEventQueue(CWsClient* aOwner) : CEventBase(aOwner)
895 __DECLARE_NAME(_S("CEventQueue"));
898 CEventQueue::~CEventQueue()
903 void CEventQueue::InitStaticsL()
905 User::LeaveIfError(iMutex.CreateLocal());
908 void CEventQueue::DeleteStaticsL()
913 void CEventQueue::ConstructL()
916 Mem::FillZ(&iNullEvent,sizeof(iNullEvent));
919 TWsEvent* CEventQueue::EventPtr(TInt index)
921 return(iEventPtr+((iHead+index)%iQueueSize));
924 TBool CEventQueue::QueueEvent(const TWsEvent &event)
926 TWservEventPriorities priority=EEventPriorityLow;
927 #ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP
928 if (event.Type()==EEventPassword || event.Type()==EEventSwitchOff ||
929 event.Type()==EEventKeySwitchOff || event.Type()==EEventRestartSystem)
931 if (event.Type()==EEventPassword || event.Type()==EEventSwitchOff || event.Type()==EEventKeySwitchOff)
934 priority=EEventPriorityHigh;
936 return(QueueEvent(event,priority));
939 TBool CEventQueue::CheckRoom()
941 // If the queue is full and room is created return ETrue
946 if (iCount==iQueueSize && Expand(EEventPriorityHigh))
952 TBool CEventQueue::QueueEvent(const TWsEvent &event, TWservEventPriorities aPriority)
954 // Queue an event, returns ETrue if queued or delivered, EFalse if the queue was full.
959 if (iCount==iQueueSize && !Expand(aPriority))
962 #ifdef LOG_WSERV_EVENTS
963 RDebug::Printf("WSEVENT: CEventQueue::QueueEvent(): 0x%X: Queue Full!!!!!, iCount = %d, iQueueSize = %d", this, iCount, iQueueSize);
964 RDebug::Print(_L("WSEVENT: CEventQueue::QueueEvent(): 0x%X: Queue Full!!!!! TWsEvent.Type() = %S"), this, &WsEventName(event));
969 if (!iEventMsg.IsNull())
973 #ifdef LOG_WSERV_EVENTS
974 RDebug::Printf("_WSEVENT: CEventQueue::QueueEvent, Right before adding the event");
976 *EventPtr(iCount++)=event;
977 #ifdef LOG_WSERV_EVENTS
978 RDebug::Print(_L("_WSEVENT: CEventQueue::QueueEvent, Event %S successfully queued"), &WsEventName(event));
985 TBool CEventQueue::QueueEvent(TUint32 aTarget, TInt aEvent, TInt aIntVal)
988 event.SetType(aEvent);
989 event.SetHandle(aTarget);
991 *(event.Int()) = aIntVal;
992 return(QueueEvent(event));
995 void CEventQueue::UpdateLastEvent(const TWsEvent &event)
997 WS_ASSERT_DEBUG(iCount>0, EWsPanicQueueUpdateCount);
998 Mem::Copy(EventPtr(iCount-1)->EventData(),event.EventData(),TWsEvent::EWsEventDataSize);
1002 Replaces last pointer event related to particular pointer with new one.
1004 While searching for event to replace this method considers all events on the
1005 queue except EMove and EDrag pointer events from pointers different than aEvent.
1006 If the last of these events under consideration:
1007 (1) is a pointer event,
1008 (2) has the same type as aEvent,
1009 (3) its type is either EMove or EDrag and
1010 (4) has the same window handle as aEvent,
1011 then it is removed from the queue and aEvent is put at the end of the queue.
1013 @return ETrue if event on the queue has been replaced with aEvent, EFalse otherwise.
1015 TBool CEventQueue::UpdateLastPointerEvent(const TWsEvent &aEvent)
1017 if (aEvent.Pointer()->iType == TPointerEvent::EMove || aEvent.Pointer()->iType == TPointerEvent::EDrag)
1027 // loop through all events on the queue starting from the last one
1028 TWsEvent* evToUpdate = EventPtr(iCount);
1029 TWsEvent* evOnHead = &iEventPtr[iHead];
1030 while (evToUpdate != evOnHead)
1032 DecEventPointer(evToUpdate);
1034 // conditions that stop searching
1035 if ( (evToUpdate->Type() != EEventPointer) // found non-pointer event
1036 || (evToUpdate->Pointer()->iType != TPointerEvent::EMove && evToUpdate->Pointer()->iType != TPointerEvent::EDrag) // pointer event but wrong type
1037 || ( (TAdvancedPointerEventHelper::PointerNumber(*evToUpdate) == TAdvancedPointerEventHelper::PointerNumber(aEvent))
1038 && ( (evToUpdate->Handle() != aEvent.Handle()) // good number & bad handle
1039 || (evToUpdate->Pointer()->iType != aEvent.Pointer()->iType)))) // good number & bad type
1044 else if (TAdvancedPointerEventHelper::PointerNumber(*evToUpdate) == TAdvancedPointerEventHelper::PointerNumber(aEvent))
1046 // we found event to update: evToUpdate is pointer event with right type, pointer number
1047 // and window handle
1049 if (evToUpdate == EventPtr(iCount - 1))
1051 UpdateLastEvent(aEvent);
1055 RemoveEvent(evToUpdate);
1056 *EventPtr(iCount++) = aEvent;
1062 // evToUpdate is EMove or EDrag pointer event with different pointer id,
1063 // continue to loop through the queue
1070 void CEventQueue::GetData()
1072 // If there is an outstanding event in the queue, reply with it's data and remove it from the Q
1077 WS_ASSERT_DEBUG((iEventPtr+iHead)->Type()!=EEventMarkInvalid, EWsPanicCheckEventQueue);
1078 CEventBase::GetData(iEventPtr+iHead,sizeof(*iEventPtr));
1079 #ifdef LOG_WSERV_EVENTS
1080 RDebug::Printf("_WSEVENT: CEventQueue::GetData(), TWsEvent.Type() = %d", (iEventPtr+iHead)->Type());
1082 __ZAP_EVENT(iEventPtr+iHead);
1083 iHead=(iHead+1)%iQueueSize;
1087 CEventBase::GetData(&iNullEvent,sizeof(iNullEvent));
1090 void CEventQueue::EventReady(const RMessagePtr2& aEventMsg)
1092 // Queue a read of an event notification
1097 iEventMsg=aEventMsg;
1103 void CEventQueue::RemoveEvent(TInt index)
1105 // Remove event 'index' in the queue, this event MUST exist in the queue
1108 #ifdef LOG_WSERV_EVENTS
1109 RDebug::Printf("_WSEVENT: CEventQueue::RemoveEvent(index), Remove event index %d in the queue", index);
1111 WS_ASSERT_DEBUG(index < iCount, EWsPanicCheckEventQueue);
1112 RemoveEvent(EventPtr(index));
1115 void CEventQueue::RemoveEvent(TWsEvent* aEvToRemove)
1117 // Remove event in the queue, this event MUST exist in the queue
1120 #ifdef LOG_WSERV_EVENTS
1121 RDebug::Print(_L("_WSEVENT: CEventQueue::RemoveEvent(aEvToRemove), Remove event %S in the queue"), &WsEventName(*aEvToRemove));
1124 TWsEvent* last = EventPtr(iCount);
1126 while(aEvToRemove!=last)
1129 IncEventPointer(aEvToRemove);
1130 *prev = *aEvToRemove;
1135 const TWsEvent* CEventQueue::PeekLastEvent()
1137 // Return a read only pointer to the last event in the queue (or NULL if no event)
1142 return(EventPtr(iCount-1));
1145 void CEventQueue::Wait()
1150 void CEventQueue::Signal()
1155 void CEventQueue::WalkEventQueue(EventQueueWalk aFunc, TAny* aFuncParam)
1159 for (TInt index=0;index<iCount;index++)
1161 TWsEvent* event=EventPtr(index);
1162 switch((aFunc)(aFuncParam,event))
1164 case EEventQueueWalkDeleteEvent:
1165 RemoveEvent(index--);
1166 case EEventQueueWalkOk:
1168 case EEventQueueWalkRestart: