os/graphics/windowing/windowserver/nga/CLIENT/RDirect.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2000-2009 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
// Client side classes for handling direct screen access
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include <e32std.h>
sl@0
    19
#include <e32base.h>
sl@0
    20
#include "../SERVER/w32cmd.h"
sl@0
    21
#include "CLIENT.H"
sl@0
    22
#include "w32comm.h"
sl@0
    23
#include <e32msgqueue.h>
sl@0
    24
sl@0
    25
sl@0
    26
NONSHARABLE_CLASS(CDsaMsgQueue) : public CActive
sl@0
    27
{
sl@0
    28
	public:
sl@0
    29
		CDsaMsgQueue();
sl@0
    30
		~CDsaMsgQueue();
sl@0
    31
		void Request(TRequestStatus* aClientRequest);
sl@0
    32
		TBool Started() { return iStarted;}
sl@0
    33
		TBool Completed();
sl@0
    34
		void OpenRecQueue(TInt aHandle);
sl@0
    35
		void OpenSendQueue(TInt aHandle);
sl@0
    36
		TInt Send(TInt aData);
sl@0
    37
		RMsgQueueBase& SendQueue() {return iSendQueue; }
sl@0
    38
		RMsgQueueBase& Queue() { return iRecQueue; }
sl@0
    39
		TRequestStatus* Status() { return &iStatus; }
sl@0
    40
		TBool RequestStarted() { return iStarted;}
sl@0
    41
	private:
sl@0
    42
		void DoCancel();
sl@0
    43
		void RunL();
sl@0
    44
		void Listen();
sl@0
    45
		
sl@0
    46
	private:
sl@0
    47
		RMsgQueueBase iRecQueue;
sl@0
    48
		RMsgQueueBase iSendQueue;
sl@0
    49
		TRequestStatus* iClientRequest;
sl@0
    50
		TBool iStarted;
sl@0
    51
		RThread* iServer;
sl@0
    52
};
sl@0
    53
sl@0
    54
//
sl@0
    55
CDsaMsgQueue::CDsaMsgQueue() : CActive(RDirectScreenAccess::EPriorityVeryHigh)
sl@0
    56
	{
sl@0
    57
	CActiveScheduler::Add(this);
sl@0
    58
	}
sl@0
    59
sl@0
    60
CDsaMsgQueue::~CDsaMsgQueue()
sl@0
    61
	{
sl@0
    62
	Cancel();
sl@0
    63
	iRecQueue.Close();
sl@0
    64
	iSendQueue.Close();
sl@0
    65
	}
sl@0
    66
sl@0
    67
TInt CDsaMsgQueue::Send(TInt aData)
sl@0
    68
	{
sl@0
    69
	return iSendQueue.Send(&aData,sizeof(TInt));
sl@0
    70
	}
sl@0
    71
sl@0
    72
void CDsaMsgQueue::OpenRecQueue(TInt aHandle)
sl@0
    73
	{
sl@0
    74
	iRecQueue.SetHandle(aHandle);
sl@0
    75
// With RmessagePtr2 compelete using an RHandle the returned handle is already duplicated
sl@0
    76
	}
sl@0
    77
sl@0
    78
void CDsaMsgQueue::OpenSendQueue(TInt aHandle)
sl@0
    79
	{
sl@0
    80
	iSendQueue.SetHandle(aHandle);
sl@0
    81
// With RmessagePtr2 compelete using an RHandle the returned handle is already duplicated
sl@0
    82
	}
sl@0
    83
sl@0
    84
void CDsaMsgQueue::DoCancel()
sl@0
    85
	{
sl@0
    86
	iRecQueue.CancelDataAvailable();
sl@0
    87
	TInt ret = KErrNone;
sl@0
    88
	do
sl@0
    89
		{
sl@0
    90
		TInt data = 0;
sl@0
    91
		ret = iRecQueue.Receive(&data,sizeof(TInt));
sl@0
    92
		}while(ret == KErrNone);
sl@0
    93
	if(iClientRequest)
sl@0
    94
		{
sl@0
    95
		RThread().RequestComplete(iClientRequest,KErrCancel);
sl@0
    96
		}
sl@0
    97
	}
sl@0
    98
	
sl@0
    99
