You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

vncviewer.cxx 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690
  1. /* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved.
  2. * Copyright 2011 Pierre Ossman <ossman@cendio.se> for Cendio AB
  3. * Copyright (C) 2011 D. R. Commander. All Rights Reserved.
  4. *
  5. * This is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation; either version 2 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This software is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this software; if not, write to the Free Software
  17. * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
  18. * USA.
  19. */
  20. #ifdef HAVE_CONFIG_H
  21. #include <config.h>
  22. #endif
  23. #include <string.h>
  24. #include <stdio.h>
  25. #include <ctype.h>
  26. #include <stdlib.h>
  27. #include <errno.h>
  28. #include <signal.h>
  29. #include <locale.h>
  30. #include <fcntl.h>
  31. #include <unistd.h>
  32. #include <sys/stat.h>
  33. #ifdef WIN32
  34. #include <os/winerrno.h>
  35. #include <direct.h>
  36. #define mkdir(path, mode) _mkdir(path)
  37. #endif
  38. #if !defined(WIN32) && !defined(__APPLE__)
  39. #include <X11/Xlib.h>
  40. #include <X11/XKBlib.h>
  41. #endif
  42. #include <rfb/Logger_stdio.h>
  43. #include <rfb/SecurityClient.h>
  44. #include <rfb/Security.h>
  45. #ifdef HAVE_GNUTLS
  46. #include <rfb/CSecurityTLS.h>
  47. #endif
  48. #include <rfb/LogWriter.h>
  49. #include <rfb/Timer.h>
  50. #include <rfb/Exception.h>
  51. #include <network/TcpSocket.h>
  52. #include <os/os.h>
  53. #include <FL/Fl.H>
  54. #include <FL/Fl_Widget.H>
  55. #include <FL/Fl_PNG_Image.H>
  56. #include <FL/Fl_Sys_Menu_Bar.H>
  57. #include <FL/fl_ask.H>
  58. #include <FL/x.H>
  59. #include "i18n.h"
  60. #include "parameters.h"
  61. #include "CConn.h"
  62. #include "ServerDialog.h"
  63. #include "UserDialog.h"
  64. #include "touch.h"
  65. #include "vncviewer.h"
  66. #include "fltk_layout.h"
  67. #ifdef WIN32
  68. #include "resource.h"
  69. #include "win32.h"
  70. #endif
  71. rfb::LogWriter vlog("main");
  72. using namespace network;
  73. using namespace rfb;
  74. using namespace std;
  75. char vncServerName[VNCSERVERNAMELEN] = { '\0' };
  76. static const char *argv0 = NULL;
  77. static bool inMainloop = false;
  78. static bool exitMainloop = false;
  79. static const char *exitError = NULL;
  80. static const char *about_text()
  81. {
  82. static char buffer[1024];
  83. // This is used in multiple places with potentially different
  84. // encodings, so we need to make sure we get a fresh string every
  85. // time.
  86. snprintf(buffer, sizeof(buffer),
  87. _("TigerVNC Viewer %d-bit v%s\n"
  88. "Built on: %s\n"
  89. "Copyright (C) 1999-%d TigerVNC Team and many others (see README.rst)\n"
  90. "See https://www.tigervnc.org for information on TigerVNC."),
  91. (int)sizeof(size_t)*8, PACKAGE_VERSION,
  92. BUILD_TIMESTAMP, 2020);
  93. return buffer;
  94. }
  95. void exit_vncviewer(const char *error)
  96. {
  97. // Prioritise the first error we get as that is probably the most
  98. // relevant one.
  99. if ((error != NULL) && (exitError == NULL))
  100. exitError = strdup(error);
  101. if (inMainloop)
  102. exitMainloop = true;
  103. else {
  104. // We're early in the startup. Assume we can just exit().
  105. if (alertOnFatalError)
  106. fl_alert("%s", exitError);
  107. exit(EXIT_FAILURE);
  108. }
  109. }
  110. bool should_exit()
  111. {
  112. return exitMainloop;
  113. }
  114. void about_vncviewer()
  115. {
  116. fl_message_title(_("About TigerVNC Viewer"));
  117. fl_message("%s", about_text());
  118. }
  119. void run_mainloop()
  120. {
  121. int next_timer;
  122. next_timer = Timer::checkTimeouts();
  123. if (next_timer == 0)
  124. next_timer = INT_MAX;
  125. if (Fl::wait((double)next_timer / 1000.0) < 0.0) {
  126. vlog.error(_("Internal FLTK error. Exiting."));
  127. exit(-1);
  128. }
  129. }
  130. #ifdef __APPLE__
  131. static void about_callback(Fl_Widget *widget, void *data)
  132. {
  133. about_vncviewer();
  134. }
  135. static void new_connection_cb(Fl_Widget *widget, void *data)
  136. {
  137. const char *argv[2];
  138. pid_t pid;
  139. pid = fork();
  140. if (pid == -1) {
  141. vlog.error(_("Error starting new TigerVNC Viewer: %s"), strerror(errno));
  142. return;
  143. }
  144. if (pid != 0)
  145. return;
  146. argv[0] = argv0;
  147. argv[1] = NULL;
  148. execvp(argv[0], (char * const *)argv);
  149. vlog.error(_("Error starting new TigerVNC Viewer: %s"), strerror(errno));
  150. _exit(1);
  151. }
  152. #endif
  153. static void CleanupSignalHandler(int sig)
  154. {
  155. // CleanupSignalHandler allows C++ object cleanup to happen because it calls
  156. // exit() rather than the default which is to abort.
  157. vlog.info(_("Termination signal %d has been received. TigerVNC Viewer will now exit."), sig);
  158. exit(1);
  159. }
  160. static void init_fltk()
  161. {
  162. // Basic text size (10pt @ 96 dpi => 13px)
  163. FL_NORMAL_SIZE = 13;
  164. // Select a FLTK scheme and background color that looks somewhat
  165. // close to modern systems
  166. Fl::scheme("gtk+");
  167. Fl::background(220, 220, 220);
  168. // macOS has a slightly brighter default background though
  169. #ifdef __APPLE__
  170. Fl::background(240, 240, 240);
  171. #endif
  172. // Proper Gnome Shell integration requires that we set a sensible
  173. // WM_CLASS for the window.
  174. Fl_Window::default_xclass("vncviewer");
  175. // Set the default icon for all windows.
  176. #ifdef WIN32
  177. HICON lg, sm;
  178. lg = (HICON)LoadImage(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_ICON),
  179. IMAGE_ICON, GetSystemMetrics(SM_CXICON),
  180. GetSystemMetrics(SM_CYICON),
  181. LR_DEFAULTCOLOR | LR_SHARED);
  182. sm = (HICON)LoadImage(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_ICON),
  183. IMAGE_ICON, GetSystemMetrics(SM_CXSMICON),
  184. GetSystemMetrics(SM_CYSMICON),
  185. LR_DEFAULTCOLOR | LR_SHARED);
  186. Fl_Window::default_icons(lg, sm);
  187. #elif ! defined(__APPLE__)
  188. const int icon_sizes[] = {48, 32, 24, 16};
  189. Fl_PNG_Image *icons[4];
  190. int count;
  191. count = 0;
  192. // FIXME: Follow icon theme specification
  193. for (size_t i = 0;i < sizeof(icon_sizes)/sizeof(icon_sizes[0]);i++) {
  194. char icon_path[PATH_MAX];
  195. bool exists;
  196. sprintf(icon_path, "%s/icons/hicolor/%dx%d/apps/tigervnc.png",
  197. CMAKE_INSTALL_FULL_DATADIR, icon_sizes[i], icon_sizes[i]);
  198. #ifndef WIN32
  199. struct stat st;
  200. if (stat(icon_path, &st) != 0)
  201. #else
  202. struct _stat st;
  203. if (_stat(icon_path, &st) != 0)
  204. return(false);
  205. #endif
  206. exists = false;
  207. else
  208. exists = true;
  209. if (exists) {
  210. icons[count] = new Fl_PNG_Image(icon_path);
  211. if (icons[count]->w() == 0 ||
  212. icons[count]->h() == 0 ||
  213. icons[count]->d() != 4) {
  214. delete icons[count];
  215. continue;
  216. }
  217. count++;
  218. }
  219. }
  220. Fl_Window::default_icons((const Fl_RGB_Image**)icons, count);
  221. for (int i = 0;i < count;i++)
  222. delete icons[i];
  223. #endif
  224. // This makes the "icon" in dialogs rounded, which fits better
  225. // with the above schemes.
  226. fl_message_icon()->box(FL_UP_BOX);
  227. // Turn off the annoying behaviour where popups track the mouse.
  228. fl_message_hotspot(false);
  229. // Avoid empty titles for popups
  230. fl_message_title_default(_("TigerVNC Viewer"));
  231. #ifdef WIN32
  232. // Most "normal" Windows apps use this font for UI elements.
  233. Fl::set_font(FL_HELVETICA, "Tahoma");
  234. #endif
  235. // FLTK exposes these so that we can translate them.
  236. fl_no = _("No");
  237. fl_yes = _("Yes");
  238. fl_ok = _("OK");
  239. fl_cancel = _("Cancel");
  240. fl_close = _("Close");
  241. #ifdef __APPLE__
  242. /* Needs trailing space */
  243. static char fltk_about[16];
  244. snprintf(fltk_about, sizeof(fltk_about), "%s ", _("About"));
  245. Fl_Mac_App_Menu::about = fltk_about;
  246. static char fltk_hide[16];
  247. snprintf(fltk_hide, sizeof(fltk_hide), "%s ", _("Hide"));
  248. Fl_Mac_App_Menu::hide = fltk_hide;
  249. static char fltk_quit[16];
  250. snprintf(fltk_quit, sizeof(fltk_quit), "%s ", _("Quit"));
  251. Fl_Mac_App_Menu::quit = fltk_quit;
  252. Fl_Mac_App_Menu::print = ""; // Don't want the print item
  253. Fl_Mac_App_Menu::services = _("Services");
  254. Fl_Mac_App_Menu::hide_others = _("Hide Others");
  255. Fl_Mac_App_Menu::show = _("Show All");
  256. fl_mac_set_about(about_callback, NULL);
  257. Fl_Sys_Menu_Bar *menubar;
  258. char buffer[1024];
  259. menubar = new Fl_Sys_Menu_Bar(0, 0, 500, 25);
  260. // Fl_Sys_Menu_Bar overrides methods without them being virtual,
  261. // which means we cannot use our generic Fl_Menu_ helpers.
  262. if (fltk_menu_escape(p_("SysMenu|", "&File"),
  263. buffer, sizeof(buffer)) < sizeof(buffer))
  264. menubar->add(buffer, 0, 0, 0, FL_SUBMENU);
  265. if (fltk_menu_escape(p_("SysMenu|File|", "&New Connection"),
  266. buffer, sizeof(buffer)) < sizeof(buffer))
  267. menubar->insert(1, buffer, FL_COMMAND | 'n', new_connection_cb);
  268. #endif
  269. }
  270. static void mkvnchomedir()
  271. {
  272. // Create .vnc in the user's home directory if it doesn't already exist
  273. char* homeDir = NULL;
  274. if (getvnchomedir(&homeDir) == -1) {
  275. vlog.error(_("Could not create VNC home directory: can't obtain home "
  276. "directory path."));
  277. } else {
  278. int result = mkdir(homeDir, 0755);
  279. if (result == -1 && errno != EEXIST)
  280. vlog.error(_("Could not create VNC home directory: %s."), strerror(errno));
  281. delete [] homeDir;
  282. }
  283. }
  284. static void usage(const char *programName)
  285. {
  286. #ifdef WIN32
  287. // If we don't have a console then we need to create one for output
  288. if (GetConsoleWindow() == NULL) {
  289. HANDLE handle;
  290. int fd;
  291. AllocConsole();
  292. handle = GetStdHandle(STD_ERROR_HANDLE);
  293. fd = _open_osfhandle((intptr_t)handle, O_TEXT);
  294. *stderr = *fdopen(fd, "w");
  295. }
  296. #endif
  297. fprintf(stderr,
  298. "\n"
  299. "usage: %s [parameters] [host][:displayNum]\n"
  300. " %s [parameters] [host][::port]\n"
  301. #ifndef WIN32
  302. " %s [parameters] [unix socket]\n"
  303. #endif
  304. " %s [parameters] -listen [port]\n"
  305. " %s [parameters] [.tigervnc file]\n",
  306. programName, programName,
  307. #ifndef WIN32
  308. programName,
  309. #endif
  310. programName, programName);
  311. #if !defined(WIN32) && !defined(__APPLE__)
  312. fprintf(stderr,"\n"
  313. "Options:\n\n"
  314. " -display Xdisplay - Specifies the X display for the viewer window\n"
  315. " -geometry geometry - Initial position of the main VNC viewer window. See the\n"
  316. " man page for details.\n");
  317. #endif
  318. fprintf(stderr,"\n"
  319. "Parameters can be turned on with -<param> or off with -<param>=0\n"
  320. "Parameters which take a value can be specified as "
  321. "-<param> <value>\n"
  322. "Other valid forms are <param>=<value> -<param>=<value> "
  323. "--<param>=<value>\n"
  324. "Parameter names are case-insensitive. The parameters are:\n\n");
  325. Configuration::listParams(79, 14);
  326. #ifdef WIN32
  327. // Just wait for the user to kill the console window
  328. Sleep(INFINITE);
  329. #endif
  330. exit(1);
  331. }
  332. static void
  333. potentiallyLoadConfigurationFile(char *vncServerName)
  334. {
  335. const bool hasPathSeparator = (strchr(vncServerName, '/') != NULL ||
  336. (strchr(vncServerName, '\\')) != NULL);
  337. if (hasPathSeparator) {
  338. #ifndef WIN32
  339. struct stat sb;
  340. // This might be a UNIX socket, we need to check
  341. if (stat(vncServerName, &sb) == -1) {
  342. // Some access problem; let loadViewerParameters() deal with it...
  343. } else {
  344. if ((sb.st_mode & S_IFMT) == S_IFSOCK)
  345. return;
  346. }
  347. #endif
  348. try {
  349. const char* newServerName;
  350. newServerName = loadViewerParameters(vncServerName);
  351. // This might be empty, but we still need to clear it so we
  352. // don't try to connect to the filename
  353. strncpy(vncServerName, newServerName, VNCSERVERNAMELEN-1);
  354. vncServerName[VNCSERVERNAMELEN-1] = '\0';
  355. } catch (rfb::Exception& e) {
  356. vlog.error("%s", e.str());
  357. exit_vncviewer(e.str());
  358. }
  359. }
  360. }
  361. #ifndef WIN32
  362. static int
  363. interpretViaParam(char *remoteHost, int *remotePort, int localPort)
  364. {
  365. const int SERVER_PORT_OFFSET = 5900;
  366. char *pos = strchr(vncServerName, ':');
  367. if (pos == NULL)
  368. *remotePort = SERVER_PORT_OFFSET;
  369. else {
  370. int portOffset = SERVER_PORT_OFFSET;
  371. size_t len;
  372. *pos++ = '\0';
  373. len = strlen(pos);
  374. if (*pos == ':') {
  375. /* Two colons is an absolute port number, not an offset. */
  376. pos++;
  377. len--;
  378. portOffset = 0;
  379. }
  380. if (!len || strspn (pos, "-0123456789") != len )
  381. return 1;
  382. *remotePort = atoi(pos) + portOffset;
  383. }
  384. if (*vncServerName != '\0')
  385. strncpy(remoteHost, vncServerName, VNCSERVERNAMELEN);
  386. else
  387. strncpy(remoteHost, "localhost", VNCSERVERNAMELEN);
  388. remoteHost[VNCSERVERNAMELEN - 1] = '\0';
  389. snprintf(vncServerName, VNCSERVERNAMELEN, "localhost::%d", localPort);
  390. vncServerName[VNCSERVERNAMELEN - 1] = '\0';
  391. return 0;
  392. }
  393. static void
  394. createTunnel(const char *gatewayHost, const char *remoteHost,
  395. int remotePort, int localPort)
  396. {
  397. const char *cmd = getenv("VNC_VIA_CMD");
  398. char *cmd2, *percent;
  399. char lport[10], rport[10];
  400. sprintf(lport, "%d", localPort);
  401. sprintf(rport, "%d", remotePort);
  402. setenv("G", gatewayHost, 1);
  403. setenv("H", remoteHost, 1);
  404. setenv("R", rport, 1);
  405. setenv("L", lport, 1);
  406. if (!cmd)
  407. cmd = "/usr/bin/ssh -f -L \"$L\":\"$H\":\"$R\" \"$G\" sleep 20";
  408. /* Compatibility with TigerVNC's method. */
  409. cmd2 = strdup(cmd);
  410. while ((percent = strchr(cmd2, '%')) != NULL)
  411. *percent = '$';
  412. system(cmd2);
  413. free(cmd2);
  414. }
  415. static int mktunnel()
  416. {
  417. const char *gatewayHost;
  418. char remoteHost[VNCSERVERNAMELEN];
  419. int localPort = findFreeTcpPort();
  420. int remotePort;
  421. if (interpretViaParam(remoteHost, &remotePort, localPort) != 0)
  422. return 1;
  423. gatewayHost = (const char*)via;
  424. createTunnel(gatewayHost, remoteHost, remotePort, localPort);
  425. return 0;
  426. }
  427. #endif /* !WIN32 */
  428. int main(int argc, char** argv)
  429. {
  430. UserDialog dlg;
  431. argv0 = argv[0];
  432. setlocale(LC_ALL, "");
  433. bindtextdomain(PACKAGE_NAME, CMAKE_INSTALL_FULL_LOCALEDIR);
  434. textdomain(PACKAGE_NAME);
  435. rfb::SecurityClient::setDefaults();
  436. // Write about text to console, still using normal locale codeset
  437. fprintf(stderr,"\n%s\n", about_text());
  438. // Set gettext codeset to what our GUI toolkit uses. Since we are
  439. // passing strings from strerror/gai_strerror to the GUI, these must
  440. // be in GUI codeset as well.
  441. bind_textdomain_codeset(PACKAGE_NAME, "UTF-8");
  442. bind_textdomain_codeset("libc", "UTF-8");
  443. rfb::initStdIOLoggers();
  444. #ifdef WIN32
  445. rfb::initFileLogger("C:\\temp\\vncviewer.log");
  446. #else
  447. rfb::initFileLogger("/tmp/vncviewer.log");
  448. #endif
  449. rfb::LogWriter::setLogParams("*:stderr:30");
  450. #ifdef SIGHUP
  451. signal(SIGHUP, CleanupSignalHandler);
  452. #endif
  453. signal(SIGINT, CleanupSignalHandler);
  454. signal(SIGTERM, CleanupSignalHandler);
  455. Configuration::enableViewerParams();
  456. /* Load the default parameter settings */
  457. char defaultServerName[VNCSERVERNAMELEN] = "";
  458. try {
  459. const char* configServerName;
  460. configServerName = loadViewerParameters(NULL);
  461. if (configServerName != NULL) {
  462. strncpy(defaultServerName, configServerName, VNCSERVERNAMELEN-1);
  463. defaultServerName[VNCSERVERNAMELEN-1] = '\0';
  464. }
  465. } catch (rfb::Exception& e) {
  466. vlog.error("%s", e.str());
  467. }
  468. for (int i = 1; i < argc;) {
  469. if (Configuration::setParam(argv[i])) {
  470. i++;
  471. continue;
  472. }
  473. if (argv[i][0] == '-') {
  474. if (i+1 < argc) {
  475. if (Configuration::setParam(&argv[i][1], argv[i+1])) {
  476. i += 2;
  477. continue;
  478. }
  479. }
  480. usage(argv[0]);
  481. }
  482. strncpy(vncServerName, argv[i], VNCSERVERNAMELEN);
  483. vncServerName[VNCSERVERNAMELEN - 1] = '\0';
  484. i++;
  485. }
  486. #if !defined(WIN32) && !defined(__APPLE__)
  487. if (strcmp(display, "") != 0) {
  488. Fl::display(display);
  489. }
  490. fl_open_display();
  491. XkbSetDetectableAutoRepeat(fl_display, True, NULL);
  492. #endif
  493. init_fltk();
  494. enable_touch();
  495. // Check if the server name in reality is a configuration file
  496. potentiallyLoadConfigurationFile(vncServerName);
  497. mkvnchomedir();
  498. CSecurity::upg = &dlg;
  499. #ifdef HAVE_GNUTLS
  500. CSecurityTLS::msg = &dlg;
  501. #endif
  502. Socket *sock = NULL;
  503. #ifndef WIN32
  504. /* Specifying -via and -listen together is nonsense */
  505. if (listenMode && strlen(via.getValueStr()) > 0) {
  506. // TRANSLATORS: "Parameters" are command line arguments, or settings
  507. // from a file or the Windows registry.
  508. vlog.error(_("Parameters -listen and -via are incompatible"));
  509. exit_vncviewer(_("Parameters -listen and -via are incompatible"));
  510. return 1; /* Not reached */
  511. }
  512. #endif
  513. if (listenMode) {
  514. std::list<SocketListener*> listeners;
  515. try {
  516. int port = 5500;
  517. if (isdigit(vncServerName[0]))
  518. port = atoi(vncServerName);
  519. createTcpListeners(&listeners, 0, port);
  520. vlog.info(_("Listening on port %d"), port);
  521. /* Wait for a connection */
  522. while (sock == NULL) {
  523. fd_set rfds;
  524. FD_ZERO(&rfds);
  525. for (std::list<SocketListener*>::iterator i = listeners.begin();
  526. i != listeners.end();
  527. i++)
  528. FD_SET((*i)->getFd(), &rfds);
  529. int n = select(FD_SETSIZE, &rfds, 0, 0, 0);
  530. if (n < 0) {
  531. if (errno == EINTR) {
  532. vlog.debug("Interrupted select() system call");
  533. continue;
  534. } else {
  535. throw rdr::SystemException("select", errno);
  536. }
  537. }
  538. for (std::list<SocketListener*>::iterator i = listeners.begin ();
  539. i != listeners.end();
  540. i++)
  541. if (FD_ISSET((*i)->getFd(), &rfds)) {
  542. sock = (*i)->accept();
  543. if (sock)
  544. /* Got a connection */
  545. break;
  546. }
  547. }
  548. } catch (rdr::Exception& e) {
  549. vlog.error("%s", e.str());
  550. exit_vncviewer(e.str());
  551. return 1; /* Not reached */
  552. }
  553. while (!listeners.empty()) {
  554. delete listeners.back();
  555. listeners.pop_back();
  556. }
  557. } else {
  558. if (vncServerName[0] == '\0') {
  559. ServerDialog::run(defaultServerName, vncServerName);
  560. if (vncServerName[0] == '\0')
  561. return 1;
  562. }
  563. #ifndef WIN32
  564. if (strlen (via.getValueStr()) > 0 && mktunnel() != 0)
  565. usage(argv[0]);
  566. #endif
  567. }
  568. CConn *cc = new CConn(vncServerName, sock);
  569. inMainloop = true;
  570. while (!exitMainloop)
  571. run_mainloop();
  572. inMainloop = false;
  573. delete cc;
  574. if (exitError != NULL && alertOnFatalError)
  575. fl_alert("%s", exitError);
  576. return 0;
  577. }