persistentstorage/sql/TEST/t_sqlfserr.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Mon, 03 May 2010 14:09:14 +0300
changeset 17 55f2396f6d25
parent 15 3eacc0623088
child 13 211563e4b919
child 22 a7ba600cb39d
permissions -rw-r--r--
Revision: 201018 Kit: 201018

// Copyright (c) 2007-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 <e32test.h>
#include <bautils.h>
#include <sqldb.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h> 
#include "sqlite3.h"
#include "SqliteSymbian.h"

///////////////////////////////////////////////////////////////////////////////////////

RTest TheTest(_L("t_sqlfserr test"));
_LIT(KTestDir, "c:\\test\\");
_LIT(KTestDbName, "c:\\test\\t_fserr.db");
_LIT(KPrivateTestDbName, "c:\\private\\212A2C27\\t_fserr2.db");
_LIT(KSecureTestDbName, "c:[212A2C27]t_fserr3.db");

TFileName TheRmvMediaDbFileName;//The name of the file used for tests on a removable media
RFs TheFs;
RSqlDatabase TheDb;

//The next constants are used in the "blob write" test
const TInt KWriteCnt = 9;
const TInt KBlobSize = 397 * KWriteCnt;
_LIT(KAttachDb, "AttachDb");

//In order to be able to compile the test, the following variables are defined (used inside the OS porting layer, when _SQLPROFILER macro is defined)
#ifdef _SQLPROFILER
TInt TheSqlSrvProfilerFileRead = 0;
TInt TheSqlSrvProfilerFileWrite = 0;
TInt TheSqlSrvProfilerFileSync = 0;
TInt TheSqlSrvProfilerFileSetSize = 0;
#endif

///////////////////////////////////////////////////////////////////////////////////////

TBool FileExists(const TDesC& aFileName)
	{
	TEntry entry;
	return TheFs.Entry(aFileName, entry) == KErrNone;
	}

void DestroyTestEnv()
	{
	TheDb.Close();
    (void)RSqlDatabase::Delete(KSecureTestDbName);
    (void)RSqlDatabase::Delete(KPrivateTestDbName);
	(void)RSqlDatabase::Delete(KTestDbName);
	(void)RSqlDatabase::Delete(TheRmvMediaDbFileName);
	TheFs.Close();
	sqlite3SymbianLibFinalize();
	CloseSTDLIB();
	}

///////////////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////////////
//Test macros and functions
void Check(TInt aValue, TInt aLine)
	{
	if(!aValue)
		{
		DestroyTestEnv();
		TheTest(EFalse, aLine);
		}
	}
void Check(TInt aValue, TInt aExpected, TInt aLine)
	{
	if(aValue != aExpected)
		{
		DestroyTestEnv();
		RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
		TheTest(EFalse, aLine);
		}
	}
#define TEST(arg) ::Check((arg), __LINE__)
#define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)

void SqliteCheck(sqlite3* aDbHandle, TInt aValue, TInt aExpected, TInt aLine)
	{
	if(aValue != aExpected)
		{
		RDebug::Print(_L("*** SQLITE: Expected error: %d, got: %d\r\n"), aExpected, aValue);
		if(aDbHandle)
			{
			const char* errMsg = sqlite3_errmsg(aDbHandle);
			TPtrC8 ptr8((const TUint8*)errMsg, strlen(errMsg));
			TBuf<200> buf;
			buf.Copy(ptr8);
			RDebug::Print(_L("*** SQLITE error message: \"%S\"\r\n"), &buf);
			}
		DestroyTestEnv();
		TheTest(EFalse, aLine);
		}
	}
#define SQLITE_TEST(aDbHandle, aValue, aExpected) ::SqliteCheck(aDbHandle, aValue, aExpected, __LINE__)

///////////////////////////////////////////////////////////////////////////////////////

void SetupTestEnv()
    {
	TInt err = TheFs.Connect();
	TEST2(err, KErrNone);

	err = TheFs.MkDir(KTestDir);
	TEST(err == KErrNone || err == KErrAlreadyExists);

	sqlite3SymbianLibInit();
	}

TBool CheckRecord(const TDesC& aDbName, TInt aId, const TDesC& aExpectedName, TBool aOpenDb = ETrue)
	{
	if(aOpenDb)
		{
		TEST2(TheDb.Open(aDbName), KErrNone);
		}
	TBuf<64> sql;
	sql.Copy(_L("SELECT Name FROM A WHERE Id="));
	sql.AppendNum(aId);
	TSqlScalarFullSelectQuery q(TheDb);
	TBuf<20> name;
	TRAPD(err, (void)q.SelectTextL(sql, name));
	TEST2(err, KErrNone);
	if(aOpenDb)
		{
		TheDb.Close();
		}
	return name == aExpectedName;
	}

///////////////////////////////////////////////////////////////////////////////////////

