blob: 82286ab89f34409506521714f52e96a1602ab6e0 [file] [log] [blame]
Pierre Ossman5156d5e2011-03-09 09:42:34 +00001/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved.
2 * Copyright 2011 Pierre Ossman <ossman@cendio.se> for Cendio AB
DRCb65bb932011-06-24 03:17:00 +00003 * Copyright (C) 2011 D. R. Commander. All Rights Reserved.
Pierre Ossman5156d5e2011-03-09 09:42:34 +00004 *
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
Peter Åstrandc359f362011-08-23 12:04:46 +000021#ifdef HAVE_CONFIG_H
22#include <config.h>
23#endif
24
Pierre Ossman5156d5e2011-03-09 09:42:34 +000025#include <string.h>
26#include <stdio.h>
Pierre Ossman2a7a8d62013-02-15 08:33:39 +000027#include <ctype.h>
Pierre Ossman5156d5e2011-03-09 09:42:34 +000028#include <stdlib.h>
29#include <errno.h>
30#include <signal.h>
31#include <locale.h>
32#include <sys/stat.h>
33
34#ifdef WIN32
35#include <direct.h>
36#define mkdir(path, mode) _mkdir(path)
37#endif
38
Pierre Ossman6b9622d2014-07-21 16:42:12 +020039#if !defined(WIN32) && !defined(__APPLE__)
40#include <X11/Xlib.h>
41#include <X11/XKBlib.h>
42#endif
43
Pierre Ossman5156d5e2011-03-09 09:42:34 +000044#include <rfb/Logger_stdio.h>
45#include <rfb/SecurityClient.h>
46#include <rfb/Security.h>
47#ifdef HAVE_GNUTLS
48#include <rfb/CSecurityTLS.h>
49#endif
50#include <rfb/LogWriter.h>
51#include <rfb/Timer.h>
Peter Åstrand8a2b0812012-08-08 11:49:01 +000052#include <rfb/Exception.h>
Pierre Ossman5156d5e2011-03-09 09:42:34 +000053#include <network/TcpSocket.h>
DRC4426f002011-10-12 20:02:55 +000054#include <os/os.h>
Pierre Ossman5156d5e2011-03-09 09:42:34 +000055
56#include <FL/Fl.H>
57#include <FL/Fl_Widget.H>
Pierre Ossman8eb35082012-03-27 12:50:54 +000058#include <FL/Fl_PNG_Image.H>
Pierre Ossman5156d5e2011-03-09 09:42:34 +000059#include <FL/fl_ask.H>
Pierre Ossmanb8858222011-04-29 11:51:38 +000060#include <FL/x.H>
Pierre Ossman5156d5e2011-03-09 09:42:34 +000061
62#include "i18n.h"
63#include "parameters.h"
64#include "CConn.h"
Pierre Ossman561ff0c2011-05-13 14:04:59 +000065#include "ServerDialog.h"
Pierre Ossman5156d5e2011-03-09 09:42:34 +000066#include "UserDialog.h"
Adam Tkac8ac4b302013-01-23 13:55:46 +000067#include "vncviewer.h"
Pierre Ossman5156d5e2011-03-09 09:42:34 +000068
DRCb65bb932011-06-24 03:17:00 +000069#ifdef WIN32
Pierre Ossman8eb35082012-03-27 12:50:54 +000070#include "resource.h"
DRCb65bb932011-06-24 03:17:00 +000071#include "win32.h"
72#endif
73
Pierre Ossman5156d5e2011-03-09 09:42:34 +000074rfb::LogWriter vlog("main");
75
76using namespace network;
77using namespace rfb;
78using namespace std;
79
Pierre Ossmanf52740e2012-04-25 15:43:56 +000080static const char aboutText[] = N_("TigerVNC Viewer %d-bit v%s (%s)\n"
81 "%s\n"
Peter Åstrand2c2a1a22013-06-11 07:00:49 +000082 "Copyright (C) 1999-2013 TigerVNC Team and many others (see README.txt)\n"
Pierre Ossmanf52740e2012-04-25 15:43:56 +000083 "See http://www.tigervnc.org for information on TigerVNC.");
DRCd8e93dc2011-07-28 22:13:40 +000084extern const char* buildTime;
Pierre Ossman5156d5e2011-03-09 09:42:34 +000085
Adam Tkac8ac4b302013-01-23 13:55:46 +000086char vncServerName[VNCSERVERNAMELEN] = { '\0' };
87
Pierre Ossman5156d5e2011-03-09 09:42:34 +000088static bool exitMainloop = false;
Pierre Ossmane2ef5c12011-07-12 16:56:34 +000089static const char *exitError = NULL;
Pierre Ossman5156d5e2011-03-09 09:42:34 +000090
Pierre Ossmane2ef5c12011-07-12 16:56:34 +000091void exit_vncviewer(const char *error)
Pierre Ossman5156d5e2011-03-09 09:42:34 +000092{
Pierre Ossmane2ef5c12011-07-12 16:56:34 +000093 // Prioritise the first error we get as that is probably the most
94 // relevant one.
95 if ((error != NULL) && (exitError == NULL))
96 exitError = strdup(error);
97
Pierre Ossman5156d5e2011-03-09 09:42:34 +000098 exitMainloop = true;
99}
100
Pierre Ossmanb8858222011-04-29 11:51:38 +0000101void about_vncviewer()
102{
103 fl_message_title(_("About TigerVNC Viewer"));
Pierre Ossmanf52740e2012-04-25 15:43:56 +0000104 fl_message(gettext(aboutText), (int)sizeof(size_t)*8,
105 PACKAGE_VERSION, __BUILD__, buildTime);
Pierre Ossmanb8858222011-04-29 11:51:38 +0000106}
107
108static void about_callback(Fl_Widget *widget, void *data)
109{
110 about_vncviewer();
111}
112
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000113static void CleanupSignalHandler(int sig)
114{
115 // CleanupSignalHandler allows C++ object cleanup to happen because it calls
116 // exit() rather than the default which is to abort.
117 vlog.info("CleanupSignalHandler called");
118 exit(1);
119}
120
121static void init_fltk()
122{
123 // Basic text size (10pt @ 96 dpi => 13px)
124 FL_NORMAL_SIZE = 13;
125
126#ifndef __APPLE__
127 // Select a FLTK scheme and background color that looks somewhat
128 // close to modern Linux and Windows.
129 Fl::scheme("gtk+");
130 Fl::background(220, 220, 220);
131#else
132 // On Mac OS X there is another scheme that fits better though.
133 Fl::scheme("plastic");
134#endif
135
Henrik Andersson3b837032011-09-19 13:46:55 +0000136 // Proper Gnome Shell integration requires that we set a sensible
137 // WM_CLASS for the window.
138 Fl_Window::default_xclass("vncviewer");
139
Pierre Ossman8eb35082012-03-27 12:50:54 +0000140 // Set the default icon for all windows.
141#ifdef HAVE_FLTK_ICONS
142#ifdef WIN32
143 HICON lg, sm;
144
145 lg = (HICON)LoadImage(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_ICON),
146 IMAGE_ICON, GetSystemMetrics(SM_CXICON),
147 GetSystemMetrics(SM_CYICON),
148 LR_DEFAULTCOLOR | LR_SHARED);
149 sm = (HICON)LoadImage(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_ICON),
150 IMAGE_ICON, GetSystemMetrics(SM_CXSMICON),
151 GetSystemMetrics(SM_CYSMICON),
152 LR_DEFAULTCOLOR | LR_SHARED);
153
154 Fl_Window::default_icons(lg, sm);
155#elif ! defined(__APPLE__)
156 const int icon_sizes[] = {48, 32, 24, 16};
157
158 Fl_PNG_Image *icons[4];
159 int count;
160
161 count = 0;
162
163 // FIXME: Follow icon theme specification
164 for (size_t i = 0;i < sizeof(icon_sizes)/sizeof(icon_sizes[0]);i++) {
165 char icon_path[PATH_MAX];
166 bool exists;
167
Pierre Ossman6a007bd2012-09-11 10:56:21 +0000168 sprintf(icon_path, "%s/icons/hicolor/%dx%d/apps/tigervnc.png",
Pierre Ossman8eb35082012-03-27 12:50:54 +0000169 DATA_DIR, icon_sizes[i], icon_sizes[i]);
170
171#ifndef WIN32
172 struct stat st;
173 if (stat(icon_path, &st) != 0)
174#else
175 struct _stat st;
176 if (_stat(icon_path, &st) != 0)
177 return(false);
178#endif
179 exists = false;
180 else
181 exists = true;
182
183 if (exists) {
184 icons[count] = new Fl_PNG_Image(icon_path);
185 if (icons[count]->w() == 0 ||
186 icons[count]->h() == 0 ||
187 icons[count]->d() != 4) {
188 delete icons[count];
189 continue;
190 }
191
192 count++;
193 }
194 }
195
196 Fl_Window::default_icons((const Fl_RGB_Image**)icons, count);
197
198 for (int i = 0;i < count;i++)
199 delete icons[i];
200#endif
201#endif // FLTK_HAVE_ICONS
202
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000203 // This makes the "icon" in dialogs rounded, which fits better
204 // with the above schemes.
205 fl_message_icon()->box(FL_UP_BOX);
206
207 // Turn off the annoying behaviour where popups track the mouse.
208 fl_message_hotspot(false);
209
210 // Avoid empty titles for popups
211 fl_message_title_default(_("TigerVNC Viewer"));
212
213#ifdef WIN32
214 // Most "normal" Windows apps use this font for UI elements.
215 Fl::set_font(FL_HELVETICA, "Tahoma");
216#endif
217
218 // FLTK exposes these so that we can translate them.
219 fl_no = _("No");
220 fl_yes = _("Yes");
221 fl_ok = _("OK");
222 fl_cancel = _("Cancel");
223 fl_close = _("Close");
Pierre Ossmanb8858222011-04-29 11:51:38 +0000224
225#ifdef __APPLE__
Peter Åstrandb7c55242011-08-29 13:14:51 +0000226 /* Needs trailing space */
227 static char fltk_about[16];
228 snprintf(fltk_about, sizeof(fltk_about), "%s ", _("About"));
229 Fl_Mac_App_Menu::about = fltk_about;
230 static char fltk_hide[16];
231 snprintf(fltk_hide, sizeof(fltk_hide), "%s ", _("Hide"));
232 Fl_Mac_App_Menu::hide = fltk_hide;
233 static char fltk_quit[16];
234 snprintf(fltk_quit, sizeof(fltk_quit), "%s ", _("Quit"));
235 Fl_Mac_App_Menu::quit = fltk_quit;
236
Pierre Ossman41ba6032011-06-16 11:09:31 +0000237 Fl_Mac_App_Menu::print = ""; // Don't want the print item
238 Fl_Mac_App_Menu::services = _("Services");
Pierre Ossman41ba6032011-06-16 11:09:31 +0000239 Fl_Mac_App_Menu::hide_others = _("Hide Others");
240 Fl_Mac_App_Menu::show = _("Show All");
Pierre Ossman41ba6032011-06-16 11:09:31 +0000241
Pierre Ossmanb8858222011-04-29 11:51:38 +0000242 fl_mac_set_about(about_callback, NULL);
243#endif
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000244}
245
246static void mkvnchomedir()
247{
248 // Create .vnc in the user's home directory if it doesn't already exist
249 char* homeDir = NULL;
250
251 if (getvnchomedir(&homeDir) == -1) {
252 vlog.error(_("Could not create VNC home directory: can't obtain home "
253 "directory path."));
254 } else {
255 int result = mkdir(homeDir, 0755);
256 if (result == -1 && errno != EEXIST)
257 vlog.error(_("Could not create VNC home directory: %s."), strerror(errno));
258 delete [] homeDir;
259 }
260}
261
262static void usage(const char *programName)
263{
264 fprintf(stderr,
265 "\nusage: %s [parameters] [host:displayNum] [parameters]\n"
266 " %s [parameters] -listen [port] [parameters]\n",
267 programName, programName);
268 fprintf(stderr,"\n"
269 "Parameters can be turned on with -<param> or off with -<param>=0\n"
270 "Parameters which take a value can be specified as "
271 "-<param> <value>\n"
272 "Other valid forms are <param>=<value> -<param>=<value> "
273 "--<param>=<value>\n"
274 "Parameter names are case-insensitive. The parameters are:\n\n");
275 Configuration::listParams(79, 14);
276 exit(1);
277}
278
Adam Tkac8ac4b302013-01-23 13:55:46 +0000279#ifndef WIN32
280static int
281interpretViaParam(char *remoteHost, int *remotePort, int localPort)
282{
283 const int SERVER_PORT_OFFSET = 5900;
284 char *pos = strchr(vncServerName, ':');
285 if (pos == NULL)
286 *remotePort = SERVER_PORT_OFFSET;
287 else {
288 int portOffset = SERVER_PORT_OFFSET;
289 size_t len;
290 *pos++ = '\0';
291 len = strlen(pos);
292 if (*pos == ':') {
293 /* Two colons is an absolute port number, not an offset. */
294 pos++;
295 len--;
296 portOffset = 0;
297 }
298 if (!len || strspn (pos, "-0123456789") != len )
299 return 1;
300 *remotePort = atoi(pos) + portOffset;
301 }
302
303 if (*vncServerName != '\0')
304 strncpy(remoteHost, vncServerName, VNCSERVERNAMELEN);
305 else
306 strncpy(remoteHost, "localhost", VNCSERVERNAMELEN);
307
308 remoteHost[VNCSERVERNAMELEN - 1] = '\0';
309
310 snprintf(vncServerName, VNCSERVERNAMELEN, "localhost::%d", localPort);
311 vncServerName[VNCSERVERNAMELEN - 1] = '\0';
312 vlog.error(vncServerName);
313
314 return 0;
315}
316
317static void
318createTunnel(const char *gatewayHost, const char *remoteHost,
319 int remotePort, int localPort)
320{
Pierre Ossmanf8d525b2014-07-09 17:02:27 +0200321 const char *cmd = getenv("VNC_VIA_CMD");
322 char *cmd2, *percent;
Adam Tkac8ac4b302013-01-23 13:55:46 +0000323 char lport[10], rport[10];
324 sprintf(lport, "%d", localPort);
325 sprintf(rport, "%d", remotePort);
326 setenv("G", gatewayHost, 1);
327 setenv("H", remoteHost, 1);
328 setenv("R", rport, 1);
329 setenv("L", lport, 1);
Adam Tkac8ac4b302013-01-23 13:55:46 +0000330 if (!cmd)
331 cmd = "/usr/bin/ssh -f -L \"$L\":\"$H\":\"$R\" \"$G\" sleep 20";
332 /* Compatibility with TigerVNC's method. */
Pierre Ossmanf8d525b2014-07-09 17:02:27 +0200333 cmd2 = strdup(cmd);
334 while ((percent = strchr(cmd2, '%')) != NULL)
Adam Tkac8ac4b302013-01-23 13:55:46 +0000335 *percent = '$';
Pierre Ossmanf8d525b2014-07-09 17:02:27 +0200336 system(cmd2);
337 free(cmd2);
Adam Tkac8ac4b302013-01-23 13:55:46 +0000338}
339
340static int mktunnel()
341{
342 const char *gatewayHost;
343 char remoteHost[VNCSERVERNAMELEN];
344 int localPort = findFreeTcpPort();
345 int remotePort;
346
347 gatewayHost = strDup(via.getValueStr());
348 if (interpretViaParam(remoteHost, &remotePort, localPort) != 0)
349 return 1;
350 createTunnel(gatewayHost, remoteHost, remotePort, localPort);
351
352 return 0;
353}
354#endif /* !WIN32 */
355
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000356int main(int argc, char** argv)
357{
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000358 UserDialog dlg;
359
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000360 setlocale(LC_ALL, "");
Pierre Ossman0878eca2012-03-27 13:03:22 +0000361 bindtextdomain(PACKAGE_NAME, LOCALE_DIR);
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000362 textdomain(PACKAGE_NAME);
363
364 rfb::SecurityClient::setDefaults();
365
366 // Write about text to console, still using normal locale codeset
Pierre Ossmanf52740e2012-04-25 15:43:56 +0000367 fprintf(stderr,"\n");
368 fprintf(stderr, gettext(aboutText), (int)sizeof(size_t)*8,
369 PACKAGE_VERSION, __BUILD__, buildTime);
370 fprintf(stderr,"\n");
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000371
372 // Set gettext codeset to what our GUI toolkit uses. Since we are
373 // passing strings from strerror/gai_strerror to the GUI, these must
374 // be in GUI codeset as well.
375 bind_textdomain_codeset(PACKAGE_NAME, "UTF-8");
376 bind_textdomain_codeset("libc", "UTF-8");
377
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000378 rfb::initStdIOLoggers();
Pierre Ossmanc8719ad2012-04-26 14:27:52 +0000379#ifdef WIN32
380 rfb::initFileLogger("C:\\temp\\vncviewer.log");
381#else
382 rfb::initFileLogger("/tmp/vncviewer.log");
383#endif
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000384 rfb::LogWriter::setLogParams("*:stderr:30");
385
386#ifdef SIGHUP
387 signal(SIGHUP, CleanupSignalHandler);
388#endif
389 signal(SIGINT, CleanupSignalHandler);
390 signal(SIGTERM, CleanupSignalHandler);
391
392 init_fltk();
393
Pierre Ossman6b9622d2014-07-21 16:42:12 +0200394#if !defined(WIN32) && !defined(__APPLE__)
395 fl_open_display();
396 XkbSetDetectableAutoRepeat(fl_display, True, NULL);
397#endif
398
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000399 Configuration::enableViewerParams();
400
Peter Åstrand8a2b0812012-08-08 11:49:01 +0000401 /* Load the default parameter settings */
402 const char* defaultServerName;
403 try {
404 defaultServerName = loadViewerParameters(NULL);
405 } catch (rfb::Exception& e) {
406 fl_alert("%s", e.str());
407 }
408
DRC5aa06502011-06-23 22:04:46 +0000409 int i = 1;
410 if (!Fl::args(argc, argv, i) || i < argc)
411 for (; i < argc; i++) {
412 if (Configuration::setParam(argv[i]))
413 continue;
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000414
DRC5aa06502011-06-23 22:04:46 +0000415 if (argv[i][0] == '-') {
416 if (i+1 < argc) {
417 if (Configuration::setParam(&argv[i][1], argv[i+1])) {
418 i++;
419 continue;
420 }
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000421 }
DRC5aa06502011-06-23 22:04:46 +0000422 usage(argv[0]);
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000423 }
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000424
Adam Tkac8ac4b302013-01-23 13:55:46 +0000425 strncpy(vncServerName, argv[i], VNCSERVERNAMELEN);
426 vncServerName[VNCSERVERNAMELEN - 1] = '\0';
DRC5aa06502011-06-23 22:04:46 +0000427 }
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000428
429 if (!::autoSelect.hasBeenSet()) {
430 // Default to AutoSelect=0 if -PreferredEncoding or -FullColor is used
Peter Åstrand8a2b0812012-08-08 11:49:01 +0000431 if (::preferredEncoding.hasBeenSet() || ::fullColour.hasBeenSet() ||
432 ::fullColourAlias.hasBeenSet()) {
433 ::autoSelect.setParam(false);
434 }
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000435 }
436 if (!::fullColour.hasBeenSet() && !::fullColourAlias.hasBeenSet()) {
437 // Default to FullColor=0 if AutoSelect=0 && LowColorLevel is set
438 if (!::autoSelect && (::lowColourLevel.hasBeenSet() ||
439 ::lowColourLevelAlias.hasBeenSet())) {
440 ::fullColour.setParam(false);
441 }
442 }
443 if (!::customCompressLevel.hasBeenSet()) {
444 // Default to CustomCompressLevel=1 if CompressLevel is used.
Peter Åstrand8a2b0812012-08-08 11:49:01 +0000445 if(::compressLevel.hasBeenSet()) {
446 ::customCompressLevel.setParam(true);
447 }
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000448 }
449
450 mkvnchomedir();
451
452 CSecurity::upg = &dlg;
453#ifdef HAVE_GNUTLS
454 CSecurityTLS::msg = &dlg;
455#endif
456
Pierre Ossman2a7a8d62013-02-15 08:33:39 +0000457 Socket *sock = NULL;
458
DRC3e7ed812013-02-26 10:34:22 +0000459#ifndef WIN32
Adam Tkac571089b2013-02-19 14:30:32 +0000460 /* Specifying -via and -listen together is nonsense */
461 if (listenMode && strlen(via.getValueStr()) > 0) {
462 vlog.error("Parameters -listen and -via are incompatible");
463 fl_alert("Parameters -listen and -via are incompatible");
464 exit_vncviewer();
465 return 1;
466 }
DRC3e7ed812013-02-26 10:34:22 +0000467#endif
Adam Tkac571089b2013-02-19 14:30:32 +0000468
469 if (listenMode) {
Pierre Ossman2a7a8d62013-02-15 08:33:39 +0000470 try {
471 int port = 5500;
472 if (isdigit(vncServerName[0]))
473 port = atoi(vncServerName);
474
475 TcpListener listener(NULL, port);
476
477 vlog.info("Listening on port %d\n", port);
478 sock = listener.accept();
479 } catch (rdr::Exception& e) {
480 vlog.error("%s", e.str());
481 fl_alert("%s", e.str());
482 exit_vncviewer();
483 return 1;
484 }
485
486 } else {
487 if (vncServerName[0] == '\0') {
488 ServerDialog::run(defaultServerName, vncServerName);
489 if (vncServerName[0] == '\0')
490 return 1;
491 }
Pierre Ossman561ff0c2011-05-13 14:04:59 +0000492
Adam Tkac8ac4b302013-01-23 13:55:46 +0000493#ifndef WIN32
Pierre Ossman2a7a8d62013-02-15 08:33:39 +0000494 if (strlen (via.getValueStr()) > 0 && mktunnel() != 0)
495 usage(argv[0]);
Adam Tkac8ac4b302013-01-23 13:55:46 +0000496#endif
Pierre Ossman2a7a8d62013-02-15 08:33:39 +0000497 }
Adam Tkac8ac4b302013-01-23 13:55:46 +0000498
Pierre Ossman2a7a8d62013-02-15 08:33:39 +0000499 CConn *cc = new CConn(vncServerName, sock);
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000500
501 while (!exitMainloop) {
502 int next_timer;
503
504 next_timer = Timer::checkTimeouts();
505 if (next_timer == 0)
506 next_timer = INT_MAX;
507
508 if (Fl::wait((double)next_timer / 1000.0) < 0.0) {
509 vlog.error(_("Internal FLTK error. Exiting."));
510 break;
511 }
512 }
513
Pierre Ossmane2ef5c12011-07-12 16:56:34 +0000514 delete cc;
515
516 if (exitError != NULL)
Pierre Ossmanf52740e2012-04-25 15:43:56 +0000517 fl_alert("%s", exitError);
Pierre Ossmane2ef5c12011-07-12 16:56:34 +0000518
Pierre Ossman5156d5e2011-03-09 09:42:34 +0000519 return 0;
520}