/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 | } |