Update contrib.
1 // Copyright (c) 1995-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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
17 #include "WSGRAPHICDRAWERFACTORY.H"
22 TInt CWsGraphicDrawerObject::TransientSequence=1;
23 TBool CWsGraphicDrawerObject::RollOver=EFalse;
25 CWsGraphicDrawerObject* CWsGraphicDrawerObject::NewL(CWsClient* aClient,const TWsClCmdUnion &aParams)
29 WS_PANIC_DEBUG(EWsPanicWsGraphic);
30 User::Leave(KErrGeneral);
33 id.iId = aParams.CreateGraphic->iId;
34 id.iIsUid = !(EWsGraphicIdTransient & aParams.CreateGraphic->iFlags);
35 // check for collision
36 const CWsGraphicDrawer* dup = aClient->WindowServer().ResolveGraphic(id);
37 // request-specific checks
40 // must be exact same session
41 if(&(dup->Owner()) != aClient)
43 /* Someone is squatting on someone else's uid, or trying to,
44 but there is no good way to know which. */
45 __DEBUG_ONLY(aClient->PPanic(EWservPanicPermissionDenied);)
46 User::Leave(KErrAlreadyExists);
48 // dupping or otherwise must be explicitly intended by the client
49 if(!(EWsGraphicReplace & aParams.CreateGraphic->iFlags))
51 aClient->PPanic(EWservPanicPermissionDenied);
56 // dupping or otherwise must be explicitly intended by the client
57 if(EWsGraphicReplace & aParams.CreateGraphic->iFlags)
59 aClient->PPanic(EWservPanicPermissionDenied);
63 // police creation of artwork with UIDs
64 _LIT_SECURITY_POLICY_C1(KSecurityPolicyCreateWithUid,ECapabilityProtServ);
65 if(!KSecurityPolicyCreateWithUid.CheckPolicy(aClient->Client()))
67 __DEBUG_ONLY(aClient->PPanic(EWservPanicPermissionDenied);)
68 User::Leave(KErrPermissionDenied);
73 // allocate a new transient id
74 // assumption: that there are more values of a TInt than there are possibly Transient objects
77 id.iId = TransientSequence++;
78 RollOver |= !TransientSequence;
80 while(RollOver && aClient->WindowServer().ResolveGraphic(id)); // until no collision
83 // create the drawer object
85 HBufC8* dataBuf = NULL;
86 if(aParams.CreateGraphic->iRemoteReadData)
88 const TInt len = aParams.CreateGraphic->iDataLen;
89 if ((len >= KMaxTInt/4) || (len < 0))
90 aClient->PPanic(EWservPanicBuffer);
91 dataBuf = HBufC8::NewLC(len);
92 TPtr8 des = dataBuf->Des();
93 aClient->RemoteRead(des,0);
96 aClient->PPanic(EWservPanicBuffer);
102 data.Set(CWsClient::BufferTPtr8((TText8*)(aParams.CreateGraphic+1),aParams.CreateGraphic->iDataLen));
105 CWsGraphicDrawerObject* drawer = new(ELeave) CWsGraphicDrawerObject(aClient);
106 CleanupStack::PushL(drawer);
107 drawer->ConstructL(aParams.CreateGraphic->iType,aClient->WindowServer(),id,data,aParams.CreateGraphic->iClientHandle);
110 User::LeaveIfError(aClient->WindowServer().AddGraphicDrawer(drawer->Drawer()));
114 User::LeaveIfError(aClient->WindowServer().SwapGraphicDrawer(drawer->Drawer()));
116 // take off cleanup stack
117 CleanupStack::Pop(drawer);
120 CleanupStack::PopAndDestroy(dataBuf);
122 // delete dup, which means resolving the object that encapsulates it
125 CWsGraphicDrawerObject* obj = aClient->DrawerObject(dup);
126 WS_ASSERT_DEBUG(obj, EWsPanicWsGraphic);
132 // trigger redraws if anyone is drawing this graphic before it exists
135 aClient->WindowServer().Invalidate(id);
141 CWsGraphicDrawerObject::CWsGraphicDrawerObject(CWsClient* aOwner):
142 CWsObject(aOwner,WS_HANDLE_GRAPHIC_DRAWER)
146 CWsGraphicDrawerObject::~CWsGraphicDrawerObject()
151 void CWsGraphicDrawerObject::ConstructL(TUid aType,MWsGraphicDrawerEnvironment& aEnv,const TGraphicDrawerId& aId,const TDesC8& aData,TInt aClientHandle)
153 CWsGraphicDrawer* drawer = WsGraphicDrawer::CreateLC(aType,aEnv,aId,*WsOwner(),aData);
155 iClientHandle = aClientHandle;
157 CleanupStack::Pop(iDrawer);
160 void CWsGraphicDrawerObject::CommandL(TInt aOpcode, const TAny *aCmdData)
162 WS_ASSERT_DEBUG(iDrawer, EWsPanicWsGraphic);
167 pData.any = aCmdData;
171 WsOwner()->WindowServer().RemoveGraphicDrawer(iDrawer->Id());
175 case EWsGdOpSendSynchronMsg:
177 const TWsClCmdGdSendMessage& cmd = *pData.GraphicSendMessage;
178 __ASSERT_DEBUG(cmd.iDataLen,WsOwner()->PPanic(EWservPanicPermissionDenied));
181 if(cmd.iRemoteReadData)
183 HBufC8* wsAlloc = NULL;
184 TUint8* drawerAlloc = NULL;
186 // try to get the drawer to allocate for us
187 MWsGraphicDrawerMessageAlloc* drawerAllocator = iDrawer->ObjectInterface<MWsGraphicDrawerMessageAlloc>();
190 drawerAlloc = reinterpret_cast<TUint8*>(drawerAllocator->Alloc(cmd.iDataLen));
192 des.Set(drawerAlloc,0,cmd.iDataLen);
194 // else use the normal WSERV default heap
197 wsAlloc = HBufC8::NewLC(cmd.iDataLen);
198 des.Set(wsAlloc->Des());
201 WsOwner()->RemoteRead(des,0);
202 TBool receivedOk = (des.Size() == cmd.iDataLen);
206 if(aOpcode == EWsGdOpSendMsg)
208 iDrawer->HandleMessage(des);
212 MWsGraphicHandleSynchronMessage* obj = iDrawer->ObjectInterface<MWsGraphicHandleSynchronMessage>();
215 SetReply(obj->HandleSynchronMessage(des));
217 SetReply(KErrNotSupported);
224 drawerAllocator->Free(drawerAlloc);
228 CleanupStack::PopAndDestroy(wsAlloc);
230 // defer panic until after dealloc
233 WsOwner()->PPanic(EWservPanicBuffer);
238 const TPtrC8 data = CWsClient::BufferTPtr8((TText8*)(pData.GraphicSendMessage+1),cmd.iDataLen);
239 if(aOpcode == EWsGdOpSendMsg)
241 iDrawer->HandleMessage(data);
245 MWsGraphicHandleSynchronMessage* obj = iDrawer->ObjectInterface<MWsGraphicHandleSynchronMessage>();
247 SetReply(obj->HandleSynchronMessage(data));
249 SetReply(KErrNotSupported);
255 case EWsGdOpGetGraphicId:
257 __ASSERT_COMPILE(sizeof(TWsClCmdGdGetId) == sizeof(TGraphicDrawerId));
258 const TGraphicDrawerId id = iDrawer->Id();
259 CWsClient::ReplyBuf(&id,sizeof(id));
262 case EWsGdOpShareGlobally:
263 SetReply(iDrawer->ShareGlobally());
265 case EWsGdOpUnShareGlobally:
266 SetReply(iDrawer->UnShareGlobally());
270 const TSecureId& secid = *reinterpret_cast<const TSecureId*>(aCmdData);
271 SetReply(iDrawer->Share(secid));
276 const TSecureId& secid = *reinterpret_cast<const TSecureId*>(aCmdData);
277 SetReply(iDrawer->UnShare(secid));
281 WsOwner()->PPanic(EWservPanicOpcode);
286 CWsGraphicDrawer* CWsGraphicDrawerObject::Drawer()
291 const CWsGraphicDrawer* CWsGraphicDrawerObject::Drawer() const
296 // CWsGraphicMessageQueue \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\
298 CWsGraphicMessageQueue::CMessage* CWsGraphicMessageQueue::CMessage::New(const TDesC8& aData)
300 return new(aData.Size()) CMessage(aData);
303 CWsGraphicMessageQueue::CMessage::CMessage(const TDesC8& aData):
304 iData((reinterpret_cast<TUint8*>(this)+sizeof(*this)),aData.Size())
309 TPtrC8 CWsGraphicMessageQueue::CMessage::Data() const
314 void CWsGraphicMessageQueue::CMessage::Release()
319 CWsGraphicMessageQueue::CMessage::~CMessage()
323 CWsGraphicMessageQueue::CWsGraphicMessageQueue(CWsClient *aOwner):
328 CWsGraphicMessageQueue::~CWsGraphicMessageQueue()
332 CWsMessageData* msg = iHead;
333 iHead = iHead->iNext;
338 TInt CWsGraphicMessageQueue::TopClientHandle() const
341 return iHead->iClientHandle;
347 same functionality as CEventBase::GetData, but this one also inserts the integer header in the reply.
349 void CWsGraphicMessageQueue::GetDataWithHeader(TUint aHeader, const TDesC8& aData, TInt aDataLen)
351 TPckgBuf<TUint> hdr = aHeader | (EWsGraphMessageTypeUser & 0x03);
352 CWsClient::ReplyBuf(hdr.Ptr(), sizeof(TUint));
353 GetData((void*)aData.Ptr(), aDataLen);
356 void CWsGraphicMessageQueue::GetGraphicMessage()
358 CWsMessageData* top = Pop();
359 WS_ASSERT_DEBUG(top && top->Data().Length(), EWsPanicWsGraphic);
362 GetDataWithHeader(top->iClientHandle, top->Data(), top->Data().Size());
363 CWsGraphicDrawerObject* obj = iWsOwner->DrawerObject(top->iDrawer);
366 MWsGraphicMessageCallback* messageCallback = obj->Drawer()->ObjectInterface<MWsGraphicMessageCallback>();
369 messageCallback->HandleMessageDelivery(top->iId, KErrNone);
377 void CWsGraphicMessageQueue::AbortMessage(TInt aError)
379 CWsMessageData* top = Pop();
380 WS_ASSERT_DEBUG(top, EWsPanicWsGraphic);
383 CWsGraphicDrawerObject* obj = iWsOwner->DrawerObject(top->iDrawer);
386 MWsGraphicMessageCallback* messageCallback = obj->Drawer()->ObjectInterface<MWsGraphicMessageCallback>();
389 messageCallback->HandleMessageDelivery(top->iId, aError);
396 void CWsGraphicMessageQueue::EventReady(const RMessagePtr2& aEventMsg)
398 CEventBase::EventReady(aEventMsg);
401 SignalEvent(sizeof(TInt) + iHead->Data().Size());
406 void CWsGraphicMessageQueue::Queue(CWsMessageData* aMessage)
408 WS_ASSERT_DEBUG(aMessage && !aMessage->iNext, EWsPanicWsGraphic);
413 iTail->iNext = aMessage;
418 iHead = iTail = aMessage;
419 if(!iEventMsg.IsNull())
420 SignalEvent(sizeof(TInt) + iHead->Data().Size());
425 CWsMessageData* CWsGraphicMessageQueue::Pop()
427 CWsMessageData* ret = NULL;
431 iHead = iHead->iNext;