os/graphics/windowing/windowserver/nga/SERVER/EVQUEUE.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 1994-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Window server event queue handling
    15 // 
    16 //
    17 
    18 #include "EVQUEUE.H"
    19 
    20 #include "server.h"
    21 #include "wstop.h"
    22 #include "panics.h"
    23 #include "pointer.h"
    24 #include "advancedpointereventhelper.h"
    25 #include "debughelper.h"
    26 
    27 GLREF_D CDebugLogBase* wsDebugLog;
    28 
    29 #if defined(_DEBUG)
    30 #define __CHECK_QUEUE() CheckQueue()
    31 #define __ZAP_EVENTS(pointer,len) ZapEvents(pointer,len)
    32 #define __ZAP_EVENT(pointer) ZapEvent(pointer)
    33 #else
    34 #define __CHECK_QUEUE()
    35 #define __ZAP_EVENTS(pointer,len)
    36 #define __ZAP_EVENT(pointer)
    37 #endif
    38 
    39 const TInt ECopyBufSize=4;
    40 
    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;
    47 
    48 // CEventBase
    49 
    50 CEventBase::~CEventBase()
    51 	{
    52 	if (!iEventMsg.IsNull() && CWsTop::ShuttingDown())
    53 		{
    54 		iEventMsg.Complete(KErrServerTerminated);
    55 		}
    56 	}
    57 
    58 CEventBase::CEventBase(CWsClient* aOwner) : iWsOwner(aOwner)
    59 	{
    60 	}
    61 
    62 void CEventBase::SignalEvent(TInt aCode)
    63 	{
    64 	if (wsDebugLog)
    65 		wsDebugLog->SignalEvent(iWsOwner->ConnectionHandle());
    66 	iEventMsg.Complete(aCode);
    67 	iEventSignalledState|=EEventFlagSignalled;
    68 	}
    69 
    70 inline TBool CEventBase::IsEventCancelled()
    71 	{
    72 	return (iEventSignalledState & EEventFlagCancelled);
    73 	}
    74 
    75 /**
    76 If there is an outstanding read cancel it.
    77 */
    78 void CEventBase::CancelRead()
    79 	{
    80 	if (!iEventMsg.IsNull())
    81 		{
    82 		iEventMsg.Complete(KErrCancel);
    83 		}
    84 	iEventSignalledState|=EEventFlagCancelled;
    85 	}
    86 
    87 void CEventBase::GetData(TAny* aData, TInt aLen)
    88 	{
    89 	if (!(iEventSignalledState&EEventFlagSignalled))
    90 		{
    91 		iWsOwner->PPanic(EWservPanicUnsignalledEventData);
    92 		}
    93 	iEventSignalledState&=~EEventFlagSignalled;
    94 	CWsClient::ReplyBuf(aData,aLen);
    95 	}
    96 
    97 /**
    98 Queue a read of an event notification
    99 */
   100 void CEventBase::EventReadyCheck()
   101 	{
   102 	if ((iEventSignalledState&EEventFlagSignalled && !(iEventSignalledState&EEventFlagCancelled)) || !iEventMsg.IsNull())
   103 		{
   104 #ifdef LOG_WSERV_EVENTS
   105 		RDebug::Printf("_WSEVENT: CEventBase::EventReadyCheck, Event is not ready");
   106 #endif
   107 		iWsOwner->PPanic(EWservPanicReadOutstanding);
   108 		}
   109 	}
   110 
   111 void CEventBase::EventReady(const RMessagePtr2& aEventMsg)
   112 	{
   113 	EventReadyCheck();
   114 	iEventMsg=aEventMsg;
   115 	}
   116 
   117 // CEventQueue - Queue organisation functions
   118 
   119 #if defined(_DEBUG)
   120 void CEventQueue::CheckQueue()
   121 	{
   122 	TDblQueIter<CEventQueue> iter(iQueueList);
   123 	CEventQueue* qptr;
   124 	iter.SetToFirst();
   125 	while((qptr=iter++)!=NULL)
   126 		{
   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);
   131 		}
   132 	for(TInt index=0;index<iCount;index++)
   133 		{
   134 		TWsEvent* ev=EventPtr(index);
   135 		WS_ASSERT_DEBUG(ev->Type()!=EEventMarkInvalid, EWsPanicCheckEventQueue);
   136 		}
   137 	}
   138 
   139 void CEventQueue::ZapEvent(TWsEvent* aTarget)
   140 	{
   141 	aTarget->SetType(EEventMarkInvalid);
   142 	aTarget->SetHandle(555);
   143 	}
   144 
   145 void CEventQueue::ZapEvents(TWsEvent* aTarget, TInt aLen)
   146 	{
   147 	for(TInt index=0;index<aLen;index++)
   148 		ZapEvent(aTarget+index);
   149 	}
   150 
   151 #endif
   152 
   153 void CEventQueue::EventCopy(TWsEvent* aTarget, TWsEvent* aSource, TInt aNumEvents)
   154 	{
   155 	WS_ASSERT_DEBUG(aTarget>=iGlobalEventQueue, EWsPanicEventQueueCopy);
   156 	WS_ASSERT_DEBUG((aTarget+aNumEvents)<=(iGlobalEventQueue+iGlobalEventQueueSize), EWsPanicEventQueueCopy);
   157 	Mem::Copy(aTarget,aSource,aNumEvents*sizeof(TWsEvent));
   158 	}
   159 
   160 TInt CEventQueue::RequiredQueueSize(TInt aNumConnections)
   161 	{
   162 	return(aNumConnections*EMinQueueSize+EMaxQueueSize+EExtraQueueSize);
   163 	}
   164 
   165 /**
   166 Adjust the Global Queue Size.
   167 @param aNewSize the size for event queue. 
   168 */
   169 void CEventQueue::AdjustQueueSizeL(TInt aNewSize)
   170 	{
   171 	TWsEvent* oldQ=iGlobalEventQueue;
   172 	if (aNewSize < iGlobalEventQueueSize)
   173 		{
   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))
   178 			{
   179 			return;
   180 			}
   181 		TInt allocSize = aNewSize * sizeof(TWsEvent);
   182 		iGlobalEventQueue=static_cast<TWsEvent*>(User::AllocL(allocSize));
   183 		Mem::Copy(iGlobalEventQueue,oldQ,allocSize);
   184 		User::Free(oldQ);
   185 		}
   186 	else
   187 		{
   188 		iGlobalEventQueue = static_cast<TWsEvent*>(User::ReAllocL(iGlobalEventQueue, aNewSize * sizeof(TWsEvent)));
   189 		}
   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));
   194 	if (diff)
   195 		{
   196 		TDblQueIter<CEventQueue> iter(iQueueList);
   197 		CEventQueue* qptr;
   198 		while((qptr=iter++)!=NULL)
   199 			{
   200 			qptr->iEventPtr=reinterpret_cast<TWsEvent*>(reinterpret_cast<TInt8*>(qptr->iEventPtr)+diff);
   201 			}
   202 		}
   203 	}
   204 
   205 void CEventQueue::AddQueueL()
   206 	{
   207 	Wait();
   208 	if ((iNumConnections%EQueueGranularity)==0)
   209 		{
   210 		const TInt newSize = RequiredQueueSize(iNumConnections + EQueueGranularity);
   211 		TRAPD(err,AdjustQueueSizeL(newSize));
   212 		if (err!=KErrNone)
   213 			{
   214 			__CHECK_QUEUE();
   215 			Signal();
   216 			User::Leave(err);
   217 			}
   218 		}
   219 	iNumConnections++;
   220 	if (iQueueList.IsEmpty())
   221 		iEventPtr=iGlobalEventQueue;
   222 	else
   223 		{
   224 		CEventQueue* qptr=iQueueList.Last();
   225 		iEventPtr=qptr->iEventPtr+qptr->iQueueSize;
   226 		}
   227 	iQueueList.AddLast(*this);
   228 	
   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)
   234 		{
   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));
   240 		if (err != KErrNone)
   241 			{
   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)
   246 				{
   247 				__CHECK_QUEUE();
   248 				Signal();
   249 				User::Leave(err);
   250 				}
   251 			}
   252 		while (doExpand(ECompressNoPurge)) {};
   253 		}
   254 	__CHECK_QUEUE();
   255 	Signal();
   256 	}
   257 
   258 void CEventQueue::RemoveQueue()
   259 	{
   260 	Wait();
   261 	if (iEventPtr)	// If this is still NULL this class hasn't been linked into the Q list
   262 		{
   263 		__ZAP_EVENTS(iEventPtr, iQueueSize);
   264 		iLink.Deque();
   265 		if (--iNumConnections==0)
   266 			{
   267 			User::Free(iGlobalEventQueue);
   268 			iGlobalEventQueue=NULL;
   269 			iGlobalEventQueueSize=0;
   270 			}
   271 		else if ((iNumConnections%EQueueGranularity)==0)
   272 			{
   273 			TDblQueIter<CEventQueue> iter(iQueueList);
   274 			CEventQueue* qptr=iter++;
   275 			qptr->Compress(ECompressNoPurge);
   276 			while((qptr=iter++)!=NULL)
   277 				{
   278 				qptr->SqueezeDown();
   279 				}
   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
   285 			}
   286 		iCount=0;
   287 		}
   288 	__CHECK_QUEUE();
   289 	Signal();
   290 	}
   291 
   292 void CEventQueue::IncreaseQueueSize(TInt aNumSpaces)
   293 	{
   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;
   300 	}
   301 
   302 TBool CEventQueue::Expand(TWservEventPriorities aPriority)
   303 	{
   304 	if (iQueueSize==EMaxQueueSize)
   305 		{
   306 		if (aPriority==EEventPriorityHigh)
   307 			{
   308 			Purge();
   309 			if (iCount<iQueueSize)
   310 				return(ETrue);	// Success!
   311 			}
   312 		}
   313 	else if (doExpand(ECompressNoPurge) ||
   314 			  doExpand(ECompressPurge1) ||
   315 			   (aPriority==EEventPriorityHigh && doExpand(ECompressPurge2)))
   316 		return(ETrue);
   317 	return(EFalse);				// Failure
   318 	}
   319 
   320 TBool CEventQueue::doExpand(TCompressMode aCompressMode)
   321 	{
   322 	TDblQueIter<CEventQueue> iter(iQueueList);
   323 	iter.SetToLast();
   324 	CEventQueue* qptr=NULL;
   325 	TInt spare=0;
   326 	
   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)
   330 		{
   331 		spare=qptr->SqueezeUp();
   332 		if (spare==0)
   333 			qptr->Compress(aCompressMode);
   334 		iter--;
   335 		}
   336 
   337 	// current queue, if we have space then expand the same and return
   338 	spare=FollowingGap();
   339 	if (spare>0)
   340 		{
   341 		IncreaseQueueSize(spare);
   342 		__CHECK_QUEUE();
   343 		return(ETrue);
   344 		}
   345 
   346 	// while loop from current queue till first queue, looking for a gap 
   347 	// between queues and using the first one it finds
   348 	iter--;
   349 	TInt expanded=0;
   350 	while((qptr=iter)!=NULL)
   351 		{
   352 		expanded=qptr->SqueezeUp();
   353 		if (expanded>0)
   354 			{
   355 			return MoveDownAndExpand(iter, expanded);
   356 			}
   357 		else
   358 			qptr->Compress(aCompressMode);
   359 		iter--;
   360 		}
   361 
   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
   365 	iter.SetToFirst();
   366 	qptr=iter;
   367 	if (qptr->iEventPtr>iGlobalEventQueue)
   368 		{
   369 		return MoveDownAndExpand(iter, qptr->iEventPtr-iGlobalEventQueue);
   370 		}
   371 
   372 	__CHECK_QUEUE();
   373 	return(EFalse);	// Failed to expand enough room
   374 	}
   375 
   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)
   379 	{
   380 	CEventQueue* qptr=NULL;
   381 	FOREVER
   382 		{
   383 		qptr=aIter++;
   384 		qptr->MoveDown(aExpand);
   385 		if (qptr==this)
   386 			{
   387 			IncreaseQueueSize(aExpand);
   388 			__CHECK_QUEUE();
   389 			break;
   390 			}
   391 		}
   392 	return ETrue;
   393 	}
   394 
   395 void CEventQueue::MoveDown(TInt aMove)
   396 	{
   397 	if (!aMove)
   398 		{
   399 		return;
   400 		}
   401 	EventCopy(iEventPtr-aMove,iEventPtr,iQueueSize);
   402 	iEventPtr-=aMove;
   403 	}
   404 
   405 /*void CEventQueue::LogUpDownEvents(TChar aChar)
   406 	{
   407 	TWsEvent *event;
   408 	TBuf<128> buf;
   409 	buf.Zero();
   410 	buf.Append(aChar);
   411 	buf.Append('#');
   412 	buf.Append('#');
   413 	TBool some=EFalse;
   414 	TInt index;
   415 
   416 	for (index=0;index<iCount;index++)
   417 		{
   418 		event=EventPtr(index);
   419 		if (event->Type()==EEventPointer)
   420 			{
   421 			if (event->Pointer()->iType==TPointerEvent::EButton1Down
   422 										|| event->Pointer()->iType==TPointerEvent::EButton1Up)
   423 				{
   424 				some=ETrue;
   425 				if (event->Pointer()->iType==TPointerEvent::EButton1Down)
   426 					buf.Append('D');
   427 				else
   428 					buf.Append('U');
   429 				buf.AppendNum(index);
   430 				}
   431 			}
   432 		}
   433 	if (wsDebugLog)
   434 		wsDebugLog->MiscMessage(ELogImportant,buf);
   435 	}*/
   436 
   437 inline void CEventQueue::IncEventPointer(TWsEvent*& aEventPtr)
   438 	{
   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])
   441 		{
   442 		aEventPtr=iEventPtr;
   443 		}
   444 	}
   445 
   446 inline void CEventQueue::DecEventPointer(TWsEvent*& aEventPtr)
   447 	{
   448 	if(aEventPtr--==iEventPtr)
   449 		{
   450 		aEventPtr=&iEventPtr[iQueueSize - 1];
   451 		}
   452 	}
   453 
   454 /*
   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.
   459 
   460 Search will be stopped if an event of type aSearchTerminator is found with the same pointer number 
   461 as aEventToPurge.
   462 
   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.
   466 */
   467 void CEventQueue::PurgeEventPairs(TWsEvent* aEventToPurge, TPointerEvent::TType aMatchingType, TPointerEvent::TType aSearchTerminator)
   468 		{
   469 		TWsEvent* eventToMatch = aEventToPurge;
   470 		TWsEvent* lastEvent = EventPtr(iCount);
   471 		for(IncEventPointer(eventToMatch);eventToMatch!=lastEvent;IncEventPointer(eventToMatch))
   472 			{
   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 
   475 				{
   476 				if ((eventToMatch->Pointer()->iType==aMatchingType) // correct event type
   477 						&& (eventToMatch->Handle()==aEventToPurge->Handle())) // same window
   478 					{
   479 					*eventToMatch=iNullEvent;
   480 					return;
   481 					}
   482 				else if (eventToMatch->Pointer()->iType==aSearchTerminator) // stop searching for mathing type
   483 					{
   484 					return;
   485 					}
   486 				}
   487 			};
   488 		TWsPointer::UnmatchedEventPurged(aMatchingType, aEventToPurge);
   489 		}
   490 
   491 /**
   492 Starting from pointer event aBasePointerEvent searches the queue for next pointer event for 
   493 the same pointer by iterating towards end of queue.
   494 
   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.
   499 */
   500 TWsEvent* CEventQueue::NextPointerEvent(TWsEvent* aBasePointerEvent)
   501 	{
   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))
   508 		{
   509 		if ((currentEvent->Type() == EEventPointer) && 
   510 			(TAdvancedPointerEventHelper::PointerNumber(*currentEvent) == pointerNumber))
   511 			{
   512 			if (currentEvent->Handle() == handle)
   513 				{
   514 				return currentEvent;
   515 				}
   516 			else
   517 				{
   518 				return NULL;
   519 				}
   520 			}
   521 		};
   522 	return NULL;
   523 	}
   524 	
   525 /**
   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.
   528 
   529 @return ETrue if aEventToPurge should be purged, EFalse otherwise.
   530 */
   531 TBool CEventQueue::CheckPurgePointerEvent(TWsEvent* aEventToPurge)
   532 	{
   533 	switch(aEventToPurge->Pointer()->iType)
   534 		{
   535 		case TPointerEvent::EDrag:
   536 		case TPointerEvent::EMove:
   537 		case TPointerEvent::EButtonRepeat:
   538 		case TPointerEvent::ESwitchOn:
   539 			return ETrue;
   540 		case TPointerEvent::EButton1Down:
   541 			PurgeEventPairs(aEventToPurge,TPointerEvent::EButton1Up, TPointerEvent::ENullType);
   542 			return ETrue;
   543 		case TPointerEvent::EButton2Down:
   544 			PurgeEventPairs(aEventToPurge,TPointerEvent::EButton2Up, TPointerEvent::ENullType);
   545 			return ETrue;
   546 		case TPointerEvent::EButton3Down:
   547 			PurgeEventPairs(aEventToPurge,TPointerEvent::EButton3Up, TPointerEvent::ENullType);
   548 			return ETrue;
   549 		case TPointerEvent::EEnterHighPressure:
   550 			PurgeEventPairs(aEventToPurge,TPointerEvent::EExitHighPressure, TPointerEvent::EButton1Up);
   551 			return ETrue;
   552 		case TPointerEvent::EEnterCloseProximity:
   553 			{
   554 			TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
   555 			if (nextEvent != NULL)
   556 				{
   557 				switch(nextEvent->Pointer()->iType)
   558 					{
   559 					case TPointerEvent::EExitCloseProximity: 
   560 						*nextEvent = iNullEvent;
   561 					case TPointerEvent::EOutOfRange:
   562 						return ETrue;
   563 					}
   564 				}
   565 			break;
   566 			}
   567 		case TPointerEvent::EExitCloseProximity:
   568 			{
   569 			TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
   570 			if (nextEvent != NULL)
   571 				{
   572 				switch(nextEvent->Pointer()->iType)
   573 					{
   574 					case TPointerEvent::EEnterCloseProximity: 
   575 						*nextEvent = iNullEvent;
   576 					case TPointerEvent::EOutOfRange:
   577 						return ETrue;
   578 					}
   579 				}
   580 			break;
   581 			}
   582 		case TPointerEvent::EOutOfRange:
   583 			{
   584 			TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
   585 			if ((nextEvent != NULL) &&
   586 				(nextEvent->Pointer()->iType == TPointerEvent::EOutOfRange))
   587 				{
   588 				return ETrue;
   589 				}
   590 			break;
   591 			}
   592 		case TPointerEvent::EExitHighPressure:
   593 			{
   594 			TWsEvent* nextEvent = NextPointerEvent(aEventToPurge);
   595 			if ((nextEvent != NULL) &&
   596 				(nextEvent->Pointer()->iType == TPointerEvent::EButton1Up))
   597 				{
   598 				return ETrue;
   599 				}
   600 			break;
   601 			}
   602 		case TPointerEvent::EButton1Up:
   603 		case TPointerEvent::EButton2Up:
   604 		case TPointerEvent::EButton3Up:
   605 			break;
   606 		}
   607 	return EFalse;
   608 	}
   609 
   610 /** Purgable events are:
   611 	Pointer Up/Down pairs belonging to the same pointer, button and window
   612     Pointer moves & drags
   613     Key messages
   614     Key Up/Down pairs
   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
   622 
   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
   629 */	
   630 void CEventQueue::Purge()
   631 	{
   632 	TWsEvent* eventToPurge;
   633 	TWsEvent* eventToMatch;
   634 	TInt indexToMatch;
   635 	TInt indexToPurge=iCount;
   636 	while(indexToPurge>0)
   637 		{
   638 		eventToPurge=EventPtr(--indexToPurge);
   639 		switch(eventToPurge->Type())
   640 			{
   641 			case EEventPassword:
   642 				break;
   643 			case EEventMarkInvalid:
   644 #if defined(_DEBUG)
   645 				WS_PANIC_DEBUG(EWsPanicCheckEventQueue);
   646 #endif
   647 			case EEventNull:
   648 			case EEventKey:
   649 			case EEventPointerEnter:
   650 			case EEventPointerExit:
   651 			case EEventDragDrop:
   652 breakLoopAndRemoveEvent:
   653 #ifdef LOG_WSERV_EVENTS
   654 				RDebug::Print(_L("_WSEVENT: CEventQueue::Purge(), The event to be purged is %S"), &WsEventName(*eventToPurge));
   655 #endif
   656 				RemoveEvent(indexToPurge);
   657 				return;
   658 			case EEventKeyUp:
   659 				if (iQueueList.First()!=this)
   660 					goto breakLoopAndRemoveEvent;
   661 				break;
   662 			case EEventKeyDown:
   663 				if (iQueueList.First()!=this)
   664 					goto breakLoopAndRemoveEvent;
   665 				for (indexToMatch=indexToPurge+1;indexToMatch<iCount;indexToMatch++)
   666 					{
   667 					eventToMatch=EventPtr(indexToMatch);
   668 					if (eventToMatch->Type()==EEventKeyUp && eventToMatch->Key()->iScanCode==eventToPurge->Key()->iScanCode)
   669 						{
   670 						*eventToMatch=iNullEvent;
   671 						goto breakLoopAndRemoveEvent;
   672 						}
   673 					}
   674 				break;
   675 			case EEventModifiersChanged:
   676 				for (indexToMatch=indexToPurge;indexToMatch>0;)
   677 					{
   678 					eventToMatch=EventPtr(--indexToMatch);
   679 					if (eventToMatch->Type()==EEventModifiersChanged)
   680 						{
   681 						eventToPurge->ModifiersChanged()->iChangedModifiers|=eventToMatch->ModifiersChanged()->iChangedModifiers;
   682 						indexToPurge=indexToMatch;
   683 						goto breakLoopAndRemoveEvent;
   684 						}
   685 					}
   686 				break;
   687 			case EEventPointerBufferReady:
   688 				CWsPointerBuffer::DiscardPointerMoveBuffer(eventToPurge->Handle());
   689 				goto breakLoopAndRemoveEvent;
   690 			case EEventFocusLost:
   691 			case EEventFocusGained:
   692 				if ((indexToPurge+1)<iCount)
   693 					{
   694 					eventToMatch=EventPtr(indexToPurge+1);
   695 					if (eventToMatch->Type()==EEventFocusLost || eventToMatch->Type()==EEventFocusGained)
   696 						{
   697 						*eventToMatch=iNullEvent;
   698 						goto breakLoopAndRemoveEvent;
   699 						}
   700 					}
   701 				break;
   702 			case EEventSwitchOn:
   703 				if ((indexToPurge+1)<iCount && EventPtr(indexToPurge+1)->Type()==EEventSwitchOn)
   704 					goto breakLoopAndRemoveEvent;
   705 				break;
   706 			case EEventPointer:
   707 				if (CheckPurgePointerEvent(eventToPurge))
   708 					{
   709 					goto breakLoopAndRemoveEvent;
   710 					}
   711 				break;
   712 			}
   713 		}
   714 	}
   715 
   716 void CEventQueue::PurgePointerEvents()
   717 	{
   718 	TWsEvent* eventToPurge;
   719 	TInt indexToPurge=iCount;
   720 	while(indexToPurge>0)
   721 		{
   722 		eventToPurge=EventPtr(--indexToPurge);
   723 		switch(eventToPurge->Type())
   724 			{
   725 			case EEventPointerBufferReady:
   726 				CWsPointerBuffer::DiscardPointerMoveBuffer(eventToPurge->Handle());
   727 				RemoveEvent(indexToPurge);
   728 				break;
   729 			case EEventPointer:
   730 				if (CheckPurgePointerEvent(eventToPurge))
   731 					{
   732 					RemoveEvent(indexToPurge);
   733 					}
   734 				break;
   735 			}
   736 		}
   737 	}
   738 
   739 /**
   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.
   743 */
   744 TInt CEventQueue::PurgeInactiveEvents(const TInt& aSizeRequired)
   745 	{
   746 	TInt numEventsCleared = 0;
   747 	CEventQueue* qptr;
   748 	TBool isRemoved;
   749 	do	{
   750 		TDblQueIter<CEventQueue> iter(iQueueList);
   751 		isRemoved = EFalse;
   752 		while ((qptr = iter++) != NULL && (aSizeRequired > numEventsCleared))
   753 			{
   754 			if ((qptr->IsEventCancelled() || (qptr->iEventMsg.IsNull() && !qptr->iEventSignalledState)) &&
   755 				(qptr->iQueueSize > EMinQueueSize))
   756 				{
   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);
   760 				numEventsCleared++;
   761 				isRemoved = ETrue;
   762 				}
   763 			}
   764 		} while ((aSizeRequired > numEventsCleared) && isRemoved);
   765 	return numEventsCleared;
   766 	}
   767 
   768 void CEventQueue::Compress(TCompressMode aCompressMode)
   769 	{
   770 //
   771 // The different purge modes are
   772 //
   773 // ECompressNoPurge,	// Don't purge anything
   774 // ECompressPurge1,		// Don't purge foreground queue
   775 // ECompressPurge2,		// Purge all queues
   776 //
   777 	if (aCompressMode==ECompressPurge2 ||
   778 		(this!=iQueueList.First() && aCompressMode==ECompressPurge1))
   779 		Purge();
   780 	TInt compress=iQueueSize-(iCount>EMinQueueSize?iCount:EMinQueueSize);
   781 	if (compress>0)
   782 		{
   783 		compress=(compress+1)/2;	// Compress half the free space in the queue
   784 		TWsEvent* head=EventPtr(0);
   785 		TWsEvent* tail=EventPtr(iCount);
   786 		if (head>tail)
   787 			{
   788 			EventCopy(iEventPtr+compress,iEventPtr,tail-iEventPtr);
   789 			iHead-=compress;
   790 			}
   791 		else
   792 			{
   793 			EventCopy(iEventPtr+compress,head,iCount);
   794 			iHead=0;
   795 			}
   796 		iEventPtr+=compress;
   797 		iQueueSize-=compress;
   798 		}
   799 	}
   800 
   801 void CEventQueue::MoveUp(TInt aNumEvents)
   802 	{
   803 	if (!aNumEvents)
   804 		{
   805 		return;
   806 		}
   807 	EventCopy(iEventPtr+aNumEvents,iEventPtr,iQueueSize);
   808 	iEventPtr+=aNumEvents;
   809 	}
   810 
   811 TInt CEventQueue::FollowingGap() const
   812 	{
   813 	TDblQueIter<CEventQueue> iter(iQueueList);
   814 	CEventQueue* qptr;
   815 	iter.Set(*(CEventQueue *)this);
   816 	iter++;
   817 	TWsEvent* end;
   818 	if ((qptr=iter)!=NULL)
   819 		end=qptr->iEventPtr;
   820 	else
   821 		end=iGlobalEventQueue+iGlobalEventQueueSize;
   822 	return(end-(iEventPtr+iQueueSize));
   823 	}
   824 
   825 TInt CEventQueue::SqueezeUp()
   826 	{
   827 	TInt gap=FollowingGap();
   828 	MoveUp(gap);
   829 	return(gap);
   830 	}
   831 
   832 void CEventQueue::SqueezeDown()
   833 	{
   834 	TDblQueIter<CEventQueue> iter(iQueueList);
   835 	iter.Set(*this);
   836 	iter--;
   837 	CEventQueue* qptr=iter;
   838 	if (qptr!=NULL)
   839 		{
   840 		Compress(ECompressNoPurge);
   841 		TInt gap=qptr->FollowingGap();
   842 		MoveDown(gap);
   843 		}
   844 	}
   845 
   846 void CEventQueue::MoveToFront()
   847 	{
   848 	if (this==iQueueList.First())
   849 		return;
   850 	Wait();
   851 	CEventQueue* qptr;
   852 	TInt gap=0;
   853 	TDblQueIter<CEventQueue> iter(iQueueList);
   854 	iter.SetToLast();
   855 	while((qptr=iter--)!=NULL)
   856 		{
   857 		if (gap<iQueueSize)
   858 			qptr->Compress(ECompressNoPurge);
   859 		gap=qptr->SqueezeUp();
   860 		}
   861 	if (gap>=iQueueSize)
   862 		EventCopy(iGlobalEventQueue,iEventPtr,iQueueSize);
   863 	else
   864 		{
   865 		EventCopy(iGlobalEventQueue,iEventPtr,gap);
   866 		iEventPtr+=gap;
   867 		TWsEvent copyBuf[ECopyBufSize];	// temp buffer, can copy upto ECopyBufSize events at a time
   868 		TInt eventsToGo=iQueueSize-gap;
   869 		iQueueSize=gap;
   870 		do
   871 			{
   872 			TInt copy=Min(eventsToGo,ECopyBufSize);
   873 			Mem::Copy(&copyBuf[0],iEventPtr,copy*sizeof(TWsEvent));
   874 			iter.Set(*this);
   875 			iter--;
   876 			while((qptr=iter--)!=NULL)
   877 				qptr->MoveUp(copy);
   878 			EventCopy(iGlobalEventQueue+iQueueSize,&copyBuf[0],copy);
   879 			iQueueSize+=copy;
   880 			eventsToGo-=copy;
   881 			iEventPtr+=copy;
   882 			} while(eventsToGo>0);
   883 		}
   884 	iEventPtr=iGlobalEventQueue;
   885 	this->iLink.Deque();
   886 	iQueueList.AddFirst(*this);
   887 	__CHECK_QUEUE();
   888 	Signal();
   889 	}
   890 
   891 // CEventQueue
   892 
   893 CEventQueue::CEventQueue(CWsClient* aOwner) : CEventBase(aOwner)
   894 	{
   895 	__DECLARE_NAME(_S("CEventQueue"));
   896 	}
   897 
   898 CEventQueue::~CEventQueue()
   899 	{
   900 	RemoveQueue();
   901 	}
   902 
   903 void CEventQueue::InitStaticsL()
   904 	{
   905 	User::LeaveIfError(iMutex.CreateLocal());
   906 	}
   907 
   908 void CEventQueue::DeleteStaticsL()
   909 	{
   910 	iMutex.Close();
   911 	}
   912 
   913 void CEventQueue::ConstructL()
   914 	{
   915 	AddQueueL();
   916 	Mem::FillZ(&iNullEvent,sizeof(iNullEvent));
   917 	}
   918 
   919 TWsEvent* CEventQueue::EventPtr(TInt index)	
   920 	{
   921 	return(iEventPtr+((iHead+index)%iQueueSize));
   922 	}
   923 
   924 TBool CEventQueue::QueueEvent(const TWsEvent &event)
   925 	{
   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)
   930 #else
   931 	if (event.Type()==EEventPassword || event.Type()==EEventSwitchOff || event.Type()==EEventKeySwitchOff)
   932 #endif
   933 		{
   934 		priority=EEventPriorityHigh;
   935 		}
   936 	return(QueueEvent(event,priority));
   937 	}
   938 
   939 TBool CEventQueue::CheckRoom()
   940 //
   941 // If the queue is full and room is created return ETrue
   942 //
   943 	{
   944 	TBool ret=EFalse;
   945 	Wait();
   946 	if (iCount==iQueueSize && Expand(EEventPriorityHigh))
   947 		ret=ETrue;
   948 	Signal();
   949 	return(ret);
   950 	}
   951 
   952 TBool CEventQueue::QueueEvent(const TWsEvent &event, TWservEventPriorities aPriority)
   953 //
   954 // Queue an event, returns ETrue if queued or delivered, EFalse if the queue was full.
   955 //
   956 	{
   957 	TBool ret=ETrue;
   958 	Wait();
   959 	if (iCount==iQueueSize && !Expand(aPriority))
   960 		{
   961 		ret=EFalse;
   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));
   965 #endif
   966 		}
   967 	else
   968 		{
   969 		if (!iEventMsg.IsNull())
   970 			{
   971 			SignalEvent();
   972 			}
   973 #ifdef LOG_WSERV_EVENTS
   974 		RDebug::Printf("_WSEVENT: CEventQueue::QueueEvent, Right before adding the event");
   975 #endif
   976 		*EventPtr(iCount++)=event;
   977 #ifdef LOG_WSERV_EVENTS
   978 		RDebug::Print(_L("_WSEVENT: CEventQueue::QueueEvent, Event %S successfully queued"), &WsEventName(event));
   979 #endif
   980 		}
   981 	Signal();
   982 	return(ret);
   983 	}
   984 
   985 TBool CEventQueue::QueueEvent(TUint32 aTarget, TInt aEvent, TInt aIntVal)
   986 	{
   987 	TWsEvent event;
   988 	event.SetType(aEvent);
   989 	event.SetHandle(aTarget);
   990 	event.SetTimeNow();
   991 	*(event.Int()) = aIntVal;
   992 	return(QueueEvent(event));
   993 	}
   994 
   995 void CEventQueue::UpdateLastEvent(const TWsEvent &event)
   996 	{
   997 	WS_ASSERT_DEBUG(iCount>0, EWsPanicQueueUpdateCount);
   998 	Mem::Copy(EventPtr(iCount-1)->EventData(),event.EventData(),TWsEvent::EWsEventDataSize);
   999 	}
  1000 
  1001 /*
  1002 Replaces last pointer event related to particular pointer with new one.
  1003 
  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.
  1012 
  1013 @return ETrue if event on the queue has been replaced with aEvent, EFalse otherwise. 
  1014 */
  1015 TBool CEventQueue::UpdateLastPointerEvent(const TWsEvent &aEvent)
  1016 	{
  1017 	if (aEvent.Pointer()->iType == TPointerEvent::EMove || aEvent.Pointer()->iType == TPointerEvent::EDrag)
  1018 		{
  1019 		Wait();
  1020 		
  1021 		if (iCount == 0)
  1022 			{
  1023 			Signal();
  1024 			return EFalse;
  1025 			}
  1026 		
  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)
  1031 			{
  1032 			DecEventPointer(evToUpdate);
  1033 			
  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
  1040 				{
  1041 				Signal();
  1042 				return EFalse;
  1043 				}
  1044 			else if (TAdvancedPointerEventHelper::PointerNumber(*evToUpdate) == TAdvancedPointerEventHelper::PointerNumber(aEvent))
  1045 				{
  1046 				// we found event to update: evToUpdate is pointer event with right type, pointer number
  1047 				// and window handle
  1048 				
  1049 				if (evToUpdate == EventPtr(iCount - 1))
  1050 					{
  1051 					UpdateLastEvent(aEvent);
  1052 					}
  1053 				else
  1054 					{
  1055 					RemoveEvent(evToUpdate);
  1056 					*EventPtr(iCount++) = aEvent;
  1057 					}
  1058 				Signal();
  1059 				return ETrue;
  1060 				}
  1061 			
  1062 			// evToUpdate is EMove or EDrag pointer event with different pointer id,
  1063 			// continue to loop through the queue
  1064 			}
  1065 		Signal();
  1066 		}
  1067 	return EFalse;
  1068 	}
  1069 
  1070 void CEventQueue::GetData()
  1071 //
  1072 // If there is an outstanding event in the queue, reply with it's data and remove it from the Q
  1073 //
  1074 	{
  1075 	if (iCount>0)
  1076 		{
  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());
  1081 #endif
  1082 		__ZAP_EVENT(iEventPtr+iHead);
  1083 		iHead=(iHead+1)%iQueueSize;
  1084 		iCount--;
  1085 		}
  1086 	else
  1087 		CEventBase::GetData(&iNullEvent,sizeof(iNullEvent));
  1088 	}
  1089 
  1090 void CEventQueue::EventReady(const RMessagePtr2& aEventMsg)
  1091 //
  1092 // Queue a read of an event notification
  1093 //
  1094 	{
  1095 	EventReadyCheck();
  1096 	Wait();
  1097 	iEventMsg=aEventMsg;
  1098 	if (iCount>0)
  1099 		SignalEvent();
  1100 	Signal();
  1101 	}
  1102 
  1103 void CEventQueue::RemoveEvent(TInt index)
  1104 //
  1105 // Remove event 'index' in the queue, this event MUST exist in the queue
  1106 //
  1107 	{
  1108 #ifdef LOG_WSERV_EVENTS
  1109 	RDebug::Printf("_WSEVENT: CEventQueue::RemoveEvent(index), Remove event index %d in the queue", index);
  1110 #endif
  1111 	WS_ASSERT_DEBUG(index < iCount, EWsPanicCheckEventQueue);
  1112 	RemoveEvent(EventPtr(index));
  1113 	}
  1114 
  1115 void CEventQueue::RemoveEvent(TWsEvent* aEvToRemove)
  1116 //
  1117 // Remove event in the queue, this event MUST exist in the queue
  1118 //
  1119 	{
  1120 #ifdef LOG_WSERV_EVENTS
  1121 	RDebug::Print(_L("_WSEVENT: CEventQueue::RemoveEvent(aEvToRemove), Remove event %S in the queue"), &WsEventName(*aEvToRemove));
  1122 #endif
  1123 	iCount--;
  1124 	TWsEvent* last = EventPtr(iCount);
  1125 	TWsEvent* prev;
  1126 	while(aEvToRemove!=last)
  1127 		{
  1128 		prev = aEvToRemove;
  1129 		IncEventPointer(aEvToRemove);
  1130 		*prev = *aEvToRemove;
  1131 		}
  1132 	__ZAP_EVENT(last);
  1133 	}
  1134 
  1135 const TWsEvent* CEventQueue::PeekLastEvent()
  1136 //
  1137 // Return a read only pointer to the last event in the queue (or NULL if no event)
  1138 //
  1139 	{
  1140 	if (iCount==0)
  1141 		return(NULL);
  1142 	return(EventPtr(iCount-1));
  1143 	}
  1144 
  1145 void CEventQueue::Wait()
  1146 	{
  1147 	iMutex.Wait();
  1148 	}
  1149 
  1150 void CEventQueue::Signal()
  1151 	{
  1152 	iMutex.Signal();
  1153 	}
  1154 
  1155 void CEventQueue::WalkEventQueue(EventQueueWalk aFunc, TAny* aFuncParam)
  1156 	{
  1157 	Wait();
  1158 restart:
  1159 	for (TInt index=0;index<iCount;index++)
  1160 		{
  1161 		TWsEvent* event=EventPtr(index);
  1162 		switch((aFunc)(aFuncParam,event))
  1163 			{
  1164 			case EEventQueueWalkDeleteEvent:
  1165 				RemoveEvent(index--);
  1166 			case EEventQueueWalkOk:
  1167 				break;
  1168 			case EEventQueueWalkRestart:
  1169 				goto restart;
  1170 			}
  1171 		}
  1172 	Signal();
  1173 	}