os/persistentdata/persistentstorage/sql/TEST/t_sqloom3.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) 2005-2010 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
//
sl@0
    15
sl@0
    16
#include "t_sqloom.h"
sl@0
    17
sl@0
    18
RTest TheTest(_L("t_sqloom3 test"));
sl@0
    19
sl@0
    20
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    21
///////////////         RSqlDatabase OOM tests         ////////////////////////////////
sl@0
    22
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    23
sl@0
    24
/**
sl@0
    25
@SYMTestCaseID			SYSLIB-SQL-CT-1615, SYSLIB-SQL-CT-1639
sl@0
    26
@SYMTestCaseDesc		RSqlDatabase::Create() OOM test - secure and non-secure databases.
sl@0
    27
						Precondition: the database does not exist.
sl@0
    28
						The test calls RSqlDatabase::Create() while simulating OOM failures and checks
sl@0
    29
						that there are no memory and resource leaks.
sl@0
    30
						Note: It's possible for a database to be created even after memory allocation
sl@0
    31
						has failed. This is because SQLITE reuses some pages of the page cache which
sl@0
    32
						have been allocated but are curently not in use. This means it is necessary
sl@0
    33
						to delete the database and continue checking for memory and resource leaks
sl@0
    34
						even after a database has been created successfully.
sl@0
    35
@SYMTestPriority		High
sl@0
    36
@SYMTestActions			RSqlDatabase::Create() OOM test
sl@0
    37
@SYMTestExpectedResults Test must not fail
sl@0
    38
@SYMREQ					REQ5792
sl@0
    39
                        REQ5793
sl@0
    40
                        REQ10271
sl@0
    41
                        REQ10273
sl@0
    42
                        REQ10274
sl@0
    43
*/
sl@0
    44
void DoCreateDatabaseOomTest(const TDesC& aDbFileName, TDbType aDbType, TInt aExpectedError, const TDesC8* aConfigStr = NULL)
sl@0
    45
	{
sl@0
    46
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1639 RSqlDatabase::Create() - OOM test"));
sl@0
    47
	RSqlSecurityPolicy securityPolicy;
sl@0
    48
	CreateTestSecurityPolicy(securityPolicy);
sl@0
    49
	enum TMethodType {ENonLeavingMethod, ELeavingMethod};
sl@0
    50
	const TMethodType KMethodType[] = {ENonLeavingMethod, ELeavingMethod};
sl@0
    51
	for(TInt j=0;j<sizeof(KMethodType)/sizeof(KMethodType[0]);++j)
sl@0
    52
		{
sl@0
    53
		for(TInt i=0;i<(TInt)(sizeof(TheOomTestType)/sizeof(TheOomTestType[0]));++i)
sl@0
    54
			{
sl@0
    55
			if(aExpectedError != KErrAlreadyExists)
sl@0
    56
				{
sl@0
    57
				(void)RSqlDatabase::Delete(aDbFileName);
sl@0
    58
				}
sl@0
    59
			TInt err = KErrNone;
sl@0
    60
			TInt failingAllocationNo = 0;//the real exit point of the OOM test. allocationNo is set maxAllocationNo times.
sl@0
    61
			TInt allocationNo = 0;
sl@0
    62
			TInt maxAllocationNo = TheOomTestType[i] == EServerSideTest ? KDoCreateDatabaseOomTestAllocLimitServer : KDoCreateDatabaseOomTestAllocLimitClient;
sl@0
    63
			while(allocationNo < maxAllocationNo)
sl@0
    64
				{
sl@0
    65
				MarkHandles();
sl@0
    66
				MarkAllocatedCells();
sl@0
    67
sl@0
    68
				__UHEAP_MARK;
sl@0
    69
sl@0
    70
				RSqlDatabase db;
sl@0
    71
sl@0
    72
				SetDbHeapFailure(TheOomTestType[i], ++allocationNo);
sl@0
    73
sl@0
    74
				if(KMethodType[j] == ENonLeavingMethod)
sl@0
    75
					{
sl@0
    76
					err =  aDbType == ESecureDb ? db.Create(aDbFileName, securityPolicy, aConfigStr) : db.Create(aDbFileName, aConfigStr);
sl@0
    77
					}
sl@0
    78
				else
sl@0
    79
					{
sl@0
    80
					TRAP(err, aDbType == ESecureDb ? db.CreateL(aDbFileName, securityPolicy, aConfigStr) : db.CreateL(aDbFileName, aConfigStr));
sl@0
    81
					}
sl@0
    82
sl@0
    83
				db.Close();
sl@0
    84
				if(err != KErrNoMemory)
sl@0
    85
					{
sl@0
    86
					TEST2(err, aExpectedError);
sl@0
    87
					}
sl@0
    88
				else
sl@0
    89
					{
sl@0
    90
					failingAllocationNo = allocationNo;
sl@0
    91
					}
sl@0
    92
sl@0
    93
				ResetDbHeapFailure(TheOomTestType[i]);
sl@0
    94
sl@0
    95
				if(err == KErrNone && aExpectedError != KErrAlreadyExists)
sl@0
    96
					{
sl@0
    97
					err = db.Delete(aDbFileName);
sl@0
    98
					TEST2(err, KErrNone);
sl@0
    99
					}
sl@0
   100
sl@0
   101
				__UHEAP_MARKEND;
sl@0
   102
sl@0
   103
				CheckAllocatedCells();
sl@0
   104
				CheckHandles();
sl@0
   105
				}
sl@0
   106
			TEST2(err, aExpectedError);
sl@0
   107
			PrintEndOfOomTest(TheOomTestType[i], failingAllocationNo + 1);
sl@0
   108
			}
sl@0
   109
		}
sl@0
   110
	RSqlDatabase::Delete(aDbFileName);
sl@0
   111
	securityPolicy.Close();
sl@0
   112
	}
