graphicscomposition/openwfsupport/test/tstreamoperation/tnativestream.cpp
author Pat Downey <patd@symbian.org>
Tue, 04 May 2010 16:06:01 +0100
changeset 46 ca46a316904d
parent 0 5d03bc08d59c
child 98 bf7481649c98
child 163 bbf46f59e123
permissions -rw-r--r--
Merge KhronosRI and bld.inf fix.

// Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// Implementation of Test class for OpenWfc Native Stream 
// 
//

#include <graphics/symbianstream.h>
#include <graphics/streammap.h>
#include <graphics/compositionsurfaceupdate.h>
#include <graphics/extensioncontainer.h>
#include <graphics/suerror.h>
#include <test/extendtef.h>
#include <dispchannel.h>
#include <hal.h>
#include "tnativestream.h"
#include "surfaceutility.h"
#include "owftestexithelper.inl"

#define BUFFER_READ_HANDLE_BASE  0x100
#define BUFFER_WRITE_HANDLE_BASE 0x200
#define INDEX_TO_READ_HANDLE(x)  ((SymbianStreamBuffer) ((x)+BUFFER_READ_HANDLE_BASE))
#define INDEX_TO_WRITE_HANDLE(x)  ((SymbianStreamBuffer) ((x)+BUFFER_WRITE_HANDLE_BASE))
#define BUFFER_READ_HANDLE_TO_INDEX(x)  (TInt) (x > 0 ? (x&0xFF) : (x-BUFFER_READ_HANDLE_BASE))
#define BUFFER_WRITE_HANDLE_TO_INDEX(x)  (TInt) (x > 0 ? (x&0xFF) : (x-BUFFER_WRITE_HANDLE_BASE))
#define WFC_INVALID_HANDLE NULL
#define KGrowCleanupStack   12
#define KCompositorVersion 0x01023456
#define KCompositorVersionMajor 0x1
#define KCompositorVersionMinor 0x2
#define KCompositorVersionRevision 0x3456

void PopHeap(void* aHeapPtr)
    {
    User::SwitchHeap((RHeap*)aHeapPtr);
    }

void GrowCleanupStackL()
    {
    TInt n = KGrowCleanupStack;
    while(n--)
        {
        CleanupStack::PushL((CBase*)NULL);
        }
    CleanupStack::Pop(KGrowCleanupStack);
    }

// Helper functions
/* supported external image formats */
enum OWF_PIXEL_FORMAT   {
    OWF_IMAGE_NOT_SUPPORTED = 0,
    OWF_IMAGE_ARGB8888      = 0x8888,
    OWF_IMAGE_XRGB8888      = 0xf888,
    OWF_IMAGE_RGB888        = 0x888,
    OWF_IMAGE_RGB565        = 0x565,
    OWF_IMAGE_L32           = 0xA32,
    OWF_IMAGE_L16           = 0xA16,
    OWF_IMAGE_L8            = 0xA8,
    OWF_IMAGE_L1            = 0xA1,
    OWF_IMAGE_ARGB_INTERNAL = 0x666 /* OWFpixel rep */
} ;

struct CTestNativeStream::OWF_IMAGE_FORMAT{
    OWF_PIXEL_FORMAT        pixelFormat;
    bool              linear;
    bool              premultiplied;
    int                  rowPadding; /* row alignment, in bytes */
} ;

TInt CTestNativeStream::BytesPerPixel(TUidPixelFormat aPixelFormat)
    {
    switch (aPixelFormat)
        {
        case EUidPixelFormatXRGB_8888:
        case EUidPixelFormatARGB_8888:
        case EUidPixelFormatBGRX_8888:
        case EUidPixelFormatXBGR_8888:
        case EUidPixelFormatBGRA_8888:
        case EUidPixelFormatABGR_8888:  
        case EUidPixelFormatABGR_8888_PRE:
        case EUidPixelFormatARGB_8888_PRE:
        case EUidPixelFormatBGRA_8888_PRE:
        case EUidPixelFormatARGB_2101010:
        case EUidPixelFormatABGR_2101010:
            return 4;
        case EUidPixelFormatBGR_888:
        case EUidPixelFormatRGB_888:
            return 3;
        case EUidPixelFormatXRGB_4444:
        case EUidPixelFormatARGB_4444:
        case EUidPixelFormatXBGR_4444:
        case EUidPixelFormatRGB_565:
        case EUidPixelFormatBGR_565:
        case EUidPixelFormatARGB_1555:
        case EUidPixelFormatXRGB_1555:
        case EUidPixelFormatARGB_8332:
        case EUidPixelFormatBGRX_5551:
        case EUidPixelFormatBGRA_5551:
        case EUidPixelFormatBGRA_4444:
        case EUidPixelFormatBGRX_4444:
        case EUidPixelFormatAP_88:
            return  2;
        case EUidPixelFormatRGB_332:
        case EUidPixelFormatBGR_332:
        case EUidPixelFormatA_8:
        case EUidPixelFormatL_8:
            return  1;
        case EUidPixelFormatP_8:
            return -1;
        case EUidPixelFormatP_4:
        case EUidPixelFormatL_4:
            return -2;
        case EUidPixelFormatL_2:
        case EUidPixelFormatP_2:
            return -4;
        case EUidPixelFormatL_1 :
            return -8;
        default:
            {
            return 0;
            }
        }
    }

/*****************************************
 * Helper Creates Surface from OWF spec 
 * 
 *
 */
TSurfaceId CTestNativeStream::helperCreateSurfaceL(khronos_int32_t width,
        khronos_int32_t height,
        const OWF_IMAGE_FORMAT*   format,
        khronos_int32_t nbufs,
        TUidPixelFormat overridePixelFormat)
	{
	RSurfaceManager::TSurfaceCreationAttributesBuf bf;
	RSurfaceManager::TSurfaceCreationAttributes& b = bf();
	
	TBool premultiplied = format->premultiplied;
	OWF_PIXEL_FORMAT pixelFormat = format->pixelFormat;
	khronos_int32_t bytesPerPixel = 0;
	
	if (overridePixelFormat != EUidPixelFormatUnknown)
	    {
        bytesPerPixel = BytesPerPixel(overridePixelFormat);
        b.iAlignment = 4;
        b.iPixelFormat = overridePixelFormat;
	    }
	else
	    {
        switch(pixelFormat)
            {
            case OWF_IMAGE_RGB565:
                b.iPixelFormat = EUidPixelFormatRGB_565;
                bytesPerPixel = 2;
                b.iAlignment = 4;
                break;
            case OWF_IMAGE_ARGB8888:
                {
                if (premultiplied)
                    {
                    b.iPixelFormat = EUidPixelFormatARGB_8888_PRE;
                    }
                else
                    {
                    b.iPixelFormat = EUidPixelFormatARGB_8888;
                    }
                bytesPerPixel = 4;
                b.iAlignment = 4;
                break;
                }
            case OWF_IMAGE_XRGB8888 :
                b.iPixelFormat = EUidPixelFormatXRGB_8888;
                bytesPerPixel = 4;
                b.iAlignment = 4;
                break;
            case OWF_IMAGE_L8 :
                b.iPixelFormat = EUidPixelFormatA_8;
                bytesPerPixel = 1;
                b.iAlignment = 4;
                break;
            case OWF_IMAGE_L1 :
                b.iPixelFormat = EUidPixelFormatL_1;
                bytesPerPixel = -8;
                b.iAlignment = 4;
                break;
            default:
                User::Leave(KErrNotSupported);	
                break;
            }
	    }
	
	b.iSize.iWidth = width;
	b.iSize.iHeight = height;
	b.iBuffers = nbufs;								// number of buffers in the surface
	b.iOffsetToFirstBuffer = 0;						// way of reserving space before the surface pixel data
	if (bytesPerPixel >= 0)
	    {
	    b.iStride = bytesPerPixel * width;	// number of bytes between start of one line and start of next	
	    }
	else
	    {
	    b.iStride = (width-(bytesPerPixel+1)) / (-bytesPerPixel);
	    }
	b.iContiguous = EFalse;
	b.iMappable = ETrue;
	
	TSurfaceId surface = TSurfaceId::CreateNullId();
	User::LeaveIfError(iUtility->Manager().CreateSurface(bf, surface));
	
	return surface;
	}

SymbianStreamType CTestNativeStream::helperCreateImageStream(khronos_int32_t width,
                                 khronos_int32_t height,
                                 const OWF_IMAGE_FORMAT*  format,
                                 khronos_int32_t nbufs,
                                 TUidPixelFormat overridePixelFormat)
	{
	TSurfaceId surface;
	
	TRAPD(err,surface = helperCreateSurfaceL(width, height, format, nbufs, overridePixelFormat));
	if (err)
	    {
	    return WFC_INVALID_HANDLE;
	    }
	SymbianStreamType ns;
	SymbianStreamAcquire(&surface, &ns);
	
	iUtility->Manager().CloseSurface(surface);
	
	return ns;
	}

RSemaphore      gSemaphore;

TGlobalNativeStreamVar	gVarInstance={0};
const TGlobalNativeStreamVar& TGlobalNativeStreamVar::Instance()
	{
	return gVarInstance;
	}

void TGlobalNativeStreamVar::SetSurfaceID(TSurfaceId aSurfaceID)
    {
    iSurfaceID = aSurfaceID;
    }

void TGlobalNativeStreamVar::SetTestComplete(TBool aTestComplete)
    {
    iTestComplete = aTestComplete;
    }

void TGlobalNativeStreamVar::SetBuffers(TInt aBuffers)
    {
    iBuffers = aBuffers;
    }

TSurfaceId TGlobalNativeStreamVar::SurfaceID() const
    {
    return iSurfaceID;
    }

TBool TGlobalNativeStreamVar::TestComplete() const
    {
    return iTestComplete;
    }

TInt TGlobalNativeStreamVar::Buffers() const
    {
    return iBuffers;
    }

/*
 * CTestNativeStream implementation
 * 
 * 
 * 
 * 
 */

CTestNativeStream::CTestNativeStream(): iUtility(this)
	{
	// No implementation required
	}

CTestNativeStream::~CTestNativeStream()
	{
	DeleteOwfSingletons();
	}

void CTestNativeStream::SetupL()
	{
	TRAPD(err_FailedToCreateSurfaceUtility, iUtility = CSurfaceUtility::NewL( NULL ));
	ASSERT_EQUALS(err_FailedToCreateSurfaceUtility,KErrNone);
	DefineOwfSingletonKeys();
	}

/**
 * test Suite furniture 
 **/
void CTestNativeStream::TearDownL()
	{
	delete iUtility();
	}

/**
WFC context callback function invoked by native stream when the stream's content
is updated. For testing, we simply call a class member function that checks that the
correct native stream handle and events mask were supplied.
*/
void CTestNativeStream::SourceStreamUpdatedCallback(
        SymbianStreamType aNs, khronos_int32_t aEvents, void* aData, void* aParam)
    {
    (void)aData;
    if (aEvents == ESOWF_ObserverReturnDefaultEvent && aParam)
        {
        SYMOWF_DEFAULT_EVENT_PARAM* parameter = (SYMOWF_DEFAULT_EVENT_PARAM*) aParam;
        if ((parameter->length) == sizeof(SYMOWF_DEFAULT_EVENT_PARAM))
            {
            parameter->event = ESOWF_EventUpdated;
            }
        return;
        }
        
    ASSERT(CTestNativeStream::iTester);
    CTestNativeStream::iTester->CheckSourceStreamUpdated(aNs, aParam);
    }

void CTestNativeStream::CheckSourceStreamUpdated(SymbianStreamType aNs, void* aParam)
    {
    RHeap *h1 = &(User::Heap());
    if (aParam)
        {
        ASSERT_EQUALS(iNs, aNs);
        SYMOWF_CONTENT_UPDATED_PARAM* param = (SYMOWF_CONTENT_UPDATED_PARAM*) aParam;
        ASSERT_TRUE(param->id == SYM_CONTENT_UPDATE_BEGIN || 
                    param->id == SYM_CONTENT_UPDATE_END ||
                    param->id == SYM_CONTENT_UPDATE);

        iSourceStreamUpdatedCalled++;
        iStreamUpdatedParameter = param->id; 
        switch(param->id)
            {
            case SYM_CONTENT_UPDATE_BEGIN:
                param->immediateAvailable = iImmediateAvailable;
                param->immediateVisibility = iImmediateVisible;
                param->serialNumber = iStreamUpdatedSerialNumber;
                break;
                
            case SYM_CONTENT_UPDATE:
            case SYM_CONTENT_UPDATE_END:
                ASSERT_EQUALS(iExpectedSourceStreamUpdatedEventMask, param->par);        
                iContextUpdatedFlags |= param->par & (~ESOWF_EventUpdated);    
                break;
            default:
                break;
            }
        }
    else
        {
        iSourceStreamUpdatedCalled++;
        }
        
    }

/**
Remove the native stream notifications. The creator of the source is responsible
for destroying the native stream. 

Now with the new SymbianStreamRemoveObserver function we need to pass in an observer
*/
void CTestNativeStream::RemoveNsNotifications()
    {
    SymbianStreamRemoveObserver(iNs, &iScreenNo, ESOWF_EventAvailable);
    SymbianStreamRemoveObserver(iNs, &iScreenNo, ESOWF_EventDisplayed);
    SymbianStreamRemoveObserver(iNs, &iScreenNo, ESOWF_EventDisplayedX);
    }

CTestNativeStream* CTestNativeStream::iTester = NULL;

// Create a suite of all the tests
CTestSuite* CTestNativeStream::CreateSuiteL(const TDesC& aName)
	{
	SymbianStreamRegisterScreenNotifications(0, 10, KCompositorVersion);
	SUB_SUITE_OPT(CTestNativeStream,NULL);

		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0100L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0101L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0102L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0103L);
		
		// Test with 1, 2 and 3 buffers
		ADD_TEST_STEP_PARAM_RANGE(GRAPHICS_OPENWFC_NATIVESTREAM_0104L,1,4);
		ADD_TEST_STEP_PARAM_RANGE(GRAPHICS_OPENWFC_NATIVESTREAM_0105L,1,4);
		ADD_TEST_STEP_PARAM_RANGE(GRAPHICS_OPENWFC_NATIVESTREAM_0106L,1,4);
		
		// Concurrent tests
		ADD_TEST_STEP_PARAM_RANGE(CreateSharedNativeStreamL,1,4);
		ADD_THIS_TEST_STEP(DestroySharedNativeStreamL);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0107_1L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0107_2L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0107_3L);
		
		// Test SUS with OpenWF pipeline
		ADD_TEST_STEP_PARAM_RANGE(GRAPHICS_OPENWFC_NATIVESTREAM_0108L, 1, 4);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0109L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0110L);
		
		// Test various cases for Native Stream callbacks
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0111L);
		// Test multithreaded cases for Native Stream callbacks
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0112_1L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0112_2L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0112_3L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0112_4L);
		
		// Notification tests
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0113L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0114L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0115L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0116L);
		
		// Notification cancel
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0117_1L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0117_2L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0117_3L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0117_4L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0118_1L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0118_2L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0118_3L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0118_4L);
		
		// Notification overflow
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0119_1L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0119_2L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0119_3L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0120_1L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0120_2L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0120_3L);
		
		// Notifications cancelled due to removal of source observer
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0122L);
		
		// Sym native stream add/remove observers
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0130L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0131L);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0132L);
		
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0133L);

		// Test with 1, 2 and 3 buffers
		ADD_TEST_STEP_PARAM_RANGE(GRAPHICS_OPENWFC_NATIVESTREAM_0140L,1,4);
		ADD_TEST_STEP_PARAM_RANGE(GRAPHICS_OPENWFC_NATIVESTREAM_0141L,1,4);
		ADD_THIS_TEST_STEP(GRAPHICS_OPENWFC_NATIVESTREAM_0142L);

	END_SUITE;	
	
	}

// This handles any non-member uses of the extended ASSERT_XXX macros
void TefUnitFailLeaveL()
	{
	User::Leave(KErrTEFUnitFail);
	}

