/**
 * @file llformat.cpp
 * @date   January 2007
 * @brief string formatting utility
 *
 * $LicenseInfo:firstyear=2007&license=viewerlgpl$
 * Second Life Viewer Source Code
 * Copyright (C) 2010, Linden Research, Inc.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation;
 * version 2.1 of the License only.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 * $/LicenseInfo$
 */

#include "linden_common.h"

#include "llapr.h" // thread-related functions
#include "llcrashlock.h"
#include "lldir.h"
#include "llsd.h"
#include "llsdserialize.h"
#include "llframetimer.h"
#include <boost/filesystem.hpp>
#include <string>
#include <iostream>
#include <stdio.h>


#if LL_WINDOWS   //For windows platform.
#include <llwin32headers.h>
#include <TlHelp32.h>

bool LLCrashLock::isProcessAlive(U32 pid, const std::string& pname)
{
    std::wstring wpname = ll_convert_string_to_wide(pname);

    HANDLE snapshot;
    PROCESSENTRY32 pe32{};

    bool matched = false;

    snapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
    if (snapshot == INVALID_HANDLE_VALUE)
    {
        return false;
    }
    else
    {
        pe32.dwSize = sizeof(PROCESSENTRY32);
        if (Process32First(snapshot, &pe32))
        {
            do {
                std::wstring wexecname = pe32.szExeFile;
                std::string execname = ll_convert_wide_to_string(wexecname);
                if (!wpname.compare(pe32.szExeFile))
                {
                    if (pid == (U32)pe32.th32ProcessID)
                    {
                        matched = true;
                        break;
                    }
                }
            } while (Process32Next(snapshot, &pe32));
        }
    }

    CloseHandle(snapshot);
    return matched;
}

#else   //Everyone Else
bool LLCrashLock::isProcessAlive(U32 pid, const std::string& pname)
{
    //Will boost.process ever become a reality?
    std::stringstream cmd;

    cmd << "pgrep '" << pname << "' | grep '^" << pid << "$'";
    return (!system(cmd.str().c_str()));
}
#endif //Everyone else.


LLCrashLock::LLCrashLock() : mCleanUp(true), mWaitingPID(0)
{
}

void LLCrashLock::setCleanUp(bool cleanup)
{
    mCleanUp = cleanup;  //Allow cleanup to be disabled for debugging.
}

LLSD LLCrashLock::getLockFile(std::string filename)
{
    LLSD lock_sd = LLSD::emptyMap();

    llifstream ifile(filename.c_str());

    if (ifile.is_open())
    {
        LLSDSerialize::fromXML(lock_sd, ifile);
        ifile.close();
    }

    return lock_sd;
}

bool LLCrashLock::putLockFile(std::string filename, const LLSD& data)
{
    bool result = true;
    llofstream ofile(filename.c_str());

    if (!LLSDSerialize::toXML(data,ofile))
    {
        result=false;
    }
    ofile.close();
    return result;
}

bool LLCrashLock::requestMaster( F32 timeout )
{
    if (mMaster.empty())
    {
            mMaster = gDirUtilp->getExpandedFilename(LL_PATH_LOGS,
                                            "crash_master.lock");
    }

    LLSD lock_sd=getLockFile(mMaster);

    if (lock_sd.has("pid"))
    {
        mWaitingPID = lock_sd["pid"].asInteger();
        if ( isProcessAlive(mWaitingPID, gDirUtilp->getExecutableFilename()) )
        {
            mTimer.resetWithExpiry(timeout);
            return false;
        }
    }

    U32 pid = getpid();
    lock_sd["pid"] = (LLSD::Integer)pid;
    return putLockFile(mMaster,lock_sd);
}

bool LLCrashLock::checkMaster()
{
    if (mWaitingPID)
    {
        return (!isProcessAlive(mWaitingPID, gDirUtilp->getExecutableFilename()));
    }
    return  false;
}

bool LLCrashLock::isWaiting()
{
    return !mTimer.hasExpired();
}

void LLCrashLock::releaseMaster()
{
    //Yeeeeeeehaw
    unlink(mMaster.c_str());
}

LLSD LLCrashLock::getProcessList()
{
    if (mDumpTable.empty())
    {
        mDumpTable= gDirUtilp->getExpandedFilename(LL_PATH_LOGS,
                                                "crash_table.lock");
    }
   return getLockFile(mDumpTable);
}

//static
bool LLCrashLock::fileExists(std::string filename)
{
#ifdef LL_WINDOWS // or BOOST_WINDOWS_API
    boost::filesystem::path file_path(utf8str_to_utf16str(filename));
#else
    boost::filesystem::path file_path(filename);
#endif
    return boost::filesystem::exists(file_path);
}

void LLCrashLock::cleanupProcess(std::string proc_dir)
{
#ifdef LL_WINDOWS // or BOOST_WINDOWS_API
    boost::filesystem::path dir_path(utf8str_to_utf16str(proc_dir));
#else
    boost::filesystem::path dir_path(proc_dir);
#endif
    boost::filesystem::remove_all(dir_path);
}

bool LLCrashLock::putProcessList(const LLSD& proc_sd)
{
    return putLockFile(mDumpTable,proc_sd);
}