void CDsaMsgQueue::RunL()
sl@0
   100
	{
sl@0
   101
	// get the data from the msg queue
sl@0
   102
	TInt reason = 0;
sl@0
   103
	iRecQueue.Receive(&reason,sizeof(TInt));
sl@0
   104
	
sl@0
   105
	if(iClientRequest)
sl@0
   106
		{
sl@0
   107
		// if there is an outstanding client request, complete and pass on the abort reason
sl@0
   108
		User::RequestComplete(iClientRequest,reason);
sl@0
   109
		iClientRequest = NULL;
sl@0
   110
		}
sl@0
   111
	}
sl@0
   112
sl@0
   113
void CDsaMsgQueue::Listen()
sl@0
   114
	{
sl@0
   115
	if(!IsActive())
sl@0
   116
		{
sl@0
   117
		SetActive();	
sl@0
   118
		iRecQueue.NotifyDataAvailable(iStatus);
sl@0
   119
		}
sl@0
   120
	}
sl@0
   121
sl@0
   122
void CDsaMsgQueue::Request(TRequestStatus* aClientRequest)
sl@0
   123
	{
sl@0
   124
	__ASSERT_ALWAYS(!IsActive(),User::Invariant());
sl@0
   125
	iClientRequest = aClientRequest;
sl@0
   126
	iStarted = ETrue;
sl@0
   127
	Listen();
sl@0
   128
	}
sl@0
   129
sl@0
   130
TBool CDsaMsgQueue::Completed()
sl@0
   131
	{
sl@0
   132
	if(iStarted)
sl@0
   133
		{
sl@0
   134
		Send(KErrNone);
sl@0
   135
		iStarted = EFalse;
sl@0
   136
		return ETrue;
sl@0
   137
		}
sl@0
   138
	return EFalse;
sl@0
   139
	}
sl@0
   140
sl@0
   141
sl@0
   142
LOCAL_C inline TDeviceOrientation Graphics2DeviceOrientation(CFbsBitGc::TGraphicsOrientation aGraphicsOrientation)
sl@0
   143
    {
sl@0
   144
    return ((TDeviceOrientation)(1 << aGraphicsOrientation));
sl@0
   145
    }
sl@0
   146
sl@0
   147
//
sl@0
   148
// RDirectScreenAccess
sl@0
   149
//
sl@0
   150
sl@0
   151
EXPORT_C RDirectScreenAccess::RDirectScreenAccess()
sl@0
   152
/** Default constructor.
sl@0
   153
sl@0
   154
Developers should use the other constructor overload instead. */
sl@0
   155
	{
sl@0
   156
	}
sl@0
   157
sl@0
   158
EXPORT_C RDirectScreenAccess::RDirectScreenAccess(RWsSession& aWs) : MWsClientClass(aWs.iBuffer), iWs(&aWs), iMsgQueue(NULL)
sl@0
   159
/** C++ constructor with a connected window server session.
sl@0
   160
sl@0
   161
Construct() must be called to complete construction.
sl@0
   162
sl@0
   163
@param aWs Connected session with the window server. */
sl@0
   164
	{
sl@0
   165
	}
sl@0
   166
sl@0
   167
EXPORT_C TInt RDirectScreenAccess::Construct()
sl@0
   168
/** Second phase constructor.
sl@0
   169
sl@0
   170
Creates the server side resource and initialises the client's handle to it.
sl@0
   171
sl@0
   172
This function always causes a flush of the window server buffer.
sl@0
   173
sl@0
   174
@return KErrNone if successful, otherwise one of the system wide error codes. 
sl@0
   175
@panic TW32Panic 17 in debug builds if called on an already constructed object.*/
sl@0
   176
	{
sl@0
   177
	__ASSERT_DEBUG(iWsHandle == KNullHandle, Panic(EW32PanicGraphicDoubleConstruction));
sl@0
   178
	return Construct(EFalse);
sl@0
   179
	}
sl@0
   180
sl@0
   181
EXPORT_C TInt RDirectScreenAccess::Construct(TBool aRegionTrackingOnly)
sl@0
   182
