X-Git-Url: http://git.hungrycats.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=driver%2Fxscreensaver-command.c;h=d5d3ba9c93b4c912a847e95f5fdeeb39035a3771;hb=2a991a811de4c7b22f812682b267b616a809fd9a;hp=15e86063d33b3cbc7b75423d3cf3784ed5dace62;hpb=6edc84f12f15860a71430c45e8392a5e4ef8203c;p=xscreensaver diff --git a/driver/xscreensaver-command.c b/driver/xscreensaver-command.c index 15e86063..d5d3ba9c 100644 --- a/driver/xscreensaver-command.c +++ b/driver/xscreensaver-command.c @@ -1,4 +1,5 @@ -/* xscreensaver-command, Copyright (c) 1991-1995 Jamie Zawinski +/* xscreensaver-command, Copyright (c) 1991-1998 + * by Jamie Zawinski * * Permission to use, copy, modify, distribute, and sell this software and its * documentation for any purpose is hereby granted without fee, provided that @@ -9,55 +10,57 @@ * implied warranty. */ -#include "version.h" +#define STANDALONE + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + #include +#include +#include +#include + +#ifdef HAVE_UNISTD_H +# include +#endif + +#include /* for CARD32 */ #include #include +#include /* for XGetClassHint() */ #include -#if __STDC__ -# include -#endif + +#include /* only needed to get through xscreensaver.h */ + +#include "version.h" + +#ifdef STANDALONE + static char *progname; + static Atom XA_VROOT; + static Atom XA_SCREENSAVER, XA_SCREENSAVER_RESPONSE, XA_SCREENSAVER_VERSION; + static Atom XA_SCREENSAVER_TIME, XA_SELECT; +#else /* !STANDALONE */ +# include "xscreensaver.h" +#endif /* !STANDALONE */ + #ifdef _VROOT_H_ ERROR! you must not include vroot.h in this file #endif -static char *screensaver_version; -static char *usage = "usage: %s -\n\ -\n\ - This program provides external control of a running xscreensaver process.\n\ - Version %s, copyright (c) 1991-1994 Jamie Zawinski .\n\ -\n\ - -demo Enter interactive demo mode.\n\ - -deactivate Turns off the screensaver if it is on, as user input would.\n\ - -activate Turns it on as if the user had been idle for long enough.\n\ - -cycle Stops the current hack and runs a new one.\n\ - -next Like either -activate or -cycle, depending on which is more\n\ - appropriate, except that the screenhack that will be run is\n\ - the next one in the list of hacks, instead of a randomly-\n\ - chosen one. This option is good for looking at a demo of\n\ - each of the hacks in place.\n\ - -prev Like -next, but goes in the other direction.\n\ - -exit Causes the screensaver process to exit. It should be ok to\n\ - just kill the process (NOT with -9!) but this is a slightly\n\ - easier way.\n\ - -restart Causes the screensaver process to exit and then restart with\n\ - the same command line arguments. This is a good way of \n\ - causing the screensaver to re-read the resource database.\n\ - -lock Same as -activate, but with immediate locking.\n\ -\n\ - See the man page for more details.\n\n"; + static Window -find_screensaver_window (dpy, progname) - Display *dpy; - char *progname; +find_screensaver_window (Display *dpy, char **version) { int i; Window root = RootWindowOfScreen (DefaultScreenOfDisplay (dpy)); Window root2, parent, *kids; unsigned int nkids; + if (version) *version = 0; + if (! XQueryTree (dpy, root, &root2, &parent, &kids, &nkids)) abort (); if (root != root2) @@ -71,80 +74,457 @@ find_screensaver_window (dpy, progname) Atom type; int format; unsigned long nitems, bytesafter; - char *version; + char *v; if (XGetWindowProperty (dpy, kids[i], - XInternAtom (dpy, "_SCREENSAVER_VERSION", False), - 0, 1, False, XA_STRING, + XA_SCREENSAVER_VERSION, + 0, 200, False, XA_STRING, &type, &format, &nitems, &bytesafter, - (unsigned char **) &version) + (unsigned char **) &v) == Success && type != None) - return kids[i]; + { + if (version) + *version = v; + return kids[i]; + } } fprintf (stderr, "%s: no screensaver is running on display %s\n", progname, DisplayString (dpy)); - exit (1); + return 0; +} + + +static int +send_xscreensaver_command (Display *dpy, Atom command, long argument, + Window *window_ret) +{ + char *v = 0; + Window window = find_screensaver_window (dpy, &v); + XWindowAttributes xgwa; + + if (window_ret) + *window_ret = window; + + if (!window) + return -1; + + /* Select for property change events, so that we can read the response. */ + XGetWindowAttributes (dpy, window, &xgwa); + XSelectInput (dpy, window, xgwa.your_event_mask | PropertyChangeMask); + + if (command == XA_SCREENSAVER_TIME || + command == XA_SCREENSAVER_VERSION) + { + XClassHint hint; + memset (&hint, 0, sizeof(hint)); + if (!v || !*v) + { + fprintf (stderr, "%s: version property not set on window 0x%x?\n", + progname, (unsigned int) window); + return -1; + } + + XGetClassHint(dpy, window, &hint); + if (!hint.res_class) + { + fprintf (stderr, "%s: class hints not set on window 0x%x?\n", + progname, (unsigned int) window); + return -1; + } + + fprintf (stdout, "%s %s", hint.res_class, v); + + if (command != XA_SCREENSAVER_TIME) + { + fprintf (stdout, "\n"); + } + else + { + Atom type; + int format; + unsigned long nitems, bytesafter; + unsigned char *data = 0; + Bool active_p = False; + + if (XGetWindowProperty (dpy, window, XA_VROOT, + 0, 0, False, XA_WINDOW, + &type, &format, &nitems, &bytesafter, + &data) + == Success + && type != None) + active_p = True; + + if (data) free (data); + data = 0; + + if (XGetWindowProperty (dpy, window, + XA_SCREENSAVER_TIME, + 0, 1, False, XA_INTEGER, + &type, &format, &nitems, &bytesafter, + &data) + == Success + && type == XA_INTEGER + && data) + { + CARD32 time32 = *((CARD32 *)data); + time_t tt = (time_t) time32; + + if (active_p) + fprintf (stdout, ": screen blanked since "); + else + /* suggestions for a better way to phrase this are welcome. */ + fprintf (stdout, ": screen non-blanked since "); + fprintf (stdout, "%s", ctime(&tt)); + if (data) free (data); + } + else + { + if (data) free (data); + fprintf (stdout, "\n"); + fflush (stdout); + fprintf (stderr, "%s: no time on window 0x%x (%s %s).\n", + progname, (unsigned int) window, + hint.res_class, (v ? v : "???")); + return -1; + } + } + + /* No need to read a response for these commands. */ + return 1; + } + else + { + XEvent event; + long arg1 = (command == XA_SELECT ? argument : 0L); + event.xany.type = ClientMessage; + event.xclient.display = dpy; + event.xclient.window = window; + event.xclient.message_type = XA_SCREENSAVER; + event.xclient.format = 32; + memset (&event.xclient.data, 0, sizeof(event.xclient.data)); + event.xclient.data.l[0] = (long) command; + event.xclient.data.l[1] = arg1; + if (! XSendEvent (dpy, window, False, 0L, &event)) + { + fprintf (stderr, "%s: XSendEvent(dpy, 0x%x ...) failed.\n", + progname, (unsigned int) window); + return -1; + } + } + XSync (dpy, 0); + return 0; +} + + +static XErrorHandler old_handler = 0; +static Bool got_badwindow = False; +static int +BadWindow_ehandler (Display *dpy, XErrorEvent *error) +{ + /* When we notice a window being created, we spawn a timer that waits + 30 seconds or so, and then selects events on that window. This error + handler is used so that we can cope with the fact that the window + may have been destroyed <30 seconds after it was created. + */ + if (error->error_code == BadWindow) + { + got_badwindow = True; + return 0; + } + else + { + fprintf (stderr, "%s: ", progname); + return (*old_handler) (dpy, error); + } } +static int +xscreensaver_command_response (Display *dpy, Window window) +{ + int fd = ConnectionNumber (dpy); + int timeout = 10; + int status; + fd_set fds; + struct timeval tv; + + while (1) + { + FD_ZERO(&fds); + FD_SET(fd, &fds); + memset(&tv, 0, sizeof(tv)); + tv.tv_sec = timeout; + status = select (fd+1, &fds, 0, &fds, &tv); + + if (status < 0) + { + char buf[1024]; + sprintf (buf, "%s: waiting for reply", progname); + perror (buf); + return status; + } + else if (status == 0) + { + fprintf (stderr, "%s: no response to command.\n", progname); + return -1; + } + else + { + XEvent event; + XNextEvent (dpy, &event); + switch (event.xany.type) { + case PropertyNotify: + if (event.xproperty.state == PropertyNewValue && + event.xproperty.atom == XA_SCREENSAVER_RESPONSE) + { + Status st2; + Atom type; + int format; + unsigned long nitems, bytesafter; + char *msg = 0; + + old_handler = XSetErrorHandler (BadWindow_ehandler); + XSync (dpy, False); + + st2 = XGetWindowProperty (dpy, window, + XA_SCREENSAVER_RESPONSE, + 0, 1024, True, + AnyPropertyType, + &type, &format, &nitems, &bytesafter, + (unsigned char **) &msg); + + if (got_badwindow) + { + fprintf (stdout, + "%s: xscreensaver window has been deleted.\n", + progname); + return 0; + } + + if (st2 == Success && type != None) + { + if (type != XA_STRING || format != 8) + { + fprintf (stderr, + "%s: unrecognized response property.\n", + progname); + if (msg) XFree (msg); + return -1; + } + else if (!msg || (msg[0] != '+' && msg[0] != '-')) + { + fprintf (stderr, + "%s: unrecognized response message.\n", + progname); + if (msg) XFree (msg); + return -1; + } + else + { + int ret = (msg[0] == '+' ? 0 : -1); + fprintf ((ret < 0 ? stderr : stdout), + "%s: %s\n", + progname, + msg+1); + XFree (msg); + return ret; + } + } + } + break; + + default: + fprintf (stderr, "%s: got unexpected response event %d.\n", + progname, event.xany.type); + return -1; + break; + } + } + } +} + + +int +xscreensaver_command (Display *dpy, Atom command, long argument) +{ + Window w = 0; + int status = send_xscreensaver_command (dpy, command, argument, &w); + if (status == 0) + status = xscreensaver_command_response (dpy, w); + fflush (stdout); + fflush (stderr); + return status; +} + + +#ifdef STANDALONE +static Atom XA_ACTIVATE, XA_DEACTIVATE, XA_CYCLE, XA_NEXT, XA_PREV; +static Atom XA_EXIT, XA_RESTART, XA_DEMO, XA_PREFS, XA_LOCK; + +static char *progname; +static char *screensaver_version; +static char *usage = "\n\ +usage: %s -