SymbianStreamType CTestNativeStream::NsCheckL(const TSurfaceId aId, TInt aCheck, TBool aFind)
	{
	// Acquire (create OR find) the stream
	SymbianStreamType ns = NULL;
	TInt err;
	if (aFind)
		{
		err = (TInt)SymbianStreamFind(&aId, &ns);
		}
	else
		{
		err = SymbianStreamAcquire(&aId, &ns);
		}
	ASSERT_TRUE(err == KErrNone);
	// Check the hash map count
	ASSERT_EQUALS((COpenWfcStreamMap::InstanceL().Count()), aCheck);
    SymbianStreamBuffer bufferHandle;
    err = SymbianStreamAcquireReadBuffer(ns,&bufferHandle);
    ASSERT_TRUE(err == KErrNone);
    long bufferIndex;
    const TSurfaceId* checkId = NULL;
    err = SymbianStreamGetBufferId(ns,bufferHandle,&bufferIndex,&checkId);
    ASSERT_TRUE(err == KErrNone);
    SymbianStreamReleaseReadBuffer(ns,bufferHandle);
	ASSERT_NOT_NULL(checkId);
	ASSERT_EQUALS(*checkId, aId);
	return ns;
	}

void CTestNativeStream::CreateSharedNativeStreamL(TInt aBuffers)
	{
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    GrowCleanupStackL();
    
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	
	gVarInstance.SetTestComplete(EFalse);
	gVarInstance.SetBuffers(aBuffers);
	
	TSize surfaceSize(TSize(100,100));
	khronos_int32_t width = surfaceSize.iWidth;
	khronos_int32_t height = surfaceSize.iHeight;
	
	OWF_IMAGE_FORMAT pixelFormat =
		{
		OWF_IMAGE_ARGB8888,
		ETrue,
		ETrue,
		4
		};	
	
	// Create the first stream
	SymbianStreamType ns=helperCreateImageStream(width,
													height,
													&pixelFormat,
													aBuffers);
	ASSERT_TRUE(ns);
	
    SymbianStreamBuffer bufferHandle;
    SymbianStreamAcquireReadBuffer(ns,&bufferHandle);
    long bufferIndex;
    const TSurfaceId* checkId = NULL;
    TInt err = SymbianStreamGetBufferId(ns,bufferHandle,&bufferIndex,&checkId);
    ASSERT_TRUE(err == KErrNone);
    SymbianStreamReleaseReadBuffer(ns,bufferHandle);
	gVarInstance.SetSurfaceID(*checkId);
    gSemaphore.CreateLocal(1);
    gSemaphore.Wait(); 
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

void CTestNativeStream::DestroySharedNativeStreamL()
	{
	GrowCleanupStackL();
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(COpenWfcStreamMap::InstanceL().GetMainHeap())));
	
	gSemaphore.Close();
	TSurfaceId id = gVarInstance.SurfaceID();
	SymbianStreamType ns;
	TInt err = SymbianStreamFind(&id,&ns);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns);
	// Decrease stream's reference count by one. This removes reference added by owfNativeStreamFind()
	SymbianStreamRemoveReference(ns);
	// Decrease stream's reference count by one to make reference count zero, and destroy the stream
	SymbianStreamRemoveReference(ns);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

// Test observers
void TestUpdateCallback(SymbianStreamType aStream, khronos_int32_t aEvent, void* aData, void* aParam)
    {
    (void) aStream;
    switch (aEvent)
        {
        case ESOWF_ObserverReturnDefaultEvent:
            if (aParam)
                {
                SYMOWF_DEFAULT_EVENT_PARAM* parameter = (SYMOWF_DEFAULT_EVENT_PARAM*) aParam;
                if ((parameter->length) == sizeof(SYMOWF_DEFAULT_EVENT_PARAM))
                    {
                    parameter->event = ESOWF_EventUpdated;
                    }
                }
            return;
            
        case ESOWF_EventUpdated:
            {
            TInt* localNumber = (TInt*)aData;
             ++(*localNumber);
            }
            break;
        default:
            break;
        }
    }

// Test compose target
void TestComposedCallback(SymbianStreamType aStream, khronos_int32_t aEvent, void* aData, void* aParam)
    {
    (void) aStream;
    switch (aEvent)
        {
        case ESOWF_ObserverReturnDefaultEvent:
            if (aParam)
                {
                SYMOWF_DEFAULT_EVENT_PARAM* parameter = (SYMOWF_DEFAULT_EVENT_PARAM*) aParam;
                if ((parameter->length) == sizeof(SYMOWF_DEFAULT_EVENT_PARAM))
                    {
                    parameter->event = ESOWF_EventComposed;
                    }
                }
            return;
            
        case ESOWF_EventComposed:
            {
            TInt* localNumber = (TInt*)aData;
             ++(*localNumber);
            }
            break;
        default:
            break;
        }
    }

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0100
@SYMTestCaseDesc		Create a native stream using SymbianStreamAcquire()
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify native stream objects can be created and persist unique surface ID values
@SYMTestActions	
		Create two surfaces,
		Create two streams from the surfaces
		Read back the surface Ids from the streams
@SYMTestExpectedResults
		The surface IDs should be non-null and unique
		The streams should be non-null and unique
		The returned surface Ids should match the constructed IDs
 **/		
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0100L()
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);
	
	TInt count = COpenWfcStreamMap::InstanceL().Count();
	TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100),EUidPixelFormatARGB_8888_PRE,400,2);
	ASSERT_FALSE(surface.IsNull());
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	SymbianStreamType ns;
	err = SymbianStreamAcquire(&surface, &ns);
	ASSERT_TRUE(err == KErrNone);
    SymbianStreamBuffer bufferHandle;
    err = SymbianStreamAcquireReadBuffer(ns, &bufferHandle);
    ASSERT_TRUE(err == KErrNone);
    long bufferIndex;
    const TSurfaceId* getId = NULL;
    err = SymbianStreamGetBufferId(ns,bufferHandle,&bufferIndex,&getId);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns,bufferHandle);
    ASSERT_TRUE(err == KErrNone);
	ASSERT_EQUALS(*getId,surface);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	
	TSurfaceId surface2=iUtility->CreateSurfaceL(TSize(100,100),EUidPixelFormatARGB_8888_PRE,400,2);
	ASSERT_FALSE(surface2.IsNull());
	ASSERT_NOT_EQUALS(surface,surface2);
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	SymbianStreamType ns2;
	err = SymbianStreamAcquire(&surface2,&ns2);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns2);
	ASSERT_NOT_EQUALS(ns,ns2);
	ASSERT_EQUALS((COpenWfcStreamMap::InstanceL().Count()), count + 2);
	err = SymbianStreamAcquireReadBuffer(ns2,&bufferHandle);
	ASSERT_TRUE(err == KErrNone);
    const TSurfaceId* getId2 = NULL;
    err = SymbianStreamGetBufferId(ns2,bufferHandle,&bufferIndex,&getId2);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns2,bufferHandle);
    ASSERT_TRUE(err == KErrNone);
	ASSERT_NOT_NULL(getId2);
	ASSERT_EQUALS(*getId2,surface2);
	
	SymbianStreamRemoveReference(ns);
	ASSERT_EQUALS((COpenWfcStreamMap::InstanceL().Count()), count + 1);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	iUtility->DestroySurface(surface);
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	SymbianStreamRemoveReference(ns2);
	ASSERT_EQUALS((COpenWfcStreamMap::InstanceL().Count()), count);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	iUtility->DestroySurface(surface2);
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0101
@SYMTestCaseDesc		Create a native stream using SymbianStreamCreateImageStream()
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify native stream objects can be created and persist unique surface ID values
@SYMTestActions	
		Create two streams from the parameters passed in
		Read back the surface Ids from the streams
@SYMTestExpectedResults
		The surface IDs should be non-null and unique
		The streams should be non-null and unique
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0101L()
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);
	
	TSize surfaceSize(TSize(100,100));
	TInt width = surfaceSize.iWidth;
	TInt height = surfaceSize.iHeight;
	TInt buffers = 2;

	OWF_IMAGE_FORMAT pixelFormat =
		{
		OWF_IMAGE_ARGB8888,
	    ETrue,
	    ETrue,
	    4
	    };	
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	
	// Create the first stream
	SymbianStreamType ns=helperCreateImageStream(width,
													height,
													&pixelFormat,
													buffers);
	ASSERT_TRUE(ns);
    SymbianStreamBuffer bufferHandle;
    err = SymbianStreamAcquireReadBuffer(ns,&bufferHandle);
    ASSERT_TRUE(err == KErrNone);
    long bufferIndex;
    const TSurfaceId* getId = NULL;
    err = SymbianStreamGetBufferId(ns,bufferHandle,&bufferIndex,&getId);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns,bufferHandle);
    ASSERT_TRUE(err == KErrNone);
	
	// Create the second stream
	SymbianStreamType ns2=helperCreateImageStream(width,
													height,
													&pixelFormat,
													buffers);
	ASSERT_TRUE(ns2);
	
    err = SymbianStreamAcquireReadBuffer(ns2,&bufferHandle);
    ASSERT_TRUE(err == KErrNone);
    const TSurfaceId* getId2 = NULL;
    err = SymbianStreamGetBufferId(ns2,bufferHandle,&bufferIndex,&getId2);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns2,bufferHandle);
    ASSERT_TRUE(err == KErrNone);
	ASSERT_NOT_NULL(getId2);
	
	ASSERT_NOT_EQUALS(ns,ns2);
	ASSERT_NOT_EQUALS(getId,getId2);
	
	SymbianStreamRemoveReference(ns);
	SymbianStreamRemoveReference(ns2);
	
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0102
@SYMTestCaseDesc		Create a native stream 2
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify native stream objects can be created and persist unique surface ID values
@SYMTestActions	
		Create two surfaces,
		Create two streams from the surfaces
		Read back the surface Ids from the streams
		Acquire multiple time the native streams
		Find native streams
@SYMTestExpectedResults
		The surface IDs should be non-null and unique
		The streams should be non-null and unique
		The returned surface Ids should match the constructed IDs
		The hash map counter should be updated accordingly when native streams are created or destroyed
 **/		
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0102L()
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);
	
	COpenWfcStreamMap& singleton = COpenWfcStreamMap::InstanceL();
	TInt check = COpenWfcStreamMap::InstanceL().Count() + 1;
	// Create the first surface
	TSurfaceId surfaceId1 = iUtility->CreateSurfaceL(TSize(100, 100), EUidPixelFormatARGB_8888_PRE, 400, 2);
	ASSERT_FALSE(surfaceId1.IsNull());
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	SymbianStreamType ns1 = NsCheckL(surfaceId1, check, EFalse);
	
	TSurfaceId surfaceId2;
	surfaceId2.CreateNullId();
	SymbianStreamType ns2;
	err = SymbianStreamFind(&surfaceId2,&ns2);
	ASSERT_TRUE(err == KErrNotFound);
	ASSERT_FALSE(ns2);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	
	// Create the second surface
	surfaceId2 = iUtility->CreateSurfaceL(TSize(100, 100), EUidPixelFormatARGB_8888_PRE, 400, 2);
	ASSERT_FALSE(surfaceId2.IsNull()); 
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	err = SymbianStreamFind(&surfaceId2,&ns2);
	ASSERT_TRUE(err == KErrNotFound);
	ASSERT_FALSE(ns2);
	
	ns2 = NsCheckL(surfaceId2, ++check, EFalse);
	
	NsCheckL(surfaceId1, check, EFalse);
	
	NsCheckL(surfaceId1, check, ETrue);
	
	NsCheckL(surfaceId2, check, ETrue);
	
	SymbianStreamRemoveReference(ns1);
	ASSERT_EQUALS((singleton.Count()), check);
	
	SymbianStreamRemoveReference(ns2);
	ASSERT_EQUALS((singleton.Count()), check);
	
	SymbianStreamRemoveReference(ns2);
	ASSERT_EQUALS((singleton.Count()), --check);

	SymbianStreamRemoveReference(ns1);
	ASSERT_EQUALS((singleton.Count()), check);
	
	SymbianStreamRemoveReference(ns1);
	ASSERT_EQUALS((singleton.Count()), --check);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	
	iUtility->DestroySurface(surfaceId1);
	iUtility->DestroySurface(surfaceId2);
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	err = SymbianStreamFind(&surfaceId2,&ns2);
	ASSERT_TRUE(err == KErrNotFound);
	ASSERT_FALSE(ns2);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

struct SupportedFormats
    {
    TUidPixelFormat symbianPixelFormat;
    } supportedFormats[]=
    {
        EUidPixelFormatXRGB_8888,
        EUidPixelFormatARGB_8888,
        EUidPixelFormatBGRX_8888,
        EUidPixelFormatXBGR_8888,
        EUidPixelFormatBGRA_8888,
        EUidPixelFormatABGR_8888, 
        EUidPixelFormatABGR_8888_PRE,
        EUidPixelFormatARGB_8888_PRE,
        EUidPixelFormatBGRA_8888_PRE,
        EUidPixelFormatARGB_2101010,
        EUidPixelFormatABGR_2101010,
        EUidPixelFormatBGR_888,
        EUidPixelFormatRGB_888,
        EUidPixelFormatXRGB_4444,
        EUidPixelFormatARGB_4444,
        EUidPixelFormatXBGR_4444,
        EUidPixelFormatRGB_565,
        EUidPixelFormatBGR_565,
        EUidPixelFormatARGB_1555,
        EUidPixelFormatXRGB_1555,
        EUidPixelFormatARGB_8332,
        EUidPixelFormatBGRX_5551,
        EUidPixelFormatBGRA_5551,
        EUidPixelFormatBGRA_4444,
        EUidPixelFormatBGRX_4444,
        EUidPixelFormatAP_88,
        EUidPixelFormatRGB_332,
        EUidPixelFormatBGR_332,
        EUidPixelFormatA_8,
        EUidPixelFormatL_8,
        EUidPixelFormatP_8,
        EUidPixelFormatP_4,
        EUidPixelFormatL_4,
        EUidPixelFormatL_2,
        EUidPixelFormatP_2,
        EUidPixelFormatL_1
    };

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0103
@SYMTestCaseDesc		Retrieve stream attributes using SymbianStreamGetHeader()
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify native stream object attributes can be retrieved.
@SYMTestActions	
		Create a native stream based on a supported image/pixel format
		Retrieve all of the stream attributes
		Retreive none of the stream attributes