/** Second phase constructor.
sl@0
   183
sl@0
   184
Creates the server side resource and initialises the client's handle to it.
sl@0
   185
sl@0
   186
This function always causes a flush of the window server buffer.
sl@0
   187
sl@0
   188
@param aRegionTrackingOnly ETrue if the DSA is intended to be used for region tracking purposes only,
sl@0
   189
EFalse if the DSA will be used to perform actual drawing.
sl@0
   190
@return KErrNone if successful, otherwise one of the system wide error codes. 
sl@0
   191
@panic TW32Panic 17 in debug builds if called on an already constructed object.*/
sl@0
   192
	{
sl@0
   193
	__ASSERT_DEBUG(iWsHandle == KNullHandle, Panic(EW32PanicGraphicDoubleConstruction));
sl@0
   194
	TInt ret = KErrNone;
sl@0
   195
	TWsClientOpcodes requestedOpCode = aRegionTrackingOnly? EWsClOpCreateDirectScreenAccessRegionTrackingOnly : EWsClOpCreateDirectScreenAccess;
sl@0
   196
	
sl@0
   197
	if ((ret = iBuffer->WriteReplyWs(requestedOpCode)) >= 0)
sl@0
   198
		{
sl@0
   199
		iWsHandle = ret;
sl@0
   200
		TRAP(ret,iMsgQueue = new (ELeave)CDsaMsgQueue);
sl@0
   201
		if(ret == KErrNone)
sl@0
   202
			{
sl@0
   203
			// the servers send queue is the client receive queue
sl@0
   204
			TInt h = WriteReply(EWsDirectOpGetSendQueue);
sl@0
   205
			iMsgQueue->OpenRecQueue(h);	
sl@0
   206
		
sl@0
   207
			// servers receive queue is the clients send queue
sl@0
   208
			h = WriteReply(EWsDirectOpGetRecQueue);
sl@0
   209
			iMsgQueue->OpenSendQueue(h);	
sl@0
   210
			}
sl@0
   211
		else
sl@0
   212
			{
sl@0
   213
			Close();
sl@0
   214
			}
sl@0
   215
		}
sl@0
   216
	return(ret);
sl@0
   217
	}
sl@0
   218
sl@0
   219
EXPORT_C TInt RDirectScreenAccess::Request(RRegion*& aRegion,TRequestStatus& aStatus,const RWindowBase& aWindow)
sl@0
   220
