xref: /OK3568_Linux_fs/external/xserver/hw/dmx/dmxcb.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright 2001-2004 Red Hat Inc., Durham, North Carolina.
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * All Rights Reserved.
5*4882a593Smuzhiyun  *
6*4882a593Smuzhiyun  * Permission is hereby granted, free of charge, to any person obtaining
7*4882a593Smuzhiyun  * a copy of this software and associated documentation files (the
8*4882a593Smuzhiyun  * "Software"), to deal in the Software without restriction, including
9*4882a593Smuzhiyun  * without limitation on the rights to use, copy, modify, merge,
10*4882a593Smuzhiyun  * publish, distribute, sublicense, and/or sell copies of the Software,
11*4882a593Smuzhiyun  * and to permit persons to whom the Software is furnished to do so,
12*4882a593Smuzhiyun  * subject to the following conditions:
13*4882a593Smuzhiyun  *
14*4882a593Smuzhiyun  * The above copyright notice and this permission notice (including the
15*4882a593Smuzhiyun  * next paragraph) shall be included in all copies or substantial
16*4882a593Smuzhiyun  * portions of the Software.
17*4882a593Smuzhiyun  *
18*4882a593Smuzhiyun  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
19*4882a593Smuzhiyun  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20*4882a593Smuzhiyun  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
21*4882a593Smuzhiyun  * NON-INFRINGEMENT.  IN NO EVENT SHALL RED HAT AND/OR THEIR SUPPLIERS
22*4882a593Smuzhiyun  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
23*4882a593Smuzhiyun  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
24*4882a593Smuzhiyun  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
25*4882a593Smuzhiyun  * SOFTWARE.
26*4882a593Smuzhiyun  */
27*4882a593Smuzhiyun 
28*4882a593Smuzhiyun /*
29*4882a593Smuzhiyun  * Authors:
30*4882a593Smuzhiyun  *   Rickard E. (Rik) Faith <faith@redhat.com>
31*4882a593Smuzhiyun  *
32*4882a593Smuzhiyun  */
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun /** \file
35*4882a593Smuzhiyun  * This code queries and modifies the connection block. */
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun #ifdef HAVE_DMX_CONFIG_H
38*4882a593Smuzhiyun #include <dmx-config.h>
39*4882a593Smuzhiyun #endif
40*4882a593Smuzhiyun 
41*4882a593Smuzhiyun #include "dmx.h"
42*4882a593Smuzhiyun #include "dmxcb.h"
43*4882a593Smuzhiyun #include "dmxinput.h"
44*4882a593Smuzhiyun #include "dmxlog.h"
45*4882a593Smuzhiyun 
46*4882a593Smuzhiyun extern int connBlockScreenStart;
47*4882a593Smuzhiyun 
48*4882a593Smuzhiyun #ifdef PANORAMIX
49*4882a593Smuzhiyun #include "panoramiXsrv.h"
50*4882a593Smuzhiyun #endif
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun int dmxGlobalWidth, dmxGlobalHeight;
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun /** We may want the wall dimensions to be different from the bounding
55*4882a593Smuzhiyun  * box dimensions that Xinerama computes, so save those and update them
56*4882a593Smuzhiyun  * here.
57*4882a593Smuzhiyun  */
58*4882a593Smuzhiyun void
dmxSetWidthHeight(int width,int height)59*4882a593Smuzhiyun dmxSetWidthHeight(int width, int height)
60*4882a593Smuzhiyun {
61*4882a593Smuzhiyun     dmxGlobalWidth = width;
62*4882a593Smuzhiyun     dmxGlobalHeight = height;
63*4882a593Smuzhiyun }
64*4882a593Smuzhiyun 
65*4882a593Smuzhiyun /** Computes the global bounding box for DMX.  This may be larger than
66*4882a593Smuzhiyun  * the one computed by Xinerama because of the DMX configuration
67*4882a593Smuzhiyun  * file. */
68*4882a593Smuzhiyun void
dmxComputeWidthHeight(DMXRecomputeFlag flag)69*4882a593Smuzhiyun dmxComputeWidthHeight(DMXRecomputeFlag flag)
70*4882a593Smuzhiyun {
71*4882a593Smuzhiyun     int i;
72*4882a593Smuzhiyun     DMXScreenInfo *dmxScreen;
73*4882a593Smuzhiyun     int w = 0;
74*4882a593Smuzhiyun     int h = 0;
75*4882a593Smuzhiyun 
76*4882a593Smuzhiyun     for (i = 0; i < dmxNumScreens; i++) {
77*4882a593Smuzhiyun         /* Don't use root* here because this is
78*4882a593Smuzhiyun          * the global bounding box. */
79*4882a593Smuzhiyun         dmxScreen = &dmxScreens[i];
80*4882a593Smuzhiyun         if (w < dmxScreen->scrnWidth + dmxScreen->rootXOrigin)
81*4882a593Smuzhiyun             w = dmxScreen->scrnWidth + dmxScreen->rootXOrigin;
82*4882a593Smuzhiyun         if (h < dmxScreen->scrnHeight + dmxScreen->rootYOrigin)
83*4882a593Smuzhiyun             h = dmxScreen->scrnHeight + dmxScreen->rootYOrigin;
84*4882a593Smuzhiyun     }
85*4882a593Smuzhiyun     if (!dmxGlobalWidth && !dmxGlobalHeight) {
86*4882a593Smuzhiyun         dmxLog(dmxInfo, "Using %dx%d as global bounding box\n", w, h);
87*4882a593Smuzhiyun     }
88*4882a593Smuzhiyun     else {
89*4882a593Smuzhiyun         switch (flag) {
90*4882a593Smuzhiyun         case DMX_NO_RECOMPUTE_BOUNDING_BOX:
91*4882a593Smuzhiyun             dmxLog(dmxInfo,
92*4882a593Smuzhiyun                    "Using old bounding box (%dx%d) instead of new (%dx%d)\n",
93*4882a593Smuzhiyun                    dmxGlobalWidth, dmxGlobalHeight, w, h);
94*4882a593Smuzhiyun             w = dmxGlobalWidth;
95*4882a593Smuzhiyun             h = dmxGlobalHeight;
96*4882a593Smuzhiyun             break;
97*4882a593Smuzhiyun         case DMX_RECOMPUTE_BOUNDING_BOX:
98*4882a593Smuzhiyun             dmxLog(dmxInfo,
99*4882a593Smuzhiyun                    "Using %dx%d as global bounding box, instead of %dx%d\n",
100*4882a593Smuzhiyun                    w, h, dmxGlobalWidth, dmxGlobalHeight);
101*4882a593Smuzhiyun             break;
102*4882a593Smuzhiyun         }
103*4882a593Smuzhiyun     }
104*4882a593Smuzhiyun 
105*4882a593Smuzhiyun     dmxGlobalWidth = w;
106*4882a593Smuzhiyun     dmxGlobalHeight = h;
107*4882a593Smuzhiyun }
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun /** A callback routine that hooks into Xinerama and provides a
110*4882a593Smuzhiyun  * convenient place to print summary log information during server
111*4882a593Smuzhiyun  * startup.  This routine does not modify any values. */
112*4882a593Smuzhiyun void
dmxConnectionBlockCallback(void)113*4882a593Smuzhiyun dmxConnectionBlockCallback(void)
114*4882a593Smuzhiyun {
115*4882a593Smuzhiyun     xWindowRoot *root = (xWindowRoot *) (ConnectionInfo + connBlockScreenStart);
116*4882a593Smuzhiyun     int offset = connBlockScreenStart + sizeof(xWindowRoot);
117*4882a593Smuzhiyun     int i;
118*4882a593Smuzhiyun     Bool *found = NULL;
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun     MAXSCREENSALLOC(found);
121*4882a593Smuzhiyun     if (!found)
122*4882a593Smuzhiyun         dmxLog(dmxFatal, "dmxConnectionBlockCallback: out of memory\n");
123*4882a593Smuzhiyun 
124*4882a593Smuzhiyun     dmxLog(dmxInfo, "===== Start of Summary =====\n");
125*4882a593Smuzhiyun #ifdef PANORAMIX
126*4882a593Smuzhiyun     if (!noPanoramiXExtension) {
127*4882a593Smuzhiyun         if (dmxGlobalWidth && dmxGlobalHeight
128*4882a593Smuzhiyun             && (dmxGlobalWidth != PanoramiXPixWidth
129*4882a593Smuzhiyun                 || dmxGlobalHeight != PanoramiXPixHeight)) {
130*4882a593Smuzhiyun             dmxLog(dmxInfo,
131*4882a593Smuzhiyun                    "Changing Xinerama dimensions from %d %d to %d %d\n",
132*4882a593Smuzhiyun                    PanoramiXPixWidth, PanoramiXPixHeight,
133*4882a593Smuzhiyun                    dmxGlobalWidth, dmxGlobalHeight);
134*4882a593Smuzhiyun             PanoramiXPixWidth = root->pixWidth = dmxGlobalWidth;
135*4882a593Smuzhiyun             PanoramiXPixHeight = root->pixHeight = dmxGlobalHeight;
136*4882a593Smuzhiyun         }
137*4882a593Smuzhiyun         else {
138*4882a593Smuzhiyun             dmxGlobalWidth = PanoramiXPixWidth;
139*4882a593Smuzhiyun             dmxGlobalHeight = PanoramiXPixHeight;
140*4882a593Smuzhiyun         }
141*4882a593Smuzhiyun         dmxLog(dmxInfo, "%d screens configured with Xinerama (%d %d)\n",
142*4882a593Smuzhiyun                PanoramiXNumScreens, PanoramiXPixWidth, PanoramiXPixHeight);
143*4882a593Smuzhiyun         FOR_NSCREENS(i) found[i] = FALSE;
144*4882a593Smuzhiyun     }
145*4882a593Smuzhiyun     else {
146*4882a593Smuzhiyun #endif
147*4882a593Smuzhiyun         /* This never happens because we're
148*4882a593Smuzhiyun          * either called from a Xinerama
149*4882a593Smuzhiyun          * callback or during reconfiguration
150*4882a593Smuzhiyun          * (which only works with Xinerama on).
151*4882a593Smuzhiyun          * In any case, be reasonable. */
152*4882a593Smuzhiyun         dmxLog(dmxInfo, "%d screens configured (%d %d)\n",
153*4882a593Smuzhiyun                screenInfo.numScreens, root->pixWidth, root->pixHeight);
154*4882a593Smuzhiyun #ifdef PANORAMIX
155*4882a593Smuzhiyun     }
156*4882a593Smuzhiyun #endif
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun     for (i = 0; i < root->nDepths; i++) {
159*4882a593Smuzhiyun         xDepth *depth = (xDepth *) (ConnectionInfo + offset);
160*4882a593Smuzhiyun         int voffset = offset + sizeof(xDepth);
161*4882a593Smuzhiyun         xVisualType *visual = (xVisualType *) (ConnectionInfo + voffset);
162*4882a593Smuzhiyun         int j;
163*4882a593Smuzhiyun 
164*4882a593Smuzhiyun         dmxLog(dmxInfo, "%d visuals at depth %d:\n",
165*4882a593Smuzhiyun                depth->nVisuals, depth->depth);
166*4882a593Smuzhiyun         for (j = 0; j < depth->nVisuals; j++, visual++) {
167*4882a593Smuzhiyun             XVisualInfo vi;
168*4882a593Smuzhiyun 
169*4882a593Smuzhiyun             vi.visual = NULL;
170*4882a593Smuzhiyun             vi.visualid = visual->visualID;
171*4882a593Smuzhiyun             vi.screen = 0;
172*4882a593Smuzhiyun             vi.depth = depth->depth;
173*4882a593Smuzhiyun             vi.class = visual->class;
174*4882a593Smuzhiyun             vi.red_mask = visual->redMask;
175*4882a593Smuzhiyun             vi.green_mask = visual->greenMask;
176*4882a593Smuzhiyun             vi.blue_mask = visual->blueMask;
177*4882a593Smuzhiyun             vi.colormap_size = visual->colormapEntries;
178*4882a593Smuzhiyun             vi.bits_per_rgb = visual->bitsPerRGB;
179*4882a593Smuzhiyun             dmxLogVisual(NULL, &vi, 0);
180*4882a593Smuzhiyun 
181*4882a593Smuzhiyun #ifdef PANORAMIX
182*4882a593Smuzhiyun             if (!noPanoramiXExtension) {
183*4882a593Smuzhiyun                 int k;
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun                 FOR_NSCREENS(k) {
186*4882a593Smuzhiyun                     DMXScreenInfo *dmxScreen = &dmxScreens[k];
187*4882a593Smuzhiyun 
188*4882a593Smuzhiyun                     if (dmxScreen->beDisplay) {
189*4882a593Smuzhiyun                         XVisualInfo *pvi =
190*4882a593Smuzhiyun                             &dmxScreen->beVisuals[dmxScreen->beDefVisualIndex];
191*4882a593Smuzhiyun                         if (pvi->depth == depth->depth &&
192*4882a593Smuzhiyun                             pvi->class == visual->class)
193*4882a593Smuzhiyun                             found[k] = TRUE;
194*4882a593Smuzhiyun                     }
195*4882a593Smuzhiyun                     else {
196*4882a593Smuzhiyun                         /* Screen #k is detatched, so it always succeeds */
197*4882a593Smuzhiyun                         found[k] = TRUE;
198*4882a593Smuzhiyun                     }
199*4882a593Smuzhiyun                 }
200*4882a593Smuzhiyun             }
201*4882a593Smuzhiyun #endif
202*4882a593Smuzhiyun         }
203*4882a593Smuzhiyun         offset = voffset + depth->nVisuals * sizeof(xVisualType);
204*4882a593Smuzhiyun     }
205*4882a593Smuzhiyun 
206*4882a593Smuzhiyun     dmxInputLogDevices();
207*4882a593Smuzhiyun     dmxLog(dmxInfo, "===== End of Summary =====\n");
208*4882a593Smuzhiyun 
209*4882a593Smuzhiyun #ifdef PANORAMIX
210*4882a593Smuzhiyun     if (!noPanoramiXExtension) {
211*4882a593Smuzhiyun         Bool fatal = FALSE;
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun         FOR_NSCREENS(i) {
214*4882a593Smuzhiyun             fatal |= !found[i];
215*4882a593Smuzhiyun             if (!found[i]) {
216*4882a593Smuzhiyun                 dmxLog(dmxError,
217*4882a593Smuzhiyun                        "The default visual for screen #%d does not match "
218*4882a593Smuzhiyun                        "any of the\n", i);
219*4882a593Smuzhiyun                 dmxLog(dmxError,
220*4882a593Smuzhiyun                        "consolidated visuals from Xinerama (listed above)\n");
221*4882a593Smuzhiyun             }
222*4882a593Smuzhiyun         }
223*4882a593Smuzhiyun         if (fatal)
224*4882a593Smuzhiyun             dmxLog(dmxFatal,
225*4882a593Smuzhiyun                    "dmxConnectionBlockCallback: invalid screen(s) found");
226*4882a593Smuzhiyun     }
227*4882a593Smuzhiyun #endif
228*4882a593Smuzhiyun     MAXSCREENSFREE(found);
229*4882a593Smuzhiyun }
230