Update contrib.
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
23 static RTest TheTest(_L("t_dbperf3"));
24 static CTrapCleanup* TheTrapCleanup;
27 static RDbNamedDatabase TheDatabase;
28 static RDbRowSet TheRowSet, TheRowSet2;
29 static RFile TheTestFile;
30 static TFileName TheDatabaseFileName;
32 #define COUNT_OF(array) (sizeof(array)/sizeof(array[0]))
34 const TInt KTrackRecordCount = 200;
35 const TInt KCategoryRecordCount = 50;
36 const TInt KStatsRecordCount = 50;
37 const TInt KTestBlobSize = 4096;
39 //_LIT(KCreateTrackTable, "CREATE TABLE TRACKS (id INTEGER, marked_2_play INTEGER, category_id INTEGER, artist_last_name CHAR(15) NOT NULL,artist_first_name CHAR(15) NOT NULL, title CHAR(16) NOT NULL,download_site CHAR(30) NOT NULL,band_name CHAR(20) NOT NULL,origin CHAR(16),autostart INTEGER, init_volume INTEGER, music_file LONG VARCHAR)");
40 _LIT(KCreateTrackIndex, "CREATE INDEX IDX1 ON TRACKS(id,marked_2_play,category_id)");
41 _LIT(KCreateTrackTable, "CREATE TABLE TRACKS (id INTEGER, marked_2_play INTEGER, category_id INTEGER, artist_last_name CHAR(15) NOT NULL,artist_first_name CHAR(15) NOT NULL, title CHAR(16) NOT NULL,download_site CHAR(30) NOT NULL,band_name CHAR(20) NOT NULL,origin CHAR(16),autostart INTEGER, init_volume INTEGER)");
42 _LIT(KCreateTrackTable2, "CREATE TABLE TRACKS2 (id INTEGER, music_file LONG VARCHAR)");
43 //_LIT(KCreateTrackIndex2, "CREATE INDEX IDX4 ON TRACKS2(id)");
45 //_LIT(KTrackTable,"TRACKS");
48 _LIT(KLastName,"artist_last_name");
49 _LIT(KFirstName,"artist_first_name");
51 _LIT(KDownloadSite,"download_site");
52 _LIT(KBandName,"band_name");
53 _LIT(KOrigin,"origin");
54 _LIT(KAutoStart,"autostart");
55 _LIT(KInitVolume,"init_volume");
56 _LIT(KMarked2Play,"marked_2_play");
57 _LIT(KCategoryId,"category_id");
58 _LIT(KMusicFile,"music_file");
62 _LIT(KCreateCategoryTable, "CREATE TABLE CATEGORY (category_id INTEGER,category_name CHAR(20),genre INTEGER)");
63 _LIT(KCreateCategoryIndex, "CREATE INDEX IDX2 ON CATEGORY(category_id)");
65 //_LIT(KCategoryTable,"CATEGORY");
67 //KCategoryId defined for category table
68 _LIT(KCategoryName,"category_name");
73 _LIT(KCreatestatsTable, "CREATE TABLE STATS (category_id INTEGER, no_of_tracks INTEGER, no_autostart INTEGER,no_manualstart INTEGER,no_marked_2_play INTEGER, no_unmarked_2_play INTEGER, size_of_musicfiles INTEGER)");
74 _LIT(KCreatestatsIndex, "CREATE UNIQUE INDEX IDX3 ON STATS(category_id)");
76 //_LIT(KStatsTable,"STATS");
78 //KCategoryId defined for category table
79 _LIT(KNoOfTracks,"no_of_tracks");
80 _LIT(KNoMarked2Play,"no_marked_2_play");
81 _LIT(KNoUnmarked2Play,"no_unmarked_2_play");
82 _LIT(KNoAutostart,"no_autostart");
83 _LIT(KNoManualStart,"no_manualstart");
84 _LIT(KSizeOfMusicFiles,"size_of_musicfiles");
86 //////////////////////////////////////////////////////
88 static TInt TheCounterFreq = -10000000;
89 const TInt KMicroSecIn1Sec = 1000000;
91 TUint32 CalcTickDiff(TUint32 aStartTicks, TUint32 aEndTicks)
93 TInt64 diffTicks = (TInt64)aEndTicks - (TInt64)aStartTicks;
96 diffTicks = KMaxTUint32 + diffTicks + 1;
98 return (TUint32)diffTicks;
101 //Prints aFastCount parameter (converted to us)
102 void PrintFcDiffAsUs(const TDesC& aFormatStr, TUint32 aFastCount)
104 double v = ((double)aFastCount * KMicroSecIn1Sec) / (double)TheCounterFreq;
106 TheTest.Printf(aFormatStr, v2);
109 ///////////////////////////////////////////////////////////////////////////////////////
111 //Delete "aFullName" file.
112 static void DeleteFile(const TDesC& aFullName)
115 TInt err = fsSession.Connect();
119 if(fsSession.Entry(aFullName, entry) == KErrNone)
121 err = fsSession.SetAtt(aFullName, 0, KEntryAttReadOnly);
124 TheTest.Printf(_L("Error %d changing \"%S\" file attributes.\n"), err, &aFullName);
126 err = fsSession.Delete(aFullName);
129 TheTest.Printf(_L("Error %d deleting \"%S\" file.\n"), err, &aFullName);
136 TheTest.Printf(_L("Error %d connecting file session. File: %S.\n"), err, &aFullName);
140 ///////////////////////////////////////////////////////////////////////////////////////
142 static void CloseAll()
151 ///////////////////////////////////////////////////////////////////////////////////////
152 ///////////////////////////////////////////////////////////////////////////////////////
153 //Tests macros and functions.
154 //If (!aValue) then the test will be panicked, the test data files will be deleted.
155 static void Check(TInt aValue, TInt aLine)
160 DeleteFile(TheDatabaseFileName);
161 TheTest(EFalse, aLine);
164 //If (aValue != aExpected) then the test will be panicked, the test data files will be deleted.
165 static void Check(TInt aValue, TInt aExpected, TInt aLine)
167 if(aValue != aExpected)
169 TheTest.Printf(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
171 DeleteFile(TheDatabaseFileName);
172 TheTest(EFalse, aLine);
175 //Use these to test conditions.
176 #define TEST(arg) ::Check((arg), __LINE__)
177 #define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
179 ///////////////////////////////////////////////////////////////////////////////////////
183 RDbDatabase::TSize s = TheDatabase.Size();
184 TheTest.Printf(_L("####FileSize: %d\r\n"), s.iSize);
187 ///////////////////////////////////////////////////////////////////////////////////////
189 void FillRandomData(TDes& aData, TInt64 aSeed)
192 for (TInt i=0; i<aData.MaxLength(); ++i)
194 // add next character (we stick to lowercase alphabet for now)
195 aData.Append(TChar(Math::FRand(aSeed)*25 + 'a'));
199 void FillRandomData(TDes& aData)
203 time.UniversalTime();
204 TInt64 seed = time.Int64();
206 FillRandomData(aData, seed);
209 ///////////////////////////////////////////////////////////////////////////////////////
211 void CreateDatabase()
213 TInt err = TheDatabase.Replace(TheFs, TheDatabaseFileName);
214 TEST2(err, KErrNone);
216 err = TheDbs.Connect();
217 TEST2(err, KErrNone);
218 err = TheDatabase.Open(TheDbs, TheDatabaseFileName);
219 TEST2(err, KErrNone);
220 err = TheDatabase.Execute(KCreateTrackTable);
221 TEST2(err, KErrNone);
222 err = TheDatabase.Execute(KCreateTrackIndex);
223 TEST2(err, KErrNone);
224 err = TheDatabase.Execute(KCreateTrackTable2);
225 TEST2(err, KErrNone);
226 //err = TheDatabase.Execute(KCreateTrackIndex2);
227 //TEST2(err, KErrNone);
228 err = TheDatabase.Execute(KCreateCategoryTable);
229 TEST2(err, KErrNone);
230 err = TheDatabase.Execute(KCreateCategoryIndex);
231 TEST2(err, KErrNone);
232 err = TheDatabase.Execute(KCreatestatsTable);
233 TEST2(err, KErrNone);
234 err = TheDatabase.Execute(KCreatestatsIndex);
235 TEST2(err, KErrNone);
236 //err = TheDatabase.Compact();
237 //TEST2(err, KErrNone);
240 void InsertTrackTableL()
242 HBufC* randomDataBuf = HBufC::NewLC(KTestBlobSize);
243 TPtr randomData(randomDataBuf->Des());
244 FillRandomData(randomData);
247 TInt err = view.Prepare(TheDatabase, _L("select * from TRACKS"), view.EInsertOnly);
248 TEST2(err, KErrNone);
251 CDbColSet* colSet = TheRowSet.ColSetL();
252 const TInt KIdIdx = colSet->ColNo(KId);
253 const TInt KLastNameIdx = colSet->ColNo(KLastName);
254 const TInt KFirstNameIdx = colSet->ColNo(KFirstName);
255 const TInt KTitleIdx = colSet->ColNo(KTitle);
256 const TInt KDownloadSiteIdx = colSet->ColNo(KDownloadSite);
257 const TInt KBandNameIdx = colSet->ColNo(KBandName);
258 const TInt KOriginIdx = colSet->ColNo(KOrigin);
259 const TInt KAutoStartIdx = colSet->ColNo(KAutoStart);
260 const TInt KInitVolumeIdx = colSet->ColNo(KInitVolume);
261 const TInt KMarkedToPlayIdx = colSet->ColNo(KMarked2Play);
262 const TInt KCategoryIdIdx = colSet->ColNo(KCategoryId);
263 //const TInt KMusicFileIdx = colSet->ColNo(KMusicFile);
267 err = TheDatabase.Begin();
268 TEST2(err, KErrNone);
270 for (TInt ii=1;ii<=KTrackRecordCount;++ii)
273 TheRowSet.SetColL(KIdIdx, ii);
274 TheRowSet.SetColL(KLastNameIdx, _L("Dummy"));
275 TheRowSet.SetColL(KFirstNameIdx,_L("Dummy"));
276 TheRowSet.SetColL(KTitleIdx,_L("Dummy"));
277 TheRowSet.SetColL(KDownloadSiteIdx,_L("Dummy"));
278 TheRowSet.SetColL(KBandNameIdx,_L("Dummy"));
279 TheRowSet.SetColL(KOriginIdx,_L("Dummy"));
280 TheRowSet.SetColL(KAutoStartIdx,(ii%2));
281 TheRowSet.SetColL(KInitVolumeIdx,(ii%2));
282 TheRowSet.SetColL(KMarkedToPlayIdx,(ii%2));
283 TheRowSet.SetColL(KCategoryIdIdx,(ii%KCategoryRecordCount));
285 //RDbColWriteStream musicfile;
286 //musicfile.OpenLC(TheRowSet, KMusicFileIdx);
287 //musicfile.WriteL(randomData,KTestBlobSize);
288 //musicfile.CommitL();
289 //CleanupStack::PopAndDestroy(&musicfile);
294 err = TheDatabase.Commit();
295 TEST2(err, KErrNone);
297 //err = TheDatabase.Compact();
298 //TEST2(err, KErrNone);
302 ////////////////////////////////////////////////////////////////////////////////////////////////
304 err = view.Prepare(TheDatabase, _L("select * from TRACKS2"), view.EInsertOnly);
305 TEST2(err, KErrNone);
308 colSet = TheRowSet.ColSetL();
309 const TInt KIdIdx2 = colSet->ColNo(KId);
310 const TInt KMusicFileIdx2 = colSet->ColNo(KMusicFile);
313 err = TheDatabase.Begin();
314 TEST2(err, KErrNone);
316 for (TInt ii=1;ii<=KTrackRecordCount;++ii)
319 TheRowSet.SetColL(KIdIdx2, ii);
321 RDbColWriteStream musicfile;
322 musicfile.OpenLC(TheRowSet, KMusicFileIdx2);
323 musicfile.WriteL(randomData,KTestBlobSize);
325 CleanupStack::PopAndDestroy(&musicfile);
330 err = TheDatabase.Commit();
331 TEST2(err, KErrNone);
333 //err = TheDatabase.Compact();
334 //TEST2(err, KErrNone);
338 CleanupStack::PopAndDestroy(randomDataBuf);
341 void InsertCategoryTableL()
344 TInt err = view.Prepare(TheDatabase, _L("select * from CATEGORY"), view.EInsertOnly);
345 TEST2(err, KErrNone);
348 CDbColSet* colSet = TheRowSet.ColSetL();
349 const TInt KCategoryIdIdx = colSet->ColNo(KCategoryId);
350 const TInt KCategoryNameIdx = colSet->ColNo(KCategoryName);
351 const TInt KGenreIdx = colSet->ColNo(KGenre);
354 err = TheDatabase.Begin();
355 TEST2(err, KErrNone);
357 for (TInt ii=1;ii<=KCategoryRecordCount;++ii)
360 TheRowSet.SetColL(KCategoryIdIdx, ii);
361 TheRowSet.SetColL(KCategoryNameIdx, _L("History"));
362 TheRowSet.SetColL(KGenreIdx,(ii*500));
366 err = TheDatabase.Commit();
367 TEST2(err, KErrNone);
369 //err = TheDatabase.Compact();
370 //TEST2(err, KErrNone);
375 void InsertStatsTableL()
378 TInt err = view.Prepare(TheDatabase, _L("select * from STATS"), view.EInsertOnly);
379 TEST2(err, KErrNone);
382 CDbColSet* colSet = TheRowSet.ColSetL();
383 const TInt KCategoryIdIdx = colSet->ColNo(KCategoryId);
384 const TInt KTrackCntIdx = colSet->ColNo(KNoOfTracks);
385 const TInt KMarkedToPlayCntIdx = colSet->ColNo(KNoMarked2Play);
386 const TInt KUnmarkedToPlayCntIdx = colSet->ColNo(KNoUnmarked2Play);
387 const TInt KAutoStartCntIdx = colSet->ColNo(KNoAutostart);
388 const TInt KManualStartCntIdx = colSet->ColNo(KNoManualStart);
389 const TInt KSizeMusicFilesIdx = colSet->ColNo(KSizeOfMusicFiles);
392 TInt default_Stat = 0;
394 err = TheDatabase.Begin();
395 TEST2(err, KErrNone);
397 for (TInt ii=0;ii<KStatsRecordCount;++ii)
400 TheRowSet.SetColL(KCategoryIdIdx, ii);
401 TheRowSet.SetColL(KTrackCntIdx,default_Stat);
402 TheRowSet.SetColL(KMarkedToPlayCntIdx,default_Stat);
403 TheRowSet.SetColL(KUnmarkedToPlayCntIdx,default_Stat);
404 TheRowSet.SetColL(KAutoStartCntIdx,default_Stat);
405 TheRowSet.SetColL(KManualStartCntIdx,default_Stat);
406 TheRowSet.SetColL(KSizeMusicFilesIdx,default_Stat);
410 err = TheDatabase.Commit();
411 TEST2(err, KErrNone);
413 //err = TheDatabase.Compact();
414 //TEST2(err, KErrNone);
422 InsertCategoryTableL();
426 void DestroyDatabase()
431 TInt err = TheFs.Delete(TheDatabaseFileName);
432 TEST2(err, KErrNone);
435 ///////////////////////////////////////////////////////////////////////////////////////
437 void GetFastCounterFrequency()
439 TEST2(HAL::Get(HAL::EFastCounterFrequency, TheCounterFreq), KErrNone);
440 TheTest.Printf(_L("Counter frequency=%d\r\n"), TheCounterFreq);
443 ///////////////////////////////////////////////////////////////////////////////////////
445 void SelectTracksL(TInt aCount, RArray<TInt>& aTracks)
447 TUint32 fc = User::FastCounter();
449 TUint32 fc2 = User::FastCounter();
450 TInt err = view.Prepare(TheDatabase, _L("select id from TRACKS"), view.EReadOnly);
451 TEST2(err, KErrNone);
452 PrintFcDiffAsUs(_L("###\"Prepare()\",time=%d us\r\n"), CalcTickDiff(fc2, User::FastCounter()));
453 fc2 = User::FastCounter();
454 err = view.EvaluateAll();
455 TEST2(err, KErrNone);
456 PrintFcDiffAsUs(_L("###\"EvaluateAll()\",time=%d us\r\n"), CalcTickDiff(fc2, User::FastCounter()));
459 TUint32 diff1 = 0, diff2 = 0;
461 while(count < aCount)
463 fc2 = User::FastCounter();
464 if(!TheRowSet.NextL())
468 diff1 += CalcTickDiff(fc2, User::FastCounter());
469 fc2 = User::FastCounter();
471 diff2 += CalcTickDiff(fc2, User::FastCounter());
472 aTracks.Append(TheRowSet.ColInt(1));
475 TEST2(count, aCount);
476 PrintFcDiffAsUs(_L("###\"Total NextL()\",time=%d us\r\n"), diff1);
477 PrintFcDiffAsUs(_L("###\"Total GetL()\",time=%d us\r\n"), diff2);
479 fc2 = User::FastCounter();
481 PrintFcDiffAsUs(_L("###\"Close()\",time=%d us\r\n"), CalcTickDiff(fc2, User::FastCounter()));
482 PrintFcDiffAsUs(_L("###\"SELECT FROM TRACKS\",time=%d us\r\n"), CalcTickDiff(fc, User::FastCounter()));
486 @SYMTestCaseID PDS-DBMS-UT-4009
487 @SYMTestCaseDesc DBMS performance tests.
488 @SYMTestPriority High
489 @SYMTestActions The test opens the test database and:
490 - selects the ids of the tracks to be deleted and collects them into an array;
491 - deletes the recods with matching track ids from TRACK table;
492 - deletes the recods with matching track ids from TRACK2 table;
493 The execution times are printed out.
494 @SYMTestExpectedResults Test must not fail
499 TheTest.Printf(_L("Record count: %d\r\n"), KTrackRecordCount);
502 tracks.ReserveL(KTrackRecordCount);
503 CleanupClosePushL(tracks);
504 SelectTracksL(KTrackRecordCount, tracks);
506 _LIT(KDeleteSql, "DELETE FROM tracks WHERE id>=%d AND id<=%d");
509 sql.Format(KDeleteSql, tracks[0], tracks[tracks.Count() - 1]);
511 TUint32 fc2 = User::FastCounter();
513 TInt err = TheDatabase.Begin();
514 TEST2(err, KErrNone);
516 TUint32 fc = User::FastCounter();
517 TInt rc = TheDatabase.Execute(sql);
518 PrintFcDiffAsUs(_L("###\"DELETE FROM TRACKS\",time=%d us\r\n"), CalcTickDiff(fc, User::FastCounter()));
519 TEST2(rc, KTrackRecordCount);
520 TheTest.Printf(_L("Deleted record count: %d\r\n"), rc);
522 sql.Replace(12, 6, _L("TRACKS2"));
523 fc = User::FastCounter();
524 rc = TheDatabase.Execute(sql);
525 PrintFcDiffAsUs(_L("###\"DELETE FROM TRACKS2\",time=%d us\r\n"), CalcTickDiff(fc, User::FastCounter()));
526 TEST2(rc, KTrackRecordCount);
527 TheTest.Printf(_L("Deleted record count: %d\r\n"), rc);
529 err = TheDatabase.Commit();
530 TEST2(err, KErrNone);
532 PrintFcDiffAsUs(_L("###Total \"DELETE FROM TRACKS\",time=%d us\r\n"), CalcTickDiff(fc2, User::FastCounter()));
534 CleanupStack::PopAndDestroy(&tracks);
537 ///////////////////////////////////////////////////////////////////////////////////////
539 //This test checks how RDbDatabase::Commit() works if there are some active RDbRowSet objects -
540 //read-only and updatable. The expectation is that the Commit() call won't fail, the RDbRowSet objects
541 //retain their pre-commit positions.
544 //Create 2 test tables, insert some records in the first table
546 TInt err = TheDatabase.Replace(TheFs, TheDatabaseFileName);
547 TEST2(err, KErrNone);
548 err = TheDatabase.Execute(_L("CREATE TABLE AA1(Id INTEGER)"));
549 TEST2(err, KErrNone);
550 err = TheDatabase.Execute(_L("CREATE TABLE AA2(Id INTEGER)"));
551 TEST2(err, KErrNone);
552 err = TheDatabase.Execute(_L("INSERT INTO AA1(Id) VALUES(1)"));
554 err = TheDatabase.Execute(_L("INSERT INTO AA1(Id) VALUES(2)"));
556 err = TheDatabase.Execute(_L("INSERT INTO AA1(Id) VALUES(3)"));
559 err = TheDatabase.Begin();
560 TEST2(err, KErrNone);
561 //Prepare read-only view and call FirstL() (TheRowSet object)
563 err = view.Prepare(TheDatabase, _L("select * from AA1"), RDbRowSet::EReadOnly);
564 TEST2(err, KErrNone);
566 err = view.EvaluateAll(); //DBMS can use FirstL() without the EvaluateAll() call in this case
567 TEST2(err, KErrNone);
568 TBool rc = TheRowSet.FirstL();
570 //Prepare updatable view and call NextL() (TheRowSet2 object)
571 err = view.Prepare(TheDatabase, _L("select * from AA1"), RDbRowSet::EUpdatable);
572 TEST2(err, KErrNone);
574 err = view.EvaluateAll(); //DBMS can use NextL() without the EvaluateAll() call in this case
575 TEST2(err, KErrNone);
576 rc = TheRowSet2.FirstL();
578 rc = TheRowSet2.NextL();
580 //Execute one INSERT statement
581 err = TheDatabase.Execute(_L("INSERT INTO AA2(Id) VALUES(1)"));
584 err = TheDatabase.Commit();
585 TEST2(err, KErrNone);
586 //Check the retrieved by TheRowSet record
588 TEST2(TheRowSet.ColInt(1), 1);
589 //Check the retrieved by TheRowSet2 record
591 TEST2(TheRowSet2.ColInt(1), 2);
598 ///////////////////////////////////////////////////////////////////////////////////////
602 TheTest.Start(_L("Get fast counter frequency"));
603 GetFastCounterFrequency();
605 TheTest.Next(_L("Create&Fill test database"));
609 TheTest.Next(_L(" @SYMTestCaseID:PDS-DBMS-UT-4009 Delete tracks"));
612 TheTest.Next(_L("Commit() test (not a performance test)"));
616 ///////////////////////////////////////////////////////////////////////////////////////
618 //Usage: "t_dbperf3 [<drive letter>:]"
623 TheTrapCleanup = CTrapCleanup::New();
624 TEST(TheTrapCleanup != NULL);
626 //Construct test database file name
627 _LIT(KTestDatabase, "c:\\dbms-tst\\t_dbperf3.db");
629 User::CommandLine(fname);
631 parse.Set(fname, &KTestDatabase, 0);
632 const TDesC& dbFilePath = parse.FullName();
633 TheDatabaseFileName.Copy(dbFilePath);
634 TheTest.Printf(_L("Test database: %S\r\n"), &TheDatabaseFileName);
638 TInt err = TheFs.Connect();
639 TEST2(err, KErrNone);
640 err = TheFs.MkDir(TheDatabaseFileName);
641 TheTest.Printf(_L("MkDir(): err=%d\r\n"), err);
642 TEST(err == KErrNone || err == KErrAlreadyExists);
644 DeleteFile(TheDatabaseFileName);
646 TRAP(err, DoTestL());
647 TEST2(err, KErrNone);
650 DeleteFile(TheDatabaseFileName);
657 delete TheTrapCleanup;