kerneltest/e32test/pccd/t_media.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Wed, 09 Jun 2010 11:10:19 +0300
branchRCL_3
changeset 147 bbf8bed59bcb
parent 33 0173bcd7697c
permissions -rw-r--r--
Revision: 201023 Kit: 2010123

// Copyright (c) 1996-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "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:
// e32test\pccd\t_media.cpp
// Test the Compact Flash card (ATA) media driver
// 
//

#include <e32test.h>
#include <e32svr.h>
#include "u32std.h"
#include "../misc/prbs.h"

const TInt KSectorSize=512;

LOCAL_D RTest test(_L("T_MEDIA"));
LOCAL_D TBusLocalDrive TheDrive;
LOCAL_D TBool MediaChange=EFalse;
LOCAL_D TUint Seed[2];

LOCAL_D TUint8 Background1[64*KSectorSize];
LOCAL_D TUint8 Background2[64*KSectorSize];
LOCAL_D TUint8 Foreground1[64*KSectorSize];
LOCAL_D TUint8 Foreground2[64*KSectorSize];
LOCAL_D TUint8 VerifyBuffer[64*KSectorSize];

inline TUint RoundDownToSector(TUint aPos)
	{ return aPos&~0x1ff; }
inline TUint RoundUpToSector(TUint aPos)
	{ return (aPos+0x1ff)&~0x1ff; }

LOCAL_C void TestPattern(TUint8* aBuf, TInt aLength)
	{
	while(aLength--)
		*aBuf++=(TUint8)Random(Seed);
	}

LOCAL_C void Write(TUint aPos, TInt aLength, const TUint8* aBuffer)
	{
	TPtrC8 p(aBuffer,aLength);
	TInt r=TheDrive.Write(aPos,p);
	if (r!=KErrNone)
		{
		test.Printf(_L("Write failed with error %d\n"),r);
		test.Printf(_L("Pos=%08x, Length=%x\n"),aPos,aLength);
		test(0);
		}
	}

LOCAL_C void DebugDump(TUint aPos, TInt aLength, const TUint8* aBuf, const TDesC& aTitle)
	{
	RDebug::Print(aTitle);
	TUint end=aPos+aLength;
	TInt i;
	TInt j=0;
	while(aPos<end)
		{
		TBuf<80> buf;
		buf.NumFixedWidthUC(aPos,EHex,8);
		buf+=_L(": ");
		for (i=0; i<16; i++)
			{
			buf.AppendNumFixedWidthUC(aBuf[j+i],EHex,2);
			buf+=_L(" ");
			}
		RDebug::Print(buf);
		aPos+=16;
		j+=16;
		if ((aPos&(KSectorSize-1))==0)
			RDebug::Print(_L(""));
		}
	}

LOCAL_C void Verify(TUint aPos, TInt aLength, const TUint8* aRef)
	{
	TPtr8 p(VerifyBuffer,0,64*KSectorSize);
	TInt r=TheDrive.Read(aPos,aLength,p);
	if (r!=KErrNone)
		{
		test.Printf(_L("Read failed with error %d\n"),r);
		test.Printf(_L("Pos=%08x, Length=%x\n"),aPos,aLength);
		test(0);
		}
	if (p.Length()!=aLength)
		{
		test.Printf(_L("Incorrect length after read: Was %08x Expected %08x\n"),p.Length(),aLength);
		test.Printf(_L("Pos=%08x, Length=%x\n"),aPos,aLength);
		test(0);
		}
	r=Mem::Compare(VerifyBuffer,aLength,aRef,aLength);
	if (r==0)
		return;
	TInt i=0;
	while(i<aLength && VerifyBuffer[i]==aRef[i])
		i++;
	test.Printf(_L("Verify error: aPos=%08x, aLength=%08x\n"),aPos,aLength);
	test.Printf(_L("First difference at offset %x\n"),i);
	test.Printf(_L("Press <ENTER> for debug dump "));
	TInt k=test.Getch();
	if (k==EKeyEnter)
		{
		DebugDump(aPos,aLength,VerifyBuffer,_L("Actual:"));
		DebugDump(aPos,aLength,aRef,_L("Expected:"));
		}
	test(0);
	}

