1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/CLIENT/RBUFFER.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,439 @@
1.4 +// Copyright (c) 1994-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +// RWsBuffer class, handles buffering and flushing of window server commands
1.18 +//
1.19 +//
1.20 +
1.21 +#include <e32std.h>
1.22 +#include "../SERVER/w32cmd.h"
1.23 +#include "CLIENT.H"
1.24 +#include "w32comm.h"
1.25 +
1.26 +// Global functions
1.27 +
1.28 +GLDEF_C void Assert(TW32Assert aAssert)
1.29 + {
1.30 + _LIT(KW32AssertCategory,"W32 Assert");
1.31 + User::Panic(KW32AssertCategory,aAssert);
1.32 + }
1.33 +
1.34 +GLDEF_C void Panic(TW32Panic aPanic)
1.35 + {
1.36 + _LIT(KW32PanicCategory,"W32");
1.37 + User::Panic(KW32PanicCategory,aPanic);
1.38 + }
1.39 +
1.40 +// Public functions
1.41 +
1.42 +RWsBuffer::RWsBuffer(RWsSession *aSession) : iSession(aSession), iManager(NULL),
1.43 + #if defined(__AUTO_FLUSH)
1.44 + iAutoFlush(ETrue),
1.45 + #else
1.46 + iAutoFlush(EFalse),
1.47 + #endif
1.48 + iBuf(NULL,0,0), iNext(NULL), iPreviousHandle(0), iBufSize(0), iMaxBufSize(EMinBufferSize),
1.49 + iDirectAcessCount(0), iInvalidBitmapArray(EFalse)
1.50 + {
1.51 + }
1.52 +
1.53 +TInt WsFbsDestroyCallBack(TAny* aBitmapHandle)
1.54 + {
1.55 + TInt* bitmapHandle=static_cast<TInt*>(aBitmapHandle);
1.56 + RWsBuffer::FlushAllBuffers(aBitmapHandle ? *bitmapHandle : 0);
1.57 +
1.58 + return(0);
1.59 + }
1.60 +
1.61 +void RWsBuffer::SetCallBack()
1.62 + {
1.63 + for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
1.64 + if (buffer==this)
1.65 + return; // Already linked
1.66 + iNext=(RWsBuffer *)Dll::Tls();
1.67 + Dll::SetTls(this);
1.68 + if (iNext==NULL) // First connection so set callback
1.69 + RFbsSession::GetSession()->SetCallBack(TCallBack(WsFbsDestroyCallBack,NULL));
1.70 + }
1.71 +
1.72 +void RWsBuffer::CancelCallBack()
1.73 + {
1.74 + RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();
1.75 + if (buffer==this)
1.76 + {
1.77 + Dll::SetTls(iNext);
1.78 + if (iNext==NULL)
1.79 + RFbsSession::GetSession()->ResetCallBack(); // Last connection closing so cancel the callback
1.80 + }
1.81 + else
1.82 + {
1.83 + RWsBuffer *prev;
1.84 + while(buffer)
1.85 + {
1.86 + prev=buffer;
1.87 + buffer=buffer->iNext;
1.88 + if (buffer==this)
1.89 + {
1.90 + prev->iNext=iNext;
1.91 + break;
1.92 + }
1.93 + }
1.94 + }
1.95 + }
1.96 +
1.97 +void RWsBuffer::Close()
1.98 + {
1.99 + User::Free((TAny *)iBuf.Ptr());
1.100 + }
1.101 +
1.102 +void RWsBuffer::Destroy()
1.103 + {
1.104 + Flush();
1.105 + Close();
1.106 + delete this;
1.107 + }
1.108 +
1.109 +
1.110 +TInt RWsBuffer::Flush(const TIpcArgs* aIpcArgs,TBool aRequestFinish)
1.111 + {
1.112 + iBitmapArray.Reset();
1.113 + iInvalidBitmapArray=EFalse;
1.114 + if (iBuf.Length()==0)
1.115 + {
1.116 + return(KErrNone);
1.117 + }
1.118 + TIpcArgs ipcArgs;
1.119 + if (aIpcArgs!=NULL)
1.120 + {
1.121 + ipcArgs=*aIpcArgs;
1.122 + // check that the caller hasn't used the first slot
1.123 + ipcArgs.Set(KBufferMessageSlot,TIpcArgs::ENothing);
1.124 + __ASSERT_ALWAYS(Mem::Compare(REINTERPRET_CAST(const TUint8*, &ipcArgs), sizeof(TIpcArgs), REINTERPRET_CAST(const TUint8*, aIpcArgs), sizeof(TIpcArgs))==0,Panic(EW32PanicUsingReservedIpcSlot));
1.125 + }
1.126 + ipcArgs.Set(KBufferMessageSlot,&iBuf);
1.127 + TInt ret;
1.128 + if(aRequestFinish)
1.129 + ret=iSession->DoFlush(ipcArgs);
1.130 + else
1.131 + ret=iSession->DoSyncMsgBuf(ipcArgs);
1.132 + iBuf.Zero();
1.133 + iPreviousHandle=0;
1.134 + return(ret);
1.135 + }
1.136 +
1.137 +void RWsBuffer::FlushAllBuffers(TInt aBitmapHandle)
1.138 + {
1.139 + for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
1.140 + {
1.141 + if(!aBitmapHandle || buffer->iInvalidBitmapArray || (buffer->iBitmapArray.FindInOrder(aBitmapHandle)!=KErrNotFound))
1.142 +
1.143 + buffer->Flush();
1.144 + }
1.145 + }
1.146 +
1.147 +inline void RWsBuffer::SetAndLimitMaxBufSize(TInt aMaxBufSize)
1.148 + { // apply upper & lower limits to input buffer size
1.149 + if (aMaxBufSize < EMinBufferSize)
1.150 + {
1.151 + iMaxBufSize = EMinBufferSize;
1.152 + }
1.153 + else if (aMaxBufSize > EMaxBufferSize)
1.154 + {
1.155 + iMaxBufSize = EMaxBufferSize;
1.156 + }
1.157 + else
1.158 + {
1.159 + iMaxBufSize = aMaxBufSize;
1.160 + }
1.161 + }
1.162 +
1.163 +void RWsBuffer::SetBufferSizeL(TInt aBufSize)
1.164 + {
1.165 + SetAndLimitMaxBufSize(aBufSize);
1.166 + ReAllocBufferL(iMaxBufSize);
1.167 + }
1.168 +
1.169 +void RWsBuffer::SetMaxBufferSizeL(TInt aMaxBufSize)
1.170 + {
1.171 + SetAndLimitMaxBufSize(aMaxBufSize);
1.172 +
1.173 + if (iMaxBufSize < iBufSize)
1.174 + { // shrink to new maximum
1.175 + ReAllocBufferL(iMaxBufSize);
1.176 + }
1.177 + else
1.178 + {
1.179 + // initial allocation should be (at least) a quarter of the requested size
1.180 + TInt minSize = Max( (iMaxBufSize + 3) >> 2, EMinBufferSize);
1.181 + if (minSize > iBufSize)
1.182 + { // new or enlarged buffer
1.183 + ReAllocBufferL(minSize);
1.184 + }
1.185 + }
1.186 +
1.187 + __ASSERT_DEBUG((iBufSize >= EMinBufferSize) && (iBufSize <= iMaxBufSize), Assert(EW32AssertBufferLogic));
1.188 + }
1.189 +
1.190 +// Flush() buffer, try to ReAlloc, Leave if the ReAlloc fails.
1.191 +void RWsBuffer::ReAllocBufferL(TInt aNewSize)
1.192 + {
1.193 + if (aNewSize != iBufSize)
1.194 + {
1.195 + Flush();
1.196 + if (!ReAllocBuffer(aNewSize))
1.197 + {
1.198 + User::LeaveNoMemory();
1.199 + }
1.200 + }
1.201 + }
1.202 +
1.203 +TBool RWsBuffer::ReAllocBuffer(TInt aNewSize)
1.204 + {
1.205 + TUint8* ptr = const_cast<TUint8*>(iBuf.Ptr());
1.206 + __ASSERT_DEBUG((iBufSize == 0) || (ptr != NULL), Assert(EW32AssertBufferLogic));
1.207 + const TInt len = iBuf.Length();
1.208 + TUint8* newPtr = static_cast<TUint8*>(User::ReAlloc(ptr, aNewSize));
1.209 + if (newPtr != NULL)
1.210 + { // realloc was successful
1.211 + iBuf.Set(newPtr, len, aNewSize);
1.212 + iBufSize = aNewSize;
1.213 + return ETrue;
1.214 + }
1.215 + return EFalse;
1.216 + }
1.217 +
1.218 +/* Expand the buffer, to allow new drawing command to fit.
1.219 +
1.220 + Called either when trying to store additional commands to queue for Wserv, or
1.221 + the trying to send a command bigger than the current buffer size.
1.222 +
1.223 + Failure to expand the buffer is a minor problem in the first case but a big
1.224 + problem in the second.
1.225 +
1.226 + @param aRequiredSpace Size of buffer increase required.
1.227 + @param aMsgSize If expanding the buffer fails then needs this value to know whether Flush() is good enough.
1.228 + @return ETrue if there is enough space, EFalse if not.
1.229 + */
1.230 +void RWsBuffer::GrowBuffer(TInt aRequiredSpace, TInt aMsgSize)
1.231 + {
1.232 + __ASSERT_DEBUG(iBufSize < iMaxBufSize, Assert(EW32AssertBufferLogic));
1.233 + // maximum size will be big enough?
1.234 + __ASSERT_ALWAYS(aMsgSize <= iMaxBufSize, Panic(EW32PanicDataExceedsBufferLength));
1.235 +
1.236 + // double or quad the current size, then limit it
1.237 + TInt newSize = Min((iBufSize >= aRequiredSpace) ? iBufSize << 1 : iBufSize << 2, iMaxBufSize);
1.238 +
1.239 + if (!ReAllocBuffer(newSize))
1.240 + { // OOM error
1.241 + Flush();
1.242 + if (aMsgSize > iBufSize)
1.243 + { // message is too big for buffer
1.244 + Panic(EW32PanicDataExceedsBufferLength);
1.245 + }
1.246 + }
1.247 + }
1.248 +
1.249 +
1.250 +TBool RWsBuffer::SetAutoFlush(TBool aState)
1.251 + {
1.252 + TBool old;
1.253 +
1.254 + old=iAutoFlush;
1.255 +#if defined(__AUTO_FLUSH)
1.256 + if (aState)
1.257 +#else
1.258 + iAutoFlush=aState;
1.259 + if (iAutoFlush)
1.260 +#endif
1.261 + Flush();
1.262 + return(old);
1.263 + }
1.264 +
1.265 +TInt RWsBuffer::DoWrite(TInt aHandle, TUint aOpcode, TBool aFlush, const TIpcArgs* aIpcArgs, const TAny* aData, TInt aLength, const TAny* aData2, TInt aLength2)
1.266 + {
1.267 + __ASSERT_DEBUG(((TUint32) aOpcode) < 0x8000, Assert(EW32AssertIllegalOpcode));
1.268 + __ASSERT_DEBUG((aLength&0x3) == 0, Assert(EW32AssertOddLengthData));
1.269 + TInt xtra(0);
1.270 + if (aLength2 > 0)
1.271 + {
1.272 + xtra = 4 - (aLength2&0x3); // Round data upto a multiple of 4
1.273 + if (xtra==4)
1.274 + {
1.275 + xtra=0;
1.276 + }
1.277 + }
1.278 +
1.279 + const TInt msgSize = aLength + aLength2 + xtra + static_cast<TInt>(sizeof(TWsCmdHeader));
1.280 + TInt available = iBuf.MaxLength() - iBuf.Length();
1.281 + if (msgSize > available)
1.282 + {
1.283 + if (iBufSize >= iMaxBufSize)
1.284 + { // buffer is maximum size already
1.285 + Flush();
1.286 + }
1.287 + else
1.288 + { // try to grow buffer
1.289 + if ( (iBuf.Length() + msgSize) > iMaxBufSize)
1.290 + { // growing alone will not make enough extra space
1.291 + Flush();
1.292 + available = iBufSize;
1.293 + }
1.294 +
1.295 + const TInt requiredSpace = msgSize - available;
1.296 + if (requiredSpace > 0)
1.297 + {
1.298 + GrowBuffer(requiredSpace, msgSize);
1.299 + }
1.300 + }
1.301 + }
1.302 +
1.303 + TWsCmdHeader cmdHeader;
1.304 + cmdHeader.iBase.iOpcode = (TInt16)aOpcode;
1.305 + cmdHeader.iBase.iCmdLength = (TInt16)(aLength + aLength2 + xtra);
1.306 +
1.307 + // For performance reasons we only pass in the handle if it is different
1.308 + // from the previous command
1.309 + if (aHandle == iPreviousHandle)
1.310 + {
1.311 + iBuf.Append((TUint8 *)&cmdHeader.iBase,sizeof(cmdHeader.iBase));
1.312 + }
1.313 + else
1.314 + {
1.315 + iPreviousHandle = aHandle;
1.316 + cmdHeader.iBase.iOpcode|=EWsOpcodeHandle;
1.317 + cmdHeader.iDestHandle = aHandle;
1.318 + iBuf.Append((TUint8 *)&cmdHeader,sizeof(cmdHeader));
1.319 + }
1.320 +
1.321 + if (aLength)
1.322 + {
1.323 + iBuf.Append((TUint8 *)aData, aLength);
1.324 + }
1.325 + if (aLength2 > 0)
1.326 + {
1.327 + iBuf.Append((TUint8 *)aData2, aLength2);
1.328 + iBuf.AppendFill(0,xtra);
1.329 + }
1.330 +
1.331 + if (aFlush)
1.332 + {
1.333 + return Flush(aIpcArgs);
1.334 + }
1.335 + return KErrNone;
1.336 + }
1.337 +
1.338 +void RWsBuffer::Write(TInt handle,TUint opcode)
1.339 + {
1.340 + DoWrite(handle, opcode, iAutoFlush, NULL);
1.341 + }
1.342 +
1.343 +void RWsBuffer::Write(TInt handle,TUint opcode,const TAny *pData, TInt length)
1.344 + {
1.345 + DoWrite(handle, opcode, iAutoFlush, NULL, pData, length);
1.346 + }
1.347 +
1.348 +void RWsBuffer::Write(TInt handle,TUint opcode,const TAny *pData, TInt length,const TAny *pData2, TInt length2)
1.349 + {
1.350 + DoWrite(handle, opcode, iAutoFlush, NULL, pData, length, pData2, length2);
1.351 + }
1.352 +
1.353 +TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TIpcArgs* aIpcArgs)
1.354 + {
1.355 + return DoWrite(handle, opcode, ETrue, aIpcArgs);
1.356 + }
1.357 +
1.358 +TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData, TInt length,const TIpcArgs* aIpcArgs)
1.359 + {
1.360 + return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length);
1.361 + }
1.362 +
1.363 +TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData,TInt length,const TAny *pData2,TInt length2,const TIpcArgs* aIpcArgs)
1.364 + {
1.365 + return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length, pData2, length2);
1.366 + }
1.367 +
1.368 +TInt RWsBuffer::WriteReplyP(TInt aHandle, TUint aOpcode, const TWriteDescriptorType& aReplyBuffer)
1.369 + {
1.370 + TIpcArgs ipcArgs;
1.371 + aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.372 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs);
1.373 + }
1.374 +
1.375 +TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData,TInt aLength,const TWriteDescriptorType& aReplyBuffer)
1.376 + {
1.377 + TIpcArgs ipcArgs;
1.378 + aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.379 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
1.380 + }
1.381 +
1.382 +TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData1,TInt aLengthData1,const TAny *aData2,TInt aLengthData2,const TWriteDescriptorType& aReplyBuffer)
1.383 + {
1.384 + TIpcArgs ipcArgs;
1.385 + aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.386 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData1, aLengthData1, aData2, aLengthData2);
1.387 + }
1.388 +
1.389 +TInt RWsBuffer::WriteReplyWs(TUint opcode)
1.390 +//
1.391 +// Do a WriteReply using the sessions handle
1.392 +//
1.393 + {
1.394 + return(iSession->WriteReply(opcode));
1.395 + }
1.396 +
1.397 +TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, TUint aOpcode)
1.398 +//
1.399 +// Do a WriteReply using the sessions handle
1.400 +//
1.401 + {
1.402 + return(iSession->WriteReply(pData,aLength,aOpcode));
1.403 + }
1.404 +
1.405 +TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, const TAny *pData2, TInt aLength2, TUint aOpcode)
1.406 +//
1.407 +// Do a WriteReply using the sessions handle
1.408 +//
1.409 + {
1.410 + return(iSession->WriteReply(pData,aLength,pData2,aLength2,aOpcode));
1.411 + }
1.412 +
1.413 +TInt RWsBuffer::WriteReplyByProvidingRemoteReadAccess(TInt aHandle,TUint aOpcode,const TAny *aData, TInt aLength,const TReadDescriptorType& aRemoteReadBuffer)
1.414 + {
1.415 + TIpcArgs ipcArgs;
1.416 + aRemoteReadBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.417 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
1.418 + }
1.419 +
1.420 +void RWsBuffer::AddToBitmapArray(TInt aBitmapHandle)
1.421 + {
1.422 + if(aBitmapHandle && !iInvalidBitmapArray)
1.423 + {
1.424 + if(iBitmapArray.InsertInOrder(aBitmapHandle)==KErrNoMemory)
1.425 +
1.426 + iInvalidBitmapArray=ETrue;
1.427 + }
1.428 + }
1.429 +
1.430 +void RWsBuffer::SetWsGraphicManager(CWsGraphic::CManager* aManager)
1.431 + {
1.432 + __ASSERT_DEBUG(!WsGraphicManager(),Panic(EW32PanicGraphicInternal));
1.433 + iManager = aManager;
1.434 + }
1.435 +
1.436 +CWsGraphic::CManager* RWsBuffer::WsGraphicManager()
1.437 + {
1.438 + for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
1.439 + if (buffer->iManager)
1.440 + return buffer->iManager;
1.441 + return NULL; // does not yet exist
1.442 + }