465 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			465 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright © 2006 Sun Microsystems
 | |
|  *
 | |
|  * Permission to use, copy, modify, distribute, and sell this software and its
 | |
|  * documentation for any purpose is hereby granted without fee, provided that
 | |
|  * the above copyright notice appear in all copies and that both that
 | |
|  * copyright notice and this permission notice appear in supporting
 | |
|  * documentation, and that the name of Sun Microsystems not be used in
 | |
|  * advertising or publicity pertaining to distribution of the software without
 | |
|  * specific, written prior permission.  Sun Microsystems makes no
 | |
|  * representations about the suitability of this software for any purpose.  It
 | |
|  * is provided "as is" without express or implied warranty.
 | |
|  *
 | |
|  * SUN MICROSYSTEMS DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
 | |
|  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
 | |
|  * EVENT SHALL SUN MICROSYSTEMS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
 | |
|  * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
 | |
|  * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
 | |
|  * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
 | |
|  * PERFORMANCE OF THIS SOFTWARE.
 | |
|  *
 | |
|  * Copyright © 2003 Keith Packard
 | |
|  *
 | |
|  * Permission to use, copy, modify, distribute, and sell this software and its
 | |
|  * documentation for any purpose is hereby granted without fee, provided that
 | |
|  * the above copyright notice appear in all copies and that both that
 | |
|  * copyright notice and this permission notice appear in supporting
 | |
|  * documentation, and that the name of Keith Packard not be used in
 | |
|  * advertising or publicity pertaining to distribution of the software without
 | |
|  * specific, written prior permission.  Keith Packard makes no
 | |
|  * representations about the suitability of this software for any purpose.  It
 | |
|  * is provided "as is" without express or implied warranty.
 | |
|  *
 | |
|  * KEITH PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
 | |
|  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
 | |
|  * EVENT SHALL KEITH PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR
 | |
|  * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
 | |
|  * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
 | |
|  * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
 | |
|  * PERFORMANCE OF THIS SOFTWARE.
 | |
|  */
 | |
| 
 | |
| #ifdef HAVE_DIX_CONFIG_H
 | |
| #include <dix-config.h>
 | |
| #endif
 | |
| 
 | |
| #include "compint.h"
 | |
| 
 | |
| int	CompScreenPrivateIndex;
 | |
| int	CompWindowPrivateIndex;
 | |
| int	CompSubwindowsPrivateIndex;
 | |
| int	CompGeneration;
 | |
| 
 | |
| 
 | |
| static Bool
 | |
| compCloseScreen (int index, ScreenPtr pScreen)
 | |