@SYMTestExpectedResults
		The retrieved attributes should match the parameters used to create the surface stream
		Retrieving no attributes should not cause a crash
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0103L()
	{
	// Native stream attributes
    TSize surfaceSize(TSize(100,50));
    khronos_int32_t width = surfaceSize.iWidth;
    khronos_int32_t height = surfaceSize.iHeight;
    khronos_int32_t numBuffers = 2;
    
    for (TInt x=0; x<sizeof(supportedFormats)/sizeof(supportedFormats[0]); x++)
        {
        INFO_PRINTF2(_L("Pixel format %x"),supportedFormats[x]);
        khronos_int32_t streamPixelSize = BytesPerPixel(supportedFormats[x].symbianPixelFormat);
        
        OWF_IMAGE_FORMAT pixelFormat =
            {
            OWF_IMAGE_NOT_SUPPORTED,
            EFalse,
            EFalse,
            2
            };  
        
        SymbianStreamType ns=helperCreateImageStream(width,
                                                     height,
                                                     &pixelFormat,
                                                     numBuffers,
                                                     supportedFormats[x].symbianPixelFormat);      //will ignore pixelFormat
        ASSERT_TRUE(ns);
        
        // Store the retrieved attributes
        TInt32 attWidth = 0;
        TInt32 attHeight = 0;
        TInt32 attStreamStride = 0;
        TUidPixelFormat attStreamFormat = EUidPixelFormatUnknown;
        TInt32 attStreamPixelSize = 0;
        
        SymbianStreamGetHeader(ns, &attWidth, &attHeight, &attStreamStride, &attStreamFormat, &attStreamPixelSize);		
            
        ASSERT_EQUALS(attWidth, width);
        ASSERT_EQUALS(attHeight, height);
        ASSERT_EQUALS(attStreamFormat, supportedFormats[x].symbianPixelFormat);
        if (BytesPerPixel(supportedFormats[x].symbianPixelFormat) > 0)
            {
            ASSERT_EQUALS(attStreamStride, (streamPixelSize * width));
            }
        else
            {
            ASSERT_EQUALS(attStreamStride, (width-(streamPixelSize+1)) / (-streamPixelSize));
            }
        ASSERT_EQUALS(attStreamPixelSize, (TInt32)BytesPerPixel(supportedFormats[x].symbianPixelFormat));	
            
        SymbianStreamGetHeader(ns, NULL, NULL, NULL, NULL, NULL);
        
        SymbianStreamRemoveReference(ns);
        }
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0104
@SYMTestCaseDesc		Acquire write/read buffers and retrieve the buffer ptr (single threaded test).
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose         Verify owfNativeStreamAcquireWriteBuffer(), owfNativeStreamReleaseWriteBuffer(),
                        owfNativeStreamAcquireReadBuffer() and owfNativeStreamReleaseReadBuffer() methods
						work as expected.
						
@SYMTestActions	
		Create a native stream 
		For each buffer:
		- Acquire the write buffer (wB)
		- Get the write buffer ptr (pWB)
		- Release the write buffer
		- Acquire the read buffer (rB)
		- Get the read buffer ptr (pRB)
		- Release the read buffer	
		Repeat for each buffer. Finally:
		Acquire the write buffer
		Release the write buffer	
@SYMTestExpectedResults
		For each buffer of the native stream, check:
		- The read buffer (rB) should be the same as the write buffer (wB)
		- The read buffer address (pRB) should be the same as the write buffer address (pWB)
		If the number of buffers > 1, check:
		- The write buffer number from the previous acquire write buffer call should not be the same 
		as the write buffer number from the next acquire write buffer call
		The final acquire/release write/read calls should check:
		The write buffer number should acquire the first buffer number
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0104L(TInt aNumBuffers)
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);

	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	
	ASSERT_TRUE(aNumBuffers > 0);
	
	TSize surfaceSize(TSize(100,100));
	khronos_int32_t width = surfaceSize.iWidth;
	khronos_int32_t height = surfaceSize.iHeight;
	
	OWF_IMAGE_FORMAT pixelFormat =
		{
		OWF_IMAGE_ARGB8888,
	    ETrue,
	    ETrue,
	    aNumBuffers
	    };	
	
	SymbianStreamType ns=helperCreateImageStream(width,
													height,
													&pixelFormat,
													aNumBuffers);
	ASSERT_TRUE(ns);
	
	TUint8 *pWriteBuffer = NULL;
	TUint8 *pReadBuffer = NULL;
	TUint8 *pPrevWriteBuffer = pWriteBuffer;
	khronos_int32_t writeBuffer;
	khronos_int32_t readBuffer;
	khronos_int32_t bufferIndexWrite;
	khronos_int32_t bufferIndexRead;
	khronos_int32_t bufferIndexWriteFirst;
	khronos_int32_t bufferIndexWriteFinal;
	khronos_int32_t finalWriteBuffer;
	const TSurfaceId* getId = NULL;
	
	TInt bufferCount = aNumBuffers;
	
	// Loop through the buffers
	for (TInt count=0; count<bufferCount; count++)
		{
		// Acquire the write buffer
		err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer);
		ASSERT_TRUE(err == KErrNone);
		err = SymbianStreamGetBufferId(ns,writeBuffer,&bufferIndexWrite,&getId);
		ASSERT_TRUE(err == KErrNone);
		
		if (count == 0)
		    {
		    bufferIndexWriteFirst = bufferIndexWrite;
		    }
		
		err = SymbianStreamGetBufferPointer(ns,writeBuffer,reinterpret_cast<void**>(&pWriteBuffer));
		ASSERT_TRUE(err == KErrNone);
		ASSERT_NOT_NULL(pWriteBuffer);
		
		err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer);
		ASSERT_TRUE(err == KErrNone);
		
		// Acquire the read buffer
		err = SymbianStreamAcquireReadBuffer(ns,&readBuffer);
		ASSERT_TRUE(err == KErrNone);
		err = SymbianStreamGetBufferId(ns,writeBuffer,&bufferIndexRead,&getId);
		ASSERT_TRUE(err == KErrNone);
		
		err = SymbianStreamGetBufferPointer(ns,readBuffer,reinterpret_cast<void**>(&pReadBuffer));
		ASSERT_TRUE(err == KErrNone);
		ASSERT_NOT_NULL(pReadBuffer);
		
		err = SymbianStreamReleaseReadBuffer(ns,readBuffer);
		ASSERT_TRUE(err == KErrNone);
		
		// Life-cycle checks
		ASSERT_EQUALS(bufferIndexWrite, bufferIndexRead)
		ASSERT_SAME(pWriteBuffer, pReadBuffer);
		
		if (count > 0)
			{
			ASSERT_NOT_SAME(pWriteBuffer, pPrevWriteBuffer);
			}
		
		pPrevWriteBuffer = pWriteBuffer;
		}	

	// The next acquire write/reads should return the first buffer (0).

	err = SymbianStreamAcquireWriteBuffer(ns,&finalWriteBuffer);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,finalWriteBuffer,&bufferIndexWriteFinal,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseWriteBuffer(ns,finalWriteBuffer);
	ASSERT_TRUE(err == KErrNone);
	
	// Final checks
	ASSERT_EQUALS(bufferIndexWriteFinal, bufferIndexWriteFirst);
	
	SymbianStreamRemoveReference(ns);
	
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0105
@SYMTestCaseDesc		Create a native stream and acquire a write buffer which is written to (single threaded test)
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify native stream buffers can be written to and read from.
@SYMTestActions	
		Create a surface and fill it Red
		Create a native stream surface with the same surface properties (size, pixel format etc.)
		Acquire the native stream write buffer
		Fill the native stream surface Red
		Release the write buffer
		Acquire the native stream read buffer
		Compare the pixel data of the surface to the native stream surface
		Fill the Blue
		Compare the pixel data of the surface to the native stream surface
@SYMTestExpectedResults
		The surface pixel data is the same as the native stream surface pixel data
		After the surface pixel data is changed to Blue, the native stream surface should not be the same
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0105L(TInt aNumBuffers)
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);
	
	// Surface properties
	TSize surfaceSize(TSize(100,100));
	
	// Create the comparison surface and fill it Red
	TSurfaceId surface;
	TRAP(err, surface = iUtility->CreateSurfaceL(TSize(surfaceSize.iWidth,surfaceSize.iHeight), 
			EUidPixelFormatXRGB_8888, surfaceSize.iWidth * 4));
	
	TRAP(err, iUtility->FillSurfaceL(surface, 0, KRgbRed));
	
	// Native stream
	khronos_int32_t width = surfaceSize.iWidth;
	khronos_int32_t height = surfaceSize.iHeight;
	
	OWF_IMAGE_FORMAT pixelFormatXRGB888 =
		{
		OWF_IMAGE_XRGB8888,
	    ETrue,
	    EFalse,
	    4
	    };	
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	
	SymbianStreamType ns1=helperCreateImageStream(width,
													height,
													&pixelFormatXRGB888,
													aNumBuffers);
	ASSERT_TRUE(ns1);
	
	
    SymbianStreamBuffer bufferHandle;
    err = SymbianStreamAcquireReadBuffer(ns1,&bufferHandle);
    ASSERT_TRUE(err == KErrNone);
    long bufferIndex;
    const TSurfaceId* nSurface = NULL;
    err = SymbianStreamGetBufferId(ns1,bufferHandle,&bufferIndex,&nSurface);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns1,bufferHandle);
    ASSERT_TRUE(err == KErrNone);
	ASSERT_NOT_NULL(nSurface);
	
	TSurfaceId* nsSurface = const_cast<TSurfaceId*>(nSurface);
	
	// Acquire write buffer. With 3 buffers we should return buffer 1
	khronos_int32_t writeBuffer1;
	err = SymbianStreamAcquireWriteBuffer(ns1,&writeBuffer1);
	ASSERT_TRUE(err == KErrNone);
	
	TUint8 *pWriteBuffer1 = NULL;
	err = SymbianStreamGetBufferPointer(ns1,writeBuffer1,reinterpret_cast<void**>(&pWriteBuffer1));
	ASSERT_TRUE(err == KErrNone);
	ASSERT_NOT_NULL(pWriteBuffer1);
	
	TRAP(err, iUtility->FillNativeStreamSurfaceL(*nsSurface, pWriteBuffer1, KRgbRed));
	
	err = SymbianStreamReleaseWriteBuffer(ns1, writeBuffer1);
	ASSERT_TRUE(err == KErrNone);
	
	// Now we should compare to see if the pixels are the same
	khronos_int32_t readBuffer1;
	err = SymbianStreamAcquireReadBuffer(ns1,&readBuffer1);
	ASSERT_TRUE(err == KErrNone);
	
	TUint8 *pReadBuffer1 = NULL;
	err = SymbianStreamGetBufferPointer(ns1,readBuffer1,reinterpret_cast<void**>(&pReadBuffer1));
	ASSERT_TRUE(err == KErrNone);
	ASSERT_NOT_NULL(pReadBuffer1);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	ASSERT_TRUE(iUtility->CompareSurfacesL(surface, 0, *nsSurface, pReadBuffer1));	
	
	// Finally, change the surface to blue. The pixels should now be different
	TRAP(err, iUtility->FillSurfaceL(surface, 0, KRgbBlue));
	ASSERT_FALSE(iUtility->CompareSurfacesL(surface, 0, *nsSurface, pReadBuffer1));	
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	err = SymbianStreamReleaseReadBuffer(ns1,readBuffer1);
	ASSERT_TRUE(err == KErrNone);
	SymbianStreamRemoveReference(ns1);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0106
@SYMTestCaseDesc		Negative test - Attempt to acquire a write buffer when the write buffer has not been released (single threaded test)
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify an invalid handle is returned to the calling thread if the write buffer has not been released
@SYMTestActions	
		Create a native stream surface with 1 buffer
		Acquire the native stream write buffer
		Acquire the native stream write buffer again
@SYMTestExpectedResults
		The 2nd acquire write buffer call should return OWF_INVALID_HANDLE if not single buffered.
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0106L(TInt aNumBuffers)
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);

	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	
	// Create the first stream (ARGB_8888_PRE)
	TSize surfaceSize(TSize(100,100));
	khronos_int32_t width = surfaceSize.iWidth;
	khronos_int32_t height = surfaceSize.iHeight;
	
	OWF_IMAGE_FORMAT pixelFormatARGB888Pre =
		{
		OWF_IMAGE_ARGB8888,
	    ETrue,
	    EFalse,
	    4
	    };	
	
	SymbianStreamType ns1=helperCreateImageStream(width,
													height,
													&pixelFormatARGB888Pre,
													aNumBuffers);
	ASSERT_TRUE(ns1);
	
	// Acquire write buffer. With just 1 buffer we should return buffer 0
	khronos_int32_t writeBuffer1;
	err = SymbianStreamAcquireWriteBuffer(ns1,&writeBuffer1);
	ASSERT_TRUE(err != KErrBadHandle);
	
	// Try and acquire the write buffer again before we have released it
	khronos_int32_t writeBuffer2;
	err = SymbianStreamAcquireWriteBuffer(ns1,&writeBuffer2);
	ASSERT_TRUE(err != KErrBadHandle);
	if (aNumBuffers == 1)
	    {
	    ASSERT_EQUALS(writeBuffer2, writeBuffer1); 	    
	    }
	else
	    {
	    ASSERT_EQUALS(writeBuffer2, (khronos_int32_t)0); 	    
	    }
	
	err = SymbianStreamReleaseWriteBuffer(ns1,writeBuffer1);
	ASSERT_TRUE(err != KErrBadHandle);
	
	err = SymbianStreamAcquireWriteBuffer(ns1, &writeBuffer2);
	ASSERT_TRUE(err != KErrBadHandle);
	TUint8 *pWriteBuffer2 = NULL;
	err = SymbianStreamGetBufferPointer(ns1,writeBuffer2,reinterpret_cast<void**>(&pWriteBuffer2));
	ASSERT_TRUE(err != KErrBadHandle);
	ASSERT_NOT_NULL(pWriteBuffer2);
	
	SymbianStreamRemoveReference(ns1);
	
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0107
@SYMTestCaseDesc		Attempt to acquire a write buffer when the write buffer has not been released (multi threaded tests)
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify an invalid handle is returned to the calling thread if the write buffer has not been released
@SYMTestActions	
		Create a shared native stream surface to be used by multiple threads
		Thread 1 acquires the shared native stream and acquires the write buffer
		Thread 2 acquires the shared native stream and attempts to acquire the write buffer
		Thread 3 acquires the shared native stream and attempts to acquire the write buffer
		Thread 1 releases the write buffer
		Thread 2 acquires the write buffer
		Thread 2 releases the write buffer
@SYMTestExpectedResults
		OWF_INVALID_HANDLE returned when Thread 2 and Thread 3 attempt to acquire the write buffer
		when Thread 1 has already acquired it
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0107_1L()
	{
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
        
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	INFO_PRINTF2(_L("** GRAPHICS_OPENWFC_NATIVESTREAM_0107 - %i buffers **"), gVarInstance.Buffers());
	INFO_PRINTF1(_L("Thread 1 - start"));
	TSurfaceId surface = gVarInstance.SurfaceID();
	SymbianStreamType ns;
	err = SymbianStreamAcquire(&surface,&ns);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns);
	
	khronos_int32_t writeBuffer1;
	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer1);
	ASSERT_TRUE(err != KErrBadHandle);
	
	khronos_int32_t bufferIndex;
	const TSurfaceId* getId = NULL;
	err = SymbianStreamGetBufferId(ns,writeBuffer1,&bufferIndex,&getId);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_EQUALS(*getId, surface);
	ASSERT_EQUALS(bufferIndex, (khronos_int32_t)1);
	INFO_PRINTF2(_L("Thread 1 - Write buffer %i acquired"), bufferIndex);
	
	gSemaphore.Signal(2); // Thread 2 and 3 ready to run
	
	gSemaphore.Wait();
	err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer1);
	ASSERT_TRUE(err == KErrNone);
	SymbianStreamRemoveReference(ns);
	INFO_PRINTF2(_L("Thread 1 - Write buffer %i released"), bufferIndex);

	gSemaphore.Signal();	
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one    
	}

void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0107_2L()
	{
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
        
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	gSemaphore.Wait(); // Semaphore count becomes -1
	INFO_PRINTF1(_L("Thread 2 - Start"));
	TSurfaceId surface = gVarInstance.SurfaceID();
	SymbianStreamType ns;
	err = SymbianStreamAcquire(&surface,&ns);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns);

	khronos_int32_t writeBuffer1;
	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer1);
	ASSERT_TRUE(err != KErrBadHandle);
	INFO_PRINTF1(_L("Thread 2 - Attempt to acquire the write buffer"));
	ASSERT_FALSE(writeBuffer1);
	INFO_PRINTF1(_L("Thread 2 - Write buffer already in use by Thread 1!"));
    
	gSemaphore.Signal();
	
	gSemaphore.Wait();
	
	khronos_int32_t writeBuffer2;
	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer2);
	ASSERT_TRUE(err != KErrBadHandle);
	khronos_int32_t bufferIndex;
	const TSurfaceId* getId = NULL;
	err = SymbianStreamGetBufferId(ns,writeBuffer2,&bufferIndex,&getId);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_FALSE(bufferIndex);
	INFO_PRINTF2(_L("Thread 2 - Write buffer %i acquired"), bufferIndex);
	
	err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer2);
	ASSERT_TRUE(err == KErrNone);
	SymbianStreamRemoveReference(ns);
	INFO_PRINTF2(_L("Thread 2 - Write buffer %i released"), bufferIndex);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one    
	}

void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0107_3L()
	{
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
        
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	gSemaphore.Wait(); // Semaphore count is -2
	INFO_PRINTF1(_L("Thread 3 - start"));
	TSurfaceId surface = gVarInstance.SurfaceID();
	SymbianStreamType ns;
	err = SymbianStreamAcquire(&surface,&ns);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns);

 	khronos_int32_t writeBuffer1;
 	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer1);
 	ASSERT_TRUE(err != KErrBadHandle);
 	INFO_PRINTF1(_L("Thread 3 - Attempt to acquire the write buffer"));
	ASSERT_FALSE(writeBuffer1);
	INFO_PRINTF1(_L("Thread 3 - Write buffer already in use by Thread 1!"));

    SymbianStreamRemoveReference(ns);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one    
	}


/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0108
@SYMTestCaseDesc		Content updates on valid and invalid buffers
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify native streams can handle content updates for valid and invalid buffer numbers
@SYMTestActions	
		Create a surface with 4 buffers,
		Create a stream from this surface,
		Add an observer to listen out for content updates to the stream,
		Set valid and invalid buffer numbers to be used in content updates
