1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/SERVER/redrawmsgwindow.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1707 @@
1.4 +// Copyright (c) 1995-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 +// Window redraw code, three sorts of redrawing are supported
1.18 +// CRedrawMsgWindow handles it via sending a redraw message to the client
1.19 +//
1.20 +//
1.21 +
1.22 +#include "redrawmsgwindow.h"
1.23 +#include "gc.h"
1.24 +#include "playbackgc.h"
1.25 +#include "inifile.h"
1.26 +#include "rootwin.h"
1.27 +#include "wstop.h"
1.28 +#include "ANIM.H"
1.29 +#include "EVQUEUE.H"
1.30 +#include <s32mem.h>
1.31 +#include <gdi.h>
1.32 +#include "panics.h"
1.33 +#include "rootwin.h"
1.34 +#include "EVENT.H"
1.35 +#include "wsfont.h"
1.36 +#include <graphics/wsgraphicdrawerinterface.h>
1.37 +#include "../debuglog/DEBUGLOG.H"
1.38 +
1.39 +const TUint KDrawBufferGranularity = 240;
1.40 +const TInt KRedrawRegionGranularity = 8;
1.41 +const TInt KReadBufferMaxLen=0x100;
1.42 +const TInt KRegionCompressThreshold = 6; // number of rectangles in region at which we try to compress it
1.43 +
1.44 +/** Max number of non-redraw segments allowed before starting to throw away the oldest */
1.45 +const TInt KNonRedrawSegMaxLimit = 16;
1.46 +/** The number of non-redraw segments to spare from deletion once their number
1.47 + have grown beyond KNonRedrawSegMaxLimit */
1.48 +const TInt KNonRedrawSegThreshold = 8;
1.49 +
1.50 +TInt CWsRedrawMsgWindow::iNonRedrawAgeLimit = 0;
1.51 +CWsRedrawMsgWindow::TAtomicityType CWsRedrawMsgWindow::iAtomicity = ENoAtomicity;
1.52 +
1.53 +#if defined(__WINS__) && defined(_DEBUG)
1.54 +# include "offscreenbitmap.h"
1.55 +# define DEBUGOSB { CWsOffScreenBitmap * ofb = Screen()->OffScreenBitmap(); if (ofb) ofb->Update(); }
1.56 +#else
1.57 +# define DEBUGOSB
1.58 +#endif
1.59 +
1.60 +#ifndef _DEBUG
1.61 +
1.62 +#define LOG_WINDOW_REDRAW_START(wswin,region)
1.63 +#define LOG_WINDOW_REDRAW_END(wswin)
1.64 +#define LOG_REDRAW_SEGMENT(segmentIndex,segmentType)
1.65 +#define LOG_REDRAW_SEGMENT_REGION(region)
1.66 +#define LOG_PLAYBACK_GC_COMMAND(opcode,data)
1.67 +
1.68 +#else
1.69 +
1.70 +#define LOG_WINDOW_REDRAW_START(wswin,region) LogDrawCommandsStart(wswin,region)
1.71 +#define LOG_WINDOW_REDRAW_END(wswin) LogDrawCommandsEnd(wswin)
1.72 +#define LOG_REDRAW_SEGMENT(segmentIndex,segmentType) LogRedrawSegment(segmentIndex, segmentType)
1.73 +#define LOG_REDRAW_SEGMENT_REGION(region) {if(wsDebugLog){ LogRegion(region);}}
1.74 +#define LOG_PLAYBACK_GC_COMMAND(opcode,data) {if (wsDebugLog) {wsDebugLog->Command(WS_HANDLE_GC, opcode, data, NULL);}}
1.75 +
1.76 +extern CDebugLogBase *wsDebugLog;
1.77 +
1.78 +class TTruncateOverflow : public TDesOverflow
1.79 + {
1.80 + public:
1.81 + virtual void Overflow(TDes&) {};
1.82 + };
1.83 +
1.84 +LOCAL_C void LogRedrawSegment(TUint aSegmentIndex, CWsRedrawMsgWindow::TRedrawSegmentType aSegmentType)
1.85 + {
1.86 + if (wsDebugLog)
1.87 + {
1.88 + TBuf<LogTBufSize> log;
1.89 + TTruncateOverflow overflow;
1.90 + _LIT(KLogRedrawSegment, ">> CRedrawSegment[%d] ");
1.91 + log.AppendFormat(KLogRedrawSegment, &overflow, aSegmentIndex);
1.92 + _LIT(KLogRedrawSegmentPending, "Pending");
1.93 + _LIT(KLogRedrawSegmentRedraw, "Redraw");
1.94 + _LIT(KLogRedrawSegmentNonRedraw, "NonRedraw");
1.95 + switch(aSegmentType)
1.96 + {
1.97 + case CWsRedrawMsgWindow::ESegmentTypePendingRedraw :
1.98 + log.AppendFormat(KLogRedrawSegmentPending, &overflow);
1.99 + break;
1.100 + case CWsRedrawMsgWindow::ESegmentTypeRedraw :
1.101 + log.AppendFormat(KLogRedrawSegmentRedraw, &overflow);
1.102 + break;
1.103 + case CWsRedrawMsgWindow::ESegmentTypeNonRedraw :
1.104 + log.AppendFormat(KLogRedrawSegmentNonRedraw, &overflow);
1.105 + break;
1.106 + default :
1.107 + {
1.108 + }
1.109 + }
1.110 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.111 + }
1.112 + }
1.113 +
1.114 +LOCAL_C void LogRegion(const TRegion* aRegion)
1.115 + {
1.116 + TBuf<LogTBufSize> log;
1.117 + TTruncateOverflow overflow;
1.118 + TInt rectCount = (aRegion == NULL ? 0 : aRegion->Count());
1.119 + log.AppendFormat(_L("region [%d,"), &overflow, rectCount);
1.120 + if (rectCount > 0)
1.121 + {
1.122 + const TRect* rectangles = aRegion->RectangleList();
1.123 + TBuf<1> comma;
1.124 + for (TInt ii = 0; ii < rectCount; ii++)
1.125 + {
1.126 + TRect current = rectangles[ii];
1.127 + log.AppendFormat(_L("%S{{%d,%d},{%d,%d}}"), &overflow, &comma,
1.128 + current.iTl.iX,current.iTl.iY,current.iBr.iX,current.iBr.iY);
1.129 + comma = _L(",");
1.130 + }
1.131 + }
1.132 + else
1.133 + {
1.134 + log.AppendFormat(_L("NULL"), &overflow);
1.135 + }
1.136 + log.AppendFormat(_L("]"), &overflow);
1.137 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.138 + }
1.139 +
1.140 +LOCAL_C void LogDrawCommandsStart(const CWsWindow* aWsWin, const TRegion* aRegion)
1.141 + {
1.142 + if (wsDebugLog)
1.143 + {
1.144 + _LIT(KLogDrawCommandsStart, ">> CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
1.145 + const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
1.146 + TBuf<LogTBufSize> log;
1.147 + TTruncateOverflow overflow;
1.148 + log.AppendFormat(KLogDrawCommandsStart, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
1.149 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.150 + LogRegion(aRegion);
1.151 + }
1.152 + }
1.153 +
1.154 +LOCAL_C void LogDrawCommandsEnd(const CWsWindow* aWsWin)
1.155 + {
1.156 + if (wsDebugLog)
1.157 + {
1.158 + _LIT(KLogDrawCommandsEnd, "<< CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
1.159 + const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
1.160 + TBuf<LogTBufSize> log;
1.161 + TTruncateOverflow overflow;
1.162 + log.AppendFormat(KLogDrawCommandsEnd, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
1.163 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.164 + }
1.165 + }
1.166 +
1.167 +#endif
1.168 +
1.169 +//
1.170 +// Redraw windows //
1.171 +//
1.172 +
1.173 +CWsRedrawMsgWindow::CRedrawSegment::CRedrawSegment()
1.174 + {
1.175 + }
1.176 +
1.177 +CWsRedrawMsgWindow::CRedrawSegment* CWsRedrawMsgWindow::CRedrawSegment::NewLC(const TRect& aRect, TRedrawSegmentType aNewRegionType)
1.178 + {
1.179 + CRedrawSegment* self = new (ELeave) CRedrawSegment();
1.180 + CleanupStack::PushL(self);
1.181 + self->ConstructL(aRect, aNewRegionType);
1.182 + return self;
1.183 + }
1.184 +
1.185 +void CWsRedrawMsgWindow::StaticInitL()
1.186 + {
1.187 + _LIT(KNonRedrawAgeLimit, "NONREDRAWAGELIMIT");
1.188 + const TInt KDefaultNonRedrawAgeLimit = 1000000;
1.189 + if (!WsIniFile->FindVar(KNonRedrawAgeLimit, iNonRedrawAgeLimit))
1.190 + iNonRedrawAgeLimit = KDefaultNonRedrawAgeLimit;
1.191 +
1.192 + _LIT(KAtomicRedraws,"ATOMICREDRAWS");
1.193 + _LIT(KAtomicSegment,"SEGMENT");
1.194 + _LIT(KAtomicWindow,"WINDOW");
1.195 +
1.196 + TPtrC atomicityTypeString;
1.197 + if (WsIniFile->FindVar(KAtomicRedraws,atomicityTypeString))
1.198 + {
1.199 + if(atomicityTypeString.CompareF(KAtomicSegment)==0 || atomicityTypeString.Length()==0)
1.200 + iAtomicity = ESegment;
1.201 + else if(atomicityTypeString.CompareF(KAtomicWindow)==0)
1.202 + iAtomicity = EWindow;
1.203 + }
1.204 + }
1.205 +
1.206 +void CWsRedrawMsgWindow::CRedrawSegment::ReleaseFontsAndBitmaps()
1.207 + {
1.208 + // release Bitmap and Font handles
1.209 + TInt count = iWsBitmapArray.Count();
1.210 + TInt ii;
1.211 + for (ii = count - 1; ii >= 0; ii--)
1.212 + {
1.213 + iWsBitmapArray[ii]->DecRefCount();
1.214 + iWsBitmapArray.Remove(ii);
1.215 + }
1.216 +
1.217 + count = iWsFontArray.Count();
1.218 + for (ii = count - 1; ii >= 0; --ii)
1.219 + {
1.220 + CWsFontCache::Instance()->ReleaseFont(iWsFontArray[ii]);
1.221 + iWsFontArray.Remove(ii);
1.222 + }
1.223 +
1.224 + count = iFbsBitmapArray.Count();
1.225 + for(ii = count - 1 ; ii >= 0; ii--)
1.226 + {
1.227 + delete iFbsBitmapArray[ii];
1.228 + iFbsBitmapArray.Remove(ii);
1.229 + }
1.230 + }
1.231 +
1.232 +/* Set new rectangle and region type for initial or reset redraw region
1.233 + @leave KErrNoMemory no memory to update region details
1.234 + */
1.235 +void CWsRedrawMsgWindow::CRedrawSegment::ConstructL(const TRect& aRect, TRedrawSegmentType aNewRegionType)
1.236 + {
1.237 + iDrawCommands = CBufSeg::NewL(KDrawBufferGranularity);
1.238 + iCreationTime.UniversalTime();
1.239 +
1.240 + iRegion.AddRect(aRect);
1.241 + if (iRegion.CheckError())
1.242 + {
1.243 + User::Leave(KErrNoMemory);
1.244 + }
1.245 + iRedrawSegmentType = aNewRegionType;
1.246 + }
1.247 +
1.248 +CWsRedrawMsgWindow::CRedrawSegment::~CRedrawSegment()
1.249 + {
1.250 + delete iDrawCommands;
1.251 +
1.252 + iRegion.Close();
1.253 +
1.254 + // Release Font and Bitmap handles, close arrays
1.255 + ReleaseFontsAndBitmaps();
1.256 + iFbsBitmapArray.Close();
1.257 + iWsBitmapArray.Close();
1.258 + iWsFontArray.Close();
1.259 + iDrawerArray.Close();
1.260 + }
1.261 +
1.262 +void CWsRedrawMsgWindow::WindowClosing()
1.263 + {
1.264 + iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
1.265 + }
1.266 +
1.267 +
1.268 +TInt CWsRedrawMsgWindow::CRedrawSegment::SizeInBytes() const
1.269 + {
1.270 + TInt size = sizeof(CWsRedrawMsgWindow::CRedrawSegment);
1.271 + size += iDrawCommands->Size();
1.272 + size += iFbsBitmapArray.Count() * sizeof(CFbsBitmap);
1.273 + size += iWsBitmapArray.Count() * sizeof(DWsBitmap);
1.274 + size += iWsFontArray.Count() * sizeof(CWsFbsFont);
1.275 + size += iDrawerArray.Count() * sizeof(TGraphicDrawerId);
1.276 + return size;
1.277 + }
1.278 +
1.279 +CWsRedrawMsgWindow::CWsRedrawMsgWindow(CWsWindow *aWin)
1.280 + :CWsWindowRedraw(aWin), iBackColor(aWin->RootWindow()->DefaultBackgroundColor()), iFlags(EBackgroundClear|EStoringEntireWindow),
1.281 + iRedrawSegments(KRedrawRegionGranularity),
1.282 + iCurrentSegment(0),
1.283 + iOSBStatus(ETrue)
1.284 + {
1.285 + }
1.286 +
1.287 +void CWsRedrawMsgWindow::ConstructL()
1.288 + {
1.289 + CWsWindowRedraw::ConstructL();
1.290 + Invalidate(&WsWin()->Rel());
1.291 + iWsWin->WsOwner()->RedrawQueue()->ReCalcOrder();
1.292 + }
1.293 +
1.294 +CWsRedrawMsgWindow::~CWsRedrawMsgWindow()
1.295 + {
1.296 + WS_ASSERT_DEBUG(iMemoryLock == 0, EWsPanicMemoryLock);
1.297 + RemoveFromRedrawQueueIfEmpty();
1.298 + iInvalid.Close();
1.299 + iLocalRedrawRegion.Close();
1.300 + iRedrawSegments.ResetAndDestroy();
1.301 + iCurrentSegment=NULL;
1.302 + }
1.303 +
1.304 +/**
1.305 +These three functions actually check for a value they have already asserted on. This is intentional.
1.306 +*/
1.307 +void CWsRedrawMsgWindow::ExpandCommandBufferL(TInt aLength)
1.308 + {
1.309 + WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
1.310 +
1.311 + if (iCurrentSegment)
1.312 + {
1.313 + // need more space?
1.314 + if (iCurrentSegment->iCurrentCommandBufferWritePos + aLength > iCurrentSegment->iDrawCommands->Size())
1.315 + {
1.316 + iCurrentSegment->iDrawCommands->ResizeL(iCurrentSegment->iCurrentCommandBufferWritePos + aLength);
1.317 + }
1.318 + }
1.319 + }
1.320 +
1.321 +void CWsRedrawMsgWindow::CommandBufferWrite(const TDesC8& aDes, TInt aLength)
1.322 + {
1.323 + WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
1.324 + WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
1.325 + if (iCurrentSegment)
1.326 + {
1.327 + iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aDes, aLength);
1.328 + iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
1.329 + }
1.330 + }
1.331 +
1.332 +void CWsRedrawMsgWindow::CommandBufferWrite(const TAny* aPtr,TInt aLength)
1.333 + {
1.334 + WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
1.335 + WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
1.336 + if (iCurrentSegment)
1.337 + {
1.338 + iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aPtr, aLength);
1.339 + iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
1.340 + }
1.341 + }
1.342 +
1.343 +/*------------------------------------------------------------------------------
1.344 + Description: Processes draw commands. These are received as opcodes.
1.345 + -----------------------------------------------------------------------------*/
1.346 +TBool CWsRedrawMsgWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
1.347 + {
1.348 + switch(aOpcode)
1.349 + {
1.350 + case EWsWinOpEnableOSB:
1.351 + iOSBStatus = ETrue;
1.352 + break;
1.353 + case EWsWinOpDisableOSB:
1.354 + iOSBStatus = EFalse;
1.355 + break;
1.356 + case EWsWinOpSetBackgroundColor:
1.357 + iBackColor = *aCmd.rgb;
1.358 + iFlags |= EBackgroundClear;
1.359 + break;
1.360 + case EWsWinOpSetNoBackgroundColor:
1.361 + iFlags &= ~EBackgroundClear;
1.362 + break;
1.363 + case EWsWinOpInvalidate:
1.364 + Invalidate(aCmd.rect);
1.365 + break;
1.366 + case EWsWinOpInvalidateFull:
1.367 + Invalidate();
1.368 + break;
1.369 + case EWsWinOpBeginRedraw:
1.370 + BeginRedraw(aCmd.rect);
1.371 + ValidateRect(aCmd.rect);
1.372 + break;
1.373 + case EWsWinOpBeginRedrawFull:
1.374 + BeginRedraw(NULL);
1.375 + ValidateRect(NULL);
1.376 + break;
1.377 + case EWsWinOpEndRedraw:
1.378 + EndRedraw();
1.379 + break;
1.380 + case EWsWinOpGetInvalidRegionCount:
1.381 + {
1.382 + SetReply(iInvalid.Count());
1.383 + }
1.384 + break;
1.385 + case EWsWinOpGetInvalidRegion:
1.386 + {
1.387 + if ((*aCmd.Int) <= 0)
1.388 + OwnerPanic(EWservPanicInvalidRegionCount);
1.389 + if ((!iInvalid.CheckError()) && iInvalid.Count() == (*aCmd.Int))
1.390 + {
1.391 + CWsClient::ReplyBuf(iInvalid.RectangleList(),(*aCmd.Int) * sizeof(TRect));
1.392 + SetReply(EFalse);
1.393 + }
1.394 + else
1.395 + SetReply(ETrue);
1.396 + }
1.397 + break;
1.398 + case EWsWinOpStoreDrawCommands:
1.399 + /* If the client asks us not to store commands, we still store the commands
1.400 + for the region of the window which can be seen through the parent, but
1.401 + won't attempt to obtain the entire window.
1.402 + */
1.403 + if (*aCmd.Bool)
1.404 + {
1.405 + SetScope(EStoreEntireWindow);
1.406 + }
1.407 + else
1.408 + {
1.409 + // Clients that turn their redraw store off will still get one,
1.410 + // but it will only attempt to store the current viewport.
1.411 + SetScope(EStoreViewport);
1.412 + }
1.413 + break;
1.414 + case EWsWinOpHandleTransparencyUpdate: // deprecated
1.415 + case EWsWinOpSetTransparencyBitmap: // deprecated
1.416 + case EWsWinOpSetTransparencyFactor: // deprecated
1.417 + case EWsWinOpSetTransparencyBitmapCWs: // deprecated
1.418 + break; // do nothing.
1.419 + case EWsWinOpIsRedrawStoreEnabled:
1.420 + SetReply(ETrue);
1.421 + break;
1.422 + case EWsWinOpClearRedrawStore:
1.423 + DiscardStoredCommands();
1.424 + break;
1.425 + default:
1.426 + return(EFalse);
1.427 + }
1.428 + return(ETrue);
1.429 + }
1.430 +
1.431 +/**
1.432 +*/
1.433 +void CWsRedrawMsgWindow::BeginRedraw(const TRect* aRect)
1.434 + {
1.435 + if(InRedraw())
1.436 + OwnerPanic(EWservPanicDrawCommandsInvalidState);
1.437 + iFlags|=EBeginEndRedraw;
1.438 + TRAPD(err,DoBeginRedrawL(aRect));
1.439 + DiscardStoredCommandsIfError(err);
1.440 + }
1.441 +
1.442 +void CWsRedrawMsgWindow::DoBeginRedrawL(const TRect* aRect)
1.443 + {
1.444 + const TRect redrawRect = (aRect ? *aRect : TRect(WsWin()->Size()));
1.445 + if (redrawRect.IsEmpty())
1.446 + {
1.447 + //Skip empty rects since they are not added to the region
1.448 + iCurrentSegment = NULL;
1.449 + }
1.450 + else
1.451 + {
1.452 + CreateNewSegmentL(redrawRect, CWsRedrawMsgWindow::ESegmentTypePendingRedraw);
1.453 + if (iAtomicity==ENoAtomicity)
1.454 + PromoteLastPendingSegment();
1.455 + }
1.456 + }
1.457 +
1.458 +void CWsRedrawMsgWindow::Invalidate(const TRect * aRect)
1.459 + {
1.460 + //The memory allocation in this function can trigger a call to ReleaseMemory(), which would
1.461 + //recursively call this function again. This would cause a memory leak. To avoid this
1.462 + //we call Lock() to block ReleaseMemory() from this object while executing this function.
1.463 + Lock();
1.464 + if (!aRect)
1.465 + {
1.466 + iInvalid.Clear();
1.467 + iInvalid.Copy(iWsWin->WindowArea());
1.468 + iInvalid.Offset(-iWsWin->Origin());
1.469 + }
1.470 + else if((!aRect->IsEmpty()) && aRect->IsNormalized())
1.471 + {
1.472 + iInvalid.AddRect(*aRect);
1.473 + iInvalid.Tidy();
1.474 + }
1.475 + if (iWsWin->IsVisible())
1.476 + {
1.477 + QueueRedraw();
1.478 + iWsWin->WsOwner()->TriggerRedraw(); //wtf isn't the redrawq already scheduling itself?
1.479 + }
1.480 + Unlock();
1.481 + }
1.482 +
1.483 +/**
1.484 +If a draw command is received outside a begin/end redraw pair, then it is stored in a non-redraw
1.485 +segment. This function creates such a segment if it isn't already available.
1.486 +*/
1.487 +void CWsRedrawMsgWindow::HandleNonRedrawCommand(TWsGcOpcodes aOpcode)
1.488 + {
1.489 + // calling code should check the Window State
1.490 + WS_ASSERT_DEBUG(!InRedraw(), EWsPanicDrawCommandsInvalidState);
1.491 +
1.492 + // Attempting to draw part of a polygon in a new segment - only a very bad client can do this:
1.493 + TBool canCreate = !(aOpcode == EWsGcOpSegmentedDrawPolygonData || aOpcode == EWsGcOpDrawSegmentedPolygon);
1.494 + if ((!iCurrentSegment) && (!canCreate))
1.495 + {
1.496 + OwnerPanic(EWservPanicBadPolyData);
1.497 + }
1.498 +
1.499 + if (canCreate)
1.500 + AgeNonRedrawSegments();
1.501 +
1.502 + // If current redraw region is not for Non-Redraw drawing, add a region corresponding to the
1.503 + // full window to the redraw store. Need to make sure that this is done
1.504 + // before the AppendCommand, or bitmap/font handles are recorded, in StoreDrawCommandL
1.505 + TInt err = KErrNone;
1.506 + if ((!iCurrentSegment) || iCurrentSegment->iRedrawSegmentType != ESegmentTypeNonRedraw)
1.507 + {
1.508 + TRAP(err,CreateNewSegmentL(TRect(WsWin()->Size()), ESegmentTypeNonRedraw));
1.509 + }
1.510 +
1.511 + if (err != KErrNone)
1.512 + {
1.513 + Invalidate();
1.514 + }
1.515 +
1.516 + if(iWsWin->VisibleRegion().CheckError())
1.517 + iWsWin->Screen()->AddRedrawRegion(iWsWin->WindowArea());
1.518 + else if(iWsWin->VisibleRegion().Count())
1.519 + iWsWin->Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
1.520 + else if(!iWsWin->HasBeenDrawnToScreen())
1.521 + CliWin()->ScheduleRegionUpdate(&iWsWin->VisibleRegion());
1.522 + }
1.523 +
1.524 +/**
1.525 +This function attempts to prevent non-redraw segments from growing indefinitely by requesting redraws
1.526 +and throwing away old ones.
1.527 +*/
1.528 +void CWsRedrawMsgWindow::AgeNonRedrawSegments()
1.529 + {
1.530 + if (!iRedrawSegments.Count())
1.531 + return;
1.532 +
1.533 + //Count the number of non-redraw segs
1.534 + TInt nonRedrawSegCount = 0;
1.535 + for (TInt i = iRedrawSegments.Count()-1; i >= 0; --i)
1.536 + {
1.537 + CRedrawSegment* segment = iRedrawSegments[i];
1.538 + if (segment->iRedrawSegmentType == ESegmentTypeNonRedraw)
1.539 + {
1.540 + ++nonRedrawSegCount;
1.541 + }
1.542 + }
1.543 +
1.544 + TBool callInvalidate = EFalse;
1.545 +
1.546 + //To prevent the number of non redraw segements to grow indefinitely,
1.547 + //delete the oldest if their number exceeds KNonRedrawSegMaxLimit
1.548 + if (nonRedrawSegCount > KNonRedrawSegMaxLimit)
1.549 + {
1.550 + TInt keep = KNonRedrawSegThreshold; // keep this many, the most recent ones
1.551 + for (TInt i = iRedrawSegments.Count()-1; i >= 0; --i)
1.552 + {
1.553 + CRedrawSegment* segment = iRedrawSegments[i];
1.554 + if (segment->iRedrawSegmentType == ESegmentTypeNonRedraw)
1.555 + {
1.556 + if (keep-- > 0)
1.557 + {
1.558 + continue;
1.559 + }
1.560 + else if (segment!=iCurrentSegment) //never delete the current segment
1.561 + {
1.562 + callInvalidate = ETrue;
1.563 + iRedrawSegments.Remove(i);
1.564 + delete segment;
1.565 + }
1.566 + }
1.567 + }
1.568 + }
1.569 +
1.570 + if(iCurrentSegment && iCurrentSegment->iRedrawSegmentType == ESegmentTypeNonRedraw)
1.571 + {
1.572 + // If the current segment is an old non-redraw segment, try to get rid of it
1.573 + TTime now;
1.574 + now.UniversalTime();
1.575 + TTimeIntervalMicroSeconds age = now.MicroSecondsFrom(iCurrentSegment->iCreationTime);
1.576 + if (age > iNonRedrawAgeLimit)
1.577 + {
1.578 + // First, find any even older non redraw segments and discard them.
1.579 + for (TInt seg = iRedrawSegments.Count() - 2; seg >= 0; --seg)
1.580 + {
1.581 + CRedrawSegment * segment = iRedrawSegments[seg];
1.582 + if (segment->iRedrawSegmentType == ESegmentTypeNonRedraw)
1.583 + {
1.584 + age = now.MicroSecondsFrom(segment->iCreationTime);
1.585 + if ((age > iNonRedrawAgeLimit * 2) && (segment!=iCurrentSegment))
1.586 + {
1.587 + iRedrawSegments.Remove(seg);
1.588 + delete segment;
1.589 + }
1.590 + }
1.591 + }
1.592 +
1.593 + // Then force the creation of a new segment, so that the current one can be allowed to age and eventually vanish.
1.594 + iCurrentSegment->iCreationTime = now;
1.595 + iCurrentSegment = NULL;
1.596 + callInvalidate = ETrue;
1.597 + }
1.598 + }
1.599 +
1.600 + if(callInvalidate)
1.601 + Invalidate(); // Invalidate the window so that a complete redraw should occur
1.602 + }
1.603 +
1.604 +/**
1.605 +Obtains a region from the redraw store, intersects it with the global redraw region which
1.606 +we have been asked to draw to, and returns the intersection.
1.607 +*/
1.608 +const TRegion * CWsRedrawMsgWindow::ReadRegion(const TInt aRegionNum)
1.609 + {
1.610 + // Catch anyone calling this without first deciding where they want to draw
1.611 + WS_ASSERT_DEBUG(iWsWin->ScheduledRegion(), EWsPanicScheduledRedraw);
1.612 +
1.613 + // We are drawing to the global region, and we have a region in the redraw store to clip to.
1.614 + // We want the intersection of these to actually draw to.
1.615 + // Andy - this allocates memory during drawing, which is bad in OOM conditions.
1.616 + iLocalRedrawRegion.Copy(iRedrawSegments[aRegionNum]->iRegion);
1.617 + iLocalRedrawRegion.Offset(WsWin()->Origin());
1.618 + iLocalRedrawRegion.Intersect(*iGlobalRedrawRegion);
1.619 + iLocalRedrawRegion.Tidy();
1.620 +
1.621 + // If the resulting region is empty there is no point drawing its corresponding commands
1.622 + if (iLocalRedrawRegion.IsEmpty())
1.623 + return NULL;
1.624 + else
1.625 + return &iLocalRedrawRegion;
1.626 + }
1.627 +
1.628 +TInt CWsRedrawMsgWindow::SubtractRectFromSegmentArray(const TRect& aRect)
1.629 + {
1.630 + TInt numOfRegionsRemoved =0;
1.631 + for (TInt regionNum = iRedrawSegments.Count() - 1; regionNum >= 0; --regionNum)
1.632 + {
1.633 + if (iRedrawSegments[regionNum]->iRedrawSegmentType != ESegmentTypePendingRedraw)
1.634 + {
1.635 + RWsRegion& region = iRedrawSegments[regionNum]->iRegion;
1.636 + region.SubRect(aRect);
1.637 + if (region.CheckError())
1.638 + {
1.639 + // Ouch. Drop the now broken segment and ask for a full redraw
1.640 + // Andy - This is an error condition and needs to check for infinite loops
1.641 + delete iRedrawSegments[regionNum];
1.642 + iRedrawSegments.Remove(regionNum);
1.643 + numOfRegionsRemoved++;
1.644 + Invalidate();
1.645 + }
1.646 + else
1.647 + {
1.648 + // check if region has zero uncovered rectangles left
1.649 + if (region.IsEmpty())
1.650 + { // delete draw commands, release bitmaps and fonts
1.651 + delete iRedrawSegments[regionNum];
1.652 + iRedrawSegments.Remove(regionNum);
1.653 + numOfRegionsRemoved++;
1.654 + }
1.655 + else
1.656 + {
1.657 + if (region.Count() > KRegionCompressThreshold)
1.658 + { // tidy up the rectangles
1.659 + region.Tidy();
1.660 + }
1.661 + }
1.662 + }
1.663 + }
1.664 + }
1.665 + return numOfRegionsRemoved;
1.666 + }
1.667 +
1.668 +/*------------------------------------------------------------------------------
1.669 + Description: Clears out the command buffer if aError indicates an
1.670 + error has occurred whilst storing commands.
1.671 + -----------------------------------------------------------------------------*/
1.672 +void CWsRedrawMsgWindow::DiscardStoredCommandsIfError(TInt aError)
1.673 + {
1.674 + if (aError != KErrNone)
1.675 + {
1.676 + // Discard stored commands by clearing out the command buffer
1.677 + DiscardStoredCommands();
1.678 +
1.679 + if (!(iFlags&ENoRepeatRedraw))
1.680 + Invalidate();
1.681 + iFlags |= ENoRepeatRedraw;
1.682 + }
1.683 + }
1.684 +
1.685 +/*------------------------------------------------------------------------------
1.686 + Description: If the graphics context has changed and we are currently storing
1.687 + commands, store the data given by aCmdData.
1.688 +
1.689 + -----------------------------------------------------------------------------*/
1.690 +TBool CWsRedrawMsgWindow::DrawCommand(CWsGc* aGc,const TAny *aCmdData)
1.691 + {
1.692 + // Store commands, clearing out buffer if error occurs.
1.693 + TRAPD(err,StoreDrawCommandL(aGc,aCmdData));
1.694 + DiscardStoredCommandsIfError(err);
1.695 + return EFalse;
1.696 + }
1.697 +
1.698 +void CWsRedrawMsgWindow::GcAttributeChange(CWsGc* aGc,const TAny *aCmdData)
1.699 + {
1.700 + if (iLastDrawGc == aGc && iCurrentSegment)
1.701 + {
1.702 + TInt err = KErrNone;
1.703 + if (IsWsFontOperation(CWsClient::iCurrentCommand.iOpcode))
1.704 + {
1.705 + TWsGcCmdUnion pData;
1.706 + pData.any=aCmdData;
1.707 + TRAP(err,AddWsFontL(*pData.UInt));
1.708 + }
1.709 + if (KErrNone == err)
1.710 + TRAP(err,AppendCommandL(aCmdData));
1.711 + DiscardStoredCommandsIfError(err);
1.712 + }
1.713 +
1.714 + // INC135845:
1.715 + // Retain the bitmap handle for the lifetime of the redraw store
1.716 + // If the client destroys it, we will still have a reference to it
1.717 + if (iCurrentSegment && CWsClient::iCurrentCommand.iOpcode == EWsGcOpUseBrushPattern)
1.718 + {
1.719 + TInt err = KErrNone;
1.720 + TWsGcCmdUnion pData;
1.721 + pData.any=aCmdData;
1.722 + TRAP(err, AddFbsBitmapsL(*pData.handle, 0));
1.723 + DiscardStoredCommandsIfError(err);
1.724 + }
1.725 + }
1.726 +
1.727 +void CWsRedrawMsgWindow::GcDeactivate(CWsGc* aGc)
1.728 + {
1.729 + if (iLastDrawGc==aGc)
1.730 + iLastDrawGc=NULL;
1.731 + }
1.732 +
1.733 +inline TBool CWsRedrawMsgWindow::IsFbsBitmapOperation(TInt aOpCode) const
1.734 + {
1.735 + WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
1.736 + &&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
1.737 + return (aOpCode>=EWsGcOpGdiBlt2&&aOpCode<=EWsGcOpGdiBltMasked)||(aOpCode>=EWsGcOpDrawBitmap&&aOpCode<=EWsGcOpDrawBitmapMasked)||(aOpCode==EWsGcOpGdiAlphaBlendBitmaps);
1.738 + }
1.739 +
1.740 +inline TBool CWsRedrawMsgWindow::IsWsBitmapOperation(TInt aOpCode) const
1.741 + {
1.742 + WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
1.743 + &&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
1.744 + return (aOpCode>=EWsGcOpGdiWsBlt2&&aOpCode<=EWsGcOpGdiWsBltMasked)||(aOpCode==EWsGcOpGdiWsAlphaBlendBitmaps)||(aOpCode==EWsGcOpWsDrawBitmapMasked);
1.745 + }
1.746 +
1.747 +inline TBool CWsRedrawMsgWindow::IsWsFontOperation(TInt aOpCode) const
1.748 + {
1.749 + return aOpCode==EWsGcOpUseFont;
1.750 + }
1.751 +
1.752 +inline TBool CWsRedrawMsgWindow::IsDrawWsGraphicOperation(TInt aOpCode) const
1.753 + {
1.754 + return (aOpCode == EWsGcOpDrawWsGraphic) || (aOpCode == EWsGcOpDrawWsGraphicPtr);
1.755 + }
1.756 +
1.757 +void CWsRedrawMsgWindow::ReplaceAndAppendCommandL(TInt aOpcode,const TAny* aCmdData)
1.758 + {
1.759 + const TInt KCharWidthInBytes = 2; // # of bytes for a Unicode character
1.760 + CWsClient* owner=iWsWin->WsOwner();
1.761 + WS_ASSERT_DEBUG(owner,EWsPanicDrawCommandsNullSession);
1.762 +
1.763 + // aCmdData doesn't contain data, it should be retrieved from client space using remote read
1.764 + TWsGcCmdUnion cmd;
1.765 + cmd.any=aCmdData;
1.766 + TUint16 newOpcode=EWsGcOpDrawText;
1.767 + TInt strLen=0;
1.768 + switch (aOpcode)
1.769 + {
1.770 + case EWsGcOpDrawTextPtr:
1.771 + newOpcode=EWsGcOpDrawText;
1.772 + strLen=cmd.DrawText->length;
1.773 + break;
1.774 + case EWsGcOpDrawTextVerticalPtr:
1.775 + newOpcode=EWsGcOpDrawTextVertical;
1.776 + strLen=cmd.DrawTextVertical->length;
1.777 + break;
1.778 + case EWsGcOpDrawBoxTextPtr:
1.779 + newOpcode=EWsGcOpDrawBoxText;
1.780 + strLen=cmd.BoxText->length;
1.781 + break;
1.782 + case EWsGcOpDrawBoxTextVerticalPtr:
1.783 + newOpcode=EWsGcOpDrawBoxTextVertical;
1.784 + strLen=cmd.DrawBoxTextVertical->length;
1.785 + break;
1.786 + }
1.787 + TInt strSize = strLen * KCharWidthInBytes;
1.788 + TInt oldCmdLen=CWsClient::iCurrentCommand.iCmdLength;
1.789 + TInt newCmdLen=sizeof(TWsCmdHeaderBase)+oldCmdLen+strSize;
1.790 + // resize buffer
1.791 + ExpandCommandBufferL(newCmdLen);
1.792 + // update current command to reflect the new command and data
1.793 + CWsClient::iCurrentCommand.iOpcode=newOpcode;
1.794 + CWsClient::iCurrentCommand.iOpcode|=EWsGcOpFlagDrawOp;
1.795 + CWsClient::iCurrentCommand.iCmdLength=(TInt16)(oldCmdLen+strSize);
1.796 + // write command header
1.797 + CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
1.798 + // write command
1.799 + CommandBufferWrite(aCmdData, oldCmdLen);
1.800 +
1.801 + // remote read
1.802 + TBuf<KReadBufferMaxLen> buf;
1.803 + TInt len=KReadBufferMaxLen;
1.804 + TInt bufOffset=0;
1.805 + TInt toGo=strLen;
1.806 + while(toGo>0)
1.807 + {
1.808 + if (len>toGo)
1.809 + len=toGo;
1.810 + owner->RemoteRead(buf,bufOffset);
1.811 + CommandBufferWrite(buf.Ptr(), len * KCharWidthInBytes);
1.812 + bufOffset+=len;
1.813 + toGo-=len;
1.814 + }
1.815 + }
1.816 +
1.817 +/*------------------------------------------------------------------------------
1.818 + Description: Stores drawing related commands into the command buffer
1.819 + -----------------------------------------------------------------------------*/
1.820 +void CWsRedrawMsgWindow::StoreDrawCommandL(CWsGc* aGc,const TAny *aCmdData)
1.821 + {
1.822 + TWsGcOpcodes currentOpcode = static_cast<TWsGcOpcodes>(CWsClient::iCurrentCommand.iOpcode);
1.823 +
1.824 + // If we get an extra command after the redraw has finished then redraw strategy needs to know
1.825 + if (!InRedraw())
1.826 + {
1.827 +#ifdef __WINS__
1.828 + TBool isDrawingCommand = (currentOpcode != EWsGcOpSegmentedDrawPolygonData) && (currentOpcode != EWsGcOpDrawSegmentedPolygon);
1.829 +
1.830 + if( CWsClient::DebugEnforceRedrawCallingConvention() && isDrawingCommand)
1.831 + CWsClient::PanicCurrentClient(EWservPanicWindowBeginRedrawNotCalled);
1.832 +#endif
1.833 + HandleNonRedrawCommand(currentOpcode);
1.834 + }
1.835 +
1.836 + // If there is no current segment then we have discarded it at some point
1.837 + // since beginning this redraw.
1.838 + if (iCurrentSegment)
1.839 + {
1.840 + TWsGcCmdUnion pData;
1.841 + pData.any = aCmdData;
1.842 + if (IsFbsBitmapOperation(currentOpcode))
1.843 + {
1.844 + TInt maskHandle = 0;
1.845 + TInt handle = aGc->FbsBitmapHandle(currentOpcode, pData, maskHandle);
1.846 + AddFbsBitmapsL(handle, maskHandle);
1.847 + }
1.848 + else if (IsWsBitmapOperation(currentOpcode))
1.849 + {
1.850 + TInt maskHandle = 0;
1.851 + TInt handle = aGc->WsBitmapHandle(currentOpcode, pData, maskHandle);
1.852 + AddWsBitmapsL(handle, maskHandle);
1.853 + }
1.854 + else if (IsDrawWsGraphicOperation(currentOpcode))
1.855 + {
1.856 + TGraphicDrawerId drawerId;
1.857 + drawerId.iId = pData.WsGraphic->iId;
1.858 + drawerId.iIsUid = (pData.WsGraphic->iFlags & EWsGraphicIdUid);
1.859 + iCurrentSegment->AddDrawerL(drawerId);
1.860 + }
1.861 +
1.862 + // If the graphics context has changed since last time store the new graphics
1.863 + // context attributes.
1.864 + if (aGc != iLastDrawGc)
1.865 + {
1.866 + StoreAllGcAttributesL(aGc);
1.867 + iLastDrawGc = aGc;
1.868 + }
1.869 +
1.870 + // For operation which requires remote read from client space, we must retrieve that data and store
1.871 + // it in command buffer at server side and change opcode if necessary e.g EWsGcOpDrawTextPtr to EWsGcOpDrawText
1.872 + // to avoid remote read during DoDrawing operation
1.873 + if (IsRemoteReadRequired(currentOpcode))
1.874 + ReplaceAndAppendCommandL(currentOpcode,aCmdData);
1.875 + else
1.876 + // Append the command data to the command buffer
1.877 + AppendCommandL(aCmdData, EWsGcOpFlagDrawOp);
1.878 + }
1.879 + }
1.880 +
1.881 +/*------------------------------------------------------------------------------
1.882 + Description: Stores given drawing command data into the command buffer.
1.883 + -----------------------------------------------------------------------------*/
1.884 +void CWsRedrawMsgWindow::AppendCommandL(const TAny* aCmdData, const TUint16 aOpcodeFlags)
1.885 + {
1.886 + if (CWsClient::iCurrentCommand.iOpcode == EWsGcOpSetClippingRegion)
1.887 + {
1.888 + // The client is defining a clipping region
1.889 +
1.890 + // make room for the header
1.891 + ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
1.892 +
1.893 + // Externalize the clipping region data from position after the header
1.894 + RBufWriteStream bufWriteStream;
1.895 + bufWriteStream.Open(*CurrentDrawCommandBuffer(), CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
1.896 + CleanupClosePushL(bufWriteStream);
1.897 + TInt dataLen = iLastDrawGc->ExternalizeClippingRegionL(bufWriteStream);
1.898 +
1.899 + // Setup the clipping region data header
1.900 + CWsClient::iCurrentCommand.iOpcode = EWsStoreClippingRegion;
1.901 + CWsClient::iCurrentCommand.iCmdLength = REINTERPRET_CAST(TInt16&,dataLen);
1.902 +
1.903 + // Store command header for clipping region data at current write position
1.904 + CommandBufferWrite(&CWsClient::iCurrentCommand,sizeof(TWsCmdHeaderBase));
1.905 +
1.906 + // Update write position for command data
1.907 + iCurrentSegment->iCurrentCommandBufferWritePos += dataLen;
1.908 +
1.909 + CleanupStack::PopAndDestroy(&bufWriteStream);
1.910 + }
1.911 + else
1.912 + {
1.913 + TUint16 opcode = CWsClient::iCurrentCommand.iOpcode;
1.914 + CWsClient::iCurrentCommand.iOpcode |= aOpcodeFlags;
1.915 +
1.916 + // ensure room in command buffer
1.917 + ExpandCommandBufferL(sizeof(TWsCmdHeaderBase) + CWsClient::iCurrentCommand.iCmdLength);
1.918 +
1.919 + // Store command header to current position
1.920 + CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
1.921 +
1.922 + // If there's command data (other than header), store it
1.923 + if (CWsClient::iCurrentCommand.iCmdLength > 0)
1.924 + {
1.925 + CommandBufferWrite(aCmdData, CWsClient::iCurrentCommand.iCmdLength);
1.926 + }
1.927 +
1.928 + CWsClient::iCurrentCommand.iOpcode = opcode;
1.929 + }
1.930 + }
1.931 +
1.932 +
1.933 +/*------------------------------------------------------------------------------
1.934 + Description: Stores graphics context information into the command buffer
1.935 + from the current write position.
1.936 + -----------------------------------------------------------------------------*/
1.937 +void CWsRedrawMsgWindow::StoreAllGcAttributesL(CWsGc* aGc)
1.938 + {
1.939 + // In order for the externalize below to work correctly from
1.940 + // a non-zero position we have to create the header placeholder
1.941 + ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
1.942 +
1.943 + // Externalise GC attribute data. We do this before writing the
1.944 + // header as we do not know the size of the data yet and it is
1.945 + // part of the header.
1.946 + TInt numOfBytesAdded = aGc->ExternalizeL(*CurrentDrawCommandBuffer(),
1.947 + CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
1.948 +
1.949 + // Setup the header
1.950 + TWsCmdHeaderBase cmdHeader;
1.951 + cmdHeader.iCmdLength = (TInt16) numOfBytesAdded; // as calculated above
1.952 + cmdHeader.iOpcode = (TInt16) EWsStoreAllGcAttributes;
1.953 +
1.954 + // Store the header for the GC data into the space we created
1.955 + CommandBufferWrite(&cmdHeader, sizeof(TWsCmdHeaderBase));
1.956 +
1.957 + // Update write position for command data
1.958 + iCurrentSegment->iCurrentCommandBufferWritePos += numOfBytesAdded;
1.959 + }
1.960 +
1.961 +/*------------------------------------------------------------------------------
1.962 + Description: Loops through the whole of the current command buffer, processing
1.963 + each in turn.
1.964 + -----------------------------------------------------------------------------*/
1.965 +void CWsRedrawMsgWindow::DrawCommandsL()
1.966 + {
1.967 + LOG_WINDOW_REDRAW_START(WsWin(), iGlobalRedrawRegion);
1.968 + WS_ASSERT_DEBUG(iMemoryLock > 0, EWsPanicMemoryLock);
1.969 + static TBuf8<EClientBufferMaxSize> buf;
1.970 + TInt regionCount = iRedrawSegments.Count();
1.971 +
1.972 + for (TInt regionNum = 0; regionNum < regionCount; ++regionNum)
1.973 + {
1.974 + CRedrawSegment* segment = iRedrawSegments[regionNum];
1.975 + LOG_REDRAW_SEGMENT(regionNum, segment->iRedrawSegmentType);
1.976 + if (segment->iRedrawSegmentType == ESegmentTypePendingRedraw)
1.977 + continue;
1.978 +
1.979 + // The amount of commands we process is given by the value of the
1.980 + // current write position rather than the size of the command buffer.
1.981 + // Note: the write position is incremented as each command is stored and
1.982 + // will typically be less than the buffer size.
1.983 + const TInt length = segment->iCurrentCommandBufferWritePos;
1.984 +
1.985 + // need to draw this region?
1.986 + const TRegion * localDrawRegion = 0;
1.987 + if (length)
1.988 + localDrawRegion = ReadRegion(regionNum);
1.989 + if (localDrawRegion)
1.990 + {
1.991 + CPlaybackGc::Instance()->SetTargetRegion(localDrawRegion);
1.992 + LOG_REDRAW_SEGMENT_REGION(localDrawRegion)
1.993 +
1.994 + TWsCmdHeaderBase header;
1.995 + TInt pos = 0; // Set to first command position in buffer
1.996 + CBufSeg* drawCmdBuffer = segment->iDrawCommands;
1.997 +
1.998 +#ifdef _DEBUG
1.999 + // Read the first command header. The associated opcode must always be
1.1000 + // EWsStoreAllGcAttributes as this is always the first stored item.
1.1001 + drawCmdBuffer->Read(pos,&header,sizeof(TWsCmdHeaderBase));
1.1002 + WS_ASSERT_DEBUG(header.iOpcode == EWsStoreAllGcAttributes, EWsPanicDrawCommandsBufferCorrupt);
1.1003 +#endif
1.1004 +
1.1005 + // Read through remaining commands
1.1006 + while (pos < length)
1.1007 + {
1.1008 + // Get header of command
1.1009 + drawCmdBuffer->Read(pos, &header, sizeof(TWsCmdHeaderBase));
1.1010 + pos += sizeof(TWsCmdHeaderBase);
1.1011 +
1.1012 + switch(header.iOpcode)
1.1013 + {
1.1014 + case EWsStoreAllGcAttributes:
1.1015 + {
1.1016 + // Header indicates command encapsulates gc data
1.1017 + CPlaybackGc::Instance()->Reset();
1.1018 +
1.1019 + // Read gc data
1.1020 + CPlaybackGc::Instance()->InternalizeL(*drawCmdBuffer,pos);
1.1021 +
1.1022 + }
1.1023 + break;
1.1024 + case EWsStoreClippingRegion:
1.1025 + {
1.1026 + // Clipping region data read in from current position via stream
1.1027 + RBufReadStream bufReadStream;
1.1028 + bufReadStream.Open(*drawCmdBuffer,pos);
1.1029 + CleanupClosePushL(bufReadStream);
1.1030 + CPlaybackGc::Instance()->InternalizeClippingRegionL(bufReadStream);
1.1031 + CleanupStack::PopAndDestroy(&bufReadStream);
1.1032 + }
1.1033 + break;
1.1034 + default:
1.1035 + {
1.1036 + // Another type of command. Read it.
1.1037 + CWsClient::iCurrentCommand.iCmdLength = header.iCmdLength;
1.1038 + drawCmdBuffer->Read(pos,buf,header.iCmdLength);
1.1039 +
1.1040 + TInt opcode = header.iOpcode;
1.1041 +
1.1042 + // Drawing command?
1.1043 + if (opcode & EWsGcOpFlagDrawOp)
1.1044 + {
1.1045 + opcode &= ~EWsGcOpFlagDrawOp;
1.1046 + }
1.1047 + if (opcode > -1)
1.1048 + {
1.1049 + LOG_PLAYBACK_GC_COMMAND(opcode, buf.Ptr())
1.1050 + CPlaybackGc::Instance()->CommandL(static_cast<TWsGcOpcodes>(opcode),buf);
1.1051 + }
1.1052 + }
1.1053 + break;
1.1054 + }
1.1055 + pos += header.iCmdLength; // Move on, header indicates length
1.1056 + }
1.1057 + DEBUGOSB // per-redraw-segment debug osb updates
1.1058 + }
1.1059 + }
1.1060 + LOG_WINDOW_REDRAW_END(WsWin());
1.1061 + }
1.1062 +
1.1063 +/*------------------------------------------------------------------------------
1.1064 + Description: Called when the currently stored graphics commands
1.1065 + are no longer required.
1.1066 + -----------------------------------------------------------------------------*/
1.1067 +void CWsRedrawMsgWindow::DiscardStoredCommands()
1.1068 + {
1.1069 + iCurrentSegment = NULL;
1.1070 + if (iRedrawSegments.Count() > 0)
1.1071 + {
1.1072 + // First of all, if we have any redraws pending, update the screen with
1.1073 + // whatever commands we have before we throw them away:
1.1074 + if (iFlags & EPendingScheduledDraw)
1.1075 + {
1.1076 + Screen()->DoRedrawNow();
1.1077 + }
1.1078 +
1.1079 + // for all regions or just Partial Redraw regions > index 0: delete bitmaps and draw commands
1.1080 + iRedrawSegments.ResetAndDestroy();
1.1081 +
1.1082 + iLastDrawGc = NULL;
1.1083 + }
1.1084 + }
1.1085 +
1.1086 +void CWsRedrawMsgWindow::CreateNewSegmentL(const TRect& aRect, TRedrawSegmentType aNewRedrawRegionType)
1.1087 + {
1.1088 + CWsRedrawMsgWindow::CRedrawSegment* newRegion = CWsRedrawMsgWindow::CRedrawSegment::NewLC(aRect, aNewRedrawRegionType);
1.1089 +
1.1090 + iRedrawSegments.AppendL(newRegion);
1.1091 + iCurrentSegment = newRegion;
1.1092 + CleanupStack::Pop(newRegion);
1.1093 +
1.1094 + // Set iLastDrawGc to NULL. This will cause all GC attributes to be stored
1.1095 + // in redraw store when the window receives the next command
1.1096 + iLastDrawGc = NULL;
1.1097 + }
1.1098 +
1.1099 +static TInt FindBitmapByHandle(const TInt* aKey, const CFbsBitmap& aBitmap)
1.1100 + { // compare handles
1.1101 + return *aKey - aBitmap.Handle();
1.1102 + }
1.1103 +
1.1104 +static TInt InsertBitmapByHandle(const CFbsBitmap& aFirst, const CFbsBitmap& aSecond)
1.1105 + {
1.1106 + return aFirst.Handle() - aSecond.Handle();
1.1107 + }
1.1108 +
1.1109 +void CWsRedrawMsgWindow::AddFbsBitmapsL(TInt aHandle, TInt aMaskHandle)
1.1110 + {
1.1111 + iCurrentSegment->AddFbsBitmapL(aHandle, this);
1.1112 + if (aMaskHandle)
1.1113 + {
1.1114 + iCurrentSegment->AddFbsBitmapL(aMaskHandle, this);
1.1115 + }
1.1116 + }
1.1117 +
1.1118 +void CWsRedrawMsgWindow::CRedrawSegment::AddFbsBitmapL(TInt aHandle, CWsRedrawMsgWindow* aWindow)
1.1119 + {
1.1120 + if (iFbsBitmapArray.FindInOrder(aHandle, &FindBitmapByHandle) >= 0)
1.1121 + {
1.1122 + // Bitmap already in the store
1.1123 + return;
1.1124 + }
1.1125 +
1.1126 + CFbsBitmap* bitmap = new(ELeave) CFbsBitmap;
1.1127 + CleanupStack::PushL(bitmap);
1.1128 + if (bitmap->Duplicate(aHandle)!=KErrNone)
1.1129 + aWindow->OwnerPanic(EWservPanicBitmap);
1.1130 + iFbsBitmapArray.InsertInOrderL(bitmap, TLinearOrder<CFbsBitmap>(InsertBitmapByHandle));
1.1131 + CleanupStack::Pop(bitmap);
1.1132 + }
1.1133 +
1.1134 +void CWsRedrawMsgWindow::AddWsBitmapsL(TInt aHandle, TInt aMaskHandle)
1.1135 + {
1.1136 + if (iWsWin->WsOwner() == NULL)
1.1137 + Panic(EWsPanicDrawCommandsInvalidState);
1.1138 + DWsBitmap * bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aHandle, WS_HANDLE_BITMAP));
1.1139 + if (!bmp)
1.1140 + OwnerPanic(EWservPanicBitmap);
1.1141 + iCurrentSegment->AddWsBitmapL(bmp);
1.1142 + if (aMaskHandle)
1.1143 + {
1.1144 + bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aMaskHandle, WS_HANDLE_BITMAP));
1.1145 + if (!bmp)
1.1146 + OwnerPanic(EWservPanicBitmap);
1.1147 + iCurrentSegment->AddWsBitmapL(bmp);
1.1148 + }
1.1149 + }
1.1150 +
1.1151 +void CWsRedrawMsgWindow::CRedrawSegment::AddWsBitmapL(DWsBitmap* bitmap)
1.1152 + {
1.1153 + iWsBitmapArray.AppendL(bitmap);
1.1154 + bitmap->IncRefCount();
1.1155 + }
1.1156 +
1.1157 +void CWsRedrawMsgWindow::AddWsFontL(TInt aHandle)
1.1158 + {
1.1159 + if (iWsWin->WsOwner()==NULL)
1.1160 + Panic(EWsPanicDrawCommandsInvalidState);
1.1161 + TDblQueIter<CWsFbsFont> iter(CWsFontCache::List());
1.1162 + CWsFbsFont* font=NULL;
1.1163 + while((font=iter++)!=NULL)
1.1164 + {
1.1165 + if (font->Handle()==aHandle)
1.1166 + break;
1.1167 + }
1.1168 + if (font)
1.1169 + {
1.1170 + iCurrentSegment->iWsFontArray.AppendL(font);
1.1171 + ++(font->iCount);
1.1172 + }
1.1173 + }
1.1174 +
1.1175 +void CWsRedrawMsgWindow::CRedrawSegment::AddDrawerL(TGraphicDrawerId aDrawerId)
1.1176 + {
1.1177 + TInt error = iDrawerArray.InsertInOrder(aDrawerId, TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare));
1.1178 + if (error != KErrAlreadyExists && error != KErrNone)
1.1179 + {
1.1180 + User::Leave(error);
1.1181 + }
1.1182 + }
1.1183 +
1.1184 +TBool CWsRedrawMsgWindow::CRedrawSegment::ContainsDrawers(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
1.1185 + {
1.1186 + TBool result = EFalse;
1.1187 + if (iDrawerArray.Count() > 0)
1.1188 + {
1.1189 + STACK_REGION tempRegion;
1.1190 + tempRegion.Intersection(iRegion, aRegion);
1.1191 + if (tempRegion.CheckError() || (tempRegion.Count() > 0) )
1.1192 + { // regions do intersect, (presumed if region had an error); so check for a matching Id
1.1193 + const TInt drawersCount = aDrawers.Count();
1.1194 + for (TInt idx = 0; idx < drawersCount; ++idx)
1.1195 + { // (iDrawerArray is kept sorted)
1.1196 + if (KErrNotFound != iDrawerArray.FindInOrder(aDrawers[idx], TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
1.1197 + {
1.1198 + result = ETrue;
1.1199 + break;
1.1200 + }
1.1201 +
1.1202 + const TInt count = iDrawerArray.Count();
1.1203 + for(TInt i = 0; i < count; i++)
1.1204 + {
1.1205 + const CWsGraphicDrawer* drawer = CWsTop::WindowServer()->ResolveGraphic(iDrawerArray[i]);
1.1206 + if(drawer && drawer->Contains(aDrawers))
1.1207 + {
1.1208 + result = ETrue;
1.1209 + break;
1.1210 + }
1.1211 + }
1.1212 + }
1.1213 + }
1.1214 + tempRegion.Close();
1.1215 + }
1.1216 + return result;
1.1217 + }
1.1218 +
1.1219 +inline TBool CWsRedrawMsgWindow::NoBuffer() const
1.1220 + {
1.1221 + return (iRedrawSegments.Count() == 0);
1.1222 + }
1.1223 +
1.1224 +void CWsRedrawMsgWindow::ClientExposing()
1.1225 + {
1.1226 + Invalidate();
1.1227 + }
1.1228 +
1.1229 +/*------------------------------------------------------------------------------
1.1230 + Description: If a complete set of drawing commands have been stored
1.1231 + this method attempts to draw ALL the commands via DrawCommandsL().
1.1232 + It also draws the window in the background colour if the window is
1.1233 + opaque.
1.1234 + -----------------------------------------------------------------------------*/
1.1235 +void CWsRedrawMsgWindow::DrawWindow()
1.1236 + {
1.1237 + iFlags &= ~EPendingScheduledDraw;
1.1238 + // This is a happy window - it can draw itself whenever we ask.
1.1239 + if(iFlags&EBackgroundClear)
1.1240 + {
1.1241 + DrawBackgroundColor(iGlobalRedrawRegion);
1.1242 + }
1.1243 + // If valid commands have been stored, draw them.
1.1244 + if (iRedrawSegments.Count() > 0)
1.1245 + {
1.1246 + Lock();
1.1247 + TRAP_IGNORE(DrawCommandsL());
1.1248 + Unlock();
1.1249 + }
1.1250 + }
1.1251 +
1.1252 +void CWsRedrawMsgWindow::RemoveFromRedrawQueueIfEmpty()
1.1253 + {
1.1254 + if (iInvalid.Count()==0)
1.1255 + {
1.1256 + iInvalid.Clear(); // Ensures heap cell is freed, otherwise may be left as an empty cell
1.1257 + iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
1.1258 + }
1.1259 + }
1.1260 +
1.1261 +TBool CWsRedrawMsgWindow::NeedsRedraw() const
1.1262 +// If iInvalid has an persistant error it will not be reported as needing a redraw,
1.1263 +// this is needed as otherwise cases where validation of a window results
1.1264 +// in iInvalid having an error will get into an endless cycle of redraws.
1.1265 +// The down side of this is that sometimes a window will not be sent a redraw
1.1266 +// message when it needs it, some things can't be perfect!
1.1267 +//
1.1268 + {
1.1269 + if ((!iWsWin->IsVisible()) || iInvalid.IsEmpty())
1.1270 + return EFalse;
1.1271 +
1.1272 + TRect nextRedrawRect;
1.1273 + return GetRedrawRect(nextRedrawRect);
1.1274 + }
1.1275 +
1.1276 +TBool CWsRedrawMsgWindow::GetRedrawRect(TRect &aRect) const
1.1277 + {
1.1278 + if (iWsWin->ClientSetInvisible())
1.1279 + return EFalse;
1.1280 +
1.1281 + if(InRedraw())
1.1282 + {
1.1283 + aRect = iRedrawRect;
1.1284 + return (!aRect.IsEmpty());
1.1285 + }
1.1286 + else if(iInvalid.CheckError())
1.1287 + {
1.1288 + if (iFlags & EStoringEntireWindow || iWsWin->VisibleRegion().CheckError())
1.1289 + {
1.1290 + aRect = iWsWin->AbsRect();
1.1291 + }
1.1292 + else
1.1293 + {
1.1294 + aRect = iWsWin->VisibleRegion().BoundingRect();
1.1295 + }
1.1296 + if (!(iFlags & EStoringEntireWindow))
1.1297 + iWsWin->ClipRectToViewport(aRect);
1.1298 + aRect.Move(-iWsWin->Origin());
1.1299 + return (!aRect.IsEmpty());
1.1300 + }
1.1301 + else if(iInvalid.Count())
1.1302 + {
1.1303 + if (iFlags & EStoringEntireWindow)
1.1304 + {
1.1305 + aRect = iInvalid.BoundingRect();
1.1306 + }
1.1307 + else
1.1308 + {
1.1309 + RWsRegion region;
1.1310 + region.Copy(iInvalid);
1.1311 + region.Offset(iWsWin->Origin());
1.1312 + region.Intersect(iWsWin->VisibleRegion());
1.1313 + if (region.CheckError())
1.1314 + {
1.1315 + aRect = iInvalid.BoundingRect();
1.1316 + aRect.Move(iWsWin->Origin());
1.1317 + }
1.1318 + else
1.1319 + {
1.1320 + aRect = region.BoundingRect();
1.1321 + }
1.1322 + region.Close();
1.1323 + iWsWin->ClipRectToViewport(aRect);
1.1324 + aRect.Move(-iWsWin->Origin());
1.1325 + }
1.1326 + return (!aRect.IsEmpty());
1.1327 + }
1.1328 + else
1.1329 + {
1.1330 + return EFalse;
1.1331 + }
1.1332 + }
1.1333 +
1.1334 +void CWsRedrawMsgWindow::ClipInvalidRegion(const TRect &aRect)
1.1335 + {
1.1336 + if (iInvalid.Count()>0)
1.1337 + {
1.1338 + iInvalid.ClipRect(aRect);
1.1339 + RemoveFromRedrawQueueIfEmpty();
1.1340 + }
1.1341 + }
1.1342 +
1.1343 +void CWsRedrawMsgWindow::EndRedraw()
1.1344 + {
1.1345 + ++iCount;
1.1346 + if(!InRedraw())
1.1347 + OwnerPanic(EWservPanicDrawCommandsInvalidState);
1.1348 + if (iCurrentSegment)
1.1349 + {
1.1350 + iCurrentSegment->iDrawCommands->Compress();
1.1351 + if (iAtomicity==ENoAtomicity)
1.1352 + {
1.1353 + ScheduleUpdateOfSegment(iCurrentSegment);
1.1354 + }
1.1355 + else if(iAtomicity==ESegment)
1.1356 + {
1.1357 + PromoteLastPendingSegment();
1.1358 + ScheduleUpdateOfSegment(iCurrentSegment);
1.1359 + }
1.1360 + else if(iAtomicity==EWindow)
1.1361 + {
1.1362 + //only promote all pending segments when there are no invalid regions left in the window.
1.1363 + STACK_REGION regionAwaitingRedraws;
1.1364 + regionAwaitingRedraws.Copy(WsWin()->VisibleRegion());
1.1365 + regionAwaitingRedraws.Offset(-WsWin()->Origin());
1.1366 + regionAwaitingRedraws.Intersect(iInvalid);
1.1367 + if(regionAwaitingRedraws.IsEmpty())
1.1368 + PromoteAndUpdateAllPendingSegments();
1.1369 + regionAwaitingRedraws.Close();
1.1370 + }
1.1371 + }
1.1372 +
1.1373 + iCurrentSegment = NULL;
1.1374 + iFlags&=~(ENoRepeatRedraw|EBeginEndRedraw);
1.1375 + }
1.1376 +
1.1377 +void CWsRedrawMsgWindow::ScheduleUpdateOfSegment(CRedrawSegment* aSegment)
1.1378 + {
1.1379 + // Schedule an update of the area of the screen we just drew to:
1.1380 + iFlags |= EPendingScheduledDraw;
1.1381 + if(iWsWin->VisibleRegion().Count() || iWsWin->VisibleRegion().CheckError())
1.1382 + {
1.1383 + STACK_REGION draw; //### in low memory where VisibleRegion() is intact we can degrade much better than this!
1.1384 + draw.Copy(aSegment->iRegion);
1.1385 + draw.Offset(iWsWin->Origin());
1.1386 + draw.Intersect(iWsWin->VisibleRegion());
1.1387 + if(!draw.CheckError())
1.1388 + Screen()->AddRedrawRegion(draw);
1.1389 + else
1.1390 + Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
1.1391 + draw.Close();
1.1392 + }
1.1393 + }
1.1394 +
1.1395 +void CWsRedrawMsgWindow::ValidateRect(const TRect *aRect)
1.1396 + {
1.1397 + if (!WsWin()->BaseParent())
1.1398 + OwnerPanic(EWservPanicParentDeleted);
1.1399 + if (aRect)
1.1400 + iRedrawRect = *aRect;
1.1401 + if (!iInvalid.IsEmpty())
1.1402 + {
1.1403 + STACK_REGION validated;
1.1404 + validated.Copy(iInvalid);
1.1405 + if (aRect)
1.1406 + validated.ClipRect(iRedrawRect);
1.1407 +
1.1408 + if (iInvalid.CheckError())
1.1409 + {
1.1410 + iInvalid.Copy(iWsWin->VisibleRegion());
1.1411 + iInvalid.Offset(-iWsWin->Origin());
1.1412 + }
1.1413 + iInvalid.SubRegion(validated);
1.1414 + validated.Close();
1.1415 + }
1.1416 + RemoveFromRedrawQueueIfEmpty();
1.1417 + }
1.1418 +
1.1419 +TRgb CWsRedrawMsgWindow::BackColor() const
1.1420 + {
1.1421 + return(iBackColor);
1.1422 + }
1.1423 +
1.1424 +/**
1.1425 +This function used to be quite clever about what it invalidated and what it redrew by copying
1.1426 +rectangles of the screen around. This is a lot less subtle, and makes calling Scroll pretty much
1.1427 +pointless, but it IS functionally correct.
1.1428 +*/
1.1429 +void CWsRedrawMsgWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset,const TRect &aRect)
1.1430 + {
1.1431 + TRect rect = aRect;
1.1432 + rect.Intersection(aClipRect);
1.1433 + Invalidate(&rect);
1.1434 + rect = aRect;
1.1435 + rect.Move(aOffset);
1.1436 + rect.Intersection(aClipRect);
1.1437 + Invalidate(&rect);
1.1438 + }
1.1439 +
1.1440 +void CWsRedrawMsgWindow::ClearRedrawStore(TBool aClearPendingRedraw)
1.1441 + {
1.1442 + if(aClearPendingRedraw && (iFlags & EPendingScheduledDraw))
1.1443 + iFlags &= ~EPendingScheduledDraw;
1.1444 +
1.1445 + DiscardStoredCommands();
1.1446 + Invalidate();
1.1447 + }
1.1448 +
1.1449 +
1.1450 +void CWsRedrawMsgWindow::PrepareForResizeL(const TSize& aSize, TSize& /*aOldSize*/)
1.1451 + {
1.1452 + TBool anyIncreases(EFalse);
1.1453 + if (aSize.iWidth>iWsWin->Size().iWidth||aSize.iHeight>iWsWin->Size().iHeight)
1.1454 + {
1.1455 + anyIncreases = ETrue;
1.1456 + }
1.1457 +
1.1458 + TRect newWinRect(TPoint(0,0),aSize);
1.1459 + iInvalid.ClipRect(newWinRect);
1.1460 + if (anyIncreases)
1.1461 + {
1.1462 + // add new invalid region to iInvalid
1.1463 + iInvalid.AddRect(newWinRect);
1.1464 + QueueRedraw();
1.1465 + iWsWin->WsOwner()->TriggerRedraw();
1.1466 + }
1.1467 + }
1.1468 +
1.1469 +void CWsRedrawMsgWindow::Moved()
1.1470 + {
1.1471 + if (!(iFlags & EStoringEntireWindow))
1.1472 + {
1.1473 + DiscardSegmentsOutsideViewport();
1.1474 + }
1.1475 + if (iInvalid.Count())
1.1476 + {
1.1477 + QueueRedraw();
1.1478 + iWsWin->WsOwner()->TriggerRedraw();
1.1479 + }
1.1480 + }
1.1481 +
1.1482 +TBool CWsRedrawMsgWindow::Contains(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
1.1483 + {
1.1484 + if (iRedrawSegments.Count() > 0)
1.1485 + {
1.1486 + // scan redraw store: calls Contains() on every region drawing commands are stored for,
1.1487 + // looking for a DrawWsGraphic command that intersects the aRegion
1.1488 + TBool contains = EFalse;
1.1489 + const TInt regionCount = iRedrawSegments.Count();
1.1490 + // loop through regions, stops when a match is found
1.1491 + for (TInt regionNum = 0; (regionNum < regionCount) && !contains; ++regionNum)
1.1492 + {
1.1493 + contains = iRedrawSegments[regionNum]->ContainsDrawers(aDrawers, aRegion);
1.1494 + }
1.1495 + return contains;
1.1496 + }
1.1497 + else
1.1498 + {
1.1499 + return CWsWindowRedraw::Contains(aDrawers,aRegion);
1.1500 + }
1.1501 + }
1.1502 +
1.1503 +
1.1504 +void CWsRedrawMsgWindow::SetScope(TScope aScope)
1.1505 + {
1.1506 + if (aScope == EStoreEntireWindow)
1.1507 + {
1.1508 + if (!(iFlags & EStoringEntireWindow))
1.1509 + {
1.1510 + iFlags |= EStoringEntireWindow;
1.1511 + Invalidate();
1.1512 + }
1.1513 + }
1.1514 + else
1.1515 + {
1.1516 + if (iFlags & EStoringEntireWindow)
1.1517 + {
1.1518 + iFlags &= ~ EStoringEntireWindow;
1.1519 + DiscardSegmentsOutsideViewport();
1.1520 + }
1.1521 + }
1.1522 + }
1.1523 +
1.1524 +/**
1.1525 +Removes all segments from the redraw store which are outside the viewport onto the window.
1.1526 +Note that this doesn't clip the regions of those segments which are partly outside, since
1.1527 +this wouldn't actually achieve anything useful.
1.1528 +
1.1529 +This function allocates memory so it is not suitable to run as part of ReleaseMemory.
1.1530 +*/
1.1531 +TBool CWsRedrawMsgWindow::DiscardSegmentsOutsideViewport()
1.1532 + {
1.1533 + TBool discarded = EFalse;
1.1534 + TInt count = iRedrawSegments.Count();
1.1535 + STACK_REGION viewport;
1.1536 + CliWin()->SetClippedBaseArea(viewport);
1.1537 + viewport.Offset(-iWsWin->Origin());
1.1538 + STACK_REGION intersect;
1.1539 + for (TInt idx = count - 1; idx >= 0; --idx)
1.1540 + {
1.1541 + CRedrawSegment * segment = iRedrawSegments[idx];
1.1542 + intersect.Intersection(segment->iRegion, viewport);
1.1543 + if (!intersect.CheckError() && intersect.IsEmpty())
1.1544 + {
1.1545 + iInvalid.Union(segment->iRegion);
1.1546 + delete segment;
1.1547 + iRedrawSegments.Remove(idx);
1.1548 + if (iCurrentSegment == segment)
1.1549 + iCurrentSegment = NULL;
1.1550 + discarded = ETrue;
1.1551 + }
1.1552 + }
1.1553 + intersect.Close();
1.1554 + viewport.Close();
1.1555 + return discarded;
1.1556 + }
1.1557 +
1.1558 +/**
1.1559 +Statements encapsulated in between Lock() and Unlock() is guaranteed to execute in an
1.1560 +atomic way without being interupted by a call to ReleaseMemory from CWsMemoryManager.
1.1561 +Locking will prevent memory belonging to this object to be freed during a
1.1562 +memory alloc/realloc originating from self.
1.1563 +*/
1.1564 +void CWsRedrawMsgWindow::Lock()
1.1565 + {
1.1566 + ++iMemoryLock;
1.1567 + }
1.1568 +
1.1569 +void CWsRedrawMsgWindow::Unlock()
1.1570 + {
1.1571 + --iMemoryLock;
1.1572 + WS_ASSERT_DEBUG(iMemoryLock >= 0, EWsPanicMemoryLock);
1.1573 + }
1.1574 +
1.1575 +TBool CWsRedrawMsgWindow::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel aLevel)
1.1576 + {
1.1577 + //When this function is called, wserv is in the middle of executing something.
1.1578 + //Therefore we can not safely do anything that alters the state of any shared
1.1579 + //resouces (like e.g. CScreenRedraw::iInvalid).
1.1580 + //In addition, we should refrain from anything that might try to allocate memory.
1.1581 + TBool released = EFalse;
1.1582 + //Don't release iRedrawSegments from this win if its currently being rendered,
1.1583 + //is releasing memory or is receiving drawcommands.
1.1584 + if (iMemoryLock == 0 && !iCurrentSegment)
1.1585 + {
1.1586 + Lock();
1.1587 + switch (aLevel)
1.1588 + {
1.1589 + case MWsMemoryRelease::ELow:
1.1590 + break;
1.1591 + case MWsMemoryRelease::EMedium:
1.1592 + break;
1.1593 + case MWsMemoryRelease::EHigh:
1.1594 + //Only release memory from background windows.
1.1595 + if (iRedrawSegments.Count() > 0 && iWsWin->VisibleRegion().IsEmpty())
1.1596 + {
1.1597 + ReleaseRedrawSegments();
1.1598 + released = ETrue;
1.1599 + }
1.1600 + break;
1.1601 + }
1.1602 + Unlock();
1.1603 + }
1.1604 + return released;
1.1605 + }
1.1606 +
1.1607 +void CWsRedrawMsgWindow::ReleaseRedrawSegments()
1.1608 + {
1.1609 + iLastDrawGc = NULL;
1.1610 + iCurrentSegment = NULL;
1.1611 + iRedrawSegments.ResetAndDestroy();
1.1612 +
1.1613 + //The call to ResetAndDestroy just freed some memory so it should be
1.1614 + //possible to call Invalidate() now.
1.1615 + Invalidate();
1.1616 +
1.1617 + //Releasing the same window over and over again could quickly end up in
1.1618 + //a never ending loop with a high-prio client before we find the window
1.1619 + //that has nicked all memory. So call accessed now to prevent that.
1.1620 + iWsWin->Accessed();
1.1621 + }
1.1622 +
1.1623 +void CWsRedrawMsgWindow::VisibleRegionChange()
1.1624 + {
1.1625 + if (!iFlags & EStoringEntireWindow)
1.1626 + {
1.1627 + DiscardSegmentsOutsideViewport();
1.1628 + }
1.1629 + if ((!iInvalid.IsEmpty()) && (!iWsWin->VisibleRegion().IsEmpty()))
1.1630 + {
1.1631 + STACK_REGION exposed;
1.1632 + exposed.Copy(iInvalid);
1.1633 + exposed.Offset(iWsWin->Origin());
1.1634 + exposed.Intersect(iWsWin->VisibleRegion());
1.1635 + if (!exposed.IsEmpty())
1.1636 + {
1.1637 + QueueRedraw();
1.1638 + }
1.1639 + exposed.Close();
1.1640 + }
1.1641 + }
1.1642 +
1.1643 +TBool CWsRedrawMsgWindow::ReadyToDraw() const
1.1644 + {
1.1645 + //We are only ready to draw when we have a complete segment.
1.1646 + if (iWsWin->HasBeenDrawnToScreen())
1.1647 + return ETrue;
1.1648 +
1.1649 + if (iRedrawSegments.Count() == 0)
1.1650 + return EFalse;
1.1651 +
1.1652 + if (iRedrawSegments.Count() > 1)
1.1653 + return ETrue;
1.1654 +
1.1655 + if (iRedrawSegments[0]->iRedrawSegmentType == ESegmentTypePendingRedraw)
1.1656 + return EFalse;
1.1657 +
1.1658 + return ETrue;
1.1659 + }
1.1660 +
1.1661 +TInt CWsRedrawMsgWindow::SizeInBytes() const
1.1662 + {
1.1663 + TInt size = sizeof(CWsRedrawMsgWindow);
1.1664 + for(TInt i = iRedrawSegments.Count()-1; i >= 0; i--)
1.1665 + {
1.1666 + size += iRedrawSegments[i]->SizeInBytes();
1.1667 + }
1.1668 + size += iInvalid.Count() * sizeof(TRect);
1.1669 + size += iLocalRedrawRegion.Count() * sizeof(TRect);
1.1670 + return size;
1.1671 + }
1.1672 +
1.1673 +void CWsRedrawMsgWindow::PromoteLastPendingSegment()
1.1674 + {
1.1675 + if (iRedrawSegments.Count() > 0 && iRedrawSegments[iRedrawSegments.Count() - 1]->iRedrawSegmentType == ESegmentTypePendingRedraw)
1.1676 + {
1.1677 + CRedrawSegment * segment = iRedrawSegments[iRedrawSegments.Count() - 1];
1.1678 + const TRect * rect = segment->iRegion.RectangleList();
1.1679 + // when we get here there should only ever be one rectangle in the region, but we are playing safe
1.1680 + for (TInt r = 0; r < segment->iRegion.Count(); ++r)
1.1681 + {
1.1682 + SubtractRectFromSegmentArray(*rect);
1.1683 + ++rect;
1.1684 + }
1.1685 + segment->iRedrawSegmentType = ESegmentTypeRedraw;
1.1686 + }
1.1687 + }
1.1688 +
1.1689 +void CWsRedrawMsgWindow::PromoteAndUpdateAllPendingSegments()
1.1690 + {
1.1691 + for(TInt i =0; i<iRedrawSegments.Count(); i++)
1.1692 + {
1.1693 + CRedrawSegment * segment = iRedrawSegments[i];
1.1694 + if (segment->iRedrawSegmentType == ESegmentTypePendingRedraw)
1.1695 + {
1.1696 + const TRect * rect = segment->iRegion.RectangleList();
1.1697 + TInt totalRemovedSegments = 0;
1.1698 + for (TInt r = 0; r < segment->iRegion.Count(); ++r)
1.1699 + {
1.1700 + totalRemovedSegments += SubtractRectFromSegmentArray(*rect);
1.1701 + ++rect;
1.1702 + }
1.1703 + //we need to decrement the loop count to take into account any removed segments so we
1.1704 + //make sure we iterate over every segment in the array.
1.1705 + i-=totalRemovedSegments;
1.1706 + segment->iRedrawSegmentType = ESegmentTypeRedraw;
1.1707 + ScheduleUpdateOfSegment(segment);
1.1708 + }
1.1709 + }
1.1710 + }