/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.04k | { |
37 | 2.04k | if (!s || !header) |
38 | 0 | return FALSE; |
39 | | |
40 | 2.04k | if (!Stream_CheckAndLogRequiredLength(TAG, s, 24)) |
41 | 95 | return FALSE; |
42 | | |
43 | 1.94k | Stream_Read_UINT32(s, header->highUniqueId); |
44 | 1.94k | Stream_Read_UINT32(s, header->lowUniqueId); |
45 | 1.94k | Stream_Read_UINT64(s, header->tmMilliseconds); |
46 | 1.94k | Stream_Read_UINT64(s, header->tmSeconds); |
47 | 1.94k | return TRUE; |
48 | 2.04k | } |
49 | | |
50 | | static BOOL update_recv_surfcmd_bitmap_ex(wStream* s, TS_BITMAP_DATA_EX* bmp) |
51 | 3.34k | { |
52 | 3.34k | if (!s || !bmp) |
53 | 0 | return FALSE; |
54 | | |
55 | 3.34k | if (!Stream_CheckAndLogRequiredLength(TAG, s, 12)) |
56 | 275 | return FALSE; |
57 | | |
58 | 3.07k | Stream_Read_UINT8(s, bmp->bpp); |
59 | 3.07k | Stream_Read_UINT8(s, bmp->flags); |
60 | 3.07k | Stream_Seek(s, 1); /* reserved */ |
61 | 3.07k | Stream_Read_UINT8(s, bmp->codecID); |
62 | 3.07k | Stream_Read_UINT16(s, bmp->width); |
63 | 3.07k | Stream_Read_UINT16(s, bmp->height); |
64 | 3.07k | Stream_Read_UINT32(s, bmp->bitmapDataLength); |
65 | | |
66 | 3.07k | if ((bmp->width == 0) || (bmp->height == 0)) |
67 | 134 | { |
68 | 134 | WLog_ERR(TAG, "invalid size value width=%" PRIu16 ", height=%" PRIu16, bmp->width, |
69 | 134 | bmp->height); |
70 | 134 | return FALSE; |
71 | 134 | } |
72 | | |
73 | 2.93k | if ((bmp->bpp < 1) || (bmp->bpp > 32)) |
74 | 339 | { |
75 | 339 | WLog_ERR(TAG, "invalid bpp value %" PRIu32 "", bmp->bpp); |
76 | 339 | return FALSE; |
77 | 339 | } |
78 | | |
79 | 2.60k | if (bmp->flags & EX_COMPRESSED_BITMAP_HEADER_PRESENT) |
80 | 2.04k | { |
81 | 2.04k | if (!update_recv_surfcmd_bitmap_header_ex(s, &bmp->exBitmapDataHeader)) |
82 | 95 | return FALSE; |
83 | 2.04k | } |
84 | | |
85 | 2.50k | bmp->bitmapData = Stream_Pointer(s); |
86 | 2.50k | if (!Stream_SafeSeek(s, bmp->bitmapDataLength)) |
87 | 1.86k | { |
88 | 1.86k | WLog_ERR(TAG, "expected bitmapDataLength %" PRIu32 ", not enough data", |
89 | 1.86k | bmp->bitmapDataLength); |
90 | 1.86k | return FALSE; |
91 | 1.86k | } |
92 | 636 | return TRUE; |
93 | 2.50k | } |
94 | | |
95 | | static BOOL update_recv_surfcmd_is_rect_valid(const rdpContext* context, |
96 | | const SURFACE_BITS_COMMAND* cmd) |
97 | 5.13k | { |
98 | 5.13k | WINPR_ASSERT(context); |
99 | 5.13k | WINPR_ASSERT(context->settings); |
100 | 5.13k | 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 | 5.13k | if ((cmd->destTop >= cmd->destBottom) || (cmd->destLeft >= cmd->destRight)) |
105 | 847 | { |
106 | 847 | WLog_WARN(TAG, |
107 | 847 | "Empty surface bits command rectangle: %" PRIu16 "x%" PRIu16 "-%" PRIu16 |
108 | 847 | "x%" PRIu16, |
109 | 847 | cmd->destLeft, cmd->destTop, cmd->destRight, cmd->destBottom); |
110 | 847 | return FALSE; |
111 | 847 | } |
112 | | |
113 | | /* The rectangle needs to fit into our session size */ |
114 | 4.28k | const DWORD DesktopWidth = freerdp_settings_get_uint32(context->settings, FreeRDP_DesktopWidth); |
115 | 4.28k | const DWORD DesktopHeight = |
116 | 4.28k | freerdp_settings_get_uint32(context->settings, FreeRDP_DesktopHeight); |
117 | 4.28k | if ((cmd->destRight > DesktopWidth) || (cmd->destBottom > DesktopHeight)) |
118 | 940 | { |
119 | 940 | WLog_WARN(TAG, |
120 | 940 | "Invalid surface bits command rectangle: %" PRIu16 "x%" PRIu16 "-%" PRIu16 |
121 | 940 | "x%" PRIu16 " does not fit %" PRIu32 "x%" PRIu32, |
122 | 940 | cmd->destLeft, cmd->destTop, cmd->destRight, cmd->destBottom, DesktopWidth, |
123 | 940 | DesktopHeight); |
124 | 940 | return FALSE; |
125 | 940 | } |
126 | | |
127 | 3.34k | return TRUE; |
128 | 4.28k | } |
129 | | |
130 | | static BOOL update_recv_surfcmd_surface_bits(rdpUpdate* update, wStream* s, UINT16 cmdType) |
131 | 5.36k | { |
132 | 5.36k | BOOL rc = FALSE; |
133 | 5.36k | SURFACE_BITS_COMMAND cmd = WINPR_C_ARRAY_INIT; |
134 | | |
135 | 5.36k | if (!Stream_CheckAndLogRequiredLength(TAG, s, 8)) |
136 | 230 | goto fail; |
137 | | |
138 | 5.13k | cmd.cmdType = cmdType; |
139 | 5.13k | Stream_Read_UINT16(s, cmd.destLeft); |
140 | 5.13k | Stream_Read_UINT16(s, cmd.destTop); |
141 | 5.13k | Stream_Read_UINT16(s, cmd.destRight); |
142 | 5.13k | Stream_Read_UINT16(s, cmd.destBottom); |
143 | | |
144 | 5.13k | if (!update_recv_surfcmd_is_rect_valid(update->context, &cmd)) |
145 | 1.78k | goto fail; |
146 | | |
147 | 3.34k | if (!update_recv_surfcmd_bitmap_ex(s, &cmd.bmp)) |
148 | 2.71k | goto fail; |
149 | | |
150 | 636 | if (!IFCALLRESULT(TRUE, update->SurfaceBits, update->context, &cmd)) |
151 | 174 | { |
152 | 174 | WLog_DBG(TAG, "update->SurfaceBits implementation failed"); |
153 | 174 | goto fail; |
154 | 174 | } |
155 | | |
156 | 462 | rc = TRUE; |
157 | 5.36k | fail: |
158 | 5.36k | return rc; |
159 | 462 | } |
160 | | |
161 | | static BOOL update_recv_surfcmd_frame_marker(rdpUpdate* update, wStream* s) |
162 | 1.04k | { |
163 | 1.04k | SURFACE_FRAME_MARKER marker = WINPR_C_ARRAY_INIT; |
164 | 1.04k | rdp_update_internal* up = update_cast(update); |
165 | | |
166 | 1.04k | WINPR_ASSERT(s); |
167 | | |
168 | 1.04k | if (!Stream_CheckAndLogRequiredLength(TAG, s, 2)) |
169 | 222 | return FALSE; |
170 | | |
171 | 818 | Stream_Read_UINT16(s, marker.frameAction); |
172 | 818 | if (!Stream_CheckAndLogRequiredLength(TAG, s, 4)) |
173 | 344 | WLog_WARN(TAG, |
174 | 818 | "[SERVER-BUG]: got %" PRIuz ", expected %u" |
175 | 818 | " bytes. [MS-RDPBCGR] 2.2.9.2.3 Frame Marker Command (TS_FRAME_MARKER) is " |
176 | 818 | "missing frameId, ignoring", |
177 | 818 | Stream_GetRemainingLength(s), 4u); |
178 | 474 | else |
179 | 474 | Stream_Read_UINT32(s, marker.frameId); |
180 | 818 | WLog_Print(up->log, WLOG_DEBUG, "SurfaceFrameMarker: action: %s (%" PRIu32 ") id: %" PRIu32 "", |
181 | 818 | (!marker.frameAction) ? "Begin" : "End", marker.frameAction, marker.frameId); |
182 | | |
183 | 818 | if (!update->SurfaceFrameMarker) |
184 | 128 | { |
185 | 128 | WINPR_ASSERT(update->context); |
186 | 128 | if (freerdp_settings_get_bool(update->context->settings, FreeRDP_DeactivateClientDecoding)) |
187 | 0 | return TRUE; |
188 | 128 | WLog_ERR(TAG, "Missing callback update->SurfaceFrameMarker"); |
189 | 128 | return FALSE; |
190 | 128 | } |
191 | | |
192 | 690 | if (!update->SurfaceFrameMarker(update->context, &marker)) |
193 | 690 | { |
194 | 690 | WLog_DBG(TAG, "update->SurfaceFrameMarker implementation failed"); |
195 | 690 | return FALSE; |
196 | 690 | } |
197 | | |
198 | 0 | return TRUE; |
199 | 690 | } |
200 | | |
201 | | BOOL update_recv_surfcmds(rdpUpdate* update, wStream* s) |
202 | 31.9k | { |
203 | 31.9k | UINT16 cmdType = 0; |
204 | 31.9k | rdp_update_internal* up = update_cast(update); |
205 | | |
206 | 31.9k | WINPR_ASSERT(s); |
207 | | |
208 | 32.4k | while (Stream_GetRemainingLength(s) >= 2) |
209 | 16.6k | { |
210 | 16.6k | const size_t start = Stream_GetPosition(s); |
211 | 16.6k | const BYTE* mark = Stream_ConstPointer(s); |
212 | | |
213 | 16.6k | Stream_Read_UINT16(s, cmdType); |
214 | | |
215 | 16.6k | switch (cmdType) |
216 | 16.6k | { |
217 | 4.35k | case CMDTYPE_SET_SURFACE_BITS: |
218 | 5.36k | case CMDTYPE_STREAM_SURFACE_BITS: |
219 | 5.36k | if (!update_recv_surfcmd_surface_bits(update, s, cmdType)) |
220 | 4.90k | return FALSE; |
221 | | |
222 | 462 | break; |
223 | | |
224 | 1.04k | case CMDTYPE_FRAME_MARKER: |
225 | 1.04k | if (!update_recv_surfcmd_frame_marker(update, s)) |
226 | 1.04k | return FALSE; |
227 | | |
228 | 0 | break; |
229 | | |
230 | 10.2k | default: |
231 | 10.2k | WLog_ERR(TAG, "unknown cmdType 0x%04" PRIX16 "", cmdType); |
232 | 10.2k | return FALSE; |
233 | 16.6k | } |
234 | | |
235 | 462 | if (up->dump_rfx) |
236 | 0 | { |
237 | 0 | const size_t size = Stream_GetPosition(s) - start; |
238 | | /* TODO: treat return values */ |
239 | 0 | if (!pcap_add_record(up->pcap_rfx, mark, size)) |
240 | 0 | return FALSE; |
241 | 0 | pcap_flush(up->pcap_rfx); |
242 | 0 | } |
243 | 462 | } |
244 | | |
245 | 15.8k | return TRUE; |
246 | 31.9k | } |
247 | | |
248 | | static BOOL update_write_surfcmd_bitmap_header_ex(wStream* s, |
249 | | const TS_COMPRESSED_BITMAP_HEADER_EX* header) |
250 | 75 | { |
251 | 75 | if (!s || !header) |
252 | 0 | return FALSE; |
253 | | |
254 | 75 | if (!Stream_EnsureRemainingCapacity(s, 24)) |
255 | 0 | return FALSE; |
256 | | |
257 | 75 | Stream_Write_UINT32(s, header->highUniqueId); |
258 | 75 | Stream_Write_UINT32(s, header->lowUniqueId); |
259 | 75 | Stream_Write_UINT64(s, header->tmMilliseconds); |
260 | 75 | Stream_Write_UINT64(s, header->tmSeconds); |
261 | 75 | return TRUE; |
262 | 75 | } |
263 | | |
264 | | static BOOL update_write_surfcmd_bitmap_ex(wStream* s, const TS_BITMAP_DATA_EX* bmp) |
265 | 173 | { |
266 | 173 | if (!s || !bmp) |
267 | 0 | return FALSE; |
268 | | |
269 | 173 | if (!Stream_EnsureRemainingCapacity(s, 12)) |
270 | 0 | return FALSE; |
271 | | |
272 | 173 | if (bmp->codecID > UINT8_MAX) |
273 | 0 | { |
274 | 0 | WLog_ERR(TAG, "Invalid TS_BITMAP_DATA_EX::codecID=0x%04" PRIx16 "", bmp->codecID); |
275 | 0 | return FALSE; |
276 | 0 | } |
277 | 173 | Stream_Write_UINT8(s, bmp->bpp); |
278 | 173 | Stream_Write_UINT8(s, bmp->flags); |
279 | 173 | Stream_Write_UINT8(s, 0); /* reserved1, reserved2 */ |
280 | 173 | Stream_Write_UINT8(s, (UINT8)bmp->codecID); |
281 | 173 | Stream_Write_UINT16(s, bmp->width); |
282 | 173 | Stream_Write_UINT16(s, bmp->height); |
283 | 173 | Stream_Write_UINT32(s, bmp->bitmapDataLength); |
284 | | |
285 | 173 | if (bmp->flags & EX_COMPRESSED_BITMAP_HEADER_PRESENT) |
286 | 75 | { |
287 | 75 | if (!update_write_surfcmd_bitmap_header_ex(s, &bmp->exBitmapDataHeader)) |
288 | 0 | return FALSE; |
289 | 75 | } |
290 | | |
291 | 173 | if (!Stream_EnsureRemainingCapacity(s, bmp->bitmapDataLength)) |
292 | 0 | return FALSE; |
293 | | |
294 | 173 | Stream_Write(s, bmp->bitmapData, bmp->bitmapDataLength); |
295 | 173 | return TRUE; |
296 | 173 | } |
297 | | |
298 | | BOOL update_write_surfcmd_surface_bits(wStream* s, const SURFACE_BITS_COMMAND* cmd) |
299 | 173 | { |
300 | 173 | if (!Stream_EnsureRemainingCapacity(s, SURFCMD_SURFACE_BITS_HEADER_LENGTH)) |
301 | 0 | return FALSE; |
302 | | |
303 | 173 | WINPR_ASSERT(cmd->cmdType <= UINT16_MAX); |
304 | 173 | UINT16 cmdType = (UINT16)cmd->cmdType; |
305 | 173 | switch (cmdType) |
306 | 173 | { |
307 | 108 | case CMDTYPE_SET_SURFACE_BITS: |
308 | 173 | case CMDTYPE_STREAM_SURFACE_BITS: |
309 | 173 | break; |
310 | 0 | default: |
311 | 0 | { |
312 | 0 | const UINT16 defaultCmdType = CMDTYPE_STREAM_SURFACE_BITS; |
313 | 0 | WLog_WARN(TAG, |
314 | 0 | "SURFACE_BITS_COMMAND->cmdType 0x%04" PRIx16 |
315 | 0 | " not allowed, correcting to 0x%04" PRIx16, |
316 | 0 | cmdType, defaultCmdType); |
317 | 0 | cmdType = defaultCmdType; |
318 | 0 | } |
319 | 0 | break; |
320 | 173 | } |
321 | | |
322 | 173 | Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmdType)); |
323 | 173 | Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destLeft)); |
324 | 173 | Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destTop)); |
325 | 173 | Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destRight)); |
326 | 173 | Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->destBottom)); |
327 | 173 | return update_write_surfcmd_bitmap_ex(s, &cmd->bmp); |
328 | 173 | } |
329 | | |
330 | | BOOL update_write_surfcmd_frame_marker(wStream* s, UINT16 frameAction, UINT32 frameId) |
331 | 680 | { |
332 | 680 | if (!Stream_EnsureRemainingCapacity(s, SURFCMD_FRAME_MARKER_LENGTH)) |
333 | 0 | return FALSE; |
334 | | |
335 | 680 | Stream_Write_UINT16(s, CMDTYPE_FRAME_MARKER); |
336 | 680 | Stream_Write_UINT16(s, frameAction); |
337 | 680 | Stream_Write_UINT32(s, frameId); |
338 | 680 | return TRUE; |
339 | 680 | } |