fbs/fontandbitmapserver/trasterizer/test/trasterizer.cpp
author William Roberts <williamr@symbian.org>
Tue, 20 Apr 2010 16:24:43 +0100
branchNewGraphicsArchitecture
changeset 34 76efc8f9f7b4
parent 0 5d03bc08d59c
permissions -rw-r--r--
Apply Faisal's first patch from Bug 2354 - First resolve some the the bit rot in graphics MCL to get it to compile, then fix some performance issues in OpenWF

// Copyright (c) 2008-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:
//

#include "trasterizer.h"
#include "testutils.h"
#include "examplerasterizer.h"
#include <fbs.h>
#include <gdi.h>
#include <s32mem.h>
#include <e32math.h>
#include <e32svr.h>
#include <u32hal.h>

const TDisplayMode KDisplayModes[] = {EGray256, EColor64K, EColor16MU, EColor16MAP};
const TInt KNumDisplayModes = sizeof(KDisplayModes)/sizeof(KDisplayModes[0]);
const TRgb KStripeColours[3] = {TRgb(255,255,0), TRgb(255,0,255), TRgb(0,255,255)};

CTRasterizer::CTRasterizer()
	{
	SetTestStepName(KTRasterizerStep);
	}

/**
@SYMTestCaseID		
	GRAPHICS-FBSERV-RASTERIZER-001

@SYMTestPriority
	High

@SYMPREQ
	PREQ2096

@SYMREQ
	REQ10860
	REQ10861 
	
@SYMTestCaseDesc
	Test that the example rasterizer dll "fbsrasterizer_test.dll" has been loaded.

@SYMTestActions	
	On the emulator check that the rasterizer setting is "fbsrasterizer_test.dll".
	Get a pointer to the rasterizer.

@SYMTestExpectedResults
	"fbsrasterizer_test.dll" has been loaded successfully
*/
void CTRasterizer::TestLoadRasterizerDllL()
	{
	INFO_PRINTF1(_L("TestLoadRasterizerDll"));
#ifdef __WINS__
	_LIT8(KTestRasterizerName, "fbsrasterizer_test.dll");
	TUint8* rasterizerSetting = NULL;
	UserSvr::HalFunction(EHalGroupEmulator, EEmulatorHalStringProperty, (TAny*)"FBSRASTERIZER_DLL", &rasterizerSetting);
	TESTL(rasterizerSetting != NULL);
	TESTL(TPtrC8(rasterizerSetting).CompareF(KTestRasterizerName) == 0);
#endif
	GetExampleRasterizerL();
	}

