First public contribution.
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 = CWsTop::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 && CWsTop::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,*CWsTop::WindowServer(),id,data,aParams.CreateGraphic->iClientHandle);
110 User::LeaveIfError(CWsTop::WindowServer()->AddGraphicDrawer(drawer->Drawer()));
114 User::LeaveIfError(CWsTop::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 CWsTop::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 // trigger redraws if anyone is drawing this graphic before it exists
172 CWsTop::WindowServer()->RemoveGraphicDrawer(iDrawer->Id());
176 case EWsGdOpSendSynchronMsg:
178 const TWsClCmdGdSendMessage& cmd = *pData.GraphicSendMessage;
179 __ASSERT_DEBUG(cmd.iDataLen,WsOwner()->PPanic(EWservPanicPermissionDenied));
182 if(cmd.iRemoteReadData)
184 HBufC8* wsAlloc = NULL;
185 TUint8* drawerAlloc = NULL;
187 // try to get the drawer to allocate for us
188 MWsGraphicDrawerMessageAlloc* drawerAllocator = iDrawer->ObjectInterface<MWsGraphicDrawerMessageAlloc>();
191 drawerAlloc = reinterpret_cast<TUint8*>(drawerAllocator->Alloc(cmd.iDataLen));
193 des.Set(drawerAlloc,0,cmd.iDataLen);
195 // else use the normal WSERV default heap
198 wsAlloc = HBufC8::NewLC(cmd.iDataLen);
199 des.Set(wsAlloc->Des());
202 WsOwner()->RemoteRead(des,0);
203 TBool receivedOk = (des.Size() == cmd.iDataLen);
207 if(aOpcode == EWsGdOpSendMsg)
209 iDrawer->HandleMessage(des);
213 MWsGraphicHandleSynchronMessage* obj = iDrawer->ObjectInterface<MWsGraphicHandleSynchronMessage>();
216 SetReply(obj->HandleSynchronMessage(des));
218 SetReply(KErrNotSupported);
225 drawerAllocator->Free(drawerAlloc);
229 CleanupStack::PopAndDestroy(wsAlloc);
231 // defer panic until after dealloc
234 WsOwner()->PPanic(EWservPanicBuffer);
239 const TPtrC8 data = CWsClient::BufferTPtr8((TText8*)(pData.GraphicSendMessage+1),cmd.iDataLen);
240 if(aOpcode == EWsGdOpSendMsg)
242 iDrawer->HandleMessage(data);
246 MWsGraphicHandleSynchronMessage* obj = iDrawer->ObjectInterface<MWsGraphicHandleSynchronMessage>();
248 SetReply(obj->HandleSynchronMessage(data));
250 SetReply(KErrNotSupported);
256 case EWsGdOpGetGraphicId:
258 __ASSERT_COMPILE(sizeof(TWsClCmdGdGetId) == sizeof(TGraphicDrawerId));
259 const TGraphicDrawerId id = iDrawer->Id();
260 CWsClient::ReplyBuf(&id,sizeof(id));
263 case EWsGdOpShareGlobally:
264 SetReply(iDrawer->ShareGlobally());
266 case EWsGdOpUnShareGlobally:
267 SetReply(iDrawer->UnShareGlobally());
271 const TSecureId& secid = *reinterpret_cast<const TSecureId*>(aCmdData);
272 SetReply(iDrawer->Share(secid));
277 const TSecureId& secid = *reinterpret_cast<const TSecureId*>(aCmdData);
278 SetReply(iDrawer->UnShare(secid));
282 WsOwner()->PPanic(EWservPanicOpcode);
287 CWsGraphicDrawer* CWsGraphicDrawerObject::Drawer()
292 const CWsGraphicDrawer* CWsGraphicDrawerObject::Drawer() const
297 // CWsGraphicMessageQueue \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\
299 CWsGraphicMessageQueue::CMessage* CWsGraphicMessageQueue::CMessage::New(const TDesC8& aData)
301 return new(aData.Size()) CMessage(aData);
304 CWsGraphicMessageQueue::CMessage::CMessage(const TDesC8& aData):
305 iData((reinterpret_cast<TUint8*>(this)+sizeof(*this)),aData.Size())
310 TPtrC8 CWsGraphicMessageQueue::CMessage::Data() const
315 void CWsGraphicMessageQueue::CMessage::Release()
320 CWsGraphicMessageQueue::CMessage::~CMessage()
324 CWsGraphicMessageQueue::CWsGraphicMessageQueue(CWsClient *aOwner):
329 CWsGraphicMessageQueue::~CWsGraphicMessageQueue()
333 CWsMessageData* msg = iHead;
334 iHead = iHead->iNext;
339 TInt CWsGraphicMessageQueue::TopClientHandle() const
342 return iHead->iClientHandle;
348 same functionality as CEventBase::GetData, but this one also inserts the integer header in the reply.
350 void CWsGraphicMessageQueue::GetDataWithHeader(TUint aHeader, const TDesC8& aData, TInt aDataLen)
352 TPckgBuf<TUint> hdr = aHeader | (EWsGraphMessageTypeUser & 0x03);
353 CWsClient::ReplyBuf(hdr.Ptr(), sizeof(TUint));
354 GetData((void*)aData.Ptr(), aDataLen);
357 void CWsGraphicMessageQueue::GetGraphicMessage()
359 CWsMessageData* top = Pop();
360 WS_ASSERT_DEBUG(top && top->Data().Length(), EWsPanicWsGraphic);
363 GetDataWithHeader(top->iClientHandle, top->Data(), top->Data().Size());
364 CWsGraphicDrawerObject* obj = iWsOwner->DrawerObject(top->iDrawer);
367 MWsGraphicMessageCallback* messageCallback = obj->Drawer()->ObjectInterface<MWsGraphicMessageCallback>();
370 messageCallback->HandleMessageDelivery(top->iId, KErrNone);
378 void CWsGraphicMessageQueue::AbortMessage(TInt aError)
380 CWsMessageData* top = Pop();
381 WS_ASSERT_DEBUG(top, EWsPanicWsGraphic);
384 CWsGraphicDrawerObject* obj = iWsOwner->DrawerObject(top->iDrawer);
387 MWsGraphicMessageCallback* messageCallback = obj->Drawer()->ObjectInterface<MWsGraphicMessageCallback>();
390 messageCallback->HandleMessageDelivery(top->iId, aError);
397 void CWsGraphicMessageQueue::EventReady(const RMessagePtr2& aEventMsg)
399 CEventBase::EventReady(aEventMsg);
402 SignalEvent(sizeof(TInt) + iHead->Data().Size());
407 void CWsGraphicMessageQueue::Queue(CWsMessageData* aMessage)
409 WS_ASSERT_DEBUG(aMessage && !aMessage->iNext, EWsPanicWsGraphic);
414 iTail->iNext = aMessage;
419 iHead = iTail = aMessage;
420 if(!iEventMsg.IsNull())
421 SignalEvent(sizeof(TInt) + iHead->Data().Size());
426 CWsMessageData* CWsGraphicMessageQueue::Pop()
428 CWsMessageData* ret = NULL;
432 iHead = iHead->iNext;