summaryrefslogtreecommitdiffstats
path: root/contrib/x11vnc.c
diff options
context:
space:
mode:
Diffstat (limited to 'contrib/x11vnc.c')
-rw-r--r--contrib/x11vnc.c1065
1 files changed, 805 insertions, 260 deletions
diff --git a/contrib/x11vnc.c b/contrib/x11vnc.c
index c094147..06505b4 100644
--- a/contrib/x11vnc.c
+++ b/contrib/x11vnc.c
@@ -108,11 +108,24 @@
#include <fcntl.h>
#include <rfb/rfb.h>
+#include <rfb/rfbregion.h>
#ifdef LIBVNCSERVER_HAVE_XKEYBOARD
#include <X11/XKBlib.h>
#endif
+/*
+ * Temporary kludge: to run with -xinerama define the following macro
+ * and be sure to link with * -lXinerama (e.g. LDFLAGS=-lXinerama before
+ * configure). Support for this is being added to libvncserver 'configure.ac'
+ * so it will all be done automatically.
+
+#define LIBVNCSERVER_HAVE_LIBXINERAMA
+ */
+#ifdef LIBVNCSERVER_HAVE_LIBXINERAMA
+#include <X11/extensions/Xinerama.h>
+#endif
+
/* X and rfb framebuffer */
Display *dpy = 0;
@@ -132,7 +145,9 @@ XImage *scanline;
XImage *fullscreen;
int fs_factor = 0;
+#ifdef SINGLE_TILE_SHM
XShmSegmentInfo tile_shm;
+#endif
XShmSegmentInfo *tile_row_shm; /* for all possible row runs */
XShmSegmentInfo scanline_shm;
XShmSegmentInfo fullscreen_shm;
@@ -149,6 +164,20 @@ int ntiles, ntiles_x, ntiles_y;
/* arrays that indicate changed or checked tiles. */
unsigned char *tile_has_diff, *tile_tried;
+/* blacked-out region things */
+typedef struct bout {
+ int x1, y1, x2, y2;
+} blackout_t;
+typedef struct tbout {
+ blackout_t bo[10]; /* hardwired max rectangles. */
+ int cover;
+ int count;
+} tile_blackout_t;
+blackout_t black[100]; /* hardwired max blackouts */
+int blackouts = 0;
+tile_blackout_t *tile_blackout;
+
+
typedef struct tile_change_region {
/* start and end lines, along y, of the changed area inside a tile. */
unsigned short first_line, last_line;
@@ -180,21 +209,25 @@ int flash_cmap = 0; /* follow installed colormaps */
int force_indexed_color = 0; /* whether to force indexed color for 8bpp */
int use_modifier_tweak = 0; /* use the altgr_keyboard modifier tweak */
-char *remap_file = NULL; /* user supplied remapping file */
+char *remap_file = NULL; /* user supplied remapping file or list */
int nofb = 0; /* do not send any fb updates */
+char *blackout_string = NULL; /* -blackout */
+int xinerama = 0; /* -xinerama */
+
char *client_connect = NULL; /* strings for -connect option */
char *client_connect_file = NULL;
int vnc_connect = 0; /* -vncconnect option */
int local_cursor = 1; /* whether the viewer draws a local cursor */
int show_mouse = 0; /* display a cursor for the real mouse */
+int use_xwarppointer = 0; /* use XWarpPointer instead of XTestFake... */
int show_root_cursor = 0; /* show X when on root background */
int show_dragging = 1; /* process mouse movement events */
int watch_bell = 1; /* watch for the bell using XKEYBOARD */
int old_pointer = 0; /* use the old way of updating the pointer */
-int single_copytile = 0; /* use the old way copy_tile() */
+int single_copytile = 0; /* use the old way copy_tiles() */
int using_shm = 1; /* whether mit-shm is used */
int flip_byte_order = 0; /* sometimes needed when using_shm = 0 */
@@ -252,6 +285,8 @@ int debug_keyboard = 0;
int quiet = 0;
double dtime(double *);
+void zero_fb(int, int, int, int);
+
#if defined(LIBVNCSERVER_X11VNC_THREADED) && ! defined(X11VNC_THREADED)
#define X11VNC_THREADED
#endif
@@ -294,7 +329,9 @@ void clean_up_exit (int ret) {
exit_flag = 1;
/* remove the shm areas: */
+#ifdef SINGLE_TILE_SHM
shm_clean(&tile_shm, tile);
+#endif
shm_clean(&scanline_shm, scanline);
shm_clean(&fullscreen_shm, fullscreen);
@@ -336,7 +373,9 @@ void interrupted (int sig) {
* to avoid deadlock, etc, just delete the shm areas and
* leave the X stuff hanging.
*/
+#ifdef SINGLE_TILE_SHM
shm_delete(&tile_shm);
+#endif
shm_delete(&scanline_shm);
shm_delete(&fullscreen_shm);
@@ -387,7 +426,7 @@ void set_signals(void) {
void client_gone(rfbClientPtr client) {
if (connect_once) {
- fprintf(stderr, "viewer exited.\n");
+ rfbLog("viewer exited.\n");
clean_up_exit(0);
}
}
@@ -729,9 +768,26 @@ void initialize_remap(char *infile) {
in = fopen(infile, "r");
if (in == NULL) {
- rfbLog("remap: cannot open: %s\n", infile);
- perror("fopen");
- clean_up_exit(1);
+ /* assume cmd line key1:key2,key3:key4 */
+ if (! strchr(infile, ':') || (in = tmpfile()) == NULL) {
+ rfbLog("remap: cannot open: %s\n", infile);
+ perror("fopen");
+ clean_up_exit(1);
+ }
+ p = infile;
+ while (*p) {
+ if (*p == ':') {
+ fprintf(in, " ");
+ } else if (*p == ',') {
+ fprintf(in, "\n");
+ } else {
+ fprintf(in, "%c", *p);
+ }
+ p++;
+ }
+ fprintf(in, "\n");
+ fflush(in);
+ rewind(in);
}
while (fgets(line, 256, in) != NULL) {
int blank = 1;
@@ -804,13 +860,19 @@ void DebugXTestFakeKeyEvent(Display* dpy, KeyCode key, Bool down, time_t cur_tim
void tweak_mod(signed char mod, rfbBool down) {
rfbBool is_shift = mod_state & (LEFTSHIFT|RIGHTSHIFT);
Bool dn = (Bool) down;
- if (debug_keyboard) {
- rfbLog("tweak_mod: down=%d mod=0x%x\n", down, mod);
- }
if (mod < 0) {
+ if (debug_keyboard) {
+ rfbLog("tweak_mod: Skip: down=%d mod=0x%x\n", down,
+ (int) mod);
+ }
return;
}
+ if (debug_keyboard) {
+ rfbLog("tweak_mod: Start: down=%d mod=0x%x mod_state=0x%x"
+ " is_shift=%d\n", down, (int) mod, (int) mod_state,
+ is_shift);
+ }
X_LOCK;
if (is_shift && mod != 1) {
@@ -831,6 +893,11 @@ void tweak_mod(signed char mod, rfbBool down) {
XTestFakeKeyEvent(dpy, altgr_code, dn, CurrentTime);
}
X_UNLOCK;
+ if (debug_keyboard) {
+ rfbLog("tweak_mod: Finish: down=%d mod=0x%x mod_state=0x%x"
+ " is_shift=%d\n", down, (int) mod, (int) mod_state,
+ is_shift);
+ }
}
static void modifier_tweak_keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) {
@@ -867,11 +934,16 @@ static void modifier_tweak_keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr
k = XKeysymToKeycode(dpy, (KeySym) keysym);
X_UNLOCK;
}
+ if (debug_keyboard) {
+ rfbLog("modifier_tweak_keyboard: KeySym 0x%x \"%s\" -> "
+ "KeyCode 0x%x%s\n", (int) keysym, XKeysymToString(keysym),
+ (int) k, k ? "" : " *ignored*");
+ }
if ( k != NoSymbol ) {
X_LOCK;
XTestFakeKeyEvent(dpy, k, (Bool) down, CurrentTime);
X_UNLOCK;
- }
+ }
if ( tweak ) {
tweak_mod(modifiers[keysym], False);
@@ -928,8 +1000,9 @@ static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) {
k = XKeysymToKeycode(dpy, (KeySym) keysym);
if (debug_keyboard) {
- rfbLog("keyboard(): KeySym 0x%x \"%s\" -> KeyCode 0x%x\n",
- (int) keysym, XKeysymToString(keysym), (int) k);
+ rfbLog("keyboard(): KeySym 0x%x \"%s\" -> KeyCode 0x%x%s\n",
+ (int) keysym, XKeysymToString(keysym), (int) k,
+ k ? "" : " *ignored*");
}
if ( k != NoSymbol ) {
@@ -1140,7 +1213,11 @@ void update_pointer(int mask, int x, int y) {
X_LOCK;
- XTestFakeMotionEvent(dpy, scr, x + off_x, y + off_y, CurrentTime);
+ if (! use_xwarppointer) {
+ XTestFakeMotionEvent(dpy, scr, x+off_x, y+off_y, CurrentTime);
+ } else {
+ XWarpPointer(dpy, None, window, 0, 0, 0, 0, x+off_x, y+off_y);
+ }
cursor_x = x;
cursor_y = y;
@@ -1812,7 +1889,7 @@ int cur_save_x, cur_save_y, cur_save_w, cur_save_h;
int cur_save_cx, cur_save_cy, cur_save_which, cur_saved = 0;
/*
- * save current cursor info and the patch of data it covers
+ * save current cursor info and the patch of non-cursor data it covers
*/
void save_mouse_patch(int x, int y, int w, int h, int cx, int cy, int which) {
int pixelsize = bpp >> 3;
@@ -1887,6 +1964,43 @@ int tree_descend_cursor(void) {
return descend;
}
+void blackout_nearby_tiles(x, y, dt) {
+ int sx, sy, n, b;
+ int tx = x/tile_x;
+ int ty = y/tile_y;
+
+ if (! blackouts) {
+ return;
+ }
+ if (dt < 1) {
+ dt = 1;
+ }
+ /* loop over a range of tiles, blacking out as needed */
+
+ for (sx = tx - dt; sx <= tx + dt; sx++) {
+ if (sx < 0 || sx >= tile_x) {
+ continue;
+ }
+ for (sy = ty - dt; sy <= ty + dt; sy++) {
+ if (sy < 0 || sy >= tile_y) {
+ continue;
+ }
+ n = sx + sy * ntiles_x;
+ if (tile_blackout[n].cover == 0) {
+ continue;
+ }
+ for (b=0; b <= tile_blackout[n].count; b++) {
+ int x1, y1, x2, y2;
+ x1 = tile_blackout[n].bo[b].x1;
+ y1 = tile_blackout[n].bo[b].y1;
+ x2 = tile_blackout[n].bo[b].x2;
+ y2 = tile_blackout[n].bo[b].y2;
+ zero_fb(x1, y1, x2, y2);
+ }
+ }
+ }
+}
+
/*
* draw one of the mouse cursors into the rfb fb
*/
@@ -1975,6 +2089,27 @@ void draw_mouse(int x, int y, int which, int update) {
}
}
+ if (blackouts) {
+ /*
+ * loop over a range of tiles, blacking out as needed
+ * note we currently disable mouse drawing under blackouts.
+ */
+ static int mx = -1, my = -1;
+ int skip = 0;
+ if (mx < 0) {
+ mx = x;
+ my = y;
+ } else if (mx == x && my == y) {
+ skip = 1;
+ }
+ mx = x;
+ my = y;
+
+ if (! skip) {
+ blackout_nearby_tiles(x, y, 2);
+ }
+ }
+
if (update) {
/* x and y of the real (X server) mouse */
static int mouse_x = -1;
@@ -2260,10 +2395,42 @@ void nofb_hook(rfbClientPtr cl) {
*/
void initialize_screen(int *argc, char **argv, XImage *fb) {
int have_masks = 0;
+ int argc_orig = *argc;
screen = rfbGetScreen(argc, argv, fb->width, fb->height,
fb->bits_per_pixel, 8, fb->bits_per_pixel/8);
+ fprintf(stderr, "\n");
+
+ if (! screen) {
+ int i;
+ rfbLog("\n");
+ rfbLog("failed to create rfb screen.\n");
+ for (i=0; i< *argc; i++) {
+ rfbLog("\t[%d] %s\n", i, argv[i]);
+ }
+ clean_up_exit(1);
+ }
+
+/*
+ * This ifdef is a transient for source compatibility for people who download
+ * 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".
+ */
+#ifdef LIBVNCSERVER_VERSION
+if (strcmp(LIBVNCSERVER_VERSION, "0.5") && strcmp(LIBVNCSERVER_VERSION, "0.6")) {
+ if (*argc != 1) {
+ int i;
+ rfbLog("*** unrecognized option(s) ***\n");
+ for (i=1; i< *argc; i++) {
+ rfbLog("\t[%d] %s\n", i, argv[i]);
+ }
+ rfbLog("for a list of options run: x11vnc -help\n");
+ clean_up_exit(1);
+ }
+}
+#endif
+
screen->paddedWidthInBytes = fb->bytes_per_line;
screen->rfbServerFormat.bitsPerPixel = fb->bits_per_pixel;
screen->rfbServerFormat.depth = fb->depth;
@@ -2276,14 +2443,14 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
if ( ! have_masks && screen->rfbServerFormat.bitsPerPixel == 8
&& CellsOfScreen(ScreenOfDisplay(dpy,scr)) ) {
/* indexed colour */
- if (! quiet) fprintf(stderr, "using 8bpp indexed colour\n");
+ if (!quiet) rfbLog("Using X display with 8bpp indexed color\n");
indexed_colour = 1;
set_colormap();
} else {
/* general case ... */
if (! quiet) {
- fprintf(stderr, "using %dbpp depth=%d true colour\n",
- fb->bits_per_pixel, fb->depth);
+ rfbLog("Using X display with %dbpp depth=%d true "
+ "color\n", fb->bits_per_pixel, fb->depth);
}
/* convert masks to bit shifts and max # colors */
@@ -2378,6 +2545,316 @@ void initialize_screen(int *argc, char **argv, XImage *fb) {
}
/*
+ * 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) {
+ char *p, *blist = strdup(list);
+ int x, y, X, Y, h, w;
+ int tx, ty;
+
+ p = strtok(blist, ",");
+ while (p) {
+ /* handle +/-x and +/-y */
+ 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) {
+ x = dpy_x - x - w;
+ } else if (sscanf(p, "%dx%d+%d-%d", &w, &h, &x, &y) == 4) {
+ y = dpy_y - y - h;
+ } else if (sscanf(p, "%dx%d-%d-%d", &w, &h, &x, &y) == 4) {
+ x = dpy_x - x - w;
+ y = dpy_y - y - h;
+ } else {
+ if (*p != '\0') {
+ rfbLog("skipping invalid geometry: %s\n", p);
+ }
+ p = strtok(NULL, ",");
+ continue;
+ }
+ X = x + w;
+ Y = y + h;
+ if (x < 0 || x > dpy_x || y < 0 || y > dpy_y ||
+ X < 0 || X > dpy_x || Y < 0 || Y > dpy_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 {
+ rfbLog("blackout rect: %s: x=%d-%d y=%d-%d\n", p,
+ x, X, y, Y);
+
+ /*
+ * note that the black out is x1 <= x but x < x2
+ * for the region. i.e. the x2, y2 are outside
+ * by 1 pixel.
+ */
+ black[blackouts].x1 = x;
+ black[blackouts].y1 = y;
+ black[blackouts].x2 = X;
+ black[blackouts].y2 = Y;
+ blackouts++;
+ if (blackouts >= 100) {
+ rfbLog("too many blackouts: %d\n", blackouts);
+ break;
+ }
+ }
+ p = strtok(NULL, ",");
+ }
+ free(blist);
+}
+
+/*
+ * Now that all blackout rectangles have been constructed, see what overlap
+ * they have with the tiles in the system. If a tile is touched by a
+ * blackout, record information.
+ */
+void blackout_tiles() {
+ int tx, ty;
+ if (! blackouts) {
+ return;
+ }
+
+ /*
+ * to simplify things drop down to single copy mode, no vcr, etc...
+ */
+ single_copytile = 1;
+ if (show_mouse) {
+ rfbLog("disabling remote mouse drawing due to blackouts\n");
+ show_mouse = 0;
+ }
+
+ /* loop over all tiles. */
+ for (ty=0; ty < ntiles_y; ty++) {
+ for (tx=0; tx < ntiles_x; tx++) {
+ sraRegionPtr tile_reg, black_reg;
+ sraRect rect;
+ sraRectangleIterator *iter;
+ int n, b, x1, y1, x2, y2, cnt;
+
+ /* tile number and coordinates: */
+ n = tx + ty * ntiles_x;
+ x1 = tx * tile_x;
+ y1 = ty * tile_y;
+ x2 = x1 + tile_x;
+ y2 = y1 + tile_y;
+ if (x2 > dpy_x) {
+ x2 = dpy_x;
+ }
+ if (y2 > dpy_y) {
+ y2 = dpy_y;
+ }
+
+ /* make regions for the tile and the blackouts: */
+ black_reg = (sraRegionPtr) sraRgnCreate();
+ tile_reg = (sraRegionPtr) sraRgnCreateRect(x1, y1,
+ x2, y2);
+
+ tile_blackout[n].cover = 0;
+ tile_blackout[n].count = 0;
+
+ /* union of blackouts */
+ for (b=0; b < blackouts; b++) {
+ sraRegionPtr tmp_reg = (sraRegionPtr)
+ sraRgnCreateRect(black[b].x1, black[b].y1,
+ black[b].x2, black[b].y2);
+
+ sraRgnOr(black_reg, tmp_reg);
+ sraRgnDestroy(tmp_reg);
+ }
+
+ if (! sraRgnAnd(black_reg, tile_reg)) {
+ /*
+ * no intersection for this tile, so we
+ * are done.
+ */
+ sraRgnDestroy(black_reg);
+ sraRgnDestroy(tile_reg);
+ continue;
+ }
+
+ /*
+ * loop over rectangles that make up the blackout
+ * region:
+ */
+ cnt = 0;
+ iter = sraRgnGetIterator(black_reg);
+ while (sraRgnIteratorNext(iter, &rect)) {
+
+ /* make sure x1 < x2 and y1 < y2 */
+ if (rect.x1 > rect.x2) {
+ int tmp = rect.x2;
+ rect.x2 = rect.x1;
+ rect.x1 = tmp;
+ }
+ if (rect.y1 > rect.y2) {
+ int tmp = rect.y2;
+ rect.y2 = rect.y1;
+ rect.y1 = tmp;
+ }
+
+ /* store coordinates */
+ tile_blackout[n].bo[cnt].x1 = rect.x1;
+ tile_blackout[n].bo[cnt].y1 = rect.y1;
+ tile_blackout[n].bo[cnt].x2 = rect.x2;
+ tile_blackout[n].bo[cnt].y2 = rect.y2;
+
+ /* note if the tile is completely obscured */
+ if (rect.x1 == x1 && rect.y1 == y1 &&
+ rect.x2 == x2 && rect.y2 == y2) {
+ tile_blackout[n].cover = 2;
+ } else {
+ tile_blackout[n].cover = 1;
+ }
+
+ if (++cnt >= 10) {
+ rfbLog("too many blackout rectangles "
+ "for tile %d=%d,%d.\n", n, tx, ty);
+ break;
+ }
+ }
+
+ sraRgnReleaseIterator(iter);
+ sraRgnDestroy(black_reg);
+ sraRgnDestroy(tile_reg);
+
+ tile_blackout[n].count = cnt;
+ }
+ }
+}
+
+void initialize_xinerama () {
+#ifdef LIBVNCSERVER_HAVE_LIBXINERAMA
+ XineramaScreenInfo *sc, *xineramas;
+#endif
+ sraRegionPtr black_region, tmp_region;
+ sraRectangleIterator *iter;
+ sraRect rect;
+ char *bstr, *tstr;
+ int ev, er, i, n, rcnt;
+
+#ifndef LIBVNCSERVER_HAVE_LIBXINERAMA
+ rfbLog("Xinerama: Library libXinerama is not available to determine\n");
+ rfbLog("Xinerama: the head geometries, consider using -blackout\n");
+ rfbLog("Xinerama: if the screen is non-rectangular.\n");
+#else
+ if (! XineramaQueryExtension(dpy, &ev, &er)) {
+ rfbLog("Xinerama: disabling: display does not support it.\n");
+ xinerama = 0;
+ return;
+ }
+ if (! XineramaIsActive(dpy)) {
+ /* n.b. change to XineramaActive(dpy, window) someday */
+ rfbLog("Xinerama: disabling: not active on display.\n");
+ xinerama = 0;
+ return;
+ }
+
+ /* n.b. change to XineramaGetData() someday */
+ xineramas = XineramaQueryScreens(dpy, &n);
+ rfbLog("Xinerama: number of sub-screens: %d\n", n);
+
+ if (n == 1) {
+ rfbLog("Xinerama: no blackouts needed (only one"
+ " sub-screen)\n");
+ XFree(xineramas);
+ return; /* must be OK w/o change */
+ }
+
+ black_region = sraRgnCreateRect(0, 0, dpy_x, dpy_y);
+
+ sc = xineramas;
+ for (i=0; i<n; i++) {
+ int x, y, w, h;
+
+ x = sc->x_org;
+ y = sc->y_org;
+ w = sc->width;
+ h = sc->height;
+
+ tmp_region = sraRgnCreateRect(x, y, x + w, y + h);
+
+ sraRgnSubtract(black_region, tmp_region);
+ sraRgnDestroy(tmp_region);
+ sc++;
+ }
+ XFree(xineramas);
+
+ if (sraRgnEmpty(black_region)) {
+ rfbLog("Xinerama: no blackouts needed (screen fills"
+ " rectangle)\n");
+ sraRgnDestroy(black_region);
+ return;
+ }
+
+ /* max len is 10000x10000+10000+10000 (23 chars) per geometry */
+ rcnt = (int) sraRgnCountRects(black_region);
+ bstr = (char *) malloc(30 * rcnt * sizeof(char));
+ tstr = (char *) malloc(30 * sizeof(char));
+ bstr[0] = '\0';
+
+ iter = sraRgnGetIterator(black_region);
+ while (sraRgnIteratorNext(iter, &rect)) {
+ int x, y, w, h;
+
+ /* make sure x1 < x2 and y1 < y2 */
+ if (rect.x1 > rect.x2) {
+ int tmp = rect.x2;
+ rect.x2 = rect.x1;
+ rect.x1 = tmp;
+ }
+ if (rect.y1 > rect.y2) {
+ int tmp = rect.y2;
+ rect.y2 = rect.y1;
+ rect.y1 = tmp;
+ }
+ x = rect.x1;
+ y = rect.y1;
+ w = rect.x2 - x;
+ h = rect.y2 - y;
+ sprintf(tstr, "%dx%d+%d+%d,", w, h, x, y);
+ strcat(bstr, tstr);
+ }
+ initialize_blackout(bstr);
+
+ free(bstr);
+ free(tstr);
+#endif
+
+}
+
+/*
+ * Fill the framebuffer with zero for the prescribed rectangle
+ */
+void zero_fb(x1, y1, x2, y2) {
+ int pixelsize = bpp >> 3;
+ int line, fill = 0;
+ char *dst;
+
+ if (x1 < 0 || x2 <= x1 || x2 > dpy_x) {
+ return;
+ }
+ if (y1 < 0 || y2 <= y1 || y2 > dpy_y) {
+ return;
+ }
+
+ dst = screen->frameBuffer + y1 * bytes_per_line + x1 * pixelsize;
+ line = y1;
+ while (line++ < y2) {
+ memset(dst, fill, (size_t) (x2 - x1) * pixelsize);
+ dst += bytes_per_line;
+ }
+}
+
+/*
+ * Fill the framebuffer with zeros for each blackout region
+ */
+void blackout_regions() {
+ int i;
+ for (i=0; i < blackouts; i++) {
+ zero_fb(black[i].x1, black[i].y1, black[i].x2, black[i].y2);
+ }
+}
+
+/*
* setup tile numbers and allocate the tile and hint arrays:
*/
void initialize_tiles() {
@@ -2390,6 +2867,8 @@ void initialize_tiles() {
malloc((size_t) (ntiles * sizeof(unsigned char)));
tile_tried = (unsigned char *)
malloc((size_t) (ntiles * sizeof(unsigned char)));
+ tile_blackout = (tile_blackout_t *)
+ malloc((size_t) (ntiles * sizeof(tile_blackout_t)));
tile_region = (region_t *) malloc((size_t) (ntiles * sizeof(region_t)));
tile_row = (XImage **)
@@ -2472,7 +2951,7 @@ int shm_create(XShmSegmentInfo *shm, XImage **ximg_ptr, int w, int h,
xim->bitmap_bit_order = LSBFirst;
}
if (! reported && ! quiet) {
- fprintf(stderr, "changing XImage byte order"
+ rfbLog("changing XImage byte order"
" to %s\n", bo);
reported = 1;
}
@@ -2575,9 +3054,11 @@ void initialize_shm() {
int i;
/* set all shm areas to "none" before trying to create any */
+#ifdef SINGLE_TILE_SHM
tile_shm.shmid = -1;
tile_shm.shmaddr = (char *) -1;
tile = NULL;
+#endif
scanline_shm.shmid = -1;
scanline_shm.shmaddr = (char *) -1;
scanline = NULL;
@@ -2590,11 +3071,13 @@ void initialize_shm() {
tile_row[i] = NULL;
}
+#ifdef SINGLE_TILE_SHM
/* the tile (e.g. 32x32) shared memory area image: */
if (! shm_create(&tile_shm, &tile, tile_x, tile_y, "tile")) {
clean_up_exit(1);
}
+#endif
/* the scanline (e.g. 1280x1) shared memory area image: */
@@ -2608,8 +3091,12 @@ void initialize_shm() {
* limits, e.g. the default 1MB on Solaris)
*/
set_fs_factor(1024 * 1024);
+ if (fs_frac >= 1.0) {
+ fs_frac = 1.1;
+ fs_factor = 0;
+ }
if (! fs_factor) {
- fprintf(stderr, "warning: fullscreen updates are disabled.\n");
+ rfbLog("warning: fullscreen updates are disabled.\n");
} else {
if (! shm_create(&fullscreen_shm, &fullscreen, dpy_x,
dpy_y/fs_factor, "fullscreen")) {
@@ -2800,168 +3287,6 @@ void tile_updates() {
/*
- * copy_tile() is called on a tile with a known change (from a scanline
- * diff) or a suspected change (from our various heuristics).
- *
- * Examine the whole tile for the y-range of difference, copy that
- * image difference to the rfb framebuffer, and do bookkeepping wrt
- * the y-range and edge differences.
- *
- * This call is somewhat costly, maybe 1-2 ms. Primarily the XShmGetImage
- * and then the memcpy/memcmp.
- */
-
-void copy_tile(int tx, int ty) {
- int x, y, line, first_line, last_line;
- int size_x, size_y, n, dw, dx;
- int pixelsize = bpp >> 3;
- unsigned short l_diff = 0, r_diff = 0;
-
- int restored_patch = 0; /* for show_mouse */
-
- char *src, *dst, *s_src, *s_dst, *m_src, *m_dst;
- char *h_src, *h_dst;
-
- x = tx * tile_x;
- y = ty * tile_y;
-
- size_x = dpy_x - x;
- if ( size_x > tile_x ) size_x = tile_x;
-
- size_y = dpy_y - y;
- if ( size_y > tile_y ) size_y = tile_y;
-
- n = tx + ty * ntiles_x; /* number of the tile */
-
- X_LOCK;
- if ( using_shm && size_x == tile_x && size_y == tile_y ) {
- /* general case: */
- XShmGetImage(dpy, window, tile, x, y, AllPlanes);
- } else {
- /*
- * No shm or near bottom/rhs edge case:
- * (but only if tile size does not divide screen size)
- */
- XGetSubImage(dpy, window, x, y, size_x, size_y, AllPlanes,
- ZPixmap, tile, 0, 0);
- }
- X_UNLOCK;
-
- /*
- * Some awkwardness wrt the little remote mouse patch we display.
- * When threaded we want to have as small a window of time
- * as possible when the mouse image is not in the fb, otherwise
- * a libvncserver thread may send the uncorrected patch to the
- * clients.
- */
- if (show_mouse && use_threads && cur_saved) {
- /* check for overlap */
- if (cur_save_x + cur_save_w > x && x + size_x > cur_save_x &&
- cur_save_y + cur_save_h > y && y + size_y > cur_save_y) {
-
- /* restore the real data to the rfb fb */
- restore_mouse_patch();
- restored_patch = 1;
- }
- }
-
- src = tile->data;
- dst = screen->frameBuffer + y * bytes_per_line + x * pixelsize;
-
- s_src = src;
- s_dst = dst;
-
- first_line = -1;
-
- /* find the first line with difference: */
- for (line = 0; line < size_y; line++) {
- if ( memcmp(s_dst, s_src, size_x * pixelsize) ) {
- first_line = line;
- break;
- }
- s_src += tile->bytes_per_line;
- s_dst += bytes_per_line;
- }
-
- tile_tried[n] = 1;
- if (first_line == -1) {
- /* tile has no difference, note it and get out: */
- tile_has_diff[n] = 0;
- if (restored_patch) {
- redraw_mouse();
- }
- return;
- } else {
- /*
- * make sure it is recorded (e.g. sometimes we guess tiles
- * and they came in with tile_has_diff 0)
- */
- tile_has_diff[n] = 1;
- }
-
- m_src = src + (tile->bytes_per_line * size_y);
- m_dst = dst + (bytes_per_line * size_y);
- last_line = first_line;
-
- /* find the last line with difference: */
- for (line = size_y - 1; line > first_line; line--) {
- m_src -= tile->bytes_per_line;
- m_dst -= bytes_per_line;
- if ( memcmp(m_dst, m_src, size_x * pixelsize) ) {
- last_line = line;
- break;
- }
- }
-
- /* look for differences on left and right hand edges: */
- dx = (size_x - tile_fuzz) * pixelsize;
- dw = tile_fuzz * pixelsize;
-
- h_src = src;
- h_dst = dst;
- for (line = 0; line < size_y; line++) {
- if (! l_diff && memcmp(h_dst, h_src, dw) ) {
- l_diff = 1;
- }
- if (! r_diff && memcmp(h_dst + dx, h_src + dx, dw) ) {
- r_diff = 1;
- }
- if (l_diff && r_diff) {
- break;
- }
- h_src += tile->bytes_per_line;
- h_dst += bytes_per_line;
- }
-
- /* now copy the difference to the rfb framebuffer: */
- for (line = first_line; line <= last_line; line++) {
- memcpy(s_dst, s_src, size_x * pixelsize);
- s_src += tile->bytes_per_line;
- s_dst += bytes_per_line;
- }
-
- if (restored_patch) {
- redraw_mouse();
- }
-
- /* record all the info in the region array for this tile: */
- tile_region[n].first_line = first_line;
- tile_region[n].last_line = last_line;
- tile_region[n].left_diff = l_diff;
- tile_region[n].right_diff = r_diff;
-
- tile_region[n].top_diff = 0;
- tile_region[n].bot_diff = 0;
- if ( first_line < tile_fuzz ) {
- tile_region[n].top_diff = 1;
- }
- if ( last_line > (size_y - 1) - tile_fuzz ) {
- tile_region[n].bot_diff = 1;
- }
-}
-
-
-/*
* copy_tiles() gives a slight improvement over copy_tile() since
* adjacent runs of tiles are done all at once there is some savings
* due to contiguous memory access. Not a great speedup, but in
@@ -3019,6 +3344,16 @@ void copy_tiles(int tx, int ty, int nt) {
n = tx + ty * ntiles_x; /* number of the first tile */
+ if (blackouts && tile_blackout[n].cover == 2) {
+ /*
+ * If there are blackouts and this tile is completely covered
+ * no need to poll screen or do anything else..
+ * n.b. we are int single copy_tile mode: nt=1
+ */
+ tile_has_diff[n] = 0;
+ return;
+ }
+
X_LOCK;
/* read in the whole tile run at once: */
if ( using_shm && size_x == tile_x * nt && size_y == tile_y ) {
@@ -3034,6 +3369,35 @@ void copy_tiles(int tx, int ty, int nt) {
}
X_UNLOCK;
+ if (blackouts && tile_blackout[n].cover == 1) {
+ /*
+ * If there are blackouts and this tile is partially covered
+ * we should re-black-out the portion.
+ * n.b. we are int single copy_tile mode: nt=1
+ */
+ int x1, x2, y1, y2, b;
+ int w, s, fill = 0;
+
+ for (b=0; b < tile_blackout[n].count; b++) {
+ char *b_dst = tile_row[nt]->data;
+
+ x1 = tile_blackout[n].bo[b].x1 - x;
+ y1 = tile_blackout[n].bo[b].y1 - y;
+ x2 = tile_blackout[n].bo[b].x2 - x;
+ y2 = tile_blackout[n].bo[b].y2 - y;
+
+ w = (x2 - x1) * pixelsize;
+ s = x1 * pixelsize;
+
+ for (line = 0; line < size_y; line++) {
+ if (y1 <= line && line < y2) {
+ memset(b_dst + s, fill, (size_t) w);
+ }
+ b_dst += tile_row[nt]->bytes_per_line;
+ }
+ }
+ }
+
/*
* Some awkwardness wrt the little remote mouse patch we display.
* When threaded we want to have as small a window of time
@@ -3285,12 +3649,11 @@ int copy_all_tiles() {
n = x + y * ntiles_x;
if (tile_has_diff[n]) {
- copy_tile(x, y);
- /* later: copy_tiles(x, y, 1); */
+ copy_tiles(x, y, 1);
}
if (! tile_has_diff[n]) {
/*
- * n.b. copy_tile() may have detected
+ * n.b. copy_tiles() may have detected
* no change and reset tile_has_diff to 0.
*/
continue;
@@ -3404,7 +3767,7 @@ int copy_tiles_backward_pass() {
if (y >= 1 && ! tile_has_diff[m] && tile_region[n].top_diff) {
if (! tile_tried[m]) {
tile_has_diff[m] = 2;
- copy_tile(x, y-1);
+ copy_tiles(x, y-1, 1);
}
}
@@ -3413,7 +3776,7 @@ int copy_tiles_backward_pass() {
if (x >= 1 && ! tile_has_diff[m] && tile_region[n].left_diff) {
if (! tile_tried[m]) {
tile_has_diff[m] = 2;
- copy_tile(x-1, y);
+ copy_tiles(x-1, y, 1);
}
}
}
@@ -3457,7 +3820,7 @@ void gap_try(int x, int y, int *run, int *saw, int along_x) {
continue;
}
- copy_tile(xt, yt);
+ copy_tiles(xt, yt, 1);
}
*run = 0;
*saw = 1;
@@ -3522,7 +3885,7 @@ void island_try(int x, int y, int u, int v, int *run) {
return;
}
- copy_tile(u, v);
+ copy_tiles(u, v, 1);
}
}
@@ -3597,6 +3960,10 @@ void copy_screen() {
X_UNLOCK;
+ if (blackouts) {
+ blackout_regions();
+ }
+
rfbMarkRectAsModified(screen, 0, 0, dpy_x, dpy_y);
}
@@ -3704,7 +4071,7 @@ void ping_clients(int tile_cnt) {
if (rfbMaxClientWait <= 3000) {
rfbMaxClientWait = 3000;
- fprintf(stderr, "reset rfbMaxClientWait to %d ms.\n",
+ rfbLog("reset rfbMaxClientWait to %d ms.\n",
rfbMaxClientWait);
}
if (tile_cnt) {
@@ -3717,6 +4084,114 @@ void ping_clients(int tile_cnt) {
}
/*
+ * scan_display() wants to know if this tile can be skipped due to
+ * blackout regions: (no data compare is done, just a quick geometric test)
+ */
+int blackout_line_skip(int n, int x, int y, int rescan, int *tile_count) {
+
+ if (tile_blackout[n].cover == 2) {
+ tile_has_diff[n] = 0;
+ return 1; /* skip it */
+
+ } else if (tile_blackout[n].cover == 1) {
+ int w, x1, y1, x2, y2, b, hit = 0;
+ if (x + NSCAN > dpy_x) {
+ w = dpy_x - x;
+ } else {
+ w = NSCAN;
+ }
+
+ for (b=0; b < tile_blackout[n].count; b++) {
+
+ /* n.b. these coords are in full display space: */
+ x1 = tile_blackout[n].bo[b].x1;
+ x2 = tile_blackout[n].bo[b].x2;
+ y1 = tile_blackout[n].bo[b].y1;
+ y2 = tile_blackout[n].bo[b].y2;
+
+ if (x2 - x1 < w) {
+ /* need to cover full width */
+ continue;
+ }
+ if (y1 <= y && y < y2) {
+ hit = 1;
+ break;
+ }
+ }
+ if (hit) {
+ if (! rescan) {
+ tile_has_diff[n] = 0;
+ } else {
+ *tile_count += tile_has_diff[n];
+ }
+ return 1; /* skip */
+ }
+ }
+ return 0; /* do not skip */
+}
+
+/*
+ * scan_display() wants to know if this changed tile can be skipped due
+ * to blackout regions (we do an actual compare to find the changed region).
+ */
+int blackout_line_cmpskip(int n, int x, int y, char *dst, char *src,
+ int w, int pixelsize) {
+
+ int i, x1, y1, x2, y2, b, hit = 0;
+ int beg = -1, end = -1;
+
+ if (tile_blackout[n].cover == 0) {
+ return 0; /* 0 means do not skip it. */
+ } else if (tile_blackout[n].cover == 2) {
+ return 1; /* 1 means skip it. */
+ }
+
+ /* tile has partial coverage: */
+
+ for (i=0; i < w * pixelsize; i++) {
+ if (*(dst+i) != *(src+i)) {
+ beg = i/pixelsize; /* beginning difference */
+ break;
+ }
+ }
+ for (i = w * pixelsize - 1; i >= 0; i--) {
+ if (*(dst+i) != *(src+i)) {
+ end = i/pixelsize; /* ending difference */
+ break;
+ }
+ }
+ if (beg < 0 || end < 0) {
+ /* problem finding range... */
+ return 0;
+ }
+
+ /* loop over blackout rectangles: */
+ for (b=0; b < tile_blackout[n].count; b++) {
+
+ /* y in full display space: */
+ y1 = tile_blackout[n].bo[b].y1;
+ y2 = tile_blackout[n].bo[b].y2;
+
+ /* x relative to tile origin: */
+ x1 = tile_blackout[n].bo[b].x1 - x;
+ x2 = tile_blackout[n].bo[b].x2 - x;
+
+ if (y1 > y || y >= y2) {
+ continue;
+ }
+ if (x1 <= beg && end <= x2) {
+ hit = 1;
+ break;
+ }
+ }
+ if (hit) {
+ return 1;
+ } else {
+ return 0;
+ }
+}
+
+/*
* Loop over 1-pixel tall horizontal scanlines looking for changes.
* Record the changes in tile_has_diff[]. Scanlines in the loop are
* equally spaced along y by NSCAN pixels, but have a slightly random
@@ -3727,7 +4202,7 @@ int scan_display(int ystart, int rescan) {
int pixelsize = bpp >> 3;
int x, y, w, n;
int tile_count = 0;
- int whole_line = 1, nodiffs;
+ int whole_line = 1, nodiffs = 0;
y = ystart;
@@ -3748,7 +4223,6 @@ int scan_display(int ystart, int rescan) {
src = scanline->data;
dst = screen->frameBuffer + y * bytes_per_line;
- nodiffs = 0;
if (whole_line && ! memcmp(dst, src, bytes_per_line)) {
/* no changes anywhere in scan line */
nodiffs = 1;
@@ -3762,6 +4236,14 @@ int scan_display(int ystart, int rescan) {
while (x < dpy_x) {
n = (x/tile_x) + (y/tile_y) * ntiles_x;
+ if (blackouts) {
+ if (blackout_line_skip(n, x, y, rescan,
+ &tile_count)) {
+ x += NSCAN;
+ continue;
+ }
+ }
+
if (rescan) {
if (nodiffs || tile_has_diff[n]) {
tile_count += tile_has_diff[n];
@@ -3784,8 +4266,18 @@ int scan_display(int ystart, int rescan) {
if (memcmp(dst, src, w * pixelsize)) {
/* found a difference, record it: */
- tile_has_diff[n] = 1;
- tile_count++;
+ if (! blackouts) {
+ tile_has_diff[n] = 1;
+ tile_count++;
+ } else {
+ if (blackout_line_cmpskip(n, x, y,
+ dst, src, w, pixelsize)) {
+ tile_has_diff[n] = 0;
+ } else {
+ tile_has_diff[n] = 1;
+ tile_count++;
+ }
+ }
}
x += NSCAN;
}
@@ -3844,7 +4336,7 @@ void scan_for_updates() {
if (tile_count > frac1 * ntiles) {
/*
* many tiles have changed, so try a rescan (since it should
- * be short compared to the many upcoming copy_tile() calls)
+ * be short compared to the many upcoming copy_tiles() calls)
*/
/* this check is done to skip the extra scan_display() call */
@@ -3867,7 +4359,7 @@ void scan_for_updates() {
/*
* At some number of changed tiles it is better to just
* copy the full screen at once. I.e. time = c1 + m * r1
- * where m is number of tiles, r1 is the copy_tile()
+ * where m is number of tiles, r1 is the copy_tiles()
* time, and c1 is the scan_display() time: for some m
* it crosses the full screen update time.
*
@@ -3939,6 +4431,19 @@ void scan_for_updates() {
pointer(-1, 0, 0, NULL);
}
+ if (blackouts) {
+ /* ignore any diffs in completely covered tiles */
+ int x, y, n;
+ for (y=0; y < ntiles_y; y++) {
+ for (x=0; x < ntiles_x; x++) {
+ n = x + y * ntiles_x;
+ if (tile_blackout[n].cover == 2) {
+ tile_has_diff[n] = 0;
+ }
+ }
+ }
+ }
+
if (use_hints) {
hint_updates(); /* use krfb/x0rfbserver hints algorithm */
} else {
@@ -4231,11 +4736,17 @@ void print_help() {
" for more control build libvncserver with libwrap support.\n"
"-localhost Same as -allow 127.0.0.1\n"
"-inetd Launched by inetd(1): stdio instead of listening socket.\n"
+"\n"
"-noshm Do not use the MIT-SHM extension for the polling.\n"
" remote displays can be polled this way: be careful\n"
" this can use large amounts of network bandwidth.\n"
"-flipbyteorder Sometimes needed if remotely polled host has different\n"
" endianness. Ignored unless -noshm is set.\n"
+"-blackout string Black out rectangles on the screen. string is a comma\n"
+" separated list of WxH+X+Y type geometries for each rect.\n"
+"-xinerama If your screen is composed of multiple monitors glued\n"
+" together via XINERAMA, and that screen is non-rectangular\n"
+" this option will try to guess the areas to black out.\n"
"\n"
"-q Be quiet by printing less informational output.\n"
"-bg Go into the background after screen setup.\n"
@@ -4248,8 +4759,9 @@ void print_help() {
"-modtweak Handle AltGr/Shift modifiers for differing languages\n"
" between client and host (default %s).\n"
"-nomodtweak Send the keysym directly to the X server.\n"
-"-remap file Read keysym remappings from file. Format is one pair\n"
-" of keysyms per line (can be string or the hex value).\n"
+"-remap string Read keysym remappings from file \"string\". Format is\n"
+" one pair of keysyms per line (can be name or hex value).\n"
+" \"string\" can also be of form: key1:key2,key3:key4...\n"
"-nobell Do not watch for XBell events.\n"
"-nofb Ignore framebuffer: only process keyboard and pointer.\n"
"-nosel Do not manage exchange of X selection/cutbuffer.\n"
@@ -4260,6 +4772,9 @@ void print_help() {
"-mouse Draw a 2nd cursor at the current X pointer position.\n"
"-mouseX As -mouse, but also draw an X on root background.\n"
"-X Shorthand for -mouseX -nocursor.\n"
+"-xwarppointer Move the pointer with XWarpPointer instead of XTEST\n"
+" (try as a workaround if pointer behaves poorly, e.g.\n"
+" on touchscreens or other non-standard setups).\n"
"-buttonmap str String to remap mouse buttons. Format: IJK-LMN, this\n"
" maps buttons I -> L, etc., e.g. -buttonmap 13-31\n"
"-nodragging Do not update the display during mouse dragging events.\n"
@@ -4270,7 +4785,6 @@ void print_help() {
"-input_skip n For the old 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"
-"-old_copytile Do not use the new copy_tiles() framebuffer mechanism.\n"
"-debug_pointer Print debugging output for every pointer event.\n"
"-debug_keyboard Print debugging output for every keyboard event.\n"
"\n"
@@ -4287,6 +4801,8 @@ void print_help() {
"\n"
"-fs f If the fraction of changed tiles in a poll is greater\n"
" than f, the whole screen is updated (default %.2f).\n"
+"-onetile Do not use the new copy_tiles() framebuffer mechanism,\n"
+" just use 1 shm tile for polling. Same as -old_copytile.\n"
"-gaps n Heuristic to fill in gaps in rows or cols of n or less\n"
" tiles. Used to improve text paging (default %d).\n"
"-grow n Heuristic to grow islands of changed tiles n or wider\n"
@@ -4366,7 +4882,7 @@ int main(int argc, char** argv) {
XImage *fb;
int i, op, ev, er, maj, min;
char *use_dpy = NULL;
- char *visual_str = NULL;
+ char *arg, *visual_str = NULL;
int pw_loc = -1;
int dt = 0;
int bg = 0;
@@ -4375,12 +4891,18 @@ int main(int argc, char** argv) {
/* used to pass args we do not know about to rfbGetScreen(): */
int argc2 = 1; char *argv2[100];
- argv2[0] = argv[0];
+ argv2[0] = strdup(argv[0]);
for (i=1; i < argc; i++) {
- if (!strcmp(argv[i], "-display")) {
+ /* quick-n-dirty --option handling. */
+ arg = argv[i];
+ if (strstr(arg, "--") == arg) {
+ arg++;
+ }
+
+ if (!strcmp(arg, "-display")) {
use_dpy = argv[++i];
- } else if (!strcmp(argv[i], "-id")) {
+ } else if (!strcmp(arg, "-id")) {
if (sscanf(argv[++i], "0x%x", &subwin) != 1) {
if (sscanf(argv[i], "%d", &subwin) != 1) {
fprintf(stderr, "bad -id arg: %s\n",
@@ -4388,126 +4910,133 @@ int main(int argc, char** argv) {
exit(1);
}
}
- } else if (!strcmp(argv[i], "-visual")) {
+ } else if (!strcmp(arg, "-visual")) {
visual_str = argv[++i];
- } else if (!strcmp(argv[i], "-flashcmap")) {
+ } else if (!strcmp(arg, "-flashcmap")) {
flash_cmap = 1;
- } else if (!strcmp(argv[i], "-notruecolor")) {
+ } else if (!strcmp(arg, "-notruecolor")) {
force_indexed_color = 1;
- } else if (!strcmp(argv[i], "-viewonly")) {
+ } else if (!strcmp(arg, "-viewonly")) {
view_only = 1;
- } else if (!strcmp(argv[i], "-shared")) {
+ } else if (!strcmp(arg, "-shared")) {
shared = 1;
- } else if (!strcmp(argv[i], "-allow")) {
+ } else if (!strcmp(arg, "-allow")) {
allow_list = argv[++i];
- } else if (!strcmp(argv[i], "-localhost")) {
+ } else if (!strcmp(arg, "-localhost")) {
allow_list = "127.0.0.1";
- } else if (!strcmp(argv[i], "-many")
- || !strcmp(argv[i], "-forever")) {
+ } else if (!strcmp(arg, "-many")
+ || !strcmp(arg, "-forever")) {
connect_once = 0;
- } else if (!strcmp(argv[i], "-connect")) {
+ } else if (!strcmp(arg, "-connect")) {
i++;
- if (strchr(argv[i], '/')) {
- client_connect_file = argv[i];
+ if (strchr(arg, '/')) {
+ client_connect_file = arg;
} else {
- client_connect = strdup(argv[i]);
+ client_connect = strdup(arg);
}
- } else if (!strcmp(argv[i], "-vncconnect")) {
+ } else if (!strcmp(arg, "-vncconnect")) {
vnc_connect = 1;
- } else if (!strcmp(argv[i], "-inetd")) {
+ } else if (!strcmp(arg, "-inetd")) {
inetd = 1;
- } else if (!strcmp(argv[i], "-noshm")) {
+ } else if (!strcmp(arg, "-noshm")) {
using_shm = 0;
- } else if (!strcmp(argv[i], "-flipbyteorder")) {
+ } else if (!strcmp(arg, "-flipbyteorder")) {
flip_byte_order = 1;
- } else if (!strcmp(argv[i], "-modtweak")) {
+ } else if (!strcmp(arg, "-modtweak")) {
use_modifier_tweak = 1;
- } else if (!strcmp(argv[i], "-nomodtweak")) {
+ } else if (!strcmp(arg, "-nomodtweak")) {
use_modifier_tweak = 0;
- } else if (!strcmp(argv[i], "-remap")) {
+ } else if (!strcmp(arg, "-remap")) {
remap_file = argv[++i];
- } else if (!strcmp(argv[i], "-nobell")) {
+ } else if (!strcmp(arg, "-blackout")) {
+ blackout_string = argv[++i];
+ } else if (!strcmp(arg, "-xinerama")) {
+ xinerama = 1;
+ } else if (!strcmp(arg, "-nobell")) {
watch_bell = 0;
- } else if (!strcmp(argv[i], "-nofb")) {
+ } else if (!strcmp(arg, "-nofb")) {
nofb = 1;
- } else if (!strcmp(argv[i], "-nosel")) {
+ } else if (!strcmp(arg, "-nosel")) {
watch_selection = 0;
- } else if (!strcmp(argv[i], "-noprimary")) {
+ } else if (!strcmp(arg, "-noprimary")) {
watch_primary = 0;
- } else if (!strcmp(argv[i], "-nocursor")) {
+ } else if (!strcmp(arg, "-nocursor")) {
local_cursor = 0;
- } else if (!strcmp(argv[i], "-mouse")) {
+ } else if (!strcmp(arg, "-mouse")) {
show_mouse = 1;
- } else if (!strcmp(argv[i], "-mouseX")) {
+ } else if (!strcmp(arg, "-mouseX")) {
show_mouse = 1;
show_root_cursor = 1;
- } else if (!strcmp(argv[i], "-X")) {
+ } else if (!strcmp(arg, "-X")) {
show_mouse = 1;
show_root_cursor = 1;
local_cursor = 0;
- } else if (!strcmp(argv[i], "-buttonmap")) {
+ } else if (!strcmp(arg, "-xwarppointer")) {
+ use_xwarppointer = 1;
+ } else if (!strcmp(arg, "-buttonmap")) {
pointer_remap = argv[++i];
- } else if (!strcmp(argv[i], "-nodragging")) {
+ } else if (!strcmp(arg, "-nodragging")) {
show_dragging = 0;
- } else if (!strcmp(argv[i], "-input_skip")) {
+ } else if (!strcmp(arg, "-input_skip")) {
ui_skip = atoi(argv[++i]);
if (! ui_skip) ui_skip = 1;
- } else if (!strcmp(argv[i], "-old_pointer")) {
+ } else if (!strcmp(arg, "-old_pointer")) {
old_pointer = 1;
- } else if (!strcmp(argv[i], "-old_copytile")) {
+ } else if (!strcmp(arg, "-onetile")
+ || !strcmp(arg, "-old_copytile")) {
single_copytile = 1;
- } else if (!strcmp(argv[i], "-debug_pointer")) {
+ } else if (!strcmp(arg, "-debug_pointer")) {
debug_pointer++;
- } else if (!strcmp(argv[i], "-debug_keyboard")) {
+ } else if (!strcmp(arg, "-debug_keyboard")) {
debug_keyboard++;
- } else if (!strcmp(argv[i], "-defer")) {
+ } else if (!strcmp(arg, "-defer")) {
defer_update = atoi(argv[++i]);
- } else if (!strcmp(argv[i], "-wait")) {
+ } else if (!strcmp(arg, "-wait")) {
waitms = atoi(argv[++i]);
got_waitms = 1;
- } else if (!strcmp(argv[i], "-nap")) {
+ } else if (!strcmp(arg, "-nap")) {
take_naps = 1;
#ifdef LIBVNCSERVER_HAVE_LIBPTHREAD
- } else if (!strcmp(argv[i], "-threads")) {
+ } else if (!strcmp(arg, "-threads")) {
use_threads = 1;
- } else if (!strcmp(argv[i], "-nothreads")) {
+ } else if (!strcmp(arg, "-nothreads")) {
use_threads = 0;
#endif
- } else if (!strcmp(argv[i], "-fs")) {
+ } else if (!strcmp(arg, "-fs")) {
fs_frac = atof(argv[++i]);
- } else if (!strcmp(argv[i], "-gaps")) {
+ } else if (!strcmp(arg, "-gaps")) {
gaps_fill = atoi(argv[++i]);
- } else if (!strcmp(argv[i], "-grow")) {
+ } else if (!strcmp(arg, "-grow")) {
grow_fill = atoi(argv[++i]);
- } else if (!strcmp(argv[i], "-fuzz")) {
+ } else if (!strcmp(arg, "-fuzz")) {
tile_fuzz = atoi(argv[++i]);
- } else if (!strcmp(argv[i], "-hints")) {
+ } else if (!strcmp(arg, "-hints")) {
use_hints = 1;
- } else if (!strcmp(argv[i], "-nohints")) {
+ } else if (!strcmp(arg, "-nohints")) {
use_hints = 0;
- } else if (!strcmp(argv[i], "-h") || !strcmp(argv[i], "-help")
- || !strcmp(argv[i], "-?")) {
+ } else if (!strcmp(arg, "-h") || !strcmp(arg, "-help")
+ || !strcmp(arg, "-?")) {
print_help();
- } else if (!strcmp(argv[i], "-q")) {
+ } else if (!strcmp(arg, "-q") || !strcmp(arg, "-quiet")) {
quiet = 1;
#ifdef LIBVNCSERVER_HAVE_SETSID
- } else if (!strcmp(argv[i], "-bg")) {
+ } else if (!strcmp(arg, "-bg") || !strcmp(arg, "-background")) {
bg = 1;
#endif
} else {
- if (!strcmp(argv[i], "-desktop")) {
+ if (!strcmp(arg, "-desktop")) {
dt = 1;
}
- if (!strcmp(argv[i], "-passwd")) {
+ if (!strcmp(arg, "-passwd")) {
pw_loc = i;
}
/* otherwise copy it for use below. */
if (! quiet && i != pw_loc && i != pw_loc+1) {
fprintf(stderr, "passing arg to libvncserver: %s\n",
- argv[i]);
+ arg);
}
if (argc2 < 100) {
- argv2[argc2++] = strdup(argv[i]);
+ argv2[argc2++] = strdup(arg);
}
}
}
@@ -4548,6 +5077,7 @@ int main(int argc, char** argv) {
}
if (! quiet) {
+ fprintf(stderr, "\n");
fprintf(stderr, "viewonly: %d\n", view_only);
fprintf(stderr, "shared: %d\n", shared);
fprintf(stderr, "allow: %s\n", allow_list ? allow_list
@@ -4594,9 +5124,9 @@ int main(int argc, char** argv) {
use_dpy ? use_dpy:"null");
exit(1);
} else if (use_dpy) {
- if (! quiet) fprintf(stderr, "Using display %s\n", use_dpy);
+ if (! quiet) fprintf(stderr, "Using X display %s\n", use_dpy);
} else {
- if (! quiet) fprintf(stderr, "Using default display.\n");
+ if (! quiet) fprintf(stderr, "Using default X display.\n");
}
/* check for XTEST */
@@ -4727,7 +5257,7 @@ int main(int argc, char** argv) {
fb = XGetImage(dpy, window, 0, 0, dpy_x, dpy_y, AllPlanes,
ZPixmap);
if (! quiet) {
- fprintf(stderr, "Read initial data from display into"
+ fprintf(stderr, "Read initial data from X display into"
" framebuffer.\n");
}
}
@@ -4751,10 +5281,25 @@ int main(int argc, char** argv) {
initialize_tiles();
+ /* rectangular blackout regions */
+ if (blackout_string != NULL) {
+ initialize_blackout(blackout_string);
+ }
+ if (xinerama) {
+ initialize_xinerama();
+ }
+ if (blackouts) {
+ blackout_tiles();
+ }
+
initialize_shm(); /* also creates XImages when using_shm = 0 */
set_signals();
+ if (blackouts) { /* blackout fb as needed. */
+ copy_screen();
+ }
+
if (use_modifier_tweak) {
initialize_modtweak();
}
@@ -4767,7 +5312,7 @@ int main(int argc, char** argv) {
fflush(stdout);
}
if (! quiet) {
- fprintf(stderr, "screen setup finished.\n");
+ rfbLog("screen setup finished.\n");
}
#if defined(LIBVNCSERVER_HAVE_FORK) && defined(LIBVNCSERVER_HAVE_SETSID)