os/persistentdata/persistentstorage/sql/TEST/t_sqldefect2.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2006-2010 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 <f32file.h>
    18 #include <sqldb.h>
    19 #include <f32file.h>
    20 ///////////////////////////////////////////////////////////////////////////////////////
    21 
    22 static RFs TheFs;
    23 static RTest TheTest(_L("t_sqldefect2 test"));
    24 static RSqlDatabase TheDb1;
    25 static RSqlDatabase TheDb2;
    26 
    27 _LIT(KTestDir, "c:\\test\\");
    28 _LIT(KTestDatabase1, "c:\\test\\t_sqldefect2.db");
    29 _LIT(KTestDatabaseJournal1, "c:\\test\\t_sqldefect2.db-journal");
    30 _LIT(KServerTempDir, "c:\\private\\10281e17\\temp\\");
    31 
    32 ///////////////////////////////////////////////////////////////////////////////////////
    33 
    34 //Deletes all created test files.
    35 void DestroyTestEnv()
    36 	{
    37     TheDb2.Close();
    38     TheDb1.Close();
    39 	(void)RSqlDatabase::Delete(KTestDatabase1);
    40 	TheFs.Close();
    41 	}
    42 
    43 ///////////////////////////////////////////////////////////////////////////////////////
    44 ///////////////////////////////////////////////////////////////////////////////////////
    45 //Test macros and functions
    46 void Check1(TInt aValue, TInt aLine)
    47 	{
    48 	if(!aValue)
    49 		{
    50 		DestroyTestEnv();
    51 		RDebug::Print(_L("*** Line %d\r\n"), aLine);
    52 		TheTest(EFalse, aLine);
    53 		}
    54 	}
    55 void Check2(TInt aValue, TInt aExpected, TInt aLine)
    56 	{
    57 	if(aValue != aExpected)
    58 		{
    59 		DestroyTestEnv();
    60 		RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
    61 		TheTest(EFalse, aLine);
    62 		}
    63 	}
    64 #define TEST(arg) ::Check1((arg), __LINE__)
    65 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
    66 
    67 ///////////////////////////////////////////////////////////////////////////////////////
    68 
    69 //Creates file session instance and the test directory
    70 void CreateTestEnv()
    71     {
    72 	TInt err = TheFs.Connect();
    73 	TEST2(err, KErrNone);
    74 
    75 	err = TheFs.MkDir(KTestDir);
    76 	TEST(err == KErrNone || err == KErrAlreadyExists);
    77 	}
    78 
    79 /**
    80 @SYMTestCaseID          PDS-SQL-CT-4154
    81 @SYMTestCaseDesc        Test for DEF143062: SQL, "CREATE INDEX" sql crashes SQL server.
    82                         The test creates a database with one empty table and establishes two connections
    83                         to that database. Then, while the first connection is at the middle of a read
    84                         transaction, the second connection attempts to create an index.
    85                         If the defect is not fixed, the SQL server will crash.
    86 @SYMTestPriority        High
    87 @SYMTestActions         DEF143062: SQL, "CREATE INDEX" sql crashes SQL server.
    88 @SYMTestExpectedResults Test must not fail
    89 @SYMDEF                 DEF143062
    90 */
    91 void DEF143062()
    92     {
    93     (void)RSqlDatabase::Delete(KTestDatabase1);
    94     TInt err = TheDb1.Create(KTestDatabase1);
    95     TEST2(err, KErrNone);
    96     err = TheDb1.Exec(_L("CREATE TABLE T0(Thread INTEGER, LocalIndex INTEGER, Inserts INTEGER, Updates INTEGER, IndexMod8 INTEGER)"));
    97     TEST(err >= 0);
    98     
    99     err = TheDb2.Open(KTestDatabase1);
   100     TEST2(err, KErrNone);
   101 
   102     RSqlStatement stmt;
   103     err = stmt.Prepare(TheDb1, _L8("SELECT COUNT(Thread) FROM T0 WHERE Thread = 0"));
   104     TEST2(err, KErrNone);
   105     err = stmt.Next();
   106     TEST2(err, KSqlAtRow);
   107     
   108     err = TheDb2.Exec(_L8("CREATE INDEX T0INDEX ON T0(Thread,IndexMod8)"));//crashes the SQL server if the defect is not fixed 
   109     TEST2(err, KSqlErrLocked);
   110     
   111     stmt.Close();
   112     
   113     TheDb2.Close();
   114     TheDb1.Close();
   115     err = RSqlDatabase::Delete(KTestDatabase1);
   116     TEST2(err, KErrNone);
   117     }
   118 
   119 /**
   120 @SYMTestCaseID          PDS-SQL-CT-4155
   121 @SYMTestCaseDesc        Test for DEF143061: SQL, SQLITE_DEFAULT_JOURNAL_SIZE_LIMIT value is too big.
   122                                          The test verifies that after comitting a big transaction, the journal file size is made equal the 
   123                                           max journal file size limit of 64Kb.
   124 @SYMTestPriority        High
   125 @SYMTestActions         DEF143061: SQL, SQLITE_DEFAULT_JOURNAL_SIZE_LIMIT value is too big..
   126 @SYMTestExpectedResults Test must not fail
   127 @SYMDEF                 DEF143061
   128 */
   129 void DEF143061()
   130     {
   131     (void)RSqlDatabase::Delete(KTestDatabase1);
   132     //"Auto" compaction is used in order to see how the journal file is immediatelly used.
   133     _LIT8(KConfig, "compaction=auto");
   134     TInt err = TheDb1.Create(KTestDatabase1, &KConfig);
   135     TEST2(err, KErrNone);
   136     err = TheDb1.Exec(_L("CREATE TABLE A(I INTEGER, B BLOB)"));
   137     TEST(err >= 0);
   138 
   139     const TInt KBlobSize = 100000;//bigger than the journal size limit
   140     HBufC8* buf = HBufC8::New(KBlobSize);
   141     TEST(buf != NULL);
   142     TPtr8 ptr = buf->Des();
   143     ptr.SetLength(KBlobSize);
   144         
   145     RSqlStatement stmt;
   146     err = stmt.Prepare(TheDb1, _L("INSERT INTO A VALUES(1, :Prm)"));
   147     TEST2(err, KErrNone);
   148     ptr.Fill(TChar('N'));
   149     err = stmt.BindBinary(0, ptr);
   150     TEST2(err, KErrNone);
   151     err = stmt.Exec();
   152     TEST2(err, 1);
   153     stmt.Close();
   154     
   155     //Try to update the BLOB in the record that was just inserted. This operation should create a big journal file.
   156     err = stmt.Prepare(TheDb1, _L("UPDATE A SET B=:Prm WHERE I=1"));
   157     TEST2(err, KErrNone);
   158     ptr.Fill(TChar('Y'));
   159     err = stmt.BindBinary(0, ptr);
   160     TEST2(err, KErrNone);
   161     err = stmt.Exec();
   162     TEST2(err, 1);
   163     stmt.Close();
   164     
   165     //Check the journal file size. It should be less than the 64Kb limit defined in sqlite_macro.mmh file.  
   166     TEntry entry;
   167     err = TheFs.Entry(KTestDatabaseJournal1, entry);
   168     TEST2(err, KErrNone);
   169     TInt64 fsize = entry.FileSize();
   170     TEST(fsize <= SQLITE_DEFAULT_JOURNAL_SIZE_LIMIT);
   171     
   172     delete buf;
   173     TheDb1.Close();
   174     err = RSqlDatabase::Delete(KTestDatabase1);
   175     TEST2(err, KErrNone);
   176     }
   177 
   178 /**
   179 @SYMTestCaseID          PDS-SQL-CT-4156
   180 @SYMTestCaseDesc        Test for DEF143150: SQL, strftime() returns incorrect result.
   181                         The test takes the current universal time (using TTime) 
   182                         and the current time retrieved from the SQL  server.
   183                         The test compares the times and expects the difference to be no more than
   184                         1 second. 
   185 @SYMTestPriority        High
   186 @SYMTestActions         DEF143150: SQL, strftime() returns incorrect result
   187 @SYMTestExpectedResults Test must not fail
   188 @SYMDEF                 DEF143150
   189 */
   190 void DEF143150()
   191     {
   192     (void)RSqlDatabase::Delete(KTestDatabase1);
   193     TInt err = TheDb1.Create(KTestDatabase1);
   194     TEST2(err, KErrNone);
   195 
   196     //Home date & time
   197     TBuf<50> dtstr1;
   198     TTime time;
   199     time.UniversalTime();
   200     TDateTime dt = time.DateTime();
   201     
   202     RSqlStatement stmt;
   203     err = stmt.Prepare(TheDb1, _L("SELECT strftime('%Y-%m-%d,%H:%M:%S','now')"));
   204     TEST2(err, KErrNone);
   205     err = stmt.Next();
   206     TEST2(err, KSqlAtRow);
   207     
   208     //SQLite date & time
   209     TBuf<50> dtstr2;
   210     err = stmt.ColumnText(0, dtstr2);
   211     TEST2(err, KErrNone);
   212     stmt.Close();
   213 
   214     TheDb1.Close();
   215     err = RSqlDatabase::Delete(KTestDatabase1);
   216     TEST2(err, KErrNone);
   217     
   218     dtstr1.Format(_L("%04d-%02d-%02d,%02d:%02d:%02d"), dt.Year(), dt.Month() + 1, dt.Day() + 1, dt.Hour(), dt.Minute(), dt.Second());
   219     TheTest.Printf(_L("Universal date&time=\"%S\"\n"), &dtstr1);
   220     TheTest.Printf(_L("SQLite    date&time=\"%S\"\n"), &dtstr2);
   221     
   222     //Comapare and fail if dates are not equal (+- 1 second)
   223     TLex lex;
   224     lex = dtstr2.Mid(0, 4);
   225     TInt sqlyear;
   226     err = lex.Val(sqlyear);
   227     TEST2(err, KErrNone);
   228     
   229     lex = dtstr2.Mid(5, 2);
   230     TInt sqlmonth;
   231     err = lex.Val(sqlmonth);
   232     TEST2(err, KErrNone);
   233     
   234     lex = dtstr2.Mid(8, 2);
   235     TInt sqlday;
   236     err = lex.Val(sqlday);
   237     TEST2(err, KErrNone);
   238     
   239     lex = dtstr2.Mid(11, 2);
   240     TInt sqlhour;
   241     err = lex.Val(sqlhour);
   242     TEST2(err, KErrNone);
   243     
   244     lex = dtstr2.Mid(14, 2);
   245     TInt sqlminute;
   246     err = lex.Val(sqlminute);
   247     TEST2(err, KErrNone);
   248     
   249     lex = dtstr2.Mid(17, 2);
   250     TInt sqlsecond;
   251     err = lex.Val(sqlsecond);
   252     TEST2(err, KErrNone);
   253     
   254     TDateTime sqldt(sqlyear, (TMonth)(sqlmonth - 1), sqlday - 1, sqlhour, sqlminute, sqlsecond, 0);
   255     TTime sqltime(sqldt);
   256     TTimeIntervalSeconds diff;
   257     err = sqltime.SecondsFrom(time, diff);
   258     TEST2(err, KErrNone);
   259     TEST(diff.Int() <= 1);
   260     }
   261 
   262 static TInt KillProcess(const TDesC& aProcessName)
   263     {
   264     TFullName name;
   265     TBuf<64> pattern(aProcessName);
   266     TInt length = pattern.Length();
   267     pattern += _L("*");
   268     TFindProcess procFinder(pattern);
   269 
   270     while (procFinder.Next(name) == KErrNone)
   271         {
   272         if (name.Length() > length)
   273             {//If found name is a string containing aProcessName string.
   274             TChar c(name[length]);
   275             if (c.IsAlphaDigit() ||
   276                 c == TChar('_') ||
   277                 c == TChar('-'))
   278                 {
   279                 // If the found name is other valid application name
   280                 // starting with aProcessName string.
   281                 continue;
   282                 }
   283             }
   284         RProcess proc;
   285         if (proc.Open(name) == KErrNone)
   286             {
   287             proc.Kill(0);
   288             }
   289         proc.Close();
   290         }
   291     return KErrNone;
   292     }
   293 
   294 /**
   295 @SYMTestCaseID          PDS-SQL-CT-4210
   296 @SYMTestCaseDesc        Test for the change "Temp files created during sql operations are not deleted after rebooting the phone" 
   297 @SYMTestPriority        High
   298 @SYMTestActions         Kill the sql server
   299                         Create two temp files in sql server's private directory
   300                         Start the sql server
   301                         Test that the temp files do not exist.
   302 @SYMTestExpectedResults Test must not fail
   303 */
   304 void DeleteTempFile()
   305     {   
   306     _LIT(KSqlSrvName, "sqlsrv.exe");
   307     _LIT(KTempFile1, "TMP00052.$$$");
   308     _LIT(KTempFile2, "TMP00044.$$$");
   309     
   310     KillProcess(KSqlSrvName);
   311  
   312     //Create two temp file in c:\\private\\10281e17\\temp\\ folder
   313     TInt err = TheFs.MkDir(KServerTempDir);
   314     TEST(err == KErrNone || err == KErrAlreadyExists);
   315     RFile file;
   316     TFileName filename1(KServerTempDir);
   317     TFileName filename2(KServerTempDir);
   318     filename1.Append(KTempFile1);
   319     filename2.Append(KTempFile2);
   320     err = file.Replace(TheFs, filename1, 0);
   321     file.Close();
   322     TEST2(err, KErrNone);
   323     err = file.Replace(TheFs, filename2, 0);
   324     file.Close();
   325     TEST2(err, KErrNone);
   326     
   327     //Create a database that should start sql server
   328     err = TheDb1.Create(KTestDatabase1);
   329     TEST(err == KErrNone || err == KErrAlreadyExists);
   330     //Test that the temp files have been deleted during server's start-up
   331     TUint dummy;
   332     err = TheFs.Att(filename1, dummy);
   333     TEST2(err, KErrNotFound);
   334     err = TheFs.Att(filename2, dummy);
   335     TEST2(err, KErrNotFound);
   336     
   337     TheDb1.Close();
   338     err = RSqlDatabase::Delete(KTestDatabase1);
   339     TEST2(err, KErrNone);
   340     }
   341 
   342 TInt TempFilesCount()
   343 	{
   344     _LIT(KServerTempDirMask, "c:\\private\\10281e17\\temp\\*.*");
   345 	CDir* dir = NULL;
   346 	TInt err = TheFs.GetDir(KServerTempDirMask, KEntryAttNormal, ESortNone, dir);
   347 	TEST2(err, KErrNone);
   348 	TInt tmpFileCount = dir->Count();
   349 	delete dir;
   350 	return tmpFileCount;
   351 	}
   352 
   353 /**
   354 @SYMTestCaseID          PDS-SQL-CT-4211
   355 @SYMTestCaseDesc        Test for the change "Temp files created during sql operations are not deleted after rebooting the phone" 
   356 @SYMTestPriority        High
   357 @SYMTestActions         The test creates a database and runs a set of statements that
   358 						will lead to a delayed creation of a temp file.
   359 						At the end the test checks that the temp file was created.
   360 @SYMTestExpectedResults Test must not fail
   361 */
   362 void TempFileTest()
   363 	{
   364     (void)RSqlDatabase::Delete(KTestDatabase1);
   365     TInt err = TheDb1.Create(KTestDatabase1);
   366     TEST2(err, KErrNone);
   367     //Get the number of the files in the SQL temp directory 
   368 	TInt tmpFileCount = TempFilesCount();
   369     //    
   370     err = TheDb1.Exec(_L("CREATE TABLE t1(x UNIQUE); INSERT INTO t1 VALUES(1)"));
   371     TEST(err >= 0);
   372     err = TheDb1.Exec(_L("BEGIN; UPDATE t1 SET x = 2; UPDATE t1 SET x = 3; COMMIT"));
   373     TEST(err >= 0);
   374     //Check that a temp file really was created
   375 	TInt tmpFileCount2 = TempFilesCount();
   376 	TEST(tmpFileCount2 > tmpFileCount);
   377     //
   378     TheDb1.Close();
   379     err = RSqlDatabase::Delete(KTestDatabase1);
   380     TEST2(err, KErrNone);
   381 	}
   382 
   383 /**
   384 @SYMTestCaseID          PDS-SQL-CT-4213
   385 @SYMTestCaseDesc        Tests the ability of the SQL server to store empty strings and retrieve them as 
   386 						text column values, not NULLs.
   387 						Change: ou1cimx1#504388. 
   388 @SYMTestPriority        High
   389 @SYMTestActions         The test creates a database and a table and stores there empty strings.
   390 						Then the test retrieves the stored column values and verifies that the column type is
   391 						"text", not "null".
   392 @SYMTestExpectedResults Test must not fail
   393 */
   394 void EmptyTextColumnTest()
   395 	{
   396 	_LIT8(KEncUtf16, "encoding=\"UTF-16\"");
   397 	_LIT8(KEncUtf8, "encoding=\"UTF-8\"");
   398 	TPtrC8 enc[] = {KEncUtf16(), KEncUtf8()};
   399 	for(TInt i=0;i<(sizeof(enc)/sizeof(enc[0]));++i)
   400 		{
   401 		(void)RSqlDatabase::Delete(KTestDatabase1);
   402 		TInt err = TheDb1.Create(KTestDatabase1, &enc[i]);
   403 		TEST2(err, KErrNone);
   404 		//Insert records with empty text column values using RSqlDatabase::Exec()
   405 	    err = TheDb1.Exec(_L("CREATE TABLE A(ID INTEGER, T TEXT)"));
   406 	    TEST(err >= 0);
   407 	    err = TheDb1.Exec(_L("INSERT INTO A VALUES(1, '')"));
   408 	    TEST2(err, 1);
   409 	    err = TheDb1.Exec(_L8("INSERT INTO A VALUES(2, '')"));
   410 	    TEST2(err, 1);
   411 		//Insert a record with empty text column value using RSqlParamWriteStream
   412 	    RSqlStatement stmt;
   413 	    err = stmt.Prepare(TheDb1, _L("INSERT INTO A(ID, T) VALUES(:P1, :P2)"));
   414 	    TEST2(err, KErrNone);
   415 		err = stmt.BindInt(0, 3);
   416 	    TEST2(err, KErrNone);
   417 	    RSqlParamWriteStream strm;
   418 	    err = strm.BindText(stmt, 1);
   419 	    TEST2(err, KErrNone);
   420 	    TRAP(err, strm.WriteL(KNullDesC));
   421 	    TEST2(err, KErrNone);
   422 	    strm.Close();
   423 	    err = stmt.Exec();
   424 	    TEST2(err, 1);
   425 	    stmt.Close();
   426 		//Insert records with empty text column values using RSqlStatement::Bind()
   427 	    err = stmt.Prepare(TheDb1, _L("INSERT INTO A(ID, T) VALUES(:P1, :P2)"));
   428 	    TEST2(err, KErrNone);
   429 		err = stmt.BindInt(0, 4);
   430 	    TEST2(err, KErrNone);
   431 		err = stmt.BindText(1, KNullDesC);
   432 	    TEST2(err, KErrNone);
   433 	    err = stmt.Exec();
   434 	    TEST2(err, 1);
   435 	    //
   436 	    err = stmt.Reset();
   437 	    TEST2(err, KErrNone);
   438 		err = stmt.BindInt(0, 5);
   439 	    TEST2(err, KErrNone);
   440 	    _LIT(KEmptyStr, "");
   441 		err = stmt.BindText(1, KEmptyStr);
   442 	    TEST2(err, KErrNone);
   443 	    err = stmt.Exec();
   444 	    TEST2(err, 1);
   445 	    stmt.Close();
   446 	    //Read the empty text column values
   447 	    err = stmt.Prepare(TheDb1, _L("SELECT T FROM A"));
   448 	    TEST2(err, KErrNone);
   449 	    TInt cnt = 0;
   450 	    while((err = stmt.Next()) == KSqlAtRow)
   451 	    	{
   452 			++cnt;
   453 			TPtrC val;
   454 			err = stmt.ColumnText(0, val);
   455 			TEST2(err, KErrNone);
   456 			TEST2(val.Length(), 0);
   457 			TSqlColumnType type = stmt.ColumnType(0);
   458 			TEST2(type, ESqlText);
   459 	    	}
   460 	    stmt.Close();
   461 	    TEST2(err, KSqlAtEnd);
   462 	    TEST2(cnt, 5);
   463 	    //
   464 	    TheDb1.Close();
   465 	    err = RSqlDatabase::Delete(KTestDatabase1);
   466 	    TEST2(err, KErrNone);
   467 		}
   468 	}
   469 
   470 /**
   471 @SYMTestCaseID          PDS-SQL-CT-4214
   472 @SYMTestCaseDesc        Test for the change "After *#7370# Java apps are not preinstalled again" 
   473 @SYMTestPriority        High
   474 @SYMTestActions         The test makes sure there are no issues if the temp folder is removed after the server 
   475                         has already started. The test performs the following actions - 
   476                         1. Delete the 'temp' directory.
   477                         2. Create a transaction which creates temp files.
   478                         3. Check 'temp' folder exists at the end
   479 @SYMTestExpectedResults Test must not fail
   480 */
   481 void DeleteTempFolder()
   482     {
   483     //1. Delete 'temp' folder
   484     TInt err = TheFs.RmDir(KServerTempDir);
   485     TEST2(err, KErrNone);
   486 	
   487     //2. Create a transaction which creates temp files.
   488     (void)RSqlDatabase::Delete(KTestDatabase1);
   489     err = TheDb1.Create(KTestDatabase1);
   490     TEST2(err, KErrNone);
   491     
   492     err = TheDb1.Exec(_L("CREATE TABLE t1(x UNIQUE); INSERT INTO t1 VALUES(1)"));
   493     TEST(err >= 0);
   494     err = TheDb1.Exec(_L("BEGIN; UPDATE t1 SET x = 2; UPDATE t1 SET x = 3; COMMIT"));
   495     TEST(err >= 0);
   496     
   497     TheDb1.Close();
   498     err = RSqlDatabase::Delete(KTestDatabase1);
   499     TEST2(err, KErrNone);
   500     
   501     //3. Check 'temp' folder exists
   502     err = TheFs.MkDir(KServerTempDir);
   503     TEST2(err, KErrAlreadyExists);
   504     }	
   505 	
   506 void DoTestsL()
   507 	{
   508 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-4154 DEF143062: SQL, \"CREATE INDEX\" sql crashes SQL server"));
   509 	DEF143062();
   510 
   511     TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-4155 DEF143061: SQL, SQLITE_DEFAULT_JOURNAL_SIZE_LIMIT value is too big"));
   512     DEF143061();
   513 
   514     TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-4156 DEF143150: SQL, strftime() returns incorrect result"));
   515     DEF143150();
   516     
   517     TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4210 Temp files created during sql operations are not deleted after rebooting the phone - 1"));
   518     DeleteTempFile();
   519     
   520     TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4211 Temp files created during sql operations are not deleted after rebooting the phone - 2"));
   521     TempFileTest();
   522     
   523     TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4213 No support to store an empty string in symbian's sqlite."));
   524     EmptyTextColumnTest();
   525 	
   526 	TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4214 After *#7370# Java apps are not preinstalled again"));
   527     DeleteTempFolder();
   528 	}
   529 
   530 TInt E32Main()
   531 	{
   532 	TheTest.Title();
   533 	
   534 	CTrapCleanup* tc = CTrapCleanup::New();
   535 	
   536 	__UHEAP_MARK;
   537 	
   538 	CreateTestEnv();
   539 	TRAPD(err, DoTestsL());
   540 	DestroyTestEnv();
   541 	TEST2(err, KErrNone);
   542 
   543 	__UHEAP_MARKEND;
   544 	
   545 	TheTest.End();
   546 	TheTest.Close();
   547 	
   548 	delete tc;
   549 
   550 	User::Heap().Check();
   551 	return KErrNone;
   552 	}