os/graphics/windowing/windowserver/nga/CLIENT/WSGRAPHIC.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) 2005-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
// The client-side representation of a WsGraphic artwork
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include <w32std.h>
sl@0
    19
#include "../SERVER/w32cmd.h"
sl@0
    20
#include "w32comm.h"
sl@0
    21
#include "CLIENT.H"
sl@0
    22
#include <s32mem.h>
sl@0
    23
#include <graphics/WSGRAPHICDRAWERINTERFACE.H>
sl@0
    24
sl@0
    25
NONSHARABLE_STRUCT(CWsGraphic::CPimpl): public CBase, public MWsClientClass
sl@0
    26
/** @internalComponent @released */
sl@0
    27
	{
sl@0
    28
	friend class CWsGraphic;
sl@0
    29
	enum
sl@0
    30
		{
sl@0
    31
		/** is registered in the graphic manager's array */
sl@0
    32
		ERegistered = 0x01,
sl@0
    33
		/** has a peer CWsGraphicDrawer on the server */
sl@0
    34
		EHasPeer	= 0x02
sl@0
    35
		};
sl@0
    36
	CPimpl(CWsGraphic& aGraphic);
sl@0
    37
	~CPimpl();
sl@0
    38
	void ConstructL();		//LeaveScan: Member of macroised structure declaration.
sl@0
    39
	CManager* iManager;
sl@0
    40
	TInt WriteCreateGraphic(TWsClCmdCreateGraphic& aCreateGraphic,const TDesC8& aData,TInt aWsHandle) const;
sl@0
    41
	CWsGraphic& iGraphic;
sl@0
    42
	TWsGraphicId iId;
sl@0
    43
	TUint iFlags;
sl@0
    44
	MWsObjectProvider* iExt;
sl@0
    45
	};
sl@0
    46
sl@0
    47
LOCAL_C void MWsGraphicMessageAllocRelease(TAny* aAny)
sl@0
    48
	{
sl@0
    49
	MWsGraphicMessageAlloc::MBuffer* buf = static_cast<MWsGraphicMessageAlloc::MBuffer*>(aAny);
sl@0
    50
	if(buf)
sl@0
    51
		{
sl@0
    52
		buf->Release();
sl@0
    53
		}
sl@0
    54
	}
sl@0
    55
sl@0
    56
NONSHARABLE_CLASS(CWsGraphic::CManager): private CActive, public RWsSession
sl@0
    57
/** Client-side manager singleton for marshalling messages for all CWsGraphics owned by a client
sl@0
    58
sl@0
    59
@publishedAll
sl@0
    60
@released
sl@0
    61
*/	{
sl@0
    62
public:
sl@0
    63
	static CManager* StaticL();			  //LeaveScan:  Member of macroised structure declaration.
sl@0
    64
	// used by CWsGraphic
sl@0
    65
	void AddL(CWsGraphic* aGraphic);	  //LeaveScan:  Member of macroised structure declaration.
sl@0
    66
	void Replace(CWsGraphic* aGraphic);
sl@0
    67
	void Remove(CWsGraphic* aGraphic);
sl@0
    68
	CWsGraphic* Find(const TWsGraphicId& aId);
sl@0
    69
	void Inc();
sl@0
    70
	void Dec();
sl@0
    71
	RWsBuffer* Buffer();
sl@0
    72
	void ScheduleFlush();
sl@0
    73
private:
sl@0
    74
	CManager();
sl@0
    75
	~CManager();
sl@0
    76
	void ConstructL();					 //LeaveScan:  Member of macroised structure declaration.
sl@0
    77
	void Queue();
sl@0
    78
	void RunL();						 //LeaveScan:  Member of macroised structure declaration.
sl@0
    79
	void DoCancel();
sl@0
    80
	static TInt GraphicCompare(const CWsGraphic& aFirst,const CWsGraphic& aSecond);
sl@0
    81
	static TInt FlushOnIdle(TAny* aArg);
sl@0
    82
private:
sl@0
    83
	TInt iRefCount;
sl@0
    84
	RPointerArray<CWsGraphic> iArray;
sl@0
    85
	CIdle* iFlusher;
sl@0
    86
	TBool iFlushScheduled;
sl@0
    87
	};
sl@0
    88
sl@0
    89
// TWsGraphicId \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
    90
sl@0
    91
EXPORT_C TWsGraphicId::TWsGraphicId(TUid aUid):
sl@0
    92
/** Construct a UID
sl@0
    93
    @param aUid UID of the graphic artwork.
sl@0
    94
    @publishedAll @released
sl@0
    95
*/	iFlags(EWsGraphicIdUid), iId(aUid.iUid)
sl@0
    96
	{
sl@0
    97
	}
sl@0
    98
sl@0
    99
EXPORT_C TWsGraphicId::TWsGraphicId(TInt aId):
sl@0
   100
/** Construct a transient Id
sl@0
   101
	@publishedAll @released
sl@0
   102
*/	iFlags(EWsGraphicIdTransient), iId(aId)
sl@0
   103
	{
sl@0
   104
	}
sl@0
   105
sl@0
   106
/** 
sl@0
   107
Copy constructor.
sl@0
   108
@param aCopy Graphic artwork Id.
sl@0
   109
*/
sl@0
   110
EXPORT_C TWsGraphicId::TWsGraphicId(const TWsGraphicId& aCopy):
sl@0
   111
	iFlags(aCopy.iFlags), iId(aCopy.iId)
sl@0
   112
	{
sl@0
   113
	}
sl@0
   114
sl@0
   115
EXPORT_C TUid TWsGraphicId::Uid() const
sl@0
   116
/** Returns UID.
sl@0
   117
    @return UID of graphic artwork. KNullUid if graphic artwork is transient.
sl@0
   118
    @publishedAll @released
sl@0
   119
*/	{
sl@0
   120
	if(IsUid())
sl@0
   121
		{
sl@0
   122
		return TUid::Uid(iId);
sl@0
   123
		}
sl@0
   124
	return KNullUid;
sl@0
   125
	}
sl@0
   126
sl@0
   127
EXPORT_C TBool TWsGraphicId::IsUid() const
sl@0
   128
/** Identifies whether graphic artwork is non-transient.
sl@0
   129
    @return ETrue if graphic artwork is non-transient.
sl@0
   130
    @publishedAll @released
sl@0
   131
*/	{
sl@0
   132
	return (iFlags & EWsGraphicIdUid);
sl@0
   133
	}
sl@0
   134
sl@0
   135
EXPORT_C void TWsGraphicId::Set(TUid aUid)
sl@0
   136
/** Set to be a UID
sl@0
   137
	@publishedAll @released
sl@0
   138
*/	{
sl@0
   139
	iId = aUid.iUid;
sl@0
   140
	iFlags = EWsGraphicIdUid;
sl@0
   141
	}
sl@0
   142
sl@0
   143
EXPORT_C TInt TWsGraphicId::Id() const
sl@0
   144
/** Returns the transient Id.
sl@0
   145
    @return Id of transient graphic artwork. Zero if graphic artwork is non-transient.
sl@0
   146
	@publishedAll @released
sl@0
   147
*/	{
sl@0
   148
	if(IsId())
sl@0
   149
		{
sl@0
   150
		return iId;
sl@0
   151
		}
sl@0
   152
	return 0;
sl@0
   153
	}
sl@0
   154
sl@0
   155
