os/persistentdata/persistentstorage/dbms/tdbms/t_dbood.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// Testing new RDbs methods, which handle "Out of disk space" situations.
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include <e32test.h>
sl@0
    19
#include <f32file.h>
sl@0
    20
#include <d32dbms.h>
sl@0
    21
sl@0
    22
/////////////////////////////////////////////////////////////////
sl@0
    23
//Globals
sl@0
    24
sl@0
    25
//If you change KTestDrive, don't forget to change KTestDatabase too!!!
sl@0
    26
sl@0
    27
#if  defined __WINSCW__ || defined __WINS__
sl@0
    28
sl@0
    29
	//The C: drive may be too big and may be used concurently by other applications. 
sl@0
    30
	//The T: drive is more suitable for the test if running on the emulator
sl@0
    31
	const TInt				KTestDrive = EDriveT;
sl@0
    32
	_LIT(					KTestDatabase, "T:\\DBMS-TST\\T_DbmsOOD.DB");
sl@0
    33
	
sl@0
    34
#elif defined __X86GCC__
sl@0
    35
sl@0
    36
	const TInt				KTestDrive = EDriveG;
sl@0
    37
	_LIT(					KTestDatabase, "G:\\DBMS-TST\\T_DbmsOOD.DB");
sl@0
    38
	
sl@0
    39
#else
sl@0
    40
sl@0
    41
	const TInt				KTestDrive = EDriveE;
sl@0
    42
	_LIT(					KTestDatabase, "E:\\DBMS-TST\\T_DbmsOOD.DB");
sl@0
    43
	
sl@0
    44
#endif
sl@0
    45
sl@0
    46
const TInt				KReservedSpaceSize = 0; //The aSpace parameter of RDbs::ReserveDriveSpace()
sl@0
    47
                                                //is not used at the moment and shall be set to 0.
sl@0
    48
sl@0
    49
static RTest			TheTest(_L("t_dbood - \"Out of Disk space\" test"));
sl@0
    50
static RFs				TheFs;
sl@0
    51
static RDbNamedDatabase TheDb;
sl@0
    52
static RDbs				TheDbSession;
sl@0
    53
sl@0
    54
//Test table defs
sl@0
    55
_LIT(KTestTableName, "TABLE1");
sl@0
    56
sl@0
    57
const TInt KTestRecordsCount = 350;
sl@0
    58
sl@0
    59
struct TColDef
sl@0
    60
	{
sl@0
    61
	const TText*	iName;
sl@0
    62
	TDbColType		iType;
sl@0
    63
	TInt			iAttributes;
sl@0
    64
	};
sl@0
    65
static TColDef const KColDefs[]=
sl@0
    66
	{
sl@0
    67
		{_S("ID"), EDbColUint32, TDbCol::EAutoIncrement},
sl@0
    68
		{_S("DATA"), EDbColBinary, TDbCol::ENotNull},
sl@0
    69
		{0}
sl@0
    70
	};
sl@0
    71
sl@0
    72
//One or more files with KLargeFileName name and ".<n>" extension, where n is
sl@0
    73
//000, 001, 002, 003...
sl@0
    74
//will be created and they will occupy almost all available disk space.
sl@0
    75
//The idea is to perform after that "delete"
sl@0
    76
//transaction, which has to fail, because of "out of disk space" condition.
sl@0
    77
#if  defined __WINSCW__ || defined __WINS__
sl@0
    78
sl@0
    79
	_LIT(KLargeFileName, "T:\\DBMS-TST\\DeleteMe");
sl@0
    80
sl@0
    81
#elif defined  __X86GCC__
sl@0
    82
sl@0
    83
	_LIT(KLargeFileName, "G:\\DBMS-TST\\DeleteMe");
sl@0
    84
sl@0
    85
#else
sl@0
    86
sl@0
    87
	_LIT(KLargeFileName, "E:\\DBMS-TST\\DeleteMe");
sl@0
    88
sl@0
    89
#endif
sl@0
    90
sl@0
    91
static void AssembleLargeFileName(const TDesC& aFileName, TInt aFileNumber, TDes& aResultPath)
sl@0
    92
	{
sl@0
    93
	_LIT(KFormatStr, "%S.%03d");
sl@0
    94
	aResultPath.Format(KFormatStr, &aFileName, aFileNumber);
sl@0
    95
	}
sl@0
    96
sl@0
    97
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    98
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    99
//Create/Destroy test environment - global functions
sl@0
   100
sl@0
   101
//Deletes "aFullName" file.
sl@0
   102
static TInt DeleteDataFile(const TDesC& aFullName)
sl@0
   103
	{
sl@0
   104
	RFs fsSession;
sl@0
   105
	TInt err = fsSession.Connect();
sl@0
   106
	if(err == KErrNone)
sl@0
   107
		{
sl@0
   108
		TEntry entry;
sl@0
   109
		err = fsSession.Entry(aFullName, entry);
sl@0
   110
		if(err == KErrNone)
sl@0
   111
			{
sl@0
   112
			RDebug::Print(_L("Deleting \"%S\" file.\n"), &aFullName);
sl@0
   113
			err = fsSession.SetAtt(aFullName, 0, KEntryAttReadOnly);
sl@0
   114
			if(err != KErrNone)
sl@0
   115
				{
sl@0
   116
				RDebug::Print(_L("Error %d changing \"%S\" file attributes.\n"), err, &aFullName);
sl@0
   117
				}
sl@0
   118
			err = fsSession.Delete(aFullName);
sl@0
   119
			if(err != KErrNone)
sl@0
   120
				{
sl@0
   121
				RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err, &aFullName);
sl@0
   122
				}
sl@0
   123
			}
sl@0
   124
		fsSession.Close();
sl@0
   125
		}
