Commit d43fc7ef authored by runge's avatar runge

x11vnc: mouse -> keystroke and keystroke -> mouse remappings.

parent f5cfa4bc
2004-03-28 Karl Runge <runge@karlrunge.com> 2004-05-05 Karl Runge <runge@karlrunge.com>
* mouse button -> keystrokes and keystroke -> mouse button mappings
in -buttonmap and -remap
* shm OS blacklist revert to -onetile
2004-04-28 Karl Runge <runge@karlrunge.com>
* x11vnc: -auth, more -cursopor and -nofb work * x11vnc: -auth, more -cursopor and -nofb work
2004-03-19 Karl Runge <runge@karlrunge.com> 2004-04-19 Karl Runge <runge@karlrunge.com>
* x11vnc: -cursorpos, -sigpipe * x11vnc: -cursorpos, -sigpipe
2004-03-13 Karl Runge <runge@karlrunge.com> 2004-04-13 Karl Runge <runge@karlrunge.com>
* x11vnc: do not send selection unless all clients * x11vnc: do not send selection unless all clients
are in RFB_NORMAL state. are in RFB_NORMAL state.
* increase rfbMaxClientWait when threaded to avoid * increase rfbMaxClientWait when threaded to avoid
ReadExact() timeouts for some viewers. ReadExact() timeouts for some viewers.
2004-03-10 Karl Runge <runge@karlrunge.com> 2004-04-08 Karl Runge <runge@karlrunge.com>
* x11vnc options -blackout, -xinerama, -xwarppointer * x11vnc options -blackout, -xinerama, -xwarppointer
* modify configure.ac to pick up -lXinerama * modify configure.ac to pick up -lXinerama
* extend -remap to take mapping list. * extend -remap to take mapping list.
......
2004-05-05 Karl Runge <runge@karlrunge.com>
* enable mouse button -> keystrokes mapping in -buttonmap (mousewheel)
* enable keystroke -> mouse button mapping in -remap (touchpad paste)
(-remap incompat ':' -> '-', sorry...)
* shm OS blacklist (i.e. <= SunOS 5.8) -> -onetile
* revert to check_user_input() under -nofb
* cleanup: lastmod, remove tile_shm and update_client_pointer,
debug output, rfbPort failure.
* user friendly last line: 'The VNC desktop is hostname:0'
2004-04-28 Karl Runge <runge@karlrunge.com> 2004-04-28 Karl Runge <runge@karlrunge.com>
* -auth cmdline option for xauthority. * -auth cmdline option for xauthority.
* decrease default deferupdate under -nofb. * decrease default deferupdate under -nofb.
......
...@@ -59,7 +59,7 @@ ...@@ -59,7 +59,7 @@
* can be very painful; one has to modify one's behavior a bit. * can be very painful; one has to modify one's behavior a bit.
* *
* General audio at the remote display is lost unless one separately * General audio at the remote display is lost unless one separately
* sets up some audio side-channel. * sets up some audio side-channel such as esd.
* *
* It does not appear possible to query the X server for the current * 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 shape. We can use XTest to compare cursor to current window's
...@@ -95,8 +95,11 @@ ...@@ -95,8 +95,11 @@
#include <unistd.h> #include <unistd.h>
#include <signal.h> #include <signal.h>
#include <sys/utsname.h>
#include <sys/ipc.h> #include <sys/ipc.h>
#include <sys/shm.h> #include <sys/shm.h>
#include <X11/Xlib.h> #include <X11/Xlib.h>
#include <X11/Xutil.h> #include <X11/Xutil.h>
#include <X11/extensions/XShm.h> #include <X11/extensions/XShm.h>
...@@ -126,6 +129,9 @@ ...@@ -126,6 +129,9 @@
#include <X11/extensions/Xinerama.h> #include <X11/extensions/Xinerama.h>
#endif #endif
/* date +'"lastmod: %Y-%m-%d";' */
char lastmod[] = "lastmod: 2004-05-05";
/* X and rfb framebuffer */ /* X and rfb framebuffer */
Display *dpy = 0; Display *dpy = 0;
...@@ -145,9 +151,6 @@ XImage *scanline; ...@@ -145,9 +151,6 @@ XImage *scanline;
XImage *fullscreen; XImage *fullscreen;
int fs_factor = 0; int fs_factor = 0;
#ifdef SINGLE_TILE_SHM
XShmSegmentInfo tile_shm;
#endif
XShmSegmentInfo *tile_row_shm; /* for all possible row runs */ XShmSegmentInfo *tile_row_shm; /* for all possible row runs */
XShmSegmentInfo scanline_shm; XShmSegmentInfo scanline_shm;
XShmSegmentInfo fullscreen_shm; XShmSegmentInfo fullscreen_shm;
...@@ -333,9 +336,6 @@ void clean_up_exit (int ret) { ...@@ -333,9 +336,6 @@ void clean_up_exit (int ret) {
exit_flag = 1; exit_flag = 1;
/* remove the shm areas: */ /* remove the shm areas: */
#ifdef SINGLE_TILE_SHM
shm_clean(&tile_shm, tile);
#endif
shm_clean(&scanline_shm, scanline); shm_clean(&scanline_shm, scanline);
shm_clean(&fullscreen_shm, fullscreen); shm_clean(&fullscreen_shm, fullscreen);
...@@ -377,9 +377,6 @@ void interrupted (int sig) { ...@@ -377,9 +377,6 @@ void interrupted (int sig) {
* to avoid deadlock, etc, just delete the shm areas and * to avoid deadlock, etc, just delete the shm areas and
* leave the X stuff hanging. * leave the X stuff hanging.
*/ */
#ifdef SINGLE_TILE_SHM
shm_delete(&tile_shm);
#endif
shm_delete(&scanline_shm); shm_delete(&scanline_shm);
shm_delete(&fullscreen_shm); shm_delete(&fullscreen_shm);
...@@ -424,7 +421,6 @@ void set_signals(void) { ...@@ -424,7 +421,6 @@ void set_signals(void) {
if (sigpipe == 1) { if (sigpipe == 1) {
#ifdef SIG_IGN #ifdef SIG_IGN
rfbLog("set_signals: ignoring SIGPIPE\n");
signal(SIGPIPE, SIG_IGN); signal(SIGPIPE, SIG_IGN);
#endif #endif
} else if (sigpipe == 2) { } else if (sigpipe == 2) {
...@@ -771,6 +767,7 @@ void initialize_modtweak() { ...@@ -771,6 +767,7 @@ void initialize_modtweak() {
typedef struct keyremap { typedef struct keyremap {
KeySym before; KeySym before;
KeySym after; KeySym after;
int isbutton;
struct keyremap *next; struct keyremap *next;
} keyremap_t; } keyremap_t;
...@@ -778,22 +775,22 @@ keyremap_t *keyremaps = NULL; ...@@ -778,22 +775,22 @@ keyremap_t *keyremaps = NULL;
void initialize_remap(char *infile) { void initialize_remap(char *infile) {
FILE *in; FILE *in;
char *p, line[256], str1[256], str2[256]; char *p, *q, line[256], str1[256], str2[256];
int i; int i;
KeySym ksym1, ksym2; KeySym ksym1, ksym2;
keyremap_t *remap, *current; keyremap_t *remap, *current;
in = fopen(infile, "r"); in = fopen(infile, "r");
if (in == NULL) { if (in == NULL) {
/* assume cmd line key1:key2,key3:key4 */ /* assume cmd line key1-key2,key3-key4 */
if (! strchr(infile, ':') || (in = tmpfile()) == NULL) { if (! strchr(infile, '-') || (in = tmpfile()) == NULL) {
rfbLog("remap: cannot open: %s\n", infile); rfbLog("remap: cannot open: %s\n", infile);
perror("fopen"); perror("fopen");
clean_up_exit(1); clean_up_exit(1);
} }
p = infile; p = infile;
while (*p) { while (*p) {
if (*p == ':') { if (*p == '-') {
fprintf(in, " "); fprintf(in, " ");
} else if (*p == ',') { } else if (*p == ',') {
fprintf(in, "\n"); fprintf(in, "\n");
...@@ -807,7 +804,7 @@ void initialize_remap(char *infile) { ...@@ -807,7 +804,7 @@ void initialize_remap(char *infile) {
rewind(in); rewind(in);
} }
while (fgets(line, 256, in) != NULL) { while (fgets(line, 256, in) != NULL) {
int blank = 1; int blank = 1, isbtn = 0;
p = line; p = line;
while (*p) { while (*p) {
if (! isspace(*p)) { if (! isspace(*p)) {
...@@ -822,6 +819,10 @@ void initialize_remap(char *infile) { ...@@ -822,6 +819,10 @@ void initialize_remap(char *infile) {
if (strchr(line, '#')) { if (strchr(line, '#')) {
continue; continue;
} }
if ( (q = strchr(line, '-')) != NULL) {
/* allow Keysym1-Keysym2 notation */
*q = ' ';
}
if (sscanf(line, "%s %s", str1, str2) != 2) { if (sscanf(line, "%s %s", str1, str2) != 2) {
rfbLog("remap: bad line: %s\n", line); rfbLog("remap: bad line: %s\n", line);
...@@ -838,6 +839,13 @@ void initialize_remap(char *infile) { ...@@ -838,6 +839,13 @@ void initialize_remap(char *infile) {
} else { } else {
ksym2 = XStringToKeysym(str2); ksym2 = XStringToKeysym(str2);
} }
if (ksym2 == NoSymbol) {
int i;
if (sscanf(str2, "Button%d", &i) == 1) {
ksym2 = (KeySym) i;
isbtn = 1;
}
}
if (ksym1 == NoSymbol || ksym2 == NoSymbol) { if (ksym1 == NoSymbol || ksym2 == NoSymbol) {
rfbLog("warning: skipping bad remap line: %s", line); rfbLog("warning: skipping bad remap line: %s", line);
continue; continue;
...@@ -845,9 +853,10 @@ void initialize_remap(char *infile) { ...@@ -845,9 +853,10 @@ void initialize_remap(char *infile) {
remap = (keyremap_t *) malloc((size_t) sizeof(keyremap_t)); remap = (keyremap_t *) malloc((size_t) sizeof(keyremap_t));
remap->before = ksym1; remap->before = ksym1;
remap->after = ksym2; remap->after = ksym2;
remap->isbutton = isbtn;
remap->next = NULL; remap->next = NULL;
rfbLog("remapping: (%s, 0x%x) -> (%s, 0x%x)\n", str1, ksym1, rfbLog("remapping: (%s, 0x%x) -> (%s, 0x%x) isbtn=%d\n", str1,
str2, ksym2); ksym1, str2, ksym2, isbtn);
if (keyremaps == NULL) { if (keyremaps == NULL) {
keyremaps = remap; keyremaps = remap;
} else { } else {
...@@ -972,9 +981,11 @@ static void modifier_tweak_keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr ...@@ -972,9 +981,11 @@ static void modifier_tweak_keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr
* running under -modtweak. * running under -modtweak.
*/ */
rfbClientPtr last_keyboard_client = NULL; rfbClientPtr last_keyboard_client = NULL;
int num_buttons = -1;
static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) { static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) {
KeyCode k; KeyCode k;
int isbutton = 0;
if (debug_keyboard) { if (debug_keyboard) {
X_LOCK; X_LOCK;
...@@ -993,13 +1004,17 @@ static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) { ...@@ -993,13 +1004,17 @@ static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) {
while (remap != NULL) { while (remap != NULL) {
if (remap->before == keysym) { if (remap->before == keysym) {
keysym = remap->after; keysym = remap->after;
isbutton = remap->isbutton;
if (debug_keyboard) { if (debug_keyboard) {
X_LOCK;
rfbLog("keyboard(): remapping keysym: " rfbLog("keyboard(): remapping keysym: "
"0x%x \"%s\" -> 0x%x \"%s\"\n", "0x%x \"%s\" -> 0x%x \"%s\"\n",
(int) remap->before, (int) remap->before,
XKeysymToString(remap->before), XKeysymToString(remap->before),
(int) remap->after, (int) remap->after,
remap->isbutton ? "button" :
XKeysymToString(remap->after)); XKeysymToString(remap->after));
X_UNLOCK;
} }
break; break;
} }
...@@ -1007,6 +1022,28 @@ static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) { ...@@ -1007,6 +1022,28 @@ static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) {
} }
} }
if (isbutton) {
int button = (int) keysym;
if (! down) {
return; /* nothing to send */
}
if (debug_keyboard) {
rfbLog("keyboard(): remapping keystroke to button %d"
" click\n", button);
}
if (button < 1 || button > num_buttons) {
rfbLog("keyboard(): ignoring mouse button out of "
"bounds: %d\n", button);
return;
}
X_LOCK;
XTestFakeButtonEvent(dpy, button, True, CurrentTime);
XTestFakeButtonEvent(dpy, button, False, CurrentTime);
XFlush(dpy);
X_UNLOCK;
return;
}
if (use_modifier_tweak) { if (use_modifier_tweak) {
modifier_tweak_keyboard(down, keysym, client); modifier_tweak_keyboard(down, keysym, client);
X_LOCK; X_LOCK;
...@@ -1040,16 +1077,178 @@ static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) { ...@@ -1040,16 +1077,178 @@ static void keyboard(rfbBool down, rfbKeySym keysym, rfbClientPtr client) {
/* /*
* pointer event handling routines. * pointer event handling routines.
*/ */
typedef struct ptrremap {
KeySym keysym;
KeyCode keycode;
int end;
int button;
int down;
int up;
} prtremap_t;
MUTEX(pointerMutex); MUTEX(pointerMutex);
#define MAX_BUTTONS 5 #define MAX_BUTTONS 5
int pointer_map[MAX_BUTTONS+1]; #define MAX_BUTTON_EVENTS 50
int num_buttons = -1; prtremap_t pointer_map[MAX_BUTTONS+1][MAX_BUTTON_EVENTS];
char *pointer_remap = NULL; char *pointer_remap = NULL;
void update_pointer(int, int, int); void update_pointer(int, int, int);
void init_pointer(void) {
/* based on IsModifierKey in Xutil.h */
#define ismodkey(keysym) \
((((KeySym)(keysym) >= XK_Shift_L) && ((KeySym)(keysym) <= XK_Hyper_R)))
/*
* For parsing the -buttonmap sections, e.g. "4" or ":Up+Up+Up:"
*/
void buttonparse(int from, char **s) {
char *q;
int to, i;
int modisdown[256];
q = *s;
for (i=0; i<256; i++) {
modisdown[i] = 0;
}
if (*q == ':') {
/* :sym1+sym2+...+symN: format */
int l = 0, n = 0;
char list[1000];
char *t, *kp = q + 1;
KeyCode kcode;
while (*(kp+l) != ':' && *(kp+l) != '\0') {
/* loop to the matching ':' */
l++;
if (l >= 1000) {
rfbLog("buttonparse: keysym list too long: "
"%s\n", q);
break;
}
}
*(kp+l) = '\0';
strncpy(list, kp, l);
list[l] = '\0';
rfbLog("remap button %d using \"%s\"\n", from, list);
/* loop over tokens separated by '+' */
t = strtok(list, "+");
while (t) {
KeySym ksym;
int i;
if (n >= MAX_BUTTON_EVENTS - 20) {
rfbLog("buttonparse: too many button map "
"events: %s\n", list);
break;
}
if (sscanf(t, "0x%x", &i) == 1) {
ksym = (KeySym) i; /* hex value */
} else {
ksym = XStringToKeysym(t); /* string value */
}
if (ksym == NoSymbol) {
/* see if Button<N> "keysym" was used: */
if (sscanf(t, "Button%d", &i) == 1) {
rfbLog(" event %d: button %d\n",
from, n+1, i);
if (i == 0) i = -1; /* bah */
pointer_map[from][n].keysym = NoSymbol;
pointer_map[from][n].keycode = NoSymbol;
pointer_map[from][n].button = i;
pointer_map[from][n].end = 0;
pointer_map[from][n].down = 0;
pointer_map[from][n].up = 0;
} else {
rfbLog("buttonparse: ignoring unknown "
"keysym: %s\n", t);
n--;
}
} else {
kcode = XKeysymToKeycode(dpy, ksym);
pointer_map[from][n].keysym = ksym;
pointer_map[from][n].keycode = kcode;
pointer_map[from][n].button = 0;
pointer_map[from][n].end = 0;
if (! ismodkey(ksym) ) {
/* do both down then up */
pointer_map[from][n].down = 1;
pointer_map[from][n].up = 1;
} else {
if (modisdown[kcode]) {
pointer_map[from][n].down = 0;
pointer_map[from][n].up = 1;
modisdown[kcode] = 0;
} else {
pointer_map[from][n].down = 1;
pointer_map[from][n].up = 0;
modisdown[kcode] = 1;
}
}
rfbLog(" event %d: keysym %s (0x%x) -> "
"keycode 0x%x down=%d up=%d\n", n+1,
XKeysymToString(ksym), ksym, kcode,
pointer_map[from][n].down,
pointer_map[from][n].up);
}
t = strtok(NULL, "+");
n++;
}
/* we must release any modifiers that are still down: */
for (i=0; i<256; i++) {
kcode = (KeyCode) i;
if (n >= MAX_BUTTON_EVENTS) {
rfbLog("buttonparse: too many button map "
"events: %s\n", list);
break;
}
if (modisdown[kcode]) {
pointer_map[from][n].keysym = NoSymbol;
pointer_map[from][n].keycode = kcode;
pointer_map[from][n].button = 0;
pointer_map[from][n].end = 0;
pointer_map[from][n].down = 0;
pointer_map[from][n].up = 1;
modisdown[kcode] = 0;
n++;
}
}
/* advance the source pointer position */
(*s) += l+2;
} else {
/* single digit format */
char str[2];
str[0] = *q;
str[1] = '\0';
to = atoi(str);
if (to < 1) {
rfbLog("skipping invalid remap button \"%d\" for button"
" %d from string \"%s\"\n",
to, from, str);
} else {
rfbLog("remap button %d using \"%s\"\n", from, str);
rfbLog(" button: %d -> %d\n", from, to);
pointer_map[from][0].keysym = NoSymbol;
pointer_map[from][0].keycode = NoSymbol;
pointer_map[from][0].button = to;
pointer_map[from][0].end = 0;
pointer_map[from][0].down = 0;
pointer_map[from][0].up = 0;
}
/* advance the source pointer position */
(*s)++;
}
}
void initialize_pointer_map(void) {
unsigned char map[MAX_BUTTONS]; unsigned char map[MAX_BUTTONS];
int i; int i, k;
/* /*
* This routine counts the number of pointer buttons on the X * This routine counts the number of pointer buttons on the X
* server (to avoid problems, even crashes, if a client has more * server (to avoid problems, even crashes, if a client has more
...@@ -1059,23 +1258,33 @@ void init_pointer(void) { ...@@ -1059,23 +1258,33 @@ void init_pointer(void) {
X_LOCK; X_LOCK;
num_buttons = XGetPointerMapping(dpy, map, MAX_BUTTONS); num_buttons = XGetPointerMapping(dpy, map, MAX_BUTTONS);
X_UNLOCK;
if (num_buttons < 0) { if (num_buttons < 0) {
num_buttons = 0; num_buttons = 0;
} }
/* FIXME: should use info in map[] */ /* FIXME: should use info in map[] */
for (i=1; i<= MAX_BUTTONS; i++) { for (i=1; i<= MAX_BUTTONS; i++) {
pointer_map[i] = i; for (k=0; k < MAX_BUTTON_EVENTS; k++) {
pointer_map[i][k].end = 1;
}
pointer_map[i][0].keysym = NoSymbol;
pointer_map[i][0].keycode = NoSymbol;
pointer_map[i][0].button = i;
pointer_map[i][0].end = 0;
pointer_map[i][0].down = 0;
pointer_map[i][0].up = 0;
} }
if (pointer_remap) { if (pointer_remap) {
/* -buttonmap, format is like: 12-21:2 */ /* -buttonmap, format is like: 12-21=2 */
char *p, *q; char *p, *q, *remap = pointer_remap;
int n; int n;
if ((p = strchr(pointer_remap, ':')) != NULL) {
if ((p = strchr(remap, '=')) != NULL) {
/* undocumented max button number */ /* undocumented max button number */
n = atoi(p+1); n = atoi(p+1);
*p = '\0';
if (n < num_buttons || num_buttons == 0) { if (n < num_buttons || num_buttons == 0) {
num_buttons = n; num_buttons = n;
} else { } else {
...@@ -1084,30 +1293,31 @@ void init_pointer(void) { ...@@ -1084,30 +1293,31 @@ void init_pointer(void) {
num_buttons = n; num_buttons = n;
} }
} }
if ((q = strchr(pointer_remap, '-')) != NULL) { if ((q = strchr(remap, '-')) != NULL) {
/* /*
* The '-' separates the 'from' and 'to' lists, * The '-' separates the 'from' and 'to' lists,
* then it is kind of like tr(1). * then it is kind of like tr(1).
*/ */
char str[2]; char str[2];
int from, to; int from, to;
p = pointer_remap;
rfbLog("remapping pointer buttons using string:\n");
rfbLog(" \"%s\"\n", remap);
p = remap;
q++; q++;
i = 0; i = 0;
str[1] = '\0'; str[1] = '\0';
while (*p != '-') { while (*p != '-') {
int n;
str[0] = *p; str[0] = *p;
from = atoi(str); from = atoi(str);
str[0] = *(q+i); buttonparse(from, &q);
to = atoi(str);
rfbLog("button remap: %d -> %d using: "
"%s\n", from, to, pointer_remap);
pointer_map[from] = to;
p++; p++;
i++;
} }
} }
} }
X_UNLOCK;
} }
/* /*
...@@ -1125,9 +1335,6 @@ static void pointer(int mask, int x, int y, rfbClientPtr client) { ...@@ -1125,9 +1335,6 @@ static void pointer(int mask, int x, int y, rfbClientPtr client) {
return; return;
} }
if (num_buttons < 0) {
init_pointer();
}
if (mask >= 0) { if (mask >= 0) {
/* /*
...@@ -1248,22 +1455,67 @@ void update_pointer(int mask, int x, int y) { ...@@ -1248,22 +1455,67 @@ void update_pointer(int mask, int x, int y) {
last_event = last_input = time(0); last_event = last_input = time(0);
for (i=0; i < MAX_BUTTONS; i++) { for (i=0; i < MAX_BUTTONS; i++) {
/* look for buttons that have be clicked or released: */ /* look for buttons that have be clicked or released: */
if ( (button_mask & (1<<i)) != (mask & (1<<i)) ) { if ( (button_mask & (1<<i)) != (mask & (1<<i)) ) {
if (debug_pointer) { int k;
rfbLog("pointer(): mask change: mask: 0x%x -> " if (debug_pointer) {
"0x%x button: %d\n", button_mask, mask,i+1); rfbLog("pointer(): mask change: mask: 0x%x -> "
"0x%x button: %d\n", button_mask, mask,i+1);
}
for (k=0; k < MAX_BUTTON_EVENTS; k++) {
int bmask = (mask & (1<<i));
if (pointer_map[i+1][k].end) {
break;
} }
mb = pointer_map[i+1];
if (num_buttons && mb > num_buttons) { if (pointer_map[i+1][k].button) {
rfbLog("ignoring mouse button out of bounds: %d" /* sent button up or down */
">%d mask: 0x%x -> 0x%x\n", mb, num_buttons, mb = pointer_map[i+1][k].button;
button_mask, mask); if ((num_buttons && mb > num_buttons)
continue; || mb < 1) {
rfbLog("ignoring mouse button out of "
"bounds: %d>%d mask: 0x%x -> 0x%x\n",
mb, num_buttons, button_mask, mask);
continue;
}
if (debug_pointer) {
rfbLog("pointer(): sending button %d"
" %s (event %d)\n", mb, bmask
? "down" : "up", k+1);
}
XTestFakeButtonEvent(dpy, mb, (mask & (1<<i))
? True : False, CurrentTime);
} else {
/* sent keysym up or down */
KeyCode key = pointer_map[i+1][k].keycode;
int up = pointer_map[i+1][k].up;
int down = pointer_map[i+1][k].down;
if (! bmask) {
/* do not send keysym on button up */
continue;
}
if (debug_pointer) {
rfbLog("pointer(): sending button %d "
"down as keycode 0x%x (event %d)\n",
i+1, key, k+1);
rfbLog(" down=%d up=%d "
"keysym: %s\n", down, up,
XKeysymToString(XKeycodeToKeysym(
dpy, key, 0)));
}
if (down) {
XTestFakeKeyEvent(dpy, key, True,
CurrentTime);
}
if (up) {
XTestFakeKeyEvent(dpy, key, False,
CurrentTime);
}
} }
XTestFakeButtonEvent(dpy, mb,
(mask & (1<<i)) ? True : False, CurrentTime);
} }
}
} }
if (nofb) { if (nofb) {
...@@ -1294,13 +1546,17 @@ void initialize_watch_bell() { ...@@ -1294,13 +1546,17 @@ void initialize_watch_bell() {
int ir, reason; int ir, reason;
if (! XkbSelectEvents(dpy, XkbUseCoreKbd, XkbBellNotifyMask, if (! XkbSelectEvents(dpy, XkbUseCoreKbd, XkbBellNotifyMask,
XkbBellNotifyMask) ) { XkbBellNotifyMask) ) {
fprintf(stderr, "warning: disabling bell.\n"); if (! quiet) {
fprintf(stderr, "warning: disabling bell.\n");
}
watch_bell = 0; watch_bell = 0;
return; return;
} }
if (! XkbOpenDisplay(DisplayString(dpy), &xkb_base_event_type, &ir, if (! XkbOpenDisplay(DisplayString(dpy), &xkb_base_event_type, &ir,
NULL, NULL, &reason) ) { NULL, NULL, &reason) ) {
fprintf(stderr, "warning: disabling bell.\n"); if (! quiet) {
fprintf(stderr, "warning: disabling bell.\n");
}
watch_bell = 0; watch_bell = 0;
} }
} }
...@@ -2056,46 +2312,6 @@ void blackout_nearby_tiles(x, y, dt) { ...@@ -2056,46 +2312,6 @@ void blackout_nearby_tiles(x, y, dt) {
} }
} }
/*
* This is a special workaround to quickly send rfbCursorPosUpdates
* to client(s) when in -nofb mode, e.g. Win2VNC. It sends the updates
* immediately (by-passing libvncserver). Requires a customized Win2VNC
* to interpret the rfbCursorPosUpdates. Currently no big reason to
* do this for non-nofb mode (i.e. normal viewing) and so the normal
* libvncserver mechanism is used. Thanks to Edoardo Tirtarahardja.
*/
void update_client_pointer(rfbClientPtr cl, rfbBool send) {
rfbFramebufferUpdateMsg *fu;
if (! nofb) {
/* normal libvncserver mechanism: */
cl->cursorWasMoved = send;
return;
}
if (! send) {
return;
}
if (cl->state != RFB_NORMAL) {
/* bad idea to force this data to initializing clients */
return;
}
fu = (rfbFramebufferUpdateMsg*) cl->updateBuf;
cl->rfbFramebufferUpdateMessagesSent++;
fu->type = rfbFramebufferUpdate;
fu->nRects = Swap16IfLE(1);
memcpy(&cl->updateBuf[cl->ublen], (char*) fu,
sz_rfbFramebufferUpdateMsg);
cl->ublen = sz_rfbFramebufferUpdateMsg;
rfbSendCursorPos(cl);
if (debug_pointer) {
rfbLog("Sending rfbEncodingPointerPos message to host %s with "
"x=%d,y=%d\n", cl->host, screen->cursorX, screen->cursorY);
}
}
/* /*
* Send rfbCursorPosUpdates back to clients that understand them. This * Send rfbCursorPosUpdates back to clients that understand them. This
* seems to be TightVNC specific. * seems to be TightVNC specific.
...@@ -2134,7 +2350,7 @@ void cursor_pos_updates(int x, int y) { ...@@ -2134,7 +2350,7 @@ void cursor_pos_updates(int x, int y) {
* send a pointer position. * send a pointer position.
*/ */
if (x == cursor_x && y == cursor_y) { if (x == cursor_x && y == cursor_y) {
update_client_pointer(cl, FALSE); cl->cursorWasMoved = FALSE;
} else { } else {
/* an X11 app evidently warped the pointer */ /* an X11 app evidently warped the pointer */
if (debug_pointer) { if (debug_pointer) {
...@@ -2142,11 +2358,11 @@ void cursor_pos_updates(int x, int y) { ...@@ -2142,11 +2358,11 @@ void cursor_pos_updates(int x, int y) {
"detected dx=%3d dy=%3d\n", "detected dx=%3d dy=%3d\n",
cursor_x - x, cursor_y - y); cursor_x - x, cursor_y - y);
} }
update_client_pointer(cl, TRUE); cl->cursorWasMoved = TRUE;
cnt++; cnt++;
} }
} else { } else {
update_client_pointer(cl, TRUE); cl->cursorWasMoved = TRUE;
cnt++; cnt++;
} }
} }
...@@ -3224,11 +3440,6 @@ void initialize_shm() { ...@@ -3224,11 +3440,6 @@ void initialize_shm() {
int i; int i;
/* set all shm areas to "none" before trying to create any */ /* 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.shmid = -1;
scanline_shm.shmaddr = (char *) -1; scanline_shm.shmaddr = (char *) -1;
scanline = NULL; scanline = NULL;
...@@ -3241,14 +3452,6 @@ void initialize_shm() { ...@@ -3241,14 +3452,6 @@ void initialize_shm() {
tile_row[i] = NULL; 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: */ /* the scanline (e.g. 1280x1) shared memory area image: */
if (! shm_create(&scanline_shm, &scanline, dpy_x, 1, "scanline")) { if (! shm_create(&scanline_shm, &scanline, dpy_x, 1, "scanline")) {
...@@ -4659,7 +4862,7 @@ void watch_loop(void) { ...@@ -4659,7 +4862,7 @@ void watch_loop(void) {
if (! use_threads) { if (! use_threads) {
rfbProcessEvents(screen, -1); rfbProcessEvents(screen, -1);
if (! nofb && check_user_input(dt, &cnt)) { if (check_user_input(dt, &cnt)) {
/* true means loop back for more input */ /* true means loop back for more input */
continue; continue;
} }
...@@ -4726,12 +4929,14 @@ void watch_loop(void) { ...@@ -4726,12 +4929,14 @@ void watch_loop(void) {
* to eat as much user input here as we can using some hints from the * to eat as much user input here as we can using some hints from the
* duration of the previous scan_for_updates() call (in dt). * duration of the previous scan_for_updates() call (in dt).
* *
* note: we do this even under -nofb
*
* return of 1 means watch_loop should short-circuit and reloop, * return of 1 means watch_loop should short-circuit and reloop,
* return of 0 means watch_loop should proceed to scan_for_updates(). * return of 0 means watch_loop should proceed to scan_for_updates().
*/ */
int check_user_input(double dt, int *cnt) { int check_user_input(double dt, int *cnt) {
if (old_pointer) { if (old_pointer) {
/* every n-th drops thru to scan */ /* every n-th drops thru to scan */
if ((got_user_input || ui_skip < 0) && *cnt % ui_skip != 0) { if ((got_user_input || ui_skip < 0) && *cnt % ui_skip != 0) {
...@@ -4881,10 +5086,14 @@ void print_help() { ...@@ -4881,10 +5086,14 @@ void print_help() {
"exit as soon as a client disconnects. See -shared and -forever below\n" "exit as soon as a client disconnects. See -shared and -forever below\n"
"to override these protections.\n" "to override these protections.\n"
"\n" "\n"
"For additional info see: http://www.karlrunge.com/x11vnc/\n"
" and http://www.karlrunge.com/x11vnc/#faq\n"
"\n"
"Options:\n" "Options:\n"
"\n" "\n"
"-display disp X11 server display to connect to, the X server process\n" "-display disp X11 server display to connect to, the X server process\n"
" must be running on same machine and support MIT-SHM.\n" " must be running on same machine and support MIT-SHM.\n"
" Equivalent to setting the DISPLAY env. variable.\n"
"-id windowid Show the window corresponding to <windowid> not the\n" "-id windowid Show the window corresponding to <windowid> not the\n"
" entire display. Warning: bugs! new toplevels missed!...\n" " entire display. Warning: bugs! new toplevels missed!...\n"
"-flashcmap In 8bpp indexed color, let the installed colormap flash\n" "-flashcmap In 8bpp indexed color, let the installed colormap flash\n"
...@@ -4920,7 +5129,8 @@ void print_help() { ...@@ -4920,7 +5129,8 @@ void print_help() {
"\n" "\n"
"-noshm Do not use the MIT-SHM extension for the polling.\n" "-noshm Do not use the MIT-SHM extension for the polling.\n"
" remote displays can be polled this way: be careful\n" " remote displays can be polled this way: be careful\n"
" this can use large amounts of network bandwidth.\n" " this can use large amounts of network bandwidth. Also\n"
" of use if machine has a limited number of shm segments.\n"
"-flipbyteorder Sometimes needed if remotely polled host has different\n" "-flipbyteorder Sometimes needed if remotely polled host has different\n"
" endianness. Ignored unless -noshm is set.\n" " endianness. Ignored unless -noshm is set.\n"
"-blackout string Black out rectangles on the screen. string is a comma\n" "-blackout string Black out rectangles on the screen. string is a comma\n"
...@@ -4942,7 +5152,9 @@ void print_help() { ...@@ -4942,7 +5152,9 @@ void print_help() {
"-nomodtweak Send the keysym directly to the X server.\n" "-nomodtweak Send the keysym directly to the X server.\n"
"-remap string Read keysym remappings from file \"string\". Format is\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" " one pair of keysyms per line (can be name or hex value).\n"
" \"string\" can also be of form: key1:key2,key3:key4...\n" " \"string\" can also be of form: key1-key2,key3-key4...\n"
" To map a key to a button click, use the fake keysyms\n"
" \"Button1\", ..., etc. E.g. -remap Super_R-Button2\n"
"-nobell Do not watch for XBell events.\n" "-nobell Do not watch for XBell events.\n"
"-nofb Ignore framebuffer: only process keyboard and pointer.\n" "-nofb Ignore framebuffer: only process keyboard and pointer.\n"
"-nosel Do not manage exchange of X selection/cutbuffer.\n" "-nosel Do not manage exchange of X selection/cutbuffer.\n"
...@@ -4958,8 +5170,24 @@ void print_help() { ...@@ -4958,8 +5170,24 @@ void print_help() {
" on touchscreens or other non-standard setups).\n" " on touchscreens or other non-standard setups).\n"
"-cursorpos Send the X cursor position back to all vnc clients that\n" "-cursorpos Send the X cursor position back to all vnc clients that\n"
" support the TightVNC CursorPosUpdates extension.\n" " support the TightVNC CursorPosUpdates extension.\n"
"-buttonmap str String to remap mouse buttons. Format: IJK-LMN, this\n" "-buttonmap string String to remap mouse buttons. Format: IJK-LMN, this\n"
" maps buttons I -> L, etc., e.g. -buttonmap 13-31\n" " maps buttons I -> L, etc., e.g. -buttonmap 13-31\n"
"\n"
" Button presses can also be mapped to keysyms: replace\n"
" a button digit on the right of the dash with :<sym>:\n"
" or :<sym1>+<sym2>: etc. for multiple keys. For example,\n"
" if the viewing machine has a mouse-wheel (buttons 4 5)\n"
" but the x11vnc side does not, these will do scrolls:\n"
" -buttonmap 12345-123:Prior::Next:\n"
" -buttonmap 12345-123:Up+Up+Up::Down+Down+Down:\n"
"\n"
" If you include a modifier like \"Shift_L\" the modifier's\n"
" up/down state is toggled, e.g. to send \"The\" use\n"
" :Shift_L+t+Shift_L+h+e: (the 1st one is shift down and\n"
" the 2nd one is shift up. (note: the initial state of\n"
" the modifier is ignored and not reset)\n"
" To include button events use \"Button1\", ... etc.\n"
"\n"
"-nodragging Do not update the display during mouse dragging events\n" "-nodragging Do not update the display during mouse dragging events\n"
" (mouse motion with a button held down). Greatly\n" " (mouse motion with a button held down). Greatly\n"
" improves response on slow setups, but you lose all\n" " improves response on slow setups, but you lose all\n"
...@@ -4994,6 +5222,7 @@ void print_help() { ...@@ -4994,6 +5222,7 @@ void print_help() {
" than f, the whole screen is updated (default %.2f).\n" " than f, the whole screen is updated (default %.2f).\n"
"-onetile Do not use the new copy_tiles() framebuffer mechanism,\n" "-onetile Do not use the new copy_tiles() framebuffer mechanism,\n"
" just use 1 shm tile for polling. Same as -old_copytile.\n" " just use 1 shm tile for polling. Same as -old_copytile.\n"
" Limits shm segments used to 3.\n"
"-gaps n Heuristic to fill in gaps in rows or cols of n or less\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" " tiles. Used to improve text paging (default %d).\n"
"-grow n Heuristic to grow islands of changed tiles n or wider\n" "-grow n Heuristic to grow islands of changed tiles n or wider\n"
...@@ -5038,6 +5267,17 @@ void print_help() { ...@@ -5038,6 +5267,17 @@ void print_help() {
* choose a desktop name * choose a desktop name
*/ */
#define MAXN 256 #define MAXN 256
char *this_host() {
char host[MAXN];
#ifdef LIBVNCSERVER_HAVE_GETHOSTNAME
if (gethostname(host, MAXN) == 0) {
return strdup(host);
}
#endif
return NULL;
}
char *choose_title(char *display) { char *choose_title(char *display) {
static char title[(MAXN+10)]; static char title[(MAXN+10)];
strcpy(title, "x11vnc"); strcpy(title, "x11vnc");
...@@ -5051,11 +5291,9 @@ char *choose_title(char *display) { ...@@ -5051,11 +5291,9 @@ char *choose_title(char *display) {
title[0] = '\0'; title[0] = '\0';
if (display[0] == ':') { if (display[0] == ':') {
char host[MAXN]; char host[MAXN];
#ifdef LIBVNCSERVER_HAVE_GETHOSTNAME if (this_host() != NULL) {
if (gethostname(host, MAXN) == 0) { strncpy(title, this_host(), MAXN - strlen(title));
strncpy(title, host, MAXN - strlen(title));
} }
#endif
} }
strncat(title, display, MAXN - strlen(title)); strncat(title, display, MAXN - strlen(title));
if (subwin) { if (subwin) {
...@@ -5068,6 +5306,31 @@ char *choose_title(char *display) { ...@@ -5068,6 +5306,31 @@ char *choose_title(char *display) {
return title; return title;
} }
/*
* check blacklist for OSs with tight shm limits.
*/
int limit_shm(void) {
struct utsname ut;
int limit = 0;
if (uname(&ut) == -1) {
return 0;
}
if (!strcmp(ut.sysname, "SunOS")) {
char *r = ut.release;
if (*r == '5' && *(r+1) == '.') {
if (strchr("2345678", *(r+2)) != NULL) {
limit = 1;
}
}
}
if (limit) {
fprintf(stderr, "reducing shm usage on %s %s (adding "
"-onetile)\n", ut.sysname, ut.release);
}
return limit;
}
int main(int argc, char** argv) { int main(int argc, char** argv) {
XImage *fb; XImage *fb;
...@@ -5078,7 +5341,7 @@ int main(int argc, char** argv) { ...@@ -5078,7 +5341,7 @@ int main(int argc, char** argv) {
int pw_loc = -1; int pw_loc = -1;
int dt = 0; int dt = 0;
int bg = 0; int bg = 0;
int got_waitms = 0, got_rfbwait = 0; int got_rfbwait = 0;
int got_deferupdate = 0, got_defer = 0; int got_deferupdate = 0, got_defer = 0;
/* used to pass args we do not know about to rfbGetScreen(): */ /* used to pass args we do not know about to rfbGetScreen(): */
...@@ -5191,7 +5454,6 @@ int main(int argc, char** argv) { ...@@ -5191,7 +5454,6 @@ int main(int argc, char** argv) {
got_defer = 1; got_defer = 1;
} else if (!strcmp(arg, "-wait")) { } else if (!strcmp(arg, "-wait")) {
waitms = atoi(argv[++i]); waitms = atoi(argv[++i]);
got_waitms = 1;
} else if (!strcmp(arg, "-nap")) { } else if (!strcmp(arg, "-nap")) {
take_naps = 1; take_naps = 1;
#ifdef LIBVNCSERVER_HAVE_LIBPTHREAD #ifdef LIBVNCSERVER_HAVE_LIBPTHREAD
...@@ -5295,10 +5557,6 @@ int main(int argc, char** argv) { ...@@ -5295,10 +5557,6 @@ int main(int argc, char** argv) {
if (waitms < 0) { if (waitms < 0) {
waitms = 0; waitms = 0;
} }
if (! using_shm && ! got_waitms) {
/* try to cut down on polling over network... */
waitms *= 2;
}
if (inetd) { if (inetd) {
shared = 0; shared = 0;
connect_once = 1; connect_once = 1;
...@@ -5311,6 +5569,13 @@ int main(int argc, char** argv) { ...@@ -5311,6 +5569,13 @@ int main(int argc, char** argv) {
argv2[argc2++] = "604800000"; /* one week... */ argv2[argc2++] = "604800000"; /* one week... */
} }
/* check for OS with small shm limits */
if (using_shm && ! single_copytile) {
if (limit_shm()) {
single_copytile = 1;
}
}
if (nofb && ! got_deferupdate && ! got_defer) { if (nofb && ! got_deferupdate && ! got_defer) {
/* reduce defer time under -nofb */ /* reduce defer time under -nofb */
defer_update = defer_update_nofb; defer_update = defer_update_nofb;
...@@ -5322,6 +5587,14 @@ int main(int argc, char** argv) { ...@@ -5322,6 +5587,14 @@ int main(int argc, char** argv) {
argv2[argc2++] = "-deferupdate"; argv2[argc2++] = "-deferupdate";
argv2[argc2++] = strdup(tmp); argv2[argc2++] = strdup(tmp);
} }
if (debug_pointer || debug_keyboard) {
if (bg || quiet) {
fprintf(stderr, "disabling -bg/-q under -debug_pointer"
"/-debug_keyboard\n");
bg = 0;
quiet = 0;
}
}
if (! quiet) { if (! quiet) {
fprintf(stderr, "\n"); fprintf(stderr, "\n");
...@@ -5381,6 +5654,7 @@ int main(int argc, char** argv) { ...@@ -5381,6 +5654,7 @@ int main(int argc, char** argv) {
fprintf(stderr, "tile_fuzz: %d\n", tile_fuzz); fprintf(stderr, "tile_fuzz: %d\n", tile_fuzz);
fprintf(stderr, "use_hints: %d\n", use_hints); fprintf(stderr, "use_hints: %d\n", use_hints);
fprintf(stderr, "bg: %d\n", bg); fprintf(stderr, "bg: %d\n", bg);
fprintf(stderr, "%s\n", lastmod);
} else { } else {
rfbLogEnable(0); rfbLogEnable(0);
} }
...@@ -5589,14 +5863,48 @@ int main(int argc, char** argv) { ...@@ -5589,14 +5863,48 @@ int main(int argc, char** argv) {
if (remap_file != NULL) { if (remap_file != NULL) {
initialize_remap(remap_file); initialize_remap(remap_file);
} }
initialize_pointer_map();
if (screen->rfbPort) { if (! inetd) {
fprintf(stdout, "PORT=%d\n", screen->rfbPort); if (! screen->rfbPort || screen->rfbListenSock < 0) {
fflush(stdout); rfbLog("Error: could not obtain listening port.\n");
clean_up_exit(1);
}
} }
if (! quiet) { if (! quiet) {
rfbLog("screen setup finished.\n"); rfbLog("screen setup finished.\n");
} }
if (screen->rfbPort) {
char *host = this_host();
int port = screen->rfbPort;
if (host != NULL) {
/* note that vncviewer special cases 5900-5999 */
if (quiet) {
if (port >= 5900) {
fprintf(stderr, "The VNC desktop is "
"%s:%d\n", host, port - 5900);
} else {
fprintf(stderr, "The VNC desktop is "
"%s:%d\n", host, port);
}
} else if (port >= 5900) {
rfbLog("The VNC desktop is %s:%d\n", host,
port - 5900);
if (port >= 6000) {
rfbLog("possible aliases: %s:%d, "
"%s::%d\n", host, port, host, port);
}
} else {
rfbLog("The VNC desktop is %s:%d\n", host,
port);
rfbLog("possible alias: %s::%d\n",
host, port);
}
}
fflush(stderr);
fprintf(stdout, "PORT=%d\n", screen->rfbPort);
fflush(stdout);
}
#if defined(LIBVNCSERVER_HAVE_FORK) && defined(LIBVNCSERVER_HAVE_SETSID) #if defined(LIBVNCSERVER_HAVE_FORK) && defined(LIBVNCSERVER_HAVE_SETSID)
if (bg) { if (bg) {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment