launcher/engine/src/launcherengine.cpp
author hgs
Fri, 11 Jun 2010 17:49:08 +0300
changeset 29 1c71b77fbc93
parent 28 4cc0d1a608c1
permissions -rw-r--r--
201023

/*
* Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of "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:
* Nokia Corporation - initial contribution.
*
* Contributors:
*
* Description:
*
*/

#include <coemain.h>
#include <eikenv.h>
#include <pathinfo.h> 
#include <apgtask.h>
#include <bautils.h>
 
#include "launchertraces.h"
#include "e32image.h"
#include "launcherxmlparser.h"
#include "launcherdllparser.h"
#include "launcherengine.h"

_LIT(KLogFileName, "LauncherLog.txt");
_LIT(KBCLogFileName, "LauncherBCLog.txt");
_LIT(KSystemDllsFileName, "SystemDlls.txt");
_LIT(KRequiredDllsFileName, "RequiredDlls.xml");
_LIT(KDotXML,".xml");
_LIT(KDotLauncherXML,".launcherxml");

_LIT(KFileSeparator, "\t");
_LIT(KFileNewLine, "\r\n");

_LIT(KNewLine, "\n");

const TInt KMaxAppsArraySize=250;
const TInt KMaxDllArraySize=5000;
const TInt KLauncherLogBufferSize = 4096;

// After this many issues, issues are buffered and printed 
// in the end of analysis
const TInt KBigBufferUsageThreshold=10;

// Buffer allocation unit
const TInt KBigBufferAllocBytes=1024;

// ---------------------------------------------------------------------------

CLauncherEngine* CLauncherEngine::NewL(MLauncherUI *aLauncherUI)
    {
    CLauncherEngine* self = new(ELeave) CLauncherEngine;
    CleanupStack::PushL(self);
    self->ConstructL(aLauncherUI);
    CleanupStack::Pop();
    return self;
    }

// ---------------------------------------------------------------------------

CLauncherEngine::CLauncherEngine() : CActive(EActivePriorityIpcEventsHigh)
    {
    }

// ---------------------------------------------------------------------------

