/[rdesktop]/jpeg/rdesktop/trunk/rdesktop.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /jpeg/rdesktop/trunk/rdesktop.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 567 by matthewc, Wed Jan 21 11:08:39 2004 UTC revision 1272 by stargo, Thu Sep 21 19:00:11 2006 UTC
# Line 1  Line 1 
1  /* -*- c-basic-offset: 8 -*-  /* -*- c-basic-offset: 8 -*-
2     rdesktop: A Remote Desktop Protocol client.     rdesktop: A Remote Desktop Protocol client.
3     Entrypoint and utility functions     Entrypoint and utility functions
4     Copyright (C) Matthew Chapman 1999-2004     Copyright (C) Matthew Chapman 1999-2005
5    
6     This program is free software; you can redistribute it and/or modify     This program is free software; you can redistribute it and/or modify
7     it under the terms of the GNU General Public License as published by     it under the terms of the GNU General Public License as published by
# Line 26  Line 26 
26  #include <sys/stat.h>           /* stat */  #include <sys/stat.h>           /* stat */
27  #include <sys/time.h>           /* gettimeofday */  #include <sys/time.h>           /* gettimeofday */
28  #include <sys/times.h>          /* times */  #include <sys/times.h>          /* times */
29    #include <ctype.h>              /* toupper */
30  #include <errno.h>  #include <errno.h>
31  #include "rdesktop.h"  #include "rdesktop.h"
32    
33    #ifdef HAVE_LOCALE_H
34    #include <locale.h>
35    #endif
36    #ifdef HAVE_ICONV
37    #ifdef HAVE_LANGINFO_H
38    #include <langinfo.h>
39    #endif
40    #endif
41    
42  #ifdef EGD_SOCKET  #ifdef EGD_SOCKET
43    #include <sys/types.h>
44  #include <sys/socket.h>         /* socket connect */  #include <sys/socket.h>         /* socket connect */
45  #include <sys/un.h>             /* sockaddr_un */  #include <sys/un.h>             /* sockaddr_un */
46  #endif  #endif
47    
 #ifdef WITH_OPENSSL  
48  #include <openssl/md5.h>  #include <openssl/md5.h>
 #else  
 #include "crypto/md5.h"  
 #endif  
