FreeRDP
Loading...
Searching...
No Matches
cliprdr_format.c
1
23#include <freerdp/config.h>
24
25#include <winpr/crt.h>
26#include <winpr/print.h>
27#include <winpr/clipboard.h>
28
29#include <freerdp/types.h>
30#include <freerdp/freerdp.h>
31#include <freerdp/settings.h>
32#include <freerdp/constants.h>
33#include <freerdp/client/cliprdr.h>
34
35#include "cliprdr_main.h"
36#include "cliprdr_format.h"
37#include "../cliprdr_common.h"
38
39CLIPRDR_FORMAT_LIST cliprdr_filter_format_list(const CLIPRDR_FORMAT_LIST* list, const UINT32 mask,
40 const UINT32 checkMask)
41{
42 const UINT32 maskData =
43 checkMask & (CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL);
44 const UINT32 maskFiles =
45 checkMask & (CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES | CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES);
46 WINPR_ASSERT(list);
47
48 CLIPRDR_FORMAT_LIST filtered = { 0 };
49 filtered.common.msgType = CB_FORMAT_LIST;
50 filtered.numFormats = list->numFormats;
51 filtered.formats = calloc(filtered.numFormats, sizeof(CLIPRDR_FORMAT));
52
53 size_t wpos = 0;
54 if ((mask & checkMask) == checkMask)
55 {
56 for (size_t x = 0; x < list->numFormats; x++)
57 {
58 const CLIPRDR_FORMAT* format = &list->formats[x];
59 CLIPRDR_FORMAT* cur = &filtered.formats[x];
60 cur->formatId = format->formatId;
61 if (format->formatName)
62 cur->formatName = _strdup(format->formatName);
63 wpos++;
64 }
65 }
66 else if ((mask & maskFiles) != 0)
67 {
68 for (size_t x = 0; x < list->numFormats; x++)
69 {
70 const CLIPRDR_FORMAT* format = &list->formats[x];
71 CLIPRDR_FORMAT* cur = &filtered.formats[wpos];
72
73 if (!format->formatName)
74 continue;
75 if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0 ||
76 strcmp(format->formatName, type_FileContents) == 0)
77 {
78 cur->formatId = format->formatId;
79 cur->formatName = _strdup(format->formatName);
80 wpos++;
81 }
82 }
83 }
84 else if ((mask & maskData) != 0)
85 {
86 for (size_t x = 0; x < list->numFormats; x++)
87 {
88 const CLIPRDR_FORMAT* format = &list->formats[x];
89 CLIPRDR_FORMAT* cur = &filtered.formats[wpos];
90
91 if (!format->formatName ||
92 (strcmp(format->formatName, type_FileGroupDescriptorW) != 0 &&
93 strcmp(format->formatName, type_FileContents) != 0))
94 {
95 cur->formatId = format->formatId;
96 if (format->formatName)
97 cur->formatName = _strdup(format->formatName);
98 wpos++;
99 }
100 }
101 }
102 WINPR_ASSERT(wpos <= UINT32_MAX);
103 filtered.numFormats = (UINT32)wpos;
104 return filtered;
105}
106
112UINT cliprdr_process_format_list(cliprdrPlugin* cliprdr, wStream* s, UINT32 dataLen,
113 UINT16 msgFlags)
114{
115 CLIPRDR_FORMAT_LIST formatList = { 0 };
116 CLIPRDR_FORMAT_LIST filteredFormatList = { 0 };
117 CliprdrClientContext* context = cliprdr_get_client_interface(cliprdr);
118 UINT error = CHANNEL_RC_OK;
119
120 formatList.common.msgType = CB_FORMAT_LIST;
121 formatList.common.msgFlags = msgFlags;
122 formatList.common.dataLen = dataLen;
123
124 if ((error =
125 cliprdr_read_format_list(cliprdr->log, s, &formatList, cliprdr->useLongFormatNames)))
126 goto error_out;
127
128 const UINT32 mask =
129 freerdp_settings_get_uint32(context->rdpcontext->settings, FreeRDP_ClipboardFeatureMask);
130 filteredFormatList = cliprdr_filter_format_list(
131 &formatList, mask, CLIPRDR_FLAG_REMOTE_TO_LOCAL | CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES);
132 if (filteredFormatList.numFormats == 0)
133 goto error_out;
134
135 const DWORD level = WLOG_DEBUG;
136 if (WLog_IsLevelActive(cliprdr->log, level))
137 {
138 WLog_Print(cliprdr->log, level, "ServerFormatList: numFormats: %" PRIu32 "",
139 formatList.numFormats);
140 for (size_t x = 0; x < formatList.numFormats; x++)
141 {
142 const CLIPRDR_FORMAT* format = &formatList.formats[x];
143 WLog_Print(cliprdr->log, level, "[%" PRIu32 "]: id=0x%08" PRIx32 " [%s|%s]", x,
144 format->formatId, ClipboardGetFormatIdString(format->formatId),
145 format->formatName);
146 }
147
148 WLog_Print(cliprdr->log, level, "ServerFormatList [filtered]: numFormats: %" PRIu32 "",
149 filteredFormatList.numFormats);
150 for (size_t x = 0; x < filteredFormatList.numFormats; x++)
151 {
152 const CLIPRDR_FORMAT* format = &filteredFormatList.formats[x];
153 WLog_Print(cliprdr->log, level, "[%" PRIu32 "]: id=0x%08" PRIx32 " [%s|%s]", x,
154 format->formatId, ClipboardGetFormatIdString(format->formatId),
155 format->formatName);
156 }
157 }
158
159 if (context->ServerFormatList)
160 {
161 if ((error = context->ServerFormatList(context, &filteredFormatList)))
162 WLog_Print(cliprdr->log, WLOG_ERROR, "ServerFormatList failed with error %" PRIu32 "",
163 error);
164 }
165
166error_out:
167 cliprdr_free_format_list(&filteredFormatList);
168 cliprdr_free_format_list(&formatList);
169 return error;
170}
171
177UINT cliprdr_process_format_list_response(cliprdrPlugin* cliprdr, WINPR_ATTR_UNUSED wStream* s,
178 UINT32 dataLen, UINT16 msgFlags)
179{
180 CLIPRDR_FORMAT_LIST_RESPONSE formatListResponse = { 0 };
181 CliprdrClientContext* context = cliprdr_get_client_interface(cliprdr);
182 UINT error = CHANNEL_RC_OK;
183
184 WLog_Print(cliprdr->log, WLOG_DEBUG, "ServerFormatListResponse");
185
186 formatListResponse.common.msgType = CB_FORMAT_LIST_RESPONSE;
187 formatListResponse.common.msgFlags = msgFlags;
188 formatListResponse.common.dataLen = dataLen;
189
190 IFCALLRET(context->ServerFormatListResponse, error, context, &formatListResponse);
191 if (error)
192 WLog_Print(cliprdr->log, WLOG_ERROR,
193 "ServerFormatListResponse failed with error %" PRIu32 "!", error);
194
195 return error;
196}
197
203UINT cliprdr_process_format_data_request(cliprdrPlugin* cliprdr, wStream* s, UINT32 dataLen,
204 UINT16 msgFlags)
205{
206 CLIPRDR_FORMAT_DATA_REQUEST formatDataRequest = { 0 };
207 CliprdrClientContext* context = cliprdr_get_client_interface(cliprdr);
208 UINT error = CHANNEL_RC_OK;
209
210 formatDataRequest.common.msgType = CB_FORMAT_DATA_REQUEST;
211 formatDataRequest.common.msgFlags = msgFlags;
212 formatDataRequest.common.dataLen = dataLen;
213
214 if ((error = cliprdr_read_format_data_request(s, &formatDataRequest)))
215 return error;
216
217 WLog_Print(cliprdr->log, WLOG_DEBUG, "ServerFormatDataRequest (0x%08" PRIx32 " [%s])",
218 formatDataRequest.requestedFormatId,
219 ClipboardGetFormatIdString(formatDataRequest.requestedFormatId));
220
221 const UINT32 mask =
222 freerdp_settings_get_uint32(context->rdpcontext->settings, FreeRDP_ClipboardFeatureMask);
223 if ((mask & (CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES)) == 0)
224 {
225 return cliprdr_send_error_response(cliprdr, CB_FORMAT_DATA_RESPONSE);
226 }
227
228 context->lastRequestedFormatId = formatDataRequest.requestedFormatId;
229 IFCALLRET(context->ServerFormatDataRequest, error, context, &formatDataRequest);
230 if (error)
231 WLog_Print(cliprdr->log, WLOG_ERROR,
232 "ServerFormatDataRequest failed with error %" PRIu32 "!", error);
233
234 return error;
235}
236
242UINT cliprdr_process_format_data_response(cliprdrPlugin* cliprdr, wStream* s, UINT32 dataLen,
243 UINT16 msgFlags)
244{
245 CLIPRDR_FORMAT_DATA_RESPONSE formatDataResponse = { 0 };
246 CliprdrClientContext* context = cliprdr_get_client_interface(cliprdr);
247 UINT error = CHANNEL_RC_OK;
248
249 WLog_Print(cliprdr->log, WLOG_DEBUG,
250 "ServerFormatDataResponse: msgFlags=0x%08" PRIx32 ", dataLen=%" PRIu32, msgFlags,
251 dataLen);
252
253 formatDataResponse.common.msgType = CB_FORMAT_DATA_RESPONSE;
254 formatDataResponse.common.msgFlags = msgFlags;
255 formatDataResponse.common.dataLen = dataLen;
256
257 if ((error = cliprdr_read_format_data_response(s, &formatDataResponse)))
258 return error;
259
260 const UINT32 mask =
261 freerdp_settings_get_uint32(context->rdpcontext->settings, FreeRDP_ClipboardFeatureMask);
262 if ((mask & (CLIPRDR_FLAG_REMOTE_TO_LOCAL | CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES)) == 0)
263 {
264 WLog_Print(cliprdr->log, WLOG_WARN,
265 "Received ServerFormatDataResponse but remote -> local clipboard is disabled");
266 return CHANNEL_RC_OK;
267 }
268
269 IFCALLRET(context->ServerFormatDataResponse, error, context, &formatDataResponse);
270 if (error)
271 WLog_Print(cliprdr->log, WLOG_ERROR,
272 "ServerFormatDataResponse failed with error %" PRIu32 "!", error);
273
274 return error;
275}
FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.