/**
@SYMTestCaseID		
	GRAPHICS-FBSERV-RASTERIZER-002

@SYMTestPriority
	High

@SYMPREQ
	PREQ2096

@SYMREQ
	REQ10860
	REQ10861 
	
@SYMTestCaseDesc
	Test that a scanline retrieved from the example rasterizer is what we expect.

@SYMTestActions		
	Create an extended bitmap and set its extended data to three colours followed by a TUint8.
	The example rasterizer interprets this data as the three colours in a tricolour flag drawn
	with either horizontal or vertical stripes depending on the value of the TUint8.	
	Draw a scanline of the extended bitmap by calling:
		- CFbsRasterizer::BeginBitmap()
		- CFbsRasterizer::ScanLine()
		- CFbsRasterizer::EndBitmap()

@SYMTestExpectedResults
	The scan line returned by the example rasterizer should be exactly the same as the
	corresponding scanline in the test data.
*/
void CTRasterizer::TestScanLineL()
	{	
	TBuf<32> displayModeDes(TestUtils::DisplayModeToString(iDisplayMode)); 
	INFO_PRINTF2(_L("TestScanLine, display mode %S"), &displayModeDes);
	
	const TInt KNumBeginBitmapCalls = 5;
	
	// Get the example rasterizer
	GetExampleRasterizerL();	
	
	// Test for both stripe styles, 0 for vertical stripes and 1 for horizontal stripes
	for (TInt stripeStyle = 0; stripeStyle < 2; stripeStyle++)
		{
		if (stripeStyle == 0)
			{
			INFO_PRINTF1(_L("... Vertical stripes"));
			}
		else
			{
			INFO_PRINTF1(_L("... Horizontal stripes"));
			}
				
		const TSize sizeInPixels(64,32);
		CFbsBitmap* extendedBitmap = CreateExtendedBitmapLC(KStripeColours, stripeStyle, sizeInPixels);					
			
		// Begin scanline access to the extended bitmap
		CFbsRasterizer::TBitmapDesc bitmapDesc;
		TInt64 bitmapId = extendedBitmap->SerialNumber();
		bitmapDesc.iSizeInPixels = sizeInPixels;
		bitmapDesc.iDispMode = iDisplayMode;
		bitmapDesc.iDataType = KUidExampleExtendedBitmap;
		bitmapDesc.iData = extendedBitmap->DataAddress();
		bitmapDesc.iDataSize = extendedBitmap->DataSize();
		extendedBitmap->BeginDataAccess();
		// test calling BeginBitmap more than once as each additional call should just increment as reference count
		for (TInt i = 0; i < KNumBeginBitmapCalls; i++)
			{			
			iRasterizer->BeginBitmap(bitmapId, bitmapDesc, NULL);
			}				
		
		TInt scanlineLength;
		TPoint startPixel;
		TRgb bufferColour;
		// The width or height of a stripe in the tricolour flag that the extended bitmap rasterizer draws
		TInt stripeSize = (stripeStyle == 0) ? sizeInPixels.iWidth/3 : sizeInPixels.iHeight/3; 				
		for (TInt h = 0; h < sizeInPixels.iHeight; h++)
			{
			// Get a full scanline from the extended bitmap using the rasterizer	
			startPixel = TPoint(0, h);
			scanlineLength = sizeInPixels.iWidth;
			const TUint8* scanline = reinterpret_cast<const TUint8*>(iRasterizer->ScanLine(bitmapId, startPixel, scanlineLength));
			TESTL(scanline != NULL);								
			
			// Make sure each pixel in the scanline is the colour that we expect						
			TRgb compareColour;
			
			for (TInt w = 0; w < scanlineLength; w++)
				{																			
				if (((stripeStyle == 0) && (w < stripeSize)) 
					|| ((stripeStyle == 1) && (h < stripeSize)))
					{					
					compareColour = KStripeColours[0]; // left or top
					}
				else if (((stripeStyle == 0) && (w >= sizeInPixels.iWidth-stripeSize)) 
					|| ((stripeStyle == 1) && (h >= sizeInPixels.iHeight-stripeSize)))
					{
					compareColour = KStripeColours[2]; // right or bottom
					}
				else
					{
					compareColour = KStripeColours[1]; // middle
					}					
				bufferColour = ExtractRgb(scanline, w, iDisplayMode);
				if (((iDisplayMode != EGray256) && (bufferColour != compareColour))
					|| ((iDisplayMode == EGray256) && (bufferColour != TRgb::Gray256(compareColour.Gray256()))))
					{					
					INFO_PRINTF3(_L("Pixel comparison failed for pixel %d,%d"), w, h);
					TEST(EFalse);
					}			
				}							
			}
		
		// End scanline access to the extended bitmap
		for (TInt i = 0; i < KNumBeginBitmapCalls; i++)
			{
			iRasterizer->EndBitmap(bitmapId);			
			}
		extendedBitmap->EndDataAccess();
		
		// Clean up
		CleanupStack::Pop(1, extendedBitmap);
		}
	}

