os/graphics/windowing/windowserver/nga/SERVER/wsdisplaychangeao.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
/**
sl@0
    17
 @file
sl@0
    18
*/
sl@0
    19
sl@0
    20
#include "wsdisplaychangeao.h"
sl@0
    21
#include "ScrDev.H"
sl@0
    22
#include "server.h"
sl@0
    23
sl@0
    24
CWsDisplayChangeNotifier::CWsDisplayChangeNotifier(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
sl@0
    25
:CActive(EEventPriority), iNextLevelInterface(aNextLevelInterface), iOwner(aOwner)
sl@0
    26
	{
sl@0
    27
	CActiveScheduler::Add(this);
sl@0
    28
	}
sl@0
    29
sl@0
    30
CWsDisplayChangeNotifier* CWsDisplayChangeNotifier::NewL(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
sl@0
    31
	{
sl@0
    32
	CWsDisplayChangeNotifier* self = new(ELeave) CWsDisplayChangeNotifier(aNextLevelInterface, aOwner);
sl@0
    33
	CleanupStack::PushL(self);
sl@0
    34
	self->ConstructL();
sl@0
    35
	CleanupStack::Pop();
sl@0
    36
	return self;
sl@0
    37
	}
sl@0
    38
sl@0
    39
void CWsDisplayChangeNotifier::ConstructL()
sl@0
    40
	{
sl@0
    41
	iRetry = CEventQueueRetry::NewL();
sl@0
    42
	}
sl@0
    43
sl@0
    44
void CWsDisplayChangeNotifier::IssueNotificationRequest()
sl@0
    45
	{
sl@0
    46
	//before submitting request, save the last resolution list
sl@0
    47
	iLastErr = iNextLevelInterface->GetResolutions(iLastResList);
sl@0
    48
	if(iLastErr != KErrNone)
sl@0
    49
		{
sl@0
    50
		//probabally KErrDisconnect
sl@0
    51
		iLastResList.Reset();
sl@0
    52
		}
sl@0
    53
	iNextLevelInterface->NotifyOnDisplayChange(iStatus);
sl@0
    54
	SetActive();
sl@0
    55
	}
sl@0
    56
sl@0
    57
void CWsDisplayChangeNotifier::CancelNotificationRequest()
sl@0
    58
	{
sl@0
    59
	Cancel();
sl@0
    60
	}
sl@0
    61
sl@0
    62
void CWsDisplayChangeNotifier::RunL()
sl@0
    63
	{
sl@0
    64
	if(iStatus == KErrNone)
sl@0
    65
		{
sl@0
    66
		//IssueNotificationRequest() will overwrite iLastResList and iLastErr, save a copy first
sl@0
    67
		TInt lastErr = iLastErr;
sl@0
    68
		RArray<MWsDisplayControl::TResolution> lastResList;
sl@0
    69
		CleanupClosePushL(lastResList);
sl@0
    70
		lastResList.ReserveL(iLastResList.Count());
sl@0
    71
		for(TInt i = 0; i < iLastResList.Count(); i++)
sl@0
    72
			{
sl@0
    73
			lastResList.Append(iLastResList[i]);
sl@0
    74
			}
sl@0
    75
		//submit request again, this should be called as early as possible in RunL(). Otherwise display change occurs
sl@0
    76
		//before submitting the request will be missed.
sl@0
    77
		IssueNotificationRequest();
sl@0
    78
		
sl@0
    79
		RArray<MWsDisplayControl::TResolution> currentResList;
sl@0
    80
		CleanupClosePushL(currentResList);
sl@0
    81
		TInt err = iNextLevelInterface->GetResolutions(currentResList);
sl@0
    82
		if(err != KErrNone)
sl@0
    83
			{
sl@0
    84
			//probabally KErrDisconnect
sl@0
    85
			currentResList.Reset();
sl@0
    86
			}
sl@0
    87
		//return code not equal also counts as an event
sl@0
    88
		if(!IsResListEqual(currentResList, lastResList) || lastErr != err)
sl@0
    89
			{
sl@0
    90
			//first thing is to cancel the retry AO if it's running for the last event
sl@0
    91
			iRetry->CancelRetry();
sl@0
    92
			
sl@0
    93
			iOwner->IncreaseDisplaySpinner();
sl@0
    94
			//put display change event on queue
sl@0
    95
			RPointerArray<CWsClient> clientArray;
sl@0
    96
			CleanupClosePushL(clientArray);
sl@0
    97
			User::LeaveIfError(iOwner->GetNotificationClients(clientArray));
sl@0
    98
			TBool eventOnAllQueues = ETrue;
sl@0
    99
			for(TInt i = 0; i < clientArray.Count(); i++)
sl@0
   100
				{
sl@0
   101
				if(!TWindowServerEvent::SendDisplayChangedEvents(clientArray[i], iOwner->ScreenNumber(),
sl@0
   102
						iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
sl@0
   103
					{
sl@0
   104
					eventOnAllQueues = EFalse;
sl@0
   105
					clientArray[i]->SetRetryFlag(EEventDisplayChanged);
sl@0
   106
					}
sl@0
   107
				}
sl@0
   108
			CleanupStack::PopAndDestroy(&clientArray);
sl@0
   109
			
sl@0
   110
			//some event queues are full, kick off retry AO
sl@0
   111
			if(!eventOnAllQueues)
sl@0
   112
				{
sl@0
   113
				iRetry->Init(iOwner);
sl@0
   114
				iRetry->Retry(KRetryInitialDelay);
sl@0
   115
				}
sl@0
   116
						
sl@0
   117
			//codes below are dealing with detach/attach
sl@0
   118
			MWsDisplayPolicy* policy = iOwner->DisplayPolicy();
sl@0
   119
			if((err == KErrDisconnected && lastErr != KErrDisconnected)
sl@0
   120
					||(currentResList.Count() == 0 && lastResList.Count() != 0))
sl@0
   121
				{
sl@0
   122
				//The display is disconnected
sl@0
   123
				if(policy)
sl@0
   124
					{
sl@0
   125
					TInt appMode = policy->SuitableAppMode(MWsDisplayPolicy::EDetach);
sl@0
   126
					//Last app mode can be resumed when display is connected again from disconnection
sl@0
   127
					iOwner->DisplayPolicy()->SetLastAppMode(iOwner->ScreenSizeMode());
sl@0
   128
					//This gonna set the screen mode to smallest app mode
sl@0
   129
					if(appMode >= 0)
sl@0
   130
						{
sl@0
   131
						iOwner->doSetScreenMode(appMode);
sl@0
   132
						}
sl@0
   133
					}
sl@0
   134
				else
sl@0
   135
					{
sl@0
   136
					//if policy is not available, do a SetConfiguration set to the same config before disconnect
sl@0
   137
					//this will update parameters for MDisplayMapping stored in CDisplayPolicy
sl@0
   138
					TDisplayConfiguration config;
sl@0
   139
					iNextLevelInterface->GetConfiguration(config);
sl@0
   140
					iNextLevelInterface->SetConfiguration(config);
sl@0
   141
					}
sl@0
   142
				//stop DSA drawing as the above set screen mode is under disconnection and it won't
sl@0
   143
				//go the normal routine to stop DSA drawing
sl@0
   144
				iOwner->AbortAllDirectDrawing(RDirectScreenAccess::ETerminateScreenMode);
sl@0
   145
				}
sl@0
   146
			
sl@0
   147
			if(currentResList.Count() > 0 && lastResList.Count() == 0)
sl@0
   148
				{
sl@0
   149
				//The display is connected
sl@0
   150
				if(policy)
sl@0
   151
					{
sl@0
   152
					TInt appMode = policy->SuitableAppMode(MWsDisplayPolicy::EAttach);
sl@0
   153
					//This gonna resume the screen mode to the one before disconnection
sl@0
   154
					if(appMode >= 0)
sl@0
   155
						{
sl@0
   156
						iOwner->doSetScreenMode(appMode);
sl@0
   157
						}
sl@0
   158
					}
sl@0
   159
				else
sl@0
   160
					{
sl@0
   161
					//if policy is not available, force a SetConfiguration to trigger a config change notification
sl@0
   162
					//as the twips size in config is changed on attaching display though resolution may remain 0x0 
sl@0
   163
					TDisplayConfiguration config;
sl@0
   164
					config.SetResolution(currentResList[0].iPixelSize);
sl@0
   165
					iOwner->SetConfiguration(config);
sl@0
   166
					}
sl@0
   167
				iOwner->RecalculateModeTwips();
sl@0
   168
				}
sl@0
   169
			}
sl@0
   170
		CleanupStack::PopAndDestroy(2, &lastResList);
sl@0
   171
		}
sl@0
   172
	else if(iStatus != KErrCancel && iStatus != KErrNotSupported)
sl@0
   173
		{
sl@0
   174
		IssueNotificationRequest();	//This should be ok, not deadlock
sl@0
   175
		}
sl@0
   176
		
sl@0
   177
	}
sl@0
   178
sl@0
   179
TBool CWsDisplayChangeNotifier::IsResListEqual(RArray<MWsDisplayControl::TResolution>& aResListA, RArray<MWsDisplayControl::TResolution>& aResListB)
sl@0
   180
	{
sl@0
   181
	if (aResListA.Count() != aResListB.Count())
sl@0
   182
		return EFalse;
sl@0
   183
	
sl@0
   184
	for(TInt i = 0; i < aResListA.Count(); i++)
sl@0
   185
		{
sl@0
   186
		if(aResListA[i].iPixelSize != aResListB[i].iPixelSize)
sl@0
   187
			{
sl@0
   188
			return EFalse;
sl@0
   189
			}
sl@0
   190
		if(aResListA[i].iTwipsSize != aResListB[i].iTwipsSize)
sl@0
   191
			{
sl@0
   192
			return EFalse;
sl@0
   193
			}
sl@0
   194
		if(!(aResListA[i].iFlags == aResListB[i].iFlags))
sl@0
   195
			{
sl@0
   196
			return EFalse;
sl@0
   197
			}
sl@0
   198
		}
sl@0
   199
	
sl@0
   200
	return ETrue;
sl@0
   201
	}
sl@0
   202
sl@0
   203
void CWsDisplayChangeNotifier::DoCancel()
sl@0
   204
	{}
sl@0
   205
sl@0
   206
CWsDisplayChangeNotifier::~CWsDisplayChangeNotifier()
sl@0
   207
	{
sl@0
   208
	Cancel();
sl@0
   209
	iLastResList.Close();
sl@0
   210
	delete iRetry;
sl@0
   211
	}
sl@0
   212
sl@0
   213
CWsConfigChangeNotifier::CWsConfigChangeNotifier(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
sl@0
   214
:CActive(EEventPriority), iNextLevelInterface(aNextLevelInterface), iOwner(aOwner)
sl@0
   215
	{
sl@0
   216
	CActiveScheduler::Add(this);
sl@0
   217
	}
sl@0
   218
sl@0
   219
CWsConfigChangeNotifier* CWsConfigChangeNotifier::NewL(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
sl@0
   220
	{
sl@0
   221
	CWsConfigChangeNotifier* self = new(ELeave) CWsConfigChangeNotifier(aNextLevelInterface, aOwner);
sl@0
   222
	CleanupStack::PushL(self);
sl@0
   223
	self->ConstructL();
sl@0
   224
	CleanupStack::Pop();
sl@0
   225
	return self;
sl@0
   226
	}
sl@0
   227
sl@0
   228
void CWsConfigChangeNotifier::ConstructL()
sl@0
   229
	{
sl@0
   230
	iRetry = CEventQueueRetry::NewL();
sl@0
   231
	}
sl@0
   232
sl@0
   233
void CWsConfigChangeNotifier::IssueNotificationRequest()
sl@0
   234
	{
sl@0
   235
	//Before submitting the request, save last configuration
sl@0
   236
	iNextLevelInterface->GetConfiguration(iLastConfig);
sl@0
   237
	iNextLevelInterface->NotifyOnConfigChange(iStatus);
sl@0
   238
	SetActive();
sl@0
   239
	}
sl@0
   240
sl@0
   241
void CWsConfigChangeNotifier::CancelNotificationRequest()
sl@0
   242
	{
sl@0
   243
	Cancel();
sl@0
   244
	}
sl@0
   245
sl@0
   246
void CWsConfigChangeNotifier::UpdateLastSetConfiguration(TDisplayConfiguration& aNewConfig)
sl@0
   247
    {
sl@0
   248
    TSize resolution(0,0);
sl@0
   249
    if (aNewConfig.GetResolution(resolution))
sl@0
   250
        {
sl@0
   251
        iLastSetConfig.SetResolution(resolution);
sl@0
   252
        }
sl@0
   253
    
sl@0
   254
    TSize resolutionTwips(0,0);
sl@0
   255
    if (aNewConfig.GetResolutionTwips(resolutionTwips))
sl@0
   256
        {
sl@0
   257
        iLastSetConfig.SetResolutionTwips(resolutionTwips);   
sl@0
   258
        }
sl@0
   259
    
sl@0
   260
    TDisplayConfiguration1::TRotation rotation;
sl@0
   261
    if (aNewConfig.GetRotation(rotation))
sl@0
   262
        {
sl@0
   263
        iLastSetConfig.SetRotation(rotation);
sl@0
   264
        }
sl@0
   265
    }
sl@0
   266
sl@0
   267
void CWsConfigChangeNotifier::RunL()
sl@0
   268
	{
sl@0
   269
	if(iStatus == KErrNone)
sl@0
   270
		{
sl@0
   271
		//IssueNotificationRequest() will overwrite iLastConfig, save a copy first
sl@0
   272
		TDisplayConfiguration lastConfig(iLastConfig);
sl@0
   273
		IssueNotificationRequest();
sl@0
   274
		
sl@0
   275
		TDisplayConfiguration currentConfig;
sl@0
   276
		iNextLevelInterface->GetConfiguration(currentConfig);
sl@0
   277
		if(!(currentConfig == lastConfig))
sl@0
   278
			{
sl@0
   279
			//first thing is to cancel the retry AO if it's running for the last event
sl@0
   280
			iRetry->CancelRetry();
sl@0
   281
			
sl@0
   282
			iOwner->IncreaseConfigSpinner();
sl@0
   283
			
sl@0
   284
			//if the config change comes from a render stage then ensure screen device size 
sl@0
   285
			//is also updated
sl@0
   286
			TSize currentRes;
sl@0
   287
			currentConfig.GetResolution(currentRes);
sl@0
   288
			TBool disconnected = (currentRes.iHeight == 0 || currentRes.iWidth == 0) ? ETrue : EFalse;
sl@0
   289
            
sl@0
   290
			//if the config change is due to CScreen::SetConfiguration() being called then we
sl@0
   291
			//don't want to update it again. Only update if the configs are different and the
sl@0
   292
			//display is connected...
sl@0
   293
			TDisplayConfiguration lastSetConfig(iLastSetConfig);
sl@0
   294
			if (!((currentConfig == lastSetConfig) || (disconnected)))
sl@0
   295
			    {
sl@0
   296
                TDisplayConfiguration1::TRotation rotation;
sl@0
   297
                if (lastSetConfig.GetRotation(rotation))
sl@0
   298
                    {
sl@0
   299
                    //use the latest rotation value to ensure we don't get any
sl@0
   300
                    //inconsistencies with the layer extents
sl@0
   301
                    currentConfig.SetRotation(rotation);
sl@0
   302
                    }
sl@0
   303
                iOwner->UpdateConfiguration(currentConfig);
sl@0
   304
			    }		
sl@0
   305
			
sl@0
   306
			//put config change event on queue
sl@0
   307
			RPointerArray<CWsClient> clientArray;
sl@0
   308
			CleanupClosePushL(clientArray);
sl@0
   309
			User::LeaveIfError(iOwner->GetNotificationClients(clientArray));
sl@0
   310
			TBool eventOnAllQueues = ETrue;
sl@0
   311
			for(TInt i = 0; i < clientArray.Count(); i++)
sl@0
   312
				{
sl@0
   313
				if(!TWindowServerEvent::SendDisplayChangedEvents(clientArray[i], iOwner->ScreenNumber(),
sl@0
   314
						iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
sl@0
   315
					{
sl@0
   316
					eventOnAllQueues = EFalse;
sl@0
   317
					clientArray[i]->SetRetryFlag(EEventDisplayChanged);					
sl@0
   318
					}
sl@0
   319
				}
sl@0
   320
			
sl@0
   321
			CleanupStack::PopAndDestroy(&clientArray);
sl@0
   322
			//some event queues are full, kick off retry AO
sl@0
   323
			if(!eventOnAllQueues)
sl@0
   324
				{
sl@0
   325
				iRetry->Init(iOwner);
sl@0
   326
				iRetry->Retry(KRetryInitialDelay);
sl@0
   327
				}
sl@0
   328
			}
sl@0
   329
		iNextLevelInterface->GetConfiguration(iLastSetConfig);
sl@0
   330
		}
sl@0
   331
	else if(iStatus != KErrCancel && iStatus != KErrNotSupported)
sl@0
   332
		{
sl@0
   333
		IssueNotificationRequest();
sl@0
   334
		}
sl@0
   335
	
sl@0
   336
	}
sl@0
   337
sl@0
   338
sl@0
   339
void CWsConfigChangeNotifier::DoCancel()
sl@0
   340
	{}
sl@0
   341
sl@0
   342
CWsConfigChangeNotifier::~CWsConfigChangeNotifier()
sl@0
   343
	{
sl@0
   344
	Cancel();
sl@0
   345
	delete iRetry;
sl@0
   346
	}
sl@0
   347
sl@0
   348