23#include <freerdp/config.h>
33#include <winpr/assert.h>
34#include <winpr/library.h>
37#include <winpr/tchar.h>
38#include <winpr/stream.h>
40#include <freerdp/log.h>
41#include <freerdp/client/cliprdr.h>
45#include "wf_cliprdr.h"
47#define TAG CLIENT_TAG("windows")
49#ifdef WITH_DEBUG_CLIPRDR
50#define DEBUG_CLIPRDR(...) WLog_DBG(TAG, __VA_ARGS__)
52#define DEBUG_CLIPRDR(...) \
58typedef BOOL(WINAPI* fnAddClipboardFormatListener)(HWND hwnd);
59typedef BOOL(WINAPI* fnRemoveClipboardFormatListener)(HWND hwnd);
60typedef BOOL(WINAPI* fnGetUpdatedClipboardFormats)(PUINT lpuiFormats, UINT cFormats,
65 UINT32 remote_format_id;
66 UINT32 local_format_id;
72 IEnumFORMATETC iEnumFORMATETC;
77 FORMATETC* m_pFormatEtc;
78} CliprdrEnumFORMATETC;
94 IDataObject iDataObject;
97 FORMATETC* m_pFormatEtc;
98 STGMEDIUM* m_pStgMedium;
108 rdpChannels* channels;
109 CliprdrClientContext* context;
116 formatMapping* format_mappings;
118 UINT32 requestedFormatId;
123 HANDLE response_data_event;
125 LPDATAOBJECT data_obj;
131 size_t file_array_size;
138 fnAddClipboardFormatListener AddClipboardFormatListener;
139 fnRemoveClipboardFormatListener RemoveClipboardFormatListener;
140 fnGetUpdatedClipboardFormats GetUpdatedClipboardFormats;
143#define WM_CLIPRDR_MESSAGE (WM_USER + 156)
144#define OLE_SETCLIPBOARD 1
146static BOOL wf_create_file_obj(wfClipboard* cliprdrrdr, IDataObject** ppDataObject);
147static void wf_destroy_file_obj(IDataObject* instance);
148static UINT32 get_remote_format_id(wfClipboard* clipboard, UINT32 local_format);
149static UINT cliprdr_send_data_request(wfClipboard* clipboard, UINT32 format);
150static UINT cliprdr_send_lock(wfClipboard* clipboard);
151static UINT cliprdr_send_unlock(wfClipboard* clipboard);
152static UINT cliprdr_send_request_filecontents(wfClipboard* clipboard,
const void* streamid,
153 ULONG index, UINT32 flag, UINT64 position,
156static void CliprdrDataObject_Delete(CliprdrDataObject* instance);
158static CliprdrEnumFORMATETC* CliprdrEnumFORMATETC_New(ULONG nFormats, FORMATETC* pFormatEtc);
159static void CliprdrEnumFORMATETC_Delete(CliprdrEnumFORMATETC* instance);
161static void CliprdrStream_Delete(CliprdrStream* instance);
163static BOOL try_open_clipboard(HWND hwnd)
165 for (
size_t x = 0; x < 10; x++)
167 if (OpenClipboard(hwnd))
178static HRESULT STDMETHODCALLTYPE CliprdrStream_QueryInterface(IStream* This, REFIID riid,
181 if (IsEqualIID(riid, &IID_IStream) || IsEqualIID(riid, &IID_IUnknown))
183 IStream_AddRef(This);
190 return E_NOINTERFACE;
194static ULONG STDMETHODCALLTYPE CliprdrStream_AddRef(IStream* This)
196 CliprdrStream* instance = (CliprdrStream*)This;
201 return InterlockedIncrement(&instance->m_lRefCount);
204static ULONG STDMETHODCALLTYPE CliprdrStream_Release(IStream* This)
207 CliprdrStream* instance = (CliprdrStream*)This;
212 count = InterlockedDecrement(&instance->m_lRefCount);
216 CliprdrStream_Delete(instance);
225static HRESULT STDMETHODCALLTYPE CliprdrStream_Read(IStream* This,
void* pv, ULONG cb,
229 CliprdrStream* instance = (CliprdrStream*)This;
230 wfClipboard* clipboard;
232 if (!pv || !pcbRead || !instance)
235 clipboard = (wfClipboard*)instance->m_pData;
238 if (instance->m_lOffset.QuadPart >= instance->m_lSize.QuadPart)
241 ret = cliprdr_send_request_filecontents(clipboard, (
void*)This, instance->m_lIndex,
242 FILECONTENTS_RANGE, instance->m_lOffset.QuadPart, cb);
247 if (clipboard->req_fdata)
249 CopyMemory(pv, clipboard->req_fdata, clipboard->req_fsize);
250 free(clipboard->req_fdata);
253 *pcbRead = clipboard->req_fsize;
254 instance->m_lOffset.QuadPart += clipboard->req_fsize;
256 if (clipboard->req_fsize < cb)
262static HRESULT STDMETHODCALLTYPE CliprdrStream_Write(IStream* This,
const void* pv, ULONG cb,
269 return STG_E_ACCESSDENIED;
272static HRESULT STDMETHODCALLTYPE CliprdrStream_Seek(IStream* This,
LARGE_INTEGER dlibMove,
276 CliprdrStream* instance = (CliprdrStream*)This;
281 newoffset = instance->m_lOffset.QuadPart;
285 case STREAM_SEEK_SET:
286 newoffset = dlibMove.QuadPart;
289 case STREAM_SEEK_CUR:
290 newoffset += dlibMove.QuadPart;
293 case STREAM_SEEK_END:
294 newoffset = instance->m_lSize.QuadPart + dlibMove.QuadPart;
301 if (newoffset < 0 || newoffset >= instance->m_lSize.QuadPart)
304 instance->m_lOffset.QuadPart = newoffset;
307 plibNewPosition->QuadPart = instance->m_lOffset.QuadPart;
312static HRESULT STDMETHODCALLTYPE CliprdrStream_SetSize(IStream* This,
ULARGE_INTEGER libNewSize)
319static HRESULT STDMETHODCALLTYPE CliprdrStream_CopyTo(IStream* This, IStream* pstm,
331static HRESULT STDMETHODCALLTYPE CliprdrStream_Commit(IStream* This, DWORD grfCommitFlags)
334 (void)grfCommitFlags;
338static HRESULT STDMETHODCALLTYPE CliprdrStream_Revert(IStream* This)
344static HRESULT STDMETHODCALLTYPE CliprdrStream_LockRegion(IStream* This,
ULARGE_INTEGER libOffset,
354static HRESULT STDMETHODCALLTYPE CliprdrStream_UnlockRegion(IStream* This,
ULARGE_INTEGER libOffset,
364static HRESULT STDMETHODCALLTYPE CliprdrStream_Stat(IStream* This, STATSTG* pstatstg,
367 CliprdrStream* instance = (CliprdrStream*)This;
372 if (pstatstg ==
nullptr)
373 return STG_E_INVALIDPOINTER;
375 ZeroMemory(pstatstg,
sizeof(STATSTG));
379 case STATFLAG_DEFAULT:
380 return STG_E_INSUFFICIENTMEMORY;
382 case STATFLAG_NONAME:
383 pstatstg->cbSize.QuadPart = instance->m_lSize.QuadPart;
384 pstatstg->grfLocksSupported = LOCK_EXCLUSIVE;
385 pstatstg->grfMode = GENERIC_READ;
386 pstatstg->grfStateBits = 0;
387 pstatstg->type = STGTY_STREAM;
390 case STATFLAG_NOOPEN:
391 return STG_E_INVALIDFLAG;
394 return STG_E_INVALIDFLAG;
400static HRESULT STDMETHODCALLTYPE CliprdrStream_Clone(IStream* This, IStream** ppstm)
407static CliprdrStream* CliprdrStream_New(ULONG index,
void* pData,
const FILEDESCRIPTORW* dsc)
410 BOOL success = FALSE;
412 CliprdrStream* instance;
413 wfClipboard* clipboard = (wfClipboard*)pData;
414 instance = (CliprdrStream*)calloc(1,
sizeof(CliprdrStream));
418 instance->m_Dsc = *dsc;
419 iStream = &instance->iStream;
420 iStream->lpVtbl = (IStreamVtbl*)calloc(1,
sizeof(IStreamVtbl));
424 iStream->lpVtbl->QueryInterface = CliprdrStream_QueryInterface;
425 iStream->lpVtbl->AddRef = CliprdrStream_AddRef;
426 iStream->lpVtbl->Release = CliprdrStream_Release;
427 iStream->lpVtbl->Read = CliprdrStream_Read;
428 iStream->lpVtbl->Write = CliprdrStream_Write;
429 iStream->lpVtbl->Seek = CliprdrStream_Seek;
430 iStream->lpVtbl->SetSize = CliprdrStream_SetSize;
431 iStream->lpVtbl->CopyTo = CliprdrStream_CopyTo;
432 iStream->lpVtbl->Commit = CliprdrStream_Commit;
433 iStream->lpVtbl->Revert = CliprdrStream_Revert;
434 iStream->lpVtbl->LockRegion = CliprdrStream_LockRegion;
435 iStream->lpVtbl->UnlockRegion = CliprdrStream_UnlockRegion;
436 iStream->lpVtbl->Stat = CliprdrStream_Stat;
437 iStream->lpVtbl->Clone = CliprdrStream_Clone;
438 instance->m_lRefCount = 1;
439 instance->m_lIndex = index;
440 instance->m_pData = pData;
441 instance->m_lOffset.QuadPart = 0;
443 if (instance->m_Dsc.dwFlags & FD_ATTRIBUTES)
445 if (instance->m_Dsc.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
449 if (((instance->m_Dsc.dwFlags & FD_FILESIZE) == 0) && !isDir)
452 if (cliprdr_send_request_filecontents(clipboard, (
void*)instance,
453 instance->m_lIndex, FILECONTENTS_SIZE, 0,
459 instance->m_lSize.QuadPart = *((LONGLONG*)clipboard->req_fdata);
460 free(clipboard->req_fdata);
464 instance->m_lSize.QuadPart =
465 ((UINT64)instance->m_Dsc.nFileSizeHigh << 32) | instance->m_Dsc.nFileSizeLow;
473 CliprdrStream_Delete(instance);
480void CliprdrStream_Delete(CliprdrStream* instance)
484 free(instance->iStream.lpVtbl);
493static LONG cliprdr_lookup_format(CliprdrDataObject* instance, FORMATETC* pFormatEtc)
495 if (!instance || !pFormatEtc)
498 for (ULONG i = 0; i < instance->m_nNumFormats; i++)
500 if ((pFormatEtc->tymed & instance->m_pFormatEtc[i].tymed) &&
501 pFormatEtc->cfFormat == instance->m_pFormatEtc[i].cfFormat &&
502 pFormatEtc->dwAspect & instance->m_pFormatEtc[i].dwAspect)
511static HRESULT STDMETHODCALLTYPE CliprdrDataObject_QueryInterface(IDataObject* This, REFIID riid,
519 if (IsEqualIID(riid, &IID_IDataObject) || IsEqualIID(riid, &IID_IUnknown))
521 IDataObject_AddRef(This);
528 return E_NOINTERFACE;
532static ULONG STDMETHODCALLTYPE CliprdrDataObject_AddRef(IDataObject* This)
534 CliprdrDataObject* instance = (CliprdrDataObject*)This;
539 return InterlockedIncrement(&instance->m_lRefCount);
542static ULONG STDMETHODCALLTYPE CliprdrDataObject_Release(IDataObject* This)
545 CliprdrDataObject* instance = (CliprdrDataObject*)This;
550 count = InterlockedDecrement(&instance->m_lRefCount);
554 CliprdrDataObject_Delete(instance);
561static HRESULT STDMETHODCALLTYPE CliprdrDataObject_GetData(IDataObject* This, FORMATETC* pFormatEtc,
565 CliprdrDataObject* instance = (CliprdrDataObject*)This;
566 wfClipboard* clipboard;
568 if (!pFormatEtc || !pMedium || !instance)
571 clipboard = (wfClipboard*)instance->m_pData;
576 if ((idx = cliprdr_lookup_format(instance, pFormatEtc)) == -1)
577 return DV_E_FORMATETC;
579 pMedium->tymed = instance->m_pFormatEtc[idx].tymed;
580 pMedium->pUnkForRelease = 0;
582 if (instance->m_pFormatEtc[idx].cfFormat == RegisterClipboardFormat(CFSTR_FILEDESCRIPTORW))
584 FILEGROUPDESCRIPTOR* dsc;
585 DWORD remote = get_remote_format_id(clipboard, instance->m_pFormatEtc[idx].cfFormat);
587 if (cliprdr_send_data_request(clipboard, remote) != 0)
590 pMedium->u.hGlobal = clipboard->hmem;
595 dsc = (FILEGROUPDESCRIPTOR*)GlobalLock(clipboard->hmem);
596 instance->m_nStreams = dsc->cItems;
597 GlobalUnlock(clipboard->hmem);
599 if (instance->m_nStreams > 0)
601 if (!instance->m_pStream)
603 instance->m_pStream = (LPSTREAM*)calloc(instance->m_nStreams,
sizeof(LPSTREAM));
605 if (instance->m_pStream)
607 for (ULONG i = 0; i < instance->m_nStreams; i++)
609 instance->m_pStream[i] =
610 (IStream*)CliprdrStream_New(i, clipboard, &dsc->fgd[i]);
612 if (!instance->m_pStream[i])
613 return E_OUTOFMEMORY;
619 if (!instance->m_pStream)
623 GlobalFree(clipboard->hmem);
624 clipboard->hmem =
nullptr;
627 pMedium->u.hGlobal =
nullptr;
628 return E_OUTOFMEMORY;
631 else if (instance->m_pFormatEtc[idx].cfFormat == RegisterClipboardFormat(CFSTR_FILECONTENTS))
633 if ((pFormatEtc->lindex >= 0) && ((ULONG)pFormatEtc->lindex < instance->m_nStreams))
635 pMedium->u.pstm = instance->m_pStream[pFormatEtc->lindex];
636 IDataObject_AddRef(instance->m_pStream[pFormatEtc->lindex]);
647static HRESULT STDMETHODCALLTYPE CliprdrDataObject_GetDataHere(IDataObject* This,
648 FORMATETC* pformatetc,
657static HRESULT STDMETHODCALLTYPE CliprdrDataObject_QueryGetData(IDataObject* This,
658 FORMATETC* pformatetc)
660 CliprdrDataObject* instance = (CliprdrDataObject*)This;
665 if (cliprdr_lookup_format(instance, pformatetc) == -1)
666 return DV_E_FORMATETC;
671static HRESULT STDMETHODCALLTYPE CliprdrDataObject_GetCanonicalFormatEtc(IDataObject* This,
672 FORMATETC* pformatectIn,
673 FORMATETC* pformatetcOut)
681 pformatetcOut->ptd =
nullptr;
685static HRESULT STDMETHODCALLTYPE CliprdrDataObject_SetData(IDataObject* This, FORMATETC* pformatetc,
686 STGMEDIUM* pmedium, BOOL fRelease)
695static HRESULT STDMETHODCALLTYPE CliprdrDataObject_EnumFormatEtc(IDataObject* This,
697 IEnumFORMATETC** ppenumFormatEtc)
699 CliprdrDataObject* instance = (CliprdrDataObject*)This;
701 if (!instance || !ppenumFormatEtc)
704 if (dwDirection == DATADIR_GET)
706 *ppenumFormatEtc = (IEnumFORMATETC*)CliprdrEnumFORMATETC_New(instance->m_nNumFormats,
707 instance->m_pFormatEtc);
708 return (*ppenumFormatEtc) ? S_OK : E_OUTOFMEMORY;
716static HRESULT STDMETHODCALLTYPE CliprdrDataObject_DAdvise(IDataObject* This, FORMATETC* pformatetc,
717 DWORD advf, IAdviseSink* pAdvSink,
718 DWORD* pdwConnection)
725 return OLE_E_ADVISENOTSUPPORTED;
728static HRESULT STDMETHODCALLTYPE CliprdrDataObject_DUnadvise(IDataObject* This, DWORD dwConnection)
732 return OLE_E_ADVISENOTSUPPORTED;
735static HRESULT STDMETHODCALLTYPE CliprdrDataObject_EnumDAdvise(IDataObject* This,
736 IEnumSTATDATA** ppenumAdvise)
740 return OLE_E_ADVISENOTSUPPORTED;
743static CliprdrDataObject* CliprdrDataObject_New(FORMATETC* fmtetc, STGMEDIUM* stgmed, ULONG count,
746 CliprdrDataObject* instance;
747 IDataObject* iDataObject;
748 instance = (CliprdrDataObject*)calloc(1,
sizeof(CliprdrDataObject));
753 iDataObject = &instance->iDataObject;
754 iDataObject->lpVtbl = (IDataObjectVtbl*)calloc(1,
sizeof(IDataObjectVtbl));
756 if (!iDataObject->lpVtbl)
759 iDataObject->lpVtbl->QueryInterface = CliprdrDataObject_QueryInterface;
760 iDataObject->lpVtbl->AddRef = CliprdrDataObject_AddRef;
761 iDataObject->lpVtbl->Release = CliprdrDataObject_Release;
762 iDataObject->lpVtbl->GetData = CliprdrDataObject_GetData;
763 iDataObject->lpVtbl->GetDataHere = CliprdrDataObject_GetDataHere;
764 iDataObject->lpVtbl->QueryGetData = CliprdrDataObject_QueryGetData;
765 iDataObject->lpVtbl->GetCanonicalFormatEtc = CliprdrDataObject_GetCanonicalFormatEtc;
766 iDataObject->lpVtbl->SetData = CliprdrDataObject_SetData;
767 iDataObject->lpVtbl->EnumFormatEtc = CliprdrDataObject_EnumFormatEtc;
768 iDataObject->lpVtbl->DAdvise = CliprdrDataObject_DAdvise;
769 iDataObject->lpVtbl->DUnadvise = CliprdrDataObject_DUnadvise;
770 iDataObject->lpVtbl->EnumDAdvise = CliprdrDataObject_EnumDAdvise;
771 instance->m_lRefCount = 1;
772 instance->m_nNumFormats = count;
773 instance->m_pData = data;
774 instance->m_nStreams = 0;
775 instance->m_pStream =
nullptr;
779 instance->m_pFormatEtc = (FORMATETC*)calloc(count,
sizeof(FORMATETC));
781 if (!instance->m_pFormatEtc)
784 instance->m_pStgMedium = (STGMEDIUM*)calloc(count,
sizeof(STGMEDIUM));
786 if (!instance->m_pStgMedium)
789 for (ULONG i = 0; i < count; i++)
791 instance->m_pFormatEtc[i] = fmtetc[i];
792 instance->m_pStgMedium[i] = stgmed[i];
798 CliprdrDataObject_Delete(instance);
802void CliprdrDataObject_Delete(CliprdrDataObject* instance)
806 free(instance->iDataObject.lpVtbl);
807 free(instance->m_pFormatEtc);
808 free(instance->m_pStgMedium);
810 if (instance->m_pStream)
812 for (ULONG i = 0; i < instance->m_nStreams; i++)
813 CliprdrStream_Release(instance->m_pStream[i]);
815 free(instance->m_pStream);
822static BOOL wf_create_file_obj(wfClipboard* clipboard, IDataObject** ppDataObject)
824 FORMATETC fmtetc[2] = WINPR_C_ARRAY_INIT;
825 STGMEDIUM stgmeds[2] = WINPR_C_ARRAY_INIT;
830 fmtetc[0].cfFormat = RegisterClipboardFormat(CFSTR_FILEDESCRIPTORW);
831 fmtetc[0].dwAspect = DVASPECT_CONTENT;
833 fmtetc[0].tymed = TYMED_HGLOBAL;
834 stgmeds[0].tymed = TYMED_HGLOBAL;
836 fmtetc[1].cfFormat = RegisterClipboardFormat(CFSTR_FILECONTENTS);
837 fmtetc[1].dwAspect = DVASPECT_CONTENT;
839 fmtetc[1].tymed = TYMED_ISTREAM;
840 stgmeds[1].tymed = TYMED_ISTREAM;
842 *ppDataObject = (IDataObject*)CliprdrDataObject_New(fmtetc, stgmeds, 2, clipboard);
843 return (*ppDataObject) ? TRUE : FALSE;
846static void wf_destroy_file_obj(IDataObject* instance)
849 IDataObject_Release(instance);
856static void cliprdr_format_deep_copy(FORMATETC* dest, FORMATETC* source)
862 dest->ptd = (DVTARGETDEVICE*)CoTaskMemAlloc(
sizeof(DVTARGETDEVICE));
865 *(dest->ptd) = *(source->ptd);
869static HRESULT STDMETHODCALLTYPE CliprdrEnumFORMATETC_QueryInterface(IEnumFORMATETC* This,
870 REFIID riid,
void** ppvObject)
874 if (IsEqualIID(riid, &IID_IEnumFORMATETC) || IsEqualIID(riid, &IID_IUnknown))
876 IEnumFORMATETC_AddRef(This);
883 return E_NOINTERFACE;
887static ULONG STDMETHODCALLTYPE CliprdrEnumFORMATETC_AddRef(IEnumFORMATETC* This)
889 CliprdrEnumFORMATETC* instance = (CliprdrEnumFORMATETC*)This;
894 return InterlockedIncrement(&instance->m_lRefCount);
897static ULONG STDMETHODCALLTYPE CliprdrEnumFORMATETC_Release(IEnumFORMATETC* This)
900 CliprdrEnumFORMATETC* instance = (CliprdrEnumFORMATETC*)This;
905 count = InterlockedDecrement(&instance->m_lRefCount);
909 CliprdrEnumFORMATETC_Delete(instance);
918static HRESULT STDMETHODCALLTYPE CliprdrEnumFORMATETC_Next(IEnumFORMATETC* This, ULONG celt,
919 FORMATETC* rgelt, ULONG* pceltFetched)
922 CliprdrEnumFORMATETC* instance = (CliprdrEnumFORMATETC*)This;
924 if (!instance || !celt || !rgelt)
927 while ((instance->m_nIndex < instance->m_nNumFormats) && (copied < celt))
929 cliprdr_format_deep_copy(&rgelt[copied++], &instance->m_pFormatEtc[instance->m_nIndex++]);
932 if (pceltFetched != 0)
933 *pceltFetched = copied;
935 return (copied == celt) ? S_OK : E_FAIL;
938static HRESULT STDMETHODCALLTYPE CliprdrEnumFORMATETC_Skip(IEnumFORMATETC* This, ULONG celt)
940 CliprdrEnumFORMATETC* instance = (CliprdrEnumFORMATETC*)This;
945 if (instance->m_nIndex + (LONG)celt > instance->m_nNumFormats)
948 instance->m_nIndex += celt;
952static HRESULT STDMETHODCALLTYPE CliprdrEnumFORMATETC_Reset(IEnumFORMATETC* This)
954 CliprdrEnumFORMATETC* instance = (CliprdrEnumFORMATETC*)This;
959 instance->m_nIndex = 0;
963static HRESULT STDMETHODCALLTYPE CliprdrEnumFORMATETC_Clone(IEnumFORMATETC* This,
964 IEnumFORMATETC** ppEnum)
966 CliprdrEnumFORMATETC* instance = (CliprdrEnumFORMATETC*)This;
968 if (!instance || !ppEnum)
972 (IEnumFORMATETC*)CliprdrEnumFORMATETC_New(instance->m_nNumFormats, instance->m_pFormatEtc);
975 return E_OUTOFMEMORY;
977 ((CliprdrEnumFORMATETC*)*ppEnum)->m_nIndex = instance->m_nIndex;
981CliprdrEnumFORMATETC* CliprdrEnumFORMATETC_New(ULONG nFormats, FORMATETC* pFormatEtc)
983 CliprdrEnumFORMATETC* instance;
984 IEnumFORMATETC* iEnumFORMATETC;
986 if ((nFormats != 0) && !pFormatEtc)
989 instance = (CliprdrEnumFORMATETC*)calloc(1,
sizeof(CliprdrEnumFORMATETC));
994 iEnumFORMATETC = &instance->iEnumFORMATETC;
995 iEnumFORMATETC->lpVtbl = (IEnumFORMATETCVtbl*)calloc(1,
sizeof(IEnumFORMATETCVtbl));
997 if (!iEnumFORMATETC->lpVtbl)
1000 iEnumFORMATETC->lpVtbl->QueryInterface = CliprdrEnumFORMATETC_QueryInterface;
1001 iEnumFORMATETC->lpVtbl->AddRef = CliprdrEnumFORMATETC_AddRef;
1002 iEnumFORMATETC->lpVtbl->Release = CliprdrEnumFORMATETC_Release;
1003 iEnumFORMATETC->lpVtbl->Next = CliprdrEnumFORMATETC_Next;
1004 iEnumFORMATETC->lpVtbl->Skip = CliprdrEnumFORMATETC_Skip;
1005 iEnumFORMATETC->lpVtbl->Reset = CliprdrEnumFORMATETC_Reset;
1006 iEnumFORMATETC->lpVtbl->Clone = CliprdrEnumFORMATETC_Clone;
1007 instance->m_lRefCount = 1;
1008 instance->m_nIndex = 0;
1009 instance->m_nNumFormats = nFormats;
1013 instance->m_pFormatEtc = (FORMATETC*)calloc(nFormats,
sizeof(FORMATETC));
1015 if (!instance->m_pFormatEtc)
1018 for (ULONG i = 0; i < nFormats; i++)
1019 cliprdr_format_deep_copy(&instance->m_pFormatEtc[i], &pFormatEtc[i]);
1024 CliprdrEnumFORMATETC_Delete(instance);
1028void CliprdrEnumFORMATETC_Delete(CliprdrEnumFORMATETC* instance)
1032 free(instance->iEnumFORMATETC.lpVtbl);
1034 if (instance->m_pFormatEtc)
1036 for (LONG i = 0; i < instance->m_nNumFormats; i++)
1038 if (instance->m_pFormatEtc[i].ptd)
1039 CoTaskMemFree(instance->m_pFormatEtc[i].ptd);
1042 free(instance->m_pFormatEtc);
1051static UINT32 get_local_format_id_by_name(wfClipboard* clipboard,
const TCHAR* format_name)
1054 WCHAR* unicode_name;
1055#if !defined(UNICODE)
1059 if (!clipboard || !format_name)
1063 unicode_name = _wcsdup(format_name);
1065 size = _tcslen(format_name);
1066 unicode_name = calloc(size + 1,
sizeof(WCHAR));
1071 MultiByteToWideChar(CP_OEMCP, 0, format_name, strlen(format_name), unicode_name, size);
1077 for (
size_t i = 0; i < clipboard->map_size; i++)
1079 map = &clipboard->format_mappings[i];
1083 if (wcscmp(map->name, unicode_name) == 0)
1086 return map->local_format_id;
1095static inline BOOL file_transferring(wfClipboard* clipboard)
1097 return get_local_format_id_by_name(clipboard, CFSTR_FILEDESCRIPTORW) ? TRUE : FALSE;
1100static UINT32 get_remote_format_id(wfClipboard* clipboard, UINT32 local_format)
1107 for (UINT32 i = 0; i < clipboard->map_size; i++)
1109 map = &clipboard->format_mappings[i];
1111 if (map->local_format_id == local_format)
1112 return map->remote_format_id;
1115 return local_format;
1118static void map_ensure_capacity(wfClipboard* clipboard)
1123 if (clipboard->map_size >= clipboard->map_capacity)
1125 size_t new_size = clipboard->map_capacity;
1128 WINPR_ASSERT(new_size <= SIZE_MAX - 128ull);
1130 }
while (new_size <= clipboard->map_size);
1131 formatMapping* new_map =
1132 (formatMapping*)realloc(clipboard->format_mappings,
sizeof(formatMapping) * new_size);
1137 clipboard->format_mappings = new_map;
1138 clipboard->map_capacity = new_size;
1142static BOOL clear_format_map(wfClipboard* clipboard)
1149 if (clipboard->format_mappings)
1151 for (
size_t i = 0; i < clipboard->map_capacity; i++)
1153 map = &clipboard->format_mappings[i];
1154 map->remote_format_id = 0;
1155 map->local_format_id = 0;
1157 map->name =
nullptr;
1161 clipboard->map_size = 0;
1165static UINT cliprdr_send_tempdir(wfClipboard* clipboard)
1172 if (GetEnvironmentVariableA(
"TEMP", tempDirectory.szTempDir,
sizeof(tempDirectory.szTempDir)) ==
1176 return clipboard->context->TempDirectory(clipboard->context, &tempDirectory);
1179static BOOL cliprdr_GetUpdatedClipboardFormats(wfClipboard* clipboard, PUINT lpuiFormats,
1180 UINT cFormats, PUINT pcFormatsOut)
1184 BOOL clipboardOpen = FALSE;
1186 if (!clipboard->legacyApi)
1187 return clipboard->GetUpdatedClipboardFormats(lpuiFormats, cFormats, pcFormatsOut);
1189 clipboardOpen = try_open_clipboard(clipboard->hwnd);
1197 while (index < cFormats)
1199 format = EnumClipboardFormats(format);
1204 lpuiFormats[index] = format;
1208 *pcFormatsOut = index;
1213static UINT cliprdr_send_format_list(wfClipboard* clipboard)
1217 UINT32 numFormats = 0;
1218 UINT32 formatId = 0;
1219 char formatName[1024];
1224 return ERROR_INTERNAL_ERROR;
1227 if (try_open_clipboard(clipboard->hwnd))
1229 count = CountClipboardFormats();
1230 numFormats = (UINT32)count;
1236 return CHANNEL_RC_NO_MEMORY;
1242 if (IsClipboardFormatAvailable(CF_HDROP))
1244 formats[index++].formatId = RegisterClipboardFormat(CFSTR_FILEDESCRIPTORW);
1245 formats[index++].formatId = RegisterClipboardFormat(CFSTR_FILECONTENTS);
1249 while ((formatId = EnumClipboardFormats(formatId)) != 0)
1250 formats[index++].formatId = formatId;
1256 if (!CloseClipboard())
1259 return ERROR_INTERNAL_ERROR;
1262 for (UINT index = 0; index < numFormats; index++)
1264 if (GetClipboardFormatNameA(formats[index].formatId, formatName,
sizeof(formatName)))
1266 formats[index].formatName = _strdup(formatName);
1271 formatList.numFormats = numFormats;
1272 formatList.formats = formats;
1273 formatList.common.msgType = CB_FORMAT_LIST;
1274 rc = clipboard->context->ClientFormatList(clipboard->context, &formatList);
1276 for (UINT index = 0; index < numFormats; index++)
1277 free(formats[index].formatName);
1283static UINT cliprdr_send_data_request(wfClipboard* clipboard, UINT32 formatId)
1286 UINT32 remoteFormatId;
1289 if (!clipboard || !clipboard->context || !clipboard->context->ClientFormatDataRequest)
1290 return ERROR_INTERNAL_ERROR;
1292 remoteFormatId = get_remote_format_id(clipboard, formatId);
1294 formatDataRequest.requestedFormatId = remoteFormatId;
1295 clipboard->requestedFormatId = formatId;
1296 rc = clipboard->context->ClientFormatDataRequest(clipboard->context, &formatDataRequest);
1298 if (WaitForSingleObject(clipboard->response_data_event, INFINITE) != WAIT_OBJECT_0)
1299 rc = ERROR_INTERNAL_ERROR;
1300 else if (!ResetEvent(clipboard->response_data_event))
1301 rc = ERROR_INTERNAL_ERROR;
1306UINT cliprdr_send_request_filecontents(wfClipboard* clipboard,
const void* streamid, ULONG index,
1307 UINT32 flag, UINT64 position, ULONG nreq)
1312 if (!clipboard || !clipboard->context || !clipboard->context->ClientFileContentsRequest)
1313 return ERROR_INTERNAL_ERROR;
1315 fileContentsRequest.streamId = (UINT32)(ULONG_PTR)streamid;
1316 fileContentsRequest.listIndex = index;
1317 fileContentsRequest.dwFlags = flag;
1318 fileContentsRequest.nPositionLow = position & 0xFFFFFFFF;
1319 fileContentsRequest.nPositionHigh = (position >> 32) & 0xFFFFFFFF;
1320 fileContentsRequest.cbRequested = nreq;
1321 fileContentsRequest.clipDataId = 0;
1322 fileContentsRequest.common.msgFlags = 0;
1323 rc = clipboard->context->ClientFileContentsRequest(clipboard->context, &fileContentsRequest);
1325 if (WaitForSingleObject(clipboard->req_fevent, INFINITE) != WAIT_OBJECT_0)
1326 rc = ERROR_INTERNAL_ERROR;
1327 else if (!ResetEvent(clipboard->req_fevent))
1328 rc = ERROR_INTERNAL_ERROR;
1333static UINT cliprdr_send_response_filecontents(wfClipboard* clipboard, UINT32 streamId, UINT32 size,
1338 if (!clipboard || !clipboard->context || !clipboard->context->ClientFileContentsResponse)
1339 return ERROR_INTERNAL_ERROR;
1341 fileContentsResponse.streamId = streamId;
1342 fileContentsResponse.cbRequested = size;
1343 fileContentsResponse.requestedData = data;
1344 fileContentsResponse.common.msgFlags = CB_RESPONSE_OK;
1345 return clipboard->context->ClientFileContentsResponse(clipboard->context,
1346 &fileContentsResponse);
1349static LRESULT CALLBACK cliprdr_proc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
1351 static wfClipboard* clipboard =
nullptr;
1356 DEBUG_CLIPRDR(
"info: WM_CREATE");
1357 clipboard = (wfClipboard*)((CREATESTRUCT*)lParam)->lpCreateParams;
1358 clipboard->hwnd = hWnd;
1360 if (!clipboard->legacyApi)
1361 clipboard->AddClipboardFormatListener(hWnd);
1363 clipboard->hWndNextViewer = SetClipboardViewer(hWnd);
1368 DEBUG_CLIPRDR(
"info: WM_CLOSE");
1370 if (!clipboard->legacyApi)
1371 clipboard->RemoveClipboardFormatListener(hWnd);
1376 if (clipboard->legacyApi)
1377 ChangeClipboardChain(hWnd, clipboard->hWndNextViewer);
1381 case WM_CLIPBOARDUPDATE:
1382 DEBUG_CLIPRDR(
"info: WM_CLIPBOARDUPDATE");
1384 if (clipboard->sync)
1386 if ((GetClipboardOwner() != clipboard->hwnd) &&
1387 (S_FALSE == OleIsCurrentClipboard(clipboard->data_obj)))
1389 if (clipboard->hmem)
1391 GlobalFree(clipboard->hmem);
1392 clipboard->hmem =
nullptr;
1395 cliprdr_send_format_list(clipboard);
1401 case WM_RENDERALLFORMATS:
1402 DEBUG_CLIPRDR(
"info: WM_RENDERALLFORMATS");
1405 if (!try_open_clipboard(clipboard->hwnd))
1407 DEBUG_CLIPRDR(
"OpenClipboard failed with 0x%x", GetLastError());
1415 case WM_RENDERFORMAT:
1416 DEBUG_CLIPRDR(
"info: WM_RENDERFORMAT");
1418 if (cliprdr_send_data_request(clipboard, (UINT32)wParam) != 0)
1420 DEBUG_CLIPRDR(
"error: cliprdr_send_data_request failed.");
1424 if (!SetClipboardData((UINT)wParam, clipboard->hmem))
1426 DEBUG_CLIPRDR(
"SetClipboardData failed with 0x%x", GetLastError());
1428 if (clipboard->hmem)
1430 GlobalFree(clipboard->hmem);
1431 clipboard->hmem =
nullptr;
1438 case WM_DRAWCLIPBOARD:
1439 if (clipboard->legacyApi)
1441 if ((GetClipboardOwner() != clipboard->hwnd) &&
1442 (S_FALSE == OleIsCurrentClipboard(clipboard->data_obj)))
1444 cliprdr_send_format_list(clipboard);
1447 SendMessage(clipboard->hWndNextViewer, Msg, wParam, lParam);
1452 case WM_CHANGECBCHAIN:
1453 if (clipboard->legacyApi)
1455 HWND hWndCurrViewer = (HWND)wParam;
1456 HWND hWndNextViewer = (HWND)lParam;
1458 if (hWndCurrViewer == clipboard->hWndNextViewer)
1459 clipboard->hWndNextViewer = hWndNextViewer;
1460 else if (clipboard->hWndNextViewer)
1461 SendMessage(clipboard->hWndNextViewer, Msg, wParam, lParam);
1466 case WM_CLIPRDR_MESSAGE:
1467 DEBUG_CLIPRDR(
"info: WM_CLIPRDR_MESSAGE");
1471 case OLE_SETCLIPBOARD:
1472 DEBUG_CLIPRDR(
"info: OLE_SETCLIPBOARD");
1474 if (wf_create_file_obj(clipboard, &clipboard->data_obj))
1476 if (OleSetClipboard(clipboard->data_obj) != S_OK)
1478 wf_destroy_file_obj(clipboard->data_obj);
1479 clipboard->data_obj =
nullptr;
1491 case WM_DESTROYCLIPBOARD:
1492 case WM_ASKCBFORMATNAME:
1493 case WM_HSCROLLCLIPBOARD:
1494 case WM_PAINTCLIPBOARD:
1495 case WM_SIZECLIPBOARD:
1496 case WM_VSCROLLCLIPBOARD:
1498 return DefWindowProc(hWnd, Msg, wParam, lParam);
1504static int create_cliprdr_window(wfClipboard* clipboard)
1506 WNDCLASSEX wnd_cls = WINPR_C_ARRAY_INIT;
1508 wnd_cls.cbSize =
sizeof(WNDCLASSEX);
1509 wnd_cls.style = CS_OWNDC;
1510 wnd_cls.lpfnWndProc = cliprdr_proc;
1511 wnd_cls.cbClsExtra = 0;
1512 wnd_cls.cbWndExtra = 0;
1513 wnd_cls.hIcon =
nullptr;
1514 wnd_cls.hCursor =
nullptr;
1515 wnd_cls.hbrBackground =
nullptr;
1516 wnd_cls.lpszMenuName =
nullptr;
1517 wnd_cls.lpszClassName = _T(
"ClipboardHiddenMessageProcessor");
1518 wnd_cls.hInstance = GetModuleHandle(
nullptr);
1519 wnd_cls.hIconSm =
nullptr;
1520 RegisterClassEx(&wnd_cls);
1522 CreateWindowEx(WS_EX_LEFT, _T(
"ClipboardHiddenMessageProcessor"), _T(
"rdpclip"), 0, 0, 0, 0,
1523 0, HWND_MESSAGE,
nullptr, GetModuleHandle(
nullptr), clipboard);
1525 if (!clipboard->hwnd)
1527 DEBUG_CLIPRDR(
"error: CreateWindowEx failed with %x.", GetLastError());
1534static DWORD WINAPI cliprdr_thread_func(LPVOID arg)
1539 wfClipboard* clipboard = (wfClipboard*)arg;
1542 if ((ret = create_cliprdr_window(clipboard)) != 0)
1545 DEBUG_CLIPRDR(
"error: create clipboard window failed.");
1549 while ((mcode = GetMessage(&msg, 0, 0, 0)) != 0)
1553 DEBUG_CLIPRDR(
"error: clipboard thread GetMessage failed.");
1558 TranslateMessage(&msg);
1559 DispatchMessage(&msg);
1567static void clear_file_array(wfClipboard* clipboard)
1573 if (clipboard->file_names)
1575 for (
size_t i = 0; i < clipboard->nFiles; i++)
1577 free(clipboard->file_names[i]);
1578 clipboard->file_names[i] =
nullptr;
1581 free(clipboard->file_names);
1582 clipboard->file_names =
nullptr;
1586 if (clipboard->fileDescriptor)
1588 for (
size_t i = 0; i < clipboard->nFiles; i++)
1590 free(clipboard->fileDescriptor[i]);
1591 clipboard->fileDescriptor[i] =
nullptr;
1594 free(clipboard->fileDescriptor);
1595 clipboard->fileDescriptor =
nullptr;
1598 clipboard->file_array_size = 0;
1599 clipboard->nFiles = 0;
1602static BOOL wf_cliprdr_get_file_contents(WCHAR* file_name, BYTE* buffer, LONG positionLow,
1603 LONG positionHigh, DWORD nRequested, DWORD* puSize)
1609 if (!file_name || !buffer || !puSize)
1611 WLog_ERR(TAG,
"get file contents Invalid Arguments.");
1615 hFile = CreateFileW(file_name, GENERIC_READ, FILE_SHARE_READ,
nullptr, OPEN_EXISTING,
1616 FILE_ATTRIBUTE_NORMAL | FILE_FLAG_BACKUP_SEMANTICS,
nullptr);
1618 if (hFile == INVALID_HANDLE_VALUE)
1621 rc = SetFilePointer(hFile, positionLow, &positionHigh, FILE_BEGIN);
1623 if (rc == INVALID_SET_FILE_POINTER)
1626 if (!ReadFile(hFile, buffer, nRequested, &nGet,
nullptr))
1628 DEBUG_CLIPRDR(
"ReadFile failed with 0x%08lX.", GetLastError());
1635 if (!CloseHandle(hFile))
1645static FILEDESCRIPTORW* wf_cliprdr_get_file_descriptor(WCHAR* file_name,
size_t pathLen)
1654 hFile = CreateFileW(file_name, GENERIC_READ, FILE_SHARE_READ,
nullptr, OPEN_EXISTING,
1655 FILE_ATTRIBUTE_NORMAL | FILE_FLAG_BACKUP_SEMANTICS,
nullptr);
1657 if (hFile == INVALID_HANDLE_VALUE)
1663 fd->dwFlags = FD_ATTRIBUTES | FD_FILESIZE | FD_WRITESTIME | FD_PROGRESSUI;
1664 fd->dwFileAttributes = GetFileAttributes(file_name);
1666 if (!GetFileTime(hFile,
nullptr,
nullptr, &fd->ftLastWriteTime))
1668 fd->dwFlags &= ~FD_WRITESTIME;
1671 fd->nFileSizeLow = GetFileSize(hFile, &fd->nFileSizeHigh);
1672 wcscpy_s(fd->cFileName,
sizeof(fd->cFileName) / 2, file_name + pathLen);
1673 (void)CloseHandle(hFile);
1677static BOOL wf_cliprdr_array_ensure_capacity(wfClipboard* clipboard)
1682 if (clipboard->nFiles == clipboard->file_array_size)
1687 new_size = (clipboard->file_array_size + 1) * 2;
1692 clipboard->fileDescriptor = new_fd;
1694 new_name = (WCHAR**)realloc(clipboard->file_names, new_size *
sizeof(WCHAR*));
1697 clipboard->file_names = new_name;
1699 if (!new_fd || !new_name)
1702 clipboard->file_array_size = new_size;
1708static BOOL wf_cliprdr_add_to_file_arrays(wfClipboard* clipboard, WCHAR* full_file_name,
1711 if (!wf_cliprdr_array_ensure_capacity(clipboard))
1715 clipboard->file_names[clipboard->nFiles] = (LPWSTR)malloc(MAX_PATH * 2);
1717 if (!clipboard->file_names[clipboard->nFiles])
1720 wcscpy_s(clipboard->file_names[clipboard->nFiles], MAX_PATH, full_file_name);
1722 clipboard->fileDescriptor[clipboard->nFiles] =
1723 wf_cliprdr_get_file_descriptor(full_file_name, pathLen);
1725 if (!clipboard->fileDescriptor[clipboard->nFiles])
1727 free(clipboard->file_names[clipboard->nFiles]);
1731 clipboard->nFiles++;
1735static BOOL wf_cliprdr_traverse_directory(wfClipboard* clipboard, WCHAR* Dir,
size_t pathLen)
1738 WCHAR DirSpec[MAX_PATH];
1739 WIN32_FIND_DATA FindFileData;
1741 if (!clipboard || !Dir)
1744 StringCchCopy(DirSpec, MAX_PATH, Dir);
1745 StringCchCat(DirSpec, MAX_PATH, TEXT(
"\\*"));
1746 hFind = FindFirstFile(DirSpec, &FindFileData);
1748 if (hFind == INVALID_HANDLE_VALUE)
1750 DEBUG_CLIPRDR(
"FindFirstFile failed with 0x%x.", GetLastError());
1754 while (FindNextFile(hFind, &FindFileData))
1756 if ((((FindFileData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0) &&
1757 (wcscmp(FindFileData.cFileName, _T(
".")) == 0)) ||
1758 (wcscmp(FindFileData.cFileName, _T(
"..")) == 0))
1763 if ((FindFileData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0)
1765 WCHAR DirAdd[MAX_PATH];
1766 StringCchCopy(DirAdd, MAX_PATH, Dir);
1767 StringCchCat(DirAdd, MAX_PATH, _T(
"\\"));
1768 StringCchCat(DirAdd, MAX_PATH, FindFileData.cFileName);
1770 if (!wf_cliprdr_add_to_file_arrays(clipboard, DirAdd, pathLen))
1773 if (!wf_cliprdr_traverse_directory(clipboard, DirAdd, pathLen))
1778 WCHAR fileName[MAX_PATH];
1779 StringCchCopy(fileName, MAX_PATH, Dir);
1780 StringCchCat(fileName, MAX_PATH, _T(
"\\"));
1781 StringCchCat(fileName, MAX_PATH, FindFileData.cFileName);
1783 if (!wf_cliprdr_add_to_file_arrays(clipboard, fileName, pathLen))
1792static UINT wf_cliprdr_send_client_capabilities(wfClipboard* clipboard)
1797 if (!clipboard || !clipboard->context || !clipboard->context->ClientCapabilities)
1798 return ERROR_INTERNAL_ERROR;
1800 capabilities.cCapabilitiesSets = 1;
1802 generalCapabilitySet.capabilitySetType = CB_CAPSTYPE_GENERAL;
1803 generalCapabilitySet.capabilitySetLength = 12;
1804 generalCapabilitySet.version = CB_CAPS_VERSION_2;
1805 generalCapabilitySet.generalFlags =
1806 CB_USE_LONG_FORMAT_NAMES | CB_STREAM_FILECLIP_ENABLED | CB_FILECLIP_NO_FILE_PATHS;
1807 return clipboard->context->ClientCapabilities(clipboard->context, &capabilities);
1815static UINT wf_cliprdr_monitor_ready(CliprdrClientContext* context,
1819 wfClipboard* clipboard = (wfClipboard*)context->custom;
1821 if (!context || !monitorReady)
1822 return ERROR_INTERNAL_ERROR;
1824 clipboard->sync = TRUE;
1825 rc = wf_cliprdr_send_client_capabilities(clipboard);
1827 if (rc != CHANNEL_RC_OK)
1830 return cliprdr_send_format_list(clipboard);
1838static UINT wf_cliprdr_server_capabilities(CliprdrClientContext* context,
1842 wfClipboard* clipboard = (wfClipboard*)context->custom;
1844 if (!context || !capabilities)
1845 return ERROR_INTERNAL_ERROR;
1847 for (UINT32 index = 0; index < capabilities->cCapabilitiesSets; index++)
1849 capabilitySet = &(capabilities->capabilitySets[index]);
1851 if ((capabilitySet->capabilitySetType == CB_CAPSTYPE_GENERAL) &&
1852 (capabilitySet->capabilitySetLength >= CB_CAPSTYPE_GENERAL_LEN))
1856 clipboard->capabilities = generalCapabilitySet->generalFlags;
1861 return CHANNEL_RC_OK;
1869static UINT wf_cliprdr_server_format_list(CliprdrClientContext* context,
1872 UINT rc = ERROR_INTERNAL_ERROR;
1873 formatMapping* mapping;
1875 wfClipboard* clipboard = (wfClipboard*)context->custom;
1877 if (!clear_format_map(clipboard))
1878 return ERROR_INTERNAL_ERROR;
1880 for (UINT32 i = 0; i < formatList->numFormats; i++)
1882 format = &(formatList->formats[i]);
1883 mapping = &(clipboard->format_mappings[i]);
1884 mapping->remote_format_id = format->formatId;
1886 if (format->formatName)
1888 mapping->name = ConvertUtf8ToWCharAlloc(format->formatName,
nullptr);
1891 mapping->local_format_id = RegisterClipboardFormatW((LPWSTR)mapping->name);
1895 mapping->name =
nullptr;
1896 mapping->local_format_id = mapping->remote_format_id;
1899 clipboard->map_size++;
1900 map_ensure_capacity(clipboard);
1903 if (file_transferring(clipboard))
1905 if (PostMessage(clipboard->hwnd, WM_CLIPRDR_MESSAGE, OLE_SETCLIPBOARD, 0))
1910 if (!try_open_clipboard(clipboard->hwnd))
1911 return CHANNEL_RC_OK;
1913 if (EmptyClipboard())
1915 for (UINT32 i = 0; i < (UINT32)clipboard->map_size; i++)
1916 SetClipboardData(clipboard->format_mappings[i].local_format_id,
nullptr);
1921 if (!CloseClipboard() && GetLastError())
1922 return ERROR_INTERNAL_ERROR;
1934wf_cliprdr_server_format_list_response(CliprdrClientContext* context,
1938 (void)formatListResponse;
1940 if (formatListResponse->common.msgFlags != CB_RESPONSE_OK)
1941 WLog_WARN(TAG,
"format list update failed");
1943 return CHANNEL_RC_OK;
1952wf_cliprdr_server_lock_clipboard_data(CliprdrClientContext* context,
1956 (void)lockClipboardData;
1957 return CHANNEL_RC_OK;
1966wf_cliprdr_server_unlock_clipboard_data(CliprdrClientContext* context,
1970 (void)unlockClipboardData;
1971 return CHANNEL_RC_OK;
1974static BOOL wf_cliprdr_process_filename(wfClipboard* clipboard, WCHAR* wFileName,
size_t str_len)
1977 size_t offset = str_len;
1979 if (!clipboard || !wFileName)
1985 if (wFileName[offset] == L
'\\')
1991 pathLen = offset + 1;
1993 if (!wf_cliprdr_add_to_file_arrays(clipboard, wFileName, pathLen))
1996 if ((clipboard->fileDescriptor[clipboard->nFiles - 1]->dwFileAttributes &
1997 FILE_ATTRIBUTE_DIRECTORY) != 0)
2000 if (!wf_cliprdr_traverse_directory(clipboard, wFileName, pathLen))
2007static SSIZE_T wf_cliprdr_tryopen(wfClipboard* clipboard, UINT32 requestedFormatId, BYTE** pData)
2010 WINPR_ASSERT(clipboard);
2011 WINPR_ASSERT(pData);
2016 if (!try_open_clipboard(clipboard->hwnd))
2019 HANDLE hClipdata = GetClipboardData(requestedFormatId);
2024 char* globlemem = (
char*)GlobalLock(hClipdata);
2025 const SSIZE_T size = GlobalSize(hClipdata);
2029 BYTE* buff = malloc(size);
2030 if (buff ==
nullptr)
2032 CopyMemory(buff, globlemem, size);
2037 GlobalUnlock(hClipdata);
2045static SSIZE_T wf_cliprdr_get_filedescriptor(wfClipboard* clipboard, BYTE** pData)
2047 WINPR_ASSERT(clipboard);
2048 WINPR_ASSERT(pData);
2051 LPDATAOBJECT dataObj =
nullptr;
2052 FORMATETC format_etc = WINPR_C_ARRAY_INIT;
2053 STGMEDIUM stg_medium = WINPR_C_ARRAY_INIT;
2057 HRESULT result = OleGetClipboard(&dataObj);
2062 format_etc.cfFormat = CF_HDROP;
2063 format_etc.tymed = TYMED_HGLOBAL;
2064 format_etc.dwAspect = 1;
2065 format_etc.lindex = -1;
2066 result = IDataObject_GetData(dataObj, &format_etc, &stg_medium);
2070 DEBUG_CLIPRDR(
"dataObj->GetData failed.");
2074 HGLOBAL hdl = stg_medium.u.hGlobal;
2075 DROPFILES* dropFiles = (DROPFILES*)GlobalLock(hdl);
2079 ReleaseStgMedium(&stg_medium);
2080 clipboard->nFiles = 0;
2084 clear_file_array(clipboard);
2086 if (dropFiles->fWide)
2090 for (WCHAR* wFileName = (WCHAR*)((
char*)dropFiles + dropFiles->pFiles);
2091 (len = wcslen(wFileName)) > 0; wFileName += len + 1)
2093 wf_cliprdr_process_filename(clipboard, wFileName, wcslen(wFileName));
2099 for (
char* p = (
char*)((
char*)dropFiles + dropFiles->pFiles); (len = strlen(p)) > 0;
2100 p += len + 1, clipboard->nFiles++)
2102 const int ilen = WINPR_ASSERTING_INT_CAST(
int, len);
2103 const int cchWideChar = MultiByteToWideChar(CP_ACP, MB_COMPOSITE, p, ilen,
nullptr, 0);
2104 WCHAR* wFileName = (LPWSTR)calloc(cchWideChar,
sizeof(WCHAR));
2105 MultiByteToWideChar(CP_ACP, MB_COMPOSITE, p, ilen, wFileName, cchWideChar);
2106 wf_cliprdr_process_filename(clipboard, wFileName, cchWideChar);
2112 ReleaseStgMedium(&stg_medium);
2115 const size_t size = 4ull + clipboard->nFiles *
sizeof(
FILEDESCRIPTORW);
2116 FILEGROUPDESCRIPTORW* groupDsc = (FILEGROUPDESCRIPTORW*)calloc(size, 1);
2120 groupDsc->cItems = WINPR_ASSERTING_INT_CAST(UINT, clipboard->nFiles);
2122 for (
size_t i = 0; i < clipboard->nFiles; i++)
2124 if (clipboard->fileDescriptor[i])
2125 groupDsc->fgd[i] = *clipboard->fileDescriptor[i];
2128 *pData = (BYTE*)groupDsc;
2133 IDataObject_Release(dataObj);
2143wf_cliprdr_server_format_data_request(CliprdrClientContext* context,
2146 if (!context || !formatDataRequest)
2147 return ERROR_INTERNAL_ERROR;
2149 wfClipboard* clipboard = (wfClipboard*)context->custom;
2152 return ERROR_INTERNAL_ERROR;
2154 const UINT32 requestedFormatId = formatDataRequest->requestedFormatId;
2155 BYTE* requestedFormatData =
nullptr;
2157 if (requestedFormatId == RegisterClipboardFormat(CFSTR_FILEDESCRIPTORW))
2159 res = wf_cliprdr_get_filedescriptor(clipboard, &requestedFormatData);
2163 res = wf_cliprdr_tryopen(clipboard, requestedFormatId, &requestedFormatData);
2166 UINT rc = ERROR_INTERNAL_ERROR;
2170 .common = { .msgType = CB_FORMAT_DATA_RESPONSE,
2171 .msgFlags = CB_RESPONSE_OK,
2172 .dataLen = (uint32_t)res },
2173 .requestedFormatData = requestedFormatData
2176 rc = clipboard->context->ClientFormatDataResponse(clipboard->context, &response);
2181 CB_FORMAT_DATA_RESPONSE,
2182 .msgFlags = CB_RESPONSE_FAIL,
2184 .requestedFormatData =
nullptr };
2186 rc = clipboard->context->ClientFormatDataResponse(clipboard->context, &response);
2189 free(requestedFormatData);
2199wf_cliprdr_server_format_data_response(CliprdrClientContext* context,
2204 wfClipboard* clipboard;
2206 if (!context || !formatDataResponse)
2207 return ERROR_INTERNAL_ERROR;
2209 clipboard = (wfClipboard*)context->custom;
2212 return ERROR_INTERNAL_ERROR;
2214 if (formatDataResponse->common.msgFlags != CB_RESPONSE_OK)
2216 clipboard->hmem =
nullptr;
2218 if (!SetEvent(clipboard->response_data_event))
2219 return ERROR_INTERNAL_ERROR;
2221 return CHANNEL_RC_OK;
2224 hMem = GlobalAlloc(GMEM_MOVEABLE, formatDataResponse->common.dataLen);
2227 return ERROR_INTERNAL_ERROR;
2229 data = (BYTE*)GlobalLock(hMem);
2234 return ERROR_INTERNAL_ERROR;
2237 CopyMemory(data, formatDataResponse->requestedFormatData, formatDataResponse->common.dataLen);
2239 if (!GlobalUnlock(hMem) && GetLastError())
2242 return ERROR_INTERNAL_ERROR;
2245 clipboard->hmem = hMem;
2247 if (!SetEvent(clipboard->response_data_event))
2248 return ERROR_INTERNAL_ERROR;
2250 return CHANNEL_RC_OK;
2259wf_cliprdr_server_file_contents_request(CliprdrClientContext* context,
2263 BYTE* pData =
nullptr;
2264 HRESULT hRet = S_OK;
2265 FORMATETC vFormatEtc = WINPR_C_ARRAY_INIT;
2266 LPDATAOBJECT pDataObj =
nullptr;
2267 STGMEDIUM vStgMedium = WINPR_C_ARRAY_INIT;
2268 BOOL bIsStreamFile = TRUE;
2269 static LPSTREAM pStreamStc =
nullptr;
2270 static UINT32 uStreamIdStc = 0;
2271 wfClipboard* clipboard;
2272 UINT rc = ERROR_INTERNAL_ERROR;
2276 if (!context || !fileContentsRequest)
2277 return ERROR_INTERNAL_ERROR;
2279 clipboard = (wfClipboard*)context->custom;
2282 return ERROR_INTERNAL_ERROR;
2284 cbRequested = fileContentsRequest->cbRequested;
2285 if (fileContentsRequest->dwFlags == FILECONTENTS_SIZE)
2286 cbRequested =
sizeof(UINT64);
2288 pData = (BYTE*)calloc(1, cbRequested);
2293 hRet = OleGetClipboard(&pDataObj);
2297 WLog_ERR(TAG,
"filecontents: get ole clipboard failed.");
2301 vFormatEtc.cfFormat = RegisterClipboardFormat(CFSTR_FILECONTENTS);
2302 vFormatEtc.tymed = TYMED_ISTREAM;
2303 vFormatEtc.dwAspect = 1;
2304 vFormatEtc.lindex = fileContentsRequest->listIndex;
2305 vFormatEtc.ptd =
nullptr;
2307 if ((uStreamIdStc != fileContentsRequest->streamId) || !pStreamStc)
2309 LPENUMFORMATETC pEnumFormatEtc;
2311 FORMATETC vFormatEtc2;
2315 IStream_Release(pStreamStc);
2316 pStreamStc =
nullptr;
2319 bIsStreamFile = FALSE;
2320 hRet = IDataObject_EnumFormatEtc(pDataObj, DATADIR_GET, &pEnumFormatEtc);
2326 hRet = IEnumFORMATETC_Next(pEnumFormatEtc, 1, &vFormatEtc2, &CeltFetched);
2330 if (vFormatEtc2.cfFormat == RegisterClipboardFormat(CFSTR_FILECONTENTS))
2332 hRet = IDataObject_GetData(pDataObj, &vFormatEtc, &vStgMedium);
2336 pStreamStc = vStgMedium.u.pstm;
2337 uStreamIdStc = fileContentsRequest->streamId;
2338 bIsStreamFile = TRUE;
2344 }
while (hRet == S_OK);
2348 if (bIsStreamFile == TRUE)
2350 if (fileContentsRequest->dwFlags == FILECONTENTS_SIZE)
2352 STATSTG vStatStg = WINPR_C_ARRAY_INIT;
2353 hRet = IStream_Stat(pStreamStc, &vStatStg, STATFLAG_NONAME);
2357 *((UINT32*)&pData[0]) = vStatStg.cbSize.QuadPart & 0xFFFFFFFF;
2358 *((UINT32*)&pData[4]) = (vStatStg.cbSize.QuadPart >> 32) & 0xFFFFFFFF;
2359 uSize = cbRequested;
2362 else if (fileContentsRequest->dwFlags == FILECONTENTS_RANGE)
2366 dlibMove.QuadPart = (INT64)(((UINT64)fileContentsRequest->nPositionHigh << 32) |
2367 fileContentsRequest->nPositionLow);
2368 hRet = IStream_Seek(pStreamStc, dlibMove, STREAM_SEEK_SET, &dlibNewPosition);
2370 if (SUCCEEDED(hRet))
2371 hRet = IStream_Read(pStreamStc, pData, cbRequested, (PULONG)&uSize);
2376 if (fileContentsRequest->dwFlags == FILECONTENTS_SIZE)
2378 if (clipboard->nFiles <= fileContentsRequest->listIndex)
2380 *((UINT32*)&pData[0]) =
2381 clipboard->fileDescriptor[fileContentsRequest->listIndex]->nFileSizeLow;
2382 *((UINT32*)&pData[4]) =
2383 clipboard->fileDescriptor[fileContentsRequest->listIndex]->nFileSizeHigh;
2384 uSize = cbRequested;
2386 else if (fileContentsRequest->dwFlags == FILECONTENTS_RANGE)
2389 if (clipboard->nFiles <= fileContentsRequest->listIndex)
2391 bRet = wf_cliprdr_get_file_contents(
2392 clipboard->file_names[fileContentsRequest->listIndex], pData,
2393 fileContentsRequest->nPositionLow, fileContentsRequest->nPositionHigh, cbRequested,
2398 WLog_ERR(TAG,
"get file contents failed.");
2409 IDataObject_Release(pDataObj);
2418 cliprdr_send_response_filecontents(clipboard, fileContentsRequest->streamId, uSize, pData);
2421 if (sRc != CHANNEL_RC_OK)
2433wf_cliprdr_server_file_contents_response(CliprdrClientContext* context,
2436 wfClipboard* clipboard;
2438 if (!context || !fileContentsResponse)
2439 return ERROR_INTERNAL_ERROR;
2441 if (fileContentsResponse->common.msgFlags != CB_RESPONSE_OK)
2444 clipboard = (wfClipboard*)context->custom;
2447 return ERROR_INTERNAL_ERROR;
2449 clipboard->req_fsize = fileContentsResponse->cbRequested;
2450 clipboard->req_fdata = (
char*)malloc(fileContentsResponse->cbRequested);
2452 if (!clipboard->req_fdata)
2453 return ERROR_INTERNAL_ERROR;
2455 CopyMemory(clipboard->req_fdata, fileContentsResponse->requestedData,
2456 fileContentsResponse->cbRequested);
2458 if (!SetEvent(clipboard->req_fevent))
2460 free(clipboard->req_fdata);
2461 return ERROR_INTERNAL_ERROR;
2464 return CHANNEL_RC_OK;
2467BOOL wf_cliprdr_init(wfContext* wfc, CliprdrClientContext* cliprdr)
2469 wfClipboard* clipboard;
2470 rdpContext* context = (rdpContext*)wfc;
2472 if (!context || !cliprdr)
2475 wfc->clipboard = (wfClipboard*)calloc(1,
sizeof(wfClipboard));
2477 if (!wfc->clipboard)
2480 clipboard = wfc->clipboard;
2481 clipboard->wfc = wfc;
2482 clipboard->context = cliprdr;
2483 clipboard->channels = context->channels;
2484 clipboard->sync = FALSE;
2485 clipboard->map_capacity = 32;
2486 clipboard->map_size = 0;
2487 clipboard->hUser32 = LoadLibraryA(
"user32.dll");
2489 if (clipboard->hUser32)
2491 clipboard->AddClipboardFormatListener = GetProcAddressAs(
2492 clipboard->hUser32,
"AddClipboardFormatListener", fnAddClipboardFormatListener);
2493 clipboard->RemoveClipboardFormatListener = GetProcAddressAs(
2494 clipboard->hUser32,
"RemoveClipboardFormatListener", fnRemoveClipboardFormatListener);
2495 clipboard->GetUpdatedClipboardFormats = GetProcAddressAs(
2496 clipboard->hUser32,
"GetUpdatedClipboardFormats", fnGetUpdatedClipboardFormats);
2499 if (!(clipboard->hUser32 && clipboard->AddClipboardFormatListener &&
2500 clipboard->RemoveClipboardFormatListener && clipboard->GetUpdatedClipboardFormats))
2501 clipboard->legacyApi = TRUE;
2503 if (!(clipboard->format_mappings =
2504 (formatMapping*)calloc(clipboard->map_capacity,
sizeof(formatMapping))))
2507 if (!(clipboard->response_data_event = CreateEvent(
nullptr, TRUE, FALSE,
nullptr)))
2510 if (!(clipboard->req_fevent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr)))
2513 if (!(clipboard->thread = CreateThread(
nullptr, 0, cliprdr_thread_func, clipboard, 0,
nullptr)))
2516 cliprdr->MonitorReady = wf_cliprdr_monitor_ready;
2517 cliprdr->ServerCapabilities = wf_cliprdr_server_capabilities;
2518 cliprdr->ServerFormatList = wf_cliprdr_server_format_list;
2519 cliprdr->ServerFormatListResponse = wf_cliprdr_server_format_list_response;
2520 cliprdr->ServerLockClipboardData = wf_cliprdr_server_lock_clipboard_data;
2521 cliprdr->ServerUnlockClipboardData = wf_cliprdr_server_unlock_clipboard_data;
2522 cliprdr->ServerFormatDataRequest = wf_cliprdr_server_format_data_request;
2523 cliprdr->ServerFormatDataResponse = wf_cliprdr_server_format_data_response;
2524 cliprdr->ServerFileContentsRequest = wf_cliprdr_server_file_contents_request;
2525 cliprdr->ServerFileContentsResponse = wf_cliprdr_server_file_contents_response;
2526 cliprdr->custom = (
void*)wfc->clipboard;
2529 wf_cliprdr_uninit(wfc, cliprdr);
2533BOOL wf_cliprdr_uninit(wfContext* wfc, CliprdrClientContext* cliprdr)
2535 wfClipboard* clipboard;
2537 if (!wfc || !cliprdr)
2540 clipboard = wfc->clipboard;
2545 cliprdr->custom =
nullptr;
2547 if (clipboard->hwnd)
2548 PostMessage(clipboard->hwnd, WM_QUIT, 0, 0);
2550 if (clipboard->thread)
2552 (void)WaitForSingleObject(clipboard->thread, INFINITE);
2553 (void)CloseHandle(clipboard->thread);
2556 if (clipboard->response_data_event)
2557 (void)CloseHandle(clipboard->response_data_event);
2559 if (clipboard->req_fevent)
2560 (void)CloseHandle(clipboard->req_fevent);
2562 clear_file_array(clipboard);
2563 clear_format_map(clipboard);
2564 free(clipboard->format_mappings);