Implemented RObjectIx-based memoryaccess APIs.
Upshot is that objinfo now works again on platforms that define FSHELL_NO_DOBJECTIX_SUPPORT.
// shutapp.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 <w32std.h>
#include <apgtask.h>
#include <apgwgnam.h>
#include "shutapp.h"
CCommandBase* CCmdShutApp::NewLC()
	{
	CCmdShutApp* self = new(ELeave) CCmdShutApp();
	CleanupStack::PushL(self);
	self->BaseConstructL();
	return self;
	}
CCmdShutApp::~CCmdShutApp()
	{
	iWgIds.Close();
	iWgNames.ResetAndDestroy();
	}
CCmdShutApp::CCmdShutApp()
	: iTimeout(5)
	{
	}
const TDesC& CCmdShutApp::Name() const
	{
	_LIT(KName, "shutapp");
	return KName;
	}
void CCmdShutApp::DoRunL()
	{
	RWsSession wsSession;
	User::LeaveIfError(wsSession.Connect());
	CleanupClosePushL(wsSession);
	const TInt numIds = iWgIds.Count();
	for (TInt i = 0; i < numIds; ++i)
		{
		ShutL(wsSession, iWgIds[i]);
		}
	const TInt numNames = iWgNames.Count();
	for (TInt i = 0; i < numNames; ++i)
		{
		const TDesC& name = *(iWgNames[i]);
		TInt id = 0;
		while ((id = wsSession.FindWindowGroupIdentifier(id, name)) != KErrNotFound)
			{
			ShutL(wsSession, id);
			}
		}
	if (iAll)
		{
		TInt numWgs = wsSession.NumWindowGroups(0);
		if (numWgs > 0)
			{
			CArrayFixFlat<TInt>* wgIds = new(ELeave) CArrayFixFlat<TInt>(numWgs);
			CleanupStack::PushL(wgIds);
			User::LeaveIfError(wsSession.WindowGroupList(0, wgIds));
			numWgs = wgIds->Count();
			for (TInt i = 0; i < numWgs; ++i)
				{
				ShutL(wsSession, (*wgIds)[i]);
				}
			CleanupStack::PopAndDestroy(wgIds);
			}
		}
	CleanupStack::PopAndDestroy(&wsSession);
	}
	
void CCmdShutApp::ShutL(RWsSession& aWsSession, TInt aWindowGroupId)
	{
	CApaWindowGroupName* wgn = CApaWindowGroupName::NewLC(aWsSession, aWindowGroupId);
	if (iVerbose)
		{
		_LIT(KUnknown, "not named");
		TPtrC caption(wgn->Caption());
		if (caption.Length() == 0)
			{
			caption.Set(KUnknown);
			}
		Printf(_L("Shutting down %d (%S)\r\n"), aWindowGroupId, &caption);
		}
	TThreadId threadId;
	User::LeaveIfError(aWsSession.GetWindowGroupClientThreadId(aWindowGroupId, threadId));
	RThread thread;
	User::LeaveIfError(thread.Open(threadId));
	CleanupClosePushL(thread);
	TRequestStatus threadStatus;
	thread.Logon(threadStatus);
	if (threadStatus != KRequestPending)
		{
		User::WaitForRequest(threadStatus);
		User::Leave(KErrGeneral);
		}
	RTimer timer;
	User::LeaveIfError(timer.CreateLocal());
	CleanupClosePushL(timer);
	TRequestStatus timeoutStatus;
	timer.After(timeoutStatus, iTimeout * 1000000);
	TApaTask task(aWsSession);
	task.SetWgId(aWindowGroupId);
#ifdef EKA2
	if (iForce)
		{
		task.SendSystemEvent(EApaSystemEventSecureShutdown);
		}
	else
		{
		task.SendSystemEvent(EApaSystemEventShutdown);
		}
#else
	task.SendSystemEvent(EApaSystemEventShutdown);
#endif
	User::WaitForRequest(threadStatus, timeoutStatus);
	if (timeoutStatus != KRequestPending)
		{
		// Timeout occurred.
		thread.LogonCancel(threadStatus);
		User::WaitForRequest(threadStatus);
		User::Leave(KErrTimedOut);
		}
	else
		{
		timer.Cancel();
		User::WaitForRequest(timeoutStatus);
		}
	CleanupStack::PopAndDestroy(3, wgn);
	}
void CCmdShutApp::OptionsL(RCommandOptionList& aOptions)
	{
	_LIT(KOptWgId, "id");
	aOptions.AppendUintL(iWgIds, KOptWgId);
	_LIT(KOptWgPattern, "match");
	aOptions.AppendStringL(iWgNames, KOptWgPattern);
	_LIT(KOptTimeout, "timeout");
	aOptions.AppendIntL(iTimeout, KOptTimeout);
	_LIT(KOptVerbose, "verbose");
	aOptions.AppendBoolL(iVerbose, KOptVerbose);
	_LIT(KOptAll, "all");
	aOptions.AppendBoolL(iAll, KOptAll);
#ifdef EKA2
	_LIT(KOptForce, "force");
	aOptions.AppendBoolL(iForce, KOptForce);
#endif
	}
void CCmdShutApp::ArgumentsL(RCommandArgumentList&)
	{
	}
#ifdef EXE_BUILD
EXE_BOILER_PLATE(CCmdShutApp)
#endif