sl@0
   113
sl@0
   114
//"RSqlDatabase::Open()" OOM test
sl@0
   115
void OpenDatabaseL(RSqlDatabase& aDb, const TDesC& aDbFileName, TDbType)
sl@0
   116
	{
sl@0
   117
	TInt err = aDb.Open(aDbFileName);
sl@0
   118
	User::LeaveIfError(err);
sl@0
   119
	}
sl@0
   120
sl@0
   121
//"RSqlDatabase::Exec()" OOM test (8-bit SQL statements), syntax error
sl@0
   122
void ExecBadStatement8L(RSqlDatabase& aDb, const TDesC&, TDbType)
sl@0
   123
	{
sl@0
   124
	_LIT8(KSqlString, "CREATE TABL BBB(Fld1 INTEGER, Fld2 BIGINT, Fld3 DOUBLE, Fld4 TEXT)");
sl@0
   125
	TInt err = aDb.Exec(KSqlString);
sl@0
   126
	User::LeaveIfError(err);
sl@0
   127
	}
sl@0
   128
sl@0
   129
//"RSqlDatabase::Exec()" OOM test (16-bit SQL statements), syntax error
sl@0
   130
void ExecBadStatement16L(RSqlDatabase& aDb, const TDesC&, TDbType)
sl@0
   131
	{
sl@0
   132
	_LIT(KSqlString, "CREATE TABLE B!B!B(Fld1 INTEGER, Fld2 BIGINT, Fld3 DOUBLE, Fld4 TEXT)");
sl@0
   133
	TInt err = aDb.Exec(KSqlString);
sl@0
   134
	User::LeaveIfError(err);
sl@0
   135
	}
sl@0
   136
sl@0
   137
