Coverage Report

Created: 2026-06-15 06:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/FreeRDP/libfreerdp/core/surface.c
Line
Count
Source
1
/**
2
 * FreeRDP: A Remote Desktop Protocol Implementation
3
 * Surface Commands
4
 *
5
 * Copyright 2011 Vic Lee
6
 *
7
 * Licensed under the Apache License, Version 2.0 (the "License");
8
 * you may not use this file except in compliance with the License.
9
 * You may obtain a copy of the License at
10
 *
11
 *     http://www.apache.org/licenses/LICENSE-2.0
12
 *
13
 * Unless required by applicable law or agreed to in writing, software
14
 * distributed under the License is distributed on an "AS IS" BASIS,
15
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
 * See the License for the specific language governing permissions and
17
 * limitations under the License.
18
 */
19
20
#include <freerdp/config.h>
21
22
#include "settings.h"
23
24
#include <winpr/assert.h>
25
#include <winpr/cast.h>
26
27
#include <freerdp/utils/pcap.h>
28
#include <freerdp/log.h>
29
30
#include "../cache/cache.h"
31
#include "surface.h"
32
33
#define TAG FREERDP_TAG("core.surface")
34
35
static BOOL update_recv_surfcmd_bitmap_header_ex(wStream* s, TS_COMPRESSED_BITMAP_HEADER_EX* header)
36
2.64k
{
37
2.64k
  if (!s || !header)
38
0
    return FALSE;
39
40
2.64k
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
41
107
    return FALSE;
42
43
2.53k
  Stream_Read_UINT32(s, header->highUniqueId);
44
2.53k
  Stream_Read_UINT32(s, header->lowUniqueId);
45
2.53k
  Stream_Read_UINT64(s, header->tmMilliseconds);
46
2.53k
  Stream_Read_UINT64(s, header->tmSeconds);
47
2.53k
  return TRUE;
48
2.64k
}
49
50
static BOOL update_recv_surfcmd_bitmap_ex(wStream* s, TS_BITMAP_DATA_EX* bmp)
51
4.57k
{
52
4.57k
  if (!s || !bmp)
53
0
    return FALSE;
54
55
4.57k
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 12))
56
281
    return FALSE;
57
58
4.29k
  Stream_Read_UINT8(s, bmp->bpp);
59
4.29k
  Stream_Read_UINT8(s, bmp->flags);
60
4.29k
  Stream_Seek(s, 1); /* reserved */
61
4.29k
  Stream_Read_UINT8(s, bmp->codecID);
62
4.29k
  Stream_Read_UINT16(s, bmp->width);
63
4.29k
  Stream_Read_UINT16(s, bmp->height);
64
4.29k
  Stream_Read_UINT32(s, bmp->bitmapDataLength);
65
66
4.29k
  if ((bmp->width == 0) || (bmp->height == 0))
67
190
  {
68
190
    WLog_ERR(TAG, "invalid size value width=%" PRIu16 ", height=%" PRIu16, bmp->width,
69
190
             bmp->height);
70
190
    return FALSE;
71
190
  }
72
73
4.10k
  if ((bmp->bpp < 1) || (bmp->bpp > 32))
74
539
  {
75
539
    WLog_ERR(TAG, "invalid bpp value %" PRIu32 "", bmp->bpp);
76
539
    return FALSE;
77
539
  }
78
79
3.56k
  if (bmp->flags & EX_COMPRESSED_BITMAP_HEADER_PRESENT)
80
2.64k
  {
81
2.64k
    if (!update_recv_surfcmd_bitmap_header_ex(s, &bmp->exBitmapDataHeader))
82
107
      return FALSE;
83
2.64k
  }
84
85
3.45k
  bmp->bitmapData = Stream_Pointer(s);
86
3.45k
  if (!Stream_SafeSeek(s, bmp->bitmapDataLength))
87
2.40k
  {
88
2.40k
    WLog_ERR(TAG, "expected bitmapDataLength %" PRIu32 ", not enough data",
89
2.40k
             bmp->bitmapDataLength);
90
2.40k
    return FALSE;
91
2.40k
  }
92
1.05k
  return TRUE;
93
3.45k
}
94
95
static BOOL update_recv_surfcmd_is_rect_valid(const rdpContext* context,
96
                                              const SURFACE_BITS_COMMAND* cmd)