@SYMTestExpectedResults
		For valid buffers, the read buffers should be set correctly and observer 
		callback method is called.
 **/		
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0108L(TInt aBuffers)
	{
	aBuffers = 3;
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);

	iScreenNo = 0;
	TInt localNumber = 0;
	
	TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, aBuffers);
	ASSERT_FALSE(surface.IsNull());
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	SymbianStreamType ns;
	err = SymbianStreamAcquire(&surface,&ns);
    ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns);
	
	err = SymbianStreamAddObserver(ns, TestUpdateCallback, &localNumber);
	ASSERT_TRUE(err == KErrNone);
	khronos_int32_t bufferIndex = 0; 
	khronos_int32_t buffersIndex = 0;
	khronos_int32_t lastValidBufnum = 0;
	
	const TSurfaceId* getId = NULL;
    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
	// Valid inputs
	for (TInt i = 0; i < aBuffers; ++i)
		{
		buffersIndex = (aBuffers + i) % aBuffers;

 		
        updateProxy->ContentUpdated(surface,                    //aSurface
                                    buffersIndex,               //aBuffer
                                    NULL,                       //aRegion
                                    NULL,                       //aStatusConsumed
                                    NULL,                       //aStatusDisplayed
                                    NULL,                       //aTimeStamp
                                    NULL,                       //aStatusDispXTimes
                                    NULL                        //aDisplayedXTimes
                                    );	
		
		// The test can pass without a delay when running locally, but fails on ONB. Insert a delay here temporarily to see if it makes any difference
        User::After(10000); //10 ms delay
		khronos_int32_t readBuffer;
		err = SymbianStreamAcquireReadBuffer(ns,&readBuffer);
		ASSERT_TRUE(err == KErrNone);
		err = SymbianStreamGetBufferId(ns,readBuffer,&bufferIndex,&getId);
		ASSERT_TRUE(err == KErrNone);
		err = SymbianStreamReleaseReadBuffer(ns, readBuffer);
		ASSERT_TRUE(err == KErrNone);
		ASSERT_TRUE(bufferIndex == buffersIndex);
		
        // The test can pass without a delay when running locally, but fails on ONB. Insert a delay here temporarily to see if it makes any difference
        User::After(10000); //10 ms delay
		ASSERT_EQUALS((i+1),localNumber);
		}
	
	// Reset number for negative tests
	localNumber = 0;
	lastValidBufnum = buffersIndex;
	
	// Invalid inputs
	TInt invalidBuffersIndex[] = {-1, -256, aBuffers+29, 10000, -10000, aBuffers};
	TInt size = sizeof(invalidBuffersIndex) / sizeof(invalidBuffersIndex[0]);
	
	for (TInt i = 0; i < size; ++i)
		{
		updateProxy->ContentUpdated(surface,                    //aSurface
                                    invalidBuffersIndex[i],     //aBuffer
                                    NULL,                       //aRegion
                                    NULL,                       //aStatusConsumed
                                    NULL,                       //aStatusDisplayed
                                    NULL,                       //aTimeStamp
                                    NULL,                       //aStatusDispXTimes
                                    NULL                        //aDisplayedXTimes
                                    );
        
		User::After(10000); //10 ms delay
		khronos_int32_t readBuffer;
		err = SymbianStreamAcquireReadBuffer(ns,&readBuffer);
		ASSERT_TRUE(err == KErrNone);
		err = SymbianStreamGetBufferId(ns,readBuffer,&bufferIndex,&getId);
		ASSERT_TRUE(err == KErrNone);
		err = SymbianStreamReleaseReadBuffer(ns, readBuffer);
		ASSERT_TRUE(err == KErrNone);
		ASSERT_TRUE(bufferIndex == lastValidBufnum);
		ASSERT_EQUALS(0,localNumber);
		}
	SymbianStreamRemoveReference(ns);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0109
@SYMTestCaseDesc		Test multiple acquire/release read/write buffer calls
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			Verify the buffer index is correctly set
@SYMTestActions	
		Create a surface with 4 buffers,
		Call acquire/release read/write buffer functions and to Submit an update to the surface
@SYMTestExpectedResults
		Verify that acquire/release read/write buffer functions honour SUS update
		Verify that the update callback function is called when the native stream is updated
 **/		
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0109L()
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);
	
	iScreenNo = 0;
	TInt localNumber = 0;
	TInt count = 0;
	khronos_int32_t bufferIndexRead;
	khronos_int32_t bufferIndexWrite;
	
	TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
	ASSERT_FALSE(surface.IsNull());
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	SymbianStreamType ns;
	err = SymbianStreamAcquire(&surface,&ns);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns);
	
	err = SymbianStreamAddObserver(ns, TestComposedCallback, &localNumber);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamAddObserver(ns, TestUpdateCallback, &localNumber);
	ASSERT_TRUE(err == KErrNone);
	
	khronos_int32_t readBuffer1;
	err = SymbianStreamAcquireReadBuffer(ns,&readBuffer1);
	ASSERT_TRUE(err == KErrNone);
	const TSurfaceId* getId = NULL;
	err = SymbianStreamGetBufferId(ns,readBuffer1,&bufferIndexRead,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseReadBuffer(ns, readBuffer1);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexRead == 0);
	
	khronos_int32_t writeBuffer1;
	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer1);
	ASSERT_TRUE(err == KErrNone);
	
	khronos_int32_t readBuffer2;
	err = SymbianStreamAcquireReadBuffer(ns,&readBuffer2);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,readBuffer2,&bufferIndexRead,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseReadBuffer(ns, readBuffer2);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexRead == 0);
	err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer1);
	ASSERT_TRUE(err == KErrNone);
	
	ASSERT_EQUALS((++count),localNumber);
	
	khronos_int32_t readBuffer3;
	err = SymbianStreamAcquireReadBuffer(ns,&readBuffer3);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,readBuffer3,&bufferIndexRead,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseReadBuffer(ns, readBuffer3);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexRead == 1);
	
	khronos_int32_t writeBuffer2;
	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer2);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,writeBuffer2,&bufferIndexWrite,&getId);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE((bufferIndexRead + 1) == bufferIndexWrite);
	
	khronos_int32_t readBuffer4;
	err = SymbianStreamAcquireReadBuffer(ns,&readBuffer4);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,readBuffer4,&bufferIndexRead,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseReadBuffer(ns, readBuffer4);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexRead == (bufferIndexWrite - 1));

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );  	
	
	ASSERT_EQUALS((++count),localNumber);
	
	khronos_int32_t readBuffer5;
	err = SymbianStreamAcquireReadBuffer(ns,&readBuffer5);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,readBuffer5,&bufferIndexRead,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseReadBuffer(ns, readBuffer5);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexRead == 0);
	err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer2);
	ASSERT_TRUE(err == KErrNone);
	
	ASSERT_EQUALS((++count),localNumber);
	
	khronos_int32_t readBuffer6;
	err = SymbianStreamAcquireReadBuffer(ns,&readBuffer6);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,readBuffer6,&bufferIndexRead,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseReadBuffer(ns, readBuffer6);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexRead == 0);
	
	khronos_int32_t writeBuffer3;
	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer3);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,writeBuffer3,&bufferIndexWrite,&getId);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexWrite == 1);
	err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer3);
	ASSERT_TRUE(err == KErrNone);
	
	khronos_int32_t readBuffer7;
	err = SymbianStreamAcquireReadBuffer(ns,&readBuffer7);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamGetBufferId(ns,readBuffer7,&bufferIndexRead,&getId);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseReadBuffer(ns, readBuffer7);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(bufferIndexRead == 1);
	
	SymbianStreamRemoveReference(ns);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID			GFX_OPENWFC_NATIVESTREAM_0110
@SYMTestCaseDesc		Verify that the update observer callback function is called after a
                        acquireWriteBuffer/releaseWriteBuffer call.
@SYMREQ					
@SYMPREQ				PREQ2400
@SYMTestType			CT
@SYMTestPriority		
@SYMTestPurpose			
@SYMTestActions	
		Create a surface with 1 buffer
		Add an observer to listen out for content updates to the stream
		Call acquire/release write buffer functions
@SYMTestExpectedResults
		Verify that the release write buffer function notifies any observers listening and the 
		observer callback function is called.
 **/		
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0110L()
	{
	RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
	TRAPD(err, GrowCleanupStackL());
	ASSERT_TRUE(err == KErrNone);
	
	TInt localNumber = 0;
	
	TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 1);
	ASSERT_FALSE(surface.IsNull());
	
	CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
	SymbianStreamType ns;
	err = SymbianStreamAcquire(&surface,&ns);
	ASSERT_TRUE(err == KErrNone);
	ASSERT_TRUE(ns);
	
	err = SymbianStreamAddObserver(ns, TestComposedCallback, &localNumber);
	ASSERT_TRUE(err == KErrNone);
	
	khronos_int32_t writeBuffer1;
	err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer1);
	ASSERT_TRUE(err == KErrNone);
	err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer1);
	ASSERT_TRUE(err == KErrNone);
	
	if (localNumber == 0)
		{
		User::After(1000000);
		}
	
	ASSERT_TRUE(localNumber == 1);    
	SymbianStreamRemoveReference(ns);
	CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0111
@SYMTestCaseDesc        Tests various cases including some negative cases for Native Stream callbacks
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         
@SYMTestActions 
        Create a surface with 1 buffer
        Add an observer N times
        Remove the same observer M (M <= N) times
        Register Null observer
        Unregister something that was never registered
@SYMTestExpectedResults
        Verify that the observer is called (N - M) times
        Verify that error case behaviour is correct
 **/        
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0111L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;
    
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 1);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    SymbianStreamType ns;
    err = SymbianStreamAcquire(&surface,&ns);
    ASSERT_TRUE(err == KErrNone);    
    ASSERT_TRUE(ns);
    
    TInt localNumber = 0;
    err = SymbianStreamAddObserver(ns, NULL, &localNumber);
    ASSERT_TRUE(err == KErrBadHandle);
    err = SymbianStreamRemoveObserver(ns, &localNumber, ESOWF_EventComposed);
    ASSERT_TRUE(err == KErrNotFound);
    err = SymbianStreamRemoveObserver(ns, NULL, ESOWF_EventComposed);
    ASSERT_TRUE(err == KErrNotFound);
    err = SymbianStreamAddObserver(ns, TestUpdateCallback, &localNumber);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamRemoveObserver(ns, &localNumber, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamRemoveObserver(ns, &localNumber, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNotFound);
    
#define N_TIMES 7
#define M_TIMES 3
    
    localNumber = 0;
    TInt k = N_TIMES;
    while (k--)
        {
        err = SymbianStreamAddObserver(ns, TestUpdateCallback, &localNumber);
        ASSERT_TRUE(err != KErrBadHandle);
        }
    k = M_TIMES;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );  
    
    User::After(10000);
    ASSERT_TRUE(localNumber == 1);
    
    while (k--)
        {
        err = SymbianStreamRemoveObserver(ns, &localNumber, ESOWF_EventUpdated);
        ASSERT_TRUE(err != KErrBadHandle);
        }
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );      
    
    ASSERT_TRUE(localNumber == 1);
    
    SymbianStreamRemoveReference(ns);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0112
@SYMTestCaseDesc        Tests multithreaded cases for Native Stream callbacks
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         
@SYMTestActions 
        Create 3 threads and call add, remove, notify observers from
        respective threads multiple times
@SYMTestExpectedResults
        Verify that the observers work correctly in multithreaded environment
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0112_1L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
        
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_1L: Thread 1 start, Register Observer"));
    TSurfaceId surface = gVarInstance.SurfaceID();
    SymbianStreamType ns;
    err = SymbianStreamAcquire(&surface,&ns);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(ns);
    
    gVarInstance.iMultithreadCounter = 0;

    TTimeIntervalMicroSeconds32 delay(32000), zeroDelay(0);
    while (delay >= zeroDelay)
        {
        INFO_PRINTF2(_L("Thread 1 is going to add another observer after %i microsecond"), delay.Int());
        User::AfterHighRes(delay);
        SymbianStreamAddObserver(ns, TestUpdateCallback, &gVarInstance.iMultithreadCounter);
        delay = delay.Int() - 43;
        }
    TInt count = 50;
    while (count--)
        {
        INFO_PRINTF1(_L("Thread 1 is going to add another observer without delay"));
        SymbianStreamAddObserver(ns, TestUpdateCallback, &gVarInstance.iMultithreadCounter);
        }
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_1L: Thread 1 exits"));
    
    SymbianStreamRemoveReference(ns);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0112_2L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);

    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_2L: Thread 2 start, Unregister Observer"));
    TSurfaceId surface = gVarInstance.SurfaceID();
    SymbianStreamType ns;
    err = SymbianStreamAcquire(&surface,&ns);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(ns);
    
    TTimeIntervalMicroSeconds32 delay(32000), zeroDelay(0);
    while (delay > zeroDelay)
        {
        INFO_PRINTF2(_L("Thread 2 is going to remove one observer after %i microsecond"), delay.Int());
        User::AfterHighRes(delay);
        SymbianStreamRemoveObserver(ns, &gVarInstance.iMultithreadCounter, ESOWF_EventUpdated);
        delay = delay.Int() - 49;
        }
    TInt count = 50;
    while (count--)
        {
        INFO_PRINTF1(_L("Thread 2 is going to remove one observer without delay"));
        SymbianStreamRemoveObserver(ns, &gVarInstance.iMultithreadCounter, ESOWF_EventUpdated);
        }
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_2L: Thread 2 exits"));    
    SymbianStreamRemoveReference(ns);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0112_3L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);

    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_2L: Thread 3 start, Notify Observer"));
    TSurfaceId surface = gVarInstance.SurfaceID();
    SymbianStreamType ns;
    err = SymbianStreamAcquire(&surface,&ns);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(ns);
    
    khronos_int32_t bufferIndex;
    TTimeIntervalMicroSeconds32 delay(32000), zeroDelay(0);
    while (delay > zeroDelay)
        {
        khronos_int32_t writeBuffer;
        err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer);
        ASSERT_TRUE(err == KErrNone);
        const TSurfaceId* getId = NULL;
        err = SymbianStreamGetBufferId(ns,writeBuffer,&bufferIndex,&getId);
        ASSERT_TRUE(err == KErrNone);
        INFO_PRINTF2(_L("Thread 3 - Write buffer %i acquired"), bufferIndex);
        
        INFO_PRINTF2(_L("Thread 3 going to send notification after %i second"), delay.Int());
        User::AfterHighRes(delay);
        delay = delay.Int() - 58;
        err = SymbianStreamReleaseWriteBuffer(ns, writeBuffer);
        ASSERT_TRUE(err == KErrNone);
        }
    TInt count = 50;
    while (count--)
        {
        khronos_int32_t writeBuffer;
        err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer);
        ASSERT_TRUE(err == KErrNone);
        const TSurfaceId* getId = NULL;
        err = SymbianStreamGetBufferId(ns,writeBuffer,&bufferIndex,&getId);
        ASSERT_TRUE(err == KErrNone);
        INFO_PRINTF2(_L("Thread 3 - Write buffer %i acquired"), bufferIndex);
        
        INFO_PRINTF1(_L("Thread 3 going to send notification without delay"));
        err = SymbianStreamReleaseWriteBuffer(ns, writeBuffer);
        ASSERT_TRUE(err == KErrNone);
        }
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_2L: Thread 3 exits"));
    
    SymbianStreamRemoveReference(ns);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
