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