/** Issues a request to the window server for permission to perform direct screen 
sl@0
   221
access on a window.
sl@0
   222
sl@0
   223
Direct access to the screen may be refused due to lack of memory or if the 
sl@0
   224
target window is completely obscured.
sl@0
   225
sl@0
   226
If direct access is allowed, the function passes back a clipping region which 
sl@0
   227
is the part of the screen the caller can draw to. 
sl@0
   228
sl@0
   229
When direct screen access must stop, for instance because a dialog is to be 
sl@0
   230
displayed in front of the region where direct screen access is taking place, 
sl@0
   231
the window server completes the request. The recommended way to check for 
sl@0
   232
this is for aStatus to be the request status of an active object that will 
sl@0
   233
be run when the request completes, i.e. if Request() returns KErrNone, call 
sl@0
   234
SetActive(), and in the object's RunL(), you should immediately abort direct 
sl@0
   235
screen access.
sl@0
   236
sl@0
   237
While the DSA is in operation, it is strongly advised that the client should 
sl@0
   238
not make any call to WSERV that will affect the visible area of the window in 
sl@0
   239
which the DSA is taking place. 
sl@0
   240
sl@0
   241
When WSERV tells the client that it needs to abort its DSA, it waits to receive
sl@0
   242
the acknowledgment from the client that it has done so. However, it doesn't wait 
sl@0
   243
for ever, since the client may have entered some long running calculation or even
sl@0
   244
an infinite loop. So WSERV also waits on a timer: if the timer expires before the
sl@0
   245
client acknowledges, then WSERV continues; if, later on, WSERV gets notification
sl@0
   246
from the client that it has aborted the DSA, then WSERV will invalidate the region
sl@0
   247
in which the DSA was taking place, just in case there had been a conflict between
sl@0
   248
the DSA and another client.
sl@0
   249
sl@0
   250
sl@0
   251
This function always causes a flush of the window server buffer.
sl@0
   252
sl@0
   253
@param aRegion On return, the clipping region that the caller can draw to. 
sl@0
   254
NULL if the function was not successful.
sl@0
   255
If the target window is invisible or completely covered by other windows
sl@0
   256
then the region will be empty.
sl@0
   257
@param aStatus A request status that is set to a completion code by the window 
sl@0
   258
server when direct screen access must stop.
sl@0
   259
@param aWindow The window that you want to perform the direct screen access 
sl@0
   260
on.
sl@0
   261
@return KErrNone if the request was successful, KErrNone with empty region if 
sl@0
   262
none of the window is currently visible, otherwise one of the system wide error codes,
sl@0
   263
e.g. KErrNoMemory if out of memory. */
sl@0
   264
	{
sl@0
   265
	__ASSERT_ALWAYS(iMsgQueue,Panic(EW32PanicDirectMisuse));
sl@0
   266
sl@0
   267
	aRegion = NULL;
sl@0
   268
sl@0
   269
	// Allocate the memory for the RRegion here so it is simple to back out
sl@0
   270
	// in case of failure
sl@0
   271
	TAny* regionMem = User::Alloc (sizeof (RRegion));
sl@0
   272
	if (!regionMem)
sl@0
   273
		{
sl@0
   274
		return KErrNoMemory;
sl@0
   275
		}
sl@0
   276
sl@0
   277
	TInt ret = WriteReplyInt(aWindow.WsHandle(),EWsDirectOpRequest);
sl@0
   278
	if (ret<KErrNone)
sl@0
   279
		{
sl@0
   280
		User::Free (regionMem);
sl@0
   281
		return ret;
sl@0
   282
		}
sl@0
   283
	TRect* rectList = NULL;
sl@0
   284
	TRect* newRectList;
sl@0
   285
	TInt numRect;
sl@0
   286
sl@0
   287
	do
sl@0
   288
		{
sl@0
   289
		numRect = ret;
sl@0
   290
		newRectList = STATIC_CAST(TRect*,User::ReAlloc(rectList,numRect*sizeof(TRect)));
sl@0
   291
		if (!newRectList)
sl@0
   292
			{
sl@0
   293
			Write(EWsDirectOpInitFailed);
sl@0
   294
			User::Free (regionMem);
sl@0
   295
			delete rectList;
sl@0
   296
			return KErrNoMemory;
sl@0
   297
			}
sl@0
   298
		rectList = newRectList;
sl@0
   299
		TPtr8 ptr(REINTERPRET_CAST(TUint8*,rectList),ret*sizeof(TRect));
sl@0
   300
		ret = WriteReplyIntP(ret,&ptr,EWsDirectOpGetRegion);
sl@0
   301
		} while(ret >=0 && ret != KMaxTInt);
sl@0
   302
	if (ret<0)
sl@0
   303
		{
sl@0
   304
		User::Free (regionMem);
sl@0
   305
		delete rectList;
sl@0
   306
		return ret;
sl@0
   307
		}
sl@0
   308
sl@0
   309
	aRegion = new (regionMem) RRegion (numRect, rectList);
sl@0
   310
	aStatus = KRequestPending;
sl@0
   311
	iMsgQueue->Request(&aStatus);
sl@0
   312
	iWs->DirectAcessActivation(ETrue);
sl@0
   313
	return KErrNone;
sl@0
   314
	}
sl@0
   315
sl@0
   316
EXPORT_C void RDirectScreenAccess::Completed()
sl@0
   317
/** Indicates to the window server that you have responded to the completion of 
sl@0
   318
the request status passed to Request(), by stopping direct screen access. */
sl@0
   319
	{
sl@0
   320
	__ASSERT_ALWAYS(iMsgQueue->Started(),Panic(EW32PanicDirectMisuse));
sl@0
   321
	if(iMsgQueue->Completed())
sl@0
   322
		{
sl@0
   323
		iWs->DirectAcessActivation(EFalse);
sl@0
   324
		}
sl@0
   325
	}
sl@0
   326
sl@0
   327
EXPORT_C void RDirectScreenAccess::Cancel()
sl@0
   328
/** Indicates to the window server that you have finished performing direct screen 
sl@0
   329
access. */
sl@0
   330
	{
sl@0
   331
	if(iMsgQueue->Started())
sl@0
   332
		{
sl@0
   333
		Completed();
sl@0
   334
		}
sl@0
   335
	TInt ret = WriteReply(EWsDirectOpCancel);
sl@0
   336
	if(ret != 0) // the server is sending us some data.
sl@0
   337
		{
sl@0
   338
		iMsgQueue->Queue().CancelDataAvailable();
sl@0
   339
		TInt data = 0;
sl@0
   340
		iMsgQueue->Queue().ReceiveBlocking(&data,sizeof(TInt));
sl@0
   341
		}
sl@0
   342
	iMsgQueue->Cancel();
sl@0
   343
	}
sl@0
   344
sl@0
   345
EXPORT_C void RDirectScreenAccess::Close()
sl@0
   346