sl@0
   126
	else
sl@0
   127
		{
sl@0
   128
		RDebug::Print(_L("Error %d connecting file session. File: %S.\n"), err, &aFullName);
sl@0
   129
		}
sl@0
   130
	return err;
sl@0
   131
	}
sl@0
   132
sl@0
   133
//Deletes large data files only
sl@0
   134
static void DeleteLargeDataFiles()
sl@0
   135
	{
sl@0
   136
	for(TInt i=0;i<1000;++i)
sl@0
   137
		{
sl@0
   138
		TBuf<KMaxFileName> filePath;
sl@0
   139
		AssembleLargeFileName(KLargeFileName, i, filePath);
sl@0
   140
		if(DeleteDataFile(filePath) != KErrNone)
sl@0
   141
			{
sl@0
   142
			break;
sl@0
   143
			}
sl@0
   144
		}
sl@0
   145
	}
sl@0
   146
sl@0
   147
//Deletes data files used by the test
sl@0
   148
static void DeleteDataFiles()
sl@0
   149
	{
sl@0
   150
	if(TheDbSession.Handle())
sl@0
   151
		{
sl@0
   152
		TheDb.Close();
sl@0
   153
		}
sl@0
   154
	TheDbSession.Close();
sl@0
   155
	DeleteDataFile(KTestDatabase);
sl@0
   156
	DeleteLargeDataFiles();
sl@0
   157
	}
sl@0
   158
sl@0
   159
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   160
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   161
//Tests macros and functions.
sl@0
   162
//If (!aValue) then the test will be panicked, the test data files will be deleted.
sl@0
   163
static void Check(TInt aValue, TInt aLine)
sl@0
   164
	{
sl@0
   165
	if(!aValue)
sl@0
   166
		{
sl@0
   167
		DeleteDataFiles();
sl@0
   168
		TheTest(EFalse, aLine);
sl@0
   169
		}
sl@0
   170
	}
sl@0
   171
//If (aValue != aExpected) then the test will be panicked, the test data files will be deleted.
sl@0
   172
static void Check(TInt aValue, TInt aExpected, TInt aLine)
sl@0
   173
	{
sl@0
   174
	if(aValue != aExpected)
sl@0
   175
		{
sl@0
   176
		RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
sl@0
   177
		DeleteDataFiles();
sl@0
   178
		TheTest(EFalse, aLine);
sl@0
   179
		}
sl@0
   180
	}
sl@0
   181
//Use these to test conditions.
sl@0
   182
#define TEST(arg) Check((arg), __LINE__)
sl@0
   183
#define TEST2(aValue, aExpected) Check(aValue, aExpected, __LINE__)
sl@0
   184
sl@0
   185
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   186
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   187
//Global functions
sl@0
   188
sl@0
   189
//Prepares the test directory.
sl@0
   190
//TheFs.Connect() has to be called already.
sl@0
   191
static void SetupTestDirectory()
sl@0
   192
    {
sl@0
   193
	TInt err = TheFs.MkDir(KTestDatabase);
sl@0
   194
	if(err != KErrNone)
sl@0
   195
	    {
sl@0
   196
	    RDebug::Print(_L("*** SetupTestDirectory(), RFs::MkDir(), err=%d\r\n"), err);
sl@0
   197
	    }
sl@0
   198
	TEST(err == KErrNone || err == KErrAlreadyExists);
sl@0
   199
	}
sl@0
   200
sl@0
   201
//Leaves with info message printed out
sl@0
   202
static void LeaveL(TInt aError, TInt aLine)
sl@0
   203
	{
sl@0
   204
	RDebug::Print(_L("*** Leave. Error: %d, Line: %d\r\n"), aError, aLine);
sl@0
   205
	User::Leave(aError);
sl@0
   206
	}
sl@0
   207
sl@0
   208
//Leaves if aError < 0 with info message printed out
sl@0
   209
static void LeaveIfErrorL(TInt aError, TInt aLine)
sl@0
   210
	{
sl@0
   211
	if(aError < KErrNone)
sl@0
   212
		{
sl@0
   213
		LeaveL(aError, aLine);
sl@0
   214
		}
sl@0
   215
	}
sl@0
   216
sl@0
   217
//Use LEAVE() macro instead of User::Leave() and LEAVE_IF_ERROR() macro instead of
sl@0
   218
//User::LeaveIfError(). They will print the line number, where the "leave" was called.
sl@0
   219
#define LEAVE(aError) LeaveL(aError, __LINE__)
sl@0
   220
#define LEAVE_IF_ERROR(aError) LeaveIfErrorL(aError, __LINE__)
sl@0
   221
sl@0
   222
//Creates one or more large files with the total size near to the size of the available disk space.
sl@0
   223
//The idea is to cause  an "out of disk space" condition.
sl@0
   224
static void FillLargeDataFileL(RFile& aFile, TInt aSize)
sl@0
   225
	{
sl@0
   226
    TInt err = KErrDiskFull;
sl@0
   227
    while(err == KErrDiskFull)
sl@0
   228
        {
sl@0
   229
        err = aFile.SetSize(aSize);
sl@0
   230
        aSize -= 100;
sl@0
   231
        if(aSize <= 0)
sl@0
   232
            {
sl@0
   233
            break;
sl@0
   234
            }
sl@0
   235
        }
sl@0
   236
    TEST(err == KErrNone || err == KErrDiskFull);
sl@0
   237
	}
sl@0
   238
sl@0
   239
//Gets the available space of the tested drive.
sl@0
   240