/**
@SYMTestCaseID			SYSLIB-SQL-UT-3419
@SYMTestCaseDesc		Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
						The test creates a test database with one table, inserts one record.
						Then the test attempts to update the existing record while simulating file I/O failures.
						After each test iteration the database content is tested and is expected to be the same
						as it was before the test. RSqlDatabase::Exec() is used for the update operation.
@SYMTestPriority		High
@SYMTestActions			Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
@SYMTestExpectedResults The test must not fail
@SYMDEF					DEF103859
*/
void AlterDatabaseTest()
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	TInt err = TheDb.Create(KTestDbName);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
	TEST(err >= 0);
	TheDb.Close();
	err = KErrNotFound;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{
		TheTest.Printf(_L("%d \r"), cnt);		
		for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
			{
			//Preprocessing
			TEST2(TheDb.Open(KTestDbName), KErrNone);
			(void)TheDb.Exec(_L("DELETE FROM A WHERE Id=1"));
			err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
			TEST2(err, 1);
			//The test
			(void)TheFs.SetErrorCondition(fsError, cnt);
			err = TheDb.Exec(_L("UPDATE A SET Name='Name2' WHERE Id=1"));
			(void)TheFs.SetErrorCondition(KErrNone);
			if(err < 1)
				{
				TheDb.Close();//close the database to recover from the last error
				//check the database content - all bets are off in a case of an I/O error. 
				//The existing record might have been updated.
				TEST(CheckRecord(KTestDbName, 1, _L("Name")) || CheckRecord(KTestDbName, 1, _L("Name2")));
				}
			else
				{
				TEST2(err, 1);
				//check the database content has been modified by the operation. 
				TEST(CheckRecord(KTestDbName, 1, _L("Name2"), EFalse));
				TheDb.Close();
				}
			}
		}
	(void)TheFs.SetErrorCondition(KErrNone);
	TEST2(err, 1);
	//check the database content (transaction durability). 
	TEST(CheckRecord(KTestDbName, 1, _L("Name2")));
	err = RSqlDatabase::Delete(KTestDbName);
	TEST2(err, KErrNone);
	TheTest.Printf(_L("\r\n"));
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-3420
@SYMTestCaseDesc		Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
						The test creates a test database with one table, inserts one record.
						Then the test attempts to update the existing record while simulating file I/O failures.
						After each test iteration the database content is tested and is expected to be the same
						as it was before the test. RSqlStatement::Exec() is used for the update operation.
@SYMTestPriority		High
@SYMTestActions			Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
@SYMTestExpectedResults The test must not fail
@SYMDEF					DEF103859
*/
void AlterDatabaseTest2()
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	TInt err = TheDb.Create(KTestDbName);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
	TEST(err >= 0);
	TheDb.Close();
	err = KErrNotFound;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{
		TheTest.Printf(_L("%d \r"), cnt);		
		for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
			{
			//Preprocessing
			TEST2(TheDb.Open(KTestDbName), KErrNone);
			(void)TheDb.Exec(_L("DELETE FROM A WHERE Id=1"));
			err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
			TEST2(err, 1);
			//The test
			(void)TheFs.SetErrorCondition(fsError, cnt);
			RSqlStatement stmt;
			err = stmt.Prepare(TheDb, _L("UPDATE A SET Name='Name2' WHERE Id=1"));
			if(err == KErrNone)
				{
				err = stmt.Exec();	
				}
			(void)TheFs.SetErrorCondition(KErrNone);
			stmt.Close();
			if(err < 1)
				{
				TheDb.Close();//close the database to recover from the last error
				//check the database content - all bets are off in a case of an I/O error. 
				//The existing record might have been updated.
				TEST(CheckRecord(KTestDbName, 1, _L("Name")) || CheckRecord(KTestDbName, 1, _L("Name2")));
				}
			else
				{
				TEST2(err, 1);
				//check the database content has been modified by the operation. 
				TEST(CheckRecord(KTestDbName, 1, _L("Name2"), EFalse));
				TheDb.Close();
				}
			}
		}
	(void)TheFs.SetErrorCondition(KErrNone);
	TEST2(err, 1);
	//check the database content has been modified by the operation. 
	TEST(CheckRecord(KTestDbName, 1, _L("Name2")));
	err = RSqlDatabase::Delete(KTestDbName);
	TEST2(err, KErrNone);
	TheTest.Printf(_L("\r\n"));
	}

void CreateTestSecurityPolicy(RSqlSecurityPolicy& aSecurityPolicy)
    {
    TSecurityPolicy alwaysPassPolicy(TSecurityPolicy::EAlwaysPass);
    TInt err = aSecurityPolicy.Create(alwaysPassPolicy);
    TEST2(err, KErrNone);

    err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::ESchemaPolicy, alwaysPassPolicy);
    TEST2(err, KErrNone);
    err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::EWritePolicy, alwaysPassPolicy);
    TEST2(err, KErrNone);
    err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::EReadPolicy, alwaysPassPolicy);
    TEST2(err, KErrNone);
    }

