Coverage Report

Created: 2025-07-01 06:46

/src/FreeRDP/channels/cliprdr/cliprdr_common.c
Line
Count
Source (jump to first uncovered line)
1
/**
2
 * FreeRDP: A Remote Desktop Protocol Implementation
3
 * Cliprdr common
4
 *
5
 * Copyright 2013 Marc-Andre Moreau <marcandre.moreau@gmail.com>
6
 * Copyright 2015 Thincast Technologies GmbH
7
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
8
 * Copyright 2019 Kobi Mizrachi <kmizrachi18@gmail.com>
9
 *
10
 * Licensed under the Apache License, Version 2.0 (the "License");
11
 * you may not use this file except in compliance with the License.
12
 * You may obtain a copy of the License at
13
 *
14
 *     http://www.apache.org/licenses/LICENSE-2.0
15
 *
16
 * Unless required by applicable law or agreed to in writing, software
17
 * distributed under the License is distributed on an "AS IS" BASIS,
18
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
19
 * See the License for the specific language governing permissions and
20
 * limitations under the License.
21
 */
22
23
#include <winpr/crt.h>
24
#include <winpr/stream.h>
25
#include <freerdp/channels/log.h>
26
27
#define TAG CHANNELS_TAG("cliprdr.common")
28
29
#include "cliprdr_common.h"
30
31
static const char* CB_MSG_TYPE_STR(UINT32 type)
32
0
{
33
0
  switch (type)
34
0
  {
35
0
    case CB_TYPE_NONE:
36
0
      return "CB_TYPE_NONE";
37
0
    case CB_MONITOR_READY:
38
0
      return "CB_MONITOR_READY";
39
0
    case CB_FORMAT_LIST:
40
0
      return "CB_FORMAT_LIST";
41
0
    case CB_FORMAT_LIST_RESPONSE:
42
0
      return "CB_FORMAT_LIST_RESPONSE";
43
0
    case CB_FORMAT_DATA_REQUEST:
44
0
      return "CB_FORMAT_DATA_REQUEST";
45
0
    case CB_FORMAT_DATA_RESPONSE:
46
0
      return "CB_FORMAT_DATA_RESPONSE";
47
0
    case CB_TEMP_DIRECTORY:
48
0
      return "CB_TEMP_DIRECTORY";
49
0
    case CB_CLIP_CAPS:
50
0
      return "CB_CLIP_CAPS";
51
0
    case CB_FILECONTENTS_REQUEST:
52
0
      return "CB_FILECONTENTS_REQUEST";
53
0
    case CB_FILECONTENTS_RESPONSE:
54
0
      return "CB_FILECONTENTS_RESPONSE";
55
0
    case CB_LOCK_CLIPDATA:
56
0
      return "CB_LOCK_CLIPDATA";
57
0
    case CB_UNLOCK_CLIPDATA:
58
0
      return "CB_UNLOCK_CLIPDATA";
59
0
    default:
60
0
      return "UNKNOWN";
61
0
  }
62
0
}
63
64
const char* CB_MSG_TYPE_STRING(UINT16 type, char* buffer, size_t size)
65
0
{
66
0
  (void)_snprintf(buffer, size, "%s [0x%04" PRIx16 "]", CB_MSG_TYPE_STR(type), type);
67
0
  return buffer;
68
0
}
69
70
const char* CB_MSG_FLAGS_STRING(UINT16 msgFlags, char* buffer, size_t size)
71
0
{
72
0
  if ((msgFlags & CB_RESPONSE_OK) != 0)
73
0
    winpr_str_append("CB_RESPONSE_OK", buffer, size, "|");
74
0
  if ((msgFlags & CB_RESPONSE_FAIL) != 0)
75
0
    winpr_str_append("CB_RESPONSE_FAIL", buffer, size, "|");
76
0
  if ((msgFlags & CB_ASCII_NAMES) != 0)
77
0
    winpr_str_append("CB_ASCII_NAMES", buffer, size, "|");
78
79
0
  const size_t len = strnlen(buffer, size);
80
0
  if (!len)
81
0
    winpr_str_append("NONE", buffer, size, "");
82
83
0
  char val[32] = { 0 };
84
0
  (void)_snprintf(val, sizeof(val), "[0x%04" PRIx16 "]", msgFlags);
85
0
  winpr_str_append(val, buffer, size, "|");
86
0
  return buffer;
87
0
}
88
89
static BOOL cliprdr_validate_file_contents_request(const CLIPRDR_FILE_CONTENTS_REQUEST* request)
90
0
{
91
  /*
92
   * [MS-RDPECLIP] 2.2.5.3 File Contents Request PDU (CLIPRDR_FILECONTENTS_REQUEST).
93
   *
94
   * A request for the size of the file identified by the lindex field. The size MUST be
95
   * returned as a 64-bit, unsigned integer. The cbRequested field MUST be set to
96
   * 0x00000008 and both the nPositionLow and nPositionHigh fields MUST be
97
   * set to 0x00000000.
98
   */
99
100
0
  if (request->dwFlags & FILECONTENTS_SIZE)
101
0
  {
102
0
    if (request->cbRequested != sizeof(UINT64))
103
0
    {
104
0
      WLog_ERR(TAG, "cbRequested must be %" PRIu32 ", got %" PRIu32 "", sizeof(UINT64),
105
0
               request->cbRequested);
106
0
      return FALSE;
107
0
    }
108
109
0
    if (request->nPositionHigh != 0 || request->nPositionLow != 0)
110
0
    {
111
0
      WLog_ERR(TAG, "nPositionHigh and nPositionLow must be set to 0");
112
0
      return FALSE;
113
0
    }
114
0
  }
115
116
0
  return TRUE;
117
0
}
118
119
wStream* cliprdr_packet_new(UINT16 msgType, UINT16 msgFlags, size_t dataLen)
120
0
{
121
0
  WINPR_ASSERT(dataLen < UINT32_MAX);
122
0
  wStream* s = Stream_New(NULL, dataLen + 8ULL);
123
124
0
  if (!s)
125
0
  {
126
0
    WLog_ERR(TAG, "Stream_New failed!");
127
0
    return NULL;
128
0
  }
129
130
0
  Stream_Write_UINT16(s, msgType);
131
0
  Stream_Write_UINT16(s, msgFlags);
132
  /* Write actual length after the entire packet has been constructed. */
133
0
  Stream_Write_UINT32(s, 0);
134
0
  return s;
135
0
}
136
137
static void cliprdr_write_file_contents_request(wStream* s,
138
                                                const CLIPRDR_FILE_CONTENTS_REQUEST* request)