| {
 | |
|     CompScreenPtr   cs = GetCompScreen (pScreen);
 | |
|     Bool	    ret;
 | |
| 
 | |
|     xfree (cs->alternateVisuals);
 | |
| 
 | |
|     pScreen->CloseScreen = cs->CloseScreen;
 | |
|     pScreen->BlockHandler = cs->BlockHandler;
 | |
|     pScreen->InstallColormap = cs->InstallColormap;
 | |
|     pScreen->ReparentWindow = cs->ReparentWindow;
 | |
|     pScreen->MoveWindow = cs->MoveWindow;
 | |
|     pScreen->ResizeWindow = cs->ResizeWindow;
 | |
|     pScreen->ChangeBorderWidth = cs->ChangeBorderWidth;
 | |
|     
 | |
|     pScreen->ClipNotify = cs->ClipNotify;
 | |
|     pScreen->PaintWindowBackground = cs->PaintWindowBackground;
 | |
|     pScreen->UnrealizeWindow = cs->UnrealizeWindow;
 | |
|     pScreen->RealizeWindow = cs->RealizeWindow;
 | |
|     pScreen->DestroyWindow = cs->DestroyWindow;
 | |
|     pScreen->CreateWindow = cs->CreateWindow;
 | |
|     pScreen->CopyWindow = cs->CopyWindow;
 | |
|     pScreen->PositionWindow = cs->PositionWindow;
 | |
| 
 | |
|     deleteCompOverlayClientsForScreen(pScreen);
 | |
| 
 | |
|     /* 
 | |
|     ** Note: no need to call DeleteWindow; the server has
 | |
|     ** already destroyed it.
 | |
|     */
 | |
|     cs->pOverlayWin = NULL;
 | |
| 
 | |
|     xfree (cs);
 | |
|     pScreen->devPrivates[CompScreenPrivateIndex].ptr = 0;
 | |
|     ret = (*pScreen->CloseScreen) (index, pScreen);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| static void
 | |
| compInstallColormap (ColormapPtr pColormap)
 | |
| {
 | |
|     VisualPtr	    pVisual = pColormap->pVisual;
 | |
|     ScreenPtr	    pScreen = pColormap->pScreen;
 | |
|     CompScreenPtr   cs = GetCompScreen (pScreen);
 | |
|     int		    a;
 | |
| 
 | |
|     for (a = 0; a < cs->numAlternateVisuals; a++)
 | |
| 	if (pVisual->vid == cs->alternateVisuals[a])
 | |
| 	    return;
 | |
|     pScreen->InstallColormap = cs->InstallColormap;
 | |
|     (*pScreen->InstallColormap) (pColormap);
 | |
|     cs->InstallColormap = pScreen->InstallColormap;
 | |
|     pScreen->InstallColormap = compInstallColormap;
 | |
| }
 | |
| 
 | |
| static void
 | |
| compScreenUpdate (ScreenPtr pScreen)
 | |
| {
 | |
|     CompScreenPtr   cs = GetCompScreen (pScreen);
 | |
| 
 | |
|     compCheckTree (pScreen);
 | |
|     if (cs->damaged)
 | |
|     {
 | |
| 	compWindowUpdate (WindowTable[pScreen->myNum]);
 | |
| 	cs->damaged = FALSE;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| compBlockHandler (int	    i,
 | |
| 		  pointer   blockData,
 | |
| 		  pointer   pTimeout,
 | |
| 		  pointer   pReadmask)
 | |
| {
 | |
|     ScreenPtr	    pScreen = screenInfo.screens[i];
 | |
|     CompScreenPtr   cs = GetCompScreen (pScreen);
 | |
| 
 | |
|     pScreen->BlockHandler = cs->BlockHandler;
 | |
|     compScreenUpdate (pScreen);
 | |
|     (*pScreen->BlockHandler) (i, blockData, pTimeout, pReadmask);
 | |
|     cs->BlockHandler = pScreen->BlockHandler;
 | |
|     pScreen->BlockHandler = compBlockHandler;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Add alternate visuals -- always expose an ARGB32 and RGB24 visual
 | |
|  */
 | |
| 
 | |
| static DepthPtr
 | |
| compFindVisuallessDepth (ScreenPtr pScreen, int d)
 | |
| {
 | |
|     int		i;
 | |
| 
 | |
|     for (i = 0; i < pScreen->numDepths; i++)
 | |
|     {
 | |
| 	DepthPtr    depth = &pScreen->allowedDepths[i];
 | |
| 	if (depth->depth == d)
 | |
| 	{
 | |
| 	    /*
 | |
| 	     * Make sure it doesn't have visuals already
 | |
| 	     */
 | |
| 	    if (depth->numVids)
 | |
| 		return 0;
 | |
| 	    /*
 | |
| 	     * looks fine
 | |
| 	     */
 | |
| 	    return depth;
 | |
| 	}
 | |
|     }
 | |
|     /*
 | |
|      * If there isn't one, then it's gonna be hard to have 
 | |
|      * an associated visual
 | |
|      */
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Add a list of visual IDs to the list of visuals to implicitly redirect.
 | |
|  */
 | |
| static Bool
 | |
| compRegisterAlternateVisuals (CompScreenPtr cs, VisualID *vids, int nVisuals)
 | |
| {
 | |
|     VisualID *p;
 | |
| 
 | |
|     p = xrealloc(cs->alternateVisuals,
 | |
| 		 sizeof(VisualID) * (cs->numAlternateVisuals + nVisuals));
 | |
|     if(p == NULL)
 | |
| 	return FALSE;
 | |
| 
 | |
|     memcpy(&p[cs->numAlternateVisuals], vids, sizeof(VisualID) * nVisuals);
 | |
| 
 | |
|     cs->alternateVisuals = p;
 | |
|     cs->numAlternateVisuals += nVisuals;
 | |
| 
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| _X_EXPORT
 | |
| Bool CompositeRegisterAlternateVisuals (ScreenPtr pScreen, VisualID *vids,
 | |
| 					int nVisuals)
 | |
| {
 | |
|     CompScreenPtr cs = GetCompScreen (pScreen);
 | |
|     return compRegisterAlternateVisuals(cs, vids, nVisuals);
 | |
| }
 | |
| 
 | |
| #if COMP_INCLUDE_RGB24_VISUAL
 | |
| #define NUM_COMP_ALTERNATE_VISUALS 2
 | |
| #else
 | |
| #define NUM_COMP_ALTERNATE_VISUALS 1
 | |
| #endif
 | |
| 
 | |
| typedef struct _alternateVisual {
 | |
|     int		depth;
 | |
|     CARD32	format;
 | |
| } CompAlternateVisual;
 | |
| 
 | |
| static CompAlternateVisual  altVisuals[NUM_COMP_ALTERNATE_VISUALS] = {
 | |
| #if COMP_INCLUDE_RGB24_VISUAL
 | |
|     {	24,	PICT_r8g8b8 },
 | |
| #endif
 | |
|     {	32,	PICT_a8r8g8b8 },
 | |
| };
 | |
| 
 | |
| static Bool
 | |
| compAddAlternateVisuals (ScreenPtr pScreen, CompScreenPtr cs)
 | |
| {
 | |
|     VisualPtr	    visuals;
 | |
|     DepthPtr	    depths[NUM_COMP_ALTERNATE_VISUALS];
 | |
|     PictFormatPtr   pPictFormats[NUM_COMP_ALTERNATE_VISUALS];
 | |
|     int		    i;
 | |
|     int		    numVisuals;
 | |
|     VisualID	    *vids[NUM_COMP_ALTERNATE_VISUALS];
 | |
|     XID		    *installedCmaps;
 | |
|     ColormapPtr	    installedCmap;
 | |
|     int		    numInstalledCmaps;
 | |
|     int		    numAlternate = 0;
 | |
|     int		    alt;
 | |
|     
 | |
|     for (alt = 0; alt < NUM_COMP_ALTERNATE_VISUALS; alt++)
 | |
|     {
 | |
| 	DepthPtr	depth;
 | |
| 	PictFormatPtr   pPictFormat;
 | |
| 	
 | |
| 	depth = compFindVisuallessDepth (pScreen, altVisuals[alt].depth);
 | |
| 	if (!depth)
 | |
| 	    continue;
 | |
| 	/*
 | |
| 	 * Find the right picture format
 | |
| 	 */
 | |
| 	pPictFormat = PictureMatchFormat (pScreen, altVisuals[alt].depth,
 | |
| 					  altVisuals[alt].format);
 | |
| 	if (!pPictFormat)
 | |
| 	    continue;
 | |
| 
 | |
| 	/*
 | |
| 	 * Allocate vid list for this depth
 | |
| 	 */
 | |
| 	vids[numAlternate] = xalloc (sizeof (VisualID));
 | |
| 	if (!vids[numAlternate])
 | |
| 	    continue;
 | |
| 	depths[numAlternate] = depth;
 | |
| 	pPictFormats[numAlternate] = pPictFormat;
 | |
| 	numAlternate++;
 | |
|     }
 | |
|     
 | |
|     if (!numAlternate)
 | |
| 	return TRUE;
 | |
| 
 | |
|     /*
 | |
|      * Find the installed colormaps
 | |
|      */
 | |
|     installedCmaps = xalloc (pScreen->maxInstalledCmaps * sizeof (XID));
 | |
|     if (!installedCmaps)
 | |
|     {
 | |
| 	for (alt = 0; alt < numAlternate; alt++)
 | |
| 	    xfree (vids[alt]);
 | |
| 	return FALSE;
 | |
|     }
 | |
|     numInstalledCmaps = (*pScreen->ListInstalledColormaps) (pScreen, 
 | |
| 							    installedCmaps);
 | |
|     
 | |
|     /*
 | |
|      * realloc the visual array to fit the new one in place
 | |
|      */
 | |
|     numVisuals = pScreen->numVisuals;
 | |
|     visuals = xrealloc (pScreen->visuals,
 | |
| 			(numVisuals + numAlternate) * sizeof (VisualRec));
 | |
|     if (!visuals)
 | |
|     {
 | |
| 	for (alt = 0; alt < numAlternate; alt++)
 | |
| 	    xfree (vids[alt]);
 | |
| 	xfree (installedCmaps);
 | |
| 	return FALSE;
 | |
|     }
 | |
|     
 | |
|     /*
 | |
|      * Fix up any existing installed colormaps -- we'll assume that
 | |
|      * the only ones created so far have been installed.  If this
 | |
|      * isn't true, we'll have to walk the resource database looking
 | |
|      * for all colormaps.
 | |
|      */
 | |
|     for (i = 0; i < numInstalledCmaps; i++)
 | |
|     {
 | |
| 	int j;
 | |
| 	
 | |
| 	installedCmap = LookupIDByType (installedCmaps[i], RT_COLORMAP);
 | |
| 	if (!installedCmap)
 | |
| 	    continue;
 | |
| 	j = installedCmap->pVisual - pScreen->visuals;
 | |
| 	installedCmap->pVisual = &visuals[j];
 | |
|     }
 | |
| 
 | |
|     xfree (installedCmaps);
 | |
| 
 | |
|     pScreen->visuals = visuals;
 | |
|     pScreen->numVisuals = numVisuals + numAlternate;
 | |
| 
 | |
|     for (alt = 0; alt < numAlternate; alt++)
 | |
|     {
 | |
| 	DepthPtr	depth = depths[alt];
 | |
| 	PictFormatPtr	pPictFormat = pPictFormats[alt];
 | |
| 	VisualPtr	visual = &visuals[numVisuals + alt];
 | |
| 	unsigned long	alphaMask;
 | |
| 
 | |
| 	/*
 | |
| 	 * Initialize the visual
 | |
| 	 */
 | |
| 	visual->class = TrueColor;
 | |
| 	visual->bitsPerRGBValue = 8;
 | |
| 
 | |
| 	visual->vid = FakeClientID (0);
 | |
| 	visual->redMask   = (((unsigned long) pPictFormat->direct.redMask) << 
 | |
| 			     pPictFormat->direct.red);
 | |
| 	visual->greenMask = (((unsigned long) pPictFormat->direct.greenMask) << 
 | |
| 			     pPictFormat->direct.green);
 | |
| 	visual->blueMask  = (((unsigned long) pPictFormat->direct.blueMask) << 
 | |
| 			     pPictFormat->direct.blue);
 | |
| 	alphaMask =  (((unsigned long) pPictFormat->direct.alphaMask) << 
 | |
| 		      pPictFormat->direct.alpha);
 | |
| 	visual->offsetRed   = pPictFormat->direct.red;
 | |
| 	visual->offsetGreen = pPictFormat->direct.green;
 | |
| 	visual->offsetBlue  = pPictFormat->direct.blue;
 | |
| 	/*
 | |
| 	 * Include A bits in this (unlike GLX which includes only RGB)
 | |
| 	 * This lets DIX compute suitable masks for colormap allocations
 | |
| 	 */
 | |
| 	visual->nplanes = Ones (visual->redMask |
 | |
| 				visual->greenMask |
 | |
| 				visual->blueMask |
 | |
| 				alphaMask);
 | |
| 	/*
 | |
| 	 * find widest component
 | |
| 	 */
 | |
| 	visual->ColormapEntries = (1 << max (Ones (visual->redMask),
 | |
| 					     max (Ones (visual->greenMask),
 | |
| 						  Ones (visual->blueMask))));
 | |
| 
 | |
| 	/*
 | |
| 	 * remember the visual ID to detect auto-update windows
 | |
| 	 */
 | |
| 	compRegisterAlternateVisuals(cs, &visual->vid, 1);
 | |
| 	
 | |
| 	/*
 | |
| 	 * Fix up the depth
 | |
| 	 */
 | |
| 	vids[alt][0] = visual->vid;
 | |
| 	depth->numVids = 1;
 | |
| 	depth->vids = vids[alt];
 | |
|     }
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| Bool
 | |
| compScreenInit (ScreenPtr pScreen)
 | |
| {
 | |
|     CompScreenPtr   cs;
 | |
| 
 | |
|     if (CompGeneration != serverGeneration)
 | |
|     {
 | |
| 	CompScreenPrivateIndex = AllocateScreenPrivateIndex ();
 | |
| 	if (CompScreenPrivateIndex == -1)
 | |
| 	    return FALSE;
 | |
| 	CompWindowPrivateIndex = AllocateWindowPrivateIndex ();
 | |
| 	if (CompWindowPrivateIndex == -1)
 | |
| 	    return FALSE;
 | |
| 	CompSubwindowsPrivateIndex = AllocateWindowPrivateIndex ();
 | |
| 	if (CompSubwindowsPrivateIndex == -1)
 | |
| 	    return FALSE;
 | |
| 	CompGeneration = serverGeneration;
 | |
|     }
 | |
|     if (!AllocateWindowPrivate (pScreen, CompWindowPrivateIndex, 0))
 | |
| 	return FALSE;
 | |
| 
 | |
|     if (!AllocateWindowPrivate (pScreen, CompSubwindowsPrivateIndex, 0))
 | |
| 	return FALSE;
 | |
| 
 | |
|     if (GetCompScreen (pScreen))
 | |
| 	return TRUE;
 | |
|     cs = (CompScreenPtr) xalloc (sizeof (CompScreenRec));
 | |
|     if (!cs)
 | |
| 	return FALSE;
 | |
| 
 | |
|     cs->damaged = FALSE;
 | |
|     cs->pOverlayWin = NULL;
 | |
|     cs->pOverlayClients = NULL;
 | |
| 
 | |
|     cs->numAlternateVisuals = 0;
 | |
|     cs->alternateVisuals = NULL;
 | |
| 
 | |
|     if (!compAddAlternateVisuals (pScreen, cs))
 | |
|     {
 | |
| 	xfree (cs);
 | |
| 	return FALSE;
 | |
|     }
 | |
| 
 | |
|     cs->PositionWindow = pScreen->PositionWindow;
 | |
|     pScreen->PositionWindow = compPositionWindow;
 | |
| 
 | |
|     cs->CopyWindow = pScreen->CopyWindow;
 | |
|     pScreen->CopyWindow = compCopyWindow;
 | |
| 
 | |
|     cs->CreateWindow = pScreen->CreateWindow;
 | |
|     pScreen->CreateWindow = compCreateWindow;
 | |
| 
 | |
|     cs->DestroyWindow = pScreen->DestroyWindow;
 | |
|     pScreen->DestroyWindow = compDestroyWindow;
 | |
| 
 | |
|     cs->RealizeWindow = pScreen->RealizeWindow;
 | |
|     pScreen->RealizeWindow = compRealizeWindow;
 | |
| 
 | |
|     cs->UnrealizeWindow = pScreen->UnrealizeWindow;
 | |
|     pScreen->UnrealizeWindow = compUnrealizeWindow;
 | |
| 
 | |
|     cs->PaintWindowBackground = pScreen->PaintWindowBackground;
 | |
|     pScreen->PaintWindowBackground = compPaintWindowBackground;
 | |
| 
 | |
|     cs->ClipNotify = pScreen->ClipNotify;
 | |
|     pScreen->ClipNotify = compClipNotify;
 | |
| 
 | |
|     cs->MoveWindow = pScreen->MoveWindow;
 | |
|     pScreen->MoveWindow = compMoveWindow;
 | |
| 
 | |
|     cs->ResizeWindow = pScreen->ResizeWindow;
 | |
|     pScreen->ResizeWindow = compResizeWindow;
 | |
| 
 | |
|     cs->ChangeBorderWidth = pScreen->ChangeBorderWidth;
 | |
|     pScreen->ChangeBorderWidth = compChangeBorderWidth;
 | |
| 
 | |
|     cs->ReparentWindow = pScreen->ReparentWindow;
 | |
|     pScreen->ReparentWindow = compReparentWindow;
 | |
| 
 | |
|     cs->InstallColormap = pScreen->InstallColormap;
 | |
|     pScreen->InstallColormap = compInstallColormap;
 | |
| 
 | |
|     cs->BlockHandler = pScreen->BlockHandler;
 | |
|     pScreen->BlockHandler = compBlockHandler;
 | |
| 
 | |
|     cs->CloseScreen = pScreen->CloseScreen;
 | |
|     pScreen->CloseScreen = compCloseScreen;
 | |
| 
 | |
|     pScreen->devPrivates[CompScreenPrivateIndex].ptr = (pointer) cs;
 | |
| 
 | |
|     RegisterRealChildHeadProc(CompositeRealChildHead);
 | |
| 
 | |
|     return TRUE;
 | |
| }
 |