/**
@SYMTestCaseID			SYSLIB-SQL-UT-3421
@SYMTestCaseDesc		Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
						The test creates a test database with one table, inserts one record.
						Then the test attempts to open the database while simulating file I/O failures.
						At the end of the test the database content is tested and is expected to be the same
						as it was before the test. RSqlStatement::Open() is used in the test.
@SYMTestPriority		High
@SYMTestActions			Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
@SYMTestExpectedResults The test must not fail
@SYMDEF					DEF103859
*/
void OpenDatabaseTest()
	{
    TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
    const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
    for(TInt k=0;k<KDbNameCnt;++k)
        {	
        TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);       
        (void)RSqlDatabase::Delete(dbName[k]);
        TInt err = KErrGeneral;
        if(k == (KDbNameCnt - 1))
            {
            RSqlSecurityPolicy policy;
            CreateTestSecurityPolicy(policy);
            err = TheDb.Create(dbName[k], policy);
            policy.Close();
            }
        else
            {
            err = TheDb.Create(dbName[k]);
            }
        TEST2(err, KErrNone);
        err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
        TEST(err >= 0);
        err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
        TEST2(err, 1);
        TheDb.Close();
    
        err = KErrNotFound;
        for(TInt cnt=1;err<KErrNone;++cnt)
            {		
            TheTest.Printf(_L("%d \r"), cnt);		
            for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
                {
                (void)TheFs.SetErrorCondition(fsError, cnt);
                err = TheDb.Open(dbName[k]);
                (void)TheFs.SetErrorCondition(KErrNone);
                if(err != KErrNone)
                    {
                    TheDb.Close();//close the database to recover from the last error
                    //check the database content is still the same as before the "open" call
                    TEST(CheckRecord(dbName[k], 1, _L("Name")));
                    }
                else
                    {
                    TEST2(err, KErrNone);
                    //check the database content is still the same as before the operation, without closing the database
                    TEST(CheckRecord(dbName[k], 1, _L("Name"), EFalse));
                    TheDb.Close();
                    }
                }
            }
        (void)TheFs.SetErrorCondition(KErrNone);
        TEST2(err, KErrNone);
        //check the database content is the same as before the operation, after reopening the database.
        TEST(CheckRecord(dbName[k], 1, _L("Name")));
        err = RSqlDatabase::Delete(dbName[k]);
        TEST2(err, KErrNone);
        TheTest.Printf(_L("\r\n"));
        }
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-3434
@SYMTestCaseDesc		Test for DEF104820 "SQL, RSqlDatabase::Create() does not delete the file if fails".
						Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
						Then the test attempts to create a database while simulating file I/O failures.
						When the test succeeds, the test verifies that the database file does exist.
@SYMTestPriority		High
@SYMTestActions			Test for DEF104820 "SQL, RSqlDatabase::Create() does not delete the file if fails".
						Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
@SYMTestExpectedResults The test must not fail
@SYMDEF					DEF103859
*/
void CreateDatabaseTest()
	{
	RSqlSecurityPolicy policy;
	CreateTestSecurityPolicy(policy);
	
	TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
	const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
	for(TInt k=0;k<KDbNameCnt;++k)
	    {
        TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);       
        TInt err = -1;
        for(TInt cnt=1;err<KErrNone;++cnt)
            {		
            TheTest.Printf(_L("%d \r"), cnt);		
            for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
                {
                //Ideally, the database should be deleted by the SQL server, if RSqlDatabase::Create() fails.
                //But SetErrorCondition() makes the error persistent, so the SQL server will fail to delete the file.
                //This is the reason, RSqlDatabase::Delete()to be used, before simulating file I/O error.
                (void)RSqlDatabase::Delete(dbName[k]);
                (void)TheFs.SetErrorCondition(fsError, cnt);
                err = (k == (KDbNameCnt - 1)) ? TheDb.Create(dbName[k], policy) : TheDb.Create(dbName[k]);
                (void)TheFs.SetErrorCondition(KErrNone);
                TheDb.Close();
                //If err != KErrNone, the database file should have been already deleted by the server and here is 
                //the place to check that. But since the file I/O failure simulation makes the file I/O error 
                //persistent, the file cannot be deleted by the server, because the "file delete" operation also fails.
                }
            }
        (void)TheFs.SetErrorCondition(KErrNone);
        TheDb.Close();
        TEST2(err, KErrNone);
        if( k != (KDbNameCnt - 1))
            {
            TEST(FileExists(dbName[k]));
            }
        err = RSqlDatabase::Delete(dbName[k]);
        TEST2(err, KErrNone);
        TheTest.Printf(_L("\r\n"));
	    }
	policy.Close();
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-3462
@SYMTestCaseDesc		Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
						The test creates a test database with one table, inserts one record.
						Then the test attempts to retrieve the existing record while simulating file I/O failures.
						After each iteration, the database content is tested, that it has not been modified by the operation.
@SYMTestPriority		High
@SYMTestActions			Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
@SYMTestExpectedResults The test must not fail
@SYMDEF					DEF105434
*/
void SelectRecordTest()
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	TInt err = TheDb.Create(KTestDbName);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER, Name TEXT)"));
	TEST(err >= 0);
	err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
	TEST2(err, 1);
	TheDb.Close();
	err = -1;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{		
		TheTest.Printf(_L("%d \r"), cnt);		
		err = TheDb.Open(KTestDbName);
		TEST2(err, KErrNone);
		RSqlStatement stmt;
		(void)TheFs.SetErrorCondition(KErrGeneral, cnt);
		err = stmt.Prepare(TheDb, _L("SELECT * FROM A WHERE Id=?"));
		if(err == KErrNone)
			{
			err = stmt.BindInt(0, 1);
			if(err == KErrNone)
				{
				err = stmt.Next();
				TEST(err == KSqlAtRow || err < 0);
				if(err == KSqlAtRow)
					{
					TInt id = stmt.ColumnInt(0);
					TEST2(id, 1);
					TPtrC name;
					err = stmt.ColumnText(1, name);
					TEST2(err, KErrNone);
					TEST(name == _L("Name"));
					}
				}
			}
		(void)TheFs.SetErrorCondition(KErrNone);
		stmt.Close();
		TheDb.Close();
		//check the database content is the same as before the operation
		TEST(CheckRecord(KTestDbName, 1, _L("Name")));
		}
	(void)TheFs.SetErrorCondition(KErrNone);
	TEST(err >= 0);
	TheDb.Close();
	//check the database content is the same as before the operation, after reopening the database.
	TEST(CheckRecord(KTestDbName, 1, _L("Name")));
	err = RSqlDatabase::Delete(KTestDbName);
	TEST2(err, KErrNone);
	TheTest.Printf(_L("\r\n"));
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-3463
@SYMTestCaseDesc		Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
						The test creates a test database with one table, inserts one record.
						Then the test attempts to insert another while simulating file I/O failures.
						After each iteration, the database content is tested, that it has not been modified by the operation.
						If the operation succeeds, the database content is tested again to check that the inserted record is there.