/**
sl@0
   138
@SYMTestCaseID			SYSLIB-SQL-CT-1813
sl@0
   139
@SYMTestCaseDesc		RSqlDatabase methods - negative OOM test
sl@0
   140
						Precondition: the database exists.
sl@0
   141
						The test calls the given as an argument function while simulating OOM failures
sl@0
   142
						and checks that there are no memory and resource leaks. The calling function is expected to fail
sl@0
   143
						with aExpectedError error.
sl@0
   144
@SYMTestPriority		High
sl@0
   145
@SYMTestActions			RSqlDatabase methods - negative OOM tests
sl@0
   146
@SYMTestExpectedResults Test must not fail
sl@0
   147
@SYMREQ					REQ5792
sl@0
   148
                        REQ5793
sl@0
   149
*/
sl@0
   150
void DoDbOomNegativeTest(TDbFuncPtrL aTestFunctionPtrL, const TDesC& aDbFileName, TDbAction aDbAction, TInt aExpectedError)
sl@0
   151
	{
sl@0
   152
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1813 RSqlDatabase - negative OOM test"));
sl@0
   153
	for(TInt i=0;i<(TInt)(sizeof(TheOomTestType)/sizeof(TheOomTestType[0]));++i)
sl@0
   154
		{
sl@0
   155
		TInt err = KErrNoMemory;
sl@0
   156
		TInt failingAllocationNo = 0;
sl@0
   157
		while(err == KErrNoMemory)
sl@0
   158
			{
sl@0
   159
			MarkHandles();
sl@0
   160
			MarkAllocatedCells();
sl@0
   161
sl@0
   162
			__UHEAP_MARK;
sl@0
   163
sl@0
   164
			if(TheOomTestType[i] == EServerSideTest)
sl@0
   165
				{//If aDbAction is EOpenDb, then we will delay the heap failure simulation, until the database is opened
sl@0
   166
				SetDbHeapFailure(TheOomTestType[i], ++failingAllocationNo, aDbAction == EOpenDb);
sl@0
   167
				}
sl@0
   168
sl@0
   169
			RSqlDatabase db;
sl@0
   170
			//if aDbAction is EOpenDb then this is a OOM test different than a test for RSqlDatabase::Open
sl@0
   171
			if(aDbAction == EOpenDb)
sl@0
   172
				{
sl@0
   173
				err = db.Open(aDbFileName);
sl@0
   174
				TEST2(err, KErrNone);
sl@0
   175
				}
sl@0
   176
sl@0
   177
			if(TheOomTestType[i] == EClientSideTest)
sl@0
   178
				{
sl@0
   179
				SetDbHeapFailure(TheOomTestType[i], ++failingAllocationNo);
sl@0
   180
				}
sl@0
   181
sl@0
   182
			TRAP(err, (*aTestFunctionPtrL)(db, aDbFileName, ENonSecureDb));
sl@0
   183
			db.Close();
sl@0
   184
			if(err != KErrNoMemory)
sl@0
   185
				{
sl@0
   186
				TEST2(err, aExpectedError);
sl@0
   187
				}
sl@0
   188
sl@0
   189
			ResetDbHeapFailure(TheOomTestType[i]);
sl@0
   190
sl@0
   191
			__UHEAP_MARKEND;
sl@0
   192
sl@0
   193
			CheckAllocatedCells();
sl@0
   194
			CheckHandles();
sl@0
   195
			}
sl@0
   196
		TEST2(err, aExpectedError);
sl@0
   197
		PrintEndOfOomTest(TheOomTestType[i], failingAllocationNo);
sl@0
   198
		}
sl@0
   199
	RSqlDatabase::Delete(aDbFileName);
sl@0
   200
	}
sl@0
   201
sl@0
   202
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   203
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   204
sl@0
   205
//RSqlDatabase - negative OOM tests
sl@0
   206
