summaryrefslogtreecommitdiffstats
path: root/x11vnc/x11vnc.c
diff options
context:
space:
mode:
authorrunge <runge>2004-12-17 03:55:35 +0000
committerrunge <runge>2004-12-17 03:55:35 +0000
commit3a84b0ccc87a023be3ac84fba82895fafade30c1 (patch)
tree054ec71b3ad716623433ed8b8e39da417afb71fe /x11vnc/x11vnc.c
parent7e13b8a594805e25e928b8bcac394eef0a1ee5c8 (diff)
downloadlibtdevnc-3a84b0ccc87a023be3ac84fba82895fafade30c1.tar.gz
libtdevnc-3a84b0ccc87a023be3ac84fba82895fafade30c1.zip
x11vnc: XFIXES cursorshape, XRANDR resize, remote control, gui
Diffstat (limited to 'x11vnc/x11vnc.c')
-rw-r--r--x11vnc/x11vnc.c6876
1 files changed, 5818 insertions, 1058 deletions
diff --git a/x11vnc/x11vnc.c b/x11vnc/x11vnc.c
index 5df5db3..c26db31 100644
--- a/x11vnc/x11vnc.c
+++ b/x11vnc/x11vnc.c
@@ -47,27 +47,33 @@
* gcc should be used on all platforms. To build a threaded version put
* "-D_REENTRANT -DX11VNC_THREADED" in the environment variable CFLAGS
* or CPPFLAGS (e.g. before running the libvncserver configure). The
- * threaded mode is a bit more responsive, but can be unstable.
+ * threaded mode is a bit more responsive, but can be unstable (e.g.
+ * if more than one client the same tight or zrle encoding).
*
* Known shortcomings:
*
* The screen updates are good, but of course not perfect since the X
* display must be continuously polled and read for changes (as opposed to
* receiving a change callback from the X server, if that were generally
- * possible...). So, e.g., opaque moves and similar window activity
- * can be very painful; one has to modify one's behavior a bit.
+ * possible... (Update: this seems to be handled now with the X DAMAGE
+ * extension, but unfortunately that doesn't seem to address the slow
+ * read from the video h/w). So, e.g., opaque moves and similar window
+ * activity can be very painful; one has to modify one's behavior a bit.
*
* General audio at the remote display is lost unless one separately
* sets up some audio side-channel such as esd.
*
* It does not appear possible to query the X server for the current
* cursor shape. We can use XTest to compare cursor to current window's
- * cursor, but we cannot extract what the cursor is...
+ * cursor, but we cannot extract what the cursor is... (Update: we now
+ * use XFIXES extension for this. Also on Solaris and IRIX Overlay
+ * extensions exists that allow drawing the mouse into the framebuffer)
*
- * Nevertheless, the current *position* of the remote X mouse pointer
- * is shown with the -cursor option. Further, if -cursorX or -X is used, a
- * trick is done to at least show the root window cursor vs non-root cursor.
- * (perhaps some heuristic can be done to further distinguish cases...)
+ * The current *position* of the remote X mouse pointer is shown with
+ * the -cursor option. Further, if -cursorX or -X is used, a trick
+ * is done to at least show the root window cursor vs non-root cursor.
+ * (perhaps some heuristic can be done to further distinguish cases...,
+ * currently -cursor some is a first hack at this)
*
* Windows using visuals other than the default X visual may have
* their colors messed up. When using 8bpp indexed color, the colormap
@@ -75,11 +81,11 @@
* -flashcmap option to have colormap flashing as the pointer moves
* windows with private colormaps (slow). Displays with mixed depth 8 and
* 24 visuals will incorrectly display windows using the non-default one.
- * On Sun hardware we try to work around this with -overlay.
+ * On Sun and Sgi hardware we can to work around this with -overlay.
*
* Feature -id <windowid> can be picky: it can crash for things like the
- * window not sufficiently mapped into server memory, use of -cursor, etc.
- * SaveUnders menus, popups, etc will not be seen.
+ * window not sufficiently mapped into server memory, etc. SaveUnders
+ * menus, popups, etc will not be seen.
*
* Occasionally, a few tile updates can be missed leaving a patch of
* color that needs to be refreshed. This may only be when threaded,
@@ -92,11 +98,11 @@
*/
/*
- * These ' -- filename -- ' comments represent a partial cleanup:
+ * These ' -- filename.[ch] -- ' comments represent a partial cleanup:
* they are an odd way to indicate how this huge file would be split up
* someday into multiple files. Not finished, externs and other things
- * would need to be done, but it indicates the breakup, including static
- * keyword for local items.
+ * would need to be done, but it indicates a breakup, including static
+ * keyword for some items.
*
* The primary reason we do not break up this file is for user
* convenience: those wanting to use the latest version download a single
@@ -105,52 +111,37 @@
/* -- x11vnc.h -- */
-#define NON_CVS 0
/*
- * At some point beyond 0.7pre remove these two definitions since we
- * have them set in configure (for all users of this x11vnc.c file).
- * Then move them to the comment below.
+ * if you are inserting this file, x11vnc.c into an old CVS tree you
+ * may need to set OLD_TREE to 1.
*/
-#if NON_CVS
-#define LIBVNCSERVER_HAVE_XSHM
-#define LIBVNCSERVER_HAVE_XTEST
-#endif
+#define OLD_TREE 0
+#if OLD_TREE
/*
* If you are building in an older libvncserver tree with this newer
- * x11vnc.c file you may need to uncomment some of these lines since
- * your older libvncserver configure is not setting them.
- *
- * For LIBVNCSERVER_HAVE_LIBXINERAMA you may also need to add to the
- * linking -lXinerama (by setting LDFLAGS=-lXinerama before configure).
- *
-#define LIBVNCSERVER_HAVE_LIBXINERAMA
-#define LIBVNCSERVER_HAVE_XFIXES
-#define LIBVNCSERVER_HAVE_XDAMAGE
+ * x11vnc.c file using OLD_TREE=1 you may need to set some of these lines
+ * since your older libvncserver configure is not setting them.
*
+ * For the features LIBVNCSERVER_HAVE_LIBXINERAMA and
+ * LIBVNCSERVER_HAVE_XFIXES you may also need to add
+ * -lXinerama or -lXfixes, respectively, to the linking line, e.g.
+ * by setting them in LD_FLAGS before running configure.
*/
+#define LIBVNCSERVER_HAVE_XSHM
+#define LIBVNCSERVER_HAVE_XTEST
/*
- * This is another transient for building in older libvncserver trees.
+#define LIBVNCSERVER_HAVE_LIBXINERAMA
+#define LIBVNCSERVER_HAVE_XFIXES
+#define LIBVNCSERVER_HAVE_XDAMAGE
*/
-#if NON_CVS
-#define dontDisconnect rfbDontDisconnect
-#define neverShared rfbNeverShared
-#define alwaysShared rfbAlwaysShared
-#define clientHead rfbClientHead
-#define serverFormat rfbServerFormat
-#define port rfbPort
-#define listenSock rfbListenSock
-#define deferUpdateTime rfbDeferUpdateTime
-#define authPasswdData rfbAuthPasswdData
-#define rfbEncryptAndStorePasswd vncEncryptAndStorePasswd
#endif
#include <unistd.h>
#include <signal.h>
#include <sys/utsname.h>
-
#include <X11/Xlib.h>
#include <X11/Xutil.h>
@@ -164,6 +155,23 @@
#include <rfb/rfb.h>
#include <rfb/rfbregion.h>
+/*
+ * This is another transient for building in older libvncserver trees,
+ * due to the API change:
+ */
+#if OLD_TREE
+#define dontDisconnect rfbDontDisconnect
+#define neverShared rfbNeverShared
+#define alwaysShared rfbAlwaysShared
+#define clientHead rfbClientHead
+#define serverFormat rfbServerFormat
+#define port rfbPort
+#define listenSock rfbListenSock
+#define deferUpdateTime rfbDeferUpdateTime
+#define authPasswdData rfbAuthPasswdData
+#define rfbEncryptAndStorePasswd vncEncryptAndStorePasswd
+#endif
+
#ifdef LIBVNCSERVER_HAVE_XSHM
#include <sys/ipc.h>
#include <sys/shm.h>
@@ -182,23 +190,73 @@
#include <X11/extensions/Xinerama.h>
#endif
-#if defined (__SVR4) && defined (__sun)
-#define SOLARIS
-#include <X11/extensions/transovl.h>
-#endif
-
#ifdef LIBVNCSERVER_HAVE_SYS_SOCKET_H
#include <sys/socket.h>
#endif
+#include <netdb.h>
+extern int h_errno;
+
#ifdef LIBVNCSERVER_HAVE_NETINET_IN_H
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <arpa/inet.h>
#endif
+/*
+ * overlay/multi-depth screen reading support
+ * undef SOLARIS_OVERLAY or IRIX_OVERLAY if there are problems building.
+ */
+
+/* solaris/sun */
+#if defined (__SVR4) && defined (__sun)
+#define SOLARIS
+#define SOLARIS_OVERLAY
+#define OVERLAY_OS
+#endif
+#ifdef SOLARIS_OVERLAY
+#include <X11/extensions/transovl.h>
+#endif
+
+/* irix/sgi */
+#if defined(__sgi)
+#define IRIX
+#define IRIX_OVERLAY
+#define OVERLAY_OS
+#endif
+#ifdef IRIX_OVERLAY
+#include <X11/extensions/readdisplay.h>
+#endif
+
+int overlay_present = 0;
+
+/*
+ * Ditto for librandr.
+ * (e.g. LDFLAGS=-lXrandr before configure).
+#define LIBVNCSERVER_HAVE_LIBXRANDR
+ */
+#ifdef LIBVNCSERVER_HAVE_LIBXRANDR
+#include <X11/extensions/Xrandr.h>
+static int xrandr_base_event_type;
+#endif
+
+int xfixes_present = 0;
+int use_xfixes = 1;
+int got_xfixes_cursor_notify = 0;
+
+#ifdef LIBVNCSERVER_HAVE_LIBXFIXES
+#include <X11/extensions/Xfixes.h>
+static int xfixes_base_event_type;
+#endif
+
+int xdamage_present = 0;
+#ifdef LIBVNCSERVER_HAVE_LIBXDAMAGE
+#include <X11/extensions/Xdamage.h>
+static int xdamage_base_event_type;
+#endif
+
/* date +'lastmod: %Y-%m-%d' */
-char lastmod[] = "0.6.3pre lastmod: 2004-08-31";
+char lastmod[] = "0.6.3pre lastmod: 2004-12-17";
/* X display info */
@@ -207,7 +265,7 @@ int scr;
Window window, rootwin; /* polled window, root window (usu. same) */
Visual *default_visual; /* the default visual (unless -visual) */
int bpp, depth;
-int indexed_colour = 0;
+int indexed_color = 0;
int dpy_x, dpy_y; /* size of display */
int off_x, off_y; /* offsets for -sid */
int button_mask = 0; /* button state and info */
@@ -217,10 +275,11 @@ int num_buttons = -1;
XImage *scanline;
XImage *fullscreen;
XImage **tile_row; /* for all possible row runs */
+XImage *fb0;
#ifndef LIBVNCSERVER_HAVE_XSHM
/*
- * for simplicity, define these since we'll never use them
+ * for simplicity, define this struct since we'll never use them
* under using_shm = 0.
*/
typedef struct {
@@ -234,16 +293,27 @@ XShmSegmentInfo fullscreen_shm;
XShmSegmentInfo *tile_row_shm; /* for all possible row runs */
/* rfb screen info */
-rfbScreenInfoPtr screen;
+rfbScreenInfoPtr screen = NULL;
+char *rfb_desktop_name = NULL;
+char vnc_desktop_name[256];
char *main_fb; /* our copy of the X11 fb */
char *rfb_fb; /* same as main_fb unless transformation */
+char *fake_fb = NULL; /* used under -padgeom */
int rfb_bytes_per_line;
int main_bytes_per_line;
unsigned long main_red_mask, main_green_mask, main_blue_mask;
unsigned short main_red_max, main_green_max, main_blue_max;
unsigned short main_red_shift, main_green_shift, main_blue_shift;
+/* we now have a struct with client specific data: */
+typedef struct _ClientData {
+ int had_cursor_shape_updates;
+ int had_cursor_pos_updates;
+ int uid;
+} ClientData;
+
/* scaling parameters */
+char *scale_str = NULL;
double scale_fac = 1.0;
int scaling = 0;
int scaling_noblend = 0; /* no blending option (very course) */
@@ -253,7 +323,6 @@ int scaling_interpolate = 0; /* use interpolation scheme when shrinking */
int scaled_x = 0, scaled_y = 0; /* dimensions of scaled display */
int scale_numer = 0, scale_denom = 0; /* n/m */
-
/* size of the basic tile unit that is polled for changes: */
int tile_x = 32;
int tile_y = 32;
@@ -268,30 +337,46 @@ time_t last_event, last_input, last_client = 0;
/* last client to move pointer */
rfbClientPtr last_pointer_client = NULL;
+/* more transient kludge variables: */
int cursor_x, cursor_y; /* x and y from the viewer(s) */
int got_user_input = 0;
int got_pointer_input = 0;
int got_keyboard_input = 0;
int last_keyboard_input = 0;
int fb_copy_in_progress = 0;
+int drag_in_progress = 0;
int shut_down = 0;
+int do_copy_screen = 0;
+time_t damage_time = 0;
+int damage_delay = 0;
+
+char *program_name = NULL;
+char *program_cmdline = NULL;
/* string for the VNC_CONNECT property */
-#define VNC_CONNECT_MAX 512
+#define VNC_CONNECT_MAX 16384
char vnc_connect_str[VNC_CONNECT_MAX+1];
Atom vnc_connect_prop = None;
/* function prototypes (see filename comment above) */
int all_clients_initialized(void);
+void close_all_clients(void);
+void close_clients(char *);
void autorepeat(int restore);
-char *bitprint(unsigned int);
+char *bitprint(unsigned int, int);
void blackout_tiles(void);
void check_connect_inputs(void);
+void check_padded_fb(void);
void clean_up_exit(int);
void clear_modifiers(int init);
void clear_keys(void);
-void copy_screen(void);
+int copy_screen(void);
+void check_black_fb(void);
+void do_new_fb(int);
+void install_padded_fb(char *);
+void install_fake_fb(int, int, int);
+void remove_fake_fb(void);
int add_keysym(KeySym);
void delete_keycode(KeyCode);
@@ -299,16 +384,23 @@ void delete_added_keycodes(void);
double dtime(double *);
-void initialize_blackout(char *);
+void initialize_blackouts(char *);
+void initialize_blackouts_and_xinerama(void);
+void initialize_keyboard_and_pointer(void);
void initialize_modtweak(void);
void initialize_pointer_map(char *);
+void initialize_cursors_mode(void);
void initialize_remap(char *);
void initialize_screen(int *argc, char **argv, XImage *fb);
-void initialize_shm(void);
+void initialize_polling_images(void);
void initialize_signals(void);
void initialize_tiles(void);
+void free_tiles(void);
void initialize_watch_bell(void);
void initialize_xinerama(void);
+void initialize_xfixes(void);
+void initialize_xrandr(void);
+XImage *initialize_xdisplay_fb(void);
void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client);
@@ -335,14 +427,24 @@ void pointer(int mask, int x, int y, rfbClientPtr client);
void cursor_position(int, int);
void read_vnc_connect_prop(void);
+void set_vnc_connect_prop(char *);
+char *process_remote_cmd(char *, int);
void rfbPE(rfbScreenInfoPtr, long);
void rfbCFD(rfbScreenInfoPtr, long);
int scan_for_updates(void);
-void set_colormap(void);
+void set_colormap(int);
void set_offset(void);
+void set_rfb_cursor(int);
void set_visual(char *vstring);
void set_cursor(int, int, int);
+void set_no_cursor(void);
+void set_cursor_was_changed(rfbScreenInfoPtr);
int get_which_cursor(void);
+int get_xfixes_cursor(int);
+
+void disable_cursor_shape_updates(rfbScreenInfoPtr);
+void restore_cursor_shape_updates(rfbScreenInfoPtr);
+int new_fb_size_clients(rfbScreenInfoPtr);
void shm_clean(XShmSegmentInfo *, XImage *);
void shm_delete(XShmSegmentInfo *);
@@ -350,18 +452,40 @@ void shm_delete(XShmSegmentInfo *);
void check_x11_pointer(void);
void check_bell_event(void);
void check_xevents(void);
+char *this_host(void);
+
+int get_remote_port(int sock);
+int get_local_port(int sock);
+char *get_remote_host(int sock);
+char *get_local_host(int sock);
void xcut_receive(char *text, int len, rfbClientPtr client);
void zero_fb(int, int, int, int);
+void push_black_screen(int);
+void push_sleep(int);
+void refresh_screen(void);
/* -- options.h -- */
/*
* variables for the command line options
*/
+char *use_dpy = NULL;
+char *auth_file = NULL;
+char *visual_str = NULL;
+char *logfile = NULL;
+char *passwdfile = NULL;
+char *blackout_string = NULL;
+char *rc_rcfile = NULL;
+int rc_norc = 0;
+int opts_bg = 0;
int shared = 0; /* share vnc display. */
+int deny_all = 0; /* global locking of new clients */
+int accept_remote_cmds = 1; /* -noremote */
+int safe_remote_only = 0; /* -safer, -unsafe */
char *allow_list = NULL; /* for -allow and -localhost */
+char *allow_once = NULL; /* one time -allow */
char *accept_cmd = NULL; /* for -accept */
char *gone_cmd = NULL; /* for -gone */
int view_only = 0; /* clients can only watch. */
@@ -370,15 +494,26 @@ int inetd = 0; /* spawned from inetd(1) */
int connect_once = 1; /* disconnect after first connection session. */
int flash_cmap = 0; /* follow installed colormaps */
int force_indexed_color = 0; /* whether to force indexed color for 8bpp */
+int launch_gui = 0; /* -gui */
int use_modifier_tweak = 1; /* use the shift/altgr modifier tweak */
int use_iso_level3 = 0; /* ISO_Level3_Shift instead of Mode_switch */
int clear_mods = 0; /* -clear_mods (1) and -clear_keys (2) */
int nofb = 0; /* do not send any fb updates */
-int subwin = 0; /* -id */
+unsigned long subwin = 0x0; /* -id, -sid */
int xinerama = 0; /* -xinerama */
+int xrandr = 0; /* -xrandr */
+int xrandr_present = 0;
+int xrandr_width = -1;
+int xrandr_height = -1;
+int xrandr_rotation = -1;
+Time xrandr_timestamp = 0;
+Time xrandr_cfg_time = 0;
+char *xrandr_mode = NULL;
+char *pad_geometry = NULL;
+time_t pad_geometry_time;
char *client_connect = NULL; /* strings for -connect option */
char *client_connect_file = NULL;
@@ -386,21 +521,28 @@ int vnc_connect = 1; /* -vncconnect option */
int show_cursor = 1; /* show cursor shapes */
int show_multiple_cursors = 0; /* show X when on root background, etc */
-char *multiple_cursors_mode = "default";
+char *multiple_cursors_mode = NULL;
int cursor_pos_updates = 1; /* cursor position updates -cursorpos */
int cursor_shape_updates = 1; /* cursor shape updates -nocursorshape */
int use_xwarppointer = 0; /* use XWarpPointer instead of XTestFake... */
int show_dragging = 1; /* process mouse movement events */
-int no_autorepeat = 0; /* turn off autorepeat with clients */
+int no_autorepeat = 1; /* turn off autorepeat with clients */
int watch_bell = 1; /* watch for the bell using XKEYBOARD */
+int sound_bell = 1; /* actually send it */
int xkbcompat = 0; /* ignore XKEYBOARD extension */
int use_xkb = 0; /* try to open Xkb connection (for bell or other) */
int use_xkb_modtweak = 0; /* -xkb */
char *skip_keycodes = NULL;
int add_keysyms = 0; /* automatically add keysyms to X server */
-int old_pointer = 0; /* use the old ways of updating the pointer */
+char *remap_file = NULL; /* -remap */
+char *pointer_remap = NULL;
+int pointer_mode = 2; /* use the various ways of updating pointer */
+int pointer_mode_max = 4;
int single_copytile = 0; /* use the old way copy_tiles() */
+int single_copytile_orig = 0;
+int single_copytile_count = 0;
+int tile_shm_count = 0;
int using_shm = 1; /* whether mit-shm is used */
int flip_byte_order = 0; /* sometimes needed when using_shm = 0 */
@@ -422,21 +564,33 @@ int ui_skip = 10; /* see watchloop. negative means ignore input */
int watch_selection = 1; /* normal selection/cutbuffer maintenance */
int watch_primary = 1; /* more dicey, poll for changes in PRIMARY */
-int sigpipe = 1; /* 0=skip, 1=ignore, 2=exit */
+char *sigpipe = NULL; /* skip, ignore, exit */
/* visual stuff for -visual override or -overlay */
VisualID visual_id = (VisualID) 0;
int visual_depth = 0;
-/* for -overlay mode on Solaris. X server draws cursor correctly. */
+/* for -overlay mode on Solaris/IRIX. X server draws cursor correctly. */
int overlay = 0;
int overlay_cursor = 1;
+#ifdef LIBVNCSERVER_HAVE_XSHM
+int xshm_present = 1;
+#else
+int xshm_present = 0;
+#endif
#ifdef LIBVNCSERVER_HAVE_XTEST
int xtest_present = 1;
#else
int xtest_present = 0;
#endif
+#ifdef LIBVNCSERVER_HAVE_XKEYBOARD
+int xkb_present = 1;
+#else
+int xkb_present = 0;
+#endif
+int xinerama_present = 0;
+
/* tile heuristics: */
double fs_frac = 0.75; /* threshold tile fraction to do fullscreen updates. */
@@ -469,6 +623,7 @@ int got_nevershared = 0;
/* -- util.h -- */
+#define NONUL(x) ((x) ? (x) : "")
/* XXX usleep(3) is not thread safe on some older systems... */
struct timeval _mysleep;
@@ -497,7 +652,7 @@ MUTEX(x11Mutex);
#define X_UNLOCK UNLOCK(x11Mutex)
#define X_INIT INIT_MUTEX(x11Mutex)
-/* -- util.c -- ? */
+/* -- util.c -- */
/*
* routine to keep 0 <= i < n, should use in more places...
@@ -511,6 +666,14 @@ int nfix(int i, int n) {
return i;
}
+int nabs(int n) {
+ if (n < 0) {
+ return -n;
+ } else {
+ return n;
+ }
+}
+
void lowercase(char *str) {
char *p;
if (str == NULL) {
@@ -523,6 +686,235 @@ void lowercase(char *str) {
}
}
+int scan_hexdec(char *str, unsigned long *num) {
+ if (sscanf(str, "0x%lx", num) != 1) {
+ if (sscanf(str, "%ld", num) != 1) {
+ return 0;
+ }
+ }
+ return 1;
+}
+
+void set_env(char *name, char *value) {
+ char *str;
+ str = malloc(strlen(name)+strlen(value)+2);
+ sprintf(str, "%s=%s", name, value);
+ putenv(str);
+}
+
+int pick_windowid(unsigned long *num) {
+ char line[512];
+ int ok = 0, n = 0, msec = 10, secmax = 15;
+ FILE *p;
+
+ if (use_dpy) {
+ set_env("DISPLAY", use_dpy);
+ }
+ p = popen("xwininfo", "r");
+
+ if (! p) {
+ return 0;
+ }
+
+ fprintf(stderr, "\n");
+ fprintf(stderr, " Please select the window for x11vnc to poll\n");
+ fprintf(stderr, " by clicking the mouse in that window.\n");
+ fprintf(stderr, "\n");
+
+ while (msec * n++ < 1000 * secmax) {
+ unsigned long tmp;
+ char *q;
+ fd_set set;
+ struct timeval tv;
+
+ if (screen && screen->clientHead) {
+ /* they may be doing the pointer-pick thru vnc: */
+ tv.tv_sec = 0;
+ tv.tv_usec = msec * 1000;
+ FD_ZERO(&set);
+ FD_SET(fileno(p), &set);
+ if (select(fileno(p)+1, &set, NULL, NULL, &tv) == 0) {
+ /* note that rfbPE takes about 30ms too */
+ rfbPE(screen, -1);
+ continue;
+ }
+ }
+
+ if (fgets(line, 512, p) == NULL) {
+ break;
+ }
+ q = strstr(line, " id: 0x");
+ if (q) {
+ q += 5;
+ if (sscanf(q, "0x%lx ", &tmp) == 1) {
+ ok = 1;
+ *num = tmp;
+ fprintf(stderr, " Picked: 0x%lx\n\n", tmp);
+ break;
+ }
+ }
+ }
+ pclose(p);
+ return ok;
+}
+
+char *bitprint(unsigned int st, int nbits) {
+ static char str[33];
+ int i, mask;
+ if (nbits > 32) {
+ nbits = 32;
+ }
+ for (i=0; i<nbits; i++) {
+ str[i] = '0';
+ }
+ str[nbits] = '\0';
+ mask = 1;
+ for (i=nbits-1; i>=0; i--) {
+ if (st & mask) {
+ str[i] = '1';
+ }
+ mask = mask << 1;
+ }
+ return str; /* take care to use or copy immediately */
+}
+
+/*
+ * Simple utility to map host name to dotted IP address. Ignores aliases.
+ * Up to caller to free returned string.
+ */
+char *host2ip(char *host) {
+ struct hostent *hp;
+ struct sockaddr_in addr;
+ char *str;
+
+ hp = gethostbyname(host);
+ if (!hp) {
+ return NULL;
+ }
+ memset(&addr, 0, sizeof(addr));
+ addr.sin_family = AF_INET;
+ addr.sin_addr.s_addr = *(unsigned long *)hp->h_addr;
+ str = strdup(inet_ntoa(addr.sin_addr));
+ return str;
+}
+
+int dotted_ip(char *host) {
+ char *p = host;
+ while (*p != '\0') {
+ if (*p == '.' || isdigit(*p)) {
+ p++;
+ continue;
+ }
+ return 0;
+ }
+ return 1;
+}
+
+int get_remote_port(int sock) {
+ struct sockaddr_in saddr;
+ int saddr_len, saddr_port;
+
+ saddr_len = sizeof(saddr);
+ memset(&saddr, 0, sizeof(saddr));
+ saddr_port = -1;
+ if (!getpeername(sock, (struct sockaddr *)&saddr, &saddr_len)) {
+ saddr_port = ntohs(saddr.sin_port);
+ }
+ return saddr_port;
+}
+
+char *get_remote_host(int sock) {
+ struct sockaddr_in saddr;
+ int saddr_len, saddr_port;
+ char *saddr_ip_str = NULL;
+
+ saddr_len = sizeof(saddr);
+ memset(&saddr, 0, sizeof(saddr));
+ saddr_port = -1;
+ if (!getsockname(sock, (struct sockaddr *)&saddr, &saddr_len)) {
+#ifdef LIBVNCSERVER_HAVE_NETINET_IN_H
+ saddr_ip_str = inet_ntoa(saddr.sin_addr);
+#endif
+ }
+ if (! saddr_ip_str) {
+ saddr_ip_str = strdup("unknown");
+ }
+ return saddr_ip_str;
+}
+
+int get_local_port(int sock) {
+ struct sockaddr_in saddr;
+ int saddr_len, saddr_port;
+
+ saddr_len = sizeof(saddr);
+ memset(&saddr, 0, sizeof(saddr));
+ saddr_port = -1;
+ if (!getsockname(sock, (struct sockaddr *)&saddr, &saddr_len)) {
+ saddr_port = ntohs(saddr.sin_port);
+ }
+ return saddr_port;
+}
+
+char *get_local_host(int sock) {
+ struct sockaddr_in saddr;
+ int saddr_len, saddr_port;
+ char *saddr_ip_str = NULL;
+
+ saddr_len = sizeof(saddr);
+ memset(&saddr, 0, sizeof(saddr));
+ saddr_port = -1;
+ if (!getsockname(sock, (struct sockaddr *)&saddr, &saddr_len)) {
+#ifdef LIBVNCSERVER_HAVE_NETINET_IN_H
+ saddr_ip_str = inet_ntoa(saddr.sin_addr);
+#endif
+ }
+ if (! saddr_ip_str) {
+ saddr_ip_str = strdup("unknown");
+ }
+ return saddr_ip_str;
+}
+
+/*
+ * used in rfbGetScreen and rfbNewFramebuffer: and estimate to the number
+ * of bits per color, of course for some visuals, e.g. 565, the number
+ * is not the same for each color. This is just a sane default.
+ */
+int guess_bits_per_color(int bits_per_pixel) {
+ int bits_per_color;
+
+ /* first guess, spread them "evenly" over R, G, and B */
+ bits_per_color = bits_per_pixel/3;
+ if (bits_per_color < 1) {
+ bits_per_color = 1; /* 1bpp, 2bpp... */
+ }
+
+ /* choose safe values for usual cases: */
+ if (bits_per_pixel == 8) {
+ bits_per_color = 2;
+ } else if (bits_per_pixel == 15 || bits_per_pixel == 16) {
+ bits_per_color = 5;
+ } else if (bits_per_pixel == 24 || bits_per_pixel == 32) {
+ bits_per_color = 8;
+ }
+ return bits_per_color;
+}
+
+/* count number of clients supporting NewFBSize */
+int new_fb_size_clients(rfbScreenInfoPtr s) {
+ rfbClientIteratorPtr iter;
+ rfbClientPtr cl;
+ int count = 0;
+
+ iter = rfbGetClientIterator(s);
+ while( (cl = rfbClientIteratorNext(iter)) ) {
+ if (cl->useNewFBSize) {
+ count++;
+ }
+ }
+ rfbReleaseClientIterator(iter);
+ return count;
+}
+
/*
* Kludge to interpose image gets and limit to a subset rectangle of
* the rootwin. This is the -sid option trying to work around invisible
@@ -537,6 +929,8 @@ int rootshift = 0;
y += off_y; \
}
+/* -- ximage.c -- */
+
/*
* Wrappers for Image related X calls
*/
@@ -545,7 +939,7 @@ Status XShmGetImage_wr(Display *disp, Drawable d, XImage *image, int x, int y,
ADJUST_ROOTSHIFT
- /* The Solaris overlay stuff is all non-shm (using_shm = 0) */
+ /* Note: the Solaris overlay stuff is all non-shm (using_shm = 0) */
#ifdef LIBVNCSERVER_HAVE_XSHM
return XShmGetImage(disp, d, image, x, y, mask);
@@ -590,18 +984,39 @@ Bool XShmQueryExtension_wr(Display *disp) {
#endif
}
+/* wrapper for overlay screen reading: */
+
+XImage *xreadscreen(Display *disp, Drawable d, int x, int y,
+ unsigned int width, unsigned int height, Bool show_cursor) {
+#ifdef SOLARIS_OVERLAY
+ return XReadScreen(disp, d, x, y, width, height,
+ show_cursor);
+#endif
+#ifdef IRIX_OVERLAY
+ { unsigned long hints = 0, hints_ret;
+ if (show_cursor) hints |= XRD_READ_POINTER;
+ return XReadDisplay(disp, d, x, y, width, height,
+ hints, &hints_ret);
+ }
+#endif
+ return NULL;
+}
+
XImage *XGetSubImage_wr(Display *disp, Drawable d, int x, int y,
unsigned int width, unsigned int height, unsigned long plane_mask,
int format, XImage *dest_image, int dest_x, int dest_y) {
ADJUST_ROOTSHIFT
-#ifdef SOLARIS
if (overlay && dest_x == 0 && dest_y == 0) {
size_t size = dest_image->height * dest_image->bytes_per_line;
- XImage *xi = XReadScreen(disp, d, x, y, width, height,
+ XImage *xi;
+
+ xi = xreadscreen(disp, d, x, y, width, height,
(Bool) overlay_cursor);
+ if (! xi) return NULL;
+
/*
* There is extra overhead from memcpy and free...
* this is not like the real XGetSubImage(). We hope
@@ -613,7 +1028,6 @@ XImage *XGetSubImage_wr(Display *disp, Drawable d, int x, int y,
XDestroyImage(xi);
return (dest_image);
}
-#endif
return XGetSubImage(disp, d, x, y, width, height, plane_mask,
format, dest_image, dest_x, dest_y);
}
@@ -624,28 +1038,26 @@ XImage *XGetImage_wr(Display *disp, Drawable d, int x, int y,
ADJUST_ROOTSHIFT
-#ifdef SOLARIS
if (overlay) {
- return XReadScreen(disp, d, x, y, width, height,
+ return xreadscreen(disp, d, x, y, width, height,
(Bool) overlay_cursor);
}
-#endif
return XGetImage(disp, d, x, y, width, height, plane_mask, format);
}
XImage *XCreateImage_wr(Display *disp, Visual *visual, unsigned int depth,
int format, int offset, char *data, unsigned int width,
unsigned int height, int bitmap_pad, int bytes_per_line) {
-/*
- * This is a kludge to get a created XImage to exactly match what
- * XReadScreen returns: we noticed the rgb masks are different from
- * XCreateImage with the high color visual (red mask <-> blue mask).
- * Note we read from the root window(!) then free the data.
- */
-#ifdef SOLARIS
+ /*
+ * This is a kludge to get a created XImage to exactly match what
+ * XReadScreen returns: we noticed the rgb masks are different
+ * from XCreateImage with the high color visual (red mask <->
+ * blue mask). Note we read from the root window(!) then free
+ * the data.
+ */
if (overlay) {
XImage *xi;
- xi = XReadScreen(disp, window, 0, 0, width, height, False);
+ xi = xreadscreen(disp, window, 0, 0, width, height, False);
if (xi == NULL) {
return xi;
}
@@ -655,7 +1067,6 @@ XImage *XCreateImage_wr(Display *disp, Visual *visual, unsigned int depth,
xi->data = data;
return xi;
}
-#endif
return XCreateImage(disp, visual, depth, format, offset, data,
width, height, bitmap_pad, bytes_per_line);
@@ -764,23 +1175,53 @@ void XTestDiscard_wr(Display *dpy) {
static int exit_flag = 0;
int exit_sig = 0;
+void clean_shm(int quick) {
+ int i, cnt = 0;
+
+ /*
+ * to avoid deadlock, etc, under quick=1 we just delete the shm
+ * areas and leave the X stuff hanging.
+ */
+ if (quick) {
+ shm_delete(&scanline_shm);
+ shm_delete(&fullscreen_shm);
+ } else {
+ shm_clean(&scanline_shm, scanline);
+ shm_clean(&fullscreen_shm, fullscreen);
+ }
+
+ /*
+ * Here we have to clean up quite a few shm areas for all
+ * the possible tile row runs (40 for 1280), not as robust
+ * as one might like... sometimes need to run ipcrm(1).
+ */
+ for(i=1; i<=ntiles_x; i++) {
+ if (i > tile_shm_count) {
+ break;
+ }
+ if (quick) {
+ shm_delete(&tile_row_shm[i]);
+ } else {
+ shm_clean(&tile_row_shm[i], tile_row[i]);
+ }
+ cnt++;
+ if (single_copytile_count && i >= single_copytile_count) {
+ break;
+ }
+ }
+ if (!quiet && !quick) {
+ rfbLog("deleted %d tile_row polling images.\n", cnt);
+ }
+}
+
/*
* Normal exiting
*/
void clean_up_exit (int ret) {
- int i;
exit_flag = 1;
/* remove the shm areas: */
- shm_clean(&scanline_shm, scanline);
- shm_clean(&fullscreen_shm, fullscreen);
-
- for(i=1; i<=ntiles_x; i++) {
- shm_clean(&tile_row_shm[i], tile_row[i]);
- if (single_copytile && i >= single_copytile) {
- break;
- }
- }
+ clean_shm(0);
/* X keyboard cleanups */
delete_added_keycodes();
@@ -807,7 +1248,6 @@ void clean_up_exit (int ret) {
* General problem handler
*/
static void interrupted (int sig) {
- int i;
exit_sig = sig;
if (exit_flag) {
exit_flag++;
@@ -828,24 +1268,11 @@ static void interrupted (int sig) {
shut_down = 1;
return;
}
- /*
- * to avoid deadlock, etc, just delete the shm areas and
- * leave the X stuff hanging.
- */
- shm_delete(&scanline_shm);
- shm_delete(&fullscreen_shm);
- /*
- * Here we have to clean up quite a few shm areas for all
- * the possible tile row runs (40 for 1280), not as robust
- * as one might like... sometimes need to run ipcrm(1).
- */
- for(i=1; i<=ntiles_x; i++) {
- shm_delete(&tile_row_shm[i]);
- if (single_copytile && i >= single_copytile) {
- break;
- }
- }
+ X_UNLOCK;
+
+ /* remove the shm areas with quick=1: */
+ clean_shm(1);
/* X keyboard cleanups */
delete_added_keycodes();
@@ -858,6 +1285,7 @@ static void interrupted (int sig) {
if (no_autorepeat) {
autorepeat(1);
}
+
if (sig) {
exit(2);
}
@@ -867,10 +1295,19 @@ static void interrupted (int sig) {
static XErrorHandler Xerror_def;
static XIOErrorHandler XIOerr_def;
+XErrorEvent *trapped_xerror_event;
int trapped_xerror = 0;
+int trapped_getimage_xerror = 0;
int trap_xerror(Display *d, XErrorEvent *error) {
trapped_xerror = 1;
+ trapped_xerror_event = error;
+ return 0;
+}
+
+int trap_getimage_xerror(Display *d, XErrorEvent *error) {
+ trapped_getimage_xerror = 1;
+ trapped_xerror_event = error;
return 0;
}
@@ -886,6 +1323,74 @@ static int XIOerr(Display *d) {
return (*XIOerr_def)(d);
}
+char *xerrors[] = {
+ "Success",
+ "BadRequest",
+ "BadValue",
+ "BadWindow",
+ "BadPixmap",
+ "BadAtom",
+ "BadCursor",
+ "BadFont",
+ "BadMatch",
+ "BadDrawable",
+ "BadAccess",
+ "BadAlloc",
+ "BadColor",
+ "BadGC",
+ "BadIDChoice",
+ "BadName",
+ "BadLength",
+ "BadImplementation",
+ "unknown"
+};
+int xerrors_max = BadImplementation;
+
+char *xerror_string(XErrorEvent *error) {
+ int index = -1;
+ if (error) {
+ index = (int) error->error_code;
+ }
+ if (0 <= index && index <= xerrors_max) {
+ return xerrors[index];
+ } else {
+ return xerrors[xerrors_max+1];
+ }
+}
+
+/* trapping utility to check for a valid window: */
+int valid_window(Window win) {
+ XErrorHandler old_handler;
+ XWindowAttributes attr;
+ int ok = 0;
+
+ trapped_xerror = 0;
+ old_handler = XSetErrorHandler(trap_xerror);
+ if (XGetWindowAttributes(dpy, win, &attr)) {
+ ok = 1;
+ }
+ if (trapped_xerror && trapped_xerror_event && ! quiet) {
+ rfbLog("trapped XError: %s (0x%lx)\n",
+ xerror_string(trapped_xerror_event), win);
+ }
+ XSetErrorHandler(old_handler);
+ trapped_xerror = 0;
+
+ return ok;
+}
+
+int get_window_size(Window win, int *x, int *y) {
+ XWindowAttributes attr;
+ /* valid_window? */
+ if (XGetWindowAttributes(dpy, win, &attr)) {
+ *x = attr.width;
+ *y = attr.height;
+ return 1;
+ } else {
+ return 0;
+ }
+}
+
/* signal handlers */
void initialize_signals(void) {
signal(SIGHUP, interrupted);
@@ -897,11 +1402,13 @@ void initialize_signals(void) {
signal(SIGSEGV, interrupted);
signal(SIGFPE, interrupted);
- if (sigpipe == 1) {
+ if (!sigpipe || *sigpipe == '\0' || !strcmp(sigpipe, "skip")) {
+ ;
+ } else if (!strcmp(sigpipe, "ignore")) {
#ifdef SIG_IGN
signal(SIGPIPE, SIG_IGN);
#endif
- } else if (sigpipe == 2) {
+ } else if (!strcmp(sigpipe, "exit")) {
rfbLog("initialize_signals: will exit on SIGPIPE\n");
signal(SIGPIPE, interrupted);
}
@@ -919,6 +1426,7 @@ void initialize_signals(void) {
static int accepted_client = 0;
static int client_count = 0;
+static int clients_served = 0;
/*
* check that all clients are in RFB_NORMAL state
@@ -940,11 +1448,110 @@ int all_clients_initialized(void) {
return ok;
}
+char *list_clients(void) {
+ rfbClientIteratorPtr iter;
+ rfbClientPtr cl;
+ char *list, tmp[32];
+ int count = 0;
+
+ iter = rfbGetClientIterator(screen);
+ while( (cl = rfbClientIteratorNext(iter)) ) {
+ count++;
+ }
+ rfbReleaseClientIterator(iter);
+
+ list = (char *) malloc((count+1)*100);
+
+ list[0] = '\0';
+
+ iter = rfbGetClientIterator(screen);
+ while( (cl = rfbClientIteratorNext(iter)) ) {
+ ClientData *cd = (ClientData *) cl->clientData;
+ if (*list != '\0') {
+ strcat(list, ",");
+ }
+ strcat(list, cl->host);
+ sprintf(tmp, ":%d/0x%x", get_remote_port(cl->sock), cd->uid);
+ strcat(list, tmp);
+ if (cl->viewOnly) {
+ strcat(list, "-ro");
+ } else {
+ strcat(list, "-rw");
+ }
+ }
+ rfbReleaseClientIterator(iter);
+ return list;
+}
+
+void close_all_clients(void) {
+ rfbClientIteratorPtr iter;
+ rfbClientPtr cl;
+
+ iter = rfbGetClientIterator(screen);
+ while( (cl = rfbClientIteratorNext(iter)) ) {
+ rfbCloseClient(cl);
+ rfbClientConnectionGone(cl);
+ }
+ rfbReleaseClientIterator(iter);
+}
+
+void close_clients(char *str) {
+ rfbClientIteratorPtr iter;
+ rfbClientPtr cl;
+ int host_warn = 0, hex_warn = 0;
+
+ if (!strcmp(str, "all") || !strcmp(str, "*")) {
+ close_all_clients();
+ return;
+ }
+
+ iter = rfbGetClientIterator(screen);
+ while( (cl = rfbClientIteratorNext(iter)) ) {
+ if (strstr(str, "0x")) {
+ int id;
+ ClientData *cd = (ClientData *) cl->clientData;
+ if (sscanf(str, "0x%x", &id) != 1) {
+ if (hex_warn++) {
+ continue;
+ }
+ rfbLog("skipping bad client hex id: %s\n", str);
+ continue;
+ }
+ if ( cd->uid == id) {
+ rfbCloseClient(cl);
+ rfbClientConnectionGone(cl);
+ }
+ } else {
+ char *rstr = str;
+ if (! dotted_ip(str)) {
+ rstr = host2ip(str);
+ if (rstr == NULL) {
+ if (host_warn++) {
+ continue;
+ }
+ rfbLog("skipping bad lookup: \"%s\"\n",
+ str);
+ continue;
+ }
+ rfbLog("lookup: %s -> %s\n", str, rstr);
+ }
+ if (!strcmp(rstr, cl->host)) {
+ rfbCloseClient(cl);
+ rfbClientConnectionGone(cl);
+ }
+ if (rstr != str) {
+ free(rstr);
+ }
+ }
+ }
+ rfbReleaseClientIterator(iter);
+}
+
/*
* utility to run a user supplied command setting some RFB_ env vars.
* used by, e.g., accept_client() and client_gone()
*/
-static int run_user_command(char *cmd, rfbClientPtr client) {
+static int run_user_command(char *cmd, rfbClientPtr client, char *mode) {
char *dpystr = DisplayString(dpy);
static char *display_env = NULL;
static char env_rfb_client_id[100];
@@ -954,18 +1561,24 @@ static int run_user_command(char *cmd, rfbClientPtr client) {
static char env_rfb_server_port[100];
static char env_rfb_x11vnc_pid[100];
static char env_rfb_client_count[100];
+ static char env_rfb_mode[100];
char *addr = client->host;
- int rc;
- char *saddr_ip_str = NULL;
- int saddr_len, saddr_port;
- struct sockaddr_in saddr;
+ int rc, hport;
+ char *ip_str = NULL;
+ ClientData *cd = (ClientData *) client->clientData;
if (addr == NULL || addr[0] == '\0') {
addr = "unknown-host";
}
/* set RFB_CLIENT_ID to semi unique id for command to use */
- sprintf(env_rfb_client_id, "RFB_CLIENT_ID=%p", (void *) client);
+ if (cd && cd->uid) {
+ sprintf(env_rfb_client_id, "RFB_CLIENT_ID=0x%x", cd->uid);
+ } else {
+ /* not accepted yet: */
+ sprintf(env_rfb_client_id, "RFB_CLIENT_ID=0x%x",
+ clients_served);
+ }
putenv(env_rfb_client_id);
/* set RFB_CLIENT_IP to IP addr for command to use */
@@ -977,34 +1590,25 @@ static int run_user_command(char *cmd, rfbClientPtr client) {
putenv(env_rfb_x11vnc_pid);
/* set RFB_CLIENT_PORT to peer port for command to use */
- saddr_len = sizeof(saddr);
- memset(&saddr, 0, sizeof(saddr));
- saddr_port = -1;
- if (!getpeername(client->sock, (struct sockaddr *)&saddr, &saddr_len)) {
- saddr_port = ntohs(saddr.sin_port);
- }
- sprintf(env_rfb_client_port, "RFB_CLIENT_PORT=%d", saddr_port);
+ hport = get_remote_port(client->sock);
+ sprintf(env_rfb_client_port, "RFB_CLIENT_PORT=%d", hport);
putenv(env_rfb_client_port);
+ sprintf(env_rfb_mode, "RFB_MODE=%s", mode);
+ putenv(env_rfb_mode);
+
/*
* now do RFB_SERVER_IP and RFB_SERVER_PORT (i.e. us!)
* This will establish a 5-tuple (including tcp) the external
* program can potentially use to work out the virtual circuit
* for this connection.
*/
- saddr_len = sizeof(saddr);
- memset(&saddr, 0, sizeof(saddr));
- saddr_port = -1;
- saddr_ip_str = "unknown";
- if (!getsockname(client->sock, (struct sockaddr *)&saddr, &saddr_len)) {
- saddr_port = ntohs(saddr.sin_port);
-#ifdef LIBVNCSERVER_HAVE_NETINET_IN_H
- saddr_ip_str = inet_ntoa(saddr.sin_addr);
-#endif
- }
- sprintf(env_rfb_server_ip, "RFB_SERVER_IP=%s", saddr_ip_str);
+ ip_str = get_local_host(client->sock);
+ hport = get_local_port(client->sock);
+
+ sprintf(env_rfb_server_ip, "RFB_SERVER_IP=%s", ip_str);
putenv(env_rfb_server_ip);
- sprintf(env_rfb_server_port, "RFB_SERVER_PORT=%d", saddr_port);
+ sprintf(env_rfb_server_port, "RFB_SERVER_PORT=%d", hport);
putenv(env_rfb_server_port);
@@ -1039,7 +1643,6 @@ static int run_user_command(char *cmd, rfbClientPtr client) {
return rc;
}
-
/*
* callback for when a client disconnects
*/
@@ -1048,12 +1651,16 @@ static void client_gone(rfbClientPtr client) {
client_count--;
rfbLog("client_count: %d\n", client_count);
+ if (client->clientData) {
+ free(client->clientData);
+ }
+
if (no_autorepeat && client_count == 0) {
autorepeat(1);
}
- if (gone_cmd) {
+ if (gone_cmd && *gone_cmd != '\0') {
rfbLog("client_gone: using cmd for: %s\n", client->host);
- run_user_command(gone_cmd, client);
+ run_user_command(gone_cmd, client, "gone");
}
if (inetd) {
@@ -1074,6 +1681,11 @@ static void client_gone(rfbClientPtr client) {
accepted_client = 0;
return;
}
+ if (shared && client_count > 0) {
+ rfbLog("connect_once: other shared clients still "
+ "connected, not exiting.\n");
+ return;
+ }
rfbLog("viewer exited.\n");
clean_up_exit(0);
@@ -1088,8 +1700,19 @@ static int check_access(char *addr) {
int allowed = 0;
char *p, *list;
+ if (deny_all) {
+ rfbLog("check_access: new connections are currently "
+ "blocked.\n");
+ return 0;
+ }
+
if (allow_list == NULL || *allow_list == '\0') {
- return 1;
+ if (allow_once == NULL) {
+ return 1;
+ }
+ }
+ if (allow_list == NULL) {
+ allow_list = strdup("");
}
if (addr == NULL || *addr == '\0') {
rfbLog("check_access: denying empty host IP address string.\n");
@@ -1110,6 +1733,9 @@ static int check_access(char *addr) {
clean_up_exit(1);
}
len = sbuf.st_size + 1; /* 1 more for '\0' at end */
+ if (allow_once) {
+ len += strlen(allow_once) + 2;
+ }
list = malloc(len);
list[0] = '\0';
@@ -1129,17 +1755,46 @@ static int check_access(char *addr) {
strcat(list, line);
}
fclose(in);
+ if (allow_once) {
+ strcat(list, allow_once);
+ strcat(list, "\n");
+ }
} else {
- list = strdup(allow_list);
+ int len = strlen(allow_list);
+ if (allow_once) {
+ len += strlen(allow_once) + 2;
+ }
+ list = malloc(len);
+ list[0] = '\0';
+ strcat(list, allow_list);
+ if (allow_once) {
+ strcat(list, ",");
+ strcat(list, allow_once);
+ }
}
+ if (allow_once) {
+ free(allow_once);
+ allow_once = NULL;
+ }
p = strtok(list, ", \t\n\r");
while (p) {
- char *q;
+ char *q, *r = NULL;
if (*p == '\0') {
+ p = strtok(NULL, ", \t\n\r");
continue;
}
+ if (! dotted_ip(p)) {
+ r = host2ip(p);
+ if (r == NULL) {
+ rfbLog("check_access: bad lookup \"%s\"\n", p);
+ p = strtok(NULL, ", \t\n\r");
+ continue;
+ }
+ rfbLog("check_access: lookup %s -> %s\n", p, r);
+ p = r;
+ }
q = strstr(addr, p);
if (q == addr) {
rfbLog("check_access: client %s matches pattern %s\n",
@@ -1150,6 +1805,12 @@ static int check_access(char *addr) {
allowed = 1;
}
p = strtok(NULL, ", \t\n\r");
+ if (r) {
+ free(r);
+ }
+ if (allowed) {
+ break;
+ }
}
free(list);
return allowed;
@@ -1157,7 +1818,7 @@ static int check_access(char *addr) {
/*
* x11vnc's first (and only) visible widget: accept/reject dialog window.
- * We go through this pain to avoid dependency on libXt.
+ * We go through this pain to avoid dependency on libXt...
*/
static int ugly_accept_window(char *addr, int X, int Y, int timeout,
char *mode) {
@@ -1364,8 +2025,8 @@ static char t2x2_bits[] = {
strlen(str_v));
tw = (Vi_w - tw)/2;
if (tw < 0) tw = 1;
- XDrawString(dpy, awin, gc, Vi_x+tw, Vi_y+Vi_h-5,
- str_v, strlen(str_v));
+ XDrawString(dpy, awin, gc, Vi_x+tw,
+ Vi_y+Vi_h-5, str_v, strlen(str_v));
}
break;
@@ -1529,7 +2190,7 @@ static int accept_client(rfbClientPtr client) {
char *addr = client->host;
char *action = NULL;
- if (accept_cmd == NULL) {
+ if (accept_cmd == NULL || *accept_cmd == '\0') {
return 1; /* no command specified, so we accept */
}
@@ -1637,7 +2298,7 @@ static int accept_client(rfbClientPtr client) {
int rc;
rfbLog("accept_client: using cmd for: %s\n", addr);
- rc = run_user_command(cmd, client);
+ rc = run_user_command(cmd, client, "accept");
if (action) {
int result;
@@ -1689,6 +2350,9 @@ static void check_connect_file(char *file) {
static time_t last_time = 0;
time_t now = time(0);
+ if (last_time == 0) {
+ last_time = now;
+ }
if (now - last_time < 1) {
/* check only once a second */
return;
@@ -1717,8 +2381,9 @@ static void check_connect_file(char *file) {
if (fgets(line, 1024, in) != NULL) {
if (sscanf(line, "%s", host) == 1) {
if (strlen(host) > 0) {
- client_connect = strdup(host);
+ char *str = strdup(host);
rfbLog("read connect file: %s\n", host);
+ client_connect = str;
}
}
}
@@ -1836,6 +2501,11 @@ static void reverse_connect(char *str) {
* Routines for monitoring the VNC_CONNECT property for changes.
* The vncconnect(1) will set it on our X display.
*/
+void set_vnc_connect_prop(char *str) {
+ XChangeProperty(dpy, rootwin, vnc_connect_prop, XA_STRING, 8,
+ PropModeReplace, (unsigned char *)str, strlen(str));
+}
+
void read_vnc_connect_prop(void) {
Atom type;
int format, slen, dlen;
@@ -1873,7 +2543,15 @@ void read_vnc_connect_prop(void) {
} while (bytes_after > 0);
vnc_connect_str[VNC_CONNECT_MAX] = '\0';
- rfbLog("read property VNC_CONNECT: %s\n", vnc_connect_str);
+ if (strlen(vnc_connect_str) > 100) {
+ char trim[101];
+ trim[0] = '\0';
+ strncat(trim, vnc_connect_str, 100);
+ rfbLog("read X property VNC_CONNECT: %s ...\n", trim);
+
+ } else {
+ rfbLog("read X property VNC_CONNECT: %s\n", vnc_connect_str);
+ }
}
/*
@@ -1881,7 +2559,14 @@ void read_vnc_connect_prop(void) {
*/
static void send_client_connect(void) {
if (client_connect != NULL) {
- reverse_connect(client_connect);
+ char *str = client_connect;
+ if (strstr(str, "cmd=") == str || strstr(str, "qry=") == str) {
+ process_remote_cmd(client_connect, 0);
+ } else if (strstr(str, "ans=") || strstr(str, "aro=") == str) {
+ ;
+ } else {
+ reverse_connect(client_connect);
+ }
free(client_connect);
client_connect = NULL;
}
@@ -1913,6 +2598,7 @@ void check_connect_inputs(void) {
* libvncserver callback for when a new client connects
*/
enum rfbNewClientAction new_client(rfbClientPtr client) {
+ ClientData *cd;
last_event = last_input = time(0);
if (inetd) {
@@ -1923,6 +2609,8 @@ enum rfbNewClientAction new_client(rfbClientPtr client) {
client->clientGoneHook = client_gone;
}
+ clients_served++;
+
if (connect_once) {
if (screen->dontDisconnect && screen->neverShared) {
if (! shared && accepted_client) {
@@ -1946,11 +2634,12 @@ enum rfbNewClientAction new_client(rfbClientPtr client) {
}
if (view_only) {
- client->clientData = (void *) -1;
client->viewOnly = TRUE;
- } else {
- client->clientData = (void *) 0;
}
+ client->clientData = (void *) calloc(sizeof(ClientData), 1);
+ cd = (ClientData *) client->clientData;
+
+ cd->uid = clients_served;
client->clientGoneHook = client_gone;
client_count++;
@@ -1960,6 +2649,10 @@ enum rfbNewClientAction new_client(rfbClientPtr client) {
autorepeat(0);
}
+ if (pad_geometry) {
+ install_padded_fb(pad_geometry);
+ }
+
accepted_client = 1;
last_client = time(0);
@@ -2253,6 +2946,21 @@ void initialize_remap(char *infile) {
KeySym ksym1, ksym2;
keyremap_t *remap, *current;
+ if (keyremaps != NULL) {
+ /* free last remapping */
+ keyremap_t *next_remap, *curr_remap = keyremaps;
+ while (curr_remap != NULL) {
+ next_remap = curr_remap->next;
+ free(curr_remap);
+ curr_remap = next_remap;
+ }
+ keyremaps = NULL;
+ }
+ if (infile == NULL || *infile == '\0') {
+ /* just unset remapping */
+ return;
+ }
+
in = fopen(infile, "r");
if (in == NULL) {
/* assume cmd line key1-key2,key3-key4 */
@@ -2581,11 +3289,11 @@ xkbmodifiers[] For the KeySym bound to this (keycode,group,level) store
if (debug_keyboard > 1) {
fprintf(stderr, " %03d G%d L%d mod=%s ",
- kc, grp+1, lvl+1, bitprint(ms));
+ kc, grp+1, lvl+1, bitprint(ms, 8));
fprintf(stderr, "state=%s ",
- bitprint(xkbstate[kc][grp][lvl]));
+ bitprint(xkbstate[kc][grp][lvl], 8));
fprintf(stderr, "ignore=%s ",
- bitprint(xkbignore[kc][grp][lvl]));
+ bitprint(xkbignore[kc][grp][lvl], 8));
fprintf(stderr, " ks=0x%08lx \"%s\"\n",
ks, XKeysymToString(ks));
}
@@ -2688,9 +3396,9 @@ static void xkb_tweak_keyboard(rfbBool down, rfbKeySym keysym,
XKeysymToString(keysym), XKeysymToString(
XKeycodeToKeysym(dpy, kc, 0)));
fprintf(stderr, " need state: %s\n",
- bitprint(state));
+ bitprint(state, 8));
fprintf(stderr, " ignorable : %s\n",
- bitprint(xkbignore[kc][grp][lvl]));
+ bitprint(xkbignore[kc][grp][lvl], 8));
}
/* save it if state is OK and not told to skip */
@@ -2719,11 +3427,11 @@ static void xkb_tweak_keyboard(rfbBool down, rfbKeySym keysym,
#define PKBSTATE \
fprintf(stderr, " --- current mod state:\n"); \
- fprintf(stderr, " mods : %s\n", bitprint(kbstate.mods)); \
- fprintf(stderr, " base_mods : %s\n", bitprint(kbstate.base_mods)); \
- fprintf(stderr, " latch_mods: %s\n", bitprint(kbstate.latched_mods)); \
- fprintf(stderr, " lock_mods : %s\n", bitprint(kbstate.locked_mods)); \
- fprintf(stderr, " compat : %s\n", bitprint(kbstate.compat_state));
+ fprintf(stderr, " mods : %s\n", bitprint(kbstate.mods, 8)); \
+ fprintf(stderr, " base_mods : %s\n", bitprint(kbstate.base_mods, 8)); \
+ fprintf(stderr, " latch_mods: %s\n", bitprint(kbstate.latched_mods, 8)); \
+ fprintf(stderr, " lock_mods : %s\n", bitprint(kbstate.locked_mods, 8)); \
+ fprintf(stderr, " compat : %s\n", bitprint(kbstate.compat_state, 8));
/*
* Now get the current state of the keyboard from the X server.
@@ -2902,7 +3610,7 @@ static void xkb_tweak_keyboard(rfbBool down, rfbKeySym keysym,
if (debug_keyboard > 1) {
fprintf(stderr, " +++ needmods: mod=%d %s "
- "need it to be: %d %s\n", i, bitprint(b),
+ "need it to be: %d %s\n", i, bitprint(b, 8),
needmods[i], needmods[i] ? "down" : "up");
}
@@ -2958,7 +3666,7 @@ static void xkb_tweak_keyboard(rfbBool down, rfbKeySym keysym,
fprintf(stderr, " === for "
"mod=%s found kc=%03d/G%d"
"/L%d it is %d %s skip=%d "
- "(%s)\n", bitprint(b), kc,
+ "(%s)\n", bitprint(b,8), kc,
grp+1, lvl+1, kt, kt ?
"down" : "up ", skip,
str ? str : "null");
@@ -3104,21 +3812,41 @@ static char modifiers[0x100];
static KeyCode keycodes[0x100];
static KeyCode left_shift_code, right_shift_code, altgr_code, iso_level3_code;
-char *bitprint(unsigned int st) {
- static char str[9];
- int i, mask;
- for (i=0; i<8; i++) {
- str[i] = '0';
+/* workaround for X11R5, Latin 1 only */
+#ifndef XConvertCase
+#define XConvertCase(sym, lower, upper) \
+*(lower) = sym; \
+*(upper) = sym; \
+if (sym >> 8 == 0) { \
+ if ((sym >= XK_A) && (sym <= XK_Z)) \
+ *(lower) += (XK_a - XK_A); \
+ else if ((sym >= XK_a) && (sym <= XK_z)) \
+ *(upper) -= (XK_a - XK_A); \
+ else if ((sym >= XK_Agrave) && (sym <= XK_Odiaeresis)) \
+ *(lower) += (XK_agrave - XK_Agrave); \
+ else if ((sym >= XK_agrave) && (sym <= XK_odiaeresis)) \
+ *(upper) -= (XK_agrave - XK_Agrave); \
+ else if ((sym >= XK_Ooblique) && (sym <= XK_Thorn)) \
+ *(lower) += (XK_oslash - XK_Ooblique); \
+ else if ((sym >= XK_oslash) && (sym <= XK_thorn)) \
+ *(upper) -= (XK_oslash - XK_Ooblique); \
+}
+#endif
+
+void initialize_keyboard_and_pointer(void) {
+ if (use_modifier_tweak) {
+ initialize_modtweak();
}
- str[8] = '\0';
- mask = 1;
- for (i=7; i>=0; i--) {
- if (st & mask) {
- str[i] = '1';
- }
- mask = mask << 1;
+ if (remap_file != NULL) {
+ initialize_remap(remap_file);
+ }
+
+ initialize_pointer_map(pointer_remap);
+
+ clear_modifiers(1);
+ if (clear_mods == 1) {
+ clear_modifiers(0);
}
- return str;
}
void initialize_modtweak(void) {
@@ -3278,7 +4006,7 @@ static void modifier_tweak_keyboard(rfbBool down, rfbKeySym keysym,
if (view_only) {
return;
}
- if (client->viewOnly) {
+ if (client && client->viewOnly) {
return;
}
@@ -3348,7 +4076,7 @@ void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) {
if (view_only) {
return;
}
- if (client->viewOnly) {
+ if (client && client->viewOnly) {
return;
}
@@ -3641,7 +4369,7 @@ void initialize_pointer_map(char *pointer_remap) {
pointer_map[i][0].up = 0;
}
- if (pointer_remap) {
+ if (pointer_remap && *pointer_remap != '\0') {
/* -buttonmap, format is like: 12-21=2 */
char *p, *q, *remap = pointer_remap;
int n;
@@ -3843,7 +4571,7 @@ void pointer(int mask, int x, int y, rfbClientPtr client) {
* See check_user_input() for the more complicated things we do
* in the non-threaded case.
*/
- if (use_threads && old_pointer != 1) {
+ if (use_threads && pointer_mode != 1) {
# define NEV 32
/* storage for the event queue */
static int mutex_init = 0;
@@ -3940,22 +4668,24 @@ void initialize_xkb(void) {
int ir, reason;
int op, ev, er, maj, min;
- if (! use_xkb) {
- return;
- }
if (! XkbQueryExtension(dpy, &op, &ev, &er, &maj, &min)) {
- if (! quiet) {
- fprintf(stderr, "warning: XKEYBOARD"
- " extension not present.\n");
+ if (! quiet && use_xkb) {
+ rfbLog("warning: XKEYBOARD extension not present.\n");
}
+ xkb_present = 0;
use_xkb = 0;
return;
+ } else {
+ xkb_present = 1;
+ }
+ if (! use_xkb) {
+ return;
}
if (! XkbOpenDisplay(DisplayString(dpy), &xkb_base_event_type, &ir,
NULL, NULL, &reason) ) {
if (! quiet) {
- fprintf(stderr, "warning: disabling XKEYBOARD."
- " XkbOpenDisplay failed.\n");
+ rfbLog("warning: disabling XKEYBOARD. XkbOpenDisplay"
+ " failed.\n");
}
use_xkb = 0;
}
@@ -3964,8 +4694,8 @@ void initialize_xkb(void) {
void initialize_watch_bell(void) {
if (! use_xkb) {
if (! quiet) {
- fprintf(stderr, "warning: disabling bell."
- " XKEYBOARD ext. not present.\n");
+ rfbLog("warning: disabling bell. XKEYBOARD ext. "
+ "not present.\n");
}
watch_bell = 0;
return;
@@ -3973,8 +4703,8 @@ void initialize_watch_bell(void) {
if (! XkbSelectEvents(dpy, XkbUseCoreKbd, XkbBellNotifyMask,
XkbBellNotifyMask) ) {
if (! quiet) {
- fprintf(stderr, "warning: disabling bell."
- " XkbSelectEvents failed.\n");
+ rfbLog("warning: disabling bell. XkbSelectEvents"
+ " failed.\n");
}
watch_bell = 0;
}
@@ -3994,7 +4724,7 @@ void check_bell_event(void) {
}
X_LOCK;
- if (! XCheckTypedEvent(dpy, xkb_base_event_type , &xev)) {
+ if (! XCheckTypedEvent(dpy, xkb_base_event_type, &xev)) {
X_UNLOCK;
return;
}
@@ -4004,7 +4734,7 @@ void check_bell_event(void) {
}
X_UNLOCK;
- if (got_bell) {
+ if (got_bell && sound_bell) {
if (! all_clients_initialized()) {
rfbLog("check_bell_event: not sending bell: "
"uninitialized clients\n");
@@ -4017,6 +4747,260 @@ void check_bell_event(void) {
void check_bell_event(void) {}
#endif
+/* -- xrandr.h -- */
+
+time_t last_subwin_trap = 0;
+int subwin_trap_count = 0;
+
+XErrorHandler old_getimage_handler;
+#define XRANDR_SET_TRAP_RET(x,y) \
+ if (subwin || xrandr) { \
+ if (0) fprintf(stderr, " SET_TRAP: '%d' '%s'\n", x, y); \
+ trapped_getimage_xerror = 0; \
+ old_getimage_handler = XSetErrorHandler(trap_getimage_xerror); \
+ if (check_xrandr_event(y)) { \
+ trapped_getimage_xerror = 0; \
+ XSetErrorHandler(old_getimage_handler); \
+ return(x); \
+ } \
+ }
+#define XRANDR_CHK_TRAP_RET(x,y) \
+ if (subwin || xrandr) { \
+ if (0) fprintf(stderr, " CHK_TRAP: '%d' '%s'\n", x, y); \
+ if (trapped_getimage_xerror) { \
+ if (subwin) { \
+ static int last = 0; \
+ subwin_trap_count++; \
+ if (time(0) > last_subwin_trap + 60) { \
+ rfbLog("trapped GetImage xerror" \
+ " in SUBWIN mode. [%d]\n", \
+ subwin_trap_count); \
+ last_subwin_trap = time(0); \
+ last = subwin_trap_count; \
+ } \
+ if (subwin_trap_count - last > 30) { \
+ /* window probably iconified */ \
+ usleep(1000*1000); \
+ } \
+ } else { \
+ rfbLog("trapped GetImage xerror" \
+ " in XRANDR mode.\n"); \
+ } \
+ trapped_getimage_xerror = 0; \
+ XSetErrorHandler(old_getimage_handler); \
+ check_xrandr_event(y); \
+ X_UNLOCK; \
+ return(x); \
+ } \
+ }
+
+/* -- xrandr.c -- */
+
+void initialize_xrandr(void) {
+#ifdef LIBVNCSERVER_HAVE_LIBXRANDR
+ if (xrandr_present) {
+ Rotation rot;
+
+ xrandr_width = XDisplayWidth(dpy, scr);
+ xrandr_height = XDisplayHeight(dpy, scr);
+ XRRRotations(dpy, scr, &rot);
+ xrandr_rotation = (int) rot;
+ if (xrandr) {
+ XRRSelectInput(dpy, rootwin, RRScreenChangeNotifyMask);
+ } else {
+ XRRSelectInput(dpy, rootwin, 0);
+ }
+ }
+#endif
+}
+
+void handle_xrandr_change(int, int);
+
+int handle_subwin_resize(char *msg) {
+ int new_x, new_y;
+ int i, check = 10, ms = 250; /* 2.5 secs... */
+
+ if (0) fprintf(stderr, "IN handle_subwin_resize('%s')\n", msg);
+ if (! subwin) {
+ return 0; /* hmmm... */
+ }
+ if (! valid_window(subwin)) {
+ rfbLog("subwin 0x%lx went away!\n", subwin);
+ X_UNLOCK;
+ clean_up_exit(1);
+ }
+ if (! get_window_size(subwin, &new_x, &new_y)) {
+ rfbLog("could not get size of subwin 0x%lx\n", subwin);
+ X_UNLOCK;
+ clean_up_exit(1);
+ }
+ if (dpy_x == new_x && dpy_y == new_y) {
+ /* no change */
+ return 0;
+ }
+
+ /* window may still be changing (e.g. drag resize) */
+ for (i=0; i < check; i++) {
+ int newer_x, newer_y;
+ usleep(ms * 1000);
+
+ if (! get_window_size(subwin, &newer_x, &newer_y)) {
+ rfbLog("could not get size of subwin 0x%lx\n", subwin);
+ clean_up_exit(1);
+ }
+ if (new_x == newer_x && new_y == newer_y) {
+ /* go for it... */
+ break;
+ } else {
+ rfbLog("subwin 0x%lx still changing size...\n", subwin);
+ new_x = newer_x;
+ new_y = newer_y;
+ }
+ }
+
+ rfbLog("subwin 0x%lx new size: x: %d -> %d, y: %d -> %d\n",
+ subwin, dpy_x, new_x, dpy_y, new_y);
+ rfbLog("calling handle_xrandr_change() for resizing\n");
+
+ X_UNLOCK;
+ handle_xrandr_change(new_x, new_y);
+ return 1;
+}
+
+int known_xrandr_mode(char *);
+
+void handle_xrandr_change(int new_x, int new_y) {
+ rfbClientIteratorPtr iter;
+ rfbClientPtr cl;
+
+ /* sanity check xrandr_mode */
+ if (! xrandr_mode) {
+ xrandr_mode = strdup("default");
+ } else if (! known_xrandr_mode(xrandr_mode)) {
+ free(xrandr_mode);
+ xrandr_mode = strdup("default");
+ }
+ rfbLog("xrandr_mode: %s\n", xrandr_mode);
+ if (!strcmp(xrandr_mode, "exit")) {
+ close_all_clients();
+ rfbLog(" shutting down due to XRANDR event.\n");
+ clean_up_exit(0);
+ }
+ if (!strcmp(xrandr_mode, "newfbsize")) {
+ iter = rfbGetClientIterator(screen);
+ while( (cl = rfbClientIteratorNext(iter)) ) {
+ if (cl->useNewFBSize) {
+ continue;
+ }
+ rfbLog(" closing client %s (no useNewFBSize"
+ " support).\n", cl->host);
+ rfbCloseClient(cl);
+ rfbClientConnectionGone(cl);
+ }
+ rfbReleaseClientIterator(iter);
+ }
+
+ /* default, resize, and newfbsize create a new fb: */
+ rfbLog("check_xrandr_event: trying to create new framebuffer...\n");
+ if (new_x < dpy_x || new_y < dpy_y) {
+ check_black_fb();
+ }
+ do_new_fb(1);
+ rfbLog("check_xrandr_event: fb WxH: %dx%d\n", dpy_x, dpy_y);
+}
+
+int check_xrandr_event(char *msg) {
+ XEvent xev;
+ if (subwin) {
+ return handle_subwin_resize(msg);
+ }
+#ifdef LIBVNCSERVER_HAVE_LIBXRANDR
+ if (! xrandr || ! xrandr_present) {
+ return 0;
+ }
+ if (0) fprintf(stderr, "IN check_xrandr_event('%s')\n", msg);
+ if (XCheckTypedEvent(dpy, xrandr_base_event_type +
+ RRScreenChangeNotify, &xev)) {
+ int do_change;
+ XRRScreenChangeNotifyEvent *rev;
+
+ rev = (XRRScreenChangeNotifyEvent *) &xev;
+ rfbLog("check_xrandr_event():\n");
+ rfbLog("Detected XRANDR event at location '%s':\n", msg);
+ rfbLog(" serial: %d\n", (int) rev->serial);
+ rfbLog(" timestamp: %d\n", (int) rev->timestamp);
+ rfbLog(" cfg_timestamp: %d\n", (int) rev->config_timestamp);
+ rfbLog(" size_id: %d\n", (int) rev->size_index);
+ rfbLog(" sub_pixel: %d\n", (int) rev->subpixel_order);
+ rfbLog(" rotation: %d\n", (int) rev->rotation);
+ rfbLog(" width: %d\n", (int) rev->width);
+ rfbLog(" height: %d\n", (int) rev->height);
+ rfbLog(" mwidth: %d mm\n", (int) rev->mwidth);
+ rfbLog(" mheight: %d mm\n", (int) rev->mheight);
+ rfbLog("\n");
+ rfbLog("check_xrandr_event: previous WxH: %dx%d\n",
+ dpy_x, dpy_y);
+ if (dpy_x == rev->width && dpy_y == rev->height &&
+ xrandr_rotation == (int) rev->rotation) {
+ rfbLog("check_xrandr_event: no change detected.\n");
+ do_change = 0;
+ } else {
+ do_change = 1;
+ }
+
+ xrandr_width = rev->width;
+ xrandr_height = rev->height;
+ xrandr_timestamp = rev->timestamp;
+ xrandr_cfg_time = rev->config_timestamp;
+ xrandr_rotation = (int) rev->rotation;
+
+ rfbLog("check_xrandr_event: updating config...\n");
+ XRRUpdateConfiguration(&xev);
+
+ if (do_change) {
+ X_UNLOCK;
+ handle_xrandr_change(rev->width, rev->height);
+ }
+ rfbLog("check_xrandr_event: current WxH: %dx%d\n",
+ XDisplayWidth(dpy, scr), XDisplayHeight(dpy, scr));
+ rfbLog("check_xrandr_event(): returning control to"
+ " caller...\n");
+ if (0) fprintf(stderr, "OUT-%d check_xrandr_event('%s')\n",
+ do_change, msg);
+ return do_change;
+ }
+#endif
+ if (0) fprintf(stderr, "OUT-0 check_xrandr_event('%s')\n", msg);
+ return 0;
+}
+
+int known_xrandr_mode(char *s) {
+/*
+ * default:
+ * resize: the default
+ * exit: shutdown clients and exit.
+ * newfbsize: shutdown clients that do not support NewFBSize encoding.
+ */
+ if (strcmp(s, "default") && strcmp(s, "resize") &&
+ strcmp(s, "exit") && strcmp(s, "newfbsize")) {
+ return 0;
+ } else {
+ return 1;
+ }
+}
+
+int known_sigpipe_mode(char *s) {
+/*
+ * skip, ignore, exit
+ */
+ if (strcmp(s, "skip") && strcmp(s, "ignore") &&
+ strcmp(s, "exit")) {
+ return 0;
+ } else {
+ return 1;
+ }
+}
+
/* -- selection.c -- */
/*
* Selection/Cutbuffer/Clipboard handlers.
@@ -4176,6 +5160,8 @@ static void cutbuffer_send(void) {
*
* TODO: if we were willing to use libXt, we could perhaps get selection
* timestamps to speed up the checking... XtGetSelectionValue().
+ *
+ * Also: XFIXES has XFixesSelectSelectionInput().
*/
#define CHKSZ 32
static void selection_send(XEvent *ev) {
@@ -4258,36 +5244,70 @@ static void selection_send(XEvent *ev) {
rfbSendServerCutText(screen, selection_str, newlen);
}
-/*
- * This routine is periodically called to check for selection related
- * and other X11 events and respond to them as needed.
- */
-void check_xevents(void) {
- XEvent xev;
- static int first = 1, sent_some_sel = 0;
- static time_t last_request = 0;
- time_t now = time(0);
- int have_clients = screen->clientHead ? 1 : 0;
+/* -- xevents.c -- */
+
+void initialize_vnc_connect_prop() {
+ vnc_connect_str[0] = '\0';
+ vnc_connect_prop = XInternAtom(dpy, "VNC_CONNECT", False);
+}
+
+void initialize_xevents(void) {
+ static int did_xselect_input = 0;
+ static int did_xcreate_simple_window = 0;
+ static int did_vnc_connect_prop = 0;
+ static int did_xfixes = 0;
+ static int did_xrandr = 0;
X_LOCK;
- if (first && (watch_selection || vnc_connect)) {
+ if ((watch_selection || vnc_connect) && !did_xselect_input) {
/*
* register desired event(s) for notification.
* PropertyChangeMask is for CUT_BUFFER0 changes.
- * TODO: does this cause a flood of other stuff?
+ * XXX: does this cause a flood of other stuff?
*/
XSelectInput(dpy, rootwin, PropertyChangeMask);
+ did_xselect_input = 1;
}
- if (first && watch_selection) {
+ if (watch_selection && !did_xcreate_simple_window) {
/* create fake window for our selection ownership, etc */
+
selwin = XCreateSimpleWindow(dpy, rootwin, 0, 0, 1, 1, 0, 0, 0);
+ did_xcreate_simple_window = 1;
}
- if (first && vnc_connect) {
- vnc_connect_str[0] = '\0';
- vnc_connect_prop = XInternAtom(dpy, "VNC_CONNECT", False);
+ X_UNLOCK;
+
+ if (xrandr && !did_xrandr) {
+ initialize_xrandr();
+ did_xrandr = 1;
+ }
+ if (vnc_connect && !did_vnc_connect_prop) {
+ initialize_vnc_connect_prop();
+ did_vnc_connect_prop = 1;
+ }
+ if (xfixes_present && use_xfixes && !did_xfixes) {
+ initialize_xfixes();
+ did_xfixes = 1;
+ }
+}
+
+/*
+ * This routine is periodically called to check for selection related
+ * and other X11 events and respond to them as needed.
+ */
+void check_xevents(void) {
+ XEvent xev;
+ static int first = 1, sent_some_sel = 0;
+ static time_t last_request = 0;
+ time_t now = time(0);
+ int have_clients = screen->clientHead ? 1 : 0;
+
+
+ if (first) {
+ initialize_xevents();
}
first = 0;
+ X_LOCK;
/*
* There is a bug where we have to wait before sending text to
* the client... so instead of sending right away we wait a
@@ -4337,6 +5357,18 @@ void check_xevents(void) {
}
}
+#ifdef LIBVNCSERVER_HAVE_LIBXRANDR
+ if (xrandr) {
+ check_xrandr_event("check_xevents");
+ }
+#endif
+#ifdef LIBVNCSERVER_HAVE_LIBXFIXES
+ if (XCheckTypedEvent(dpy, xfixes_base_event_type +
+ XFixesCursorNotify, &xev)) {
+ got_xfixes_cursor_notify++;
+ }
+#endif
+
/* check for our PRIMARY request notification: */
if (watch_primary) {
if (XCheckTypedEvent(dpy, SelectionNotify, &xev)) {
@@ -4400,7 +5432,7 @@ void check_xevents(void) {
*/
void xcut_receive(char *text, int len, rfbClientPtr cl) {
- if (cl->viewOnly) {
+ if (cl && cl->viewOnly) {
return;
}
if (text == NULL || len == 0) {
@@ -4436,6 +5468,1806 @@ void xcut_receive(char *text, int len, rfbClientPtr cl) {
set_cutbuffer = 1;
}
+/* -- remote.c -- */
+
+/*
+ * for the wild-n-crazy -remote/-R interface.
+ */
+int send_remote_cmd(char *cmd, int query) {
+ char *str;
+ FILE *in = NULL;
+
+ if (client_connect_file) {
+ in = fopen(client_connect_file, "w");
+ if (in == NULL) {
+ fprintf(stderr, "send_remote_cmd: could not open "
+ "connect file \"%s\" for writing\n",
+ client_connect_file);
+ perror("fopen");
+ return 1;
+ }
+ } else if (vnc_connect_prop == None) {
+ initialize_vnc_connect_prop();
+ if (vnc_connect_prop == None) {
+ fprintf(stderr, "send_remote_cmd: could not obtain "
+ "VNC_CONNECT X property\n");
+ return 1;
+ }
+ }
+ str = (char *) malloc((strlen(cmd)+5));
+ if (query) {
+ strcpy(str, "qry=");
+ } else {
+ strcpy(str, "cmd=");
+ }
+ strcat(str, cmd);
+
+ if (in != NULL) {
+ fprintf(stderr, "sending remote command: \"%s\"\nvia connect"
+ " file: %s\n", cmd, client_connect_file);
+ fprintf(in, "%s\n", str);
+ fclose(in);
+ } else {
+ fprintf(stderr, "sending remote command: \"%s\" via VNC_CONNECT"
+ " X property.\n", cmd);
+ set_vnc_connect_prop(str);
+ XFlush(dpy);
+ }
+
+ if (strstr(str, "qry=") == str) {
+ char line[VNC_CONNECT_MAX];
+ int rc=1, i=0, max=20, ms_sl=100;
+ for (i=0; i<max; i++) {
+ usleep(ms_sl * 1000);
+ if (client_connect_file) {
+ char *q;
+ in = fopen(client_connect_file, "r");
+ if (in == NULL) {
+ fprintf(stderr, "send_remote_cmd: could"
+ " not open connect file \"%s\" for"
+ " writing\n", client_connect_file);
+ perror("fopen");
+ return 1;
+ }
+ fgets(line, VNC_CONNECT_MAX, in);
+ fclose(in);
+ q = line;
+ while (*q != '\0') {
+ if (*q == '\n') *q = '\0';
+ q++;
+ }
+ } else {
+ read_vnc_connect_prop();
+ strncpy(line, vnc_connect_str, VNC_CONNECT_MAX);
+ }
+ if (strcmp(str, line)){
+ fprintf(stdout, "%s\n", line);
+ fflush(stdout);
+ rc = 0;
+ break;
+ }
+ }
+ free(str);
+ if (rc) {
+ fprintf(stderr, "error: could not connect to "
+ "an x11vnc server at %s (rc=%d)\n",
+ client_connect_file ? client_connect_file
+ : DisplayString(dpy), rc);
+ }
+ return rc;
+ }
+ free(str);
+ return 0;
+}
+
+char *add_item(char *instr, char *item) {
+ char *p, *str;
+ int len, saw_item = 0;
+
+ if (! instr) {
+ str = strdup(item);
+ return str;
+ }
+ len = strlen(instr) + strlen(item) + 2;
+ str = (char *)malloc(len);
+ str[0] = '\0';
+
+ p = strtok(instr, ",");
+ while (p) {
+ if (!strcmp(p, item)) {
+ if (saw_item) {
+ p = strtok(NULL, ",");
+ continue;
+ }
+ saw_item = 1;
+ } else if (*p == '\0') {
+ p = strtok(NULL, ",");
+ continue;
+ }
+ if (str[0]) {
+ strcat(str, ",");
+ }
+ strcat(str, p);
+ p = strtok(NULL, ",");
+ }
+ if (! saw_item) {
+ if (str[0]) {
+ strcat(str, ",");
+ }
+ strcat(str, item);
+
+ }
+ return str;
+}
+
+char *delete_item(char *instr, char *item) {
+ char *p, *str;
+ int len;
+
+ if (! instr) {
+ str = strdup("");
+ return str;
+ }
+ len = strlen(instr) + 1;
+ str = (char *)malloc(len);
+ str[0] = '\0';
+
+ p = strtok(instr, ",");
+ while (p) {
+ if (!strcmp(p, item) || *p == '\0') {
+ p = strtok(NULL, ",");
+ continue;
+ }
+ if (str[0]) {
+ strcat(str, ",");
+ }
+ strcat(str, p);
+ p = strtok(NULL, ",");
+ }
+ return str;
+}
+
+void if_8bpp_do_new_fb(void) {
+ if (bpp == 8) {
+ do_new_fb(0);
+ } else {
+ rfbLog(" bpp(%d) is not 8bpp, not resetting fb\n", bpp);
+ }
+}
+
+void check_black_fb(void) {
+ if (new_fb_size_clients(screen) != client_count) {
+ rfbLog("trying to send a black fb for non-newfbsize"
+ " clients %d != %d\n", client_count,
+ new_fb_size_clients(screen));
+ push_black_screen(4);
+ }
+}
+
+/*
+ * Huge, ugly switch to handle all remote commands and queries
+ * -remote/-R and -query/-Q.
+ */
+char *process_remote_cmd(char *cmd, int stringonly) {
+ char *p = cmd;
+ char *co = "";
+ char buf[VNC_CONNECT_MAX];
+ int do_vnc_connect = 0;
+ int query = 0;
+ static char *prev_cursors_mode = NULL;
+
+ if (! accept_remote_cmds) {
+ rfbLog("remote commands disabled: %s\n", cmd);
+ return NULL;
+ }
+
+ if (strstr(cmd, "cmd=") == cmd) {
+ p += strlen("cmd=");
+ } else if (strstr(cmd, "qry=") == cmd) {
+ query = 1;
+ if (strchr(cmd, ',')) {
+ /* comma separated batch mode */
+ char *s, *q, *res;
+ char tmp[512];
+ strcpy(buf, "");
+ s = strdup(cmd + strlen("qry="));
+ q = strtok(s, ",");
+ while (q) {
+ sprintf(tmp, "qry=%s", q);
+ res = process_remote_cmd(tmp, 1);
+ if (res) {
+ strcat(buf, res);
+ free(res);
+ }
+ q = strtok(NULL, ",");
+ if (q) {
+ strcat(buf, ",");
+ }
+ }
+ free(s);
+ goto qry;
+ }
+ p += strlen("qry=");
+ } else {
+ rfbLog("ignoring malformed command: %s\n", cmd);
+ return NULL;
+ }
+
+ if (!query && strstr(p, "file:") == p) {
+ /* process command from a script file */
+ FILE *in;
+ char line[VNC_CONNECT_MAX], cmd[VNC_CONNECT_MAX];
+ int sz = VNC_CONNECT_MAX - 6;
+ p += strlen("file:");
+ in = fopen(p, "r");
+ if (in == NULL) {
+ rfbLog("process_remote_cmd: could not open cmd file"
+ " '%s'\n", p);
+ return NULL;
+ }
+ rfbLog("process_remote_cmd: running command from '%s'\n", p);
+ while (fgets(line, sz, in) != NULL) {
+ char *q;
+ if (strlen(line)) {
+ q = line + strlen(line) - 1;
+ if (*q == '\n') {
+ *q = '\0';
+ }
+ while (q > line && isspace(*q)) {
+ *q = '\0';
+ q--;
+ }
+ }
+ q = line;
+ while (isspace(*q)) {
+ q++;
+ }
+ if ( *q == '#' || *q == '\0') {
+ continue;
+ }
+ strcpy(cmd, "cmd=");
+ strcat(cmd, q);
+ process_remote_cmd(cmd, 0);
+ }
+ fclose(in);
+ return NULL;
+ }
+
+ /* always call like: COLON_CHECK("foobar:") */
+#define COLON_CHECK(str) \
+ if (strstr(p, str) != p) { \
+ co = ":"; \
+ if (! query) { \
+ goto done; \
+ } \
+ } else { \
+ char *q = strchr(p, ':'); \
+ if (query && q != NULL) { \
+ *(q+1) = '\0'; \
+ } \
+ }
+
+#define NOTAPP \
+ if (query) { \
+ if (strchr(p, ':')) { \
+ sprintf(buf, "ans=%sN/A", p); \
+ } else { \
+ sprintf(buf, "ans=%s:N/A", p); \
+ } \
+ goto qry; \
+ }
+
+#define NOTAPPRO \
+ if (query) { \
+ if (strchr(p, ':')) { \
+ sprintf(buf, "aro=%sN/A", p); \
+ } else { \
+ sprintf(buf, "aro=%s:N/A", p); \
+ } \
+ goto qry; \
+ }
+
+/*
+ * Add: passwdfile logfile bg nofb rfbauth passwd noshm...
+ */
+ if (!strcmp(p, "stop") || !strcmp(p, "quit") ||
+ !strcmp(p, "exit") || !strcmp(p, "shutdown")) {
+ NOTAPP
+ close_all_clients();
+ rfbLog("process_remote_cmd: setting shut_down flag\n");
+ shut_down = 1;
+
+ } else if (!strcmp(p, "ping")) {
+ query = 1;
+ if (rfb_desktop_name) {
+ sprintf(buf, "ans=%s:%s", p, rfb_desktop_name);
+ } else {
+ sprintf(buf, "ans=%s:%s", p, "unknown");
+ }
+ goto qry;
+
+ } else if (!strcmp(p, "blacken") || !strcmp(p, "zero")) {
+ NOTAPP
+ push_black_screen(4);
+ } else if (!strcmp(p, "refresh")) {
+ NOTAPP
+ refresh_screen();
+ } else if (!strcmp(p, "reset")) {
+ NOTAPP
+ do_new_fb(1);
+ } else if (strstr(p, "zero:") == p) { /* skip-cmd-list */
+ int x1, y1, x2, y2;
+ NOTAPP
+ p += strlen("zero:");
+ if (sscanf(p, "%d,%d,%d,%d", &x1, &y1, &x2, &y2) == 4) {
+ int mark = 1;
+ rfbLog("zeroing rect: %s\n", p);
+ if (x1 < 0 || x2 < 0) {
+ x1 = nabs(x1);
+ x2 = nabs(x2);
+ mark = 0; /* hack for testing */
+ }
+
+ zero_fb(x1, y1, x2, y2);
+ if (mark) {
+ mark_rect_as_modified(x1, y1, x2, y2, 1);
+ }
+ push_sleep(4);
+ }
+ } else if (strstr(p, "damagefb:") == p) { /* skip-cmd-list */
+ int delay;
+ NOTAPP
+ p += strlen("damagefb:");
+ if (sscanf(p, "%d", &delay) == 1) {
+ rfbLog("damaging client fb's for %d secs "
+ "(by not marking rects.)\n", delay);
+ damage_time = time(0);
+ damage_delay = delay;
+ }
+
+ } else if (strstr(p, "close") == p) {
+ NOTAPP
+ COLON_CHECK("close:")
+ p += strlen("close:");
+ close_clients(p);
+ } else if (strstr(p, "disconnect") == p) {
+ NOTAPP
+ COLON_CHECK("disconnect:")
+ p += strlen("disconnect:");
+ close_clients(p);
+
+ } else if (strstr(p, "id") == p) {
+ int ok = 0;
+ Window twin;
+ COLON_CHECK("id:")
+ if (query) {
+ sprintf(buf, "ans=%s%s0x%lx", p, co, subwin);
+ goto qry;
+ }
+ p += strlen("id:");
+ if (*p == '\0' || !strcmp("root", p)) {
+ /* back to root win */
+ twin = 0x0;
+ ok = 1;
+ } else if (!strcmp("pick", p)) {
+ twin = 0x0;
+ if (safe_remote_only) {
+ rfbLog("unsafe: '-id pick'\n");
+ } else if (pick_windowid(&twin)) {
+ ok = 1;
+ }
+ } else if (! scan_hexdec(p, &twin)) {
+ rfbLog("-id: incorrect hex/dec number: %s\n", p);
+ } else {
+ ok = 1;
+ }
+ if (ok) {
+ if (twin && ! valid_window(twin)) {
+ rfbLog("skipping invalid sub-window: 0x%lx\n",
+ twin);
+ } else {
+ subwin = twin;
+ rootshift = 0;
+ check_black_fb();
+ do_new_fb(1);
+ }
+ }
+ } else if (strstr(p, "sid") == p) {
+ int ok = 0;
+ Window twin;
+ COLON_CHECK("sid:")
+ if (query) {
+ sprintf(buf, "ans=%s%s0x%lx", p, co, subwin);
+ goto qry;
+ }
+ p += strlen("sid:");
+ if (*p == '\0' || !strcmp("root", p)) {
+ /* back to root win */
+ twin = 0x0;
+ ok = 1;
+ } else if (!strcmp("pick", p)) {
+ twin = 0x0;
+ if (safe_remote_only) {
+ rfbLog("unsafe: '-sid pick'\n");
+ } else if (pick_windowid(&twin)) {
+ ok = 1;
+ }
+ } else if (! scan_hexdec(p, &twin)) {
+ rfbLog("-sid: incorrect hex/dec number: %s\n", p);
+ } else {
+ ok = 1;
+ }
+ if (ok) {
+ if (twin && ! valid_window(twin)) {
+ rfbLog("skipping invalid sub-window: 0x%lx\n",
+ twin);
+ } else {
+ subwin = twin;
+ rootshift = 1;
+ check_black_fb();
+ do_new_fb(1);
+ }
+ }
+
+ } else if (!strcmp(p, "flashcmap")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, flash_cmap);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on flashcmap mode.\n");
+ flash_cmap = 1;
+ } else if (!strcmp(p, "noflashcmap")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !flash_cmap);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off flashcmap mode.\n");
+ flash_cmap = 0;
+
+ } else if (!strcmp(p, "truecolor")) {
+ int orig = force_indexed_color;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !force_indexed_color);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off notruecolor mode.\n");
+ force_indexed_color = 0;
+ if (orig != force_indexed_color) {
+ if_8bpp_do_new_fb();
+ }
+ } else if (!strcmp(p, "notruecolor")) {
+ int orig = force_indexed_color;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, force_indexed_color);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on notruecolor mode.\n");
+ force_indexed_color = 1;
+ if (orig != force_indexed_color) {
+ if_8bpp_do_new_fb();
+ }
+
+ } else if (!strcmp(p, "overlay")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, overlay);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on -overlay mode.\n");
+ if (!overlay_present) {
+ rfbLog("skipping: overlay extension not present.\n");
+ } else if (overlay) {
+ rfbLog("skipping: already in -overlay mode.\n");
+ } else {
+ int reset_mem = 0;
+ /* here we go... */
+ if (using_shm) {
+ rfbLog("setting -noshm mode.\n");
+ using_shm = 0;
+ reset_mem = 1;
+ }
+ overlay = 1;
+ do_new_fb(reset_mem);
+ }
+ } else if (!strcmp(p, "nooverlay")) {
+ int orig = overlay;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !overlay);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off overlay mode\n");
+ overlay = 0;
+ if (!overlay_present) {
+ rfbLog("warning: overlay extension not present.\n");
+ } else if (!orig) {
+ rfbLog("skipping: already not in -overlay mode.\n");
+ } else {
+ /* here we go... */
+ do_new_fb(0);
+ }
+
+ } else if (!strcmp(p, "overlay_cursor") ||
+ !strcmp(p, "overlay_yescursor")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, overlay_cursor);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on overlay_cursor mode.\n");
+ overlay_cursor = 1;
+ if (!overlay_present) {
+ rfbLog("warning: overlay extension not present.\n");
+ } else if (!overlay) {
+ rfbLog("warning: not in -overlay mode.\n");
+ } else {
+ rfbLog("You may want to run -R noshow_cursor or\n");
+ rfbLog(" -R cursor:none to disable any extra "
+ "cursors.\n");
+ }
+ } else if (!strcmp(p, "nooverlay_cursor") ||
+ !strcmp(p, "overlay_nocursor")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !overlay_cursor);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off overlay_cursor mode\n");
+ overlay_cursor = 0;
+ if (!overlay_present) {
+ rfbLog("warning: overlay extension not present.\n");
+ } else if (!overlay) {
+ rfbLog("warning: not in -overlay mode.\n");
+ } else {
+ rfbLog("You may want to run -R show_cursor or\n");
+ rfbLog(" -R cursor:... to re-enable any cursors.\n");
+ }
+
+ } else if (strstr(p, "visual") == p) {
+ COLON_CHECK("visual:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(visual_str));
+ goto qry;
+ }
+ p += strlen("visual:");
+ if (visual_str) free(visual_str);
+ visual_str = strdup(p);
+
+ /* OK, this requires a new fb... */
+ do_new_fb(0);
+
+ } else if (strstr(p, "scale") == p) {
+ COLON_CHECK("scale:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(scale_str));
+ goto qry;
+ }
+ p += strlen("scale:");
+ if (scale_str) free(scale_str);
+ scale_str = strdup(p);
+
+ /* OK, this requires a new fb... */
+ check_black_fb();
+ do_new_fb(0);
+
+ } else if (!strcmp(p, "viewonly")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, view_only); goto qry;
+ }
+ rfbLog("process_remote_cmd: enable viewonly mode.\n");
+ view_only = 1;
+ } else if (!strcmp(p, "noviewonly")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !view_only); goto qry;
+ }
+ rfbLog("process_remote_cmd: disable viewonly mode.\n");
+ view_only = 0;
+
+ } else if (!strcmp(p, "shared")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, shared); goto qry;
+ }
+ rfbLog("process_remote_cmd: enable sharing.\n");
+ shared = 1;
+ screen->alwaysShared = TRUE;
+ screen->neverShared = FALSE;
+ } else if (!strcmp(p, "noshared")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !shared); goto qry;
+ }
+ rfbLog("process_remote_cmd: disable sharing.\n");
+ shared = 0;
+ screen->alwaysShared = FALSE;
+ screen->neverShared = TRUE;
+
+ } else if (!strcmp(p, "forever")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, 1-connect_once); goto qry;
+ }
+ rfbLog("process_remote_cmd: enable -forever mode.\n");
+ connect_once = 0;
+ } else if (!strcmp(p, "noforever") || !strcmp(p, "once")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, connect_once); goto qry;
+ }
+ rfbLog("process_remote_cmd: disable -forever mode.\n");
+ connect_once = 1;
+
+ } else if (!strcmp(p, "deny") || !strcmp(p, "lock")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, deny_all); goto qry;
+ }
+ rfbLog("process_remote_cmd: denying new connections.\n");
+ deny_all = 1;
+ } else if (!strcmp(p, "nodeny") || !strcmp(p, "unlock")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !deny_all); goto qry;
+ }
+ rfbLog("process_remote_cmd: allowing new connections.\n");
+ deny_all = 0;
+
+ } else if (strstr(p, "connect") == p) {
+ NOTAPP
+ COLON_CHECK("connect:")
+ p += strlen("connect:"); /* handled at end */
+ do_vnc_connect = 1;
+
+ } else if (strstr(p, "allowonce") == p) {
+ NOTAPP
+ COLON_CHECK("allowonce:")
+ p += strlen("allowonce:");
+ allow_once = strdup(p);
+ rfbLog("process_remote_cmd: set allow_once %s\n", allow_once);
+
+ } else if (strstr(p, "allow") == p) {
+ char *before, *old;
+ COLON_CHECK("allow:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(allow_list));
+ goto qry;
+ }
+ p += strlen("allow:");
+ if (allow_list && strchr(allow_list, '/')) {
+ rfbLog("process_remote_cmd: cannot use allow:host\n");
+ rfbLog("in '-allow %s' mode.\n", allow_list);
+ return NULL;
+ }
+ if (allow_list) {
+ before = strdup(allow_list);
+ } else {
+ before = strdup("");
+ }
+
+ old = allow_list;
+ if (*p == '+') {
+ p++;
+ allow_list = add_item(allow_list, p);
+ } else if (*p == '-') {
+ p++;
+ allow_list = delete_item(allow_list, p);
+ } else {
+ allow_list = strdup(p);
+ }
+
+ if (strcmp(before, allow_list)) {
+ rfbLog("process_remote_cmd: modified allow_list:\n");
+ rfbLog(" from: \"%s\"\n", before);
+ rfbLog(" to: \"%s\"\n", allow_list);
+ }
+ if (old) free(old);
+ free(before);
+
+ } else if (!strcmp(p, "localhost")) {
+ char *before, *old;
+ if (query) {
+ int state = 0;
+ if (allow_list && !strcmp(allow_list, "127.0.0.1")) {
+ state = 1;
+ }
+ sprintf(buf, "ans=%s:%d", p, state);
+ goto qry;
+ }
+ if (allow_list) {
+ before = strdup(allow_list);
+ } else {
+ before = strdup("");
+ }
+ old = allow_list;
+
+ allow_list = strdup("127.0.0.1");
+
+ if (strcmp(before, allow_list)) {
+ rfbLog("process_remote_cmd: modified allow_list:\n");
+ rfbLog(" from: \"%s\"\n", before);
+ rfbLog(" to: \"%s\"\n", allow_list);
+ }
+ if (old) free(old);
+ free(before);
+ } else if (!strcmp(p, "nolocalhost")) {
+ char *before, *old;
+ if (query) {
+ int state = 0;
+ if (allow_list && !strcmp(allow_list, "127.0.0.1")) {
+ state = 1;
+ }
+ sprintf(buf, "ans=%s:%d", p, !state);
+ goto qry;
+ }
+ if (allow_list) {
+ before = strdup(allow_list);
+ } else {
+ before = strdup("");
+ }
+ old = allow_list;
+
+ allow_list = strdup("");
+
+ if (strcmp(before, allow_list)) {
+ rfbLog("process_remote_cmd: modified allow_list:\n");
+ rfbLog(" from: \"%s\"\n", before);
+ rfbLog(" to: \"%s\"\n", allow_list);
+ }
+ if (old) free(old);
+ free(before);
+
+ } else if (strstr(p, "accept") == p) {
+ COLON_CHECK("accept:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(accept_cmd));
+ goto qry;
+ }
+ if (safe_remote_only) {
+ rfbLog("unsafe: %s\n", p);
+ } else {
+ p += strlen("accept:");
+ if (accept_cmd) free(accept_cmd);
+ accept_cmd = strdup(p);
+ }
+
+ } else if (strstr(p, "gone") == p) {
+ COLON_CHECK("gone:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(gone_cmd));
+ goto qry;
+ }
+ if (safe_remote_only) {
+ rfbLog("unsafe: %s\n", p);
+ } else {
+ p += strlen("gone:");
+ if (gone_cmd) free(gone_cmd);
+ gone_cmd = strdup(p);
+ }
+
+ } else if (!strcmp(p, "shm")) {
+ int orig = using_shm;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, using_shm);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off noshm mode.\n");
+ using_shm = 1;
+ if (orig != using_shm) {
+ do_new_fb(1);
+ } else {
+ rfbLog(" already in shm mode.\n");
+ }
+ } else if (!strcmp(p, "noshm")) {
+ int orig = using_shm;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !using_shm);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on noshm mode.\n");
+ using_shm = 0;
+ if (orig != using_shm) {
+ do_new_fb(1);
+ } else {
+ rfbLog(" already in noshm mode.\n");
+ }
+
+ } else if (!strcmp(p, "flipbyteorder")) {
+ int orig = flip_byte_order;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, flip_byte_order);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on flipbyteorder mode.\n");
+ flip_byte_order = 1;
+ if (orig != flip_byte_order) {
+ if (! using_shm) {
+ do_new_fb(1);
+ } else {
+ rfbLog(" using shm, not resetting fb\n");
+ }
+ }
+ } else if (!strcmp(p, "noflipbyteorder")) {
+ int orig = flip_byte_order;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !flip_byte_order);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off flipbyteorder mode.\n");
+ flip_byte_order = 0;
+ if (orig != flip_byte_order) {
+ if (! using_shm) {
+ do_new_fb(1);
+ } else {
+ rfbLog(" using shm, not resetting fb\n");
+ }
+ }
+
+ } else if (!strcmp(p, "onetile")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, single_copytile); goto qry;
+ }
+ rfbLog("process_remote_cmd: enable -onetile mode.\n");
+ single_copytile = 1;
+ } else if (!strcmp(p, "noonetile")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !single_copytile);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disable -onetile mode.\n");
+ if (tile_shm_count < ntiles_x) {
+ rfbLog(" this has no effect: tile_shm_count=%d"
+ " ntiles_x=%d\n", tile_shm_count, ntiles_x);
+
+ }
+ single_copytile = 0;
+
+ } else if (strstr(p, "blackout") == p) {
+ char *before, *old;
+ COLON_CHECK("blackout:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co,
+ NONUL(blackout_string));
+ goto qry;
+ }
+ p += strlen("blackout:");
+ if (blackout_string) {
+ before = strdup(blackout_string);
+ } else {
+ before = strdup("");
+ }
+ old = blackout_string;
+ if (*p == '+') {
+ p++;
+ blackout_string = add_item(blackout_string, p);
+ } else if (*p == '-') {
+ p++;
+ blackout_string = delete_item(blackout_string, p);
+ } else {
+ blackout_string = strdup(p);
+ }
+ if (strcmp(before, blackout_string)) {
+ rfbLog("process_remote_cmd: changing -blackout\n");
+ rfbLog(" from: %s\n", before);
+ rfbLog(" to: %s\n", blackout_string);
+ if (0 && !strcmp(blackout_string, "") &&
+ single_copytile_orig != single_copytile) {
+ rfbLog("resetting single_copytile to: %d\n",
+ single_copytile_orig);
+ single_copytile = single_copytile_orig;
+ }
+ initialize_blackouts_and_xinerama();
+ }
+ if (old) free(old);
+ free(before);
+
+ } else if (!strcmp(p, "xinerama")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, xinerama); goto qry;
+ }
+ rfbLog("process_remote_cmd: enable xinerama mode."
+ "(if applicable).\n");
+ xinerama = 1;
+ initialize_blackouts_and_xinerama();
+ } else if (!strcmp(p, "noxinerama")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !xinerama); goto qry;
+ }
+ rfbLog("process_remote_cmd: disable xinerama mode."
+ "(if applicable).\n");
+ xinerama = 0;
+ initialize_blackouts_and_xinerama();
+
+ } else if (!strcmp(p, "xrandr")) {
+ int orig = xrandr;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, xrandr); goto qry;
+ }
+ if (xrandr_present) {
+ rfbLog("process_remote_cmd: enable xrandr mode.\n");
+ xrandr = 1;
+ if (! xrandr_mode) {
+ xrandr_mode = strdup("default");
+ }
+ if (orig != xrandr) {
+ initialize_xrandr();
+ }
+ } else {
+ rfbLog("process_remote_cmd: XRANDR ext. not "
+ "present.\n");
+ }
+ } else if (!strcmp(p, "noxrandr")) {
+ int orig = xrandr;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !xrandr); goto qry;
+ }
+ xrandr = 0;
+ if (xrandr_present) {
+ rfbLog("process_remote_cmd: disable xrandr mode.\n");
+ if (orig != xrandr) {
+ initialize_xrandr();
+ }
+ } else {
+ rfbLog("process_remote_cmd: XRANDR ext. not "
+ "present.\n");
+ }
+ } else if (strstr(p, "xrandr_mode") == p) {
+ int orig = xrandr;
+ COLON_CHECK("xrandr_mode:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(xrandr_mode));
+ goto qry;
+ }
+ p += strlen("xrandr_mode:");
+ if (!strcmp("none", p)) {
+ xrandr = 0;
+ } else {
+ if (known_xrandr_mode(p)) {
+ if (xrandr_mode) free(xrandr_mode);
+ xrandr_mode = strdup(p);
+ } else {
+ rfbLog("skipping unknown xrandr mode: %s\n", p);
+ return NULL;
+ }
+ xrandr = 1;
+ }
+ if (xrandr_present) {
+ if (xrandr) {
+ rfbLog("process_remote_cmd: enable xrandr"
+ " mode.\n");
+ } else {
+ rfbLog("process_remote_cmd: disable xrandr"
+ " mode.\n");
+ }
+ if (! xrandr_mode) {
+ xrandr_mode = strdup("default");
+ }
+ if (orig != xrandr) {
+ initialize_xrandr();
+ }
+ } else {
+ rfbLog("process_remote_cmd: XRANDR ext. not "
+ "present.\n");
+ }
+
+ } else if (strstr(p, "padgeom") == p) {
+ COLON_CHECK("padgeom:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(pad_geometry));
+ goto qry;
+ }
+ p += strlen("padgeom:");
+ if (!strcmp("force", p) || !strcmp("do",p) || !strcmp("go",p)) {
+ rfbLog("process_remote_cmd: invoking "
+ "install_padded_fb()\n");
+ install_padded_fb(pad_geometry);
+ } else {
+ if (pad_geometry) free(pad_geometry);
+ pad_geometry = strdup(p);
+ rfbLog("process_remote_cmd: set padgeom to: %s\n",
+ pad_geometry);
+ }
+
+ } else if (!strcmp(p, "quiet") || !strcmp(p, "q")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, quiet); goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on quiet mode.\n");
+ quiet = 1;
+ } else if (!strcmp(p, "noquiet")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !quiet); goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off quiet mode.\n");
+ quiet = 0;
+
+ } else if (!strcmp(p, "modtweak")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, use_modifier_tweak);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling -modtweak mode.\n");
+ if (! use_modifier_tweak) {
+ use_modifier_tweak = 1;
+ initialize_modtweak();
+ }
+ use_modifier_tweak = 1;
+
+ } else if (!strcmp(p, "nomodtweak")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !use_modifier_tweak);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling -nomodtweak mode.\n");
+ use_modifier_tweak = 0;
+
+ } else if (!strcmp(p, "xkb")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, use_xkb_modtweak);
+ goto qry;
+ }
+ if (! xkb_present) {
+ rfbLog("process_remote_cmd: cannot enable -xkb "
+ "modtweak mode (not supported on X display)\n");
+ return NULL;
+ }
+ rfbLog("process_remote_cmd: enabling -xkb modtweak mode"
+ " (if supported).\n");
+ if (! use_modifier_tweak || ! use_xkb_modtweak) {
+ use_modifier_tweak = 1;
+ use_xkb_modtweak = 1;
+ initialize_modtweak();
+ }
+ use_modifier_tweak = 1;
+ use_xkb_modtweak = 1;
+
+ } else if (!strcmp(p, "noxkb")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !use_xkb_modtweak);
+ goto qry;
+ }
+ if (! xkb_present) {
+ rfbLog("process_remote_cmd: cannot disable -xkb "
+ "modtweak mode (not supported on X display)\n");
+ return NULL;
+ }
+ rfbLog("process_remote_cmd: disabling -xkb modtweak mode.\n");
+ use_xkb_modtweak = 0;
+
+ } else if (strstr(p, "skip_keycodes") == p) {
+ COLON_CHECK("skip_keycodes:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(skip_keycodes));
+ goto qry;
+ }
+ p += strlen("skip_keycodes:");
+ rfbLog("process_remote_cmd: setting xkb -skip_keycodes"
+ " to:\n\t'%s'\n", p);
+ if (! xkb_present) {
+ rfbLog("process_remote_cmd: warning xkb not present\n");
+ } else if (! use_xkb_modtweak) {
+ rfbLog("process_remote_cmd: turning on xkb.\n");
+ use_xkb_modtweak = 1;
+ if (! use_modifier_tweak) {
+ rfbLog("process_remote_cmd: turning on "
+ "modtweak.\n");
+ use_modifier_tweak = 1;
+ }
+ }
+ if (skip_keycodes) free(skip_keycodes);
+ skip_keycodes = strdup(p);
+ initialize_modtweak();
+
+ } else if (!strcmp(p, "add_keysyms")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, add_keysyms);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling -add_keysyms mode.\n");
+ add_keysyms = 1;
+
+ } else if (!strcmp(p, "noadd_keysyms")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !add_keysyms);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disabling -add_keysyms mode.\n");
+ add_keysyms = 0;
+
+ } else if (!strcmp(p, "clear_mods")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, clear_mods == 1);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling -clear_mods mode.\n");
+ clear_mods = 1;
+ clear_modifiers(0);
+
+ } else if (!strcmp(p, "noclear_mods")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !(clear_mods == 1));
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disabling -clear_mods mode.\n");
+ clear_mods = 0;
+
+ } else if (!strcmp(p, "clear_keys")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, clear_mods == 2);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling -clear_keys mode.\n");
+ clear_mods = 2;
+ clear_keys();
+
+ } else if (!strcmp(p, "noclear_keys")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !(clear_mods == 2));
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disabling -clear_keys mode.\n");
+ clear_mods = 0;
+
+ } else if (strstr(p, "remap") == p) {
+ char *before, *old;
+ COLON_CHECK("remap:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(remap_file));
+ goto qry;
+ }
+ p += strlen("remap:");
+ if ((*p == '+' || *p == '-') && remap_file &&
+ strchr(remap_file, '/')) {
+ rfbLog("process_remote_cmd: cannot use remap:+/-\n");
+ rfbLog("in '-remap %s' mode.\n", remap_file);
+ return NULL;
+ }
+ if (remap_file) {
+ before = strdup(remap_file);
+ } else {
+ before = strdup("");
+ }
+ old = remap_file;
+ if (*p == '+') {
+ p++;
+ remap_file = add_item(remap_file, p);
+ } else if (*p == '-') {
+ p++;
+ remap_file = delete_item(remap_file, p);
+ if (! strchr(remap_file, '-')) {
+ *remap_file = '\0';
+ }
+ } else {
+ remap_file = strdup(p);
+ }
+ if (strcmp(before, remap_file)) {
+ rfbLog("process_remote_cmd: changed -remap\n");
+ rfbLog(" from: %s\n", before);
+ rfbLog(" to: %s\n", remap_file);
+ initialize_remap(remap_file);
+ }
+ if (old) free(old);
+ free(before);
+
+ } else if (!strcmp(p, "repeat")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !no_autorepeat);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling -repeat mode.\n");
+ if (no_autorepeat) {
+ autorepeat(1);
+ }
+ no_autorepeat = 0;
+
+ } else if (!strcmp(p, "norepeat")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, no_autorepeat);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling -norepeat mode.\n");
+ if (! no_autorepeat && client_count) {
+ no_autorepeat = 1;
+ autorepeat(0);
+ }
+ no_autorepeat = 1;
+
+ } else if (!strcmp(p, "bell")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, sound_bell);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling bell (if supported).\n");
+ sound_bell = 1;
+
+ } else if (!strcmp(p, "nobell")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !sound_bell);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disabling bell.\n");
+ sound_bell = 0;
+
+ } else if (!strcmp(p, "sel")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, watch_selection);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling watch_selection.\n");
+ watch_selection = 1;
+
+ } else if (!strcmp(p, "nosel")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !watch_selection);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disabling watch_selection.\n");
+ watch_selection = 0;
+
+ } else if (!strcmp(p, "primary")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, watch_primary);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling watch_primary.\n");
+ watch_primary = 1;
+
+ } else if (!strcmp(p, "noprimary")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !watch_primary);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disabling watch_primary.\n");
+ watch_primary = 0;
+
+ } else if (!strcmp(p, "set_no_cursor")) { /* skip-cmd-list */
+ rfbLog("process_remote_cmd: calling set_no_cursor()\n");
+ set_no_cursor();
+
+ } else if (!strcmp(p, "cursorshape")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, cursor_shape_updates);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on cursorshape mode.\n");
+
+ rfbUndrawCursor(screen);
+ set_no_cursor();
+ cursor_shape_updates = 1;
+ restore_cursor_shape_updates(screen);
+ } else if (!strcmp(p, "nocursorshape")) {
+ int i, max = 5;
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !cursor_shape_updates);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off cursorshape mode.\n");
+
+ rfbUndrawCursor(screen);
+ set_no_cursor();
+ for (i=0; i<max; i++) {
+ /* XXX: try to force empty cursor back to client */
+ rfbPE(screen, -1);
+ }
+ cursor_shape_updates = 0;
+ disable_cursor_shape_updates(screen);
+
+ } else if (!strcmp(p, "cursorpos")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, cursor_pos_updates);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on cursorpos mode.\n");
+ cursor_pos_updates = 1;
+ } else if (!strcmp(p, "nocursorpos")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !cursor_pos_updates);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off cursorpos mode.\n");
+ cursor_pos_updates = 0;
+
+ } else if (strstr(p, "cursor") == p) {
+ COLON_CHECK("cursor:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co,
+ NONUL(multiple_cursors_mode));
+ goto qry;
+ }
+ p += strlen("cursor:");
+ if (multiple_cursors_mode) {
+ if (prev_cursors_mode) free(prev_cursors_mode);
+ prev_cursors_mode = strdup(multiple_cursors_mode);
+ free(multiple_cursors_mode);
+ }
+ multiple_cursors_mode = strdup(p);
+
+ rfbLog("process_remote_cmd: changed -cursor mode "
+ "to: %s\n", multiple_cursors_mode);
+
+ if (strcmp(multiple_cursors_mode, "none") && !show_cursor) {
+ show_cursor = 1;
+ rfbLog("process_remote_cmd: changed show_cursor "
+ "to: %d\n", show_cursor);
+ }
+ initialize_cursors_mode();
+
+ } else if (!strcmp(p, "show_cursor")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, show_cursor);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling show_cursor.\n");
+ show_cursor = 1;
+ if (multiple_cursors_mode && !strcmp(multiple_cursors_mode,
+ "none")) {
+ free(multiple_cursors_mode);
+ if (prev_cursors_mode) {
+ multiple_cursors_mode =
+ strdup(prev_cursors_mode);
+ } else {
+ multiple_cursors_mode = strdup("default");
+ }
+ rfbLog("process_remote_cmd: changed -cursor mode "
+ "to: %s\n", multiple_cursors_mode);
+ }
+ initialize_cursors_mode();
+ } else if (!strcmp(p, "noshow_cursor") || !strcmp(p, "nocursor")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !show_cursor);
+ goto qry;
+ }
+ if (prev_cursors_mode) free(prev_cursors_mode);
+ prev_cursors_mode = strdup(multiple_cursors_mode);
+
+ rfbLog("process_remote_cmd: disabling show_cursor.\n");
+ show_cursor = 0;
+ initialize_cursors_mode();
+
+ } else if (!strcmp(p, "xfixes")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, use_xfixes);
+ goto qry;
+ }
+ if (! xfixes_present) {
+ rfbLog("process_remote_cmd: cannot enable xfixes "
+ "(not supported on X display)\n");
+ return NULL;
+ }
+ rfbLog("process_remote_cmd: enabling -xfixes"
+ " (if supported).\n");
+ use_xfixes = 1;
+ initialize_xfixes();
+ } else if (!strcmp(p, "noxfixes")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !use_xfixes);
+ goto qry;
+ }
+ if (! xfixes_present) {
+ rfbLog("process_remote_cmd: disabling xfixes "
+ "(but not supported on X display)\n");
+ return NULL;
+ }
+ rfbLog("process_remote_cmd: disabling -xfixes.\n");
+ use_xfixes = 0;
+ initialize_xfixes();
+
+ } else if (strstr(p, "xwarp") == p || strstr(p, "xwarppointer")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, use_xwarppointer);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on xwarppointer mode.\n");
+ use_xwarppointer = 1;
+ } else if (strstr(p, "noxwarp") == p || strstr(p, "noxwarppointer")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !use_xwarppointer);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off xwarppointer mode.\n");
+ use_xwarppointer = 0;
+
+ } else if (strstr(p, "buttonmap") == p) {
+ COLON_CHECK("buttonmap:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%s", p, co, NONUL(pointer_remap));
+ goto qry;
+ }
+ p += strlen("buttonmap:");
+ if (pointer_remap) free(pointer_remap);
+ pointer_remap = strdup(p);
+
+ rfbLog("process_remote_cmd: setting -buttonmap to:\n"
+ "\t'%s'\n", p);
+ initialize_pointer_map(p);
+
+ } else if (!strcmp(p, "dragging")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, show_dragging);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling mouse dragging mode.\n");
+ show_dragging = 1;
+ } else if (!strcmp(p, "nodragging")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !show_dragging);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: enabling mouse nodragging mode.\n");
+ show_dragging = 0;
+
+ } else if (strstr(p, "pointer_mode") == p) {
+ int pm;
+ COLON_CHECK("pointer_mode:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, pointer_mode);
+ goto qry;
+ }
+ p += strlen("pointer_mode:");
+ pm = atoi(p);
+ if (pm < 1 || pm > pointer_mode_max) {
+ rfbLog("process_remote_cmd: pointer_mode out of range:"
+ " 1-%d: %d\n", pointer_mode_max, pm);
+ } else {
+ rfbLog("process_remote_cmd: setting pointer_mode %d\n",
+ pm);
+ pointer_mode = pm;
+ }
+ } else if (strstr(p, "input_skip") == p) {
+ int is;
+ COLON_CHECK("input_skip:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, ui_skip);
+ goto qry;
+ }
+ p += strlen("input_skip:");
+ is = atoi(p);
+ rfbLog("process_remote_cmd: setting input_skip %d\n", is);
+ ui_skip = is;
+
+ } else if (!strcmp(p, "debug_pointer") || !strcmp(p, "dp")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, debug_pointer);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on debug_pointer.\n");
+ debug_pointer = 1;
+ } else if (!strcmp(p, "nodebug_pointer") || !strcmp(p, "nodp")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !debug_pointer);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off debug_pointer.\n");
+ debug_pointer = 0;
+
+ } else if (!strcmp(p, "debug_keyboard") || !strcmp(p, "dk")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, debug_keyboard);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on debug_keyboard.\n");
+ debug_keyboard = 1;
+ } else if (!strcmp(p, "nodebug_keyboard") || !strcmp(p, "nodk")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !debug_keyboard);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off debug_keyboard.\n");
+ debug_keyboard = 0;
+
+ } else if (strstr(p, "deferupdate") == p) {
+ int d;
+ COLON_CHECK("deferupdate:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co,
+ screen->deferUpdateTime);
+ goto qry;
+ }
+ p += strlen("deferupdate:");
+ d = atoi(p);
+ if (d < 0) d = 0;
+ rfbLog("process_remote_cmd: setting defer to %d ms.\n", d);
+ screen->deferUpdateTime = d;
+
+ } else if (strstr(p, "defer") == p) {
+ int d;
+ COLON_CHECK("defer:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co,
+ screen->deferUpdateTime);
+ goto qry;
+ }
+ p += strlen("defer:");
+ d = atoi(p);
+ if (d < 0) d = 0;
+ rfbLog("process_remote_cmd: setting defer to %d ms.\n", d);
+ /* XXX not part of API? */
+ screen->deferUpdateTime = d;
+
+ } else if (strstr(p, "wait") == p) {
+ int w;
+ COLON_CHECK("wait:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, waitms);
+ goto qry;
+ }
+ p += strlen("wait:");
+ w = atoi(p);
+ if (w < 0) w = 0;
+ rfbLog("process_remote_cmd: setting wait %d -> %d ms.\n",
+ waitms, w);
+ waitms = w;
+
+ } else if (!strcmp(p, "nap")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, take_naps); goto qry;
+ }
+ rfbLog("process_remote_cmd: turning on nap mode.\n");
+ take_naps = 1;
+ } else if (!strcmp(p, "nonap")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !take_naps); goto qry;
+ }
+ rfbLog("process_remote_cmd: turning off nap mode.\n");
+ take_naps = 0;
+
+ } else if (strstr(p, "sb") == p) {
+ int w;
+ COLON_CHECK("sb:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, screen_blank);
+ goto qry;
+ }
+ p += strlen("sb:");
+ w = atoi(p);
+ if (w < 0) w = 0;
+ rfbLog("process_remote_cmd: setting screen_blank %d -> %d"
+ " sec.\n", screen_blank, w);
+ screen_blank = w;
+ } else if (strstr(p, "screen_blank") == p) {
+ int w;
+ COLON_CHECK("screen_blank:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, screen_blank);
+ goto qry;
+ }
+ p += strlen("screen_blank:");
+ w = atoi(p);
+ if (w < 0) w = 0;
+ rfbLog("process_remote_cmd: setting screen_blank %d -> %d"
+ " sec.\n", screen_blank, w);
+ screen_blank = w;
+
+ } else if (strstr(p, "fs") == p) {
+ COLON_CHECK("fs:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%f", p, co, fs_frac);
+ goto qry;
+ }
+ p += strlen("fs:");
+ fs_frac = atof(p);
+ rfbLog("process_remote_cmd: setting -fs frac to %f\n", fs_frac);
+
+ } else if (strstr(p, "gaps") == p) {
+ int g;
+ COLON_CHECK("gaps:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, gaps_fill);
+ goto qry;
+ }
+ p += strlen("gaps:");
+ g = atoi(p);
+ if (g < 0) g = 0;
+ rfbLog("process_remote_cmd: setting gaps_fill %d -> %d.\n",
+ gaps_fill, g);
+ gaps_fill = g;
+ } else if (strstr(p, "grow") == p) {
+ int g;
+ COLON_CHECK("grow:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, grow_fill);
+ goto qry;
+ }
+ p += strlen("grow:");
+ g = atoi(p);
+ if (g < 0) g = 0;
+ rfbLog("process_remote_cmd: setting grow_fill %d -> %d.\n",
+ grow_fill, g);
+ grow_fill = g;
+ } else if (strstr(p, "fuzz") == p) {
+ int f;
+ COLON_CHECK("fuzz:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co, tile_fuzz);
+ goto qry;
+ }
+ p += strlen("fuzz:");
+ f = atoi(p);
+ if (f < 0) f = 0;
+ rfbLog("process_remote_cmd: setting tile_fuzz %d -> %d.\n",
+ tile_fuzz, f);
+ grow_fill = f;
+
+ } else if (strstr(p, "progressive") == p) {
+ int f;
+ COLON_CHECK("progressive:")
+ if (query) {
+ sprintf(buf, "ans=%s%s%d", p, co,
+ screen->progressiveSliceHeight);
+ goto qry;
+ }
+ p += strlen("progressive:");
+ f = atoi(p);
+ if (f < 0) f = 0;
+ rfbLog("process_remote_cmd: setting progressive %d -> %d.\n",
+ screen->progressiveSliceHeight, f);
+ screen->progressiveSliceHeight = f;
+
+ } else if (!strcmp(p, "noremote")) {
+ if (query) {
+ sprintf(buf, "ans=%s:%d", p, !accept_remote_cmds);
+ goto qry;
+ }
+ rfbLog("process_remote_cmd: disabling remote commands.\n");
+ accept_remote_cmds = 0; /* cannot be turned back on. */
+
+ } else if (query) {
+ /* read-only variables that can only be queried: */
+
+ if (!strcmp(p, "display")) {
+ char *d = DisplayString(dpy);
+ if (! d) d = "unknown";
+ if (*d == ':') {
+ sprintf(buf, "aro=%s:%s%s", p, this_host(), d);
+ } else {
+ sprintf(buf, "aro=%s:%s", p, d);
+ }
+ } else if (!strcmp(p, "vncdisplay")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(vnc_desktop_name));
+ } else if (!strcmp(p, "desktopname")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(rfb_desktop_name));
+ } else if (!strcmp(p, "desktop")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(rfb_desktop_name));
+ } else if (!strcmp(p, "auth")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(auth_file));
+ } else if (!strcmp(p, "rootshift")) {
+ sprintf(buf, "aro=%s:%d", p, rootshift);
+ } else if (!strcmp(p, "scale_str")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(scale_str));
+ } else if (!strcmp(p, "scaled_x")) {
+ sprintf(buf, "aro=%s:%d", p, scaled_x);
+ } else if (!strcmp(p, "scaled_y")) {
+ sprintf(buf, "aro=%s:%d", p, scaled_y);
+ } else if (!strcmp(p, "scale_numer")) {
+ sprintf(buf, "aro=%s:%d", p, scale_numer);
+ } else if (!strcmp(p, "scale_denom")) {
+ sprintf(buf, "aro=%s:%d", p, scale_denom);
+ } else if (!strcmp(p, "scale_fac")) {
+ sprintf(buf, "aro=%s:%f", p, scale_fac);
+ } else if (!strcmp(p, "scaling_noblend")) {
+ sprintf(buf, "aro=%s:%d", p, scaling_noblend);
+ } else if (!strcmp(p, "scaling_nomult4")) {
+ sprintf(buf, "aro=%s:%d", p, scaling_nomult4);
+ } else if (!strcmp(p, "scaling_pad")) {
+ sprintf(buf, "aro=%s:%d", p, scaling_pad);
+ } else if (!strcmp(p, "scaling_interpolate")) {
+ sprintf(buf, "aro=%s:%d", p, scaling_interpolate);
+ } else if (!strcmp(p, "inetd")) {
+ sprintf(buf, "aro=%s:%d", p, inetd);
+ } else if (!strcmp(p, "safer")) {
+ sprintf(buf, "aro=%s:%d", p, safe_remote_only);
+ } else if (!strcmp(p, "unsafe")) {
+ sprintf(buf, "aro=%s:%d", p, !safe_remote_only);
+ } else if (!strcmp(p, "passwdfile")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(passwdfile));
+ } else if (!strcmp(p, "using_shm")) {
+ sprintf(buf, "aro=%s:%d", p, !using_shm);
+ } else if (!strcmp(p, "logfile") || !strcmp(p, "o")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(logfile));
+ } else if (!strcmp(p, "rc")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(rc_rcfile));
+ } else if (!strcmp(p, "norc")) {
+ sprintf(buf, "aro=%s:%d", p, rc_norc);
+ } else if (!strcmp(p, "h") || !strcmp(p, "help") ||
+ !strcmp(p, "V") || !strcmp(p, "version") ||
+ !strcmp(p, "lastmod")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(lastmod));
+ } else if (!strcmp(p, "bg")) {
+ sprintf(buf, "aro=%s:%d", p, opts_bg);
+ } else if (!strcmp(p, "nofb")) {
+ sprintf(buf, "aro=%s:%d", p, nofb);
+ } else if (!strcmp(p, "sigpipe")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(sigpipe));
+ } else if (!strcmp(p, "threads")) {
+ sprintf(buf, "aro=%s:%d", p, use_threads);
+ } else if (!strcmp(p, "clients")) {
+ char *str = list_clients();
+ sprintf(buf, "aro=%s:%s", p, str);
+ free(str);
+ } else if (!strcmp(p, "client_count")) {
+ sprintf(buf, "aro=%s:%d", p, client_count);
+ } else if (!strcmp(p, "pid")) {
+ sprintf(buf, "aro=%s:%d", p, (int) getpid());
+ } else if (!strcmp(p, "ext_xtest")) {
+ sprintf(buf, "aro=%s:%d", p, xtest_present);
+ } else if (!strcmp(p, "ext_xkb")) {
+ sprintf(buf, "aro=%s:%d", p, xkb_present);
+ } else if (!strcmp(p, "ext_xshm")) {
+ sprintf(buf, "aro=%s:%d", p, xshm_present);
+ } else if (!strcmp(p, "ext_xinerama")) {
+ sprintf(buf, "aro=%s:%d", p, xinerama_present);
+ } else if (!strcmp(p, "ext_overlay")) {
+ sprintf(buf, "aro=%s:%d", p, overlay_present);
+ } else if (!strcmp(p, "ext_xfixes")) {
+ sprintf(buf, "aro=%s:%d", p, xfixes_present);
+ } else if (!strcmp(p, "ext_xdamage")) {
+ sprintf(buf, "aro=%s:%d", p, xdamage_present);
+ } else if (!strcmp(p, "ext_xrandr")) {
+ sprintf(buf, "aro=%s:%d", p, xrandr_present);
+ } else if (!strcmp(p, "rootwin")) {
+ sprintf(buf, "aro=%s:0x%x", p, (unsigned int) rootwin);
+ } else if (!strcmp(p, "num_buttons")) {
+ sprintf(buf, "aro=%s:%d", p, num_buttons);
+ } else if (!strcmp(p, "button_mask")) {
+ sprintf(buf, "aro=%s:%d", p, button_mask);
+ } else if (!strcmp(p, "mouse_x")) {
+ sprintf(buf, "aro=%s:%d", p, cursor_x);
+ } else if (!strcmp(p, "mouse_y")) {
+ sprintf(buf, "aro=%s:%d", p, cursor_y);
+ } else if (!strcmp(p, "bpp")) {
+ sprintf(buf, "aro=%s:%d", p, bpp);
+ } else if (!strcmp(p, "depth")) {
+ sprintf(buf, "aro=%s:%d", p, depth);
+ } else if (!strcmp(p, "indexed_color")) {
+ sprintf(buf, "aro=%s:%d", p, indexed_color);
+ } else if (!strcmp(p, "dpy_x")) {
+ sprintf(buf, "aro=%s:%d", p, dpy_x);
+ } else if (!strcmp(p, "dpy_y")) {
+ sprintf(buf, "aro=%s:%d", p, dpy_y);
+ } else if (!strcmp(p, "rfbport")) {
+ sprintf(buf, "aro=%s:%d", p, screen->port);
+ } else if (!strcmp(p, "rfbwait")) {
+ NOTAPPRO
+ } else if (!strcmp(p, "rfbauth")) {
+ NOTAPPRO
+ } else if (!strcmp(p, "passwd")) {
+ NOTAPPRO
+ } else if (!strcmp(p, "alwaysshared")) {
+ sprintf(buf, "aro=%s:%d", p, screen->alwaysShared);
+ } else if (!strcmp(p, "dontdisconnect")) {
+ sprintf(buf, "aro=%s:%d", p, screen->dontDisconnect);
+ } else if (!strcmp(p, "httpdir")) {
+ sprintf(buf, "aro=%s:%s", p, NONUL(screen->httpDir));
+ } else if (!strcmp(p, "enablehttpproxy")) {
+ sprintf(buf, "aro=%s:%d", p,
+ screen->httpEnableProxyConnect);
+ } else {
+ NOTAPP
+ }
+ goto qry;
+ } else {
+ NOTAPP
+ rfbLog("process_remote_cmd: warning unknown\n");
+ rfbLog("command \"%s\"\n", p);
+ return NULL;
+ }
+
+ if (*p != '\0' && do_vnc_connect) {
+ /* this is a reverse connection */
+ strncpy(vnc_connect_str, p, VNC_CONNECT_MAX);
+ vnc_connect_str[VNC_CONNECT_MAX] = '\0';
+ }
+
+ done:
+ return NULL;
+
+ qry:
+
+ if (stringonly) {
+ return strdup(buf);
+ } else if (client_connect_file) {
+ FILE *out = fopen(client_connect_file, "w");
+ if (out != NULL) {
+ fprintf(out, "%s\n", buf);
+ fclose(out);
+ usleep(200*1000);
+ }
+ } else {
+ set_vnc_connect_prop(buf);
+ XFlush(dpy);
+ }
+ return NULL;
+}
+
/* -- cursor.c -- */
/*
* Here begins a bit of a mess to experiment with multiple cursors
@@ -4450,6 +7282,26 @@ typedef struct cursor_info {
rfbCursorPtr rfb;
} cursor_info_t;
+/* empty cursor */
+static char* curs_empty_data =
+" "
+" ";
+
+static char* curs_empty_mask =
+" "
+" ";
+static cursor_info_t cur_empty = {NULL, NULL, 2, 2, 0, 0, 0, NULL};
+
+/* dot cursor */
+static char* curs_dot_data =
+" "
+" x";
+
+static char* curs_dot_mask =
+" "
+" x";
+static cursor_info_t cur_dot = {NULL, NULL, 2, 2, 0, 0, 0, NULL};
+
/* main cursor */
static char* curs_arrow_data =
" "
@@ -4642,18 +7494,107 @@ static char* curs_xterm_mask =
static cursor_info_t cur_xterm = {NULL, NULL, 16, 16, 8, 8, 1, NULL};
enum cursor_names {
- CURS_ARROW = 0,
+ CURS_EMPTY = 0,
+ CURS_DOT,
+
+ CURS_ARROW,
CURS_ROOT,
CURS_WM,
- CURS_TERM
+ CURS_TERM,
+
+ CURS_DYN1,
+ CURS_DYN2,
+ CURS_DYN3,
+ CURS_DYN4,
+ CURS_DYN5,
+ CURS_DYN6,
+ CURS_DYN7,
+ CURS_DYN8,
+ CURS_DYN9,
+ CURS_DYN10,
+ CURS_DYN11,
+ CURS_DYN12,
+ CURS_DYN13,
+ CURS_DYN14,
+ CURS_DYN15,
+ CURS_DYN16
};
-static cursor_info_t *cursors[10];
+#define CURS_DYN_MIN CURS_DYN1
+#define CURS_DYN_MAX CURS_DYN16
+#define CURS_DYN_NUM (CURS_DYN_MAX - CURS_DYN_MIN + 1)
+
+#define CURS_MAX 32
+static cursor_info_t *cursors[CURS_MAX];
static void setup_cursors(void) {
- /* TODO clean this up if we ever do more cursors... */
rfbCursorPtr rfb_curs;
int i, n = 0;
+ static int first = 1;
+
+ rfbLog("setting up %d cursors...\n", CURS_MAX);
+
+ if (first) {
+ for (i=0; i<CURS_MAX; i++) {
+ cursors[i] = NULL;
+ }
+ }
+ first = 0;
+
+ if (screen) {
+ rfbUndrawCursor(screen);
+ screen->cursor = NULL;
+ LOCK(screen->cursorMutex);
+ }
+
+ for (i=0; i<CURS_MAX; i++) {
+ cursor_info_t *ci;
+ if (cursors[i]) {
+ /* clear out any existing ones: */
+ ci = cursors[i];
+ if (ci->rfb) {
+ /* this is the rfbCursor part: */
+ if (ci->rfb->richSource) {
+ free(ci->rfb->richSource);
+ }
+ if (ci->rfb->source) {
+ free(ci->rfb->source);
+ }
+ if (ci->rfb->mask) {
+ free(ci->rfb->mask);
+ }
+ free(ci->rfb);
+ }
+ if (ci->data) {
+ free(ci->data);
+ }
+ if (ci->mask) {
+ free(ci->mask);
+ }
+ }
+
+ /* create new struct: */
+ ci = (cursor_info_t *) malloc(sizeof(cursor_info_t));
+ ci->data = NULL;
+ ci->mask = NULL;
+ ci->wx = 0;
+ ci->wy = 0;
+ ci->sx = 0;
+ ci->sy = 0;
+ ci->reverse = 0;
+ ci->rfb = NULL;
+ cursors[i] = ci;
+ }
+
+ /* clear any xfixes cursor cache (no freeing is done) */
+ get_xfixes_cursor(1);
+
+ /* manually fill in the data+masks: */
+ cur_empty.data = curs_empty_data;
+ cur_empty.mask = curs_empty_mask;
+
+ cur_dot.data = curs_dot_data;
+ cur_dot.mask = curs_dot_mask;
cur_arrow.data = curs_arrow_data;
cur_arrow.mask = curs_arrow_mask;
@@ -4670,12 +7611,16 @@ static void setup_cursors(void) {
cur_xterm.data = curs_xterm_data;
cur_xterm.mask = curs_xterm_mask;
+ cursors[CURS_EMPTY] = &cur_empty; n++;
+ cursors[CURS_DOT] = &cur_dot; n++;
cursors[CURS_ARROW] = &cur_arrow; n++;
cursors[CURS_ROOT] = &cur_root; n++;
cursors[CURS_WM] = &cur_fleur; n++;
cursors[CURS_TERM] = &cur_xterm; n++;
for (i=0; i<n; i++) {
+ /* create rfbCursors for the special cursors: */
+
cursor_info_t *ci = cursors[i];
ci->data = strdup(ci->data);
@@ -4698,8 +7643,49 @@ static void setup_cursors(void) {
rfb_curs->cleanupMask = FALSE;
rfb_curs->cleanupRichSource = FALSE;
+ if (bpp == 8 && indexed_color) {
+ /*
+ * use richsource in PseudoColor for better
+ * looking cursors (i.e. two-color).
+ */
+ int x, y, k = 0, bw;
+ char d, m;
+ int black = BlackPixel(dpy, scr);
+ int white = WhitePixel(dpy, scr);
+
+ rfb_curs->richSource
+ = (char *)calloc(ci->wx * ci->wy, 1);
+
+ for (y = 0; y < ci->wy; y++) {
+ for (x = 0; x < ci->wx; x++) {
+ d = *(ci->data + k);
+ m = *(ci->mask + k);
+ if (d == ' ' && m == ' ') {
+ k++;
+ continue;
+ } else if (m != ' ' && d == ' ') {
+ bw = black;
+ } else {
+ bw = white;
+ }
+ if (ci->reverse) {
+ if (bw == black) {
+ bw = white;
+ } else {
+ bw = black;
+ }
+ }
+ *(rfb_curs->richSource+k) = (unsigned char) bw;
+ k++;
+ }
+ }
+ }
ci->rfb = rfb_curs;
}
+ if (screen) {
+ UNLOCK(screen->cursorMutex);
+ }
+ rfbLog(" done.\n");
}
typedef struct win_str_info {
@@ -4722,15 +7708,14 @@ void tree_descend_cursor(int *depth, Window *w, win_str_info_t *winfo) {
unsigned int mask;
Window wins[10];
int descend, maxtries = 10;
- char *name, a;
+ char *name, *s = multiple_cursors_mode;
static XClassHint *classhint = NULL;
int nm_info = 1;
XErrorHandler old_handler;
X_LOCK;
- a = multiple_cursors_mode[0];
- if (a == 'd' || a == 'X') {
+ if (!strcmp(s, "default") || !strcmp(s, "X") || !strcmp(s, "arrow")) {
nm_info = 0;
}
@@ -4814,6 +7799,304 @@ void tree_descend_cursor(int *depth, Window *w, win_str_info_t *winfo) {
*w = wins[descend];
}
+void initialize_xfixes(void) {
+#ifdef LIBVNCSERVER_HAVE_LIBXFIXES
+ if (xfixes_present) {
+ if (use_xfixes) {
+ XFixesSelectCursorInput(dpy, rootwin,
+ XFixesDisplayCursorNotifyMask);
+ } else {
+ XFixesSelectCursorInput(dpy, rootwin, 0);
+ }
+ }
+#endif
+}
+
+int get_xfixes_cursor(int init) {
+ static unsigned long last_cursor = 0;
+ static int last_index = 0;
+ static time_t curs_times[CURS_MAX];
+ static unsigned long curs_index[CURS_MAX];
+ int which = CURS_ARROW;
+
+ if (init) {
+ /* zero out our cache (cursors are not freed) */
+ int i;
+ for (i=0; i<CURS_MAX; i++) {
+ curs_times[i] = 0;
+ curs_index[i] = 0;
+ }
+ last_cursor = 0;
+ last_index = 0;
+ return -1;
+ }
+
+ if (xfixes_present) {
+#ifdef LIBVNCSERVER_HAVE_LIBXFIXES
+ int use, oldest, i, x, y, w, h, len;
+ int Bpp = bpp/8;
+ time_t oldtime, now;
+ char *bitmap, *rich;
+ unsigned long black, white;
+ rfbCursorPtr c;
+ XFixesCursorImage *xfc;
+
+ if (! got_xfixes_cursor_notify) {
+ /* try again for XFixesCursorNotify event */
+ XEvent xev;
+ X_LOCK;
+ if (XCheckTypedEvent(dpy, xfixes_base_event_type +
+ XFixesCursorNotify, &xev)) {
+ got_xfixes_cursor_notify++;
+ }
+ X_UNLOCK;
+ }
+ if (! got_xfixes_cursor_notify) {
+ /* evidently no cursor change, just return last one */
+ if (last_index) {
+ return last_index;
+ } else {
+ return CURS_ARROW;
+ }
+ }
+ got_xfixes_cursor_notify = 0;
+
+ /* retrieve the cursor info + pixels from server: */
+ X_LOCK;
+ xfc = XFixesGetCursorImage(dpy);
+ X_UNLOCK;
+ if (! xfc) {
+ /* failure. */
+ return(which);
+ }
+
+ if (xfc->cursor_serial == last_cursor) {
+ /* same serial index: no change */
+ X_LOCK;
+ XFree(xfc);
+ X_UNLOCK;
+ if (last_index) {
+ return last_index;
+ } else {
+ return CURS_ARROW;
+ }
+ }
+
+ oldest = CURS_DYN_MIN;
+ oldtime = curs_times[oldest];
+ now = time(0);
+ for (i = CURS_DYN_MIN; i <= CURS_DYN_MAX; i++) {
+ if (curs_times[i] < oldtime) {
+ /* watch for oldest one to overwrite */
+ oldest = i;
+ oldtime = curs_times[i];
+ }
+ if (xfc->cursor_serial == curs_index[i]) {
+ /*
+ * got a hit with an existing cursor,
+ * use that one.
+ */
+ last_cursor = curs_index[i];
+ curs_times[i] = now;
+ last_index = i;
+ X_LOCK;
+ XFree(xfc);
+ X_UNLOCK;
+ return last_index;
+ }
+ }
+
+ if (screen) {
+ rfbUndrawCursor(screen);
+ }
+ /* we need to create the cursor and overwrite oldest */
+ use = oldest;
+ if (cursors[use]->rfb) {
+ /* clean up oldest if it exists */
+ if (cursors[use]->rfb->richSource) {
+ free(cursors[use]->rfb->richSource);
+ }
+ if (cursors[use]->rfb->source) {
+ free(cursors[use]->rfb->source);
+ }
+ if (cursors[use]->rfb->mask) {
+ free(cursors[use]->rfb->mask);
+ }
+ free(cursors[use]->rfb);
+ }
+
+ X_LOCK;
+ black = BlackPixel(dpy, scr);
+ white = WhitePixel(dpy, scr);
+ X_UNLOCK;
+
+ w = xfc->width;
+ h = xfc->height;
+ len = w * h;
+
+ /* for bitmap data */
+ bitmap = (char *)malloc(len+1);
+ bitmap[len] = '\0';
+
+ /* for rich cursor pixel data */
+ rich = (char *)calloc(Bpp*len, 1);
+
+ i = 0;
+ for (y = 0; y < h; y++) {
+ for (x = 0; x < w; x++) {
+ unsigned long r, g, b, a;
+ unsigned int ui;
+ char *p;
+
+ a = 0xff000000 & (*(xfc->pixels+i));
+ a = a >> 24; /* alpha channel */
+
+ if (a == 0) {
+ bitmap[i] = ' ';
+ i++;
+ continue;
+ } else {
+ bitmap[i] = 'x';
+ }
+
+ r = 0x00ff0000 & (*(xfc->pixels+i));
+ g = 0x0000ff00 & (*(xfc->pixels+i));
+ b = 0x000000ff & (*(xfc->pixels+i));
+ r = r >> 16; /* red */
+ g = g >> 8; /* green */
+ b = b >> 0; /* blue */
+
+ if (indexed_color) {
+ /*
+ * Choose black or white for
+ * PseudoColor case.
+ */
+ int value = (r+g+b)/3;
+ if (value > 127) {
+ ui = white;
+ } else {
+ ui = black;
+ }
+ } else {
+ /*
+ * Otherwise map the RGB data onto
+ * the framebuffer format:
+ */
+ r = (main_red_max * r)/255;
+ g = (main_green_max * g)/255;
+ b = (main_blue_max * b)/255;
+ ui = 0;
+ ui |= (r << main_red_shift);
+ ui |= (g << main_green_shift);
+ ui |= (b << main_blue_shift);
+ }
+
+ /* insert value into rich source: */
+ p = rich + Bpp*i;
+ if (Bpp == 1) {
+ *((unsigned char *)p)
+ = (unsigned char) ui;
+ } else if (Bpp == 2) {
+ *((unsigned short *)p)
+ = (unsigned short) ui;
+ } else if (Bpp == 4) {
+ *((unsigned int *)p)
+ = (unsigned int) ui;
+ }
+ i++;
+ }
+ }
+
+ /* create the cursor with the bitmap: */
+ c = rfbMakeXCursor(w, h, bitmap, bitmap);
+ free(bitmap);
+
+ /* set up the cursor parameters: */
+ c->xhot = xfc->xhot;
+ c->yhot = xfc->yhot;
+ c->cleanup = FALSE;
+ c->cleanupSource = FALSE;
+ c->cleanupMask = FALSE;
+ c->cleanupRichSource = FALSE;
+ c->richSource = rich;
+
+ /* place cursor into our collection */
+ cursors[use]->rfb = c;
+
+ /* update time and serial index: */
+ curs_times[use] = now;
+ curs_index[use] = xfc->cursor_serial;
+ last_index = use;
+ last_cursor = xfc->cursor_serial;
+
+ which = last_index;
+
+ X_LOCK;
+ XFree(xfc);
+ X_UNLOCK;
+#endif
+ }
+ return(which);
+}
+
+int known_cursors_mode(char *s) {
+/*
+ * default: see initialize_cursors_mode() for default behavior.
+ * arrow: unchanging white arrow.
+ * Xn*: show X on root background. Optional n sets treedepth.
+ * some: do the heuristics for root, wm, term detection.
+ * most: if display have overlay or xfixes, show all cursors,
+ * otherwise do the same as "some"
+ * none: show no cursor.
+ */
+ if (strcmp(s, "default") && strcmp(s, "arrow") && *s != 'X' &&
+ strcmp(s, "some") && strcmp(s, "most") && strcmp(s, "none")) {
+ return 0;
+ } else {
+ return 1;
+ }
+}
+
+void initialize_cursors_mode(void) {
+ char *s = multiple_cursors_mode;
+ if (!s || !known_cursors_mode(s)) {
+ rfbLog("unknown cursors mode: %s\n", s);
+ rfbLog("resetting cursors mode to \"default\"\n");
+ if (multiple_cursors_mode) free(multiple_cursors_mode);
+ multiple_cursors_mode = strdup("default");
+ s = multiple_cursors_mode;
+ }
+ if (!strcmp(s, "none")) {
+ show_cursor = 0;
+ } else {
+ /* we do NOT set show_cursor = 1, let the caller do that */
+ }
+
+ show_multiple_cursors = 0;
+ if (show_cursor) {
+ if (!strcmp(s, "default")) {
+ if(multiple_cursors_mode) free(multiple_cursors_mode);
+ multiple_cursors_mode = strdup("X");
+ s = multiple_cursors_mode;
+ }
+ if (*s == 'X' || !strcmp(s, "some") || !strcmp(s, "most")) {
+ show_multiple_cursors = 1;
+ } else {
+ show_multiple_cursors = 0;
+ /* hmmm, some bug going back to arrow mode.. */
+ set_rfb_cursor(CURS_ARROW);
+ }
+ if (screen) {
+ set_cursor_was_changed(screen);
+ }
+ } else {
+ if (screen) {
+ screen->cursor = NULL; /* dangerous? */
+ set_cursor_was_changed(screen);
+ }
+ }
+}
+
int get_which_cursor(void) {
int which = CURS_ARROW;
@@ -4824,14 +8107,26 @@ int get_which_cursor(void) {
Window win;
XErrorHandler old_handler;
int mode = 0;
- char a = multiple_cursors_mode[0];
- if (a == 'd') {
+ if (drag_in_progress) {
+ return -1;
+ }
+
+ if (!strcmp(multiple_cursors_mode, "arrow")) {
+ /* should not happen... */
+ return CURS_ARROW;
+ } else if (!strcmp(multiple_cursors_mode, "default")) {
mode = 0;
- } else if (a == 'X') {
+ } else if (!strcmp(multiple_cursors_mode, "X")) {
mode = 1;
- } else {
+ } else if (!strcmp(multiple_cursors_mode, "some")) {
mode = 2;
+ } else if (!strcmp(multiple_cursors_mode, "most")) {
+ mode = 3;
+ }
+
+ if (mode == 3 && xfixes_present && use_xfixes) {
+ return get_xfixes_cursor(0);
}
if (depth_cutoff < 0) {
@@ -4852,11 +8147,13 @@ int get_which_cursor(void) {
tree_descend_cursor(&depth, &win, &winfo);
- if (depth <= depth_cutoff) {
+ if (depth <= depth_cutoff && !subwin) {
which = CURS_ROOT;
- } else if (mode >= 2) {
+
+ } else if (mode == 2 || mode == 3) {
int which0 = which;
+ /* apply crude heuristics to choose a cursor... */
if (win) {
int ratio = 10, x, y;
unsigned int w, h, bw, d;
@@ -4864,6 +8161,8 @@ int get_which_cursor(void) {
trapped_xerror = 0;
old_handler = XSetErrorHandler(trap_xerror);
+
+ /* "narrow" windows are WM */
if (XGetGeometry(dpy, win, &r, &x, &y, &w, &h,
&bw, &d)) {
if (w > ratio * h || h > ratio * w) {
@@ -4874,12 +8173,17 @@ int get_which_cursor(void) {
trapped_xerror = 0;
}
if (which == which0) {
+ /* the string "term" mean I-beam. */
lowercase(winfo.res_name);
lowercase(winfo.res_class);
if (strstr(winfo.res_name, "term")) {
which = CURS_TERM;
} else if (strstr(winfo.res_class, "term")) {
which = CURS_TERM;
+ } else if (strstr(winfo.res_name, "text")) {
+ which = CURS_TERM;
+ } else if (strstr(winfo.res_class, "text")) {
+ which = CURS_TERM;
}
}
}
@@ -4948,12 +8252,52 @@ void set_cursor_was_moved(rfbScreenInfoPtr s) {
rfbReleaseClientIterator(iter);
}
-void unset_cursor_shape_updates(rfbScreenInfoPtr s) {
+void restore_cursor_shape_updates(rfbScreenInfoPtr s) {
rfbClientIteratorPtr iter;
rfbClientPtr cl;
+ int count = 0;
iter = rfbGetClientIterator(s);
while( (cl = rfbClientIteratorNext(iter)) ) {
+ int changed = 0;
+ ClientData *cd = (ClientData *) cl->clientData;
+
+ if (cd->had_cursor_shape_updates) {
+ rfbLog("restoring enableCursorShapeUpdates for client"
+ " 0x%x\n", cl);
+ cl->enableCursorShapeUpdates = TRUE;
+ changed = 1;
+ }
+ if (cd->had_cursor_pos_updates) {
+ rfbLog("restoring enableCursorPosUpdates for client"
+ " 0x%x\n", cl);
+ cl->enableCursorPosUpdates = TRUE;
+ changed = 1;
+ }
+ if (changed) {
+ cl->cursorWasChanged = TRUE;
+ count++;
+ }
+ }
+ rfbReleaseClientIterator(iter);
+}
+
+void disable_cursor_shape_updates(rfbScreenInfoPtr s) {
+ rfbClientIteratorPtr iter;
+ rfbClientPtr cl;
+
+ iter = rfbGetClientIterator(s);
+ while( (cl = rfbClientIteratorNext(iter)) ) {
+ ClientData *cd;
+ cd = (ClientData *) cl->clientData;
+
+ if (cl->enableCursorShapeUpdates) {
+ cd->had_cursor_shape_updates = 1;
+ }
+ if (cl->enableCursorPosUpdates) {
+ cd->had_cursor_pos_updates = 1;
+ }
+
cl->enableCursorShapeUpdates = FALSE;
cl->enableCursorPosUpdates = FALSE;
cl->cursorWasChanged = FALSE;
@@ -4961,14 +8305,14 @@ void unset_cursor_shape_updates(rfbScreenInfoPtr s) {
rfbReleaseClientIterator(iter);
}
-int cursor_pos_updates_clients(rfbScreenInfoPtr s) {
+int cursor_shape_updates_clients(rfbScreenInfoPtr s) {
rfbClientIteratorPtr iter;
rfbClientPtr cl;
int count = 0;
iter = rfbGetClientIterator(s);
while( (cl = rfbClientIteratorNext(iter)) ) {
- if (cl->enableCursorPosUpdates) {
+ if (cl->enableCursorShapeUpdates) {
count++;
}
}
@@ -4976,14 +8320,14 @@ int cursor_pos_updates_clients(rfbScreenInfoPtr s) {
return count;
}
-int cursor_shape_updates_clients(rfbScreenInfoPtr s) {
+int cursor_pos_updates_clients(rfbScreenInfoPtr s) {
rfbClientIteratorPtr iter;
rfbClientPtr cl;
int count = 0;
iter = rfbGetClientIterator(s);
while( (cl = rfbClientIteratorNext(iter)) ) {
- if (cl->enableCursorShapeUpdates) {
+ if (cl->enableCursorPosUpdates) {
count++;
}
}
@@ -5074,7 +8418,7 @@ void cursor_position(int x, int y) {
}
void set_rfb_cursor(int which) {
- int workaround = 1; /* if rfbSetCursor does not mark modified */
+ int workaround = 2; /* if rfbSetCursor does not mark modified */
if (! show_cursor) {
return;
@@ -5101,12 +8445,21 @@ void set_rfb_cursor(int which) {
}
}
- rfbSetCursor(screen, cursors[which]->rfb, FALSE);
+ if (!cursors[which] || !cursors[which]->rfb) {
+ rfbLog("non-existent cursor: which=%d\n", which);
+ return;
+ } else {
+ rfbSetCursor(screen, cursors[which]->rfb, FALSE);
+ }
/* this is a 2nd workaround for rfbSetCursor() */
- if (screen->underCursorBuffer == NULL &&
- screen->underCursorBufferLen != 0) {
- screen->underCursorBufferLen = 0;
+ if (workaround > 1) {
+ if (screen->underCursorBuffer == NULL &&
+ screen->underCursorBufferLen != 0) {
+ LOCK(screen->cursorMutex);
+ screen->underCursorBufferLen = 0;
+ UNLOCK(screen->cursorMutex);
+ }
}
if (workaround) {
@@ -5114,8 +8467,15 @@ void set_rfb_cursor(int which) {
}
}
+void set_no_cursor(void) {
+ set_rfb_cursor(CURS_EMPTY);
+}
+
void set_cursor(int x, int y, int which) {
static int last = -1;
+ if (which < 0) {
+ which = last;
+ }
if (last < 0 || which != last) {
set_rfb_cursor(which);
}
@@ -5172,13 +8532,20 @@ void check_x11_pointer(void) {
* the clients and dynamically if -flashcmap is specified.
*/
#define NCOLOR 256
-void set_colormap(void) {
+void set_colormap(int reset) {
static int first = 1;
static XColor color[NCOLOR], prev[NCOLOR];
Colormap cmap;
Visual *vis;
int i, ncells, diffs = 0;
+ if (reset) {
+ first = 1;
+ if (screen->colourMap.data.shorts) {
+ free(screen->colourMap.data.shorts);
+ }
+ }
+
if (first) {
screen->colourMap.count = NCOLOR;
screen->serverFormat.trueColour = FALSE;
@@ -5300,8 +8667,13 @@ void set_visual(char *str) {
XVisualInfo vinfo;
char *p, *vstring = strdup(str);
- if (! quiet) {
- fprintf(stderr, "set_visual: %s\n", vstring);
+ visual_id = (VisualID) 0;
+ visual_depth = 0;
+
+ if (!strcmp(vstring, "ignore") || !strcmp(vstring, "default")
+ || !strcmp(vstring, "")) {
+ free(vstring);
+ return;
}
/* set visual depth */
@@ -5312,6 +8684,11 @@ void set_visual(char *str) {
} else {
vdepth = defdepth;
}
+ if (! quiet) {
+ fprintf(stderr, "\nVisual Info:\n");
+ fprintf(stderr, " set_visual(\"%s\")\n", str);
+ fprintf(stderr, " visual_depth: %d\n", vdepth);
+ }
/* set visual id number */
if (strcmp(vstring, "StaticGray") == 0) {
@@ -5333,21 +8710,22 @@ void set_visual(char *str) {
visual_id = (VisualID) v_in;
return;
}
- fprintf(stderr, "bad -visual arg: %s\n", vstring);
- exit(1);
+ rfbLog("bad -visual arg: %s\n", vstring);
+ clean_up_exit(1);
}
visual_id = (VisualID) v_in;
free(vstring);
return;
}
+ if (! quiet) fprintf(stderr, " visual: %d\n", vis);
if (XMatchVisualInfo(dpy, scr, visual_depth, vis, &vinfo)) {
;
} else if (XMatchVisualInfo(dpy, scr, defdepth, vis, &vinfo)) {
;
} else {
- fprintf(stderr, "could not find visual: %s\n", vstring);
- exit(1);
+ rfbLog("could not find visual: %s\n", vstring);
+ clean_up_exit(1);
}
free(vstring);
@@ -5369,6 +8747,7 @@ void nofb_hook(rfbClientPtr cl) {
return;
}
rfbLog("framebuffer requested in -nofb mode by client %s\n", cl->host);
+ /* ignore xrandr */
fb = XGetImage_wr(dpy, window, 0, 0, dpy_x, dpy_y, AllPlanes, ZPixmap);
main_fb = fb->data;
rfb_fb = main_fb;
@@ -5377,19 +8756,437 @@ void nofb_hook(rfbClientPtr cl) {
screen->displayHook = NULL;
}
+void do_new_fb(int reset_mem) {
+ XImage *fb;
+ char *old_main = main_fb;
+ char *old_rfb = rfb_fb;
+
+ /* for threaded we really should lock libvncserver out. */
+ if (use_threads) {
+ rfbLog("warning: changing framebuffers while threaded may\n");
+ rfbLog(" not work, do not use -threads if problems arise.\n");
+ }
+
+ if (reset_mem) {
+ clean_shm(0);
+ free_tiles();
+ }
+
+ fb = initialize_xdisplay_fb();
+
+ initialize_screen(NULL, NULL, fb);
+
+ if (reset_mem) {
+ initialize_tiles();
+ initialize_blackouts_and_xinerama();
+ initialize_polling_images();
+ }
+
+ if (old_main != old_rfb && old_main) {
+ free(old_main);
+ }
+ if (old_rfb) {
+ free(old_rfb);
+ }
+ fb0 = fb;
+}
+
+void remove_fake_fb(void) {
+ if (! screen) {
+ return;
+ }
+ rfbLog("removing fake fb: 0x%x\n", fake_fb);
+
+ do_new_fb(1);
+
+ /*
+ * fake_fb is freed in do_new_fb(), but we set to NULL here to
+ * indicate it is gone.
+ */
+ fake_fb = NULL;
+}
+
+void install_fake_fb(int w, int h, int bpp) {
+ int bpc;
+ if (! screen) {
+ return;
+ }
+ if (fake_fb) {
+ free(fake_fb);
+ }
+ fake_fb = (char *) calloc(w*h*bpp/8, 1);
+ if (! fake_fb) {
+ rfbLog("could not create fake fb: %dx%d %d\n", w, h, bpp);
+ return;
+ }
+ bpc = guess_bits_per_color(bpp);
+ rfbLog("installing fake fb: %dx%d %d\n", w, h, bpp);
+ rfbLog("rfbNewFramebuffer(0x%x, 0x%x, %d, %d, %d, %d, %d)\n",
+ screen, fake_fb, w, h, bpc, 1, bpp/8);
+
+ rfbNewFramebuffer(screen, fake_fb, w, h, bpc, 1, bpp/8);
+}
+
+void check_padded_fb(void) {
+ if (! fake_fb) {
+ return;
+ }
+ if (time(0) > pad_geometry_time+1 && all_clients_initialized()) {
+ remove_fake_fb();
+ }
+}
+
+void install_padded_fb(char *geom) {
+ int w, h;
+ int ok = 1;
+ if (! geom || *geom == '\0') {
+ ok = 0;
+ } else if (sscanf(geom, "%dx%d", &w, &h) != 2) {
+ ok = 0;
+ }
+ w = nabs(w);
+ h = nabs(h);
+
+ if (w < 5) w = 5;
+ if (h < 5) h = 5;
+
+ if (!ok) {
+ rfbLog("skipping invalid pad geometry: '%s'\n", NONUL(geom));
+ return;
+ }
+ install_fake_fb(w, h, bpp);
+ pad_geometry_time = time(0);
+}
+
/*
- * initialize the rfb framebuffer/screen
+ * initialize a fb for the X display
*/
-void initialize_screen(int *argc, char **argv, XImage *fb) {
- int have_masks = 0;
- int width = fb->width;
- int height = fb->height;
+XImage *initialize_xdisplay_fb(void) {
+ XImage *fb;
+ char *vis_str = visual_str;
+ int try = 0, subwin_tries = 3;
+ XErrorHandler old_handler;
+ int subwin_bs;
+
+ X_LOCK;
+ if (subwin && !valid_window((Window) subwin)) {
+ rfbLog("invalid sub-window: 0x%lx\n", subwin);
+ X_UNLOCK;
+ clean_up_exit(1);
+ }
- main_bytes_per_line = fb->bytes_per_line;
+ if (overlay) {
+ /*
+ * ideally we'd like to not have to cook up the
+ * visual variables but rather let it all come out
+ * of XReadScreen(), however there is no way to get
+ * a default visual out of it, so we pretend -visual
+ * TrueColor:NN was supplied with NN usually 24.
+ */
+ char str[32];
+ Window twin = subwin ? subwin : rootwin;
+ XImage *xi;
+
+ xi = xreadscreen(dpy, twin, 0, 0, 8, 8, False);
+ sprintf(str, "TrueColor:%d", xi->depth);
+ if (xi->depth != 24 && ! quiet) {
+ rfbLog("warning: overlay image has depth %d "
+ "instead of 24.\n", xi->depth);
+ }
+ XDestroyImage(xi);
+ if (visual_str != NULL && ! quiet) {
+ rfbLog("warning: replacing '-visual %s' by '%s' "
+ "for use with -overlay\n", visual_str, str);
+ }
+ vis_str = strdup(str);
+ }
+
+ if (vis_str != NULL) {
+ set_visual(vis_str);
+ if (vis_str != visual_str) {
+ free(vis_str);
+ }
+ }
+
+ /* set up parameters for subwin or non-subwin cases: */
+
+ if (! subwin) {
+ /* full screen */
+ window = rootwin;
+ dpy_x = DisplayWidth(dpy, scr);
+ dpy_y = DisplayHeight(dpy, scr);
+ off_x = 0;
+ off_y = 0;
+ /* this may be overridden via visual_id below */
+ default_visual = DefaultVisual(dpy, scr);
+ } else {
+ /* single window */
+ XWindowAttributes attr;
+
+ window = (Window) subwin;
+ if (! XGetWindowAttributes(dpy, window, &attr)) {
+ rfbLog("invalid window: 0x%lx\n", window);
+ X_UNLOCK;
+ clean_up_exit(1);
+ }
+ dpy_x = attr.width;
+ dpy_y = attr.height;
+
+ subwin_bs = attr.backing_store;
+
+ /* this may be overridden via visual_id below */
+ default_visual = attr.visual;
+
+ X_UNLOCK;
+ set_offset();
+ X_LOCK;
+ }
+
+ /* initialize depth to reasonable value, visual_id may override */
+ depth = DefaultDepth(dpy, scr);
+
+ if (visual_id) {
+ int n;
+ XVisualInfo vinfo_tmpl, *vinfo;
+
+ /*
+ * we are in here from -visual or -overlay options
+ * visual_id and visual_depth were set in set_visual().
+ */
+
+ vinfo_tmpl.visualid = visual_id;
+ vinfo = XGetVisualInfo(dpy, VisualIDMask, &vinfo_tmpl, &n);
+ if (vinfo == NULL || n == 0) {
+ rfbLog("could not match visual_id: 0x%x\n",
+ (int) visual_id);
+ X_UNLOCK;
+ clean_up_exit(1);
+ }
+ default_visual = vinfo->visual;
+ depth = vinfo->depth;
+ if (visual_depth) {
+ /* force it from -visual MooColor:NN */
+ depth = visual_depth;
+ }
+ if (! quiet) {
+ fprintf(stderr, " initialize_xdisplay_fb()\n");
+ fprintf(stderr, " Visual*: 0x%x\n",
+ (int) vinfo->visual);
+ fprintf(stderr, " visualid: 0x%x\n",
+ (int) vinfo->visualid);
+ fprintf(stderr, " screen: %d\n", vinfo->screen);
+ fprintf(stderr, " depth: %d\n", vinfo->depth);
+ fprintf(stderr, " class: %d\n", vinfo->class);
+ fprintf(stderr, " red_mask: 0x%08lx %s\n",
+ vinfo->red_mask, bitprint(vinfo->red_mask, 32));
+ fprintf(stderr, " green_mask: 0x%08lx %s\n",
+ vinfo->green_mask, bitprint(vinfo->green_mask, 32));
+ fprintf(stderr, " blue_mask: 0x%08lx %s\n",
+ vinfo->blue_mask, bitprint(vinfo->blue_mask, 32));
+ fprintf(stderr, " cmap_size: %d\n",
+ vinfo->colormap_size);
+ fprintf(stderr, " bits b/rgb: %d\n",
+ vinfo->bits_per_rgb);
+ fprintf(stderr, "\n");
+ }
+ XFree(vinfo);
+ }
+
+ if (! quiet) {
+ rfbLog("default visual ID: 0x%x\n",
+ (int) XVisualIDFromVisual(default_visual));
+ }
+
+ again:
+ if (subwin) {
+ int shift = 0;
+ int subwin_x, subwin_y;
+ int disp_x = DisplayWidth(dpy, scr);
+ int disp_y = DisplayHeight(dpy, scr);
+ Window twin;
+ /* subwins can be a dicey if they are changing size... */
+ XTranslateCoordinates(dpy, window, rootwin, 0, 0, &subwin_x,
+ &subwin_y, &twin);
+ if (subwin_x + dpy_x > disp_x) {
+ shift = 1;
+ subwin_x = disp_x - dpy_x - 3;
+ }
+ if (subwin_y + dpy_y > disp_y) {
+ shift = 1;
+ subwin_y = disp_y - dpy_y - 3;
+ }
+ if (subwin_x < 0) {
+ shift = 1;
+ subwin_x = 1;
+ }
+ if (subwin_y < 0) {
+ shift = 1;
+ subwin_y = 1;
+ }
+
+ trapped_xerror = 0;
+ old_handler = XSetErrorHandler(trap_xerror);
+ if (shift) {
+ XMoveWindow(dpy, window, subwin_x, subwin_y);
+ }
+ XMapRaised(dpy, window);
+ XRaiseWindow(dpy, window);
+ XFlush(dpy);
+ }
+ try++;
+
+ if (nofb) {
+ /*
+ * For -nofb we do not allocate the framebuffer, so we
+ * can save a few MB of memory.
+ */
+ fb = XCreateImage_wr(dpy, default_visual, depth, ZPixmap,
+ 0, NULL, dpy_x, dpy_y, BitmapPad(dpy), 0);
+
+ } else if (visual_id) {
+ /*
+ * we need to call XCreateImage to supply the visual
+ */
+ fb = XCreateImage_wr(dpy, default_visual, depth, ZPixmap,
+ 0, NULL, dpy_x, dpy_y, BitmapPad(dpy), 0);
+ fb->data = (char *) malloc(fb->bytes_per_line * fb->height);
- main_red_mask = fb->red_mask;
- main_green_mask = fb->green_mask;
- main_blue_mask = fb->blue_mask;
+ } else {
+ fb = XGetImage_wr(dpy, window, 0, 0, dpy_x, dpy_y, AllPlanes,
+ ZPixmap);
+ if (! quiet) {
+ rfbLog("Read initial data from X display into"
+ " framebuffer.\n");
+ }
+ }
+
+ if (subwin) {
+ XSetErrorHandler(old_handler);
+ if (trapped_xerror) {
+ rfbLog("trapped GetImage at SUBWIN creation.\n");
+ if (try < subwin_tries) {
+ usleep(250 * 1000);
+ if (!get_window_size(window, &dpy_x, &dpy_y)) {
+ rfbLog("could not get size of subwin "
+ "0x%lx\n", subwin);
+ X_UNLOCK;
+ clean_up_exit(1);
+ }
+ goto again;
+ }
+ }
+ trapped_xerror = 0;
+
+ } else if (! fb && try == 1) {
+ /* try once more */
+ usleep(250 * 1000);
+ goto again;
+ }
+ X_UNLOCK;
+
+ if (fb->bits_per_pixel == 24 && ! quiet) {
+ rfbLog("warning: 24 bpp may have poor performance.\n");
+ }
+ return fb;
+}
+
+void parse_scale_string(char *str) {
+ int m, n;
+ char *p;
+ double f;
+
+ scale_fac = 1.0;
+ scaling = 0;
+ scaling_noblend = 0;
+ scaling_nomult4 = 0;
+ scaling_pad = 0;
+ scaling_interpolate = 0;
+ scaled_x = 0, scaled_y = 0;
+ scale_numer = 0, scale_denom = 0;
+
+ if (str == NULL || str[0] == '\0') {
+ return;
+ }
+
+ if ( (p = strchr(str, ':')) != NULL) {
+ /* options */
+ if (strstr(p+1, "nb") != NULL) {
+ scaling_noblend = 1;
+ }
+ if (strstr(p+1, "n4") != NULL) {
+ scaling_nomult4 = 1;
+ }
+ if (strstr(p+1, "in") != NULL) {
+ scaling_interpolate = 1;
+ }
+ if (strstr(p+1, "pad") != NULL) {
+ scaling_pad = 1;
+ }
+ *p = '\0';
+ }
+ if (strchr(str, '.') != NULL) {
+ double test, diff, eps = 1.0e-7;
+ if (sscanf(str, "%lf", &f) != 1) {
+ rfbLog("bad -scale arg: %s\n", str);
+ clean_up_exit(1);
+ }
+ scale_fac = (double) f;
+ /* look for common fractions from small ints: */
+ for (n=2; n<=10; n++) {
+ for (m=1; m<n; m++) {
+ test = ((double) m)/ n;
+ diff = scale_fac - test;
+ if (-eps < diff && diff < eps) {
+ scale_numer = m;
+ scale_denom = n;
+ break;
+
+ }
+ }
+ if (scale_denom) {
+ break;
+ }
+ }
+ if (scale_fac < 0.01) {
+ rfbLog("-scale factor too small: %f\n", scale_fac);
+ clean_up_exit(1);
+ }
+ } else {
+ if (sscanf(str, "%d/%d", &m, &n) != 2) {
+ if (sscanf(str, "%d", &m) != 1) {
+ rfbLog("bad -scale arg: %s\n", str);
+ clean_up_exit(1);
+ } else {
+ /* e.g. -scale 1 or -scale 2 */
+ n = 1;
+ }
+ }
+ if (n <= 0 || m <=0) {
+ rfbLog("bad -scale arg: %s\n", str);
+ clean_up_exit(1);
+ }
+ scale_fac = ((double) m)/ n;
+ if (scale_fac < 0.01) {
+ rfbLog("-scale factor too small: %f\n", scale_fac);
+ clean_up_exit(1);
+ }
+ scale_numer = m;
+ scale_denom = n;
+ }
+ if (scale_fac == 1.0) {
+ if (! quiet) {
+ rfbLog("scaling disabled for factor %f\n", scale_fac);
+ }
+ } else {
+ scaling = 1;
+ }
+}
+
+void setup_scaling(int *width_in, int *height_in) {
+ int width = *width_in;
+ int height = *height_in;
+
+ parse_scale_string(scale_str);
if (scaling) {
double eps = 0.000001;
@@ -5439,20 +9236,60 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
}
scaled_x = width;
scaled_y = height;
- rfb_bytes_per_line = (main_bytes_per_line / fb->width) * width;
+
+ *width_in = width;
+ *height_in = height;
+ }
+}
+
+/*
+ * initialize the rfb framebuffer/screen
+ */
+void initialize_screen(int *argc, char **argv, XImage *fb) {
+ int have_masks = 0;
+ int width = fb->width;
+ int height = fb->height;
+ int create_screen = screen ? 0 : 1;
+ int bits_per_color;
+
+ main_bytes_per_line = fb->bytes_per_line;
+
+ setup_scaling(&width, &height);
+
+ if (scaling) {
rfbLog("scaling screen: %dx%d -> %dx%d scale_fac=%.5f\n",
fb->width, fb->height, scaled_x, scaled_y, scale_fac);
+
+ rfb_bytes_per_line = (main_bytes_per_line / fb->width) * width;
} else {
rfb_bytes_per_line = main_bytes_per_line;
}
- screen = rfbGetScreen(argc, argv, width, height, fb->bits_per_pixel,
- 8, fb->bits_per_pixel/8);
+ /*
+ * These are just hints wrt pixel format just to let
+ * rfbGetScreen/rfbNewFramebuffer proceed with reasonable
+ * defaults. We manually set them in painful detail
+ * below.
+ */
+ bits_per_color = guess_bits_per_color(fb->bits_per_pixel);
- if (! quiet) {
- fprintf(stderr, "\n");
- }
+ /* n.b. samplesPerPixel (set = 1 here) seems to be unused. */
+ if (create_screen) {
+ screen = rfbGetScreen(argc, argv, width, height,
+ bits_per_color, 1, (int) fb->bits_per_pixel/8);
+ } else {
+ /* set set frameBuffer member below. */
+ rfbLog("rfbNewFramebuffer(0x%x, 0x%x, %d, %d, %d, %d, %d)\n",
+ screen, NULL, width, height,
+ bits_per_color, 1, fb->bits_per_pixel/8);
+ /* these are probably overwritten, but just to be safe: */
+ screen->bitsPerPixel = fb->bits_per_pixel;
+ screen->depth = fb->depth;
+
+ rfbNewFramebuffer(screen, NULL, width, height,
+ bits_per_color, 1, (int) fb->bits_per_pixel/8);
+ }
if (! screen) {
int i;
rfbLog("\n");
@@ -5468,11 +9305,11 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
* the x11vnc.c file by itself and plop it down into their libvncserver tree.
* Remove at some point. BTW, this assumes no usage of earlier "0.7pre".
*/
-#if NON_CVS && defined(LIBVNCSERVER_VERSION)
+#if OLD_TREE && defined(LIBVNCSERVER_VERSION)
if (strcmp(LIBVNCSERVER_VERSION, "0.6"))
#endif
{
- if (*argc != 1) {
+ if (create_screen && *argc != 1) {
int i;
rfbLog("*** unrecognized option(s) ***\n");
for (i=1; i< *argc; i++) {
@@ -5491,23 +9328,45 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
rfbLog("renamed: -mouseX, use -cursor X\n");
rfbLog("renamed: -X, use -cursor X\n");
rfbLog("renamed: -nomouse, use -nocursor\n");
+ rfbLog("renamed: -old_pointer, use -pointer_mode 1\n");
clean_up_exit(1);
}
}
+ /* set up format from scratch: */
screen->paddedWidthInBytes = rfb_bytes_per_line;
screen->serverFormat.bitsPerPixel = fb->bits_per_pixel;
screen->serverFormat.depth = fb->depth;
- screen->serverFormat.trueColour = (uint8_t) TRUE;
+ screen->serverFormat.trueColour = TRUE;
+
+ screen->serverFormat.redShift = 0;
+ screen->serverFormat.greenShift = 0;
+ screen->serverFormat.blueShift = 0;
+ screen->serverFormat.redMax = 0;
+ screen->serverFormat.greenMax = 0;
+ screen->serverFormat.blueMax = 0;
+
+ /* these main_* formats are use elsewhere by us. */
+ main_red_shift = 0;
+ main_green_shift = 0;
+ main_blue_shift = 0;
+ main_red_max = 0;
+ main_green_max = 0;
+ main_blue_max = 0;
+ main_red_mask = fb->red_mask;
+ main_green_mask = fb->green_mask;
+ main_blue_mask = fb->blue_mask;
+
+
have_masks = ((fb->red_mask|fb->green_mask|fb->blue_mask) != 0);
if (force_indexed_color) {
have_masks = 0;
}
- if ( ! have_masks && screen->serverFormat.bitsPerPixel == 8
- && CellsOfScreen(ScreenOfDisplay(dpy,scr)) ) {
- /* indexed colour */
+ if (! have_masks && screen->serverFormat.bitsPerPixel == 8
+ && CellsOfScreen(ScreenOfDisplay(dpy, scr))) {
+ /* indexed color */
if (!quiet) {
rfbLog("X display %s is 8bpp indexed color\n",
DisplayString(dpy));
@@ -5522,53 +9381,115 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
rfbLog("\n");
}
}
- indexed_colour = 1;
- set_colormap();
+ screen->serverFormat.trueColour = FALSE;
+ indexed_color = 1;
+ set_colormap(1);
} else {
- /* general case ... */
+ /*
+ * general case, we call it truecolor, but could be direct
+ * color, static color, etc....
+ */
if (! quiet) {
rfbLog("X display %s is %dbpp depth=%d true "
"color\n", DisplayString(dpy), fb->bits_per_pixel,
fb->depth);
}
+ indexed_color = 0;
+
/* convert masks to bit shifts and max # colors */
- screen->serverFormat.redShift = 0;
- if ( fb->red_mask ) {
- while ( ! (fb->red_mask
- & (1 << screen->serverFormat.redShift) ) ) {
+ if (fb->red_mask) {
+ while (! (fb->red_mask
+ & (1 << screen->serverFormat.redShift))) {
screen->serverFormat.redShift++;
}
}
- screen->serverFormat.greenShift = 0;
- if ( fb->green_mask ) {
- while ( ! (fb->green_mask
- & (1 << screen->serverFormat.greenShift) ) ) {
+ if (fb->green_mask) {
+ while (! (fb->green_mask
+ & (1 << screen->serverFormat.greenShift))) {
screen->serverFormat.greenShift++;
}
}
- screen->serverFormat.blueShift = 0;
- if ( fb->blue_mask ) {
- while ( ! (fb->blue_mask
- & (1 << screen->serverFormat.blueShift) ) ) {
+ if (fb->blue_mask) {
+ while (! (fb->blue_mask
+ & (1 << screen->serverFormat.blueShift))) {
screen->serverFormat.blueShift++;
}
}
screen->serverFormat.redMax
- = fb->red_mask >> screen->serverFormat.redShift;
+ = fb->red_mask >> screen->serverFormat.redShift;
screen->serverFormat.greenMax
= fb->green_mask >> screen->serverFormat.greenShift;
screen->serverFormat.blueMax
- = fb->blue_mask >> screen->serverFormat.blueShift;
+ = fb->blue_mask >> screen->serverFormat.blueShift;
- main_red_max = screen->serverFormat.redMax;
- main_green_max = screen->serverFormat.greenMax;
- main_blue_max = screen->serverFormat.blueMax;
+ main_red_max = screen->serverFormat.redMax;
+ main_green_max = screen->serverFormat.greenMax;
+ main_blue_max = screen->serverFormat.blueMax;
main_red_shift = screen->serverFormat.redShift;
main_green_shift = screen->serverFormat.greenShift;
main_blue_shift = screen->serverFormat.blueShift;
}
+
+ if (!quiet) {
+ fprintf(stderr, "\n");
+ fprintf(stderr, "FrameBuffer Info:\n");
+ fprintf(stderr, " width: %d\n", fb->width);
+ fprintf(stderr, " height: %d\n", fb->height);
+ fprintf(stderr, " scaled_width: %d\n", width);
+ fprintf(stderr, " scaled_height: %d\n", height);
+ fprintf(stderr, " indexed_color: %d\n", indexed_color);
+ fprintf(stderr, " bits_per_pixel: %d\n", fb->bits_per_pixel);
+ fprintf(stderr, " depth: %d\n", fb->depth);
+ fprintf(stderr, " red_mask: 0x%08lx %s\n", fb->red_mask,
+ bitprint(fb->red_mask, 32));
+ fprintf(stderr, " green_mask: 0x%08lx %s\n", fb->green_mask,
+ bitprint(fb->green_mask, 32));
+ fprintf(stderr, " blue_mask: 0x%08lx %s\n", fb->blue_mask,
+ bitprint(fb->blue_mask, 32));
+ fprintf(stderr, " red: max: %3d shift: %2d\n",
+ main_red_max, main_red_shift);
+ fprintf(stderr, " green: max: %3d shift: %2d\n",
+ main_green_max, main_green_shift);
+ fprintf(stderr, " blue: max: %3d shift: %2d\n",
+ main_blue_max, main_blue_shift);
+ fprintf(stderr, " mainfb_bytes_per_line: %d\n",
+ main_bytes_per_line);
+ fprintf(stderr, " rfb_fb_bytes_per_line: %d\n",
+ rfb_bytes_per_line);
+ switch(fb->format) {
+ case XYBitmap:
+ fprintf(stderr, " format: XYBitmap\n"); break;
+ case XYPixmap:
+ fprintf(stderr, " format: XYPixmap\n"); break;
+ case ZPixmap:
+ fprintf(stderr, " format: ZPixmap\n"); break;
+ default:
+ fprintf(stderr, " format: %d\n", fb->format); break;
+ }
+ switch(fb->byte_order) {
+ case LSBFirst:
+ fprintf(stderr, " byte_order: LSBFirst\n"); break;
+ case MSBFirst:
+ fprintf(stderr, " byte_order: MSBFirst\n"); break;
+ default:
+ fprintf(stderr, " byte_order: %d\n", fb->byte_order);
+ break;
+ }
+ fprintf(stderr, " bitmap_pad: %d\n", fb->bitmap_pad);
+ fprintf(stderr, " bitmap_unit: %d\n", fb->bitmap_unit);
+ switch(fb->bitmap_bit_order) {
+ case LSBFirst:
+ fprintf(stderr, " bitmap_bit_order: LSBFirst\n"); break;
+ case MSBFirst:
+ fprintf(stderr, " bitmap_bit_order: MSBFirst\n"); break;
+ default:
+ fprintf(stderr, " bitmap_bit_order: %d\n",
+ fb->bitmap_bit_order); break;
+ }
+ fprintf(stderr, "\n");
+ }
if (overlay && ! quiet) {
rfbLog("\n");
rfbLog("Overlay mode enabled: If you experience color\n");
@@ -5595,6 +9516,50 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
}
screen->frameBuffer = rfb_fb;
+ bpp = screen->serverFormat.bitsPerPixel;
+ depth = screen->serverFormat.depth;
+
+ setup_cursors();
+ if (! show_cursor) {
+ screen->cursor = NULL;
+ } else {
+ /* just set it to the arrow for now. */
+ if (0 && !quiet) rfbLog("calling set_rfb_cursor...\n");
+ set_rfb_cursor(CURS_ARROW);
+ if (0 && !quiet) rfbLog(" done.\n");
+ }
+
+ if (scaling) {
+ mark_rect_as_modified(0, 0, dpy_x, dpy_y, 0);
+ }
+
+ if (! create_screen) {
+ rfbClientIteratorPtr iter;
+ rfbClientPtr cl;
+
+ /*
+ * since bits_per_color above may have been approximate,
+ * try to reset the individual translation tables...
+ * we do not seem to need this with rfbGetScreen()...
+ */
+ if (!quiet) rfbLog("calling setTranslateFunction()...\n");
+ iter = rfbGetClientIterator(screen);
+ while ((cl = rfbClientIteratorNext(iter)) != NULL) {
+ screen->setTranslateFunction(cl);
+ }
+ rfbReleaseClientIterator(iter);
+ if (!quiet) rfbLog(" done.\n");
+ do_copy_screen = 1;
+
+ /* done for framebuffer change case */
+ return;
+ }
+
+ /*
+ * the rest is screen server initialization, etc, only needed
+ * at screen creation time.
+ */
+
/* called from inetd, we need to treat stdio as our socket */
if (inetd) {
int fd = dup(0);
@@ -5635,23 +9600,8 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
screen->setXCutText = xcut_receive;
}
- setup_cursors();
- if (! show_cursor) {
- screen->cursor = NULL;
- } else {
- screen->cursor = cursors[0]->rfb;
- }
-
rfbInitServer(screen);
-
- bpp = screen->serverFormat.bitsPerPixel;
- depth = screen->serverFormat.depth;
-
- if (scaling) {
- mark_rect_as_modified(0, 0, dpy_x, dpy_y, 0);
- }
-
if (viewonly_passwd) {
/* append the view only passwd after the normal passwd */
char **passwds_new = malloc(3*sizeof(char**));
@@ -5688,9 +9638,11 @@ int blackouts = 0;
* Take a comma separated list of geometries: WxH+X+Y and register them as
* rectangles to black out from the screen.
*/
-void initialize_blackout (char *list) {
+void initialize_blackouts(char *list) {
char *p, *blist = strdup(list);
- int x, y, X, Y, h, w;
+ int x, y, X, Y, h, w, t;
+
+ blackouts = 0;
p = strtok(blist, ", \t");
while (p) {
@@ -5698,10 +9650,14 @@ void initialize_blackout (char *list) {
if (sscanf(p, "%dx%d+%d+%d", &w, &h, &x, &y) == 4) {
;
} else if (sscanf(p, "%dx%d-%d+%d", &w, &h, &x, &y) == 4) {
+ w = nabs(w);
x = dpy_x - x - w;
} else if (sscanf(p, "%dx%d+%d-%d", &w, &h, &x, &y) == 4) {
+ h = nabs(h);
y = dpy_y - y - h;
} else if (sscanf(p, "%dx%d-%d-%d", &w, &h, &x, &y) == 4) {
+ w = nabs(w);
+ h = nabs(h);
x = dpy_x - x - w;
y = dpy_y - y - h;
} else {
@@ -5711,10 +9667,23 @@ void initialize_blackout (char *list) {
p = strtok(NULL, ", \t");
continue;
}
+ w = nabs(w);
+ h = nabs(h);
+ x = nfix(x, dpy_x);
+ y = nfix(y, dpy_y);
X = x + w;
Y = y + h;
+ X = nfix(X, dpy_x);
+ Y = nfix(Y, dpy_y);
+ if (x > X) {
+ t = X; X = x; x = t;
+ }
+ if (y > Y) {
+ t = Y; Y = y; y = t;
+ }
if (x < 0 || x > dpy_x || y < 0 || y > dpy_y ||
- X < 0 || X > dpy_x || Y < 0 || Y > dpy_y) {
+ X < 0 || X > dpy_x || Y < 0 || Y > dpy_y ||
+ x == X || y == Y) {
rfbLog("skipping invalid blackout geometry: %s x="
"%d-%d,y=%d-%d,w=%d,h=%d\n", p, x, X, y, Y, w, h);
} else {
@@ -5748,9 +9717,13 @@ void initialize_blackout (char *list) {
*/
void blackout_tiles(void) {
int tx, ty;
+ int debug_bo = 0;
if (! blackouts) {
return;
}
+ if (getenv("DEBUG_BLACKOUT") != NULL) {
+ debug_bo = 1;
+ }
/*
* to simplify things drop down to single copy mode, no vcr, etc...
@@ -5836,8 +9809,18 @@ void blackout_tiles(void) {
if (rect.x1 == x1 && rect.y1 == y1 &&
rect.x2 == x2 && rect.y2 == y2) {
tile_blackout[n].cover = 2;
+ if (debug_bo) {
+ fprintf(stderr, "full: %d=%d,%d"
+ " (%d-%d) (%d-%d)\n",
+ n, tx, ty, x1, x2, y1, y2);
+ }
} else {
tile_blackout[n].cover = 1;
+ if (debug_bo) {
+ fprintf(stderr, "part: %d=%d,%d"
+ " (%d-%d) (%d-%d)\n",
+ n, tx, ty, x1, x2, y1, y2);
+ }
}
if (++cnt >= BO_MAX) {
@@ -5852,6 +9835,10 @@ void blackout_tiles(void) {
sraRgnDestroy(tile_reg);
tile_blackout[n].count = cnt;
+ if (debug_bo && cnt > 1) {
+ rfbLog("warning: multiple region overlaps[%d] "
+ "for tile %d=%d,%d.\n", cnt, n, tx, ty);
+ }
}
}
}
@@ -5872,14 +9859,17 @@ void initialize_xinerama (void) {
if (! XineramaQueryExtension(dpy, &ev, &er)) {
rfbLog("Xinerama: disabling: display does not support it.\n");
xinerama = 0;
+ xinerama_present = 0;
return;
}
if (! XineramaIsActive(dpy)) {
/* n.b. change to XineramaActive(dpy, window) someday */
rfbLog("Xinerama: disabling: not active on display.\n");
xinerama = 0;
+ xinerama_present = 0;
return;
- }
+ }
+ xinerama_present = 1;
/* n.b. change to XineramaGetData() someday */
xineramas = XineramaQueryScreens(dpy, &n);
@@ -5946,13 +9936,51 @@ void initialize_xinerama (void) {
sprintf(tstr, "%dx%d+%d+%d,", w, h, x, y);
strcat(bstr, tstr);
}
- initialize_blackout(bstr);
+ initialize_blackouts(bstr);
free(bstr);
free(tstr);
#endif
}
+void initialize_blackouts_and_xinerama(void) {
+ if (blackout_string != NULL) {
+ initialize_blackouts(blackout_string);
+ }
+ if (xinerama) {
+ initialize_xinerama();
+ }
+ if (blackouts) {
+ blackout_tiles();
+ /* schedule a copy_screen(), now is too early. */
+ do_copy_screen = 1;
+ }
+}
+
+void push_sleep(n) {
+ int i;
+ for (i=0; i<n; i++) {
+ rfbPE(screen, -1);
+ if (i != n-1 && defer_update) {
+ usleep(defer_update * 1000);
+ }
+ }
+}
+
+/*
+ * try to forcefully push a black screen to all connected clients
+ */
+void push_black_screen(int n) {
+ zero_fb(0, 0, dpy_x, dpy_y);
+ mark_rect_as_modified(0, 0, dpy_x, dpy_y, 1);
+ push_sleep(n);
+}
+
+void refresh_screen(void) {
+ mark_rect_as_modified(0, 0, dpy_x, dpy_y, 1);
+ rfbPE(screen, -1);
+}
+
/*
* Fill the framebuffer with zero for the prescribed rectangle
*/
@@ -6038,6 +10066,37 @@ void initialize_tiles(void) {
hint_list = (hint_t *) malloc((size_t) (ntiles * sizeof(hint_t)));
}
+void free_tiles(void) {
+ if (tile_has_diff) {
+ free(tile_has_diff);
+ tile_has_diff = NULL;
+ }
+ if (tile_tried) {
+ free(tile_tried);
+ tile_tried = NULL;
+ }
+ if (tile_blackout) {
+ free(tile_blackout);
+ tile_blackout = NULL;
+ }
+ if (tile_region) {
+ free(tile_region);
+ tile_region = NULL;
+ }
+ if (tile_row) {
+ free(tile_row);
+ tile_row = NULL;
+ }
+ if (tile_row_shm) {
+ free(tile_row_shm);
+ tile_row_shm = NULL;
+ }
+ if (hint_list) {
+ free(hint_list);
+ hint_list = NULL;
+ }
+}
+
/*
* silly function to factor dpy_y until fullscreen shm is not bigger than max.
* should always work unless dpy_y is a large prime or something... under
@@ -6191,38 +10250,32 @@ static int shm_create(XShmSegmentInfo *shm, XImage **ximg_ptr, int w, int h,
}
void shm_delete(XShmSegmentInfo *shm) {
- if (! using_shm) {
- return;
- }
#ifdef LIBVNCSERVER_HAVE_XSHM
- if (shm->shmaddr != (char *) -1) {
+ if (shm != NULL && shm->shmaddr != (char *) -1) {
shmdt(shm->shmaddr);
}
- if (shm->shmid != -1) {
+ if (shm != NULL && shm->shmid != -1) {
shmctl(shm->shmid, IPC_RMID, 0);
}
#endif
}
void shm_clean(XShmSegmentInfo *shm, XImage *xim) {
- if (! using_shm) {
- return;
- }
-#ifdef LIBVNCSERVER_HAVE_XSHM
X_LOCK;
- if (shm->shmid != -1) {
+#ifdef LIBVNCSERVER_HAVE_XSHM
+ if (shm != NULL && shm->shmid != -1) {
XShmDetach_wr(dpy, shm);
}
+#endif
if (xim != NULL) {
XDestroyImage(xim);
}
X_UNLOCK;
shm_delete(shm);
-#endif
}
-void initialize_shm(void) {
+void initialize_polling_images(void) {
int i;
/* set all shm areas to "none" before trying to create any */
@@ -6243,6 +10296,13 @@ void initialize_shm(void) {
if (! shm_create(&scanline_shm, &scanline, dpy_x, 1, "scanline")) {
clean_up_exit(1);
}
+ if (0 && !quiet) {
+ if (using_shm) {
+ rfbLog("created \"scanline\" shm polling image.\n");
+ } else {
+ rfbLog("created \"scanline\" polling image.\n");
+ }
+ }
/*
* the fullscreen (e.g. 1280x1024/fs_factor) shared memory area image:
@@ -6261,6 +10321,13 @@ void initialize_shm(void) {
dpy_y/fs_factor, "fullscreen")) {
clean_up_exit(1);
}
+ if (0 && !quiet) {
+ if (using_shm) {
+ rfbLog("created \"scanline\" shm polling image.\n");
+ } else {
+ rfbLog("created \"scanline\" polling image.\n");
+ }
+ }
}
/*
@@ -6269,6 +10336,7 @@ void initialize_shm(void) {
* and 40 for 1280x1024, etc.
*/
+ tile_shm_count = 0;
for (i=1; i<=ntiles_x; i++) {
if (! shm_create(&tile_row_shm[i], &tile_row[i], tile_x * i,
tile_y, "tile_row")) {
@@ -6284,14 +10352,23 @@ void initialize_shm(void) {
rfbLog("shm: shared memory usage, or run ipcrm(1)"
" to manually\n");
rfbLog("shm: delete unattached shm segments.\n");
- /* n.b.: "i" not "1", a kludge for cleanup */
- single_copytile = i;
+ single_copytile_count = i;
}
+ tile_shm_count++;
if (single_copytile && i >= 1) {
/* only need 1x1 tiles */
break;
}
}
+ if (!quiet) {
+ if (using_shm) {
+ rfbLog("created %d tile_row shm polling images.\n",
+ tile_shm_count);
+ } else {
+ rfbLog("created %d tile_row polling images.\n",
+ tile_shm_count);
+ }
+ }
}
/*
@@ -6895,6 +10972,23 @@ static void scale_and_mark_rect(int X1, int Y1, int X2, int Y2) {
void mark_rect_as_modified(int x1, int y1, int x2, int y2, int force) {
+ if (damage_time != 0) {
+ int debug = 0;
+ if (time(0) > damage_time + damage_delay) {
+ if (! quiet) {
+ rfbLog("damaging turned off.\n");
+ }
+ damage_time = 0;
+ damage_delay = 0;
+ } else {
+ if (debug) {
+ rfbLog("damaging viewer fb by not marking "
+ "rect: %d,%d,%d,%d\n", x1, y1, x2, y2);
+ }
+ return;
+ }
+ }
+
if (rfb_fb == main_fb || force) {
rfbMarkRectAsModified(screen, x1, y1, x2, y2);
} else if (scaling) {
@@ -6926,7 +11020,7 @@ void mark_hint(hint_t hint) {
static int *first_line = NULL, *last_line;
static unsigned short *left_diff, *right_diff;
-static void copy_tiles(int tx, int ty, int nt) {
+static int copy_tiles(int tx, int ty, int nt) {
int x, y, line;
int size_x, size_y, width1, width2;
int off, len, n, dw, dx, t;
@@ -6975,12 +11069,13 @@ static void copy_tiles(int tx, int ty, int nt) {
* n.b. we are int single copy_tile mode: nt=1
*/
tile_has_diff[n] = 0;
- return;
+ return(0);
}
X_LOCK;
+ XRANDR_SET_TRAP_RET(-1, "copy_tile-set");
/* read in the whole tile run at once: */
- if ( using_shm && size_x == tile_x * nt && size_y == tile_y ) {
+ if (using_shm && size_x == tile_x * nt && size_y == tile_y) {
/* general case: */
XShmGetImage_wr(dpy, window, tile_row[nt], x, y, AllPlanes);
} else {
@@ -6991,6 +11086,7 @@ static void copy_tiles(int tx, int ty, int nt) {
XGetSubImage_wr(dpy, window, x, y, size_x, size_y, AllPlanes,
ZPixmap, tile_row[nt], 0, 0);
}
+ XRANDR_CHK_TRAP_RET(-1, "copy_tile-chk");
X_UNLOCK;
if (blackouts && tile_blackout[n].cover == 1) {
@@ -7074,7 +11170,7 @@ static void copy_tiles(int tx, int ty, int nt) {
for (t=1; t <= nt; t++) {
tile_has_diff[n+(t-1)] = 0;
}
- return;
+ return(0);
} else {
/*
* at least one tile has a difference. make sure info
@@ -7224,6 +11320,7 @@ static void copy_tiles(int tx, int ty, int nt) {
tile_region[n+s].right_diff = right_diff[t];
}
+ return(1);
}
/*
@@ -7241,14 +11338,15 @@ static void copy_tiles(int tx, int ty, int nt) {
*/
static int copy_all_tiles(void) {
int x, y, n, m;
- int diffs = 0;
+ int diffs = 0, ct;
for (y=0; y < ntiles_y; y++) {
for (x=0; x < ntiles_x; x++) {
n = x + y * ntiles_x;
if (tile_has_diff[n]) {
- copy_tiles(x, y, 1);
+ ct = copy_tiles(x, y, 1);
+ if (ct < 0) return ct; /* fatal */
}
if (! tile_has_diff[n]) {
/*
@@ -7284,7 +11382,7 @@ static int copy_all_tiles(void) {
*/
static int copy_all_tile_runs(void) {
int x, y, n, m, i;
- int diffs = 0;
+ int diffs = 0, ct;
int in_run = 0, run = 0;
int ntave = 0, ntcnt = 0;
@@ -7301,7 +11399,8 @@ static int copy_all_tile_runs(void) {
run = 0;
continue;
}
- copy_tiles(x - run, y, run);
+ ct = copy_tiles(x - run, y, run);
+ if (ct < 0) return ct; /* fatal */
ntcnt++;
ntave += run;
@@ -7351,7 +11450,7 @@ static int copy_all_tile_runs(void) {
*/
static int copy_tiles_backward_pass(void) {
int x, y, n, m;
- int diffs = 0;
+ int diffs = 0, ct;
for (y = ntiles_y - 1; y >= 0; y--) {
for (x = ntiles_x - 1; x >= 0; x--) {
@@ -7366,7 +11465,8 @@ static int copy_tiles_backward_pass(void) {
if (y >= 1 && ! tile_has_diff[m] && tile_region[n].top_diff) {
if (! tile_tried[m]) {
tile_has_diff[m] = 2;
- copy_tiles(x, y-1, 1);
+ ct = copy_tiles(x, y-1, 1);
+ if (ct < 0) return ct; /* fatal */
}
}
@@ -7375,7 +11475,8 @@ static int copy_tiles_backward_pass(void) {
if (x >= 1 && ! tile_has_diff[m] && tile_region[n].left_diff) {
if (! tile_tried[m]) {
tile_has_diff[m] = 2;
- copy_tiles(x-1, y, 1);
+ ct = copy_tiles(x-1, y, 1);
+ if (ct < 0) return ct; /* fatal */
}
}
}
@@ -7388,8 +11489,8 @@ static int copy_tiles_backward_pass(void) {
return diffs;
}
-static void gap_try(int x, int y, int *run, int *saw, int along_x) {
- int n, m, i, xt, yt;
+static int gap_try(int x, int y, int *run, int *saw, int along_x) {
+ int n, m, i, xt, yt, ct;
n = x + y * ntiles_x;
@@ -7397,12 +11498,12 @@ static void gap_try(int x, int y, int *run, int *saw, int along_x) {
if (*saw) {
(*run)++; /* extend the gap run. */
}
- return;
+ return 0;
}
if (! *saw || *run == 0 || *run > gaps_fill) {
*run = 0; /* unacceptable run. */
*saw = 1;
- return;
+ return 0;
}
for (i=1; i <= *run; i++) { /* iterate thru the run. */
@@ -7419,10 +11520,12 @@ static void gap_try(int x, int y, int *run, int *saw, int along_x) {
continue;
}
- copy_tiles(xt, yt, 1);
+ ct = copy_tiles(xt, yt, 1);
+ if (ct < 0) return ct; /* fatal */
}
*run = 0;
*saw = 1;
+ return 1;
}
/*
@@ -7435,14 +11538,15 @@ static void gap_try(int x, int y, int *run, int *saw, int along_x) {
*/
static int fill_tile_gaps(void) {
int x, y, run, saw;
- int n, diffs = 0;
+ int n, diffs = 0, ct;
/* horizontal: */
for (y=0; y < ntiles_y; y++) {
run = 0;
saw = 0;
for (x=0; x < ntiles_x; x++) {
- gap_try(x, y, &run, &saw, 1);
+ ct = gap_try(x, y, &run, &saw, 1);
+ if (ct < 0) return ct; /* fatal */
}
}
@@ -7451,7 +11555,8 @@ static int fill_tile_gaps(void) {
run = 0;
saw = 0;
for (y=0; y < ntiles_y; y++) {
- gap_try(x, y, &run, &saw, 0);
+ ct = gap_try(x, y, &run, &saw, 0);
+ if (ct < 0) return ct; /* fatal */
}
}
@@ -7463,8 +11568,8 @@ static int fill_tile_gaps(void) {
return diffs;
}
-static void island_try(int x, int y, int u, int v, int *run) {
- int n, m;
+static int island_try(int x, int y, int u, int v, int *run) {
+ int n, m, ct;
n = x + y * ntiles_x;
m = u + v * ntiles_x;
@@ -7479,13 +11584,15 @@ static void island_try(int x, int y, int u, int v, int *run) {
/* found a discontinuity */
if (tile_tried[m]) {
- return;
+ return 0;
} else if (*run < grow_fill) {
- return;
+ return 0;
}
- copy_tiles(u, v, 1);
+ ct = copy_tiles(u, v, 1);
+ if (ct < 0) return ct; /* fatal */
}
+ return 1;
}
/*
@@ -7495,7 +11602,7 @@ static void island_try(int x, int y, int u, int v, int *run) {
*/
static int grow_islands(void) {
int x, y, n, run;
- int diffs = 0;
+ int diffs = 0, ct;
/*
* n.b. the way we scan here should keep an extension going,
@@ -7506,14 +11613,16 @@ static int grow_islands(void) {
for (y=0; y < ntiles_y; y++) {
run = 0;
for (x=0; x <= ntiles_x - 2; x++) {
- island_try(x, y, x+1, y, &run);
+ ct = island_try(x, y, x+1, y, &run);
+ if (ct < 0) return ct; /* fatal */
}
}
/* right to left: */
for (y=0; y < ntiles_y; y++) {
run = 0;
for (x = ntiles_x - 1; x >= 1; x--) {
- island_try(x, y, x-1, y, &run);
+ ct = island_try(x, y, x-1, y, &run);
+ if (ct < 0) return ct; /* fatal */
}
}
for (n=0; n < ntiles; n++) {
@@ -7540,11 +11649,15 @@ static void blackout_regions(void) {
* the info from the X server. Bandwidth to client and compression time
* are other issues... use -fs 1.0 to disable.
*/
-void copy_screen(void) {
+int copy_screen(void) {
int pixelsize = bpp >> 3;
char *fbp;
int i, y, block_size;
+ if (! fs_factor) {
+ return 0;
+ }
+
block_size = (dpy_x * (dpy_y/fs_factor) * pixelsize);
fbp = main_fb;
@@ -7554,6 +11667,7 @@ void copy_screen(void) {
/* screen may be too big for 1 shm area, so broken into fs_factor */
for (i=0; i < fs_factor; i++) {
+ XRANDR_SET_TRAP_RET(-1, "copy_screen-set");
if (using_shm) {
XShmGetImage_wr(dpy, window, fullscreen, 0, y,
AllPlanes);
@@ -7562,6 +11676,7 @@ void copy_screen(void) {
fullscreen->height, AllPlanes, ZPixmap, fullscreen,
0, 0);
}
+ XRANDR_CHK_TRAP_RET(-1, "copy_screen-chk");
memcpy(fbp, fullscreen->data, (size_t) block_size);
y += dpy_y / fs_factor;
@@ -7575,6 +11690,7 @@ void copy_screen(void) {
}
mark_rect_as_modified(0, 0, dpy_x, dpy_y, 0);
+ return 0;
}
@@ -7818,6 +11934,7 @@ static int scan_display(int ystart, int rescan) {
/* grab the horizontal scanline from the display: */
X_LOCK;
+ XRANDR_SET_TRAP_RET(-1, "scan_display-set");
if (using_shm) {
XShmGetImage_wr(dpy, window, scanline, 0, y, AllPlanes);
} else {
@@ -7825,6 +11942,7 @@ static int scan_display(int ystart, int rescan) {
scanline->height, AllPlanes, ZPixmap, scanline,
0, 0);
}
+ XRANDR_CHK_TRAP_RET(-1, "scan_display-chk");
X_UNLOCK;
/* for better memory i/o try the whole line at once */
@@ -7921,14 +12039,22 @@ int scan_for_updates(void) {
if (subwin) {
set_offset(); /* follow the subwindow */
}
- if (indexed_colour) { /* check for changed colormap */
- set_colormap();
+ if (indexed_color) { /* check for changed colormap */
+ set_colormap(0);
}
}
+#define SCAN_FATAL(x) \
+ if (x < 0) { \
+ scan_in_progress = 0; \
+ fb_copy_in_progress = 0; \
+ return 0; \
+ }
+
/* scan with the initial y to the jitter value from scanlines: */
scan_in_progress = 1;
tile_count = scan_display(scanlines[scan_count], 0);
+ SCAN_FATAL(tile_count);
nap_set(tile_count);
@@ -7951,11 +12077,13 @@ int scan_for_updates(void) {
cp = (NSCAN - scan_count) % NSCAN;
tile_count = scan_display(scanlines[cp], 1);
+ SCAN_FATAL(tile_count);
if (tile_count >= (1 + frac2) * tile_count_old) {
/* on a roll... do a 3rd scan */
cp = (NSCAN - scan_count + 7) % NSCAN;
tile_count = scan_display(scanlines[cp], 1);
+ SCAN_FATAL(tile_count);
}
}
scan_in_progress = 0;
@@ -7975,10 +12103,12 @@ int scan_for_updates(void) {
* Use -fs 1.0 to disable on slow links.
*/
if (fs_factor && tile_count > fs_frac * ntiles) {
+ int cs;
fb_copy_in_progress = 1;
- copy_screen();
+ cs = copy_screen();
fb_copy_in_progress = 0;
- if (use_threads && old_pointer != 1) {
+ SCAN_FATAL(cs);
+ if (use_threads && pointer_mode != 1) {
pointer(-1, 0, 0, NULL);
}
nap_check(tile_count);
@@ -7990,7 +12120,7 @@ int scan_for_updates(void) {
/* copy all tiles with differences from display to rfb framebuffer: */
fb_copy_in_progress = 1;
- if (single_copytile) {
+ if (single_copytile || tile_shm_count < ntiles_x) {
/*
* Old way, copy I/O one tile at a time.
*/
@@ -8003,25 +12133,29 @@ int scan_for_updates(void) {
*/
tile_diffs = copy_all_tile_runs();
}
+ SCAN_FATAL(tile_diffs);
/*
* This backward pass for upward and left tiles complements what
* was done in copy_all_tiles() for downward and right tiles.
*/
tile_diffs = copy_tiles_backward_pass();
+ SCAN_FATAL(tile_diffs);
/* Given enough tile diffs, try the islands: */
if (grow_fill && tile_diffs > 4) {
tile_diffs = grow_islands();
}
+ SCAN_FATAL(tile_diffs);
/* Given enough tile diffs, try the gaps: */
if (gaps_fill && tile_diffs > 4) {
tile_diffs = fill_tile_gaps();
}
+ SCAN_FATAL(tile_diffs);
fb_copy_in_progress = 0;
- if (use_threads && old_pointer != 1) {
+ if (use_threads && pointer_mode != 1) {
/*
* tell the pointer handler it can process any queued
* pointer events:
@@ -8054,6 +12188,220 @@ int scan_for_updates(void) {
return tile_diffs;
}
+/* -- gui.c -- */
+#if OLD_TREE
+char gui_code[] = "";
+#else
+#include "tkx11vnc.h"
+#endif
+
+void run_gui(char *gui_xdisplay, int connect_to_x11vnc) {
+ char *x11vnc_xdisplay = NULL;
+ char extra_path[] = ":/usr/local/bin:/usr/sfw/bin:/usr/X11R6/bin";
+ char cmd[100];
+ char *wish, *orig_path, *full_path, *tpath, *p;
+ int try_max = 9, sleep = 300;
+ FILE *pipe;
+
+
+ if (getenv("DISPLAY") != NULL) {
+ x11vnc_xdisplay = strdup(getenv("DISPLAY"));
+ }
+ if (use_dpy) {
+ x11vnc_xdisplay = strdup(use_dpy);
+ }
+ if (connect_to_x11vnc) {
+ int rc, i;
+ if (! client_connect_file) {
+ dpy = XOpenDisplay(x11vnc_xdisplay);
+ if (! dpy) {
+ fprintf(stderr, "could not open display: %s\n",
+ x11vnc_xdisplay);
+ exit(1);
+ }
+ scr = DefaultScreen(dpy);
+ rootwin = RootWindow(dpy, scr);
+ initialize_vnc_connect_prop();
+ }
+ for (i=0; i<try_max; i++) {
+ usleep(sleep*1000);
+ fprintf(stderr, "pinging %s ...\n",
+ NONUL(x11vnc_xdisplay));
+ rc = send_remote_cmd("ping", 1);
+ if (rc == 0) {
+ break;
+ }
+ }
+ if (rc == 0) {
+ fprintf(stderr, "success.\n");
+ set_env("X11VNC_XDISPLAY", x11vnc_xdisplay);
+ set_env("X11VNC_CONNECT", "1");
+ } else {
+ fprintf(stderr, "could not connect to: '%s', try again"
+ " in the gui.\n", x11vnc_xdisplay);
+ }
+ }
+
+ orig_path = getenv("PATH");
+ if (! orig_path) {
+ orig_path = strdup("/bin:/usr/bin:/usr/bin/X11");
+ }
+ full_path = (char *) malloc(strlen(orig_path)+strlen(extra_path)+1);
+ strcpy(full_path, orig_path);
+ strcat(full_path, extra_path);
+
+ tpath = strdup(full_path);
+ p = strtok(tpath, ":");
+
+ while (p) {
+ char *try;
+ struct stat sbuf;
+ char *wishes[] = {"wish", "wish8.3", "wish8.4"};
+ int nwishes = 3, i;
+
+ try = (char *)malloc(strlen(p) + 10);
+ for (i=0; i<nwishes; i++) {
+ sprintf(try, "%s/%s", p, wishes[i]);
+ if (stat(try, &sbuf)) {
+ wish = wishes[i];
+ break;
+ }
+ }
+ free(try);
+ if (wish) {
+ break;
+ }
+ p = strtok(NULL, ":");
+ }
+ free(tpath);
+ if (!wish) {
+ wish = strdup("wish");
+ }
+ set_env("PATH", full_path);
+ set_env("DISPLAY", gui_xdisplay);
+ set_env("X11VNC_PROG", program_name);
+ set_env("X11VNC_CMDLINE", program_cmdline);
+
+if (0) system("env");
+ sprintf(cmd, "%s -", wish);
+ pipe = popen(cmd, "w");
+ if (! pipe) {
+ fprintf(stderr, "could not run: %s\n", cmd);
+ perror("popen");
+ }
+ fprintf(pipe, "%s", gui_code);
+ pclose(pipe);
+ exit(0);
+}
+
+void do_gui(char *opts) {
+ char *s, *p;
+ char *old_xauth = NULL;
+ char *gui_xdisplay = NULL;
+ int start_x11vnc = 1;
+ int connect_to_x11vnc = 0;
+ Display *test_dpy;
+
+ if (*gui_code == '\0') {
+ rfbLog("gui not available in this program.\n");
+ clean_up_exit(1);
+ }
+ if (opts) {
+ s = strdup(opts);
+ } else {
+ s = strdup("");
+ }
+
+ if (use_dpy) {
+ /* worst case */
+ gui_xdisplay = strdup(use_dpy);
+
+ }
+ if (getenv("DISPLAY") != NULL) {
+ /* better */
+ gui_xdisplay = strdup(getenv("DISPLAY"));
+ }
+
+ p = strtok(s, ",");
+
+ while(p) {
+ if (*p == '\0') {
+ ;
+ } else if (strchr(p, ':') != NULL) {
+ /* best */
+ gui_xdisplay = strdup(p);
+ } else if (!strcmp(p, "wait")) {
+ start_x11vnc = 0;
+ connect_to_x11vnc = 0;
+ } else if (!strcmp(p, "conn") || !strcmp(p, "connect")) {
+ start_x11vnc = 0;
+ connect_to_x11vnc = 1;
+ } else {
+ fprintf(stderr, "unrecognized gui opt: %s\n", p);
+ }
+
+ p = strtok(NULL, ",");
+ }
+ free(s);
+ if (start_x11vnc) {
+ connect_to_x11vnc = 1;
+ }
+
+ if (! gui_xdisplay) {
+ fprintf(stderr, "error: cannot determine X DISPLAY for gui"
+ " to display on.\n");
+ exit(1);
+ }
+ test_dpy = XOpenDisplay(gui_xdisplay);
+ if (! test_dpy && auth_file) {
+ if (getenv("XAUTHORITY") != NULL) {
+ old_xauth = strdup(getenv("XAUTHORITY"));
+ }
+ set_env("XAUTHORITY", auth_file);
+ test_dpy = XOpenDisplay(gui_xdisplay);
+ }
+ if (! test_dpy) {
+ if (! old_xauth && getenv("XAUTHORITY") != NULL) {
+ old_xauth = strdup(getenv("XAUTHORITY"));
+ }
+ set_env("XAUTHORITY", "");
+ test_dpy = XOpenDisplay(gui_xdisplay);
+ }
+ if (! test_dpy) {
+ fprintf(stderr, "error: cannot connect to gui X DISPLAY: %s\n",
+ gui_xdisplay);
+ }
+ XCloseDisplay(test_dpy);
+
+ if (start_x11vnc) {
+#if defined(LIBVNCSERVER_HAVE_FORK)
+ /* fork into the background now */
+ int p;
+ if ((p = fork()) > 0) {
+ ; /* parent */
+ } else if (p == -1) {
+ fprintf(stderr, "could not fork\n");
+ perror("fork");
+ clean_up_exit(1);
+ } else {
+ run_gui(gui_xdisplay, connect_to_x11vnc);
+ exit(1);
+ }
+#else
+ fprintf(stderr, "system does not support fork: start "
+ "x11vnc in the gui.\n");
+ start_x11vnc = 0;
+#endif
+ }
+ if (!start_x11vnc) {
+ run_gui(gui_xdisplay, connect_to_x11vnc);
+ exit(1);
+ }
+ if (old_xauth) {
+ set_env("XAUTHORITY", old_xauth);
+ }
+}
+
/* -- x11vnc.c -- */
/*
* main routine for the x11vnc program
@@ -8077,43 +12425,143 @@ static int defer_update_nofb = 6; /* defer a shorter time under -nofb */
*
* return of 1 means watch_loop should short-circuit and reloop,
* return of 0 means watch_loop should proceed to scan_for_updates().
- * (this is for old_pointer == 1 mode, the others do it all internally,
+ * (this is for pointer_mode == 1 mode, the others do it all internally,
* cnt is also only for that mode).
*/
-static int check_user_input_old(double dt, int *cnt);
-static int check_user_input(double dt, int tile_diffs, int *cnt) {
+static void check_user_input2(double dt) {
- if (old_pointer == 1) {
- /* every n-th drops thru to scan */
- if ((got_user_input || ui_skip < 0) && *cnt % ui_skip != 0) {
- *cnt++;
+
+ if (got_pointer_input) {
+ int eaten = 0, miss = 0, max_eat = 50;
+ int g, g_in;
+ double spin = 0.0, tm = 0.0;
+ double quick_spin_fac = 0.40;
+ double grind_spin_time = 0.175;
+
+ dtime(&tm);
+ g = g_in = got_pointer_input;
+
+ /*
+ * Try for some "quick" pointer input processing.
+ *
+ * About as fast as we can, we try to process user input
+ * calling rfbProcessEvents or rfbCheckFds. We do this
+ * for a time on order of the last scan_for_updates() time,
+ * dt, but if we stop getting user input we break out.
+ * We will also break out if we have processed max_eat
+ * inputs.
+ *
+ * Note that rfbCheckFds() does not send any framebuffer
+ * updates, so is more what we want here, although it is
+ * likely they have all be sent already.
+ */
+ while (1) {
+ if (show_multiple_cursors) {
+ rfbPE(screen, 1000);
+ } else {
+ rfbCFD(screen, 1000);
+ }
XFlush(dpy);
- return 1; /* short circuit watch_loop */
- } else {
- return 0;
+
+ spin += dtime(&tm);
+
+ if (spin > quick_spin_fac * dt) {
+ /* get out if spin time comparable to last scan time */
+ break;
+ }
+ if (got_pointer_input > g) {
+ g = got_pointer_input;
+ if (eaten++ < max_eat) {
+ continue;
+ }
+ } else {
+ miss++;
+ }
+ if (miss > 1) { /* 1 means out on 2nd miss */
+ break;
+ }
}
- } else if (old_pointer == 2) {
- /* this older way is similar to the method below */
- return check_user_input_old(dt, cnt);
- }
- if (got_keyboard_input) {
- if (*cnt % ui_skip != 0) {
- *cnt++;
- return 1; /* short circuit watch_loop */
+
+ /*
+ * Probably grinding with a lot of fb I/O if dt is
+ * this large. (need to do this more elegantly)
+ *
+ * Current idea is to spin our wheels here *not* processing
+ * any fb I/O, but still processing the user input.
+ * This user input goes to the X display and changes it,
+ * but we don't poll it while we "rest" here for a time
+ * on order of dt, the previous scan_for_updates() time.
+ * We also break out if we miss enough user input.
+ */
+ if (dt > grind_spin_time) {
+ int i, ms, split = 30;
+ double shim;
+
+ /*
+ * Break up our pause into 'split' steps.
+ * We get at most one input per step.
+ */
+ shim = 0.75 * dt / split;
+
+ ms = (int) (1000 * shim);
+
+ /* cutoff how long the pause can be */
+ if (split * ms > 300) {
+ ms = 300 / split;
+ }
+
+ spin = 0.0;
+ tm = 0.0;
+ dtime(&tm);
+
+ g = got_pointer_input;
+ miss = 0;
+ for (i=0; i<split; i++) {
+ usleep(ms * 1000);
+ if (show_multiple_cursors) {
+ rfbPE(screen, 1000);
+ } else {
+ rfbCFD(screen, 1000);
+ }
+ spin += dtime(&tm);
+ if (got_pointer_input > g) {
+ XFlush(dpy);
+ miss = 0;
+ } else {
+ miss++;
+ }
+ g = got_pointer_input;
+ if (miss > 2) {
+ break;
+ }
+ if (1000 * spin > ms * split) {
+ break;
+ }
+ }
}
- /* otherwise continue with pointer input */
}
+}
+
+static void check_user_input3(double dt, int tile_diffs) {
if (got_pointer_input) {
int spun_out, missed_out, allowed_misses, g, g_in;
double spin, spin_max, tm, to, dtm, rpe_last;
static int rfb_wait_ms = 2;
- static double grind_spin_time = 0.30, dt_min = 0.15;
+ static double grind_spin_time = 0.30, dt_min = 0.075;
static double quick_spin_fac = 0.65, spin_max_fac = 2.0;
static double rpe_wait = 0.15;
int grinding, gcnt, ms, split = 200;
+ static int first = 1;
+ if (first) {
+ char *p = getenv("SPIN");
+ if (p) {
+ sscanf(p, "%lf,%lf,%lf", &grind_spin_time, &dt_min, &quick_spin_fac);
+ }
+ first = 0;
+ }
/*
@@ -8150,6 +12598,7 @@ static int check_user_input(double dt, int tile_diffs, int *cnt) {
rpe_last = to = tm; /* last time we did rfbPE() */
g = g_in = got_pointer_input;
+
while (1) {
int got_input = 0;
@@ -8161,6 +12610,10 @@ static int check_user_input(double dt, int tile_diffs, int *cnt) {
usleep(ms * 1000);
}
+ if (button_mask) {
+ drag_in_progress = 1;
+ }
+
if (show_multiple_cursors && tm > rpe_last + rpe_wait) {
rfbPE(screen, rfb_wait_ms * 1000);
rpe_last = tm;
@@ -8195,6 +12648,7 @@ static int check_user_input(double dt, int tile_diffs, int *cnt) {
wms = 1000 * (0.5 * (spin_max - spin));
} else if (button_mask) {
wms = 10;
+ } else {
}
if (wms) {
usleep(wms * 1000);
@@ -8219,7 +12673,7 @@ static int check_user_input(double dt, int tile_diffs, int *cnt) {
/* reset for second pass */
spun_out = 0;
missed_out = 0;
- allowed_misses = 5;
+ allowed_misses = 3;
g = got_pointer_input;
gcnt = 0;
} else if (spun_out && grinding) {
@@ -8231,129 +12685,39 @@ static int check_user_input(double dt, int tile_diffs, int *cnt) {
}
}
}
- return 0;
+ drag_in_progress = 0;
}
-/* this is the -old_pointer2 way */
-static int check_user_input_old(double dt, int *cnt) {
+static void check_user_input4(double dt, int tile_diffs) {
+ return;
+}
- if (got_keyboard_input) {
- if (*cnt % ui_skip != 0) {
+static int check_user_input(double dt, int tile_diffs, int *cnt) {
+ if (pointer_mode == 1) {
+ if ((got_user_input || ui_skip < 0) && *cnt % ui_skip != 0) {
+ /* every ui_skip-th drops thru to scan */
*cnt++;
+ XFlush(dpy);
return 1; /* short circuit watch_loop */
+ } else {
+ return 0;
}
- /* otherwise continue with pointer input */
}
-
- if (got_pointer_input) {
- int eaten = 0, miss = 0, max_eat = 50;
- int g, g_in;
- double spin = 0.0, tm = 0.0;
- double quick_spin_fac = 0.40;
- double grind_spin_time = 0.175;
-
- dtime(&tm);
- g = g_in = got_pointer_input;
-
- /*
- * Try for some "quick" pointer input processing.
- *
- * About as fast as we can, we try to process user input
- * calling rfbProcessEvents or rfbCheckFds. We do this
- * for a time on order of the last scan_for_updates() time,
- * dt, but if we stop getting user input we break out.
- * We will also break out if we have processed max_eat
- * inputs.
- *
- * Note that rfbCheckFds() does not send any framebuffer
- * updates, so is more what we want here, although it is
- * likely they have all be sent already.
- */
- while (1) {
- if (show_multiple_cursors) {
- rfbPE(screen, 1000);
- } else {
- rfbCFD(screen, 1000);
- }
- XFlush(dpy);
-
- spin += dtime(&tm);
-
- if (spin > quick_spin_fac * dt) {
- /* get out if spin time comparable to last scan time */
- break;
- }
- if (got_pointer_input > g) {
- g = got_pointer_input;
- if (eaten++ < max_eat) {
- continue;
- }
- } else {
- miss++;
- }
- if (miss > 1) { /* 1 means out on 2nd miss */
- break;
- }
- }
-
-
- /*
- * Probably grinding with a lot of fb I/O if dt is
- * this large. (need to do this more elegantly)
- *
- * Current idea is to spin our wheels here *not* processing
- * any fb I/O, but still processing the user input.
- * This user input goes to the X display and changes it,
- * but we don't poll it while we "rest" here for a time
- * on order of dt, the previous scan_for_updates() time.
- * We also break out if we miss enough user input.
- */
- if (dt > grind_spin_time) {
- int i, ms, split = 30;
- double shim;
-
- /*
- * Break up our pause into 'split' steps.
- * We get at most one input per step.
- */
- shim = 0.75 * dt / split;
-
- ms = (int) (1000 * shim);
-
- /* cutoff how long the pause can be */
- if (split * ms > 300) {
- ms = 300 / split;
- }
-
- spin = 0.0;
- tm = 0.0;
- dtime(&tm);
-
- g = got_pointer_input;
- miss = 0;
- for (i=0; i<split; i++) {
- usleep(ms * 1000);
- if (show_multiple_cursors) {
- rfbPE(screen, 1000);
- } else {
- rfbCFD(screen, 1000);
- }
- spin += dtime(&tm);
- if (got_pointer_input > g) {
- XFlush(dpy);
- miss = 0;
- } else {
- miss++;
- }
- g = got_pointer_input;
- if (miss > 2) {
- break;
- }
- if (1000 * spin > ms * split) {
- break;
- }
+ if (pointer_mode >= 2 && pointer_mode <= 4) {
+ if (got_keyboard_input) {
+ if (*cnt % ui_skip != 0) {
+ *cnt++;
+ return 1; /* short circuit watch_loop */
}
}
+ /* otherwise continue with pointer input */
+ }
+ if (pointer_mode == 2) {
+ check_user_input2(dt);
+ } else if (pointer_mode == 3) {
+ check_user_input3(dt, tile_diffs);
+ } else if (pointer_mode == 4) {
+ check_user_input4(dt, tile_diffs);
}
return 0;
}
@@ -8418,7 +12782,7 @@ static void watch_loop(void) {
rfbPE(screen, -1);
if (! cursor_shape_updates) {
/* undo any cursor shape requests */
- unset_cursor_shape_updates(screen);
+ disable_cursor_shape_updates(screen);
}
if (check_user_input(dt, tile_diffs, &cnt)) {
/* true means loop back for more input */
@@ -8430,8 +12794,14 @@ static void watch_loop(void) {
clean_up_exit(0);
}
+ if (do_copy_screen) {
+ do_copy_screen = 0;
+ copy_screen();
+ }
+
check_xevents();
check_connect_inputs();
+ check_padded_fb();
if (! screen->clientHead) { /* waiting for a client */
usleep(200 * 1000);
@@ -8497,15 +12867,17 @@ static void print_help(void) {
"\n"
" vncviewer far-host:0\n"
"\n"
-"Once x11vnc establishes connections with the X11 server and starts\n"
-"listening as a VNC server it will print out a string: PORT=XXXX where\n"
-"XXXX is typically 5900 (the default VNC port). One would next run something\n"
-"like this on the local machine: \"vncviewer host:N\" where N is XXXX - 5900,\n"
-"i.e. usually \"vncviewer host:0\"\n"
+"Once x11vnc establishes connections with the X11 server and starts listening\n"
+"as a VNC server it will print out a string: PORT=XXXX where XXXX is typically\n"
+"5900 (the default VNC server port). One would next run something like\n"
+"this on the local machine: \"vncviewer hostname:N\" where \"hostname\" is\n"
+"the name of the machine running x11vnc and N is XXXX - 5900, i.e. usually\n"
+"\"vncviewer hostname:0\".\n"
"\n"
-"By default x11vnc will not allow the screen to be shared and it will\n"
-"exit as soon as a client disconnects. See -shared and -forever below\n"
-"to override these protections.\n"
+"By default x11vnc will not allow the screen to be shared and it will exit\n"
+"as soon as a client disconnects. See -shared and -forever below to override\n"
+"these protections. See the FAQ on how to tunnel the VNC connection through\n"
+"an encrypted channel such as ssh(1).\n"
"\n"
"For additional info see: http://www.karlrunge.com/x11vnc/\n"
" and http://www.karlrunge.com/x11vnc/#faq\n"
@@ -8532,36 +12904,54 @@ static void print_help(void) {
"\n"
"-id windowid Show the window corresponding to \"windowid\" not\n"
" the entire display. New windows like popup menus,\n"
-" etc may not be seen, or will be clipped. x11vnc may\n"
-" crash if the window changes size, is iconified, etc.\n"
-" Use xwininfo(1) to get the window id. Primarily useful\n"
-" for exporting very simple applications.\n"
+" transient toplevels, etc, may not be seen or may be\n"
+" clipped. Disabling SaveUnders or BackingStore in the\n"
+" X server may help show them. x11vnc may crash if the\n"
+" window is initially partially obscured, changes size,\n"
+" is iconified, etc. Some steps are taken to avoid this\n"
+" and the -xrandr mechanism is used to track resizes. Use\n"
+" xwininfo(1) to get the window id, or use \"-id pick\"\n"
+" to have x11vnc run xwininfo(1) for you and extract\n"
+" the id. The -id option is useful for exporting very\n"
+" simple applications (e.g. the current view on a webcam).\n"
"-sid windowid As -id, but instead of using the window directly it\n"
-" shifts a root view to it: this shows saveUnders menus,\n"
+" shifts a root view to it: this shows SaveUnders menus,\n"
" etc, although they will be clipped if they extend beyond\n"
" the window.\n"
"-flashcmap In 8bpp indexed color, let the installed colormap flash\n"
" as the pointer moves from window to window (slow).\n"
"-notruecolor For 8bpp displays, force indexed color (i.e. a colormap)\n"
" even if it looks like 8bpp TrueColor. (rare problem)\n"
+"-visual n Experimental option: probably does not do what you\n"
+" think. It simply *forces* the visual used for the\n"
+" framebuffer; this may be a bad thing... (e.g. messes\n"
+" up colors or cause a crash). It is useful for testing\n"
+" and for some workarounds. n may be a decimal number,\n"
+" or 0x hex. Run xdpyinfo(1) for the values. One may\n"
+" also use \"TrueColor\", etc. see <X11/X.h> for a list.\n"
+" If the string ends in \":m\" for better or for worse\n"
+" the visual depth is forced to be m.\n"
"-overlay Handle multiple depth visuals on one screen, e.g. 8+24\n"
" and 24+8 overlay visuals (the 32 bits per pixel are\n"
" packed with 8 for PseudoColor and 24 for TrueColor).\n"
"\n"
-" Currently -overlay only works on Solaris (it uses\n"
-" XReadScreen(3X11)). There is a problem with image\n"
-" \"bleeding\" around transient popup menus (but not\n"
-" for the menu itself): a workaround is to disable\n"
-" SaveUnders by passing the \"-su\" argument to Xsun\n"
-" (in /etc/dt/config/Xservers, say). Also note that,\n"
-" the mouse cursor shape is exactly correct in this mode.\n"
+" Currently -overlay only works on Solaris via\n"
+" XReadScreen(3X11) and IRIX using XReadDisplay(3).\n"
+" On Solaris there is a problem with image \"bleeding\"\n"
+" around transient popup menus (but not for the menu\n"
+" itself): a workaround is to disable SaveUnders\n"
+" by passing the \"-su\" argument to Xsun (in\n"
+" /etc/dt/config/Xservers). Also note that the mouse\n"
+" cursor shape is exactly correct in this mode.\n"
"\n"
" Use -overlay as a workaround for situations like these:\n"
-" Some legacy applications require the default visual\n"
+" Some legacy applications require the default visual to\n"
" be 8bpp (8+24), or they will use 8bpp PseudoColor even\n"
" when the default visual is depth 24 TrueColor (24+8).\n"
" In these cases colors in some windows will be messed\n"
-" up in x11vnc unless -overlay is used.\n"
+" up in x11vnc unless -overlay is used. Another use of\n"
+" -overlay is to enable showing the exact mouse cursor\n"
+" shape (details below).\n"
"\n"
" Under -overlay, performance will be somewhat degraded\n"
" due to the extra image transformations required.\n"
@@ -8571,22 +12961,15 @@ static void print_help(void) {
" visual (some apps have -use24 or -visual options).\n"
"-overlay_nocursor Sets -overlay, but does not try to draw the exact mouse\n"
" cursor shape using the overlay mechanism.\n"
-"-visual n Experimental option: probably does not do what you\n"
-" think. It simply *forces* the visual used for the\n"
-" framebuffer; this may be a bad thing... It is useful for\n"
-" testing and for some workarounds. n may be a decimal\n"
-" number, or 0x hex. Run xdpyinfo(1) for the values.\n"
-" One may also use \"TrueColor\", etc. see <X11/X.h>\n"
-" for a list. If the string ends in \":m\" for better\n"
-" or for worse the visual depth is forced to be m.\n"
"\n"
-"-scale fraction Scale the framebuffer by factor \"fraction\".\n"
-" Values less than 1 shrink the fb. Note: image may not\n"
-" be sharp and response may be slower. Currently the\n"
-" cursor shape is not scaled. If \"fraction\" contains\n"
-" a decimal point \".\" it is taken as a floating point\n"
-" number, alternatively the notation \"m/n\" may be used\n"
-" to denote fractions exactly, e.g. -scale 2/3.\n"
+"-scale fraction Scale the framebuffer by factor \"fraction\". Values\n"
+" less than 1 shrink the fb, larger ones expand it.\n"
+" Note: image may not be sharp and response may be\n"
+" slower. Currently the cursor shape is not scaled.\n"
+" If \"fraction\" contains a decimal point \".\" it\n"
+" is taken as a floating point number, alternatively\n"
+" the notation \"m/n\" may be used to denote fractions\n"
+" exactly, e.g. -scale 2/3.\n"
"\n"
" Scaling Options: can be added after \"fraction\" via\n"
" \":\", to supply multiple \":\" options use commas.\n"
@@ -8605,6 +12988,10 @@ static void print_help(void) {
" disconnects, opposite of -forever. This is the Default.\n"
"-forever Keep listening for more connections rather than exiting\n"
" as soon as the first client(s) disconnect. Same as -many\n"
+"-inetd Launched by inetd(1): stdio instead of listening socket.\n"
+" Note: if you are not redirecting stderr to a log file\n"
+" (via shell 2> or -o option) you must also specify the\n"
+" -q option, otherwise the stderr goes to the viewer.\n"
"-connect string For use with \"vncviewer -listen\" reverse connections.\n"
" If \"string\" has the form \"host\" or \"host:port\"\n"
" the connection is made once at startup. Use commas\n"
@@ -8616,21 +13003,17 @@ static void print_help(void) {
"-novncconnect VNC program vncconnect(1). When the property is\n"
" set to \"host\" or \"host:port\" establish a reverse\n"
" connection. Using xprop(1) instead of vncconnect may\n"
-" work, see the FAQ. Default: %s\n"
-"-inetd Launched by inetd(1): stdio instead of listening socket.\n"
-" Note: if you are not redirecting stderr to a log file\n"
-" (via shell 2> or -o option) you must also specify the\n"
-" -q option.\n"
+" work (see the FAQ). Default: %s\n"
"\n"
-"-allow addr1[,addr2..] Only allow client connections from IP addresses matching\n"
-" the comma separated list of numerical addresses.\n"
-" Can be a prefix, e.g. \"192.168.100.\" to match a\n"
-" simple subnet, for more control build libvncserver\n"
-" with libwrap support. If the list contains a \"/\"\n"
-" it instead is a interpreted as a file containing\n"
-" addresses or prefixes that is re-read each time a new\n"
-" client connects. Lines can be commented out with the\n"
-" \"#\" character in the usual way.\n"
+"-allow host1[,host2..] Only allow client connections from hosts matching\n"
+" the comma separated list of hostnames or IP addresses.\n"
+" Can also be a numerical IP prefix, e.g. \"192.168.100.\"\n"
+" to match a simple subnet, for more control build\n"
+" libvncserver with libwrap support (See the FAQ). If the\n"
+" list contains a \"/\" it instead is a interpreted as a\n"
+" file containing addresses or prefixes that is re-read\n"
+" each time a new client connects. Lines can be commented\n"
+" out with the \"#\" character in the usual way.\n"
"-localhost Same as -allow 127.0.0.1\n"
"-viewpasswd string Supply a 2nd password for view-only logins. The -passwd\n"
" (full-access) password must also be supplied.\n"
@@ -8639,7 +13022,7 @@ static void print_help(void) {
" If a second non blank line exists in the file it is\n"
" taken as a view-only password (i.e. -viewpasswd) Note:\n"
" this is a simple plaintext passwd, see also -rfbauth\n"
-" and -storepasswd below.\n"
+" and -storepasswd below for obfuscated passwords.\n"
"-storepasswd pass file Store password \"pass\" as the VNC password in the\n"
" file \"file\". Once the password is stored the\n"
" program exits. Use the password via \"-rfbauth file\"\n"
@@ -8648,10 +13031,10 @@ static void print_help(void) {
" should be allowed to connect or not. \"string\" is\n"
" an external command run via system(3) or some special\n"
" cases described below. Be sure to quote \"string\"\n"
-" if it contains spaces, etc. If the external command\n"
-" returns 0 the client is accepted, otherwise the client\n"
-" is rejected. See below for an extension to accept a\n"
-" client view-only.\n"
+" if it contains spaces, shell characters, etc. If the\n"
+" external command returns 0 the client is accepted,\n"
+" otherwise the client is rejected. See below for an\n"
+" extension to accept a client view-only.\n"
"\n"
" Environment: The RFB_CLIENT_IP environment variable will\n"
" be set to the incoming client IP number and the port\n"
@@ -8661,7 +13044,7 @@ static void print_help(void) {
" of the tcp virtual circuit. The x11vnc process\n"
" id will be in RFB_X11VNC_PID, a client id number in\n"
" RFB_CLIENT_ID, and the number of other connected clients\n"
-" in RFB_CLIENT_COUNT.\n"
+" in RFB_CLIENT_COUNT. RFB_MODE will be \"accept\"\n"
"\n"
" If \"string\" is \"popup\" then a builtin popup window\n"
" is used. The popup will time out after 120 seconds,\n"
@@ -8669,7 +13052,8 @@ static void print_help(void) {
" (use 0 for no timeout)\n"
"\n"
" If \"string\" is \"xmessage\" then an xmessage(1)\n"
-" invocation is used for the command.\n"
+" invocation is used for the command. xmessage must be\n"
+" installed on the machine for this to work.\n"
"\n"
" Both \"popup\" and \"xmessage\" will present an option\n"
" for accepting the client \"View-Only\" (the client\n"
@@ -8685,19 +13069,23 @@ static void print_help(void) {
" the default action (in case the command returns an\n"
" unexpected value). E.g. \"no:*\" is a good choice.\n"
"\n"
-" Note that x11vnc blocks while the external command or\n"
+" Note that x11vnc blocks while the external command\n"
" or popup is running (other clients may see no updates\n"
" during this period).\n"
"\n"
" More -accept tricks: use \"popupmouse\" to only allow\n"
" mouse clicks in the builtin popup to be recognized.\n"
-" Similarly use \"popupkey\" to only recognize keystroke\n"
-" responses. All 3 of the popup keywords can be followed\n"
+" Similarly use \"popupkey\" to only recognize\n"
+" keystroke responses. These are to help avoid the\n"
+" user accidentally accepting a client by typing or\n"
+" clicking. All 3 of the popup keywords can be followed\n"
" by +N+M to supply a position for the popup window.\n"
" The default is to center the popup window.\n"
"-gone string As -accept, except to run a user supplied command when\n"
-" a client goes away (disconnects). Unlike -accept,\n"
-" the command return code is not interpreted by x11vnc.\n"
+" a client goes away (disconnects). RFB_MODE will be\n"
+" set to \"gone\" and the other RFB_* variables are as\n"
+" in -accept. Unlike -accept, the command return code\n"
+" is not interpreted by x11vnc. Example: -gone 'xlock &'\n"
"\n"
"-noshm Do not use the MIT-SHM extension for the polling.\n"
" Remote displays can be polled this way: be careful this\n"
@@ -8720,8 +13108,44 @@ static void print_help(void) {
" In general on XINERAMA displays you may need to use the\n"
" -xwarppointer option if the mouse pointer misbehaves.\n"
"\n"
+"-xrandr [mode] If the display supports the XRANDR (X Resize, Rotate\n"
+" and Reflection) extension, and you expect XRANDR events\n"
+" to occur to the display while x11vnc is running, this\n"
+" options indicates x11vnc should try to respond to\n"
+" them (as opposed to simply crashing by assuming the\n"
+" old screen size). See the xrandr(1) manpage and run\n"
+" 'xrandr -q' for more info. [mode] is optional and\n"
+" described below.\n"
+"\n"
+" Since watching for XRANDR events and errors increases\n"
+" polling overhead, only use this option if XRANDR changes\n"
+" are expected. For example on a rotatable screen PDA or\n"
+" laptop, or using a XRANDR-aware Desktop where you resize\n"
+" often. It is best to be viewing with a vncviewer that\n"
+" supports the NewFBSize encoding, since it knows how to\n"
+" react to screen size changes. Otherwise, libvncserver\n"
+" tries to do so something reasonable for viewers that\n"
+" cannot do this (portions of the screen may be clipped,\n"
+" unused, etc).\n"
+"\n"
+" \"mode\" defaults to \"resize\", which means create a\n"
+" new, resized, framebuffer and hope all viewers can cope\n"
+" with the change. \"newfbsize\" means first disconnect\n"
+" all viewers that do not support the NewFBSize VNC\n"
+" encoding, and then resize the framebuffer. \"exit\"\n"
+" means disconnect all viewer clients, and then terminate\n"
+" x11vnc.\n"
+"-padgeom WxH Whenever a new vncviewer connects, the framebuffer is\n"
+" replaced with a fake, solid black one of geometry WxH.\n"
+" Shortly afterwards the framebuffer is replaced with the\n"
+" real one. This is intended for use with vncviewers\n"
+" that do not support NewFBSize and one wants to make\n"
+" sure the initial viewer geometry will be big enough\n"
+" to handle all subsequent resizes (e.g. under -xrandr,\n"
+" -remote id:windowid, rescaling, etc.\n"
+"\n"
"-o logfile Write stderr messages to file \"logfile\" instead of\n"
-" to the terminal. Same as -logfile \"file\".\n"
+" to the terminal. Same as \"-logfile file\".\n"
"-rc filename Use \"filename\" instead of $HOME/.x11vncrc for rc file.\n"
"-norc Do not process any .x11vncrc file for options.\n"
"-h, -help Print this help text.\n"
@@ -8749,16 +13173,18 @@ static void print_help(void) {
"-isolevel3 When in modtweak mode, always send ISO_Level3_Shift to\n"
" the X server instead of Mode_switch (AltGr).\n"
#endif
-"-xkb When in modtweak mode, use the XKEYBOARD extension\n"
-" (if it exists) to do the modifier tweaking. This is\n"
-" powerful and should be tried if there are still\n"
-" keymapping problems when using the simpler -modtweak.\n"
-"-skip_keycodes string Skip keycodes not on your keyboard but your X server\n"
-" thinks exist. Currently only applies to -xkb mode.\n"
-" \"string\" is a comma separated list of decimal\n"
-" keycodes. Use this option to help x11vnc in the reverse\n"
-" problem it tries to solve: Keysym -> Keycode(s) when\n"
-" ambiguities exist. E.g. -skip_keycodes 94,114\n"
+"-xkb When in modtweak mode, use the XKEYBOARD extension (if\n"
+" the X display supports it) to do the modifier tweaking.\n"
+" This is powerful and should be tried if there are still\n"
+" keymapping problems when using -modtweak by itself.\n"
+"-skip_keycodes string Ignore the comma separated list of decimal keycodes.\n"
+" Perhaps these are keycodes not on your keyboard but\n"
+" your X server thinks exist. Currently only applies\n"
+" to -xkb mode. Use this option to help x11vnc in the\n"
+" reverse problem it tries to solve: Keysym -> Keycode(s)\n"
+" when ambiguities exist (more than one Keycode per\n"
+" Keysym). Run 'xmodmap -pk' to see your keymapping.\n"
+" E.g. \"-skip_keycodes 94,114\"\n"
"-add_keysyms If a Keysym is received from a VNC viewer and\n"
" that Keysym does not exist in the X server, then\n"
" add the Keysym to the X server's keyboard mapping.\n"
@@ -8785,8 +13211,8 @@ static void print_help(void) {
" form: key1-key2,key3-key4,... See <X11/keysymdef.h>\n"
" header file for a list of Keysym names, or use\n"
" xev(1). To map a key to a button click, use the\n"
-" fake Keysyms \"Button1\", ..., etc.\n"
-" E.g. -remap Super_R-Button2\n"
+" fake Keysyms \"Button1\", ..., etc. E.g. \"-remap\n"
+" Super_R-Button2\" (useful for pasting on a laptop)\n"
"-norepeat Option -norepeat disables X server key auto repeat\n"
"-repeat when VNC clients are connected. This works around a\n"
" repeating keystrokes bug (triggered by long processing\n"
@@ -8819,26 +13245,35 @@ static void print_help(void) {
" network traffic by not having to send the cursor image\n"
" every time the pointer is moved), in which case these\n"
" extensions are used (see -nocursorshape and -nocursorpos\n"
-" below). For other viewers the cursor shape is written\n"
-" directly to the framebuffer every time the pointer is\n"
-" moved or changed and gets sent along with the other\n"
-" framebuffer updates. In this case, there will be\n"
-" some lag between the vnc viewer pointer and the remote\n"
-" cursor position.\n"
+" below to disable). For other viewers the cursor shape\n"
+" is written directly to the framebuffer every time the\n"
+" pointer is moved or changed and gets sent along with\n"
+" the other framebuffer updates. In this case, there\n"
+" will be some lag between the vnc viewer pointer and\n"
+" the remote cursor position.\n"
"\n"
" If the X display supports retrieving the cursor shape\n"
-" information from the X server, then the default\n"
-" is to use that mode. On Solaris this requires\n"
-" the SUN_OVL extension and the -overlay option to be\n"
-" supplied. (see also the -overlay_nomouse option). (Soon)\n"
-" on XFree86/Xorg the XFIXES extension is required.\n"
-" Either can be disabled with -nocursor, and also some\n"
-" values of the \"mode\" option below.\n"
+" information from the X server, then the default is\n"
+" to use that mode. On Solaris this can be done with\n"
+" the SUN_OVL extension using -overlay (see also the\n"
+" -overlay_nomouse option). A similar overlay scheme\n"
+" is used on IRIX. Xorg (e.g. Linux) and recent Solaris\n"
+" Xsun servers support the XFIXES extension to retrieve\n"
+" the exact cursor shape from the X server. If XFIXES\n"
+" is present it is preferred over Overlay and is used by\n"
+" default (see -noxfixes below). This can be disabled\n"
+" with -nocursor, and also some values of the \"mode\"\n"
+" option below.\n"
"\n"
" The \"mode\" string can be used to fine-tune the\n"
" displaying of cursor shapes. It can be used the\n"
" following ways:\n"
"\n"
+" \"-cursor arrow\" - just show the standard arrow\n"
+" nothing more or nothing less.\n"
+"\n"
+" \"-cursor none\" - same as \"-nocursor\"\n"
+"\n"
" \"-cursor X\" - when the cursor appears to be on the\n"
" root window, draw the familiar X shape. Some desktops\n"
" such as GNOME cover up the root window completely,\n"
@@ -8855,21 +13290,24 @@ static void print_help(void) {
" more feedback about the cursor shape.\n"
"\n"
" \"-cursor most\" - try to show as many cursors as\n"
-" possible. Often this will only be the same as \"some\".\n"
-" On Solaris if XFIXES is not available, -overlay mode\n"
-" will be used.\n"
+" possible. Often this will only be the same as \"some\"\n"
+" unless the display has overlay visuals or XFIXES\n"
+" extensions available. On Solaris and IRIX if XFIXES\n"
+" is not available, -overlay mode will be attempted.\n"
"\n"
+"-noxfixes Do not use the XFIXES extension to draw the exact cursor\n"
+" shape even if it is available.\n"
"-nocursorshape Do not use the TightVNC CursorShapeUpdates extension\n"
" even if clients support it. See -cursor above.\n"
"-cursorpos Option -cursorpos enables sending the X cursor position\n"
"-nocursorpos back to all vnc clients that support the TightVNC\n"
" CursorPosUpdates extension. Other clients will be able\n"
" to see the pointer motions. Default: %s\n"
-"-xwarppointer Move the pointer with XWarpPointer(3X) instead of XTEST\n"
-" extension. Use this as a workaround if the pointer\n"
-" motion behaves incorrectly, e.g. on touchscreens or\n"
-" other non-standard setups. Also sometimes needed on\n"
-" XINERAMA displays.\n"
+"-xwarppointer Move the pointer with XWarpPointer(3X) instead of\n"
+" the XTEST extension. Use this as a workaround\n"
+" if the pointer motion behaves incorrectly, e.g.\n"
+" on touchscreens or other non-standard setups.\n"
+" Also sometimes needed on XINERAMA displays.\n"
"\n"
"-buttonmap string String to remap mouse buttons. Format: IJK-LMN, this\n"
" maps buttons I -> L, etc., e.g. -buttonmap 13-31\n"
@@ -8899,14 +13337,24 @@ static void print_help(void) {
" improves response on slow setups, but you lose all\n"
" visual feedback for drags, text selection, and some\n"
" menu traversals.\n"
-"-old_pointer Use the original pointer input handling mechanism.\n"
-" See check_input() and pointer() in source file for\n"
-" details.\n"
-"-old_pointer2 The default pointer input handling algorithm was changed\n"
-" again, this option indicates to use the second one.\n"
-"-input_skip n For the old pointer handling when non-threaded: try to\n"
+"-pointer_mode n Various pointer update schemes. The problem is pointer\n"
+" motion can cause rapid changes on the screen, e.g. a\n"
+" window drag. Neither x11vnc nor the bandwidth to the\n"
+" vncviewers can keep up these rapid screen changes:\n"
+" everything bogs down when dragging or scrolling.\n"
+" Note that most video h/w is optimized for writing, not\n"
+" reading (a 50X rate difference is possible) and x11vnc\n"
+" is reading all the time. So a scheme has to be used to\n"
+" \"eat\" much of that pointer input before re-polling the\n"
+" screen. n can be 1 to %d. n=1 was the original scheme\n"
+" used to about Jan 2004. n=2 is an improved scheme.\n"
+" n=3 is basically a dynamic -nodragging mode: it detects\n"
+" if the mouse drag motion has paused and refreshes\n"
+" the display. n=4 is TBD. The default n is %d.\n"
+"-input_skip n For the pointer handling when non-threaded: try to\n"
" read n user input events before scanning display. n < 0\n"
" means to act as though there is always user input.\n"
+" Default: %d\n"
"\n"
"-debug_pointer Print debugging output for every pointer event.\n"
"-debug_keyboard Print debugging output for every keyboard event.\n"
@@ -8919,6 +13367,9 @@ static void print_help(void) {
" down on load. Default: %d\n"
"-nap Monitor activity and if low take longer naps between\n"
" polls to really cut down load when idle. Default: %s\n"
+"-sb time Time in seconds after NO activity (e.g. screen blank)\n"
+" to really throttle down the screen polls (i.e. sleep\n"
+" for about 1.5 secs). Use 0 to disable. Default: %d\n"
"\n"
"-sigpipe string Broken pipe (SIGPIPE) handling. \"string\" can be\n"
" \"ignore\" or \"exit\". For \"ignore\" libvncserver\n"
@@ -8937,6 +13388,313 @@ static void print_help(void) {
" by checking the tile near the boundary. Default: %d\n"
"-fuzz n Tolerance in pixels to mark a tiles edges as changed.\n"
" Default: %d\n"
+"\n"
+"-gui [gui-opts] Start up a simple tcl/tk gui based on the the remote\n"
+" control options -remote/-query described below.\n"
+" Requires the \"wish\" program to be installed on the\n"
+" machine. \"gui-opts\" is not required: the default is\n"
+" to start up both the gui and x11vnc with the gui showing\n"
+" up on the X display in the environment variable DISPLAY.\n"
+"\n"
+" \"gui-opts\" can be a comma separated list of items.\n"
+" Currently there are only two types of items: 1) a gui\n"
+" mode and 2) the X display the gui should display on.\n"
+" The gui mode can be \"start\", \"conn\", or \"wait\"\n"
+" \"start\" is the default mode above and is not required.\n"
+" \"conn\" means do not automatically start up x11vnc,\n"
+" but instead just try to connect to an existing x11vnc\n"
+" process. \"wait\" means just start the gui and nothing\n"
+" else (you will later instruct the gui to start x11vnc\n"
+" or connect to an existing one.)\n"
+"\n"
+" Note the possible confusion regarding the potentially\n"
+" two different X displays: x11vnc polls one, but you\n"
+" may want the gui to appear on another. For example, if\n"
+" you ssh in and x11vnc is not running yet you may want\n"
+" the gui to come back to you via your ssh redirected X\n"
+" display (e.g. localhost:10).\n"
+"\n"
+" Examples: \"x11vnc -gui\", \"x11vnc -gui localhost:10\",\n"
+" \"x11vnc -gui :10\", \"x11vnc -gui wait,:10\",\n"
+" \"x11vnc -gui <x11vnc-opts...>\"\n"
+"\n"
+" If you do not specify a gui X display in \"gui-opts\"\n"
+" then the DISPLAY environment variable and -display\n"
+" option are tried (in that order). Regarding the x11vnc\n"
+" X display the gui will try to connect to, it first\n"
+" tries -display and then DISPLAY. For example, \"x11vnc\n"
+" -display :0 -gui otherhost:0\", will remote control an\n"
+" x11vnc polling :0 and display the gui on otherhost:0\n"
+"\n"
+" If you do not intend to start x11vnc from the gui\n"
+" (i.e. just remote control an existing one), then the\n"
+" gui process can run on a different machine from the\n"
+" x11vnc server as long as X permissions, etc. permit\n"
+" communication between the two.\n"
+"\n"
+"-remote command Remotely control some aspects of an already running\n"
+" x11vnc server. \"-R\" and \"-r\" are aliases for\n"
+" \"-remote\". After the remote control command is\n"
+" sent to the running server the 'x11vnc -remote ...'\n"
+" command exits. You can often use the -query command\n"
+" (see below) to see if the x11vnc server processed your\n"
+" -remote command.\n"
+"\n"
+" The default communication channel is that of X\n"
+" properties (specifically VNC_CONNECT), and so this\n"
+" command must be run with correct settings for DISPLAY\n"
+" and possibly XAUTHORITY to connect to the X server\n"
+" and set the property. Alternatively, use the -display\n"
+" and -auth options to set them to the correct values.\n"
+" The running server cannot use the -novncconnect option\n"
+" because that disables the communication channel.\n"
+" See below for alternate channels.\n"
+"\n"
+" For example: 'x11vnc -remote stop' (which is the same as\n"
+" 'x11vnc -R stop') will close down the x11vnc server.\n"
+" 'x11vnc -R shared' will enable shared connections, and\n"
+" 'x11vnc -R scale:3/4' will rescale the desktop.\n"
+"\n"
+" Note: the more drastic the change induced by the -remote\n"
+" command, the bigger the chance for bugs or crashes.\n"
+" Please report reproducible bugs.\n"
+"\n"
+" The following -remote/-R commands are supported:\n"
+"\n"
+" stop terminate the server, same as \"quit\"\n"
+" \"exit\" or \"shutdown\"\n"
+" ping see if the x11vnc server responds.\n"
+" Return is: ans=ping:<xdisplay>\n"
+" blacken try to push a black fb update to all\n"
+" clients (due to timings a client\n"
+" could miss it). Same as \"zero\", also\n"
+" \"zero:x1,y1,x2,y2\" for a rectangle.\n"
+" refresh send the entire fb to all clients.\n"
+" reset recreate the fb, polling memory, etc.\n"
+" id:windowid set -id window to \"windowid\". empty\n"
+" or \"root\" to go back to root window\n"
+" sid:windowid set -sid window to \"windowid\"\n"
+" flashcmap enable -flashcmap mode.\n"
+" noflashcmap disable -flashcmap mode.\n"
+" notruecolor enable -notruecolor mode.\n"
+" truecolor disable -notruecolor mode.\n"
+" overlay enable -overlay mode (if applicable).\n"
+" nooverlay disable -overlay mode.\n"
+" overlay_cursor in -overlay mode, enable cursor drawing.\n"
+" overlay_nocursor disable cursor drawing. same as\n"
+" nooverlay_cursor.\n"
+" visual:vis set -visual to \"vis\"\n"
+" scale:frac set -scale to \"frac\"\n"
+" viewonly enable -viewonly mode.\n"
+" noviewonly disable -viewonly mode.\n"
+" shared enable -shared mode.\n"
+" noshared disable -shared mode.\n"
+" forever enable -forever mode.\n"
+" noforever disable -forever mode.\n"
+" deny deny any new connections, same as \"lock\"\n"
+" nodeny allow new connections, same as \"unlock\"\n"
+" connect:host do reverse connection to host, \"host\"\n"
+" may be a comma separated list of hosts\n"
+" or host:ports. See -connect.\n"
+" disconnect:host disconnect any clients from \"host\"\n"
+" same as \"close:host\". Use host\n"
+" \"all\" to close all current clients.\n"
+" If you know the client internal hex ID,\n"
+" e.g. 0x3 (returned by -query clients and\n"
+" RFB_CLIENT_ID), you can use that too.\n"
+" allowonce:host For the next connection only, allow\n"
+" connection from \"host\".\n"
+" allow:hostlist set -allow list to (comma separated)\n"
+" \"hostlist\". See -allow and -localhost.\n"
+" Do not use with -allow /path/to/file\n"
+" Use \"+host\" to add a single host, and\n"
+" use \"-host\" to delete a single host\n"
+" localhost enable -localhost mode\n"
+" nolocalhost disable -localhost mode\n"
+" accept:cmd set -accept \"cmd\" (empty to disable).\n"
+" gone:cmd set -gone \"cmd\" (empty to disable).\n"
+" noshm enable -noshm mode.\n"
+" shm disable -noshm mode (i.e. use shm).\n"
+" flipbyteorder enable -flipbyteorder mode, you may need\n"
+" to set noshm for this to do something.\n"
+" noflipbyteorder disable -flipbyteorder mode.\n"
+" onetile enable -onetile mode. (you may need to\n"
+" set shm for this to do something)\n"
+" noonetile disable -onetile mode.\n"
+" blackout:str set -blackout \"str\" (empty to disable).\n"
+" See -blackout for the form of \"str\"\n"
+" (basically: WxH+X+Y,...)\n"
+" Use \"+WxH+X+Y\" to append a single\n"
+" rectangle use \"-WxH+X+Y\" to delete one\n"
+" xinerama enable -xinerama mode. (if applicable)\n"
+" noxinerama disable -xinerama mode.\n"
+" xrandr enable -xrandr mode. (if applicable)\n"
+" noxrandr disable -xrandr mode.\n"
+" xrandr_mode:mode set the -xrandr mode to \"mode\".\n"
+" padgeom:WxH set -padgeom to WxH (empty to disable)\n"
+" If WxH is \"force\" or \"do\" the padded\n"
+" geometry fb is immediately applied.\n"
+" quiet enable -quiet mode.\n"
+" noquiet disable -quiet mode.\n"
+" modtweak enable -modtweak mode.\n"
+" nomodtweak enable -nomodtweak mode.\n"
+" xkb enable -xkb modtweak mode.\n"
+" noxkb disable -xkb modtweak mode.\n"
+" skip_keycodes:str enable -xkb -skip_keycodes \"str\".\n"
+" add_keysyms enable -add_keysyms mode.\n"
+" noadd_keysyms stop adding keysyms. those added will\n"
+" still be removed at exit.\n"
+" clear_mods enable -clear_mods mode and clear them.\n"
+" noclear_mods disable -clear_mods mode.\n"
+" clear_keys enable -clear_keys mode and clear them.\n"
+" noclear_keys disable -clear_keys mode.\n"
+" remap:str set -remap \"str\" (empty to disable).\n"
+" See -remap for the form of \"str\"\n"
+" (basically: key1-key2,key3-key4,...)\n"
+" Use \"+key1-key2\" to append a single\n"
+" keymapping, use \"-key1-key2\" to delete.\n"
+" norepeat enable -norepeat mode.\n"
+" repeat disable -norepeat mode.\n"
+" bell enable bell (if supported).\n"
+" nobell disable bell.\n"
+" sel disable -nosel mode.\n"
+" nosel enable -nosel mode.\n"
+" primary disable -noprimary mode.\n"
+" noprimary enable -noprimary mode.\n"
+" cursor:mode enable -cursor \"mode\".\n"
+" show_cursor enable showing a cursor.\n"
+" noshow_cursor disable showing a cursor. (same as\n"
+" \"nocursor\")\n"
+" xfixes enable xfixes cursor shape mode.\n"
+" noxfixes disable xfixes cursor shape mode.\n"
+" cursorshape disable -nocursorshape mode.\n"
+" nocursorshape enable -nocursorshape mode.\n"
+" cursorpos disable -nocursorpos mode.\n"
+" nocursorpos enable -nocursorpos mode.\n"
+" xwarp enable -xwarppointer mode.\n"
+" noxwarp disable -xwarppointer mode.\n"
+" buttonmap:str set -buttonmap \"str\", empty to disable\n"
+" dragging disable -nodragging mode.\n"
+" nodragging enable -nodragging mode.\n"
+" pointer_mode n set -pointer_mode to n.\n"
+" input_skip n set -input_skip to n.\n"
+" debug_pointer enable -debug_pointer, same as \"dp\"\n"
+" nodebug_pointer disable -debug_pointer, same as \"nodp\"\n"
+" debug_keyboard enable -debug_keyboard, same as \"dk\"\n"
+" nodebug_keyboard disable -debug_keyboard, same as \"nodk\"\n"
+" defer:n set -defer to n ms,same as deferupdate:n\n"
+" wait:n set -wait to n ms.\n"
+" nap enable -nap mode.\n"
+" nonap disable -nap mode.\n"
+" sb:n set -sb to n s, same as screen_blank:n\n"
+" fs:frac set -fs fraction to \"frac\", e.g. 0.5\n"
+" gaps:n set -gaps to n.\n"
+" grow:n set -grow to n.\n"
+" fuzz:n set -fuzz to n.\n"
+" progressive:n set libvncserver -progressive slice\n"
+" height parameter to n.\n"
+" file:name run -remote commands from file \"name\",\n"
+" one command per line,blank and # skipped\n"
+" noremote disable the -remote command processing,\n"
+" it cannot be turned back on.\n"
+"\n"
+" The vncconnect(1) command from standard VNC\n"
+" distributions may also be used if string is prefixed\n"
+" with \"cmd=\" E.g. 'vncconnect cmd=stop'. Under some\n"
+" circumstances xprop(1) can used if it supports -set\n"
+" (see the FAQ).\n"
+"\n"
+" If \"-connect /path/to/file\" has been supplied to the\n"
+" running x11vnc server then that file can be used as a\n"
+" communication channel (this is the only way to remote\n"
+" control one of many x11vnc's polling the same X display)\n"
+" Simply run: 'x11vnc -connect /path/to/file -remote ...'\n"
+" or you can directly write to the file via something\n"
+" like: \"echo cmd=stop > /path/to/file\", etc.\n"
+"\n"
+"-query variable Like -remote, except just query the value of\n"
+" \"variable\". \"-Q\" is an alias for \"-query\".\n"
+" Multiple queries can be done by separating variables\n"
+" by commas, e.g. -query var1,var2. The results come\n"
+" back in the form ans=var1:value1,ans=var2:value2,...\n"
+" to the standard output. If a variable is read-only,\n"
+" it comes back with prefix \"aro=\" instead of \"ans=\".\n"
+"\n"
+" Some -remote commands are pure actions that do not make\n"
+" sense as variables, e.g. \"stop\" or \"disconnect\",\n"
+" in these cases the value returned is \"N/A\". To direct\n"
+" a query straight to the VNC_CONNECT property or connect\n"
+" file use \"qry=...\" instead of \"cmd=...\"\n"
+"\n"
+" Here is the current list of \"variables\" that can\n"
+" be supplied to the -query command. This includes the\n"
+" \"N/A\" ones that return no useful info. For variables\n"
+" names that do not correspond to an x11vnc option or\n"
+" remote command, we hope the name makes it obvious what\n"
+" the returned value corresponds to (hint: the ext_*\n"
+" variables correspond to the presence of X extensions):\n"
+"\n"
+" ans= stop quit exit shutdown ping blacken zero refresh\n"
+" reset close disconnect id sid flashcmap noflashcmap\n"
+" truecolor notruecolor overlay nooverlay overlay_cursor\n"
+" overlay_yescursor nooverlay_cursor overlay_nocursor\n"
+" visual scale viewonly noviewonly shared noshared\n"
+" forever noforever once deny lock nodeny unlock connect\n"
+" allowonce allow localhost nolocalhost accept gone shm\n"
+" noshm flipbyteorder noflipbyteorder onetile noonetile\n"
+" blackout xinerama noxinerama xrandr noxrandr xrandr_mode\n"
+" padgeom quiet q noquiet modtweak nomodtweak xkb noxkb\n"
+" skip_keycodes add_keysyms noadd_keysyms clear_mods\n"
+" noclear_mods clear_keys noclear_keys remap repeat\n"
+" norepeat bell nobell sel nosel primary noprimary\n"
+" cursorshape nocursorshape cursorpos nocursorpos cursor\n"
+" show_cursor noshow_cursor nocursor xfixes noxfixes xwarp\n"
+" xwarppointer noxwarp noxwarppointer buttonmap dragging\n"
+" nodragging pointer_mode input_skip debug_pointer dp\n"
+" nodebug_pointer nodp debug_keyboard dk nodebug_keyboard\n"
+" nodk deferupdate defer wait nap nonap sb screen_blank\n"
+" fs gaps grow fuzz progressive noremote\n"
+"\n"
+" aro= display vncdisplay desktopname desktop auth\n"
+" rootshift scale_str scaled_x scaled_y scale_numer\n"
+" scale_denom scale_fac scaling_noblend scaling_nomult4\n"
+" scaling_pad scaling_interpolate inetd safer unsafe\n"
+" passwdfile using_shm logfile o rc norc h help V version\n"
+" lastmod bg nofb sigpipe threads clients client_count\n"
+" pid ext_xtest ext_xkb ext_xshm ext_xinerama ext_overlay\n"
+" ext_xfixes ext_xdamage ext_xrandr rootwin num_buttons\n"
+" button_mask mouse_x mouse_y bpp depth indexed_color\n"
+" dpy_x dpy_y rfbport rfbwait rfbauth passwd alwaysshared\n"
+" dontdisconnect httpdir enablehttpproxy\n"
+"\n"
+"-noremote Do not process any remote control commands or queries.\n"
+"\n"
+" A note about security wrt remote control commands.\n"
+" If someone can connect to the X display and change the\n"
+" property VNC_CONNECT, then they can remotely control\n"
+" x11vnc. Normally access to the X display is protected.\n"
+" Note that if they can modify VNC_CONNECT, they could\n"
+" also run their own x11vnc and have complete control\n"
+" of the desktop. If the \"-connect /path/to/file\"\n"
+" channel is being used, obviously anyone who can write\n"
+" to /path/to/file can remotely control x11vnc. So be\n"
+" sure to protect the X display and that file's write\n"
+" permissions.\n"
+"\n"
+"-unsafe If x11vnc is running as root (e.g. inetd or Xsetup for\n"
+" a display manager) a few remote commands are disabled\n"
+" (currently: id:pick, accept:<cmd>, and gone:<cmd>)\n"
+" because they are associated with running external\n"
+" programs. If you specify -unsafe, then these remote\n"
+" control commands are allowed when running as root.\n"
+" When running as non-root all commands are allowed.\n"
+" See -safer below.\n"
+"-safer Even if not running as root, disable the above unsafe\n"
+" remote control commands.\n"
+"\n"
+"-deny_all For use with -remote nodeny: start out denying all\n"
+" incoming clients until \"-remote nodeny\" is used to\n"
+" let them in.\n"
"%s\n"
"\n"
"These options are passed to libvncserver:\n"
@@ -8951,9 +13709,12 @@ static void print_help(void) {
use_modifier_tweak ? "-modtweak":"-nomodtweak",
no_autorepeat ? "-norepeat":"-repeat",
cursor_pos_updates ? "-cursorpos":"-nocursorpos",
+ pointer_mode_max, pointer_mode,
+ ui_skip,
defer_update,
waitms,
take_naps ? "on":"off",
+ screen_blank,
use_threads ? "-threads":"-nothreads",
fs_frac,
gaps_fill,
@@ -8971,7 +13732,7 @@ static void print_help(void) {
*/
#define MAXN 256
-static char *this_host(void) {
+char *this_host(void) {
char host[MAXN];
#ifdef LIBVNCSERVER_HAVE_GETHOSTNAME
if (gethostname(host, MAXN) == 0) {
@@ -9001,7 +13762,7 @@ static char *choose_title(char *display) {
}
}
strncat(title, display, MAXN - strlen(title));
- if (subwin) {
+ if (subwin && valid_window(subwin)) {
char *name;
if (XFetchName(dpy, subwin, &name)) {
strncat(title, " ", MAXN - strlen(title));
@@ -9036,6 +13797,7 @@ static int limit_shm(void) {
return limit;
}
+
/*
* quick-n-dirty ~/.x11vncrc: each line (except # comments) is a cmdline option.
*/
@@ -9043,7 +13805,7 @@ static int argc2 = 0;
static char **argv2;
static void check_rcfile(int argc, char **argv) {
- int i, norc = 0, argmax = 1024;
+ int i, pwlast, norc = 0, argmax = 1024;
char *infile = NULL;
char rcfile[1024];
FILE *rc;
@@ -9062,10 +13824,12 @@ static void check_rcfile(int argc, char **argv) {
}
}
}
+ rc_norc = norc;
if (norc) {
;
} else if (infile != NULL) {
rc = fopen(infile, "r");
+ rc_rcfile = strdup(infile);
if (rc == NULL) {
fprintf(stderr, "could not open rcfile: %s\n", infile);
perror("fopen");
@@ -9078,6 +13842,7 @@ static void check_rcfile(int argc, char **argv) {
strcat(rcfile, "/.x11vncrc");
infile = rcfile;
rc = fopen(rcfile, "r");
+ rc_rcfile = strdup(rcfile);
if (rc == NULL) {
norc = 1;
}
@@ -9188,8 +13953,21 @@ static void check_rcfile(int argc, char **argv) {
fclose(rc);
free(buf);
}
+ pwlast = 0;
for (i=1; i < argc; i++) {
argv2[argc2++] = strdup(argv[i]);
+
+ if (pwlast || !strcmp("-passwd", argv[i])
+ || !strcmp("-viewpasswd", argv[i])) {
+ char *p = argv[i];
+ if (pwlast) {
+ pwlast = 0;
+ } else {
+ pwlast = 1;
+ }
+ while (*p != '\0')
+ *p++ = '\0';
+ }
if (argc2 >= argmax) {
fprintf(stderr, "too many rcfile options\n");
exit(1);
@@ -9199,28 +13977,47 @@ static void check_rcfile(int argc, char **argv) {
int main(int argc, char* argv[]) {
- XImage *fb;
- int i;
+ int i, len;
int ev, er, maj, min;
- char *use_dpy = NULL;
- char *auth_file = NULL;
- char *arg, *visual_str = NULL;
- char *logfile = NULL;
- char *passwdfile = NULL;
- char *blackout_string = NULL;
- char *remap_file = NULL;
- char *pointer_remap = NULL;
+ char *arg;
+ char *remote_cmd = NULL;
+ char *gui_string = NULL;
+ int remote_query = 0;
int pw_loc = -1;
int vpw_loc = -1;
- int dt = 0;
- int bg = 0;
- int got_rfbwait = 0;
- int got_deferupdate = 0, got_defer = 0;
+ int dt = 0, bg = 0;
+ int got_rfbwait = 0, got_deferupdate = 0, got_defer = 0;
/* used to pass args we do not know about to rfbGetScreen(): */
int argc_vnc = 1; char *argv_vnc[128];
+ /* if we are root limit some remote commands: */
+ if (!getuid() || !geteuid()) {
+ safe_remote_only = 1;
+ }
+
argv_vnc[0] = strdup(argv[0]);
+ program_name = strdup(argv[0]);
+
+ len = 0;
+ for (i=1; i < argc; i++) {
+ len += strlen(argv[i]) + 4 + 1;
+ }
+ program_cmdline = (char *)malloc(len+1);
+ program_cmdline[0] = '\0';
+ for (i=1; i < argc; i++) {
+ char *s = argv[i];
+ if (program_cmdline[0]) {
+ strcat(program_cmdline, " ");
+ }
+ if (*s == '-') {
+ strcat(program_cmdline, s);
+ } else {
+ strcat(program_cmdline, "{{");
+ strcat(program_cmdline, s);
+ strcat(program_cmdline, "}}");
+ }
+ }
check_rcfile(argc, argv);
/* kludge for the new array argv2 set in check_rcfile() */
@@ -9241,110 +14038,83 @@ int main(int argc, char* argv[]) {
if (!strcmp(arg, "-display")) {
CHECK_ARGC
- use_dpy = argv[++i];
- } else if (!strcmp(arg, "-id")) {
+ use_dpy = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-auth")) {
CHECK_ARGC
- if (sscanf(argv[++i], "0x%x", &subwin) != 1) {
- if (sscanf(argv[i], "%d", &subwin) != 1) {
- fprintf(stderr, "bad -id arg: %s\n",
- argv[i]);
- exit(1);
- }
- }
- } else if (!strcmp(arg, "-sid")) {
- rootshift = 1;
+ auth_file = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-id") || !strcmp(arg, "-sid")) {
CHECK_ARGC
- if (sscanf(argv[++i], "0x%x", &subwin) != 1) {
- if (sscanf(argv[i], "%d", &subwin) != 1) {
- fprintf(stderr, "bad -id arg: %s\n",
- argv[i]);
- exit(1);
- }
- }
- } else if (!strcmp(arg, "-scale")) {
- int m, n;
- char *p;
- double f;
- CHECK_ARGC
- if ( (p = strchr(argv[++i], ':')) != NULL) {
- /* options */
- if (strstr(p+1, "nb") != NULL) {
- scaling_noblend = 1;
- }
- if (strstr(p+1, "n4") != NULL) {
- scaling_nomult4 = 1;
- }
- if (strstr(p+1, "in") != NULL) {
- scaling_interpolate = 1;
- }
- if (strstr(p+1, "pad") != NULL) {
- scaling_pad = 1;
- }
- *p = '\0';
+ if (!strcmp(arg, "-sid")) {
+ rootshift = 1;
+ } else {
+ rootshift = 0;
}
- if (strchr(argv[i], '.') != NULL) {
- double test, diff, eps = 1.0e-7;
- if (sscanf(argv[i], "%lf", &f) != 1) {
- fprintf(stderr, "bad -scale arg: %s\n",
- argv[i]);
+ i++;
+ if (!strcmp("pick", argv[i])) {
+ if (safe_remote_only) {
+ fprintf(stderr, "unsafe: %s pick\n",
+ arg);
exit(1);
- }
- scale_fac = (double) f;
- /* look for common fractions from small ints: */
- for (n=2; n<=10; n++) {
- for (m=1; m<n; m++) {
- test = ((double) m)/ n;
- diff = scale_fac - test;
- if (-eps < diff && diff < eps) {
- scale_numer = m;
- scale_denom = n;
- break;
-
- }
- }
- if (scale_denom) {
- break;
- }
- }
- } else {
- if (sscanf(argv[i], "%d/%d", &m, &n) != 2) {
- fprintf(stderr, "bad -scale arg: %s\n",
- argv[i]);
+ } else if (! pick_windowid(&subwin)) {
+ fprintf(stderr, "bad %s pick\n", arg);
exit(1);
}
- scale_fac = ((double) m)/ n;
- scale_numer = m;
- scale_denom = n;
- }
- if (scale_fac == 1.0) {
- if (! quiet) {
- rfbLog("scaling disabled for factor "
- " %f\n", scale_fac);
- }
- } else {
- scaling = 1;
+ } else if (! scan_hexdec(argv[i], &subwin)) {
+ fprintf(stderr, "bad %s arg: %s\n", arg,
+ argv[i]);
+ exit(1);
}
- } else if (!strcmp(arg, "-visual")) {
- CHECK_ARGC
- visual_str = argv[++i];
+ } else if (!strcmp(arg, "-flashcmap")) {
+ flash_cmap = 1;
+ } else if (!strcmp(arg, "-notruecolor")) {
+ force_indexed_color = 1;
} else if (!strcmp(arg, "-overlay")) {
overlay = 1;
} else if (!strcmp(arg, "-overlay_nocursor")) {
overlay = 1;
overlay_cursor = 0;
- } else if (!strcmp(arg, "-flashcmap")) {
- flash_cmap = 1;
- } else if (!strcmp(arg, "-notruecolor")) {
- force_indexed_color = 1;
+ } else if (!strcmp(arg, "-overlay_yescursor")) {
+ overlay = 1;
+ overlay_cursor = 2;
+ } else if (!strcmp(arg, "-visual")) {
+ CHECK_ARGC
+ visual_str = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-scale")) {
+ CHECK_ARGC
+ scale_str = strdup(argv[++i]);
} else if (!strcmp(arg, "-viewonly")) {
view_only = 1;
+ } else if (!strcmp(arg, "-shared")) {
+ shared = 1;
+ } else if (!strcmp(arg, "-once")) {
+ connect_once = 1;
+ } else if (!strcmp(arg, "-many") || !strcmp(arg, "-forever")) {
+ connect_once = 0;
+ } else if (!strcmp(arg, "-inetd")) {
+ inetd = 1;
+ } else if (!strcmp(arg, "-connect")) {
+ CHECK_ARGC
+ if (strchr(argv[++i], '/')) {
+ client_connect_file = strdup(argv[i]);
+ } else {
+ client_connect = strdup(argv[i]);
+ }
+ } else if (!strcmp(arg, "-vncconnect")) {
+ vnc_connect = 1;
+ } else if (!strcmp(arg, "-novncconnect")) {
+ vnc_connect = 0;
+ } else if (!strcmp(arg, "-allow")) {
+ CHECK_ARGC
+ allow_list = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-localhost")) {
+ allow_list = strdup("127.0.0.1");
} else if (!strcmp(arg, "-viewpasswd")) {
vpw_loc = i;
CHECK_ARGC
viewonly_passwd = strdup(argv[++i]);
} else if (!strcmp(arg, "-passwdfile")) {
CHECK_ARGC
- passwdfile = argv[++i];
+ passwdfile = strdup(argv[++i]);
} else if (!strcmp(arg, "-storepasswd")) {
if (i+2 >= argc || rfbEncryptAndStorePasswd(argv[i+1],
argv[i+2]) != 0) {
@@ -9355,44 +14125,58 @@ int main(int argc, char* argv[]) {
argv[i+2]);
exit(0);
}
- } else if (!strcmp(arg, "-shared")) {
- shared = 1;
- } else if (!strcmp(arg, "-auth")) {
- CHECK_ARGC
- auth_file = argv[++i];
- } else if (!strcmp(arg, "-allow")) {
- CHECK_ARGC
- allow_list = argv[++i];
- } else if (!strcmp(arg, "-localhost")) {
- allow_list = "127.0.0.1";
} else if (!strcmp(arg, "-accept")) {
CHECK_ARGC
- accept_cmd = argv[++i];
+ accept_cmd = strdup(argv[++i]);
} else if (!strcmp(arg, "-gone")) {
CHECK_ARGC
- gone_cmd = argv[++i];
- } else if (!strcmp(arg, "-once")) {
- connect_once = 1;
- } else if (!strcmp(arg, "-many")
- || !strcmp(arg, "-forever")) {
- connect_once = 0;
- } else if (!strcmp(arg, "-connect")) {
- CHECK_ARGC
- if (strchr(argv[++i], '/')) {
- client_connect_file = argv[i];
- } else {
- client_connect = strdup(argv[i]);
- }
- } else if (!strcmp(arg, "-vncconnect")) {
- vnc_connect = 1;
- } else if (!strcmp(arg, "-novncconnect")) {
- vnc_connect = 0;
- } else if (!strcmp(arg, "-inetd")) {
- inetd = 1;
+ gone_cmd = strdup(argv[++i]);
} else if (!strcmp(arg, "-noshm")) {
using_shm = 0;
} else if (!strcmp(arg, "-flipbyteorder")) {
flip_byte_order = 1;
+ } else if (!strcmp(arg, "-onetile")) {
+ single_copytile = 1;
+ } else if (!strcmp(arg, "-blackout")) {
+ CHECK_ARGC
+ blackout_string = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-xinerama")) {
+ xinerama = 1;
+ } else if (!strcmp(arg, "-xrandr")) {
+ xrandr = 1;
+ if (i < argc-1) {
+ char *s = argv[i+1];
+ if (known_xrandr_mode(s)) {
+ xrandr_mode = strdup(s);
+ i++;
+ }
+ }
+ } else if (!strcmp(arg, "-padgeom")
+ || !strcmp(arg, "-padgeometry")) {
+ CHECK_ARGC
+ pad_geometry = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-o") || !strcmp(arg, "-logfile")) {
+ CHECK_ARGC
+ logfile = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-rc")) {
+ i++; /* done above */
+ } else if (!strcmp(arg, "-norc")) {
+ ; /* done above */
+ } else if (!strcmp(arg, "-h") || !strcmp(arg, "-help")
+ || !strcmp(arg, "-?")) {
+ print_help();
+ } else if (!strcmp(arg, "-V") || !strcmp(arg, "-version")) {
+ fprintf(stderr, "x11vnc: %s\n", lastmod);
+ exit(0);
+ } else if (!strcmp(arg, "-q") || !strcmp(arg, "-quiet")) {
+ quiet = 1;
+ } else if (!strcmp(arg, "-bg") || !strcmp(arg, "-background")) {
+#ifdef LIBVNCSERVER_HAVE_SETSID
+ bg = 1;
+ opts_bg = bg;
+#else
+ fprintf(stderr, "warning: -bg mode not supported.\n");
+#endif
} else if (!strcmp(arg, "-modtweak")) {
use_modifier_tweak = 1;
} else if (!strcmp(arg, "-nomodtweak")) {
@@ -9401,11 +14185,11 @@ int main(int argc, char* argv[]) {
use_iso_level3 = 1;
} else if (!strcmp(arg, "-xkb")) {
use_xkb_modtweak = 1;
- } else if (!strcmp(arg, "-skip_keycodes")) {
- CHECK_ARGC
- skip_keycodes = argv[++i];
} else if (!strcmp(arg, "-xkbcompat")) {
xkbcompat = 1;
+ } else if (!strcmp(arg, "-skip_keycodes")) {
+ CHECK_ARGC
+ skip_keycodes = strdup(argv[++i]);
} else if (!strcmp(arg, "-add_keysyms")) {
add_keysyms++;
} else if (!strcmp(arg, "-clear_mods")) {
@@ -9414,65 +14198,68 @@ int main(int argc, char* argv[]) {
clear_mods = 2;
} else if (!strcmp(arg, "-remap")) {
CHECK_ARGC
- remap_file = argv[++i];
- } else if (!strcmp(arg, "-blackout")) {
- CHECK_ARGC
- blackout_string = argv[++i];
- } else if (!strcmp(arg, "-xinerama")) {
- xinerama = 1;
- } else if (!strcmp(arg, "-norc")) {
- ; /* done above */
- } else if (!strcmp(arg, "-rc")) {
- i++; /* done above */
- } else if (!strcmp(arg, "-nobell")) {
- watch_bell = 0;
+ remap_file = strdup(argv[++i]);
+ } else if (!strcmp(arg, "-norepeat")) {
+ no_autorepeat = 1;
+ } else if (!strcmp(arg, "-repeat")) {
+ no_autorepeat = 0;
} else if (!strcmp(arg, "-nofb")) {
nofb = 1;
+ } else if (!strcmp(arg, "-nobell")) {
+ watch_bell = 0;
} else if (!strcmp(arg, "-nosel")) {
watch_selection = 0;
} else if (!strcmp(arg, "-noprimary")) {
watch_primary = 0;
} else if (!strcmp(arg, "-cursor")) {
show_cursor = 1;
- if (i >= argc-1) {
- ;
- } else {
+ if (i < argc-1) {
char *s = argv[i+1];
- if (*s == 'X' || !strcmp(s, "default") ||
- !strcmp(s, "some") || !strcmp(s, "most")) {
+ if (known_cursors_mode(s)) {
multiple_cursors_mode = strdup(s);
i++;
- }
+ if (!strcmp(s, "none")) {
+ show_cursor = 0;
+ }
+ }
}
} else if (!strcmp(arg, "-nocursor")) {
+ multiple_cursors_mode = strdup("none");
show_cursor = 0;
+ } else if (!strcmp(arg, "-noxfixes")) {
+ use_xfixes = 0;
+ } else if (!strcmp(arg, "-nocursorshape")) {
+ cursor_shape_updates = 0;
} else if (!strcmp(arg, "-cursorpos")) {
cursor_pos_updates = 1;
} else if (!strcmp(arg, "-nocursorpos")) {
cursor_pos_updates = 0;
- } else if (!strcmp(arg, "-nocursorshape")) {
- cursor_shape_updates = 0;
} else if (!strcmp(arg, "-xwarppointer")) {
use_xwarppointer = 1;
} else if (!strcmp(arg, "-buttonmap")) {
CHECK_ARGC
- pointer_remap = argv[++i];
+ pointer_remap = strdup(argv[++i]);
} else if (!strcmp(arg, "-nodragging")) {
show_dragging = 0;
+ } else if (!strcmp(arg, "-pointer_mode")) {
+ char *p, *s;
+ CHECK_ARGC
+ s = argv[++i];
+ if ((p = strchr(s, ':')) != NULL) {
+ ui_skip = atoi(p+1);
+ if (! ui_skip) ui_skip = 1;
+ *p = '\0';
+ }
+ if (atoi(s) < 1 || atoi(s) > pointer_mode_max) {
+ rfbLog("pointer_mode out of range 1-%d: %d\n",
+ pointer_mode_max, atoi(s));
+ } else {
+ pointer_mode = atoi(s);
+ }
} else if (!strcmp(arg, "-input_skip")) {
CHECK_ARGC
ui_skip = atoi(argv[++i]);
if (! ui_skip) ui_skip = 1;
- } else if (!strcmp(arg, "-old_pointer")) {
- old_pointer = 1;
- } else if (!strcmp(arg, "-old_pointer2")) {
- old_pointer = 2;
- } else if (!strcmp(arg, "-norepeat")) {
- no_autorepeat = 1;
- } else if (!strcmp(arg, "-repeat")) {
- no_autorepeat = 0;
- } else if (!strcmp(arg, "-onetile")) {
- single_copytile = 1;
} else if (!strcmp(arg, "-debug_pointer")
|| !strcmp(arg, "-dp")) {
debug_pointer++;
@@ -9488,25 +14275,24 @@ int main(int argc, char* argv[]) {
waitms = atoi(argv[++i]);
} else if (!strcmp(arg, "-nap")) {
take_naps = 1;
-#ifdef LIBVNCSERVER_HAVE_LIBPTHREAD
- } else if (!strcmp(arg, "-threads")) {
- use_threads = 1;
- } else if (!strcmp(arg, "-nothreads")) {
- use_threads = 0;
-#endif
+ } else if (!strcmp(arg, "-sb")) {
+ CHECK_ARGC
+ screen_blank = atoi(argv[++i]);
} else if (!strcmp(arg, "-sigpipe")) {
CHECK_ARGC
- if (!strcmp(argv[++i], "ignore")) {
- sigpipe = 1;
- } else if (!strcmp(argv[i], "exit")) {
- sigpipe = 2;
- } else if (!strcmp(argv[i], "skip")) {
- sigpipe = 0;
+ if (known_sigpipe_mode(argv[++i])) {
+ sigpipe = strdup(argv[i]);
} else {
fprintf(stderr, "bad -sigpipe arg: %s, must "
"be \"ignore\" or \"exit\"\n", argv[i]);
exit(1);
}
+#ifdef LIBVNCSERVER_HAVE_LIBPTHREAD
+ } else if (!strcmp(arg, "-threads")) {
+ use_threads = 1;
+ } else if (!strcmp(arg, "-nothreads")) {
+ use_threads = 0;
+#endif
} else if (!strcmp(arg, "-fs")) {
CHECK_ARGC
fs_frac = atof(argv[++i]);
@@ -9519,24 +14305,42 @@ int main(int argc, char* argv[]) {
} else if (!strcmp(arg, "-fuzz")) {
CHECK_ARGC
tile_fuzz = atoi(argv[++i]);
- } else if (!strcmp(arg, "-h") || !strcmp(arg, "-help")
- || !strcmp(arg, "-?")) {
- print_help();
- } else if (!strcmp(arg, "-V") || !strcmp(arg, "-version")) {
- fprintf(stderr, "x11vnc: %s\n", lastmod);
- exit(0);
- } else if (!strcmp(arg, "-o") || !strcmp(arg, "-logfile")) {
+ } else if (!strcmp(arg, "-gui")) {
+ launch_gui = 1;
+ if (i < argc-1) {
+ char *s = argv[i+1];
+ if (*s != '-') {
+ gui_string = strdup(s);
+ i++;
+ }
+ }
+ } else if (!strcmp(arg, "-remote") || !strcmp(arg, "-R")
+ || !strcmp(arg, "-r")) {
CHECK_ARGC
- logfile = argv[++i];
- } else if (!strcmp(arg, "-q") || !strcmp(arg, "-quiet")) {
+ remote_cmd = strdup(argv[++i]);
+ if (!strcmp(remote_cmd, "ping")) {
+ remote_query = 1;
+ }
quiet = 1;
-#ifdef LIBVNCSERVER_HAVE_SETSID
- } else if (!strcmp(arg, "-bg") || !strcmp(arg, "-background")) {
- bg = 1;
-#endif
+ xkbcompat = 0;
+ } else if (!strcmp(arg, "-query") || !strcmp(arg, "-Q")) {
+ CHECK_ARGC
+ remote_cmd = strdup(argv[++i]);
+ remote_query = 1;
+ quiet = 1;
+ xkbcompat = 0;
+ } else if (!strcmp(arg, "-noremote")) {
+ accept_remote_cmds = 0;
+ } else if (!strcmp(arg, "-unsafe")) {
+ safe_remote_only = 0;
+ } else if (!strcmp(arg, "-safer")) {
+ safe_remote_only = 1;
+ } else if (!strcmp(arg, "-deny_all")) {
+ deny_all = 1;
} else {
- if (!strcmp(arg, "-desktop")) {
+ if (!strcmp(arg, "-desktop") && i < argc-1) {
dt = 1;
+ rfb_desktop_name = strdup(argv[i+1]);
}
if (!strcmp(arg, "-passwd")) {
pw_loc = i;
@@ -9562,6 +14366,10 @@ int main(int argc, char* argv[]) {
}
}
}
+
+ if (launch_gui) {
+ do_gui(gui_string);
+ }
if (logfile) {
int n;
if ((n = open(logfile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) < 0) {
@@ -9588,20 +14396,27 @@ int main(int argc, char* argv[]) {
}
}
+ if (client_connect_file && remote_cmd) {
+ /* no need to open DISPLAY, just write it to the file now */
+ int rc = send_remote_cmd(remote_cmd, remote_query);
+ fflush(stderr);
+ fflush(stdout);
+ exit(rc);
+ }
+
/*
* If -passwd was used, clear it out of argv. This does not
* work on all UNIX, have to use execvp() in general...
*/
if (pw_loc > 0) {
- char *p = argv[pw_loc];
- while (*p != '\0') {
- *p++ = '\0';
- }
- if (pw_loc+1 < argc) {
- p = argv[pw_loc+1];
- while (*p != '\0') {
- *p++ = '\0';
+ int i;
+ for (i=pw_loc; i <= pw_loc+1; i++) {
+ if (i < argc) {
+ char *p = argv[i];
+ while (*p != '\0') {
+ *p++ = '\0';
+ }
}
}
} else if (passwdfile) {
@@ -9611,7 +14426,7 @@ int main(int argc, char* argv[]) {
in = fopen(passwdfile, "r");
if (in == NULL) {
rfbLog("cannot open passwdfile: %s\n", passwdfile);
- rfbLog("fopen");
+ rfbLogPerror("fopen");
exit(1);
}
if (fgets(line, 1024, in) != NULL) {
@@ -9619,7 +14434,7 @@ int main(int argc, char* argv[]) {
if (len > 0 && line[len-1] == '\n') {
line[len-1] = '\0';
}
- argv_vnc[argc_vnc++] = "-passwd";
+ argv_vnc[argc_vnc++] = strdup("-passwd");
argv_vnc[argc_vnc++] = strdup(line);
pw_loc = 100; /* just for pw_loc check below */
if (fgets(line, 1024, in) != NULL) {
@@ -9656,14 +14471,13 @@ int main(int argc, char* argv[]) {
fclose(in);
}
if (vpw_loc > 0) {
- char *p = argv[vpw_loc];
- while (*p != '\0') {
- *p++ = '\0';
- }
- if (vpw_loc+1 < argc) {
- p = argv[vpw_loc+1];
- while (*p != '\0') {
- *p++ = '\0';
+ int i;
+ for (i=vpw_loc; i <= vpw_loc+1; i++) {
+ if (i < argc) {
+ char *p = argv[i];
+ while (*p != '\0') {
+ *p++ = '\0';
+ }
}
}
}
@@ -9698,77 +14512,20 @@ int main(int argc, char* argv[]) {
/* increase rfbwait if threaded */
if (use_threads && ! got_rfbwait) {
- argv_vnc[argc_vnc++] = "-rfbwait";
- argv_vnc[argc_vnc++] = "604800000"; /* one week... */
- }
-
- /* cursor shapes setup */
-
-#ifdef SOLARIS
- if (show_cursor && ! overlay && overlay_cursor &&
- !strcmp(multiple_cursors_mode, "most")) {
- overlay = 1;
- if (! quiet) {
- rfbLog("enabling -overlay mode to achieve "
- "'-cursor most'\n");
- rfbLog("disable with: -overlay_nocursor.\n");
- }
- }
-#endif
-
- if (overlay) {
-#ifdef SOLARIS
- using_shm = 0;
-
- if (flash_cmap && ! quiet) {
- rfbLog("warning: -flashcmap may be incompatible "
- "with -overlay\n");
- }
-
- if (show_cursor && overlay_cursor) {
- char *s = multiple_cursors_mode;
- if (*s == 'X' || !strcmp(s, "some")) {
- /*
- * user wants these modes, so disable fb cursor
- */
- overlay_cursor = 0;
- } else {
- /*
- * "default" and "most", we turn off
- * show_cursor since it will automatically
- * be in the framebuffer.
- */
- show_cursor = 0;
- }
- }
-#else
- if (! quiet) {
- rfbLog("disabling -overlay: only available on "
- "Solaris Xsun.\n");
- }
- overlay = 0;
-#endif
- }
-
- if (show_cursor) {
- char *s = multiple_cursors_mode;
- if (*s == 'X' || !strcmp(s, "some")) {
- show_multiple_cursors = 1;
- } else if (!strcmp(s, "most")) {
- /* later check for XFIXES, for now "some" */
- show_multiple_cursors = 1;
- }
+ argv_vnc[argc_vnc++] = strdup("-rfbwait");
+ argv_vnc[argc_vnc++] = strdup("604800000"); /* one week... */
}
/* no framebuffer (Win2VNC) mode */
if (nofb) {
- /* disable things that do not make sense */
+ /* disable things that do not make sense with no fb */
using_shm = 0;
flash_cmap = 0;
show_cursor = 0;
show_multiple_cursors = 0;
overlay = 0;
+ overlay_cursor = 0;
if (! quiet) {
rfbLog("disabling -cursor, fb, shm, etc. in "
"-nofb mode.\n");
@@ -9780,18 +14537,11 @@ int main(int argc, char* argv[]) {
}
}
- /* check for OS with small shm limits */
- if (using_shm && ! single_copytile) {
- if (limit_shm()) {
- single_copytile = 1;
- }
- }
-
if (! got_deferupdate) {
char tmp[40];
/* XXX not working yet in libvncserver */
sprintf(tmp, "%d", defer_update);
- argv_vnc[argc_vnc++] = "-deferupdate";
+ argv_vnc[argc_vnc++] = strdup("-deferupdate");
argv_vnc[argc_vnc++] = strdup(tmp);
}
@@ -9809,24 +14559,26 @@ int main(int argc, char* argv[]) {
fprintf(stderr, "Settings:\n");
fprintf(stderr, " display: %s\n", use_dpy ? use_dpy
: "null");
- fprintf(stderr, " subwin: 0x%x\n", subwin);
+ fprintf(stderr, " authfile: %s\n", auth_file ? auth_file
+ : "null");
+ fprintf(stderr, " subwin: 0x%lx\n", subwin);
+ fprintf(stderr, " rootshift: %d\n", rootshift);
fprintf(stderr, " flashcmap: %d\n", flash_cmap);
fprintf(stderr, " force_idx: %d\n", force_indexed_color);
- fprintf(stderr, " scaling: %d %.5f\n", scaling, scale_fac);
- fprintf(stderr, " visual: %s\n", visual_str ? visual_str
- : "null");
fprintf(stderr, " overlay: %d\n", overlay);
fprintf(stderr, " ovl_cursor: %d\n", overlay_cursor);
+ fprintf(stderr, " visual: %s\n", visual_str ? visual_str
+ : "null");
+ fprintf(stderr, " scaling: %d %.5f\n", scaling, scale_fac);
fprintf(stderr, " viewonly: %d\n", view_only);
fprintf(stderr, " shared: %d\n", shared);
fprintf(stderr, " conn_once: %d\n", connect_once);
+ fprintf(stderr, " inetd: %d\n", inetd);
fprintf(stderr, " connect: %s\n", client_connect
? client_connect : "null");
fprintf(stderr, " connectfile %s\n", client_connect_file
? client_connect_file : "null");
fprintf(stderr, " vnc_conn: %d\n", vnc_connect);
- fprintf(stderr, " authfile: %s\n", auth_file ? auth_file
- : "null");
fprintf(stderr, " allow: %s\n", allow_list ? allow_list
: "null");
fprintf(stderr, " passfile: %s\n", passwdfile ? passwdfile
@@ -9835,14 +14587,20 @@ int main(int argc, char* argv[]) {
: "null");
fprintf(stderr, " gone: %s\n", gone_cmd ? gone_cmd
: "null");
- fprintf(stderr, " inetd: %d\n", inetd);
fprintf(stderr, " using_shm: %d\n", using_shm);
fprintf(stderr, " flipbytes: %d\n", flip_byte_order);
+ fprintf(stderr, " onetile: %d\n", single_copytile);
fprintf(stderr, " blackout: %s\n", blackout_string
? blackout_string : "null");
fprintf(stderr, " xinerama: %d\n", xinerama);
+ fprintf(stderr, " xrandr: %d\n", xrandr);
+ fprintf(stderr, " xrandrmode: %s\n", xrandr_mode ? xrandr_mode
+ : "null");
fprintf(stderr, " logfile: %s\n", logfile ? logfile
: "null");
+ fprintf(stderr, " rc_file: %s\n", rc_rcfile ? rc_rcfile
+ : "null");
+ fprintf(stderr, " norc: %d\n", rc_norc);
fprintf(stderr, " bg: %d\n", bg);
fprintf(stderr, " mod_tweak: %d\n", use_modifier_tweak);
fprintf(stderr, " isolevel3: %d\n", use_iso_level3);
@@ -9854,6 +14612,7 @@ int main(int argc, char* argv[]) {
fprintf(stderr, " clearmods: %d\n", clear_mods);
fprintf(stderr, " remap: %s\n", remap_file ? remap_file
: "null");
+ fprintf(stderr, " norepeat: %d\n", no_autorepeat);
fprintf(stderr, " nofb: %d\n", nofb);
fprintf(stderr, " watchbell: %d\n", watch_bell);
fprintf(stderr, " watchsel: %d\n", watch_selection);
@@ -9862,27 +14621,30 @@ int main(int argc, char* argv[]) {
fprintf(stderr, " root_curs: %d\n", show_multiple_cursors);
fprintf(stderr, " curs_mode: %s\n", multiple_cursors_mode
? multiple_cursors_mode : "null");
- fprintf(stderr, " xwarpptr: %d\n", use_xwarppointer);
- fprintf(stderr, " cursorpos: %d\n", cursor_pos_updates);
+ fprintf(stderr, " xfixes: %d\n", use_xfixes);
fprintf(stderr, " cursorshp: %d\n", cursor_shape_updates);
+ fprintf(stderr, " cursorpos: %d\n", cursor_pos_updates);
+ fprintf(stderr, " xwarpptr: %d\n", use_xwarppointer);
fprintf(stderr, " buttonmap: %s\n", pointer_remap
? pointer_remap : "null");
fprintf(stderr, " dragging: %d\n", show_dragging);
- fprintf(stderr, " old_ptr: %d\n", old_pointer);
+ fprintf(stderr, " ptr_mode: %d\n", pointer_mode);
fprintf(stderr, " inputskip: %d\n", ui_skip);
- fprintf(stderr, " norepeat: %d\n", no_autorepeat);
fprintf(stderr, " debug_ptr: %d\n", debug_pointer);
fprintf(stderr, " debug_key: %d\n", debug_keyboard);
fprintf(stderr, " defer: %d\n", defer_update);
fprintf(stderr, " waitms: %d\n", waitms);
fprintf(stderr, " take_naps: %d\n", take_naps);
- fprintf(stderr, " sigpipe: %d\n", sigpipe);
+ fprintf(stderr, " sb: %d\n", screen_blank);
+ fprintf(stderr, " sigpipe: %s\n", sigpipe
+ ? sigpipe : "null");
fprintf(stderr, " threads: %d\n", use_threads);
fprintf(stderr, " fs_frac: %.2f\n", fs_frac);
- fprintf(stderr, " onetile: %d\n", single_copytile);
fprintf(stderr, " gaps_fill: %d\n", gaps_fill);
fprintf(stderr, " grow_fill: %d\n", grow_fill);
fprintf(stderr, " tile_fuzz: %d\n", tile_fuzz);
+ fprintf(stderr, " deny_all: %d\n", deny_all);
+ fprintf(stderr, " noremote: %d\n", !accept_remote_cmds);
fprintf(stderr, "\n");
rfbLog("x11vnc version: %s\n", lastmod);
} else {
@@ -9946,21 +14708,140 @@ int main(int argc, char* argv[]) {
scr = DefaultScreen(dpy);
rootwin = RootWindow(dpy, scr);
+ if (remote_cmd) {
+ int rc = send_remote_cmd(remote_cmd, remote_query);
+ XFlush(dpy);
+ fflush(stderr);
+ usleep(30 * 1000); /* still needed? */
+ XCloseDisplay(dpy);
+ exit(rc);
+ }
+
if (! dt) {
static char str[] = "-desktop";
argv_vnc[argc_vnc++] = str;
argv_vnc[argc_vnc++] = choose_title(use_dpy);
+ rfb_desktop_name = strdup(argv_vnc[argc_vnc-1]);
+ }
+
+#ifdef LIBVNCSERVER_HAVE_LIBXFIXES
+ if (! XFixesQueryExtension(dpy, &xfixes_base_event_type, &er)) {
+ if (! quiet) {
+ rfbLog("disabling xfixes mode: display does not "
+ "support it.\n");
+ }
+ xfixes_present = 0;
+ } else {
+ xfixes_present = 1;
+ }
+#endif
+
+#ifdef LIBVNCSERVER_HAVE_LIBXDAMAGE
+ if (! XDamageQueryExtension(dpy, &xdamage_base_event_type, &er)) {
+ if (0 && ! quiet) {
+ rfbLog("disabling xdamage mode: display does not "
+ "support it.\n");
+ }
+ xdamage_present = 0;
+ } else {
+ xdamage_present = 1;
+ }
+#endif
+
+ overlay_present = 0;
+#ifdef SOLARIS_OVERLAY
+ if (! XQueryExtension(dpy, "SUN_OVL", &maj, &ev, &er)) {
+ if (! quiet && overlay) {
+ rfbLog("disabling -overlay: SUN_OVL "
+ "extension not available.\n");
+ }
+ } else {
+ overlay_present = 1;
+ }
+#endif
+#ifdef IRIX_OVERLAY
+ if (! XReadDisplayQueryExtension(dpy, &ev, &er)) {
+ if (! quiet && overlay) {
+ rfbLog("disabling -overlay: IRIX ReadDisplay "
+ "extension not available.\n");
+ }
+ } else {
+ overlay_present = 1;
+ }
+#endif
+ if (overlay && !overlay_present) {
+ overlay = 0;
+ overlay_cursor = 0;
+ }
+
+ /* cursor shapes setup */
+ if (! multiple_cursors_mode) {
+ multiple_cursors_mode = strdup("default");
+ }
+ if (show_cursor) {
+ if(!strcmp(multiple_cursors_mode, "default")
+ && xfixes_present && use_xfixes) {
+ free(multiple_cursors_mode);
+ multiple_cursors_mode = strdup("most");
+
+ if (! quiet) {
+ rfbLog("XFIXES available on display, resetting"
+ " cursor mode\n");
+ rfbLog(" to: '-cursor most'.\n");
+ rfbLog(" to disable this behavior use: "
+ "'-cursor arrow'.\n");
+ }
+ }
+ if(!strcmp(multiple_cursors_mode, "most")) {
+ if (xfixes_present && use_xfixes &&
+ overlay_cursor == 1) {
+ if (! quiet) {
+ rfbLog("using XFIXES for cursor "
+ "drawing.\n");
+ }
+ overlay_cursor = 0;
+ }
+ }
+ }
+
+ if (overlay) {
+ using_shm = 0;
+ if (flash_cmap && ! quiet) {
+ rfbLog("warning: -flashcmap may be "
+ "incompatible with -overlay\n");
+ }
+ if (show_cursor && overlay_cursor) {
+ char *s = multiple_cursors_mode;
+ if (*s == 'X' || !strcmp(s, "some") ||
+ !strcmp(s, "arrow")) {
+ /*
+ * user wants these modes, so disable fb cursor
+ */
+ overlay_cursor = 0;
+ } else {
+ /*
+ * "default" and "most", we turn off
+ * show_cursor since it will automatically
+ * be in the framebuffer.
+ */
+ show_cursor = 0;
+ }
+ }
}
+ initialize_cursors_mode();
+
/* check for XTEST */
if (! XTestQueryExtension_wr(dpy, &ev, &er, &maj, &min)) {
+ if (! quiet) {
rfbLog("warning: XTest extension not available, most user"
" input\n");
rfbLog("(pointer and keyboard) will be discarded.\n");
- xtest_present = 0;
rfbLog("no XTest extension, switching to -xwarppointer mode\n");
rfbLog("for pointer motion input.\n");
- use_xwarppointer = 1;
+ }
+ xtest_present = 0;
+ use_xwarppointer = 1;
}
/*
* Window managers will often grab the display during resize, etc.
@@ -9969,14 +14850,25 @@ int main(int argc, char* argv[]) {
*/
XTestGrabControl_wr(dpy, True);
+ /* check for OS with small shm limits */
+ if (using_shm && ! single_copytile) {
+ if (limit_shm()) {
+ single_copytile = 1;
+ }
+ }
+
+ single_copytile_orig = single_copytile;
+
/* check for MIT-SHM */
if (! nofb && ! XShmQueryExtension_wr(dpy)) {
+ xshm_present = 0;
if (! using_shm) {
if (! quiet) {
rfbLog("info: display does not support"
" XShm.\n");
}
} else {
+ if (! quiet) {
rfbLog("warning: XShm extension is not available.\n");
rfbLog("For best performance the X Display should be"
" local. (i.e.\n");
@@ -9985,46 +14877,16 @@ int main(int argc, char* argv[]) {
rfbLog("the same machine.)\n");
#ifdef LIBVNCSERVER_HAVE_XSHM
rfbLog("Restart with -noshm to override this.\n");
- exit(1);
+ }
+ exit(1);
#else
rfbLog("Switching to -noshm mode.\n");
- using_shm = 0;
+ }
+ using_shm = 0;
#endif
}
}
- if (overlay) {
- /*
- * ideally we'd like to not have to cook up the visual variables
- * but rather let it all come out of XReadScreen(), however
- * there is no way to get a default visual out of it, so we
- * pretend -visual TrueColor:NN was supplied with NN usually 24.
- */
-#ifdef SOLARIS
- char str[16];
- XImage *xi;
- Window twin = subwin ? subwin : rootwin;
-
- xi = XReadScreen(dpy, twin, 0, 0, 8, 8, False);
-
- sprintf(str, "TrueColor:%d", xi->depth);
- if (xi->depth != 24 && ! quiet) {
- fprintf(stderr, "warning XReadScreen() image has "
- "depth %d instead of 24.\n", xi->depth);
- }
- XDestroyImage(xi);
- if (visual_str != NULL && ! quiet) {
- fprintf(stderr, "warning: replacing '-visual %s' by "
- "'%s' for use with -overlay\n", visual_str, str);
- }
- visual_str = strdup(str);
-#endif
- }
-
- if (visual_str != NULL) {
- set_visual(visual_str);
- }
-
#ifdef LIBVNCSERVER_HAVE_XKEYBOARD
/* check for XKEYBOARD */
if (use_xkb) {
@@ -10033,163 +14895,51 @@ int main(int argc, char* argv[]) {
initialize_watch_bell();
if (!use_xkb && use_xkb_modtweak) {
if (! quiet) {
- fprintf(stderr, "warning: disabling xkb modtweak."
+ rfbLog("warning: disabling xkb modtweak."
" XKEYBOARD ext. not present.\n");
}
use_xkb_modtweak = 0;
}
#endif
- /* set up parameters for subwin or non-subwin cases: */
- if (! subwin) {
- window = rootwin;
- dpy_x = DisplayWidth(dpy, scr);
- dpy_y = DisplayHeight(dpy, scr);
- off_x = 0;
- off_y = 0;
- /* this may be overridden via visual_id below */
- default_visual = DefaultVisual(dpy, scr);
- } else {
- /* experiment to share just one window */
- XWindowAttributes attr;
-
- window = (Window) subwin;
- if ( ! XGetWindowAttributes(dpy, window, &attr) ) {
- fprintf(stderr, "bad window: 0x%lx\n", window);
- exit(1);
+#ifdef LIBVNCSERVER_HAVE_LIBXRANDR
+ if (! XRRQueryExtension(dpy, &xrandr_base_event_type, &er)) {
+ if (xrandr && ! quiet) {
+ rfbLog("disabling -xrandr mode: display does not"
+ " support it.\n");
}
- dpy_x = attr.width;
- dpy_y = attr.height;
-
- /* this may be overridden via visual_id below */
- default_visual = attr.visual;
-
- if (show_multiple_cursors) {
- show_multiple_cursors = 0;
- if (! quiet) {
- fprintf(stderr, "disabling root cursor drawing"
- " for subwindow\n");
- }
- }
- set_offset();
- }
-
- /* initialize depth to reasonable value, visual_id may override */
- depth = DefaultDepth(dpy, scr);
-
- if (visual_id) {
- int n;
- XVisualInfo vinfo_tmpl, *vinfo;
-
- /*
- * we are in here from -visual or -overlay options
- * visual_id and visual_depth were set in set_visual().
- */
-
- vinfo_tmpl.visualid = visual_id;
- vinfo = XGetVisualInfo(dpy, VisualIDMask, &vinfo_tmpl, &n);
- if (vinfo == NULL || n == 0) {
- fprintf(stderr, "could not match visual_id: 0x%x\n",
- (int) visual_id);
- exit(1);
- }
- default_visual = vinfo->visual;
- depth = vinfo->depth;
- if (visual_depth) {
- /* force it from -visual FooColor:NN */
- depth = visual_depth;
- }
- if (! quiet) {
- fprintf(stderr, "vis id: 0x%x\n",
- (int) vinfo->visualid);
- fprintf(stderr, "vis scr: %d\n", vinfo->screen);
- fprintf(stderr, "vis depth %d\n", vinfo->depth);
- fprintf(stderr, "vis class %d\n", vinfo->class);
- fprintf(stderr, "vis rmask 0x%lx\n", vinfo->red_mask);
- fprintf(stderr, "vis gmask 0x%lx\n", vinfo->green_mask);
- fprintf(stderr, "vis bmask 0x%lx\n", vinfo->blue_mask);
- fprintf(stderr, "vis cmap_sz %d\n", vinfo->colormap_size);
- fprintf(stderr, "vis b/rgb %d\n", vinfo->bits_per_rgb);
- }
-
- XFree(vinfo);
- }
-
- if (! quiet) {
- rfbLog("default visual ID: 0x%x\n",
- (int) XVisualIDFromVisual(default_visual));
+ xrandr = 0;
+ xrandr_present = 0;
+ } else {
+ xrandr_present = 1;
}
+#endif
- if (nofb) {
- /*
- * For -nofb we do not allocate the framebuffer, so we
- * can save a few MB of memory.
- */
- fb = XCreateImage_wr(dpy, default_visual, depth, ZPixmap,
- 0, NULL, dpy_x, dpy_y, BitmapPad(dpy), 0);
+ /*
+ * Create the XImage corresponding to the display framebuffer.
+ */
- } else if (visual_id) {
- /*
- * we need to call XCreateImage to supply the visual
- */
- fb = XCreateImage_wr(dpy, default_visual, depth, ZPixmap,
- 0, NULL, dpy_x, dpy_y, BitmapPad(dpy), 0);
- fb->data = (char *) malloc(fb->bytes_per_line * fb->height);
- } else {
- fb = XGetImage_wr(dpy, window, 0, 0, dpy_x, dpy_y, AllPlanes,
- ZPixmap);
- if (! quiet) {
- rfbLog("Read initial data from X display into"
- " framebuffer.\n");
- }
- }
- if (fb->bits_per_pixel == 24 && ! quiet) {
- fprintf(stderr, "warning: 24 bpp may have poor"
- " performance.\n");
- }
+ fb0 = initialize_xdisplay_fb();
/*
* n.b. we do not have to X_LOCK any X11 calls until watch_loop()
* is called since we are single-threaded until then.
*/
- initialize_screen(&argc_vnc, argv_vnc, fb);
+ initialize_screen(&argc_vnc, argv_vnc, fb0);
initialize_tiles();
/* rectangular blackout regions */
- if (blackout_string != NULL) {
- initialize_blackout(blackout_string);
- }
- if (xinerama) {
- initialize_xinerama();
- }
- if (blackouts) {
- blackout_tiles();
- }
+ initialize_blackouts_and_xinerama();
- initialize_shm(); /* also creates XImages when using_shm = 0 */
+ /* created shm or XImages when using_shm = 0 */
+ initialize_polling_images();
initialize_signals();
- if (blackouts) { /* blackout fb as needed. */
- copy_screen();
- }
-
- if (use_modifier_tweak) {
- initialize_modtweak();
- }
- if (remap_file != NULL) {
- initialize_remap(remap_file);
- }
-
- initialize_pointer_map(pointer_remap);
-
- clear_modifiers(1);
- if (clear_mods == 1) {
- clear_modifiers(0);
- }
+ initialize_keyboard_and_pointer();
if (! inetd) {
if (! screen->port || screen->listenSock < 0) {
@@ -10200,6 +14950,7 @@ int main(int argc, char* argv[]) {
if (! quiet) {
rfbLog("screen setup finished.\n");
}
+ sprintf(vnc_desktop_name, "unknown");
if (screen->port) {
char *host = this_host();
int lport = screen->port;
@@ -10209,22 +14960,31 @@ int main(int argc, char* argv[]) {
; /* should not occur (port) */
} else if (quiet) {
if (lport >= 5900) {
+ sprintf(vnc_desktop_name, "%s:%d",
+ host, lport - 5900);
fprintf(stderr, "The VNC desktop is "
- "%s:%d\n", host, lport - 5900);
+ "%s\n", vnc_desktop_name);
} else {
+ sprintf(vnc_desktop_name, "%s:%d",
+ host, lport);
fprintf(stderr, "The VNC desktop is "
- "%s:%d\n", host, lport);
+ "%s\n", vnc_desktop_name);
}
} else if (lport >= 5900) {
- rfbLog("The VNC desktop is %s:%d\n", host,
- lport - 5900);
+ sprintf(vnc_desktop_name, "%s:%d",
+ host, lport - 5900);
+ rfbLog("\n");
+ rfbLog("The VNC desktop is %s\n",
+ vnc_desktop_name);
if (lport >= 6000) {
rfbLog("possible aliases: %s:%d, "
"%s::%d\n", host, lport, host, lport);
}
} else {
- rfbLog("The VNC desktop is %s:%d\n", host,
- lport);
+ sprintf(vnc_desktop_name, "%s:%d", host, lport);
+ rfbLog("\n");
+ rfbLog("The VNC desktop is %s\n",
+ vnc_desktop_name);
rfbLog("possible alias: %s::%d\n",
host, lport);
}