@SYMTestPriority		High
@SYMTestActions			Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
@SYMTestExpectedResults The test must not fail
@SYMDEF					DEF105434
*/
void InsertRecordTest()
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	TInt err = TheDb.Create(KTestDbName);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
	TEST(err >= 0);
	err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
	TEST2(err, 1);
	TheDb.Close();
	err = -1;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{		
		TheTest.Printf(_L("%d \r"), cnt);		
		err = TheDb.Open(KTestDbName);
		TEST2(err, KErrNone);
		RSqlStatement stmt;
		(void)TheFs.SetErrorCondition(KErrGeneral, cnt);
		err = stmt.Prepare(TheDb, _L("INSERT INTO A(Id,Name) VALUES(2, 'Name2')"));
		if(err == KErrNone)
			{
			err = TheDb.Exec(_L("BEGIN TRANSACTION"));
			if(err == KErrNone)
				{
				err = stmt.Exec();
				TEST(err == 1 || err < 0);
				if(err == 1)
					{
					err = TheDb.Exec(_L("COMMIT TRANSACTION"));
					}
				}
			}
		(void)TheFs.SetErrorCondition(KErrNone);
		stmt.Close();
		if(err < 1)
			{
			TheDb.Close();//close the database to recover from the last error
			//check that the database contains the "name" record that has been inserted before the file I/O failure test.
			TEST(CheckRecord(KTestDbName, 1, _L("Name")));
			}
		else
			{
			TEST2(err, 1);
			//check the database content has been modified by the operation, without closing the database.
			TEST(CheckRecord(KTestDbName, 1, _L("Name"), EFalse));
			TEST(CheckRecord(KTestDbName, 2, _L("Name2"), EFalse));
			TheDb.Close();
			}
		}
	(void)TheFs.SetErrorCondition(KErrNone);
	TEST2(err, 1);
	//check the database content (transaction durability).
	TEST(CheckRecord(KTestDbName, 1, _L("Name")));
	TEST(CheckRecord(KTestDbName, 2, _L("Name2")));
	(void)RSqlDatabase::Delete(KTestDbName);
	TheTest.Printf(_L("\r\n"));
	}

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//////////////////////////////////         Removable media robustness test      /////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

_LIT8(KNameColData, "A123456789012345678901234567890");
_LIT8(KUpdatedNameColData, "1234");

//TRemovableMediaTest class is used for testing the SQLITE behaviour when the database file is created on a removable media where
//the cluster size is bigger than the page size and in case of a power failure is not guaranteed that the content
//of the last updated cluster will be preserved.
class TRemovableMediaTest
	{
	enum {KTestRecCnt = 200};
	enum {KMinCachePageSize = 512};
	
public:	
	void Run();
	
private:	
	TInt GetRemovableMediaDriveNo();
	TInt ClusterSize(TInt aDriveNo);
	void CreateDatabase(TInt aDriveNo, TInt aCachePageSize);
	void CheckRecord(sqlite3_stmt* aStmt, TInt aRecId);
	void VerifyDatabase();
	void DoTest();

	};

//Returns the number of the removable media drive, KErrNotFound otherwise.
TInt TRemovableMediaTest::GetRemovableMediaDriveNo()
	{
	for(TInt driveNo=EDriveA;driveNo<=EDriveZ;++driveNo)
		{
		TDriveInfo driveInfo;
		TInt err = TheFs.Drive(driveInfo, driveNo);
		if(err == KErrNone)
			{
			_LIT(KType1, "Not present");
			_LIT(KType2, "Unknown");
			_LIT(KType3, "Floppy");
			_LIT(KType4, "Hard disk");
			_LIT(KType5, "CD ROM");
			_LIT(KType6, "RAM disk");
			_LIT(KType7, "Flash");
			_LIT(KType8, "ROM drive");
			_LIT(KType9, "Remote drive");
			_LIT(KType10,"NAND flash");
			_LIT(KType11,"Rotating media");
			_LIT(KYes, "Yes");
			_LIT(KNo,  "No ");
			TPtrC KMediaTypeNames[] = {KType1(), KType2(), KType3(), KType4(), KType5(), KType6(), KType7(), KType8(), KType9(), KType10(), KType11()};
			TheTest.Printf(_L("Drive: %C:, %S, Removable: %S\r\n"), 'A' + driveNo, &KMediaTypeNames[driveInfo.iType], 
																	driveInfo.iDriveAtt & KDriveAttRemovable ? &KYes : &KNo);
			if(driveInfo.iDriveAtt & KDriveAttRemovable)
				{
				TheTest.Printf(_L("Removable drive to test: %C:\r\n"), 'A' + driveNo);
				return driveNo;	
				}
			}
		}
	return KErrNotFound;
	}