static TInt64 FreeDiskSpaceL()
sl@0
   241
	{
sl@0
   242
	TVolumeInfo volInfoBefore;
sl@0
   243
	LEAVE_IF_ERROR(TheFs.Volume(volInfoBefore, KTestDrive));
sl@0
   244
	return volInfoBefore.iFree;
sl@0
   245
	}
sl@0
   246
sl@0
   247
//Creates large data file with aSize size (in bytes).
sl@0
   248
static void DoCreateLargeFileL(const TDesC& aPath, TInt aSize)
sl@0
   249
	{
sl@0
   250
	RFile file;
sl@0
   251
	CleanupClosePushL(file);
sl@0
   252
	LEAVE_IF_ERROR(file.Replace(TheFs, aPath, EFileRead | EFileWrite));
sl@0
   253
	FillLargeDataFileL(file, aSize);
sl@0
   254
	LEAVE_IF_ERROR(file.Flush());
sl@0
   255
	CleanupStack::PopAndDestroy(&file);
sl@0
   256
	}
sl@0
   257
sl@0
   258
//Creates enough number of large data files to fill the available disk space.
sl@0
   259
//It will change FilesCount global variable's value.
sl@0
   260
static void CreateLargeFileL()
sl@0
   261
	{
sl@0
   262
	TInt fileNo = 0;
sl@0
   263
	const TInt KLargeFileSize = 1000000000;
sl@0
   264
	TInt64 diskSpace = FreeDiskSpaceL();
sl@0
   265
	RDebug::Print(_L("CreateLargeFileL: free space before = %ld\n"), diskSpace);
sl@0
   266
	TBuf<KMaxFileName> filePath;
sl@0
   267
    const TInt64 KMinDiskSpace = 200;
sl@0
   268
	//Reserve almost all disk space, except a small amount - 200 bytes.
sl@0
   269
	while(diskSpace > KMinDiskSpace)
sl@0
   270
		{
sl@0
   271
		AssembleLargeFileName(KLargeFileName, fileNo++, filePath);
sl@0
   272
		TInt fileSize = KLargeFileSize;
sl@0
   273
        if(diskSpace < (TInt64)KLargeFileSize)
sl@0
   274
            {
sl@0
   275
		    TInt64 lastFileSize = diskSpace - KMinDiskSpace;
sl@0
   276
            fileSize = I64LOW(lastFileSize);
sl@0
   277
            }
sl@0
   278
		DoCreateLargeFileL(filePath, fileSize);
sl@0
   279
		diskSpace = FreeDiskSpaceL();
sl@0
   280
		RDebug::Print(_L("----CreateLargeFileL, step %d, free space = %ld\n"), fileNo, diskSpace);
sl@0
   281
		}
sl@0
   282
	diskSpace = FreeDiskSpaceL();
sl@0
   283
	RDebug::Print(_L("CreateLargeFileL: free space after = %ld\n"), diskSpace);
sl@0
   284
	}
sl@0
   285
sl@0
   286
//Reserves disk space for TheDbSession instance.
sl@0
   287
//TheDbSession instance has to be connected already.
sl@0
   288
static void ReserveDiskSpace()
sl@0
   289
	{
sl@0
   290
	TInt err = TheDbSession.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   291
	TEST2(err, KErrNone);
sl@0
   292
	}
sl@0
   293
sl@0
   294
//Frees already reserved disk space for TheDbSession instance.
sl@0
   295
//TheDbSession instance has to be connected already.
sl@0
   296
static void FreeReservedSpace()
sl@0
   297
	{
sl@0
   298
	TheDbSession.FreeReservedSpace(KTestDrive);
sl@0
   299
	}
sl@0
   300
sl@0
   301
//Gets an access to the reserved disk space for TheDbSession instance.
sl@0
   302
//TheDbSession instance has to be connected already.
sl@0
   303
static void UnlockReservedSpace()
sl@0
   304
	{
sl@0
   305
	TInt err = TheDbSession.GetReserveAccess(KTestDrive);
sl@0
   306
	TEST2(err, KErrNone);
sl@0
   307
	}
sl@0
   308
sl@0
   309
//Releases the access to the reserved disk space.
sl@0
   310
//TheDbSession instance has to be connected already.
sl@0
   311
static void LockReservedSpace()
sl@0
   312
	{
sl@0
   313
	(void)TheDbSession.ReleaseReserveAccess(KTestDrive);
sl@0
   314
	}
sl@0
   315
sl@0
   316
//Creates the test DBMS session
sl@0
   317
static void CreateTestDbSession()
sl@0
   318
	{
sl@0
   319
	TInt err = TheDbSession.Connect();
sl@0
   320
	TEST2(err, KErrNone);
sl@0
   321
	}
sl@0
   322
sl@0
   323
sl@0
   324
//Creates the test database
sl@0
   325
//TheDbSession instance has to be connected already.
sl@0
   326
//TheFs.Connect() has to be called already.
sl@0
   327
static void CreateTestDatabase(RDbs& aDbs, RDbNamedDatabase& aDb)
sl@0
   328
	{
sl@0
   329
	//Create the test database.
sl@0
   330
	TInt err = aDb.Replace(TheFs, KTestDatabase);
sl@0
   331
	TEST2(err, KErrNone);
sl@0
   332
	TheDb.Close();
sl@0
   333
	//Open it now using DBMS session (so, on DBMS server side), because we want to test
sl@0
   334
	//server side RFs sessions - handling "out of disk space" situations.
sl@0
   335
	err = aDb.Open(aDbs, KTestDatabase);
sl@0
   336
	TEST2(err, KErrNone);
sl@0
   337
	}
sl@0
   338
sl@0
   339
//Creates a test table
sl@0
   340
