Added bsym file format spec.
// topfshell.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 <e32hashtab.h>
#include <fshell/ioutils.h>
#include <fshell/memoryaccesscmd.h>
#include "sampler.h"
using namespace IoUtils;
class CCmdTop : public CMemoryAccessCommandBase
{
public:
static CCommandBase* NewLC();
~CCmdTop();
private:
CCmdTop();
static TInt TimerCallback(TAny* aSelf);
void TimerCallback();
void UpdateL();
struct SThreadData;
static TInt SortReverse(const CCmdTop::SThreadData& aLeft, const CCmdTop::SThreadData& aRight);
private: // From CCommandBase.
virtual const TDesC& Name() const;
virtual void DoRunL();
virtual void ArgumentsL(RCommandArgumentList& aArguments);
virtual void OptionsL(RCommandOptionList& aOptions);
private:
RBuf8 iReadBuf;
struct SThreadData
{
TUint iId;
TInt iNumSamples;
TFullName iName; // Saves recalculating every time
};
RHashMap<TUint, SThreadData*> iSeenThreads; // Saves linear searching through iThreads for each sample (Maps thread id to SThreadData
RPointerArray<SThreadData> iThreads;
RSampler iSampler;
CPeriodic* iTimer;
TInt iRate;
CTextBuffer* iBuffer;
CTextFormatter* iFormatter;
TInt iNumConsoleLines;
TInt iNumLinesInLastUpdate;
};
CCommandBase* CCmdTop::NewLC()
{
CCmdTop* self = new(ELeave) CCmdTop();
CleanupStack::PushL(self);
self->BaseConstructL();
return self;
}
CCmdTop::~CCmdTop()
{
if (iTimer)
{
iTimer->Cancel();
}
delete iTimer;
if (iSampler.Handle())
{
iSampler.Stop();
iSampler.Close();
User::FreeLogicalDevice(KSamplerName);
}
delete iBuffer;
delete iFormatter;
iReadBuf.Close();
iThreads.ResetAndDestroy();
iSeenThreads.Close();
}
CCmdTop::CCmdTop()
: CMemoryAccessCommandBase(EManualComplete), iRate(1000)
{
}
const TDesC& CCmdTop::Name() const
{
_LIT(KName, "top");
return KName;
}
void CCmdTop::ArgumentsL(RCommandArgumentList& /*aArguments*/)
{
}
void CCmdTop::OptionsL(RCommandOptionList& aOptions)
{
aOptions.AppendIntL(iRate, _L("rate"));
}
EXE_BOILER_PLATE(CCmdTop)
void CCmdTop::DoRunL()
{
#ifdef FSHELL_MEMORY_ACCESS_SUPPORT
LoadMemoryAccessL();
#endif
TInt err = User::LoadLogicalDevice(KSamplerName);
if (err != KErrNone && err != KErrAlreadyExists)
{
LeaveIfErr(err, _L("Couldn't load sampler ldd %S"), &KSamplerName);
}
iReadBuf.CreateL(iRate * 2 * sizeof(TUint32)); // go twice as big as needed just to be safe, allows for AO not getting to run on time
iTimer = CPeriodic::NewL(CActive::EPriorityStandard);
iTimer->Start(iRate*1000, iRate*1000, TCallBack(&CCmdTop::TimerCallback, this));
LeaveIfErr(iSampler.Open(), _L("Couldn't open sampler"));
iSampler.Reset(ETrue); // Parameter is ignored
iSampler.Start(1000); // Always tell sampler to sample every millisecond, regardless of the frequency we update the UI
TSize size;
User::LeaveIfError(Stdout().GetScreenSize(size));
iNumConsoleLines = size.iHeight;
iBuffer = CTextBuffer::NewL(512);
iFormatter = CTextFormatter::NewL(size.iWidth);
Stdout().ClearScreen();
UpdateL(); // Just so we display something on screen before the first update
}
TInt CCmdTop::TimerCallback(TAny* aSelf)
{
static_cast<CCmdTop*>(aSelf)->TimerCallback();
return 0;
}
void CCmdTop::TimerCallback()
{
iReadBuf.Zero();
TRequestStatus stat;
iSampler.Read(iReadBuf, stat);
User::WaitForRequest(stat); // API used to be async, it's now sync but I haven't changed the interface
// First off clear all the iNumSamples (as that was for the previous sampling period)
for (TInt i = 0; i < iThreads.Count(); i++)
{
iThreads[i]->iNumSamples = 0;
}
TInt numSamples = iReadBuf.Length() / sizeof(TUint32);
TUint32* buf = (TUint32*)iReadBuf.Ptr();
for (TInt i = 0; i < numSamples; i++)
{
TUint32 threadId = buf[i];
SThreadData** thread = iSeenThreads.Find(threadId);
if (thread)
{
(*thread)->iNumSamples++;
}
else
{
// New thread id
SThreadData* thread = new SThreadData;
if (!thread) continue;
TInt err = iThreads.Append(thread);
if (err)
{
delete thread;
continue;
}
err = iSeenThreads.Insert(threadId, thread);
if (err)
{
iThreads.Remove(iThreads.Count()-1);
delete thread;
continue;
}
thread->iNumSamples = 1;
thread->iId = threadId;
RThread rthread;
#ifdef FSHELL_MEMORY_ACCESS_SUPPORT
err = iMemAccess.RThreadForceOpen(rthread, threadId);
#else
err = rthread.Open(TThreadId(threadId));
#endif
if (err == KErrNone)
{
thread->iName = rthread.FullName();
//PrettyName(EListThread, thread->iName);
rthread.Close();
}
else
{
thread->iName = _L("Tid: ");
thread->iName.AppendNum(threadId);
}
}
}
TRAP_IGNORE(UpdateL());
}
TInt CCmdTop::SortReverse(const CCmdTop::SThreadData& aLeft, const CCmdTop::SThreadData& aRight)
{
// This sorts largest number of samples first
TInt res = - (aLeft.iNumSamples - aRight.iNumSamples);
if (res == 0) return aLeft.iId - aRight.iId; // make sure there is a total ordering in event of tie break
return res;
}
void CCmdTop::UpdateL()
{
// First, sort iThreads
iThreads.Sort(TLinearOrder<SThreadData>(&SortReverse));
TInt numSamples = iReadBuf.Size() / sizeof(TUint32);
iBuffer->Zero();
iFormatter->Zero();
iBuffer->AppendL(_L("Tid\tThread name\tCPU usage\r\n"));
for (TInt i = 0; i < iThreads.Count() && i < iNumConsoleLines-2; i++) // minus one for title, one for last line of screen which I can't seem to figure out how to make use of
{
SThreadData& thread = *iThreads[i];
TReal percent = 100 * thread.iNumSamples / (TReal)numSamples;
iBuffer->AppendFormatL(_L("%d\t%S\t%00.2f%%\r\n"), thread.iId, &thread.iName, percent);
}
Stdout().SetCursorHeight(0);
iFormatter->TabulateL(0, 1, iBuffer->Descriptor(), ETruncateLongestColumn);
User::LeaveIfError(Stdout().SetCursorPosAbs(TPoint(0, 0)));
Stdout().Write(iFormatter->Descriptor());
TInt numLines = 1 + iThreads.Count(); // plus 1 for the title line
TInt numOldLines = iNumLinesInLastUpdate - numLines;
while (numOldLines > 0)
{
Stdout().ClearToEndOfLine();
Stdout().SetCursorPosRel(TPoint(0, 1));
--numOldLines;
}
iNumLinesInLastUpdate = numLines;
}