kerneltest/e32test/timestamp/t_timestamp.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 31 Aug 2010 16:34:26 +0300
branchRCL_3
changeset 43 c1f20ce4abcf
permissions -rw-r--r--
Revision: 201035 Kit: 201035

// Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "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:
//  
// This test checks the accuracy if NKern::Timestamp (in SMP systems) or
// NKern::Fastcounter (in unicore), when low power modes are being selected
// during idle.
// Platforms requiring this test need to provide a d_timestamp.pdd which
// implements the functions required to test the accurary. If no pdd is 
// supplied the test is skipped and claims to succeed.
// Overview:

//-------------------------------------------------------------------------------------------
//! @SYMTestCaseID				KBASE-t_timestamp-2706
//! @SYMTestCaseDesc			verifying that NKern::Timestamp/FastCounter works correctly 
//!                             accross low power modes
//! @SYMPREQ					417-52765
//! @SYMTestPriority			High
//! @SYMTestActions				
//! 1. This test harness first gets information about NKern::Timestamp or
//!     NKern::FastCounter in particular its frequency. It also obtains
//!     the following constants: Number of timer to run the test, number
//!     of retries before giving up if no low power modes are detected,
//!     and acceptable error percent. Length in nanokernel ticks of each cycle
//! 2. Then it measures the timestamp before and after a time interval 
//!    controlled by NTimerIntervalInS (constant is in seconds)
//!    2.1 If in period of KTimerInterval a low power mode was entered and 
//!        the timer underlying NKern::Timestamp had to be restated then before
//!        and after time is stored. The cycle is considered successful.
//! 3. If KNMaxentries occur with no successful entries the test fails
//! 4. If any valid entry has an interval as measured with NKern::Timestamp that
//!    is outside KNErrPercent of the same interval as measured with nanokernel 
//!    ticks then the test fails
//! 5. If KNValidRuns valid cycles have an acceptable error the test succeeds
//! 
//! @SYMTestExpectedResults 
//!     test passed
//-------------------------------------------------------------------------------------------

#include <e32std.h>
#include <e32base.h>
#include <e32debug.h>
#define __E32TEST_EXTENSION__
#include <e32test.h>
#include <hal.h>
#include <u32hal.h>
#include <e32svr.h>
#include "d_timestamp.h"

_LIT(KTimestampTestLddFileName,"D_TIMESTAMP.LDD");
_LIT(KTimestampTestPddFileName,"D_TIMESTAMP.PDD");

RTest test(_L("T_TIMESTAMP"));

LOCAL_C void UnloadDrivers()
	{
    test.Printf(_L("Unloading LDD\n"));

	TInt r=User::FreeLogicalDevice(RTimestampTest::Name());
    test_KErrNone(r);
    
	TName pddName(RTimestampTest::Name());
	_LIT(KPddWildcardExtension,".*");
	pddName.Append(KPddWildcardExtension);
	TFindPhysicalDevice findPD(pddName);
	TFullName findResult;
	r=findPD.Next(findResult);
	while (r==KErrNone)
		{
        test.Printf(_L("Unloading PDD: %S\n"),&findResult);
		r=User::FreePhysicalDevice(findResult);
        test_KErrNone(r);
		findPD.Find(pddName); // Reset the find handle now that we have deleted something from the container.
		r=findPD.Next(findResult);
		} 
	}


GLDEF_C TInt E32Main()
    {

    TBool dontFail = (User::CommandLineLength()!=0);
    
    test.Title();

    test.Start(_L("Timestamp accuracy test"));
    

    TInt r;
    TRequestStatus st;
    
    r = User::LoadPhysicalDevice(KTimestampTestPddFileName);
    if (KErrNotFound == r)
        {
        test.Printf(_L("No timestamp pdd, test skipped\n"));
        test.End(); // skip test if this platform does not supply a PDD
        return 0;
        }
    

	r=User::LoadLogicalDevice(KTimestampTestLddFileName);
	test(r==KErrNone || r==KErrAlreadyExists);
    
    RTimestampTest ldd;
    r = ldd.Open();
    test_KErrNone(r);
    
    test.Next(_L("Get timestamp frequency"));
    STimestampTestConfig info;
    r = ldd.Config(info);
    test_KErrNone(r);
    

    TUint retries = 0;
    TUint validruns = 0;
    
    test.Next(_L("Get nanotick frequency"));
	TInt tickPeriod = 0;
	r = HAL::Get(HAL::ENanoTickPeriod, tickPeriod);
    test_KErrNone(r);
    test.Printf(_L(" tick period in uS== %d\n"), tickPeriod);

    TInt ticks = info.iTimerDurationS*1000000/tickPeriod;
    TUint64 expected = info.iTimerDurationS*info.iFreq;
    TUint64 acceptError = info.iErrorPercent*expected/100;
    
    test.Printf(_L("running at %dHz for %d interations, each lasting %d seconds and with %d retries\n"),
                info.iFreq,
                info.iIterations,
                info.iTimerDurationS,
                info.iRetries
        );
    
    test.Printf(_L("expecting %lu with up to %lu error\n"),expected,acceptError);

    test.Next(_L("test timer interval"));
    STimestampResult result;
    STimestampResult* validResults = new STimestampResult[info.iIterations];
    memset(&validResults[0],0,sizeof(validResults));
    ldd.Start(st,ticks);
    User::WaitForRequest(st);
    test_KErrNone(st.Int());
    
    FOREVER
        {
        ldd.WaitOnTimer(st,result);
        User::WaitForRequest(st);
        test_KErrNone(st.Int());
        TUint64 error = (result.iDelta>expected) ? result.iDelta-expected : expected - result.iDelta;

        if (error < acceptError)
            {
            test.Printf(_L("Got %lu expected %lu, LPM Entered:%d, error %lu is OK \n"),
                        result.iDelta,expected,result.iLPMEntered,error);
            }
        else 
            {
            test.Printf(_L("Got %lu expected %lu, LPM Entered:%d, error %lu is BAD\n"),
                        result.iDelta,expected,result.iLPMEntered,error);
            if (!dontFail) 
                {
                delete [] validResults;
                ldd.Close();
                UnloadDrivers();
                test(error < acceptError);
                }
            }

        if (result.iLPMEntered)
            {
            retries = 0;
            validResults[validruns] = result;
            if (++validruns==info.iIterations) break;
            }
        else
            {
            retries++;
            if (retries==info.iRetries) 
                {
                test.Printf(_L("several retries with no power mode entry ... aborting ...\n"));
                ldd.Close();
                delete [] validResults;
                UnloadDrivers();
                test_Compare(retries,<,info.iRetries);
                }
            
            }
        }

    delete [] validResults;
    ldd.Close();
    UnloadDrivers();
    test.End();
	return(0);
    }