1 /* screens.c --- dealing with RANDR, Xinerama, and VidMode Viewports.
2 * xscreensaver, Copyright (c) 1991-2008 Jamie Zawinski <jwz@jwz.org>
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
13 /* There are a bunch of different mechanisms for multiple monitors
14 * available in X. XScreenSaver needs to care about this for two
15 * reasons: first, to ensure that all visible areas go black; and
16 * second, so that the windows of screen savers exactly fill the
17 * glass of each monitor (instead of one saver spanning multiple
18 * monitors, or a monitor displaying only a sub-rectangle of the
23 * This is the original way. Each monitor gets its own display
24 * number. :0.0 is the first one, :0.1 is the next, etc. The
25 * value of $DISPLAY determines which screen windows open on by
26 * default. A single app can open windows on multiple screens
27 * with the same display connection, but windows cannot be moved
28 * from one screen to another. The mouse can be moved from one
29 * screen to another, though. Screens may be different depths
30 * (e.g., one can be TrueColor and one can be PseudoColor.)
31 * Screens cannot be resized or moved without restarting X.
33 * Everyone hates this way of doing things because of the
34 * inability to move a window from one screen to another without
35 * restarting the application.
39 * There is a single giant root window that spans all the
40 * monitors. All monitors are the same depth, and windows can be
41 * moved around. Applications can learn which rectangles are
42 * actually visible on monitors by querying the Xinerama server
43 * extension. (If you don't do that, you end up with dialog
44 * boxes that try to appear in the middle of the screen actually
45 * spanning the gap between two monitors.)
47 * Xinerama doesn't work with DRI, which means that if you use
48 * it, you lose hardware acceleration on OpenGL programs. Also,
49 * screens can't be resized or moved without restarting X.
51 * 3) Vidmode Viewports:
53 * With this extension, the root window can be bigger than the
54 * monitor. Moving the mouse near the edges of the screen
55 * scrolls around, like a pan-and-scan movie. There can also be
56 * a hot key for changing the monitor's resolution (zooming
59 * Trying to combine this with Xinerama crashes the server, so
60 * you can only use this if you have only a single screen, or are
61 * in old-multi-screen mode.
63 * Also, half the time it doesn't work at all: it tends to lie
64 * about the size of the rectangle in use.
68 * The first version of the "Resize and Rotate" extension let you
69 * change the resolution of a screen on the fly. The root window
70 * would actually resize. However, it was also incompatible with
71 * Xinerama (did it crash, or just do nothing? I can't remember)
72 * so you needed to be in single-screen or old multi-screen mode.
73 * I believe RANDR could co-exist with Vidmode Viewports, but I'm
78 * Finally, RANDR added the functionality of Xinerama, plus some.
79 * Each X screen (in the sense of #1, "multi-screen") can have a
80 * number of sub-rectangles that are displayed on monitors, and
81 * each of those sub-rectangles can be displayed on more than one
82 * monitor. So it's possible (I think) to have a hybrid of
83 * multi-screen and Xinerama (e.g., to have two monitors running
84 * in one depth, and three monitors running in another?)
85 * Typically though, there will be a single X screen, with
86 * Xinerama-like division of that large root window onto multiple
87 * monitors. Also everything's dynamic: monitors can be added,
88 * removed, and resized at runtime.
90 * I believe that as of RANDR 1.2, the Xinerama extension still
91 * exists but only as a compatiblity layer: it's actually
92 * returning data from the RANDR extension.
94 * Though RANDR 1.2 allows the same image to be cloned onto more
95 * than one monitor, and also allows one monitor to show a
96 * subsection of something on another monitor (e.g., the
97 * rectangles can be enclosed or overlap). Since there's no way
98 * to put seperate savers on those duplicated-or-overlapping
99 * monitors, xscreensaver just ignores them (which allows them to
100 * display duplicates or overlaps).
102 * 5a) Nvidia fucks it up:
104 * Nvidia drivers as of Aug 2008 running in "TwinView" mode
105 * apparently report correct screen geometry via Xinerama, but
106 * report one giant screen via RANDR. The response from the
107 * nvidia developers is, "we don't support RANDR, use Xinerama
108 * instead." Which is a seriously lame answer. So, xscreensaver
109 * has to query *both* extensions, and make a guess as to which
112 * 5b) Also sometimes RANDR says stupid shit like, "You have one
113 * screen, and it has no available orientations or sizes."
121 #include <X11/Xlib.h>
124 # include <X11/extensions/Xrandr.h>
125 #endif /* HAVE_RANDR */
128 # include <X11/extensions/Xinerama.h>
129 #endif /* HAVE_XINERAMA */
131 #ifdef HAVE_XF86VMODE
132 # include <X11/extensions/xf86vmode.h>
133 #endif /* HAVE_XF86VMODE */
135 /* This file doesn't need the Xt headers, so stub these types out... */
137 #define XtAppContext void*
138 #define XrmDatabase void*
139 #define XtIntervalId void*
140 #define XtPointer void*
143 #include "xscreensaver.h"
147 typedef enum { S_SANE, S_ENCLOSED, S_DUPLICATE, S_OVERLAP,
148 S_OFFSCREEN, S_DISABLED } monitor_sanity;
150 /* 'typedef monitor' is in types.h */
155 int x, y, width, height;
156 monitor_sanity sanity; /* I'm not crazy you're the one who's crazy */
157 int enemy; /* which monitor it overlaps or duplicates */
158 char *err; /* msg to print at appropriate later time;
159 exists only on monitor #0. */
162 static Bool layouts_differ_p (monitor **a, monitor **b);
166 free_monitors (monitor **monitors)
168 monitor **m2 = monitors;
169 if (! monitors) return;
172 if ((*m2)->desc) free ((*m2)->desc);
173 if ((*m2)->err) free ((*m2)->err);
182 append (char *s1, const char *s2)
184 char *s = (char *) malloc ((s1 ? strlen(s1) : 0) +
185 (s2 ? strlen(s2) : 0) + 3);
187 if (s1) strcat (s, s1);
188 if (s1 && s2) strcat (s, "\n");
189 if (s2) strcat (s, s2);
198 xinerama_scan_monitors (Display *dpy, char **errP)
200 Screen *screen = DefaultScreenOfDisplay (dpy);
201 int event, error, nscreens, i;
202 XineramaScreenInfo *xsi;
205 if (! XineramaQueryExtension (dpy, &event, &error))
208 if (! XineramaIsActive (dpy))
211 xsi = XineramaQueryScreens (dpy, &nscreens);
214 monitors = (monitor **) calloc (nscreens + 1, sizeof(*monitors));
215 if (!monitors) return 0;
217 for (i = 0; i < nscreens; i++)
219 monitor *m = (monitor *) calloc (1, sizeof (monitor));
225 m->width = xsi[i].width;
226 m->height = xsi[i].height;
231 #endif /* HAVE_XINERAMA */
234 #ifdef HAVE_XF86VMODE
237 vidmode_scan_monitors (Display *dpy, char **errP)
239 int event, error, nscreens, i;
242 /* Note that XF86VidModeGetViewPort() tends to be full of lies on laptops
243 that have a docking station or external monitor that runs in a different
244 resolution than the laptop's screen:
246 http://bugzilla.redhat.com/bugzilla/show_bug.cgi?id=81593
247 http://bugs.debian.org/cgi-bin/bugreport.cgi?bug=208417
248 http://bugs.xfree86.org/show_bug.cgi?id=421
250 Presumably this is fixed by using RANDR instead of VidMode.
253 # ifdef HAVE_XINERAMA
254 /* Attempts to use the VidMode extension when the Xinerama extension is
255 active can result in a server crash! Yay! */
256 if (XQueryExtension (dpy, "XINERAMA", &error, &event, &error))
258 # endif /* !HAVE_XINERAMA */
260 if (! XF86VidModeQueryExtension (dpy, &event, &error))
263 nscreens = ScreenCount (dpy);
264 monitors = (monitor **) calloc (nscreens + 1, sizeof(*monitors));
265 if (!monitors) return 0;
267 for (i = 0; i < nscreens; i++)
269 monitor *m = (monitor *) calloc (1, sizeof (monitor));
270 XF86VidModeModeLine ml;
272 Screen *screen = ScreenOfDisplay (dpy, i);
278 if (! safe_XF86VidModeGetViewPort (dpy, i, &m->x, &m->y))
281 if (XF86VidModeGetModeLine (dpy, i, &dot, &ml))
283 m->width = ml.hdisplay;
284 m->height = ml.vdisplay;
287 /* Apparently, though the server stores the X position in increments of
288 1 pixel, it will only make changes to the *display* in some other
289 increment. With XF86_SVGA on a Thinkpad, the display only updates
290 in multiples of 8 pixels when in 8-bit mode, and in multiples of 4
291 pixels in 16-bit mode. I don't know what it does in 24- and 32-bit
292 mode, because I don't have enough video memory to find out.
294 I consider it a bug that XF86VidModeGetViewPort() is telling me the
295 server's *target* scroll position rather than the server's *actual*
296 scroll position. David Dawes agrees, and says they may fix this in
297 XFree86 4.0, but it's notrivial.
299 He also confirms that this behavior is server-dependent, so the
300 actual scroll position cannot be reliably determined by the client.
301 So... that means the only solution is to provide a ``sandbox''
302 around the blackout window -- we make the window be up to N pixels
303 larger than the viewport on both the left and right sides. That
304 means some part of the outer edges of each hack might not be
305 visible, but screw it.
307 I'm going to guess that 16 pixels is enough, and that the Y dimension
308 doesn't have this problem.
310 The drawback of doing this, of course, is that some of the screenhacks
311 will still look pretty stupid -- for example, "slidescreen" will cut
312 off the left and right edges of the grid, etc.
315 if (m->x > 0 && m->x < m->width - ml.hdisplay)
317 /* Not at left edge or right edge:
318 Round X position down to next lower multiple of FUDGE.
319 Increase width by 2*FUDGE in case some server rounds up.
321 m->x = ((m->x - 1) / FUDGE) * FUDGE;
322 m->width += (FUDGE * 2);
330 #endif /* HAVE_XF86VMODE */
336 randr_scan_monitors (Display *dpy, char **errP)
338 int event, error, major, minor, nscreens, i, j;
340 Bool new_randr_p = False;
342 if (! XRRQueryExtension (dpy, &event, &error))
345 if (! XRRQueryVersion (dpy, &major, &minor))
348 if (major <= 0) /* Protocol was still in flux back then -- fuck it. */
351 # ifdef HAVE_RANDR_12
352 new_randr_p = (major > 1 || (major == 1 && minor >= 2));
356 /* RANDR 1.0 -- no Xinerama-like virtual screens. */
357 nscreens = ScreenCount (dpy);
358 else /* RANDR 1.2 or newer -- built-in Xinerama */
360 # ifdef HAVE_RANDR_12
361 int xsc = ScreenCount (dpy);
363 /* Add up the virtual screens on each X screen. */
364 for (i = 0; i < xsc; i++)
366 XRRScreenResources *res =
367 XRRGetScreenResources (dpy, RootWindow (dpy, i));
368 nscreens += res->noutput;
369 XRRFreeScreenResources (res);
371 # endif /* HAVE_RANDR_12 */
376 *errP = append (*errP,
377 "WARNING: RANDR reported no screens! Ignoring it.");
381 monitors = (monitor **) calloc (nscreens + 1, sizeof(*monitors));
382 if (!monitors) return 0;
384 for (i = 0, j = 0; i < ScreenCount (dpy); i++)
386 Screen *screen = ScreenOfDisplay (dpy, i);
388 if (! new_randr_p) /* RANDR 1.0 */
390 XRRScreenConfiguration *rrc;
391 monitor *m = (monitor *) calloc (1, sizeof (monitor));
396 rrc = XRRGetScreenInfo (dpy, RootWindowOfScreen (screen));
401 XRRScreenSize *rrsizes;
404 size = XRRConfigCurrentConfiguration (rrc, &rot);
405 rrsizes = XRRConfigSizes (rrc, &nsizes);
407 if (nsizes <= 0) /* WTF? Shouldn't happen but does. */
409 m->width = DisplayWidth (dpy, i);
410 m->height = DisplayHeight (dpy, i);
412 else if (rot & (RR_Rotate_90|RR_Rotate_270))
414 m->width = rrsizes[size].height;
415 m->height = rrsizes[size].width;
419 m->width = rrsizes[size].width;
420 m->height = rrsizes[size].height;
423 /* don't free 'rrsizes' */
424 XRRFreeScreenConfigInfo (rrc);
427 else /* RANDR 1.2 or newer */
429 # ifdef HAVE_RANDR_12
431 XRRScreenResources *res =
432 XRRGetScreenResources (dpy, RootWindowOfScreen (screen));
433 for (k = 0; k < res->noutput; k++, j++)
435 monitor *m = (monitor *) calloc (1, sizeof (monitor));
436 XRROutputInfo *rroi = XRRGetOutputInfo (dpy, res,
438 RRCrtc crtc = (rroi->crtc ? rroi->crtc :
439 rroi->ncrtc ? rroi->crtcs[0] : 0);
440 XRRCrtcInfo *crtci = (crtc ? XRRGetCrtcInfo(dpy, res, crtc) : 0);
444 m->id = (i * 1000) + j;
445 m->desc = (rroi->name ? strdup (rroi->name) : 0);
449 /* Note: if the screen is rotated, XRRConfigSizes contains
450 the unrotated WxH, but XRRCrtcInfo contains rotated HxW.
454 m->width = crtci->width;
455 m->height = crtci->height;
458 if (rroi->connection == RR_Disconnected)
459 m->sanity = S_DISABLED;
460 /* #### do the same for RR_UnknownConnection? */
463 XRRFreeCrtcInfo (crtci);
464 XRRFreeOutputInfo (rroi);
466 XRRFreeScreenResources (res);
467 # endif /* HAVE_RANDR_12 */
471 /* Work around more fucking brain damage. */
477 if (monitors[i]->width != 0 && monitors[i]->height != 0)
483 *errP = append (*errP,
484 "WARNING: RANDR says all screens are 0x0! Ignoring it.");
485 free_monitors (monitors);
493 #endif /* HAVE_RANDR */
497 basic_scan_monitors (Display *dpy, char **errP)
499 int nscreens = ScreenCount (dpy);
501 monitor **monitors = (monitor **) calloc (nscreens + 1, sizeof(*monitors));
502 if (!monitors) return 0;
504 for (i = 0; i < nscreens; i++)
506 Screen *screen = ScreenOfDisplay (dpy, i);
507 monitor *m = (monitor *) calloc (1, sizeof (monitor));
513 m->width = WidthOfScreen (screen);
514 m->height = HeightOfScreen (screen);
520 #if defined(HAVE_RANDR) && defined(HAVE_XINERAMA)
522 /* From: Aaron Plattner <aplattner@nvidia.com>
523 Date: August 7, 2008 10:21:25 AM PDT
524 To: linux-bugs@nvidia.com
526 The NVIDIA X driver does not yet support RandR 1.2. The X server has
527 a compatibility layer in it that allows RandR 1.2 clients to talk to
528 RandR 1.1 drivers through an RandR 1.2 pseudo-output called "default".
529 This reports the total combined resolution of the TwinView display,
530 since it doesn't have any visibility into TwinView metamodes. There
531 is no way for the driver to prevent the server from turning on this
534 The intention is for X client applications to continue to use the
535 Xinerama extension to query the screen geometry. RandR 1.2 reports
536 its own Xinerama info for this purpose. I would recommend against
537 modifying xscreensaver to try to get this information from RandR.
540 randr_versus_xinerama_fight (Display *dpy, monitor **randr_monitors,
543 monitor **xinerama_monitors;
548 xinerama_monitors = xinerama_scan_monitors (dpy, errP);
549 if (!xinerama_monitors)
550 return randr_monitors;
552 if (! layouts_differ_p (randr_monitors, xinerama_monitors))
554 free_monitors (xinerama_monitors);
555 return randr_monitors;
557 else if ( randr_monitors[0] && !randr_monitors[1] && /* 1 monitor */
558 xinerama_monitors[0] && xinerama_monitors[1]) /* >1 monitor */
560 *errP = append (*errP,
561 "WARNING: RANDR reports 1 screen but Xinerama\n"
562 "\t\treports multiple. Believing Xinerama.");
563 free_monitors (randr_monitors);
564 return xinerama_monitors;
568 *errP = append (*errP,
569 "WARNING: RANDR and Xinerama report different\n"
570 "\t\tscreen layouts! Believing RANDR.");
571 free_monitors (xinerama_monitors);
572 return randr_monitors;
576 #endif /* HAVE_RANDR && HAVE_XINERAMA */
579 #ifdef DEBUG_MULTISCREEN
581 /* If DEBUG_MULTISCREEN is defined, then in "-debug" mode, xscreensaver
582 will pretend that it is changing the number of connected monitors
583 every few seconds, using the geometries in the following list,
584 for stress-testing purposes.
587 debug_scan_monitors (Display *dpy, char **errP)
589 static const char * const geoms[] = {
593 "800x600+0+22,800x600+800+22",
594 "800x600+0+22,800x600+800+22,800x600+300+622",
595 "800x600+0+22,800x600+800+22,800x600+0+622,800x600+800+622",
596 "640x480+0+22,640x480+640+22,640x480+0+502,640x480+640+502",
597 "640x480+240+22,640x480+0+502,640x480+640+502",
598 "640x480+0+200,640x480+640+200",
600 "320x200+0+22,320x200+320+22,320x200+640+22,320x200+960+22,320x200+0+222,320x200+320+222,320x200+640+222,320x200+960+222,320x200+0+422,320x200+320+422,320x200+640+422,320x200+960+422,320x200+0+622,320x200+320+622,320x200+640+622,320x200+960+622,320x200+0+822,320x200+320+822,320x200+640+822,320x200+960+822"
602 static int index = 0;
603 monitor **monitors = (monitor **) calloc (100, sizeof(*monitors));
605 Screen *screen = DefaultScreenOfDisplay (dpy);
607 char *s = strdup (geoms[index]);
608 char *token = strtok (s, ",");
611 monitor *m = calloc (1, sizeof (monitor));
615 if (4 != sscanf (token, "%dx%d+%d+%d%c",
616 &m->width, &m->height, &m->x, &m->y, &c))
620 monitors[nscreens++] = m;
621 token = strtok (0, ",");
625 index = (index+1) % countof(geoms);
629 #endif /* DEBUG_MULTISCREEN */
634 quadruple (monitor **monitors, Bool debug_p, char **errP)
638 while (monitors[count])
640 monitors2 = (monitor **) calloc (count * 4 + 1, sizeof(*monitors));
641 if (!monitors2) abort();
643 for (i = 0, j = 0; i < count; i++)
646 for (k = 0; k < 4; k++)
648 monitors2[j+k] = (monitor *) calloc (1, sizeof (monitor));
649 *monitors2[j+k] = *monitors[i];
650 monitors2[j+k]->width /= (debug_p ? 4 : 2);
651 monitors2[j+k]->height /= 2;
652 monitors2[j+k]->id = (monitors[i]->id * 4) + k;
653 monitors2[j+k]->name = (monitors[i]->name
654 ? strdup (monitors[i]->name) : 0);
656 monitors2[j+1]->x += monitors2[j]->width;
657 monitors2[j+2]->y += monitors2[j]->height;
658 monitors2[j+3]->x += monitors2[j]->width;
659 monitors2[j+3]->y += monitors2[j]->height;
663 free_monitors (monitors);
666 #endif /* QUAD_MODE */
670 scan_monitors (saver_info *si)
672 saver_preferences *p = &si->prefs;
673 monitor **monitors = 0;
676 # ifdef DEBUG_MULTISCREEN
677 if (! monitors) monitors = debug_scan_monitors (si->dpy, &err);
681 if (! p->getviewport_full_of_lies_p)
682 if (! monitors) monitors = randr_scan_monitors (si->dpy, &err);
684 # ifdef HAVE_XINERAMA
685 monitors = randr_versus_xinerama_fight (si->dpy, monitors, &err);
687 # endif /* HAVE_RANDR */
689 # ifdef HAVE_XF86VMODE
690 if (! monitors) monitors = vidmode_scan_monitors (si->dpy, &err);
693 # ifdef HAVE_XINERAMA
694 if (! monitors) monitors = xinerama_scan_monitors (si->dpy, &err);
697 if (! monitors) monitors = basic_scan_monitors (si->dpy, &err);
701 monitors = quadruple (monitors, p->debug_p, &err);
704 if (monitors && err) monitors[0]->err = err;
711 monitors_overlap_p (monitor *a, monitor *b)
713 /* Two rectangles overlap if the max of the tops is less than the
714 min of the bottoms and the max of the lefts is less than the min
719 # define MAX(A,B) ((A)>(B)?(A):(B))
720 # define MIN(A,B) ((A)<(B)?(A):(B))
722 int maxleft = MAX(a->x, b->x);
723 int maxtop = MAX(a->y, b->y);
724 int minright = MIN(a->x + a->width - 1, b->x + b->width);
725 int minbot = MIN(a->y + a->height - 1, b->y + b->height);
726 return (maxtop < minbot && maxleft < minright);
731 plausible_aspect_ratio_p (monitor **monitors)
733 /* Modern wide-screen monitors come in the following aspect ratios:
735 One monitor: If you tack a 640x480 monitor
736 onto the right, the ratio is:
738 852 x 480 --> 1.77 852+640 x 480 --> 3.11 "SD 480p"
739 1280 x 720 --> 1.78 1280+640 x 720 --> 2.67 "HD 720p"
740 1280 x 920 --> 1.39 1280+640 x 920 --> 2.09
741 1366 x 768 --> 1.78 1366+640 x 768 --> 2.61 "HD 768p"
742 1440 x 900 --> 1.60 1440+640 x 900 --> 2.31
743 1680 x 1050 --> 1.60 1680+640 x 1050 --> 2.21
744 1690 x 1050 --> 1.61 1690+640 x 1050 --> 2.22
745 1920 x 1080 --> 1.78 1920+640 x 1080 --> 2.37 "HD 1080p"
746 1920 x 1200 --> 1.60 1920+640 x 1200 --> 2.13
747 2560 x 1600 --> 1.60 2560+640 x 1600 --> 2.00
749 So that implies that if we ever see an aspect ratio >= 2.0,
750 we can be pretty sure that the X server is lying to us, and
751 that's actually two monitors, not one.
753 if (monitors[0] && !monitors[1] && /* exactly 1 monitor */
754 monitors[0]->height &&
755 monitors[0]->width / (double) monitors[0]->height >= 1.9)
762 /* Mark the ones that overlap, etc.
765 check_monitor_sanity (monitor **monitors)
769 while (monitors[count])
772 # define X1 monitors[i]->x
773 # define X2 monitors[j]->x
774 # define Y1 monitors[i]->y
775 # define Y2 monitors[j]->y
776 # define W1 monitors[i]->width
777 # define W2 monitors[j]->width
778 # define H1 monitors[i]->height
779 # define H2 monitors[j]->height
781 /* If a monitor is enclosed by any other monitor, that's insane.
783 for (i = 0; i < count; i++)
784 for (j = 0; j < count; j++)
786 monitors[i]->sanity == S_SANE &&
787 monitors[j]->sanity == S_SANE &&
788 monitors[i]->screen == monitors[j]->screen &&
791 (X2+W2) <= (X1+W1) &&
798 monitors[j]->sanity = S_DUPLICATE;
800 monitors[j]->sanity = S_ENCLOSED;
801 monitors[j]->enemy = i;
804 /* After checking for enclosure, check for other lossage against earlier
805 monitors. We do enclosure first so that we make sure to pick the
808 for (i = 0; i < count; i++)
809 for (j = 0; j < i; j++)
811 if (monitors[i]->sanity != S_SANE) continue; /* already marked */
812 if (monitors[j]->sanity != S_SANE) continue;
813 if (monitors[i]->screen != monitors[j]->screen) continue;
815 if (monitors_overlap_p (monitors[i], monitors[j]))
817 monitors[i]->sanity = S_OVERLAP;
818 monitors[i]->enemy = j;
822 /* Finally, make sure all monitors have sane positions and sizes.
823 Xinerama sometimes reports 1024x768 VPs at -1936862040, -1953705044.
825 for (i = 0; i < count; i++)
827 if (monitors[i]->sanity != S_SANE) continue; /* already marked */
828 if (X1 < 0 || Y1 < 0 ||
829 W1 <= 0 || H1 <= 0 ||
830 X1+W1 >= 0x7FFF || Y1+H1 >= 0x7FFF)
832 monitors[i]->sanity = S_OFFSCREEN;
833 monitors[i]->enemy = 0;
849 layouts_differ_p (monitor **a, monitor **b)
851 if (!a || !b) return True;
856 if ((*a)->screen != (*b)->screen ||
857 (*a)->x != (*b)->x ||
858 (*a)->y != (*b)->y ||
859 (*a)->width != (*b)->width ||
860 (*a)->height != (*b)->height)
873 describe_monitor_layout (saver_info *si)
875 monitor **monitors = si->monitor_layout;
879 int implausible_p = !plausible_aspect_ratio_p (monitors);
881 while (monitors[count])
883 if (monitors[count]->sanity == S_SANE)
890 if (monitors[0]->err) /* deferred error msg */
892 char *token = strtok (monitors[0]->err, "\n");
895 fprintf (stderr, "%s: %s\n", blurb(), token);
896 token = strtok (0, "\n");
898 free (monitors[0]->err);
899 monitors[0]->err = 0;
903 fprintf (stderr, "%s: no screens!\n", blurb());
907 fprintf (stderr, "%s: screens in use: %d\n", blurb(), good_count);
908 for (i = 0; i < count; i++)
910 monitor *m = monitors[i];
911 if (m->sanity != S_SANE) continue;
912 fprintf (stderr, "%s: %3d/%d: %dx%d+%d+%d",
913 blurb(), m->id, screen_number (m->screen),
914 m->width, m->height, m->x, m->y);
915 if (m->desc && *m->desc) fprintf (stderr, " (%s)", m->desc);
916 fprintf (stderr, "\n");
920 fprintf (stderr, "%s: rejected screens: %d\n", blurb(), bad_count);
921 for (i = 0; i < count; i++)
923 monitor *m = monitors[i];
924 monitor *e = monitors[m->enemy];
925 if (m->sanity == S_SANE) continue;
926 fprintf (stderr, "%s: %3d/%d: %dx%d+%d+%d",
927 blurb(), m->id, screen_number (m->screen),
928 m->width, m->height, m->x, m->y);
929 if (m->desc && *m->desc) fprintf (stderr, " (%s)", m->desc);
930 fprintf (stderr, " -- ");
933 case S_SANE: abort(); break;
935 fprintf (stderr, "enclosed by %d (%dx%d+%d+%d)\n",
936 e->id, e->width, e->height, e->x, e->y);
939 fprintf (stderr, "duplicate of %d\n", e->id);
942 fprintf (stderr, "overlaps %d (%dx%d+%d+%d)\n",
943 e->id, e->width, e->height, e->x, e->y);
946 fprintf (stderr, "off screen (%dx%d)\n",
947 WidthOfScreen (e->screen),
948 HeightOfScreen (e->screen));
951 fprintf (stderr, "output disabled\n");
959 "%s: WARNING: single screen aspect ratio is %dx%d = %.2f\n"
960 "%s: probable X server bug in Xinerama/RANDR!\n",
961 blurb(), monitors[0]->width, monitors[0]->height,
962 monitors[0]->width / (double) monitors[0]->height,
968 /* Synchronize the contents of si->ssi to the current state of the monitors.
969 Doesn't change anything if nothing has changed; otherwise, alters and
970 reuses existing saver_screen_info structs as much as possible.
971 Returns True if anything changed.
974 update_screen_layout (saver_info *si)
976 monitor **monitors = scan_monitors (si);
980 int seen_screens[100] = { 0, };
982 if (! layouts_differ_p (monitors, si->monitor_layout))
984 free_monitors (monitors);
988 free_monitors (si->monitor_layout);
989 si->monitor_layout = monitors;
990 check_monitor_sanity (si->monitor_layout);
992 while (monitors[count])
994 if (monitors[count]->sanity == S_SANE)
999 if (si->ssi_count == 0)
1002 si->screens = (saver_screen_info *)
1003 calloc (sizeof(*si->screens), si->ssi_count);
1006 if (si->ssi_count <= good_count)
1008 si->ssi_count = good_count + 10;
1009 si->screens = (saver_screen_info *)
1010 realloc (si->screens, sizeof(*si->screens) * si->ssi_count);
1011 memset (si->screens + si->nscreens, 0,
1012 sizeof(*si->screens) * (si->ssi_count - si->nscreens));
1015 if (! si->screens) abort();
1017 si->nscreens = good_count;
1019 /* Regenerate the list of GL visuals as needed. */
1020 if (si->best_gl_visuals)
1021 free (si->best_gl_visuals);
1022 si->best_gl_visuals = 0;
1024 for (i = 0, j = 0; i < count; i++)
1026 monitor *m = monitors[i];
1027 saver_screen_info *ssi = &si->screens[j];
1028 Screen *old_screen = ssi->screen;
1030 if (monitors[i]->sanity != S_SANE) continue;
1035 sn = screen_number (m->screen);
1036 ssi->screen = m->screen;
1037 ssi->real_screen_number = sn;
1038 ssi->real_screen_p = (seen_screens[sn] == 0);
1041 ssi->default_visual =
1042 get_visual_resource (ssi->screen, "visualID", "VisualID", False);
1043 ssi->current_visual = ssi->default_visual;
1044 ssi->current_depth = visual_depth (ssi->screen, ssi->current_visual);
1046 /* If the screen changed (or if this is the first time) we need
1047 a new toplevel shell for this screen's depth.
1049 if (ssi->screen != old_screen)
1050 initialize_screen_root_widget (ssi);
1052 ssi->last_poll_mouse.root_x = -1;
1053 ssi->last_poll_mouse.root_y = -1;
1057 ssi->width = m->width;
1058 ssi->height = m->height;
1060 # ifndef DEBUG_MULTISCREEN
1062 saver_preferences *p = &si->prefs;
1075 si->default_screen = &si->screens[0];