49    
50  char g_title[64] = "";  char g_title[64] = "";
51  char g_username[64];  char g_username[64];
52  char hostname[16];  char g_hostname[16];
53  char keymapname[16];  char g_keymapname[PATH_MAX] = "";
54  int keylayout = 0x409;          /* Defaults to US keyboard layout */  unsigned int g_keylayout = 0x409;       /* Defaults to US keyboard layout */
55    int g_keyboard_type = 0x4;      /* Defaults to US keyboard layout */
56    int g_keyboard_subtype = 0x0;   /* Defaults to US keyboard layout */
57    int g_keyboard_functionkeys = 0xc;      /* Defaults to US keyboard layout */
58    
59  int g_width = 800;              /* width is special: If 0, the  int g_width = 800;              /* width is special: If 0, the
60                                     geometry will be fetched from                                     geometry will be fetched from
# Line 52  int g_width = 800;             /* width is special: Line 62  int g_width = 800;             /* width is special:
62                                     absolute value specifies the                                     absolute value specifies the
63                                     percent of the whole screen. */                                     percent of the whole screen. */
64  int g_height = 600;  int g_height = 600;
65  int tcp_port_rdp = TCP_PORT_RDP;  int g_xpos = 0;
66  int g_server_bpp = 8;  int g_ypos = 0;
67    int g_pos = 0;                  /* 0 position unspecified,
68                                       1 specified,
69                                       2 xpos neg,
70                                       4 ypos neg  */
71    extern int g_tcp_port_rdp;
72    int g_server_depth = -1;
73  int g_win_button_size = 0;      /* If zero, disable single app mode */  int g_win_button_size = 0;      /* If zero, disable single app mode */
74  BOOL g_bitmap_compression = True;  BOOL g_bitmap_compression = True;
75  BOOL g_sendmotion = True;  BOOL g_sendmotion = True;
76  BOOL g_orders = True;  BOOL g_bitmap_cache = True;
77    BOOL g_bitmap_cache_persist_enable = False;
78    BOOL g_bitmap_cache_precache = True;
79  BOOL g_encryption = True;  BOOL g_encryption = True;
80  BOOL packet_encryption = True;  BOOL packet_encryption = True;
81  BOOL g_desktop_save = True;  BOOL g_desktop_save = True;     /* desktop save order */
82    BOOL g_polygon_ellipse_orders = True;   /* polygon / ellipse orders */
83  BOOL g_fullscreen = False;  BOOL g_fullscreen = False;
84  BOOL g_grab_keyboard = True;  BOOL g_grab_keyboard = True;
85  BOOL g_hide_decorations = False;  BOOL g_hide_decorations = False;
86  BOOL g_use_rdp5 = True;  BOOL g_use_rdp5 = True;
87    BOOL g_rdpclip = True;
88  BOOL g_console_session = False;  BOOL g_console_session = False;
89  BOOL g_numlock_sync = False;  BOOL g_numlock_sync = False;
90  extern BOOL g_owncolmap;  BOOL lspci_enabled = False;
91    BOOL g_owncolmap = False;
92    BOOL g_ownbackstore = True;     /* We can't rely on external BackingStore */
93    BOOL g_seamless_rdp = False;
94    uint32 g_embed_wnd;
95    uint32 g_rdp5_performanceflags =
96            RDP5_NO_WALLPAPER | RDP5_NO_FULLWINDOWDRAG | RDP5_NO_MENUANIMATIONS;
97    /* Session Directory redirection */
98    BOOL g_redirect = False;
99    char g_redirect_server[64];
100    char g_redirect_domain[16];
101    char g_redirect_password[64];
102    char g_redirect_username[64];
103    char g_redirect_cookie[128];
104    uint32 g_redirect_flags = 0;
105    
106  #ifdef WITH_RDPSND  #ifdef WITH_RDPSND
107  BOOL g_rdpsnd = False;  BOOL g_rdpsnd = False;
108  #endif  #endif
109    
110    #ifdef HAVE_ICONV
111    char g_codepage[16] = "";
112    #endif
113    
114    extern RDPDR_DEVICE g_rdpdr_device[];
115    extern uint32 g_num_devices;
116    extern char *g_rdpdr_clientname;
117    
118  #ifdef RDP2VNC  #ifdef RDP2VNC
119  extern int rfb_port;  extern int rfb_port;
120  extern int defer_time;  extern int defer_time;
# Line 85  static void Line 127  static void
127  usage(char *program)  usage(char *program)
128  {  {
129          fprintf(stderr, "rdesktop: A Remote Desktop Protocol client.\n");          fprintf(stderr, "rdesktop: A Remote Desktop Protocol client.\n");
130          fprintf(stderr, "Version " VERSION ". Copyright (C) 1999-2003 Matt Chapman.\n");          fprintf(stderr, "Version " VERSION ". Copyright (C) 1999-2005 Matt Chapman.\n");
131          fprintf(stderr, "See http://www.rdesktop.org/ for more information.\n\n");          fprintf(stderr, "See http://www.rdesktop.org/ for more information.\n\n");
132    
133          fprintf(stderr, "Usage: %s [options] server[:port]\n", program);          fprintf(stderr, "Usage: %s [options] server[:port]\n", program);
# Line 103  usage(char *program) Line 145  usage(char *program)
145          fprintf(stderr, "   -g: desktop geometry (WxH)\n");          fprintf(stderr, "   -g: desktop geometry (WxH)\n");
146          fprintf(stderr, "   -f: full-screen mode\n");          fprintf(stderr, "   -f: full-screen mode\n");
147          fprintf(stderr, "   -b: force bitmap updates\n");          fprintf(stderr, "   -b: force bitmap updates\n");
148    #ifdef HAVE_ICONV
149            fprintf(stderr, "   -L: local codepage\n");
150    #endif
151            fprintf(stderr, "   -A: enable SeamlessRDP mode\n");
152            fprintf(stderr, "   -B: use BackingStore of X-server (if available)\n");
153          fprintf(stderr, "   -e: disable encryption (French TS)\n");          fprintf(stderr, "   -e: disable encryption (French TS)\n");
154          fprintf(stderr, "   -E: disable encryption from client to server\n");          fprintf(stderr, "   -E: disable encryption from client to server\n");
155          fprintf(stderr, "   -m: do not send motion events\n");          fprintf(stderr, "   -m: do not send motion events\n");
# Line 111  usage(char *program) Line 158  usage(char *program)
158          fprintf(stderr, "   -K: keep window manager key bindings\n");          fprintf(stderr, "   -K: keep window manager key bindings\n");
159          fprintf(stderr, "   -S: caption button size (single application mode)\n");          fprintf(stderr, "   -S: caption button size (single application mode)\n");
160          fprintf(stderr, "   -T: window title\n");          fprintf(stderr, "   -T: window title\n");
161          fprintf(stderr, "   -N: enable numlock synchronisation\n");          fprintf(stderr, "   -N: enable numlock syncronization\n");
162            fprintf(stderr, "   -X: embed into another window with a given id.\n");
163          fprintf(stderr, "   -a: connection colour depth\n");          fprintf(stderr, "   -a: connection colour depth\n");
164          fprintf(stderr, "   -r: enable specified device redirection (currently: sound)\n");          fprintf(stderr, "   -z: enable rdp compression\n");
165            fprintf(stderr, "   -x: RDP5 experience (m[odem 28.8], b[roadband], l[an] or hex nr.)\n");
166            fprintf(stderr, "   -P: use persistent bitmap caching\n");
167            fprintf(stderr, "   -r: enable specified device redirection (this flag can be repeated)\n");
168            fprintf(stderr,
169                    "         '-r comport:COM1=/dev/ttyS0': enable serial redirection of /dev/ttyS0 to COM1\n");
170            fprintf(stderr, "             or      COM1=/dev/ttyS0,COM2=/dev/ttyS1\n");
171            fprintf(stderr,
172                    "         '-r disk:floppy=/mnt/floppy': enable redirection of /mnt/floppy to 'floppy' share\n");
173            fprintf(stderr, "             or   'floppy=/mnt/floppy,cdrom=/mnt/cdrom'\n");
174            fprintf(stderr, "         '-r clientname=<client name>': Set the client name displayed\n");
175            fprintf(stderr, "             for redirected disks\n");
176            fprintf(stderr,
177                    "         '-r lptport:LPT1=/dev/lp0': enable parallel redirection of /dev/lp0 to LPT1\n");
178            fprintf(stderr, "             or      LPT1=/dev/lp0,LPT2=/dev/lp1\n");
179            fprintf(stderr, "         '-r printer:mydeskjet': enable printer redirection\n");
180            fprintf(stderr,
181                    "             or      mydeskjet=\"HP LaserJet IIIP\" to enter server driver as well\n");
182    #ifdef WITH_RDPSND
183            fprintf(stderr,
184                    "         '-r sound:[local[:driver[:device]]|off|remote]': enable sound redirection\n");
185            fprintf(stderr, "                     remote would leave sound on server\n");
186            fprintf(stderr, "                     available drivers for 'local':\n");
187            rdpsnd_show_help();
188    #endif
189            fprintf(stderr,
190                    "         '-r clipboard:[off|PRIMARYCLIPBOARD|CLIPBOARD]': enable clipboard\n");
191            fprintf(stderr, "                      redirection.\n");
192            fprintf(stderr,
193                    "                      'PRIMARYCLIPBOARD' looks at both PRIMARY and CLIPBOARD\n");
194            fprintf(stderr, "                      when sending data to server.\n");
195            fprintf(stderr, "                      'CLIPBOARD' looks at only CLIPBOARD.\n");
196          fprintf(stderr, "   -0: attach to console\n");          fprintf(stderr, "   -0: attach to console\n");
197          fprintf(stderr, "   -4: use RDP version 4\n");          fprintf(stderr, "   -4: use RDP version 4\n");
198          fprintf(stderr, "   -5: use RDP version 5 (default)\n");          fprintf(stderr, "   -5: use RDP version 5 (default)\n");
199  }  }
200    
201    static void
202    print_disconnect_reason(uint16 reason)
203    {
204            char *text;
205    
206            switch (reason)
207            {
208                    case exDiscReasonNoInfo:
209                            text = "No information available";
210                            break;
211    
212                    case exDiscReasonAPIInitiatedDisconnect:
213                            text = "Server initiated disconnect";
214                            break;
215    
216                    case exDiscReasonAPIInitiatedLogoff:
217                            text = "Server initiated logoff";
218                            break;
219    
220                    case exDiscReasonServerIdleTimeout:
221                            text = "Server idle timeout reached";
222                            break;
223    
224                    case exDiscReasonServerLogonTimeout:
225                            text = "Server logon timeout reached";
226                            break;
227    
228                    case exDiscReasonReplacedByOtherConnection:
229                            text = "The session was replaced";
230                            break;
231    
232                    case exDiscReasonOutOfMemory:
233                            text = "The server is out of memory";
234                            break;
235    
236                    case exDiscReasonServerDeniedConnection:
237                            text = "The server denied the connection";
238                            break;
239    
240                    case exDiscReasonServerDeniedConnectionFips:
241                            text = "The server denied the connection for security reason";
242                            break;
243    
244                    case exDiscReasonLicenseInternal:
245                            text = "Internal licensing error";
246                            break;
247    
248                    case exDiscReasonLicenseNoLicenseServer:
249                            text = "No license server available";
250                            break;
251    
252                    case exDiscReasonLicenseNoLicense:
253                            text = "No valid license available";
254                            break;
255    
256                    case exDiscReasonLicenseErrClientMsg:
257                            text = "Invalid licensing message";
258                            break;
259    
260                    case exDiscReasonLicenseHwidDoesntMatchLicense:
261                            text = "Hardware id doesn't match software license";
262                            break;
263    
264                    case exDiscReasonLicenseErrClientLicense:
265                            text = "Client license error";
266                            break;
267    
268                    case exDiscReasonLicenseCantFinishProtocol:
269                            text = "Network error during licensing protocol";
270                            break;
271    
272                    case exDiscReasonLicenseClientEndedProtocol:
273                            text = "Licensing protocol was not completed";
274                            break;
275    
276                    case exDiscReasonLicenseErrClientEncryption:
277                            text = "Incorrect client license enryption";
278                            break;
279    
280                    case exDiscReasonLicenseCantUpgradeLicense:
281                            text = "Can't upgrade license";
282                            break;
283    
284                    case exDiscReasonLicenseNoRemoteConnections:
285                            text = "The server is not licensed to accept remote connections";
286                            break;
287    
288                    default:
289                            if (reason > 0x1000 && reason < 0x7fff)
290                            {
291                                    text = "Internal protocol error";
292                            }
293                            else
294                            {
295                                    text = "Unknown reason";
296                            }
297            }
298            fprintf(stderr, "disconnect: %s.\n", text);
299    }
300    
301    static void
302    rdesktop_reset_state(void)
303    {
304            rdp_reset_state();
305    }
306    
307  static BOOL  static BOOL
308  read_password(char *password, int size)  read_password(char *password, int size)
309  {  {
# Line 176  parse_server_and_port(char *server) Line 361  parse_server_and_port(char *server)
361                  if (*server == '[' && p != NULL)                  if (*server == '[' && p != NULL)
362                  {                  {
363                          if (*(p + 1) == ':' && *(p + 2) != '\0')                          if (*(p + 1) == ':' && *(p + 2) != '\0')
364                                  tcp_port_rdp = strtol(p + 2, NULL, 10);                                  g_tcp_port_rdp = strtol(p + 2, NULL, 10);
365                          /* remove the port number and brackets from the address */                          /* remove the port number and brackets from the address */
366                          *p = '\0';                          *p = '\0';
367                          strncpy(server, server + 1, strlen(server));                          strncpy(server, server + 1, strlen(server));
# Line 188  parse_server_and_port(char *server) Line 373  parse_server_and_port(char *server)
373                  p = strchr(server, ':');                  p = strchr(server, ':');
374                  if (p != NULL)                  if (p != NULL)
375                  {                  {
376                          tcp_port_rdp = strtol(p + 1, NULL, 10);                          g_tcp_port_rdp = strtol(p + 1, NULL, 10);
377                          *p = 0;                          *p = 0;
378                  }                  }
379          }          }
# Line 196  parse_server_and_port(char *server) Line 381  parse_server_and_port(char *server)
381          p = strchr(server, ':');          p = strchr(server, ':');
382          if (p != NULL)          if (p != NULL)
383          {          {
384                  tcp_port_rdp = strtol(p + 1, NULL, 10);                  g_tcp_port_rdp = strtol(p + 1, NULL, 10);
385                  *p = 0;                  *p = 0;
386          }          }
387  #endif /* IPv6 */  #endif /* IPv6 */
# Line 211  main(int argc, char *argv[]) Line 396  main(int argc, char *argv[])
396          char fullhostname[64];          char fullhostname[64];
397          char domain[16];          char domain[16];
398          char password[64];          char password[64];
399          char shell[128];          char shell[256];
400          char directory[32];          char directory[256];
401          BOOL prompt_password, rdp_retval = False;          BOOL prompt_password, deactivated;
402          struct passwd *pw;          struct passwd *pw;
403          uint32 flags;          uint32 flags, ext_disc_reason = 0;
404          char *p;          char *p;
405          int c;          int c;
406            char *locale = NULL;
407          int username_option = 0;          int username_option = 0;
408            BOOL geometry_option = False;
409            int run_count = 0;      /* Session Directory support */
410            BOOL continue_connect = True;   /* Session Directory support */
411            char *rdpsnd_optarg = NULL;
412    
413    #ifdef HAVE_LOCALE_H
414            /* Set locale according to environment */
415            locale = setlocale(LC_ALL, "");
416            if (locale)
417            {
418                    locale = xstrdup(locale);
419            }
420    
421    #endif
422          flags = RDP_LOGON_NORMAL;          flags = RDP_LOGON_NORMAL;
423          prompt_password = False;          prompt_password = False;
424          domain[0] = password[0] = shell[0] = directory[0] = 0;          domain[0] = password[0] = shell[0] = directory[0] = 0;
425          strcpy(keymapname, "en-us");          g_embed_wnd = 0;
426    
427            g_num_devices = 0;
428    
429  #ifdef RDP2VNC  #ifdef RDP2VNC
430  #define VNCOPT "V:Q:"  #define VNCOPT "V:Q:"
# Line 231  main(int argc, char *argv[]) Line 432  main(int argc, char *argv[])
432  #define VNCOPT  #define VNCOPT
433  #endif  #endif
434    
435          while ((c = getopt(argc, argv, VNCOPT "u:d:s:c:p:n:k:g:fbeEmCDKS:T:Na:r:045h?")) != -1)          while ((c = getopt(argc, argv,
436                               VNCOPT "Au:L:d:s:c:p:n:k:g:fbBeEmzCDKS:T:NX:a:x:Pr:045h?")) != -1)
437          {          {
438                  switch (c)                  switch (c)
439                  {                  {
# Line 249  main(int argc, char *argv[]) Line 451  main(int argc, char *argv[])
451                                  break;                                  break;
452  #endif  #endif
453    
454                            case 'A':
455                                    g_seamless_rdp = True;
456                                    break;
457    
458                          case 'u':                          case 'u':
459                                  STRNCPY(g_username, optarg, sizeof(g_username));                                  STRNCPY(g_username, optarg, sizeof(g_username));
460                                  username_option = 1;                                  username_option = 1;
461                                  break;                                  break;
462    
463                            case 'L':
464    #ifdef HAVE_ICONV
465                                    STRNCPY(g_codepage, optarg, sizeof(g_codepage));
466    #else
467                                    error("iconv support not available\n");
468    #endif
469                                    break;
470    
471                          case 'd':                          case 'd':
472                                  STRNCPY(domain, optarg, sizeof(domain));                                  STRNCPY(domain, optarg, sizeof(domain));
473                                  break;                                  break;
# Line 283  main(int argc, char *argv[]) Line 497  main(int argc, char *argv[])
497                                  break;                                  break;
498    
499                          case 'n':                          case 'n':
500                                  STRNCPY(hostname, optarg, sizeof(hostname));                                  STRNCPY(g_hostname, optarg, sizeof(g_hostname));
501                                  break;                                  break;
502    
503                          case 'k':                          case 'k':
504                                  STRNCPY(keymapname, optarg, sizeof(keymapname));                                  STRNCPY(g_keymapname, optarg, sizeof(g_keymapname));
505                                  break;                                  break;
506    
507                          case 'g':                          case 'g':
508                                    geometry_option = True;
509                                  g_fullscreen = False;                                  g_fullscreen = False;
510                                  if (!strcmp(optarg, "workarea"))                                  if (!strcmp(optarg, "workarea"))
511                                  {                                  {
# Line 306  main(int argc, char *argv[]) Line 521  main(int argc, char *argv[])
521                                  }                                  }
522    
523                                  if (*p == 'x')                                  if (*p == 'x')
524                                          g_height = strtol(p + 1, NULL, 10);                                          g_height = strtol(p + 1, &p, 10);
525    
526                                  if (g_height <= 0)                                  if (g_height <= 0)
527                                  {                                  {
# Line 315  main(int argc, char *argv[]) Line 530  main(int argc, char *argv[])
530                                  }                                  }
531    
532                                  if (*p == '%')                                  if (*p == '%')
533                                    {
534                                          g_width = -g_width;                                          g_width = -g_width;
535                                            p++;
536                                    }
537    
538                                    if (*p == '+' || *p == '-')
539                                    {
540                                            g_pos |= (*p == '-') ? 2 : 1;
541                                            g_xpos = strtol(p, &p, 10);
542    
543                                    }
544                                    if (*p == '+' || *p == '-')
545                                    {
546                                            g_pos |= (*p == '-') ? 4 : 1;
547                                            g_ypos = strtol(p, NULL, 10);
548                                    }
549    
550                                  break;                                  break;
551    
# Line 324  main(int argc, char *argv[]) Line 554  main(int argc, char *argv[])
554                                  break;                                  break;
555    
556                          case 'b':                          case 'b':
557                                  g_orders = False;                                  g_bitmap_cache = False;
558                                    break;
559    
560                            case 'B':
561                                    g_ownbackstore = False;
562                                  break;                                  break;
563    
564                          case 'e':                          case 'e':
# Line 374  main(int argc, char *argv[]) Line 608  main(int argc, char *argv[])
608                                  g_numlock_sync = True;                                  g_numlock_sync = True;
609                                  break;                                  break;
610    
611                            case 'X':
612                                    g_embed_wnd = strtol(optarg, NULL, 0);
613                                    break;
614    
615                          case 'a':                          case 'a':
616                                  g_server_bpp = strtol(optarg, NULL, 10);                                  g_server_depth = strtol(optarg, NULL, 10);
617                                  if (g_server_bpp != 8 && g_server_bpp != 16 && g_server_bpp != 15                                  if (g_server_depth != 8 &&
618                                      && g_server_bpp != 24)                                      g_server_depth != 16 &&
619                                        g_server_depth != 15 && g_server_depth != 24)
620                                  {                                  {
621                                          error("invalid server bpp\n");                                          error("Invalid server colour depth.\n");
622                                          return 1;                                          return 1;
623                                  }                                  }
624                                  break;                                  break;
625    
626                            case 'z':
627                                    DEBUG(("rdp compression enabled\n"));
628                                    flags |= (RDP_LOGON_COMPRESSION | RDP_LOGON_COMPRESSION2);
629                                    break;
630    
631                            case 'x':
632                                    if (str_startswith(optarg, "m"))        /* modem */
633                                    {
634                                            g_rdp5_performanceflags =
635                                                    RDP5_NO_WALLPAPER | RDP5_NO_FULLWINDOWDRAG |
636                                                    RDP5_NO_MENUANIMATIONS | RDP5_NO_THEMING;
637                                    }
638                                    else if (str_startswith(optarg, "b"))   /* broadband */
639                                    {
640                                            g_rdp5_performanceflags = RDP5_NO_WALLPAPER;
641                                    }
642                                    else if (str_startswith(optarg, "l"))   /* lan */
643                                    {
644                                            g_rdp5_performanceflags = RDP5_DISABLE_NOTHING;
645                                    }
646                                    else
647                                    {
648                                            g_rdp5_performanceflags = strtol(optarg, NULL, 16);
649                                    }
650                                    break;
651    
652                            case 'P':
653                                    g_bitmap_cache_persist_enable = True;
654                                    break;
655    
656                          case 'r':                          case 'r':
657                                  if (!strcmp(optarg, "sound"))  
658                                    if (str_startswith(optarg, "sound"))
659                                    {
660                                            optarg += 5;
661    
662                                            if (*optarg == ':')
663                                            {
664                                                    optarg++;
665                                                    while ((p = next_arg(optarg, ',')))
666                                                    {
667                                                            if (str_startswith(optarg, "remote"))
668                                                                    flags |= RDP_LOGON_LEAVE_AUDIO;
669    
670                                                            if (str_startswith(optarg, "local"))
671  #ifdef WITH_RDPSND  #ifdef WITH_RDPSND
672                                          g_rdpsnd = True;                                                          {
673                                                                    rdpsnd_optarg =
674                                                                            next_arg(optarg, ':');
675                                                                    g_rdpsnd = True;
676                                                            }
677    
678  #else  #else
679                                          warning("Not compiled with sound support");                                                                  warning("Not compiled with sound support\n");
680  #endif  #endif
681    
682                                                            if (str_startswith(optarg, "off"))
683    #ifdef WITH_RDPSND
684                                                                    g_rdpsnd = False;
685    #else
686                                                                    warning("Not compiled with sound support\n");
687    #endif
688    
689                                                            optarg = p;
690                                                    }
691                                            }
692                                            else
693                                            {
694    #ifdef WITH_RDPSND
695                                                    g_rdpsnd = True;
696    #else
697                                                    warning("Not compiled with sound support\n");
698    #endif
699                                            }
700                                    }
701                                    else if (str_startswith(optarg, "disk"))
702                                    {
703                                            /* -r disk:h:=/mnt/floppy */
704                                            disk_enum_devices(&g_num_devices, optarg + 4);
705                                    }
706                                    else if (str_startswith(optarg, "comport"))
707                                    {
708                                            serial_enum_devices(&g_num_devices, optarg + 7);
709                                    }
710                                    else if (str_startswith(optarg, "lspci"))
711                                    {
712                                            lspci_enabled = True;
713                                    }
714                                    else if (str_startswith(optarg, "lptport"))
715                                    {
716                                            parallel_enum_devices(&g_num_devices, optarg + 7);
717                                    }
718                                    else if (str_startswith(optarg, "printer"))
719                                    {
720                                            printer_enum_devices(&g_num_devices, optarg + 7);
721                                    }
722                                    else if (str_startswith(optarg, "clientname"))
723                                    {
724                                            g_rdpdr_clientname = xmalloc(strlen(optarg + 11) + 1);
725                                            strcpy(g_rdpdr_clientname, optarg + 11);
726                                    }
727                                    else if (str_startswith(optarg, "clipboard"))
728                                    {
729                                            optarg += 9;
730    
731                                            if (*optarg == ':')
732                                            {
733                                                    optarg++;
734    
735                                                    if (str_startswith(optarg, "off"))
736                                                            g_rdpclip = False;
737                                                    else
738                                                            cliprdr_set_mode(optarg);
739                                            }
740                                            else
741                                                    g_rdpclip = True;
742                                    }
743                                    else
744                                    {
745                                            warning("Unknown -r argument\n\n\tPossible arguments are: comport, disk, lptport, printer, sound, clipboard\n");
746                                    }
747                                  break;                                  break;
748    
749                          case '0':                          case '0':
# Line 413  main(int argc, char *argv[]) Line 766  main(int argc, char *argv[])
766                  }                  }
767          }          }
768    
769          if (argc - optind < 1)          if (argc - optind != 1)
770          {          {
771                  usage(argv[0]);                  usage(argv[0]);
772                  return 1;                  return 1;
# Line 422  main(int argc, char *argv[]) Line 775  main(int argc, char *argv[])
775          STRNCPY(server, argv[optind], sizeof(server));          STRNCPY(server, argv[optind], sizeof(server));
776          parse_server_and_port(server);          parse_server_and_port(server);
777    
778            if (g_seamless_rdp)
779            {
780                    if (g_win_button_size)
781                    {
782                            error("You cannot use -S and -A at the same time\n");
783                            return 1;
784                    }
785                    g_rdp5_performanceflags &= ~RDP5_NO_FULLWINDOWDRAG;
786                    if (geometry_option)
787                    {
788                            error("You cannot use -g and -A at the same time\n");
789                            return 1;
790                    }
791                    if (g_fullscreen)
792                    {
793                            error("You cannot use -f and -A at the same time\n");
794                            return 1;
795                    }
796                    if (g_hide_decorations)
797                    {
798                            error("You cannot use -D and -A at the same time\n");
799                            return 1;
800                    }
801                    if (g_embed_wnd)
802                    {
803                            error("You cannot use -X and -A at the same time\n");
804                            return 1;
805                    }
806                    if (!g_use_rdp5)
807                    {
808                            error("You cannot use -4 and -A at the same time\n");
809                            return 1;
810                    }
811                    g_width = -100;
812                    g_grab_keyboard = False;
813            }
814    
815          if (!username_option)          if (!username_option)
816          {          {
817                  pw = getpwuid(getuid());                  pw = getpwuid(getuid());
# Line 434  main(int argc, char *argv[]) Line 824  main(int argc, char *argv[])
824                  STRNCPY(g_username, pw->pw_name, sizeof(g_username));                  STRNCPY(g_username, pw->pw_name, sizeof(g_username));
825          }          }
826    
827          if (hostname[0] == 0)  #ifdef HAVE_ICONV
828            if (g_codepage[0] == 0)
829            {
830                    if (setlocale(LC_CTYPE, ""))
831                    {
832                            STRNCPY(g_codepage, nl_langinfo(CODESET), sizeof(g_codepage));
833                    }
834                    else
835                    {
836                            STRNCPY(g_codepage, DEFAULT_CODEPAGE, sizeof(g_codepage));
837                    }
838            }
839    #endif
840    
841            if (g_hostname[0] == 0)
842          {          {
843                  if (gethostname(fullhostname, sizeof(fullhostname)) == -1)                  if (gethostname(fullhostname, sizeof(fullhostname)) == -1)
844                  {                  {
# Line 446  main(int argc, char *argv[]) Line 850  main(int argc, char *argv[])
850                  if (p != NULL)                  if (p != NULL)
851                          *p = 0;                          *p = 0;
852    
853                  STRNCPY(hostname, fullhostname, sizeof(hostname));                  STRNCPY(g_hostname, fullhostname, sizeof(g_hostname));
854          }          }
855    
856            if (g_keymapname[0] == 0)
857            {
858                    if (locale && xkeymap_from_locale(locale))
859                    {
860                            fprintf(stderr, "Autoselected keyboard map %s\n", g_keymapname);
861                    }
862                    else
863                    {
864                            STRNCPY(g_keymapname, "en-us", sizeof(g_keymapname));
865                    }
866            }
867            if (locale)
868                    xfree(locale);
869    
870    
871          if (prompt_password && read_password(password, sizeof(password)))          if (prompt_password && read_password(password, sizeof(password)))
872                  flags |= RDP_LOGON_AUTO;                  flags |= RDP_LOGON_AUTO;
873    
# Line 468  main(int argc, char *argv[]) Line 887  main(int argc, char *argv[])
887    
888  #ifdef WITH_RDPSND  #ifdef WITH_RDPSND
889          if (g_rdpsnd)          if (g_rdpsnd)
890                  rdpsnd_init();          {
891                    if (!rdpsnd_init(rdpsnd_optarg))
892                    {
893                            warning("Initializing sound-support failed!\n");
894                    }
895            }
896  #endif  #endif
         /* rdpdr_init(); */  
897    
898          if (!rdp_connect(server, flags, domain, password, shell, directory))          if (lspci_enabled)
899                  return 1;                  lspci_init();
900    
901            rdpdr_init();
902    
903          /* By setting encryption to False here, we have an encrypted login          while (run_count < 2 && continue_connect)       /* add support for Session Directory; only reconnect once */
904             packet but unencrypted transfer of other packets */          {
905          if (!packet_encryption)                  if (run_count == 0)
906                  g_encryption = False;                  {
907                            if (!rdp_connect(server, flags, domain, password, shell, directory))
908                                    return 1;
909                    }
910                    else if (!rdp_reconnect
911                             (server, flags, domain, password, shell, directory, g_redirect_cookie))
912                            return 1;
913    
914                    /* By setting encryption to False here, we have an encrypted login
915                       packet but unencrypted transfer of other packets */
916                    if (!packet_encryption)
917                            g_encryption = False;
918    
         DEBUG(("Connection successful.\n"));  
         memset(password, 0, sizeof(password));  
919    
920          if (ui_create_window())                  DEBUG(("Connection successful.\n"));
921          {                  memset(password, 0, sizeof(password));
922                  rdp_retval = rdp_main_loop();  
923                  ui_destroy_window();                  if (run_count == 0)
924                            if (!ui_create_window())
925                                    continue_connect = False;
926    
927                    if (continue_connect)
928                            rdp_main_loop(&deactivated, &ext_disc_reason);
929    
930                    DEBUG(("Disconnecting...\n"));
931                    rdp_disconnect();
932    
933                    if ((g_redirect == True) && (run_count == 0))   /* Support for Session Directory */
934                    {
935                            /* reset state of major globals */
936                            rdesktop_reset_state();
937    
938                            STRNCPY(domain, g_redirect_domain, sizeof(domain));
939                            STRNCPY(g_username, g_redirect_username, sizeof(g_username));
940                            STRNCPY(password, g_redirect_password, sizeof(password));
941                            STRNCPY(server, g_redirect_server, sizeof(server));
942                            flags |= RDP_LOGON_AUTO;
943    
944                            g_redirect = False;
945                    }
946                    else
947                    {
948                            continue_connect = False;
949                            ui_destroy_window();
950                            break;
951                    }
952    
953                    run_count++;
954          }          }
955    
956          DEBUG(("Disconnecting...\n"));          cache_save_state();
         rdp_disconnect();  
957          ui_deinit();          ui_deinit();
958    
959          if (True == rdp_retval)          if (ext_disc_reason >= 2)
960                    print_disconnect_reason(ext_disc_reason);
961    
962            if (deactivated)
963            {
964                    /* clean disconnect */
965                  return 0;                  return 0;
966            }
967          else          else
968                  return 2;          {
969                    if (ext_disc_reason == exDiscReasonAPIInitiatedDisconnect
970                        || ext_disc_reason == exDiscReasonAPIInitiatedLogoff)
971                    {
972                            /* not so clean disconnect, but nothing to worry about */
973                            return 0;
974                    }
975                    else
976                    {
977                            /* return error */
978                            return 2;
979                    }
980            }
981    
982  #endif  #endif
983    
# Line 599  xmalloc(int size) Line 1079  xmalloc(int size)
1079          return mem;          return mem;
1080  }  }
1081    
1082    /* strdup */
1083    char *
1084    xstrdup(const char *s)
1085    {
1086            char *mem = strdup(s);
1087            if (mem == NULL)
1088            {
1089                    perror("strdup");
1090                    exit(1);
1091            }
1092            return mem;
1093    }
1094    
1095  /* realloc; exit if out of memory */  /* realloc; exit if out of memory */
1096  void *  void *
1097  xrealloc(void *oldmem, int size)  xrealloc(void *oldmem, int size)
1098  {  {
1099          void *mem = realloc(oldmem, size);          void *mem;
1100    
1101            if (size < 1)
1102                    size = 1;
1103            mem = realloc(oldmem, size);
1104          if (mem == NULL)          if (mem == NULL)
1105          {          {
1106                  error("xrealloc %d\n", size);                  error("xrealloc %d\n", size);
# Line 660  unimpl(char *format, ...) Line 1157  unimpl(char *format, ...)
1157    
1158  /* produce a hex dump */  /* produce a hex dump */
1159  void  void
1160  hexdump(unsigned char *p, int len)  hexdump(unsigned char *p, unsigned int len)
1161  {  {
1162          unsigned char *line = p;          unsigned char *line = p;
1163          int i, thisline, offset = 0;          int i, thisline, offset = 0;
# Line 687  hexdump(unsigned char *p, int len) Line 1184  hexdump(unsigned char *p, int len)
1184          }          }
1185  }  }
1186    
1187    /*
1188      input: src is the string we look in for needle.
1189             Needle may be escaped by a backslash, in
1190             that case we ignore that particular needle.
1191      return value: returns next src pointer, for
1192            succesive executions, like in a while loop
1193            if retval is 0, then there are no more args.
1194      pitfalls:
1195            src is modified. 0x00 chars are inserted to
1196            terminate strings.
1197            return val, points on the next val chr after ins
1198            0x00
1199    
1200            example usage:
1201            while( (pos = next_arg( optarg, ',')) ){
1202                    printf("%s\n",optarg);
1203                    optarg=pos;
1204            }
1205    
1206    */
1207    char *
1208    next_arg(char *src, char needle)
1209    {
1210            char *nextval;
1211            char *p;
1212            char *mvp = 0;
1213    
1214            /* EOS */
1215            if (*src == (char) 0x00)
1216                    return 0;
1217    
1218            p = src;
1219            /*  skip escaped needles */
1220            while ((nextval = strchr(p, needle)))
1221            {
1222                    mvp = nextval - 1;
1223                    /* found backslashed needle */
1224                    if (*mvp == '\\' && (mvp > src))
1225                    {
1226                            /* move string one to the left */
1227                            while (*(mvp + 1) != (char) 0x00)
1228                            {
1229                                    *mvp = *(mvp + 1);
1230                                    mvp++;
1231                            }
1232                            *mvp = (char) 0x00;
1233                            p = nextval;
1234                    }
1235                    else
1236                    {
1237                            p = nextval + 1;
1238                            break;
1239                    }
1240    
1241            }
1242    
1243            /* more args available */
1244            if (nextval)
1245            {
1246                    *nextval = (char) 0x00;
1247                    return ++nextval;
1248            }
1249    
1250            /* no more args after this, jump to EOS */
1251            nextval = src + strlen(src);
1252            return nextval;
1253    }
1254    
1255    
1256    void
1257    toupper_str(char *p)
1258    {
1259            while (*p)
1260            {
1261                    if ((*p >= 'a') && (*p <= 'z'))
1262                            *p = toupper((int) *p);
1263                    p++;
1264            }
1265    }
1266    
1267    
1268    BOOL
1269    str_startswith(const char *s, const char *prefix)
1270    {
1271            return (strncmp(s, prefix, strlen(prefix)) == 0);
1272    }
1273    
1274    
1275    /* Split input into lines, and call linehandler for each
1276       line. Incomplete lines are saved in the rest variable, which should
1277       initially point to NULL. When linehandler returns False, stop and
1278       return False. Otherwise, return True.  */
1279    BOOL
1280    str_handle_lines(const char *input, char **rest, str_handle_lines_t linehandler, void *data)
1281    {
1282            char *buf, *p;
1283            char *oldrest;
1284            size_t inputlen;
1285            size_t buflen;
1286            size_t restlen = 0;
1287            BOOL ret = True;
1288    
1289            /* Copy data to buffer */
1290            inputlen = strlen(input);
1291            if (*rest)
1292                    restlen = strlen(*rest);
1293            buflen = restlen + inputlen + 1;
1294            buf = (char *) xmalloc(buflen);
1295            buf[0] = '\0';
1296            if (*rest)
1297                    STRNCPY(buf, *rest, buflen);
1298            strncat(buf, input, inputlen);
1299            p = buf;
1300    
1301            while (1)
1302            {
1303                    char *newline = strchr(p, '\n');
1304                    if (newline)
1305                    {
1306                            *newline = '\0';
1307                            if (!linehandler(p, data))
1308                            {
1309                                    p = newline + 1;
1310                                    ret = False;
1311                                    break;
1312                            }
1313                            p = newline + 1;
1314                    }
1315                    else
1316                    {
1317                            break;
1318    
1319                    }
1320            }
1321    
1322            /* Save in rest */
1323            oldrest = *rest;
1324            restlen = buf + buflen - p;
1325            *rest = (char *) xmalloc(restlen);
1326            STRNCPY((*rest), p, restlen);
1327            xfree(oldrest);
1328    
1329            xfree(buf);
1330            return ret;
1331    }
1332    
1333    /* Execute the program specified by argv. For each line in
1334       stdout/stderr output, call linehandler. Returns false on failure. */
1335    BOOL
1336    subprocess(char *const argv[], str_handle_lines_t linehandler, void *data)
1337    {
1338            pid_t child;
1339            int fd[2];
1340            int n = 1;
1341            char output[256];
1342            char *rest = NULL;
1343    
1344            if (pipe(fd) < 0)
1345            {
1346                    perror("pipe");
1347                    return False;
1348            }
1349    
1350            if ((child = fork()) < 0)
1351            {
1352                    perror("fork");
1353                    return False;
1354            }
1355    
1356            /* Child */
1357            if (child == 0)
1358            {
1359                    /* Close read end */
1360                    close(fd[0]);
1361    
1362                    /* Redirect stdout and stderr to pipe */
1363                    dup2(fd[1], 1);
1364                    dup2(fd[1], 2);
1365    
1366                    /* Execute */
1367                    execvp(argv[0], argv);
1368                    perror("Error executing child");
1369                    _exit(128);
1370            }
1371    
1372            /* Parent. Close write end. */
1373            close(fd[1]);
1374            while (n > 0)
1375            {
1376                    n = read(fd[0], output, 255);
1377                    output[n] = '\0';
1378                    str_handle_lines(output, &rest, linehandler, data);
1379            }
1380            xfree(rest);
1381    
1382            return True;
1383    }
1384    
1385    
1386    /* not all clibs got ltoa */
1387    #define LTOA_BUFSIZE (sizeof(long) * 8 + 1)
1388    
1389    char *
1390    l_to_a(long N, int base)
1391    {
1392            static char ret[LTOA_BUFSIZE];
1393    
1394            char *head = ret, buf[LTOA_BUFSIZE], *tail = buf + sizeof(buf);
1395    
1396            register int divrem;
1397    
1398            if (base < 36 || 2 > base)
1399                    base = 10;
1400    
1401            if (N < 0)
1402            {
1403                    *head++ = '-';
1404                    N = -N;
1405            }
1406    
1407            tail = buf + sizeof(buf);
1408            *--tail = 0;
1409    
1410            do
1411            {
1412                    divrem = N % base;
1413                    *--tail = (divrem <= 9) ? divrem + '0' : divrem + 'a' - 10;
1414                    N /= base;
1415            }
1416            while (N);
1417    
1418            strcpy(head, tail);
1419            return ret;
1420    }
1421    
1422    
1423  int  int
1424  load_licence(unsigned char **data)  load_licence(unsigned char **data)
# Line 699  load_licence(unsigned char **data) Line 1431  load_licence(unsigned char **data)
1431          if (home == NULL)          if (home == NULL)
1432                  return -1;                  return -1;
1433    
1434          path = (char *) xmalloc(strlen(home) + strlen(hostname) + sizeof("/.rdesktop/licence."));          path = (char *) xmalloc(strlen(home) + strlen(g_hostname) + sizeof("/.rdesktop/licence."));
1435          sprintf(path, "%s/.rdesktop/licence.%s", home, hostname);          sprintf(path, "%s/.rdesktop/licence.%s", home, g_hostname);
1436    
1437          fd = open(path, O_RDONLY);          fd = open(path, O_RDONLY);
1438          if (fd == -1)          if (fd == -1)
# Line 726  save_licence(unsigned char *data, int le Line 1458  save_licence(unsigned char *data, int le
1458          if (home == NULL)          if (home == NULL)
1459                  return;                  return;
1460    
1461          path = (char *) xmalloc(strlen(home) + strlen(hostname) + sizeof("/.rdesktop/licence."));          path = (char *) xmalloc(strlen(home) + strlen(g_hostname) + sizeof("/.rdesktop/licence."));
1462    
1463          sprintf(path, "%s/.rdesktop", home);          sprintf(path, "%s/.rdesktop", home);
1464          if ((mkdir(path, 0700) == -1) && errno != EEXIST)          if ((mkdir(path, 0700) == -1) && errno != EEXIST)
# Line 737  save_licence(unsigned char *data, int le Line 1469  save_licence(unsigned char *data, int le
1469    
1470          /* write licence to licence.hostname.new, then atomically rename to licence.hostname */          /* write licence to licence.hostname.new, then atomically rename to licence.hostname */
1471    
1472          sprintf(path, "%s/.rdesktop/licence.%s", home, hostname);          sprintf(path, "%s/.rdesktop/licence.%s", home, g_hostname);
1473          tmppath = (char *) xmalloc(strlen(path) + sizeof(".new"));          tmppath = (char *) xmalloc(strlen(path) + sizeof(".new"));
1474          strcpy(tmppath, path);          strcpy(tmppath, path);
1475          strcat(tmppath, ".new");          strcat(tmppath, ".new");
# Line 764  save_licence(unsigned char *data, int le Line 1496  save_licence(unsigned char *data, int le
1496          xfree(tmppath);          xfree(tmppath);
1497          xfree(path);          xfree(path);
1498  }  }
1499    
1500    /* Create the bitmap cache directory */
1501    BOOL
1502    rd_pstcache_mkdir(void)
1503    {
1504            char *home;
1505            char bmpcache_dir[256];
1506    
1507            home = getenv("HOME");
1508    
1509            if (home == NULL)
1510                    return False;
1511    
1512            sprintf(bmpcache_dir, "%s/%s", home, ".rdesktop");
1513    
1514            if ((mkdir(bmpcache_dir, S_IRWXU) == -1) && errno != EEXIST)
1515            {
1516                    perror(bmpcache_dir);
1517                    return False;
1518            }
1519    
1520            sprintf(bmpcache_dir, "%s/%s", home, ".rdesktop/cache");
1521    
1522            if ((mkdir(bmpcache_dir, S_IRWXU) == -1) && errno != EEXIST)
1523            {
1524                    perror(bmpcache_dir);
1525                    return False;
1526            }
1527    
1528            return True;
1529    }
1530    
1531    /* open a file in the .rdesktop directory */
1532    int
1533    rd_open_file(char *filename)
1534    {
1535            char *home;
1536            char fn[256];
1537            int fd;
1538    
1539            home = getenv("HOME");
1540            if (home == NULL)
1541                    return -1;
1542            sprintf(fn, "%s/.rdesktop/%s", home, filename);
1543            fd = open(fn, O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
1544            if (fd == -1)
1545                    perror(fn);
1546            return fd;
1547    }
1548    
1549    /* close file */
1550    void
1551    rd_close_file(int fd)
1552    {
1553            close(fd);
1554    }
1555    
1556    /* read from file*/
1557    int
1558    rd_read_file(int fd, void *ptr, int len)
1559    {
1560            return read(fd, ptr, len);
1561    }
1562    
1563    /* write to file */
1564    int
1565    rd_write_file(int fd, void *ptr, int len)
1566    {
1567            return write(fd, ptr, len);
1568    }
1569    
1570    /* move file pointer */
1571    int
1572    rd_lseek_file(int fd, int offset)
1573    {
1574            return lseek(fd, offset, SEEK_SET);
1575    }
1576    
1577    /* do a write lock on a file */
1578    BOOL
1579    rd_lock_file(int fd, int start, int len)
1580    {
1581            struct flock lock;
1582    
1583            lock.l_type = F_WRLCK;
1584            lock.l_whence = SEEK_SET;
1585            lock.l_start = start;
1586            lock.l_len = len;
1587            if (fcntl(fd, F_SETLK, &lock) == -1)
1588                    return False;
1589            return True;
1590    }

Legend:
Removed from v.567  
changed lines
  Added in v.1272

  ViewVC Help
Powered by ViewVC 1.1.26