/**
@SYMTestCaseID		
	GRAPHICS-FBSERV-RASTERIZER-003

@SYMTestPriority
	High

@SYMPREQ
	PREQ2096

@SYMREQ
	REQ10860
	REQ10861 
	
@SYMTestCaseDesc
	Test that we can get scanlines from 3 extended bitmaps simultaneously using the example rasterizer.

@SYMTestActions	
	Load a normal bitmap.	
	Create 3 extended bitmaps and set their extended data to be vertically striped colours.	
	Draw a scanline of each extended bitmap by calling:
		- CFbsRasterizer::BeginBitmap() for each extended bitmap
		- CFbsRasterizer::ScanLine() for each extended bitmap
		- CFbsRasterizer::EndBitmap() for each extended bitmap
	Compare the first pixel of each scanline and the last pixel of each scanline against the
	expected colour.

@SYMTestExpectedResults
	The rasterizer should be able to handle rasterizing up to 3 extended bitmaps at once,
	all three scanlines should be returned with no errors.
	The colours compared should match exactly.
*/
void CTRasterizer::TestMultipleScanLinesL()
	{
	TBuf<32> displayModeDes(TestUtils::DisplayModeToString(iDisplayMode)); 
	INFO_PRINTF2(_L("TestMultipleScanLines, vertical stripes, display mode %S"), &displayModeDes);
	
	const TInt KNumBitmaps = 3;
	const TInt KBaseSize = 10;			
	RArray<CFbsBitmap*> bitmaps;
	CleanupClosePushL(bitmaps);	
	
	// Get the example rasterizer
	GetExampleRasterizerL();
	
	// Create the three differently sized extended bitmaps each coloured with vertical stripes 
	for (TInt i = 0; i < KNumBitmaps; i++)
		{
		const TSize KSizeInPixels = TSize(KBaseSize*(i+1), KBaseSize*(i+1));			
		CFbsBitmap* extendedBitmap = CreateExtendedBitmapLC(KStripeColours, EFalse, KSizeInPixels);		
		bitmaps.AppendL(extendedBitmap);			
		}
	
	// Begin scanline access to the extended bitmaps
	for (TInt i = 0; i < KNumBitmaps; i++)
		{
		bitmaps[i]->BeginDataAccess();
		
		CFbsRasterizer::TBitmapDesc bitmapDesc;
		TInt64 bitmapId = bitmaps[i]->SerialNumber();
		bitmapDesc.iSizeInPixels = bitmaps[i]->SizeInPixels();
		bitmapDesc.iDispMode = iDisplayMode;
		bitmapDesc.iDataType = KUidExampleExtendedBitmap;
		bitmapDesc.iData = bitmaps[i]->DataAddress();
		bitmapDesc.iDataSize = bitmaps[i]->DataSize();	
		iRasterizer->BeginBitmap(bitmapId, bitmapDesc, NULL);
		}
	
	// Get scanlines from each extended bitmap and check that the first and last pixel in each is the correct colour
	for (TInt h = 0; h < KBaseSize; h++)
		{
		for (TInt i = 0; i < KNumBitmaps; i++)
			{	
			const TUint8* scanline = reinterpret_cast<const TUint8*>(iRasterizer->ScanLine(bitmaps[i]->SerialNumber(), TPoint(0,h), KBaseSize*(i+1)));			
			TESTL(scanline != NULL);					
			
			// Test that the first pixel is the correct colour (the first pixel in each scanline should be KStripeColours[0]
			// as the bitmaps should have all been drawn with vertical stripes with the first stripe being KStripeColours[0]
			TRgb bufferColour = ExtractRgb(scanline, 0, iDisplayMode);			
			if (((iDisplayMode != EGray256) && (bufferColour != KStripeColours[0]))
				|| ((iDisplayMode == EGray256) && (bufferColour != TRgb::Gray256(KStripeColours[0].Gray256()))))	
				{
				INFO_PRINTF3(_L("Pixel comparison failed for bitmap %d, pixel 0,%d"), i, h);
				TEST(EFalse);
				}	
			// The last pixel should be KStripeColours[2]							
			bufferColour = ExtractRgb(scanline, (KBaseSize*(i+1))-1, iDisplayMode);
			if (((iDisplayMode != EGray256) && (bufferColour != KStripeColours[2]))
				|| ((iDisplayMode == EGray256) && (bufferColour != TRgb::Gray256(KStripeColours[2].Gray256()))))	
				{
				INFO_PRINTF4(_L("Pixel comparison failed for bitmap %d, pixel %d,%d"), i, (KBaseSize*(i+1))-1, h);
				TEST(EFalse);
				}
			}
		}
		
	// Unregister each extended bitmap
	for (TInt i = 0; i < KNumBitmaps; i++)
		{					
		iRasterizer->EndBitmap(bitmaps[i]->SerialNumber());
		bitmaps[i]->EndDataAccess();
		}
	
	// Cleanup
	CleanupStack::PopAndDestroy(KNumBitmaps+1);
	}