/** Calls Completed() then deletes the server side resource and sets the client's 
sl@0
   347
handle to it to NULL. */
sl@0
   348
	{
sl@0
   349
	if (iBuffer && iWsHandle)
sl@0
   350
		{
sl@0
   351
		if(iMsgQueue && iMsgQueue->Started())
sl@0
   352
			{
sl@0
   353
			Completed();
sl@0
   354
			}
sl@0
   355
		Write(EWsDirectOpFree);
sl@0
   356
		delete iMsgQueue;
sl@0
   357
		iMsgQueue = NULL;
sl@0
   358
		}
sl@0
   359
	iWsHandle = NULL;
sl@0
   360
	}
sl@0
   361
sl@0
   362
//
sl@0
   363
// CDirectScreenAccess
sl@0
   364
//
sl@0
   365
sl@0
   366
EXPORT_C CDirectScreenAccess* CDirectScreenAccess::NewL(RWsSession& aWs,CWsScreenDevice& aScreenDevice,RWindowBase& aWin,MDirectScreenAccess& aAbort)
sl@0
   367
/** Allocates and constructs the object and adds it to the current active scheduler.
sl@0
   368
sl@0
   369
This function always causes a flush of the window server buffer.
sl@0
   370
sl@0
   371
@param aWs A session with the window server.
sl@0
   372
@param aScreenDevice Specifies the characteristics of the screen device to 
sl@0
   373
draw to.
sl@0
   374
@param aWin The window to draw to directly.
sl@0
   375
@param aAbort Defines an AbortNow() and a Restart() function which are both 
sl@0
   376
called on aborting, as part of the RunL(). Restart() is called from an idle 
sl@0
   377
time active object (CIdle).
sl@0
   378
@return The newly constructed object. */
sl@0
   379
	{
sl@0
   380
	return CDirectScreenAccess::NewL(aWs,aScreenDevice,aWin,aAbort,EFalse);
sl@0
   381
	}
sl@0
   382
sl@0
   383
EXPORT_C CDirectScreenAccess* CDirectScreenAccess::NewL(RWsSession& aWs,CWsScreenDevice& aScreenDevice,RWindowBase& aWin,MDirectScreenAccess& aAbort,TBool aRegionTrackingOnly)
sl@0
   384
/** Allocates and constructs the object and adds it to the current active scheduler.
sl@0
   385
sl@0
   386
This function always causes a flush of the window server buffer.
sl@0
   387
sl@0
   388
@param aWs A session with the window server.
sl@0
   389
@param aScreenDevice Specifies the characteristics of the screen device to 
sl@0
   390
draw to.
sl@0
   391
@param aWin The window to draw to directly.
sl@0
   392
@param aAbort Defines an AbortNow() and a Restart() function which are both 
sl@0
   393
called on aborting, as part of the RunL(). Restart() is called from an idle 
sl@0
   394
time active object (CIdle).
sl@0
   395
@param aRegionTrackingOnly The screen device and GC are allocated if this is EFalse,
sl@0
   396
but not if it is ETrue. Only the DSA region data and updates to that are
sl@0
   397
available in the latter case. Creating the screen device will trigger the dsa
sl@0
   398
buffer allocationand it is an operation that could fail, should this happen
sl@0
   399
the function will leave.
sl@0
   400
@return The newly constructed object. */
sl@0
   401
	{
sl@0
   402
	CDirectScreenAccess* self = new(ELeave) CDirectScreenAccess(aWs,&aScreenDevice,aWin,aAbort);
sl@0
   403
	CleanupStack::PushL(self);
sl@0
   404
	self->ConstructL(aWs,aRegionTrackingOnly);
sl@0
   405
	CleanupStack::Pop(self);
sl@0
   406
	return self;
sl@0
   407
	}
sl@0
   408
sl@0
   409
CDirectScreenAccess::~CDirectScreenAccess()
sl@0
   410
	{
sl@0
   411
	__ASSERT_ALWAYS(!iAborting,Panic(EW32PanicDirectMisuse));
sl@0
   412
	Cancel();
sl@0
   413
	delete iGc;
sl@0
   414
	delete iScreenDevice;
sl@0
   415
	if (iDrawingRegion)
sl@0
   416
		iDrawingRegion->Destroy();
sl@0
   417
	iDirectAccess.Close();
sl@0
   418
	delete iRestart;
sl@0
   419
	}