//Get the cluster size of aDriveNo drive	
TInt TRemovableMediaTest::ClusterSize(TInt aDriveNo)
	{
	__ASSERT_DEBUG((TUint)aDriveNo <= EDriveZ, User::Invariant());
	TVolumeIOParamInfo volIoParams;
	TInt err = TheFs.VolumeIOParam(aDriveNo, volIoParams);
	return (err == KErrNone) ? volIoParams.iClusterSize : err;
	}

//Create a test database on aDriveNo with aCachePageSize page size.
//Insert KTestRecCnt records.
void TRemovableMediaTest::CreateDatabase(TInt aDriveNo, TInt aCachePageSize)
	{
	__ASSERT_DEBUG((TUint)aDriveNo <= EDriveZ, User::Invariant());
	__ASSERT_DEBUG(aCachePageSize > 0, User::Invariant());
	TDriveUnit drvUnit(aDriveNo);
	_LIT(KDbName, "\\flashmedia.db");
	TParse parse;
	parse.Set(drvUnit.Name(), &KDbName, 0);
	TheRmvMediaDbFileName.Copy(parse.FullName());
	TBuf8<KMaxFileName + 1> dbFileName8;
	dbFileName8.Copy(TheRmvMediaDbFileName);
	(void)TheFs.Delete(TheRmvMediaDbFileName);
	
	sqlite3* dbHandle = NULL;
	TInt rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
	__ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
	
	TBuf8<40> config;
	config.Copy(_L8("PRAGMA PAGE_SIZE="));
	config.AppendNum(aCachePageSize);
	rc = sqlite3_exec(dbHandle, (const char*)config.PtrZ(), 0, 0, 0);
	SQLITE_TEST(dbHandle, rc, SQLITE_OK);

	rc = sqlite3_exec(dbHandle, "CREATE TABLE A(Id INTEGER,Name TEXT)", 0, 0, 0);
	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
	rc = sqlite3_exec(dbHandle, "BEGIN", 0, 0, 0);
	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
	for(TInt recid=1;recid<=KTestRecCnt;++recid)
		{
		
		TBuf8<100> sql;
		sql.Copy(_L8("INSERT INTO A VALUES("));
		sql.AppendNum(recid);
		sql.Append(_L8(",'"));
		sql.Append(KNameColData);
		sql.Append(_L8("')"));
		rc = sqlite3_exec(dbHandle, (const char*)sql.PtrZ(), 0, 0, 0);
		SQLITE_TEST(dbHandle, rc, SQLITE_OK);
		}
	rc = sqlite3_exec(dbHandle, "COMMIT", 0, 0, 0);
	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
	sqlite3_close(dbHandle);
	}

//Checks the content of a single record
void TRemovableMediaTest::CheckRecord(sqlite3_stmt* aStmt, TInt aRecId)
	{
	__ASSERT_DEBUG(aStmt != NULL, User::Invariant());
	TInt id = sqlite3_column_int(aStmt, 0);
	TEST2(id, aRecId);
	const TUint8* text = (const TUint8*)sqlite3_column_text(aStmt, 1);
	TPtrC8 name(text, User::StringLength(text));
	TEST(KNameColData() == name || KUpdatedNameColData() == name);
	}

//Verifies that the database content is either the same as it was before the UPDATE operation or 
//it has been updated with the new data.
void TRemovableMediaTest::VerifyDatabase()
	{
	TBuf8<KMaxFileName + 1> dbFileName8;
	dbFileName8.Copy(TheRmvMediaDbFileName);
	
	sqlite3* dbHandle = NULL;
	TInt rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
	__ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
	
	sqlite3_stmt* stmtHandle = NULL;
	rc = sqlite3_prepare(dbHandle, "SELECT Id,Name FROM A", -1, &stmtHandle, 0);
	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
	__ASSERT_DEBUG(stmtHandle != NULL, User::Invariant());
	
	for(TInt recid=1;recid<=KTestRecCnt;++recid)
		{
		rc = sqlite3_step(stmtHandle);
		SQLITE_TEST(dbHandle, rc, SQLITE_ROW);
		CheckRecord(stmtHandle, recid);
		}
	rc = sqlite3_step(stmtHandle);
	SQLITE_TEST(dbHandle, rc, SQLITE_DONE);
	
	sqlite3_finalize(stmtHandle);
	sqlite3_close(dbHandle);
	}