Submit updates to the native stream whilst observers are being added and removed. 
Do not run in parallel with 0112_3L 
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0112_4L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);

    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_2L: Thread 3 start, Notify Observer"));
    
    TTimeIntervalMicroSeconds32 delay(32000), zeroDelay(0);
    
    iScreenNo = 0;
    TRequestStatus displayedStatus, availableStatus, displayedXStatus;
    TUint32 timeStamp = 0;
    TInt bufferNo = 0;
    TInt numBuffers = 2;
    TInt displayedX = 5;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    while (delay > zeroDelay)
        {        
        INFO_PRINTF2(_L("Thread 0112_4 submitting update after %i second"), delay.Int());
        User::AfterHighRes(delay);
        delay = delay.Int() - 58;                
        
        updateProxy->ContentUpdated(gVarInstance.SurfaceID(),                    //aSurface
                                    bufferNo,                   //aBuffer
                                    NULL,                       //aRegion
                                    &availableStatus,           //aStatusConsumed
                                    &displayedStatus,           //aStatusDisplayed
                                    &timeStamp,                 //aTimeStamp
                                    &displayedXStatus,          //aStatusDispXTimes
                                    &displayedX                 //aDisplayedXTimes
                                    );                
        
        bufferNo = (bufferNo + 1) % numBuffers;        
        }   
    
    TInt count = 50;
    while (count--)
        {
        INFO_PRINTF1(_L("Thread 0112_4 Set notifications"));
        INFO_PRINTF1(_L("Thread 0112_4 submitting update without delay"));

        updateProxy->ContentUpdated(gVarInstance.SurfaceID(),   //aSurface
                                    bufferNo,                   //aBuffer
                                    NULL,                       //aRegion
                                    &availableStatus,           //aStatusConsumed
                                    &displayedStatus,           //aStatusDisplayed
                                    &timeStamp,                 //aTimeStamp
                                    &displayedXStatus,          //aStatusDispXTimes
                                    &displayedX                 //aDisplayedXTimes
                                    );         
        
        bufferNo = (bufferNo + 1) % numBuffers;
        }    
    INFO_PRINTF1(_L("GRAPHICS_OPENWFC_NATIVESTREAM_0112_4L: Thread exits"));
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0113
@SYMTestCaseDesc        Test displayed notification
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Tests an end to end displayed notification.
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for displayed notifications
     5.  Submit an update to the surface
     6.  The observer function should be invoked and verifies that stream and event parameters.
     7.  The tester simulates the end of a composition by firing owfNativeStreanProcessNotifications
     8.  Wait on the displayed request status. 
     9.  Remove the source stream updated observer
     10. Destroy the native stream.
     
     The test is then repeated but the compositor claims the native stream is not visible.
        
@SYMTestExpectedResults
    No errors, displayed status completed with KErrNone.
        
 **/

void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0113L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iSourceStreamUpdatedCalled = 0;
    iImmediateAvailable = EFalse;
    iImmediateVisible = SYM_CONTENT_NOT_VISIBLE;
    iContextUpdatedFlags = 0;
    iScreenNo = 0;    
    
    for (TInt i = 0; i < 2; ++i)
        {
        TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
        ASSERT_FALSE(surface.IsNull());
        CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
        
        CTestNativeStream::iTester = this;
        TBool visible = (i == 0);
        
        RHeap* threadHeap3 = &User::Heap();
        err = SymbianStreamAcquire(&surface,&iNs);
        ASSERT_TRUE(err == KErrNone);
        ASSERT_TRUE(iNs);
        
        iExpectedSourceStreamUpdatedEventMask = 0;
        err = SymbianStreamAddObserver(iNs, SourceStreamUpdatedCallback, this);
        ASSERT_TRUE(err == KErrNone);
        
        TRequestStatus statusDisplayed;
        TUint32 displayedTime;    

        CExtensionContainer* updateExtension = NULL;
        err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
        ASSERT_TRUE(err == KErrNone);
        MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
        ASSERT_NOT_NULL(updateProxy);
        
        iSourceStreamUpdatedCalled = 0;
        updateProxy->ContentUpdated(surface,                    //aSurface
                                    0,                          //aBuffer
                                    NULL,                       //aRegion
                                    NULL,                       //aStatusConsumed
                                    &statusDisplayed,           //aStatusDisplayed
                                    &displayedTime,             //aTimeStamp
                                    NULL,                       //aStatusDispXTimes
                                    NULL                        //aDisplayedXTimes
                                    );
        
        ASSERT_TRUE(iSourceStreamUpdatedCalled == 1);
        err = SymbianStreamRemoveObserver(iNs, this, ESOWF_EventUpdated);
        ASSERT_TRUE(err == KErrNone);
    
        iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayed;
        err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
        ASSERT_TRUE(err == KErrNone);
    
        updateProxy->ContentUpdated(surface,                    //aSurface
                                    0,                          //aBuffer
                                    NULL,                       //aRegion
                                    NULL,                       //aStatusConsumed
                                    &statusDisplayed,           //aStatusDisplayed
                                    &displayedTime,             //aTimeStamp
                                    NULL,                       //aStatusDispXTimes
                                    NULL                        //aDisplayedXTimes
                                    );
        
        ASSERT_TRUE(iSourceStreamUpdatedCalled);
        
        // Pretend that a composition has occured
        ++iStreamUpdatedSerialNumber; 
        if (visible)
            {
            khronos_int32_t newNotificationsMask = 0;
            SymbianStreamProcessNotifications(iNs, 
                                                ESOWF_EventDisplayed,
                                                iScreenNo,
                                                iStreamUpdatedSerialNumber, 
                                                &newNotificationsMask);
    
            // No updates during composition so newNotificationMask should still be zero 
            ASSERT_TRUE(newNotificationsMask == 0);
            }        
        // Simulate multiple sources
        SymbianStreamCheckVisible(iNs, ESOWF_EventDisplayed, iScreenNo, iStreamUpdatedSerialNumber); 
        SymbianStreamCheckVisible(iNs, ESOWF_EventDisplayed, iScreenNo, iStreamUpdatedSerialNumber);
        
        // Make sure displayed event was completed
        User::WaitForRequest(statusDisplayed);
        if (visible)
            {
            ASSERT_EQUALS(statusDisplayed.Int(), KErrNone);
            }
        else
            {
            ASSERT_EQUALS(statusDisplayed.Int(), KErrNotVisible);
            }
        
        err = SymbianStreamRemoveObserver(iNs, this, ESOWF_EventUpdated);
        ASSERT_TRUE(err == KErrNone);
        SymbianStreamRemoveReference(iNs);
        CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one   
        iUtility->DestroySurface(surface);
        }
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0114
@SYMTestCaseDesc        Test available notification
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Tests an end to end available notification.
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for displayed notifications
     5.  Submit an update to the surface
     6.  The observer function should be invoked and verifies that stream and event parameters.
     7.  The tester simulates the end of a composition by firing owfNativeStreanProcessNotifications
     8.  Verify that available notification has not been sent.
     9.  Send another display update to change the read buffer to buffer 1
     10. Verify that the source-stream updated callback is invoked.
     11. The tester simulates the end of a composition by firing owfNativeStreanProcessNotifications
     12. Wait for available status to be completed
     13. Remove the source stream updated observer
     14. Destroy the native stream.
     
     The test is then repeated but the compositor claims the native stream is not visible.
        
@SYMTestExpectedResults
    No errors, available status completed with KErrNone.
        
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0114L()
    {  
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    // during compositio or first time after a commit
    iImmediateAvailable = EFalse;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    
    for (TInt i = 0; i < 2; ++i)
        {
        TBool visible = (i == 0);
        TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
        ASSERT_FALSE(surface.IsNull());
        CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
        
        CTestNativeStream::iTester = this;
        
        err = SymbianStreamAcquire(&surface,&iNs);
        ASSERT_TRUE(err == KErrNone);
        ASSERT_TRUE(iNs);
        
        err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
        ASSERT_TRUE(err == KErrNone);
        
        TRequestStatus statusAvailable;
        iSourceStreamUpdatedCalled = 0;

        CExtensionContainer* updateExtension = NULL;
        err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
        ASSERT_TRUE(err == KErrNone);
        MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
        ASSERT_NOT_NULL(updateProxy);
        
        ASSERT_TRUE(iSourceStreamUpdatedCalled == 0);
        
        iExpectedSourceStreamUpdatedEventMask = 0;
        //we are during a composition
        ++iStreamUpdatedSerialNumber;
        updateProxy->ContentUpdated(surface,                    //aSurface
                                    0,                          //aBuffer
                                    NULL,                       //aRegion
                                    &statusAvailable,           //aStatusConsumed
                                    NULL,                       //aStatusDisplayed
                                    NULL,                       //aTimeStamp
                                    NULL,                       //aStatusDispXTimes
                                    NULL                        //aDisplayedXTimes
                                    );
        
        // Verify that the context's callback is invoked when SubmitUpdate is called.
        ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
        
        // Available should only be compled when submit update is called with a different buffer no.
        // if the stream is multi-buffered.
        ASSERT_EQUALS(statusAvailable.Int(), KRequestPending);
        
        // Pretend that a composition has occured
        khronos_int32_t newNotificationsMask = 0;
        if (visible)
            {            
            SymbianStreamProcessNotifications(iNs, 
                                              0, 
                                              iScreenNo,
                                              iStreamUpdatedSerialNumber,
                                              &newNotificationsMask);
    
            // No updates during composition so newNotificationMask should still be zero 
            ASSERT_TRUE(newNotificationsMask == 0);
            SymbianStreamProcessNotifications(iNs, 
                                              0, 
                                              iScreenNo,
                                              iStreamUpdatedSerialNumber,
                                              &newNotificationsMask);
    
            // No updates during composition so newNotificationMask should still be zero 
            ASSERT_TRUE(newNotificationsMask == 0);
            }
        SymbianStreamCheckVisible(iNs, ESOWF_EventAvailable, iScreenNo, iStreamUpdatedSerialNumber); 
        SymbianStreamCheckVisible(iNs, ESOWF_EventAvailable, iScreenNo, iStreamUpdatedSerialNumber);        
                
        // Available for buffer zero should not be completed yet
        ASSERT_EQUALS(statusAvailable.Int(), KRequestPending);
        
        // Update and switch to buffer 1
        iExpectedSourceStreamUpdatedEventMask = ESOWF_EventAvailable;
        //we are during a composition
        ++iStreamUpdatedSerialNumber;
        updateProxy->ContentUpdated(surface,                    //aSurface
                                    1,                          //aBuffer
                                    NULL,                       //aRegion
                                    NULL,                       //aStatusConsumed
                                    NULL,                       //aStatusDisplayed
                                    NULL,                       //aTimeStamp
                                    NULL,                       //aStatusDispXTimes
                                    NULL                        //aDisplayedXTimes
                                    );
        
        ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
        
        // Consume update on buffer 1. This should make buffer 0 available
        if (visible)
            {
            SymbianStreamProcessNotifications(iNs, 
                                                ESOWF_EventAvailable, 
                                                iScreenNo,
                                                iStreamUpdatedSerialNumber,
                                                &newNotificationsMask);
            
            ASSERT_TRUE(newNotificationsMask == 0);

            SymbianStreamProcessNotifications(iNs, 
                                                0, 
                                                iScreenNo,
                                                ++iStreamUpdatedSerialNumber,
                                                &newNotificationsMask);
            ASSERT_TRUE(newNotificationsMask == 0);
            }
        SymbianStreamCheckVisible(iNs, ESOWF_EventAvailable, iScreenNo, iStreamUpdatedSerialNumber); 
        SymbianStreamCheckVisible(iNs, ESOWF_EventAvailable, iScreenNo, iStreamUpdatedSerialNumber);        
        
        // Make sure displayed event was completed
        User::WaitForRequest(statusAvailable);
        if (visible)
            {
            ASSERT_EQUALS(statusAvailable.Int(), KErrNone);
            }
        else
            {
            ASSERT_EQUALS(statusAvailable.Int(), KErrNotVisible);
            }

        err = SymbianStreamRemoveObserver(iNs, this, ESOWF_EventUpdated);
        ASSERT_TRUE(err == KErrNone);
        SymbianStreamRemoveReference(iNs);
        CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one    
        iUtility->DestroySurface(surface);
        }
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0115
@SYMTestCaseDesc        Test displayed x times notification
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation processes displayed x times
                        notifications correctly.
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for displayed notifications
     5.  Submit an update to the surface
     6.  The observer function should be invoked and verifies that stream and event parameters.
     7.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     8.  If X has not been reached yet verify that displayed x status is not completed then goto 6. 
         Otherwise, goto step 9.   
     9.  Verify displayed-x-times status is completed with KErrNone
     10. Remove the observer
     11. Destroy the native stream.
     
     The test is then repeated but the compositor claims the native stream is not visible.
        
@SYMTestExpectedResults
    No errors, displayed-x-times status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0115L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;  
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iImmediateAvailable = EFalse;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    
    for (TInt i = 0; i < 2; ++i)
        {
        TBool visible = (i == 0);        
        iSourceStreamUpdatedCalled = 0;
        TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
        ASSERT_FALSE(surface.IsNull());
        
        CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
        err = SymbianStreamAcquire(&surface, &iNs);
        ASSERT_TRUE(err == KErrNone);
        ASSERT_TRUE(iNs);
    
        err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
        ASSERT_TRUE(err == KErrNone);
        
        TRequestStatus statusDisplayedX;
        TInt X = 5;

        CExtensionContainer* updateExtension = NULL;
        err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
        ASSERT_TRUE(err == KErrNone);
        MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
        ASSERT_NOT_NULL(updateProxy);
        
        iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayedX;
        
        updateProxy->ContentUpdated(surface,                    //aSurface
                                    0,                          //aBuffer
                                    NULL,                       //aRegion
                                    NULL,                       //aStatusConsumed
                                    NULL,                       //aStatusDisplayed
                                    NULL,                       //aTimeStamp
                                    &statusDisplayedX,          //aStatusDispXTimes
                                    &X                          //aDisplayedXTimes
                                    );        
        
        ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
        
        
        khronos_int32_t events = ESOWF_EventDisplayedX;
        for (TInt i = 0; i < X; ++i)
            {            
            // Pretend that a composition has occured
            ++iStreamUpdatedSerialNumber;
            
            if (visible)
                {
                khronos_int32_t newNotificationsMask = 0;                        
                SymbianStreamProcessNotifications(iNs, 
                                                    events, 
                                                    iScreenNo, 
                                                    iStreamUpdatedSerialNumber,
                                                    &newNotificationsMask);
            
                SymbianStreamCheckVisible(iNs, events, iScreenNo, iStreamUpdatedSerialNumber); 
                SymbianStreamCheckVisible(iNs, events, iScreenNo, iStreamUpdatedSerialNumber);   
                
                if (i < X -1)
                    {
                    ASSERT_TRUE(newNotificationsMask == ESOWF_EventDisplayedX);
                    ASSERT_EQUALS(statusDisplayedX.Int(), KRequestPending);
                    }
                else
                    {
                    ASSERT_TRUE(newNotificationsMask == 0);
                    User::WaitForRequest(statusDisplayedX);
                    ASSERT_EQUALS(statusDisplayedX.Int(), KErrNone);            
                    }
                }
            else
                {
                SymbianStreamCheckVisible(iNs, events, iScreenNo, iStreamUpdatedSerialNumber); 
                SymbianStreamCheckVisible(iNs, events, iScreenNo, iStreamUpdatedSerialNumber);            
                User::WaitForRequest(statusDisplayedX);
                ASSERT_EQUALS(statusDisplayedX.Int(), KErrNotVisible);
                break;
                }
            }
            
        err = SymbianStreamRemoveObserver(iNs, this, ESOWF_EventUpdated);
        ASSERT_TRUE(err == KErrNone);
        SymbianStreamRemoveReference(iNs);
        CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
        
        iUtility->DestroySurface(surface);
        }
    }


/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0116
@SYMTestCaseDesc        Test all notifications together
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation processes displayed x times
                        notifications correctly.
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register available, displayed and displayed-x-times notifications.
     5.  Submit an update to the surface
     6.  Wait for the displayed notification to complete.
     7.  The observer function should be invoked and verifies that stream and event parameters.
     8.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     9.  If X has not been reached yet verify that displayedx and available status is not completed 
         then goto 5; otherwise, goto step 11.   
     10. Verify displayed-x-times status is completed with KErrNone
     11. Submit an update on a different buffer number
     12. The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     13. Verify that the available status is completed with KErrNone
     14. Destroy the native stream.
        
