24#include <freerdp/config.h>
30#define FUSE_USE_VERSION 30
31#include <fuse_lowlevel.h>
42#include <winpr/string.h>
43#include <winpr/assert.h>
44#include <winpr/image.h>
45#include <winpr/stream.h>
46#include <winpr/clipboard.h>
47#include <winpr/path.h>
49#include <freerdp/utils/signal.h>
50#include <freerdp/log.h>
51#include <freerdp/client/cliprdr.h>
52#include <freerdp/channels/channels.h>
53#include <freerdp/channels/cliprdr.h>
55#include <freerdp/client/client_cliprdr_file.h>
57#define MAX_CLIP_DATA_DIR_LEN 10
58#define NO_CLIP_DATA_ID (UINT64_C(1) << 32)
59#define WIN32_FILETIME_TO_UNIX_EPOCH INT64_C(11644473600)
61#ifdef WITH_DEBUG_CLIPRDR
62#define DEBUG_CLIPRDR(log, ...) WLog_Print(log, WLOG_DEBUG, __VA_ARGS__)
64#define DEBUG_CLIPRDR(log, ...) \
71typedef enum eFuseLowlevelOperationType
73 FUSE_LL_OPERATION_NONE,
74 FUSE_LL_OPERATION_LOOKUP,
75 FUSE_LL_OPERATION_GETATTR,
76 FUSE_LL_OPERATION_READ,
77} FuseLowlevelOperationType;
79typedef struct sCliprdrFuseFile CliprdrFuseFile;
81struct sCliprdrFuseFile
83 CliprdrFuseFile* parent;
87 char* filename_with_root;
97 BOOL has_last_write_time;
98 INT64 last_write_time_unix;
100 BOOL has_clip_data_id;
106 CliprdrFileContext* file_context;
108 CliprdrFuseFile* clip_data_dir;
110 BOOL has_clip_data_id;
112} CliprdrFuseClipDataEntry;
116 CliprdrFileContext* file_context;
118 wArrayList* fuse_files;
121 BOOL has_clip_data_id;
123} FuseFileClearContext;
127 FuseLowlevelOperationType operation_type;
128 CliprdrFuseFile* fuse_file;
135 CliprdrFuseFile* parent;
137} CliprdrFuseFindParentContext;
145 CliprdrFileContext* context;
153 CliprdrLocalFile* files;
154 CliprdrFileContext* context;
157struct cliprdr_file_context
159#if defined(WITH_FUSE)
161 HANDLE fuse_start_sync;
162 HANDLE fuse_stop_sync;
164 struct fuse_session* fuse_sess;
165#if FUSE_USE_VERSION < 30
166 struct fuse_chan* ch;
169 wHashTable* inode_table;
170 wHashTable* clip_data_table;
171 wHashTable* request_table;
173 CliprdrFuseFile* root_dir;
174 CliprdrFuseClipDataEntry* clip_data_entry_without_id;
175 UINT32 current_clip_data_id;
178 UINT32 next_clip_data_id;
179 UINT32 next_stream_id;
183 BOOL file_formats_registered;
184 UINT32 file_capability_flags;
186 UINT32 local_lock_id;
188 wHashTable* local_streams;
191 CliprdrClientContext* context;
194 BYTE server_data_hash[WINPR_SHA256_DIGEST_LENGTH];
195 BYTE client_data_hash[WINPR_SHA256_DIGEST_LENGTH];
198#if defined(WITH_FUSE)
199static void fuse_file_free(
void* data)
201 CliprdrFuseFile* fuse_file = data;
206 ArrayList_Free(fuse_file->children);
207 free(fuse_file->filename_with_root);
212static CliprdrFuseFile* fuse_file_new(
void)
214 CliprdrFuseFile* fuse_file = NULL;
216 fuse_file = calloc(1,
sizeof(CliprdrFuseFile));
220 fuse_file->children = ArrayList_New(FALSE);
221 if (!fuse_file->children)
230static void clip_data_entry_free(
void* data)
232 CliprdrFuseClipDataEntry* clip_data_entry = data;
234 if (!clip_data_entry)
237 if (clip_data_entry->has_clip_data_id)
239 CliprdrFileContext* file_context = clip_data_entry->file_context;
242 WINPR_ASSERT(file_context);
244 unlock_clipboard_data.common.msgType = CB_UNLOCK_CLIPDATA;
245 unlock_clipboard_data.clipDataId = clip_data_entry->clip_data_id;
247 file_context->context->ClientUnlockClipboardData(file_context->context,
248 &unlock_clipboard_data);
249 clip_data_entry->has_clip_data_id = FALSE;
251 WLog_Print(file_context->log, WLOG_DEBUG,
"Destroyed ClipDataEntry with id %u",
252 clip_data_entry->clip_data_id);
255 free(clip_data_entry);
258static BOOL does_server_support_clipdata_locking(CliprdrFileContext* file_context)
260 WINPR_ASSERT(file_context);
262 if (cliprdr_file_context_remote_get_flags(file_context) & CB_CAN_LOCK_CLIPDATA)
268static UINT32 get_next_free_clip_data_id(CliprdrFileContext* file_context)
270 UINT32 clip_data_id = 0;
272 WINPR_ASSERT(file_context);
274 HashTable_Lock(file_context->inode_table);
275 clip_data_id = file_context->next_clip_data_id;
276 while (clip_data_id == 0 ||
277 HashTable_GetItemValue(file_context->clip_data_table, (
void*)(uintptr_t)clip_data_id))
280 file_context->next_clip_data_id = clip_data_id + 1;
281 HashTable_Unlock(file_context->inode_table);
286static CliprdrFuseClipDataEntry* clip_data_entry_new(CliprdrFileContext* file_context,
287 BOOL needs_clip_data_id)
289 CliprdrFuseClipDataEntry* clip_data_entry = NULL;
292 WINPR_ASSERT(file_context);
294 clip_data_entry = calloc(1,
sizeof(CliprdrFuseClipDataEntry));
295 if (!clip_data_entry)
298 clip_data_entry->file_context = file_context;
299 clip_data_entry->clip_data_id = get_next_free_clip_data_id(file_context);
301 if (!needs_clip_data_id)
302 return clip_data_entry;
304 lock_clipboard_data.common.msgType = CB_LOCK_CLIPDATA;
305 lock_clipboard_data.clipDataId = clip_data_entry->clip_data_id;
307 if (file_context->context->ClientLockClipboardData(file_context->context, &lock_clipboard_data))
309 HashTable_Lock(file_context->inode_table);
310 clip_data_entry_free(clip_data_entry);
311 HashTable_Unlock(file_context->inode_table);
314 clip_data_entry->has_clip_data_id = TRUE;
316 WLog_Print(file_context->log, WLOG_DEBUG,
"Created ClipDataEntry with id %u",
317 clip_data_entry->clip_data_id);
319 return clip_data_entry;
322static BOOL should_remove_fuse_file(CliprdrFuseFile* fuse_file, BOOL all_files,
323 BOOL has_clip_data_id, UINT32 clip_data_id)
328 if (fuse_file->ino == FUSE_ROOT_ID)
330 if (!fuse_file->has_clip_data_id && !has_clip_data_id)
332 if (fuse_file->has_clip_data_id && has_clip_data_id &&
333 (fuse_file->clip_data_id == clip_data_id))
339static BOOL maybe_clear_fuse_request(
const void* key,
void* value,
void* arg)
341 CliprdrFuseRequest* fuse_request = value;
342 FuseFileClearContext* clear_context = arg;
343 CliprdrFileContext* file_context = clear_context->file_context;
344 CliprdrFuseFile* fuse_file = fuse_request->fuse_file;
346 WINPR_ASSERT(file_context);
348 if (!should_remove_fuse_file(fuse_file, clear_context->all_files,
349 clear_context->has_clip_data_id, clear_context->clip_data_id))
352 DEBUG_CLIPRDR(file_context->log,
"Clearing FileContentsRequest for file \"%s\"",
353 fuse_file->filename_with_root);
355 fuse_reply_err(fuse_request->fuse_req, EIO);
356 HashTable_Remove(file_context->request_table, key);
361static BOOL maybe_steal_inode(
const void* key,
void* value,
void* arg)
363 CliprdrFuseFile* fuse_file = value;
364 FuseFileClearContext* clear_context = arg;
365 CliprdrFileContext* file_context = clear_context->file_context;
367 WINPR_ASSERT(file_context);
369 if (should_remove_fuse_file(fuse_file, clear_context->all_files,
370 clear_context->has_clip_data_id, clear_context->clip_data_id))
372 if (!ArrayList_Append(clear_context->fuse_files, fuse_file))
373 WLog_Print(file_context->log, WLOG_ERROR,
374 "Failed to append FUSE file to list for deletion");
376 HashTable_Remove(file_context->inode_table, key);
382static BOOL notify_delete_child(
void* data, WINPR_ATTR_UNUSED
size_t index, va_list ap)
384 CliprdrFuseFile* child = data;
388 CliprdrFileContext* file_context = va_arg(ap, CliprdrFileContext*);
389 CliprdrFuseFile* parent = va_arg(ap, CliprdrFuseFile*);
391 WINPR_ASSERT(file_context);
392 WINPR_ASSERT(parent);
394 WINPR_ASSERT(file_context->fuse_sess);
395 fuse_lowlevel_notify_delete(file_context->fuse_sess, parent->ino, child->ino, child->filename,
396 strlen(child->filename));
401static BOOL invalidate_inode(
void* data, WINPR_ATTR_UNUSED
size_t index, va_list ap)
403 CliprdrFuseFile* fuse_file = data;
405 WINPR_ASSERT(fuse_file);
407 CliprdrFileContext* file_context = va_arg(ap, CliprdrFileContext*);
408 WINPR_ASSERT(file_context);
409 WINPR_ASSERT(file_context->fuse_sess);
411 ArrayList_ForEach(fuse_file->children, notify_delete_child, file_context, fuse_file);
413 DEBUG_CLIPRDR(file_context->log,
"Invalidating inode %lu for file \"%s\"", fuse_file->ino,
414 fuse_file->filename);
415 fuse_lowlevel_notify_inval_inode(file_context->fuse_sess, fuse_file->ino, 0, 0);
416 WLog_Print(file_context->log, WLOG_DEBUG,
"Inode %lu invalidated", fuse_file->ino);
421static void clear_selection(CliprdrFileContext* file_context, BOOL all_selections,
422 CliprdrFuseClipDataEntry* clip_data_entry)
424 FuseFileClearContext clear_context = { 0 };
425 CliprdrFuseFile* root_dir = NULL;
426 CliprdrFuseFile* clip_data_dir = NULL;
428 WINPR_ASSERT(file_context);
430 root_dir = file_context->root_dir;
431 WINPR_ASSERT(root_dir);
433 clear_context.file_context = file_context;
434 clear_context.fuse_files = ArrayList_New(FALSE);
435 WINPR_ASSERT(clear_context.fuse_files);
437 wObject* aobj = ArrayList_Object(clear_context.fuse_files);
439 aobj->fnObjectFree = fuse_file_free;
443 clip_data_dir = clip_data_entry->clip_data_dir;
444 clip_data_entry->clip_data_dir = NULL;
446 WINPR_ASSERT(clip_data_dir);
448 ArrayList_Remove(root_dir->children, clip_data_dir);
450 clear_context.has_clip_data_id = clip_data_dir->has_clip_data_id;
451 clear_context.clip_data_id = clip_data_dir->clip_data_id;
453 clear_context.all_files = all_selections;
455 if (clip_data_entry && clip_data_entry->has_clip_data_id)
456 WLog_Print(file_context->log, WLOG_DEBUG,
"Clearing selection for clipDataId %u",
457 clip_data_entry->clip_data_id);
459 WLog_Print(file_context->log, WLOG_DEBUG,
"Clearing selection%s",
460 all_selections ?
"s" :
"");
462 HashTable_Foreach(file_context->request_table, maybe_clear_fuse_request, &clear_context);
463 HashTable_Foreach(file_context->inode_table, maybe_steal_inode, &clear_context);
464 HashTable_Unlock(file_context->inode_table);
466 if (file_context->fuse_sess)
476 ArrayList_ForEach(clear_context.fuse_files, invalidate_inode, file_context);
479 fuse_lowlevel_notify_delete(file_context->fuse_sess, file_context->root_dir->ino,
480 clip_data_dir->ino, clip_data_dir->filename,
481 strlen(clip_data_dir->filename));
484 ArrayList_Free(clear_context.fuse_files);
486 HashTable_Lock(file_context->inode_table);
487 if (clip_data_entry && clip_data_entry->has_clip_data_id)
488 WLog_Print(file_context->log, WLOG_DEBUG,
"Selection cleared for clipDataId %u",
489 clip_data_entry->clip_data_id);
491 WLog_Print(file_context->log, WLOG_DEBUG,
"Selection%s cleared", all_selections ?
"s" :
"");
494static void clear_entry_selection(CliprdrFuseClipDataEntry* clip_data_entry)
496 WINPR_ASSERT(clip_data_entry);
498 if (!clip_data_entry->clip_data_dir)
501 clear_selection(clip_data_entry->file_context, FALSE, clip_data_entry);
504static void clear_no_cdi_entry(CliprdrFileContext* file_context)
506 WINPR_ASSERT(file_context);
508 WINPR_ASSERT(file_context->inode_table);
509 HashTable_Lock(file_context->inode_table);
510 if (file_context->clip_data_entry_without_id)
512 clear_entry_selection(file_context->clip_data_entry_without_id);
514 clip_data_entry_free(file_context->clip_data_entry_without_id);
515 file_context->clip_data_entry_without_id = NULL;
517 HashTable_Unlock(file_context->inode_table);
520static BOOL clear_clip_data_entries(WINPR_ATTR_UNUSED
const void* key,
void* value,
521 WINPR_ATTR_UNUSED
void* arg)
523 clear_entry_selection(value);
528static void clear_cdi_entries(CliprdrFileContext* file_context)
530 WINPR_ASSERT(file_context);
532 HashTable_Lock(file_context->inode_table);
533 HashTable_Foreach(file_context->clip_data_table, clear_clip_data_entries, NULL);
535 HashTable_Clear(file_context->clip_data_table);
536 HashTable_Unlock(file_context->inode_table);
539static UINT prepare_clip_data_entry_with_id(CliprdrFileContext* file_context)
541 CliprdrFuseClipDataEntry* clip_data_entry = NULL;
543 WINPR_ASSERT(file_context);
545 clip_data_entry = clip_data_entry_new(file_context, TRUE);
546 if (!clip_data_entry)
548 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to create clipDataEntry");
549 return ERROR_INTERNAL_ERROR;
552 HashTable_Lock(file_context->inode_table);
553 if (!HashTable_Insert(file_context->clip_data_table,
554 (
void*)(uintptr_t)clip_data_entry->clip_data_id, clip_data_entry))
556 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to insert clipDataEntry");
557 clip_data_entry_free(clip_data_entry);
558 HashTable_Unlock(file_context->inode_table);
559 return ERROR_INTERNAL_ERROR;
561 HashTable_Unlock(file_context->inode_table);
565 file_context->current_clip_data_id = clip_data_entry->clip_data_id;
567 return CHANNEL_RC_OK;
570static UINT prepare_clip_data_entry_without_id(CliprdrFileContext* file_context)
572 WINPR_ASSERT(file_context);
573 WINPR_ASSERT(!file_context->clip_data_entry_without_id);
575 file_context->clip_data_entry_without_id = clip_data_entry_new(file_context, FALSE);
576 if (!file_context->clip_data_entry_without_id)
578 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to create clipDataEntry");
579 return ERROR_INTERNAL_ERROR;
582 return CHANNEL_RC_OK;
586UINT cliprdr_file_context_notify_new_server_format_list(CliprdrFileContext* file_context)
588 WINPR_ASSERT(file_context);
589 WINPR_ASSERT(file_context->context);
591#if defined(WITH_FUSE)
592 clear_no_cdi_entry(file_context);
594 clear_cdi_entries(file_context);
596 if (does_server_support_clipdata_locking(file_context))
597 return prepare_clip_data_entry_with_id(file_context);
599 return prepare_clip_data_entry_without_id(file_context);
601 return CHANNEL_RC_OK;
605UINT cliprdr_file_context_notify_new_client_format_list(CliprdrFileContext* file_context)
607 WINPR_ASSERT(file_context);
608 WINPR_ASSERT(file_context->context);
610#if defined(WITH_FUSE)
611 clear_no_cdi_entry(file_context);
613 clear_cdi_entries(file_context);
616 return CHANNEL_RC_OK;
619static CliprdrLocalStream* cliprdr_local_stream_new(CliprdrFileContext* context, UINT32 streamID,
620 const char* data,
size_t size);
621static void cliprdr_file_session_terminate(CliprdrFileContext* file, BOOL stop_thread);
622static BOOL local_stream_discard(
const void* key,
void* value,
void* arg);
624static void writelog(wLog* log, DWORD level,
const char* fname,
const char* fkt,
size_t line, ...)
626 if (!WLog_IsLevelActive(log, level))
631 WLog_PrintMessageVA(log, WLOG_MESSAGE_TEXT, level, line, fname, fkt, ap);
635#if defined(WITH_FUSE)
636static void cliprdr_file_fuse_lookup(fuse_req_t req, fuse_ino_t parent,
const char* name);
637static void cliprdr_file_fuse_getattr(fuse_req_t req, fuse_ino_t ino,
struct fuse_file_info* fi);
638static void cliprdr_file_fuse_readdir(fuse_req_t req, fuse_ino_t ino,
size_t size, off_t off,
639 struct fuse_file_info* fi);
640static void cliprdr_file_fuse_read(fuse_req_t req, fuse_ino_t ino,
size_t size, off_t off,
641 struct fuse_file_info* fi);
642static void cliprdr_file_fuse_open(fuse_req_t req, fuse_ino_t ino,
struct fuse_file_info* fi);
643static void cliprdr_file_fuse_opendir(fuse_req_t req, fuse_ino_t ino,
struct fuse_file_info* fi);
645static const struct fuse_lowlevel_ops cliprdr_file_fuse_oper = {
646 .lookup = cliprdr_file_fuse_lookup,
647 .getattr = cliprdr_file_fuse_getattr,
648 .readdir = cliprdr_file_fuse_readdir,
649 .open = cliprdr_file_fuse_open,
650 .read = cliprdr_file_fuse_read,
651 .opendir = cliprdr_file_fuse_opendir,
654static CliprdrFuseFile* get_fuse_file_by_ino(CliprdrFileContext* file_context, fuse_ino_t fuse_ino)
656 WINPR_ASSERT(file_context);
658 return HashTable_GetItemValue(file_context->inode_table, (
void*)(uintptr_t)fuse_ino);
661static CliprdrFuseFile*
662get_fuse_file_by_name_from_parent(WINPR_ATTR_UNUSED CliprdrFileContext* file_context,
663 CliprdrFuseFile* parent,
const char* name)
665 WINPR_ASSERT(file_context);
666 WINPR_ASSERT(parent);
668 for (
size_t i = 0; i < ArrayList_Count(parent->children); ++i)
670 CliprdrFuseFile* child = ArrayList_GetItem(parent->children, i);
674 if (strcmp(name, child->filename) == 0)
678 DEBUG_CLIPRDR(file_context->log,
"Requested file \"%s\" in directory \"%s\" does not exist",
679 name, parent->filename);
684static CliprdrFuseRequest* cliprdr_fuse_request_new(CliprdrFileContext* file_context,
685 CliprdrFuseFile* fuse_file, fuse_req_t fuse_req,
686 FuseLowlevelOperationType operation_type)
688 CliprdrFuseRequest* fuse_request = NULL;
689 UINT32 stream_id = file_context->next_stream_id;
691 WINPR_ASSERT(file_context);
692 WINPR_ASSERT(fuse_file);
694 fuse_request = calloc(1,
sizeof(CliprdrFuseRequest));
697 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to allocate FUSE request for file \"%s\"",
698 fuse_file->filename_with_root);
702 fuse_request->fuse_file = fuse_file;
703 fuse_request->fuse_req = fuse_req;
704 fuse_request->operation_type = operation_type;
706 while (stream_id == 0 ||
707 HashTable_GetItemValue(file_context->request_table, (
void*)(uintptr_t)stream_id))
709 fuse_request->stream_id = stream_id;
711 file_context->next_stream_id = stream_id + 1;
713 if (!HashTable_Insert(file_context->request_table, (
void*)(uintptr_t)fuse_request->stream_id,
716 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to track FUSE request for file \"%s\"",
717 fuse_file->filename_with_root);
725static BOOL request_file_size_async(CliprdrFileContext* file_context, CliprdrFuseFile* fuse_file,
726 fuse_req_t fuse_req, FuseLowlevelOperationType operation_type)
728 CliprdrFuseRequest* fuse_request = NULL;
731 WINPR_ASSERT(file_context);
732 WINPR_ASSERT(fuse_file);
734 fuse_request = cliprdr_fuse_request_new(file_context, fuse_file, fuse_req, operation_type);
738 file_contents_request.common.msgType = CB_FILECONTENTS_REQUEST;
739 file_contents_request.streamId = fuse_request->stream_id;
740 file_contents_request.listIndex = fuse_file->list_idx;
741 file_contents_request.dwFlags = FILECONTENTS_SIZE;
742 file_contents_request.cbRequested = 0x8;
743 file_contents_request.haveClipDataId = fuse_file->has_clip_data_id;
744 file_contents_request.clipDataId = fuse_file->clip_data_id;
746 if (file_context->context->ClientFileContentsRequest(file_context->context,
747 &file_contents_request))
749 WLog_Print(file_context->log, WLOG_ERROR,
750 "Failed to send FileContentsRequest for file \"%s\"",
751 fuse_file->filename_with_root);
752 HashTable_Remove(file_context->request_table, (
void*)(uintptr_t)fuse_request->stream_id);
756 DEBUG_CLIPRDR(file_context->log,
"Requested file size for file \"%s\" with stream id %u",
757 fuse_file->filename, fuse_request->stream_id);
763static void write_file_attributes(CliprdrFuseFile* fuse_file,
struct stat* attr)
765 memset(attr, 0,
sizeof(
struct stat));
770 attr->st_ino = fuse_file->ino;
771 if (fuse_file->is_directory)
773 attr->st_mode = S_IFDIR | (fuse_file->is_readonly ? 0555 : 0755);
778 attr->st_mode = S_IFREG | (fuse_file->is_readonly ? 0444 : 0644);
780 attr->st_size = WINPR_ASSERTING_INT_CAST(off_t, fuse_file->size);
782 attr->st_uid = getuid();
783 attr->st_gid = getgid();
784 attr->st_atime = attr->st_mtime = attr->st_ctime =
785 (fuse_file->has_last_write_time ? fuse_file->last_write_time_unix : time(NULL));
788static void cliprdr_file_fuse_lookup(fuse_req_t fuse_req, fuse_ino_t parent_ino,
const char* name)
790 CliprdrFileContext* file_context = fuse_req_userdata(fuse_req);
791 CliprdrFuseFile* parent = NULL;
792 CliprdrFuseFile* fuse_file = NULL;
793 struct fuse_entry_param entry = { 0 };
795 WINPR_ASSERT(file_context);
797 HashTable_Lock(file_context->inode_table);
798 if (!(parent = get_fuse_file_by_ino(file_context, parent_ino)) || !parent->is_directory)
800 HashTable_Unlock(file_context->inode_table);
801 fuse_reply_err(fuse_req, ENOENT);
804 if (!(fuse_file = get_fuse_file_by_name_from_parent(file_context, parent, name)))
806 HashTable_Unlock(file_context->inode_table);
807 fuse_reply_err(fuse_req, ENOENT);
811 DEBUG_CLIPRDR(file_context->log,
"lookup() has been called for \"%s\"", name);
812 DEBUG_CLIPRDR(file_context->log,
"Parent is \"%s\", child is \"%s\"",
813 parent->filename_with_root, fuse_file->filename_with_root);
815 if (!fuse_file->is_directory && !fuse_file->has_size)
820 request_file_size_async(file_context, fuse_file, fuse_req, FUSE_LL_OPERATION_LOOKUP);
821 HashTable_Unlock(file_context->inode_table);
824 fuse_reply_err(fuse_req, EIO);
829 entry.ino = fuse_file->ino;
830 write_file_attributes(fuse_file, &entry.attr);
831 entry.attr_timeout = 1.0;
832 entry.entry_timeout = 1.0;
833 HashTable_Unlock(file_context->inode_table);
835 fuse_reply_entry(fuse_req, &entry);
838static void cliprdr_file_fuse_getattr(fuse_req_t fuse_req, fuse_ino_t fuse_ino,
839 WINPR_ATTR_UNUSED
struct fuse_file_info* file_info)
841 CliprdrFileContext* file_context = fuse_req_userdata(fuse_req);
842 CliprdrFuseFile* fuse_file = NULL;
843 struct stat attr = { 0 };
845 WINPR_ASSERT(file_context);
847 HashTable_Lock(file_context->inode_table);
848 if (!(fuse_file = get_fuse_file_by_ino(file_context, fuse_ino)))
850 HashTable_Unlock(file_context->inode_table);
851 fuse_reply_err(fuse_req, ENOENT);
855 DEBUG_CLIPRDR(file_context->log,
"getattr() has been called for file \"%s\"",
856 fuse_file->filename_with_root);
858 if (!fuse_file->is_directory && !fuse_file->has_size)
863 request_file_size_async(file_context, fuse_file, fuse_req, FUSE_LL_OPERATION_GETATTR);
864 HashTable_Unlock(file_context->inode_table);
867 fuse_reply_err(fuse_req, EIO);
872 write_file_attributes(fuse_file, &attr);
873 HashTable_Unlock(file_context->inode_table);
875 fuse_reply_attr(fuse_req, &attr, 1.0);
878static void cliprdr_file_fuse_open(fuse_req_t fuse_req, fuse_ino_t fuse_ino,
879 struct fuse_file_info* file_info)
881 CliprdrFileContext* file_context = fuse_req_userdata(fuse_req);
882 CliprdrFuseFile* fuse_file = NULL;
884 WINPR_ASSERT(file_context);
886 HashTable_Lock(file_context->inode_table);
887 if (!(fuse_file = get_fuse_file_by_ino(file_context, fuse_ino)))
889 HashTable_Unlock(file_context->inode_table);
890 fuse_reply_err(fuse_req, ENOENT);
893 if (fuse_file->is_directory)
895 HashTable_Unlock(file_context->inode_table);
896 fuse_reply_err(fuse_req, EISDIR);
899 HashTable_Unlock(file_context->inode_table);
901 if ((file_info->flags & O_ACCMODE) != O_RDONLY)
903 fuse_reply_err(fuse_req, EACCES);
908 file_info->direct_io = 1;
910 fuse_reply_open(fuse_req, file_info);
913static BOOL request_file_range_async(CliprdrFileContext* file_context, CliprdrFuseFile* fuse_file,
914 fuse_req_t fuse_req, off_t offset,
size_t requested_size)
918 WINPR_ASSERT(file_context);
919 WINPR_ASSERT(fuse_file);
921 if (requested_size > UINT32_MAX)
924 CliprdrFuseRequest* fuse_request =
925 cliprdr_fuse_request_new(file_context, fuse_file, fuse_req, FUSE_LL_OPERATION_READ);
929 file_contents_request.common.msgType = CB_FILECONTENTS_REQUEST;
930 file_contents_request.streamId = fuse_request->stream_id;
931 file_contents_request.listIndex = fuse_file->list_idx;
932 file_contents_request.dwFlags = FILECONTENTS_RANGE;
933 file_contents_request.nPositionLow = (UINT32)(offset & 0xFFFFFFFF);
934 file_contents_request.nPositionHigh = (UINT32)((offset >> 32) & 0xFFFFFFFF);
935 file_contents_request.cbRequested = (UINT32)requested_size;
936 file_contents_request.haveClipDataId = fuse_file->has_clip_data_id;
937 file_contents_request.clipDataId = fuse_file->clip_data_id;
939 if (file_context->context->ClientFileContentsRequest(file_context->context,
940 &file_contents_request))
942 WLog_Print(file_context->log, WLOG_ERROR,
943 "Failed to send FileContentsRequest for file \"%s\"",
944 fuse_file->filename_with_root);
945 HashTable_Remove(file_context->request_table, (
void*)(uintptr_t)fuse_request->stream_id);
953 "Requested file range (%zu Bytes at offset %lu) for file \"%s\" with stream id %u",
954 requested_size, offset, fuse_file->filename, fuse_request->stream_id);
960static void cliprdr_file_fuse_read(fuse_req_t fuse_req, fuse_ino_t fuse_ino,
size_t size,
961 off_t offset, WINPR_ATTR_UNUSED
struct fuse_file_info* file_info)
963 CliprdrFileContext* file_context = fuse_req_userdata(fuse_req);
964 CliprdrFuseFile* fuse_file = NULL;
967 WINPR_ASSERT(file_context);
969 HashTable_Lock(file_context->inode_table);
970 if (!(fuse_file = get_fuse_file_by_ino(file_context, fuse_ino)))
972 HashTable_Unlock(file_context->inode_table);
973 fuse_reply_err(fuse_req, ENOENT);
976 if (fuse_file->is_directory)
978 HashTable_Unlock(file_context->inode_table);
979 fuse_reply_err(fuse_req, EISDIR);
982 if (!fuse_file->has_size || (offset < 0) || ((
size_t)offset > fuse_file->size))
984 HashTable_Unlock(file_context->inode_table);
985 fuse_reply_err(fuse_req, EINVAL);
989 size = MIN(size, 8ULL * 1024ULL * 1024ULL);
991 result = request_file_range_async(file_context, fuse_file, fuse_req, offset, size);
992 HashTable_Unlock(file_context->inode_table);
995 fuse_reply_err(fuse_req, EIO);
998static void cliprdr_file_fuse_opendir(fuse_req_t fuse_req, fuse_ino_t fuse_ino,
999 struct fuse_file_info* file_info)
1001 CliprdrFileContext* file_context = fuse_req_userdata(fuse_req);
1002 CliprdrFuseFile* fuse_file = NULL;
1004 WINPR_ASSERT(file_context);
1006 HashTable_Lock(file_context->inode_table);
1007 if (!(fuse_file = get_fuse_file_by_ino(file_context, fuse_ino)))
1009 HashTable_Unlock(file_context->inode_table);
1010 fuse_reply_err(fuse_req, ENOENT);
1013 if (!fuse_file->is_directory)
1015 HashTable_Unlock(file_context->inode_table);
1016 fuse_reply_err(fuse_req, ENOTDIR);
1019 HashTable_Unlock(file_context->inode_table);
1021 if ((file_info->flags & O_ACCMODE) != O_RDONLY)
1023 fuse_reply_err(fuse_req, EACCES);
1027 fuse_reply_open(fuse_req, file_info);
1030static void cliprdr_file_fuse_readdir(fuse_req_t fuse_req, fuse_ino_t fuse_ino,
size_t max_size,
1032 WINPR_ATTR_UNUSED
struct fuse_file_info* file_info)
1034 CliprdrFileContext* file_context = fuse_req_userdata(fuse_req);
1035 CliprdrFuseFile* fuse_file = NULL;
1036 CliprdrFuseFile* child = NULL;
1037 struct stat attr = { 0 };
1038 size_t written_size = 0;
1039 size_t entry_size = 0;
1040 char* filename = NULL;
1043 WINPR_ASSERT(file_context);
1045 HashTable_Lock(file_context->inode_table);
1046 if (!(fuse_file = get_fuse_file_by_ino(file_context, fuse_ino)))
1048 HashTable_Unlock(file_context->inode_table);
1049 fuse_reply_err(fuse_req, ENOENT);
1052 if (!fuse_file->is_directory)
1054 HashTable_Unlock(file_context->inode_table);
1055 fuse_reply_err(fuse_req, ENOTDIR);
1059 DEBUG_CLIPRDR(file_context->log,
"Reading directory \"%s\" at offset %lu",
1060 fuse_file->filename_with_root, offset);
1062 if ((offset < 0) || ((
size_t)offset >= ArrayList_Count(fuse_file->children)))
1064 HashTable_Unlock(file_context->inode_table);
1065 fuse_reply_buf(fuse_req, NULL, 0);
1069 buf = calloc(max_size,
sizeof(
char));
1072 HashTable_Unlock(file_context->inode_table);
1073 fuse_reply_err(fuse_req, ENOMEM);
1078 for (off_t i = offset; i < 2; ++i)
1082 write_file_attributes(fuse_file, &attr);
1087 write_file_attributes(fuse_file->parent, &attr);
1088 attr.st_ino = fuse_file->parent ? attr.st_ino : FUSE_ROOT_ID;
1089 attr.st_mode = fuse_file->parent ? attr.st_mode : 0555;
1094 WINPR_ASSERT(FALSE);
1101 entry_size = fuse_add_direntry(fuse_req, buf + written_size, max_size - written_size,
1102 filename, &attr, i + 1);
1103 if (entry_size > max_size - written_size)
1106 written_size += entry_size;
1109 for (
size_t j = 0, i = 2; j < ArrayList_Count(fuse_file->children); ++j, ++i)
1111 if (i < (
size_t)offset)
1114 child = ArrayList_GetItem(fuse_file->children, j);
1116 write_file_attributes(child, &attr);
1118 fuse_add_direntry(fuse_req, buf + written_size, max_size - written_size,
1119 child->filename, &attr, WINPR_ASSERTING_INT_CAST(off_t, i + 1));
1120 if (entry_size > max_size - written_size)
1123 written_size += entry_size;
1125 HashTable_Unlock(file_context->inode_table);
1127 fuse_reply_buf(fuse_req, buf, written_size);
1131static void fuse_abort(
int sig,
const char* signame,
void* context)
1133 CliprdrFileContext* file = (CliprdrFileContext*)context;
1137 WLog_Print(file->log, WLOG_INFO,
"signal %s [%d] aborting session", signame, sig);
1138 cliprdr_file_session_terminate(file, FALSE);
1142static DWORD WINAPI cliprdr_file_fuse_thread(LPVOID arg)
1144 CliprdrFileContext* file = (CliprdrFileContext*)arg;
1148 DEBUG_CLIPRDR(file->log,
"Starting fuse with mountpoint '%s'", file->path);
1150 struct fuse_args args = FUSE_ARGS_INIT(0, NULL);
1151 fuse_opt_add_arg(&args, file->path);
1152 file->fuse_sess = fuse_session_new(&args, &cliprdr_file_fuse_oper,
1153 sizeof(cliprdr_file_fuse_oper), (
void*)file);
1154 (void)SetEvent(file->fuse_start_sync);
1156 if (file->fuse_sess != NULL)
1158 freerdp_add_signal_cleanup_handler(file, fuse_abort);
1159 if (0 == fuse_session_mount(file->fuse_sess, file->path))
1161 fuse_session_loop(file->fuse_sess);
1162 fuse_session_unmount(file->fuse_sess);
1164 freerdp_del_signal_cleanup_handler(file, fuse_abort);
1166 WLog_Print(file->log, WLOG_DEBUG,
"Waiting for FUSE stop sync");
1167 if (WaitForSingleObject(file->fuse_stop_sync, INFINITE) == WAIT_FAILED)
1168 WLog_Print(file->log, WLOG_ERROR,
"Failed to wait for stop sync");
1169 fuse_session_destroy(file->fuse_sess);
1171 fuse_opt_free_args(&args);
1173 DEBUG_CLIPRDR(file->log,
"Quitting fuse with mountpoint '%s'", file->path);
1179static UINT cliprdr_file_context_server_file_contents_response(
1180 CliprdrClientContext* cliprdr_context,
1183 CliprdrFileContext* file_context = NULL;
1184 CliprdrFuseRequest* fuse_request = NULL;
1185 struct fuse_entry_param entry = { 0 };
1187 WINPR_ASSERT(cliprdr_context);
1188 WINPR_ASSERT(file_contents_response);
1190 file_context = cliprdr_context->custom;
1191 WINPR_ASSERT(file_context);
1193 HashTable_Lock(file_context->inode_table);
1194 fuse_request = HashTable_GetItemValue(file_context->request_table,
1195 (
void*)(uintptr_t)file_contents_response->streamId);
1198 HashTable_Unlock(file_context->inode_table);
1199 return CHANNEL_RC_OK;
1202 if (!(file_contents_response->common.msgFlags & CB_RESPONSE_OK))
1204 WLog_Print(file_context->log, WLOG_WARN,
1205 "FileContentsRequests for file \"%s\" was unsuccessful",
1206 fuse_request->fuse_file->filename);
1208 fuse_reply_err(fuse_request->fuse_req, EIO);
1209 HashTable_Remove(file_context->request_table,
1210 (
void*)(uintptr_t)file_contents_response->streamId);
1211 HashTable_Unlock(file_context->inode_table);
1212 return CHANNEL_RC_OK;
1215 if ((fuse_request->operation_type == FUSE_LL_OPERATION_LOOKUP ||
1216 fuse_request->operation_type == FUSE_LL_OPERATION_GETATTR) &&
1217 file_contents_response->cbRequested !=
sizeof(UINT64))
1219 WLog_Print(file_context->log, WLOG_WARN,
1220 "Received invalid file size for file \"%s\" from the client",
1221 fuse_request->fuse_file->filename);
1222 fuse_reply_err(fuse_request->fuse_req, EIO);
1223 HashTable_Remove(file_context->request_table,
1224 (
void*)(uintptr_t)file_contents_response->streamId);
1225 HashTable_Unlock(file_context->inode_table);
1226 return CHANNEL_RC_OK;
1229 if (fuse_request->operation_type == FUSE_LL_OPERATION_LOOKUP ||
1230 fuse_request->operation_type == FUSE_LL_OPERATION_GETATTR)
1232 DEBUG_CLIPRDR(file_context->log,
"Received file size for file \"%s\" with stream id %u",
1233 fuse_request->fuse_file->filename, file_contents_response->streamId);
1235 fuse_request->fuse_file->size = *((
const UINT64*)file_contents_response->requestedData);
1236 fuse_request->fuse_file->has_size = TRUE;
1238 entry.ino = fuse_request->fuse_file->ino;
1239 write_file_attributes(fuse_request->fuse_file, &entry.attr);
1240 entry.attr_timeout = 1.0;
1241 entry.entry_timeout = 1.0;
1243 else if (fuse_request->operation_type == FUSE_LL_OPERATION_READ)
1245 DEBUG_CLIPRDR(file_context->log,
"Received file range for file \"%s\" with stream id %u",
1246 fuse_request->fuse_file->filename, file_contents_response->streamId);
1248 HashTable_Unlock(file_context->inode_table);
1250 switch (fuse_request->operation_type)
1252 case FUSE_LL_OPERATION_NONE:
1254 case FUSE_LL_OPERATION_LOOKUP:
1255 fuse_reply_entry(fuse_request->fuse_req, &entry);
1257 case FUSE_LL_OPERATION_GETATTR:
1258 fuse_reply_attr(fuse_request->fuse_req, &entry.attr, entry.attr_timeout);
1260 case FUSE_LL_OPERATION_READ:
1261 fuse_reply_buf(fuse_request->fuse_req,
1262 (
const char*)file_contents_response->requestedData,
1263 file_contents_response->cbRequested);
1269 HashTable_Remove(file_context->request_table,
1270 (
void*)(uintptr_t)file_contents_response->streamId);
1272 return CHANNEL_RC_OK;
1276static UINT cliprdr_file_context_send_file_contents_failure(
1282 WINPR_ASSERT(fileContentsRequest);
1284 const UINT64 offset = (((UINT64)fileContentsRequest->nPositionHigh) << 32) |
1285 ((UINT64)fileContentsRequest->nPositionLow);
1286 writelog(file->log, WLOG_WARN, __FILE__, __func__, __LINE__,
1287 "server file contents request [lockID %" PRIu32
", streamID %" PRIu32
1288 ", index %" PRIu32
"] offset %" PRIu64
", size %" PRIu32
" failed",
1289 fileContentsRequest->clipDataId, fileContentsRequest->streamId,
1290 fileContentsRequest->listIndex, offset, fileContentsRequest->cbRequested);
1292 response.common.msgFlags = CB_RESPONSE_FAIL;
1293 response.streamId = fileContentsRequest->streamId;
1295 WINPR_ASSERT(file->context);
1296 WINPR_ASSERT(file->context->ClientFileContentsResponse);
1297 return file->context->ClientFileContentsResponse(file->context, &response);
1301cliprdr_file_context_send_contents_response(CliprdrFileContext* file,
1303 const void* data,
size_t size)
1305 if (size > UINT32_MAX)
1306 return ERROR_INVALID_PARAMETER;
1309 .requestedData = data,
1310 .cbRequested = (UINT32)size,
1311 .common.msgFlags = CB_RESPONSE_OK };
1313 WINPR_ASSERT(request);
1316 WLog_Print(file->log, WLOG_DEBUG,
"send contents response streamID=%" PRIu32
", size=%" PRIu32,
1317 response.streamId, response.cbRequested);
1318 WINPR_ASSERT(file->context);
1319 WINPR_ASSERT(file->context->ClientFileContentsResponse);
1320 return file->context->ClientFileContentsResponse(file->context, &response);
1323static BOOL dump_streams(
const void* key,
void* value, WINPR_ATTR_UNUSED
void* arg)
1325 const UINT32* ukey = key;
1326 CliprdrLocalStream* cur = value;
1328 writelog(cur->context->log, WLOG_WARN, __FILE__, __func__, __LINE__,
1329 "[key %" PRIu32
"] lockID %" PRIu32
", count %" PRIuz
", locked %d", *ukey,
1330 cur->lockId, cur->count, cur->locked);
1331 for (
size_t x = 0; x < cur->count; x++)
1333 const CliprdrLocalFile* file = &cur->files[x];
1334 writelog(cur->context->log, WLOG_WARN, __FILE__, __func__, __LINE__,
"file [%" PRIuz
"] ",
1335 x, file->name, file->size);
1340static CliprdrLocalFile* file_info_for_request(CliprdrFileContext* file, UINT32 lockId,
1345 CliprdrLocalStream* cur = HashTable_GetItemValue(file->local_streams, &lockId);
1348 if (listIndex < cur->count)
1350 CliprdrLocalFile* f = &cur->files[listIndex];
1355 writelog(file->log, WLOG_WARN, __FILE__, __func__, __LINE__,
1356 "invalid entry index for lockID %" PRIu32
", index %" PRIu32
" [count %" PRIu32
1358 lockId, listIndex, cur->count, cur->locked);
1363 writelog(file->log, WLOG_WARN, __FILE__, __func__, __LINE__,
1364 "missing entry for lockID %" PRIu32
", index %" PRIu32, lockId, listIndex);
1365 HashTable_Foreach(file->local_streams, dump_streams, file);
1371static CliprdrLocalFile* file_for_request(CliprdrFileContext* file, UINT32 lockId, UINT32 listIndex)
1373 CliprdrLocalFile* f = file_info_for_request(file, lockId, listIndex);
1378 const char* name = f->name;
1379 f->fp = winpr_fopen(name,
"rb");
1383 char ebuffer[256] = { 0 };
1384 writelog(file->log, WLOG_WARN, __FILE__, __func__, __LINE__,
1385 "[lockID %" PRIu32
", index %" PRIu32
1386 "] failed to open file '%s' [size %" PRId64
"] %s [%d]",
1387 lockId, listIndex, f->name, f->size,
1388 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
1396static void cliprdr_local_file_try_close(CliprdrLocalFile* file, UINT res, UINT64 offset,
1403 WINPR_ASSERT(file->context);
1404 WLog_Print(file->context->log, WLOG_DEBUG,
"closing file %s after error %" PRIu32,
1407 else if (((file->size > 0) && (offset + size >= (UINT64)file->size)))
1409 WINPR_ASSERT(file->context);
1410 WLog_Print(file->context->log, WLOG_DEBUG,
"closing file %s after read", file->name);
1418 (void)fclose(file->fp);
1422static UINT cliprdr_file_context_server_file_size_request(
1425 WINPR_ASSERT(fileContentsRequest);
1427 if (fileContentsRequest->cbRequested !=
sizeof(UINT64))
1429 WLog_Print(file->log, WLOG_WARN,
"unexpected FILECONTENTS_SIZE request: %" PRIu32
" bytes",
1430 fileContentsRequest->cbRequested);
1433 HashTable_Lock(file->local_streams);
1435 UINT res = CHANNEL_RC_OK;
1436 CliprdrLocalFile* rfile =
1437 file_for_request(file, fileContentsRequest->clipDataId, fileContentsRequest->listIndex);
1439 res = cliprdr_file_context_send_file_contents_failure(file, fileContentsRequest);
1442 if (_fseeki64(rfile->fp, 0, SEEK_END) < 0)
1443 res = cliprdr_file_context_send_file_contents_failure(file, fileContentsRequest);
1446 const INT64 size = _ftelli64(rfile->fp);
1448 cliprdr_local_file_try_close(rfile, res, 0, 0);
1450 res = cliprdr_file_context_send_contents_response(file, fileContentsRequest, &size,
1455 HashTable_Unlock(file->local_streams);
1459static UINT cliprdr_file_context_server_file_range_request(
1464 WINPR_ASSERT(fileContentsRequest);
1466 HashTable_Lock(file->local_streams);
1467 const UINT64 offset = (((UINT64)fileContentsRequest->nPositionHigh) << 32) |
1468 ((UINT64)fileContentsRequest->nPositionLow);
1470 CliprdrLocalFile* rfile =
1471 file_for_request(file, fileContentsRequest->clipDataId, fileContentsRequest->listIndex);
1475 if (_fseeki64(rfile->fp, WINPR_ASSERTING_INT_CAST(int64_t, offset), SEEK_SET) < 0)
1478 data = malloc(fileContentsRequest->cbRequested);
1482 const size_t r = fread(data, 1, fileContentsRequest->cbRequested, rfile->fp);
1483 const UINT rc = cliprdr_file_context_send_contents_response(file, fileContentsRequest, data, r);
1486 cliprdr_local_file_try_close(rfile, rc, offset, fileContentsRequest->cbRequested);
1487 HashTable_Unlock(file->local_streams);
1491 cliprdr_local_file_try_close(rfile, ERROR_INTERNAL_ERROR, offset,
1492 fileContentsRequest->cbRequested);
1494 HashTable_Unlock(file->local_streams);
1495 return cliprdr_file_context_send_file_contents_failure(file, fileContentsRequest);
1498static void cliprdr_local_stream_free(
void* obj);
1500static UINT change_lock(CliprdrFileContext* file, UINT32 lockId, BOOL lock)
1502 UINT rc = CHANNEL_RC_OK;
1506 HashTable_Lock(file->local_streams);
1509 CliprdrLocalStream* stream = HashTable_GetItemValue(file->local_streams, &lockId);
1510 if (lock && !stream)
1512 stream = cliprdr_local_stream_new(file, lockId, NULL, 0);
1513 if (!HashTable_Insert(file->local_streams, &lockId, stream))
1515 rc = ERROR_INTERNAL_ERROR;
1516 cliprdr_local_stream_free(stream);
1519 file->local_lock_id = lockId;
1523 stream->locked = lock;
1524 stream->lockId = lockId;
1531 if (!HashTable_Foreach(file->local_streams, local_stream_discard, file))
1532 rc = ERROR_INTERNAL_ERROR;
1535 HashTable_Unlock(file->local_streams);
1539static UINT cliprdr_file_context_lock(CliprdrClientContext* context,
1542 WINPR_ASSERT(context);
1543 WINPR_ASSERT(lockClipboardData);
1544 CliprdrFileContext* file = (context->custom);
1545 return change_lock(file, lockClipboardData->clipDataId, TRUE);
1548static UINT cliprdr_file_context_unlock(CliprdrClientContext* context,
1551 WINPR_ASSERT(context);
1552 WINPR_ASSERT(unlockClipboardData);
1553 CliprdrFileContext* file = (context->custom);
1554 return change_lock(file, unlockClipboardData->clipDataId, FALSE);
1557static UINT cliprdr_file_context_server_file_contents_request(
1560 UINT error = NO_ERROR;
1562 WINPR_ASSERT(context);
1563 WINPR_ASSERT(fileContentsRequest);
1565 CliprdrFileContext* file = (context->custom);
1572 if ((fileContentsRequest->dwFlags & (FILECONTENTS_SIZE | FILECONTENTS_RANGE)) ==
1573 (FILECONTENTS_SIZE | FILECONTENTS_RANGE))
1575 WLog_Print(file->log, WLOG_ERROR,
"invalid CLIPRDR_FILECONTENTS_REQUEST.dwFlags");
1576 return cliprdr_file_context_send_file_contents_failure(file, fileContentsRequest);
1579 if (fileContentsRequest->dwFlags & FILECONTENTS_SIZE)
1580 error = cliprdr_file_context_server_file_size_request(file, fileContentsRequest);
1582 if (fileContentsRequest->dwFlags & FILECONTENTS_RANGE)
1583 error = cliprdr_file_context_server_file_range_request(file, fileContentsRequest);
1587 WLog_Print(file->log, WLOG_ERROR,
"failed to handle CLIPRDR_FILECONTENTS_REQUEST: 0x%08X",
1589 return cliprdr_file_context_send_file_contents_failure(file, fileContentsRequest);
1592 return CHANNEL_RC_OK;
1595BOOL cliprdr_file_context_init(CliprdrFileContext* file, CliprdrClientContext* cliprdr)
1598 WINPR_ASSERT(cliprdr);
1600 cliprdr->custom = file;
1601 file->context = cliprdr;
1603 cliprdr->ServerLockClipboardData = cliprdr_file_context_lock;
1604 cliprdr->ServerUnlockClipboardData = cliprdr_file_context_unlock;
1605 cliprdr->ServerFileContentsRequest = cliprdr_file_context_server_file_contents_request;
1606#if defined(WITH_FUSE)
1607 cliprdr->ServerFileContentsResponse = cliprdr_file_context_server_file_contents_response;
1613#if defined(WITH_FUSE)
1614static void clear_all_selections(CliprdrFileContext* file_context)
1616 WINPR_ASSERT(file_context);
1617 WINPR_ASSERT(file_context->inode_table);
1619 HashTable_Lock(file_context->inode_table);
1620 clear_selection(file_context, TRUE, NULL);
1622 HashTable_Clear(file_context->clip_data_table);
1623 HashTable_Unlock(file_context->inode_table);
1627BOOL cliprdr_file_context_uninit(CliprdrFileContext* file, CliprdrClientContext* cliprdr)
1630 WINPR_ASSERT(cliprdr);
1634#if defined(WITH_FUSE)
1635 if (file->inode_table)
1637 clear_no_cdi_entry(file);
1638 clear_all_selections(file);
1642 HashTable_Clear(file->local_streams);
1644 file->context = NULL;
1645#if defined(WITH_FUSE)
1646 cliprdr->ServerFileContentsResponse = NULL;
1652static BOOL cliprdr_file_content_changed_and_update(
void* ihash,
size_t hsize,
const void* data,
1656 BYTE hash[WINPR_SHA256_DIGEST_LENGTH] = { 0 };
1658 if (hsize <
sizeof(hash))
1661 if (!winpr_Digest(WINPR_MD_SHA256, data, size, hash,
sizeof(hash)))
1664 const BOOL changed = memcmp(hash, ihash,
sizeof(hash)) != 0;
1666 memcpy(ihash, hash,
sizeof(hash));
1670static BOOL cliprdr_file_client_content_changed_and_update(CliprdrFileContext* file,
1671 const void* data,
size_t size)
1674 return cliprdr_file_content_changed_and_update(file->client_data_hash,
1675 sizeof(file->client_data_hash), data, size);
1678#if defined(WITH_FUSE)
1679static fuse_ino_t get_next_free_inode(CliprdrFileContext* file_context)
1683 WINPR_ASSERT(file_context);
1685 ino = file_context->next_ino;
1686 while (ino == 0 || ino == FUSE_ROOT_ID ||
1687 HashTable_GetItemValue(file_context->inode_table, (
void*)(uintptr_t)ino))
1690 file_context->next_ino = ino + 1;
1695static CliprdrFuseFile* clip_data_dir_new(CliprdrFileContext* file_context, BOOL has_clip_data_id,
1696 UINT32 clip_data_id)
1698 CliprdrFuseFile* root_dir = NULL;
1699 CliprdrFuseFile* clip_data_dir = NULL;
1700 size_t path_length = 0;
1702 WINPR_ASSERT(file_context);
1704 clip_data_dir = fuse_file_new();
1708 path_length = 1 + MAX_CLIP_DATA_DIR_LEN + 1;
1710 clip_data_dir->filename_with_root = calloc(path_length,
sizeof(
char));
1711 if (!clip_data_dir->filename_with_root)
1713 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to allocate filename");
1714 fuse_file_free(clip_data_dir);
1718 if (has_clip_data_id)
1719 (void)_snprintf(clip_data_dir->filename_with_root, path_length,
"/%u",
1720 (
unsigned)clip_data_id);
1722 (
void)_snprintf(clip_data_dir->filename_with_root, path_length,
"/%" PRIu64,
1725 clip_data_dir->filename = strrchr(clip_data_dir->filename_with_root,
'/') + 1;
1727 clip_data_dir->ino = get_next_free_inode(file_context);
1728 clip_data_dir->is_directory = TRUE;
1729 clip_data_dir->is_readonly = TRUE;
1730 clip_data_dir->has_clip_data_id = has_clip_data_id;
1731 clip_data_dir->clip_data_id = clip_data_id;
1733 root_dir = file_context->root_dir;
1734 if (!ArrayList_Append(root_dir->children, clip_data_dir))
1736 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to append FUSE file");
1737 fuse_file_free(clip_data_dir);
1740 clip_data_dir->parent = root_dir;
1742 if (!HashTable_Insert(file_context->inode_table, (
void*)(uintptr_t)clip_data_dir->ino,
1745 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to insert inode into inode table");
1746 ArrayList_Remove(root_dir->children, clip_data_dir);
1747 fuse_file_free(clip_data_dir);
1751 return clip_data_dir;
1754static char* get_parent_path(
const char* filepath)
1756 const char* base = strrchr(filepath,
'/');
1759 while ((base > filepath) && (*base ==
'/'))
1762 WINPR_ASSERT(base >= filepath);
1763 const size_t parent_path_length = 1ULL + (size_t)(base - filepath);
1764 char* parent_path = calloc(parent_path_length + 1,
sizeof(
char));
1768 memcpy(parent_path, filepath, parent_path_length);
1773static BOOL is_fuse_file_not_parent(WINPR_ATTR_UNUSED
const void* key,
void* value,
void* arg)
1775 CliprdrFuseFile* fuse_file = value;
1776 CliprdrFuseFindParentContext* find_context = arg;
1778 if (!fuse_file->is_directory)
1781 if (strcmp(find_context->parent_path, fuse_file->filename_with_root) == 0)
1783 find_context->parent = fuse_file;
1790static CliprdrFuseFile* get_parent_directory(CliprdrFileContext* file_context,
const char* path)
1792 CliprdrFuseFindParentContext find_context = { 0 };
1794 WINPR_ASSERT(file_context);
1797 find_context.parent_path = get_parent_path(path);
1798 if (!find_context.parent_path)
1801 WINPR_ASSERT(!find_context.parent);
1803 if (HashTable_Foreach(file_context->inode_table, is_fuse_file_not_parent, &find_context))
1805 free(find_context.parent_path);
1808 WINPR_ASSERT(find_context.parent);
1810 free(find_context.parent_path);
1812 return find_context.parent;
1815static BOOL set_selection_for_clip_data_entry(CliprdrFileContext* file_context,
1816 CliprdrFuseClipDataEntry* clip_data_entry,
1819 CliprdrFuseFile* clip_data_dir = NULL;
1821 WINPR_ASSERT(file_context);
1822 WINPR_ASSERT(clip_data_entry);
1823 WINPR_ASSERT(files);
1825 clip_data_dir = clip_data_entry->clip_data_dir;
1826 WINPR_ASSERT(clip_data_dir);
1828 if (clip_data_entry->has_clip_data_id)
1829 WLog_Print(file_context->log, WLOG_DEBUG,
"Setting selection for clipDataId %u",
1830 clip_data_entry->clip_data_id);
1832 WLog_Print(file_context->log, WLOG_DEBUG,
"Setting selection");
1835 for (UINT32 i = 0; i < n_files; ++i)
1838 CliprdrFuseFile* fuse_file = NULL;
1839 char* filename = NULL;
1840 size_t path_length = 0;
1842 fuse_file = fuse_file_new();
1845 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to create FUSE file");
1846 clear_entry_selection(clip_data_entry);
1850 filename = ConvertWCharToUtf8Alloc(file->cFileName, NULL);
1853 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to convert filename");
1854 fuse_file_free(fuse_file);
1855 clear_entry_selection(clip_data_entry);
1859 for (
size_t j = 0; filename[j]; ++j)
1861 if (filename[j] ==
'\\')
1865 path_length = strlen(clip_data_dir->filename_with_root) + 1 + strlen(filename) + 1;
1866 fuse_file->filename_with_root = calloc(path_length,
sizeof(
char));
1867 if (!fuse_file->filename_with_root)
1869 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to allocate filename");
1871 fuse_file_free(fuse_file);
1872 clear_entry_selection(clip_data_entry);
1876 (void)_snprintf(fuse_file->filename_with_root, path_length,
"%s/%s",
1877 clip_data_dir->filename_with_root, filename);
1880 fuse_file->filename = strrchr(fuse_file->filename_with_root,
'/') + 1;
1882 fuse_file->parent = get_parent_directory(file_context, fuse_file->filename_with_root);
1883 if (!fuse_file->parent)
1885 WLog_Print(file_context->log, WLOG_ERROR,
"Found no parent for FUSE file");
1886 fuse_file_free(fuse_file);
1887 clear_entry_selection(clip_data_entry);
1891 if (!ArrayList_Append(fuse_file->parent->children, fuse_file))
1893 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to append FUSE file");
1894 fuse_file_free(fuse_file);
1895 clear_entry_selection(clip_data_entry);
1899 fuse_file->list_idx = i;
1900 fuse_file->ino = get_next_free_inode(file_context);
1901 fuse_file->has_clip_data_id = clip_data_entry->has_clip_data_id;
1902 fuse_file->clip_data_id = clip_data_entry->clip_data_id;
1903 if (file->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
1904 fuse_file->is_directory = TRUE;
1905 if (file->dwFileAttributes & FILE_ATTRIBUTE_READONLY)
1906 fuse_file->is_readonly = TRUE;
1907 if (file->dwFlags & FD_FILESIZE)
1909 fuse_file->size = ((UINT64)file->nFileSizeHigh << 32) + file->nFileSizeLow;
1910 fuse_file->has_size = TRUE;
1912 if (file->dwFlags & FD_WRITESTIME)
1916 filetime = file->ftLastWriteTime.dwHighDateTime;
1918 filetime += file->ftLastWriteTime.dwLowDateTime;
1920 fuse_file->last_write_time_unix =
1921 1LL * filetime / (10LL * 1000LL * 1000LL) - WIN32_FILETIME_TO_UNIX_EPOCH;
1922 fuse_file->has_last_write_time = TRUE;
1925 if (!HashTable_Insert(file_context->inode_table, (
void*)(uintptr_t)fuse_file->ino,
1928 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to insert inode into inode table");
1929 fuse_file_free(fuse_file);
1930 clear_entry_selection(clip_data_entry);
1936 if (clip_data_entry->has_clip_data_id)
1937 WLog_Print(file_context->log, WLOG_DEBUG,
"Selection set for clipDataId %u",
1938 clip_data_entry->clip_data_id);
1940 WLog_Print(file_context->log, WLOG_DEBUG,
"Selection set");
1945static BOOL update_exposed_path(CliprdrFileContext* file_context, wClipboard* clip,
1946 CliprdrFuseClipDataEntry* clip_data_entry)
1948 wClipboardDelegate* delegate = NULL;
1949 CliprdrFuseFile* clip_data_dir = NULL;
1951 WINPR_ASSERT(file_context);
1953 WINPR_ASSERT(clip_data_entry);
1955 delegate = ClipboardGetDelegate(clip);
1956 WINPR_ASSERT(delegate);
1958 clip_data_dir = clip_data_entry->clip_data_dir;
1959 WINPR_ASSERT(clip_data_dir);
1961 free(file_context->exposed_path);
1962 file_context->exposed_path = GetCombinedPath(file_context->path, clip_data_dir->filename);
1963 if (file_context->exposed_path)
1964 WLog_Print(file_context->log, WLOG_DEBUG,
"Updated exposed path to \"%s\"",
1965 file_context->exposed_path);
1967 delegate->basePath = file_context->exposed_path;
1969 return delegate->basePath != NULL;
1973BOOL cliprdr_file_context_update_server_data(CliprdrFileContext* file_context, wClipboard* clip,
1974 const void* data,
size_t size)
1976#if defined(WITH_FUSE)
1977 CliprdrFuseClipDataEntry* clip_data_entry = NULL;
1982 WINPR_ASSERT(file_context);
1984 if (size > UINT32_MAX)
1987 if (cliprdr_parse_file_list(data, (UINT32)size, &files, &n_files))
1989 WLog_Print(file_context->log, WLOG_ERROR,
"Failed to parse file list");
1993 HashTable_Lock(file_context->inode_table);
1994 HashTable_Lock(file_context->clip_data_table);
1995 if (does_server_support_clipdata_locking(file_context))
1996 clip_data_entry = HashTable_GetItemValue(
1997 file_context->clip_data_table, (
void*)(uintptr_t)file_context->current_clip_data_id);
1999 clip_data_entry = file_context->clip_data_entry_without_id;
2001 WINPR_ASSERT(clip_data_entry);
2003 clear_entry_selection(clip_data_entry);
2004 WINPR_ASSERT(!clip_data_entry->clip_data_dir);
2006 clip_data_entry->clip_data_dir =
2007 clip_data_dir_new(file_context, does_server_support_clipdata_locking(file_context),
2008 file_context->current_clip_data_id);
2009 if (!clip_data_entry->clip_data_dir)
2011 if (!update_exposed_path(file_context, clip, clip_data_entry))
2013 if (!set_selection_for_clip_data_entry(file_context, clip_data_entry, files, n_files))
2019 HashTable_Unlock(file_context->clip_data_table);
2020 HashTable_Unlock(file_context->inode_table);
2028void* cliprdr_file_context_get_context(CliprdrFileContext* file)
2031 return file->clipboard;
2034void cliprdr_file_session_terminate(CliprdrFileContext* file, BOOL stop_thread)
2039#if defined(WITH_FUSE)
2040 WINPR_ASSERT(file->fuse_stop_sync);
2042 WLog_Print(file->log, WLOG_DEBUG,
"Setting FUSE exit flag");
2043 if (file->fuse_sess)
2044 fuse_session_exit(file->fuse_sess);
2048 WLog_Print(file->log, WLOG_DEBUG,
"Setting FUSE stop event");
2049 (void)SetEvent(file->fuse_stop_sync);
2055#if defined(WITH_FUSE)
2056 WLog_Print(file->log, WLOG_DEBUG,
"Forcing FUSE to check exit flag");
2058 (void)winpr_PathFileExists(file->path);
2061void cliprdr_file_context_free(CliprdrFileContext* file)
2066#if defined(WITH_FUSE)
2067 if (file->inode_table)
2069 clear_no_cdi_entry(file);
2070 clear_all_selections(file);
2073 if (file->fuse_thread)
2075 WINPR_ASSERT(file->fuse_stop_sync);
2077 WLog_Print(file->log, WLOG_DEBUG,
"Stopping FUSE thread");
2078 cliprdr_file_session_terminate(file, TRUE);
2080 WLog_Print(file->log, WLOG_DEBUG,
"Waiting on FUSE thread");
2081 (void)WaitForSingleObject(file->fuse_thread, INFINITE);
2082 (void)CloseHandle(file->fuse_thread);
2084 if (file->fuse_stop_sync)
2085 (void)CloseHandle(file->fuse_stop_sync);
2086 if (file->fuse_start_sync)
2087 (void)CloseHandle(file->fuse_start_sync);
2089 HashTable_Free(file->request_table);
2090 HashTable_Free(file->clip_data_table);
2091 HashTable_Free(file->inode_table);
2093 HashTable_Free(file->local_streams);
2094 winpr_RemoveDirectory(file->path);
2096 free(file->exposed_path);
2100static BOOL create_base_path(CliprdrFileContext* file)
2104 char base[64] = { 0 };
2105 (void)_snprintf(base,
sizeof(base),
"com.freerdp.client.cliprdr.%" PRIu32,
2106 GetCurrentProcessId());
2108 file->path = GetKnownSubPath(KNOWN_PATH_TEMP, base);
2112 if (!winpr_PathFileExists(file->path) && !winpr_PathMakePath(file->path, 0))
2114 WLog_Print(file->log, WLOG_ERROR,
"Failed to create directory '%s'", file->path);
2120static void cliprdr_local_file_free(CliprdrLocalFile* file)
2122 const CliprdrLocalFile empty = { 0 };
2127 WLog_Print(file->context->log, WLOG_DEBUG,
"closing file %s, discarding entry", file->name);
2128 (void)fclose(file->fp);
2134static BOOL cliprdr_local_file_new(CliprdrFileContext* context, CliprdrLocalFile* f,
2137 const CliprdrLocalFile empty = { 0 };
2139 WINPR_ASSERT(context);
2143 f->context = context;
2144 f->name = winpr_str_url_decode(path, strlen(path));
2150 cliprdr_local_file_free(f);
2154static void cliprdr_local_files_free(CliprdrLocalStream* stream)
2156 WINPR_ASSERT(stream);
2158 for (
size_t x = 0; x < stream->count; x++)
2159 cliprdr_local_file_free(&stream->files[x]);
2160 free(stream->files);
2162 stream->files = NULL;
2166static void cliprdr_local_stream_free(
void* obj)
2168 CliprdrLocalStream* stream = (CliprdrLocalStream*)obj;
2170 cliprdr_local_files_free(stream);
2175static BOOL append_entry(CliprdrLocalStream* stream,
const char* path)
2177 CliprdrLocalFile* tmp = realloc(stream->files,
sizeof(CliprdrLocalFile) * (stream->count + 1));
2180 stream->files = tmp;
2181 CliprdrLocalFile* f = &stream->files[stream->count++];
2183 return cliprdr_local_file_new(stream->context, f, path);
2186static BOOL is_directory(
const char* path)
2188 WCHAR* wpath = ConvertUtf8ToWCharAlloc(path, NULL);
2193 CreateFileW(wpath, 0, FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
2196 if (hFile == INVALID_HANDLE_VALUE)
2200 const BOOL status = GetFileInformationByHandle(hFile, &fileInformation);
2201 (void)CloseHandle(hFile);
2205 return (fileInformation.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) ? TRUE : FALSE;
2208static BOOL add_directory(CliprdrLocalStream* stream,
const char* path)
2210 char* wildcardpath = GetCombinedPath(path,
"*");
2213 WCHAR* wpath = ConvertUtf8ToWCharAlloc(wildcardpath, NULL);
2219 HANDLE hFind = FindFirstFileW(wpath, &FindFileData);
2222 if (hFind == INVALID_HANDLE_VALUE)
2228 WCHAR dotbuffer[6] = { 0 };
2229 WCHAR dotdotbuffer[6] = { 0 };
2230 const WCHAR* dot = InitializeConstWCharFromUtf8(
".", dotbuffer, ARRAYSIZE(dotbuffer));
2231 const WCHAR* dotdot = InitializeConstWCharFromUtf8(
"..", dotdotbuffer, ARRAYSIZE(dotdotbuffer));
2234 if (_wcscmp(FindFileData.cFileName, dot) == 0)
2236 if (_wcscmp(FindFileData.cFileName, dotdot) == 0)
2239 char cFileName[MAX_PATH] = { 0 };
2240 (void)ConvertWCharNToUtf8(FindFileData.cFileName, ARRAYSIZE(FindFileData.cFileName),
2241 cFileName, ARRAYSIZE(cFileName));
2244 next = GetCombinedPath(path, cFileName);
2248 if (!append_entry(stream, next))
2250 if (is_directory(next))
2252 if (!add_directory(stream, next))
2255 }
while (FindNextFileW(hFind, &FindFileData));
2265static BOOL cliprdr_local_stream_update(CliprdrLocalStream* stream,
const char* data,
size_t size)
2268 WINPR_ASSERT(stream);
2272 cliprdr_local_files_free(stream);
2274 stream->files = calloc(size,
sizeof(CliprdrLocalFile));
2278 char* copy = strndup(data, size);
2282 char* saveptr = NULL;
2283 char* ptr = strtok_s(copy,
"\r\n", &saveptr);
2286 const char* name = ptr;
2287 if (strncmp(
"file:///", ptr, 8) == 0)
2289 else if (strncmp(
"file:/", ptr, 6) == 0)
2292 if (!append_entry(stream, name))
2295 if (is_directory(name))
2297 const BOOL res = add_directory(stream, name);
2301 ptr = strtok_s(NULL,
"\r\n", &saveptr);
2310CliprdrLocalStream* cliprdr_local_stream_new(CliprdrFileContext* context, UINT32 streamID,
2311 const char* data,
size_t size)
2313 WINPR_ASSERT(context);
2314 CliprdrLocalStream* stream = calloc(1,
sizeof(CliprdrLocalStream));
2318 stream->context = context;
2319 if (!cliprdr_local_stream_update(stream, data, size))
2322 stream->lockId = streamID;
2326 cliprdr_local_stream_free(stream);
2330static UINT32 UINTPointerHash(
const void*
id)
2333 return *((
const UINT32*)
id);
2336static BOOL UINTPointerCompare(
const void* pointer1,
const void* pointer2)
2338 if (!pointer1 || !pointer2)
2339 return pointer1 == pointer2;
2341 const UINT32* a = pointer1;
2342 const UINT32* b = pointer2;
2346static void* UINTPointerClone(
const void* other)
2348 const UINT32* src = other;
2352 UINT32* copy = calloc(1,
sizeof(UINT32));
2360#if defined(WITH_FUSE)
2361static CliprdrFuseFile* fuse_file_new_root(CliprdrFileContext* file_context)
2363 CliprdrFuseFile* root_dir = NULL;
2365 root_dir = fuse_file_new();
2369 root_dir->filename_with_root = calloc(2,
sizeof(
char));
2370 if (!root_dir->filename_with_root)
2372 fuse_file_free(root_dir);
2376 (void)_snprintf(root_dir->filename_with_root, 2,
"/");
2377 root_dir->filename = root_dir->filename_with_root;
2379 root_dir->ino = FUSE_ROOT_ID;
2380 root_dir->is_directory = TRUE;
2381 root_dir->is_readonly = TRUE;
2383 if (!HashTable_Insert(file_context->inode_table, (
void*)(uintptr_t)root_dir->ino, root_dir))
2385 fuse_file_free(root_dir);
2393CliprdrFileContext* cliprdr_file_context_new(
void* context)
2395 CliprdrFileContext* file = calloc(1,
sizeof(CliprdrFileContext));
2399 file->log = WLog_Get(CLIENT_TAG(
"common.cliprdr.file"));
2400 file->clipboard = context;
2402 file->local_streams = HashTable_New(FALSE);
2403 if (!file->local_streams)
2406 if (!HashTable_SetHashFunction(file->local_streams, UINTPointerHash))
2409 wObject* hkobj = HashTable_KeyObject(file->local_streams);
2410 WINPR_ASSERT(hkobj);
2411 hkobj->fnObjectEquals = UINTPointerCompare;
2412 hkobj->fnObjectFree = free;
2413 hkobj->fnObjectNew = UINTPointerClone;
2415 wObject* hobj = HashTable_ValueObject(file->local_streams);
2417 hobj->fnObjectFree = cliprdr_local_stream_free;
2419#if defined(WITH_FUSE)
2420 file->inode_table = HashTable_New(FALSE);
2421 file->clip_data_table = HashTable_New(FALSE);
2422 file->request_table = HashTable_New(FALSE);
2423 if (!file->inode_table || !file->clip_data_table || !file->request_table)
2427 wObject* ctobj = HashTable_ValueObject(file->request_table);
2428 WINPR_ASSERT(ctobj);
2429 ctobj->fnObjectFree = free;
2432 wObject* ctobj = HashTable_ValueObject(file->clip_data_table);
2433 WINPR_ASSERT(ctobj);
2434 ctobj->fnObjectFree = clip_data_entry_free;
2437 file->root_dir = fuse_file_new_root(file);
2438 if (!file->root_dir)
2442 if (!create_base_path(file))
2445#if defined(WITH_FUSE)
2446 if (!(file->fuse_start_sync = CreateEvent(NULL, TRUE, FALSE, NULL)))
2448 if (!(file->fuse_stop_sync = CreateEvent(NULL, TRUE, FALSE, NULL)))
2450 if (!(file->fuse_thread = CreateThread(NULL, 0, cliprdr_file_fuse_thread, file, 0, NULL)))
2453 if (WaitForSingleObject(file->fuse_start_sync, INFINITE) == WAIT_FAILED)
2454 WLog_Print(file->log, WLOG_ERROR,
"Failed to wait for start sync");
2459 WINPR_PRAGMA_DIAG_PUSH
2460 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2461 cliprdr_file_context_free(file);
2462 WINPR_PRAGMA_DIAG_POP
2466BOOL local_stream_discard(
const void* key,
void* value,
void* arg)
2468 CliprdrFileContext* file = arg;
2469 CliprdrLocalStream* stream = value;
2471 WINPR_ASSERT(stream);
2473 if (!stream->locked)
2474 HashTable_Remove(file->local_streams, key);
2478BOOL cliprdr_file_context_clear(CliprdrFileContext* file)
2482 WLog_Print(file->log, WLOG_DEBUG,
"clear file clipboard...");
2484 HashTable_Lock(file->local_streams);
2485 HashTable_Foreach(file->local_streams, local_stream_discard, file);
2486 HashTable_Unlock(file->local_streams);
2488 memset(file->server_data_hash, 0,
sizeof(file->server_data_hash));
2489 memset(file->client_data_hash, 0,
sizeof(file->client_data_hash));
2493BOOL cliprdr_file_context_update_client_data(CliprdrFileContext* file,
const char* data,
2499 if (!cliprdr_file_client_content_changed_and_update(file, data, size))
2502 if (!cliprdr_file_context_clear(file))
2505 UINT32 lockId = file->local_lock_id;
2507 HashTable_Lock(file->local_streams);
2508 CliprdrLocalStream* stream = HashTable_GetItemValue(file->local_streams, &lockId);
2510 WLog_Print(file->log, WLOG_DEBUG,
"update client file list (stream=%p)...", stream);
2512 rc = cliprdr_local_stream_update(stream, data, size);
2515 stream = cliprdr_local_stream_new(file, lockId, data, size);
2516 rc = HashTable_Insert(file->local_streams, &stream->lockId, stream);
2518 cliprdr_local_stream_free(stream);
2522 HashTable_Unlock(file->local_streams);
2526UINT32 cliprdr_file_context_current_flags(CliprdrFileContext* file)
2530 if ((file->file_capability_flags & CB_STREAM_FILECLIP_ENABLED) == 0)
2533 if (!file->file_formats_registered)
2536 return CB_STREAM_FILECLIP_ENABLED | CB_FILECLIP_NO_FILE_PATHS |
2537 CB_HUGE_FILE_SUPPORT_ENABLED;
2540BOOL cliprdr_file_context_set_locally_available(CliprdrFileContext* file, BOOL available)
2543 file->file_formats_registered = available;
2547BOOL cliprdr_file_context_remote_set_flags(CliprdrFileContext* file, UINT32 flags)
2550 file->file_capability_flags = flags;
2554UINT32 cliprdr_file_context_remote_get_flags(CliprdrFileContext* file)
2557 return file->file_capability_flags;
2560BOOL cliprdr_file_context_has_local_support(CliprdrFileContext* file)
2564#if defined(WITH_FUSE)
This struct contains function pointer to initialize/free objects.