Tidied iocli exports, build macro tweaks.
Removed 4 overloads of CCommandBase::RunCommand[L] that are no longer used at all, and changed one more to not be exported as it's only used internally to iocli.dll.
fixed builds on platforms that don't support btrace or any form of tracing.
// kerninfo.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/ioutils.h>
#include <fshell/memoryaccesscmd.h>
#include <fshell/qr3dll.h>
#include <fshell/ltkutils.h>
#include <fshell/descriptorutils.h>
#ifdef FSHELL_SPCRE_SUPPORT
#include <fshell/pcre/cregex.h>
#endif
using namespace IoUtils;
class CRegEx;
class CCmdKerninfo : public CMemoryAccessCommandBase
{
public:
static CCommandBase* NewLC();
~CCmdKerninfo();
private:
CCmdKerninfo();
TInt InfoL(TInt aIndex, TBool aUseId, TBool aVerbose);
void PrintInfoForL(TInt aIndex, TBool aUseId=EFalse);
static TBool SupportsIds(TKernelObjectType aType);
static void ChompBlankLines(TDes& aDes);
private: // From CCommandBase.
virtual const TDesC& Name() const;
virtual void DoRunL();
virtual void ArgumentsL(RCommandArgumentList& aArguments);
virtual void OptionsL(RCommandOptionList& aOptions);
private:
TKernelObjectType iType;
TInt iObjectIndex;
TBool iVerbose;
TBool iVerboseMatch;
HBufC* iMatch;
HBufC* iRegexMatch;
CKernelObjectList* iModel;
RBuf iTitle, iInfo;
LtkUtils::RLtkBuf8 iNarrowBuf;
CRegEx* iRegex;
};
CCommandBase* CCmdKerninfo::NewLC()
{
CCmdKerninfo* self = new(ELeave) CCmdKerninfo();
CleanupStack::PushL(self);
self->BaseConstructL();
return self;
}
CCmdKerninfo::~CCmdKerninfo()
{
delete iModel;
iTitle.Close();
iInfo.Close();
iNarrowBuf.Close();
delete iMatch;
delete iRegexMatch;
#ifdef FSHELL_SPCRE_SUPPORT
delete iRegex;
#endif
}
CCmdKerninfo::CCmdKerninfo()
{
}
const TDesC& CCmdKerninfo::Name() const
{
_LIT(KName, "kerninfo");
return KName;
}
void CCmdKerninfo::ArgumentsL(RCommandArgumentList& aArguments)
{
aArguments.AppendEnumL((TInt&)iType, _L("object-type"));
aArguments.AppendIntL(iObjectIndex, _L("object-index"));
}
void CCmdKerninfo::OptionsL(RCommandOptionList& aOptions)
{
aOptions.AppendStringL(iMatch, _L("match"));
#ifdef FSHELL_SPCRE_SUPPORT
aOptions.AppendStringL(iRegexMatch, _L("regex-match"));
#endif
aOptions.AppendBoolL(iVerbose, _L("verbose"));
aOptions.AppendBoolL(iVerboseMatch, _L("verbosematch"));
}
EXE_BOILER_PLATE(CCmdKerninfo)
void CCmdKerninfo::DoRunL()
{
LoadMemoryAccessL();
iModel = CKernelObjectList::NewL(&iMemAccess);
iModel->SetCurrentType(iType);
TRAPL(iModel->RefreshDataL(), _L("Couldn't get model data"));
iTitle.CreateL(256);
iInfo.CreateL(256);
#ifdef FSHELL_SPCRE_SUPPORT
if (iRegexMatch)
{
if (iMatch) LeaveIfErr(KErrArgument, _L("Can't specify both --match and --regex-match, use one or the other"));
iRegex = CRegEx::NewL(*iRegexMatch, TRegExOptions(EPcreExtended|EPcreNewlineAny));
}
#endif
if (iVerboseMatch) iVerbose = ETrue; // verbosematch implies verbose
const TInt n = iModel->Count();
TBool listAll = !iArguments.IsPresent(&iObjectIndex);
if (listAll)
{
for (TInt i = 0; i < n; i++)
{
PrintInfoForL(i);
}
}
else
{
TBool usingId = SupportsIds(iType);
if (!usingId && (iObjectIndex < 0 || iObjectIndex >= n))
{
LeaveIfErr(KErrArgument, _L("Bad argument %d. Number of objects in this category: %d"), iObjectIndex, n);
}
PrintInfoForL(iObjectIndex, usingId);
}
}
TInt CCmdKerninfo::InfoL(TInt aIndex, TBool aUseId, TBool aVerbose)
{
TInt err = KErrNone;
TInt itemid = 0;
if (aUseId)
{
TRAP(err, iModel->GetInfoByIdL(aIndex, aVerbose, iTitle, iInfo));
}
else
{
TRAP(err, itemid = iModel->GetInfoByIndexL(aIndex, aVerbose, iTitle, iInfo));
}
LeaveIfErr(err, _L("Couldn't get info for item %d"), aIndex);
return itemid;
}
void CCmdKerninfo::PrintInfoForL(TInt aIndex, TBool aUseId /* =EFalse */)
{
TInt itemid = 0;
if (iMatch || iRegexMatch)
{
// Filter out anything not matching iMatch/iRegexMatch
// Unless --verbosematch is specified, match on the non-verbose (but display the verbose if specified). This is because calculating the verbose can be really slow so don't do it until we know we match.
itemid = InfoL(aIndex, aUseId, (TBool)iVerboseMatch);
const TDesC& toMatchAgainst = iVerboseMatch ? iInfo : iTitle;
TBool match = EFalse;
if (iMatch)
{
match = (toMatchAgainst.MatchF(*iMatch) != KErrNotFound);
}
#ifdef FSHELL_SPCRE_SUPPORT
else if (iRegexMatch)
{
iNarrowBuf.Zero();
iNarrowBuf.AppendL(toMatchAgainst);
match = iRegex->PartialMatchL(iNarrowBuf);
}
#endif
if (!match) return;
}
if ((!iMatch && !iRegexMatch) || (iVerbose && !iVerboseMatch))
{
// Need to get info for first time (if !iMatch) or re-get the verbose info (in the case of we just did a match on the non-verbose info)
itemid = InfoL(aIndex, aUseId, iVerbose);
}
TInt idxToDisplay = (SupportsIds(iType) && !aUseId) ? itemid : aIndex;
if (iVerbose)
{
// In verbose mode the title is always just "Process info" (etc) so no point displaying it
Printf(_L("%d:\n"), idxToDisplay);
ChompBlankLines(iInfo); // When doing a verbose listing (specially multiple ones) it's easier to remove the blank lines that the model adds for readability in the QResources dialog
Write(iInfo);
Write(_L("\n\n"));
}
else
{
Printf(_L("%d: %S\t%S\n"), idxToDisplay, &iTitle, &iInfo);
}
}
TBool CCmdKerninfo::SupportsIds(TKernelObjectType aType)
{
// These types can (and should) be looked up directly by id and not by index into the list.
return (aType == EListThread || aType == EListProcess || aType == EListHal);
}
void CCmdKerninfo::ChompBlankLines(TDes& aDes)
{
_LIT(KLfLf, "\n\n");
TInt found;
while ((found = aDes.Find(KLfLf)) != KErrNotFound)
{
aDes.Delete(found, 1); // Nuke one of the newlines
}
}