//Simulates a file system error in a loop.
//Attempts to update single record in a transaction.
//If the UPDATE operation fails - verifies the database content on each iteration.
//Note: pages are stored at the moment, not clusters. The database operations are not more robust if
//      clusters are stored in a case of a removable media.
void TRemovableMediaTest::DoTest()
	{
	TheTest.Printf(_L("Update 1 record in a file I/o simulation loop\r\n"));		
	TInt rc = -1;
	TBuf8<KMaxFileName + 1> dbFileName8;
	dbFileName8.Copy(TheRmvMediaDbFileName);
	for(TInt cnt=1;rc!=SQLITE_OK;++cnt)
		{		
		TheTest.Printf(_L("%d \r"), cnt);		
		sqlite3* dbHandle = NULL;
		rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
		SQLITE_TEST(dbHandle, rc, SQLITE_OK);
		__ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
		rc = sqlite3_exec(dbHandle, "BEGIN IMMEDIATE", 0, 0, 0);
		if(rc == SQLITE_OK)
			{
			rc = sqlite3_exec(dbHandle, "UPDATE A SET Name='1234' WHERE Id=1", 0, 0, 0);
			if(rc == SQLITE_OK)
				{
				TInt cnt = sqlite3_changes(dbHandle);
				TEST2(cnt, 1);
				rc = sqlite3_exec(dbHandle, "COMMIT", 0, 0, 0);
				}
			}
		(void)TheFs.SetErrorCondition(KErrNone);
		sqlite3_close(dbHandle);
		if(rc != SQLITE_OK)
			{
			VerifyDatabase();
			}
		}
	TEST2(rc, SQLITE_OK);
	VerifyDatabase();
	}
	
void TRemovableMediaTest::Run()
	{
	TInt driveNo = GetRemovableMediaDriveNo();
	if(driveNo == KErrNotFound)
		{
		TheTest.Printf(_L("No removable media discovered. Test case not executed.\r\n"));	
		return;
		}
	TInt clusterSize = ClusterSize(driveNo);
	if(clusterSize < 0)
		{
		TheTest.Printf(_L("Error %d retrieving the cluster size of drive %C. Test case not executed.\r\n"), clusterSize, 'A' + driveNo);
		return;
		}
	if(clusterSize <= KMinCachePageSize)
		{
		TheTest.Printf(_L("Cluster size: %d. No appropriate cache page size found. Test case not executed.\r\n"), clusterSize);	
		return;
		}
		
	TheTest.Printf(_L("Cluster size: %d. Cache page size %d.\r\nBegin test.\r\n"), clusterSize, KMinCachePageSize);	
	CreateDatabase(driveNo, KMinCachePageSize);
	DoTest();
	(void)TheFs.Delete(TheRmvMediaDbFileName);
	TheTest.Printf(_L("End test.\r\n"));
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-3516
@SYMTestCaseDesc		Removable media robustness test
						The test creates a test database with a table with some records. Then the test verifies
						that the database content cannot be corrupted by file I/O failures during database updates, 
						when the database file is on a removable media and the media cluster size is bigger than the 
						database page size.
@SYMTestPriority		High
@SYMTestActions			Removable media robustness test
@SYMTestExpectedResults The test must not fail
@SYMREQ					REQ7913
*/
void RemovableMediaRobustnessTest()
	{
	TRemovableMediaTest removableMediaTest;
	removableMediaTest.Run();
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-4044
@SYMTestCaseDesc		RSqlDatabase::Size(TSize&), file I/O error simulation test.
						The test creates a database and executes RSqldatabase::Size(TSize&)
						during a file I/O error simulation. The database should not be corrupted 
						by the call.
@SYMTestPriority		High
@SYMTestActions			RSqlDatabase::Size(TSize&), file I/O error simulation test.
@SYMTestExpectedResults Test must not fail
@SYMREQ					REQ10407
*/
void SizeTest()
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	TInt err = TheDb.Create(KTestDbName);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("BEGIN;CREATE TABLE A(Id INTEGER,Data BLOB);INSERT INTO A VALUES(1, x'11223344');COMMIT;"));
	TEST(err >= 0);
	RSqlDatabase::TSize size1 = {-1, -1};
	err = TheDb.Size(size1);
	TEST2(err, KErrNone);
	TEST(size1.iSize > 0);
	TEST2(size1.iFree, 0);
	TheDb.Close();
	//"File I/O" error simulation loop
	err = KErrCorrupt;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{
		TheTest.Printf(_L("%d \r"), cnt);		
		TEST2(TheDb.Open(KTestDbName), KErrNone);
		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
		RSqlDatabase::TSize size2 = {-1, -1};
		err = TheDb.Size(size2);
		(void)TheFs.SetErrorCondition(KErrNone);
		TheDb.Close();
		if(err == KErrNone)
			{
			TEST(size2.iSize == size1.iSize);
			TEST(size2.iFree == size1.iFree);
			break;
			}
		else
			{
			//check the database content - all bets are off in a case of an I/O error. 
			TEST2(TheDb.Open(KTestDbName), KErrNone);
			TSqlScalarFullSelectQuery q(TheDb);
			TInt recCnt = 0;
			TRAPD(err2, recCnt = q.SelectIntL(_L8("SELECT COUNT(*) FROM A")));
			TheDb.Close();
			TEST2(err2, KErrNone);
			TEST2(recCnt, 1);
			}
		}
	(void)RSqlDatabase::Delete(KTestDbName);
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-4045
@SYMTestCaseDesc		RSqlDatabase::Compact(), file I/O error simulation test.
						The test creates a database and executes RSqlDatabase::Compact()
						during a file I/O error simulation. The database should not be corrupted 
						by the call.
@SYMTestPriority		High
@SYMTestActions			RSqlDatabase::Compact(), file I/O error simulation test.
@SYMTestExpectedResults Test must not fail
@SYMREQ					REQ10405
*/
void CompactTest()
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	_LIT8(KConfig, "compaction=manual");
	TInt err = TheDb.Create(KTestDbName, &KConfig);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
	TEST(err >= 0);
	//Insert records
	err = TheDb.Exec(_L8("BEGIN"));
	TEST(err >= 0);
	const TInt KRecLen = 1000;
	TBuf8<KRecLen> sqlfmt;
	sqlfmt.Copy(_L8("INSERT INTO A VALUES(%d,x'"));
	for(TInt j=0;j<(KRecLen-50);++j)
		{
		sqlfmt.Append(_L8("A"));
		}
	sqlfmt.Append(_L8("')"));
	const TInt KRecCount = 100;	
	for(TInt i=0;i<KRecCount;++i)
		{
		TBuf8<KRecLen> sql;
		sql.Format(sqlfmt, i + 1);
		err = TheDb.Exec(sql);
		TEST2(err, 1);
		}
	err = TheDb.Exec(_L8("COMMIT"));
	TEST(err >= 0);
	//Free some space
	const TInt KDeletedRecCnt = KRecCount - 10;
	err = TheDb.Exec(_L8("DELETE FROM A WHERE Id > 10"));
	TEST(err >= 0);
	//Get the database size
	RSqlDatabase::TSize size;
	err = TheDb.Size(size);
	TEST2(err, KErrNone);
	TheDb.Close();
	TEST(size.iSize > 0);
	TEST(size.iFree > 0);
	//"File I/O" error simulation loop
	err = KErrCorrupt;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{
		TheTest.Printf(_L("%d \r"), cnt);		
		TEST2(TheDb.Open(KTestDbName), KErrNone);
		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
		err = TheDb.Compact(RSqlDatabase::EMaxCompaction);
		(void)TheFs.SetErrorCondition(KErrNone);
		TheDb.Close();
		if(err == KErrNone)
			{
			break;
			}
		else
			{
			//check the database content - all bets are off in a case of an I/O error. 
			//The database maight have been compacted, so - no check for that.
			TEST2(TheDb.Open(KTestDbName), KErrNone);
			TSqlScalarFullSelectQuery q(TheDb);
			TInt recCnt = 0;
			TRAPD(err2, recCnt = q.SelectIntL(_L8("SELECT COUNT(*) FROM A")));
			TheDb.Close();
			TEST2(err2, KErrNone);
			TEST2(recCnt, (KRecCount - KDeletedRecCnt));
			}
		}
	TheTest.Printf(_L("\r\n"));
	//Check that the database has been really compacted
	TEST2(TheDb.Open(KTestDbName), KErrNone);
	RSqlDatabase::TSize size2;
	err = TheDb.Size(size2);
	TEST2(err, KErrNone);
	TheDb.Close();
	(void)RSqlDatabase::Delete(KTestDbName);
	TEST(size.iSize > size2.iSize);
	TEST2(size2.iFree, 0);
	}

