kerneltest/e32test/nkernsa/testutils.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Mon, 21 Jun 2010 17:12:14 +0300
branchRCL_3
changeset 198 2bb754abd467
parent 0 a41df078684a
permissions -rw-r--r--
Revision: 201025 Kit: 2010125

// Copyright (c) 2006-2009 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:
// e32test\nkernsa\testutils.cpp
// 
//

#include <nktest/nkutils.h>

extern "C" {
TUint32 random(TUint32* aSeed)
	{
	TUint32 x = aSeed[0];
	TUint32 r3 = x >> 1;
	r3 |= (aSeed[1] << 31);
	aSeed[1] = x & 1;
	r3 ^= (x << 12);
	x = r3 ^ (r3 >> 20);
	aSeed[0] = x;
	return x;
	}

void setup_block(TUint32* aBlock, TInt aNumWords)
	{
	TUint32 seed[2];
	seed[0] = aBlock[0];
	seed[1] = 0;
	TInt i;
	for (i=1; i<aNumWords; ++i)
		*++aBlock = random(seed);
	}

void setup_block_cpu(TUint32* aBlock, TInt aNumWords)
	{
	TUint32 seed[2];
	seed[0] = aBlock[0];
	seed[1] = 0;
	aBlock[1] = NKern::CurrentCpu();
	TInt i;
	for (i=2; i<aNumWords; ++i)
		aBlock[i] = random(seed) ^ NKern::CurrentCpu();
	}

TBool verify_block(const TUint32* aBlock, TInt aNumWords)
	{
	TUint32 seed[2];
	seed[0] = aBlock[0];
	seed[1] = 0;
	TInt i;
	for (i=1; i<aNumWords; ++i)
		{
		TUint32 x = random(seed);
		if (aBlock[i] != x)
			{
			KPrintf("Verify block failed: expected %08x got %08x", x, aBlock[i]);
			return FALSE;
			}
		}
	return TRUE;
	}

TBool verify_block_no_trace(const TUint32* aBlock, TInt aNumWords)
	{
	TUint32 seed[2];
	seed[0] = aBlock[0];
	seed[1] = 0;
	TInt i;
	for (i=1; i<aNumWords; ++i)
		{
		TUint32 x = random(seed);
		if (aBlock[i] != x)
			return FALSE;
		}
	return TRUE;
	}

TInt verify_block_cpu(const TUint32* aBlock, TInt aNumWords)
	{
	TUint32 seed[2];
	seed[0] = aBlock[0];
	seed[1] = 0;
	TUint32 cpu = aBlock[1];
	TInt i;
	for (i=2; i<aNumWords; ++i)
		{
		TUint32 x = random(seed);
		TUint32 y = aBlock[i] ^ x;
		if (y != cpu)
			{
			KPrintf("Verify block with CPU failed: expected %08x got %08x (XOR %08x)", x, aBlock[i], y);
			return -1;
			}
		}
	return cpu;
	}

TInt verify_block_cpu_no_trace(const TUint32* aBlock, TInt aNumWords)
	{
	TUint32 seed[2];
	seed[0] = aBlock[0];
	seed[1] = 0;
	TUint32 cpu = aBlock[1];
	TInt i;
	for (i=2; i<aNumWords; ++i)
		{
		TUint32 x = random(seed);
		TUint32 y = aBlock[i] ^ x;
		if (y != cpu)
			return -1;
		}
	return cpu;
	}
}


CircBuf* CircBuf::New(TInt aSlots)
	{
	CircBuf* p = new CircBuf();
	p->iSlotCount = aSlots;
	p->iGetIndex = 0;
	p->iPutIndex = 0;
	p->iBufBase = (TUint32*)malloc(aSlots*sizeof(TUint32));
	if (!p->iBufBase)
		{
		delete p;
		p = 0;
		}
	return p;
	}

CircBuf::CircBuf()
	:	iLock(TSpinLock::EOrderGenericIrqLow1)
	{
	}

CircBuf::~CircBuf()
	{
	free(iBufBase);
	}

TInt CircBuf::TryGet(TUint32& aOut)
	{
	TInt r = KErrUnderflow;
	TInt irq = __SPIN_LOCK_IRQSAVE(iLock);
	if (iGetIndex != iPutIndex)
		{
		aOut = iBufBase[iGetIndex++];
		if (iGetIndex == iSlotCount)
			iGetIndex = 0;
		r = KErrNone;
		}
	__SPIN_UNLOCK_IRQRESTORE(iLock,irq);
	return r;
	}

TInt CircBuf::TryPut(TUint32 aIn)
	{
	TInt r = KErrOverflow;
	TInt irq = __SPIN_LOCK_IRQSAVE(iLock);
	TInt nextPut = iPutIndex + 1;
	if (nextPut == iSlotCount)
		nextPut = 0;
	if (iGetIndex != nextPut)
		{
		iBufBase[iPutIndex] = aIn;
		iPutIndex = nextPut;
		r = KErrNone;
		}
	__SPIN_UNLOCK_IRQRESTORE(iLock,irq);
	return r;
	}

TUint32 CircBuf::Get()
	{
	TUint32 x;
	while(TryGet(x)!=KErrNone)
		NKern::Sleep(1);
	return x;
	}

void CircBuf::Put(TUint32 aIn)
	{
	while(TryPut(aIn)!=KErrNone)
		NKern::Sleep(1);
	}

void CircBuf::Reset()
	{
	TInt irq = __SPIN_LOCK_IRQSAVE(iLock);
	iPutIndex = iGetIndex = 0;
	__SPIN_UNLOCK_IRQRESTORE(iLock,irq);
	}

TInt CircBuf::Count()
	{
	TInt irq = __SPIN_LOCK_IRQSAVE(iLock);
	TInt r = iPutIndex - iGetIndex;
	if (r < 0)
		r += iSlotCount;
	__SPIN_UNLOCK_IRQRESTORE(iLock,irq);
	return r;
	}