os/graphics/graphicscomposition/openwfsupport/test/tstreamoperation/surfaceutility.cpp
Update contrib.
1 // Copyright (c) 2007-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.
21 #include <imageconversion.h>
22 #include "surfaceutility.h"
24 CSurfaceUtility::CSurfaceUtility(CSurfaceUtility* aClone)
25 : iSurfaces(aClone?&(aClone->iSurfaces):NULL)
29 CSurfaceUtility* CSurfaceUtility::NewL(CSurfaceUtility* aClone/*=NULL*/)
31 CSurfaceUtility* utility = new (ELeave)CSurfaceUtility(aClone);
32 CleanupStack::PushL(utility);
33 utility->ConstructL();
34 CleanupStack::Pop(utility);
38 void CSurfaceUtility::ConstructL()
40 TInt r = iManager.Open();
43 LOG(("Surface manager failed to open: %d", r));
47 r = iSurfaceUpdateSession.Connect();
50 LOG(("Failed to connect to update server: %d", r));
55 CSurfaceUtility::~CSurfaceUtility()
63 iSurfaceUpdateSession.Close();
66 TBool CSurfaceUtility::DestroyAll()
69 TInt jj = iSurfaces.Count() - 1;
74 err = iManager.CloseSurface(iSurfaces[jj]);
77 LOG(("Error closing surface: 0x%X\n", err));
84 /***************************************
85 * The aim of the THeapSurfaceArray is to locally switch in the specified heap for any array operation
86 ***************************************/
88 CSurfaceUtility::RHeapSurfaceArray::RHeapSurfaceArray(RHeapSurfaceArray* aUseExternalArray)
89 : iUseArray(aUseExternalArray?aUseExternalArray->iUseArray:&this->iLocalArray),
90 iExternalHeapRef(aUseExternalArray?aUseExternalArray->iExternalHeapRef:User::Heap())
94 /************************************
95 * The following methods have been used by the surfaceutility... some require the heap wrapping, and some don't
96 * I actually need three different startegies (count em) for 7 methods...
97 * Some methods only read the existing objects, so don't need a heap swap at all
98 * Leaving methods have to use PopAndDestroy strategy to restore the heap on leaving or success
99 * Non-leaving methods must not call PushL, so directly make SwitchHeap calls!
100 ************************************/
102 // PopAndDestroy method to restore the heap
103 /*static*/ void CSurfaceUtility::RHeapSurfaceArray::PopHeap(void* aHeapPtr)
105 RHeap* heapPtr=(RHeap*)aHeapPtr;
106 User::SwitchHeap(heapPtr);
109 // Switches and pushes the previous heap so it can be restored with PopAndDestroy
110 /*static*/ void CSurfaceUtility::RHeapSurfaceArray::SwitchHeapLC(RHeap* aNewHeap)
112 CleanupStack::PushL(TCleanupItem(PopHeap,NULL));
113 CleanupStack::PushL(TCleanupItem(PopHeap,NULL));
114 CleanupStack::PushL(TCleanupItem(PopHeap,NULL));
115 CleanupStack::Pop(3);
116 RHeap* oldHeap=User::SwitchHeap(aNewHeap);
118 CleanupStack::PushL(TCleanupItem(PopHeap,oldHeap));
122 TSurfaceId& CSurfaceUtility::RHeapSurfaceArray::operator[](TUint aIndex)
124 return iUseArray->operator[](aIndex);
127 // Close only closes the local array, while Reset resets the active array (may be external)
128 void CSurfaceUtility::RHeapSurfaceArray::Close()
133 TInt CSurfaceUtility::RHeapSurfaceArray::Count() const
135 return iUseArray->Count();
138 // Close only closes the local array, while Reset resets the active array (may be external)
139 inline void CSurfaceUtility::RHeapSurfaceArray::Reset()
144 void CSurfaceUtility::RHeapSurfaceArray::AppendL(const TSurfaceId &anEntry)
146 iUseArray->AppendL(anEntry);
149 TInt CSurfaceUtility::RHeapSurfaceArray::Find(const TSurfaceId &anEntry) const
151 return iUseArray->Find(anEntry);
154 void CSurfaceUtility::RHeapSurfaceArray::Remove(TInt anIndex)
156 iUseArray->Remove(anIndex);
160 Cleanup stack helper object, holding references to both utility and surface, so
161 that the standard Close() semantics can be used.
163 class TSurfaceCleanup
166 TSurfaceCleanup(CSurfaceUtility& aUtility, TSurfaceId& aSurface)
167 : iUtility(aUtility), iSurface(aSurface)
171 // Removes the surface from the list of surfaces to clean up, and closes
172 // the surface reference.
173 iUtility.DestroySurface(iSurface);
176 CSurfaceUtility& iUtility;
177 TSurfaceId& iSurface;
181 Get the size of a surface.
183 @param aSurface The surface to get the size for.
184 @return The size in pixels, or empty on failure.
186 TSize CSurfaceUtility::SurfaceSize(const TSurfaceId& aSurface)
188 RSurfaceManager::TInfoBuf infoBuf;
189 RSurfaceManager::TSurfaceInfoV01& info = infoBuf();
191 if (iManager.SurfaceInfo(aSurface, infoBuf) == KErrNone)
200 Create a surface using the surface manager.
202 Stores the ID for tear down, as well as returning it.
204 @param aSize Dimensions of the surface.
205 @param aPixelFormat UID of the pixel format.
206 @param aStride Stride value for the surface (usually bytes per pixel * width)
207 @leave May leave due to lack of memory.
208 @return New surface's ID.
210 TSurfaceId CSurfaceUtility::CreateSurfaceL(const TSize& aSize, TUidPixelFormat aPixelFormat, TInt aStride, TInt aBuffers)
212 RSurfaceManager::TSurfaceCreationAttributesBuf bf;
213 RSurfaceManager::TSurfaceCreationAttributes& b = bf();
214 if (aStride<aSize.iWidth*BytesPerPixelL(aPixelFormat))
216 User::Leave(KErrOverflow);
218 b.iSize.iWidth = aSize.iWidth;
219 b.iSize.iHeight = aSize.iHeight;
220 b.iBuffers = aBuffers; // number of buffers in the surface
221 b.iPixelFormat = aPixelFormat;
222 b.iStride = aStride; // Number of bytes between start of one line and start of next
223 b.iOffsetToFirstBuffer = 0; // way of reserving space before the surface pixel data
224 b.iAlignment = 4; // alignment, 1,2,4,8 byte aligned
225 b.iContiguous = EFalse;
228 TSurfaceId surface = TSurfaceId::CreateNullId();
230 User::LeaveIfError(iManager.CreateSurface(bf, surface));
231 iSurfaces.AppendL(surface);
236 A helper function that returns the bytes per pixel for a given pixel format uid
238 @param aPixelFormat Pixel format UID to convert
239 @return The bytes per pixel
241 TInt CSurfaceUtility::BytesPerPixelL(TUidPixelFormat aPixelFormat)
243 TInt bytesPerPixel = 0;
244 switch (aPixelFormat)
246 case EUidPixelFormatXRGB_8888:
247 case EUidPixelFormatARGB_8888:
248 case EUidPixelFormatARGB_8888_PRE:
253 case EUidPixelFormatXRGB_4444:
254 case EUidPixelFormatARGB_4444:
255 case EUidPixelFormatRGB_565:
262 User::Leave(KErrNotSupported);
266 return bytesPerPixel;
270 Fill the given surface with a color.
272 @param aSurface The surface to be filled.
273 @param aColor The color to fill it with.
275 void CSurfaceUtility::FillSurfaceL(TSurfaceId& aSurface, const TRgb& aColor)
277 RSurfaceManager::TInfoBuf infoBuf;
278 RSurfaceManager::TSurfaceInfoV01& info = infoBuf();
280 User::LeaveIfError(iManager.SurfaceInfo(aSurface, infoBuf));
282 TBool use16 = EFalse;
284 if (info.iSize.iHeight<0 || info.iSize.iWidth<0 || info.iStride<0)
286 User::Leave(KErrCorrupt);
288 if (info.iSize.iHeight==0 || info.iSize.iWidth==0 || info.iStride==0)
290 User::Leave(KErrNotReady);
293 switch (info.iPixelFormat)
295 case EUidPixelFormatXRGB_8888:
297 color = aColor.Color16MU();
298 #ifdef ALPHA_FIX_24BIT
299 color |= ((ALPHA_FIX_24BIT)&0xff)<<24;
303 case EUidPixelFormatARGB_8888:
305 color = aColor.Color16MA();
308 case EUidPixelFormatARGB_8888_PRE:
310 color = aColor.Color16MAP();
313 case EUidPixelFormatXRGB_4444:
314 case EUidPixelFormatARGB_4444:
316 color = aColor.Color4K();
320 case EUidPixelFormatRGB_565:
322 color = aColor.Color64K();
328 User::Leave(KErrNotSupported);
334 User::LeaveIfError(iManager.MapSurface(aSurface, chunk));
335 CleanupClosePushL(chunk);
337 TInt offsetToFirstBuffer;
338 User::LeaveIfError(iManager.GetBufferOffset(aSurface, 0, offsetToFirstBuffer));
339 TUint8* surfacePtr = chunk.Base() + offsetToFirstBuffer;
340 TUint8* linePtr = surfacePtr;
344 if ( info.iSize.iWidth*2>info.iStride)
346 User::Leave(KErrOverflow);
348 TUint16* ptr = reinterpret_cast<TUint16*>(surfacePtr);
351 for (TInt xx = 0; xx < info.iSize.iWidth; xx++)
353 ptr[xx] = (TUint16)color;
358 if ( info.iSize.iWidth*4>info.iStride)
360 User::Leave(KErrOverflow);
362 TUint32* ptr = reinterpret_cast<TUint32*>(surfacePtr);
365 for (TInt xx = 0; xx < info.iSize.iWidth; xx++)
371 // Now copy that to the other lines
372 for (TInt yy = 1; yy < info.iSize.iHeight; yy++)
374 linePtr += info.iStride;
375 Mem::Copy(linePtr, surfacePtr, info.iSize.iWidth * BytesPerPixelL(info.iPixelFormat));
378 TInt err = SubmitUpdate(KAllScreens, aSurface, 0, NULL);
380 LOG(("Error submitting update: 0x%X\n", err));
382 CleanupStack::PopAndDestroy(/* chunk */);
386 Fill the given surface with a color.
388 @param aSurface The surface to be filled.
389 @param aBuffer The buffer to fill.
390 @param aColor The color to fill it with.
392 void CSurfaceUtility::FillSurfaceL(TSurfaceId& aSurface, TInt aBuffer, const TRgb& aColor)
394 RSurfaceManager::TInfoBuf infoBuf;
395 RSurfaceManager::TSurfaceInfoV01& info = infoBuf();
397 User::LeaveIfError(iManager.SurfaceInfo(aSurface, infoBuf));
399 TBool use16 = EFalse;
400 TInt numBuffers = info.iBuffers;
401 if (aBuffer < 0 || aBuffer >= numBuffers)
403 User::Leave(KErrArgument);
406 if (info.iSize.iHeight<0 || info.iSize.iWidth<0 || info.iStride<0)
408 User::Leave(KErrCorrupt);
410 if (info.iSize.iHeight==0 || info.iSize.iWidth==0 || info.iStride==0)
412 User::Leave(KErrNotReady);
415 switch (info.iPixelFormat)
417 case EUidPixelFormatXRGB_8888:
419 color = aColor.Color16MU();
420 #ifdef ALPHA_FIX_24BIT
421 color |= ((ALPHA_FIX_24BIT)&0xff)<<24;
425 case EUidPixelFormatARGB_8888:
427 color = aColor.Color16MA();
430 case EUidPixelFormatARGB_8888_PRE:
432 color = aColor.Color16MAP();
435 case EUidPixelFormatXRGB_4444:
436 case EUidPixelFormatARGB_4444:
438 color = aColor.Color4K();
442 case EUidPixelFormatRGB_565:
444 color = aColor.Color64K();
450 User::Leave(KErrNotSupported);
456 User::LeaveIfError(iManager.MapSurface(aSurface, chunk));
457 CleanupClosePushL(chunk);
460 User::LeaveIfError(iManager.GetBufferOffset(aSurface, aBuffer, offsetToBuffer));
461 TUint8* surfacePtr = chunk.Base() + offsetToBuffer;
462 TUint8* linePtr = surfacePtr;
466 if ( info.iSize.iWidth*2>info.iStride)
468 User::Leave(KErrOverflow);
470 TUint16* ptr = reinterpret_cast<TUint16*>(surfacePtr);
473 for (TInt xx = 0; xx < info.iSize.iWidth; xx++)
475 ptr[xx] = (TUint16)color;
480 if ( info.iSize.iWidth*4>info.iStride)
482 User::Leave(KErrOverflow);
484 TUint32* ptr = reinterpret_cast<TUint32*>(surfacePtr);
487 for (TInt xx = 0; xx < info.iSize.iWidth; xx++)
493 // Now copy that to the other lines
494 for (TInt yy = 1; yy < info.iSize.iHeight; yy++)
496 linePtr += info.iStride;
497 Mem::Copy(linePtr, surfacePtr, info.iSize.iWidth * BytesPerPixelL(info.iPixelFormat));
500 TInt err = SubmitUpdate(KAllScreens, aSurface, 0, NULL);
502 LOG(("Error submitting update: 0x%X\n", err));
504 CleanupStack::PopAndDestroy(/* chunk */);
511 As well as destroying the surface, it is removed from the set held for
512 destruction during tear down.
514 @param aSurface The surface to be destroyed.
516 void CSurfaceUtility::DestroySurface(TSurfaceId& aSurface)
518 TInt index = iSurfaces.Find(aSurface);
520 if (index != KErrNotFound)
522 iSurfaces.Remove(index);
525 TInt err = iManager.CloseSurface(aSurface);
527 LOG(("Error closing surfaces: 0x%X\n", err));
532 Submit an update to a surface to the update server.
534 @param aScreenNumber The screen to be updated where the surface is shown.
535 @param aSurface The surface which has been updated.
536 @param aRegion The area of the surface affected, or NULL for all of it.
538 TInt CSurfaceUtility::SubmitUpdate(TInt /* aScreenNumber */, const TSurfaceId& aSurface,TInt aBufferNumber, TInt aNullRegion)
542 return SubmitUpdate(KAllScreens, aSurface, aBufferNumber);
545 if (aBufferNumber==0)
547 return SubmitUpdate(KAllScreens, aSurface, aNullRegion);
551 return KErrNotSupported;
555 TInt CSurfaceUtility::SubmitUpdate(TInt /* aScreenNumber */, const TSurfaceId& aSurface, const TRegion* aRegion,TInt aBufferNumber)
557 return SubmitUpdate(KAllScreens, aSurface, aBufferNumber, aRegion);
560 TInt CSurfaceUtility::SubmitUpdate(TInt /* aScreenNumber */, const TSurfaceId& aSurface,TInt aBufferNumber, const TRegion* aRegion)
562 if (!iSurfaceUpdateSession.Handle())
564 iSurfaceUpdateSession.Connect();
566 if (!iSurfaceUpdateSession.Handle())
568 LOG(("Error - SUS client not started!"));
573 TInt err =iSurfaceUpdateSession.SubmitUpdate(KAllScreens, aSurface, aBufferNumber, aRegion);
575 LOG(("Error submitting update: 0x%X\n", err));
580 void CSurfaceUtility::FillNativeStreamSurfaceL(TSurfaceId& aSurface, TUint8* aBufferPtr, const TRgb& aColor)
582 RSurfaceManager::TInfoBuf infoBuf;
583 RSurfaceManager::TSurfaceInfoV01& info = infoBuf();
585 User::LeaveIfError(iManager.SurfaceInfo(aSurface, infoBuf));
587 TBool use16 = EFalse;
589 if (info.iSize.iHeight<0 || info.iSize.iWidth<0 || info.iStride<0)
591 User::Leave(KErrCorrupt);
593 if (info.iSize.iHeight==0 || info.iSize.iWidth==0 || info.iStride==0)
595 User::Leave(KErrNotReady);
598 switch (info.iPixelFormat)
600 case EUidPixelFormatXRGB_8888:
602 color = aColor.Color16MU();
603 #ifdef ALPHA_FIX_24BIT
604 color |= ((ALPHA_FIX_24BIT)&0xff)<<24;
608 case EUidPixelFormatARGB_8888:
610 color = aColor.Color16MA();
613 case EUidPixelFormatARGB_8888_PRE:
615 color = aColor.Color16MAP();
618 case EUidPixelFormatXRGB_4444:
619 case EUidPixelFormatARGB_4444:
621 color = aColor.Color4K();
625 case EUidPixelFormatRGB_565:
627 color = aColor.Color64K();
633 User::Leave(KErrNotSupported);
638 TUint8* surfacePtr = aBufferPtr;
639 TUint8* linePtr = surfacePtr;
643 if ( info.iSize.iWidth*2>info.iStride)
645 User::Leave(KErrOverflow);
647 TUint16* ptr = reinterpret_cast<TUint16*>(surfacePtr);
650 for (TInt xx = 0; xx < info.iSize.iWidth; xx++)
652 ptr[xx] = (TUint16)color;
657 if ( info.iSize.iWidth*4>info.iStride)
659 User::Leave(KErrOverflow);
661 TUint32* ptr = reinterpret_cast<TUint32*>(surfacePtr);
664 for (TInt xx = 0; xx < info.iSize.iWidth; xx++)
670 // Now copy that to the other lines
671 for (TInt yy = 1; yy < info.iSize.iHeight; yy++)
673 linePtr += info.iStride;
674 Mem::Copy(linePtr, surfacePtr, info.iSize.iWidth * BytesPerPixelL(info.iPixelFormat));
678 TBool CSurfaceUtility::CompareSurfacesL(TSurfaceId& aSurface, TInt aBuffer, TSurfaceId& aStreamSurface, TUint8* aBufferPtr)
680 RSurfaceManager::TInfoBuf infoBuf1;
681 RSurfaceManager::TSurfaceInfoV01& info1 = infoBuf1();
683 RSurfaceManager::TInfoBuf infoBuf2;
684 RSurfaceManager::TSurfaceInfoV01& info2 = infoBuf2();
686 User::LeaveIfError(iManager.SurfaceInfo(aSurface, infoBuf1));
687 User::LeaveIfError(iManager.SurfaceInfo(aStreamSurface, infoBuf2));
688 TBool use16 = EFalse;
690 if (aBuffer < 0 || aBuffer >= info1.iBuffers)
692 User::Leave(KErrArgument);
695 if (info1.iPixelFormat != info2.iPixelFormat)
697 User::Leave(KErrArgument);
700 if ((info1.iSize.iHeight<0 || info1.iSize.iWidth<0 || info1.iStride<0) ||
701 (info2.iSize.iHeight<0 || info2.iSize.iWidth<0 || info2.iStride<0))
703 User::Leave(KErrCorrupt);
705 if ((info1.iSize.iHeight==0 || info1.iSize.iWidth==0 || info1.iStride==0) ||
706 (info2.iSize.iHeight==0 || info2.iSize.iWidth==0 || info2.iStride==0))
708 User::Leave(KErrNotReady);
710 if (info1.iSize != info2.iSize)
712 User::Leave(KErrArgument);
715 switch (info1.iPixelFormat)
717 case EUidPixelFormatXRGB_8888:
718 case EUidPixelFormatARGB_8888:
719 case EUidPixelFormatARGB_8888_PRE:
723 case EUidPixelFormatXRGB_4444:
724 case EUidPixelFormatARGB_4444:
725 case EUidPixelFormatRGB_565:
732 User::Leave(KErrNotSupported);
739 User::LeaveIfError(iManager.MapSurface(aSurface, chunk));
740 CleanupClosePushL(chunk);
743 User::LeaveIfError(iManager.GetBufferOffset(aSurface, aBuffer, offsetToBuffer));
744 TUint8* surfacePtr1 = chunk.Base() + offsetToBuffer;
747 TUint8* surfacePtr2 = aBufferPtr;
754 if ((info1.iSize.iWidth*2>info1.iStride) ||
755 (info2.iSize.iWidth*2>info2.iStride))
757 User::Leave(KErrOverflow);
760 TUint16* ptr1 = reinterpret_cast<TUint16*>(surfacePtr1);
761 TUint16* ptr2 = reinterpret_cast<TUint16*>(surfacePtr2);
764 for (TInt xx = 0; xx < info1.iSize.iWidth; xx++)
766 for (TInt yy = 0; yy < info1.iSize.iHeight; yy++)
768 color1 = (TUint16)ptr1[xx];
769 color2 = (TUint16)ptr2[xx];
770 if (color1 != color2)
779 if ((info1.iSize.iWidth*4>info1.iStride) ||
780 (info2.iSize.iWidth*4>info2.iStride))
782 User::Leave(KErrOverflow);
785 TUint32* ptr1 = reinterpret_cast<TUint32*>(surfacePtr1);
786 TUint32* ptr2 = reinterpret_cast<TUint32*>(surfacePtr2);
789 for (TInt xx = 0; xx < info1.iSize.iWidth; xx++)
791 for (TInt yy = 0; yy < info1.iSize.iHeight; yy++)
795 if (color1 != color2)
797 CleanupStack::PopAndDestroy(/* chunk */);
804 CleanupStack::PopAndDestroy(/* chunk */);
808 void CSurfaceUtility::NotifyWhenDisplayed(TRequestStatus& aStatusDisplayed, TTimeStamp& aTimeStamp)
810 iSurfaceUpdateSession.NotifyWhenDisplayed(aStatusDisplayed, aTimeStamp);
813 void CSurfaceUtility::NotifyWhenDisplayedXTimes(TInt aCount, TRequestStatus& aStatusDisplayedX)
815 iSurfaceUpdateSession.NotifyWhenDisplayedXTimes(aCount, aStatusDisplayedX);
818 void CSurfaceUtility::NotifyWhenAvailable(TRequestStatus& aStatusAvailable)
820 iSurfaceUpdateSession.NotifyWhenAvailable(aStatusAvailable);