FreeRDP
Loading...
Searching...
No Matches
RDPSession.m
1/*
2 RDP Session object
3
4 Copyright 2013 Thincast Technologies GmbH, Authors: Martin Fleisz, Dorian Johnson
5
6 This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
7 If a copy of the MPL was not distributed with this file, You can obtain one at
8 http://mozilla.org/MPL/2.0/.
9 */
10
11#import "ios_freerdp.h"
12#import "ios_freerdp_ui.h"
13#import "ios_freerdp_events.h"
14
15#import "RDPSession.h"
16#import "RDPCursor.h"
17#import "TSXTypes.h"
18#import "Bookmark.h"
19#import "ConnectionParams.h"
20
21NSString *TSXSessionDidDisconnectNotification = @"TSXSessionDidDisconnect";
22NSString *TSXSessionDidFailToConnectNotification = @"TSXSessionDidFailToConnect";
23
24@interface RDPSession (Private)
25- (void)runSession;
26- (void)runSessionFinished:(NSNumber *)result;
27- (mfInfo *)mfi;
28
29// The connection thread calls these on the main thread.
30- (void)sessionWillConnect;
31- (void)sessionDidConnect;
32- (void)sessionDidDisconnect;
33- (void)sessionDidFailToConnect:(int)reason;
34- (void)sessionBitmapContextWillChange;
35- (void)sessionBitmapContextDidChange;
36@end
37
38@implementation RDPSession
39
40@synthesize delegate = _delegate, params = _params, toolbarVisible = _toolbar_visible,
41 uiRequestCompleted = _ui_request_completed, bookmark = _bookmark;
42
43+ (void)initialize
44{
45 ios_init_freerdp();
46}
47
48static BOOL addArgument(int *argc, char ***argv, const char *fmt, ...)
49{
50 va_list ap = WINPR_C_ARRAY_INIT;
51 char *arg = nullptr;
52 char **tmp = realloc(*argv, (*argc + 1) * sizeof(char *));
53
54 if (!tmp)
55 return FALSE;
56
57 *argv = tmp;
58 *argc = *argc + 1;
59 va_start(ap, fmt);
60 vasprintf(&arg, fmt, ap);
61 va_end(ap);
62 (*argv)[*argc - 1] = arg;
63 return TRUE;
64}
65
66static BOOL addFlag(int *argc, char ***argv, const char *str, BOOL flag)
67{
68 return addArgument(argc, argv, "%s%s", flag ? "+" : "-", str);
69}
70
71static void freeArguments(int argc, char **argv)
72{
73 for (int i = 0; i < argc; i++)
74 free(argv[i]);
75
76 free(argv);
77}
78
79// Designated initializer.
80- (id)initWithBookmark:(ComputerBookmark *)bookmark
81{
82 int status;
83 char **argv = nullptr;
84 int argc = 0;
85
86 if (!(self = [super init]))
87 return nil;
88
89 if (!bookmark)
90 [NSException raise:NSInvalidArgumentException
91 format:@"%s: params may not be nil.", __func__];
92
93 _bookmark = [bookmark retain];
94 _params = [[bookmark params] copy];
95 _name = [[bookmark label] retain];
96 _delegate = nil;
97 _toolbar_visible = YES;
98 _freerdp = ios_freerdp_new();
99 _ui_request_completed = [[NSCondition alloc] init];
100 BOOL connected_via_3g = ![bookmark conntectedViaWLAN];
101
102 if (!addArgument(&argc, &argv, "iFreeRDP"))
103 goto out_free;
104
105 if (!addArgument(&argc, &argv, "/gdi:sw"))
106 goto out_free;
107
108 if (!addArgument(&argc, &argv, "/relax-order-checks"))
109 goto out_free;
110 // Screen Size is set on connect (we need a valid delegate in case the user choose an automatic
111 // screen size)
112
113 // Other simple numeric settings
114 if ([_params hasValueForKey:@"colors"])
115 if (!addArgument(&argc, &argv, "/bpp:%d",
116 [_params intForKey:@"colors" with3GEnabled:connected_via_3g]))
117 goto out_free;
118
119 if ([_params hasValueForKey:@"port"])
120 if (!addArgument(&argc, &argv, "/port:%d", [_params intForKey:@"port"]))
121 goto out_free;
122
123 if ([_params boolForKey:@"console"])
124 if (!addArgument(&argc, &argv, "/admin"))
125 goto out_free;
126
127 if (!addArgument(&argc, &argv, "/v:%s", [_params UTF8StringForKey:@"hostname"]))
128 goto out_free;
129
130 // String settings
131 if ([[_params StringForKey:@"username"] length])
132 {
133 if (!addArgument(&argc, &argv, "/u:%s", [_params UTF8StringForKey:@"username"]))
134 goto out_free;
135 }
136
137 if ([[_params StringForKey:@"password"] length])
138 {
139 if (!addArgument(&argc, &argv, "/p:%s", [_params UTF8StringForKey:@"password"]))
140 goto out_free;
141 }
142
143 if ([[_params StringForKey:@"domain"] length])
144 {
145 if (!addArgument(&argc, &argv, "/d:%s", [_params UTF8StringForKey:@"domain"]))
146 goto out_free;
147 }
148
149 if ([[_params StringForKey:@"working_directory"] length])
150 {
151 if (!addArgument(&argc, &argv, "/shell-dir:%s",
152 [_params UTF8StringForKey:@"working_directory"]))
153 goto out_free;
154 }
155
156 if ([[_params StringForKey:@"remote_program"] length])
157 {
158 if (!addArgument(&argc, &argv, "/shell:%s", [_params UTF8StringForKey:@"remote_program"]))
159 goto out_free;
160 }
161
162 // RemoteFX
163 if ([_params boolForKey:@"perf_remotefx" with3GEnabled:connected_via_3g])
164 if (!addArgument(&argc, &argv, "/rfx"))
165 goto out_free;
166
167 if ([_params boolForKey:@"perf_gfx" with3GEnabled:connected_via_3g])
168 if (!addArgument(&argc, &argv, "/gfx"))
169 goto out_free;
170
171 if ([_params boolForKey:@"perf_h264" with3GEnabled:connected_via_3g])
172 if (!addArgument(&argc, &argv, "/gfx:AVC444"))
173 goto out_free;
174
175 if (![_params boolForKey:@"perf_remotefx" with3GEnabled:connected_via_3g] &&
176 ![_params boolForKey:@"perf_gfx" with3GEnabled:connected_via_3g] &&
177 ![_params boolForKey:@"perf_h264" with3GEnabled:connected_via_3g])
178 if (!addArgument(&argc, &argv, "/nsc"))
179 goto out_free;
180
181 if (!addArgument(&argc, &argv, "/cache:bitmap:on,glyph:on"))
182 goto out_free;
183
184 if (!addFlag(&argc, &argv, "wallpaper",
185 [_params boolForKey:@"perf_show_desktop" with3GEnabled:connected_via_3g]))
186 goto out_free;
187
188 if (!addFlag(&argc, &argv, "window-drag",
189 [_params boolForKey:@"perf_window_dragging" with3GEnabled:connected_via_3g]))
190 goto out_free;
191
192 if (!addFlag(&argc, &argv, "menu-anims",
193 [_params boolForKey:@"perf_menu_animation" with3GEnabled:connected_via_3g]))
194 goto out_free;
195
196 if (!addFlag(&argc, &argv, "themes",
197 [_params boolForKey:@"perf_windows_themes" with3GEnabled:connected_via_3g]))
198 goto out_free;
199
200 if (!addFlag(&argc, &argv, "fonts",
201 [_params boolForKey:@"perf_font_smoothing" with3GEnabled:connected_via_3g]))
202 goto out_free;
203
204 if (!addFlag(&argc, &argv, "aero",
205 [_params boolForKey:@"perf_desktop_composition" with3GEnabled:connected_via_3g]))
206 goto out_free;
207
208 if ([_params hasValueForKey:@"width"])
209 if (!addArgument(&argc, &argv, "/w:%d", [_params intForKey:@"width"]))
210 goto out_free;
211
212 if ([_params hasValueForKey:@"height"])
213 if (!addArgument(&argc, &argv, "/h:%d", [_params intForKey:@"height"]))
214 goto out_free;
215
216 // security
217 switch ([_params intForKey:@"security"])
218 {
219 case TSXProtocolSecurityNLA:
220 if (!addArgument(&argc, &argv, "/sec:NLA"))
221 goto out_free;
222
223 break;
224
225 case TSXProtocolSecurityTLS:
226 if (!addArgument(&argc, &argv, "/sec:TLS"))
227 goto out_free;
228
229 break;
230
231 case TSXProtocolSecurityRDP:
232 if (!addArgument(&argc, &argv, "/sec:RDP"))
233 goto out_free;
234
235 break;
236
237 default:
238 break;
239 }
240
241 // ts gateway settings
242 if ([_params boolForKey:@"enable_tsg_settings"])
243 {
244 if (!addArgument(&argc, &argv, "/gateway:g:%s:%d,u:%s,d:%s,p:%s",
245 [_params UTF8StringForKey:@"tsg_hostname"],
246 [_params intForKey:@"tsg_port"],
247 [_params UTF8StringForKey:@"tsg_username"],
248 [_params UTF8StringForKey:@"tsg_domain"],
249 [_params UTF8StringForKey:@"tsg_password"]))
250 goto out_free;
251 }
252
253 // Remote keyboard layout
254 if (!addArgument(&argc, &argv, "/kbd:layout:%d", 0x409))
255 goto out_free;
256
257 status =
258 freerdp_client_settings_parse_command_line(_freerdp->context->settings, argc, argv, FALSE);
259
260 if (0 != status)
261 goto out_free;
262
263 freeArguments(argc, argv);
264 [self mfi]->session = self;
265 return self;
266out_free:
267 freeArguments(argc, argv);
268 [self release];
269 return nil;
270}
271
272- (void)dealloc
273{
274 [self setDelegate:nil];
275 [_bookmark release];
276 [_name release];
277 [_params release];
278 [_ui_request_completed release];
279 ios_freerdp_free(_freerdp);
280 [super dealloc];
281}
282
283- (CGContextRef)bitmapContext
284{
285 return [self mfi]->bitmap_context;
286}
287
288#pragma mark -
289#pragma mark Connecting and disconnecting
290
291- (void)connect
292{
293 // Set Screen Size to automatic if width or height are still 0
294 rdpSettings *settings = _freerdp->context->settings;
295
296 if (freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth) == 0 ||
297 freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight) == 0)
298 {
299 CGSize size = CGSizeZero;
300
301 if ([[self delegate] respondsToSelector:@selector(sizeForFitScreenForSession:)])
302 size = [[self delegate] sizeForFitScreenForSession:self];
303
304 if (!CGSizeEqualToSize(CGSizeZero, size))
305 {
306 [_params setInt:size.width forKey:@"width"];
307 [_params setInt:size.height forKey:@"height"];
308 (void)freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, size.width);
309 (void)freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, size.height);
310 }
311 }
312
313 // TODO: This is a hack to ensure connections to RDVH with 16bpp don't have an odd screen
314 // resolution width
315 // Otherwise this could result in screen corruption ..
316 if (freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth) <= 16)
317 {
318 const UINT32 w = freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth) & (~1);
319 (void)freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, w);
320 }
321
322 [self performSelectorInBackground:@selector(runSession) withObject:nil];
323}
324
325- (void)disconnect
326{
327 mfInfo *mfi = [self mfi];
328 ios_events_send(mfi, [NSDictionary dictionaryWithObject:@"disconnect" forKey:@"type"]);
329
330 if (mfi->connection_state == TSXConnectionConnecting)
331 {
332 mfi->unwanted = YES;
333 [self sessionDidDisconnect];
334 return;
335 }
336}
337
338- (TSXConnectionState)connectionState
339{
340 return [self mfi]->connection_state;
341}
342
343// suspends the session
344- (void)suspend
345{
346 if (!_suspended)
347 {
348 _suspended = YES;
349 // instance->update->SuppressOutput(instance->context, 0, nullptr);
350 }
351}
352
353// resumes a previously suspended session
354- (void)resume
355{
356 if (_suspended)
357 {
358 /* RECTANGLE_16 rec;
359 rec.left = 0;
360 rec.top = 0;
361 rec.right = freerdp_settings_get_uint32(instance->settings, FreeRDP_DesktopWidth);
362 rec.bottom = freerdp_settings_get_uint32(instance->settings, FreeRDP_DesktopHeight);
363 */
364 _suspended = NO;
365 // instance->update->SuppressOutput(instance->context, 1, &rec);
366 // [delegate sessionScreenSettingsChanged:self];
367 }
368}
369
370// returns YES if the session is started
371- (BOOL)isSuspended
372{
373 return _suspended;
374}
375
376#pragma mark -
377#pragma mark Input events
378
379- (void)sendInputEvent:(NSDictionary *)eventDescriptor
380{
381 if ([self mfi]->connection_state == TSXConnectionConnected)
382 ios_events_send([self mfi], eventDescriptor);
383}
384
385#pragma mark -
386#pragma mark Server events (main thread)
387
388- (void)setNeedsDisplayInRectAsValue:(NSValue *)rect_value
389{
390 if ([[self delegate] respondsToSelector:@selector(session:needsRedrawInRect:)])
391 [[self delegate] session:self needsRedrawInRect:[rect_value CGRectValue]];
392}
393
394- (void)setRemoteCursor:(RDPCursor *)cursor
395{
396 if ([[self delegate] respondsToSelector:@selector(session:didSetRemoteCursor:)])
397 [[self delegate] session:self didSetRemoteCursor:cursor];
398}
399
400- (void)setRemoteCursorPositionValue:(NSValue *)positionValue
401{
402 if ([[self delegate] respondsToSelector:@selector(session:didMoveRemoteCursor:)])
403 [[self delegate] session:self didMoveRemoteCursor:[positionValue CGPointValue]];
404}
405
406- (void)hideRemoteCursor
407{
408 if ([[self delegate] respondsToSelector:@selector(sessionDidHideRemoteCursor:)])
409 [[self delegate] sessionDidHideRemoteCursor:self];
410}
411
412- (void)setDefaultRemoteCursor
413{
414 if ([[self delegate] respondsToSelector:@selector(sessionDidSetDefaultRemoteCursor:)])
415 [[self delegate] sessionDidSetDefaultRemoteCursor:self];
416}
417
418#pragma mark -
419#pragma mark interface functions
420
421- (UIImage *)getScreenshotWithSize:(CGSize)size
422{
423 NSAssert([self mfi]->bitmap_context != nil,
424 @"Screenshot requested while having no valid RDP drawing context");
425 CGImageRef cgImage = CGBitmapContextCreateImage([self mfi]->bitmap_context);
426 UIGraphicsBeginImageContext(size);
427 CGContextTranslateCTM(UIGraphicsGetCurrentContext(), 0, size.height);
428 CGContextScaleCTM(UIGraphicsGetCurrentContext(), 1.0, -1.0);
429 CGContextDrawImage(UIGraphicsGetCurrentContext(), CGRectMake(0, 0, size.width, size.height),
430 cgImage);
431 UIImage *viewImage = UIGraphicsGetImageFromCurrentImageContext();
432 UIGraphicsEndImageContext();
433 CGImageRelease(cgImage);
434 return viewImage;
435}
436
437- (rdpSettings *)getSessionParams
438{
439 return _freerdp->context->settings;
440}
441
442- (NSString *)sessionName
443{
444 return _name;
445}
446
447@end
448
449#pragma mark -
450@implementation RDPSession (Private)
451
452- (mfInfo *)mfi
453{
454 return MFI_FROM_INSTANCE(_freerdp);
455}
456
457// Blocks until rdp session finishes.
458- (void)runSession
459{
460 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
461 // Run the session
462 [self performSelectorOnMainThread:@selector(sessionWillConnect)
463 withObject:nil
464 waitUntilDone:YES];
465 int result_code = ios_run_freerdp(_freerdp);
466 [self mfi]->connection_state = TSXConnectionDisconnected;
467 [self performSelectorOnMainThread:@selector(runSessionFinished:)
468 withObject:[NSNumber numberWithInt:result_code]
469 waitUntilDone:YES];
470 [pool release];
471}
472
473// Main thread.
474- (void)runSessionFinished:(NSNumber *)result
475{
476 int result_code = [result intValue];
477
478 switch (result_code)
479 {
480 case MF_EXIT_CONN_CANCELED:
481 [self sessionDidDisconnect];
482 break;
483
484 case MF_EXIT_LOGON_TIMEOUT:
485 case MF_EXIT_CONN_FAILED:
486 [self sessionDidFailToConnect:result_code];
487 break;
488
489 case MF_EXIT_SUCCESS:
490 default:
491 [self sessionDidDisconnect];
492 break;
493 }
494}
495
496#pragma mark -
497#pragma mark Session management (main thread)
498
499- (void)sessionWillConnect
500{
501 if ([[self delegate] respondsToSelector:@selector(sessionWillConnect:)])
502 [[self delegate] sessionWillConnect:self];
503}
504
505- (void)sessionDidConnect
506{
507 if ([[self delegate] respondsToSelector:@selector(sessionDidConnect:)])
508 [[self delegate] sessionDidConnect:self];
509}
510
511- (void)sessionDidFailToConnect:(int)reason
512{
513 [[NSNotificationCenter defaultCenter]
514 postNotificationName:TSXSessionDidFailToConnectNotification
515 object:self];
516
517 if ([[self delegate] respondsToSelector:@selector(session:didFailToConnect:)])
518 [[self delegate] session:self didFailToConnect:reason];
519}
520
521- (void)sessionDidDisconnect
522{
523 [[NSNotificationCenter defaultCenter] postNotificationName:TSXSessionDidDisconnectNotification
524 object:self];
525
526 if ([[self delegate] respondsToSelector:@selector(sessionDidDisconnect:)])
527 [[self delegate] sessionDidDisconnect:self];
528}
529
530- (void)sessionBitmapContextWillChange
531{
532 if ([[self delegate] respondsToSelector:@selector(sessionBitmapContextWillChange:)])
533 [[self delegate] sessionBitmapContextWillChange:self];
534}
535
536- (void)sessionBitmapContextDidChange
537{
538 if ([[self delegate] respondsToSelector:@selector(sessionBitmapContextDidChange:)])
539 [[self delegate] sessionBitmapContextDidChange:self];
540}
541
542- (void)sessionRequestsAuthenticationWithParams:(NSMutableDictionary *)params
543{
544 if ([[self delegate] respondsToSelector:@selector(session:requestsAuthenticationWithParams:)])
545 [[self delegate] session:self requestsAuthenticationWithParams:params];
546}
547
548- (void)sessionVerifyCertificateWithParams:(NSMutableDictionary *)params
549{
550 if ([[self delegate] respondsToSelector:@selector(session:verifyCertificateWithParams:)])
551 [[self delegate] session:self verifyCertificateWithParams:params];
552}
553
554@end
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_uint32(rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id, UINT32 val)
Sets a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.