1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/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 +#include "wstraces.h"
1.30 +
1.31 +const TInt KGranularity = 10;
1.32 +
1.33 +void TWsRedrawEvent::SetHandle(TUint aHandle)
1.34 + {
1.35 + iHandle=aHandle;
1.36 + }
1.37 +
1.38 +void TWsRedrawEvent::SetRect(TRect aRect)
1.39 + {
1.40 + iRect=aRect;
1.41 + }
1.42 +
1.43 +TWsRedrawEvent CRedrawQueue::iNullRedrawEvent;
1.44 +
1.45 +CRedrawQueue::CRedrawQueue(CWsClient *aOwner) : CEventBase(aOwner)
1.46 + {
1.47 + __DECLARE_NAME(_S("CRedrawQueue"));
1.48 + }
1.49 +
1.50 +void CRedrawQueue::ConstructL()
1.51 + {
1.52 + iRedrawTrigger=EFalse;
1.53 + iRedraws=new(ELeave) CArrayFixSeg<TRedraw>(KGranularity);
1.54 + iKeyPriority=new(ELeave) TKeyArrayFix(_FOFF(TRedraw,iPriority),ECmpTUint32);
1.55 + iKeyWindow=new(ELeave) TKeyArrayFix(_FOFF(TRedraw,iRedraw),ECmpTUint32);
1.56 + Mem::FillZ(&iNullRedrawEvent,sizeof(iNullRedrawEvent));
1.57 + }
1.58 +
1.59 +CRedrawQueue::~CRedrawQueue()
1.60 + {
1.61 + delete iRedraws;
1.62 + delete iKeyPriority;
1.63 + delete iKeyWindow;
1.64 + }
1.65 +
1.66 +void CRedrawQueue::ReCalcOrder()
1.67 + {
1.68 + const TInt redrawsCount=iRedraws->Count();
1.69 + for(TInt index=0;index<redrawsCount;index++)
1.70 + {
1.71 + TRedraw *redraw=&iRedraws->At(index);
1.72 + redraw->iPriority=static_cast<CWsClientWindow*>(redraw->iRedraw->WsWin())->RedrawPriority();
1.73 + }
1.74 + iRedraws->Sort(*iKeyPriority);
1.75 + }
1.76 +
1.77 +void CRedrawQueue::AddInvalid(CWsWindowRedraw *aRedrawWin)
1.78 +//
1.79 +// Add a window to the update list.
1.80 +//
1.81 + {
1.82 + TInt index;
1.83 + TRedraw redraw;
1.84 +
1.85 + redraw.iRedraw=aRedrawWin;
1.86 + if (iRedraws->Find(redraw,*iKeyWindow,index)!=0)
1.87 + {
1.88 + redraw.iPriority=static_cast<CWsClientWindow*>(aRedrawWin->WsWin())->RedrawPriority();
1.89 + TRAPD(err,iRedraws->InsertIsqAllowDuplicatesL(redraw,*iKeyPriority));
1.90 + if (err!=KErrNone)
1.91 + {
1.92 + WS_ASSERT_DEBUG(err==KErrNoMemory,EWsPanicRedrawQueueError);
1.93 + iAllocError=ETrue;
1.94 + }
1.95 + iRedrawTrigger=ETrue;
1.96 + }
1.97 + }
1.98 +
1.99 +void CRedrawQueue::DeleteFromQueue(TInt aIndex)
1.100 + {
1.101 + iRedraws->Delete(aIndex,1);
1.102 +
1.103 + //We are certain we will need iRedraws again, so it would be silly to compress away the last KGranularity slots.
1.104 + const TInt count = iRedraws->Count();
1.105 + if((count >= KGranularity) && (count % KGranularity == 0))
1.106 + {
1.107 + iRedraws->Compress();
1.108 + }
1.109 + }
1.110 +
1.111 +void CRedrawQueue::RemoveInvalid(CWsWindowRedraw *aRedrawWin)
1.112 +//
1.113 +// remove the window from the invalid window list.
1.114 +// harmless to call if the window is not in the list.
1.115 +//
1.116 + {
1.117 + TInt index;
1.118 + TRedraw redraw;
1.119 +
1.120 + redraw.iRedraw=aRedrawWin;
1.121 + redraw.iPriority=0;
1.122 + if ((iRedraws->Find(redraw,*iKeyWindow,index))==KErrNone)
1.123 + DeleteFromQueue(index);
1.124 + }
1.125 +
1.126 +TBool CRedrawQueue::TriggerRedraw()
1.127 +//
1.128 +// Trigger any pending redraw messages in the queue
1.129 +// Returns ETrue if a redraw was sent, EFalse if not.
1.130 +//
1.131 + {
1.132 + WS_TRACE_SERVER_TRIGGERREDRAW();
1.133 + TBool ret=EFalse;
1.134 + if (iRedrawTrigger)
1.135 + {
1.136 + iRedrawTrigger=EFalse;
1.137 + if (!iEventMsg.IsNull() && (iRedraws->Count()>0 || iAllocError))
1.138 + {
1.139 + SignalEvent();
1.140 + ret=ETrue;
1.141 + }
1.142 + }
1.143 + return(ret);
1.144 + }
1.145 +
1.146 +void CRedrawQueue::EventReady(const RMessagePtr2& aEventMsg)
1.147 +//
1.148 +// Queue a read of an event from the queue
1.149 +//
1.150 + {
1.151 + CEventBase::EventReady(aEventMsg);
1.152 + iRedrawTrigger=ETrue;
1.153 + TriggerRedraw();
1.154 + }
1.155 +
1.156 +TBool CRedrawQueue::FindOutstandingRedrawEvent(CWsWindowRedraw& aRedraw, TWsRedrawEvent& aEvent)
1.157 + {
1.158 + TRect rect;
1.159 + if (aRedraw.GetRedrawRect(rect))
1.160 + {
1.161 + aEvent.SetRect(rect);
1.162 + aEvent.SetHandle(aRedraw.WsWin()->ClientHandle());
1.163 + CEventBase::GetData(&aEvent,sizeof(aEvent));
1.164 + return ETrue;
1.165 + }
1.166 + return EFalse;
1.167 + }
1.168 +
1.169 +TBool CRedrawQueue::FindWindowNeedingRedrawEvent(TWsRedrawEvent& aEvent)
1.170 + {
1.171 +#if defined(_DEBUG)
1.172 + CWsWindowRedraw* previousRedraw = NULL;
1.173 +#endif
1.174 + // search all screens
1.175 + TInt invalidWindows = 0;
1.176 + for (TInt screenNo = 0; screenNo < CWsTop::NumberOfScreens(); ++screenNo)
1.177 + {
1.178 + CWsRootWindow* rootWindow = CWsTop::Screen(screenNo)->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 + }