static void CreateTestTableL(RDbNamedDatabase& aDb)
sl@0
   341
	{
sl@0
   342
	CDbColSet* colSet = CDbColSet::NewLC();
sl@0
   343
	for(const TColDef* colDef=KColDefs;colDef->iName;++colDef)
sl@0
   344
		{
sl@0
   345
		TDbCol col(TPtrC(colDef->iName), colDef->iType);
sl@0
   346
		col.iAttributes = colDef->iAttributes;
sl@0
   347
		colSet->AddL(col);
sl@0
   348
		}
sl@0
   349
	TEST2(aDb.CreateTable(KTestTableName, *colSet), KErrNone);
sl@0
   350
	CleanupStack::PopAndDestroy(colSet);
sl@0
   351
	}
sl@0
   352
sl@0
   353
//Adds some data to the test table
sl@0
   354
static void AddTestDataL(RDbNamedDatabase& aDb)
sl@0
   355
	{
sl@0
   356
	RDbTable tbl;
sl@0
   357
	CleanupClosePushL(tbl);
sl@0
   358
	TEST2(tbl.Open(aDb, KTestTableName, RDbRowSet::EUpdatable), KErrNone);
sl@0
   359
	for(TInt i=0;i<KTestRecordsCount;++i)
sl@0
   360
		{
sl@0
   361
		tbl.InsertL();
sl@0
   362
		tbl.SetColL(2, _L8("1ABCDEFGHI2ABCDEFGHI3ABCDEFGHI4ABCDEFGHI5ABCDEFGHI6ABCDEFGHI7ABCDEFGHI8ABCDEFGHI9ABCDEFGHI0ABCDEFGHI"));
sl@0
   363
		tbl.PutL();
sl@0
   364
		}
sl@0
   365
	TEST(tbl.CountL() == KTestRecordsCount);
sl@0
   366
	CleanupStack::PopAndDestroy(&tbl);
sl@0
   367
	}
sl@0
   368
sl@0
   369
//Deletes some records from the test table using "delete" transaction.
sl@0
   370
//Do not put TEST or TEST2 macro calls here (except for record count checks)!
sl@0
   371
//The method must leave if some of the calls inside fail.
sl@0
   372
static void DeleteRecordsL()
sl@0
   373
	{
sl@0
   374
	RDbTable tbl;
sl@0
   375
	CleanupClosePushL(tbl);
sl@0
   376
	LEAVE_IF_ERROR(tbl.Open(TheDb, KTestTableName, RDbRowSet::EUpdatable));
sl@0
   377
	TEST(tbl.CountL() == KTestRecordsCount);
sl@0
   378
	TheDb.Begin();
sl@0
   379
	tbl.FirstL();
sl@0
   380
	for(TInt i=0;i<(KTestRecordsCount/2);++i)
sl@0
   381
		{
sl@0
   382
		tbl.DeleteL();
sl@0
   383
		tbl.NextL();
sl@0
   384
		}
sl@0
   385
	TInt err = TheDb.Commit();
sl@0
   386
	if(err != KErrNone)
sl@0
   387
		{
sl@0
   388
		TheDb.Rollback();
sl@0
   389
		LEAVE(err);
sl@0
   390
		}
sl@0
   391
	TEST(tbl.CountL() == (KTestRecordsCount / 2));
sl@0
   392
	CleanupStack::PopAndDestroy(&tbl);
sl@0
   393
	}
sl@0
   394
sl@0
   395
/**
sl@0
   396
The function simply calls RDbs::ReserveDriveSpace(), RDbs::GetReserveAccess(),
sl@0
   397
RDbs::ReleaseReserveAccess() methods and checks the return values.
sl@0
   398
It might be usefull for debugging in case if something gets wrong.
sl@0
   399
sl@0
   400
@SYMTestCaseID          SYSLIB-DBMS-CT-0647
sl@0
   401
@SYMTestCaseDesc        Tests for attempting to reserve disk space
sl@0
   402
@SYMTestPriority        Medium
sl@0
   403
@SYMTestActions         Calls up RDbs::ReserveDriveSpace(), RDbs::GetReserveAccess(),
sl@0
   404
                        RDbs::ReleaseReserveAccess() methods and checks the return values.
sl@0
   405
@SYMTestExpectedResults Test must not fail
sl@0
   406
@SYMREQ                 REQ0000
sl@0
   407
*/
sl@0
   408
static void SimpleCallsL()
sl@0
   409
	{
sl@0
   410
	RDbs dbs;
sl@0
   411
	CleanupClosePushL(dbs);
sl@0
   412
	LEAVE_IF_ERROR(dbs.Connect());
sl@0
   413
sl@0
   414
	//Reserve disk space
sl@0
   415
	TInt err = dbs.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   416
	TEST2(err, KErrNone);
sl@0
   417
sl@0
   418
	//An attempt to re-reserve it
sl@0
   419
   	err = dbs.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   420
	TEST2(err, KErrInUse);
sl@0
   421
sl@0
   422
	//Get an access to the reserved disk space
sl@0
   423
	err = dbs.GetReserveAccess(KTestDrive);
sl@0
   424
	TEST2(err, KErrNone);
sl@0
   425
sl@0
   426
	//An attempt to get an access to the reserved space twice.
sl@0
   427
	err = dbs.GetReserveAccess(KTestDrive);
sl@0
   428
	TEST2(err, KErrInUse);
sl@0
   429
sl@0
   430
	//This call must fail, because it tries to get an access to the reserved space of
sl@0
   431
	//not the same drive, for which ReserveDriveSpace() was called.
sl@0
   432
	err = dbs.GetReserveAccess(KTestDrive + 1);
sl@0
   433
	TEST(err != KErrNone);
sl@0
   434
sl@0
   435
	(void)dbs.ReleaseReserveAccess(KTestDrive);
sl@0
   436
sl@0
   437
	//An attempt to release the reserved space twice. This call will panic in debug mode.
sl@0
   438
	//(void)dbs.ReleaseReserveAccess(KTestDrive);
sl@0
   439
sl@0
   440
	//Cancel reserving an additional disk space
sl@0
   441
	dbs.FreeReservedSpace(KTestDrive);
sl@0
   442
sl@0
   443
	//Cancel reserving an additional disk space twice
sl@0
   444
    //This call will panic in debug mode.
sl@0
   445
	//dbs.FreeReservedSpace(KTestDrive);
sl@0
   446
sl@0
   447
	CleanupStack::PopAndDestroy(&dbs);
sl@0
   448
	}