EXPORT_C TBool TWsGraphicId::IsId() const
sl@0
   156
/** Identifies whether graphic artwork is transient.
sl@0
   157
    @return ETrue if graphic artwork is transient.
sl@0
   158
    @publishedAll @released
sl@0
   159
*/	{
sl@0
   160
	return (iFlags & EWsGraphicIdTransient);
sl@0
   161
	}
sl@0
   162
sl@0
   163
EXPORT_C void TWsGraphicId::Set(TInt aId)
sl@0
   164
/** Set to be a transient Id
sl@0
   165
	@publishedAll @released
sl@0
   166
*/	{
sl@0
   167
	iId = aId;
sl@0
   168
	iFlags = EWsGraphicIdTransient;
sl@0
   169
	}
sl@0
   170
sl@0
   171
EXPORT_C TInt TWsGraphicId::Compare(const TWsGraphicId& aOther) const
sl@0
   172
/** Compares another Id with this one.
sl@0
   173
	@return 0 if the other Id is identical, else -1 if the other Id is to greater than or 1 if the other Id is less than
sl@0
   174
	@publishedAll @released
sl@0
   175
*/	{
sl@0
   176
	if(iId < aOther.iId)
sl@0
   177
		{
sl@0
   178
		return -1;
sl@0
   179
		}
sl@0
   180
	else if(iId > aOther.iId)
sl@0
   181
		{
sl@0
   182
		return 1;
sl@0
   183
		}
sl@0
   184
	// else we have to compare the iIsUid flag too; again, expect it to be a match 99.99% of these times
sl@0
   185
	else if(IsUid() == aOther.IsUid())
sl@0
   186
		{
sl@0
   187
		return 0;
sl@0
   188
		}
sl@0
   189
	// collisions of id but not iIsUid are going to be really really rare
sl@0
   190
	else if(IsUid())
sl@0
   191
		{
sl@0
   192
		return 1;
sl@0
   193
		}
sl@0
   194
	else
sl@0
   195
		{
sl@0
   196
		return -1;
sl@0
   197
		}
sl@0
   198
	}
sl@0
   199
sl@0
   200
// CWsGraphicManager \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   201
sl@0
   202
CWsGraphic::CManager* CWsGraphic::CManager::StaticL()
sl@0
   203
	{
sl@0
   204
	CManager* singleton = RWsBuffer::WsGraphicManager();
sl@0
   205
	if(!singleton)
sl@0
   206
		{
sl@0
   207
		singleton = new(ELeave) CManager;
sl@0
   208
		CleanupStack::PushL(singleton);
sl@0
   209
		singleton->ConstructL();
sl@0
   210
		CleanupStack::Pop(singleton);
sl@0
   211
		__ASSERT_DEBUG(singleton == RWsBuffer::WsGraphicManager(),Panic(EW32PanicGraphicInternal));
sl@0
   212
		}
sl@0
   213
	return singleton;
sl@0
   214
	}
sl@0
   215
sl@0
   216
CWsGraphic::CManager::~CManager()
sl@0
   217
	{
sl@0
   218
	__ASSERT_DEBUG(!ResourceCount(),Panic(EW32PanicGraphicInternal));
sl@0
   219
	__ASSERT_DEBUG(!iArray.Count(),Panic(EW32PanicGraphicOrphaned));
sl@0
   220
	Close();
sl@0
   221
	__ASSERT_DEBUG(!RWsBuffer::WsGraphicManager(),Panic(EW32PanicGraphicInternal));
sl@0
   222
	iArray.Close();
sl@0
   223
	delete iFlusher;
sl@0
   224
	}
sl@0
   225
sl@0
   226
CWsGraphic::CManager::CManager(): CActive(CActive::EPriorityStandard)
sl@0
   227
	{
sl@0
   228
	}
sl@0
   229
sl@0
   230
void CWsGraphic::CManager::ConstructL()
sl@0
   231
	{
sl@0
   232
	User::LeaveIfError(Connect());
sl@0
   233
	iBuffer->SetWsGraphicManager(this);
sl@0
   234
	CActiveScheduler::Add(this);
sl@0
   235
	// coverity[negative_returns]
sl@0
   236
	iFlusher = CIdle::NewL(CActive::EPriorityIdle);
sl@0
   237
	}
sl@0
   238
sl@0
   239
void CWsGraphic::CManager::Inc()
sl@0
   240
	{
sl@0
   241
	iRefCount++;
sl@0
   242
	}
sl@0
   243
sl@0
   244
void CWsGraphic::CManager::Dec()
sl@0
   245
	{
sl@0
   246
	if(!--iRefCount)
sl@0
   247
		{
sl@0
   248
		delete this;
sl@0
   249
		}
sl@0
   250
	}
sl@0
   251
sl@0
   252
RWsBuffer* CWsGraphic::CManager::Buffer()
sl@0
   253
	{
sl@0
   254
	return iBuffer;
sl@0
   255
	}
sl@0
   256
sl@0
   257
// used by CWsGraphic
sl@0
   258
sl@0
   259
void CWsGraphic::CManager::AddL(CWsGraphic* aGraphic)
sl@0
   260
/** Leaves if the graphic couldn't be added to the list
sl@0
   261
	@internalComponent @released */
