os/graphics/windowing/windowserver/nga/CLIENT/RBUFFER.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // RWsBuffer class, handles buffering and flushing of window server commands
    15 // 
    16 //
    17 
    18 #include <e32std.h>
    19 #include "../SERVER/w32cmd.h"
    20 #include "CLIENT.H"
    21 #include "w32comm.h"
    22 
    23 // Global functions
    24 
    25 GLDEF_C void Assert(TW32Assert aAssert)
    26 	{
    27 	_LIT(KW32AssertCategory,"W32 Assert");
    28 	User::Panic(KW32AssertCategory,aAssert);
    29 	}
    30 
    31 GLDEF_C void Panic(TW32Panic aPanic)
    32 	{
    33 	_LIT(KW32PanicCategory,"W32");
    34 	User::Panic(KW32PanicCategory,aPanic);
    35 	}
    36 
    37 // Public functions
    38 
    39 RWsBuffer::RWsBuffer(RWsSession *aSession) : iSession(aSession), iManager(NULL),
    40 	#if defined(__AUTO_FLUSH)
    41 		iAutoFlush(ETrue),
    42 	#else
    43 		iAutoFlush(EFalse),
    44 	#endif
    45 	iBuf(NULL,0,0), iNext(NULL), iPreviousHandle(0), iBufSize(0), iMaxBufSize(EMinBufferSize),
    46 	#if defined(_DEBUG)
    47 	iAppendDataLength(0),
    48 	#endif
    49 	iDirectAcessCount(0), iInvalidBitmapArray(EFalse), iWindowSizeCache(NULL)
    50 	{
    51 	}
    52 
    53 TInt WsFbsDestroyCallBack(TAny* aBitmapHandle)
    54 	{
    55 	TInt* bitmapHandle=static_cast<TInt*>(aBitmapHandle);
    56 	RWsBuffer::FlushAllBuffers(aBitmapHandle ? *bitmapHandle : 0);
    57 
    58 	return(0);
    59 	}
    60 
    61 void RWsBuffer::SetCallBack()
    62 	{
    63 	for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
    64 		if (buffer==this)
    65 			return;	// Already linked
    66 	iNext=(RWsBuffer *)Dll::Tls();
    67 	Dll::SetTls(this);
    68 	if (iNext==NULL)	// First connection so set callback
    69 		RFbsSession::GetSession()->SetCallBack(TCallBack(WsFbsDestroyCallBack,NULL));
    70 	}
    71 
    72 void RWsBuffer::CancelCallBack()
    73 	{
    74 	RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();
    75 	if (buffer==this)
    76 		{
    77 		Dll::SetTls(iNext);
    78 		if (iNext==NULL)
    79 			RFbsSession::GetSession()->ResetCallBack();	// Last connection closing so cancel the callback
    80 		}
    81 	else
    82 		{
    83 		RWsBuffer *prev;
    84 		while(buffer)
    85 			{
    86 			prev=buffer;
    87 			buffer=buffer->iNext;
    88 			if (buffer==this)
    89 				{
    90 				prev->iNext=iNext;
    91 				break;
    92 				}
    93 			}
    94 		}
    95 	}
    96 
    97 void RWsBuffer::Close()
    98 	{
    99 	User::Free((TAny *)iBuf.Ptr());
   100 	if (iWindowSizeCache)
   101 	    {
   102 	    iWindowSizeCache->Close();
   103 	    delete iWindowSizeCache;
   104 	    iWindowSizeCache = NULL;
   105 	    }
   106 	}
   107 
   108 void RWsBuffer::Destroy()
   109 	{
   110 	Flush();
   111 	Close();
   112 	delete this;
   113 	}
   114 
   115 
   116 TInt RWsBuffer::Flush(const TIpcArgs* aIpcArgs,TBool aRequestFinish)
   117 	{
   118 	iBitmapArray.Reset();
   119 	iInvalidBitmapArray=EFalse;
   120 	if (iBuf.Length()==0)
   121 		{
   122 		return(KErrNone);
   123 		}
   124 	TIpcArgs ipcArgs;
   125 	if (aIpcArgs!=NULL)
   126 		{
   127 		ipcArgs=*aIpcArgs;
   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));
   131 		}
   132 	ipcArgs.Set(KBufferMessageSlot,&iBuf);
   133 	TInt ret;
   134 	if(aRequestFinish)
   135 		ret=iSession->DoFlush(ipcArgs);
   136 	else
   137 		ret=iSession->DoSyncMsgBuf(ipcArgs);
   138 	iBuf.Zero();
   139 	iPreviousHandle=0;
   140 	return(ret);
   141 	}
   142 
   143 void RWsBuffer::FlushAllBuffers(TInt aBitmapHandle)
   144 	{
   145 	for(RWsBuffer *buffer=(RWsBuffer *)Dll::Tls();buffer;buffer=buffer->iNext)
   146 		{
   147 		if(!aBitmapHandle || buffer->iInvalidBitmapArray || (buffer->iBitmapArray.FindInOrder(aBitmapHandle)!=KErrNotFound))
   148 
   149 			buffer->Flush();
   150 		}
   151 	}
   152 
   153 inline void RWsBuffer::SetAndLimitMaxBufSize(TInt aMaxBufSize)
   154 	{ // apply upper & lower limits to input buffer size
   155 	if (aMaxBufSize < EMinBufferSize)
   156 		{
   157 		iMaxBufSize = EMinBufferSize;
   158 		}
   159 	else if	(aMaxBufSize > EMaxBufferSize)
   160 		{
   161 		iMaxBufSize = EMaxBufferSize;
   162 		}
   163 	else
   164 		{
   165 		iMaxBufSize = aMaxBufSize;
   166 		}
   167 	}
   168 
   169 void RWsBuffer::SetBufferSizeL(TInt aBufSize)
   170 	{
   171 	SetAndLimitMaxBufSize(aBufSize);
   172 	ReAllocBufferL(iMaxBufSize);
   173 	}
   174 
   175 void RWsBuffer::SetMaxBufferSizeL(TInt aMaxBufSize)
   176 	{
   177 	SetAndLimitMaxBufSize(aMaxBufSize);
   178 
   179 	if (iMaxBufSize < iBufSize)
   180 		{ // shrink to new maximum
   181 		ReAllocBufferL(iMaxBufSize);
   182 		}
   183 	else
   184 		{
   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);
   190 			}
   191 		}
   192 
   193 	__ASSERT_DEBUG((iBufSize >= EMinBufferSize) && (iBufSize <= iMaxBufSize), Assert(EW32AssertBufferLogic));
   194 	}
   195 
   196 // Flush() buffer, try to ReAlloc, Leave if the ReAlloc fails.
   197 void RWsBuffer::ReAllocBufferL(TInt aNewSize)
   198 	{
   199 	if (aNewSize != iBufSize)
   200 		{
   201 		Flush();
   202 		if (!ReAllocBuffer(aNewSize))
   203 			{
   204 			User::LeaveNoMemory();
   205 			}
   206 		}
   207 	}
   208 
   209 TBool RWsBuffer::ReAllocBuffer(TInt aNewSize)
   210 	{
   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));
   215 	if (newPtr != NULL)
   216 		{ // realloc was successful
   217 		iBuf.Set(newPtr, len, aNewSize);
   218 		iBufSize = aNewSize;
   219 		return ETrue;
   220 		}
   221 	return EFalse;
   222 	}
   223 
   224 /* Expand the buffer, to allow new drawing command to fit.
   225  
   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.
   228 
   229  Failure to expand the buffer is a minor problem in the first case but a big 
   230  problem in the second.
   231 
   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.
   235  */
   236 void RWsBuffer::GrowBuffer(TInt aRequiredSpace, TInt aMsgSize)
   237 	{
   238 	__ASSERT_DEBUG(iBufSize < iMaxBufSize, Assert(EW32AssertBufferLogic));
   239 	// maximum size will be big enough?
   240 	__ASSERT_ALWAYS(aMsgSize <= iMaxBufSize, Panic(EW32PanicDataExceedsBufferLength));
   241 	
   242 	 // double or quad the current size, then limit it
   243 	TInt newSize = Min((iBufSize >= aRequiredSpace) ? iBufSize << 1 : iBufSize << 2, iMaxBufSize);
   244 
   245 	if (!ReAllocBuffer(newSize))
   246 		{ // OOM error
   247 		Flush();
   248 		if (aMsgSize > iBufSize)
   249 			{ // message is too big for buffer
   250 			Panic(EW32PanicDataExceedsBufferLength);
   251 			}
   252 		}
   253 	}
   254 
   255 
   256 TBool RWsBuffer::SetAutoFlush(TBool aState)
   257 	{
   258 	TBool old;
   259 
   260 	old=iAutoFlush;
   261 #if defined(__AUTO_FLUSH)
   262 	if (aState)
   263 #else
   264 	iAutoFlush=aState;
   265 	if (iAutoFlush)
   266 #endif
   267 		Flush();
   268 	return(old);
   269 	}
   270 
   271 TInt RWsBuffer::DoWrite(TInt aHandle, TUint aOpcode, TBool aFlush, const TIpcArgs* aIpcArgs, const TAny* aData, TInt aLength, const TAny* aData2, TInt aLength2)
   272 	{
   273 	__ASSERT_DEBUG(((TUint32) aOpcode) < 0x8000, Assert(EW32AssertIllegalOpcode));
   274 	__ASSERT_DEBUG((aLength&0x3) == 0, Assert(EW32AssertOddLengthData));
   275 	TInt xtra(0);
   276 	if (aLength2>0)
   277 		xtra = PadValue(aLength2);		// Round data upto a multiple of 4
   278 
   279 	const TInt msgSize = aLength + aLength2 + xtra + static_cast<TInt>(sizeof(TWsCmdHeader));
   280 	TInt available = iBuf.MaxLength() - iBuf.Length();
   281 	if (msgSize > available)
   282 		{
   283 		if (iBufSize >= iMaxBufSize)
   284 			{ // buffer is maximum size already
   285 			Flush();
   286 			}
   287 		else
   288 			{ // try to grow buffer
   289 			 if ( (iBuf.Length() + msgSize) > iMaxBufSize)
   290 				{ // growing alone will not make enough extra space
   291 				Flush();
   292 				available = iBufSize;
   293 				}
   294 
   295 			const TInt requiredSpace = msgSize - available;
   296 			if (requiredSpace > 0)
   297 				{
   298 				GrowBuffer(requiredSpace, msgSize);
   299 				}
   300 			}
   301 		}
   302 
   303 	TWsCmdHeader cmdHeader;
   304 	cmdHeader.iBase.iOpcode = (TInt16)aOpcode;
   305 	cmdHeader.iBase.iCmdLength = (TInt16)(aLength + aLength2 + xtra);
   306 
   307 	// For performance reasons we only pass in the handle if it is different
   308 	// from the previous command
   309 	if (aHandle == iPreviousHandle)
   310 		{
   311 		iBuf.Append((TUint8 *)&cmdHeader.iBase,sizeof(cmdHeader.iBase));
   312 		}
   313 	else
   314 		{
   315 		iPreviousHandle = aHandle;
   316 		cmdHeader.iBase.iOpcode|=EWsOpcodeHandle;
   317 		cmdHeader.iDestHandle = aHandle;
   318 		iBuf.Append((TUint8 *)&cmdHeader,sizeof(cmdHeader));
   319 		}
   320 
   321 	if (aLength)
   322 		{
   323 		iBuf.Append((TUint8 *)aData, aLength);
   324 		}
   325 	if (aLength2>0 && aData2!=NULL)
   326 		{
   327 		iBuf.Append((TUint8 *)aData2, aLength2);
   328 		iBuf.AppendFill(0,xtra);
   329 		}
   330 #if defined(_DEBUG)
   331 	else if (aLength2>0 && aData2==NULL)
   332 		{
   333 		iAppendDataLength = aLength2;
   334 		}
   335 #endif
   336 	if (aFlush)
   337 		{
   338 		return Flush(aIpcArgs);
   339 		}
   340 	return KErrNone;
   341 	}
   342 
   343 void RWsBuffer::Write(TInt handle,TUint opcode)
   344 	{
   345 	DoWrite(handle, opcode, iAutoFlush, NULL);
   346 	}
   347 
   348 /**
   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.
   353 
   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
   360 */
   361 void RWsBuffer::Write(TInt aHandle, TUint aOpcode, const TAny *aData, TInt aLength, const TIpcArgs* aIpcArgs/*=NULL*/)
   362 	{
   363 	TBool flush = (aIpcArgs != NULL ? ETrue : iAutoFlush);	// If aIpcArgs contains data then we do explicit flush
   364 	DoWrite(aHandle, aOpcode, flush, aIpcArgs, aData, aLength);
   365 	}
   366 
   367 /**
   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.
   372 
   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
   381 */
   382 void RWsBuffer::Write(TInt aHandle, TUint aOpcode, const TAny *aData, TInt aLength, const TAny *aData2, TInt aLength2, const TIpcArgs* aIpcArgs/*=NULL*/)
   383 	{
   384 	__ASSERT_DEBUG(!((aIpcArgs != NULL) &&  (aLength2 > 0 && aData2 == NULL)), Assert(EW32AssertBufferLogic));
   385 	TBool flush = iAutoFlush;
   386 	if (aLength2 > 0 && aData2 == NULL)
   387 		{
   388 		flush = EFalse;		// if just length2 is sent then we should not flush
   389 		}
   390 	else if (aIpcArgs != NULL)
   391 		{
   392 		flush = ETrue;		// If aIpcArgs contains data then we do explicit flush
   393 		}
   394 	DoWrite(aHandle, aOpcode, flush, aIpcArgs, aData, aLength, aData2, aLength2);
   395 	}
   396 
   397 /**
   398 Appends data directly to wserv buffer for the current command. So this function 
   399 should be used after adding the current command. 
   400 
   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
   408 
   409 Note: When data is added using this API, it pads out buffer to multiple of 4 bytes  
   410 */
   411 void RWsBuffer::AppendData(const TAny *aData,TInt aLength,TBool aFinished)
   412 	{
   413 	__ASSERT_ALWAYS(iBuf.MaxLength()-iBuf.Length()>=PaddedValue(aLength),Assert(EW32AssertBufferLogic));
   414 #if defined(_DEBUG)
   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));
   419 	if (aFinished)
   420 		iAppendDataLength = 0; 
   421 	else
   422 		iAppendDataLength -= aLength;
   423 #endif
   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)
   427 		Flush(NULL);
   428 	}
   429 
   430 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TIpcArgs* aIpcArgs)
   431 	{
   432 	return DoWrite(handle, opcode, ETrue, aIpcArgs);
   433 	}
   434 
   435 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData, TInt length,const TIpcArgs* aIpcArgs)
   436 	{
   437 	return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length);
   438 	}
   439 	
   440 TInt RWsBuffer::WriteReply(TInt handle,TUint opcode,const TAny *pData,TInt length,const TAny *pData2,TInt length2,const TIpcArgs* aIpcArgs)
   441 	{
   442 	return DoWrite(handle, opcode, ETrue, aIpcArgs, pData, length, pData2, length2);
   443 	}
   444 
   445 TInt RWsBuffer::WriteReplyP(TInt aHandle, TUint aOpcode, const TWriteDescriptorType& aReplyBuffer)
   446 	{
   447 	TIpcArgs ipcArgs;
   448 	aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
   449 	return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs);
   450 	}
   451 
   452 TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData,TInt aLength,const TWriteDescriptorType& aReplyBuffer)
   453 	{
   454 	TIpcArgs ipcArgs;
   455 	aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
   456 	return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
   457 	}
   458 
   459 TInt RWsBuffer::WriteReplyP(TInt aHandle,TUint aOpcode,const TAny *aData1,TInt aLengthData1,const TAny *aData2,TInt aLengthData2,const TWriteDescriptorType& aReplyBuffer)
   460 	{
   461 	TIpcArgs ipcArgs;
   462 	aReplyBuffer.SetDescriptorOnIpcArgs(ipcArgs);
   463 	return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData1, aLengthData1, aData2, aLengthData2);
   464 	}
   465 
   466 TInt RWsBuffer::WriteReplyWs(TUint opcode)
   467 //
   468 // Do a WriteReply using the sessions handle
   469 //
   470 	{
   471 	return(iSession->WriteReply(opcode));
   472 	}
   473 
   474 TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, TUint aOpcode)
   475 //
   476 // Do a WriteReply using the sessions handle
   477 //
   478 	{
   479 	return(iSession->WriteReply(pData,aLength,aOpcode));
   480 	}
   481 
   482 TInt RWsBuffer::WriteReplyWs(const TAny *pData, TInt aLength, const TAny *pData2, TInt aLength2, TUint aOpcode)
   483 //
   484 // Do a WriteReply using the sessions handle
   485 //
   486 	{
   487 	return(iSession->WriteReply(pData,aLength,pData2,aLength2,aOpcode));
   488 	}
   489 
   490 TInt RWsBuffer::WriteReplyByProvidingRemoteReadAccess(TInt aHandle,TUint aOpcode,const TAny *aData, TInt aLength,const TReadDescriptorType& aRemoteReadBuffer)
   491 	{
   492 	TIpcArgs ipcArgs;
   493 	aRemoteReadBuffer.SetDescriptorOnIpcArgs(ipcArgs);
   494 	return DoWrite(aHandle, aOpcode, ETrue, &ipcArgs, aData, aLength);
   495 	}
   496 
   497 void RWsBuffer::AddToBitmapArray(TInt aBitmapHandle)
   498 	{
   499 	if(aBitmapHandle && !iInvalidBitmapArray)
   500 		{
   501 		if(iBitmapArray.InsertInOrder(aBitmapHandle)==KErrNoMemory)
   502 
   503 			iInvalidBitmapArray=ETrue;
   504 		}
   505 	}
   506 	
   507 void RWsBuffer::SetWsGraphicManager(CWsGraphic::CManager* aManager)
   508 	{
   509 	__ASSERT_DEBUG(!WsGraphicManager(),Panic(EW32PanicGraphicInternal));
   510 	iManager = aManager;
   511 	}
   512 
   513 CWsGraphic::CManager* RWsBuffer::WsGraphicManager()
   514 	{
   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
   519 	}
   520 
   521 void RWsBuffer::AsyncRequest(TInt aHandle, TUint aOpcode, TRequestStatus& aStatus)
   522 	{
   523 	aStatus = KRequestPending;
   524 	__ASSERT_DEBUG((aOpcode&EWservMessAsynchronousService)==0, Assert(EW32AssertIllegalOpcode));
   525 	Flush(); 
   526 
   527 	const TInt function = EWservMessAsynchronousService | aOpcode;
   528 	TIpcArgs ipcArgs(aHandle);
   529 
   530 	iSession->SendReceive(function, ipcArgs, aStatus);
   531 	}
   532 
   533 void RWsBuffer::EnableWindowSizeCacheL()
   534     {
   535     if (iWindowSizeCache == NULL)
   536         {
   537         iWindowSizeCache = new (ELeave) RHashMap<TInt, TWindowSizeCacheEntry>();        
   538         }
   539     }