core/extra/pipsrun/pipsrun.cpp
author Tom Sutcliffe <thomas.sutcliffe@accenture.com>
Sat, 31 Jul 2010 19:07:57 +0100
changeset 23 092bcc217d9d
parent 0 7f656887cf89
permissions -rw-r--r--
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.

// pipsrun.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/ltkutils.h>
#include <stdapis/unistd.h>
#include <stdapis/stdio.h>
#include <stdapis/pthread.h>

const TInt KBlockSize = 512;

using namespace IoUtils;

class CCmdPipsRun : public CCommandBase
	{
public:
	static CCommandBase* NewLC();
	~CCmdPipsRun();
private:
	CCmdPipsRun();
private: // From CCommandBase.
	virtual const TDesC& Name() const;
	virtual const TDesC& Description() const;
	virtual void DoRunL();
	};


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

CCmdPipsRun::~CCmdPipsRun()
	{
	}

CCmdPipsRun::CCmdPipsRun() : CCommandBase(ESharableIoSession)
	{
	}

const TDesC& CCmdPipsRun::Name() const
	{
	_LIT(KName, "pipsrun");	
	return KName;
	}

const TDesC& CCmdPipsRun::Description() const
	{
	_LIT(KDescription, "Internal helper executable for launching processes that use PIPS in such a way that they can interact with fshell's I/O primitives. You should not call it directly.");
	return KDescription;
	}

const TUint8* GetDesParamLC(TInt aSlot)
	{
	TInt paramLen = User::ParameterLength(aSlot);
	StaticLeaveIfErr(paramLen, _L("Couldn't read environment slot %d"), aSlot);
	HBufC* buf = HBufC::NewLC(paramLen + 1); // + 1 for the terminating null character.
	TPtr bufPtr(buf->Des());
	User::LeaveIfError(User::GetDesParameter(aSlot, bufPtr));
	TPtr8 narrowPtr = buf->Des().Collapse();
	narrowPtr.ZeroTerminate();
	return narrowPtr.Ptr();
	}

void AppendNarrowDes(TUint8*& aTarget, const TDesC& aDes)
	{
	TInt len = aDes.Length();
	const TUint16 *pS = aDes.Ptr();
	const TUint16 *pE = pS + len;
	while (pS < pE)
		{
		TUint c = (*pS++);
		if (c >= 0x100)
			{
			c = 1;
			}
		*aTarget++ = (TUint8)c;
		}
	}

class TGlueParams
	{
public:
	TGlueParams(TInt aFileDescriptor, RIoWriteHandle aWriteHandle);
public:
	TInt iFileDescriptor;
	RIoWriteHandle iWriteHandle;
	};

TGlueParams::TGlueParams(TInt aFileDescriptor, RIoWriteHandle aWriteHandle)
	: iFileDescriptor(aFileDescriptor), iWriteHandle(aWriteHandle)
	{
	}

void* GlueFileDescriptor(void* aParams)
	{
	TGlueParams* params = (TGlueParams*)aParams;
	TBuf8<KBlockSize * 2> buf;
	ssize_t s;
	TInt err = KErrNone;
	do
		{
		s = read((int)params->iFileDescriptor, (char*)buf.Ptr(), KBlockSize);
		if (s > 0)
			{
			buf.SetLength(s);
			err = params->iWriteHandle.Write(buf.Expand());
			}
		}
		while ((s > 0) && (err == KErrNone));
	return (void*)err;
	}

void CCmdPipsRun::DoRunL()
	{
	// Pack the envronment object we inherited from fshell in the way POSIX expects.
	_LIT(KEquals, "=");
	_LIT(KNull, "\x0");
	const CEnvironment& env = Env();
	RPointerArray<HBufC> keys;
	LtkUtils::CleanupResetAndDestroyPushL(keys);
	env.GetKeysL(keys);
	const TInt numVars = keys.Count();
	TUint8** envp = (TUint8**)User::AllocLC((numVars  + 1) * sizeof(TInt*)); // + 1 because the last pointer must be NULL.
	for (TInt i = 0; i < numVars; ++i)
		{
		const TDesC& key = *keys[i];
		const TDesC& val = env.GetAsDesL(key);
		const TInt length = (key.Length() + 2 + val.Length()); // 2 for the delimiting '=' and the terminating null.
		TUint8* p = (TUint8*)User::AllocLC(length);
		envp[i] = p;
		AppendNarrowDes(p, key);
		AppendNarrowDes(p, KEquals);
		AppendNarrowDes(p, val);
		AppendNarrowDes(p, KNull);
		}
	envp[numVars] = NULL;

	const TUint8* commandName(GetDesParamLC(IoUtils::KPipsCommandNameProcessSlot));
	const TUint8* commandLine(GetDesParamLC(IoUtils::KPipsCommandLineProcessSlot));
	TInt fds[3];
	TInt pid = popen3((const char*)commandName, (const char*)commandLine, (char**)envp, fds);
	if (pid == -1)
		{
		User::Leave(KErrGeneral);
		}
	CleanupStack::PopAndDestroy(2 + numVars + 1);
	CleanupStack::PopAndDestroy(&keys);

	RProcess childProcess;
	User::LeaveIfError(childProcess.Open(pid));
	CleanupClosePushL(childProcess);
	TRequestStatus childProcessStatus;
	childProcess.Logon(childProcessStatus);
	RProcess::Rendezvous(pid);

	pthread_t stdoutReaderThreadId = 0;
	pthread_t stderrReaderThreadId = 0;
	pthread_attr_t threadAttr;
	pthread_attr_init(&threadAttr);
	pthread_attr_setdetachstate(&threadAttr, PTHREAD_CREATE_DETACHED);
	TGlueParams stdoutParams(fds[1], Stdout());
	TGlueParams stderrParams(fds[2], Stderr());
	TInt err = pthread_create(&stdoutReaderThreadId, &threadAttr, GlueFileDescriptor, (void*)&stdoutParams);
	if (err == 0)
		{
		err = pthread_create(&stderrReaderThreadId, &threadAttr, GlueFileDescriptor, (void*)&stderrParams);
		}
	if (err != 0)
		{
		User::Leave(err);
		}

	Stdin().SetReadMode(RIoReadHandle::EOneOrMore);
	TBuf<KBlockSize> buf;
	TRequestStatus stdinReadStatus;
	TBool finished(EFalse);
	while (!finished)
		{
		Stdin().Read(buf, stdinReadStatus);
		User::WaitForRequest(childProcessStatus, stdinReadStatus);
		if (childProcessStatus == KRequestPending)
			{
			TPtr8 narrowPtr(buf.Collapse());
			narrowPtr.ZeroTerminate();			
			do
				{
				TInt ret = write(fds[0], narrowPtr.Ptr(), narrowPtr.Length());
				if (ret < 0)
					{
					finished = ETrue;
					}
				else
					{
					narrowPtr.Set(narrowPtr.MidTPtr(ret));
					}
				}
				while (!finished && narrowPtr.Length());
			}
		else
			{
			finished = ETrue;
			Stdin().ReadCancel();
			User::WaitForRequest(stdinReadStatus);
			}
		}

	CleanupStack::PopAndDestroy(&childProcess);
	pthread_join(stdoutReaderThreadId, NULL);
	pthread_join(stderrReaderThreadId, NULL);
	Complete(childProcessStatus.Int());
	}

EXE_BOILER_PLATE(CCmdPipsRun)