@SYMTestExpectedResults
    No errors, displayed-x-times status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0116L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0; 
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateAvailable = EFalse;
    iImmediateVisible = SYM_CONTENT_VISIBLE;   
    
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusDisplayedX;
    TRequestStatus statusAvailable;
    TRequestStatus statusDisplayed;
        
    TInt X = 5;
    TUint32 displayedTime = 0;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    // the composition is signalled as ongoing (busy system)
    iImmediateAvailable = EFalse;
    // we expect, initially that the composer is asked to check only for the displayed notifications
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayedX | ESOWF_EventDisplayed;
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable,           //aStatusConsumed
                                &statusDisplayed,           //aStatusDisplayed
                                &displayedTime,             //aTimeStamp
                                &statusDisplayedX,          //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );     
    
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    // simulating the ongoing composition, the processing to be deferred for the following one
    khronos_int32_t newNotificationsMask = 0;
    SymbianStreamProcessNotifications(iNs, 
                                        iExpectedSourceStreamUpdatedEventMask, 
                                        iScreenNo,
                                        iStreamUpdatedSerialNumber++,
                                        &newNotificationsMask);

    ASSERT_TRUE(newNotificationsMask == iExpectedSourceStreamUpdatedEventMask);
    
    for (TInt i = 0; i < X; ++i)
        {                                
        // Pretend that a composition has occured
                
        khronos_int32_t events = newNotificationsMask;
        // we process the expected notifications
        newNotificationsMask = 0;
        SymbianStreamProcessNotifications(iNs, 
                                            events, 
                                            iScreenNo,
                                            iStreamUpdatedSerialNumber++,
                                            &newNotificationsMask);
                
        // No updates during composition so newNotificationMask should still be zero 
         
        if (i == 0)
            {
            User::WaitForRequest(statusDisplayed);
            ASSERT_EQUALS(statusDisplayed.Int(), KErrNone);
            }
        
        
        if (i < X - 1)
            {
            ASSERT_TRUE(newNotificationsMask == ESOWF_EventDisplayedX);
            // Displayed X times for buffer zero should not be completed yet
            ASSERT_EQUALS(statusDisplayedX.Int(), KRequestPending);
            ASSERT_EQUALS(statusAvailable.Int(), KRequestPending);
            }
        else
            {
            ASSERT_TRUE(newNotificationsMask == 0);
            User::WaitForRequest(statusDisplayedX);
            ASSERT_EQUALS(statusDisplayedX.Int(), KErrNone);            
            }
        }
    
    // the composition is, still, signalled as ongoing (busy system)
    iImmediateAvailable = EFalse;
    // we expect, initially that the composer is asked to check only for the displayed notifications
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventAvailable;
    iUtility->SubmitUpdate(KAllScreens, surface, 1, 0);
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
    
    SymbianStreamProcessNotifications(iNs, 
                                        iExpectedSourceStreamUpdatedEventMask, 
                                        iScreenNo,
                                        iStreamUpdatedSerialNumber++,
                                        &newNotificationsMask);
    
    User::WaitForRequest(statusAvailable);
    ASSERT_EQUALS(statusAvailable.Int(), KErrNone);    
    ASSERT_TRUE(newNotificationsMask == 0);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    
    iUtility->DestroySurface(surface);
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0117_1
@SYMTestCaseDesc        Test displayed notification is cancelled when there are no registered SUS 
                        notification observers
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels the displayed notification
                        when the native stream has not registered any observers for SUS notifications
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Do not add any SUS notification observers to the native stream
     5.  Register for displayed notification.
     6.  Submit an update to the surface
     7.  The observer function should be invoked and verifies that stream and event parameters.
     8.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     9.  Verify displayed status is completed with KErrCancel
     10. Destroy the native stream.
        
@SYMTestExpectedResults
    No errors, displayed status completed with KErrCancel.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0117_1L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    // Do not add observers for SUS notifications!
    
    iExpectedSourceStreamUpdatedEventMask = 0;
    err = SymbianStreamAddObserver(iNs, SourceStreamUpdatedCallback, this);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusDisplayed;
    TUint32 timeStamp = 0;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                &statusDisplayed,           //aStatusDisplayed
                                &timeStamp,                 //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );    
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 1);
    
    // Pretend that a composition has occured
    khronos_int32_t notificationsMask = 0;
    khronos_int32_t newNotificationsMask = 0;
    
    SymbianStreamProcessNotifications(iNs, 
                                        ESOWF_EventAvailable | ESOWF_EventDisplayed |ESOWF_EventDisplayedX, 
                                        iScreenNo,
                                        ++iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    // No updates during composition so newNotificationMask should still be zero 
    ASSERT_EQUALS(newNotificationsMask, notificationsMask);
    
    // The displayed notification should be cancelled as we have no observers registered
    ASSERT_EQUALS(statusDisplayed.Int(), KErrCancel);
    ASSERT_TRUE(timeStamp == 0);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0117_2
@SYMTestCaseDesc        Test displayed x times notification is cancelled when there are no registered SUS 
                        notification observers
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels the displayed x times notification
                        when the native stream has not registered any observers for SUS notifications
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Do not add any SUS notification observers to the native stream
     5.  Register for displayed x times notification.
     6.  Submit an update to the surface
     7.  The observer function should be invoked and verifies that stream and event parameters.
     8.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     9.  Verify displayed x time status is completed with KErrCancel
     10. Destroy the native stream.
        
@SYMTestExpectedResults
    No errors, displayed x times status completed with KErrCancel.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0117_2L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    // Do not add observers for SUS notifications!
    
    iExpectedSourceStreamUpdatedEventMask = 0;
    err = SymbianStreamAddObserver(iNs, SourceStreamUpdatedCallback, this);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusDisplayedX;
    TInt X = 5;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                &statusDisplayedX,          //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );     

    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 1);
    
    User::WaitForRequest(statusDisplayedX);
    ASSERT_EQUALS(statusDisplayedX.Int(), KErrCancel);
    
    for (TInt i = 0; i < X; ++i)
        {
        // Pretend that a composition has occured
        khronos_int32_t newNotificationsMask = 0;
        SymbianStreamProcessNotifications(iNs, 
                                            ESOWF_EventAvailable | ESOWF_EventDisplayed |ESOWF_EventDisplayedX, 
                                            iScreenNo,
                                            ++iStreamUpdatedSerialNumber,
                                            &newNotificationsMask);
        
        // No updates during composition so newNotificationMask should still be zero 
        ASSERT_TRUE(newNotificationsMask == 0);
        }
   
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0117_3
@SYMTestCaseDesc        Test available notification is cancelled when there are no registered SUS 
                        notification observers
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels the available notification
                        when the native stream has not registered any observers for SUS notifications
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Do not add any SUS notification observers to the native stream
     5.  Register for available notification.
     6.  Submit an update to the surface
     7.  The observer function should be invoked and verifies that stream and event parameters.
     8.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     9.  Verify available status is completed with KErrCancel
     10. Destroy the native stream.
        
@SYMTestExpectedResults
    No errors, available status completed with KErrCancel.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0117_3L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    // Do not add observers for SUS notifications!
    
    iExpectedSourceStreamUpdatedEventMask = 0;
    err = SymbianStreamAddObserver(iNs, SourceStreamUpdatedCallback, this);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusAvailable;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable,           //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                ); 
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 1);
    
    // Available should only be completed when submit update is called with a different buffer no.
    // But, when there are no registered SUS observers, the notification should complete immediately
    // with KErrCancel
    ASSERT_EQUALS(statusAvailable.Int(), KErrCancel);
    
    // Pretend that a composition has occured
    khronos_int32_t newNotificationsMask = 0;
    SymbianStreamProcessNotifications(iNs, 
                                        ESOWF_EventAvailable | ESOWF_EventDisplayed |ESOWF_EventDisplayedX, 
                                        iScreenNo,
                                        ++iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    ASSERT_TRUE(newNotificationsMask == 0);
    
    // Update and switch to buffer 1
    iExpectedSourceStreamUpdatedEventMask = 0;
    iUtility->SubmitUpdate(KAllScreens, surface, 1, 0);    
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    // Consume update on buffer 1. This should make buffer 0 available
    SymbianStreamProcessNotifications(iNs, 
                                        ESOWF_EventAvailable | ESOWF_EventDisplayed |ESOWF_EventDisplayedX, 
                                        iScreenNo,
                                        ++iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    ASSERT_TRUE(newNotificationsMask == 0);
    
    User::WaitForRequest(statusAvailable);
    ASSERT_EQUALS(statusAvailable.Int(), KErrCancel);
   
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**  
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0117_4
@SYMTestCaseDesc        Test all notifications are cancelled when there are no registered SUS 
                        notification observers
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels all notifications
                        when the native stream has not registered any observers for SUS notifications
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Do not add any SUS notification observers to the native stream
     5.  Register for displayed, available and displayed x times notifications
     6.  Submit an update to the surface
     7.  The observer function should be invoked and verifies that stream and event parameters
     8.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     9.  Verify all notification statuses completed with KErrCancel
     10. Destroy the native stream.
        
@SYMTestExpectedResults
    No errors, all notification statuses completed with KErrCancel.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0117_4L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    // Do not add observers for SUS notifications!
    
    iExpectedSourceStreamUpdatedEventMask = 0;
    err = SymbianStreamAddObserver(iNs, SourceStreamUpdatedCallback, this);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusDisplayedX;
    TRequestStatus statusAvailable;
    TRequestStatus statusDisplayed;
    TInt X = 5;
    TUint32 displayedTime = 0;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable,           //aStatusConsumed
                                &statusDisplayed,           //aStatusDisplayed
                                &displayedTime,             //aTimeStamp
                                &statusDisplayedX,          //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );     

    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 1);
    
    for (TInt i = 0; i < X; ++i)
        {                                
        // Pretend that a composition has occured
        khronos_int32_t newNotificationsMask = 0;
                
        SymbianStreamProcessNotifications(iNs, 
                                            ESOWF_EventAvailable | ESOWF_EventDisplayed |ESOWF_EventDisplayedX, 
                                            iScreenNo,
                                            ++iStreamUpdatedSerialNumber,
                                            &newNotificationsMask);
                
        // No updates during composition so newNotificationMask should still be zero 
        ASSERT_TRUE(newNotificationsMask == 0);
                
        if (i == 0)
            {
            User::WaitForRequest(statusDisplayed);
            User::WaitForRequest(statusDisplayedX);
            User::WaitForRequest(statusAvailable);
            }

        ASSERT_EQUALS(statusAvailable.Int(), KErrCancel);            
        ASSERT_EQUALS(statusDisplayed.Int(), KErrCancel);            
        ASSERT_EQUALS(statusDisplayedX.Int(), KErrCancel);
        }
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );    
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    khronos_int32_t newNotificationsMask = 0;
    SymbianStreamProcessNotifications(iNs, 
                                        ESOWF_EventAvailable | ESOWF_EventDisplayed |ESOWF_EventDisplayedX, 
                                        iScreenNo,
                                        ++iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    ASSERT_EQUALS(statusAvailable.Int(), KErrCancel);
    ASSERT_TRUE(displayedTime == 0);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0118_1
@SYMTestCaseDesc        Test displayed notification is cancelled when content updated is called on  
                        surface which is not registered with a native stream
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels the displayed notification
                        added to a surface which does not have a native stream associated with it
@SYMTestActions

     1.  Create a surface
     2.  Register for displayed notification.
     3.  Submit an update to the surface
     4.  Verify displayed status is completed with KErrSurfaceNotRegistered
        
@SYMTestExpectedResults
    No errors, displayed status completed with KErrSurfaceNotRegistered.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0118_1L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    TRequestStatus statusDisplayed;
    TUint32 timeStamp = 0;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                &statusDisplayed,           //aStatusDisplayed
                                &timeStamp,                 //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );    
    
    User::WaitForRequest(statusDisplayed);
    
    // The displayed notification should return with a surface registration error as there 
    // isn't a native stream registered with surface...
    ASSERT_EQUALS(statusDisplayed.Int(), KErrSurfaceNotRegistered);
    ASSERT_TRUE(timeStamp == 0);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0118_2
@SYMTestCaseDesc        Test displayed x times notification is cancelled when content updated is called on  
                        surface which is not registered with a native stream
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels the displayed x times notification
                        added to a surface which does not have a native stream associated with it
@SYMTestActions

     1.  Create a surface
     2.  Register for displayed x times notification.
     3.  Submit an update to the surface
     4.  Verify displayed x times status is completed with KErrSurfaceNotRegistered
        
@SYMTestExpectedResults
    No errors, displayed x times status completed with KErrSurfaceNotRegistered.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0118_2L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    // Do not create a native stream for surface!
    
    TRequestStatus statusDisplayedXTimes;
    TInt X = 5;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                &statusDisplayedXTimes,     //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );
    
    User::WaitForRequest(statusDisplayedXTimes);
    
    // The displayed x times notification should return with a surface registration error as there 
    // isn't a native stream registered with surface...
    ASSERT_EQUALS(statusDisplayedXTimes.Int(), KErrSurfaceNotRegistered);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0118_3
@SYMTestCaseDesc        Test available notification is cancelled when content updated is called on  
                        surface which is not registered with a native stream
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels the available notification
                        added to a surface which does not have a native stream associated with it
@SYMTestActions

     1.  Create a surface
     2.  Rregister for available notification.
     3.  Submit an update to the surface
     4.  Verify available status is completed with KErrSurfaceNotRegistered
        
@SYMTestExpectedResults
    No errors, available status completed with KErrSurfaceNotRegistered.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0118_3L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    // Do not create a native stream for surface!
    
    TRequestStatus statusAvailable;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable,           //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );
    
    User::WaitForRequest(statusAvailable);  
    
    // The available notification should return with a surface registration error as there 
    // isn't a native stream registered with surface...
    ASSERT_EQUALS(statusAvailable.Int(), KErrSurfaceNotRegistered);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0118_4
@SYMTestCaseDesc        Test all notifications are cancelled when content updated is called on  
                        surface which is not registered with a native stream
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation cancels all notifications
                        added to a surface which does not have a native stream associated with it
@SYMTestActions

     1.  Create a surface
     2.  Register for displayed, available and displayed x times notifications.
     3.  Submit an update to the surface
     4.  Verify all notification statuses completed with KErrSurfaceNotRegistered
        
@SYMTestExpectedResults
    No errors, all notification statuses completed with KErrSurfaceNotRegistered.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0118_4L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    // Do not create a native stream for surface!
    
    TRequestStatus statusDisplayed, statusAvailable, statusDisplayedXTimes;
    TInt X = 5;
    TUint32 timeStamp = 0;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable,           //aStatusConsumed
                                &statusDisplayed,           //aStatusDisplayed
                                &timeStamp,                 //aTimeStamp
                                &statusDisplayedXTimes,     //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                ); 
    
    User::WaitForRequest(statusDisplayed);
    User::WaitForRequest(statusAvailable);
    User::WaitForRequest(statusDisplayedXTimes);
    
    // All notifications should return with a surface registration error as there 
    // isn't a native stream registered with surface...
    ASSERT_EQUALS(statusDisplayed.Int(), KErrSurfaceNotRegistered);
    ASSERT_EQUALS(statusAvailable.Int(), KErrSurfaceNotRegistered);
    ASSERT_EQUALS(statusDisplayedXTimes.Int(), KErrSurfaceNotRegistered);
    ASSERT_TRUE(timeStamp == 0);
    
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0119_1
@SYMTestCaseDesc        Test overflow conditions on notifications for displayed
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation correctly handles duplicate displayed 
                        notifications registered by SUS for content updates on the same screen and buffer
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for displayed notification (d1)
     5.  Submit an update to the surface
     6.  Register for displayed notification (d2)
     7.  Submit an update to the surface
     8.  The observer function should be invoked and verifies that stream and event parameters.
     9.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     10. Verify d1 status is completed with KErrOverflow and d2 status is completed with KErrNone
     11. Destroy the native stream
        
@SYMTestExpectedResults
    No errors, d1 status completed with KErrOverflow, d2 status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0119_1L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);;
    
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayed;
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusDisplayed1, statusDisplayed2;
    TUint32 timeStamp1 = 0;
    TUint32 timeStamp2 = 0;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                &statusDisplayed1,          //aStatusDisplayed
                                &timeStamp1,                //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );
    
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                &statusDisplayed2,          //aStatusDisplayed
                                &timeStamp2,                //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
    
    User::WaitForRequest(statusDisplayed1);
    ASSERT_EQUALS(statusDisplayed1.Int(), KErrOverflow);
    ASSERT_TRUE(timeStamp1 == 0);
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled);
    
    // Pretend that a composition has occured
    khronos_int32_t newNotificationsMask = 0;
    SymbianStreamProcessNotifications(iNs, 
                                        iExpectedSourceStreamUpdatedEventMask, 
                                        iScreenNo,
                                        ++iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    ASSERT_TRUE(newNotificationsMask == 0);
    
    // Make sure displayed event was completed
    User::WaitForRequest(statusDisplayed2);
    ASSERT_EQUALS(statusDisplayed2.Int(), KErrNone);
    ASSERT_TRUE(timeStamp2);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0119_2
@SYMTestCaseDesc        Test overflow conditions on notifications for displayed-x-times
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation correctly handles duplicate displayed 
                        x times notifications registered by SUS for content updates on the same screen
                        and buffer
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for displayed x times notification (d1)
     5.  Submit an update to the surface
     6.  Register for displayed x times notification (d2)
     7.  Submit an update to the surface
     8.  The observer function should be invoked and verifies that stream and event parameters.
     9.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     10. Verify d1 status is completed with KErrOverflow and d2 status is completed with KErrNone
     11. Destroy the native stream
        
@SYMTestExpectedResults
    No errors, d1 status completed with KErrOverflow, d2 status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0119_2L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    TInt X = 5;
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayedX;
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusDisplayedX1, statusDisplayedX2;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                &statusDisplayedX1,         //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );
    
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                &statusDisplayedX2,         //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );
    
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
    
    User::WaitForRequest(statusDisplayedX1);
    ASSERT_EQUALS(statusDisplayedX1.Int(), KErrOverflow);
    
    for (TInt i = 0; i < X; ++i)
        {
        ASSERT_EQUALS(statusDisplayedX2.Int(), KRequestPending);
        
        // Pretend that a composition has occured
        khronos_int32_t newNotificationsMask = 0;
        SymbianStreamProcessNotifications(iNs, 
                                            ESOWF_EventDisplayedX, 
                                            iScreenNo,
                                            ++iStreamUpdatedSerialNumber,
                                            &newNotificationsMask);        

        if (i < X -1)
            {
            ASSERT_TRUE(newNotificationsMask == ESOWF_EventDisplayedX);
            }
        else
            {
            ASSERT_TRUE(newNotificationsMask == 0);
            }
        }
    
    // Make sure displayed event was completed
    User::WaitForRequest(statusDisplayedX2);
    ASSERT_EQUALS(statusDisplayedX2.Int(), KErrNone);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0119_3
@SYMTestCaseDesc        Test overflow conditions on notifications for available
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation correctly handles duplicate available 
                        notifications registered by SUS for content updates on the same screen and buffer
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for available notification (a1)
     5.  Submit an update to the surface
     6.  Register for available notification (a2)
     7.  Submit an update to the surface
     8.  The observer function should be invoked and verifies that stream and event parameters.
     9.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     10. Verify d1 status is completed with KErrOverflow and d2 status is completed with KErrNone
     11. Destroy the native stream
        
@SYMTestExpectedResults
    No errors, a1 status completed with KErrOverflow, a2 status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0119_3L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateAvailable = EFalse;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusAvailable1, statusAvailable2, statusAvailable3;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    iExpectedSourceStreamUpdatedEventMask = 0;
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable1,          //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );    
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventAvailable;
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable2,          //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );   
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
    
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventAvailable;
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable3,          //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );   
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 6);
    
    User::WaitForRequest(statusAvailable1);
    ASSERT_EQUALS(statusAvailable1.Int(), KErrOverflow);
    
    // Pretend that a composition has occured
    khronos_int32_t newNotificationsMask = 0;
    SymbianStreamProcessNotifications(iNs, 
                                        ESOWF_EventAvailable, 
                                        iScreenNo,
                                        iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    ASSERT_TRUE(newNotificationsMask == 0);
    
    // Make sure displayed event was completed
    User::WaitForRequest(statusAvailable2);
    ASSERT_EQUALS(statusAvailable2.Int(), KErrNone);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0120_1
@SYMTestCaseDesc        Test overflow conditions on notifications for displayed
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation correctly handles duplicate notifications
                        registered by SUS for content updates on different buffers
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for displayed notification (d1)
     5.  Submit an update to the surface on buffer 0
     6.  Register for displayed notification (d2)
     7.  Submit an update to the surface on buffer 1
     8.  The observer function should be invoked and verifies that stream and event parameters.
     9.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     10. Verify d1 status is completed with KErrOverflow and d2 status is completed with KErrNone
     11. Destroy the native stream
        
@SYMTestExpectedResults
    No errors, d1 status completed with KErrOverflow, d2 status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0120_1L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayed;
    
    TRequestStatus statusDisplayed1, statusDisplayed2;
    TUint32 timeStamp1 = 0;
    TUint32 timeStamp2 = 0;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                &statusDisplayed1,          //aStatusDisplayed
                                &timeStamp1,                //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                1,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                &statusDisplayed2,          //aStatusDisplayed
                                &timeStamp2,                //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );  
    
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
    
    User::WaitForRequest(statusDisplayed1);
    ASSERT_EQUALS(statusDisplayed1.Int(), KErrOverflow);
    
    // Pretend that a composition has occured
    khronos_int32_t newNotificationsMask = 0;
    SymbianStreamProcessNotifications(iNs, 
                                        iExpectedSourceStreamUpdatedEventMask, 
                                        iScreenNo,
                                        ++iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    ASSERT_TRUE(newNotificationsMask == 0);
    
    // Make sure displayed event was completed
    User::WaitForRequest(statusDisplayed2);
    ASSERT_EQUALS(statusDisplayed2.Int(), KErrNone);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }


/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0120_2
@SYMTestCaseDesc        Test overflow conditions on notifications for displayed-x-times
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation correctly handles duplicate notifications
                        registered by SUS for content updates on different buffers
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for displayed-x-times notification (d1)
     5.  Submit an update to the surface on buffer 0
     6.  Register for displayed-x-times notification (d2)
     7.  Submit an update to the surface on buffer 1
     8.  The observer function should be invoked and verifies that stream and event parameters.
     9.  Verify d1 status is completed with KErrOverflow
     10. The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     11. Step 11. is repeated 5 times to simulate 5 compositions.
     12. Verify that d2 status is completed with KErrNone
     13. Destroy the native stream
        
@SYMTestExpectedResults
    No errors, d1 status completed with KErrOverflow, d2 status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0120_2L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    TInt X = 5;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayedX;
    
    TRequestStatus statusDisplayedX1, statusDisplayedX2;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                &statusDisplayedX1,         //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                1,                          //aBuffer
                                NULL,                       //aRegion
                                NULL,                       //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                &statusDisplayedX2,         //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
    
    User::WaitForRequest(statusDisplayedX1);
    ASSERT_EQUALS(statusDisplayedX1.Int(), KErrOverflow);
    
    for (TInt i = 0; i < X; ++i)
        {
        ASSERT_EQUALS(statusDisplayedX2.Int(), KRequestPending);
        
        // Pretend that a composition has occured
        khronos_int32_t newNotificationsMask = 0;
        SymbianStreamProcessNotifications(iNs, 
                                            iExpectedSourceStreamUpdatedEventMask, 
                                            iScreenNo,
                                            ++iStreamUpdatedSerialNumber,
                                            &newNotificationsMask);
        if (i < X -1)
            {
            ASSERT_TRUE(newNotificationsMask == ESOWF_EventDisplayedX);
            }
        else
            {
            ASSERT_TRUE(newNotificationsMask == 0);
            }
        }
    
    // Make sure displayed event was completed
    User::WaitForRequest(statusDisplayedX2);
    ASSERT_EQUALS(statusDisplayedX2.Int(), KErrNone);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0120_3
@SYMTestCaseDesc        Test overflow conditions on notifications for available
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Verify that the surface stream adaptation correctly handles duplicate notifications
                        registered by SUS for content updates on different buffers
@SYMTestActions

     1.  Create a surface
     2.  Create a native stream for the surface
     3.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     4.  Register for available notification (d1)
     5.  Submit an update to the surface on buffer 0
     6.  Register for available notification (d2)
     7.  Submit an update to the surface on buffer 1
     8.  The observer function should be invoked and verifies that stream and event parameters.
     9.  The tester simulates the end of a composition by firing SymbianStreamProcessNotifications
     10. Verify d1 status is completed with KErrOverflow and d2 status is completed with KErrNone
     11. Destroy the native stream
        
@SYMTestExpectedResults
    No errors, d1 status completed with KErrOverflow, d2 status completed with KErrNone.
        
*/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0120_3L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    //force composition, otherwise we may never be able to experience the overflow
    iImmediateAvailable = EFalse;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusAvailable1, statusAvailable2, statusAvailable3;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    iExpectedSourceStreamUpdatedEventMask = 0;

    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable1,          //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventAvailable;
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                1,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable2,          //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 4);
   
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventAvailable;
    
    updateProxy->ContentUpdated(surface,                    //aSurface
                                2,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable3,          //aStatusConsumed
                                NULL,                       //aStatusDisplayed
                                NULL,                       //aTimeStamp
                                NULL,                       //aStatusDispXTimes
                                NULL                        //aDisplayedXTimes
                                );
    
    // Verify that the context's callback is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 6);
   
    // Theoretically KErrNone could be returned because buffer 0 is actually available.
    // However, this would be a change in behaviour.
    User::WaitForRequest(statusAvailable1);
    ASSERT_EQUALS(statusAvailable1.Int(), KErrOverflow);
    
    // Pretend that a composition has occured
    khronos_int32_t newNotificationsMask = 0;
    SymbianStreamProcessNotifications(iNs, 
                                        iExpectedSourceStreamUpdatedEventMask, 
                                        iScreenNo,
                                        iStreamUpdatedSerialNumber,
                                        &newNotificationsMask);
    
    ASSERT_TRUE(newNotificationsMask == 0);
    
    // Make sure displayed event was completed
    User::WaitForRequest(statusAvailable2);
    ASSERT_EQUALS(statusAvailable2.Int(), KErrNone);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }



