Blame client/X11/xf_cliprdr.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * X11 Clipboard Redirection
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2010-2011 Vic Lee
Packit Service fa4841
 * Copyright 2015 Thincast Technologies GmbH
Packit Service fa4841
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
Packit Service fa4841
 *
Packit Service fa4841
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service fa4841
 * you may not use this file except in compliance with the License.
Packit Service fa4841
 * You may obtain a copy of the License at
Packit Service fa4841
 *
Packit Service fa4841
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service fa4841
 *
Packit Service fa4841
 * Unless required by applicable law or agreed to in writing, software
Packit Service fa4841
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service fa4841
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service fa4841
 * See the License for the specific language governing permissions and
Packit Service fa4841
 * limitations under the License.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
#ifdef HAVE_CONFIG_H
Packit Service fa4841
#include "config.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <stdlib.h>
Packit Service fa4841
#include <X11/Xlib.h>
Packit Service fa4841
#include <X11/Xatom.h>
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_XFIXES
Packit Service fa4841
#include <X11/extensions/Xfixes.h>
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <winpr/crt.h>
Packit Service fa4841
#include <winpr/image.h>
Packit Service fa4841
#include <winpr/stream.h>
Packit Service fa4841
#include <winpr/clipboard.h>
Packit Service fa4841
Packit Service fa4841
#include <freerdp/log.h>
Packit Service fa4841
#include <freerdp/client/cliprdr.h>
Packit Service fa4841
#include <freerdp/channels/channels.h>
Packit Service fa4841
#include <freerdp/channels/cliprdr.h>
Packit Service fa4841
Packit Service fa4841
#include "xf_cliprdr.h"
Packit Service fa4841
Packit Service fa4841
#define TAG CLIENT_TAG("x11")
Packit Service fa4841
Packit Service fa4841
#define MAX_CLIPBOARD_FORMATS 255
Packit Service fa4841
Packit Service fa4841
struct xf_cliprdr_format
Packit Service fa4841
{
Packit Service fa4841
	Atom atom;
Packit Service fa4841
	UINT32 formatId;
Packit Service fa4841
	char* formatName;
Packit Service fa4841
};
Packit Service fa4841
typedef struct xf_cliprdr_format xfCliprdrFormat;
Packit Service fa4841
Packit Service fa4841
struct xf_clipboard
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc;
Packit Service fa4841
	rdpChannels* channels;
Packit Service fa4841
	CliprdrClientContext* context;
Packit Service fa4841
Packit Service fa4841
	wClipboard* system;
Packit Service fa4841
	wClipboardDelegate* delegate;
Packit Service fa4841
Packit Service fa4841
	Window root_window;
Packit Service fa4841
	Atom clipboard_atom;
Packit Service fa4841
	Atom property_atom;
Packit Service fa4841
Packit Service fa4841
	Atom raw_transfer_atom;
Packit Service fa4841
	Atom raw_format_list_atom;
Packit Service fa4841
Packit Service fa4841
	int numClientFormats;
Packit Service fa4841
	xfCliprdrFormat clientFormats[20];
Packit Service fa4841
Packit Service fa4841
	int numServerFormats;
Packit Service fa4841
	CLIPRDR_FORMAT* serverFormats;
Packit Service fa4841
Packit Service fa4841
	int numTargets;
Packit Service fa4841
	Atom targets[20];
Packit Service fa4841
Packit Service fa4841
	int requestedFormatId;
Packit Service fa4841
Packit Service fa4841
	BYTE* data;
Packit Service fa4841
	BYTE* data_raw;
Packit Service fa4841
	BOOL data_raw_format;
Packit Service fa4841
	UINT32 data_format_id;
Packit Service fa4841
	const char* data_format_name;
Packit Service fa4841
	int data_length;
Packit Service fa4841
	int data_raw_length;
Packit Service fa4841
	XSelectionEvent* respond;
Packit Service fa4841
Packit Service fa4841
	Window owner;
Packit Service fa4841
	BOOL sync;
Packit Service fa4841
Packit Service fa4841
	/* INCR mechanism */
Packit Service fa4841
	Atom incr_atom;
Packit Service fa4841
	BOOL incr_starts;
Packit Service fa4841
	BYTE* incr_data;
Packit Service fa4841
	int incr_data_length;
Packit Service fa4841
Packit Service fa4841
	/* XFixes extension */
Packit Service fa4841
	int xfixes_event_base;
Packit Service fa4841
	int xfixes_error_base;
Packit Service fa4841
	BOOL xfixes_supported;
Packit Service fa4841
Packit Service fa4841
	/* File clipping */
Packit Service fa4841
	BOOL streams_supported;
Packit Service fa4841
	BOOL file_formats_registered;