sl@0
   262
	{
sl@0
   263
	__ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
sl@0
   264
	__ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
sl@0
   265
	__ASSERT_ALWAYS(!(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
sl@0
   266
	iArray.InsertInOrderL(aGraphic,GraphicCompare);
sl@0
   267
	__ASSERT_ALWAYS(0 <= iArray.FindInOrder(aGraphic,GraphicCompare),Panic(EW32PanicGraphicInternal));
sl@0
   268
	aGraphic->iPimpl->iFlags |= CWsGraphic::CPimpl::ERegistered;
sl@0
   269
	Queue();
sl@0
   270
	}
sl@0
   271
sl@0
   272
void CWsGraphic::CManager::Replace(CWsGraphic* aGraphic)
sl@0
   273
/** @internalComponent @released */
sl@0
   274
	{
sl@0
   275
	__ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
sl@0
   276
	__ASSERT_ALWAYS(!(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
sl@0
   277
	__ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
sl@0
   278
	const TInt idx = iArray.FindInOrder(aGraphic,GraphicCompare);
sl@0
   279
	__ASSERT_ALWAYS(0 <= idx,Panic(EW32PanicGraphicInternal));
sl@0
   280
	__ASSERT_ALWAYS(iArray[idx]->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered,Panic(EW32PanicGraphicInternal));
sl@0
   281
	iArray[idx]->iPimpl->iFlags &= ~CWsGraphic::CPimpl::ERegistered;
sl@0
   282
	iArray[idx] = aGraphic;
sl@0
   283
	iArray[idx]->iPimpl->iFlags |= CWsGraphic::CPimpl::ERegistered;
sl@0
   284
	Queue();
sl@0
   285
	}
sl@0
   286
sl@0
   287
void CWsGraphic::CManager::Remove(CWsGraphic* aGraphic)
sl@0
   288
/** @internalComponent @released */
sl@0
   289
	{
sl@0
   290
	__ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
sl@0
   291
	__ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
sl@0
   292
	__ASSERT_ALWAYS(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered,Panic(EW32PanicGraphicInternal));
sl@0
   293
	const TInt idx = iArray.FindInOrder(aGraphic,GraphicCompare);
sl@0
   294
	__ASSERT_ALWAYS(0 <= idx,Panic(EW32PanicGraphicInternal));
sl@0
   295
	iArray[idx]->iPimpl->iFlags &= ~CWsGraphic::CPimpl::ERegistered;
sl@0
   296
	iArray.Remove(idx);
sl@0
   297
	if(!iArray.Count())
sl@0
   298
		{
sl@0
   299
		Cancel();
sl@0
   300
		}
sl@0
   301
	}
sl@0
   302
sl@0
   303
CWsGraphic* CWsGraphic::CManager::Find(const TWsGraphicId& aId)
sl@0
   304
/** Find the active artwork identified by the Id
sl@0
   305
	@return NULL if no active artwork has the Id
sl@0
   306
	@publishedAll @released */
sl@0
   307
	{
sl@0
   308
	/*	RPointerArray can only FindInOrder other T*, which is a needless shame.  Therefore this search is
sl@0
   309
		dumb sequential */
sl@0
   310
	const TInt count = iArray.Count();
sl@0
   311
	for(TInt i=0; i<count; i++)
sl@0
   312
		{
sl@0
   313
		CWsGraphic* graphic = iArray[i];
sl@0
   314
		__ASSERT_ALWAYS(graphic && graphic->iPimpl && (graphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
sl@0
   315
		const TWsGraphicId& candidate = graphic->Id();
sl@0
   316
		if(0 == candidate.Compare(aId))
sl@0
   317
			{
sl@0
   318
			// found
sl@0
   319
			__ASSERT_DEBUG(i == iArray.FindInOrder(graphic,GraphicCompare),Panic(EW32PanicGraphicInternal));
sl@0
   320
			return graphic;
sl@0
   321
			}
sl@0
   322
		}
sl@0
   323
	// not found
sl@0
   324
	return NULL;
sl@0
   325
	}
sl@0
   326
sl@0
   327
void CWsGraphic::CManager::Queue()
sl@0
   328
	{
sl@0
   329
	if(!IsActive())
sl@0
   330
		{
sl@0
   331
		GraphicMessageReady(&iStatus);
sl@0
   332
		SetActive();
sl@0
   333
		}
sl@0
   334
	}
sl@0
   335
sl@0
   336
void CWsGraphic::CManager::RunL()
sl@0
   337
   	{
sl@0
   338
   	if(0 < iStatus.Int())
sl@0
   339
   		{
sl@0
   340
   		// a message to fetch!
sl@0
   341
   		const TInt msgLen = iStatus.Int();
sl@0
   342
 		TPtr8 buf(NULL,0);
sl@0
   343
 		MWsGraphicMessageAlloc* allocator = NULL;
sl@0
   344
 		MWsGraphicMessageAlloc::MBuffer* theirBuf = NULL;
sl@0
   345
 		HBufC8* ourBuf = HBufC8::New(msgLen);
sl@0
   346
 		if(ourBuf)
sl@0
   347
 			{
sl@0
   348
 			CleanupStack::PushL(ourBuf);
sl@0
   349
 			buf.Set(ourBuf->Des());
sl@0
   350
 			}
sl@0
   351
 		else // try to see if the destination CWsGraphic can allocate for us
sl@0
   352
 			{
sl@0
   353
			const TInt handle = GraphicFetchHeaderMessage();
sl@0
   354
			CWsGraphic* dest = reinterpret_cast<CWsGraphic*>(handle & ~0x03);
sl@0
   355
 			// check if it's valid
sl@0
   356
 			if(KErrNotFound != iArray.Find(dest) && dest->iPimpl && dest->iPimpl->iExt)
sl@0
   357
 				{
sl@0
   358
 				// check if the client is able to alloc memory for us
sl@0
   359
 				allocator = dest->iPimpl->iExt->ObjectInterface<MWsGraphicMessageAlloc>();
sl@0
   360
 				if(allocator)
sl@0
   361
 					{
sl@0
   362
 					// allocate memory
sl@0
   363
 					theirBuf = allocator->Alloc(msgLen);
sl@0
   364
 					if(theirBuf)
sl@0
   365
 						{
sl@0
   366
 						CleanupStack::PushL(TCleanupItem(MWsGraphicMessageAllocRelease,theirBuf));
sl@0
   367
 						buf.Set(theirBuf->Buffer());
sl@0
   368
 						}
sl@0
   369
 					}
sl@0
   370
 				}
sl@0
   371
 			}
sl@0
   372
 		if(!ourBuf && !theirBuf)
sl@0
   373
 			{
sl@0
   374
 			GraphicMessageCancel();
sl@0
   375
 			GraphicAbortMessage(KErrNoMemory);
sl@0
   376
 			}
sl@0
   377
 		else
sl@0
   378
 			{
sl@0
   379
 			GetGraphicMessage(buf);
sl@0
   380
 			// decode header and body
sl@0
   381
 			RDesReadStream in(buf);
sl@0
   382
 			in.PushL();
sl@0
   383
 			const TInt header = in.ReadInt32L();
sl@0
   384
 			__ASSERT_COMPILE(sizeof(header) == sizeof(TInt32));
sl@0
   385
 			const TInt clientHandle = (header & ~0x03);
sl@0
   386
 			const TInt msgType = (header & 0x03);
sl@0
   387
 			const TPtr8 body = buf.MidTPtr(sizeof(header));
sl@0
   388
 			// dispatch
sl@0
   389
 			CWsGraphic* dest = (CWsGraphic*)clientHandle;
sl@0
   390
 			if(KErrNotFound != iArray.Find(dest))
sl@0
   391
 				{
sl@0
   392
 				switch(msgType)
sl@0
   393
 					{
sl@0
   394
 					case EWsGraphMessageTypeUser:
sl@0
   395
 						dest->HandleMessage(body);
sl@0
   396
 						break;
sl@0
   397
 					default:
sl@0
   398
 						Panic(EW32PanicGraphicInternal);
sl@0
   399
 					}
sl@0
   400
 				}
sl@0
   401
 			// done
sl@0
   402
 			in.Pop();
sl@0
   403
 			}
sl@0
   404
 		if(ourBuf)
sl@0
   405
 			{
sl@0
   406
 			CleanupStack::PopAndDestroy(ourBuf);
sl@0
   407
 			}
sl@0
   408
 		else if(theirBuf)
sl@0
   409
 			{
sl@0
   410
 			CleanupStack::PopAndDestroy(theirBuf);
sl@0
   411
 			}
sl@0
   412
 		// done, wait for next message
sl@0
   413
 		Queue();
sl@0
   414
 		}
sl@0
   415
 	}
sl@0
   416
sl@0
   417
void CWsGraphic::CManager::DoCancel()
sl@0
   418
	{
sl@0
   419
	GraphicMessageCancel();
sl@0
   420
	}
sl@0
   421
sl@0
   422
TInt CWsGraphic::CManager::GraphicCompare(const CWsGraphic& aFirst,const CWsGraphic& aSecond)
sl@0
   423
/** Compares two graphics for id equality
sl@0
   424
@internalComponent
sl@0
   425
@released
sl@0
   426
*/	{
sl@0
   427
	return aFirst.Id().Compare(aSecond.Id());
sl@0
   428
	}
sl@0
   429
sl@0
   430
void CWsGraphic::CManager::ScheduleFlush()
sl@0
   431
/** Request to schedule flush when idle
sl@0
   432
@internalComponent
sl@0
   433
@released
sl@0
   434
*/
sl@0
   435
	{
sl@0
   436
	if (iFlushScheduled)
sl@0
   437
		return;
sl@0
   438
sl@0
   439
	iFlushScheduled = ETrue;
sl@0
   440
	iFlusher->Start(TCallBack(CWsGraphic::CManager::FlushOnIdle,this));
sl@0
   441
	}
sl@0
   442
sl@0
   443
TInt CWsGraphic::CManager::FlushOnIdle(TAny* aArg)
sl@0
   444
/** Flush buffer when idle and there is outstanding data in it
sl@0
   445
@internalComponent
sl@0
   446
@released
sl@0
   447
*/
sl@0
   448
	{
sl@0
   449
	CWsGraphic::CManager* mgr = reinterpret_cast<CWsGraphic::CManager*>(aArg);
sl@0
   450
	if (mgr)
sl@0
   451
		{
sl@0
   452
		mgr->iFlushScheduled = EFalse;
sl@0
   453
		if (mgr->iBuffer && !mgr->iBuffer->IsEmpty())
sl@0
   454
			mgr->iBuffer->Flush();
sl@0
   455
		}
sl@0
   456
sl@0
   457
	return 0; // complete
sl@0
   458
	}
sl@0
   459
sl@0
   460
// CWsGraphic::CPimpl \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   461
sl@0
   462
CWsGraphic::CPimpl::CPimpl(CWsGraphic& aGraphic):
sl@0
   463
	iGraphic(aGraphic), iId(TWsGraphicId::EUninitialized)
sl@0
   464
	{
sl@0
   465
	}
sl@0
   466
sl@0
   467
CWsGraphic::CPimpl::~CPimpl()
sl@0
   468
	{
sl@0
   469
	if(iManager)
sl@0
   470
		{
sl@0
   471
		iManager->Dec();
sl@0
   472
		}
sl@0
   473
	}
sl@0
   474
sl@0
   475
void CWsGraphic::CPimpl::ConstructL()
sl@0
   476
	{
sl@0
   477
	iManager = CManager::StaticL();
sl@0
   478
	iManager->Inc();
sl@0
   479
	iBuffer = iManager->Buffer();
sl@0
   480
	}
sl@0
   481
sl@0
   482
TInt CWsGraphic::CPimpl::WriteCreateGraphic(TWsClCmdCreateGraphic& aCreateGraphic,const TDesC8& aData,TInt aWsHandle) const
sl@0
   483
/** Writes the CreateGraphic message to the server.  If the data will not fit in the buffer, uses remote-read
sl@0
   484
	@internalComponent @released */
sl@0
   485
	{
sl@0
   486
	aCreateGraphic.iDataLen = aData.Size();
sl@0
   487
	aCreateGraphic.iRemoteReadData = ((aData.Size()+sizeof(aCreateGraphic))>(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
sl@0
   488
	if(aCreateGraphic.iRemoteReadData)
sl@0
   489
		{
sl@0
   490
		return iBuffer->WriteReplyByProvidingRemoteReadAccess(aWsHandle,EWsClOpCreateGraphic,&aCreateGraphic,sizeof(aCreateGraphic),&aData);
sl@0
   491
		}
sl@0
   492
	else if(aCreateGraphic.iDataLen)
sl@0
   493
		{
sl@0
   494
		return iBuffer->WriteReplyWs(&aCreateGraphic,sizeof(aCreateGraphic),aData.Ptr(),aData.Size(),EWsClOpCreateGraphic);
sl@0
   495
		}
sl@0
   496
	else
sl@0
   497
		{
sl@0
   498
		return iBuffer->WriteReplyWs(&aCreateGraphic,sizeof(aCreateGraphic),EWsClOpCreateGraphic);
sl@0
   499
		}
sl@0
   500
	}
sl@0
   501
sl@0
   502
// CWsGraphic \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   503
sl@0
   504
/** 
sl@0
   505
Default Constructor.
sl@0
   506
*/
sl@0
   507
EXPORT_C CWsGraphic::CWsGraphic()
sl@0
   508
	{
sl@0
   509
	}
sl@0
   510
sl@0
   511
/**
sl@0
   512
Destructor.
sl@0
   513
*/
sl@0
   514
EXPORT_C CWsGraphic::~CWsGraphic()
sl@0
   515
	{
sl@0
   516
	Destroy();
sl@0
   517
	delete iPimpl;
sl@0
   518
	}
sl@0
   519
	
sl@0
   520
/**
sl@0
   521
Completes construction of the baseclass. All the overloaded BaseConstructL() methods
sl@0
   522
should invoke this method to complete the construction of the baseclass.
sl@0
   523
*/
sl@0
   524
void CWsGraphic::BaseConstructL()
sl@0
   525
	{
sl@0
   526
	iPimpl = new(ELeave) CPimpl(*this);
sl@0
   527
	iPimpl->ConstructL();
sl@0
   528
	}
sl@0
   529
sl@0
   530
EXPORT_C void CWsGraphic::BaseConstructL(TUid aUid,TUid aType,const TDesC8& aData)
sl@0
   531
/** 
sl@0
   532
Constructs a piece of non-transient graphic artwork.
sl@0
   533
@capability ProtServ
sl@0
   534
@param aUid	Graphic artwork UID.
sl@0
   535
@param aType Graphic artwork type.
sl@0
   536
@param aData User specific data.
sl@0
   537
*/	{
sl@0
   538
	BaseConstructL();
sl@0
   539
sl@0
   540
	TWsClCmdCreateGraphic createGraphic;
sl@0
   541
	createGraphic.iFlags = EWsGraphicIdUid;
sl@0
   542
	createGraphic.iId = aUid.iUid;
sl@0
   543
	createGraphic.iType = aType;
sl@0
   544
	createGraphic.iClientHandle = (TInt)this;
sl@0
   545
sl@0
   546
	TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
sl@0
   547
	User::LeaveIfError(ret);
sl@0
   548
sl@0
   549
	iPimpl->iWsHandle=ret;
sl@0
   550
	iPimpl->iId = aUid;
sl@0
   551
	iPimpl->iFlags = CPimpl::EHasPeer;
sl@0
   552
	iPimpl->iManager->AddL(this);
sl@0
   553
	}
sl@0
   554
sl@0
   555
EXPORT_C void CWsGraphic::BaseConstructL(TUid aType,const TDesC8& aData)
sl@0
   556
/** 
sl@0
   557
Constructs a piece of transient graphic artwork.
sl@0
   558
@param aType Graphic artwork type.
sl@0
   559
@param aData User specific data.
sl@0
   560
*/
sl@0
   561
	{
sl@0
   562
	BaseConstructL();
sl@0
   563
sl@0
   564
	TWsClCmdCreateGraphic createGraphic;
sl@0
   565
	createGraphic.iFlags = EWsGraphicIdTransient;
sl@0
   566
	createGraphic.iId = 0;
sl@0
   567
	createGraphic.iType = aType;
sl@0
   568
	createGraphic.iClientHandle = (TInt)this;
sl@0
   569
sl@0
   570
	TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
sl@0
   571
	User::LeaveIfError(ret);
sl@0
   572
sl@0
   573
	iPimpl->iWsHandle = ret;
sl@0
   574
sl@0
   575
	// fetch id from server
sl@0
   576
	TPckgBuf<TWsClCmdGdGetId> cmd;
sl@0
   577
	User::LeaveIfError(iPimpl->WriteReplyP(TWriteDescriptorType(&cmd),EWsGdOpGetGraphicId));
sl@0
   578
	if(cmd().iIsUid)
sl@0
   579
		{
sl@0
   580
		__DEBUG_ONLY(Panic(EW32PanicGraphicInternal));
sl@0
   581
		User::Leave(KErrGeneral);
sl@0
   582
		}
sl@0
   583
	iPimpl->iId = cmd().iId;
sl@0
   584
sl@0
   585
	iPimpl->iFlags = CPimpl::EHasPeer;
sl@0
   586
	iPimpl->iManager->AddL(this);
sl@0
   587
	}
sl@0
   588
sl@0
   589
EXPORT_C void CWsGraphic::BaseConstructL(const TWsGraphicId& aReplace,TUid aType,const TDesC8& aData)
sl@0
   590
/** 
sl@0
   591
Atomically replace the artwork that already exists with this artwork.
sl@0
   592
If failure to properly construct the replacement artwork occurs, the replacee artwork will remain
sl@0
   593
@param aReplace Graphic artwork which will be replaced.
sl@0
   594
@param aType New graphic artwork type.
sl@0
   595
@param aData User specific data.
sl@0
   596
*/
sl@0
   597
	{
sl@0
   598
	BaseConstructL();
sl@0
   599
sl@0
   600
	CWsGraphic* dup = iPimpl->iManager->Find(aReplace);
sl@0
   601
	if(!dup || !dup->iPimpl)
sl@0
   602
		{
sl@0
   603
		Panic(EW32PanicGraphicInternal);
sl@0
   604
		}
sl@0
   605
sl@0
   606
	const TUint flags = aReplace.IsUid()?EWsGraphicIdUid:EWsGraphicIdTransient;
sl@0
   607
sl@0
   608
	TWsClCmdCreateGraphic createGraphic;
sl@0
   609
	createGraphic.iFlags = EWsGraphicReplace|flags;
sl@0
   610
	createGraphic.iId = aReplace.IsUid()?aReplace.Uid().iUid:aReplace.Id();
sl@0
   611
	createGraphic.iType = aType;
sl@0
   612
	createGraphic.iClientHandle = (TInt)this;
sl@0
   613
sl@0
   614
	TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
sl@0
   615
	if(0 > ret)
sl@0
   616
		{
sl@0
   617
		User::Leave(ret);
sl@0
   618
		}
sl@0
   619
sl@0
   620
	iPimpl->iWsHandle = ret;
sl@0
   621
	iPimpl->iId = aReplace;
sl@0
   622
sl@0
   623
	iPimpl->iFlags = CPimpl::EHasPeer;
sl@0
   624
	iPimpl->iManager->Replace(this);
sl@0
   625
sl@0
   626
	// when WriteCreateGraphic succeeds, the replacee drawer has already been destroyed serverside
sl@0
   627
	// so this cleanup is not quite the same as Destroy(), as it doesn't free server side
sl@0
   628
	// coverity[var_deref_op]
sl@0
   629
	dup->iPimpl->iWsHandle = 0;
sl@0
   630
	dup->OnReplace();
sl@0
   631
	dup->iPimpl->iFlags &= ~CPimpl::EHasPeer;
sl@0
   632
	}
sl@0
   633
sl@0
   634
/**
sl@0
   635
Shares the graphic artwork with all the client sessions.
sl@0
   636
Sharing globally trumps explicit shares.
sl@0
   637
@return KErrNone if the graphic is globally shared, else one of the system-wide error codes.
sl@0
   638
*/
sl@0
   639
EXPORT_C TInt CWsGraphic::ShareGlobally()
sl@0
   640
	{
sl@0
   641
	if(!IsActive())
sl@0
   642
		{
sl@0
   643
		return KErrNotReady;
sl@0
   644
		}
sl@0
   645
	return iPimpl->WriteReply(EWsGdOpShareGlobally);
sl@0
   646
	}
sl@0
   647
sl@0
   648
/**
sl@0
   649
Prevents this graphic artwork from being shared with all the client sessions.
sl@0
   650
A graphic artwork that isn't shared explicitly is only available to clients it
sl@0
   651
has been explicitly shared with using Share().
sl@0
   652
@return KErrNone if the graphic is not globally shared, else one of the system-wide error codes.
sl@0
   653
*/
sl@0
   654
EXPORT_C TInt CWsGraphic::UnShareGlobally()
sl@0
   655
	{
sl@0
   656
	if(!IsActive())
sl@0
   657
		{
sl@0
   658
		return KErrNotReady;
sl@0
   659
		}
sl@0
   660
	return iPimpl->WriteReply(EWsGdOpUnShareGlobally);
sl@0
   661
	}
sl@0
   662
	
sl@0
   663
/** 
sl@0
   664
Explicitly shares this graphic artwork with client sessions with the specified Secure ID.
sl@0
   665
@param aClientId the Secure ID of the client sessions to share with.
sl@0
   666
@return KErrNone If the graphic artwork was shared, else one of the system-wide error codes.
sl@0
   667
*/
sl@0
   668
EXPORT_C TInt CWsGraphic::Share(TSecureId aClientId)
sl@0
   669
	{
sl@0
   670
	if(!IsActive())
sl@0
   671
		{
sl@0
   672
		return KErrNotReady;
sl@0
   673
		}
sl@0
   674
	return iPimpl->WriteReply(&aClientId,sizeof(TSecureId),EWsGdOpShare);
sl@0
   675
	}
sl@0
   676
sl@0
   677
/** 
sl@0
   678
Stops this graphic artwork from being shared with all client sessions with the specific Secure ID.
sl@0
   679
ShareGlobally() trumps explicit sharing.
sl@0
   680
@param aClientId the Secure ID of the client sessions to not share with
sl@0
   681
@return KErrNone if the graphic artwork is no longer shared, KErrNotFound if the graphic was not shared anyway, else one of the system-wide error codes
sl@0
   682
*/
sl@0
   683
EXPORT_C TInt CWsGraphic::UnShare(TSecureId aClientId)
sl@0
   684
	{
sl@0
   685
	if(!IsActive())
sl@0
   686
		{
sl@0
   687
		return KErrNotReady;
sl@0
   688
		}
sl@0
   689
	return iPimpl->WriteReply(&aClientId,sizeof(TSecureId),EWsGdOpUnShare);
sl@0
   690
	}
sl@0
   691
sl@0
   692
/**
sl@0
   693
Returns graphic artwork Id.
sl@0
   694
@return Graphic artwork Id. KNullWsGraphicId if graphic artwork is not active.
sl@0
   695
*/
sl@0
   696
EXPORT_C const TWsGraphicId& CWsGraphic::Id() const
sl@0
   697
	{
sl@0
   698
	if(IsActive())
sl@0
   699
		{
sl@0
   700
		return iPimpl->iId;
sl@0
   701
		}
sl@0
   702
	else
sl@0
   703
		{
sl@0
   704
		// fallback
sl@0
   705
		static const TInt KNullWsGraphicId[4] = //binary compatible with TWsGraphicId
sl@0
   706
			{
sl@0
   707
			0, 0, 0, 0
sl@0
   708
			};
sl@0
   709
		__ASSERT_COMPILE(sizeof(KNullWsGraphicId) == sizeof(TWsGraphicId));
sl@0
   710
		return reinterpret_cast<const TWsGraphicId&>(KNullWsGraphicId);
sl@0
   711
		}
sl@0
   712
	}
sl@0
   713
	
sl@0
   714
/** 
sl@0
   715
Checks whether a peer of this graphic artwork has been fully constructed on the server.
sl@0
   716
@return ETrue if this graphic artwork has a peer CWsGraphic on the server.
sl@0
   717
*/
sl@0
   718
EXPORT_C TBool CWsGraphic::IsActive() const
sl@0
   719
	{
sl@0
   720
	return (iPimpl && iPimpl->iWsHandle && (iPimpl->iFlags & CPimpl::EHasPeer));
sl@0
   721
	}
sl@0
   722
	
sl@0
   723
/**
sl@0
   724
Derived class can override this method to provide custom operations when the client is closed.
sl@0
   725
*/
sl@0
   726
EXPORT_C void CWsGraphic::OnClientClose()
sl@0
   727
	{
sl@0
   728
	}
sl@0
   729
	
sl@0
   730
/** 
sl@0
   731
Sends message to this graphic artwork peer on the server.
sl@0
   732
@param aData User specific data.
sl@0
   733
*/
sl@0
   734
EXPORT_C void CWsGraphic::SendMessage(const TDesC8& aData) const
sl@0
   735
	{
sl@0
   736
	TWsClCmdGdSendMessage cmd;
sl@0
   737
	cmd.iDataLen = aData.Size();
sl@0
   738
	__ASSERT_DEBUG(cmd.iDataLen, Panic(EW32PanicGraphicNullData));
sl@0
   739
	cmd.iRemoteReadData = ((aData.Size()+sizeof(cmd))>(TInt)(iPimpl->iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
sl@0
   740
	if(cmd.iRemoteReadData)
sl@0
   741
		{
sl@0
   742
		iPimpl->WriteReplyByProvidingRemoteReadAccess(&cmd,sizeof(cmd),&aData,EWsGdOpSendMsg);
sl@0
   743
		//ignore return value!
sl@0
   744
		}
sl@0
   745
	else
sl@0
   746
		{
sl@0
   747
		iPimpl->Write(&cmd,sizeof(cmd),aData.Ptr(),aData.Size(),EWsGdOpSendMsg);
sl@0
   748
		}
sl@0
   749
	}
sl@0
   750
sl@0
   751
EXPORT_C TInt CWsGraphic::SendSynchronMessage(const TDesC8& aData) const
sl@0
   752
	 {
sl@0
   753
	 TWsClCmdGdSendMessage cmd;
sl@0
   754
	 cmd.iDataLen = aData.Size();
sl@0
   755
	 __ASSERT_DEBUG(cmd.iDataLen, Panic(EW32PanicGraphicNullData));
sl@0
   756
	 cmd.iRemoteReadData = ((aData.Size()+sizeof(cmd))>(TInt)(iPimpl->iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
sl@0
   757
	 if(cmd.iRemoteReadData)
sl@0
   758
		{
sl@0
   759
		return iPimpl->WriteReplyByProvidingRemoteReadAccess(&cmd,sizeof(cmd),&aData,EWsGdOpSendSynchronMsg);
sl@0
   760
		}
sl@0
   761
	 else
sl@0
   762
		{
sl@0
   763
		return iPimpl->WriteReply(&cmd,sizeof(cmd),aData.Ptr(),aData.Size(),EWsGdOpSendSynchronMsg);
sl@0
   764
		}
sl@0
   765
	 }
sl@0
   766
sl@0
   767
/** 
sl@0
   768
Flushes window server command buffer
sl@0
   769
@return One of system-wide error codes.
sl@0
   770
*/
sl@0
   771
EXPORT_C TInt CWsGraphic::Flush() const
sl@0
   772
	{
sl@0
   773
	return iPimpl->iBuffer->Flush();
sl@0
   774
	}
sl@0
   775
sl@0
   776
EXPORT_C void CWsGraphic::Destroy()
sl@0
   777
/** Destroys the corresponding CWsGraphicDrawer instance on the server
sl@0
   778
@released
sl@0
   779
@publishedAll
sl@0
   780
*/	{
sl@0
   781
	if(iPimpl && (iPimpl->iFlags & CPimpl::ERegistered))
sl@0
   782
		{
sl@0
   783
		iPimpl->iManager->Remove(this);
sl@0
   784
		}
sl@0
   785
	if(IsActive()) // if iPimpl==NULL, IsActive() returns false
sl@0
   786
		{
sl@0
   787
		__ASSERT_DEBUG(iPimpl != NULL, Panic(EW32PanicGraphicInternal));
sl@0
   788
sl@0
   789
		iPimpl->Write(EWsGdOpFree);
sl@0
   790
		iPimpl->iWsHandle = 0;
sl@0
   791
		iPimpl->iFlags &= ~CPimpl::EHasPeer;
sl@0
   792
		iPimpl->iManager->ScheduleFlush();
sl@0
   793
		}
sl@0
   794
	}
sl@0
   795
sl@0
   796
EXPORT_C void CWsGraphic::SetGraphicExtension(MWsObjectProvider* aExt)
sl@0
   797
	{
sl@0
   798
	iPimpl->iExt = aExt;
sl@0
   799
	}
sl@0
   800
EXPORT_C RWsSession&  CWsGraphic::Session()
sl@0
   801
	{
sl@0
   802
	return *iPimpl->iManager;
sl@0
   803
	}
sl@0
   804
sl@0
   805
EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved1()
sl@0
   806
	{
sl@0
   807
	return KErrNotSupported;
sl@0
   808
	}
sl@0
   809
sl@0
   810
EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved2()
sl@0
   811
	{
sl@0
   812
	return KErrNotSupported;
sl@0
   813
	}
sl@0
   814
sl@0
   815
EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved3()
sl@0
   816
	{
sl@0
   817
	return KErrNotSupported;
sl@0
   818
	}
sl@0
   819
sl@0
   820
// TWsGraphicMsgFixedBase \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   821
sl@0
   822
EXPORT_C TWsGraphicMsgFixedBase::TWsGraphicMsgFixedBase(TUid aTypeId,TInt aSizeOfDerived):
sl@0
   823
/** Protected constructor for subclasses to call
sl@0
   824
@param aTypeId The UID representing this type of data
sl@0
   825
@param aSizeOf The size of the derived class
sl@0
   826
sl@0
   827
Example:
sl@0
   828
@code
sl@0
   829
TMyDerivedFixedMsg::TMyDerivedFixedMsg(): TWsGraphicMsgFixedBase(KUidMyDerivedType,sizeof(TMyDerivedFixedMsg)), ...
sl@0
   830
@endcode
sl@0
   831
*/	iTypeId(aTypeId), iSize(aSizeOfDerived-sizeof(TWsGraphicMsgFixedBase))
sl@0
   832
	{
sl@0
   833
	__ASSERT_COMPILE(sizeof(*this) == (sizeof(TInt32)*2));
sl@0
   834
	}
sl@0
   835
sl@0
   836
EXPORT_C TPtrC8 TWsGraphicMsgFixedBase::Pckg() const
sl@0
   837
/** @return this fixed message as a descriptor so that it can be passed as draw data in the CWindowGc::DrawWsGraphic command directly if only one such message is to be sent
sl@0
   838
*/	{
sl@0
   839
	return TPtrC8(reinterpret_cast<const TUint8*>(this),sizeof(*this) + iSize);
sl@0
   840
	}
sl@0
   841
sl@0
   842
EXPORT_C TUid TWsGraphicMsgFixedBase::TypeId() const
sl@0
   843
/** @return the UID identifying the type of the data that follows */
sl@0
   844
	{
sl@0
   845
	return iTypeId;
sl@0
   846
	}
sl@0
   847
sl@0
   848
EXPORT_C TInt TWsGraphicMsgFixedBase::Size() const
sl@0
   849
/** @return the size of the derived class (not including this fixed base class size) */
sl@0
   850
	{
sl@0
   851
	return iSize;
sl@0
   852
	}
sl@0
   853
sl@0
   854
// RWsGraphicMsgBuf \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   855
sl@0
   856
EXPORT_C RWsGraphicMsgBuf::RWsGraphicMsgBuf()
sl@0
   857
/** Default Constructor */
sl@0
   858
	{
sl@0
   859
	}
sl@0
   860
sl@0
   861
TInt RWsGraphicMsgBuf::IntAt(TInt aOfs) const
sl@0
   862
/** @internalComponent @released */
sl@0
   863
	{
sl@0
   864
	if((aOfs < 0) || ((aOfs+sizeof(TInt)) > Length()))
sl@0
   865
		{
sl@0
   866
		Panic(EW32PanicGraphicBadBuffer);
sl@0
   867
		}
sl@0
   868
	TInt ret;
sl@0
   869
	memcpy(&ret,Ptr()+aOfs,sizeof(TInt));
sl@0
   870
	return ret;
sl@0
   871
	}
sl@0
   872
sl@0
   873
EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,const TDesC8& aData)
sl@0
   874
/** Append a descriptor as data
sl@0
   875
	@param aTypeId the type of the message to append
sl@0
   876
	@param aData arbitrary length data consisting of the whole message
sl@0
   877
	@return KErrNone if successful, else a system-wide error code
sl@0
   878
*/	{
sl@0
   879
	TInt err = ExpandForAppend(aData.Length());
sl@0
   880
	if (err)
sl@0
   881
		{
sl@0
   882
		return err;
sl@0
   883
		}
sl@0
   884
	WriteHeader(aTypeId,aData.Length());
sl@0
   885
	// append data
sl@0
   886
	Insert(Length(),aData);
sl@0
   887
	return KErrNone;
sl@0
   888
	}
sl@0
   889
sl@0
   890
EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,const TDesC16& aData)
sl@0
   891
/** Append a descriptor as data
sl@0
   892
	@param aTypeId the type of the message to append
sl@0
   893
	@param aData arbitrary length data consisting of the whole message
sl@0
   894
	@return KErrNone if successful, else a system-wide error code
sl@0
   895
*/	{
sl@0
   896
	TPtr8 data(NULL,0);
sl@0
   897
	TInt err = Append(aTypeId,aData.Size(),data);
sl@0
   898
	if (err)
sl@0
   899
		{
sl@0
   900
		return err;
sl@0
   901
		}
sl@0
   902
	// append data
sl@0
   903
	data.Copy(reinterpret_cast<const TUint8*>(aData.Ptr()),aData.Size());
sl@0
   904
	return KErrNone;
sl@0
   905
	}
sl@0
   906
sl@0
   907
TInt RWsGraphicMsgBuf::ExpandForAppend(TInt aDataLen)
sl@0
   908
/** @internalComponent @released */
sl@0
   909
	{
sl@0
   910
	__ASSERT_COMPILE(sizeof(TInt) == sizeof(TInt32));
sl@0
   911
	const TInt required = (sizeof(TUid) + sizeof(TInt) + aDataLen);
sl@0
   912
	if(MaxLength() < (Length() + required))
sl@0
   913
		{
sl@0
   914
		TInt err = ReAlloc(Length()+required);
sl@0
   915
		if(KErrNone != err)
sl@0
   916
			{
sl@0
   917
			return err;
sl@0
   918
			}
sl@0
   919
		}
sl@0
   920
	return KErrNone;
sl@0
   921
	}
sl@0
   922
sl@0
   923
void RWsGraphicMsgBuf::WriteHeader(TUid aTypeId,TInt aLen)
sl@0
   924
/** @internalComponent @released */
sl@0
   925
	{
sl@0
   926
	__ASSERT_COMPILE(sizeof(TInt) == sizeof(TInt32));
sl@0
   927
	// append header
sl@0
   928
	TPckgBuf<TInt32> i(aTypeId.iUid);
sl@0
   929
	Insert(Length(),i);
sl@0
   930
	i() = aLen;
sl@0
   931
	Insert(Length(),i);
sl@0
   932
	}
sl@0
   933
sl@0
   934
EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,TInt aLen,TPtr8& aPtr)
sl@0
   935
/** Append a message of the specified length and type, and return a pointer to
sl@0
   936
	allow client code to modify the message.
sl@0
   937
	aPtr is only set if the append succeeds.
sl@0
   938
	aPtr is only valid until the next message is appended to the buffer.
sl@0
   939
	@param aTypeId the type of the message to append
sl@0
   940
	@param aLen the length of the message to be reserved
sl@0
   941
	@param aPtr a modifiable descriptor to be used by the client code to write into the message body
sl@0
   942
	@return KErrNone if successful, else a system-wide error code
sl@0
   943
*/	{
sl@0
   944
	TInt err = ExpandForAppend(aLen);
sl@0
   945
	if (err)
sl@0
   946
		{
sl@0
   947
		return err;
sl@0
   948
		}
sl@0
   949
	WriteHeader(aTypeId,aLen);
sl@0
   950
	// set aPtr
sl@0
   951
	TInt usedLen = Length();
sl@0
   952
	SetLength(usedLen+aLen);
sl@0
   953
	aPtr.Set(MidTPtr(usedLen,aLen));
sl@0
   954
	aPtr.Fill('_'); // prettier when debugging, but want consistant behaviour in release builds
sl@0
   955
	aPtr.Zero();
sl@0
   956
	return KErrNone;
sl@0
   957
	}
sl@0
   958
sl@0
   959
EXPORT_C void RWsGraphicMsgBuf::Remove(TInt aIndex)
sl@0
   960
/** Remove a message from the buffer
sl@0
   961
	@panic if the index is out of bounds
sl@0
   962
	@param aIndex the ordinal position of message to be removed
sl@0
   963
*/	{
sl@0
   964
	if((aIndex < 0) || (aIndex >= Count()))
sl@0
   965
		{
sl@0
   966
		Panic(EW32PanicGraphicBadBuffer);
sl@0
   967
		}
sl@0
   968
	TPtrC8 ptr = Data(aIndex);
sl@0
   969
	const TInt ofs = (ptr.Ptr()-(sizeof(TInt)*2)-Ptr());
sl@0
   970
	Delete(ofs,ptr.Length() + (sizeof(TInt)*2));
sl@0
   971
	}
sl@0
   972
sl@0
   973
EXPORT_C TInt RWsGraphicMsgBuf::Count() const
sl@0
   974
/** Returns the number of messages in the buffer
sl@0
   975
	@return the number of messages in the buffer
sl@0
   976
*/	{
sl@0
   977
	const TInt length = Length();
sl@0
   978
	TInt ofs = 0, count = 0;
sl@0
   979
	while(ofs < length)
sl@0
   980
		{
sl@0
   981
		count++;
sl@0
   982
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
   983
		}
sl@0
   984
	if(ofs != length)
sl@0
   985
		{
sl@0
   986
		Panic(EW32PanicGraphicBadBuffer);
sl@0
   987
		}
sl@0
   988
	return count;
sl@0
   989
	}
sl@0
   990
sl@0
   991
EXPORT_C TUid RWsGraphicMsgBuf::TypeId(TInt aIndex) const
sl@0
   992
/** Returns the Type ID of a message in the buffer
sl@0
   993
	@param aIndex the ordinal position of the message
sl@0
   994
	@return the Type ID of the message
sl@0
   995
	@panic if the index is out of bounds
sl@0
   996
*/	{
sl@0
   997
	const TInt length = Length();
sl@0
   998
	TInt ofs = 0, count = 0;
sl@0
   999
	while(ofs < length)
sl@0
  1000
		{
sl@0
  1001
		if(count == aIndex)
sl@0
  1002
			{
sl@0
  1003
			return TUid::Uid(IntAt(ofs));
sl@0
  1004
			}
sl@0
  1005
		count++;
sl@0
  1006
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1007
		}
sl@0
  1008
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1009
	return KNullUid; //dumb compiler
sl@0
  1010
	}
sl@0
  1011
sl@0
  1012
EXPORT_C TPtrC8 RWsGraphicMsgBuf::Data(TInt aIndex) const
sl@0
  1013
/** Returns a non-modifiable descriptor of a message body in the buffer
sl@0
  1014
	@param aIndex the ordinal position of the message
sl@0
  1015
	@return the message body
sl@0
  1016
	@panic if the index is out of bounds
sl@0
  1017
*/	{
sl@0
  1018
	const TInt length = Length();
sl@0
  1019
	TInt ofs = 0, count = 0;
sl@0
  1020
	while(ofs < length)
sl@0
  1021
		{
sl@0
  1022
		if(count == aIndex)
sl@0
  1023
			{
sl@0
  1024
			return Mid(ofs+(sizeof(TInt)*2),IntAt(ofs+sizeof(TInt)));
sl@0
  1025
			}
sl@0
  1026
		count++;
sl@0
  1027
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1028
		}
sl@0
  1029
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1030
	return TPtrC8(KNullDesC8()); //dumb compiler
sl@0
  1031
	}
sl@0
  1032
sl@0
  1033
EXPORT_C TPtr8 RWsGraphicMsgBuf::Data(TInt aIndex)
sl@0
  1034
/** Returns a modifiable descriptor of a message body in the buffer
sl@0
  1035
	The returned TPtr8 is only valid until the next message is appended to the buffer.
sl@0
  1036
	@param aIndex the ordinal position of the message
sl@0
  1037
	@return the message body
sl@0
  1038
	@panic if the index is out of bounds
sl@0
  1039
*/	{
sl@0
  1040
	const TInt length = Length();
sl@0
  1041
	TInt ofs = 0, count = 0;
sl@0
  1042
	while(ofs < length)
sl@0
  1043
		{
sl@0
  1044
		if(count == aIndex)
sl@0
  1045
			{
sl@0
  1046
			return MidTPtr(ofs+(sizeof(TInt)*2),IntAt(ofs+sizeof(TInt)));
sl@0
  1047
			}
sl@0
  1048
		count++;
sl@0
  1049
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1050
		}
sl@0
  1051
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1052
	return TPtr8(NULL,0); //dumb compiler
sl@0
  1053
	}
sl@0
  1054
sl@0
  1055
EXPORT_C const TDesC8& RWsGraphicMsgBuf::Pckg() const
sl@0
  1056
/** Returns the message buffer as a descriptor.  Example:
sl@0
  1057
	@code
sl@0
  1058
	RWsGraphicMsgBuf msgBuf;
sl@0
  1059
	msgBuf.Append(...);
sl@0
  1060
	...
sl@0
  1061
	TWsGraphicId id(...);
sl@0
  1062
	SystemGc().DrawWsGraphic(id,Rect(),msgBuf.Pckg());
sl@0
  1063
	msgBuf.Close();
sl@0
  1064
	@endcode
sl@0
  1065
	@see CWindowGc::DrawWsGraphic
sl@0
  1066
	@see CCoeControl::Draw
sl@0
  1067
	@return the message buffer to be attached a command to draw a CWsGraphic
sl@0
  1068
*/	{
sl@0
  1069
	return *this;
sl@0
  1070
	}
sl@0
  1071
sl@0
  1072
EXPORT_C TInt RWsGraphicMsgBuf::Append(const TWsGraphicMsgFixedBase& aMsg)
sl@0
  1073
/** Append a fixed-size message
sl@0
  1074
	@param aMsg the fixed-size message to append
sl@0
  1075
	@return KErrNone if successful, else a system-wide error code
sl@0
  1076
*/	{
sl@0
  1077
	__ASSERT_COMPILE(sizeof(TWsGraphicMsgFixedBase) == (sizeof(TInt)*2));
sl@0
  1078
	TInt err = ExpandForAppend(aMsg.Size());
sl@0
  1079
	if (err)
sl@0
  1080
		{
sl@0
  1081
		return err;
sl@0
  1082
		}
sl@0
  1083
	// append data
sl@0
  1084
	RBuf8::Append(reinterpret_cast<const TUint8*>(&aMsg),sizeof(TWsGraphicMsgFixedBase) + aMsg.Size());
sl@0
  1085
	Count();
sl@0
  1086
	return KErrNone;
sl@0
  1087
	}
sl@0
  1088
sl@0
  1089
EXPORT_C void RWsGraphicMsgBuf::GetFixedMsg(TWsGraphicMsgFixedBase& aMsg,TInt aIndex) const
sl@0
  1090
/** Returns a copy of a fixed-size message in the buffer
sl@0
  1091
	@param a copy of the message
sl@0
  1092
	@param aIndex the ordinal position of the message
sl@0
  1093
	@panic if the index is out of bounds
sl@0
  1094
	@panic the message specified is not of the correct type
sl@0
  1095
*/	{
sl@0
  1096
	__ASSERT_COMPILE(sizeof(TWsGraphicMsgFixedBase) == (sizeof(TInt32)*2));
sl@0
  1097
	const TInt KHeaderSize = sizeof(TWsGraphicMsgFixedBase);
sl@0
  1098
	const TInt length = Length();
sl@0
  1099
	TInt ofs = 0, count = 0;
sl@0
  1100
	while(ofs < length)
sl@0
  1101
		{
sl@0
  1102
		if(count == aIndex)
sl@0
  1103
			{
sl@0
  1104
			if((TUid::Uid(IntAt(ofs)) != aMsg.TypeId()) ||
sl@0
  1105
				(IntAt(ofs+sizeof(TInt)) != aMsg.Size()) ||
sl@0
  1106
				((ofs + KHeaderSize + aMsg.Size()) > length))
sl@0
  1107
				{
sl@0
  1108
				Panic(EW32PanicGraphicBadBuffer);
sl@0
  1109
				}
sl@0
  1110
			memcpy(&aMsg,(Ptr()+ofs),KHeaderSize + aMsg.Size());
sl@0
  1111
			return;
sl@0
  1112
			}
sl@0
  1113
		count++;
sl@0
  1114
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1115
		}
sl@0
  1116
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1117
	}
sl@0
  1118