97
6.52k
{
98
6.52k
  WINPR_ASSERT(context);
99
6.52k
  WINPR_ASSERT(context->settings);
100
6.52k
  WINPR_ASSERT(cmd);
101
102
  /* We need a rectangle with left/top being smaller than right/bottom.
103
   * Also do not allow empty rectangles. */
104
6.52k
  if ((cmd->destTop >= cmd->destBottom) || (cmd->destLeft >= cmd->destRight))
105
1.36k
  {
106
1.36k
    WLog_WARN(TAG,
107
1.36k
              "Empty surface bits command rectangle: %" PRIu16 "x%" PRIu16 "-%" PRIu16
108
1.36k
              "x%" PRIu16,
109
1.36k
              cmd->destLeft, cmd->destTop, cmd->destRight, cmd->destBottom);
110
1.36k
    return FALSE;
111
1.36k
  }
112
113
  /* The rectangle needs to fit into our session size */
114
5.15k
  const DWORD DesktopWidth = freerdp_settings_get_uint32(context->settings, FreeRDP_DesktopWidth);
115
5.15k
  const DWORD DesktopHeight =
116
5.15k
      freerdp_settings_get_uint32(context->settings, FreeRDP_DesktopHeight);
117
5.15k
  if ((cmd->destRight > DesktopWidth) || (cmd->destBottom > DesktopHeight))
118
580
  {
119
580
    WLog_WARN(TAG,
120
580
              "Invalid surface bits command rectangle: %" PRIu16 "x%" PRIu16 "-%" PRIu16
121
580
              "x%" PRIu16 " does not fit %" PRIu32 "x%" PRIu32,
122
580
              cmd->destLeft, cmd->destTop, cmd->destRight, cmd->destBottom, DesktopWidth,
123
580
              DesktopHeight);
124
580
    return FALSE;
125
580
  }
126
127
4.57k
  return TRUE;
128
5.15k
}
129
130
static BOOL update_recv_surfcmd_surface_bits(rdpUpdate* update, wStream* s, UINT16 cmdType)
131
6.88k
{
132
6.88k
  rdp_update_internal* up = update_cast(update);
133
6.88k
  BOOL rc = FALSE;
134
6.88k
  SURFACE_BITS_COMMAND cmd = WINPR_C_ARRAY_INIT;
135
136
6.88k
  WINPR_ASSERT(up);
137
138
6.88k
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
139
364
    goto fail;
140
141
6.52k
  cmd.cmdType = cmdType;
142
6.52k
  Stream_Read_UINT16(s, cmd.destLeft);
143
6.52k
  Stream_Read_UINT16(s, cmd.destTop);
144
6.52k
  Stream_Read_UINT16(s, cmd.destRight);
145
6.52k
  Stream_Read_UINT16(s, cmd.destBottom);
146
147
6.52k
  if (!update_recv_surfcmd_is_rect_valid(update->context, &cmd))
148
1.94k
    goto fail;
149
150
4.57k
  if (!update_recv_surfcmd_bitmap_ex(s, &cmd.bmp))
151
3.52k
    goto fail;
152
153
1.05k
  up->stats.base[RDP_STATS_SURFACE_BITS]++;
154
1.05k
  switch (cmd.bmp.codecID)
155
1.05k
  {
156
2
    case RDP_CODEC_ID_REMOTEFX:
157
2
      up->stats.base[RDP_STATS_SURFACE_BITS_RFX]++;
158
2
      break;
159
3
    case RDP_CODEC_ID_IMAGE_REMOTEFX:
160
3
      up->stats.base[RDP_STATS_SURFACE_BITS_RFX_IMAGE]++;
161
3
      break;
162
2
    case RDP_CODEC_ID_NSCODEC:
163
2
      up->stats.base[RDP_STATS_SURFACE_BITS_NSC]++;
164
2
      break;
165
17
    case RDP_CODEC_ID_NONE:
166
17
      up->stats.base[RDP_STATS_SURFACE_BITS_NONE]++;
167
17
      break;
168
1.03k
    default:
169
1.03k
      up->stats.base[RDP_STATS_SURFACE_BITS_UNKNOWN]++;
170
1.03k
      break;
171
1.05k
  }
172
173
1.05k
  if (!IFCALLRESULT(TRUE, update->SurfaceBits, update->context, &cmd))
174
508
  {
175
508
    WLog_DBG(TAG, "update->SurfaceBits implementation failed");
176
508
    goto fail;
177
508
  }
178
179
546
  rc = TRUE;
180
6.88k
fail:
181
6.88k
  return rc;
182
546
}
183
184
static BOOL update_recv_surfcmd_frame_marker(rdpUpdate* update, wStream* s)
185
1.40k
{
186
1.40k
  SURFACE_FRAME_MARKER marker = WINPR_C_ARRAY_INIT;
187
1.40k
  rdp_update_internal* up = update_cast(update);
188
189
1.40k
  WINPR_ASSERT(s);
190
191
1.40k
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
192
278
    return FALSE;
193
194
1.13k
  Stream_Read_UINT16(s, marker.frameAction);
195
1.13k
  if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
196
589
    WLog_WARN(TAG,
197
1.13k
              "[SERVER-BUG]: got %" PRIuz ", expected %u"
198
1.13k
              " bytes. [MS-RDPBCGR] 2.2.9.2.3 Frame Marker Command (TS_FRAME_MARKER) is "
199
1.13k
              "missing frameId, ignoring",
200
1.13k
              Stream_GetRemainingLength(s), 4u);
201
542
  else
202
542
    Stream_Read_UINT32(s, marker.frameId);
203
1.13k
  WLog_Print(up->log, WLOG_DEBUG, "SurfaceFrameMarker: action: %s (%" PRIu32 ") id: %" PRIu32 "",
204
1.13k
             (!marker.frameAction) ? "Begin" : "End", marker.frameAction, marker.frameId);
205
206
1.13k
  if (!update->SurfaceFrameMarker)
207
413
  {
208
413
    WINPR_ASSERT(update->context);
209
413
    if (freerdp_settings_get_bool(update->context->settings, FreeRDP_DeactivateClientDecoding))
210
0
      return TRUE;
211
413
    WLog_ERR(TAG, "Missing callback update->SurfaceFrameMarker");
212
413
    return FALSE;
213
413
  }
214
215
718
  if (!update->SurfaceFrameMarker(update->context, &marker))
216
718
  {
217
718
    WLog_DBG(TAG, "update->SurfaceFrameMarker implementation failed");
218
718
    return FALSE;
219
718
  }
220
221
0
  return TRUE;
222
718
}
223
224
BOOL update_recv_surfcmds(rdpUpdate* update, wStream* s)
225
29.6k
{
226
29.6k
  UINT16 cmdType = 0;
227
29.6k
  rdp_update_internal* up = update_cast(update);
228
229
29.6k
  WINPR_ASSERT(s);
230
231
30.1k
  while (Stream_GetRemainingLength(s) >= 2)
232
18.0k
  {
233
18.0k
    const size_t start = Stream_GetPosition(s);
234
18.0k
    const BYTE* mark = Stream_ConstPointer(s);
235
236
18.0k
    Stream_Read_UINT16(s, cmdType);
237
238
18.0k
    switch (cmdType)
239
18.0k
    {
240
6.04k
      case CMDTYPE_SET_SURFACE_BITS:
241
6.88k
      case CMDTYPE_STREAM_SURFACE_BITS:
242
6.88k
        if (!update_recv_surfcmd_surface_bits(update, s, cmdType))
243
6.34k
          return FALSE;
244
546
        break;
245
246
1.40k
      case CMDTYPE_FRAME_MARKER:
247
1.40k
        up->stats.base[RDP_STATS_SURFACE_FRAME_MARKER]++;
248
1.40k
        if (!update_recv_surfcmd_frame_marker(update, s))
249
1.40k
          return FALSE;
250
251
0
        break;
252
253
9.73k
      default:
254
9.73k
        WLog_ERR(TAG, "unknown cmdType 0x%04" PRIX16 "", cmdType);
255
9.73k
        return FALSE;
256
18.0k
    }
257
258
546
    if (up->dump_rfx)
259
0
    {
260
0
      const size_t size = Stream_GetPosition(s) - start;
261
      /* TODO: treat return values */
262
0
      if (!pcap_add_record(up->pcap_rfx, mark, size))
263
0
        return FALSE;
264
0
      pcap_flush(up->pcap_rfx);
265
0
    }
266
546
  }
267
268
12.1k
  return TRUE;
269
29.6k
}
270
271
static BOOL update_write_surfcmd_bitmap_header_ex(wStream* s,
272
                                                  const TS_COMPRESSED_BITMAP_HEADER_EX* header)