void DbOomNegativeTestsL()
sl@0
   207
	{
sl@0
   208
	TheTest.Printf(_L("===RSqlDatabase::Open(), non-existing drive\r\n"));
sl@0
   209
	_LIT(KDbName1, "A:[1111CCCC]db1.db");
sl@0
   210
	DoDbOomNegativeTest(&OpenDatabaseL, KDbName1, ENotOpenDb, KErrNotReady);
sl@0
   211
sl@0
   212
	TheTest.Printf(_L("===RSqlDatabase::Open(), non-existing file\r\n"));
sl@0
   213
	_LIT(KDbName2, "c:\\test\\nofile.db");
sl@0
   214
	DoDbOomNegativeTest(&OpenDatabaseL, KDbName2, ENotOpenDb, KErrNotFound);
sl@0
   215
sl@0
   216
	TheTest.Printf(_L("===RSqlDatabase::Open(), zero-length name\r\n"));
sl@0
   217
	_LIT(KDbName3, "");
sl@0
   218
	DoDbOomNegativeTest(&OpenDatabaseL, KDbName3, ENotOpenDb, KErrBadName);
sl@0
   219
sl@0
   220
	TheTest.Printf(_L("===RSqlDatabase::Open(), directory name\r\n"));
sl@0
   221
	_LIT(KDbName4, "C:\\TEST\\");
sl@0
   222
	DoDbOomNegativeTest(&OpenDatabaseL, KDbName4, ENotOpenDb, KErrBadName);
sl@0
   223
sl@0
   224
	TheTest.Printf(_L("===RSqlDatabase::Create(), secure database already exists\r\n"));
sl@0
   225
	RSqlSecurityPolicy securityPolicy;
sl@0
   226
	CreateTestSecurityPolicy(securityPolicy);
sl@0
   227
	RSqlDatabase db;
sl@0
   228
	TInt err = db.Create(KSecureDb2, securityPolicy);
sl@0
   229
	TEST2(err, KErrNone);
sl@0
   230
	db.Close();
sl@0
   231
	securityPolicy.Close();
sl@0
   232
	DoCreateDatabaseOomTest(KSecureDb2, ESecureDb, KErrAlreadyExists);
sl@0
   233
sl@0
   234
	TheTest.Printf(_L("===RSqlDatabase::Create(), database already exists\r\n"));
sl@0
   235
	err = db.Create(KTestDb2);
sl@0
   236
	TEST2(err, KErrNone);
sl@0
   237
	db.Close();
sl@0
   238
	DoCreateDatabaseOomTest(KTestDb2, ENonSecureDb, KErrAlreadyExists);
sl@0
   239
sl@0
   240
	TheTest.Printf(_L("===RSqlDatabase::Exec(), 8-bit SQL, syntax error\r\n"));
sl@0
   241
	err = db.Create(KTestDb);
sl@0
   242
	TEST2(err, KErrNone);
sl@0
   243
	db.Close();
sl@0
   244
	DoDbOomNegativeTest(&ExecBadStatement8L, KTestDb, EOpenDb, KSqlErrGeneral);
sl@0
   245
sl@0
   246
	TheTest.Printf(_L("===RSqlDatabase::Exec(), 16-bit SQL, syntax error\r\n"));
sl@0
   247
	err = db.Create(KTestDb);
sl@0
   248
	TEST2(err, KErrNone);
sl@0
   249
	db.Close();
sl@0
   250
	DoDbOomNegativeTest(&ExecBadStatement16L, KTestDb, EOpenDb, KSqlErrGeneral);
sl@0
   251
	}
sl@0
   252
sl@0
   253
void DEF114297PrepareStmtL(RSqlDatabase& aDb, RSqlStatement& aStmt)
sl@0
   254
	{
sl@0
   255
	_LIT(KSelectSql, "SELECT e.* FROM edge AS e, node AS n1, node AS n2 WHERE n1.name = 'alice' AND n2.name = 'bob' AND e.orig = n1.id AND e.dest = n2.id ORDER BY n2.name DESC");
sl@0
   256
	TInt err = aStmt.Prepare(aDb, KSelectSql);
sl@0
   257
	User::LeaveIfError(err);
sl@0
   258
	}
