diff options
Diffstat (limited to 'x11vnc/userinput.c')
| -rw-r--r-- | x11vnc/userinput.c | 686 | 
1 files changed, 469 insertions, 217 deletions
| diff --git a/x11vnc/userinput.c b/x11vnc/userinput.c index 57a302a..7ee7db3 100644 --- a/x11vnc/userinput.c +++ b/x11vnc/userinput.c @@ -1970,6 +1970,7 @@ void batch_copyregion(sraRegionPtr* region, int *dx, int *dy, int ncr, double de  	rfbClientIteratorPtr i;  	rfbClientPtr cl;  	int k, direct, mode, nrects = 0, bad = 0; +	double start = dnow();  /* XXX Y */ @@ -2039,13 +2040,15 @@ void batch_copyregion(sraRegionPtr* region, int *dx, int *dy, int ncr, double de  	}  	rfbReleaseClientIterator(i); -fprintf(stderr, "batch_copyregion: nrects: %d nregions: %d\n", nrects, ncr); +fprintf(stderr, "batch_copyregion: nrects: %d nregions: %d  dt=%.4f\n", nrects, ncr, dnow() - start);  }  void batch_push(int nreg, double delay) {  	int k;  	batch_copyregion(batch_reg, batch_dxs, batch_dys, nreg, delay); +	/* XXX Y */ +	fb_push();  	for (k=0; k < nreg; k++) {  		sraRgnDestroy(batch_reg[k]);  	} @@ -4157,6 +4160,8 @@ void snap_old(void);  int check_copyrect_raise(int idx, Window orig_frame, int try_batch) {  	char *no = "none";  	int doraise = 1; +	int valid; +	XWindowAttributes attr;  	if (! ncache_wf_raises) {  		doraise = 0; @@ -4177,7 +4182,8 @@ fprintf(stderr, "--YES, wf_raise\n");  		if (try_batch) {  			nb = &nr;  		} -		bs_restore(idx, nb, 0, 1, 1, 1); +		valid = 1; +		bs_restore(idx, nb, &attr, 0, 1, &valid, 1);  		try_to_fix_su(orig_frame, idx, 0x1, nb, NULL);	  		if (nb && nr) {  			batch_push(nr, -1.0); @@ -5906,6 +5912,7 @@ int lookup_old_stack_index(int ic) {  }  #define STORE(k, w, attr) \ +	if (0) fprintf(stderr, "STORE(%d) = 0x%x\n", k, w); \  	cache_list[k].win = w;  \  	cache_list[k].fetched = 1;  \  	cache_list[k].valid = 1;  \ @@ -5917,6 +5924,7 @@ int lookup_old_stack_index(int ic) {  	cache_list[k].time = dnow();  #define CLEAR(k) \ +	if (0) fprintf(stderr, "CLEAR(%d)\n", k); \  	cache_list[k].bs_x = -1;  \  	cache_list[k].bs_y = -1;  \  	cache_list[k].bs_w = -1;  \ @@ -5930,6 +5938,7 @@ int lookup_old_stack_index(int ic) {  	cache_list[k].su_time = 0.0;  #define DELETE(k) \ +	if (0) fprintf(stderr, "DELETE(%d) = 0x%x\n", k, cache_list[k].win); \  	cache_list[k].win = None;  \  	cache_list[k].fetched = 0;  \  	cache_list[k].valid = 0; \ @@ -6878,9 +6887,8 @@ void clip_region(sraRegionPtr r, Window win) {  	}  } -int bs_save(int idx, int *nbatch, int clip, int only_if_tracking, int verb) { +int bs_save(int idx, int *nbatch, XWindowAttributes *attr, int clip, int only_if_tracking, int *valid, int verb) {  	Window win = cache_list[idx].win; -	XWindowAttributes attr;  	int x1, y1, w1, h1;  	int x2, y2, w2, h2;  	int x, y, w, h; @@ -6892,27 +6900,33 @@ int bs_save(int idx, int *nbatch, int clip, int only_if_tracking, int verb) {  	w1 = cache_list[idx].width;  	h1 = cache_list[idx].height; -	if (only_if_tracking && cache_list[idx].bs_x < 0) { -		return; -	} -  if (verb) fprintf(stderr, "backingstore save:       0x%x  %3d clip=%d\n", (unsigned int) win, idx, clip);  	X_LOCK; -	if (! valid_wr(idx, win, &attr)) { +	if (*valid) { +		attr->x = x1; +		attr->y = y1; +		attr->width = w1; +		attr->height = h1; +	} else if (! valid_wr(idx, win, attr)) {  fprintf(stderr, "bs_save:    not a valid X window: 0x%x\n", (unsigned int) win); -/* XXX Y */ -/*		DELETE(idx); */  		X_UNLOCK; +		*valid = 0;  		cache_list[idx].valid = 0;  		return 0; +	} else { +		*valid = 1;  	}  	X_UNLOCK; -	x2 = attr.x; -	y2 = attr.y; -	w2 = attr.width; -	h2 = attr.height; +	if (only_if_tracking && cache_list[idx].bs_x < 0) { +		return 0; +	} + +	x2 = attr->x; +	y2 = attr->y; +	w2 = attr->width; +	h2 = attr->height;  	if (cache_list[idx].bs_x < 0) {  		rc = find_rect(idx, x2, y2, w2, h2); @@ -6927,7 +6941,9 @@ fprintf(stderr, "bs_save:    not a valid X window: 0x%x\n", (unsigned int) win);  	h = cache_list[idx].bs_h;  	if (x < 0 || ! rc) { -		STORE(idx, win, attr); +//		if (!novalidate) { +//			STORE(idx, win, attr); +//		}  fprintf(stderr, "BS_save: FAIL FOR: %d\n", idx);  		return 0;  	} @@ -6968,15 +6984,16 @@ if (verb) fprintf(stderr, "BS_save: %.4f %.2f %d done.  %dx%d+%d+%d %dx%d+%d+%d  	sraRgnDestroy(r0);  	sraRgnDestroy(r); -	STORE(idx, win, attr); +//	if (!novalidate) { +//		STORE(idx, win, attr); +//	}  	cache_list[idx].bs_time = dnow();  	return 1;  } -int su_save(int idx, int *nbatch, int clip, int verb) { +int su_save(int idx, int *nbatch, XWindowAttributes *attr, int clip, int *valid, int verb) {  	Window win = cache_list[idx].win; -	XWindowAttributes attr;  	int x1, y1, w1, h1;  	int x2, y2, w2, h2;  	int x, y, w, h; @@ -6991,20 +7008,26 @@ if (verb) fprintf(stderr, "save-unders save:        0x%x  %3d \n", (unsigned int  	h1 = cache_list[idx].height;  	X_LOCK; -	if (! valid_wr(idx, win, &attr)) { +	if (*valid) { +		attr->x = x1; +		attr->y = y1; +		attr->width = w1; +		attr->height = h1; +	} else if (! valid_wr(idx, win, attr)) {  fprintf(stderr, "su_save:    not a valid X window: 0x%x\n", (unsigned int) win); -/* XXX Y */ -/*		DELETE(idx); */  		X_UNLOCK; +		*valid = 0;  		cache_list[idx].valid = 0;  		return 0; +	} else { +		*valid = 1;  	}  	X_UNLOCK; -	x2 = attr.x; -	y2 = attr.y; -	w2 = attr.width; -	h2 = attr.height; +	x2 = attr->x; +	y2 = attr->y; +	w2 = attr->width; +	h2 = attr->height;  	if (cache_list[idx].bs_x < 0) {  		rc = find_rect(idx, x2, y2, w2, h2); @@ -7018,7 +7041,9 @@ fprintf(stderr, "su_save:    not a valid X window: 0x%x\n", (unsigned int) win);  	h = cache_list[idx].su_h;  	if (x < 0 || ! rc) { -		STORE(idx, win, attr); +//		if (!novalidate) { +//			STORE(idx, win, attr); +//		}  fprintf(stderr, "SU_save: FAIL FOR: %d\n", idx);  		return 0;  	} @@ -7059,15 +7084,16 @@ if (verb) fprintf(stderr, "SU_save: %.4f %.2f %d done.  %dx%d+%d+%d %dx%d+%d+%d  	sraRgnDestroy(r0);  	sraRgnDestroy(r); -	STORE(idx, win, attr); +//	if (!novalidate) { +//		STORE(idx, win, attr); +//	}  	cache_list[idx].su_time = dnow();  	return 1;  } -int bs_restore(int idx, int *nbatch, int clip, int nopad, int novalidate, int verb) { +int bs_restore(int idx, int *nbatch, XWindowAttributes *attr, int clip, int nopad, int *valid, int verb) {  	Window win = cache_list[idx].win; -	XWindowAttributes attr;  	int x1, y1, w1, h1;  	int x2, y2, w2, h2;  	int x, y, w, h; @@ -7082,23 +7108,25 @@ if (verb) fprintf(stderr, "backingstore restore:    0x%x  %3d \n", (unsigned int  	h1 = cache_list[idx].height;  	X_LOCK; -	if (novalidate) { -		attr.x = x1; -		attr.y = y1; -		attr.width = w1; -		attr.height = h1; -	} else if (! valid_wr(idx, win, &attr)) { +	if (*valid) { +		attr->x = x1; +		attr->y = y1; +		attr->width = w1; +		attr->height = h1; +	} else if (! valid_wr(idx, win, attr)) {  fprintf(stderr, "BS_restore: not a valid X window: 0x%x\n", (unsigned int) win); -		DELETE(idx); +		*valid = 0;  		X_UNLOCK;  		return 0; +	} else { +		*valid = 1;  	}  	X_UNLOCK; -	x2 = attr.x; -	y2 = attr.y; -	w2 = attr.width; -	h2 = attr.height; +	x2 = attr->x; +	y2 = attr->y; +	w2 = attr->width; +	h2 = attr->height;  	x = cache_list[idx].bs_x;  	y = cache_list[idx].bs_y; @@ -7106,9 +7134,9 @@ fprintf(stderr, "BS_restore: not a valid X window: 0x%x\n", (unsigned int) win);  	h = cache_list[idx].bs_h;  	if (x < 0 || cache_list[idx].bs_time == 0.0) { -		if (!novalidate) { -			STORE(idx, win, attr); -		} +//		if (!novalidate) { +//			STORE(idx, win, attr); +//		}  		return 0;  	} @@ -7162,22 +7190,20 @@ if (verb) fprintf(stderr, "BS_rest: %.4f %.2f %d done.  %dx%d+%d+%d %dx%d+%d+%d  	sraRgnDestroy(r0);  	sraRgnDestroy(r); -	if (!novalidate) { -		STORE(idx, win, attr); -	} +//	if (!novalidate) { +//		STORE(idx, win, attr); +//	}  	return 1;  } -int su_restore(int idx, int *nbatch, int clip, int nopad, int verb) { +int su_restore(int idx, int *nbatch, XWindowAttributes *attr, int clip, int nopad, int *valid, int verb) {  	Window win = cache_list[idx].win; -	XWindowAttributes attr;  	int x1, y1, w1, h1;  	int x2, y2, w2, h2;  	int x, y, w, h;  	int dx, dy;  	sraRegionPtr r, r0; -	int invalid = 0;  if (verb) fprintf(stderr, "save-unders  restore:    0x%x  %3d \n", (unsigned int) win, idx); @@ -7187,18 +7213,24 @@ if (verb) fprintf(stderr, "save-unders  restore:    0x%x  %3d \n", (unsigned int  	h1 = cache_list[idx].height;  	X_LOCK; -	if (! valid_wr(idx, win, &attr)) { +	if (*valid) { +		attr->x = x1; +		attr->y = y1; +		attr->width = w1; +		attr->height = h1; +	} else if (! valid_wr(idx, win, attr)) {  fprintf(stderr, "SU_restore: not a valid X window: 0x%x\n", (unsigned int) win); -		invalid = 1; +		*valid = 0;  		x2 = x1;  		y2 = y1;  		w2 = w1;  		h2 = h1;  	} else { -		x2 = attr.x; -		y2 = attr.y; -		w2 = attr.width; -		h2 = attr.height; +		x2 = attr->x; +		y2 = attr->y; +		w2 = attr->width; +		h2 = attr->height; +		*valid = 1;  	}  	X_UNLOCK; @@ -7209,9 +7241,9 @@ fprintf(stderr, "SU_restore: not a valid X window: 0x%x\n", (unsigned int) win);  	if (x < 0 || cache_list[idx].bs_x < 0 || cache_list[idx].su_time == 0.0) {  fprintf(stderr, "SU_rest: su_x/bs_x/su_time: %d %d %.3f\n", x, cache_list[idx].bs_x, cache_list[idx].su_time); -		if (invalid) { -			DELETE(idx); -		} +//		if (invalid) { +//			DELETE(idx); +//		}  		return 0;  	} @@ -7265,11 +7297,11 @@ if (verb) fprintf(stderr, "SU_rest: %.4f %.2f %d done.  %dx%d+%d+%d %dx%d+%d+%d  	sraRgnDestroy(r0);  	sraRgnDestroy(r); -	if (invalid) { -		DELETE(idx); -	} else { -		STORE(idx, win, attr); -	} +//	if (invalid) { +//		DELETE(idx); +//	} else if (!novalidate) { +//		STORE(idx, win, attr); +//	}  	return 1;  } @@ -7357,10 +7389,10 @@ void block_stats(void) {  				double t2 = cache_list[k].bs_time;  				if (t1 > 0.0) {t1 = dnow() - t1;} else {t1 = -1.0;}  				if (t2 > 0.0) {t2 = dnow() - t2;} else {t2 = -1.0;} -				fprintf(stderr, "     [%02d] %04d 0x%08x bs: %04dx%04d+%04d+%05d vw: %04dx%04d+%04d+%04d cl: %04dx%04d+%04d+%04d map=%d su=%9.3f bs=%9.3f\n", +				fprintf(stderr, "     [%02d] %04d 0x%08x bs: %04dx%04d+%04d+%05d vw: %04dx%04d+%04d+%04d cl: %04dx%04d+%04d+%04d map=%d su=%9.3f bs=%9.3f cnt=%d/%d\n",  				    n, k, (unsigned int) win, w, h, x, y, attr.width, attr.height, attr.x, attr.y,  				    cache_list[k].width, cache_list[k].height, cache_list[k].x, cache_list[k].y, -				    attr.map_state == IsViewable, t1, t2);  +				    attr.map_state == IsViewable, t1, t2, cache_list[k].create_cnt, cache_list[k].map_cnt);   			}  		}  		frac = area /(dpy_x * dpy_y); @@ -7638,13 +7670,13 @@ int try_to_fix_su(Window win, int idx, Window above, int *nbatch, char *mode) {  	int unmapped = 0;  	int moved = 0; -fprintf(stderr, "TRY_TO_FIX_SU(%d)  0x%x  0x%x unmapped=%d\n", idx, win, above, unmapped);  	if (mode && !strcmp(mode, "unmapped")) {  		unmapped = 1;  	} else if (mode && !strcmp(mode, "moved")) {  		moved = 1;  	} +fprintf(stderr, "TRY_TO_FIX_SU(%d)  0x%x  0x%x unmapped=%d\n", idx, win, above, unmapped);  	if (idx < 0) {  		return 0; @@ -7851,25 +7883,176 @@ sraRegionPtr idx_create_rgn(sraRegionPtr r0, int idx) {  	return rtmp;  } +#define EVLISTMAX 256 +#define EV_RESET	0 +#define EV_CREATE	1 +#define EV_DESTROY	2 +#define EV_UNMAP	3  +#define EV_MAP		4  +#define EV_REPARENT	5  +#define EV_CONFIGURE	6  +#define EV_VISIBILITY_UNOBS	7  +#define EV_VISIBILITY_OBS	8  +#define EV_PROPERTY	9  +Window _ev_list[EVLISTMAX]; +int _ev_case[EVLISTMAX]; +int _ev_list_cnt; + +int n_CN = 0, n_RN = 0, n_DN = 0, n_ON = 0, n_MN = 0, n_UN = 0; +int n_VN = 0, n_VN_p = 0, n_VN_u = 0, n_ST = 0, n_PN = 0; + +int ev_store(Window win, int type) { +	if (type == EV_RESET)  { +		n_CN = 0; n_RN = 0; n_DN = 0; n_ON = 0; n_MN = 0; n_UN = 0; +		n_VN = 0; n_VN_p = 0; n_VN_u = 0; n_ST = 0; n_PN = 0; +		_ev_list_cnt = 0; +		return 1; +	} +	if (_ev_list_cnt >= EVLISTMAX) { +		return 0; +	} +	_ev_list[_ev_list_cnt] = win; +	_ev_case[_ev_list_cnt++] = type; +} + +int ev_lookup(Window win, int type) { +	int i; +	for(i=0; i < _ev_list_cnt; i++) { +		if (_ev_list[i] == win && _ev_case[i] == type) 	{ +			return 1; +		} +	} +	return 0; +} + +unsigned long all_ev = SubstructureNotifyMask|StructureNotifyMask|VisibilityChangeMask; +unsigned long win_ev = StructureNotifyMask|VisibilityChangeMask; + +void read_events(int *n_in) { +	int n = *n_in; +	Window win, win2; +	XEvent ev; +	 +	while (xcheckmaskevent(dpy, all_ev, &Ev[n])) { +		int type = Ev[n].type;  +		Window w = None; +		win = Ev[n].xany.window; +		Ev_done[n] = 0; +		Ev_area[n] = 0; +		Ev_win[n] = win; +		Ev_map[n] = None; +		Ev_unmap[n] = None; +		Ev_order[n] = n; + +		ev = Ev[n]; + +if (type == DestroyNotify)  w = Ev[n].xcreatewindow.window; +if (type == CreateNotify)   w = Ev[n].xdestroywindow.window; +if (type == ReparentNotify) w = Ev[n].xreparent.window; +if (type == UnmapNotify)    w = Ev[n].xunmap.window; +if (type == MapNotify)      w = Ev[n].xmap.window; +if (type == Expose)         w = Ev[n].xexpose.window; +if (type == ConfigureNotify) w = Ev[n].xconfigure.window; +if (n == *n_in) fprintf(stderr, "\n"); +fprintf(stderr, "----- %d inputev 0x%08x w: 0x%08x %s\n", n, win, w, Etype(type)); + +		if (win == rootwin) { +			if (type == CreateNotify) { +				win2 = ev.xcreatewindow.window; +				ev_store(win2, EV_CREATE); +				n++; +				n_CN++; +			} else if (type == ReparentNotify) { +				if (ev.xreparent.parent != rootwin) { +					win2 = ev.xreparent.window; +					if (win2 != rootwin) { +						ev_store(win2, EV_REPARENT); +					} +				} +				n++; +				n_RN++; +			} else if (type == PropertyNotify) { +				set_prop_atom(Ev[n].xproperty.atom); +				n++; +				n_PN++; +			} else if (type == MapNotify) { +				win2 = ev.xmap.window; +				ev_store(win2, EV_MAP); +				n++; +				n_CN++; +			} else { +				/* skip rest */ +#if 0 +				Window w = None; +if (type == DestroyNotify) w = Ev[n].xdestroywindow.window; +if (type == UnmapNotify)   w = Ev[n].xunmap.window; +if (type == MapNotify)     w = Ev[n].xmap.window; +if (type == Expose)        w = Ev[n].xexpose.window; +if (type == ConfigureNotify) w = Ev[n].xconfigure.window; +if (type != ConfigureNotify) fprintf(stderr, "root: skip %s  for 0x%x\n", Etype(type), (unsigned int) w); +#endif + +			} +		} else { +			if (type == ReparentNotify) { +				ev_store(win, EV_REPARENT); +				n++; +				n_RN++; +			} else if (type == DestroyNotify) { +				ev_store(win, EV_DESTROY); +				n++; +				n_DN++; +			} else if (type == ConfigureNotify) { +				ev_store(win, EV_CONFIGURE); +				n++; +				n_ON++; +			} else if (type == VisibilityNotify) { +				if (Ev[n].xvisibility.state == VisibilityUnobscured) { +					ev_store(win, EV_VISIBILITY_UNOBS); +					n_VN_u++; +				} else { +					ev_store(win, EV_VISIBILITY_OBS); +					n_VN_p++; +				} +				n++; +				n_VN++; +			} else if (type == MapNotify) { +				ev_store(win, EV_MAP); +				Ev_map[n] = win; +				n++; +				n_MN++; +			} else if (type == UnmapNotify) { +				ev_store(win, EV_UNMAP); +				Ev_unmap[n] = win; +				n++; +				n_UN++; +			} else { +				/* skip rest */ +fprintf(stderr, "----- skip %s\n", Etype(type)); +			} +		} +		if (n >= EVMAX) { +			break; +		} +	} +	*n_in = n; +} +  int check_ncache(int reset, int mode) {  	static double last_root = 0.0;  	static int first = 1;  	static int last_client_count = -1;  	int i, k, n;  -	int n_CN = 0, n_RN = 0, n_DN = 0, n_ON = 0, n_MN = 0, n_UN = 0; -	int n_VN = 0, n_VN_p = 0, n_VN_u = 0, n_ST = 0, n_PN = 0;  	int did_sched = 0;  	double now, refresh = 60.0;  	Window win, win2;  	XWindowAttributes attr; -	unsigned long all_ev = SubstructureNotifyMask|StructureNotifyMask|VisibilityChangeMask; -	unsigned long win_ev = StructureNotifyMask|VisibilityChangeMask; - +	int valid;  	int try_batch = 1; /* XXX Y */  	int use_batch = 0;  	int nreg = 0, *nbatch; -	int create_cnt, create_tot; +	int create_cnt;  	int pixels = 0, ttot;  	int desktop_change = 0, n1, n2;  	static int saw_desktop_change = 0; @@ -7880,10 +8063,12 @@ int check_ncache(int reset, int mode) {  	sraRegionPtr missed_bs_restore_rgn;  	int nrects = 0; +	int nsave, nxsel; +	int did_vis_snap = 0;  	int skipwins_n = 0; -	int skipwins_max = 16; -	Window skipwins[16]; +	int skipwins_max = 256; +	Window skipwins[256];  	if (unixpw_in_progress) return -1; @@ -7995,7 +8180,7 @@ if (c) fprintf(stderr, "check_ncache purged %d events\n", c);  			    "This is the Pixel buffer cache region. Your VNC Viewer is not hiding it from you.",  			    white_pixel());  			rfbDrawString(screen, &default8x16Font, dx, ds + Dy+2*dy, -			    "Try resizing your VNC Viewer so you don't see it !!", +			    "Try resizing your VNC Viewer so you don't see it!! Pay no attention to the man behind the curtain...",  			    white_pixel());  			rfbDrawString(screen, &default8x16Font, dx, ds + Dy+3*dy,  			    "To disable run the server with:  x11vnc -ncache 0 ...", @@ -8045,13 +8230,13 @@ fprintf(stderr, "\n**** checking cache_list[%d]\n\n", cache_list_num);  		block_stats();  		for(k=0; k<cache_list_num; k++) { -			int valid = 0; +			valid = 0;  			win = cache_list[k].win;   			X_LOCK;  			if (win == None) {  				;  			} else if (cache_list[k].selectinput && cache_list[k].time > now - refresh) { -				; +				valid = 1;  			} else if (valid_window(win, &attr, 1)) {  				STORE(k, win, attr);  				if (! cache_list[k].selectinput) { @@ -8069,7 +8254,7 @@ fprintf(stderr, "DELETE(%d) %dx%d+%d+%d\n", k, cache_list[k].width, cache_list[k  			X_UNLOCK;  /* XXX Y */  			if (valid) { -				if (cache_list[k].create_cnt && attr.map_state != IsViewable && cache_list[k].map_cnt == 0) { +				if (cache_list[k].create_cnt && cache_list[k].map_state != IsViewable && cache_list[k].map_cnt == 0) {  					if (cache_list[k].bs_x >= 0) {  fprintf(stderr, "Created window never mapped: freeing(%d) 0x%x\n", k, (unsigned int) win);  						free_rect(k); @@ -8129,9 +8314,17 @@ fprintf(stderr, "Created window never mapped: freeing(%d) 0x%x\n", k, (unsigned  					} else if (aw * ah < 64 * 64) {  						;  					} else { -//fprintf(stderr, "*SNAP BS_save: 0x%x %d %d %d\n", (unsigned int) win, aw, ah, cache_list[idx].map_state);  -						bs_save(idx, bat, 1, 0, 0); +fprintf(stderr, "*SNAP BS_save: 0x%x %d %d %d\n", (unsigned int) win, aw, ah, cache_list[idx].map_state);  +						valid = 0; +						bs_save(idx, bat, &attr, 1, 0, &valid, 0); +						if (valid) { +							STORE(idx, win, attr); +						} else { +							DELETE(idx); +						}  					} +				} else { +fprintf(stderr, "*SCHED LOOKUP FAIL: i=%d 0x%x\n", i, win);  				}  			}  			sched_bs[i] = None; @@ -8139,12 +8332,21 @@ fprintf(stderr, "Created window never mapped: freeing(%d) 0x%x\n", k, (unsigned  		did_sched = 1;  		if (now > last_sched_vis + 3.0 && now > last_wireframe + 2.0) { +			static double last_vis = 0.0; +			int vis_now[32], top_now[32]; +			static int vis_prev[32]; +			int diff, nv = 32, vis_now_n = 0; +			Window win; + +			did_vis_snap = 1;  			for (i=0; i < cache_list_num; i++) { -				Window win = cache_list[i].win;  				int ok = 0;  				int top_only = 1;  				int aw = cache_list[i].width;   				int ah = cache_list[i].height;  +				int map_prev = cache_list[i].map_state; + +				win = cache_list[i].win;  				if (saw_desktop_change) {  					top_only = 0; @@ -8163,6 +8365,10 @@ fprintf(stderr, "Created window never mapped: freeing(%d) 0x%x\n", k, (unsigned  				if (cache_list[i].map_state != IsViewable) {  					continue;  				} +				if (map_prev != IsViewable) { +					/* we hope to catch it below in the normal event processing */ +					continue; +				}  				if (aw * ah < 64 * 64) {  					continue;  				} @@ -8176,8 +8382,33 @@ fprintf(stderr, "Created window never mapped: freeing(%d) 0x%x\n", k, (unsigned  					ok = 1;  				}  				if (ok) { -					bs_save(i, bat, !top_only, 0, 0); +					if (vis_now_n < nv) { +						vis_now[vis_now_n] = i; +						top_now[vis_now_n++] = top_only; +					} +				} +			} +			diff = 0; +			for (k = 0; k < vis_now_n; k++) { +				if (vis_now[k] != vis_prev[k]) { +					diff = 1; +				} +			} +			if (diff || now > last_vis + 45.0) { +				for (k = 0; k < vis_now_n; k++) { +					i = vis_now[k]; +					win = cache_list[i].win; +					valid = 0; +fprintf(stderr, "*VIS  BS_save: 0x%x %d %d %d\n", win, cache_list[i].width, cache_list[i].height, cache_list[i].map_state);  +					bs_save(i, bat, &attr, !top_now[k], 0, &valid, 0); +					if (valid) { +						STORE(i, win, attr); +					} else { +						DELETE(i); +					} +					vis_prev[k] = vis_now[k];  				} +				last_vis = dnow();  			}  			last_sched_vis = dnow();  			saw_desktop_change = 0; @@ -8191,78 +8422,74 @@ fprintf(stderr, "Created window never mapped: freeing(%d) 0x%x\n", k, (unsigned  	n = 0;  	ttot = 0; -	create_tot = 0; + +	ev_store(None, EV_RESET); +  	X_LOCK; -	while (xcheckmaskevent(dpy, all_ev, &Ev[n])) { -		int type = Ev[n].type;  -		win = Ev[n].xany.window; -		Ev_done[n] = 0; -		Ev_area[n] = 0; -		Ev_win[n] = win; -		Ev_map[n] = None; -		Ev_unmap[n] = None; -		Ev_order[n] = n; +	for (k = 1; k <= 3; k++) { -//fprintf(stderr, "----- %d/%d inputev 0x%x %s\n", n, ttot++, win, Etype(type)); +		nsave = n; -		if (win == rootwin) { -			if (type == CreateNotify) { -				create_tot++; -				n++; -				n_CN++; -			} else if (type == ReparentNotify) { -				n++; -				n_RN++; -			} else if (type == PropertyNotify) { -				set_prop_atom(Ev[n].xproperty.atom); -				n++; -				n_PN++; -			} else { -				/* skip rest */ -#if 0 -				Window w = None; -if (type == DestroyNotify) w = Ev[n].xdestroywindow.window; -if (type == UnmapNotify)   w = Ev[n].xunmap.window; -if (type == MapNotify)     w = Ev[n].xmap.window; -if (type == Expose)        w = Ev[n].xexpose.window; -if (type == ConfigureNotify) w = Ev[n].xconfigure.window; -if (type != ConfigureNotify) fprintf(stderr, "root: skip %s  for 0x%x\n", Etype(type), (unsigned int) w); -#endif +		if (k > 1) fprintf(stderr, "read_events-%d\n", k); +		read_events(&n); -			} -		} else { -			if (type == ReparentNotify) { -				n++; -				n_RN++; -			} else if (type == DestroyNotify) { -				n++; -				n_DN++; -			} else if (type == ConfigureNotify) { -				n++; -				n_ON++; -			} else if (type == VisibilityNotify) { -				if (Ev[n].xvisibility.state == VisibilityUnobscured) { -					n_VN_u++; +		nxsel = 0; +			 +		/* handle creates and reparenting: */ +		for (n1 = nsave; n1 < n; n1++) { +			Window win2; +			int idx; +			XEvent ev = Ev[n1]; +			win = Ev_win[n1]; +			if (ev.type == CreateNotify) { +				win2 = ev.xcreatewindow.window; +				if (ev_lookup(win2, EV_REPARENT) || ev_lookup(win2, EV_DESTROY)) { +					if (skipwins_n < skipwins_max) { +fprintf(stderr, "SKIPWINS: CreateNotify: 0x%x %d\n", win2, n1); +						skipwins[skipwins_n++] = win2; +					}  				} else { -					n_VN_p++; +					idx = lookup_win_index(win); +					if (idx < 0) { +						idx = lookup_free_index(); +						if (idx < 0) { +							continue; +						} +						CLEAR(idx); +					} +fprintf(stderr, "PRELOOP:  CreateNotify: 0x%x %d valid_window\n", win2, n1); +					if (valid_window(win2, &attr, 1)) { +						STORE(idx, win2, attr); +						CLEAR(idx); +						cache_list[idx].selectinput = 1; +						cache_list[idx].create_cnt = 1; +fprintf(stderr, "PRELOOP:  CreateNotify: 0x%x %d xselectinput\n", win2, n1); +						xselectinput(win2, win_ev, 1); +						nxsel++; +					} else { +						DELETE(idx); +					} +					nxsel++; +				} +			} else if (ev.type == ReparentNotify) { +				if (ev.xreparent.parent != rootwin) { +					win2 = ev.xreparent.window; +					if (win2 != rootwin) { +						idx = lookup_win_index(win2); +fprintf(stderr, "PRELOOP:  RepartNotify: 0x%x %d idx=%d\n", win2, n1, idx); +						if (idx >= 0) { +							DELETE(idx); +						} +						if (! ev_lookup(win2, EV_CREATE)) { +							xselectinput(win2, 0, 1); +							nxsel++; +						} +					}  				} -				n++; -				n_VN++; -			} else if (type == MapNotify) { -				Ev_map[n] = win; -				n++; -				n_MN++; -			} else if (type == UnmapNotify) { -				Ev_unmap[n] = win; -				n++; -				n_UN++; -			} else { -				/* skip rest */ -fprintf(stderr, "----- skip %s\n", Etype(type));  			}  		} -		if (n >= EVMAX) { +		if (nxsel == 0) {  			break;  		}  	} @@ -8281,7 +8508,8 @@ fprintf(stderr, "----- skip %s\n", Etype(type));  	if (n == 0) {  		return 0;  	} -fprintf(stderr, "\n"); rfbLog("IN  check_ncache() %d events.\n", n); +fprintf(stderr, "\n"); rfbLog("IN  check_ncache() %d events.  %.4f\n", n, now - x11vnc_start); +if (did_vis_snap) fprintf(stderr, "VIS snapshot all %.4f\n", dnowx());  	if (try_batch) {  		use_batch = 1; @@ -8310,6 +8538,7 @@ fprintf(stderr, "\n"); rfbLog("IN  check_ncache() %d events.\n", n);  		for (n2 = 0; n2 < n; n2++) {  			if (Ev_unmap[n2] == twin) {  				if (skipwins_n < skipwins_max) { +fprintf(stderr, "SKIPWINS: Ev_unmap/map: 0x%x %d\n", twin, n2);  					skipwins[skipwins_n++] = twin;  					break;  				} @@ -8435,6 +8664,10 @@ fprintf(stderr, "\n"); rfbLog("IN  check_ncache() %d events.\n", n);  	missed_bs_restore_rgn = sraRgnCreate();  	r0 = sraRgnCreateRect(0, 0, dpy_x, dpy_y); +for (k = 0; k < skipwins_n; k++) { +	fprintf(stderr, "skipwins[%d] 0x%x\n", k, skipwins[k]); +} +  	X_LOCK;  	for (i=0; i < n; i++) {  		XEvent ev; @@ -8448,57 +8681,50 @@ fprintf(stderr, "\n"); rfbLog("IN  check_ncache() %d events.\n", n);  		type = ev.type;  		Ev_done[ik] = 1; +		win2 = win; +		if (win == rootwin) { +			if (type == CreateNotify) { +				win2 = ev.xcreatewindow.window; +			} else if (type == ReparentNotify) { +				win2 = ev.xreparent.window; +			} +		}  		for (ns = 0; ns < skipwins_n; ns++) { -			if (win == skipwins[ns]) { +			if (win2 == skipwins[ns]) {  				skip = 1;  				break;  			}  		}  		if (skip) { -fprintf(stderr, "skip%02d: ** SpecialSkip   0x%x type: %s\n", ik, (unsigned int) win, Etype(type)); +fprintf(stderr, "skip%02d: ** SpecialSkip   0x%x/0x%x type: %s\n", ik, (unsigned int) win, (unsigned int) win2, Etype(type));  			continue;  		}  		if (win == rootwin) {  			if (type == CreateNotify) {  				int x=0, y=0, w=0, h=0; -				int valid = 0; +				valid = 0;  				win2 = ev.xcreatewindow.window;  				idx = lookup_win_index(win2);  				if (idx < 0) { -					idx = lookup_free_index(); -					if (idx < 0) { -						continue; -					} -					CLEAR(idx); +					continue;  				} -				if (valid_window(win2, &attr, 1)) { -					STORE(idx, win2, attr); -					CLEAR(idx); -					x=attr.x; -					y=attr.y; -					w=attr.width; -					h=attr.height; -					if (create_tot <= 6 && create_cnt++ < 3) { -						if (w*h > 64 * 64) { -							X_UNLOCK; -							su_save(idx, nbatch, 0, 1); -							X_LOCK; -							if (cache_list[idx].valid) { -								if (! desktop_change) { -									SCHED(win2, 1)  -								} -							} -							create_cnt++; -						} -					} -					if (cache_list[idx].valid) { -						xselectinput(win2, win_ev, 1); -						cache_list[idx].selectinput = 1; -						cache_list[idx].create_cnt = 1; +				if (cache_list[idx].valid) { +					valid = 1; +					x=cache_list[idx].x; +					y=cache_list[idx].y; +					w=cache_list[idx].width; +					h=cache_list[idx].height; +					if (w*h > 64 * 64 && ev_lookup(win2, EV_MAP)) { +						X_UNLOCK;  						valid = 1; -					} else { -						DELETE(idx); +						su_save(idx, nbatch, &attr, 0, &valid, 1); +						STORE(idx, win2, attr); +						X_LOCK; +						if (! desktop_change) { +							SCHED(win2, 1)  +						} +						create_cnt++;  					}  				}  fprintf(stderr, "root%02d: ** CreateNotify  0x%x  %3d  -- %dx%d+%d+%d valid=%d\n", ik, (unsigned int) win2, idx, w, h, x, y, valid); @@ -8506,14 +8732,8 @@ fprintf(stderr, "root%02d: ** CreateNotify  0x%x  %3d  -- %dx%d+%d+%d valid=%d\n  			} else if (type == ReparentNotify) {  				if (ev.xreparent.parent != rootwin) {  					win2 = ev.xreparent.window; -					if (win2 != rootwin) { -						idx = lookup_win_index(win2); +					idx = lookup_win_index(win2);  fprintf(stderr, "root%02d: ReparentNotifyRM 0x%x  %3d\n", ik, (unsigned int) win2, idx); -						if (idx >= 0) { -							DELETE(idx); -						} -						xselectinput(win2, 0, 1); -					}  				}  			} else {  fprintf(stderr, "root%02d: ** IgnoringRoot  0x%x type: %s\n", ik, (unsigned int) win, Etype(type)); @@ -8542,6 +8762,7 @@ fprintf(stderr, "----%02d: ConfigureNotify  0x%x  %3d  -- above: 0x%x -> 0x%x  %  				y_new = ev.xconfigure.y;   				w_new = ev.xconfigure.width;   				h_new = ev.xconfigure.height;  +  				x_old = cache_list[idx].x;  				y_old = cache_list[idx].y;  				w_old = cache_list[idx].width; @@ -8557,6 +8778,7 @@ fprintf(stderr, "          INVALIDATE su: 0x%x xy: %d/%d  %d/%d \n", (unsigned i  					cache_list[idx].bs_time = 0.0;  fprintf(stderr, "          INVALIDATE bs: 0x%x wh: %d/%d  %d/%d \n", (unsigned int) win, w_old, h_old, w_new, h_new);  				} +  				stack_change = 0;  				if (cache_list[idx].above != ev.xconfigure.above) {  					stack_change = 1; @@ -8581,6 +8803,7 @@ fprintf(stderr, "          INVALIDATE bs: 0x%x wh: %d/%d  %d/%d \n", (unsigned i  				cache_list[idx].y = y_new;  				cache_list[idx].width = w_new;  				cache_list[idx].height = h_new; +  				cache_list[idx].above = ev.xconfigure.above;  				cache_list[idx].time = dnow(); @@ -8598,28 +8821,35 @@ fprintf(stderr, "----%02d: VisibilityNotify 0x%x  %3d  state: %s U/P %d/%d\n", i  					;	/* XXXX not working well yet with UnmapNotify ... */  				} else if (state == VisibilityUnobscured) {  					int i2, ok = 1; -					for (i2=0; i2 < n; i2++)  { -						if (Ev_map[i2] == win) { -							ok = 0; -							break; -						} -					}  					if (ncache <= 2) {  						ok = 0; +					} else if (ev_lookup(win, EV_MAP)) { +						ok = 0; +					} else if (ev_lookup(win, EV_UNMAP)) { +						ok = 0; +					} else if (ev_lookup(win, EV_DESTROY)) { +						ok = 0;  					}  					if (ok) {  						X_UNLOCK; -						bs_restore(idx, nbatch, 0, 1, 0, 1); +						valid = 0; +						bs_restore(idx, nbatch, &attr, 0, 1, &valid, 1);  						X_LOCK; -						cache_list[idx].time = dnow(); -						cache_list[idx].vis_cnt++; -						Ev_map[ik] = win; -						Ev_rects[nrects].x1 = cache_list[idx].x; -						Ev_rects[nrects].y1 = cache_list[idx].y; -						Ev_rects[nrects].x2 = cache_list[idx].width; -						Ev_rects[nrects].y2 = cache_list[idx].height; -						nrects++; -						SCHED(win, 1)  +						if (valid) { +							STORE(idx, win, attr); + +							cache_list[idx].time = dnow(); +							cache_list[idx].vis_cnt++; +							Ev_map[ik] = win; +							Ev_rects[nrects].x1 = cache_list[idx].x; +							Ev_rects[nrects].y1 = cache_list[idx].y; +							Ev_rects[nrects].x2 = cache_list[idx].width; +							Ev_rects[nrects].y2 = cache_list[idx].height; +							nrects++; +							SCHED(win, 1)  +						} else { +							DELETE(idx); +						}  					}  				}  				cache_list[idx].vis_state = state; @@ -8632,7 +8862,8 @@ fprintf(stderr, "----%02d: MapNotify        0x%x  %3d\n", ik, (unsigned int) win  					continue;  				} -				if (cache_list[idx].map_state == IsUnmapped || desktop_change || macosx_console) { +//				if (cache_list[idx].map_state == IsUnmapped || desktop_change || macosx_console) +				if (1) {  					X_UNLOCK;  					if (desktop_change) {  						/* XXX Y */ @@ -8655,17 +8886,29 @@ fprintf(stderr, "----%02d: MapNotify        0x%x  %3d\n", ik, (unsigned int) win  							sraRgnDestroy(r);  						}   						if (save) { -							su_save(idx, nbatch, 1, 1); +							valid = 0; +							su_save(idx, nbatch, &attr, 1, &valid, 1); +							if (valid) { +								STORE(idx, win, attr); +							}  						}  					} else { -						su_save(idx, nbatch, 0, 1); +						valid = 0; +						su_save(idx, nbatch, &attr, 0, &valid, 1); +						if (valid) { +							STORE(idx, win, attr); +						}  					} -					if (bs_restore(idx, nbatch, 0, 0, 0, 1)) { /* XXX clip? */ +					valid = 0; +					if (bs_restore(idx, nbatch, &attr, 0, 0, &valid, 1)) { /* XXX clip? */  						;  					} else {  						idx_add_rgn(missed_bs_restore_rgn, r0, idx);  						missed_bs_restore++;  					} +					if (valid) { +						STORE(idx, win, attr); +					}  					if (macosx_console) {  #ifdef MACOSX @@ -8690,6 +8933,10 @@ fprintf(stderr, "----%02d: MapNotify        0x%x  %3d\n", ik, (unsigned int) win  					Ev_rects[nrects].x2 = cache_list[idx].width;  					Ev_rects[nrects].y2 = cache_list[idx].height;  					nrects++; + +					if (! valid) { +						DELETE(idx); +					}  				}  				cache_list[idx].map_state = IsViewable; @@ -8706,7 +8953,8 @@ fprintf(stderr, "----%02d: UnmapNotify      0x%x  %3d\n", ik, (unsigned int) win  					}  				} -				if (cache_list[idx].map_state == IsViewable || desktop_change || macosx_console) { +//				if (cache_list[idx].map_state == IsViewable || desktop_change || macosx_console) +				if (1) {  					X_UNLOCK;  					if (desktop_change) {  						int save = 1; @@ -8728,13 +8976,21 @@ fprintf(stderr, "----%02d: UnmapNotify      0x%x  %3d\n", ik, (unsigned int) win  							sraRgnDestroy(r);  						}   						if (save) { -							bs_save(idx, nbatch, 1, 0, 1); +							valid = 0; +							bs_save(idx, nbatch, &attr, 1, 0, &valid, 1);  						}  					} else { -						bs_save(idx, nbatch, 1, 0, 1); +						valid = 0; +						bs_save(idx, nbatch, &attr, 1, 0, &valid, 1);  					} -					if (su_restore(idx, nbatch, 1, 0, 1)) { +					valid = 0; +					if (su_restore(idx, nbatch, &attr, 1, 0, &valid, 1)) {  						try_to_fix_su(win, idx, None, nbatch, "unmapped");	 +						if (valid) { +							STORE(idx, win, attr); +						} else { +							DELETE(idx); +						}  					} else {  						idx_add_rgn(missed_su_restore_rgn, r0, idx);  						missed_su_restore++; @@ -8759,10 +9015,6 @@ fprintf(stderr, "----%02d: UnmapNotify      0x%x  %3d\n", ik, (unsigned int) win  					if (win2 != rootwin) {  						idx = lookup_win_index(win2);  fprintf(stderr, "----%02d: ReparentNotifyRM 0x%x  %3d\n", ik, (unsigned int) win2, idx); -						if (idx >= 0) { -							DELETE(idx); -						} -						xselectinput(win2, 0, 1);  					}  				} @@ -8799,7 +9051,7 @@ fprintf(stderr, "igno%02d: ** Ignoring      0x%x type: %s\n", ik, (unsigned int)  	sraRgnDestroy(missed_su_restore_rgn);  	sraRgnDestroy(missed_bs_restore_rgn); -rfbLog("OUT check_ncache(): %.6f events: %d  pixels: %d\n", dnow() - now, n, pixels); +rfbLog("OUT check_ncache(): %.4f %.6f events: %d  pixels: %d\n", dnowx(), dnow() - now, n, pixels);  	return pixels;  }  #endif | 