LOCAL_C void DoTest(TUint aBasePos, TInt anOffset, TInt aSize)
	{
	TBuf<80> buf;
	buf.Format(_L("Offset %3x Size %04x"),anOffset,aSize);
	test.Next(buf);
	TUint block1=aBasePos;
	TUint block2=aBasePos+64*KSectorSize;
	TUint totalSectorSize=RoundUpToSector(anOffset+aSize);
	TestPattern(Background1,totalSectorSize);
	TestPattern(Background2,totalSectorSize);
	TestPattern(Foreground1,totalSectorSize);
	TestPattern(Foreground2,totalSectorSize);
	Write(block1,totalSectorSize,Background1);
	Write(block2,totalSectorSize,Background2);
	Verify(block1,totalSectorSize,Background1);
	Verify(block2,totalSectorSize,Background2);
	Write(block1+anOffset,aSize,Foreground1);
	Write(block2+anOffset,aSize,Foreground2);
	Mem::Copy(Background1+anOffset,Foreground1,aSize);
	Mem::Copy(Background2+anOffset,Foreground2,aSize);
	Verify(block1,totalSectorSize,Background1);
	Verify(block2,totalSectorSize,Background2);
	}

GLDEF_C TInt E32Main()
	{
	Seed[0]=0xadf85458;
	Seed[1]=0;
	test.Title();
	
	TChar driveToTest;

	// Get the list of drives
	TDriveInfoV1Buf diBuf;
	UserHal::DriveInfo(diBuf);
	TDriveInfoV1 &di=diBuf();
	TInt driveCount = di.iTotalSupportedDrives;
	
	test.Printf(_L("\nDRIVES USED AT PRESENT :\r\n"));
	for (TInt i=0; i < driveCount; i++)
		{
		TBool flag=EFalse;
		RLocalDrive d;
		TInt r=d.Connect(i,flag);
		//Not all the drives are used at present
		if (r == KErrNotSupported)
			continue;

		test.Printf(_L("%d : DRIVE NAME  :%- 16S\r\n"), i, &di.iDriveName[i]);
		}

	test.Printf(_L("\n<<<Hit required drive number to continue>>>\r\n"));

	driveToTest=(TUint)test.Getch();
	
	TInt driveNumber=((TUint)driveToTest) - '0';

	TBuf<0x100> buf;
	buf.Format(_L("Connect to local drive (%d)"),driveNumber);
	test.Start(buf);
	
	TInt r=TheDrive.Connect(driveNumber,MediaChange);
	test(r==KErrNone);
	
	test.Next(_L("Get capabilities"));
	TLocalDriveCapsV2 driveCaps;
	TPckg<TLocalDriveCapsV2> capsPckg(driveCaps);
	r=TheDrive.Caps(capsPckg);
	test(r==KErrNone);
	TUint driveSize=I64LOW(driveCaps.iSize);
	test.Printf(_L("Drive size       = %08x (%dK)\n"),driveSize,driveSize>>10);
	test.Printf(_L("Media type       = %d\n"),driveCaps.iType);
	test.Printf(_L("Connection Bus   = %d\n"),driveCaps.iConnectionBusType);
	test.Printf(_L("Drive attributes = %08x\n"),driveCaps.iDriveAtt);
	test.Printf(_L("Media attributes = %08x\n"),driveCaps.iMediaAtt);
	test.Printf(_L("Base address     = %08x\n"),driveCaps.iBaseAddress);
	test.Printf(_L("File system ID   = %08x\n"),driveCaps.iFileSystemId);
	test.Printf(_L("Hidden sectors   = %08x\n"),driveCaps.iHiddenSectors);
	test.Printf(_L("Press any key...\n"));
	test.Getch();
	TUint basePos=RoundDownToSector(driveSize)-128*KSectorSize;
	test.Printf(_L("Base position    = %08x\n"),basePos);

	TInt offset;
	TInt size;
	for (size=KSectorSize/4; size<=23*KSectorSize/2; size+=KSectorSize/4)
		{
		for (offset=0; offset<KSectorSize; offset+=KSectorSize/2)
			{
			DoTest(basePos,offset,size);
			}
		}

	for (size=12*KSectorSize; size<=33*KSectorSize; size+=KSectorSize/2)
		{
		for (offset=0; offset<KSectorSize; offset+=KSectorSize/2)
			{
			DoTest(basePos,offset,size);
			}
		}

	buf.Format(_L("Disconnect from local drive (%d)"),driveNumber);
	test.Next(buf);
	TheDrive.Disconnect();
	test.End();
	return 0;
	}