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