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