1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/wsdisplaychangeao.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,348 @@
1.4 +// Copyright (c) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +/**
1.20 + @file
1.21 +*/
1.22 +
1.23 +#include "wsdisplaychangeao.h"
1.24 +#include "ScrDev.H"
1.25 +#include "server.h"
1.26 +
1.27 +CWsDisplayChangeNotifier::CWsDisplayChangeNotifier(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
1.28 +:CActive(EEventPriority), iNextLevelInterface(aNextLevelInterface), iOwner(aOwner)
1.29 + {
1.30 + CActiveScheduler::Add(this);
1.31 + }
1.32 +
1.33 +CWsDisplayChangeNotifier* CWsDisplayChangeNotifier::NewL(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
1.34 + {
1.35 + CWsDisplayChangeNotifier* self = new(ELeave) CWsDisplayChangeNotifier(aNextLevelInterface, aOwner);
1.36 + CleanupStack::PushL(self);
1.37 + self->ConstructL();
1.38 + CleanupStack::Pop();
1.39 + return self;
1.40 + }
1.41 +
1.42 +void CWsDisplayChangeNotifier::ConstructL()
1.43 + {
1.44 + iRetry = CEventQueueRetry::NewL();
1.45 + }
1.46 +
1.47 +void CWsDisplayChangeNotifier::IssueNotificationRequest()
1.48 + {
1.49 + //before submitting request, save the last resolution list
1.50 + iLastErr = iNextLevelInterface->GetResolutions(iLastResList);
1.51 + if(iLastErr != KErrNone)
1.52 + {
1.53 + //probabally KErrDisconnect
1.54 + iLastResList.Reset();
1.55 + }
1.56 + iNextLevelInterface->NotifyOnDisplayChange(iStatus);
1.57 + SetActive();
1.58 + }
1.59 +
1.60 +void CWsDisplayChangeNotifier::CancelNotificationRequest()
1.61 + {
1.62 + Cancel();
1.63 + }
1.64 +
1.65 +void CWsDisplayChangeNotifier::RunL()
1.66 + {
1.67 + if(iStatus == KErrNone)
1.68 + {
1.69 + //IssueNotificationRequest() will overwrite iLastResList and iLastErr, save a copy first
1.70 + TInt lastErr = iLastErr;
1.71 + RArray<MWsDisplayControl::TResolution> lastResList;
1.72 + CleanupClosePushL(lastResList);
1.73 + lastResList.ReserveL(iLastResList.Count());
1.74 + for(TInt i = 0; i < iLastResList.Count(); i++)
1.75 + {
1.76 + lastResList.Append(iLastResList[i]);
1.77 + }
1.78 + //submit request again, this should be called as early as possible in RunL(). Otherwise display change occurs
1.79 + //before submitting the request will be missed.
1.80 + IssueNotificationRequest();
1.81 +
1.82 + RArray<MWsDisplayControl::TResolution> currentResList;
1.83 + CleanupClosePushL(currentResList);
1.84 + TInt err = iNextLevelInterface->GetResolutions(currentResList);
1.85 + if(err != KErrNone)
1.86 + {
1.87 + //probabally KErrDisconnect
1.88 + currentResList.Reset();
1.89 + }
1.90 + //return code not equal also counts as an event
1.91 + if(!IsResListEqual(currentResList, lastResList) || lastErr != err)
1.92 + {
1.93 + //first thing is to cancel the retry AO if it's running for the last event
1.94 + iRetry->CancelRetry();
1.95 +
1.96 + iOwner->IncreaseDisplaySpinner();
1.97 + //put display change event on queue
1.98 + RPointerArray<CWsClient> clientArray;
1.99 + CleanupClosePushL(clientArray);
1.100 + User::LeaveIfError(iOwner->GetNotificationClients(clientArray));
1.101 + TBool eventOnAllQueues = ETrue;
1.102 + for(TInt i = 0; i < clientArray.Count(); i++)
1.103 + {
1.104 + if(!TWindowServerEvent::SendDisplayChangedEvents(clientArray[i], iOwner->ScreenNumber(),
1.105 + iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
1.106 + {
1.107 + eventOnAllQueues = EFalse;
1.108 + clientArray[i]->SetRetryFlag(EEventDisplayChanged);
1.109 + }
1.110 + }
1.111 + CleanupStack::PopAndDestroy(&clientArray);
1.112 +
1.113 + //some event queues are full, kick off retry AO
1.114 + if(!eventOnAllQueues)
1.115 + {
1.116 + iRetry->Init(iOwner);
1.117 + iRetry->Retry(KRetryInitialDelay);
1.118 + }
1.119 +
1.120 + //codes below are dealing with detach/attach
1.121 + MWsDisplayPolicy* policy = iOwner->DisplayPolicy();
1.122 + if((err == KErrDisconnected && lastErr != KErrDisconnected)
1.123 + ||(currentResList.Count() == 0 && lastResList.Count() != 0))
1.124 + {
1.125 + //The display is disconnected
1.126 + if(policy)
1.127 + {
1.128 + TInt appMode = policy->SuitableAppMode(MWsDisplayPolicy::EDetach);
1.129 + //Last app mode can be resumed when display is connected again from disconnection
1.130 + iOwner->DisplayPolicy()->SetLastAppMode(iOwner->ScreenSizeMode());
1.131 + //This gonna set the screen mode to smallest app mode
1.132 + if(appMode >= 0)
1.133 + {
1.134 + iOwner->doSetScreenMode(appMode);
1.135 + }
1.136 + }
1.137 + else
1.138 + {
1.139 + //if policy is not available, do a SetConfiguration set to the same config before disconnect
1.140 + //this will update parameters for MDisplayMapping stored in CDisplayPolicy
1.141 + TDisplayConfiguration config;
1.142 + iNextLevelInterface->GetConfiguration(config);
1.143 + iNextLevelInterface->SetConfiguration(config);
1.144 + }
1.145 + //stop DSA drawing as the above set screen mode is under disconnection and it won't
1.146 + //go the normal routine to stop DSA drawing
1.147 + iOwner->AbortAllDirectDrawing(RDirectScreenAccess::ETerminateScreenMode);
1.148 + }
1.149 +
1.150 + if(currentResList.Count() > 0 && lastResList.Count() == 0)
1.151 + {
1.152 + //The display is connected
1.153 + if(policy)
1.154 + {
1.155 + TInt appMode = policy->SuitableAppMode(MWsDisplayPolicy::EAttach);
1.156 + //This gonna resume the screen mode to the one before disconnection
1.157 + if(appMode >= 0)
1.158 + {
1.159 + iOwner->doSetScreenMode(appMode);
1.160 + }
1.161 + }
1.162 + else
1.163 + {
1.164 + //if policy is not available, force a SetConfiguration to trigger a config change notification
1.165 + //as the twips size in config is changed on attaching display though resolution may remain 0x0
1.166 + TDisplayConfiguration config;
1.167 + config.SetResolution(currentResList[0].iPixelSize);
1.168 + iOwner->SetConfiguration(config);
1.169 + }
1.170 + iOwner->RecalculateModeTwips();
1.171 + }
1.172 + }
1.173 + CleanupStack::PopAndDestroy(2, &lastResList);
1.174 + }
1.175 + else if(iStatus != KErrCancel && iStatus != KErrNotSupported)
1.176 + {
1.177 + IssueNotificationRequest(); //This should be ok, not deadlock
1.178 + }
1.179 +
1.180 + }
1.181 +
1.182 +TBool CWsDisplayChangeNotifier::IsResListEqual(RArray<MWsDisplayControl::TResolution>& aResListA, RArray<MWsDisplayControl::TResolution>& aResListB)
1.183 + {
1.184 + if (aResListA.Count() != aResListB.Count())
1.185 + return EFalse;
1.186 +
1.187 + for(TInt i = 0; i < aResListA.Count(); i++)
1.188 + {
1.189 + if(aResListA[i].iPixelSize != aResListB[i].iPixelSize)
1.190 + {
1.191 + return EFalse;
1.192 + }
1.193 + if(aResListA[i].iTwipsSize != aResListB[i].iTwipsSize)
1.194 + {
1.195 + return EFalse;
1.196 + }
1.197 + if(!(aResListA[i].iFlags == aResListB[i].iFlags))
1.198 + {
1.199 + return EFalse;
1.200 + }
1.201 + }
1.202 +
1.203 + return ETrue;
1.204 + }
1.205 +
1.206 +void CWsDisplayChangeNotifier::DoCancel()
1.207 + {}
1.208 +
1.209 +CWsDisplayChangeNotifier::~CWsDisplayChangeNotifier()
1.210 + {
1.211 + Cancel();
1.212 + iLastResList.Close();
1.213 + delete iRetry;
1.214 + }
1.215 +
1.216 +CWsConfigChangeNotifier::CWsConfigChangeNotifier(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
1.217 +:CActive(EEventPriority), iNextLevelInterface(aNextLevelInterface), iOwner(aOwner)
1.218 + {
1.219 + CActiveScheduler::Add(this);
1.220 + }
1.221 +
1.222 +CWsConfigChangeNotifier* CWsConfigChangeNotifier::NewL(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
1.223 + {
1.224 + CWsConfigChangeNotifier* self = new(ELeave) CWsConfigChangeNotifier(aNextLevelInterface, aOwner);
1.225 + CleanupStack::PushL(self);
1.226 + self->ConstructL();
1.227 + CleanupStack::Pop();
1.228 + return self;
1.229 + }
1.230 +
1.231 +void CWsConfigChangeNotifier::ConstructL()
1.232 + {
1.233 + iRetry = CEventQueueRetry::NewL();
1.234 + }
1.235 +
1.236 +void CWsConfigChangeNotifier::IssueNotificationRequest()
1.237 + {
1.238 + //Before submitting the request, save last configuration
1.239 + iNextLevelInterface->GetConfiguration(iLastConfig);
1.240 + iNextLevelInterface->NotifyOnConfigChange(iStatus);
1.241 + SetActive();
1.242 + }
1.243 +
1.244 +void CWsConfigChangeNotifier::CancelNotificationRequest()
1.245 + {
1.246 + Cancel();
1.247 + }
1.248 +
1.249 +void CWsConfigChangeNotifier::UpdateLastSetConfiguration(TDisplayConfiguration& aNewConfig)
1.250 + {
1.251 + TSize resolution(0,0);
1.252 + if (aNewConfig.GetResolution(resolution))
1.253 + {
1.254 + iLastSetConfig.SetResolution(resolution);
1.255 + }
1.256 +
1.257 + TSize resolutionTwips(0,0);
1.258 + if (aNewConfig.GetResolutionTwips(resolutionTwips))
1.259 + {
1.260 + iLastSetConfig.SetResolutionTwips(resolutionTwips);
1.261 + }
1.262 +
1.263 + TDisplayConfiguration1::TRotation rotation;
1.264 + if (aNewConfig.GetRotation(rotation))
1.265 + {
1.266 + iLastSetConfig.SetRotation(rotation);
1.267 + }
1.268 + }
1.269 +
1.270 +void CWsConfigChangeNotifier::RunL()
1.271 + {
1.272 + if(iStatus == KErrNone)
1.273 + {
1.274 + //IssueNotificationRequest() will overwrite iLastConfig, save a copy first
1.275 + TDisplayConfiguration lastConfig(iLastConfig);
1.276 + IssueNotificationRequest();
1.277 +
1.278 + TDisplayConfiguration currentConfig;
1.279 + iNextLevelInterface->GetConfiguration(currentConfig);
1.280 + if(!(currentConfig == lastConfig))
1.281 + {
1.282 + //first thing is to cancel the retry AO if it's running for the last event
1.283 + iRetry->CancelRetry();
1.284 +
1.285 + iOwner->IncreaseConfigSpinner();
1.286 +
1.287 + //if the config change comes from a render stage then ensure screen device size
1.288 + //is also updated
1.289 + TSize currentRes;
1.290 + currentConfig.GetResolution(currentRes);
1.291 + TBool disconnected = (currentRes.iHeight == 0 || currentRes.iWidth == 0) ? ETrue : EFalse;
1.292 +
1.293 + //if the config change is due to CScreen::SetConfiguration() being called then we
1.294 + //don't want to update it again. Only update if the configs are different and the
1.295 + //display is connected...
1.296 + TDisplayConfiguration lastSetConfig(iLastSetConfig);
1.297 + if (!((currentConfig == lastSetConfig) || (disconnected)))
1.298 + {
1.299 + TDisplayConfiguration1::TRotation rotation;
1.300 + if (lastSetConfig.GetRotation(rotation))
1.301 + {
1.302 + //use the latest rotation value to ensure we don't get any
1.303 + //inconsistencies with the layer extents
1.304 + currentConfig.SetRotation(rotation);
1.305 + }
1.306 + iOwner->UpdateConfiguration(currentConfig);
1.307 + }
1.308 +
1.309 + //put config change event on queue
1.310 + RPointerArray<CWsClient> clientArray;
1.311 + CleanupClosePushL(clientArray);
1.312 + User::LeaveIfError(iOwner->GetNotificationClients(clientArray));
1.313 + TBool eventOnAllQueues = ETrue;
1.314 + for(TInt i = 0; i < clientArray.Count(); i++)
1.315 + {
1.316 + if(!TWindowServerEvent::SendDisplayChangedEvents(clientArray[i], iOwner->ScreenNumber(),
1.317 + iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
1.318 + {
1.319 + eventOnAllQueues = EFalse;
1.320 + clientArray[i]->SetRetryFlag(EEventDisplayChanged);
1.321 + }
1.322 + }
1.323 +
1.324 + CleanupStack::PopAndDestroy(&clientArray);
1.325 + //some event queues are full, kick off retry AO
1.326 + if(!eventOnAllQueues)
1.327 + {
1.328 + iRetry->Init(iOwner);
1.329 + iRetry->Retry(KRetryInitialDelay);
1.330 + }
1.331 + }
1.332 + iNextLevelInterface->GetConfiguration(iLastSetConfig);
1.333 + }
1.334 + else if(iStatus != KErrCancel && iStatus != KErrNotSupported)
1.335 + {
1.336 + IssueNotificationRequest();
1.337 + }
1.338 +
1.339 + }
1.340 +
1.341 +
1.342 +void CWsConfigChangeNotifier::DoCancel()
1.343 + {}
1.344 +
1.345 +CWsConfigChangeNotifier::~CWsConfigChangeNotifier()
1.346 + {
1.347 + Cancel();
1.348 + delete iRetry;
1.349 + }
1.350 +
1.351 +