sl@0
   420
sl@0
   421
void CDirectScreenAccess::ConstructL(RWsSession& aWs,TBool aRegionTrackingOnly)
sl@0
   422
	{
sl@0
   423
	iScreenNumber = iWsScreenDevice->GetScreenNumber();
sl@0
   424
	
sl@0
   425
	if(aRegionTrackingOnly)
sl@0
   426
		{
sl@0
   427
		iFlags |= EDirectRegionTrackingOnly;
sl@0
   428
		}
sl@0
   429
	User::LeaveIfError(iDirectAccess.Construct(aRegionTrackingOnly));
sl@0
   430
	
sl@0
   431
	iRestart = CIdle::NewL(RDirectScreenAccess::EPriorityVeryHigh-5);
sl@0
   432
	CActiveScheduler::Add(this);
sl@0
   433
	if (aWs.GetColorModeList(NULL)>1)
sl@0
   434
		iFlags |= EDirectCheckModeChange;
sl@0
   435
	if (iWsScreenDevice->NumScreenModes() == 1)
sl@0
   436
		{
sl@0
   437
		if ((iWsScreenDevice->GetRotationsList(0,NULL) == 1) && !aRegionTrackingOnly)
sl@0
   438
			{
sl@0
   439
			return;
sl@0
   440
			}
sl@0
   441
		}
sl@0
   442
	iFlags |= EDirectCheckSizeModeChange;
sl@0
   443
	}
sl@0
   444
sl@0
   445
void CDirectScreenAccess::CreateScreenObjectsL(TDisplayMode aCurrentMode)
sl@0
   446
	{
sl@0
   447
	__ASSERT_DEBUG(!(iFlags&EDirectRegionTrackingOnly),Panic(EW32PanicDirectMisuse));
sl@0
   448
	delete iScreenDevice;
sl@0
   449
	iScreenDevice = NULL;
sl@0
   450
	
sl@0
   451
	iScreenDevice = CFbsScreenDevice::NewL(iScreenNumber,aCurrentMode);
sl@0
   452
	
sl@0
   453
	if (iGc)
sl@0
   454
		{
sl@0
   455
		iGc->Activate(iScreenDevice);
sl@0
   456
		}
sl@0
   457
	else
sl@0
   458
		{
sl@0
   459
		User::LeaveIfError(iScreenDevice->CreateContext(iGc));
sl@0
   460
		if (!(iFlags&EDirectCheckSizeModeChange))
sl@0
   461
			UpdateSizeAndRotation(iGc);
sl@0
   462
		}
sl@0
   463
	}
sl@0
   464
sl@0
   465
EXPORT_C void CDirectScreenAccess::StartL()
sl@0
   466
/** Informs the window server that you are going to start direct screen access 
sl@0
   467
and sets up a graphics context with which you can draw to the screen.
sl@0
   468
sl@0
   469
It should also be called to restart direct screen access after Cancel() has 
sl@0
   470
been called to stop it. 
sl@0
   471
sl@0
   472
While the DSA is in operation, it is strongly advised that the client should 
sl@0
   473
not make any call to WSERV that will affect the visible area of the window in 
sl@0
   474
which the DSA is taking place. 
sl@0
   475
sl@0
   476
When WSERV tells the client that it needs to abort its DSA, it waits to receive
sl@0
   477
the acknowledgment from the client that it has done so. However, it doesn't wait
sl@0
   478
for ever, since the client may have entered some long running calculation or even
sl@0
   479
an infinite loop. So WSERV also waits on a timer: if the timer expires before the
sl@0
   480
client acknowledges, then WSERV continues; if, later on, WSERV gets notification
sl@0
   481
from the client that it has aborted the DSA, then WSERV will invalidate the region
sl@0
   482
in which the DSA was taking place, just in case there had been a conflict between
sl@0
   483
the DSA and another client.
sl@0
   484
sl@0
   485
sl@0
   486
This function always causes a flush of the window server buffer. */
sl@0
   487
	{
sl@0
   488
	if (iDrawingRegion)
sl@0
   489
		iDrawingRegion->Destroy();
sl@0
   490
	User::LeaveIfError(iDirectAccess.Request(iDrawingRegion,iStatus,iWindow));
sl@0
   491
	SetActive();
sl@0
   492
	if(!(iFlags&EDirectRegionTrackingOnly))
sl@0
   493
		{
sl@0
   494
		if((iFlags&EDirectCheckModeChange) || iScreenDevice == NULL)
sl@0
   495
			{
sl@0
   496
			TDisplayMode currentDisplayMode = iWsScreenDevice->DisplayMode();
sl@0
   497
			if (iScreenDevice == NULL || currentDisplayMode != iScreenDevice->DisplayMode())
sl@0
   498
				{
sl@0
   499
				TRAPD(err,CreateScreenObjectsL(currentDisplayMode));
sl@0
   500
				if (err != KErrNone)
sl@0
   501
					{
sl@0
   502
					Cancel();
sl@0
   503
					User::Leave(err);
sl@0
   504
					}
sl@0
   505
				}
sl@0
   506
			}
sl@0
   507
		if (iFlags&EDirectCheckSizeModeChange)
sl@0
   508
			{
sl@0
   509
			UpdateSizeAndRotation(iGc);
sl@0
   510
			}
sl@0
   511
		iGc->SetOrigin(iWindow.AbsPosition());
sl@0
   512
		}
sl@0
   513
	iDrawingRegion->ClipRect(iScreenSize);
sl@0
   514
	if(!(iFlags&EDirectRegionTrackingOnly))
sl@0
   515
			{
sl@0
   516
			iGc->SetClippingRegion(iDrawingRegion);
sl@0
   517
			}
sl@0
   518
	}