/**
@SYMTestCaseID		
	GRAPHICS-FBSERV-RASTERIZER-004

@SYMTestPriority
	High

@SYMPREQ
	PREQ2096

@SYMREQ
	REQ10860
	REQ10861 
	
@SYMTestCaseDesc
	Test that the rasterizer returns portions of scanline correctly
	when a region of interest is specified.

@SYMTestActions	
	Create an extended bitmap.
	Rasterize it specifying a region of interest that has three regions that roughly
	form a squared off C shape.		

@SYMTestExpectedResults
	Check that scanlines returned are correct in the specified region of interest.
	(Anything outside the region of interest does not matter).
*/
void CTRasterizer::TestRegionOfInterestL()
	{
	TBuf<32> displayModeDes(TestUtils::DisplayModeToString(iDisplayMode)); 
	INFO_PRINTF2(_L("TestRegionOfInterest, horizontal stripes, display mode %S"), &displayModeDes);
	
	// Get the example rasterizer
	GetExampleRasterizerL();	
	
	// Create an extended bitmap with horizontal stripes
	const TSize sizeInPixels(24,24);
	CFbsBitmap* extendedBitmap = CreateExtendedBitmapLC(KStripeColours, ETrue, sizeInPixels);		
	
	// Create a region of interest in the shape of a square with a hole in the middle
	TRegionFix<4> regionOfInterest;
	const TInt KStripeSize = sizeInPixels.iHeight/3;
	const TInt KEdgeOffset = 2;	
	regionOfInterest.AddRect(TRect(TPoint(KEdgeOffset,KEdgeOffset), TSize(sizeInPixels.iWidth-(KEdgeOffset*2),KStripeSize)));	// top of the square
	regionOfInterest.AddRect(TRect(TPoint(KEdgeOffset,sizeInPixels.iHeight-KEdgeOffset-KStripeSize), TSize(sizeInPixels.iWidth-(KEdgeOffset*2),KStripeSize)));	// bottom of the square
	regionOfInterest.AddRect(TRect(TPoint(KEdgeOffset,KEdgeOffset+KStripeSize), TSize(KStripeSize,sizeInPixels.iHeight-(KEdgeOffset*2)-(KStripeSize*2))));	// left of the square
	regionOfInterest.AddRect(TRect(TPoint(sizeInPixels.iWidth-KEdgeOffset-KStripeSize,KEdgeOffset+KStripeSize), TSize(KStripeSize,sizeInPixels.iHeight-(KEdgeOffset*2)-(KStripeSize*2)))); // right of the square		
	TEST(regionOfInterest.CheckError() == EFalse);
		
	// Begin scanline access to the extended bitmap
	CFbsRasterizer::TBitmapDesc bitmapDesc;
	TInt64 bitmapId = extendedBitmap->SerialNumber();
	bitmapDesc.iSizeInPixels = sizeInPixels;
	bitmapDesc.iDispMode = iDisplayMode;
	bitmapDesc.iDataType = KUidExampleExtendedBitmap;
	bitmapDesc.iData = extendedBitmap->DataAddress();
	bitmapDesc.iDataSize = extendedBitmap->DataSize();
	extendedBitmap->BeginDataAccess();
	iRasterizer->BeginBitmap(bitmapId, bitmapDesc, &regionOfInterest);	
	
	TInt scanlineLength;
	TPoint startPixel;
	TRgb bufferColour;
	 			
	for (TInt h = 0; h < sizeInPixels.iHeight; h++)
		{
		// Get scanlines from the extended bitmap using the rasterizer	
		startPixel = TPoint(0, h);
		scanlineLength = sizeInPixels.iWidth;
		const TUint8* scanline = reinterpret_cast<const TUint8*>(iRasterizer->ScanLine(bitmapId, startPixel, scanlineLength));
		TESTL(scanline != NULL);								
		
		// Make sure each pixel in the scanline is the colour that we expect					
		for (TInt w = 0; w < sizeInPixels.iWidth; w++)
			{										
			bufferColour = ExtractRgb(scanline, w, iDisplayMode);
			
			// Pixels that lie inside the region should be set to a flag stripe colour, pixels
			// outside of the region should be white			
			if (regionOfInterest.Contains(TPoint(w,h)))
				{				
				if (((iDisplayMode != EGray256) && (bufferColour != KStripeColours[h/KStripeSize]))
					|| ((iDisplayMode == EGray256) && (bufferColour != TRgb::Gray256(KStripeColours[h/KStripeSize].Gray256()))))
					{
					INFO_PRINTF3(_L("Pixel comparison failed for pixel %d,%d"), w, h);
					TEST(EFalse);
					}			
				}
			else
				{
				if (bufferColour != KRgbWhite)
					{
					INFO_PRINTF3(_L("Pixel comparison failed for pixel %d,%d"), w, h);
					TEST(EFalse);
					}
				}
			}		
		}
	
	// End scanline access to the extended bitmap	
	iRasterizer->EndBitmap(bitmapId);
	extendedBitmap->EndDataAccess();
	
	// Clean up
	CleanupStack::Pop(1, extendedBitmap);
	}

/**
Override of base class pure virtual
Lists the tests to be run
*/
TVerdict CTRasterizer::doTestStepL()
	{
	TestLoadRasterizerDllL();
	
	// Tests that require testing in multiple display modes
	for (TInt i = 0; i < KNumDisplayModes; i++)
		{
		iDisplayMode = KDisplayModes[i];		
				
		TestScanLineL();
		TestMultipleScanLinesL();
		TestRegionOfInterestL();
		}
	
	return TestStepResult();
	}

/** Helper function for getting a TRgb colour value from a buffer given a pixel offset and 
a display mode.
@param aBuffer A buffer to extract the TRgb from
@param aPixelOffset The pixel position in the buffer to extract the TRgb from
@param aDispMode The display mode of the buffer
@return The TRgb value found at the pixel position specified
 */