/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0122
@SYMTestCaseDesc        
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        High
@SYMTestPurpose         Tests an end to end displayed notification.
@SYMTestActions

     1.  Create a surface.
     2.  Create a native stream for the surface.
     3.  Register an observer for content-update events.
     4.  Register notifications for displayed, available and displayed-x-times.
     5.  Submit an update to the surface
     6.  Remove the content-updated observer.
     7.  Wait for the request status objects to be completed.
        
@SYMTestExpectedResults
    The notification status objects are completed with KErrCancel.
        
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0122L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    CTestNativeStream::iTester = this;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    err = SymbianStreamAddExtendedObserver(iNs, SourceStreamUpdatedCallback, this, iScreenNo, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    
    TRequestStatus statusDisplayed, statusAvailable, statusDisplayedX;
    TUint32 displayedTime = 0;
    TInt X = 42;

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(iScreenNo,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* updateProxy=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(updateProxy);
    
    iExpectedSourceStreamUpdatedEventMask = ESOWF_EventDisplayed | ESOWF_EventDisplayedX;

    updateProxy->ContentUpdated(surface,                    //aSurface
                                0,                          //aBuffer
                                NULL,                       //aRegion
                                &statusAvailable,           //aStatusConsumed
                                &statusDisplayed,           //aStatusDisplayed
                                &displayedTime,             //aTimeStamp
                                &statusDisplayedX,          //aStatusDispXTimes
                                &X                          //aDisplayedXTimes
                                );     
    
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 2);
    err = SymbianStreamRemoveObserver(iNs, this, ESOWF_EventUpdated);
    ASSERT_TRUE(err == KErrNone);
    
    User::WaitForRequest(statusDisplayed);
    ASSERT_EQUALS(statusDisplayed.Int(), KErrCancel);
    User::WaitForRequest(statusAvailable);
    ASSERT_EQUALS(statusAvailable.Int(), KErrCancel);
    User::WaitForRequest(statusDisplayedX);
    ASSERT_EQUALS(statusDisplayedX.Int(), KErrCancel);
    
    SymbianStreamRemoveReference(iNs);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    
    iUtility->DestroySurface(surface);
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0130
@SYMTestCaseDesc        Add/remove Sym observers - positive test cases
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         Verify that Sym observers can be added, removed and re-added
@SYMTestActions 
        Create a surface with 1 buffer
        Add a Sym observer (displayed notification) 
        Remove the Sym observer
        Add the Sym observer
@SYMTestExpectedResults
        KErrNone is returned when the Sym observer is added
        KErrNone is returned when the Sym observer is removed
        KErrNone is returned when the Sym observer is re-added
 **/ 
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0130L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 1);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    SymbianStreamType ns;
    err = SymbianStreamAcquire(&surface,&ns);
    ASSERT_TRUE(err == KErrNone);    
    ASSERT_TRUE(ns);
    
    TInt32 screenNumber = 0;
    TInt localnumber = 0;
    err = SymbianStreamAddExtendedObserver(ns, TestUpdateCallback, &localnumber, screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamRemoveObserver(ns, &localnumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamAddExtendedObserver(ns, TestUpdateCallback, &localnumber, screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrNone);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0131
@SYMTestCaseDesc        Add/remove Sym observers - negative test cases
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         Verify that Sym observers reject invalid parameters
@SYMTestActions 
        Create a surface with 1 buffer
        Add an invalid Sym observer (invalid event flag)
        Remove a non-existent Sym observer
        Add a valid Sym observer
        Add the same Sym observer
        Remove the valid Sym observer
        Remove the same Sym observer
@SYMTestExpectedResults
        KErrArgument is returned and the observer list is un-changed
        KErrNotFound is returned and the observer list is un-changed
        KErrOverflow is returned when the Sym observer is added again and the observer list is un-changed
        KErrNotFound is returned when the Sym observer is removed again the observer list is un-changed
 **/ 
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0131L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 1);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    SymbianStreamType ns;
    err = SymbianStreamAcquire(&surface, &ns);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(ns);
    
    TInt32 screenNumber = 0;
    err = SymbianStreamAddExtendedObserver(ns, NULL, NULL, screenNumber, ESOWF_NoEvent);
    ASSERT_TRUE(err == KErrArgument);
    err = SymbianStreamRemoveObserver(ns, &screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrNotFound);
    
    // add the same observer twice
    err = SymbianStreamAddExtendedObserver(ns, NULL, NULL, screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrArgument);
    err = SymbianStreamAddExtendedObserver(ns, NULL, NULL, screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrArgument);
    
    // remove the same observer
    err = SymbianStreamRemoveObserver(ns, &screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrNotFound);
    err = SymbianStreamRemoveObserver(ns, &screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrNotFound);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0132
@SYMTestCaseDesc        Destroy a native stream when a Sym observer is registered
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         Verify that Sym observers are safely removed before native stream destruction
@SYMTestActions 
        Create a surface with 1 buffer
        Add a Sym observer (displayed notification) 
        Destroy the native stream
        Create another native stream using the surface created previously
        Remove the same Sym observer added in step 2
@SYMTestExpectedResults
        KErrNone is returned when the Sym observer is added
        KErrNotFound is returned when the Sym observer is attempting to be removed
 **/ 
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0132L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);

    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 1);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    SymbianStreamType ns;
    err = SymbianStreamAcquire(&surface, &ns);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(ns);
    
    TInt screenNumber = 0;
    err = SymbianStreamAddExtendedObserver(ns, NULL, NULL, screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrArgument);
   
    SymbianStreamRemoveReference(ns);
    
    SymbianStreamType ns2;
    err = SymbianStreamAcquire(&surface, &ns2);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(ns2);
    
    err = SymbianStreamRemoveObserver(ns2, &screenNumber, ESOWF_EventDisplayed);
    ASSERT_TRUE(err == KErrNotFound);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
	}

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0133
@SYMTestCaseDesc        Test the construction/destruction of a CContentUpdateProxy
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         Verify that additional screens can be created and destroyed
@SYMTestActions

     1.  Create a new CContentUpdateProxy (screen 1)
     2.  Verify API version and Internal version are correct
     3.  Create a surface
     4.  Create a native stream for the surface
     5.  Register the test class as an observer for ESOWF_EventUpdated to act as a fake composer
     6.  Submit an update to the surface
     7.  The observer function should be invoked and verifies that stream and event parameters.
     8.  Verify that the source-stream updated callback is invoked.
     9.  Destroy the native stream.
     10. Destroy the CContentUpdateProxy (screen 1)