sl@0
   519
sl@0
   520
TInt CDirectScreenAccess::Restart(TAny* aDirect)		//static
sl@0
   521
	{
sl@0
   522
	STATIC_CAST(CDirectScreenAccess*,aDirect)->Restart();
sl@0
   523
	return(KErrNone);
sl@0
   524
	}
sl@0
   525
sl@0
   526
void CDirectScreenAccess::Restart()
sl@0
   527
	{
sl@0
   528
	iAbort.Restart(iReason);
sl@0
   529
	}
sl@0
   530
sl@0
   531
void CDirectScreenAccess::UpdateSizeAndRotation(CFbsBitGc* aGc)
sl@0
   532
	{
sl@0
   533
	TPixelsAndRotation sizeAndRotation;
sl@0
   534
	iWsScreenDevice->GetDefaultScreenSizeAndRotation(sizeAndRotation);
sl@0
   535
	iScreenSize = sizeAndRotation.iPixelSize;
sl@0
   536
	__ASSERT_ALWAYS(iScreenDevice,Panic(EW32PanicDirectMisuse));
sl@0
   537
	iScreenDevice->SetDeviceOrientation(Graphics2DeviceOrientation(sizeAndRotation.iRotation));
sl@0
   538
	MDisplayMapping* interface = static_cast<MDisplayMapping*>
sl@0
   539
				(iWsScreenDevice->GetInterface(MDisplayMapping::ETypeId));
sl@0
   540
	
sl@0
   541
	if(interface)
sl@0
   542
		{
sl@0
   543
		TRect appAreaInDsa;
sl@0
   544
		interface->MapCoordinates(EApplicationSpace, iScreenSize, EDirectScreenAccessSpace, appAreaInDsa);
sl@0
   545
		if(!iDrawingRegion->BoundingRect().IsEmpty())
sl@0
   546
			{
sl@0
   547
			//no point to set draw origin if draw region is empty
sl@0
   548
			//this also indicates the place to draw might be outside DSA buffer
sl@0
   549
			iScreenDevice->SetDrawDeviceOffset(appAreaInDsa.iTl);
sl@0
   550
			}
sl@0
   551
		}
sl@0
   552
	
sl@0
   553
	aGc->Activate(iScreenDevice);
sl@0
   554
	}
sl@0
   555
sl@0
   556
void CDirectScreenAccess::RunL()
sl@0
   557
	{
sl@0
   558
	iAborting = ETrue;
sl@0
   559
	iReason = REINTERPRET_CAST(RDirectScreenAccess::TTerminationReasons&,iStatus);
sl@0
   560
	iAbort.AbortNow(iReason);
sl@0
   561
	iAborting = EFalse;
sl@0
   562
	iDirectAccess.Completed();
sl@0
   563
	iRestart->Start(TCallBack(CDirectScreenAccess::Restart,this));
sl@0
   564
	}
sl@0
   565
sl@0
   566
void CDirectScreenAccess::DoCancel()
sl@0
   567
	{
sl@0
   568
	iDirectAccess.Cancel();
sl@0
   569
	}