X-Git-Url: http://git.hungrycats.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=driver%2Fxscreensaver-command.c;h=d5d3ba9c93b4c912a847e95f5fdeeb39035a3771;hb=2a991a811de4c7b22f812682b267b616a809fd9a;hp=2d7f3326064d97916b44782cf7530bed434d2e14;hpb=5b7bc6e70fb439cf4c4bf771ae9f94077fe4fe08;p=xscreensaver diff --git a/driver/xscreensaver-command.c b/driver/xscreensaver-command.c index 2d7f3326..d5d3ba9c 100644 --- a/driver/xscreensaver-command.c +++ b/driver/xscreensaver-command.c @@ -1,5 +1,5 @@ -/* xscreensaver-command, Copyright (c) 1991-1997 - * by 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 @@ -10,61 +10,49 @@ * implied warranty. */ +#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 -#include + +#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-1997 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 graphics 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 could be used for looking at a demo\n\ - of each of the configured hacks.\n\ - -prev Like -next, but goes in the other direction.\n\ - -exit Causes the screensaver process to exit. This is the same as\n\ - killing the process with `kill', but it's easier, since you\n\ - don't need to first figure out the pid. (Note that one\n\ - must *never* kill xscreensaver with -9!)\n\ - -restart Causes the screensaver process to exit and then restart with\n\ - the same command line arguments. Do this after you've\n\ - changed the resource database, to cause the screensaver to\n\ - notice the changes.\n\ - -lock Same as -activate, but with immediate locking.\n\ - -version Prints the version of XScreenSaver that is currently running\n\ - on the display.\n\ - -time Prints the time at which the screensaver last activated or\n\ - deactivated (roughly, how long the user has been idle or\n\ - non-idle.)\n\ -\n\ - See the man page for more details.\n\ - For updates, check http://people.netscape.com/jwz/xscreensaver/\n\ -\n"; + static Window -find_screensaver_window (Display *dpy, char *progname, char **version) +find_screensaver_window (Display *dpy, char **version) { int i; Window root = RootWindowOfScreen (DefaultScreenOfDisplay (dpy)); @@ -89,7 +77,7 @@ find_screensaver_window (Display *dpy, char *progname, char **version) char *v; if (XGetWindowProperty (dpy, kids[i], - XInternAtom (dpy, "_SCREENSAVER_VERSION", False), + XA_SCREENSAVER_VERSION, 0, 200, False, XA_STRING, &type, &format, &nitems, &bytesafter, (unsigned char **) &v) @@ -103,97 +91,51 @@ find_screensaver_window (Display *dpy, char *progname, char **version) } fprintf (stderr, "%s: no screensaver is running on display %s\n", progname, DisplayString (dpy)); - exit (1); + return 0; } -#define USAGE() \ - { fprintf (stderr, usage, argv[0], screensaver_version); exit (1); } - -int -main (int argc, char **argv) +static int +send_xscreensaver_command (Display *dpy, Atom command, long argument, + Window *window_ret) { - Display *dpy; - Window window; - int i; - int query = 0; -#define Q_version 1 -#define Q_time 2 - char *message = 0, *dpyname = 0; char *v = 0; + Window window = find_screensaver_window (dpy, &v); + XWindowAttributes xgwa; - screensaver_version = (char *) malloc (5); - memcpy (screensaver_version, screensaver_id + 17, 4); - screensaver_version [4] = 0; - for (i = 1; i < argc; i++) - { - char *s = argv [i]; - int L = strlen (s); - if (L < 2) USAGE (); - if (!strncmp (s, "-display", L)) dpyname = argv [++i]; - else if (message) USAGE () - else if (!strncmp (s, "-activate", L)) message = "ACTIVATE"; - else if (!strncmp (s, "-deactivate", L)) message = "DEACTIVATE"; - else if (!strncmp (s, "-cycle", L)) message = "CYCLE"; - else if (!strncmp (s, "-next", L)) message = "NEXT"; - else if (!strncmp (s, "-prev", L)) message = "PREV"; - else if (!strncmp (s, "-exit", L)) message = "EXIT"; - else if (!strncmp (s, "-restart", L)) message = "RESTART"; - else if (!strncmp (s, "-demo", L)) message = "DEMO"; - else if (!strncmp (s, "-lock", L)) message = "LOCK"; - else if (!strncmp (s, "-version", L)) query = Q_version; - else if (!strncmp (s, "-time", L)) query = Q_time; - else USAGE (); - } - if (!message && !query) USAGE (); - if (!dpyname) dpyname = (char *) getenv ("DISPLAY"); - dpy = XOpenDisplay (dpyname); - if (!dpy) - { - fprintf (stderr, "%s: can't open display %s\n", argv[0], - (dpyname ? dpyname : "(null)")); - exit (1); - } - window = find_screensaver_window (dpy, argv[0], &v); + if (window_ret) + *window_ret = window; - if (message) - { - XEvent event; - event.xany.type = ClientMessage; - event.xclient.display = dpy; - event.xclient.window = window; - event.xclient.message_type = XInternAtom (dpy, "SCREENSAVER", False); - event.xclient.format = 32; - event.xclient.data.l[0] = (long) XInternAtom (dpy, message, False); - if (! XSendEvent (dpy, window, False, 0L, &event)) - { - fprintf (stderr, "%s: XSendEvent(dpy, 0x%x ...) failed.\n", argv [0], - (unsigned int) window); - exit (1); - } - } - else if (query) + 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", - argv [0], (unsigned int) window); - exit (1); + 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", - argv [0], (unsigned int) window); - exit (1); + progname, (unsigned int) window); + return -1; } fprintf (stdout, "%s %s", hint.res_class, v); - if (query != Q_time) + if (command != XA_SCREENSAVER_TIME) { fprintf (stdout, "\n"); } @@ -203,14 +145,9 @@ main (int argc, char **argv) int format; unsigned long nitems, bytesafter; unsigned char *data = 0; - XWindowAttributes xgwa; Bool active_p = False; - xgwa.map_state = IsViewable; - XGetWindowAttributes (dpy, window, &xgwa); - - if (XGetWindowProperty (dpy, window, - XInternAtom (dpy, "__SWM_VROOT", False), + if (XGetWindowProperty (dpy, window, XA_VROOT, 0, 0, False, XA_WINDOW, &type, &format, &nitems, &bytesafter, &data) @@ -222,7 +159,7 @@ main (int argc, char **argv) data = 0; if (XGetWindowProperty (dpy, window, - XInternAtom (dpy, "_SCREENSAVER_TIME",False), + XA_SCREENSAVER_TIME, 0, 1, False, XA_INTEGER, &type, &format, &nitems, &bytesafter, &data) @@ -234,9 +171,10 @@ main (int argc, char **argv) time_t tt = (time_t) time32; if (active_p) - fprintf (stdout, ": active since "); + fprintf (stdout, ": screen blanked since "); else - fprintf (stdout, ": inactive since "); + /* 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); } @@ -246,15 +184,347 @@ main (int argc, char **argv) fprintf (stdout, "\n"); fflush (stdout); fprintf (stderr, "%s: no time on window 0x%x (%s %s).\n", - argv[0], (unsigned int) window, + progname, (unsigned int) window, hint.res_class, (v ? v : "???")); - exit (1); + 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); - exit (0); + 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 -