sl@0
   259
sl@0
   260
/**
sl@0
   261
@SYMTestCaseID			SYSLIB-SQL-UT-4004
sl@0
   262
@SYMTestCaseDesc		Test for DEF114297 - SqlSrv.EXE::!SQL Server OOM Test for PrepareL.
sl@0
   263
						The test does an OOM test for RSqlStatement::Prepare() using a specific SELECT SQL statement.
sl@0
   264
@SYMTestPriority		High
sl@0
   265
@SYMTestActions			Test for DEF114297 - SqlSrv.EXE::!SQL Server OOM Test for PrepareL.
sl@0
   266
@SYMTestExpectedResults Test must not fail
sl@0
   267
@SYMDEF					DEF114297
sl@0
   268
*/
sl@0
   269
void DEF114297()
sl@0
   270
	{
sl@0
   271
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4004  ===DEF114297 - SqlSrv.EXE::!SQL Server OOM Test for PrepareL "));
sl@0
   272
	(void)RSqlDatabase::Delete(KTestDb);
sl@0
   273
	RSqlDatabase db;
sl@0
   274
	TInt err = db.Create(KTestDb);
sl@0
   275
	TEST2(err, KErrNone);
sl@0
   276
	err = db.Exec(_L("CREATE TABLE node(id INTEGER PRIMARY KEY,name TEXT)"));
sl@0
   277
	TEST2(err, 1);
sl@0
   278
	err = db.Exec(_L("CREATE INDEX node_idx ON node(name)"));
sl@0
   279
	TEST2(err, 1);
sl@0
   280
	err = db.Exec(_L("CREATE TABLE edge(orig INTEGER REFERENCES node,dest INTEGER REFERENCES node,PRIMARY KEY(orig, dest))"));
sl@0
   281
	TEST2(err, 1);
sl@0
   282
	err = db.Exec(_L("CREATE INDEX edge_idx ON edge(dest,orig)"));
sl@0
   283
	TEST2(err, 1);
sl@0
   284
	err = db.Exec(_L("INSERT INTO node(id,name) VALUES(1,'alice')"));
sl@0
   285
	TEST2(err, 1);
sl@0
   286
	err = db.Exec(_L("INSERT INTO node(id,name) VALUES(2,'bob')"));
sl@0
   287
	TEST2(err, 1);
sl@0
   288
	err = KErrNoMemory;
sl@0
   289
	TInt failingAllocationNo = 0;
sl@0
   290
	while(err == KErrNoMemory)
sl@0
   291
		{
sl@0
   292
		MarkHandles();
sl@0
   293
		MarkAllocatedCells();
sl@0
   294
sl@0
   295
		__UHEAP_MARK;
sl@0
   296
sl@0
   297
		SetHeapFailure(EServerSideTest, ++failingAllocationNo);
sl@0
   298
sl@0
   299
    	RSqlStatement stmt;
sl@0
   300
		TRAP(err, DEF114297PrepareStmtL(db, stmt));
sl@0
   301
		stmt.Close();
sl@0
   302
		if(err != KErrNoMemory)
sl@0
   303
			{
sl@0
   304
			TEST2(err, KErrNone);
sl@0
   305
			}
sl@0
   306
sl@0
   307
		ResetHeapFailure(EServerSideTest);
sl@0
   308
sl@0
   309
		__UHEAP_MARKEND;
sl@0
   310
sl@0
   311
		CheckAllocatedCells();
sl@0
   312
		CheckHandles();
sl@0
   313
		}
sl@0
   314
	db.Close();
sl@0
   315
	(void)RSqlDatabase::Delete(KTestDb);
sl@0
   316
	}
sl@0
   317
sl@0
   318