sl@0
   449
sl@0
   450
/**
sl@0
   451
@SYMTestCaseID          SYSLIB-DBMS-CT-0648
sl@0
   452
@SYMTestCaseDesc        Transactions test
sl@0
   453
						Simulating  an "out of disk space" situation
sl@0
   454
@SYMTestPriority        Medium
sl@0
   455
@SYMTestActions         Transaction test under "out of disk space" circumstances
sl@0
   456
						while reserving disk space.
sl@0
   457
@SYMTestExpectedResults Test must not fail
sl@0
   458
@SYMREQ                 REQ0000
sl@0
   459
*/
sl@0
   460
static void TransactionTestL()
sl@0
   461
	{
sl@0
   462
    TVolumeIOParamInfo volIoPrm;
sl@0
   463
    TInt err = TheFs.VolumeIOParam(KTestDrive, volIoPrm);
sl@0
   464
    TEST2(err, KErrNone);
sl@0
   465
    RDebug::Print(_L("--Drive %d. BlockSize=%d, ClusterSize=%d, RecReadBufSize=%d, RecWriteBufSize=%d\r\n"), KTestDrive, volIoPrm.iBlockSize, volIoPrm.iClusterSize, volIoPrm.iRecReadBufSize, volIoPrm.iRecWriteBufSize);
sl@0
   466
    /////////////////////////////////////////////////////////
sl@0
   467
	CreateTestDbSession();
sl@0
   468
    //Rserve disk space
sl@0
   469
	ReserveDiskSpace();
sl@0
   470
    //Create test database and table. Add some test data to them.
sl@0
   471
	CreateTestDatabase(TheDbSession, TheDb);
sl@0
   472
	CreateTestTableL(TheDb);
sl@0
   473
	AddTestDataL(TheDb);
sl@0
   474
    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"));
sl@0
   475
	CreateLargeFileL();
sl@0
   476
    RDebug::Print(_L("--Attempt to delete test data records. The transaction must fail, because of \"out of disk space\".\r\n"));
sl@0
   477
    TInt64 diskSpace = FreeDiskSpaceL();
sl@0
   478
	RDebug::Print(_L("--Attempt to delete test data records. Free disk space = %ld\n"), diskSpace);
sl@0
   479
	TRAP(err, DeleteRecordsL());
sl@0
   480
	RDebug::Print(_L("--DeleteRecordsL() returned %d error\r\n"), err);
sl@0
   481
	TEST(err != KErrNone);
sl@0
   482
    RDebug::Print(_L("--The attempt failed with err=%d. Get an access to the reserved disk space.\r\n"), err);
sl@0
   483
    UnlockReservedSpace();
sl@0
   484
	RDebug::Print(_L("--Try again with getting an access to the reserved disk space.\n"));
sl@0
   485
    diskSpace = FreeDiskSpaceL();
sl@0
   486
    RDebug::Print(_L("After GetReserveAccess(), free disk space = %ld\r\n"), diskSpace);
sl@0
   487
	DeleteRecordsL();
sl@0
   488
	RDebug::Print(_L("--\"Delete\" transaction was completed successfully.\n"));
sl@0
   489
    //Free the resources, used in the test
sl@0
   490
	DeleteLargeDataFiles();
sl@0
   491
	LockReservedSpace();
sl@0
   492
    FreeReservedSpace();
sl@0
   493
	}
sl@0
   494
sl@0
   495
/**
sl@0
   496
@SYMTestCaseID          SYSLIB-DBMS-CT-0649
sl@0
   497
@SYMTestCaseDesc        OOD tests with two DBMS sessions.
sl@0
   498
@SYMTestPriority        Medium
sl@0
   499
@SYMTestActions         The test actually checks that the DBMS server is in a stable state, when there is more
sl@0
   500
						than one RDbs session and a shared database is accessed.
sl@0
   501
						The first check is that the shared database can be accessed without any problem through
sl@0
   502
						any of the sessions: first or second.
sl@0
   503
						Then the second session is closed and the shared database is accessed
sl@0
   504
						through the first DBMS session - the operations should not fail.
sl@0
   505
@SYMTestExpectedResults Test must not fail
sl@0
   506
@SYMREQ                 REQ0000
sl@0
   507
*/
sl@0
   508
