First public contribution.
1 // Copyright (c) 1994-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.
14 // RWsBuffer class, handles buffering and flushing of window server commands
19 #include "../SERVER/w32cmd.h"
25 GLDEF_C void Assert(TW32Assert aAssert)
27 _LIT(KW32AssertCategory,"W32 Assert");
28 User::Panic(KW32AssertCategory,aAssert);
31 GLDEF_C void Panic(TW32Panic aPanic)
33 _LIT(KW32PanicCategory,"W32");
34 User::Panic(KW32PanicCategory,aPanic);
39 RWsBuffer::RWsBuffer(RWsSession *aSession) : iSession(aSession), iManager(NULL),
40 #if defined(__AUTO_FLUSH)
45 iBuf(NULL,0,0), iNext(NULL), iPreviousHandle(0), iBufSize(0), iMaxBufSize(EMinBufferSize),
46 iDirectAcessCount(0), iInvalidBitmapArray(EFalse)
50 TInt WsFbsDestroyCallBack(TAny* aBitmapHandle)
52 TInt* bitmapHandle=static_cast<TInt*>(aBitmapHandle);
53 RWsBuffer::FlushAllBuffers(aBitmapHandle ? *bitmapHandle : 0);
58 void RWsBuffer::SetCallBack()
60 for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
62 return; // Already linked
63 iNext=(RWsBuffer *)Dll::Tls();
65 if (iNext==NULL) // First connection so set callback
66 RFbsSession::GetSession()->SetCallBack(TCallBack(WsFbsDestroyCallBack,NULL));
69 void RWsBuffer::CancelCallBack()
71 RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();
76 RFbsSession::GetSession()->ResetCallBack(); // Last connection closing so cancel the callback
94 void RWsBuffer::Close()
96 User::Free((TAny *)iBuf.Ptr());
99 void RWsBuffer::Destroy()
107 TInt RWsBuffer::Flush(const TIpcArgs* aIpcArgs,TBool aRequestFinish)
109 iBitmapArray.Reset();
110 iInvalidBitmapArray=EFalse;
111 if (iBuf.Length()==0)
119 // check that the caller hasn't used the first slot
120 ipcArgs.Set(KBufferMessageSlot,TIpcArgs::ENothing);
121 __ASSERT_ALWAYS(Mem::Compare(REINTERPRET_CAST(const TUint8*, &ipcArgs), sizeof(TIpcArgs), REINTERPRET_CAST(const TUint8*, aIpcArgs), sizeof(TIpcArgs))==0,Panic(EW32PanicUsingReservedIpcSlot));
123 ipcArgs.Set(KBufferMessageSlot,&iBuf);
126 ret=iSession->DoFlush(ipcArgs);
128 ret=iSession->DoSyncMsgBuf(ipcArgs);
134 void RWsBuffer::FlushAllBuffers(TInt aBitmapHandle)
136 for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
138 if(!aBitmapHandle || buffer->iInvalidBitmapArray || (buffer->iBitmapArray.FindInOrder(aBitmapHandle)!=KErrNotFound))
144 inline void RWsBuffer::SetAndLimitMaxBufSize(TInt aMaxBufSize)
145 { // apply upper & lower limits to input buffer size
146 if (aMaxBufSize < EMinBufferSize)
148 iMaxBufSize = EMinBufferSize;
150 else if (aMaxBufSize > EMaxBufferSize)
152 iMaxBufSize = EMaxBufferSize;
156 iMaxBufSize = aMaxBufSize;
160 void RWsBuffer::SetBufferSizeL(TInt aBufSize)
162 SetAndLimitMaxBufSize(aBufSize);
163 ReAllocBufferL(iMaxBufSize);
166 void RWsBuffer::SetMaxBufferSizeL(TInt aMaxBufSize)
168 SetAndLimitMaxBufSize(aMaxBufSize);
170 if (iMaxBufSize < iBufSize)
171 { // shrink to new maximum
172 ReAllocBufferL(iMaxBufSize);
176 // initial allocation should be (at least) a quarter of the requested size
177 TInt minSize = Max( (iMaxBufSize + 3) >> 2, EMinBufferSize);
178 if (minSize > iBufSize)
179 { // new or enlarged buffer
180 ReAllocBufferL(minSize);
184 __ASSERT_DEBUG((iBufSize >= EMinBufferSize) && (iBufSize <= iMaxBufSize), Assert(EW32AssertBufferLogic));
187 // Flush() buffer, try to ReAlloc, Leave if the ReAlloc fails.
188 void RWsBuffer::ReAllocBufferL(TInt aNewSize)
190 if (aNewSize != iBufSize)
193 if (!ReAllocBuffer(aNewSize))
195 User::LeaveNoMemory();
200 TBool RWsBuffer::ReAllocBuffer(TInt aNewSize)
202 TUint8* ptr = const_cast<TUint8*>(iBuf.Ptr());
203 __ASSERT_DEBUG((iBufSize == 0) || (ptr != NULL), Assert(EW32AssertBufferLogic));
204 const TInt len = iBuf.Length();
205 TUint8* newPtr = static_cast<TUint8*>(User::ReAlloc(ptr, aNewSize));
207 { // realloc was successful
208 iBuf.Set(newPtr, len, aNewSize);
215 /* Expand the buffer, to allow new drawing command to fit.
217 Called either when trying to store additional commands to queue for Wserv, or
218 the trying to send a command bigger than the current buffer size.
220 Failure to expand the buffer is a minor problem in the first case but a big
221 problem in the second.
223 @param aRequiredSpace Size of buffer increase required.
224 @param aMsgSize If expanding the buffer fails then needs this value to know whether Flush() is good enough.
225 @return ETrue if there is enough space, EFalse if not.
227 void RWsBuffer::GrowBuffer(TInt aRequiredSpace, TInt aMsgSize)
229 __ASSERT_DEBUG(iBufSize < iMaxBufSize, Assert(EW32AssertBufferLogic));
230 // maximum size will be big enough?
231 __ASSERT_ALWAYS(aMsgSize <= iMaxBufSize, Panic(EW32PanicDataExceedsBufferLength));
233 // double or quad the current size, then limit it
234 TInt newSize = Min((iBufSize >= aRequiredSpace) ? iBufSize << 1 : iBufSize << 2, iMaxBufSize);
236 if (!ReAllocBuffer(newSize))
239 if (aMsgSize > iBufSize)
240 { // message is too big for buffer
241 Panic(EW32PanicDataExceedsBufferLength);
247 TBool RWsBuffer::SetAutoFlush(TBool aState)
252 #if defined(__AUTO_FLUSH)
262 TInt RWsBuffer::DoWrite(TInt aHandle, TUint aOpcode, TBool aFlush, const TIpcArgs* aIpcArgs, const TAny* aData, TInt aLength, const TAny* aData2, TInt aLength2)
264 __ASSERT_DEBUG(((TUint32) aOpcode) < 0x8000, Assert(EW32AssertIllegalOpcode));
265 __ASSERT_DEBUG((aLength&0x3) == 0, Assert(EW32AssertOddLengthData));
269 xtra = 4 - (aLength2&0x3); // Round data upto a multiple of 4
276 const TInt msgSize = aLength + aLength2 + xtra + static_cast<TInt>(sizeof(TWsCmdHeader));
277 TInt available = iBuf.MaxLength() - iBuf.Length();
278 if (msgSize > available)
280 if (iBufSize >= iMaxBufSize)
281 { // buffer is maximum size already
285 { // try to grow buffer
286 if ( (iBuf.Length() + msgSize) > iMaxBufSize)
287 { // growing alone will not make enough extra space
289 available = iBufSize;
292 const TInt requiredSpace = msgSize - available;
293 if (requiredSpace > 0)
295 GrowBuffer(requiredSpace, msgSize);
300 TWsCmdHeader cmdHeader;
301 cmdHeader.iBase.iOpcode = (TInt16)aOpcode;
302 cmdHeader.iBase.iCmdLength = (TInt16)(aLength + aLength2 + xtra);
304 // For performance reasons we only pass in the handle if it is different
305 // from the previous command
306 if (aHandle == iPreviousHandle)
308 iBuf.Append((TUint8 *)&cmdHeader.iBase,sizeof(cmdHeader.iBase));
312 iPreviousHandle = aHandle;
313 cmdHeader.iBase.iOpcode|=EWsOpcodeHandle;
314 cmdHeader.iDestHandle = aHandle;
315 iBuf.Append((TUint8 *)&cmdHeader,sizeof(cmdHeader));
320 iBuf.Append((TUint8 *)aData, aLength);
324 iBuf.Append((TUint8 *)aData2, aLength2);
325 iBuf.AppendFill(0,xtra);
330 return Flush(aIpcArgs);
335 void RWsBuffer::Write(TInt handle,TUint opcode)
337 DoWrite(handle, opcode, iAutoFlush, NULL);
340 void RWsBuffer::Write(TInt handle,TUint opcode,const TAny *pData, TInt length)
342 DoWrite(handle, opcode, iAutoFlush, NULL, pData, length);
345 void RWsBuffer::Write(TInt handle,TUint opcode,const TAny *pData, TInt length,const TAny *pData2, TInt length2)
347 DoWrite(handle, opcode, iAutoFlush, NULL, pData, length, pData2, length2);
350 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TIpcArgs* aIpcArgs)
352 return DoWrite(handle, opcode, ETrue, aIpcArgs);
355 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData, TInt length,const TIpcArgs* aIpcArgs)
357 return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length);
360 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData,TInt length,const TAny *pData2,TInt length2,const TIpcArgs* aIpcArgs)
362 return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length, pData2, length2);
365 TInt RWsBuffer::WriteReplyP(TInt aHandle, TUint aOpcode, const TWriteDescriptorType& aReplyBuffer)
368 aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
369 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs);
372 TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData,TInt aLength,const TWriteDescriptorType& aReplyBuffer)
375 aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
376 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
379 TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData1,TInt aLengthData1,const TAny *aData2,TInt aLengthData2,const TWriteDescriptorType& aReplyBuffer)
382 aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
383 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData1, aLengthData1, aData2, aLengthData2);
386 TInt RWsBuffer::WriteReplyWs(TUint opcode)
388 // Do a WriteReply using the sessions handle
391 return(iSession->WriteReply(opcode));
394 TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, TUint aOpcode)
396 // Do a WriteReply using the sessions handle
399 return(iSession->WriteReply(pData,aLength,aOpcode));
402 TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, const TAny *pData2, TInt aLength2, TUint aOpcode)
404 // Do a WriteReply using the sessions handle
407 return(iSession->WriteReply(pData,aLength,pData2,aLength2,aOpcode));
410 TInt RWsBuffer::WriteReplyByProvidingRemoteReadAccess(TInt aHandle,TUint aOpcode,const TAny *aData, TInt aLength,const TReadDescriptorType& aRemoteReadBuffer)
413 aRemoteReadBuffer.SetDescriptorOnIpcArgs(ipcArgs);
414 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
417 void RWsBuffer::AddToBitmapArray(TInt aBitmapHandle)
419 if(aBitmapHandle && !iInvalidBitmapArray)
421 if(iBitmapArray.InsertInOrder(aBitmapHandle)==KErrNoMemory)
423 iInvalidBitmapArray=ETrue;
427 void RWsBuffer::SetWsGraphicManager(CWsGraphic::CManager* aManager)
429 __ASSERT_DEBUG(!WsGraphicManager(),Panic(EW32PanicGraphicInternal));
433 CWsGraphic::CManager* RWsBuffer::WsGraphicManager()
435 for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
436 if (buffer->iManager)
437 return buffer->iManager;
438 return NULL; // does not yet exist