/**
sl@0
   319
@SYMTestCaseID			SYSLIB-SQL-UT-4011
sl@0
   320
@SYMTestCaseDesc		Test for DEF115815 - SELECT random()&1==-1 causes sql server to crash.
sl@0
   321
						The test does an OOM test for RSqlStatement::Prepare() using a specific SELECT SQL statement.
sl@0
   322
@SYMTestPriority		High
sl@0
   323
@SYMTestActions			Test for DEF115815 - SELECT random()&1==-1 causes sql server to crash.
sl@0
   324
@SYMTestExpectedResults Test must not fail
sl@0
   325
@SYMDEF					DEF115815
sl@0
   326
*/
sl@0
   327
void DEF115815()
sl@0
   328
	{
sl@0
   329
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4011 ===DEF115815 - SELECT random()&1==-1 causes sql server to crash "));
sl@0
   330
	(void)RSqlDatabase::Delete(KTestDb);
sl@0
   331
	RSqlDatabase db;
sl@0
   332
	TInt err = db.Create(KTestDb);
sl@0
   333
	TEST2(err, KErrNone);
sl@0
   334
	err = db.Exec(_L("CREATE TABLE node(id INTEGER)"));
sl@0
   335
	TEST2(err, 1);
sl@0
   336
	err = KErrNoMemory;
sl@0
   337
	TInt failingAllocationNo = 0;
sl@0
   338
	while(err == KErrNoMemory)
sl@0
   339
		{
sl@0
   340
		MarkHandles();
sl@0
   341
		MarkAllocatedCells();
sl@0
   342
sl@0
   343
		__UHEAP_MARK;
sl@0
   344
sl@0
   345
		SetHeapFailure(EServerSideTest, ++failingAllocationNo);
sl@0
   346
sl@0
   347
    	RSqlStatement stmt;
sl@0
   348
		err = stmt.Prepare(db, _L("SELECT random()&1==-1"));
sl@0
   349
		stmt.Close();
sl@0
   350
		if(err != KErrNoMemory)
sl@0
   351
			{
sl@0
   352
			TEST2(err, KErrNone);
sl@0
   353
			}
sl@0
   354
sl@0
   355
		ResetHeapFailure(EServerSideTest);
sl@0
   356
sl@0
   357
		__UHEAP_MARKEND;
sl@0
   358
sl@0
   359
		CheckAllocatedCells();
sl@0
   360
		CheckHandles();
sl@0
   361
		}
sl@0
   362
	db.Close();
sl@0
   363
	(void)RSqlDatabase::Delete(KTestDb);
sl@0
   364
	}
sl@0
   365
sl@0
   366
void DoTestsL()
sl@0
   367
	{
sl@0
   368
	TheTest.Start(_L("SQL OOM-3 tests"));
sl@0
   369
sl@0
   370
	DbOomNegativeTestsL();
sl@0
   371
sl@0
   372
	DEF114297();
sl@0
   373
sl@0
   374
	DEF115815();
sl@0
   375
	}
sl@0
   376
sl@0
   377
TInt E32Main()
sl@0
   378
	{
sl@0
   379
	TheTest.Title();
sl@0
   380
sl@0
   381
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   382
sl@0
   383
	__UHEAP_MARK;
sl@0
   384
sl@0
   385
	CreateTestDir();
sl@0
   386
	DeleteTestFiles();
sl@0
   387
sl@0
   388
	TRAPD(err, DoTestsL());
sl@0
   389
	DeleteTestFiles();
sl@0
   390
	TEST2(err, KErrNone);
sl@0
   391
sl@0
   392
	__UHEAP_MARKEND;
sl@0
   393
sl@0
   394
	TheTest.End();
sl@0
   395
	TheTest.Close();
sl@0
   396
sl@0
   397
	delete tc;
sl@0
   398
sl@0
   399
	User::Heap().Check();
sl@0
   400
	return KErrNone;
sl@0
   401
	}