static void TwoSessTestL()
sl@0
   509
    {
sl@0
   510
    //Create session1, open a shared database, open a shared table through session 1
sl@0
   511
    RDbs dbSess1;
sl@0
   512
    CleanupClosePushL(dbSess1);
sl@0
   513
    LEAVE_IF_ERROR(dbSess1.Connect());
sl@0
   514
sl@0
   515
    RDbNamedDatabase db1;
sl@0
   516
    CleanupClosePushL(db1);
sl@0
   517
	TInt err = db1.Open(dbSess1, KTestDatabase);
sl@0
   518
	TEST2(err, KErrNone);
sl@0
   519
sl@0
   520
	RDbTable tbl1;
sl@0
   521
	CleanupClosePushL(tbl1);
sl@0
   522
	TEST2(tbl1.Open(db1, KTestTableName, RDbRowSet::EUpdatable), KErrNone);
sl@0
   523
sl@0
   524
    //Create session2, open shared database, open shared table through session 2
sl@0
   525
    RDbs dbSess2;
sl@0
   526
    CleanupClosePushL(dbSess2);
sl@0
   527
    LEAVE_IF_ERROR(dbSess2.Connect());
sl@0
   528
sl@0
   529
    RDbNamedDatabase db2;
sl@0
   530
    CleanupClosePushL(db2);
sl@0
   531
	err = db2.Open(dbSess2, KTestDatabase);
sl@0
   532
	TEST2(err, KErrNone);
sl@0
   533
sl@0
   534
	RDbTable tbl2;
sl@0
   535
	CleanupClosePushL(tbl2);
sl@0
   536
	TEST2(tbl2.Open(db2, KTestTableName, RDbRowSet::EUpdatable), KErrNone);
sl@0
   537
sl@0
   538
    //Here we have two sessions and two instances of RDbNamedDatabase type, which
sl@0
   539
    //operate on a shared database. Insert a record through the sessions.
sl@0
   540
sl@0
   541
	tbl1.InsertL();
sl@0
   542
	tbl1.SetColL(2, _L8("--------------------------1----------------------------------------"));
sl@0
   543
	tbl1.PutL();
sl@0
   544
sl@0
   545
	tbl2.InsertL();
sl@0
   546
	tbl2.SetColL(2, _L8("========================2======================================"));
sl@0
   547
	tbl2.PutL();
sl@0
   548
sl@0
   549
    //Close the second session. It should be able to access the shared database via the
sl@0
   550
    //first session.
sl@0
   551
sl@0
   552
	CleanupStack::PopAndDestroy(&tbl2);
sl@0
   553
    CleanupStack::PopAndDestroy(&db2);
sl@0
   554
    CleanupStack::PopAndDestroy(&dbSess2);
sl@0
   555
sl@0
   556
    //Try to access again the shared database.
sl@0
   557
	tbl1.InsertL();
sl@0
   558
	tbl1.SetColL(2, _L8("+++++++++++++++++++++++++++++++++++3++++++++++++++++++++++++++++++++++++++++++"));
sl@0
   559
	tbl1.PutL();
sl@0
   560
sl@0
   561
	CleanupStack::PopAndDestroy(&tbl1);
sl@0
   562
    CleanupStack::PopAndDestroy(&db1);
sl@0
   563
    CleanupStack::PopAndDestroy(&dbSess1);
sl@0
   564
    }
sl@0
   565
sl@0
   566
/**
sl@0
   567
@SYMTestCaseID          SYSLIB-DBMS-CT-0650
sl@0
   568
@SYMTestCaseDesc        OOD tests with more than one DBMS session.
sl@0
   569
@SYMTestPriority        Medium
sl@0
   570
@SYMTestActions         The test calls ReserveDriveSpace/GetReserveAccess/ReleaseReserveAccess in a different
sl@0
   571
						combinations on four DBMS sessions. The test should not fail or panic.
sl@0
   572
@SYMTestExpectedResults Test must not fail
sl@0
   573
@SYMREQ                 REQ0000
sl@0
   574
*/
sl@0
   575
static void TwoSessTest2L()
sl@0
   576
    {
sl@0
   577
    //Create session1
sl@0
   578
    RDbs dbSess1;
sl@0
   579
    CleanupClosePushL(dbSess1);
sl@0
   580
    LEAVE_IF_ERROR(dbSess1.Connect());
sl@0
   581
sl@0
   582
    //Create session2
sl@0
   583
    RDbs dbSess2;
sl@0
   584
    CleanupClosePushL(dbSess2);
sl@0
   585
    LEAVE_IF_ERROR(dbSess2.Connect());
sl@0
   586
sl@0
   587
    //Play with "ReserveDriveSpace" on both sessions
sl@0
   588
    TInt err = dbSess1.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   589
    TEST2(err, KErrNone);
sl@0
   590
    err = dbSess2.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   591
    TEST2(err, KErrNone);
sl@0
   592
    dbSess2.FreeReservedSpace(KTestDrive);
sl@0
   593
    err = dbSess2.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   594
    TEST2(err, KErrNone);
sl@0
   595
sl@0
   596
    //Get an access to the reserved space through session 2
sl@0
   597
	err = dbSess2.GetReserveAccess(KTestDrive);
sl@0
   598
    TEST2(err, KErrNone);
sl@0
   599
    //Free/re-reserve disk space for session 1.
sl@0
   600
    dbSess1.FreeReservedSpace(KTestDrive);
sl@0
   601
    err = dbSess1.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   602
    TEST2(err, KErrNone);
sl@0
   603
sl@0
   604
    //Create session4
sl@0
   605
    RDbs dbSess4;
sl@0
   606
    CleanupClosePushL(dbSess4);
sl@0
   607
    LEAVE_IF_ERROR(dbSess4.Connect());
sl@0
   608
sl@0
   609
    //Try to reserve space for session 4.
sl@0
   610
    err = dbSess4.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   611
    TEST2(err, KErrNone);
sl@0
   612
sl@0
   613
    //Create session3
sl@0
   614
    RDbs dbSess3;
sl@0
   615
    CleanupClosePushL(dbSess3);
sl@0
   616
    LEAVE_IF_ERROR(dbSess3.Connect());
sl@0
   617
    //Try to reserve space for session 3.
sl@0
   618
    err = dbSess3.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   619
    TEST2(err, KErrNone);
sl@0
   620
sl@0
   621
    //Release and free session 2 access to the reserved space.
sl@0
   622
    (void)dbSess2.ReleaseReserveAccess(KTestDrive);
sl@0
   623
    dbSess2.FreeReservedSpace(KTestDrive);
sl@0
   624
sl@0
   625
    dbSess3.FreeReservedSpace(KTestDrive);
sl@0
   626
    CleanupStack::PopAndDestroy(&dbSess3);
sl@0
   627
sl@0
   628
    dbSess4.FreeReservedSpace(KTestDrive);
sl@0
   629
    CleanupStack::PopAndDestroy(&dbSess4);
sl@0
   630
sl@0
   631
    //Get an access to the reserved space through session 2.
sl@0
   632
    //But it was freed, so the call will fail.
sl@0
   633
	err = dbSess2.GetReserveAccess(KTestDrive);
sl@0
   634
    TEST(err != KErrNone);
sl@0
   635
sl@0
   636
    //Free/re-reserve disk space for session 1.
sl@0
   637
    dbSess1.FreeReservedSpace(KTestDrive);
sl@0
   638
    err = dbSess1.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   639
    TEST2(err, KErrNone);
sl@0
   640
sl@0
   641
    //Grant/release the access to the reserved space for session 1.
sl@0
   642
	err = dbSess1.GetReserveAccess(KTestDrive);
sl@0
   643
    TEST2(err, KErrNone);
sl@0
   644
    (void)dbSess1.ReleaseReserveAccess(KTestDrive);
sl@0
   645
sl@0
   646
    //Grant an access to the reserved space for session 2.
sl@0
   647
    //The call will fail because there is no reserved disk space for session 2.
sl@0
   648
	err = dbSess2.GetReserveAccess(KTestDrive);
sl@0
   649
    TEST(err != KErrNone);
sl@0
   650
sl@0
   651
    //Free the reserved space - session 1
sl@0
   652
    dbSess1.FreeReservedSpace(KTestDrive);
sl@0
   653
sl@0
   654
    CleanupStack::PopAndDestroy(&dbSess2);
sl@0
   655
    CleanupStack::PopAndDestroy(&dbSess1);
sl@0
   656
    }
