First submission to Symbian Foundation staging server.
// memsamplerdd.cpp
// 
// Copyright (c) 2008 - 2010 Accenture. All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the "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:
// Accenture - Initial contribution
//
#include <fshell/extrabtrace.h>
#include <platform.h>
#include <kern_priv.h>
#include "memsamplerdd.h"
#include <fshell/common.mmh>
const TInt KMinRate = 10;
const TInt KMaxRate = 10000;
const TInt KMajorVersionNumber = 1;
const TInt KMinorVersionNumber = 0;
const TInt KBuildVersionNumber = 0;
const TInt KArrayGranularity = 32;
const TInt KDfcThreadPriority = 26;
_LIT(KDfcThreadName, "MemSamplerDD");
class DDeviceMemSampler : public DLogicalDevice
	{
public:
	DDeviceMemSampler();
	virtual TInt Install();
	virtual void GetCaps(TDes8& aDes) const;
	virtual TInt Create(DLogicalChannelBase*& aChannel);
	};
class TChunkInfo
	{
public:
	TChunkInfo(DChunk& aChunk);
	TBool operator==(const TChunkInfo& aChunkInfo);
public:
	DChunk* iAddress;
	TInt iSize;
	TInt iHighWaterMark;
	TBool iSeen;
	};
TChunkInfo::TChunkInfo(DChunk& aChunk)
	: iAddress(&aChunk), iSize(aChunk.Size()), iHighWaterMark(aChunk.Size()), iSeen(ETrue)
	{
	}
TBool TChunkInfo::operator==(const TChunkInfo& aChunkInfo)
	{
	return (iSize == aChunkInfo.iSize);
	}
class DMemSamplerChannel : public DLogicalChannel
	{
public:
	DMemSamplerChannel();
	~DMemSamplerChannel();
protected:
	virtual TInt DoCreate(TInt aUnit, const TDesC8* anInfo, const TVersion& aVer);
	virtual void HandleMsg(TMessageBase* aMsg);
private:
	TInt StartSampling(TInt aRate);
	TInt StopSampling();
	TInt ResetSampling();
	TInt ReadChunks();
	void TraceNewChunk(const TChunkInfo& aChunkInfo) const;
	void TraceChangedChunk(const TChunkInfo& aChunkInfo) const;
	void TraceDeletedChunk(const TChunkInfo& aChunkInfo) const;
	void TraceNewSample() const;
#if FSHELL_PLATFORM_SYMTB >= 92
	inline TBool Running()
		{return iTimer.IsPending();}
#else
	inline TBool Running()
		{return iTimer.iState!=NTimer::EIdle;}
#endif
private:
	static void Sample(TAny*);
	static void DoSample(TAny*);
	static void ExitDfcThread(TAny*);
private:
	RArray<TChunkInfo> iChunkInfo;
	NTimer iTimer;
	TInt iPeriod;
	TInt iPeriodNumber;
	TDfcQue iPrivateDfcQ;
	TDfc iDoSampleDfc;
	TDfc iExitDfc;
	DThread* iClient;
	mutable TBool iNewSampleTraced;
	};
DECLARE_STANDARD_LDD()
	{
	return new DDeviceMemSampler;
	}
DDeviceMemSampler::DDeviceMemSampler()
	{
	iVersion=TVersion(KMajorVersionNumber, KMinorVersionNumber, KBuildVersionNumber);
	}
TInt DDeviceMemSampler::Install()
	{
	TInt r = SetName(&KMemSamplerName);
	return r;
	}
void DDeviceMemSampler::GetCaps(TDes8&) const
	{
	}
TInt DDeviceMemSampler::Create(DLogicalChannelBase*& aChannel)
	{
	aChannel = new DMemSamplerChannel;
	return aChannel ? KErrNone : KErrNoMemory;
	}
DMemSamplerChannel::DMemSamplerChannel()
	: iChunkInfo(KArrayGranularity, _FOFF(TChunkInfo, iAddress)), iTimer(Sample, this), iPeriodNumber(-1), iDoSampleDfc(DoSample, this, 1), iExitDfc(ExitDfcThread, this, 0), iNewSampleTraced(EFalse)
	{
	}
DMemSamplerChannel::~DMemSamplerChannel()
	{
	Kern::SafeClose((DObject*&)iClient, NULL);
	iExitDfc.Enque(); 
	}
void DMemSamplerChannel::ExitDfcThread(TAny*)
	{
	Kern::Exit(KErrNone);
	} 
TInt DMemSamplerChannel::DoCreate(TInt /*aUnit*/, const TDesC8* /*anInfo*/, const TVersion& aVer)
	{
	if (!Kern::QueryVersionSupported(TVersion(1,0,0),aVer))
		{
		return KErrNotSupported;
		}
	iClient=&Kern::CurrentThread();
	iClient->Open();
	TInt err = Kern::DfcQInit(&iPrivateDfcQ, KDfcThreadPriority, &KDfcThreadName);
	if (err == KErrNone)
		{
    	SetDfcQ(&iPrivateDfcQ);
		iDoSampleDfc.SetDfcQ(&iPrivateDfcQ);
		iExitDfc.SetDfcQ(&iPrivateDfcQ);
    	iMsgQ.Receive();		
		}
	
	return err;
	}
TInt DMemSamplerChannel::StartSampling(TInt aRate)
	{
	DEBUG_MEMSAMPLER(Kern::Printf("START");)
	ReadChunks();
	aRate = Min(KMaxRate, Max(KMinRate, aRate));
	iPeriod = NKern::TimerTicks(aRate);
	if (!Running())
		{
		iTimer.OneShot(iPeriod, EFalse);
		}
	DEBUG_MEMSAMPLER(Kern::Printf("START end");)
	return KErrNone;
	}
