os/graphics/windowing/windowserver/nga/SERVER/openwfc/screen.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2006-2010 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 #include "screen.h"
    17 
    18 #include <hal.h>
    19 #include <graphics/wsscreendevice.h>
    20 #include <graphics/wsscene.h>
    21 #include <graphics/wselement.h>
    22 #include "server.h"
    23 #include "wstop.h"
    24 #include "rootwin.h"
    25 #include "walkwindowtree.h"
    26 #include "EVENT.H"
    27 #include "windowgroup.h"
    28 #include "inifile.h"
    29 #include "pointer.h"
    30 #include "windowelementset.h"
    31 #include "registeredsurfacemap.h"
    32 #include "debugbar.h"
    33 #include "ScreenRedraw.h"
    34 #include "wspluginmanager.h"
    35 #include "devicemap.h"
    36 #include <graphics/wsdisplaymapping.h>
    37 #if defined(__WINS__) && defined(_DEBUG)
    38 #include "../../debuglog/osbwin.h"
    39 #endif
    40 #include <graphics/wsdisplaycontrol.h>
    41 
    42 GLREF_D CDebugLogBase *wsDebugLog;
    43 
    44 LOCAL_C inline MWsScene::TSceneRotation GcToScreen(CFbsBitGc::TGraphicsOrientation aGcOrientation)
    45 	{
    46 	MWsScene::TSceneRotation screenRotation = MWsScene::ESceneAntiClockwise0;
    47 
    48 	switch (aGcOrientation)
    49 		{
    50 		case CFbsBitGc::EGraphicsOrientationRotated90:
    51 			screenRotation = MWsScene::ESceneAntiClockwise90;
    52 			break;
    53 		case CFbsBitGc::EGraphicsOrientationRotated180:
    54 			screenRotation = MWsScene::ESceneAntiClockwise180;
    55 			break;
    56 		case CFbsBitGc::EGraphicsOrientationRotated270:
    57 			screenRotation = MWsScene::ESceneAntiClockwise270;
    58 			break;
    59 		}
    60 
    61 	return screenRotation;
    62 	}
    63 
    64 LOCAL_C inline TDeviceOrientation GcToDevice(CFbsBitGc::TGraphicsOrientation aGcOrientation)
    65 	{
    66 	// Each device orientation is defined to be the value where just the bit
    67 	// corresponding to the GDI orientation value is set.
    68 	return (TDeviceOrientation)(1 << aGcOrientation);
    69 	}
    70 
    71 LOCAL_D TBool FindNextValue(TLex& aLex, TInt& aValue) // assumes the list cannot contain *negative* integers
    72 	{
    73 	while (!aLex.Eos())
    74 		{
    75 		const TUint character=aLex.Peek();
    76 		if (Rng(TUint('0'), character, TUint('9')) || (character=='-'))
    77 			{
    78 			break;
    79 			}
    80 		aLex.Inc();
    81 		}
    82 
    83 	return (aLex.Val(aValue)==KErrNone);
    84 	}
    85 
    86 CScreen::CFallbackMap * CScreen::CFallbackMap::NewL(CScreen* aScreen)
    87 	{
    88 	CFallbackMap * self = new (ELeave) CFallbackMap(aScreen);
    89 	CleanupStack::PushL(self);
    90 	self->ConstructL();
    91 	CleanupStack::Pop(self);
    92 	return self;
    93 	}
    94 	
    95 CScreen::CFallbackMap::CFallbackMap(CScreen* aScreen) :
    96 	iScreen(aScreen),
    97 	iRegion(TRect(TPoint(0,0), TSize(1,1))),
    98 	iCount(0)
    99 	{
   100 	}
   101 	
   102 CScreen::CFallbackMap::~CFallbackMap()
   103 	{
   104 	delete []iMap;
   105 	}
   106 	
   107 void CScreen::CFallbackMap::ConstructL()
   108 	{
   109 	iMapSize = 0;
   110 	
   111 	for (TInt num = 0; num < iScreen->NumScreenSizeModes(); ++num)
   112 		{
   113 		if (iScreen->IsValidScreenSizeMode(num))
   114 			{
   115 			const TSizeMode & mode = iScreen->ScreenSizeModeData(num);
   116 			TInt width = mode.iScreenSize.iWidth / 32;
   117 			if (mode.iScreenSize.iWidth & (32 - 1))
   118 				++width;
   119 			TInt size = width * mode.iScreenSize.iHeight;
   120 			if (size > iMapSize)
   121 				iMapSize = size;
   122 			}
   123 		}
   124 		
   125 	iMap = new (ELeave) TInt [iMapSize];
   126 	}
   127 
   128 void CScreen::CFallbackMap::Prepare()
   129 	{
   130 	const TSizeMode & mode = iScreen->ScreenSizeModeData();
   131 	Mem::FillZ(iMap, iMapSize * sizeof(TInt));
   132 	iCount = mode.iScreenSize.iHeight * mode.iScreenSize.iWidth;
   133 	WS_ASSERT_DEBUG(iRegion.Count() == 1, EWsPanicScreenFallback);
   134 	iRegion.Clear();
   135 	iRegion.AddRect(TRect(mode.iOrigin, mode.iScreenSize));
   136 	}
   137 
   138 TBool CScreen::CFallbackMap::FillRegion(const TRegion& aRegion)
   139 	{
   140 	WS_ASSERT_DEBUG(!aRegion.CheckError(), EWsPanicScreenFallback);
   141 	if (aRegion.Count() > 20 || aRegion.CheckError())
   142 		return ETrue;
   143 	TBool hit = EFalse;
   144 	if (iCount > 0)
   145 		{
   146 		const TRect * rect = aRegion.RectangleList();
   147 		for (TInt num = 0; num < aRegion.Count(); ++num)
   148 			{
   149 			hit = FillRect(*rect) || hit;
   150 			if (iCount < 1)
   151 				break;
   152 			++rect;
   153 			}
   154 		}
   155 	return hit;
   156 	}
   157 	
   158 // x >> 5 is equivalent to x / 32
   159 // 0x1F is the rounding error when dividing by 32
   160 // 0x20 is 32.
   161 // The compiler might do all the optimizations for us - not checked.
   162 TBool CScreen::CFallbackMap::FillRect(const TRect& aRect)
   163 	{
   164 	TBool hit = EFalse;
   165 	const TSizeMode & mode = iScreen->ScreenSizeModeData();
   166 	TRect scrrect(mode.iOrigin, mode.iScreenSize);
   167 	TRect rect = aRect;
   168 	rect.Intersection(scrrect);
   169 	TInt rowWidthInInts = mode.iScreenSize.iWidth;
   170 	if (rowWidthInInts & 0x1F)
   171 		rowWidthInInts += 0x20;
   172 	rowWidthInInts >>= 5;
   173 
   174 	TInt colStartInInts = rect.iTl.iX >> 5;
   175 	TInt firstOffsetInBits = rect.iTl.iX & 0x1F;
   176 
   177 	for(TInt row = rect.iTl.iY; row < rect.iBr.iY; ++row)
   178 		{
   179 		TInt * map = iMap + row * rowWidthInInts + colStartInInts;
   180 		TInt offsetShift = 31 - firstOffsetInBits;
   181 		for (TInt col = rect.iTl.iX; col < rect.iBr.iX; ++col)
   182 			{
   183 			WS_ASSERT_DEBUG(map - iMap < iMapSize, EWsPanicScreenFallback);
   184 			if (!(*map & 1 << offsetShift))
   185 				{
   186 				--iCount;
   187 				hit = ETrue;
   188 				if (iCount < 1)
   189 					break;
   190 				(*map) |= (1 << offsetShift);
   191 				}
   192 			--offsetShift;
   193 			if (offsetShift < 0)
   194 				{
   195 				offsetShift = 31;
   196 				++map;
   197 				}
   198 			}
   199 		}
   200 	return hit;
   201 	}
   202 
   203 TInt CScreen::CFallbackMap::Count() const
   204 	{
   205 	return iCount;
   206 	}
   207 
   208 const TRect * CScreen::CFallbackMap::Rect() const
   209 	{
   210 	return iRegion.RectangleList();
   211 	}
   212 	
   213 const RRegion * CScreen::CFallbackMap::Region() const
   214 	{
   215 	return &iRegion;
   216 	}
   217 
   218 TInt CScreen::CFallbackMap::Resize(const TSize& aSize)
   219 	{
   220 	TInt err = KErrNone;
   221 	TInt width = (aSize.iWidth+31) >> 5;	// divide by 32
   222 	TInt invertedWidth = (aSize.iHeight+31) >> 5;	// divide by 32
   223 		
   224 	TInt maxSize = Max(width * aSize.iHeight,invertedWidth * aSize.iWidth);
   225 	if (maxSize > iMapSize)
   226 		{
   227 		TInt* newMap=NULL;
   228 		newMap = new  TInt [maxSize];
   229 		if (newMap)
   230 			{
   231 			delete []iMap;
   232 			iMap = newMap;
   233 			iMapSize = maxSize;
   234 			}
   235 		else
   236 			{
   237 			err = KErrNoMemory;
   238 			}
   239 		}
   240 	return err;
   241 	}
   242 	
   243 //
   244 // CScreen
   245 //
   246 CScreen::CScreen(): iDirects(_FOFF(CWsDirectScreenAccess,iLink)), iMaxContrast(-1), iMaxBrightness(-1)
   247 , iDisplayChangeSpinner(0), iConfigChangeSpinner(0)
   248 	{
   249 	}
   250 
   251 CScreen::~CScreen()
   252 	{
   253 	delete iDebugBar;
   254 	TInt ii;
   255 	if(iModes)
   256 		{
   257 		for(ii=iNumScreenSizeModes-1;ii>=0;--ii)
   258 			{
   259 			delete (*iModes)[ii];
   260 			}
   261 		iModes->Close();
   262 		delete iModes;
   263 		}
   264 	delete iRootWindow;
   265 	iScreenDevice = NULL;
   266 	delete iDsaDevice;
   267 	delete iDeviceMap;
   268 	delete iFallbackMap;
   269 	delete iSpriteManager;
   270 	delete iWindowElementSet;
   271 	if (!iDsaSurface.IsNull())
   272 	    {
   273 	    TInt err = iScene->UnregisterSurface(iDsaSurface);
   274 		WS_ASSERT_DEBUG(KErrNone == err, EWsPanicDirectScreenAccess);
   275 	    }
   276 
   277 	delete iSurfaceMap;
   278 	delete iRedraw;
   279 #if defined(__WINS__) && defined(_DEBUG)
   280 	delete iDebugWin;
   281 #endif
   282 	
   283 	if(iDisplayChangeNotifier)
   284 		delete iDisplayChangeNotifier;
   285 	if(iConfigChangeNotifier)
   286 		delete iConfigChangeNotifier;
   287 	iWsClientList.Close();
   288 	}
   289 
   290 void CScreen::ConstructL(const TRect& aDigitiserArea, TInt aScreenNumber)
   291 	{
   292 	iScreenNumber = aScreenNumber ;
   293 
   294 	if (wsDebugLog)
   295 		{
   296 		_LIT(KWSERVInitScreen,"Initialising for Screen %d");
   297 		wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant, KWSERVInitScreen, aScreenNumber);
   298 		}
   299 
   300 	// create screen redraw with render stages
   301 	iRedraw = CScreenRedraw::NewL(*this);
   302 	iScreenDevice = iRedraw->ObjectInterface<MWsScreenDevice>();
   303 	WS_ASSERT_ALWAYS(iScreenDevice, EWsPanicScreenDeviceMissing);
   304 	iScene = iRedraw->ObjectInterface<MWsScene>();
   305 	WS_ASSERT_ALWAYS(iScene, EWsPanicSceneMissing);
   306 	
   307 	iDeviceMap = CGraphicsDeviceMap::NewL(*iScreenDevice);
   308 
   309 	iDisplayControl = MWsScreen::ObjectInterface<MWsDisplayControl>();
   310 	iDisplayMapping = MWsScreen::ObjectInterface<MWsDisplayMapping>();
   311 	iDisplayPolicy = MWsScreen::ObjectInterface<MWsDisplayPolicy>();
   312 
   313 
   314 	// initialize screen size mode data
   315 	LoadScreenSizesL(iScreenDevice->SizeInPixels());
   316 	iFallbackMap = CFallbackMap::NewL(this);
   317 
   318 	LoadScreenSizeProperties(iScreenDevice->DisplayMode());
   319 
   320 	if (iDisplayPolicy)
   321 		{
   322 		iDisplayPolicy->NewAppModesAvailable();
   323 		}
   324 	
   325 	iDigitiserArea = aDigitiserArea;
   326 	SetInitialScreenSizeModeAndRotation(); //get the first/lowest valid mode.  Also calls SetDigitiserAreas
   327 	
   328 	ApplyRemainingWsiniSettingsL();
   329 	
   330 	iRootWindow = new (ELeave) CWsRootWindow(NULL, this);
   331 	iRootWindow->ConstructL();
   332 	
   333 	// Default fading parameters
   334 	iBlackMap=128;
   335 	iWhiteMap=255;
   336 	
   337 	iSpriteManager = CWsSpriteManager::NewL();
   338 
   339 	InitializeSceneL();
   340 	InitializeUiElementsL();
   341 	iSurfaceMap = new (ELeave) CRegisteredSurfaceMap(*iScene);
   342 	
   343 	//if the interface for notification is available. start notification AO.
   344 	if (iDisplayControl)
   345 		{
   346 		iDisplayChangeNotifier = CWsDisplayChangeNotifier::NewL(iDisplayControl, this);
   347 		iDisplayChangeNotifier->IssueNotificationRequest();
   348 		iConfigChangeNotifier = CWsConfigChangeNotifier::NewL(iDisplayControl, this);
   349 		iConfigChangeNotifier->IssueNotificationRequest();
   350 		}
   351 	doSetScreenMode(iScreenSizeMode,ETrue);
   352 	}
   353 
   354 void CScreen::ApplyRemainingWsiniSettingsL()
   355 	{
   356 #if defined(__WINS__) && defined(_DEBUG)
   357 	_LIT(KDebugOsb,"DEBUGOSB");
   358 	if(WsIniFile->FindVar(iScreenNumber, KDebugOsb))
   359 		{
   360 		_LIT(KDebugWinTitleFormat, "Screen %d, DSA surface");
   361 		TBuf<32> title;
   362 		title.Format(KDebugWinTitleFormat, iScreenNumber);
   363 		iDebugWin = CDebugOsbWin::NewL(title, iScreenDevice->SizeInPixels());
   364 		}
   365 #endif
   366 	
   367    	TInt autoClear = 1;
   368    	_LIT(KWSERVIniFileVarAutoClear,"AUTOCLEAR");
   369    	WsIniFile->FindVar(iScreenNumber,KWSERVIniFileVarAutoClear,autoClear);
   370    	if (autoClear != 0)
   371    		{
   372    		iFlags|=EAutoClear;
   373    		}
   374 		
   375 	_LIT(KBackLight,"BACKLIGHTCONTROL");
   376 	iBackLightFlag=WsIniFile->FindVar( iScreenNumber, KBackLight);
   377 
   378 	_LIT(KWSERVIniFileVarBlankScreen, "BLANKSCREENONROTATION");
   379 	if (WsIniFile->FindVar(iScreenNumber, KWSERVIniFileVarBlankScreen))
   380 		{
   381 		iFlags|=EBlankScreenOnRotation;
   382 		}
   383 
   384 	//Cache pointers to renderstage APIs required in CHANGETRACKING mode
   385 	iWindowTreeObserver = iRedraw->ObjectInterface<MWsWindowTreeObserver>(); 
   386 	iDrawAnnotationObserver = iRedraw->ObjectInterface<MWsDrawAnnotationObserver>();
   387 	iWindowVisibilityNotifier = iRedraw->ObjectInterface<MWsWindowVisibilityNotifier>();
   388 	if(iWindowVisibilityNotifier)
   389 		{
   390 		iWindowVisibilityNotifier->RegisterWindowVisibilityObserver(iRedraw);
   391 		}
   392 	if (WsIniFile->FindVar(iScreenNumber, KWSERVIniFileVarChangeTracking))
   393 		{
   394 		iFlags|=EChangeTracking;
   395 		}
   396 
   397 	//coverity[const]
   398 	TInt refreshRate = 1000000;
   399 	_LIT(KDebugBar, "DEBUGBAR");
   400 	if (WsIniFile->FindVar(KDebugBar, refreshRate))
   401 		{
   402 		if (refreshRate < 100000)
   403 			// coverity [dead_error_line]
   404 			refreshRate = 50000;
   405 		iDebugBar = CDebugBar::NewL(this, refreshRate);
   406 		}
   407 	}
   408 
   409 void CScreen::AcquireDsaScreenDeviceL()
   410 	{
   411 	//creates WSERV's DSA buffer handle
   412 	//registers the DSA surface into the scene accordingly to the value of aRegisterSurface
   413 	if(!iDsaDevice)
   414 		{
   415 		TDisplayMode screenMode = ENone;
   416 		screenMode = iScreenDevice->DisplayMode();
   417 		if(screenMode != ENone)
   418 			{
   419 			CreateDsaScreenDeviceIfSupportedL(screenMode);
   420 			// initialize DSA
   421 			iDsaDevice->SetAutoUpdate(EFalse);
   422 			iDsaDevice->SetDeviceOrientation(GcToDevice(ScreenSizeModeData().iRotation));
   423 			iDsaDevice->ChangeScreenDevice(NULL);    //This is necessary to initialise the screen
   424 			// register DSA Surface
   425 			TInt err = InitializeDsaSurface();
   426 			// create a graphics context to clear the DSA surface
   427 			if (!err)
   428 				err = iDsaDevice->CreateContext(iDsaGc);
   429 			if (!err)
   430 				iDsaGc->Activate(iDsaDevice);
   431 			if (err != KErrNone)
   432 				{
   433 				//Creation of the DSA surface failed
   434 				//Cleanup the DSA Surface ID
   435 				iDsaSurface = TSurfaceId::CreateNullId();
   436 				//and the iDsaDevice
   437 				delete iDsaDevice;
   438 				iDsaDevice = NULL;
   439 				User::Leave(err);
   440 				}
   441 			}
   442 		else
   443 			{
   444 			User::Leave(KErrNotSupported);
   445 			}
   446 		}
   447 	iNumberDrawingDsa++;
   448 	}
   449 
   450 void CScreen::CreateDsaScreenDeviceIfSupportedL(TDisplayMode aScreenMode)
   451 	{
   452 	if(DoCreateDsaScreenDevice(aScreenMode))
   453 		return;
   454 	// Try creating the screen device with all available display modes, going from best to worst
   455 	__ASSERT_COMPILE(EColorLast == 14); // if any display mode is added to TDisplayMode we must update the list below
   456 	// (the list below contains all enums in TDisplayMode except ENone, ERgb, EColorLast)
   457 	if(DoCreateDsaScreenDevice(EColor16MAP))
   458 		return;
   459 	if(DoCreateDsaScreenDevice(EColor16MA))
   460 		return;
   461 	if(DoCreateDsaScreenDevice(EColor16MU))
   462 		return;
   463 	if(DoCreateDsaScreenDevice(EColor16M))
   464 		return;
   465 	if(DoCreateDsaScreenDevice(EColor64K))
   466 		return;
   467 	if(DoCreateDsaScreenDevice(EColor4K))
   468 		return;
   469 	if(DoCreateDsaScreenDevice(EColor256))
   470 		return;
   471 	if(DoCreateDsaScreenDevice(EColor16))
   472 		return;
   473 	if(DoCreateDsaScreenDevice(EGray256))
   474 		return;
   475 	if(DoCreateDsaScreenDevice(EGray16))
   476 		return;
   477 	if(DoCreateDsaScreenDevice(EGray4))
   478 		return;
   479 	if(DoCreateDsaScreenDevice(EGray2))
   480 		return;
   481 	User::Leave(KErrNotSupported);
   482 	}
   483 
   484 TBool CScreen::DoCreateDsaScreenDevice(TDisplayMode aScreenMode)
   485 	{
   486 	TRAPD(err, iDsaDevice = CFbsScreenDevice::NewL(iScreenNumber, aScreenMode));
   487 	if(err == KErrNone)
   488 		{
   489 		TUint supportedDsaRotationModes = iDsaDevice->DeviceOrientationsAvailable();
   490 		MWsScene::TSceneRotation currenTSceneRotation = iScene->SceneRotation();
   491 		TBool doesDsaSupportThisMode = EFalse;
   492 		switch(currenTSceneRotation)
   493 			{
   494 			case MWsScene::ESceneAntiClockwise0:
   495 				if(supportedDsaRotationModes & EDeviceOrientationNormal)
   496 					{
   497 					doesDsaSupportThisMode = ETrue;
   498 					}
   499 				break;
   500 			case MWsScene::ESceneAntiClockwise90:
   501 				if(supportedDsaRotationModes & EDeviceOrientation90CW)
   502 					{
   503 					doesDsaSupportThisMode = ETrue;
   504 					}
   505 				break;				
   506 			case MWsScene::ESceneAntiClockwise180:
   507 				if(supportedDsaRotationModes & EDeviceOrientation180)
   508 					{
   509 					doesDsaSupportThisMode = ETrue;
   510 					}
   511 				break;				
   512 			case MWsScene::ESceneAntiClockwise270:
   513 				if(supportedDsaRotationModes & EDeviceOrientation270CW)
   514 					{
   515 					doesDsaSupportThisMode = ETrue;
   516 					}
   517 				break;				
   518 			default:
   519 				RDebug::Print(_L("** CScreen::DoCreateDsaScreenDevice Panic, non existing rotation mode"));
   520 				WS_PANIC_ALWAYS(EWsPanicInvalidOperation);
   521 				break;
   522 			}
   523 		if(!doesDsaSupportThisMode)
   524 			{
   525 			delete iDsaDevice;
   526 			iDsaDevice = NULL;
   527 			RDebug::Print(_L("** Current Rotation Mode not supported by the DSA device"));
   528 			err = KErrNotSupported;
   529 			}
   530 		}
   531 	return (err == KErrNone);
   532 	}
   533 
   534 void CScreen::AbortDSAs(RDirectScreenAccess::TTerminationReasons aReason, TSglQue<CWsDirectScreenAccess>& aDirects)
   535 	{
   536 	if (aDirects.IsEmpty())
   537 		return;
   538 
   539 	TInt nofDSAs = 0;
   540 	CWsDirectScreenAccess* direct= NULL;
   541 	TSglQueIter<CWsDirectScreenAccess> iter(aDirects);
   542 	while ((direct=iter++)!=NULL)
   543 		{
   544 		nofDSAs++;
   545 		direct->SignalAbort(aReason);
   546 		}
   547 
   548 	TRequestStatus timerStatus;
   549 	RTimer& timer=CWsTop::Timer();
   550 	timer.Cancel();
   551 
   552 	TRequestStatus** cancelReqList = (TRequestStatus**) User::AllocZ(sizeof(TRequestStatus*) * (nofDSAs + 1));
   553 	if (NULL != cancelReqList)
   554 		{
   555 		TInt dsaNo = 1;
   556 		timer.After(timerStatus, KDSAAbortingImmediateRespAwaitFrameMicrosec);
   557 		iter.SetToFirst();
   558 		while ((direct=iter++)!=NULL)
   559 			{
   560 			WS_ASSERT_DEBUG((dsaNo<=(nofDSAs)),EWsPanicDirectScreenAccess);
   561 			cancelReqList[ dsaNo ] = &direct->AbortStatus();
   562 			dsaNo++;
   563 			}
   564 		cancelReqList[ 0 ] = &timerStatus;
   565 
   566 		//wait for response or timeout
   567 		User::WaitForNRequest(cancelReqList, nofDSAs + 1);
   568 
   569 		iter.SetToFirst();
   570 		while ((direct=iter++)!=NULL)
   571 			{
   572 			if (direct->AbortStatus() != KRequestPending)
   573 				direct->CancelAbortObject(); // responded
   574 			else
   575 				direct->Abort();
   576 			}
   577 
   578 		if (timerStatus == KRequestPending)
   579 			{
   580 			timer.Cancel();
   581 			User::WaitForRequest(timerStatus);
   582 			}
   583 
   584 		User::Free(cancelReqList);
   585 		}
   586 	else
   587 		{
   588 		iter.SetToFirst();
   589 		while ((direct=iter++) != NULL)
   590 			{
   591 			TRequestStatus timerStatus;
   592 			RTimer& timer=CWsTop::Timer();
   593 			timer.Cancel();
   594 			timer.After(timerStatus, KDSAAbortingImmediateRespAwaitFrameMicrosec);
   595 			
   596 			//wait for response or timeout
   597 			User::WaitForRequest(direct->AbortStatus(), timerStatus);
   598 			
   599 			if (direct->AbortStatus() != KRequestPending)
   600 				direct->CancelAbortObject(); //responded
   601 			else
   602 				direct->Abort(); //timed out
   603 
   604 			if (timerStatus == KRequestPending)
   605 				{
   606 				timer.Cancel();
   607 				User::WaitForRequest(timerStatus);
   608 				}
   609 			}
   610 		}
   611 	}
   612 
   613 void CScreen::AbortAllDirectDrawing(RDirectScreenAccess::TTerminationReasons aReason)
   614 	{
   615 	AbortDSAs(aReason,iDirects);
   616 	}
   617 
   618 void CScreen::AddDirect(CWsDirectScreenAccess& aDirect)
   619 	{
   620 	TBool emptyBefore = iDirects.IsEmpty();
   621 	iDirects.AddLast(aDirect);
   622 	TBool emptyAfter = iDirects.IsEmpty();
   623 	if (emptyBefore && ! emptyAfter)
   624 		{
   625 		TWsEvent wsevent;
   626 		wsevent.SetType(EEventDirectScreenAccessBegin);
   627 		*(wsevent.Int()) = iScreenNumber;
   628 		TWindowServerEvent::PublishNotification(wsevent);
   629 		}
   630 
   631 	if (iDsaDrawState==EDsaDrawStateIdle && aDirect.IsVisible())
   632 		{
   633 		iDsaDrawState = EDsaDrawStateDrawing;
   634 		TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EDsaDrawingBegin, iScreenNumber));
   635 		}
   636 	}
   637 
   638 void CScreen::RemoveDirect(CWsDirectScreenAccess& aDirect)
   639 	{
   640 	TBool emptyBefore = iDirects.IsEmpty();
   641 	iDirects.Remove(aDirect);
   642 	TBool emptyAfter = iDirects.IsEmpty();
   643 	if (emptyAfter && ! emptyBefore)
   644 		{
   645 		TWsEvent wsevent;
   646 		wsevent.SetType(EEventDirectScreenAccessEnd);
   647 		*(wsevent.Int()) = iScreenNumber;
   648 		TWindowServerEvent::PublishNotification(wsevent);
   649 		}
   650 
   651 	if (iDsaDrawState==EDsaDrawStateDrawing && aDirect.IsVisible() && !HasVisibleDirectOnQueue())
   652 		{
   653 		iDsaDrawState = EDsaDrawStateIdle;
   654 		TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EDsaDrawingEnd, iScreenNumber));
   655 		}
   656 	}
   657 
   658 TBool CScreen::HasVisibleDirectOnQueue()
   659 	{
   660 	if (iDirects.IsEmpty())
   661 		return EFalse;
   662 
   663 	TSglQueIter<CWsDirectScreenAccess> iter(iDirects);
   664 	CWsDirectScreenAccess* dsa;
   665 	while ((dsa=iter++)!=NULL)
   666 		{
   667 		if (dsa->IsVisible())
   668 			return ETrue;
   669 		}
   670 
   671 	return EFalse;
   672 	}
   673 
   674 #if defined(_DEBUG)
   675 TBool CScreen::IsDirectOnQueue(const CWsDirectScreenAccess* aDirect)
   676 	{
   677 	TSglQueIter<CWsDirectScreenAccess> iter(iDirects);
   678 	CWsDirectScreenAccess* direct;
   679 	while ((direct=iter++)!=NULL)
   680 		{
   681 		if (direct==aDirect)
   682 			return ETrue;
   683 		}
   684 	return EFalse;
   685 	}
   686 #endif
   687 
   688 void CScreen::KillForegroundSession()
   689 	{
   690 	if (iCurrentFocus)
   691 		{
   692 		_LIT(KWSERVKillWinGp,"Killing Session owning Window Group with Id=%d");
   693 		if (wsDebugLog)
   694 			wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVKillWinGp,iCurrentFocus->Identifier());
   695 		iCurrentFocus->WsOwner()->SessionTerminate();
   696 		}
   697 	}
   698 
   699 CWsWindowGroup* CScreen::FindNewFocus(CWsRootWindow* aRootWindow)
   700 	{
   701 	CWsWindowGroup* newFocus;
   702 	for(newFocus=aRootWindow->Child();newFocus && newFocus->CanReceiveFocus()==EFalse;newFocus=newFocus->NextSibling()) {}
   703 
   704 	return newFocus;
   705 	}
   706 
   707 void CScreen::ResetFocus(CWsWindowGroup* aClosingWindow)
   708 	{
   709 	CWsWindowGroup* oldFocus=iCurrentFocus;
   710 	CWsWindowGroup* newFocus=NULL;
   711 	CScreen* newFocusedScreen=NULL;
   712 	iCurrentFocus=FindNewFocus(iRootWindow);
   713 	TBool focusedScreen= EFalse;
   714 	/*Focus policy is specified in the wsini.ini file using the keyword 'MULTIFOCUSPOLICY'.
   715 	If the keyword is not specified, then the default policy is run.
   716 	*/
   717 	if(!CWsTop::MultiFocusPolicy())
   718 		{
   719 		focusedScreen=(this==CWsTop::CurrentFocusScreen()); //check if this screen is the focus screen
   720 		if (!iCurrentFocus && focusedScreen)
   721 			{
   722 			/*If this screen is the focused screen but does not have a focusable window group, then search for the
   723 			next screen that has a focusable window group and set that screen as the focused screen.
   724 			*/
   725 			CScreen* screen=NULL;
   726 			TInt screenNo;
   727 			for (screenNo=0; screenNo<CWsTop::NumberOfScreens() && !newFocus; ++screenNo)
   728 				{
   729 				if (screenNo!=iScreenNumber)
   730 					{
   731 					screen=CWsTop::Screen(screenNo);
   732 					newFocus=FindNewFocus(screen->RootWindow());
   733 					}
   734 				}
   735 			if (newFocus)
   736 				newFocusedScreen=screen;
   737 			}
   738 		}
   739 	/*Scenario A: multi-focus policy
   740 			newFocusedScreen is NULL
   741 			focusedScreen is EFalse
   742 			CWsTop::MultiFocusPolicy() returns ETrue
   743 			Check if the new focusable window group is not the same, send focus lost message to window group
   744 			that has just lost focus and send focus gain message to window group that can receive focus.
   745 	  Scenario B: single-focus policy (default)
   746 			CWsTop::MultiFocusPolicy() returns EFalse
   747 			Check if the new focusable window group is not the same or if there is a new focused screen, send focus lost
   748 			message to window group that has just lost focus and send focus gain message to window group that can receive focus.
   749 	*/
   750 	if (iCurrentFocus!=oldFocus || newFocusedScreen)
   751 		{
   752 		if (oldFocus && (focusedScreen || CWsTop::MultiFocusPolicy()) && oldFocus!=aClosingWindow)
   753 			{
   754 			oldFocus->LostFocus();
   755 			}
   756 		if (newFocusedScreen)
   757 			{
   758 			CWsTop::SetCurrentFocusScreen(newFocusedScreen);
   759 			newFocus->ReceivedFocus();
   760 			}
   761 		else if (iCurrentFocus && (focusedScreen || CWsTop::MultiFocusPolicy()))
   762 			{
   763 			iCurrentFocus->ReceivedFocus();
   764 			}
   765 		TWsPointer::UpdatePointerCursor();
   766 		TWindowServerEvent::SendFocusChangedEvents();
   767 		}
   768 	TWindowServerEvent::SendGroupListChangedEvents();
   769 	}
   770 
   771 void CScreen::RemoveFromDefaultOwningList(CWsWindowGroup *aDestroyedGroup)
   772 	{
   773 	for (CWsWindowGroup **group=&iDefaultOwningWindow;*group;group=(*group)->NextDefaultOwningWindowPtr())
   774 		{
   775 		if (*group==aDestroyedGroup)
   776 			{
   777 			*group=*aDestroyedGroup->NextDefaultOwningWindowPtr();
   778 			break;
   779 			}
   780 		}
   781 	}
   782 
   783 void CScreen::SetDefaultOwningWindow(CWsWindowGroup *aGroup)
   784 	{
   785 	RemoveFromDefaultOwningList(aGroup);
   786 	aGroup->SetNextDefaultOwningWindow(iDefaultOwningWindow);
   787 	iDefaultOwningWindow=aGroup;
   788 	}
   789 
   790 void CScreen::GetScanLine(const TWsSdCmdGetScanLine *aGetScanLine)
   791 	{
   792 	TRgb buf[EGetScanLineBufLen];
   793 	TPtr8 des((TUint8 *)&buf[0],EGetScanLineBufLen*sizeof(TRgb));
   794 	TPoint pos(aGetScanLine->pos);
   795 	TInt read=0;
   796 	TInt len=(des.MaxLength()*EGetScanLineBufLen)/CFbsBitmap::ScanLineLength(EGetScanLineBufLen,aGetScanLine->dispMode);
   797 	if (aGetScanLine->len < 0 || (CFbsBitmap::ScanLineLength(aGetScanLine->len, aGetScanLine->dispMode) >
   798 									CWsClient::CurrentClient()->ClientMessage().GetDesMaxLength(1)))
   799 		{
   800 		CWsClient::PanicCurrentClient(EWservPanicInvalidParameter);
   801 		}
   802 	FOREVER
   803 		{
   804 		if ((aGetScanLine->len-read)<len)
   805 			len=aGetScanLine->len-read;
   806 		iScreenDevice->GetScanLine(des,pos,len,aGetScanLine->dispMode);
   807 		CWsClient::ReplyBuf(des);
   808 		read+=len;
   809 		if (read==aGetScanLine->len)
   810 			break;
   811 		pos.iX+=len;
   812 		}
   813 	}
   814 
   815 void CScreen::MaxNumColors(TInt& aColors,TInt& aGrays)
   816 	{
   817 	aGrays=0;
   818 	aColors=TDisplayModeUtils::NumDisplayModeColors(DisplayMode());
   819 	if (!TDisplayModeUtils::IsDisplayModeColor(DisplayMode()))
   820 		{
   821 		aGrays=aColors;
   822 		aColors=0;
   823 		}
   824 	}
   825 
   826 #define MODE_TO_FLAG(x) 1<<(x-1)
   827 #define ROTATION_TO_FLAG(x) 1<<x
   828 const TUint allRotationsMask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationNormal)
   829 						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated90)
   830 						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated180)
   831 						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated270);
   832 const TUint KRotation0_180Mask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationNormal)
   833 						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated180);
   834 const TUint KRotation90_270Mask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated90)
   835 						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated270);
   836 TInt CScreen::ColorModesFlag()
   837 	{
   838 	return MODE_TO_FLAG(DisplayMode());
   839 	}
   840 
   841 void CScreen::UpdateDsa()
   842 	{
   843 	if(iDsaDevice)
   844 		{
   845 		#if defined(__WINS__) && defined(_DEBUG)
   846 			if (iDebugWin)
   847 				iDebugWin->Refresh(iDsaDevice->SizeInPixels(), iDsaDevice->DisplayMode(), iDsaDevice->Bits());
   848 		#endif
   849 		
   850 		iDsaDevice->Update();
   851 		}
   852 	}
   853 
   854 const CGraphicsDeviceMap& CScreen::DeviceMap() const
   855 	{
   856 	return *iDeviceMap;
   857 	}
   858 
   859 const MWsScreenDevice& CScreen::ScreenDevice() const
   860 	{
   861 	return *iScreenDevice; 
   862 	}
   863 
   864 void CScreen::SetPointerCursorArea(TInt aMode,const TRect& aRect)
   865 	{
   866 	(*iModes)[aMode]->iPointerCursorArea=aRect;
   867 	(*iModes)[aMode]->iFlags |= EClientDefinedDigitiserArea;
   868 	TWsPointer::SetPointerCursorPos(TWsPointer::PointerCursorPos());
   869 	}
   870 
   871 CFbsBitGc::TGraphicsOrientation CScreen::Orientation() const
   872 	{
   873   	WS_ASSERT_DEBUG(IsValidScreenSizeMode(iScreenSizeMode),EWsPanicInvalidScreenSizeMode);
   874 	return (*iModes)[iScreenSizeMode]->iRotation;
   875 	}
   876 
   877 TRect CScreen::DrawableArea() const
   878 	{
   879 	TRect drawRect=iScreenDevice->SizeInPixels();
   880 	if (iDisplayMapping)
   881 		{
   882 		iDisplayMapping->MapCoordinates(EFullScreenSpace,drawRect,EApplicationSpace,drawRect);                   
   883 		}
   884 	return drawRect;
   885 	}
   886 
   887 TClientPanic CScreen::SetModeRotation(TInt aMode,CFbsBitGc::TGraphicsOrientation aRotation)
   888 	{
   889 	if (!IsValidScreenSizeMode(aMode))
   890 		return EWservPanicScreenModeNumber;
   891 	TSizeMode& mode=*(*iModes)[aMode];
   892 	if (!(ROTATION_TO_FLAG(aRotation)&mode.iAlternativeRotations))
   893 		return EWservPanicRotation;
   894 	CFbsBitGc::TGraphicsOrientation oldRotation=mode.iRotation;
   895 	mode.iRotation=aRotation;
   896 	CWsWindowGroup::NewOrientation(aMode,aRotation, iRootWindow);
   897 	if (aMode==ScreenSizeMode())
   898 		{
   899 		if(iDisplayPolicy && iDisplayControl && (mode.iAlternativeRotations == allRotationsMask))
   900 			{
   901 			//square mode supports all 4 rotations. We'd better do a complete Setconfiguration
   902 			//all parameters are recalculated for 90 degree rotation
   903 			//The most important one is the offset, since we need to re-center the appmode and it's a policy behaviour
   904 			TDisplayConfiguration config;
   905 			iDisplayControl->GetConfiguration(config);
   906 			//update rotation
   907 			config.SetRotation((TDisplayConfiguration::TRotation)aRotation);
   908 			SetConfiguration(config);
   909 			}
   910 		else if (!UpdateOrientation())
   911 			{
   912 			// Roll back earlier change
   913 			mode.iRotation=oldRotation;
   914 			CWsWindowGroup::NewOrientation(aMode, oldRotation, iRootWindow);
   915 			}
   916 		}
   917 	return EWservNoPanic;
   918 	}
   919 
   920 void CScreen::CycleDisplaySize()
   921 	{
   922 	TInt newMode = iScreenSizeMode;
   923 	TSizeMode* sizeMode = NULL;
   924 	do
   925 		{
   926 		newMode = (newMode+1)%iModes->Count();
   927 		sizeMode = (*iModes)[newMode];
   928 		}
   929 	while (sizeMode==NULL);
   930 	doSetScreenMode(newMode);
   931 	}
   932 
   933 void CScreen::doSetScreenMode(TInt aMode,TBool aInsideStartup)
   934 	{
   935   	WS_ASSERT_DEBUG(IsValidScreenSizeMode(aMode),EWsPanicInvalidScreenSizeMode);
   936  	TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, aMode));
   937 	
   938 	if (iDisplayControl && iDisplayPolicy)
   939 		{
   940 		TDisplayConfiguration config;
   941 		TRect sizeModePosition;
   942 		TInt error;
   943 		TSizeMode& mode=*(*iModes)[aMode];
   944 		config.SetRotation((TDisplayConfiguration::TRotation)mode.iRotation);
   945 		//This will return suitable config when display is connected
   946 		//and UI size equal to smallest appmode when disconnected
   947 		error = iDisplayPolicy->GetSizeModeConfiguration(aMode,config,sizeModePosition);
   948 		//set appmode in policy
   949 		if (iDisplayMapping)
   950 			{
   951 			iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
   952 			}
   953 		MWsScene::TSceneRotation oldRotation;
   954 		TSize newUiSize;
   955 		config.GetResolution(newUiSize);
   956 		if (error == KErrNone)
   957 			{
   958 			oldRotation = iScene->SceneRotation();
   959 			//The config info will always be set in policy. If display is connected, policy will have
   960 			//correct composition, Ui and app res. otherwise Ui and appmode will be both set to smallest
   961 			//app mode, composition will be set to zero
   962 			if (iFlags&EHasDynamicSizeModes)
   963 				{
   964 				error = iFallbackMap->Resize(newUiSize);
   965 				}
   966 			if (error == KErrNone)
   967 				{
   968 				error = iDisplayControl->SetConfiguration(config);
   969 				}
   970 			}
   971 		if (error == KErrNone)
   972 			{
   973 			UpdateDynamicScreenModes();
   974 			AbortAllDirectDrawing(RDirectScreenAccess::ETerminateScreenMode);
   975 
   976 			if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
   977 				{
   978 				iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
   979 				}
   980 
   981 			iScreenSizeMode=aMode;
   982 			//This could fail (to set orientation on the context, or to register the rotated DSA).
   983 			//Previously, the update rotation was deferred, leaving the size mode out of step with the actual rotation
   984 			//It also returns false if the orientation is "intentionally" not modified.
   985 			(void) UpdateOrientation(&oldRotation);	
   986 			
   987 			//SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
   988             //on a screen with touch input.
   989             //SetDigitiserAreas(newUiSize);
   990 			
   991 			CWsWindowGroup::SetScreenDeviceValidStates(this);
   992 			
   993 			if (!aInsideStartup)
   994 				{
   995 				iWindowElementSet->ResubmitAllElementExtents();
   996 				//TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
   997 				if(BlankScreenOnRotation())
   998 					{
   999 					iRootWindow->ClearDisplay();
  1000 					}
  1001 	
  1002 				CWsTop::ClearAllRedrawStores();
  1003 				DiscardAllSchedules();
  1004 				iRootWindow->InvalidateWholeScreen();
  1005 				}
  1006 			}
  1007 		}
  1008 	else
  1009 		{
  1010 		if (iDisplayMapping)
  1011 			{
  1012 			TRect sizeModePosition;
  1013 			TRAPD(err,sizeModePosition=TRect(OriginL(aMode),ScreenModeSizeInPixelsL(aMode)));
  1014 			if (err==KErrNone)
  1015 				{
  1016 				iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
  1017 				}
  1018 			}
  1019 		if (!aInsideStartup && (*iModes)[aMode]->iOrigin != (*iModes)[iScreenSizeMode]->iOrigin)
  1020 			{
  1021 			iWindowElementSet->ResubmitAllElementExtents();
  1022 			if ((*iModes)[aMode]->iRotation == (*iModes)[iScreenSizeMode]->iRotation)
  1023 				{
  1024 				//TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
  1025 				if(BlankScreenOnRotation())
  1026 					{
  1027 					iRootWindow->ClearDisplay();
  1028 					}
  1029 	
  1030 				CWsTop::ClearAllRedrawStores();
  1031 				DiscardAllSchedules();
  1032 				iRootWindow->InvalidateWholeScreen();
  1033 				}
  1034 			}
  1035 		iScreenSizeMode=aMode;
  1036 		//This could fail (to set orientation on the context, or to register the rotated DSA).
  1037 		//Previously, the update rotation was deferred, leaving the size mode out of step with the actual rotation
  1038 		//It also returns false if the orientation is not modified.
  1039 		(void)UpdateOrientation();
  1040 		CWsWindowGroup::SetScreenDeviceValidStates(this);
  1041 		}
  1042 	TWindowServerEvent::SendScreenDeviceChangedEvents(this);
  1043 	ResetFocus(NULL);
  1044 	}
  1045 
  1046 void CScreen::CycleOrientation()
  1047 	{
  1048   	WS_ASSERT_DEBUG(IsValidScreenSizeMode(iScreenSizeMode),EWsPanicInvalidScreenSizeMode);
  1049 	TSizeMode& currentSizeMode=*(*iModes)[iScreenSizeMode];
  1050 	TUint rotations=currentSizeMode.iAlternativeRotations;
  1051 	TInt currentRotation=currentSizeMode.iRotation;
  1052 	TInt rotation=currentRotation+1;
  1053 	while (rotation!=currentRotation)
  1054 		{
  1055 		if (rotation>CFbsBitGc::EGraphicsOrientationRotated270)
  1056 			rotation=CFbsBitGc::EGraphicsOrientationNormal;
  1057 		if (ROTATION_TO_FLAG(rotation)&rotations)
  1058 			break;
  1059 		++rotation;
  1060 		}
  1061 	if (rotation==currentRotation)
  1062 		{
  1063 		if (rotation>CFbsBitGc::EGraphicsOrientationRotated90)
  1064 			rotation-=2;
  1065 		else
  1066 			rotation+=2;
  1067 		}
  1068 	currentSizeMode.iRotation=REINTERPRET_CAST(CFbsBitGc::TGraphicsOrientation&,rotation);
  1069 	CWsWindowGroup::NewOrientation(iScreenSizeMode,currentSizeMode.iRotation, iRootWindow);
  1070 
  1071 	if (!UpdateOrientation())
  1072 		{
  1073 		// Roll back earlier changes
  1074 		currentSizeMode.iRotation=REINTERPRET_CAST(CFbsBitGc::TGraphicsOrientation&,currentRotation);
  1075 		CWsWindowGroup::NewOrientation(iScreenSizeMode, currentSizeMode.iRotation, iRootWindow);
  1076 		}
  1077 	}
  1078 
  1079 /**
  1080  * This method is called either when switching screen size mode, or when the 
  1081  * orientation of the currently active screen size mode is changed.
  1082  */
  1083 TBool CScreen::UpdateOrientation(MWsScene::TSceneRotation* aOldRotation)
  1084 	{
  1085 	CFbsBitGc::TGraphicsOrientation gcOrientation = Orientation();
  1086 
  1087 	MWsScene::TSceneRotation oldRotation = aOldRotation? (*aOldRotation):(iScene->SceneRotation());
  1088 	MWsScene::TSceneRotation newRotation = GcToScreen(gcOrientation);
  1089 	TDeviceOrientation newDeviceOrientation = GcToDevice(gcOrientation);
  1090 
  1091 	// We have to disable the text cursor here while we are still in the
  1092 	// same orientation or offset as we drew it.
  1093 	RWsTextCursor* cursor = CWsTop::CurrentTextCursor();
  1094 	if (cursor)
  1095 		{
  1096 		cursor->Disable();
  1097 		}
  1098 	
  1099 // Some of this method has to be done when changing mode even if not changing rotation
  1100 	TBool rotating=(oldRotation != newRotation);
  1101 	if (rotating)
  1102 		{
  1103 	
  1104 		// Set the new screen rotation and update the UI element extent
  1105 		if (iScene->SetSceneRotation(newRotation) != KErrNone)
  1106 			return EFalse;
  1107 		// Set the new orientation for the DSA device and update the DSA surface
  1108 		if(iDsaDevice)
  1109 			{
  1110 			iDsaDevice->SetDeviceOrientation(newDeviceOrientation);
  1111 	
  1112 			TSurfaceId newSurface;
  1113 			iDsaDevice->GetSurface(newSurface);
  1114 			TInt errRegister = iScene->RegisterSurface(newSurface);
  1115 			WS_ASSERT_DEBUG(KErrNone == errRegister, EWsPanicDirectScreenAccess);
  1116 			// This will remove all the DSA elements from the scene
  1117             AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
  1118 	
  1119 			TInt errUnregister = iScene->UnregisterSurface(iDsaSurface);
  1120 			WS_ASSERT_DEBUG(KErrNone == errUnregister, EWsPanicDirectScreenAccess);
  1121 			iDsaSurface = newSurface;
  1122 			}
  1123 		
  1124 		//updaterotation should not fail after this point (no cleanup)
  1125 			
  1126         //update the last set config with the new rotation change so we don't incorrectly
  1127         //change the layer extents
  1128         if (iDisplayControl)
  1129             {
  1130             TDisplayConfiguration config;
  1131             config.SetRotation(static_cast<TDisplayConfiguration::TRotation>(newRotation));           
  1132             iConfigChangeNotifier->UpdateLastSetConfiguration(config);
  1133             }		
  1134 		
  1135 		TWservCrEvent crEvent(TWservCrEvent::EDeviceOrientationChanged,iScreenNumber,&gcOrientation);
  1136 		TWindowServerEvent::NotifyDrawer(crEvent);
  1137 		
  1138 		if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
  1139 			{
  1140 			iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
  1141 			}
  1142 		
  1143 		}
  1144 	
  1145 	iRootWindow->AdjustCoordsDueToRotation();
  1146 	if (rotating)
  1147 		{
  1148 		if(BlankScreenOnRotation())
  1149 			{
  1150 			iRootWindow->ClearDisplay();
  1151 			}
  1152 		
  1153 		CWsTop::ClearAllRedrawStores();	
  1154 		DiscardAllSchedules();
  1155 		iRootWindow->InvalidateWholeScreen();
  1156 		}
  1157 	return ETrue;
  1158 	}
  1159 
  1160 TPoint CScreen::PhysicalToLogical(TPoint aPhysicalPt)
  1161 	{
  1162 	const TSizeMode& mode=ScreenSizeModeData();
  1163 	TPoint logicalPt;
  1164 	if(!iDisplayMapping)
  1165 		{
  1166 		//old behaviour
  1167 		logicalPt=aPhysicalPt-mode.iOrigin;
  1168 		if (mode.iScreenScale.iWidth!=1)
  1169 			logicalPt.iX=(logicalPt.iX>=0 ? logicalPt.iX/mode.iScreenScale.iWidth : (logicalPt.iX-(mode.iScreenScale.iWidth-1))/mode.iScreenScale.iWidth);
  1170 		if (mode.iScreenScale.iHeight!=1)
  1171 			logicalPt.iY=(logicalPt.iY>=0 ? logicalPt.iY/mode.iScreenScale.iHeight : (logicalPt.iY-(mode.iScreenScale.iHeight-1))/mode.iScreenScale.iHeight);
  1172 		}
  1173 	else
  1174 		{
  1175 		//rect with dummy size for coordinates mapping purpose
  1176 		TRect rectInComp(aPhysicalPt, TSize(1,1));
  1177 		TRect rectInApp;
  1178 		iDisplayMapping->MapCoordinates(ECompositionSpace, rectInComp, EApplicationSpace,rectInApp);
  1179 		logicalPt = rectInApp.iTl;
  1180 		}
  1181 	
  1182 	return logicalPt;
  1183 	}
  1184 
  1185 void CScreen::LoadScreenSizesL(TSize aScreenSize)
  1186 	{
  1187 	_LIT(KWSERVNumScrSizeMode, "NUMSCREENMODES");
  1188 	TBool allowScrGap=WsIniFile->FindVar(iScreenNumber, KWSERVNumScrSizeMode, iNumScreenSizeModes);
  1189 	iModes=new(ELeave) RPointerArray<TInternalSizeMode>(1);
  1190 	WS_ASSERT_DEBUG(!allowScrGap || (allowScrGap && iNumScreenSizeModes>0), EWsPanicInvalidScreenSizeMode);
  1191 	TInt screenNum=0;
  1192 	FOREVER
  1193 		{
  1194 		++screenNum;
  1195 		TBuf<32> varNameWidth;
  1196 		TBuf<32> varNameHeight;
  1197 		_LIT(KWSERVScreenWidthPattern,"SCR_WIDTH%d");
  1198 		varNameWidth.Format(KWSERVScreenWidthPattern,screenNum);
  1199 		_LIT(KWSERVScreenHeightPattern,"SCR_HEIGHT%d");
  1200 		varNameHeight.Format(KWSERVScreenHeightPattern,screenNum);
  1201 		TSize screenSize;
  1202 		if (!WsIniFile->FindVar(iScreenNumber, varNameWidth, screenSize.iWidth) ||
  1203 			!WsIniFile->FindVar(iScreenNumber, varNameHeight, screenSize.iHeight))
  1204 			{
  1205 			if (allowScrGap && screenNum<=iNumScreenSizeModes)
  1206 				{
  1207 				iModes->AppendL(NULL);
  1208 				continue;
  1209 				}
  1210 			else
  1211 				break;
  1212 			}
  1213 		TInt flags=0;
  1214 		if (screenSize.iWidth==0 && screenSize.iHeight==0)
  1215 			{
  1216 			screenSize=aScreenSize;
  1217 			flags|=EHalDefault;
  1218 			}
  1219 		if (screenSize.iWidth==-1 && screenSize.iHeight==-1)
  1220 			{
  1221 			screenSize=aScreenSize;
  1222 			flags|=EDynamic;
  1223 			iFlags|=EHasDynamicSizeModes;
  1224 			}
  1225 		TInternalSizeMode* newSizeMode=new(ELeave) TInternalSizeMode(screenSize);
  1226 		newSizeMode->iFlags|=flags;
  1227 		CleanupStack::PushL(newSizeMode);
  1228 		iModes->AppendL(newSizeMode);
  1229 		CleanupStack::Pop(newSizeMode);
  1230 		++iNumSupportedScreenSizeModes;
  1231 		}
  1232 	// If sparse index is enabled and no screen size mode defined, all iModes entries will be NULL
  1233 	// Otherwise iModes will be empty
  1234 	if (iModes->Count()==0 || iNumSupportedScreenSizeModes==0)
  1235 		{
  1236 		TInternalSizeMode* defaultSizeMode=new(ELeave) TInternalSizeMode(aScreenSize);
  1237 		defaultSizeMode->iFlags|=EHalDefault;
  1238 		if (iModes->Count()>0)
  1239 			(*iModes)[0]=defaultSizeMode;
  1240 		else
  1241 			{
  1242 			CleanupStack::PushL(defaultSizeMode);
  1243 			iModes->AppendL(defaultSizeMode);
  1244 			CleanupStack::Pop(defaultSizeMode);
  1245 			}
  1246 		++iNumSupportedScreenSizeModes;
  1247 		}
  1248 	if (!allowScrGap)
  1249 		iNumScreenSizeModes=iNumSupportedScreenSizeModes;
  1250 	}
  1251 
  1252 void CScreen::LoadScreenRotationProperties(TInternalSizeMode& aMode, const TInt aModeIndex)
  1253 	{
  1254 	TBuf<32> varRotation;
  1255 	_LIT(KWSERVScreenRotationPattern,"SCR_ROTATION%d");
  1256 	varRotation.Format(KWSERVScreenRotationPattern,aModeIndex+1);		
  1257 	TInt rotation=CFbsBitGc::EGraphicsOrientationNormal;
  1258 	TUint allRotations=0;
  1259 	TPtrC rotList(NULL,0);
  1260 	if (WsIniFile->FindVar( iScreenNumber, varRotation,rotList))
  1261 		{
  1262 		TLex lex(rotList);
  1263 		TBool foundOne=EFalse;
  1264 		TInt rot;
  1265 
  1266 		while (!lex.Eos())
  1267 			{
  1268 			if (!FindNextValue(lex, rot))
  1269 				{
  1270 				break;
  1271 				}
  1272 			if (rot<0 || rot>360)
  1273 				{
  1274 				continue;
  1275 				}
  1276 			if (rot>4)
  1277 				{
  1278 				rot/=90;
  1279 				}
  1280 			if (!foundOne)
  1281 				{
  1282 				rotation=rot;
  1283 				foundOne=ETrue;
  1284 				}
  1285 			if (rot<=CFbsBitGc::EGraphicsOrientationRotated270)
  1286 				{
  1287 				allRotations|=ROTATION_TO_FLAG(rot);
  1288 				}
  1289 			}
  1290 		}
  1291 	if (allRotations==0)
  1292 		allRotations=ROTATION_TO_FLAG(rotation);
  1293 	const TInt KAllRotationsMask = 0xF;	//Used to keep the old behaviour
  1294 	WS_ASSERT_ALWAYS((ROTATION_TO_FLAG(rotation)&KAllRotationsMask)>0, EWsPanicFailedToInitialise);
  1295 	aMode.iRotation=reinterpret_cast<CFbsBitGc::TGraphicsOrientation&>(rotation);
  1296 	aMode.iAlternativeRotations=allRotations&KAllRotationsMask;
  1297 	}
  1298 
  1299 void CScreen::LoadScreenTwipsProperties(TInternalSizeMode& aMode, const TInt aModeIndex)
  1300 	{
  1301 	TBuf<32> varNameWidth;
  1302 	_LIT(KWSERVScreenTwipWidthPattern,"SCR_TWIP_WIDTH%d");
  1303 	varNameWidth.Format(KWSERVScreenTwipWidthPattern,aModeIndex+1);
  1304 	TBuf<32> varNameHeight;
  1305 	_LIT(KWSERVScreenTwipHeightPattern,"SCR_TWIP_HEIGHT%d");
  1306 	varNameHeight.Format(KWSERVScreenTwipHeightPattern,aModeIndex+1);
  1307 
  1308 	TSize twipsSize;
  1309 	TBool widthFound = WsIniFile->FindVar(iScreenNumber,varNameWidth,twipsSize.iWidth);
  1310 	TBool heightFound = WsIniFile->FindVar(iScreenNumber,varNameHeight,twipsSize.iHeight);	
  1311 	
  1312 	// if either of the width or height wsini reads has failed we need to generate default values
  1313 	switch(aMode.iRotation)
  1314 		{
  1315 		// CFbsBitGc::TGraphicsOrientation
  1316 		case CFbsBitGc::EGraphicsOrientationRotated90:	// deliberate drop-through
  1317 		case CFbsBitGc::EGraphicsOrientationRotated270:
  1318 			{
  1319 			// CFbsScreenDevice knows nothing about rotation, so we can't use it's PixelsTo Twips methods
  1320 			// So swap the axis here to use the correct twips per pixel ratio
  1321 			if (!widthFound)
  1322 				twipsSize.iWidth = DeviceMap().VerticalPixelsToTwips(aMode.iScreenSize.iWidth);
  1323 			if (!heightFound)
  1324 				twipsSize.iHeight = DeviceMap().HorizontalPixelsToTwips(aMode.iScreenSize.iHeight);	
  1325 			break;
  1326 			}
  1327 		case CFbsBitGc::EGraphicsOrientationNormal:		// deliberate drop-through
  1328 		case CFbsBitGc::EGraphicsOrientationRotated180:
  1329 			if (!widthFound)
  1330 				twipsSize.iWidth = DeviceMap().HorizontalPixelsToTwips(aMode.iScreenSize.iWidth);
  1331 			if (!heightFound)
  1332 				twipsSize.iHeight = DeviceMap().VerticalPixelsToTwips(aMode.iScreenSize.iHeight);
  1333 			break;
  1334 		default:
  1335 			RDebug::Print(_L("** CScreen::LoadScreenTwipsProperties Panic"));
  1336 			WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
  1337 			break;			
  1338 		}
  1339 	if (widthFound&&heightFound)
  1340 		{
  1341 		aMode.iFlags|=this->ETwipsSpecified;
  1342 		}
  1343 	aMode.iScreenTwipsSize=twipsSize;
  1344 	}
  1345 
  1346 
  1347 void CScreen::LoadScreenSizeProperties(TDisplayMode aDefaultDisplayMode)
  1348 	{
  1349 	for(TInt sizeLoop=0;sizeLoop<iModes->Count();sizeLoop++)
  1350 		{
  1351 		TInternalSizeMode* modePtr=(*iModes)[sizeLoop];
  1352 		if (!modePtr)
  1353 			continue;
  1354 		TInternalSizeMode& mode=*modePtr;
  1355 		TBuf<32> varDisplayMode;
  1356 		_LIT(KWSERVScreenDisplayModePattern,"SCR_WINDOWMODE%d");
  1357 		
  1358 		varDisplayMode.Format(KWSERVScreenDisplayModePattern,sizeLoop+1);
  1359 		mode.iScreenScale.iWidth=1;
  1360 		mode.iScreenScale.iHeight=1;
  1361 
  1362 		TBuf<32> varLeft;
  1363 		TBuf<32> varTop;
  1364 		_LIT(KWSERVScreenLeftPattern,"SCR_LEFT%d");
  1365 		_LIT(KWSERVScreenTopPattern,"SCR_TOP%d");
  1366 		varLeft.Format(KWSERVScreenLeftPattern,sizeLoop+1);
  1367 		varTop.Format(KWSERVScreenTopPattern,sizeLoop+1);
  1368 		if (!WsIniFile->FindVar( iScreenNumber, varLeft,mode.iOrigin.iX))
  1369 			mode.iOrigin.iX=0;
  1370 		if (!WsIniFile->FindVar( iScreenNumber, varTop,mode.iOrigin.iY))
  1371 			mode.iOrigin.iY=0;
  1372 			
  1373 		TPtrC displayModeName(NULL,0);
  1374 		mode.iDefaultDisplayMode = aDefaultDisplayMode;
  1375 		// must know rotation before parsing twips
  1376 		LoadScreenRotationProperties(mode, sizeLoop);
  1377 		LoadScreenTwipsProperties(mode, sizeLoop);
  1378 		
  1379 		
  1380 		if(mode.iScreenSize.iWidth == mode.iScreenSize.iHeight && mode.iAlternativeRotations == allRotationsMask)
  1381 			{
  1382 			//square appmode with all four rotations allowed must have square twipsize
  1383 			if((mode.iFlags&ETwipsSpecified) && mode.iScreenTwipsSize.iWidth != mode.iScreenTwipsSize.iHeight)
  1384 				{
  1385 				RDebug::Print(_L("**Panic: Square appmode with all four rotations must have square twip size"));
  1386 				WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
  1387 				}
  1388 			//square appmode with all four rotations allowed must have square offset
  1389 			if(mode.iOrigin.iX != mode.iOrigin.iY)
  1390 				{
  1391 				RDebug::Print(_L("**Panic: Square appmode with all four rotations must have square offset"));
  1392 				WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
  1393 				}
  1394 			}
  1395 		else
  1396 			{
  1397 			//Everything else is treated as rectangle appmode. Square appmode not supporting all 4 rotations is considered
  1398 			//as rectangle appmode as well. Rectangle appmode suports 2 rotations at most (0 and 180, or 90 and 270)
  1399 			//first rotation of the appmode is taken to apply the corresponding rotation mask
  1400 			if(!((mode.iAlternativeRotations&KRotation0_180Mask) == mode.iAlternativeRotations
  1401 					|| (mode.iAlternativeRotations&KRotation90_270Mask) == mode.iAlternativeRotations))
  1402 				{
  1403 				RDebug::Print(_L("**Panic_DEBUG: non square appmode can only define (0,180) or (90,270) rotations"));
  1404 				WS_PANIC_DEBUG(EWsPanicFailedToInitialise);
  1405 				//in relase build, no panic, just correct the rotations set
  1406 
  1407 				}
  1408 			//correct the rotations set
  1409 			mode.iAlternativeRotations &= ((ROTATION_TO_FLAG(mode.iRotation) & KRotation0_180Mask)? KRotation0_180Mask:
  1410 												KRotation90_270Mask);
  1411 			}
  1412 			
  1413 		}
  1414 //
  1415 	TInt intForFindVar=0;
  1416 	_LIT(KWSERVIniFileVarSizeMode,"SIZE_MODE");
  1417 	WsIniFile->FindVar( iScreenNumber, KWSERVIniFileVarSizeMode,intForFindVar);
  1418 	iSizeEnforcementMode=(TScreenModeEnforcement)intForFindVar;
  1419 	}
  1420 
  1421 void CScreen::SetDigitiserAreas(const TSize& aUiSize)
  1422 	{  //aUiSize should be the unrotated current ui size
  1423 	//SetDigitiserAreas needs revisiting if/when we support dynamic resolutions on a screen 
  1424     //with touch input. It is not known how digitiser coordinates will be represented if the 
  1425     //physical display resolution changes. Currently digitisers are only supported on screen 0, 
  1426     //and dynamic resolution only applies to higher screen numbers on real hardware.
  1427 	for(TInt sizeLoop=0;sizeLoop<iModes->Count();sizeLoop++)
  1428 		{
  1429 		TInternalSizeMode* modePtr=(*iModes)[sizeLoop];
  1430 		if (!modePtr)
  1431 			continue;
  1432 		TInternalSizeMode& mode=*modePtr;
  1433 		if(mode.iFlags & EClientDefinedDigitiserArea)
  1434 			{
  1435 			//if it's client set, keep it unchanged.
  1436 			continue;
  1437 			}
  1438 		switch (mode.iRotation)
  1439 			{
  1440 			case CFbsBitGc::EGraphicsOrientationNormal:
  1441 				mode.iPointerCursorArea=iDigitiserArea;
  1442 				continue;
  1443 			case CFbsBitGc::EGraphicsOrientationRotated90:
  1444 				mode.iPointerCursorArea.SetRect(iDigitiserArea.iTl.iY,aUiSize.iWidth-iDigitiserArea.iBr.iX,
  1445 																iDigitiserArea.iBr.iY,aUiSize.iWidth-iDigitiserArea.iTl.iX);
  1446 				break;
  1447 			case CFbsBitGc::EGraphicsOrientationRotated180:
  1448 				mode.iPointerCursorArea.SetRect(-(iDigitiserArea.iBr-aUiSize),-(iDigitiserArea.iTl-aUiSize));
  1449 				break;
  1450 			case CFbsBitGc::EGraphicsOrientationRotated270:
  1451 				mode.iPointerCursorArea.SetRect(aUiSize.iHeight-iDigitiserArea.iBr.iY,iDigitiserArea.iTl.iX,
  1452 				        aUiSize.iHeight-iDigitiserArea.iTl.iY,iDigitiserArea.iBr.iX);
  1453 				break;
  1454 			default:
  1455 			    WS_PANIC_ALWAYS(EWsPanicInvalidRotation);
  1456 			}
  1457 		}
  1458 	}
  1459 
  1460 void CScreen::GetScreenSizeAndRotation(TPixelsTwipsAndRotation &aSar, TInt aScreenMode)
  1461 	{
  1462 	TSizeMode& mode=*(*iModes)[aScreenMode];
  1463 	aSar.iRotation=mode.iRotation;
  1464 	aSar.iPixelSize=mode.iScreenSize;
  1465 	aSar.iTwipsSize=mode.iScreenTwipsSize;
  1466 	if (aSar.iTwipsSize.iWidth==0)
  1467 		{
  1468 		aSar.iTwipsSize.iWidth  = iDeviceMap->HorizontalPixelsToTwips(aSar.iPixelSize.iWidth);
  1469 		aSar.iTwipsSize.iHeight = iDeviceMap->VerticalPixelsToTwips(aSar.iPixelSize.iHeight);
  1470 		}
  1471 	}
  1472 
  1473 void CScreen::GetScreenSizeAndRotation(TPixelsAndRotation &aSar, TInt aScreenMode)
  1474 	{
  1475 	TSizeMode& mode=*(*iModes)[aScreenMode];
  1476 	aSar.iRotation=mode.iRotation;
  1477 	aSar.iPixelSize=mode.iScreenSize;
  1478 	}
  1479 
  1480 TBool CScreen::SetScreenModeEnforcement(TInt aMode)
  1481 	{
  1482 	if (aMode<0 || aMode>ESizeEnforcementPixelsTwipsAndRotation)
  1483 		return EFalse;
  1484 	TScreenModeEnforcement newMode=(TScreenModeEnforcement)aMode;
  1485 	if (newMode!=iSizeEnforcementMode)
  1486 		{
  1487 		iSizeEnforcementMode=newMode;
  1488 		CWsWindowGroup::SetScreenDeviceValidStates(this);
  1489 		ResetFocus(NULL);
  1490 		}
  1491 	return ETrue;
  1492 	}
  1493 
  1494 void CScreen::IncContrast()
  1495 	{
  1496 	TInt contrast;
  1497 	if (iMaxContrast<0)			//If failed to get it sofar get it again
  1498 		TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrastMax,iMaxContrast));
  1499 	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrast,contrast)))
  1500 		return;
  1501 	if (contrast==iMaxContrast)
  1502 		contrast=-1;
  1503 	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Set(iScreenNumber,HALData::EDisplayContrast,++contrast));
  1504 	}
  1505 
  1506 void CScreen::DecContrast()
  1507 	{
  1508 	TInt contrast;
  1509 	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrast,contrast)))
  1510 		return;
  1511 	if (contrast==0)
  1512 		{
  1513 		if (iMaxContrast<0 && TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast,
  1514 															HAL::Get(iScreenNumber,HALData::EDisplayContrastMax,iMaxContrast)))
  1515 			return;
  1516 		contrast=iMaxContrast+1;
  1517 		}
  1518 	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Set(iScreenNumber,HALData::EDisplayContrast,--contrast));
  1519 	}
  1520 
  1521 void CScreen::IncBrightness()
  1522 	{
  1523 	TInt brightness;
  1524 	if (iMaxBrightness<0)			//If failed to get it sofar get it again
  1525 		TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightnessMax,iMaxBrightness));
  1526 	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightness,brightness)))
  1527 		return;
  1528 	if (brightness==iMaxBrightness)
  1529 		brightness=-1;
  1530 	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(iScreenNumber,HALData::EDisplayBrightness,++brightness));
  1531 	}
  1532 
  1533 void CScreen::DecBrightness()
  1534 	{
  1535 	TInt brightness;
  1536 	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightness,brightness)))
  1537 		return;
  1538 	if (brightness==0)
  1539 		{
  1540 		if (iMaxBrightness<0 && TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight,
  1541 													HAL::Get(iScreenNumber,HALData::EDisplayBrightnessMax,iMaxBrightness)))
  1542 			return;
  1543 		brightness=iMaxBrightness+1;
  1544 		}
  1545 	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(iScreenNumber,HALData::EDisplayBrightness,--brightness));
  1546 	}
  1547 TInt CScreen::GetScreenSizeModeList(RArray<TInt>& aList) const
  1548 	{
  1549 	aList.Reset();
  1550 	TInt numModes=iNumScreenSizeModes;
  1551 	TInt err=aList.Reserve(numModes);
  1552 	if (err!=KErrNone)
  1553 		return err;
  1554 	TInt index;
  1555 	for (index=0; index<numModes; ++index)
  1556 		{
  1557 		TSizeMode* modePtr=(*iModes)[index];
  1558 		if (modePtr)
  1559 			aList.Append(index);	//Can't fail due to reserve
  1560 		}
  1561 	TInt count=aList.Count();
  1562 	return count;
  1563 	}
  1564 
  1565 TInt CScreen::GetScreenSizeModeListL()
  1566 	{
  1567 	RArray<TInt> list;
  1568 	CleanupClosePushL(list);
  1569 	TInt count=GetScreenSizeModeList(list);
  1570 		User::LeaveIfError(count);
  1571 	CWsClient::ReplyBuf(&list[0], count*sizeof(TInt));
  1572 	CleanupStack::PopAndDestroy(&list);
  1573 	return count;
  1574 	}
  1575 
  1576 void CScreen::SetInitialScreenSizeModeAndRotation()
  1577 	{	//Set first app mode that supports current supported rotations if available
  1578 	TInt index;
  1579 	TInt firstMode = -1;
  1580 	TInt bestMode = -1;
  1581 	// Since all screen rotation modes are supported.
  1582 	TBitFlags32 rotationFlags = CFbsBitGc::EGraphicsOrientationNormal|CFbsBitGc::EGraphicsOrientationRotated90|CFbsBitGc::EGraphicsOrientationRotated180|CFbsBitGc::EGraphicsOrientationRotated270;
  1583 	for (index=0; index<iModes->Count(); ++index)
  1584 		{
  1585 		TSizeMode* modePtr=(*iModes)[index];
  1586 		if (modePtr)
  1587 			{
  1588 			if (firstMode == -1)
  1589 				{
  1590 				firstMode = index;
  1591 				}
  1592 			if (rotationFlags.IsSet((TInt)modePtr->iRotation))
  1593 				{
  1594 				bestMode = index;
  1595 				break;
  1596 				}
  1597 			}
  1598 		}
  1599 
  1600 	if (bestMode != -1)	//found a mode that supports current supported rotations
  1601 		{
  1602 		iScreenSizeMode = bestMode;
  1603 		}
  1604 	else
  1605 		{
  1606 		if (firstMode != -1)	//could only find a mode that doesnt support current supported rotations
  1607 			{
  1608 			iScreenSizeMode = firstMode;
  1609 			}
  1610 		else
  1611 			{
  1612 			return;	//couldn't find a mode at all
  1613 			}
  1614 		}
  1615 	if(iDisplayPolicy)
  1616 		{
  1617 		iDisplayPolicy->SetLastAppMode(iScreenSizeMode);
  1618 		}
  1619 
  1620     SetDigitiserAreas(iScreenDevice->SizeInPixels()); //unrotated size in pixels
  1621     
  1622     // Here we are mixing CFbsBitGc::TGraphicsOrientation with MWsScene::TSceneRotation
  1623     // As they both have same values it is fine for now
  1624     iScene->SetSceneRotation(GcToScreen(ScreenSizeModeData().iRotation)); //set rotation
  1625 	}
  1626 
  1627 
  1628 TDisplayMode CScreen::FirstDefaultDisplayMode() const
  1629 	{
  1630 	TInt mode=-1;
  1631 	while ((*iModes)[++mode]==NULL)
  1632 		{
  1633 		WS_ASSERT_DEBUG(mode<iModes->Count()-1,EWsPanicInvalidScreenSizeMode);
  1634 		}
  1635 	return((*iModes)[mode]->iDefaultDisplayMode);
  1636 	}
  1637 
  1638 void CScreen::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
  1639 	{
  1640 	iRedraw->AddRedrawRegion(aRegion, aSchedule, aDepth);
  1641 	}
  1642 
  1643 void CScreen::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
  1644 	{
  1645 	iRedraw->ScheduleRender(aFromNow);
  1646 	}
  1647 
  1648 void CScreen::DoRedrawNow()
  1649 	{
  1650 	iRedraw->DoRedrawNow();
  1651 	}
  1652 
  1653 // See CScreenRedraw::IsUpdatePending() for important notes on usage.
  1654 void CScreen::RedrawNowIfPending()
  1655 	{
  1656 	if(iRedraw->IsUpdatePending())
  1657 		DoRedrawNow();
  1658 	}
  1659 
  1660 TBool CScreen::IsQuickFadeScheduled( CWsWindow* aWin ) const
  1661 	{
  1662 	return iRedraw->IsQuickFadeScheduled( aWin );
  1663 	}
  1664 
  1665 void CScreen::RemoveFromQuickFadeList( CWsWindow* aWin )
  1666 	{
  1667 	iRedraw->RemoveFromQuickFadeList( aWin );
  1668 	}
  1669 
  1670 void CScreen::AcceptFadeRequest( CWsWindow* aWin, TBool aIsFaded )
  1671 	{
  1672 	iRedraw->AcceptFadeRequest( aWin, aIsFaded );
  1673 	}
  1674 
  1675 // implementing MWsScreen
  1676 
  1677 const TTime& CScreen::Now() const
  1678 	{
  1679 	return iRedraw->Now();
  1680 	}
  1681 
  1682 void CScreen::ScheduleAnimation(TAnimType aType, const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& aFreq,const TTimeIntervalMicroSeconds& aStop, CWsWindow* aWindow)
  1683 	{
  1684 	iRedraw->ScheduleAnimation(aType, aRect,aFromNow,aFreq,aStop, aWindow);
  1685 	}
  1686 
  1687 TBool CScreen::IsScheduled(TAnimType aType, const TRect& aRect, CWsWindow* aWindow) const
  1688 	{
  1689 	return iRedraw->IsScheduled(aType, aRect, aWindow);
  1690 	}
  1691 
  1692 void CScreen::OnAnimation(TRequestStatus* aFinished)
  1693 	{
  1694 	iRedraw->OnAnimation(aFinished);
  1695 	}
  1696 	
  1697 void CScreen::Redraw()
  1698 	{
  1699 	STACK_REGION bounds;
  1700 	bounds.AddRect(DrawableArea());
  1701 	AddRedrawRegion(bounds);
  1702 	bounds.Close();
  1703 	}
  1704 
  1705 TBool CScreen::RedrawInvalid(const TArray<TGraphicDrawerId>& aInvalid)
  1706 	{
  1707 	TBool wasDirty = EFalse;
  1708 	STACK_REGION bounds;
  1709 	bounds.AddRect(DrawableArea());
  1710 	STACK_REGION dirty;
  1711 	TWalkWindowTreeCalcInvalidGraphics calc(&bounds,dirty,aInvalid);
  1712 	if(calc.CreateSubRegion())
  1713 		{
  1714 		calc.CalcInvalid(*this);
  1715 		if(dirty.CheckError() || dirty.Count())
  1716 			{
  1717 			Redraw();
  1718 			wasDirty = ETrue;
  1719 			}
  1720 		calc.DestroyRegions();
  1721 		}
  1722 	dirty.Close();
  1723 	bounds.Close();
  1724 	return wasDirty;
  1725 	}
  1726 
  1727 /**
  1728  Overidding MWsObjectProvider
  1729 */
  1730 TAny* CScreen::ResolveObjectInterface(TUint aTypeId)
  1731 	{
  1732 	TAny* interface = NULL;
  1733 
  1734 	switch (aTypeId)
  1735 		{
  1736 		case MWsWindow::EWsObjectInterfaceId:
  1737 			interface = static_cast<MWsWindow*>(RootWindow());
  1738 			break;
  1739 		case MWsScreenConfigList::EWsObjectInterfaceId:
  1740 			interface = static_cast<MWsScreenConfigList*>(this);
  1741 			break;
  1742 		case MWsScreenConfig::EWsObjectInterfaceId:
  1743 			interface = static_cast<MWsScreenConfig*>(this);
  1744 			break;
  1745 		case MWsWindowTree::EWsObjectInterfaceId:
  1746 			interface = static_cast<MWsWindowTree*>(this);
  1747 		}
  1748 
  1749 	if (!interface)
  1750 		interface = iRedraw->ResolveObjectInterface(aTypeId);
  1751 
  1752 	return interface;
  1753 	}
  1754 
  1755 void CScreen::SendTree() const
  1756 	{
  1757 	if(!iWindowTreeObserver)
  1758 		return;
  1759 	
  1760 	TWalkWindowTreeSendState wtw(*iWindowTreeObserver);
  1761 	RootWindow()->WalkWindowTreeBackToFront(wtw, EVisitParentNodesFirst);
  1762 
  1763 	//Standard text cursors
  1764 	RWsTextCursor* cursor = CWsTop::CurrentTextCursor();
  1765 	if(cursor)
  1766 		cursor->SendState(*iWindowTreeObserver);
  1767 	
  1768 	//Floating Sprites	
  1769 	SpriteManager()->SendState(*iWindowTreeObserver);
  1770 	
  1771 	//Window Group Chains
  1772 	for(CWsWindowGroup *group=RootWindow()->Child(); group!=NULL; group=group->NextSibling())
  1773 		{
  1774 		group->SendStateWindowGroupChain(*iWindowTreeObserver);
  1775 		}
  1776 	}
  1777 
  1778 TDisplayMode CScreen::DisplayMode() const
  1779 	{
  1780 	return iScreenDevice->DisplayMode();
  1781 	}
  1782 
  1783 TSize CScreen::SizeInPixels() const
  1784 	{
  1785 	return iScreenDevice->SizeInPixels();
  1786 	}
  1787 
  1788 TSize CScreen::SizeInTwips() const
  1789 	{
  1790 	return iScreenDevice->SizeInTwips();
  1791 	}
  1792 
  1793 void CScreen::DiscardAllSchedules()
  1794 	{
  1795 	iRedraw->DiscardAllSchedules();
  1796 	}
  1797 
  1798 void CScreen::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
  1799 	{
  1800 	iRedraw->ScheduleRegionUpdate(aDefinitelyDirty);
  1801 	}
  1802 
  1803 TBool CScreen::IsDSAClientWindow( const CWsClientWindow* aWin ) const
  1804 	{
  1805 	TBool res = EFalse; 
  1806 	if ( ! iDirects.IsEmpty() )
  1807 		{
  1808 		TSglQueIter<CWsDirectScreenAccess> iter( (TSglQueBase&)iDirects );
  1809 		iter.SetToFirst();
  1810 		CWsDirectScreenAccess* dsa;
  1811 		while ( (dsa = iter++) != NULL && !res )
  1812 			{
  1813 			res =  (dsa->ClientWindow() == aWin) && (dsa->IsVisible());
  1814 			}
  1815 		}
  1816 	return res;	
  1817 	}
  1818 
  1819 /**
  1820 Update the UI element composition method based on whether
  1821 there are any externals surfaces present and the current display mode. 
  1822 If the method changes, recomposition is triggered.
  1823 */
  1824 void CScreen::UpdateCompositionMode()
  1825 	{
  1826 	// do nothing
  1827 	}
  1828 
  1829 void CScreen::ElementAdded()
  1830 	{
  1831 	UpdateCompositionMode();
  1832 	}
  1833 
  1834 void CScreen::ElementRemoved()
  1835 	{
  1836 	UpdateCompositionMode();
  1837 	}
  1838 
  1839 CRegisteredSurfaceMap* CScreen::SurfaceMap()
  1840 	{
  1841 	return iSurfaceMap;
  1842 	}
  1843 
  1844 void CScreen::InitializeSceneL()
  1845 	{
  1846 	// Ensure the surface is not valid to start with.
  1847 	iDsaSurface = TSurfaceId::CreateNullId();
  1848 	iWindowElementSet = CWindowElementSet::NewL(*iScene);
  1849 	}
  1850 
  1851 MWsElement* CScreen::CreateUiElementL(const TRect& aExtent)
  1852 	{
  1853 	MWsElement* pElement = iScene->CreateSceneElementL();
  1854 	
  1855 	TUint32 flags = 0;
  1856 	pElement->GetRenderStageFlags(flags);
  1857 	flags |= MWsElement::EElementIsIndirectlyRenderedUserInterface;
  1858 	pElement->SetRenderStageFlags(flags);
  1859 	    
  1860     iScene->InsertSceneElement(pElement, NULL);
  1861 
  1862 	pElement->SetDestinationRectangle(aExtent);
  1863 	pElement->SetSourceRectangle(aExtent);	//initial guess... updated by PositionUiElements
  1864 
  1865 	return pElement;
  1866 	}
  1867 
  1868 void CScreen::InitializeUiElementsL()
  1869 	{
  1870 	const TRect screenRect(iScreenDevice->SizeInPixels());
  1871 	MWsElement* pElement = CreateUiElementL(screenRect);
  1872 	
  1873 	if(HasAlpha())
  1874 		{
  1875 		TUint32 flags = 0;
  1876 		pElement->GetTargetRendererFlags(flags);
  1877 		flags |= MWsElement::EElementTransparencySource;
  1878 		pElement->SetTargetRendererFlags(flags);
  1879 		}
  1880 	}
  1881 
  1882 TInt CScreen::InitializeDsaSurface()
  1883 	{
  1884 	WS_ASSERT_DEBUG(iDsaSurface.IsNull(),EWsPanicInvalidOperation);
  1885 	iDsaDevice->GetSurface(iDsaSurface);
  1886 	// Currently Surface Manager does not recognise DSA surface IDs originating
  1887 	// from the Screen Driver.  This causes it to fail to register such
  1888 	// surfaces.  OpenWF should be amended to properly register DSA surfaces.
  1889 	iScene->RegisterSurface(iDsaSurface);
  1890 
  1891 	return KErrNone;
  1892 	}
  1893 
  1894 TSize CScreen::DSASizeInPixels() const
  1895 	{
  1896 	if(iDsaDevice)
  1897 		{
  1898 		return iDsaDevice->SizeInPixels();
  1899 		}
  1900 	else
  1901 		{
  1902 		return TSize(0,0);
  1903 		}
  1904 	}
  1905 
  1906 MWsTextCursor* CScreen::RenderStageTextCursor() const
  1907 	{
  1908 	return iRedraw->RenderStageTextCursor();
  1909 	}
  1910 
  1911 void CScreen::ClearDsaSurface(const TRect& area, const TRgb& color)
  1912 	{
  1913 	WS_ASSERT_DEBUG(iDsaGc, EWsPanicInvalidOperation);
  1914 	iDsaGc->SetBrushStyle(CFbsBitGc::ESolidBrush);
  1915 	iDsaGc->SetPenStyle(CFbsBitGc::ENullPen);
  1916 	iDsaGc->SetBrushColor(color);
  1917 	iDsaGc->DrawRect(area);
  1918 	iDsaDevice->Update();
  1919 	}
  1920 
  1921 void CScreen::ReleaseDsaScreenDevice()
  1922 	{
  1923 	//This function checks if any of the DSA currently active on the screen is actually used to draw
  1924 	//If not it unregister the DSA surface and destroys the iDsaDevice.
  1925 	//This function should be called only by a drawing DSA so a surface should be in place.
  1926 	iNumberDrawingDsa--;
  1927 	if(iNumberDrawingDsa == 0)
  1928 		{
  1929 		WS_ASSERT_DEBUG(!iDsaSurface.IsNull(),EWsPanicInvalidOperation);
  1930 		delete iDsaGc;
  1931 		iDsaGc = NULL;
  1932 		// Currently Surface Manager does not recognise DSA surface IDs originating
  1933 		// from the Screen Driver.  This causes it to fail to unregister such
  1934 		// surfaces.  OpenWF should be amended to properly register DSA surfaces.
  1935 		iScene->UnregisterSurface(iDsaSurface);
  1936 
  1937 		delete iDsaDevice;
  1938 		iDsaDevice = NULL;
  1939 		//the old surface Id is now meaningless
  1940 		iDsaSurface = TSurfaceId::CreateNullId();
  1941 		}
  1942 	}
  1943 
  1944 TInt CScreen::SetConfiguration(const TDisplayConfiguration& aConfigInput)
  1945 	{
  1946 	TInt reply = KErrNone;
  1947 	if(iDisplayControl)
  1948 		{
  1949 		TDisplayConfiguration config(aConfigInput);
  1950 		TRect sizeModePosition;
  1951 		if (iDisplayPolicy)
  1952 			{	//validate config and update to a valid hardware config
  1953 			reply = iDisplayPolicy->GetSizeModeConfiguration(iScreenSizeMode,config,sizeModePosition);
  1954 			if (reply >= KErrNone)
  1955 				{//set appmode in policy
  1956 				if (iDisplayMapping)
  1957 					{
  1958 					iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
  1959 					}
  1960 				}
  1961 			}
  1962 		else
  1963 			{	//exessive strategy: limit rotation agains curr app mode.
  1964 				//really we want the system to accept the rotation change regardless of the app mode.
  1965 			TDisplayConfiguration::TRotation newRot;
  1966 			if (aConfigInput.GetRotation(newRot))
  1967 				{	//This should cast between rotation enumertaions "properly"
  1968 				if (!(iModes[0][iScreenSizeMode]->iAlternativeRotations&(1<<newRot)))
  1969 					{
  1970 					reply=KErrArgument;
  1971 					}
  1972 				}
  1973 			}
  1974 		if (reply < KErrNone)
  1975 			{
  1976 			return reply;
  1977 			}
  1978 		MWsScene::TSceneRotation oldRotation;
  1979 		oldRotation = iScene->SceneRotation();
  1980 		TSize newUiSize;
  1981 		config.GetResolution(newUiSize);
  1982 		TDisplayConfiguration oldConfig;
  1983 		iDisplayControl->GetConfiguration(oldConfig);
  1984 		if(iFlags&EHasDynamicSizeModes)
  1985 			{
  1986 			reply = iFallbackMap->Resize(newUiSize);
  1987 			}
  1988 		if (reply >= KErrNone)
  1989 			{
  1990 			reply=iDisplayControl->SetConfiguration(config);
  1991 			}	
  1992 		if (reply==KErrNone)
  1993 			{
  1994 			TSize oldConfigRes;
  1995 			oldConfig.GetResolution(oldConfigRes);
  1996             TDisplayConfiguration newConfig;
  1997 			if (oldConfigRes.iWidth == 0 || oldConfigRes.iHeight == 0)
  1998 				{
  1999 				iDisplayControl->GetConfiguration(newConfig);
  2000 				RecalculateModeTwips(&newConfig);	//needs res and twips information
  2001 				}
  2002 			UpdateDynamicScreenModes();
  2003 			
  2004 			//update the last set config in the config change notifier to 
  2005 			//prevent SetConfiguration() from being called again!
  2006 			newConfig.ClearAll();
  2007 			iDisplayControl->GetConfiguration(newConfig);
  2008 			iConfigChangeNotifier->UpdateLastSetConfiguration(newConfig); 			
  2009 			
  2010 			TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, iScreenSizeMode));
  2011 			// This will remove all the DSA elements from the scene
  2012 			AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
  2013 			
  2014 			//SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
  2015 			//on a screen with touch input.
  2016 			//SetDigitiserAreas(newUiSize);
  2017 			
  2018 			//failure here should only be because of DSA orientation change failure, which shouldn't happen, either.
  2019 			//Or there may be no change to do.
  2020 			(void)UpdateOrientation(&oldRotation);
  2021 			
  2022 			iWindowElementSet->ResubmitAllElementExtents();
  2023 			if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
  2024 				{
  2025 				iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
  2026 				}
  2027 			
  2028 			iRootWindow->AdjustCoordsDueToRotation();
  2029 
  2030 			//TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
  2031 			if(BlankScreenOnRotation())
  2032 				{
  2033 				iRootWindow->ClearDisplay();
  2034 				}
  2035 
  2036 			CWsTop::ClearAllRedrawStores();
  2037 			DiscardAllSchedules();
  2038 			iRootWindow->InvalidateWholeScreen();
  2039 			CWsWindowGroup::SetScreenDeviceValidStates(this);
  2040 			TWindowServerEvent::SendScreenDeviceChangedEvents(this);
  2041 			}
  2042 		else
  2043 			{
  2044 			return reply;
  2045 			}
  2046 		}
  2047 	else
  2048 		{
  2049 		reply = KErrNotSupported;
  2050 		}
  2051 	return reply;
  2052 	}
  2053 
  2054 /**
  2055  * Updates the screen device display properties. This is to ensure the screen device is 
  2056  * consistent with any configuration changes not made using CScreen::SetConfiguration.
  2057  * 
  2058  * @param aConfigInput a fully populated display configuration
  2059  **/
  2060 TInt CScreen::UpdateConfiguration(const TDisplayConfiguration& aConfigInput)
  2061     {
  2062     TInt reply = KErrNone;
  2063     if(iDisplayControl)
  2064         {
  2065         TDisplayConfiguration config(aConfigInput);
  2066         TRect sizeModePosition;
  2067         if (iDisplayPolicy)
  2068             {   //validate config and update to a valid hardware config
  2069             reply = iDisplayPolicy->GetSizeModeConfiguration(iScreenSizeMode,config,sizeModePosition);
  2070             if (reply >= KErrNone)
  2071                 {//set appmode in policy
  2072                 if (iDisplayMapping)
  2073                     {
  2074                     iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
  2075                     }
  2076                 }
  2077             }
  2078         else
  2079             {   //exessive strategy: limit rotation agains curr app mode.
  2080                 //really we want the system to accept the rotation change regardless of the app mode.
  2081             TDisplayConfiguration::TRotation newRot;
  2082             if (aConfigInput.GetRotation(newRot))
  2083                 {   //This should cast between rotation enumertaions "properly"
  2084                 if (!(iModes[0][iScreenSizeMode]->iAlternativeRotations&(1<<newRot)))
  2085                     {
  2086                     reply=KErrArgument;
  2087                     }
  2088                 }
  2089             }
  2090 
  2091         MWsScene::TSceneRotation oldRotation;
  2092         oldRotation = iScene->SceneRotation();
  2093         TSize newUiSize;
  2094         config.GetResolution(newUiSize);
  2095         if(iFlags&EHasDynamicSizeModes)
  2096             {
  2097             reply = iFallbackMap->Resize(newUiSize);
  2098             }
  2099 
  2100         RecalculateModeTwips(&config);   //needs res and twips information
  2101         UpdateDynamicScreenModes();        
  2102         
  2103         TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, iScreenSizeMode));
  2104         // This will remove all the DSA elements from the scene
  2105         AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
  2106         
  2107         //SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
  2108         //on a screen with touch input.
  2109         //SetDigitiserAreas(newUiSize);
  2110         
  2111         //failure here should only be because of DSA orientation change failure, which shouldn't happen, either.
  2112         //Or there may be no change to do.
  2113         (void)UpdateOrientation(&oldRotation);
  2114         
  2115         iWindowElementSet->ResubmitAllElementExtents();
  2116         if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
  2117             {
  2118             iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
  2119             }
  2120         
  2121         iRootWindow->AdjustCoordsDueToRotation();
  2122 
  2123         //TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
  2124         if(BlankScreenOnRotation())
  2125             {
  2126             iRootWindow->ClearDisplay();
  2127             }
  2128 
  2129         CWsTop::ClearAllRedrawStores();
  2130         DiscardAllSchedules();
  2131         iRootWindow->InvalidateWholeScreen();
  2132         CWsWindowGroup::SetScreenDeviceValidStates(this);
  2133         TWindowServerEvent::SendScreenDeviceChangedEvents(this);
  2134         }
  2135     else
  2136         {
  2137         reply = KErrNotSupported;
  2138         }
  2139     return reply;  
  2140     }
  2141 
  2142 void CScreen::UpdateDynamicScreenModes()
  2143 	{
  2144 	WS_ASSERT_DEBUG(iDisplayControl,EWsPanicNoDisplayControl);
  2145 	TDisplayConfiguration newConfig;
  2146 	iDisplayControl->GetConfiguration(newConfig);
  2147 	TSize res;
  2148 	TSize twips;
  2149 	newConfig.GetResolution(res);
  2150 	newConfig.GetResolutionTwips(twips);
  2151 	for (TInt i=0; i<iModes->Count(); i++)
  2152 		{
  2153 		if ((*iModes)[i] && ((*iModes)[i]->iFlags & EDynamic))
  2154 			{
  2155 			(*iModes)[i]->iScreenSize = res;
  2156 			(*iModes)[i]->iScreenTwipsSize = twips;
  2157 			}
  2158 		}
  2159 	}
  2160 
  2161 void CScreen::RecalculateModeTwips(const TDisplayConfiguration* aConfig)
  2162 	{
  2163 	TDisplayConfiguration config;
  2164 	iDisplayControl->GetConfiguration(config);
  2165 	TSize res;
  2166 	TSize twips;
  2167 	if (aConfig)	//called from SetConfiguration
  2168 		{
  2169 		aConfig->GetResolution(res);
  2170 		if (res.iWidth == 0 || res.iHeight == 0)
  2171 			{
  2172 			return;
  2173 			}
  2174 		aConfig->GetResolutionTwips(twips);
  2175 		}
  2176 	else	//called from DisplayChangeNotifier during attach
  2177 		{
  2178 		config.GetResolution(res);
  2179 		if ((res.iWidth == 0 || res.iHeight == 0) && !iDisplayPolicy)
  2180 			{
  2181 			return;
  2182 			}
  2183 		config.GetResolutionTwips(twips);
  2184 		}
  2185 	TInt err=KErrNone;
  2186 	TInt flags=0;
  2187 	for (TInt ii=0; ii<iModes->Count(); ii++)
  2188 		{	//for every mode
  2189 		TRAP(err, flags = ModePtrL(ii)->iFlags);
  2190 		if (err != KErrNone || flags&(ETwipsSpecified|EDynamic))
  2191 			{	//continue if mode doesnt exist,twips specified or dynamic mode specified
  2192 			continue;
  2193 			}
  2194 
  2195 		if (iDisplayPolicy)
  2196 			{	//get ideal config for app mode from policy
  2197 			TRect modePosition;
  2198 			config.ClearAll();
  2199 			TInt err = iDisplayPolicy->GetSizeModeConfiguration(ii,config,modePosition);
  2200 			if (err != KErrNone)
  2201 				{	//nothing we can do, the twips will not be calculated correctly
  2202 				continue;
  2203 				}
  2204 			config.GetResolution(res);
  2205 			config.GetResolutionTwips(twips);
  2206 			}
  2207 		TSizeMode* modePtr=(*iModes)[ii];
  2208 		modePtr->iScreenTwipsSize.iWidth = (twips.iWidth * modePtr->iScreenSize.iWidth)/
  2209 				res.iWidth;
  2210 		modePtr->iScreenTwipsSize.iHeight = (twips.iHeight * modePtr->iScreenSize.iHeight)/
  2211 				res.iHeight;
  2212 		}
  2213 	
  2214 	}
  2215 
  2216 TInt CScreen::AddNotificationClient(CWsClient *aClient)
  2217 	{
  2218 	TInt err = iWsClientList.InsertInAddressOrder(aClient);
  2219 	if(!(err == KErrNone || err == KErrAlreadyExists))
  2220 		{
  2221 		return err;
  2222 		}
  2223 	return KErrNone;
  2224 	}
  2225 void CScreen::RemoveNotificationClient(CWsClient *aClient)
  2226 	{
  2227 	TInt index = iWsClientList.FindInAddressOrder(aClient);
  2228 	if(index != KErrNotFound)
  2229 		{
  2230 		iWsClientList.Remove(index);
  2231 		}
  2232 	}
  2233 TInt CScreen::GetNotificationClients(RPointerArray<CWsClient>& aClientsArray)
  2234 	{
  2235 	TInt err = aClientsArray.Reserve(iWsClientList.Count());
  2236 	if(err != KErrNone)
  2237 		return err;
  2238 	
  2239 	for(TInt i = 0; i < iWsClientList.Count(); i++)
  2240 		{
  2241 		aClientsArray.Append(iWsClientList[i]);
  2242 		}
  2243 	return KErrNone;
  2244 	}
  2245 
  2246 TInt CScreen::FindNotificationClient (CWsClient *aClient)
  2247 	{
  2248 	return iWsClientList.FindInAddressOrder(aClient);
  2249 	}
  2250 
  2251 // implementing MWsScreenConfig... this might be better as RS interface
  2252 TSize CScreen::ScreenModeSizeInPixels() const
  2253 	{
  2254 	return (*iModes)[iScreenSizeMode]->iScreenSize;
  2255 	}
  2256 TInt CScreen::Stride() const
  2257 	{
  2258 	return 0;
  2259 	}
  2260 TInt CScreen::SizeMode() const
  2261 	{
  2262 	return iScreenSizeMode;
  2263 
  2264 	}
  2265 TSize CScreen::ScalingFactor() const
  2266 	{
  2267 	return (*iModes)[iScreenSizeMode]->iScreenScale;
  2268 	}
  2269 TPoint CScreen::Origin() const
  2270 	{
  2271 	return (*iModes)[iScreenSizeMode]->iOrigin;
  2272 	}
  2273 TPoint CScreen::ScaledOrigin() const
  2274 	{
  2275 	return (*iModes)[iScreenSizeMode]->ScaledOrigin();
  2276 	}
  2277 
  2278 const CScreen::TInternalSizeMode* CScreen::ModePtrL(TInt aIndex) const
  2279 	{
  2280 	if (aIndex>=iModes->Count() || aIndex<0)
  2281 		{
  2282 		User::Leave(KErrArgument);
  2283 		}
  2284 	if (iModes==NULL)
  2285 		{
  2286 		User::Leave(KErrNotReady);
  2287 		}
  2288 	TInternalSizeMode* modePtr=(*iModes)[aIndex];
  2289 	if (modePtr==NULL)
  2290 		{
  2291 		User::Leave(KErrArgument);
  2292 		}
  2293 	return modePtr;
  2294 	}
  2295 
  2296 TDisplayMode CScreen::DisplayModeL(TInt aIndex) const
  2297 	{
  2298 	return ModePtrL(aIndex)->iDefaultDisplayMode;
  2299 	}
  2300 TSize CScreen::ScreenModeSizeInPixelsL(TInt aIndex) const
  2301 	{
  2302 	return ModePtrL(aIndex)->iScreenSize;
  2303 	}
  2304 TSize CScreen::ScreenModeSizeInTwipsL(TInt aIndex) const 
  2305 	{
  2306 	return ModePtrL(aIndex)->iScreenTwipsSize;
  2307 	}
  2308 
  2309 CFbsBitGc::TGraphicsOrientation CScreen::OrientationL(TInt aIndex) const
  2310 	{
  2311 	return ModePtrL(aIndex)->iRotation;
  2312 	}
  2313 TInt CScreen::AvailableOrientationsL(TInt aIndex) const
  2314 	{
  2315 	return ModePtrL(aIndex)->iAlternativeRotations;
  2316 	}
  2317 TSize CScreen::ScalingFactorL(TInt aIndex) const
  2318 	{
  2319 	return ModePtrL(aIndex)->iScreenScale;
  2320 	}
  2321 TPoint CScreen::OriginL(TInt aIndex) const
  2322 	{
  2323 	return ModePtrL(aIndex)->iOrigin;
  2324 	}
  2325 TPoint CScreen::ScaledOriginL(TInt aIndex) const
  2326 	{
  2327 	return ModePtrL(aIndex)->ScaledOrigin();
  2328 	}
  2329 TInt CScreen::ModeFlagsL(TInt aIndex) const
  2330 	{
  2331 	return ModePtrL(aIndex)->iFlags;
  2332 	}
  2333 void CScreen::SetCurrentScreenModeAttributes(const TSizeMode &aModeData)
  2334 	{	
  2335 	TSizeMode* modeToOverwrite=(*iModes)[iScreenSizeMode];
  2336 	*modeToOverwrite=aModeData;
  2337 	}
  2338 
  2339 void CScreen::ScheduleWindow(CWsWindow* aWindow)
  2340 	{
  2341 	iRedraw->ScheduleWindow(aWindow);
  2342 	}
  2343 
  2344 void CScreen::RemoveFromScheduledList(CWsWindow* aWindow)
  2345 	{
  2346 	iRedraw->RemoveFromScheduledList(aWindow);
  2347 	}
  2348 
  2349 void CScreen::RemoveFromTimedDrawList(CWsWindow* aWindow)
  2350 	{
  2351 	iRedraw->RemoveFromTimedDrawList(aWindow);
  2352 	}
  2353 
  2354 void CScreen::SetupVisibleRegionTracking(CWsWindow& aWindow, TBool aRegister) const
  2355 	{
  2356 	if(ChangeTracking() && iWindowVisibilityNotifier)
  2357 		{
  2358 		if(aRegister)
  2359 			{
  2360 			iWindowVisibilityNotifier->RegisterWindow(aWindow);
  2361 			}
  2362 		else
  2363 			{
  2364 			iWindowVisibilityNotifier->UnregisterWindow(aWindow);
  2365 			}
  2366 		}
  2367 	}
  2368 
  2369 TBool CScreen::IsAnimating() const
  2370     {
  2371     return iRedraw->IsAnimating();
  2372     }