/[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 223 by matthewc, Fri Oct 11 05:13:19 2002 UTC revision 1417 by jsorg71, Thu Aug 30 04:47:36 2007 UTC
# Line 1  Line 1 
1  /*  /* -*- 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-2002     Copyright (C) Matthew Chapman 1999-2007
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 22  Line 22 
22  #include <unistd.h>             /* read close getuid getgid getpid getppid gethostname */  #include <unistd.h>             /* read close getuid getgid getpid getppid gethostname */
23  #include <fcntl.h>              /* open */  #include <fcntl.h>              /* open */
24  #include <pwd.h>                /* getpwuid */  #include <pwd.h>                /* getpwuid */
 #include <limits.h>             /* PATH_MAX */  
25  #include <termios.h>            /* tcgetattr tcsetattr */  #include <termios.h>            /* tcgetattr tcsetattr */
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>
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    
48  #ifdef WITH_OPENSSL  #include "ssl.h"
49  #include <openssl/md5.h>  
50  #else  char g_title[64] = "";
51  #include "crypto/md5.h"  char g_username[64];
52    char g_hostname[16];
53    char g_keymapname[PATH_MAX] = "";
54    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
60                                       geometry will be fetched from
61                                       _NET_WORKAREA. If negative,
62                                       absolute value specifies the
63                                       percent of the whole screen. */
64    int g_height = 600;
65    int g_xpos = 0;
66    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 */
74    RD_BOOL g_bitmap_compression = True;
75    RD_BOOL g_sendmotion = True;
76    RD_BOOL g_bitmap_cache = True;
77    RD_BOOL g_bitmap_cache_persist_enable = False;
78    RD_BOOL g_bitmap_cache_precache = True;
79    RD_BOOL g_encryption = True;
80    RD_BOOL g_packet_encryption = True;
81    RD_BOOL g_desktop_save = True;  /* desktop save order */
82    RD_BOOL g_polygon_ellipse_orders = True;        /* polygon / ellipse orders */
83    RD_BOOL g_fullscreen = False;
84    RD_BOOL g_grab_keyboard = True;
85    RD_BOOL g_hide_decorations = False;
86    RD_BOOL g_use_rdp5 = True;
87    RD_BOOL g_rdpclip = True;
88    RD_BOOL g_console_session = False;
89    RD_BOOL g_numlock_sync = False;
90    RD_BOOL g_lspci_enabled = False;
91    RD_BOOL g_owncolmap = False;
92    RD_BOOL g_ownbackstore = True;  /* We can't rely on external BackingStore */
93    RD_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    RD_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
107    RD_BOOL g_rdpsnd = False;
108  #endif  #endif
109    
110  char title[32] = "";  #ifdef HAVE_ICONV
111  char username[16];  char g_codepage[16] = "";
112  char hostname[16];  #endif
 char keymapname[16];  
 int keylayout = 0x409;          /* Defaults to US keyboard layout */  
 int width = 800;  
 int height = 600;  
 int tcp_port_rdp = TCP_PORT_RDP;  
 BOOL bitmap_compression = True;  
 BOOL sendmotion = True;  
 BOOL orders = True;  
 BOOL encryption = True;  
 BOOL desktop_save = True;  
 BOOL fullscreen = False;  
 BOOL grab_keyboard = True;  
113    
114    extern RDPDR_DEVICE g_rdpdr_device[];
115    extern uint32 g_num_devices;
116    extern char *g_rdpdr_clientname;
117    
118    #ifdef RDP2VNC
119    extern int rfb_port;
120    extern int defer_time;
121    void
122    rdp2vnc_connect(char *server, uint32 flags, char *domain, char *password,
123                    char *shell, char *directory);
124    #endif
125  /* Display usage information */  /* Display usage information */
126  static void  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-2002 Matt Chapman.\n");          fprintf(stderr, "Version " VERSION ". Copyright (C) 1999-2007 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);
134    #ifdef RDP2VNC
135            fprintf(stderr, "   -V: vnc port\n");
136            fprintf(stderr, "   -Q: defer time (ms)\n");
137    #endif
138          fprintf(stderr, "   -u: user name\n");          fprintf(stderr, "   -u: user name\n");
139          fprintf(stderr, "   -d: domain\n");          fprintf(stderr, "   -d: domain\n");
140          fprintf(stderr, "   -s: shell\n");          fprintf(stderr, "   -s: shell\n");
141          fprintf(stderr, "   -c: working directory\n");          fprintf(stderr, "   -c: working directory\n");
142          fprintf(stderr, "   -p: password (- to prompt)\n");          fprintf(stderr, "   -p: password (- to prompt)\n");
143          fprintf(stderr, "   -n: client hostname\n");          fprintf(stderr, "   -n: client hostname\n");
144          fprintf(stderr, "   -k: keyboard layout on terminal server (us,sv,gr,etc.)\n");          fprintf(stderr, "   -k: keyboard layout on server (en-us, de, sv, etc.)\n");
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");
155          fprintf(stderr, "   -m: do not send motion events\n");          fprintf(stderr, "   -m: do not send motion events\n");
156            fprintf(stderr, "   -C: use private colour map\n");
157            fprintf(stderr, "   -D: hide window manager decorations\n");
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");
160          fprintf(stderr, "   -T: window title\n");          fprintf(stderr, "   -T: window title\n");
161            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");
164            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    #ifdef WITH_SCARD
197            fprintf(stderr, "         '-r scard[:\"Scard Name\"=\"Alias Name[;Vendor Name]\"[,...]]\n");
198            fprintf(stderr, "          example: -r scard:\"eToken PRO 00 00\"=\"AKS ifdh 0\"\n");
199            fprintf(stderr,
200                    "                   \"eToken PRO 00 00\" -> Device in Linux/Unix enviroment\n");
201            fprintf(stderr,
202                    "                   \"AKS ifdh 0\"       -> Device shown in Windows enviroment \n");
203            fprintf(stderr, "          example: -r scard:\"eToken PRO 00 00\"=\"AKS ifdh 0;AKS\"\n");
204            fprintf(stderr,
205                    "                   \"eToken PRO 00 00\" -> Device in Linux/Unix enviroment\n");
206            fprintf(stderr,
207                    "                   \"AKS ifdh 0\"       -> Device shown in Windows enviroment \n");
208            fprintf(stderr,
209                    "                   \"AKS\"              -> Device vendor name                 \n");
210    #endif
211            fprintf(stderr, "   -0: attach to console\n");
212            fprintf(stderr, "   -4: use RDP version 4\n");
213            fprintf(stderr, "   -5: use RDP version 5 (default)\n");
214  }  }
215    
216  static BOOL  static void
217    print_disconnect_reason(uint16 reason)
218    {
219            char *text;
220    
221            switch (reason)
222            {
223                    case exDiscReasonNoInfo:
224                            text = "No information available";
225                            break;
226    
227                    case exDiscReasonAPIInitiatedDisconnect:
228                            text = "Server initiated disconnect";
229                            break;
230    
231                    case exDiscReasonAPIInitiatedLogoff:
232                            text = "Server initiated logoff";
233                            break;
234    
235                    case exDiscReasonServerIdleTimeout:
236                            text = "Server idle timeout reached";
237                            break;
238    
239                    case exDiscReasonServerLogonTimeout:
240                            text = "Server logon timeout reached";
241                            break;
242    
243                    case exDiscReasonReplacedByOtherConnection:
244                            text = "The session was replaced";
245                            break;
246    
247                    case exDiscReasonOutOfMemory:
248                            text = "The server is out of memory";
249                            break;
250    
251                    case exDiscReasonServerDeniedConnection:
252                            text = "The server denied the connection";
253                            break;
254    
255                    case exDiscReasonServerDeniedConnectionFips:
256                            text = "The server denied the connection for security reason";
257                            break;
258    
259                    case exDiscReasonLicenseInternal:
260                            text = "Internal licensing error";
261                            break;
262    
263                    case exDiscReasonLicenseNoLicenseServer:
264                            text = "No license server available";
265                            break;
266    
267                    case exDiscReasonLicenseNoLicense:
268                            text = "No valid license available";
269                            break;
270    
271                    case exDiscReasonLicenseErrClientMsg:
272                            text = "Invalid licensing message";
273                            break;
274    
275                    case exDiscReasonLicenseHwidDoesntMatchLicense:
276                            text = "Hardware id doesn't match software license";
277                            break;
278    
279                    case exDiscReasonLicenseErrClientLicense:
280                            text = "Client license error";
281                            break;
282    
283                    case exDiscReasonLicenseCantFinishProtocol:
284                            text = "Network error during licensing protocol";
285                            break;
286    
287                    case exDiscReasonLicenseClientEndedProtocol:
288                            text = "Licensing protocol was not completed";
289                            break;
290    
291                    case exDiscReasonLicenseErrClientEncryption:
292                            text = "Incorrect client license enryption";
293                            break;
294    
295                    case exDiscReasonLicenseCantUpgradeLicense:
296                            text = "Can't upgrade license";
297                            break;
298    
299                    case exDiscReasonLicenseNoRemoteConnections:
300                            text = "The server is not licensed to accept remote connections";
301                            break;
302    
303                    default:
304                            if (reason > 0x1000 && reason < 0x7fff)
305                            {
306                                    text = "Internal protocol error";
307                            }
308                            else
309                            {
310                                    text = "Unknown reason";
311                            }
312            }
313            fprintf(stderr, "disconnect: %s.\n", text);
314    }
315    
316    static void
317    rdesktop_reset_state(void)
318    {
319            rdp_reset_state();
320    }
321    
322    static RD_BOOL
323  read_password(char *password, int size)  read_password(char *password, int size)
324  {  {
325          struct termios tios;          struct termios tios;
326          BOOL ret = False;          RD_BOOL ret = False;
327          int istty = 0;          int istty = 0;
328          char *p;          char *p;
329    
# Line 117  read_password(char *password, int size) Line 355  read_password(char *password, int size)
355          return ret;          return ret;
356  }  }
357    
358    static void
359    parse_server_and_port(char *server)
360    {
361            char *p;
362    #ifdef IPv6
363            int addr_colons;
364    #endif
365    
366    #ifdef IPv6
367            p = server;
368            addr_colons = 0;
369            while (*p)
370                    if (*p++ == ':')
371                            addr_colons++;
372            if (addr_colons >= 2)
373            {
374                    /* numeric IPv6 style address format - [1:2:3::4]:port */
375                    p = strchr(server, ']');
376                    if (*server == '[' && p != NULL)
377                    {
378                            if (*(p + 1) == ':' && *(p + 2) != '\0')
379                                    g_tcp_port_rdp = strtol(p + 2, NULL, 10);
380                            /* remove the port number and brackets from the address */
381                            *p = '\0';
382                            strncpy(server, server + 1, strlen(server));
383                    }
384            }
385            else
386            {
387                    /* dns name or IPv4 style address format - server.example.com:port or 1.2.3.4:port */
388                    p = strchr(server, ':');
389                    if (p != NULL)
390                    {
391                            g_tcp_port_rdp = strtol(p + 1, NULL, 10);
392                            *p = 0;
393                    }
394            }
395    #else /* no IPv6 support */
396            p = strchr(server, ':');
397            if (p != NULL)
398            {
399                    g_tcp_port_rdp = strtol(p + 1, NULL, 10);
400                    *p = 0;
401            }
402    #endif /* IPv6 */
403    
404    }
405    
406  /* Client program */  /* Client program */
407  int  int
408  main(int argc, char *argv[])  main(int argc, char *argv[])
# Line 124  main(int argc, char *argv[]) Line 410  main(int argc, char *argv[])
410          char server[64];          char server[64];
411          char fullhostname[64];          char fullhostname[64];
412          char domain[16];          char domain[16];
413          char password[16];          char password[64];
414          char shell[32];          char shell[256];
415          char directory[32];          char directory[256];
416          BOOL prompt_password;          RD_BOOL prompt_password, deactivated;
417          struct passwd *pw;          struct passwd *pw;
418          uint32 flags;          uint32 flags, ext_disc_reason = 0;
419          char *p;          char *p;
420          int c;          int c;
421            char *locale = NULL;
422            int username_option = 0;
423            RD_BOOL geometry_option = False;
424            int run_count = 0;      /* Session Directory support */
425            RD_BOOL continue_connect = True;        /* Session Directory support */
426    #ifdef WITH_RDPSND
427            char *rdpsnd_optarg = NULL;
428    #endif
429    
430    #ifdef HAVE_LOCALE_H
431            /* Set locale according to environment */
432            locale = setlocale(LC_ALL, "");
433            if (locale)
434            {
435                    locale = xstrdup(locale);
436            }
437    
438    #endif
439          flags = RDP_LOGON_NORMAL;          flags = RDP_LOGON_NORMAL;
440          prompt_password = False;          prompt_password = False;
441          domain[0] = password[0] = shell[0] = directory[0] = 0;          domain[0] = password[0] = shell[0] = directory[0] = 0;
442          strcpy(keymapname, "us");          g_embed_wnd = 0;
443    
444          while ((c = getopt(argc, argv, "u:d:s:c:p:n:k:g:fbemKT:h?")) != -1)          g_num_devices = 0;
445    
446    #ifdef RDP2VNC
447    #define VNCOPT "V:Q:"
448    #else
449    #define VNCOPT
450    #endif
451    
452            while ((c = getopt(argc, argv,
453                               VNCOPT "Au:L:d:s:c:p:n:k:g:fbBeEmzCDKS:T:NX:a:x:Pr:045h?")) != -1)
454          {          {
455                  switch (c)                  switch (c)
456                  {                  {
457    #ifdef RDP2VNC
458                            case 'V':
459                                    rfb_port = strtol(optarg, NULL, 10);
460                                    if (rfb_port < 100)
461                                            rfb_port += 5900;
462                                    break;
463    
464                            case 'Q':
465                                    defer_time = strtol(optarg, NULL, 10);
466                                    if (defer_time < 0)
467                                            defer_time = 0;
468                                    break;
469    #endif
470    
471                            case 'A':
472                                    g_seamless_rdp = True;
473                                    break;
474    
475                          case 'u':                          case 'u':
476                                  STRNCPY(username, optarg, sizeof(username));                                  STRNCPY(g_username, optarg, sizeof(g_username));
477                                    username_option = 1;
478                                    break;
479    
480                            case 'L':
481    #ifdef HAVE_ICONV
482                                    STRNCPY(g_codepage, optarg, sizeof(g_codepage));
483    #else
484                                    error("iconv support not available\n");
485    #endif
486                                  break;                                  break;
487    
488                          case 'd':                          case 'd':
# Line 175  main(int argc, char *argv[]) Line 514  main(int argc, char *argv[])
514                                  break;                                  break;
515    
516                          case 'n':                          case 'n':
517                                  STRNCPY(hostname, optarg, sizeof(hostname));                                  STRNCPY(g_hostname, optarg, sizeof(g_hostname));
518                                  break;                                  break;
519    
520                          case 'k':                          case 'k':
521                                  STRNCPY(keymapname, optarg, sizeof(keymapname));                                  STRNCPY(g_keymapname, optarg, sizeof(g_keymapname));
522                                  break;                                  break;
523    
524                          case 'g':                          case 'g':
525                                  width = strtol(optarg, &p, 10);                                  geometry_option = True;
526                                    g_fullscreen = False;
527                                    if (!strcmp(optarg, "workarea"))
528                                    {
529                                            g_width = g_height = 0;
530                                            break;
531                                    }
532    
533                                    g_width = strtol(optarg, &p, 10);
534                                    if (g_width <= 0)
535                                    {
536                                            error("invalid geometry\n");
537                                            return 1;
538                                    }
539    
540                                  if (*p == 'x')                                  if (*p == 'x')
541                                          height = strtol(p + 1, NULL, 10);                                          g_height = strtol(p + 1, &p, 10);
542    
543                                  if ((width == 0) || (height == 0))                                  if (g_height <= 0)
544                                  {                                  {
545                                          error("invalid geometry\n");                                          error("invalid geometry\n");
546                                          return 1;                                          return 1;
547                                  }                                  }
548    
549                                    if (*p == '%')
550                                    {
551                                            g_width = -g_width;
552                                            p++;
553                                    }
554    
555                                    if (*p == '+' || *p == '-')
556                                    {
557                                            g_pos |= (*p == '-') ? 2 : 1;
558                                            g_xpos = strtol(p, &p, 10);
559    
560                                    }
561                                    if (*p == '+' || *p == '-')
562                                    {
563                                            g_pos |= (*p == '-') ? 4 : 1;
564                                            g_ypos = strtol(p, NULL, 10);
565                                    }
566    
567                                  break;                                  break;
568    
569                          case 'f':                          case 'f':
570                                  fullscreen = True;                                  g_fullscreen = True;
571                                  break;                                  break;
572    
573                          case 'b':                          case 'b':
574                                  orders = False;                                  g_bitmap_cache = False;
575                                  break;                                  break;
576    
577                          case 'e':                          case 'B':
578                                  encryption = False;                                  g_ownbackstore = False;
579                                  break;                                  break;
580    
581                            case 'e':
582                                    g_encryption = False;
583                                    break;
584                            case 'E':
585                                    g_packet_encryption = False;
586                                    break;
587                          case 'm':                          case 'm':
588                                  sendmotion = False;                                  g_sendmotion = False;
589                                    break;
590    
591                            case 'C':
592                                    g_owncolmap = True;
593                                    break;
594    
595                            case 'D':
596                                    g_hide_decorations = True;
597                                  break;                                  break;
598    
599                          case 'K':                          case 'K':
600                                  grab_keyboard = False;                                  g_grab_keyboard = False;
601                                    break;
602    
603                            case 'S':
604                                    if (!strcmp(optarg, "standard"))
605                                    {
606                                            g_win_button_size = 18;
607                                            break;
608                                    }
609    
610                                    g_win_button_size = strtol(optarg, &p, 10);
611    
612                                    if (*p)
613                                    {
614                                            error("invalid button size\n");
615                                            return 1;
616                                    }
617    
618                                  break;                                  break;
619    
620                          case 'T':                          case 'T':
621                                  STRNCPY(title, optarg, sizeof(title));                                  STRNCPY(g_title, optarg, sizeof(g_title));
622                                    break;
623    
624                            case 'N':
625                                    g_numlock_sync = True;
626                                    break;
627    
628                            case 'X':
629                                    g_embed_wnd = strtol(optarg, NULL, 0);
630                                    break;
631    
632                            case 'a':
633                                    g_server_depth = strtol(optarg, NULL, 10);
634                                    if (g_server_depth != 8 &&
635                                        g_server_depth != 16 &&
636                                        g_server_depth != 15 && g_server_depth != 24
637                                        && g_server_depth != 32)
638                                    {
639                                            error("Invalid server colour depth.\n");
640                                            return 1;
641                                    }
642                                    break;
643    
644                            case 'z':
645                                    DEBUG(("rdp compression enabled\n"));
646                                    flags |= (RDP_LOGON_COMPRESSION | RDP_LOGON_COMPRESSION2);
647                                    break;
648    
649                            case 'x':
650                                    if (str_startswith(optarg, "m"))        /* modem */
651                                    {
652                                            g_rdp5_performanceflags =
653                                                    RDP5_NO_WALLPAPER | RDP5_NO_FULLWINDOWDRAG |
654                                                    RDP5_NO_MENUANIMATIONS | RDP5_NO_THEMING;
655                                    }
656                                    else if (str_startswith(optarg, "b"))   /* broadband */
657                                    {
658                                            g_rdp5_performanceflags = RDP5_NO_WALLPAPER;
659                                    }
660                                    else if (str_startswith(optarg, "l"))   /* lan */
661                                    {
662                                            g_rdp5_performanceflags = RDP5_DISABLE_NOTHING;
663                                    }
664                                    else
665                                    {
666                                            g_rdp5_performanceflags = strtol(optarg, NULL, 16);
667                                    }
668                                    break;
669    
670                            case 'P':
671                                    g_bitmap_cache_persist_enable = True;
672                                    break;
673    
674                            case 'r':
675    
676                                    if (str_startswith(optarg, "sound"))
677                                    {
678                                            optarg += 5;
679    
680                                            if (*optarg == ':')
681                                            {
682                                                    optarg++;
683                                                    while ((p = next_arg(optarg, ',')))
684                                                    {
685                                                            if (str_startswith(optarg, "remote"))
686                                                                    flags |= RDP_LOGON_LEAVE_AUDIO;
687    
688                                                            if (str_startswith(optarg, "local"))
689    #ifdef WITH_RDPSND
690                                                            {
691                                                                    rdpsnd_optarg =
692                                                                            next_arg(optarg, ':');
693                                                                    g_rdpsnd = True;
694                                                            }
695    
696    #else
697                                                                    warning("Not compiled with sound support\n");
698    #endif
699    
700                                                            if (str_startswith(optarg, "off"))
701    #ifdef WITH_RDPSND
702                                                                    g_rdpsnd = False;
703    #else
704                                                                    warning("Not compiled with sound support\n");
705    #endif
706    
707                                                            optarg = p;
708                                                    }
709                                            }
710                                            else
711                                            {
712    #ifdef WITH_RDPSND
713                                                    g_rdpsnd = True;
714    #else
715                                                    warning("Not compiled with sound support\n");
716    #endif
717                                            }
718                                    }
719                                    else if (str_startswith(optarg, "disk"))
720                                    {
721                                            /* -r disk:h:=/mnt/floppy */
722                                            disk_enum_devices(&g_num_devices, optarg + 4);
723                                    }
724                                    else if (str_startswith(optarg, "comport"))
725                                    {
726                                            serial_enum_devices(&g_num_devices, optarg + 7);
727                                    }
728                                    else if (str_startswith(optarg, "lspci"))
729                                    {
730                                            g_lspci_enabled = True;
731                                    }
732                                    else if (str_startswith(optarg, "lptport"))
733                                    {
734                                            parallel_enum_devices(&g_num_devices, optarg + 7);
735                                    }
736                                    else if (str_startswith(optarg, "printer"))
737                                    {
738                                            printer_enum_devices(&g_num_devices, optarg + 7);
739                                    }
740                                    else if (str_startswith(optarg, "clientname"))
741                                    {
742                                            g_rdpdr_clientname = xmalloc(strlen(optarg + 11) + 1);
743                                            strcpy(g_rdpdr_clientname, optarg + 11);
744                                    }
745                                    else if (str_startswith(optarg, "clipboard"))
746                                    {
747                                            optarg += 9;
748    
749                                            if (*optarg == ':')
750                                            {
751                                                    optarg++;
752    
753                                                    if (str_startswith(optarg, "off"))
754                                                            g_rdpclip = False;
755                                                    else
756                                                            cliprdr_set_mode(optarg);
757                                            }
758                                            else
759                                                    g_rdpclip = True;
760                                    }
761                                    else if (strncmp("scard", optarg, 5) == 0)
762                                    {
763    #ifdef WITH_SCARD
764                                            scard_enum_devices(&g_num_devices, optarg + 5);
765    #else
766                                            warning("Not compiled with smartcard support\n");
767    #endif
768                                    }
769                                    else
770                                    {
771                                            warning("Unknown -r argument\n\n\tPossible arguments are: comport, disk, lptport, printer, sound, clipboard, scard\n");
772                                    }
773                                    break;
774    
775                            case '0':
776                                    g_console_session = True;
777                                    break;
778    
779                            case '4':
780                                    g_use_rdp5 = False;
781                                    break;
782    
783                            case '5':
784                                    g_use_rdp5 = True;
785                                  break;                                  break;
786    
787                          case 'h':                          case 'h':
# Line 226  main(int argc, char *argv[]) Line 792  main(int argc, char *argv[])
792                  }                  }
793          }          }
794    
795          if (argc - optind < 1)          if (argc - optind != 1)
796          {          {
797                  usage(argv[0]);                  usage(argv[0]);
798                  return 1;                  return 1;
799          }          }
800    
801          STRNCPY(server, argv[optind], sizeof(server));          STRNCPY(server, argv[optind], sizeof(server));
802          p = strchr(server, ':');          parse_server_and_port(server);
803          if (p != NULL)  
804            if (g_seamless_rdp)
805          {          {
806                  tcp_port_rdp = strtol(p + 1, NULL, 10);                  if (g_win_button_size)
807                  *p = 0;                  {
808                            error("You cannot use -S and -A at the same time\n");
809                            return 1;
810                    }
811                    g_rdp5_performanceflags &= ~RDP5_NO_FULLWINDOWDRAG;
812                    if (geometry_option)
813                    {
814                            error("You cannot use -g and -A at the same time\n");
815                            return 1;
816                    }
817                    if (g_fullscreen)
818                    {
819                            error("You cannot use -f and -A at the same time\n");
820                            return 1;
821                    }
822                    if (g_hide_decorations)
823                    {
824                            error("You cannot use -D and -A at the same time\n");
825                            return 1;
826                    }
827                    if (g_embed_wnd)
828                    {
829                            error("You cannot use -X and -A at the same time\n");
830                            return 1;
831                    }
832                    if (!g_use_rdp5)
833                    {
834                            error("You cannot use -4 and -A at the same time\n");
835                            return 1;
836                    }
837                    g_width = -100;
838                    g_grab_keyboard = False;
839          }          }
840    
841          if (username[0] == 0)          if (!username_option)
842          {          {
843                  pw = getpwuid(getuid());                  pw = getpwuid(getuid());
844                  if ((pw == NULL) || (pw->pw_name == NULL))                  if ((pw == NULL) || (pw->pw_name == NULL))
# Line 249  main(int argc, char *argv[]) Line 847  main(int argc, char *argv[])
847                          return 1;                          return 1;
848                  }                  }
849    
850                  STRNCPY(username, pw->pw_name, sizeof(username));                  STRNCPY(g_username, pw->pw_name, sizeof(g_username));
851          }          }
852    
853          if (hostname[0] == 0)  #ifdef HAVE_ICONV
854            if (g_codepage[0] == 0)
855            {
856                    if (setlocale(LC_CTYPE, ""))
857                    {
858                            STRNCPY(g_codepage, nl_langinfo(CODESET), sizeof(g_codepage));
859                    }
860                    else
861                    {
862                            STRNCPY(g_codepage, DEFAULT_CODEPAGE, sizeof(g_codepage));
863                    }
864            }
865    #endif
866    
867            if (g_hostname[0] == 0)
868          {          {
869                  if (gethostname(fullhostname, sizeof(fullhostname)) == -1)                  if (gethostname(fullhostname, sizeof(fullhostname)) == -1)
870                  {                  {
# Line 264  main(int argc, char *argv[]) Line 876  main(int argc, char *argv[])
876                  if (p != NULL)                  if (p != NULL)
877                          *p = 0;                          *p = 0;
878    
879                  STRNCPY(hostname, fullhostname, sizeof(hostname));                  STRNCPY(g_hostname, fullhostname, sizeof(g_hostname));
880          }          }
881    
882            if (g_keymapname[0] == 0)
883            {
884                    if (locale && xkeymap_from_locale(locale))
885                    {
886                            fprintf(stderr, "Autoselected keyboard map %s\n", g_keymapname);
887                    }
888                    else
889                    {
890                            STRNCPY(g_keymapname, "en-us", sizeof(g_keymapname));
891                    }
892            }
893            if (locale)
894                    xfree(locale);
895    
896    
897          if (prompt_password && read_password(password, sizeof(password)))          if (prompt_password && read_password(password, sizeof(password)))
898                  flags |= RDP_LOGON_AUTO;                  flags |= RDP_LOGON_AUTO;
899    
900          if (title[0] == 0)          if (g_title[0] == 0)
901          {          {
902                  strcpy(title, "rdesktop - ");                  strcpy(g_title, "rdesktop - ");
903                  strncat(title, server, sizeof(title) - sizeof("rdesktop - "));                  strncat(g_title, server, sizeof(g_title) - sizeof("rdesktop - "));
904          }          }
905    
906    #ifdef RDP2VNC
907            rdp2vnc_connect(server, flags, domain, password, shell, directory);
908            return 0;
909    #else
910    
911          if (!ui_init())          if (!ui_init())
912                  return 1;                  return 1;
913    
914          if (!rdp_connect(server, flags, domain, password, shell, directory))  #ifdef WITH_RDPSND
915                  return 1;          if (g_rdpsnd)
916            {
917                    if (!rdpsnd_init(rdpsnd_optarg))
918                    {
919                            warning("Initializing sound-support failed!\n");
920                    }
921            }
922    #endif
923    
924          DEBUG(("Connection successful.\n"));          if (g_lspci_enabled)
925          memset(password, 0, sizeof(password));                  lspci_init();
926    
927          if (ui_create_window())          rdpdr_init();
928    
929            while (run_count < 2 && continue_connect)       /* add support for Session Directory; only reconnect once */
930          {          {
931                  rdp_main_loop();                  if (run_count == 0)
932                  ui_destroy_window();                  {
933                            if (!rdp_connect(server, flags, domain, password, shell, directory))
934                                    return 1;
935                    }
936                    else if (!rdp_reconnect
937                             (server, flags, domain, password, shell, directory, g_redirect_cookie))
938                            return 1;
939    
940                    /* By setting encryption to False here, we have an encrypted login
941                       packet but unencrypted transfer of other packets */
942                    if (!g_packet_encryption)
943                            g_encryption = False;
944    
945    
946                    DEBUG(("Connection successful.\n"));
947                    memset(password, 0, sizeof(password));
948    
949                    if (run_count == 0)
950                            if (!ui_create_window())
951                                    continue_connect = False;
952    
953                    if (continue_connect)
954                            rdp_main_loop(&deactivated, &ext_disc_reason);
955    
956                    DEBUG(("Disconnecting...\n"));
957                    rdp_disconnect();
958    
959                    if ((g_redirect == True) && (run_count == 0))   /* Support for Session Directory */
960                    {
961                            /* reset state of major globals */
962                            rdesktop_reset_state();
963    
964                            STRNCPY(domain, g_redirect_domain, sizeof(domain));
965                            STRNCPY(g_username, g_redirect_username, sizeof(g_username));
966                            STRNCPY(password, g_redirect_password, sizeof(password));
967                            STRNCPY(server, g_redirect_server, sizeof(server));
968                            flags |= RDP_LOGON_AUTO;
969    
970                            g_redirect = False;
971                    }
972                    else
973                    {
974                            continue_connect = False;
975                            ui_destroy_window();
976                            break;
977                    }
978    
979                    run_count++;
980          }          }
981    
982          DEBUG(("Disconnecting...\n"));          cache_save_state();
         rdp_disconnect();  
983          ui_deinit();          ui_deinit();
984          return 0;  
985            if (ext_disc_reason >= 2)
986                    print_disconnect_reason(ext_disc_reason);
987    
988            if (deactivated)
989            {
990                    /* clean disconnect */
991                    return 0;
992            }
993            else
994            {
995                    if (ext_disc_reason == exDiscReasonAPIInitiatedDisconnect
996                        || ext_disc_reason == exDiscReasonAPIInitiatedLogoff)
997                    {
998                            /* not so clean disconnect, but nothing to worry about */
999                            return 0;
1000                    }
1001                    else
1002                    {
1003                            /* return error */
1004                            return 2;
1005                    }
1006            }
1007    
1008    #endif
1009    
1010  }  }
1011    
1012  #ifdef EGD_SOCKET  #ifdef EGD_SOCKET
1013  /* Read 32 random bytes from PRNGD or EGD socket (based on OpenSSL RAND_egd) */  /* Read 32 random bytes from PRNGD or EGD socket (based on OpenSSL RAND_egd) */
1014  static BOOL  static RD_BOOL
1015  generate_random_egd(uint8 * buf)  generate_random_egd(uint8 * buf)
1016  {  {
1017          struct sockaddr_un addr;          struct sockaddr_un addr;
1018          BOOL ret = False;          RD_BOOL ret = False;
1019          int fd;          int fd;
1020    
1021          fd = socket(AF_UNIX, SOCK_STREAM, 0);          fd = socket(AF_UNIX, SOCK_STREAM, 0);
# Line 312  generate_random_egd(uint8 * buf) Line 1024  generate_random_egd(uint8 * buf)
1024    
1025          addr.sun_family = AF_UNIX;          addr.sun_family = AF_UNIX;
1026          memcpy(addr.sun_path, EGD_SOCKET, sizeof(EGD_SOCKET));          memcpy(addr.sun_path, EGD_SOCKET, sizeof(EGD_SOCKET));
1027          if (connect(fd, (struct sockaddr *)&addr, sizeof(addr)) == -1)          if (connect(fd, (struct sockaddr *) &addr, sizeof(addr)) == -1)
1028                  goto err;                  goto err;
1029    
1030          /* PRNGD and EGD use a simple communications protocol */          /* PRNGD and EGD use a simple communications protocol */
1031          buf[0] = 1;  /* Non-blocking (similar to /dev/urandom) */          buf[0] = 1;             /* Non-blocking (similar to /dev/urandom) */
1032          buf[1] = 32; /* Number of requested random bytes */          buf[1] = 32;            /* Number of requested random bytes */
1033          if (write(fd, buf, 2) != 2)          if (write(fd, buf, 2) != 2)
1034                  goto err;                  goto err;
1035    
1036          if ((read(fd, buf, 1) != 1) || (buf[0] == 0)) /* Available? */          if ((read(fd, buf, 1) != 1) || (buf[0] == 0))   /* Available? */
1037                  goto err;                  goto err;
1038    
1039          if (read(fd, buf, 32) != 32)          if (read(fd, buf, 32) != 32)
# Line 329  generate_random_egd(uint8 * buf) Line 1041  generate_random_egd(uint8 * buf)
1041    
1042          ret = True;          ret = True;
1043    
1044  err:        err:
1045          close(fd);          close(fd);
1046          return ret;          return ret;
1047  }  }
# Line 341  generate_random(uint8 * random) Line 1053  generate_random(uint8 * random)
1053  {  {
1054          struct stat st;          struct stat st;
1055          struct tms tmsbuf;          struct tms tmsbuf;
1056          MD5_CTX md5;          SSL_MD5 md5;
1057          uint32 *r;          uint32 *r;
1058          int fd, n;          int fd, n;
1059    
# Line 362  generate_random(uint8 * random) Line 1074  generate_random(uint8 * random)
1074  #endif  #endif
1075    
1076          /* Otherwise use whatever entropy we can gather - ideas welcome. */          /* Otherwise use whatever entropy we can gather - ideas welcome. */
1077          r = (uint32 *)random;          r = (uint32 *) random;
1078          r[0] = (getpid()) | (getppid() << 16);          r[0] = (getpid()) | (getppid() << 16);
1079          r[1] = (getuid()) | (getgid() << 16);          r[1] = (getuid()) | (getgid() << 16);
1080          r[2] = times(&tmsbuf);  /* system uptime (clocks) */          r[2] = times(&tmsbuf);  /* system uptime (clocks) */
# Line 373  generate_random(uint8 * random) Line 1085  generate_random(uint8 * random)
1085          r[7] = st.st_ctime;          r[7] = st.st_ctime;
1086    
1087          /* Hash both halves with MD5 to obscure possible patterns */          /* Hash both halves with MD5 to obscure possible patterns */
1088          MD5_Init(&md5);          ssl_md5_init(&md5);
1089          MD5_Update(&md5, random, 16);          ssl_md5_update(&md5, random, 16);
1090          MD5_Final(random, &md5);          ssl_md5_final(&md5, random);
1091          MD5_Update(&md5, random+16, 16);          ssl_md5_update(&md5, random + 16, 16);
1092          MD5_Final(random+16, &md5);          ssl_md5_final(&md5, random + 16);
1093  }  }
1094    
1095  /* malloc; exit if out of memory */  /* malloc; exit if out of memory */
# Line 393  xmalloc(int size) Line 1105  xmalloc(int size)
1105          return mem;          return mem;
1106  }  }
1107    
1108    /* Exit on NULL pointer. Use to verify result from XGetImage etc */
1109    void
1110    exit_if_null(void *ptr)
1111    {
1112            if (ptr == NULL)
1113            {
1114                    error("unexpected null pointer. Out of memory?\n");
1115                    exit(1);
1116            }
1117    }
1118    
1119    /* strdup */
1120    char *
1121    xstrdup(const char *s)
1122    {
1123            char *mem = strdup(s);
1124            if (mem == NULL)
1125            {
1126                    perror("strdup");
1127                    exit(1);
1128            }
1129            return mem;
1130    }
1131    
1132  /* realloc; exit if out of memory */  /* realloc; exit if out of memory */
1133  void *  void *
1134  xrealloc(void *oldmem, int size)  xrealloc(void *oldmem, int size)
1135  {  {
1136          void *mem = realloc(oldmem, size);          void *mem;
1137    
1138            if (size < 1)
1139                    size = 1;
1140            mem = realloc(oldmem, size);
1141          if (mem == NULL)          if (mem == NULL)
1142          {          {
1143                  error("xrealloc %d\n", size);                  error("xrealloc %d\n", size);
# Line 426  error(char *format, ...) Line 1166  error(char *format, ...)
1166          va_end(ap);          va_end(ap);
1167  }  }
1168    
1169    /* report a warning */
1170    void
1171    warning(char *format, ...)
1172    {
1173            va_list ap;
1174    
1175            fprintf(stderr, "WARNING: ");
1176    
1177            va_start(ap, format);
1178            vfprintf(stderr, format, ap);
1179            va_end(ap);
1180    }
1181    
1182  /* report an unimplemented protocol feature */  /* report an unimplemented protocol feature */
1183  void  void
1184  unimpl(char *format, ...)  unimpl(char *format, ...)
# Line 444  void Line 1197  void
1197  hexdump(unsigned char *p, unsigned int len)  hexdump(unsigned char *p, unsigned int len)
1198  {  {
1199          unsigned char *line = p;          unsigned char *line = p;
1200          unsigned int thisline, offset = 0;          int i, thisline, offset = 0;
         int i;  
1201    
1202          while (offset < len)          while (offset < len)
1203          {          {
# Line 469  hexdump(unsigned char *p, unsigned int l Line 1221  hexdump(unsigned char *p, unsigned int l
1221          }          }
1222  }  }
1223    
1224  #ifdef SAVE_LICENCE  /*
1225      input: src is the string we look in for needle.
1226             Needle may be escaped by a backslash, in
1227             that case we ignore that particular needle.
1228      return value: returns next src pointer, for
1229            succesive executions, like in a while loop
1230            if retval is 0, then there are no more args.
1231      pitfalls:
1232            src is modified. 0x00 chars are inserted to
1233            terminate strings.
1234            return val, points on the next val chr after ins
1235            0x00
1236    
1237            example usage:
1238            while( (pos = next_arg( optarg, ',')) ){
1239                    printf("%s\n",optarg);
1240                    optarg=pos;
1241            }
1242    
1243    */
1244    char *
1245    next_arg(char *src, char needle)
1246    {
1247            char *nextval;
1248            char *p;
1249            char *mvp = 0;
1250    
1251            /* EOS */
1252            if (*src == (char) 0x00)
1253                    return 0;
1254    
1255            p = src;
1256            /*  skip escaped needles */
1257            while ((nextval = strchr(p, needle)))
1258            {
1259                    mvp = nextval - 1;
1260                    /* found backslashed needle */
1261                    if (*mvp == '\\' && (mvp > src))
1262                    {
1263                            /* move string one to the left */
1264                            while (*(mvp + 1) != (char) 0x00)
1265                            {
1266                                    *mvp = *(mvp + 1);
1267                                    mvp++;
1268                            }
1269                            *mvp = (char) 0x00;
1270                            p = nextval;
1271                    }
1272                    else
1273                    {
1274                            p = nextval + 1;
1275                            break;
1276                    }
1277    
1278            }
1279    
1280            /* more args available */
1281            if (nextval)
1282            {
1283                    *nextval = (char) 0x00;
1284                    return ++nextval;
1285            }
1286    
1287            /* no more args after this, jump to EOS */
1288            nextval = src + strlen(src);
1289            return nextval;
1290    }
1291    
1292    
1293    void
1294    toupper_str(char *p)
1295    {
1296            while (*p)
1297            {
1298                    if ((*p >= 'a') && (*p <= 'z'))
1299                            *p = toupper((int) *p);
1300                    p++;
1301            }
1302    }
1303    
1304    
1305    RD_BOOL
1306    str_startswith(const char *s, const char *prefix)
1307    {
1308            return (strncmp(s, prefix, strlen(prefix)) == 0);
1309    }
1310    
1311    
1312    /* Split input into lines, and call linehandler for each
1313       line. Incomplete lines are saved in the rest variable, which should
1314       initially point to NULL. When linehandler returns False, stop and
1315       return False. Otherwise, return True.  */
1316    RD_BOOL
1317    str_handle_lines(const char *input, char **rest, str_handle_lines_t linehandler, void *data)
1318    {
1319            char *buf, *p;
1320            char *oldrest;
1321            size_t inputlen;
1322            size_t buflen;
1323            size_t restlen = 0;
1324            RD_BOOL ret = True;
1325    
1326            /* Copy data to buffer */
1327            inputlen = strlen(input);
1328            if (*rest)
1329                    restlen = strlen(*rest);
1330            buflen = restlen + inputlen + 1;
1331            buf = (char *) xmalloc(buflen);
1332            buf[0] = '\0';
1333            if (*rest)
1334                    STRNCPY(buf, *rest, buflen);
1335            strncat(buf, input, inputlen);
1336            p = buf;
1337    
1338            while (1)
1339            {
1340                    char *newline = strchr(p, '\n');
1341                    if (newline)
1342                    {
1343                            *newline = '\0';
1344                            if (!linehandler(p, data))
1345                            {
1346                                    p = newline + 1;
1347                                    ret = False;
1348                                    break;
1349                            }
1350                            p = newline + 1;
1351                    }
1352                    else
1353                    {
1354                            break;
1355    
1356                    }
1357            }
1358    
1359            /* Save in rest */
1360            oldrest = *rest;
1361            restlen = buf + buflen - p;
1362            *rest = (char *) xmalloc(restlen);
1363            STRNCPY((*rest), p, restlen);
1364            xfree(oldrest);
1365    
1366            xfree(buf);
1367            return ret;
1368    }
1369    
1370    /* Execute the program specified by argv. For each line in
1371       stdout/stderr output, call linehandler. Returns false on failure. */
1372    RD_BOOL
1373    subprocess(char *const argv[], str_handle_lines_t linehandler, void *data)
1374    {
1375            pid_t child;
1376            int fd[2];
1377            int n = 1;
1378            char output[256];
1379            char *rest = NULL;
1380    
1381            if (pipe(fd) < 0)
1382            {
1383                    perror("pipe");
1384                    return False;
1385            }
1386    
1387            if ((child = fork()) < 0)
1388            {
1389                    perror("fork");
1390                    return False;
1391            }
1392    
1393            /* Child */
1394            if (child == 0)
1395            {
1396                    /* Close read end */
1397                    close(fd[0]);
1398    
1399                    /* Redirect stdout and stderr to pipe */
1400                    dup2(fd[1], 1);
1401                    dup2(fd[1], 2);
1402    
1403                    /* Execute */
1404                    execvp(argv[0], argv);
1405                    perror("Error executing child");
1406                    _exit(128);
1407            }
1408    
1409            /* Parent. Close write end. */
1410            close(fd[1]);
1411            while (n > 0)
1412            {
1413                    n = read(fd[0], output, 255);
1414                    output[n] = '\0';
1415                    str_handle_lines(output, &rest, linehandler, data);
1416            }
1417            xfree(rest);
1418    
1419            return True;
1420    }
1421    
1422    
1423    /* not all clibs got ltoa */
1424    #define LTOA_BUFSIZE (sizeof(long) * 8 + 1)
1425    
1426    char *
1427    l_to_a(long N, int base)
1428    {
1429            static char ret[LTOA_BUFSIZE];
1430    
1431            char *head = ret, buf[LTOA_BUFSIZE], *tail = buf + sizeof(buf);
1432    
1433            register int divrem;
1434    
1435            if (base < 36 || 2 > base)
1436                    base = 10;
1437    
1438            if (N < 0)
1439            {
1440                    *head++ = '-';
1441                    N = -N;
1442            }
1443    
1444            tail = buf + sizeof(buf);
1445            *--tail = 0;
1446    
1447            do
1448            {
1449                    divrem = N % base;
1450                    *--tail = (divrem <= 9) ? divrem + '0' : divrem + 'a' - 10;
1451                    N /= base;
1452            }
1453            while (N);
1454    
1455            strcpy(head, tail);
1456            return ret;
1457    }
1458    
1459    
1460  int  int
1461  load_licence(unsigned char **data)  load_licence(unsigned char **data)
1462  {  {
1463          char path[PATH_MAX];          char *home, *path;
         char *home;  
1464          struct stat st;          struct stat st;
1465          int fd;          int fd, length;
1466    
1467          home = getenv("HOME");          home = getenv("HOME");
1468          if (home == NULL)          if (home == NULL)
1469                  return -1;                  return -1;
1470    
1471          STRNCPY(path, home, sizeof(path));          path = (char *) xmalloc(strlen(home) + strlen(g_hostname) + sizeof("/.rdesktop/licence."));
1472          strncat(path, "/.rdesktop/licence", sizeof(path) - strlen(path) - 1);          sprintf(path, "%s/.rdesktop/licence.%s", home, g_hostname);
1473    
1474          fd = open(path, O_RDONLY);          fd = open(path, O_RDONLY);
1475          if (fd == -1)          if (fd == -1)
# Line 492  load_licence(unsigned char **data) Line 1478  load_licence(unsigned char **data)
1478          if (fstat(fd, &st))          if (fstat(fd, &st))
1479                  return -1;                  return -1;
1480    
1481          *data = xmalloc(st.st_size);          *data = (uint8 *) xmalloc(st.st_size);
1482          return read(fd, *data, st.st_size);          length = read(fd, *data, st.st_size);
1483            close(fd);
1484            xfree(path);
1485            return length;
1486  }  }
1487    
1488  void  void
1489  save_licence(unsigned char *data, int length)  save_licence(unsigned char *data, int length)
1490  {  {
1491          char path[PATH_MAX];          char *home, *path, *tmppath;
         char *home;  
1492          int fd;          int fd;
1493    
1494          home = getenv("HOME");          home = getenv("HOME");
1495          if (home == NULL)          if (home == NULL)
1496                  return;                  return;
1497    
1498          STRNCPY(path, home, sizeof(path));          path = (char *) xmalloc(strlen(home) + strlen(g_hostname) + sizeof("/.rdesktop/licence."));
1499          strncat(path, "/.rdesktop", sizeof(path) - strlen(path) - 1);  
1500          mkdir(path, 0700);          sprintf(path, "%s/.rdesktop", home);
1501            if ((mkdir(path, 0700) == -1) && errno != EEXIST)
1502            {
1503                    perror(path);
1504                    return;
1505            }
1506    
1507            /* write licence to licence.hostname.new, then atomically rename to licence.hostname */
1508    
1509          strncat(path, "/licence", sizeof(path) - strlen(path) - 1);          sprintf(path, "%s/.rdesktop/licence.%s", home, g_hostname);
1510            tmppath = (char *) xmalloc(strlen(path) + sizeof(".new"));
1511            strcpy(tmppath, path);
1512            strcat(tmppath, ".new");
1513    
1514          fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, 0600);          fd = open(tmppath, O_WRONLY | O_CREAT | O_TRUNC, 0600);
1515          if (fd == -1)          if (fd == -1)
1516          {          {
1517                  perror("open");                  perror(tmppath);
1518                  return;                  return;
1519          }          }
1520    
1521          write(fd, data, length);          if (write(fd, data, length) != length)
1522            {
1523                    perror(tmppath);
1524                    unlink(tmppath);
1525            }
1526            else if (rename(tmppath, path) == -1)
1527            {
1528                    perror(path);
1529                    unlink(tmppath);
1530            }
1531    
1532          close(fd);          close(fd);
1533            xfree(tmppath);
1534            xfree(path);
1535    }
1536    
1537    /* Create the bitmap cache directory */
1538    RD_BOOL
1539    rd_pstcache_mkdir(void)
1540    {
1541            char *home;
1542            char bmpcache_dir[256];
1543    
1544            home = getenv("HOME");
1545    
1546            if (home == NULL)
1547                    return False;
1548    
1549            sprintf(bmpcache_dir, "%s/%s", home, ".rdesktop");
1550    
1551            if ((mkdir(bmpcache_dir, S_IRWXU) == -1) && errno != EEXIST)
1552            {
1553                    perror(bmpcache_dir);
1554                    return False;
1555            }
1556    
1557            sprintf(bmpcache_dir, "%s/%s", home, ".rdesktop/cache");
1558    
1559            if ((mkdir(bmpcache_dir, S_IRWXU) == -1) && errno != EEXIST)
1560            {
1561                    perror(bmpcache_dir);
1562                    return False;
1563            }
1564    
1565            return True;
1566    }
1567    
1568    /* open a file in the .rdesktop directory */
1569    int
1570    rd_open_file(char *filename)
1571    {
1572            char *home;
1573            char fn[256];
1574            int fd;
1575    
1576            home = getenv("HOME");
1577            if (home == NULL)
1578                    return -1;
1579            sprintf(fn, "%s/.rdesktop/%s", home, filename);
1580            fd = open(fn, O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
1581            if (fd == -1)
1582                    perror(fn);
1583            return fd;
1584    }
1585    
1586    /* close file */
1587    void
1588    rd_close_file(int fd)
1589    {
1590            close(fd);
1591    }
1592    
1593    /* read from file*/
1594    int
1595    rd_read_file(int fd, void *ptr, int len)
1596    {
1597            return read(fd, ptr, len);
1598    }
1599    
1600    /* write to file */
1601    int
1602    rd_write_file(int fd, void *ptr, int len)
1603    {
1604            return write(fd, ptr, len);
1605    }
1606    
1607    /* move file pointer */
1608    int
1609    rd_lseek_file(int fd, int offset)
1610    {
1611            return lseek(fd, offset, SEEK_SET);
1612    }
1613    
1614    /* do a write lock on a file */
1615    RD_BOOL
1616    rd_lock_file(int fd, int start, int len)
1617    {
1618            struct flock lock;
1619    
1620            lock.l_type = F_WRLCK;
1621            lock.l_whence = SEEK_SET;
1622            lock.l_start = start;
1623            lock.l_len = len;
1624            if (fcntl(fd, F_SETLK, &lock) == -1)
1625                    return False;
1626            return True;
1627  }  }
 #endif  

Legend:
Removed from v.223  
changed lines
  Added in v.1417

  ViewVC Help
Powered by ViewVC 1.1.26