First public contribution.
1 // Copyright (c) 1994-2010 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),
49 iDirectAcessCount(0), iInvalidBitmapArray(EFalse), iWindowSizeCache(NULL)
53 TInt WsFbsDestroyCallBack(TAny* aBitmapHandle)
55 TInt* bitmapHandle=static_cast<TInt*>(aBitmapHandle);
56 RWsBuffer::FlushAllBuffers(aBitmapHandle ? *bitmapHandle : 0);
61 void RWsBuffer::SetCallBack()
63 for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
65 return; // Already linked
66 iNext=(RWsBuffer *)Dll::Tls();
68 if (iNext==NULL) // First connection so set callback
69 RFbsSession::GetSession()->SetCallBack(TCallBack(WsFbsDestroyCallBack,NULL));
72 void RWsBuffer::CancelCallBack()
74 RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();
79 RFbsSession::GetSession()->ResetCallBack(); // Last connection closing so cancel the callback
97 void RWsBuffer::Close()
99 User::Free((TAny *)iBuf.Ptr());
100 if (iWindowSizeCache)
102 iWindowSizeCache->Close();
103 delete iWindowSizeCache;
104 iWindowSizeCache = NULL;
108 void RWsBuffer::Destroy()
116 TInt RWsBuffer::Flush(const TIpcArgs* aIpcArgs,TBool aRequestFinish)
118 iBitmapArray.Reset();
119 iInvalidBitmapArray=EFalse;
120 if (iBuf.Length()==0)
128 // check that the caller hasn't used the first slot
129 ipcArgs.Set(KBufferMessageSlot,TIpcArgs::ENothing);
130 __ASSERT_ALWAYS(Mem::Compare(REINTERPRET_CAST(const TUint8*, &ipcArgs), sizeof(TIpcArgs), REINTERPRET_CAST(const TUint8*, aIpcArgs), sizeof(TIpcArgs))==0,Panic(EW32PanicUsingReservedIpcSlot));
132 ipcArgs.Set(KBufferMessageSlot,&iBuf);
135 ret=iSession->DoFlush(ipcArgs);
137 ret=iSession->DoSyncMsgBuf(ipcArgs);
143 void RWsBuffer::FlushAllBuffers(TInt aBitmapHandle)
145 for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
147 if(!aBitmapHandle || buffer->iInvalidBitmapArray || (buffer->iBitmapArray.FindInOrder(aBitmapHandle)!=KErrNotFound))
153 inline void RWsBuffer::SetAndLimitMaxBufSize(TInt aMaxBufSize)
154 { // apply upper & lower limits to input buffer size
155 if (aMaxBufSize < EMinBufferSize)
157 iMaxBufSize = EMinBufferSize;
159 else if (aMaxBufSize > EMaxBufferSize)
161 iMaxBufSize = EMaxBufferSize;
165 iMaxBufSize = aMaxBufSize;
169 void RWsBuffer::SetBufferSizeL(TInt aBufSize)
171 SetAndLimitMaxBufSize(aBufSize);
172 ReAllocBufferL(iMaxBufSize);
175 void RWsBuffer::SetMaxBufferSizeL(TInt aMaxBufSize)
177 SetAndLimitMaxBufSize(aMaxBufSize);
179 if (iMaxBufSize < iBufSize)
180 { // shrink to new maximum
181 ReAllocBufferL(iMaxBufSize);
185 // initial allocation should be (at least) a quarter of the requested size
186 TInt minSize = Max( (iMaxBufSize + 3) >> 2, EMinBufferSize);
187 if (minSize > iBufSize)
188 { // new or enlarged buffer
189 ReAllocBufferL(minSize);
193 __ASSERT_DEBUG((iBufSize >= EMinBufferSize) && (iBufSize <= iMaxBufSize), Assert(EW32AssertBufferLogic));
196 // Flush() buffer, try to ReAlloc, Leave if the ReAlloc fails.
197 void RWsBuffer::ReAllocBufferL(TInt aNewSize)
199 if (aNewSize != iBufSize)
202 if (!ReAllocBuffer(aNewSize))
204 User::LeaveNoMemory();
209 TBool RWsBuffer::ReAllocBuffer(TInt aNewSize)
211 TUint8* ptr = const_cast<TUint8*>(iBuf.Ptr());
212 __ASSERT_DEBUG((iBufSize == 0) || (ptr != NULL), Assert(EW32AssertBufferLogic));
213 const TInt len = iBuf.Length();
214 TUint8* newPtr = static_cast<TUint8*>(User::ReAlloc(ptr, aNewSize));
216 { // realloc was successful
217 iBuf.Set(newPtr, len, aNewSize);
224 /* Expand the buffer, to allow new drawing command to fit.
226 Called either when trying to store additional commands to queue for Wserv, or
227 the trying to send a command bigger than the current buffer size.
229 Failure to expand the buffer is a minor problem in the first case but a big
230 problem in the second.
232 @param aRequiredSpace Size of buffer increase required.
233 @param aMsgSize If expanding the buffer fails then needs this value to know whether Flush() is good enough.
234 @return ETrue if there is enough space, EFalse if not.
236 void RWsBuffer::GrowBuffer(TInt aRequiredSpace, TInt aMsgSize)
238 __ASSERT_DEBUG(iBufSize < iMaxBufSize, Assert(EW32AssertBufferLogic));
239 // maximum size will be big enough?
240 __ASSERT_ALWAYS(aMsgSize <= iMaxBufSize, Panic(EW32PanicDataExceedsBufferLength));
242 // double or quad the current size, then limit it
243 TInt newSize = Min((iBufSize >= aRequiredSpace) ? iBufSize << 1 : iBufSize << 2, iMaxBufSize);
245 if (!ReAllocBuffer(newSize))
248 if (aMsgSize > iBufSize)
249 { // message is too big for buffer
250 Panic(EW32PanicDataExceedsBufferLength);
256 TBool RWsBuffer::SetAutoFlush(TBool aState)
261 #if defined(__AUTO_FLUSH)
271 TInt RWsBuffer::DoWrite(TInt aHandle, TUint aOpcode, TBool aFlush, const TIpcArgs* aIpcArgs, const TAny* aData, TInt aLength, const TAny* aData2, TInt aLength2)
273 __ASSERT_DEBUG(((TUint32) aOpcode) < 0x8000, Assert(EW32AssertIllegalOpcode));
274 __ASSERT_DEBUG((aLength&0x3) == 0, Assert(EW32AssertOddLengthData));
277 xtra = PadValue(aLength2); // Round data upto a multiple of 4
279 const TInt msgSize = aLength + aLength2 + xtra + static_cast<TInt>(sizeof(TWsCmdHeader));
280 TInt available = iBuf.MaxLength() - iBuf.Length();
281 if (msgSize > available)
283 if (iBufSize >= iMaxBufSize)
284 { // buffer is maximum size already
288 { // try to grow buffer
289 if ( (iBuf.Length() + msgSize) > iMaxBufSize)
290 { // growing alone will not make enough extra space
292 available = iBufSize;
295 const TInt requiredSpace = msgSize - available;
296 if (requiredSpace > 0)
298 GrowBuffer(requiredSpace, msgSize);
303 TWsCmdHeader cmdHeader;
304 cmdHeader.iBase.iOpcode = (TInt16)aOpcode;
305 cmdHeader.iBase.iCmdLength = (TInt16)(aLength + aLength2 + xtra);
307 // For performance reasons we only pass in the handle if it is different
308 // from the previous command
309 if (aHandle == iPreviousHandle)
311 iBuf.Append((TUint8 *)&cmdHeader.iBase,sizeof(cmdHeader.iBase));
315 iPreviousHandle = aHandle;
316 cmdHeader.iBase.iOpcode|=EWsOpcodeHandle;
317 cmdHeader.iDestHandle = aHandle;
318 iBuf.Append((TUint8 *)&cmdHeader,sizeof(cmdHeader));
323 iBuf.Append((TUint8 *)aData, aLength);
325 if (aLength2>0 && aData2!=NULL)
327 iBuf.Append((TUint8 *)aData2, aLength2);
328 iBuf.AppendFill(0,xtra);
331 else if (aLength2>0 && aData2==NULL)
333 iAppendDataLength = aLength2;
338 return Flush(aIpcArgs);
343 void RWsBuffer::Write(TInt handle,TUint opcode)
345 DoWrite(handle, opcode, iAutoFlush, NULL);
349 Writes data sent in aData of length aLength1 for the specifed aOpcode
350 into wserv buffer. It also takes an TIpcArgs by which you can send additional
351 data. But one thing needs to be noted that if aIpcArgs has some content then
352 this function flushes the wserv buffer.
354 @param aHandle aHandle of class derived from MWsClientClass
355 @param aOpcode Opcode for the current command
356 @param aData Data to be added to the buffer
357 @param aLength Length of the data to be added to buffer
358 @param aIpcArgs Additional data sent from client to server. It has default argument NULL.
359 And if some data is sent in aIpcArgs, it flushes wserv buffer
361 void RWsBuffer::Write(TInt aHandle, TUint aOpcode, const TAny *aData, TInt aLength, const TIpcArgs* aIpcArgs/*=NULL*/)
363 TBool flush = (aIpcArgs != NULL ? ETrue : iAutoFlush); // If aIpcArgs contains data then we do explicit flush
364 DoWrite(aHandle, aOpcode, flush, aIpcArgs, aData, aLength);
368 Writes data sent in aData and aData2 of lengths aLength1 and aLength2
369 for the specifed aOpcode into wserv buffer. It also takes an TIpcArgs by which
370 you can send additional data. But one thing needs to be noted that if aIpcArgs
371 has some content then this function flushes the wserv buffer.
373 @param aHandle Handle of class derived from MWsClientClass
374 @param aOpcode Opcode for the current command
375 @param aData Data to be added to the buffer
376 @param aLength Length of the data to be added to buffer
377 @param aData2 second Data to be added to the buffer
378 @param aLength2 Length of the second data to be added to buffer
379 @param aIpcArgs Additional data sent from client to server. It has default argument NULL.
380 And if some data is sent in aIpcArgs, it flushes wserv buffer
382 void RWsBuffer::Write(TInt aHandle, TUint aOpcode, const TAny *aData, TInt aLength, const TAny *aData2, TInt aLength2, const TIpcArgs* aIpcArgs/*=NULL*/)
384 __ASSERT_DEBUG(!((aIpcArgs != NULL) && (aLength2 > 0 && aData2 == NULL)), Assert(EW32AssertBufferLogic));
385 TBool flush = iAutoFlush;
386 if (aLength2 > 0 && aData2 == NULL)
388 flush = EFalse; // if just length2 is sent then we should not flush
390 else if (aIpcArgs != NULL)
392 flush = ETrue; // If aIpcArgs contains data then we do explicit flush
394 DoWrite(aHandle, aOpcode, flush, aIpcArgs, aData, aLength, aData2, aLength2);
398 Appends data directly to wserv buffer for the current command. So this function
399 should be used after adding the current command.
401 @param aData Data to be added to the buffer
402 @param aLength Length of the data to be added to buffer. Make sure that its length
403 is less than availabe buffer.
404 @param aFinished EFalse, adds data to buffer and disables flushing even if auto flush is on,
405 basically this notfies that more data is pending to be added.
406 ETrue, adds data to buffer and resume normal service for flushing
407 ie. Signals that this is the last bit of data to be added
409 Note: When data is added using this API, it pads out buffer to multiple of 4 bytes
411 void RWsBuffer::AppendData(const TAny *aData,TInt aLength,TBool aFinished)
413 __ASSERT_ALWAYS(iBuf.MaxLength()-iBuf.Length()>=PaddedValue(aLength),Assert(EW32AssertBufferLogic));
415 // Check if this function is called only after setting iAppendDataLength
416 __ASSERT_DEBUG(iAppendDataLength > 0, Assert(EW32AssertBufferLogic));
417 // Check if length passed in is less then iAppendDataLength
418 __ASSERT_DEBUG(iAppendDataLength >= aLength, Assert(EW32AssertBufferLogic));
420 iAppendDataLength = 0;
422 iAppendDataLength -= aLength;
424 iBuf.Append((TUint8*)(aData),aLength);
425 iBuf.AppendFill(0,PadValue(iBuf.Length())); // Padout out buffer to multiple of 4 bytes
426 if (aFinished && iAutoFlush)
430 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TIpcArgs* aIpcArgs)
432 return DoWrite(handle, opcode, ETrue, aIpcArgs);
435 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData, TInt length,const TIpcArgs* aIpcArgs)
437 return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length);
440 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData,TInt length,const TAny *pData2,TInt length2,const TIpcArgs* aIpcArgs)
442 return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length, pData2, length2);
445 TInt RWsBuffer::WriteReplyP(TInt aHandle, TUint aOpcode, const TWriteDescriptorType& aReplyBuffer)
448 aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
449 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs);
452 TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData,TInt aLength,const TWriteDescriptorType& aReplyBuffer)
455 aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
456 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
459 TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData1,TInt aLengthData1,const TAny *aData2,TInt aLengthData2,const TWriteDescriptorType& aReplyBuffer)
462 aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
463 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData1, aLengthData1, aData2, aLengthData2);
466 TInt RWsBuffer::WriteReplyWs(TUint opcode)
468 // Do a WriteReply using the sessions handle
471 return(iSession->WriteReply(opcode));
474 TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, TUint aOpcode)
476 // Do a WriteReply using the sessions handle
479 return(iSession->WriteReply(pData,aLength,aOpcode));
482 TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, const TAny *pData2, TInt aLength2, TUint aOpcode)
484 // Do a WriteReply using the sessions handle
487 return(iSession->WriteReply(pData,aLength,pData2,aLength2,aOpcode));
490 TInt RWsBuffer::WriteReplyByProvidingRemoteReadAccess(TInt aHandle,TUint aOpcode,const TAny *aData, TInt aLength,const TReadDescriptorType& aRemoteReadBuffer)
493 aRemoteReadBuffer.SetDescriptorOnIpcArgs(ipcArgs);
494 return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
497 void RWsBuffer::AddToBitmapArray(TInt aBitmapHandle)
499 if(aBitmapHandle && !iInvalidBitmapArray)
501 if(iBitmapArray.InsertInOrder(aBitmapHandle)==KErrNoMemory)
503 iInvalidBitmapArray=ETrue;
507 void RWsBuffer::SetWsGraphicManager(CWsGraphic::CManager* aManager)
509 __ASSERT_DEBUG(!WsGraphicManager(),Panic(EW32PanicGraphicInternal));
513 CWsGraphic::CManager* RWsBuffer::WsGraphicManager()
515 for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
516 if (buffer->iManager)
517 return buffer->iManager;
518 return NULL; // does not yet exist
521 void RWsBuffer::AsyncRequest(TInt aHandle, TUint aOpcode, TRequestStatus& aStatus)
523 aStatus = KRequestPending;
524 __ASSERT_DEBUG((aOpcode&EWservMessAsynchronousService)==0, Assert(EW32AssertIllegalOpcode));
527 const TInt function = EWservMessAsynchronousService | aOpcode;
528 TIpcArgs ipcArgs(aHandle);
530 iSession->SendReceive(function, ipcArgs, aStatus);
533 void RWsBuffer::EnableWindowSizeCacheL()
535 if (iWindowSizeCache == NULL)
537 iWindowSizeCache = new (ELeave) RHashMap<TInt, TWindowSizeCacheEntry>();