void DoBlobWriteStreamTestL(TBool aAttachDb)
	{
	RSqlBlobWriteStream strm;
	CleanupClosePushL(strm);
	if(aAttachDb)
		{
		strm.OpenL(TheDb, _L("A"), _L("Data"), 1, KAttachDb);
		}
	else
		{
		strm.OpenL(TheDb, _L("A"), _L("Data"), 1);
		}

	TBuf8<KBlobSize / KWriteCnt> data;
	data.SetLength(KBlobSize / KWriteCnt);
	data.Fill(0xA5);
	
	for(TInt i=0;i<KWriteCnt;++i)
		{
		strm.WriteL(data);
		}
	
	strm.CommitL();
	
	CleanupStack::PopAndDestroy(&strm);
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-4089
@SYMTestCaseDesc		RSqlBlobWriteStream::WriteL(), file I/O error simulation test.
						The test creates a database and executes RSqlBlobWriteStream::WriteL()
						during a file I/O error simulation. The database should not be corrupted 
						by the call.
@SYMTestPriority		High
@SYMTestActions			RSqlBlobWriteStream::WriteL(), file I/O error simulation test.
@SYMTestExpectedResults Test must not fail
@SYMREQ					REQ5792
                        REQ10418
*/
void BlobWriteStreamTest(TBool aAttachDb)
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	TInt err = TheDb.Create(KTestDbName);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
	TEST2(err, 1);
	TBuf8<100> sql;
	sql.Format(_L8("INSERT INTO A VALUES(1, zeroblob(%d))"), KBlobSize);
	err = TheDb.Exec(sql);
	TEST2(err, 1);
	TheDb.Close();
	
	err = KErrCorrupt;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{
		TheTest.Printf(_L("%d \r"), cnt);		
		TEST2(TheDb.Open(KTestDbName), KErrNone);
		if(aAttachDb)
			{
			TEST2(TheDb.Attach(KTestDbName, KAttachDb), KErrNone);	
			}
		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
		TRAP(err, DoBlobWriteStreamTestL(aAttachDb));
		(void)TheFs.SetErrorCondition(KErrNone);
		if(aAttachDb)
			{
			TEST2(TheDb.Detach(KAttachDb), KErrNone);	
			}
		TheDb.Close();
		}
	TheTest.Printf(_L("\r\n"));

	TEST2(TheDb.Open(KTestDbName), KErrNone);
	
	RSqlStatement stmt;
	err = stmt.Prepare(TheDb, _L8("SELECT * FROM A"));
	TEST2(err, KErrNone);
	err = stmt.Next();
	TEST2(err, KSqlAtRow);
	TPtrC8 data;
	err = stmt.ColumnBinary(1, data);
	TEST2(err, KErrNone);
	TEST2(data.Length(), KBlobSize);
	for(TInt j=0;j<KBlobSize;++j)
		{
		TUint8 d = data[j];
		TEST2(d, 0xA5);	
		}
	stmt.Close();
	
	TheDb.Close();
	(void)RSqlDatabase::Delete(KTestDbName);
	}