sl@0
   657
sl@0
   658
/**
sl@0
   659
@SYMTestCaseID          SYSLIB-DBMS-CT-0651
sl@0
   660
@SYMTestCaseDesc        Out of memory tests
sl@0
   661
@SYMTestPriority        Medium
sl@0
   662
@SYMTestActions         Checks RDbs::ReserveDriveSpace() behaviour under OOM circumstances
sl@0
   663
@SYMTestExpectedResults Test must not fail
sl@0
   664
@SYMREQ                 REQ0000
sl@0
   665
*/
sl@0
   666
static void OOMTest1()
sl@0
   667
    {
sl@0
   668
    RDbs dbs;
sl@0
   669
    TEST2(dbs.Connect(), KErrNone);
sl@0
   670
	dbs.ResourceMark();
sl@0
   671
	for(TInt count=1;;++count)
sl@0
   672
		{
sl@0
   673
        RDebug::Print(_L("OOMTest1. Count=%d\n"), count);
sl@0
   674
		dbs.SetHeapFailure(RHeap::EFailNext, count);
sl@0
   675
sl@0
   676
		TInt ret = dbs.ReserveDriveSpace(KTestDrive, KReservedSpaceSize);
sl@0
   677
sl@0
   678
		if(ret == KErrNoMemory)
sl@0
   679
			{
sl@0
   680
	        dbs.ResourceCheck();
sl@0
   681
			}
sl@0
   682
		else if(ret == KErrNone)
sl@0
   683
			{
sl@0
   684
			dbs.FreeReservedSpace(KTestDrive);
sl@0
   685
			break;
sl@0
   686
			}
sl@0
   687
		else
sl@0
   688
			{
sl@0
   689
			TEST2(ret, KErrNone);
sl@0
   690
			}
sl@0
   691
		}
sl@0
   692
sl@0
   693
    dbs.SetHeapFailure(RHeap::ENone, 0);
sl@0
   694
    dbs.Close();
sl@0
   695
    }
sl@0
   696
sl@0
   697
/**
sl@0
   698
@SYMTestCaseID          SYSLIB-DBMS-CT-0652
sl@0
   699
@SYMTestCaseDesc        Out of memory tests
sl@0
   700
@SYMTestPriority        Medium
sl@0
   701
@SYMTestActions         Checks RDbs::GetReserveAccess() behaviour under OOM circumstances
sl@0
   702
@SYMTestExpectedResults Test must not fail
sl@0
   703
@SYMREQ                 REQ0000
sl@0
   704
*/
sl@0
   705
static void OOMTest2()
sl@0
   706
    {
sl@0
   707
    RDbs dbs;
sl@0
   708
    TEST2(dbs.Connect(), KErrNone);
sl@0
   709
	TEST2(dbs.ReserveDriveSpace(KTestDrive, KReservedSpaceSize), KErrNone);
sl@0
   710
	dbs.ResourceMark();
sl@0
   711
	for(TInt count=1;;++count)
sl@0
   712
		{
sl@0
   713
        RDebug::Print(_L("OOMTest2. Count=%d\n"), count);
sl@0
   714
		dbs.SetHeapFailure(RHeap::EFailNext, count);
sl@0
   715
sl@0
   716
		TInt ret = dbs.GetReserveAccess(KTestDrive);
sl@0
   717
sl@0
   718
		if(ret == KErrNoMemory)
sl@0
   719
			{
sl@0
   720
	        dbs.ResourceCheck();
sl@0
   721
			}
sl@0
   722
		else if(ret == KErrNone)
sl@0
   723
			{
sl@0
   724
			(void)dbs.ReleaseReserveAccess(KTestDrive);
sl@0
   725
			break;
sl@0
   726
			}
sl@0
   727
		else
sl@0
   728
			{
sl@0
   729
			TEST2(ret, KErrNone);
sl@0
   730
			}
sl@0
   731
		}
sl@0
   732
sl@0
   733
	dbs.FreeReservedSpace(KTestDrive);
sl@0
   734
    dbs.SetHeapFailure(RHeap::ENone, 0);
sl@0
   735
    dbs.Close();
sl@0
   736
    }
