2011-09-23 07:37:17 +04:00
|
|
|
/**
|
2012-10-09 07:02:04 +04:00
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
2011-09-23 07:37:17 +04:00
|
|
|
* X11 Clipboard Redirection
|
|
|
|
*
|
|
|
|
* Copyright 2010-2011 Vic Lee
|
2015-06-02 14:05:10 +03:00
|
|
|
* Copyright 2015 Thincast Technologies GmbH
|
|
|
|
* Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
|
2011-09-23 07:37:17 +04:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2022-02-16 13:20:38 +03:00
|
|
|
#include <freerdp/config.h>
|
2012-08-15 01:20:53 +04:00
|
|
|
|
2011-09-23 07:37:17 +04:00
|
|
|
#include <stdlib.h>
|
2021-01-18 10:51:51 +03:00
|
|
|
#include <errno.h>
|
|
|
|
|
2011-09-23 07:37:17 +04:00
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xatom.h>
|
2012-08-15 01:20:53 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
#ifdef WITH_XFIXES
|
|
|
|
#include <X11/extensions/Xfixes.h>
|
|
|
|
#endif
|
|
|
|
|
2012-11-20 08:49:08 +04:00
|
|
|
#include <winpr/crt.h>
|
2022-01-19 11:27:39 +03:00
|
|
|
#include <winpr/assert.h>
|
2014-10-17 05:45:47 +04:00
|
|
|
#include <winpr/image.h>
|
2013-03-29 05:13:56 +04:00
|
|
|
#include <winpr/stream.h>
|
2014-10-18 04:40:11 +04:00
|
|
|
#include <winpr/clipboard.h>
|
2020-12-27 09:08:35 +03:00
|
|
|
#include <winpr/path.h>
|
2012-11-20 08:49:08 +04:00
|
|
|
|
2023-02-22 12:13:18 +03:00
|
|
|
#include <freerdp/utils/signal.h>
|
2014-09-12 19:13:01 +04:00
|
|
|
#include <freerdp/log.h>
|
2012-10-09 04:33:58 +04:00
|
|
|
#include <freerdp/client/cliprdr.h>
|
2013-05-13 05:23:12 +04:00
|
|
|
#include <freerdp/channels/channels.h>
|
2017-04-09 02:29:51 +03:00
|
|
|
#include <freerdp/channels/cliprdr.h>
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2023-02-25 18:57:52 +03:00
|
|
|
#include <freerdp/client/client_cliprdr_file.h>
|
|
|
|
|
2011-09-23 07:37:17 +04:00
|
|
|
#include "xf_cliprdr.h"
|
2023-02-24 10:59:36 +03:00
|
|
|
#include "xf_event.h"
|
2023-03-01 18:01:08 +03:00
|
|
|
#include "xf_utils.h"
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
#define TAG CLIENT_TAG("x11.cliprdr")
|
2013-04-02 23:13:10 +04:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
#define MAX_CLIPBOARD_FORMATS 255
|
2021-01-16 10:14:36 +03:00
|
|
|
#define WIN32_FILETIME_TO_UNIX_EPOCH_USEC UINT64_C(116444736000000000)
|
2016-02-23 01:14:30 +03:00
|
|
|
|
2023-02-21 11:53:52 +03:00
|
|
|
#ifdef WITH_DEBUG_CLIPRDR
|
|
|
|
#define DEBUG_CLIPRDR(...) WLog_DBG(TAG, __VA_ARGS__)
|
|
|
|
#else
|
|
|
|
#define DEBUG_CLIPRDR(...) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
} while (0)
|
|
|
|
#endif
|
|
|
|
|
2022-02-14 16:59:22 +03:00
|
|
|
typedef struct
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-15 23:49:57 +04:00
|
|
|
Atom atom;
|
2023-02-22 09:15:59 +03:00
|
|
|
UINT32 formatToRequest;
|
|
|
|
UINT32 localFormat;
|
2014-10-16 05:30:11 +04:00
|
|
|
char* formatName;
|
2022-02-14 16:59:22 +03:00
|
|
|
} xfCliprdrFormat;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
BYTE* data;
|
|
|
|
UINT32 data_length;
|
|
|
|
} xfCachedData;
|
|
|
|
|
2024-08-12 16:55:12 +03:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
UINT32 localFormat;
|
|
|
|
UINT32 formatToRequest;
|
|
|
|
char* formatName;
|
|
|
|
} RequestedFormat;
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
struct xf_clipboard
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-15 06:24:07 +04:00
|
|
|
xfContext* xfc;
|
2011-11-01 03:15:27 +04:00
|
|
|
rdpChannels* channels;
|
2014-10-15 06:58:01 +04:00
|
|
|
CliprdrClientContext* context;
|
|
|
|
|
2014-10-18 04:40:11 +04:00
|
|
|
wClipboard* system;
|
|
|
|
|
2011-09-23 07:37:17 +04:00
|
|
|
Window root_window;
|
|
|
|
Atom clipboard_atom;
|
|
|
|
Atom property_atom;
|
|
|
|
|
2020-03-13 20:59:22 +03:00
|
|
|
Atom timestamp_property_atom;
|
|
|
|
Time selection_ownership_timestamp;
|
|
|
|
|
2015-08-14 22:57:42 +03:00
|
|
|
Atom raw_transfer_atom;
|
2015-08-19 10:59:49 +03:00
|
|
|
Atom raw_format_list_atom;
|
2015-08-14 22:57:42 +03:00
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
UINT32 numClientFormats;
|
2014-10-15 23:49:57 +04:00
|
|
|
xfCliprdrFormat clientFormats[20];
|
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
UINT32 numServerFormats;
|
2014-10-15 23:49:57 +04:00
|
|
|
CLIPRDR_FORMAT* serverFormats;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
size_t numTargets;
|
2011-09-23 07:37:17 +04:00
|
|
|
Atom targets[20];
|
2014-10-15 23:49:57 +04:00
|
|
|
|
|
|
|
int requestedFormatId;
|
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
wHashTable* cachedData;
|
|
|
|
wHashTable* cachedRawData;
|
|
|
|
|
2015-08-19 10:25:59 +03:00
|
|
|
BOOL data_raw_format;
|
2023-02-22 09:15:59 +03:00
|
|
|
|
2024-08-12 16:55:12 +03:00
|
|
|
RequestedFormat* requestedFormat;
|
2023-02-22 09:15:59 +03:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
XSelectionEvent* respond;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
|
|
|
Window owner;
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL sync;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
|
|
|
/* INCR mechanism */
|
|
|
|
Atom incr_atom;
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL incr_starts;
|
2012-10-09 11:01:37 +04:00
|
|
|
BYTE* incr_data;
|
2011-09-23 07:37:17 +04:00
|
|
|
int incr_data_length;
|
2024-07-17 15:19:58 +03:00
|
|
|
long event_mask;
|
2014-07-07 22:26:41 +04:00
|
|
|
|
2014-10-16 05:30:11 +04:00
|
|
|
/* XFixes extension */
|
2014-07-07 22:26:41 +04:00
|
|
|
int xfixes_event_base;
|
|
|
|
int xfixes_error_base;
|
|
|
|
BOOL xfixes_supported;
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
/* last sent data */
|
|
|
|
CLIPRDR_FORMAT* lastSentFormats;
|
|
|
|
UINT32 lastSentNumFormats;
|
2023-02-25 14:42:04 +03:00
|
|
|
CliprdrFileContext* file;
|
2011-09-23 07:37:17 +04:00
|
|
|
};
|
|
|
|
|
2024-01-31 11:49:16 +03:00
|
|
|
static const char mime_text_plain[] = "text/plain";
|
|
|
|
static const char mime_uri_list[] = "text/uri-list";
|
|
|
|
static const char mime_html[] = "text/html";
|
|
|
|
static const char* mime_bitmap[] = { "image/bmp", "image/x-bmp", "image/x-MS-bmp",
|
|
|
|
"image/x-win-bitmap" };
|
|
|
|
static const char mime_webp[] = "image/webp";
|
|
|
|
static const char mime_png[] = "image/png";
|
|
|
|
static const char mime_jpeg[] = "image/jpeg";
|
|
|
|
static const char mime_tiff[] = "image/tiff";
|
|
|
|
static const char* mime_images[] = { mime_webp, mime_png, mime_jpeg, mime_tiff };
|
|
|
|
|
|
|
|
static const char mime_gnome_copied_files[] = "x-special/gnome-copied-files";
|
|
|
|
static const char mime_mate_copied_files[] = "x-special/mate-copied-files";
|
|
|
|
|
|
|
|
static const char type_FileGroupDescriptorW[] = "FileGroupDescriptorW";
|
|
|
|
static const char type_HtmlFormat[] = "HTML Format";
|
2023-02-22 09:15:59 +03:00
|
|
|
|
2023-03-01 16:08:02 +03:00
|
|
|
static void xf_cliprdr_clear_cached_data(xfClipboard* clipboard);
|
2022-12-03 11:57:43 +03:00
|
|
|
static UINT xf_cliprdr_send_client_format_list(xfClipboard* clipboard, BOOL force);
|
2020-03-13 20:59:22 +03:00
|
|
|
static void xf_cliprdr_set_selection_owner(xfContext* xfc, xfClipboard* clipboard, Time timestamp);
|
2014-10-16 06:48:18 +04:00
|
|
|
|
2024-08-12 16:55:12 +03:00
|
|
|
static void requested_format_free(RequestedFormat** ppRequestedFormat)
|
|
|
|
{
|
|
|
|
if (!ppRequestedFormat)
|
|
|
|
return;
|
|
|
|
if (!(*ppRequestedFormat))
|
|
|
|
return;
|
|
|
|
|
|
|
|
free((*ppRequestedFormat)->formatName);
|
|
|
|
*ppRequestedFormat = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL requested_format_replace(RequestedFormat** ppRequestedFormat, UINT32 formatId,
|
|
|
|
const char* formatName)
|
|
|
|
{
|
|
|
|
if (!ppRequestedFormat)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
requested_format_free(ppRequestedFormat);
|
|
|
|
RequestedFormat* requested = calloc(1, sizeof(RequestedFormat));
|
|
|
|
if (!requested)
|
|
|
|
return FALSE;
|
|
|
|
requested->localFormat = formatId;
|
|
|
|
requested->formatToRequest = formatId;
|
|
|
|
if (formatName)
|
|
|
|
{
|
|
|
|
requested->formatName = _strdup(formatName);
|
|
|
|
if (!requested->formatName)
|
|
|
|
{
|
|
|
|
free(requested);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*ppRequestedFormat = requested;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL requested_format_matches(const RequestedFormat* pRequestedFormat, UINT32 formatId,
|
|
|
|
const char* formatName)
|
|
|
|
{
|
|
|
|
if (!pRequestedFormat)
|
|
|
|
return FALSE;
|
|
|
|
if (pRequestedFormat->formatToRequest != formatId)
|
|
|
|
return FALSE;
|
|
|
|
if (formatName || pRequestedFormat->formatName)
|
|
|
|
{
|
|
|
|
if (!formatName || !pRequestedFormat->formatName)
|
|
|
|
return FALSE;
|
|
|
|
if (strcmp(formatName, pRequestedFormat->formatName) != 0)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2023-09-20 16:57:39 +03:00
|
|
|
static void xf_cached_data_free(void* ptr)
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
{
|
2023-09-20 16:57:39 +03:00
|
|
|
xfCachedData* cached_data = ptr;
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
if (!cached_data)
|
|
|
|
return;
|
|
|
|
|
|
|
|
free(cached_data->data);
|
|
|
|
free(cached_data);
|
|
|
|
}
|
|
|
|
|
2023-05-23 13:05:51 +03:00
|
|
|
static xfCachedData* xf_cached_data_new(BYTE* data, UINT32 data_length)
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfCachedData* cached_data = NULL;
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
|
|
|
|
cached_data = calloc(1, sizeof(xfCachedData));
|
|
|
|
if (!cached_data)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
cached_data->data = data;
|
|
|
|
cached_data->data_length = data_length;
|
|
|
|
|
|
|
|
return cached_data;
|
|
|
|
}
|
|
|
|
|
2024-04-06 21:49:48 +03:00
|
|
|
static xfCachedData* xf_cached_data_new_copy(const BYTE* data, size_t data_length)
|
2024-01-24 10:21:47 +03:00
|
|
|
{
|
|
|
|
BYTE* copy = NULL;
|
|
|
|
if (data_length > 0)
|
|
|
|
{
|
|
|
|
copy = malloc(data_length);
|
|
|
|
if (!copy)
|
|
|
|
return NULL;
|
|
|
|
memcpy(copy, data, data_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
xfCachedData* cache = xf_cached_data_new(copy, data_length);
|
|
|
|
if (!cache)
|
|
|
|
free(copy);
|
|
|
|
return cache;
|
|
|
|
}
|
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
static void xf_clipboard_free_server_formats(xfClipboard* clipboard)
|
|
|
|
{
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
if (clipboard->serverFormats)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < clipboard->numServerFormats; i++)
|
|
|
|
{
|
|
|
|
CLIPRDR_FORMAT* format = &clipboard->serverFormats[i];
|
|
|
|
free(format->formatName);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(clipboard->serverFormats);
|
|
|
|
clipboard->serverFormats = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-06 21:49:48 +03:00
|
|
|
static BOOL xf_cliprdr_update_owner(xfClipboard* clipboard)
|
2014-10-17 05:45:47 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2024-04-06 21:49:48 +03:00
|
|
|
xfContext* xfc = clipboard->xfc;
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
2014-10-17 05:45:47 +04:00
|
|
|
|
2024-04-06 21:49:48 +03:00
|
|
|
if (!clipboard->sync)
|
|
|
|
return FALSE;
|
2014-10-17 05:45:47 +04:00
|
|
|
|
2024-04-06 21:49:48 +03:00
|
|
|
Window owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
|
|
|
|
if (clipboard->owner == owner)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
clipboard->owner = owner;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xf_cliprdr_check_owner(xfClipboard* clipboard)
|
|
|
|
{
|
|
|
|
if (xf_cliprdr_update_owner(clipboard))
|
|
|
|
xf_cliprdr_send_client_format_list(clipboard, FALSE);
|
2014-10-17 05:45:47 +04:00
|
|
|
}
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
static BOOL xf_cliprdr_is_self_owned(xfClipboard* clipboard)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2019-11-06 17:24:51 +03:00
|
|
|
return XGetSelectionOwner(xfc->display, clipboard->clipboard_atom) == xfc->drawable;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
static void xf_cliprdr_set_raw_transfer_enabled(xfClipboard* clipboard, BOOL enabled)
|
2015-08-14 22:57:42 +03:00
|
|
|
{
|
|
|
|
UINT32 data = enabled;
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2023-03-01 18:01:08 +03:00
|
|
|
LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->raw_transfer_atom,
|
|
|
|
XA_INTEGER, 32, PropModeReplace, (BYTE*)&data, 1);
|
2015-08-14 22:57:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL xf_cliprdr_is_raw_transfer_available(xfClipboard* clipboard)
|
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
Atom type = 0;
|
|
|
|
int format = 0;
|
2015-08-14 22:57:42 +03:00
|
|
|
int result = 0;
|
2024-01-23 18:49:54 +03:00
|
|
|
unsigned long length = 0;
|
|
|
|
unsigned long bytes_left = 0;
|
2015-08-14 22:57:42 +03:00
|
|
|
UINT32* data = NULL;
|
|
|
|
UINT32 is_enabled = 0;
|
|
|
|
Window owner = None;
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
2015-08-14 22:57:42 +03:00
|
|
|
owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
|
|
|
|
|
|
|
|
if (owner != None)
|
|
|
|
{
|
2023-03-01 18:25:19 +03:00
|
|
|
result = LogTagAndXGetWindowProperty(TAG, xfc->display, owner, clipboard->raw_transfer_atom,
|
|
|
|
0, 4, 0, XA_INTEGER, &type, &format, &length,
|
|
|
|
&bytes_left, (BYTE**)&data);
|
2015-08-14 22:57:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (data)
|
|
|
|
{
|
|
|
|
is_enabled = *data;
|
|
|
|
XFree(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((owner == None) || (owner == xfc->drawable))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (result != Success)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return is_enabled ? TRUE : FALSE;
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
static BOOL xf_cliprdr_formats_equal(const CLIPRDR_FORMAT* server, const xfCliprdrFormat* client)
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(server);
|
|
|
|
WINPR_ASSERT(client);
|
|
|
|
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
if (server->formatName && client->formatName)
|
|
|
|
{
|
|
|
|
/* The server may be using short format names while we store them in full form. */
|
2019-11-06 17:24:51 +03:00
|
|
|
return (0 == strncmp(server->formatName, client->formatName, strlen(server->formatName)));
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!server->formatName && !client->formatName)
|
|
|
|
{
|
2023-02-22 09:15:59 +03:00
|
|
|
return (server->formatId == client->formatToRequest);
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2021-03-10 13:36:23 +03:00
|
|
|
static const xfCliprdrFormat* xf_cliprdr_get_client_format_by_id(xfClipboard* clipboard,
|
|
|
|
UINT32 formatId)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
for (size_t index = 0; index < clipboard->numClientFormats; index++)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
const xfCliprdrFormat* format = &(clipboard->clientFormats[index]);
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
if (format->formatToRequest == formatId)
|
2014-10-15 23:49:57 +04:00
|
|
|
return format;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
2011-11-01 03:15:27 +04:00
|
|
|
|
2014-10-15 23:49:57 +04:00
|
|
|
return NULL;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2021-03-10 13:36:23 +03:00
|
|
|
static const xfCliprdrFormat* xf_cliprdr_get_client_format_by_atom(xfClipboard* clipboard,
|
|
|
|
Atom atom)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
for (UINT32 i = 0; i < clipboard->numClientFormats; i++)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
const xfCliprdrFormat* format = &(clipboard->clientFormats[i]);
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2015-08-16 11:56:36 +03:00
|
|
|
if (format->atom == atom)
|
2014-10-15 23:49:57 +04:00
|
|
|
return format;
|
2015-08-16 11:56:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-03-10 13:36:23 +03:00
|
|
|
static const CLIPRDR_FORMAT* xf_cliprdr_get_server_format_by_atom(xfClipboard* clipboard, Atom atom)
|
2015-08-16 11:56:36 +03:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
for (size_t i = 0; i < clipboard->numClientFormats; i++)
|
2015-08-16 11:56:36 +03:00
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
const xfCliprdrFormat* client_format = &(clipboard->clientFormats[i]);
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2015-08-16 11:56:36 +03:00
|
|
|
if (client_format->atom == atom)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2023-02-21 18:24:41 +03:00
|
|
|
for (size_t j = 0; j < clipboard->numServerFormats; j++)
|
2015-08-16 11:56:36 +03:00
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
const CLIPRDR_FORMAT* server_format = &(clipboard->serverFormats[j]);
|
2015-08-16 11:56:36 +03:00
|
|
|
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
if (xf_cliprdr_formats_equal(server_format, client_format))
|
2015-08-16 11:56:36 +03:00
|
|
|
return server_format;
|
|
|
|
}
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
}
|
2011-11-01 03:15:27 +04:00
|
|
|
|
2014-10-15 23:49:57 +04:00
|
|
|
return NULL;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2023-02-22 09:15:59 +03:00
|
|
|
static UINT xf_cliprdr_send_data_request(xfClipboard* clipboard, UINT32 formatId,
|
|
|
|
const xfCliprdrFormat* cformat)
|
2014-10-17 05:45:47 +04:00
|
|
|
{
|
2019-01-23 18:22:29 +03:00
|
|
|
CLIPRDR_FORMAT_DATA_REQUEST request = { 0 };
|
2014-10-17 05:45:47 +04:00
|
|
|
request.requestedFormatId = formatId;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
DEBUG_CLIPRDR("requesting format 0x%08" PRIx32 " [%s] {local 0x%08" PRIx32 "} [%s]", formatId,
|
|
|
|
ClipboardGetFormatIdString(formatId), cformat->localFormat, cformat->formatName);
|
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(clipboard->context);
|
|
|
|
WINPR_ASSERT(clipboard->context->ClientFormatDataRequest);
|
2019-11-06 17:24:51 +03:00
|
|
|
return clipboard->context->ClientFormatDataRequest(clipboard->context, &request);
|
2014-10-17 05:45:47 +04:00
|
|
|
}
|
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2023-02-22 09:15:59 +03:00
|
|
|
static UINT xf_cliprdr_send_data_response(xfClipboard* clipboard, const xfCliprdrFormat* format,
|
|
|
|
const BYTE* data, size_t size)
|
2014-10-17 05:45:47 +04:00
|
|
|
{
|
2019-01-23 18:22:29 +03:00
|
|
|
CLIPRDR_FORMAT_DATA_RESPONSE response = { 0 };
|
2020-11-26 18:34:28 +03:00
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2020-11-26 18:34:28 +03:00
|
|
|
/* No request currently pending, do not send a response. */
|
|
|
|
if (clipboard->requestedFormatId < 0)
|
|
|
|
return CHANNEL_RC_OK;
|
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
if (size == 0)
|
|
|
|
{
|
|
|
|
if (format)
|
|
|
|
DEBUG_CLIPRDR("send CB_RESPONSE_FAIL response {format 0x%08" PRIx32
|
|
|
|
" [%s] {local 0x%08" PRIx32 "} [%s]",
|
|
|
|
format->formatToRequest,
|
|
|
|
ClipboardGetFormatIdString(format->formatToRequest), format->localFormat,
|
|
|
|
format->formatName);
|
|
|
|
else
|
|
|
|
DEBUG_CLIPRDR("send CB_RESPONSE_FAIL response");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WINPR_ASSERT(format);
|
|
|
|
DEBUG_CLIPRDR("send response format 0x%08" PRIx32 " [%s] {local 0x%08" PRIx32 "} [%s]",
|
|
|
|
format->formatToRequest, ClipboardGetFormatIdString(format->formatToRequest),
|
|
|
|
format->localFormat, format->formatName);
|
|
|
|
}
|
2020-11-26 18:34:28 +03:00
|
|
|
/* Request handled, reset to invalid */
|
|
|
|
clipboard->requestedFormatId = -1;
|
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
response.common.msgFlags = (data) ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
|
|
|
|
response.common.dataLen = size;
|
2014-10-17 05:45:47 +04:00
|
|
|
response.requestedFormatData = data;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard->context);
|
|
|
|
WINPR_ASSERT(clipboard->context->ClientFormatDataResponse);
|
2019-11-06 17:24:51 +03:00
|
|
|
return clipboard->context->ClientFormatDataResponse(clipboard->context, &response);
|
2014-10-17 05:45:47 +04:00
|
|
|
}
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
static wStream* xf_cliprdr_serialize_server_format_list(xfClipboard* clipboard)
|
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
UINT32 formatCount = 0;
|
2015-08-19 10:59:49 +03:00
|
|
|
wStream* s = NULL;
|
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
/* Typical MS Word format list is about 80 bytes long. */
|
|
|
|
if (!(s = Stream_New(NULL, 128)))
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "failed to allocate serialized format list");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If present, the last format is always synthetic CF_RAW. Do not include it. */
|
2019-11-06 17:24:51 +03:00
|
|
|
formatCount = (clipboard->numServerFormats > 0) ? clipboard->numServerFormats - 1 : 0;
|
2015-08-19 10:59:49 +03:00
|
|
|
Stream_Write_UINT32(s, formatCount);
|
|
|
|
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 i = 0; i < formatCount; i++)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
|
|
|
CLIPRDR_FORMAT* format = &clipboard->serverFormats[i];
|
|
|
|
size_t name_length = format->formatName ? strlen(format->formatName) : 0;
|
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
DEBUG_CLIPRDR("server announced 0x%08" PRIx32 " [%s][%s]", format->formatId,
|
|
|
|
ClipboardGetFormatIdString(format->formatId), format->formatName);
|
2015-08-19 10:59:49 +03:00
|
|
|
if (!Stream_EnsureRemainingCapacity(s, sizeof(UINT32) + name_length + 1))
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "failed to expand serialized format list");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
Stream_Write_UINT32(s, format->formatId);
|
2017-11-14 15:51:37 +03:00
|
|
|
|
|
|
|
if (format->formatName)
|
|
|
|
Stream_Write(s, format->formatName, name_length);
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
Stream_Write_UINT8(s, '\0');
|
|
|
|
}
|
|
|
|
|
|
|
|
Stream_SealLength(s);
|
|
|
|
return s;
|
|
|
|
error:
|
|
|
|
Stream_Free(s, TRUE);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
static CLIPRDR_FORMAT* xf_cliprdr_parse_server_format_list(BYTE* data, size_t length,
|
|
|
|
UINT32* numFormats)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
|
|
|
wStream* s = NULL;
|
|
|
|
CLIPRDR_FORMAT* formats = NULL;
|
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(data || (length == 0));
|
|
|
|
WINPR_ASSERT(numFormats);
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
if (!(s = Stream_New(data, length)))
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "failed to allocate stream for parsing serialized format list");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2022-04-19 15:29:17 +03:00
|
|
|
if (!Stream_CheckAndLogRequiredLength(TAG, s, sizeof(UINT32)))
|
2015-08-19 10:59:49 +03:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
Stream_Read_UINT32(s, *numFormats);
|
|
|
|
|
2016-02-23 01:14:30 +03:00
|
|
|
if (*numFormats > MAX_CLIPBOARD_FORMATS)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
WLog_ERR(TAG, "unexpectedly large number of formats: %" PRIu32 "", *numFormats);
|
2016-02-23 01:14:30 +03:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (!(formats = (CLIPRDR_FORMAT*)calloc(*numFormats, sizeof(CLIPRDR_FORMAT))))
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "failed to allocate format list");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 i = 0; i < *numFormats; i++)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
2016-02-23 01:14:30 +03:00
|
|
|
const char* formatName = NULL;
|
|
|
|
size_t formatNameLength = 0;
|
|
|
|
|
2022-04-19 15:29:17 +03:00
|
|
|
if (!Stream_CheckAndLogRequiredLength(TAG, s, sizeof(UINT32)))
|
2015-08-19 10:59:49 +03:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
Stream_Read_UINT32(s, formats[i].formatId);
|
2019-11-06 17:24:51 +03:00
|
|
|
formatName = (const char*)Stream_Pointer(s);
|
2016-02-23 01:14:30 +03:00
|
|
|
formatNameLength = strnlen(formatName, Stream_GetRemainingLength(s));
|
|
|
|
|
|
|
|
if (formatNameLength == Stream_GetRemainingLength(s))
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
WLog_ERR(TAG, "missing terminating null byte, %" PRIuz " bytes left to read",
|
2016-10-06 14:31:25 +03:00
|
|
|
formatNameLength);
|
2016-02-23 01:14:30 +03:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
formats[i].formatName = strndup(formatName, formatNameLength);
|
|
|
|
Stream_Seek(s, formatNameLength + 1);
|
2015-08-19 10:59:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Stream_Free(s, FALSE);
|
|
|
|
return formats;
|
|
|
|
error:
|
|
|
|
Stream_Free(s, FALSE);
|
|
|
|
free(formats);
|
|
|
|
*numFormats = 0;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xf_cliprdr_free_formats(CLIPRDR_FORMAT* formats, UINT32 numFormats)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(formats || (numFormats == 0));
|
|
|
|
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 i = 0; i < numFormats; i++)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
|
|
|
free(formats[i].formatName);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(formats);
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
static CLIPRDR_FORMAT* xf_cliprdr_get_raw_server_formats(xfClipboard* clipboard, UINT32* numFormats)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
|
|
|
Atom type = None;
|
|
|
|
int format = 0;
|
|
|
|
unsigned long length = 0;
|
2024-01-23 18:49:54 +03:00
|
|
|
unsigned long remaining = 0;
|
2015-08-19 10:59:49 +03:00
|
|
|
BYTE* data = NULL;
|
|
|
|
CLIPRDR_FORMAT* formats = NULL;
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(numFormats);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
*numFormats = 0;
|
2024-04-06 21:49:48 +03:00
|
|
|
|
|
|
|
Window owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
|
|
|
|
LogTagAndXGetWindowProperty(TAG, xfc->display, owner, clipboard->raw_format_list_atom, 0, 4096,
|
|
|
|
False, clipboard->raw_format_list_atom, &type, &format, &length,
|
|
|
|
&remaining, &data);
|
2015-08-19 10:59:49 +03:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (data && length > 0 && format == 8 && type == clipboard->raw_format_list_atom)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
|
|
|
formats = xf_cliprdr_parse_server_format_list(data, length, numFormats);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-10-06 14:31:25 +03:00
|
|
|
WLog_ERR(TAG,
|
2019-11-06 17:24:51 +03:00
|
|
|
"failed to retrieve raw format list: data=%p, length=%lu, format=%d, type=%lu "
|
|
|
|
"(expected=%lu)",
|
|
|
|
(void*)data, length, format, (unsigned long)type,
|
|
|
|
(unsigned long)clipboard->raw_format_list_atom);
|
2015-08-19 10:59:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (data)
|
|
|
|
XFree(data);
|
|
|
|
|
|
|
|
return formats;
|
|
|
|
}
|
|
|
|
|
2024-07-17 11:30:34 +03:00
|
|
|
static BOOL xf_cliprdr_should_add_format(const CLIPRDR_FORMAT* formats, size_t count,
|
|
|
|
const xfCliprdrFormat* xformat)
|
|
|
|
{
|
|
|
|
WINPR_ASSERT(formats);
|
|
|
|
|
|
|
|
if (!xformat)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
for (size_t x = 0; x < count; x++)
|
|
|
|
{
|
|
|
|
const CLIPRDR_FORMAT* format = &formats[x];
|
|
|
|
if (format->formatId == xformat->formatToRequest)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
static CLIPRDR_FORMAT* xf_cliprdr_get_formats_from_targets(xfClipboard* clipboard,
|
|
|
|
UINT32* numFormats)
|
2014-10-17 05:45:47 +04:00
|
|
|
{
|
2023-06-05 13:16:57 +03:00
|
|
|
Atom atom = None;
|
2014-10-17 05:45:47 +04:00
|
|
|
BYTE* data = NULL;
|
2023-06-05 13:16:57 +03:00
|
|
|
int format_property = 0;
|
|
|
|
unsigned long length = 0;
|
|
|
|
unsigned long bytes_left = 0;
|
2014-10-17 05:45:47 +04:00
|
|
|
CLIPRDR_FORMAT* formats = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(numFormats);
|
|
|
|
|
2023-06-05 13:16:57 +03:00
|
|
|
xfContext* xfc = clipboard->xfc;
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
*numFormats = 0;
|
2023-03-01 18:25:19 +03:00
|
|
|
LogTagAndXGetWindowProperty(TAG, xfc->display, xfc->drawable, clipboard->property_atom, 0, 200,
|
|
|
|
0, XA_ATOM, &atom, &format_property, &length, &bytes_left, &data);
|
2014-10-17 05:45:47 +04:00
|
|
|
|
|
|
|
if (length > 0)
|
2015-10-20 22:28:29 +03:00
|
|
|
{
|
|
|
|
if (!data)
|
|
|
|
{
|
2016-11-25 14:40:11 +03:00
|
|
|
WLog_ERR(TAG, "XGetWindowProperty set length = %lu but data is NULL", length);
|
2015-10-20 22:28:29 +03:00
|
|
|
goto out;
|
|
|
|
}
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (!(formats = (CLIPRDR_FORMAT*)calloc(length, sizeof(CLIPRDR_FORMAT))))
|
2015-10-20 22:28:29 +03:00
|
|
|
{
|
2016-11-25 14:40:11 +03:00
|
|
|
WLog_ERR(TAG, "failed to allocate %lu CLIPRDR_FORMAT structs", length);
|
2015-10-20 22:28:29 +03:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
2014-10-17 05:45:47 +04:00
|
|
|
|
2023-06-05 13:16:57 +03:00
|
|
|
for (unsigned long i = 0; i < length; i++)
|
2014-10-17 05:45:47 +04:00
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
Atom tatom = ((Atom*)data)[i];
|
|
|
|
const xfCliprdrFormat* format = xf_cliprdr_get_client_format_by_atom(clipboard, tatom);
|
2014-10-17 05:45:47 +04:00
|
|
|
|
2024-07-17 11:30:34 +03:00
|
|
|
if (xf_cliprdr_should_add_format(formats, *numFormats, format))
|
2014-10-17 05:45:47 +04:00
|
|
|
{
|
2024-07-17 11:30:34 +03:00
|
|
|
CLIPRDR_FORMAT* cformat = &formats[*numFormats];
|
|
|
|
cformat->formatId = format->formatToRequest;
|
|
|
|
if (format->formatName)
|
|
|
|
{
|
|
|
|
cformat->formatName = _strdup(format->formatName);
|
|
|
|
WINPR_ASSERT(cformat->formatName);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
cformat->formatName = NULL;
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
*numFormats += 1;
|
2014-10-17 05:45:47 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
out:
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
if (data)
|
|
|
|
XFree(data);
|
|
|
|
|
|
|
|
return formats;
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
static CLIPRDR_FORMAT* xf_cliprdr_get_client_formats(xfClipboard* clipboard, UINT32* numFormats)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
|
|
|
CLIPRDR_FORMAT* formats = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(numFormats);
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
*numFormats = 0;
|
|
|
|
|
|
|
|
if (xf_cliprdr_is_raw_transfer_available(clipboard))
|
|
|
|
{
|
|
|
|
formats = xf_cliprdr_get_raw_server_formats(clipboard, numFormats);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*numFormats == 0)
|
|
|
|
{
|
|
|
|
xf_cliprdr_free_formats(formats, *numFormats);
|
|
|
|
formats = xf_cliprdr_get_formats_from_targets(clipboard, numFormats);
|
|
|
|
}
|
|
|
|
|
|
|
|
return formats;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xf_cliprdr_provide_server_format_list(xfClipboard* clipboard)
|
|
|
|
{
|
|
|
|
wStream* formats = NULL;
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
2015-08-19 10:59:49 +03:00
|
|
|
formats = xf_cliprdr_serialize_server_format_list(clipboard);
|
|
|
|
|
|
|
|
if (formats)
|
|
|
|
{
|
2023-03-01 18:01:08 +03:00
|
|
|
LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->raw_format_list_atom,
|
|
|
|
clipboard->raw_format_list_atom, 8, PropModeReplace,
|
|
|
|
Stream_Buffer(formats), Stream_Length(formats));
|
2015-08-19 10:59:49 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-01 18:17:03 +03:00
|
|
|
LogTagAndXDeleteProperty(TAG, xfc->display, xfc->drawable, clipboard->raw_format_list_atom);
|
2015-08-19 10:59:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Stream_Free(formats, TRUE);
|
|
|
|
}
|
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
static BOOL xf_clipboard_format_equal(const CLIPRDR_FORMAT* a, const CLIPRDR_FORMAT* b)
|
|
|
|
{
|
2022-04-28 09:00:39 +03:00
|
|
|
WINPR_ASSERT(a);
|
|
|
|
WINPR_ASSERT(b);
|
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
if (a->formatId != b->formatId)
|
|
|
|
return FALSE;
|
|
|
|
if (!a->formatName && !b->formatName)
|
|
|
|
return TRUE;
|
2022-04-28 09:00:39 +03:00
|
|
|
if (!a->formatName || !b->formatName)
|
|
|
|
return FALSE;
|
2020-11-26 19:38:04 +03:00
|
|
|
return strcmp(a->formatName, b->formatName) == 0;
|
|
|
|
}
|
2022-07-01 12:04:22 +03:00
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
static BOOL xf_clipboard_changed(xfClipboard* clipboard, const CLIPRDR_FORMAT* formats,
|
|
|
|
UINT32 numFormats)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(formats || (numFormats == 0));
|
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
if (clipboard->lastSentNumFormats != numFormats)
|
|
|
|
return TRUE;
|
|
|
|
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 x = 0; x < numFormats; x++)
|
2020-11-26 19:38:04 +03:00
|
|
|
{
|
|
|
|
const CLIPRDR_FORMAT* cur = &clipboard->lastSentFormats[x];
|
|
|
|
BOOL contained = FALSE;
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 y = 0; y < numFormats; y++)
|
2020-11-26 19:38:04 +03:00
|
|
|
{
|
|
|
|
if (xf_clipboard_format_equal(cur, &formats[y]))
|
|
|
|
{
|
|
|
|
contained = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!contained)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xf_clipboard_formats_free(xfClipboard* clipboard)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
xf_cliprdr_free_formats(clipboard->lastSentFormats, clipboard->lastSentNumFormats);
|
|
|
|
clipboard->lastSentFormats = NULL;
|
|
|
|
clipboard->lastSentNumFormats = 0;
|
|
|
|
}
|
2022-07-01 12:04:22 +03:00
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
static BOOL xf_clipboard_copy_formats(xfClipboard* clipboard, const CLIPRDR_FORMAT* formats,
|
|
|
|
UINT32 numFormats)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(formats || (numFormats == 0));
|
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
xf_clipboard_formats_free(clipboard);
|
2024-09-11 21:39:01 +03:00
|
|
|
if (numFormats > 0)
|
|
|
|
clipboard->lastSentFormats = calloc(numFormats, sizeof(CLIPRDR_FORMAT));
|
2020-11-26 19:38:04 +03:00
|
|
|
if (!clipboard->lastSentFormats)
|
|
|
|
return FALSE;
|
|
|
|
clipboard->lastSentNumFormats = numFormats;
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 x = 0; x < numFormats; x++)
|
2020-11-26 19:38:04 +03:00
|
|
|
{
|
|
|
|
CLIPRDR_FORMAT* lcur = &clipboard->lastSentFormats[x];
|
|
|
|
const CLIPRDR_FORMAT* cur = &formats[x];
|
|
|
|
*lcur = *cur;
|
|
|
|
if (cur->formatName)
|
|
|
|
lcur->formatName = _strdup(cur->formatName);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-11-26 18:34:28 +03:00
|
|
|
static UINT xf_cliprdr_send_format_list(xfClipboard* clipboard, const CLIPRDR_FORMAT* formats,
|
2022-12-03 11:57:43 +03:00
|
|
|
UINT32 numFormats, BOOL force)
|
2015-08-19 10:59:49 +03:00
|
|
|
{
|
2022-04-27 22:02:18 +03:00
|
|
|
union
|
|
|
|
{
|
|
|
|
const CLIPRDR_FORMAT* cpv;
|
|
|
|
CLIPRDR_FORMAT* pv;
|
|
|
|
} cnv = { .cpv = formats };
|
2024-05-22 17:30:21 +03:00
|
|
|
const CLIPRDR_FORMAT_LIST formatList = { .common.msgFlags = 0,
|
2022-04-27 22:02:18 +03:00
|
|
|
.numFormats = numFormats,
|
|
|
|
.formats = cnv.pv,
|
2022-07-01 12:04:22 +03:00
|
|
|
.common.msgType = CB_FORMAT_LIST };
|
2024-01-23 18:49:54 +03:00
|
|
|
UINT ret = 0;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(formats || (numFormats == 0));
|
2020-11-26 18:34:28 +03:00
|
|
|
|
2022-12-03 11:57:43 +03:00
|
|
|
if (!force && !xf_clipboard_changed(clipboard, formats, numFormats))
|
2020-11-26 19:38:04 +03:00
|
|
|
return CHANNEL_RC_OK;
|
|
|
|
|
2023-03-13 14:53:40 +03:00
|
|
|
#if defined(WITH_DEBUG_CLIPRDR)
|
2023-02-22 09:15:59 +03:00
|
|
|
for (UINT32 x = 0; x < numFormats; x++)
|
|
|
|
{
|
|
|
|
const CLIPRDR_FORMAT* format = &formats[x];
|
|
|
|
DEBUG_CLIPRDR("announcing format 0x%08" PRIx32 " [%s] [%s]", format->formatId,
|
|
|
|
ClipboardGetFormatIdString(format->formatId), format->formatName);
|
|
|
|
}
|
2023-03-13 14:53:40 +03:00
|
|
|
#endif
|
2023-02-22 09:15:59 +03:00
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
xf_clipboard_copy_formats(clipboard, formats, numFormats);
|
2020-11-26 18:34:28 +03:00
|
|
|
/* Ensure all pending requests are answered. */
|
2023-02-22 09:15:59 +03:00
|
|
|
xf_cliprdr_send_data_response(clipboard, NULL, NULL, 0);
|
2023-03-01 16:08:02 +03:00
|
|
|
|
|
|
|
xf_cliprdr_clear_cached_data(clipboard);
|
2022-07-01 12:04:22 +03:00
|
|
|
|
X11/cliprdr: Rework server to client clipboard handling
The purpose of clipboard data locking is to make the other peer
retaining the current file list until a pending paste operation is done,
even though the clipboard selection changed.
As it may be difficult to determine, when a lock is needed, imitate the
same behaviour as mstsc:
When the server side supports clipboard data locking, always attempt to
lock the file list on the server regardless of what is advertised in a
FormatList PDU.
The Lock Clipboard Data PDU can even be already sent, before the
Format List Response PDU is sent.
This is also what mstsc, does: First, lock the new (potential) file
list, then unlock the file list, when the pending paste operation is
done.
So, rework the current clipboard implementation in that direction.
Since the implementation for timeouts for old file lists is a bit hard,
for now always force unlock pending locks, when the selection changes.
However, timeouts for old file lists can still be added in the future.
The reworked clipboard handling is done with the help of three hash
tables:
1. The inode table: This hash table manages all inodes for each file.
The keys in this table are the inodes themselves, while the values
the files and directories and their attributes (file size, last write
time, etc.).
2. The clipdata table: This table manages the locks for each file list.
The keys in this table represent the clip data id and the values the
clip data entries, which have a reference to the clip data dir, a
directory containing the whole selection, and some helper attributes,
like the clip data id itself.
3. The request table: Every file size or file range request is managed
here. When a FileContentsRequest is made, its stream id with the
respective details are added to this table. When a response is
received, these details can then be easily looked up here.
2023-07-19 11:38:33 +03:00
|
|
|
ret = cliprdr_file_context_notify_new_client_format_list(clipboard->file);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard->context);
|
|
|
|
WINPR_ASSERT(clipboard->context->ClientFormatList);
|
2020-11-26 18:34:28 +03:00
|
|
|
return clipboard->context->ClientFormatList(clipboard->context, &formatList);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xf_cliprdr_get_requested_targets(xfClipboard* clipboard)
|
|
|
|
{
|
|
|
|
UINT32 numFormats = 0;
|
2022-07-01 12:04:22 +03:00
|
|
|
CLIPRDR_FORMAT* formats = xf_cliprdr_get_client_formats(clipboard, &numFormats);
|
2022-12-03 11:57:43 +03:00
|
|
|
xf_cliprdr_send_format_list(clipboard, formats, numFormats, FALSE);
|
2015-08-19 10:59:49 +03:00
|
|
|
xf_cliprdr_free_formats(formats, numFormats);
|
2014-10-17 05:45:47 +04:00
|
|
|
}
|
|
|
|
|
2024-07-17 15:19:58 +03:00
|
|
|
static void xf_cliprdr_process_requested_data(xfClipboard* clipboard, BOOL hasData,
|
|
|
|
const BYTE* data, size_t size)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
BOOL bSuccess = 0;
|
|
|
|
UINT32 SrcSize = 0;
|
2021-09-08 16:47:03 +03:00
|
|
|
UINT32 DstSize = 0;
|
2024-08-12 16:55:12 +03:00
|
|
|
INT64 srcFormatId = -1;
|
2014-10-18 04:40:11 +04:00
|
|
|
BYTE* pDstData = NULL;
|
2024-01-23 18:49:54 +03:00
|
|
|
const xfCliprdrFormat* format = NULL;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2014-10-18 04:40:11 +04:00
|
|
|
if (clipboard->incr_starts && hasData)
|
2011-09-23 07:37:17 +04:00
|
|
|
return;
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
format = xf_cliprdr_get_client_format_by_id(clipboard, clipboard->requestedFormatId);
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2014-10-18 04:40:11 +04:00
|
|
|
if (!hasData || !data || !format)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2023-02-22 09:15:59 +03:00
|
|
|
xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
|
2011-09-23 07:37:17 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
switch (format->formatToRequest)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2015-08-19 10:25:59 +03:00
|
|
|
case CF_RAW:
|
|
|
|
srcFormatId = CF_RAW;
|
|
|
|
break;
|
|
|
|
|
2014-10-17 06:20:12 +04:00
|
|
|
case CF_TEXT:
|
2014-12-26 19:30:09 +03:00
|
|
|
case CF_OEMTEXT:
|
2014-10-17 06:20:12 +04:00
|
|
|
case CF_UNICODETEXT:
|
2024-07-17 15:19:58 +03:00
|
|
|
size = strlen((const char*)data) + 1;
|
2023-02-22 09:15:59 +03:00
|
|
|
srcFormatId = format->localFormat;
|
2011-09-23 07:37:17 +04:00
|
|
|
break;
|
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
default:
|
|
|
|
srcFormatId = format->localFormat;
|
2011-09-23 07:37:17 +04:00
|
|
|
break;
|
2023-02-22 09:15:59 +03:00
|
|
|
}
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2024-08-12 16:55:12 +03:00
|
|
|
if (srcFormatId < 0)
|
2023-02-22 09:15:59 +03:00
|
|
|
{
|
|
|
|
xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
|
|
|
|
return;
|
2014-10-18 04:40:11 +04:00
|
|
|
}
|
|
|
|
|
2023-02-28 13:37:42 +03:00
|
|
|
ClipboardLock(clipboard->system);
|
2019-11-06 17:24:51 +03:00
|
|
|
SrcSize = (UINT32)size;
|
2024-08-12 16:55:12 +03:00
|
|
|
bSuccess = ClipboardSetData(clipboard->system, (UINT32)srcFormatId, data, SrcSize);
|
2014-10-18 04:40:11 +04:00
|
|
|
|
2015-08-19 10:25:59 +03:00
|
|
|
if (bSuccess)
|
2014-07-07 22:26:41 +04:00
|
|
|
{
|
2014-10-18 04:40:11 +04:00
|
|
|
DstSize = 0;
|
2023-02-22 09:15:59 +03:00
|
|
|
pDstData = (BYTE*)ClipboardGetData(clipboard->system, format->formatToRequest, &DstSize);
|
2014-07-07 22:26:41 +04:00
|
|
|
}
|
2023-02-28 13:37:42 +03:00
|
|
|
ClipboardUnlock(clipboard->system);
|
2014-10-18 04:40:11 +04:00
|
|
|
|
|
|
|
if (!pDstData)
|
|
|
|
{
|
2023-02-22 09:15:59 +03:00
|
|
|
xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
|
2014-10-18 04:40:11 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-04-09 02:29:51 +03:00
|
|
|
/*
|
|
|
|
* File lists require a bit of postprocessing to convert them from WinPR's FILDESCRIPTOR
|
|
|
|
* format to CLIPRDR_FILELIST expected by the server.
|
|
|
|
*
|
|
|
|
* We check for "FileGroupDescriptorW" format being registered (i.e., nonzero) in order
|
|
|
|
* to not process CF_RAW as a file list in case WinPR does not support file transfers.
|
|
|
|
*/
|
2023-03-22 11:53:30 +03:00
|
|
|
ClipboardLock(clipboard->system);
|
2023-02-22 09:15:59 +03:00
|
|
|
if (format->formatToRequest &&
|
|
|
|
(format->formatToRequest ==
|
|
|
|
ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW)))
|
2017-04-09 02:29:51 +03:00
|
|
|
{
|
|
|
|
UINT error = NO_ERROR;
|
2020-09-17 16:21:45 +03:00
|
|
|
FILEDESCRIPTORW* file_array = (FILEDESCRIPTORW*)pDstData;
|
|
|
|
UINT32 file_count = DstSize / sizeof(FILEDESCRIPTORW);
|
2017-04-09 02:29:51 +03:00
|
|
|
pDstData = NULL;
|
|
|
|
DstSize = 0;
|
2023-02-25 18:57:52 +03:00
|
|
|
|
|
|
|
const UINT32 flags = cliprdr_file_context_remote_get_flags(clipboard->file);
|
|
|
|
error = cliprdr_serialize_file_list_ex(flags, file_array, file_count, &pDstData, &DstSize);
|
2017-04-09 02:29:51 +03:00
|
|
|
|
|
|
|
if (error)
|
|
|
|
WLog_ERR(TAG, "failed to serialize CLIPRDR_FILELIST: 0x%08X", error);
|
2024-07-17 10:49:40 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
UINT32 formatId = ClipboardGetFormatId(clipboard->system, mime_uri_list);
|
|
|
|
UINT32 url_size = 0;
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2024-07-17 10:49:40 +03:00
|
|
|
char* url = ClipboardGetData(clipboard->system, formatId, &url_size);
|
|
|
|
cliprdr_file_context_update_client_data(clipboard->file, url, url_size);
|
|
|
|
free(url);
|
|
|
|
}
|
2023-02-25 18:57:52 +03:00
|
|
|
|
2017-04-09 02:29:51 +03:00
|
|
|
free(file_array);
|
|
|
|
}
|
2023-03-22 11:53:30 +03:00
|
|
|
ClipboardUnlock(clipboard->system);
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
xf_cliprdr_send_data_response(clipboard, format, pDstData, DstSize);
|
2014-12-04 21:19:10 +03:00
|
|
|
free(pDstData);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2024-07-17 15:19:58 +03:00
|
|
|
static BOOL xf_add_input_flags(xfClipboard* clipboard, long mask)
|
|
|
|
{
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfContext* xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
|
|
|
XWindowAttributes attr = { 0 };
|
|
|
|
XGetWindowAttributes(xfc->display, xfc->drawable, &attr);
|
|
|
|
if ((attr.all_event_masks & mask) == 0)
|
|
|
|
clipboard->event_mask = attr.all_event_masks;
|
|
|
|
|
|
|
|
XSelectInput(xfc->display, xfc->drawable, attr.all_event_masks | mask);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL xf_restore_input_flags(xfClipboard* clipboard)
|
|
|
|
{
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfContext* xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
|
|
|
if (clipboard->event_mask != 0)
|
|
|
|
{
|
|
|
|
XSelectInput(xfc->display, xfc->drawable, clipboard->event_mask);
|
|
|
|
clipboard->event_mask = 0;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
static BOOL xf_cliprdr_get_requested_data(xfClipboard* clipboard, Atom target)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
Atom type = 0;
|
2012-10-09 11:01:37 +04:00
|
|
|
BYTE* data = NULL;
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL has_data = FALSE;
|
2024-01-23 18:49:54 +03:00
|
|
|
int format_property = 0;
|
|
|
|
unsigned long dummy = 0;
|
|
|
|
unsigned long length = 0;
|
|
|
|
unsigned long bytes_left = 0;
|
|
|
|
const xfCliprdrFormat* format = NULL;
|
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2020-11-26 18:34:28 +03:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
format = xf_cliprdr_get_client_format_by_id(clipboard, clipboard->requestedFormatId);
|
2014-10-15 23:49:57 +04:00
|
|
|
|
|
|
|
if (!format || (format->atom != target))
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2023-02-22 09:15:59 +03:00
|
|
|
xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
|
2012-10-09 10:31:28 +04:00
|
|
|
return FALSE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2023-03-01 18:25:19 +03:00
|
|
|
LogTagAndXGetWindowProperty(TAG, xfc->display, xfc->drawable, clipboard->property_atom, 0, 0, 0,
|
|
|
|
target, &type, &format_property, &length, &bytes_left, &data);
|
2011-11-01 03:15:27 +04:00
|
|
|
|
2011-09-23 07:37:17 +04:00
|
|
|
if (data)
|
|
|
|
{
|
|
|
|
XFree(data);
|
|
|
|
data = NULL;
|
|
|
|
}
|
2014-12-04 21:19:10 +03:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
if (bytes_left <= 0 && !clipboard->incr_starts)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
|
|
|
}
|
2014-10-15 06:24:07 +04:00
|
|
|
else if (type == clipboard->incr_atom)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-15 06:24:07 +04:00
|
|
|
clipboard->incr_starts = TRUE;
|
2014-10-16 05:30:11 +04:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
if (clipboard->incr_data)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-15 06:24:07 +04:00
|
|
|
free(clipboard->incr_data);
|
|
|
|
clipboard->incr_data = NULL;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
2014-10-16 05:30:11 +04:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
clipboard->incr_data_length = 0;
|
2014-10-16 05:30:11 +04:00
|
|
|
has_data = TRUE; /* data will be followed in PropertyNotify event */
|
2024-07-17 15:19:58 +03:00
|
|
|
xf_add_input_flags(clipboard, PropertyChangeMask);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (bytes_left <= 0)
|
|
|
|
{
|
|
|
|
/* INCR finish */
|
2014-10-15 06:24:07 +04:00
|
|
|
data = clipboard->incr_data;
|
|
|
|
clipboard->incr_data = NULL;
|
|
|
|
bytes_left = clipboard->incr_data_length;
|
|
|
|
clipboard->incr_data_length = 0;
|
2024-07-17 15:19:58 +03:00
|
|
|
clipboard->incr_starts = FALSE;
|
|
|
|
|
|
|
|
/* Restore previous event mask */
|
|
|
|
xf_restore_input_flags(clipboard);
|
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
has_data = TRUE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
2023-03-01 18:25:19 +03:00
|
|
|
else if (LogTagAndXGetWindowProperty(
|
|
|
|
TAG, xfc->display, xfc->drawable, clipboard->property_atom, 0, bytes_left, 0,
|
|
|
|
target, &type, &format_property, &length, &dummy, &data) == Success)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-15 06:24:07 +04:00
|
|
|
if (clipboard->incr_starts)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
BYTE* new_data = NULL;
|
2014-10-15 23:49:57 +04:00
|
|
|
bytes_left = length * format_property / 8;
|
2019-11-06 17:24:51 +03:00
|
|
|
new_data =
|
|
|
|
(BYTE*)realloc(clipboard->incr_data, clipboard->incr_data_length + bytes_left);
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
if (new_data)
|
|
|
|
{
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
clipboard->incr_data = new_data;
|
|
|
|
CopyMemory(clipboard->incr_data + clipboard->incr_data_length, data,
|
|
|
|
bytes_left);
|
|
|
|
clipboard->incr_data_length += bytes_left;
|
|
|
|
XFree(data);
|
|
|
|
data = NULL;
|
|
|
|
}
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
has_data = TRUE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2023-03-01 18:17:03 +03:00
|
|
|
LogTagAndXDeleteProperty(TAG, xfc->display, xfc->drawable, clipboard->property_atom);
|
2020-11-26 19:38:04 +03:00
|
|
|
xf_cliprdr_process_requested_data(clipboard, has_data, data, bytes_left);
|
2011-09-23 07:37:17 +04:00
|
|
|
|
|
|
|
if (data)
|
|
|
|
XFree(data);
|
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
static void xf_cliprdr_append_target(xfClipboard* clipboard, Atom target)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
2019-02-07 16:40:36 +03:00
|
|
|
|
2023-06-07 11:20:43 +03:00
|
|
|
if (clipboard->numTargets >= ARRAYSIZE(clipboard->targets))
|
2011-09-23 07:37:17 +04:00
|
|
|
return;
|
|
|
|
|
2023-06-07 11:20:43 +03:00
|
|
|
for (size_t i = 0; i < clipboard->numTargets; i++)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-15 06:24:07 +04:00
|
|
|
if (clipboard->targets[i] == target)
|
2011-09-23 07:37:17 +04:00
|
|
|
return;
|
|
|
|
}
|
2011-11-01 03:15:27 +04:00
|
|
|
|
2014-10-15 23:49:57 +04:00
|
|
|
clipboard->targets[clipboard->numTargets++] = target;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
static void xf_cliprdr_provide_targets(xfClipboard* clipboard, const XSelectionEvent* respond)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (respond->property != None)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2023-03-01 18:01:08 +03:00
|
|
|
LogTagAndXChangeProperty(TAG, xfc->display, respond->requestor, respond->property, XA_ATOM,
|
|
|
|
32, PropModeReplace, (BYTE*)clipboard->targets,
|
|
|
|
clipboard->numTargets);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-13 20:59:22 +03:00
|
|
|
static void xf_cliprdr_provide_timestamp(xfClipboard* clipboard, const XSelectionEvent* respond)
|
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2020-03-13 20:59:22 +03:00
|
|
|
|
|
|
|
if (respond->property != None)
|
|
|
|
{
|
2023-03-01 18:01:08 +03:00
|
|
|
LogTagAndXChangeProperty(TAG, xfc->display, respond->requestor, respond->property,
|
|
|
|
XA_INTEGER, 32, PropModeReplace,
|
|
|
|
(BYTE*)&clipboard->selection_ownership_timestamp, 1);
|
2020-03-13 20:59:22 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
static void xf_cliprdr_provide_data(xfClipboard* clipboard, const XSelectionEvent* respond,
|
|
|
|
const BYTE* data, UINT32 size)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (respond->property != None)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2023-03-01 18:01:08 +03:00
|
|
|
LogTagAndXChangeProperty(TAG, xfc->display, respond->requestor, respond->property,
|
|
|
|
respond->target, 8, PropModeReplace, data, size);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-24 10:59:36 +03:00
|
|
|
static void log_selection_event(xfContext* xfc, const XEvent* event)
|
|
|
|
{
|
|
|
|
const DWORD level = WLOG_TRACE;
|
|
|
|
static wLog* _log_cached_ptr = NULL;
|
|
|
|
if (!_log_cached_ptr)
|
|
|
|
_log_cached_ptr = WLog_Get(TAG);
|
|
|
|
if (WLog_IsLevelActive(_log_cached_ptr, level))
|
|
|
|
{
|
|
|
|
|
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case SelectionClear:
|
|
|
|
{
|
|
|
|
const XSelectionClearEvent* xevent = &event->xselectionclear;
|
2024-06-28 10:16:48 +03:00
|
|
|
char* selection =
|
|
|
|
Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->selection);
|
2023-02-24 10:59:36 +03:00
|
|
|
WLog_Print(_log_cached_ptr, level, "got event %s [selection %s]",
|
|
|
|
x11_event_string(event->type), selection);
|
|
|
|
XFree(selection);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SelectionNotify:
|
|
|
|
{
|
|
|
|
const XSelectionEvent* xevent = &event->xselection;
|
2024-06-28 10:16:48 +03:00
|
|
|
char* selection =
|
|
|
|
Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->selection);
|
|
|
|
char* target = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->target);
|
|
|
|
char* property = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->property);
|
2023-02-24 10:59:36 +03:00
|
|
|
WLog_Print(_log_cached_ptr, level,
|
|
|
|
"got event %s [selection %s, target %s, property %s]",
|
|
|
|
x11_event_string(event->type), selection, target, property);
|
|
|
|
XFree(selection);
|
|
|
|
XFree(target);
|
|
|
|
XFree(property);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SelectionRequest:
|
|
|
|
{
|
|
|
|
const XSelectionRequestEvent* xevent = &event->xselectionrequest;
|
2024-06-28 10:16:48 +03:00
|
|
|
char* selection =
|
|
|
|
Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->selection);
|
|
|
|
char* target = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->target);
|
|
|
|
char* property = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->property);
|
2023-02-24 10:59:36 +03:00
|
|
|
WLog_Print(_log_cached_ptr, level,
|
|
|
|
"got event %s [selection %s, target %s, property %s]",
|
|
|
|
x11_event_string(event->type), selection, target, property);
|
|
|
|
XFree(selection);
|
|
|
|
XFree(target);
|
|
|
|
XFree(property);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case PropertyNotify:
|
|
|
|
{
|
|
|
|
const XPropertyEvent* xevent = &event->xproperty;
|
2024-06-28 10:16:48 +03:00
|
|
|
char* atom = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->atom);
|
2023-02-24 10:59:36 +03:00
|
|
|
WLog_Print(_log_cached_ptr, level, "got event %s [atom %s]",
|
|
|
|
x11_event_string(event->type), atom);
|
|
|
|
XFree(atom);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
static BOOL xf_cliprdr_process_selection_notify(xfClipboard* clipboard,
|
|
|
|
const XSelectionEvent* xevent)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(xevent);
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (xevent->target == clipboard->targets[1])
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2020-02-27 10:28:52 +03:00
|
|
|
if (xevent->property == None)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2022-12-03 11:57:43 +03:00
|
|
|
xf_cliprdr_send_client_format_list(clipboard, FALSE);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-10-15 06:24:07 +04:00
|
|
|
xf_cliprdr_get_requested_targets(clipboard);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-02-27 10:28:52 +03:00
|
|
|
return xf_cliprdr_get_requested_data(clipboard, xevent->target);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-01 16:08:02 +03:00
|
|
|
void xf_cliprdr_clear_cached_data(xfClipboard* clipboard)
|
2016-08-15 22:45:20 +03:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2023-03-01 16:08:02 +03:00
|
|
|
ClipboardLock(clipboard->system);
|
|
|
|
ClipboardEmpty(clipboard->system);
|
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
HashTable_Clear(clipboard->cachedData);
|
|
|
|
HashTable_Clear(clipboard->cachedRawData);
|
|
|
|
|
|
|
|
cliprdr_file_context_clear(clipboard->file);
|
|
|
|
ClipboardUnlock(clipboard->system);
|
|
|
|
}
|
|
|
|
|
2024-09-10 14:24:50 +03:00
|
|
|
static void* format_to_cache_slot(UINT32 format)
|
|
|
|
{
|
|
|
|
union
|
|
|
|
{
|
|
|
|
uintptr_t uptr;
|
|
|
|
void* vptr;
|
|
|
|
} cnv;
|
|
|
|
cnv.uptr = 0x100000000ULL + format;
|
|
|
|
return cnv.vptr;
|
|
|
|
}
|
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
static UINT32 get_dst_format_id_for_local_request(xfClipboard* clipboard,
|
|
|
|
const xfCliprdrFormat* format)
|
|
|
|
{
|
|
|
|
UINT32 dstFormatId = 0;
|
|
|
|
|
|
|
|
WINPR_ASSERT(format);
|
|
|
|
|
|
|
|
if (!format->formatName)
|
|
|
|
return format->localFormat;
|
|
|
|
|
|
|
|
ClipboardLock(clipboard->system);
|
|
|
|
if (strcmp(format->formatName, type_HtmlFormat) == 0)
|
|
|
|
dstFormatId = ClipboardGetFormatId(clipboard->system, mime_html);
|
|
|
|
ClipboardUnlock(clipboard->system);
|
|
|
|
|
|
|
|
if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0)
|
|
|
|
dstFormatId = format->localFormat;
|
|
|
|
|
|
|
|
return dstFormatId;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void get_src_format_info_for_local_request(xfClipboard* clipboard,
|
|
|
|
const xfCliprdrFormat* format,
|
|
|
|
UINT32* srcFormatId, BOOL* nullTerminated)
|
|
|
|
{
|
|
|
|
*srcFormatId = 0;
|
|
|
|
*nullTerminated = FALSE;
|
|
|
|
|
|
|
|
if (format->formatName)
|
2016-08-15 22:45:20 +03:00
|
|
|
{
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
ClipboardLock(clipboard->system);
|
|
|
|
if (strcmp(format->formatName, type_HtmlFormat) == 0)
|
|
|
|
{
|
|
|
|
*srcFormatId = ClipboardGetFormatId(clipboard->system, type_HtmlFormat);
|
|
|
|
*nullTerminated = TRUE;
|
|
|
|
}
|
|
|
|
else if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0)
|
|
|
|
{
|
|
|
|
*srcFormatId = ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
|
|
|
|
*nullTerminated = TRUE;
|
|
|
|
}
|
|
|
|
ClipboardUnlock(clipboard->system);
|
2016-08-15 22:45:20 +03:00
|
|
|
}
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
*srcFormatId = format->formatToRequest;
|
|
|
|
switch (format->formatToRequest)
|
|
|
|
{
|
|
|
|
case CF_TEXT:
|
|
|
|
case CF_OEMTEXT:
|
|
|
|
case CF_UNICODETEXT:
|
|
|
|
*nullTerminated = TRUE;
|
|
|
|
break;
|
|
|
|
case CF_DIB:
|
|
|
|
*srcFormatId = CF_DIB;
|
|
|
|
break;
|
2024-01-31 11:49:16 +03:00
|
|
|
case CF_TIFF:
|
|
|
|
*srcFormatId = CF_TIFF;
|
|
|
|
break;
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-11-14 15:51:37 +03:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
static xfCachedData* convert_data_from_existing_raw_data(xfClipboard* clipboard,
|
|
|
|
xfCachedData* cached_raw_data,
|
|
|
|
UINT32 srcFormatId, BOOL nullTerminated,
|
|
|
|
UINT32 dstFormatId)
|
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfCachedData* cached_data = NULL;
|
|
|
|
BOOL success = 0;
|
|
|
|
BYTE* dst_data = NULL;
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
UINT32 dst_size = 0;
|
2016-08-15 22:45:20 +03:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(cached_raw_data);
|
|
|
|
WINPR_ASSERT(cached_raw_data->data);
|
|
|
|
|
|
|
|
ClipboardLock(clipboard->system);
|
|
|
|
success = ClipboardSetData(clipboard->system, srcFormatId, cached_raw_data->data,
|
|
|
|
cached_raw_data->data_length);
|
|
|
|
if (!success)
|
2016-08-15 22:45:20 +03:00
|
|
|
{
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
WLog_WARN(TAG, "Failed to set clipboard data (formatId: %u, data: %p, data_length: %u)",
|
|
|
|
srcFormatId, cached_raw_data->data, cached_raw_data->data_length);
|
|
|
|
ClipboardUnlock(clipboard->system);
|
|
|
|
return NULL;
|
2016-08-15 22:45:20 +03:00
|
|
|
}
|
2017-11-14 15:51:37 +03:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
dst_data = ClipboardGetData(clipboard->system, dstFormatId, &dst_size);
|
|
|
|
if (!dst_data)
|
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "Failed to get converted clipboard data");
|
|
|
|
ClipboardUnlock(clipboard->system);
|
|
|
|
return NULL;
|
|
|
|
}
|
2023-03-01 16:08:02 +03:00
|
|
|
ClipboardUnlock(clipboard->system);
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
|
|
|
|
if (nullTerminated)
|
|
|
|
{
|
|
|
|
BYTE* nullTerminator = memchr(dst_data, '\0', dst_size);
|
|
|
|
if (nullTerminator)
|
|
|
|
dst_size = nullTerminator - dst_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
cached_data = xf_cached_data_new(dst_data, dst_size);
|
|
|
|
if (!cached_data)
|
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "Failed to allocate cache entry");
|
|
|
|
free(dst_data);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2024-09-10 14:24:50 +03:00
|
|
|
if (!HashTable_Insert(clipboard->cachedData, format_to_cache_slot(dstFormatId), cached_data))
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "Failed to cache clipboard data");
|
|
|
|
xf_cached_data_free(cached_data);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return cached_data;
|
2016-08-15 22:45:20 +03:00
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
static BOOL xf_cliprdr_process_selection_request(xfClipboard* clipboard,
|
|
|
|
const XSelectionRequestEvent* xevent)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
int fmt = 0;
|
|
|
|
Atom type = 0;
|
|
|
|
UINT32 formatId = 0;
|
|
|
|
XSelectionEvent* respond = NULL;
|
2012-10-09 11:01:37 +04:00
|
|
|
BYTE* data = NULL;
|
2024-01-23 18:49:54 +03:00
|
|
|
BOOL delayRespond = 0;
|
|
|
|
BOOL rawTransfer = 0;
|
|
|
|
unsigned long length = 0;
|
|
|
|
unsigned long bytes_left = 0;
|
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(xevent);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (xevent->owner != xfc->drawable)
|
2012-10-09 10:31:28 +04:00
|
|
|
return FALSE;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2014-10-16 05:30:11 +04:00
|
|
|
delayRespond = FALSE;
|
2012-11-20 08:49:08 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (!(respond = (XSelectionEvent*)calloc(1, sizeof(XSelectionEvent))))
|
2015-10-20 22:28:29 +03:00
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "failed to allocate XEvent data");
|
|
|
|
return FALSE;
|
|
|
|
}
|
2012-11-20 08:49:08 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
respond->property = None;
|
|
|
|
respond->type = SelectionNotify;
|
|
|
|
respond->display = xevent->display;
|
|
|
|
respond->requestor = xevent->requestor;
|
|
|
|
respond->selection = xevent->selection;
|
|
|
|
respond->target = xevent->target;
|
|
|
|
respond->time = xevent->time;
|
2011-11-01 03:15:27 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (xevent->target == clipboard->targets[0]) /* TIMESTAMP */
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2020-03-13 20:59:22 +03:00
|
|
|
/* Someone else requests the selection's timestamp */
|
|
|
|
respond->property = xevent->property;
|
|
|
|
xf_cliprdr_provide_timestamp(clipboard, respond);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
2020-02-27 10:28:52 +03:00
|
|
|
else if (xevent->target == clipboard->targets[1]) /* TARGETS */
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
|
|
|
/* Someone else requests our available formats */
|
2020-02-27 10:28:52 +03:00
|
|
|
respond->property = xevent->property;
|
2014-10-15 06:24:07 +04:00
|
|
|
xf_cliprdr_provide_targets(clipboard, respond);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
const CLIPRDR_FORMAT* format =
|
|
|
|
xf_cliprdr_get_server_format_by_atom(clipboard, xevent->target);
|
2023-02-22 09:15:59 +03:00
|
|
|
const xfCliprdrFormat* cformat =
|
|
|
|
xf_cliprdr_get_client_format_by_atom(clipboard, xevent->target);
|
2011-11-23 21:36:36 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (format && (xevent->requestor != xfc->drawable))
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-15 23:49:57 +04:00
|
|
|
formatId = format->formatId;
|
2015-08-19 10:25:59 +03:00
|
|
|
rawTransfer = FALSE;
|
2024-01-23 18:49:54 +03:00
|
|
|
xfCachedData* cached_data = NULL;
|
|
|
|
UINT32 dstFormatId = 0;
|
2013-02-12 05:38:19 +04:00
|
|
|
|
2015-08-16 11:56:36 +03:00
|
|
|
if (formatId == CF_RAW)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2023-03-01 18:25:19 +03:00
|
|
|
if (LogTagAndXGetWindowProperty(
|
|
|
|
TAG, xfc->display, xevent->requestor, clipboard->property_atom, 0, 4, 0,
|
|
|
|
XA_INTEGER, &type, &fmt, &length, &bytes_left, &data) != Success)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
|
|
|
}
|
2014-10-16 06:48:18 +04:00
|
|
|
|
2011-09-23 07:37:17 +04:00
|
|
|
if (data)
|
|
|
|
{
|
2015-08-19 10:25:59 +03:00
|
|
|
rawTransfer = TRUE;
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
CopyMemory(&formatId, data, 4);
|
2011-09-23 07:37:17 +04:00
|
|
|
XFree(data);
|
|
|
|
}
|
|
|
|
}
|
2013-02-12 05:38:19 +04:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
dstFormatId = get_dst_format_id_for_local_request(clipboard, cformat);
|
|
|
|
DEBUG_CLIPRDR("formatId: %u, dstFormatId: %u", formatId, dstFormatId);
|
|
|
|
|
|
|
|
if (!rawTransfer)
|
2024-09-10 14:24:50 +03:00
|
|
|
cached_data = HashTable_GetItemValue(clipboard->cachedData,
|
|
|
|
format_to_cache_slot(dstFormatId));
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
else
|
2024-09-10 14:24:50 +03:00
|
|
|
cached_data = HashTable_GetItemValue(clipboard->cachedRawData,
|
|
|
|
format_to_cache_slot(formatId));
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
|
|
|
|
DEBUG_CLIPRDR("hasCachedData: %u, rawTransfer: %u", cached_data ? 1 : 0, rawTransfer);
|
|
|
|
|
|
|
|
if (!cached_data && !rawTransfer)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
UINT32 srcFormatId = 0;
|
|
|
|
BOOL nullTerminated = FALSE;
|
2024-01-23 18:49:54 +03:00
|
|
|
xfCachedData* cached_raw_data = NULL;
|
2023-02-21 18:24:41 +03:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
get_src_format_info_for_local_request(clipboard, cformat, &srcFormatId,
|
|
|
|
&nullTerminated);
|
|
|
|
cached_raw_data =
|
|
|
|
HashTable_GetItemValue(clipboard->cachedRawData, (void*)(UINT_PTR)srcFormatId);
|
2023-02-21 18:24:41 +03:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
DEBUG_CLIPRDR("hasCachedRawData: %u, rawDataLength: %u", cached_raw_data ? 1 : 0,
|
|
|
|
cached_raw_data ? cached_raw_data->data_length : 0);
|
2023-02-28 13:37:42 +03:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
if (cached_raw_data && cached_raw_data->data_length != 0)
|
2024-02-20 12:45:08 +03:00
|
|
|
cached_data = convert_data_from_existing_raw_data(
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
clipboard, cached_raw_data, srcFormatId, nullTerminated, dstFormatId);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
|
|
|
|
DEBUG_CLIPRDR("hasCachedData: %u", cached_data ? 1 : 0);
|
|
|
|
|
|
|
|
if (cached_data)
|
2016-08-15 22:45:20 +03:00
|
|
|
{
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
/* Cached clipboard data available. Send it now */
|
2020-02-27 10:28:52 +03:00
|
|
|
respond->property = xevent->property;
|
2024-01-24 10:21:47 +03:00
|
|
|
|
|
|
|
// NOLINTNEXTLINE(clang-analyzer-unix.Malloc)
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
xf_cliprdr_provide_data(clipboard, respond, cached_data->data,
|
|
|
|
cached_data->data_length);
|
2016-08-15 22:45:20 +03:00
|
|
|
}
|
2014-10-15 06:24:07 +04:00
|
|
|
else if (clipboard->respond)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2014-10-16 06:48:18 +04:00
|
|
|
/* duplicate request */
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-02-22 09:15:59 +03:00
|
|
|
WINPR_ASSERT(cformat);
|
2023-02-24 11:27:27 +03:00
|
|
|
|
2011-09-23 07:37:17 +04:00
|
|
|
/**
|
|
|
|
* Send clipboard data request to the server.
|
|
|
|
* Response will be postponed after receiving the data
|
|
|
|
*/
|
2020-02-27 10:28:52 +03:00
|
|
|
respond->property = xevent->property;
|
2014-10-15 06:24:07 +04:00
|
|
|
clipboard->respond = respond;
|
2024-08-12 16:55:12 +03:00
|
|
|
requested_format_replace(&clipboard->requestedFormat, formatId,
|
|
|
|
cformat->formatName);
|
2015-08-19 10:25:59 +03:00
|
|
|
clipboard->data_raw_format = rawTransfer;
|
2014-10-16 05:30:11 +04:00
|
|
|
delayRespond = TRUE;
|
2023-02-22 09:15:59 +03:00
|
|
|
xf_cliprdr_send_data_request(clipboard, formatId, cformat);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-16 05:30:11 +04:00
|
|
|
if (!delayRespond)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2020-12-31 17:40:39 +03:00
|
|
|
union
|
|
|
|
{
|
2020-02-27 10:28:52 +03:00
|
|
|
XEvent* ev;
|
|
|
|
XSelectionEvent* sev;
|
|
|
|
} conv;
|
|
|
|
|
|
|
|
conv.sev = respond;
|
|
|
|
XSendEvent(xfc->display, xevent->requestor, 0, 0, conv.ev);
|
2013-06-13 02:57:25 +04:00
|
|
|
XFlush(xfc->display);
|
2012-10-09 07:21:26 +04:00
|
|
|
free(respond);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
static BOOL xf_cliprdr_process_selection_clear(xfClipboard* clipboard,
|
|
|
|
const XSelectionClearEvent* xevent)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(xevent);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2019-02-07 19:53:21 +03:00
|
|
|
WINPR_UNUSED(xevent);
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
if (xf_cliprdr_is_self_owned(clipboard))
|
2012-10-09 10:31:28 +04:00
|
|
|
return FALSE;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2023-03-01 18:17:03 +03:00
|
|
|
LogTagAndXDeleteProperty(TAG, xfc->display, clipboard->root_window, clipboard->property_atom);
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
static BOOL xf_cliprdr_process_property_notify(xfClipboard* clipboard, const XPropertyEvent* xevent)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
const xfCliprdrFormat* format = NULL;
|
2017-01-26 12:44:19 +03:00
|
|
|
xfContext* xfc = NULL;
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
if (!clipboard)
|
2013-02-12 05:38:19 +04:00
|
|
|
return TRUE;
|
|
|
|
|
2017-01-26 12:44:19 +03:00
|
|
|
xfc = clipboard->xfc;
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
WINPR_ASSERT(xevent);
|
2017-01-26 12:44:19 +03:00
|
|
|
|
2020-03-13 20:59:22 +03:00
|
|
|
if (xevent->atom == clipboard->timestamp_property_atom)
|
|
|
|
{
|
|
|
|
/* This is the response to the property change we did
|
|
|
|
* in xf_cliprdr_prepare_to_set_selection_owner. Now
|
|
|
|
* we can set ourselves as the selection owner. (See
|
|
|
|
* comments in those functions below.) */
|
|
|
|
xf_cliprdr_set_selection_owner(xfc, clipboard, xevent->time);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (xevent->atom != clipboard->property_atom)
|
2012-10-09 10:31:28 +04:00
|
|
|
return FALSE; /* Not cliprdr-related */
|
2011-09-23 07:37:17 +04:00
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
if (xevent->window == clipboard->root_window)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2022-12-03 11:57:43 +03:00
|
|
|
xf_cliprdr_send_client_format_list(clipboard, FALSE);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
2020-02-27 10:28:52 +03:00
|
|
|
else if ((xevent->window == xfc->drawable) && (xevent->state == PropertyNewValue) &&
|
|
|
|
clipboard->incr_starts)
|
2011-09-23 07:37:17 +04:00
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
format = xf_cliprdr_get_client_format_by_id(clipboard, clipboard->requestedFormatId);
|
2014-10-15 23:49:57 +04:00
|
|
|
|
|
|
|
if (format)
|
|
|
|
xf_cliprdr_get_requested_data(clipboard, format->atom);
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-09-23 07:37:17 +04:00
|
|
|
}
|
|
|
|
|
2020-02-27 10:28:52 +03:00
|
|
|
void xf_cliprdr_handle_xevent(xfContext* xfc, const XEvent* event)
|
2014-07-07 22:26:41 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfClipboard* clipboard = NULL;
|
2014-07-07 22:26:41 +04:00
|
|
|
|
|
|
|
if (!xfc || !event)
|
|
|
|
return;
|
|
|
|
|
2014-10-15 23:49:57 +04:00
|
|
|
clipboard = xfc->clipboard;
|
|
|
|
|
|
|
|
if (!clipboard)
|
2014-07-07 22:26:41 +04:00
|
|
|
return;
|
|
|
|
|
|
|
|
#ifdef WITH_XFIXES
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (clipboard->xfixes_supported &&
|
|
|
|
event->type == XFixesSelectionNotify + clipboard->xfixes_event_base)
|
2014-07-07 22:26:41 +04:00
|
|
|
{
|
2021-08-02 13:13:34 +03:00
|
|
|
const XFixesSelectionNotifyEvent* se = (const XFixesSelectionNotifyEvent*)event;
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
if (se->subtype == XFixesSetSelectionOwnerNotify)
|
|
|
|
{
|
2014-10-15 06:24:07 +04:00
|
|
|
if (se->selection != clipboard->clipboard_atom)
|
2014-07-07 22:26:41 +04:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (XGetSelectionOwner(xfc->display, se->selection) == xfc->drawable)
|
|
|
|
return;
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
clipboard->owner = None;
|
2014-10-15 23:49:57 +04:00
|
|
|
xf_cliprdr_check_owner(clipboard);
|
2014-07-07 22:26:41 +04:00
|
|
|
}
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
return;
|
|
|
|
}
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
#endif
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case SelectionNotify:
|
2023-02-24 10:59:36 +03:00
|
|
|
log_selection_event(xfc, event);
|
2020-02-27 10:28:52 +03:00
|
|
|
xf_cliprdr_process_selection_notify(clipboard, &event->xselection);
|
2014-07-07 22:26:41 +04:00
|
|
|
break;
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
case SelectionRequest:
|
2023-02-24 10:59:36 +03:00
|
|
|
log_selection_event(xfc, event);
|
2020-02-27 10:28:52 +03:00
|
|
|
xf_cliprdr_process_selection_request(clipboard, &event->xselectionrequest);
|
2014-07-07 22:26:41 +04:00
|
|
|
break;
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
case SelectionClear:
|
2023-02-24 10:59:36 +03:00
|
|
|
log_selection_event(xfc, event);
|
2020-02-27 10:28:52 +03:00
|
|
|
xf_cliprdr_process_selection_clear(clipboard, &event->xselectionclear);
|
2014-07-07 22:26:41 +04:00
|
|
|
break;
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
case PropertyNotify:
|
2023-02-24 10:59:36 +03:00
|
|
|
log_selection_event(xfc, event);
|
2020-02-27 10:28:52 +03:00
|
|
|
xf_cliprdr_process_property_notify(clipboard, &event->xproperty);
|
2014-07-07 22:26:41 +04:00
|
|
|
break;
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2014-07-07 22:26:41 +04:00
|
|
|
case FocusIn:
|
2014-10-15 06:24:07 +04:00
|
|
|
if (!clipboard->xfixes_supported)
|
2014-07-07 22:26:41 +04:00
|
|
|
{
|
2014-10-15 23:49:57 +04:00
|
|
|
xf_cliprdr_check_owner(clipboard);
|
2014-07-07 22:26:41 +04:00
|
|
|
}
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2023-02-24 10:59:36 +03:00
|
|
|
break;
|
|
|
|
default:
|
2014-07-07 22:26:41 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2017-05-24 23:05:42 +03:00
|
|
|
static UINT xf_cliprdr_send_client_capabilities(xfClipboard* clipboard)
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
CLIPRDR_CAPABILITIES capabilities = { 0 };
|
|
|
|
CLIPRDR_GENERAL_CAPABILITY_SET generalCapabilitySet = { 0 };
|
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2014-10-15 06:58:01 +04:00
|
|
|
capabilities.cCapabilitiesSets = 1;
|
2019-11-06 17:24:51 +03:00
|
|
|
capabilities.capabilitySets = (CLIPRDR_CAPABILITY_SET*)&(generalCapabilitySet);
|
2014-10-15 06:58:01 +04:00
|
|
|
generalCapabilitySet.capabilitySetType = CB_CAPSTYPE_GENERAL;
|
|
|
|
generalCapabilitySet.capabilitySetLength = 12;
|
|
|
|
generalCapabilitySet.version = CB_CAPS_VERSION_2;
|
|
|
|
generalCapabilitySet.generalFlags = CB_USE_LONG_FORMAT_NAMES;
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
2023-02-25 18:57:52 +03:00
|
|
|
generalCapabilitySet.generalFlags |= cliprdr_file_context_current_flags(clipboard->file);
|
2022-07-01 12:04:22 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard->context);
|
|
|
|
WINPR_ASSERT(clipboard->context->ClientCapabilities);
|
2019-11-06 17:24:51 +03:00
|
|
|
return clipboard->context->ClientCapabilities(clipboard->context, &capabilities);
|
2014-10-15 06:58:01 +04:00
|
|
|
}
|
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2022-12-03 11:57:43 +03:00
|
|
|
static UINT xf_cliprdr_send_client_format_list(xfClipboard* clipboard, BOOL force)
|
2014-10-15 06:58:01 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
2023-06-05 13:16:57 +03:00
|
|
|
xfContext* xfc = clipboard->xfc;
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2023-05-30 14:18:08 +03:00
|
|
|
UINT32 numFormats = 0;
|
|
|
|
CLIPRDR_FORMAT* formats = xf_cliprdr_get_client_formats(clipboard, &numFormats);
|
2014-10-16 06:48:18 +04:00
|
|
|
|
2023-06-05 13:16:57 +03:00
|
|
|
const UINT ret = xf_cliprdr_send_format_list(clipboard, formats, numFormats, force);
|
2014-10-16 06:48:18 +04:00
|
|
|
|
2014-12-04 02:19:23 +03:00
|
|
|
if (clipboard->owner && clipboard->owner != xfc->drawable)
|
2014-10-16 06:48:18 +04:00
|
|
|
{
|
|
|
|
/* Request the owner for TARGETS, and wait for SelectionNotify event */
|
2024-07-17 12:05:57 +03:00
|
|
|
LogTagAndXConvertSelection(TAG, xfc->display, clipboard->clipboard_atom,
|
|
|
|
clipboard->targets[1], clipboard->property_atom, xfc->drawable,
|
|
|
|
CurrentTime);
|
2014-10-16 06:48:18 +04:00
|
|
|
}
|
2014-10-15 06:58:01 +04:00
|
|
|
|
2023-05-30 14:18:08 +03:00
|
|
|
xf_cliprdr_free_formats(formats, numFormats);
|
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
return ret;
|
2014-10-15 06:58:01 +04:00
|
|
|
}
|
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2019-11-06 17:24:51 +03:00
|
|
|
static UINT xf_cliprdr_send_client_format_list_response(xfClipboard* clipboard, BOOL status)
|
2014-10-15 06:58:01 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
CLIPRDR_FORMAT_LIST_RESPONSE formatListResponse = { 0 };
|
|
|
|
|
|
|
|
formatListResponse.common.msgType = CB_FORMAT_LIST_RESPONSE;
|
|
|
|
formatListResponse.common.msgFlags = status ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
|
|
|
|
formatListResponse.common.dataLen = 0;
|
|
|
|
|
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
WINPR_ASSERT(clipboard->context);
|
|
|
|
WINPR_ASSERT(clipboard->context->ClientFormatListResponse);
|
2019-11-06 17:24:51 +03:00
|
|
|
return clipboard->context->ClientFormatListResponse(clipboard->context, &formatListResponse);
|
2014-10-15 06:58:01 +04:00
|
|
|
}
|
|
|
|
|
2015-08-27 15:25:09 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2016-10-06 14:31:25 +03:00
|
|
|
static UINT xf_cliprdr_monitor_ready(CliprdrClientContext* context,
|
2019-01-23 18:22:29 +03:00
|
|
|
const CLIPRDR_MONITOR_READY* monitorReady)
|
2014-10-15 06:58:01 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
UINT ret = 0;
|
|
|
|
xfClipboard* clipboard = NULL;
|
2021-12-17 15:47:20 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(context);
|
|
|
|
WINPR_ASSERT(monitorReady);
|
|
|
|
|
2023-02-25 14:42:04 +03:00
|
|
|
clipboard = cliprdr_file_context_get_context(context->custom);
|
2021-12-17 15:47:20 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
2015-10-20 22:28:29 +03:00
|
|
|
|
2019-02-07 19:53:21 +03:00
|
|
|
WINPR_UNUSED(monitorReady);
|
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
if ((ret = xf_cliprdr_send_client_capabilities(clipboard)) != CHANNEL_RC_OK)
|
|
|
|
return ret;
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2022-10-17 17:36:38 +03:00
|
|
|
xf_clipboard_formats_free(clipboard);
|
|
|
|
|
2022-12-03 11:57:43 +03:00
|
|
|
if ((ret = xf_cliprdr_send_client_format_list(clipboard, TRUE)) != CHANNEL_RC_OK)
|
2015-10-20 22:28:29 +03:00
|
|
|
return ret;
|
2014-10-15 06:58:01 +04:00
|
|
|
|
2014-10-15 23:49:57 +04:00
|
|
|
clipboard->sync = TRUE;
|
2015-06-02 14:05:10 +03:00
|
|
|
return CHANNEL_RC_OK;
|
2014-10-15 06:58:01 +04:00
|
|
|
}
|
|
|
|
|
2015-08-27 15:25:09 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2016-10-06 14:31:25 +03:00
|
|
|
static UINT xf_cliprdr_server_capabilities(CliprdrClientContext* context,
|
2019-11-06 17:24:51 +03:00
|
|
|
const CLIPRDR_CAPABILITIES* capabilities)
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
const CLIPRDR_GENERAL_CAPABILITY_SET* generalCaps = NULL;
|
|
|
|
const BYTE* capsPtr = NULL;
|
|
|
|
xfClipboard* clipboard = NULL;
|
2021-12-17 15:47:20 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(context);
|
|
|
|
WINPR_ASSERT(capabilities);
|
|
|
|
|
2023-02-25 14:42:04 +03:00
|
|
|
clipboard = cliprdr_file_context_get_context(context->custom);
|
2021-12-17 15:47:20 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
capsPtr = (const BYTE*)capabilities->capabilitySets;
|
|
|
|
WINPR_ASSERT(capsPtr);
|
|
|
|
|
2023-02-25 18:57:52 +03:00
|
|
|
cliprdr_file_context_remote_set_flags(clipboard->file, 0);
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 i = 0; i < capabilities->cCapabilitiesSets; i++)
|
2017-04-09 02:29:51 +03:00
|
|
|
{
|
2021-12-17 15:47:20 +03:00
|
|
|
const CLIPRDR_CAPABILITY_SET* caps = (const CLIPRDR_CAPABILITY_SET*)capsPtr;
|
2017-04-09 02:29:51 +03:00
|
|
|
|
|
|
|
if (caps->capabilitySetType == CB_CAPSTYPE_GENERAL)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
generalCaps = (const CLIPRDR_GENERAL_CAPABILITY_SET*)caps;
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2023-02-25 18:57:52 +03:00
|
|
|
cliprdr_file_context_remote_set_flags(clipboard->file, generalCaps->generalFlags);
|
2017-04-09 02:29:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
capsPtr += caps->capabilitySetLength;
|
|
|
|
}
|
|
|
|
|
2015-06-02 14:05:10 +03:00
|
|
|
return CHANNEL_RC_OK;
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
2020-03-13 20:59:22 +03:00
|
|
|
static void xf_cliprdr_prepare_to_set_selection_owner(xfContext* xfc, xfClipboard* clipboard)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
WINPR_ASSERT(clipboard);
|
2020-03-13 20:59:22 +03:00
|
|
|
/*
|
|
|
|
* When you're writing to the selection in response to a
|
|
|
|
* normal X event like a mouse click or keyboard action, you
|
|
|
|
* get the selection timestamp by copying the time field out
|
|
|
|
* of that X event. Here, we're doing it on our own
|
|
|
|
* initiative, so we have to _request_ the X server time.
|
|
|
|
*
|
|
|
|
* There isn't a GetServerTime request in the X protocol, so I
|
|
|
|
* work around it by setting a property on our own window, and
|
|
|
|
* waiting for a PropertyNotify event to come back telling me
|
|
|
|
* it's been done - which will have a timestamp we can use.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* We have to set the property to some value, but it doesn't
|
|
|
|
* matter what. Set it to its own name, which we have here
|
|
|
|
* anyway! */
|
|
|
|
Atom value = clipboard->timestamp_property_atom;
|
|
|
|
|
2023-03-01 18:01:08 +03:00
|
|
|
LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->timestamp_property_atom,
|
|
|
|
XA_ATOM, 32, PropModeReplace, (BYTE*)&value, 1);
|
2020-03-13 20:59:22 +03:00
|
|
|
XFlush(xfc->display);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xf_cliprdr_set_selection_owner(xfContext* xfc, xfClipboard* clipboard, Time timestamp)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
WINPR_ASSERT(clipboard);
|
2020-03-13 20:59:22 +03:00
|
|
|
/*
|
|
|
|
* Actually set ourselves up as the selection owner, now that
|
|
|
|
* we have a timestamp to use.
|
|
|
|
*/
|
|
|
|
|
|
|
|
clipboard->selection_ownership_timestamp = timestamp;
|
|
|
|
XSetSelectionOwner(xfc->display, clipboard->clipboard_atom, xfc->drawable, timestamp);
|
|
|
|
XFlush(xfc->display);
|
|
|
|
}
|
|
|
|
|
2015-08-27 15:25:09 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2016-10-06 14:31:25 +03:00
|
|
|
static UINT xf_cliprdr_server_format_list(CliprdrClientContext* context,
|
2019-11-06 17:24:51 +03:00
|
|
|
const CLIPRDR_FORMAT_LIST* formatList)
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
xfContext* xfc = NULL;
|
|
|
|
UINT ret = 0;
|
|
|
|
xfClipboard* clipboard = NULL;
|
2021-12-17 15:47:20 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(context);
|
|
|
|
WINPR_ASSERT(formatList);
|
|
|
|
|
2023-02-25 14:42:04 +03:00
|
|
|
clipboard = cliprdr_file_context_get_context(context->custom);
|
2021-12-17 15:47:20 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
X11/cliprdr: Clear selection requests, when they become invalid
When a FormatDataRequest by xfreerdp is answered with the
CB_RESPONSE_FAIL flag, then the request was answered, but xfreerdp does
not mark it as answered, by leaving the request as pending.
This results in non-functional copy-paste actions, when copying
something from the server side, after the failed request.
A similar situation can happen, when a new FormatList is received,
while there is a pending FormatDataRequest.
Fix these situations by clearing the pending request, when a form of an
answer, either via CB_RESPONSE_FAIL flag or via new FormatList, is
received.
Fixes: https://github.com/FreeRDP/FreeRDP/issues/7757
2022-03-29 13:13:33 +03:00
|
|
|
/* Clear the active SelectionRequest, as it is now invalid */
|
|
|
|
free(clipboard->respond);
|
|
|
|
clipboard->respond = NULL;
|
|
|
|
|
2020-11-26 19:38:04 +03:00
|
|
|
xf_clipboard_formats_free(clipboard);
|
2016-08-15 22:45:20 +03:00
|
|
|
xf_cliprdr_clear_cached_data(clipboard);
|
2024-08-12 16:55:12 +03:00
|
|
|
requested_format_free(&clipboard->requestedFormat);
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
xf_clipboard_free_server_formats(clipboard);
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2015-08-16 11:56:36 +03:00
|
|
|
clipboard->numServerFormats = formatList->numFormats + 1; /* +1 for CF_RAW */
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (!(clipboard->serverFormats =
|
|
|
|
(CLIPRDR_FORMAT*)calloc(clipboard->numServerFormats, sizeof(CLIPRDR_FORMAT))))
|
2016-10-06 14:31:25 +03:00
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
WLog_ERR(TAG, "failed to allocate %d CLIPRDR_FORMAT structs", clipboard->numServerFormats);
|
2015-08-16 11:56:36 +03:00
|
|
|
return CHANNEL_RC_NO_MEMORY;
|
2015-10-20 22:28:29 +03:00
|
|
|
}
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
for (size_t i = 0; i < formatList->numFormats; i++)
|
2014-10-15 23:49:57 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
const CLIPRDR_FORMAT* format = &formatList->formats[i];
|
|
|
|
CLIPRDR_FORMAT* srvFormat = &clipboard->serverFormats[i];
|
|
|
|
|
|
|
|
srvFormat->formatId = format->formatId;
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2015-07-02 16:39:35 +03:00
|
|
|
if (format->formatName)
|
2015-06-17 23:08:02 +03:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
srvFormat->formatName = _strdup(format->formatName);
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
if (!srvFormat->formatName)
|
2015-07-02 16:39:35 +03:00
|
|
|
{
|
2024-01-30 12:25:38 +03:00
|
|
|
for (UINT32 k = 0; k < i; k++)
|
2017-11-14 15:51:37 +03:00
|
|
|
free(clipboard->serverFormats[k].formatName);
|
2015-06-17 23:08:02 +03:00
|
|
|
|
2015-07-02 16:39:35 +03:00
|
|
|
clipboard->numServerFormats = 0;
|
|
|
|
free(clipboard->serverFormats);
|
|
|
|
clipboard->serverFormats = NULL;
|
2015-10-20 22:28:29 +03:00
|
|
|
return CHANNEL_RC_NO_MEMORY;
|
2015-07-02 16:39:35 +03:00
|
|
|
}
|
2015-06-17 23:08:02 +03:00
|
|
|
}
|
2014-10-15 23:49:57 +04:00
|
|
|
}
|
|
|
|
|
X11/cliprdr: Rework server to client clipboard handling
The purpose of clipboard data locking is to make the other peer
retaining the current file list until a pending paste operation is done,
even though the clipboard selection changed.
As it may be difficult to determine, when a lock is needed, imitate the
same behaviour as mstsc:
When the server side supports clipboard data locking, always attempt to
lock the file list on the server regardless of what is advertised in a
FormatList PDU.
The Lock Clipboard Data PDU can even be already sent, before the
Format List Response PDU is sent.
This is also what mstsc, does: First, lock the new (potential) file
list, then unlock the file list, when the pending paste operation is
done.
So, rework the current clipboard implementation in that direction.
Since the implementation for timeouts for old file lists is a bit hard,
for now always force unlock pending locks, when the selection changes.
However, timeouts for old file lists can still be added in the future.
The reworked clipboard handling is done with the help of three hash
tables:
1. The inode table: This hash table manages all inodes for each file.
The keys in this table are the inodes themselves, while the values
the files and directories and their attributes (file size, last write
time, etc.).
2. The clipdata table: This table manages the locks for each file list.
The keys in this table represent the clip data id and the values the
clip data entries, which have a reference to the clip data dir, a
directory containing the whole selection, and some helper attributes,
like the clip data id itself.
3. The request table: Every file size or file range request is managed
here. When a FileContentsRequest is made, its stream id with the
respective details are added to this table. When a response is
received, these details can then be easily looked up here.
2023-07-19 11:38:33 +03:00
|
|
|
ret = cliprdr_file_context_notify_new_server_format_list(clipboard->file);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2015-08-16 11:56:36 +03:00
|
|
|
/* CF_RAW is always implicitly supported by the server */
|
2017-11-14 15:51:37 +03:00
|
|
|
{
|
|
|
|
CLIPRDR_FORMAT* format = &clipboard->serverFormats[formatList->numFormats];
|
|
|
|
format->formatId = CF_RAW;
|
|
|
|
format->formatName = NULL;
|
|
|
|
}
|
2015-08-19 10:59:49 +03:00
|
|
|
xf_cliprdr_provide_server_format_list(clipboard);
|
2014-10-15 23:49:57 +04:00
|
|
|
clipboard->numTargets = 2;
|
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
for (size_t i = 0; i < formatList->numFormats; i++)
|
2014-10-15 23:49:57 +04:00
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
const CLIPRDR_FORMAT* format = &formatList->formats[i];
|
2014-10-15 23:49:57 +04:00
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
for (size_t j = 0; j < clipboard->numClientFormats; j++)
|
2014-10-15 23:49:57 +04:00
|
|
|
{
|
2021-03-10 13:36:23 +03:00
|
|
|
const xfCliprdrFormat* clientFormat = &clipboard->clientFormats[j];
|
|
|
|
if (xf_cliprdr_formats_equal(format, clientFormat))
|
2014-10-15 23:49:57 +04:00
|
|
|
{
|
2023-03-01 16:22:42 +03:00
|
|
|
if ((clientFormat->formatName != NULL) &&
|
|
|
|
(strcmp(type_FileGroupDescriptorW, clientFormat->formatName) == 0))
|
2023-02-28 11:31:09 +03:00
|
|
|
{
|
|
|
|
if (!cliprdr_file_context_has_local_support(clipboard->file))
|
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 13:36:23 +03:00
|
|
|
xf_cliprdr_append_target(clipboard, clientFormat->atom);
|
2014-10-15 23:49:57 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-20 22:28:29 +03:00
|
|
|
ret = xf_cliprdr_send_client_format_list_response(clipboard, TRUE);
|
2021-04-16 11:50:08 +03:00
|
|
|
if (xfc->remote_app)
|
|
|
|
xf_cliprdr_set_selection_owner(xfc, clipboard, CurrentTime);
|
|
|
|
else
|
|
|
|
xf_cliprdr_prepare_to_set_selection_owner(xfc, clipboard);
|
2015-10-20 22:28:29 +03:00
|
|
|
return ret;
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
2015-08-27 15:25:09 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2019-11-06 17:24:51 +03:00
|
|
|
static UINT
|
|
|
|
xf_cliprdr_server_format_list_response(CliprdrClientContext* context,
|
|
|
|
const CLIPRDR_FORMAT_LIST_RESPONSE* formatListResponse)
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(context);
|
|
|
|
WINPR_ASSERT(formatListResponse);
|
2019-11-06 17:24:51 +03:00
|
|
|
// xfClipboard* clipboard = (xfClipboard*) context->custom;
|
2015-06-02 14:05:10 +03:00
|
|
|
return CHANNEL_RC_OK;
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
2015-08-27 15:25:09 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2019-11-06 17:24:51 +03:00
|
|
|
static UINT
|
|
|
|
xf_cliprdr_server_format_data_request(CliprdrClientContext* context,
|
|
|
|
const CLIPRDR_FORMAT_DATA_REQUEST* formatDataRequest)
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
BOOL rawTransfer = 0;
|
2021-03-10 13:36:23 +03:00
|
|
|
const xfCliprdrFormat* format = NULL;
|
2024-01-23 18:49:54 +03:00
|
|
|
UINT32 formatId = 0;
|
|
|
|
xfContext* xfc = NULL;
|
|
|
|
xfClipboard* clipboard = NULL;
|
2021-12-17 15:47:20 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(context);
|
|
|
|
WINPR_ASSERT(formatDataRequest);
|
|
|
|
|
2023-02-25 14:42:04 +03:00
|
|
|
clipboard = cliprdr_file_context_get_context(context->custom);
|
2021-12-17 15:47:20 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
|
|
|
formatId = formatDataRequest->requestedFormatId;
|
|
|
|
|
2015-08-19 10:25:59 +03:00
|
|
|
rawTransfer = xf_cliprdr_is_raw_transfer_available(clipboard);
|
|
|
|
|
|
|
|
if (rawTransfer)
|
2014-10-16 01:42:55 +04:00
|
|
|
{
|
2015-08-16 11:56:36 +03:00
|
|
|
format = xf_cliprdr_get_client_format_by_id(clipboard, CF_RAW);
|
2023-03-01 18:01:08 +03:00
|
|
|
LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->property_atom,
|
|
|
|
XA_INTEGER, 32, PropModeReplace, (BYTE*)&formatId, 1);
|
2014-10-16 01:42:55 +04:00
|
|
|
}
|
|
|
|
else
|
2015-08-16 11:56:36 +03:00
|
|
|
format = xf_cliprdr_get_client_format_by_id(clipboard, formatId);
|
2014-10-16 01:42:55 +04:00
|
|
|
|
2020-11-26 18:34:28 +03:00
|
|
|
clipboard->requestedFormatId = rawTransfer ? CF_RAW : formatId;
|
2014-10-16 01:42:55 +04:00
|
|
|
if (!format)
|
2023-02-22 09:15:59 +03:00
|
|
|
return xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
|
2014-10-16 01:42:55 +04:00
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
DEBUG_CLIPRDR("requested format 0x%08" PRIx32 " [%s] {local 0x%08" PRIx32 "} [%s]",
|
|
|
|
format->formatToRequest, ClipboardGetFormatIdString(format->formatToRequest),
|
|
|
|
format->localFormat, format->formatName);
|
2024-07-17 12:05:57 +03:00
|
|
|
LogTagAndXConvertSelection(TAG, xfc->display, clipboard->clipboard_atom, format->atom,
|
|
|
|
clipboard->property_atom, xfc->drawable, CurrentTime);
|
2014-10-16 01:42:55 +04:00
|
|
|
XFlush(xfc->display);
|
|
|
|
/* After this point, we expect a SelectionNotify event from the clipboard owner. */
|
2015-06-02 14:05:10 +03:00
|
|
|
return CHANNEL_RC_OK;
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
2015-08-27 15:25:09 +03:00
|
|
|
/**
|
|
|
|
* Function description
|
|
|
|
*
|
|
|
|
* @return 0 on success, otherwise a Win32 error code
|
|
|
|
*/
|
2019-11-06 17:24:51 +03:00
|
|
|
static UINT
|
|
|
|
xf_cliprdr_server_format_data_response(CliprdrClientContext* context,
|
|
|
|
const CLIPRDR_FORMAT_DATA_RESPONSE* formatDataResponse)
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
BOOL bSuccess = 0;
|
|
|
|
BYTE* pDstData = NULL;
|
|
|
|
UINT32 DstSize = 0;
|
|
|
|
UINT32 SrcSize = 0;
|
|
|
|
UINT32 srcFormatId = 0;
|
|
|
|
UINT32 dstFormatId = 0;
|
2014-12-21 21:49:22 +03:00
|
|
|
BOOL nullTerminated = FALSE;
|
2024-01-23 18:49:54 +03:00
|
|
|
UINT32 size = 0;
|
|
|
|
const BYTE* data = NULL;
|
|
|
|
xfContext* xfc = NULL;
|
|
|
|
xfClipboard* clipboard = NULL;
|
|
|
|
xfCachedData* cached_data = NULL;
|
2021-12-17 15:47:20 +03:00
|
|
|
|
|
|
|
WINPR_ASSERT(context);
|
|
|
|
WINPR_ASSERT(formatDataResponse);
|
|
|
|
|
2023-02-25 14:42:04 +03:00
|
|
|
clipboard = cliprdr_file_context_get_context(context->custom);
|
2021-12-17 15:47:20 +03:00
|
|
|
WINPR_ASSERT(clipboard);
|
|
|
|
|
|
|
|
xfc = clipboard->xfc;
|
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
size = formatDataResponse->common.dataLen;
|
2021-12-17 15:47:20 +03:00
|
|
|
data = formatDataResponse->requestedFormatData;
|
2014-10-16 01:42:55 +04:00
|
|
|
|
2022-07-01 12:04:22 +03:00
|
|
|
if (formatDataResponse->common.msgFlags == CB_RESPONSE_FAIL)
|
2021-01-16 10:14:36 +03:00
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "Format Data Response PDU msgFlags is CB_RESPONSE_FAIL");
|
X11/cliprdr: Clear selection requests, when they become invalid
When a FormatDataRequest by xfreerdp is answered with the
CB_RESPONSE_FAIL flag, then the request was answered, but xfreerdp does
not mark it as answered, by leaving the request as pending.
This results in non-functional copy-paste actions, when copying
something from the server side, after the failed request.
A similar situation can happen, when a new FormatList is received,
while there is a pending FormatDataRequest.
Fix these situations by clearing the pending request, when a form of an
answer, either via CB_RESPONSE_FAIL flag or via new FormatList, is
received.
Fixes: https://github.com/FreeRDP/FreeRDP/issues/7757
2022-03-29 13:13:33 +03:00
|
|
|
free(clipboard->respond);
|
|
|
|
clipboard->respond = NULL;
|
2021-01-16 10:14:36 +03:00
|
|
|
return CHANNEL_RC_OK;
|
|
|
|
}
|
|
|
|
|
2014-10-16 01:42:55 +04:00
|
|
|
if (!clipboard->respond)
|
2015-06-02 14:05:10 +03:00
|
|
|
return CHANNEL_RC_OK;
|
2014-10-16 01:42:55 +04:00
|
|
|
|
2014-10-18 04:55:12 +04:00
|
|
|
pDstData = NULL;
|
2014-12-07 02:29:28 +03:00
|
|
|
DstSize = 0;
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
srcFormatId = 0;
|
|
|
|
dstFormatId = 0;
|
2014-10-18 04:55:12 +04:00
|
|
|
|
2024-08-12 16:55:12 +03:00
|
|
|
const RequestedFormat* format = clipboard->requestedFormat;
|
2015-08-19 10:25:59 +03:00
|
|
|
if (clipboard->data_raw_format)
|
|
|
|
{
|
|
|
|
srcFormatId = CF_RAW;
|
|
|
|
dstFormatId = CF_RAW;
|
|
|
|
}
|
2023-03-13 14:53:40 +03:00
|
|
|
else if (!format)
|
2023-02-22 09:15:59 +03:00
|
|
|
return ERROR_INTERNAL_ERROR;
|
2023-03-13 14:53:40 +03:00
|
|
|
else if (format->formatName)
|
2014-10-18 04:55:12 +04:00
|
|
|
{
|
2023-03-22 11:53:30 +03:00
|
|
|
ClipboardLock(clipboard->system);
|
2023-03-13 14:53:40 +03:00
|
|
|
if (strcmp(format->formatName, type_HtmlFormat) == 0)
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
{
|
2023-02-22 09:15:59 +03:00
|
|
|
srcFormatId = ClipboardGetFormatId(clipboard->system, type_HtmlFormat);
|
|
|
|
dstFormatId = ClipboardGetFormatId(clipboard->system, mime_html);
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
nullTerminated = TRUE;
|
|
|
|
}
|
2019-02-20 19:05:47 +03:00
|
|
|
|
2023-03-13 14:53:40 +03:00
|
|
|
if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0)
|
2019-02-20 19:05:47 +03:00
|
|
|
{
|
2023-03-01 11:12:02 +03:00
|
|
|
if (!cliprdr_file_context_update_server_data(clipboard->file, clipboard->system, data,
|
|
|
|
size))
|
2023-02-28 13:37:42 +03:00
|
|
|
WLog_WARN(TAG, "failed to update file descriptors");
|
2020-12-27 09:08:35 +03:00
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
srcFormatId = ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
|
2023-02-24 10:59:36 +03:00
|
|
|
const xfCliprdrFormat* dstTargetFormat =
|
2020-12-27 09:08:35 +03:00
|
|
|
xf_cliprdr_get_client_format_by_atom(clipboard, clipboard->respond->target);
|
|
|
|
if (!dstTargetFormat)
|
|
|
|
{
|
2023-02-21 13:09:34 +03:00
|
|
|
dstFormatId = ClipboardGetFormatId(clipboard->system, mime_uri_list);
|
2020-12-27 09:08:35 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-02-22 09:15:59 +03:00
|
|
|
dstFormatId = dstTargetFormat->localFormat;
|
2020-12-27 09:08:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
nullTerminated = TRUE;
|
2019-02-20 19:05:47 +03:00
|
|
|
}
|
2023-03-22 11:53:30 +03:00
|
|
|
ClipboardUnlock(clipboard->system);
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-13 14:53:40 +03:00
|
|
|
srcFormatId = format->formatToRequest;
|
|
|
|
dstFormatId = format->localFormat;
|
|
|
|
switch (format->formatToRequest)
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
{
|
2016-10-06 14:31:25 +03:00
|
|
|
case CF_TEXT:
|
|
|
|
nullTerminated = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CF_OEMTEXT:
|
|
|
|
nullTerminated = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CF_UNICODETEXT:
|
|
|
|
nullTerminated = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CF_DIB:
|
|
|
|
srcFormatId = CF_DIB;
|
|
|
|
break;
|
2018-09-24 10:53:28 +03:00
|
|
|
|
2024-01-31 11:49:16 +03:00
|
|
|
case CF_TIFF:
|
|
|
|
srcFormatId = CF_TIFF;
|
|
|
|
break;
|
|
|
|
|
2018-09-24 10:53:28 +03:00
|
|
|
default:
|
|
|
|
break;
|
client/X11: improve named clipboard format support
Clipboard formats are identified by numerical IDs and literal names.
We can keep using arbitrary defined IDs for local clipboard formats
as we are sure that they have some fixed meaning, but the server can
and will be using its own IDs, which can be different from ours for
the named formats.
Therefore:
1) A correct way to compare a local format to a remote one is
to check the names first, and only then compare their IDs.
(Extra care should be taken to support short format names.)
2) Server IDs cannot be used with wClipboard directly when dealing
with named formats. Format name should be used to extract correct
local ID for the use with Clipboard{Set,Get}Data().
Also, I find the notion of 'alternate' format IDs to be confusing.
We either deal with a fixed ID format (declared in <winpr/user.h>),
or a format that was given an arbitrary fixed ID for local use
(defined in <freerdp/channels/cliprdr.h>), or a remote format
identified by a pair of an ID and a name. Format IDs can be local
and remote, but there are no 'alternates'.
So now:
1) A new function xf_cliprdr_formats_equal() is used to compare
formats correctly in xf_cliprdr_get_server_format_by_atom()
when searching for a server format corresponding to a local
one, and in xf_cliprdr_server_format_list() when constructing
a local TARGETS list from the server format list.
2) Correct local format IDs are used with wClipboard conversions
by xf_cliprdr_process_requested_data() and
xf_cliprdr_server_format_data_response().
3) We refer to formatId and formatName when doing requests,
and srcFormatId and dstFormatId when doing conversions,
instead of using formatId and altFormatId for both purposes.
4) Server format ID and name are used to identify cached clipboard
contents. The name is compared directly as a pointer because it
will be a pointer from the same clipboard->serverFormats array.
Also, the clipboard contents are invalidated when format list
arrives, so xf_cliprdr_server_format_list() now also clears
the format ID and name together with the data.
2015-08-17 11:35:19 +03:00
|
|
|
}
|
2014-10-18 04:55:12 +04:00
|
|
|
}
|
2014-10-16 01:42:55 +04:00
|
|
|
|
2023-02-22 09:15:59 +03:00
|
|
|
DEBUG_CLIPRDR("requested format 0x%08" PRIx32 " [%s] {local 0x%08" PRIx32 "} [%s]",
|
|
|
|
format->formatToRequest, ClipboardGetFormatIdString(format->formatToRequest),
|
|
|
|
format->localFormat, format->formatName);
|
2024-08-29 12:11:11 +03:00
|
|
|
SrcSize = size;
|
2023-02-28 13:37:42 +03:00
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
DEBUG_CLIPRDR("srcFormatId: %u, dstFormatId: %u", srcFormatId, dstFormatId);
|
|
|
|
|
2023-02-28 13:37:42 +03:00
|
|
|
ClipboardLock(clipboard->system);
|
2016-10-06 14:31:25 +03:00
|
|
|
bSuccess = ClipboardSetData(clipboard->system, srcFormatId, data, SrcSize);
|
2014-10-16 23:05:06 +04:00
|
|
|
|
2023-02-28 13:37:42 +03:00
|
|
|
BOOL willQuit = FALSE;
|
2015-08-19 10:25:59 +03:00
|
|
|
if (bSuccess)
|
2014-10-18 04:55:12 +04:00
|
|
|
{
|
2018-09-24 10:53:28 +03:00
|
|
|
if (SrcSize == 0)
|
|
|
|
{
|
2022-03-03 12:48:15 +03:00
|
|
|
WLog_DBG(TAG, "skipping, empty data detected!");
|
2020-03-26 20:50:44 +03:00
|
|
|
free(clipboard->respond);
|
|
|
|
clipboard->respond = NULL;
|
2023-02-28 13:37:42 +03:00
|
|
|
willQuit = TRUE;
|
2018-09-24 10:53:28 +03:00
|
|
|
}
|
2023-02-28 13:37:42 +03:00
|
|
|
else
|
2018-05-02 10:21:42 +03:00
|
|
|
{
|
2023-02-28 13:37:42 +03:00
|
|
|
pDstData = (BYTE*)ClipboardGetData(clipboard->system, dstFormatId, &DstSize);
|
2018-05-02 10:21:42 +03:00
|
|
|
|
2023-02-28 13:37:42 +03:00
|
|
|
if (!pDstData)
|
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "failed to get clipboard data in format %s [source format %s]",
|
|
|
|
ClipboardGetFormatName(clipboard->system, dstFormatId),
|
|
|
|
ClipboardGetFormatName(clipboard->system, srcFormatId));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nullTerminated && pDstData)
|
|
|
|
{
|
|
|
|
BYTE* nullTerminator = memchr(pDstData, '\0', DstSize);
|
|
|
|
if (nullTerminator)
|
|
|
|
DstSize = nullTerminator - pDstData;
|
|
|
|
}
|
2015-08-19 17:25:17 +03:00
|
|
|
}
|
2014-10-16 01:42:55 +04:00
|
|
|
}
|
2023-02-28 13:37:42 +03:00
|
|
|
ClipboardUnlock(clipboard->system);
|
|
|
|
if (willQuit)
|
|
|
|
return CHANNEL_RC_OK;
|
2014-10-16 01:42:55 +04:00
|
|
|
|
2016-08-15 22:45:20 +03:00
|
|
|
/* Cache converted and original data to avoid doing a possibly costly
|
|
|
|
* conversion again on subsequent requests */
|
2024-01-31 11:49:16 +03:00
|
|
|
if (pDstData)
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
{
|
2024-01-31 11:49:16 +03:00
|
|
|
cached_data = xf_cached_data_new(pDstData, DstSize);
|
|
|
|
if (!cached_data)
|
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "Failed to allocate cache entry");
|
|
|
|
free(pDstData);
|
|
|
|
return CHANNEL_RC_OK;
|
|
|
|
}
|
2024-09-10 14:24:50 +03:00
|
|
|
if (!HashTable_Insert(clipboard->cachedData, format_to_cache_slot(dstFormatId),
|
|
|
|
cached_data))
|
2024-01-31 11:49:16 +03:00
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "Failed to cache clipboard data");
|
|
|
|
xf_cached_data_free(cached_data);
|
|
|
|
return CHANNEL_RC_OK;
|
|
|
|
}
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
}
|
2023-02-24 11:27:27 +03:00
|
|
|
|
2016-08-15 22:45:20 +03:00
|
|
|
/* We have to copy the original data again, as pSrcData is now owned
|
|
|
|
* by clipboard->system. Memory allocation failure is not fatal here
|
|
|
|
* as this is only a cached value. */
|
|
|
|
{
|
2024-01-24 10:21:47 +03:00
|
|
|
// clipboard->cachedData owns cached_data
|
|
|
|
// NOLINTNEXTLINE(clang-analyzer-unix.Malloc
|
|
|
|
xfCachedData* cached_raw_data = xf_cached_data_new_copy(data, size);
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
if (!cached_raw_data)
|
|
|
|
WLog_WARN(TAG, "Failed to allocate cache entry");
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!HashTable_Insert(clipboard->cachedRawData, (void*)(UINT_PTR)srcFormatId,
|
|
|
|
cached_raw_data))
|
|
|
|
{
|
|
|
|
WLog_WARN(TAG, "Failed to cache clipboard data");
|
2023-09-20 16:36:05 +03:00
|
|
|
xf_cached_data_free(cached_raw_data);
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
}
|
|
|
|
}
|
2016-08-15 22:45:20 +03:00
|
|
|
}
|
|
|
|
|
2024-01-24 10:21:47 +03:00
|
|
|
// clipboard->cachedRawData owns cached_raw_data
|
|
|
|
// NOLINTNEXTLINE(clang-analyzer-unix.Malloc)
|
2014-10-18 04:55:12 +04:00
|
|
|
xf_cliprdr_provide_data(clipboard, clipboard->respond, pDstData, DstSize);
|
2020-02-27 10:28:52 +03:00
|
|
|
{
|
2020-12-31 17:40:39 +03:00
|
|
|
union
|
|
|
|
{
|
2020-02-27 10:28:52 +03:00
|
|
|
XEvent* ev;
|
|
|
|
XSelectionEvent* sev;
|
|
|
|
} conv;
|
|
|
|
|
|
|
|
conv.sev = clipboard->respond;
|
|
|
|
|
|
|
|
XSendEvent(xfc->display, clipboard->respond->requestor, 0, 0, conv.ev);
|
|
|
|
XFlush(xfc->display);
|
|
|
|
}
|
2014-10-16 01:42:55 +04:00
|
|
|
free(clipboard->respond);
|
|
|
|
clipboard->respond = NULL;
|
2015-06-02 14:05:10 +03:00
|
|
|
return CHANNEL_RC_OK;
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
2023-04-01 12:36:27 +03:00
|
|
|
static BOOL xf_cliprdr_is_valid_unix_filename(LPCWSTR filename)
|
|
|
|
{
|
|
|
|
if (!filename)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (filename[0] == L'\0')
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* Reserved characters */
|
2024-01-30 12:25:38 +03:00
|
|
|
for (const WCHAR* c = filename; *c; ++c)
|
2023-04-01 12:36:27 +03:00
|
|
|
{
|
|
|
|
if (*c == L'/')
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2022-07-06 14:46:39 +03:00
|
|
|
xfClipboard* xf_clipboard_new(xfContext* xfc, BOOL relieveFilenameRestriction)
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2023-02-22 12:13:18 +03:00
|
|
|
int n = 0;
|
2024-01-23 18:49:54 +03:00
|
|
|
rdpChannels* channels = NULL;
|
|
|
|
xfClipboard* clipboard = NULL;
|
|
|
|
const char* selectionAtom = NULL;
|
|
|
|
xfCliprdrFormat* clientFormat = NULL;
|
|
|
|
wObject* obj = NULL;
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2022-01-19 11:27:39 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
WINPR_ASSERT(xfc->common.context.settings);
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (!(clipboard = (xfClipboard*)calloc(1, sizeof(xfClipboard))))
|
2015-10-20 22:28:29 +03:00
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "failed to allocate xfClipboard data");
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2023-02-25 18:57:52 +03:00
|
|
|
clipboard->file = cliprdr_file_context_new(clipboard);
|
|
|
|
if (!clipboard->file)
|
|
|
|
goto fail;
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
xfc->clipboard = clipboard;
|
|
|
|
clipboard->xfc = xfc;
|
2022-07-01 12:04:22 +03:00
|
|
|
channels = xfc->common.context.channels;
|
2014-10-15 06:24:07 +04:00
|
|
|
clipboard->channels = channels;
|
2014-10-18 04:40:11 +04:00
|
|
|
clipboard->system = ClipboardCreate();
|
2014-10-15 23:49:57 +04:00
|
|
|
clipboard->requestedFormatId = -1;
|
2014-10-15 06:24:07 +04:00
|
|
|
clipboard->root_window = DefaultRootWindow(xfc->display);
|
2023-10-19 13:01:03 +03:00
|
|
|
|
|
|
|
selectionAtom =
|
|
|
|
freerdp_settings_get_string(xfc->common.context.settings, FreeRDP_ClipboardUseSelection);
|
|
|
|
if (!selectionAtom)
|
|
|
|
selectionAtom = "CLIPBOARD";
|
|
|
|
|
2024-07-25 08:36:09 +03:00
|
|
|
clipboard->clipboard_atom = Logging_XInternAtom(xfc->log, xfc->display, selectionAtom, FALSE);
|
2014-10-15 06:24:07 +04:00
|
|
|
|
|
|
|
if (clipboard->clipboard_atom == None)
|
|
|
|
{
|
2020-03-18 23:59:12 +03:00
|
|
|
WLog_ERR(TAG, "unable to get %s atom", selectionAtom);
|
2023-02-21 18:24:41 +03:00
|
|
|
goto fail;
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
2020-03-13 20:59:22 +03:00
|
|
|
clipboard->timestamp_property_atom =
|
2024-07-25 08:36:09 +03:00
|
|
|
Logging_XInternAtom(xfc->log, xfc->display, "_FREERDP_TIMESTAMP_PROPERTY", FALSE);
|
|
|
|
clipboard->property_atom =
|
|
|
|
Logging_XInternAtom(xfc->log, xfc->display, "_FREERDP_CLIPRDR", FALSE);
|
|
|
|
clipboard->raw_transfer_atom =
|
|
|
|
Logging_XInternAtom(xfc->log, xfc->display, "_FREERDP_CLIPRDR_RAW", FALSE);
|
|
|
|
clipboard->raw_format_list_atom =
|
|
|
|
Logging_XInternAtom(xfc->log, xfc->display, "_FREERDP_CLIPRDR_FORMATS", FALSE);
|
2015-08-14 22:57:42 +03:00
|
|
|
xf_cliprdr_set_raw_transfer_enabled(clipboard, TRUE);
|
2014-10-15 06:24:07 +04:00
|
|
|
XSelectInput(xfc->display, clipboard->root_window, PropertyChangeMask);
|
2017-04-09 02:29:51 +03:00
|
|
|
#ifdef WITH_XFIXES
|
2017-11-14 15:51:37 +03:00
|
|
|
|
2016-10-06 14:31:25 +03:00
|
|
|
if (XFixesQueryExtension(xfc->display, &clipboard->xfixes_event_base,
|
|
|
|
&clipboard->xfixes_error_base))
|
2014-10-15 06:24:07 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
int xfmajor = 0;
|
|
|
|
int xfminor = 0;
|
2014-10-16 06:48:18 +04:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
if (XFixesQueryVersion(xfc->display, &xfmajor, &xfminor))
|
|
|
|
{
|
|
|
|
XFixesSelectSelectionInput(xfc->display, clipboard->root_window,
|
2019-11-06 17:24:51 +03:00
|
|
|
clipboard->clipboard_atom,
|
|
|
|
XFixesSetSelectionOwnerNotifyMask);
|
2014-10-15 06:24:07 +04:00
|
|
|
clipboard->xfixes_supported = TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "Error querying X Fixes extension version");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "Error loading X Fixes extension");
|
|
|
|
}
|
2016-10-06 14:31:25 +03:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
#else
|
2019-11-06 17:24:51 +03:00
|
|
|
WLog_ERR(
|
|
|
|
TAG,
|
|
|
|
"Warning: Using clipboard redirection without XFIXES extension is strongly discouraged!");
|
2014-10-15 06:24:07 +04:00
|
|
|
#endif
|
2021-03-10 13:36:23 +03:00
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, "_FREERDP_RAW", False);
|
2023-02-22 09:15:59 +03:00
|
|
|
clientFormat->localFormat = clientFormat->formatToRequest = CF_RAW;
|
2021-03-10 13:36:23 +03:00
|
|
|
|
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, "UTF8_STRING", False);
|
2023-02-22 09:15:59 +03:00
|
|
|
clientFormat->formatToRequest = CF_UNICODETEXT;
|
|
|
|
clientFormat->localFormat = ClipboardGetFormatId(xfc->clipboard->system, mime_text_plain);
|
2021-03-10 13:36:23 +03:00
|
|
|
|
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
|
|
|
clientFormat->atom = XA_STRING;
|
2023-02-22 09:15:59 +03:00
|
|
|
clientFormat->formatToRequest = CF_TEXT;
|
|
|
|
clientFormat->localFormat = ClipboardGetFormatId(xfc->clipboard->system, mime_text_plain);
|
2021-03-10 13:36:23 +03:00
|
|
|
|
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_tiff, False);
|
2024-01-31 11:49:16 +03:00
|
|
|
clientFormat->formatToRequest = clientFormat->localFormat = CF_TIFF;
|
2021-03-10 13:36:23 +03:00
|
|
|
|
2024-01-31 11:49:16 +03:00
|
|
|
for (size_t x = 0; x < ARRAYSIZE(mime_bitmap); x++)
|
|
|
|
{
|
|
|
|
const char* mime_bmp = mime_bitmap[x];
|
|
|
|
const DWORD format = ClipboardGetFormatId(xfc->clipboard->system, mime_bmp);
|
|
|
|
if (format == 0)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "skipping local bitmap format %s [NOT SUPPORTED]", mime_bmp);
|
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 13:36:23 +03:00
|
|
|
|
2024-01-31 11:49:16 +03:00
|
|
|
WLog_DBG(TAG, "register local bitmap format %s [0x%08" PRIx32 "]", mime_bmp, format);
|
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
|
|
|
clientFormat->localFormat = format;
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_bmp, False);
|
2024-01-31 11:49:16 +03:00
|
|
|
clientFormat->formatToRequest = CF_DIB;
|
|
|
|
}
|
2021-03-10 13:36:23 +03:00
|
|
|
|
2024-01-31 11:49:16 +03:00
|
|
|
for (size_t x = 0; x < ARRAYSIZE(mime_images); x++)
|
|
|
|
{
|
|
|
|
const char* mime_bmp = mime_images[x];
|
|
|
|
const DWORD format = ClipboardGetFormatId(xfc->clipboard->system, mime_bmp);
|
|
|
|
if (format == 0)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "skipping local bitmap format %s [NOT SUPPORTED]", mime_bmp);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
WLog_DBG(TAG, "register local bitmap format %s [0x%08" PRIx32 "]", mime_bmp, format);
|
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
|
|
|
clientFormat->localFormat = format;
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_bmp, False);
|
2024-01-31 11:49:16 +03:00
|
|
|
clientFormat->formatToRequest = CF_DIB;
|
|
|
|
}
|
2021-03-10 13:36:23 +03:00
|
|
|
|
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_html, False);
|
2023-02-22 09:15:59 +03:00
|
|
|
clientFormat->formatToRequest = ClipboardGetFormatId(xfc->clipboard->system, type_HtmlFormat);
|
|
|
|
clientFormat->localFormat = ClipboardGetFormatId(xfc->clipboard->system, mime_html);
|
|
|
|
clientFormat->formatName = _strdup(type_HtmlFormat);
|
2021-03-10 13:36:23 +03:00
|
|
|
|
|
|
|
if (!clientFormat->formatName)
|
2023-02-21 18:24:41 +03:00
|
|
|
goto fail;
|
2017-11-14 15:51:37 +03:00
|
|
|
|
2021-03-10 13:36:23 +03:00
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
2017-04-09 02:29:51 +03:00
|
|
|
|
2017-04-09 02:29:51 +03:00
|
|
|
/*
|
|
|
|
* Existence of registered format IDs for file formats does not guarantee that they are
|
|
|
|
* in fact supported by wClipboard (as further initialization may have failed after format
|
|
|
|
* registration). However, they are definitely not supported if there are no registered
|
|
|
|
* formats. In this case we should not list file formats in TARGETS.
|
|
|
|
*/
|
2023-02-22 09:15:59 +03:00
|
|
|
const UINT32 fgid = ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
|
2023-02-21 18:24:41 +03:00
|
|
|
const UINT32 uid = ClipboardGetFormatId(clipboard->system, mime_uri_list);
|
|
|
|
if (uid)
|
2017-04-09 02:29:51 +03:00
|
|
|
{
|
2023-02-25 18:57:52 +03:00
|
|
|
cliprdr_file_context_set_locally_available(clipboard->file, TRUE);
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_uri_list, False);
|
2023-02-22 09:15:59 +03:00
|
|
|
clientFormat->localFormat = uid;
|
|
|
|
clientFormat->formatToRequest = fgid;
|
|
|
|
clientFormat->formatName = _strdup(type_FileGroupDescriptorW);
|
2017-11-14 15:51:37 +03:00
|
|
|
|
2021-03-10 13:36:23 +03:00
|
|
|
if (!clientFormat->formatName)
|
2023-02-21 18:24:41 +03:00
|
|
|
goto fail;
|
2017-11-14 15:51:37 +03:00
|
|
|
|
2021-03-10 13:36:23 +03:00
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
2015-06-17 23:08:02 +03:00
|
|
|
}
|
2014-10-15 06:24:07 +04:00
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
const UINT32 gid = ClipboardGetFormatId(clipboard->system, mime_gnome_copied_files);
|
|
|
|
if (gid != 0)
|
|
|
|
{
|
2023-02-25 18:57:52 +03:00
|
|
|
cliprdr_file_context_set_locally_available(clipboard->file, TRUE);
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom =
|
|
|
|
Logging_XInternAtom(xfc->log, xfc->display, mime_gnome_copied_files, False);
|
2023-02-22 09:15:59 +03:00
|
|
|
clientFormat->localFormat = gid;
|
|
|
|
clientFormat->formatToRequest = fgid;
|
|
|
|
clientFormat->formatName = _strdup(type_FileGroupDescriptorW);
|
2023-02-21 18:24:41 +03:00
|
|
|
|
|
|
|
if (!clientFormat->formatName)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
clientFormat = &clipboard->clientFormats[n++];
|
|
|
|
}
|
|
|
|
|
|
|
|
const UINT32 mid = ClipboardGetFormatId(clipboard->system, mime_mate_copied_files);
|
|
|
|
if (mid != 0)
|
|
|
|
{
|
2023-02-25 18:57:52 +03:00
|
|
|
cliprdr_file_context_set_locally_available(clipboard->file, TRUE);
|
2024-07-25 08:36:09 +03:00
|
|
|
clientFormat->atom =
|
|
|
|
Logging_XInternAtom(xfc->log, xfc->display, mime_mate_copied_files, False);
|
2023-02-22 09:15:59 +03:00
|
|
|
clientFormat->localFormat = mid;
|
|
|
|
clientFormat->formatToRequest = fgid;
|
|
|
|
clientFormat->formatName = _strdup(type_FileGroupDescriptorW);
|
2023-02-21 18:24:41 +03:00
|
|
|
|
|
|
|
if (!clientFormat->formatName)
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2014-10-15 23:49:57 +04:00
|
|
|
clipboard->numClientFormats = n;
|
2024-07-25 08:36:09 +03:00
|
|
|
clipboard->targets[0] = Logging_XInternAtom(xfc->log, xfc->display, "TIMESTAMP", FALSE);
|
|
|
|
clipboard->targets[1] = Logging_XInternAtom(xfc->log, xfc->display, "TARGETS", FALSE);
|
2014-10-15 23:49:57 +04:00
|
|
|
clipboard->numTargets = 2;
|
2024-07-25 08:36:09 +03:00
|
|
|
clipboard->incr_atom = Logging_XInternAtom(xfc->log, xfc->display, "INCR", FALSE);
|
2020-12-27 09:08:35 +03:00
|
|
|
|
2023-04-01 12:36:27 +03:00
|
|
|
if (relieveFilenameRestriction)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "Relieving CLIPRDR filename restriction");
|
|
|
|
ClipboardGetDelegate(clipboard->system)->IsFileNameComponentValid =
|
|
|
|
xf_cliprdr_is_valid_unix_filename;
|
|
|
|
}
|
|
|
|
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
clipboard->cachedData = HashTable_New(TRUE);
|
|
|
|
if (!clipboard->cachedData)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
obj = HashTable_ValueObject(clipboard->cachedData);
|
2023-09-20 16:36:05 +03:00
|
|
|
obj->fnObjectFree = xf_cached_data_free;
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
|
|
|
|
clipboard->cachedRawData = HashTable_New(TRUE);
|
|
|
|
if (!clipboard->cachedRawData)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
obj = HashTable_ValueObject(clipboard->cachedRawData);
|
2023-09-20 16:36:05 +03:00
|
|
|
obj->fnObjectFree = xf_cached_data_free;
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
return clipboard;
|
2021-01-05 16:13:58 +03:00
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
fail:
|
2024-02-04 13:11:29 +03:00
|
|
|
WINPR_PRAGMA_DIAG_PUSH
|
|
|
|
WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
|
2023-02-21 18:24:41 +03:00
|
|
|
xf_clipboard_free(clipboard);
|
2024-02-04 13:11:29 +03:00
|
|
|
WINPR_PRAGMA_DIAG_POP
|
2017-04-09 02:29:51 +03:00
|
|
|
return NULL;
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void xf_clipboard_free(xfClipboard* clipboard)
|
|
|
|
{
|
|
|
|
if (!clipboard)
|
|
|
|
return;
|
|
|
|
|
2023-02-21 18:24:41 +03:00
|
|
|
xf_clipboard_free_server_formats(clipboard);
|
2014-10-16 06:56:25 +04:00
|
|
|
|
2014-12-03 22:17:27 +03:00
|
|
|
if (clipboard->numClientFormats)
|
|
|
|
{
|
2023-02-22 12:13:18 +03:00
|
|
|
for (UINT32 i = 0; i < clipboard->numClientFormats; i++)
|
2021-03-10 13:36:23 +03:00
|
|
|
{
|
|
|
|
xfCliprdrFormat* format = &clipboard->clientFormats[i];
|
|
|
|
free(format->formatName);
|
|
|
|
}
|
2014-12-03 22:17:27 +03:00
|
|
|
}
|
|
|
|
|
2023-02-25 14:42:04 +03:00
|
|
|
cliprdr_file_context_free(clipboard->file);
|
2020-12-27 09:08:35 +03:00
|
|
|
|
2014-10-18 04:40:11 +04:00
|
|
|
ClipboardDestroy(clipboard->system);
|
2020-11-26 19:38:04 +03:00
|
|
|
xf_clipboard_formats_free(clipboard);
|
client/X11: Fix IO errors during server to client file transfer
When receiving a file list, xfreerdp3 rebuilds the content of the FUSE
filesystem.
Since fetching uri-lists can happen during a paste action too, xfreerdp3
caches the content of the last fetched mime type.
However, uri-lists exists in different variations, e.g. nautilus uses a
different mime type, than gnome-terminal does.
Furthermore, FormatLists can also contain other formats in addition to
file lists.
Fetching those contents during a paste operation leads to IO errors in
the paste operation.
In order to fix those errors, cache every mimetype in two hash tables:
One for the raw (unconverted) data, and one for the converted data.
When a content request is received, xfreerdp3 can with the already
cached converted data, directly serve the data.
If the content is not available as cached data, but its source data was
already fetched, use that source data to create the converted data.
Then serve the converted data.
Only if no cached converted data or cached raw data is available, issue
a new FormatDataRequest.
2023-04-14 14:02:26 +03:00
|
|
|
HashTable_Free(clipboard->cachedRawData);
|
|
|
|
HashTable_Free(clipboard->cachedData);
|
2014-10-15 06:24:07 +04:00
|
|
|
free(clipboard->respond);
|
|
|
|
free(clipboard->incr_data);
|
|
|
|
free(clipboard);
|
|
|
|
}
|
|
|
|
|
|
|
|
void xf_cliprdr_init(xfContext* xfc, CliprdrClientContext* cliprdr)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
WINPR_ASSERT(cliprdr);
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
xfc->cliprdr = cliprdr;
|
2014-10-15 06:58:01 +04:00
|
|
|
xfc->clipboard->context = cliprdr;
|
2023-02-25 14:42:04 +03:00
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
cliprdr->MonitorReady = xf_cliprdr_monitor_ready;
|
|
|
|
cliprdr->ServerCapabilities = xf_cliprdr_server_capabilities;
|
|
|
|
cliprdr->ServerFormatList = xf_cliprdr_server_format_list;
|
|
|
|
cliprdr->ServerFormatListResponse = xf_cliprdr_server_format_list_response;
|
|
|
|
cliprdr->ServerFormatDataRequest = xf_cliprdr_server_format_data_request;
|
|
|
|
cliprdr->ServerFormatDataResponse = xf_cliprdr_server_format_data_response;
|
2023-02-25 14:42:04 +03:00
|
|
|
|
|
|
|
cliprdr_file_context_init(xfc->clipboard->file, cliprdr);
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void xf_cliprdr_uninit(xfContext* xfc, CliprdrClientContext* cliprdr)
|
|
|
|
{
|
2022-07-01 12:04:22 +03:00
|
|
|
WINPR_ASSERT(xfc);
|
|
|
|
WINPR_ASSERT(cliprdr);
|
|
|
|
|
2014-10-15 06:24:07 +04:00
|
|
|
xfc->cliprdr = NULL;
|
2014-10-16 01:42:55 +04:00
|
|
|
|
|
|
|
if (xfc->clipboard)
|
2023-02-25 14:42:04 +03:00
|
|
|
{
|
|
|
|
cliprdr_file_context_uninit(xfc->clipboard->file, cliprdr);
|
2014-10-16 01:42:55 +04:00
|
|
|
xfc->clipboard->context = NULL;
|
2023-02-25 14:42:04 +03:00
|
|
|
}
|
2014-10-15 06:24:07 +04:00
|
|
|
}
|