[PATCH 16/18] Add synxrrd daemon

Takashi Iwai tiwai at suse.de
Fri Oct 8 10:22:40 PDT 2010


This is a daemon program to sync Xrandr rotation with the synaptics device.

Signed-off-by: Takashi Iwai <tiwai at suse.de>
---
 tools/Makefile.am |    5 +-
 tools/synxrrd.c   |  186 +++++++++++++++++++++++++++++++++++++++++++++++++++++
 2 files changed, 190 insertions(+), 1 deletions(-)
 create mode 100644 tools/synxrrd.c

diff --git a/tools/Makefile.am b/tools/Makefile.am
index 2ad48f6..1111576 100644
--- a/tools/Makefile.am
+++ b/tools/Makefile.am
@@ -18,7 +18,7 @@
 #  IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 #  CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 
-bin_PROGRAMS = synclient syndaemon
+bin_PROGRAMS = synclient syndaemon synxrrd
 
 AM_CPPFLAGS = -I$(top_srcdir)/include -I$(sdkdir)
 AM_CFLAGS = $(XI_CFLAGS)
@@ -30,3 +30,6 @@ syndaemon_SOURCES = syndaemon.c
 syndaemon_CFLAGS = $(AM_CFLAGS) $(XTST_CFLAGS)
 syndaemon_LDFLAGS = $(AM_LDFLAGS) $(XTST_LIBS)
 
+synxrrd_SOURCES = synxrrd.c
+synxrrd_LDFLAGS = -lXrandr $(XLIB_LIBS) $(XI_LIBS)
+
diff --git a/tools/synxrrd.c b/tools/synxrrd.c
new file mode 100644
index 0000000..9f00e5b
--- /dev/null
+++ b/tools/synxrrd.c
@@ -0,0 +1,186 @@
+/*
+ * synxrrd.c
+ * Copyright (C) 2010 Takashi Iwai <tiwai at suse.de>
+ */
+
+#include <stdio.h>
+#include <stdarg.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+#include <X11/Xlib.h>
+#include <X11/extensions/Xrandr.h>
+
+#define progname	"synxrrd"
+
+static int verbose;
+static const char *laptop_name;
+
+static int is_laptop(const char *name)
+{
+	if (laptop_name) {
+		if (!strcmp(name, laptop_name))
+			return 1;
+	} else {
+		if (strstr(name, "LVDS") ||
+		    strstr(name, "Lvds") ||
+		    strstr(name, "LVDs"))
+			return 1;
+		if (strstr(name, "LCD"))
+			return 1;
+	}
+	return 0;
+}
+
+static int current_rotation = -1; /* unknown */
+
+static int (*xerror_saved)(Display *, XErrorEvent *);
+static int xerror_code;
+
+static int xerror_ignore(Display *dpy, XErrorEvent *err)
+{
+	xerror_code = err->error_code;
+	return 0;
+}
+
+extern int (*_XErrorFunction)(Display *, XErrorEvent *);
+extern int _XDefaultError(Display *dpy, XErrorEvent *event);
+
+static void xerror_save(void)
+{
+	xerror_code = 0;
+	xerror_saved = _XErrorFunction;
+	_XErrorFunction = xerror_ignore;
+}
+
+static int xerror_restore(void)
+{
+	_XErrorFunction = xerror_saved;
+	return xerror_code;
+}
+
+static void notify_synaptics(XRRCrtcInfo *cres)
+{
+	int rotation;
+	char buf[256];
+
+	if (cres->rotation & 8)
+		rotation = 3;
+	else if (cres->rotation & 4)
+		rotation = 2;
+	else if (cres->rotation & 2)
+		rotation = 1;
+	else
+		rotation = 0;
+	if (current_rotation == rotation)
+		return;
+	current_rotation = rotation;
+	if (verbose)
+		fprintf(stderr, "%s: notified rotation %d\n",
+			progname, current_rotation);
+	sprintf(buf, "synclient Orientation=%d", current_rotation);
+	system(buf);
+}
+
+static void update_screen(Display *dpy, Window root)
+{
+	XRRScreenResources *res;
+	int i;
+
+	res = XRRGetScreenResources(dpy, root);
+	if (!res)
+		return;
+	for (i = 0; i < res->noutput; i++) {
+		XRROutputInfo *info;
+
+	       	info = XRRGetOutputInfo(dpy, res, res->outputs[i]);
+		if (!info)
+			continue;
+		if (is_laptop(info->name)) {
+			RRCrtc crtc = info->crtc;
+			XRRCrtcInfo *cres;
+
+			XRRFreeOutputInfo(info);
+			cres = XRRGetCrtcInfo(dpy, res, crtc);
+			if (!cres)
+				continue;
+			notify_synaptics(cres);
+			XRRFreeCrtcInfo(cres);
+			break;
+		}
+		XRRFreeOutputInfo(info);
+	}
+	XRRFreeScreenResources(res);
+}
+
+static void synxrr_daemon(Display *dpy, Window root)
+{
+	int event_base, error_base;
+
+	XRRQueryExtension(dpy, &event_base, &error_base);
+
+	XRRSelectInput(dpy, root, RRScreenChangeNotifyMask);
+
+	for (;;) {
+		XEvent event;
+
+		XNextEvent(dpy, &event);
+		xerror_save();
+		XRRUpdateConfiguration(&event);
+		switch (event.type - event_base) {
+		case RRScreenChangeNotify:
+			update_screen(dpy, root);
+			break;
+		}
+		xerror_restore();
+	}
+}
+
+static void usage(void)
+{
+	fprintf(stderr, "usage: %s [options]\n", progname);
+	fprintf(stderr, "  options\n");
+	fprintf(stderr, "  -d display  set X display name\n");
+	fprintf(stderr, "  -l name     xrandr output name for laptop display\n");
+	fprintf(stderr, "  -v          verbose mode\n");
+}
+
+int main(int argc, char **argv)
+{
+	Display *dpy;
+	Window root;
+	const char *display_name = NULL;
+	const char *device_name;
+	int c;
+
+	while ((c = getopt(argc, argv, "d:l:v")) != -1) {
+		switch (c) {
+		case 'd':
+			display_name = optarg;
+			break;
+		case 'l':
+			laptop_name = optarg;
+			break;
+		case 'v':
+			verbose++;
+			break;
+		default:
+			usage();
+			return 1;
+		}
+	}
+
+	device_name = argv[optind];
+
+	dpy = XOpenDisplay(display_name);
+	if (!dpy) {
+		perror("open display");
+		return 1;
+	}
+
+	root = DefaultRootWindow(dpy);
+	update_screen(dpy, root);
+	synxrr_daemon(dpy, root);
+
+	return 0;
+}
-- 
1.7.3.1



More information about the xorg-devel mailing list