1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/CLIENT/RBUFFER.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,539 @@
1.4 +// Copyright (c) 1994-2010 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 + #if defined(_DEBUG)
1.50 + iAppendDataLength(0),
1.51 + #endif
1.52 + iDirectAcessCount(0), iInvalidBitmapArray(EFalse), iWindowSizeCache(NULL)
1.53 + {
1.54 + }
1.55 +
1.56 +TInt WsFbsDestroyCallBack(TAny* aBitmapHandle)
1.57 + {
1.58 + TInt* bitmapHandle=static_cast<TInt*>(aBitmapHandle);
1.59 + RWsBuffer::FlushAllBuffers(aBitmapHandle ? *bitmapHandle : 0);
1.60 +
1.61 + return(0);
1.62 + }
1.63 +
1.64 +void RWsBuffer::SetCallBack()
1.65 + {
1.66 + for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
1.67 + if (buffer==this)
1.68 + return; // Already linked
1.69 + iNext=(RWsBuffer *)Dll::Tls();
1.70 + Dll::SetTls(this);
1.71 + if (iNext==NULL) // First connection so set callback
1.72 + RFbsSession::GetSession()->SetCallBack(TCallBack(WsFbsDestroyCallBack,NULL));
1.73 + }
1.74 +
1.75 +void RWsBuffer::CancelCallBack()
1.76 + {
1.77 + RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();
1.78 + if (buffer==this)
1.79 + {
1.80 + Dll::SetTls(iNext);
1.81 + if (iNext==NULL)
1.82 + RFbsSession::GetSession()->ResetCallBack(); // Last connection closing so cancel the callback
1.83 + }
1.84 + else
1.85 + {
1.86 + RWsBuffer *prev;
1.87 + while(buffer)
1.88 + {
1.89 + prev=buffer;
1.90 + buffer=buffer->iNext;
1.91 + if (buffer==this)
1.92 + {
1.93 + prev->iNext=iNext;
1.94 + break;
1.95 + }
1.96 + }
1.97 + }
1.98 + }
1.99 +
1.100 +void RWsBuffer::Close()
1.101 + {
1.102 + User::Free((TAny *)iBuf.Ptr());
1.103 + if (iWindowSizeCache)
1.104 + {
1.105 + iWindowSizeCache->Close();
1.106 + delete iWindowSizeCache;
1.107 + iWindowSizeCache = NULL;
1.108 + }
1.109 + }
1.110 +
1.111 +void RWsBuffer::Destroy()
1.112 + {
1.113 + Flush();
1.114 + Close();
1.115 + delete this;
1.116 + }
1.117 +
1.118 +
1.119 +TInt RWsBuffer::Flush(const TIpcArgs* aIpcArgs,TBool aRequestFinish)
1.120 + {
1.121 + iBitmapArray.Reset();
1.122 + iInvalidBitmapArray=EFalse;
1.123 + if (iBuf.Length()==0)
1.124 + {
1.125 + return(KErrNone);
1.126 + }
1.127 + TIpcArgs ipcArgs;
1.128 + if (aIpcArgs!=NULL)
1.129 + {
1.130 + ipcArgs=*aIpcArgs;
1.131 + // check that the caller hasn't used the first slot
1.132 + ipcArgs.Set(KBufferMessageSlot,TIpcArgs::ENothing);
1.133 + __ASSERT_ALWAYS(Mem::Compare(REINTERPRET_CAST(const TUint8*, &ipcArgs), sizeof(TIpcArgs), REINTERPRET_CAST(const TUint8*, aIpcArgs), sizeof(TIpcArgs))==0,Panic(EW32PanicUsingReservedIpcSlot));
1.134 + }
1.135 + ipcArgs.Set(KBufferMessageSlot,&iBuf);
1.136 + TInt ret;
1.137 + if(aRequestFinish)
1.138 + ret=iSession->DoFlush(ipcArgs);
1.139 + else
1.140 + ret=iSession->DoSyncMsgBuf(ipcArgs);
1.141 + iBuf.Zero();
1.142 + iPreviousHandle=0;
1.143 + return(ret);
1.144 + }
1.145 +
1.146 +void RWsBuffer::FlushAllBuffers(TInt aBitmapHandle)
1.147 + {
1.148 + for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
1.149 + {
1.150 + if(!aBitmapHandle || buffer->iInvalidBitmapArray || (buffer->iBitmapArray.FindInOrder(aBitmapHandle)!=KErrNotFound))
1.151 +
1.152 + buffer->Flush();
1.153 + }
1.154 + }
1.155 +
1.156 +inline void RWsBuffer::SetAndLimitMaxBufSize(TInt aMaxBufSize)
1.157 + { // apply upper & lower limits to input buffer size
1.158 + if (aMaxBufSize < EMinBufferSize)
1.159 + {
1.160 + iMaxBufSize = EMinBufferSize;
1.161 + }
1.162 + else if (aMaxBufSize > EMaxBufferSize)
1.163 + {
1.164 + iMaxBufSize = EMaxBufferSize;
1.165 + }
1.166 + else
1.167 + {
1.168 + iMaxBufSize = aMaxBufSize;
1.169 + }
1.170 + }
1.171 +
1.172 +void RWsBuffer::SetBufferSizeL(TInt aBufSize)
1.173 + {
1.174 + SetAndLimitMaxBufSize(aBufSize);
1.175 + ReAllocBufferL(iMaxBufSize);
1.176 + }
1.177 +
1.178 +void RWsBuffer::SetMaxBufferSizeL(TInt aMaxBufSize)
1.179 + {
1.180 + SetAndLimitMaxBufSize(aMaxBufSize);
1.181 +
1.182 + if (iMaxBufSize < iBufSize)
1.183 + { // shrink to new maximum
1.184 + ReAllocBufferL(iMaxBufSize);
1.185 + }
1.186 + else
1.187 + {
1.188 + // initial allocation should be (at least) a quarter of the requested size
1.189 + TInt minSize = Max( (iMaxBufSize + 3) >> 2, EMinBufferSize);
1.190 + if (minSize > iBufSize)
1.191 + { // new or enlarged buffer
1.192 + ReAllocBufferL(minSize);
1.193 + }
1.194 + }
1.195 +
1.196 + __ASSERT_DEBUG((iBufSize >= EMinBufferSize) && (iBufSize <= iMaxBufSize), Assert(EW32AssertBufferLogic));
1.197 + }
1.198 +
1.199 +// Flush() buffer, try to ReAlloc, Leave if the ReAlloc fails.
1.200 +void RWsBuffer::ReAllocBufferL(TInt aNewSize)
1.201 + {
1.202 + if (aNewSize != iBufSize)
1.203 + {
1.204 + Flush();
1.205 + if (!ReAllocBuffer(aNewSize))
1.206 + {
1.207 + User::LeaveNoMemory();
1.208 + }
1.209 + }
1.210 + }
1.211 +
1.212 +TBool RWsBuffer::ReAllocBuffer(TInt aNewSize)
1.213 + {
1.214 + TUint8* ptr = const_cast<TUint8*>(iBuf.Ptr());
1.215 + __ASSERT_DEBUG((iBufSize == 0) || (ptr != NULL), Assert(EW32AssertBufferLogic));
1.216 + const TInt len = iBuf.Length();
1.217 + TUint8* newPtr = static_cast<TUint8*>(User::ReAlloc(ptr, aNewSize));
1.218 + if (newPtr != NULL)
1.219 + { // realloc was successful
1.220 + iBuf.Set(newPtr, len, aNewSize);
1.221 + iBufSize = aNewSize;
1.222 + return ETrue;
1.223 + }
1.224 + return EFalse;
1.225 + }
1.226 +
1.227 +/* Expand the buffer, to allow new drawing command to fit.
1.228 +
1.229 + Called either when trying to store additional commands to queue for Wserv, or
1.230 + the trying to send a command bigger than the current buffer size.
1.231 +
1.232 + Failure to expand the buffer is a minor problem in the first case but a big
1.233 + problem in the second.
1.234 +
1.235 + @param aRequiredSpace Size of buffer increase required.
1.236 + @param aMsgSize If expanding the buffer fails then needs this value to know whether Flush() is good enough.
1.237 + @return ETrue if there is enough space, EFalse if not.
1.238 + */
1.239 +void RWsBuffer::GrowBuffer(TInt aRequiredSpace, TInt aMsgSize)
1.240 + {
1.241 + __ASSERT_DEBUG(iBufSize < iMaxBufSize, Assert(EW32AssertBufferLogic));
1.242 + // maximum size will be big enough?
1.243 + __ASSERT_ALWAYS(aMsgSize <= iMaxBufSize, Panic(EW32PanicDataExceedsBufferLength));
1.244 +
1.245 + // double or quad the current size, then limit it
1.246 + TInt newSize = Min((iBufSize >= aRequiredSpace) ? iBufSize << 1 : iBufSize << 2, iMaxBufSize);
1.247 +
1.248 + if (!ReAllocBuffer(newSize))
1.249 + { // OOM error
1.250 + Flush();
1.251 + if (aMsgSize > iBufSize)
1.252 + { // message is too big for buffer
1.253 + Panic(EW32PanicDataExceedsBufferLength);
1.254 + }
1.255 + }
1.256 + }
1.257 +
1.258 +
1.259 +TBool RWsBuffer::SetAutoFlush(TBool aState)
1.260 + {
1.261 + TBool old;
1.262 +
1.263 + old=iAutoFlush;
1.264 +#if defined(__AUTO_FLUSH)
1.265 + if (aState)
1.266 +#else
1.267 + iAutoFlush=aState;
1.268 + if (iAutoFlush)
1.269 +#endif
1.270 + Flush();
1.271 + return(old);
1.272 + }
1.273 +
1.274 +TInt RWsBuffer::DoWrite(TInt aHandle, TUint aOpcode, TBool aFlush, const TIpcArgs* aIpcArgs, const TAny* aData, TInt aLength, const TAny* aData2, TInt aLength2)
1.275 + {
1.276 + __ASSERT_DEBUG(((TUint32) aOpcode) < 0x8000, Assert(EW32AssertIllegalOpcode));
1.277 + __ASSERT_DEBUG((aLength&0x3) == 0, Assert(EW32AssertOddLengthData));
1.278 + TInt xtra(0);
1.279 + if (aLength2>0)
1.280 + xtra = PadValue(aLength2); // Round data upto a multiple of 4
1.281 +
1.282 + const TInt msgSize = aLength + aLength2 + xtra + static_cast<TInt>(sizeof(TWsCmdHeader));
1.283 + TInt available = iBuf.MaxLength() - iBuf.Length();
1.284 + if (msgSize > available)
1.285 + {
1.286 + if (iBufSize >= iMaxBufSize)
1.287 + { // buffer is maximum size already
1.288 + Flush();
1.289 + }
1.290 + else
1.291 + { // try to grow buffer
1.292 + if ( (iBuf.Length() + msgSize) > iMaxBufSize)
1.293 + { // growing alone will not make enough extra space
1.294 + Flush();
1.295 + available = iBufSize;
1.296 + }
1.297 +
1.298 + const TInt requiredSpace = msgSize - available;
1.299 + if (requiredSpace > 0)
1.300 + {
1.301 + GrowBuffer(requiredSpace, msgSize);
1.302 + }
1.303 + }
1.304 + }
1.305 +
1.306 + TWsCmdHeader cmdHeader;
1.307 + cmdHeader.iBase.iOpcode = (TInt16)aOpcode;
1.308 + cmdHeader.iBase.iCmdLength = (TInt16)(aLength + aLength2 + xtra);
1.309 +
1.310 + // For performance reasons we only pass in the handle if it is different
1.311 + // from the previous command
1.312 + if (aHandle == iPreviousHandle)
1.313 + {
1.314 + iBuf.Append((TUint8 *)&cmdHeader.iBase,sizeof(cmdHeader.iBase));
1.315 + }
1.316 + else
1.317 + {
1.318 + iPreviousHandle = aHandle;
1.319 + cmdHeader.iBase.iOpcode|=EWsOpcodeHandle;
1.320 + cmdHeader.iDestHandle = aHandle;
1.321 + iBuf.Append((TUint8 *)&cmdHeader,sizeof(cmdHeader));
1.322 + }
1.323 +
1.324 + if (aLength)
1.325 + {
1.326 + iBuf.Append((TUint8 *)aData, aLength);
1.327 + }
1.328 + if (aLength2>0 && aData2!=NULL)
1.329 + {
1.330 + iBuf.Append((TUint8 *)aData2, aLength2);
1.331 + iBuf.AppendFill(0,xtra);
1.332 + }
1.333 +#if defined(_DEBUG)
1.334 + else if (aLength2>0 && aData2==NULL)
1.335 + {
1.336 + iAppendDataLength = aLength2;
1.337 + }
1.338 +#endif
1.339 + if (aFlush)
1.340 + {
1.341 + return Flush(aIpcArgs);
1.342 + }
1.343 + return KErrNone;
1.344 + }
1.345 +
1.346 +void RWsBuffer::Write(TInt handle,TUint opcode)
1.347 + {
1.348 + DoWrite(handle, opcode, iAutoFlush, NULL);
1.349 + }
1.350 +
1.351 +/**
1.352 +Writes data sent in aData of length aLength1 for the specifed aOpcode
1.353 +into wserv buffer. It also takes an TIpcArgs by which you can send additional
1.354 +data. But one thing needs to be noted that if aIpcArgs has some content then
1.355 +this function flushes the wserv buffer.
1.356 +
1.357 +@param aHandle aHandle of class derived from MWsClientClass
1.358 +@param aOpcode Opcode for the current command
1.359 +@param aData Data to be added to the buffer
1.360 +@param aLength Length of the data to be added to buffer
1.361 +@param aIpcArgs Additional data sent from client to server. It has default argument NULL.
1.362 + And if some data is sent in aIpcArgs, it flushes wserv buffer
1.363 +*/
1.364 +void RWsBuffer::Write(TInt aHandle, TUint aOpcode, const TAny *aData, TInt aLength, const TIpcArgs* aIpcArgs/*=NULL*/)
1.365 + {
1.366 + TBool flush = (aIpcArgs != NULL ? ETrue : iAutoFlush); // If aIpcArgs contains data then we do explicit flush
1.367 + DoWrite(aHandle, aOpcode, flush, aIpcArgs, aData, aLength);
1.368 + }
1.369 +
1.370 +/**
1.371 +Writes data sent in aData and aData2 of lengths aLength1 and aLength2
1.372 +for the specifed aOpcode into wserv buffer. It also takes an TIpcArgs by which
1.373 +you can send additional data. But one thing needs to be noted that if aIpcArgs
1.374 +has some content then this function flushes the wserv buffer.
1.375 +
1.376 +@param aHandle Handle of class derived from MWsClientClass
1.377 +@param aOpcode Opcode for the current command
1.378 +@param aData Data to be added to the buffer
1.379 +@param aLength Length of the data to be added to buffer
1.380 +@param aData2 second Data to be added to the buffer
1.381 +@param aLength2 Length of the second data to be added to buffer
1.382 +@param aIpcArgs Additional data sent from client to server. It has default argument NULL.
1.383 + And if some data is sent in aIpcArgs, it flushes wserv buffer
1.384 +*/
1.385 +void RWsBuffer::Write(TInt aHandle, TUint aOpcode, const TAny *aData, TInt aLength, const TAny *aData2, TInt aLength2, const TIpcArgs* aIpcArgs/*=NULL*/)
1.386 + {
1.387 + __ASSERT_DEBUG(!((aIpcArgs != NULL) && (aLength2 > 0 && aData2 == NULL)), Assert(EW32AssertBufferLogic));
1.388 + TBool flush = iAutoFlush;
1.389 + if (aLength2 > 0 && aData2 == NULL)
1.390 + {
1.391 + flush = EFalse; // if just length2 is sent then we should not flush
1.392 + }
1.393 + else if (aIpcArgs != NULL)
1.394 + {
1.395 + flush = ETrue; // If aIpcArgs contains data then we do explicit flush
1.396 + }
1.397 + DoWrite(aHandle, aOpcode, flush, aIpcArgs, aData, aLength, aData2, aLength2);
1.398 + }
1.399 +
1.400 +/**
1.401 +Appends data directly to wserv buffer for the current command. So this function
1.402 +should be used after adding the current command.
1.403 +
1.404 +@param aData Data to be added to the buffer
1.405 +@param aLength Length of the data to be added to buffer. Make sure that its length
1.406 + is less than availabe buffer.
1.407 +@param aFinished EFalse, adds data to buffer and disables flushing even if auto flush is on,
1.408 + basically this notfies that more data is pending to be added.
1.409 + ETrue, adds data to buffer and resume normal service for flushing
1.410 + ie. Signals that this is the last bit of data to be added
1.411 +
1.412 +Note: When data is added using this API, it pads out buffer to multiple of 4 bytes
1.413 +*/
1.414 +void RWsBuffer::AppendData(const TAny *aData,TInt aLength,TBool aFinished)
1.415 + {
1.416 + __ASSERT_ALWAYS(iBuf.MaxLength()-iBuf.Length()>=PaddedValue(aLength),Assert(EW32AssertBufferLogic));
1.417 +#if defined(_DEBUG)
1.418 + // Check if this function is called only after setting iAppendDataLength
1.419 + __ASSERT_DEBUG(iAppendDataLength > 0, Assert(EW32AssertBufferLogic));
1.420 + // Check if length passed in is less then iAppendDataLength
1.421 + __ASSERT_DEBUG(iAppendDataLength >= aLength, Assert(EW32AssertBufferLogic));
1.422 + if (aFinished)
1.423 + iAppendDataLength = 0;
1.424 + else
1.425 + iAppendDataLength -= aLength;
1.426 +#endif
1.427 + iBuf.Append((TUint8*)(aData),aLength);
1.428 + iBuf.AppendFill(0,PadValue(iBuf.Length())); // Padout out buffer to multiple of 4 bytes
1.429 + if (aFinished && iAutoFlush)
1.430 + Flush(NULL);
1.431 + }
1.432 +
1.433 +TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TIpcArgs* aIpcArgs)
1.434 + {
1.435 + return DoWrite(handle, opcode, ETrue, aIpcArgs);
1.436 + }
1.437 +
1.438 +TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData, TInt length,const TIpcArgs* aIpcArgs)
1.439 + {
1.440 + return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length);
1.441 + }
1.442 +
1.443 +TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData,TInt length,const TAny *pData2,TInt length2,const TIpcArgs* aIpcArgs)
1.444 + {
1.445 + return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length, pData2, length2);
1.446 + }
1.447 +
1.448 +TInt RWsBuffer::WriteReplyP(TInt aHandle, TUint aOpcode, const TWriteDescriptorType& aReplyBuffer)
1.449 + {
1.450 + TIpcArgs ipcArgs;
1.451 + aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.452 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs);
1.453 + }
1.454 +
1.455 +TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData,TInt aLength,const TWriteDescriptorType& aReplyBuffer)
1.456 + {
1.457 + TIpcArgs ipcArgs;
1.458 + aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.459 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
1.460 + }
1.461 +
1.462 +TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData1,TInt aLengthData1,const TAny *aData2,TInt aLengthData2,const TWriteDescriptorType& aReplyBuffer)
1.463 + {
1.464 + TIpcArgs ipcArgs;
1.465 + aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.466 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData1, aLengthData1, aData2, aLengthData2);
1.467 + }
1.468 +
1.469 +TInt RWsBuffer::WriteReplyWs(TUint opcode)
1.470 +//
1.471 +// Do a WriteReply using the sessions handle
1.472 +//
1.473 + {
1.474 + return(iSession->WriteReply(opcode));
1.475 + }
1.476 +
1.477 +TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, TUint aOpcode)
1.478 +//
1.479 +// Do a WriteReply using the sessions handle
1.480 +//
1.481 + {
1.482 + return(iSession->WriteReply(pData,aLength,aOpcode));
1.483 + }
1.484 +
1.485 +TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, const TAny *pData2, TInt aLength2, TUint aOpcode)
1.486 +//
1.487 +// Do a WriteReply using the sessions handle
1.488 +//
1.489 + {
1.490 + return(iSession->WriteReply(pData,aLength,pData2,aLength2,aOpcode));
1.491 + }
1.492 +
1.493 +TInt RWsBuffer::WriteReplyByProvidingRemoteReadAccess(TInt aHandle,TUint aOpcode,const TAny *aData, TInt aLength,const TReadDescriptorType& aRemoteReadBuffer)
1.494 + {
1.495 + TIpcArgs ipcArgs;
1.496 + aRemoteReadBuffer.SetDescriptorOnIpcArgs(ipcArgs);
1.497 + return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
1.498 + }
1.499 +
1.500 +void RWsBuffer::AddToBitmapArray(TInt aBitmapHandle)
1.501 + {
1.502 + if(aBitmapHandle && !iInvalidBitmapArray)
1.503 + {
1.504 + if(iBitmapArray.InsertInOrder(aBitmapHandle)==KErrNoMemory)
1.505 +
1.506 + iInvalidBitmapArray=ETrue;
1.507 + }
1.508 + }
1.509 +
1.510 +void RWsBuffer::SetWsGraphicManager(CWsGraphic::CManager* aManager)
1.511 + {
1.512 + __ASSERT_DEBUG(!WsGraphicManager(),Panic(EW32PanicGraphicInternal));
1.513 + iManager = aManager;
1.514 + }
1.515 +
1.516 +CWsGraphic::CManager* RWsBuffer::WsGraphicManager()
1.517 + {
1.518 + for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
1.519 + if (buffer->iManager)
1.520 + return buffer->iManager;
1.521 + return NULL; // does not yet exist
1.522 + }
1.523 +
1.524 +void RWsBuffer::AsyncRequest(TInt aHandle, TUint aOpcode, TRequestStatus& aStatus)
1.525 + {
1.526 + aStatus = KRequestPending;
1.527 + __ASSERT_DEBUG((aOpcode&EWservMessAsynchronousService)==0, Assert(EW32AssertIllegalOpcode));
1.528 + Flush();
1.529 +
1.530 + const TInt function = EWservMessAsynchronousService | aOpcode;
1.531 + TIpcArgs ipcArgs(aHandle);
1.532 +
1.533 + iSession->SendReceive(function, ipcArgs, aStatus);
1.534 + }
1.535 +
1.536 +void RWsBuffer::EnableWindowSizeCacheL()
1.537 + {
1.538 + if (iWindowSizeCache == NULL)
1.539 + {
1.540 + iWindowSizeCache = new (ELeave) RHashMap<TInt, TWindowSizeCacheEntry>();
1.541 + }
1.542 + }