void DoBlobReadStreamTestL(TBool aAttachDb, TDes8& aDes)
	{
	RSqlBlobReadStream strm;
	CleanupClosePushL(strm);
	if(aAttachDb)
		{
		strm.OpenL(TheDb, _L("A"), _L("Data"), 1, KAttachDb);
		}
	else
		{
		strm.OpenL(TheDb, _L("A"), _L("Data"), 1);
		}

	TBuf8<KBlobSize / KWriteCnt> data;
	aDes.SetLength(0);
	
	for(TInt i=0;i<KWriteCnt;++i)
		{
		strm.ReadL(data);
		aDes.Append(data);
		}
	
	CleanupStack::PopAndDestroy(&strm);
	}

/**
@SYMTestCaseID			SYSLIB-SQL-UT-4090
@SYMTestCaseDesc		RSqlBlobReadStream::ReadL(), file I/O error simulation test.
						The test creates a database and executes RSqlBlobReadStream::ReadL()
						during a file I/O error simulation. The database should not be corrupted 
						by the call.
@SYMTestPriority		High
@SYMTestActions			RSqlBlobReadStream::ReadL(), file I/O error simulation test.
@SYMTestExpectedResults Test must not fail
@SYMREQ					REQ5792
                        REQ10410
                        REQ10411
*/
void BlobReadStreamTest(TBool aAttachDb)
	{
	(void)RSqlDatabase::Delete(KTestDbName);
	TInt err = TheDb.Create(KTestDbName);
	TEST2(err, KErrNone);
	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
	TEST2(err, 1);
	TBuf8<100> sql;
	sql.Format(_L8("INSERT INTO A VALUES(1, zeroblob(%d))"), KBlobSize);
	err = TheDb.Exec(sql);
	TEST2(err, 1);
	TRAP(err, DoBlobWriteStreamTestL(EFalse));
	TEST2(err, KErrNone);
	TheDb.Close();
	
	HBufC8* buf = HBufC8::New(KBlobSize);
	TEST(buf != NULL);
	TPtr8 bufptr = buf->Des();
	
	err = KErrCorrupt;
	for(TInt cnt=1;err<KErrNone;++cnt)
		{
		TheTest.Printf(_L("%d \r"), cnt);		
		TEST2(TheDb.Open(KTestDbName), KErrNone);
		if(aAttachDb)
			{
			TEST2(TheDb.Attach(KTestDbName, KAttachDb), KErrNone);	
			}
		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
		TRAP(err, DoBlobReadStreamTestL(aAttachDb, bufptr));
		(void)TheFs.SetErrorCondition(KErrNone);
		if(aAttachDb)
			{
			TEST2(TheDb.Detach(KAttachDb), KErrNone);	
			}
		TheDb.Close();
		}
	TheTest.Printf(_L("\r\n"));

	TEST2(bufptr.Length(), KBlobSize);
	for(TInt j=0;j<KBlobSize;++j)
		{
		TUint8 d = bufptr[j];
		TEST2(d, 0xA5);	
		}
		
	delete buf;
		
	(void)RSqlDatabase::Delete(KTestDbName);
	}

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

void DoTests()
	{
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3419 Alter database during file I/O error "));
	AlterDatabaseTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3420 Alter database during file I/O error (using statement object) "));
	AlterDatabaseTest2();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3421 Open database during file I/O error "));
	OpenDatabaseTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3434 Create database during file I/O error "));
	CreateDatabaseTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3462 Select record test during file I/O error "));
	SelectRecordTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3463 Insert record test during file I/O error "));
	InsertRecordTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3516 Removable Media robustness test "));
	RemovableMediaRobustnessTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4044 Database size test during file I/O error"));	
	SizeTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4045 Compact database test during file I/O error"));	
	CompactTest();
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4089 RSqlBlobWriteStream::WriteL() test during file I/O error"));	
	BlobWriteStreamTest(EFalse);
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4089 RSqlBlobWriteStream::WriteL()+attached database test during file I/O error"));	
	BlobWriteStreamTest(ETrue);
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4090 RSqlBlobReadStream::ReadL() test during file I/O error"));	
	BlobReadStreamTest(EFalse);
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4090 RSqlBlobReadStream::ReadL()+attached database test during file I/O error"));	
	BlobReadStreamTest(ETrue);
	}

TInt E32Main()
	{
	TheTest.Title();
	
	CTrapCleanup* tc = CTrapCleanup::New();
	
	__UHEAP_MARK;
	
	SetupTestEnv();
	DoTests();
	DestroyTestEnv();
	
	__UHEAP_MARKEND;
	
	TheTest.End();
	TheTest.Close();
	
	delete tc;
	
	User::Heap().Check();
	return KErrNone;
	}