GuestProcessImpl.h revision dea3e7faa80f4aab41e08945b9308fd2e3ffe7fc
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync/* $Id$ */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync/** @file
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * VirtualBox Main - Guest process handling.
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync/*
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * Copyright (C) 2012-2013 Oracle Corporation
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync *
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * This file is part of VirtualBox Open Source Edition (OSE), as
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * available from http://www.virtualbox.org. This file is free software;
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * you can redistribute it and/or modify it under the terms of the GNU
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * General Public License (GPL) as published by the Free Software
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * Foundation, in version 2 as it comes in the "COPYING" file of the
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync#ifndef ____H_GUESTPROCESSIMPL
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync#define ____H_GUESTPROCESSIMPL
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync#include "GuestCtrlImplPrivate.h"
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync#include "GuestProcessWrap.h"
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsyncclass Console;
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsyncclass GuestSession;
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync/**
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * Class for handling a guest process.
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsyncclass ATL_NO_VTABLE GuestProcess :
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync public GuestProcessWrap,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync public GuestObject
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync{
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsyncpublic:
a917e98de4393d1090e536fdbeb04c285f98c92evboxsync /** @name COM and internal init/term/mapping cruft.
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * @{ */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync DECLARE_EMPTY_CTOR_DTOR(GuestProcess)
a917e98de4393d1090e536fdbeb04c285f98c92evboxsync
a917e98de4393d1090e536fdbeb04c285f98c92evboxsync int init(Console *aConsole, GuestSession *aSession, ULONG aProcessID, const GuestProcessStartupInfo &aProcInfo);
a917e98de4393d1090e536fdbeb04c285f98c92evboxsync void uninit(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT FinalConstruct(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync void FinalRelease(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync /** @} */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsyncpublic:
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync /** @name Public internal methods.
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * @{ */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_callbackDispatcher(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCb);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync inline int i_checkPID(uint32_t uPID);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync static Utf8Str i_guestErrorToString(int guestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_onRemove(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_readData(uint32_t uHandle, uint32_t uSize, uint32_t uTimeoutMS, void *pvData, size_t cbData, uint32_t *pcbRead, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync static HRESULT i_setErrorExternal(VirtualBoxBase *pInterface, int guestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_startProcess(uint32_t uTimeoutMS, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_startProcessAsync(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_terminateProcess(uint32_t uTimeoutMS, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync static ProcessWaitResult_T i_waitFlagsToResultEx(uint32_t fWaitFlags, ProcessStatus_T oldStatus, ProcessStatus_T newStatus, uint32_t uProcFlags, uint32_t uProtocol);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ProcessWaitResult_T i_waitFlagsToResult(uint32_t fWaitFlags);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_waitFor(uint32_t fWaitFlags, ULONG uTimeoutMS, ProcessWaitResult_T &waitResult, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_waitForInputNotify(GuestWaitEvent *pEvent, uint32_t uHandle, uint32_t uTimeoutMS, ProcessInputStatus_T *pInputStatus, uint32_t *pcbProcessed);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_waitForOutput(GuestWaitEvent *pEvent, uint32_t uHandle, uint32_t uTimeoutMS, void* pvData, size_t cbData, uint32_t *pcbRead);
956230513f1e35e9d694c219c7009034b71bb836vboxsync int i_waitForStatusChange(GuestWaitEvent *pEvent, uint32_t uTimeoutMS, ProcessStatus_T *pProcessStatus, int *pGuestRc);
956230513f1e35e9d694c219c7009034b71bb836vboxsync static bool i_waitResultImpliesEx(ProcessWaitResult_T waitResult, ProcessStatus_T procStatus, uint32_t uProcFlags, uint32_t uProtocol);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_writeData(uint32_t uHandle, uint32_t uFlags, void *pvData, size_t cbData, uint32_t uTimeoutMS, uint32_t *puWritten, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync /** @} */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsyncprotected:
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync /** @name Protected internal methods.
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * @{ */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync inline bool i_isAlive(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync inline bool i_hasEnded(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_onGuestDisconnected(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCbData);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_onProcessInputStatus(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCbData);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_onProcessNotifyIO(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCbData);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_onProcessStatusChange(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCbData);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_onProcessOutput(PVBOXGUESTCTRLHOSTCBCTX pCbCtx, PVBOXGUESTCTRLHOSTCALLBACK pSvcCbData);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_prepareExecuteEnv(const char *pszEnv, void **ppvList, ULONG *pcbList, ULONG *pcEnvVars);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_setProcessStatus(ProcessStatus_T procStatus, int procRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync static DECLCALLBACK(int) i_startProcessThread(RTTHREAD Thread, void *pvUser);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync /** @} */
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsyncprivate:
956230513f1e35e9d694c219c7009034b71bb836vboxsync /** Wrapped @name IProcess data .
956230513f1e35e9d694c219c7009034b71bb836vboxsync * @{ */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getArguments(std::vector<com::Utf8Str> &aArguments);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getEnvironment(std::vector<com::Utf8Str> &aEnvironment);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getEventSource(ComPtr<IEventSource> &aEventSource);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getExecutablePath(com::Utf8Str &aExecutablePath);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getExitCode(LONG *aExitCode);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getName(com::Utf8Str &aName);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getPID(ULONG *aPID);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT getStatus(ProcessStatus_T *aStatus);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync /** Wrapped @name IProcess methods.
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync * @{ */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT waitFor(ULONG aWaitFor,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ULONG aTimeoutMS,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ProcessWaitResult_T *aReason);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT waitForArray(const std::vector<ProcessWaitForFlag_T> &aWaitFor,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ULONG aTimeoutMS,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ProcessWaitResult_T *aReason);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT read(ULONG aHandle,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ULONG aToRead,
956230513f1e35e9d694c219c7009034b71bb836vboxsync ULONG aTimeoutMS,
956230513f1e35e9d694c219c7009034b71bb836vboxsync std::vector<BYTE> &aData);
956230513f1e35e9d694c219c7009034b71bb836vboxsync HRESULT write(ULONG aHandle,
956230513f1e35e9d694c219c7009034b71bb836vboxsync ULONG aFlags,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync const std::vector<BYTE> &aData,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ULONG aTimeoutMS,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync ULONG *aWritten);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync HRESULT writeArray(ULONG aHandle,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync const std::vector<ProcessInputFlag_T> &aFlags,
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync const std::vector<BYTE> &aData,
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync ULONG aTimeoutMS,
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync ULONG *aWritten);
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync HRESULT terminate();
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync /**
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync * This can safely be used without holding any locks.
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync * An AutoCaller suffices to prevent it being destroy while in use and
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync * internally there is a lock providing the necessary serialization.
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync */
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync const ComObjPtr<EventSource> mEventSource;
110a95f133a07d2c98a580d226abf8850bc2bd47vboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync struct Data
956230513f1e35e9d694c219c7009034b71bb836vboxsync {
956230513f1e35e9d694c219c7009034b71bb836vboxsync /** The process startup information. */
956230513f1e35e9d694c219c7009034b71bb836vboxsync GuestProcessStartupInfo mProcess;
956230513f1e35e9d694c219c7009034b71bb836vboxsync /** Exit code if process has been terminated. */
956230513f1e35e9d694c219c7009034b71bb836vboxsync LONG mExitCode;
956230513f1e35e9d694c219c7009034b71bb836vboxsync /** PID reported from the guest. */
956230513f1e35e9d694c219c7009034b71bb836vboxsync ULONG mPID;
956230513f1e35e9d694c219c7009034b71bb836vboxsync /** The current process status. */
956230513f1e35e9d694c219c7009034b71bb836vboxsync ProcessStatus_T mStatus;
956230513f1e35e9d694c219c7009034b71bb836vboxsync /** The last returned process status
956230513f1e35e9d694c219c7009034b71bb836vboxsync * returned from the guest side. */
956230513f1e35e9d694c219c7009034b71bb836vboxsync int mLastError;
956230513f1e35e9d694c219c7009034b71bb836vboxsync } mData;
956230513f1e35e9d694c219c7009034b71bb836vboxsync};
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsync/**
956230513f1e35e9d694c219c7009034b71bb836vboxsync * Guest process tool flags.
956230513f1e35e9d694c219c7009034b71bb836vboxsync */
956230513f1e35e9d694c219c7009034b71bb836vboxsync/** No flags specified; wait until process terminates.
956230513f1e35e9d694c219c7009034b71bb836vboxsync * The maximum waiting time is set in the process' startup
956230513f1e35e9d694c219c7009034b71bb836vboxsync * info. */
956230513f1e35e9d694c219c7009034b71bb836vboxsync#define GUESTPROCESSTOOL_FLAG_NONE 0
956230513f1e35e9d694c219c7009034b71bb836vboxsync/** Wait until next stream block from stdout has been
956230513f1e35e9d694c219c7009034b71bb836vboxsync * read in completely, then return.
956230513f1e35e9d694c219c7009034b71bb836vboxsync */
956230513f1e35e9d694c219c7009034b71bb836vboxsync#define GUESTPROCESSTOOL_FLAG_STDOUT_BLOCK RT_BIT(0)
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsync/**
956230513f1e35e9d694c219c7009034b71bb836vboxsync * Internal class for handling a VBoxService tool ("vbox_ls", vbox_stat", ...).
956230513f1e35e9d694c219c7009034b71bb836vboxsync */
956230513f1e35e9d694c219c7009034b71bb836vboxsyncclass GuestProcessTool
956230513f1e35e9d694c219c7009034b71bb836vboxsync{
956230513f1e35e9d694c219c7009034b71bb836vboxsyncpublic:
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsync GuestProcessTool(void);
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsync virtual ~GuestProcessTool(void);
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsyncpublic:
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsync int Init(GuestSession *pGuestSession, const GuestProcessStartupInfo &startupInfo, bool fAsync, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync GuestProcessStream &i_getStdOut(void) { return mStdOut; }
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync GuestProcessStream &i_getStdErr(void) { return mStdErr; }
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_wait(uint32_t fFlags, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_waitEx(uint32_t fFlags, GuestProcessStreamBlock *pStreamBlock, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_getCurrentBlock(uint32_t uHandle, GuestProcessStreamBlock &strmBlock);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync bool i_isRunning(void);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync static int i_run(GuestSession *pGuestSession, const GuestProcessStartupInfo &startupInfo, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync static int i_runEx(GuestSession *pGuestSession, const GuestProcessStartupInfo &startupInfo, GuestCtrlStreamObjects *pStrmOutObjects,
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync uint32_t cStrmOutObjects, int *pGuestRc);
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync int i_terminatedOk(LONG *pExitCode);
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsync int i_terminate(uint32_t uTimeoutMS, int *pGuestRc);
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsyncprotected:
956230513f1e35e9d694c219c7009034b71bb836vboxsync
956230513f1e35e9d694c219c7009034b71bb836vboxsync ComObjPtr<GuestSession> pSession;
956230513f1e35e9d694c219c7009034b71bb836vboxsync ComObjPtr<GuestProcess> pProcess;
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync GuestProcessStartupInfo mStartupInfo;
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync GuestProcessStream mStdOut;
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync GuestProcessStream mStdErr;
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync};
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync#endif /* !____H_GUESTPROCESSIMPL */
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync
f1f5335f9ec8e56fe0e3e27f253e24b10ff20f2evboxsync