commands/getstack/getstack.cpp
author Tom Sutcliffe <thomas.sutcliffe@accenture.com>
Tue, 07 Dec 2010 17:29:09 +0000
changeset 90 ceac7084e2e5
parent 0 7f656887cf89
permissions -rw-r--r--
Implemented RObjectIx-based memoryaccess APIs. Upshot is that objinfo now works again on platforms that define FSHELL_NO_DOBJECTIX_SUPPORT.

// getstack.cpp
// 
// Copyright (c) 2007 - 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 "getstack.h"
#include <fshell/memoryaccess.h>


CCommandBase* CCmdGetStack::NewLC()
	{
	CCmdGetStack* self = new(ELeave) CCmdGetStack();
	CleanupStack::PushL(self);
	self->BaseConstructL();
	return self;
	}

CCmdGetStack::~CCmdGetStack()
	{
	}

CCmdGetStack::CCmdGetStack()
	{
	}

const TDesC& CCmdGetStack::Name() const
	{
	_LIT(KName, "getstack");
	return KName;
	}

void CCmdGetStack::DoRunL()
	{
	RMemoryAccess memoryAccess;
	TInt err = RMemoryAccess::LoadDriver();
	if ((err == KErrNone) || (err == KErrAlreadyExists))
		{
		User::LeaveIfError(memoryAccess.Open());
		CleanupClosePushL(memoryAccess);
		}
	else
		{
		User::LeaveIfError(err);
		}
	RThread thread;
	User::LeaveIfError(thread.Open(iThreadId));
	CleanupClosePushL(thread);

	RFile file;
	User::LeaveIfError(file.Create(FsL(), iFileName, EFileWrite | EFileStream));
	CleanupClosePushL(file);

	TThreadStackInfo stackInfo;
	User::LeaveIfError(thread.StackInfo(stackInfo));
	const TInt stackSize = stackInfo.iBase - stackInfo.iLimit;

	const TInt KBufSize = 4096; // The largest amount RMemoryAccess allows us to copy in one go.
	HBufC8* stackBuf = HBufC8::NewLC(KBufSize);
	TPtr8 stackBufPtr(stackBuf->Des());

	TThreadMemoryAccessParamsBuf accessParamsBuf;
	TThreadMemoryAccessParams& accessParams = accessParamsBuf();
	accessParams.iId = (TInt)thread.Id().Id();

	TInt numBytesRead = 0;
	while (numBytesRead < stackSize)
		{
		accessParams.iAddr = (TUint8*)stackInfo.iLimit + numBytesRead;
		accessParams.iSize = Min(KBufSize, stackSize - numBytesRead);
		stackBufPtr.Zero();
		User::LeaveIfError(memoryAccess.GetThreadMem(accessParamsBuf, stackBufPtr));
		User::LeaveIfError(file.Write(*stackBuf));
		numBytesRead += stackBuf->Length();
		}

	CleanupStack::PopAndDestroy(4, &memoryAccess);
	}

void CCmdGetStack::ArgumentsL(RCommandArgumentList& aArguments)
	{
	_LIT(KArgFileName, "file_name");
	aArguments.AppendFileNameL(iFileName, KArgFileName);

	_LIT(KArgThreadId, "thread_id");
	aArguments.AppendUintL(iThreadId, KArgThreadId);
	}


#ifdef EXE_BUILD
EXE_BOILER_PLATE(CCmdGetStack)
#endif