TInt DMemSamplerChannel::StopSampling()
	{
	DEBUG_MEMSAMPLER(Kern::Printf("STOP");)
	if (Running())
		{
		iTimer.Cancel();
		}
	DEBUG_MEMSAMPLER(Kern::Printf("STOP end");)
	return KErrNone;
	}
TInt DMemSamplerChannel::ResetSampling()
	{
	DEBUG_MEMSAMPLER(Kern::Printf("RESET");)
	iChunkInfo.Reset();
	DEBUG_MEMSAMPLER(Kern::Printf("RESET end");)
	return KErrNone;
	}
TInt DMemSamplerChannel::ReadChunks()
	{
	++iPeriodNumber;
	iNewSampleTraced = EFalse;
	for (TInt i = (iChunkInfo.Count() - 1); i >= 0; --i)
		{
		iChunkInfo[i].iSeen = EFalse;
		}
	DObjectCon* const * containers = Kern::Containers();
	DObjectCon& container = *containers[EChunk];
	container.Wait();
	for (TInt i = (container.Count() - 1); i >= 0; --i)
		{
		TChunkInfo thisChunk(*(DChunk*)container[i]);
		TInt pos;
		TInt err = iChunkInfo.FindInUnsignedKeyOrder(thisChunk, pos);
		if (err == KErrNotFound)
			{
			if (pos < iChunkInfo.Count())
				{
				iChunkInfo.Insert(thisChunk, pos);
				}
			else
				{
				TInt err = iChunkInfo.Append(thisChunk);
				if (err)
					{
					container.Signal();
					return err;
					}
				}
			TraceNewChunk(thisChunk);
			}
		else
			{
			if (thisChunk == iChunkInfo[pos])
				{
				// Chunk size hasn't changed - ignore.
				iChunkInfo[pos].iSeen = ETrue;
				}
			else
				{
				TChunkInfo& c = iChunkInfo[pos];
				if (thisChunk.iSize > c.iHighWaterMark)
					{
					c.iHighWaterMark = thisChunk.iSize;
					}
				c.iSize = thisChunk.iSize;
				c.iSeen = ETrue;
				TraceChangedChunk(c);
				}
			}
		}
	container.Signal();
	for (TInt i = (iChunkInfo.Count() - 1); i >= 0; --i)
		{
		const TChunkInfo& chunkInfo = iChunkInfo[i];
		if (!chunkInfo.iSeen)
			{
			TraceDeletedChunk(chunkInfo);
			iChunkInfo.Remove(i);
			}
		}
	return KErrNone;
	}
void DMemSamplerChannel::TraceNewChunk(const TChunkInfo& aChunkInfo) const
	{
	TraceNewSample();
	TFullName nameBuf;
	aChunkInfo.iAddress->FullName(nameBuf);				
	BTraceN(RMemSampler::EBtraceCategory, RMemSampler::ENewChunk, aChunkInfo.iAddress, aChunkInfo.iAddress->MaxSize(), nameBuf.Ptr(), nameBuf.Size());
	TraceChangedChunk(aChunkInfo);
	}
void DMemSamplerChannel::TraceChangedChunk(const TChunkInfo& aChunkInfo) const
	{
	TraceNewSample();
	BTraceContext12(RMemSampler::EBtraceCategory, RMemSampler::EChangedChunk, aChunkInfo.iAddress, aChunkInfo.iSize, aChunkInfo.iHighWaterMark);
	}
void DMemSamplerChannel::TraceDeletedChunk(const TChunkInfo& aChunkInfo) const
	{
	TraceNewSample();
	BTraceContext4(RMemSampler::EBtraceCategory, RMemSampler::EDeletedChunk, aChunkInfo.iAddress);
	}
void DMemSamplerChannel::TraceNewSample() const
	{
	if (!iNewSampleTraced)
		{
		iNewSampleTraced = ETrue;
		BTraceContext8(RMemSampler::EBtraceCategory, RMemSampler::ENewSample, iPeriod, iPeriodNumber);
		}
	}
void DMemSamplerChannel::HandleMsg(TMessageBase* aMsg)
	{
	TInt r=KErrNone;
	TThreadMessage& m=*(TThreadMessage*)aMsg;
	TInt id=m.iValue;
	if (id==(TInt)ECloseMsg)
		{
		DEBUG_MEMSAMPLER(Kern::Printf("CLOSE");)
		iTimer.Cancel();
		m.Complete(KErrNone,EFalse);
		iMsgQ.CompleteAll(KErrServerTerminated);
		DEBUG_MEMSAMPLER(Kern::Printf("CLOSE end");)
		return;
		}
	else
		{
		switch(id)
			{
			case RMemSampler::EControlStartProfile:
				r = StartSampling(m.Int0());
				break;
			case RMemSampler::EControlStopProfile:
				r = StopSampling();
				break;
			case RMemSampler::EControlResetProfile:
				r = ResetSampling();
				break;
			default:
				r = KErrNotSupported;
				break;
			}
		}
	m.Complete(r,ETrue);
	}
void DMemSamplerChannel::Sample(TAny* aPtr)
	{
	DMemSamplerChannel& d = *(DMemSamplerChannel*)aPtr;
	d.iTimer.Again(d.iPeriod);
	d.iDoSampleDfc.Add(); 
	}
void DMemSamplerChannel::DoSample(TAny* aPtr)
	{
	DMemSamplerChannel& d = *(DMemSamplerChannel*)aPtr;
	d.ReadChunks();
	}