template <typename T>
void AppendLogBufferL(const TDesC& aText, T*& aBuf, TInt aIncreaseStep = KLauncherLogBufferSize)
    {
    if( aBuf == 0 )
        {
        aBuf = T::NewL(aIncreaseStep);
        }
    
    TInt currentMaxLength = aBuf->Des().MaxLength();
    if( aBuf->Des().Length() + aText.Length() > currentMaxLength )
        {
        TInt increaseSize = Max(aText.Length(), aIncreaseStep);                
        aBuf = aBuf->ReAllocL(aBuf->Des().MaxLength() + increaseSize );
        }
    aBuf->Des().Append(aText);        
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::ConstructL(MLauncherUI *aLauncherUI)
    {
    LOGSTRING("Launcher: CLauncherEngine::ConstructL");
    
    User::LeaveIfNull( aLauncherUI );
    
    iLauncherUI = aLauncherUI;
    

    iEnv = CEikonEnv::Static();
    iLaunchingIsActive = EFalse;
    iDLLAnalysisIsActive = EFalse;
    iSkipHiddenAndEmbedOnly = ETrue;

    User::LeaveIfError(iTimer.CreateLocal());

    User::LeaveIfError(iLs.Connect());
    User::LeaveIfError(iWs.Connect());

    iAppThreadChecker = CAppThreadChecker::NewL(this);
    iAppRunningChecker = CAppRunningChecker::NewL(this);

    iAllAppsArray = new(ELeave) CDesCArrayFlat(KMaxAppsArraySize);
    iAppsArray = new(ELeave) CDesCArrayFlat(KMaxAppsArraySize);
    iSystemDllArray = new(ELeave) CDesCArrayFlat(KMaxDllArraySize);

    iLogWriteBuf = HBufC8::NewL(KLauncherLogBufferSize);

    iLogFilePath = PathInfo::PhoneMemoryRootPath();
    iLogFilePath.Append( KLogFileName );
    
    iBCLogFilePath = PathInfo::PhoneMemoryRootPath();
    iBCLogFilePath.Append( KBCLogFileName );

    iSystemDllsFilePath = PathInfo::PhoneMemoryRootPath();
    iSystemDllsFilePath.Append( KSystemDllsFileName );

    iRequiredDllsFilePath = PathInfo::PhoneMemoryRootPath();
    iRequiredDllsFilePath.Append( KRequiredDllsFileName );
            
    iDLLParser = CLauncherDLLParser::NewL();

    CActiveScheduler::Add(this);
    }

// ---------------------------------------------------------------------------

CLauncherEngine::~CLauncherEngine()
    {
    LOGSTRING("Launcher: CLauncherEngine::~CLauncherEngine");

    Cancel();

    // close the log
    iLogFile.Close();
    iBCLogFile.Close();
    if (iLogWriteBuf)
        delete iLogWriteBuf;

    if (iAppRunningChecker)
        delete iAppRunningChecker;
    
    if (iAppThreadChecker)
        delete iAppThreadChecker;

    if (iSystemDllArray)
        {
        iSystemDllArray->Reset();
        delete iSystemDllArray;
        }

    if (iAppsArray)
        {
        iAppsArray->Reset();
        delete iAppsArray;
        }

    
    if (iAllAppsArray)
        {
        iAllAppsArray->Reset();
        delete iAllAppsArray;
        }
    
    delete iXMLParser;
   // delete iDLLParser;
    //delete iDLLElement;
    delete iBCIssuesBigBuffer;            
    
    iWs.Close();
    iLs.Close();
    iTimer.Close();
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::DoCancel()
    {
    LOGSTRING("Launcher: CLauncherEngine::DoCancel");
    
    if( iXMLParser )
        {
        iXMLParser->Cancel();
        }
    iTimer.Cancel();
    
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::RunL()
    {
    
    LOGSTRING("Launcher: CLauncherEngine::RunL");  
    TInt err = iStatus.Int();
    User::LeaveIfError(err);
    
    // write full app path to the log file
    WriteInitialStuffToTheLogL(iAppsArray->MdcaPoint(iAppLaunchCounter), iLogFile);

    // get the uid of the current app
    iCurrentAppUid = KNullUid;
    TApaAppInfo appInfo;
    iLs.GetAllApps();
    while (iLs.GetNextApp(appInfo) == KErrNone)
        {
        if (appInfo.iFullName.CompareF( iAppsArray->MdcaPoint(iAppLaunchCounter) ) == 0)
            {
            iCurrentAppUid = appInfo.iUid;
            break;
            }
        }

    if (iCurrentAppUid == KNullUid)
        {        
        iLogWriteBuf->Des().Append(_L("[WARN: App has no UID] "));
        }

    // parse the filename
    TParse nameParser;
    nameParser.SetNoWild(iAppsArray->MdcaPoint(iAppLaunchCounter), NULL, NULL);
    iCurrentAppNameAndExt.Copy(nameParser.Drive());
    iCurrentAppNameAndExt.Append(nameParser.NameAndExt());

    // do not try to launch these apps
    if (iAppsArray->MdcaPoint(iAppLaunchCounter).FindF(_L("\\Launcher.")) != KErrNotFound
        || iAppsArray->MdcaPoint(iAppLaunchCounter).FindF(_L("\\Phone.")) != KErrNotFound
        || iAppsArray->MdcaPoint(iAppLaunchCounter).FindF(_L("\\Startup.")) != KErrNotFound  
        || iAppsArray->MdcaPoint(iAppLaunchCounter).FindF(_L("\\SplashScreen.")) != KErrNotFound  
        || iAppsArray->MdcaPoint(iAppLaunchCounter).FindF(_L("\\eshell.")) != KErrNotFound)  
        {

        iLauncherUI->PrintText(iCurrentAppNameAndExt);
        iLauncherUI->PrintText(_L(": SKIPPED automatically\n"));
        
        iLogWriteBuf->Des().Append(_L("[SKIPPED automatically] "));
        iLogWriteBuf->Des().Append(KFileNewLine);
        iLogFile.Write(iLogWriteBuf->Des());

        iSkippedCases++;
        CheckForMoreAppsL();
        }

    else
        {

        // dependency check not needed if the app is in the ROM/ROFS, because they'll be checked automatically
        // when the rom image is built
        
        if (iCurrentAppNameAndExt[0] != 'Z' && iCurrentAppNameAndExt[0] != 'z')
            {

            if (iCurrentAppNameAndExt[2] == '[')  // this is quite likely a Java application, no dependency test needed 
                {
                iLogWriteBuf->Des().Append(_L("[Dependency check not done] "));
                }
            else
                {  // otherwise check depencies
            
                #ifdef __WINS__
                
                    // emulator not supported

                #else

                    // ELF binaries

                    CDesCArray* missingDllArray = NULL;

                    TRAPD(err, missingDllArray = DependencyCheckForE32ImageL());

                    // some error happened while processing the E32 image
                    if (err != KErrNone)
                        {

                        iLauncherUI->PrintText(iCurrentAppNameAndExt);
                        iLauncherUI->PrintText(_L(": unable to read import table!\n"));
                        
                        iLogWriteBuf->Des().Append(_L("[Unable to read import table!] "));
                        }

                    // print missing dependencies
                    else if (err==KErrNone && missingDllArray)
                        {
                        if (missingDllArray->MdcaCount() > 0)
                            {
                            iLauncherUI->PrintText(iCurrentAppNameAndExt);
                            iLauncherUI->PrintText(_L(": missing dependencies: "));
                            
                            iLogWriteBuf->Des().Append(_L("[Missing dependencies: "));

                            for (TInt k=0; k<missingDllArray->MdcaCount(); k++)
                                {
                                iLauncherUI->PrintText(missingDllArray->MdcaPoint(k));
                                iLauncherUI->PrintText(_L(" "));
                                
                                iLogWriteBuf->Des().Append(missingDllArray->MdcaPoint(k));
                                iLogWriteBuf->Des().Append(_L(" "));
                                }

                            iLauncherUI->PrintText(_L("\n"));

                            iLogWriteBuf->Des().Append(_L("] "));
                            }
                        }

                    if (missingDllArray)
                        delete missingDllArray;

                #endif

                }  // if '['
            }  // if 'Z'



        // check if the app is already running
        TApaTaskList taskList(iWs);
        TApaTask thisTask = taskList.FindApp(iCurrentAppUid);
        if (thisTask.Exists())
            {
            iLogWriteBuf->Des().Append(_L(" [OK: App already running]"));
            iLogWriteBuf->Des().Append(KFileNewLine);
            iLogFile.Write(iLogWriteBuf->Des());

            iOkCases++;
            CheckForMoreAppsL();
            }
        
        else
            {
            // check the program's capabilities
            TApaAppCapabilityBuf buf;
            iLs.GetAppCapability(buf, iCurrentAppUid);
            TApaAppCapability cap = buf();

            // if it's embeddable only, don't launch if setting is enabled
            if (cap.iEmbeddability == TApaAppCapability::EEmbeddableOnly && SkipHiddenAndEmbedOnly())  
                {
              
                iLauncherUI->PrintText(iCurrentAppNameAndExt);
                iLauncherUI->PrintText(_L(": SKIPPED: embeddable only\n"));

                iLogWriteBuf->Des().Append(_L("[SKIPPED: embeddable only] "));
                iLogWriteBuf->Des().Append(KFileNewLine);
                iLogFile.Write(iLogWriteBuf->Des());

                iSkippedCases++;
                CheckForMoreAppsL();
                }
        
            // if it's hidden, don't launch if setting is enabled
            else if (cap.iAppIsHidden && SkipHiddenAndEmbedOnly())  
                {
                
                iLauncherUI->PrintText(iCurrentAppNameAndExt);
                iLauncherUI->PrintText(_L(": SKIPPED: hidden\n"));
                
                iLogWriteBuf->Des().Append(_L("[SKIPPED: hidden] "));
                iLogWriteBuf->Des().Append(KFileNewLine);
                iLogFile.Write(iLogWriteBuf->Des());

                iSkippedCases++;
                CheckForMoreAppsL();
                }

            // otherwise do the normal launch test
            else
                {
                LaunchApplicationL();
                }
            
            } //if (thisTask.Exists())
        
        } //if (iAppsArray->MdcaPoint(iAppLaunchCounter).FindF
        
    }

// ---------------------------------------------------------------------------

CDesCArray* CLauncherEngine::ListOfAllAppsL()
    {
    LOGSTRING("Launcher: CLauncherEngine::ListOfAllAppsL");

    // find all DLLs from the system
    iSystemDllArray->Reset();
    TRAP_IGNORE( FindFiles(_L("*.dll"), _L("\\sys\\bin\\")) );

    // write the list of DLLs to a file
    RFile dllFile;
    if (dllFile.Replace(iEnv->FsSession(), iSystemDllsFilePath, EFileWrite) == KErrNone)
        {
        TBuf8<KMaxFileName> dllName;

        for (TInt i=0; i<iSystemDllArray->MdcaCount(); i++)
            {
            dllName.Copy( iSystemDllArray->MdcaPoint(i) );
            dllName.Append( KFileNewLine );

            dllFile.Write( dllName );
            }

        dllFile.Close();
        }

    // reset the apps list
    iAllAppsArray->Reset();

    // search all apps
    TApaAppInfo appInfo;
    iLs.GetAllApps();

    while (iLs.GetNextApp(appInfo) == KErrNone)
        {
        iAllAppsArray->AppendL(appInfo.iFullName);
        }


    // remove launcher.app / launcher.exe from the list
    for (TInt i=0; i<iAllAppsArray->MdcaCount(); i++)
        {
        if (iAllAppsArray->MdcaPoint(i).FindF(_L("\\Launcher.")) != KErrNotFound)
            {
            iAllAppsArray->Delete(i);
            iAllAppsArray->Compress();
            break;
            }
        }

    // sort the elements
    iAllAppsArray->Sort();

    return iAllAppsArray;
    }

// ---------------------------------------------------------------------------

TInt CLauncherEngine::FindFiles(const TDesC& aFileName, const TDesC& aPath)
    {
    TFindFile fileFinder(iEnv->FsSession());
    CDir* fileList; 
    TInt err = fileFinder.FindWildByDir(aFileName, aPath, fileList);

    while (err == KErrNone)
        {
        for (TInt i=0; i<fileList->Count(); i++)
            {
            TParse fullentry;
            fullentry.Set((*fileList)[i].iName, &fileFinder.File(), NULL);

            TRAP(err, iSystemDllArray->AppendL(fullentry.NameAndExt())); 
            }

        delete fileList;
        err = fileFinder.FindWild(fileList);
        }
    return err;
    }

// ---------------------------------------------------------------------------


void CLauncherEngine::StartAppLaunchingL(const CArrayFix<TInt>* aSelectedApps, TBool aAutoClose)
    {
    LOGSTRING("Launcher: CLauncherEngine::StartAppLaunchingL");

    // check that we have something to launch
    if (aSelectedApps->Count() <= 0)
        {
        _LIT(message, "Nothing selected");
        iLauncherUI->ShowErrorMessage(message);
        }
    else
        {
        // update the list of applications to be tested
        iAppsArray->Reset();

        TInt ref(0);
        TKeyArrayFix key(0, ECmpTUint16);
        TInt index(0);

        for (TInt i=0; i<iAllAppsArray->MdcaCount(); i++)
            {
            ref = i;

            // if the application is selected, append it to the apps array
            if (aSelectedApps->Find(ref, key, index) == 0)  
                {
                iAppsArray->AppendL(iAllAppsArray->MdcaPoint(i));
                }
            }


        // to make sure that our algorithm works correctly
        if (iAppsArray->MdcaCount() != aSelectedApps->Count())
            {
            _LIT(message, "Something went wrong...");
            iLauncherUI->ShowErrorMessage(message);
            }

        
        // init
        Cancel();
        iLaunchingIsActive = ETrue;
        iAutoClose = aAutoClose;
        iAppLaunchCounter = 0;
        iFailedCases = 0;
        iOkCases = 0;
        iSkippedCases = 0;
        iTotalNumberOfCases = iAppsArray->MdcaCount();
        iCurrentAppUid = KNullUid;

        // open the log file for writing
        if (iLogFile.Open(iEnv->FsSession(), iLogFilePath, EFileWrite) != KErrNone)
            {
            iEnv->FsSession().MkDirAll(iLogFilePath);
            iLogFile.Replace(iEnv->FsSession(), iLogFilePath, EFileWrite);
            }
        else
            {
            // file opens correctly, seek to the end
            TInt fileSize=0;
            iLogFile.Size(fileSize);
            iLogFile.Seek(ESeekCurrent, fileSize);
            }

        // change focus to output view!!
        iLauncherUI->ChangeFocusToOutputView();

        //text to log
        iLauncherUI->PrintText(_L("New test started.\n"));

        // start the first launch!
        IssueLaunch();

        }
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::IssueLaunch()
    {
    LOGSTRING("Launcher: CLauncherEngine::IssueLaunch");

    __ASSERT_ALWAYS(!IsActive(), User::Panic(_L("Timing error?"), 100));

    // this should keep the backlight on
    User::ResetInactivityTime();

    iDLLAnalysisIsActive = EFalse;
    // some delay
    iTimer.After(iStatus, 1000000);    
    SetActive();
    }

// ---------------------------------------------------------------------------

// ---------------------------------------------------------------------------

void CLauncherEngine::WriteInitialStuffToTheLogL(const TDesC& aOwnData, RFile& aFile)
    {
    LOGSTRING("Launcher: CLauncherEngine::WriteInitialStuffToTheLog");

    TTime time;
    time.HomeTime();
    TBuf<32> currentTime;
    TBuf<32> currentDate;

    // current date            
    _LIT(KCurrentDate,"%D%M%Y%/0%1%/1%2%/2%3%/3");
    time.FormatL(currentDate, KCurrentDate);    
    iLogWriteBuf->Des().Copy(currentDate);    
    AppendLogBufferL(KFileSeparator(), iLogWriteBuf);

    // current time
    _LIT(KCurrentTime,"%-B%:0%J%:1%T%:2%S%:3%+B");
    time.FormatL(currentTime, KCurrentTime);    
    AppendLogBufferL(currentTime, iLogWriteBuf);
    AppendLogBufferL(KFileSeparator(), iLogWriteBuf);

    // available RAM memory
    TMemoryInfoV1Buf memory;
    UserHal::MemoryInfo(memory);
    iLogWriteBuf->Des().AppendNum(memory().iFreeRamInBytes);  
    AppendLogBufferL(KFileSeparator(), iLogWriteBuf);

    // own data, eg. application name
    AppendLogBufferL(aOwnData, iLogWriteBuf);
    AppendLogBufferL(KFileSeparator(), iLogWriteBuf);
    AppendLogBufferL(KFileNewLine(), iLogWriteBuf);    

    // write the buffer to the file
    aFile.Write(iLogWriteBuf->Des()); 

    // clear the buffer
    iLogWriteBuf->Des().Copy(_L(""));
    }

// ---------------------------------------------------------------------------


void CLauncherEngine::CheckForMoreAppsL()
    {
    LOGSTRING("Launcher: CLauncherEngine::CheckForMoreAppsL");

    // make sure the launcher app is in the foreground
    TApaTaskList taskList(iWs);
    TUid launcherAppUid;
    launcherAppUid.iUid = 0x101FB74F;
    TApaTask launcherTask = taskList.FindApp(launcherAppUid);
    launcherTask.BringToForeground();


    // check if we have more test to be executed
    if ( iAppLaunchCounter >= iTotalNumberOfCases-1 )
        {
        
        // all done, show stats
        TBuf<200> message;
        message.Append( _L("Done: ") );
        message.AppendNum( iOkCases );
        message.Append( _L(" ok, ") );
        message.AppendNum( iFailedCases );
        message.Append( _L(" failed, ") );
        message.AppendNum( iSkippedCases );
        message.Append( _L(" skipped.") );

        // print the message to the output screen
        iLauncherUI->PrintText(message);
        iLauncherUI->PrintText(_L("\n\n"));
        
        //write same stuff to the log
        WriteInitialStuffToTheLogL(message, iLogFile);
        iLogWriteBuf->Des().Copy(KFileNewLine);
        iLogWriteBuf->Des().Append(KFileNewLine);        
        iLogFile.Write(iLogWriteBuf->Des());

        // close the log
        iLogFile.Close();

        iLaunchingIsActive = EFalse;

        // all apps launched
        _LIT(KAllDoneMessage, "All apps launched");
        iLauncherUI->ShowInfoMessage(KAllDoneMessage);
        }
    else
        {
        // more apps to be launched, maintain requests
        iAppLaunchCounter++;
        IssueLaunch();
        }
  
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::LaunchApplicationL()
    {
    LOGSTRING("Launcher: CLauncherEngine::LaunchApplication");
    LOGSTRING3("Launcher: Trying to launch %S, UID: %d", &iCurrentAppNameAndExt, iCurrentAppUid.iUid);

    TRAPD(err, DoLaunchApplicationL());
    
    if (err!=KErrNone)
        {
        iLogWriteBuf->Des().Append(_L("[FAIL: Cannot launch the app] "));        
       
        iLauncherUI->PrintText(iCurrentAppNameAndExt);
        iLauncherUI->PrintText(_L(": cannot launch\n"));
        
        // write the buffer to the log
        iLogWriteBuf->Des().Append(KFileNewLine);        
        iLogFile.Write(iLogWriteBuf->Des());

        // this application isn't even launchable, go to next one
        iFailedCases++;
        CheckForMoreAppsL();        
        }
    
    }
// ---------------------------------------------------------------------------


void CLauncherEngine::DoLaunchApplicationL()
    {
    LOGSTRING("Launcher: CLauncherEngine::DoLaunchApplicationL");

    // create a new handle
    RThread newThreadHandle;
    iCurrentAppThread = newThreadHandle;


    // Find the task with uid3
    TApaTaskList tasklist(iWs);
    TApaTask task=tasklist.FindApp(iCurrentAppUid);

    if (task.Exists())
        // Task exists, bring it to foreground
        {
        task.BringToForeground();
        }
    else
        // Task doesn't exist, launch a new instance of an application
        {
        TApaAppInfo appInfo;
        User::LeaveIfError(iLs.GetAppInfo(appInfo, iCurrentAppUid));
        TApaAppCapabilityBuf capBuf;
        User::LeaveIfError(iLs.GetAppCapability(capBuf, iCurrentAppUid));
        TApaAppCapability& caps = capBuf();

        CApaCommandLine* cmdLine=CApaCommandLine::NewLC();
        cmdLine->SetExecutableNameL(appInfo.iFullName);

        if ( caps.iLaunchInBackground )
            // Apps capability defines that the app is launched in background
            {
            cmdLine->SetCommandL(EApaCommandBackground);
            }
        else
            {
            cmdLine->SetCommandL(EApaCommandRun);
            }

        // start the app
        User::LeaveIfError(iLs.StartApp(*cmdLine, iCurrentAppThreadId));

        // activate thread checker active object
        iAppThreadChecker->ActivateChecking();

        // now open a handle to the thread and register death notifier
        TInt err = iCurrentAppThread.Open(iCurrentAppThreadId);
        if (err == KErrNone)
            iCurrentAppThread.Logon(iAppThreadChecker->iStatus);
        else
            {
            iCurrentAppThread.Close();
            TRequestStatus* status = &iAppThreadChecker->iStatus;
            User::RequestComplete(status, KErrNone);
            iAppThreadChecker->Cancel();
            User::Leave(err);                
            }

        CleanupStack::PopAndDestroy(); // cmdLine
        }

    // the application is now running, start a check to see if it's still alive
    iAppRunningChecker->StartTesting(); 

    }

// ---------------------------------------------------------------------------

void CLauncherEngine::CheckIfAppIsRunningL()
    {
    LOGSTRING("Launcher: CLauncherEngine::CheckIfAppIsRunningL");

    // cancel the death notifier since it isn't needed anymore
    if( iCurrentAppThread.Handle() != 0 )
        {
        iCurrentAppThread.LogonCancel(iAppThreadChecker->iStatus);
        }

    // cancel checkers
    iAppThreadChecker->Cancel();
    iAppRunningChecker->Cancel(); 



    // check from the window server if the app is running
    TApaTaskList taskList(iWs);
    TApaTask thisTask = taskList.FindApp(iCurrentAppUid);

    if( !thisTask.Exists() ) // application not running -> FAIL
        {
        // check from the thread why it quit
        CheckWhyThreadDiedL();

        // --> CheckForMoreApps() and all the other stuff called from CheckWhyThreadDied() !

        }
    else
        {
        // app is running!
        iOkCases++;

        // close handle to the thread        
        iCurrentAppThread.Close();


        iLogWriteBuf->Des().Append(_L(" [OK]"));
        iLogWriteBuf->Des().Append(KFileNewLine);        
        iLogFile.Write(iLogWriteBuf->Des());


        // close the running application if needed
        if (iAutoClose)
            {
            // since the application is still open, let's close it
            thisTask.EndTask();
            //User::After(1000);
            //thisTask.SendSystemEvent(EApaSystemEventShutdown);
            //thisTask.KillTask();
            //User::After(1000);
            }

        // this app is done now, move to the next one!
        CheckForMoreAppsL();
                
        }    
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::CheckWhyThreadDiedL()
    {
    LOGSTRING("Launcher: CLauncherEngine::CheckWhyThreadDiedL");

    // cancel the death notifier since it isn't needed anymore
    if( iCurrentAppThread.Handle() != 0 )
        {
        iCurrentAppThread.LogonCancel(iAppThreadChecker->iStatus);
        }

    // make sure all checkers are cancelled
    iAppRunningChecker->Cancel(); 
    iAppThreadChecker->Cancel(); 

    TBuf<256> outputText;
    outputText.Append(_L("App.Closed. "));


    if (iCurrentAppThread.ExitType() == EExitKill)
        {
        outputText.Append(_L("\"EExitKill\""));
        }
    else if (iCurrentAppThread.ExitType() == EExitTerminate)
        {
        outputText.Append(_L("\"EExitTerminate\""));
        }
    else if (iCurrentAppThread.ExitType() == EExitPanic)
        {
        outputText.Append(_L("\"EExitPanic\""));
        }
    else if (iCurrentAppThread.ExitType() == EExitPending)
        {
        outputText.Append(_L("\"EExitPending\""));
        }
    else // unknown reason
        {
        outputText.Append(_L("\"Exit_Unknown_Reason\""));
        }        
    
    outputText.Append(_L(" code:"));
    TInt exitReason = iCurrentAppThread.ExitReason();
    outputText.AppendNum(exitReason);
    outputText.Append(_L(" \""));

    TPtrC exitCategory = iCurrentAppThread.ExitCategory();
    outputText.Append(exitCategory);

    outputText.Append(_L("\""));


    // print to screen
    iLauncherUI->PrintText(iCurrentAppNameAndExt);
    iLauncherUI->PrintText(_L(": "));
    iLauncherUI->PrintText(outputText);
    iLauncherUI->PrintText(_L("\n"));
    
    // write to the log also
    iLogWriteBuf->Des().Append(_L(" [FAIL: "));
    iLogWriteBuf->Des().Append(outputText);
    iLogWriteBuf->Des().Append(_L("]"));
    iLogWriteBuf->Des().Append(KFileNewLine);
    
    iLogFile.Write(iLogWriteBuf->Des());

    // close handle to the thread
    //iCurrentAppThread.Close();    <-- not safe the close the handle because of the "App.Closed" dialog
    //                                  somehow takes ownership of the thread or something

    // nothing to do anymore, move to the next app
    iFailedCases++;
    CheckForMoreAppsL();
    }

// ---------------------------------------------------------------------------
// ---------------------------------------------------------------------------

CDesCArray* CLauncherEngine::DependencyCheckForE32ImageL()
    {
    LOGSTRING("Launcher: CLauncherEngine::DependencyCheckForE32ImageL");

    // create an empty array for the missing dll names
    CDesCArray* missingDllArray = new(ELeave) CDesCArrayFlat(100);
    CleanupStack::PushL(missingDllArray);

    // get a list of DLLs from the E32 image file
    E32ImageReader* reader = E32ImageReader::NewLC();
    CDesCArray* dllArray = reader->ListOfDLLsL( iAppsArray->MdcaPoint(iAppLaunchCounter) );
    CleanupStack::PopAndDestroy(); // reader
    CleanupStack::PushL(dllArray);

    // compare system DLL and image DLL arrays
    TInt pos(0);
    for (TInt j=0; j<dllArray->MdcaCount(); j++)
        {
        if (iSystemDllArray->Find(dllArray->MdcaPoint(j), pos, ECmpFolded) != 0)  
            {
            // DLL not found, append the name to the list of missing DLLs
            missingDllArray->AppendL(dllArray->MdcaPoint(j));  
            }
        }

    CleanupStack::PopAndDestroy(); // dllArray
    CleanupStack::Pop();  // missingDllArray

    LOGSTRING("Launcher: CLauncherEngine::DependencyCheckForE32ImageL returns");

    return missingDllArray;
    }

// ---------------------------------------------------------------------------

TInt CLauncherEngine::DeleteLogFile()
    {
    LOGSTRING("Launcher: CLauncherEngine::DeleteLogFile");
    return BaflUtils::DeleteFile(iEnv->FsSession(), iLogFilePath);
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::StopLaunchingL()
    {
    LOGSTRING("Launcher: CLauncherEngine::StopLaunchingL");
    if(iLaunchingIsActive)
        {
        //write to the log
        WriteInitialStuffToTheLogL(_L("Cancelled by the user !!! "), iLogFile);
        iLogWriteBuf->Des().Copy(KFileNewLine);
        iLogWriteBuf->Des().Append(KFileNewLine);
        iLogFile.Write(iLogWriteBuf->Des());
    
        // close the log
        iLogFile.Close();
    
        // print to the screen
        iLauncherUI->PrintText(_L("Launching cancelled.\n\n"));
    
        // cancel all active objects
        if( iCurrentAppThread.Handle() != 0 )
            {
            iCurrentAppThread.LogonCancel(iAppThreadChecker->iStatus);
            }
        Cancel();
        iAppRunningChecker->Cancel(); 
        iAppThreadChecker->Cancel(); 
        
        iLaunchingIsActive = EFalse;
    
        _LIT(KMessage, "Launching cancelled");
        iLauncherUI->ShowInfoMessage(KMessage);

        }
    }

// ---------------------------------------------------------------------------

TBool CLauncherEngine::LogFileExists()
    {
    LOGSTRING("Launcher: CLauncherEngine::LogFileExists");
    return BaflUtils::FileExists(iEnv->FsSession(), iLogFilePath);
    }

// ---------------------------------------------------------------------------

TInt CLauncherEngine::DeleteBCLogFile()
    {
    LOGSTRING("Launcher: CLauncherEngine::DeleteBCLogFile");
    return BaflUtils::DeleteFile(iEnv->FsSession(), iBCLogFilePath);
    }


// ---------------------------------------------------------------------------

TBool CLauncherEngine::BCLogFileExists()
    {
    LOGSTRING("Launcher: CLauncherEngine::BCLogFileExists");
    return BaflUtils::FileExists(iEnv->FsSession(), iBCLogFilePath);
    }

// ---------------------------------------------------------------------------
void CLauncherEngine::AnalyseDLLsL( const TDesC& aFileName )
    {
    LOGSTRING("Launcher: CLauncherEngine::AnalyseDLLsL");
    _LIT(KStartSeparator, "******************************");
    _LIT(KStartingAnalysis, "Starting BC Analysis for DLLs.");    
    _LIT(KInputFileSelected, "Input file selected: ");
    
    // Reset found issues counter and buffer
    iFoundBCIssues = 0;    
    delete iBCIssuesBigBuffer;
    iBCIssuesBigBuffer = 0;
    
    // Reset log writing buffer:
    iLogWriteBuf->Des().Zero();
                
    if( IsActive() )
        {
        Cancel();
        }   
    
    iRequiredDllsFilePath = aFileName;
    
    if( BaflUtils::FileExists(iEnv->FsSession(), iRequiredDllsFilePath) == EFalse )
        {
        LOGSTRING2("Launcher: CLauncherEngine::AnalyseDLLsL - can't find input file: %S", &iRequiredDllsFilePath);
        User::Leave( KErrNotFound );
        }
    
    // open the log file for writing
    if (iBCLogFile.Open(iEnv->FsSession(), iBCLogFilePath, EFileWrite) != KErrNone)
        {
        iEnv->FsSession().MkDirAll(iLogFilePath);
        User::LeaveIfError( iBCLogFile.Replace(iEnv->FsSession(), iBCLogFilePath, EFileWrite) );
        }
    else
        {
        // file opens correctly, seek to the end
        TInt fileSize=0;
        iBCLogFile.Size(fileSize);
        iBCLogFile.Seek(ESeekCurrent, fileSize);
        }
    
    AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
    AppendLogBufferL(KStartSeparator(), iLogWriteBuf);
    AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
    iBCLogFile.Write(iLogWriteBuf->Des());
    iLogWriteBuf->Des().Zero();
    
    // Resolve file type. Should we use XML parsing or just compare DLL list
    TBool xmlParsing =
        KDotXML().Compare(iRequiredDllsFilePath.Right(KDotXML().Length())) == 0 ||
        KDotLauncherXML().Compare(iRequiredDllsFilePath.Right(KDotLauncherXML().Length())) == 0;
    
    // Log analysis starting time and selected input file:
    WriteInitialStuffToTheLogL(KStartingAnalysis, iBCLogFile);    
    AppendLogBufferL(KInputFileSelected(), iLogWriteBuf);
    AppendLogBufferL(iRequiredDllsFilePath, iLogWriteBuf);
    AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
    iBCLogFile.Write(iLogWriteBuf->Des());
    iLogWriteBuf->Des().Zero();
    
    if( xmlParsing )
        {
        iLauncherUI->ChangeFocusToOutputView();
        DoBCAnalysisL();
        }
    else
        {
        DoCompareDLLListsL();
        }
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::DoBCAnalysisL()
    {
    LOGSTRING("Launcher: CLauncherEngine::DoBCAnalysisL");
    if( iXMLParser == 0 )
        {
        iXMLParser = CLauncherXMLParser::NewL(iEnv->FsSession());
        }
    
    TEntry entry;
    User::LeaveIfError(iEnv->FsSession().Entry(iRequiredDllsFilePath, entry));
    TInt fileSize = entry.iSize;
    
    _LIT(KAnalysing,"Analysing DLLs");
    iLauncherUI->ShowProgressBar(KAnalysing, 0, fileSize);
    iDLLAnalysisIsActive = ETrue;
    
    iXMLParser->ParseL(iRequiredDllsFilePath, this);
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::CancelBCAnalysis()
    {
    LOGSTRING("Launcher: CLauncherEngine::CancelBCAnalysis");
    if( iXMLParser )
        {
        iXMLParser->Cancel();
        }
    iLauncherUI->PrintText(_L("\nAnalysis cancelled.\n\n"));
    WriteInitialStuffToTheLogL(_L("Analysis cancelled by user"), iBCLogFile);
    iLogWriteBuf->Des().Zero();
    iBCLogFile.Close();
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::DoCompareDLLListsL()
    {
    LOGSTRING("Launcher: CLauncherEngine::DoCompareDLLListsL");
    
    RFile file;
    CleanupClosePushL(file);
        
    // Show progress bar 
    _LIT(KAnalysing,"Analysing DLLs");
    iLauncherUI->ShowWaitDialog(KAnalysing);

    if(file.Open(iEnv->FsSession(), iRequiredDllsFilePath, EFileRead) != KErrNone)
        {
        // Hide wait dialog  
        iLauncherUI->HideWaitDialog();
        
        iLauncherUI->ChangeFocusToOutputView();

        TBuf<200> msg;

        
        msg.Format(_L("Unable to open %S for reading!\n\n"), &iRequiredDllsFilePath);
        iLauncherUI->PrintText( msg );
        }
    else
        {
        
        CDesCArray* requiredDllArray = new(ELeave) CDesCArrayFlat(KMaxDllArraySize);
        CleanupStack::PushL(requiredDllArray);

        // read all lines the text file
        TFileName dllName;
        TInt i(0);
        while( ReadLineFromFileL(file, dllName) == KErrNone && i<KMaxDllArraySize )
            {
            dllName.TrimAll();

            if (dllName.Length() > 1)
                requiredDllArray->AppendL(dllName);
            
            i++;
            }

        if (requiredDllArray->MdcaCount() == 0)
            {
            // Hide wait dialog  
            iLauncherUI->HideWaitDialog();
            
            iLauncherUI->ChangeFocusToOutputView();

            TBuf<200> msg;
            msg.Format(_L("File %S is empty!\n\n"), &iRequiredDllsFilePath);
            iLauncherUI->PrintText( msg );
            }
        else
            {
            // compare the arrays and print any missing items
            CDesCArray* missingDllArray = new(ELeave) CDesCArrayFlat(KMaxDllArraySize);
            CleanupStack::PushL(missingDllArray);

            TInt pos(0);
            for (TInt j=0; j<requiredDllArray->MdcaCount(); j++)
                {
                if (iSystemDllArray->Find(requiredDllArray->MdcaPoint(j), pos, ECmpFolded) != 0)  
                    {
                    // DLL not found, append the name to the list of missing DLLs
                    missingDllArray->AppendL(requiredDllArray->MdcaPoint(j));  
                    }
                }

            
            if (missingDllArray->MdcaCount() == 0)
                {
                _LIT(KMessage, "No missing files found");
                AppendLogBufferL(KMessage(), iLogWriteBuf);
                iLauncherUI->ShowInfoMessage(KMessage);
                }
            else
                {
                iLauncherUI->ChangeFocusToOutputView();
                _LIT(KMissingFiles, "Missing files:\n");

                //iLauncherUI->PrintText( KMissingFiles );
                AppendLogBufferL(KMissingFiles(), iBCIssuesBigBuffer, KBigBufferAllocBytes);
                
                AppendLogBufferL(KMissingFiles(), iLogWriteBuf);
                
                for (TInt i=0; i<missingDllArray->MdcaCount(); i++)
                    {
                    //iLauncherUI->PrintText( missingDllArray->MdcaPoint(i) );
                    AppendLogBufferL(missingDllArray->MdcaPoint(i), iBCIssuesBigBuffer, KBigBufferAllocBytes);
                    //iLauncherUI->PrintText( _L("\n") );         
                    AppendLogBufferL(KNewLine(), iBCIssuesBigBuffer, KBigBufferAllocBytes);
                    
                    AppendLogBufferL(missingDllArray->MdcaPoint(i), iLogWriteBuf);
                    AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
                    }
                AppendLogBufferL(KNewLine(), iBCIssuesBigBuffer, KBigBufferAllocBytes);
                iLauncherUI->PrintText( iBCIssuesBigBuffer->Des() );
                }
            
            // Hide wait dialog  
            iLauncherUI->HideWaitDialog();
            
            CleanupStack::PopAndDestroy(); // missingDllArray            
            }

        CleanupStack::PopAndDestroy(); // requiredDllArray
        }

    CleanupStack::PopAndDestroy(); //file
    iBCLogFile.Write(iLogWriteBuf->Des());
    iLogWriteBuf->Des().Zero();
    iBCLogFile.Close();
    }

// ---------------------------------------------------------------------------

void CLauncherEngine::ParsingProgressedL(TInt aBytes)
    {
    LOGSTRING2("Launcher: CLauncherEngine::ParsingProgressedL - Bytes: %d", aBytes);
    iLauncherUI->SetProgressBarValue(aBytes);
    }

// ---------------------------------------------------------------------------


void CLauncherEngine::ElementParsedL(const CLauncherDLLElement& aDllElement)
    {
    LOGSTRING("Launcher: CLauncherEngine::ElementParsedL");
    User::ResetInactivityTime();

    _LIT(KIndent, "    ");
    _LIT(KIssuesFound, "Binary compatibility issues found:"); 
    _LIT(KDLLMissing,"DLL is missing");
    _LIT(KUID1Changed,"UID1 changed");
    _LIT(KUID2Changed,"UID2 changed");
    _LIT(KUID3Changed,"UID3 changed");
    _LIT(KSIDChanged,"SID changed");
    _LIT(KCapabilityChanged,"Capability changed");
    
    if( iDLLElement == 0 )
        {
        iDLLElement = CLauncherDLLElement::NewL();
        }
    
    TFindFile fileFinder(iEnv->FsSession());
    _LIT(KDLLPath, "\\sys\\bin\\");
    TInt err = fileFinder.FindByPath(aDllElement.Name(), &KDLLPath); 
    
    TBuf<256> issueStr;
    
    if( err == KErrNotFound )
        {
        if( iFoundBCIssues++ == 0 )
            {                        
            iLauncherUI->PrintText( KIssuesFound );
            iLauncherUI->PrintText( KNewLine );            
            AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
            AppendLogBufferL(KIssuesFound(), iLogWriteBuf);
            AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
            }
        issueStr.Copy(aDllElement.Name());
        issueStr.Append(KNewLine);
        issueStr.Append(KIndent);
        issueStr.Append(KDLLMissing);
        issueStr.Append(KNewLine);
                
        AppendLogBufferL(issueStr, iLogWriteBuf);
        
        TFileName dllName = aDllElement.Name();
        LOGSTRING2("Launcher: DLL not found: %S", &dllName);
        if( iFoundBCIssues > KBigBufferUsageThreshold )
            {
            if( iBCIssuesBigBuffer == 0)
                {
                iBCIssuesBigBuffer = HBufC::NewL(KBigBufferAllocBytes);
                }
            TInt maxSize = iBCIssuesBigBuffer->Des().Length() + issueStr.Length();
            if( maxSize >= iBCIssuesBigBuffer->Des().MaxLength())
                {
                iBCIssuesBigBuffer = iBCIssuesBigBuffer->ReAllocL(maxSize + KBigBufferAllocBytes );                
                }
            TPtr ptr(iBCIssuesBigBuffer->Des());
            ptr += issueStr;            
            }
        else
            {
            iLauncherUI->PrintText( issueStr);
            }
        }
    else if( err == KErrNone)
        {
        // File is found, so let's try to open it:
        RFile dllFile;
        CleanupClosePushL(dllFile);        
        if( dllFile.Open(iEnv->FsSession(), fileFinder.File(), EFileRead) == KErrNone )
            {
            // Parse DLL:
            iDLLParser->ParseL(iEnv->FsSession(), dllFile, *iDLLElement);
            CleanupStack::PopAndDestroy(); // dllFile            
            RArray<CLauncherDLLElement::TDifference> diffs;
            CleanupClosePushL(diffs);
                        
            // Compare DLLs:              
            if( iDLLElement->CompareL(aDllElement, diffs))
                {                
                if( iFoundBCIssues++ == 0 )
                    {
                    iLauncherUI->PrintText( KIssuesFound );
                    iLauncherUI->PrintText( KNewLine );                   
                    AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
                    AppendLogBufferL(KIssuesFound(), iLogWriteBuf);
                    AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
                    }
                // Differencies found:
                for( TInt i = 0; i < diffs.Count(); ++i )
                    {                
                    // Print DLL name:
                    if( i == 0 )
                        {
                        issueStr.Copy(aDllElement.Name());
                        issueStr.Append(KNewLine);
                        }
                    
                    // Print differencies:
                    issueStr.Append(KIndent);
                    switch(diffs[i])
                        {
                        case CLauncherDLLElement::EDifference_UID1:
                            issueStr.Append(KUID1Changed);
                            break;
                        case CLauncherDLLElement::EDifference_UID2:
                            issueStr.Append( KUID2Changed );
                            break;
                        case CLauncherDLLElement::EDifference_UID3:
                            issueStr.Append( KUID3Changed );
                            break;
                        case CLauncherDLLElement::EDifference_SID:
                            issueStr.Append( KSIDChanged );
                            break;
                        case CLauncherDLLElement::EDifference_Capability:
                            issueStr.Append( KCapabilityChanged );
                            break;
                        }
                    issueStr.Append( KNewLine );                    
                    }
                AppendLogBufferL(issueStr, iLogWriteBuf);
                if( iFoundBCIssues > KBigBufferUsageThreshold )
                    {
                    // To improve performance, don't print issues to output anymore.
                    // Instead, store the issues in buffer and print them when analysis is done.
                    if( iBCIssuesBigBuffer == 0)
                        {
                        iBCIssuesBigBuffer = HBufC::NewL(KBigBufferAllocBytes);
                        }
                    TInt maxSize = iBCIssuesBigBuffer->Des().Length() + issueStr.Length();
                    if( maxSize >= iBCIssuesBigBuffer->Des().MaxLength())
                        {
                        TInt increaseSize = Max(issueStr.Length(), KBigBufferAllocBytes);
                        iBCIssuesBigBuffer = iBCIssuesBigBuffer->ReAllocL(maxSize + increaseSize );                            
                        }
                    TPtr ptr(iBCIssuesBigBuffer->Des());
                    ptr += issueStr;
                    }
                else
                    {
                    iLauncherUI->PrintText( issueStr);
                    }
                }
            CleanupStack::Pop(); // diffs
            diffs.Close();            
            }
        else
            {
            CleanupStack::PopAndDestroy(); // dllFile
            }
        }
    }

// ---------------------------------------------------------------------------
void CLauncherEngine::DocumentParsedL(TInt aErrorCode)
    {            
    LOGSTRING2("Launcher: CLauncherEngine::DocumentParsedL (Error code: %d)", aErrorCode);
        
    iDLLAnalysisIsActive = EFalse;
    _LIT(KParseError, "Parse error: ");
    _LIT(KNoIssues, "No binary compatibility issues found");
    
    if( IsActive() )
        {
        Cancel();
        }
     
    //hide progress bar 
    iLauncherUI->HideProgressBar();

    if( aErrorCode != KErrNone )
        {
        TBuf<16> errorCodeString;
        errorCodeString.AppendNum(aErrorCode);
        iLauncherUI->PrintText( KNewLine );
        iLauncherUI->PrintText( KParseError );
        iLauncherUI->PrintText( errorCodeString );
        iLauncherUI->PrintText( KNewLine );        
        AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
        AppendLogBufferL(KParseError(), iLogWriteBuf);
        AppendLogBufferL(errorCodeString, iLogWriteBuf);
        AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
        }
    else if(iFoundBCIssues == 0)
        {
        iLauncherUI->PrintText( KNoIssues );
        iLauncherUI->PrintText( KNewLine );        
        AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
        AppendLogBufferL(KNoIssues(), iLogWriteBuf);
        AppendLogBufferL(KFileNewLine(), iLogWriteBuf);
        }
    else if( iBCIssuesBigBuffer && iBCIssuesBigBuffer->Des().Length() > 0 )
        {
        iLauncherUI->PrintText(iBCIssuesBigBuffer->Des());
        delete iBCIssuesBigBuffer;
        iBCIssuesBigBuffer = 0;
        }
    if( iLogWriteBuf->Length() > 0 && iBCLogFile.SubSessionHandle() != 0)
        {
        iBCLogFile.Write(iLogWriteBuf->Des());
        }
    WriteInitialStuffToTheLogL(_L("Analysis ready"), iBCLogFile);
    iLogWriteBuf->Des().Zero();
    iBCLogFile.Close();
    }

// ---------------------------------------------------------------------------

TInt CLauncherEngine::ReadLineFromFileL(RFile& aFile, TDes& aReadBuf)
    {
    LOGSTRING("Launcher: CLauncherEngine::ReadLineFromFile");

    _LIT8(KImcvCRLF, "\r\n");
    TInt err(KErrNone);

    HBufC8* tempLine = HBufC8::NewLC(1000);
    TPtr8 buffer = tempLine->Des();

    // clear the target buffer
    aReadBuf.Zero();

    // get the current file position
    TInt filePos(0);
    aFile.Seek(ESeekCurrent, filePos);

    // read the buffer
    err = aFile.Read(buffer);

    // check if it's the end of file
    TInt s = buffer.Length();
    if (s == 0)
        err = KErrEof;

    if (err == KErrNone)
        {
        // copy to the lfcr and then set the file pointer to the point after that
        TInt pos = buffer.Find(KImcvCRLF);
        if (pos != -1)
            {
            TFileName tempBuf;
            buffer.SetLength(pos);
            tempBuf.Copy(buffer);
            aReadBuf.Justify(tempBuf, pos, ELeft, ' ');
            filePos += (pos+2);

            // set the file pointer back to after the lfcr
            aFile.Seek(ESeekStart, filePos);
            }
        
        // else fill the whole buffer
        else
            {
            aReadBuf.Copy(buffer);
            }
        }

    CleanupStack::PopAndDestroy(); // tempLine
    return err;
    }


// ---------------------------------------------------------------------------
// ---------------------------------------------------------------------------


//////////////////////////////////////////////////////////////////////////////////////

// ---------------------------------------------------------------------------

CAppThreadChecker* CAppThreadChecker::NewL(CLauncherEngine* aLauncherEngine)
    {
    LOGSTRING("Launcher: CAppThreadChecker::NewL");

    CAppThreadChecker* self = new(ELeave) CAppThreadChecker;
    CleanupStack::PushL(self);
    self->ConstructL(aLauncherEngine);
    CleanupStack::Pop();
    return self;
    }

// ---------------------------------------------------------------------------

CAppThreadChecker::CAppThreadChecker() : CActive(EActivePriorityIpcEventsHigh)
    {
    }

// ---------------------------------------------------------------------------

void CAppThreadChecker::ConstructL(CLauncherEngine* aLauncherEngine)
    {
    LOGSTRING("Launcher: CAppThreadChecker::ConstructL");

    iEnv = CEikonEnv::Static();

    iLauncherEngine = aLauncherEngine;

    CActiveScheduler::Add(this);
    }

// ---------------------------------------------------------------------------

CAppThreadChecker::~CAppThreadChecker()
    {
    LOGSTRING("Launcher: CAppThreadChecker::~CAppThreadChecker");

    Cancel();
    }

// ---------------------------------------------------------------------------

void CAppThreadChecker::ActivateChecking()
    {
    LOGSTRING("Launcher: CAppThreadChecker::ActivateChecking");

    __ASSERT_ALWAYS(!IsActive(), User::Panic(_L("Thread Checker"), 300));

    SetActive();
    }

// ---------------------------------------------------------------------------

void CAppThreadChecker::RunL()
    {
    LOGSTRING("Launcher: CAppThreadChecker::RunL");

    // check the state of the thread
    iLauncherEngine->CheckWhyThreadDiedL();
    }

// ---------------------------------------------------------------------------

void CAppThreadChecker::DoCancel()
    {
    LOGSTRING("Launcher: CAppThreadChecker::DoCancel");
    }

// ---------------------------------------------------------------------------



// ---------------------------------------------------------------------------

//////////////////////////////////////////////////////////////////////////////////////

// ---------------------------------------------------------------------------

CAppRunningChecker* CAppRunningChecker::NewL(CLauncherEngine* aLauncherEngine)
    {
    LOGSTRING("Launcher: CAppRunningChecker::NewL");

    CAppRunningChecker* self = new(ELeave) CAppRunningChecker;
    CleanupStack::PushL(self);
    self->ConstructL(aLauncherEngine);
    CleanupStack::Pop();
    return self;
    }

// ---------------------------------------------------------------------------

CAppRunningChecker::CAppRunningChecker() : CActive(EActivePriorityIpcEventsHigh)
    {
    }

// ---------------------------------------------------------------------------

void CAppRunningChecker::ConstructL(CLauncherEngine* aLauncherEngine)
    {
    LOGSTRING("Launcher: CAppRunningChecker::ConstructL");

    iEnv = CEikonEnv::Static();
    User::LeaveIfError(iTimer.CreateLocal());

    iLauncherEngine = aLauncherEngine;

    CActiveScheduler::Add(this);
    }

// ---------------------------------------------------------------------------

CAppRunningChecker::~CAppRunningChecker()
    {
    LOGSTRING("Launcher: CAppRunningChecker::~CAppRunningChecker");

    Cancel();

    iTimer.Close();
    }

// ---------------------------------------------------------------------------

void CAppRunningChecker::StartTesting()
    {
    LOGSTRING("Launcher: CAppRunningChecker::StartTesting");

    __ASSERT_ALWAYS(!IsActive(), User::Panic(_L("Running Checker"), 200));

    // async delay of seven seconds
    iTimer.After(iStatus, 7000000);    
    SetActive();
    }

// ---------------------------------------------------------------------------

void CAppRunningChecker::RunL()
    {
    LOGSTRING("Launcher: CAppRunningChecker::RunL");

    // check if the application is running
    iLauncherEngine->CheckIfAppIsRunningL();
    }

// ---------------------------------------------------------------------------

void CAppRunningChecker::DoCancel()
    {
    LOGSTRING("Launcher: CAppRunningChecker::DoCancel");
    iTimer.Cancel();
    }

// ---------------------------------------------------------------------------



//////////////////////////////////////////////////////////////////////////////////////