Packit Service fa4841
};
Packit Service fa4841
Packit Service fa4841
static UINT xf_cliprdr_send_client_format_list(xfClipboard* clipboard);
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_check_owner(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	Window owner;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->sync)
Packit Service fa4841
	{
Packit Service fa4841
		owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
Packit Service fa4841
Packit Service fa4841
		if (clipboard->owner != owner)
Packit Service fa4841
		{
Packit Service fa4841
			clipboard->owner = owner;
Packit Service fa4841
			xf_cliprdr_send_client_format_list(clipboard);
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_is_self_owned(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	return XGetSelectionOwner(xfc->display, clipboard->clipboard_atom) == xfc->drawable;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_set_raw_transfer_enabled(xfClipboard* clipboard, BOOL enabled)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 data = enabled;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	XChangeProperty(xfc->display, xfc->drawable, clipboard->raw_transfer_atom, XA_INTEGER, 32,
Packit Service fa4841
	                PropModeReplace, (BYTE*)&data, 1);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_is_raw_transfer_available(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	Atom type;
Packit Service fa4841
	int format;
Packit Service fa4841
	int result = 0;
Packit Service fa4841
	unsigned long length;
Packit Service fa4841
	unsigned long bytes_left;
Packit Service fa4841
	UINT32* data = NULL;
Packit Service fa4841
	UINT32 is_enabled = 0;
Packit Service fa4841
	Window owner = None;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
Packit Service fa4841
Packit Service fa4841
	if (owner != None)
Packit Service fa4841
	{
Packit Service fa4841
		result =
Packit Service fa4841
		    XGetWindowProperty(xfc->display, owner, clipboard->raw_transfer_atom, 0, 4, 0,
Packit Service fa4841
		                       XA_INTEGER, &type, &format, &length, &bytes_left, (BYTE**)&data);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (data)
Packit Service fa4841
	{
Packit Service fa4841
		is_enabled = *data;
Packit Service fa4841
		XFree(data);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if ((owner == None) || (owner == xfc->drawable))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (result != Success)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	return is_enabled ? TRUE : FALSE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_formats_equal(const CLIPRDR_FORMAT* server, const xfCliprdrFormat* client)
Packit Service fa4841
{
Packit Service fa4841
	if (server->formatName && client->formatName)
Packit Service fa4841
	{
Packit Service fa4841
		/* The server may be using short format names while we store them in full form. */
Packit Service fa4841
		return (0 == strncmp(server->formatName, client->formatName, strlen(server->formatName)));
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!server->formatName && !client->formatName)
Packit Service fa4841
	{
Packit Service fa4841
		return (server->formatId == client->formatId);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return FALSE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static xfCliprdrFormat* xf_cliprdr_get_client_format_by_id(xfClipboard* clipboard, UINT32 formatId)
Packit Service fa4841
{
Packit Service fa4841
	int index;
Packit Service fa4841
	xfCliprdrFormat* format;
Packit Service fa4841
Packit Service fa4841
	for (index = 0; index < clipboard->numClientFormats; index++)
Packit Service fa4841
	{
Packit Service fa4841
		format = &(clipboard->clientFormats[index]);
Packit Service fa4841
Packit Service fa4841
		if (format->formatId == formatId)
Packit Service fa4841
			return format;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static xfCliprdrFormat* xf_cliprdr_get_client_format_by_atom(xfClipboard* clipboard, Atom atom)
Packit Service fa4841
{
Packit Service fa4841
	int i;
Packit Service fa4841
	xfCliprdrFormat* format;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < clipboard->numClientFormats; i++)
Packit Service fa4841
	{
Packit Service fa4841
		format = &(clipboard->clientFormats[i]);
Packit Service fa4841
Packit Service fa4841
		if (format->atom == atom)
Packit Service fa4841
			return format;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static CLIPRDR_FORMAT* xf_cliprdr_get_server_format_by_atom(xfClipboard* clipboard, Atom atom)
Packit Service fa4841
{
Packit Service fa4841
	int i, j;
Packit Service fa4841
	xfCliprdrFormat* client_format;
Packit Service fa4841
	CLIPRDR_FORMAT* server_format;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < clipboard->numClientFormats; i++)
Packit Service fa4841
	{
Packit Service fa4841
		client_format = &(clipboard->clientFormats[i]);
Packit Service fa4841
Packit Service fa4841
		if (client_format->atom == atom)
Packit Service fa4841
		{
Packit Service fa4841
			for (j = 0; j < clipboard->numServerFormats; j++)
Packit Service fa4841
			{
Packit Service fa4841
				server_format = &(clipboard->serverFormats[j]);
Packit Service fa4841
Packit Service fa4841
				if (xf_cliprdr_formats_equal(server_format, client_format))
Packit Service fa4841
					return server_format;
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_send_data_request(xfClipboard* clipboard, UINT32 formatId)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FORMAT_DATA_REQUEST request = { 0 };
Packit Service fa4841
	request.requestedFormatId = formatId;
Packit Service fa4841
	return clipboard->context->ClientFormatDataRequest(clipboard->context, &request);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_send_data_response(xfClipboard* clipboard, BYTE* data, int size)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FORMAT_DATA_RESPONSE response = { 0 };
Packit Service fa4841
	response.msgFlags = (data) ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
Packit Service fa4841
	response.dataLen = size;
Packit Service fa4841
	response.requestedFormatData = data;
Packit Service fa4841
	return clipboard->context->ClientFormatDataResponse(clipboard->context, &response);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static wStream* xf_cliprdr_serialize_server_format_list(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i;
Packit Service fa4841
	UINT32 formatCount;
Packit Service fa4841
	wStream* s = NULL;
Packit Service fa4841
Packit Service fa4841
	/* Typical MS Word format list is about 80 bytes long. */
Packit Service fa4841
	if (!(s = Stream_New(NULL, 128)))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "failed to allocate serialized format list");
Packit Service fa4841
		goto error;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* If present, the last format is always synthetic CF_RAW. Do not include it. */
Packit Service fa4841
	formatCount = (clipboard->numServerFormats > 0) ? clipboard->numServerFormats - 1 : 0;
Packit Service fa4841
	Stream_Write_UINT32(s, formatCount);
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < formatCount; i++)
Packit Service fa4841
	{
Packit Service fa4841
		CLIPRDR_FORMAT* format = &clipboard->serverFormats[i];
Packit Service fa4841
		size_t name_length = format->formatName ? strlen(format->formatName) : 0;
Packit Service fa4841
Packit Service fa4841
		if (!Stream_EnsureRemainingCapacity(s, sizeof(UINT32) + name_length + 1))
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "failed to expand serialized format list");
Packit Service fa4841
			goto error;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		Stream_Write_UINT32(s, format->formatId);
Packit Service fa4841
Packit Service fa4841
		if (format->formatName)
Packit Service fa4841
			Stream_Write(s, format->formatName, name_length);
Packit Service fa4841
Packit Service fa4841
		Stream_Write_UINT8(s, '\0');
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	Stream_SealLength(s);
Packit Service fa4841
	return s;
Packit Service fa4841
error:
Packit Service fa4841
	Stream_Free(s, TRUE);
Packit Service fa4841
	return NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static CLIPRDR_FORMAT* xf_cliprdr_parse_server_format_list(BYTE* data, size_t length,
Packit Service fa4841
                                                           UINT32* numFormats)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i;
Packit Service fa4841
	wStream* s = NULL;
Packit Service fa4841
	CLIPRDR_FORMAT* formats = NULL;
Packit Service fa4841
Packit Service fa4841
	if (!(s = Stream_New(data, length)))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "failed to allocate stream for parsing serialized format list");
Packit Service fa4841
		goto error;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (Stream_GetRemainingLength(s) < sizeof(UINT32))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "too short serialized format list");
Packit Service fa4841
		goto error;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	Stream_Read_UINT32(s, *numFormats);
Packit Service fa4841
Packit Service fa4841
	if (*numFormats > MAX_CLIPBOARD_FORMATS)
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "unexpectedly large number of formats: %" PRIu32 "", *numFormats);
Packit Service fa4841
		goto error;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!(formats = (CLIPRDR_FORMAT*)calloc(*numFormats, sizeof(CLIPRDR_FORMAT))))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "failed to allocate format list");
Packit Service fa4841
		goto error;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < *numFormats; i++)
Packit Service fa4841
	{
Packit Service fa4841
		const char* formatName = NULL;
Packit Service fa4841
		size_t formatNameLength = 0;
Packit Service fa4841
Packit Service fa4841
		if (Stream_GetRemainingLength(s) < sizeof(UINT32))
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "unexpected end of serialized format list");
Packit Service fa4841
			goto error;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		Stream_Read_UINT32(s, formats[i].formatId);
Packit Service fa4841
		formatName = (const char*)Stream_Pointer(s);
Packit Service fa4841
		formatNameLength = strnlen(formatName, Stream_GetRemainingLength(s));
Packit Service fa4841
Packit Service fa4841
		if (formatNameLength == Stream_GetRemainingLength(s))
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "missing terminating null byte, %" PRIuz " bytes left to read",
Packit Service fa4841
			         formatNameLength);
Packit Service fa4841
			goto error;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		formats[i].formatName = strndup(formatName, formatNameLength);
Packit Service fa4841
		Stream_Seek(s, formatNameLength + 1);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	Stream_Free(s, FALSE);
Packit Service fa4841
	return formats;
Packit Service fa4841
error:
Packit Service fa4841
	Stream_Free(s, FALSE);
Packit Service fa4841
	free(formats);
Packit Service fa4841
	*numFormats = 0;
Packit Service fa4841
	return NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_free_formats(CLIPRDR_FORMAT* formats, UINT32 numFormats)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < numFormats; i++)
Packit Service fa4841
	{
Packit Service fa4841
		free(formats[i].formatName);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	free(formats);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static CLIPRDR_FORMAT* xf_cliprdr_get_raw_server_formats(xfClipboard* clipboard, UINT32* numFormats)
Packit Service fa4841
{
Packit Service fa4841
	Atom type = None;
Packit Service fa4841
	int format = 0;
Packit Service fa4841
	unsigned long length = 0;
Packit Service fa4841
	unsigned long remaining;
Packit Service fa4841
	BYTE* data = NULL;
Packit Service fa4841
	CLIPRDR_FORMAT* formats = NULL;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	*numFormats = 0;
Packit Service fa4841
	XGetWindowProperty(xfc->display, clipboard->owner, clipboard->raw_format_list_atom, 0, 4096,
Packit Service fa4841
	                   False, clipboard->raw_format_list_atom, &type, &format, &length, &remaining,
Packit Service fa4841
	                   &data);
Packit Service fa4841
Packit Service fa4841
	if (data && length > 0 && format == 8 && type == clipboard->raw_format_list_atom)
Packit Service fa4841
	{
Packit Service fa4841
		formats = xf_cliprdr_parse_server_format_list(data, length, numFormats);
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG,
Packit Service fa4841
		         "failed to retrieve raw format list: data=%p, length=%lu, format=%d, type=%lu "
Packit Service fa4841
		         "(expected=%lu)",
Packit Service fa4841
		         (void*)data, length, format, (unsigned long)type,
Packit Service fa4841
		         (unsigned long)clipboard->raw_format_list_atom);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (data)
Packit Service fa4841
		XFree(data);
Packit Service fa4841
Packit Service fa4841
	return formats;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static CLIPRDR_FORMAT* xf_cliprdr_get_formats_from_targets(xfClipboard* clipboard,
Packit Service fa4841
                                                           UINT32* numFormats)
Packit Service fa4841
{
Packit Service fa4841
	unsigned long i;
Packit Service fa4841
	Atom atom;
Packit Service fa4841
	BYTE* data = NULL;
Packit Service fa4841
	int format_property;
Packit Service fa4841
	unsigned long length;
Packit Service fa4841
	unsigned long bytes_left;
Packit Service fa4841
	xfCliprdrFormat* format = NULL;
Packit Service fa4841
	CLIPRDR_FORMAT* formats = NULL;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	*numFormats = 0;
Packit Service fa4841
	XGetWindowProperty(xfc->display, xfc->drawable, clipboard->property_atom, 0, 200, 0, XA_ATOM,
Packit Service fa4841
	                   &atom, &format_property, &length, &bytes_left, &data);
Packit Service fa4841
Packit Service fa4841
	if (length > 0)
Packit Service fa4841
	{
Packit Service fa4841
		if (!data)
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "XGetWindowProperty set length = %lu but data is NULL", length);
Packit Service fa4841
			goto out;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (!(formats = (CLIPRDR_FORMAT*)calloc(length, sizeof(CLIPRDR_FORMAT))))
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "failed to allocate %lu CLIPRDR_FORMAT structs", length);
Packit Service fa4841
			goto out;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < length; i++)
Packit Service fa4841
	{
Packit Service fa4841
		atom = ((Atom*)data)[i];
Packit Service fa4841
		format = xf_cliprdr_get_client_format_by_atom(clipboard, atom);
Packit Service fa4841
Packit Service fa4841
		if (format)
Packit Service fa4841
		{
Packit Service fa4841
			formats[*numFormats].formatId = format->formatId;
Packit Service fa4841
			formats[*numFormats].formatName = _strdup(format->formatName);
Packit Service fa4841
			*numFormats += 1;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
out:
Packit Service fa4841
Packit Service fa4841
	if (data)
Packit Service fa4841
		XFree(data);
Packit Service fa4841
Packit Service fa4841
	return formats;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static CLIPRDR_FORMAT* xf_cliprdr_get_client_formats(xfClipboard* clipboard, UINT32* numFormats)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FORMAT* formats = NULL;
Packit Service fa4841
	*numFormats = 0;
Packit Service fa4841
Packit Service fa4841
	if (xf_cliprdr_is_raw_transfer_available(clipboard))
Packit Service fa4841
	{
Packit Service fa4841
		formats = xf_cliprdr_get_raw_server_formats(clipboard, numFormats);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (*numFormats == 0)
Packit Service fa4841
	{
Packit Service fa4841
		xf_cliprdr_free_formats(formats, *numFormats);
Packit Service fa4841
		formats = xf_cliprdr_get_formats_from_targets(clipboard, numFormats);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return formats;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_provide_server_format_list(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	wStream* formats = NULL;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	formats = xf_cliprdr_serialize_server_format_list(clipboard);
Packit Service fa4841
Packit Service fa4841
	if (formats)
Packit Service fa4841
	{
Packit Service fa4841
		XChangeProperty(xfc->display, xfc->drawable, clipboard->raw_format_list_atom,
Packit Service fa4841
		                clipboard->raw_format_list_atom, 8, PropModeReplace, Stream_Buffer(formats),
Packit Service fa4841
		                Stream_Length(formats));
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		XDeleteProperty(xfc->display, xfc->drawable, clipboard->raw_format_list_atom);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	Stream_Free(formats, TRUE);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_get_requested_targets(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 numFormats = 0;
Packit Service fa4841
	CLIPRDR_FORMAT* formats = NULL;
Packit Service fa4841
	CLIPRDR_FORMAT_LIST formatList = { 0 };
Packit Service fa4841
	formats = xf_cliprdr_get_client_formats(clipboard, &numFormats);
Packit Service fa4841
	formatList.msgFlags = CB_RESPONSE_OK;
Packit Service fa4841
	formatList.numFormats = numFormats;
Packit Service fa4841
	formatList.formats = formats;
Packit Service fa4841
	formatList.msgType = CB_FORMAT_LIST;
Packit Service fa4841
	clipboard->context->ClientFormatList(clipboard->context, &formatList);
Packit Service fa4841
	xf_cliprdr_free_formats(formats, numFormats);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_process_requested_data(xfClipboard* clipboard, BOOL hasData, BYTE* data,
Packit Service fa4841
                                              int size)
Packit Service fa4841
{
Packit Service fa4841
	BOOL bSuccess;
Packit Service fa4841
	UINT32 SrcSize;
Packit Service fa4841
	UINT32 DstSize;
Packit Service fa4841
	UINT32 srcFormatId;
Packit Service fa4841
	UINT32 dstFormatId;
Packit Service fa4841
	BYTE* pDstData = NULL;
Packit Service fa4841
	xfCliprdrFormat* format;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->incr_starts && hasData)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	format = xf_cliprdr_get_client_format_by_id(clipboard, clipboard->requestedFormatId);
Packit Service fa4841
Packit Service fa4841
	if (!hasData || !data || !format)
Packit Service fa4841
	{
Packit Service fa4841
		xf_cliprdr_send_data_response(clipboard, NULL, 0);
Packit Service fa4841
		return;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	srcFormatId = 0;
Packit Service fa4841
Packit Service fa4841
	switch (format->formatId)
Packit Service fa4841
	{
Packit Service fa4841
		case CF_RAW:
Packit Service fa4841
			srcFormatId = CF_RAW;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case CF_TEXT:
Packit Service fa4841
		case CF_OEMTEXT:
Packit Service fa4841
		case CF_UNICODETEXT:
Packit Service fa4841
			size = strlen((char*)data) + 1;
Packit Service fa4841
			srcFormatId = ClipboardGetFormatId(clipboard->system, "UTF8_STRING");
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case CF_DIB:
Packit Service fa4841
			srcFormatId = ClipboardGetFormatId(clipboard->system, "image/bmp");
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case CB_FORMAT_HTML:
Packit Service fa4841
			srcFormatId = ClipboardGetFormatId(clipboard->system, "text/html");
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case CB_FORMAT_TEXTURILIST:
Packit Service fa4841
			srcFormatId = ClipboardGetFormatId(clipboard->system, "text/uri-list");
Packit Service fa4841
			break;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	SrcSize = (UINT32)size;
Packit Service fa4841
	bSuccess = ClipboardSetData(clipboard->system, srcFormatId, data, SrcSize);
Packit Service fa4841
Packit Service fa4841
	if (format->formatName)
Packit Service fa4841
		dstFormatId = ClipboardGetFormatId(clipboard->system, format->formatName);
Packit Service fa4841
	else
Packit Service fa4841
		dstFormatId = format->formatId;
Packit Service fa4841
Packit Service fa4841
	if (bSuccess)
Packit Service fa4841
	{
Packit Service fa4841
		DstSize = 0;
Packit Service fa4841
		pDstData = (BYTE*)ClipboardGetData(clipboard->system, dstFormatId, &DstSize);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!pDstData)
Packit Service fa4841
	{
Packit Service fa4841
		xf_cliprdr_send_data_response(clipboard, NULL, 0);
Packit Service fa4841
		return;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/*
Packit Service fa4841
	 * File lists require a bit of postprocessing to convert them from WinPR's FILDESCRIPTOR
Packit Service fa4841
	 * format to CLIPRDR_FILELIST expected by the server.
Packit Service fa4841
	 *
Packit Service fa4841
	 * We check for "FileGroupDescriptorW" format being registered (i.e., nonzero) in order
Packit Service fa4841
	 * to not process CF_RAW as a file list in case WinPR does not support file transfers.
Packit Service fa4841
	 */
Packit Service fa4841
	if (dstFormatId &&
Packit Service fa4841
	    (dstFormatId == ClipboardGetFormatId(clipboard->system, "FileGroupDescriptorW")))
Packit Service fa4841
	{
Packit Service fa4841
		UINT error = NO_ERROR;
Packit Service fa4841
		FILEDESCRIPTOR* file_array = (FILEDESCRIPTOR*)pDstData;
Packit Service fa4841
		UINT32 file_count = DstSize / sizeof(FILEDESCRIPTOR);
Packit Service fa4841
		pDstData = NULL;
Packit Service fa4841
		DstSize = 0;
Packit Service fa4841
		error = cliprdr_serialize_file_list(file_array, file_count, &pDstData, &DstSize);
Packit Service fa4841
Packit Service fa4841
		if (error)
Packit Service fa4841
			WLog_ERR(TAG, "failed to serialize CLIPRDR_FILELIST: 0x%08X", error);
Packit Service fa4841
Packit Service fa4841
		free(file_array);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	xf_cliprdr_send_data_response(clipboard, pDstData, (int)DstSize);
Packit Service fa4841
	free(pDstData);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_get_requested_data(xfClipboard* clipboard, Atom target)
Packit Service fa4841
{
Packit Service fa4841
	Atom type;
Packit Service fa4841
	BYTE* data = NULL;
Packit Service fa4841
	BOOL has_data = FALSE;
Packit Service fa4841
	int format_property;
Packit Service fa4841
	unsigned long dummy;
Packit Service fa4841
	unsigned long length;
Packit Service fa4841
	unsigned long bytes_left;
Packit Service fa4841
	xfCliprdrFormat* format;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	format = xf_cliprdr_get_client_format_by_id(clipboard, clipboard->requestedFormatId);
Packit Service fa4841
Packit Service fa4841
	if (!format || (format->atom != target))
Packit Service fa4841
	{
Packit Service fa4841
		xf_cliprdr_send_data_response(clipboard, NULL, 0);
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	XGetWindowProperty(xfc->display, xfc->drawable, clipboard->property_atom, 0, 0, 0, target,
Packit Service fa4841
	                   &type, &format_property, &length, &bytes_left, &data);
Packit Service fa4841
Packit Service fa4841
	if (data)
Packit Service fa4841
	{
Packit Service fa4841
		XFree(data);
Packit Service fa4841
		data = NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (bytes_left <= 0 && !clipboard->incr_starts)
Packit Service fa4841
	{
Packit Service fa4841
	}
Packit Service fa4841
	else if (type == clipboard->incr_atom)
Packit Service fa4841
	{
Packit Service fa4841
		clipboard->incr_starts = TRUE;
Packit Service fa4841
Packit Service fa4841
		if (clipboard->incr_data)
Packit Service fa4841
		{
Packit Service fa4841
			free(clipboard->incr_data);
Packit Service fa4841
			clipboard->incr_data = NULL;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		clipboard->incr_data_length = 0;
Packit Service fa4841
		has_data = TRUE; /* data will be followed in PropertyNotify event */
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		if (bytes_left <= 0)
Packit Service fa4841
		{
Packit Service fa4841
			/* INCR finish */
Packit Service fa4841
			data = clipboard->incr_data;
Packit Service fa4841
			clipboard->incr_data = NULL;
Packit Service fa4841
			bytes_left = clipboard->incr_data_length;
Packit Service fa4841
			clipboard->incr_data_length = 0;
Packit Service fa4841
			clipboard->incr_starts = 0;
Packit Service fa4841
			has_data = TRUE;
Packit Service fa4841
		}
Packit Service fa4841
		else if (XGetWindowProperty(xfc->display, xfc->drawable, clipboard->property_atom, 0,
Packit Service fa4841
		                            bytes_left, 0, target, &type, &format_property, &length, &dummy,
Packit Service fa4841
		                            &data) == Success)
Packit Service fa4841
		{
Packit Service fa4841
			if (clipboard->incr_starts)
Packit Service fa4841
			{
Packit Service fa4841
				BYTE* new_data;
Packit Service fa4841
				bytes_left = length * format_property / 8;
Packit Service fa4841
				new_data =
Packit Service fa4841
				    (BYTE*)realloc(clipboard->incr_data, clipboard->incr_data_length + bytes_left);
Packit Service fa4841
Packit Service fa4841
				if (!new_data)
Packit Service fa4841
					return FALSE;
Packit Service fa4841
Packit Service fa4841
				clipboard->incr_data = new_data;
Packit Service fa4841
				CopyMemory(clipboard->incr_data + clipboard->incr_data_length, data, bytes_left);
Packit Service fa4841
				clipboard->incr_data_length += bytes_left;
Packit Service fa4841
				XFree(data);
Packit Service fa4841
				data = NULL;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			has_data = TRUE;
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	XDeleteProperty(xfc->display, xfc->drawable, clipboard->property_atom);
Packit Service fa4841
	xf_cliprdr_process_requested_data(clipboard, has_data, data, (int)bytes_left);
Packit Service fa4841
Packit Service fa4841
	if (data)
Packit Service fa4841
		XFree(data);
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_append_target(xfClipboard* clipboard, Atom target)
Packit Service fa4841
{
Packit Service fa4841
	int i;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->numTargets < 0)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if ((size_t)clipboard->numTargets >= ARRAYSIZE(clipboard->targets))
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < clipboard->numTargets; i++)
Packit Service fa4841
	{
Packit Service fa4841
		if (clipboard->targets[i] == target)
Packit Service fa4841
			return;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	clipboard->targets[clipboard->numTargets++] = target;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_provide_targets(xfClipboard* clipboard, const XSelectionEvent* respond)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
Packit Service fa4841
	if (respond->property != None)
Packit Service fa4841
	{
Packit Service fa4841
		XChangeProperty(xfc->display, respond->requestor, respond->property, XA_ATOM, 32,
Packit Service fa4841
		                PropModeReplace, (BYTE*)clipboard->targets, clipboard->numTargets);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_provide_data(xfClipboard* clipboard, const XSelectionEvent* respond,
Packit Service fa4841
                                    const BYTE* data, UINT32 size)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
Packit Service fa4841
	if (respond->property != None)
Packit Service fa4841
	{
Packit Service fa4841
		XChangeProperty(xfc->display, respond->requestor, respond->property, respond->target, 8,
Packit Service fa4841
		                PropModeReplace, data, size);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_process_selection_notify(xfClipboard* clipboard,
Packit Service fa4841
                                                const XSelectionEvent* xevent)
Packit Service fa4841
{
Packit Service fa4841
	if (xevent->target == clipboard->targets[1])
Packit Service fa4841
	{
Packit Service fa4841
		if (xevent->property == None)
Packit Service fa4841
		{
Packit Service fa4841
			xf_cliprdr_send_client_format_list(clipboard);
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			xf_cliprdr_get_requested_targets(clipboard);
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		return TRUE;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		return xf_cliprdr_get_requested_data(clipboard, xevent->target);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_cliprdr_clear_cached_data(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	if (clipboard->data)
Packit Service fa4841
	{
Packit Service fa4841
		free(clipboard->data);
Packit Service fa4841
		clipboard->data = NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	clipboard->data_length = 0;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->data_raw)
Packit Service fa4841
	{
Packit Service fa4841
		free(clipboard->data_raw);
Packit Service fa4841
		clipboard->data_raw = NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	clipboard->data_raw_length = 0;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_process_selection_request(xfClipboard* clipboard,
Packit Service fa4841
                                                 const XSelectionRequestEvent* xevent)
Packit Service fa4841
{
Packit Service fa4841
	int fmt;
Packit Service fa4841
	Atom type;
Packit Service fa4841
	UINT32 formatId;
Packit Service fa4841
	const char* formatName;
Packit Service fa4841
	XSelectionEvent* respond;
Packit Service fa4841
	BYTE* data = NULL;
Packit Service fa4841
	BOOL delayRespond;
Packit Service fa4841
	BOOL rawTransfer;
Packit Service fa4841
	BOOL matchingFormat;
Packit Service fa4841
	unsigned long length;
Packit Service fa4841
	unsigned long bytes_left;
Packit Service fa4841
	CLIPRDR_FORMAT* format;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
Packit Service fa4841
	if (xevent->owner != xfc->drawable)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	delayRespond = FALSE;
Packit Service fa4841
Packit Service fa4841
	if (!(respond = (XSelectionEvent*)calloc(1, sizeof(XSelectionEvent))))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "failed to allocate XEvent data");
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	respond->property = None;
Packit Service fa4841
	respond->type = SelectionNotify;
Packit Service fa4841
	respond->display = xevent->display;
Packit Service fa4841
	respond->requestor = xevent->requestor;
Packit Service fa4841
	respond->selection = xevent->selection;
Packit Service fa4841
	respond->target = xevent->target;
Packit Service fa4841
	respond->time = xevent->time;
Packit Service fa4841
Packit Service fa4841
	if (xevent->target == clipboard->targets[0]) /* TIMESTAMP */
Packit Service fa4841
	{
Packit Service fa4841
		/* TODO */
Packit Service fa4841
	}
Packit Service fa4841
	else if (xevent->target == clipboard->targets[1]) /* TARGETS */
Packit Service fa4841
	{
Packit Service fa4841
		/* Someone else requests our available formats */
Packit Service fa4841
		respond->property = xevent->property;
Packit Service fa4841
		xf_cliprdr_provide_targets(clipboard, respond);
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		format = xf_cliprdr_get_server_format_by_atom(clipboard, xevent->target);
Packit Service fa4841
Packit Service fa4841
		if (format && (xevent->requestor != xfc->drawable))
Packit Service fa4841
		{
Packit Service fa4841
			formatId = format->formatId;
Packit Service fa4841
			formatName = format->formatName;
Packit Service fa4841
			rawTransfer = FALSE;
Packit Service fa4841
Packit Service fa4841
			if (formatId == CF_RAW)
Packit Service fa4841
			{
Packit Service fa4841
				if (XGetWindowProperty(xfc->display, xevent->requestor, clipboard->property_atom, 0,
Packit Service fa4841
				                       4, 0, XA_INTEGER, &type, &fmt, &length, &bytes_left,
Packit Service fa4841
				                       &data) != Success)
Packit Service fa4841
				{
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				if (data)
Packit Service fa4841
				{
Packit Service fa4841
					rawTransfer = TRUE;
Packit Service fa4841
					CopyMemory(&formatId, data, 4);
Packit Service fa4841
					XFree(data);
Packit Service fa4841
				}
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			/* We can compare format names by pointer value here as they are both
Packit Service fa4841
			 * taken from the same clipboard->serverFormats array */
Packit Service fa4841
			matchingFormat = (formatId == clipboard->data_format_id) &&
Packit Service fa4841
			                 (formatName == clipboard->data_format_name);
Packit Service fa4841
Packit Service fa4841
			if (matchingFormat && (clipboard->data != 0) && !rawTransfer)
Packit Service fa4841
			{
Packit Service fa4841
				/* Cached converted clipboard data available. Send it now */
Packit Service fa4841
				respond->property = xevent->property;
Packit Service fa4841
				xf_cliprdr_provide_data(clipboard, respond, clipboard->data,
Packit Service fa4841
				                        clipboard->data_length);
Packit Service fa4841
			}
Packit Service fa4841
			else if (matchingFormat && (clipboard->data_raw != 0) && rawTransfer)
Packit Service fa4841
			{
Packit Service fa4841
				/* Cached raw clipboard data available. Send it now */
Packit Service fa4841
				respond->property = xevent->property;
Packit Service fa4841
				xf_cliprdr_provide_data(clipboard, respond, clipboard->data_raw,
Packit Service fa4841
				                        clipboard->data_raw_length);
Packit Service fa4841
			}
Packit Service fa4841
			else if (clipboard->respond)
Packit Service fa4841
			{
Packit Service fa4841
				/* duplicate request */
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				/**
Packit Service fa4841
				 * Send clipboard data request to the server.
Packit Service fa4841
				 * Response will be postponed after receiving the data
Packit Service fa4841
				 */
Packit Service fa4841
				xf_cliprdr_clear_cached_data(clipboard);
Packit Service fa4841
				respond->property = xevent->property;
Packit Service fa4841
				clipboard->respond = respond;
Packit Service fa4841
				clipboard->data_format_id = formatId;
Packit Service fa4841
				clipboard->data_format_name = formatName;
Packit Service fa4841
				clipboard->data_raw_format = rawTransfer;
Packit Service fa4841
				delayRespond = TRUE;
Packit Service fa4841
				xf_cliprdr_send_data_request(clipboard, formatId);
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!delayRespond)
Packit Service fa4841
	{
Packit Service fa4841
		union {
Packit Service fa4841
			XEvent* ev;
Packit Service fa4841
			XSelectionEvent* sev;
Packit Service fa4841
		} conv;
Packit Service fa4841
Packit Service fa4841
		conv.sev = respond;
Packit Service fa4841
		XSendEvent(xfc->display, xevent->requestor, 0, 0, conv.ev);
Packit Service fa4841
		XFlush(xfc->display);
Packit Service fa4841
		free(respond);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_process_selection_clear(xfClipboard* clipboard,
Packit Service fa4841
                                               const XSelectionClearEvent* xevent)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
Packit Service fa4841
	WINPR_UNUSED(xevent);
Packit Service fa4841
Packit Service fa4841
	if (xf_cliprdr_is_self_owned(clipboard))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	XDeleteProperty(xfc->display, clipboard->root_window, clipboard->property_atom);
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_cliprdr_process_property_notify(xfClipboard* clipboard, const XPropertyEvent* xevent)
Packit Service fa4841
{
Packit Service fa4841
	xfCliprdrFormat* format;
Packit Service fa4841
	xfContext* xfc = NULL;
Packit Service fa4841
Packit Service fa4841
	if (!clipboard)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	xfc = clipboard->xfc;
Packit Service fa4841
Packit Service fa4841
	if (xevent->atom != clipboard->property_atom)
Packit Service fa4841
		return FALSE; /* Not cliprdr-related */
Packit Service fa4841
Packit Service fa4841
	if (xevent->window == clipboard->root_window)
Packit Service fa4841
	{
Packit Service fa4841
		xf_cliprdr_send_client_format_list(clipboard);
Packit Service fa4841
	}
Packit Service fa4841
	else if ((xevent->window == xfc->drawable) && (xevent->state == PropertyNewValue) &&
Packit Service fa4841
	         clipboard->incr_starts)
Packit Service fa4841
	{
Packit Service fa4841
		format = xf_cliprdr_get_client_format_by_id(clipboard, clipboard->requestedFormatId);
Packit Service fa4841
Packit Service fa4841
		if (format)
Packit Service fa4841
			xf_cliprdr_get_requested_data(clipboard, format->atom);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_cliprdr_handle_xevent(xfContext* xfc, const XEvent* event)
Packit Service fa4841
{
Packit Service fa4841
	xfClipboard* clipboard;
Packit Service fa4841
Packit Service fa4841
	if (!xfc || !event)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	clipboard = xfc->clipboard;
Packit Service fa4841
Packit Service fa4841
	if (!clipboard)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_XFIXES
Packit Service fa4841
Packit Service fa4841
	if (clipboard->xfixes_supported &&
Packit Service fa4841
	    event->type == XFixesSelectionNotify + clipboard->xfixes_event_base)
Packit Service fa4841
	{
Packit Service fa4841
		XFixesSelectionNotifyEvent* se = (XFixesSelectionNotifyEvent*)event;
Packit Service fa4841
Packit Service fa4841
		if (se->subtype == XFixesSetSelectionOwnerNotify)
Packit Service fa4841
		{
Packit Service fa4841
			if (se->selection != clipboard->clipboard_atom)
Packit Service fa4841
				return;
Packit Service fa4841
Packit Service fa4841
			if (XGetSelectionOwner(xfc->display, se->selection) == xfc->drawable)
Packit Service fa4841
				return;
Packit Service fa4841
Packit Service fa4841
			clipboard->owner = None;
Packit Service fa4841
			xf_cliprdr_check_owner(clipboard);
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		return;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
	switch (event->type)
Packit Service fa4841
	{
Packit Service fa4841
		case SelectionNotify:
Packit Service fa4841
			xf_cliprdr_process_selection_notify(clipboard, &event->xselection);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case SelectionRequest:
Packit Service fa4841
			xf_cliprdr_process_selection_request(clipboard, &event->xselectionrequest);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case SelectionClear:
Packit Service fa4841
			xf_cliprdr_process_selection_clear(clipboard, &event->xselectionclear);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case PropertyNotify:
Packit Service fa4841
			xf_cliprdr_process_property_notify(clipboard, &event->xproperty);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case FocusIn:
Packit Service fa4841
			if (!clipboard->xfixes_supported)
Packit Service fa4841
			{
Packit Service fa4841
				xf_cliprdr_check_owner(clipboard);
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_send_client_capabilities(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_CAPABILITIES capabilities;
Packit Service fa4841
	CLIPRDR_GENERAL_CAPABILITY_SET generalCapabilitySet;
Packit Service fa4841
	capabilities.cCapabilitiesSets = 1;
Packit Service fa4841
	capabilities.capabilitySets = (CLIPRDR_CAPABILITY_SET*)&(generalCapabilitySet);
Packit Service fa4841
	generalCapabilitySet.capabilitySetType = CB_CAPSTYPE_GENERAL;
Packit Service fa4841
	generalCapabilitySet.capabilitySetLength = 12;
Packit Service fa4841
	generalCapabilitySet.version = CB_CAPS_VERSION_2;
Packit Service fa4841
	generalCapabilitySet.generalFlags = CB_USE_LONG_FORMAT_NAMES;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->streams_supported && clipboard->file_formats_registered)
Packit Service fa4841
		generalCapabilitySet.generalFlags |= CB_STREAM_FILECLIP_ENABLED | CB_FILECLIP_NO_FILE_PATHS;
Packit Service fa4841
Packit Service fa4841
	return clipboard->context->ClientCapabilities(clipboard->context, &capabilities);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_send_client_format_list(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i, numFormats;
Packit Service fa4841
	CLIPRDR_FORMAT* formats = NULL;
Packit Service fa4841
	CLIPRDR_FORMAT_LIST formatList = { 0 };
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	UINT ret;
Packit Service fa4841
	numFormats = clipboard->numClientFormats;
Packit Service fa4841
Packit Service fa4841
	if (numFormats)
Packit Service fa4841
	{
Packit Service fa4841
		if (!(formats = (CLIPRDR_FORMAT*)calloc(numFormats, sizeof(CLIPRDR_FORMAT))))
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "failed to allocate %" PRIu32 " CLIPRDR_FORMAT structs", numFormats);
Packit Service fa4841
			return CHANNEL_RC_NO_MEMORY;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < numFormats; i++)
Packit Service fa4841
	{
Packit Service fa4841
		formats[i].formatId = clipboard->clientFormats[i].formatId;
Packit Service fa4841
		formats[i].formatName = clipboard->clientFormats[i].formatName;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	formatList.msgFlags = CB_RESPONSE_OK;
Packit Service fa4841
	formatList.numFormats = numFormats;
Packit Service fa4841
	formatList.formats = formats;
Packit Service fa4841
	formatList.msgType = CB_FORMAT_LIST;
Packit Service fa4841
	ret = clipboard->context->ClientFormatList(clipboard->context, &formatList);
Packit Service fa4841
	free(formats);
Packit Service fa4841
Packit Service fa4841
	if (clipboard->owner && clipboard->owner != xfc->drawable)
Packit Service fa4841
	{
Packit Service fa4841
		/* Request the owner for TARGETS, and wait for SelectionNotify event */
Packit Service fa4841
		XConvertSelection(xfc->display, clipboard->clipboard_atom, clipboard->targets[1],
Packit Service fa4841
		                  clipboard->property_atom, xfc->drawable, CurrentTime);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return ret;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_send_client_format_list_response(xfClipboard* clipboard, BOOL status)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FORMAT_LIST_RESPONSE formatListResponse;
Packit Service fa4841
	formatListResponse.msgType = CB_FORMAT_LIST_RESPONSE;
Packit Service fa4841
	formatListResponse.msgFlags = status ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
Packit Service fa4841
	formatListResponse.dataLen = 0;
Packit Service fa4841
	return clipboard->context->ClientFormatListResponse(clipboard->context, &formatListResponse);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_monitor_ready(CliprdrClientContext* context,
Packit Service fa4841
                                     const CLIPRDR_MONITOR_READY* monitorReady)
Packit Service fa4841
{
Packit Service fa4841
	xfClipboard* clipboard = (xfClipboard*)context->custom;
Packit Service fa4841
	UINT ret;
Packit Service fa4841
Packit Service fa4841
	WINPR_UNUSED(monitorReady);
Packit Service fa4841
Packit Service fa4841
	if ((ret = xf_cliprdr_send_client_capabilities(clipboard)) != CHANNEL_RC_OK)
Packit Service fa4841
		return ret;
Packit Service fa4841
Packit Service fa4841
	if ((ret = xf_cliprdr_send_client_format_list(clipboard)) != CHANNEL_RC_OK)
Packit Service fa4841
		return ret;
Packit Service fa4841
Packit Service fa4841
	clipboard->sync = TRUE;
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_server_capabilities(CliprdrClientContext* context,
Packit Service fa4841
                                           const CLIPRDR_CAPABILITIES* capabilities)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i;
Packit Service fa4841
	const CLIPRDR_CAPABILITY_SET* caps;
Packit Service fa4841
	const CLIPRDR_GENERAL_CAPABILITY_SET* generalCaps;
Packit Service fa4841
	const BYTE* capsPtr = (const BYTE*)capabilities->capabilitySets;
Packit Service fa4841
	xfClipboard* clipboard = (xfClipboard*)context->custom;
Packit Service fa4841
	clipboard->streams_supported = FALSE;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < capabilities->cCapabilitiesSets; i++)
Packit Service fa4841
	{
Packit Service fa4841
		caps = (const CLIPRDR_CAPABILITY_SET*)capsPtr;
Packit Service fa4841
Packit Service fa4841
		if (caps->capabilitySetType == CB_CAPSTYPE_GENERAL)
Packit Service fa4841
		{
Packit Service fa4841
			generalCaps = (const CLIPRDR_GENERAL_CAPABILITY_SET*)caps;
Packit Service fa4841
Packit Service fa4841
			if (generalCaps->generalFlags & CB_STREAM_FILECLIP_ENABLED)
Packit Service fa4841
			{
Packit Service fa4841
				clipboard->streams_supported = TRUE;
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		capsPtr += caps->capabilitySetLength;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT xf_cliprdr_server_format_list(CliprdrClientContext* context,
Packit Service fa4841
                                          const CLIPRDR_FORMAT_LIST* formatList)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i;
Packit Service fa4841
	int j;
Packit Service fa4841
	xfClipboard* clipboard = (xfClipboard*)context->custom;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	UINT ret;
Packit Service fa4841
	xf_cliprdr_clear_cached_data(clipboard);
Packit Service fa4841
	clipboard->data_format_id = -1;
Packit Service fa4841
	clipboard->data_format_name = NULL;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->serverFormats)
Packit Service fa4841
	{
Packit Service fa4841
		for (j = 0; j < clipboard->numServerFormats; j++)
Packit Service fa4841
			free(clipboard->serverFormats[j].formatName);
Packit Service fa4841
Packit Service fa4841
		free(clipboard->serverFormats);
Packit Service fa4841
		clipboard->serverFormats = NULL;
Packit Service fa4841
		clipboard->numServerFormats = 0;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	clipboard->numServerFormats = formatList->numFormats + 1; /* +1 for CF_RAW */
Packit Service fa4841
Packit Service fa4841
	if (!(clipboard->serverFormats =
Packit Service fa4841
	          (CLIPRDR_FORMAT*)calloc(clipboard->numServerFormats, sizeof(CLIPRDR_FORMAT))))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "failed to allocate %d CLIPRDR_FORMAT structs", clipboard->numServerFormats);
Packit Service fa4841
		return CHANNEL_RC_NO_MEMORY;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < formatList->numFormats; i++)
Packit Service fa4841
	{
Packit Service fa4841
		CLIPRDR_FORMAT* format = &formatList->formats[i];
Packit Service fa4841
		clipboard->serverFormats[i].formatId = format->formatId;
Packit Service fa4841
Packit Service fa4841
		if (format->formatName)
Packit Service fa4841
		{
Packit Service fa4841
			clipboard->serverFormats[i].formatName = _strdup(format->formatName);
Packit Service fa4841
Packit Service fa4841
			if (!clipboard->serverFormats[i].formatName)
Packit Service fa4841
			{
Packit Service fa4841
				UINT32 k;
Packit Service fa4841
Packit Service fa4841
				for (k = 0; k < i; k++)
Packit Service fa4841
					free(clipboard->serverFormats[k].formatName);
Packit Service fa4841
Packit Service fa4841
				clipboard->numServerFormats = 0;
Packit Service fa4841
				free(clipboard->serverFormats);
Packit Service fa4841
				clipboard->serverFormats = NULL;
Packit Service fa4841
				return CHANNEL_RC_NO_MEMORY;
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* CF_RAW is always implicitly supported by the server */
Packit Service fa4841
	{
Packit Service fa4841
		CLIPRDR_FORMAT* format = &clipboard->serverFormats[formatList->numFormats];
Packit Service fa4841
		format->formatId = CF_RAW;
Packit Service fa4841
		format->formatName = NULL;
Packit Service fa4841
	}
Packit Service fa4841
	xf_cliprdr_provide_server_format_list(clipboard);
Packit Service fa4841
	clipboard->numTargets = 2;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < formatList->numFormats; i++)
Packit Service fa4841
	{
Packit Service fa4841
		CLIPRDR_FORMAT* format = &formatList->formats[i];
Packit Service fa4841
Packit Service fa4841
		for (j = 0; j < clipboard->numClientFormats; j++)
Packit Service fa4841
		{
Packit Service fa4841
			if (xf_cliprdr_formats_equal(format, &clipboard->clientFormats[j]))
Packit Service fa4841
			{
Packit Service fa4841
				xf_cliprdr_append_target(clipboard, clipboard->clientFormats[j].atom);
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	ret = xf_cliprdr_send_client_format_list_response(clipboard, TRUE);
Packit Service fa4841
	XSetSelectionOwner(xfc->display, clipboard->clipboard_atom, xfc->drawable, CurrentTime);
Packit Service fa4841
	XFlush(xfc->display);
Packit Service fa4841
	return ret;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT
Packit Service fa4841
xf_cliprdr_server_format_list_response(CliprdrClientContext* context,
Packit Service fa4841
                                       const CLIPRDR_FORMAT_LIST_RESPONSE* formatListResponse)
Packit Service fa4841
{
Packit Service fa4841
	// xfClipboard* clipboard = (xfClipboard*) context->custom;
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT
Packit Service fa4841
xf_cliprdr_server_format_data_request(CliprdrClientContext* context,
Packit Service fa4841
                                      const CLIPRDR_FORMAT_DATA_REQUEST* formatDataRequest)
Packit Service fa4841
{
Packit Service fa4841
	BOOL rawTransfer;
Packit Service fa4841
	xfCliprdrFormat* format = NULL;
Packit Service fa4841
	UINT32 formatId = formatDataRequest->requestedFormatId;
Packit Service fa4841
	xfClipboard* clipboard = (xfClipboard*)context->custom;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
	rawTransfer = xf_cliprdr_is_raw_transfer_available(clipboard);
Packit Service fa4841
Packit Service fa4841
	if (rawTransfer)
Packit Service fa4841
	{
Packit Service fa4841
		format = xf_cliprdr_get_client_format_by_id(clipboard, CF_RAW);
Packit Service fa4841
		XChangeProperty(xfc->display, xfc->drawable, clipboard->property_atom, XA_INTEGER, 32,
Packit Service fa4841
		                PropModeReplace, (BYTE*)&formatId, 1);
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
		format = xf_cliprdr_get_client_format_by_id(clipboard, formatId);
Packit Service fa4841
Packit Service fa4841
	if (!format)
Packit Service fa4841
		return xf_cliprdr_send_data_response(clipboard, NULL, 0);
Packit Service fa4841
Packit Service fa4841
	clipboard->requestedFormatId = rawTransfer ? CF_RAW : formatId;
Packit Service fa4841
	XConvertSelection(xfc->display, clipboard->clipboard_atom, format->atom,
Packit Service fa4841
	                  clipboard->property_atom, xfc->drawable, CurrentTime);
Packit Service fa4841
	XFlush(xfc->display);
Packit Service fa4841
	/* After this point, we expect a SelectionNotify event from the clipboard owner. */
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT
Packit Service fa4841
xf_cliprdr_server_format_data_response(CliprdrClientContext* context,
Packit Service fa4841
                                       const CLIPRDR_FORMAT_DATA_RESPONSE* formatDataResponse)
Packit Service fa4841
{
Packit Service fa4841
	BOOL bSuccess;
Packit Service fa4841
	BYTE* pDstData;
Packit Service fa4841
	UINT32 DstSize;
Packit Service fa4841
	UINT32 SrcSize;
Packit Service fa4841
	UINT32 srcFormatId;
Packit Service fa4841
	UINT32 dstFormatId;
Packit Service fa4841
	BOOL nullTerminated = FALSE;
Packit Service fa4841
	UINT32 size = formatDataResponse->dataLen;
Packit Service fa4841
	const BYTE* data = formatDataResponse->requestedFormatData;
Packit Service fa4841
	xfClipboard* clipboard = (xfClipboard*)context->custom;
Packit Service fa4841
	xfContext* xfc = clipboard->xfc;
Packit Service fa4841
Packit Service fa4841
	if (!clipboard->respond)
Packit Service fa4841
		return CHANNEL_RC_OK;
Packit Service fa4841
Packit Service fa4841
	xf_cliprdr_clear_cached_data(clipboard);
Packit Service fa4841
	pDstData = NULL;
Packit Service fa4841
	DstSize = 0;
Packit Service fa4841
	srcFormatId = 0;
Packit Service fa4841
	dstFormatId = 0;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->data_raw_format)
Packit Service fa4841
	{
Packit Service fa4841
		srcFormatId = CF_RAW;
Packit Service fa4841
		dstFormatId = CF_RAW;
Packit Service fa4841
	}
Packit Service fa4841
	else if (clipboard->data_format_name)
Packit Service fa4841
	{
Packit Service fa4841
		if (strcmp(clipboard->data_format_name, "HTML Format") == 0)
Packit Service fa4841
		{
Packit Service fa4841
			srcFormatId = ClipboardGetFormatId(clipboard->system, "HTML Format");
Packit Service fa4841
			dstFormatId = ClipboardGetFormatId(clipboard->system, "text/html");
Packit Service fa4841
			nullTerminated = TRUE;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (strcmp(clipboard->data_format_name, "FileGroupDescriptorW") == 0)
Packit Service fa4841
		{
Packit Service fa4841
			srcFormatId = ClipboardGetFormatId(clipboard->system, "FileGroupDescriptorW");
Packit Service fa4841
			dstFormatId = ClipboardGetFormatId(clipboard->system, "text/uri-list");
Packit Service fa4841
			nullTerminated = FALSE;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		switch (clipboard->data_format_id)
Packit Service fa4841
		{
Packit Service fa4841
			case CF_TEXT:
Packit Service fa4841
				srcFormatId = CF_TEXT;
Packit Service fa4841
				dstFormatId = ClipboardGetFormatId(clipboard->system, "UTF8_STRING");
Packit Service fa4841
				nullTerminated = TRUE;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case CF_OEMTEXT:
Packit Service fa4841
				srcFormatId = CF_OEMTEXT;
Packit Service fa4841
				dstFormatId = ClipboardGetFormatId(clipboard->system, "UTF8_STRING");
Packit Service fa4841
				nullTerminated = TRUE;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case CF_UNICODETEXT:
Packit Service fa4841
				srcFormatId = CF_UNICODETEXT;
Packit Service fa4841
				dstFormatId = ClipboardGetFormatId(clipboard->system, "UTF8_STRING");
Packit Service fa4841
				nullTerminated = TRUE;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case CF_DIB:
Packit Service fa4841
				srcFormatId = CF_DIB;
Packit Service fa4841
				dstFormatId = ClipboardGetFormatId(clipboard->system, "image/bmp");
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			default:
Packit Service fa4841
				break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	SrcSize = (UINT32)size;
Packit Service fa4841
	bSuccess = ClipboardSetData(clipboard->system, srcFormatId, data, SrcSize);
Packit Service fa4841
Packit Service fa4841
	if (bSuccess)
Packit Service fa4841
	{
Packit Service fa4841
		if (SrcSize == 0)
Packit Service fa4841
		{
Packit Service fa4841
			WLog_INFO(TAG, "skipping, empty data detected!!!");
Packit Service fa4841
			free(clipboard->respond);
Packit Service fa4841
			clipboard->respond = NULL;
Packit Service fa4841
			return CHANNEL_RC_OK;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		pDstData = (BYTE*)ClipboardGetData(clipboard->system, dstFormatId, &DstSize);
Packit Service fa4841
Packit Service fa4841
		if (!pDstData)
Packit Service fa4841
		{
Packit Service fa4841
			WLog_WARN(TAG, "failed to get clipboard data in format %s [source format %s]",
Packit Service fa4841
			          ClipboardGetFormatName(clipboard->system, dstFormatId),
Packit Service fa4841
			          ClipboardGetFormatName(clipboard->system, srcFormatId));
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (nullTerminated && pDstData)
Packit Service fa4841
		{
Packit Service fa4841
			BYTE* nullTerminator = memchr(pDstData, '\0', DstSize);
Packit Service fa4841
			if (nullTerminator)
Packit Service fa4841
				DstSize = nullTerminator - pDstData;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* Cache converted and original data to avoid doing a possibly costly
Packit Service fa4841
	 * conversion again on subsequent requests */
Packit Service fa4841
	clipboard->data = pDstData;
Packit Service fa4841
	clipboard->data_length = DstSize;
Packit Service fa4841
	/* We have to copy the original data again, as pSrcData is now owned
Packit Service fa4841
	 * by clipboard->system. Memory allocation failure is not fatal here
Packit Service fa4841
	 * as this is only a cached value. */
Packit Service fa4841
	clipboard->data_raw = (BYTE*)malloc(size);
Packit Service fa4841
Packit Service fa4841
	if (clipboard->data_raw)
Packit Service fa4841
	{
Packit Service fa4841
		CopyMemory(clipboard->data_raw, data, size);
Packit Service fa4841
		clipboard->data_raw_length = size;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		WLog_WARN(TAG, "failed to allocate %" PRIu32 " bytes for a copy of raw clipboard data",
Packit Service fa4841
		          size);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	xf_cliprdr_provide_data(clipboard, clipboard->respond, pDstData, DstSize);
Packit Service fa4841
	{
Packit Service fa4841
		union {
Packit Service fa4841
			XEvent* ev;
Packit Service fa4841
			XSelectionEvent* sev;
Packit Service fa4841
		} conv;
Packit Service fa4841
Packit Service fa4841
		conv.sev = clipboard->respond;
Packit Service fa4841
Packit Service fa4841
		XSendEvent(xfc->display, clipboard->respond->requestor, 0, 0, conv.ev);
Packit Service fa4841
		XFlush(xfc->display);
Packit Service fa4841
	}
Packit Service fa4841
	free(clipboard->respond);
Packit Service fa4841
	clipboard->respond = NULL;
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT
Packit Service fa4841
xf_cliprdr_server_file_size_request(xfClipboard* clipboard,
Packit Service fa4841
                                    const CLIPRDR_FILE_CONTENTS_REQUEST* fileContentsRequest)
Packit Service fa4841
{
Packit Service fa4841
	wClipboardFileSizeRequest request = { 0 };
Packit Service fa4841
	request.streamId = fileContentsRequest->streamId;
Packit Service fa4841
	request.listIndex = fileContentsRequest->listIndex;
Packit Service fa4841
Packit Service fa4841
	if (fileContentsRequest->cbRequested != sizeof(UINT64))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_WARN(TAG, "unexpected FILECONTENTS_SIZE request: %" PRIu32 " bytes",
Packit Service fa4841
		          fileContentsRequest->cbRequested);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return clipboard->delegate->ClientRequestFileSize(clipboard->delegate, &request);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT
Packit Service fa4841
xf_cliprdr_server_file_range_request(xfClipboard* clipboard,
Packit Service fa4841
                                     const CLIPRDR_FILE_CONTENTS_REQUEST* fileContentsRequest)
Packit Service fa4841
{
Packit Service fa4841
	wClipboardFileRangeRequest request = { 0 };
Packit Service fa4841
	request.streamId = fileContentsRequest->streamId;
Packit Service fa4841
	request.listIndex = fileContentsRequest->listIndex;
Packit Service fa4841
	request.nPositionLow = fileContentsRequest->nPositionLow;
Packit Service fa4841
	request.nPositionHigh = fileContentsRequest->nPositionHigh;
Packit Service fa4841
	request.cbRequested = fileContentsRequest->cbRequested;
Packit Service fa4841
	return clipboard->delegate->ClientRequestFileRange(clipboard->delegate, &request);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT
Packit Service fa4841
xf_cliprdr_send_file_contents_failure(CliprdrClientContext* context,
Packit Service fa4841
                                      const CLIPRDR_FILE_CONTENTS_REQUEST* fileContentsRequest)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FILE_CONTENTS_RESPONSE response = { 0 };
Packit Service fa4841
	response.msgFlags = CB_RESPONSE_FAIL;
Packit Service fa4841
	response.streamId = fileContentsRequest->streamId;
Packit Service fa4841
	return context->ClientFileContentsResponse(context, &response);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT
Packit Service fa4841
xf_cliprdr_server_file_contents_request(CliprdrClientContext* context,
Packit Service fa4841
                                        const CLIPRDR_FILE_CONTENTS_REQUEST* fileContentsRequest)
Packit Service fa4841
{
Packit Service fa4841
	UINT error = NO_ERROR;
Packit Service fa4841
	xfClipboard* clipboard = context->custom;
Packit Service fa4841
Packit Service fa4841
	/*
Packit Service fa4841
	 * MS-RDPECLIP 2.2.5.3 File Contents Request PDU (CLIPRDR_FILECONTENTS_REQUEST):
Packit Service fa4841
	 * The FILECONTENTS_SIZE and FILECONTENTS_RANGE flags MUST NOT be set at the same time.
Packit Service fa4841
	 */
Packit Service fa4841
	if ((fileContentsRequest->dwFlags & (FILECONTENTS_SIZE | FILECONTENTS_RANGE)) ==
Packit Service fa4841
	    (FILECONTENTS_SIZE | FILECONTENTS_RANGE))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "invalid CLIPRDR_FILECONTENTS_REQUEST.dwFlags");
Packit Service fa4841
		return xf_cliprdr_send_file_contents_failure(context, fileContentsRequest);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (fileContentsRequest->dwFlags & FILECONTENTS_SIZE)
Packit Service fa4841
		error = xf_cliprdr_server_file_size_request(clipboard, fileContentsRequest);
Packit Service fa4841
Packit Service fa4841
	if (fileContentsRequest->dwFlags & FILECONTENTS_RANGE)
Packit Service fa4841
		error = xf_cliprdr_server_file_range_request(clipboard, fileContentsRequest);
Packit Service fa4841
Packit Service fa4841
	if (error)
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "failed to handle CLIPRDR_FILECONTENTS_REQUEST: 0x%08X", error);
Packit Service fa4841
		return xf_cliprdr_send_file_contents_failure(context, fileContentsRequest);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT xf_cliprdr_clipboard_file_size_success(wClipboardDelegate* delegate,
Packit Service fa4841
                                                   const wClipboardFileSizeRequest* request,
Packit Service fa4841
                                                   UINT64 fileSize)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FILE_CONTENTS_RESPONSE response = { 0 };
Packit Service fa4841
	xfClipboard* clipboard = delegate->custom;
Packit Service fa4841
	response.msgFlags = CB_RESPONSE_OK;
Packit Service fa4841
	response.streamId = request->streamId;
Packit Service fa4841
	response.cbRequested = sizeof(UINT64);
Packit Service fa4841
	response.requestedData = (BYTE*)&fileSize;
Packit Service fa4841
	return clipboard->context->ClientFileContentsResponse(clipboard->context, &response);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT xf_cliprdr_clipboard_file_size_failure(wClipboardDelegate* delegate,
Packit Service fa4841
                                                   const wClipboardFileSizeRequest* request,
Packit Service fa4841
                                                   UINT errorCode)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FILE_CONTENTS_RESPONSE response = { 0 };
Packit Service fa4841
	xfClipboard* clipboard = delegate->custom;
Packit Service fa4841
	WINPR_UNUSED(errorCode);
Packit Service fa4841
Packit Service fa4841
	response.msgFlags = CB_RESPONSE_FAIL;
Packit Service fa4841
	response.streamId = request->streamId;
Packit Service fa4841
	return clipboard->context->ClientFileContentsResponse(clipboard->context, &response);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT xf_cliprdr_clipboard_file_range_success(wClipboardDelegate* delegate,
Packit Service fa4841
                                                    const wClipboardFileRangeRequest* request,
Packit Service fa4841
                                                    const BYTE* data, UINT32 size)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FILE_CONTENTS_RESPONSE response = { 0 };
Packit Service fa4841
	xfClipboard* clipboard = delegate->custom;
Packit Service fa4841
	response.msgFlags = CB_RESPONSE_OK;
Packit Service fa4841
	response.streamId = request->streamId;
Packit Service fa4841
	response.cbRequested = size;
Packit Service fa4841
	response.requestedData = (BYTE*)data;
Packit Service fa4841
	return clipboard->context->ClientFileContentsResponse(clipboard->context, &response);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT xf_cliprdr_clipboard_file_range_failure(wClipboardDelegate* delegate,
Packit Service fa4841
                                                    const wClipboardFileRangeRequest* request,
Packit Service fa4841
                                                    UINT errorCode)
Packit Service fa4841
{
Packit Service fa4841
	CLIPRDR_FILE_CONTENTS_RESPONSE response = { 0 };
Packit Service fa4841
	xfClipboard* clipboard = delegate->custom;
Packit Service fa4841
	WINPR_UNUSED(errorCode);
Packit Service fa4841
Packit Service fa4841
	response.msgFlags = CB_RESPONSE_FAIL;
Packit Service fa4841
	response.streamId = request->streamId;
Packit Service fa4841
	return clipboard->context->ClientFileContentsResponse(clipboard->context, &response);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
xfClipboard* xf_clipboard_new(xfContext* xfc)
Packit Service fa4841
{
Packit Service fa4841
	int i, n = 0;
Packit Service fa4841
	rdpChannels* channels;
Packit Service fa4841
	xfClipboard* clipboard;
Packit Service fa4841
Packit Service fa4841
	if (!(clipboard = (xfClipboard*)calloc(1, sizeof(xfClipboard))))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "failed to allocate xfClipboard data");
Packit Service fa4841
		return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	xfc->clipboard = clipboard;
Packit Service fa4841
	clipboard->xfc = xfc;
Packit Service fa4841
	channels = ((rdpContext*)xfc)->channels;
Packit Service fa4841
	clipboard->channels = channels;
Packit Service fa4841
	clipboard->system = ClipboardCreate();
Packit Service fa4841
	clipboard->requestedFormatId = -1;
Packit Service fa4841
	clipboard->root_window = DefaultRootWindow(xfc->display);
Packit Service fa4841
	clipboard->clipboard_atom = XInternAtom(xfc->display, "CLIPBOARD", FALSE);
Packit Service fa4841
Packit Service fa4841
	if (clipboard->clipboard_atom == None)
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "unable to get CLIPBOARD atom");
Packit Service fa4841
		goto error;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	clipboard->property_atom = XInternAtom(xfc->display, "_FREERDP_CLIPRDR", FALSE);
Packit Service fa4841
	clipboard->raw_transfer_atom = XInternAtom(xfc->display, "_FREERDP_CLIPRDR_RAW", FALSE);
Packit Service fa4841
	clipboard->raw_format_list_atom = XInternAtom(xfc->display, "_FREERDP_CLIPRDR_FORMATS", FALSE);
Packit Service fa4841
	xf_cliprdr_set_raw_transfer_enabled(clipboard, TRUE);
Packit Service fa4841
	XSelectInput(xfc->display, clipboard->root_window, PropertyChangeMask);
Packit Service fa4841
#ifdef WITH_XFIXES
Packit Service fa4841
Packit Service fa4841
	if (XFixesQueryExtension(xfc->display, &clipboard->xfixes_event_base,
Packit Service fa4841
	                         &clipboard->xfixes_error_base))
Packit Service fa4841
	{
Packit Service fa4841
		int xfmajor, xfminor;
Packit Service fa4841
Packit Service fa4841
		if (XFixesQueryVersion(xfc->display, &xfmajor, &xfminor))
Packit Service fa4841
		{
Packit Service fa4841
			XFixesSelectSelectionInput(xfc->display, clipboard->root_window,
Packit Service fa4841
			                           clipboard->clipboard_atom,
Packit Service fa4841
			                           XFixesSetSelectionOwnerNotifyMask);
Packit Service fa4841
			clipboard->xfixes_supported = TRUE;
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "Error querying X Fixes extension version");
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "Error loading X Fixes extension");
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
#else
Packit Service fa4841
	WLog_ERR(
Packit Service fa4841
	    TAG,
Packit Service fa4841
	    "Warning: Using clipboard redirection without XFIXES extension is strongly discouraged!");
Packit Service fa4841
#endif
Packit Service fa4841
	clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "_FREERDP_RAW", False);
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CF_RAW;
Packit Service fa4841
	n++;
Packit Service fa4841
	clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "UTF8_STRING", False);
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CF_UNICODETEXT;
Packit Service fa4841
	n++;
Packit Service fa4841
	clipboard->clientFormats[n].atom = XA_STRING;
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CF_TEXT;
Packit Service fa4841
	n++;
Packit Service fa4841
	clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "image/png", False);
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CB_FORMAT_PNG;
Packit Service fa4841
	n++;
Packit Service fa4841
	clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "image/jpeg", False);
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CB_FORMAT_JPEG;
Packit Service fa4841
	n++;
Packit Service fa4841
	clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "image/gif", False);
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CB_FORMAT_GIF;
Packit Service fa4841
	n++;
Packit Service fa4841
	clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "image/bmp", False);
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CF_DIB;
Packit Service fa4841
	n++;
Packit Service fa4841
	clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "text/html", False);
Packit Service fa4841
	clipboard->clientFormats[n].formatId = CB_FORMAT_HTML;
Packit Service fa4841
	clipboard->clientFormats[n].formatName = _strdup("HTML Format");
Packit Service fa4841
Packit Service fa4841
	if (!clipboard->clientFormats[n].formatName)
Packit Service fa4841
		goto error;
Packit Service fa4841
Packit Service fa4841
	n++;
Packit Service fa4841
Packit Service fa4841
	/*
Packit Service fa4841
	 * Existence of registered format IDs for file formats does not guarantee that they are
Packit Service fa4841
	 * in fact supported by wClipboard (as further initialization may have failed after format
Packit Service fa4841
	 * registration). However, they are definitely not supported if there are no registered
Packit Service fa4841
	 * formats. In this case we should not list file formats in TARGETS.
Packit Service fa4841
	 */
Packit Service fa4841
	if (ClipboardGetFormatId(clipboard->system, "text/uri-list"))
Packit Service fa4841
	{
Packit Service fa4841
		clipboard->file_formats_registered = TRUE;
Packit Service fa4841
		clipboard->clientFormats[n].atom = XInternAtom(xfc->display, "text/uri-list", False);
Packit Service fa4841
		clipboard->clientFormats[n].formatId = CB_FORMAT_TEXTURILIST;
Packit Service fa4841
		clipboard->clientFormats[n].formatName = _strdup("FileGroupDescriptorW");
Packit Service fa4841
Packit Service fa4841
		if (!clipboard->clientFormats[n].formatName)
Packit Service fa4841
			goto error;
Packit Service fa4841
Packit Service fa4841
		n++;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	clipboard->numClientFormats = n;
Packit Service fa4841
	clipboard->targets[0] = XInternAtom(xfc->display, "TIMESTAMP", FALSE);
Packit Service fa4841
	clipboard->targets[1] = XInternAtom(xfc->display, "TARGETS", FALSE);
Packit Service fa4841
	clipboard->numTargets = 2;
Packit Service fa4841
	clipboard->incr_atom = XInternAtom(xfc->display, "INCR", FALSE);
Packit Service fa4841
	clipboard->delegate = ClipboardGetDelegate(clipboard->system);
Packit Service fa4841
	clipboard->delegate->custom = clipboard;
Packit Service fa4841
	/* TODO: set up a filesystem base path for local URI */
Packit Service fa4841
	/* clipboard->delegate->basePath = "file:///tmp/foo/bar/gaga"; */
Packit Service fa4841
	clipboard->delegate->ClipboardFileSizeSuccess = xf_cliprdr_clipboard_file_size_success;
Packit Service fa4841
	clipboard->delegate->ClipboardFileSizeFailure = xf_cliprdr_clipboard_file_size_failure;
Packit Service fa4841
	clipboard->delegate->ClipboardFileRangeSuccess = xf_cliprdr_clipboard_file_range_success;
Packit Service fa4841
	clipboard->delegate->ClipboardFileRangeFailure = xf_cliprdr_clipboard_file_range_failure;
Packit Service fa4841
	return clipboard;
Packit Service fa4841
error:
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < n; i++)
Packit Service fa4841
		free(clipboard->clientFormats[i].formatName);
Packit Service fa4841
Packit Service fa4841
	ClipboardDestroy(clipboard->system);
Packit Service fa4841
	free(clipboard);
Packit Service fa4841
	return NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_clipboard_free(xfClipboard* clipboard)
Packit Service fa4841
{
Packit Service fa4841
	int i;
Packit Service fa4841
Packit Service fa4841
	if (!clipboard)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if (clipboard->serverFormats)
Packit Service fa4841
	{
Packit Service fa4841
		for (i = 0; i < clipboard->numServerFormats; i++)
Packit Service fa4841
			free(clipboard->serverFormats[i].formatName);
Packit Service fa4841
Packit Service fa4841
		free(clipboard->serverFormats);
Packit Service fa4841
		clipboard->serverFormats = NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (clipboard->numClientFormats)
Packit Service fa4841
	{
Packit Service fa4841
		for (i = 0; i < clipboard->numClientFormats; i++)
Packit Service fa4841
			free(clipboard->clientFormats[i].formatName);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	ClipboardDestroy(clipboard->system);
Packit Service fa4841
	free(clipboard->data);
Packit Service fa4841
	free(clipboard->data_raw);
Packit Service fa4841
	free(clipboard->respond);
Packit Service fa4841
	free(clipboard->incr_data);
Packit Service fa4841
	free(clipboard);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_cliprdr_init(xfContext* xfc, CliprdrClientContext* cliprdr)
Packit Service fa4841
{
Packit Service fa4841
	xfc->cliprdr = cliprdr;
Packit Service fa4841
	xfc->clipboard->context = cliprdr;
Packit Service fa4841
	cliprdr->custom = (void*)xfc->clipboard;
Packit Service fa4841
	cliprdr->MonitorReady = xf_cliprdr_monitor_ready;
Packit Service fa4841
	cliprdr->ServerCapabilities = xf_cliprdr_server_capabilities;
Packit Service fa4841
	cliprdr->ServerFormatList = xf_cliprdr_server_format_list;
Packit Service fa4841
	cliprdr->ServerFormatListResponse = xf_cliprdr_server_format_list_response;
Packit Service fa4841
	cliprdr->ServerFormatDataRequest = xf_cliprdr_server_format_data_request;
Packit Service fa4841
	cliprdr->ServerFormatDataResponse = xf_cliprdr_server_format_data_response;
Packit Service fa4841
	cliprdr->ServerFileContentsRequest = xf_cliprdr_server_file_contents_request;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_cliprdr_uninit(xfContext* xfc, CliprdrClientContext* cliprdr)
Packit Service fa4841
{
Packit Service fa4841
	xfc->cliprdr = NULL;
Packit Service fa4841
	cliprdr->custom = NULL;
Packit Service fa4841
Packit Service fa4841
	if (xfc->clipboard)
Packit Service fa4841
		xfc->clipboard->context = NULL;
Packit Service fa4841
}