139
0
{
140
0
  Stream_Write_UINT32(s, request->streamId);      /* streamId (4 bytes) */
141
0
  Stream_Write_UINT32(s, request->listIndex);     /* listIndex (4 bytes) */
142
0
  Stream_Write_UINT32(s, request->dwFlags);       /* dwFlags (4 bytes) */
143
0
  Stream_Write_UINT32(s, request->nPositionLow);  /* nPositionLow (4 bytes) */
144
0
  Stream_Write_UINT32(s, request->nPositionHigh); /* nPositionHigh (4 bytes) */
145
0
  Stream_Write_UINT32(s, request->cbRequested);   /* cbRequested (4 bytes) */
146
147
0
  if (request->haveClipDataId)
148
0
    Stream_Write_UINT32(s, request->clipDataId); /* clipDataId (4 bytes) */
149
0
}
150
151
static INLINE void cliprdr_write_lock_unlock_clipdata(wStream* s, UINT32 clipDataId)
152
0
{
153
0
  Stream_Write_UINT32(s, clipDataId);
154
0
}
155
156
static void cliprdr_write_lock_clipdata(wStream* s,
157
                                        const CLIPRDR_LOCK_CLIPBOARD_DATA* lockClipboardData)
158
0
{
159
0
  cliprdr_write_lock_unlock_clipdata(s, lockClipboardData->clipDataId);
160
0
}
161
162
static void cliprdr_write_unlock_clipdata(wStream* s,
163
                                          const CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
164
0
{
165
0
  cliprdr_write_lock_unlock_clipdata(s, unlockClipboardData->clipDataId);
166
0
}
167
168
static void cliprdr_write_file_contents_response(wStream* s,
169
                                                 const CLIPRDR_FILE_CONTENTS_RESPONSE* response)
170
0
{
171
0
  Stream_Write_UINT32(s, response->streamId); /* streamId (4 bytes) */
172
0
  Stream_Write(s, response->requestedData, response->cbRequested);
173
0
}
174
175
wStream* cliprdr_packet_lock_clipdata_new(const CLIPRDR_LOCK_CLIPBOARD_DATA* lockClipboardData)
176
0
{
177
0
  wStream* s = NULL;
178
179
0
  if (!lockClipboardData)
180
0
    return NULL;
181
182
0
  s = cliprdr_packet_new(CB_LOCK_CLIPDATA, 0, 4);
183
184
0
  if (!s)
185
0
    return NULL;
186
187
0
  cliprdr_write_lock_clipdata(s, lockClipboardData);
188
0
  return s;
189
0
}
190
191
wStream*
192
cliprdr_packet_unlock_clipdata_new(const CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
193
0
{
194
0
  wStream* s = NULL;
195
196
0
  if (!unlockClipboardData)
197
0
    return NULL;
198
199
0
  s = cliprdr_packet_new(CB_UNLOCK_CLIPDATA, 0, 4);
200
201
0
  if (!s)
202
0
    return NULL;
203
204
0
  cliprdr_write_unlock_clipdata(s, unlockClipboardData);
205
0
  return s;
206
0
}
207
208
wStream* cliprdr_packet_file_contents_request_new(const CLIPRDR_FILE_CONTENTS_REQUEST* request)
209
0
{
210
0
  wStream* s = NULL;
211
212
0
  if (!request)
213
0
    return NULL;
214
215
0
  s = cliprdr_packet_new(CB_FILECONTENTS_REQUEST, 0, 28);
216
217
0
  if (!s)
218
0
    return NULL;
219
220
0
  cliprdr_write_file_contents_request(s, request);
221
0
  return s;
222
0
}
223
224
wStream* cliprdr_packet_file_contents_response_new(const CLIPRDR_FILE_CONTENTS_RESPONSE* response)
225
0
{
226
0
  wStream* s = NULL;
227
228
0
  if (!response)
229
0
    return NULL;
230
231
0
  s = cliprdr_packet_new(CB_FILECONTENTS_RESPONSE, response->common.msgFlags,
232
0
                         4 + response->cbRequested);
233
234
0
  if (!s)
235
0
    return NULL;
236
237
0
  cliprdr_write_file_contents_response(s, response);
238
0
  return s;
239
0
}
240
241
wStream* cliprdr_packet_format_list_new(const CLIPRDR_FORMAT_LIST* formatList,
242
                                        BOOL useLongFormatNames, BOOL useAsciiNames)
243
0
{
244
0
  WINPR_ASSERT(formatList);
245
246
0
  if (formatList->common.msgType != CB_FORMAT_LIST)
247
0
    WLog_WARN(TAG, "called with invalid type %08" PRIx32, formatList->common.msgType);
248
249
0
  if (useLongFormatNames && useAsciiNames)
250
0
    WLog_WARN(TAG, "called with invalid arguments useLongFormatNames=true && "
251
0
                   "useAsciiNames=true. useAsciiNames requires "
252
0
                   "useLongFormatNames=false, ignoring argument.");
253
254
0
  const UINT32 length = formatList->numFormats * 36;
255
0
  const size_t formatNameCharSize =
256
0
      (useLongFormatNames || !useAsciiNames) ? sizeof(WCHAR) : sizeof(CHAR);
257
258
0
  wStream* s = cliprdr_packet_new(CB_FORMAT_LIST, 0, length);
259
0
  if (!s)
260
0
  {
261
0
    WLog_ERR(TAG, "cliprdr_packet_new failed!");
262
0
    return NULL;
263
0
  }
264
265
0
  for (UINT32 index = 0; index < formatList->numFormats; index++)
266
0
  {
267
0
    const CLIPRDR_FORMAT* format = &(formatList->formats[index]);
268
269
0
    const char* szFormatName = format->formatName;
270
0
    size_t formatNameLength = 0;
271
0
    if (szFormatName)
272
0
      formatNameLength = strlen(szFormatName);
273
274
0
    size_t formatNameMaxLength = formatNameLength + 1; /* Ensure '\0' termination in output */
275
0
    if (!Stream_EnsureRemainingCapacity(s,
276
0
                                        4 + MAX(32, formatNameMaxLength * formatNameCharSize)))
277
0
      goto fail;
278
279
0
    Stream_Write_UINT32(s, format->formatId); /* formatId (4 bytes) */
280
281
0
    if (!useLongFormatNames)
282
0
    {
283
0
      formatNameMaxLength = useAsciiNames ? 32 : 16;
284
0
      formatNameLength = MIN(formatNameMaxLength - 1, formatNameLength);
285
0
    }
286
287
0
    if (szFormatName && (formatNameLength > 0))
288
0
    {
289
0
      if (useAsciiNames)
290
0
      {
291
0
        Stream_Write(s, szFormatName, formatNameLength);
292
0
        Stream_Zero(s, formatNameMaxLength - formatNameLength);
293
0
      }
294
0
      else
295
0
      {
296
0
        if (Stream_Write_UTF16_String_From_UTF8(s, formatNameMaxLength, szFormatName,
297
0
                                                formatNameLength, TRUE) < 0)
298
0
          goto fail;
299
0
      }
300
0
    }
301
0
    else
302
0
      Stream_Zero(s, formatNameMaxLength * formatNameCharSize);
303
0
  }
304
305
0
  return s;
306
307
0
fail:
308
0
  Stream_Free(s, TRUE);
309
0
  return NULL;
310
0
}
311
312
UINT cliprdr_read_unlock_clipdata(wStream* s, CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
313
0
{
314
0
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
315
0
    return ERROR_INVALID_DATA;
316
317
0
  Stream_Read_UINT32(s, unlockClipboardData->clipDataId); /* clipDataId (4 bytes) */
318
0
  return CHANNEL_RC_OK;
319
0
}
320
321
UINT cliprdr_read_format_data_request(wStream* s, CLIPRDR_FORMAT_DATA_REQUEST* request)
322
0
{
323
0
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
324
0
    return ERROR_INVALID_DATA;
325
326
0
  Stream_Read_UINT32(s, request->requestedFormatId); /* requestedFormatId (4 bytes) */
327
0
  return CHANNEL_RC_OK;
328
0
}
329
330
UINT cliprdr_read_format_data_response(wStream* s, CLIPRDR_FORMAT_DATA_RESPONSE* response)
331
0
{
332
0
  response->requestedFormatData = NULL;
333
334
0
  if (!Stream_CheckAndLogRequiredLength(TAG, s, response->common.dataLen))
335
0
    return ERROR_INVALID_DATA;
336
337
0
  if (response->common.dataLen)
338
0
    response->requestedFormatData = Stream_ConstPointer(s);
339
340
0
  return CHANNEL_RC_OK;
341
0
}
342
343
UINT cliprdr_read_file_contents_request(wStream* s, CLIPRDR_FILE_CONTENTS_REQUEST* request)
344
0
{
345
0
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
346
0
    return ERROR_INVALID_DATA;
347
348
0
  request->haveClipDataId = FALSE;
349
0
  Stream_Read_UINT32(s, request->streamId);      /* streamId (4 bytes) */
350
0
  Stream_Read_UINT32(s, request->listIndex);     /* listIndex (4 bytes) */
351
0
  Stream_Read_UINT32(s, request->dwFlags);       /* dwFlags (4 bytes) */
352
0
  Stream_Read_UINT32(s, request->nPositionLow);  /* nPositionLow (4 bytes) */
353
0
  Stream_Read_UINT32(s, request->nPositionHigh); /* nPositionHigh (4 bytes) */
354
0
  Stream_Read_UINT32(s, request->cbRequested);   /* cbRequested (4 bytes) */
355
356
0
  if (Stream_GetRemainingLength(s) >= 4)
357
0
  {
358
0
    Stream_Read_UINT32(s, request->clipDataId); /* clipDataId (4 bytes) */
359
0
    request->haveClipDataId = TRUE;
360
0
  }
361
362
0
  if (!cliprdr_validate_file_contents_request(request))
363
0
    return ERROR_BAD_ARGUMENTS;
364
365
0
  return CHANNEL_RC_OK;
366
0
}
367
368
UINT cliprdr_read_file_contents_response(wStream* s, CLIPRDR_FILE_CONTENTS_RESPONSE* response)
369
0
{
370
0
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
371
0
    return ERROR_INVALID_DATA;
372
373
0
  Stream_Read_UINT32(s, response->streamId);   /* streamId (4 bytes) */
374
0
  response->requestedData = Stream_ConstPointer(s); /* requestedFileContentsData */
375
376
0
  WINPR_ASSERT(response->common.dataLen >= 4);
377
0
  response->cbRequested = response->common.dataLen - 4;
378
0
  return CHANNEL_RC_OK;
379
0
}
380
381
UINT cliprdr_read_format_list(wLog* log, wStream* s, CLIPRDR_FORMAT_LIST* formatList,
382
                              BOOL useLongFormatNames)
383
0
{
384
0
  UINT32 index = 0;
385
0
  size_t formatNameLength = 0;
386
0
  const char* szFormatName = NULL;
387
0
  const WCHAR* wszFormatName = NULL;
388
0
  wStream sub1buffer = { 0 };
389
0
  CLIPRDR_FORMAT* formats = NULL;
390
0
  UINT error = ERROR_INTERNAL_ERROR;
391
392
0
  const BOOL asciiNames = (formatList->common.msgFlags & CB_ASCII_NAMES) ? TRUE : FALSE;
393
394
0
  index = 0;
395
  /* empty format list */
396
0
  formatList->formats = NULL;
397
0
  formatList->numFormats = 0;
398
399
0
  wStream* sub1 =
400
0
      Stream_StaticConstInit(&sub1buffer, Stream_ConstPointer(s), formatList->common.dataLen);
401
0
  if (!Stream_SafeSeek(s, formatList->common.dataLen))
402
0
    return ERROR_INVALID_DATA;
403
404
0
  if (!formatList->common.dataLen)
405
0
  {
406
0
  }
407
0
  else if (!useLongFormatNames)
408
0
  {
409
0
    const size_t cap = Stream_Capacity(sub1) / 36ULL;
410
0
    if (cap > UINT32_MAX)
411
0
    {
412
0
      WLog_Print(log, WLOG_ERROR, "Invalid short format list length: %" PRIuz "", cap);
413
0
      return ERROR_INTERNAL_ERROR;
414
0
    }
415
0
    formatList->numFormats = (UINT32)cap;
416
417
0
    if (formatList->numFormats)
418
0
      formats = (CLIPRDR_FORMAT*)calloc(formatList->numFormats, sizeof(CLIPRDR_FORMAT));
419
420
0
    if (!formats)
421
0
    {
422
0
      WLog_Print(log, WLOG_ERROR, "calloc failed!");
423
0
      return CHANNEL_RC_NO_MEMORY;
424
0
    }
425
426
0
    formatList->formats = formats;
427
428
0
    while (Stream_GetRemainingLength(sub1) >= 4)
429
0
    {
430
0
      if (index >= formatList->numFormats)
431
0
        goto error_out;
432
433
0
      CLIPRDR_FORMAT* format = &formats[index];
434
435
0
      Stream_Read_UINT32(sub1, format->formatId); /* formatId (4 bytes) */
436
437
      /* According to MS-RDPECLIP 2.2.3.1.1.1 formatName is "a 32-byte block containing
438
       * the *null-terminated* name assigned to the Clipboard Format: (32 ASCII 8 characters
439
       * or 16 Unicode characters)"
440
       * However, both Windows RDSH and mstsc violate this specs as seen in the following
441
       * example of a transferred short format name string: [R.i.c.h. .T.e.x.t. .F.o.r.m.a.t.]
442
       * These are 16 unicode characters - *without* terminating null !
443
       */
444
445
0
      szFormatName = Stream_ConstPointer(sub1);
446
0
      wszFormatName = Stream_ConstPointer(sub1);
447
0
      if (!Stream_SafeSeek(sub1, 32))
448
0
        goto error_out;
449
450
0
      free(format->formatName);
451
0
      format->formatName = NULL;
452
453
0
      if (asciiNames)
454
0
      {
455
0
        if (szFormatName[0])
456
0
        {
457
          /* ensure null termination */
458
0
          format->formatName = strndup(szFormatName, 31);
459
0
          if (!format->formatName)
460
0
          {
461
0
            WLog_Print(log, WLOG_ERROR, "malloc failed!");
462
0
            error = CHANNEL_RC_NO_MEMORY;
463
0
            goto error_out;
464
0
          }
465
0
        }
466
0
      }
467
0
      else
468
0
      {
469
0
        if (wszFormatName[0])
470
0
        {
471
0
          format->formatName = ConvertWCharNToUtf8Alloc(wszFormatName, 16, NULL);
472
0
          if (!format->formatName)
473
0
            goto error_out;
474
0
        }
475
0
      }
476
477
0
      index++;
478
0
    }
479
0
  }
480
0
  else
481
0
  {
482
0
    wStream sub2buffer = sub1buffer;
483
0
    wStream* sub2 = &sub2buffer;
484
485
0
    while (Stream_GetRemainingLength(sub1) > 0)
486
0
    {
487
0
      size_t rest = 0;
488
0
      if (!Stream_SafeSeek(sub1, 4)) /* formatId (4 bytes) */
489
0
        goto error_out;
490
491
0
      wszFormatName = Stream_ConstPointer(sub1);
492
0
      rest = Stream_GetRemainingLength(sub1);
493
0
      formatNameLength = _wcsnlen(wszFormatName, rest / sizeof(WCHAR));
494
495
0
      if (!Stream_SafeSeek(sub1, (formatNameLength + 1) * sizeof(WCHAR)))
496
0
        goto error_out;
497
0
      formatList->numFormats++;
498
0
    }
499
500
0
    if (formatList->numFormats)
501
0
      formats = (CLIPRDR_FORMAT*)calloc(formatList->numFormats, sizeof(CLIPRDR_FORMAT));
502
503
0
    if (!formats)
504
0
    {
505
0
      WLog_Print(log, WLOG_ERROR, "calloc failed!");
506
0
      return CHANNEL_RC_NO_MEMORY;
507
0
    }
508
509
0
    formatList->formats = formats;
510
511
0
    while (Stream_GetRemainingLength(sub2) >= 4)
512
0
    {
513
0
      if (index >= formatList->numFormats)
514
0
        goto error_out;
515
516
0
      size_t rest = 0;
517
0
      CLIPRDR_FORMAT* format = &formats[index];
518
519
0
      Stream_Read_UINT32(sub2, format->formatId); /* formatId (4 bytes) */
520
521
0
      free(format->formatName);
522
0
      format->formatName = NULL;
523
524
0
      wszFormatName = Stream_ConstPointer(sub2);
525
0
      rest = Stream_GetRemainingLength(sub2);
526
0
      formatNameLength = _wcsnlen(wszFormatName, rest / sizeof(WCHAR));
527
0
      if (!Stream_SafeSeek(sub2, (formatNameLength + 1) * sizeof(WCHAR)))
528
0
        goto error_out;
529
530
0
      if (formatNameLength)
531
0
      {
532
0
        format->formatName =
533
0
            ConvertWCharNToUtf8Alloc(wszFormatName, formatNameLength, NULL);
534
0
        if (!format->formatName)
535
0
          goto error_out;
536
0
      }
537
538
0
      index++;
539
0
    }
540
0
  }
541
542
0
  return CHANNEL_RC_OK;
543
544
0
error_out:
545
0
  cliprdr_free_format_list(formatList);
546
0
  return error;
547
0
}
548
549
void cliprdr_free_format_list(CLIPRDR_FORMAT_LIST* formatList)
550
0
{
551
0
  if (formatList == NULL)
552
0
    return;
553
554
0
  if (formatList->formats)
555
0
  {
556
0
    for (UINT32 index = 0; index < formatList->numFormats; index++)
557
0
    {
558
0
      free(formatList->formats[index].formatName);
559
0
    }
560
561
0
    free(formatList->formats);
562
0
    formatList->formats = NULL;
563
0
    formatList->numFormats = 0;
564
0
  }
565
0
}