os/graphics/windowing/windowserver/nga/SERVER/REDRAWQ.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) 1996-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 //
    15 
    16 // Window server redraw queue handling
    17 //
    18 #include <e32std.h>
    19 #include "server.h"
    20 #include "rootwin.h"
    21 #include "windowgroup.h"
    22 #include "walkwindowtree.h"
    23 #include "EVQUEUE.H"
    24 #include "panics.h"
    25 #include "wstop.h"
    26 
    27 const TInt KGranularity = 10;
    28 
    29 void TWsRedrawEvent::SetHandle(TUint aHandle)
    30 	{
    31 	iHandle=aHandle;
    32 	}
    33 
    34 void TWsRedrawEvent::SetRect(TRect aRect)
    35 	{
    36 	iRect=aRect;
    37 	}
    38 
    39 TWsRedrawEvent CRedrawQueue::iNullRedrawEvent;
    40 
    41 CRedrawQueue::CRedrawQueue(CWsClient *aOwner) : CEventBase(aOwner)
    42 	{
    43 	__DECLARE_NAME(_S("CRedrawQueue"));
    44 	}
    45 
    46 void CRedrawQueue::ConstructL()
    47 	{
    48 	iRedrawTrigger=EFalse;
    49 	iRedraws=new(ELeave) CArrayFixSeg<TRedraw>(KGranularity);
    50 	iKeyPriority=new(ELeave) TKeyArrayFix(_FOFF(TRedraw,iPriority),ECmpTUint32);
    51 	iKeyWindow=new(ELeave) TKeyArrayFix(_FOFF(TRedraw,iRedraw),ECmpTUint32);
    52 	Mem::FillZ(&iNullRedrawEvent,sizeof(iNullRedrawEvent));
    53 	}
    54 
    55 CRedrawQueue::~CRedrawQueue()
    56 	{
    57 	delete iRedraws;
    58 	delete iKeyPriority;
    59 	delete iKeyWindow;
    60 	}
    61 
    62 void CRedrawQueue::ReCalcOrder()
    63 	{
    64 	const TInt redrawsCount=iRedraws->Count();
    65 	for(TInt index=0;index<redrawsCount;index++)
    66 		{
    67 		TRedraw *redraw=&iRedraws->At(index);
    68 		redraw->iPriority=static_cast<CWsClientWindow*>(redraw->iRedraw->WsWin())->RedrawPriority();
    69 		}
    70 	iRedraws->Sort(*iKeyPriority);
    71 	}
    72 
    73 void CRedrawQueue::AddInvalid(CWsWindowRedraw *aRedrawWin) 
    74 //
    75 // Add a window to the update list.
    76 //
    77 	{
    78 	TInt index;
    79 	TRedraw redraw;
    80 
    81 	redraw.iRedraw=aRedrawWin;
    82 	if (iRedraws->Find(redraw,*iKeyWindow,index)!=0)
    83 		{
    84 		redraw.iPriority=static_cast<CWsClientWindow*>(aRedrawWin->WsWin())->RedrawPriority();
    85 		TRAPD(err,iRedraws->InsertIsqAllowDuplicatesL(redraw,*iKeyPriority));
    86 		if (err!=KErrNone)
    87 			{
    88 			WS_ASSERT_DEBUG(err==KErrNoMemory,EWsPanicRedrawQueueError);
    89 			iAllocError=ETrue;
    90 			}
    91 		iRedrawTrigger=ETrue;
    92 		}
    93 	}
    94 
    95 void CRedrawQueue::DeleteFromQueue(TInt aIndex)
    96 	{
    97 	iRedraws->Delete(aIndex,1);
    98 
    99 	//We are certain we will need iRedraws again, so it would be silly to compress away the last KGranularity slots.
   100 	const TInt count = iRedraws->Count();
   101 	if((count >= KGranularity) && (count % KGranularity == 0))
   102 		{
   103 		iRedraws->Compress();
   104 		}
   105 	}
   106 
   107 void CRedrawQueue::RemoveInvalid(CWsWindowRedraw *aRedrawWin)
   108 //
   109 // remove the window from the invalid window list.
   110 // harmless to call if the window is not in the list.
   111 //
   112 	{
   113 	TInt index;
   114 	TRedraw redraw;
   115 
   116 	redraw.iRedraw=aRedrawWin;
   117 	redraw.iPriority=0;
   118 	if ((iRedraws->Find(redraw,*iKeyWindow,index))==KErrNone)
   119 		DeleteFromQueue(index);
   120 	}
   121 
   122 TBool CRedrawQueue::TriggerRedraw()
   123 //
   124 // Trigger any pending redraw messages in the queue
   125 // Returns ETrue if a redraw was sent, EFalse if not.
   126 //
   127 	{
   128 	TBool ret=EFalse;
   129 	if (iRedrawTrigger)
   130 		{
   131 		iRedrawTrigger=EFalse;
   132 		if (!iEventMsg.IsNull() && (iRedraws->Count()>0 || iAllocError))
   133 			{
   134 			SignalEvent();
   135 			ret=ETrue;
   136 			}
   137 		}
   138 	return(ret);
   139 	}
   140 
   141 void CRedrawQueue::EventReady(const RMessagePtr2& aEventMsg)
   142 //
   143 // Queue a read of an event from the queue
   144 //
   145 	{
   146 	CEventBase::EventReady(aEventMsg);
   147 	iRedrawTrigger=ETrue;
   148 	TriggerRedraw();
   149 	}
   150 
   151 TBool CRedrawQueue::FindOutstandingRedrawEvent(CWsWindowRedraw& aRedraw, TWsRedrawEvent& aEvent)
   152 	{
   153 	TRect rect;
   154 	if (aRedraw.GetRedrawRect(rect))
   155 		{
   156 		aEvent.SetRect(rect);
   157 		aEvent.SetHandle(aRedraw.WsWin()->ClientHandle());
   158 		CEventBase::GetData(&aEvent,sizeof(aEvent));
   159 		return ETrue;
   160 		}
   161 	return EFalse;
   162 	}
   163 
   164 TBool CRedrawQueue::FindWindowNeedingRedrawEvent(TWsRedrawEvent& aEvent)
   165 	{
   166 #if defined(_DEBUG)
   167 	CWsWindowRedraw* previousRedraw = NULL;
   168 #endif
   169 	// search all screens
   170 	TInt invalidWindows = 0;
   171 	for (TInt screenNo = 0; screenNo < CWsTop::NumberOfScreens(); ++screenNo)
   172 		{
   173 		const CScreen* screen = CWsTop::Screen(screenNo);
   174 		WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
   175 		CWsRootWindow* rootWindow = screen->RootWindow();
   176 		for (CWsWindowGroup *groupWin = rootWindow->Child(); groupWin; groupWin = groupWin->NextSibling())
   177 			{
   178 			if (groupWin->WsOwner() == iWsOwner)
   179 				{
   180 				CWsWindowRedraw* redrawWin = NULL;
   181 				// use a window tree walk that can be resumed to find windows with an invalid region
   182 				TResumableWalkWindowTreeFindInvalid wwt(&redrawWin);
   183 				groupWin->WalkWindowTree(wwt, EWalkChildren, EFalse);
   184 				while (redrawWin != NULL)
   185 					{
   186 					WS_ASSERT_DEBUG(redrawWin != previousRedraw, EWsPanicRedrawQueueError);
   187 					// (the window may not actually need the client to redraw it, e.g. a CWsBlankWindow can redraw itself)
   188 					if (FindOutstandingRedrawEvent(*redrawWin, aEvent))
   189 						{
   190 						return ETrue;
   191 						}
   192 					else
   193 						{ // continue the Tree Walk
   194 #if defined(_DEBUG)
   195 						previousRedraw = redrawWin;
   196 #endif
   197 						if (redrawWin->NeedsRedraw())
   198 							{ // needs to be redrawn later?
   199 							++invalidWindows;
   200 							}
   201 						redrawWin = NULL;
   202 						groupWin->WalkWindowTree(wwt, EWalkChildren, ETrue);
   203 						}
   204 					}
   205 				}
   206 			}
   207 		}
   208 
   209 	if (invalidWindows == 0)
   210 		{ // error recovery is complete
   211 		iAllocError = 0;
   212 		}
   213 	return EFalse;
   214 	}
   215 
   216 void CRedrawQueue::GetData()
   217 //
   218 // If there is an outstanding redraw event in the queue, reply with it's data
   219 //
   220 	{
   221 	CWsWindowRedraw *redraw;
   222 	TWsRedrawEvent event;
   223 
   224 	while (iRedraws->Count()>0)
   225 		{
   226 		redraw=(*iRedraws)[0].iRedraw;
   227 		if (FindOutstandingRedrawEvent(*redraw,event))
   228 			{
   229 			return;
   230 			}
   231 		TInt toDelete=0;
   232 		if (redraw!=(*iRedraws)[0].iRedraw)
   233 			{		//In low memory conditions calls to FindOutstandingRedrawEvent can cause extra entries to be added to the array
   234 			TRedraw redrawFind;
   235 			redrawFind.iRedraw=redraw;
   236 			iRedraws->Find(redrawFind,*iKeyWindow,toDelete);
   237 			}
   238 		DeleteFromQueue(toDelete);
   239 		}
   240 
   241 	if (iAllocError && FindWindowNeedingRedrawEvent(event))
   242 		{
   243 		return;
   244 		}
   245 
   246 	CEventBase::GetData(&iNullRedrawEvent,sizeof(iNullRedrawEvent));
   247 	}
   248 
   249 TUint CRedrawQueue::RedrawPriority(CWsWindowRedraw *aRedrawWin)
   250 	{
   251 	TInt index;
   252 	TRedraw redraw;
   253 	TUint priority=0;
   254 	redraw.iRedraw=aRedrawWin;
   255 
   256 	if ((iRedraws->Find(redraw,*iKeyWindow,index))==KErrNone)
   257 		{
   258 		priority=iRedraws->At(index).iPriority;
   259 		}
   260 	return priority;
   261 	}