TRgb CTRasterizer::ExtractRgb(const TUint8* aBuffer, TInt aPixelOffset, TDisplayMode aDispMode)
	{
	switch (aDispMode)
		{
	case EGray2:
		{
		TUint8 byte = *(aBuffer + (aPixelOffset >> 3));
		if (byte & (1 << (aPixelOffset & 7)))
			return KRgbWhite;
		return KRgbBlack;
		}
	case EGray4:
		{
		TUint8 byte = *(aBuffer + (aPixelOffset >> 2));
		byte >>= ((aPixelOffset & 3) << 1);
		return TRgb::Gray4(byte & 3);
		}
	case EGray16:
		{
		TUint8 byte = *(aBuffer + (aPixelOffset >> 1));
		if (aPixelOffset & 1)
			byte >>= 4;
		return TRgb::Gray16(byte & 0xf);
		}
	case EGray256:
		return TRgb::Gray256(*(aBuffer + aPixelOffset));
	case EColor16:
		{
		TUint8 byte = *(aBuffer + (aPixelOffset >> 1));
		if (aPixelOffset & 1)
			byte >>= 4;
		return TRgb::Color16(byte & 0xf);
		}
	case EColor256:
		return TRgb::Color256(*(aBuffer + aPixelOffset));
	case EColor4K:
		{
		TUint16 doubleByte = *(((TUint16*)aBuffer) + aPixelOffset);
		return TRgb::Color4K(doubleByte & 0xfff);
		}
	case EColor64K:
		{
		TUint16 doubleByte = *(((TUint16*)aBuffer) + aPixelOffset);
		return TRgb::Color64K(doubleByte);
		}
	case EColor16M:
		{
		aBuffer += aPixelOffset * 3;
		TInt value = *aBuffer++;
		value |= *aBuffer++ << 8;
		value |= *aBuffer << 16;
		return TRgb::Color16M(value);
		}
	case ERgb:
		return *(((TRgb*)aBuffer) + aPixelOffset);
	case EColor16MU:
		{
		// Note this is | with 0xFF000000 to match the example rasterizer which sets
		// the alpha to 0xFF when drawing using EColor16MU
		return TRgb::Color16MU((*(((TUint32*)aBuffer) + aPixelOffset)) | 0xFF000000);
		}
	case EColor16MA:
		{
		return TRgb::Color16MA(*(((TUint32*)aBuffer) + aPixelOffset));
		}	
	case EColor16MAP:
		{
		return TRgb::_Color16MAP(*(((TUint32*)aBuffer) + aPixelOffset));
		}	
	default:
		break;
		};
	return KRgbBlack;
	}

/** Helper function for creating an extended bitmap. 
@param aColours A pointer to an array of three colours to use when creating the extended bitmap, 
these colours define the colours used when drawing the stripes in the extended bitmap.
@param aHorizontalStripe ETrue for horizontal stripes, EFalse for vertical stripes.
@param aSizeInPixels The size of the extended bitmap to create.
@return A pointer to a newly created extended bitmap that has been pushed on to the cleanup stack.
 */
CFbsBitmap* CTRasterizer::CreateExtendedBitmapLC(const TRgb* aColours, TBool aHorizontalStripe, TSize aSizeInPixels)
	{	
	// Set up the buffer containing the three TRgb values and one TUint8 value that the rasterizer expects
	TInt dataSize = (sizeof(TRgb)*3) + sizeof(TUint8); // estimate the size to be written
	TUint8* data = new (ELeave)TUint8[dataSize];
	CleanupArrayDeletePushL(data);
	RMemWriteStream writeStream;
	writeStream.Open(data, dataSize);
	CleanupClosePushL(writeStream);		
	writeStream << aColours[0]; 		
	writeStream << aColours[1];		
	writeStream << aColours[2];
	TUint8 stripe = aHorizontalStripe; // EFalse is vertical stripes, ETrue is horizontal stripes
	writeStream << stripe;
	dataSize = writeStream.Sink()->TellL(MStreamBuf::EWrite).Offset(); // get the actual size written
	CleanupStack::PopAndDestroy(&writeStream);					

	CFbsBitmap* extendedBitmap = new(ELeave) CFbsBitmap;
		
	TInt err = extendedBitmap->CreateExtendedBitmap(aSizeInPixels, 
													iDisplayMode,
													KUidExampleExtendedBitmap,
													data,
													dataSize);
	CleanupStack::PopAndDestroy(data);
	CleanupStack::PushL(extendedBitmap);
	TESTL(err == KErrNone);
			
	return extendedBitmap;
	}