/* $Id: ClipboardStreamImpl-win.cpp 80359 2019-08-21 08:37:54Z vboxsync $ */ /** @file * ClipboardStreamImpl-win.cpp - Shared Clipboard IStream object implementation (guest and host side). */ /* * Copyright (C) 2019 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ /********************************************************************************************************************************* * Header Files * *********************************************************************************************************************************/ #define LOG_GROUP LOG_GROUP_SHARED_CLIPBOARD #include #include #include #include #include #include #include #include /********************************************************************************************************************************* * Structures and Typedefs * *********************************************************************************************************************************/ /********************************************************************************************************************************* * Static variables * *********************************************************************************************************************************/ VBoxClipboardWinStreamImpl::VBoxClipboardWinStreamImpl(VBoxClipboardWinDataObject *pParent, PSHAREDCLIPBOARDURITRANSFER pTransfer, const Utf8Str &strPath, PSHAREDCLIPBOARDFSOBJINFO pObjInfo) : m_pParent(pParent) , m_lRefCount(1) /* Our IDataObjct *always* holds the last reference to this object; needed for the callbacks. */ , m_pURITransfer(pTransfer) , m_strPath(strPath) , m_hObj(SHAREDCLIPBOARDOBJHANDLE_INVALID) , m_objInfo(*pObjInfo) , m_cbProcessed(0) , m_fNotifiedComplete(false) { AssertPtr(m_pURITransfer); LogFunc(("m_strPath=%s\n", m_strPath.c_str())); } VBoxClipboardWinStreamImpl::~VBoxClipboardWinStreamImpl(void) { LogFlowThisFuncEnter(); } /* * IUnknown methods. */ STDMETHODIMP VBoxClipboardWinStreamImpl::QueryInterface(REFIID iid, void **ppvObject) { AssertPtrReturn(ppvObject, E_INVALIDARG); if (iid == IID_IUnknown) { LogFlowFunc(("IID_IUnknown\n")); *ppvObject = (IUnknown *)(ISequentialStream *)this; } else if (iid == IID_ISequentialStream) { LogFlowFunc(("IID_ISequentialStream\n")); *ppvObject = (ISequentialStream *)this; } else if (iid == IID_IStream) { LogFlowFunc(("IID_IStream\n")); *ppvObject = (IStream *)this; } else { *ppvObject = NULL; return E_NOINTERFACE; } AddRef(); return S_OK; } STDMETHODIMP_(ULONG) VBoxClipboardWinStreamImpl::AddRef(void) { LONG lCount = InterlockedIncrement(&m_lRefCount); LogFlowFunc(("lCount=%RI32\n", lCount)); return lCount; } STDMETHODIMP_(ULONG) VBoxClipboardWinStreamImpl::Release(void) { LONG lCount = InterlockedDecrement(&m_lRefCount); LogFlowFunc(("lCount=%RI32\n", m_lRefCount)); if (lCount == 0) { if ( !m_fNotifiedComplete && m_pParent) { m_pParent->OnTransferComplete(); } delete this; return 0; } return lCount; } /* * IStream methods. */ STDMETHODIMP VBoxClipboardWinStreamImpl::Clone(IStream** ppStream) { RT_NOREF(ppStream); LogFlowFuncEnter(); return E_NOTIMPL; } STDMETHODIMP VBoxClipboardWinStreamImpl::Commit(DWORD dwFrags) { RT_NOREF(dwFrags); LogFlowThisFuncEnter(); return E_NOTIMPL; } STDMETHODIMP VBoxClipboardWinStreamImpl::CopyTo(IStream *pDestStream, ULARGE_INTEGER nBytesToCopy, ULARGE_INTEGER *nBytesRead, ULARGE_INTEGER *nBytesWritten) { RT_NOREF(pDestStream, nBytesToCopy, nBytesRead, nBytesWritten); LogFlowThisFuncEnter(); return E_NOTIMPL; } STDMETHODIMP VBoxClipboardWinStreamImpl::LockRegion(ULARGE_INTEGER nStart, ULARGE_INTEGER nBytes,DWORD dwFlags) { RT_NOREF(nStart, nBytes, dwFlags); LogFlowThisFuncEnter(); return STG_E_INVALIDFUNCTION; } /* Note: Windows seems to assume EOF if nBytesRead < nBytesToRead. */ STDMETHODIMP VBoxClipboardWinStreamImpl::Read(void *pvBuffer, ULONG nBytesToRead, ULONG *nBytesRead) { LogFlowThisFunc(("Enter: m_cbProcessed=%RU64\n", m_cbProcessed)); /** @todo Is there any locking required so that parallel reads aren't possible? */ if (!pvBuffer) return STG_E_INVALIDPOINTER; if (nBytesToRead == 0) { if (nBytesRead) *nBytesRead = 0; return S_OK; } int rc; try { if ( m_hObj == SHAREDCLIPBOARDOBJHANDLE_INVALID && m_pURITransfer->ProviderIface.pfnObjOpen) { VBOXCLIPBOARDOBJOPENCREATEPARMS openParms; rc = SharedClipboardURIObjectOpenParmsInit(&openParms); if (RT_SUCCESS(rc)) { openParms.fCreate = SHAREDCLIPBOARD_OBJ_CF_ACT_OPEN_IF_EXISTS | SHAREDCLIPBOARD_OBJ_CF_ACT_FAIL_IF_NEW | SHAREDCLIPBOARD_OBJ_CF_ACCESS_READ | SHAREDCLIPBOARD_OBJ_CF_ACCESS_DENYWRITE; rc = RTStrCopy(openParms.pszPath, openParms.cbPath, m_strPath.c_str()); if (RT_SUCCESS(rc)) { rc = m_pURITransfer->ProviderIface.pfnObjOpen(&m_pURITransfer->ProviderCtx, &openParms, &m_hObj); } SharedClipboardURIObjectOpenParmsDestroy(&openParms); } } else rc = VINF_SUCCESS; uint32_t cbRead = 0; const uint64_t cbSize = (uint64_t)m_objInfo.cbObject; const uint32_t cbToRead = RT_MIN(cbSize - m_cbProcessed, nBytesToRead); bool fComplete = false; if (RT_SUCCESS(rc)) { if (cbToRead) { rc = m_pURITransfer->ProviderIface.pfnObjRead(&m_pURITransfer->ProviderCtx, m_hObj, pvBuffer, cbToRead, 0 /* fFlags */, &cbRead); if (RT_SUCCESS(rc)) { m_cbProcessed += cbRead; Assert(m_cbProcessed <= cbSize); } } /* Transfer complete? Make sure to close the object again. */ fComplete = m_cbProcessed == cbSize; if (fComplete) { if (m_pURITransfer->ProviderIface.pfnObjClose) { int rc2 = m_pURITransfer->ProviderIface.pfnObjClose(&m_pURITransfer->ProviderCtx, m_hObj); AssertRC(rc2); } if (m_pParent) { m_pParent->OnTransferComplete(); m_fNotifiedComplete = true; } } } LogFlowThisFunc(("Leave: rc=%Rrc, cbSize=%RU64, cbProcessed=%RU64 -> nBytesToRead=%RU32, cbToRead=%RU32, cbRead=%RU32\n", rc, cbSize, m_cbProcessed, nBytesToRead, cbToRead, cbRead)); if (nBytesRead) *nBytesRead = (ULONG)cbRead; if (nBytesToRead != cbRead) return S_FALSE; return S_OK; } catch (...) { LogFunc(("Caught exception\n")); } return E_FAIL; } STDMETHODIMP VBoxClipboardWinStreamImpl::Revert(void) { LogFlowThisFuncEnter(); return E_NOTIMPL; } STDMETHODIMP VBoxClipboardWinStreamImpl::Seek(LARGE_INTEGER nMove, DWORD dwOrigin, ULARGE_INTEGER* nNewPos) { RT_NOREF(nMove, dwOrigin, nNewPos); LogFlowThisFunc(("nMove=%RI64, dwOrigin=%RI32\n", nMove, dwOrigin)); return E_NOTIMPL; } STDMETHODIMP VBoxClipboardWinStreamImpl::SetSize(ULARGE_INTEGER nNewSize) { RT_NOREF(nNewSize); LogFlowThisFuncEnter(); return E_NOTIMPL; } STDMETHODIMP VBoxClipboardWinStreamImpl::Stat(STATSTG *pStatStg, DWORD dwFlags) { HRESULT hr = S_OK; if (pStatStg) { RT_BZERO(pStatStg, sizeof(STATSTG)); switch (dwFlags) { case STATFLAG_NONAME: pStatStg->pwcsName = NULL; break; case STATFLAG_DEFAULT: { int rc2 = RTStrToUtf16(m_strPath.c_str(), &pStatStg->pwcsName); if (RT_FAILURE(rc2)) hr = E_FAIL; break; } default: hr = STG_E_INVALIDFLAG; break; } if (SUCCEEDED(hr)) { pStatStg->type = STGTY_STREAM; pStatStg->grfMode = STGM_READ; pStatStg->grfLocksSupported = 0; pStatStg->cbSize.QuadPart = (uint64_t)m_objInfo.cbObject; } } else hr = STG_E_INVALIDPOINTER; LogFlowThisFunc(("hr=%Rhrc\n", hr)); return hr; } STDMETHODIMP VBoxClipboardWinStreamImpl::UnlockRegion(ULARGE_INTEGER nStart, ULARGE_INTEGER nBytes, DWORD dwFlags) { RT_NOREF(nStart, nBytes, dwFlags); LogFlowThisFuncEnter(); return E_NOTIMPL; } STDMETHODIMP VBoxClipboardWinStreamImpl::Write(const void *pvBuffer, ULONG nBytesToRead, ULONG *nBytesRead) { RT_NOREF(pvBuffer, nBytesToRead, nBytesRead); LogFlowThisFuncEnter(); return E_NOTIMPL; } /* * Own stuff. */ /** * Factory to create our own IStream implementation. * * @returns HRESULT * @param pParent Pointer to the parent data object. * @param pTransfer Pointer to URI transfer object to use. * @param strPath Path of object to handle for the stream. * @param pObjInfo Pointer to object information. * @param ppStream Where to return the created stream object on success. */ /* static */ HRESULT VBoxClipboardWinStreamImpl::Create(VBoxClipboardWinDataObject *pParent, PSHAREDCLIPBOARDURITRANSFER pTransfer, const Utf8Str &strPath, PSHAREDCLIPBOARDFSOBJINFO pObjInfo, IStream **ppStream) { AssertPtrReturn(pTransfer, E_POINTER); VBoxClipboardWinStreamImpl *pStream = new VBoxClipboardWinStreamImpl(pParent, pTransfer, strPath, pObjInfo); if (pStream) { pStream->AddRef(); *ppStream = pStream; return S_OK; } return E_FAIL; }