http://www.tienza.es/crux/src/www.jwz.org/xscreensaver/xscreensaver-5.05.tar.gz
[xscreensaver] / driver / timers.c
1 /* timers.c --- detecting when the user is idle, and other timer-related tasks.
2  * xscreensaver, Copyright (c) 1991-2008 Jamie Zawinski <jwz@jwz.org>
3  *
4  * Permission to use, copy, modify, distribute, and sell this software and its
5  * documentation for any purpose is hereby granted without fee, provided that
6  * the above copyright notice appear in all copies and that both that
7  * copyright notice and this permission notice appear in supporting
8  * documentation.  No representations are made about the suitability of this
9  * software for any purpose.  It is provided "as is" without express or 
10  * implied warranty.
11  */
12
13 #ifdef HAVE_CONFIG_H
14 # include "config.h"
15 #endif
16
17 #include <stdio.h>
18 #include <X11/Xlib.h>
19 #include <X11/Intrinsic.h>
20 #include <X11/Xos.h>
21 #include <time.h>
22 #include <sys/time.h>
23 #ifdef HAVE_XMU
24 # ifndef VMS
25 #  include <X11/Xmu/Error.h>
26 # else /* VMS */
27 #  include <Xmu/Error.h>
28 # endif /* VMS */
29 # else /* !HAVE_XMU */
30 # include "xmu.h"
31 #endif /* !HAVE_XMU */
32
33 #ifdef HAVE_XIDLE_EXTENSION
34 #include <X11/extensions/xidle.h>
35 #endif /* HAVE_XIDLE_EXTENSION */
36
37 #ifdef HAVE_MIT_SAVER_EXTENSION
38 #include <X11/extensions/scrnsaver.h>
39 #endif /* HAVE_MIT_SAVER_EXTENSION */
40
41 #ifdef HAVE_SGI_SAVER_EXTENSION
42 #include <X11/extensions/XScreenSaver.h>
43 #endif /* HAVE_SGI_SAVER_EXTENSION */
44
45 #ifdef HAVE_RANDR
46 #include <X11/extensions/Xrandr.h>
47 #endif /* HAVE_RANDR */
48
49 #include "xscreensaver.h"
50
51 #undef ABS
52 #define ABS(x)((x)<0?-(x):(x))
53
54 #undef MAX
55 #define MAX(x,y)((x)>(y)?(x):(y))
56
57
58 #ifdef HAVE_PROC_INTERRUPTS
59 static Bool proc_interrupts_activity_p (saver_info *si);
60 #endif /* HAVE_PROC_INTERRUPTS */
61
62 static void check_for_clock_skew (saver_info *si);
63
64
65 void
66 idle_timer (XtPointer closure, XtIntervalId *id)
67 {
68   saver_info *si = (saver_info *) closure;
69
70   /* What an amazingly shitty design.  Not only does Xt execute timeout
71      events from XtAppNextEvent() instead of from XtDispatchEvent(), but
72      there is no way to tell Xt to block until there is an X event OR a
73      timeout happens.  Once your timeout proc is called, XtAppNextEvent()
74      still won't return until a "real" X event comes in.
75
76      So this function pushes a stupid, gratuitous, unnecessary event back
77      on the event queue to force XtAppNextEvent to return Right Fucking Now.
78      When the code in sleep_until_idle() sees an event of type XAnyEvent,
79      which the server never generates, it knows that a timeout has occurred.
80    */
81   XEvent fake_event;
82   fake_event.type = 0;  /* XAnyEvent type, ignored. */
83   fake_event.xany.display = si->dpy;
84   fake_event.xany.window  = 0;
85   XPutBackEvent (si->dpy, &fake_event);
86
87   /* If we are the timer that just went off, clear the pointer to the id. */
88   if (id)
89     {
90       if (si->timer_id && *id != si->timer_id)
91         abort();  /* oops, scheduled timer twice?? */
92       si->timer_id = 0;
93     }
94 }
95
96
97 void
98 schedule_wakeup_event (saver_info *si, Time when, Bool verbose_p)
99 {
100   if (si->timer_id)
101     {
102       if (verbose_p)
103         fprintf (stderr, "%s: idle_timer already running\n", blurb());
104       return;
105     }
106
107   /* Wake up periodically to ask the server if we are idle. */
108   si->timer_id = XtAppAddTimeOut (si->app, when, idle_timer,
109                                   (XtPointer) si);
110
111   if (verbose_p)
112     fprintf (stderr, "%s: starting idle_timer (%ld, %ld)\n",
113              blurb(), when, si->timer_id);
114 }
115
116
117 static void
118 notice_events (saver_info *si, Window window, Bool top_p)
119 {
120   saver_preferences *p = &si->prefs;
121   XWindowAttributes attrs;
122   unsigned long events;
123   Window root, parent, *kids;
124   unsigned int nkids;
125   int screen_no;
126
127   if (XtWindowToWidget (si->dpy, window))
128     /* If it's one of ours, don't mess up its event mask. */
129     return;
130
131   if (!XQueryTree (si->dpy, window, &root, &parent, &kids, &nkids))
132     return;
133   if (window == root)
134     top_p = False;
135
136   /* Figure out which screen this window is on, for the diagnostics. */
137   for (screen_no = 0; screen_no < si->nscreens; screen_no++)
138     if (root == RootWindowOfScreen (si->screens[screen_no].screen))
139       break;
140
141   XGetWindowAttributes (si->dpy, window, &attrs);
142   events = ((attrs.all_event_masks | attrs.do_not_propagate_mask)
143             & KeyPressMask);
144
145   /* Select for SubstructureNotify on all windows.
146      Select for KeyPress on all windows that already have it selected.
147
148      Note that we can't select for ButtonPress, because of X braindamage:
149      only one client at a time may select for ButtonPress on a given
150      window, though any number can select for KeyPress.  Someone explain
151      *that* to me.
152
153      So, if the user spends a while clicking the mouse without ever moving
154      the mouse or touching the keyboard, we won't know that they've been
155      active, and the screensaver will come on.  That sucks, but I don't
156      know how to get around it.
157
158      Since X presents mouse wheels as clicks, this applies to those, too:
159      scrolling through a document using only the mouse wheel doesn't
160      count as activity...  Fortunately, /proc/interrupts helps, on
161      systems that have it.  Oh, if it's a PS/2 mouse, not serial or USB.
162      This sucks!
163    */
164   XSelectInput (si->dpy, window, SubstructureNotifyMask | events);
165
166   if (top_p && p->debug_p && (events & KeyPressMask))
167     {
168       /* Only mention one window per tree (hack hack). */
169       fprintf (stderr, "%s: %d: selected KeyPress on 0x%lX\n",
170                blurb(), screen_no, (unsigned long) window);
171       top_p = False;
172     }
173
174   if (kids)
175     {
176       while (nkids)
177         notice_events (si, kids [--nkids], top_p);
178       XFree ((char *) kids);
179     }
180 }
181
182
183 int
184 BadWindow_ehandler (Display *dpy, XErrorEvent *error)
185 {
186   /* When we notice a window being created, we spawn a timer that waits
187      30 seconds or so, and then selects events on that window.  This error
188      handler is used so that we can cope with the fact that the window
189      may have been destroyed <30 seconds after it was created.
190    */
191   if (error->error_code == BadWindow ||
192       error->error_code == BadMatch ||
193       error->error_code == BadDrawable)
194     return 0;
195   else
196     return saver_ehandler (dpy, error);
197 }
198
199
200 struct notice_events_timer_arg {
201   saver_info *si;
202   Window w;
203 };
204
205 static void
206 notice_events_timer (XtPointer closure, XtIntervalId *id)
207 {
208   struct notice_events_timer_arg *arg =
209     (struct notice_events_timer_arg *) closure;
210
211   XErrorHandler old_handler = XSetErrorHandler (BadWindow_ehandler);
212
213   saver_info *si = arg->si;
214   Window window = arg->w;
215
216   free(arg);
217   notice_events (si, window, True);
218   XSync (si->dpy, False);
219   XSetErrorHandler (old_handler);
220 }
221
222 void
223 start_notice_events_timer (saver_info *si, Window w, Bool verbose_p)
224 {
225   saver_preferences *p = &si->prefs;
226   struct notice_events_timer_arg *arg =
227     (struct notice_events_timer_arg *) malloc(sizeof(*arg));
228   arg->si = si;
229   arg->w = w;
230   XtAppAddTimeOut (si->app, p->notice_events_timeout, notice_events_timer,
231                    (XtPointer) arg);
232
233   if (verbose_p)
234     fprintf (stderr, "%s: starting notice_events_timer for 0x%X (%lu)\n",
235              blurb(), (unsigned int) w, p->notice_events_timeout);
236 }
237
238
239 /* When the screensaver is active, this timer will periodically change
240    the running program.
241  */
242 void
243 cycle_timer (XtPointer closure, XtIntervalId *id)
244 {
245   saver_info *si = (saver_info *) closure;
246   saver_preferences *p = &si->prefs;
247   Time how_long = p->cycle;
248
249   if (si->selection_mode > 0 &&
250       screenhack_running_p (si))
251     /* If we're in "SELECT n" mode, the cycle timer going off will just
252        restart this same hack again.  There's not much point in doing this
253        every 5 or 10 minutes, but on the other hand, leaving one hack running
254        for days is probably not a great idea, since they tend to leak and/or
255        crash.  So, restart the thing once an hour. */
256     how_long = 1000 * 60 * 60;
257
258   if (si->dbox_up_p)
259     {
260       if (p->verbose_p)
261         fprintf (stderr, "%s: dialog box up; delaying hack change.\n",
262                  blurb());
263       how_long = 30000; /* 30 secs */
264     }
265   else
266     {
267       maybe_reload_init_file (si);
268       kill_screenhack (si);
269
270       if (!si->throttled_p)
271         spawn_screenhack (si, False);
272       else
273         {
274           raise_window (si, True, True, False);
275           if (p->verbose_p)
276             fprintf (stderr, "%s: not launching new hack (throttled.)\n",
277                      blurb());
278         }
279     }
280
281   if (how_long > 0)
282     {
283       si->cycle_id = XtAppAddTimeOut (si->app, how_long, cycle_timer,
284                                       (XtPointer) si);
285
286       if (p->debug_p)
287         fprintf (stderr, "%s: starting cycle_timer (%ld, %ld)\n",
288                  blurb(), how_long, si->cycle_id);
289     }
290   else
291     {
292       if (p->debug_p)
293         fprintf (stderr, "%s: not starting cycle_timer: how_long == %ld\n",
294                  blurb(), (unsigned long) how_long);
295     }
296 }
297
298
299 void
300 activate_lock_timer (XtPointer closure, XtIntervalId *id)
301 {
302   saver_info *si = (saver_info *) closure;
303   saver_preferences *p = &si->prefs;
304
305   if (p->verbose_p)
306     fprintf (stderr, "%s: timed out; activating lock.\n", blurb());
307   set_locked_p (si, True);
308 }
309
310
311 /* Call this when user activity (or "simulated" activity) has been noticed.
312  */
313 void
314 reset_timers (saver_info *si)
315 {
316   saver_preferences *p = &si->prefs;
317   if (si->using_mit_saver_extension || si->using_sgi_saver_extension)
318     return;
319
320   if (si->timer_id)
321     {
322       if (p->debug_p)
323         fprintf (stderr, "%s: killing idle_timer  (%ld, %ld)\n",
324                  blurb(), p->timeout, si->timer_id);
325       XtRemoveTimeOut (si->timer_id);
326       si->timer_id = 0;
327     }
328
329   schedule_wakeup_event (si, p->timeout, p->debug_p); /* sets si->timer_id */
330
331   if (si->cycle_id) abort ();   /* no cycle timer when inactive */
332
333   si->last_activity_time = time ((time_t *) 0);
334
335   /* This will (hopefully, supposedly) tell the server to re-set its
336      DPMS timer.  Without this, the -deactivate clientmessage would
337      prevent xscreensaver from blanking, but would not prevent the
338      monitor from powering down. */
339 #if 0
340   /* #### With some servers, this causes the screen to flicker every
341      time a key is pressed!  Ok, I surrender.  I give up on ever
342      having DPMS work properly.
343    */
344   XForceScreenSaver (si->dpy, ScreenSaverReset);
345
346   /* And if the monitor is already powered off, turn it on.
347      You'd think the above would do that, but apparently not? */
348   monitor_power_on (si);
349 #endif
350
351 }
352
353
354 /* Returns true if the mouse has moved since the last time we checked.
355    Small motions (of less than "hysteresis" pixels/second) are ignored.
356  */
357 static Bool
358 pointer_moved_p (saver_screen_info *ssi, Bool mods_p)
359 {
360   saver_info *si = ssi->global;
361   saver_preferences *p = &si->prefs;
362
363   Window root, child;
364   int root_x, root_y, x, y;
365   unsigned int mask;
366   time_t now = time ((time_t *) 0);
367   unsigned int distance, dps;
368   unsigned long seconds = 0;
369   Bool moved_p = False;
370
371   /* don't check xinerama pseudo-screens. */
372   if (!ssi->real_screen_p) return False;
373
374   if (!XQueryPointer (si->dpy, ssi->screensaver_window, &root, &child,
375                       &root_x, &root_y, &x, &y, &mask))
376     {
377       /* If XQueryPointer() returns false, the mouse is not on this screen.
378        */
379       x = root_x = -1;
380       y = root_y = -1;
381       root = child = 0;
382       mask = 0;
383     }
384
385   distance = MAX (ABS (ssi->poll_mouse_last_root_x - root_x),
386                   ABS (ssi->poll_mouse_last_root_y - root_y));
387   seconds = (now - ssi->poll_mouse_last_time);
388
389
390   /* When the screen is blanked, we get MotionNotify events, but when not
391      blanked, we poll only every 5 seconds, and that's not enough resolution
392      to do hysteresis based on a 1 second interval.  So, assume that any
393      motion we've seen during the 5 seconds when our eyes were closed happened
394      in the last 1 second instead.
395    */
396   if (seconds > 1) seconds = 1;
397
398   dps = (seconds <= 0 ? distance : (distance / seconds));
399
400   /* Motion only counts if the rate is more than N pixels per second.
401    */
402   if (dps >= p->pointer_hysteresis &&
403       distance > 0)
404     moved_p = True;
405
406   /* If the mouse is not on this screen but used to be, that's motion.
407      If the mouse was not on this screen, but is now, that's motion.
408    */
409   {
410     Bool on_screen_p  = (root_x != -1 && root_y != -1);
411     Bool was_on_screen_p = (ssi->poll_mouse_last_root_x != -1 &&
412                             ssi->poll_mouse_last_root_y != -1);
413
414     if (on_screen_p != was_on_screen_p)
415       moved_p = True;
416   }
417
418   if (p->debug_p && (distance != 0 || moved_p))
419     {
420       fprintf (stderr, "%s: %d: pointer %s", blurb(), ssi->number,
421                (moved_p ? "moved:  " : "ignored:"));
422       if (ssi->poll_mouse_last_root_x == -1)
423         fprintf (stderr, "off screen");
424       else
425         fprintf (stderr, "%d,%d",
426                  ssi->poll_mouse_last_root_x,
427                  ssi->poll_mouse_last_root_y);
428       fprintf (stderr, " -> ");
429       if (root_x == -1)
430         fprintf (stderr, "off screen");
431       else
432         fprintf (stderr, "%d,%d", root_x, root_y);
433       if (ssi->poll_mouse_last_root_x != -1 && root_x != -1)
434         fprintf (stderr, " (%d,%d; %d/%lu=%d)",
435                  ABS(ssi->poll_mouse_last_root_x - root_x),
436                  ABS(ssi->poll_mouse_last_root_y - root_y),
437                  distance, seconds, dps);
438
439       fprintf (stderr, ".\n");
440     }
441
442   if (!moved_p &&
443       mods_p &&
444       mask != ssi->poll_mouse_last_mask)
445     {
446       moved_p = True;
447
448       if (p->debug_p)
449         fprintf (stderr, "%s: %d: modifiers changed: 0x%04x -> 0x%04x.\n",
450                  blurb(), ssi->number, ssi->poll_mouse_last_mask, mask);
451     }
452
453   si->last_activity_screen   = ssi;
454   ssi->poll_mouse_last_child = child;
455   ssi->poll_mouse_last_mask  = mask;
456
457   if (moved_p || seconds > 0)
458     {
459       ssi->poll_mouse_last_time   = now;
460       ssi->poll_mouse_last_root_x = root_x;
461       ssi->poll_mouse_last_root_y = root_y;
462     }
463
464   return moved_p;
465 }
466
467
468 /* When we aren't using a server extension, this timer is used to periodically
469    wake up and poll the mouse position, which is possibly more reliable than
470    selecting motion events on every window.
471  */
472 static void
473 check_pointer_timer (XtPointer closure, XtIntervalId *id)
474 {
475   int i;
476   saver_info *si = (saver_info *) closure;
477   saver_preferences *p = &si->prefs;
478   Bool active_p = False;
479
480   if (!si->using_proc_interrupts &&
481       (si->using_xidle_extension ||
482        si->using_mit_saver_extension ||
483        si->using_sgi_saver_extension))
484     /* If an extension is in use, we should not be polling the mouse.
485        Unless we're also checking /proc/interrupts, in which case, we should.
486      */
487     abort ();
488
489   if (id && *id == si->check_pointer_timer_id)  /* this is us - it's expired */
490     si->check_pointer_timer_id = 0;
491
492   if (si->check_pointer_timer_id)               /* only queue one at a time */
493     XtRemoveTimeOut (si->check_pointer_timer_id);
494
495   si->check_pointer_timer_id =                  /* now re-queue */
496     XtAppAddTimeOut (si->app, p->pointer_timeout, check_pointer_timer,
497                      (XtPointer) si);
498
499   for (i = 0; i < si->nscreens; i++)
500     {
501       saver_screen_info *ssi = &si->screens[i];
502       if (pointer_moved_p (ssi, True))
503         active_p = True;
504     }
505
506 #ifdef HAVE_PROC_INTERRUPTS
507   if (!active_p &&
508       si->using_proc_interrupts &&
509       proc_interrupts_activity_p (si))
510     {
511       active_p = True;
512     }
513 #endif /* HAVE_PROC_INTERRUPTS */
514
515   if (active_p)
516     reset_timers (si);
517
518   check_for_clock_skew (si);
519 }
520
521
522 /* An unfortunate situation is this: the saver is not active, because the
523    user has been typing.  The machine is a laptop.  The user closes the lid
524    and suspends it.  The CPU halts.  Some hours later, the user opens the
525    lid.  At this point, Xt's timers will fire, and xscreensaver will blank
526    the screen.
527
528    So far so good -- well, not really, but it's the best that we can do,
529    since the OS doesn't send us a signal *before* shutdown -- but if the
530    user had delayed locking (lockTimeout > 0) then we should start off
531    in the locked state, rather than only locking N minutes from when the
532    lid was opened.  Also, eschewing fading is probably a good idea, to
533    clamp down as soon as possible.
534
535    We only do this when we'd be polling the mouse position anyway.
536    This amounts to an assumption that machines with APM support also
537    have /proc/interrupts.
538  */
539 static void
540 check_for_clock_skew (saver_info *si)
541 {
542   saver_preferences *p = &si->prefs;
543   time_t now = time ((time_t *) 0);
544   long shift = now - si->last_wall_clock_time;
545
546   if (p->debug_p)
547     {
548       int i = (si->last_wall_clock_time == 0 ? 0 : shift);
549       fprintf (stderr,
550                "%s: checking wall clock for hibernation (%d:%02d:%02d).\n",
551                blurb(),
552                (i / (60 * 60)), ((i / 60) % 60), (i % 60));
553     }
554
555   if (si->last_wall_clock_time != 0 &&
556       shift > (p->timeout / 1000))
557     {
558       if (p->verbose_p)
559         fprintf (stderr, "%s: wall clock has jumped by %ld:%02ld:%02ld!\n",
560                  blurb(),
561                  (shift / (60 * 60)), ((shift / 60) % 60), (shift % 60));
562
563       si->emergency_lock_p = True;
564       idle_timer ((XtPointer) si, 0);
565     }
566
567   si->last_wall_clock_time = now;
568 }
569
570
571
572 static void
573 dispatch_event (saver_info *si, XEvent *event)
574 {
575   /* If this is for the splash dialog, pass it along.
576      Note that the password dialog is handled with its own event loop,
577      so events for that window will never come through here.
578    */
579   if (si->splash_dialog && event->xany.window == si->splash_dialog)
580     handle_splash_event (si, event);
581
582   XtDispatchEvent (event);
583 }
584
585
586 static void
587 swallow_unlock_typeahead_events (saver_info *si, XEvent *e)
588 {
589   XEvent event;
590   char buf [100];
591   int i = 0;
592
593   memset (buf, 0, sizeof(buf));
594
595   event = *e;
596
597   do
598     {
599       if (event.xany.type == KeyPress)
600         {
601           char s[2];
602           int size = XLookupString ((XKeyEvent *) &event, s, 1, 0, 0);
603           if (size != 1) continue;
604           switch (*s)
605             {
606             case '\010': case '\177':                   /* Backspace */
607               if (i > 0) i--;
608               break;
609             case '\025': case '\030':                   /* Erase line */
610             case '\012': case '\015':                   /* Enter */
611               i = 0;
612               break;
613             case '\040':                                /* Space */
614               if (i == 0)
615                 break;  /* ignore space at beginning of line */
616               /* else, fall through */
617             default:
618               buf [i++] = *s;
619               break;
620             }
621         }
622
623     } while (i < sizeof(buf)-1 &&
624              XCheckMaskEvent (si->dpy, KeyPressMask, &event));
625
626   buf[i] = 0;
627
628   if (si->unlock_typeahead)
629     {
630       memset (si->unlock_typeahead, 0, strlen(si->unlock_typeahead));
631       free (si->unlock_typeahead);
632     }
633
634   if (i > 0)
635     si->unlock_typeahead = strdup (buf);
636   else
637     si->unlock_typeahead = 0;
638
639   memset (buf, 0, sizeof(buf));
640 }
641
642
643 /* methods of detecting idleness:
644
645       explicitly informed by SGI SCREEN_SAVER server event;
646       explicitly informed by MIT-SCREEN-SAVER server event;
647       poll server idle time with XIDLE extension;
648       select events on all windows, and note absence of recent events;
649       note that /proc/interrupts has not changed in a while;
650       activated by clientmessage.
651
652    methods of detecting non-idleness:
653
654       read events on the xscreensaver window;
655       explicitly informed by SGI SCREEN_SAVER server event;
656       explicitly informed by MIT-SCREEN-SAVER server event;
657       select events on all windows, and note events on any of them;
658       note that /proc/interrupts has changed;
659       deactivated by clientmessage.
660
661    I trust that explains why this function is a big hairy mess.
662  */
663 void
664 sleep_until_idle (saver_info *si, Bool until_idle_p)
665 {
666   saver_preferences *p = &si->prefs;
667   XEvent event;
668
669   /* We need to select events on all windows if we're not using any extensions.
670      Otherwise, we don't need to. */
671   Bool scanning_all_windows = !(si->using_xidle_extension ||
672                                 si->using_mit_saver_extension ||
673                                 si->using_sgi_saver_extension);
674
675   /* We need to periodically wake up and check for idleness if we're not using
676      any extensions, or if we're using the XIDLE extension.  The other two
677      extensions explicitly deliver events when we go idle/non-idle, so we
678      don't need to poll. */
679   Bool polling_for_idleness = !(si->using_mit_saver_extension ||
680                                 si->using_sgi_saver_extension);
681
682   /* Whether we need to periodically wake up and check to see if the mouse has
683      moved.  We only need to do this when not using any extensions.  The reason
684      this isn't the same as `polling_for_idleness' is that the "idleness" poll
685      can happen (for example) 5 minutes from now, whereas the mouse-position
686      poll should happen with low periodicity.  We don't need to poll the mouse
687      position with the XIDLE extension, but we do need to periodically wake up
688      and query the server with that extension.  For our purposes, polling
689      /proc/interrupts is just like polling the mouse position.  It has to
690      happen on the same kind of schedule. */
691   Bool polling_mouse_position = (si->using_proc_interrupts ||
692                                  !(si->using_xidle_extension ||
693                                    si->using_mit_saver_extension ||
694                                    si->using_sgi_saver_extension));
695
696   if (until_idle_p)
697     {
698       if (polling_for_idleness)
699         /* This causes a no-op event to be delivered to us in a while, so that
700            we come back around through the event loop again.  */
701         schedule_wakeup_event (si, p->timeout, p->debug_p);
702
703       if (polling_mouse_position)
704         /* Check to see if the mouse has moved, and set up a repeating timer
705            to do so periodically (typically, every 5 seconds.) */
706         check_pointer_timer ((XtPointer) si, 0);
707     }
708
709   while (1)
710     {
711       XtAppNextEvent (si->app, &event);
712
713       switch (event.xany.type) {
714       case 0:           /* our synthetic "timeout" event has been signalled */
715         if (until_idle_p)
716           {
717             Time idle;
718
719             /* We may be idle; check one last time to see if the mouse has
720                moved, just in case the idle-timer went off within the 5 second
721                window between mouse polling.  If the mouse has moved, then
722                check_pointer_timer() will reset last_activity_time.
723              */
724             if (polling_mouse_position)
725               check_pointer_timer ((XtPointer) si, 0);
726
727 #ifdef HAVE_XIDLE_EXTENSION
728             if (si->using_xidle_extension)
729               {
730                 /* The XIDLE extension uses the synthetic event to prod us into
731                    re-asking the server how long the user has been idle. */
732                 if (! XGetIdleTime (si->dpy, &idle))
733                   {
734                     fprintf (stderr, "%s: XGetIdleTime() failed.\n", blurb());
735                     saver_exit (si, 1, 0);
736                   }
737               }
738             else
739 #endif /* HAVE_XIDLE_EXTENSION */
740 #ifdef HAVE_MIT_SAVER_EXTENSION
741               if (si->using_mit_saver_extension)
742                 {
743                   /* We don't need to do anything in this case - the synthetic
744                      event isn't necessary, as we get sent specific events
745                      to wake us up.  In fact, this event generally shouldn't
746                      be being delivered when the MIT extension is in use. */
747                   idle = 0;
748                 }
749             else
750 #endif /* HAVE_MIT_SAVER_EXTENSION */
751 #ifdef HAVE_SGI_SAVER_EXTENSION
752               if (si->using_sgi_saver_extension)
753                 {
754                   /* We don't need to do anything in this case - the synthetic
755                      event isn't necessary, as we get sent specific events
756                      to wake us up.  In fact, this event generally shouldn't
757                      be being delivered when the SGI extension is in use. */
758                   idle = 0;
759                 }
760             else
761 #endif /* HAVE_SGI_SAVER_EXTENSION */
762               {
763                 /* Otherwise, no server extension is in use.  The synthetic
764                    event was to tell us to wake up and see if the user is now
765                    idle.  Compute the amount of idle time by comparing the
766                    `last_activity_time' to the wall clock.  The l_a_t was set
767                    by calling `reset_timers()', which is called only in only
768                    two situations: when polling the mouse position has revealed
769                    the the mouse has moved (user activity) or when we have read
770                    an event (again, user activity.)
771                  */
772                 idle = 1000 * (si->last_activity_time - time ((time_t *) 0));
773               }
774
775             if (idle >= p->timeout)
776               {
777                 /* Look, we've been idle long enough.  We're done. */
778                 goto DONE;
779               }
780             else if (si->emergency_lock_p)
781               {
782                 /* Oops, the wall clock has jumped far into the future, so
783                    we need to lock down in a hurry! */
784                 goto DONE;
785               }
786             else
787               {
788                 /* The event went off, but it turns out that the user has not
789                    yet been idle for long enough.  So re-signal the event.
790                    Be economical: if we should blank after 5 minutes, and the
791                    user has been idle for 2 minutes, then set this timer to
792                    go off in 3 minutes.
793                  */
794                 if (polling_for_idleness)
795                   schedule_wakeup_event (si, p->timeout - idle, p->debug_p);
796               }
797           }
798         break;
799
800       case ClientMessage:
801         if (handle_clientmessage (si, &event, until_idle_p))
802           goto DONE;
803         break;
804
805       case CreateNotify:
806         /* A window has been created on the screen somewhere.  If we're
807            supposed to scan all windows for events, prepare this window. */
808         if (scanning_all_windows)
809           {
810             Window w = event.xcreatewindow.window;
811             start_notice_events_timer (si, w, p->debug_p);
812           }
813         break;
814
815       case KeyPress:
816       case KeyRelease:
817       case ButtonPress:
818       case ButtonRelease:
819       case MotionNotify:
820
821         if (p->debug_p)
822           {
823             Window root=0, window=0;
824             int x=-1, y=-1;
825             const char *type = 0;
826             if (event.xany.type == MotionNotify)
827               {
828                 /*type = "MotionNotify";*/
829                 root = event.xmotion.root;
830                 window = event.xmotion.window;
831                 x = event.xmotion.x_root;
832                 y = event.xmotion.y_root;
833               }
834             else if (event.xany.type == KeyPress)
835               {
836                 type = "KeyPress";
837                 root = event.xkey.root;
838                 window = event.xkey.window;
839                 x = y = -1;
840               }
841             else if (event.xany.type == ButtonPress)
842               {
843                 type = "ButtonPress";
844                 root = event.xkey.root;
845                 window = event.xkey.window;
846                 x = event.xmotion.x_root;
847                 y = event.xmotion.y_root;
848               }
849
850             if (type)
851               {
852                 int i;
853                 for (i = 0; i < si->nscreens; i++)
854                   if (root == RootWindowOfScreen (si->screens[i].screen))
855                     break;
856                 fprintf (stderr,"%s: %d: %s on 0x%lx",
857                          blurb(), i, type, (unsigned long) window);
858
859                 /* Be careful never to do this unless in -debug mode, as
860                    this could expose characters from the unlock password. */
861                 if (p->debug_p && event.xany.type == KeyPress)
862                   {
863                     KeySym keysym;
864                     char c = 0;
865                     XLookupString (&event.xkey, &c, 1, &keysym, 0);
866                     fprintf (stderr, " (%s%s)",
867                              (event.xkey.send_event ? "synthetic " : ""),
868                              XKeysymToString (keysym));
869                   }
870
871                 if (x == -1)
872                   fprintf (stderr, "\n");
873                 else
874                   fprintf (stderr, " at %d,%d.\n", x, y);
875               }
876           }
877
878         /* If any widgets want to handle this event, let them. */
879         dispatch_event (si, &event);
880
881         
882         /* If we got a MotionNotify event, figure out what screen it
883            was on and poll the mouse there: if the mouse hasn't moved
884            far enough to count as "real" motion, then ignore this
885            event.
886          */
887         if (event.xany.type == MotionNotify)
888           {
889             int i;
890             for (i = 0; i < si->nscreens; i++)
891               if (event.xmotion.root ==
892                   RootWindowOfScreen (si->screens[i].screen))
893                 break;
894             if (i < si->nscreens)
895               {
896                 if (!pointer_moved_p (&si->screens[i], False))
897                   continue;
898               }
899           }
900
901
902         /* We got a user event.
903            If we're waiting for the user to become active, this is it.
904            If we're waiting until the user becomes idle, reset the timers
905            (since now we have longer to wait.)
906          */
907         if (!until_idle_p)
908           {
909             if (si->demoing_p &&
910                 (event.xany.type == MotionNotify ||
911                  event.xany.type == KeyRelease))
912               /* When we're demoing a single hack, mouse motion doesn't
913                  cause deactivation.  Only clicks and keypresses do. */
914               ;
915             else
916               /* If we're not demoing, then any activity causes deactivation.
917                */
918               goto DONE;
919           }
920         else
921           reset_timers (si);
922
923         break;
924
925       default:
926
927 #ifdef HAVE_MIT_SAVER_EXTENSION
928         if (event.type == si->mit_saver_ext_event_number)
929           {
930             /* This event's number is that of the MIT-SCREEN-SAVER server
931                extension.  This extension has one event number, and the event
932                itself contains sub-codes that say what kind of event it was
933                (an "idle" or "not-idle" event.)
934              */
935             XScreenSaverNotifyEvent *sevent =
936               (XScreenSaverNotifyEvent *) &event;
937             if (sevent->state == ScreenSaverOn)
938               {
939                 int i = 0;
940                 if (p->verbose_p)
941                   fprintf (stderr, "%s: MIT ScreenSaverOn event received.\n",
942                            blurb());
943
944                 /* Get the "real" server window(s) out of the way as soon
945                    as possible. */
946                 for (i = 0; i < si->nscreens; i++)
947                   {
948                     saver_screen_info *ssi = &si->screens[i];
949                     if (ssi->server_mit_saver_window &&
950                         window_exists_p (si->dpy,
951                                          ssi->server_mit_saver_window))
952                       XUnmapWindow (si->dpy, ssi->server_mit_saver_window);
953                   }
954
955                 if (sevent->kind != ScreenSaverExternal)
956                   {
957                     fprintf (stderr,
958                          "%s: ScreenSaverOn event wasn't of type External!\n",
959                              blurb());
960                   }
961
962                 if (until_idle_p)
963                   goto DONE;
964               }
965             else if (sevent->state == ScreenSaverOff)
966               {
967                 if (p->verbose_p)
968                   fprintf (stderr, "%s: MIT ScreenSaverOff event received.\n",
969                            blurb());
970                 if (!until_idle_p)
971                   goto DONE;
972               }
973             else
974               fprintf (stderr,
975                        "%s: unknown MIT-SCREEN-SAVER event %d received!\n",
976                        blurb(), sevent->state);
977           }
978         else
979
980 #endif /* HAVE_MIT_SAVER_EXTENSION */
981
982
983 #ifdef HAVE_SGI_SAVER_EXTENSION
984         if (event.type == (si->sgi_saver_ext_event_number + ScreenSaverStart))
985           {
986             /* The SGI SCREEN_SAVER server extension has two event numbers,
987                and this event matches the "idle" event. */
988             if (p->verbose_p)
989               fprintf (stderr, "%s: SGI ScreenSaverStart event received.\n",
990                        blurb());
991
992             if (until_idle_p)
993               goto DONE;
994           }
995         else if (event.type == (si->sgi_saver_ext_event_number +
996                                 ScreenSaverEnd))
997           {
998             /* The SGI SCREEN_SAVER server extension has two event numbers,
999                and this event matches the "idle" event. */
1000             if (p->verbose_p)
1001               fprintf (stderr, "%s: SGI ScreenSaverEnd event received.\n",
1002                        blurb());
1003             if (!until_idle_p)
1004               goto DONE;
1005           }
1006         else
1007 #endif /* HAVE_SGI_SAVER_EXTENSION */
1008
1009 #ifdef HAVE_RANDR
1010         if (event.type == (si->randr_event_number + RRScreenChangeNotify))
1011           {
1012             /* The Resize and Rotate extension sends an event when the
1013                size, rotation, or refresh rate of the screen has changed. */
1014
1015             XRRScreenChangeNotifyEvent *xrr_event =
1016               (XRRScreenChangeNotifyEvent *) &event;
1017             /* XRRRootToScreen is in Xrandr.h 1.4, 2001/06/07 */
1018             int screen = XRRRootToScreen (si->dpy, xrr_event->window);
1019
1020             if (p->verbose_p)
1021               {
1022                 if (si->screens[screen].width  == xrr_event->width &&
1023                     si->screens[screen].height == xrr_event->height)
1024                   fprintf (stderr,
1025                           "%s: %d: no-op screen size change event (%dx%d)\n",
1026                            blurb(), screen,
1027                            xrr_event->width, xrr_event->height);
1028                 else
1029                   fprintf (stderr,
1030                        "%s: %d: screen size changed from %dx%d to %dx%d\n",
1031                            blurb(), screen,
1032                            si->screens[screen].width,
1033                            si->screens[screen].height,
1034                            xrr_event->width, xrr_event->height);
1035               }
1036
1037 # ifdef RRScreenChangeNotifyMask
1038             /* Inform Xlib that it's ok to update its data structures. */
1039             XRRUpdateConfiguration (&event); /* Xrandr.h 1.9, 2002/09/29 */
1040 # endif /* RRScreenChangeNotifyMask */
1041
1042             /* Resize the existing xscreensaver windows and cached ssi data. */
1043             resize_screensaver_window (si);
1044           }
1045         else
1046 #endif /* HAVE_RANDR */
1047
1048           /* Just some random event.  Let the Widgets handle it, if desired. */
1049           dispatch_event (si, &event);
1050       }
1051     }
1052  DONE:
1053
1054
1055   /* If there's a user event on the queue, swallow it.
1056      If we're using a server extension, and the user becomes active, we
1057      get the extension event before the user event -- so the keypress or
1058      motion or whatever is still on the queue.  This makes "unfade" not
1059      work, because it sees that event, and bugs out.  (This problem
1060      doesn't exhibit itself without an extension, because in that case,
1061      there's only one event generated by user activity, not two.)
1062    */
1063   if (!until_idle_p && si->locked_p)
1064     swallow_unlock_typeahead_events (si, &event);
1065   else
1066     while (XCheckMaskEvent (si->dpy,
1067                             (KeyPressMask|ButtonPressMask|PointerMotionMask),
1068                      &event))
1069       ;
1070
1071
1072   if (si->check_pointer_timer_id)
1073     {
1074       XtRemoveTimeOut (si->check_pointer_timer_id);
1075       si->check_pointer_timer_id = 0;
1076     }
1077   if (si->timer_id)
1078     {
1079       XtRemoveTimeOut (si->timer_id);
1080       si->timer_id = 0;
1081     }
1082
1083   if (until_idle_p && si->cycle_id)     /* no cycle timer when inactive */
1084     abort ();
1085
1086   return;
1087 }
1088
1089
1090 \f
1091 /* Some crap for dealing with /proc/interrupts.
1092
1093    On Linux systems, it's possible to see the hardware interrupt count
1094    associated with the keyboard.  We can therefore use that as another method
1095    of detecting idleness.
1096
1097    Why is it a good idea to do this?  Because it lets us detect keyboard
1098    activity that is not associated with X events.  For example, if the user
1099    has switched to another virtual console, it's good for xscreensaver to not
1100    be running graphics hacks on the (non-visible) X display.  The common
1101    complaint that checking /proc/interrupts addresses is that the user is
1102    playing Quake on a non-X console, and the GL hacks are perceptibly slowing
1103    the game...
1104
1105    This is tricky for a number of reasons.
1106
1107      * First, we must be sure to only do this when running on an X server that
1108        is on the local machine (because otherwise, we'd be reacting to the
1109        wrong keyboard.)  The way we do this is by noting that the $DISPLAY is
1110        pointing to display 0 on the local machine.  It *could* be that display
1111        1 is also on the local machine (e.g., two X servers, each on a different
1112        virtual-terminal) but it's also possible that screen 1 is an X terminal,
1113        using this machine as the host.  So we can't take that chance.
1114
1115      * Second, one can only access these interrupt numbers in a completely
1116        and utterly brain-damaged way.  You would think that one would use an
1117        ioctl for this.  But no.  The ONLY way to get this information is to
1118        open the pseudo-file /proc/interrupts AS A FILE, and read the numbers
1119        out of it TEXTUALLY.  Because this is Unix, and all the world's a file,
1120        and the only real data type is the short-line sequence of ASCII bytes.
1121
1122        Now it's all well and good that the /proc/interrupts pseudo-file
1123        exists; that's a clever idea, and a useful API for things that are
1124        already textually oriented, like shell scripts, and users doing
1125        interactive debugging sessions.  But to make a *C PROGRAM* open a file
1126        and parse the textual representation of integers out of it is just
1127        insane.
1128
1129      * Third, you can't just hold the file open, and fseek() back to the
1130        beginning to get updated data!  If you do that, the data never changes.
1131        And I don't want to call open() every five seconds, because I don't want
1132        to risk going to disk for any inodes.  It turns out that if you dup()
1133        it early, then each copy gets fresh data, so we can get around that in
1134        this way (but for how many releases, one might wonder?)
1135
1136      * Fourth, the format of the output of the /proc/interrupts file is
1137        undocumented, and has changed several times already!  In Linux 2.0.33,
1138        even on a multiprocessor machine, it looks like this:
1139
1140           0:  309453991   timer
1141           1:    4771729   keyboard
1142    
1143        but in Linux 2.2 and 2.4 kernels with MP machines, it looks like this:
1144
1145                    CPU0       CPU1
1146           0:    1671450    1672618    IO-APIC-edge  timer
1147           1:      13037      13495    IO-APIC-edge  keyboard
1148
1149        and in Linux 2.6, it's gotten even goofier: now there are two lines
1150        labelled "i8042".  One of them is the keyboard, and one of them is
1151        the PS/2 mouse -- and of course, you can't tell them apart, except
1152        by wiggling the mouse and noting which one changes:
1153
1154                    CPU0       CPU1
1155           1:      32051      30864    IO-APIC-edge  i8042
1156          12:     476577     479913    IO-APIC-edge  i8042
1157
1158        Joy!  So how are we expected to parse that?  Well, this code doesn't
1159        parse it: it saves the first line with the string "keyboard" (or
1160        "i8042") in it, and does a string-comparison to note when it has
1161        changed.  If there are two "i8042" lines, we assume the first is
1162        the keyboard and the second is the mouse (doesn't matter which is
1163        which, really, as long as we don't compare them against each other.)
1164
1165    Thanks to Nat Friedman <nat@nat.org> for figuring out most of this crap.
1166
1167    Note that if you have a serial or USB mouse, or a USB keyboard, it won't
1168    detect it.  That's because there's no way to tell the difference between a
1169    serial mouse and a general serial port, and all USB devices look the same
1170    from here.  It would be somewhat unfortunate to have the screensaver turn
1171    off when the modem on COM1 burped, or when a USB disk was accessed.
1172  */
1173
1174
1175 #ifdef HAVE_PROC_INTERRUPTS
1176
1177 #define PROC_INTERRUPTS "/proc/interrupts"
1178
1179 Bool
1180 query_proc_interrupts_available (saver_info *si, const char **why)
1181 {
1182   /* We can use /proc/interrupts if $DISPLAY points to :0, and if the
1183      "/proc/interrupts" file exists and is readable.
1184    */
1185   FILE *f;
1186   if (why) *why = 0;
1187
1188   if (!display_is_on_console_p (si))
1189     {
1190       if (why) *why = "not on primary console";
1191       return False;
1192     }
1193
1194   f = fopen (PROC_INTERRUPTS, "r");
1195   if (!f)
1196     {
1197       if (why) *why = "does not exist";
1198       return False;
1199     }
1200
1201   fclose (f);
1202   return True;
1203 }
1204
1205
1206 static Bool
1207 proc_interrupts_activity_p (saver_info *si)
1208 {
1209   static FILE *f0 = 0;
1210   FILE *f1 = 0;
1211   int fd;
1212   static char last_kbd_line[255] = { 0, };
1213   static char last_ptr_line[255] = { 0, };
1214   char new_line[sizeof(last_kbd_line)];
1215   Bool checked_kbd = False, kbd_changed = False;
1216   Bool checked_ptr = False, ptr_changed = False;
1217   int i8042_count = 0;
1218
1219   if (!f0)
1220     {
1221       /* First time -- open the file. */
1222       f0 = fopen (PROC_INTERRUPTS, "r");
1223       if (!f0)
1224         {
1225           char buf[255];
1226           sprintf(buf, "%s: error opening %s", blurb(), PROC_INTERRUPTS);
1227           perror (buf);
1228           goto FAIL;
1229         }
1230     }
1231
1232   if (f0 == (FILE *) -1)            /* means we got an error initializing. */
1233     return False;
1234
1235   fd = dup (fileno (f0));
1236   if (fd < 0)
1237     {
1238       char buf[255];
1239       sprintf(buf, "%s: could not dup() the %s fd", blurb(), PROC_INTERRUPTS);
1240       perror (buf);
1241       goto FAIL;
1242     }
1243
1244   f1 = fdopen (fd, "r");
1245   if (!f1)
1246     {
1247       char buf[255];
1248       sprintf(buf, "%s: could not fdopen() the %s fd", blurb(),
1249               PROC_INTERRUPTS);
1250       perror (buf);
1251       goto FAIL;
1252     }
1253
1254   /* Actually, I'm unclear on why this fseek() is necessary, given the timing
1255      of the dup() above, but it is. */
1256   if (fseek (f1, 0, SEEK_SET) != 0)
1257     {
1258       char buf[255];
1259       sprintf(buf, "%s: error rewinding %s", blurb(), PROC_INTERRUPTS);
1260       perror (buf);
1261       goto FAIL;
1262     }
1263
1264   /* Now read through the pseudo-file until we find the "keyboard",
1265      "PS/2 mouse", or "i8042" lines. */
1266
1267   while (fgets (new_line, sizeof(new_line)-1, f1))
1268     {
1269       Bool i8042_p = !!strstr (new_line, "i8042");
1270       if (i8042_p) i8042_count++;
1271
1272       if (strchr (new_line, ','))
1273         {
1274           /* Ignore any line that has a comma on it: this is because
1275              a setup like this:
1276
1277                  12:     930935          XT-PIC  usb-uhci, PS/2 Mouse
1278
1279              is really bad news.  It *looks* like we can note mouse
1280              activity from that line, but really, that interrupt gets
1281              fired any time any USB device has activity!  So we have
1282              to ignore any shared IRQs.
1283            */
1284         }
1285       else if (!checked_kbd &&
1286                (strstr (new_line, "keyboard") ||
1287                 (i8042_p && i8042_count == 1)))
1288         {
1289           /* Assume the keyboard interrupt is the line that says "keyboard",
1290              or the *first* line that says "i8042".
1291            */
1292           kbd_changed = (*last_kbd_line && !!strcmp (new_line, last_kbd_line));
1293           strcpy (last_kbd_line, new_line);
1294           checked_kbd = True;
1295         }
1296       else if (!checked_ptr &&
1297                (strstr (new_line, "PS/2 Mouse") ||
1298                 (i8042_p && i8042_count == 2)))
1299         {
1300           /* Assume the mouse interrupt is the line that says "PS/2 mouse",
1301              or the *second* line that says "i8042".
1302            */
1303           ptr_changed = (*last_ptr_line && !!strcmp (new_line, last_ptr_line));
1304           strcpy (last_ptr_line, new_line);
1305           checked_ptr = True;
1306         }
1307
1308       if (checked_kbd && checked_ptr)
1309         break;
1310     }
1311
1312   if (checked_kbd || checked_ptr)
1313     {
1314       fclose (f1);
1315
1316       if (si->prefs.debug_p && (kbd_changed || ptr_changed))
1317         fprintf (stderr, "%s: /proc/interrupts activity: %s\n",
1318                  blurb(),
1319                  ((kbd_changed && ptr_changed) ? "mouse and kbd" :
1320                   kbd_changed ? "kbd" :
1321                   ptr_changed ? "mouse" : "ERR"));
1322
1323       return (kbd_changed || ptr_changed);
1324     }
1325
1326
1327   /* If we got here, we didn't find either a "keyboard" or a "PS/2 Mouse"
1328      line in the file at all. */
1329   fprintf (stderr, "%s: no keyboard or mouse data in %s?\n",
1330            blurb(), PROC_INTERRUPTS);
1331
1332  FAIL:
1333   if (f1)
1334     fclose (f1);
1335
1336   if (f0 && f0 != (FILE *) -1)
1337     fclose (f0);
1338
1339   f0 = (FILE *) -1;
1340   return False;
1341 }
1342
1343 #endif /* HAVE_PROC_INTERRUPTS */
1344
1345 \f
1346 /* This timer goes off every few minutes, whether the user is idle or not,
1347    to try and clean up anything that has gone wrong.
1348
1349    It calls disable_builtin_screensaver() so that if xset has been used,
1350    or some other program (like xlock) has messed with the XSetScreenSaver()
1351    settings, they will be set back to sensible values (if a server extension
1352    is in use, messing with xlock can cause xscreensaver to never get a wakeup
1353    event, and could cause monitor power-saving to occur, and all manner of
1354    heinousness.)
1355
1356    If the screen is currently blanked, it raises the window, in case some
1357    other window has been mapped on top of it.
1358
1359    If the screen is currently blanked, and there is no hack running, it
1360    clears the window, in case there is an error message printed on it (we
1361    don't want the error message to burn in.)
1362  */
1363
1364 static void
1365 watchdog_timer (XtPointer closure, XtIntervalId *id)
1366 {
1367   saver_info *si = (saver_info *) closure;
1368   saver_preferences *p = &si->prefs;
1369
1370   disable_builtin_screensaver (si, False);
1371
1372   /* If the DPMS settings on the server have changed, change them back to
1373      what ~/.xscreensaver says they should be. */
1374   sync_server_dpms_settings (si->dpy,
1375                              (p->dpms_enabled_p  &&
1376                               p->mode != DONT_BLANK),
1377                              p->dpms_standby / 1000,
1378                              p->dpms_suspend / 1000,
1379                              p->dpms_off / 1000,
1380                              False);
1381
1382   if (si->screen_blanked_p)
1383     {
1384       Bool running_p = screenhack_running_p (si);
1385
1386       if (si->dbox_up_p)
1387         {
1388           if (si->prefs.debug_p)
1389             fprintf (stderr, "%s: dialog box is up: not raising screen.\n",
1390                      blurb());
1391         }
1392       else
1393         {
1394           if (si->prefs.debug_p)
1395             fprintf (stderr, "%s: watchdog timer raising %sscreen.\n",
1396                      blurb(), (running_p ? "" : "and clearing "));
1397
1398           raise_window (si, True, True, running_p);
1399         }
1400
1401       if (screenhack_running_p (si) &&
1402           !monitor_powered_on_p (si))
1403         {
1404           if (si->prefs.verbose_p)
1405             fprintf (stderr,
1406                      "%s: X says monitor has powered down; "
1407                      "killing running hacks.\n", blurb());
1408           kill_screenhack (si);
1409         }
1410
1411       /* Re-schedule this timer.  The watchdog timer defaults to a bit less
1412          than the hack cycle period, but is never longer than one hour.
1413        */
1414       si->watchdog_id = 0;
1415       reset_watchdog_timer (si, True);
1416     }
1417 }
1418
1419
1420 void
1421 reset_watchdog_timer (saver_info *si, Bool on_p)
1422 {
1423   saver_preferences *p = &si->prefs;
1424
1425   if (si->watchdog_id)
1426     {
1427       XtRemoveTimeOut (si->watchdog_id);
1428       si->watchdog_id = 0;
1429     }
1430
1431   if (on_p && p->watchdog_timeout)
1432     {
1433       si->watchdog_id = XtAppAddTimeOut (si->app, p->watchdog_timeout,
1434                                          watchdog_timer, (XtPointer) si);
1435
1436       if (p->debug_p)
1437         fprintf (stderr, "%s: restarting watchdog_timer (%ld, %ld)\n",
1438                  blurb(), p->watchdog_timeout, si->watchdog_id);
1439     }
1440 }
1441
1442
1443 /* It's possible that a race condition could have led to the saver
1444    window being unexpectedly still mapped.  This can happen like so:
1445
1446     - screen is blanked
1447     - hack is launched
1448     - that hack tries to grab a screen image (it does this by
1449       first unmapping the saver window, then remapping it.)
1450     - hack unmaps window
1451     - hack waits
1452     - user becomes active
1453     - hack re-maps window (*)
1454     - driver kills subprocess
1455     - driver unmaps window (**)
1456
1457    The race is that (*) might have been sent to the server before
1458    the client process was killed, but, due to scheduling randomness,
1459    might not have been received by the server until after (**).
1460    In other words, (*) and (**) might happen out of order, meaning
1461    the driver will unmap the window, and then after that, the
1462    recently-dead client will re-map it.  This leaves the user
1463    locked out (it looks like a desktop, but it's not!)
1464
1465    To avoid this: after un-blanking the screen, we launch a timer
1466    that wakes up once a second for ten seconds, and makes damned
1467    sure that the window is still unmapped.
1468  */
1469
1470 void
1471 de_race_timer (XtPointer closure, XtIntervalId *id)
1472 {
1473   saver_info *si = (saver_info *) closure;
1474   saver_preferences *p = &si->prefs;
1475   int secs = 1;
1476
1477   if (id == 0)  /* if id is 0, this is the initialization call. */
1478     {
1479       si->de_race_ticks = 10;
1480       if (p->verbose_p)
1481         fprintf (stderr, "%s: starting de-race timer (%d seconds.)\n",
1482                  blurb(), si->de_race_ticks);
1483     }
1484   else
1485     {
1486       int i;
1487       XSync (si->dpy, False);
1488       for (i = 0; i < si->nscreens; i++)
1489         {
1490           saver_screen_info *ssi = &si->screens[i];
1491           Window w = ssi->screensaver_window;
1492           XWindowAttributes xgwa;
1493           XGetWindowAttributes (si->dpy, w, &xgwa);
1494           if (xgwa.map_state != IsUnmapped)
1495             {
1496               if (p->verbose_p)
1497                 fprintf (stderr,
1498                          "%s: %d: client race! emergency unmap 0x%lx.\n",
1499                          blurb(), i, (unsigned long) w);
1500               XUnmapWindow (si->dpy, w);
1501             }
1502           else if (p->debug_p)
1503             fprintf (stderr, "%s: %d: (de-race of 0x%lx is cool.)\n",
1504                      blurb(), i, (unsigned long) w);
1505         }
1506       XSync (si->dpy, False);
1507
1508       si->de_race_ticks--;
1509     }
1510
1511   if (id && *id == si->de_race_id)
1512     si->de_race_id = 0;
1513
1514   if (si->de_race_id) abort();
1515
1516   if (si->de_race_ticks <= 0)
1517     {
1518       si->de_race_id = 0;
1519       if (p->verbose_p)
1520         fprintf (stderr, "%s: de-race completed.\n", blurb());
1521     }
1522   else
1523     {
1524       si->de_race_id = XtAppAddTimeOut (si->app, secs * 1000,
1525                                         de_race_timer, closure);
1526     }
1527 }