273
128
{
274
128
  if (!s || !header)
275
0
    return FALSE;
276
277
128
  if (!Stream_EnsureRemainingCapacity(s, 24))
278
0
    return FALSE;
279
280
128
  Stream_Write_UINT32(s, header->highUniqueId);
281
128
  Stream_Write_UINT32(s, header->lowUniqueId);
282
128
  Stream_Write_UINT64(s, header->tmMilliseconds);
283
128
  Stream_Write_UINT64(s, header->tmSeconds);
284
128
  return TRUE;
285
128
}
286
287
static BOOL update_write_surfcmd_bitmap_ex(wStream* s, const TS_BITMAP_DATA_EX* bmp)
288
507
{
289
507
  if (!s || !bmp)
290
0
    return FALSE;
291
292
507
  if (!Stream_EnsureRemainingCapacity(s, 12))
293
0
    return FALSE;
294
295
507
  if (bmp->codecID > UINT8_MAX)
296
0
  {
297
0
    WLog_ERR(TAG, "Invalid TS_BITMAP_DATA_EX::codecID=0x%04" PRIx16 "", bmp->codecID);
298
0
    return FALSE;
299
0
  }
300
507
  Stream_Write_UINT8(s, bmp->bpp);
301
507
  Stream_Write_UINT8(s, bmp->flags);
302
507
  Stream_Write_UINT8(s, 0); /* reserved1, reserved2 */
303
507
  Stream_Write_UINT8(s, (UINT8)bmp->codecID);
304
507
  Stream_Write_UINT16(s, bmp->width);
305
507
  Stream_Write_UINT16(s, bmp->height);
306
507
  Stream_Write_UINT32(s, bmp->bitmapDataLength);
307
308
507
  if (bmp->flags & EX_COMPRESSED_BITMAP_HEADER_PRESENT)
309
128
  {
310
128
    if (!update_write_surfcmd_bitmap_header_ex(s, &bmp->exBitmapDataHeader))
311
0
      return FALSE;
312
128
  }
313
314
507
  if (!Stream_EnsureRemainingCapacity(s, bmp->bitmapDataLength))
315
0
    return FALSE;
316
317
507
  Stream_Write(s, bmp->bitmapData, bmp->bitmapDataLength);
318
507
  return TRUE;
319
507
}
320
321
BOOL update_write_surfcmd_surface_bits(wStream* s, const SURFACE_BITS_COMMAND* cmd)
322
507
{
323
507
  if (!Stream_EnsureRemainingCapacity(s, SURFCMD_SURFACE_BITS_HEADER_LENGTH))
324
0
    return FALSE;
325
326
507
  WINPR_ASSERT(cmd->cmdType <= UINT16_MAX);
327
507
  UINT16 cmdType = (UINT16)cmd->cmdType;
328
507
  switch (cmdType)
329
507
  {
330
476
    case CMDTYPE_SET_SURFACE_BITS:
331
507
    case CMDTYPE_STREAM_SURFACE_BITS:
332
507
      break;
333
0
    default:
334
0
    {
335
0
      const UINT16 defaultCmdType = CMDTYPE_STREAM_SURFACE_BITS;
336
0
      WLog_WARN(TAG,
337
0
                "SURFACE_BITS_COMMAND->cmdType 0x%04" PRIx16
338
0
                " not allowed, correcting to 0x%04" PRIx16,
339
0
                cmdType, defaultCmdType);
340
0
      cmdType = defaultCmdType;
341
0
    }
342
0
    break;
343
507
  }
344
345
507
  Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmdType));
346
507
  Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destLeft));
347
507
  Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destTop));
348
507
  Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destRight));
349
507
  Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destBottom));
350
507
  return update_write_surfcmd_bitmap_ex(s, &cmd->bmp);
351
507
}
352
353
BOOL update_write_surfcmd_frame_marker(wStream* s, UINT16 frameAction, UINT32 frameId)
354
713
{
355
713
  if (!Stream_EnsureRemainingCapacity(s, SURFCMD_FRAME_MARKER_LENGTH))
356
0
    return FALSE;
357
358
713
  Stream_Write_UINT16(s, CMDTYPE_FRAME_MARKER);
359
713
  Stream_Write_UINT16(s, frameAction);
360
713
  Stream_Write_UINT32(s, frameId);
361
713
  return TRUE;
362
713
}