sl@0
   737
sl@0
   738
sl@0
   739
//Used in DEF057265().
sl@0
   740
static TInt ThreadFunc(void*)
sl@0
   741
	{
sl@0
   742
	User::SetJustInTime(EFalse);	// disable debugger panic handling
sl@0
   743
	//Create DBMS session. Reserve drive space.
sl@0
   744
    RDbs dbs;
sl@0
   745
    TEST2(dbs.Connect(), KErrNone);
sl@0
   746
	TEST2(dbs.ReserveDriveSpace(KTestDrive, KReservedSpaceSize), KErrNone);
sl@0
   747
	//Panic thread. See DBMS server behaviour - will it panic or not?
sl@0
   748
	//If DBMS server panics in _DEBUG mode - DEF057265 is not properly fixed.
sl@0
   749
	User::Panic(_L("Simulate DBMS client failuer"), 0);
sl@0
   750
	return KErrNone;
sl@0
   751
	}
sl@0
   752
sl@0
   753
//DEF057265 - Panics when uninstalling a java midlet while it is running.
sl@0
   754
//The test will run one thread. Inside the thread's function the test will create
sl@0
   755
//DBMS session and reserve some disk space. Then the test will panic the thread
sl@0
   756
//(without freeing the reserved disk space).
sl@0
   757
//If DBMS server panics in _DEBUG mode - the defect is not fixed.
sl@0
   758
void DEF057265()
sl@0
   759
	{
sl@0
   760
	_LIT(KSessThreadName,"SessThrd");
sl@0
   761
	RThread sessThread;
sl@0
   762
	TEST2(sessThread.Create(KSessThreadName, &ThreadFunc, 0x2000, 0, 0), KErrNone);
sl@0
   763
sl@0
   764
	TRequestStatus sessThreadStatus;
sl@0
   765
	sessThread.Logon(sessThreadStatus);
sl@0
   766
	TEST2(sessThreadStatus.Int(), KRequestPending);
sl@0
   767
sl@0
   768
	sessThread.Resume();
sl@0
   769
	User::WaitForRequest(sessThreadStatus);
sl@0
   770
	TEST2(sessThread.ExitType(), EExitPanic);
sl@0
   771
sl@0
   772
	User::SetJustInTime(EFalse);	// disable debugger panic handling
sl@0
   773
	sessThread.Close();//This Close() operation will force DBMS server to close
sl@0
   774
					   //created in ThreadFunc() DBMS session.
sl@0
   775
	}
sl@0
   776
sl@0
   777
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   778
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   779
//The main test function.
sl@0
   780
//Call your new test functions from here
sl@0
   781
static void RunTestsL()
sl@0
   782
	{
sl@0
   783
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0647 RDbs OOD methods calls "));
sl@0
   784
	SimpleCallsL();
sl@0
   785
sl@0
   786
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0648 Transaction test with reserving disk space "));
sl@0
   787
	TransactionTestL();
sl@0
   788
sl@0
   789
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0649 Two DBMS sessions test "));
sl@0
   790
    TwoSessTestL();
sl@0
   791
sl@0
   792
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0650 Two DBMS sessions test-2 "));
sl@0
   793
    TwoSessTest2L();
sl@0
   794
sl@0
   795
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0651 DBMS OOD - OOM test 1 "));
sl@0
   796
    OOMTest1();
sl@0
   797
sl@0
   798
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0652 DBMS OOD - OOM test 2 "));
sl@0
   799
    OOMTest2();
sl@0
   800
sl@0
   801
	TheTest.Next(_L("DEF057265  Panics when uninstalling a java midlet while it is running"));
sl@0
   802
	DEF057265();
sl@0
   803
sl@0
   804
	//Add tests here!
sl@0
   805
	}
sl@0
   806
sl@0
   807
TInt E32Main()
sl@0
   808
	{
sl@0
   809
	TheTest.Title();
sl@0
   810
sl@0
   811
	__UHEAP_MARK;
sl@0
   812
sl@0
   813
	CTrapCleanup* trapCleanup = CTrapCleanup::New();
sl@0
   814
	TEST(trapCleanup != NULL);
sl@0
   815
sl@0
   816
	DeleteLargeDataFiles();
sl@0
   817
sl@0
   818
	TInt err = TheFs.Connect();
sl@0
   819
	TEST2(err, KErrNone);
sl@0
   820
	SetupTestDirectory();
sl@0
   821
sl@0
   822
	TRAP(err, RunTestsL());
sl@0
   823
	TheDb.Close();
sl@0
   824
	TheDbSession.Close();
sl@0
   825
	TheFs.Close();
sl@0
   826
	TEST2(err, KErrNone);
sl@0
   827
sl@0
   828
	DeleteDataFiles();//delete the data files used by this test
sl@0
   829
sl@0
   830
	TheTest.End();
sl@0
   831
	TheTest.Close();
sl@0
   832
sl@0
   833
	delete trapCleanup;
sl@0
   834
sl@0
   835
	__UHEAP_MARKEND;
sl@0
   836
sl@0
   837
	return 0;
sl@0
   838
	}
sl@0
   839
sl@0
   840