@SYMTestExpectedResults
        KErrNone is returned when screen 1 is created
        API version and Internal version match expected results
        KErrNone is returned when screen 1 is destroyed
 **/ 
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0133L()
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    // Register a new screen
    iScreenNo = 0;    
    iContextUpdatedFlags = 0;
    iStreamUpdatedSerialNumber = 1;
    iSourceStreamUpdatedCalled = 0;
    iImmediateVisible = SYM_CONTENT_VISIBLE;
    
    TSurfaceId surface=iUtility->CreateSurfaceL(TSize(100,100), EUidPixelFormatARGB_8888_PRE, 400, 4);
    ASSERT_FALSE(surface.IsNull());
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    err = SymbianStreamRegisterScreenNotifications(1, 0, KCompositorVersion);
    ASSERT_TRUE(err == KErrNone);   
    

    CExtensionContainer* updateExtension = NULL;
    err = SymbianStreamHasRegisteredScreenNotifications(1,reinterpret_cast<void**>(&updateExtension));
    ASSERT_TRUE(err == KErrNone);
    MCompositionSurfaceUpdate* screen1=updateExtension->GetInterface<MCompositionSurfaceUpdate>();
    ASSERT_NOT_NULL(screen1);

    TVersion expectedVersion(KCompositorVersionMajor,KCompositorVersionMinor,KCompositorVersionRevision);
    
    ASSERT_EQUALS(screen1->ApiVersion(), 3);
    ASSERT_TRUE(screen1->InternalVersion().iBuild == expectedVersion.iBuild &&
                screen1->InternalVersion().iMajor == expectedVersion.iMajor &&
                screen1->InternalVersion().iMinor == expectedVersion.iMinor);
        
    err = SymbianStreamAcquire(&surface, &iNs);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(iNs);
    
    iExpectedSourceStreamUpdatedEventMask = 0;
    err = SymbianStreamAddObserver(iNs, SourceStreamUpdatedCallback, this);
    ASSERT_TRUE(err == KErrNone);
    
    iSourceStreamUpdatedCalled = EFalse;    
    screen1->ContentUpdated(surface,0,NULL,NULL,NULL,NULL,NULL,NULL);
    
    // Verify that the context's callback (screen 1) is invoked when SubmitUpdate is called.
    ASSERT_TRUE(iSourceStreamUpdatedCalled == 1);
    
    SymbianStreamRemoveReference(iNs);
    
    // Destroy the context (screen 1)
    err = SymbianStreamUnregisterScreenNotifications(1);
    ASSERT_TRUE(err == KErrNone);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }              

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0140
@SYMTestCaseDesc        Negative test - Acquire write/read buffers and retrieve the buffer ptr (single threaded test).
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         Verify SymbianStreamAcquireWriteBuffer(), SymbianStreamReleaseWriteBuffer(),
                        SymbianStreamAcquireReadBuffer() and SymbianStreamReleaseReadBuffer() methods
                        work as expected.
                        
@SYMTestActions 
        Create a native stream 
        For each buffer:
        - Acquire the read buffer 
        - Acquire the write buffer 
        - Release the write buffer
        - Acquire the read buffer

        Repeat for each buffer. Finally:
        Acquire the write buffer
        Release the write buffer    
@SYMTestExpectedResults
        For each buffer of the native stream, check:
        - The read buffer should be aquired successfully
        If the number of buffers = 1, check:
        - The write buffer should be aquired successfully
        If the number of buffers > 1, check:
        - The write buffer before the final one should be aquired successfully
        - The final acquire write buffer call should return OWF_INVALID_HANDLE if not single buffered
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0140L(TInt aNumBuffers)
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);

    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    ASSERT_TRUE(aNumBuffers > 0);
    
    TSize surfaceSize(TSize(100,100));
    khronos_int32_t width = surfaceSize.iWidth;
    khronos_int32_t height = surfaceSize.iHeight;
    
    OWF_IMAGE_FORMAT pixelFormat =
        {
        OWF_IMAGE_ARGB8888,
        ETrue,
        ETrue,
        aNumBuffers
        };  
    
    SymbianStreamType ns=helperCreateImageStream(width,
                                                    height,
                                                    &pixelFormat,
                                                    aNumBuffers);
    ASSERT_TRUE(ns);
    
    khronos_int32_t writeBuffer = 0;
    khronos_int32_t readBuffer = 0;    
    TInt bufferCount;
    
    if (aNumBuffers == 1)
        {
        bufferCount = aNumBuffers;
        }
    else
        {
        bufferCount = aNumBuffers - 1;
        }
    
    RArray<TInt> bufferHandleArray;

    // Loop through the buffers
    for (TInt count=0; count<bufferCount; count++)
        {
        err = SymbianStreamAcquireReadBuffer(ns, &readBuffer);
        ASSERT_TRUE(err == KErrNone);
        ASSERT_TRUE(readBuffer);
        bufferHandleArray.Append(readBuffer);
        
        // Acquire the write buffer
        err = SymbianStreamAcquireWriteBuffer(ns, &writeBuffer);
        ASSERT_TRUE(err == KErrNone);
        ASSERT_TRUE(writeBuffer);     
        SymbianStreamReleaseWriteBuffer(ns,writeBuffer);        
        }   

    // Acquire final read buffer
    khronos_int32_t finalReadBuffer;
    err = SymbianStreamAcquireReadBuffer(ns, &finalReadBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(finalReadBuffer);
    bufferHandleArray.Append(finalReadBuffer);        

    khronos_int32_t finalWriteBuffer;
    err = SymbianStreamAcquireWriteBuffer(ns, &finalWriteBuffer);
    ASSERT_TRUE(err != KErrBadHandle);
    
    // Final checks
    if (aNumBuffers == 1)
        {
        ASSERT_TRUE(finalWriteBuffer);
        err =  SymbianStreamReleaseWriteBuffer(ns,finalWriteBuffer);
        ASSERT_TRUE(err == KErrNone);
        }
    else
        {
        ASSERT_FALSE(finalWriteBuffer);        
        }
    
    for (TInt x=0; x < bufferHandleArray.Count(); ++x)
        {
        err = SymbianStreamReleaseReadBuffer(ns,bufferHandleArray[x]);
        ASSERT_TRUE(err == KErrNone);
        }

    err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(writeBuffer);     
    err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(writeBuffer);     

    bufferHandleArray.Close();
    SymbianStreamRemoveReference(ns);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0141
@SYMTestCaseDesc        Negative test - Acquire multiple read buffers before releasing write buffer (single threaded test).
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            CT
@SYMTestPriority        
@SYMTestPurpose         Verify SymbianStreamAcquireWriteBuffer(), SymbianStreamReleaseWriteBuffer(),
                        SymbianStreamAcquireReadBuffer() and SymbianStreamReleaseReadBuffer() methods
                        work as expected.
                        
@SYMTestActions 
        Create a native stream 
        For each buffer:
        - Acquire the read buffer multiple times
        - Acquire the write buffer 
        - Acquire the read buffer       
        - Release the write buffer
        - Acquire the read buffer

@SYMTestExpectedResults
        For each buffer of the native stream, check:
        - The read buffer index should remain the same before release write bufffer is called
        If the number of buffers = 1, check:
        - The buffer index is always the same
        If the number of buffers > 1, check:
        - The buffer index increments after release write buffer is called
 **/
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0141L(TInt aNumBuffers)
    {
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);

    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));
    
    ASSERT_TRUE(aNumBuffers > 0);
    
    TSize surfaceSize(TSize(100,100));
    khronos_int32_t width = surfaceSize.iWidth;
    khronos_int32_t height = surfaceSize.iHeight;
    
    OWF_IMAGE_FORMAT pixelFormat =
        {
        OWF_IMAGE_ARGB8888,
        ETrue,
        ETrue,
        aNumBuffers
        };  
    
    SymbianStreamType ns=helperCreateImageStream(width,
                                                    height,
                                                    &pixelFormat,
                                                    aNumBuffers);
    ASSERT_TRUE(ns);
    
    khronos_int32_t writeBuffer = 0;
    khronos_int32_t readBuffer = 0; 
    khronos_int32_t bufferIndexRead1, bufferIndexRead2, bufferIndexRead3, bufferIndexRead4;
    khronos_int32_t bufferIndexWrite;
    
    err = SymbianStreamAcquireReadBuffer(ns, &readBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(readBuffer);
    const TSurfaceId* getId = NULL;
    err = SymbianStreamGetBufferId(ns,readBuffer,&bufferIndexRead1,&getId);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns,readBuffer);
    ASSERT_TRUE(err == KErrNone);
    
    err = SymbianStreamAcquireReadBuffer(ns, &readBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(readBuffer);
    err = SymbianStreamGetBufferId(ns,readBuffer,&bufferIndexRead2,&getId);
    ASSERT_TRUE(err == KErrNone);
    SymbianStreamReleaseReadBuffer(ns,readBuffer);
    ASSERT_EQUALS(bufferIndexRead1, bufferIndexRead2);

    err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(writeBuffer);     
    err = SymbianStreamGetBufferId(ns,writeBuffer,&bufferIndexWrite,&getId);
    ASSERT_TRUE(err == KErrNone);
    if (aNumBuffers == 1)
        {
        ASSERT_EQUALS(bufferIndexRead2, bufferIndexWrite);                
        }
    else
        {
        ASSERT_NOT_EQUALS((bufferIndexRead2), bufferIndexWrite);        
        }

    err = SymbianStreamAcquireReadBuffer(ns,&readBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(readBuffer);
    err = SymbianStreamGetBufferId(ns,readBuffer,&bufferIndexRead3,&getId);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns,readBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_EQUALS(bufferIndexRead3, bufferIndexRead2);
    
    err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer);
    ASSERT_TRUE(err == KErrNone);
    
    err = SymbianStreamAcquireReadBuffer(ns,&readBuffer);
    ASSERT_TRUE(err == KErrNone);
    ASSERT_TRUE(readBuffer);
    err = SymbianStreamGetBufferId(ns,readBuffer,&bufferIndexRead4,&getId);
    ASSERT_TRUE(err == KErrNone);
    err = SymbianStreamReleaseReadBuffer(ns,readBuffer);
    ASSERT_TRUE(err == KErrNone);
    if (aNumBuffers == 1)
        {
        ASSERT_EQUALS(bufferIndexRead3, bufferIndexRead4);                
        }
    else
        {
        ASSERT_NOT_EQUALS((bufferIndexRead3), bufferIndexRead4);        
        }

    SymbianStreamRemoveReference(ns);
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }

/**
@SYMTestCaseID          GFX_OPENWFC_NATIVESTREAM_0142
@SYMTestCaseDesc        Special surface ID unit tests
@SYMREQ                 
@SYMPREQ                PREQ2400
@SYMTestType            
@SYMTestPriority        
@SYMTestPurpose         Exercise special surface creation and flip operations
                        
@SYMTestActions 
        Generate a special surface ID with pixel formats of type 
        EUidPixelFormatARGB_8888_PRE, EUidPixelFormatRGB_888, 
        EUidPixelFormatRGB_565, and EUidPixelFormatRGB_332. Then
        - Acquire the read buffer
        - Acquire the write buffer 
        - Check that they are the same
        - Add a value to the read buffer
        - Check that the write buffer has the same value       
        - Release the write buffer
        - Release the read buffer
        Then reset the special surface ID, acquire a stream, and read header data.
        Set the flip flag to true and re-acquire the stream.
        Compare header data of pre-flip and post-flip cases.

@SYMTestExpectedResults
        Read/write buffer pairs are always identical.
        Flipped stream has width and height switched and stride is different
 **/ 
void CTestNativeStream::GRAPHICS_OPENWFC_NATIVESTREAM_0142L()
    {    
    _LIT(KDisplayChannelLdd, "display0.ldd");
    
    RHeap* mainHeap = COpenWfcStreamMap::InstanceL().GetMainHeap();
    TRAPD(err, GrowCleanupStackL());
    ASSERT_TRUE(err == KErrNone);
    
    CleanupStack::PushL(TCleanupItem(PopHeap, User::SwitchHeap(mainHeap)));

    err = User::LoadLogicalDevice(KDisplayChannelLdd);
    RDisplayChannel testChannel;
    User::LeaveIfError(testChannel.Open(0));
    
    TSurfaceId testId;
    SymbianStreamType ns;
    TUidPixelFormat testGuids[] = {EUidPixelFormatARGB_8888_PRE, EUidPixelFormatRGB_888, EUidPixelFormatRGB_565, EUidPixelFormatRGB_332};
    TInt size = sizeof(testGuids) / sizeof(testGuids[0]);
    TUint8 *pWriteBuffer = NULL;
    khronos_int32_t writeBuffer;
    TUint8 *pReadBuffer = NULL;
    khronos_int32_t readBuffer;

    TInt halStride = 0;
    TInt halMode = 0;

    // For header variables before flipping
    TInt32 preFlipWidth = 0;
    TInt32 preFlipHeight = 0;
    TInt32 preFlipStreamStride = 0;
    TUidPixelFormat preFlipStreamFormat = EUidPixelFormatUnknown;
    TInt32 preFlipStreamPixelSize = 0;
    
    // For header variables after flipping
    TInt32 width = 0;
    TInt32 height = 0;
    TInt32 streamStride = 0;
    TUidPixelFormat streamFormat = EUidPixelFormatUnknown;
    TInt32 streamPixelSize = 0;
    
    for (TInt ii = 0; ii < size; ii++)
        {
        HAL::Get(0, HALData::EDisplayMode, halMode);
        
        testId.iInternal[TSurfaceId::TScreenSurfaceUsage::EScreenField] = 0;      // Screen number
        testId.iInternal[TSurfaceId::TScreenSurfaceUsage::EHalField] = halMode;          // Rotation and hal mode index
        testId.iInternal[TSurfaceId::TScreenSurfaceUsage::ETypeGuidField] = testGuids[ii]; //May be zero for non-GCE modes
        testId.iInternal[TSurfaceId::TScreenSurfaceUsage::ETypeClassField] = ((TUint32)(TSurfaceId::EScreenSurface) << TSurfaceId::TScreenSurfaceUsage::ETypeClassShift);  // Type
                        
        err = SymbianStreamAcquire(&testId, &ns);
        ASSERT_TRUE(err == KErrNone);
        ASSERT_TRUE(ns);
        
        err = SymbianStreamAcquireWriteBuffer(ns,&writeBuffer);
        ASSERT_TRUE(err == KErrNone);
        
        err = SymbianStreamGetBufferPointer(ns,writeBuffer,reinterpret_cast<void**>(&pWriteBuffer));
        ASSERT_TRUE(err == KErrNone);
        ASSERT_NOT_NULL(pWriteBuffer);
        
        err = SymbianStreamReleaseWriteBuffer(ns,writeBuffer);
        ASSERT_TRUE(err == KErrNone);

        err = SymbianStreamAcquireReadBuffer(ns,&readBuffer);
        ASSERT_TRUE(err == KErrNone);
                    
        err = SymbianStreamGetBufferPointer(ns,readBuffer,reinterpret_cast<void**>(&pReadBuffer));
        ASSERT_TRUE(err == KErrNone);
        ASSERT_NOT_NULL(pReadBuffer);

        err = SymbianStreamReleaseReadBuffer(ns,readBuffer);
        ASSERT_TRUE(err == KErrNone);
        
        ASSERT_SAME(pWriteBuffer, pReadBuffer);
        
        *pWriteBuffer = 0xFF;
        ASSERT_TRUE(*pReadBuffer == 0xFF);
        
        halStride = halMode;
        HAL::Get(0, HALData::EDisplayOffsetBetweenLines, halStride);

        SymbianStreamGetHeader(ns, &preFlipWidth, &preFlipHeight, &preFlipStreamStride, &preFlipStreamFormat, &preFlipStreamPixelSize);
        ASSERT_TRUE(preFlipStreamStride == halStride);

        // Now the flipping test
        halMode = halMode^TSurfaceId::TScreenSurfaceUsage::EHalFlippedFlag;
        testId.iInternal[TSurfaceId::TScreenSurfaceUsage::EHalField] = halMode; 
                
        err = SymbianStreamAcquire(&testId, &ns);
        ASSERT_TRUE(err == KErrNone);
        ASSERT_TRUE(ns);

        halStride = halMode;
        HAL::Get(0, HALData::EDisplayOffsetBetweenLines, halStride);
        SymbianStreamGetHeader(ns, &width, &height, &streamStride, &streamFormat, &streamPixelSize);
        
        ASSERT_EQUALS(preFlipWidth, height);
        ASSERT_EQUALS(preFlipHeight, width);
        ASSERT_EQUALS(preFlipStreamFormat, streamFormat);
        ASSERT_TRUE(streamStride == halStride);
        ASSERT_EQUALS(preFlipStreamPixelSize, streamPixelSize); 
            
        SymbianStreamGetHeader(ns, NULL, NULL, NULL, NULL, NULL);
        // Clearing the flip flag
        halMode = halMode&~TSurfaceId::TScreenSurfaceUsage::EHalFlippedFlag;
        
        SymbianStreamRemoveReference(ns);     

        *pWriteBuffer = NULL;
        pWriteBuffer = NULL;
        *pReadBuffer = NULL;
        pReadBuffer = NULL;
        }
    
    testChannel.Close();
    CleanupStack::PopAndDestroy(); // switch the heap back to current thread's one
    }