void passphrase_read_reads_from_stdin_when_no_tty() { static const int read_nbyte = 11; int stdin_pipe[2], stderr_pipe[2], result_pipe[2]; char read_buf[read_nbyte]; struct sigaction ignore, orig; ignore.sa_handler = SIG_IGN; sigemptyset(&ignore.sa_mask); if (pipe(stdin_pipe) != 0 || pipe(stderr_pipe) != 0 || pipe(result_pipe) !=0) CU_FAIL_FATAL("Could not create pipe"); switch (fork()) { case -1: CU_FAIL_FATAL("Could not fork"); case 0: { static const int password_size = 512; char buffer[password_size]; close(stderr_pipe[0]); close(result_pipe[0]); close(stdin_pipe[1]); if (setsid() == (pid_t) -1) CU_FAIL_FATAL("Could not create new session"); dup2(stdin_pipe[0], STDIN_FILENO); dup2(stderr_pipe[1], STDERR_FILENO); freerdp_passphrase_read("Password: "******"Nothing written to pipe"); sigaction(SIGPIPE, &ignore, &orig); write(stdin_pipe[1], "passw0rd\n", sizeof "passw0rd\n"); sigaction(SIGPIPE, &orig, NULL); if (read(result_pipe[0], read_buf, read_nbyte) == (ssize_t) -1) CU_FAIL_FATAL("Nothing written to pipe"); CU_ASSERT_STRING_EQUAL(read_buf, "passw0rd"); close(stderr_pipe[0]); close(stdin_pipe[1]); return; }
void passphrase_read_prompts_to_tty() { static const int read_nbyte = 11; int masterfd; char* slavedevice = NULL; char read_buf[read_nbyte]; fd_set fd_set_write; masterfd = posix_openpt(O_RDWR|O_NOCTTY); if (masterfd == -1 || grantpt (masterfd) == -1 || unlockpt (masterfd) == -1 || (slavedevice = ptsname (masterfd)) == NULL) CU_FAIL_FATAL("Could not create pty"); switch (fork()) { case -1: CU_FAIL_FATAL("Could not fork"); case 0: { static const int password_size = 512; char buffer[password_size]; int slavefd; if (setsid() == (pid_t) -1) CU_FAIL_FATAL("Could not create new session"); if ((slavefd = open(slavedevice, O_RDWR)) == 0) CU_FAIL_FATAL("Could not open slave end of pty"); close(STDIN_FILENO); close(STDOUT_FILENO); close(STDERR_FILENO); close(masterfd); freerdp_passphrase_read("Password: "******"Master end of pty not writeable"); if (read(masterfd, read_buf, read_nbyte) == (ssize_t) -1) CU_FAIL_FATAL("Nothing written to slave end of pty"); CU_ASSERT_STRING_EQUAL(read_buf, "Password: "******"\n", (size_t) 2); close(masterfd); return; }
void passphrase_read_turns_on_newline_echo_during_read() { static const int read_nbyte = 11; int masterfd, slavefd; char* slavedevice = NULL; char read_buf[read_nbyte]; fd_set fd_set_write; struct termios term_flags; masterfd = posix_openpt(O_RDWR|O_NOCTTY); if (masterfd == -1 || grantpt (masterfd) == -1 || unlockpt (masterfd) == -1 || (slavedevice = ptsname (masterfd)) == NULL) CU_FAIL_FATAL("Could not create pty"); slavefd = open(slavedevice, O_RDWR|O_NOCTTY); if (slavefd == -1) CU_FAIL_FATAL("Could not open slave end of pty"); if (tcgetattr(slavefd, &term_flags) != 0) CU_FAIL_FATAL("Could not get slave pty attributes"); if (term_flags.c_lflag & ECHONL) { term_flags.c_lflag &= ~ECHONL; if (tcsetattr(slavefd, TCSANOW, &term_flags) != 0) CU_FAIL_FATAL("Could not turn ECHO on on slave pty"); } switch (fork()) { case -1: CU_FAIL_FATAL("Could not fork"); case 0: { static const int password_size = 512; int child_slavefd; char buffer[password_size]; if (setsid() == (pid_t) -1) CU_FAIL_FATAL("Could not create new session"); if ((child_slavefd = open(slavedevice, O_RDWR)) == 0) CU_FAIL_FATAL("Could not open slave end of pty"); close(STDIN_FILENO); close(STDOUT_FILENO); close(STDERR_FILENO); close(masterfd); close(slavefd); freerdp_passphrase_read("Password: "******"Master end of pty not writeable"); if (read(masterfd, read_buf, read_nbyte) == (ssize_t) -1) CU_FAIL_FATAL("Nothing written to slave end of pty"); if (tcgetattr(slavefd, &term_flags) != 0) CU_FAIL_FATAL("Could not get slave pty attributes"); CU_ASSERT(term_flags.c_lflag & ECHONL) write(masterfd, "\n", (size_t) 2); close(masterfd); close(slavefd); return; }
/** * Parse command-line arguments and update rdpSettings members accordingly. * @param settings pointer to rdpSettings struct to be updated. * @param argc number of arguments available. * @param argv string array of the arguments. * @param plugin_callback function to be called when a plugin needs to be loaded. * @param plugin_user_data pointer to be passed to the plugin_callback function. * @param ui_callback function to be called when a UI-specific argument is being processed. * @param ui_user_data pointer to be passed to the ui_callback function. * @return number of arguments that were parsed, or FREERDP_ARGS_PARSE_RESULT on failure or --version/--help */ int freerdp_parse_args(rdpSettings* settings, int argc, char** argv, ProcessPluginArgs plugin_callback, void* plugin_user_data, ProcessUIArgs ui_callback, void* ui_user_data) { int t; char* p; int i, j; int index = 1; int num_extensions = 0; RDP_PLUGIN_DATA* plugin_data; while (index < argc) { if ((strcmp("-h", argv[index]) == 0 ) || (strcmp("--help", argv[index]) == 0 )) { printf("\n" "FreeRDP - A Free Remote Desktop Protocol Client\n" "See http://www.freerdp.com for more information\n" "\n" "Usage: %s [options] server:port\n" " -0: connect to console session\n" " -a: set color depth in bit, default is 16\n" " -c: initial working directory\n" " -D: hide window decorations\n" " -T: window title\n" " -d: domain\n" " -f: fullscreen mode\n" " -g: set geometry, using format WxH or X%% or 'workarea', default is 1024x768\n" " -h: print this help\n" " -k: set keyboard layout ID\n" " -K: do not interfere with window manager bindings\n" " -n: hostname\n" " -o: console audio\n" " -p: password\n" " -s: set startup-shell\n" " -t: alternative port number, default is 3389\n" " -u: username\n" " -x: performance flags (m[odem], b[roadband] or l[an])\n" " -X: embed into another window with a given XID.\n" " -z: enable compression\n" " --app: RemoteApp connection. This implies -g workarea\n" " --ext: load an extension\n" " --no-auth: disable authentication\n" " --authonly: authentication only, no UI\n" " --from-stdin: unspecified username, password, domain and hostname params are prompted\n" " --no-fastpath: disable fast-path\n" " --no-motion: don't send mouse motion events\n" " --gdi: graphics rendering (hw, sw)\n" " --no-osb: disable offscreen bitmaps\n" " --no-bmp-cache: disable bitmap cache\n" " --bcv3: codec for bitmap cache v3 (rfx, nsc, jpeg)\n" " --plugin: load a virtual channel plugin\n" " --rfx: enable RemoteFX\n" " --rfx-mode: RemoteFX operational flags (v[ideo], i[mage]), default is video\n" " --frame-ack: number of frames pending to be acknowledged, default is 2 (disable with 0)\n" " --nsc: enable NSCodec (experimental)\n" #ifdef WITH_JPEG " --jpeg: enable jpeg codec, uses 75 quality\n" " --jpegex: enable jpeg and set quality(1..99)\n" #endif " --disable-wallpaper: disables wallpaper\n" " --composition: enable desktop composition\n" " --disable-full-window-drag: disables full window drag\n" " --disable-menu-animations: disables menu animations\n" " --disable-theming: disables theming\n" " --no-nego: disable negotiation of security layer and enforce highest enabled security protocol\n" " --no-rdp: disable Standard RDP encryption\n" " --no-tls: disable TLS encryption\n" " --no-nla: disable network level authentication\n" " --ntlm: force NTLM authentication protocol version (1 or 2)\n" " --ignore-certificate: ignore verification of logon certificate\n" " --certificate-name: use this name for the logon certificate, instead of the server name\n" " --sec: force protocol security (rdp, tls or nla)\n" " --tsg: Terminal Server Gateway (<username> <password> <hostname>)\n" " --kbd-list: list all keyboard layout ids used by -k\n" " --no-salted-checksum: disable salted checksums with Standard RDP encryption\n" " --pcid: preconnection id\n" " --pcb: preconnection blob\n" " --version: print version information\n" "\n", argv[0]); return FREERDP_ARGS_PARSE_HELP; /* TODO: What is the correct return? */ } else if (strcmp("-a", argv[index]) == 0) { index++; if (index == argc) { printf("missing color depth\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->color_depth = atoi(argv[index]); } else if (strcmp("-u", argv[index]) == 0) { index++; if (index == argc) { printf("missing username\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->username = _strdup(argv[index]); } else if (strcmp("-p", argv[index]) == 0) { index++; if (index == argc) { printf("missing password\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->password = _strdup(argv[index]); settings->autologon = 1; /* * Overwrite original password which could be revealed by a simple "ps aux" command. * This approach won't hide the password length, but it is better than nothing. */ memset(argv[index], '*', strlen(argv[index])); } else if (strcmp("-d", argv[index]) == 0) { index++; if (index == argc) { printf("missing domain\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->domain = _strdup(argv[index]); } else if (strcmp("-s", argv[index]) == 0) { index++; if (index == argc) { printf("missing shell\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->shell = _strdup(argv[index]); } else if (strcmp("-c", argv[index]) == 0) { index++; if (index == argc) { printf("missing directory\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->directory = _strdup(argv[index]); } else if (strcmp("-g", argv[index]) == 0) { index++; if (index == argc) { printf("missing dimensions\n"); return FREERDP_ARGS_PARSE_FAILURE; } if (strncmp("workarea", argv[index], 1) == 0) { settings->workarea = TRUE; } else { settings->width = (UINT16) strtol(argv[index], &p, 10); if (*p == 'x') { settings->height = (UINT16) strtol(p + 1, &p, 10); } if (*p == '%') { settings->percent_screen = settings->width; if (settings->percent_screen <= 0 || settings->percent_screen > 100) { printf("invalid geometry percentage\n"); return FREERDP_ARGS_PARSE_FAILURE; } } else { if (ui_callback != NULL) ui_callback(settings, "-g", p, ui_user_data); } } } else if (strcmp("-f", argv[index]) == 0) { settings->fullscreen = TRUE; } else if (strcmp("-D", argv[index]) == 0) { settings->decorations = FALSE; } else if (strcmp("-T", argv[index]) == 0) { index++; if (index == argc) { printf("missing window title\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->window_title = _strdup(argv[index]); } else if (strcmp("-t", argv[index]) == 0) { index++; if (index == argc) { printf("missing port number\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->port = atoi(argv[index]); } else if (strcmp("-k", argv[index]) == 0) { index++; if (index == argc) { printf("missing keyboard layout id\n"); return FREERDP_ARGS_PARSE_FAILURE; } sscanf(argv[index], "%X", &(settings->kbd_layout)); } else if (strcmp("-K", argv[index]) == 0) { settings->grab_keyboard = FALSE; } else if (strcmp("-n", argv[index]) == 0) { index++; if (index == argc) { printf("missing client hostname\n"); return FREERDP_ARGS_PARSE_FAILURE; } strncpy(settings->client_hostname, argv[index], 31); settings->client_hostname[31] = 0; } else if (strcmp("-o", argv[index]) == 0) { settings->console_audio = TRUE; } else if (strcmp("-0", argv[index]) == 0) { settings->console_session = TRUE; } else if (strcmp("-z", argv[index]) == 0) { settings->compression = TRUE; } else if (strcmp("--ntlm", argv[index]) == 0) { index++; settings->ntlm_version = atoi(argv[index]); if (settings->ntlm_version != 2) settings->ntlm_version = 1; } else if (strcmp("--no-glyph-cache", argv[index]) == 0) { settings->glyphSupportLevel = GLYPH_SUPPORT_NONE; } else if (strcmp("--no-osb", argv[index]) == 0) { settings->offscreen_bitmap_cache = FALSE; } else if (strcmp("--no-bmp-cache", argv[index]) == 0) { settings->bitmap_cache = FALSE; } else if (strcmp("--no-auth", argv[index]) == 0) { settings->authentication = FALSE; } else if (strcmp("--authonly", argv[index]) == 0) { settings->authentication_only = TRUE; } else if (strcmp("--from-stdin", argv[index]) == 0) { settings->from_stdin = TRUE; } else if (strcmp("--ignore-certificate", argv[index]) == 0) { settings->ignore_certificate = TRUE; } else if (strcmp("--certificate-name", argv[index]) == 0) { index++; if (index == argc) { printf("missing certificate name\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->certificate_name = _strdup(argv[index]); } else if (strcmp("--no-fastpath", argv[index]) == 0) { settings->fastpath_input = FALSE; settings->fastpath_output = FALSE; } else if (strcmp("--gdi", argv[index]) == 0) { index++; if (index == argc) { printf("missing GDI backend\n"); return FREERDP_ARGS_PARSE_FAILURE; } if (strncmp("sw", argv[index], 1) == 0) /* software */ { settings->sw_gdi = TRUE; } else if (strncmp("hw", argv[index], 1) == 0) /* hardware */ { settings->sw_gdi = FALSE; } else { printf("unknown GDI backend\n"); return FREERDP_ARGS_PARSE_FAILURE; } } else if (strcmp("--bcv3", argv[index]) == 0) { index++; if (index == argc) { printf("missing codec name\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->bitmap_cache_v3 = TRUE; if (strcmp("rfx", argv[index]) == 0) { printf("setting rfx\n"); settings->v3_codec_id = CODEC_ID_REMOTEFX; settings->rfx_codec = TRUE; } else if (strcmp("nsc", argv[index]) == 0) { printf("setting codec nsc\n"); settings->v3_codec_id = CODEC_ID_NSCODEC; settings->ns_codec = TRUE; } #ifdef WITH_JPEG else if (strcmp("jpeg", argv[index]) == 0) { printf("setting codec jpeg\n"); settings->v3_codec_id = CODEC_ID_JPEG; settings->jpeg_codec = TRUE; if (settings->jpeg_quality == 0) settings->jpeg_quality = 75; } #endif else { printf("bad codec name\n"); return FREERDP_ARGS_PARSE_FAILURE; } } #ifdef WITH_JPEG else if (strcmp("--jpeg", argv[index]) == 0) { settings->jpeg_codec = TRUE; settings->jpeg_quality = 75; } else if (strcmp("--jpegex", argv[index]) == 0) { index++; if (index == argc) { printf("missing codec name\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->jpeg_codec = TRUE; settings->jpeg_quality = atoi(argv[index]); } #endif else if (strcmp("--rfx", argv[index]) == 0) { settings->rfx_codec = TRUE; settings->fastpath_output = TRUE; settings->color_depth = 32; settings->performance_flags = PERF_FLAG_NONE; settings->large_pointer = TRUE; } else if (strcmp("--rfx-mode", argv[index]) == 0) { index++; if (index == argc) { printf("missing RemoteFX mode flag\n"); return FREERDP_ARGS_PARSE_FAILURE; } if (argv[index][0] == 'v') /* video */ { settings->rfx_codec_mode = 0x00; } else if (argv[index][0] == 'i') /* image */ { settings->rfx_codec_mode = 0x02; } else { printf("unknown RemoteFX mode flag\n"); return FREERDP_ARGS_PARSE_FAILURE; } } else if (strcmp("--frame-ack", argv[index]) == 0) { index++; if (index == argc) { printf("missing frame acknowledge number\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->frame_acknowledge = atoi(argv[index]); } else if (strcmp("--nsc", argv[index]) == 0) { settings->ns_codec = TRUE; } else if (strcmp("--dump-rfx", argv[index]) == 0) { index++; if (index == argc) { printf("missing file name\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->dump_rfx_file = _strdup(argv[index]); settings->dump_rfx = TRUE; settings->rfx_codec_only = TRUE; } else if (strcmp("--play-rfx", argv[index]) == 0) { index++; if (index == argc) { printf("missing file name\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->play_rfx_file = _strdup(argv[index]); settings->play_rfx = TRUE; } else if (strcmp("--fonts", argv[index]) == 0) { settings->smooth_fonts = TRUE; } else if (strcmp("--disable-wallpaper", argv[index]) == 0) { settings->disable_wallpaper = TRUE; } else if (strcmp("--disable-full-window-drag", argv[index]) == 0) { settings->disable_full_window_drag = TRUE; } else if (strcmp("--disable-menu-animations", argv[index]) == 0) { settings->disable_menu_animations = TRUE; } else if (strcmp("--disable-theming", argv[index]) == 0) { settings->disable_theming = TRUE; } else if (strcmp("--composition", argv[index]) == 0) { settings->desktop_composition = TRUE; } else if (strcmp("--no-motion", argv[index]) == 0) { settings->mouse_motion = FALSE; } else if (strcmp("--app", argv[index]) == 0) { settings->remote_app = TRUE; settings->rail_langbar_supported = TRUE; settings->workarea = TRUE; settings->performance_flags = PERF_DISABLE_WALLPAPER | PERF_DISABLE_FULLWINDOWDRAG; } else if (strcmp("-x", argv[index]) == 0) { index++; if (index == argc) { printf("missing performance flag\n"); return FREERDP_ARGS_PARSE_FAILURE; } if (argv[index][0] == 'm') /* modem */ { settings->performance_flags = PERF_DISABLE_WALLPAPER | PERF_DISABLE_FULLWINDOWDRAG | PERF_DISABLE_MENUANIMATIONS | PERF_DISABLE_THEMING; settings->connection_type = CONNECTION_TYPE_MODEM; } else if (argv[index][0] == 'b') /* broadband */ { settings->performance_flags = PERF_DISABLE_WALLPAPER; settings->connection_type = CONNECTION_TYPE_BROADBAND_HIGH; } else if (argv[index][0] == 'l') /* lan */ { settings->performance_flags = PERF_FLAG_NONE; settings->connection_type = CONNECTION_TYPE_LAN; } else { settings->performance_flags = strtol(argv[index], 0, 16); } } else if (strcmp("-X", argv[index]) == 0) { index++; if (index == argc) { printf("missing parent window XID\n"); return FREERDP_ARGS_PARSE_FAILURE; } settings->parent_window_xid = strtol(argv[index], NULL, 0); if (settings->parent_window_xid == 0) { printf("invalid parent window XID\n"); return FREERDP_ARGS_PARSE_FAILURE; } } else if (strcmp("--no-rdp", argv[index]) == 0) { settings->rdp_security = FALSE; } else if (strcmp("--no-tls", argv[index]) == 0) { settings->tls_security = FALSE; } else if (strcmp("--no-nla", argv[index]) == 0) { settings->nla_security = FALSE; } else if (strcmp("--sec", argv[index]) == 0) { index++; if (index == argc) { printf("missing protocol security\n"); return FREERDP_ARGS_PARSE_FAILURE; } if (strncmp("rdp", argv[index], 1) == 0) /* Standard RDP */ { settings->rdp_security = TRUE; settings->tls_security = FALSE; settings->nla_security = FALSE; settings->encryption = TRUE; settings->encryption_method = ENCRYPTION_METHOD_40BIT | ENCRYPTION_METHOD_128BIT | ENCRYPTION_METHOD_FIPS; settings->encryption_level = ENCRYPTION_LEVEL_CLIENT_COMPATIBLE; } else if (strncmp("tls", argv[index], 1) == 0) /* TLS */ { settings->rdp_security = FALSE; settings->tls_security = TRUE; settings->nla_security = FALSE; } else if (strncmp("nla", argv[index], 1) == 0) /* NLA */ { settings->rdp_security = FALSE; settings->tls_security = FALSE; settings->nla_security = TRUE; } else { printf("unknown protocol security\n"); return FREERDP_ARGS_PARSE_FAILURE; } } else if (strcmp("--no-nego", argv[index]) == 0) { settings->security_layer_negotiation = FALSE; } else if (strcmp("--tsg", argv[index]) == 0) { settings->ts_gateway = TRUE; index++; if (index == argc) { printf("missing TSG username\n"); return -1; } settings->tsg_username = _strdup(argv[index]); index++; if (index == argc) { printf("missing TSG password\n"); return -1; } settings->tsg_password = _strdup(argv[index]); index++; if (index == argc) { printf("missing TSG server\n"); return -1; } settings->tsg_hostname = _strdup(argv[index]); } else if (strcmp("--plugin", argv[index]) == 0) { index++; t = index; if (index == argc) { printf("missing plugin name\n"); return FREERDP_ARGS_PARSE_FAILURE; } plugin_data = NULL; if (strstr(argv[t], "rdpsnd")) settings->audio_playback = TRUE; if (index < argc - 1 && strcmp("--data", argv[index + 1]) == 0) { index += 2; i = 0; while (index < argc && strcmp("--", argv[index]) != 0) { if (plugin_data == NULL) plugin_data = (RDP_PLUGIN_DATA*) malloc(sizeof(RDP_PLUGIN_DATA) * (i + 2)); else plugin_data = (RDP_PLUGIN_DATA*) realloc(plugin_data, sizeof(RDP_PLUGIN_DATA) * (i + 2)); if (strstr(argv[t], "drdynvc") && strstr(argv[index], "audin")) settings->audio_capture = TRUE; plugin_data[i].size = sizeof(RDP_PLUGIN_DATA); plugin_data[i].data[0] = NULL; plugin_data[i].data[1] = NULL; plugin_data[i].data[2] = NULL; plugin_data[i].data[3] = NULL; plugin_data[i + 1].size = 0; for (j = 0, p = argv[index]; j < 4 && p != NULL; j++) { if (*p == '\'') { plugin_data[i].data[j] = p + 1; p = strchr(p + 1, '\''); if (p) *p++ = 0; } else plugin_data[i].data[j] = p; p = strchr(p, ':'); if (p != NULL) *p++ = 0; } index++; i++; } } if (plugin_callback != NULL) { if (!plugin_callback(settings, argv[t], plugin_data, plugin_user_data)) return FREERDP_ARGS_PARSE_FAILURE; } } else if (strcmp("--ext", argv[index]) == 0) { index++; if (index == argc) { printf("missing extension name\n"); return FREERDP_ARGS_PARSE_FAILURE; } if (num_extensions >= ARRAY_SIZE(settings->extensions)) { printf("maximum extensions reached\n"); return FREERDP_ARGS_PARSE_FAILURE; } snprintf(settings->extensions[num_extensions].name, sizeof(settings->extensions[num_extensions].name), "%s", argv[index]); settings->extensions[num_extensions].data = NULL; if (index < argc - 1 && strcmp("--data", argv[index + 1]) == 0) { index += 2; settings->extensions[num_extensions].data = argv[index]; i = 0; while (index < argc && strcmp("--", argv[index]) != 0) { index++; i++; } } num_extensions++; } else if (strcmp("--no-salted-checksum", argv[index]) == 0) { settings->salted_checksum = FALSE; } else if (strcmp("--pcid", argv[index]) == 0) { index++; if (index == argc) { printf("missing preconnection id value\n"); return -1; } settings->send_preconnection_pdu = TRUE; settings->preconnection_id = atoi(argv[index]); } else if (strcmp("--pcb", argv[index]) == 0) { index++; if (index == argc) { printf("missing preconnection blob value\n"); return -1; } settings->send_preconnection_pdu = TRUE; settings->preconnection_blob = _strdup(argv[index]); } else if (strcmp("--version", argv[index]) == 0) { printf("This is FreeRDP version %s (git %s)\n", FREERDP_VERSION_FULL, GIT_REVISION); return FREERDP_ARGS_PARSE_VERSION; } else if (argv[index][0] != '-') { freerdp_parse_hostname(settings, argv[index]); /* server is the last argument for the current session. arguments followed will be parsed for the next session. */ index++; if (settings->smooth_fonts) settings->performance_flags |= PERF_ENABLE_FONT_SMOOTHING; if (settings->desktop_composition) settings->performance_flags |= PERF_ENABLE_DESKTOP_COMPOSITION; if (settings->disable_wallpaper) settings->performance_flags |= PERF_DISABLE_WALLPAPER; if (settings->disable_full_window_drag) settings->performance_flags |= PERF_DISABLE_FULLWINDOWDRAG; if (settings->disable_menu_animations) settings->performance_flags |= PERF_DISABLE_MENUANIMATIONS; if (settings->disable_theming) settings->performance_flags |= PERF_DISABLE_THEMING; break; /* post process missing arguments */ } else { if (ui_callback != NULL) { t = ui_callback(settings, argv[index], (index + 1 < argc && argv[index + 1][0] != '-' ? argv[index + 1] : NULL), ui_user_data); if (t == 0) { printf("invalid option: %s\n", argv[index]); return FREERDP_ARGS_PARSE_FAILURE; } index += t - 1; } } index++; } /* --from-stdin will prompt for missing arguments only. You can prompt for username, password, domain and hostname to avoid disclosing these settings to ps. */ if (settings->from_stdin) { /* username */ if (NULL == settings->username) { char input[512]; input[0] = '\0'; printf("username: "******"%511s%*c", input) > 0) { settings->username = _strdup(input); } } /* password */ if (NULL == settings->password) { settings->password = malloc(512 * sizeof(char)); if (isatty(STDIN_FILENO)) { freerdp_passphrase_read("password: "******"password: "******"%511s%*c", settings->password) <= 0) { free(settings->password); settings->password = NULL; } } } /* domain */ if (NULL == settings->domain) { char input[512]; input[0] = '\0'; printf("domain (control-D to skip): "); if (scanf("%511s%*c", input) > 0) { /* Try to catch the cases where the string is NULL-ish right at the get go */ if (input[0] != '\0' && !(input[0] == '.' && input[1] == '\0')) { settings->domain = _strdup(input); } } if (feof(stdin)) { printf("\n"); clearerr(stdin); } } /* hostname */ if (NULL == settings->hostname) { char input[512]; input[0] = '\0'; printf("hostname: "); if (scanf("%511s%*c", input) > 0) { freerdp_parse_hostname(settings, input); } } } /* Must have a hostname. Do you? */ if (NULL == settings->hostname) { printf("missing server name\n"); return FREERDP_ARGS_PARSE_FAILURE; } else { return index; } }
/** Callback set in the rdp_freerdp structure, and used to get the user's password, * if required to establish the connection. * This function is actually called in credssp_ntlmssp_client_init() * @see rdp_server_accept_nego() and rdp_check_fds() * @param instance - pointer to the rdp_freerdp structure that contains the connection settings * @param username - unused * @param password - on return: pointer to a character string that will be filled by the password entered by the user. * Note that this character string will be allocated inside the function, and needs to be deallocated by the caller * using free(), even in case this function fails. * @param domain - unused * @return TRUE if a password was successfully entered. See freerdp_passphrase_read() for more details. */ static BOOL client_cli_authenticate_raw(freerdp* instance, BOOL gateway, char** username, char** password, char** domain) { static const size_t password_size = 512; const char* auth[] = { "Username: "******"Domain: ", "Password: "******"GatewayUsername: "******"GatewayDomain: ", "GatewayPassword: "******"%s", prompt[0]); if (GetLine(username, &username_size, stdin) < 0) { WLog_ERR(TAG, "GetLine returned %s [%d]", strerror(errno), errno); goto fail; } if (*username) { *username = StrSep(username, "\r"); *username = StrSep(username, "\n"); } } if (!*domain) { size_t domain_size = 0; printf("%s", prompt[1]); if (GetLine(domain, &domain_size, stdin) < 0) { WLog_ERR(TAG, "GetLine returned %s [%d]", strerror(errno), errno); goto fail; } if (*domain) { *domain = StrSep(domain, "\r"); *domain = StrSep(domain, "\n"); } } if (!*password) { *password = calloc(password_size, sizeof(char)); if (!*password) goto fail; if (freerdp_passphrase_read(prompt[2], *password, password_size, instance->settings->CredentialsFromStdin) == NULL) goto fail; } return TRUE; fail: free(*username); free(*domain); free(*password); *username = NULL; *domain = NULL; *password = NULL; return FALSE; }