1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/REDRAWQ.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,261 @@
1.4 +// Copyright (c) 1996-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +// Window server redraw queue handling
1.20 +//
1.21 +#include <e32std.h>
1.22 +#include "server.h"
1.23 +#include "rootwin.h"
1.24 +#include "windowgroup.h"
1.25 +#include "walkwindowtree.h"
1.26 +#include "EVQUEUE.H"
1.27 +#include "panics.h"
1.28 +#include "wstop.h"
1.29 +
1.30 +const TInt KGranularity = 10;
1.31 +
1.32 +void TWsRedrawEvent::SetHandle(TUint aHandle)
1.33 + {
1.34 + iHandle=aHandle;
1.35 + }
1.36 +
1.37 +void TWsRedrawEvent::SetRect(TRect aRect)
1.38 + {
1.39 + iRect=aRect;
1.40 + }
1.41 +
1.42 +TWsRedrawEvent CRedrawQueue::iNullRedrawEvent;
1.43 +
1.44 +CRedrawQueue::CRedrawQueue(CWsClient *aOwner) : CEventBase(aOwner)
1.45 + {
1.46 + __DECLARE_NAME(_S("CRedrawQueue"));
1.47 + }
1.48 +
1.49 +void CRedrawQueue::ConstructL()
1.50 + {
1.51 + iRedrawTrigger=EFalse;
1.52 + iRedraws=new(ELeave) CArrayFixSeg<TRedraw>(KGranularity);
1.53 + iKeyPriority=new(ELeave) TKeyArrayFix(_FOFF(TRedraw,iPriority),ECmpTUint32);
1.54 + iKeyWindow=new(ELeave) TKeyArrayFix(_FOFF(TRedraw,iRedraw),ECmpTUint32);
1.55 + Mem::FillZ(&iNullRedrawEvent,sizeof(iNullRedrawEvent));
1.56 + }
1.57 +
1.58 +CRedrawQueue::~CRedrawQueue()
1.59 + {
1.60 + delete iRedraws;
1.61 + delete iKeyPriority;
1.62 + delete iKeyWindow;
1.63 + }
1.64 +
1.65 +void CRedrawQueue::ReCalcOrder()
1.66 + {
1.67 + const TInt redrawsCount=iRedraws->Count();
1.68 + for(TInt index=0;index<redrawsCount;index++)
1.69 + {
1.70 + TRedraw *redraw=&iRedraws->At(index);
1.71 + redraw->iPriority=static_cast<CWsClientWindow*>(redraw->iRedraw->WsWin())->RedrawPriority();
1.72 + }
1.73 + iRedraws->Sort(*iKeyPriority);
1.74 + }
1.75 +
1.76 +void CRedrawQueue::AddInvalid(CWsWindowRedraw *aRedrawWin)
1.77 +//
1.78 +// Add a window to the update list.
1.79 +//
1.80 + {
1.81 + TInt index;
1.82 + TRedraw redraw;
1.83 +
1.84 + redraw.iRedraw=aRedrawWin;
1.85 + if (iRedraws->Find(redraw,*iKeyWindow,index)!=0)
1.86 + {
1.87 + redraw.iPriority=static_cast<CWsClientWindow*>(aRedrawWin->WsWin())->RedrawPriority();
1.88 + TRAPD(err,iRedraws->InsertIsqAllowDuplicatesL(redraw,*iKeyPriority));
1.89 + if (err!=KErrNone)
1.90 + {
1.91 + WS_ASSERT_DEBUG(err==KErrNoMemory,EWsPanicRedrawQueueError);
1.92 + iAllocError=ETrue;
1.93 + }
1.94 + iRedrawTrigger=ETrue;
1.95 + }
1.96 + }
1.97 +
1.98 +void CRedrawQueue::DeleteFromQueue(TInt aIndex)
1.99 + {
1.100 + iRedraws->Delete(aIndex,1);
1.101 +
1.102 + //We are certain we will need iRedraws again, so it would be silly to compress away the last KGranularity slots.
1.103 + const TInt count = iRedraws->Count();
1.104 + if((count >= KGranularity) && (count % KGranularity == 0))
1.105 + {
1.106 + iRedraws->Compress();
1.107 + }
1.108 + }
1.109 +
1.110 +void CRedrawQueue::RemoveInvalid(CWsWindowRedraw *aRedrawWin)
1.111 +//
1.112 +// remove the window from the invalid window list.
1.113 +// harmless to call if the window is not in the list.
1.114 +//
1.115 + {
1.116 + TInt index;
1.117 + TRedraw redraw;
1.118 +
1.119 + redraw.iRedraw=aRedrawWin;
1.120 + redraw.iPriority=0;
1.121 + if ((iRedraws->Find(redraw,*iKeyWindow,index))==KErrNone)
1.122 + DeleteFromQueue(index);
1.123 + }
1.124 +
1.125 +TBool CRedrawQueue::TriggerRedraw()
1.126 +//
1.127 +// Trigger any pending redraw messages in the queue
1.128 +// Returns ETrue if a redraw was sent, EFalse if not.
1.129 +//
1.130 + {
1.131 + TBool ret=EFalse;
1.132 + if (iRedrawTrigger)
1.133 + {
1.134 + iRedrawTrigger=EFalse;
1.135 + if (!iEventMsg.IsNull() && (iRedraws->Count()>0 || iAllocError))
1.136 + {
1.137 + SignalEvent();
1.138 + ret=ETrue;
1.139 + }
1.140 + }
1.141 + return(ret);
1.142 + }
1.143 +
1.144 +void CRedrawQueue::EventReady(const RMessagePtr2& aEventMsg)
1.145 +//
1.146 +// Queue a read of an event from the queue
1.147 +//
1.148 + {
1.149 + CEventBase::EventReady(aEventMsg);
1.150 + iRedrawTrigger=ETrue;
1.151 + TriggerRedraw();
1.152 + }
1.153 +
1.154 +TBool CRedrawQueue::FindOutstandingRedrawEvent(CWsWindowRedraw& aRedraw, TWsRedrawEvent& aEvent)
1.155 + {
1.156 + TRect rect;
1.157 + if (aRedraw.GetRedrawRect(rect))
1.158 + {
1.159 + aEvent.SetRect(rect);
1.160 + aEvent.SetHandle(aRedraw.WsWin()->ClientHandle());
1.161 + CEventBase::GetData(&aEvent,sizeof(aEvent));
1.162 + return ETrue;
1.163 + }
1.164 + return EFalse;
1.165 + }
1.166 +
1.167 +TBool CRedrawQueue::FindWindowNeedingRedrawEvent(TWsRedrawEvent& aEvent)
1.168 + {
1.169 +#if defined(_DEBUG)
1.170 + CWsWindowRedraw* previousRedraw = NULL;
1.171 +#endif
1.172 + // search all screens
1.173 + TInt invalidWindows = 0;
1.174 + for (TInt screenNo = 0; screenNo < CWsTop::NumberOfScreens(); ++screenNo)
1.175 + {
1.176 + const CScreen* screen = CWsTop::Screen(screenNo);
1.177 + WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
1.178 + CWsRootWindow* rootWindow = screen->RootWindow();
1.179 + for (CWsWindowGroup *groupWin = rootWindow->Child(); groupWin; groupWin = groupWin->NextSibling())
1.180 + {
1.181 + if (groupWin->WsOwner() == iWsOwner)
1.182 + {
1.183 + CWsWindowRedraw* redrawWin = NULL;
1.184 + // use a window tree walk that can be resumed to find windows with an invalid region
1.185 + TResumableWalkWindowTreeFindInvalid wwt(&redrawWin);
1.186 + groupWin->WalkWindowTree(wwt, EWalkChildren, EFalse);
1.187 + while (redrawWin != NULL)
1.188 + {
1.189 + WS_ASSERT_DEBUG(redrawWin != previousRedraw, EWsPanicRedrawQueueError);
1.190 + // (the window may not actually need the client to redraw it, e.g. a CWsBlankWindow can redraw itself)
1.191 + if (FindOutstandingRedrawEvent(*redrawWin, aEvent))
1.192 + {
1.193 + return ETrue;
1.194 + }
1.195 + else
1.196 + { // continue the Tree Walk
1.197 +#if defined(_DEBUG)
1.198 + previousRedraw = redrawWin;
1.199 +#endif
1.200 + if (redrawWin->NeedsRedraw())
1.201 + { // needs to be redrawn later?
1.202 + ++invalidWindows;
1.203 + }
1.204 + redrawWin = NULL;
1.205 + groupWin->WalkWindowTree(wwt, EWalkChildren, ETrue);
1.206 + }
1.207 + }
1.208 + }
1.209 + }
1.210 + }
1.211 +
1.212 + if (invalidWindows == 0)
1.213 + { // error recovery is complete
1.214 + iAllocError = 0;
1.215 + }
1.216 + return EFalse;
1.217 + }
1.218 +
1.219 +void CRedrawQueue::GetData()
1.220 +//
1.221 +// If there is an outstanding redraw event in the queue, reply with it's data
1.222 +//
1.223 + {
1.224 + CWsWindowRedraw *redraw;
1.225 + TWsRedrawEvent event;
1.226 +
1.227 + while (iRedraws->Count()>0)
1.228 + {
1.229 + redraw=(*iRedraws)[0].iRedraw;
1.230 + if (FindOutstandingRedrawEvent(*redraw,event))
1.231 + {
1.232 + return;
1.233 + }
1.234 + TInt toDelete=0;
1.235 + if (redraw!=(*iRedraws)[0].iRedraw)
1.236 + { //In low memory conditions calls to FindOutstandingRedrawEvent can cause extra entries to be added to the array
1.237 + TRedraw redrawFind;
1.238 + redrawFind.iRedraw=redraw;
1.239 + iRedraws->Find(redrawFind,*iKeyWindow,toDelete);
1.240 + }
1.241 + DeleteFromQueue(toDelete);
1.242 + }
1.243 +
1.244 + if (iAllocError && FindWindowNeedingRedrawEvent(event))
1.245 + {
1.246 + return;
1.247 + }
1.248 +
1.249 + CEventBase::GetData(&iNullRedrawEvent,sizeof(iNullRedrawEvent));
1.250 + }
1.251 +
1.252 +TUint CRedrawQueue::RedrawPriority(CWsWindowRedraw *aRedrawWin)
1.253 + {
1.254 + TInt index;
1.255 + TRedraw redraw;
1.256 + TUint priority=0;
1.257 + redraw.iRedraw=aRedrawWin;
1.258 +
1.259 + if ((iRedraws->Find(redraw,*iKeyWindow,index))==KErrNone)
1.260 + {
1.261 + priority=iRedraws->At(index).iPriority;
1.262 + }
1.263 + return priority;
1.264 + }