os/persistentdata/persistentstorage/sql/TEST/t_sqlood.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include <e32test.h>
    17 #include <bautils.h>
    18 #include <sqldb.h>
    19 
    20 ///////////////////////////////////////////////////////////////////////////////////////
    21 
    22 static RFs TheFs;
    23 RTest TheTest(_L("t_sqlood test"));
    24 
    25 #if  defined __WINSCW__ || defined __WINS__
    26 
    27 	//The C: drive may be too big and may be used concurently by other applications. 
    28 	//The T: drive is more suitable for the test if running on the emulator
    29 	const TInt KTestDrive = EDriveT;
    30 	_LIT(KTestDir, "t:\\test\\");
    31 	_LIT(KTestDatabase, "t:\\test\\t_sql_ood.db");
    32 	
    33 #elif defined __X86GCC__
    34 
    35 	const TInt KTestDrive = EDriveG;
    36 	_LIT(KTestDir, "g:\\test\\");
    37 	_LIT(KTestDatabase, "g:\\test\\t_sql_ood.db");
    38 	
    39 #else
    40 
    41 	const TInt KTestDrive = EDriveE;
    42 	_LIT(KTestDir, "e:\\test\\");
    43 	_LIT(KTestDatabase, "e:\\test\\t_sql_ood.db");
    44 	
    45 #endif
    46 
    47 //One or more files with KLargeFileName name and ".<n>" extension, where n is 
    48 //000, 001, 002, 003...
    49 //will be created and they will occupy almost all available disk space.
    50 //The idea is to perform after that one "delete"
    51 //transaction, which must to fail, because there won't be enough available disk space to complete the transaction.
    52 #if  defined __WINSCW__ || defined __WINS__
    53 
    54 	_LIT(KLargeFileName, "t:\\test\\DeleteMe");
    55 
    56 #elif defined __X86GCC__
    57 
    58 	_LIT(KLargeFileName, "g:\\test\\DeleteMe");
    59 
    60 #else
    61 
    62 	_LIT(KLargeFileName, "e:\\test\\DeleteMe");
    63 
    64 #endif
    65 
    66 _LIT8(KDatabasePageSizeConfig, "page_size=1024");
    67 
    68 const TInt KMaxTestRecordsCount = 350;
    69 TInt TestRecordsCount = 0;
    70 
    71 ///////////////////////////////////////////////////////////////////////////////////////
    72 
    73 //Assemblesd a file name from "aFileName" and "aFileNumber" parameters and places the resulting string in "aResultPath".
    74 void AssembleLargeFileName(const TDesC& aFileName, TInt aFileNumber, TDes& aResultPath)
    75 	{
    76 	_LIT(KFormatStr, "%S.%03d");
    77 	aResultPath.Format(KFormatStr, &aFileName, aFileNumber);
    78 	}
    79 
    80 //Deletes all created large data files.
    81 void DeleteLargeDataFiles()
    82 	{
    83 	TInt err = KErrNone;
    84 	TInt i = -1;
    85 	while(err == KErrNone)
    86 		{
    87 		TBuf<KMaxFileName> filePath;
    88 		::AssembleLargeFileName(KLargeFileName, ++i, filePath);
    89 		err = TheFs.Delete(filePath);
    90 		}
    91 	}
    92 
    93 //Deletes all created test files.
    94 void DeleteTestFiles()
    95 	{
    96 	DeleteLargeDataFiles();
    97 	(void)RSqlDatabase::Delete(KTestDatabase);
    98 	}
    99 
   100 ///////////////////////////////////////////////////////////////////////////////////////
   101 ///////////////////////////////////////////////////////////////////////////////////////
   102 //Test macros and functions
   103 void Check1(TInt aValue, TInt aLine)
   104 	{
   105 	if(!aValue)
   106 		{
   107 		DeleteTestFiles();
   108 		RDebug::Print(_L("*** Line %d\r\n"), aLine);
   109 		TheTest(EFalse, aLine);
   110 		}
   111 	}
   112 void Check2(TInt aValue, TInt aExpected, TInt aLine)
   113 	{
   114 	if(aValue != aExpected)
   115 		{
   116 		DeleteTestFiles();
   117 		RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
   118 		TheTest(EFalse, aLine);
   119 		}
   120 	}
   121 #define TEST(arg) ::Check1((arg), __LINE__)
   122 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
   123 
   124 ///////////////////////////////////////////////////////////////////////////////////////
   125 
   126 //Creates file session instance and the test directory
   127 void CreateTestEnv()
   128     {
   129 	TInt err = TheFs.Connect();
   130 	TEST2(err, KErrNone);
   131 
   132 	err = TheFs.MkDir(KTestDir);
   133 	if(err != KErrNone)
   134 	    {
   135 	    RDebug::Print(_L("*** CreateTestEnv(), RFs::MkDir(), err=%d\r\n"), err);
   136 	    }
   137 	TEST(err == KErrNone || err == KErrAlreadyExists);
   138 	}
   139 
   140 //Creates one or more large files with the total size near to the size of the available disk space.
   141 //The idea is to cause an "out of disk space" condition.
   142 void FillLargeDataFile(RFile& aFile, TInt aSize)
   143 	{
   144 	TInt err = KErrDiskFull;
   145 	while(err == KErrDiskFull)
   146 		{
   147 		err = aFile.SetSize(aSize);
   148 		aSize -= 100;
   149 		if(aSize <= 0)
   150 			{
   151 			break;
   152 			}
   153 		}
   154 	TEST(err == KErrNone || err == KErrDiskFull);
   155 	}
   156 
   157 //Gets and returns the available disk space of the tested drive.
   158 TInt64 FreeDiskSpace()
   159 	{
   160 	TVolumeInfo volInfoBefore;
   161 	TInt err = TheFs.Volume(volInfoBefore, KTestDrive);
   162 	TEST2(err, KErrNone);
   163 	return volInfoBefore.iFree;
   164 	}
   165 
   166 //Creates a large data file with aSize size (in bytes).
   167 void DoCreateLargeFile(const TDesC& aPath, TInt aSize)
   168 	{
   169 	RFile file;
   170 	TInt err = file.Replace(TheFs, aPath, EFileRead | EFileWrite);
   171 	TEST2(err, KErrNone);
   172 	FillLargeDataFile(file, aSize);
   173 	err = file.Flush();
   174 	TEST2(err, KErrNone);
   175 	file.Close();
   176 	}
   177 
   178 //Creates enough number of large data files to fill the available disk space.
   179 void CreateLargeFile()
   180 	{
   181 	TInt fileNo = 0;
   182 	const TInt KLargeFileSize = 1000000000;
   183 	TInt64 diskSpace = ::FreeDiskSpace();
   184 	RDebug::Print(_L("CreateLargeFile: free space before = %ld\r\n"), diskSpace);
   185 	TBuf<KMaxFileName> filePath;
   186 	while(diskSpace > KLargeFileSize)
   187 		{
   188 		AssembleLargeFileName(KLargeFileName, fileNo++, filePath);
   189 		DoCreateLargeFile(filePath, KLargeFileSize);
   190 		diskSpace = ::FreeDiskSpace();
   191 		RDebug::Print(_L("----CreateLargeFile, step %d, free space = %ld\r\n"), fileNo, diskSpace);
   192 		}
   193 	//Reserve almost all disk space, except a small amount - 200 bytes.
   194 	if(diskSpace > 0)
   195 		{
   196 		::AssembleLargeFileName(KLargeFileName, fileNo++, filePath);
   197 		const TInt64 KSpaceLeft = 200;
   198 		TInt64 lastFileSize = diskSpace - KSpaceLeft;
   199         TInt lastFileSize32 = I64LOW(lastFileSize);
   200 		RDebug::Print(_L("----file size32 = %d\r\n"), lastFileSize32);
   201 		::DoCreateLargeFile(filePath, lastFileSize32);
   202 		RDebug::Print(_L("----CreateLargeFile, last step (%d), file size = %ld\r\n"), fileNo, lastFileSize);
   203 		}
   204 	diskSpace = ::FreeDiskSpace();
   205 	RDebug::Print(_L("CreateLargeFile: free space after = %ld\r\n"), diskSpace);
   206 	}
   207 
   208 
   209 // Number of bytes in the default journal header size.
   210 const TInt KJournalHeaderSize = 0x200;
   211 
   212 // Number of bytes added to each database page in the journal.
   213 const TInt KJournalPageOverhead = 8; 
   214 
   215 // The default amount of reserved space provided by the ReserveDriveSpace API
   216 const TInt KReserveDriveSpaceAmount = 64*1024;
   217 
   218 //Creates and fills with some records a test database
   219 void CreateAndFillTestDatabase(RSqlDatabase& aDb)
   220 	{
   221 	TInt err = aDb.Create(KTestDatabase, &KDatabasePageSizeConfig);
   222 	TEST2(err, KErrNone);
   223 	err = aDb.Exec(_L("CREATE TABLE A(Id INTEGER, Data TEXT)"));
   224 	TEST(err >= 0);
   225 
   226 	//
   227 	// Find the page size of the database on this media
   228 	//
   229 	TBuf<200> sql;
   230 	sql.Copy(_L("PRAGMA page_size"));
   231 	TSqlScalarFullSelectQuery q(aDb);
   232 	TInt pageSize = 0;
   233 	TRAP(err, pageSize = q.SelectIntL(sql););
   234 	//RDebug::Print(_L("Error %d Page Size %d"),err,pageSize);
   235 	TEST2(err, KErrNone);
   236 	TEST(pageSize > 0);
   237 	//RDebug::Print(_L("Page Size %d"),pageSize);
   238 	
   239 	//
   240 	// Find the sector size of this media
   241 	//
   242 	TDriveInfo driveInfo;
   243 	err = TheFs.Drive(driveInfo, KTestDrive);
   244 	TEST2(err, KErrNone);
   245 	TVolumeIOParamInfo volumeInfo;
   246 	err = TheFs.VolumeIOParam(KTestDrive, volumeInfo);
   247 	TEST2(err, KErrNone);
   248 	TInt sectorSize = volumeInfo.iBlockSize;
   249 	//RDebug::Print(_L("Sector Size %d"),sectorSize);	
   250 
   251 	TInt journalHeaderSize = Max(sectorSize, KJournalHeaderSize);
   252 	//RDebug::Print(_L("Journal Header Size %d"),journalHeaderSize);
   253 
   254 	//
   255 	// Keep adding to database until it is a size such that all the data can still be deleted within the reserved disk space size. 
   256 	// Do this piece-wise not in a transaction.
   257 	//
   258 	TInt i;
   259 	for(i=0;i<KMaxTestRecordsCount;++i)
   260 		{
   261 		sql.Format(_L("INSERT INTO A(Id, Data) VALUES(%d, 'A0123456789B0123456789C0123456789D0123456789E0123456789F0123456789G0123456789H0123456789')"), i + 1);
   262 		err = aDb.Exec(sql);
   263 		TEST2(err, 1);
   264 
   265 		TInt size = aDb.Size();
   266 		TInt numberOfPages = size/pageSize;
   267 		TInt predictedJournalSize = journalHeaderSize + numberOfPages * (pageSize + KJournalPageOverhead);
   268 		//RDebug::Print(_L("Size %d, Pages %d, predictedJournalSize %d"),size, numberOfPages, predictedJournalSize);
   269 		
   270 		// Will another page take us over the limit ?
   271 		if ((predictedJournalSize + (pageSize + KJournalPageOverhead)) >= (KReserveDriveSpaceAmount))
   272 			{
   273 			break;
   274 			}
   275 		}
   276 	TestRecordsCount = i + 1;
   277 	
   278 	//RDebug::Print(_L("TestRecordsCount %d"),TestRecordsCount);
   279 	
   280 	}
   281 
   282 //Tries to delete test database records
   283 TInt DeleteTestRecords(RSqlDatabase& aDb)
   284 	{
   285 	TInt err = aDb.Exec(_L("BEGIN TRANSACTION"));
   286 	TEST(err >= 0);
   287 	for(TInt i=0;i<TestRecordsCount;++i)
   288 		{
   289 		TBuf<100> sql;
   290 		sql.Format(_L("DELETE FROM A WHERE Id = %d"), i + 1);
   291 		err = aDb.Exec(sql);// May fail with KErrDiskFull
   292 		if(err < 0)
   293 			{
   294 			(void)aDb.Exec(_L("ROLLBACK TRANSACTION"));
   295 			return err;
   296 			}
   297 		}
   298 	err = aDb.Exec(_L("COMMIT TRANSACTION"));// May fail with KErrDiskFull
   299 	return err;
   300 	}
   301 
   302 ///////////////////////////////////////////////////////////////////////////////////////
   303 
   304 //The function simply calls RSqlDatabase::ReserveDriveSpace(), RSqlDatabase::GetReserveAccess(),
   305 //RSqlDatabase::ReleaseReserveAccess() methods and checks the return values.
   306 //It might be usefull for debugging in case if something gets wrong.
   307 void SimpleCallsTest()
   308 	{
   309 	RSqlDatabase db, db2;
   310 	TInt err = db.Create(KTestDatabase, &KDatabasePageSizeConfig);
   311 	TEST2(err, KErrNone);
   312 
   313 	err = db2.Open(KTestDatabase);
   314 	TEST2(err, KErrNone);
   315 	
   316 	//An attempt to get an access to the reserved space (which is not reserved yet).
   317 	err = db.GetReserveAccess();
   318 	TEST2(err, KErrNotFound);
   319 
   320 	//Reserve disk space
   321 	err = db.ReserveDriveSpace(0);
   322 	TEST2(err, KErrNone);
   323 
   324 	//An attempt to re-reserve it
   325    	err = db.ReserveDriveSpace(0);
   326 	TEST2(err, KErrAlreadyExists);
   327 
   328 	//Get an access to the reserved disk space
   329 	err = db.GetReserveAccess();
   330 	TEST2(err, KErrNone);
   331 
   332 	//Reserve disk space from the second connection
   333 	err = db2.ReserveDriveSpace(0);
   334 	TEST2(err, KErrNone);
   335 
   336 	//An attempt to get an access to the reserved space twice.
   337 	err = db.GetReserveAccess();
   338 	TEST2(err, KErrInUse);
   339 
   340 	//An attempt to get an access to the reserved space from the second connection.
   341 	err = db2.GetReserveAccess();
   342 	TEST2(err, KErrNone);
   343 
   344 	db.ReleaseReserveAccess();
   345 
   346 	//An attempt to release the reserved space twice.
   347 	db.ReleaseReserveAccess();
   348 
   349 	//Free the reserved disk space
   350 	db.FreeReservedSpace();
   351 
   352 	//Free the reserved disk space twice.
   353 	db.FreeReservedSpace();
   354 
   355 	//Free the reserved disk space from the second connection.
   356 	db2.FreeReservedSpace();
   357 
   358 	db2.Close();
   359 	db.Close();
   360 	(void)RSqlDatabase::Delete(KTestDatabase);
   361 	}
   362 
   363 /**
   364 @SYMTestCaseID			SYSLIB-SQL-CT-1649
   365 @SYMTestCaseDesc		SQL database "out of disk space" tests.
   366 						The test creates and fills with some records a test database and then reserves a disk space.
   367 						The second step: the test fills almost all available disk space creting large data files.
   368 						The third step: the test attempts to delete all records from the test database and fails with
   369 						KErrDiskFull error.
   370 						The fourth step: the test gets an access to the reserved disk space and attempts to delete
   371 						records again. This time the test should not fail.
   372 @SYMTestPriority		High
   373 @SYMTestActions			SQL database "out of disk space" tests.
   374 @SYMTestExpectedResults Test must not fail
   375 @SYMREQ					REQ5792
   376                         REQ5793
   377 */
   378 void DeleteTransactionTest()
   379 	{
   380 	TVolumeIOParamInfo volIoPrm;
   381 	TInt err = TheFs.VolumeIOParam(KTestDrive, volIoPrm);
   382     TEST2(err, KErrNone);
   383     RDebug::Print(_L("--Drive %d. BlockSize=%d, ClusterSize=%d, RecReadBufSize=%d, RecWriteBufSize=%d\r\n"), KTestDrive, volIoPrm.iBlockSize, volIoPrm.iClusterSize, volIoPrm.iRecReadBufSize, volIoPrm.iRecWriteBufSize);
   384 	/////////////////////////////////////////////////////////
   385     RDebug::Print(_L("--Create and fill database \"%S\".\r\n"), &KTestDatabase);
   386 	RSqlDatabase db;
   387 	CreateAndFillTestDatabase(db);
   388 	db.Close();//When the database gets closed, the persisted journal file will be deleted.
   389     RDebug::Print(_L("--Close and reopen database \"%S\" (in order to get the persisted journal file deleted).\r\n"), &KTestDatabase);
   390     err = db.Open(KTestDatabase);
   391     TEST2(err, KErrNone);
   392     RDebug::Print(_L("--Reserve disk space for database \"%S\".\r\n"), &KTestDatabase);
   393     err = db.ReserveDriveSpace(0);
   394 	TEST2(err, KErrNone);
   395     RDebug::Print(_L("--Simulate an \"out of disk space\" situation with creating a very large data file, which occupies almost the all the available disk space.\r\n"));
   396 	CreateLargeFile();
   397 	RDebug::Print(_L("--Attempt to delete test data records. The transaction must fail, because of \"out of disk space\".\r\n"));
   398 	err = DeleteTestRecords(db);
   399 	TEST2(err, KErrDiskFull);
   400     RDebug::Print(_L("--Get an access to the reserved disk space.\r\n"));
   401 	err = db.GetReserveAccess();
   402 	TEST2(err, KErrNone);
   403     TInt64 diskSpace = ::FreeDiskSpace();
   404     RDebug::Print(_L("After GetReserveAccess(), free disk space = %ld. Try again \"Delete records\" transaction. The transaction must not fail.\r\n"), diskSpace);
   405 	err = DeleteTestRecords(db);
   406 	RDebug::Print(_L("--DeleteTestRecords() returned %d error.\r\n"), err);
   407 	TEST(err >= 0);
   408 	//Releases the access to the reserved disk space
   409 	db.ReleaseReserveAccess();
   410 	//Frees the reserved disk space
   411 	db.FreeReservedSpace();
   412     //Free the resources, used in the test
   413 	DeleteLargeDataFiles();
   414 	//Verify that the records have been deleted
   415 	RSqlStatement stmt;
   416 	err = stmt.Prepare(db, _L("SELECT COUNT(*) FROM A"));
   417 	TEST2(err, KErrNone);
   418 	err = stmt.Next();
   419 	TEST2(err, KSqlAtRow);
   420 	TInt recCount = stmt.ColumnInt(0);
   421 	TEST2(recCount, 0);
   422 	stmt.Close();
   423 	db.Close();
   424 	(void)RSqlDatabase::Delete(KTestDatabase);
   425 	}
   426 
   427 //OOD API tests with more than one connection to the same SQL database.
   428 //The test calls ReserveDriveSpace/GetReserveAccess/ReleaseReserveAccess in a different
   429 //combinations on four RSqlDatabase objects, connected to the same database .
   430 //The test should not fail or panic.
   431 void MultiDbTest()
   432     {
   433     RSqlDatabase db1;
   434 	CreateAndFillTestDatabase(db1);
   435 
   436     RSqlDatabase db2;
   437     TInt err = db2.Open(KTestDatabase);
   438     TEST2(err, KErrNone);
   439 
   440     //Play with "ReserveDriveSpace" on both sessions
   441     err = db1.ReserveDriveSpace(0);
   442     TEST2(err, KErrNone);
   443     err = db2.ReserveDriveSpace(0);
   444     TEST2(err, KErrNone);
   445     db2.FreeReservedSpace();
   446     err = db2.ReserveDriveSpace(0);
   447     TEST2(err, KErrNone);
   448 
   449     //Get an access to the reserved space through db2
   450 	err = db2.GetReserveAccess();
   451     TEST2(err, KErrNone);
   452     //Free/re-reserve disk space for db1.
   453     db1.FreeReservedSpace();
   454     err = db1.ReserveDriveSpace(0);
   455     TEST2(err, KErrNone);
   456 
   457     RSqlDatabase db4;
   458     err = db4.Open(KTestDatabase);
   459     TEST2(err, KErrNone);
   460 
   461     //Try to reserve space for db4.
   462     err = db4.ReserveDriveSpace(0);
   463     TEST2(err, KErrNone);
   464 
   465     RSqlDatabase db3;
   466     err = db3.Open(KTestDatabase);
   467     TEST2(err, KErrNone);
   468 
   469     //Try to reserve space for session db3.
   470     err = db3.ReserveDriveSpace(0);
   471     TEST2(err, KErrNone);
   472 
   473     //Release and free db2 access to the reserved space.
   474     db2.ReleaseReserveAccess();
   475     db2.FreeReservedSpace();
   476 
   477     db3.FreeReservedSpace();
   478     db3.Close();
   479 
   480     db4.FreeReservedSpace();
   481     db4.Close();
   482 
   483     //Get an access to the reserved space through db2.
   484     //But it was freed, so the call will fail.
   485 	err = db2.GetReserveAccess();
   486     TEST2(err, KErrNotFound);
   487 
   488     //Free/re-reserve disk space for db1.
   489     db1.FreeReservedSpace();
   490     err = db1.ReserveDriveSpace(0);
   491     TEST2(err, KErrNone);
   492 
   493     //Get/release the access to the reserved space for db1.
   494 	err = db1.GetReserveAccess();
   495     TEST2(err, KErrNone);
   496     db1.ReleaseReserveAccess();
   497 
   498     //Get an access to the reserved space for db2.
   499     //The call will fail because there is no reserved disk space for db2.
   500 	err = db2.GetReserveAccess();
   501     TEST2(err, KErrNotFound);
   502 
   503     //Free the reserved space - db1
   504     db1.FreeReservedSpace();
   505 
   506 	db2.Close();
   507 	db1.Close();
   508 
   509 	(void)RSqlDatabase::Delete(KTestDatabase);
   510     }
   511 
   512 void DoTests()
   513 	{
   514 	TheTest.Start(_L(" \"Simple calls\" OOD test "));
   515 	SimpleCallsTest();
   516 
   517 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1649 \"Delete transaction\" OOD test "));
   518 	DeleteTransactionTest();
   519 
   520 	TheTest.Next(_L(" Multi db OOD test "));
   521 	MultiDbTest();
   522 	}
   523 
   524 TInt E32Main()
   525 	{
   526 	TheTest.Title();
   527 
   528 	CTrapCleanup* tc = CTrapCleanup::New();
   529 
   530 	__UHEAP_MARK;
   531 
   532 	CreateTestEnv();
   533 	DeleteTestFiles();
   534 	DoTests();
   535 	DeleteTestFiles();
   536 	TheFs.Close();
   537 
   538 	__UHEAP_MARKEND;
   539 
   540 	TheTest.End();
   541 	TheTest.Close();
   542 
   543 	delete tc;
   544 
   545 	User::Heap().Check();
   546 	return KErrNone;
   547 	}