persistentstorage/sqlite3api/TEST/t_sqlitewsd.cpp
author Pat Downey <patd@symbian.org>
Wed, 01 Sep 2010 12:39:58 +0100
branchRCL_3
changeset 24 cc28652e0254
parent 23 26645d81f48d
permissions -rw-r--r--
Revert incorrect RCL_3 drop: Revision: 201035 Kit: 201035

// Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
//

#include <e32test.h>
#include <e32uid.h>
#include <f32file.h>
#include <e32math.h>
#include <sqlite3.h>
#include "t_sqlitewsd.h"

#include <spawn.h>
#include <sys/wait.h>

///////////////////////////////////////////////////////////////////////////////////////

static RTest	TheTest(_L("t_sqlitewsd test"));
static RFs		TheFs;

static pid_t	TheKSqliteWsdProc2Pid = 0;
const char* 	KSqliteWsdProc2Name = "z:\\sys\\bin\\t_sqlitewsd2.exe";

const char* KTestDir = "c:\\test\\";
const char* KTestDb  = "c:\\test\\t_sqlitewsd.db";

sqlite3* TheDb = 0;

///////////////////////////////////////////////////////////////////////////////////////

static void DestroyTestEnv()
	{
	if(TheDb)
		{
		(void)sqlite3_close(TheDb);
		TheDb = 0;
		}
	if(TheFs.Handle() != KNullHandle)
		{
		TFileName fname;
		fname.Copy(TPtrC8((const TUint8*)KTestDb));
		(void)TheFs.Delete(fname);
		}
	TheFs.Close();
	}

///////////////////////////////////////////////////////////////////////////////////////

//Test macros and functions
void Check(TInt aValue, TInt aLine)
	{
	if(!aValue)
		{
		DestroyTestEnv();
		TheTest(EFalse, aLine);
		}
	}
	
void Check(TInt aValue, TInt aExpected, TInt aLine)
	{
	if(aValue != aExpected)
		{
		RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
		const char* errMsg = sqlite3_errmsg(TheDb);
		if(errMsg)
			{
			TBuf<200> msgBuf;
			msgBuf.Copy(TPtrC8((const TUint8*)errMsg));
			RDebug::Print(_L("*** SQLITE error msg: \"%S\".\r\n"), &msgBuf);
			}
		DestroyTestEnv();
		TheTest(EFalse, aLine);
		}
	}

///////////////////////////////////////////////////////////////////////////////////////

static void CreateTestEnv()
    {
	TInt err = TheFs.Connect();
	TEST2(err, KErrNone);

    TFileName testDir;
    testDir.Copy(TPtrC8((const TUint8*)KTestDir));
	err = TheFs.MkDir(testDir);
	TEST(err == KErrNone || err == KErrAlreadyExists);

	TFileName fname;
	fname.Copy(TPtrC8((const TUint8*)KTestDb));
	(void)TheFs.Delete(fname);

	err = sqlite3_open(KTestDb, &TheDb);
	TEST2(err, SQLITE_OK);
	TEST(TheDb != 0);
	}

///////////////////////////////////////////////////////////////////////////////////////

static void CreateDb()
	{
	TEST(TheDb != 0);
	TInt err = sqlite3_exec(TheDb, "CREATE TABLE A(F1 INTEGER)", 0, 0, 0);
	TEST2(err, SQLITE_OK);
	}

static void RunSqliteWsd2()
	{
	TInt err = posix_spawn(&TheKSqliteWsdProc2Pid, KSqliteWsdProc2Name, 0, 0, 0, 0);
	TEST2(err, 0);
	}
	
static void DestroySqliteWsd2()
	{
	(void)waitpid(TheKSqliteWsdProc2Pid, 0, 0);
	}

void DoVerify()
	{
	sqlite3_stmt* stmt = 0;
  	const char* tail = 0;
	TInt err = sqlite3_prepare(TheDb, "SELECT * FROM A", -1, &stmt, &tail);
	TEST2(err, SQLITE_OK);
	TEST(!tail || tail[0] == 0);
	TInt proc1Id1recCnt = 0;
	TInt proc1Id2recCnt = 0;
	TInt proc2Id1recCnt = 0;
	TInt proc2Id2recCnt = 0;
	while((err = sqlite3_step(stmt)) == SQLITE_ROW)
		{
		TInt val = sqlite3_column_int(stmt, 0);
		switch(val)
			{
			case KWsdProc1RecId1: 
				++proc1Id1recCnt;
				break;
			case KWsdProc1RecId2: 
				++proc1Id2recCnt;
				break;
			case KWsdProc2RecId1: 
				++proc2Id1recCnt;
				break;
			case KWsdProc2RecId2: 
				++proc2Id2recCnt;
				break;
			default:
				TEST(0);
				break;
			}
		}
	sqlite3_finalize(stmt);
	TEST2(err, SQLITE_DONE);
	TEST2((proc1Id1recCnt + proc1Id2recCnt), KTestRecordCnt);
	TEST2((proc2Id1recCnt + proc2Id2recCnt), KTestRecordCnt);
	}

/**
@SYMTestCaseID			SYSLIB-SQLITE3-UT-4026
@SYMTestCaseDesc		SQLITE OS porting layer - WSD test.
						The test verifies that the WSD object allocation and access	inside the OS porting layer 
						works properly on both the emulator and the hardware.
						The test runs two separate processes. Each process establishes a connection to the same
						database and inserts 500 records simultaneously.
						During the inserts, the SQLITE OS porting layer will use a mutex to synchronise the database
						operations between the two processes. If the WSD implementation does not work properly,
						then the mutex object won't be allocated per process and the same mutex instance will be used by
						both processes on the emulator. This will lead to panics/asserts inside the OS porting layer.
						The number of the inserted record and record ids is verified at the end of the test.
@SYMTestPriority		High
@SYMTestActions			SQLITE OS porting layer - WSD test.
@SYMTestExpectedResults Test must not fail
@SYMREQ					REQ8782
*/
static void DoWsdTests()
	{
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQLITE3-UT-4026 Create the test database "));
	CreateDb();
	TheTest.Next(_L("Run the second process: t_sqlitewsd2"));
	RunSqliteWsd2();
	TheTest.Next(_L("Insert the records"));
	DoInserts(KWsdProc1Id, KWsdProc1RecId1, KWsdProc1RecId2);
	DestroySqliteWsd2();
	TheTest.Next(_L("Verify the inserted records"));
	DoVerify();
	}

///////////////////////////////////////////////////////////////////////////////////////

TInt E32Main()
	{
	TheTest.Title();
	
	CTrapCleanup* tc = CTrapCleanup::New();
	
	__UHEAP_MARK;
	
	CreateTestEnv();
	DoWsdTests();
	DestroyTestEnv();
	
	__UHEAP_MARKEND;
	
	TheTest.End();
	TheTest.Close();
	
	delete tc;
	
	User::Heap().Check();
	return KErrNone;
	}